mirror of
https://github.com/docmost/docmost.git
synced 2026-05-07 14:43:06 +08:00
Compare commits
41 Commits
fix-245
...
space-watch
| Author | SHA1 | Date | |
|---|---|---|---|
| eb8a834d2f | |||
| 4966f9b152 | |||
| e1bbceb9a6 | |||
| 895c1817ae | |||
| 642024ba9d | |||
| 147d028036 | |||
| 992691e6e0 | |||
| 9aaa6c731c | |||
| fd91b11c6c | |||
| af8b0ddf3a | |||
| 879aa2c3d8 | |||
| c180d0e487 | |||
| a062f7a165 | |||
| cbd0dd4a0b | |||
| 2d6d829581 | |||
| 5cea30cc5c | |||
| bca85a49d6 | |||
| c9cdfa0f17 | |||
| 412962204c | |||
| a42ac3d450 | |||
| 642c92f779 | |||
| ccb35517bb | |||
| cbdb37ed0a | |||
| aa27d57624 | |||
| 3829b6cbef | |||
| 17da762984 | |||
| 859f16740b | |||
| 7981ef462e | |||
| 2d835da0e3 | |||
| a3559b7c33 | |||
| 803f1f0b81 | |||
| 4e8f533b91 | |||
| 7b0d8fe140 | |||
| 2f92278a9d | |||
| 53608eae35 | |||
| 0e4a1e7419 | |||
| 9125996e97 | |||
| fa4872e89e | |||
| 6d6f3a8a8e | |||
| 975b4dcaab | |||
| 6683c515cf |
+39
-39
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "client",
|
"name": "client",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "0.70.3",
|
"version": "0.71.1",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
"build": "tsc && vite build",
|
"build": "tsc && vite build",
|
||||||
@@ -10,76 +10,76 @@
|
|||||||
"format": "prettier --write \"src/**/*.tsx\" \"src/**/*.ts\""
|
"format": "prettier --write \"src/**/*.tsx\" \"src/**/*.ts\""
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@casl/react": "^4.0.0",
|
"@casl/react": "^5.0.1",
|
||||||
"@docmost/editor-ext": "workspace:*",
|
"@docmost/editor-ext": "workspace:*",
|
||||||
"@emoji-mart/data": "^1.2.1",
|
"@emoji-mart/data": "^1.2.1",
|
||||||
"@emoji-mart/react": "^1.1.1",
|
"@emoji-mart/react": "^1.1.1",
|
||||||
"@excalidraw/excalidraw": "0.18.0-3a5ef40",
|
"@excalidraw/excalidraw": "0.18.0-3a5ef40",
|
||||||
"@mantine/core": "^8.3.14",
|
"@mantine/core": "^8.3.18",
|
||||||
"@mantine/dates": "^8.3.14",
|
"@mantine/dates": "^8.3.18",
|
||||||
"@mantine/form": "^8.3.14",
|
"@mantine/form": "^8.3.18",
|
||||||
"@mantine/hooks": "^8.3.14",
|
"@mantine/hooks": "^8.3.18",
|
||||||
"@mantine/modals": "^8.3.14",
|
"@mantine/modals": "^8.3.18",
|
||||||
"@mantine/notifications": "^8.3.14",
|
"@mantine/notifications": "^8.3.18",
|
||||||
"@mantine/spotlight": "^8.3.14",
|
"@mantine/spotlight": "^8.3.18",
|
||||||
"@tabler/icons-react": "^3.36.1",
|
"@tabler/icons-react": "^3.40.0",
|
||||||
"@tanstack/react-query": "^5.90.17",
|
"@tanstack/react-query": "5.90.17",
|
||||||
"alfaaz": "^1.1.0",
|
"alfaaz": "^1.1.0",
|
||||||
"axios": "^1.13.5",
|
"axios": "1.13.6",
|
||||||
"blueimp-load-image": "^5.16.0",
|
"blueimp-load-image": "^5.16.0",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
"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": "^23.16.8",
|
"i18next": "^25.10.1",
|
||||||
"i18next-http-backend": "^2.7.3",
|
"i18next-http-backend": "^3.0.2",
|
||||||
"jotai": "^2.16.2",
|
"jotai": "^2.18.1",
|
||||||
"jotai-optics": "^0.4.0",
|
"jotai-optics": "^0.4.0",
|
||||||
"js-cookie": "^3.0.5",
|
"js-cookie": "^3.0.5",
|
||||||
"jwt-decode": "^4.0.0",
|
"jwt-decode": "^4.0.0",
|
||||||
"katex": "0.16.27",
|
"katex": "0.16.40",
|
||||||
"lowlight": "^3.3.0",
|
"lowlight": "^3.3.0",
|
||||||
"mantine-form-zod-resolver": "^1.3.0",
|
"mantine-form-zod-resolver": "^1.3.0",
|
||||||
"mermaid": "^11.12.2",
|
"mermaid": "^11.13.0",
|
||||||
"mitt": "^3.0.1",
|
"mitt": "^3.0.1",
|
||||||
"posthog-js": "1.345.5",
|
"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.17",
|
"react-clear-modal": "^2.0.18",
|
||||||
"react-dom": "^18.3.1",
|
"react-dom": "^18.3.1",
|
||||||
"react-drawio": "^1.0.7",
|
"react-drawio": "^1.0.7",
|
||||||
"react-error-boundary": "^4.1.2",
|
"react-error-boundary": "^6.1.1",
|
||||||
"react-helmet-async": "^2.0.5",
|
"react-helmet-async": "^3.0.0",
|
||||||
"react-i18next": "^15.0.1",
|
"react-i18next": "^16.5.8",
|
||||||
"react-router-dom": "^7.12.0",
|
"react-router-dom": "^7.13.1",
|
||||||
"semver": "^7.7.3",
|
"semver": "^7.7.4",
|
||||||
"socket.io-client": "^4.8.3",
|
"socket.io-client": "^4.8.3",
|
||||||
"tiptap-extension-global-drag-handle": "^0.1.18",
|
"tiptap-extension-global-drag-handle": "^0.1.18",
|
||||||
"zod": "^4.3.6"
|
"zod": "^4.3.6"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@eslint/js": "^9.16.0",
|
"@eslint/js": "^9.28.0",
|
||||||
"@tanstack/eslint-plugin-query": "^5.62.1",
|
"@tanstack/eslint-plugin-query": "^5.94.4",
|
||||||
"@types/blueimp-load-image": "^5.16.0",
|
"@types/blueimp-load-image": "^5.16.6",
|
||||||
"@types/file-saver": "^2.0.7",
|
"@types/file-saver": "^2.0.7",
|
||||||
"@types/js-cookie": "^3.0.6",
|
"@types/js-cookie": "^3.0.6",
|
||||||
"@types/katex": "^0.16.7",
|
"@types/katex": "^0.16.8",
|
||||||
"@types/node": "22.19.1",
|
"@types/node": "22.19.1",
|
||||||
"@types/react": "^18.3.12",
|
"@types/react": "^18.3.12",
|
||||||
"@types/react-dom": "^18.3.1",
|
"@types/react-dom": "^18.3.1",
|
||||||
"@vitejs/plugin-react": "^5.1.1",
|
"@vitejs/plugin-react": "^6.0.1",
|
||||||
"eslint": "^9.39.2",
|
"eslint": "^9.28.0",
|
||||||
"eslint-plugin-react": "^7.37.2",
|
"eslint-plugin-react": "^7.37.5",
|
||||||
"eslint-plugin-react-hooks": "^5.1.0",
|
"eslint-plugin-react-hooks": "^7.0.1",
|
||||||
"eslint-plugin-react-refresh": "^0.4.16",
|
"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.4.49",
|
"postcss": "^8.5.8",
|
||||||
"postcss-preset-mantine": "^1.17.0",
|
"postcss-preset-mantine": "^1.18.0",
|
||||||
"postcss-simple-vars": "^7.0.1",
|
"postcss-simple-vars": "^7.0.1",
|
||||||
"prettier": "^3.4.1",
|
"prettier": "^3.8.1",
|
||||||
"typescript": "^5.7.2",
|
"typescript": "^5.9.3",
|
||||||
"typescript-eslint": "^8.17.0",
|
"typescript-eslint": "^8.57.1",
|
||||||
"vite": "^7.2.4"
|
"vite": "8.0.5"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,6 +289,11 @@
|
|||||||
"Save & Exit": "Speichern & Beenden",
|
"Save & Exit": "Speichern & Beenden",
|
||||||
"Double-click to edit Excalidraw diagram": "Zum Bearbeiten des Excalidraw-Diagramms doppelklicken",
|
"Double-click to edit Excalidraw diagram": "Zum Bearbeiten des Excalidraw-Diagramms doppelklicken",
|
||||||
"Paste link": "Link einfügen",
|
"Paste link": "Link einfügen",
|
||||||
|
"Paste link or search pages": "Link einfügen oder Seiten durchsuchen",
|
||||||
|
"Link to web page": "Link zur Webseite",
|
||||||
|
"Recents": "Zuletzt verwendet",
|
||||||
|
"Page or URL": "Seite oder URL",
|
||||||
|
"Link title": "Linktitel",
|
||||||
"Edit link": "Link bearbeiten",
|
"Edit link": "Link bearbeiten",
|
||||||
"Remove link": "Link entfernen",
|
"Remove link": "Link entfernen",
|
||||||
"Add link": "Link hinzufügen",
|
"Add link": "Link hinzufügen",
|
||||||
@@ -336,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Horizontale Trennlinie einfügen",
|
"Insert horizontal rule divider": "Horizontale Trennlinie einfügen",
|
||||||
"Upload any image from your device.": "Laden Sie ein beliebiges Bild von Ihrem Gerät hoch.",
|
"Upload any image from your device.": "Laden Sie ein beliebiges Bild von Ihrem Gerät hoch.",
|
||||||
"Upload any video from your device.": "Laden Sie ein beliebiges Video von Ihrem Gerät hoch.",
|
"Upload any video from your device.": "Laden Sie ein beliebiges Video von Ihrem Gerät hoch.",
|
||||||
|
"Upload any audio from your device.": "Laden Sie beliebige Audiodateien von Ihrem Gerät hoch.",
|
||||||
"Upload any file from your device.": "Laden Sie eine beliebige Datei von Ihrem Gerät hoch.",
|
"Upload any file from your device.": "Laden Sie eine beliebige Datei von Ihrem Gerät hoch.",
|
||||||
"Uploading {{name}}": "Lade {{name}} hoch",
|
"Uploading {{name}}": "Lade {{name}} hoch",
|
||||||
"Uploading file": "Datei wird hochgeladen",
|
"Uploading file": "Datei wird hochgeladen",
|
||||||
@@ -346,6 +352,12 @@
|
|||||||
"Divider": "Trennlinie",
|
"Divider": "Trennlinie",
|
||||||
"Quote": "Zitat",
|
"Quote": "Zitat",
|
||||||
"Image": "Bild",
|
"Image": "Bild",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "PDF einbetten",
|
||||||
|
"Upload and embed a PDF file.": "Laden Sie eine PDF-Datei hoch und betten Sie sie ein.",
|
||||||
|
"Embed as PDF": "Als PDF einbetten",
|
||||||
|
"Failed to load PDF": "Fehler beim Laden der PDF",
|
||||||
|
"Convert to attachment": "In Anhang umwandeln",
|
||||||
"File attachment": "Dateianhang",
|
"File attachment": "Dateianhang",
|
||||||
"Toggle block": "Block umschalten",
|
"Toggle block": "Block umschalten",
|
||||||
"Callout": "Hinweisbox",
|
"Callout": "Hinweisbox",
|
||||||
@@ -437,9 +449,11 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Verhindern Sie, dass Mitglieder Seiten öffentlich teilen.",
|
"Prevent members from sharing pages publicly.": "Verhindern Sie, dass Mitglieder Seiten öffentlich teilen.",
|
||||||
"Toggle public sharing": "Öffentliches Teilen umschalten",
|
"Toggle public sharing": "Öffentliches Teilen umschalten",
|
||||||
"Toggle space public sharing": "Öffentliches Teilen im Bereich umschalten",
|
"Toggle space public sharing": "Öffentliches Teilen im Bereich umschalten",
|
||||||
|
"Allow viewers to comment": "Zuschauern erlauben, Kommentare zu hinterlassen",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Erlauben Sie Zuschauern, Kommentare auf Seiten in diesem Bereich hinzuzufügen.",
|
||||||
|
"Toggle viewer comments": "Zuschauerkommentare umschalten",
|
||||||
"Public sharing is disabled at the workspace level": "Öffentliches Teilen ist auf der Arbeitsbereichsebene deaktiviert",
|
"Public sharing is disabled at the workspace level": "Öffentliches Teilen ist auf der Arbeitsbereichsebene deaktiviert",
|
||||||
"Prevent pages in this space from being shared publicly.": "Verhindern Sie, dass Seiten in diesem Bereich öffentlich geteilt werden.",
|
"Prevent pages in this space from being shared publicly.": "Verhindern Sie, dass Seiten in diesem Bereich öffentlich geteilt werden.",
|
||||||
"Requires an enterprise license": "Erfordert eine Unternehmenslizenz",
|
|
||||||
"Page permissions": "Seitenberechtigungen",
|
"Page permissions": "Seitenberechtigungen",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "Steuern Sie, wer einzelne Seiten ansehen und bearbeiten kann. Verfügbar mit einer Enterprise-Lizenz.",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Steuern Sie, wer einzelne Seiten ansehen und bearbeiten kann. Verfügbar mit einer Enterprise-Lizenz.",
|
||||||
"Enable public sharing": "Öffentliches Teilen aktivieren",
|
"Enable public sharing": "Öffentliches Teilen aktivieren",
|
||||||
@@ -621,7 +635,9 @@
|
|||||||
"Generative AI (Ask AI)": "Generative KI (KI fragen)",
|
"Generative AI (Ask AI)": "Generative KI (KI fragen)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Aktivieren Sie die KI-unterstützte Inhaltserstellung im Editor. Ermöglicht Benutzern das Erzeugen, Verbessern, Übersetzen und Transformieren von Text.",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Aktivieren Sie die KI-unterstützte Inhaltserstellung im Editor. Ermöglicht Benutzern das Erzeugen, Verbessern, Übersetzen und Transformieren von Text.",
|
||||||
"Toggle generative AI": "Generative KI umschalten",
|
"Toggle generative AI": "Generative KI umschalten",
|
||||||
"Enterprise feature": "Enterprise-Funktion",
|
"Upgrade your plan": "Upgrade Ihres Plans",
|
||||||
|
"Available with a paid license": "Verfügbar mit einer kostenpflichtigen Lizenz",
|
||||||
|
"Upgrade your license tier.": "Stufen Sie Ihre Lizenz hoch.",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "KI ist nur in der Docmost Enterprise-Edition verfügbar. Kontaktieren Sie sales@docmost.com.",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "KI ist nur in der Docmost Enterprise-Edition verfügbar. Kontaktieren Sie sales@docmost.com.",
|
||||||
"AI & MCP": "KI & MCP",
|
"AI & MCP": "KI & MCP",
|
||||||
"AI": "KI",
|
"AI": "KI",
|
||||||
@@ -629,17 +645,15 @@
|
|||||||
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Aktivieren Sie den MCP-Server, damit KI-Assistenten und -Tools mit den Inhalten Ihres Arbeitsbereichs interagieren können.",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Aktivieren Sie den MCP-Server, damit KI-Assistenten und -Tools mit den Inhalten Ihres Arbeitsbereichs interagieren können.",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP ist nur in der Docmost Enterprise-Edition verfügbar. Kontaktieren Sie sales@docmost.com.",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP ist nur in der Docmost Enterprise-Edition verfügbar. Kontaktieren Sie sales@docmost.com.",
|
||||||
"MCP documentation": "MCP-Dokumentation",
|
|
||||||
"MCP Server URL": "MCP-Server-URL",
|
"MCP Server URL": "MCP-Server-URL",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "Verwenden Sie Ihren API-Schlüssel zur Authentifizierung. API-Schlüssel können in Ihren Kontoeinstellungen verwaltet werden.",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Verwenden Sie Ihren API-Schlüssel zur Authentifizierung. API-Schlüssel können in Ihren Kontoeinstellungen verwaltet werden.",
|
||||||
"Supported tools": "Unterstützte Tools",
|
"Supported tools": "Unterstützte Tools",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "In Ihrem Arbeitsbereich ist MCP aktiviert. Verwenden Sie Ihren API-Schlüssel, um KI-Assistenten anzubinden.",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "In Ihrem Arbeitsbereich ist MCP aktiviert. Verwenden Sie Ihren API-Schlüssel, um KI-Assistenten anzubinden.",
|
||||||
"MCP server URL:": "MCP-Server-URL:",
|
"MCP server URL:": "MCP-Server-URL:",
|
||||||
"Learn more": "Mehr erfahren",
|
"Learn more": "Mehr erfahren",
|
||||||
"View the": "Anzeigen",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Verwalten Sie API-Schlüssel für alle Nutzer im Arbeitsbereich. Siehe die <anchor>API-Dokumentation</anchor> für Details zur Verwendung.",
|
||||||
"for usage details.": "für Informationen zur Nutzung.",
|
"View the <anchor>API documentation</anchor> for usage details.": "Siehe die <anchor>API-Dokumentation</anchor> für Details zur Verwendung.",
|
||||||
"for setup instructions.": "für Einrichtungshinweise.",
|
"View the <anchor>MCP documentation</anchor>.": "Sehen Sie die <anchor>MCP-Dokumentation</anchor> ein.",
|
||||||
"API documentation": "API-Dokumentation",
|
|
||||||
"Sources": "Quellen",
|
"Sources": "Quellen",
|
||||||
"AI Answers not available for attachments": "KI-Antworten sind für Anhänge nicht verfügbar",
|
"AI Answers not available for attachments": "KI-Antworten sind für Anhänge nicht verfügbar",
|
||||||
"No answer available": "Keine Antwort verfügbar",
|
"No answer available": "Keine Antwort verfügbar",
|
||||||
@@ -654,12 +668,30 @@
|
|||||||
"Mark all as read": "Alle als gelesen markieren",
|
"Mark all as read": "Alle als gelesen markieren",
|
||||||
"Mark as read": "Als gelesen markieren",
|
"Mark as read": "Als gelesen markieren",
|
||||||
"More options": "Weitere Optionen",
|
"More options": "Weitere Optionen",
|
||||||
"mentioned you in a comment": "hat Sie in einem Kommentar erwähnt",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> hat Sie in einem Kommentar erwähnt",
|
||||||
"commented on a page": "hat auf einer Seite kommentiert",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> hat einen Kommentar auf einer Seite hinterlassen",
|
||||||
"resolved a comment": "hat einen Kommentar gelöst",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> hat einen Kommentar als erledigt markiert",
|
||||||
"mentioned you on a page": "hat Sie auf einer Seite erwähnt",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> hat Sie auf einer Seite erwähnt",
|
||||||
"gave you edit access to a page": "hat Ihnen Bearbeitungsrechte für eine Seite gegeben",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> hat Ihnen Bearbeitungszugriff auf eine Seite gegeben",
|
||||||
"gave you view access to a page": "hat Ihnen Leserechte für eine Seite gewährt",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> hat Ihnen Ansichtsrechte für eine Seite gegeben",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> hat eine Seite aktualisiert.",
|
||||||
|
"Watch page": "Seite beobachten",
|
||||||
|
"Stop watching": "Beobachtung beenden",
|
||||||
|
"Email notifications": "E-Mail-Benachrichtigungen",
|
||||||
|
"Page updates": "Seitenaktualisierungen",
|
||||||
|
"Get notified when pages you watch are updated.": "Erhalten Sie eine Benachrichtigung, wenn Seiten, die Sie beobachten, aktualisiert werden.",
|
||||||
|
"Page mentions": "Seiten-Erwähnungen",
|
||||||
|
"Get notified when someone mentions you on a page.": "Erhalten Sie eine Benachrichtigung, wenn Sie jemand auf einer Seite erwähnt.",
|
||||||
|
"Comment mentions": "Kommentar-Erwähnungen",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Erhalten Sie eine Benachrichtigung, wenn Sie jemand in einem Kommentar erwähnt.",
|
||||||
|
"New comments": "Neue Kommentare",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Erhalten Sie eine Benachrichtigung über neue Kommentare in Threads, an denen Sie teilnehmen.",
|
||||||
|
"Resolved comments": "Erledigte Kommentare",
|
||||||
|
"Get notified when your comment is resolved.": "Erhalten Sie eine Benachrichtigung, wenn Ihr Kommentar erledigt wurde.",
|
||||||
|
"You are now watching this page": "Sie beobachten diese Seite jetzt",
|
||||||
|
"You are no longer watching this page": "Sie beobachten diese Seite nicht mehr",
|
||||||
|
"Direct": "Direkt",
|
||||||
|
"Updates": "Aktualisierungen",
|
||||||
"Today": "Heute",
|
"Today": "Heute",
|
||||||
"Yesterday": "Gestern",
|
"Yesterday": "Gestern",
|
||||||
"This week": "Diese Woche",
|
"This week": "Diese Woche",
|
||||||
@@ -693,5 +725,31 @@
|
|||||||
"Failed to update trash retention": "Aktualisierung der Aufbewahrungsdauer des Papierkorbs fehlgeschlagen",
|
"Failed to update trash retention": "Aktualisierung der Aufbewahrungsdauer des Papierkorbs fehlgeschlagen",
|
||||||
"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",
|
||||||
|
"Verifying your email": "E-Mail wird überprüft",
|
||||||
|
"Please wait...": "Bitte warten...",
|
||||||
|
"Verification failed. The link may have expired.": "Überprüfung fehlgeschlagen. Der Link ist möglicherweise abgelaufen.",
|
||||||
|
"Check your email": "Prüfen Sie Ihr E-Mail-Postfach",
|
||||||
|
"We sent a verification link to {{email}}.": "Wir haben einen Bestätigungslink an {{email}} gesendet.",
|
||||||
|
"We sent a verification link to your email.": "Wir haben einen Bestätigungslink an Ihre E-Mail-Adresse gesendet.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Klicken Sie auf den Link, um Ihre E-Mail zu bestätigen und auf Ihren Arbeitsbereich zuzugreifen.",
|
||||||
|
"Resend verification email": "Bestätigungs-E-Mail erneut senden",
|
||||||
|
"Verification email sent. Please check your inbox.": "Bestätigungs-E-Mail gesendet. Bitte überprüfen Sie Ihr Postfach.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Fehler beim erneuten Senden der Bestätigungs-E-Mail. Bitte versuchen Sie es erneut.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Wir haben Ihnen eine E-Mail mit Ihren zugehörigen Arbeitsbereichen gesendet.",
|
||||||
|
"Load more": "Mehr laden",
|
||||||
|
"Log out of all devices": "Von allen Geräten abmelden",
|
||||||
|
"Log out of all sessions except this device": "Von allen Sitzungen außer diesem Gerät abmelden",
|
||||||
|
"This Device": "Dieses Gerät",
|
||||||
|
"Unknown device": "Unbekanntes Gerät",
|
||||||
|
"No active sessions": "Keine aktiven Sitzungen",
|
||||||
|
"Session revoked": "Sitzung widerrufen",
|
||||||
|
"All other sessions revoked": "Alle anderen Sitzungen widerrufen",
|
||||||
|
"Last used": "Zuletzt verwendet",
|
||||||
|
"Created": "Erstellt",
|
||||||
|
"Rename": "Umbenennen",
|
||||||
|
"Publish": "Veröffentlichen",
|
||||||
|
"Security": "Sicherheit",
|
||||||
|
"Enforce SSO": "SSO erzwingen",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Nach dem Erzwingen können sich Mitglieder nicht mehr mit E-Mail und Passwort anmelden."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Insert horizontal rule divider",
|
"Insert horizontal rule divider": "Insert horizontal rule divider",
|
||||||
"Upload any image from your device.": "Upload any image from your device.",
|
"Upload any image from your device.": "Upload any image from your device.",
|
||||||
"Upload any video from your device.": "Upload any video from your device.",
|
"Upload any video from your device.": "Upload any video from your device.",
|
||||||
|
"Upload any audio from your device.": "Upload any audio from your device.",
|
||||||
"Upload any file from your device.": "Upload any file from your device.",
|
"Upload any file from your device.": "Upload any file from your device.",
|
||||||
"Uploading {{name}}": "Uploading {{name}}",
|
"Uploading {{name}}": "Uploading {{name}}",
|
||||||
"Uploading file": "Uploading file",
|
"Uploading file": "Uploading file",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "Divider",
|
"Divider": "Divider",
|
||||||
"Quote": "Quote",
|
"Quote": "Quote",
|
||||||
"Image": "Image",
|
"Image": "Image",
|
||||||
|
"Audio": "Audio",
|
||||||
|
"Embed PDF": "Embed PDF",
|
||||||
|
"Upload and embed a PDF file.": "Upload and embed a PDF file.",
|
||||||
|
"Embed as PDF": "Embed as PDF",
|
||||||
|
"Failed to load PDF": "Failed to load PDF",
|
||||||
|
"Convert to attachment": "Convert to attachment",
|
||||||
"File attachment": "File attachment",
|
"File attachment": "File attachment",
|
||||||
"Toggle block": "Toggle block",
|
"Toggle block": "Toggle block",
|
||||||
"Callout": "Callout",
|
"Callout": "Callout",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Prevent members from sharing pages publicly.",
|
"Prevent members from sharing pages publicly.": "Prevent members from sharing pages publicly.",
|
||||||
"Toggle public sharing": "Toggle public sharing",
|
"Toggle public sharing": "Toggle public sharing",
|
||||||
"Toggle space public sharing": "Toggle space public sharing",
|
"Toggle space public sharing": "Toggle space public sharing",
|
||||||
|
"Allow viewers to comment": "Allow viewers to comment",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Allow viewers to add comments on pages in this space.",
|
||||||
|
"Toggle viewer comments": "Toggle viewer comments",
|
||||||
"Public sharing is disabled at the workspace level": "Public sharing is disabled at the workspace level",
|
"Public sharing is disabled at the workspace level": "Public sharing is disabled at the workspace level",
|
||||||
"Prevent pages in this space from being shared publicly.": "Prevent pages in this space from being shared publicly.",
|
"Prevent pages in this space from being shared publicly.": "Prevent pages in this space from being shared publicly.",
|
||||||
"Page permissions": "Page permissions",
|
"Page permissions": "Page permissions",
|
||||||
@@ -664,6 +674,28 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> mentioned you on a page",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> mentioned you on a page",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> gave you edit access to a page",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> gave you edit access to a page",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> gave you view access to a page",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> gave you view access to a page",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> updated a page",
|
||||||
|
"Watch page": "Watch page",
|
||||||
|
"Stop watching": "Stop watching",
|
||||||
|
"Watch space": "Watch space",
|
||||||
|
"Stop watching space": "Stop watching space",
|
||||||
|
"Email notifications": "Email notifications",
|
||||||
|
"Page updates": "Page updates",
|
||||||
|
"Get notified when pages you watch are updated.": "Receive notifications when the pages you watch are updated.",
|
||||||
|
"Page mentions": "Page mentions",
|
||||||
|
"Get notified when someone mentions you on a page.": "Receive notifications when someone mentions you on a page.",
|
||||||
|
"Comment mentions": "Comment mentions",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Receive notifications when someone mentions you in a comment.",
|
||||||
|
"New comments": "New comments",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Receive notifications about new comments in threads you are participating in.",
|
||||||
|
"Resolved comments": "Resolved comments",
|
||||||
|
"Get notified when your comment is resolved.": "Receive a notification when your comment is resolved.",
|
||||||
|
"You are now watching this page": "You’re now watching this page",
|
||||||
|
"You are no longer watching this page": "You’re no longer watching this page",
|
||||||
|
"You are now watching this space": "You’re now watching this space",
|
||||||
|
"You are no longer watching this space": "You’re no longer watching this space",
|
||||||
|
"Direct": "Direct",
|
||||||
|
"Updates": "Updates",
|
||||||
"Today": "Today",
|
"Today": "Today",
|
||||||
"Yesterday": "Yesterday",
|
"Yesterday": "Yesterday",
|
||||||
"This week": "This week",
|
"This week": "This week",
|
||||||
@@ -708,5 +740,20 @@
|
|||||||
"Resend verification email": "Resend verification email",
|
"Resend verification email": "Resend verification email",
|
||||||
"Verification email sent. Please check your inbox.": "Verification email sent. Please check your inbox.",
|
"Verification email sent. Please check your inbox.": "Verification email sent. Please check your inbox.",
|
||||||
"Failed to resend verification email. Please try again.": "Failed to resend verification email. Please try again.",
|
"Failed to resend verification email. Please try again.": "Failed to resend verification email. Please try again.",
|
||||||
"We've sent you an email with your associated workspaces.": "We've sent you an email with your associated workspaces."
|
"We've sent you an email with your associated workspaces.": "We've sent you an email with your associated workspaces.",
|
||||||
|
"Load more": "Load more",
|
||||||
|
"Log out of all devices": "Log out of all devices",
|
||||||
|
"Log out of all sessions except this device": "Log out of all sessions except this device",
|
||||||
|
"This Device": "This Device",
|
||||||
|
"Unknown device": "Unknown device",
|
||||||
|
"No active sessions": "No active sessions",
|
||||||
|
"Session revoked": "Session revoked",
|
||||||
|
"All other sessions revoked": "All other sessions revoked",
|
||||||
|
"Last used": "Last used",
|
||||||
|
"Created": "Created",
|
||||||
|
"Rename": "Rename",
|
||||||
|
"Publish": "Publish",
|
||||||
|
"Security": "Security",
|
||||||
|
"Enforce SSO": "Enforce SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Once enforced, members will not be able to login with email and password."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,6 +289,11 @@
|
|||||||
"Save & Exit": "Guardar y Salir",
|
"Save & Exit": "Guardar y Salir",
|
||||||
"Double-click to edit Excalidraw diagram": "Doble clic para editar el diagrama de Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Doble clic para editar el diagrama de Excalidraw",
|
||||||
"Paste link": "Pegar enlace",
|
"Paste link": "Pegar enlace",
|
||||||
|
"Paste link or search pages": "Pega un enlace o busca páginas",
|
||||||
|
"Link to web page": "Enlazar a una página web",
|
||||||
|
"Recents": "Recientes",
|
||||||
|
"Page or URL": "Página o URL",
|
||||||
|
"Link title": "Título del enlace",
|
||||||
"Edit link": "Editar enlace",
|
"Edit link": "Editar enlace",
|
||||||
"Remove link": "Eliminar enlace",
|
"Remove link": "Eliminar enlace",
|
||||||
"Add link": "Agregar enlace",
|
"Add link": "Agregar enlace",
|
||||||
@@ -336,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Insertar regla horizontal",
|
"Insert horizontal rule divider": "Insertar regla horizontal",
|
||||||
"Upload any image from your device.": "Sube cualquier imagen desde tu dispositivo.",
|
"Upload any image from your device.": "Sube cualquier imagen desde tu dispositivo.",
|
||||||
"Upload any video from your device.": "Sube cualquier video desde tu dispositivo.",
|
"Upload any video from your device.": "Sube cualquier video desde tu dispositivo.",
|
||||||
|
"Upload any audio from your device.": "Sube cualquier audio desde tu dispositivo.",
|
||||||
"Upload any file from your device.": "Sube cualquier archivo desde tu dispositivo.",
|
"Upload any file from your device.": "Sube cualquier archivo desde tu dispositivo.",
|
||||||
"Uploading {{name}}": "Subiendo {{name}}",
|
"Uploading {{name}}": "Subiendo {{name}}",
|
||||||
"Uploading file": "Subiendo archivo",
|
"Uploading file": "Subiendo archivo",
|
||||||
@@ -346,6 +352,12 @@
|
|||||||
"Divider": "Divisor",
|
"Divider": "Divisor",
|
||||||
"Quote": "Cita",
|
"Quote": "Cita",
|
||||||
"Image": "Imagen",
|
"Image": "Imagen",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "Adjuntar PDF",
|
||||||
|
"Upload and embed a PDF file.": "Sube y adjunta un archivo PDF.",
|
||||||
|
"Embed as PDF": "Adjuntar como PDF",
|
||||||
|
"Failed to load PDF": "Error al cargar el PDF",
|
||||||
|
"Convert to attachment": "Convertir en adjunto",
|
||||||
"File attachment": "Adjunto de archivo",
|
"File attachment": "Adjunto de archivo",
|
||||||
"Toggle block": "Alternar bloque",
|
"Toggle block": "Alternar bloque",
|
||||||
"Callout": "Aviso",
|
"Callout": "Aviso",
|
||||||
@@ -437,9 +449,11 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Evitar que los miembros compartan páginas públicamente.",
|
"Prevent members from sharing pages publicly.": "Evitar que los miembros compartan páginas públicamente.",
|
||||||
"Toggle public sharing": "Alternar el uso compartido público",
|
"Toggle public sharing": "Alternar el uso compartido público",
|
||||||
"Toggle space public sharing": "Alternar el uso compartido público del espacio",
|
"Toggle space public sharing": "Alternar el uso compartido público del espacio",
|
||||||
|
"Allow viewers to comment": "Permitir que los espectadores comenten",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Permitir que los espectadores agreguen comentarios en las páginas de este espacio.",
|
||||||
|
"Toggle viewer comments": "Activar/desactivar comentarios de los espectadores",
|
||||||
"Public sharing is disabled at the workspace level": "El uso compartido público está desactivado a nivel de espacio de trabajo",
|
"Public sharing is disabled at the workspace level": "El uso compartido público está desactivado a nivel de espacio de trabajo",
|
||||||
"Prevent pages in this space from being shared publicly.": "Evitar que las páginas en este espacio se compartan públicamente.",
|
"Prevent pages in this space from being shared publicly.": "Evitar que las páginas en este espacio se compartan públicamente.",
|
||||||
"Requires an enterprise license": "Requiere una licencia empresarial",
|
|
||||||
"Page permissions": "Permisos de la página},{",
|
"Page permissions": "Permisos de la página},{",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "Controla quién puede ver y editar páginas individuales. Disponible con una licencia empresarial.",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Controla quién puede ver y editar páginas individuales. Disponible con una licencia empresarial.",
|
||||||
"Enable public sharing": "Activar el uso compartido público",
|
"Enable public sharing": "Activar el uso compartido público",
|
||||||
@@ -621,7 +635,9 @@
|
|||||||
"Generative AI (Ask AI)": "IA generativa (Preguntar a la IA)",
|
"Generative AI (Ask AI)": "IA generativa (Preguntar a la IA)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Habilitar la generación de contenido impulsada por IA en el editor. Permite a los usuarios generar, mejorar, traducir y transformar texto.",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Habilitar la generación de contenido impulsada por IA en el editor. Permite a los usuarios generar, mejorar, traducir y transformar texto.",
|
||||||
"Toggle generative AI": "Activar IA generativa",
|
"Toggle generative AI": "Activar IA generativa",
|
||||||
"Enterprise feature": "Función empresarial",
|
"Upgrade your plan": "Mejora tu plan",
|
||||||
|
"Available with a paid license": "Disponible con una licencia de pago",
|
||||||
|
"Upgrade your license tier.": "Mejora el nivel de tu licencia.",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "La IA solo está disponible en la edición empresarial de Docmost. Contacte con sales@docmost.com.",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "La IA solo está disponible en la edición empresarial de Docmost. Contacte con sales@docmost.com.",
|
||||||
"AI & MCP": "IA y MCP",
|
"AI & MCP": "IA y MCP",
|
||||||
"AI": "IA",
|
"AI": "IA",
|
||||||
@@ -629,17 +645,15 @@
|
|||||||
"Model Context Protocol (MCP)": "Protocolo de Contexto del Modelo (MCP)",
|
"Model Context Protocol (MCP)": "Protocolo de Contexto del Modelo (MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Habilite el servidor MCP para permitir que asistentes de IA y herramientas interactúen con el contenido de su espacio de trabajo.",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Habilite el servidor MCP para permitir que asistentes de IA y herramientas interactúen con el contenido de su espacio de trabajo.",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP solo está disponible en la edición empresarial de Docmost. Contacte con sales@docmost.com.",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP solo está disponible en la edición empresarial de Docmost. Contacte con sales@docmost.com.",
|
||||||
"MCP documentation": "Documentación de MCP",
|
|
||||||
"MCP Server URL": "URL del servidor MCP",
|
"MCP Server URL": "URL del servidor MCP",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "Use su clave API para la autenticación. Puede gestionar las claves API en la configuración de su cuenta.",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Use su clave API para la autenticación. Puede gestionar las claves API en la configuración de su cuenta.",
|
||||||
"Supported tools": "Herramientas compatibles",
|
"Supported tools": "Herramientas compatibles",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Su espacio de trabajo tiene MCP habilitado. Use su clave API para conectar asistentes de IA.",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Su espacio de trabajo tiene MCP habilitado. Use su clave API para conectar asistentes de IA.",
|
||||||
"MCP server URL:": "URL del servidor MCP:",
|
"MCP server URL:": "URL del servidor MCP:",
|
||||||
"Learn more": "Más información",
|
"Learn more": "Más información",
|
||||||
"View the": "Ver la",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Gestiona las claves de API para todos los usuarios en el espacio de trabajo. Consulta la <anchor>documentación de la API</anchor> para detalles de uso.",
|
||||||
"for usage details.": "para detalles de uso.",
|
"View the <anchor>API documentation</anchor> for usage details.": "Consulta la <anchor>documentación de la API</anchor> para detalles de uso.",
|
||||||
"for setup instructions.": "para instrucciones de configuración.",
|
"View the <anchor>MCP documentation</anchor>.": "Consulta la <anchor>documentación de MCP</anchor>.",
|
||||||
"API documentation": "Documentación de la API",
|
|
||||||
"Sources": "Fuentes",
|
"Sources": "Fuentes",
|
||||||
"AI Answers not available for attachments": "Respuestas de IA no disponibles para archivos adjuntos",
|
"AI Answers not available for attachments": "Respuestas de IA no disponibles para archivos adjuntos",
|
||||||
"No answer available": "No hay respuesta disponible",
|
"No answer available": "No hay respuesta disponible",
|
||||||
@@ -654,12 +668,30 @@
|
|||||||
"Mark all as read": "Marcar todo como leído",
|
"Mark all as read": "Marcar todo como leído",
|
||||||
"Mark as read": "Marcar como leído",
|
"Mark as read": "Marcar como leído",
|
||||||
"More options": "Más opciones",
|
"More options": "Más opciones",
|
||||||
"mentioned you in a comment": "te mencionó en un comentario",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> te mencionó en un comentario",
|
||||||
"commented on a page": "comentó en una página",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> comentó en una página",
|
||||||
"resolved a comment": "resolvió un comentario",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> resolvió un comentario",
|
||||||
"mentioned you on a page": "te mencionó en una página",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> te mencionó en una página",
|
||||||
"gave you edit access to a page": "Te dio acceso para editar una página.",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> te dio acceso de edición a una página",
|
||||||
"gave you view access to a page": "Te dio acceso para ver una página.",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> te dio acceso de visualización a una página",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> actualizó una página.",
|
||||||
|
"Watch page": "Seguir página",
|
||||||
|
"Stop watching": "Dejar de seguir",
|
||||||
|
"Email notifications": "Notificaciones por correo electrónico",
|
||||||
|
"Page updates": "Actualizaciones de página",
|
||||||
|
"Get notified when pages you watch are updated.": "Recibe una notificación cuando se actualicen las páginas que sigues.",
|
||||||
|
"Page mentions": "Menciones en la página",
|
||||||
|
"Get notified when someone mentions you on a page.": "Recibe una notificación cuando alguien te mencione en una página.",
|
||||||
|
"Comment mentions": "Menciones en comentarios",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Recibe una notificación cuando alguien te mencione en un comentario.",
|
||||||
|
"New comments": "Nuevos comentarios",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Recibe una notificación sobre nuevos comentarios en los hilos donde participas.",
|
||||||
|
"Resolved comments": "Comentarios resueltos",
|
||||||
|
"Get notified when your comment is resolved.": "Recibe una notificación cuando tu comentario sea resuelto.",
|
||||||
|
"You are now watching this page": "Ahora sigues esta página",
|
||||||
|
"You are no longer watching this page": "Ya no sigues esta página",
|
||||||
|
"Direct": "Directo",
|
||||||
|
"Updates": "Actualizaciones",
|
||||||
"Today": "Hoy",
|
"Today": "Hoy",
|
||||||
"Yesterday": "Ayer",
|
"Yesterday": "Ayer",
|
||||||
"This week": "Esta semana",
|
"This week": "Esta semana",
|
||||||
@@ -693,5 +725,31 @@
|
|||||||
"Failed to update trash retention": "No se pudo actualizar la retención de la papelera.",
|
"Failed to update trash retention": "No se pudo actualizar la retención de la papelera.",
|
||||||
"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",
|
||||||
|
"Verifying your email": "Verificando tu correo electrónico",
|
||||||
|
"Please wait...": "Por favor, espera...",
|
||||||
|
"Verification failed. The link may have expired.": "La verificación ha fallado. Es posible que el enlace haya expirado.",
|
||||||
|
"Check your email": "Revisa tu correo electrónico",
|
||||||
|
"We sent a verification link to {{email}}.": "Te enviamos un enlace de verificación a {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Te enviamos un enlace de verificación a tu correo.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Haz clic en el enlace para verificar tu correo electrónico y acceder a tu espacio de trabajo.",
|
||||||
|
"Resend verification email": "Reenviar correo de verificación",
|
||||||
|
"Verification email sent. Please check your inbox.": "Correo de verificación enviado. Por favor, revisa tu bandeja de entrada.",
|
||||||
|
"Failed to resend verification email. Please try again.": "No se pudo reenviar el correo de verificación. Por favor, intente de nuevo.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Te hemos enviado un correo electrónico con tus espacios de trabajo asociados.",
|
||||||
|
"Load more": "Cargar más",
|
||||||
|
"Log out of all devices": "Cerrar sesión en todos los dispositivos",
|
||||||
|
"Log out of all sessions except this device": "Cerrar sesión en todos los dispositivos excepto este",
|
||||||
|
"This Device": "Este dispositivo",
|
||||||
|
"Unknown device": "Dispositivo desconocido",
|
||||||
|
"No active sessions": "No hay sesiones activas",
|
||||||
|
"Session revoked": "Sesión revocada",
|
||||||
|
"All other sessions revoked": "Todas las demás sesiones revocadas",
|
||||||
|
"Last used": "Último uso",
|
||||||
|
"Created": "Creado",
|
||||||
|
"Rename": "Renombrar",
|
||||||
|
"Publish": "Publicar",
|
||||||
|
"Security": "Seguridad",
|
||||||
|
"Enforce SSO": "Forzar SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Una vez forzado, los miembros no podrán iniciar sesión con correo electrónico y contraseña."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,6 +289,11 @@
|
|||||||
"Save & Exit": "Enregistrer & Quitter",
|
"Save & Exit": "Enregistrer & Quitter",
|
||||||
"Double-click to edit Excalidraw diagram": "Double-cliquez pour modifier le diagramme Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Double-cliquez pour modifier le diagramme Excalidraw",
|
||||||
"Paste link": "Coller le lien",
|
"Paste link": "Coller le lien",
|
||||||
|
"Paste link or search pages": "Coller le lien ou rechercher des pages",
|
||||||
|
"Link to web page": "Lien vers une page web",
|
||||||
|
"Recents": "Récents",
|
||||||
|
"Page or URL": "Page ou URL",
|
||||||
|
"Link title": "Titre du lien",
|
||||||
"Edit link": "Modifier le lien",
|
"Edit link": "Modifier le lien",
|
||||||
"Remove link": "Supprimer le lien",
|
"Remove link": "Supprimer le lien",
|
||||||
"Add link": "Ajouter un lien",
|
"Add link": "Ajouter un lien",
|
||||||
@@ -336,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Insérer un séparateur de règle horizontale",
|
"Insert horizontal rule divider": "Insérer un séparateur de règle horizontale",
|
||||||
"Upload any image from your device.": "Téléchargez n'importe quelle image depuis votre appareil.",
|
"Upload any image from your device.": "Téléchargez n'importe quelle image depuis votre appareil.",
|
||||||
"Upload any video from your device.": "Téléchargez n'importe quelle vidéo depuis votre appareil.",
|
"Upload any video from your device.": "Téléchargez n'importe quelle vidéo depuis votre appareil.",
|
||||||
|
"Upload any audio from your device.": "Téléchargez n'importe quel fichier audio depuis votre appareil.",
|
||||||
"Upload any file from your device.": "Téléchargez n'importe quel fichier depuis votre appareil.",
|
"Upload any file from your device.": "Téléchargez n'importe quel fichier depuis votre appareil.",
|
||||||
"Uploading {{name}}": "Téléchargement de {{name}}",
|
"Uploading {{name}}": "Téléchargement de {{name}}",
|
||||||
"Uploading file": "Téléchargement du fichier",
|
"Uploading file": "Téléchargement du fichier",
|
||||||
@@ -346,6 +352,12 @@
|
|||||||
"Divider": "Diviseur",
|
"Divider": "Diviseur",
|
||||||
"Quote": "Citation",
|
"Quote": "Citation",
|
||||||
"Image": "Image",
|
"Image": "Image",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "Intégrer un PDF",
|
||||||
|
"Upload and embed a PDF file.": "Téléchargez et intégrez un fichier PDF.",
|
||||||
|
"Embed as PDF": "Intégrer comme PDF",
|
||||||
|
"Failed to load PDF": "Échec du chargement du PDF",
|
||||||
|
"Convert to attachment": "Convertir en pièce jointe",
|
||||||
"File attachment": "Pièce jointe",
|
"File attachment": "Pièce jointe",
|
||||||
"Toggle block": "Basculer le bloc",
|
"Toggle block": "Basculer le bloc",
|
||||||
"Callout": "Appel",
|
"Callout": "Appel",
|
||||||
@@ -410,7 +422,7 @@
|
|||||||
"Move page": "Déplacer la page",
|
"Move page": "Déplacer la page",
|
||||||
"Move page to a different space.": "Déplacer la page vers un autre espace.",
|
"Move page to a different space.": "Déplacer la page vers un autre espace.",
|
||||||
"Real-time editor connection lost. Retrying...": "Connexion avec l'éditeur en temps réel perdue. Nouvelle tentative...",
|
"Real-time editor connection lost. Retrying...": "Connexion avec l'éditeur en temps réel perdue. Nouvelle tentative...",
|
||||||
"Table of contents": "",
|
"Table of contents": "Table des matières.",
|
||||||
"Add headings (H1, H2, H3) to generate a table of contents.": "Ajoutez des titres (H1, H2, H3) pour générer une table des matières.",
|
"Add headings (H1, H2, H3) to generate a table of contents.": "Ajoutez des titres (H1, H2, H3) pour générer une table des matières.",
|
||||||
"Share": "Partager",
|
"Share": "Partager",
|
||||||
"Public sharing": "Partage public",
|
"Public sharing": "Partage public",
|
||||||
@@ -437,9 +449,11 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Empêcher les membres de partager des pages publiquement.",
|
"Prevent members from sharing pages publicly.": "Empêcher les membres de partager des pages publiquement.",
|
||||||
"Toggle public sharing": "Basculer le partage public",
|
"Toggle public sharing": "Basculer le partage public",
|
||||||
"Toggle space public sharing": "Basculer le partage public de l'espace",
|
"Toggle space public sharing": "Basculer le partage public de l'espace",
|
||||||
|
"Allow viewers to comment": "Autoriser les spectateurs à commenter",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Autoriser les spectateurs à ajouter des commentaires sur les pages de cet espace.",
|
||||||
|
"Toggle viewer comments": "Basculer les commentaires des spectateurs",
|
||||||
"Public sharing is disabled at the workspace level": "Le partage public est désactivé au niveau de l'espace de travail",
|
"Public sharing is disabled at the workspace level": "Le partage public est désactivé au niveau de l'espace de travail",
|
||||||
"Prevent pages in this space from being shared publicly.": "Empêcher les pages de cet espace d'être partagées publiquement.",
|
"Prevent pages in this space from being shared publicly.": "Empêcher les pages de cet espace d'être partagées publiquement.",
|
||||||
"Requires an enterprise license": "Nécessite une licence d'entreprise",
|
|
||||||
"Page permissions": "Autorisations de la page",
|
"Page permissions": "Autorisations de la page",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "Contrôlez qui peut consulter et modifier chaque page. Disponible avec une licence Entreprise.",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Contrôlez qui peut consulter et modifier chaque page. Disponible avec une licence Entreprise.",
|
||||||
"Enable public sharing": "Activer le partage public",
|
"Enable public sharing": "Activer le partage public",
|
||||||
@@ -621,7 +635,9 @@
|
|||||||
"Generative AI (Ask AI)": "IA générative (Demandez à l'IA)",
|
"Generative AI (Ask AI)": "IA générative (Demandez à l'IA)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Activer la génération de contenu assistée par IA dans l'éditeur. Permet aux utilisateurs de générer, améliorer, traduire et transformer du texte.",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Activer la génération de contenu assistée par IA dans l'éditeur. Permet aux utilisateurs de générer, améliorer, traduire et transformer du texte.",
|
||||||
"Toggle generative AI": "Activer/désactiver l'IA générative",
|
"Toggle generative AI": "Activer/désactiver l'IA générative",
|
||||||
"Enterprise feature": "Fonctionnalité entreprise",
|
"Upgrade your plan": "Mettez à niveau votre forfait",
|
||||||
|
"Available with a paid license": "Disponible avec une licence payante",
|
||||||
|
"Upgrade your license tier.": "Mettez à niveau votre niveau de licence.",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "L'IA n'est disponible que dans l'édition Entreprise de Docmost. Contactez sales@docmost.com.",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "L'IA n'est disponible que dans l'édition Entreprise de Docmost. Contactez sales@docmost.com.",
|
||||||
"AI & MCP": "IA & MCP",
|
"AI & MCP": "IA & MCP",
|
||||||
"AI": "IA",
|
"AI": "IA",
|
||||||
@@ -629,17 +645,15 @@
|
|||||||
"Model Context Protocol (MCP)": "Protocole de contexte de modèle (MCP)",
|
"Model Context Protocol (MCP)": "Protocole de contexte de modèle (MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Activez le serveur MCP pour permettre aux assistants et outils IA d'interagir avec le contenu de votre espace de travail.",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Activez le serveur MCP pour permettre aux assistants et outils IA d'interagir avec le contenu de votre espace de travail.",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP n'est disponible que dans l'édition Entreprise de Docmost. Contactez sales@docmost.com.",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP n'est disponible que dans l'édition Entreprise de Docmost. Contactez sales@docmost.com.",
|
||||||
"MCP documentation": "Documentation MCP",
|
|
||||||
"MCP Server URL": "URL du serveur MCP",
|
"MCP Server URL": "URL du serveur MCP",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "Utilisez votre clé API pour l'authentification. Vous pouvez gérer les clés API dans les paramètres de votre compte.",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Utilisez votre clé API pour l'authentification. Vous pouvez gérer les clés API dans les paramètres de votre compte.",
|
||||||
"Supported tools": "Outils pris en charge",
|
"Supported tools": "Outils pris en charge",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Votre espace de travail a MCP activé. Utilisez votre clé API pour connecter des assistants IA.",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Votre espace de travail a MCP activé. Utilisez votre clé API pour connecter des assistants IA.",
|
||||||
"MCP server URL:": "URL du serveur MCP :",
|
"MCP server URL:": "URL du serveur MCP :",
|
||||||
"Learn more": "En savoir plus",
|
"Learn more": "En savoir plus",
|
||||||
"View the": "Voir la",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Gérez les clés API pour tous les utilisateurs de l'espace de travail. Consultez la <anchor>documentation API</anchor> pour plus de détails sur l'utilisation.",
|
||||||
"for usage details.": "pour les détails d'utilisation.",
|
"View the <anchor>API documentation</anchor> for usage details.": "Consultez la <anchor>documentation API</anchor> pour plus de détails sur l'utilisation.",
|
||||||
"for setup instructions.": "pour les instructions de configuration.",
|
"View the <anchor>MCP documentation</anchor>.": "Consultez la <anchor>documentation MCP</anchor>.",
|
||||||
"API documentation": "Documentation de l'API",
|
|
||||||
"Sources": "Sources",
|
"Sources": "Sources",
|
||||||
"AI Answers not available for attachments": "Réponses IA non disponibles pour les pièces jointes",
|
"AI Answers not available for attachments": "Réponses IA non disponibles pour les pièces jointes",
|
||||||
"No answer available": "Pas de réponse disponible",
|
"No answer available": "Pas de réponse disponible",
|
||||||
@@ -654,12 +668,30 @@
|
|||||||
"Mark all as read": "Tout marquer comme lu",
|
"Mark all as read": "Tout marquer comme lu",
|
||||||
"Mark as read": "Marquer comme lu",
|
"Mark as read": "Marquer comme lu",
|
||||||
"More options": "Plus d'options",
|
"More options": "Plus d'options",
|
||||||
"mentioned you in a comment": "vous a mentionné dans un commentaire",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> vous a mentionné dans un commentaire",
|
||||||
"commented on a page": "a commenté une page",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> a commenté une page",
|
||||||
"resolved a comment": "a résolu un commentaire",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> a résolu un commentaire",
|
||||||
"mentioned you on a page": "vous a mentionné sur une page",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> vous a mentionné sur une page",
|
||||||
"gave you edit access to a page": "vous a donné l'accès pour modifier une page",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> vous a donné l'accès en modification à une page",
|
||||||
"gave you view access to a page": "vous a donné l'accès pour consulter une page",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> vous a donné l'accès en lecture à une page",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> a mis à jour une page.",
|
||||||
|
"Watch page": "Surveiller la page",
|
||||||
|
"Stop watching": "Ne plus surveiller",
|
||||||
|
"Email notifications": "Notifications par e-mail",
|
||||||
|
"Page updates": "Mises à jour de la page",
|
||||||
|
"Get notified when pages you watch are updated.": "Recevez une notification lorsque les pages que vous surveillez sont mises à jour.",
|
||||||
|
"Page mentions": "Mentions sur la page",
|
||||||
|
"Get notified when someone mentions you on a page.": "Recevez une notification lorsqu'une personne vous mentionne sur une page.",
|
||||||
|
"Comment mentions": "Mentions dans les commentaires",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Recevez une notification lorsqu'une personne vous mentionne dans un commentaire.",
|
||||||
|
"New comments": "Nouveaux commentaires",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Recevez une notification concernant les nouveaux commentaires dans les fils auxquels vous participez.",
|
||||||
|
"Resolved comments": "Commentaires résolus",
|
||||||
|
"Get notified when your comment is resolved.": "Recevez une notification lorsque votre commentaire est résolu.",
|
||||||
|
"You are now watching this page": "Vous surveillez désormais cette page",
|
||||||
|
"You are no longer watching this page": "Vous ne surveillez plus cette page",
|
||||||
|
"Direct": "Direct",
|
||||||
|
"Updates": "Mises à jour",
|
||||||
"Today": "Aujourd'hui",
|
"Today": "Aujourd'hui",
|
||||||
"Yesterday": "Hier",
|
"Yesterday": "Hier",
|
||||||
"This week": "Cette semaine",
|
"This week": "Cette semaine",
|
||||||
@@ -693,5 +725,31 @@
|
|||||||
"Failed to update trash retention": "Échec de la mise à jour de la durée de conservation de la corbeille",
|
"Failed to update trash retention": "Échec de la mise à jour de la durée de conservation de la corbeille",
|
||||||
"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",
|
||||||
|
"Verifying your email": "Vérification de votre e-mail",
|
||||||
|
"Please wait...": "Veuillez patienter...",
|
||||||
|
"Verification failed. The link may have expired.": "Échec de la vérification. Le lien a peut-être expiré.",
|
||||||
|
"Check your email": "Vérifiez votre e-mail",
|
||||||
|
"We sent a verification link to {{email}}.": "Nous avons envoyé un lien de vérification à {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Nous avons envoyé un lien de vérification à votre adresse e-mail.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Cliquez sur le lien pour vérifier votre adresse et accéder à votre espace de travail.",
|
||||||
|
"Resend verification email": "Renvoyer l'e-mail de vérification",
|
||||||
|
"Verification email sent. Please check your inbox.": "E-mail de vérification envoyé. Veuillez vérifier votre boîte de réception.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Échec de l'envoi du nouvel e-mail de vérification. Veuillez réessayer.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Nous vous avons envoyé un e-mail avec vos espaces de travail associés.",
|
||||||
|
"Load more": "Charger plus",
|
||||||
|
"Log out of all devices": "Déconnexion de tous les appareils",
|
||||||
|
"Log out of all sessions except this device": "Déconnexion de toutes les sessions sauf cet appareil",
|
||||||
|
"This Device": "Cet appareil",
|
||||||
|
"Unknown device": "Appareil inconnu",
|
||||||
|
"No active sessions": "Aucune session active",
|
||||||
|
"Session revoked": "Session révoquée",
|
||||||
|
"All other sessions revoked": "Toutes les autres sessions révoquées",
|
||||||
|
"Last used": "Dernière utilisation",
|
||||||
|
"Created": "Créé",
|
||||||
|
"Rename": "Renommer",
|
||||||
|
"Publish": "Publier",
|
||||||
|
"Security": "Sécurité",
|
||||||
|
"Enforce SSO": "Imposer SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Une fois imposé, les membres ne pourront plus se connecter par e-mail et mot de passe."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,6 +289,11 @@
|
|||||||
"Save & Exit": "Salva ed esci",
|
"Save & Exit": "Salva ed esci",
|
||||||
"Double-click to edit Excalidraw diagram": "Fai doppio clic per modificare il diagramma di Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Fai doppio clic per modificare il diagramma di Excalidraw",
|
||||||
"Paste link": "Incolla link",
|
"Paste link": "Incolla link",
|
||||||
|
"Paste link or search pages": "Incolla il link o cerca le pagine",
|
||||||
|
"Link to web page": "Collega a una pagina web",
|
||||||
|
"Recents": "Recenti",
|
||||||
|
"Page or URL": "Pagina o URL",
|
||||||
|
"Link title": "Titolo del link",
|
||||||
"Edit link": "Modifica link",
|
"Edit link": "Modifica link",
|
||||||
"Remove link": "Rimuovi link",
|
"Remove link": "Rimuovi link",
|
||||||
"Add link": "Aggiungi link",
|
"Add link": "Aggiungi link",
|
||||||
@@ -336,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Inserisci divisore di regola orizzontale",
|
"Insert horizontal rule divider": "Inserisci divisore di regola orizzontale",
|
||||||
"Upload any image from your device.": "Carica un'immagine dal tuo dispositivo.",
|
"Upload any image from your device.": "Carica un'immagine dal tuo dispositivo.",
|
||||||
"Upload any video from your device.": "Carica qualsiasi video dal tuo dispositivo.",
|
"Upload any video from your device.": "Carica qualsiasi video dal tuo dispositivo.",
|
||||||
|
"Upload any audio from your device.": "Carica qualsiasi audio dal tuo dispositivo.",
|
||||||
"Upload any file from your device.": "Carica qualsiasi file dal tuo dispositivo.",
|
"Upload any file from your device.": "Carica qualsiasi file dal tuo dispositivo.",
|
||||||
"Uploading {{name}}": "Caricamento di {{name}}",
|
"Uploading {{name}}": "Caricamento di {{name}}",
|
||||||
"Uploading file": "Caricamento file",
|
"Uploading file": "Caricamento file",
|
||||||
@@ -346,6 +352,12 @@
|
|||||||
"Divider": "Divisore",
|
"Divider": "Divisore",
|
||||||
"Quote": "Preventivo",
|
"Quote": "Preventivo",
|
||||||
"Image": "Immagine",
|
"Image": "Immagine",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "Incorpora PDF",
|
||||||
|
"Upload and embed a PDF file.": "Carica e incorpora un file PDF.",
|
||||||
|
"Embed as PDF": "Incorpora come PDF",
|
||||||
|
"Failed to load PDF": "Caricamento del PDF non riuscito",
|
||||||
|
"Convert to attachment": "Converti in allegato",
|
||||||
"File attachment": "Allegato file",
|
"File attachment": "Allegato file",
|
||||||
"Toggle block": "Attiva blocco",
|
"Toggle block": "Attiva blocco",
|
||||||
"Callout": "Avviso",
|
"Callout": "Avviso",
|
||||||
@@ -437,9 +449,11 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Impedisci ai membri di condividere pubblicamente le pagine.",
|
"Prevent members from sharing pages publicly.": "Impedisci ai membri di condividere pubblicamente le pagine.",
|
||||||
"Toggle public sharing": "Attiva/disattiva la condivisione pubblica",
|
"Toggle public sharing": "Attiva/disattiva la condivisione pubblica",
|
||||||
"Toggle space public sharing": "Attiva/disattiva la condivisione pubblica nello spazio",
|
"Toggle space public sharing": "Attiva/disattiva la condivisione pubblica nello spazio",
|
||||||
|
"Allow viewers to comment": "Consenti agli utenti di commentare",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Consenti agli utenti di aggiungere commenti alle pagine in questo spazio.",
|
||||||
|
"Toggle viewer comments": "Attiva/disattiva i commenti degli utenti",
|
||||||
"Public sharing is disabled at the workspace level": "La condivisione pubblica è disabilitata a livello di area di lavoro",
|
"Public sharing is disabled at the workspace level": "La condivisione pubblica è disabilitata a livello di area di lavoro",
|
||||||
"Prevent pages in this space from being shared publicly.": "Impedisci che le pagine in questo spazio vengano condivise pubblicamente.",
|
"Prevent pages in this space from being shared publicly.": "Impedisci che le pagine in questo spazio vengano condivise pubblicamente.",
|
||||||
"Requires an enterprise license": "Richiede una licenza enterprise",
|
|
||||||
"Page permissions": "Autorizzazioni della pagina.",
|
"Page permissions": "Autorizzazioni della pagina.",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "Controlla chi può visualizzare e modificare le singole pagine. Disponibile con una licenza Enterprise.",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Controlla chi può visualizzare e modificare le singole pagine. Disponibile con una licenza Enterprise.",
|
||||||
"Enable public sharing": "Abilita la condivisione pubblica",
|
"Enable public sharing": "Abilita la condivisione pubblica",
|
||||||
@@ -621,7 +635,9 @@
|
|||||||
"Generative AI (Ask AI)": "AI generativa (Chiedi AI)",
|
"Generative AI (Ask AI)": "AI generativa (Chiedi AI)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Abilita la generazione di contenuti con AI nell'editor. Consente agli utenti di generare, migliorare, tradurre e trasformare il testo.",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Abilita la generazione di contenuti con AI nell'editor. Consente agli utenti di generare, migliorare, tradurre e trasformare il testo.",
|
||||||
"Toggle generative AI": "Attiva/Disattiva AI generativa",
|
"Toggle generative AI": "Attiva/Disattiva AI generativa",
|
||||||
"Enterprise feature": "Funzionalità Enterprise",
|
"Upgrade your plan": "Aggiorna il tuo piano",
|
||||||
|
"Available with a paid license": "Disponibile con una licenza a pagamento",
|
||||||
|
"Upgrade your license tier.": "Aggiorna il livello della tua licenza.",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "L'IA è disponibile solo nell'edizione Enterprise di Docmost. Contatta sales@docmost.com.",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "L'IA è disponibile solo nell'edizione Enterprise di Docmost. Contatta sales@docmost.com.",
|
||||||
"AI & MCP": "IA e MCP",
|
"AI & MCP": "IA e MCP",
|
||||||
"AI": "IA",
|
"AI": "IA",
|
||||||
@@ -629,17 +645,15 @@
|
|||||||
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Abilita il server MCP per consentire ad assistenti e strumenti IA di interagire con i contenuti del tuo spazio di lavoro.",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Abilita il server MCP per consentire ad assistenti e strumenti IA di interagire con i contenuti del tuo spazio di lavoro.",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP è disponibile solo nell'edizione Enterprise di Docmost. Contatta sales@docmost.com.",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP è disponibile solo nell'edizione Enterprise di Docmost. Contatta sales@docmost.com.",
|
||||||
"MCP documentation": "Documentazione MCP",
|
|
||||||
"MCP Server URL": "URL del server MCP",
|
"MCP Server URL": "URL del server MCP",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "Usa la tua chiave API per l'autenticazione. Puoi gestire le chiavi API nelle impostazioni del tuo account.",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Usa la tua chiave API per l'autenticazione. Puoi gestire le chiavi API nelle impostazioni del tuo account.",
|
||||||
"Supported tools": "Strumenti supportati",
|
"Supported tools": "Strumenti supportati",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Il tuo spazio di lavoro ha MCP abilitato. Usa la tua chiave API per collegare gli assistenti IA.",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Il tuo spazio di lavoro ha MCP abilitato. Usa la tua chiave API per collegare gli assistenti IA.",
|
||||||
"MCP server URL:": "URL del server MCP:",
|
"MCP server URL:": "URL del server MCP:",
|
||||||
"Learn more": "Scopri di più",
|
"Learn more": "Scopri di più",
|
||||||
"View the": "Visualizza la",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Gestisci le API key per tutti gli utenti nello spazio di lavoro. Consulta la <anchor>documentazione API</anchor> per i dettagli sull'utilizzo.",
|
||||||
"for usage details.": "per i dettagli sull'utilizzo.",
|
"View the <anchor>API documentation</anchor> for usage details.": "Consulta la <anchor>documentazione API</anchor> per i dettagli sull'utilizzo.",
|
||||||
"for setup instructions.": "per le istruzioni di configurazione.",
|
"View the <anchor>MCP documentation</anchor>.": "Consulta la <anchor>documentazione MCP</anchor>.",
|
||||||
"API documentation": "Documentazione API",
|
|
||||||
"Sources": "Fonti",
|
"Sources": "Fonti",
|
||||||
"AI Answers not available for attachments": "Risposte AI non disponibili per gli allegati",
|
"AI Answers not available for attachments": "Risposte AI non disponibili per gli allegati",
|
||||||
"No answer available": "Nessuna risposta disponibile",
|
"No answer available": "Nessuna risposta disponibile",
|
||||||
@@ -654,12 +668,30 @@
|
|||||||
"Mark all as read": "Segna tutto come letto",
|
"Mark all as read": "Segna tutto come letto",
|
||||||
"Mark as read": "Segna come letto",
|
"Mark as read": "Segna come letto",
|
||||||
"More options": "Altre opzioni",
|
"More options": "Altre opzioni",
|
||||||
"mentioned you in a comment": "ti ha menzionato in un commento",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> ti ha menzionato in un commento",
|
||||||
"commented on a page": "ha commentato una pagina",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> ha commentato una pagina",
|
||||||
"resolved a comment": "ha risolto un commento",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> ha risolto un commento",
|
||||||
"mentioned you on a page": "ti ha menzionato in una pagina",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> ti ha menzionato su una pagina",
|
||||||
"gave you edit access to a page": "ti ha concesso l'accesso per modificare una pagina",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> ti ha dato l'accesso di modifica a una pagina",
|
||||||
"gave you view access to a page": "ti ha concesso l'accesso per visualizzare una pagina",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> ti ha dato l'accesso di visualizzazione a una pagina",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> ha aggiornato una pagina.",
|
||||||
|
"Watch page": "Segui pagina",
|
||||||
|
"Stop watching": "Smetti di seguire",
|
||||||
|
"Email notifications": "Notifiche email",
|
||||||
|
"Page updates": "Aggiornamenti pagina",
|
||||||
|
"Get notified when pages you watch are updated.": "Ricevi una notifica quando le pagine che segui vengono aggiornate.",
|
||||||
|
"Page mentions": "Menzioni nella pagina",
|
||||||
|
"Get notified when someone mentions you on a page.": "Ricevi una notifica quando qualcuno ti menziona su una pagina.",
|
||||||
|
"Comment mentions": "Menzioni nei commenti",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Ricevi una notifica quando qualcuno ti menziona in un commento.",
|
||||||
|
"New comments": "Nuovi commenti",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Ricevi una notifica sui nuovi commenti nelle discussioni a cui partecipi.",
|
||||||
|
"Resolved comments": "Commenti risolti",
|
||||||
|
"Get notified when your comment is resolved.": "Ricevi una notifica quando il tuo commento viene risolto.",
|
||||||
|
"You are now watching this page": "Ora stai seguendo questa pagina",
|
||||||
|
"You are no longer watching this page": "Non stai più seguendo questa pagina",
|
||||||
|
"Direct": "Diretto",
|
||||||
|
"Updates": "Aggiornamenti",
|
||||||
"Today": "Oggi",
|
"Today": "Oggi",
|
||||||
"Yesterday": "Ieri",
|
"Yesterday": "Ieri",
|
||||||
"This week": "Questa settimana",
|
"This week": "Questa settimana",
|
||||||
@@ -693,5 +725,31 @@
|
|||||||
"Failed to update trash retention": "Impossibile aggiornare la conservazione del cestino",
|
"Failed to update trash retention": "Impossibile aggiornare la conservazione del cestino",
|
||||||
"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",
|
||||||
|
"Verifying your email": "Verifica della tua email",
|
||||||
|
"Please wait...": "Attendere...",
|
||||||
|
"Verification failed. The link may have expired.": "Verifica non riuscita. Il link potrebbe essere scaduto.",
|
||||||
|
"Check your email": "Controlla la tua email",
|
||||||
|
"We sent a verification link to {{email}}.": "Abbiamo inviato un link di verifica a {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Abbiamo inviato un link di verifica alla tua email.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Clicca sul link per verificare la tua email e accedere al tuo workspace.",
|
||||||
|
"Resend verification email": "Invia nuovamente l'email di verifica",
|
||||||
|
"Verification email sent. Please check your inbox.": "Email di verifica inviata. Controlla la tua casella di posta.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Invio dell'email di verifica non riuscito. Si prega di riprovare.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Ti abbiamo inviato un'email con i workspace associati.",
|
||||||
|
"Load more": "Carica altro",
|
||||||
|
"Log out of all devices": "Disconnetti da tutti i dispositivi",
|
||||||
|
"Log out of all sessions except this device": "Disconnetti da tutte le sessioni tranne questo dispositivo",
|
||||||
|
"This Device": "Questo dispositivo",
|
||||||
|
"Unknown device": "Dispositivo sconosciuto",
|
||||||
|
"No active sessions": "Nessuna sessione attiva",
|
||||||
|
"Session revoked": "Sessione revocata",
|
||||||
|
"All other sessions revoked": "Tutte le altre sessioni revocate",
|
||||||
|
"Last used": "Ultimo utilizzo",
|
||||||
|
"Created": "Creato",
|
||||||
|
"Rename": "Rinomina",
|
||||||
|
"Publish": "Pubblica",
|
||||||
|
"Security": "Sicurezza",
|
||||||
|
"Enforce SSO": "Forza SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Una volta attivata, i membri non potranno più accedere con email e password."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,6 +289,11 @@
|
|||||||
"Save & Exit": "保存して終了",
|
"Save & Exit": "保存して終了",
|
||||||
"Double-click to edit Excalidraw diagram": "ダブルクリックして Excalidraw 図を編集",
|
"Double-click to edit Excalidraw diagram": "ダブルクリックして Excalidraw 図を編集",
|
||||||
"Paste link": "リンクを貼り付け",
|
"Paste link": "リンクを貼り付け",
|
||||||
|
"Paste link or search pages": "リンクを貼り付けるかページを検索してください。 ",
|
||||||
|
"Link to web page": "ウェブページへのリンク",
|
||||||
|
"Recents": "最近使用したもの",
|
||||||
|
"Page or URL": "ページまたはURL",
|
||||||
|
"Link title": "リンクタイトル",
|
||||||
"Edit link": "リンクを編集",
|
"Edit link": "リンクを編集",
|
||||||
"Remove link": "リンクを削除",
|
"Remove link": "リンクを削除",
|
||||||
"Add link": "リンクを追加",
|
"Add link": "リンクを追加",
|
||||||
@@ -336,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "区切り線を挿入します",
|
"Insert horizontal rule divider": "区切り線を挿入します",
|
||||||
"Upload any image from your device.": "デバイスから画像をアップロードします",
|
"Upload any image from your device.": "デバイスから画像をアップロードします",
|
||||||
"Upload any video from your device.": "デバイスから動画をアップロードします",
|
"Upload any video from your device.": "デバイスから動画をアップロードします",
|
||||||
|
"Upload any audio from your device.": "デバイスから音声ファイルをアップロードします。",
|
||||||
"Upload any file from your device.": "デバイスからファイルをアップロードします",
|
"Upload any file from your device.": "デバイスからファイルをアップロードします",
|
||||||
"Uploading {{name}}": "{{name}} をアップロード中",
|
"Uploading {{name}}": "{{name}} をアップロード中",
|
||||||
"Uploading file": "ファイルをアップロード中",
|
"Uploading file": "ファイルをアップロード中",
|
||||||
@@ -346,6 +352,12 @@
|
|||||||
"Divider": "区切り線",
|
"Divider": "区切り線",
|
||||||
"Quote": "引用",
|
"Quote": "引用",
|
||||||
"Image": "画像",
|
"Image": "画像",
|
||||||
|
"Audio": "音声。",
|
||||||
|
"Embed PDF": "PDFを埋め込む",
|
||||||
|
"Upload and embed a PDF file.": "PDFファイルをアップロードして埋め込みます。",
|
||||||
|
"Embed as PDF": "PDFとして埋め込む",
|
||||||
|
"Failed to load PDF": "PDFの読み込みに失敗しました",
|
||||||
|
"Convert to attachment": "添付ファイルに変換",
|
||||||
"File attachment": "ファイル添付",
|
"File attachment": "ファイル添付",
|
||||||
"Toggle block": "ブロックを切り替える",
|
"Toggle block": "ブロックを切り替える",
|
||||||
"Callout": "コールアウト",
|
"Callout": "コールアウト",
|
||||||
@@ -437,9 +449,11 @@
|
|||||||
"Prevent members from sharing pages publicly.": "メンバーがページを公開で共有するのを防ぐ。",
|
"Prevent members from sharing pages publicly.": "メンバーがページを公開で共有するのを防ぐ。",
|
||||||
"Toggle public sharing": "公開共有を切り替える",
|
"Toggle public sharing": "公開共有を切り替える",
|
||||||
"Toggle space public sharing": "スペースの公開共有を切り替える",
|
"Toggle space public sharing": "スペースの公開共有を切り替える",
|
||||||
|
"Allow viewers to comment": "閲覧者によるコメントを許可",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "このスペース内のページに閲覧者がコメントを追加できるようにします。",
|
||||||
|
"Toggle viewer comments": "閲覧者コメントの切り替え",
|
||||||
"Public sharing is disabled at the workspace level": "ワークスペースレベルで公開共有が無効になっています",
|
"Public sharing is disabled at the workspace level": "ワークスペースレベルで公開共有が無効になっています",
|
||||||
"Prevent pages in this space from being shared publicly.": "このスペース内のページが公開で共有されるのを防ぐ。",
|
"Prevent pages in this space from being shared publicly.": "このスペース内のページが公開で共有されるのを防ぐ。",
|
||||||
"Requires an enterprise license": "エンタープライズライセンスが必要です",
|
|
||||||
"Page permissions": "ページのアクセス権",
|
"Page permissions": "ページのアクセス権",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "個々のページを誰が表示・編集できるかを制御します。エンタープライズライセンスで利用可能です。",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "個々のページを誰が表示・編集できるかを制御します。エンタープライズライセンスで利用可能です。",
|
||||||
"Enable public sharing": "公開共有を有効にする",
|
"Enable public sharing": "公開共有を有効にする",
|
||||||
@@ -621,7 +635,9 @@
|
|||||||
"Generative AI (Ask AI)": "生成AI (Ask AI)",
|
"Generative AI (Ask AI)": "生成AI (Ask AI)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "エディターでAIを活用したコンテンツ生成を有効にします。ユーザーがテキストの生成、改善、翻訳、および変換を行うことができます。",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "エディターでAIを活用したコンテンツ生成を有効にします。ユーザーがテキストの生成、改善、翻訳、および変換を行うことができます。",
|
||||||
"Toggle generative AI": "生成AIを切り替える",
|
"Toggle generative AI": "生成AIを切り替える",
|
||||||
"Enterprise feature": "エンタープライズ機能",
|
"Upgrade your plan": "プランをアップグレードする",
|
||||||
|
"Available with a paid license": "有料ライセンスで利用可能",
|
||||||
|
"Upgrade your license tier.": "ライセンスタイアをアップグレードしてください。",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI は Docmost のエンタープライズ版でのみ利用可能です。sales@docmost.com までお問い合わせください。",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI は Docmost のエンタープライズ版でのみ利用可能です。sales@docmost.com までお問い合わせください。",
|
||||||
"AI & MCP": "AI と MCP",
|
"AI & MCP": "AI と MCP",
|
||||||
"AI": "AI",
|
"AI": "AI",
|
||||||
@@ -629,17 +645,15 @@
|
|||||||
"Model Context Protocol (MCP)": "モデルコンテキストプロトコル(MCP)",
|
"Model Context Protocol (MCP)": "モデルコンテキストプロトコル(MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "MCP サーバーを有効にして、AI アシスタントやツールがワークスペースのコンテンツとやり取りできるようにします。",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "MCP サーバーを有効にして、AI アシスタントやツールがワークスペースのコンテンツとやり取りできるようにします。",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP は Docmost のエンタープライズ版でのみ利用可能です。sales@docmost.com までお問い合わせください。",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP は Docmost のエンタープライズ版でのみ利用可能です。sales@docmost.com までお問い合わせください。",
|
||||||
"MCP documentation": "MCP ドキュメント",
|
|
||||||
"MCP Server URL": "MCP サーバーの URL",
|
"MCP Server URL": "MCP サーバーの URL",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "認証には API キーを使用してください。API キーはアカウント設定で管理できます。",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "認証には API キーを使用してください。API キーはアカウント設定で管理できます。",
|
||||||
"Supported tools": "サポートされているツール",
|
"Supported tools": "サポートされているツール",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "このワークスペースでは MCP が有効になっています。AI アシスタントを接続するには API キーを使用してください。",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "このワークスペースでは MCP が有効になっています。AI アシスタントを接続するには API キーを使用してください。",
|
||||||
"MCP server URL:": "MCP サーバーの URL:",
|
"MCP server URL:": "MCP サーバーの URL:",
|
||||||
"Learn more": "詳細を見る",
|
"Learn more": "詳細を見る",
|
||||||
"View the": "表示",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "ワークスペース内のすべてのユーザーのAPIキーを管理します。利用方法の詳細は<anchor>APIドキュメント</anchor>をご覧ください。",
|
||||||
"for usage details.": "使用方法の詳細については。",
|
"View the <anchor>API documentation</anchor> for usage details.": "利用方法の詳細は<anchor>APIドキュメント</anchor>をご覧ください。",
|
||||||
"for setup instructions.": "設定手順については。",
|
"View the <anchor>MCP documentation</anchor>.": "<anchor>MCPドキュメント</anchor>をご覧ください。",
|
||||||
"API documentation": "API ドキュメント",
|
|
||||||
"Sources": "ソース",
|
"Sources": "ソース",
|
||||||
"AI Answers not available for attachments": "添付ファイルにはAI回答を利用できません",
|
"AI Answers not available for attachments": "添付ファイルにはAI回答を利用できません",
|
||||||
"No answer available": "回答がありません",
|
"No answer available": "回答がありません",
|
||||||
@@ -654,12 +668,30 @@
|
|||||||
"Mark all as read": "すべてを既読にする",
|
"Mark all as read": "すべてを既読にする",
|
||||||
"Mark as read": "既読にする",
|
"Mark as read": "既読にする",
|
||||||
"More options": "その他のオプション",
|
"More options": "その他のオプション",
|
||||||
"mentioned you in a comment": "コメントであなたに言及しました",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold>さんがコメントであなたに言及しました",
|
||||||
"commented on a page": "ページにコメントしました",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold>さんがページにコメントしました",
|
||||||
"resolved a comment": "コメントを解決しました",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold>さんがコメントを解決しました",
|
||||||
"mentioned you on a page": "ページ上であなたに言及しました",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>さんがページであなたに言及しました",
|
||||||
"gave you edit access to a page": "あなたにページの編集アクセス権を付与しました",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>さんがページの編集権限をあなたに付与しました",
|
||||||
"gave you view access to a page": "あなたにページの閲覧アクセス権を付与しました",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>さんがページの閲覧権限をあなたに付与しました",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold>さんがページを更新しました。",
|
||||||
|
"Watch page": "ページをウォッチ",
|
||||||
|
"Stop watching": "ウォッチを解除",
|
||||||
|
"Email notifications": "メール通知",
|
||||||
|
"Page updates": "ページの更新",
|
||||||
|
"Get notified when pages you watch are updated.": "ウォッチしているページが更新されたときに通知を受け取ります。",
|
||||||
|
"Page mentions": "ページでの言及",
|
||||||
|
"Get notified when someone mentions you on a page.": "誰かがページであなたに言及したとき通知を受け取ります。",
|
||||||
|
"Comment mentions": "コメントでの言及",
|
||||||
|
"Get notified when someone mentions you in a comment.": "誰かがコメントであなたに言及したとき通知を受け取ります。",
|
||||||
|
"New comments": "新しいコメント",
|
||||||
|
"Get notified about new comments on threads you participate in.": "参加しているスレッドに新しいコメントがあると通知されます。",
|
||||||
|
"Resolved comments": "解決済みコメント",
|
||||||
|
"Get notified when your comment is resolved.": "あなたのコメントが解決されたとき通知を受け取ります。",
|
||||||
|
"You are now watching this page": "このページをウォッチしています",
|
||||||
|
"You are no longer watching this page": "このページのウォッチを解除しました",
|
||||||
|
"Direct": "直接",
|
||||||
|
"Updates": "アップデート",
|
||||||
"Today": "今日",
|
"Today": "今日",
|
||||||
"Yesterday": "昨日",
|
"Yesterday": "昨日",
|
||||||
"This week": "今週",
|
"This week": "今週",
|
||||||
@@ -693,5 +725,31 @@
|
|||||||
"Failed to update trash retention": "ゴミ箱保持期間の更新に失敗しました",
|
"Failed to update trash retention": "ゴミ箱保持期間の更新に失敗しました",
|
||||||
"Removed page restriction": "ページの制限を解除しました",
|
"Removed page restriction": "ページの制限を解除しました",
|
||||||
"Added page permission": "ページの権限を追加しました",
|
"Added page permission": "ページの権限を追加しました",
|
||||||
"Removed page permission": "ページの権限を削除しました"
|
"Removed page permission": "ページの権限を削除しました",
|
||||||
|
"Verifying your email": "メールを確認中",
|
||||||
|
"Please wait...": "お待ちください…",
|
||||||
|
"Verification failed. The link may have expired.": "認証に失敗しました。リンクの有効期限が切れている可能性があります。",
|
||||||
|
"Check your email": "メールを確認してください",
|
||||||
|
"We sent a verification link to {{email}}.": "確認用リンクを{{email}}に送信しました。",
|
||||||
|
"We sent a verification link to your email.": "確認用リンクをあなたのメールアドレスに送信しました。",
|
||||||
|
"Click the link to verify your email and access your workspace.": "リンクをクリックしてメールを認証し、ワークスペースにアクセスしてください。",
|
||||||
|
"Resend verification email": "確認メールを再送信",
|
||||||
|
"Verification email sent. Please check your inbox.": "確認メールを送信しました。受信箱をご確認ください。",
|
||||||
|
"Failed to resend verification email. Please try again.": "確認メールの再送信に失敗しました。もう一度お試しください。",
|
||||||
|
"We've sent you an email with your associated workspaces.": "紐づいているワークスペース情報をメールでお送りしました。",
|
||||||
|
"Load more": "もっと見る",
|
||||||
|
"Log out of all devices": "すべての端末からログアウト",
|
||||||
|
"Log out of all sessions except this device": "この端末以外の全セッションからログアウト",
|
||||||
|
"This Device": "このデバイス",
|
||||||
|
"Unknown device": "不明な端末",
|
||||||
|
"No active sessions": "アクティブなセッションはありません",
|
||||||
|
"Session revoked": "セッションが取り消されました",
|
||||||
|
"All other sessions revoked": "他のすべてのセッションが取り消されました",
|
||||||
|
"Last used": "最終使用",
|
||||||
|
"Created": "作成日",
|
||||||
|
"Rename": "名前を変更",
|
||||||
|
"Publish": "公開する",
|
||||||
|
"Security": "セキュリティ",
|
||||||
|
"Enforce SSO": "SSOを強制する",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "一度SSOが強制されると、メールとパスワードでログインできなくなります。"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,6 +289,11 @@
|
|||||||
"Save & Exit": "저장 후 나가기",
|
"Save & Exit": "저장 후 나가기",
|
||||||
"Double-click to edit Excalidraw diagram": "Excalidraw diagram을 편집하려면 더블 클릭하세요",
|
"Double-click to edit Excalidraw diagram": "Excalidraw diagram을 편집하려면 더블 클릭하세요",
|
||||||
"Paste link": "링크 붙여넣기",
|
"Paste link": "링크 붙여넣기",
|
||||||
|
"Paste link or search pages": "링크를 붙여넣거나 페이지를 검색",
|
||||||
|
"Link to web page": "웹페이지에 링크하기",
|
||||||
|
"Recents": "최근 항목",
|
||||||
|
"Page or URL": "페이지 또는 URL",
|
||||||
|
"Link title": "링크 제목",
|
||||||
"Edit link": "링크 수정",
|
"Edit link": "링크 수정",
|
||||||
"Remove link": "링크 제거",
|
"Remove link": "링크 제거",
|
||||||
"Add link": "링크 추가",
|
"Add link": "링크 추가",
|
||||||
@@ -336,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "가로 구분선 삽입",
|
"Insert horizontal rule divider": "가로 구분선 삽입",
|
||||||
"Upload any image from your device.": "기기에서 이미지를 업로드하세요.",
|
"Upload any image from your device.": "기기에서 이미지를 업로드하세요.",
|
||||||
"Upload any video from your device.": "기기에서 비디오를 업로드하세요.",
|
"Upload any video from your device.": "기기에서 비디오를 업로드하세요.",
|
||||||
|
"Upload any audio from your device.": "기기에서 오디오를 업로드하세요.",
|
||||||
"Upload any file from your device.": "기기에서 파일을 업로드하세요.",
|
"Upload any file from your device.": "기기에서 파일을 업로드하세요.",
|
||||||
"Uploading {{name}}": "{{name}} 업로드 중",
|
"Uploading {{name}}": "{{name}} 업로드 중",
|
||||||
"Uploading file": "파일 업로드 중",
|
"Uploading file": "파일 업로드 중",
|
||||||
@@ -346,6 +352,12 @@
|
|||||||
"Divider": "구분선",
|
"Divider": "구분선",
|
||||||
"Quote": "인용",
|
"Quote": "인용",
|
||||||
"Image": "이미지",
|
"Image": "이미지",
|
||||||
|
"Audio": "오디오.",
|
||||||
|
"Embed PDF": "PDF 임베드",
|
||||||
|
"Upload and embed a PDF file.": "PDF 파일을 업로드하고 임베드하세요.",
|
||||||
|
"Embed as PDF": "PDF로 임베드",
|
||||||
|
"Failed to load PDF": "PDF 로드 실패",
|
||||||
|
"Convert to attachment": "첨부 파일로 변환",
|
||||||
"File attachment": "파일 첨부",
|
"File attachment": "파일 첨부",
|
||||||
"Toggle block": "블록 토글",
|
"Toggle block": "블록 토글",
|
||||||
"Callout": "경고 상자",
|
"Callout": "경고 상자",
|
||||||
@@ -437,9 +449,11 @@
|
|||||||
"Prevent members from sharing pages publicly.": "멤버들이 페이지를 공개적으로 공유하지 못하도록 방지하십시오.",
|
"Prevent members from sharing pages publicly.": "멤버들이 페이지를 공개적으로 공유하지 못하도록 방지하십시오.",
|
||||||
"Toggle public sharing": "공유 전환",
|
"Toggle public sharing": "공유 전환",
|
||||||
"Toggle space public sharing": "공간 공유 전환",
|
"Toggle space public sharing": "공간 공유 전환",
|
||||||
|
"Allow viewers to comment": "뷰어가 댓글을 달 수 있도록 허용",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "이 공간 내 페이지에 뷰어가 댓글을 추가할 수 있도록 허용합니다.",
|
||||||
|
"Toggle viewer comments": "뷰어 댓글 전환",
|
||||||
"Public sharing is disabled at the workspace level": "워크스페이스 수준에서 공유가 비활성화되었습니다.",
|
"Public sharing is disabled at the workspace level": "워크스페이스 수준에서 공유가 비활성화되었습니다.",
|
||||||
"Prevent pages in this space from being shared publicly.": "이 공간의 페이지가 공개적으로 공유되지 않도록 방지하십시오.",
|
"Prevent pages in this space from being shared publicly.": "이 공간의 페이지가 공개적으로 공유되지 않도록 방지하십시오.",
|
||||||
"Requires an enterprise license": "기업 라이센스가 필요합니다.",
|
|
||||||
"Page permissions": "페이지 권한},{",
|
"Page permissions": "페이지 권한},{",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "개별 페이지의 조회 및 편집 권한을 제어합니다. 엔터프라이즈 라이선스에서 이용 가능합니다.",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "개별 페이지의 조회 및 편집 권한을 제어합니다. 엔터프라이즈 라이선스에서 이용 가능합니다.",
|
||||||
"Enable public sharing": "공유 활성화",
|
"Enable public sharing": "공유 활성화",
|
||||||
@@ -621,7 +635,9 @@
|
|||||||
"Generative AI (Ask AI)": "생성 AI (Ask AI)",
|
"Generative AI (Ask AI)": "생성 AI (Ask AI)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "편집기에서 AI 구동 콘텐츠 생성을 활성화합니다. 사용자가 텍스트를 생성, 개선, 번역 및 변환할 수 있습니다.",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "편집기에서 AI 구동 콘텐츠 생성을 활성화합니다. 사용자가 텍스트를 생성, 개선, 번역 및 변환할 수 있습니다.",
|
||||||
"Toggle generative AI": "생성 AI 토글",
|
"Toggle generative AI": "생성 AI 토글",
|
||||||
"Enterprise feature": "엔터프라이즈 기능",
|
"Upgrade your plan": "요금제를 업그레이드하세요",
|
||||||
|
"Available with a paid license": "유료 라이선스에서만 사용 가능합니다",
|
||||||
|
"Upgrade your license tier.": "라이선스 등급을 업그레이드하세요.",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI는 Docmost 엔터프라이즈 에디션에서만 제공됩니다. sales@docmost.com으로 문의하세요.",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI는 Docmost 엔터프라이즈 에디션에서만 제공됩니다. sales@docmost.com으로 문의하세요.",
|
||||||
"AI & MCP": "AI 및 MCP",
|
"AI & MCP": "AI 및 MCP",
|
||||||
"AI": "AI",
|
"AI": "AI",
|
||||||
@@ -629,17 +645,15 @@
|
|||||||
"Model Context Protocol (MCP)": "모델 컨텍스트 프로토콜(MCP)",
|
"Model Context Protocol (MCP)": "모델 컨텍스트 프로토콜(MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "AI 어시스턴트와 도구가 워크스페이스 콘텐츠와 상호작용할 수 있도록 MCP 서버를 활성화하세요.",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "AI 어시스턴트와 도구가 워크스페이스 콘텐츠와 상호작용할 수 있도록 MCP 서버를 활성화하세요.",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP는 Docmost 엔터프라이즈 에디션에서만 제공됩니다. sales@docmost.com으로 문의하세요.",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP는 Docmost 엔터프라이즈 에디션에서만 제공됩니다. sales@docmost.com으로 문의하세요.",
|
||||||
"MCP documentation": "MCP 문서",
|
|
||||||
"MCP Server URL": "MCP 서버 URL",
|
"MCP Server URL": "MCP 서버 URL",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "인증을 위해 API 키를 사용하세요. API 키는 계정 설정에서 관리할 수 있습니다.",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "인증을 위해 API 키를 사용하세요. API 키는 계정 설정에서 관리할 수 있습니다.",
|
||||||
"Supported tools": "지원되는 도구",
|
"Supported tools": "지원되는 도구",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "워크스페이스에 MCP가 활성화되어 있습니다. AI 어시스턴트를 연결하려면 API 키를 사용하세요.",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "워크스페이스에 MCP가 활성화되어 있습니다. AI 어시스턴트를 연결하려면 API 키를 사용하세요.",
|
||||||
"MCP server URL:": "MCP 서버 URL:",
|
"MCP server URL:": "MCP 서버 URL:",
|
||||||
"Learn more": "자세히 알아보기",
|
"Learn more": "자세히 알아보기",
|
||||||
"View the": "다음을",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "워크스페이스의 모든 사용자를 위한 API 키를 관리하세요. 사용 방법은 <anchor>API 문서</anchor>를 참고하세요.",
|
||||||
"for usage details.": "에서 사용 방법을 확인하세요.",
|
"View the <anchor>API documentation</anchor> for usage details.": "사용 방법은 <anchor>API 문서</anchor>를 참고하세요.",
|
||||||
"for setup instructions.": "에서 설정 지침을 확인하세요.",
|
"View the <anchor>MCP documentation</anchor>.": "<anchor>MCP 문서</anchor>를 확인하세요.",
|
||||||
"API documentation": "API 문서",
|
|
||||||
"Sources": "출처",
|
"Sources": "출처",
|
||||||
"AI Answers not available for attachments": "첨부 파일에 대해 AI 답변을 사용할 수 없습니다",
|
"AI Answers not available for attachments": "첨부 파일에 대해 AI 답변을 사용할 수 없습니다",
|
||||||
"No answer available": "답변을 제공할 수 없습니다",
|
"No answer available": "답변을 제공할 수 없습니다",
|
||||||
@@ -654,12 +668,30 @@
|
|||||||
"Mark all as read": "모두 읽음으로 표시",
|
"Mark all as read": "모두 읽음으로 표시",
|
||||||
"Mark as read": "읽음으로 표시",
|
"Mark as read": "읽음으로 표시",
|
||||||
"More options": "추가 옵션",
|
"More options": "추가 옵션",
|
||||||
"mentioned you in a comment": "댓글에서 당신을 언급했습니다",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold>님이 댓글에서 당신을 언급했습니다",
|
||||||
"commented on a page": "페이지에 댓글을 달았습니다",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold>님이 페이지에 댓글을 남겼습니다",
|
||||||
"resolved a comment": "댓글을 해결했습니다",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold>님이 댓글을 해결했습니다",
|
||||||
"mentioned you on a page": "페이지에서 당신을 언급했습니다",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>님이 페이지에서 당신을 언급했습니다",
|
||||||
"gave you edit access to a page": "페이지 편집 권한을 부여했습니다",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>님이 페이지 편집 권한을 부여했습니다",
|
||||||
"gave you view access to a page": "페이지 보기 권한을 부여했습니다",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>님이 페이지 조회 권한을 부여했습니다",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold>님이 페이지를 업데이트했습니다.",
|
||||||
|
"Watch page": "페이지 구독",
|
||||||
|
"Stop watching": "구독 취소",
|
||||||
|
"Email notifications": "이메일 알림",
|
||||||
|
"Page updates": "페이지 업데이트",
|
||||||
|
"Get notified when pages you watch are updated.": "구독한 페이지가 업데이트될 때 알림을 받으세요.",
|
||||||
|
"Page mentions": "페이지 언급",
|
||||||
|
"Get notified when someone mentions you on a page.": "누군가가 페이지에서 당신을 언급하면 알림을 받으세요.",
|
||||||
|
"Comment mentions": "댓글 언급",
|
||||||
|
"Get notified when someone mentions you in a comment.": "누군가가 댓글에서 당신을 언급하면 알림을 받으세요.",
|
||||||
|
"New comments": "새 댓글",
|
||||||
|
"Get notified about new comments on threads you participate in.": "참여 중인 스레드에 새 댓글이 달리면 알림을 받으세요.",
|
||||||
|
"Resolved comments": "해결된 댓글",
|
||||||
|
"Get notified when your comment is resolved.": "내 댓글이 해결되었을 때 알림을 받으세요.",
|
||||||
|
"You are now watching this page": "이제 이 페이지를 주시합니다.",
|
||||||
|
"You are no longer watching this page": "더 이상 이 페이지를 주시하지 않습니다.",
|
||||||
|
"Direct": "직접",
|
||||||
|
"Updates": "업데이트",
|
||||||
"Today": "오늘",
|
"Today": "오늘",
|
||||||
"Yesterday": "어제",
|
"Yesterday": "어제",
|
||||||
"This week": "이번 주",
|
"This week": "이번 주",
|
||||||
@@ -693,5 +725,31 @@
|
|||||||
"Failed to update trash retention": "휴지통 보관 기간 업데이트에 실패했습니다.",
|
"Failed to update trash retention": "휴지통 보관 기간 업데이트에 실패했습니다.",
|
||||||
"Removed page restriction": "페이지 제한이 제거됨",
|
"Removed page restriction": "페이지 제한이 제거됨",
|
||||||
"Added page permission": "페이지 권한이 추가됨",
|
"Added page permission": "페이지 권한이 추가됨",
|
||||||
"Removed page permission": "페이지 권한이 제거됨"
|
"Removed page permission": "페이지 권한이 제거됨",
|
||||||
|
"Verifying your email": "이메일 인증 중",
|
||||||
|
"Please wait...": "잠시만 기다려 주세요...",
|
||||||
|
"Verification failed. The link may have expired.": "인증에 실패했습니다. 링크가 만료되었을 수 있습니다.",
|
||||||
|
"Check your email": "이메일을 확인하세요",
|
||||||
|
"We sent a verification link to {{email}}.": "{{email}} 주소로 인증 링크를 보냈습니다.",
|
||||||
|
"We sent a verification link to your email.": "이메일로 인증 링크를 보냈습니다.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "이메일의 링크를 클릭하여 인증하고 워크스페이스에 접속하세요.",
|
||||||
|
"Resend verification email": "인증 이메일 재전송",
|
||||||
|
"Verification email sent. Please check your inbox.": "인증 이메일이 전송되었습니다. 받은 편지함을 확인하세요.",
|
||||||
|
"Failed to resend verification email. Please try again.": "인증 이메일 재전송에 실패했습니다. 다시 시도해 주세요.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "연결된 워크스페이스 목록이 포함된 이메일을 보내드렸습니다.",
|
||||||
|
"Load more": "더 불러오기",
|
||||||
|
"Log out of all devices": "모든 기기에서 로그아웃",
|
||||||
|
"Log out of all sessions except this device": "이 기기를 제외한 모든 세션에서 로그아웃",
|
||||||
|
"This Device": "이 기기",
|
||||||
|
"Unknown device": "알 수 없는 기기",
|
||||||
|
"No active sessions": "활성 세션이 없습니다",
|
||||||
|
"Session revoked": "세션이 해제되었습니다",
|
||||||
|
"All other sessions revoked": "나머지 모든 세션이 해제되었습니다",
|
||||||
|
"Last used": "최근 사용",
|
||||||
|
"Created": "생성됨",
|
||||||
|
"Rename": "이름 바꾸기",
|
||||||
|
"Publish": "게시",
|
||||||
|
"Security": "보안",
|
||||||
|
"Enforce SSO": "SSO 강제 적용",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "강제 적용 시, 멤버는 이메일과 비밀번호로는 로그인할 수 없습니다."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,6 +289,11 @@
|
|||||||
"Save & Exit": "Opslaan & Afsluiten",
|
"Save & Exit": "Opslaan & Afsluiten",
|
||||||
"Double-click to edit Excalidraw diagram": "Dubbelklik om Excalidraw diagram te bewerken",
|
"Double-click to edit Excalidraw diagram": "Dubbelklik om Excalidraw diagram te bewerken",
|
||||||
"Paste link": "Link plakken",
|
"Paste link": "Link plakken",
|
||||||
|
"Paste link or search pages": "Plak link of zoek pagina's",
|
||||||
|
"Link to web page": "Link naar webpagina",
|
||||||
|
"Recents": "Recent",
|
||||||
|
"Page or URL": "Pagina of URL",
|
||||||
|
"Link title": "Kop van de link",
|
||||||
"Edit link": "Link bewerken",
|
"Edit link": "Link bewerken",
|
||||||
"Remove link": "Link verwijderen",
|
"Remove link": "Link verwijderen",
|
||||||
"Add link": "Link toevoegen",
|
"Add link": "Link toevoegen",
|
||||||
@@ -336,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Horizontale lijn invoegen",
|
"Insert horizontal rule divider": "Horizontale lijn invoegen",
|
||||||
"Upload any image from your device.": "Upload een afbeelding vanaf uw apparaat.",
|
"Upload any image from your device.": "Upload een afbeelding vanaf uw apparaat.",
|
||||||
"Upload any video from your device.": "Upload een video vanaf uw apparaat.",
|
"Upload any video from your device.": "Upload een video vanaf uw apparaat.",
|
||||||
|
"Upload any audio from your device.": "Upload een audio vanaf uw apparaat.",
|
||||||
"Upload any file from your device.": "Upload een bestand vanaf uw apparaat.",
|
"Upload any file from your device.": "Upload een bestand vanaf uw apparaat.",
|
||||||
"Uploading {{name}}": "Uploaden {{name}}",
|
"Uploading {{name}}": "Uploaden {{name}}",
|
||||||
"Uploading file": "Bestand uploaden",
|
"Uploading file": "Bestand uploaden",
|
||||||
@@ -346,6 +352,12 @@
|
|||||||
"Divider": "Scheidingslijn",
|
"Divider": "Scheidingslijn",
|
||||||
"Quote": "Quote",
|
"Quote": "Quote",
|
||||||
"Image": "Afbeelding",
|
"Image": "Afbeelding",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "PDF insluiten",
|
||||||
|
"Upload and embed a PDF file.": "Upload en sluit een PDF-bestand in.",
|
||||||
|
"Embed as PDF": "Insluiten als PDF",
|
||||||
|
"Failed to load PDF": "Laden van PDF mislukt",
|
||||||
|
"Convert to attachment": "Converteren naar bijlage",
|
||||||
"File attachment": "Bestand bijlage",
|
"File attachment": "Bestand bijlage",
|
||||||
"Toggle block": "Schakel blok in/uit",
|
"Toggle block": "Schakel blok in/uit",
|
||||||
"Callout": "Opmerking",
|
"Callout": "Opmerking",
|
||||||
@@ -437,9 +449,11 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Voorkom dat leden pagina's openbaar delen.",
|
"Prevent members from sharing pages publicly.": "Voorkom dat leden pagina's openbaar delen.",
|
||||||
"Toggle public sharing": "Wissel openbaar delen",
|
"Toggle public sharing": "Wissel openbaar delen",
|
||||||
"Toggle space public sharing": "Wissel openbaar delen van ruimte",
|
"Toggle space public sharing": "Wissel openbaar delen van ruimte",
|
||||||
|
"Allow viewers to comment": "Toestaan dat kijkers reageren",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Sta kijkers toe om reacties toe te voegen op pagina\u0019s in deze ruimte.",
|
||||||
|
"Toggle viewer comments": "Reacties van kijkers in- of uitschakelen",
|
||||||
"Public sharing is disabled at the workspace level": "Openbaar delen is uitgeschakeld op werkruimteniveau",
|
"Public sharing is disabled at the workspace level": "Openbaar delen is uitgeschakeld op werkruimteniveau",
|
||||||
"Prevent pages in this space from being shared publicly.": "Voorkom dat pagina's in deze ruimte openbaar worden gedeeld.",
|
"Prevent pages in this space from being shared publicly.": "Voorkom dat pagina's in deze ruimte openbaar worden gedeeld.",
|
||||||
"Requires an enterprise license": "Vereist een bedrijfslicentie",
|
|
||||||
"Page permissions": "Pagina rechten",
|
"Page permissions": "Pagina rechten",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "Beheer wie individuele pagina's kan bekijken en bewerken. Beschikbaar met een Enterprise-licentie.",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Beheer wie individuele pagina's kan bekijken en bewerken. Beschikbaar met een Enterprise-licentie.",
|
||||||
"Enable public sharing": "Openbaar delen inschakelen",
|
"Enable public sharing": "Openbaar delen inschakelen",
|
||||||
@@ -621,7 +635,9 @@
|
|||||||
"Generative AI (Ask AI)": "Generatieve AI (Vraag het AI)",
|
"Generative AI (Ask AI)": "Generatieve AI (Vraag het AI)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Schakel AI-gestuurde inhoudsgeneratie in de editor in. Hiermee kunnen gebruikers tekst genereren, verbeteren, vertalen en transformeren.",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Schakel AI-gestuurde inhoudsgeneratie in de editor in. Hiermee kunnen gebruikers tekst genereren, verbeteren, vertalen en transformeren.",
|
||||||
"Toggle generative AI": "Generatieve AI schakelen",
|
"Toggle generative AI": "Generatieve AI schakelen",
|
||||||
"Enterprise feature": "Enterprise-functie",
|
"Upgrade your plan": "Upgrade je abonnement",
|
||||||
|
"Available with a paid license": "Beschikbaar met een betaalde licentie",
|
||||||
|
"Upgrade your license tier.": "Upgrade je licentieniveau.",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI is alleen beschikbaar in de Docmost Enterprise-editie. Neem contact op met sales@docmost.com.",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI is alleen beschikbaar in de Docmost Enterprise-editie. Neem contact op met sales@docmost.com.",
|
||||||
"AI & MCP": "AI & MCP",
|
"AI & MCP": "AI & MCP",
|
||||||
"AI": "AI",
|
"AI": "AI",
|
||||||
@@ -629,17 +645,15 @@
|
|||||||
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Schakel de MCP-server in zodat AI-assistenten en tools kunnen interageren met de inhoud van uw werkruimte.",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Schakel de MCP-server in zodat AI-assistenten en tools kunnen interageren met de inhoud van uw werkruimte.",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP is alleen beschikbaar in de Docmost Enterprise-editie. Neem contact op met sales@docmost.com.",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP is alleen beschikbaar in de Docmost Enterprise-editie. Neem contact op met sales@docmost.com.",
|
||||||
"MCP documentation": "MCP-documentatie",
|
|
||||||
"MCP Server URL": "MCP-server-URL",
|
"MCP Server URL": "MCP-server-URL",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "Gebruik uw API-sleutel voor authenticatie. U kunt API-sleutels beheren in uw accountinstellingen.",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Gebruik uw API-sleutel voor authenticatie. U kunt API-sleutels beheren in uw accountinstellingen.",
|
||||||
"Supported tools": "Ondersteunde tools",
|
"Supported tools": "Ondersteunde tools",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "In uw werkruimte is MCP ingeschakeld. Gebruik uw API-sleutel om AI-assistenten te koppelen.",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "In uw werkruimte is MCP ingeschakeld. Gebruik uw API-sleutel om AI-assistenten te koppelen.",
|
||||||
"MCP server URL:": "MCP-server-URL:",
|
"MCP server URL:": "MCP-server-URL:",
|
||||||
"Learn more": "Meer informatie",
|
"Learn more": "Meer informatie",
|
||||||
"View the": "Bekijk de",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Beheer API-sleutels voor alle gebruikers in de werkruimte. Bekijk de <anchor>API-documentatie</anchor> voor gebruiksdetails.",
|
||||||
"for usage details.": "voor details over het gebruik.",
|
"View the <anchor>API documentation</anchor> for usage details.": "Bekijk de <anchor>API-documentatie</anchor> voor gebruiksdetails.",
|
||||||
"for setup instructions.": "voor installatie-instructies.",
|
"View the <anchor>MCP documentation</anchor>.": "Bekijk de <anchor>MCP-documentatie</anchor>.",
|
||||||
"API documentation": "API-documentatie",
|
|
||||||
"Sources": "Bronnen",
|
"Sources": "Bronnen",
|
||||||
"AI Answers not available for attachments": "AI Antwoorden niet beschikbaar voor bijlagen",
|
"AI Answers not available for attachments": "AI Antwoorden niet beschikbaar voor bijlagen",
|
||||||
"No answer available": "Geen antwoord beschikbaar",
|
"No answer available": "Geen antwoord beschikbaar",
|
||||||
@@ -654,12 +668,30 @@
|
|||||||
"Mark all as read": "Markeer alles als gelezen",
|
"Mark all as read": "Markeer alles als gelezen",
|
||||||
"Mark as read": "Markeer als gelezen",
|
"Mark as read": "Markeer als gelezen",
|
||||||
"More options": "Meer opties",
|
"More options": "Meer opties",
|
||||||
"mentioned you in a comment": "noemde je in een reactie",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> noemde je in een reactie",
|
||||||
"commented on a page": "reageerde op een pagina",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> heeft een reactie geplaatst op een pagina",
|
||||||
"resolved a comment": "heeft een opmerking opgelost",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> heeft een reactie opgelost",
|
||||||
"mentioned you on a page": "noemde je op een pagina",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> noemde je op een pagina",
|
||||||
"gave you edit access to a page": "heeft je toegang gegeven om een pagina te bewerken",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> heeft je toegang gegeven om een pagina te bewerken",
|
||||||
"gave you view access to a page": "heeft je toegang gegeven om een pagina te bekijken",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> heeft je toegang gegeven om een pagina te bekijken",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> heeft een pagina bijgewerkt.",
|
||||||
|
"Watch page": "Pagina volgen",
|
||||||
|
"Stop watching": "Volgen stoppen",
|
||||||
|
"Email notifications": "E-mailmeldingen",
|
||||||
|
"Page updates": "Pagina-updates",
|
||||||
|
"Get notified when pages you watch are updated.": "Ontvang een melding wanneer pagina's die je volgt worden bijgewerkt.",
|
||||||
|
"Page mentions": "Pagina-vermeldingen",
|
||||||
|
"Get notified when someone mentions you on a page.": "Ontvang een melding wanneer iemand je noemt op een pagina.",
|
||||||
|
"Comment mentions": "Vermeldingen in opmerkingen",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Ontvang een melding wanneer iemand je noemt in een opmerking.",
|
||||||
|
"New comments": "Nieuwe opmerkingen",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Ontvang meldingen over nieuwe reacties in threads waaraan je deelneemt.",
|
||||||
|
"Resolved comments": "Opgeloste opmerkingen",
|
||||||
|
"Get notified when your comment is resolved.": "Ontvang een melding wanneer je reactie is opgelost.",
|
||||||
|
"You are now watching this page": "Je volgt nu deze pagina",
|
||||||
|
"You are no longer watching this page": "Je volgt deze pagina niet meer",
|
||||||
|
"Direct": "Direct",
|
||||||
|
"Updates": "Updates",
|
||||||
"Today": "Vandaag",
|
"Today": "Vandaag",
|
||||||
"Yesterday": "Gisteren",
|
"Yesterday": "Gisteren",
|
||||||
"This week": "Deze week",
|
"This week": "Deze week",
|
||||||
@@ -693,5 +725,31 @@
|
|||||||
"Failed to update trash retention": "Bijwerken van de bewaartermijn voor de prullenbak is mislukt.",
|
"Failed to update trash retention": "Bijwerken van de bewaartermijn voor de prullenbak is mislukt.",
|
||||||
"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",
|
||||||
|
"Verifying your email": "Je e-mailadres wordt geverifieerd",
|
||||||
|
"Please wait...": "Even geduld...",
|
||||||
|
"Verification failed. The link may have expired.": "Verificatie mislukt. De link is mogelijk verlopen.",
|
||||||
|
"Check your email": "Controleer je e-mail",
|
||||||
|
"We sent a verification link to {{email}}.": "We hebben een verificatielink naar {{email}} gestuurd.",
|
||||||
|
"We sent a verification link to your email.": "We hebben een verificatielink naar je e-mailadres gestuurd.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Klik op de link om je e-mailadres te verifiëren en toegang te krijgen tot je werkruimte.",
|
||||||
|
"Resend verification email": "Verificatie-e-mail opnieuw verzenden",
|
||||||
|
"Verification email sent. Please check your inbox.": "Verificatie-e-mail verzonden. Controleer je inbox.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Het verzenden van de verificatie-e-mail is mislukt. Probeer het opnieuw.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "We hebben je een e-mail gestuurd met je gekoppelde werkruimtes.",
|
||||||
|
"Load more": "Meer laden",
|
||||||
|
"Log out of all devices": "Log uit op alle apparaten",
|
||||||
|
"Log out of all sessions except this device": "Log uit op alle sessies behalve dit apparaat",
|
||||||
|
"This Device": "Dit apparaat",
|
||||||
|
"Unknown device": "Onbekend apparaat",
|
||||||
|
"No active sessions": "Geen actieve sessies",
|
||||||
|
"Session revoked": "Sessie ingetrokken",
|
||||||
|
"All other sessions revoked": "Alle andere sessies ingetrokken",
|
||||||
|
"Last used": "Laatst gebruikt",
|
||||||
|
"Created": "Aangemaakt",
|
||||||
|
"Rename": "Hernoemen",
|
||||||
|
"Publish": "Publiceren",
|
||||||
|
"Security": "Beveiliging",
|
||||||
|
"Enforce SSO": "SSO afdwingen",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Zodra dit is afgedwongen, kunnen leden niet meer inloggen met e-mail en wachtwoord."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,6 +289,11 @@
|
|||||||
"Save & Exit": "Salvar e Sair",
|
"Save & Exit": "Salvar e Sair",
|
||||||
"Double-click to edit Excalidraw diagram": "Clique duas vezes para editar o diagrama Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Clique duas vezes para editar o diagrama Excalidraw",
|
||||||
"Paste link": "Colar link",
|
"Paste link": "Colar link",
|
||||||
|
"Paste link or search pages": "Cole o link ou pesquise páginas",
|
||||||
|
"Link to web page": "Link para página da web",
|
||||||
|
"Recents": "Recentes",
|
||||||
|
"Page or URL": "Página ou URL",
|
||||||
|
"Link title": "Título do link",
|
||||||
"Edit link": "Editar link",
|
"Edit link": "Editar link",
|
||||||
"Remove link": "Remover link",
|
"Remove link": "Remover link",
|
||||||
"Add link": "Adicionar link",
|
"Add link": "Adicionar link",
|
||||||
@@ -336,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Insira um divisor horizontal",
|
"Insert horizontal rule divider": "Insira um divisor horizontal",
|
||||||
"Upload any image from your device.": "Envie qualquer imagem do seu dispositivo.",
|
"Upload any image from your device.": "Envie qualquer imagem do seu dispositivo.",
|
||||||
"Upload any video from your device.": "Envie qualquer vídeo do seu dispositivo.",
|
"Upload any video from your device.": "Envie qualquer vídeo do seu dispositivo.",
|
||||||
|
"Upload any audio from your device.": "Envie qualquer áudio do seu dispositivo.",
|
||||||
"Upload any file from your device.": "Envie qualquer arquivo do seu dispositivo.",
|
"Upload any file from your device.": "Envie qualquer arquivo do seu dispositivo.",
|
||||||
"Uploading {{name}}": "Enviando {{name}}",
|
"Uploading {{name}}": "Enviando {{name}}",
|
||||||
"Uploading file": "Enviando arquivo",
|
"Uploading file": "Enviando arquivo",
|
||||||
@@ -346,6 +352,12 @@
|
|||||||
"Divider": "Divisor",
|
"Divider": "Divisor",
|
||||||
"Quote": "Citação",
|
"Quote": "Citação",
|
||||||
"Image": "Imagem",
|
"Image": "Imagem",
|
||||||
|
"Audio": "Áudio.",
|
||||||
|
"Embed PDF": "Incorporar PDF",
|
||||||
|
"Upload and embed a PDF file.": "Envie e incorpore um arquivo PDF.",
|
||||||
|
"Embed as PDF": "Incorporar como PDF",
|
||||||
|
"Failed to load PDF": "Falha ao carregar PDF",
|
||||||
|
"Convert to attachment": "Converter em anexo",
|
||||||
"File attachment": "Anexo de arquivo",
|
"File attachment": "Anexo de arquivo",
|
||||||
"Toggle block": "Bloco colapsável",
|
"Toggle block": "Bloco colapsável",
|
||||||
"Callout": "Aviso",
|
"Callout": "Aviso",
|
||||||
@@ -437,9 +449,11 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Impedir que os membros compartilhem páginas publicamente.",
|
"Prevent members from sharing pages publicly.": "Impedir que os membros compartilhem páginas publicamente.",
|
||||||
"Toggle public sharing": "Alternar compartilhamento público",
|
"Toggle public sharing": "Alternar compartilhamento público",
|
||||||
"Toggle space public sharing": "Alternar compartilhamento público do espaço",
|
"Toggle space public sharing": "Alternar compartilhamento público do espaço",
|
||||||
|
"Allow viewers to comment": "Permitir que os visualizadores comentem",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Permitir que os visualizadores adicionem comentários em páginas deste espaço.",
|
||||||
|
"Toggle viewer comments": "Ativar/desativar comentários de visualizadores",
|
||||||
"Public sharing is disabled at the workspace level": "O compartilhamento público está desativado no nível do espaço de trabalho",
|
"Public sharing is disabled at the workspace level": "O compartilhamento público está desativado no nível do espaço de trabalho",
|
||||||
"Prevent pages in this space from being shared publicly.": "Impedir que as páginas neste espaço sejam compartilhadas publicamente.",
|
"Prevent pages in this space from being shared publicly.": "Impedir que as páginas neste espaço sejam compartilhadas publicamente.",
|
||||||
"Requires an enterprise license": "Requer uma licença empresarial",
|
|
||||||
"Page permissions": "Permissões da página},{",
|
"Page permissions": "Permissões da página},{",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "Controle quem pode visualizar e editar páginas individuais. Disponível com licença empresarial.",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Controle quem pode visualizar e editar páginas individuais. Disponível com licença empresarial.",
|
||||||
"Enable public sharing": "Ativar compartilhamento público",
|
"Enable public sharing": "Ativar compartilhamento público",
|
||||||
@@ -621,7 +635,9 @@
|
|||||||
"Generative AI (Ask AI)": "IA generativa (Perguntar à IA)",
|
"Generative AI (Ask AI)": "IA generativa (Perguntar à IA)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Habilitar geração de conteúdo com IA no editor. Permite aos usuários gerar, melhorar, traduzir e transformar texto.",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Habilitar geração de conteúdo com IA no editor. Permite aos usuários gerar, melhorar, traduzir e transformar texto.",
|
||||||
"Toggle generative AI": "Alternar IA generativa",
|
"Toggle generative AI": "Alternar IA generativa",
|
||||||
"Enterprise feature": "Recurso empresarial",
|
"Upgrade your plan": "Faça upgrade do seu plano",
|
||||||
|
"Available with a paid license": "Disponível com uma licença paga",
|
||||||
|
"Upgrade your license tier.": "Faça upgrade do seu nível de licença.",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "A IA está disponível apenas na edição empresarial do Docmost. Contate sales@docmost.com.",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "A IA está disponível apenas na edição empresarial do Docmost. Contate sales@docmost.com.",
|
||||||
"AI & MCP": "IA e MCP",
|
"AI & MCP": "IA e MCP",
|
||||||
"AI": "IA",
|
"AI": "IA",
|
||||||
@@ -629,17 +645,15 @@
|
|||||||
"Model Context Protocol (MCP)": "Protocolo de Contexto de Modelo (MCP)",
|
"Model Context Protocol (MCP)": "Protocolo de Contexto de Modelo (MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Ative o servidor MCP para permitir que assistentes de IA e ferramentas interajam com o conteúdo do seu espaço de trabalho.",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Ative o servidor MCP para permitir que assistentes de IA e ferramentas interajam com o conteúdo do seu espaço de trabalho.",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "O MCP está disponível apenas na edição empresarial do Docmost. Contate sales@docmost.com.",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "O MCP está disponível apenas na edição empresarial do Docmost. Contate sales@docmost.com.",
|
||||||
"MCP documentation": "Documentação do MCP",
|
|
||||||
"MCP Server URL": "URL do servidor MCP",
|
"MCP Server URL": "URL do servidor MCP",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "Use sua chave de API para autenticação. Você pode gerenciar chaves de API nas configurações da sua conta.",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Use sua chave de API para autenticação. Você pode gerenciar chaves de API nas configurações da sua conta.",
|
||||||
"Supported tools": "Ferramentas compatíveis",
|
"Supported tools": "Ferramentas compatíveis",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Seu espaço de trabalho tem MCP habilitado. Use sua chave de API para conectar assistentes de IA.",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Seu espaço de trabalho tem MCP habilitado. Use sua chave de API para conectar assistentes de IA.",
|
||||||
"MCP server URL:": "URL do servidor MCP:",
|
"MCP server URL:": "URL do servidor MCP:",
|
||||||
"Learn more": "Saiba mais",
|
"Learn more": "Saiba mais",
|
||||||
"View the": "Veja o",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Gerencie as chaves de API de todos os usuários do workspace. Veja a <anchor>documentação da API</anchor> para detalhes de uso.",
|
||||||
"for usage details.": "para detalhes de uso.",
|
"View the <anchor>API documentation</anchor> for usage details.": "Veja a <anchor>documentação da API</anchor> para detalhes de uso.",
|
||||||
"for setup instructions.": "para instruções de configuração.",
|
"View the <anchor>MCP documentation</anchor>.": "Veja a <anchor>documentação MCP</anchor>.",
|
||||||
"API documentation": "Documentação da API",
|
|
||||||
"Sources": "Fontes",
|
"Sources": "Fontes",
|
||||||
"AI Answers not available for attachments": "Respostas de IA não disponíveis para anexos",
|
"AI Answers not available for attachments": "Respostas de IA não disponíveis para anexos",
|
||||||
"No answer available": "Nenhuma resposta disponível",
|
"No answer available": "Nenhuma resposta disponível",
|
||||||
@@ -654,12 +668,30 @@
|
|||||||
"Mark all as read": "Marcar todas como lidas",
|
"Mark all as read": "Marcar todas como lidas",
|
||||||
"Mark as read": "Marcar como lida",
|
"Mark as read": "Marcar como lida",
|
||||||
"More options": "Mais opções",
|
"More options": "Mais opções",
|
||||||
"mentioned you in a comment": "mencionou você em um comentário",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> mencionou você em um comentário",
|
||||||
"commented on a page": "comentou em uma página",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> comentou em uma página",
|
||||||
"resolved a comment": "resolveu um comentário",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> resolveu um comentário",
|
||||||
"mentioned you on a page": "mencionou você em uma página",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> mencionou você em uma página",
|
||||||
"gave you edit access to a page": "concedeu a você acesso para editar a página",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> concedeu acesso de edição a uma página",
|
||||||
"gave you view access to a page": "concedeu a você acesso para visualizar a página",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> concedeu acesso de visualização a uma página",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> atualizou uma página.",
|
||||||
|
"Watch page": "Observar página",
|
||||||
|
"Stop watching": "Parar de observar",
|
||||||
|
"Email notifications": "Notificações por e-mail",
|
||||||
|
"Page updates": "Atualizações da página",
|
||||||
|
"Get notified when pages you watch are updated.": "Receba notificações quando as páginas que você observa forem atualizadas.",
|
||||||
|
"Page mentions": "Menções na página",
|
||||||
|
"Get notified when someone mentions you on a page.": "Receba notificações quando alguém mencionar você em uma página.",
|
||||||
|
"Comment mentions": "Menções em comentários",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Receba notificações quando alguém mencionar você em um comentário.",
|
||||||
|
"New comments": "Novos comentários",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Receba notificações sobre novos comentários nas discussões em que você participa.",
|
||||||
|
"Resolved comments": "Comentários resolvidos",
|
||||||
|
"Get notified when your comment is resolved.": "Receba notificações quando seu comentário for resolvido.",
|
||||||
|
"You are now watching this page": "Agora você está observando esta página",
|
||||||
|
"You are no longer watching this page": "Você não está mais observando esta página",
|
||||||
|
"Direct": "Direto",
|
||||||
|
"Updates": "Atualizações",
|
||||||
"Today": "Hoje",
|
"Today": "Hoje",
|
||||||
"Yesterday": "Ontem",
|
"Yesterday": "Ontem",
|
||||||
"This week": "Esta semana",
|
"This week": "Esta semana",
|
||||||
@@ -693,5 +725,31 @@
|
|||||||
"Failed to update trash retention": "Falha ao atualizar a retenção da lixeira",
|
"Failed to update trash retention": "Falha ao atualizar a retenção da lixeira",
|
||||||
"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",
|
||||||
|
"Verifying your email": "Verificando seu e-mail",
|
||||||
|
"Please wait...": "Por favor, aguarde...",
|
||||||
|
"Verification failed. The link may have expired.": "Falha na verificação. O link pode ter expirado.",
|
||||||
|
"Check your email": "Verifique seu e-mail",
|
||||||
|
"We sent a verification link to {{email}}.": "Enviamos um link de verificação para {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Enviamos um link de verificação para seu e-mail.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Clique no link para verificar seu e-mail e acessar seu workspace.",
|
||||||
|
"Resend verification email": "Reenviar e-mail de verificação",
|
||||||
|
"Verification email sent. Please check your inbox.": "E-mail de verificação enviado. Por favor, verifique sua caixa de entrada.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Falha ao reenviar o e-mail de verificação. Por favor, tente novamente.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Enviamos um e-mail para você com seus workspaces associados.",
|
||||||
|
"Load more": "Carregar mais",
|
||||||
|
"Log out of all devices": "Sair de todos os dispositivos",
|
||||||
|
"Log out of all sessions except this device": "Sair de todas as sessões, exceto neste dispositivo",
|
||||||
|
"This Device": "Este dispositivo",
|
||||||
|
"Unknown device": "Dispositivo desconhecido",
|
||||||
|
"No active sessions": "Sem sessões ativas",
|
||||||
|
"Session revoked": "Sessão revogada",
|
||||||
|
"All other sessions revoked": "Todas as outras sessões revogadas",
|
||||||
|
"Last used": "Último uso",
|
||||||
|
"Created": "Criado",
|
||||||
|
"Rename": "Renomear",
|
||||||
|
"Publish": "Publicar",
|
||||||
|
"Security": "Segurança",
|
||||||
|
"Enforce SSO": "Exigir SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Uma vez exigido, os membros não poderão entrar com e-mail e senha."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,6 +289,11 @@
|
|||||||
"Save & Exit": "Сохранить и выйти",
|
"Save & Exit": "Сохранить и выйти",
|
||||||
"Double-click to edit Excalidraw diagram": "Кликните дважды для редактирования диаграммы Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Кликните дважды для редактирования диаграммы Excalidraw",
|
||||||
"Paste link": "Вставить ссылку",
|
"Paste link": "Вставить ссылку",
|
||||||
|
"Paste link or search pages": "Вставьте ссылку или найдите страницы",
|
||||||
|
"Link to web page": "Ссылка на веб-страницу",
|
||||||
|
"Recents": "Недавние",
|
||||||
|
"Page or URL": "Страница или URL",
|
||||||
|
"Link title": "Заголовок ссылки",
|
||||||
"Edit link": "Редактировать ссылку",
|
"Edit link": "Редактировать ссылку",
|
||||||
"Remove link": "Удалить ссылку",
|
"Remove link": "Удалить ссылку",
|
||||||
"Add link": "Добавить ссылку",
|
"Add link": "Добавить ссылку",
|
||||||
@@ -336,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Вставить горизонтальный разделитель",
|
"Insert horizontal rule divider": "Вставить горизонтальный разделитель",
|
||||||
"Upload any image from your device.": "Загрузить любое изображение с вашего устройства.",
|
"Upload any image from your device.": "Загрузить любое изображение с вашего устройства.",
|
||||||
"Upload any video from your device.": "Загрузить любое видео с вашего устройства.",
|
"Upload any video from your device.": "Загрузить любое видео с вашего устройства.",
|
||||||
|
"Upload any audio from your device.": "Загрузите любой аудиофайл с вашего устройства.",
|
||||||
"Upload any file from your device.": "Загрузить любой файл с вашего устройства.",
|
"Upload any file from your device.": "Загрузить любой файл с вашего устройства.",
|
||||||
"Uploading {{name}}": "Загрузка {{name}}",
|
"Uploading {{name}}": "Загрузка {{name}}",
|
||||||
"Uploading file": "Загрузка файла",
|
"Uploading file": "Загрузка файла",
|
||||||
@@ -346,6 +352,12 @@
|
|||||||
"Divider": "Разделитель",
|
"Divider": "Разделитель",
|
||||||
"Quote": "Цитата",
|
"Quote": "Цитата",
|
||||||
"Image": "Изображение",
|
"Image": "Изображение",
|
||||||
|
"Audio": "Аудио.",
|
||||||
|
"Embed PDF": "Встроить PDF",
|
||||||
|
"Upload and embed a PDF file.": "Загрузите и встроите PDF-файл.",
|
||||||
|
"Embed as PDF": "Встроить как PDF",
|
||||||
|
"Failed to load PDF": "Не удалось загрузить PDF",
|
||||||
|
"Convert to attachment": "Преобразовать в вложение",
|
||||||
"File attachment": "Прикрепленный файл",
|
"File attachment": "Прикрепленный файл",
|
||||||
"Toggle block": "Сворачиваемый блок",
|
"Toggle block": "Сворачиваемый блок",
|
||||||
"Callout": "Выноска",
|
"Callout": "Выноска",
|
||||||
@@ -437,9 +449,11 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Запретить участникам делиться страницами публично.",
|
"Prevent members from sharing pages publicly.": "Запретить участникам делиться страницами публично.",
|
||||||
"Toggle public sharing": "Переключить общий доступ",
|
"Toggle public sharing": "Переключить общий доступ",
|
||||||
"Toggle space public sharing": "Переключить общий доступ для пространства",
|
"Toggle space public sharing": "Переключить общий доступ для пространства",
|
||||||
|
"Allow viewers to comment": "Разрешить зрителям комментировать",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Разрешить зрителям добавлять комментарии на страницах в этом пространстве.",
|
||||||
|
"Toggle viewer comments": "Переключить комментарии зрителей",
|
||||||
"Public sharing is disabled at the workspace level": "Общий доступ отключен на уровне рабочего пространства",
|
"Public sharing is disabled at the workspace level": "Общий доступ отключен на уровне рабочего пространства",
|
||||||
"Prevent pages in this space from being shared publicly.": "Запретить делиться страницами в этом пространстве публично.",
|
"Prevent pages in this space from being shared publicly.": "Запретить делиться страницами в этом пространстве публично.",
|
||||||
"Requires an enterprise license": "Требуется корпоративная лицензия",
|
|
||||||
"Page permissions": "Права доступа к странице},{",
|
"Page permissions": "Права доступа к странице},{",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "Контролируйте, кто может просматривать и редактировать отдельные страницы. Доступно при наличии лицензии Enterprise.",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Контролируйте, кто может просматривать и редактировать отдельные страницы. Доступно при наличии лицензии Enterprise.",
|
||||||
"Enable public sharing": "Включить общий доступ",
|
"Enable public sharing": "Включить общий доступ",
|
||||||
@@ -621,7 +635,9 @@
|
|||||||
"Generative AI (Ask AI)": "Генеративный ИИ (Спросить ИИ)",
|
"Generative AI (Ask AI)": "Генеративный ИИ (Спросить ИИ)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Включите создание контента на базе ИИ в редакторе. Позволяет пользователям генерировать, улучшать, переводить и преобразовывать текст.",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Включите создание контента на базе ИИ в редакторе. Позволяет пользователям генерировать, улучшать, переводить и преобразовывать текст.",
|
||||||
"Toggle generative AI": "Переключить генеративный ИИ",
|
"Toggle generative AI": "Переключить генеративный ИИ",
|
||||||
"Enterprise feature": "Корпоративная функция",
|
"Upgrade your plan": "Обновите свой тарифный план",
|
||||||
|
"Available with a paid license": "Доступно с платной лицензией",
|
||||||
|
"Upgrade your license tier.": "Обновите уровень вашей лицензии.",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "ИИ доступен только в корпоративной версии Docmost. Свяжитесь по адресу sales@docmost.com.",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "ИИ доступен только в корпоративной версии Docmost. Свяжитесь по адресу sales@docmost.com.",
|
||||||
"AI & MCP": "ИИ и MCP",
|
"AI & MCP": "ИИ и MCP",
|
||||||
"AI": "ИИ",
|
"AI": "ИИ",
|
||||||
@@ -629,17 +645,15 @@
|
|||||||
"Model Context Protocol (MCP)": "Протокол контекста модели (MCP)",
|
"Model Context Protocol (MCP)": "Протокол контекста модели (MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Включите сервер MCP, чтобы ИИ-ассистенты и инструменты могли взаимодействовать с содержимым вашего рабочего пространства.",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Включите сервер MCP, чтобы ИИ-ассистенты и инструменты могли взаимодействовать с содержимым вашего рабочего пространства.",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP доступен только в корпоративной версии Docmost. Свяжитесь по адресу sales@docmost.com.",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP доступен только в корпоративной версии Docmost. Свяжитесь по адресу sales@docmost.com.",
|
||||||
"MCP documentation": "Документация MCP",
|
|
||||||
"MCP Server URL": "URL сервера MCP",
|
"MCP Server URL": "URL сервера MCP",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "Используйте ваш API-ключ для аутентификации. Управлять API-ключами можно в настройках аккаунта.",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Используйте ваш API-ключ для аутентификации. Управлять API-ключами можно в настройках аккаунта.",
|
||||||
"Supported tools": "Поддерживаемые инструменты",
|
"Supported tools": "Поддерживаемые инструменты",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "В вашем рабочем пространстве включён MCP. Используйте свой API-ключ для подключения ИИ-ассистентов.",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "В вашем рабочем пространстве включён MCP. Используйте свой API-ключ для подключения ИИ-ассистентов.",
|
||||||
"MCP server URL:": "URL сервера MCP:",
|
"MCP server URL:": "URL сервера MCP:",
|
||||||
"Learn more": "Подробнее",
|
"Learn more": "Подробнее",
|
||||||
"View the": "Просмотреть",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Управляйте API-ключами для всех пользователей в рабочем пространстве. Смотрите <anchor>документацию по API</anchor> для получения информации об использовании.",
|
||||||
"for usage details.": "для подробностей использования.",
|
"View the <anchor>API documentation</anchor> for usage details.": "Смотрите <anchor>документацию по API</anchor> для получения информации об использовании.",
|
||||||
"for setup instructions.": "для инструкций по настройке.",
|
"View the <anchor>MCP documentation</anchor>.": "Смотрите <anchor>документацию по MCP</anchor>.",
|
||||||
"API documentation": "Документация API",
|
|
||||||
"Sources": "Источники",
|
"Sources": "Источники",
|
||||||
"AI Answers not available for attachments": "Ответы ИИ недоступны для вложений",
|
"AI Answers not available for attachments": "Ответы ИИ недоступны для вложений",
|
||||||
"No answer available": "Ответ недоступен",
|
"No answer available": "Ответ недоступен",
|
||||||
@@ -654,12 +668,30 @@
|
|||||||
"Mark all as read": "Отметить все как прочитанные",
|
"Mark all as read": "Отметить все как прочитанные",
|
||||||
"Mark as read": "Отметить как прочитанное",
|
"Mark as read": "Отметить как прочитанное",
|
||||||
"More options": "Больше возможностей",
|
"More options": "Больше возможностей",
|
||||||
"mentioned you in a comment": "упомянул вас в комментарии",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> упомянул вас в комментарии",
|
||||||
"commented on a page": "прокомментировал на странице",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> оставил комментарий на странице",
|
||||||
"resolved a comment": "разрешил комментарий",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> решил комментарий",
|
||||||
"mentioned you on a page": "упомянул вас на странице",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> упомянул вас на странице",
|
||||||
"gave you edit access to a page": "предоставил вам доступ на редактирование страницы",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> предоставил вам доступ для редактирования страницы",
|
||||||
"gave you view access to a page": "предоставил вам доступ для просмотра страницы",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> предоставил вам доступ к просмотру страницы",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> обновил страницу.",
|
||||||
|
"Watch page": "Следить за страницей",
|
||||||
|
"Stop watching": "Прекратить отслеживание",
|
||||||
|
"Email notifications": "Уведомления на email",
|
||||||
|
"Page updates": "Обновления страницы",
|
||||||
|
"Get notified when pages you watch are updated.": "Получайте уведомления, когда отслеживаемые вами страницы обновляются.",
|
||||||
|
"Page mentions": "Упоминания на странице",
|
||||||
|
"Get notified when someone mentions you on a page.": "Получайте уведомления, когда кто-то упоминает вас на странице.",
|
||||||
|
"Comment mentions": "Упоминания в комментариях",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Получайте уведомления, когда кто-то упоминает вас в комментарии.",
|
||||||
|
"New comments": "Новые комментарии",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Получайте уведомления о новых комментариях в цепочках, в которых вы участвуете.",
|
||||||
|
"Resolved comments": "Разрешённые комментарии",
|
||||||
|
"Get notified when your comment is resolved.": "Получайте уведомление, когда ваш комментарий разрешён.",
|
||||||
|
"You are now watching this page": "Вы теперь следите за этой страницей",
|
||||||
|
"You are no longer watching this page": "Вы больше не следите за этой страницей",
|
||||||
|
"Direct": "Прямые",
|
||||||
|
"Updates": "Обновления",
|
||||||
"Today": "Сегодня",
|
"Today": "Сегодня",
|
||||||
"Yesterday": "Вчера",
|
"Yesterday": "Вчера",
|
||||||
"This week": "На этой неделе",
|
"This week": "На этой неделе",
|
||||||
@@ -693,5 +725,31 @@
|
|||||||
"Failed to update trash retention": "Не удалось обновить срок хранения корзины",
|
"Failed to update trash retention": "Не удалось обновить срок хранения корзины",
|
||||||
"Removed page restriction": "Ограничение доступа к странице удалено",
|
"Removed page restriction": "Ограничение доступа к странице удалено",
|
||||||
"Added page permission": "Добавлено разрешение доступа к странице",
|
"Added page permission": "Добавлено разрешение доступа к странице",
|
||||||
"Removed page permission": "Удалено разрешение доступа к странице"
|
"Removed page permission": "Удалено разрешение доступа к странице",
|
||||||
|
"Verifying your email": "Проверка вашей электронной почты",
|
||||||
|
"Please wait...": "Пожалуйста, подождите...",
|
||||||
|
"Verification failed. The link may have expired.": "Ошибка проверки. Ссылка могла устареть.",
|
||||||
|
"Check your email": "Проверьте вашу электронную почту",
|
||||||
|
"We sent a verification link to {{email}}.": "Мы отправили ссылку для подтверждения на {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Мы отправили ссылку для подтверждения на вашу электронную почту.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Перейдите по ссылке, чтобы подтвердить электронную почту и получить доступ к рабочему пространству.",
|
||||||
|
"Resend verification email": "Отправить письмо для подтверждения повторно",
|
||||||
|
"Verification email sent. Please check your inbox.": "Письмо для подтверждения отправлено. Пожалуйста, проверьте ваш почтовый ящик.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Не удалось отправить письмо для подтверждения. Пожалуйста, попробуйте снова.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Мы отправили вам электронное письмо с привязанными рабочими пространствами.",
|
||||||
|
"Load more": "Загрузить ещё",
|
||||||
|
"Log out of all devices": "Выйти со всех устройств",
|
||||||
|
"Log out of all sessions except this device": "Выйти из всех сессий, кроме этого устройства",
|
||||||
|
"This Device": "Это устройство",
|
||||||
|
"Unknown device": "Неизвестное устройство",
|
||||||
|
"No active sessions": "Нет активных сессий",
|
||||||
|
"Session revoked": "Сессия отозвана",
|
||||||
|
"All other sessions revoked": "Все другие сессии отозваны",
|
||||||
|
"Last used": "Последнее использование",
|
||||||
|
"Created": "Создано",
|
||||||
|
"Rename": "Переименовать",
|
||||||
|
"Publish": "Опубликовать",
|
||||||
|
"Security": "Безопасность",
|
||||||
|
"Enforce SSO": "Принудительно использовать SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "После включения участники не смогут войти с помощью электронной почты и пароля."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,6 +289,11 @@
|
|||||||
"Save & Exit": "Зберегти та вийти",
|
"Save & Exit": "Зберегти та вийти",
|
||||||
"Double-click to edit Excalidraw diagram": "Клацніть двічі для редагування діаграми Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Клацніть двічі для редагування діаграми Excalidraw",
|
||||||
"Paste link": "Вставити посилання",
|
"Paste link": "Вставити посилання",
|
||||||
|
"Paste link or search pages": "Вставте посилання або знайдіть сторінки",
|
||||||
|
"Link to web page": "Посилання на веб-сторінку",
|
||||||
|
"Recents": "Нещодавні",
|
||||||
|
"Page or URL": "Сторінка або URL",
|
||||||
|
"Link title": "Назва посилання",
|
||||||
"Edit link": "Редагувати посилання",
|
"Edit link": "Редагувати посилання",
|
||||||
"Remove link": "Видалити посилання",
|
"Remove link": "Видалити посилання",
|
||||||
"Add link": "Додати посилання",
|
"Add link": "Додати посилання",
|
||||||
@@ -336,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Вставити горизонтальний роздільник",
|
"Insert horizontal rule divider": "Вставити горизонтальний роздільник",
|
||||||
"Upload any image from your device.": "Завантажити будь-яке зображення з вашого пристрою.",
|
"Upload any image from your device.": "Завантажити будь-яке зображення з вашого пристрою.",
|
||||||
"Upload any video from your device.": "Завантажити будь-яке відео з вашого пристрою.",
|
"Upload any video from your device.": "Завантажити будь-яке відео з вашого пристрою.",
|
||||||
|
"Upload any audio from your device.": "Завантажте будь-який аудіофайл зі свого пристрою.",
|
||||||
"Upload any file from your device.": "Завантажити будь-який файл з вашого пристрою.",
|
"Upload any file from your device.": "Завантажити будь-який файл з вашого пристрою.",
|
||||||
"Uploading {{name}}": "Завантаження {{name}}",
|
"Uploading {{name}}": "Завантаження {{name}}",
|
||||||
"Uploading file": "Завантаження файлу",
|
"Uploading file": "Завантаження файлу",
|
||||||
@@ -346,6 +352,12 @@
|
|||||||
"Divider": "Роздільник",
|
"Divider": "Роздільник",
|
||||||
"Quote": "Цитата",
|
"Quote": "Цитата",
|
||||||
"Image": "Зображення",
|
"Image": "Зображення",
|
||||||
|
"Audio": "Аудіо.",
|
||||||
|
"Embed PDF": "Вбудувати PDF",
|
||||||
|
"Upload and embed a PDF file.": "Завантажте та вбудуйте файл PDF.",
|
||||||
|
"Embed as PDF": "Вбудувати як PDF",
|
||||||
|
"Failed to load PDF": "Не вдалося завантажити PDF",
|
||||||
|
"Convert to attachment": "Перетворити на вкладення",
|
||||||
"File attachment": "Прикріплений файл",
|
"File attachment": "Прикріплений файл",
|
||||||
"Toggle block": "Блок, що згортається",
|
"Toggle block": "Блок, що згортається",
|
||||||
"Callout": "Виноска",
|
"Callout": "Виноска",
|
||||||
@@ -437,9 +449,11 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Перешкодити учасникам публічно ділитися сторінками.",
|
"Prevent members from sharing pages publicly.": "Перешкодити учасникам публічно ділитися сторінками.",
|
||||||
"Toggle public sharing": "Перемикання публічного доступу",
|
"Toggle public sharing": "Перемикання публічного доступу",
|
||||||
"Toggle space public sharing": "Перемикання публічного доступу до просторів",
|
"Toggle space public sharing": "Перемикання публічного доступу до просторів",
|
||||||
|
"Allow viewers to comment": "Дозволити глядачам коментувати",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Дозволити глядачам додавати коментарі на сторінках у цьому просторі.",
|
||||||
|
"Toggle viewer comments": "Увімкнути або вимкнути коментарі глядачів",
|
||||||
"Public sharing is disabled at the workspace level": "Публічний доступ вимкнуто на рівні робочого простору",
|
"Public sharing is disabled at the workspace level": "Публічний доступ вимкнуто на рівні робочого простору",
|
||||||
"Prevent pages in this space from being shared publicly.": "Перешкодити публічному поширенню сторінок у цьому просторі.",
|
"Prevent pages in this space from being shared publicly.": "Перешкодити публічному поширенню сторінок у цьому просторі.",
|
||||||
"Requires an enterprise license": "Потребує корпоративної ліцензії",
|
|
||||||
"Page permissions": "Права доступу до сторінки.",
|
"Page permissions": "Права доступу до сторінки.",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "Керуйте тим, хто може переглядати та редагувати окремі сторінки. Доступно з корпоративною ліцензією.",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Керуйте тим, хто може переглядати та редагувати окремі сторінки. Доступно з корпоративною ліцензією.",
|
||||||
"Enable public sharing": "Увімкнути публічний доступ",
|
"Enable public sharing": "Увімкнути публічний доступ",
|
||||||
@@ -621,7 +635,9 @@
|
|||||||
"Generative AI (Ask AI)": "Генеративний ШІ (Запитати ШІ)",
|
"Generative AI (Ask AI)": "Генеративний ШІ (Запитати ШІ)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Увімкнути генерацію контенту за допомогою ШІ в редакторі. Дозволяє користувачам генерувати, покращувати, перекладати та трансформувати текст.",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Увімкнути генерацію контенту за допомогою ШІ в редакторі. Дозволяє користувачам генерувати, покращувати, перекладати та трансформувати текст.",
|
||||||
"Toggle generative AI": "Переключити генеративний ШІ",
|
"Toggle generative AI": "Переключити генеративний ШІ",
|
||||||
"Enterprise feature": "Функція корпоративної версії",
|
"Upgrade your plan": "Оновіть свій тарифний план",
|
||||||
|
"Available with a paid license": "Доступно за платною ліцензією",
|
||||||
|
"Upgrade your license tier.": "Оновіть рівень своєї ліцензії.",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "ШІ доступний лише в корпоративній редакції Docmost. Зверніться до sales@docmost.com.",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "ШІ доступний лише в корпоративній редакції Docmost. Зверніться до sales@docmost.com.",
|
||||||
"AI & MCP": "ШІ та MCP",
|
"AI & MCP": "ШІ та MCP",
|
||||||
"AI": "ШІ",
|
"AI": "ШІ",
|
||||||
@@ -629,17 +645,15 @@
|
|||||||
"Model Context Protocol (MCP)": "Протокол контексту моделі (MCP)",
|
"Model Context Protocol (MCP)": "Протокол контексту моделі (MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Увімкніть MCP‑сервер, щоб дозволити ШІ‑помічникам та інструментам взаємодіяти з вмістом вашого робочого простору.",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Увімкніть MCP‑сервер, щоб дозволити ШІ‑помічникам та інструментам взаємодіяти з вмістом вашого робочого простору.",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP доступний лише в корпоративній редакції Docmost. Зверніться до sales@docmost.com.",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP доступний лише в корпоративній редакції Docmost. Зверніться до sales@docmost.com.",
|
||||||
"MCP documentation": "Документація MCP",
|
|
||||||
"MCP Server URL": "URL сервера MCP",
|
"MCP Server URL": "URL сервера MCP",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "Використовуйте свій API‑ключ для аутентифікації. Ви можете керувати API‑ключами в налаштуваннях облікового запису.",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Використовуйте свій API‑ключ для аутентифікації. Ви можете керувати API‑ключами в налаштуваннях облікового запису.",
|
||||||
"Supported tools": "Підтримувані інструменти",
|
"Supported tools": "Підтримувані інструменти",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "У вашому робочому просторі MCP увімкнено. Використайте свій API‑ключ, щоб підключити ШІ‑помічників.",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "У вашому робочому просторі MCP увімкнено. Використайте свій API‑ключ, щоб підключити ШІ‑помічників.",
|
||||||
"MCP server URL:": "URL сервера MCP:",
|
"MCP server URL:": "URL сервера MCP:",
|
||||||
"Learn more": "Дізнатися більше",
|
"Learn more": "Дізнатися більше",
|
||||||
"View the": "Переглянути",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Керуйте ключами API для всіх користувачів у робочому просторі. Перегляньте <anchor>документацію API</anchor> для деталей використання.",
|
||||||
"for usage details.": "для відомостей про використання.",
|
"View the <anchor>API documentation</anchor> for usage details.": "Перегляньте <anchor>документацію API</anchor> для деталей використання.",
|
||||||
"for setup instructions.": "для інструкцій з налаштування.",
|
"View the <anchor>MCP documentation</anchor>.": "Перегляньте <anchor>документацію MCP</anchor>.",
|
||||||
"API documentation": "Документація API",
|
|
||||||
"Sources": "Джерела",
|
"Sources": "Джерела",
|
||||||
"AI Answers not available for attachments": "Відповіді ШІ недоступні для вкладень",
|
"AI Answers not available for attachments": "Відповіді ШІ недоступні для вкладень",
|
||||||
"No answer available": "Відповідь недоступна",
|
"No answer available": "Відповідь недоступна",
|
||||||
@@ -654,12 +668,30 @@
|
|||||||
"Mark all as read": "Позначити все як прочитане",
|
"Mark all as read": "Позначити все як прочитане",
|
||||||
"Mark as read": "Позначити як прочитане",
|
"Mark as read": "Позначити як прочитане",
|
||||||
"More options": "Більше опцій",
|
"More options": "Більше опцій",
|
||||||
"mentioned you in a comment": "згадали вас у коментарі",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> згадав вас у коментарі",
|
||||||
"commented on a page": "прокоментували на сторінці",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> залишив коментар на сторінці",
|
||||||
"resolved a comment": "вирішили коментар",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> вирішив коментар",
|
||||||
"mentioned you on a page": "згадали вас на сторінці",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> згадав вас на сторінці",
|
||||||
"gave you edit access to a page": "надав вам доступ для редагування сторінки",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> надав вам доступ до редагування сторінки",
|
||||||
"gave you view access to a page": "надав вам доступ для перегляду сторінки",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> надав вам доступ до перегляду сторінки",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> оновив сторінку.",
|
||||||
|
"Watch page": "Стежити за сторінкою",
|
||||||
|
"Stop watching": "Припинити стежити",
|
||||||
|
"Email notifications": "Сповіщення електронною поштою",
|
||||||
|
"Page updates": "Оновлення сторінки",
|
||||||
|
"Get notified when pages you watch are updated.": "Отримуйте сповіщення, коли сторінки, за якими ви стежите, оновлюються.",
|
||||||
|
"Page mentions": "Згадки на сторінці",
|
||||||
|
"Get notified when someone mentions you on a page.": "Отримуйте сповіщення, коли хтось згадує вас на сторінці.",
|
||||||
|
"Comment mentions": "Згадки у коментарях",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Отримуйте сповіщення, коли хтось згадує вас у коментарі.",
|
||||||
|
"New comments": "Нові коментарі",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Отримуйте сповіщення про нові коментарі у темах, у яких ви берете участь.",
|
||||||
|
"Resolved comments": "Вирішені коментарі",
|
||||||
|
"Get notified when your comment is resolved.": "Отримайте сповіщення, коли ваш коментар вирішено.",
|
||||||
|
"You are now watching this page": "Ви зараз стежите за цією сторінкою",
|
||||||
|
"You are no longer watching this page": "Ви більше не стежите за цією сторінкою",
|
||||||
|
"Direct": "Прямі",
|
||||||
|
"Updates": "Оновлення",
|
||||||
"Today": "Сьогодні",
|
"Today": "Сьогодні",
|
||||||
"Yesterday": "Вчора",
|
"Yesterday": "Вчора",
|
||||||
"This week": "Цього тижня",
|
"This week": "Цього тижня",
|
||||||
@@ -693,5 +725,31 @@
|
|||||||
"Failed to update trash retention": "Не вдалося оновити термін зберігання у кошику",
|
"Failed to update trash retention": "Не вдалося оновити термін зберігання у кошику",
|
||||||
"Removed page restriction": "Обмеження сторінки видалено",
|
"Removed page restriction": "Обмеження сторінки видалено",
|
||||||
"Added page permission": "Додано дозвіл на сторінку",
|
"Added page permission": "Додано дозвіл на сторінку",
|
||||||
"Removed page permission": "Дозвіл на сторінку видалено"
|
"Removed page permission": "Дозвіл на сторінку видалено",
|
||||||
|
"Verifying your email": "Підтвердження вашої електронної пошти",
|
||||||
|
"Please wait...": "Будь ласка, зачекайте...",
|
||||||
|
"Verification failed. The link may have expired.": "Підтвердження не вдалося. Посилання могло втратити чинність.",
|
||||||
|
"Check your email": "Перевірте свою електронну пошту",
|
||||||
|
"We sent a verification link to {{email}}.": "Ми надіслали посилання для підтвердження на {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Ми надіслали посилання для підтвердження на вашу електронну пошту.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Клікніть на посилання, щоб підтвердити електронну пошту та отримати доступ до робочого простору.",
|
||||||
|
"Resend verification email": "Повторно надіслати лист для підтвердження",
|
||||||
|
"Verification email sent. Please check your inbox.": "Лист для підтвердження надіслано. Будь ласка, перевірте свою скриньку.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Не вдалося повторно надіслати лист для підтвердження. Будь ласка, спробуйте ще раз.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Ми надіслали вам лист із переліком пов’язаних робочих просторів.",
|
||||||
|
"Load more": "Завантажити ще",
|
||||||
|
"Log out of all devices": "Вийти з усіх пристроїв",
|
||||||
|
"Log out of all sessions except this device": "Вийти з усіх сесій, окрім цього пристрою",
|
||||||
|
"This Device": "Цей пристрій",
|
||||||
|
"Unknown device": "Невідомий пристрій",
|
||||||
|
"No active sessions": "Немає активних сесій",
|
||||||
|
"Session revoked": "Сесію скасовано",
|
||||||
|
"All other sessions revoked": "Всі інші сесії скасовано",
|
||||||
|
"Last used": "Останнє використання",
|
||||||
|
"Created": "Створено",
|
||||||
|
"Rename": "Перейменувати",
|
||||||
|
"Publish": "Опублікувати",
|
||||||
|
"Security": "Безпека",
|
||||||
|
"Enforce SSO": "Вимагати SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Після активування учасники не зможуть увійти за допомогою електронної пошти та паролю."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,6 +289,11 @@
|
|||||||
"Save & Exit": "保存并退出",
|
"Save & Exit": "保存并退出",
|
||||||
"Double-click to edit Excalidraw diagram": "双击以编辑 Excalidraw 图表",
|
"Double-click to edit Excalidraw diagram": "双击以编辑 Excalidraw 图表",
|
||||||
"Paste link": "粘贴链接",
|
"Paste link": "粘贴链接",
|
||||||
|
"Paste link or search pages": "粘贴链接或搜索页面",
|
||||||
|
"Link to web page": "链接到网页",
|
||||||
|
"Recents": "最近使用",
|
||||||
|
"Page or URL": "页面或网址",
|
||||||
|
"Link title": "链接标题",
|
||||||
"Edit link": "编辑链接",
|
"Edit link": "编辑链接",
|
||||||
"Remove link": "移除链接",
|
"Remove link": "移除链接",
|
||||||
"Add link": "添加链接",
|
"Add link": "添加链接",
|
||||||
@@ -336,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "插入水平分割线",
|
"Insert horizontal rule divider": "插入水平分割线",
|
||||||
"Upload any image from your device.": "从设备上传任何图像",
|
"Upload any image from your device.": "从设备上传任何图像",
|
||||||
"Upload any video from your device.": "从设备上传任何视频",
|
"Upload any video from your device.": "从设备上传任何视频",
|
||||||
|
"Upload any audio from your device.": "从您的设备上传任意音频文件。",
|
||||||
"Upload any file from your device.": "从设备上传任何文件",
|
"Upload any file from your device.": "从设备上传任何文件",
|
||||||
"Uploading {{name}}": "正在上传{{name}}",
|
"Uploading {{name}}": "正在上传{{name}}",
|
||||||
"Uploading file": "正在上传文件",
|
"Uploading file": "正在上传文件",
|
||||||
@@ -346,6 +352,12 @@
|
|||||||
"Divider": "分割线",
|
"Divider": "分割线",
|
||||||
"Quote": "引用",
|
"Quote": "引用",
|
||||||
"Image": "图像",
|
"Image": "图像",
|
||||||
|
"Audio": "音频。",
|
||||||
|
"Embed PDF": "嵌入 PDF",
|
||||||
|
"Upload and embed a PDF file.": "上传并嵌入 PDF 文件。",
|
||||||
|
"Embed as PDF": "作为 PDF 嵌入",
|
||||||
|
"Failed to load PDF": "加载 PDF 失败",
|
||||||
|
"Convert to attachment": "转换为附件",
|
||||||
"File attachment": "文件附件",
|
"File attachment": "文件附件",
|
||||||
"Toggle block": "切换块",
|
"Toggle block": "切换块",
|
||||||
"Callout": "标注块",
|
"Callout": "标注块",
|
||||||
@@ -437,9 +449,11 @@
|
|||||||
"Prevent members from sharing pages publicly.": "阻止成员公开分享页面。",
|
"Prevent members from sharing pages publicly.": "阻止成员公开分享页面。",
|
||||||
"Toggle public sharing": "切换公开分享",
|
"Toggle public sharing": "切换公开分享",
|
||||||
"Toggle space public sharing": "切换空间公开分享",
|
"Toggle space public sharing": "切换空间公开分享",
|
||||||
|
"Allow viewers to comment": "允许观众评论",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "允许观众在此空间的页面上添加评论。",
|
||||||
|
"Toggle viewer comments": "切换观众评论",
|
||||||
"Public sharing is disabled at the workspace level": "公开分享在工作区级别被禁用",
|
"Public sharing is disabled at the workspace level": "公开分享在工作区级别被禁用",
|
||||||
"Prevent pages in this space from being shared publicly.": "阻止此空间中的页面被公开分享。",
|
"Prevent pages in this space from being shared publicly.": "阻止此空间中的页面被公开分享。",
|
||||||
"Requires an enterprise license": "需要企业许可证",
|
|
||||||
"Page permissions": "页面权限},{",
|
"Page permissions": "页面权限},{",
|
||||||
"Control who can view and edit individual pages. Available with an enterprise license.": "控制谁可以查看和编辑单个页面。此功能在企业版许可下可用。",
|
"Control who can view and edit individual pages. Available with an enterprise license.": "控制谁可以查看和编辑单个页面。此功能在企业版许可下可用。",
|
||||||
"Enable public sharing": "启用公开分享",
|
"Enable public sharing": "启用公开分享",
|
||||||
@@ -621,7 +635,9 @@
|
|||||||
"Generative AI (Ask AI)": "生成型AI (询问AI)",
|
"Generative AI (Ask AI)": "生成型AI (询问AI)",
|
||||||
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "在编辑器中启用AI驱动的内容生成。允许用户生成、改进、翻译和转换文本。",
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "在编辑器中启用AI驱动的内容生成。允许用户生成、改进、翻译和转换文本。",
|
||||||
"Toggle generative AI": "切换生成型AI",
|
"Toggle generative AI": "切换生成型AI",
|
||||||
"Enterprise feature": "企业版功能",
|
"Upgrade your plan": "升级您的方案",
|
||||||
|
"Available with a paid license": "需付费许可才可用",
|
||||||
|
"Upgrade your license tier.": "升级您的许可等级。",
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI 仅在 Docmost 企业版中提供。请联系 sales@docmost.com。",
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI 仅在 Docmost 企业版中提供。请联系 sales@docmost.com。",
|
||||||
"AI & MCP": "AI 与 MCP",
|
"AI & MCP": "AI 与 MCP",
|
||||||
"AI": "AI",
|
"AI": "AI",
|
||||||
@@ -629,17 +645,15 @@
|
|||||||
"Model Context Protocol (MCP)": "模型上下文协议(MCP)",
|
"Model Context Protocol (MCP)": "模型上下文协议(MCP)",
|
||||||
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "启用 MCP 服务器以允许 AI 助手和工具与您的工作区内容交互。",
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "启用 MCP 服务器以允许 AI 助手和工具与您的工作区内容交互。",
|
||||||
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP 仅在 Docmost 企业版中提供。请联系 sales@docmost.com。",
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP 仅在 Docmost 企业版中提供。请联系 sales@docmost.com。",
|
||||||
"MCP documentation": "MCP 文档",
|
|
||||||
"MCP Server URL": "MCP 服务器 URL",
|
"MCP Server URL": "MCP 服务器 URL",
|
||||||
"Use your API key for authentication. You can manage API keys in your account settings.": "使用您的 API 密钥进行身份验证。您可以在账户设置中管理 API 密钥。",
|
"Use your API key for authentication. You can manage API keys in your account settings.": "使用您的 API 密钥进行身份验证。您可以在账户设置中管理 API 密钥。",
|
||||||
"Supported tools": "支持的工具",
|
"Supported tools": "支持的工具",
|
||||||
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "您的工作区已启用 MCP。使用您的 API 密钥连接 AI 助手。",
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "您的工作区已启用 MCP。使用您的 API 密钥连接 AI 助手。",
|
||||||
"MCP server URL:": "MCP 服务器 URL:",
|
"MCP server URL:": "MCP 服务器 URL:",
|
||||||
"Learn more": "了解更多",
|
"Learn more": "了解更多",
|
||||||
"View the": "查看",
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "为工作区内所有用户管理 API 密钥。有关使用详情,请查阅<anchor>API 文档</anchor>。",
|
||||||
"for usage details.": "以获取使用详情。",
|
"View the <anchor>API documentation</anchor> for usage details.": "有关使用详情,请查阅<anchor>API 文档</anchor>。",
|
||||||
"for setup instructions.": "以获取设置说明。",
|
"View the <anchor>MCP documentation</anchor>.": "查看<anchor>MCP 文档</anchor>。",
|
||||||
"API documentation": "API 文档",
|
|
||||||
"Sources": "来源",
|
"Sources": "来源",
|
||||||
"AI Answers not available for attachments": "AI答案不适用于附件",
|
"AI Answers not available for attachments": "AI答案不适用于附件",
|
||||||
"No answer available": "无可用答案",
|
"No answer available": "无可用答案",
|
||||||
@@ -654,12 +668,30 @@
|
|||||||
"Mark all as read": "标记所有为已读",
|
"Mark all as read": "标记所有为已读",
|
||||||
"Mark as read": "标记为已读",
|
"Mark as read": "标记为已读",
|
||||||
"More options": "更多选项",
|
"More options": "更多选项",
|
||||||
"mentioned you in a comment": "在评论中提到你",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold>在评论中提到你",
|
||||||
"commented on a page": "在页面上评论",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold>在页面上评论了",
|
||||||
"resolved a comment": "解决了一个评论",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold>已解决一条评论",
|
||||||
"mentioned you on a page": "在页面上提到你",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>在页面上提到你",
|
||||||
"gave you edit access to a page": "已授予你编辑该页面的权限",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>授予你页面编辑权限",
|
||||||
"gave you view access to a page": "已授予你查看该页面的权限",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>授予你页面查看权限",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold>更新了一个页面。",
|
||||||
|
"Watch page": "关注页面",
|
||||||
|
"Stop watching": "取消关注",
|
||||||
|
"Email notifications": "邮件通知",
|
||||||
|
"Page updates": "页面更新",
|
||||||
|
"Get notified when pages you watch are updated.": "当你关注的页面有更新时收到通知。",
|
||||||
|
"Page mentions": "页面提及",
|
||||||
|
"Get notified when someone mentions you on a page.": "当有人在页面上提到你时收到通知。",
|
||||||
|
"Comment mentions": "评论提及",
|
||||||
|
"Get notified when someone mentions you in a comment.": "当有人在评论中提到你时收到通知。",
|
||||||
|
"New comments": "新评论",
|
||||||
|
"Get notified about new comments on threads you participate in.": "当你参与的讨论有新评论时收到通知。",
|
||||||
|
"Resolved comments": "已解决的评论",
|
||||||
|
"Get notified when your comment is resolved.": "当你的评论被解决时收到通知。",
|
||||||
|
"You are now watching this page": "你现在正在关注此页面",
|
||||||
|
"You are no longer watching this page": "你已取消关注此页面",
|
||||||
|
"Direct": "直接",
|
||||||
|
"Updates": "更新",
|
||||||
"Today": "今天",
|
"Today": "今天",
|
||||||
"Yesterday": "昨天",
|
"Yesterday": "昨天",
|
||||||
"This week": "本周",
|
"This week": "本周",
|
||||||
@@ -693,5 +725,31 @@
|
|||||||
"Failed to update trash retention": "更新垃圾箱保留期失败",
|
"Failed to update trash retention": "更新垃圾箱保留期失败",
|
||||||
"Removed page restriction": "已移除页面限制",
|
"Removed page restriction": "已移除页面限制",
|
||||||
"Added page permission": "已添加页面权限",
|
"Added page permission": "已添加页面权限",
|
||||||
"Removed page permission": "已移除页面权限"
|
"Removed page permission": "已移除页面权限",
|
||||||
|
"Verifying your email": "正在验证您的邮箱",
|
||||||
|
"Please wait...": "请稍候……",
|
||||||
|
"Verification failed. The link may have expired.": "验证失败。该链接可能已过期。",
|
||||||
|
"Check your email": "查看您的邮箱",
|
||||||
|
"We sent a verification link to {{email}}.": "我们已向{{email}}发送了一封验证邮件。",
|
||||||
|
"We sent a verification link to your email.": "我们已向您的邮箱发送了一封验证邮件。",
|
||||||
|
"Click the link to verify your email and access your workspace.": "请点击链接以验证邮箱并访问您的工作区。",
|
||||||
|
"Resend verification email": "重新发送验证邮件",
|
||||||
|
"Verification email sent. Please check your inbox.": "验证邮件已发送。请检查您的收件箱。",
|
||||||
|
"Failed to resend verification email. Please try again.": "重新发送验证邮件失败。请重试。",
|
||||||
|
"We've sent you an email with your associated workspaces.": "我们已向您发送包含关联工作区的邮件。",
|
||||||
|
"Load more": "加载更多",
|
||||||
|
"Log out of all devices": "退出所有设备登录",
|
||||||
|
"Log out of all sessions except this device": "除本设备外,退出所有会话",
|
||||||
|
"This Device": "本设备",
|
||||||
|
"Unknown device": "未知设备",
|
||||||
|
"No active sessions": "无活动会话",
|
||||||
|
"Session revoked": "会话已被撤销",
|
||||||
|
"All other sessions revoked": "所有其他会话已被撤销",
|
||||||
|
"Last used": "上次使用",
|
||||||
|
"Created": "创建时间",
|
||||||
|
"Rename": "重命名",
|
||||||
|
"Publish": "发布",
|
||||||
|
"Security": "安全性",
|
||||||
|
"Enforce SSO": "强制启用 SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "一旦强制,成员将无法用邮箱和密码登录。"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ import { useTranslation } from "react-i18next";
|
|||||||
import JoinedWorkspaces from "@/ee/components/joined-workspaces.tsx";
|
import JoinedWorkspaces from "@/ee/components/joined-workspaces.tsx";
|
||||||
import { useJoinedWorkspacesQuery } from "@/ee/cloud/query/cloud-query.ts";
|
import { useJoinedWorkspacesQuery } from "@/ee/cloud/query/cloud-query.ts";
|
||||||
import { findWorkspacesByEmail } from "@/ee/cloud/service/cloud-service.ts";
|
import { findWorkspacesByEmail } from "@/ee/cloud/service/cloud-service.ts";
|
||||||
|
import { AuthLayout } from "@/features/auth/components/auth-layout.tsx";
|
||||||
|
|
||||||
const formSchema = z.object({
|
const formSchema = z.object({
|
||||||
hostname: z.string().min(1, { message: "subdomain is required" }),
|
hostname: z.string().min(1, { message: "subdomain is required" }),
|
||||||
@@ -82,7 +83,7 @@ export function CloudLoginForm() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<AuthLayout>
|
||||||
<Container size={420} className={classes.container}>
|
<Container size={420} className={classes.container}>
|
||||||
<Box p="xl" className={classes.containerBox}>
|
<Box p="xl" className={classes.containerBox}>
|
||||||
<Title order={2} ta="center" fw={500} mb="md">
|
<Title order={2} ta="center" fw={500} mb="md">
|
||||||
@@ -145,12 +146,12 @@ export function CloudLoginForm() {
|
|||||||
</Box>
|
</Box>
|
||||||
</Container>
|
</Container>
|
||||||
|
|
||||||
<Text ta="center">
|
<Text ta="center" mb="xl">
|
||||||
{t("Don't have a workspace?")}{" "}
|
{t("Don't have a workspace?")}{" "}
|
||||||
<Anchor component={Link} to={APP_ROUTE.AUTH.CREATE_WORKSPACE} fw={500}>
|
<Anchor component={Link} to={APP_ROUTE.AUTH.CREATE_WORKSPACE} fw={500}>
|
||||||
{t("Create new workspace")}
|
{t("Create new workspace")}
|
||||||
</Anchor>
|
</Anchor>
|
||||||
</Text>
|
</Text>
|
||||||
</div>
|
</AuthLayout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,4 +16,5 @@ export const Feature = {
|
|||||||
AUDIT_LOGS: 'audit:logs',
|
AUDIT_LOGS: 'audit:logs',
|
||||||
RETENTION: 'retention',
|
RETENTION: 'retention',
|
||||||
SHARING_CONTROLS: 'sharing:controls',
|
SHARING_CONTROLS: 'sharing:controls',
|
||||||
|
VIEWER_COMMENTS: 'comment:viewer',
|
||||||
} as const;
|
} as const;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { z } from "zod/v4";
|
import { z } from "zod/v4";
|
||||||
import React from "react";
|
import React, { useRef } from "react";
|
||||||
import { Button, Group, Modal, Textarea } from "@mantine/core";
|
import { Button, Divider, Group, Modal, Stack, Textarea } from "@mantine/core";
|
||||||
import { useForm } from "@mantine/form";
|
import { useForm } from "@mantine/form";
|
||||||
import { zod4Resolver } from "mantine-form-zod-resolver";
|
import { zod4Resolver } from "mantine-form-zod-resolver";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
@@ -49,6 +49,7 @@ interface ActivateLicenseFormProps {
|
|||||||
export function ActivateLicenseForm({ onClose }: ActivateLicenseFormProps) {
|
export function ActivateLicenseForm({ onClose }: ActivateLicenseFormProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const activateLicenseMutation = useActivateMutation();
|
const activateLicenseMutation = useActivateMutation();
|
||||||
|
const fileInputRef = useRef<HTMLInputElement>(null);
|
||||||
|
|
||||||
const form = useForm<FormValues>({
|
const form = useForm<FormValues>({
|
||||||
validate: zod4Resolver(formSchema),
|
validate: zod4Resolver(formSchema),
|
||||||
@@ -63,29 +64,68 @@ export function ActivateLicenseForm({ onClose }: ActivateLicenseFormProps) {
|
|||||||
onClose?.();
|
onClose?.();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function handleFileUpload(event: React.ChangeEvent<HTMLInputElement>) {
|
||||||
|
const file = event.target.files?.[0];
|
||||||
|
if (!file) return;
|
||||||
|
|
||||||
|
const reader = new FileReader();
|
||||||
|
reader.onload = (e) => {
|
||||||
|
const content = (e.target?.result as string)?.trim();
|
||||||
|
if (content) {
|
||||||
|
form.setFieldValue("licenseKey", content);
|
||||||
|
handleSubmit({ licenseKey: content });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
reader.readAsText(file);
|
||||||
|
|
||||||
|
if (fileInputRef.current) {
|
||||||
|
fileInputRef.current.value = "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<form onSubmit={form.onSubmit(handleSubmit)}>
|
<form onSubmit={form.onSubmit(handleSubmit)}>
|
||||||
<Textarea
|
<input
|
||||||
label={t("License key")}
|
type="file"
|
||||||
description="Enter a valid enterprise license key. Contact sales@docmost.com to purchase one."
|
accept=".txt"
|
||||||
placeholder={t("e.g eyJhb.....")}
|
ref={fileInputRef}
|
||||||
variant="filled"
|
onChange={handleFileUpload}
|
||||||
autosize
|
hidden
|
||||||
minRows={3}
|
|
||||||
maxRows={5}
|
|
||||||
data-autofocus
|
|
||||||
{...form.getInputProps("licenseKey")}
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Group justify="flex-end" mt="md">
|
<Stack gap="xs">
|
||||||
<Button
|
<Textarea
|
||||||
type="submit"
|
label={t("License key")}
|
||||||
disabled={activateLicenseMutation.isPending}
|
placeholder={t("e.g eyJhb.....")}
|
||||||
loading={activateLicenseMutation.isPending}
|
variant="filled"
|
||||||
>
|
autosize
|
||||||
{t("Save")}
|
minRows={3}
|
||||||
</Button>
|
maxRows={5}
|
||||||
</Group>
|
data-autofocus
|
||||||
|
{...form.getInputProps("licenseKey")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Group justify="flex-end">
|
||||||
|
<Button
|
||||||
|
type="submit"
|
||||||
|
disabled={activateLicenseMutation.isPending}
|
||||||
|
loading={activateLicenseMutation.isPending}
|
||||||
|
>
|
||||||
|
{t("Save")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
|
||||||
|
<Divider label={t("Or")} labelPosition="center" />
|
||||||
|
|
||||||
|
<Group justify="center">
|
||||||
|
<Button
|
||||||
|
variant="light"
|
||||||
|
onClick={() => fileInputRef.current?.click()}
|
||||||
|
>
|
||||||
|
{t("Upload license file")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
</Stack>
|
||||||
</form>
|
</form>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -68,7 +68,11 @@ export default function OssDetails() {
|
|||||||
</List>
|
</List>
|
||||||
|
|
||||||
<Text size="sm" c="dimmed">
|
<Text size="sm" c="dimmed">
|
||||||
Contact <a href="mailto:sales@docmost.com?subject=Enterprise%20License%20Inquiry">sales@docmost.com </a> to purchase an enterprise license.
|
Get an enterprise trial key at <a href="https://customers.docmost.com/" target="_blank" rel="noopener noreferrer">customers.docmost.com</a>.
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
Visit <a href="https://docmost.com/pricing" target="_blank" rel="noopener noreferrer">docmost.com/pricing</a> to purchase an enterprise license.
|
||||||
</Text>
|
</Text>
|
||||||
</Stack>
|
</Stack>
|
||||||
</Stack>
|
</Stack>
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ import APP_ROUTE, { getPostLoginRedirect } from "@/lib/app-route";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { z } from "zod/v4";
|
import { z } from "zod/v4";
|
||||||
import { MfaBackupCodeInput } from "./mfa-backup-code-input";
|
import { MfaBackupCodeInput } from "./mfa-backup-code-input";
|
||||||
|
import { AuthLayout } from "@/features/auth/components/auth-layout.tsx";
|
||||||
|
|
||||||
const formSchema = z.object({
|
const formSchema = z.object({
|
||||||
code: z
|
code: z
|
||||||
@@ -66,6 +67,7 @@ export function MfaChallenge() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<AuthLayout>
|
||||||
<Container size={420} className={classes.container}>
|
<Container size={420} className={classes.container}>
|
||||||
<Paper radius="lg" p={40} className={classes.paper}>
|
<Paper radius="lg" p={40} className={classes.paper}>
|
||||||
<Stack align="center" gap="xl">
|
<Stack align="center" gap="xl">
|
||||||
@@ -157,5 +159,6 @@ export function MfaChallenge() {
|
|||||||
</Stack>
|
</Stack>
|
||||||
</Paper>
|
</Paper>
|
||||||
</Container>
|
</Container>
|
||||||
|
</AuthLayout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import { useTranslation } from "react-i18next";
|
|||||||
import { MfaSetupModal } from "@/ee/mfa";
|
import { MfaSetupModal } from "@/ee/mfa";
|
||||||
import APP_ROUTE, { getPostLoginRedirect } from "@/lib/app-route.ts";
|
import APP_ROUTE, { getPostLoginRedirect } from "@/lib/app-route.ts";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
|
import { AuthLayout } from "@/features/auth/components/auth-layout.tsx";
|
||||||
|
|
||||||
export default function MfaSetupRequired() {
|
export default function MfaSetupRequired() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -15,6 +16,7 @@ export default function MfaSetupRequired() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<AuthLayout>
|
||||||
<Container size="sm" py="xl">
|
<Container size="sm" py="xl">
|
||||||
<Paper shadow="sm" p="xl" radius="md" withBorder>
|
<Paper shadow="sm" p="xl" radius="md" withBorder>
|
||||||
<Stack>
|
<Stack>
|
||||||
@@ -44,5 +46,6 @@ export default function MfaSetupRequired() {
|
|||||||
</Stack>
|
</Stack>
|
||||||
</Paper>
|
</Paper>
|
||||||
</Container>
|
</Container>
|
||||||
|
</AuthLayout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -71,7 +71,10 @@ export function PageShareModal({ readOnly }: PageShareModalProps) {
|
|||||||
) : null
|
) : null
|
||||||
}
|
}
|
||||||
variant="default"
|
variant="default"
|
||||||
onClick={open}
|
onClick={() => {
|
||||||
|
setActiveTab(isPubliclyShared ? "publish" : hasPagePermissions ? "access" : "publish");
|
||||||
|
open();
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
{t("Share")}
|
{t("Share")}
|
||||||
</Button>
|
</Button>
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import {
|
|||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import APP_ROUTE from "@/lib/app-route.ts";
|
import APP_ROUTE from "@/lib/app-route.ts";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { AuthLayout } from "@/features/auth/components/auth-layout.tsx";
|
||||||
|
|
||||||
export default function VerifyEmail() {
|
export default function VerifyEmail() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -59,20 +60,23 @@ export default function VerifyEmail() {
|
|||||||
|
|
||||||
if (token) {
|
if (token) {
|
||||||
return (
|
return (
|
||||||
<Container size={420} className={classes.container}>
|
<AuthLayout>
|
||||||
<Box p="xl" className={classes.containerBox}>
|
<Container size={420} className={classes.container}>
|
||||||
<Title order={2} ta="center" fw={500} mb="md">
|
<Box p="xl" className={classes.containerBox}>
|
||||||
{t("Verifying your email")}
|
<Title order={2} ta="center" fw={500} mb="md">
|
||||||
</Title>
|
{t("Verifying your email")}
|
||||||
<Text ta="center" c="dimmed">
|
</Title>
|
||||||
{t("Please wait...")}
|
<Text ta="center" c="dimmed">
|
||||||
</Text>
|
{t("Please wait...")}
|
||||||
</Box>
|
</Text>
|
||||||
</Container>
|
</Box>
|
||||||
|
</Container>
|
||||||
|
</AuthLayout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<AuthLayout>
|
||||||
<Container size={420} className={classes.container}>
|
<Container size={420} className={classes.container}>
|
||||||
<Box p="xl" className={classes.containerBox}>
|
<Box p="xl" className={classes.containerBox}>
|
||||||
<Title order={2} ta="center" fw={500} mb="md">
|
<Title order={2} ta="center" fw={500} mb="md">
|
||||||
@@ -103,5 +107,6 @@ export default function VerifyEmail() {
|
|||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
</Container>
|
</Container>
|
||||||
|
</AuthLayout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,61 @@
|
|||||||
|
import { Group, Text, Switch, Tooltip } from "@mantine/core";
|
||||||
|
import React, { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { ISpace } from "@/features/space/types/space.types.ts";
|
||||||
|
import { useUpdateSpaceMutation } from "@/features/space/queries/space-query.ts";
|
||||||
|
import { useHasFeature } from "@/ee/hooks/use-feature.ts";
|
||||||
|
import { Feature } from "@/ee/features.ts";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label.ts";
|
||||||
|
|
||||||
|
type SpaceViewerCommentsToggleProps = {
|
||||||
|
space: ISpace;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function SpaceViewerCommentsToggle({
|
||||||
|
space,
|
||||||
|
}: SpaceViewerCommentsToggleProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const hasViewerComments = useHasFeature(Feature.VIEWER_COMMENTS);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
const isDisabled = !hasViewerComments;
|
||||||
|
const [checked, setChecked] = useState(
|
||||||
|
space.settings?.comments?.allowViewerComments === true,
|
||||||
|
);
|
||||||
|
const updateSpaceMutation = useUpdateSpaceMutation();
|
||||||
|
|
||||||
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = event.currentTarget.checked;
|
||||||
|
try {
|
||||||
|
await updateSpaceMutation.mutateAsync({
|
||||||
|
spaceId: space.id,
|
||||||
|
allowViewerComments: value,
|
||||||
|
});
|
||||||
|
setChecked(value);
|
||||||
|
} catch {
|
||||||
|
// error handled by mutation
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
||||||
|
<div>
|
||||||
|
<Text size="md">{t("Allow viewers to comment")}</Text>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t("Allow viewers to add comments on pages in this space.")}
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
<Tooltip
|
||||||
|
label={upgradeLabel}
|
||||||
|
disabled={!isDisabled}
|
||||||
|
refProp="rootRef"
|
||||||
|
>
|
||||||
|
<Switch
|
||||||
|
checked={checked}
|
||||||
|
onChange={handleChange}
|
||||||
|
disabled={isDisabled}
|
||||||
|
aria-label={t("Toggle viewer comments")}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { Group, Text } from "@mantine/core";
|
||||||
|
import classes from "./auth.module.css";
|
||||||
|
|
||||||
|
type AuthLayoutProps = {
|
||||||
|
children: React.ReactNode;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function AuthLayout({ children }: AuthLayoutProps) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Group justify="center" gap={8} className={classes.logo}>
|
||||||
|
<img
|
||||||
|
src="/icons/favicon-32x32.png"
|
||||||
|
alt="Docmost"
|
||||||
|
width={22}
|
||||||
|
height={22}
|
||||||
|
/>
|
||||||
|
<Text size="28px" fw={700} style={{ userSelect: "none" }}>
|
||||||
|
Docmost
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
{children}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,12 +1,20 @@
|
|||||||
|
.logo {
|
||||||
|
margin-top: 80px;
|
||||||
|
|
||||||
|
@media (max-width: $mantine-breakpoint-sm) {
|
||||||
|
margin-top: 30px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.container {
|
.container {
|
||||||
box-shadow: rgba(0, 0, 0, 0.07) 0px 2px 45px 4px;
|
box-shadow: rgba(0, 0, 0, 0.07) 0px 2px 45px 4px;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
background: light-dark(var(--mantine-color-body), rgba(0, 0, 0, 0.1));
|
background: light-dark(var(--mantine-color-body), rgba(0, 0, 0, 0.1));
|
||||||
margin-top: 150px;
|
margin-top: 40px;
|
||||||
margin-bottom: 20px;
|
margin-bottom: 20px;
|
||||||
|
|
||||||
@media (max-width: $mantine-breakpoint-sm) {
|
@media (max-width: $mantine-breakpoint-sm) {
|
||||||
margin-top: 50px;
|
margin-top: 20px;
|
||||||
margin-bottom: 20px;
|
margin-bottom: 20px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import { Box, Button, Container, Text, TextInput, Title } from "@mantine/core";
|
|||||||
import classes from "./auth.module.css";
|
import classes from "./auth.module.css";
|
||||||
import { useRedirectIfAuthenticated } from "@/features/auth/hooks/use-redirect-if-authenticated.ts";
|
import { useRedirectIfAuthenticated } from "@/features/auth/hooks/use-redirect-if-authenticated.ts";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { AuthLayout } from "./auth-layout.tsx";
|
||||||
|
|
||||||
const formSchema = z.object({
|
const formSchema = z.object({
|
||||||
email: z
|
email: z
|
||||||
@@ -35,6 +36,7 @@ export function ForgotPasswordForm() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<AuthLayout>
|
||||||
<Container size={420} className={classes.container}>
|
<Container size={420} className={classes.container}>
|
||||||
<Box p="xl" className={classes.containerBox}>
|
<Box p="xl" className={classes.containerBox}>
|
||||||
<Title order={2} ta="center" fw={500} mb="md">
|
<Title order={2} ta="center" fw={500} mb="md">
|
||||||
@@ -69,5 +71,6 @@ export function ForgotPasswordForm() {
|
|||||||
</form>
|
</form>
|
||||||
</Box>
|
</Box>
|
||||||
</Container>
|
</Container>
|
||||||
|
</AuthLayout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ import { useGetInvitationQuery } from "@/features/workspace/queries/workspace-qu
|
|||||||
import { useRedirectIfAuthenticated } from "@/features/auth/hooks/use-redirect-if-authenticated.ts";
|
import { useRedirectIfAuthenticated } from "@/features/auth/hooks/use-redirect-if-authenticated.ts";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import SsoLogin from "@/ee/components/sso-login.tsx";
|
import SsoLogin from "@/ee/components/sso-login.tsx";
|
||||||
|
import { AuthLayout } from "./auth-layout.tsx";
|
||||||
|
|
||||||
const formSchema = z.object({
|
const formSchema = z.object({
|
||||||
name: z.string().trim().min(1),
|
name: z.string().trim().min(1),
|
||||||
@@ -66,6 +67,7 @@ export function InviteSignUpForm() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<AuthLayout>
|
||||||
<Container size={420} className={classes.container}>
|
<Container size={420} className={classes.container}>
|
||||||
<Box p="xl" className={classes.containerBox}>
|
<Box p="xl" className={classes.containerBox}>
|
||||||
<Title order={2} ta="center" fw={500} mb="md">
|
<Title order={2} ta="center" fw={500} mb="md">
|
||||||
@@ -111,5 +113,6 @@ export function InviteSignUpForm() {
|
|||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
</Container>
|
</Container>
|
||||||
|
</AuthLayout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ import SsoLogin from "@/ee/components/sso-login.tsx";
|
|||||||
import { useWorkspacePublicDataQuery } from "@/features/workspace/queries/workspace-query.ts";
|
import { useWorkspacePublicDataQuery } from "@/features/workspace/queries/workspace-query.ts";
|
||||||
import { Error404 } from "@/components/ui/error-404.tsx";
|
import { Error404 } from "@/components/ui/error-404.tsx";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
|
import { AuthLayout } from "./auth-layout.tsx";
|
||||||
|
|
||||||
const formSchema = z.object({
|
const formSchema = z.object({
|
||||||
email: z
|
email: z
|
||||||
@@ -62,52 +63,54 @@ export function LoginForm() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Container size={420} className={classes.container}>
|
<AuthLayout>
|
||||||
<Box p="xl" className={classes.containerBox}>
|
<Container size={420} className={classes.container}>
|
||||||
<Title order={2} ta="center" fw={500} mb="md">
|
<Box p="xl" className={classes.containerBox}>
|
||||||
{t("Login")}
|
<Title order={2} ta="center" fw={500} mb="md">
|
||||||
</Title>
|
{t("Login")}
|
||||||
|
</Title>
|
||||||
|
|
||||||
<SsoLogin />
|
<SsoLogin />
|
||||||
|
|
||||||
{!data?.enforceSso && (
|
{!data?.enforceSso && (
|
||||||
<>
|
<>
|
||||||
<form onSubmit={form.onSubmit(onSubmit)}>
|
<form onSubmit={form.onSubmit(onSubmit)}>
|
||||||
<TextInput
|
<TextInput
|
||||||
id="email"
|
id="email"
|
||||||
type="email"
|
type="email"
|
||||||
label={t("Email")}
|
label={t("Email")}
|
||||||
placeholder="email@example.com"
|
placeholder="email@example.com"
|
||||||
variant="filled"
|
variant="filled"
|
||||||
{...form.getInputProps("email")}
|
{...form.getInputProps("email")}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<PasswordInput
|
<PasswordInput
|
||||||
label={t("Password")}
|
label={t("Password")}
|
||||||
placeholder={t("Your password")}
|
placeholder={t("Your password")}
|
||||||
variant="filled"
|
variant="filled"
|
||||||
mt="md"
|
mt="md"
|
||||||
{...form.getInputProps("password")}
|
{...form.getInputProps("password")}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Group justify="flex-end" mt="sm">
|
<Group justify="flex-end" mt="sm">
|
||||||
<Anchor
|
<Anchor
|
||||||
to={APP_ROUTE.AUTH.FORGOT_PASSWORD}
|
to={APP_ROUTE.AUTH.FORGOT_PASSWORD}
|
||||||
component={Link}
|
component={Link}
|
||||||
underline="never"
|
underline="never"
|
||||||
size="sm"
|
size="sm"
|
||||||
>
|
>
|
||||||
{t("Forgot your password?")}
|
{t("Forgot your password?")}
|
||||||
</Anchor>
|
</Anchor>
|
||||||
</Group>
|
</Group>
|
||||||
|
|
||||||
<Button type="submit" fullWidth mt="md" loading={isLoading}>
|
<Button type="submit" fullWidth mt="md" loading={isLoading}>
|
||||||
{t("Sign In")}
|
{t("Sign In")}
|
||||||
</Button>
|
</Button>
|
||||||
</form>
|
</form>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
</Container>
|
</Container>
|
||||||
|
</AuthLayout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import { Box, Button, Container, PasswordInput, Title } from "@mantine/core";
|
|||||||
import classes from "./auth.module.css";
|
import classes from "./auth.module.css";
|
||||||
import { useRedirectIfAuthenticated } from "@/features/auth/hooks/use-redirect-if-authenticated.ts";
|
import { useRedirectIfAuthenticated } from "@/features/auth/hooks/use-redirect-if-authenticated.ts";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { AuthLayout } from "./auth-layout.tsx";
|
||||||
|
|
||||||
const formSchema = z.object({
|
const formSchema = z.object({
|
||||||
newPassword: z
|
newPassword: z
|
||||||
@@ -38,6 +39,7 @@ export function PasswordResetForm({ resetToken }: PasswordResetFormProps) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<AuthLayout>
|
||||||
<Container size={420} className={classes.container}>
|
<Container size={420} className={classes.container}>
|
||||||
<Box p="xl" className={classes.containerBox}>
|
<Box p="xl" className={classes.containerBox}>
|
||||||
<Title order={2} ta="center" fw={500} mb="md">
|
<Title order={2} ta="center" fw={500} mb="md">
|
||||||
@@ -59,5 +61,6 @@ export function PasswordResetForm({ resetToken }: PasswordResetFormProps) {
|
|||||||
</form>
|
</form>
|
||||||
</Box>
|
</Box>
|
||||||
</Container>
|
</Container>
|
||||||
|
</AuthLayout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ import SsoCloudSignup from "@/ee/components/sso-cloud-signup.tsx";
|
|||||||
import { isCloud } from "@/lib/config.ts";
|
import { isCloud } from "@/lib/config.ts";
|
||||||
import { Link } from "react-router-dom";
|
import { Link } from "react-router-dom";
|
||||||
import APP_ROUTE from "@/lib/app-route.ts";
|
import APP_ROUTE from "@/lib/app-route.ts";
|
||||||
|
import { AuthLayout } from "./auth-layout.tsx";
|
||||||
|
|
||||||
const formSchema = z.object({
|
const formSchema = z.object({
|
||||||
workspaceName: z.string().trim().max(50).optional(),
|
workspaceName: z.string().trim().max(50).optional(),
|
||||||
@@ -50,7 +51,7 @@ export function SetupWorkspaceForm() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<AuthLayout>
|
||||||
<Container size={420} className={classes.container}>
|
<Container size={420} className={classes.container}>
|
||||||
<Box p="xl" className={classes.containerBox}>
|
<Box p="xl" className={classes.containerBox}>
|
||||||
<Title order={2} ta="center" fw={500} mb="md">
|
<Title order={2} ta="center" fw={500} mb="md">
|
||||||
@@ -117,6 +118,6 @@ export function SetupWorkspaceForm() {
|
|||||||
</Anchor>
|
</Anchor>
|
||||||
</Text>
|
</Text>
|
||||||
)}
|
)}
|
||||||
</div>
|
</AuthLayout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,3 +3,15 @@ import { atom } from 'jotai';
|
|||||||
export const showCommentPopupAtom = atom<boolean>(false);
|
export const showCommentPopupAtom = atom<boolean>(false);
|
||||||
export const activeCommentIdAtom = atom<string>('');
|
export const activeCommentIdAtom = atom<string>('');
|
||||||
export const draftCommentIdAtom = atom<string>('');
|
export const draftCommentIdAtom = atom<string>('');
|
||||||
|
|
||||||
|
// Read-only comment state
|
||||||
|
export const showReadOnlyCommentPopupAtom = atom<boolean>(false);
|
||||||
|
export type YjsSelection = {
|
||||||
|
anchor: any;
|
||||||
|
head: any;
|
||||||
|
};
|
||||||
|
export type ReadOnlyCommentData = {
|
||||||
|
yjsSelection: YjsSelection;
|
||||||
|
selectedText: string;
|
||||||
|
};
|
||||||
|
export const readOnlyCommentDataAtom = atom<ReadOnlyCommentData | null>(null);
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ import {
|
|||||||
activeCommentIdAtom,
|
activeCommentIdAtom,
|
||||||
draftCommentIdAtom,
|
draftCommentIdAtom,
|
||||||
showCommentPopupAtom,
|
showCommentPopupAtom,
|
||||||
|
showReadOnlyCommentPopupAtom,
|
||||||
|
readOnlyCommentDataAtom,
|
||||||
} from "@/features/comment/atoms/comment-atom";
|
} from "@/features/comment/atoms/comment-atom";
|
||||||
import CommentEditor from "@/features/comment/components/comment-editor";
|
import CommentEditor from "@/features/comment/components/comment-editor";
|
||||||
import CommentActions from "@/features/comment/components/comment-actions";
|
import CommentActions from "@/features/comment/components/comment-actions";
|
||||||
@@ -19,12 +21,15 @@ import { useTranslation } from "react-i18next";
|
|||||||
interface CommentDialogProps {
|
interface CommentDialogProps {
|
||||||
editor: ReturnType<typeof useEditor>;
|
editor: ReturnType<typeof useEditor>;
|
||||||
pageId: string;
|
pageId: string;
|
||||||
|
readOnly?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
function CommentDialog({ editor, pageId, readOnly }: CommentDialogProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [comment, setComment] = useState("");
|
const [comment, setComment] = useState("");
|
||||||
const [, setShowCommentPopup] = useAtom(showCommentPopupAtom);
|
const [, setShowCommentPopup] = useAtom(showCommentPopupAtom);
|
||||||
|
const [, setShowReadOnlyCommentPopup] = useAtom(showReadOnlyCommentPopupAtom);
|
||||||
|
const [readOnlyCommentData, setReadOnlyCommentData] = useAtom(readOnlyCommentDataAtom);
|
||||||
const [, setActiveCommentId] = useAtom(activeCommentIdAtom);
|
const [, setActiveCommentId] = useAtom(activeCommentIdAtom);
|
||||||
const [draftCommentId, setDraftCommentId] = useAtom(draftCommentIdAtom);
|
const [draftCommentId, setDraftCommentId] = useAtom(draftCommentIdAtom);
|
||||||
const [currentUser] = useAtom(currentUserAtom);
|
const [currentUser] = useAtom(currentUserAtom);
|
||||||
@@ -34,11 +39,17 @@ function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
|||||||
handleDialogClose();
|
handleDialogClose();
|
||||||
});
|
});
|
||||||
const createCommentMutation = useCreateCommentMutation();
|
const createCommentMutation = useCreateCommentMutation();
|
||||||
const { isPending } = createCommentMutation;
|
const isPending = createCommentMutation.isPending;
|
||||||
|
|
||||||
const handleDialogClose = () => {
|
const handleDialogClose = () => {
|
||||||
setShowCommentPopup(false);
|
if (readOnly) {
|
||||||
editor.chain().focus().unsetCommentDecoration().run();
|
setShowReadOnlyCommentPopup(false);
|
||||||
|
// @ts-ignore
|
||||||
|
setReadOnlyCommentData(null);
|
||||||
|
} else {
|
||||||
|
setShowCommentPopup(false);
|
||||||
|
editor.chain().focus().unsetCommentDecoration().run();
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const getSelectedText = () => {
|
const getSelectedText = () => {
|
||||||
@@ -47,6 +58,11 @@ function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleAddComment = async () => {
|
const handleAddComment = async () => {
|
||||||
|
if (readOnly) {
|
||||||
|
await handleAddReadOnlyComment();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const selectedText = getSelectedText();
|
const selectedText = getSelectedText();
|
||||||
const commentData = {
|
const commentData = {
|
||||||
@@ -65,7 +81,6 @@ function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
|||||||
.run();
|
.run();
|
||||||
setActiveCommentId(createdComment.id);
|
setActiveCommentId(createdComment.id);
|
||||||
|
|
||||||
//unselect text to close bubble menu
|
|
||||||
editor.commands.setTextSelection({ from: editor.view.state.selection.from, to: editor.view.state.selection.from });
|
editor.commands.setTextSelection({ from: editor.view.state.selection.from, to: editor.view.state.selection.from });
|
||||||
|
|
||||||
setAsideState({ tab: "comments", isAsideOpen: true });
|
setAsideState({ tab: "comments", isAsideOpen: true });
|
||||||
@@ -85,6 +100,33 @@ function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleAddReadOnlyComment = async () => {
|
||||||
|
if (!readOnlyCommentData) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const createdComment = await createCommentMutation.mutateAsync({
|
||||||
|
pageId,
|
||||||
|
content: JSON.stringify(comment),
|
||||||
|
selection: readOnlyCommentData.selectedText,
|
||||||
|
type: "inline",
|
||||||
|
yjsSelection: readOnlyCommentData.yjsSelection,
|
||||||
|
});
|
||||||
|
|
||||||
|
setActiveCommentId(createdComment.id);
|
||||||
|
setAsideState({ tab: "comments", isAsideOpen: true });
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
const selector = `div[data-comment-id="${createdComment.id}"]`;
|
||||||
|
const commentElement = document.querySelector(selector);
|
||||||
|
commentElement?.scrollIntoView({ behavior: "smooth", block: "center" });
|
||||||
|
}, 400);
|
||||||
|
} finally {
|
||||||
|
setShowReadOnlyCommentPopup(false);
|
||||||
|
// @ts-ignore
|
||||||
|
setReadOnlyCommentData(null);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const handleCommentEditorChange = (newContent: any) => {
|
const handleCommentEditorChange = (newContent: any) => {
|
||||||
setComment(newContent);
|
setComment(newContent);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -44,7 +44,9 @@ function CommentListWithTabs() {
|
|||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const { data: space } = useGetSpaceBySlugQuery(page?.space?.slug);
|
const { data: space } = useGetSpaceBySlugQuery(page?.space?.slug);
|
||||||
|
|
||||||
const canComment = page?.permissions?.canEdit ?? false;
|
const canComment =
|
||||||
|
(page?.permissions?.canEdit ?? false) ||
|
||||||
|
(space?.settings?.comments?.allowViewerComments === true);
|
||||||
|
|
||||||
// Separate active and resolved comments
|
// Separate active and resolved comments
|
||||||
const { activeComments, resolvedComments } = useMemo(() => {
|
const { activeComments, resolvedComments } = useMemo(() => {
|
||||||
@@ -153,7 +155,7 @@ function CommentListWithTabs() {
|
|||||||
)}
|
)}
|
||||||
</Paper>
|
</Paper>
|
||||||
),
|
),
|
||||||
[comments, handleAddReply, isLoading, space?.membership?.role],
|
[comments, handleAddReply, isLoading, space?.membership?.role, canComment],
|
||||||
);
|
);
|
||||||
|
|
||||||
if (isCommentsLoading) {
|
if (isCommentsLoading) {
|
||||||
|
|||||||
@@ -75,7 +75,7 @@ function CommentMenu({
|
|||||||
{isResolved ? t("Re-open comment") : t("Resolve comment")}
|
{isResolved ? t("Re-open comment") : t("Resolve comment")}
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
) : (
|
) : (
|
||||||
<Tooltip label={upgradeLabel} position="left">
|
<Tooltip label={upgradeLabel} position="left" withinPortal={false}>
|
||||||
<Menu.Item disabled leftSection={<IconCircleCheck size={14} />}>
|
<Menu.Item disabled leftSection={<IconCircleCheck size={14} />}>
|
||||||
{t("Resolve comment")}
|
{t("Resolve comment")}
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
|
|||||||
@@ -65,6 +65,11 @@ export function useCreateCommentMutation() {
|
|||||||
) as InfiniteData<IPagination<IComment>> | undefined;
|
) as InfiniteData<IPagination<IComment>> | undefined;
|
||||||
|
|
||||||
if (cache && cache.pages.length > 0) {
|
if (cache && cache.pages.length > 0) {
|
||||||
|
const alreadyExists = cache.pages.some((page) =>
|
||||||
|
page.items.some((c) => c.id === newComment.id),
|
||||||
|
);
|
||||||
|
if (alreadyExists) return;
|
||||||
|
|
||||||
const lastIdx = cache.pages.length - 1;
|
const lastIdx = cache.pages.length - 1;
|
||||||
queryClient.setQueryData(RQ_KEY(newComment.pageId), {
|
queryClient.setQueryData(RQ_KEY(newComment.pageId), {
|
||||||
...cache,
|
...cache,
|
||||||
|
|||||||
@@ -17,6 +17,10 @@ export interface IComment {
|
|||||||
deletedAt?: Date;
|
deletedAt?: Date;
|
||||||
creator: IUser;
|
creator: IUser;
|
||||||
resolvedBy?: IUser;
|
resolvedBy?: IUser;
|
||||||
|
yjsSelection?: {
|
||||||
|
anchor: any;
|
||||||
|
head: any;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ICommentData {
|
export interface ICommentData {
|
||||||
|
|||||||
@@ -1,17 +1,43 @@
|
|||||||
import { NodeViewProps, NodeViewWrapper } from "@tiptap/react";
|
import { NodeViewProps, NodeViewWrapper } from "@tiptap/react";
|
||||||
import { Group, Text, Paper, ActionIcon, Loader } from "@mantine/core";
|
import { Group, Text, Paper, ActionIcon, Loader, Tooltip } from "@mantine/core";
|
||||||
import { getFileUrl } from "@/lib/config.ts";
|
import { getFileUrl } from "@/lib/config.ts";
|
||||||
import { IconDownload, IconPaperclip } from "@tabler/icons-react";
|
import { IconDownload, IconFileTypePdf, IconPaperclip } from "@tabler/icons-react";
|
||||||
import { useHover } from "@mantine/hooks";
|
import { useHover } from "@mantine/hooks";
|
||||||
import { formatBytes } from "@/lib";
|
import { formatBytes } from "@/lib";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { useCallback } from "react";
|
||||||
|
|
||||||
export default function AttachmentView(props: NodeViewProps) {
|
export default function AttachmentView(props: NodeViewProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { node, selected } = props;
|
const { editor, node, getPos, selected } = props;
|
||||||
const { url, name, size } = node.attrs;
|
const { url, name, size, mime, attachmentId, placeholder } = node.attrs;
|
||||||
const { hovered, ref } = useHover();
|
const { hovered, ref } = useHover();
|
||||||
|
|
||||||
|
const isPdf = mime === "application/pdf" || name?.toLowerCase().endsWith(".pdf");
|
||||||
|
|
||||||
|
const handleEmbedAsPdf = useCallback(() => {
|
||||||
|
const pos = getPos();
|
||||||
|
if (pos === undefined || !url) return;
|
||||||
|
|
||||||
|
const nodeSize = node.nodeSize;
|
||||||
|
|
||||||
|
editor
|
||||||
|
.chain()
|
||||||
|
.insertContentAt(
|
||||||
|
{ from: pos, to: pos + nodeSize },
|
||||||
|
{
|
||||||
|
type: "pdf",
|
||||||
|
attrs: {
|
||||||
|
src: url,
|
||||||
|
name,
|
||||||
|
attachmentId,
|
||||||
|
size,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.run();
|
||||||
|
}, [editor, getPos, node, url, name, attachmentId]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<NodeViewWrapper>
|
<NodeViewWrapper>
|
||||||
<Paper withBorder p="4px" ref={ref} data-drag-handle>
|
<Paper withBorder p="4px" ref={ref} data-drag-handle>
|
||||||
@@ -23,14 +49,14 @@ export default function AttachmentView(props: NodeViewProps) {
|
|||||||
h={25}
|
h={25}
|
||||||
>
|
>
|
||||||
<Group wrap="nowrap" gap="sm" style={{ minWidth: 0, flex: 1 }}>
|
<Group wrap="nowrap" gap="sm" style={{ minWidth: 0, flex: 1 }}>
|
||||||
{url ? (
|
{!url && placeholder ? (
|
||||||
<IconPaperclip size={20} style={{ flexShrink: 0 }} />
|
|
||||||
) : (
|
|
||||||
<Loader size={20} style={{ flexShrink: 0 }} />
|
<Loader size={20} style={{ flexShrink: 0 }} />
|
||||||
|
) : (
|
||||||
|
<IconPaperclip size={20} style={{ flexShrink: 0 }} />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Text component="span" size="md" truncate="end" style={{ minWidth: 0 }}>
|
<Text component="span" size="md" truncate="end" style={{ minWidth: 0 }}>
|
||||||
{url ? name : t("Uploading {{name}}", { name })}
|
{!url && placeholder ? t("Uploading {{name}}", { name }) : name}
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
<Text component="span" size="sm" c="dimmed" style={{ flexShrink: 0 }}>
|
<Text component="span" size="sm" c="dimmed" style={{ flexShrink: 0 }}>
|
||||||
@@ -39,11 +65,20 @@ export default function AttachmentView(props: NodeViewProps) {
|
|||||||
</Group>
|
</Group>
|
||||||
|
|
||||||
{url && (selected || hovered) && (
|
{url && (selected || hovered) && (
|
||||||
<a href={getFileUrl(url)} target="_blank">
|
<Group gap={4} wrap="nowrap" style={{ flexShrink: 0 }}>
|
||||||
<ActionIcon variant="default" aria-label="download file">
|
{isPdf && editor.isEditable && (
|
||||||
<IconDownload size={18} />
|
<Tooltip label={t("Embed as PDF")} position="top" withinPortal={false}>
|
||||||
</ActionIcon>
|
<ActionIcon variant="default" aria-label={t("Embed as PDF")} onClick={handleEmbedAsPdf}>
|
||||||
</a>
|
<IconFileTypePdf size={18} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
)}
|
||||||
|
<a href={getFileUrl(url)} target="_blank">
|
||||||
|
<ActionIcon variant="default" aria-label="download file">
|
||||||
|
<IconDownload size={18} />
|
||||||
|
</ActionIcon>
|
||||||
|
</a>
|
||||||
|
</Group>
|
||||||
)}
|
)}
|
||||||
</Group>
|
</Group>
|
||||||
</Paper>
|
</Paper>
|
||||||
|
|||||||
@@ -0,0 +1,123 @@
|
|||||||
|
import { BubbleMenu as BaseBubbleMenu } from "@tiptap/react/menus";
|
||||||
|
import { findParentNode, posToDOMRect, useEditorState } from "@tiptap/react";
|
||||||
|
import { useCallback } from "react";
|
||||||
|
import { Node as PMNode } from "@tiptap/pm/model";
|
||||||
|
import {
|
||||||
|
EditorMenuProps,
|
||||||
|
ShouldShowProps,
|
||||||
|
} from "@/features/editor/components/table/types/types.ts";
|
||||||
|
import { ActionIcon, Tooltip } from "@mantine/core";
|
||||||
|
import {
|
||||||
|
IconDownload,
|
||||||
|
IconTrash,
|
||||||
|
} from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { getFileUrl } from "@/lib/config.ts";
|
||||||
|
import classes from "../common/toolbar-menu.module.css";
|
||||||
|
|
||||||
|
export function AudioMenu({ editor }: EditorMenuProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const editorState = useEditorState({
|
||||||
|
editor,
|
||||||
|
selector: (ctx) => {
|
||||||
|
if (!ctx.editor) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const audioAttrs = ctx.editor.getAttributes("audio");
|
||||||
|
|
||||||
|
return {
|
||||||
|
isAudio: ctx.editor.isActive("audio"),
|
||||||
|
src: audioAttrs?.src || null,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const shouldShow = useCallback(
|
||||||
|
({ state }: ShouldShowProps) => {
|
||||||
|
if (!state) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return editor.isActive("audio") && editor.getAttributes("audio").src;
|
||||||
|
},
|
||||||
|
[editor],
|
||||||
|
);
|
||||||
|
|
||||||
|
const getReferencedVirtualElement = useCallback(() => {
|
||||||
|
if (!editor) return;
|
||||||
|
const { selection } = editor.state;
|
||||||
|
const predicate = (node: PMNode) => node.type.name === "audio";
|
||||||
|
const parent = findParentNode(predicate)(selection);
|
||||||
|
|
||||||
|
if (parent) {
|
||||||
|
const dom = editor.view.nodeDOM(parent?.pos) as HTMLElement;
|
||||||
|
const domRect = dom.getBoundingClientRect();
|
||||||
|
return {
|
||||||
|
getBoundingClientRect: () => domRect,
|
||||||
|
getClientRects: () => [domRect],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const domRect = posToDOMRect(editor.view, selection.from, selection.to);
|
||||||
|
return {
|
||||||
|
getBoundingClientRect: () => domRect,
|
||||||
|
getClientRects: () => [domRect],
|
||||||
|
};
|
||||||
|
}, [editor]);
|
||||||
|
|
||||||
|
const handleDownload = useCallback(() => {
|
||||||
|
if (!editorState?.src) return;
|
||||||
|
const url = getFileUrl(editorState.src);
|
||||||
|
const a = document.createElement("a");
|
||||||
|
a.href = url;
|
||||||
|
a.download = "";
|
||||||
|
a.click();
|
||||||
|
}, [editorState?.src]);
|
||||||
|
|
||||||
|
const handleDelete = useCallback(() => {
|
||||||
|
editor.commands.deleteSelection();
|
||||||
|
}, [editor]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<BaseBubbleMenu
|
||||||
|
editor={editor}
|
||||||
|
pluginKey={`audio-menu`}
|
||||||
|
updateDelay={0}
|
||||||
|
getReferencedVirtualElement={getReferencedVirtualElement}
|
||||||
|
options={{
|
||||||
|
placement: "top",
|
||||||
|
offset: 8,
|
||||||
|
flip: false,
|
||||||
|
}}
|
||||||
|
shouldShow={shouldShow}
|
||||||
|
>
|
||||||
|
<div className={classes.toolbar}>
|
||||||
|
<Tooltip position="top" label={t("Download")} withinPortal={false}>
|
||||||
|
<ActionIcon
|
||||||
|
onClick={handleDownload}
|
||||||
|
size="lg"
|
||||||
|
aria-label={t("Download")}
|
||||||
|
variant="subtle"
|
||||||
|
>
|
||||||
|
<IconDownload size={18} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
|
||||||
|
<Tooltip position="top" label={t("Delete")} withinPortal={false}>
|
||||||
|
<ActionIcon
|
||||||
|
onClick={handleDelete}
|
||||||
|
size="lg"
|
||||||
|
aria-label={t("Delete")}
|
||||||
|
variant="subtle"
|
||||||
|
>
|
||||||
|
<IconTrash size={18} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
</BaseBubbleMenu>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default AudioMenu;
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
.audioWrapper {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
max-width: 100%;
|
||||||
|
border-radius: 8px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.skeleton {
|
||||||
|
animation: pulse 1.2s ease-in-out infinite;
|
||||||
|
|
||||||
|
@mixin light {
|
||||||
|
background: linear-gradient(-90deg, var(--mantine-color-gray-3) 0%, var(--mantine-color-gray-1) 50%, var(--mantine-color-gray-3) 100%);
|
||||||
|
background-size: 400% 400%;
|
||||||
|
}
|
||||||
|
|
||||||
|
@mixin dark {
|
||||||
|
background: linear-gradient(-90deg, var(--mantine-color-dark-6) 0%, var(--mantine-color-dark-5) 50%, var(--mantine-color-dark-6) 100%);
|
||||||
|
background-size: 400% 400%;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes pulse {
|
||||||
|
0% {
|
||||||
|
background-position: 0% 0%;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
background-position: -135% 0%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.audio {
|
||||||
|
display: block;
|
||||||
|
width: 100%;
|
||||||
|
border-radius: 8px;
|
||||||
|
}
|
||||||
@@ -0,0 +1,68 @@
|
|||||||
|
import { NodeViewProps, NodeViewWrapper } from "@tiptap/react";
|
||||||
|
import { Group, Loader, Text } from "@mantine/core";
|
||||||
|
import { useMemo } from "react";
|
||||||
|
import { getFileUrl } from "@/lib/config.ts";
|
||||||
|
import { isInternalFileUrl } from "@docmost/editor-ext";
|
||||||
|
import classes from "./audio-view.module.css";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
|
export default function AudioView(props: NodeViewProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { editor, node } = props;
|
||||||
|
const { src, placeholder } = node.attrs;
|
||||||
|
|
||||||
|
const safeSrc = useMemo(() => {
|
||||||
|
if (!src || !isInternalFileUrl(src)) return null;
|
||||||
|
return getFileUrl(src);
|
||||||
|
}, [src]);
|
||||||
|
|
||||||
|
const previewSrc = useMemo(() => {
|
||||||
|
editor.storage.shared.audioPreviews =
|
||||||
|
editor.storage.shared.audioPreviews || {};
|
||||||
|
|
||||||
|
if (placeholder?.id) {
|
||||||
|
return editor.storage.shared.audioPreviews[placeholder.id];
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}, [placeholder, editor]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<NodeViewWrapper data-drag-handle>
|
||||||
|
<div className={`${classes.audioWrapper} ${!safeSrc && placeholder ? classes.skeleton : ''}`}>
|
||||||
|
{safeSrc && (
|
||||||
|
<audio
|
||||||
|
className={classes.audio}
|
||||||
|
preload="metadata"
|
||||||
|
controls
|
||||||
|
src={safeSrc}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{!safeSrc && previewSrc && (
|
||||||
|
<Group pos="relative" w="100%">
|
||||||
|
<audio
|
||||||
|
className={classes.audio}
|
||||||
|
preload="metadata"
|
||||||
|
controls
|
||||||
|
src={previewSrc}
|
||||||
|
/>
|
||||||
|
<Loader size={20} pos="absolute" top={6} right={6} />
|
||||||
|
</Group>
|
||||||
|
)}
|
||||||
|
{!safeSrc && !previewSrc && placeholder && (
|
||||||
|
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md" h={54}>
|
||||||
|
<Loader size={20} style={{ flexShrink: 0 }} />
|
||||||
|
<Text component="span" size="sm" truncate="end">
|
||||||
|
{placeholder?.name
|
||||||
|
? t("Uploading {{name}}", { name: placeholder.name })
|
||||||
|
: t("Uploading file")}
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
)}
|
||||||
|
{!safeSrc && !previewSrc && !placeholder && (
|
||||||
|
<audio className={classes.audio} controls />
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</NodeViewWrapper>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
import { handleAudioUpload } from "@docmost/editor-ext";
|
||||||
|
import { uploadFile } from "@/features/page/services/page-service.ts";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { getFileUploadSizeLimit } from "@/lib/config.ts";
|
||||||
|
import { formatBytes } from "@/lib";
|
||||||
|
import i18n from "@/i18n.ts";
|
||||||
|
|
||||||
|
export const uploadAudioAction = handleAudioUpload({
|
||||||
|
onUpload: async (file: File, pageId: string): Promise<any> => {
|
||||||
|
try {
|
||||||
|
return await uploadFile(file, pageId);
|
||||||
|
} catch (err) {
|
||||||
|
notifications.show({
|
||||||
|
color: "red",
|
||||||
|
message: err?.response.data.message,
|
||||||
|
});
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
validateFn: (file) => {
|
||||||
|
if (!file.type.includes("audio/")) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (file.size > getFileUploadSizeLimit()) {
|
||||||
|
notifications.show({
|
||||||
|
color: "red",
|
||||||
|
message: i18n.t("File exceeds the {{limit}} attachment limit", {
|
||||||
|
limit: formatBytes(getFileUploadSizeLimit()),
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
});
|
||||||
@@ -0,0 +1,159 @@
|
|||||||
|
import type { Editor } from "@tiptap/react";
|
||||||
|
import { TextSelection } from "@tiptap/pm/state";
|
||||||
|
import { FC, useCallback, useEffect, useRef, useState } from "react";
|
||||||
|
import { IconMessage } from "@tabler/icons-react";
|
||||||
|
import classes from "./bubble-menu.module.css";
|
||||||
|
import { ActionIcon, Tooltip } from "@mantine/core";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import {
|
||||||
|
showReadOnlyCommentPopupAtom,
|
||||||
|
readOnlyCommentDataAtom,
|
||||||
|
} from "@/features/comment/atoms/comment-atom";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { getRelativeSelection, ySyncPluginKey } from "@tiptap/y-tiptap";
|
||||||
|
|
||||||
|
type ReadonlyBubbleMenuProps = {
|
||||||
|
editor: Editor;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ReadonlyBubbleMenu: FC<ReadonlyBubbleMenuProps> = ({ editor }) => {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [showReadOnlyCommentPopup, setShowReadOnlyCommentPopup] = useAtom(
|
||||||
|
showReadOnlyCommentPopupAtom,
|
||||||
|
);
|
||||||
|
const [, setReadOnlyCommentData] = useAtom(readOnlyCommentDataAtom);
|
||||||
|
const menuRef = useRef<HTMLDivElement>(null);
|
||||||
|
const [visible, setVisible] = useState(false);
|
||||||
|
const [position, setPosition] = useState({ top: 0, left: 0 });
|
||||||
|
const isInteractingRef = useRef(false);
|
||||||
|
|
||||||
|
const updateMenuPosition = useCallback(() => {
|
||||||
|
if (isInteractingRef.current) return;
|
||||||
|
|
||||||
|
const pmSelection = editor.state.selection;
|
||||||
|
if (!(pmSelection instanceof TextSelection) || pmSelection.empty) {
|
||||||
|
setVisible(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const selection = window.getSelection();
|
||||||
|
if (
|
||||||
|
!selection ||
|
||||||
|
selection.isCollapsed ||
|
||||||
|
selection.rangeCount === 0 ||
|
||||||
|
showReadOnlyCommentPopup
|
||||||
|
) {
|
||||||
|
setVisible(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const editorDom = editor.view.dom;
|
||||||
|
if (
|
||||||
|
!editorDom.contains(selection.anchorNode) ||
|
||||||
|
!editorDom.contains(selection.focusNode)
|
||||||
|
) {
|
||||||
|
setVisible(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const range = selection.getRangeAt(0);
|
||||||
|
const rect = range.getBoundingClientRect();
|
||||||
|
|
||||||
|
if (rect.width === 0) {
|
||||||
|
setVisible(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const editorRect = editorDom
|
||||||
|
.closest(".editor-container")
|
||||||
|
?.getBoundingClientRect();
|
||||||
|
if (!editorRect) {
|
||||||
|
setVisible(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setPosition({
|
||||||
|
top: rect.top - editorRect.top - 44,
|
||||||
|
left: rect.left - editorRect.left + rect.width / 2,
|
||||||
|
});
|
||||||
|
setVisible(true);
|
||||||
|
}, [editor, showReadOnlyCommentPopup]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleSelectionChange = () => {
|
||||||
|
updateMenuPosition();
|
||||||
|
};
|
||||||
|
|
||||||
|
document.addEventListener("selectionchange", handleSelectionChange);
|
||||||
|
return () => {
|
||||||
|
document.removeEventListener("selectionchange", handleSelectionChange);
|
||||||
|
};
|
||||||
|
}, [updateMenuPosition]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (showReadOnlyCommentPopup) {
|
||||||
|
setVisible(false);
|
||||||
|
}
|
||||||
|
}, [showReadOnlyCommentPopup]);
|
||||||
|
|
||||||
|
const handleCommentClick = () => {
|
||||||
|
if (!editor) return;
|
||||||
|
|
||||||
|
const view = editor.view;
|
||||||
|
const ystate = ySyncPluginKey.getState(view.state);
|
||||||
|
|
||||||
|
if (ystate?.binding) {
|
||||||
|
const selection = getRelativeSelection(ystate.binding, view.state);
|
||||||
|
const { from, to } = editor.state.selection;
|
||||||
|
const selectedText = editor.state.doc.textBetween(from, to);
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
setReadOnlyCommentData({
|
||||||
|
yjsSelection: {
|
||||||
|
anchor: selection.anchor,
|
||||||
|
head: selection.head,
|
||||||
|
},
|
||||||
|
selectedText,
|
||||||
|
});
|
||||||
|
|
||||||
|
setShowReadOnlyCommentPopup(true);
|
||||||
|
setVisible(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!visible) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
ref={menuRef}
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
top: position.top,
|
||||||
|
left: position.left,
|
||||||
|
transform: "translateX(-50%)",
|
||||||
|
zIndex: 199,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className={classes.bubbleMenu}>
|
||||||
|
<Tooltip label={t("Comment")} withArrow withinPortal={false}>
|
||||||
|
<ActionIcon
|
||||||
|
variant="default"
|
||||||
|
size="lg"
|
||||||
|
radius="6px"
|
||||||
|
aria-label={t("Comment")}
|
||||||
|
style={{ border: "none" }}
|
||||||
|
onMouseDown={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
isInteractingRef.current = true;
|
||||||
|
handleCommentClick();
|
||||||
|
isInteractingRef.current = false;
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<IconMessage size={16} stroke={2} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
import { uploadImageAction } from "@/features/editor/components/image/upload-image-action.tsx";
|
import { uploadImageAction } from "@/features/editor/components/image/upload-image-action.tsx";
|
||||||
import { uploadVideoAction } from "@/features/editor/components/video/upload-video-action.tsx";
|
import { uploadVideoAction } from "@/features/editor/components/video/upload-video-action.tsx";
|
||||||
import { uploadAttachmentAction } from "../attachment/upload-attachment-action";
|
import { uploadAttachmentAction } from "../attachment/upload-attachment-action";
|
||||||
|
import { uploadPdfAction } from "../pdf/upload-pdf-action";
|
||||||
import { createMentionAction } from "@/features/editor/components/link/internal-link-paste.ts";
|
import { createMentionAction } from "@/features/editor/components/link/internal-link-paste.ts";
|
||||||
import { INTERNAL_LINK_REGEX } from "@/lib/constants.ts";
|
import { INTERNAL_LINK_REGEX } from "@/lib/constants.ts";
|
||||||
import { Editor } from "@tiptap/core";
|
import { Editor } from "@tiptap/core";
|
||||||
@@ -12,6 +13,8 @@ import {
|
|||||||
const ATTACHMENT_NODE_TYPES = [
|
const ATTACHMENT_NODE_TYPES = [
|
||||||
"image",
|
"image",
|
||||||
"video",
|
"video",
|
||||||
|
"audio",
|
||||||
|
"pdf",
|
||||||
"attachment",
|
"attachment",
|
||||||
"excalidraw",
|
"excalidraw",
|
||||||
"drawio",
|
"drawio",
|
||||||
@@ -63,6 +66,7 @@ export const handlePaste = (
|
|||||||
const pos = editor.state.selection.from;
|
const pos = editor.state.selection.from;
|
||||||
uploadImageAction(file, editor, pos, pageId);
|
uploadImageAction(file, editor, pos, pageId);
|
||||||
uploadVideoAction(file, editor, pos, pageId);
|
uploadVideoAction(file, editor, pos, pageId);
|
||||||
|
uploadPdfAction(file, editor, pos, pageId);
|
||||||
uploadAttachmentAction(file, editor, pos, pageId);
|
uploadAttachmentAction(file, editor, pos, pageId);
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
@@ -229,6 +233,7 @@ export const handleFileDrop = (
|
|||||||
|
|
||||||
uploadImageAction(file, editor, coordinates?.pos ?? 0 - 1, pageId);
|
uploadImageAction(file, editor, coordinates?.pos ?? 0 - 1, pageId);
|
||||||
uploadVideoAction(file, editor, coordinates?.pos ?? 0 - 1, pageId);
|
uploadVideoAction(file, editor, coordinates?.pos ?? 0 - 1, pageId);
|
||||||
|
uploadPdfAction(file, editor, coordinates?.pos ?? 0 - 1, pageId);
|
||||||
uploadAttachmentAction(file, editor, coordinates?.pos ?? 0 - 1, pageId);
|
uploadAttachmentAction(file, editor, coordinates?.pos ?? 0 - 1, pageId);
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import type { ResizableNodeViewDirection } from "@tiptap/core";
|
|
||||||
import classes from "./node-resize.module.css";
|
import classes from "./node-resize.module.css";
|
||||||
|
import { ResizableNodeViewDirection } from "@docmost/editor-ext";
|
||||||
|
|
||||||
export function createResizeHandle(
|
export function createResizeHandle(
|
||||||
direction: ResizableNodeViewDirection,
|
direction: ResizableNodeViewDirection,
|
||||||
|
|||||||
@@ -20,8 +20,8 @@
|
|||||||
|
|
||||||
.cornerHandle {
|
.cornerHandle {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
width: 36px;
|
width: 24px;
|
||||||
height: 36px;
|
height: 24px;
|
||||||
z-index: 2;
|
z-index: 2;
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
transition: opacity 0.2s ease;
|
transition: opacity 0.2s ease;
|
||||||
@@ -42,13 +42,13 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
&::before {
|
&::before {
|
||||||
width: 28px;
|
width: 20px;
|
||||||
height: 3px;
|
height: 3px;
|
||||||
}
|
}
|
||||||
|
|
||||||
&::after {
|
&::after {
|
||||||
width: 3px;
|
width: 3px;
|
||||||
height: 28px;
|
height: 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
&:hover::before,
|
&:hover::before,
|
||||||
|
|||||||
@@ -74,6 +74,15 @@ export const ResizableWrapper: React.FC<ResizableWrapperProps> = ({
|
|||||||
const constraintsRef = useRef({ minWidth, maxWidth, minHeight, maxHeight });
|
const constraintsRef = useRef({ minWidth, maxWidth, minHeight, maxHeight });
|
||||||
constraintsRef.current = { minWidth, maxWidth, minHeight, maxHeight };
|
constraintsRef.current = { minWidth, maxWidth, minHeight, maxHeight };
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!dragRef.current && wrapperRef.current) {
|
||||||
|
widthRef.current = initialWidth;
|
||||||
|
heightRef.current = initialHeight;
|
||||||
|
wrapperRef.current.style.width = `${initialWidth}px`;
|
||||||
|
wrapperRef.current.style.height = `${initialHeight}px`;
|
||||||
|
}
|
||||||
|
}, [initialWidth, initialHeight]);
|
||||||
|
|
||||||
const handleMouseMove = useRef((e: MouseEvent) => {
|
const handleMouseMove = useRef((e: MouseEvent) => {
|
||||||
const drag = dragRef.current;
|
const drag = dragRef.current;
|
||||||
if (!drag || !wrapperRef.current) return;
|
if (!drag || !wrapperRef.current) return;
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import {
|
|||||||
} from "@/features/editor/components/table/types/types.ts";
|
} from "@/features/editor/components/table/types/types.ts";
|
||||||
import {
|
import {
|
||||||
ActionIcon,
|
ActionIcon,
|
||||||
|
LoadingOverlay,
|
||||||
Modal,
|
Modal,
|
||||||
Text,
|
Text,
|
||||||
Tooltip,
|
Tooltip,
|
||||||
@@ -46,6 +47,8 @@ export function DrawioMenu({ editor }: EditorMenuProps) {
|
|||||||
const computedColorScheme = useComputedColorScheme();
|
const computedColorScheme = useComputedColorScheme();
|
||||||
const isDirtyRef = useRef(false);
|
const isDirtyRef = useRef(false);
|
||||||
const isSavingRef = useRef(false);
|
const isSavingRef = useRef(false);
|
||||||
|
const [isSaving, setIsSaving] = useState(false);
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
|
||||||
const editorState = useEditorState({
|
const editorState = useEditorState({
|
||||||
editor,
|
editor,
|
||||||
@@ -140,6 +143,7 @@ export function DrawioMenu({ editor }: EditorMenuProps) {
|
|||||||
if (isSavingRef.current) return;
|
if (isSavingRef.current) return;
|
||||||
|
|
||||||
isSavingRef.current = true;
|
isSavingRef.current = true;
|
||||||
|
setIsSaving(true);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const svgString = decodeBase64ToSvgString(svgXml);
|
const svgString = decodeBase64ToSvgString(svgXml);
|
||||||
@@ -167,6 +171,7 @@ export function DrawioMenu({ editor }: EditorMenuProps) {
|
|||||||
isDirtyRef.current = false;
|
isDirtyRef.current = false;
|
||||||
} finally {
|
} finally {
|
||||||
isSavingRef.current = false;
|
isSavingRef.current = false;
|
||||||
|
setIsSaving(false);
|
||||||
}
|
}
|
||||||
}, [editor, editorState?.attachmentId]);
|
}, [editor, editorState?.attachmentId]);
|
||||||
|
|
||||||
@@ -196,6 +201,7 @@ export function DrawioMenu({ editor }: EditorMenuProps) {
|
|||||||
const handleOpen = useCallback(async () => {
|
const handleOpen = useCallback(async () => {
|
||||||
if (!editorState?.src) return;
|
if (!editorState?.src) return;
|
||||||
|
|
||||||
|
setIsLoading(true);
|
||||||
try {
|
try {
|
||||||
const url = getFileUrl(editorState.src);
|
const url = getFileUrl(editorState.src);
|
||||||
const request = await fetch(url, {
|
const request = await fetch(url, {
|
||||||
@@ -213,6 +219,7 @@ export function DrawioMenu({ editor }: EditorMenuProps) {
|
|||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error(err);
|
console.error(err);
|
||||||
} finally {
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
isDirtyRef.current = false;
|
isDirtyRef.current = false;
|
||||||
open();
|
open();
|
||||||
}
|
}
|
||||||
@@ -307,6 +314,7 @@ export function DrawioMenu({ editor }: EditorMenuProps) {
|
|||||||
size="lg"
|
size="lg"
|
||||||
aria-label={t("Edit")}
|
aria-label={t("Edit")}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
|
loading={isLoading}
|
||||||
>
|
>
|
||||||
<IconEdit size={18} />
|
<IconEdit size={18} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
@@ -339,7 +347,8 @@ export function DrawioMenu({ editor }: EditorMenuProps) {
|
|||||||
<Modal.Root opened={opened} onClose={handleClose} fullScreen closeOnEscape={false}>
|
<Modal.Root opened={opened} onClose={handleClose} fullScreen closeOnEscape={false}>
|
||||||
<Modal.Overlay />
|
<Modal.Overlay />
|
||||||
<Modal.Content style={{ overflow: "hidden" }}>
|
<Modal.Content style={{ overflow: "hidden" }}>
|
||||||
<Modal.Body>
|
<Modal.Body pos="relative">
|
||||||
|
<LoadingOverlay visible={isSaving} />
|
||||||
<div style={{ height: "100vh" }}>
|
<div style={{ height: "100vh" }}>
|
||||||
<DrawIoEmbed
|
<DrawIoEmbed
|
||||||
ref={drawioRef}
|
ref={drawioRef}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import { NodeViewProps, NodeViewWrapper } from "@tiptap/react";
|
|||||||
import {
|
import {
|
||||||
ActionIcon,
|
ActionIcon,
|
||||||
Card,
|
Card,
|
||||||
|
LoadingOverlay,
|
||||||
Modal,
|
Modal,
|
||||||
Text,
|
Text,
|
||||||
useComputedColorScheme,
|
useComputedColorScheme,
|
||||||
@@ -34,6 +35,7 @@ export default function DrawioView(props: NodeViewProps) {
|
|||||||
const computedColorScheme = useComputedColorScheme();
|
const computedColorScheme = useComputedColorScheme();
|
||||||
const isDirtyRef = useRef(false);
|
const isDirtyRef = useRef(false);
|
||||||
const isSavingRef = useRef(false);
|
const isSavingRef = useRef(false);
|
||||||
|
const [isSaving, setIsSaving] = useState(false);
|
||||||
|
|
||||||
const handleOpen = async () => {
|
const handleOpen = async () => {
|
||||||
if (!editor.isEditable) {
|
if (!editor.isEditable) {
|
||||||
@@ -47,6 +49,7 @@ export default function DrawioView(props: NodeViewProps) {
|
|||||||
if (isSavingRef.current) return;
|
if (isSavingRef.current) return;
|
||||||
|
|
||||||
isSavingRef.current = true;
|
isSavingRef.current = true;
|
||||||
|
setIsSaving(true);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const svgString = decodeBase64ToSvgString(svgXml);
|
const svgString = decodeBase64ToSvgString(svgXml);
|
||||||
@@ -79,6 +82,7 @@ export default function DrawioView(props: NodeViewProps) {
|
|||||||
isDirtyRef.current = false;
|
isDirtyRef.current = false;
|
||||||
} finally {
|
} finally {
|
||||||
isSavingRef.current = false;
|
isSavingRef.current = false;
|
||||||
|
setIsSaving(false);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -136,7 +140,8 @@ export default function DrawioView(props: NodeViewProps) {
|
|||||||
<Modal.Root opened={opened} onClose={handleClose} fullScreen closeOnEscape={false}>
|
<Modal.Root opened={opened} onClose={handleClose} fullScreen closeOnEscape={false}>
|
||||||
<Modal.Overlay />
|
<Modal.Overlay />
|
||||||
<Modal.Content style={{ overflow: "hidden" }}>
|
<Modal.Content style={{ overflow: "hidden" }}>
|
||||||
<Modal.Body>
|
<Modal.Body pos="relative">
|
||||||
|
<LoadingOverlay visible={isSaving} />
|
||||||
<div style={{ height: "100vh" }}>
|
<div style={{ height: "100vh" }}>
|
||||||
<DrawIoEmbed
|
<DrawIoEmbed
|
||||||
ref={drawioRef}
|
ref={drawioRef}
|
||||||
|
|||||||
@@ -86,8 +86,8 @@ export default function EmbedView(props: NodeViewProps) {
|
|||||||
{embedUrl ? (
|
{embedUrl ? (
|
||||||
<div className={classes.embedContainer}>
|
<div className={classes.embedContainer}>
|
||||||
<ResizableWrapper
|
<ResizableWrapper
|
||||||
initialWidth={nodeWidth || 640}
|
initialWidth={nodeWidth || 800}
|
||||||
initialHeight={nodeHeight || 480}
|
initialHeight={nodeHeight || 600}
|
||||||
minWidth={200}
|
minWidth={200}
|
||||||
maxWidth={1200}
|
maxWidth={1200}
|
||||||
minHeight={200}
|
minHeight={200}
|
||||||
@@ -102,8 +102,9 @@ export default function EmbedView(props: NodeViewProps) {
|
|||||||
<iframe
|
<iframe
|
||||||
className={classes.embedIframe}
|
className={classes.embedIframe}
|
||||||
src={sanitizeUrl(embedUrl)}
|
src={sanitizeUrl(embedUrl)}
|
||||||
allow="encrypted-media"
|
allow="encrypted-media; clipboard-read; clipboard-write; picture-in-picture;"
|
||||||
sandbox="allow-scripts allow-same-origin allow-forms allow-popups"
|
loading="lazy"
|
||||||
|
sandbox="allow-scripts allow-same-origin allow-forms allow-popups allow-downloads"
|
||||||
allowFullScreen
|
allowFullScreen
|
||||||
frameBorder="0"
|
frameBorder="0"
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -56,6 +56,8 @@ export function ExcalidrawMenu({ editor }: EditorMenuProps) {
|
|||||||
const computedColorScheme = useComputedColorScheme();
|
const computedColorScheme = useComputedColorScheme();
|
||||||
const isDirtyRef = useRef(false);
|
const isDirtyRef = useRef(false);
|
||||||
const isSavingRef = useRef(false);
|
const isSavingRef = useRef(false);
|
||||||
|
const [isSaving, setIsSaving] = useState(false);
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const isInitialLoadRef = useRef(true);
|
const isInitialLoadRef = useRef(true);
|
||||||
const lastFingerprintRef = useRef("");
|
const lastFingerprintRef = useRef("");
|
||||||
|
|
||||||
@@ -153,6 +155,7 @@ export function ExcalidrawMenu({ editor }: EditorMenuProps) {
|
|||||||
const handleOpen = useCallback(async () => {
|
const handleOpen = useCallback(async () => {
|
||||||
if (!editorState?.src) return;
|
if (!editorState?.src) return;
|
||||||
|
|
||||||
|
setIsLoading(true);
|
||||||
try {
|
try {
|
||||||
const url = getFileUrl(editorState.src);
|
const url = getFileUrl(editorState.src);
|
||||||
const request = await fetch(url, {
|
const request = await fetch(url, {
|
||||||
@@ -166,6 +169,7 @@ export function ExcalidrawMenu({ editor }: EditorMenuProps) {
|
|||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error(err);
|
console.error(err);
|
||||||
} finally {
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
isDirtyRef.current = false;
|
isDirtyRef.current = false;
|
||||||
isInitialLoadRef.current = true;
|
isInitialLoadRef.current = true;
|
||||||
open();
|
open();
|
||||||
@@ -178,6 +182,7 @@ export function ExcalidrawMenu({ editor }: EditorMenuProps) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
isSavingRef.current = true;
|
isSavingRef.current = true;
|
||||||
|
setIsSaving(true);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const { exportToSvg } = await import("@excalidraw/excalidraw");
|
const { exportToSvg } = await import("@excalidraw/excalidraw");
|
||||||
@@ -223,6 +228,7 @@ export function ExcalidrawMenu({ editor }: EditorMenuProps) {
|
|||||||
isDirtyRef.current = false;
|
isDirtyRef.current = false;
|
||||||
} finally {
|
} finally {
|
||||||
isSavingRef.current = false;
|
isSavingRef.current = false;
|
||||||
|
setIsSaving(false);
|
||||||
}
|
}
|
||||||
}, [editor, excalidrawAPI, editorState?.attachmentId]);
|
}, [editor, excalidrawAPI, editorState?.attachmentId]);
|
||||||
|
|
||||||
@@ -339,6 +345,7 @@ export function ExcalidrawMenu({ editor }: EditorMenuProps) {
|
|||||||
size="lg"
|
size="lg"
|
||||||
aria-label={t("Edit")}
|
aria-label={t("Edit")}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
|
loading={isLoading}
|
||||||
>
|
>
|
||||||
<IconEdit size={18} />
|
<IconEdit size={18} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
@@ -390,7 +397,7 @@ export function ExcalidrawMenu({ editor }: EditorMenuProps) {
|
|||||||
bg="var(--mantine-color-body)"
|
bg="var(--mantine-color-body)"
|
||||||
p="xs"
|
p="xs"
|
||||||
>
|
>
|
||||||
<Button onClick={handleSaveAndExit} size={"compact-sm"}>
|
<Button onClick={handleSaveAndExit} size={"compact-sm"} loading={isSaving}>
|
||||||
{t("Save & Exit")}
|
{t("Save & Exit")}
|
||||||
</Button>
|
</Button>
|
||||||
<Button onClick={handleClose} color="red" size={"compact-sm"}>
|
<Button onClick={handleClose} color="red" size={"compact-sm"}>
|
||||||
|
|||||||
@@ -52,6 +52,7 @@ export default function ExcalidrawView(props: NodeViewProps) {
|
|||||||
|
|
||||||
const isDirtyRef = useRef(false);
|
const isDirtyRef = useRef(false);
|
||||||
const isSavingRef = useRef(false);
|
const isSavingRef = useRef(false);
|
||||||
|
const [isSaving, setIsSaving] = useState(false);
|
||||||
const isInitialLoadRef = useRef(true);
|
const isInitialLoadRef = useRef(true);
|
||||||
const lastFingerprintRef = useRef("");
|
const lastFingerprintRef = useRef("");
|
||||||
|
|
||||||
@@ -70,6 +71,7 @@ export default function ExcalidrawView(props: NodeViewProps) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
isSavingRef.current = true;
|
isSavingRef.current = true;
|
||||||
|
setIsSaving(true);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const { exportToSvg } = await import("@excalidraw/excalidraw");
|
const { exportToSvg } = await import("@excalidraw/excalidraw");
|
||||||
@@ -120,6 +122,7 @@ export default function ExcalidrawView(props: NodeViewProps) {
|
|||||||
isDirtyRef.current = false;
|
isDirtyRef.current = false;
|
||||||
} finally {
|
} finally {
|
||||||
isSavingRef.current = false;
|
isSavingRef.current = false;
|
||||||
|
setIsSaving(false);
|
||||||
}
|
}
|
||||||
}, [excalidrawAPI, editor, attachmentId, updateAttributes]);
|
}, [excalidrawAPI, editor, attachmentId, updateAttributes]);
|
||||||
|
|
||||||
@@ -191,7 +194,7 @@ export default function ExcalidrawView(props: NodeViewProps) {
|
|||||||
bg="var(--mantine-color-body)"
|
bg="var(--mantine-color-body)"
|
||||||
p="xs"
|
p="xs"
|
||||||
>
|
>
|
||||||
<Button onClick={handleSaveAndExit} size={"compact-sm"}>
|
<Button onClick={handleSaveAndExit} size={"compact-sm"} loading={isSaving}>
|
||||||
{t("Save & Exit")}
|
{t("Save & Exit")}
|
||||||
</Button>
|
</Button>
|
||||||
<Button onClick={handleClose} color="red" size={"compact-sm"}>
|
<Button onClick={handleClose} color="red" size={"compact-sm"}>
|
||||||
|
|||||||
@@ -5,6 +5,9 @@
|
|||||||
max-width: 100%;
|
max-width: 100%;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.skeleton {
|
||||||
animation: pulse 1.2s ease-in-out infinite;
|
animation: pulse 1.2s ease-in-out infinite;
|
||||||
|
|
||||||
@mixin light {
|
@mixin light {
|
||||||
|
|||||||
@@ -33,6 +33,7 @@ export default function ImageView(props: NodeViewProps) {
|
|||||||
className={clsx(
|
className={clsx(
|
||||||
selected && "ProseMirror-selectednode",
|
selected && "ProseMirror-selectednode",
|
||||||
classes.imageWrapper,
|
classes.imageWrapper,
|
||||||
|
!src && placeholder && classes.skeleton,
|
||||||
alignClass,
|
alignClass,
|
||||||
)}
|
)}
|
||||||
style={{
|
style={{
|
||||||
@@ -54,7 +55,7 @@ export default function ImageView(props: NodeViewProps) {
|
|||||||
<Loader size={20} pos="absolute" bottom={6} right={6} />
|
<Loader size={20} pos="absolute" bottom={6} right={6} />
|
||||||
</Group>
|
</Group>
|
||||||
)}
|
)}
|
||||||
{!src && !previewSrc && (
|
{!src && !previewSrc && placeholder && (
|
||||||
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md">
|
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md">
|
||||||
<Loader size={20} style={{ flexShrink: 0 }} />
|
<Loader size={20} style={{ flexShrink: 0 }} />
|
||||||
<Text component="span" size="sm" truncate="end">
|
<Text component="span" size="sm" truncate="end">
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import { useAtom } from "jotai";
|
|||||||
import { isTextSelected } from "@docmost/editor-ext";
|
import { isTextSelected } from "@docmost/editor-ext";
|
||||||
import { showLinkMenuAtom } from "@/features/editor/atoms/editor-atoms";
|
import { showLinkMenuAtom } from "@/features/editor/atoms/editor-atoms";
|
||||||
import { LinkEditorPanel } from "@/features/editor/components/link/link-editor-panel";
|
import { LinkEditorPanel } from "@/features/editor/components/link/link-editor-panel";
|
||||||
import { normalizeUrl } from "@/features/editor/components/link/link-view";
|
import { normalizeUrl } from "@/lib/utils";
|
||||||
import { TextSelection } from "@tiptap/pm/state";
|
import { TextSelection } from "@tiptap/pm/state";
|
||||||
import { Paper } from "@mantine/core";
|
import { Paper } from "@mantine/core";
|
||||||
|
|
||||||
|
|||||||
@@ -29,12 +29,7 @@ import { useSharePageQuery } from "@/features/share/queries/share-query.ts";
|
|||||||
import { buildSharedPageUrl } from "@/features/page/page.utils.ts";
|
import { buildSharedPageUrl } from "@/features/page/page.utils.ts";
|
||||||
import { extractPageSlugId } from "@/lib";
|
import { extractPageSlugId } from "@/lib";
|
||||||
import { sanitizeUrl, copyToClipboard } from "@docmost/editor-ext";
|
import { sanitizeUrl, copyToClipboard } from "@docmost/editor-ext";
|
||||||
|
import { normalizeUrl } from "@/lib/utils";
|
||||||
export const normalizeUrl = (url: string): string => {
|
|
||||||
if (!url) return url;
|
|
||||||
if (url.startsWith("/") || /^(\S+):(\/\/)?\S+$/.test(url)) return url;
|
|
||||||
return `https://${url}`;
|
|
||||||
};
|
|
||||||
|
|
||||||
const parseInternalLink = (
|
const parseInternalLink = (
|
||||||
href: string,
|
href: string,
|
||||||
|
|||||||
@@ -294,6 +294,7 @@ const MentionList = forwardRef<any, MentionListProps>((props, ref) => {
|
|||||||
w={popupWidth}
|
w={popupWidth}
|
||||||
scrollbars={"y"}
|
scrollbars={"y"}
|
||||||
scrollbarSize={6}
|
scrollbarSize={6}
|
||||||
|
overscrollBehavior={"contain"}
|
||||||
styles={{ content: { minWidth: 0 } }}
|
styles={{ content: { minWidth: 0 } }}
|
||||||
>
|
>
|
||||||
{renderItems?.map((item, index) => {
|
{renderItems?.map((item, index) => {
|
||||||
|
|||||||
@@ -0,0 +1,145 @@
|
|||||||
|
import { BubbleMenu as BaseBubbleMenu } from "@tiptap/react/menus";
|
||||||
|
import { findParentNode, posToDOMRect, useEditorState } from "@tiptap/react";
|
||||||
|
import { useCallback } from "react";
|
||||||
|
import { Node as PMNode } from "@tiptap/pm/model";
|
||||||
|
import {
|
||||||
|
EditorMenuProps,
|
||||||
|
ShouldShowProps,
|
||||||
|
} from "@/features/editor/components/table/types/types.ts";
|
||||||
|
import { ActionIcon, Tooltip } from "@mantine/core";
|
||||||
|
import {
|
||||||
|
IconPaperclip,
|
||||||
|
IconTrash,
|
||||||
|
} from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import classes from "../common/toolbar-menu.module.css";
|
||||||
|
|
||||||
|
export function PdfMenu({ editor }: EditorMenuProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const editorState = useEditorState({
|
||||||
|
editor,
|
||||||
|
selector: (ctx) => {
|
||||||
|
if (!ctx.editor) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const pdfAttrs = ctx.editor.getAttributes("pdf");
|
||||||
|
|
||||||
|
return {
|
||||||
|
isPdf: ctx.editor.isActive("pdf"),
|
||||||
|
src: pdfAttrs?.src || null,
|
||||||
|
name: pdfAttrs?.name || null,
|
||||||
|
attachmentId: pdfAttrs?.attachmentId || null,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const shouldShow = useCallback(
|
||||||
|
({ state }: ShouldShowProps) => {
|
||||||
|
if (!state || !editor.isActive("pdf")) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { selection } = state;
|
||||||
|
const dom = editor.view.nodeDOM(selection.from) as HTMLElement | null;
|
||||||
|
if (!dom) return false;
|
||||||
|
|
||||||
|
return !!dom.querySelector("[data-pdf-error]");
|
||||||
|
},
|
||||||
|
[editor],
|
||||||
|
);
|
||||||
|
|
||||||
|
const getReferencedVirtualElement = useCallback(() => {
|
||||||
|
if (!editor) return;
|
||||||
|
const { selection } = editor.state;
|
||||||
|
const predicate = (node: PMNode) => node.type.name === "pdf";
|
||||||
|
const parent = findParentNode(predicate)(selection);
|
||||||
|
|
||||||
|
if (parent) {
|
||||||
|
const dom = editor.view.nodeDOM(parent?.pos) as HTMLElement;
|
||||||
|
const domRect = dom.getBoundingClientRect();
|
||||||
|
return {
|
||||||
|
getBoundingClientRect: () => domRect,
|
||||||
|
getClientRects: () => [domRect],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const domRect = posToDOMRect(editor.view, selection.from, selection.to);
|
||||||
|
return {
|
||||||
|
getBoundingClientRect: () => domRect,
|
||||||
|
getClientRects: () => [domRect],
|
||||||
|
};
|
||||||
|
}, [editor]);
|
||||||
|
|
||||||
|
const handleConvertToAttachment = useCallback(() => {
|
||||||
|
if (!editorState?.src) return;
|
||||||
|
|
||||||
|
const { selection } = editor.state;
|
||||||
|
const { from } = selection;
|
||||||
|
const node = editor.state.doc.nodeAt(from);
|
||||||
|
if (!node || node.type.name !== "pdf") return;
|
||||||
|
|
||||||
|
editor
|
||||||
|
.chain()
|
||||||
|
.insertContentAt(
|
||||||
|
{ from, to: from + node.nodeSize },
|
||||||
|
{
|
||||||
|
type: "attachment",
|
||||||
|
attrs: {
|
||||||
|
url: node.attrs.src,
|
||||||
|
name: node.attrs.name,
|
||||||
|
attachmentId: node.attrs.attachmentId,
|
||||||
|
size: node.attrs.size,
|
||||||
|
mime: "application/pdf",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.run();
|
||||||
|
}, [editor, editorState]);
|
||||||
|
|
||||||
|
const handleDelete = useCallback(() => {
|
||||||
|
editor.commands.deleteSelection();
|
||||||
|
}, [editor]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<BaseBubbleMenu
|
||||||
|
editor={editor}
|
||||||
|
pluginKey={`pdf-menu`}
|
||||||
|
updateDelay={0}
|
||||||
|
getReferencedVirtualElement={getReferencedVirtualElement}
|
||||||
|
options={{
|
||||||
|
placement: "top",
|
||||||
|
offset: 8,
|
||||||
|
flip: false,
|
||||||
|
}}
|
||||||
|
shouldShow={shouldShow}
|
||||||
|
>
|
||||||
|
<div className={classes.toolbar}>
|
||||||
|
<Tooltip position="top" label={t("Convert to attachment")} withinPortal={false}>
|
||||||
|
<ActionIcon
|
||||||
|
onClick={handleConvertToAttachment}
|
||||||
|
size="lg"
|
||||||
|
aria-label={t("Convert to attachment")}
|
||||||
|
variant="subtle"
|
||||||
|
>
|
||||||
|
<IconPaperclip size={18} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
|
||||||
|
<Tooltip position="top" label={t("Delete")} withinPortal={false}>
|
||||||
|
<ActionIcon
|
||||||
|
onClick={handleDelete}
|
||||||
|
size="lg"
|
||||||
|
aria-label={t("Delete")}
|
||||||
|
variant="subtle"
|
||||||
|
>
|
||||||
|
<IconTrash size={18} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
</BaseBubbleMenu>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default PdfMenu;
|
||||||
@@ -0,0 +1,100 @@
|
|||||||
|
.pdfWrapper {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
max-width: 100%;
|
||||||
|
border-radius: 8px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.skeleton {
|
||||||
|
animation: pulse 1.2s ease-in-out infinite;
|
||||||
|
|
||||||
|
@mixin light {
|
||||||
|
background: linear-gradient(-90deg, var(--mantine-color-gray-3) 0%, var(--mantine-color-gray-1) 50%, var(--mantine-color-gray-3) 100%);
|
||||||
|
background-size: 400% 400%;
|
||||||
|
}
|
||||||
|
|
||||||
|
@mixin dark {
|
||||||
|
background: linear-gradient(-90deg, var(--mantine-color-dark-6) 0%, var(--mantine-color-dark-5) 50%, var(--mantine-color-dark-6) 100%);
|
||||||
|
background-size: 400% 400%;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes pulse {
|
||||||
|
0% {
|
||||||
|
background-position: 0% 0%;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
background-position: -135% 0%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.pdfContainer {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pdfResizeWrapper {
|
||||||
|
@mixin light {
|
||||||
|
background-color: var(--mantine-color-gray-0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@mixin dark {
|
||||||
|
background-color: var(--mantine-color-dark-7);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.pdfIframe {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
border: none;
|
||||||
|
border-radius: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hoverMenu {
|
||||||
|
position: absolute;
|
||||||
|
top: 56px;
|
||||||
|
right: 8px;
|
||||||
|
z-index: 2;
|
||||||
|
display: flex;
|
||||||
|
gap: 4px;
|
||||||
|
padding: 4px;
|
||||||
|
border-radius: 6px;
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 0.15s ease;
|
||||||
|
background-color: rgba(0, 0, 0, 0.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
.hoverMenu::before {
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
inset: -12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hoverMenu:hover {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pdfResizeWrapper:hover .hoverMenu {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pdfError {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 8px;
|
||||||
|
padding: 12px 32px;
|
||||||
|
border-radius: 8px;
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
@mixin light {
|
||||||
|
background-color: var(--mantine-color-gray-0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@mixin dark {
|
||||||
|
background-color: var(--mantine-color-dark-7);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,170 @@
|
|||||||
|
import { NodeViewProps, NodeViewWrapper } from "@tiptap/react";
|
||||||
|
import { ActionIcon, Group, Loader, Text, Tooltip } from "@mantine/core";
|
||||||
|
import { useCallback, useMemo, useState } from "react";
|
||||||
|
import { getFileUrl } from "@/lib/config.ts";
|
||||||
|
import { ResizableWrapper } from "../common/resizable-wrapper";
|
||||||
|
import clsx from "clsx";
|
||||||
|
import classes from "./pdf-view.module.css";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { isInternalFileUrl } from "@docmost/editor-ext";
|
||||||
|
import {
|
||||||
|
IconFileTypePdf,
|
||||||
|
IconPaperclip,
|
||||||
|
IconTrash,
|
||||||
|
} from "@tabler/icons-react";
|
||||||
|
|
||||||
|
export default function PdfView(props: NodeViewProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { editor, node, getPos, selected, updateAttributes } = props;
|
||||||
|
const { src, placeholder, width: nodeWidth, height: nodeHeight } = node.attrs;
|
||||||
|
const [hasError, setHasError] = useState(false);
|
||||||
|
|
||||||
|
const safeSrc = useMemo(() => {
|
||||||
|
if (!src || !isInternalFileUrl(src)) return null;
|
||||||
|
return getFileUrl(src);
|
||||||
|
}, [src]);
|
||||||
|
|
||||||
|
const handleSelect = useCallback(() => {
|
||||||
|
const pos = getPos();
|
||||||
|
if (pos !== undefined) {
|
||||||
|
editor.commands.setNodeSelection(pos);
|
||||||
|
}
|
||||||
|
}, [editor, getPos]);
|
||||||
|
|
||||||
|
const handleResize = useCallback(
|
||||||
|
(newWidth: number, newHeight: number) => {
|
||||||
|
updateAttributes({ width: newWidth, height: newHeight });
|
||||||
|
},
|
||||||
|
[updateAttributes],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleConvertToAttachment = useCallback(() => {
|
||||||
|
if (!src) return;
|
||||||
|
const pos = getPos();
|
||||||
|
if (pos === undefined) return;
|
||||||
|
const currentNode = editor.state.doc.nodeAt(pos);
|
||||||
|
if (!currentNode || currentNode.type.name !== "pdf") return;
|
||||||
|
|
||||||
|
editor
|
||||||
|
.chain()
|
||||||
|
.insertContentAt(
|
||||||
|
{ from: pos, to: pos + currentNode.nodeSize },
|
||||||
|
{
|
||||||
|
type: "attachment",
|
||||||
|
attrs: {
|
||||||
|
url: currentNode.attrs.src,
|
||||||
|
name: currentNode.attrs.name,
|
||||||
|
attachmentId: currentNode.attrs.attachmentId,
|
||||||
|
size: currentNode.attrs.size,
|
||||||
|
mime: "application/pdf",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.run();
|
||||||
|
}, [editor, src, getPos]);
|
||||||
|
|
||||||
|
const handleDelete = useCallback(() => {
|
||||||
|
const pos = getPos();
|
||||||
|
if (pos === undefined) return;
|
||||||
|
editor.commands.setNodeSelection(pos);
|
||||||
|
editor.commands.deleteSelection();
|
||||||
|
}, [editor, getPos]);
|
||||||
|
|
||||||
|
if (!src || !safeSrc) {
|
||||||
|
return (
|
||||||
|
<NodeViewWrapper data-drag-handle>
|
||||||
|
<div className={`${classes.pdfWrapper} ${placeholder ? classes.skeleton : ''}`} style={{ height: placeholder ? 600 : undefined }}>
|
||||||
|
{placeholder && (
|
||||||
|
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md">
|
||||||
|
<Loader size={20} style={{ flexShrink: 0 }} />
|
||||||
|
<Text component="span" size="sm" truncate="end">
|
||||||
|
{placeholder?.name
|
||||||
|
? t("Uploading {{name}}", { name: placeholder.name })
|
||||||
|
: t("Uploading file")}
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</NodeViewWrapper>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hasError) {
|
||||||
|
return (
|
||||||
|
<NodeViewWrapper data-drag-handle>
|
||||||
|
<div data-pdf-error className={clsx(classes.pdfError, { "ProseMirror-selectednode": selected })} onClick={handleSelect}>
|
||||||
|
<IconFileTypePdf size={32} stroke={1.5} />
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t("Failed to load PDF")}
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
</NodeViewWrapper>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<NodeViewWrapper data-drag-handle className={classes.pdfNodeView}>
|
||||||
|
<div className={classes.pdfContainer}>
|
||||||
|
<ResizableWrapper
|
||||||
|
initialWidth={nodeWidth || 800}
|
||||||
|
initialHeight={nodeHeight || 600}
|
||||||
|
minWidth={200}
|
||||||
|
maxWidth={1200}
|
||||||
|
minHeight={200}
|
||||||
|
maxHeight={1200}
|
||||||
|
onResize={handleResize}
|
||||||
|
isEditable={editor.isEditable}
|
||||||
|
selected={selected}
|
||||||
|
className={clsx(classes.pdfResizeWrapper, {
|
||||||
|
"ProseMirror-selectednode": selected,
|
||||||
|
})}
|
||||||
|
>
|
||||||
|
<iframe
|
||||||
|
className={classes.pdfIframe}
|
||||||
|
src={safeSrc}
|
||||||
|
loading="lazy"
|
||||||
|
frameBorder="0"
|
||||||
|
onError={() => setHasError(true)}
|
||||||
|
onLoad={(e) => {
|
||||||
|
try {
|
||||||
|
const iframe = e.currentTarget;
|
||||||
|
const status = iframe.contentDocument?.querySelector("pre")?.textContent;
|
||||||
|
if (status && status.includes('"statusCode":404')) {
|
||||||
|
setHasError(true);
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
// cross-origin - can't inspect, assume OK
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
{editor.isEditable && (
|
||||||
|
<div className={classes.hoverMenu}>
|
||||||
|
<Tooltip position="top" label={t("Convert to attachment")} withinPortal>
|
||||||
|
<ActionIcon
|
||||||
|
size="sm"
|
||||||
|
variant="filled"
|
||||||
|
color="dark"
|
||||||
|
onClick={handleConvertToAttachment}
|
||||||
|
aria-label={t("Convert to attachment")}
|
||||||
|
>
|
||||||
|
<IconPaperclip size={14} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
<Tooltip position="top" label={t("Delete")} withinPortal>
|
||||||
|
<ActionIcon
|
||||||
|
size="sm"
|
||||||
|
variant="filled"
|
||||||
|
color="dark"
|
||||||
|
onClick={handleDelete}
|
||||||
|
aria-label={t("Delete")}
|
||||||
|
>
|
||||||
|
<IconTrash size={14} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</ResizableWrapper>
|
||||||
|
</div>
|
||||||
|
</NodeViewWrapper>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
import { handlePdfUpload } from "@docmost/editor-ext";
|
||||||
|
import { uploadFile } from "@/features/page/services/page-service.ts";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { getFileUploadSizeLimit } from "@/lib/config.ts";
|
||||||
|
import { formatBytes } from "@/lib";
|
||||||
|
import i18n from "@/i18n.ts";
|
||||||
|
|
||||||
|
export const uploadPdfAction = handlePdfUpload({
|
||||||
|
onUpload: async (file: File, pageId: string): Promise<any> => {
|
||||||
|
try {
|
||||||
|
return await uploadFile(file, pageId);
|
||||||
|
} catch (err) {
|
||||||
|
notifications.show({
|
||||||
|
color: "red",
|
||||||
|
message: err?.response.data.message,
|
||||||
|
});
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
validateFn: (file) => {
|
||||||
|
if (file.type !== "application/pdf") {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (file.size > getFileUploadSizeLimit()) {
|
||||||
|
notifications.show({
|
||||||
|
color: "red",
|
||||||
|
message: i18n.t("File exceeds the {{limit}} attachment limit", {
|
||||||
|
limit: formatBytes(getFileUploadSizeLimit()),
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
});
|
||||||
@@ -87,7 +87,13 @@ const CommandList = ({
|
|||||||
|
|
||||||
return flatItems.length > 0 ? (
|
return flatItems.length > 0 ? (
|
||||||
<Paper id="slash-command" shadow="md" p="xs" withBorder>
|
<Paper id="slash-command" shadow="md" p="xs" withBorder>
|
||||||
<ScrollArea viewportRef={viewportRef} h={350} w={270} scrollbarSize={8}>
|
<ScrollArea
|
||||||
|
viewportRef={viewportRef}
|
||||||
|
h={350}
|
||||||
|
w={270}
|
||||||
|
scrollbarSize={8}
|
||||||
|
overscrollBehavior="contain"
|
||||||
|
>
|
||||||
{Object.entries(items).map(([category, categoryItems]) => (
|
{Object.entries(items).map(([category, categoryItems]) => (
|
||||||
<div key={category}>
|
<div key={category}>
|
||||||
<Text c="dimmed" mb={4} fw={500} tt="capitalize">
|
<Text c="dimmed" mb={4} fw={500} tt="capitalize">
|
||||||
@@ -103,10 +109,7 @@ const CommandList = ({
|
|||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<Group>
|
<Group>
|
||||||
<ActionIcon
|
<ActionIcon variant="default" component="div">
|
||||||
variant="default"
|
|
||||||
component="div"
|
|
||||||
>
|
|
||||||
<item.icon size={18} />
|
<item.icon size={18} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
|
|
||||||
|
|||||||
@@ -12,7 +12,9 @@ import {
|
|||||||
IconMath,
|
IconMath,
|
||||||
IconMathFunction,
|
IconMathFunction,
|
||||||
IconMovie,
|
IconMovie,
|
||||||
|
IconMusic,
|
||||||
IconPaperclip,
|
IconPaperclip,
|
||||||
|
IconFileTypePdf,
|
||||||
IconPhoto,
|
IconPhoto,
|
||||||
IconTable,
|
IconTable,
|
||||||
IconTypography,
|
IconTypography,
|
||||||
@@ -30,7 +32,9 @@ import {
|
|||||||
} from "@/features/editor/components/slash-menu/types";
|
} from "@/features/editor/components/slash-menu/types";
|
||||||
import { uploadImageAction } from "@/features/editor/components/image/upload-image-action.tsx";
|
import { uploadImageAction } from "@/features/editor/components/image/upload-image-action.tsx";
|
||||||
import { uploadVideoAction } from "@/features/editor/components/video/upload-video-action.tsx";
|
import { uploadVideoAction } from "@/features/editor/components/video/upload-video-action.tsx";
|
||||||
|
import { uploadAudioAction } from "@/features/editor/components/audio/upload-audio-action.tsx";
|
||||||
import { uploadAttachmentAction } from "@/features/editor/components/attachment/upload-attachment-action.tsx";
|
import { uploadAttachmentAction } from "@/features/editor/components/attachment/upload-attachment-action.tsx";
|
||||||
|
import { uploadPdfAction } from "@/features/editor/components/pdf/upload-pdf-action.tsx";
|
||||||
import IconExcalidraw from "@/components/icons/icon-excalidraw";
|
import IconExcalidraw from "@/components/icons/icon-excalidraw";
|
||||||
import IconMermaid from "@/components/icons/icon-mermaid";
|
import IconMermaid from "@/components/icons/icon-mermaid";
|
||||||
import IconDrawio from "@/components/icons/icon-drawio";
|
import IconDrawio from "@/components/icons/icon-drawio";
|
||||||
@@ -161,7 +165,7 @@ const CommandGroups: SlashMenuGroupedItemsType = {
|
|||||||
{
|
{
|
||||||
title: "Image",
|
title: "Image",
|
||||||
description: "Upload any image from your device.",
|
description: "Upload any image from your device.",
|
||||||
searchTerms: ["photo", "picture", "media"],
|
searchTerms: ["photo", "picture", "media", "file", "attachment"],
|
||||||
icon: IconPhoto,
|
icon: IconPhoto,
|
||||||
command: ({ editor, range }) => {
|
command: ({ editor, range }) => {
|
||||||
editor.chain().focus().deleteRange(range).run();
|
editor.chain().focus().deleteRange(range).run();
|
||||||
@@ -194,7 +198,7 @@ const CommandGroups: SlashMenuGroupedItemsType = {
|
|||||||
{
|
{
|
||||||
title: "Video",
|
title: "Video",
|
||||||
description: "Upload any video from your device.",
|
description: "Upload any video from your device.",
|
||||||
searchTerms: ["video", "mp4", "media"],
|
searchTerms: ["video", "mp4", "media", "file", "attachment"],
|
||||||
icon: IconMovie,
|
icon: IconMovie,
|
||||||
command: ({ editor, range }) => {
|
command: ({ editor, range }) => {
|
||||||
editor.chain().focus().deleteRange(range).run();
|
editor.chain().focus().deleteRange(range).run();
|
||||||
@@ -224,10 +228,74 @@ const CommandGroups: SlashMenuGroupedItemsType = {
|
|||||||
input.click();
|
input.click();
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
title: "Audio",
|
||||||
|
description: "Upload any audio from your device.",
|
||||||
|
searchTerms: ["audio", "music", "sound", "mp3", "media", "file", "attachment"],
|
||||||
|
icon: IconMusic,
|
||||||
|
command: ({ editor, range }) => {
|
||||||
|
editor.chain().focus().deleteRange(range).run();
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const pageId = editor.storage?.pageId;
|
||||||
|
if (!pageId) return;
|
||||||
|
|
||||||
|
// upload audio
|
||||||
|
const input = document.createElement("input");
|
||||||
|
input.type = "file";
|
||||||
|
input.accept = "audio/*";
|
||||||
|
input.multiple = true;
|
||||||
|
input.style.display = "none";
|
||||||
|
document.body.appendChild(input);
|
||||||
|
input.onchange = async () => {
|
||||||
|
if (input.files?.length) {
|
||||||
|
for (const file of input.files) {
|
||||||
|
const pos = editor.view.state.selection.from;
|
||||||
|
|
||||||
|
uploadAudioAction(file, editor, pos, pageId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
input.remove();
|
||||||
|
};
|
||||||
|
input.click();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: "Embed PDF",
|
||||||
|
description: "Upload and embed a PDF file.",
|
||||||
|
searchTerms: ["pdf", "document", "embed"],
|
||||||
|
icon: IconFileTypePdf,
|
||||||
|
command: ({ editor, range }) => {
|
||||||
|
editor.chain().focus().deleteRange(range).run();
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const pageId = editor.storage?.pageId;
|
||||||
|
if (!pageId) return;
|
||||||
|
|
||||||
|
const input = document.createElement("input");
|
||||||
|
input.type = "file";
|
||||||
|
input.accept = "application/pdf";
|
||||||
|
input.style.display = "none";
|
||||||
|
document.body.appendChild(input);
|
||||||
|
input.onchange = async () => {
|
||||||
|
if (input.files?.length) {
|
||||||
|
for (const file of input.files) {
|
||||||
|
const pos = editor.view.state.selection.from;
|
||||||
|
|
||||||
|
uploadPdfAction(file, editor, pos, pageId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
input.remove();
|
||||||
|
};
|
||||||
|
input.click();
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
title: "File attachment",
|
title: "File attachment",
|
||||||
description: "Upload any file from your device.",
|
description: "Upload any file from your device.",
|
||||||
searchTerms: ["file", "attachment", "upload", "pdf", "csv", "zip"],
|
searchTerms: ["file", "attachment", "upload", "csv", "zip"],
|
||||||
icon: IconPaperclip,
|
icon: IconPaperclip,
|
||||||
command: ({ editor, range }) => {
|
command: ({ editor, range }) => {
|
||||||
editor.chain().focus().deleteRange(range).run();
|
editor.chain().focus().deleteRange(range).run();
|
||||||
@@ -359,7 +427,7 @@ const CommandGroups: SlashMenuGroupedItemsType = {
|
|||||||
editor.chain().focus().deleteRange(range).setDrawio().run(),
|
editor.chain().focus().deleteRange(range).setDrawio().run(),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
title: "Excalidraw diagram",
|
title: "Excalidraw (Whiteboard)",
|
||||||
description: "Draw and sketch excalidraw diagrams",
|
description: "Draw and sketch excalidraw diagrams",
|
||||||
searchTerms: ["diagrams", "draw", "sketch", "whiteboard"],
|
searchTerms: ["diagrams", "draw", "sketch", "whiteboard"],
|
||||||
icon: IconExcalidraw,
|
icon: IconExcalidraw,
|
||||||
@@ -548,7 +616,7 @@ const CommandGroups: SlashMenuGroupedItemsType = {
|
|||||||
{
|
{
|
||||||
title: "YouTube",
|
title: "YouTube",
|
||||||
description: "Embed YouTube video",
|
description: "Embed YouTube video",
|
||||||
searchTerms: ["youtube", "yt"],
|
searchTerms: ["youtube", "yt", "media", "video"],
|
||||||
icon: YoutubeIcon,
|
icon: YoutubeIcon,
|
||||||
command: ({ editor, range }: CommandProps) => {
|
command: ({ editor, range }: CommandProps) => {
|
||||||
editor
|
editor
|
||||||
@@ -647,7 +715,11 @@ export const getSuggestionItems = ({
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (filteredItems.length) {
|
if (filteredItems.length) {
|
||||||
filteredGroups[group] = filteredItems;
|
filteredGroups[group] = filteredItems.sort((a, b) => {
|
||||||
|
const aTitle = a.title.toLowerCase().includes(search) ? 0 : 1;
|
||||||
|
const bTitle = b.title.toLowerCase().includes(search) ? 0 : 1;
|
||||||
|
return aTitle - bTitle;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -49,7 +49,7 @@ const renderItems = () => {
|
|||||||
getReferenceClientRect = props.clientRect;
|
getReferenceClientRect = props.clientRect;
|
||||||
|
|
||||||
popup = document.createElement("div");
|
popup = document.createElement("div");
|
||||||
popup.style.zIndex = "9999";
|
popup.style.zIndex = "199";
|
||||||
popup.style.position = "absolute";
|
popup.style.position = "absolute";
|
||||||
popup.style.top = "0";
|
popup.style.top = "0";
|
||||||
popup.style.left = "0";
|
popup.style.left = "0";
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ export const TableMenu = React.memo(
|
|||||||
if (isTextSelected(editor)) return false;
|
if (isTextSelected(editor)) return false;
|
||||||
return editor.isActive("table") && !isCellSelection(state.selection);
|
return editor.isActive("table") && !isCellSelection(state.selection);
|
||||||
},
|
},
|
||||||
[editor]
|
[editor],
|
||||||
);
|
);
|
||||||
|
|
||||||
const getReferencedVirtualElement = useCallback(() => {
|
const getReferencedVirtualElement = useCallback(() => {
|
||||||
@@ -121,7 +121,11 @@ export const TableMenu = React.memo(
|
|||||||
shouldShow={shouldShow}
|
shouldShow={shouldShow}
|
||||||
>
|
>
|
||||||
<div className={classes.toolbar}>
|
<div className={classes.toolbar}>
|
||||||
<Tooltip position="top" label={t("Add left column")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Add left column")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={addColumnLeft}
|
onClick={addColumnLeft}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -132,7 +136,11 @@ export const TableMenu = React.memo(
|
|||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Add right column")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Add right column")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={addColumnRight}
|
onClick={addColumnRight}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -143,7 +151,11 @@ export const TableMenu = React.memo(
|
|||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Delete column")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Delete column")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={deleteColumn}
|
onClick={deleteColumn}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -156,7 +168,11 @@ export const TableMenu = React.memo(
|
|||||||
|
|
||||||
<div className={classes.divider} />
|
<div className={classes.divider} />
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Add row above")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Add row above")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={addRowAbove}
|
onClick={addRowAbove}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -167,7 +183,11 @@ export const TableMenu = React.memo(
|
|||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Add row below")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Add row below")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={addRowBelow}
|
onClick={addRowBelow}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -178,7 +198,7 @@ export const TableMenu = React.memo(
|
|||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Delete row")}>
|
<Tooltip position="top" label={t("Delete row")} withinPortal={false}>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={deleteRow}
|
onClick={deleteRow}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -191,7 +211,11 @@ export const TableMenu = React.memo(
|
|||||||
|
|
||||||
<div className={classes.divider} />
|
<div className={classes.divider} />
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Toggle header row")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Toggle header row")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={toggleHeaderRow}
|
onClick={toggleHeaderRow}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -202,7 +226,11 @@ export const TableMenu = React.memo(
|
|||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Toggle header column")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Toggle header column")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={toggleHeaderColumn}
|
onClick={toggleHeaderColumn}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -215,7 +243,11 @@ export const TableMenu = React.memo(
|
|||||||
|
|
||||||
<div className={classes.divider} />
|
<div className={classes.divider} />
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Delete table")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Delete table")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={deleteTable}
|
onClick={deleteTable}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -228,7 +260,7 @@ export const TableMenu = React.memo(
|
|||||||
</div>
|
</div>
|
||||||
</BubbleMenu>
|
</BubbleMenu>
|
||||||
);
|
);
|
||||||
}
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
export default TableMenu;
|
export default TableMenu;
|
||||||
|
|||||||
@@ -5,6 +5,9 @@
|
|||||||
max-width: 100%;
|
max-width: 100%;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.skeleton {
|
||||||
animation: pulse 1.2s ease-in-out infinite;
|
animation: pulse 1.2s ease-in-out infinite;
|
||||||
|
|
||||||
@mixin light {
|
@mixin light {
|
||||||
@@ -26,6 +29,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.video {
|
.video {
|
||||||
display: block;
|
display: block;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
|||||||
@@ -33,6 +33,7 @@ export default function VideoView(props: NodeViewProps) {
|
|||||||
className={clsx(
|
className={clsx(
|
||||||
selected && "ProseMirror-selectednode",
|
selected && "ProseMirror-selectednode",
|
||||||
classes.videoWrapper,
|
classes.videoWrapper,
|
||||||
|
!src && placeholder && classes.skeleton,
|
||||||
alignClass,
|
alignClass,
|
||||||
)}
|
)}
|
||||||
style={{
|
style={{
|
||||||
@@ -59,7 +60,7 @@ export default function VideoView(props: NodeViewProps) {
|
|||||||
<Loader size={20} pos="absolute" top={6} right={6} />
|
<Loader size={20} pos="absolute" top={6} right={6} />
|
||||||
</Group>
|
</Group>
|
||||||
)}
|
)}
|
||||||
{!src && !previewSrc && (
|
{!src && !previewSrc && placeholder && (
|
||||||
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md">
|
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md">
|
||||||
<Loader size={20} style={{ flexShrink: 0 }} />
|
<Loader size={20} style={{ flexShrink: 0 }} />
|
||||||
<Text component="span" size="sm" truncate="end">
|
<Text component="span" size="sm" truncate="end">
|
||||||
@@ -69,6 +70,9 @@ export default function VideoView(props: NodeViewProps) {
|
|||||||
</Text>
|
</Text>
|
||||||
</Group>
|
</Group>
|
||||||
)}
|
)}
|
||||||
|
{!src && !previewSrc && !placeholder && (
|
||||||
|
<video className={classes.video} controls />
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</NodeViewWrapper>
|
</NodeViewWrapper>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -0,0 +1,105 @@
|
|||||||
|
// https://github.com/NiclasDev63/tiptap-extension-auto-joiner - MIT
|
||||||
|
import { Extension } from "@tiptap/core";
|
||||||
|
import { Plugin, PluginKey } from "@tiptap/pm/state";
|
||||||
|
import { canJoin } from "@tiptap/pm/transform";
|
||||||
|
import { getNodeType } from "@tiptap/react";
|
||||||
|
import { NodeType } from "@tiptap/pm/model";
|
||||||
|
import { Transaction } from "@tiptap/pm/state";
|
||||||
|
|
||||||
|
// https://discuss.prosemirror.net/t/how-to-autojoin-all-the-time/2957/4
|
||||||
|
// Adapted from prosemirror-commands wrapDispatchForJoin
|
||||||
|
function autoJoin(
|
||||||
|
transactions: readonly Transaction[],
|
||||||
|
newTr: Transaction,
|
||||||
|
nodeTypes: NodeType[]
|
||||||
|
) {
|
||||||
|
// Collect changed ranges across all transactions, mapping earlier ranges
|
||||||
|
// forward through later mappings so every position lands in newTr.doc space.
|
||||||
|
let ranges: number[] = [];
|
||||||
|
for (const tr of transactions) {
|
||||||
|
for (let i = 0; i < tr.mapping.maps.length; i++) {
|
||||||
|
let map = tr.mapping.maps[i];
|
||||||
|
if (!map) continue;
|
||||||
|
for (let j = 0; j < ranges.length; j++) ranges[j] = map.map(ranges[j]!);
|
||||||
|
map.forEach((_s, _e, from, to) => ranges.push(from, to));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Figure out which joinable points exist inside those ranges,
|
||||||
|
// by checking all node boundaries in their parent nodes.
|
||||||
|
// Resolve against newTr.doc — the same document we will join on.
|
||||||
|
let joinable: number[] = [];
|
||||||
|
for (let i = 0; i < ranges.length; i += 2) {
|
||||||
|
let from = ranges[i]!,
|
||||||
|
to = ranges[i + 1]!;
|
||||||
|
let $from = newTr.doc.resolve(from),
|
||||||
|
depth = $from.sharedDepth(to),
|
||||||
|
parent = $from.node(depth);
|
||||||
|
for (
|
||||||
|
let index = $from.indexAfter(depth), pos = $from.after(depth + 1);
|
||||||
|
pos <= to;
|
||||||
|
++index
|
||||||
|
) {
|
||||||
|
let after = parent.maybeChild(index);
|
||||||
|
if (!after) break;
|
||||||
|
if (index && joinable.indexOf(pos) == -1) {
|
||||||
|
let before = parent.child(index - 1);
|
||||||
|
if (before.type == after.type && nodeTypes.includes(before.type))
|
||||||
|
joinable.push(pos);
|
||||||
|
}
|
||||||
|
pos += after.nodeSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Join the joinable points (reverse order to preserve earlier positions)
|
||||||
|
let joined = false;
|
||||||
|
joinable.sort((a, b) => a - b);
|
||||||
|
for (let i = joinable.length - 1; i >= 0; i--) {
|
||||||
|
if (canJoin(newTr.doc, joinable[i]!)) {
|
||||||
|
newTr.join(joinable[i]!);
|
||||||
|
joined = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return joined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface AutoJoinerOptions {
|
||||||
|
elementsToJoin: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
const AutoJoiner = Extension.create<AutoJoinerOptions>({
|
||||||
|
name: "autoJoiner",
|
||||||
|
|
||||||
|
addOptions() {
|
||||||
|
return {
|
||||||
|
elementsToJoin: [],
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
addProseMirrorPlugins() {
|
||||||
|
const plugin = new PluginKey(this.name);
|
||||||
|
const joinableNodes = [
|
||||||
|
this.editor.schema.nodes.bulletList,
|
||||||
|
this.editor.schema.nodes.orderedList,
|
||||||
|
];
|
||||||
|
this.options.elementsToJoin.forEach((element) => {
|
||||||
|
const nodeTyp = getNodeType(element, this.editor.schema);
|
||||||
|
joinableNodes.push(nodeTyp);
|
||||||
|
});
|
||||||
|
|
||||||
|
return [
|
||||||
|
new Plugin({
|
||||||
|
key: plugin,
|
||||||
|
appendTransaction(transactions, _, newState) {
|
||||||
|
let newTr = newState.tr;
|
||||||
|
if (autoJoin(transactions, newTr, joinableNodes as NodeType[])) {
|
||||||
|
return newTr;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
];
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export default AutoJoiner;
|
||||||
@@ -30,6 +30,7 @@ import {
|
|||||||
TiptapImage,
|
TiptapImage,
|
||||||
Callout,
|
Callout,
|
||||||
TiptapVideo,
|
TiptapVideo,
|
||||||
|
TiptapAudio,
|
||||||
LinkExtension,
|
LinkExtension,
|
||||||
Selection,
|
Selection,
|
||||||
Attachment,
|
Attachment,
|
||||||
@@ -37,6 +38,7 @@ import {
|
|||||||
Drawio,
|
Drawio,
|
||||||
Excalidraw,
|
Excalidraw,
|
||||||
Embed,
|
Embed,
|
||||||
|
TiptapPdf,
|
||||||
SearchAndReplace,
|
SearchAndReplace,
|
||||||
Mention,
|
Mention,
|
||||||
TableDndExtension,
|
TableDndExtension,
|
||||||
@@ -47,7 +49,7 @@ import {
|
|||||||
SharedStorage,
|
SharedStorage,
|
||||||
Columns,
|
Columns,
|
||||||
Column,
|
Column,
|
||||||
Status
|
Status,
|
||||||
} from "@docmost/editor-ext";
|
} from "@docmost/editor-ext";
|
||||||
import {
|
import {
|
||||||
randomElement,
|
randomElement,
|
||||||
@@ -68,11 +70,13 @@ import ImageView from "@/features/editor/components/image/image-view.tsx";
|
|||||||
import CalloutView from "@/features/editor/components/callout/callout-view.tsx";
|
import CalloutView from "@/features/editor/components/callout/callout-view.tsx";
|
||||||
import StatusView from "@/features/editor/components/status/status-view.tsx";
|
import StatusView from "@/features/editor/components/status/status-view.tsx";
|
||||||
import VideoView from "@/features/editor/components/video/video-view.tsx";
|
import VideoView from "@/features/editor/components/video/video-view.tsx";
|
||||||
|
import AudioView from "@/features/editor/components/audio/audio-view.tsx";
|
||||||
import AttachmentView from "@/features/editor/components/attachment/attachment-view.tsx";
|
import AttachmentView from "@/features/editor/components/attachment/attachment-view.tsx";
|
||||||
import CodeBlockView from "@/features/editor/components/code-block/code-block-view.tsx";
|
import CodeBlockView from "@/features/editor/components/code-block/code-block-view.tsx";
|
||||||
import DrawioView from "../components/drawio/drawio-view";
|
import DrawioView from "../components/drawio/drawio-view";
|
||||||
import ExcalidrawView from "@/features/editor/components/excalidraw/excalidraw-view.tsx";
|
import ExcalidrawView from "@/features/editor/components/excalidraw/excalidraw-view.tsx";
|
||||||
import EmbedView from "@/features/editor/components/embed/embed-view.tsx";
|
import EmbedView from "@/features/editor/components/embed/embed-view.tsx";
|
||||||
|
import PdfView from "@/features/editor/components/pdf/pdf-view.tsx";
|
||||||
import SubpagesView from "@/features/editor/components/subpages/subpages-view.tsx";
|
import SubpagesView from "@/features/editor/components/subpages/subpages-view.tsx";
|
||||||
import { common, createLowlight } from "lowlight";
|
import { common, createLowlight } from "lowlight";
|
||||||
import plaintext from "highlight.js/lib/languages/plaintext";
|
import plaintext from "highlight.js/lib/languages/plaintext";
|
||||||
@@ -93,6 +97,7 @@ import i18n from "@/i18n.ts";
|
|||||||
import { MarkdownClipboard } from "@/features/editor/extensions/markdown-clipboard.ts";
|
import { MarkdownClipboard } from "@/features/editor/extensions/markdown-clipboard.ts";
|
||||||
import EmojiCommand from "./emoji-command";
|
import EmojiCommand from "./emoji-command";
|
||||||
import { countWords } from "alfaaz";
|
import { countWords } from "alfaaz";
|
||||||
|
import AutoJoiner from "@/features/editor/extensions/autojoiner.ts";
|
||||||
|
|
||||||
const lowlight = createLowlight(common);
|
const lowlight = createLowlight(common);
|
||||||
lowlight.register("mermaid", plaintext);
|
lowlight.register("mermaid", plaintext);
|
||||||
@@ -137,6 +142,25 @@ export const mainExtensions = [
|
|||||||
}),
|
}),
|
||||||
];
|
];
|
||||||
},
|
},
|
||||||
|
addKeyboardShortcuts() {
|
||||||
|
return {
|
||||||
|
Enter: ({ editor }) => {
|
||||||
|
const { from, to } = editor.state.selection;
|
||||||
|
if (from !== to) return false;
|
||||||
|
if (!editor.isActive("code")) return false;
|
||||||
|
|
||||||
|
const $from = editor.state.doc.resolve(from);
|
||||||
|
const codeType = editor.state.schema.marks.code;
|
||||||
|
const nodeAfter = $from.nodeAfter;
|
||||||
|
|
||||||
|
if (nodeAfter && codeType.isInSet(nodeAfter.marks)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return editor.chain().unsetCode().splitBlock().run();
|
||||||
|
},
|
||||||
|
};
|
||||||
|
},
|
||||||
}),
|
}),
|
||||||
SharedStorage,
|
SharedStorage,
|
||||||
Heading,
|
Heading,
|
||||||
@@ -248,8 +272,8 @@ export const mainExtensions = [
|
|||||||
resize: {
|
resize: {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
directions: ["left", "right"],
|
directions: ["left", "right"],
|
||||||
minWidth: 80,
|
minWidth: 24,
|
||||||
minHeight: 40,
|
minHeight: 16,
|
||||||
alwaysPreserveAspectRatio: true,
|
alwaysPreserveAspectRatio: true,
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
createCustomHandle: createImageHandle,
|
createCustomHandle: createImageHandle,
|
||||||
@@ -261,14 +285,17 @@ export const mainExtensions = [
|
|||||||
resize: {
|
resize: {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
directions: ["left", "right"],
|
directions: ["left", "right"],
|
||||||
minWidth: 80,
|
minWidth: 24,
|
||||||
minHeight: 40,
|
minHeight: 16,
|
||||||
alwaysPreserveAspectRatio: true,
|
alwaysPreserveAspectRatio: true,
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
createCustomHandle: createResizeHandle,
|
createCustomHandle: createResizeHandle,
|
||||||
className: buildResizeClasses("node-video"),
|
className: buildResizeClasses("node-video"),
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
|
TiptapAudio.configure({
|
||||||
|
view: AudioView,
|
||||||
|
}),
|
||||||
Callout.configure({
|
Callout.configure({
|
||||||
view: CalloutView,
|
view: CalloutView,
|
||||||
}),
|
}),
|
||||||
@@ -289,8 +316,8 @@ export const mainExtensions = [
|
|||||||
resize: {
|
resize: {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
directions: ["left", "right"],
|
directions: ["left", "right"],
|
||||||
minWidth: 80,
|
minWidth: 24,
|
||||||
minHeight: 40,
|
minHeight: 16,
|
||||||
alwaysPreserveAspectRatio: true,
|
alwaysPreserveAspectRatio: true,
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
createCustomHandle: createResizeHandle,
|
createCustomHandle: createResizeHandle,
|
||||||
@@ -302,8 +329,8 @@ export const mainExtensions = [
|
|||||||
resize: {
|
resize: {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
directions: ["left", "right"],
|
directions: ["left", "right"],
|
||||||
minWidth: 80,
|
minWidth: 24,
|
||||||
minHeight: 40,
|
minHeight: 16,
|
||||||
alwaysPreserveAspectRatio: true,
|
alwaysPreserveAspectRatio: true,
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
createCustomHandle: createResizeHandle,
|
createCustomHandle: createResizeHandle,
|
||||||
@@ -313,6 +340,9 @@ export const mainExtensions = [
|
|||||||
Embed.configure({
|
Embed.configure({
|
||||||
view: EmbedView,
|
view: EmbedView,
|
||||||
}),
|
}),
|
||||||
|
TiptapPdf.configure({
|
||||||
|
view: PdfView,
|
||||||
|
}),
|
||||||
Subpages.configure({
|
Subpages.configure({
|
||||||
view: SubpagesView,
|
view: SubpagesView,
|
||||||
}),
|
}),
|
||||||
@@ -343,6 +373,9 @@ export const mainExtensions = [
|
|||||||
}).configure(),
|
}).configure(),
|
||||||
Columns,
|
Columns,
|
||||||
Column,
|
Column,
|
||||||
|
AutoJoiner.configure({
|
||||||
|
elementsToJoin: [],
|
||||||
|
}),
|
||||||
] as any;
|
] as any;
|
||||||
|
|
||||||
type CollabExtensions = (provider: HocuspocusProvider, user: IUser) => any[];
|
type CollabExtensions = (provider: HocuspocusProvider, user: IUser) => any[];
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
// adapted from: https://github.com/aguingand/tiptap-markdown/blob/main/src/extensions/tiptap/clipboard.js - MIT
|
// adapted from: https://github.com/aguingand/tiptap-markdown/blob/main/src/extensions/tiptap/clipboard.js - MIT
|
||||||
import { Extension } from "@tiptap/core";
|
import { Extension } from "@tiptap/core";
|
||||||
import { Plugin, PluginKey } from "@tiptap/pm/state";
|
import { Plugin, PluginKey, TextSelection } from "@tiptap/pm/state";
|
||||||
import { DOMParser } from "@tiptap/pm/model";
|
import { DOMParser, DOMSerializer, Fragment, Slice } from "@tiptap/pm/model";
|
||||||
import { find } from "linkifyjs";
|
import { find } from "linkifyjs";
|
||||||
import { markdownToHtml } from "@docmost/editor-ext";
|
import { markdownToHtml, htmlToMarkdown } from "@docmost/editor-ext";
|
||||||
|
|
||||||
export const MarkdownClipboard = Extension.create({
|
export const MarkdownClipboard = Extension.create({
|
||||||
name: "markdownClipboard",
|
name: "markdownClipboard",
|
||||||
@@ -19,6 +19,27 @@ export const MarkdownClipboard = Extension.create({
|
|||||||
new Plugin({
|
new Plugin({
|
||||||
key: new PluginKey("markdownClipboard"),
|
key: new PluginKey("markdownClipboard"),
|
||||||
props: {
|
props: {
|
||||||
|
clipboardTextSerializer: (slice) => {
|
||||||
|
const listTypes = ["bulletList", "orderedList", "taskList"];
|
||||||
|
let topLevelCount = 0;
|
||||||
|
let hasList = false;
|
||||||
|
slice.content.forEach((node) => {
|
||||||
|
if (listTypes.includes(node.type.name)) {
|
||||||
|
hasList = true;
|
||||||
|
topLevelCount += node.childCount;
|
||||||
|
} else {
|
||||||
|
topLevelCount++;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!hasList || topLevelCount < 2) return null;
|
||||||
|
|
||||||
|
const div = document.createElement("div");
|
||||||
|
const serializer = DOMSerializer.fromSchema(this.editor.schema);
|
||||||
|
const fragment = serializer.serializeFragment(slice.content);
|
||||||
|
div.appendChild(fragment);
|
||||||
|
return htmlToMarkdown(div.innerHTML);
|
||||||
|
},
|
||||||
handlePaste: (view, event, slice) => {
|
handlePaste: (view, event, slice) => {
|
||||||
if (!event.clipboardData) {
|
if (!event.clipboardData) {
|
||||||
return false;
|
return false;
|
||||||
@@ -29,49 +50,80 @@ export const MarkdownClipboard = Extension.create({
|
|||||||
}
|
}
|
||||||
|
|
||||||
const text = event.clipboardData.getData("text/plain");
|
const text = event.clipboardData.getData("text/plain");
|
||||||
|
const html = event.clipboardData.getData("text/html");
|
||||||
const vscode = event.clipboardData.getData("vscode-editor-data");
|
const vscode = event.clipboardData.getData("vscode-editor-data");
|
||||||
const vscodeData = vscode ? JSON.parse(vscode) : undefined;
|
const vscodeData = vscode ? JSON.parse(vscode) : undefined;
|
||||||
const language = vscodeData?.mode;
|
const language = vscodeData?.mode;
|
||||||
|
|
||||||
if (language !== "markdown") {
|
const isVscodeMarkdown = language === "markdown";
|
||||||
|
const isPlainTextOnly = !html && !vscode && !!text;
|
||||||
|
|
||||||
|
if (!isVscodeMarkdown && !isPlainTextOnly) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (isPlainTextOnly) {
|
||||||
|
if ((view as any).input?.shiftKey || !this.options.transformPastedText) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const link = find(text, {
|
||||||
|
defaultProtocol: "http",
|
||||||
|
}).find((item) => item.isLink && item.value === text);
|
||||||
|
|
||||||
|
if (link) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const { tr } = view.state;
|
const { tr } = view.state;
|
||||||
const { from, to } = view.state.selection;
|
const { from, to } = view.state.selection;
|
||||||
|
|
||||||
const html = markdownToHtml(text);
|
const parsed = markdownToHtml(text.replace(/\n+$/, ""));
|
||||||
|
|
||||||
const contentNodes = DOMParser.fromSchema(
|
const contentNodes = DOMParser.fromSchema(
|
||||||
this.editor.schema,
|
this.editor.schema,
|
||||||
).parseSlice(elementFromString(html), {
|
).parseSlice(elementFromString(parsed), {
|
||||||
preserveWhitespace: true,
|
preserveWhitespace: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
tr.replaceRange(from, to, contentNodes);
|
tr.replaceRange(from, to, contentNodes);
|
||||||
|
const insertEnd = tr.mapping.map(from, 1);
|
||||||
|
tr.setSelection(TextSelection.near(tr.doc.resolve(Math.max(from, insertEnd - 2)), -1));
|
||||||
tr.setMeta('paste', true)
|
tr.setMeta('paste', true)
|
||||||
view.dispatch(tr);
|
view.dispatch(tr);
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
clipboardTextParser: (text, context, plainText) => {
|
// Strip trailing whitespace-only paragraphs from pasted content.
|
||||||
const link = find(text, {
|
// Terminals (GNOME Terminal, etc.) often include trailing
|
||||||
defaultProtocol: "http",
|
// whitespace in their HTML clipboard data, which ProseMirror
|
||||||
}).find((item) => item.isLink && item.value === text);
|
// parses as an extra paragraph. Inside a list item this creates
|
||||||
|
// an orphan empty line that breaks the list structure.
|
||||||
|
transformPasted: (slice) => {
|
||||||
|
let { content, openStart, openEnd } = slice;
|
||||||
|
|
||||||
if (plainText || !this.options.transformPastedText || link) {
|
// Remove trailing paragraphs that contain only whitespace
|
||||||
// don't parse plaintext link to allow link paste handler to work
|
while (content.childCount > 1) {
|
||||||
// pasting with shift key prevents formatting
|
const lastChild = content.lastChild;
|
||||||
return null;
|
if (
|
||||||
|
lastChild?.type.name === "paragraph" &&
|
||||||
|
lastChild.textContent.trim() === ""
|
||||||
|
) {
|
||||||
|
const children = [];
|
||||||
|
for (let i = 0; i < content.childCount - 1; i++) {
|
||||||
|
children.push(content.child(i));
|
||||||
|
}
|
||||||
|
content = Fragment.from(children);
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const parsed = markdownToHtml(text);
|
if (content !== slice.content) {
|
||||||
return DOMParser.fromSchema(this.editor.schema).parseSlice(
|
return new Slice(content, openStart, Math.max(openEnd, 1));
|
||||||
elementFromString(parsed),
|
}
|
||||||
{
|
|
||||||
preserveWhitespace: true,
|
return slice;
|
||||||
context,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ export interface FullEditorProps {
|
|||||||
content: string;
|
content: string;
|
||||||
spaceSlug: string;
|
spaceSlug: string;
|
||||||
editable: boolean;
|
editable: boolean;
|
||||||
|
canComment?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function FullEditor({
|
export function FullEditor({
|
||||||
@@ -25,6 +26,7 @@ export function FullEditor({
|
|||||||
content,
|
content,
|
||||||
spaceSlug,
|
spaceSlug,
|
||||||
editable,
|
editable,
|
||||||
|
canComment,
|
||||||
}: FullEditorProps) {
|
}: FullEditorProps) {
|
||||||
const [user] = useAtom(userAtom);
|
const [user] = useAtom(userAtom);
|
||||||
const fullPageWidth = user.settings?.preferences?.fullPageWidth;
|
const fullPageWidth = user.settings?.preferences?.fullPageWidth;
|
||||||
@@ -46,6 +48,7 @@ export function FullEditor({
|
|||||||
pageId={pageId}
|
pageId={pageId}
|
||||||
editable={editable}
|
editable={editable}
|
||||||
content={content}
|
content={content}
|
||||||
|
canComment={canComment}
|
||||||
/>
|
/>
|
||||||
</Container>
|
</Container>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -37,14 +37,17 @@ import { asideStateAtom } from "@/components/layouts/global/hooks/atoms/sidebar-
|
|||||||
import {
|
import {
|
||||||
activeCommentIdAtom,
|
activeCommentIdAtom,
|
||||||
showCommentPopupAtom,
|
showCommentPopupAtom,
|
||||||
|
showReadOnlyCommentPopupAtom,
|
||||||
} from "@/features/comment/atoms/comment-atom";
|
} from "@/features/comment/atoms/comment-atom";
|
||||||
import CommentDialog from "@/features/comment/components/comment-dialog";
|
import CommentDialog from "@/features/comment/components/comment-dialog";
|
||||||
import { EditorBubbleMenu } from "@/features/editor/components/bubble-menu/bubble-menu";
|
import { EditorBubbleMenu } from "@/features/editor/components/bubble-menu/bubble-menu";
|
||||||
|
import { ReadonlyBubbleMenu } from "@/features/editor/components/bubble-menu/readonly-bubble-menu";
|
||||||
import TableCellMenu from "@/features/editor/components/table/table-cell-menu.tsx";
|
import TableCellMenu from "@/features/editor/components/table/table-cell-menu.tsx";
|
||||||
import TableMenu from "@/features/editor/components/table/table-menu.tsx";
|
import TableMenu from "@/features/editor/components/table/table-menu.tsx";
|
||||||
import ImageMenu from "@/features/editor/components/image/image-menu.tsx";
|
import ImageMenu from "@/features/editor/components/image/image-menu.tsx";
|
||||||
import CalloutMenu from "@/features/editor/components/callout/callout-menu.tsx";
|
import CalloutMenu from "@/features/editor/components/callout/callout-menu.tsx";
|
||||||
import VideoMenu from "@/features/editor/components/video/video-menu.tsx";
|
import VideoMenu from "@/features/editor/components/video/video-menu.tsx";
|
||||||
|
import PdfMenu from "@/features/editor/components/pdf/pdf-menu.tsx";
|
||||||
import SubpagesMenu from "@/features/editor/components/subpages/subpages-menu.tsx";
|
import SubpagesMenu from "@/features/editor/components/subpages/subpages-menu.tsx";
|
||||||
import {
|
import {
|
||||||
handleFileDrop,
|
handleFileDrop,
|
||||||
@@ -73,12 +76,14 @@ interface PageEditorProps {
|
|||||||
pageId: string;
|
pageId: string;
|
||||||
editable: boolean;
|
editable: boolean;
|
||||||
content: any;
|
content: any;
|
||||||
|
canComment?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function PageEditor({
|
export default function PageEditor({
|
||||||
pageId,
|
pageId,
|
||||||
editable,
|
editable,
|
||||||
content,
|
content,
|
||||||
|
canComment,
|
||||||
}: PageEditorProps) {
|
}: PageEditorProps) {
|
||||||
const collaborationURL = useCollaborationUrl();
|
const collaborationURL = useCollaborationUrl();
|
||||||
const isComponentMounted = useRef(false);
|
const isComponentMounted = useRef(false);
|
||||||
@@ -93,6 +98,7 @@ export default function PageEditor({
|
|||||||
const [, setAsideState] = useAtom(asideStateAtom);
|
const [, setAsideState] = useAtom(asideStateAtom);
|
||||||
const [, setActiveCommentId] = useAtom(activeCommentIdAtom);
|
const [, setActiveCommentId] = useAtom(activeCommentIdAtom);
|
||||||
const [showCommentPopup, setShowCommentPopup] = useAtom(showCommentPopupAtom);
|
const [showCommentPopup, setShowCommentPopup] = useAtom(showCommentPopupAtom);
|
||||||
|
const [showReadOnlyCommentPopup] = useAtom(showReadOnlyCommentPopupAtom);
|
||||||
const [isLocalSynced, setIsLocalSynced] = useState(false);
|
const [isLocalSynced, setIsLocalSynced] = useState(false);
|
||||||
const [isRemoteSynced, setIsRemoteSynced] = useState(false);
|
const [isRemoteSynced, setIsRemoteSynced] = useState(false);
|
||||||
const [yjsConnectionStatus, setYjsConnectionStatus] = useAtom(
|
const [yjsConnectionStatus, setYjsConnectionStatus] = useAtom(
|
||||||
@@ -414,6 +420,7 @@ export default function PageEditor({
|
|||||||
<TableCellMenu editor={editor} appendTo={menuContainerRef} />
|
<TableCellMenu editor={editor} appendTo={menuContainerRef} />
|
||||||
<ImageMenu editor={editor} />
|
<ImageMenu editor={editor} />
|
||||||
<VideoMenu editor={editor} />
|
<VideoMenu editor={editor} />
|
||||||
|
<PdfMenu editor={editor} />
|
||||||
<CalloutMenu editor={editor} />
|
<CalloutMenu editor={editor} />
|
||||||
<SubpagesMenu editor={editor} />
|
<SubpagesMenu editor={editor} />
|
||||||
<ExcalidrawMenu editor={editor} />
|
<ExcalidrawMenu editor={editor} />
|
||||||
@@ -421,7 +428,13 @@ export default function PageEditor({
|
|||||||
<ColumnsMenu editor={editor} />
|
<ColumnsMenu editor={editor} />
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
{editor && !editorIsEditable && (editable || canComment) && providersRef.current && (
|
||||||
|
<ReadonlyBubbleMenu editor={editor} />
|
||||||
|
)}
|
||||||
{showCommentPopup && <CommentDialog editor={editor} pageId={pageId} />}
|
{showCommentPopup && <CommentDialog editor={editor} pageId={pageId} />}
|
||||||
|
{showReadOnlyCommentPopup && (
|
||||||
|
<CommentDialog editor={editor} pageId={pageId} readOnly />
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
onClick={() => editor.commands.focus("end")}
|
onClick={() => editor.commands.focus("end")}
|
||||||
|
|||||||
@@ -133,10 +133,18 @@
|
|||||||
border-top: 1px solid #68cef8;
|
border-top: 1px solid #68cef8;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&[contenteditable="false"] hr.ProseMirror-selectednode {
|
||||||
|
border-top: none;
|
||||||
|
}
|
||||||
|
|
||||||
.ProseMirror-selectednode {
|
.ProseMirror-selectednode {
|
||||||
outline: 2px solid #70cff8;
|
outline: 2px solid #70cff8;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&[contenteditable="false"] .ProseMirror-selectednode {
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
& > .react-renderer {
|
& > .react-renderer {
|
||||||
margin-top: var(--mantine-spacing-sm);
|
margin-top: var(--mantine-spacing-sm);
|
||||||
margin-bottom: var(--mantine-spacing-sm);
|
margin-bottom: var(--mantine-spacing-sm);
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.node-image, .node-video, .node-excalidraw, .node-drawio {
|
.node-image, .node-video, .node-pdf, .node-excalidraw, .node-drawio {
|
||||||
&.ProseMirror-selectednode {
|
&.ProseMirror-selectednode {
|
||||||
outline: none;
|
outline: none;
|
||||||
}
|
}
|
||||||
@@ -37,5 +37,28 @@
|
|||||||
font-size: var(--mantine-font-size-md);
|
font-size: var(--mantine-font-size-md);
|
||||||
line-height: var(--mantine-line-height-md);
|
line-height: var(--mantine-line-height-md);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.media-pulse {
|
||||||
|
animation: media-pulse 1.2s ease-in-out infinite;
|
||||||
|
|
||||||
|
@mixin light {
|
||||||
|
background: linear-gradient(-90deg, var(--mantine-color-gray-3) 0%, var(--mantine-color-gray-1) 50%, var(--mantine-color-gray-3) 100%);
|
||||||
|
background-size: 400% 400%;
|
||||||
|
}
|
||||||
|
|
||||||
|
@mixin dark {
|
||||||
|
background: linear-gradient(-90deg, var(--mantine-color-dark-6) 0%, var(--mantine-color-dark-5) 50%, var(--mantine-color-dark-6) 100%);
|
||||||
|
background-size: 400% 400%;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes media-pulse {
|
||||||
|
0% {
|
||||||
|
background-position: 0% 0%;
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
background-position: -135% 0%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ import {
|
|||||||
import { CustomAvatar } from "@/components/ui/custom-avatar";
|
import { CustomAvatar } from "@/components/ui/custom-avatar";
|
||||||
import { INotification } from "../types/notification.types";
|
import { INotification } from "../types/notification.types";
|
||||||
import { Trans, useTranslation } from "react-i18next";
|
import { Trans, useTranslation } from "react-i18next";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { Link } from "react-router-dom";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
import { useMarkReadMutation } from "../queries/notification-query";
|
import { useMarkReadMutation } from "../queries/notification-query";
|
||||||
import { buildPageUrl } from "@/features/page/page.utils";
|
import { buildPageUrl } from "@/features/page/page.utils";
|
||||||
@@ -30,7 +30,6 @@ export function NotificationItem({
|
|||||||
onNavigate,
|
onNavigate,
|
||||||
}: NotificationItemProps) {
|
}: NotificationItemProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const navigate = useNavigate();
|
|
||||||
const markRead = useMarkReadMutation();
|
const markRead = useMarkReadMutation();
|
||||||
const [hovered, setHovered] = useState(false);
|
const [hovered, setHovered] = useState(false);
|
||||||
|
|
||||||
@@ -50,37 +49,47 @@ export function NotificationItem({
|
|||||||
return notification.data?.role === "writer"
|
return notification.data?.role === "writer"
|
||||||
? "<bold>{{name}}</bold> gave you edit access to a page"
|
? "<bold>{{name}}</bold> gave you edit access to a page"
|
||||||
: "<bold>{{name}}</bold> gave you view access to a page";
|
: "<bold>{{name}}</bold> gave you view access to a page";
|
||||||
|
case "page.updated":
|
||||||
|
return "<bold>{{name}}</bold> updated a page";
|
||||||
default:
|
default:
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleClick = () => {
|
const pageUrl =
|
||||||
if (notification.page && notification.space) {
|
notification.page && notification.space
|
||||||
if (isUnread) {
|
? buildPageUrl(
|
||||||
markRead.mutate([notification.id]);
|
|
||||||
}
|
|
||||||
navigate(
|
|
||||||
buildPageUrl(
|
|
||||||
notification.space.slug,
|
notification.space.slug,
|
||||||
notification.page.slugId,
|
notification.page.slugId,
|
||||||
notification.page.title,
|
notification.page.title,
|
||||||
),
|
)
|
||||||
);
|
: undefined;
|
||||||
onNavigate();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleMarkRead = (e: React.MouseEvent) => {
|
const markReadIfNeeded = () => {
|
||||||
e.stopPropagation();
|
|
||||||
if (isUnread) {
|
if (isUnread) {
|
||||||
markRead.mutate([notification.id]);
|
markRead.mutate([notification.id]);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleClick = () => {
|
||||||
|
markReadIfNeeded();
|
||||||
|
onNavigate();
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleMarkRead = (e: React.MouseEvent) => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
markReadIfNeeded();
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<UnstyledButton
|
<UnstyledButton
|
||||||
|
component={Link}
|
||||||
|
to={pageUrl ?? ""}
|
||||||
onClick={handleClick}
|
onClick={handleClick}
|
||||||
|
// auxclick fires for all non-primary buttons; guard to middle-click only (button 1)
|
||||||
|
// so that right-click (button 2, context menu) does not mark as read
|
||||||
|
onAuxClick={(e: React.MouseEvent) => e.button === 1 && markReadIfNeeded()}
|
||||||
onMouseEnter={() => setHovered(true)}
|
onMouseEnter={() => setHovered(true)}
|
||||||
onMouseLeave={() => setHovered(false)}
|
onMouseLeave={() => setHovered(false)}
|
||||||
w="100%"
|
w="100%"
|
||||||
|
|||||||
@@ -3,17 +3,23 @@ import { IconBellOff } from "@tabler/icons-react";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { useEffect, useRef } from "react";
|
import { useEffect, useRef } from "react";
|
||||||
import { NotificationItem } from "./notification-item";
|
import { NotificationItem } from "./notification-item";
|
||||||
import { INotification, NotificationFilter } from "../types/notification.types";
|
import {
|
||||||
|
INotification,
|
||||||
|
NotificationFilter,
|
||||||
|
NotificationTab,
|
||||||
|
} from "../types/notification.types";
|
||||||
import { groupNotificationsByTime } from "../notification.utils";
|
import { groupNotificationsByTime } from "../notification.utils";
|
||||||
import { useNotificationsQuery } from "../queries/notification-query";
|
import { useNotificationsQuery } from "../queries/notification-query";
|
||||||
import classes from "../notification.module.css";
|
import classes from "../notification.module.css";
|
||||||
|
|
||||||
type NotificationListProps = {
|
type NotificationListProps = {
|
||||||
|
tab: NotificationTab;
|
||||||
filter: NotificationFilter;
|
filter: NotificationFilter;
|
||||||
onNavigate: () => void;
|
onNavigate: () => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
export function NotificationList({
|
export function NotificationList({
|
||||||
|
tab,
|
||||||
filter,
|
filter,
|
||||||
onNavigate,
|
onNavigate,
|
||||||
}: NotificationListProps) {
|
}: NotificationListProps) {
|
||||||
@@ -24,7 +30,7 @@ export function NotificationList({
|
|||||||
hasNextPage,
|
hasNextPage,
|
||||||
fetchNextPage,
|
fetchNextPage,
|
||||||
isFetchingNextPage,
|
isFetchingNextPage,
|
||||||
} = useNotificationsQuery();
|
} = useNotificationsQuery(tab as string);
|
||||||
|
|
||||||
const sentinelRef = useRef<HTMLDivElement>(null);
|
const sentinelRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import {
|
|||||||
Menu,
|
Menu,
|
||||||
Popover,
|
Popover,
|
||||||
ScrollArea,
|
ScrollArea,
|
||||||
|
Tabs,
|
||||||
Text,
|
Text,
|
||||||
Tooltip,
|
Tooltip,
|
||||||
} from "@mantine/core";
|
} from "@mantine/core";
|
||||||
@@ -18,15 +19,20 @@ import {
|
|||||||
} from "@tabler/icons-react";
|
} from "@tabler/icons-react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { NotificationList } from "./notification-list";
|
import { NotificationList } from "./notification-list";
|
||||||
import { NotificationFilter } from "../types/notification.types";
|
import {
|
||||||
|
NotificationFilter,
|
||||||
|
NotificationTab,
|
||||||
|
} from "../types/notification.types";
|
||||||
import {
|
import {
|
||||||
useMarkAllReadMutation,
|
useMarkAllReadMutation,
|
||||||
useUnreadCountQuery,
|
useUnreadCountQuery,
|
||||||
} from "../queries/notification-query";
|
} from "../queries/notification-query";
|
||||||
|
import classes from "../notification.module.css";
|
||||||
|
|
||||||
export function NotificationPopover() {
|
export function NotificationPopover() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [opened, setOpened] = useState(false);
|
const [opened, setOpened] = useState(false);
|
||||||
|
const [tab, setTab] = useState<NotificationTab>("direct");
|
||||||
const [filter, setFilter] = useState<NotificationFilter>("all");
|
const [filter, setFilter] = useState<NotificationFilter>("all");
|
||||||
|
|
||||||
const { data: unreadData } = useUnreadCountQuery();
|
const { data: unreadData } = useUnreadCountQuery();
|
||||||
@@ -125,13 +131,27 @@ export function NotificationPopover() {
|
|||||||
</Group>
|
</Group>
|
||||||
</Group>
|
</Group>
|
||||||
|
|
||||||
|
<Tabs
|
||||||
|
value={tab}
|
||||||
|
onChange={(value) => setTab(value as NotificationTab)}
|
||||||
|
variant="default"
|
||||||
|
color="dark"
|
||||||
|
>
|
||||||
|
<Tabs.List px="md">
|
||||||
|
<Tabs.Tab value="direct">{t("Direct")}</Tabs.Tab>
|
||||||
|
<Tabs.Tab value="updates">{t("Updates")}</Tabs.Tab>
|
||||||
|
</Tabs.List>
|
||||||
|
</Tabs>
|
||||||
|
|
||||||
<ScrollArea.Autosize
|
<ScrollArea.Autosize
|
||||||
mah={500}
|
mah={500}
|
||||||
type="auto"
|
type="auto"
|
||||||
offsetScrollbars
|
offsetScrollbars
|
||||||
scrollbarSize={6}
|
scrollbarSize={6}
|
||||||
|
style={{ overscrollBehavior: "contain" }}
|
||||||
>
|
>
|
||||||
<NotificationList
|
<NotificationList
|
||||||
|
tab={tab}
|
||||||
filter={filter}
|
filter={filter}
|
||||||
onNavigate={() => setOpened(false)}
|
onNavigate={() => setOpened(false)}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
.notificationItem {
|
.notificationItem {
|
||||||
|
display: block;
|
||||||
padding: 8px 12px;
|
padding: 8px 12px;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
user-select: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.notificationItem:hover {
|
.notificationItem:hover {
|
||||||
@@ -11,3 +13,4 @@
|
|||||||
.divider {
|
.divider {
|
||||||
border-color: light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
border-color: light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -15,10 +15,10 @@ import {
|
|||||||
export const NOTIFICATION_KEY = ["notifications"];
|
export const NOTIFICATION_KEY = ["notifications"];
|
||||||
export const UNREAD_COUNT_KEY = ["notifications", "unread-count"];
|
export const UNREAD_COUNT_KEY = ["notifications", "unread-count"];
|
||||||
|
|
||||||
export function useNotificationsQuery() {
|
export function useNotificationsQuery(type?: string) {
|
||||||
return useInfiniteQuery({
|
return useInfiniteQuery({
|
||||||
queryKey: NOTIFICATION_KEY,
|
queryKey: [...NOTIFICATION_KEY, type],
|
||||||
queryFn: ({ pageParam }) => getNotifications({ cursor: pageParam }),
|
queryFn: ({ pageParam }) => getNotifications({ cursor: pageParam, type }),
|
||||||
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,
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import { IPagination } from "@/lib/types";
|
|||||||
export async function getNotifications(params: {
|
export async function getNotifications(params: {
|
||||||
limit?: number;
|
limit?: number;
|
||||||
cursor?: string;
|
cursor?: string;
|
||||||
|
type?: string;
|
||||||
}): Promise<IPagination<INotification>> {
|
}): Promise<IPagination<INotification>> {
|
||||||
const req = await api.post<IPagination<INotification>>(
|
const req = await api.post<IPagination<INotification>>(
|
||||||
"/notifications",
|
"/notifications",
|
||||||
|
|||||||
@@ -3,7 +3,8 @@ export type NotificationType =
|
|||||||
| "comment.created"
|
| "comment.created"
|
||||||
| "comment.resolved"
|
| "comment.resolved"
|
||||||
| "page.user_mention"
|
| "page.user_mention"
|
||||||
| "page.permission_granted";
|
| "page.permission_granted"
|
||||||
|
| "page.updated";
|
||||||
|
|
||||||
export type INotification = {
|
export type INotification = {
|
||||||
id: string;
|
id: string;
|
||||||
@@ -38,3 +39,5 @@ export type INotification = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export type NotificationFilter = "all" | "unread";
|
export type NotificationFilter = "all" | "unread";
|
||||||
|
|
||||||
|
export type NotificationTab = "direct" | "updates" | "all";
|
||||||
|
|||||||
@@ -3,6 +3,8 @@ import {
|
|||||||
IconArrowRight,
|
IconArrowRight,
|
||||||
IconArrowsHorizontal,
|
IconArrowsHorizontal,
|
||||||
IconDots,
|
IconDots,
|
||||||
|
IconEye,
|
||||||
|
IconEyeOff,
|
||||||
IconFileExport,
|
IconFileExport,
|
||||||
IconHistory,
|
IconHistory,
|
||||||
IconLink,
|
IconLink,
|
||||||
@@ -40,6 +42,11 @@ import { PageStateSegmentedControl } from "@/features/user/components/page-state
|
|||||||
import MovePageModal from "@/features/page/components/move-page-modal.tsx";
|
import MovePageModal from "@/features/page/components/move-page-modal.tsx";
|
||||||
import { useTimeAgo } from "@/hooks/use-time-ago.tsx";
|
import { useTimeAgo } from "@/hooks/use-time-ago.tsx";
|
||||||
import { PageShareModal } from "@/ee/page-permission";
|
import { PageShareModal } from "@/ee/page-permission";
|
||||||
|
import {
|
||||||
|
useWatchStatusQuery,
|
||||||
|
useWatchPageMutation,
|
||||||
|
useUnwatchPageMutation,
|
||||||
|
} from "@/features/page/queries/watcher-query";
|
||||||
|
|
||||||
interface PageHeaderMenuProps {
|
interface PageHeaderMenuProps {
|
||||||
readOnly?: boolean;
|
readOnly?: boolean;
|
||||||
@@ -123,6 +130,9 @@ 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 { data: watchStatus } = useWatchStatusQuery(page?.id);
|
||||||
|
const watchPage = useWatchPageMutation();
|
||||||
|
const unwatchPage = useUnwatchPageMutation();
|
||||||
|
|
||||||
const handleCopyLink = () => {
|
const handleCopyLink = () => {
|
||||||
const pageUrl =
|
const pageUrl =
|
||||||
@@ -185,6 +195,23 @@ function PageActionMenu({ readOnly }: PageActionMenuProps) {
|
|||||||
>
|
>
|
||||||
{t("Copy as Markdown")}
|
{t("Copy as Markdown")}
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
|
|
||||||
|
{watchStatus?.watching ? (
|
||||||
|
<Menu.Item
|
||||||
|
leftSection={<IconEyeOff size={16} />}
|
||||||
|
onClick={() => unwatchPage.mutate(page.id)}
|
||||||
|
>
|
||||||
|
{t("Stop watching")}
|
||||||
|
</Menu.Item>
|
||||||
|
) : (
|
||||||
|
<Menu.Item
|
||||||
|
leftSection={<IconEye size={16} />}
|
||||||
|
onClick={() => watchPage.mutate(page.id)}
|
||||||
|
>
|
||||||
|
{t("Watch page")}
|
||||||
|
</Menu.Item>
|
||||||
|
)}
|
||||||
|
|
||||||
<Menu.Divider />
|
<Menu.Divider />
|
||||||
|
|
||||||
<Menu.Item leftSection={<IconArrowsHorizontal size={16} />}>
|
<Menu.Item leftSection={<IconArrowsHorizontal size={16} />}>
|
||||||
|
|||||||
@@ -110,15 +110,7 @@ export function useUpdatePageMutation() {
|
|||||||
return useMutation<IPage, Error, Partial<IPageInput>>({
|
return useMutation<IPage, Error, Partial<IPageInput>>({
|
||||||
mutationFn: (data) => updatePage(data),
|
mutationFn: (data) => updatePage(data),
|
||||||
onSuccess: (data) => {
|
onSuccess: (data) => {
|
||||||
updatePage(data);
|
updatePageData(data);
|
||||||
|
|
||||||
invalidateOnUpdatePage(
|
|
||||||
data.spaceId,
|
|
||||||
data.parentPageId,
|
|
||||||
data.id,
|
|
||||||
data.title,
|
|
||||||
data.icon,
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,43 @@
|
|||||||
|
import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query";
|
||||||
|
import {
|
||||||
|
watchPage,
|
||||||
|
unwatchPage,
|
||||||
|
getWatchStatus,
|
||||||
|
} from "@/features/page/services/watcher-service";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
|
const WATCHER_KEY = "watcher";
|
||||||
|
|
||||||
|
export function useWatchStatusQuery(pageId: string) {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: [WATCHER_KEY, pageId],
|
||||||
|
queryFn: () => getWatchStatus(pageId),
|
||||||
|
enabled: !!pageId,
|
||||||
|
staleTime: 60_000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useWatchPageMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: (pageId: string) => watchPage(pageId),
|
||||||
|
onSuccess: (_data, pageId) => {
|
||||||
|
queryClient.setQueryData([WATCHER_KEY, pageId], { watching: true });
|
||||||
|
notifications.show({ message: t("You are now watching this page") });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useUnwatchPageMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: (pageId: string) => unwatchPage(pageId),
|
||||||
|
onSuccess: (_data, pageId) => {
|
||||||
|
queryClient.setQueryData([WATCHER_KEY, pageId], { watching: false });
|
||||||
|
notifications.show({ message: t("You are no longer watching this page") });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
import api from "@/lib/api-client";
|
||||||
|
|
||||||
|
export async function watchPage(pageId: string): Promise<{ watching: boolean }> {
|
||||||
|
const req = await api.post<{ watching: boolean }>("/pages/watch", { pageId });
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function unwatchPage(pageId: string): Promise<{ watching: boolean }> {
|
||||||
|
const req = await api.post<{ watching: boolean }>("/pages/unwatch", { pageId });
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getWatchStatus(pageId: string): Promise<{ watching: boolean }> {
|
||||||
|
const req = await api.post<{ watching: boolean }>("/pages/watch-status", { pageId });
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
@@ -0,0 +1,165 @@
|
|||||||
|
import { useState } from "react";
|
||||||
|
import {
|
||||||
|
Button,
|
||||||
|
Divider,
|
||||||
|
Group,
|
||||||
|
Skeleton,
|
||||||
|
Stack,
|
||||||
|
Table,
|
||||||
|
Text,
|
||||||
|
} from "@mantine/core";
|
||||||
|
import { IconDevices } from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import {
|
||||||
|
useGetSessionsQuery,
|
||||||
|
useRevokeSessionMutation,
|
||||||
|
useRevokeAllSessionsMutation,
|
||||||
|
} from "@/features/session/queries/session-query";
|
||||||
|
import { formattedDate } from "@/lib/time";
|
||||||
|
|
||||||
|
const PAGE_SIZE = 5;
|
||||||
|
|
||||||
|
export default function SessionList() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { data: sessions, isLoading } = useGetSessionsQuery();
|
||||||
|
const revokeSessionMutation = useRevokeSessionMutation();
|
||||||
|
const revokeAllSessionsMutation = useRevokeAllSessionsMutation();
|
||||||
|
const [visibleCount, setVisibleCount] = useState(PAGE_SIZE);
|
||||||
|
|
||||||
|
const otherSessions = sessions?.filter((s) => !s?.isCurrentDevice) ?? [];
|
||||||
|
const visibleSessions = sessions?.slice(0, visibleCount) ?? [];
|
||||||
|
const hasMore = sessions && visibleCount < sessions.length;
|
||||||
|
|
||||||
|
if (isLoading) {
|
||||||
|
return (
|
||||||
|
<Table verticalSpacing="md">
|
||||||
|
<Table.Thead>
|
||||||
|
<Table.Tr>
|
||||||
|
<Table.Th>{t("Device Name")}</Table.Th>
|
||||||
|
<Table.Th>{t("Last Active")}</Table.Th>
|
||||||
|
<Table.Th />
|
||||||
|
</Table.Tr>
|
||||||
|
</Table.Thead>
|
||||||
|
<Table.Tbody>
|
||||||
|
{[1, 2, 3].map((i) => (
|
||||||
|
<Table.Tr key={i}>
|
||||||
|
<Table.Td>
|
||||||
|
<Group gap="xs">
|
||||||
|
<Skeleton height={18} width={18} radius="sm" />
|
||||||
|
<Skeleton height={14} width={140} radius="xs" />
|
||||||
|
</Group>
|
||||||
|
</Table.Td>
|
||||||
|
<Table.Td>
|
||||||
|
<Skeleton height={14} width={120} radius="xs" />
|
||||||
|
</Table.Td>
|
||||||
|
<Table.Td>
|
||||||
|
<Skeleton height={30} width={70} radius="sm" />
|
||||||
|
</Table.Td>
|
||||||
|
</Table.Tr>
|
||||||
|
))}
|
||||||
|
</Table.Tbody>
|
||||||
|
</Table>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack>
|
||||||
|
{otherSessions.length > 0 && (
|
||||||
|
<>
|
||||||
|
<div>
|
||||||
|
<Text fw={500}>{t("Log out of all devices")}</Text>
|
||||||
|
<Group justify="space-between" align="center" mt={4}>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t(
|
||||||
|
"Log out of all sessions except this device",
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
color="red"
|
||||||
|
size="xs"
|
||||||
|
loading={revokeAllSessionsMutation.isPending}
|
||||||
|
onClick={() => revokeAllSessionsMutation.mutate()}
|
||||||
|
>
|
||||||
|
{t("Log out of all devices")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
</div>
|
||||||
|
<Divider />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<Table verticalSpacing="md">
|
||||||
|
<Table.Thead>
|
||||||
|
<Table.Tr>
|
||||||
|
<Table.Th>{t("Device Name")}</Table.Th>
|
||||||
|
<Table.Th>{t("Last Active")}</Table.Th>
|
||||||
|
{otherSessions.length > 0 && <Table.Th />}
|
||||||
|
</Table.Tr>
|
||||||
|
</Table.Thead>
|
||||||
|
<Table.Tbody>
|
||||||
|
{visibleSessions.map((session) => (
|
||||||
|
<Table.Tr key={session.id}>
|
||||||
|
<Table.Td>
|
||||||
|
<Group gap="xs">
|
||||||
|
<IconDevices size={18} stroke={1.5} />
|
||||||
|
<div>
|
||||||
|
<Text size="sm">
|
||||||
|
{session.deviceName || t("Unknown device")}
|
||||||
|
</Text>
|
||||||
|
{session?.isCurrentDevice && (
|
||||||
|
<Text size="xs" c="blue">
|
||||||
|
{t("This Device")}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</Group>
|
||||||
|
</Table.Td>
|
||||||
|
<Table.Td>
|
||||||
|
<Text size="sm">
|
||||||
|
{session?.isCurrentDevice
|
||||||
|
? t("Now")
|
||||||
|
: formattedDate(new Date(session.lastActiveAt))}
|
||||||
|
</Text>
|
||||||
|
</Table.Td>
|
||||||
|
{otherSessions.length > 0 && (
|
||||||
|
<Table.Td>
|
||||||
|
{!session?.isCurrentDevice && (
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
size="xs"
|
||||||
|
loading={revokeSessionMutation.isPending}
|
||||||
|
onClick={() =>
|
||||||
|
revokeSessionMutation.mutate({
|
||||||
|
sessionId: session.id,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{t("Log out")}
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</Table.Td>
|
||||||
|
)}
|
||||||
|
</Table.Tr>
|
||||||
|
))}
|
||||||
|
</Table.Tbody>
|
||||||
|
</Table>
|
||||||
|
|
||||||
|
{hasMore && (
|
||||||
|
<Button
|
||||||
|
variant="subtle"
|
||||||
|
size="xs"
|
||||||
|
onClick={() => setVisibleCount((c) => c + PAGE_SIZE)}
|
||||||
|
>
|
||||||
|
{t("Load more")}
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{(!sessions || sessions.length === 0) && (
|
||||||
|
<Text size="sm" c="dimmed" ta="center">
|
||||||
|
{t("No active sessions")}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,55 @@
|
|||||||
|
import {
|
||||||
|
useMutation,
|
||||||
|
useQuery,
|
||||||
|
useQueryClient,
|
||||||
|
UseQueryResult,
|
||||||
|
} from "@tanstack/react-query";
|
||||||
|
import {
|
||||||
|
getSessions,
|
||||||
|
revokeSession,
|
||||||
|
revokeAllSessions,
|
||||||
|
} from "@/features/session/services/session-service";
|
||||||
|
import { ISession } from "@/features/session/types/session.types";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
|
export function useGetSessionsQuery(): UseQueryResult<ISession[], Error> {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: ["session-list"],
|
||||||
|
queryFn: () => getSessions(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useRevokeSessionMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
return useMutation<void, Error, { sessionId: string }>({
|
||||||
|
mutationFn: (data) => revokeSession(data),
|
||||||
|
onSuccess: () => {
|
||||||
|
notifications.show({ message: t("Session revoked") });
|
||||||
|
queryClient.invalidateQueries({ queryKey: ["session-list"] });
|
||||||
|
},
|
||||||
|
onError: (error) => {
|
||||||
|
const errorMessage = error["response"]?.data?.message;
|
||||||
|
notifications.show({ message: errorMessage, color: "red" });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useRevokeAllSessionsMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
return useMutation<void, Error, void>({
|
||||||
|
mutationFn: () => revokeAllSessions(),
|
||||||
|
onSuccess: () => {
|
||||||
|
notifications.show({ message: t("All other sessions revoked") });
|
||||||
|
queryClient.invalidateQueries({ queryKey: ["session-list"] });
|
||||||
|
},
|
||||||
|
onError: (error) => {
|
||||||
|
const errorMessage = error["response"]?.data?.message;
|
||||||
|
notifications.show({ message: errorMessage, color: "red" });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
import api from "@/lib/api-client";
|
||||||
|
import { ISession } from "@/features/session/types/session.types";
|
||||||
|
|
||||||
|
export async function getSessions(): Promise<ISession[]> {
|
||||||
|
const req = await api.post<{ sessions: ISession[] }>("/sessions");
|
||||||
|
return req.data.sessions;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function revokeSession(data: {
|
||||||
|
sessionId: string;
|
||||||
|
}): Promise<void> {
|
||||||
|
await api.post("/sessions/revoke", data);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function revokeAllSessions(): Promise<void> {
|
||||||
|
await api.post("/sessions/revoke-all");
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
export type ISession = {
|
||||||
|
id: string;
|
||||||
|
deviceName: string | null;
|
||||||
|
geoLocation: string | null;
|
||||||
|
lastActiveAt: string;
|
||||||
|
createdAt: string;
|
||||||
|
isCurrentDevice?: boolean;
|
||||||
|
};
|
||||||
@@ -3,6 +3,7 @@ import SpaceMembersList from "@/features/space/components/space-members.tsx";
|
|||||||
import AddSpaceMembersModal from "@/features/space/components/add-space-members-modal.tsx";
|
import AddSpaceMembersModal from "@/features/space/components/add-space-members-modal.tsx";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import SpaceDetails from "@/features/space/components/space-details.tsx";
|
import SpaceDetails from "@/features/space/components/space-details.tsx";
|
||||||
|
import SpaceSecuritySettings from "@/features/space/components/space-security-settings.tsx";
|
||||||
import { useSpaceQuery } from "@/features/space/queries/space-query.ts";
|
import { useSpaceQuery } from "@/features/space/queries/space-query.ts";
|
||||||
import { useSpaceAbility } from "@/features/space/permissions/use-space-ability.ts";
|
import { useSpaceAbility } from "@/features/space/permissions/use-space-ability.ts";
|
||||||
import {
|
import {
|
||||||
@@ -59,6 +60,14 @@ export default function SpaceSettingsModal({
|
|||||||
<Tabs.Tab fw={500} value="members">
|
<Tabs.Tab fw={500} value="members">
|
||||||
{t("Members")}
|
{t("Members")}
|
||||||
</Tabs.Tab>
|
</Tabs.Tab>
|
||||||
|
{spaceAbility.can(
|
||||||
|
SpaceCaslAction.Manage,
|
||||||
|
SpaceCaslSubject.Settings,
|
||||||
|
) && (
|
||||||
|
<Tabs.Tab fw={500} value="security">
|
||||||
|
{t("Security")}
|
||||||
|
</Tabs.Tab>
|
||||||
|
)}
|
||||||
</Tabs.List>
|
</Tabs.List>
|
||||||
|
|
||||||
<Tabs.Panel value="general">
|
<Tabs.Panel value="general">
|
||||||
@@ -91,6 +100,20 @@ export default function SpaceSettingsModal({
|
|||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
</Tabs.Panel>
|
</Tabs.Panel>
|
||||||
|
|
||||||
|
<Tabs.Panel value="security">
|
||||||
|
<ScrollArea h={580} scrollbarSize={5} pr={8}>
|
||||||
|
<div style={{ paddingBottom: "100px" }}>
|
||||||
|
<SpaceSecuritySettings
|
||||||
|
space={space}
|
||||||
|
readOnly={spaceAbility.cannot(
|
||||||
|
SpaceCaslAction.Manage,
|
||||||
|
SpaceCaslSubject.Settings,
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</ScrollArea>
|
||||||
|
</Tabs.Panel>
|
||||||
</Tabs>
|
</Tabs>
|
||||||
</div>
|
</div>
|
||||||
</Modal.Body>
|
</Modal.Body>
|
||||||
|
|||||||
@@ -9,6 +9,8 @@ import {
|
|||||||
import {
|
import {
|
||||||
IconArrowDown,
|
IconArrowDown,
|
||||||
IconDots,
|
IconDots,
|
||||||
|
IconEye,
|
||||||
|
IconEyeOff,
|
||||||
IconFileExport,
|
IconFileExport,
|
||||||
IconHome,
|
IconHome,
|
||||||
IconPlus,
|
IconPlus,
|
||||||
@@ -16,6 +18,11 @@ import {
|
|||||||
IconSettings,
|
IconSettings,
|
||||||
IconTrash,
|
IconTrash,
|
||||||
} from "@tabler/icons-react";
|
} from "@tabler/icons-react";
|
||||||
|
import {
|
||||||
|
useSpaceWatchStatusQuery,
|
||||||
|
useWatchSpaceMutation,
|
||||||
|
useUnwatchSpaceMutation,
|
||||||
|
} from "@/features/space/queries/space-watcher-query.ts";
|
||||||
import classes from "./space-sidebar.module.css";
|
import classes from "./space-sidebar.module.css";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
@@ -160,13 +167,20 @@ export function SpaceSidebar() {
|
|||||||
{t("Pages")}
|
{t("Pages")}
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
{spaceAbility.can(
|
<Group gap="xs">
|
||||||
SpaceCaslAction.Manage,
|
<SpaceMenu
|
||||||
SpaceCaslSubject.Page,
|
spaceId={space.id}
|
||||||
) && (
|
canManagePages={spaceAbility.can(
|
||||||
<Group gap="xs">
|
SpaceCaslAction.Manage,
|
||||||
<SpaceMenu spaceId={space.id} onSpaceSettings={openSettings} />
|
SpaceCaslSubject.Page,
|
||||||
|
)}
|
||||||
|
onSpaceSettings={openSettings}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{spaceAbility.can(
|
||||||
|
SpaceCaslAction.Manage,
|
||||||
|
SpaceCaslSubject.Page,
|
||||||
|
) && (
|
||||||
<Tooltip label={t("Create page")} withArrow position="right">
|
<Tooltip label={t("Create page")} withArrow position="right">
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
variant="default"
|
variant="default"
|
||||||
@@ -177,8 +191,8 @@ export function SpaceSidebar() {
|
|||||||
<IconPlus />
|
<IconPlus />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</Group>
|
)}
|
||||||
)}
|
</Group>
|
||||||
</Group>
|
</Group>
|
||||||
|
|
||||||
<div className={classes.pages}>
|
<div className={classes.pages}>
|
||||||
@@ -204,9 +218,14 @@ export function SpaceSidebar() {
|
|||||||
|
|
||||||
interface SpaceMenuProps {
|
interface SpaceMenuProps {
|
||||||
spaceId: string;
|
spaceId: string;
|
||||||
|
canManagePages: boolean;
|
||||||
onSpaceSettings: () => void;
|
onSpaceSettings: () => void;
|
||||||
}
|
}
|
||||||
function SpaceMenu({ spaceId, onSpaceSettings }: SpaceMenuProps) {
|
function SpaceMenu({
|
||||||
|
spaceId,
|
||||||
|
canManagePages,
|
||||||
|
onSpaceSettings,
|
||||||
|
}: SpaceMenuProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { spaceSlug } = useParams();
|
const { spaceSlug } = useParams();
|
||||||
const [importOpened, { open: openImportModal, close: closeImportModal }] =
|
const [importOpened, { open: openImportModal, close: closeImportModal }] =
|
||||||
@@ -214,15 +233,24 @@ function SpaceMenu({ spaceId, onSpaceSettings }: SpaceMenuProps) {
|
|||||||
const [exportOpened, { open: openExportModal, close: closeExportModal }] =
|
const [exportOpened, { open: openExportModal, close: closeExportModal }] =
|
||||||
useDisclosure(false);
|
useDisclosure(false);
|
||||||
|
|
||||||
|
const { data: watchStatus } = useSpaceWatchStatusQuery(spaceId);
|
||||||
|
const watchMutation = useWatchSpaceMutation();
|
||||||
|
const unwatchMutation = useUnwatchSpaceMutation();
|
||||||
|
const isWatching = watchStatus?.watching ?? false;
|
||||||
|
|
||||||
|
const handleToggleWatch = () => {
|
||||||
|
if (isWatching) {
|
||||||
|
unwatchMutation.mutate(spaceId);
|
||||||
|
} else {
|
||||||
|
watchMutation.mutate(spaceId);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Menu width={200} shadow="md" withArrow>
|
<Menu width={200} shadow="md" withArrow>
|
||||||
<Menu.Target>
|
<Menu.Target>
|
||||||
<Tooltip
|
<Tooltip label={t("Space menu")} withArrow position="top">
|
||||||
label={t("Import pages & space settings")}
|
|
||||||
withArrow
|
|
||||||
position="top"
|
|
||||||
>
|
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
variant="default"
|
variant="default"
|
||||||
size={18}
|
size={18}
|
||||||
@@ -235,50 +263,69 @@ function SpaceMenu({ spaceId, onSpaceSettings }: SpaceMenuProps) {
|
|||||||
|
|
||||||
<Menu.Dropdown>
|
<Menu.Dropdown>
|
||||||
<Menu.Item
|
<Menu.Item
|
||||||
onClick={openImportModal}
|
onClick={handleToggleWatch}
|
||||||
leftSection={<IconArrowDown size={16} />}
|
leftSection={
|
||||||
|
isWatching ? <IconEyeOff size={16} /> : <IconEye size={16} />
|
||||||
|
}
|
||||||
>
|
>
|
||||||
{t("Import pages")}
|
{isWatching ? t("Stop watching space") : t("Watch space")}
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
|
|
||||||
<Menu.Item
|
{canManagePages && (
|
||||||
onClick={openExportModal}
|
<>
|
||||||
leftSection={<IconFileExport size={16} />}
|
<Menu.Divider />
|
||||||
>
|
|
||||||
{t("Export space")}
|
|
||||||
</Menu.Item>
|
|
||||||
|
|
||||||
<Menu.Divider />
|
<Menu.Item
|
||||||
|
onClick={openImportModal}
|
||||||
|
leftSection={<IconArrowDown size={16} />}
|
||||||
|
>
|
||||||
|
{t("Import pages")}
|
||||||
|
</Menu.Item>
|
||||||
|
|
||||||
<Menu.Item
|
<Menu.Item
|
||||||
onClick={onSpaceSettings}
|
onClick={openExportModal}
|
||||||
leftSection={<IconSettings size={16} />}
|
leftSection={<IconFileExport size={16} />}
|
||||||
>
|
>
|
||||||
{t("Space settings")}
|
{t("Export space")}
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
|
|
||||||
<Menu.Item
|
<Menu.Divider />
|
||||||
component={Link}
|
|
||||||
to={`/s/${spaceSlug}/trash`}
|
<Menu.Item
|
||||||
leftSection={<IconTrash size={16} />}
|
onClick={onSpaceSettings}
|
||||||
>
|
leftSection={<IconSettings size={16} />}
|
||||||
{t("Trash")}
|
>
|
||||||
</Menu.Item>
|
{t("Space settings")}
|
||||||
|
</Menu.Item>
|
||||||
|
|
||||||
|
<Menu.Item
|
||||||
|
component={Link}
|
||||||
|
to={`/s/${spaceSlug}/trash`}
|
||||||
|
leftSection={<IconTrash size={16} />}
|
||||||
|
>
|
||||||
|
{t("Trash")}
|
||||||
|
</Menu.Item>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
</Menu.Dropdown>
|
</Menu.Dropdown>
|
||||||
</Menu>
|
</Menu>
|
||||||
|
|
||||||
<PageImportModal
|
{canManagePages && (
|
||||||
spaceId={spaceId}
|
<>
|
||||||
open={importOpened}
|
<PageImportModal
|
||||||
onClose={closeImportModal}
|
spaceId={spaceId}
|
||||||
/>
|
open={importOpened}
|
||||||
|
onClose={closeImportModal}
|
||||||
|
/>
|
||||||
|
|
||||||
<ExportModal
|
<ExportModal
|
||||||
type="space"
|
type="space"
|
||||||
id={spaceId}
|
id={spaceId}
|
||||||
open={exportOpened}
|
open={exportOpened}
|
||||||
onClose={closeExportModal}
|
onClose={closeExportModal}
|
||||||
/>
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ import {
|
|||||||
ResponsiveSettingsControl,
|
ResponsiveSettingsControl,
|
||||||
ResponsiveSettingsRow,
|
ResponsiveSettingsRow,
|
||||||
} from "@/components/ui/responsive-settings-row.tsx";
|
} from "@/components/ui/responsive-settings-row.tsx";
|
||||||
import SpacePublicSharingToggle from "@/ee/security/components/space-public-sharing-toggle.tsx";
|
|
||||||
|
|
||||||
interface SpaceDetailsProps {
|
interface SpaceDetailsProps {
|
||||||
spaceId: string;
|
spaceId: string;
|
||||||
@@ -27,7 +27,6 @@ interface SpaceDetailsProps {
|
|||||||
export default function SpaceDetails({ spaceId, readOnly }: SpaceDetailsProps) {
|
export default function SpaceDetails({ spaceId, readOnly }: SpaceDetailsProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { data: space, isLoading, refetch } = useSpaceQuery(spaceId);
|
const { data: space, isLoading, refetch } = useSpaceQuery(spaceId);
|
||||||
const showSharingToggle = !readOnly;
|
|
||||||
const [exportOpened, { open: openExportModal, close: closeExportModal }] =
|
const [exportOpened, { open: openExportModal, close: closeExportModal }] =
|
||||||
useDisclosure(false);
|
useDisclosure(false);
|
||||||
const [isIconUploading, setIsIconUploading] = useState(false);
|
const [isIconUploading, setIsIconUploading] = useState(false);
|
||||||
@@ -89,13 +88,6 @@ export default function SpaceDetails({ spaceId, readOnly }: SpaceDetailsProps) {
|
|||||||
|
|
||||||
<EditSpaceForm space={space} readOnly={readOnly} />
|
<EditSpaceForm space={space} readOnly={readOnly} />
|
||||||
|
|
||||||
{showSharingToggle && (
|
|
||||||
<>
|
|
||||||
<Divider my="lg" />
|
|
||||||
<SpacePublicSharingToggle space={space} />
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{!readOnly && (
|
{!readOnly && (
|
||||||
<>
|
<>
|
||||||
<Divider my="lg" />
|
<Divider my="lg" />
|
||||||
|
|||||||
@@ -0,0 +1,34 @@
|
|||||||
|
import { Text, Divider } from "@mantine/core";
|
||||||
|
import React from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { ISpace } from "@/features/space/types/space.types.ts";
|
||||||
|
import SpacePublicSharingToggle from "@/ee/security/components/space-public-sharing-toggle.tsx";
|
||||||
|
import SpaceViewerCommentsToggle from "@/ee/security/components/space-viewer-comments-toggle.tsx";
|
||||||
|
|
||||||
|
type SpaceSecuritySettingsProps = {
|
||||||
|
space: ISpace;
|
||||||
|
readOnly?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function SpaceSecuritySettings({
|
||||||
|
space,
|
||||||
|
readOnly,
|
||||||
|
}: SpaceSecuritySettingsProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
if (readOnly) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<Text my="md" fw={600}>
|
||||||
|
{t("Security")}
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<SpacePublicSharingToggle space={space} />
|
||||||
|
|
||||||
|
<Divider my="lg" />
|
||||||
|
|
||||||
|
<SpaceViewerCommentsToggle space={space} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query";
|
||||||
|
import {
|
||||||
|
watchSpace,
|
||||||
|
unwatchSpace,
|
||||||
|
getSpaceWatchStatus,
|
||||||
|
} from "@/features/space/services/space-watcher-service";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
|
const SPACE_WATCHER_KEY = "space-watcher";
|
||||||
|
|
||||||
|
export function useSpaceWatchStatusQuery(spaceId: string) {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: [SPACE_WATCHER_KEY, spaceId],
|
||||||
|
queryFn: () => getSpaceWatchStatus(spaceId),
|
||||||
|
enabled: !!spaceId,
|
||||||
|
staleTime: 60_000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useWatchSpaceMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: (spaceId: string) => watchSpace(spaceId),
|
||||||
|
onSuccess: (_data, spaceId) => {
|
||||||
|
queryClient.setQueryData([SPACE_WATCHER_KEY, spaceId], {
|
||||||
|
watching: true,
|
||||||
|
});
|
||||||
|
notifications.show({ message: t("You are now watching this space") });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useUnwatchSpaceMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: (spaceId: string) => unwatchSpace(spaceId),
|
||||||
|
onSuccess: (_data, spaceId) => {
|
||||||
|
queryClient.setQueryData([SPACE_WATCHER_KEY, spaceId], {
|
||||||
|
watching: false,
|
||||||
|
});
|
||||||
|
notifications.show({
|
||||||
|
message: t("You are no longer watching this space"),
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
import api from "@/lib/api-client";
|
||||||
|
|
||||||
|
export async function watchSpace(
|
||||||
|
spaceId: string,
|
||||||
|
): Promise<{ watching: boolean }> {
|
||||||
|
const req = await api.post<{ watching: boolean }>("/spaces/watch", {
|
||||||
|
spaceId,
|
||||||
|
});
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function unwatchSpace(
|
||||||
|
spaceId: string,
|
||||||
|
): Promise<{ watching: boolean }> {
|
||||||
|
const req = await api.post<{ watching: boolean }>("/spaces/unwatch", {
|
||||||
|
spaceId,
|
||||||
|
});
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getSpaceWatchStatus(
|
||||||
|
spaceId: string,
|
||||||
|
): Promise<{ watching: boolean }> {
|
||||||
|
const req = await api.post<{ watching: boolean }>("/spaces/watch-status", {
|
||||||
|
spaceId,
|
||||||
|
});
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
@@ -9,8 +9,13 @@ export interface ISpaceSharingSettings {
|
|||||||
disabled?: boolean;
|
disabled?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface ISpaceCommentsSettings {
|
||||||
|
allowViewerComments?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
export interface ISpaceSettings {
|
export interface ISpaceSettings {
|
||||||
sharing?: ISpaceSharingSettings;
|
sharing?: ISpaceSharingSettings;
|
||||||
|
comments?: ISpaceCommentsSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ISpace {
|
export interface ISpace {
|
||||||
@@ -29,6 +34,7 @@ export interface ISpace {
|
|||||||
settings?: ISpaceSettings;
|
settings?: ISpaceSettings;
|
||||||
// for updates
|
// for updates
|
||||||
disablePublicSharing?: boolean;
|
disablePublicSharing?: boolean;
|
||||||
|
allowViewerComments?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface IMembership {
|
interface IMembership {
|
||||||
|
|||||||
@@ -1,9 +1,8 @@
|
|||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import { focusAtom } from "jotai-optics";
|
|
||||||
import { z } from "zod/v4";
|
import { z } from "zod/v4";
|
||||||
import { useForm } from "@mantine/form";
|
import { useForm } from "@mantine/form";
|
||||||
import { zod4Resolver } from "mantine-form-zod-resolver";
|
import { zod4Resolver } from "mantine-form-zod-resolver";
|
||||||
import { currentUserAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { userAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
import { updateUser } from "@/features/user/services/user-service.ts";
|
import { updateUser } from "@/features/user/services/user-service.ts";
|
||||||
import { IUser } from "@/features/user/types/user.types.ts";
|
import { IUser } from "@/features/user/types/user.types.ts";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
@@ -17,18 +16,15 @@ const formSchema = z.object({
|
|||||||
|
|
||||||
type FormValues = z.infer<typeof formSchema>;
|
type FormValues = z.infer<typeof formSchema>;
|
||||||
|
|
||||||
const userAtom = focusAtom(currentUserAtom, (optic) => optic.prop("user"));
|
|
||||||
|
|
||||||
export default function AccountNameForm() {
|
export default function AccountNameForm() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [currentUser] = useAtom(currentUserAtom);
|
const [user, setUser] = useAtom(userAtom);
|
||||||
const [, setUser] = useAtom(userAtom);
|
|
||||||
|
|
||||||
const form = useForm<FormValues>({
|
const form = useForm<FormValues>({
|
||||||
validate: zod4Resolver(formSchema),
|
validate: zod4Resolver(formSchema),
|
||||||
initialValues: {
|
initialValues: {
|
||||||
name: currentUser?.user.name,
|
name: user?.name,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,117 @@
|
|||||||
|
import { userAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
import { updateUser } from "@/features/user/services/user-service.ts";
|
||||||
|
import { IUser, IUserSettings } from "@/features/user/types/user.types.ts";
|
||||||
|
import { Switch, Text, Title, Stack } from "@mantine/core";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import React, { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import {
|
||||||
|
ResponsiveSettingsRow,
|
||||||
|
ResponsiveSettingsContent,
|
||||||
|
ResponsiveSettingsControl,
|
||||||
|
} from "@/components/ui/responsive-settings-row";
|
||||||
|
|
||||||
|
type NotificationKey = keyof NonNullable<IUserSettings["notifications"]>;
|
||||||
|
|
||||||
|
const notificationItems: {
|
||||||
|
key: NotificationKey;
|
||||||
|
dtoField: keyof IUser;
|
||||||
|
label: string;
|
||||||
|
description: string;
|
||||||
|
}[] = [
|
||||||
|
{
|
||||||
|
key: "page.updated",
|
||||||
|
dtoField: "notificationPageUpdates",
|
||||||
|
label: "Page updates",
|
||||||
|
description: "Get notified when pages you watch are updated.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "page.userMention",
|
||||||
|
dtoField: "notificationPageUserMention",
|
||||||
|
label: "Page mentions",
|
||||||
|
description: "Get notified when someone mentions you on a page.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "comment.userMention",
|
||||||
|
dtoField: "notificationCommentUserMention",
|
||||||
|
label: "Comment mentions",
|
||||||
|
description: "Get notified when someone mentions you in a comment.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "comment.created",
|
||||||
|
dtoField: "notificationCommentCreated",
|
||||||
|
label: "New comments",
|
||||||
|
description:
|
||||||
|
"Get notified about new comments on threads you participate in.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "comment.resolved",
|
||||||
|
dtoField: "notificationCommentResolved",
|
||||||
|
label: "Resolved comments",
|
||||||
|
description: "Get notified when your comment is resolved.",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
function NotificationToggle({
|
||||||
|
settingKey,
|
||||||
|
dtoField,
|
||||||
|
label,
|
||||||
|
description,
|
||||||
|
}: {
|
||||||
|
settingKey: NotificationKey;
|
||||||
|
dtoField: keyof IUser;
|
||||||
|
label: string;
|
||||||
|
description: string;
|
||||||
|
}) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [user, setUser] = useAtom(userAtom);
|
||||||
|
const [checked, setChecked] = useState(
|
||||||
|
user.settings?.notifications?.[settingKey] !== false,
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = event.currentTarget.checked;
|
||||||
|
setChecked(value);
|
||||||
|
try {
|
||||||
|
const updatedUser = await updateUser({ [dtoField]: value } as any);
|
||||||
|
setUser(updatedUser);
|
||||||
|
} catch {
|
||||||
|
setChecked(!value);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ResponsiveSettingsRow>
|
||||||
|
<ResponsiveSettingsContent>
|
||||||
|
<Text size="md">{t(label)}</Text>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t(description)}
|
||||||
|
</Text>
|
||||||
|
</ResponsiveSettingsContent>
|
||||||
|
|
||||||
|
<ResponsiveSettingsControl>
|
||||||
|
<Switch checked={checked} onChange={handleChange} />
|
||||||
|
</ResponsiveSettingsControl>
|
||||||
|
</ResponsiveSettingsRow>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function NotificationPref() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack gap="xs">
|
||||||
|
<Title order={5}>{t("Email notifications")}</Title>
|
||||||
|
|
||||||
|
{notificationItems.map((item) => (
|
||||||
|
<NotificationToggle
|
||||||
|
key={item.key}
|
||||||
|
settingKey={item.key}
|
||||||
|
dtoField={item.dtoField}
|
||||||
|
label={item.label}
|
||||||
|
description={item.description}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -20,6 +20,11 @@ export interface IUser {
|
|||||||
deletedAt: Date;
|
deletedAt: Date;
|
||||||
fullPageWidth: boolean; // used for update
|
fullPageWidth: boolean; // used for update
|
||||||
pageEditMode: string; // used for update
|
pageEditMode: string; // used for update
|
||||||
|
notificationPageUpdates: boolean; // used for update
|
||||||
|
notificationPageUserMention: boolean; // used for update
|
||||||
|
notificationCommentUserMention: boolean; // used for update
|
||||||
|
notificationCommentCreated: boolean; // used for update
|
||||||
|
notificationCommentResolved: boolean; // used for update
|
||||||
hasGeneratedPassword?: boolean;
|
hasGeneratedPassword?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -33,6 +38,13 @@ export interface IUserSettings {
|
|||||||
fullPageWidth: boolean;
|
fullPageWidth: boolean;
|
||||||
pageEditMode: string;
|
pageEditMode: string;
|
||||||
};
|
};
|
||||||
|
notifications?: {
|
||||||
|
"page.updated"?: boolean;
|
||||||
|
"page.userMention"?: boolean;
|
||||||
|
"comment.userMention"?: boolean;
|
||||||
|
"comment.created"?: boolean;
|
||||||
|
"comment.resolved"?: boolean;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum PageEditMode {
|
export enum PageEditMode {
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ i18n
|
|||||||
.init({
|
.init({
|
||||||
fallbackLng: "en-US",
|
fallbackLng: "en-US",
|
||||||
debug: false,
|
debug: false,
|
||||||
|
showSupportNotice: false,
|
||||||
load: 'currentOnly',
|
load: 'currentOnly',
|
||||||
|
|
||||||
interpolation: {
|
interpolation: {
|
||||||
|
|||||||
@@ -74,8 +74,12 @@ function redirectToLogin() {
|
|||||||
];
|
];
|
||||||
if (!exemptPaths.some((path) => window.location.pathname.startsWith(path))) {
|
if (!exemptPaths.some((path) => window.location.pathname.startsWith(path))) {
|
||||||
const redirectTo = window.location.pathname;
|
const redirectTo = window.location.pathname;
|
||||||
const params = new URLSearchParams({ redirect: redirectTo });
|
if (redirectTo === APP_ROUTE.HOME) {
|
||||||
window.location.href = `${APP_ROUTE.AUTH.LOGIN}?${params.toString()}`;
|
window.location.href = APP_ROUTE.AUTH.LOGIN;
|
||||||
|
} else {
|
||||||
|
const params = new URLSearchParams({ redirect: redirectTo });
|
||||||
|
window.location.href = `${APP_ROUTE.AUTH.LOGIN}?${params.toString()}`;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import bytes from "bytes";
|
import bytes from "bytes";
|
||||||
import { castToBoolean } from "@/lib/utils.tsx";
|
import { castToBoolean } from "@/lib/utils.tsx";
|
||||||
import { AvatarIconType } from "@/features/attachments/types/attachment.types.ts";
|
import { AvatarIconType } from "@/features/attachments/types/attachment.types.ts";
|
||||||
|
import { sanitizeUrl } from "@docmost/editor-ext";
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface Window {
|
interface Window {
|
||||||
@@ -66,7 +67,7 @@ export function getFileUrl(src: string) {
|
|||||||
if (src.startsWith("/files/")) {
|
if (src.startsWith("/files/")) {
|
||||||
return getBackendUrl() + src;
|
return getBackendUrl() + src;
|
||||||
}
|
}
|
||||||
return src;
|
return sanitizeUrl(src);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getFileUploadSizeLimit() {
|
export function getFileUploadSizeLimit() {
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user