mirror of
https://github.com/docmost/docmost.git
synced 2026-05-08 15:23:07 +08:00
Compare commits
1 Commits
main
..
bug-352873
| Author | SHA1 | Date | |
|---|---|---|---|
| 12bd213236 |
@@ -43,9 +43,6 @@ POSTMARK_TOKEN=
|
|||||||
# for custom drawio server
|
# for custom drawio server
|
||||||
DRAWIO_URL=
|
DRAWIO_URL=
|
||||||
|
|
||||||
# Gotenberg URL for server-side PDF export
|
|
||||||
GOTENBERG_URL=
|
|
||||||
|
|
||||||
DISABLE_TELEMETRY=false
|
DISABLE_TELEMETRY=false
|
||||||
|
|
||||||
# Enable debug logging in production (default: false)
|
# Enable debug logging in production (default: false)
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "client",
|
"name": "client",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "0.80.1",
|
"version": "0.71.1",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
"build": "tsc && vite build",
|
"build": "tsc && vite build",
|
||||||
@@ -31,8 +31,8 @@
|
|||||||
"emoji-mart": "^5.6.0",
|
"emoji-mart": "^5.6.0",
|
||||||
"file-saver": "^2.0.5",
|
"file-saver": "^2.0.5",
|
||||||
"highlightjs-sap-abap": "^0.3.0",
|
"highlightjs-sap-abap": "^0.3.0",
|
||||||
"i18next": "25.10.1",
|
"i18next": "^25.10.1",
|
||||||
"i18next-http-backend": "3.0.6",
|
"i18next-http-backend": "^3.0.2",
|
||||||
"jotai": "^2.18.1",
|
"jotai": "^2.18.1",
|
||||||
"jotai-optics": "^0.4.0",
|
"jotai-optics": "^0.4.0",
|
||||||
"js-cookie": "^3.0.5",
|
"js-cookie": "^3.0.5",
|
||||||
@@ -42,7 +42,7 @@
|
|||||||
"mantine-form-zod-resolver": "^1.3.0",
|
"mantine-form-zod-resolver": "^1.3.0",
|
||||||
"mermaid": "^11.13.0",
|
"mermaid": "^11.13.0",
|
||||||
"mitt": "^3.0.1",
|
"mitt": "^3.0.1",
|
||||||
"posthog-js": "1.372.2",
|
"posthog-js": "1.363.1",
|
||||||
"react": "^18.3.1",
|
"react": "^18.3.1",
|
||||||
"react-arborist": "3.4.0",
|
"react-arborist": "3.4.0",
|
||||||
"react-clear-modal": "^2.0.18",
|
"react-clear-modal": "^2.0.18",
|
||||||
@@ -50,7 +50,7 @@
|
|||||||
"react-drawio": "^1.0.7",
|
"react-drawio": "^1.0.7",
|
||||||
"react-error-boundary": "^6.1.1",
|
"react-error-boundary": "^6.1.1",
|
||||||
"react-helmet-async": "^3.0.0",
|
"react-helmet-async": "^3.0.0",
|
||||||
"react-i18next": "16.5.8",
|
"react-i18next": "^16.5.8",
|
||||||
"react-router-dom": "^7.13.1",
|
"react-router-dom": "^7.13.1",
|
||||||
"semver": "^7.7.4",
|
"semver": "^7.7.4",
|
||||||
"socket.io-client": "^4.8.3",
|
"socket.io-client": "^4.8.3",
|
||||||
@@ -74,7 +74,7 @@
|
|||||||
"eslint-plugin-react-refresh": "^0.5.2",
|
"eslint-plugin-react-refresh": "^0.5.2",
|
||||||
"globals": "^15.13.0",
|
"globals": "^15.13.0",
|
||||||
"optics-ts": "^2.4.1",
|
"optics-ts": "^2.4.1",
|
||||||
"postcss": "^8.5.12",
|
"postcss": "^8.5.8",
|
||||||
"postcss-preset-mantine": "^1.18.0",
|
"postcss-preset-mantine": "^1.18.0",
|
||||||
"postcss-simple-vars": "^7.0.1",
|
"postcss-simple-vars": "^7.0.1",
|
||||||
"prettier": "^3.8.1",
|
"prettier": "^3.8.1",
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "Kommentar bearbeiten",
|
"Edit comment": "Kommentar bearbeiten",
|
||||||
"Delete comment": "Kommentar löschen",
|
"Delete comment": "Kommentar löschen",
|
||||||
"Are you sure you want to delete this comment?": "Sind Sie sicher, dass Sie diesen Kommentar löschen möchten?",
|
"Are you sure you want to delete this comment?": "Sind Sie sicher, dass Sie diesen Kommentar löschen möchten?",
|
||||||
"Delete chat": "Chat löschen",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "Sind Sie sicher, dass Sie '{{title}}' löschen möchten? Diese Aktion kann nicht rückgängig gemacht werden.",
|
|
||||||
"Comment created successfully": "Kommentar erfolgreich erstellt",
|
"Comment created successfully": "Kommentar erfolgreich erstellt",
|
||||||
"Error creating comment": "Fehler beim Erstellen des Kommentars",
|
"Error creating comment": "Fehler beim Erstellen des Kommentars",
|
||||||
"Comment updated successfully": "Kommentar erfolgreich aktualisiert",
|
"Comment updated successfully": "Kommentar erfolgreich aktualisiert",
|
||||||
@@ -391,7 +389,7 @@
|
|||||||
"Write anything. Enter \"/\" for commands": "Schreiben Sie etwas. Geben Sie \"/\" für Befehle ein",
|
"Write anything. Enter \"/\" for commands": "Schreiben Sie etwas. Geben Sie \"/\" für Befehle ein",
|
||||||
"Write...": "\"Schreiben...\"",
|
"Write...": "\"Schreiben...\"",
|
||||||
"Column count": "Spaltenanzahl",
|
"Column count": "Spaltenanzahl",
|
||||||
"{{count}} Columns": "{{count}} Spalten",
|
"{{count}} Columns": "{count, plural, one {# Spalte} other {# Spalten}}",
|
||||||
"Equal columns": "Gleich breite Spalten",
|
"Equal columns": "Gleich breite Spalten",
|
||||||
"Left sidebar": "Linke Seitenleiste",
|
"Left sidebar": "Linke Seitenleiste",
|
||||||
"Right sidebar": "Rechte Seitenleiste",
|
"Right sidebar": "Rechte Seitenleiste",
|
||||||
@@ -741,93 +739,6 @@
|
|||||||
"Removed page restriction": "Seitenbeschränkung entfernt",
|
"Removed page restriction": "Seitenbeschränkung entfernt",
|
||||||
"Added page permission": "Seitenberechtigung hinzugefügt",
|
"Added page permission": "Seitenberechtigung hinzugefügt",
|
||||||
"Removed page permission": "Seitenberechtigung entfernt",
|
"Removed page permission": "Seitenberechtigung entfernt",
|
||||||
"day": "Tag",
|
|
||||||
"days": "Tage",
|
|
||||||
"week": "Woche",
|
|
||||||
"weeks": "Wochen",
|
|
||||||
"month": "Monat",
|
|
||||||
"months": "Monate",
|
|
||||||
"year": "Jahr",
|
|
||||||
"years": "Jahre",
|
|
||||||
"Period": "Zeitraum",
|
|
||||||
"Fixed date": "Festes Datum",
|
|
||||||
"Indefinitely": "Unbegrenzt",
|
|
||||||
"Days": "Tage",
|
|
||||||
"Weeks": "Wochen",
|
|
||||||
"Months": "Monate",
|
|
||||||
"Years": "Jahre",
|
|
||||||
"Pick a date": "Datum auswählen",
|
|
||||||
"Maximum is {{max}} {{unit}} for this unit": "Das Maximum für diese Einheit beträgt {{max}} {{unit}}",
|
|
||||||
"Never expires. Verifiers can re-verify at any time.": "Läuft nie ab. Prüfer können die Seite jederzeit erneut verifizieren.",
|
|
||||||
"Verified": "Verifiziert",
|
|
||||||
"Review needed": "Prüfung erforderlich",
|
|
||||||
"Verification expired": "Verifizierung abgelaufen",
|
|
||||||
"Draft": "Entwurf",
|
|
||||||
"In Approval": "In Genehmigung",
|
|
||||||
"In approval": "In Genehmigung",
|
|
||||||
"Approved": "Genehmigt",
|
|
||||||
"Obsolete": "Veraltet",
|
|
||||||
"Expiring": "Läuft bald ab",
|
|
||||||
"Set up verification": "Verifizierung einrichten",
|
|
||||||
"Verify page": "Seite verifizieren",
|
|
||||||
"Page verification": "Seitenverifizierung",
|
|
||||||
"Add verification": "Verifizierung hinzufügen",
|
|
||||||
"Edit verification": "Verifizierung bearbeiten",
|
|
||||||
"Search by title": "Nach Titel suchen",
|
|
||||||
"Choose how this page should stay accurate.": "Wählen Sie aus, wie diese Seite aktuell gehalten werden soll.",
|
|
||||||
"Recurring verification": "Wiederkehrende Verifizierung",
|
|
||||||
"Verifiers re-confirm this page on a schedule.": "Prüfer bestätigen diese Seite nach einem Zeitplan erneut.",
|
|
||||||
"Re-verify on a schedule (e.g every 30 days )": "Nach einem Zeitplan erneut verifizieren (z. B. alle 30 Tage)",
|
|
||||||
"Page stays editable at all times": "Die Seite bleibt jederzeit bearbeitbar",
|
|
||||||
"Best for runbooks, FAQs, living documentation": "Am besten für Runbooks, FAQs und lebende Dokumentation geeignet",
|
|
||||||
"Approval workflow": "Genehmigungsworkflow",
|
|
||||||
"Formal document lifecycle with named approvers.": "Formaler Dokumentenlebenszyklus mit benannten Genehmigern.",
|
|
||||||
"Draft → In approval → Approved → Obsolete": "Entwurf → In Genehmigung → Genehmigt → Veraltet",
|
|
||||||
"Locked once approved, with full history": "Nach der Genehmigung gesperrt, mit vollständiger Historie",
|
|
||||||
"Designed for ISO 9001, ISO 13485, and FDA": "Entwickelt für ISO 9001, ISO 13485 und FDA",
|
|
||||||
"Best for SOPs and controlled documents": "Am besten für SOPs und kontrollierte Dokumente geeignet",
|
|
||||||
"Back": "Zurück",
|
|
||||||
"Quality management": "Qualitätsmanagement",
|
|
||||||
"Recurring": "Wiederkehrend",
|
|
||||||
"Pages move through draft, approval, and approved stages.": "Seiten durchlaufen die Phasen Entwurf, Genehmigung und Genehmigt.",
|
|
||||||
"Verifiers": "Prüfer",
|
|
||||||
"Add verifier": "Prüfer hinzufügen",
|
|
||||||
"I've reviewed this page for accuracy": "Ich habe diese Seite auf Richtigkeit geprüft",
|
|
||||||
"Set up": "Einrichten",
|
|
||||||
"Remove verification": "Verifizierung entfernen",
|
|
||||||
"Are you sure you want to remove verification from this page?": "Möchten Sie die Verifizierung wirklich von dieser Seite entfernen?",
|
|
||||||
"Assigned verifiers must periodically re-verify this page.": "Zugewiesene Prüfer müssen diese Seite regelmäßig erneut verifizieren.",
|
|
||||||
"Last verified by {{name}} {{time}} (expired)": "Zuletzt von {{name}} {{time}} verifiziert (abgelaufen)",
|
|
||||||
"The fixed expiration date has passed.": "Das feste Ablaufdatum ist überschritten.",
|
|
||||||
"Verified by {{name}} {{time}}": "Verifiziert von {{name}} {{time}}",
|
|
||||||
"Expires {{date}}": "Läuft ab am {{date}}",
|
|
||||||
"Expired {{date}}": "Abgelaufen am {{date}}",
|
|
||||||
"Mark as obsolete": "Als veraltet markieren",
|
|
||||||
"Mark obsolete": "Als veraltet markieren",
|
|
||||||
"Returned by {{name}} {{time}}": "Zurückgegeben von {{name}} {{time}}",
|
|
||||||
"No approval has been requested yet.": "Es wurde noch keine Genehmigung angefordert.",
|
|
||||||
"Submitted by {{name}} {{time}}": "Eingereicht von {{name}} {{time}}",
|
|
||||||
"Someone": "Jemand",
|
|
||||||
"Approved by {{name}} {{time}}": "Genehmigt von {{name}} {{time}}",
|
|
||||||
"This document has been marked as obsolete.": "Dieses Dokument wurde als veraltet markiert.",
|
|
||||||
"Rejection comment": "Ablehnungskommentar",
|
|
||||||
"Reason for returning this document...": "Grund für die Rückgabe dieses Dokuments...",
|
|
||||||
"Confirm rejection": "Ablehnung bestätigen",
|
|
||||||
"Submit for approval": "Zur Genehmigung einreichen",
|
|
||||||
"Reject": "Ablehnen",
|
|
||||||
"Approve": "Genehmigen",
|
|
||||||
"Re-submit for approval": "Erneut zur Genehmigung einreichen",
|
|
||||||
"Verified until": "Verifiziert bis",
|
|
||||||
"QMS": "QMS",
|
|
||||||
"Verified pages": "Verifizierte Seiten",
|
|
||||||
"Search pages...": "Seiten suchen...",
|
|
||||||
"Filter by space": "Nach Bereich filtern",
|
|
||||||
"Filter by type": "Nach Typ filtern",
|
|
||||||
"<bold>{{name}}</bold> verified a page": "<bold>{{name}}</bold> hat eine Seite verifiziert",
|
|
||||||
"<bold>{{name}}</bold> submitted a page for your approval": "<bold>{{name}}</bold> hat eine Seite zu Ihrer Genehmigung eingereicht",
|
|
||||||
"<bold>{{name}}</bold> returned a page for revision": "<bold>{{name}}</bold> hat eine Seite zur Überarbeitung zurückgegeben",
|
|
||||||
"Page verification expires soon": "Die Seitenverifizierung läuft bald ab",
|
|
||||||
"Page verification has expired": "Die Seitenverifizierung ist abgelaufen",
|
|
||||||
"Verifying your email": "Ihre E-Mail wird bestätigt",
|
"Verifying your email": "Ihre E-Mail wird bestätigt",
|
||||||
"Please wait...": "Bitte warten...",
|
"Please wait...": "Bitte warten...",
|
||||||
"Verification failed. The link may have expired.": "Überprüfung fehlgeschlagen. Der Link ist möglicherweise abgelaufen.",
|
"Verification failed. The link may have expired.": "Überprüfung fehlgeschlagen. Der Link ist möglicherweise abgelaufen.",
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "Edit comment",
|
"Edit comment": "Edit comment",
|
||||||
"Delete comment": "Delete comment",
|
"Delete comment": "Delete comment",
|
||||||
"Are you sure you want to delete this comment?": "Are you sure you want to delete this comment?",
|
"Are you sure you want to delete this comment?": "Are you sure you want to delete this comment?",
|
||||||
"Delete chat": "Delete chat",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "Are you sure you want to delete '{{title}}'? This action cannot be undone.",
|
|
||||||
"Comment created successfully": "Comment created successfully",
|
"Comment created successfully": "Comment created successfully",
|
||||||
"Error creating comment": "Error creating comment",
|
"Error creating comment": "Error creating comment",
|
||||||
"Comment updated successfully": "Comment updated successfully",
|
"Comment updated successfully": "Comment updated successfully",
|
||||||
@@ -416,7 +414,6 @@
|
|||||||
"{{latestVersion}} is available": "{{latestVersion}} is available",
|
"{{latestVersion}} is available": "{{latestVersion}} is available",
|
||||||
"Default page edit mode": "Default page edit mode",
|
"Default page edit mode": "Default page edit mode",
|
||||||
"Choose your preferred page edit mode. Avoid accidental edits.": "Choose your preferred page edit mode. Avoid accidental edits.",
|
"Choose your preferred page edit mode. Avoid accidental edits.": "Choose your preferred page edit mode. Avoid accidental edits.",
|
||||||
"Choose {{format}} file": "Choose {{format}} file",
|
|
||||||
"Reading": "Reading",
|
"Reading": "Reading",
|
||||||
"Delete member": "Delete member",
|
"Delete member": "Delete member",
|
||||||
"Member deleted successfully": "Member deleted successfully",
|
"Member deleted successfully": "Member deleted successfully",
|
||||||
@@ -609,21 +606,25 @@
|
|||||||
"Image exceeds 10MB limit.": "Image exceeds 10MB limit.",
|
"Image exceeds 10MB limit.": "Image exceeds 10MB limit.",
|
||||||
"Image removed successfully": "Image removed successfully",
|
"Image removed successfully": "Image removed successfully",
|
||||||
"API key": "API key",
|
"API key": "API key",
|
||||||
|
"API key created successfully": "API key created successfully",
|
||||||
"API keys": "API keys",
|
"API keys": "API keys",
|
||||||
"API management": "API management",
|
"API management": "API management",
|
||||||
|
"Are you sure you want to revoke this API key": "Are you sure you want to revoke this API key",
|
||||||
|
"Create API Key": "Create API Key",
|
||||||
"Custom expiration date": "Custom expiration date",
|
"Custom expiration date": "Custom expiration date",
|
||||||
"Enter a descriptive token name": "Enter a descriptive token name",
|
"Enter a descriptive token name": "Enter a descriptive token name",
|
||||||
"Expiration": "Expiration",
|
"Expiration": "Expiration",
|
||||||
"Expired": "Expired",
|
"Expired": "Expired",
|
||||||
"Expires": "Expires",
|
"Expires": "Expires",
|
||||||
|
"I've saved my API key": "I've saved my API key",
|
||||||
"Last use": "Last Used",
|
"Last use": "Last Used",
|
||||||
"No API keys found": "No API keys found",
|
"No API keys found": "No API keys found",
|
||||||
"No expiration": "No expiration",
|
"No expiration": "No expiration",
|
||||||
|
"Revoke API key": "Revoke API key",
|
||||||
"Revoked successfully": "Revoked successfully",
|
"Revoked successfully": "Revoked successfully",
|
||||||
"Select expiration date": "Select expiration date",
|
"Select expiration date": "Select expiration date",
|
||||||
"This action cannot be undone. Any applications using this API key will stop working.": "This action cannot be undone. Any applications using this API key will stop working.",
|
"This action cannot be undone. Any applications using this API key will stop working.": "This action cannot be undone. Any applications using this API key will stop working.",
|
||||||
"Update": "Update",
|
"Update API key": "Update API key",
|
||||||
"Update {{credential}}": "Update {{credential}}",
|
|
||||||
"Manage API keys for all users in the workspace": "Manage API keys for all users in the workspace",
|
"Manage API keys for all users in the workspace": "Manage API keys for all users in the workspace",
|
||||||
"Restrict API key creation to admins": "Restrict API key creation to admins",
|
"Restrict API key creation to admins": "Restrict API key creation to admins",
|
||||||
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "Only admins and owners can create new API keys. Existing member keys will continue to work.",
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "Only admins and owners can create new API keys. Existing member keys will continue to work.",
|
||||||
@@ -870,8 +871,6 @@
|
|||||||
"Previous 7 days": "Previous 7 days",
|
"Previous 7 days": "Previous 7 days",
|
||||||
"Previous 30 days": "Previous 30 days",
|
"Previous 30 days": "Previous 30 days",
|
||||||
"Search chats...": "Search chats...",
|
"Search chats...": "Search chats...",
|
||||||
"Search chats": "Search chats",
|
|
||||||
"Ask anything... Use @ to mention pages": "Ask anything... Use @ to mention pages",
|
|
||||||
"Start a new chat to see it here.": "Start a new chat to see it here.",
|
"Start a new chat to see it here.": "Start a new chat to see it here.",
|
||||||
"Summarize this page": "Summarize this page",
|
"Summarize this page": "Summarize this page",
|
||||||
"Toggle AI Chat": "Toggle AI Chat",
|
"Toggle AI Chat": "Toggle AI Chat",
|
||||||
@@ -879,54 +878,5 @@
|
|||||||
"Try a different search term.": "Try a different search term.",
|
"Try a different search term.": "Try a different search term.",
|
||||||
"Try again": "Try again",
|
"Try again": "Try again",
|
||||||
"Untitled chat": "Untitled chat",
|
"Untitled chat": "Untitled chat",
|
||||||
"What can I help you with?": "What can I help you with?",
|
"What can I help you with?": "What can I help you with?"
|
||||||
"Are you sure you want to revoke this {{credential}}": "Are you sure you want to revoke this {{credential}}",
|
|
||||||
"Automatically provision users and groups from your identity provider via SCIM.": "Automatically provision users and groups from your identity provider via SCIM.",
|
|
||||||
"Configure your identity provider with this URL to provision users and groups.": "Configure your identity provider with this URL to provision users and groups.",
|
|
||||||
"Create {{credential}}": "Create {{credential}}",
|
|
||||||
"{{credential}} created": "{{credential}} created",
|
|
||||||
"{{credential}} created successfully": "{{credential}} created successfully",
|
|
||||||
"Created by": "Created by",
|
|
||||||
"Custom": "Custom",
|
|
||||||
"Enable SCIM": "Enable SCIM",
|
|
||||||
"Enter a descriptive name": "Enter a descriptive name",
|
|
||||||
"I've saved my {{credential}}": "I've saved my {{credential}}",
|
|
||||||
"Important": "Important",
|
|
||||||
"Make sure to copy your {{credential}} now. You won't be able to see it again!": "Make sure to copy your {{credential}} now. You won't be able to see it again!",
|
|
||||||
"Never": "Never",
|
|
||||||
"Revoke {{credential}}": "Revoke {{credential}}",
|
|
||||||
"SCIM endpoint URL": "SCIM endpoint URL",
|
|
||||||
"SCIM provisioning": "SCIM provisioning",
|
|
||||||
"SCIM takes precedence over SSO group sync while enabled.": "SCIM takes precedence over SSO group sync while enabled.",
|
|
||||||
"You have reached the maximum of {{max}} SCIM tokens. Delete an existing token to create a new one.": "You have reached the maximum of {{max}} SCIM tokens. Delete an existing token to create a new one.",
|
|
||||||
"SCIM token": "SCIM token",
|
|
||||||
"SCIM tokens": "SCIM tokens",
|
|
||||||
"This action cannot be undone. Your identity provider will stop syncing immediately.": "This action cannot be undone. Your identity provider will stop syncing immediately.",
|
|
||||||
"Toggle SCIM provisioning": "Toggle SCIM provisioning",
|
|
||||||
"Token": "Token",
|
|
||||||
"Page menu": "Page menu",
|
|
||||||
"Expand": "Expand",
|
|
||||||
"Collapse": "Collapse",
|
|
||||||
"Comment menu": "Comment menu",
|
|
||||||
"Group menu": "Group menu",
|
|
||||||
"Show hidden breadcrumbs": "Show hidden breadcrumbs",
|
|
||||||
"Breadcrumbs": "Breadcrumbs",
|
|
||||||
"Page actions": "Page actions",
|
|
||||||
"Pick emoji": "Pick emoji",
|
|
||||||
"Template menu": "Template menu",
|
|
||||||
"Chat menu": "Chat menu",
|
|
||||||
"API key menu": "API key menu",
|
|
||||||
"Jump to comment selection": "Jump to comment selection",
|
|
||||||
"Slash commands": "Slash commands",
|
|
||||||
"Mention suggestions": "Mention suggestions",
|
|
||||||
"Link suggestions": "Link suggestions",
|
|
||||||
"Diagram editor": "Diagram editor",
|
|
||||||
"Add comment": "Add comment",
|
|
||||||
"Find and replace": "Find and replace",
|
|
||||||
"Main navigation": "Main navigation",
|
|
||||||
"Space navigation": "Space navigation",
|
|
||||||
"Settings navigation": "Settings navigation",
|
|
||||||
"AI navigation": "AI navigation",
|
|
||||||
"Breadcrumb": "Breadcrumb",
|
|
||||||
"Skip to main content": "Skip to main content"
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "Editar comentario",
|
"Edit comment": "Editar comentario",
|
||||||
"Delete comment": "Eliminar comentario",
|
"Delete comment": "Eliminar comentario",
|
||||||
"Are you sure you want to delete this comment?": "¿Está seguro de que desea eliminar este comentario?",
|
"Are you sure you want to delete this comment?": "¿Está seguro de que desea eliminar este comentario?",
|
||||||
"Delete chat": "Eliminar chat",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "¿Está seguro de que desea eliminar '{{title}}'? Esta acción no se puede deshacer.",
|
|
||||||
"Comment created successfully": "Comentario creado con éxito",
|
"Comment created successfully": "Comentario creado con éxito",
|
||||||
"Error creating comment": "Error al crear comentario",
|
"Error creating comment": "Error al crear comentario",
|
||||||
"Comment updated successfully": "Comentario actualizado con éxito",
|
"Comment updated successfully": "Comentario actualizado con éxito",
|
||||||
@@ -741,93 +739,6 @@
|
|||||||
"Removed page restriction": "Restricción de página eliminada",
|
"Removed page restriction": "Restricción de página eliminada",
|
||||||
"Added page permission": "Permiso de página añadido",
|
"Added page permission": "Permiso de página añadido",
|
||||||
"Removed page permission": "Permiso de página eliminado",
|
"Removed page permission": "Permiso de página eliminado",
|
||||||
"day": "día",
|
|
||||||
"days": "días",
|
|
||||||
"week": "semana",
|
|
||||||
"weeks": "semanas",
|
|
||||||
"month": "mes",
|
|
||||||
"months": "meses",
|
|
||||||
"year": "año",
|
|
||||||
"years": "años",
|
|
||||||
"Period": "Período",
|
|
||||||
"Fixed date": "Fecha fija",
|
|
||||||
"Indefinitely": "Indefinidamente",
|
|
||||||
"Days": "Días",
|
|
||||||
"Weeks": "Semanas",
|
|
||||||
"Months": "Meses",
|
|
||||||
"Years": "Años",
|
|
||||||
"Pick a date": "Selecciona una fecha",
|
|
||||||
"Maximum is {{max}} {{unit}} for this unit": "El máximo es {{max}} {{unit}} para esta unidad",
|
|
||||||
"Never expires. Verifiers can re-verify at any time.": "Nunca caduca. Los verificadores pueden volver a verificar en cualquier momento.",
|
|
||||||
"Verified": "Verificado",
|
|
||||||
"Review needed": "Revisión necesaria",
|
|
||||||
"Verification expired": "La verificación ha caducado",
|
|
||||||
"Draft": "Borrador",
|
|
||||||
"In Approval": "En aprobación",
|
|
||||||
"In approval": "En aprobación",
|
|
||||||
"Approved": "Aprobado",
|
|
||||||
"Obsolete": "Obsoleto",
|
|
||||||
"Expiring": "Próximo a caducar",
|
|
||||||
"Set up verification": "Configurar verificación",
|
|
||||||
"Verify page": "Verificar página",
|
|
||||||
"Page verification": "Verificación de página",
|
|
||||||
"Add verification": "Añadir verificación",
|
|
||||||
"Edit verification": "Editar verificación",
|
|
||||||
"Search by title": "Buscar por título",
|
|
||||||
"Choose how this page should stay accurate.": "Elige cómo debe mantenerse precisa esta página.",
|
|
||||||
"Recurring verification": "Verificación periódica",
|
|
||||||
"Verifiers re-confirm this page on a schedule.": "Los verificadores vuelven a confirmar esta página según una programación.",
|
|
||||||
"Re-verify on a schedule (e.g every 30 days )": "Volver a verificar según una programación (p. ej., cada 30 días)",
|
|
||||||
"Page stays editable at all times": "La página permanece editable en todo momento",
|
|
||||||
"Best for runbooks, FAQs, living documentation": "Ideal para runbooks, preguntas frecuentes y documentación viva",
|
|
||||||
"Approval workflow": "Flujo de aprobación",
|
|
||||||
"Formal document lifecycle with named approvers.": "Ciclo de vida formal del documento con aprobadores designados.",
|
|
||||||
"Draft → In approval → Approved → Obsolete": "Borrador → En aprobación → Aprobado → Obsoleto",
|
|
||||||
"Locked once approved, with full history": "Bloqueado una vez aprobado, con historial completo",
|
|
||||||
"Designed for ISO 9001, ISO 13485, and FDA": "Diseñado para ISO 9001, ISO 13485 y FDA",
|
|
||||||
"Best for SOPs and controlled documents": "Ideal para SOP y documentos controlados",
|
|
||||||
"Back": "Atrás",
|
|
||||||
"Quality management": "Gestión de calidad",
|
|
||||||
"Recurring": "Periódica",
|
|
||||||
"Pages move through draft, approval, and approved stages.": "Las páginas pasan por las etapas de borrador, aprobación y aprobado.",
|
|
||||||
"Verifiers": "Verificadores",
|
|
||||||
"Add verifier": "Añadir verificador",
|
|
||||||
"I've reviewed this page for accuracy": "He revisado la exactitud de esta página",
|
|
||||||
"Set up": "Configurar",
|
|
||||||
"Remove verification": "Eliminar verificación",
|
|
||||||
"Are you sure you want to remove verification from this page?": "¿Seguro que quieres eliminar la verificación de esta página?",
|
|
||||||
"Assigned verifiers must periodically re-verify this page.": "Los verificadores asignados deben volver a verificar esta página periódicamente.",
|
|
||||||
"Last verified by {{name}} {{time}} (expired)": "Última verificación por {{name}} {{time}} (caducada)",
|
|
||||||
"The fixed expiration date has passed.": "La fecha fija de vencimiento ya pasó.",
|
|
||||||
"Verified by {{name}} {{time}}": "Verificado por {{name}} {{time}}",
|
|
||||||
"Expires {{date}}": "Caduca el {{date}}",
|
|
||||||
"Expired {{date}}": "Caducó el {{date}}",
|
|
||||||
"Mark as obsolete": "Marcar como obsoleto",
|
|
||||||
"Mark obsolete": "Marcar como obsoleto",
|
|
||||||
"Returned by {{name}} {{time}}": "Devuelto por {{name}} {{time}}",
|
|
||||||
"No approval has been requested yet.": "Aún no se ha solicitado aprobación.",
|
|
||||||
"Submitted by {{name}} {{time}}": "Enviado por {{name}} {{time}}",
|
|
||||||
"Someone": "Alguien",
|
|
||||||
"Approved by {{name}} {{time}}": "Aprobado por {{name}} {{time}}",
|
|
||||||
"This document has been marked as obsolete.": "Este documento ha sido marcado como obsoleto.",
|
|
||||||
"Rejection comment": "Comentario de rechazo",
|
|
||||||
"Reason for returning this document...": "Motivo de la devolución de este documento...",
|
|
||||||
"Confirm rejection": "Confirmar rechazo",
|
|
||||||
"Submit for approval": "Enviar para aprobación",
|
|
||||||
"Reject": "Rechazar",
|
|
||||||
"Approve": "Aprobar",
|
|
||||||
"Re-submit for approval": "Volver a enviar para aprobación",
|
|
||||||
"Verified until": "Verificado hasta",
|
|
||||||
"QMS": "SGC",
|
|
||||||
"Verified pages": "Páginas verificadas",
|
|
||||||
"Search pages...": "Buscar páginas...",
|
|
||||||
"Filter by space": "Filtrar por espacio",
|
|
||||||
"Filter by type": "Filtrar por tipo",
|
|
||||||
"<bold>{{name}}</bold> verified a page": "<bold>{{name}}</bold> verificó una página",
|
|
||||||
"<bold>{{name}}</bold> submitted a page for your approval": "<bold>{{name}}</bold> envió una página para tu aprobación",
|
|
||||||
"<bold>{{name}}</bold> returned a page for revision": "<bold>{{name}}</bold> devolvió una página para revisión",
|
|
||||||
"Page verification expires soon": "La verificación de la página caduca pronto",
|
|
||||||
"Page verification has expired": "La verificación de la página ha caducado",
|
|
||||||
"Verifying your email": "Verificando tu correo electrónico",
|
"Verifying your email": "Verificando tu correo electrónico",
|
||||||
"Please wait...": "Por favor, espera...",
|
"Please wait...": "Por favor, espera...",
|
||||||
"Verification failed. The link may have expired.": "La verificación ha fallado. Es posible que el enlace haya expirado.",
|
"Verification failed. The link may have expired.": "La verificación ha fallado. Es posible que el enlace haya expirado.",
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "Modifier le commentaire",
|
"Edit comment": "Modifier le commentaire",
|
||||||
"Delete comment": "Supprimer le commentaire",
|
"Delete comment": "Supprimer le commentaire",
|
||||||
"Are you sure you want to delete this comment?": "Êtes-vous sûr de vouloir supprimer ce commentaire ?",
|
"Are you sure you want to delete this comment?": "Êtes-vous sûr de vouloir supprimer ce commentaire ?",
|
||||||
"Delete chat": "Supprimer la conversation",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "Êtes-vous sûr de vouloir supprimer '{{title}}' ? Cette action est irréversible.",
|
|
||||||
"Comment created successfully": "Commentaire créé avec succès",
|
"Comment created successfully": "Commentaire créé avec succès",
|
||||||
"Error creating comment": "Erreur lors de la création du commentaire",
|
"Error creating comment": "Erreur lors de la création du commentaire",
|
||||||
"Comment updated successfully": "Commentaire mis à jour avec succès",
|
"Comment updated successfully": "Commentaire mis à jour avec succès",
|
||||||
@@ -741,93 +739,6 @@
|
|||||||
"Removed page restriction": "Restriction de la page supprimée",
|
"Removed page restriction": "Restriction de la page supprimée",
|
||||||
"Added page permission": "Autorisation de la page ajoutée",
|
"Added page permission": "Autorisation de la page ajoutée",
|
||||||
"Removed page permission": "Autorisation de la page supprimée",
|
"Removed page permission": "Autorisation de la page supprimée",
|
||||||
"day": "jour",
|
|
||||||
"days": "jours",
|
|
||||||
"week": "semaine",
|
|
||||||
"weeks": "semaines",
|
|
||||||
"month": "mois",
|
|
||||||
"months": "mois",
|
|
||||||
"year": "an",
|
|
||||||
"years": "ans",
|
|
||||||
"Period": "Période",
|
|
||||||
"Fixed date": "Date fixe",
|
|
||||||
"Indefinitely": "Indéfiniment",
|
|
||||||
"Days": "Jours",
|
|
||||||
"Weeks": "Semaines",
|
|
||||||
"Months": "Mois",
|
|
||||||
"Years": "Ans",
|
|
||||||
"Pick a date": "Choisir une date",
|
|
||||||
"Maximum is {{max}} {{unit}} for this unit": "Le maximum est de {{max}} {{unit}} pour cette unité",
|
|
||||||
"Never expires. Verifiers can re-verify at any time.": "N’expire jamais. Les vérificateurs peuvent revérifier à tout moment.",
|
|
||||||
"Verified": "Vérifié",
|
|
||||||
"Review needed": "Révision nécessaire",
|
|
||||||
"Verification expired": "Vérification expirée",
|
|
||||||
"Draft": "Brouillon",
|
|
||||||
"In Approval": "En approbation",
|
|
||||||
"In approval": "En approbation",
|
|
||||||
"Approved": "Approuvé",
|
|
||||||
"Obsolete": "Obsolète",
|
|
||||||
"Expiring": "Expire bientôt",
|
|
||||||
"Set up verification": "Configurer la vérification",
|
|
||||||
"Verify page": "Vérifier la page",
|
|
||||||
"Page verification": "Vérification de la page",
|
|
||||||
"Add verification": "Ajouter une vérification",
|
|
||||||
"Edit verification": "Modifier la vérification",
|
|
||||||
"Search by title": "Rechercher par titre",
|
|
||||||
"Choose how this page should stay accurate.": "Choisissez comment cette page doit rester exacte.",
|
|
||||||
"Recurring verification": "Vérification récurrente",
|
|
||||||
"Verifiers re-confirm this page on a schedule.": "Les vérificateurs reconfirment cette page selon une fréquence définie.",
|
|
||||||
"Re-verify on a schedule (e.g every 30 days )": "Revérifier selon une fréquence définie (p. ex. tous les 30 jours)",
|
|
||||||
"Page stays editable at all times": "La page reste modifiable en permanence",
|
|
||||||
"Best for runbooks, FAQs, living documentation": "Idéal pour les runbooks, FAQ et la documentation évolutive",
|
|
||||||
"Approval workflow": "Flux d’approbation",
|
|
||||||
"Formal document lifecycle with named approvers.": "Cycle de vie formel du document avec des approbateurs désignés.",
|
|
||||||
"Draft → In approval → Approved → Obsolete": "Brouillon → En approbation → Approuvé → Obsolète",
|
|
||||||
"Locked once approved, with full history": "Verrouillé une fois approuvé, avec historique complet",
|
|
||||||
"Designed for ISO 9001, ISO 13485, and FDA": "Conçu pour l’ISO 9001, l’ISO 13485 et la FDA",
|
|
||||||
"Best for SOPs and controlled documents": "Idéal pour les SOP et les documents contrôlés",
|
|
||||||
"Back": "Retour",
|
|
||||||
"Quality management": "Gestion de la qualité",
|
|
||||||
"Recurring": "Récurrent",
|
|
||||||
"Pages move through draft, approval, and approved stages.": "Les pages passent par les étapes brouillon, approbation et approuvé.",
|
|
||||||
"Verifiers": "Vérificateurs",
|
|
||||||
"Add verifier": "Ajouter un vérificateur",
|
|
||||||
"I've reviewed this page for accuracy": "J’ai vérifié l’exactitude de cette page",
|
|
||||||
"Set up": "Configurer",
|
|
||||||
"Remove verification": "Supprimer la vérification",
|
|
||||||
"Are you sure you want to remove verification from this page?": "Voulez-vous vraiment supprimer la vérification de cette page ?",
|
|
||||||
"Assigned verifiers must periodically re-verify this page.": "Les vérificateurs assignés doivent revérifier périodiquement cette page.",
|
|
||||||
"Last verified by {{name}} {{time}} (expired)": "Dernière vérification par {{name}} {{time}} (expirée)",
|
|
||||||
"The fixed expiration date has passed.": "La date d’expiration fixe est passée.",
|
|
||||||
"Verified by {{name}} {{time}}": "Vérifié par {{name}} {{time}}",
|
|
||||||
"Expires {{date}}": "Expire le {{date}}",
|
|
||||||
"Expired {{date}}": "Expiré le {{date}}",
|
|
||||||
"Mark as obsolete": "Marquer comme obsolète",
|
|
||||||
"Mark obsolete": "Marquer comme obsolète",
|
|
||||||
"Returned by {{name}} {{time}}": "Renvoyé par {{name}} {{time}}",
|
|
||||||
"No approval has been requested yet.": "Aucune approbation n’a encore été demandée.",
|
|
||||||
"Submitted by {{name}} {{time}}": "Soumis par {{name}} {{time}}",
|
|
||||||
"Someone": "Quelqu’un",
|
|
||||||
"Approved by {{name}} {{time}}": "Approuvé par {{name}} {{time}}",
|
|
||||||
"This document has been marked as obsolete.": "Ce document a été marqué comme obsolète.",
|
|
||||||
"Rejection comment": "Commentaire de rejet",
|
|
||||||
"Reason for returning this document...": "Raison du renvoi de ce document...",
|
|
||||||
"Confirm rejection": "Confirmer le rejet",
|
|
||||||
"Submit for approval": "Soumettre pour approbation",
|
|
||||||
"Reject": "Rejeter",
|
|
||||||
"Approve": "Approuver",
|
|
||||||
"Re-submit for approval": "Soumettre à nouveau pour approbation",
|
|
||||||
"Verified until": "Vérifié jusqu’au",
|
|
||||||
"QMS": "SMQ",
|
|
||||||
"Verified pages": "Pages vérifiées",
|
|
||||||
"Search pages...": "Rechercher des pages...",
|
|
||||||
"Filter by space": "Filtrer par espace",
|
|
||||||
"Filter by type": "Filtrer par type",
|
|
||||||
"<bold>{{name}}</bold> verified a page": "<bold>{{name}}</bold> a vérifié une page",
|
|
||||||
"<bold>{{name}}</bold> submitted a page for your approval": "<bold>{{name}}</bold> a soumis une page à votre approbation",
|
|
||||||
"<bold>{{name}}</bold> returned a page for revision": "<bold>{{name}}</bold> a renvoyé une page pour révision",
|
|
||||||
"Page verification expires soon": "La vérification de la page expire bientôt",
|
|
||||||
"Page verification has expired": "La vérification de la page a expiré",
|
|
||||||
"Verifying your email": "Vérification de votre e-mail",
|
"Verifying your email": "Vérification de votre e-mail",
|
||||||
"Please wait...": "Veuillez patienter...",
|
"Please wait...": "Veuillez patienter...",
|
||||||
"Verification failed. The link may have expired.": "Échec de la vérification. Le lien a peut-être expiré.",
|
"Verification failed. The link may have expired.": "Échec de la vérification. Le lien a peut-être expiré.",
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "Modifica commento",
|
"Edit comment": "Modifica commento",
|
||||||
"Delete comment": "Elimina commento",
|
"Delete comment": "Elimina commento",
|
||||||
"Are you sure you want to delete this comment?": "Sei sicuro di voler eliminare questo commento?",
|
"Are you sure you want to delete this comment?": "Sei sicuro di voler eliminare questo commento?",
|
||||||
"Delete chat": "Elimina chat",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "Sei sicuro di voler eliminare '{{title}}'? Questa azione non può essere annullata.",
|
|
||||||
"Comment created successfully": "Commento creato con successo",
|
"Comment created successfully": "Commento creato con successo",
|
||||||
"Error creating comment": "Si è verificato un errore durante la creazione del commento",
|
"Error creating comment": "Si è verificato un errore durante la creazione del commento",
|
||||||
"Comment updated successfully": "Commento aggiornato con successo",
|
"Comment updated successfully": "Commento aggiornato con successo",
|
||||||
@@ -741,93 +739,6 @@
|
|||||||
"Removed page restriction": "Restrizione della pagina rimossa",
|
"Removed page restriction": "Restrizione della pagina rimossa",
|
||||||
"Added page permission": "Permesso sulla pagina aggiunto",
|
"Added page permission": "Permesso sulla pagina aggiunto",
|
||||||
"Removed page permission": "Permesso sulla pagina rimosso",
|
"Removed page permission": "Permesso sulla pagina rimosso",
|
||||||
"day": "giorno",
|
|
||||||
"days": "giorni",
|
|
||||||
"week": "settimana",
|
|
||||||
"weeks": "settimane",
|
|
||||||
"month": "mese",
|
|
||||||
"months": "mesi",
|
|
||||||
"year": "anno",
|
|
||||||
"years": "anni",
|
|
||||||
"Period": "Periodo",
|
|
||||||
"Fixed date": "Data fissa",
|
|
||||||
"Indefinitely": "A tempo indeterminato",
|
|
||||||
"Days": "Giorni",
|
|
||||||
"Weeks": "Settimane",
|
|
||||||
"Months": "Mesi",
|
|
||||||
"Years": "Anni",
|
|
||||||
"Pick a date": "Scegli una data",
|
|
||||||
"Maximum is {{max}} {{unit}} for this unit": "Il massimo consentito è {{max}} {{unit}} per questa unità",
|
|
||||||
"Never expires. Verifiers can re-verify at any time.": "Non scade mai. I verificatori possono verificare nuovamente in qualsiasi momento.",
|
|
||||||
"Verified": "Verificato",
|
|
||||||
"Review needed": "Revisione necessaria",
|
|
||||||
"Verification expired": "Verifica scaduta",
|
|
||||||
"Draft": "Bozza",
|
|
||||||
"In Approval": "In approvazione",
|
|
||||||
"In approval": "In approvazione",
|
|
||||||
"Approved": "Approvato",
|
|
||||||
"Obsolete": "Obsoleto",
|
|
||||||
"Expiring": "In scadenza",
|
|
||||||
"Set up verification": "Configura la verifica",
|
|
||||||
"Verify page": "Verifica la pagina",
|
|
||||||
"Page verification": "Verifica della pagina",
|
|
||||||
"Add verification": "Aggiungi verifica",
|
|
||||||
"Edit verification": "Modifica verifica",
|
|
||||||
"Search by title": "Cerca per titolo",
|
|
||||||
"Choose how this page should stay accurate.": "Scegli come mantenere accurata questa pagina.",
|
|
||||||
"Recurring verification": "Verifica ricorrente",
|
|
||||||
"Verifiers re-confirm this page on a schedule.": "I verificatori riconfermano questa pagina secondo una pianificazione.",
|
|
||||||
"Re-verify on a schedule (e.g every 30 days )": "Verifica nuovamente secondo una pianificazione (ad es. ogni 30 giorni)",
|
|
||||||
"Page stays editable at all times": "La pagina resta sempre modificabile",
|
|
||||||
"Best for runbooks, FAQs, living documentation": "Ideale per runbook, FAQ e documentazione dinamica",
|
|
||||||
"Approval workflow": "Flusso di approvazione",
|
|
||||||
"Formal document lifecycle with named approvers.": "Ciclo di vita formale del documento con approvatori nominati.",
|
|
||||||
"Draft → In approval → Approved → Obsolete": "Bozza → In approvazione → Approvato → Obsoleto",
|
|
||||||
"Locked once approved, with full history": "Bloccato una volta approvato, con cronologia completa",
|
|
||||||
"Designed for ISO 9001, ISO 13485, and FDA": "Progettato per ISO 9001, ISO 13485 e FDA",
|
|
||||||
"Best for SOPs and controlled documents": "Ideale per SOP e documenti controllati",
|
|
||||||
"Back": "Indietro",
|
|
||||||
"Quality management": "Gestione della qualità",
|
|
||||||
"Recurring": "Ricorrente",
|
|
||||||
"Pages move through draft, approval, and approved stages.": "Le pagine passano attraverso le fasi di bozza, approvazione e approvato.",
|
|
||||||
"Verifiers": "Verificatori",
|
|
||||||
"Add verifier": "Aggiungi verificatore",
|
|
||||||
"I've reviewed this page for accuracy": "Ho controllato l'accuratezza di questa pagina",
|
|
||||||
"Set up": "Configura",
|
|
||||||
"Remove verification": "Rimuovi verifica",
|
|
||||||
"Are you sure you want to remove verification from this page?": "Sei sicuro di voler rimuovere la verifica da questa pagina?",
|
|
||||||
"Assigned verifiers must periodically re-verify this page.": "I verificatori assegnati devono verificare nuovamente questa pagina periodicamente.",
|
|
||||||
"Last verified by {{name}} {{time}} (expired)": "Ultima verifica effettuata da {{name}} {{time}} (scaduta)",
|
|
||||||
"The fixed expiration date has passed.": "La data di scadenza fissa è trascorsa.",
|
|
||||||
"Verified by {{name}} {{time}}": "Verificato da {{name}} {{time}}",
|
|
||||||
"Expires {{date}}": "Scade il {{date}}",
|
|
||||||
"Expired {{date}}": "Scaduto il {{date}}",
|
|
||||||
"Mark as obsolete": "Contrassegna come obsoleto",
|
|
||||||
"Mark obsolete": "Contrassegna come obsoleto",
|
|
||||||
"Returned by {{name}} {{time}}": "Restituito da {{name}} {{time}}",
|
|
||||||
"No approval has been requested yet.": "Non è stata ancora richiesta alcuna approvazione.",
|
|
||||||
"Submitted by {{name}} {{time}}": "Inviato da {{name}} {{time}}",
|
|
||||||
"Someone": "Qualcuno",
|
|
||||||
"Approved by {{name}} {{time}}": "Approvato da {{name}} {{time}}",
|
|
||||||
"This document has been marked as obsolete.": "Questo documento è stato contrassegnato come obsoleto.",
|
|
||||||
"Rejection comment": "Commento di rifiuto",
|
|
||||||
"Reason for returning this document...": "Motivo della restituzione di questo documento...",
|
|
||||||
"Confirm rejection": "Conferma rifiuto",
|
|
||||||
"Submit for approval": "Invia per approvazione",
|
|
||||||
"Reject": "Rifiuta",
|
|
||||||
"Approve": "Approva",
|
|
||||||
"Re-submit for approval": "Invia nuovamente per approvazione",
|
|
||||||
"Verified until": "Verificato fino al",
|
|
||||||
"QMS": "QMS",
|
|
||||||
"Verified pages": "Pagine verificate",
|
|
||||||
"Search pages...": "Cerca pagine...",
|
|
||||||
"Filter by space": "Filtra per spazio",
|
|
||||||
"Filter by type": "Filtra per tipo",
|
|
||||||
"<bold>{{name}}</bold> verified a page": "<bold>{{name}}</bold> ha verificato una pagina",
|
|
||||||
"<bold>{{name}}</bold> submitted a page for your approval": "<bold>{{name}}</bold> ha inviato una pagina per la tua approvazione",
|
|
||||||
"<bold>{{name}}</bold> returned a page for revision": "<bold>{{name}}</bold> ha restituito una pagina per la revisione",
|
|
||||||
"Page verification expires soon": "La verifica della pagina scadrà presto",
|
|
||||||
"Page verification has expired": "La verifica della pagina è scaduta",
|
|
||||||
"Verifying your email": "Verifica della tua email in corso",
|
"Verifying your email": "Verifica della tua email in corso",
|
||||||
"Please wait...": "Attendere...",
|
"Please wait...": "Attendere...",
|
||||||
"Verification failed. The link may have expired.": "Verifica non riuscita. Il link potrebbe essere scaduto.",
|
"Verification failed. The link may have expired.": "Verifica non riuscita. Il link potrebbe essere scaduto.",
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "コメントを編集する",
|
"Edit comment": "コメントを編集する",
|
||||||
"Delete comment": "コメントを削除する",
|
"Delete comment": "コメントを削除する",
|
||||||
"Are you sure you want to delete this comment?": "このコメントを削除してもよろしいですか?",
|
"Are you sure you want to delete this comment?": "このコメントを削除してもよろしいですか?",
|
||||||
"Delete chat": "チャットを削除",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "「{{title}}」を削除してもよろしいですか?この操作は元に戻せません。",
|
|
||||||
"Comment created successfully": "コメントを作成しました",
|
"Comment created successfully": "コメントを作成しました",
|
||||||
"Error creating comment": "コメントの作成に失敗しました",
|
"Error creating comment": "コメントの作成に失敗しました",
|
||||||
"Comment updated successfully": "コメントを更新しました",
|
"Comment updated successfully": "コメントを更新しました",
|
||||||
@@ -741,93 +739,6 @@
|
|||||||
"Removed page restriction": "ページの制限を解除しました",
|
"Removed page restriction": "ページの制限を解除しました",
|
||||||
"Added page permission": "ページの権限を追加しました",
|
"Added page permission": "ページの権限を追加しました",
|
||||||
"Removed page permission": "ページの権限を削除しました",
|
"Removed page permission": "ページの権限を削除しました",
|
||||||
"day": "日",
|
|
||||||
"days": "日",
|
|
||||||
"week": "週",
|
|
||||||
"weeks": "週",
|
|
||||||
"month": "か月",
|
|
||||||
"months": "か月",
|
|
||||||
"year": "年",
|
|
||||||
"years": "年",
|
|
||||||
"Period": "期間",
|
|
||||||
"Fixed date": "指定日",
|
|
||||||
"Indefinitely": "無期限",
|
|
||||||
"Days": "日",
|
|
||||||
"Weeks": "週",
|
|
||||||
"Months": "か月",
|
|
||||||
"Years": "年",
|
|
||||||
"Pick a date": "日付を選択",
|
|
||||||
"Maximum is {{max}} {{unit}} for this unit": "この単位の最大値は{{max}}{{unit}}です",
|
|
||||||
"Never expires. Verifiers can re-verify at any time.": "有効期限はありません。検証者はいつでも再検証できます。",
|
|
||||||
"Verified": "検証済み",
|
|
||||||
"Review needed": "確認が必要",
|
|
||||||
"Verification expired": "検証期限切れ",
|
|
||||||
"Draft": "下書き",
|
|
||||||
"In Approval": "承認中",
|
|
||||||
"In approval": "承認中",
|
|
||||||
"Approved": "承認済み",
|
|
||||||
"Obsolete": "廃止",
|
|
||||||
"Expiring": "期限間近",
|
|
||||||
"Set up verification": "検証を設定",
|
|
||||||
"Verify page": "ページを検証",
|
|
||||||
"Page verification": "ページ検証",
|
|
||||||
"Add verification": "検証を追加",
|
|
||||||
"Edit verification": "検証を編集",
|
|
||||||
"Search by title": "タイトルで検索",
|
|
||||||
"Choose how this page should stay accurate.": "このページの正確性をどのように維持するか選択してください。",
|
|
||||||
"Recurring verification": "定期検証",
|
|
||||||
"Verifiers re-confirm this page on a schedule.": "検証者がこのページを定期的に再確認します。",
|
|
||||||
"Re-verify on a schedule (e.g every 30 days )": "スケジュールに従って再検証(例:30日ごと)",
|
|
||||||
"Page stays editable at all times": "ページは常に編集可能です",
|
|
||||||
"Best for runbooks, FAQs, living documentation": "運用手順書、FAQ、継続的に更新されるドキュメントに最適",
|
|
||||||
"Approval workflow": "承認ワークフロー",
|
|
||||||
"Formal document lifecycle with named approvers.": "指定された承認者による正式な文書ライフサイクルです。",
|
|
||||||
"Draft → In approval → Approved → Obsolete": "下書き → 承認中 → 承認済み → 廃止",
|
|
||||||
"Locked once approved, with full history": "承認後はロックされ、完全な履歴が残ります",
|
|
||||||
"Designed for ISO 9001, ISO 13485, and FDA": "ISO 9001、ISO 13485、FDA向けに設計",
|
|
||||||
"Best for SOPs and controlled documents": "SOPや管理文書に最適",
|
|
||||||
"Back": "戻る",
|
|
||||||
"Quality management": "品質管理",
|
|
||||||
"Recurring": "定期",
|
|
||||||
"Pages move through draft, approval, and approved stages.": "ページは下書き、承認中、承認済みの各段階を進みます。",
|
|
||||||
"Verifiers": "検証者",
|
|
||||||
"Add verifier": "検証者を追加",
|
|
||||||
"I've reviewed this page for accuracy": "このページの正確性を確認しました",
|
|
||||||
"Set up": "設定",
|
|
||||||
"Remove verification": "検証を削除",
|
|
||||||
"Are you sure you want to remove verification from this page?": "このページから検証を削除してもよろしいですか?",
|
|
||||||
"Assigned verifiers must periodically re-verify this page.": "割り当てられた検証者はこのページを定期的に再検証する必要があります。",
|
|
||||||
"Last verified by {{name}} {{time}} (expired)": "最終検証者:{{name}} {{time}}(期限切れ)",
|
|
||||||
"The fixed expiration date has passed.": "指定された有効期限を過ぎています。",
|
|
||||||
"Verified by {{name}} {{time}}": "{{name}}が{{time}}に検証",
|
|
||||||
"Expires {{date}}": "有効期限:{{date}}",
|
|
||||||
"Expired {{date}}": "{{date}}に期限切れ",
|
|
||||||
"Mark as obsolete": "廃止としてマーク",
|
|
||||||
"Mark obsolete": "廃止にする",
|
|
||||||
"Returned by {{name}} {{time}}": "{{name}}が{{time}}に差し戻し",
|
|
||||||
"No approval has been requested yet.": "まだ承認は依頼されていません。",
|
|
||||||
"Submitted by {{name}} {{time}}": "{{name}}が{{time}}に提出",
|
|
||||||
"Someone": "誰か",
|
|
||||||
"Approved by {{name}} {{time}}": "{{name}}が{{time}}に承認",
|
|
||||||
"This document has been marked as obsolete.": "この文書は廃止としてマークされています。",
|
|
||||||
"Rejection comment": "差し戻しコメント",
|
|
||||||
"Reason for returning this document...": "この文書を差し戻す理由...",
|
|
||||||
"Confirm rejection": "差し戻しを確定",
|
|
||||||
"Submit for approval": "承認を申請",
|
|
||||||
"Reject": "差し戻す",
|
|
||||||
"Approve": "承認",
|
|
||||||
"Re-submit for approval": "再度承認を申請",
|
|
||||||
"Verified until": "検証有効期限",
|
|
||||||
"QMS": "QMS",
|
|
||||||
"Verified pages": "検証済みページ",
|
|
||||||
"Search pages...": "ページを検索...",
|
|
||||||
"Filter by space": "スペースで絞り込み",
|
|
||||||
"Filter by type": "タイプで絞り込み",
|
|
||||||
"<bold>{{name}}</bold> verified a page": "<bold>{{name}}</bold>がページを検証しました",
|
|
||||||
"<bold>{{name}}</bold> submitted a page for your approval": "<bold>{{name}}</bold>があなたの承認のためにページを提出しました",
|
|
||||||
"<bold>{{name}}</bold> returned a page for revision": "<bold>{{name}}</bold>がページを修正のため差し戻しました",
|
|
||||||
"Page verification expires soon": "ページ検証の期限が間もなく切れます",
|
|
||||||
"Page verification has expired": "ページ検証の期限が切れています",
|
|
||||||
"Verifying your email": "メールアドレスを確認しています",
|
"Verifying your email": "メールアドレスを確認しています",
|
||||||
"Please wait...": "お待ちください…",
|
"Please wait...": "お待ちください…",
|
||||||
"Verification failed. The link may have expired.": "認証に失敗しました。リンクの有効期限が切れている可能性があります。",
|
"Verification failed. The link may have expired.": "認証に失敗しました。リンクの有効期限が切れている可能性があります。",
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "댓글 수정",
|
"Edit comment": "댓글 수정",
|
||||||
"Delete comment": "댓글 삭제",
|
"Delete comment": "댓글 삭제",
|
||||||
"Are you sure you want to delete this comment?": "이 댓글을 삭제하시겠습니까?",
|
"Are you sure you want to delete this comment?": "이 댓글을 삭제하시겠습니까?",
|
||||||
"Delete chat": "채팅 삭제",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "'{{title}}'을(를) 삭제하시겠습니까? 이 작업은 되돌릴 수 없습니다.",
|
|
||||||
"Comment created successfully": "댓글 생성 완료",
|
"Comment created successfully": "댓글 생성 완료",
|
||||||
"Error creating comment": "댓글 생성 오류",
|
"Error creating comment": "댓글 생성 오류",
|
||||||
"Comment updated successfully": "댓글 업데이트 완료",
|
"Comment updated successfully": "댓글 업데이트 완료",
|
||||||
@@ -741,93 +739,6 @@
|
|||||||
"Removed page restriction": "페이지 제한이 제거됨",
|
"Removed page restriction": "페이지 제한이 제거됨",
|
||||||
"Added page permission": "페이지 권한이 추가됨",
|
"Added page permission": "페이지 권한이 추가됨",
|
||||||
"Removed page permission": "페이지 권한이 제거됨",
|
"Removed page permission": "페이지 권한이 제거됨",
|
||||||
"day": "일",
|
|
||||||
"days": "일",
|
|
||||||
"week": "주",
|
|
||||||
"weeks": "주",
|
|
||||||
"month": "개월",
|
|
||||||
"months": "개월",
|
|
||||||
"year": "년",
|
|
||||||
"years": "년",
|
|
||||||
"Period": "기간",
|
|
||||||
"Fixed date": "고정 날짜",
|
|
||||||
"Indefinitely": "무기한",
|
|
||||||
"Days": "일",
|
|
||||||
"Weeks": "주",
|
|
||||||
"Months": "개월",
|
|
||||||
"Years": "년",
|
|
||||||
"Pick a date": "날짜 선택",
|
|
||||||
"Maximum is {{max}} {{unit}} for this unit": "이 단위의 최대값은 {{max}} {{unit}}입니다",
|
|
||||||
"Never expires. Verifiers can re-verify at any time.": "만료되지 않습니다. 검증자는 언제든지 다시 검증할 수 있습니다.",
|
|
||||||
"Verified": "검증됨",
|
|
||||||
"Review needed": "검토 필요",
|
|
||||||
"Verification expired": "검증 만료됨",
|
|
||||||
"Draft": "초안",
|
|
||||||
"In Approval": "승인 진행 중",
|
|
||||||
"In approval": "승인 진행 중",
|
|
||||||
"Approved": "승인됨",
|
|
||||||
"Obsolete": "폐기됨",
|
|
||||||
"Expiring": "만료 예정",
|
|
||||||
"Set up verification": "검증 설정",
|
|
||||||
"Verify page": "페이지 검증",
|
|
||||||
"Page verification": "페이지 검증",
|
|
||||||
"Add verification": "검증 추가",
|
|
||||||
"Edit verification": "검증 편집",
|
|
||||||
"Search by title": "제목으로 검색",
|
|
||||||
"Choose how this page should stay accurate.": "이 페이지의 정확성을 유지할 방법을 선택하세요.",
|
|
||||||
"Recurring verification": "반복 검증",
|
|
||||||
"Verifiers re-confirm this page on a schedule.": "검증자가 일정에 따라 이 페이지를 다시 확인합니다.",
|
|
||||||
"Re-verify on a schedule (e.g every 30 days )": "일정에 따라 다시 검증(예: 30일마다)",
|
|
||||||
"Page stays editable at all times": "페이지는 항상 편집 가능합니다",
|
|
||||||
"Best for runbooks, FAQs, living documentation": "런북, FAQ, 살아 있는 문서에 적합",
|
|
||||||
"Approval workflow": "승인 워크플로",
|
|
||||||
"Formal document lifecycle with named approvers.": "지정된 승인자가 있는 공식 문서 수명 주기입니다.",
|
|
||||||
"Draft → In approval → Approved → Obsolete": "초안 → 승인 진행 중 → 승인됨 → 폐기됨",
|
|
||||||
"Locked once approved, with full history": "승인되면 잠기며 전체 이력이 유지됩니다",
|
|
||||||
"Designed for ISO 9001, ISO 13485, and FDA": "ISO 9001, ISO 13485 및 FDA용으로 설계됨",
|
|
||||||
"Best for SOPs and controlled documents": "SOP 및 관리 문서에 적합",
|
|
||||||
"Back": "뒤로",
|
|
||||||
"Quality management": "품질 관리",
|
|
||||||
"Recurring": "반복",
|
|
||||||
"Pages move through draft, approval, and approved stages.": "페이지는 초안, 승인, 승인됨 단계를 거칩니다.",
|
|
||||||
"Verifiers": "검증자",
|
|
||||||
"Add verifier": "검증자 추가",
|
|
||||||
"I've reviewed this page for accuracy": "이 페이지의 정확성을 검토했습니다",
|
|
||||||
"Set up": "설정",
|
|
||||||
"Remove verification": "검증 제거",
|
|
||||||
"Are you sure you want to remove verification from this page?": "이 페이지에서 검증을 제거하시겠습니까?",
|
|
||||||
"Assigned verifiers must periodically re-verify this page.": "지정된 검증자는 이 페이지를 주기적으로 다시 검증해야 합니다.",
|
|
||||||
"Last verified by {{name}} {{time}} (expired)": "마지막 검증자: {{name}} {{time}} (만료됨)",
|
|
||||||
"The fixed expiration date has passed.": "고정된 만료일이 지났습니다.",
|
|
||||||
"Verified by {{name}} {{time}}": "검증자: {{name}} {{time}}",
|
|
||||||
"Expires {{date}}": "{{date}}에 만료",
|
|
||||||
"Expired {{date}}": "{{date}}에 만료됨",
|
|
||||||
"Mark as obsolete": "폐기로 표시",
|
|
||||||
"Mark obsolete": "폐기 표시",
|
|
||||||
"Returned by {{name}} {{time}}": "반려자: {{name}} {{time}}",
|
|
||||||
"No approval has been requested yet.": "아직 승인이 요청되지 않았습니다.",
|
|
||||||
"Submitted by {{name}} {{time}}": "제출자: {{name}} {{time}}",
|
|
||||||
"Someone": "누군가",
|
|
||||||
"Approved by {{name}} {{time}}": "승인자: {{name}} {{time}}",
|
|
||||||
"This document has been marked as obsolete.": "이 문서는 폐기로 표시되었습니다.",
|
|
||||||
"Rejection comment": "반려 사유",
|
|
||||||
"Reason for returning this document...": "이 문서를 반려하는 이유...",
|
|
||||||
"Confirm rejection": "반려 확인",
|
|
||||||
"Submit for approval": "승인 요청",
|
|
||||||
"Reject": "반려",
|
|
||||||
"Approve": "승인",
|
|
||||||
"Re-submit for approval": "승인 재요청",
|
|
||||||
"Verified until": "다음까지 검증됨",
|
|
||||||
"QMS": "QMS",
|
|
||||||
"Verified pages": "검증된 페이지",
|
|
||||||
"Search pages...": "페이지 검색...",
|
|
||||||
"Filter by space": "스페이스별 필터",
|
|
||||||
"Filter by type": "유형별 필터",
|
|
||||||
"<bold>{{name}}</bold> verified a page": "<bold>{{name}}</bold>님이 페이지를 검증했습니다",
|
|
||||||
"<bold>{{name}}</bold> submitted a page for your approval": "<bold>{{name}}</bold>님이 승인을 위해 페이지를 제출했습니다",
|
|
||||||
"<bold>{{name}}</bold> returned a page for revision": "<bold>{{name}}</bold>님이 수정을 위해 페이지를 반려했습니다",
|
|
||||||
"Page verification expires soon": "페이지 검토가 곧 만료됩니다",
|
|
||||||
"Page verification has expired": "페이지 검토가 만료되었습니다",
|
|
||||||
"Verifying your email": "이메일 확인 중",
|
"Verifying your email": "이메일 확인 중",
|
||||||
"Please wait...": "잠시만 기다려 주세요...",
|
"Please wait...": "잠시만 기다려 주세요...",
|
||||||
"Verification failed. The link may have expired.": "인증에 실패했습니다. 링크가 만료되었을 수 있습니다.",
|
"Verification failed. The link may have expired.": "인증에 실패했습니다. 링크가 만료되었을 수 있습니다.",
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "Bewerk reactie",
|
"Edit comment": "Bewerk reactie",
|
||||||
"Delete comment": "Verwijder reactie",
|
"Delete comment": "Verwijder reactie",
|
||||||
"Are you sure you want to delete this comment?": "Weet je zeker dat je deze reactie wilt verwijderen?",
|
"Are you sure you want to delete this comment?": "Weet je zeker dat je deze reactie wilt verwijderen?",
|
||||||
"Delete chat": "Chat verwijderen",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "Weet je zeker dat je '{{title}}' wilt verwijderen? Deze actie kan niet ongedaan worden gemaakt.",
|
|
||||||
"Comment created successfully": "Reactie succesvol aangemaakt",
|
"Comment created successfully": "Reactie succesvol aangemaakt",
|
||||||
"Error creating comment": "Fout bij het aanmaken van reactie",
|
"Error creating comment": "Fout bij het aanmaken van reactie",
|
||||||
"Comment updated successfully": "Opmerking succesvol bijgewerkt",
|
"Comment updated successfully": "Opmerking succesvol bijgewerkt",
|
||||||
@@ -741,93 +739,6 @@
|
|||||||
"Removed page restriction": "Pagina-restrictie verwijderd",
|
"Removed page restriction": "Pagina-restrictie verwijderd",
|
||||||
"Added page permission": "Paginatoestemming toegevoegd",
|
"Added page permission": "Paginatoestemming toegevoegd",
|
||||||
"Removed page permission": "Paginatoestemming verwijderd",
|
"Removed page permission": "Paginatoestemming verwijderd",
|
||||||
"day": "dag",
|
|
||||||
"days": "dagen",
|
|
||||||
"week": "week",
|
|
||||||
"weeks": "weken",
|
|
||||||
"month": "maand",
|
|
||||||
"months": "maanden",
|
|
||||||
"year": "jaar",
|
|
||||||
"years": "jaren",
|
|
||||||
"Period": "Periode",
|
|
||||||
"Fixed date": "Vaste datum",
|
|
||||||
"Indefinitely": "Voor onbepaalde tijd",
|
|
||||||
"Days": "Dagen",
|
|
||||||
"Weeks": "Weken",
|
|
||||||
"Months": "Maanden",
|
|
||||||
"Years": "Jaren",
|
|
||||||
"Pick a date": "Kies een datum",
|
|
||||||
"Maximum is {{max}} {{unit}} for this unit": "Maximum is {{max}} {{unit}} voor deze eenheid",
|
|
||||||
"Never expires. Verifiers can re-verify at any time.": "Verloopt nooit. Verificateurs kunnen op elk moment opnieuw verifiëren.",
|
|
||||||
"Verified": "Geverifieerd",
|
|
||||||
"Review needed": "Beoordeling nodig",
|
|
||||||
"Verification expired": "Verificatie verlopen",
|
|
||||||
"Draft": "Concept",
|
|
||||||
"In Approval": "In goedkeuring",
|
|
||||||
"In approval": "In goedkeuring",
|
|
||||||
"Approved": "Goedgekeurd",
|
|
||||||
"Obsolete": "Verouderd",
|
|
||||||
"Expiring": "Verloopt binnenkort",
|
|
||||||
"Set up verification": "Verificatie instellen",
|
|
||||||
"Verify page": "Pagina verifiëren",
|
|
||||||
"Page verification": "Paginaverificatie",
|
|
||||||
"Add verification": "Verificatie toevoegen",
|
|
||||||
"Edit verification": "Verificatie bewerken",
|
|
||||||
"Search by title": "Zoeken op titel",
|
|
||||||
"Choose how this page should stay accurate.": "Kies hoe deze pagina accuraat moet blijven.",
|
|
||||||
"Recurring verification": "Terugkerende verificatie",
|
|
||||||
"Verifiers re-confirm this page on a schedule.": "Verificateurs bevestigen deze pagina opnieuw volgens een schema.",
|
|
||||||
"Re-verify on a schedule (e.g every 30 days )": "Opnieuw verifiëren volgens een schema (bijv. elke 30 dagen)",
|
|
||||||
"Page stays editable at all times": "Pagina blijft altijd bewerkbaar",
|
|
||||||
"Best for runbooks, FAQs, living documentation": "Het beste voor runbooks, veelgestelde vragen en levende documentatie",
|
|
||||||
"Approval workflow": "Goedkeuringsworkflow",
|
|
||||||
"Formal document lifecycle with named approvers.": "Formele documentlevenscyclus met benoemde goedkeurders.",
|
|
||||||
"Draft → In approval → Approved → Obsolete": "Concept → In goedkeuring → Goedgekeurd → Verouderd",
|
|
||||||
"Locked once approved, with full history": "Vergrendeld zodra goedgekeurd, met volledige geschiedenis",
|
|
||||||
"Designed for ISO 9001, ISO 13485, and FDA": "Ontworpen voor ISO 9001, ISO 13485 en FDA",
|
|
||||||
"Best for SOPs and controlled documents": "Het beste voor SOP's en beheerde documenten",
|
|
||||||
"Back": "Terug",
|
|
||||||
"Quality management": "Kwaliteitsmanagement",
|
|
||||||
"Recurring": "Terugkerend",
|
|
||||||
"Pages move through draft, approval, and approved stages.": "Pagina's doorlopen de fasen concept, goedkeuring en goedgekeurd.",
|
|
||||||
"Verifiers": "Verificateurs",
|
|
||||||
"Add verifier": "Verificateur toevoegen",
|
|
||||||
"I've reviewed this page for accuracy": "Ik heb deze pagina op nauwkeurigheid beoordeeld",
|
|
||||||
"Set up": "Instellen",
|
|
||||||
"Remove verification": "Verificatie verwijderen",
|
|
||||||
"Are you sure you want to remove verification from this page?": "Weet je zeker dat je verificatie van deze pagina wilt verwijderen?",
|
|
||||||
"Assigned verifiers must periodically re-verify this page.": "Toegewezen verificateurs moeten deze pagina periodiek opnieuw verifiëren.",
|
|
||||||
"Last verified by {{name}} {{time}} (expired)": "Laatst geverifieerd door {{name}} {{time}} (verlopen)",
|
|
||||||
"The fixed expiration date has passed.": "De vaste vervaldatum is verstreken.",
|
|
||||||
"Verified by {{name}} {{time}}": "Geverifieerd door {{name}} {{time}}",
|
|
||||||
"Expires {{date}}": "Verloopt op {{date}}",
|
|
||||||
"Expired {{date}}": "Verlopen op {{date}}",
|
|
||||||
"Mark as obsolete": "Markeren als verouderd",
|
|
||||||
"Mark obsolete": "Markeer als verouderd",
|
|
||||||
"Returned by {{name}} {{time}}": "Teruggestuurd door {{name}} {{time}}",
|
|
||||||
"No approval has been requested yet.": "Er is nog geen goedkeuring aangevraagd.",
|
|
||||||
"Submitted by {{name}} {{time}}": "Ingediend door {{name}} {{time}}",
|
|
||||||
"Someone": "Iemand",
|
|
||||||
"Approved by {{name}} {{time}}": "Goedgekeurd door {{name}} {{time}}",
|
|
||||||
"This document has been marked as obsolete.": "Dit document is als verouderd gemarkeerd.",
|
|
||||||
"Rejection comment": "Afwijzingsopmerking",
|
|
||||||
"Reason for returning this document...": "Reden om dit document terug te sturen...",
|
|
||||||
"Confirm rejection": "Afwijzing bevestigen",
|
|
||||||
"Submit for approval": "Indienen voor goedkeuring",
|
|
||||||
"Reject": "Afwijzen",
|
|
||||||
"Approve": "Goedkeuren",
|
|
||||||
"Re-submit for approval": "Opnieuw indienen voor goedkeuring",
|
|
||||||
"Verified until": "Geverifieerd tot",
|
|
||||||
"QMS": "QMS",
|
|
||||||
"Verified pages": "Geverifieerde pagina's",
|
|
||||||
"Search pages...": "Pagina's zoeken...",
|
|
||||||
"Filter by space": "Filteren op ruimte",
|
|
||||||
"Filter by type": "Filteren op type",
|
|
||||||
"<bold>{{name}}</bold> verified a page": "<bold>{{name}}</bold> heeft een pagina geverifieerd",
|
|
||||||
"<bold>{{name}}</bold> submitted a page for your approval": "<bold>{{name}}</bold> heeft een pagina voor jouw goedkeuring ingediend",
|
|
||||||
"<bold>{{name}}</bold> returned a page for revision": "<bold>{{name}}</bold> heeft een pagina teruggestuurd voor revisie",
|
|
||||||
"Page verification expires soon": "Paginaverificatie verloopt binnenkort",
|
|
||||||
"Page verification has expired": "Paginaverificatie is verlopen",
|
|
||||||
"Verifying your email": "Je e-mailadres wordt geverifieerd",
|
"Verifying your email": "Je e-mailadres wordt geverifieerd",
|
||||||
"Please wait...": "Even geduld...",
|
"Please wait...": "Even geduld...",
|
||||||
"Verification failed. The link may have expired.": "Verificatie mislukt. De link is mogelijk verlopen.",
|
"Verification failed. The link may have expired.": "Verificatie mislukt. De link is mogelijk verlopen.",
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "Editar comentário",
|
"Edit comment": "Editar comentário",
|
||||||
"Delete comment": "Excluir comentário",
|
"Delete comment": "Excluir comentário",
|
||||||
"Are you sure you want to delete this comment?": "Você tem certeza de que deseja excluir este comentário?",
|
"Are you sure you want to delete this comment?": "Você tem certeza de que deseja excluir este comentário?",
|
||||||
"Delete chat": "Excluir chat",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "Tem certeza de que deseja excluir '{{title}}'? Esta ação não pode ser desfeita.",
|
|
||||||
"Comment created successfully": "Comentário criado com sucesso",
|
"Comment created successfully": "Comentário criado com sucesso",
|
||||||
"Error creating comment": "Erro ao criar comentário",
|
"Error creating comment": "Erro ao criar comentário",
|
||||||
"Comment updated successfully": "Comentário atualizado com sucesso",
|
"Comment updated successfully": "Comentário atualizado com sucesso",
|
||||||
@@ -741,93 +739,6 @@
|
|||||||
"Removed page restriction": "Restrição de página removida",
|
"Removed page restriction": "Restrição de página removida",
|
||||||
"Added page permission": "Permissão de página adicionada",
|
"Added page permission": "Permissão de página adicionada",
|
||||||
"Removed page permission": "Permissão de página removida",
|
"Removed page permission": "Permissão de página removida",
|
||||||
"day": "dia",
|
|
||||||
"days": "dias",
|
|
||||||
"week": "semana",
|
|
||||||
"weeks": "semanas",
|
|
||||||
"month": "mês",
|
|
||||||
"months": "meses",
|
|
||||||
"year": "ano",
|
|
||||||
"years": "anos",
|
|
||||||
"Period": "Período",
|
|
||||||
"Fixed date": "Data fixa",
|
|
||||||
"Indefinitely": "Indefinidamente",
|
|
||||||
"Days": "Dias",
|
|
||||||
"Weeks": "Semanas",
|
|
||||||
"Months": "Meses",
|
|
||||||
"Years": "Anos",
|
|
||||||
"Pick a date": "Escolha uma data",
|
|
||||||
"Maximum is {{max}} {{unit}} for this unit": "O máximo é {{max}} {{unit}} para esta unidade",
|
|
||||||
"Never expires. Verifiers can re-verify at any time.": "Nunca expira. Os verificadores podem verificar novamente a qualquer momento.",
|
|
||||||
"Verified": "Verificado",
|
|
||||||
"Review needed": "Revisão necessária",
|
|
||||||
"Verification expired": "A verificação expirou",
|
|
||||||
"Draft": "Rascunho",
|
|
||||||
"In Approval": "Em aprovação",
|
|
||||||
"In approval": "Em aprovação",
|
|
||||||
"Approved": "Aprovado",
|
|
||||||
"Obsolete": "Obsoleto",
|
|
||||||
"Expiring": "Expirando",
|
|
||||||
"Set up verification": "Configurar verificação",
|
|
||||||
"Verify page": "Verificar página",
|
|
||||||
"Page verification": "Verificação da página",
|
|
||||||
"Add verification": "Adicionar verificação",
|
|
||||||
"Edit verification": "Editar verificação",
|
|
||||||
"Search by title": "Pesquisar por título",
|
|
||||||
"Choose how this page should stay accurate.": "Escolha como esta página deve permanecer precisa.",
|
|
||||||
"Recurring verification": "Verificação recorrente",
|
|
||||||
"Verifiers re-confirm this page on a schedule.": "Os verificadores confirmam novamente esta página em uma programação definida.",
|
|
||||||
"Re-verify on a schedule (e.g every 30 days )": "Verificar novamente em uma programação definida (ex.: a cada 30 dias)",
|
|
||||||
"Page stays editable at all times": "A página permanece editável o tempo todo",
|
|
||||||
"Best for runbooks, FAQs, living documentation": "Ideal para runbooks, FAQs e documentação viva",
|
|
||||||
"Approval workflow": "Fluxo de aprovação",
|
|
||||||
"Formal document lifecycle with named approvers.": "Ciclo de vida formal do documento com aprovadores nomeados.",
|
|
||||||
"Draft → In approval → Approved → Obsolete": "Rascunho → Em aprovação → Aprovado → Obsoleto",
|
|
||||||
"Locked once approved, with full history": "Bloqueado após a aprovação, com histórico completo",
|
|
||||||
"Designed for ISO 9001, ISO 13485, and FDA": "Desenvolvido para ISO 9001, ISO 13485 e FDA",
|
|
||||||
"Best for SOPs and controlled documents": "Ideal para POPs e documentos controlados",
|
|
||||||
"Back": "Voltar",
|
|
||||||
"Quality management": "Gestão da qualidade",
|
|
||||||
"Recurring": "Recorrente",
|
|
||||||
"Pages move through draft, approval, and approved stages.": "As páginas passam pelos estágios de rascunho, aprovação e aprovado.",
|
|
||||||
"Verifiers": "Verificadores",
|
|
||||||
"Add verifier": "Adicionar verificador",
|
|
||||||
"I've reviewed this page for accuracy": "Revisei esta página quanto à precisão",
|
|
||||||
"Set up": "Configurar",
|
|
||||||
"Remove verification": "Remover verificação",
|
|
||||||
"Are you sure you want to remove verification from this page?": "Tem certeza de que deseja remover a verificação desta página?",
|
|
||||||
"Assigned verifiers must periodically re-verify this page.": "Os verificadores atribuídos devem verificar novamente esta página periodicamente.",
|
|
||||||
"Last verified by {{name}} {{time}} (expired)": "Verificado pela última vez por {{name}} {{time}} (expirado)",
|
|
||||||
"The fixed expiration date has passed.": "A data fixa de expiração já passou.",
|
|
||||||
"Verified by {{name}} {{time}}": "Verificado por {{name}} {{time}}",
|
|
||||||
"Expires {{date}}": "Expira em {{date}}",
|
|
||||||
"Expired {{date}}": "Expirou em {{date}}",
|
|
||||||
"Mark as obsolete": "Marcar como obsoleto",
|
|
||||||
"Mark obsolete": "Marcar como obsoleto",
|
|
||||||
"Returned by {{name}} {{time}}": "Devolvido por {{name}} {{time}}",
|
|
||||||
"No approval has been requested yet.": "Nenhuma aprovação foi solicitada ainda.",
|
|
||||||
"Submitted by {{name}} {{time}}": "Enviado por {{name}} {{time}}",
|
|
||||||
"Someone": "Alguém",
|
|
||||||
"Approved by {{name}} {{time}}": "Aprovado por {{name}} {{time}}",
|
|
||||||
"This document has been marked as obsolete.": "Este documento foi marcado como obsoleto.",
|
|
||||||
"Rejection comment": "Comentário de rejeição",
|
|
||||||
"Reason for returning this document...": "Motivo para devolver este documento...",
|
|
||||||
"Confirm rejection": "Confirmar rejeição",
|
|
||||||
"Submit for approval": "Enviar para aprovação",
|
|
||||||
"Reject": "Rejeitar",
|
|
||||||
"Approve": "Aprovar",
|
|
||||||
"Re-submit for approval": "Reenviar para aprovação",
|
|
||||||
"Verified until": "Verificado até",
|
|
||||||
"QMS": "SGQ",
|
|
||||||
"Verified pages": "Páginas verificadas",
|
|
||||||
"Search pages...": "Pesquisar páginas...",
|
|
||||||
"Filter by space": "Filtrar por espaço",
|
|
||||||
"Filter by type": "Filtrar por tipo",
|
|
||||||
"<bold>{{name}}</bold> verified a page": "<bold>{{name}}</bold> verificou uma página",
|
|
||||||
"<bold>{{name}}</bold> submitted a page for your approval": "<bold>{{name}}</bold> enviou uma página para sua aprovação",
|
|
||||||
"<bold>{{name}}</bold> returned a page for revision": "<bold>{{name}}</bold> devolveu uma página para revisão",
|
|
||||||
"Page verification expires soon": "A verificação da página expirará em breve",
|
|
||||||
"Page verification has expired": "A verificação da página expirou",
|
|
||||||
"Verifying your email": "Verificando seu e-mail",
|
"Verifying your email": "Verificando seu e-mail",
|
||||||
"Please wait...": "Por favor, aguarde...",
|
"Please wait...": "Por favor, aguarde...",
|
||||||
"Verification failed. The link may have expired.": "Falha na verificação. O link pode ter expirado.",
|
"Verification failed. The link may have expired.": "Falha na verificação. O link pode ter expirado.",
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "Редактировать комментарий",
|
"Edit comment": "Редактировать комментарий",
|
||||||
"Delete comment": "Удалить комментарий",
|
"Delete comment": "Удалить комментарий",
|
||||||
"Are you sure you want to delete this comment?": "Вы уверены, что хотите удалить этот комментарий?",
|
"Are you sure you want to delete this comment?": "Вы уверены, что хотите удалить этот комментарий?",
|
||||||
"Delete chat": "Удалить чат",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "Вы уверены, что хотите удалить '{{title}}'? Это действие нельзя отменить.",
|
|
||||||
"Comment created successfully": "Комментарий успешно создан",
|
"Comment created successfully": "Комментарий успешно создан",
|
||||||
"Error creating comment": "Ошибка при создании комментария",
|
"Error creating comment": "Ошибка при создании комментария",
|
||||||
"Comment updated successfully": "Комментарий успешно обновлён",
|
"Comment updated successfully": "Комментарий успешно обновлён",
|
||||||
@@ -741,93 +739,6 @@
|
|||||||
"Removed page restriction": "Ограничение доступа к странице удалено",
|
"Removed page restriction": "Ограничение доступа к странице удалено",
|
||||||
"Added page permission": "Добавлено разрешение доступа к странице",
|
"Added page permission": "Добавлено разрешение доступа к странице",
|
||||||
"Removed page permission": "Удалено разрешение доступа к странице",
|
"Removed page permission": "Удалено разрешение доступа к странице",
|
||||||
"day": "день",
|
|
||||||
"days": "дни",
|
|
||||||
"week": "неделя",
|
|
||||||
"weeks": "недели",
|
|
||||||
"month": "месяц",
|
|
||||||
"months": "месяцы",
|
|
||||||
"year": "год",
|
|
||||||
"years": "годы",
|
|
||||||
"Period": "Период",
|
|
||||||
"Fixed date": "Фиксированная дата",
|
|
||||||
"Indefinitely": "Бессрочно",
|
|
||||||
"Days": "Дни",
|
|
||||||
"Weeks": "Недели",
|
|
||||||
"Months": "Месяцы",
|
|
||||||
"Years": "Годы",
|
|
||||||
"Pick a date": "Выберите дату",
|
|
||||||
"Maximum is {{max}} {{unit}} for this unit": "Максимум для этой единицы измерения — {{max}} {{unit}}",
|
|
||||||
"Never expires. Verifiers can re-verify at any time.": "Срок действия не истекает. Проверяющие могут повторно подтверждать в любое время.",
|
|
||||||
"Verified": "Проверено",
|
|
||||||
"Review needed": "Требуется проверка",
|
|
||||||
"Verification expired": "Срок проверки истёк",
|
|
||||||
"Draft": "Черновик",
|
|
||||||
"In Approval": "На утверждении",
|
|
||||||
"In approval": "На утверждении",
|
|
||||||
"Approved": "Утверждено",
|
|
||||||
"Obsolete": "Устарело",
|
|
||||||
"Expiring": "Истекает",
|
|
||||||
"Set up verification": "Настроить проверку",
|
|
||||||
"Verify page": "Проверить страницу",
|
|
||||||
"Page verification": "Проверка страницы",
|
|
||||||
"Add verification": "Добавить проверку",
|
|
||||||
"Edit verification": "Изменить проверку",
|
|
||||||
"Search by title": "Поиск по заголовку",
|
|
||||||
"Choose how this page should stay accurate.": "Выберите, как поддерживать актуальность этой страницы.",
|
|
||||||
"Recurring verification": "Регулярная проверка",
|
|
||||||
"Verifiers re-confirm this page on a schedule.": "Проверяющие повторно подтверждают эту страницу по расписанию.",
|
|
||||||
"Re-verify on a schedule (e.g every 30 days )": "Повторно проверять по расписанию (например, каждые 30 дней)",
|
|
||||||
"Page stays editable at all times": "Страница остаётся редактируемой в любое время",
|
|
||||||
"Best for runbooks, FAQs, living documentation": "Лучше всего подходит для инструкций, FAQ и живой документации",
|
|
||||||
"Approval workflow": "Процесс утверждения",
|
|
||||||
"Formal document lifecycle with named approvers.": "Формальный жизненный цикл документа с назначенными утверждающими.",
|
|
||||||
"Draft → In approval → Approved → Obsolete": "Черновик → На утверждении → Утверждено → Устарело",
|
|
||||||
"Locked once approved, with full history": "После утверждения блокируется, с полной историей",
|
|
||||||
"Designed for ISO 9001, ISO 13485, and FDA": "Разработано для ISO 9001, ISO 13485 и FDA",
|
|
||||||
"Best for SOPs and controlled documents": "Лучше всего подходит для СОП и контролируемых документов",
|
|
||||||
"Back": "Назад",
|
|
||||||
"Quality management": "Управление качеством",
|
|
||||||
"Recurring": "Регулярно",
|
|
||||||
"Pages move through draft, approval, and approved stages.": "Страницы проходят стадии черновика, утверждения и утверждённого состояния.",
|
|
||||||
"Verifiers": "Проверяющие",
|
|
||||||
"Add verifier": "Добавить проверяющего",
|
|
||||||
"I've reviewed this page for accuracy": "Я проверил(а) эту страницу на точность",
|
|
||||||
"Set up": "Настроить",
|
|
||||||
"Remove verification": "Удалить проверку",
|
|
||||||
"Are you sure you want to remove verification from this page?": "Вы уверены, что хотите удалить проверку с этой страницы?",
|
|
||||||
"Assigned verifiers must periodically re-verify this page.": "Назначенные проверяющие должны периодически повторно проверять эту страницу.",
|
|
||||||
"Last verified by {{name}} {{time}} (expired)": "Последняя проверка: {{name}}, {{time}} (срок истёк)",
|
|
||||||
"The fixed expiration date has passed.": "Фиксированная дата истечения срока уже прошла.",
|
|
||||||
"Verified by {{name}} {{time}}": "Проверено: {{name}}, {{time}}",
|
|
||||||
"Expires {{date}}": "Истекает {{date}}",
|
|
||||||
"Expired {{date}}": "Срок истёк {{date}}",
|
|
||||||
"Mark as obsolete": "Отметить как устаревшее",
|
|
||||||
"Mark obsolete": "Отметить как устаревшее",
|
|
||||||
"Returned by {{name}} {{time}}": "Возвращено: {{name}}, {{time}}",
|
|
||||||
"No approval has been requested yet.": "Запрос на утверждение ещё не отправлен.",
|
|
||||||
"Submitted by {{name}} {{time}}": "Отправлено: {{name}}, {{time}}",
|
|
||||||
"Someone": "Кто-то",
|
|
||||||
"Approved by {{name}} {{time}}": "Утверждено: {{name}}, {{time}}",
|
|
||||||
"This document has been marked as obsolete.": "Этот документ был отмечен как устаревший.",
|
|
||||||
"Rejection comment": "Комментарий к отклонению",
|
|
||||||
"Reason for returning this document...": "Причина возврата этого документа...",
|
|
||||||
"Confirm rejection": "Подтвердить отклонение",
|
|
||||||
"Submit for approval": "Отправить на утверждение",
|
|
||||||
"Reject": "Отклонить",
|
|
||||||
"Approve": "Утвердить",
|
|
||||||
"Re-submit for approval": "Повторно отправить на утверждение",
|
|
||||||
"Verified until": "Проверено до",
|
|
||||||
"QMS": "QMS",
|
|
||||||
"Verified pages": "Проверенные страницы",
|
|
||||||
"Search pages...": "Поиск страниц...",
|
|
||||||
"Filter by space": "Фильтр по пространству",
|
|
||||||
"Filter by type": "Фильтр по типу",
|
|
||||||
"<bold>{{name}}</bold> verified a page": "<bold>{{name}}</bold> проверил(а) страницу",
|
|
||||||
"<bold>{{name}}</bold> submitted a page for your approval": "<bold>{{name}}</bold> отправил(а) страницу вам на утверждение",
|
|
||||||
"<bold>{{name}}</bold> returned a page for revision": "<bold>{{name}}</bold> вернул(а) страницу на доработку",
|
|
||||||
"Page verification expires soon": "Срок проверки страницы скоро истекает",
|
|
||||||
"Page verification has expired": "Срок проверки страницы истёк",
|
|
||||||
"Verifying your email": "Подтверждение вашего адреса электронной почты",
|
"Verifying your email": "Подтверждение вашего адреса электронной почты",
|
||||||
"Please wait...": "Пожалуйста, подождите...",
|
"Please wait...": "Пожалуйста, подождите...",
|
||||||
"Verification failed. The link may have expired.": "Ошибка проверки. Ссылка могла устареть.",
|
"Verification failed. The link may have expired.": "Ошибка проверки. Ссылка могла устареть.",
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "Редагувати коментар",
|
"Edit comment": "Редагувати коментар",
|
||||||
"Delete comment": "Видалити коментар",
|
"Delete comment": "Видалити коментар",
|
||||||
"Are you sure you want to delete this comment?": "Ви впевнені, що хочете видалити цей коментар?",
|
"Are you sure you want to delete this comment?": "Ви впевнені, що хочете видалити цей коментар?",
|
||||||
"Delete chat": "Видалити чат",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "Ви впевнені, що хочете видалити '{{title}}'? Цю дію неможливо скасувати.",
|
|
||||||
"Comment created successfully": "Коментар успішно створено",
|
"Comment created successfully": "Коментар успішно створено",
|
||||||
"Error creating comment": "Помилка при створенні коментаря",
|
"Error creating comment": "Помилка при створенні коментаря",
|
||||||
"Comment updated successfully": "Коментар успішно оновлено",
|
"Comment updated successfully": "Коментар успішно оновлено",
|
||||||
@@ -741,93 +739,6 @@
|
|||||||
"Removed page restriction": "Обмеження сторінки видалено",
|
"Removed page restriction": "Обмеження сторінки видалено",
|
||||||
"Added page permission": "Додано дозвіл на сторінку",
|
"Added page permission": "Додано дозвіл на сторінку",
|
||||||
"Removed page permission": "Дозвіл на сторінку видалено",
|
"Removed page permission": "Дозвіл на сторінку видалено",
|
||||||
"day": "день",
|
|
||||||
"days": "дні",
|
|
||||||
"week": "тиждень",
|
|
||||||
"weeks": "тижні",
|
|
||||||
"month": "місяць",
|
|
||||||
"months": "місяці",
|
|
||||||
"year": "рік",
|
|
||||||
"years": "роки",
|
|
||||||
"Period": "Період",
|
|
||||||
"Fixed date": "Фіксована дата",
|
|
||||||
"Indefinitely": "Безстроково",
|
|
||||||
"Days": "Дні",
|
|
||||||
"Weeks": "Тижні",
|
|
||||||
"Months": "Місяці",
|
|
||||||
"Years": "Роки",
|
|
||||||
"Pick a date": "Виберіть дату",
|
|
||||||
"Maximum is {{max}} {{unit}} for this unit": "Максимум для цієї одиниці — {{max}} {{unit}}",
|
|
||||||
"Never expires. Verifiers can re-verify at any time.": "Термін дії не спливає. Верифікатори можуть повторно перевірити будь-коли.",
|
|
||||||
"Verified": "Перевірено",
|
|
||||||
"Review needed": "Потрібен перегляд",
|
|
||||||
"Verification expired": "Термін перевірки сплив",
|
|
||||||
"Draft": "Чернетка",
|
|
||||||
"In Approval": "На погодженні",
|
|
||||||
"In approval": "На погодженні",
|
|
||||||
"Approved": "Погоджено",
|
|
||||||
"Obsolete": "Застаріло",
|
|
||||||
"Expiring": "Термін дії спливає",
|
|
||||||
"Set up verification": "Налаштувати перевірку",
|
|
||||||
"Verify page": "Перевірити сторінку",
|
|
||||||
"Page verification": "Перевірка сторінки",
|
|
||||||
"Add verification": "Додати перевірку",
|
|
||||||
"Edit verification": "Редагувати перевірку",
|
|
||||||
"Search by title": "Пошук за назвою",
|
|
||||||
"Choose how this page should stay accurate.": "Виберіть, як підтримувати актуальність цієї сторінки.",
|
|
||||||
"Recurring verification": "Регулярна перевірка",
|
|
||||||
"Verifiers re-confirm this page on a schedule.": "Верифікатори повторно підтверджують цю сторінку за розкладом.",
|
|
||||||
"Re-verify on a schedule (e.g every 30 days )": "Повторно перевіряти за розкладом (наприклад, кожні 30 днів)",
|
|
||||||
"Page stays editable at all times": "Сторінка залишається доступною для редагування в будь-який час",
|
|
||||||
"Best for runbooks, FAQs, living documentation": "Найкраще підходить для runbook-ів, FAQ і живої документації",
|
|
||||||
"Approval workflow": "Процес погодження",
|
|
||||||
"Formal document lifecycle with named approvers.": "Формальний життєвий цикл документа з призначеними погоджувачами.",
|
|
||||||
"Draft → In approval → Approved → Obsolete": "Чернетка → На погодженні → Погоджено → Застаріло",
|
|
||||||
"Locked once approved, with full history": "Після погодження блокується, із повною історією",
|
|
||||||
"Designed for ISO 9001, ISO 13485, and FDA": "Призначено для ISO 9001, ISO 13485 та FDA",
|
|
||||||
"Best for SOPs and controlled documents": "Найкраще підходить для SOP і контрольованих документів",
|
|
||||||
"Back": "Назад",
|
|
||||||
"Quality management": "Управління якістю",
|
|
||||||
"Recurring": "Регулярна",
|
|
||||||
"Pages move through draft, approval, and approved stages.": "Сторінки проходять стадії чернетки, погодження та погодженого документа.",
|
|
||||||
"Verifiers": "Верифікатори",
|
|
||||||
"Add verifier": "Додати верифікатора",
|
|
||||||
"I've reviewed this page for accuracy": "Я перевірив(ла) цю сторінку на точність",
|
|
||||||
"Set up": "Налаштувати",
|
|
||||||
"Remove verification": "Видалити перевірку",
|
|
||||||
"Are you sure you want to remove verification from this page?": "Ви впевнені, що хочете видалити перевірку з цієї сторінки?",
|
|
||||||
"Assigned verifiers must periodically re-verify this page.": "Призначені верифікатори мають періодично повторно перевіряти цю сторінку.",
|
|
||||||
"Last verified by {{name}} {{time}} (expired)": "Востаннє перевірив(-ла) {{name}} {{time}} (термін дії сплив)",
|
|
||||||
"The fixed expiration date has passed.": "Фіксована дата завершення вже минула.",
|
|
||||||
"Verified by {{name}} {{time}}": "Перевірено: {{name}} {{time}}",
|
|
||||||
"Expires {{date}}": "Термін дії спливає {{date}}",
|
|
||||||
"Expired {{date}}": "Термін дії сплив {{date}}",
|
|
||||||
"Mark as obsolete": "Позначити як застаріле",
|
|
||||||
"Mark obsolete": "Позначити як застаріле",
|
|
||||||
"Returned by {{name}} {{time}}": "Повернуто: {{name}} {{time}}",
|
|
||||||
"No approval has been requested yet.": "Запит на погодження ще не було надіслано.",
|
|
||||||
"Submitted by {{name}} {{time}}": "Надіслано: {{name}} {{time}}",
|
|
||||||
"Someone": "Хтось",
|
|
||||||
"Approved by {{name}} {{time}}": "Погоджено: {{name}} {{time}}",
|
|
||||||
"This document has been marked as obsolete.": "Цей документ позначено як застарілий.",
|
|
||||||
"Rejection comment": "Коментар щодо відхилення",
|
|
||||||
"Reason for returning this document...": "Причина повернення цього документа...",
|
|
||||||
"Confirm rejection": "Підтвердити відхилення",
|
|
||||||
"Submit for approval": "Надіслати на погодження",
|
|
||||||
"Reject": "Відхилити",
|
|
||||||
"Approve": "Погодити",
|
|
||||||
"Re-submit for approval": "Повторно надіслати на погодження",
|
|
||||||
"Verified until": "Перевірено до",
|
|
||||||
"QMS": "QMS",
|
|
||||||
"Verified pages": "Перевірені сторінки",
|
|
||||||
"Search pages...": "Шукати сторінки...",
|
|
||||||
"Filter by space": "Фільтрувати за простором",
|
|
||||||
"Filter by type": "Фільтрувати за типом",
|
|
||||||
"<bold>{{name}}</bold> verified a page": "<bold>{{name}}</bold> перевірив(-ла) сторінку",
|
|
||||||
"<bold>{{name}}</bold> submitted a page for your approval": "<bold>{{name}}</bold> надіслав(-ла) сторінку вам на погодження",
|
|
||||||
"<bold>{{name}}</bold> returned a page for revision": "<bold>{{name}}</bold> повернув(-ла) сторінку на доопрацювання",
|
|
||||||
"Page verification expires soon": "Термін перевірки сторінки скоро спливає",
|
|
||||||
"Page verification has expired": "Термін перевірки сторінки сплив",
|
|
||||||
"Verifying your email": "Перевірка вашої електронної пошти",
|
"Verifying your email": "Перевірка вашої електронної пошти",
|
||||||
"Please wait...": "Будь ласка, зачекайте...",
|
"Please wait...": "Будь ласка, зачекайте...",
|
||||||
"Verification failed. The link may have expired.": "Підтвердження не вдалося. Посилання могло втратити чинність.",
|
"Verification failed. The link may have expired.": "Підтвердження не вдалося. Посилання могло втратити чинність.",
|
||||||
|
|||||||
@@ -222,8 +222,6 @@
|
|||||||
"Edit comment": "编辑评论",
|
"Edit comment": "编辑评论",
|
||||||
"Delete comment": "删除评论",
|
"Delete comment": "删除评论",
|
||||||
"Are you sure you want to delete this comment?": "你确定要删除这条评论吗?",
|
"Are you sure you want to delete this comment?": "你确定要删除这条评论吗?",
|
||||||
"Delete chat": "删除聊天",
|
|
||||||
"Are you sure you want to delete '{{title}}'? This action cannot be undone.": "您确定要删除「{{title}}」吗?此操作无法撤销。",
|
|
||||||
"Comment created successfully": "成功创建评论",
|
"Comment created successfully": "成功创建评论",
|
||||||
"Error creating comment": "创建评论时出错",
|
"Error creating comment": "创建评论时出错",
|
||||||
"Comment updated successfully": "评论更新成功",
|
"Comment updated successfully": "评论更新成功",
|
||||||
@@ -741,93 +739,6 @@
|
|||||||
"Removed page restriction": "已移除页面限制",
|
"Removed page restriction": "已移除页面限制",
|
||||||
"Added page permission": "已添加页面权限",
|
"Added page permission": "已添加页面权限",
|
||||||
"Removed page permission": "已移除页面权限",
|
"Removed page permission": "已移除页面权限",
|
||||||
"day": "天",
|
|
||||||
"days": "天",
|
|
||||||
"week": "周",
|
|
||||||
"weeks": "周",
|
|
||||||
"month": "个月",
|
|
||||||
"months": "个月",
|
|
||||||
"year": "年",
|
|
||||||
"years": "年",
|
|
||||||
"Period": "周期",
|
|
||||||
"Fixed date": "固定日期",
|
|
||||||
"Indefinitely": "无限期",
|
|
||||||
"Days": "天",
|
|
||||||
"Weeks": "周",
|
|
||||||
"Months": "个月",
|
|
||||||
"Years": "年",
|
|
||||||
"Pick a date": "选择日期",
|
|
||||||
"Maximum is {{max}} {{unit}} for this unit": "此单位的最大值为 {{max}} {{unit}}",
|
|
||||||
"Never expires. Verifiers can re-verify at any time.": "永不过期。验证者可随时重新验证。",
|
|
||||||
"Verified": "已验证",
|
|
||||||
"Review needed": "需要审核",
|
|
||||||
"Verification expired": "验证已过期",
|
|
||||||
"Draft": "草稿",
|
|
||||||
"In Approval": "审批中",
|
|
||||||
"In approval": "审批中",
|
|
||||||
"Approved": "已批准",
|
|
||||||
"Obsolete": "已作废",
|
|
||||||
"Expiring": "即将过期",
|
|
||||||
"Set up verification": "设置验证",
|
|
||||||
"Verify page": "验证页面",
|
|
||||||
"Page verification": "页面验证",
|
|
||||||
"Add verification": "添加验证",
|
|
||||||
"Edit verification": "编辑验证",
|
|
||||||
"Search by title": "按标题搜索",
|
|
||||||
"Choose how this page should stay accurate.": "选择此页面保持准确的方式。",
|
|
||||||
"Recurring verification": "定期验证",
|
|
||||||
"Verifiers re-confirm this page on a schedule.": "验证者按计划重新确认此页面。",
|
|
||||||
"Re-verify on a schedule (e.g every 30 days )": "按计划重新验证(例如每 30 天一次)",
|
|
||||||
"Page stays editable at all times": "页面始终可编辑",
|
|
||||||
"Best for runbooks, FAQs, living documentation": "最适合运行手册、常见问题和动态文档",
|
|
||||||
"Approval workflow": "审批工作流",
|
|
||||||
"Formal document lifecycle with named approvers.": "具有指定审批人的正式文档生命周期。",
|
|
||||||
"Draft → In approval → Approved → Obsolete": "草稿 → 审批中 → 已批准 → 已作废",
|
|
||||||
"Locked once approved, with full history": "批准后锁定,并保留完整历史记录",
|
|
||||||
"Designed for ISO 9001, ISO 13485, and FDA": "专为 ISO 9001、ISO 13485 和 FDA 设计",
|
|
||||||
"Best for SOPs and controlled documents": "最适合 SOP 和受控文档",
|
|
||||||
"Back": "返回",
|
|
||||||
"Quality management": "质量管理",
|
|
||||||
"Recurring": "定期",
|
|
||||||
"Pages move through draft, approval, and approved stages.": "页面会经历草稿、审批中和已批准阶段。",
|
|
||||||
"Verifiers": "验证者",
|
|
||||||
"Add verifier": "添加验证者",
|
|
||||||
"I've reviewed this page for accuracy": "我已审核此页面的准确性",
|
|
||||||
"Set up": "设置",
|
|
||||||
"Remove verification": "移除验证",
|
|
||||||
"Are you sure you want to remove verification from this page?": "确定要移除此页面的验证吗?",
|
|
||||||
"Assigned verifiers must periodically re-verify this page.": "指定的验证者必须定期重新验证此页面。",
|
|
||||||
"Last verified by {{name}} {{time}} (expired)": "最后由 {{name}} 于 {{time}} 验证(已过期)",
|
|
||||||
"The fixed expiration date has passed.": "固定到期日已过。",
|
|
||||||
"Verified by {{name}} {{time}}": "由 {{name}} 于 {{time}} 验证",
|
|
||||||
"Expires {{date}}": "于 {{date}} 到期",
|
|
||||||
"Expired {{date}}": "已于 {{date}} 过期",
|
|
||||||
"Mark as obsolete": "标记为作废",
|
|
||||||
"Mark obsolete": "标记作废",
|
|
||||||
"Returned by {{name}} {{time}}": "由 {{name}} 于 {{time}} 退回",
|
|
||||||
"No approval has been requested yet.": "尚未请求审批。",
|
|
||||||
"Submitted by {{name}} {{time}}": "由 {{name}} 于 {{time}} 提交",
|
|
||||||
"Someone": "某人",
|
|
||||||
"Approved by {{name}} {{time}}": "由 {{name}} 于 {{time}} 批准",
|
|
||||||
"This document has been marked as obsolete.": "此文档已被标记为作废。",
|
|
||||||
"Rejection comment": "退回意见",
|
|
||||||
"Reason for returning this document...": "退回此文档的原因...",
|
|
||||||
"Confirm rejection": "确认退回",
|
|
||||||
"Submit for approval": "提交审批",
|
|
||||||
"Reject": "退回",
|
|
||||||
"Approve": "批准",
|
|
||||||
"Re-submit for approval": "重新提交审批",
|
|
||||||
"Verified until": "验证有效期至",
|
|
||||||
"QMS": "QMS",
|
|
||||||
"Verified pages": "已验证页面",
|
|
||||||
"Search pages...": "搜索页面...",
|
|
||||||
"Filter by space": "按空间筛选",
|
|
||||||
"Filter by type": "按类型筛选",
|
|
||||||
"<bold>{{name}}</bold> verified a page": "<bold>{{name}}</bold> 验证了一个页面",
|
|
||||||
"<bold>{{name}}</bold> submitted a page for your approval": "<bold>{{name}}</bold> 提交了一个页面供您审批",
|
|
||||||
"<bold>{{name}}</bold> returned a page for revision": "<bold>{{name}}</bold> 退回了一个页面以供修改",
|
|
||||||
"Page verification expires soon": "页面验证即将过期",
|
|
||||||
"Page verification has expired": "页面验证已过期",
|
|
||||||
"Verifying your email": "正在验证您的邮箱",
|
"Verifying your email": "正在验证您的邮箱",
|
||||||
"Please wait...": "请稍候……",
|
"Please wait...": "请稍候……",
|
||||||
"Verification failed. The link may have expired.": "验证失败。该链接可能已过期。",
|
"Verification failed. The link may have expired.": "验证失败。该链接可能已过期。",
|
||||||
|
|||||||
@@ -26,7 +26,6 @@ import Security from "@/ee/security/pages/security.tsx";
|
|||||||
import License from "@/ee/licence/pages/license.tsx";
|
import License from "@/ee/licence/pages/license.tsx";
|
||||||
import { useRedirectToCloudSelect } from "@/ee/hooks/use-redirect-to-cloud-select.tsx";
|
import { useRedirectToCloudSelect } from "@/ee/hooks/use-redirect-to-cloud-select.tsx";
|
||||||
import SharedPage from "@/pages/share/shared-page.tsx";
|
import SharedPage from "@/pages/share/shared-page.tsx";
|
||||||
import PdfRenderPage from "@/ee/pdf-export/pdf-render-page.tsx";
|
|
||||||
import Shares from "@/pages/settings/shares/shares.tsx";
|
import Shares from "@/pages/settings/shares/shares.tsx";
|
||||||
import ShareLayout from "@/features/share/components/share-layout.tsx";
|
import ShareLayout from "@/features/share/components/share-layout.tsx";
|
||||||
import ShareRedirect from "@/pages/share/share-redirect.tsx";
|
import ShareRedirect from "@/pages/share/share-redirect.tsx";
|
||||||
@@ -82,7 +81,6 @@ export default function App() {
|
|||||||
<Route path={"/share/p/:pageSlug"} element={<SharedPage />} />
|
<Route path={"/share/p/:pageSlug"} element={<SharedPage />} />
|
||||||
</Route>
|
</Route>
|
||||||
|
|
||||||
<Route path={"/pdf-render/:pageId"} element={<PdfRenderPage />} />
|
|
||||||
<Route path={"/share/:shareId"} element={<ShareRedirect />} />
|
<Route path={"/share/:shareId"} element={<ShareRedirect />} />
|
||||||
<Route path={"/p/:pageSlug"} element={<PageRedirect />} />
|
<Route path={"/p/:pageSlug"} element={<PageRedirect />} />
|
||||||
|
|
||||||
|
|||||||
@@ -80,12 +80,6 @@ export default function AvatarUploader({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const ariaLabel = {
|
|
||||||
[AvatarIconType.AVATAR]: t("Change avatar"),
|
|
||||||
[AvatarIconType.SPACE_ICON]: t("Change space icon"),
|
|
||||||
[AvatarIconType.WORKSPACE_ICON]: t("Change workspace icon"),
|
|
||||||
}[type];
|
|
||||||
|
|
||||||
const handleRemove = async () => {
|
const handleRemove = async () => {
|
||||||
if (disabled) return;
|
if (disabled) return;
|
||||||
|
|
||||||
@@ -110,8 +104,6 @@ export default function AvatarUploader({
|
|||||||
ref={fileInputRef}
|
ref={fileInputRef}
|
||||||
onChange={handleFileInputChange}
|
onChange={handleFileInputChange}
|
||||||
accept="image/png,image/jpeg,image/jpg"
|
accept="image/png,image/jpeg,image/jpg"
|
||||||
aria-label={ariaLabel}
|
|
||||||
tabIndex={-1}
|
|
||||||
style={{ display: "none" }}
|
style={{ display: "none" }}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
@@ -123,8 +115,6 @@ export default function AvatarUploader({
|
|||||||
size={size}
|
size={size}
|
||||||
avatarUrl={currentImageUrl}
|
avatarUrl={currentImageUrl}
|
||||||
name={fallbackName}
|
name={fallbackName}
|
||||||
aria-label={ariaLabel}
|
|
||||||
aria-haspopup="menu"
|
|
||||||
style={{
|
style={{
|
||||||
cursor: disabled || isLoading ? "default" : "pointer",
|
cursor: disabled || isLoading ? "default" : "pointer",
|
||||||
opacity: isLoading ? 0.6 : 1,
|
opacity: isLoading ? 0.6 : 1,
|
||||||
|
|||||||
@@ -25,7 +25,6 @@ export default function CopyTextButton({ text, size }: CopyProps) {
|
|||||||
variant="subtle"
|
variant="subtle"
|
||||||
onClick={copy}
|
onClick={copy}
|
||||||
size={size}
|
size={size}
|
||||||
aria-label={copied ? t("Copied") : t("Copy")}
|
|
||||||
>
|
>
|
||||||
{copied ? <IconCheck size={16} /> : <IconCopy size={16} />}
|
{copied ? <IconCheck size={16} /> : <IconCopy size={16} />}
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import {
|
|||||||
UnstyledButton,
|
UnstyledButton,
|
||||||
Badge,
|
Badge,
|
||||||
Table,
|
Table,
|
||||||
ThemeIcon,
|
ActionIcon,
|
||||||
Button,
|
Button,
|
||||||
} from "@mantine/core";
|
} from "@mantine/core";
|
||||||
import { Link } from "react-router-dom";
|
import { Link } from "react-router-dom";
|
||||||
@@ -49,9 +49,9 @@ export default function RecentChanges({ spaceId }: Props) {
|
|||||||
>
|
>
|
||||||
<Group wrap="nowrap">
|
<Group wrap="nowrap">
|
||||||
{page.icon || (
|
{page.icon || (
|
||||||
<ThemeIcon variant="transparent" color="gray" size={18}>
|
<ActionIcon variant="transparent" color="gray" size={18}>
|
||||||
<IconFileDescription size={18} />
|
<IconFileDescription size={18} />
|
||||||
</ThemeIcon>
|
</ActionIcon>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Text fw={500} size="md" lineClamp={1}>
|
<Text fw={500} size="md" lineClamp={1}>
|
||||||
|
|||||||
@@ -6,14 +6,12 @@ import { useTranslation } from "react-i18next";
|
|||||||
|
|
||||||
export interface SearchInputProps {
|
export interface SearchInputProps {
|
||||||
placeholder?: string;
|
placeholder?: string;
|
||||||
ariaLabel?: string;
|
|
||||||
debounceDelay?: number;
|
debounceDelay?: number;
|
||||||
onSearch: (value: string) => void;
|
onSearch: (value: string) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function SearchInput({
|
export function SearchInput({
|
||||||
placeholder,
|
placeholder,
|
||||||
ariaLabel,
|
|
||||||
debounceDelay = 500,
|
debounceDelay = 500,
|
||||||
onSearch,
|
onSearch,
|
||||||
}: SearchInputProps) {
|
}: SearchInputProps) {
|
||||||
@@ -30,7 +28,6 @@ export function SearchInput({
|
|||||||
<TextInput
|
<TextInput
|
||||||
size="sm"
|
size="sm"
|
||||||
placeholder={placeholder || t("Search...")}
|
placeholder={placeholder || t("Search...")}
|
||||||
aria-label={ariaLabel || placeholder || t("Search")}
|
|
||||||
leftSection={<IconSearch size={16} />}
|
leftSection={<IconSearch size={16} />}
|
||||||
value={value}
|
value={value}
|
||||||
onChange={(e) => setValue(e.currentTarget.value)}
|
onChange={(e) => setValue(e.currentTarget.value)}
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
import { ThemeIcon } from "@mantine/core";
|
import { ActionIcon, rem } from "@mantine/core";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { IconUsersGroup } from "@tabler/icons-react";
|
import { IconUsersGroup } from "@tabler/icons-react";
|
||||||
|
|
||||||
export function IconGroupCircle() {
|
export function IconGroupCircle() {
|
||||||
return (
|
return (
|
||||||
<ThemeIcon variant="light" size="lg" color="gray" radius="xl">
|
<ActionIcon variant="light" size="lg" color="gray" radius="xl">
|
||||||
<IconUsersGroup stroke={1.5} />
|
<IconUsersGroup stroke={1.5} />
|
||||||
</ThemeIcon>
|
</ActionIcon>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,22 +28,4 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.skipLink {
|
|
||||||
position: fixed;
|
|
||||||
left: 8px;
|
|
||||||
top: 8px;
|
|
||||||
padding: 8px 12px;
|
|
||||||
background: var(--mantine-color-blue-6);
|
|
||||||
color: #fff;
|
|
||||||
border-radius: 4px;
|
|
||||||
text-decoration: none;
|
|
||||||
z-index: 1000;
|
|
||||||
transform: translateY(-150%);
|
|
||||||
|
|
||||||
&:focus {
|
|
||||||
transform: translateY(0);
|
|
||||||
outline: 2px solid var(--mantine-color-blue-3);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import { AppShell, Container } from "@mantine/core";
|
import { AppShell, Container } from "@mantine/core";
|
||||||
import React, { useEffect, useRef, useState } from "react";
|
import React, { useEffect, useRef, useState } from "react";
|
||||||
import { useLocation } from "react-router-dom";
|
import { useLocation } from "react-router-dom";
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import SettingsSidebar from "@/components/settings/settings-sidebar.tsx";
|
import SettingsSidebar from "@/components/settings/settings-sidebar.tsx";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import {
|
import {
|
||||||
@@ -24,12 +23,11 @@ export default function GlobalAppShell({
|
|||||||
}: {
|
}: {
|
||||||
children: React.ReactNode;
|
children: React.ReactNode;
|
||||||
}) {
|
}) {
|
||||||
const { t } = useTranslation();
|
|
||||||
useTrialEndAction();
|
useTrialEndAction();
|
||||||
const [mobileOpened] = useAtom(mobileSidebarAtom);
|
const [mobileOpened] = useAtom(mobileSidebarAtom);
|
||||||
const toggleMobile = useToggleSidebar(mobileSidebarAtom);
|
const toggleMobile = useToggleSidebar(mobileSidebarAtom);
|
||||||
const [desktopOpened] = useAtom(desktopSidebarAtom);
|
const [desktopOpened] = useAtom(desktopSidebarAtom);
|
||||||
const [{ isAsideOpen, tab: asideTab }] = useAtom(asideStateAtom);
|
const [{ isAsideOpen }] = useAtom(asideStateAtom);
|
||||||
const [sidebarWidth, setSidebarWidth] = useAtom(sidebarWidthAtom);
|
const [sidebarWidth, setSidebarWidth] = useAtom(sidebarWidthAtom);
|
||||||
const [isResizing, setIsResizing] = useState(false);
|
const [isResizing, setIsResizing] = useState(false);
|
||||||
const sidebarRef = useRef(null);
|
const sidebarRef = useRef(null);
|
||||||
@@ -81,11 +79,7 @@ export default function GlobalAppShell({
|
|||||||
const showGlobalSidebar = !isSpaceRoute && !isSettingsRoute && !isAiRoute;
|
const showGlobalSidebar = !isSpaceRoute && !isSettingsRoute && !isAiRoute;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<AppShell
|
||||||
<a href="#main-content" className={classes.skipLink}>
|
|
||||||
{t("Skip to main content")}
|
|
||||||
</a>
|
|
||||||
<AppShell
|
|
||||||
header={{ height: 45 }}
|
header={{ height: 45 }}
|
||||||
navbar={{
|
navbar={{
|
||||||
width: isSpaceRoute ? sidebarWidth : 300,
|
width: isSpaceRoute ? sidebarWidth : 300,
|
||||||
@@ -111,15 +105,6 @@ export default function GlobalAppShell({
|
|||||||
className={classes.navbar}
|
className={classes.navbar}
|
||||||
withBorder={false}
|
withBorder={false}
|
||||||
ref={sidebarRef}
|
ref={sidebarRef}
|
||||||
aria-label={
|
|
||||||
isSpaceRoute
|
|
||||||
? t("Space navigation")
|
|
||||||
: isSettingsRoute
|
|
||||||
? t("Settings navigation")
|
|
||||||
: isAiRoute
|
|
||||||
? t("AI navigation")
|
|
||||||
: t("Main navigation")
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
{isSpaceRoute && (
|
{isSpaceRoute && (
|
||||||
<div className={classes.resizeHandle} onMouseDown={startResizing} />
|
<div className={classes.resizeHandle} onMouseDown={startResizing} />
|
||||||
@@ -129,35 +114,19 @@ export default function GlobalAppShell({
|
|||||||
{isAiRoute && <AiChatSidebar />}
|
{isAiRoute && <AiChatSidebar />}
|
||||||
{showGlobalSidebar && <GlobalSidebar />}
|
{showGlobalSidebar && <GlobalSidebar />}
|
||||||
</AppShell.Navbar>
|
</AppShell.Navbar>
|
||||||
<AppShell.Main id="main-content">
|
<AppShell.Main>
|
||||||
{isSettingsRoute ? (
|
{isSettingsRoute ? (
|
||||||
<Container size={900} pb={80}>
|
<Container size={900}>{children}</Container>
|
||||||
{children}
|
|
||||||
</Container>
|
|
||||||
) : (
|
) : (
|
||||||
children
|
children
|
||||||
)}
|
)}
|
||||||
</AppShell.Main>
|
</AppShell.Main>
|
||||||
|
|
||||||
{isPageRoute && (
|
{isPageRoute && (
|
||||||
<AppShell.Aside
|
<AppShell.Aside className={classes.aside} p="md" withBorder={false}>
|
||||||
className={classes.aside}
|
|
||||||
p="md"
|
|
||||||
withBorder={false}
|
|
||||||
aria-label={
|
|
||||||
asideTab === "comments"
|
|
||||||
? t("Comments")
|
|
||||||
: asideTab === "toc"
|
|
||||||
? t("Table of contents")
|
|
||||||
: asideTab === "chat"
|
|
||||||
? t("AI Chat")
|
|
||||||
: undefined
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<Aside />
|
<Aside />
|
||||||
</AppShell.Aside>
|
</AppShell.Aside>
|
||||||
)}
|
)}
|
||||||
</AppShell>
|
</AppShell>
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -50,7 +50,7 @@
|
|||||||
.sectionHeader {
|
.sectionHeader {
|
||||||
padding: var(--mantine-spacing-xs) var(--mantine-spacing-sm);
|
padding: var(--mantine-spacing-xs) var(--mantine-spacing-sm);
|
||||||
font-size: var(--mantine-font-size-xs);
|
font-size: var(--mantine-font-size-xs);
|
||||||
color: var(--mantine-color-dimmed);
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-3));
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
letter-spacing: 0.5px;
|
letter-spacing: 0.5px;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { ScrollArea, Text, Divider, Modal, UnstyledButton } from "@mantine/core";
|
import { ScrollArea, Text, Divider, Modal } from "@mantine/core";
|
||||||
import {
|
import {
|
||||||
IconHome,
|
IconHome,
|
||||||
IconClock,
|
IconClock,
|
||||||
@@ -119,13 +119,17 @@ export default function GlobalSidebar() {
|
|||||||
</ScrollArea>
|
</ScrollArea>
|
||||||
|
|
||||||
<div className={classes.bottomSection}>
|
<div className={classes.bottomSection}>
|
||||||
<UnstyledButton
|
<a
|
||||||
className={classes.link}
|
className={classes.link}
|
||||||
onClick={openInvite}
|
onClick={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
openInvite();
|
||||||
|
}}
|
||||||
|
href="#"
|
||||||
>
|
>
|
||||||
<IconUserPlus className={classes.linkIcon} stroke={2} />
|
<IconUserPlus className={classes.linkIcon} stroke={2} />
|
||||||
<span>{t("Invite People")}</span>
|
<span>{t("Invite People")}</span>
|
||||||
</UnstyledButton>
|
</a>
|
||||||
<Link
|
<Link
|
||||||
className={classes.link}
|
className={classes.link}
|
||||||
data-active={active.startsWith("/settings") || undefined}
|
data-active={active.startsWith("/settings") || undefined}
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ export default function AppVersion() {
|
|||||||
>
|
>
|
||||||
<Indicator
|
<Indicator
|
||||||
label={t("New update")}
|
label={t("New update")}
|
||||||
color="dark"
|
color="gray"
|
||||||
inline
|
inline
|
||||||
size={16}
|
size={16}
|
||||||
position="middle-end"
|
position="middle-end"
|
||||||
|
|||||||
@@ -13,7 +13,6 @@ import { getShares } from "@/features/share/services/share-service.ts";
|
|||||||
import { getApiKeys } from "@/ee/api-key";
|
import { getApiKeys } from "@/ee/api-key";
|
||||||
import { getAuditLogs } from "@/ee/audit/services/audit-service";
|
import { getAuditLogs } from "@/ee/audit/services/audit-service";
|
||||||
import { getVerificationList } from "@/ee/page-verification/services/page-verification-service";
|
import { getVerificationList } from "@/ee/page-verification/services/page-verification-service";
|
||||||
import { getScimTokens } from "@/ee/scim/services/scim-token-service";
|
|
||||||
|
|
||||||
export const prefetchWorkspaceMembers = () => {
|
export const prefetchWorkspaceMembers = () => {
|
||||||
const params: QueryParams = { limit: 100, query: "" };
|
const params: QueryParams = { limit: 100, query: "" };
|
||||||
@@ -99,10 +98,3 @@ export const prefetchVerifiedPages = () => {
|
|||||||
queryFn: () => getVerificationList(params),
|
queryFn: () => getVerificationList(params),
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
export const prefetchScimTokens = () => {
|
|
||||||
queryClient.prefetchQuery({
|
|
||||||
queryKey: ["scim-token-list", { cursor: undefined }],
|
|
||||||
queryFn: () => getScimTokens({}),
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|||||||
@@ -31,7 +31,6 @@ import {
|
|||||||
prefetchBilling,
|
prefetchBilling,
|
||||||
prefetchGroups,
|
prefetchGroups,
|
||||||
prefetchLicense,
|
prefetchLicense,
|
||||||
prefetchScimTokens,
|
|
||||||
prefetchShares,
|
prefetchShares,
|
||||||
prefetchSpaces,
|
prefetchSpaces,
|
||||||
prefetchSsoProviders,
|
prefetchSsoProviders,
|
||||||
@@ -205,10 +204,7 @@ export default function SettingsSidebar() {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case "Security & SSO":
|
case "Security & SSO":
|
||||||
prefetchHandler = () => {
|
prefetchHandler = prefetchSsoProviders;
|
||||||
prefetchSsoProviders();
|
|
||||||
prefetchScimTokens();
|
|
||||||
};
|
|
||||||
break;
|
break;
|
||||||
case "Public sharing":
|
case "Public sharing":
|
||||||
prefetchHandler = prefetchShares;
|
prefetchHandler = prefetchShares;
|
||||||
@@ -230,6 +226,32 @@ export default function SettingsSidebar() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const isDisabled = isItemDisabled(item);
|
const isDisabled = isItemDisabled(item);
|
||||||
|
const linkElement = (
|
||||||
|
<Link
|
||||||
|
onMouseEnter={!isDisabled ? prefetchHandler : undefined}
|
||||||
|
className={classes.link}
|
||||||
|
data-active={active.startsWith(item.path) || undefined}
|
||||||
|
data-disabled={isDisabled || undefined}
|
||||||
|
key={item.label}
|
||||||
|
to={isDisabled ? "#" : item.path}
|
||||||
|
onClick={(e) => {
|
||||||
|
if (isDisabled) {
|
||||||
|
e.preventDefault();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (mobileSidebarOpened) {
|
||||||
|
toggleMobileSidebar();
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
style={{
|
||||||
|
opacity: isDisabled ? 0.5 : 1,
|
||||||
|
cursor: isDisabled ? "not-allowed" : "pointer",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<item.icon className={classes.linkIcon} stroke={2} />
|
||||||
|
<span>{t(item.label)}</span>
|
||||||
|
</Link>
|
||||||
|
);
|
||||||
|
|
||||||
if (isDisabled) {
|
if (isDisabled) {
|
||||||
return (
|
return (
|
||||||
@@ -239,41 +261,12 @@ export default function SettingsSidebar() {
|
|||||||
position="right"
|
position="right"
|
||||||
withArrow
|
withArrow
|
||||||
>
|
>
|
||||||
<span
|
{linkElement}
|
||||||
className={classes.link}
|
|
||||||
data-disabled
|
|
||||||
role="link"
|
|
||||||
aria-disabled="true"
|
|
||||||
tabIndex={0}
|
|
||||||
style={{
|
|
||||||
opacity: 0.5,
|
|
||||||
cursor: "not-allowed",
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<item.icon className={classes.linkIcon} stroke={2} />
|
|
||||||
<span>{t(item.label)}</span>
|
|
||||||
</span>
|
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return linkElement;
|
||||||
<Link
|
|
||||||
onMouseEnter={prefetchHandler}
|
|
||||||
className={classes.link}
|
|
||||||
data-active={active.startsWith(item.path) || undefined}
|
|
||||||
key={item.label}
|
|
||||||
to={item.path}
|
|
||||||
onClick={() => {
|
|
||||||
if (mobileSidebarOpened) {
|
|
||||||
toggleMobileSidebar();
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<item.icon className={classes.linkIcon} stroke={2} />
|
|
||||||
<span>{t(item.label)}</span>
|
|
||||||
</Link>
|
|
||||||
);
|
|
||||||
})}
|
})}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
@@ -291,7 +284,7 @@ export default function SettingsSidebar() {
|
|||||||
}}
|
}}
|
||||||
variant="transparent"
|
variant="transparent"
|
||||||
c="gray"
|
c="gray"
|
||||||
aria-label={t("Back")}
|
aria-label="Back"
|
||||||
>
|
>
|
||||||
<IconArrowLeft stroke={2} />
|
<IconArrowLeft stroke={2} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import { Avatar, MantineColor } from "@mantine/core";
|
import { Avatar } from "@mantine/core";
|
||||||
import { getAvatarUrl } from "@/lib/config.ts";
|
import { getAvatarUrl } from "@/lib/config.ts";
|
||||||
import { AvatarIconType } from "@/features/attachments/types/attachment.types.ts";
|
import { AvatarIconType } from "@/features/attachments/types/attachment.types.ts";
|
||||||
|
|
||||||
@@ -16,39 +16,11 @@ interface CustomAvatarProps {
|
|||||||
mt?: string | number;
|
mt?: string | number;
|
||||||
}
|
}
|
||||||
|
|
||||||
// `color.shade` pairs whose filled background meets WCAG AA (4.5:1) against
|
|
||||||
// white text. Avoids lime/yellow/green/orange — even their dark shades have
|
|
||||||
// weak white-text contrast.
|
|
||||||
const SAFE_INITIALS_COLORS: MantineColor[] = [
|
|
||||||
"blue.8",
|
|
||||||
"cyan.9",
|
|
||||||
"grape.7",
|
|
||||||
"indigo.7",
|
|
||||||
"pink.8",
|
|
||||||
"red.8",
|
|
||||||
"violet.7",
|
|
||||||
];
|
|
||||||
|
|
||||||
function hashName(input: string) {
|
|
||||||
let hash = 0;
|
|
||||||
for (let i = 0; i < input.length; i += 1) {
|
|
||||||
hash = (hash << 5) - hash + input.charCodeAt(i);
|
|
||||||
hash |= 0;
|
|
||||||
}
|
|
||||||
return Math.abs(hash);
|
|
||||||
}
|
|
||||||
|
|
||||||
function pickInitialsColor(name: string) {
|
|
||||||
return SAFE_INITIALS_COLORS[hashName(name) % SAFE_INITIALS_COLORS.length];
|
|
||||||
}
|
|
||||||
|
|
||||||
export const CustomAvatar = React.forwardRef<
|
export const CustomAvatar = React.forwardRef<
|
||||||
HTMLInputElement,
|
HTMLInputElement,
|
||||||
CustomAvatarProps
|
CustomAvatarProps
|
||||||
>(({ avatarUrl, name, type, color, ...props }: CustomAvatarProps, ref) => {
|
>(({ avatarUrl, name, type, ...props }: CustomAvatarProps, ref) => {
|
||||||
const avatarLink = getAvatarUrl(avatarUrl, type);
|
const avatarLink = getAvatarUrl(avatarUrl, type);
|
||||||
const resolvedColor =
|
|
||||||
!color || color === "initials" ? pickInitialsColor(name ?? "") : color;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Avatar
|
<Avatar
|
||||||
@@ -56,7 +28,7 @@ export const CustomAvatar = React.forwardRef<
|
|||||||
src={avatarLink}
|
src={avatarLink}
|
||||||
name={name}
|
name={name}
|
||||||
alt={name}
|
alt={name}
|
||||||
color={resolvedColor}
|
color="initials"
|
||||||
{...props}
|
{...props}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -74,18 +74,7 @@ export function PageChildren({
|
|||||||
/>
|
/>
|
||||||
))}
|
))}
|
||||||
{hasNextPage && (
|
{hasNextPage && (
|
||||||
<div
|
<div className={classes.loadMore} onClick={() => fetchNextPage()}>
|
||||||
className={classes.loadMore}
|
|
||||||
onClick={() => fetchNextPage()}
|
|
||||||
onKeyDown={(e) => {
|
|
||||||
if (e.key === "Enter" || e.key === " ") {
|
|
||||||
e.preventDefault();
|
|
||||||
fetchNextPage();
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
role="button"
|
|
||||||
tabIndex={0}
|
|
||||||
>
|
|
||||||
{t("Load more")}
|
{t("Load more")}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -70,14 +70,11 @@ function EmojiPicker({
|
|||||||
closeOnEscape={true}
|
closeOnEscape={true}
|
||||||
>
|
>
|
||||||
<Popover.Target ref={setTarget}>
|
<Popover.Target ref={setTarget}>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
c={actionIconProps?.c || "gray"}
|
c={actionIconProps?.c || "gray"}
|
||||||
variant={actionIconProps?.variant || "transparent"}
|
variant={actionIconProps?.variant || "transparent"}
|
||||||
size={actionIconProps?.size}
|
size={actionIconProps?.size}
|
||||||
onClick={handlers.toggle}
|
onClick={handlers.toggle}
|
||||||
aria-label={t("Pick emoji")}
|
|
||||||
aria-haspopup="dialog"
|
|
||||||
aria-expanded={opened}
|
|
||||||
>
|
>
|
||||||
{icon}
|
{icon}
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import classes from "../styles/chat-sidebar.module.css";
|
|||||||
type Props = {
|
type Props = {
|
||||||
chat: AiChat;
|
chat: AiChat;
|
||||||
isActive: boolean;
|
isActive: boolean;
|
||||||
onDelete: (chatId: string, title: string | null) => void;
|
onDelete: (chatId: string) => void;
|
||||||
onRename: (chatId: string, title: string) => void;
|
onRename: (chatId: string, title: string) => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -132,7 +132,6 @@ export default function AiChatSidebarItem({
|
|||||||
size="xs"
|
size="xs"
|
||||||
color="gray"
|
color="gray"
|
||||||
onClick={(e) => e.preventDefault()}
|
onClick={(e) => e.preventDefault()}
|
||||||
aria-label={t("Chat menu")}
|
|
||||||
>
|
>
|
||||||
<IconDots size={14} />
|
<IconDots size={14} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
@@ -154,7 +153,7 @@ export default function AiChatSidebarItem({
|
|||||||
onClick={(e) => {
|
onClick={(e) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
onDelete(chat.id, chat.title);
|
onDelete(chat.id);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{t("Delete")}
|
{t("Delete")}
|
||||||
|
|||||||
@@ -1,14 +1,6 @@
|
|||||||
import { useState, useCallback, useEffect, useMemo, useRef } from "react";
|
import { useState, useCallback, useEffect, useMemo, useRef } from "react";
|
||||||
import { Link, useNavigate, useParams } from "react-router-dom";
|
import { Link, useNavigate, useParams } from "react-router-dom";
|
||||||
import {
|
import { ActionIcon, Center, TextInput, Loader, Tooltip } from "@mantine/core";
|
||||||
ActionIcon,
|
|
||||||
Center,
|
|
||||||
Text,
|
|
||||||
TextInput,
|
|
||||||
Loader,
|
|
||||||
Tooltip,
|
|
||||||
} from "@mantine/core";
|
|
||||||
import { modals } from "@mantine/modals";
|
|
||||||
import { useDebouncedValue } from "@mantine/hooks";
|
import { useDebouncedValue } from "@mantine/hooks";
|
||||||
import { IconPlus, IconSearch, IconMessageCircle2 } from "@tabler/icons-react";
|
import { IconPlus, IconSearch, IconMessageCircle2 } from "@tabler/icons-react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
@@ -81,31 +73,16 @@ export default function AiChatSidebar() {
|
|||||||
);
|
);
|
||||||
|
|
||||||
const handleDelete = useCallback(
|
const handleDelete = useCallback(
|
||||||
(id: string, title: string | null) => {
|
(id: string) => {
|
||||||
modals.openConfirmModal({
|
deleteMutation.mutate(id, {
|
||||||
title: t("Delete chat"),
|
onSuccess: () => {
|
||||||
centered: true,
|
if (chatId === id) {
|
||||||
children: (
|
navigate("/ai");
|
||||||
<Text size="sm">
|
}
|
||||||
{t("Are you sure you want to delete '{{title}}'? This action cannot be undone.", {
|
|
||||||
title: title || t("Untitled"),
|
|
||||||
})}
|
|
||||||
</Text>
|
|
||||||
),
|
|
||||||
labels: { confirm: t("Delete"), cancel: t("Cancel") },
|
|
||||||
confirmProps: { color: "red" },
|
|
||||||
onConfirm: () => {
|
|
||||||
deleteMutation.mutate(id, {
|
|
||||||
onSuccess: () => {
|
|
||||||
if (chatId === id) {
|
|
||||||
navigate("/ai");
|
|
||||||
}
|
|
||||||
},
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
[deleteMutation, chatId, navigate, t],
|
[deleteMutation, chatId, navigate],
|
||||||
);
|
);
|
||||||
|
|
||||||
const handleRename = useCallback(
|
const handleRename = useCallback(
|
||||||
@@ -137,8 +114,7 @@ export default function AiChatSidebar() {
|
|||||||
|
|
||||||
<TextInput
|
<TextInput
|
||||||
className={classes.searchInput}
|
className={classes.searchInput}
|
||||||
placeholder={t("Search chats...")}
|
placeholder="Search chats..."
|
||||||
aria-label={t("Search chats")}
|
|
||||||
leftSection={<IconSearch size={14} />}
|
leftSection={<IconSearch size={14} />}
|
||||||
size="xs"
|
size="xs"
|
||||||
value={search}
|
value={search}
|
||||||
|
|||||||
@@ -178,7 +178,6 @@ export default function AsideChatPanel() {
|
|||||||
href="/ai"
|
href="/ai"
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
color="dark"
|
color="dark"
|
||||||
aria-label={t("New chat")}
|
|
||||||
onClick={handleNewChat}
|
onClick={handleNewChat}
|
||||||
>
|
>
|
||||||
<IconPlus size={20} stroke={1.75} />
|
<IconPlus size={20} stroke={1.75} />
|
||||||
@@ -186,23 +185,13 @@ export default function AsideChatPanel() {
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip label={t("Open full page")} openDelay={250}>
|
<Tooltip label={t("Open full page")} openDelay={250}>
|
||||||
<ActionIcon
|
<ActionIcon variant="subtle" color="dark" onClick={handleExpand}>
|
||||||
variant="subtle"
|
|
||||||
color="dark"
|
|
||||||
aria-label={t("Open full page")}
|
|
||||||
onClick={handleExpand}
|
|
||||||
>
|
|
||||||
<IconArrowsDiagonal size={18} stroke={1.5} />
|
<IconArrowsDiagonal size={18} stroke={1.5} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip label={t("Close")} openDelay={250}>
|
<Tooltip label={t("Close")} openDelay={250}>
|
||||||
<ActionIcon
|
<ActionIcon variant="subtle" color="dark" onClick={handleClose}>
|
||||||
variant="subtle"
|
|
||||||
color="dark"
|
|
||||||
aria-label={t("Close")}
|
|
||||||
onClick={handleClose}
|
|
||||||
>
|
|
||||||
<IconX size={20} stroke={1.75} />
|
<IconX size={20} stroke={1.75} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|||||||
@@ -225,10 +225,6 @@ export default function ChatInput({
|
|||||||
}),
|
}),
|
||||||
],
|
],
|
||||||
editorProps: {
|
editorProps: {
|
||||||
attributes: {
|
|
||||||
"aria-label": placeholder || t("Ask anything... Use @ to mention pages"),
|
|
||||||
"aria-multiline": "true",
|
|
||||||
},
|
|
||||||
handleDOMEvents: {
|
handleDOMEvents: {
|
||||||
keydown: (_view, event) => {
|
keydown: (_view, event) => {
|
||||||
if (
|
if (
|
||||||
@@ -279,8 +275,6 @@ export default function ChatInput({
|
|||||||
type="file"
|
type="file"
|
||||||
accept={ACCEPTED_FILE_TYPES}
|
accept={ACCEPTED_FILE_TYPES}
|
||||||
multiple
|
multiple
|
||||||
aria-label={t("Add files")}
|
|
||||||
tabIndex={-1}
|
|
||||||
style={{ display: "none" }}
|
style={{ display: "none" }}
|
||||||
onChange={(e) => handleFileSelect(e.target.files)}
|
onChange={(e) => handleFileSelect(e.target.files)}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -31,16 +31,7 @@ export default function ChatToolGroup({ toolCalls, isStreaming }: Props) {
|
|||||||
<div className={classes.toolGroup}>
|
<div className={classes.toolGroup}>
|
||||||
<div
|
<div
|
||||||
className={classes.toolGroupHeader}
|
className={classes.toolGroupHeader}
|
||||||
role="button"
|
|
||||||
tabIndex={0}
|
|
||||||
aria-expanded={expanded}
|
|
||||||
onClick={() => setExpanded((prev) => !prev)}
|
onClick={() => setExpanded((prev) => !prev)}
|
||||||
onKeyDown={(event) => {
|
|
||||||
if (event.key === "Enter" || event.key === " ") {
|
|
||||||
event.preventDefault();
|
|
||||||
setExpanded((prev) => !prev);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
{activeLabel ? (
|
{activeLabel ? (
|
||||||
<IconLoader2 size={12} className={classes.processingSpinner} />
|
<IconLoader2 size={12} className={classes.processingSpinner} />
|
||||||
|
|||||||
@@ -98,7 +98,7 @@
|
|||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
letter-spacing: 0.08em;
|
letter-spacing: 0.08em;
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
color: var(--mantine-color-dimmed);
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
margin-bottom: var(--mantine-spacing-xs);
|
margin-bottom: var(--mantine-spacing-xs);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -125,7 +125,7 @@
|
|||||||
.suggestionsLabel {
|
.suggestionsLabel {
|
||||||
font-size: var(--mantine-font-size-xs);
|
font-size: var(--mantine-font-size-xs);
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
color: var(--mantine-color-dimmed);
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
letter-spacing: 0.5px;
|
letter-spacing: 0.5px;
|
||||||
margin-bottom: var(--mantine-spacing-sm);
|
margin-bottom: var(--mantine-spacing-sm);
|
||||||
|
|||||||
@@ -43,7 +43,7 @@
|
|||||||
margin-top: 6px;
|
margin-top: 6px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
font-size: var(--mantine-font-size-xs);
|
font-size: var(--mantine-font-size-xs);
|
||||||
color: var(--mantine-color-dimmed);
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
}
|
}
|
||||||
|
|
||||||
.attachmentChips {
|
.attachmentChips {
|
||||||
|
|||||||
@@ -36,7 +36,7 @@
|
|||||||
padding: 4px var(--mantine-spacing-xs);
|
padding: 4px var(--mantine-spacing-xs);
|
||||||
font-size: var(--mantine-font-size-xs);
|
font-size: var(--mantine-font-size-xs);
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
color: var(--mantine-color-dimmed);
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
user-select: none;
|
user-select: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -104,7 +104,7 @@
|
|||||||
|
|
||||||
.chatItemDate {
|
.chatItemDate {
|
||||||
font-size: var(--mantine-font-size-xs);
|
font-size: var(--mantine-font-size-xs);
|
||||||
color: var(--mantine-color-dimmed);
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
transition: opacity 150ms;
|
transition: opacity 150ms;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ export function ApiKeyCreatedModal({
|
|||||||
<Modal
|
<Modal
|
||||||
opened={opened}
|
opened={opened}
|
||||||
onClose={onClose}
|
onClose={onClose}
|
||||||
title={t("{{credential}} created", { credential: t("API key") })}
|
title={t("API key created")}
|
||||||
size="lg"
|
size="lg"
|
||||||
>
|
>
|
||||||
<Stack gap="md">
|
<Stack gap="md">
|
||||||
@@ -41,8 +41,7 @@ export function ApiKeyCreatedModal({
|
|||||||
color="red"
|
color="red"
|
||||||
>
|
>
|
||||||
{t(
|
{t(
|
||||||
"Make sure to copy your {{credential}} now. You won't be able to see it again!",
|
"Make sure to copy your API key now. You won't be able to see it again!",
|
||||||
{ credential: t("API key") },
|
|
||||||
)}
|
)}
|
||||||
</Alert>
|
</Alert>
|
||||||
|
|
||||||
@@ -65,7 +64,7 @@ export function ApiKeyCreatedModal({
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Button fullWidth onClick={onClose} mt="md">
|
<Button fullWidth onClick={onClose} mt="md">
|
||||||
{t("I've saved my {{credential}}", { credential: t("API key") })}
|
{t("I've saved my API key")}
|
||||||
</Button>
|
</Button>
|
||||||
</Stack>
|
</Stack>
|
||||||
</Modal>
|
</Modal>
|
||||||
|
|||||||
@@ -44,7 +44,7 @@ export function ApiKeyTable({
|
|||||||
<Table.Th>{t("Last used")}</Table.Th>
|
<Table.Th>{t("Last used")}</Table.Th>
|
||||||
<Table.Th>{t("Expires")}</Table.Th>
|
<Table.Th>{t("Expires")}</Table.Th>
|
||||||
<Table.Th>{t("Created")}</Table.Th>
|
<Table.Th>{t("Created")}</Table.Th>
|
||||||
<Table.Th aria-label={t("Action")} />
|
<Table.Th></Table.Th>
|
||||||
</Table.Tr>
|
</Table.Tr>
|
||||||
</Table.Thead>
|
</Table.Thead>
|
||||||
|
|
||||||
@@ -106,11 +106,7 @@ export function ApiKeyTable({
|
|||||||
<Table.Td>
|
<Table.Td>
|
||||||
<Menu position="bottom-end" withinPortal>
|
<Menu position="bottom-end" withinPortal>
|
||||||
<Menu.Target>
|
<Menu.Target>
|
||||||
<ActionIcon
|
<ActionIcon variant="subtle" color="gray">
|
||||||
variant="subtle"
|
|
||||||
color="gray"
|
|
||||||
aria-label={t("API key menu")}
|
|
||||||
>
|
|
||||||
<IconDots size={16} />
|
<IconDots size={16} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Menu.Target>
|
</Menu.Target>
|
||||||
|
|||||||
@@ -105,7 +105,7 @@ export function CreateApiKeyModal({
|
|||||||
<Modal
|
<Modal
|
||||||
opened={opened}
|
opened={opened}
|
||||||
onClose={handleClose}
|
onClose={handleClose}
|
||||||
title={t("Create {{credential}}", { credential: t("API key") })}
|
title={t("Create API Key")}
|
||||||
size="md"
|
size="md"
|
||||||
>
|
>
|
||||||
<form onSubmit={form.onSubmit((values) => handleSubmit(values))}>
|
<form onSubmit={form.onSubmit((values) => handleSubmit(values))}>
|
||||||
|
|||||||
@@ -30,14 +30,12 @@ export function RevokeApiKeyModal({
|
|||||||
<Modal
|
<Modal
|
||||||
opened={opened}
|
opened={opened}
|
||||||
onClose={onClose}
|
onClose={onClose}
|
||||||
title={t("Revoke {{credential}}", { credential: t("API key") })}
|
title={t("Revoke API key")}
|
||||||
size="md"
|
size="md"
|
||||||
>
|
>
|
||||||
<Stack gap="md">
|
<Stack gap="md">
|
||||||
<Text>
|
<Text>
|
||||||
{t("Are you sure you want to revoke this {{credential}}", {
|
{t("Are you sure you want to revoke this API key")}{" "}
|
||||||
credential: t("API key"),
|
|
||||||
})}{" "}
|
|
||||||
<strong>{apiKey?.name}</strong>?
|
<strong>{apiKey?.name}</strong>?
|
||||||
</Text>
|
</Text>
|
||||||
<Text size="sm" c="dimmed">
|
<Text size="sm" c="dimmed">
|
||||||
|
|||||||
@@ -53,7 +53,7 @@ export function UpdateApiKeyModal({
|
|||||||
<Modal
|
<Modal
|
||||||
opened={opened}
|
opened={opened}
|
||||||
onClose={onClose}
|
onClose={onClose}
|
||||||
title={t("Update {{credential}}", { credential: t("API key") })}
|
title={t("Update API key")}
|
||||||
size="md"
|
size="md"
|
||||||
>
|
>
|
||||||
<form onSubmit={form.onSubmit((values) => handleSubmit(values))}>
|
<form onSubmit={form.onSubmit((values) => handleSubmit(values))}>
|
||||||
|
|||||||
@@ -63,11 +63,7 @@ export function useCreateApiKeyMutation() {
|
|||||||
return useMutation<IApiKey, Error, ICreateApiKeyRequest>({
|
return useMutation<IApiKey, Error, ICreateApiKeyRequest>({
|
||||||
mutationFn: (data) => createApiKey(data),
|
mutationFn: (data) => createApiKey(data),
|
||||||
onSuccess: () => {
|
onSuccess: () => {
|
||||||
notifications.show({
|
notifications.show({ message: t("API key created successfully") });
|
||||||
message: t("{{credential}} created successfully", {
|
|
||||||
credential: t("API key"),
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
queryClient.invalidateQueries({
|
queryClient.invalidateQueries({
|
||||||
predicate: (item) =>
|
predicate: (item) =>
|
||||||
["api-key-list"].includes(item.queryKey[0] as string),
|
["api-key-list"].includes(item.queryKey[0] as string),
|
||||||
|
|||||||
@@ -33,10 +33,6 @@ export const auditEventLabels: Record<string, string> = {
|
|||||||
"api_key.updated": "Updated API key",
|
"api_key.updated": "Updated API key",
|
||||||
"api_key.deleted": "Deleted API key",
|
"api_key.deleted": "Deleted API key",
|
||||||
|
|
||||||
"scim_token.created": "Created SCIM token",
|
|
||||||
"scim_token.updated": "Updated SCIM token",
|
|
||||||
"scim_token.deleted": "Deleted SCIM token",
|
|
||||||
|
|
||||||
"space.created": "Created space",
|
"space.created": "Created space",
|
||||||
"space.updated": "Updated space",
|
"space.updated": "Updated space",
|
||||||
"space.deleted": "Deleted space",
|
"space.deleted": "Deleted space",
|
||||||
@@ -178,14 +174,6 @@ export const eventFilterOptions: EventGroup[] = [
|
|||||||
{ value: "api_key.deleted", label: "Deleted API key" },
|
{ value: "api_key.deleted", label: "Deleted API key" },
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
|
||||||
group: "SCIM token",
|
|
||||||
items: [
|
|
||||||
{ value: "scim_token.created", label: "Created SCIM token" },
|
|
||||||
{ value: "scim_token.updated", label: "Updated SCIM token" },
|
|
||||||
{ value: "scim_token.deleted", label: "Deleted SCIM token" },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
group: "License",
|
group: "License",
|
||||||
items: [
|
items: [
|
||||||
|
|||||||
@@ -8,7 +8,6 @@ export const Feature = {
|
|||||||
AI: 'ai',
|
AI: 'ai',
|
||||||
CONFLUENCE_IMPORT: 'import:confluence',
|
CONFLUENCE_IMPORT: 'import:confluence',
|
||||||
DOCX_IMPORT: 'import:docx',
|
DOCX_IMPORT: 'import:docx',
|
||||||
PDF_IMPORT: 'import:pdf',
|
|
||||||
ATTACHMENT_INDEXING: 'attachment:indexing',
|
ATTACHMENT_INDEXING: 'attachment:indexing',
|
||||||
SECURITY_SETTINGS: 'security:settings',
|
SECURITY_SETTINGS: 'security:settings',
|
||||||
MCP: 'mcp',
|
MCP: 'mcp',
|
||||||
|
|||||||
@@ -140,7 +140,7 @@ export function PagePermissionList({
|
|||||||
)}
|
)}
|
||||||
</Group>
|
</Group>
|
||||||
|
|
||||||
<ScrollArea.Autosize mah={400} viewportRef={viewportRef}>
|
<ScrollArea mah={250} viewportRef={viewportRef}>
|
||||||
{sortedMembers.map((member) => (
|
{sortedMembers.map((member) => (
|
||||||
<PagePermissionItem
|
<PagePermissionItem
|
||||||
key={`${member.type}-${member.id}`}
|
key={`${member.type}-${member.id}`}
|
||||||
@@ -158,7 +158,7 @@ export function PagePermissionList({
|
|||||||
<Loader size="xs" />
|
<Loader size="xs" />
|
||||||
</Center>
|
</Center>
|
||||||
)}
|
)}
|
||||||
</ScrollArea.Autosize>
|
</ScrollArea>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,4 @@
|
|||||||
import {
|
import { ActionIcon, Group, Menu, Modal, Text, Tooltip } from "@mantine/core";
|
||||||
ActionIcon,
|
|
||||||
Group,
|
|
||||||
Menu,
|
|
||||||
Modal,
|
|
||||||
Text,
|
|
||||||
ThemeIcon,
|
|
||||||
Tooltip,
|
|
||||||
} from "@mantine/core";
|
|
||||||
import { useDisclosure } from "@mantine/hooks";
|
import { useDisclosure } from "@mantine/hooks";
|
||||||
import {
|
import {
|
||||||
IconRosetteDiscountCheckFilled,
|
IconRosetteDiscountCheckFilled,
|
||||||
@@ -46,7 +38,6 @@ export function PageVerificationModal({
|
|||||||
<Modal
|
<Modal
|
||||||
opened={opened}
|
opened={opened}
|
||||||
onClose={onClose}
|
onClose={onClose}
|
||||||
aria-label={status === "none" ? t("Set up verification") : t("Verify page")}
|
|
||||||
title={
|
title={
|
||||||
<Group gap="xs">
|
<Group gap="xs">
|
||||||
<IconShieldCheck
|
<IconShieldCheck
|
||||||
@@ -106,9 +97,9 @@ export function PageVerificationBadge({
|
|||||||
withArrow
|
withArrow
|
||||||
openDelay={250}
|
openDelay={250}
|
||||||
>
|
>
|
||||||
<ThemeIcon variant="subtle" color="gray">
|
<ActionIcon variant="subtle" color="gray">
|
||||||
<IconShieldCheck size={20} stroke={1.5} />
|
<IconShieldCheck size={20} stroke={1.5} />
|
||||||
</ThemeIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -139,12 +130,7 @@ export function PageVerificationBadge({
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
) : !readOnly ? (
|
) : !readOnly ? (
|
||||||
<Tooltip label={t("Set up verification")} withArrow openDelay={250}>
|
<Tooltip label={t("Set up verification")} withArrow openDelay={250}>
|
||||||
<ActionIcon
|
<ActionIcon variant="subtle" color="gray" onClick={open}>
|
||||||
variant="subtle"
|
|
||||||
color="gray"
|
|
||||||
aria-label={t("Set up verification")}
|
|
||||||
onClick={open}
|
|
||||||
>
|
|
||||||
<IconShieldCheck size={20} stroke={1.5} />
|
<IconShieldCheck size={20} stroke={1.5} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|||||||
@@ -1,64 +0,0 @@
|
|||||||
import "@/features/editor/styles/index.css";
|
|
||||||
import { useEffect, useState } from "react";
|
|
||||||
import { useParams, useSearchParams } from "react-router-dom";
|
|
||||||
import ReadonlyPageEditor from "@/features/editor/readonly-page-editor";
|
|
||||||
import { Container } from "@mantine/core";
|
|
||||||
|
|
||||||
type PdfRenderData = {
|
|
||||||
pageId: string;
|
|
||||||
title: string;
|
|
||||||
content: any;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default function PdfRenderPage() {
|
|
||||||
const { pageId } = useParams<{ pageId: string }>();
|
|
||||||
const [searchParams] = useSearchParams();
|
|
||||||
const token = searchParams.get("token");
|
|
||||||
|
|
||||||
const [data, setData] = useState<PdfRenderData | null>(null);
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (!pageId || !token) {
|
|
||||||
setError("Missing page ID or token");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
fetch('/api/pdf-export/render', {
|
|
||||||
method: 'POST',
|
|
||||||
headers: { 'Content-Type': 'application/json' },
|
|
||||||
body: JSON.stringify({ pageId, token }),
|
|
||||||
})
|
|
||||||
.then((res) => {
|
|
||||||
if (!res.ok) throw new Error(`HTTP ${res.status}`);
|
|
||||||
return res.json();
|
|
||||||
})
|
|
||||||
.then((result) => setData(result.data))
|
|
||||||
.catch((err) => setError(err.message));
|
|
||||||
}, [pageId, token]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (data?.title) {
|
|
||||||
document.title = data.title;
|
|
||||||
}
|
|
||||||
}, [data?.title]);
|
|
||||||
|
|
||||||
if (error) {
|
|
||||||
return <div>{error}</div>;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!data) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Container size={900} p={0}>
|
|
||||||
<ReadonlyPageEditor
|
|
||||||
key={data.pageId}
|
|
||||||
title={data.title}
|
|
||||||
content={data.content}
|
|
||||||
pageId={data.pageId}
|
|
||||||
/>
|
|
||||||
</Container>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,78 +0,0 @@
|
|||||||
import { Modal, TextInput, Button, Group, Stack } from "@mantine/core";
|
|
||||||
import { useForm } from "@mantine/form";
|
|
||||||
import { zod4Resolver } from "mantine-form-zod-resolver";
|
|
||||||
import { z } from "zod/v4";
|
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import { useCreateScimTokenMutation } from "@/ee/scim/queries/scim-token-query";
|
|
||||||
import { IScimToken } from "@/ee/scim/types/scim-token.types";
|
|
||||||
|
|
||||||
interface CreateScimTokenModalProps {
|
|
||||||
opened: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
onSuccess: (response: IScimToken) => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
const formSchema = z.object({
|
|
||||||
name: z.string().min(1, "Name is required"),
|
|
||||||
});
|
|
||||||
type FormValues = z.infer<typeof formSchema>;
|
|
||||||
|
|
||||||
export function CreateScimTokenModal({
|
|
||||||
opened,
|
|
||||||
onClose,
|
|
||||||
onSuccess,
|
|
||||||
}: CreateScimTokenModalProps) {
|
|
||||||
const { t } = useTranslation();
|
|
||||||
const createMutation = useCreateScimTokenMutation();
|
|
||||||
|
|
||||||
const form = useForm<FormValues>({
|
|
||||||
validate: zod4Resolver(formSchema),
|
|
||||||
initialValues: { name: "" },
|
|
||||||
});
|
|
||||||
|
|
||||||
const handleSubmit = async (data: FormValues) => {
|
|
||||||
try {
|
|
||||||
const created = await createMutation.mutateAsync({ name: data.name });
|
|
||||||
onSuccess(created);
|
|
||||||
form.reset();
|
|
||||||
onClose();
|
|
||||||
} catch (err) {
|
|
||||||
//
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleClose = () => {
|
|
||||||
form.reset();
|
|
||||||
onClose();
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal
|
|
||||||
opened={opened}
|
|
||||||
onClose={handleClose}
|
|
||||||
title={t("Create {{credential}}", { credential: t("SCIM token") })}
|
|
||||||
size="md"
|
|
||||||
>
|
|
||||||
<form onSubmit={form.onSubmit((values) => handleSubmit(values))}>
|
|
||||||
<Stack gap="md">
|
|
||||||
<TextInput
|
|
||||||
label={t("Name")}
|
|
||||||
placeholder={t("Enter a descriptive name")}
|
|
||||||
data-autofocus
|
|
||||||
required
|
|
||||||
{...form.getInputProps("name")}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<Group justify="flex-end" mt="md">
|
|
||||||
<Button variant="default" onClick={handleClose}>
|
|
||||||
{t("Cancel")}
|
|
||||||
</Button>
|
|
||||||
<Button type="submit" loading={createMutation.isPending}>
|
|
||||||
{t("Create")}
|
|
||||||
</Button>
|
|
||||||
</Group>
|
|
||||||
</Stack>
|
|
||||||
</form>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
import { Group, Text, Switch, Tooltip } from "@mantine/core";
|
|
||||||
import { useAtom } from "jotai";
|
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
|
||||||
import React, { useState } from "react";
|
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
|
||||||
import { notifications } from "@mantine/notifications";
|
|
||||||
import { useHasFeature } from "@/ee/hooks/use-feature.ts";
|
|
||||||
import { Feature } from "@/ee/features.ts";
|
|
||||||
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label.ts";
|
|
||||||
|
|
||||||
export default function EnableScim() {
|
|
||||||
const { t } = useTranslation();
|
|
||||||
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
|
||||||
const [checked, setChecked] = useState(workspace?.isScimEnabled ?? false);
|
|
||||||
const hasAccess = useHasFeature(Feature.SCIM);
|
|
||||||
const upgradeLabel = useUpgradeLabel();
|
|
||||||
|
|
||||||
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
|
||||||
const value = event.currentTarget.checked;
|
|
||||||
try {
|
|
||||||
const updatedWorkspace = await updateWorkspace({ isScimEnabled: value });
|
|
||||||
setChecked(value);
|
|
||||||
setWorkspace(updatedWorkspace);
|
|
||||||
} catch (err) {
|
|
||||||
notifications.show({
|
|
||||||
message: err?.response?.data?.message,
|
|
||||||
color: "red",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Group justify="space-between" wrap="nowrap" gap="xl">
|
|
||||||
<div>
|
|
||||||
<Text size="md">{t("Enable SCIM")}</Text>
|
|
||||||
<Text size="sm" c="dimmed">
|
|
||||||
{t(
|
|
||||||
"Automatically provision users and groups from your identity provider via SCIM.",
|
|
||||||
)}
|
|
||||||
</Text>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<Tooltip label={upgradeLabel} disabled={hasAccess} refProp="rootRef">
|
|
||||||
<Switch
|
|
||||||
labelPosition="left"
|
|
||||||
defaultChecked={checked}
|
|
||||||
onChange={handleChange}
|
|
||||||
disabled={!hasAccess}
|
|
||||||
aria-label={t("Toggle SCIM provisioning")}
|
|
||||||
/>
|
|
||||||
</Tooltip>
|
|
||||||
</Group>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
import { Modal, Text, Button, Group, Stack } from "@mantine/core";
|
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import { useRevokeScimTokenMutation } from "@/ee/scim/queries/scim-token-query";
|
|
||||||
import { IScimToken } from "@/ee/scim/types/scim-token.types";
|
|
||||||
|
|
||||||
interface RevokeScimTokenModalProps {
|
|
||||||
opened: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
scimToken: IScimToken | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function RevokeScimTokenModal({
|
|
||||||
opened,
|
|
||||||
onClose,
|
|
||||||
scimToken,
|
|
||||||
}: RevokeScimTokenModalProps) {
|
|
||||||
const { t } = useTranslation();
|
|
||||||
const revokeMutation = useRevokeScimTokenMutation();
|
|
||||||
|
|
||||||
const handleRevoke = async () => {
|
|
||||||
if (!scimToken) return;
|
|
||||||
await revokeMutation.mutateAsync({ tokenId: scimToken.id });
|
|
||||||
onClose();
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal
|
|
||||||
opened={opened}
|
|
||||||
onClose={onClose}
|
|
||||||
title={t("Revoke {{credential}}", { credential: t("SCIM token") })}
|
|
||||||
size="md"
|
|
||||||
>
|
|
||||||
<Stack gap="md">
|
|
||||||
<Text>
|
|
||||||
{t("Are you sure you want to revoke this {{credential}}", {
|
|
||||||
credential: t("SCIM token"),
|
|
||||||
})}{" "}
|
|
||||||
<strong>{scimToken?.name}</strong>?
|
|
||||||
</Text>
|
|
||||||
<Text size="sm" c="dimmed">
|
|
||||||
{t(
|
|
||||||
"This action cannot be undone. Your identity provider will stop syncing immediately.",
|
|
||||||
)}
|
|
||||||
</Text>
|
|
||||||
|
|
||||||
<Group justify="flex-end" mt="md">
|
|
||||||
<Button variant="default" onClick={onClose}>
|
|
||||||
{t("Cancel")}
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
color="red"
|
|
||||||
onClick={handleRevoke}
|
|
||||||
loading={revokeMutation.isPending}
|
|
||||||
>
|
|
||||||
{t("Revoke")}
|
|
||||||
</Button>
|
|
||||||
</Group>
|
|
||||||
</Stack>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,69 +0,0 @@
|
|||||||
import {
|
|
||||||
Modal,
|
|
||||||
Text,
|
|
||||||
Stack,
|
|
||||||
Alert,
|
|
||||||
Group,
|
|
||||||
Button,
|
|
||||||
TextInput,
|
|
||||||
} from "@mantine/core";
|
|
||||||
import { IconAlertTriangle } from "@tabler/icons-react";
|
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import CopyTextButton from "@/components/common/copy.tsx";
|
|
||||||
import { IScimToken } from "@/ee/scim/types/scim-token.types";
|
|
||||||
|
|
||||||
interface ScimTokenCreatedModalProps {
|
|
||||||
opened: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
scimToken: IScimToken | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function ScimTokenCreatedModal({
|
|
||||||
opened,
|
|
||||||
onClose,
|
|
||||||
scimToken,
|
|
||||||
}: ScimTokenCreatedModalProps) {
|
|
||||||
const { t } = useTranslation();
|
|
||||||
if (!scimToken) return null;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal
|
|
||||||
opened={opened}
|
|
||||||
onClose={onClose}
|
|
||||||
title={t("{{credential}} created", { credential: t("SCIM token") })}
|
|
||||||
size="lg"
|
|
||||||
>
|
|
||||||
<Stack gap="md">
|
|
||||||
<Alert
|
|
||||||
icon={<IconAlertTriangle size={16} />}
|
|
||||||
title={t("Important")}
|
|
||||||
color="red"
|
|
||||||
>
|
|
||||||
{t(
|
|
||||||
"Make sure to copy your {{credential}} now. You won't be able to see it again!",
|
|
||||||
{ credential: t("SCIM token") },
|
|
||||||
)}
|
|
||||||
</Alert>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<Text size="sm" fw={500} mb="xs">
|
|
||||||
{t("SCIM token")}
|
|
||||||
</Text>
|
|
||||||
<Group gap="xs" wrap="nowrap">
|
|
||||||
<TextInput
|
|
||||||
variant="filled"
|
|
||||||
style={{ flex: 1 }}
|
|
||||||
value={scimToken.token}
|
|
||||||
readOnly
|
|
||||||
/>
|
|
||||||
<CopyTextButton text={scimToken.token} />
|
|
||||||
</Group>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<Button fullWidth onClick={onClose} mt="md">
|
|
||||||
{t("I've saved my {{credential}}", { credential: t("SCIM token") })}
|
|
||||||
</Button>
|
|
||||||
</Stack>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,130 +0,0 @@
|
|||||||
import { ActionIcon, Group, Menu, Table, Text } from "@mantine/core";
|
|
||||||
import { IconDots, IconEdit, IconTrash } from "@tabler/icons-react";
|
|
||||||
import { format } from "date-fns";
|
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import { CustomAvatar } from "@/components/ui/custom-avatar.tsx";
|
|
||||||
import React from "react";
|
|
||||||
import NoTableResults from "@/components/common/no-table-results";
|
|
||||||
import { IScimToken } from "@/ee/scim/types/scim-token.types";
|
|
||||||
|
|
||||||
interface ScimTokenTableProps {
|
|
||||||
tokens: IScimToken[];
|
|
||||||
isLoading?: boolean;
|
|
||||||
onUpdate?: (token: IScimToken) => void;
|
|
||||||
onRevoke?: (token: IScimToken) => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function ScimTokenTable({
|
|
||||||
tokens,
|
|
||||||
isLoading,
|
|
||||||
onUpdate,
|
|
||||||
onRevoke,
|
|
||||||
}: ScimTokenTableProps) {
|
|
||||||
const { t } = useTranslation();
|
|
||||||
|
|
||||||
const formatDate = (date: Date | string | null) => {
|
|
||||||
if (!date) return t("Never");
|
|
||||||
return format(new Date(date), "MMM dd, yyyy");
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Table.ScrollContainer minWidth={500}>
|
|
||||||
<Table highlightOnHover verticalSpacing="sm">
|
|
||||||
<Table.Thead>
|
|
||||||
<Table.Tr>
|
|
||||||
<Table.Th>{t("Name")}</Table.Th>
|
|
||||||
<Table.Th>{t("Token")}</Table.Th>
|
|
||||||
<Table.Th>{t("Created by")}</Table.Th>
|
|
||||||
<Table.Th>{t("Last used")}</Table.Th>
|
|
||||||
<Table.Th>{t("Created")}</Table.Th>
|
|
||||||
<Table.Th aria-label={t("Action")} />
|
|
||||||
</Table.Tr>
|
|
||||||
</Table.Thead>
|
|
||||||
|
|
||||||
<Table.Tbody>
|
|
||||||
{tokens && tokens.length > 0 ? (
|
|
||||||
tokens.map((token) => (
|
|
||||||
<Table.Tr key={token.id}>
|
|
||||||
<Table.Td>
|
|
||||||
<Text fz="sm" fw={500}>
|
|
||||||
{token.name}
|
|
||||||
</Text>
|
|
||||||
</Table.Td>
|
|
||||||
|
|
||||||
<Table.Td>
|
|
||||||
<Text fz="sm" ff="monospace" c="dimmed">
|
|
||||||
••••{token.tokenLastFour}
|
|
||||||
</Text>
|
|
||||||
</Table.Td>
|
|
||||||
|
|
||||||
{token.creator ? (
|
|
||||||
<Table.Td>
|
|
||||||
<Group gap="4" wrap="nowrap">
|
|
||||||
<CustomAvatar
|
|
||||||
avatarUrl={token.creator?.avatarUrl}
|
|
||||||
name={token.creator.name}
|
|
||||||
size="sm"
|
|
||||||
/>
|
|
||||||
<Text fz="sm" lineClamp={1}>
|
|
||||||
{token.creator.name}
|
|
||||||
</Text>
|
|
||||||
</Group>
|
|
||||||
</Table.Td>
|
|
||||||
) : (
|
|
||||||
<Table.Td>
|
|
||||||
<Text fz="sm" c="dimmed">
|
|
||||||
—
|
|
||||||
</Text>
|
|
||||||
</Table.Td>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<Table.Td>
|
|
||||||
<Text fz="sm" style={{ whiteSpace: "nowrap" }}>
|
|
||||||
{formatDate(token.lastUsedAt)}
|
|
||||||
</Text>
|
|
||||||
</Table.Td>
|
|
||||||
|
|
||||||
<Table.Td>
|
|
||||||
<Text fz="sm" style={{ whiteSpace: "nowrap" }}>
|
|
||||||
{formatDate(token.createdAt)}
|
|
||||||
</Text>
|
|
||||||
</Table.Td>
|
|
||||||
|
|
||||||
<Table.Td>
|
|
||||||
<Menu position="bottom-end" withinPortal>
|
|
||||||
<Menu.Target>
|
|
||||||
<ActionIcon variant="subtle" color="gray">
|
|
||||||
<IconDots size={16} />
|
|
||||||
</ActionIcon>
|
|
||||||
</Menu.Target>
|
|
||||||
<Menu.Dropdown>
|
|
||||||
{onUpdate && (
|
|
||||||
<Menu.Item
|
|
||||||
leftSection={<IconEdit size={16} />}
|
|
||||||
onClick={() => onUpdate(token)}
|
|
||||||
>
|
|
||||||
{t("Rename")}
|
|
||||||
</Menu.Item>
|
|
||||||
)}
|
|
||||||
{onRevoke && (
|
|
||||||
<Menu.Item
|
|
||||||
leftSection={<IconTrash size={16} />}
|
|
||||||
color="red"
|
|
||||||
onClick={() => onRevoke(token)}
|
|
||||||
>
|
|
||||||
{t("Revoke")}
|
|
||||||
</Menu.Item>
|
|
||||||
)}
|
|
||||||
</Menu.Dropdown>
|
|
||||||
</Menu>
|
|
||||||
</Table.Td>
|
|
||||||
</Table.Tr>
|
|
||||||
))
|
|
||||||
) : (
|
|
||||||
<NoTableResults colSpan={6} />
|
|
||||||
)}
|
|
||||||
</Table.Tbody>
|
|
||||||
</Table>
|
|
||||||
</Table.ScrollContainer>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,30 +0,0 @@
|
|||||||
import { Group, Stack, Text, TextInput } from "@mantine/core";
|
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import CopyTextButton from "@/components/common/copy.tsx";
|
|
||||||
|
|
||||||
export function ScimUrlPanel() {
|
|
||||||
const { t } = useTranslation();
|
|
||||||
const scimUrl = `${window.location.origin}/api/scim/v2`;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Stack gap="xs">
|
|
||||||
<Text size="sm" fw={500}>
|
|
||||||
{t("SCIM endpoint URL")}
|
|
||||||
</Text>
|
|
||||||
<Text size="xs" c="dimmed">
|
|
||||||
{t(
|
|
||||||
"Configure your identity provider with this URL to provision users and groups.",
|
|
||||||
)}
|
|
||||||
</Text>
|
|
||||||
<Group gap="xs" wrap="nowrap">
|
|
||||||
<TextInput
|
|
||||||
variant="filled"
|
|
||||||
style={{ flex: 1 }}
|
|
||||||
value={scimUrl}
|
|
||||||
readOnly
|
|
||||||
/>
|
|
||||||
<CopyTextButton text={scimUrl} />
|
|
||||||
</Group>
|
|
||||||
</Stack>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,77 +0,0 @@
|
|||||||
import { Modal, TextInput, Button, Group, Stack } from "@mantine/core";
|
|
||||||
import { useForm } from "@mantine/form";
|
|
||||||
import { zod4Resolver } from "mantine-form-zod-resolver";
|
|
||||||
import { z } from "zod/v4";
|
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
import { useEffect } from "react";
|
|
||||||
import { useUpdateScimTokenMutation } from "@/ee/scim/queries/scim-token-query";
|
|
||||||
import { IScimToken } from "@/ee/scim/types/scim-token.types";
|
|
||||||
|
|
||||||
const formSchema = z.object({
|
|
||||||
name: z.string().min(1, "Name is required"),
|
|
||||||
});
|
|
||||||
type FormValues = z.infer<typeof formSchema>;
|
|
||||||
|
|
||||||
interface UpdateScimTokenModalProps {
|
|
||||||
opened: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
scimToken: IScimToken | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function UpdateScimTokenModal({
|
|
||||||
opened,
|
|
||||||
onClose,
|
|
||||||
scimToken,
|
|
||||||
}: UpdateScimTokenModalProps) {
|
|
||||||
const { t } = useTranslation();
|
|
||||||
const updateMutation = useUpdateScimTokenMutation();
|
|
||||||
|
|
||||||
const form = useForm<FormValues>({
|
|
||||||
validate: zod4Resolver(formSchema),
|
|
||||||
initialValues: { name: "" },
|
|
||||||
});
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (opened && scimToken) {
|
|
||||||
form.setValues({ name: scimToken.name });
|
|
||||||
}
|
|
||||||
}, [opened, scimToken]);
|
|
||||||
|
|
||||||
const handleSubmit = async (data: FormValues) => {
|
|
||||||
if (!scimToken) return;
|
|
||||||
await updateMutation.mutateAsync({
|
|
||||||
tokenId: scimToken.id,
|
|
||||||
name: data.name,
|
|
||||||
});
|
|
||||||
onClose();
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal
|
|
||||||
opened={opened}
|
|
||||||
onClose={onClose}
|
|
||||||
title={t("Update {{credential}}", { credential: t("SCIM token") })}
|
|
||||||
size="md"
|
|
||||||
>
|
|
||||||
<form onSubmit={form.onSubmit((values) => handleSubmit(values))}>
|
|
||||||
<Stack gap="md">
|
|
||||||
<TextInput
|
|
||||||
label={t("Name")}
|
|
||||||
placeholder={t("Enter a descriptive name")}
|
|
||||||
required
|
|
||||||
{...form.getInputProps("name")}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<Group justify="flex-end" mt="md">
|
|
||||||
<Button variant="default" onClick={onClose}>
|
|
||||||
{t("Cancel")}
|
|
||||||
</Button>
|
|
||||||
<Button type="submit" loading={updateMutation.isPending}>
|
|
||||||
{t("Update")}
|
|
||||||
</Button>
|
|
||||||
</Group>
|
|
||||||
</Stack>
|
|
||||||
</form>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
export * from "./types/scim-token.types";
|
|
||||||
export * from "./services/scim-token-service";
|
|
||||||
@@ -1,96 +0,0 @@
|
|||||||
import { IPagination, QueryParams } from "@/lib/types.ts";
|
|
||||||
import {
|
|
||||||
keepPreviousData,
|
|
||||||
useMutation,
|
|
||||||
useQuery,
|
|
||||||
useQueryClient,
|
|
||||||
UseQueryResult,
|
|
||||||
} from "@tanstack/react-query";
|
|
||||||
import {
|
|
||||||
createScimToken,
|
|
||||||
getScimTokens,
|
|
||||||
revokeScimToken,
|
|
||||||
updateScimToken,
|
|
||||||
} from "@/ee/scim/services/scim-token-service";
|
|
||||||
import {
|
|
||||||
IScimToken,
|
|
||||||
ICreateScimTokenRequest,
|
|
||||||
IRevokeScimTokenRequest,
|
|
||||||
IUpdateScimTokenRequest,
|
|
||||||
} from "@/ee/scim/types/scim-token.types";
|
|
||||||
import { notifications } from "@mantine/notifications";
|
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
|
|
||||||
export function useGetScimTokensQuery(
|
|
||||||
params?: QueryParams,
|
|
||||||
): UseQueryResult<IPagination<IScimToken>, Error> {
|
|
||||||
return useQuery({
|
|
||||||
queryKey: ["scim-token-list", params],
|
|
||||||
queryFn: () => getScimTokens(params),
|
|
||||||
placeholderData: keepPreviousData,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useCreateScimTokenMutation() {
|
|
||||||
const queryClient = useQueryClient();
|
|
||||||
const { t } = useTranslation();
|
|
||||||
|
|
||||||
return useMutation<IScimToken, Error, ICreateScimTokenRequest>({
|
|
||||||
mutationFn: (data) => createScimToken(data),
|
|
||||||
onSuccess: () => {
|
|
||||||
notifications.show({
|
|
||||||
message: t("{{credential}} created successfully", {
|
|
||||||
credential: t("SCIM token"),
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
queryClient.invalidateQueries({
|
|
||||||
predicate: (item) =>
|
|
||||||
["scim-token-list"].includes(item.queryKey[0] as string),
|
|
||||||
});
|
|
||||||
},
|
|
||||||
onError: (error) => {
|
|
||||||
const errorMessage = error["response"]?.data?.message;
|
|
||||||
notifications.show({ message: errorMessage, color: "red" });
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useUpdateScimTokenMutation() {
|
|
||||||
const queryClient = useQueryClient();
|
|
||||||
const { t } = useTranslation();
|
|
||||||
|
|
||||||
return useMutation<void, Error, IUpdateScimTokenRequest>({
|
|
||||||
mutationFn: (data) => updateScimToken(data),
|
|
||||||
onSuccess: () => {
|
|
||||||
notifications.show({ message: t("Updated successfully") });
|
|
||||||
queryClient.invalidateQueries({
|
|
||||||
predicate: (item) =>
|
|
||||||
["scim-token-list"].includes(item.queryKey[0] as string),
|
|
||||||
});
|
|
||||||
},
|
|
||||||
onError: (error) => {
|
|
||||||
const errorMessage = error["response"]?.data?.message;
|
|
||||||
notifications.show({ message: errorMessage, color: "red" });
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useRevokeScimTokenMutation() {
|
|
||||||
const queryClient = useQueryClient();
|
|
||||||
const { t } = useTranslation();
|
|
||||||
|
|
||||||
return useMutation<void, Error, IRevokeScimTokenRequest>({
|
|
||||||
mutationFn: (data) => revokeScimToken(data),
|
|
||||||
onSuccess: () => {
|
|
||||||
notifications.show({ message: t("Revoked successfully") });
|
|
||||||
queryClient.invalidateQueries({
|
|
||||||
predicate: (item) =>
|
|
||||||
["scim-token-list"].includes(item.queryKey[0] as string),
|
|
||||||
});
|
|
||||||
},
|
|
||||||
onError: (error) => {
|
|
||||||
const errorMessage = error["response"]?.data?.message;
|
|
||||||
notifications.show({ message: errorMessage, color: "red" });
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
import api from "@/lib/api-client";
|
|
||||||
import {
|
|
||||||
IScimToken,
|
|
||||||
ICreateScimTokenRequest,
|
|
||||||
IRevokeScimTokenRequest,
|
|
||||||
IUpdateScimTokenRequest,
|
|
||||||
} from "@/ee/scim/types/scim-token.types";
|
|
||||||
import { IPagination, QueryParams } from "@/lib/types.ts";
|
|
||||||
|
|
||||||
export async function getScimTokens(
|
|
||||||
params?: QueryParams,
|
|
||||||
): Promise<IPagination<IScimToken>> {
|
|
||||||
const req = await api.post("/scim-tokens", { ...params });
|
|
||||||
return req.data;
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function createScimToken(
|
|
||||||
data: ICreateScimTokenRequest,
|
|
||||||
): Promise<IScimToken> {
|
|
||||||
const req = await api.post<IScimToken>("/scim-tokens/create", data);
|
|
||||||
return req.data;
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function updateScimToken(
|
|
||||||
data: IUpdateScimTokenRequest,
|
|
||||||
): Promise<void> {
|
|
||||||
await api.post("/scim-tokens/update", data);
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function revokeScimToken(
|
|
||||||
data: IRevokeScimTokenRequest,
|
|
||||||
): Promise<void> {
|
|
||||||
await api.post("/scim-tokens/revoke", data);
|
|
||||||
}
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
import { IUser } from "@/features/user/types/user.types.ts";
|
|
||||||
|
|
||||||
export interface IScimToken {
|
|
||||||
id: string;
|
|
||||||
name: string;
|
|
||||||
token?: string;
|
|
||||||
tokenLastFour: string;
|
|
||||||
isEnabled: boolean;
|
|
||||||
creatorId: string;
|
|
||||||
workspaceId: string;
|
|
||||||
lastUsedAt: string | null;
|
|
||||||
createdAt: string;
|
|
||||||
creator?: Partial<IUser>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface ICreateScimTokenRequest {
|
|
||||||
name: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface IUpdateScimTokenRequest {
|
|
||||||
tokenId: string;
|
|
||||||
name: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface IRevokeScimTokenRequest {
|
|
||||||
tokenId: string;
|
|
||||||
}
|
|
||||||
@@ -69,8 +69,8 @@ export default function SsoProviderList() {
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Card shadow="sm" radius="sm">
|
<Card shadow="sm" radius="sm">
|
||||||
<Table.ScrollContainer minWidth={600} maxHeight={400}>
|
<Table.ScrollContainer minWidth={600}>
|
||||||
<Table verticalSpacing="sm" stickyHeader>
|
<Table verticalSpacing="sm">
|
||||||
<Table.Thead>
|
<Table.Thead>
|
||||||
<Table.Tr>
|
<Table.Tr>
|
||||||
<Table.Th>{t("Name")}</Table.Th>
|
<Table.Th>{t("Name")}</Table.Th>
|
||||||
@@ -141,7 +141,6 @@ export default function SsoProviderList() {
|
|||||||
<ActionIcon
|
<ActionIcon
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
color="gray"
|
color="gray"
|
||||||
aria-label={t("Edit {{name}}", { name: provider.name })}
|
|
||||||
onClick={() => handleEdit(provider)}
|
onClick={() => handleEdit(provider)}
|
||||||
>
|
>
|
||||||
<IconPencil size={16} />
|
<IconPencil size={16} />
|
||||||
@@ -153,13 +152,7 @@ export default function SsoProviderList() {
|
|||||||
withinPortal
|
withinPortal
|
||||||
>
|
>
|
||||||
<Menu.Target>
|
<Menu.Target>
|
||||||
<ActionIcon
|
<ActionIcon variant="subtle" color="gray">
|
||||||
variant="subtle"
|
|
||||||
color="gray"
|
|
||||||
aria-label={t("More actions for {{name}}", {
|
|
||||||
name: provider.name,
|
|
||||||
})}
|
|
||||||
>
|
|
||||||
<IconDots size={16} />
|
<IconDots size={16} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Menu.Target>
|
</Menu.Target>
|
||||||
|
|||||||
@@ -1,18 +1,8 @@
|
|||||||
import { Helmet } from "react-helmet-async";
|
import { Helmet } from "react-helmet-async";
|
||||||
import { getAppName, isCloud } from "@/lib/config.ts";
|
import { getAppName, isCloud } from "@/lib/config.ts";
|
||||||
import SettingsTitle from "@/components/settings/settings-title.tsx";
|
import SettingsTitle from "@/components/settings/settings-title.tsx";
|
||||||
import {
|
import { Divider, Title } from "@mantine/core";
|
||||||
Alert,
|
import React from "react";
|
||||||
Button,
|
|
||||||
Card,
|
|
||||||
Divider,
|
|
||||||
Group,
|
|
||||||
Space,
|
|
||||||
Title,
|
|
||||||
Tooltip,
|
|
||||||
} from "@mantine/core";
|
|
||||||
import { IconInfoCircle } from "@tabler/icons-react";
|
|
||||||
import React, { useState } from "react";
|
|
||||||
import useUserRole from "@/hooks/use-user-role.tsx";
|
import useUserRole from "@/hooks/use-user-role.tsx";
|
||||||
import SsoProviderList from "@/ee/security/components/sso-provider-list.tsx";
|
import SsoProviderList from "@/ee/security/components/sso-provider-list.tsx";
|
||||||
import CreateSsoProvider from "@/ee/security/components/create-sso-provider.tsx";
|
import CreateSsoProvider from "@/ee/security/components/create-sso-provider.tsx";
|
||||||
@@ -22,41 +12,16 @@ import { useTranslation } from "react-i18next";
|
|||||||
import EnforceMfa from "@/ee/security/components/enforce-mfa.tsx";
|
import EnforceMfa from "@/ee/security/components/enforce-mfa.tsx";
|
||||||
import DisablePublicSharing from "@/ee/security/components/disable-public-sharing.tsx";
|
import DisablePublicSharing from "@/ee/security/components/disable-public-sharing.tsx";
|
||||||
import TrashRetention from "@/ee/security/components/trash-retention.tsx";
|
import TrashRetention from "@/ee/security/components/trash-retention.tsx";
|
||||||
import { useAtom } from "jotai";
|
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
|
||||||
import { useHasFeature } from "@/ee/hooks/use-feature";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
import { Feature } from "@/ee/features";
|
import { Feature } from "@/ee/features";
|
||||||
import { useGetScimTokensQuery } from "@/ee/scim/queries/scim-token-query";
|
|
||||||
import { ScimUrlPanel } from "@/ee/scim/components/scim-url-panel";
|
|
||||||
import { ScimTokenTable } from "@/ee/scim/components/scim-token-table";
|
|
||||||
import { CreateScimTokenModal } from "@/ee/scim/components/create-scim-token-modal";
|
|
||||||
import { ScimTokenCreatedModal } from "@/ee/scim/components/scim-token-created-modal";
|
|
||||||
import { RevokeScimTokenModal } from "@/ee/scim/components/revoke-scim-token-modal";
|
|
||||||
import { UpdateScimTokenModal } from "@/ee/scim/components/update-scim-token-modal";
|
|
||||||
import EnableScim from "@/ee/scim/components/enable-scim";
|
|
||||||
import { useCursorPaginate } from "@/hooks/use-cursor-paginate";
|
|
||||||
import Paginate from "@/components/common/paginate";
|
|
||||||
import { IScimToken } from "@/ee/scim/types/scim-token.types";
|
|
||||||
|
|
||||||
const SCIM_TOKEN_LIMIT = 5;
|
|
||||||
|
|
||||||
export default function Security() {
|
export default function Security() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { isAdmin } = useUserRole();
|
const { isAdmin } = useUserRole();
|
||||||
const hasCustomSso = useHasFeature(Feature.SSO_CUSTOM);
|
const hasCustomSso = useHasFeature(Feature.SSO_CUSTOM);
|
||||||
const hasScim = useHasFeature(Feature.SCIM);
|
const hasRetention = useHasFeature(Feature.RETENTION);
|
||||||
const [workspace] = useAtom(workspaceAtom);
|
const hasSharingControls = useHasFeature(Feature.SHARING_CONTROLS);
|
||||||
const isScimEnabled = workspace?.isScimEnabled ?? false;
|
|
||||||
|
|
||||||
const { cursor, goNext, goPrev } = useCursorPaginate();
|
|
||||||
const { data: scimData, isLoading: scimLoading } = useGetScimTokensQuery(
|
|
||||||
hasScim && isScimEnabled ? { cursor } : undefined,
|
|
||||||
);
|
|
||||||
|
|
||||||
const [createOpen, setCreateOpen] = useState(false);
|
|
||||||
const [createdToken, setCreatedToken] = useState<IScimToken | null>(null);
|
|
||||||
const [updateTarget, setUpdateTarget] = useState<IScimToken | null>(null);
|
|
||||||
const [revokeTarget, setRevokeTarget] = useState<IScimToken | null>(null);
|
|
||||||
|
|
||||||
if (!isAdmin) {
|
if (!isAdmin) {
|
||||||
return null;
|
return null;
|
||||||
@@ -80,7 +45,7 @@ export default function Security() {
|
|||||||
<Divider my="lg" />
|
<Divider my="lg" />
|
||||||
|
|
||||||
<Title order={4} my="lg">
|
<Title order={4} my="lg">
|
||||||
{t("Single sign-on (SSO)")}
|
Single sign-on (SSO)
|
||||||
</Title>
|
</Title>
|
||||||
|
|
||||||
<EnforceSso />
|
<EnforceSso />
|
||||||
@@ -101,102 +66,6 @@ export default function Security() {
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
<SsoProviderList />
|
<SsoProviderList />
|
||||||
|
|
||||||
{hasScim && (
|
|
||||||
<>
|
|
||||||
<Divider my="xl" />
|
|
||||||
|
|
||||||
<Title order={4} my="lg">
|
|
||||||
{t("SCIM provisioning")}
|
|
||||||
</Title>
|
|
||||||
|
|
||||||
<Alert
|
|
||||||
icon={<IconInfoCircle size={16} />}
|
|
||||||
color="blue"
|
|
||||||
variant="light"
|
|
||||||
mb="md"
|
|
||||||
>
|
|
||||||
{t("SCIM takes precedence over SSO group sync while enabled.")}
|
|
||||||
</Alert>
|
|
||||||
|
|
||||||
<EnableScim />
|
|
||||||
|
|
||||||
<Divider my="lg" />
|
|
||||||
|
|
||||||
<ScimUrlPanel />
|
|
||||||
|
|
||||||
{isScimEnabled && (
|
|
||||||
<>
|
|
||||||
<Divider my="lg" />
|
|
||||||
|
|
||||||
<Group justify="space-between" mb="md">
|
|
||||||
<Title order={5}>{t("SCIM tokens")}</Title>
|
|
||||||
<Tooltip
|
|
||||||
label={t(
|
|
||||||
"You have reached the maximum of {{max}} SCIM tokens. Delete an existing token to create a new one.",
|
|
||||||
{ max: SCIM_TOKEN_LIMIT },
|
|
||||||
)}
|
|
||||||
disabled={(scimData?.items.length ?? 0) < SCIM_TOKEN_LIMIT}
|
|
||||||
refProp="rootRef"
|
|
||||||
>
|
|
||||||
<Button
|
|
||||||
onClick={() => setCreateOpen(true)}
|
|
||||||
disabled={(scimData?.items.length ?? 0) >= SCIM_TOKEN_LIMIT}
|
|
||||||
>
|
|
||||||
{t("Create {{credential}}", {
|
|
||||||
credential: t("SCIM token"),
|
|
||||||
})}
|
|
||||||
</Button>
|
|
||||||
</Tooltip>
|
|
||||||
</Group>
|
|
||||||
|
|
||||||
<Card shadow="sm" radius="sm">
|
|
||||||
<ScimTokenTable
|
|
||||||
tokens={scimData?.items}
|
|
||||||
isLoading={scimLoading}
|
|
||||||
onUpdate={setUpdateTarget}
|
|
||||||
onRevoke={setRevokeTarget}
|
|
||||||
/>
|
|
||||||
</Card>
|
|
||||||
|
|
||||||
<Space h="md" />
|
|
||||||
|
|
||||||
{scimData?.items.length > 0 && (
|
|
||||||
<Paginate
|
|
||||||
hasPrevPage={scimData?.meta?.hasPrevPage}
|
|
||||||
hasNextPage={scimData?.meta?.hasNextPage}
|
|
||||||
onNext={() => goNext(scimData?.meta?.nextCursor)}
|
|
||||||
onPrev={goPrev}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<CreateScimTokenModal
|
|
||||||
opened={createOpen}
|
|
||||||
onClose={() => setCreateOpen(false)}
|
|
||||||
onSuccess={setCreatedToken}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<ScimTokenCreatedModal
|
|
||||||
opened={!!createdToken}
|
|
||||||
onClose={() => setCreatedToken(null)}
|
|
||||||
scimToken={createdToken}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<UpdateScimTokenModal
|
|
||||||
opened={!!updateTarget}
|
|
||||||
onClose={() => setUpdateTarget(null)}
|
|
||||||
scimToken={updateTarget}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<RevokeScimTokenModal
|
|
||||||
opened={!!revokeTarget}
|
|
||||||
onClose={() => setRevokeTarget(null)}
|
|
||||||
scimToken={revokeTarget}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -56,7 +56,6 @@ export default function TemplateCard({
|
|||||||
color="gray"
|
color="gray"
|
||||||
className={classes.menuTarget}
|
className={classes.menuTarget}
|
||||||
onClick={(e) => e.stopPropagation()}
|
onClick={(e) => e.stopPropagation()}
|
||||||
aria-label={t("Template menu")}
|
|
||||||
>
|
>
|
||||||
<IconDots size={16} />
|
<IconDots size={16} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ export default function TemplatePreviewModal({
|
|||||||
const title = template?.title || t("Untitled");
|
const title = template?.title || t("Untitled");
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal.Root size={1200} opened={opened} onClose={onClose} aria-label={title}>
|
<Modal.Root size={1200} opened={opened} onClose={onClose}>
|
||||||
<Modal.Overlay />
|
<Modal.Overlay />
|
||||||
<Modal.Content style={{ overflow: "hidden" }}>
|
<Modal.Content style={{ overflow: "hidden" }}>
|
||||||
<Modal.Header>
|
<Modal.Header>
|
||||||
|
|||||||
@@ -144,7 +144,6 @@ function CommentDialog({ editor, pageId, readOnly }: CommentDialogProps) {
|
|||||||
withCloseButton
|
withCloseButton
|
||||||
withBorder
|
withBorder
|
||||||
data-comment-dialog
|
data-comment-dialog
|
||||||
aria-label={t("Add comment")}
|
|
||||||
>
|
>
|
||||||
<Stack gap={2}>
|
<Stack gap={2}>
|
||||||
<Group>
|
<Group>
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ import { useTranslation } from "react-i18next";
|
|||||||
import EmojiCommand from "@/features/editor/extensions/emoji-command";
|
import EmojiCommand from "@/features/editor/extensions/emoji-command";
|
||||||
import mentionRenderItems from "@/features/editor/components/mention/mention-suggestion";
|
import mentionRenderItems from "@/features/editor/components/mention/mention-suggestion";
|
||||||
import MentionView from "@/features/editor/components/mention/mention-view";
|
import MentionView from "@/features/editor/components/mention/mention-view";
|
||||||
import { platformModifierKey } from "@/lib";
|
|
||||||
|
|
||||||
interface CommentEditorProps {
|
interface CommentEditorProps {
|
||||||
defaultContent?: any;
|
defaultContent?: any;
|
||||||
@@ -84,7 +83,7 @@ const CommentEditor = forwardRef(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (platformModifierKey(event) && event.code === "Enter") {
|
if ((event.ctrlKey || event.metaKey) && event.key === "Enter") {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
if (onSave) onSave();
|
if (onSave) onSave();
|
||||||
|
|
||||||
|
|||||||
@@ -173,15 +173,6 @@ function CommentListItem({
|
|||||||
<Box
|
<Box
|
||||||
className={classes.textSelection}
|
className={classes.textSelection}
|
||||||
onClick={() => handleCommentClick(comment)}
|
onClick={() => handleCommentClick(comment)}
|
||||||
onKeyDown={(e) => {
|
|
||||||
if (e.key === "Enter" || e.key === " ") {
|
|
||||||
e.preventDefault();
|
|
||||||
handleCommentClick(comment);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
role="button"
|
|
||||||
tabIndex={0}
|
|
||||||
aria-label={t("Jump to comment selection")}
|
|
||||||
>
|
>
|
||||||
<Text size="sm">{comment?.selection}</Text>
|
<Text size="sm">{comment?.selection}</Text>
|
||||||
</Box>
|
</Box>
|
||||||
|
|||||||
@@ -46,11 +46,7 @@ function CommentMenu({
|
|||||||
return (
|
return (
|
||||||
<Menu shadow="md" width={200}>
|
<Menu shadow="md" width={200}>
|
||||||
<Menu.Target>
|
<Menu.Target>
|
||||||
<ActionIcon
|
<ActionIcon variant="default" style={{ border: "none" }}>
|
||||||
variant="default"
|
|
||||||
style={{ border: "none" }}
|
|
||||||
aria-label={t("Comment menu")}
|
|
||||||
>
|
|
||||||
<IconDots size={20} stroke={2} />
|
<IconDots size={20} stroke={2} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Menu.Target>
|
</Menu.Target>
|
||||||
|
|||||||
@@ -19,9 +19,7 @@ export const uploadAttachmentAction = handleAttachmentUpload({
|
|||||||
},
|
},
|
||||||
validateFn: (file, allowMedia: boolean) => {
|
validateFn: (file, allowMedia: boolean) => {
|
||||||
if (
|
if (
|
||||||
(file.type.includes("image/") ||
|
(file.type.includes("image/") || file.type.includes("video/")) &&
|
||||||
file.type.includes("video/") ||
|
|
||||||
file.type === "application/pdf") &&
|
|
||||||
!allowMedia
|
!allowMedia
|
||||||
) {
|
) {
|
||||||
return false;
|
return false;
|
||||||
|
|||||||
@@ -36,7 +36,6 @@ export default function AudioView(props: NodeViewProps) {
|
|||||||
preload="metadata"
|
preload="metadata"
|
||||||
controls
|
controls
|
||||||
src={safeSrc}
|
src={safeSrc}
|
||||||
aria-label={placeholder?.name || t("Audio")}
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{!safeSrc && previewSrc && (
|
{!safeSrc && previewSrc && (
|
||||||
@@ -46,7 +45,6 @@ export default function AudioView(props: NodeViewProps) {
|
|||||||
preload="metadata"
|
preload="metadata"
|
||||||
controls
|
controls
|
||||||
src={previewSrc}
|
src={previewSrc}
|
||||||
aria-label={placeholder?.name || t("Audio")}
|
|
||||||
/>
|
/>
|
||||||
<Loader size={20} pos="absolute" top={6} right={6} />
|
<Loader size={20} pos="absolute" top={6} right={6} />
|
||||||
</Group>
|
</Group>
|
||||||
@@ -62,7 +60,7 @@ export default function AudioView(props: NodeViewProps) {
|
|||||||
</Group>
|
</Group>
|
||||||
)}
|
)}
|
||||||
{!safeSrc && !previewSrc && !placeholder && (
|
{!safeSrc && !previewSrc && !placeholder && (
|
||||||
<audio className={classes.audio} controls aria-label={t("Audio")} />
|
<audio className={classes.audio} controls />
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</NodeViewWrapper>
|
</NodeViewWrapper>
|
||||||
|
|||||||
@@ -172,9 +172,6 @@ export const ColorSelector: FC<ColorSelectorProps> = ({
|
|||||||
fontWeight: 500,
|
fontWeight: 500,
|
||||||
fontSize: rem(16),
|
fontSize: rem(16),
|
||||||
}}
|
}}
|
||||||
aria-label={t("Text color")}
|
|
||||||
aria-haspopup="dialog"
|
|
||||||
aria-expanded={isOpen}
|
|
||||||
>
|
>
|
||||||
A
|
A
|
||||||
</Button>
|
</Button>
|
||||||
@@ -189,32 +186,20 @@ export const ColorSelector: FC<ColorSelectorProps> = ({
|
|||||||
{t("Text color")}
|
{t("Text color")}
|
||||||
</Text>
|
</Text>
|
||||||
<SimpleGrid cols={5} spacing="xs">
|
<SimpleGrid cols={5} spacing="xs">
|
||||||
{TEXT_COLORS.map(({ name, color }, index) => {
|
{TEXT_COLORS.map(({ name, color }, index) => (
|
||||||
const applyTextColor = () => {
|
|
||||||
if (name === "Default") {
|
|
||||||
editor.commands.unsetColor();
|
|
||||||
} else {
|
|
||||||
editor
|
|
||||||
.chain()
|
|
||||||
.focus()
|
|
||||||
.setColor(color || "")
|
|
||||||
.run();
|
|
||||||
}
|
|
||||||
setIsOpen(false);
|
|
||||||
};
|
|
||||||
return (
|
|
||||||
<Tooltip key={index} label={t(name)} withArrow>
|
<Tooltip key={index} label={t(name)} withArrow>
|
||||||
<Box
|
<Box
|
||||||
role="button"
|
onClick={() => {
|
||||||
tabIndex={0}
|
if (name === "Default") {
|
||||||
aria-label={t(name)}
|
editor.commands.unsetColor();
|
||||||
aria-pressed={!!editorState[`text_${color}`]}
|
} else {
|
||||||
onClick={applyTextColor}
|
editor
|
||||||
onKeyDown={(e) => {
|
.chain()
|
||||||
if (e.key === "Enter" || e.key === " ") {
|
.focus()
|
||||||
e.preventDefault();
|
.setColor(color || "")
|
||||||
applyTextColor();
|
.run();
|
||||||
}
|
}
|
||||||
|
setIsOpen(false);
|
||||||
}}
|
}}
|
||||||
style={{
|
style={{
|
||||||
width: rem(28),
|
width: rem(28),
|
||||||
@@ -236,8 +221,7 @@ export const ColorSelector: FC<ColorSelectorProps> = ({
|
|||||||
A
|
A
|
||||||
</Box>
|
</Box>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
);
|
))}
|
||||||
})}
|
|
||||||
</SimpleGrid>
|
</SimpleGrid>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
@@ -246,35 +230,23 @@ export const ColorSelector: FC<ColorSelectorProps> = ({
|
|||||||
{t("Highlight color")}
|
{t("Highlight color")}
|
||||||
</Text>
|
</Text>
|
||||||
<SimpleGrid cols={5} spacing="xs">
|
<SimpleGrid cols={5} spacing="xs">
|
||||||
{HIGHLIGHT_COLORS.map(({ name, color }, index) => {
|
{HIGHLIGHT_COLORS.map(({ name, color }, index) => (
|
||||||
const applyHighlight = () => {
|
|
||||||
if (name === "Default") {
|
|
||||||
editor.commands.unsetHighlight();
|
|
||||||
} else {
|
|
||||||
editor
|
|
||||||
.chain()
|
|
||||||
.focus()
|
|
||||||
.toggleMark("highlight", {
|
|
||||||
color: color || "",
|
|
||||||
colorName: name.toLowerCase() || "",
|
|
||||||
})
|
|
||||||
.run();
|
|
||||||
}
|
|
||||||
setIsOpen(false);
|
|
||||||
};
|
|
||||||
return (
|
|
||||||
<Tooltip key={index} label={t(name)} withArrow>
|
<Tooltip key={index} label={t(name)} withArrow>
|
||||||
<Box
|
<Box
|
||||||
role="button"
|
onClick={() => {
|
||||||
tabIndex={0}
|
if (name === "Default") {
|
||||||
aria-label={t(name)}
|
editor.commands.unsetHighlight();
|
||||||
aria-pressed={!!editorState[`highlight_${color}`]}
|
} else {
|
||||||
onClick={applyHighlight}
|
editor
|
||||||
onKeyDown={(e) => {
|
.chain()
|
||||||
if (e.key === "Enter" || e.key === " ") {
|
.focus()
|
||||||
e.preventDefault();
|
.toggleMark("highlight", {
|
||||||
applyHighlight();
|
color: color || "",
|
||||||
|
colorName: name.toLowerCase() || "",
|
||||||
|
})
|
||||||
|
.run();
|
||||||
}
|
}
|
||||||
|
setIsOpen(false);
|
||||||
}}
|
}}
|
||||||
style={{
|
style={{
|
||||||
width: rem(28),
|
width: rem(28),
|
||||||
@@ -302,8 +274,7 @@ export const ColorSelector: FC<ColorSelectorProps> = ({
|
|||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
);
|
))}
|
||||||
})}
|
|
||||||
</SimpleGrid>
|
</SimpleGrid>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
|
|||||||
@@ -157,9 +157,6 @@ export const NodeSelector: FC<NodeSelectorProps> = ({
|
|||||||
radius="0"
|
radius="0"
|
||||||
rightSection={<IconChevronDown size={16} />}
|
rightSection={<IconChevronDown size={16} />}
|
||||||
onClick={() => setIsOpen(!isOpen)}
|
onClick={() => setIsOpen(!isOpen)}
|
||||||
aria-label={t("Turn into")}
|
|
||||||
aria-haspopup="menu"
|
|
||||||
aria-expanded={isOpen}
|
|
||||||
>
|
>
|
||||||
{t(activeItem?.name)}
|
{t(activeItem?.name)}
|
||||||
</Button>
|
</Button>
|
||||||
|
|||||||
@@ -92,9 +92,6 @@ export const TextAlignmentSelector: FC<TextAlignmentProps> = ({
|
|||||||
radius="0"
|
radius="0"
|
||||||
rightSection={<IconChevronDown size={16} />}
|
rightSection={<IconChevronDown size={16} />}
|
||||||
onClick={() => setIsOpen(!isOpen)}
|
onClick={() => setIsOpen(!isOpen)}
|
||||||
aria-label={t("Text align")}
|
|
||||||
aria-haspopup="menu"
|
|
||||||
aria-expanded={isOpen}
|
|
||||||
>
|
>
|
||||||
<activeItem.icon style={{ width: rem(16) }} stroke={2} />
|
<activeItem.icon style={{ width: rem(16) }} stroke={2} />
|
||||||
</Button>
|
</Button>
|
||||||
|
|||||||
@@ -137,13 +137,7 @@ export default function DrawioView(props: NodeViewProps) {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<NodeViewWrapper data-drag-handle>
|
<NodeViewWrapper data-drag-handle>
|
||||||
<Modal.Root
|
<Modal.Root opened={opened} onClose={handleClose} fullScreen closeOnEscape={false}>
|
||||||
opened={opened}
|
|
||||||
onClose={handleClose}
|
|
||||||
fullScreen
|
|
||||||
closeOnEscape={false}
|
|
||||||
aria-label={t("Diagram editor")}
|
|
||||||
>
|
|
||||||
<Modal.Overlay />
|
<Modal.Overlay />
|
||||||
<Modal.Content style={{ overflow: "hidden" }}>
|
<Modal.Content style={{ overflow: "hidden" }}>
|
||||||
<Modal.Body pos="relative">
|
<Modal.Body pos="relative">
|
||||||
|
|||||||
@@ -107,17 +107,7 @@ const EmojiList = ({
|
|||||||
}, [selectedIndex]);
|
}, [selectedIndex]);
|
||||||
|
|
||||||
return items.length > 0 || isLoading ? (
|
return items.length > 0 || isLoading ? (
|
||||||
<Paper
|
<Paper id="emoji-command" p="0" shadow="md" withBorder>
|
||||||
id="emoji-command"
|
|
||||||
p="0"
|
|
||||||
shadow="md"
|
|
||||||
withBorder
|
|
||||||
role="listbox"
|
|
||||||
aria-label="Emoji results"
|
|
||||||
aria-activedescendant={
|
|
||||||
items.length > 0 ? `emoji-command-option-${selectedIndex}` : undefined
|
|
||||||
}
|
|
||||||
>
|
|
||||||
{isLoading && <Loader m="xs" color="blue" type="dots" />}
|
{isLoading && <Loader m="xs" color="blue" type="dots" />}
|
||||||
{items.length > 0 && (
|
{items.length > 0 && (
|
||||||
<ScrollArea.Autosize
|
<ScrollArea.Autosize
|
||||||
@@ -130,10 +120,6 @@ const EmojiList = ({
|
|||||||
{items.map((item, index: number) => (
|
{items.map((item, index: number) => (
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
data-item-index={index}
|
data-item-index={index}
|
||||||
id={`emoji-command-option-${index}`}
|
|
||||||
role="option"
|
|
||||||
aria-selected={index === selectedIndex}
|
|
||||||
aria-label={item.id}
|
|
||||||
variant="transparent"
|
variant="transparent"
|
||||||
key={item.id}
|
key={item.id}
|
||||||
className={clsx(classes.menuBtn, {
|
className={clsx(classes.menuBtn, {
|
||||||
|
|||||||
@@ -102,14 +102,6 @@ export const LinkEditorPanel = ({
|
|||||||
leftSection={<IconLink size={16} stroke={1.5} color="var(--mantine-color-dimmed)" />}
|
leftSection={<IconLink size={16} stroke={1.5} color="var(--mantine-color-dimmed)" />}
|
||||||
classNames={{ input: classes.linkInput }}
|
classNames={{ input: classes.linkInput }}
|
||||||
placeholder={t("Paste link or search pages")}
|
placeholder={t("Paste link or search pages")}
|
||||||
aria-label={t("Paste link or search pages")}
|
|
||||||
role="combobox"
|
|
||||||
aria-expanded={showDropdown}
|
|
||||||
aria-controls="link-editor-results"
|
|
||||||
aria-autocomplete="list"
|
|
||||||
aria-activedescendant={
|
|
||||||
showDropdown ? `link-editor-option-${selectedIndex}` : undefined
|
|
||||||
}
|
|
||||||
value={state.url}
|
value={state.url}
|
||||||
onChange={state.onChange}
|
onChange={state.onChange}
|
||||||
onKeyDown={handleKeyDown}
|
onKeyDown={handleKeyDown}
|
||||||
@@ -133,16 +125,10 @@ export const LinkEditorPanel = ({
|
|||||||
scrollbarSize={6}
|
scrollbarSize={6}
|
||||||
mt={state.url.length > 0 ? 8 : 0}
|
mt={state.url.length > 0 ? 8 : 0}
|
||||||
styles={{ content: { minWidth: 0 } }}
|
styles={{ content: { minWidth: 0 } }}
|
||||||
id="link-editor-results"
|
|
||||||
role="listbox"
|
|
||||||
aria-label={t("Link suggestions")}
|
|
||||||
>
|
>
|
||||||
{showUrlItem && (
|
{showUrlItem && (
|
||||||
<UnstyledButton
|
<UnstyledButton
|
||||||
data-item-index={0}
|
data-item-index={0}
|
||||||
id="link-editor-option-0"
|
|
||||||
role="option"
|
|
||||||
aria-selected={selectedIndex === 0}
|
|
||||||
onClick={() => onSetLink(state.url, false)}
|
onClick={() => onSetLink(state.url, false)}
|
||||||
className={clsx(classes.searchItem, {
|
className={clsx(classes.searchItem, {
|
||||||
[classes.selectedSearchItem]: selectedIndex === 0,
|
[classes.selectedSearchItem]: selectedIndex === 0,
|
||||||
@@ -170,9 +156,6 @@ export const LinkEditorPanel = ({
|
|||||||
return (
|
return (
|
||||||
<UnstyledButton
|
<UnstyledButton
|
||||||
data-item-index={itemIndex}
|
data-item-index={itemIndex}
|
||||||
id={`link-editor-option-${itemIndex}`}
|
|
||||||
role="option"
|
|
||||||
aria-selected={itemIndex === selectedIndex}
|
|
||||||
key={page.id || index}
|
key={page.id || index}
|
||||||
onClick={() => selectPage(page)}
|
onClick={() => selectPage(page)}
|
||||||
className={clsx(classes.searchItem, {
|
className={clsx(classes.searchItem, {
|
||||||
|
|||||||
@@ -287,16 +287,7 @@ const MentionList = forwardRef<any, MentionListProps>((props, ref) => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Paper
|
<Paper id="mention" shadow="md" withBorder radius="md" py={6}>
|
||||||
id="mention"
|
|
||||||
shadow="md"
|
|
||||||
withBorder
|
|
||||||
radius="md"
|
|
||||||
py={6}
|
|
||||||
role="listbox"
|
|
||||||
aria-label={t("Mention suggestions")}
|
|
||||||
aria-activedescendant={`mention-option-${selectedIndex}`}
|
|
||||||
>
|
|
||||||
<ScrollArea.Autosize
|
<ScrollArea.Autosize
|
||||||
viewportRef={viewportRef}
|
viewportRef={viewportRef}
|
||||||
mah={350}
|
mah={350}
|
||||||
@@ -310,7 +301,7 @@ const MentionList = forwardRef<any, MentionListProps>((props, ref) => {
|
|||||||
if (item.entityType === "header") {
|
if (item.entityType === "header") {
|
||||||
const isFirst = index === 0;
|
const isFirst = index === 0;
|
||||||
return (
|
return (
|
||||||
<div key={`${item.label}-${index}`} role="presentation">
|
<div key={`${item.label}-${index}`}>
|
||||||
{!isFirst && <Divider my={6} />}
|
{!isFirst && <Divider my={6} />}
|
||||||
<Text
|
<Text
|
||||||
c="dimmed"
|
c="dimmed"
|
||||||
@@ -331,9 +322,6 @@ const MentionList = forwardRef<any, MentionListProps>((props, ref) => {
|
|||||||
<UnstyledButton
|
<UnstyledButton
|
||||||
data-item-index={index}
|
data-item-index={index}
|
||||||
key={index}
|
key={index}
|
||||||
id={`mention-option-${index}`}
|
|
||||||
role="option"
|
|
||||||
aria-selected={index === selectedIndex}
|
|
||||||
onClick={() => selectItem(index)}
|
onClick={() => selectItem(index)}
|
||||||
className={clsx(classes.menuBtn, {
|
className={clsx(classes.menuBtn, {
|
||||||
[classes.selectedItem]: index === selectedIndex,
|
[classes.selectedItem]: index === selectedIndex,
|
||||||
@@ -360,9 +348,6 @@ const MentionList = forwardRef<any, MentionListProps>((props, ref) => {
|
|||||||
<UnstyledButton
|
<UnstyledButton
|
||||||
data-item-index={index}
|
data-item-index={index}
|
||||||
key={index}
|
key={index}
|
||||||
id={`mention-option-${index}`}
|
|
||||||
role="option"
|
|
||||||
aria-selected={index === selectedIndex}
|
|
||||||
onClick={() => selectItem(index)}
|
onClick={() => selectItem(index)}
|
||||||
className={clsx(classes.menuBtn, {
|
className={clsx(classes.menuBtn, {
|
||||||
[classes.selectedItem]: index === selectedIndex,
|
[classes.selectedItem]: index === selectedIndex,
|
||||||
@@ -373,7 +358,7 @@ const MentionList = forwardRef<any, MentionListProps>((props, ref) => {
|
|||||||
<ActionIcon
|
<ActionIcon
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
component="div"
|
component="div"
|
||||||
aria-hidden="true"
|
aria-label={item.label}
|
||||||
color="gray"
|
color="gray"
|
||||||
size="sm"
|
size="sm"
|
||||||
>
|
>
|
||||||
@@ -405,11 +390,6 @@ const MentionList = forwardRef<any, MentionListProps>((props, ref) => {
|
|||||||
{(hasUsers || hasPages) && <Divider my={6} />}
|
{(hasUsers || hasPages) && <Divider my={6} />}
|
||||||
<UnstyledButton
|
<UnstyledButton
|
||||||
data-item-index={renderItems.indexOf(createPageItemData)}
|
data-item-index={renderItems.indexOf(createPageItemData)}
|
||||||
id={`mention-option-${renderItems.indexOf(createPageItemData)}`}
|
|
||||||
role="option"
|
|
||||||
aria-selected={
|
|
||||||
renderItems.indexOf(createPageItemData) === selectedIndex
|
|
||||||
}
|
|
||||||
onClick={() =>
|
onClick={() =>
|
||||||
selectItem(renderItems.indexOf(createPageItemData))
|
selectItem(renderItems.indexOf(createPageItemData))
|
||||||
}
|
}
|
||||||
@@ -425,7 +405,6 @@ const MentionList = forwardRef<any, MentionListProps>((props, ref) => {
|
|||||||
component="div"
|
component="div"
|
||||||
color="gray"
|
color="gray"
|
||||||
size="sm"
|
size="sm"
|
||||||
aria-hidden="true"
|
|
||||||
>
|
>
|
||||||
<IconPlus size={16} stroke={1.5} />
|
<IconPlus size={16} stroke={1.5} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
|
|||||||
@@ -92,20 +92,7 @@ export default function PdfView(props: NodeViewProps) {
|
|||||||
if (hasError) {
|
if (hasError) {
|
||||||
return (
|
return (
|
||||||
<NodeViewWrapper data-drag-handle>
|
<NodeViewWrapper data-drag-handle>
|
||||||
<div
|
<div data-pdf-error className={clsx(classes.pdfError, { "ProseMirror-selectednode": selected })} onClick={handleSelect}>
|
||||||
data-pdf-error
|
|
||||||
className={clsx(classes.pdfError, { "ProseMirror-selectednode": selected })}
|
|
||||||
onClick={handleSelect}
|
|
||||||
onKeyDown={(e) => {
|
|
||||||
if (e.key === "Enter" || e.key === " ") {
|
|
||||||
e.preventDefault();
|
|
||||||
handleSelect();
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
role="button"
|
|
||||||
tabIndex={0}
|
|
||||||
aria-label={t("Failed to load PDF")}
|
|
||||||
>
|
|
||||||
<IconFileTypePdf size={32} stroke={1.5} />
|
<IconFileTypePdf size={32} stroke={1.5} />
|
||||||
<Text size="sm" c="dimmed">
|
<Text size="sm" c="dimmed">
|
||||||
{t("Failed to load PDF")}
|
{t("Failed to load PDF")}
|
||||||
|
|||||||
+3
-25
@@ -187,14 +187,12 @@ function SearchAndReplaceDialog({ editor, editable = true }: PageFindDialogDialo
|
|||||||
position={{ top: 90, right: 50 }}
|
position={{ top: 90, right: 50 }}
|
||||||
withBorder
|
withBorder
|
||||||
transitionProps={{ transition: "slide-down" }}
|
transitionProps={{ transition: "slide-down" }}
|
||||||
aria-label={t("Find and replace")}
|
|
||||||
>
|
>
|
||||||
<Stack gap="xs">
|
<Stack gap="xs">
|
||||||
<Flex align="center" gap="xs">
|
<Flex align="center" gap="xs">
|
||||||
<Input
|
<Input
|
||||||
ref={inputRef}
|
ref={inputRef}
|
||||||
placeholder={t("Find")}
|
placeholder={t("Find")}
|
||||||
aria-label={t("Find")}
|
|
||||||
leftSection={<IconSearch size={16} />}
|
leftSection={<IconSearch size={16} />}
|
||||||
rightSection={
|
rightSection={
|
||||||
<Text size="xs" ta="right">
|
<Text size="xs" ta="right">
|
||||||
@@ -219,12 +217,7 @@ function SearchAndReplaceDialog({ editor, editable = true }: PageFindDialogDialo
|
|||||||
|
|
||||||
<ActionIcon.Group>
|
<ActionIcon.Group>
|
||||||
<Tooltip label={t("Previous match (Shift+Enter)")}>
|
<Tooltip label={t("Previous match (Shift+Enter)")}>
|
||||||
<ActionIcon
|
<ActionIcon variant="subtle" color="gray" onClick={previous}>
|
||||||
variant="subtle"
|
|
||||||
color="gray"
|
|
||||||
onClick={previous}
|
|
||||||
aria-label={t("Previous match (Shift+Enter)")}
|
|
||||||
>
|
|
||||||
<IconArrowNarrowUp
|
<IconArrowNarrowUp
|
||||||
style={{ width: "70%", height: "70%" }}
|
style={{ width: "70%", height: "70%" }}
|
||||||
stroke={1.5}
|
stroke={1.5}
|
||||||
@@ -232,12 +225,7 @@ function SearchAndReplaceDialog({ editor, editable = true }: PageFindDialogDialo
|
|||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
<Tooltip label={t("Next match (Enter)")}>
|
<Tooltip label={t("Next match (Enter)")}>
|
||||||
<ActionIcon
|
<ActionIcon variant="subtle" color="gray" onClick={next}>
|
||||||
variant="subtle"
|
|
||||||
color="gray"
|
|
||||||
onClick={next}
|
|
||||||
aria-label={t("Next match (Enter)")}
|
|
||||||
>
|
|
||||||
<IconArrowNarrowDown
|
<IconArrowNarrowDown
|
||||||
style={{ width: "70%", height: "70%" }}
|
style={{ width: "70%", height: "70%" }}
|
||||||
stroke={1.5}
|
stroke={1.5}
|
||||||
@@ -249,8 +237,6 @@ function SearchAndReplaceDialog({ editor, editable = true }: PageFindDialogDialo
|
|||||||
variant="subtle"
|
variant="subtle"
|
||||||
color={caseSensitive.color}
|
color={caseSensitive.color}
|
||||||
onClick={() => caseSensitiveToggle()}
|
onClick={() => caseSensitiveToggle()}
|
||||||
aria-label={t("Match case (Alt+C)")}
|
|
||||||
aria-pressed={caseSensitive.isCaseSensitive}
|
|
||||||
>
|
>
|
||||||
<IconLetterCase
|
<IconLetterCase
|
||||||
style={{ width: "70%", height: "70%" }}
|
style={{ width: "70%", height: "70%" }}
|
||||||
@@ -264,8 +250,6 @@ function SearchAndReplaceDialog({ editor, editable = true }: PageFindDialogDialo
|
|||||||
variant="subtle"
|
variant="subtle"
|
||||||
color={replaceButton.color}
|
color={replaceButton.color}
|
||||||
onClick={() => replaceButtonToggle()}
|
onClick={() => replaceButtonToggle()}
|
||||||
aria-label={t("Replace")}
|
|
||||||
aria-pressed={replaceButton.isReplaceShow}
|
|
||||||
>
|
>
|
||||||
<IconReplace
|
<IconReplace
|
||||||
style={{ width: "70%", height: "70%" }}
|
style={{ width: "70%", height: "70%" }}
|
||||||
@@ -275,12 +259,7 @@ function SearchAndReplaceDialog({ editor, editable = true }: PageFindDialogDialo
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
)}
|
)}
|
||||||
<Tooltip label={t("Close (Escape)")}>
|
<Tooltip label={t("Close (Escape)")}>
|
||||||
<ActionIcon
|
<ActionIcon variant="subtle" color="gray" onClick={closeDialog}>
|
||||||
variant="subtle"
|
|
||||||
color="gray"
|
|
||||||
onClick={closeDialog}
|
|
||||||
aria-label={t("Close (Escape)")}
|
|
||||||
>
|
|
||||||
<IconX style={{ width: "70%", height: "70%" }} stroke={1.5} />
|
<IconX style={{ width: "70%", height: "70%" }} stroke={1.5} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
@@ -290,7 +269,6 @@ function SearchAndReplaceDialog({ editor, editable = true }: PageFindDialogDialo
|
|||||||
<Flex align="center" gap="xs">
|
<Flex align="center" gap="xs">
|
||||||
<Input
|
<Input
|
||||||
placeholder={t("Replace")}
|
placeholder={t("Replace")}
|
||||||
aria-label={t("Replace")}
|
|
||||||
leftSection={<IconReplace size={16} />}
|
leftSection={<IconReplace size={16} />}
|
||||||
rightSection={<div></div>}
|
rightSection={<div></div>}
|
||||||
rightSectionPointerEvents="all"
|
rightSectionPointerEvents="all"
|
||||||
|
|||||||
@@ -86,15 +86,7 @@ const CommandList = ({
|
|||||||
}, [selectedIndex]);
|
}, [selectedIndex]);
|
||||||
|
|
||||||
return flatItems.length > 0 ? (
|
return flatItems.length > 0 ? (
|
||||||
<Paper
|
<Paper id="slash-command" shadow="md" p="xs" withBorder>
|
||||||
id="slash-command"
|
|
||||||
shadow="md"
|
|
||||||
p="xs"
|
|
||||||
withBorder
|
|
||||||
role="listbox"
|
|
||||||
aria-label={t("Slash commands")}
|
|
||||||
aria-activedescendant={`slash-command-option-${selectedIndex}`}
|
|
||||||
>
|
|
||||||
<ScrollArea
|
<ScrollArea
|
||||||
viewportRef={viewportRef}
|
viewportRef={viewportRef}
|
||||||
h={350}
|
h={350}
|
||||||
@@ -102,30 +94,22 @@ const CommandList = ({
|
|||||||
scrollbarSize={8}
|
scrollbarSize={8}
|
||||||
overscrollBehavior="contain"
|
overscrollBehavior="contain"
|
||||||
>
|
>
|
||||||
{(() => {
|
{Object.entries(items).map(([category, categoryItems]) => (
|
||||||
let flatIndex = -1;
|
<div key={category}>
|
||||||
return Object.entries(items).map(([category, categoryItems]) => (
|
|
||||||
<div key={category} role="group" aria-label={category}>
|
|
||||||
<Text c="dimmed" mb={4} fw={500} tt="capitalize">
|
<Text c="dimmed" mb={4} fw={500} tt="capitalize">
|
||||||
{category}
|
{category}
|
||||||
</Text>
|
</Text>
|
||||||
{categoryItems.map((item: SlashMenuItemType) => {
|
{categoryItems.map((item: SlashMenuItemType, index: number) => (
|
||||||
flatIndex += 1;
|
|
||||||
const itemIndex = flatIndex;
|
|
||||||
return (
|
|
||||||
<UnstyledButton
|
<UnstyledButton
|
||||||
data-item-index={itemIndex}
|
data-item-index={index}
|
||||||
key={itemIndex}
|
key={index}
|
||||||
id={`slash-command-option-${itemIndex}`}
|
onClick={() => selectItem(index)}
|
||||||
role="option"
|
|
||||||
aria-selected={itemIndex === selectedIndex}
|
|
||||||
onClick={() => selectItem(itemIndex)}
|
|
||||||
className={clsx(classes.menuBtn, {
|
className={clsx(classes.menuBtn, {
|
||||||
[classes.selectedItem]: itemIndex === selectedIndex,
|
[classes.selectedItem]: index === selectedIndex,
|
||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<Group>
|
<Group>
|
||||||
<ActionIcon variant="default" component="div" aria-hidden="true">
|
<ActionIcon variant="default" component="div">
|
||||||
<item.icon size={18} />
|
<item.icon size={18} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
|
|
||||||
@@ -140,11 +124,9 @@ const CommandList = ({
|
|||||||
</div>
|
</div>
|
||||||
</Group>
|
</Group>
|
||||||
</UnstyledButton>
|
</UnstyledButton>
|
||||||
);
|
))}
|
||||||
})}
|
|
||||||
</div>
|
</div>
|
||||||
));
|
))}
|
||||||
})()}
|
|
||||||
</ScrollArea>
|
</ScrollArea>
|
||||||
</Paper>
|
</Paper>
|
||||||
) : null;
|
) : null;
|
||||||
|
|||||||
@@ -132,7 +132,7 @@ const CommandGroups: SlashMenuGroupedItemsType = {
|
|||||||
{
|
{
|
||||||
title: "Numbered list",
|
title: "Numbered list",
|
||||||
description: "Create a list with numbering.",
|
description: "Create a list with numbering.",
|
||||||
searchTerms: ["numbered", "ordered", "list", "ol"],
|
searchTerms: ["numbered", "ordered", "list"],
|
||||||
icon: IconListNumbers,
|
icon: IconListNumbers,
|
||||||
command: ({ editor, range }: CommandProps) => {
|
command: ({ editor, range }: CommandProps) => {
|
||||||
editor.chain().focus().deleteRange(range).toggleOrderedList().run();
|
editor.chain().focus().deleteRange(range).toggleOrderedList().run();
|
||||||
@@ -471,14 +471,7 @@ const CommandGroups: SlashMenuGroupedItemsType = {
|
|||||||
{
|
{
|
||||||
title: "Subpages (Child pages)",
|
title: "Subpages (Child pages)",
|
||||||
description: "List all subpages of the current page",
|
description: "List all subpages of the current page",
|
||||||
searchTerms: [
|
searchTerms: ["subpages", "child", "children", "nested", "hierarchy"],
|
||||||
"subpages",
|
|
||||||
"child",
|
|
||||||
"children",
|
|
||||||
"nested",
|
|
||||||
"hierarchy",
|
|
||||||
"toc",
|
|
||||||
],
|
|
||||||
icon: IconSitemap,
|
icon: IconSitemap,
|
||||||
command: ({ editor, range }: CommandProps) => {
|
command: ({ editor, range }: CommandProps) => {
|
||||||
editor.chain().focus().deleteRange(range).insertSubpages().run();
|
editor.chain().focus().deleteRange(range).insertSubpages().run();
|
||||||
|
|||||||
@@ -92,17 +92,8 @@ export default function StatusView(props: NodeViewProps) {
|
|||||||
colorClassMap[color],
|
colorClassMap[color],
|
||||||
)}
|
)}
|
||||||
onClick={() => isEditable && setOpened(true)}
|
onClick={() => isEditable && setOpened(true)}
|
||||||
onKeyDown={(e) => {
|
|
||||||
if (isEditable && (e.key === "Enter" || e.key === " ")) {
|
|
||||||
e.preventDefault();
|
|
||||||
setOpened(true);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
role="button"
|
role="button"
|
||||||
tabIndex={0}
|
tabIndex={0}
|
||||||
aria-label={text || "SET STATUS"}
|
|
||||||
aria-haspopup="dialog"
|
|
||||||
aria-expanded={opened}
|
|
||||||
>
|
>
|
||||||
{text || "SET STATUS"}
|
{text || "SET STATUS"}
|
||||||
</span>
|
</span>
|
||||||
@@ -136,16 +127,6 @@ export default function StatusView(props: NodeViewProps) {
|
|||||||
)}
|
)}
|
||||||
style={{ backgroundColor: bg }}
|
style={{ backgroundColor: bg }}
|
||||||
onClick={() => handleColorChange(name)}
|
onClick={() => handleColorChange(name)}
|
||||||
onKeyDown={(e) => {
|
|
||||||
if (e.key === "Enter" || e.key === " ") {
|
|
||||||
e.preventDefault();
|
|
||||||
handleColorChange(name);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
role="button"
|
|
||||||
tabIndex={0}
|
|
||||||
aria-label={name}
|
|
||||||
aria-pressed={color === name}
|
|
||||||
>
|
>
|
||||||
{color === name && <IconCheck size={14} />}
|
{color === name && <IconCheck size={14} />}
|
||||||
</Box>
|
</Box>
|
||||||
|
|||||||
@@ -47,7 +47,6 @@ export default function VideoView(props: NodeViewProps) {
|
|||||||
preload="metadata"
|
preload="metadata"
|
||||||
controls
|
controls
|
||||||
src={getFileUrl(src)}
|
src={getFileUrl(src)}
|
||||||
aria-label={placeholder?.name || t("Video")}
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{!src && previewSrc && (
|
{!src && previewSrc && (
|
||||||
@@ -57,7 +56,6 @@ export default function VideoView(props: NodeViewProps) {
|
|||||||
preload="metadata"
|
preload="metadata"
|
||||||
controls
|
controls
|
||||||
src={previewSrc}
|
src={previewSrc}
|
||||||
aria-label={placeholder?.name || t("Video")}
|
|
||||||
/>
|
/>
|
||||||
<Loader size={20} pos="absolute" top={6} right={6} />
|
<Loader size={20} pos="absolute" top={6} right={6} />
|
||||||
</Group>
|
</Group>
|
||||||
@@ -73,7 +71,7 @@ export default function VideoView(props: NodeViewProps) {
|
|||||||
</Group>
|
</Group>
|
||||||
)}
|
)}
|
||||||
{!src && !previewSrc && !placeholder && (
|
{!src && !previewSrc && !placeholder && (
|
||||||
<video className={classes.video} controls aria-label={t("Video")} />
|
<video className={classes.video} controls />
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</NodeViewWrapper>
|
</NodeViewWrapper>
|
||||||
|
|||||||
@@ -80,12 +80,10 @@ export const MarkdownClipboard = Extension.create({
|
|||||||
const { from, to } = view.state.selection;
|
const { from, to } = view.state.selection;
|
||||||
|
|
||||||
const parsed = markdownToHtml(text.replace(/\n+$/, ""));
|
const parsed = markdownToHtml(text.replace(/\n+$/, ""));
|
||||||
const body = elementFromString(parsed);
|
|
||||||
normalizeTableColumnWidths(body);
|
|
||||||
|
|
||||||
const contentNodes = DOMParser.fromSchema(
|
const contentNodes = DOMParser.fromSchema(
|
||||||
this.editor.schema,
|
this.editor.schema,
|
||||||
).parseSlice(body, {
|
).parseSlice(elementFromString(parsed), {
|
||||||
preserveWhitespace: true,
|
preserveWhitespace: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -139,92 +137,3 @@ function elementFromString(value) {
|
|||||||
|
|
||||||
return new window.DOMParser().parseFromString(wrappedValue, "text/html").body;
|
return new window.DOMParser().parseFromString(wrappedValue, "text/html").body;
|
||||||
}
|
}
|
||||||
|
|
||||||
const DEFAULT_PASTE_COL_WIDTH_PX = 150;
|
|
||||||
|
|
||||||
function parsePixelWidth(el: Element): number | null {
|
|
||||||
const attr = el.getAttribute("width");
|
|
||||||
if (attr) {
|
|
||||||
const n = parseInt(attr, 10);
|
|
||||||
if (Number.isFinite(n) && n > 0) return n;
|
|
||||||
}
|
|
||||||
const style = el.getAttribute("style") || "";
|
|
||||||
const m = style.match(/(?:^|;)\s*width\s*:\s*([\d.]+)\s*px/i);
|
|
||||||
if (m) {
|
|
||||||
const n = parseInt(m[1], 10);
|
|
||||||
if (Number.isFinite(n) && n > 0) return n;
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
function getFirstRow(table: Element): Element | null {
|
|
||||||
const tbodyRow = table.querySelector(":scope > tbody > tr");
|
|
||||||
if (tbodyRow) return tbodyRow;
|
|
||||||
const theadRow = table.querySelector(":scope > thead > tr");
|
|
||||||
if (theadRow) return theadRow;
|
|
||||||
return table.querySelector(":scope > tr");
|
|
||||||
}
|
|
||||||
|
|
||||||
function deriveColumnWidths(table: Element): (number | null)[] | null {
|
|
||||||
const cols = table.querySelectorAll(":scope > colgroup > col");
|
|
||||||
if (cols.length > 0) {
|
|
||||||
const widths: (number | null)[] = [];
|
|
||||||
cols.forEach((col) => widths.push(parsePixelWidth(col)));
|
|
||||||
if (widths.some((w) => w !== null)) return widths;
|
|
||||||
}
|
|
||||||
|
|
||||||
const firstRow = getFirstRow(table);
|
|
||||||
if (!firstRow) return null;
|
|
||||||
|
|
||||||
const widths: (number | null)[] = [];
|
|
||||||
Array.from(firstRow.children)
|
|
||||||
.filter((c) => c.tagName === "TD" || c.tagName === "TH")
|
|
||||||
.forEach((cell) => {
|
|
||||||
const colspan = parseInt(cell.getAttribute("colspan") || "1", 10) || 1;
|
|
||||||
const w = parsePixelWidth(cell);
|
|
||||||
for (let i = 0; i < colspan; i++) {
|
|
||||||
widths.push(w !== null ? Math.round(w / colspan) : null);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
if (widths.length === 0 || widths.every((w) => w === null)) return null;
|
|
||||||
return widths;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Mirror of server normalizeTableColumnWidths (see import/utils/table-utils.ts):
|
|
||||||
// markdown source has no widths, so without this every pasted table renders
|
|
||||||
// at table-layout:fixed/100% and squashes columns to fit the editor instead of
|
|
||||||
// letting .tableWrapper's overflow-x: auto scroll.
|
|
||||||
export function normalizeTableColumnWidths(root: Element): void {
|
|
||||||
root.querySelectorAll("table").forEach((table) => {
|
|
||||||
const firstRow = getFirstRow(table);
|
|
||||||
if (!firstRow) return;
|
|
||||||
|
|
||||||
let colWidths = deriveColumnWidths(table);
|
|
||||||
if (!colWidths) {
|
|
||||||
let count = 0;
|
|
||||||
Array.from(firstRow.children)
|
|
||||||
.filter((c) => c.tagName === "TD" || c.tagName === "TH")
|
|
||||||
.forEach((cell) => {
|
|
||||||
count += parseInt(cell.getAttribute("colspan") || "1", 10) || 1;
|
|
||||||
});
|
|
||||||
if (count === 0) return;
|
|
||||||
colWidths = new Array(count).fill(DEFAULT_PASTE_COL_WIDTH_PX);
|
|
||||||
}
|
|
||||||
|
|
||||||
let col = 0;
|
|
||||||
Array.from(firstRow.children)
|
|
||||||
.filter((c) => c.tagName === "TD" || c.tagName === "TH")
|
|
||||||
.forEach((cell) => {
|
|
||||||
if (cell.getAttribute("colwidth")) {
|
|
||||||
col += parseInt(cell.getAttribute("colspan") || "1", 10) || 1;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const colspan = parseInt(cell.getAttribute("colspan") || "1", 10) || 1;
|
|
||||||
const slice = colWidths!.slice(col, col + colspan);
|
|
||||||
col += colspan;
|
|
||||||
if (slice.length === 0 || slice.every((w) => w === null)) return;
|
|
||||||
const values = slice.map((w) => (w == null ? 100 : w));
|
|
||||||
cell.setAttribute("colwidth", values.join(","));
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -102,7 +102,6 @@ function PageByline({
|
|||||||
<Group
|
<Group
|
||||||
gap="sm"
|
gap="sm"
|
||||||
mb="md"
|
mb="md"
|
||||||
className="print-hide"
|
|
||||||
style={{ marginTop: "-0.5em", paddingLeft: "3rem" }}
|
style={{ marginTop: "-0.5em", paddingLeft: "3rem" }}
|
||||||
>
|
>
|
||||||
{creator && (
|
{creator && (
|
||||||
|
|||||||
@@ -62,7 +62,7 @@ import { useIdle } from "@/hooks/use-idle.ts";
|
|||||||
import { queryClient } from "@/main.tsx";
|
import { queryClient } from "@/main.tsx";
|
||||||
import { IPage } from "@/features/page/types/page.types.ts";
|
import { IPage } from "@/features/page/types/page.types.ts";
|
||||||
import { useParams } from "react-router-dom";
|
import { useParams } from "react-router-dom";
|
||||||
import { extractPageSlugId, platformModifierKey } from "@/lib";
|
import { extractPageSlugId } from "@/lib";
|
||||||
import { FIVE_MINUTES } from "@/lib/constants.ts";
|
import { FIVE_MINUTES } from "@/lib/constants.ts";
|
||||||
import { PageEditMode } from "@/features/user/types/user.types.ts";
|
import { PageEditMode } from "@/features/user/types/user.types.ts";
|
||||||
import { jwtDecode } from "jwt-decode";
|
import { jwtDecode } from "jwt-decode";
|
||||||
@@ -232,19 +232,11 @@ export default function PageEditor({
|
|||||||
scrollMargin: 80,
|
scrollMargin: 80,
|
||||||
handleDOMEvents: {
|
handleDOMEvents: {
|
||||||
keydown: (_view, event) => {
|
keydown: (_view, event) => {
|
||||||
if (platformModifierKey(event) && event.code === "KeyS") {
|
if ((event.ctrlKey || event.metaKey) && event.code === "KeyS") {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (event.key === "Tab") {
|
if ((event.ctrlKey || event.metaKey) && event.code === "KeyK") {
|
||||||
const editor = editorRef.current;
|
|
||||||
if (!editor) return false;
|
|
||||||
event.preventDefault();
|
|
||||||
return editor.view.someProp("handleKeyDown", (f) =>
|
|
||||||
f(editor.view, event)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if (platformModifierKey(event) && event.code === "KeyK") {
|
|
||||||
searchSpotlight.open();
|
searchSpotlight.open();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -27,7 +27,6 @@ import localEmitter from "@/lib/local-emitter.ts";
|
|||||||
import { currentUserAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { currentUserAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
import { PageEditMode } from "@/features/user/types/user.types.ts";
|
import { PageEditMode } from "@/features/user/types/user.types.ts";
|
||||||
import { searchSpotlight } from "@/features/search/constants.ts";
|
import { searchSpotlight } from "@/features/search/constants.ts";
|
||||||
import { platformModifierKey } from "@/lib";
|
|
||||||
|
|
||||||
export interface TitleEditorProps {
|
export interface TitleEditorProps {
|
||||||
pageId: string;
|
pageId: string;
|
||||||
@@ -91,11 +90,11 @@ export function TitleEditor({
|
|||||||
editorProps: {
|
editorProps: {
|
||||||
handleDOMEvents: {
|
handleDOMEvents: {
|
||||||
keydown: (_view, event) => {
|
keydown: (_view, event) => {
|
||||||
if (platformModifierKey(event) && event.code === "KeyS") {
|
if ((event.ctrlKey || event.metaKey) && event.code === "KeyS") {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (platformModifierKey(event) && event.code === "KeyK") {
|
if ((event.ctrlKey || event.metaKey) && event.code === "KeyK") {
|
||||||
searchSpotlight.open();
|
searchSpotlight.open();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -53,17 +53,15 @@ export default function StarButton(props: StarButtonProps) {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const label = isFavorited
|
|
||||||
? t("Remove from favorites")
|
|
||||||
: t("Add to favorites");
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tooltip label={label} openDelay={250} withArrow>
|
<Tooltip
|
||||||
|
label={isFavorited ? t("Remove from favorites") : t("Add to favorites")}
|
||||||
|
openDelay={250}
|
||||||
|
withArrow
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
color={isFavorited ? "yellow" : "gray"}
|
color={isFavorited ? "yellow" : "gray"}
|
||||||
aria-label={label}
|
|
||||||
aria-pressed={isFavorited}
|
|
||||||
onClick={handleToggle}
|
onClick={handleToggle}
|
||||||
loading={isPending}
|
loading={isPending}
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -14,11 +14,11 @@ import {
|
|||||||
} from "../services/favorite-service";
|
} from "../services/favorite-service";
|
||||||
import { FavoriteType } from "../types/favorite.types";
|
import { FavoriteType } from "../types/favorite.types";
|
||||||
|
|
||||||
export function useFavoritesQuery(type?: FavoriteType, spaceId?: string) {
|
export function useFavoritesQuery(type?: FavoriteType) {
|
||||||
return useInfiniteQuery({
|
return useInfiniteQuery({
|
||||||
queryKey: ["favorites", type, spaceId],
|
queryKey: ["favorites", type],
|
||||||
queryFn: ({ pageParam }) =>
|
queryFn: ({ pageParam }) =>
|
||||||
getFavorites({ type, spaceId, cursor: pageParam, limit: 15 }),
|
getFavorites({ type, cursor: pageParam, limit: 15 }),
|
||||||
initialPageParam: undefined as string | undefined,
|
initialPageParam: undefined as string | undefined,
|
||||||
getNextPageParam: (lastPage) =>
|
getNextPageParam: (lastPage) =>
|
||||||
lastPage.meta.hasNextPage ? lastPage.meta.nextCursor : undefined,
|
lastPage.meta.hasNextPage ? lastPage.meta.nextCursor : undefined,
|
||||||
@@ -26,10 +26,10 @@ export function useFavoritesQuery(type?: FavoriteType, spaceId?: string) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export function useFavoriteIds(type: FavoriteType, spaceId?: string): Set<string> {
|
export function useFavoriteIds(type: FavoriteType): Set<string> {
|
||||||
const { data } = useQuery({
|
const { data } = useQuery({
|
||||||
queryKey: ["favorite-ids", type, spaceId],
|
queryKey: ["favorite-ids", type],
|
||||||
queryFn: () => getFavoriteIds(type, spaceId),
|
queryFn: () => getFavoriteIds(type),
|
||||||
refetchOnMount: true,
|
refetchOnMount: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -52,9 +52,9 @@ export function useAddFavoriteMutation() {
|
|||||||
onSuccess: (_result, variables) => {
|
onSuccess: (_result, variables) => {
|
||||||
const entityId = getEntityId(variables);
|
const entityId = getEntityId(variables);
|
||||||
if (entityId) {
|
if (entityId) {
|
||||||
queryClient.setQueriesData<{ items: string[]; meta: any }>(
|
queryClient.setQueryData(
|
||||||
{ queryKey: ["favorite-ids", variables.type] },
|
["favorite-ids", variables.type],
|
||||||
(old) => {
|
(old: { items: string[]; meta: any } | undefined) => {
|
||||||
if (!old) return old;
|
if (!old) return old;
|
||||||
if (old.items.includes(entityId)) return old;
|
if (old.items.includes(entityId)) return old;
|
||||||
return { ...old, items: [...old.items, entityId] };
|
return { ...old, items: [...old.items, entityId] };
|
||||||
@@ -76,9 +76,9 @@ export function useRemoveFavoriteMutation() {
|
|||||||
onSuccess: (_result, variables) => {
|
onSuccess: (_result, variables) => {
|
||||||
const entityId = getEntityId(variables);
|
const entityId = getEntityId(variables);
|
||||||
if (entityId) {
|
if (entityId) {
|
||||||
queryClient.setQueriesData<{ items: string[]; meta: any }>(
|
queryClient.setQueryData(
|
||||||
{ queryKey: ["favorite-ids", variables.type] },
|
["favorite-ids", variables.type],
|
||||||
(old) => {
|
(old: { items: string[]; meta: any } | undefined) => {
|
||||||
if (!old) return old;
|
if (!old) return old;
|
||||||
return { ...old, items: old.items.filter((id) => id !== entityId) };
|
return { ...old, items: old.items.filter((id) => id !== entityId) };
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -21,14 +21,13 @@ export async function removeFavorite(
|
|||||||
await api.post("/favorites/remove", params);
|
await api.post("/favorites/remove", params);
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function getFavoriteIds(type: FavoriteType, spaceId?: string): Promise<IPagination<string>> {
|
export async function getFavoriteIds(type: FavoriteType): Promise<IPagination<string>> {
|
||||||
const req = await api.post<IPagination<string>>("/favorites/ids", { type, spaceId });
|
const req = await api.post<IPagination<string>>("/favorites/ids", { type });
|
||||||
return req.data;
|
return req.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function getFavorites(params?: {
|
export async function getFavorites(params?: {
|
||||||
type?: FavoriteType;
|
type?: FavoriteType;
|
||||||
spaceId?: string;
|
|
||||||
limit?: number;
|
limit?: number;
|
||||||
cursor?: string;
|
cursor?: string;
|
||||||
}): Promise<IPagination<IFavorite>> {
|
}): Promise<IPagination<IFavorite>> {
|
||||||
|
|||||||
@@ -53,7 +53,7 @@ export default function GroupActionMenu() {
|
|||||||
arrowPosition="center"
|
arrowPosition="center"
|
||||||
>
|
>
|
||||||
<Menu.Target>
|
<Menu.Target>
|
||||||
<ActionIcon variant="light" aria-label={t("Group menu")}>
|
<ActionIcon variant="light">
|
||||||
<IconDots size={20} stroke={2} />
|
<IconDots size={20} stroke={2} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Menu.Target>
|
</Menu.Target>
|
||||||
|
|||||||
@@ -54,7 +54,7 @@ export default function GroupMembersList() {
|
|||||||
<Table.Tr>
|
<Table.Tr>
|
||||||
<Table.Th>{t("User")}</Table.Th>
|
<Table.Th>{t("User")}</Table.Th>
|
||||||
<Table.Th>{t("Status")}</Table.Th>
|
<Table.Th>{t("Status")}</Table.Th>
|
||||||
<Table.Th aria-label={t("Action")} />
|
<Table.Th></Table.Th>
|
||||||
</Table.Tr>
|
</Table.Tr>
|
||||||
</Table.Thead>
|
</Table.Thead>
|
||||||
|
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import {
|
|||||||
UnstyledButton,
|
UnstyledButton,
|
||||||
Badge,
|
Badge,
|
||||||
Table,
|
Table,
|
||||||
ThemeIcon,
|
ActionIcon,
|
||||||
Button,
|
Button,
|
||||||
} from "@mantine/core";
|
} from "@mantine/core";
|
||||||
import { Link } from "react-router-dom";
|
import { Link } from "react-router-dom";
|
||||||
@@ -61,13 +61,13 @@ export default function CreatedByMe({ spaceId }: Props) {
|
|||||||
>
|
>
|
||||||
<Group wrap="nowrap">
|
<Group wrap="nowrap">
|
||||||
{page.icon || (
|
{page.icon || (
|
||||||
<ThemeIcon
|
<ActionIcon
|
||||||
variant="transparent"
|
variant="transparent"
|
||||||
color="gray"
|
color="gray"
|
||||||
size={18}
|
size={18}
|
||||||
>
|
>
|
||||||
<IconFileDescription size={18} />
|
<IconFileDescription size={18} />
|
||||||
</ThemeIcon>
|
</ActionIcon>
|
||||||
)}
|
)}
|
||||||
<Text fw={500} size="md" lineClamp={1}>
|
<Text fw={500} size="md" lineClamp={1}>
|
||||||
{page.title || t("Untitled")}
|
{page.title || t("Untitled")}
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import {
|
|||||||
UnstyledButton,
|
UnstyledButton,
|
||||||
Badge,
|
Badge,
|
||||||
Table,
|
Table,
|
||||||
ThemeIcon,
|
ActionIcon,
|
||||||
Button,
|
Button,
|
||||||
} from "@mantine/core";
|
} from "@mantine/core";
|
||||||
import { Link } from "react-router-dom";
|
import { Link } from "react-router-dom";
|
||||||
@@ -18,11 +18,7 @@ import { getSpaceUrl } from "@/lib/config";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { getInitialsColor } from "@/lib/get-initials-color";
|
import { getInitialsColor } from "@/lib/get-initials-color";
|
||||||
|
|
||||||
interface Props {
|
export default function FavoritesPages() {
|
||||||
spaceId?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function FavoritesPages({ spaceId }: Props) {
|
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const {
|
const {
|
||||||
data,
|
data,
|
||||||
@@ -31,7 +27,7 @@ export default function FavoritesPages({ spaceId }: Props) {
|
|||||||
hasNextPage,
|
hasNextPage,
|
||||||
fetchNextPage,
|
fetchNextPage,
|
||||||
isFetchingNextPage,
|
isFetchingNextPage,
|
||||||
} = useFavoritesQuery("page", spaceId);
|
} = useFavoritesQuery("page");
|
||||||
|
|
||||||
const favorites = data?.pages.flatMap((p) => p.items) ?? [];
|
const favorites = data?.pages.flatMap((p) => p.items) ?? [];
|
||||||
|
|
||||||
@@ -62,13 +58,13 @@ export default function FavoritesPages({ spaceId }: Props) {
|
|||||||
>
|
>
|
||||||
<Group wrap="nowrap">
|
<Group wrap="nowrap">
|
||||||
{fav.page.icon || (
|
{fav.page.icon || (
|
||||||
<ThemeIcon
|
<ActionIcon
|
||||||
variant="transparent"
|
variant="transparent"
|
||||||
color="gray"
|
color="gray"
|
||||||
size={18}
|
size={18}
|
||||||
>
|
>
|
||||||
<IconFileDescription size={18} />
|
<IconFileDescription size={18} />
|
||||||
</ThemeIcon>
|
</ActionIcon>
|
||||||
)}
|
)}
|
||||||
<Text fw={500} size="md" lineClamp={1}>
|
<Text fw={500} size="md" lineClamp={1}>
|
||||||
{fav.page.title || t("Untitled")}
|
{fav.page.title || t("Untitled")}
|
||||||
@@ -76,21 +72,19 @@ export default function FavoritesPages({ spaceId }: Props) {
|
|||||||
</Group>
|
</Group>
|
||||||
</UnstyledButton>
|
</UnstyledButton>
|
||||||
</Table.Td>
|
</Table.Td>
|
||||||
{!spaceId && (
|
<Table.Td>
|
||||||
<Table.Td>
|
{fav.space && (
|
||||||
{fav.space && (
|
<Badge
|
||||||
<Badge
|
color={getInitialsColor(fav.space.name)}
|
||||||
color={getInitialsColor(fav.space.name)}
|
variant="light"
|
||||||
variant="light"
|
component={Link}
|
||||||
component={Link}
|
to={getSpaceUrl(fav.space.slug)}
|
||||||
to={getSpaceUrl(fav.space.slug)}
|
style={{ cursor: "pointer" }}
|
||||||
style={{ cursor: "pointer" }}
|
>
|
||||||
>
|
{fav.space.name}
|
||||||
{fav.space.name}
|
</Badge>
|
||||||
</Badge>
|
)}
|
||||||
)}
|
</Table.Td>
|
||||||
</Table.Td>
|
|
||||||
)}
|
|
||||||
<Table.Td>
|
<Table.Td>
|
||||||
<Text
|
<Text
|
||||||
c="dimmed"
|
c="dimmed"
|
||||||
|
|||||||
@@ -16,7 +16,7 @@
|
|||||||
|
|
||||||
.subtitle {
|
.subtitle {
|
||||||
font-size: var(--mantine-font-size-sm);
|
font-size: var(--mantine-font-size-sm);
|
||||||
color: var(--mantine-color-dimmed);
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
text-align: center;
|
text-align: center;
|
||||||
margin-top: 6px;
|
margin-top: 6px;
|
||||||
margin-bottom: var(--mantine-spacing-lg);
|
margin-bottom: var(--mantine-spacing-lg);
|
||||||
|
|||||||
@@ -58,9 +58,6 @@ export function NotificationPopover() {
|
|||||||
variant="subtle"
|
variant="subtle"
|
||||||
color="dark"
|
color="dark"
|
||||||
size="sm"
|
size="sm"
|
||||||
aria-label={t("Notifications")}
|
|
||||||
aria-haspopup="dialog"
|
|
||||||
aria-expanded={opened}
|
|
||||||
onClick={() => setOpened((o) => !o)}
|
onClick={() => setOpened((o) => !o)}
|
||||||
>
|
>
|
||||||
<Indicator
|
<Indicator
|
||||||
|
|||||||
@@ -22,7 +22,6 @@ export default function HistoryModal({ pageId, pageTitle }: Props) {
|
|||||||
opened={isModalOpen}
|
opened={isModalOpen}
|
||||||
onClose={() => setModalOpen(false)}
|
onClose={() => setModalOpen(false)}
|
||||||
fullScreen
|
fullScreen
|
||||||
aria-label={t("Page history")}
|
|
||||||
>
|
>
|
||||||
<Modal.Overlay />
|
<Modal.Overlay />
|
||||||
<Modal.Content style={{ overflow: "hidden" }}>
|
<Modal.Content style={{ overflow: "hidden" }}>
|
||||||
@@ -50,7 +49,6 @@ export default function HistoryModal({ pageId, pageTitle }: Props) {
|
|||||||
size={1400}
|
size={1400}
|
||||||
opened={isModalOpen}
|
opened={isModalOpen}
|
||||||
onClose={() => setModalOpen(false)}
|
onClose={() => setModalOpen(false)}
|
||||||
aria-label={t("Page history")}
|
|
||||||
>
|
>
|
||||||
<Modal.Overlay />
|
<Modal.Overlay />
|
||||||
<Modal.Content style={{ overflow: "hidden" }}>
|
<Modal.Content style={{ overflow: "hidden" }}>
|
||||||
|
|||||||
@@ -19,7 +19,6 @@ import { buildPageUrl } from "@/features/page/page.utils.ts";
|
|||||||
import { usePageQuery } from "@/features/page/queries/page-query.ts";
|
import { usePageQuery } from "@/features/page/queries/page-query.ts";
|
||||||
import { extractPageSlugId } from "@/lib";
|
import { extractPageSlugId } from "@/lib";
|
||||||
import { useMediaQuery } from "@mantine/hooks";
|
import { useMediaQuery } from "@mantine/hooks";
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
|
|
||||||
function getTitle(name: string, icon: string) {
|
function getTitle(name: string, icon: string) {
|
||||||
if (icon) {
|
if (icon) {
|
||||||
@@ -29,7 +28,6 @@ function getTitle(name: string, icon: string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function Breadcrumb() {
|
export default function Breadcrumb() {
|
||||||
const { t } = useTranslation();
|
|
||||||
const treeData = useAtomValue(treeDataAtom);
|
const treeData = useAtomValue(treeDataAtom);
|
||||||
const [breadcrumbNodes, setBreadcrumbNodes] = useState<
|
const [breadcrumbNodes, setBreadcrumbNodes] = useState<
|
||||||
SpaceTreeNode[] | null
|
SpaceTreeNode[] | null
|
||||||
@@ -82,7 +80,7 @@ export default function Breadcrumb() {
|
|||||||
));
|
));
|
||||||
|
|
||||||
const renderAnchor = useCallback(
|
const renderAnchor = useCallback(
|
||||||
(node: SpaceTreeNode, isCurrent = false) => (
|
(node: SpaceTreeNode) => (
|
||||||
<Tooltip label={node.name} key={node.id}>
|
<Tooltip label={node.name} key={node.id}>
|
||||||
<Anchor
|
<Anchor
|
||||||
component={Link}
|
component={Link}
|
||||||
@@ -91,7 +89,6 @@ export default function Breadcrumb() {
|
|||||||
fz="sm"
|
fz="sm"
|
||||||
key={node.id}
|
key={node.id}
|
||||||
className={classes.truncatedText}
|
className={classes.truncatedText}
|
||||||
aria-current={isCurrent ? "page" : undefined}
|
|
||||||
>
|
>
|
||||||
{getTitle(node.name, node.icon)}
|
{getTitle(node.name, node.icon)}
|
||||||
</Anchor>
|
</Anchor>
|
||||||
@@ -118,11 +115,7 @@ export default function Breadcrumb() {
|
|||||||
key="hidden-nodes"
|
key="hidden-nodes"
|
||||||
>
|
>
|
||||||
<Popover.Target>
|
<Popover.Target>
|
||||||
<ActionIcon
|
<ActionIcon color="gray" variant="transparent">
|
||||||
color="gray"
|
|
||||||
variant="transparent"
|
|
||||||
aria-label={t("Show hidden breadcrumbs")}
|
|
||||||
>
|
|
||||||
<IconDots size={20} stroke={2} />
|
<IconDots size={20} stroke={2} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Popover.Target>
|
</Popover.Target>
|
||||||
@@ -131,13 +124,11 @@ export default function Breadcrumb() {
|
|||||||
</Popover.Dropdown>
|
</Popover.Dropdown>
|
||||||
</Popover>,
|
</Popover>,
|
||||||
//renderAnchor(secondLastNode),
|
//renderAnchor(secondLastNode),
|
||||||
renderAnchor(lastNode, true),
|
renderAnchor(lastNode),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
return breadcrumbNodes.map((node, i) =>
|
return breadcrumbNodes.map(renderAnchor);
|
||||||
renderAnchor(node, i === breadcrumbNodes.length - 1),
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const getMobileBreadcrumbItems = () => {
|
const getMobileBreadcrumbItems = () => {
|
||||||
@@ -153,12 +144,8 @@ export default function Breadcrumb() {
|
|||||||
key="mobile-hidden-nodes"
|
key="mobile-hidden-nodes"
|
||||||
>
|
>
|
||||||
<Popover.Target>
|
<Popover.Target>
|
||||||
<Tooltip label={t("Breadcrumbs")}>
|
<Tooltip label="Breadcrumbs">
|
||||||
<ActionIcon
|
<ActionIcon color="gray" variant="transparent">
|
||||||
color="gray"
|
|
||||||
variant="transparent"
|
|
||||||
aria-label={t("Breadcrumbs")}
|
|
||||||
>
|
|
||||||
<IconCornerDownRightDouble size={20} stroke={2} />
|
<IconCornerDownRightDouble size={20} stroke={2} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
@@ -170,18 +157,16 @@ export default function Breadcrumb() {
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
return breadcrumbNodes.map((node, i) =>
|
return breadcrumbNodes.map(renderAnchor);
|
||||||
renderAnchor(node, i === breadcrumbNodes.length - 1),
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<nav aria-label={t("Breadcrumb")} className={classes.breadcrumbDiv}>
|
<div className={classes.breadcrumbDiv}>
|
||||||
{breadcrumbNodes && (
|
{breadcrumbNodes && (
|
||||||
<Breadcrumbs className={classes.breadcrumbs}>
|
<Breadcrumbs className={classes.breadcrumbs}>
|
||||||
{isMobile ? getMobileBreadcrumbItems() : getBreadcrumbItems()}
|
{isMobile ? getMobileBreadcrumbItems() : getBreadcrumbItems()}
|
||||||
</Breadcrumbs>
|
</Breadcrumbs>
|
||||||
)}
|
)}
|
||||||
</nav>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { ActionIcon, Group, Menu, Text, ThemeIcon, Tooltip } from "@mantine/core";
|
import { ActionIcon, Group, Menu, Text, Tooltip } from "@mantine/core";
|
||||||
import {
|
import {
|
||||||
IconArrowRight,
|
IconArrowRight,
|
||||||
IconArrowsHorizontal,
|
IconArrowsHorizontal,
|
||||||
@@ -99,7 +99,6 @@ export default function PageHeaderMenu({ readOnly }: PageHeaderMenuProps) {
|
|||||||
<ActionIcon
|
<ActionIcon
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
color="dark"
|
color="dark"
|
||||||
aria-label={t("Comments")}
|
|
||||||
onClick={() => toggleAside("comments")}
|
onClick={() => toggleAside("comments")}
|
||||||
>
|
>
|
||||||
<IconMessage size={20} stroke={2} />
|
<IconMessage size={20} stroke={2} />
|
||||||
@@ -110,7 +109,6 @@ export default function PageHeaderMenu({ readOnly }: PageHeaderMenuProps) {
|
|||||||
<ActionIcon
|
<ActionIcon
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
color="dark"
|
color="dark"
|
||||||
aria-label={t("Table of contents")}
|
|
||||||
onClick={() => toggleAside("toc")}
|
onClick={() => toggleAside("toc")}
|
||||||
>
|
>
|
||||||
<IconList size={20} stroke={2} />
|
<IconList size={20} stroke={2} />
|
||||||
@@ -147,7 +145,7 @@ function PageActionMenu({ readOnly }: PageActionMenuProps) {
|
|||||||
] = useDisclosure(false);
|
] = useDisclosure(false);
|
||||||
const [pageEditor] = useAtom(pageEditorAtom);
|
const [pageEditor] = useAtom(pageEditorAtom);
|
||||||
const pageUpdatedAt = useTimeAgo(page?.updatedAt);
|
const pageUpdatedAt = useTimeAgo(page?.updatedAt);
|
||||||
const favoriteIds = useFavoriteIds("page", page?.spaceId);
|
const favoriteIds = useFavoriteIds("page");
|
||||||
const addFavoriteMutation = useAddFavoriteMutation();
|
const addFavoriteMutation = useAddFavoriteMutation();
|
||||||
const removeFavoriteMutation = useRemoveFavoriteMutation();
|
const removeFavoriteMutation = useRemoveFavoriteMutation();
|
||||||
const isFavorited = page?.id ? favoriteIds.has(page.id) : false;
|
const isFavorited = page?.id ? favoriteIds.has(page.id) : false;
|
||||||
@@ -207,11 +205,7 @@ function PageActionMenu({ readOnly }: PageActionMenuProps) {
|
|||||||
arrowPosition="center"
|
arrowPosition="center"
|
||||||
>
|
>
|
||||||
<Menu.Target>
|
<Menu.Target>
|
||||||
<ActionIcon
|
<ActionIcon variant="subtle" color="dark">
|
||||||
variant="subtle"
|
|
||||||
color="dark"
|
|
||||||
aria-label={t("Page actions")}
|
|
||||||
>
|
|
||||||
<IconDots size={20} />
|
<IconDots size={20} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Menu.Target>
|
</Menu.Target>
|
||||||
@@ -422,15 +416,9 @@ function ConnectionWarning() {
|
|||||||
openDelay={250}
|
openDelay={250}
|
||||||
withArrow
|
withArrow
|
||||||
>
|
>
|
||||||
<ThemeIcon
|
<ActionIcon variant="default" c="red" style={{ border: "none" }}>
|
||||||
variant="default"
|
|
||||||
c="red"
|
|
||||||
role="status"
|
|
||||||
aria-label={t("Real-time editor connection lost. Retrying...")}
|
|
||||||
style={{ border: "none" }}
|
|
||||||
>
|
|
||||||
<IconWifiOff size={20} stroke={2} />
|
<IconWifiOff size={20} stroke={2} />
|
||||||
</ThemeIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,7 +12,6 @@ import {
|
|||||||
IconCheck,
|
IconCheck,
|
||||||
IconFileCode,
|
IconFileCode,
|
||||||
IconFileTypeDocx,
|
IconFileTypeDocx,
|
||||||
IconFileTypePdf,
|
|
||||||
IconFileTypeZip,
|
IconFileTypeZip,
|
||||||
IconMarkdown,
|
IconMarkdown,
|
||||||
IconX,
|
IconX,
|
||||||
@@ -67,7 +66,7 @@ export default function PageImportModal({
|
|||||||
<Modal.Content style={{ overflow: "hidden" }}>
|
<Modal.Content style={{ overflow: "hidden" }}>
|
||||||
<Modal.Header py={0}>
|
<Modal.Header py={0}>
|
||||||
<Modal.Title fw={500}>{t("Import pages")}</Modal.Title>
|
<Modal.Title fw={500}>{t("Import pages")}</Modal.Title>
|
||||||
<Modal.CloseButton aria-label={t("Close")} />
|
<Modal.CloseButton />
|
||||||
</Modal.Header>
|
</Modal.Header>
|
||||||
<Modal.Body>
|
<Modal.Body>
|
||||||
<ImportFormatSelection spaceId={spaceId} onClose={onClose} />
|
<ImportFormatSelection spaceId={spaceId} onClose={onClose} />
|
||||||
@@ -91,14 +90,12 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
const markdownFileRef = useRef<() => void>(null);
|
const markdownFileRef = useRef<() => void>(null);
|
||||||
const htmlFileRef = useRef<() => void>(null);
|
const htmlFileRef = useRef<() => void>(null);
|
||||||
const docxFileRef = useRef<() => void>(null);
|
const docxFileRef = useRef<() => void>(null);
|
||||||
const pdfFileRef = useRef<() => void>(null);
|
|
||||||
const notionFileRef = useRef<() => void>(null);
|
const notionFileRef = useRef<() => void>(null);
|
||||||
const confluenceFileRef = useRef<() => void>(null);
|
const confluenceFileRef = useRef<() => void>(null);
|
||||||
const zipFileRef = useRef<() => void>(null);
|
const zipFileRef = useRef<() => void>(null);
|
||||||
|
|
||||||
const canUseConfluence = useHasFeature(Feature.CONFLUENCE_IMPORT);
|
const canUseConfluence = useHasFeature(Feature.CONFLUENCE_IMPORT);
|
||||||
const canUseDocx = useHasFeature(Feature.DOCX_IMPORT);
|
const canUseDocx = useHasFeature(Feature.DOCX_IMPORT);
|
||||||
const canUsePdf = useHasFeature(Feature.PDF_IMPORT);
|
|
||||||
const upgradeLabel = useUpgradeLabel();
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
const handleZipUpload = async (selectedFile: File, source: string) => {
|
const handleZipUpload = async (selectedFile: File, source: string) => {
|
||||||
@@ -247,7 +244,7 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
}, 3000);
|
}, 3000);
|
||||||
}, [fileTaskId]);
|
}, [fileTaskId]);
|
||||||
|
|
||||||
const maxSingleFileSize = bytes("30mb");
|
const maxSingleFileSize = bytes("20mb");
|
||||||
|
|
||||||
const handleFileUpload = async (selectedFiles: File[]) => {
|
const handleFileUpload = async (selectedFiles: File[]) => {
|
||||||
if (!selectedFiles) {
|
if (!selectedFiles) {
|
||||||
@@ -301,7 +298,6 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
if (markdownFileRef.current) markdownFileRef.current();
|
if (markdownFileRef.current) markdownFileRef.current();
|
||||||
if (htmlFileRef.current) htmlFileRef.current();
|
if (htmlFileRef.current) htmlFileRef.current();
|
||||||
if (docxFileRef.current) docxFileRef.current();
|
if (docxFileRef.current) docxFileRef.current();
|
||||||
if (pdfFileRef.current) pdfFileRef.current();
|
|
||||||
|
|
||||||
const pageCountText =
|
const pageCountText =
|
||||||
pageCount === 1 ? `1 ${t("page")}` : `${pageCount} ${t("pages")}`;
|
pageCount === 1 ? `1 ${t("page")}` : `${pageCount} ${t("pages")}`;
|
||||||
@@ -332,15 +328,7 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<SimpleGrid cols={2}>
|
<SimpleGrid cols={2}>
|
||||||
<FileButton
|
<FileButton onChange={handleFileUpload} accept=".md" multiple resetRef={markdownFileRef}>
|
||||||
onChange={handleFileUpload}
|
|
||||||
accept=".md"
|
|
||||||
multiple
|
|
||||||
resetRef={markdownFileRef}
|
|
||||||
inputProps={{
|
|
||||||
"aria-label": t("Choose {{format}} file", { format: "Markdown" }),
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{(props) => (
|
{(props) => (
|
||||||
<Button
|
<Button
|
||||||
justify="start"
|
justify="start"
|
||||||
@@ -353,15 +341,7 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
)}
|
)}
|
||||||
</FileButton>
|
</FileButton>
|
||||||
|
|
||||||
<FileButton
|
<FileButton onChange={handleFileUpload} accept="text/html" multiple resetRef={htmlFileRef}>
|
||||||
onChange={handleFileUpload}
|
|
||||||
accept="text/html"
|
|
||||||
multiple
|
|
||||||
resetRef={htmlFileRef}
|
|
||||||
inputProps={{
|
|
||||||
"aria-label": t("Choose {{format}} file", { format: "HTML" }),
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{(props) => (
|
{(props) => (
|
||||||
<Button
|
<Button
|
||||||
justify="start"
|
justify="start"
|
||||||
@@ -379,9 +359,6 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
accept=".docx"
|
accept=".docx"
|
||||||
multiple
|
multiple
|
||||||
resetRef={docxFileRef}
|
resetRef={docxFileRef}
|
||||||
inputProps={{
|
|
||||||
"aria-label": t("Choose {{format}} file", { format: "Word (DOCX)" }),
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
{(props) => (
|
{(props) => (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
@@ -401,40 +378,10 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
)}
|
)}
|
||||||
</FileButton>
|
</FileButton>
|
||||||
|
|
||||||
<FileButton
|
|
||||||
onChange={handleFileUpload}
|
|
||||||
accept=".pdf"
|
|
||||||
multiple
|
|
||||||
resetRef={pdfFileRef}
|
|
||||||
inputProps={{
|
|
||||||
"aria-label": t("Choose {{format}} file", { format: "PDF" }),
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{(props) => (
|
|
||||||
<Tooltip
|
|
||||||
label={upgradeLabel}
|
|
||||||
disabled={canUsePdf}
|
|
||||||
>
|
|
||||||
<Button
|
|
||||||
disabled={!canUsePdf}
|
|
||||||
justify="start"
|
|
||||||
variant="default"
|
|
||||||
leftSection={<IconFileTypePdf size={18} />}
|
|
||||||
{...props}
|
|
||||||
>
|
|
||||||
PDF
|
|
||||||
</Button>
|
|
||||||
</Tooltip>
|
|
||||||
)}
|
|
||||||
</FileButton>
|
|
||||||
|
|
||||||
<FileButton
|
<FileButton
|
||||||
onChange={(file) => handleZipUpload(file, "notion")}
|
onChange={(file) => handleZipUpload(file, "notion")}
|
||||||
accept="application/zip"
|
accept="application/zip"
|
||||||
resetRef={notionFileRef}
|
resetRef={notionFileRef}
|
||||||
inputProps={{
|
|
||||||
"aria-label": t("Choose {{format}} file", { format: "Notion" }),
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
{(props) => (
|
{(props) => (
|
||||||
<Button
|
<Button
|
||||||
@@ -451,9 +398,6 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
onChange={(file) => handleZipUpload(file, "confluence")}
|
onChange={(file) => handleZipUpload(file, "confluence")}
|
||||||
accept="application/zip"
|
accept="application/zip"
|
||||||
resetRef={confluenceFileRef}
|
resetRef={confluenceFileRef}
|
||||||
inputProps={{
|
|
||||||
"aria-label": t("Choose {{format}} file", { format: "Confluence" }),
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
{(props) => (
|
{(props) => (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
@@ -491,9 +435,6 @@ function ImportFormatSelection({ spaceId, onClose }: ImportFormatSelection) {
|
|||||||
onChange={(file) => handleZipUpload(file, "generic")}
|
onChange={(file) => handleZipUpload(file, "generic")}
|
||||||
accept="application/zip"
|
accept="application/zip"
|
||||||
resetRef={zipFileRef}
|
resetRef={zipFileRef}
|
||||||
inputProps={{
|
|
||||||
"aria-label": t("Choose {{format}} file", { format: "ZIP" }),
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
{(props) => (
|
{(props) => (
|
||||||
<Group justify="center">
|
<Group justify="center">
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user