mirror of
https://github.com/docmost/docmost.git
synced 2026-05-07 06:23:06 +08:00
Compare commits
139 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| a86890d856 | |||
| bef23b6738 | |||
| 5c3942c159 | |||
| e0809e7104 | |||
| da6793ac87 | |||
| 08e94eb3c1 | |||
| 5a14186f1c | |||
| 6a0bb8d4cb | |||
| fba9f4cb2b | |||
| d8f7c4a822 | |||
| 202685b39f | |||
| fc4a428208 | |||
| 5506eb194b | |||
| f32bb298e0 | |||
| 3178cad796 | |||
| 9d7f8c62c5 | |||
| 78b1c1a453 | |||
| 96ed98619f | |||
| 60501de992 | |||
| 74e915546b | |||
| 3523600f40 | |||
| 6ccb2bb872 | |||
| 0245a183e1 | |||
| de5f71894a | |||
| 351b075ebb | |||
| 1ca7d42203 | |||
| 1e441560f6 | |||
| 54775f537d | |||
| 5dbf0027bd | |||
| 5588ec34fb | |||
| 55b8128829 | |||
| aa6a046aa6 | |||
| 657fdf8cb7 | |||
| 98f71c95fe | |||
| efb0a9317b | |||
| 063ea99b66 | |||
| aa143ad79c | |||
| 918f4508d2 | |||
| 5cd0ba6902 | |||
| a1260188ae | |||
| bdf02f593d | |||
| e24bf5ed57 | |||
| f3f74c591f | |||
| 5f966a2d89 | |||
| bcb004af21 | |||
| ac675e7d74 | |||
| bf89eff5e7 | |||
| 183787fa0c | |||
| 15aa04a5f7 | |||
| 79343a5d52 | |||
| 61e252918e | |||
| e98fa7f69a | |||
| 6d148a35eb | |||
| 0bbc1c35de | |||
| 47097969a0 | |||
| 13f529e064 | |||
| 8fc8422fbc | |||
| 732951a322 | |||
| 2544775266 | |||
| d59539f197 | |||
| b061df7f7d | |||
| 0fe1459864 | |||
| 6af7956889 | |||
| 3dbb957bd7 | |||
| f39a4cf2d5 | |||
| 724e01bd55 | |||
| 6e350f6746 | |||
| cb9f27da9a | |||
| d2629afff2 | |||
| 9139d393ef | |||
| ab96672ecd | |||
| 2ea3c2da58 | |||
| 9fb16bc842 | |||
| c3b350d943 | |||
| 8014ba3ab7 | |||
| ec3a04f7c7 | |||
| 04a17c9b92 | |||
| 520c07a0bc | |||
| 60a8ed6826 | |||
| f5684b792e | |||
| 042836cb6d | |||
| 4f1f0ba513 | |||
| 3164b6981c | |||
| 16c1e864af | |||
| c9b1cad982 | |||
| bf8cf6254f | |||
| 3135030376 | |||
| 3fae41a5ca | |||
| b50e25600a | |||
| 1f3b0c7276 | |||
| 3c4cab0d2a | |||
| 4de25a8b94 | |||
| cf5bbb10df | |||
| ac17521717 | |||
| 9ac180f719 | |||
| 46669fea56 | |||
| fe6ecdf1f1 | |||
| 04ae1d7270 | |||
| 1280f96f37 | |||
| 61d1cf88a7 | |||
| f413720e15 | |||
| 8e16ad952a | |||
| 7ada3cb1f9 | |||
| 47c54174b3 | |||
| dc0650289d | |||
| 091e790b83 | |||
| ae24ea29ba | |||
| 9df6061e1a | |||
| 31053e2b20 | |||
| eb8e8507ea | |||
| c99bfb8ef1 | |||
| 26ea04e2a3 | |||
| 6cc58c57f5 | |||
| 7d2ff346fa | |||
| b08d37fbf0 | |||
| d43ee77617 | |||
| 5d91eb4f5f | |||
| 3e9f6b11cc | |||
| db55de9406 | |||
| 1919eba340 | |||
| 7951b2e0c6 | |||
| 73b78f625d | |||
| cf7534de3d | |||
| adec36d544 | |||
| f9e10805f0 | |||
| 00e499b3e5 | |||
| 5ee6e46535 | |||
| 1f797c3d27 | |||
| f12866cf42 | |||
| dcbb65d799 | |||
| 5968764508 | |||
| 242fb6bb57 | |||
| 74cd890bdd | |||
| 509622af54 | |||
| 937386e42b | |||
| 60a373f488 | |||
| 73ee6ee8c3 | |||
| 7d1e5bce0d | |||
| aa58e272d6 |
+3
-2
@@ -1,5 +1,6 @@
|
|||||||
node_modules
|
node_modules
|
||||||
.git
|
.git
|
||||||
.gitignore
|
|
||||||
dist
|
dist
|
||||||
data
|
/data
|
||||||
|
.env*
|
||||||
|
.nx
|
||||||
|
|||||||
+7
-1
@@ -46,4 +46,10 @@ DRAWIO_URL=
|
|||||||
DISABLE_TELEMETRY=false
|
DISABLE_TELEMETRY=false
|
||||||
|
|
||||||
# Enable debug logging in production (default: false)
|
# Enable debug logging in production (default: false)
|
||||||
DEBUG_MODE=false
|
DEBUG_MODE=false
|
||||||
|
|
||||||
|
# Log database queries
|
||||||
|
DEBUG_DB=false
|
||||||
|
|
||||||
|
# Log http requests
|
||||||
|
LOG_HTTP=false
|
||||||
|
|||||||
+7
-5
@@ -1,19 +1,22 @@
|
|||||||
FROM node:22-alpine AS base
|
FROM node:22-slim AS base
|
||||||
LABEL org.opencontainers.image.source="https://github.com/docmost/docmost"
|
LABEL org.opencontainers.image.source="https://github.com/docmost/docmost"
|
||||||
|
|
||||||
|
RUN npm install -g pnpm@10.4.0
|
||||||
|
|
||||||
FROM base AS builder
|
FROM base AS builder
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
COPY . .
|
COPY . .
|
||||||
|
|
||||||
RUN npm install -g pnpm@10.4.0
|
|
||||||
RUN pnpm install --frozen-lockfile
|
RUN pnpm install --frozen-lockfile
|
||||||
RUN pnpm build
|
RUN pnpm build
|
||||||
|
|
||||||
FROM base AS installer
|
FROM base AS installer
|
||||||
|
|
||||||
RUN apk add --no-cache curl bash
|
RUN apt-get update \
|
||||||
|
&& apt-get install -y --no-install-recommends curl bash \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
@@ -29,12 +32,11 @@ COPY --from=builder /app/packages/editor-ext/package.json /app/packages/editor-e
|
|||||||
# Copy root package files
|
# Copy root package files
|
||||||
COPY --from=builder /app/package.json /app/package.json
|
COPY --from=builder /app/package.json /app/package.json
|
||||||
COPY --from=builder /app/pnpm*.yaml /app/
|
COPY --from=builder /app/pnpm*.yaml /app/
|
||||||
|
COPY --from=builder /app/.npmrc /app/.npmrc
|
||||||
|
|
||||||
# Copy patches
|
# Copy patches
|
||||||
COPY --from=builder /app/patches /app/patches
|
COPY --from=builder /app/patches /app/patches
|
||||||
|
|
||||||
RUN npm install -g pnpm@10.4.0
|
|
||||||
|
|
||||||
RUN chown -R node:node /app
|
RUN chown -R node:node /app
|
||||||
|
|
||||||
USER node
|
USER node
|
||||||
|
|||||||
+11
-3
@@ -2,10 +2,18 @@
|
|||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8" />
|
<meta charset="UTF-8" />
|
||||||
<link rel="icon" type="image/png" sizes="32x32" href="/favicon-32x32.png">
|
<link rel="icon" type="image/png" sizes="32x32" href="/icons/favicon-32x32.png" />
|
||||||
<link rel="icon" type="image/png" sizes="16x16" href="/favicon-16x16.png">
|
<link rel="icon" type="image/png" sizes="16x16" href="/icons/favicon-16x16.png" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
<meta name="viewport" content="width=device-width, initial-scale=1.0 user-scalable=no" />
|
||||||
<title>Docmost</title>
|
<title>Docmost</title>
|
||||||
|
<meta name="theme-color" content="#1f1f1f" media="(prefers-color-scheme: dark)" />
|
||||||
|
<meta name="theme-color" content="#f6f7f9" media="(prefers-color-scheme: light)" />
|
||||||
|
<link rel="manifest" href="/manifest.json" />
|
||||||
|
<meta name="mobile-web-app-capable" content="yes" />
|
||||||
|
<meta name="apple-touch-fullscreen" content="yes" />
|
||||||
|
<meta name="apple-mobile-web-app-title" content="Docmost" />
|
||||||
|
<meta name="apple-mobile-web-app-capable" content="yes" />
|
||||||
|
<meta name="apple-mobile-web-app-status-bar-style" content="default" />
|
||||||
<!--meta-tags-->
|
<!--meta-tags-->
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
|
|||||||
+26
-28
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "client",
|
"name": "client",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "0.22.2",
|
"version": "0.25.1",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
"build": "tsc && vite build",
|
"build": "tsc && vite build",
|
||||||
@@ -10,53 +10,51 @@
|
|||||||
"format": "prettier --write \"src/**/*.tsx\" \"src/**/*.ts\""
|
"format": "prettier --write \"src/**/*.tsx\" \"src/**/*.ts\""
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@casl/ability": "^6.7.2",
|
|
||||||
"@casl/react": "^4.0.0",
|
"@casl/react": "^4.0.0",
|
||||||
"@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-864353b",
|
"@excalidraw/excalidraw": "0.18.0-c158187",
|
||||||
"@mantine/core": "^8.1.3",
|
"@mantine/core": "^8.3.12",
|
||||||
"@mantine/form": "^8.1.3",
|
"@mantine/dates": "^8.3.12",
|
||||||
"@mantine/hooks": "^8.1.3",
|
"@mantine/form": "^8.3.12",
|
||||||
"@mantine/modals": "^8.1.3",
|
"@mantine/hooks": "^8.3.12",
|
||||||
"@mantine/notifications": "^8.1.3",
|
"@mantine/modals": "^8.3.12",
|
||||||
"@mantine/spotlight": "^8.1.3",
|
"@mantine/notifications": "^8.3.12",
|
||||||
"@tabler/icons-react": "^3.34.0",
|
"@mantine/spotlight": "^8.3.12",
|
||||||
"@tanstack/react-query": "^5.80.6",
|
"@tabler/icons-react": "^3.36.1",
|
||||||
"@tiptap/extension-character-count": "^2.10.3",
|
"@tanstack/react-query": "^5.90.17",
|
||||||
"alfaaz": "^1.1.0",
|
"alfaaz": "^1.1.0",
|
||||||
"axios": "^1.9.0",
|
"axios": "^1.13.2",
|
||||||
"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.14.0",
|
"i18next": "^23.16.8",
|
||||||
"i18next-http-backend": "^2.6.1",
|
"i18next-http-backend": "^2.7.3",
|
||||||
"jotai": "^2.12.5",
|
"jotai": "^2.16.2",
|
||||||
"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.22",
|
"katex": "0.16.27",
|
||||||
"lowlight": "^3.3.0",
|
"lowlight": "^3.3.0",
|
||||||
"mantine-form-zod-resolver": "^1.3.0",
|
"mantine-form-zod-resolver": "^1.3.0",
|
||||||
"mermaid": "^11.6.0",
|
"mermaid": "^11.12.2",
|
||||||
"mitt": "^3.0.1",
|
"mitt": "^3.0.1",
|
||||||
"posthog-js": "^1.255.1",
|
"posthog-js": "^1.255.1",
|
||||||
"react": "^18.3.1",
|
"react": "^18.3.1",
|
||||||
"react-arborist": "3.4.0",
|
"react-arborist": "3.4.0",
|
||||||
"react-clear-modal": "^2.0.15",
|
"react-clear-modal": "^2.0.17",
|
||||||
"react-dom": "^18.3.1",
|
"react-dom": "^18.3.1",
|
||||||
"react-drawio": "^1.0.1",
|
"react-drawio": "^1.0.7",
|
||||||
"react-error-boundary": "^4.1.2",
|
"react-error-boundary": "^4.1.2",
|
||||||
"react-helmet-async": "^2.0.5",
|
"react-helmet-async": "^2.0.5",
|
||||||
"react-i18next": "^15.0.1",
|
"react-i18next": "^15.0.1",
|
||||||
"react-router-dom": "^7.0.1",
|
"react-router-dom": "^7.12.0",
|
||||||
"semver": "^7.7.2",
|
"semver": "^7.7.3",
|
||||||
"socket.io-client": "^4.8.1",
|
"socket.io-client": "^4.8.3",
|
||||||
"tippy.js": "^6.3.7",
|
|
||||||
"tiptap-extension-global-drag-handle": "^0.1.18",
|
"tiptap-extension-global-drag-handle": "^0.1.18",
|
||||||
"zod": "^3.25.56"
|
"zod": "^3.25.76"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@eslint/js": "^9.16.0",
|
"@eslint/js": "^9.16.0",
|
||||||
@@ -64,10 +62,10 @@
|
|||||||
"@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.7",
|
||||||
"@types/node": "22.10.0",
|
"@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": "^4.4.1",
|
"@vitejs/plugin-react": "^5.1.1",
|
||||||
"eslint": "^9.15.0",
|
"eslint": "^9.15.0",
|
||||||
"eslint-plugin-react": "^7.37.2",
|
"eslint-plugin-react": "^7.37.2",
|
||||||
"eslint-plugin-react-hooks": "^5.1.0",
|
"eslint-plugin-react-hooks": "^5.1.0",
|
||||||
@@ -80,6 +78,6 @@
|
|||||||
"prettier": "^3.4.1",
|
"prettier": "^3.4.1",
|
||||||
"typescript": "^5.7.2",
|
"typescript": "^5.7.2",
|
||||||
"typescript-eslint": "^8.17.0",
|
"typescript-eslint": "^8.17.0",
|
||||||
"vite": "^6.3.5"
|
"vite": "^7.2.4"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Binary file not shown.
|
Before Width: | Height: | Size: 562 B |
Binary file not shown.
|
Before Width: | Height: | Size: 1.0 KiB |
Binary file not shown.
|
After Width: | Height: | Size: 4.6 KiB |
Binary file not shown.
|
After Width: | Height: | Size: 13 KiB |
Binary file not shown.
|
After Width: | Height: | Size: 509 B |
Binary file not shown.
|
After Width: | Height: | Size: 881 B |
@@ -29,6 +29,7 @@
|
|||||||
"Choose your preferred interface language.": "Wählen Sie Ihre bevorzugte Benutzersprache.",
|
"Choose your preferred interface language.": "Wählen Sie Ihre bevorzugte Benutzersprache.",
|
||||||
"Choose your preferred page width.": "Wählen Sie Ihre bevorzugte Seitenbreite.",
|
"Choose your preferred page width.": "Wählen Sie Ihre bevorzugte Seitenbreite.",
|
||||||
"Confirm": "Bestätigen",
|
"Confirm": "Bestätigen",
|
||||||
|
"Copy as Markdown": "Als Markdown kopieren",
|
||||||
"Copy link": "Link kopieren",
|
"Copy link": "Link kopieren",
|
||||||
"Create": "Erstellen",
|
"Create": "Erstellen",
|
||||||
"Create group": "Gruppe erstellen",
|
"Create group": "Gruppe erstellen",
|
||||||
@@ -42,7 +43,7 @@
|
|||||||
"Delete group": "Gruppe löschen",
|
"Delete group": "Gruppe löschen",
|
||||||
"Are you sure you want to delete this page? This will delete its children and page history. This action is irreversible.": "Sind Sie sicher, dass Sie diese Seite löschen möchten? Dadurch werden ihre Unterseiten und die Seitengeschichte gelöscht. Diese Aktion ist unwiderruflich.",
|
"Are you sure you want to delete this page? This will delete its children and page history. This action is irreversible.": "Sind Sie sicher, dass Sie diese Seite löschen möchten? Dadurch werden ihre Unterseiten und die Seitengeschichte gelöscht. Diese Aktion ist unwiderruflich.",
|
||||||
"Description": "Beschreibung",
|
"Description": "Beschreibung",
|
||||||
"Details": "Einzelheiten",
|
"Details": "Details",
|
||||||
"e.g ACME": "z.B. ACME",
|
"e.g ACME": "z.B. ACME",
|
||||||
"e.g ACME Inc": "z.B. ACME Inc.",
|
"e.g ACME Inc": "z.B. ACME Inc.",
|
||||||
"e.g Developers": "z.B. Entwickler",
|
"e.g Developers": "z.B. Entwickler",
|
||||||
@@ -53,6 +54,7 @@
|
|||||||
"e.g Space for product team": "z.B. Bereich für das Produktteam",
|
"e.g Space for product team": "z.B. Bereich für das Produktteam",
|
||||||
"e.g Space for sales team to collaborate": "z.B. Bereich für das Vertriebsteam zur Zusammenarbeit",
|
"e.g Space for sales team to collaborate": "z.B. Bereich für das Vertriebsteam zur Zusammenarbeit",
|
||||||
"Edit": "Bearbeiten",
|
"Edit": "Bearbeiten",
|
||||||
|
"Read": "Lesen",
|
||||||
"Edit group": "Gruppe bearbeiten",
|
"Edit group": "Gruppe bearbeiten",
|
||||||
"Email": "E-Mail",
|
"Email": "E-Mail",
|
||||||
"Enter a strong password": "Geben Sie ein starkes Passwort ein",
|
"Enter a strong password": "Geben Sie ein starkes Passwort ein",
|
||||||
@@ -121,6 +123,8 @@
|
|||||||
"page": "Seite",
|
"page": "Seite",
|
||||||
"Page deleted successfully": "Seite erfolgreich gelöscht",
|
"Page deleted successfully": "Seite erfolgreich gelöscht",
|
||||||
"Page history": "Seitengeschichte",
|
"Page history": "Seitengeschichte",
|
||||||
|
"Select version": "Version auswählen",
|
||||||
|
"Highlight changes": "Änderungen hervorheben",
|
||||||
"Page import is in progress. Please do not close this tab.": "Der Seitenimport läuft. Bitte schließen Sie diesen Tab nicht.",
|
"Page import is in progress. Please do not close this tab.": "Der Seitenimport läuft. Bitte schließen Sie diesen Tab nicht.",
|
||||||
"Pages": "Seiten",
|
"Pages": "Seiten",
|
||||||
"pages": "Seiten",
|
"pages": "Seiten",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "Export fehlgeschlagen:",
|
"Export failed:": "Export fehlgeschlagen:",
|
||||||
"export error": "Exportfehler",
|
"export error": "Exportfehler",
|
||||||
"Export page": "Seite exportieren",
|
"Export page": "Seite exportieren",
|
||||||
|
"Export successful": "Export erfolgreich",
|
||||||
"Export space": "Bereich exportieren",
|
"Export space": "Bereich exportieren",
|
||||||
"Export {{type}}": "Exportiere {{type}}",
|
"Export {{type}}": "Exportiere {{type}}",
|
||||||
"File exceeds the {{limit}} attachment limit": "Datei überschreitet das Anhängelimit von {{limit}}",
|
"File exceeds the {{limit}} attachment limit": "Datei überschreitet das Anhängelimit von {{limit}}",
|
||||||
@@ -327,6 +332,8 @@
|
|||||||
"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 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 file": "Datei wird hochgeladen",
|
||||||
"Table": "Tabelle",
|
"Table": "Tabelle",
|
||||||
"Insert a table.": "Tabelle einfügen.",
|
"Insert a table.": "Tabelle einfügen.",
|
||||||
"Insert collapsible block.": "Einklappbaren Block einfügen.",
|
"Insert collapsible block.": "Einklappbaren Block einfügen.",
|
||||||
@@ -495,5 +502,78 @@
|
|||||||
"Page restored successfully": "Seite erfolgreich wiederhergestellt",
|
"Page restored successfully": "Seite erfolgreich wiederhergestellt",
|
||||||
"Deleted by": "Gelöscht von",
|
"Deleted by": "Gelöscht von",
|
||||||
"Deleted at": "Gelöscht am",
|
"Deleted at": "Gelöscht am",
|
||||||
"Preview": "Vorschau"
|
"Preview": "Vorschau",
|
||||||
|
"Subpages": "Unterseiten",
|
||||||
|
"Failed to load subpages": "Fehler beim Laden von Unterseiten",
|
||||||
|
"No subpages": "Keine Unterseiten",
|
||||||
|
"Subpages (Child pages)": "Unterseiten (Untergeordnete Seiten)",
|
||||||
|
"List all subpages of the current page": "Alle Unterseiten der aktuellen Seite auflisten",
|
||||||
|
"Attachments": "Anhänge",
|
||||||
|
"All spaces": "Alle Bereiche",
|
||||||
|
"Unknown": "Unbekannt",
|
||||||
|
"Find a space": "Einen Bereich finden",
|
||||||
|
"Search in all your spaces": "In all deinen Bereichen suchen",
|
||||||
|
"Type": "Art",
|
||||||
|
"Enterprise": "Unternehmen",
|
||||||
|
"Download attachment": "Anhang herunterladen",
|
||||||
|
"Allowed email domains": "Erlaubte E-Mail-Domains",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "Nur Benutzer mit E-Mail-Adressen aus diesen Domains können sich über SSO registrieren.",
|
||||||
|
"Enter valid domain names separated by comma or space": "Geben Sie gültige Domainnamen ein, durch Kommas oder Leerzeichen getrennt",
|
||||||
|
"Enforce two-factor authentication": "Erzwingen der Zwei-Faktor-Authentifizierung",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "Sobald es erzwungen wird, müssen alle Mitglieder die Zwei-Faktor-Authentifizierung aktivieren, um auf den Arbeitsbereich zugreifen zu können.",
|
||||||
|
"Toggle MFA enforcement": "Umschalten der MFA-Erzwingung",
|
||||||
|
"Display name": "Anzeigename",
|
||||||
|
"Allow signup": "Registrierung erlauben",
|
||||||
|
"Enabled": "Aktiviert",
|
||||||
|
"Advanced Settings": "Erweiterte Einstellungen",
|
||||||
|
"Enable TLS/SSL": "TLS/SSL aktivieren",
|
||||||
|
"Use secure connection to LDAP server": "Sichere Verbindung zum LDAP-Server verwenden",
|
||||||
|
"Group sync": "Gruppensynchronisation",
|
||||||
|
"No SSO providers found.": "Keine SSO-Anbieter gefunden.",
|
||||||
|
"Delete SSO provider": "SSO-Anbieter löschen",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "Sind Sie sicher, dass Sie diesen SSO-Anbieter löschen möchten?",
|
||||||
|
"Action": "Aktion",
|
||||||
|
"{{ssoProviderType}} configuration": "{{ssoProviderType}}-Konfiguration",
|
||||||
|
"Icon": "Icon",
|
||||||
|
"Upload image": "Bild hochladen",
|
||||||
|
"Remove image": "Bild entfernen",
|
||||||
|
"Failed to remove image": "Fehler beim Entfernen des Bildes",
|
||||||
|
"Image exceeds 10MB limit.": "Bild überschreitet das Limit von 10 MB.",
|
||||||
|
"Image removed successfully": "Bild erfolgreich entfernt",
|
||||||
|
"API key": "API-Schlüssel",
|
||||||
|
"API key created successfully": "API-Schlüssel erfolgreich erstellt",
|
||||||
|
"API keys": "API-Schlüssel",
|
||||||
|
"API management": "API-Verwaltung",
|
||||||
|
"Are you sure you want to revoke this API key": "Sind Sie sicher, dass Sie diesen API-Schlüssel widerrufen möchten?",
|
||||||
|
"Create API Key": "API-Schlüssel erstellen",
|
||||||
|
"Custom expiration date": "Benutzerdefiniertes Ablaufdatum",
|
||||||
|
"Enter a descriptive token name": "Geben Sie einen beschreibenden Token-Namen ein",
|
||||||
|
"Expiration": "Ablauf",
|
||||||
|
"Expired": "Abgelaufen",
|
||||||
|
"Expires": "Läuft ab",
|
||||||
|
"I've saved my API key": "Ich habe meinen API-Schlüssel gespeichert",
|
||||||
|
"Last use": "Zuletzt verwendet",
|
||||||
|
"No API keys found": "Keine API-Schlüssel gefunden",
|
||||||
|
"No expiration": "Kein Ablauf",
|
||||||
|
"Revoke API key": "API-Schlüssel widerrufen",
|
||||||
|
"Revoked successfully": "Erfolgreich widerrufen",
|
||||||
|
"Select expiration date": "Ablaufdatum wählen",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "Diese Aktion kann nicht rückgängig gemacht werden. Alle Anwendungen, die diesen API-Schlüssel verwenden, werden nicht mehr funktionieren.",
|
||||||
|
"Update API key": "API-Schlüssel aktualisieren",
|
||||||
|
"Manage API keys for all users in the workspace": "Verwalten Sie API-Schlüssel für alle Benutzer im Arbeitsbereich",
|
||||||
|
"AI settings": "KI-Einstellungen",
|
||||||
|
"AI search": "KI-Suche",
|
||||||
|
"AI Answer": "KI-Antwort",
|
||||||
|
"Ask AI": "KI fragen",
|
||||||
|
"AI is thinking...": "Die KI überlegt...",
|
||||||
|
"Ask a question...": "Fragen stellen...",
|
||||||
|
"AI-powered search (Ask AI)": "KI-gestützte Suche (KI fragen)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "Die KI-Suche verwendet Vektor-Einbettungen, um semantische Suchfunktionen in Ihrem Arbeitsbereich bereitzustellen.",
|
||||||
|
"Toggle AI search": "KI-Suche umschalten",
|
||||||
|
"Sources": "Quellen",
|
||||||
|
"Ask AI not available for attachments": "KI fragen nicht für Anhänge verfügbar",
|
||||||
|
"No answer available": "Keine Antwort verfügbar",
|
||||||
|
"Background color": "Hintergrundfarbe",
|
||||||
|
"Highlight color": "Hervorhebungsfarbe",
|
||||||
|
"Remove color": "Farbe entfernen"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,6 +29,7 @@
|
|||||||
"Choose your preferred interface language.": "Choose your preferred interface language.",
|
"Choose your preferred interface language.": "Choose your preferred interface language.",
|
||||||
"Choose your preferred page width.": "Choose your preferred page width.",
|
"Choose your preferred page width.": "Choose your preferred page width.",
|
||||||
"Confirm": "Confirm",
|
"Confirm": "Confirm",
|
||||||
|
"Copy as Markdown": "Copy as Markdown",
|
||||||
"Copy link": "Copy link",
|
"Copy link": "Copy link",
|
||||||
"Create": "Create",
|
"Create": "Create",
|
||||||
"Create group": "Create group",
|
"Create group": "Create group",
|
||||||
@@ -53,6 +54,7 @@
|
|||||||
"e.g Space for product team": "e.g Space for product team",
|
"e.g Space for product team": "e.g Space for product team",
|
||||||
"e.g Space for sales team to collaborate": "e.g Space for sales team to collaborate",
|
"e.g Space for sales team to collaborate": "e.g Space for sales team to collaborate",
|
||||||
"Edit": "Edit",
|
"Edit": "Edit",
|
||||||
|
"Read": "Read",
|
||||||
"Edit group": "Edit group",
|
"Edit group": "Edit group",
|
||||||
"Email": "Email",
|
"Email": "Email",
|
||||||
"Enter a strong password": "Enter a strong password",
|
"Enter a strong password": "Enter a strong password",
|
||||||
@@ -121,6 +123,8 @@
|
|||||||
"page": "page",
|
"page": "page",
|
||||||
"Page deleted successfully": "Page deleted successfully",
|
"Page deleted successfully": "Page deleted successfully",
|
||||||
"Page history": "Page history",
|
"Page history": "Page history",
|
||||||
|
"Select version": "Select version",
|
||||||
|
"Highlight changes": "Highlight changes",
|
||||||
"Page import is in progress. Please do not close this tab.": "Page import is in progress. Please do not close this tab.",
|
"Page import is in progress. Please do not close this tab.": "Page import is in progress. Please do not close this tab.",
|
||||||
"Pages": "Pages",
|
"Pages": "Pages",
|
||||||
"pages": "pages",
|
"pages": "pages",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "Export failed:",
|
"Export failed:": "Export failed:",
|
||||||
"export error": "export error",
|
"export error": "export error",
|
||||||
"Export page": "Export page",
|
"Export page": "Export page",
|
||||||
|
"Export successful": "Export successful",
|
||||||
"Export space": "Export space",
|
"Export space": "Export space",
|
||||||
"Export {{type}}": "Export {{type}}",
|
"Export {{type}}": "Export {{type}}",
|
||||||
"File exceeds the {{limit}} attachment limit": "File exceeds the {{limit}} attachment limit",
|
"File exceeds the {{limit}} attachment limit": "File exceeds the {{limit}} attachment limit",
|
||||||
@@ -327,6 +332,8 @@
|
|||||||
"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 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 file": "Uploading file",
|
||||||
"Table": "Table",
|
"Table": "Table",
|
||||||
"Insert a table.": "Insert a table.",
|
"Insert a table.": "Insert a table.",
|
||||||
"Insert collapsible block.": "Insert collapsible block.",
|
"Insert collapsible block.": "Insert collapsible block.",
|
||||||
@@ -400,6 +407,21 @@
|
|||||||
"Share deleted successfully": "Share deleted successfully",
|
"Share deleted successfully": "Share deleted successfully",
|
||||||
"Share not found": "Share not found",
|
"Share not found": "Share not found",
|
||||||
"Failed to share page": "Failed to share page",
|
"Failed to share page": "Failed to share page",
|
||||||
|
"Disable public sharing": "Disable public sharing",
|
||||||
|
"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",
|
||||||
|
"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": "Requires an enterprise license",
|
||||||
|
"Enable public sharing": "Enable public sharing",
|
||||||
|
"Are you sure you want to enable public sharing? Members will be able to share pages publicly.": "Are you sure you want to enable public sharing? Members will be able to share pages publicly.",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this workspace will be deleted.": "Are you sure you want to disable public sharing? All existing shared links in this workspace will be deleted.",
|
||||||
|
"Are you sure you want to enable public sharing for this space?": "Are you sure you want to enable public sharing for this space?",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this space will be deleted.": "Are you sure you want to disable public sharing? All existing shared links in this space will be deleted.",
|
||||||
|
"Public sharing is disabled": "Public sharing is disabled",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "Public sharing has been disabled at the workspace level.",
|
||||||
|
"Public sharing has been disabled for this space.": "Public sharing has been disabled for this space.",
|
||||||
"Copy page": "Copy page",
|
"Copy page": "Copy page",
|
||||||
"Copy page to a different space.": "Copy page to a different space.",
|
"Copy page to a different space.": "Copy page to a different space.",
|
||||||
"Page copied successfully": "Page copied successfully",
|
"Page copied successfully": "Page copied successfully",
|
||||||
@@ -495,5 +517,78 @@
|
|||||||
"Page restored successfully": "Page restored successfully",
|
"Page restored successfully": "Page restored successfully",
|
||||||
"Deleted by": "Deleted by",
|
"Deleted by": "Deleted by",
|
||||||
"Deleted at": "Deleted at",
|
"Deleted at": "Deleted at",
|
||||||
"Preview": "Preview"
|
"Preview": "Preview",
|
||||||
|
"Subpages": "Subpages",
|
||||||
|
"Failed to load subpages": "Failed to load subpages",
|
||||||
|
"No subpages": "No subpages",
|
||||||
|
"Subpages (Child pages)": "Subpages (Child pages)",
|
||||||
|
"List all subpages of the current page": "List all subpages of the current page",
|
||||||
|
"Attachments": "Attachments",
|
||||||
|
"All spaces": "All spaces",
|
||||||
|
"Unknown": "Unknown",
|
||||||
|
"Find a space": "Find a space",
|
||||||
|
"Search in all your spaces": "Search in all your spaces",
|
||||||
|
"Type": "Type",
|
||||||
|
"Enterprise": "Enterprise",
|
||||||
|
"Download attachment": "Download attachment",
|
||||||
|
"Allowed email domains": "Allowed email domains",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "Only users with email addresses from these domains can signup via SSO.",
|
||||||
|
"Enter valid domain names separated by comma or space": "Enter valid domain names separated by comma or space",
|
||||||
|
"Enforce two-factor authentication": "Enforce two-factor authentication",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "Once enforced, all members must enable two-factor authentication to access the workspace.",
|
||||||
|
"Toggle MFA enforcement": "Toggle MFA enforcement",
|
||||||
|
"Display name": "Display name",
|
||||||
|
"Allow signup": "Allow signup",
|
||||||
|
"Enabled": "Enabled",
|
||||||
|
"Advanced Settings": "Advanced Settings",
|
||||||
|
"Enable TLS/SSL": "Enable TLS/SSL",
|
||||||
|
"Use secure connection to LDAP server": "Use secure connection to LDAP server",
|
||||||
|
"Group sync": "Group sync",
|
||||||
|
"No SSO providers found.": "No SSO providers found.",
|
||||||
|
"Delete SSO provider": "Delete SSO provider",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "Are you sure you want to delete this SSO provider?",
|
||||||
|
"Action": "Action",
|
||||||
|
"{{ssoProviderType}} configuration": "{{ssoProviderType}} configuration",
|
||||||
|
"Icon": "Icon",
|
||||||
|
"Upload image": "Upload image",
|
||||||
|
"Remove image": "Remove image",
|
||||||
|
"Failed to remove image": "Failed to remove image",
|
||||||
|
"Image exceeds 10MB limit.": "Image exceeds 10MB limit.",
|
||||||
|
"Image removed successfully": "Image removed successfully",
|
||||||
|
"API key": "API key",
|
||||||
|
"API key created successfully": "API key created successfully",
|
||||||
|
"API keys": "API keys",
|
||||||
|
"API management": "API management",
|
||||||
|
"Are you sure you want to revoke this API key": "Are you sure you want to revoke this API key",
|
||||||
|
"Create API Key": "Create API Key",
|
||||||
|
"Custom expiration date": "Custom expiration date",
|
||||||
|
"Enter a descriptive token name": "Enter a descriptive token name",
|
||||||
|
"Expiration": "Expiration",
|
||||||
|
"Expired": "Expired",
|
||||||
|
"Expires": "Expires",
|
||||||
|
"I've saved my API key": "I've saved my API key",
|
||||||
|
"Last use": "Last Used",
|
||||||
|
"No API keys found": "No API keys found",
|
||||||
|
"No expiration": "No expiration",
|
||||||
|
"Revoke API key": "Revoke API key",
|
||||||
|
"Revoked successfully": "Revoked successfully",
|
||||||
|
"Select expiration date": "Select expiration date",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "This action cannot be undone. Any applications using this API key will stop working.",
|
||||||
|
"Update API key": "Update API key",
|
||||||
|
"Manage API keys for all users in the workspace": "Manage API keys for all users in the workspace",
|
||||||
|
"AI settings": "AI settings",
|
||||||
|
"AI search": "AI search",
|
||||||
|
"AI Answer": "AI Answer",
|
||||||
|
"Ask AI": "Ask AI",
|
||||||
|
"AI is thinking...": "AI is thinking...",
|
||||||
|
"Ask a question...": "Ask a question...",
|
||||||
|
"AI-powered search (Ask AI)": "AI-powered search (Ask AI)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "AI search uses vector embeddings to provide semantic search capabilities across your workspace content.",
|
||||||
|
"Toggle AI search": "Toggle AI search",
|
||||||
|
"Sources": "Sources",
|
||||||
|
"Ask AI not available for attachments": "Ask AI not available for attachments",
|
||||||
|
"No answer available": "No answer available",
|
||||||
|
"Background color": "Background color",
|
||||||
|
"Highlight color": "Highlight color",
|
||||||
|
"Remove color": "Remove color"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,6 +29,7 @@
|
|||||||
"Choose your preferred interface language.": "Elige tu idioma de interfaz preferido.",
|
"Choose your preferred interface language.": "Elige tu idioma de interfaz preferido.",
|
||||||
"Choose your preferred page width.": "Elige el ancho de página que prefieras.",
|
"Choose your preferred page width.": "Elige el ancho de página que prefieras.",
|
||||||
"Confirm": "Confirmar",
|
"Confirm": "Confirmar",
|
||||||
|
"Copy as Markdown": "Copiar como Markdown",
|
||||||
"Copy link": "Copiar enlace",
|
"Copy link": "Copiar enlace",
|
||||||
"Create": "Crear",
|
"Create": "Crear",
|
||||||
"Create group": "Crear grupo",
|
"Create group": "Crear grupo",
|
||||||
@@ -53,6 +54,7 @@
|
|||||||
"e.g Space for product team": "ej: Espacio para el equipo de producto",
|
"e.g Space for product team": "ej: Espacio para el equipo de producto",
|
||||||
"e.g Space for sales team to collaborate": "ej: Espacio para que el equipo de ventas colabore",
|
"e.g Space for sales team to collaborate": "ej: Espacio para que el equipo de ventas colabore",
|
||||||
"Edit": "Editar",
|
"Edit": "Editar",
|
||||||
|
"Read": "Leer",
|
||||||
"Edit group": "Editar grupo",
|
"Edit group": "Editar grupo",
|
||||||
"Email": "Correo electrónico",
|
"Email": "Correo electrónico",
|
||||||
"Enter a strong password": "Introduce una contraseña fuerte",
|
"Enter a strong password": "Introduce una contraseña fuerte",
|
||||||
@@ -121,6 +123,8 @@
|
|||||||
"page": "página",
|
"page": "página",
|
||||||
"Page deleted successfully": "Página eliminada con éxito",
|
"Page deleted successfully": "Página eliminada con éxito",
|
||||||
"Page history": "Historial de la página",
|
"Page history": "Historial de la página",
|
||||||
|
"Select version": "Seleccionar versión",
|
||||||
|
"Highlight changes": "Resaltar cambios",
|
||||||
"Page import is in progress. Please do not close this tab.": "La importación de la página está en curso. Por favor, no cierre esta pestaña.",
|
"Page import is in progress. Please do not close this tab.": "La importación de la página está en curso. Por favor, no cierre esta pestaña.",
|
||||||
"Pages": "Páginas",
|
"Pages": "Páginas",
|
||||||
"pages": "páginas",
|
"pages": "páginas",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "Exportación fallida:",
|
"Export failed:": "Exportación fallida:",
|
||||||
"export error": "error de exportación",
|
"export error": "error de exportación",
|
||||||
"Export page": "Exportar página",
|
"Export page": "Exportar página",
|
||||||
|
"Export successful": "Exportación exitosa",
|
||||||
"Export space": "Exportar espacio",
|
"Export space": "Exportar espacio",
|
||||||
"Export {{type}}": "Exportar {{type}}",
|
"Export {{type}}": "Exportar {{type}}",
|
||||||
"File exceeds the {{limit}} attachment limit": "El archivo supera el límite de {{limit}} adjuntos",
|
"File exceeds the {{limit}} attachment limit": "El archivo supera el límite de {{limit}} adjuntos",
|
||||||
@@ -327,6 +332,8 @@
|
|||||||
"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 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 file": "Subiendo archivo",
|
||||||
"Table": "Tabla",
|
"Table": "Tabla",
|
||||||
"Insert a table.": "Insertar una tabla.",
|
"Insert a table.": "Insertar una tabla.",
|
||||||
"Insert collapsible block.": "Insertar bloque desplegable.",
|
"Insert collapsible block.": "Insertar bloque desplegable.",
|
||||||
@@ -495,5 +502,78 @@
|
|||||||
"Page restored successfully": "Página restaurada con éxito",
|
"Page restored successfully": "Página restaurada con éxito",
|
||||||
"Deleted by": "Eliminado por",
|
"Deleted by": "Eliminado por",
|
||||||
"Deleted at": "Eliminado en",
|
"Deleted at": "Eliminado en",
|
||||||
"Preview": "Vista previa"
|
"Preview": "Vista previa",
|
||||||
|
"Subpages": "Subpáginas",
|
||||||
|
"Failed to load subpages": "Error al cargar subpáginas",
|
||||||
|
"No subpages": "Sin subpáginas",
|
||||||
|
"Subpages (Child pages)": "Subpáginas (Páginas hijas)",
|
||||||
|
"List all subpages of the current page": "Listar todas las subpáginas de la página actual",
|
||||||
|
"Attachments": "Adjuntos",
|
||||||
|
"All spaces": "Todos los espacios",
|
||||||
|
"Unknown": "Desconocido",
|
||||||
|
"Find a space": "Encontrar un espacio",
|
||||||
|
"Search in all your spaces": "Buscar en todos tus espacios",
|
||||||
|
"Type": "Tipo",
|
||||||
|
"Enterprise": "Empresa",
|
||||||
|
"Download attachment": "Descargar adjunto",
|
||||||
|
"Allowed email domains": "Dominios de correo electrónico permitidos",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "Solo los usuarios con direcciones de correo electrónico de estos dominios pueden registrarse a través de SSO.",
|
||||||
|
"Enter valid domain names separated by comma or space": "Introduce nombres de dominio válidos separados por coma o espacio",
|
||||||
|
"Enforce two-factor authentication": "Aplicar autenticación de dos factores",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "Una vez aplicada, todos los miembros deben habilitar la autenticación de dos factores para acceder al espacio de trabajo.",
|
||||||
|
"Toggle MFA enforcement": "Alternar la aplicación de MFA",
|
||||||
|
"Display name": "Nombre para mostrar",
|
||||||
|
"Allow signup": "Permitir registro",
|
||||||
|
"Enabled": "Habilitado",
|
||||||
|
"Advanced Settings": "Configuración avanzada",
|
||||||
|
"Enable TLS/SSL": "Habilitar TLS/SSL",
|
||||||
|
"Use secure connection to LDAP server": "Usar conexión segura al servidor LDAP",
|
||||||
|
"Group sync": "Sincronización de grupos",
|
||||||
|
"No SSO providers found.": "No se encontraron proveedores de SSO.",
|
||||||
|
"Delete SSO provider": "Eliminar proveedor de SSO",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "¿Está seguro de que desea eliminar este proveedor de SSO?",
|
||||||
|
"Action": "Acción",
|
||||||
|
"{{ssoProviderType}} configuration": "Configuración de {{ssoProviderType}}",
|
||||||
|
"Icon": "Icono",
|
||||||
|
"Upload image": "Subir imagen",
|
||||||
|
"Remove image": "Eliminar imagen",
|
||||||
|
"Failed to remove image": "No se ha podido eliminar la imagen",
|
||||||
|
"Image exceeds 10MB limit.": "La imagen excede del límite de 10 MB",
|
||||||
|
"Image removed successfully": "Imagen eliminada correctamente",
|
||||||
|
"API key": "Clave API",
|
||||||
|
"API key created successfully": "Clave API creada correctamente",
|
||||||
|
"API keys": "Claves API",
|
||||||
|
"API management": "Gestión de API",
|
||||||
|
"Are you sure you want to revoke this API key": "¿Está seguro de que desea revocar esta clave API? ",
|
||||||
|
"Create API Key": "Crear clave API",
|
||||||
|
"Custom expiration date": "Fecha de vencimiento personalizada",
|
||||||
|
"Enter a descriptive token name": "Introduce un nombre descriptivo del token",
|
||||||
|
"Expiration": "Vencimiento",
|
||||||
|
"Expired": "Vencido",
|
||||||
|
"Expires": "Vence",
|
||||||
|
"I've saved my API key": "He guardado mi clave API",
|
||||||
|
"Last use": "Último uso",
|
||||||
|
"No API keys found": "No se han encontrado claves API",
|
||||||
|
"No expiration": "Sin vencimiento",
|
||||||
|
"Revoke API key": "Revocar clave API",
|
||||||
|
"Revoked successfully": "Revocada correctamente",
|
||||||
|
"Select expiration date": "Seleccionar fecha de vencimiento",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "Esta acción no se puede deshacer. Las aplicaciones que utilicen esta clave API dejarán de funcionar.",
|
||||||
|
"Update API key": "Actualizar clave API",
|
||||||
|
"Manage API keys for all users in the workspace": "Gestionar claves API para todos los usuarios en el espacio de trabajo",
|
||||||
|
"AI settings": "Configuración de IA",
|
||||||
|
"AI search": "Búsqueda de IA",
|
||||||
|
"AI Answer": "Respuesta de IA",
|
||||||
|
"Ask AI": "Preguntar a IA",
|
||||||
|
"AI is thinking...": "IA está pensando...",
|
||||||
|
"Ask a question...": "Haz una pregunta...",
|
||||||
|
"AI-powered search (Ask AI)": "Búsqueda impulsada por IA (Preguntar a IA)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "La búsqueda de IA utiliza incrustaciones vectoriales para proporcionar capacidades de búsqueda semántica en todo el contenido de su espacio de trabajo.",
|
||||||
|
"Toggle AI search": "Alternar búsqueda de IA",
|
||||||
|
"Sources": "Fuentes",
|
||||||
|
"Ask AI not available for attachments": "Preguntar a IA no está disponible para adjuntos",
|
||||||
|
"No answer available": "No hay respuesta disponible",
|
||||||
|
"Background color": "Color de fondo",
|
||||||
|
"Highlight color": "Color de resaltado",
|
||||||
|
"Remove color": "Eliminar color"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,6 +29,7 @@
|
|||||||
"Choose your preferred interface language.": "Choisissez votre langue d'interface préférée.",
|
"Choose your preferred interface language.": "Choisissez votre langue d'interface préférée.",
|
||||||
"Choose your preferred page width.": "Choisissez votre largeur de page préférée.",
|
"Choose your preferred page width.": "Choisissez votre largeur de page préférée.",
|
||||||
"Confirm": "Confirmer",
|
"Confirm": "Confirmer",
|
||||||
|
"Copy as Markdown": "Copier comme Markdown",
|
||||||
"Copy link": "Copier le lien",
|
"Copy link": "Copier le lien",
|
||||||
"Create": "Créer",
|
"Create": "Créer",
|
||||||
"Create group": "Créer groupe",
|
"Create group": "Créer groupe",
|
||||||
@@ -53,6 +54,7 @@
|
|||||||
"e.g Space for product team": "par ex. Espace pour l'équipe produit",
|
"e.g Space for product team": "par ex. Espace pour l'équipe produit",
|
||||||
"e.g Space for sales team to collaborate": "par ex. Espace pour l'équipe de vente pour collaborer",
|
"e.g Space for sales team to collaborate": "par ex. Espace pour l'équipe de vente pour collaborer",
|
||||||
"Edit": "Modifier",
|
"Edit": "Modifier",
|
||||||
|
"Read": "Lire",
|
||||||
"Edit group": "Modifier groupe",
|
"Edit group": "Modifier groupe",
|
||||||
"Email": "Email",
|
"Email": "Email",
|
||||||
"Enter a strong password": "Entrez un mot de passe fort",
|
"Enter a strong password": "Entrez un mot de passe fort",
|
||||||
@@ -121,6 +123,8 @@
|
|||||||
"page": "page",
|
"page": "page",
|
||||||
"Page deleted successfully": "Page supprimée avec succès",
|
"Page deleted successfully": "Page supprimée avec succès",
|
||||||
"Page history": "Historique de la page",
|
"Page history": "Historique de la page",
|
||||||
|
"Select version": "Sélectionner la version",
|
||||||
|
"Highlight changes": "Mettre en évidence les changements",
|
||||||
"Page import is in progress. Please do not close this tab.": "L'importation de la page est en cours. Veuillez ne pas fermer cet onglet.",
|
"Page import is in progress. Please do not close this tab.": "L'importation de la page est en cours. Veuillez ne pas fermer cet onglet.",
|
||||||
"Pages": "Pages",
|
"Pages": "Pages",
|
||||||
"pages": "pages",
|
"pages": "pages",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "Échec de l'exportation :",
|
"Export failed:": "Échec de l'exportation :",
|
||||||
"export error": "exporter l'erreur",
|
"export error": "exporter l'erreur",
|
||||||
"Export page": "Exporter la page",
|
"Export page": "Exporter la page",
|
||||||
|
"Export successful": "Exportation réussie",
|
||||||
"Export space": "Exporter l'espace",
|
"Export space": "Exporter l'espace",
|
||||||
"Export {{type}}": "Exporter {{type}}",
|
"Export {{type}}": "Exporter {{type}}",
|
||||||
"File exceeds the {{limit}} attachment limit": "Le fichier dépasse la limite de {{limit}} pièces jointes",
|
"File exceeds the {{limit}} attachment limit": "Le fichier dépasse la limite de {{limit}} pièces jointes",
|
||||||
@@ -327,6 +332,8 @@
|
|||||||
"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 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 file": "Téléchargement du fichier",
|
||||||
"Table": "Tableau",
|
"Table": "Tableau",
|
||||||
"Insert a table.": "Insérez un tableau.",
|
"Insert a table.": "Insérez un tableau.",
|
||||||
"Insert collapsible block.": "Insérer un bloc repliable.",
|
"Insert collapsible block.": "Insérer un bloc repliable.",
|
||||||
@@ -495,5 +502,78 @@
|
|||||||
"Page restored successfully": "Page restaurée avec succès",
|
"Page restored successfully": "Page restaurée avec succès",
|
||||||
"Deleted by": "Supprimé par",
|
"Deleted by": "Supprimé par",
|
||||||
"Deleted at": "Supprimé à",
|
"Deleted at": "Supprimé à",
|
||||||
"Preview": "Aperçu"
|
"Preview": "Aperçu",
|
||||||
|
"Subpages": "Sous-pages",
|
||||||
|
"Failed to load subpages": "Échec du chargement des sous-pages",
|
||||||
|
"No subpages": "Pas de sous-pages",
|
||||||
|
"Subpages (Child pages)": "Sous-pages (Pages enfants)",
|
||||||
|
"List all subpages of the current page": "Lister toutes les sous-pages de la page actuelle",
|
||||||
|
"Attachments": "Pièces jointes",
|
||||||
|
"All spaces": "Tous les espaces",
|
||||||
|
"Unknown": "Inconnu",
|
||||||
|
"Find a space": "Trouver un espace",
|
||||||
|
"Search in all your spaces": "Rechercher dans tous vos espaces",
|
||||||
|
"Type": "Type",
|
||||||
|
"Enterprise": "Entreprise",
|
||||||
|
"Download attachment": "Télécharger la pièce jointe",
|
||||||
|
"Allowed email domains": "Domaines de messagerie autorisés",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "Seuls les utilisateurs possédant des adresses e-mail provenant de ces domaines peuvent s'inscrire via SSO.",
|
||||||
|
"Enter valid domain names separated by comma or space": "Entrez des noms de domaine valides séparés par une virgule ou un espace",
|
||||||
|
"Enforce two-factor authentication": "Imposer l'authentification à deux facteurs",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "Une fois appliquée, tous les membres doivent activer l'authentification à deux facteurs pour accéder à l'espace de travail.",
|
||||||
|
"Toggle MFA enforcement": "Basculer l'application de l'AMF",
|
||||||
|
"Display name": "Nom d'affichage",
|
||||||
|
"Allow signup": "Autoriser l'inscription",
|
||||||
|
"Enabled": "Activé",
|
||||||
|
"Advanced Settings": "Paramètres avancés",
|
||||||
|
"Enable TLS/SSL": "Activer TLS/SSL",
|
||||||
|
"Use secure connection to LDAP server": "Utiliser une connexion sécurisée au serveur LDAP",
|
||||||
|
"Group sync": "Synchronisation de groupe",
|
||||||
|
"No SSO providers found.": "Aucun fournisseur SSO trouvé.",
|
||||||
|
"Delete SSO provider": "Supprimer le fournisseur SSO",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "Êtes-vous sûr de vouloir supprimer ce fournisseur SSO ?",
|
||||||
|
"Action": "Action",
|
||||||
|
"{{ssoProviderType}} configuration": "Configuration {{ssoProviderType}}",
|
||||||
|
"Icon": "Icône",
|
||||||
|
"Upload image": "Téléverser une image",
|
||||||
|
"Remove image": "Supprimer l'image",
|
||||||
|
"Failed to remove image": "Échec de la suppression de l'image",
|
||||||
|
"Image exceeds 10MB limit.": "L'image dépasse la limite de 10 Mo.",
|
||||||
|
"Image removed successfully": "Image supprimée avec succès",
|
||||||
|
"API key": "Clé API",
|
||||||
|
"API key created successfully": "Clé API créée avec succès",
|
||||||
|
"API keys": "Clés API",
|
||||||
|
"API management": "Gestion des API",
|
||||||
|
"Are you sure you want to revoke this API key": "Êtes-vous sûr de vouloir révoquer cette clé API",
|
||||||
|
"Create API Key": "Créer une clé API",
|
||||||
|
"Custom expiration date": "Date d'expiration personnalisée",
|
||||||
|
"Enter a descriptive token name": "Entrez un nom descriptif pour le jeton",
|
||||||
|
"Expiration": "Expiration",
|
||||||
|
"Expired": "Expiré(e)",
|
||||||
|
"Expires": "Expire",
|
||||||
|
"I've saved my API key": "J'ai enregistré ma clé API",
|
||||||
|
"Last use": "Dernière utilisation",
|
||||||
|
"No API keys found": "Aucune clé API trouvée",
|
||||||
|
"No expiration": "Pas d'expiration",
|
||||||
|
"Revoke API key": "Révoquer la clé API",
|
||||||
|
"Revoked successfully": "Révoqué(e) avec succès",
|
||||||
|
"Select expiration date": "Sélectionnez la date d'expiration",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "Cette action ne peut pas être annulée. Toutes les applications utilisant cette clé API cesseront de fonctionner.",
|
||||||
|
"Update API key": "Mettre à jour la clé API",
|
||||||
|
"Manage API keys for all users in the workspace": "Gérer les clés API pour tous les utilisateurs dans l'espace de travail",
|
||||||
|
"AI settings": "Paramètres de l'IA",
|
||||||
|
"AI search": "Recherche IA",
|
||||||
|
"AI Answer": "Réponse IA",
|
||||||
|
"Ask AI": "Demander à l'IA",
|
||||||
|
"AI is thinking...": "L'IA réfléchit...",
|
||||||
|
"Ask a question...": "Posez une question...",
|
||||||
|
"AI-powered search (Ask AI)": "Recherche assistée par l'IA (Demander à l'IA)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "La recherche IA utilise des incorporations vectorielles pour fournir des capacités de recherche sémantique à travers le contenu de votre espace de travail.",
|
||||||
|
"Toggle AI search": "Basculer la recherche IA",
|
||||||
|
"Sources": "Sources",
|
||||||
|
"Ask AI not available for attachments": "Demande à l'IA non disponible pour les pièces jointes",
|
||||||
|
"No answer available": "Pas de réponse disponible",
|
||||||
|
"Background color": "Couleur de fond",
|
||||||
|
"Highlight color": "Couleur de surbrillance",
|
||||||
|
"Remove color": "Supprimer la couleur"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,6 +29,7 @@
|
|||||||
"Choose your preferred interface language.": "Scegli la lingua da utilizzare per l'interfaccia.",
|
"Choose your preferred interface language.": "Scegli la lingua da utilizzare per l'interfaccia.",
|
||||||
"Choose your preferred page width.": "Scegli la larghezza della pagina che preferisci.",
|
"Choose your preferred page width.": "Scegli la larghezza della pagina che preferisci.",
|
||||||
"Confirm": "Conferma",
|
"Confirm": "Conferma",
|
||||||
|
"Copy as Markdown": "Copia come Markdown",
|
||||||
"Copy link": "Copia link",
|
"Copy link": "Copia link",
|
||||||
"Create": "Crea",
|
"Create": "Crea",
|
||||||
"Create group": "Crea gruppo",
|
"Create group": "Crea gruppo",
|
||||||
@@ -53,6 +54,7 @@
|
|||||||
"e.g Space for product team": "es. Spazio per il team di prodotto",
|
"e.g Space for product team": "es. Spazio per il team di prodotto",
|
||||||
"e.g Space for sales team to collaborate": "es. Spazio per la collaborazione del team di vendita",
|
"e.g Space for sales team to collaborate": "es. Spazio per la collaborazione del team di vendita",
|
||||||
"Edit": "Modifica",
|
"Edit": "Modifica",
|
||||||
|
"Read": "Leggi",
|
||||||
"Edit group": "Modifica gruppo",
|
"Edit group": "Modifica gruppo",
|
||||||
"Email": "Email",
|
"Email": "Email",
|
||||||
"Enter a strong password": "Inserisci una password sicura",
|
"Enter a strong password": "Inserisci una password sicura",
|
||||||
@@ -121,6 +123,8 @@
|
|||||||
"page": "pagina",
|
"page": "pagina",
|
||||||
"Page deleted successfully": "Pagina eliminata con successo",
|
"Page deleted successfully": "Pagina eliminata con successo",
|
||||||
"Page history": "Cronologia della pagina",
|
"Page history": "Cronologia della pagina",
|
||||||
|
"Select version": "Seleziona versione",
|
||||||
|
"Highlight changes": "Evidenzia modifiche",
|
||||||
"Page import is in progress. Please do not close this tab.": "L'importazione della pagina è in corso. Si prega di non chiudere questa scheda.",
|
"Page import is in progress. Please do not close this tab.": "L'importazione della pagina è in corso. Si prega di non chiudere questa scheda.",
|
||||||
"Pages": "Pagine",
|
"Pages": "Pagine",
|
||||||
"pages": "pagine",
|
"pages": "pagine",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "Esportazione fallita:",
|
"Export failed:": "Esportazione fallita:",
|
||||||
"export error": "errore di esportazione",
|
"export error": "errore di esportazione",
|
||||||
"Export page": "Esporta pagina",
|
"Export page": "Esporta pagina",
|
||||||
|
"Export successful": "Esportazione riuscita",
|
||||||
"Export space": "Esporta spazio",
|
"Export space": "Esporta spazio",
|
||||||
"Export {{type}}": "Esporta {{type}}",
|
"Export {{type}}": "Esporta {{type}}",
|
||||||
"File exceeds the {{limit}} attachment limit": "Il file supera il limite per gli allegati di {{limit}}",
|
"File exceeds the {{limit}} attachment limit": "Il file supera il limite per gli allegati di {{limit}}",
|
||||||
@@ -327,6 +332,8 @@
|
|||||||
"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 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 file": "Caricamento file",
|
||||||
"Table": "Tabella",
|
"Table": "Tabella",
|
||||||
"Insert a table.": "Inserisci una tabella.",
|
"Insert a table.": "Inserisci una tabella.",
|
||||||
"Insert collapsible block.": "Inserisci blocco comprimibile.",
|
"Insert collapsible block.": "Inserisci blocco comprimibile.",
|
||||||
@@ -495,5 +502,78 @@
|
|||||||
"Page restored successfully": "Pagina ripristinata con successo",
|
"Page restored successfully": "Pagina ripristinata con successo",
|
||||||
"Deleted by": "Eliminato da",
|
"Deleted by": "Eliminato da",
|
||||||
"Deleted at": "Eliminato il",
|
"Deleted at": "Eliminato il",
|
||||||
"Preview": "Anteprima"
|
"Preview": "Anteprima",
|
||||||
|
"Subpages": "Sottopagine",
|
||||||
|
"Failed to load subpages": "Caricamento delle sottopagine non riuscito",
|
||||||
|
"No subpages": "Nessuna sottopagina",
|
||||||
|
"Subpages (Child pages)": "Sottopagine (Pagine figlie)",
|
||||||
|
"List all subpages of the current page": "Elenca tutte le sottopagine della pagina corrente",
|
||||||
|
"Attachments": "Allegati",
|
||||||
|
"All spaces": "Tutti gli spazi",
|
||||||
|
"Unknown": "Sconosciuto",
|
||||||
|
"Find a space": "Trova uno spazio",
|
||||||
|
"Search in all your spaces": "Cerca in tutti i tuoi spazi",
|
||||||
|
"Type": "Tipo",
|
||||||
|
"Enterprise": "Impresa",
|
||||||
|
"Download attachment": "Scarica allegato",
|
||||||
|
"Allowed email domains": "Domini email consentiti",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "Solo gli utenti con indirizzi email provenienti da questi domini possono registrarsi tramite SSO.",
|
||||||
|
"Enter valid domain names separated by comma or space": "Inserisci nomi di dominio validi separati da virgole o spazi",
|
||||||
|
"Enforce two-factor authentication": "Imponi l'autenticazione a due fattori",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "Una volta impostata, tutti i membri devono abilitare l'autenticazione a due fattori per accedere all'area di lavoro.",
|
||||||
|
"Toggle MFA enforcement": "Attiva disattiva l'applicazione MFA",
|
||||||
|
"Display name": "Nome visualizzato",
|
||||||
|
"Allow signup": "Consenti iscrizione",
|
||||||
|
"Enabled": "Abilitato",
|
||||||
|
"Advanced Settings": "Impostazioni avanzate",
|
||||||
|
"Enable TLS/SSL": "Abilita TLS/SSL",
|
||||||
|
"Use secure connection to LDAP server": "Usa connessione sicura al server LDAP",
|
||||||
|
"Group sync": "Sincronizzazione gruppi",
|
||||||
|
"No SSO providers found.": "Nessun provider SSO trovato.",
|
||||||
|
"Delete SSO provider": "Elimina provider SSO",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "Sei sicuro di voler eliminare questo provider SSO?",
|
||||||
|
"Action": "Azione",
|
||||||
|
"{{ssoProviderType}} configuration": "Configurazione {{ssoProviderType}}",
|
||||||
|
"Icon": "Icona",
|
||||||
|
"Upload image": "Carica immagine",
|
||||||
|
"Remove image": "Rimuovi immagine",
|
||||||
|
"Failed to remove image": "Rimozione immagine fallita",
|
||||||
|
"Image exceeds 10MB limit.": "L'immagine supera il limite di 10MB.",
|
||||||
|
"Image removed successfully": "Immagine rimossa con successo",
|
||||||
|
"API key": "Chiave API",
|
||||||
|
"API key created successfully": "Chiave API creata con successo",
|
||||||
|
"API keys": "Chiavi API",
|
||||||
|
"API management": "Gestione API",
|
||||||
|
"Are you sure you want to revoke this API key": "Sei sicuro di voler revocare questa chiave API",
|
||||||
|
"Create API Key": "Crea Chiave API",
|
||||||
|
"Custom expiration date": "Data di scadenza personalizzata",
|
||||||
|
"Enter a descriptive token name": "Inserisci un nome descrittivo del token",
|
||||||
|
"Expiration": "Scadenza",
|
||||||
|
"Expired": "Scaduto",
|
||||||
|
"Expires": "Scade",
|
||||||
|
"I've saved my API key": "Ho salvato la mia chiave API",
|
||||||
|
"Last use": "Ultimo utilizzo",
|
||||||
|
"No API keys found": "Nessuna chiave API trovata",
|
||||||
|
"No expiration": "Nessuna scadenza",
|
||||||
|
"Revoke API key": "Revoca chiave API",
|
||||||
|
"Revoked successfully": "Revocata con successo",
|
||||||
|
"Select expiration date": "Seleziona la data di scadenza",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "Questa azione non può essere annullata. Qualsiasi applicazione che utilizza questa chiave API smetterà di funzionare.",
|
||||||
|
"Update API key": "Aggiorna chiave API",
|
||||||
|
"Manage API keys for all users in the workspace": "Gestisci le chiavi API per tutti gli utenti nell'area di lavoro",
|
||||||
|
"AI settings": "Impostazioni AI",
|
||||||
|
"AI search": "Ricerca AI",
|
||||||
|
"AI Answer": "Risposta AI",
|
||||||
|
"Ask AI": "Chiedi all'AI",
|
||||||
|
"AI is thinking...": "L'AI sta pensando...",
|
||||||
|
"Ask a question...": "Fai una domanda...",
|
||||||
|
"AI-powered search (Ask AI)": "Ricerca potenziata dall'AI (Chiedi all'AI)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "La ricerca AI utilizza embeddings vettoriali per fornire capacità di ricerca semantica nel contenuto della tua area di lavoro.",
|
||||||
|
"Toggle AI search": "Attiva/disattiva ricerca AI",
|
||||||
|
"Sources": "Fonti",
|
||||||
|
"Ask AI not available for attachments": "Chiedere all'AI non è disponibile per gli allegati",
|
||||||
|
"No answer available": "Nessuna risposta disponibile",
|
||||||
|
"Background color": "Colore di sfondo",
|
||||||
|
"Highlight color": "Colore evidenziato",
|
||||||
|
"Remove color": "Rimuovi colore"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,22 +13,23 @@
|
|||||||
"Are you sure you want to remove this user from the group? The user will lose access to resources this group has access to.": "このユーザをグループから削除してもよろしいですか? ユーザはこのグループがアクセス権を持つリソースにアクセスできなくなります。",
|
"Are you sure you want to remove this user from the group? The user will lose access to resources this group has access to.": "このユーザをグループから削除してもよろしいですか? ユーザはこのグループがアクセス権を持つリソースにアクセスできなくなります。",
|
||||||
"Are you sure you want to remove this user from the space? The user will lose all access to this space.": "このユーザをスペースから削除してもよろしいですか? ユーザはこのスペースへのアクセス権をすべて失います。",
|
"Are you sure you want to remove this user from the space? The user will lose all access to this space.": "このユーザをスペースから削除してもよろしいですか? ユーザはこのスペースへのアクセス権をすべて失います。",
|
||||||
"Are you sure you want to restore this version? Any changes not versioned will be lost.": "このバージョンを復元してもよろしいですか? バージョン管理されていない変更は失われます。",
|
"Are you sure you want to restore this version? Any changes not versioned will be lost.": "このバージョンを復元してもよろしいですか? バージョン管理されていない変更は失われます。",
|
||||||
"Can become members of groups and spaces in workspace": "ワークスペース内のグループやスペースのメンバーになることができます",
|
"Can become members of groups and spaces in workspace": "ワークスペース内のグループやスペースのメンバーになれます",
|
||||||
"Can create and edit pages in space.": "スペース内のページを作成および編集できます。",
|
"Can create and edit pages in space.": "スペース内のページを作成・編集できます",
|
||||||
"Can edit": "編集可能",
|
"Can edit": "編集可能",
|
||||||
"Can manage workspace": "ワークスペースを管理できます",
|
"Can manage workspace": "ワークスペースを管理できます",
|
||||||
"Can manage workspace but cannot delete it": "ワークスペースを管理できますが、削除はできません",
|
"Can manage workspace but cannot delete it": "ワークスペースを管理できますが削除はできません",
|
||||||
"Can view": "閲覧可能",
|
"Can view": "閲覧可能",
|
||||||
"Can view pages in space but not edit.": "スペース内のページを閲覧できますが、編集はできません。",
|
"Can view pages in space but not edit.": "スペース内のページを閲覧できますが編集はできません",
|
||||||
"Cancel": "キャンセル",
|
"Cancel": "キャンセル",
|
||||||
"Change email": "メールアドレスの変更",
|
"Change email": "メールアドレスの変更",
|
||||||
"Change password": "パスワードの変更",
|
"Change password": "パスワードの変更",
|
||||||
"Change photo": "画像の変更",
|
"Change photo": "画像の変更",
|
||||||
"Choose a role": "ロールを選んでください",
|
"Choose a role": "ロールを選んでください",
|
||||||
"Choose your preferred color scheme.": "お好みのカラースキームを選択してください。",
|
"Choose your preferred color scheme.": "お好みのカラースキームを選択してください",
|
||||||
"Choose your preferred interface language.": "お好みのインターフェース言語を選択してください。",
|
"Choose your preferred interface language.": "お好みの言語を選択してください",
|
||||||
"Choose your preferred page width.": "左右の余白を縮小する場合はオンにしてください。",
|
"Choose your preferred page width.": "お好みのページ幅を選択してください",
|
||||||
"Confirm": "確認",
|
"Confirm": "確認",
|
||||||
|
"Copy as Markdown": "Markdownとしてコピー",
|
||||||
"Copy link": "リンクをコピー",
|
"Copy link": "リンクをコピー",
|
||||||
"Create": "新規作成",
|
"Create": "新規作成",
|
||||||
"Create group": "グループを作成",
|
"Create group": "グループを作成",
|
||||||
@@ -40,23 +41,24 @@
|
|||||||
"Date": "日付",
|
"Date": "日付",
|
||||||
"Delete": "削除",
|
"Delete": "削除",
|
||||||
"Delete group": "グループを削除",
|
"Delete group": "グループを削除",
|
||||||
"Are you sure you want to delete this page? This will delete its children and page history. This action is irreversible.": "このページを削除してもよろしいですか?この操作により、子ページおよびページ履歴が削除されます。この操作は元に戻せません。",
|
"Are you sure you want to delete this page? This will delete its children and page history. This action is irreversible.": "このページを削除してもよろしいですか?子ページとページ履歴も削除されます。この操作は取り消せません。",
|
||||||
"Description": "説明",
|
"Description": "説明",
|
||||||
"Details": "詳細",
|
"Details": "詳細",
|
||||||
"e.g ACME": "例: 山田太郎",
|
"e.g ACME": "例: 山田太郎",
|
||||||
"e.g ACME Inc": "例: 株式会社サンプル",
|
"e.g ACME Inc": "例: 株式会社サンプル",
|
||||||
"e.g Developers": "例: エンジニア",
|
"e.g Developers": "例: エンジニア",
|
||||||
"e.g Group for developers": "例: エンジニアグループ",
|
"e.g Group for developers": "例: 開発チーム",
|
||||||
"e.g product": "例: product",
|
"e.g product": "例: product",
|
||||||
"e.g Product Team": "例: 製品チーム",
|
"e.g Product Team": "例: プロダクトチーム",
|
||||||
"e.g Sales": "例: 営業",
|
"e.g Sales": "例: 営業部",
|
||||||
"e.g Space for product team": "例: 製品チームのスペース",
|
"e.g Space for product team": "例: プロダクトチーム用スペース",
|
||||||
"e.g Space for sales team to collaborate": "例: 営業チーム連携用スペース",
|
"e.g Space for sales team to collaborate": "例: 営業チーム用スペース",
|
||||||
"Edit": "編集",
|
"Edit": "編集",
|
||||||
|
"Read": "閲覧",
|
||||||
"Edit group": "グループを編集",
|
"Edit group": "グループを編集",
|
||||||
"Email": "メールアドレス",
|
"Email": "メールアドレス",
|
||||||
"Enter a strong password": "強力なパスワードを入力してください",
|
"Enter a strong password": "強力なパスワードを入力してください",
|
||||||
"Enter valid email addresses separated by comma or space max_50": "有効なメールアドレスをカンマまたはスペースで区切って入力してください(最大 50 個)",
|
"Enter valid email addresses separated by comma or space max_50": "メールアドレスをカンマまたはスペース区切りで入力(最大50件)",
|
||||||
"enter valid emails addresses": "有効なメールアドレスを入力してください",
|
"enter valid emails addresses": "有効なメールアドレスを入力してください",
|
||||||
"Enter your current password": "現在のパスワードを入力してください",
|
"Enter your current password": "現在のパスワードを入力してください",
|
||||||
"enter your full name": "氏名を入力してください",
|
"enter your full name": "氏名を入力してください",
|
||||||
@@ -80,18 +82,18 @@
|
|||||||
"Group description": "グループ説明",
|
"Group description": "グループ説明",
|
||||||
"Group name": "グループ名",
|
"Group name": "グループ名",
|
||||||
"Groups": "グループ",
|
"Groups": "グループ",
|
||||||
"Has full access to space settings and pages.": "スペース設定とページにフルアクセスできます。",
|
"Has full access to space settings and pages.": "スペース設定とページにフルアクセスできます",
|
||||||
"Home": "ホーム",
|
"Home": "ホーム",
|
||||||
"Import pages": "ページをインポート",
|
"Import pages": "ページをインポート",
|
||||||
"Import pages & space settings": "ページとスペース設定をインポート",
|
"Import pages & space settings": "ページとスペース設定をインポート",
|
||||||
"Importing pages": "ページをインポートしています",
|
"Importing pages": "ページをインポートしています",
|
||||||
"invalid invitation link": "招待リンクが間違っています",
|
"invalid invitation link": "無効な招待リンクです",
|
||||||
"Invitation signup": "招待登録",
|
"Invitation signup": "招待登録",
|
||||||
"Invite by email": "メールアドレスで招待する",
|
"Invite by email": "メールアドレスで招待する",
|
||||||
"Invite members": "メンバーを招待する",
|
"Invite members": "メンバーを招待する",
|
||||||
"Invite new members": "新しいメンバーを招待する",
|
"Invite new members": "新しいメンバーを招待する",
|
||||||
"Invited members who are yet to accept their invitation will appear here.": "招待をまだ承諾していないメンバーはここに表示されます。",
|
"Invited members who are yet to accept their invitation will appear here.": "招待を承諾していないメンバーがここに表示されます",
|
||||||
"Invited members will be granted access to spaces the groups can access": "招待されたメンバーは、グループがアクセスできるスペースにアクセス権が付与されます",
|
"Invited members will be granted access to spaces the groups can access": "招待されたメンバーはグループがアクセスできるスペースにアクセスできます",
|
||||||
"Join the workspace": "ワークスペースに参加",
|
"Join the workspace": "ワークスペースに参加",
|
||||||
"Language": "言語",
|
"Language": "言語",
|
||||||
"Light": "ライト",
|
"Light": "ライト",
|
||||||
@@ -112,20 +114,22 @@
|
|||||||
"New page": "新規ページ",
|
"New page": "新規ページ",
|
||||||
"New password": "新しいパスワード",
|
"New password": "新しいパスワード",
|
||||||
"No group found": "グループが見つかりません",
|
"No group found": "グループが見つかりません",
|
||||||
"No page history saved yet.": "まだページの履歴が保存されていません。",
|
"No page history saved yet.": "ページ履歴がありません",
|
||||||
"No pages yet": "ページがありません",
|
"No pages yet": "ページがありません",
|
||||||
"No results found...": "結果が見つかりませんでした...",
|
"No results found...": "結果が見つかりません",
|
||||||
"No user found": "ユーザがいません",
|
"No user found": "ユーザーが見つかりません",
|
||||||
"Overview": "概要",
|
"Overview": "概要",
|
||||||
"Owner": "所有者",
|
"Owner": "所有者",
|
||||||
"page": "ページ",
|
"page": "ページ",
|
||||||
"Page deleted successfully": "ページが正常に削除されました",
|
"Page deleted successfully": "ページを削除しました",
|
||||||
"Page history": "ページの履歴",
|
"Page history": "ページ履歴",
|
||||||
"Page import is in progress. Please do not close this tab.": "ページのインポートが進行中です。このタブを閉じないでください。",
|
"Select version": "バージョンを選択",
|
||||||
|
"Highlight changes": "変更を強調表示",
|
||||||
|
"Page import is in progress. Please do not close this tab.": "ページをインポート中です。このタブを閉じないでください",
|
||||||
"Pages": "ページ",
|
"Pages": "ページ",
|
||||||
"pages": "ページ",
|
"pages": "ページ",
|
||||||
"Password": "パスワード",
|
"Password": "パスワード",
|
||||||
"Password changed successfully": "パスワードが正常に変更されました",
|
"Password changed successfully": "パスワードを変更しました",
|
||||||
"Pending": "保留中",
|
"Pending": "保留中",
|
||||||
"Please confirm your action": "アクションを確認してください",
|
"Please confirm your action": "アクションを確認してください",
|
||||||
"Preferences": "設定",
|
"Preferences": "設定",
|
||||||
@@ -142,95 +146,95 @@
|
|||||||
"Search for groups": "グループを検索",
|
"Search for groups": "グループを検索",
|
||||||
"Search for users": "ユーザーを検索",
|
"Search for users": "ユーザーを検索",
|
||||||
"Search for users and groups": "ユーザーとグループを検索",
|
"Search for users and groups": "ユーザーとグループを検索",
|
||||||
"Search...": "検索する語句を入力",
|
"Search...": "検索",
|
||||||
"Select language": "言語を選択",
|
"Select language": "言語を選択",
|
||||||
"Select role": "ロールを選択",
|
"Select role": "ロールを選択",
|
||||||
"Select role to assign to all invited members": "招待されたすべてのメンバーに割り当てるロールを選択してください",
|
"Select role to assign to all invited members": "招待するメンバーに割り当てるロールを選択",
|
||||||
"Select theme": "テーマを選択",
|
"Select theme": "テーマを選択",
|
||||||
"Send invitation": "招待を送る",
|
"Send invitation": "招待を送る",
|
||||||
"Invitation sent": "招待が送信されました",
|
"Invitation sent": "招待を送信しました",
|
||||||
"Settings": "設定",
|
"Settings": "設定",
|
||||||
"Setup workspace": "ワークスペースを設定する",
|
"Setup workspace": "ワークスペースを設定する",
|
||||||
"Sign In": "サインイン",
|
"Sign In": "サインイン",
|
||||||
"Sign Up": "アカウント登録",
|
"Sign Up": "新規登録",
|
||||||
"Slug": "Slug (URL用文字列)",
|
"Slug": "スラッグ(URL識別子)",
|
||||||
"Space": "スペース",
|
"Space": "スペース",
|
||||||
"Space description": "スペース説明",
|
"Space description": "スペース説明",
|
||||||
"Space menu": "スペースメニュー",
|
"Space menu": "スペースメニュー",
|
||||||
"Space name": "スペース名",
|
"Space name": "スペース名",
|
||||||
"Space settings": "スペース設定",
|
"Space settings": "スペース設定",
|
||||||
"Space slug": "スペースのSlug (URL用文字列)",
|
"Space slug": "スペースのスラッグ(URL識別子)",
|
||||||
"Spaces": "スペース",
|
"Spaces": "スペース",
|
||||||
"Spaces you belong to": "所属しているスペース",
|
"Spaces you belong to": "所属しているスペース",
|
||||||
"No space found": "スペースが見つかりません",
|
"No space found": "スペースが見つかりません",
|
||||||
"Search for spaces": "スペースを検索",
|
"Search for spaces": "スペースを検索",
|
||||||
"Start typing to search...": "検索を開始するには入力してください...",
|
"Start typing to search...": "入力して検索",
|
||||||
"Status": "ステータス",
|
"Status": "ステータス",
|
||||||
"Successfully imported": "インポートに成功しました",
|
"Successfully imported": "インポートしました",
|
||||||
"Successfully restored": "正常に復元されました",
|
"Successfully restored": "復元しました",
|
||||||
"System settings": "システム設定",
|
"System settings": "システム設定",
|
||||||
"Theme": "テーマ",
|
"Theme": "テーマ",
|
||||||
"To change your email, you have to enter your password and new email.": "メールアドレスを変更するには、パスワードと新しいメールアドレスを入力する必要があります。",
|
"To change your email, you have to enter your password and new email.": "メールアドレスを変更するには、パスワードと新しいメールアドレスを入力してください",
|
||||||
"Toggle full page width": "ページ幅を切り替える",
|
"Toggle full page width": "ページ幅を切り替え",
|
||||||
"Unable to import pages. Please try again.": "ページをインポートできません。もう一度お試しください。",
|
"Unable to import pages. Please try again.": "ページをインポートできませんでした。もう一度お試しください",
|
||||||
"untitled": "無題",
|
"untitled": "無題",
|
||||||
"Untitled": "無題",
|
"Untitled": "無題",
|
||||||
"Updated successfully": "正常に更新されました",
|
"Updated successfully": "更新しました",
|
||||||
"User": "ユーザー",
|
"User": "ユーザー",
|
||||||
"Workspace": "ワークスペース",
|
"Workspace": "ワークスペース",
|
||||||
"Workspace Name": "ワークスペース名",
|
"Workspace Name": "ワークスペース名",
|
||||||
"Workspace settings": "ワークスペース設定",
|
"Workspace settings": "ワークスペース設定",
|
||||||
"You can change your password here.": "パスワードを変更できます。",
|
"You can change your password here.": "パスワードを変更できます",
|
||||||
"Your Email": "メールアドレス",
|
"Your Email": "メールアドレス",
|
||||||
"Your import is complete.": "インポートが完了しました。",
|
"Your import is complete.": "インポートが完了しました",
|
||||||
"Your name": "名前",
|
"Your name": "名前",
|
||||||
"Your Name": "名前",
|
"Your Name": "名前",
|
||||||
"Your password": "パスワード",
|
"Your password": "パスワード",
|
||||||
"Your password must be a minimum of 8 characters.": "パスワードは最低 8 文字必要です。",
|
"Your password must be a minimum of 8 characters.": "パスワードは8文字以上にしてください",
|
||||||
"Sidebar toggle": "サイドバー切り替え",
|
"Sidebar toggle": "サイドバー切り替え",
|
||||||
"Comments": "コメント",
|
"Comments": "コメント",
|
||||||
"404 page not found": "404 ページが見つかりません",
|
"404 page not found": "404 ページが見つかりません",
|
||||||
"Sorry, we can't find the page you are looking for.": "お探しのページが見つかりません。",
|
"Sorry, we can't find the page you are looking for.": "お探しのページが見つかりません",
|
||||||
"Take me back to homepage": "ホームに戻る",
|
"Take me back to homepage": "ホームに戻る",
|
||||||
"Forgot password": "パスワードを忘れた",
|
"Forgot password": "パスワードを忘れた",
|
||||||
"Forgot your password?": "パスワードを忘れましたか?",
|
"Forgot your password?": "パスワードを忘れましたか?",
|
||||||
"A password reset link has been sent to your email. Please check your inbox.": "パスワードリセットリンクがあなたのメールアドレスに送信されました。受信箱を確認してください。",
|
"A password reset link has been sent to your email. Please check your inbox.": "パスワードリセット用のリンクをメールに送信しました。受信トレイを確認してください",
|
||||||
"Send reset link": "リセットリンクを送る",
|
"Send reset link": "リセットリンクを送信",
|
||||||
"Password reset": "パスワードリセット",
|
"Password reset": "パスワードリセット",
|
||||||
"Your new password": "新しいパスワード",
|
"Your new password": "新しいパスワード",
|
||||||
"Set password": "パスワードを設定",
|
"Set password": "パスワードを設定",
|
||||||
"Write a comment": "コメントを書く",
|
"Write a comment": "コメントを書く",
|
||||||
"Reply...": "返信...",
|
"Reply...": "返信...",
|
||||||
"Error loading comments.": "コメントの読み込み中にエラーが発生しました。",
|
"Error loading comments.": "コメントの読み込みに失敗しました",
|
||||||
"No comments yet.": "コメントがありません。",
|
"No comments yet.": "コメントがありません",
|
||||||
"Edit comment": "コメントを編集する",
|
"Edit comment": "コメントを編集する",
|
||||||
"Delete comment": "コメントを削除する",
|
"Delete comment": "コメントを削除する",
|
||||||
"Are you sure you want to delete this comment?": "このコメントを削除してもよろしいですか?",
|
"Are you sure you want to delete this comment?": "このコメントを削除してもよろしいですか?",
|
||||||
"Comment created successfully": "コメントが作成されました",
|
"Comment created successfully": "コメントを作成しました",
|
||||||
"Error creating comment": "コメントの作成中にエラーが発生しました",
|
"Error creating comment": "コメントの作成に失敗しました",
|
||||||
"Comment updated successfully": "コメントが更新されました",
|
"Comment updated successfully": "コメントを更新しました",
|
||||||
"Failed to update comment": "コメントの更新に失敗しました",
|
"Failed to update comment": "コメントの更新に失敗しました",
|
||||||
"Comment deleted successfully": "コメントが削除されました",
|
"Comment deleted successfully": "コメントを削除しました",
|
||||||
"Failed to delete comment": "コメントの削除に失敗しました",
|
"Failed to delete comment": "コメントの削除に失敗しました",
|
||||||
"Comment resolved successfully": "コメントが解決されました",
|
"Comment resolved successfully": "コメントを解決しました",
|
||||||
"Comment re-opened successfully": "コメントが再開されました",
|
"Comment re-opened successfully": "コメントを再開しました",
|
||||||
"Comment unresolved successfully": "コメントが再解決されました",
|
"Comment unresolved successfully": "コメントを未解決に戻しました",
|
||||||
"Failed to resolve comment": "コメントの解決に失敗しました",
|
"Failed to resolve comment": "コメントの解決に失敗しました",
|
||||||
"Resolve comment": "コメントを解決",
|
"Resolve comment": "コメントを解決",
|
||||||
"Unresolve comment": "コメントを再解決",
|
"Unresolve comment": "コメントを未解決に戻す",
|
||||||
"Resolve Comment Thread": "コメントスレッドを解決",
|
"Resolve Comment Thread": "コメントスレッドを解決",
|
||||||
"Unresolve Comment Thread": "コメントスレッドを再解決",
|
"Unresolve Comment Thread": "コメントスレッドを未解決に戻す",
|
||||||
"Are you sure you want to resolve this comment thread? This will mark it as completed.": "このコメントスレッドを解決しますか? これにより完了としてマークされます。",
|
"Are you sure you want to resolve this comment thread? This will mark it as completed.": "このコメントスレッドを解決しますか?完了としてマークされます",
|
||||||
"Are you sure you want to unresolve this comment thread?": "このコメントスレッドを再解決しますか?",
|
"Are you sure you want to unresolve this comment thread?": "このコメントスレッドを未解決に戻しますか?",
|
||||||
"Resolved": "解決済",
|
"Resolved": "解決済",
|
||||||
"No active comments.": "アクティブなコメントはありません。",
|
"No active comments.": "アクティブなコメントはありません",
|
||||||
"No resolved comments.": "解決されたコメントはありません。",
|
"No resolved comments.": "解決済みのコメントはありません",
|
||||||
"Revoke invitation": "招待を取り消す",
|
"Revoke invitation": "招待を取り消す",
|
||||||
"Revoke": "取り消す",
|
"Revoke": "取り消す",
|
||||||
"Don't": "取り消さない",
|
"Don't": "取り消さない",
|
||||||
"Are you sure you want to revoke this invitation? The user will not be able to join the workspace.": "この招待を取り消してもよろしいですか? ユーザはワークスペースに参加できなくなります。",
|
"Are you sure you want to revoke this invitation? The user will not be able to join the workspace.": "この招待を取り消してもよろしいですか?ユーザーはワークスペースに参加できなくなります",
|
||||||
"Resend invitation": "招待を再度送る",
|
"Resend invitation": "招待を再度送る",
|
||||||
"Anyone with this link can join this workspace.": "このリンクを持っている人は誰でもこのワークスペースに参加できます。",
|
"Anyone with this link can join this workspace.": "このリンクを知っている人は誰でもワークスペースに参加できます",
|
||||||
"Invite link": "招待リンク",
|
"Invite link": "招待リンク",
|
||||||
"Copy": "コピー",
|
"Copy": "コピー",
|
||||||
"Copy to space": "スペースにコピー",
|
"Copy to space": "スペースにコピー",
|
||||||
@@ -238,13 +242,13 @@
|
|||||||
"Duplicate": "複製",
|
"Duplicate": "複製",
|
||||||
"Select a user": "ユーザを選択",
|
"Select a user": "ユーザを選択",
|
||||||
"Select a group": "グループを選択",
|
"Select a group": "グループを選択",
|
||||||
"Export all pages and attachments in this space.": "このスペースのすべてのページと添付ファイルをエクスポートします。",
|
"Export all pages and attachments in this space.": "このスペースのすべてのページと添付ファイルをエクスポートします",
|
||||||
"Delete space": "スペースを削除",
|
"Delete space": "スペースを削除",
|
||||||
"Are you sure you want to delete this space?": "このスペースを削除してもよろしいですか?",
|
"Are you sure you want to delete this space?": "このスペースを削除してもよろしいですか?",
|
||||||
"Delete this space with all its pages and data.": "このスペースおよびスペース内のすべてのページとデータを削除します。",
|
"Delete this space with all its pages and data.": "このスペースとすべてのページ、データを削除します",
|
||||||
"All pages, comments, attachments and permissions in this space will be deleted irreversibly.": "このスペース内のすべてのページ、コメント、添付ファイル、および権限は完全に削除されます。",
|
"All pages, comments, attachments and permissions in this space will be deleted irreversibly.": "スペース内のすべてのページ、コメント、添付ファイル、権限が完全に削除されます",
|
||||||
"Confirm space name": "スペース名を確認する",
|
"Confirm space name": "スペース名を確認する",
|
||||||
"Type the space name <b>{{spaceName}}</b> to confirm your action.": "アクションを確認するためにスペース名 <b>{{spaceName}}</b> を入力してください。",
|
"Type the space name <b>{{spaceName}}</b> to confirm your action.": "確認のためスペース名 <b>{{spaceName}}</b> を入力してください",
|
||||||
"Format": "フォーマット",
|
"Format": "フォーマット",
|
||||||
"Include subpages": "サブページを含める",
|
"Include subpages": "サブページを含める",
|
||||||
"Include attachments": "添付ファイルを含める",
|
"Include attachments": "添付ファイルを含める",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "エクスポートに失敗しました:",
|
"Export failed:": "エクスポートに失敗しました:",
|
||||||
"export error": "エクスポートエラー",
|
"export error": "エクスポートエラー",
|
||||||
"Export page": "エクスポートページ",
|
"Export page": "エクスポートページ",
|
||||||
|
"Export successful": "エクスポート成功",
|
||||||
"Export space": "エクスポートスペース",
|
"Export space": "エクスポートスペース",
|
||||||
"Export {{type}}": "{{type}}をエクスポート",
|
"Export {{type}}": "{{type}}をエクスポート",
|
||||||
"File exceeds the {{limit}} attachment limit": "ファイルが{{limit}}の添付制限を超えています",
|
"File exceeds the {{limit}} attachment limit": "ファイルが{{limit}}の添付制限を超えています",
|
||||||
@@ -272,12 +277,12 @@
|
|||||||
"Success": "成功",
|
"Success": "成功",
|
||||||
"Warning": "警告",
|
"Warning": "警告",
|
||||||
"Danger": "危険",
|
"Danger": "危険",
|
||||||
"Mermaid diagram error:": "Mermaid コードエラー",
|
"Mermaid diagram error:": "Mermaid ダイアグラムエラー:",
|
||||||
"Invalid Mermaid diagram": "無効な Mermaid コードです",
|
"Invalid Mermaid diagram": "無効な Mermaid ダイアグラムです",
|
||||||
"Double-click to edit Draw.io diagram": "ダブルクリックしてDraw.ioの図を編集",
|
"Double-click to edit Draw.io diagram": "ダブルクリックして Draw.io 図を編集",
|
||||||
"Exit": "終了",
|
"Exit": "終了",
|
||||||
"Save & Exit": "保存して終了",
|
"Save & Exit": "保存して終了",
|
||||||
"Double-click to edit Excalidraw diagram": "ダブルクリックしてExcalidraw図を編集",
|
"Double-click to edit Excalidraw diagram": "ダブルクリックして Excalidraw 図を編集",
|
||||||
"Paste link": "リンクを貼り付け",
|
"Paste link": "リンクを貼り付け",
|
||||||
"Edit link": "リンクを編集",
|
"Edit link": "リンクを編集",
|
||||||
"Remove link": "リンクを削除",
|
"Remove link": "リンクを削除",
|
||||||
@@ -314,22 +319,24 @@
|
|||||||
"Bullet List": "箇条書きリスト",
|
"Bullet List": "箇条書きリスト",
|
||||||
"Numbered List": "番号付きリスト",
|
"Numbered List": "番号付きリスト",
|
||||||
"Blockquote": "引用",
|
"Blockquote": "引用",
|
||||||
"Just start typing with plain text.": "すぐに文章を書き始められます。",
|
"Just start typing with plain text.": "プレーンテキストを入力します",
|
||||||
"Track tasks with a to-do list.": "Todoリストでタスクを追跡します。",
|
"Track tasks with a to-do list.": "Todo リストでタスクを管理します",
|
||||||
"Big section heading.": "大きいフォントのセクション見出しです。",
|
"Big section heading.": "大見出し",
|
||||||
"Medium section heading.": "中くらいのフォントのセクション見出しです。",
|
"Medium section heading.": "中見出し",
|
||||||
"Small section heading.": "小さいフォントのセクション見出しです。",
|
"Small section heading.": "小見出し",
|
||||||
"Create a simple bullet list.": "シンプルな箇条書きのリストを作成します。",
|
"Create a simple bullet list.": "箇条書きリストを作成します",
|
||||||
"Create a list with numbering.": "番号付きのリストを作成します。",
|
"Create a list with numbering.": "番号付きリストを作成します",
|
||||||
"Create block quote.": "引用文を作成します。",
|
"Create block quote.": "引用ブロックを作成します",
|
||||||
"Insert code snippet.": "コードスニペットを入力します。",
|
"Insert code snippet.": "コードスニペットを挿入します",
|
||||||
"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 file from your device.": "ファイルをアップロードします。",
|
"Upload any file from your device.": "デバイスからファイルをアップロードします",
|
||||||
|
"Uploading {{name}}": "{{name}} をアップロード中",
|
||||||
|
"Uploading file": "ファイルをアップロード中",
|
||||||
"Table": "テーブル",
|
"Table": "テーブル",
|
||||||
"Insert a table.": "表を挿入します。",
|
"Insert a table.": "テーブルを挿入します",
|
||||||
"Insert collapsible block.": "折りたたみ可能なブロックを挿入します。",
|
"Insert collapsible block.": "折りたたみブロックを挿入します",
|
||||||
"Video": "動画",
|
"Video": "動画",
|
||||||
"Divider": "区切り線",
|
"Divider": "区切り線",
|
||||||
"Quote": "引用",
|
"Quote": "引用",
|
||||||
@@ -337,16 +344,16 @@
|
|||||||
"File attachment": "ファイル添付",
|
"File attachment": "ファイル添付",
|
||||||
"Toggle block": "ブロックを切り替える",
|
"Toggle block": "ブロックを切り替える",
|
||||||
"Callout": "コールアウト",
|
"Callout": "コールアウト",
|
||||||
"Insert callout notice.": "コールアウトブロックを挿入します。",
|
"Insert callout notice.": "コールアウトを挿入します",
|
||||||
"Math inline": "インライン数式",
|
"Math inline": "インライン数式",
|
||||||
"Insert inline math equation.": "インライン数式を挿入します。",
|
"Insert inline math equation.": "インライン数式を挿入します",
|
||||||
"Math block": "数式ブロック",
|
"Math block": "数式ブロック",
|
||||||
"Insert math equation": "数式を挿入します",
|
"Insert math equation": "数式を挿入します",
|
||||||
"Mermaid diagram": "Mermaidコード",
|
"Mermaid diagram": "Mermaid ダイアグラム",
|
||||||
"Insert mermaid diagram": "Mermaidコードを記述して図を挿入します",
|
"Insert mermaid diagram": "Mermaid ダイアグラムを挿入します",
|
||||||
"Insert and design Drawio diagrams": "Drawioの図を挿入してデザインします",
|
"Insert and design Drawio diagrams": "Draw.io 図を挿入・編集します",
|
||||||
"Insert current date": "今日の日付を挿入します",
|
"Insert current date": "現在の日付を挿入します",
|
||||||
"Draw and sketch excalidraw diagrams": "Excalidrawの図を埋め込みます",
|
"Draw and sketch excalidraw diagrams": "Excalidraw 図を挿入します",
|
||||||
"Multiple": "複数",
|
"Multiple": "複数",
|
||||||
"Heading {{level}}": "見出し {{level}}",
|
"Heading {{level}}": "見出し {{level}}",
|
||||||
"Toggle title": "タイトルの表示/非表示を切り替える",
|
"Toggle title": "タイトルの表示/非表示を切り替える",
|
||||||
@@ -356,29 +363,29 @@
|
|||||||
"Yesterday, {{time}}": "昨日、{{time}}",
|
"Yesterday, {{time}}": "昨日、{{time}}",
|
||||||
"Space created successfully": "スペースを作成しました",
|
"Space created successfully": "スペースを作成しました",
|
||||||
"Space updated successfully": "スペースを更新しました",
|
"Space updated successfully": "スペースを更新しました",
|
||||||
"Space deleted successfully": "スペースが削除されました",
|
"Space deleted successfully": "スペースを削除しました",
|
||||||
"Members added successfully": "メンバーを追加しました",
|
"Members added successfully": "メンバーを追加しました",
|
||||||
"Member removed successfully": "メンバーが削除されました",
|
"Member removed successfully": "メンバーを削除しました",
|
||||||
"Member role updated successfully": "メンバーのロールを更新しました",
|
"Member role updated successfully": "メンバーのロールを更新しました",
|
||||||
"Created by: <b>{{creatorName}}</b>": "作成者: <b>{{creatorName}}</b>",
|
"Created by: <b>{{creatorName}}</b>": "作成者: <b>{{creatorName}}</b>",
|
||||||
"Created at: {{time}}": "が作成しました:{{time}}",
|
"Created at: {{time}}": "作成日: {{time}}",
|
||||||
"Edited by {{name}} {{time}}": "最終編集: {{name}} {{time}}",
|
"Edited by {{name}} {{time}}": "最終編集: {{name}} {{time}}",
|
||||||
"Word count: {{wordCount}}": "ワード数: {{wordCount}}",
|
"Word count: {{wordCount}}": "単語数: {{wordCount}}",
|
||||||
"Character count: {{characterCount}}": "文字数: {{characterCount}}",
|
"Character count: {{characterCount}}": "文字数: {{characterCount}}",
|
||||||
"New update": "新規更新",
|
"New update": "新規更新",
|
||||||
"{{latestVersion}} is available": "{{latestVersion}}は利用可能です",
|
"{{latestVersion}} is available": "{{latestVersion}} が利用可能です",
|
||||||
"Default page edit mode": "デフォルトのページ編集モード",
|
"Default page edit mode": "デフォルトのページ編集モード",
|
||||||
"Choose your preferred page edit mode. Avoid accidental edits.": "希望のページ編集モードを選択してください。誤って編集を防ぎます。",
|
"Choose your preferred page edit mode. Avoid accidental edits.": "お好みのページ編集モードを選択してください(誤編集を防止します)",
|
||||||
"Reading": "読み取り",
|
"Reading": "読み取り",
|
||||||
"Delete member": "メンバーを削除する",
|
"Delete member": "メンバーを削除する",
|
||||||
"Member deleted successfully": "メンバーが削除されました",
|
"Member deleted successfully": "メンバーを削除しました",
|
||||||
"Are you sure you want to delete this workspace member? This action is irreversible.": "ワークスペースメンバーを削除してもよろしいですか?この操作は元に戻せません。",
|
"Are you sure you want to delete this workspace member? This action is irreversible.": "このメンバーを削除してもよろしいですか?この操作は取り消せません",
|
||||||
"Move": "移動",
|
"Move": "移動",
|
||||||
"Move page": "ページを移動",
|
"Move page": "ページを移動",
|
||||||
"Move page to a different space.": "ページを別のスペースに移動します。",
|
"Move page to a different space.": "ページを別のスペースに移動します",
|
||||||
"Real-time editor connection lost. Retrying...": "リアルタイムエディターの接続が失われました。再試行しています…",
|
"Real-time editor connection lost. Retrying...": "リアルタイム編集の接続が切断されました。再接続中...",
|
||||||
"Table of contents": "目次",
|
"Table of contents": "目次",
|
||||||
"Add headings (H1, H2, H3) to generate a table of contents.": "見出し(H1、H2、H3)を追加して目次を生成します。",
|
"Add headings (H1, H2, H3) to generate a table of contents.": "見出し(H1、H2、H3)を追加すると目次が生成されます",
|
||||||
"Share": "共有",
|
"Share": "共有",
|
||||||
"Public sharing": "公開共有",
|
"Public sharing": "公開共有",
|
||||||
"Shared by": "共有者",
|
"Shared by": "共有者",
|
||||||
@@ -397,13 +404,13 @@
|
|||||||
"Delete share": "共有を削除",
|
"Delete share": "共有を削除",
|
||||||
"Are you sure you want to delete this shared link?": "この共有リンクを削除してもよろしいですか?",
|
"Are you sure you want to delete this shared link?": "この共有リンクを削除してもよろしいですか?",
|
||||||
"Publicly shared pages from spaces you are a member of will appear here": "メンバーであるスペースからの公開ページがここに表示されます",
|
"Publicly shared pages from spaces you are a member of will appear here": "メンバーであるスペースからの公開ページがここに表示されます",
|
||||||
"Share deleted successfully": "共有が正常に削除されました",
|
"Share deleted successfully": "共有を削除しました",
|
||||||
"Share not found": "共有が見つかりません",
|
"Share not found": "共有が見つかりません",
|
||||||
"Failed to share page": "ページの共有に失敗しました",
|
"Failed to share page": "ページの共有に失敗しました",
|
||||||
"Copy page": "ページをコピー",
|
"Copy page": "ページをコピー",
|
||||||
"Copy page to a different space.": "ページを別のスペースにコピーします。",
|
"Copy page to a different space.": "ページを別のスペースにコピーします",
|
||||||
"Page copied successfully": "ページのコピーに成功しました",
|
"Page copied successfully": "ページをコピーしました",
|
||||||
"Page duplicated successfully": "ページが正常に複製されました",
|
"Page duplicated successfully": "ページを複製しました",
|
||||||
"Find": "検索",
|
"Find": "検索",
|
||||||
"Not found": "見つかりません",
|
"Not found": "見つかりません",
|
||||||
"Previous Match (Shift+Enter)": "前の一致 (Shift+Enter)",
|
"Previous Match (Shift+Enter)": "前の一致 (Shift+Enter)",
|
||||||
@@ -418,26 +425,26 @@
|
|||||||
"Error": "エラー",
|
"Error": "エラー",
|
||||||
"Failed to disable MFA": "MFAの無効化に失敗しました",
|
"Failed to disable MFA": "MFAの無効化に失敗しました",
|
||||||
"Disable two-factor authentication": "二要素認証を無効化",
|
"Disable two-factor authentication": "二要素認証を無効化",
|
||||||
"Disabling two-factor authentication will make your account less secure. You'll only need your password to sign in.": "二要素認証を無効化すると、アカウントのセキュリティが低下します。サインインにはパスワードのみが必要になります。",
|
"Disabling two-factor authentication will make your account less secure. You'll only need your password to sign in.": "二要素認証を無効にすると、アカウントのセキュリティが低下します。サインインにはパスワードのみが必要になります",
|
||||||
"Please enter your password to disable two-factor authentication:": "二要素認証を無効化するにはパスワードを入力してください:",
|
"Please enter your password to disable two-factor authentication:": "二要素認証を無効にするにはパスワードを入力してください",
|
||||||
"Two-factor authentication has been enabled": "二要素認証が有効になりました",
|
"Two-factor authentication has been enabled": "二要素認証を有効にしました",
|
||||||
"Two-factor authentication has been disabled": "二要素認証が無効になりました",
|
"Two-factor authentication has been disabled": "二要素認証を無効にしました",
|
||||||
"2-step verification": "2段階確認",
|
"2-step verification": "2段階認証",
|
||||||
"Protect your account with an additional verification layer when signing in.": "サインイン時に追加の認証レイヤーでアカウントを保護します。",
|
"Protect your account with an additional verification layer when signing in.": "サインイン時に追加の認証でアカウントを保護します",
|
||||||
"Two-factor authentication is active on your account.": "二要素認証がアカウントで有効です。",
|
"Two-factor authentication is active on your account.": "二要素認証が有効です",
|
||||||
"Add 2FA method": "2FAメソッドを追加",
|
"Add 2FA method": "2FAメソッドを追加",
|
||||||
"Backup codes": "バックアップコード",
|
"Backup codes": "バックアップコード",
|
||||||
"Disable": "無効にする",
|
"Disable": "無効にする",
|
||||||
"Invalid verification code": "無効な認証コード",
|
"Invalid verification code": "無効な認証コード",
|
||||||
"New backup codes have been generated": "新しいバックアップコードが生成されました",
|
"New backup codes have been generated": "新しいバックアップコードを生成しました",
|
||||||
"Failed to regenerate backup codes": "バックアップコードの再生成に失敗しました",
|
"Failed to regenerate backup codes": "バックアップコードの再生成に失敗しました",
|
||||||
"About backup codes": "バックアップコードについて",
|
"About backup codes": "バックアップコードについて",
|
||||||
"Backup codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.": "バックアップコードは、認証アプリへのアクセスを失った場合にアカウントにアクセスするために使用できます。各コードは一度しか使用できません。",
|
"Backup codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.": "認証アプリにアクセスできない場合、バックアップコードでアカウントにアクセスできます。各コードは1回のみ使用可能です",
|
||||||
"You can regenerate new backup codes at any time. This will invalidate all existing codes.": "いつでも新しいバックアップコードを再生成できます。これにより、既存のすべてのコードが無効になります。",
|
"You can regenerate new backup codes at any time. This will invalidate all existing codes.": "新しいバックアップコードはいつでも再生成できます。既存のコードはすべて無効になります",
|
||||||
"Confirm password": "パスワードを確認",
|
"Confirm password": "パスワードを確認",
|
||||||
"Generate new backup codes": "新しいバックアップコードを生成",
|
"Generate new backup codes": "新しいバックアップコードを生成",
|
||||||
"Save your new backup codes": "新しいバックアップコードを保存",
|
"Save your new backup codes": "新しいバックアップコードを保存",
|
||||||
"Make sure to save these codes in a secure place. Your old backup codes are no longer valid.": "これらのコードを安全な場所に保存してください。古いバックアップコードは無効です。",
|
"Make sure to save these codes in a secure place. Your old backup codes are no longer valid.": "これらのコードを安全な場所に保存してください。古いバックアップコードは無効になりました",
|
||||||
"Your new backup codes": "新しいバックアップコード",
|
"Your new backup codes": "新しいバックアップコード",
|
||||||
"I've saved my backup codes": "バックアップコードを保存しました",
|
"I've saved my backup codes": "バックアップコードを保存しました",
|
||||||
"Failed to setup MFA": "MFAの設定に失敗しました",
|
"Failed to setup MFA": "MFAの設定に失敗しました",
|
||||||
@@ -448,52 +455,125 @@
|
|||||||
"Enter this code manually in your authenticator app:": "このコードを認証アプリに手動で入力してください:",
|
"Enter this code manually in your authenticator app:": "このコードを認証アプリに手動で入力してください:",
|
||||||
"2. Enter the 6-digit code from your authenticator": "2. 認証アプリからの6桁のコードを入力してください",
|
"2. Enter the 6-digit code from your authenticator": "2. 認証アプリからの6桁のコードを入力してください",
|
||||||
"Verify and enable": "確認と有効化",
|
"Verify and enable": "確認と有効化",
|
||||||
"Failed to generate QR code. Please try again.": "QRコードの生成に失敗しました。再試行してください。",
|
"Failed to generate QR code. Please try again.": "QRコードの生成に失敗しました。もう一度お試しください",
|
||||||
"Backup": "バックアップ",
|
"Backup": "バックアップ",
|
||||||
"Save codes": "コードを保存",
|
"Save codes": "コードを保存",
|
||||||
"Save your backup codes": "バックアップコードを保存",
|
"Save your backup codes": "バックアップコードを保存",
|
||||||
"These codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.": "これらのコードは、認証アプリへのアクセスを失った場合にアカウントにアクセスするために使用できます。各コードは一度しか使用できません。",
|
"These codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.": "認証アプリにアクセスできない場合、これらのコードでアカウントにアクセスできます。各コードは1回のみ使用可能です",
|
||||||
"Print": "印刷",
|
"Print": "印刷",
|
||||||
"Two-factor authentication has been set up. Please log in again.": "二要素認証が設定されました。再度ログインしてください。",
|
"Two-factor authentication has been set up. Please log in again.": "二要素認証を設定しました。再度ログインしてください",
|
||||||
"Two-Factor authentication required": "二要素認証が必要です",
|
"Two-Factor authentication required": "二要素認証が必要です",
|
||||||
"Your workspace requires two-factor authentication for all users": "ワークスペースでは、すべてのユーザーに二要素認証が必要です",
|
"Your workspace requires two-factor authentication for all users": "このワークスペースではすべてのユーザーに二要素認証が必要です",
|
||||||
"To continue accessing your workspace, you must set up two-factor authentication. This adds an extra layer of security to your account.": "ワークスペースへのアクセスを続けるには、二要素認証を設定する必要があります。これにより、アカウントに追加のセキュリティ層が追加されます。",
|
"To continue accessing your workspace, you must set up two-factor authentication. This adds an extra layer of security to your account.": "ワークスペースにアクセスするには二要素認証を設定してください。アカウントのセキュリティが強化されます",
|
||||||
"Set up two-factor authentication": "二要素認証を設定",
|
"Set up two-factor authentication": "二要素認証を設定",
|
||||||
"Cancel and logout": "キャンセルしてログアウト",
|
"Cancel and logout": "キャンセルしてログアウト",
|
||||||
"Your workspace requires two-factor authentication. Please set it up to continue.": "ワークスペースでは二要素認証が必要です。続行するには設定してください。",
|
"Your workspace requires two-factor authentication. Please set it up to continue.": "このワークスペースでは二要素認証が必要です。続行するには設定してください",
|
||||||
"This adds an extra layer of security to your account by requiring a verification code from your authenticator app.": "これにより、認証アプリからの確認コードが必要となり、アカウントに追加のセキュリティ層が追加されます。",
|
"This adds an extra layer of security to your account by requiring a verification code from your authenticator app.": "認証アプリからの確認コードでアカウントのセキュリティが強化されます",
|
||||||
"Password is required": "パスワードが必要です",
|
"Password is required": "パスワードが必要です",
|
||||||
"Password must be at least 8 characters": "パスワードは8文字以上必要です",
|
"Password must be at least 8 characters": "パスワードは8文字以上必要です",
|
||||||
"Please enter a 6-digit code": "6桁のコードを入力してください",
|
"Please enter a 6-digit code": "6桁のコードを入力してください",
|
||||||
"Code must be exactly 6 digits": "コードは正確に6桁である必要があります",
|
"Code must be exactly 6 digits": "コードは6桁で入力してください",
|
||||||
"Enter the 6-digit code found in your authenticator app": "認証アプリに表示された6桁のコードを入力してください",
|
"Enter the 6-digit code found in your authenticator app": "認証アプリに表示された6桁のコードを入力してください",
|
||||||
"Need help authenticating?": "認証に関するヘルプが必要ですか?",
|
"Need help authenticating?": "認証に関するヘルプが必要ですか?",
|
||||||
"MFA QR Code": "MFA QRコード",
|
"MFA QR Code": "MFA QRコード",
|
||||||
"Account created successfully. Please log in to set up two-factor authentication.": "アカウントが正常に作成されました。二要素認証を設定するためにログインしてください。",
|
"Account created successfully. Please log in to set up two-factor authentication.": "アカウントを作成しました。二要素認証を設定するためにログインしてください",
|
||||||
"Password reset successful. Please log in with your new password and complete two-factor authentication.": "パスワードのリセットが成功しました。新しいパスワードでログインし、二要素認証を完了してください。",
|
"Password reset successful. Please log in with your new password and complete two-factor authentication.": "パスワードをリセットしました。新しいパスワードでログインして二要素認証を完了してください",
|
||||||
"Password reset successful. Please log in with your new password to set up two-factor authentication.": "パスワードのリセットが成功しました。二要素認証を設定するために新しいパスワードでログインしてください。",
|
"Password reset successful. Please log in with your new password to set up two-factor authentication.": "パスワードをリセットしました。新しいパスワードでログインして二要素認証を設定してください",
|
||||||
"Password reset was successful. Please log in with your new password.": "パスワードのリセットが成功しました。新しいパスワードでログインしてください。",
|
"Password reset was successful. Please log in with your new password.": "パスワードをリセットしました。新しいパスワードでログインしてください",
|
||||||
"Two-factor authentication": "二要素認証",
|
"Two-factor authentication": "二要素認証",
|
||||||
"Use authenticator app instead": "代わりに認証アプリを使用",
|
"Use authenticator app instead": "代わりに認証アプリを使用",
|
||||||
"Verify backup code": "バックアップコードを確認",
|
"Verify backup code": "バックアップコードを確認",
|
||||||
"Use backup code": "バックアップコードを使用",
|
"Use backup code": "バックアップコードを使用",
|
||||||
"Enter one of your backup codes": "バックアップコードのいずれかを入力してください",
|
"Enter one of your backup codes": "バックアップコードのいずれかを入力してください",
|
||||||
"Backup code": "バックアップコード",
|
"Backup code": "バックアップコード",
|
||||||
"Enter one of your backup codes. Each backup code can only be used once.": "バックアップコードのいずれかを入力してください。各バックアップコードは一度しか使用できません。",
|
"Enter one of your backup codes. Each backup code can only be used once.": "バックアップコードを入力してください。各コードは1回のみ使用可能です",
|
||||||
"Verify": "確認",
|
"Verify": "確認",
|
||||||
"Trash": "ごみ箱",
|
"Trash": "ごみ箱",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "ごみ箱内のページは30日後に完全に削除されます。",
|
"Pages in trash will be permanently deleted after 30 days.": "ごみ箱内のページは30日後に完全に削除されます",
|
||||||
"Deleted": "削除",
|
"Deleted": "削除",
|
||||||
"No pages in trash": "ごみ箱にページがありません",
|
"No pages in trash": "ごみ箱にページがありません",
|
||||||
"Permanently delete page?": "ページを完全に削除しますか?",
|
"Permanently delete page?": "ページを完全に削除しますか?",
|
||||||
"Are you sure you want to permanently delete '{{title}}'? This action cannot be undone.": "{{title}}』を完全に削除しますか? この操作は元に戻せません。",
|
"Are you sure you want to permanently delete '{{title}}'? This action cannot be undone.": "「{{title}}」を完全に削除しますか?この操作は取り消せません",
|
||||||
"Restore '{{title}}' and its sub-pages?": "{{title}}』とそのサブページを復元しますか?",
|
"Restore '{{title}}' and its sub-pages?": "「{{title}}」とそのサブページを復元しますか?",
|
||||||
"Move to trash": "ごみ箱に移動",
|
"Move to trash": "ごみ箱に移動",
|
||||||
"Move this page to trash?": "このページをごみ箱に移動しますか?",
|
"Move this page to trash?": "このページをごみ箱に移動しますか?",
|
||||||
"Restore page": "ページを復元",
|
"Restore page": "ページを復元",
|
||||||
"Page moved to trash": "ページがごみ箱に移動されました",
|
"Page moved to trash": "ページをごみ箱に移動しました",
|
||||||
"Page restored successfully": "ページが正常に復元されました",
|
"Page restored successfully": "ページを復元しました",
|
||||||
"Deleted by": "削除者",
|
"Deleted by": "削除者",
|
||||||
"Deleted at": "削除日時",
|
"Deleted at": "削除日時",
|
||||||
"Preview": "プレビュー"
|
"Preview": "プレビュー",
|
||||||
|
"Subpages": "サブページ",
|
||||||
|
"Failed to load subpages": "サブページの読み込みに失敗しました",
|
||||||
|
"No subpages": "サブページがありません",
|
||||||
|
"Subpages (Child pages)": "サブページ(子ページ)",
|
||||||
|
"List all subpages of the current page": "現在のページのすべてのサブページをリスト",
|
||||||
|
"Attachments": "添付ファイル",
|
||||||
|
"All spaces": "すべてのスペース",
|
||||||
|
"Unknown": "不明",
|
||||||
|
"Find a space": "スペースを探す",
|
||||||
|
"Search in all your spaces": "あなたのすべてのスペースで検索",
|
||||||
|
"Type": "タイプ",
|
||||||
|
"Enterprise": "エンタープライズ",
|
||||||
|
"Download attachment": "添付ファイルをダウンロード",
|
||||||
|
"Allowed email domains": "許可されたメールドメイン",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "これらのドメインのメールアドレスを持つユーザーのみSSO経由で登録できます",
|
||||||
|
"Enter valid domain names separated by comma or space": "コンマまたはスペースで区切って有効なドメイン名を入力してください",
|
||||||
|
"Enforce two-factor authentication": "二要素認証を強制する",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "有効にすると、すべてのメンバーが二要素認証を設定しないとワークスペースにアクセスできなくなります",
|
||||||
|
"Toggle MFA enforcement": "MFAの強制を切り替える",
|
||||||
|
"Display name": "表示名",
|
||||||
|
"Allow signup": "登録を許可する",
|
||||||
|
"Enabled": "有効",
|
||||||
|
"Advanced Settings": "詳細設定",
|
||||||
|
"Enable TLS/SSL": "TLS/SSLを有効にする",
|
||||||
|
"Use secure connection to LDAP server": "LDAPサーバーへの安全な接続を使用する",
|
||||||
|
"Group sync": "グループ同期",
|
||||||
|
"No SSO providers found.": "SSOプロバイダーが見つかりませんでした。",
|
||||||
|
"Delete SSO provider": "SSOプロバイダーを削除する",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "このSSOプロバイダーを削除してもよろしいですか?",
|
||||||
|
"Action": "アクション",
|
||||||
|
"{{ssoProviderType}} configuration": "{{ssoProviderType}}の構成",
|
||||||
|
"Icon": "アイコン",
|
||||||
|
"Upload image": "画像をアップロード",
|
||||||
|
"Remove image": "画像を削除",
|
||||||
|
"Failed to remove image": "画像の削除に失敗しました",
|
||||||
|
"Image exceeds 10MB limit.": "画像が10MBの制限を超えています",
|
||||||
|
"Image removed successfully": "画像を削除しました",
|
||||||
|
"API key": "APIキー",
|
||||||
|
"API key created successfully": "APIキーを作成しました",
|
||||||
|
"API keys": "APIキー",
|
||||||
|
"API management": "API管理",
|
||||||
|
"Are you sure you want to revoke this API key": "このAPIキーを無効にしてもよろしいですか",
|
||||||
|
"Create API Key": "APIキーを作成",
|
||||||
|
"Custom expiration date": "カスタム有効期限",
|
||||||
|
"Enter a descriptive token name": "説明的なトークン名を入力してください",
|
||||||
|
"Expiration": "有効期限",
|
||||||
|
"Expired": "期限切れ",
|
||||||
|
"Expires": "期限が切れます",
|
||||||
|
"I've saved my API key": "APIキーを保存しました",
|
||||||
|
"Last use": "最終使用",
|
||||||
|
"No API keys found": "APIキーが見つかりません",
|
||||||
|
"No expiration": "期限なし",
|
||||||
|
"Revoke API key": "APIキーを無効にする",
|
||||||
|
"Revoked successfully": "無効にしました",
|
||||||
|
"Select expiration date": "有効期限を選択してください",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "この操作は取り消せません。このAPIキーを使用しているアプリケーションは動作しなくなります",
|
||||||
|
"Update API key": "APIキーを更新",
|
||||||
|
"Manage API keys for all users in the workspace": "ワークスペース内のすべてのユーザーのAPIキーを管理",
|
||||||
|
"AI settings": "AI設定",
|
||||||
|
"AI search": "AI検索",
|
||||||
|
"AI Answer": "AI回答",
|
||||||
|
"Ask AI": "AIに質問する",
|
||||||
|
"AI is thinking...": "AIが考え中...",
|
||||||
|
"Ask a question...": "質問を入力...",
|
||||||
|
"AI-powered search (Ask AI)": "AIによる検索(AIに質問)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "AI検索はベクター埋め込みを使用してワークスペース全体の意味検索を実現します",
|
||||||
|
"Toggle AI search": "AI検索を切り替え",
|
||||||
|
"Sources": "ソース",
|
||||||
|
"Ask AI not available for attachments": "添付ファイルにはAI質問は利用できません",
|
||||||
|
"No answer available": "回答がありません",
|
||||||
|
"Background color": "背景色",
|
||||||
|
"Highlight color": "ハイライト色",
|
||||||
|
"Remove color": "色を削除"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,6 +29,7 @@
|
|||||||
"Choose your preferred interface language.": "선호하는 인터페이스 언어를 선택하세요.",
|
"Choose your preferred interface language.": "선호하는 인터페이스 언어를 선택하세요.",
|
||||||
"Choose your preferred page width.": "선호하는 페이지 너비를 선택하세요.",
|
"Choose your preferred page width.": "선호하는 페이지 너비를 선택하세요.",
|
||||||
"Confirm": "확인",
|
"Confirm": "확인",
|
||||||
|
"Copy as Markdown": "Markdown으로 복사",
|
||||||
"Copy link": "링크 복사",
|
"Copy link": "링크 복사",
|
||||||
"Create": "생성",
|
"Create": "생성",
|
||||||
"Create group": "팀 생성",
|
"Create group": "팀 생성",
|
||||||
@@ -53,6 +54,7 @@
|
|||||||
"e.g Space for product team": "예: 제품 팀을 위한 Space",
|
"e.g Space for product team": "예: 제품 팀을 위한 Space",
|
||||||
"e.g Space for sales team to collaborate": "예: 영업 팀의 Space",
|
"e.g Space for sales team to collaborate": "예: 영업 팀의 Space",
|
||||||
"Edit": "편집",
|
"Edit": "편집",
|
||||||
|
"Read": "읽기",
|
||||||
"Edit group": "팀 편집",
|
"Edit group": "팀 편집",
|
||||||
"Email": "이메일",
|
"Email": "이메일",
|
||||||
"Enter a strong password": "강력한 비밀번호를 입력하세요",
|
"Enter a strong password": "강력한 비밀번호를 입력하세요",
|
||||||
@@ -121,6 +123,8 @@
|
|||||||
"page": "페이지",
|
"page": "페이지",
|
||||||
"Page deleted successfully": "페이지 삭제 완료",
|
"Page deleted successfully": "페이지 삭제 완료",
|
||||||
"Page history": "페이지 기록",
|
"Page history": "페이지 기록",
|
||||||
|
"Select version": "버전 선택",
|
||||||
|
"Highlight changes": "변경 사항 강조",
|
||||||
"Page import is in progress. Please do not close this tab.": "페이지 가져오기가 진행 중입니다. 이 탭을 닫지 마세요.",
|
"Page import is in progress. Please do not close this tab.": "페이지 가져오기가 진행 중입니다. 이 탭을 닫지 마세요.",
|
||||||
"Pages": "페이지",
|
"Pages": "페이지",
|
||||||
"pages": "페이지",
|
"pages": "페이지",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "내보내기 실패:",
|
"Export failed:": "내보내기 실패:",
|
||||||
"export error": "내보내기 오류",
|
"export error": "내보내기 오류",
|
||||||
"Export page": "페이지 내보내기",
|
"Export page": "페이지 내보내기",
|
||||||
|
"Export successful": "내보내기 성공",
|
||||||
"Export space": "Space 내보내기",
|
"Export space": "Space 내보내기",
|
||||||
"Export {{type}}": "{{type}} 내보내기",
|
"Export {{type}}": "{{type}} 내보내기",
|
||||||
"File exceeds the {{limit}} attachment limit": "첨부 파일 크기 제한 {{limit}}을 초과했습니다",
|
"File exceeds the {{limit}} attachment limit": "첨부 파일 크기 제한 {{limit}}을 초과했습니다",
|
||||||
@@ -327,6 +332,8 @@
|
|||||||
"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 file from your device.": "기기에서 파일을 업로드하세요.",
|
"Upload any file from your device.": "기기에서 파일을 업로드하세요.",
|
||||||
|
"Uploading {{name}}": "{{name}} 업로드 중",
|
||||||
|
"Uploading file": "파일 업로드 중",
|
||||||
"Table": "테이블",
|
"Table": "테이블",
|
||||||
"Insert a table.": "테이블 삽입.",
|
"Insert a table.": "테이블 삽입.",
|
||||||
"Insert collapsible block.": "접을 수 있는 블록 삽입.",
|
"Insert collapsible block.": "접을 수 있는 블록 삽입.",
|
||||||
@@ -495,5 +502,78 @@
|
|||||||
"Page restored successfully": "페이지가 성공적으로 복구되었습니다",
|
"Page restored successfully": "페이지가 성공적으로 복구되었습니다",
|
||||||
"Deleted by": "삭제자",
|
"Deleted by": "삭제자",
|
||||||
"Deleted at": "삭제 시간",
|
"Deleted at": "삭제 시간",
|
||||||
"Preview": "미리보기"
|
"Preview": "미리보기",
|
||||||
|
"Subpages": "하위 페이지",
|
||||||
|
"Failed to load subpages": "하위 페이지 로드 실패",
|
||||||
|
"No subpages": "하위 페이지 없음",
|
||||||
|
"Subpages (Child pages)": "하위 페이지 (자식 페이지)",
|
||||||
|
"List all subpages of the current page": "현재 페이지의 모든 하위 페이지 목록",
|
||||||
|
"Attachments": "첨부 파일",
|
||||||
|
"All spaces": "전체 공간",
|
||||||
|
"Unknown": "알 수 없음",
|
||||||
|
"Find a space": "공간 찾기",
|
||||||
|
"Search in all your spaces": "모든 공간에서 검색",
|
||||||
|
"Type": "유형",
|
||||||
|
"Enterprise": "기업",
|
||||||
|
"Download attachment": "첨부 파일 다운로드",
|
||||||
|
"Allowed email domains": "허용된 이메일 도메인",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "이 도메인의 이메일 주소를 가진 사용자만 SSO를 통해 가입할 수 있습니다.",
|
||||||
|
"Enter valid domain names separated by comma or space": "콤마 또는 공백으로 구분하여 유효한 도메인 이름 입력",
|
||||||
|
"Enforce two-factor authentication": "이중 인증 시행",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "시행되면 모든 멤버가 작업 공간에 액세스하기 위해 이중 인증을 활성화해야 합니다.",
|
||||||
|
"Toggle MFA enforcement": "MFA 시행 전환",
|
||||||
|
"Display name": "표시 이름",
|
||||||
|
"Allow signup": "가입 허용",
|
||||||
|
"Enabled": "활성화됨",
|
||||||
|
"Advanced Settings": "고급 설정",
|
||||||
|
"Enable TLS/SSL": "TLS\\/SSL 활성화",
|
||||||
|
"Use secure connection to LDAP server": "LDAP 서버에 안전한 연결 사용",
|
||||||
|
"Group sync": "그룹 동기화",
|
||||||
|
"No SSO providers found.": "SSO 제공자를 찾을 수 없습니다.",
|
||||||
|
"Delete SSO provider": "SSO 제공자 삭제",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "이 SSO 제공자를 삭제하시겠습니까?",
|
||||||
|
"Action": "작업",
|
||||||
|
"{{ssoProviderType}} configuration": "{{ssoProviderType}} 구성",
|
||||||
|
"Icon": "아이콘",
|
||||||
|
"Upload image": "이미지 업로드",
|
||||||
|
"Remove image": "이미지 제거",
|
||||||
|
"Failed to remove image": "이미지 제거 실패",
|
||||||
|
"Image exceeds 10MB limit.": "이미지가 10MB 용량 제한을 초과합니다.",
|
||||||
|
"Image removed successfully": "이미지가 성공적으로 제거되었습니다",
|
||||||
|
"API key": "API 키",
|
||||||
|
"API key created successfully": "API 키 생성 완료",
|
||||||
|
"API keys": "API 키",
|
||||||
|
"API management": "API 관리",
|
||||||
|
"Are you sure you want to revoke this API key": "이 API 키를 취소하시겠습니까?",
|
||||||
|
"Create API Key": "API 키 생성",
|
||||||
|
"Custom expiration date": "사용자 정의 만료일",
|
||||||
|
"Enter a descriptive token name": "토큰 이름을 입력하세요",
|
||||||
|
"Expiration": "만료",
|
||||||
|
"Expired": "만료됨",
|
||||||
|
"Expires": "만료일",
|
||||||
|
"I've saved my API key": "API 키를 저장했습니다",
|
||||||
|
"Last use": "최근 사용",
|
||||||
|
"No API keys found": "API 키를 찾을 수 없습니다",
|
||||||
|
"No expiration": "유효기간 없음",
|
||||||
|
"Revoke API key": "API 키 취소",
|
||||||
|
"Revoked successfully": "성공적으로 취소되었습니다",
|
||||||
|
"Select expiration date": "만료일 선택",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "이 작업은 되돌릴 수 없습니다. 이 API 키를 사용하는 모든 응용 프로그램이 작동을 멈출 것입니다.",
|
||||||
|
"Update API key": "API 키 갱신",
|
||||||
|
"Manage API keys for all users in the workspace": "워크스페이스 내 모든 사용자의 API 키 관리",
|
||||||
|
"AI settings": "AI 설정",
|
||||||
|
"AI search": "AI 검색",
|
||||||
|
"AI Answer": "AI 답변",
|
||||||
|
"Ask AI": "AI에게 묻기",
|
||||||
|
"AI is thinking...": "AI가 생각 중입니다...",
|
||||||
|
"Ask a question...": "질문하세요...",
|
||||||
|
"AI-powered search (Ask AI)": "AI 지원 검색 (AI에게 묻기)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "AI 검색은 벡터 임베딩을 사용하여 작업공간 콘텐츠에 대한 의미 검색 기능을 제공합니다.",
|
||||||
|
"Toggle AI search": "AI 검색 전환",
|
||||||
|
"Sources": "출처",
|
||||||
|
"Ask AI not available for attachments": "AI에게 묻기 기능은 첨부 파일에 대해 사용할 수 없습니다",
|
||||||
|
"No answer available": "답변을 제공할 수 없습니다",
|
||||||
|
"Background color": "배경 색",
|
||||||
|
"Highlight color": "강조 색",
|
||||||
|
"Remove color": "색 제거"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,12 +29,13 @@
|
|||||||
"Choose your preferred interface language.": "Kies uw gewenste interfacetaal.",
|
"Choose your preferred interface language.": "Kies uw gewenste interfacetaal.",
|
||||||
"Choose your preferred page width.": "Kies uw gewenste paginabreedte.",
|
"Choose your preferred page width.": "Kies uw gewenste paginabreedte.",
|
||||||
"Confirm": "Bevestig",
|
"Confirm": "Bevestig",
|
||||||
|
"Copy as Markdown": "Kopiëren als Markdown",
|
||||||
"Copy link": "Link kopiëren",
|
"Copy link": "Link kopiëren",
|
||||||
"Create": "Aanmaken",
|
"Create": "Aanmaken",
|
||||||
"Create group": "Groep aanmaken",
|
"Create group": "Groep aanmaken",
|
||||||
"Create page": "Pagina aanmaken",
|
"Create page": "Pagina aanmaken",
|
||||||
"Create space": "Ruimte aanmaken",
|
"Create space": "Ruimte aanmaken",
|
||||||
"Create workspace": "Wwerkruimte aanmaken",
|
"Create workspace": "Werkruimte aanmaken",
|
||||||
"Current password": "Huidig wachtwoord",
|
"Current password": "Huidig wachtwoord",
|
||||||
"Dark": "Donker",
|
"Dark": "Donker",
|
||||||
"Date": "Datum",
|
"Date": "Datum",
|
||||||
@@ -53,6 +54,7 @@
|
|||||||
"e.g Space for product team": "bijv. Ruimte voor productteam",
|
"e.g Space for product team": "bijv. Ruimte voor productteam",
|
||||||
"e.g Space for sales team to collaborate": "bijv. Ruimte voor verkoopteam om samen te werken",
|
"e.g Space for sales team to collaborate": "bijv. Ruimte voor verkoopteam om samen te werken",
|
||||||
"Edit": "Bewerken",
|
"Edit": "Bewerken",
|
||||||
|
"Read": "Lezen",
|
||||||
"Edit group": "Groep bewerken",
|
"Edit group": "Groep bewerken",
|
||||||
"Email": "E-mailadres",
|
"Email": "E-mailadres",
|
||||||
"Enter a strong password": "Voer een sterk wachtwoord in",
|
"Enter a strong password": "Voer een sterk wachtwoord in",
|
||||||
@@ -90,7 +92,7 @@
|
|||||||
"Invite by email": "Uitnodigen via e-mail",
|
"Invite by email": "Uitnodigen via e-mail",
|
||||||
"Invite members": "Leden uitnodigen",
|
"Invite members": "Leden uitnodigen",
|
||||||
"Invite new members": "Nieuwe leden uitnodigen",
|
"Invite new members": "Nieuwe leden uitnodigen",
|
||||||
"Invited members who are yet to accept their invitation will appear here.": "Uigenodigde leden die hun uitnodiging nog moeten accepteren zullen hier worden getoond.",
|
"Invited members who are yet to accept their invitation will appear here.": "Uitgenodigde leden die hun uitnodiging nog moeten accepteren zullen hier worden getoond.",
|
||||||
"Invited members will be granted access to spaces the groups can access": "Uitgenodigde leden wordt toegang gegeven tot ruimtes de groepen toegang toe heeft",
|
"Invited members will be granted access to spaces the groups can access": "Uitgenodigde leden wordt toegang gegeven tot ruimtes de groepen toegang toe heeft",
|
||||||
"Join the workspace": "Word lid van de werkruimte",
|
"Join the workspace": "Word lid van de werkruimte",
|
||||||
"Language": "Taal",
|
"Language": "Taal",
|
||||||
@@ -121,6 +123,8 @@
|
|||||||
"page": "pagina",
|
"page": "pagina",
|
||||||
"Page deleted successfully": "Pagina succesvol verwijderd",
|
"Page deleted successfully": "Pagina succesvol verwijderd",
|
||||||
"Page history": "Pagina geschiedenis",
|
"Page history": "Pagina geschiedenis",
|
||||||
|
"Select version": "Selecteer versie",
|
||||||
|
"Highlight changes": "Wijzigingen markeren",
|
||||||
"Page import is in progress. Please do not close this tab.": "Importeren van pagina's is bezig. Sluit dit tabblad niet.",
|
"Page import is in progress. Please do not close this tab.": "Importeren van pagina's is bezig. Sluit dit tabblad niet.",
|
||||||
"Pages": "Pagina's",
|
"Pages": "Pagina's",
|
||||||
"pages": "pagina's",
|
"pages": "pagina's",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "Exporteren mislukt:",
|
"Export failed:": "Exporteren mislukt:",
|
||||||
"export error": "Exporteer fout",
|
"export error": "Exporteer fout",
|
||||||
"Export page": "Exporteer pagina",
|
"Export page": "Exporteer pagina",
|
||||||
|
"Export successful": "Export succesvol",
|
||||||
"Export space": "Exporteer ruimte",
|
"Export space": "Exporteer ruimte",
|
||||||
"Export {{type}}": "Exporteer {{type}}",
|
"Export {{type}}": "Exporteer {{type}}",
|
||||||
"File exceeds the {{limit}} attachment limit": "Bestand overschrijdt de bijlagelimiet van {{limit}}",
|
"File exceeds the {{limit}} attachment limit": "Bestand overschrijdt de bijlagelimiet van {{limit}}",
|
||||||
@@ -327,6 +332,8 @@
|
|||||||
"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 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 file": "Bestand uploaden",
|
||||||
"Table": "Tabel",
|
"Table": "Tabel",
|
||||||
"Insert a table.": "Voeg een tabel in.",
|
"Insert a table.": "Voeg een tabel in.",
|
||||||
"Insert collapsible block.": "Inklapbaar blok invoegen.",
|
"Insert collapsible block.": "Inklapbaar blok invoegen.",
|
||||||
@@ -495,5 +502,78 @@
|
|||||||
"Page restored successfully": "Pagina succesvol hersteld",
|
"Page restored successfully": "Pagina succesvol hersteld",
|
||||||
"Deleted by": "Verwijderd door",
|
"Deleted by": "Verwijderd door",
|
||||||
"Deleted at": "Verwijderd op",
|
"Deleted at": "Verwijderd op",
|
||||||
"Preview": "Voorbeeld"
|
"Preview": "Voorbeeld",
|
||||||
|
"Subpages": "Subpagina's",
|
||||||
|
"Failed to load subpages": "Laden van subpagina's mislukt",
|
||||||
|
"No subpages": "Geen subpagina's",
|
||||||
|
"Subpages (Child pages)": "Subpagina's (Kindpagina's)",
|
||||||
|
"List all subpages of the current page": "Lijst van alle subpagina's van de huidige pagina",
|
||||||
|
"Attachments": "Bijlagen",
|
||||||
|
"All spaces": "Alle ruimtes",
|
||||||
|
"Unknown": "Onbekend",
|
||||||
|
"Find a space": "Vind een ruimte",
|
||||||
|
"Search in all your spaces": "Zoek in al je ruimtes",
|
||||||
|
"Type": "Type",
|
||||||
|
"Enterprise": "Onderneming",
|
||||||
|
"Download attachment": "Bijlage downloaden",
|
||||||
|
"Allowed email domains": "Toegestane e-maildomeinen",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "Alleen gebruikers met e-mailadressen van deze domeinen kunnen zich aanmelden via SSO.",
|
||||||
|
"Enter valid domain names separated by comma or space": "Voer geldige domeinnamen in, gescheiden door komma of spatie",
|
||||||
|
"Enforce two-factor authentication": "Handhaaf tweefactorauthenticatie",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "Na handhaving moeten alle leden tweefactorauthenticatie inschakelen om toegang te krijgen tot de werkomgeving.",
|
||||||
|
"Toggle MFA enforcement": "Schakel MFA-handhaving in of uit",
|
||||||
|
"Display name": "Weergavenaam",
|
||||||
|
"Allow signup": "Aanmelden toestaan",
|
||||||
|
"Enabled": "Ingeschakeld",
|
||||||
|
"Advanced Settings": "Geavanceerde instellingen",
|
||||||
|
"Enable TLS/SSL": "TLS/SSL inschakelen",
|
||||||
|
"Use secure connection to LDAP server": "Gebruik een beveiligde verbinding met de LDAP-server",
|
||||||
|
"Group sync": "Groepssynchronisatie",
|
||||||
|
"No SSO providers found.": "Geen SSO-providers gevonden.",
|
||||||
|
"Delete SSO provider": "Verwijder SSO-provider",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "Weet u zeker dat u deze SSO-provider wilt verwijderen?",
|
||||||
|
"Action": "Actie",
|
||||||
|
"{{ssoProviderType}} configuration": "{{ssoProviderType}} configuratie",
|
||||||
|
"Icon": "Icoon",
|
||||||
|
"Upload image": "Afbeelding uploaden",
|
||||||
|
"Remove image": "Afbeelding verwijderen",
|
||||||
|
"Failed to remove image": "Afbeelding verwijderen mislukt",
|
||||||
|
"Image exceeds 10MB limit.": "Afbeelding overschrijdt de limiet van 10MB.",
|
||||||
|
"Image removed successfully": "Afbeelding succesvol verwijderd",
|
||||||
|
"API key": "API-sleutel",
|
||||||
|
"API key created successfully": "API-sleutel succesvol aangemaakt",
|
||||||
|
"API keys": "API-sleutels",
|
||||||
|
"API management": "API-beheer",
|
||||||
|
"Are you sure you want to revoke this API key": "Weet u zeker dat u deze API-sleutel wilt intrekken",
|
||||||
|
"Create API Key": "API-sleutel aanmaken",
|
||||||
|
"Custom expiration date": "Aangepaste vervaldatum",
|
||||||
|
"Enter a descriptive token name": "Voer een beschrijvende tokennaam in",
|
||||||
|
"Expiration": "Vervaldatum",
|
||||||
|
"Expired": "Verlopen",
|
||||||
|
"Expires": "Verloopt",
|
||||||
|
"I've saved my API key": "Ik heb mijn API-sleutel opgeslagen",
|
||||||
|
"Last use": "Laatst gebruikt",
|
||||||
|
"No API keys found": "Geen API-sleutels gevonden",
|
||||||
|
"No expiration": "Geen vervaldatum",
|
||||||
|
"Revoke API key": "API-sleutel intrekken",
|
||||||
|
"Revoked successfully": "Succesvol ingetrokken",
|
||||||
|
"Select expiration date": "Selecteer vervaldatum",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "Deze actie kan niet ongedaan worden gemaakt. Alle toepassingen die deze API-sleutel gebruiken, zullen niet meer werken.",
|
||||||
|
"Update API key": "API-sleutel bijwerken",
|
||||||
|
"Manage API keys for all users in the workspace": "Beheer API-sleutels voor alle gebruikers in de werkruimte",
|
||||||
|
"AI settings": "AI-instellingen",
|
||||||
|
"AI search": "AI-zoekopdracht",
|
||||||
|
"AI Answer": "AI Antwoord",
|
||||||
|
"Ask AI": "Vraag AI",
|
||||||
|
"AI is thinking...": "AI is aan het nadenken...",
|
||||||
|
"Ask a question...": "Stel een vraag...",
|
||||||
|
"AI-powered search (Ask AI)": "AI-ondersteunde zoekopdracht (Vraag AI)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "AI-zoekopdracht maakt gebruik van vectorembeddings om semantische zoekmogelijkheden te bieden in uw werkruimte-inhoud.",
|
||||||
|
"Toggle AI search": "Schakel AI-zoekopdracht in/uit",
|
||||||
|
"Sources": "Bronnen",
|
||||||
|
"Ask AI not available for attachments": "Vraag AI is niet beschikbaar voor bijlages",
|
||||||
|
"No answer available": "Geen antwoord beschikbaar",
|
||||||
|
"Background color": "Achtergrondkleur",
|
||||||
|
"Highlight color": "Markeerkleur",
|
||||||
|
"Remove color": "Kleur verwijderen"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,6 +29,7 @@
|
|||||||
"Choose your preferred interface language.": "Escolha o idioma da interface.",
|
"Choose your preferred interface language.": "Escolha o idioma da interface.",
|
||||||
"Choose your preferred page width.": "Escolha a largura preferida da página.",
|
"Choose your preferred page width.": "Escolha a largura preferida da página.",
|
||||||
"Confirm": "Confirmar",
|
"Confirm": "Confirmar",
|
||||||
|
"Copy as Markdown": "Copiar como Markdown",
|
||||||
"Copy link": "Copiar link",
|
"Copy link": "Copiar link",
|
||||||
"Create": "Criar",
|
"Create": "Criar",
|
||||||
"Create group": "Criar grupo",
|
"Create group": "Criar grupo",
|
||||||
@@ -53,6 +54,7 @@
|
|||||||
"e.g Space for product team": "ex.: Espaço para a equipe de produto",
|
"e.g Space for product team": "ex.: Espaço para a equipe de produto",
|
||||||
"e.g Space for sales team to collaborate": "ex.: Espaço para a equipe de vendas colaborar",
|
"e.g Space for sales team to collaborate": "ex.: Espaço para a equipe de vendas colaborar",
|
||||||
"Edit": "Editar",
|
"Edit": "Editar",
|
||||||
|
"Read": "Ler",
|
||||||
"Edit group": "Editar grupo",
|
"Edit group": "Editar grupo",
|
||||||
"Email": "Email",
|
"Email": "Email",
|
||||||
"Enter a strong password": "Insira uma senha forte",
|
"Enter a strong password": "Insira uma senha forte",
|
||||||
@@ -121,6 +123,8 @@
|
|||||||
"page": "página",
|
"page": "página",
|
||||||
"Page deleted successfully": "Página excluída com sucesso",
|
"Page deleted successfully": "Página excluída com sucesso",
|
||||||
"Page history": "Histórico da página",
|
"Page history": "Histórico da página",
|
||||||
|
"Select version": "Selecionar versão",
|
||||||
|
"Highlight changes": "Destacar alterações",
|
||||||
"Page import is in progress. Please do not close this tab.": "A importação da página está em andamento. Por favor, não feche esta aba.",
|
"Page import is in progress. Please do not close this tab.": "A importação da página está em andamento. Por favor, não feche esta aba.",
|
||||||
"Pages": "Páginas",
|
"Pages": "Páginas",
|
||||||
"pages": "páginas",
|
"pages": "páginas",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "Falha ao exportar:",
|
"Export failed:": "Falha ao exportar:",
|
||||||
"export error": "erro de exportação",
|
"export error": "erro de exportação",
|
||||||
"Export page": "Exportar página",
|
"Export page": "Exportar página",
|
||||||
|
"Export successful": "Exportação bem-sucedida",
|
||||||
"Export space": "Exportar espaço",
|
"Export space": "Exportar espaço",
|
||||||
"Export {{type}}": "Exportar para {{type}}",
|
"Export {{type}}": "Exportar para {{type}}",
|
||||||
"File exceeds the {{limit}} attachment limit": "O arquivo excede o limite de anexos {{limit}}",
|
"File exceeds the {{limit}} attachment limit": "O arquivo excede o limite de anexos {{limit}}",
|
||||||
@@ -327,6 +332,8 @@
|
|||||||
"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 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 file": "Enviando arquivo",
|
||||||
"Table": "Tabela",
|
"Table": "Tabela",
|
||||||
"Insert a table.": "Insira uma tabela.",
|
"Insert a table.": "Insira uma tabela.",
|
||||||
"Insert collapsible block.": "Insira um bloco colapsável.",
|
"Insert collapsible block.": "Insira um bloco colapsável.",
|
||||||
@@ -495,5 +502,78 @@
|
|||||||
"Page restored successfully": "Página restaurada com sucesso",
|
"Page restored successfully": "Página restaurada com sucesso",
|
||||||
"Deleted by": "Excluído por",
|
"Deleted by": "Excluído por",
|
||||||
"Deleted at": "Excluído em",
|
"Deleted at": "Excluído em",
|
||||||
"Preview": "Visualização"
|
"Preview": "Visualização",
|
||||||
|
"Subpages": "Subpáginas",
|
||||||
|
"Failed to load subpages": "Falha ao carregar subpáginas",
|
||||||
|
"No subpages": "Sem subpáginas",
|
||||||
|
"Subpages (Child pages)": "Subpáginas (Páginas filhas)",
|
||||||
|
"List all subpages of the current page": "Listar todas as subpáginas da página atual",
|
||||||
|
"Attachments": "Anexos",
|
||||||
|
"All spaces": "Todos os espaços",
|
||||||
|
"Unknown": "Desconhecido",
|
||||||
|
"Find a space": "Encontrar um espaço",
|
||||||
|
"Search in all your spaces": "Pesquisar em todos os seus espaços",
|
||||||
|
"Type": "Tipo",
|
||||||
|
"Enterprise": "Empresa",
|
||||||
|
"Download attachment": "Baixar anexo",
|
||||||
|
"Allowed email domains": "Domínios de email permitidos",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "Apenas usuários com endereços de email desses domínios podem se inscrever via SSO.",
|
||||||
|
"Enter valid domain names separated by comma or space": "Insira nomes de domínio válidos separados por vírgula ou espaço",
|
||||||
|
"Enforce two-factor authentication": "Impor autenticação de dois fatores",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "Uma vez imposto, todos os membros devem habilitar a autenticação de dois fatores para acessar o espaço de trabalho.",
|
||||||
|
"Toggle MFA enforcement": "Alternar imposição de MFA",
|
||||||
|
"Display name": "Nome de exibição",
|
||||||
|
"Allow signup": "Permitir inscrição",
|
||||||
|
"Enabled": "Habilitado",
|
||||||
|
"Advanced Settings": "Configurações Avançadas",
|
||||||
|
"Enable TLS/SSL": "Habilitar TLS/SSL",
|
||||||
|
"Use secure connection to LDAP server": "Usar conexão segura com o servidor LDAP",
|
||||||
|
"Group sync": "Sincronização de grupo",
|
||||||
|
"No SSO providers found.": "Nenhum provedor de SSO encontrado.",
|
||||||
|
"Delete SSO provider": "Excluir provedor de SSO",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "Tem certeza de que deseja excluir este provedor de SSO?",
|
||||||
|
"Action": "Ação",
|
||||||
|
"{{ssoProviderType}} configuration": "Configuração de {{ssoProviderType}}",
|
||||||
|
"Icon": "Ícone",
|
||||||
|
"Upload image": "Fazer upload da imagem",
|
||||||
|
"Remove image": "Remover imagem",
|
||||||
|
"Failed to remove image": "Falha ao remover imagem",
|
||||||
|
"Image exceeds 10MB limit.": "A imagem excede o limite de 10MB.",
|
||||||
|
"Image removed successfully": "Imagem removida com sucesso",
|
||||||
|
"API key": "Chave API",
|
||||||
|
"API key created successfully": "Chave API criada com sucesso",
|
||||||
|
"API keys": "Chaves API",
|
||||||
|
"API management": "Gestão de API",
|
||||||
|
"Are you sure you want to revoke this API key": "Tem certeza de que deseja revogar esta chave API",
|
||||||
|
"Create API Key": "Criar Chave API",
|
||||||
|
"Custom expiration date": "Data de expiração personalizada",
|
||||||
|
"Enter a descriptive token name": "Insira um nome descritivo para o token",
|
||||||
|
"Expiration": "Expiração",
|
||||||
|
"Expired": "Expirado",
|
||||||
|
"Expires": "Expira",
|
||||||
|
"I've saved my API key": "Salvei minha chave API",
|
||||||
|
"Last use": "Último uso",
|
||||||
|
"No API keys found": "Nenhuma chave API encontrada",
|
||||||
|
"No expiration": "Sem expiração",
|
||||||
|
"Revoke API key": "Revogar chave API",
|
||||||
|
"Revoked successfully": "Revogada com sucesso",
|
||||||
|
"Select expiration date": "Selecionar data de expiração",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "Esta ação não pode ser desfeita. Qualquer aplicação usando esta chave API deixará de funcionar.",
|
||||||
|
"Update API key": "Atualizar chave API",
|
||||||
|
"Manage API keys for all users in the workspace": "Gerenciar chaves API para todos os usuários no espaço de trabalho",
|
||||||
|
"AI settings": "Configurações de IA",
|
||||||
|
"AI search": "Pesquisa IA",
|
||||||
|
"AI Answer": "Resposta de IA",
|
||||||
|
"Ask AI": "Pergunte à IA",
|
||||||
|
"AI is thinking...": "IA está pensando...",
|
||||||
|
"Ask a question...": "Faça uma pergunta...",
|
||||||
|
"AI-powered search (Ask AI)": "Pesquisa com IA (Pergunte à IA)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "A pesquisa IA usa vetores de incorporação para fornecer capacidades de pesquisa semântica em todo o conteúdo do seu espaço de trabalho.",
|
||||||
|
"Toggle AI search": "Alternar pesquisa de IA",
|
||||||
|
"Sources": "Fontes",
|
||||||
|
"Ask AI not available for attachments": "Perguntar à IA não está disponível para anexos",
|
||||||
|
"No answer available": "Nenhuma resposta disponível",
|
||||||
|
"Background color": "Cor de fundo",
|
||||||
|
"Highlight color": "Cor de destaque",
|
||||||
|
"Remove color": "Remover cor"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,6 +29,7 @@
|
|||||||
"Choose your preferred interface language.": "Выберите предпочитаемый язык интерфейса.",
|
"Choose your preferred interface language.": "Выберите предпочитаемый язык интерфейса.",
|
||||||
"Choose your preferred page width.": "Выберите предпочитаемую ширину страницы.",
|
"Choose your preferred page width.": "Выберите предпочитаемую ширину страницы.",
|
||||||
"Confirm": "Подтвердить",
|
"Confirm": "Подтвердить",
|
||||||
|
"Copy as Markdown": "Копировать как Markdown",
|
||||||
"Copy link": "Копировать ссылку",
|
"Copy link": "Копировать ссылку",
|
||||||
"Create": "Создать",
|
"Create": "Создать",
|
||||||
"Create group": "Создать группу",
|
"Create group": "Создать группу",
|
||||||
@@ -53,6 +54,7 @@
|
|||||||
"e.g Space for product team": "например, Пространство для продуктовой команды",
|
"e.g Space for product team": "например, Пространство для продуктовой команды",
|
||||||
"e.g Space for sales team to collaborate": "например, Пространство для совместной работы команды продаж",
|
"e.g Space for sales team to collaborate": "например, Пространство для совместной работы команды продаж",
|
||||||
"Edit": "Редактировать",
|
"Edit": "Редактировать",
|
||||||
|
"Read": "Читать",
|
||||||
"Edit group": "Редактировать группу",
|
"Edit group": "Редактировать группу",
|
||||||
"Email": "Электронная почта",
|
"Email": "Электронная почта",
|
||||||
"Enter a strong password": "Введите надёжный пароль",
|
"Enter a strong password": "Введите надёжный пароль",
|
||||||
@@ -121,6 +123,8 @@
|
|||||||
"page": "страница",
|
"page": "страница",
|
||||||
"Page deleted successfully": "Страница успешно удалена",
|
"Page deleted successfully": "Страница успешно удалена",
|
||||||
"Page history": "История страницы",
|
"Page history": "История страницы",
|
||||||
|
"Select version": "Выбрать версию",
|
||||||
|
"Highlight changes": "Выделить изменения",
|
||||||
"Page import is in progress. Please do not close this tab.": "Импорт страницы в процессе. Пожалуйста, не закрывайте эту вкладку.",
|
"Page import is in progress. Please do not close this tab.": "Импорт страницы в процессе. Пожалуйста, не закрывайте эту вкладку.",
|
||||||
"Pages": "Страницы",
|
"Pages": "Страницы",
|
||||||
"pages": "страницы",
|
"pages": "страницы",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "Экспортирование не удалось:",
|
"Export failed:": "Экспортирование не удалось:",
|
||||||
"export error": "ошибка экспорта",
|
"export error": "ошибка экспорта",
|
||||||
"Export page": "Экспорт страницы",
|
"Export page": "Экспорт страницы",
|
||||||
|
"Export successful": "Экспорт выполнен успешно",
|
||||||
"Export space": "Экспорт пространства",
|
"Export space": "Экспорт пространства",
|
||||||
"Export {{type}}": "Экспорт {{type}}",
|
"Export {{type}}": "Экспорт {{type}}",
|
||||||
"File exceeds the {{limit}} attachment limit": "Файл превышает лимит вложений {{limit}}",
|
"File exceeds the {{limit}} attachment limit": "Файл превышает лимит вложений {{limit}}",
|
||||||
@@ -327,6 +332,8 @@
|
|||||||
"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 file from your device.": "Загрузить любой файл с вашего устройства.",
|
"Upload any file from your device.": "Загрузить любой файл с вашего устройства.",
|
||||||
|
"Uploading {{name}}": "Загрузка {{name}}",
|
||||||
|
"Uploading file": "Загрузка файла",
|
||||||
"Table": "Таблица",
|
"Table": "Таблица",
|
||||||
"Insert a table.": "Вставить таблицу.",
|
"Insert a table.": "Вставить таблицу.",
|
||||||
"Insert collapsible block.": "Вставить сворачиваемый блок.",
|
"Insert collapsible block.": "Вставить сворачиваемый блок.",
|
||||||
@@ -495,5 +502,78 @@
|
|||||||
"Page restored successfully": "Страница успешно восстановлена",
|
"Page restored successfully": "Страница успешно восстановлена",
|
||||||
"Deleted by": "Удалено пользователем",
|
"Deleted by": "Удалено пользователем",
|
||||||
"Deleted at": "Удалено в",
|
"Deleted at": "Удалено в",
|
||||||
"Preview": "Предпросмотр"
|
"Preview": "Предпросмотр",
|
||||||
|
"Subpages": "Подстраницы",
|
||||||
|
"Failed to load subpages": "Не удалось загрузить под страницы",
|
||||||
|
"No subpages": "Нет подстраниц",
|
||||||
|
"Subpages (Child pages)": "Подстраницы (вложенные страницы)",
|
||||||
|
"List all subpages of the current page": "Показать все под страницы",
|
||||||
|
"Attachments": "Вложения",
|
||||||
|
"All spaces": "Все пространства",
|
||||||
|
"Unknown": "Неизвестно",
|
||||||
|
"Find a space": "Найти пространство",
|
||||||
|
"Search in all your spaces": "Поиск во всех ваших пространствах",
|
||||||
|
"Type": "Тип",
|
||||||
|
"Enterprise": "Предприятие",
|
||||||
|
"Download attachment": "Скачать вложение",
|
||||||
|
"Allowed email domains": "Разрешенные домены электронной почты",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "Только пользователи с электронными адресами из этих доменов могут зарегистрироваться через SSO.",
|
||||||
|
"Enter valid domain names separated by comma or space": "Введите допустимые доменные имена, разделённые запятыми или пробелами",
|
||||||
|
"Enforce two-factor authentication": "Обязательная двухфакторная аутентификация",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "После введения обязательности все участники должны будут включить двухфакторную аутентификацию для доступа к рабочему пространству.",
|
||||||
|
"Toggle MFA enforcement": "Переключить обязательность MFA",
|
||||||
|
"Display name": "Отображаемое имя",
|
||||||
|
"Allow signup": "Разрешить регистрацию",
|
||||||
|
"Enabled": "Включено",
|
||||||
|
"Advanced Settings": "Расширенные настройки",
|
||||||
|
"Enable TLS/SSL": "Включить TLS/SSL",
|
||||||
|
"Use secure connection to LDAP server": "Использовать защищённое соединение с сервером LDAP",
|
||||||
|
"Group sync": "Синхронизация группы",
|
||||||
|
"No SSO providers found.": "Поставщики SSO не найдены.",
|
||||||
|
"Delete SSO provider": "Удалить поставщика SSO",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "Вы уверены, что хотите удалить этого поставщика SSO?",
|
||||||
|
"Action": "Действие",
|
||||||
|
"{{ssoProviderType}} configuration": "Настройка {{ssoProviderType}}",
|
||||||
|
"Icon": "Иконка",
|
||||||
|
"Upload image": "Загрузить изображение",
|
||||||
|
"Remove image": "Удалить изображение",
|
||||||
|
"Failed to remove image": "Не удалось удалить изображение",
|
||||||
|
"Image exceeds 10MB limit.": "Изображение превышает предел 10MB.",
|
||||||
|
"Image removed successfully": "Изображение успешно удалено",
|
||||||
|
"API key": "API ключ",
|
||||||
|
"API key created successfully": "API ключ успешно создан",
|
||||||
|
"API keys": "API ключи",
|
||||||
|
"API management": "Управление API",
|
||||||
|
"Are you sure you want to revoke this API key": "Вы уверены, что хотите отозвать этот API ключ",
|
||||||
|
"Create API Key": "Создать API ключ",
|
||||||
|
"Custom expiration date": "Пользовательская дата срока действия",
|
||||||
|
"Enter a descriptive token name": "Введите понятное имя токена",
|
||||||
|
"Expiration": "Срок действия",
|
||||||
|
"Expired": "Истек",
|
||||||
|
"Expires": "Истекает",
|
||||||
|
"I've saved my API key": "Я сохранил мой API ключ",
|
||||||
|
"Last use": "Последнее использование",
|
||||||
|
"No API keys found": "API ключи не найдены",
|
||||||
|
"No expiration": "Не истекает",
|
||||||
|
"Revoke API key": "Отозвать API ключ",
|
||||||
|
"Revoked successfully": "Отозван успешно",
|
||||||
|
"Select expiration date": "Выберете срок действия",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "Это действие необратимо. Любые приложения, использующие этот API ключ, перестанут работать.",
|
||||||
|
"Update API key": "Обновить API ключ",
|
||||||
|
"Manage API keys for all users in the workspace": "Управлять API ключами для всех пользователей в рабочей области",
|
||||||
|
"AI settings": "Настройки ИИ",
|
||||||
|
"AI search": "Поиск ИИ",
|
||||||
|
"AI Answer": "Ответ ИИ",
|
||||||
|
"Ask AI": "Спросить ИИ",
|
||||||
|
"AI is thinking...": "ИИ обрабатывает запрос...",
|
||||||
|
"Ask a question...": "Задайте вопрос...",
|
||||||
|
"AI-powered search (Ask AI)": "Поиск на базе ИИ (Спросить ИИ)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "Поиск ИИ использует векторные встраивания для обеспечения семантического поиска по содержимому вашего рабочего пространства.",
|
||||||
|
"Toggle AI search": "Переключить поиск ИИ",
|
||||||
|
"Sources": "Источники",
|
||||||
|
"Ask AI not available for attachments": "Функция \"Спросить ИИ\" недоступна для вложений",
|
||||||
|
"No answer available": "Ответ недоступен",
|
||||||
|
"Background color": "Цвет фона",
|
||||||
|
"Highlight color": "Цвет выделения",
|
||||||
|
"Remove color": "Удалить цвет"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,6 +29,7 @@
|
|||||||
"Choose your preferred interface language.": "Оберіть бажану мову інтерфейсу.",
|
"Choose your preferred interface language.": "Оберіть бажану мову інтерфейсу.",
|
||||||
"Choose your preferred page width.": "Оберіть бажану ширину сторінки.",
|
"Choose your preferred page width.": "Оберіть бажану ширину сторінки.",
|
||||||
"Confirm": "Підтвердити",
|
"Confirm": "Підтвердити",
|
||||||
|
"Copy as Markdown": "Скопіювати як Markdown",
|
||||||
"Copy link": "Копіювати посилання",
|
"Copy link": "Копіювати посилання",
|
||||||
"Create": "Створити",
|
"Create": "Створити",
|
||||||
"Create group": "Створити групу",
|
"Create group": "Створити групу",
|
||||||
@@ -53,6 +54,7 @@
|
|||||||
"e.g Space for product team": "наприклад, Простір для продуктової команди",
|
"e.g Space for product team": "наприклад, Простір для продуктової команди",
|
||||||
"e.g Space for sales team to collaborate": "наприклад, Простір для спільної роботи команди продажів",
|
"e.g Space for sales team to collaborate": "наприклад, Простір для спільної роботи команди продажів",
|
||||||
"Edit": "Редагувати",
|
"Edit": "Редагувати",
|
||||||
|
"Read": "Читати",
|
||||||
"Edit group": "Редагувати групу",
|
"Edit group": "Редагувати групу",
|
||||||
"Email": "Електронна пошта",
|
"Email": "Електронна пошта",
|
||||||
"Enter a strong password": "Введіть надійний пароль",
|
"Enter a strong password": "Введіть надійний пароль",
|
||||||
@@ -121,6 +123,8 @@
|
|||||||
"page": "сторінка",
|
"page": "сторінка",
|
||||||
"Page deleted successfully": "Сторінку успішно видалено",
|
"Page deleted successfully": "Сторінку успішно видалено",
|
||||||
"Page history": "Історія сторінки",
|
"Page history": "Історія сторінки",
|
||||||
|
"Select version": "Вибрати версію",
|
||||||
|
"Highlight changes": "Підсвітити зміни",
|
||||||
"Page import is in progress. Please do not close this tab.": "Імпорт сторінки в процесі. Будь ласка, не закривайте цю вкладку.",
|
"Page import is in progress. Please do not close this tab.": "Імпорт сторінки в процесі. Будь ласка, не закривайте цю вкладку.",
|
||||||
"Pages": "Сторінки",
|
"Pages": "Сторінки",
|
||||||
"pages": "сторінки",
|
"pages": "сторінки",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "Експортування не вдалося:",
|
"Export failed:": "Експортування не вдалося:",
|
||||||
"export error": "помилка експорту",
|
"export error": "помилка експорту",
|
||||||
"Export page": "Експорт сторінки",
|
"Export page": "Експорт сторінки",
|
||||||
|
"Export successful": "Експорт виконано успішно",
|
||||||
"Export space": "Експорт простору",
|
"Export space": "Експорт простору",
|
||||||
"Export {{type}}": "Експорт {{type}}",
|
"Export {{type}}": "Експорт {{type}}",
|
||||||
"File exceeds the {{limit}} attachment limit": "Файл перевищує ліміт вкладень {{limit}}",
|
"File exceeds the {{limit}} attachment limit": "Файл перевищує ліміт вкладень {{limit}}",
|
||||||
@@ -327,6 +332,8 @@
|
|||||||
"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 file from your device.": "Завантажити будь-який файл з вашого пристрою.",
|
"Upload any file from your device.": "Завантажити будь-який файл з вашого пристрою.",
|
||||||
|
"Uploading {{name}}": "Завантаження {{name}}",
|
||||||
|
"Uploading file": "Завантаження файлу",
|
||||||
"Table": "Таблиця",
|
"Table": "Таблиця",
|
||||||
"Insert a table.": "Вставити таблицю.",
|
"Insert a table.": "Вставити таблицю.",
|
||||||
"Insert collapsible block.": "Вставити блок, що згортається.",
|
"Insert collapsible block.": "Вставити блок, що згортається.",
|
||||||
@@ -495,5 +502,78 @@
|
|||||||
"Page restored successfully": "Сторінку успішно відновлено",
|
"Page restored successfully": "Сторінку успішно відновлено",
|
||||||
"Deleted by": "Видалено",
|
"Deleted by": "Видалено",
|
||||||
"Deleted at": "Видалено о",
|
"Deleted at": "Видалено о",
|
||||||
"Preview": "Попередній перегляд"
|
"Preview": "Попередній перегляд",
|
||||||
|
"Subpages": "Підсторінки",
|
||||||
|
"Failed to load subpages": "Не вдалося завантажити підсторінки",
|
||||||
|
"No subpages": "Немає підсторінок",
|
||||||
|
"Subpages (Child pages)": "Підсторінки (дочірні сторінки)",
|
||||||
|
"List all subpages of the current page": "Перелік всіх підсторінок поточної сторінки",
|
||||||
|
"Attachments": "Вкладення",
|
||||||
|
"All spaces": "Усі простори",
|
||||||
|
"Unknown": "Невідомо",
|
||||||
|
"Find a space": "Знайти простір",
|
||||||
|
"Search in all your spaces": "Шукати у всіх ваших просторах",
|
||||||
|
"Type": "Тип",
|
||||||
|
"Enterprise": "Підприємство",
|
||||||
|
"Download attachment": "Завантажити вкладення",
|
||||||
|
"Allowed email domains": "Дозволені домени електронної пошти",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "Лише користувачі з адресами електронної пошти з цих доменів можуть реєструватися через SSO.",
|
||||||
|
"Enter valid domain names separated by comma or space": "Введіть дійсні доменні імена, розділені комою або пробілом",
|
||||||
|
"Enforce two-factor authentication": "Вимагати двофакторну автентифікацію",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "Після увімкнення всі учасники повинні ввімкнути двофакторну автентифікацію для доступу до робочого простору.",
|
||||||
|
"Toggle MFA enforcement": "Перемикання вимоги MFA",
|
||||||
|
"Display name": "Відображуване ім'я",
|
||||||
|
"Allow signup": "Дозволити реєстрацію",
|
||||||
|
"Enabled": "Увімкнено",
|
||||||
|
"Advanced Settings": "Розширені налаштування",
|
||||||
|
"Enable TLS/SSL": "Увімкнути TLS/SSL",
|
||||||
|
"Use secure connection to LDAP server": "Використовувати захищене з'єднання з сервером LDAP",
|
||||||
|
"Group sync": "Синхронізація групи",
|
||||||
|
"No SSO providers found.": "Постачальників SSO не знайдено.",
|
||||||
|
"Delete SSO provider": "Видалити постачальника SSO",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "Ви впевнені, що хочете видалити цього постачальника SSO?",
|
||||||
|
"Action": "Дія",
|
||||||
|
"{{ssoProviderType}} configuration": "Конфігурація {{ssoProviderType}}",
|
||||||
|
"Icon": "Іконка",
|
||||||
|
"Upload image": "Завантажити зображення",
|
||||||
|
"Remove image": "Видалити зображення",
|
||||||
|
"Failed to remove image": "Не вдалося видалити зображення",
|
||||||
|
"Image exceeds 10MB limit.": "Зображення має займати менше, ніж 10 МБ.",
|
||||||
|
"Image removed successfully": "Зображення видалено",
|
||||||
|
"API key": "Ключ API",
|
||||||
|
"API key created successfully": "Ключ API успішно створено",
|
||||||
|
"API keys": "Ключі API",
|
||||||
|
"API management": "Управління API",
|
||||||
|
"Are you sure you want to revoke this API key": "Ви впевнені, що хочете відкликати цей ключ API",
|
||||||
|
"Create API Key": "Створити ключ API",
|
||||||
|
"Custom expiration date": "Користувацька дата закінчення",
|
||||||
|
"Enter a descriptive token name": "Введіть описову назву токена",
|
||||||
|
"Expiration": "Термін дії",
|
||||||
|
"Expired": "Закінчився",
|
||||||
|
"Expires": "Закінчується",
|
||||||
|
"I've saved my API key": "Я зберіг свій ключ API",
|
||||||
|
"Last use": "Останнє використання",
|
||||||
|
"No API keys found": "Ключі API не знайдено",
|
||||||
|
"No expiration": "Без терміну дії",
|
||||||
|
"Revoke API key": "Відкликати ключ API",
|
||||||
|
"Revoked successfully": "Успішно відкликано",
|
||||||
|
"Select expiration date": "Виберіть дату закінчення",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "Цю дію не можна скасувати. Будь-які додатки, що використовують цей ключ API, перестануть працювати.",
|
||||||
|
"Update API key": "Оновити ключ API",
|
||||||
|
"Manage API keys for all users in the workspace": "Керувати ключами API для всіх користувачів у робочій області",
|
||||||
|
"AI settings": "Налаштування ШІ",
|
||||||
|
"AI search": "Пошук з ШІ",
|
||||||
|
"AI Answer": "Відповідь ШІ",
|
||||||
|
"Ask AI": "Запитати ШІ",
|
||||||
|
"AI is thinking...": "ШІ думає...",
|
||||||
|
"Ask a question...": "Задайте питання...",
|
||||||
|
"AI-powered search (Ask AI)": "Пошук на базі ШІ (Запитати ШІ)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "Пошук з ШІ використовує векторні вбудовування для надання можливостей семантичного пошуку у вашому робочому вмісті.",
|
||||||
|
"Toggle AI search": "Переключити пошук з ШІ",
|
||||||
|
"Sources": "Джерела",
|
||||||
|
"Ask AI not available for attachments": "Запитати ШІ недоступно для вкладень",
|
||||||
|
"No answer available": "Відповідь недоступна",
|
||||||
|
"Background color": "Колір фону",
|
||||||
|
"Highlight color": "Колір підсвічування",
|
||||||
|
"Remove color": "Видалити колір"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,6 +29,7 @@
|
|||||||
"Choose your preferred interface language.": "选择您喜欢的界面语言。",
|
"Choose your preferred interface language.": "选择您喜欢的界面语言。",
|
||||||
"Choose your preferred page width.": "选择您喜欢的页面宽度。",
|
"Choose your preferred page width.": "选择您喜欢的页面宽度。",
|
||||||
"Confirm": "确认",
|
"Confirm": "确认",
|
||||||
|
"Copy as Markdown": "复制为Markdown",
|
||||||
"Copy link": "复制链接",
|
"Copy link": "复制链接",
|
||||||
"Create": "创建",
|
"Create": "创建",
|
||||||
"Create group": "创建群组",
|
"Create group": "创建群组",
|
||||||
@@ -53,6 +54,7 @@
|
|||||||
"e.g Space for product team": "例如:产品团队的空间",
|
"e.g Space for product team": "例如:产品团队的空间",
|
||||||
"e.g Space for sales team to collaborate": "例如:销售团队协作的空间",
|
"e.g Space for sales team to collaborate": "例如:销售团队协作的空间",
|
||||||
"Edit": "编辑",
|
"Edit": "编辑",
|
||||||
|
"Read": "阅读",
|
||||||
"Edit group": "编辑群组",
|
"Edit group": "编辑群组",
|
||||||
"Email": "电子邮箱",
|
"Email": "电子邮箱",
|
||||||
"Enter a strong password": "输入一个强密码",
|
"Enter a strong password": "输入一个强密码",
|
||||||
@@ -121,6 +123,8 @@
|
|||||||
"page": "个页面",
|
"page": "个页面",
|
||||||
"Page deleted successfully": "页面已成功删除",
|
"Page deleted successfully": "页面已成功删除",
|
||||||
"Page history": "页面历史",
|
"Page history": "页面历史",
|
||||||
|
"Select version": "选择版本",
|
||||||
|
"Highlight changes": "突出显示更改",
|
||||||
"Page import is in progress. Please do not close this tab.": "页面导入正在进行中。请不要关闭此标签页。",
|
"Page import is in progress. Please do not close this tab.": "页面导入正在进行中。请不要关闭此标签页。",
|
||||||
"Pages": "页面",
|
"Pages": "页面",
|
||||||
"pages": "个页面",
|
"pages": "个页面",
|
||||||
@@ -252,6 +256,7 @@
|
|||||||
"Export failed:": "导出失败:",
|
"Export failed:": "导出失败:",
|
||||||
"export error": "导出出错",
|
"export error": "导出出错",
|
||||||
"Export page": "导出页面",
|
"Export page": "导出页面",
|
||||||
|
"Export successful": "导出成功",
|
||||||
"Export space": "导出空间",
|
"Export space": "导出空间",
|
||||||
"Export {{type}}": "导出为 {{type}}",
|
"Export {{type}}": "导出为 {{type}}",
|
||||||
"File exceeds the {{limit}} attachment limit": "文件超出了 {{limit}} 类型附件限制",
|
"File exceeds the {{limit}} attachment limit": "文件超出了 {{limit}} 类型附件限制",
|
||||||
@@ -327,6 +332,8 @@
|
|||||||
"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 file from your device.": "从设备上传任何文件",
|
"Upload any file from your device.": "从设备上传任何文件",
|
||||||
|
"Uploading {{name}}": "正在上传{{name}}",
|
||||||
|
"Uploading file": "正在上传文件",
|
||||||
"Table": "表格",
|
"Table": "表格",
|
||||||
"Insert a table.": "插入一个表格",
|
"Insert a table.": "插入一个表格",
|
||||||
"Insert collapsible block.": "插入一个折叠块",
|
"Insert collapsible block.": "插入一个折叠块",
|
||||||
@@ -495,5 +502,78 @@
|
|||||||
"Page restored successfully": "页面恢复成功",
|
"Page restored successfully": "页面恢复成功",
|
||||||
"Deleted by": "删除人",
|
"Deleted by": "删除人",
|
||||||
"Deleted at": "删除时间",
|
"Deleted at": "删除时间",
|
||||||
"Preview": "预览"
|
"Preview": "预览",
|
||||||
|
"Subpages": "子页面",
|
||||||
|
"Failed to load subpages": "加载子页面失败",
|
||||||
|
"No subpages": "没有子页面",
|
||||||
|
"Subpages (Child pages)": "子页面(子页面)",
|
||||||
|
"List all subpages of the current page": "列出当前页面的所有子页面",
|
||||||
|
"Attachments": "附件",
|
||||||
|
"All spaces": "所有空间",
|
||||||
|
"Unknown": "未知",
|
||||||
|
"Find a space": "查找空间",
|
||||||
|
"Search in all your spaces": "在您的所有空间中搜索",
|
||||||
|
"Type": "类型",
|
||||||
|
"Enterprise": "企业",
|
||||||
|
"Download attachment": "下载附件",
|
||||||
|
"Allowed email domains": "允许的电子邮件域",
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.": "只有来自这些域的电子邮件地址的用户才能通过SSO注册。",
|
||||||
|
"Enter valid domain names separated by comma or space": "输入用逗号或空格分隔的有效域名",
|
||||||
|
"Enforce two-factor authentication": "强制实施双因素认证",
|
||||||
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "一旦实施,所有成员必须启用双因素认证才能访问工作区。",
|
||||||
|
"Toggle MFA enforcement": "切换多因素认证实施",
|
||||||
|
"Display name": "显示名称",
|
||||||
|
"Allow signup": "允许注册",
|
||||||
|
"Enabled": "已启用",
|
||||||
|
"Advanced Settings": "高级设置",
|
||||||
|
"Enable TLS/SSL": "启用TLS/SSL",
|
||||||
|
"Use secure connection to LDAP server": "使用安全连接到LDAP服务器",
|
||||||
|
"Group sync": "组同步",
|
||||||
|
"No SSO providers found.": "未找到SSO提供商。",
|
||||||
|
"Delete SSO provider": "删除SSO提供商",
|
||||||
|
"Are you sure you want to delete this SSO provider?": "您确定要删除此SSO提供商吗?",
|
||||||
|
"Action": "操作",
|
||||||
|
"{{ssoProviderType}} configuration": "{{ssoProviderType}} 配置",
|
||||||
|
"Icon": "图标",
|
||||||
|
"Upload image": "上传图片",
|
||||||
|
"Remove image": "删除图片",
|
||||||
|
"Failed to remove image": "无法删除图片",
|
||||||
|
"Image exceeds 10MB limit.": "图片超过10MB限制。",
|
||||||
|
"Image removed successfully": "图片删除成功",
|
||||||
|
"API key": "API密钥",
|
||||||
|
"API key created successfully": "API密钥创建成功",
|
||||||
|
"API keys": "API密钥",
|
||||||
|
"API management": "API管理",
|
||||||
|
"Are you sure you want to revoke this API key": "确定要撤销此API密钥吗",
|
||||||
|
"Create API Key": "创建API密钥",
|
||||||
|
"Custom expiration date": "自定义到期日期",
|
||||||
|
"Enter a descriptive token name": "输入描述性令牌名称",
|
||||||
|
"Expiration": "到期",
|
||||||
|
"Expired": "已过期",
|
||||||
|
"Expires": "到期",
|
||||||
|
"I've saved my API key": "我已保存我的API密钥",
|
||||||
|
"Last use": "上次使用",
|
||||||
|
"No API keys found": "找不到API密钥",
|
||||||
|
"No expiration": "无到期",
|
||||||
|
"Revoke API key": "撤销API密钥",
|
||||||
|
"Revoked successfully": "撤销成功",
|
||||||
|
"Select expiration date": "选择到期日期",
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.": "此操作无法撤销。使用此API密钥的任何应用程序将停止工作。",
|
||||||
|
"Update API key": "更新API密钥",
|
||||||
|
"Manage API keys for all users in the workspace": "管理工作空间中所有用户的API密钥",
|
||||||
|
"AI settings": "AI设置",
|
||||||
|
"AI search": "AI搜索",
|
||||||
|
"AI Answer": "AI回答",
|
||||||
|
"Ask AI": "询问AI",
|
||||||
|
"AI is thinking...": "AI正在思考...",
|
||||||
|
"Ask a question...": "提问...",
|
||||||
|
"AI-powered search (Ask AI)": "AI驱动的搜索(询问AI)",
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "AI搜索使用向量嵌入提供跨工作空间内容的语义搜索功能。",
|
||||||
|
"Toggle AI search": "切换AI搜索",
|
||||||
|
"Sources": "来源",
|
||||||
|
"Ask AI not available for attachments": "附件不支持询问AI",
|
||||||
|
"No answer available": "无可用答案",
|
||||||
|
"Background color": "背景颜色",
|
||||||
|
"Highlight color": "突出显示颜色",
|
||||||
|
"Remove color": "移除颜色"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,30 @@
|
|||||||
|
{
|
||||||
|
"name": "Docmost",
|
||||||
|
"short_name": "Docmost",
|
||||||
|
"start_url": "/",
|
||||||
|
"display": "standalone",
|
||||||
|
"background_color": "#222",
|
||||||
|
"theme_color": "#222",
|
||||||
|
"icons": [
|
||||||
|
{
|
||||||
|
"src": "icons/favicon-16x16.png",
|
||||||
|
"type": "image/png",
|
||||||
|
"sizes": "16x16"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "icons/favicon-32x32.png",
|
||||||
|
"type": "image/png",
|
||||||
|
"sizes": "32x32"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "icons/app-icon-192x192.png",
|
||||||
|
"type": "image/png",
|
||||||
|
"sizes": "180x180 192x192"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "icons/app-icon-512x512.png",
|
||||||
|
"type": "image/png",
|
||||||
|
"sizes": "512x512"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -35,6 +35,9 @@ import SpacesPage from "@/pages/spaces/spaces.tsx";
|
|||||||
import { MfaChallengePage } from "@/ee/mfa/pages/mfa-challenge-page";
|
import { MfaChallengePage } from "@/ee/mfa/pages/mfa-challenge-page";
|
||||||
import { MfaSetupRequiredPage } from "@/ee/mfa/pages/mfa-setup-required-page";
|
import { MfaSetupRequiredPage } from "@/ee/mfa/pages/mfa-setup-required-page";
|
||||||
import SpaceTrash from "@/pages/space/space-trash.tsx";
|
import SpaceTrash from "@/pages/space/space-trash.tsx";
|
||||||
|
import UserApiKeys from "@/ee/api-key/pages/user-api-keys";
|
||||||
|
import WorkspaceApiKeys from "@/ee/api-key/pages/workspace-api-keys";
|
||||||
|
import AiSettings from "@/ee/ai/pages/ai-settings.tsx";
|
||||||
|
|
||||||
export default function App() {
|
export default function App() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -96,13 +99,16 @@ export default function App() {
|
|||||||
path={"account/preferences"}
|
path={"account/preferences"}
|
||||||
element={<AccountPreferences />}
|
element={<AccountPreferences />}
|
||||||
/>
|
/>
|
||||||
|
<Route path={"account/api-keys"} element={<UserApiKeys />} />
|
||||||
<Route path={"workspace"} element={<WorkspaceSettings />} />
|
<Route path={"workspace"} element={<WorkspaceSettings />} />
|
||||||
<Route path={"members"} element={<WorkspaceMembers />} />
|
<Route path={"members"} element={<WorkspaceMembers />} />
|
||||||
|
<Route path={"api-keys"} element={<WorkspaceApiKeys />} />
|
||||||
<Route path={"groups"} element={<Groups />} />
|
<Route path={"groups"} element={<Groups />} />
|
||||||
<Route path={"groups/:groupId"} element={<GroupInfo />} />
|
<Route path={"groups/:groupId"} element={<GroupInfo />} />
|
||||||
<Route path={"spaces"} element={<Spaces />} />
|
<Route path={"spaces"} element={<Spaces />} />
|
||||||
<Route path={"sharing"} element={<Shares />} />
|
<Route path={"sharing"} element={<Shares />} />
|
||||||
<Route path={"security"} element={<Security />} />
|
<Route path={"security"} element={<Security />} />
|
||||||
|
<Route path={"ai"} element={<AiSettings />} />
|
||||||
{!isCloud() && <Route path={"license"} element={<License />} />}
|
{!isCloud() && <Route path={"license"} element={<License />} />}
|
||||||
{isCloud() && <Route path={"billing"} element={<Billing />} />}
|
{isCloud() && <Route path={"billing"} element={<Billing />} />}
|
||||||
</Route>
|
</Route>
|
||||||
|
|||||||
@@ -0,0 +1,165 @@
|
|||||||
|
import React, { useRef } from "react";
|
||||||
|
import { Menu, Box, Loader } from "@mantine/core";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { IconTrash, IconUpload } from "@tabler/icons-react";
|
||||||
|
import { CustomAvatar } from "@/components/ui/custom-avatar.tsx";
|
||||||
|
import { AvatarIconType } from "@/features/attachments/types/attachment.types.ts";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
|
||||||
|
interface AvatarUploaderProps {
|
||||||
|
currentImageUrl?: string | null;
|
||||||
|
fallbackName?: string;
|
||||||
|
radius?: string | number;
|
||||||
|
size?: string | number;
|
||||||
|
variant?: string;
|
||||||
|
type: AvatarIconType;
|
||||||
|
onUpload: (file: File) => Promise<void>;
|
||||||
|
onRemove: () => Promise<void>;
|
||||||
|
isLoading?: boolean;
|
||||||
|
disabled?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function AvatarUploader({
|
||||||
|
currentImageUrl,
|
||||||
|
fallbackName,
|
||||||
|
radius,
|
||||||
|
variant,
|
||||||
|
size,
|
||||||
|
type,
|
||||||
|
onUpload,
|
||||||
|
onRemove,
|
||||||
|
isLoading = false,
|
||||||
|
disabled = false,
|
||||||
|
}: AvatarUploaderProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const fileInputRef = useRef<HTMLInputElement>(null);
|
||||||
|
|
||||||
|
const handleFileInputChange = async (
|
||||||
|
event: React.ChangeEvent<HTMLInputElement>,
|
||||||
|
) => {
|
||||||
|
const file = event.target.files?.[0];
|
||||||
|
if (!file || disabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate file size (max 10MB)
|
||||||
|
const maxSizeInBytes = 10 * 1024 * 1024;
|
||||||
|
if (file.size > maxSizeInBytes) {
|
||||||
|
notifications.show({
|
||||||
|
message: t("Image exceeds 10MB limit."),
|
||||||
|
color: "red",
|
||||||
|
});
|
||||||
|
// Reset the input
|
||||||
|
if (fileInputRef.current) {
|
||||||
|
fileInputRef.current.value = "";
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await onUpload(file);
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
notifications.show({
|
||||||
|
message: t("Failed to upload image"),
|
||||||
|
color: "red",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reset the input so the same file can be selected again
|
||||||
|
if (fileInputRef.current) {
|
||||||
|
fileInputRef.current.value = "";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleUploadClick = () => {
|
||||||
|
if (fileInputRef.current) {
|
||||||
|
fileInputRef.current.click();
|
||||||
|
} else {
|
||||||
|
console.error("File input ref is null!");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleRemove = async () => {
|
||||||
|
if (disabled) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await onRemove();
|
||||||
|
notifications.show({
|
||||||
|
message: t("Image removed successfully"),
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
notifications.show({
|
||||||
|
message: t("Failed to remove image"),
|
||||||
|
color: "red",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box>
|
||||||
|
<input
|
||||||
|
type="file"
|
||||||
|
ref={fileInputRef}
|
||||||
|
onChange={handleFileInputChange}
|
||||||
|
accept="image/png,image/jpeg,image/jpg"
|
||||||
|
style={{ display: "none" }}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Menu shadow="md" width={200} withArrow disabled={disabled || isLoading}>
|
||||||
|
<Menu.Target>
|
||||||
|
<Box style={{ position: "relative", display: "inline-block" }}>
|
||||||
|
<CustomAvatar
|
||||||
|
component="button"
|
||||||
|
size={size}
|
||||||
|
avatarUrl={currentImageUrl}
|
||||||
|
name={fallbackName}
|
||||||
|
style={{
|
||||||
|
cursor: disabled || isLoading ? "default" : "pointer",
|
||||||
|
opacity: isLoading ? 0.6 : 1,
|
||||||
|
}}
|
||||||
|
radius={radius}
|
||||||
|
variant={variant}
|
||||||
|
type={type}
|
||||||
|
/>
|
||||||
|
{isLoading && (
|
||||||
|
<Box
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
top: "50%",
|
||||||
|
left: "50%",
|
||||||
|
transform: "translate(-50%, -50%)",
|
||||||
|
zIndex: 1000,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Loader size="sm" />
|
||||||
|
</Box>
|
||||||
|
)}
|
||||||
|
</Box>
|
||||||
|
</Menu.Target>
|
||||||
|
|
||||||
|
<Menu.Dropdown>
|
||||||
|
<Menu.Item
|
||||||
|
leftSection={<IconUpload size={16} />}
|
||||||
|
disabled={isLoading || disabled}
|
||||||
|
onClick={handleUploadClick}
|
||||||
|
>
|
||||||
|
{t("Upload image")}
|
||||||
|
</Menu.Item>
|
||||||
|
|
||||||
|
{currentImageUrl && (
|
||||||
|
<Menu.Item
|
||||||
|
leftSection={<IconTrash size={16} />}
|
||||||
|
color="red"
|
||||||
|
onClick={handleRemove}
|
||||||
|
disabled={isLoading || disabled}
|
||||||
|
>
|
||||||
|
{t("Remove image")}
|
||||||
|
</Menu.Item>
|
||||||
|
)}
|
||||||
|
</Menu.Dropdown>
|
||||||
|
</Menu>
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -30,9 +30,11 @@ export default function ExportModal({
|
|||||||
const [format, setFormat] = useState<ExportFormat>(ExportFormat.Markdown);
|
const [format, setFormat] = useState<ExportFormat>(ExportFormat.Markdown);
|
||||||
const [includeChildren, setIncludeChildren] = useState<boolean>(false);
|
const [includeChildren, setIncludeChildren] = useState<boolean>(false);
|
||||||
const [includeAttachments, setIncludeAttachments] = useState<boolean>(false);
|
const [includeAttachments, setIncludeAttachments] = useState<boolean>(false);
|
||||||
|
const [isExporting, setIsExporting] = useState<boolean>(false);
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
const handleExport = async () => {
|
const handleExport = async () => {
|
||||||
|
setIsExporting(true);
|
||||||
try {
|
try {
|
||||||
if (type === "page") {
|
if (type === "page") {
|
||||||
await exportPage({
|
await exportPage({
|
||||||
@@ -45,6 +47,9 @@ export default function ExportModal({
|
|||||||
if (type === "space") {
|
if (type === "space") {
|
||||||
await exportSpace({ spaceId: id, format, includeAttachments });
|
await exportSpace({ spaceId: id, format, includeAttachments });
|
||||||
}
|
}
|
||||||
|
notifications.show({
|
||||||
|
message: t("Export successful"),
|
||||||
|
});
|
||||||
onClose();
|
onClose();
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
notifications.show({
|
notifications.show({
|
||||||
@@ -52,6 +57,8 @@ export default function ExportModal({
|
|||||||
color: "red",
|
color: "red",
|
||||||
});
|
});
|
||||||
console.error("export error", err);
|
console.error("export error", err);
|
||||||
|
} finally {
|
||||||
|
setIsExporting(false);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -136,7 +143,7 @@ export default function ExportModal({
|
|||||||
<Button onClick={onClose} variant="default">
|
<Button onClick={onClose} variant="default">
|
||||||
{t("Cancel")}
|
{t("Cancel")}
|
||||||
</Button>
|
</Button>
|
||||||
<Button onClick={handleExport}>{t("Export")}</Button>
|
<Button onClick={handleExport} loading={isExporting}>{t("Export")}</Button>
|
||||||
</Group>
|
</Group>
|
||||||
</Modal.Body>
|
</Modal.Body>
|
||||||
</Modal.Content>
|
</Modal.Content>
|
||||||
|
|||||||
@@ -4,14 +4,15 @@ import { useTranslation } from "react-i18next";
|
|||||||
|
|
||||||
interface NoTableResultsProps {
|
interface NoTableResultsProps {
|
||||||
colSpan: number;
|
colSpan: number;
|
||||||
|
text?: string;
|
||||||
}
|
}
|
||||||
export default function NoTableResults({ colSpan }: NoTableResultsProps) {
|
export default function NoTableResults({ colSpan, text }: NoTableResultsProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
return (
|
return (
|
||||||
<Table.Tr>
|
<Table.Tr>
|
||||||
<Table.Td colSpan={colSpan}>
|
<Table.Td colSpan={colSpan}>
|
||||||
<Text fw={500} c="dimmed" ta="center">
|
<Text fw={500} c="dimmed" ta="center">
|
||||||
{t("No results found...")}
|
{text || t("No results found...")}
|
||||||
</Text>
|
</Text>
|
||||||
</Table.Td>
|
</Table.Td>
|
||||||
</Table.Tr>
|
</Table.Tr>
|
||||||
|
|||||||
@@ -2,17 +2,17 @@ import { Button, Group } from "@mantine/core";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
export interface PagePaginationProps {
|
export interface PagePaginationProps {
|
||||||
currentPage: number;
|
|
||||||
hasPrevPage: boolean;
|
hasPrevPage: boolean;
|
||||||
hasNextPage: boolean;
|
hasNextPage: boolean;
|
||||||
onPageChange: (newPage: number) => void;
|
onPrev: () => void;
|
||||||
|
onNext: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Paginate({
|
export default function Paginate({
|
||||||
currentPage,
|
|
||||||
hasPrevPage,
|
hasPrevPage,
|
||||||
hasNextPage,
|
hasNextPage,
|
||||||
onPageChange,
|
onPrev,
|
||||||
|
onNext,
|
||||||
}: PagePaginationProps) {
|
}: PagePaginationProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
@@ -25,7 +25,7 @@ export default function Paginate({
|
|||||||
<Button
|
<Button
|
||||||
variant="default"
|
variant="default"
|
||||||
size="compact-sm"
|
size="compact-sm"
|
||||||
onClick={() => onPageChange(currentPage - 1)}
|
onClick={onPrev}
|
||||||
disabled={!hasPrevPage}
|
disabled={!hasPrevPage}
|
||||||
>
|
>
|
||||||
{t("Prev")}
|
{t("Prev")}
|
||||||
@@ -34,7 +34,7 @@ export default function Paginate({
|
|||||||
<Button
|
<Button
|
||||||
variant="default"
|
variant="default"
|
||||||
size="compact-sm"
|
size="compact-sm"
|
||||||
onClick={() => onPageChange(currentPage + 1)}
|
onClick={onNext}
|
||||||
disabled={!hasNextPage}
|
disabled={!hasNextPage}
|
||||||
>
|
>
|
||||||
{t("Next")}
|
{t("Next")}
|
||||||
|
|||||||
@@ -5,26 +5,27 @@ import {
|
|||||||
Badge,
|
Badge,
|
||||||
Table,
|
Table,
|
||||||
ActionIcon,
|
ActionIcon,
|
||||||
} from '@mantine/core';
|
} from "@mantine/core";
|
||||||
import {Link} from 'react-router-dom';
|
import { Link } from "react-router-dom";
|
||||||
import PageListSkeleton from '@/components/ui/page-list-skeleton.tsx';
|
import PageListSkeleton from "@/components/ui/page-list-skeleton.tsx";
|
||||||
import { buildPageUrl } from '@/features/page/page.utils.ts';
|
import { buildPageUrl } from "@/features/page/page.utils.ts";
|
||||||
import { formattedDate } from '@/lib/time.ts';
|
import { formattedDate } from "@/lib/time.ts";
|
||||||
import { useRecentChangesQuery } from '@/features/page/queries/page-query.ts';
|
import { useRecentChangesQuery } from "@/features/page/queries/page-query.ts";
|
||||||
import { IconFileDescription } from '@tabler/icons-react';
|
import { IconFileDescription } from "@tabler/icons-react";
|
||||||
import { getSpaceUrl } from '@/lib/config.ts';
|
import { getSpaceUrl } from "@/lib/config.ts";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { getInitialsColor } from "@/lib/get-initials-color.ts";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
spaceId?: string;
|
spaceId?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function RecentChanges({spaceId}: Props) {
|
export default function RecentChanges({ spaceId }: Props) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const {data: pages, isLoading, isError} = useRecentChangesQuery(spaceId);
|
const { data: pages, isLoading, isError } = useRecentChangesQuery(spaceId);
|
||||||
|
|
||||||
if (isLoading) {
|
if (isLoading) {
|
||||||
return <PageListSkeleton/>;
|
return <PageListSkeleton />;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isError) {
|
if (isError) {
|
||||||
@@ -44,8 +45,8 @@ export default function RecentChanges({spaceId}: Props) {
|
|||||||
>
|
>
|
||||||
<Group wrap="nowrap">
|
<Group wrap="nowrap">
|
||||||
{page.icon || (
|
{page.icon || (
|
||||||
<ActionIcon variant='transparent' color='gray' size={18}>
|
<ActionIcon variant="transparent" color="gray" size={18}>
|
||||||
<IconFileDescription size={18}/>
|
<IconFileDescription size={18} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
@@ -58,18 +59,23 @@ export default function RecentChanges({spaceId}: Props) {
|
|||||||
{!spaceId && (
|
{!spaceId && (
|
||||||
<Table.Td>
|
<Table.Td>
|
||||||
<Badge
|
<Badge
|
||||||
color="blue"
|
color={getInitialsColor(page?.space.name)}
|
||||||
variant="light"
|
variant="light"
|
||||||
component={Link}
|
component={Link}
|
||||||
to={getSpaceUrl(page?.space.slug)}
|
to={getSpaceUrl(page?.space.slug)}
|
||||||
style={{cursor: 'pointer'}}
|
style={{ cursor: "pointer" }}
|
||||||
>
|
>
|
||||||
{page?.space.name}
|
{page?.space.name}
|
||||||
</Badge>
|
</Badge>
|
||||||
</Table.Td>
|
</Table.Td>
|
||||||
)}
|
)}
|
||||||
<Table.Td>
|
<Table.Td>
|
||||||
<Text c="dimmed" style={{whiteSpace: 'nowrap'}} size="xs" fw={500}>
|
<Text
|
||||||
|
c="dimmed"
|
||||||
|
style={{ whiteSpace: "nowrap" }}
|
||||||
|
size="xs"
|
||||||
|
fw={500}
|
||||||
|
>
|
||||||
{formattedDate(page.updatedAt)}
|
{formattedDate(page.updatedAt)}
|
||||||
</Text>
|
</Text>
|
||||||
</Table.Td>
|
</Table.Td>
|
||||||
|
|||||||
@@ -14,6 +14,14 @@ import SidebarToggle from "@/components/ui/sidebar-toggle-button.tsx";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import useTrial from "@/ee/hooks/use-trial.tsx";
|
import useTrial from "@/ee/hooks/use-trial.tsx";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
import { isCloud } from "@/lib/config.ts";
|
||||||
|
import {
|
||||||
|
SearchControl,
|
||||||
|
SearchMobileControl,
|
||||||
|
} from "@/features/search/components/search-control.tsx";
|
||||||
|
import {
|
||||||
|
searchSpotlight,
|
||||||
|
shareSearchSpotlight,
|
||||||
|
} from "@/features/search/constants.ts";
|
||||||
|
|
||||||
const links = [{ link: APP_ROUTE.HOME, label: "Home" }];
|
const links = [{ link: APP_ROUTE.HOME, label: "Home" }];
|
||||||
|
|
||||||
@@ -79,6 +87,15 @@ export function AppHeader() {
|
|||||||
</Group>
|
</Group>
|
||||||
</Group>
|
</Group>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<Group visibleFrom="sm">
|
||||||
|
<SearchControl onClick={searchSpotlight.open} />
|
||||||
|
</Group>
|
||||||
|
<Group hiddenFrom="sm">
|
||||||
|
<SearchMobileControl onSearch={searchSpotlight.open} />
|
||||||
|
</Group>
|
||||||
|
</div>
|
||||||
|
|
||||||
<Group px={"xl"} wrap="nowrap">
|
<Group px={"xl"} wrap="nowrap">
|
||||||
{isCloud() && isTrial && trialDaysLeft !== 0 && (
|
{isCloud() && isTrial && trialDaysLeft !== 0 && (
|
||||||
<Badge
|
<Badge
|
||||||
|
|||||||
@@ -1,16 +1,23 @@
|
|||||||
import { UserProvider } from "@/features/user/user-provider.tsx";
|
import { UserProvider } from "@/features/user/user-provider.tsx";
|
||||||
import { Outlet } from "react-router-dom";
|
import { Outlet, useParams } from "react-router-dom";
|
||||||
import GlobalAppShell from "@/components/layouts/global/global-app-shell.tsx";
|
import GlobalAppShell from "@/components/layouts/global/global-app-shell.tsx";
|
||||||
import { PosthogUser } from "@/ee/components/posthog-user.tsx";
|
import { PosthogUser } from "@/ee/components/posthog-user.tsx";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
import { isCloud } from "@/lib/config.ts";
|
||||||
|
import { SearchSpotlight } from "@/features/search/components/search-spotlight.tsx";
|
||||||
|
import React from "react";
|
||||||
|
import { useGetSpaceBySlugQuery } from "@/features/space/queries/space-query.ts";
|
||||||
|
|
||||||
export default function Layout() {
|
export default function Layout() {
|
||||||
|
const { spaceSlug } = useParams();
|
||||||
|
const { data: space } = useGetSpaceBySlugQuery(spaceSlug);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<UserProvider>
|
<UserProvider>
|
||||||
<GlobalAppShell>
|
<GlobalAppShell>
|
||||||
<Outlet />
|
<Outlet />
|
||||||
</GlobalAppShell>
|
</GlobalAppShell>
|
||||||
{isCloud() && <PosthogUser />}
|
{isCloud() && <PosthogUser />}
|
||||||
|
<SearchSpotlight spaceId={space?.id} />
|
||||||
</UserProvider>
|
</UserProvider>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import {
|
import {
|
||||||
Group,
|
Group,
|
||||||
Menu,
|
Menu,
|
||||||
UnstyledButton,
|
|
||||||
Text,
|
Text,
|
||||||
|
UnstyledButton,
|
||||||
useMantineColorScheme,
|
useMantineColorScheme,
|
||||||
} from "@mantine/core";
|
} from "@mantine/core";
|
||||||
import {
|
import {
|
||||||
@@ -10,7 +10,6 @@ import {
|
|||||||
IconBrush,
|
IconBrush,
|
||||||
IconCheck,
|
IconCheck,
|
||||||
IconChevronDown,
|
IconChevronDown,
|
||||||
IconChevronRight,
|
|
||||||
IconDeviceDesktop,
|
IconDeviceDesktop,
|
||||||
IconLogout,
|
IconLogout,
|
||||||
IconMoon,
|
IconMoon,
|
||||||
@@ -26,6 +25,7 @@ import APP_ROUTE from "@/lib/app-route.ts";
|
|||||||
import useAuth from "@/features/auth/hooks/use-auth.ts";
|
import useAuth from "@/features/auth/hooks/use-auth.ts";
|
||||||
import { CustomAvatar } from "@/components/ui/custom-avatar.tsx";
|
import { CustomAvatar } from "@/components/ui/custom-avatar.tsx";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { AvatarIconType } from "@/features/attachments/types/attachment.types.ts";
|
||||||
|
|
||||||
export default function TopMenu() {
|
export default function TopMenu() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -50,6 +50,7 @@ export default function TopMenu() {
|
|||||||
name={workspace?.name}
|
name={workspace?.name}
|
||||||
variant="filled"
|
variant="filled"
|
||||||
size="sm"
|
size="sm"
|
||||||
|
type={AvatarIconType.WORKSPACE_ICON}
|
||||||
/>
|
/>
|
||||||
<Text fw={500} size="sm" lh={1} mr={3} lineClamp={1}>
|
<Text fw={500} size="sm" lh={1} mr={3} lineClamp={1}>
|
||||||
{workspace?.name}
|
{workspace?.name}
|
||||||
|
|||||||
@@ -50,7 +50,7 @@ export default function AppVersion() {
|
|||||||
href="https://github.com/docmost/docmost/releases"
|
href="https://github.com/docmost/docmost/releases"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
>
|
>
|
||||||
v{APP_VERSION}
|
{appVersion?.currentVersion && <>v{appVersion?.currentVersion}</>}
|
||||||
</Text>
|
</Text>
|
||||||
</Indicator>
|
</Indicator>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|||||||
@@ -10,9 +10,10 @@ import { getWorkspaceMembers } from "@/features/workspace/services/workspace-ser
|
|||||||
import { getLicenseInfo } from "@/ee/licence/services/license-service.ts";
|
import { getLicenseInfo } from "@/ee/licence/services/license-service.ts";
|
||||||
import { getSsoProviders } from "@/ee/security/services/security-service.ts";
|
import { getSsoProviders } from "@/ee/security/services/security-service.ts";
|
||||||
import { getShares } from "@/features/share/services/share-service.ts";
|
import { getShares } from "@/features/share/services/share-service.ts";
|
||||||
|
import { getApiKeys } from "@/ee/api-key";
|
||||||
|
|
||||||
export const prefetchWorkspaceMembers = () => {
|
export const prefetchWorkspaceMembers = () => {
|
||||||
const params = { limit: 100, page: 1, query: "" } as QueryParams;
|
const params: QueryParams = { limit: 100, query: "" };
|
||||||
queryClient.prefetchQuery({
|
queryClient.prefetchQuery({
|
||||||
queryKey: ["workspaceMembers", params],
|
queryKey: ["workspaceMembers", params],
|
||||||
queryFn: () => getWorkspaceMembers(params),
|
queryFn: () => getWorkspaceMembers(params),
|
||||||
@@ -21,15 +22,15 @@ export const prefetchWorkspaceMembers = () => {
|
|||||||
|
|
||||||
export const prefetchSpaces = () => {
|
export const prefetchSpaces = () => {
|
||||||
queryClient.prefetchQuery({
|
queryClient.prefetchQuery({
|
||||||
queryKey: ["spaces", { page: 1 }],
|
queryKey: ["spaces", {}],
|
||||||
queryFn: () => getSpaces({ page: 1 }),
|
queryFn: () => getSpaces({}),
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
export const prefetchGroups = () => {
|
export const prefetchGroups = () => {
|
||||||
queryClient.prefetchQuery({
|
queryClient.prefetchQuery({
|
||||||
queryKey: ["groups", { page: 1 }],
|
queryKey: ["groups", {}],
|
||||||
queryFn: () => getGroups({ page: 1 }),
|
queryFn: () => getGroups({}),
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -61,7 +62,21 @@ export const prefetchSsoProviders = () => {
|
|||||||
|
|
||||||
export const prefetchShares = () => {
|
export const prefetchShares = () => {
|
||||||
queryClient.prefetchQuery({
|
queryClient.prefetchQuery({
|
||||||
queryKey: ["share-list", { page: 1 }],
|
queryKey: ["share-list", {}],
|
||||||
queryFn: () => getShares({ page: 1, limit: 100 }),
|
queryFn: () => getShares({}),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export const prefetchApiKeys = () => {
|
||||||
|
queryClient.prefetchQuery({
|
||||||
|
queryKey: ["api-key-list", {}],
|
||||||
|
queryFn: () => getApiKeys({}),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export const prefetchApiKeyManagement = () => {
|
||||||
|
queryClient.prefetchQuery({
|
||||||
|
queryKey: ["api-key-list", { adminView: true }],
|
||||||
|
queryFn: () => getApiKeys({ adminView: true }),
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -12,15 +12,18 @@ import {
|
|||||||
IconLock,
|
IconLock,
|
||||||
IconKey,
|
IconKey,
|
||||||
IconWorld,
|
IconWorld,
|
||||||
|
IconSparkles,
|
||||||
} from "@tabler/icons-react";
|
} from "@tabler/icons-react";
|
||||||
import { Link, useLocation } from "react-router-dom";
|
import { Link, useLocation } from "react-router-dom";
|
||||||
import classes from "./settings.module.css";
|
import classes from "./settings.module.css";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
import { isCloud } from "@/lib/config.ts";
|
||||||
import useUserRole from "@/hooks/use-user-role.tsx";
|
import useUserRole from "@/hooks/use-user-role.tsx";
|
||||||
import { useAtom } from "jotai/index";
|
import { useAtom } from "jotai";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
import {
|
import {
|
||||||
|
prefetchApiKeyManagement,
|
||||||
|
prefetchApiKeys,
|
||||||
prefetchBilling,
|
prefetchBilling,
|
||||||
prefetchGroups,
|
prefetchGroups,
|
||||||
prefetchLicense,
|
prefetchLicense,
|
||||||
@@ -60,6 +63,14 @@ const groupedData: DataGroup[] = [
|
|||||||
icon: IconBrush,
|
icon: IconBrush,
|
||||||
path: "/settings/account/preferences",
|
path: "/settings/account/preferences",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
label: "API keys",
|
||||||
|
icon: IconKey,
|
||||||
|
path: "/settings/account/api-keys",
|
||||||
|
isCloud: true,
|
||||||
|
isEnterprise: true,
|
||||||
|
showDisabledInNonEE: true,
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -90,6 +101,22 @@ const groupedData: DataGroup[] = [
|
|||||||
{ label: "Groups", icon: IconUsersGroup, path: "/settings/groups" },
|
{ label: "Groups", icon: IconUsersGroup, path: "/settings/groups" },
|
||||||
{ label: "Spaces", icon: IconSpaces, path: "/settings/spaces" },
|
{ label: "Spaces", icon: IconSpaces, path: "/settings/spaces" },
|
||||||
{ label: "Public sharing", icon: IconWorld, path: "/settings/sharing" },
|
{ label: "Public sharing", icon: IconWorld, path: "/settings/sharing" },
|
||||||
|
{
|
||||||
|
label: "API management",
|
||||||
|
icon: IconKey,
|
||||||
|
path: "/settings/api-keys",
|
||||||
|
isCloud: true,
|
||||||
|
isEnterprise: true,
|
||||||
|
isAdmin: true,
|
||||||
|
showDisabledInNonEE: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "AI settings",
|
||||||
|
icon: IconSparkles,
|
||||||
|
path: "/settings/ai",
|
||||||
|
isAdmin: true,
|
||||||
|
isSelfhosted: true,
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -195,6 +222,12 @@ export default function SettingsSidebar() {
|
|||||||
case "Public sharing":
|
case "Public sharing":
|
||||||
prefetchHandler = prefetchShares;
|
prefetchHandler = prefetchShares;
|
||||||
break;
|
break;
|
||||||
|
case "API keys":
|
||||||
|
prefetchHandler = prefetchApiKeys;
|
||||||
|
break;
|
||||||
|
case "API management":
|
||||||
|
prefetchHandler = prefetchApiKeyManagement;
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,49 @@
|
|||||||
|
import { useRef, useState, ReactNode } from "react";
|
||||||
|
import { Text, TextProps, Tooltip } from "@mantine/core";
|
||||||
|
|
||||||
|
type AutoTooltipTextProps = TextProps & {
|
||||||
|
children: ReactNode;
|
||||||
|
tooltipLabel?: string;
|
||||||
|
tooltipProps?: Omit<
|
||||||
|
React.ComponentProps<typeof Tooltip>,
|
||||||
|
"children" | "label"
|
||||||
|
>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function AutoTooltipText({
|
||||||
|
children,
|
||||||
|
tooltipLabel,
|
||||||
|
tooltipProps,
|
||||||
|
...textProps
|
||||||
|
}: AutoTooltipTextProps) {
|
||||||
|
const textRef = useRef<HTMLParagraphElement>(null);
|
||||||
|
const [isTruncated, setIsTruncated] = useState(false);
|
||||||
|
|
||||||
|
const handleMouseEnter = () => {
|
||||||
|
const element = textRef.current;
|
||||||
|
if (element) {
|
||||||
|
setIsTruncated(element.scrollWidth > element.clientWidth);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const label = tooltipLabel ?? (typeof children === "string" ? children : "");
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Tooltip
|
||||||
|
label={label}
|
||||||
|
disabled={!isTruncated || !label}
|
||||||
|
multiline
|
||||||
|
withArrow
|
||||||
|
{...tooltipProps}
|
||||||
|
>
|
||||||
|
<Text
|
||||||
|
ref={textRef}
|
||||||
|
truncate
|
||||||
|
onMouseEnter={handleMouseEnter}
|
||||||
|
{...textProps}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</Text>
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import { Avatar } from "@mantine/core";
|
import { Avatar } from "@mantine/core";
|
||||||
import { getAvatarUrl } from "@/lib/config.ts";
|
import { getAvatarUrl } from "@/lib/config.ts";
|
||||||
|
import { AvatarIconType } from "@/features/attachments/types/attachment.types.ts";
|
||||||
|
|
||||||
interface CustomAvatarProps {
|
interface CustomAvatarProps {
|
||||||
avatarUrl: string;
|
avatarUrl: string;
|
||||||
@@ -11,13 +12,15 @@ interface CustomAvatarProps {
|
|||||||
variant?: string;
|
variant?: string;
|
||||||
style?: any;
|
style?: any;
|
||||||
component?: any;
|
component?: any;
|
||||||
|
type?: AvatarIconType;
|
||||||
|
mt?: string | number;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const CustomAvatar = React.forwardRef<
|
export const CustomAvatar = React.forwardRef<
|
||||||
HTMLInputElement,
|
HTMLInputElement,
|
||||||
CustomAvatarProps
|
CustomAvatarProps
|
||||||
>(({ avatarUrl, name, ...props }: CustomAvatarProps, ref) => {
|
>(({ avatarUrl, name, type, ...props }: CustomAvatarProps, ref) => {
|
||||||
const avatarLink = getAvatarUrl(avatarUrl);
|
const avatarLink = getAvatarUrl(avatarUrl, type);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Avatar
|
<Avatar
|
||||||
|
|||||||
@@ -15,6 +15,11 @@ export interface EmojiPickerInterface {
|
|||||||
icon: ReactNode;
|
icon: ReactNode;
|
||||||
removeEmojiAction: () => void;
|
removeEmojiAction: () => void;
|
||||||
readOnly: boolean;
|
readOnly: boolean;
|
||||||
|
actionIconProps?: {
|
||||||
|
size?: string;
|
||||||
|
variant?: string;
|
||||||
|
c?: string;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
function EmojiPicker({
|
function EmojiPicker({
|
||||||
@@ -22,6 +27,7 @@ function EmojiPicker({
|
|||||||
icon,
|
icon,
|
||||||
removeEmojiAction,
|
removeEmojiAction,
|
||||||
readOnly,
|
readOnly,
|
||||||
|
actionIconProps,
|
||||||
}: EmojiPickerInterface) {
|
}: EmojiPickerInterface) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [opened, handlers] = useDisclosure(false);
|
const [opened, handlers] = useDisclosure(false);
|
||||||
@@ -64,7 +70,12 @@ function EmojiPicker({
|
|||||||
closeOnEscape={true}
|
closeOnEscape={true}
|
||||||
>
|
>
|
||||||
<Popover.Target ref={setTarget}>
|
<Popover.Target ref={setTarget}>
|
||||||
<ActionIcon c="gray" variant="transparent" onClick={handlers.toggle}>
|
<ActionIcon
|
||||||
|
c={actionIconProps?.c || "gray"}
|
||||||
|
variant={actionIconProps?.variant || "transparent"}
|
||||||
|
size={actionIconProps?.size}
|
||||||
|
onClick={handlers.toggle}
|
||||||
|
>
|
||||||
{icon}
|
{icon}
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Popover.Target>
|
</Popover.Target>
|
||||||
|
|||||||
@@ -0,0 +1,47 @@
|
|||||||
|
import { Box } from "@mantine/core";
|
||||||
|
import React from "react";
|
||||||
|
|
||||||
|
interface ResponsiveSettingsRowProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ResponsiveSettingsRow({ children }: ResponsiveSettingsRowProps) {
|
||||||
|
return (
|
||||||
|
<Box
|
||||||
|
style={{
|
||||||
|
display: "flex",
|
||||||
|
flexDirection: "row",
|
||||||
|
justifyContent: "space-between",
|
||||||
|
alignItems: "center",
|
||||||
|
gap: "1rem",
|
||||||
|
flexWrap: "wrap",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ResponsiveSettingsContentProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ResponsiveSettingsContent({ children }: ResponsiveSettingsContentProps) {
|
||||||
|
return (
|
||||||
|
<Box style={{ flex: "1 1 300px", minWidth: 0 }}>
|
||||||
|
{children}
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ResponsiveSettingsControlProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ResponsiveSettingsControl({ children }: ResponsiveSettingsControlProps) {
|
||||||
|
return (
|
||||||
|
<Box style={{ flex: "0 0 auto" }}>
|
||||||
|
{children}
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,113 @@
|
|||||||
|
import React, { useMemo } from "react";
|
||||||
|
import { Paper, Text, Group, Stack, Loader, Box } from "@mantine/core";
|
||||||
|
import { IconSparkles, IconFileText } from "@tabler/icons-react";
|
||||||
|
import { Link } from "react-router-dom";
|
||||||
|
import { IAiSearchResponse } from "../services/ai-search-service.ts";
|
||||||
|
import { buildPageUrl } from "@/features/page/page.utils.ts";
|
||||||
|
import { markdownToHtml } from "@docmost/editor-ext";
|
||||||
|
import DOMPurify from "dompurify";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
|
interface AiSearchResultProps {
|
||||||
|
result?: IAiSearchResponse;
|
||||||
|
isLoading?: boolean;
|
||||||
|
streamingAnswer?: string;
|
||||||
|
streamingSources?: any[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export function AiSearchResult({
|
||||||
|
result,
|
||||||
|
isLoading,
|
||||||
|
streamingAnswer = "",
|
||||||
|
streamingSources = [],
|
||||||
|
}: AiSearchResultProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
// Use streaming data if available, otherwise fall back to result
|
||||||
|
const answer = streamingAnswer || result?.answer || "";
|
||||||
|
const sources =
|
||||||
|
streamingSources.length > 0 ? streamingSources : result?.sources || [];
|
||||||
|
|
||||||
|
// Deduplicate sources by pageId, keeping the one with highest similarity
|
||||||
|
const deduplicatedSources = useMemo(() => {
|
||||||
|
if (!sources || sources.length === 0) return [];
|
||||||
|
|
||||||
|
const pageMap = new Map();
|
||||||
|
sources.forEach((source) => {
|
||||||
|
const existing = pageMap.get(source.pageId);
|
||||||
|
if (!existing || source.similarity > existing.similarity) {
|
||||||
|
pageMap.set(source.pageId, source);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return Array.from(pageMap.values());
|
||||||
|
}, [sources]);
|
||||||
|
|
||||||
|
if (isLoading && !answer) {
|
||||||
|
return (
|
||||||
|
<Paper p="md" radius="md" withBorder>
|
||||||
|
<Group>
|
||||||
|
<Loader size="sm" />
|
||||||
|
<Text size="sm">{t("AI is thinking...")}</Text>
|
||||||
|
</Group>
|
||||||
|
</Paper>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!answer && !isLoading) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack gap="md" p="md">
|
||||||
|
<Paper p="md" radius="md" withBorder>
|
||||||
|
<Group gap="xs" mb="sm">
|
||||||
|
<IconSparkles size={20} color="var(--mantine-color-blue-6)" />
|
||||||
|
<Text fw={600} size="sm">
|
||||||
|
{t("AI Answer")}
|
||||||
|
</Text>
|
||||||
|
{isLoading && <Loader size="xs" />}
|
||||||
|
</Group>
|
||||||
|
<div
|
||||||
|
dangerouslySetInnerHTML={{
|
||||||
|
__html: DOMPurify.sanitize(markdownToHtml(answer) as string),
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Paper>
|
||||||
|
|
||||||
|
{deduplicatedSources.length > 0 && (
|
||||||
|
<Stack gap="xs">
|
||||||
|
<Text size="xs" fw={600} c="dimmed">
|
||||||
|
{t("Sources")}
|
||||||
|
</Text>
|
||||||
|
{deduplicatedSources.map((source) => (
|
||||||
|
<Box
|
||||||
|
key={source.pageId}
|
||||||
|
component={Link}
|
||||||
|
to={buildPageUrl(source.spaceSlug, source.slugId, source.title)}
|
||||||
|
style={{
|
||||||
|
textDecoration: "none",
|
||||||
|
color: "inherit",
|
||||||
|
display: "block",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Paper
|
||||||
|
p="xs"
|
||||||
|
radius="sm"
|
||||||
|
withBorder
|
||||||
|
style={{ cursor: "pointer" }}
|
||||||
|
>
|
||||||
|
<Group gap="xs">
|
||||||
|
<IconFileText size={16} />
|
||||||
|
<Text size="sm" truncate>
|
||||||
|
{source.title}
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
</Paper>
|
||||||
|
</Box>
|
||||||
|
))}
|
||||||
|
</Stack>
|
||||||
|
)}
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,69 @@
|
|||||||
|
import { Group, Text, Switch, MantineSize, Title } from "@mantine/core";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
import React, { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { isCloud } from "@/lib/config.ts";
|
||||||
|
import useLicense from "@/ee/hooks/use-license.tsx";
|
||||||
|
|
||||||
|
export default function EnableAiSearch() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
||||||
|
<div>
|
||||||
|
<Text size="md">{t("AI-powered search (Ask AI)")}</Text>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t(
|
||||||
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.",
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<AiSearchToggle />
|
||||||
|
</Group>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
interface AiSearchToggleProps {
|
||||||
|
size?: MantineSize;
|
||||||
|
label?: string;
|
||||||
|
}
|
||||||
|
export function AiSearchToggle({ size, label }: AiSearchToggleProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
|
const [checked, setChecked] = useState(workspace?.settings?.ai?.search);
|
||||||
|
const { hasLicenseKey } = useLicense();
|
||||||
|
|
||||||
|
const hasAccess = isCloud() || (!isCloud() && hasLicenseKey);
|
||||||
|
|
||||||
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = event.currentTarget.checked;
|
||||||
|
try {
|
||||||
|
const updatedWorkspace = await updateWorkspace({ aiSearch: value });
|
||||||
|
setChecked(value);
|
||||||
|
setWorkspace(updatedWorkspace);
|
||||||
|
} catch (err) {
|
||||||
|
notifications.show({
|
||||||
|
message: err?.response?.data?.message,
|
||||||
|
color: "red",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Switch
|
||||||
|
size={size}
|
||||||
|
label={label}
|
||||||
|
labelPosition="left"
|
||||||
|
defaultChecked={checked}
|
||||||
|
onChange={handleChange}
|
||||||
|
disabled={!hasAccess}
|
||||||
|
aria-label={t("Toggle AI search")}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,46 @@
|
|||||||
|
import { useMutation, UseMutationResult } from "@tanstack/react-query";
|
||||||
|
import { useState, useCallback } from "react";
|
||||||
|
import { askAi, IAiSearchResponse } from "@/ee/ai/services/ai-search-service.ts";
|
||||||
|
import { IPageSearchParams } from "@/features/search/types/search.types.ts";
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
interface UseAiSearchResult extends UseMutationResult<IAiSearchResponse, Error, IPageSearchParams> {
|
||||||
|
streamingAnswer: string;
|
||||||
|
streamingSources: any[];
|
||||||
|
clearStreaming: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useAiSearch(): UseAiSearchResult {
|
||||||
|
const [streamingAnswer, setStreamingAnswer] = useState("");
|
||||||
|
const [streamingSources, setStreamingSources] = useState<any[]>([]);
|
||||||
|
|
||||||
|
const clearStreaming = useCallback(() => {
|
||||||
|
setStreamingAnswer("");
|
||||||
|
setStreamingSources([]);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const mutation = useMutation({
|
||||||
|
mutationFn: async (params: IPageSearchParams & { contentType?: string }) => {
|
||||||
|
setStreamingAnswer("");
|
||||||
|
setStreamingSources([]);
|
||||||
|
|
||||||
|
const { contentType, ...apiParams } = params;
|
||||||
|
|
||||||
|
return await askAi(apiParams, (chunk) => {
|
||||||
|
if (chunk.content) {
|
||||||
|
setStreamingAnswer((prev) => prev + chunk.content);
|
||||||
|
}
|
||||||
|
if (chunk.sources) {
|
||||||
|
setStreamingSources(chunk.sources);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
...mutation,
|
||||||
|
streamingAnswer,
|
||||||
|
streamingSources,
|
||||||
|
clearStreaming,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
import { useState, useCallback, useRef } from "react";
|
||||||
|
import { useAiGenerateStreamMutation } from "@/ee/ai/queries/ai-query.ts";
|
||||||
|
import { AiGenerateDto } from "@/ee/ai/types/ai.types.ts";
|
||||||
|
|
||||||
|
export function useAiStream() {
|
||||||
|
const [content, setContent] = useState("");
|
||||||
|
const [isStreaming, setIsStreaming] = useState(false);
|
||||||
|
const abortControllerRef = useRef<AbortController | null>(null);
|
||||||
|
const mutation = useAiGenerateStreamMutation();
|
||||||
|
|
||||||
|
const startStream = useCallback(
|
||||||
|
async (data: AiGenerateDto) => {
|
||||||
|
setContent("");
|
||||||
|
setIsStreaming(true);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const controller = await mutation.mutateAsync({
|
||||||
|
...data,
|
||||||
|
onChunk: (chunk) => {
|
||||||
|
setContent((prev) => prev + chunk.content);
|
||||||
|
},
|
||||||
|
onError: (error) => {
|
||||||
|
console.error("AI stream error:", error);
|
||||||
|
setIsStreaming(false);
|
||||||
|
},
|
||||||
|
onComplete: () => {
|
||||||
|
setIsStreaming(false);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
abortControllerRef.current = controller;
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to start stream:", error);
|
||||||
|
setIsStreaming(false);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[mutation]
|
||||||
|
);
|
||||||
|
|
||||||
|
const stopStream = useCallback(() => {
|
||||||
|
if (abortControllerRef.current) {
|
||||||
|
abortControllerRef.current.abort();
|
||||||
|
abortControllerRef.current = null;
|
||||||
|
setIsStreaming(false);
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const resetContent = useCallback(() => {
|
||||||
|
setContent("");
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return {
|
||||||
|
content,
|
||||||
|
isStreaming,
|
||||||
|
startStream,
|
||||||
|
stopStream,
|
||||||
|
resetContent,
|
||||||
|
isLoading: mutation.isPending,
|
||||||
|
error: mutation.error,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,46 @@
|
|||||||
|
import { Helmet } from "react-helmet-async";
|
||||||
|
import { getAppName, isCloud } from "@/lib/config.ts";
|
||||||
|
import SettingsTitle from "@/components/settings/settings-title.tsx";
|
||||||
|
import React from "react";
|
||||||
|
import useUserRole from "@/hooks/use-user-role.tsx";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import useLicense from "@/ee/hooks/use-license.tsx";
|
||||||
|
import EnableAiSearch from "@/ee/ai/components/enable-ai-search.tsx";
|
||||||
|
import { Alert } from "@mantine/core";
|
||||||
|
import { IconInfoCircle } from "@tabler/icons-react";
|
||||||
|
|
||||||
|
export default function AiSettings() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { isAdmin } = useUserRole();
|
||||||
|
const { hasLicenseKey } = useLicense();
|
||||||
|
|
||||||
|
if (!isAdmin) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const hasAccess = isCloud() || (!isCloud() && hasLicenseKey);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Helmet>
|
||||||
|
<title>AI - {getAppName()}</title>
|
||||||
|
</Helmet>
|
||||||
|
<SettingsTitle title={t("AI settings")} />
|
||||||
|
|
||||||
|
{!hasAccess && (
|
||||||
|
<Alert
|
||||||
|
icon={<IconInfoCircle />}
|
||||||
|
title={t("Enterprise feature")}
|
||||||
|
color="blue"
|
||||||
|
mb="lg"
|
||||||
|
>
|
||||||
|
{t(
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
||||||
|
)}
|
||||||
|
</Alert>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<EnableAiSearch />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
import {
|
||||||
|
useMutation,
|
||||||
|
UseMutationResult,
|
||||||
|
useQuery,
|
||||||
|
UseQueryResult,
|
||||||
|
} from "@tanstack/react-query";
|
||||||
|
import {
|
||||||
|
generateAiContent,
|
||||||
|
generateAiContentStream,
|
||||||
|
} from "@/ee/ai/services/ai-service.ts";
|
||||||
|
import {
|
||||||
|
AiConfigResponse,
|
||||||
|
AiContentResponse,
|
||||||
|
AiGenerateDto,
|
||||||
|
AiStreamChunk,
|
||||||
|
AiStreamError,
|
||||||
|
} from "@/ee/ai/types/ai.types.ts";
|
||||||
|
|
||||||
|
export function useAiGenerateMutation(): UseMutationResult<
|
||||||
|
AiContentResponse,
|
||||||
|
Error,
|
||||||
|
AiGenerateDto
|
||||||
|
> {
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: (data: AiGenerateDto) => generateAiContent(data),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
interface StreamCallbacks {
|
||||||
|
onChunk: (chunk: AiStreamChunk) => void;
|
||||||
|
onError?: (error: AiStreamError) => void;
|
||||||
|
onComplete?: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useAiGenerateStreamMutation(): UseMutationResult<
|
||||||
|
AbortController,
|
||||||
|
Error,
|
||||||
|
AiGenerateDto & StreamCallbacks
|
||||||
|
> {
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: ({ onChunk, onError, onComplete, ...data }) =>
|
||||||
|
generateAiContentStream(data, onChunk, onError, onComplete),
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,83 @@
|
|||||||
|
import api from "@/lib/api-client.ts";
|
||||||
|
import { IPageSearchParams } from "@/features/search/types/search.types.ts";
|
||||||
|
|
||||||
|
export interface IAiSearchResponse {
|
||||||
|
answer: string;
|
||||||
|
sources?: Array<{
|
||||||
|
pageId: string;
|
||||||
|
title: string;
|
||||||
|
slugId: string;
|
||||||
|
spaceSlug: string;
|
||||||
|
similarity: number;
|
||||||
|
distance: number;
|
||||||
|
chunkIndex: number;
|
||||||
|
excerpt: string;
|
||||||
|
}>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function askAi(
|
||||||
|
params: IPageSearchParams,
|
||||||
|
onChunk?: (chunk: { content?: string; sources?: any[] }) => void,
|
||||||
|
): Promise<IAiSearchResponse> {
|
||||||
|
const response = await fetch("/api/ai/ask", {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
credentials: "include",
|
||||||
|
body: JSON.stringify(params),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`HTTP error! status: ${response.status}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const reader = response.body?.getReader();
|
||||||
|
const decoder = new TextDecoder();
|
||||||
|
|
||||||
|
let answer = "";
|
||||||
|
let sources: any[] = [];
|
||||||
|
let buffer = "";
|
||||||
|
|
||||||
|
if (reader) {
|
||||||
|
while (true) {
|
||||||
|
const { done, value } = await reader.read();
|
||||||
|
if (done) break;
|
||||||
|
|
||||||
|
buffer += decoder.decode(value, { stream: true });
|
||||||
|
const lines = buffer.split("\n");
|
||||||
|
|
||||||
|
// Keep the last incomplete line in the buffer
|
||||||
|
buffer = lines.pop() || "";
|
||||||
|
|
||||||
|
for (const line of lines) {
|
||||||
|
if (line.startsWith("data: ")) {
|
||||||
|
const data = line.slice(6);
|
||||||
|
if (data === "[DONE]") break;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const parsed = JSON.parse(data);
|
||||||
|
if (parsed.error) {
|
||||||
|
throw new Error(parsed.error);
|
||||||
|
}
|
||||||
|
if (parsed.content) {
|
||||||
|
answer += parsed.content;
|
||||||
|
onChunk?.({ content: parsed.content });
|
||||||
|
}
|
||||||
|
if (parsed.sources) {
|
||||||
|
sources = parsed.sources;
|
||||||
|
onChunk?.({ sources: parsed.sources });
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
if (e instanceof Error) {
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
// Skip invalid JSON
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return { answer, sources };
|
||||||
|
}
|
||||||
@@ -0,0 +1,89 @@
|
|||||||
|
import api from "@/lib/api-client.ts";
|
||||||
|
import {
|
||||||
|
AiGenerateDto,
|
||||||
|
AiContentResponse,
|
||||||
|
AiStreamChunk,
|
||||||
|
AiStreamError,
|
||||||
|
} from "@/ee/ai/types/ai.types.ts";
|
||||||
|
|
||||||
|
export async function generateAiContent(
|
||||||
|
data: AiGenerateDto,
|
||||||
|
): Promise<AiContentResponse> {
|
||||||
|
const req = await api.post<AiContentResponse>("/ai/generate", data);
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function generateAiContentStream(
|
||||||
|
data: AiGenerateDto,
|
||||||
|
onChunk: (chunk: AiStreamChunk) => void,
|
||||||
|
onError?: (error: AiStreamError) => void,
|
||||||
|
onComplete?: () => void,
|
||||||
|
): Promise<AbortController> {
|
||||||
|
const abortController = new AbortController();
|
||||||
|
try {
|
||||||
|
const response = await fetch("/api/ai/generate/stream", {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
body: JSON.stringify(data),
|
||||||
|
signal: abortController.signal,
|
||||||
|
credentials: "include", // This ensures cookies are sent, matching axios withCredentials
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`HTTP error! status: ${response.status}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const reader = response.body?.getReader();
|
||||||
|
const decoder = new TextDecoder();
|
||||||
|
|
||||||
|
if (!reader) {
|
||||||
|
throw new Error("Response body is not readable");
|
||||||
|
}
|
||||||
|
|
||||||
|
const processStream = async () => {
|
||||||
|
try {
|
||||||
|
while (true) {
|
||||||
|
const { done, value } = await reader.read();
|
||||||
|
if (done) break;
|
||||||
|
|
||||||
|
const chunk = decoder.decode(value, { stream: true });
|
||||||
|
const lines = chunk.split("\n");
|
||||||
|
|
||||||
|
for (const line of lines) {
|
||||||
|
if (line.startsWith("data: ")) {
|
||||||
|
const data = line.slice(6);
|
||||||
|
if (data === "[DONE]") {
|
||||||
|
onComplete?.();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
const parsed = JSON.parse(data);
|
||||||
|
if (parsed.error) {
|
||||||
|
onError?.(parsed);
|
||||||
|
} else {
|
||||||
|
onChunk(parsed);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// Ignore parse errors for incomplete chunks
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
if (error.name !== "AbortError") {
|
||||||
|
onError?.({ error: error.message });
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
reader.releaseLock();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
processStream();
|
||||||
|
} catch (error) {
|
||||||
|
onError?.({ error: error.message });
|
||||||
|
}
|
||||||
|
|
||||||
|
return abortController;
|
||||||
|
}
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
export enum AiAction {
|
||||||
|
IMPROVE_WRITING = "improve_writing",
|
||||||
|
FIX_SPELLING_GRAMMAR = "fix_spelling_grammar",
|
||||||
|
MAKE_SHORTER = "make_shorter",
|
||||||
|
MAKE_LONGER = "make_longer",
|
||||||
|
SIMPLIFY = "simplify",
|
||||||
|
CHANGE_TONE = "change_tone",
|
||||||
|
SUMMARIZE = "summarize",
|
||||||
|
CONTINUE_WRITING = "continue_writing",
|
||||||
|
TRANSLATE = "translate",
|
||||||
|
CUSTOM = "custom",
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface AiGenerateDto {
|
||||||
|
action?: AiAction;
|
||||||
|
content: string;
|
||||||
|
prompt?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface AiContentResponse {
|
||||||
|
content: string;
|
||||||
|
usage?: {
|
||||||
|
promptTokens: number;
|
||||||
|
completionTokens: number;
|
||||||
|
totalTokens: number;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface AiConfigResponse {
|
||||||
|
configured: boolean;
|
||||||
|
availableActions: AiAction[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface AiStreamChunk {
|
||||||
|
content: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface AiStreamError {
|
||||||
|
error: string;
|
||||||
|
}
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
import {
|
||||||
|
Modal,
|
||||||
|
Text,
|
||||||
|
Stack,
|
||||||
|
Alert,
|
||||||
|
Group,
|
||||||
|
Button,
|
||||||
|
TextInput,
|
||||||
|
} from "@mantine/core";
|
||||||
|
import { IconAlertTriangle } from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { IApiKey } from "@/ee/api-key";
|
||||||
|
import CopyTextButton from "@/components/common/copy.tsx";
|
||||||
|
|
||||||
|
interface ApiKeyCreatedModalProps {
|
||||||
|
opened: boolean;
|
||||||
|
onClose: () => void;
|
||||||
|
apiKey: IApiKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ApiKeyCreatedModal({
|
||||||
|
opened,
|
||||||
|
onClose,
|
||||||
|
apiKey,
|
||||||
|
}: ApiKeyCreatedModalProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
if (!apiKey) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
opened={opened}
|
||||||
|
onClose={onClose}
|
||||||
|
title={t("API key created")}
|
||||||
|
size="lg"
|
||||||
|
>
|
||||||
|
<Stack gap="md">
|
||||||
|
<Alert
|
||||||
|
icon={<IconAlertTriangle size={16} />}
|
||||||
|
title={t("Important")}
|
||||||
|
color="red"
|
||||||
|
>
|
||||||
|
{t(
|
||||||
|
"Make sure to copy your API key now. You won't be able to see it again!",
|
||||||
|
)}
|
||||||
|
</Alert>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<Text size="sm" fw={500} mb="xs">
|
||||||
|
{t("API key")}
|
||||||
|
</Text>
|
||||||
|
<Group gap="xs" wrap="nowrap">
|
||||||
|
<TextInput
|
||||||
|
variant="filled"
|
||||||
|
style={{
|
||||||
|
flex: 1,
|
||||||
|
}}
|
||||||
|
value={apiKey.token}
|
||||||
|
readOnly
|
||||||
|
/>
|
||||||
|
|
||||||
|
<CopyTextButton text={apiKey.token} />
|
||||||
|
</Group>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Button fullWidth onClick={onClose} mt="md">
|
||||||
|
{t("I've saved my API key")}
|
||||||
|
</Button>
|
||||||
|
</Stack>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,143 @@
|
|||||||
|
import { ActionIcon, Group, Menu, Table, Text } from "@mantine/core";
|
||||||
|
import { IconDots, IconEdit, IconTrash } from "@tabler/icons-react";
|
||||||
|
import { format } from "date-fns";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { IApiKey } from "@/ee/api-key";
|
||||||
|
import { CustomAvatar } from "@/components/ui/custom-avatar.tsx";
|
||||||
|
import React from "react";
|
||||||
|
import NoTableResults from "@/components/common/no-table-results";
|
||||||
|
|
||||||
|
interface ApiKeyTableProps {
|
||||||
|
apiKeys: IApiKey[];
|
||||||
|
isLoading?: boolean;
|
||||||
|
showUserColumn?: boolean;
|
||||||
|
onUpdate?: (apiKey: IApiKey) => void;
|
||||||
|
onRevoke?: (apiKey: IApiKey) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ApiKeyTable({
|
||||||
|
apiKeys,
|
||||||
|
isLoading,
|
||||||
|
showUserColumn = false,
|
||||||
|
onUpdate,
|
||||||
|
onRevoke,
|
||||||
|
}: ApiKeyTableProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const formatDate = (date: Date | string | null) => {
|
||||||
|
if (!date) return t("Never");
|
||||||
|
return format(new Date(date), "MMM dd, yyyy");
|
||||||
|
};
|
||||||
|
|
||||||
|
const isExpired = (expiresAt: string | null) => {
|
||||||
|
if (!expiresAt) return false;
|
||||||
|
return new Date(expiresAt) < new Date();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Table.ScrollContainer minWidth={500}>
|
||||||
|
<Table highlightOnHover verticalSpacing="sm">
|
||||||
|
<Table.Thead>
|
||||||
|
<Table.Tr>
|
||||||
|
<Table.Th>{t("Name")}</Table.Th>
|
||||||
|
{showUserColumn && <Table.Th>{t("User")}</Table.Th>}
|
||||||
|
<Table.Th>{t("Last used")}</Table.Th>
|
||||||
|
<Table.Th>{t("Expires")}</Table.Th>
|
||||||
|
<Table.Th>{t("Created")}</Table.Th>
|
||||||
|
<Table.Th></Table.Th>
|
||||||
|
</Table.Tr>
|
||||||
|
</Table.Thead>
|
||||||
|
|
||||||
|
<Table.Tbody>
|
||||||
|
{apiKeys && apiKeys.length > 0 ? (
|
||||||
|
apiKeys.map((apiKey: IApiKey, index: number) => (
|
||||||
|
<Table.Tr key={index}>
|
||||||
|
<Table.Td>
|
||||||
|
<Text fz="sm" fw={500}>
|
||||||
|
{apiKey.name}
|
||||||
|
</Text>
|
||||||
|
</Table.Td>
|
||||||
|
|
||||||
|
{showUserColumn && apiKey.creator && (
|
||||||
|
<Table.Td>
|
||||||
|
<Group gap="4" wrap="nowrap">
|
||||||
|
<CustomAvatar
|
||||||
|
avatarUrl={apiKey.creator?.avatarUrl}
|
||||||
|
name={apiKey.creator.name}
|
||||||
|
size="sm"
|
||||||
|
/>
|
||||||
|
<Text fz="sm" lineClamp={1}>
|
||||||
|
{apiKey.creator.name}
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
</Table.Td>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<Table.Td>
|
||||||
|
<Text fz="sm" style={{ whiteSpace: "nowrap" }}>
|
||||||
|
{formatDate(apiKey.lastUsedAt)}
|
||||||
|
</Text>
|
||||||
|
</Table.Td>
|
||||||
|
|
||||||
|
<Table.Td>
|
||||||
|
{apiKey.expiresAt ? (
|
||||||
|
isExpired(apiKey.expiresAt) ? (
|
||||||
|
<Text fz="sm" style={{ whiteSpace: "nowrap" }}>
|
||||||
|
{t("Expired")}
|
||||||
|
</Text>
|
||||||
|
) : (
|
||||||
|
<Text fz="sm" style={{ whiteSpace: "nowrap" }}>
|
||||||
|
{formatDate(apiKey.expiresAt)}
|
||||||
|
</Text>
|
||||||
|
)
|
||||||
|
) : (
|
||||||
|
<Text fz="sm" style={{ whiteSpace: "nowrap" }}>
|
||||||
|
{t("Never")}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
</Table.Td>
|
||||||
|
|
||||||
|
<Table.Td>
|
||||||
|
<Text fz="sm" style={{ whiteSpace: "nowrap" }}>
|
||||||
|
{formatDate(apiKey.createdAt)}
|
||||||
|
</Text>
|
||||||
|
</Table.Td>
|
||||||
|
|
||||||
|
<Table.Td>
|
||||||
|
<Menu position="bottom-end" withinPortal>
|
||||||
|
<Menu.Target>
|
||||||
|
<ActionIcon variant="subtle" color="gray">
|
||||||
|
<IconDots size={16} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Menu.Target>
|
||||||
|
<Menu.Dropdown>
|
||||||
|
{onUpdate && (
|
||||||
|
<Menu.Item
|
||||||
|
leftSection={<IconEdit size={16} />}
|
||||||
|
onClick={() => onUpdate(apiKey)}
|
||||||
|
>
|
||||||
|
{t("Rename")}
|
||||||
|
</Menu.Item>
|
||||||
|
)}
|
||||||
|
{onRevoke && (
|
||||||
|
<Menu.Item
|
||||||
|
leftSection={<IconTrash size={16} />}
|
||||||
|
color="red"
|
||||||
|
onClick={() => onRevoke(apiKey)}
|
||||||
|
>
|
||||||
|
{t("Revoke")}
|
||||||
|
</Menu.Item>
|
||||||
|
)}
|
||||||
|
</Menu.Dropdown>
|
||||||
|
</Menu>
|
||||||
|
</Table.Td>
|
||||||
|
</Table.Tr>
|
||||||
|
))
|
||||||
|
) : (
|
||||||
|
<NoTableResults colSpan={showUserColumn ? 6 : 5} />
|
||||||
|
)}
|
||||||
|
</Table.Tbody>
|
||||||
|
</Table>
|
||||||
|
</Table.ScrollContainer>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,153 @@
|
|||||||
|
import { lazy, Suspense, useState } from "react";
|
||||||
|
import { Modal, TextInput, Button, Group, Stack, Select } from "@mantine/core";
|
||||||
|
import { useForm } from "@mantine/form";
|
||||||
|
import { zodResolver } from "mantine-form-zod-resolver";
|
||||||
|
import { z } from "zod";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { useCreateApiKeyMutation } from "@/ee/api-key/queries/api-key-query";
|
||||||
|
import { IconCalendar } from "@tabler/icons-react";
|
||||||
|
import { IApiKey } from "@/ee/api-key";
|
||||||
|
|
||||||
|
const DateInput = lazy(() =>
|
||||||
|
import("@mantine/dates").then((module) => ({
|
||||||
|
default: module.DateInput,
|
||||||
|
})),
|
||||||
|
);
|
||||||
|
|
||||||
|
interface CreateApiKeyModalProps {
|
||||||
|
opened: boolean;
|
||||||
|
onClose: () => void;
|
||||||
|
onSuccess: (response: IApiKey) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const formSchema = z.object({
|
||||||
|
name: z.string().min(1, "Name is required"),
|
||||||
|
expiresAt: z.string().optional(),
|
||||||
|
});
|
||||||
|
type FormValues = z.infer<typeof formSchema>;
|
||||||
|
|
||||||
|
export function CreateApiKeyModal({
|
||||||
|
opened,
|
||||||
|
onClose,
|
||||||
|
onSuccess,
|
||||||
|
}: CreateApiKeyModalProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [expirationOption, setExpirationOption] = useState<string>("30");
|
||||||
|
const createApiKeyMutation = useCreateApiKeyMutation();
|
||||||
|
|
||||||
|
const form = useForm<FormValues>({
|
||||||
|
validate: zodResolver(formSchema),
|
||||||
|
initialValues: {
|
||||||
|
name: "",
|
||||||
|
expiresAt: "",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const getExpirationDate = (): string | undefined => {
|
||||||
|
if (expirationOption === "never") {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
if (expirationOption === "custom") {
|
||||||
|
return form.values.expiresAt;
|
||||||
|
}
|
||||||
|
const days = parseInt(expirationOption);
|
||||||
|
const date = new Date();
|
||||||
|
date.setDate(date.getDate() + days);
|
||||||
|
return date.toISOString();
|
||||||
|
};
|
||||||
|
|
||||||
|
const getExpirationLabel = (days: number) => {
|
||||||
|
const date = new Date();
|
||||||
|
date.setDate(date.getDate() + days);
|
||||||
|
const formatted = date.toLocaleDateString("en-US", {
|
||||||
|
month: "short",
|
||||||
|
day: "2-digit",
|
||||||
|
year: "numeric",
|
||||||
|
});
|
||||||
|
return `${days} days (${formatted})`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const expirationOptions = [
|
||||||
|
{ value: "30", label: getExpirationLabel(30) },
|
||||||
|
{ value: "60", label: getExpirationLabel(60) },
|
||||||
|
{ value: "90", label: getExpirationLabel(90) },
|
||||||
|
{ value: "365", label: getExpirationLabel(365) },
|
||||||
|
{ value: "custom", label: t("Custom") },
|
||||||
|
{ value: "never", label: t("No expiration") },
|
||||||
|
];
|
||||||
|
|
||||||
|
const handleSubmit = async (data: {
|
||||||
|
name?: string;
|
||||||
|
expiresAt?: string | Date;
|
||||||
|
}) => {
|
||||||
|
const groupData = {
|
||||||
|
name: data.name,
|
||||||
|
expiresAt: getExpirationDate(),
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const createdKey = await createApiKeyMutation.mutateAsync(groupData);
|
||||||
|
onSuccess(createdKey);
|
||||||
|
form.reset();
|
||||||
|
onClose();
|
||||||
|
} catch (err) {
|
||||||
|
//
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleClose = () => {
|
||||||
|
form.reset();
|
||||||
|
setExpirationOption("30");
|
||||||
|
onClose();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
opened={opened}
|
||||||
|
onClose={handleClose}
|
||||||
|
title={t("Create API Key")}
|
||||||
|
size="md"
|
||||||
|
>
|
||||||
|
<form onSubmit={form.onSubmit((values) => handleSubmit(values))}>
|
||||||
|
<Stack gap="md">
|
||||||
|
<TextInput
|
||||||
|
label={t("Name")}
|
||||||
|
placeholder={t("Enter a descriptive name")}
|
||||||
|
data-autofocus
|
||||||
|
required
|
||||||
|
{...form.getInputProps("name")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Select
|
||||||
|
label={t("Expiration")}
|
||||||
|
data={expirationOptions}
|
||||||
|
value={expirationOption}
|
||||||
|
onChange={(value) => setExpirationOption(value || "30")}
|
||||||
|
leftSection={<IconCalendar size={16} />}
|
||||||
|
allowDeselect={false}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{expirationOption === "custom" && (
|
||||||
|
<Suspense fallback={null}>
|
||||||
|
<DateInput
|
||||||
|
label={t("Custom expiration date")}
|
||||||
|
placeholder={t("Select expiration date")}
|
||||||
|
minDate={new Date()}
|
||||||
|
{...form.getInputProps("expiresAt")}
|
||||||
|
/>
|
||||||
|
</Suspense>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<Group justify="flex-end" mt="md">
|
||||||
|
<Button variant="default" onClick={handleClose}>
|
||||||
|
{t("Cancel")}
|
||||||
|
</Button>
|
||||||
|
<Button type="submit" loading={createApiKeyMutation.isPending}>
|
||||||
|
{t("Create")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
</Stack>
|
||||||
|
</form>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,62 @@
|
|||||||
|
import { Modal, Text, Button, Group, Stack } from "@mantine/core";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
|
import { useRevokeApiKeyMutation } from "@/ee/api-key/queries/api-key-query.ts";
|
||||||
|
import { IApiKey } from "@/ee/api-key";
|
||||||
|
|
||||||
|
interface RevokeApiKeyModalProps {
|
||||||
|
opened: boolean;
|
||||||
|
onClose: () => void;
|
||||||
|
apiKey: IApiKey | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function RevokeApiKeyModal({
|
||||||
|
opened,
|
||||||
|
onClose,
|
||||||
|
apiKey,
|
||||||
|
}: RevokeApiKeyModalProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const revokeApiKeyMutation = useRevokeApiKeyMutation();
|
||||||
|
|
||||||
|
const handleRevoke = async () => {
|
||||||
|
if (!apiKey) return;
|
||||||
|
await revokeApiKeyMutation.mutateAsync({
|
||||||
|
apiKeyId: apiKey.id,
|
||||||
|
});
|
||||||
|
onClose();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
opened={opened}
|
||||||
|
onClose={onClose}
|
||||||
|
title={t("Revoke API key")}
|
||||||
|
size="md"
|
||||||
|
>
|
||||||
|
<Stack gap="md">
|
||||||
|
<Text>
|
||||||
|
{t("Are you sure you want to revoke this API key")}{" "}
|
||||||
|
<strong>{apiKey?.name}</strong>?
|
||||||
|
</Text>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t(
|
||||||
|
"This action cannot be undone. Any applications using this API key will stop working.",
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<Group justify="flex-end" mt="md">
|
||||||
|
<Button variant="default" onClick={onClose}>
|
||||||
|
{t("Cancel")}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
color="red"
|
||||||
|
onClick={handleRevoke}
|
||||||
|
loading={revokeApiKeyMutation.isPending}
|
||||||
|
>
|
||||||
|
{t("Revoke")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
</Stack>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,80 @@
|
|||||||
|
import { Modal, TextInput, Button, Group, Stack } from "@mantine/core";
|
||||||
|
import { useForm } from "@mantine/form";
|
||||||
|
import { zodResolver } from "mantine-form-zod-resolver";
|
||||||
|
import { z } from "zod";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { useUpdateApiKeyMutation } from "@/ee/api-key/queries/api-key-query";
|
||||||
|
import { IApiKey } from "@/ee/api-key";
|
||||||
|
import { useEffect } from "react";
|
||||||
|
|
||||||
|
const formSchema = z.object({
|
||||||
|
name: z.string().min(1, "Name is required"),
|
||||||
|
});
|
||||||
|
type FormValues = z.infer<typeof formSchema>;
|
||||||
|
|
||||||
|
interface UpdateApiKeyModalProps {
|
||||||
|
opened: boolean;
|
||||||
|
onClose: () => void;
|
||||||
|
apiKey: IApiKey | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function UpdateApiKeyModal({
|
||||||
|
opened,
|
||||||
|
onClose,
|
||||||
|
apiKey,
|
||||||
|
}: UpdateApiKeyModalProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const updateApiKeyMutation = useUpdateApiKeyMutation();
|
||||||
|
|
||||||
|
const form = useForm<FormValues>({
|
||||||
|
validate: zodResolver(formSchema),
|
||||||
|
initialValues: {
|
||||||
|
name: "",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (opened && apiKey) {
|
||||||
|
form.setValues({ name: apiKey.name });
|
||||||
|
}
|
||||||
|
}, [opened, apiKey]);
|
||||||
|
|
||||||
|
const handleSubmit = async (data: { name?: string }) => {
|
||||||
|
const apiKeyData = {
|
||||||
|
apiKeyId: apiKey.id,
|
||||||
|
name: data.name,
|
||||||
|
};
|
||||||
|
|
||||||
|
await updateApiKeyMutation.mutateAsync(apiKeyData);
|
||||||
|
onClose();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
opened={opened}
|
||||||
|
onClose={onClose}
|
||||||
|
title={t("Update API key")}
|
||||||
|
size="md"
|
||||||
|
>
|
||||||
|
<form onSubmit={form.onSubmit((values) => handleSubmit(values))}>
|
||||||
|
<Stack gap="md">
|
||||||
|
<TextInput
|
||||||
|
label={t("Name")}
|
||||||
|
placeholder={t("Enter a descriptive token name")}
|
||||||
|
required
|
||||||
|
{...form.getInputProps("name")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Group justify="flex-end" mt="md">
|
||||||
|
<Button variant="default" onClick={onClose}>
|
||||||
|
{t("Cancel")}
|
||||||
|
</Button>
|
||||||
|
<Button type="submit" loading={updateApiKeyMutation.isPending}>
|
||||||
|
{t("Update")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
</Stack>
|
||||||
|
</form>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
export { ApiKeyTable } from "./components/api-key-table";
|
||||||
|
export { CreateApiKeyModal } from "./components/create-api-key-modal";
|
||||||
|
export { ApiKeyCreatedModal } from "./components/api-key-created-modal";
|
||||||
|
export { UpdateApiKeyModal } from "./components/update-api-key-modal";
|
||||||
|
export { RevokeApiKeyModal } from "./components/revoke-api-key-modal";
|
||||||
|
|
||||||
|
// Services
|
||||||
|
export * from "./services/api-key-service";
|
||||||
|
|
||||||
|
// Types
|
||||||
|
export * from "./types/api-key.types";
|
||||||
@@ -0,0 +1,106 @@
|
|||||||
|
import React, { useState } from "react";
|
||||||
|
import { Button, Group, Space } from "@mantine/core";
|
||||||
|
import { Helmet } from "react-helmet-async";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import SettingsTitle from "@/components/settings/settings-title";
|
||||||
|
import { getAppName } from "@/lib/config";
|
||||||
|
import { ApiKeyTable } from "@/ee/api-key/components/api-key-table";
|
||||||
|
import { CreateApiKeyModal } from "@/ee/api-key/components/create-api-key-modal";
|
||||||
|
import { ApiKeyCreatedModal } from "@/ee/api-key/components/api-key-created-modal";
|
||||||
|
import { UpdateApiKeyModal } from "@/ee/api-key/components/update-api-key-modal";
|
||||||
|
import { RevokeApiKeyModal } from "@/ee/api-key/components/revoke-api-key-modal";
|
||||||
|
import Paginate from "@/components/common/paginate";
|
||||||
|
import { useCursorPaginate } from "@/hooks/use-cursor-paginate";
|
||||||
|
import { useGetApiKeysQuery } from "@/ee/api-key/queries/api-key-query.ts";
|
||||||
|
import { IApiKey } from "@/ee/api-key";
|
||||||
|
|
||||||
|
export default function UserApiKeys() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { cursor, goNext, goPrev } = useCursorPaginate();
|
||||||
|
const [createModalOpened, setCreateModalOpened] = useState(false);
|
||||||
|
const [createdApiKey, setCreatedApiKey] = useState<IApiKey | null>(null);
|
||||||
|
const [updateModalOpened, setUpdateModalOpened] = useState(false);
|
||||||
|
const [revokeModalOpened, setRevokeModalOpened] = useState(false);
|
||||||
|
const [selectedApiKey, setSelectedApiKey] = useState<IApiKey | null>(null);
|
||||||
|
const { data, isLoading } = useGetApiKeysQuery({ cursor });
|
||||||
|
|
||||||
|
const handleCreateSuccess = (response: IApiKey) => {
|
||||||
|
setCreatedApiKey(response);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleUpdate = (apiKey: IApiKey) => {
|
||||||
|
setSelectedApiKey(apiKey);
|
||||||
|
setUpdateModalOpened(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleRevoke = (apiKey: IApiKey) => {
|
||||||
|
setSelectedApiKey(apiKey);
|
||||||
|
setRevokeModalOpened(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Helmet>
|
||||||
|
<title>
|
||||||
|
{t("API keys")} - {getAppName()}
|
||||||
|
</title>
|
||||||
|
</Helmet>
|
||||||
|
|
||||||
|
<SettingsTitle title={t("API keys")} />
|
||||||
|
|
||||||
|
<Group justify="flex-end" mb="md">
|
||||||
|
<Button onClick={() => setCreateModalOpened(true)}>
|
||||||
|
{t("Create API Key")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
|
||||||
|
<ApiKeyTable
|
||||||
|
apiKeys={data?.items || []}
|
||||||
|
isLoading={isLoading}
|
||||||
|
onUpdate={handleUpdate}
|
||||||
|
onRevoke={handleRevoke}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Space h="md" />
|
||||||
|
|
||||||
|
{data?.items.length > 0 && (
|
||||||
|
<Paginate
|
||||||
|
hasPrevPage={data?.meta?.hasPrevPage}
|
||||||
|
hasNextPage={data?.meta?.hasNextPage}
|
||||||
|
onNext={() => goNext(data?.meta?.nextCursor)}
|
||||||
|
onPrev={goPrev}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<CreateApiKeyModal
|
||||||
|
opened={createModalOpened}
|
||||||
|
onClose={() => setCreateModalOpened(false)}
|
||||||
|
onSuccess={handleCreateSuccess}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ApiKeyCreatedModal
|
||||||
|
opened={!!createdApiKey}
|
||||||
|
onClose={() => setCreatedApiKey(null)}
|
||||||
|
apiKey={createdApiKey}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<UpdateApiKeyModal
|
||||||
|
opened={updateModalOpened}
|
||||||
|
onClose={() => {
|
||||||
|
setUpdateModalOpened(false);
|
||||||
|
setSelectedApiKey(null);
|
||||||
|
}}
|
||||||
|
apiKey={selectedApiKey}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<RevokeApiKeyModal
|
||||||
|
opened={revokeModalOpened}
|
||||||
|
onClose={() => {
|
||||||
|
setRevokeModalOpened(false);
|
||||||
|
setSelectedApiKey(null);
|
||||||
|
}}
|
||||||
|
apiKey={selectedApiKey}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,117 @@
|
|||||||
|
import React, { useState } from "react";
|
||||||
|
import { Button, Group, Space, Text } from "@mantine/core";
|
||||||
|
import { Helmet } from "react-helmet-async";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import SettingsTitle from "@/components/settings/settings-title";
|
||||||
|
import { getAppName } from "@/lib/config";
|
||||||
|
import { ApiKeyTable } from "@/ee/api-key/components/api-key-table";
|
||||||
|
import { CreateApiKeyModal } from "@/ee/api-key/components/create-api-key-modal";
|
||||||
|
import { ApiKeyCreatedModal } from "@/ee/api-key/components/api-key-created-modal";
|
||||||
|
import { UpdateApiKeyModal } from "@/ee/api-key/components/update-api-key-modal";
|
||||||
|
import { RevokeApiKeyModal } from "@/ee/api-key/components/revoke-api-key-modal";
|
||||||
|
import Paginate from "@/components/common/paginate";
|
||||||
|
import { useCursorPaginate } from "@/hooks/use-cursor-paginate";
|
||||||
|
import { useGetApiKeysQuery } from "@/ee/api-key/queries/api-key-query.ts";
|
||||||
|
import { IApiKey } from "@/ee/api-key";
|
||||||
|
import useUserRole from '@/hooks/use-user-role.tsx';
|
||||||
|
|
||||||
|
export default function WorkspaceApiKeys() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { cursor, goNext, goPrev } = useCursorPaginate();
|
||||||
|
const [createModalOpened, setCreateModalOpened] = useState(false);
|
||||||
|
const [createdApiKey, setCreatedApiKey] = useState<IApiKey | null>(null);
|
||||||
|
const [updateModalOpened, setUpdateModalOpened] = useState(false);
|
||||||
|
const [revokeModalOpened, setRevokeModalOpened] = useState(false);
|
||||||
|
const [selectedApiKey, setSelectedApiKey] = useState<IApiKey | null>(null);
|
||||||
|
const { data, isLoading } = useGetApiKeysQuery({ cursor, adminView: true });
|
||||||
|
const { isAdmin } = useUserRole();
|
||||||
|
|
||||||
|
if (!isAdmin) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleCreateSuccess = (response: IApiKey) => {
|
||||||
|
setCreatedApiKey(response);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleUpdate = (apiKey: IApiKey) => {
|
||||||
|
setSelectedApiKey(apiKey);
|
||||||
|
setUpdateModalOpened(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleRevoke = (apiKey: IApiKey) => {
|
||||||
|
setSelectedApiKey(apiKey);
|
||||||
|
setRevokeModalOpened(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Helmet>
|
||||||
|
<title>
|
||||||
|
{t("API management")} - {getAppName()}
|
||||||
|
</title>
|
||||||
|
</Helmet>
|
||||||
|
|
||||||
|
<SettingsTitle title={t("API management")} />
|
||||||
|
|
||||||
|
<Text size="md" c="dimmed" mb="md">
|
||||||
|
{t("Manage API keys for all users in the workspace")}
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<Group justify="flex-end" mb="md">
|
||||||
|
<Button onClick={() => setCreateModalOpened(true)}>
|
||||||
|
{t("Create API Key")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
|
||||||
|
<ApiKeyTable
|
||||||
|
apiKeys={data?.items}
|
||||||
|
isLoading={isLoading}
|
||||||
|
showUserColumn
|
||||||
|
onUpdate={handleUpdate}
|
||||||
|
onRevoke={handleRevoke}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Space h="md" />
|
||||||
|
|
||||||
|
{data?.items.length > 0 && (
|
||||||
|
<Paginate
|
||||||
|
hasPrevPage={data?.meta?.hasPrevPage}
|
||||||
|
hasNextPage={data?.meta?.hasNextPage}
|
||||||
|
onNext={() => goNext(data?.meta?.nextCursor)}
|
||||||
|
onPrev={goPrev}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<CreateApiKeyModal
|
||||||
|
opened={createModalOpened}
|
||||||
|
onClose={() => setCreateModalOpened(false)}
|
||||||
|
onSuccess={handleCreateSuccess}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ApiKeyCreatedModal
|
||||||
|
opened={!!createdApiKey}
|
||||||
|
onClose={() => setCreatedApiKey(null)}
|
||||||
|
apiKey={createdApiKey}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<UpdateApiKeyModal
|
||||||
|
opened={updateModalOpened}
|
||||||
|
onClose={() => {
|
||||||
|
setUpdateModalOpened(false);
|
||||||
|
setSelectedApiKey(null);
|
||||||
|
}}
|
||||||
|
apiKey={selectedApiKey}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<RevokeApiKeyModal
|
||||||
|
opened={revokeModalOpened}
|
||||||
|
onClose={() => {
|
||||||
|
setRevokeModalOpened(false);
|
||||||
|
setSelectedApiKey(null);
|
||||||
|
}}
|
||||||
|
apiKey={selectedApiKey}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,97 @@
|
|||||||
|
import { IPagination, QueryParams } from "@/lib/types.ts";
|
||||||
|
import {
|
||||||
|
keepPreviousData,
|
||||||
|
useMutation,
|
||||||
|
useQuery,
|
||||||
|
useQueryClient,
|
||||||
|
UseQueryResult,
|
||||||
|
} from "@tanstack/react-query";
|
||||||
|
import {
|
||||||
|
createApiKey,
|
||||||
|
getApiKeys,
|
||||||
|
IApiKey,
|
||||||
|
ICreateApiKeyRequest,
|
||||||
|
IUpdateApiKeyRequest,
|
||||||
|
revokeApiKey,
|
||||||
|
updateApiKey,
|
||||||
|
} from "@/ee/api-key";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
|
export function useGetApiKeysQuery(
|
||||||
|
params?: QueryParams,
|
||||||
|
): UseQueryResult<IPagination<IApiKey>, Error> {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: ["api-key-list", params],
|
||||||
|
queryFn: () => getApiKeys(params),
|
||||||
|
staleTime: 0,
|
||||||
|
gcTime: 0,
|
||||||
|
placeholderData: keepPreviousData,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useRevokeApiKeyMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
return useMutation<
|
||||||
|
void,
|
||||||
|
Error,
|
||||||
|
{
|
||||||
|
apiKeyId: string;
|
||||||
|
}
|
||||||
|
>({
|
||||||
|
mutationFn: (data) => revokeApiKey(data),
|
||||||
|
onSuccess: (data, variables) => {
|
||||||
|
notifications.show({ message: t("Revoked successfully") });
|
||||||
|
queryClient.invalidateQueries({
|
||||||
|
predicate: (item) =>
|
||||||
|
["api-key-list"].includes(item.queryKey[0] as string),
|
||||||
|
});
|
||||||
|
},
|
||||||
|
onError: (error) => {
|
||||||
|
const errorMessage = error["response"]?.data?.message;
|
||||||
|
notifications.show({ message: errorMessage, color: "red" });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useCreateApiKeyMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
return useMutation<IApiKey, Error, ICreateApiKeyRequest>({
|
||||||
|
mutationFn: (data) => createApiKey(data),
|
||||||
|
onSuccess: () => {
|
||||||
|
notifications.show({ message: t("API key created successfully") });
|
||||||
|
queryClient.invalidateQueries({
|
||||||
|
predicate: (item) =>
|
||||||
|
["api-key-list"].includes(item.queryKey[0] as string),
|
||||||
|
});
|
||||||
|
},
|
||||||
|
onError: (error) => {
|
||||||
|
const errorMessage = error["response"]?.data?.message;
|
||||||
|
notifications.show({ message: errorMessage, color: "red" });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useUpdateApiKeyMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
return useMutation<IApiKey, Error, IUpdateApiKeyRequest>({
|
||||||
|
mutationFn: (data) => updateApiKey(data),
|
||||||
|
onSuccess: (data, variables) => {
|
||||||
|
notifications.show({ message: t("Updated successfully") });
|
||||||
|
queryClient.invalidateQueries({
|
||||||
|
predicate: (item) =>
|
||||||
|
["api-key-list"].includes(item.queryKey[0] as string),
|
||||||
|
});
|
||||||
|
},
|
||||||
|
onError: (error) => {
|
||||||
|
const errorMessage = error["response"]?.data?.message;
|
||||||
|
notifications.show({ message: errorMessage, color: "red" });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,32 @@
|
|||||||
|
import api from "@/lib/api-client";
|
||||||
|
import {
|
||||||
|
ICreateApiKeyRequest,
|
||||||
|
IApiKey,
|
||||||
|
IUpdateApiKeyRequest,
|
||||||
|
} from "@/ee/api-key/types/api-key.types";
|
||||||
|
import { IPagination, QueryParams } from "@/lib/types.ts";
|
||||||
|
|
||||||
|
export async function getApiKeys(
|
||||||
|
params?: QueryParams,
|
||||||
|
): Promise<IPagination<IApiKey>> {
|
||||||
|
const req = await api.post("/api-keys", { ...params });
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function createApiKey(
|
||||||
|
data: ICreateApiKeyRequest,
|
||||||
|
): Promise<IApiKey> {
|
||||||
|
const req = await api.post<IApiKey>("/api-keys/create", data);
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function updateApiKey(
|
||||||
|
data: IUpdateApiKeyRequest,
|
||||||
|
): Promise<IApiKey> {
|
||||||
|
const req = await api.post<IApiKey>("/api-keys/update", data);
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function revokeApiKey(data: { apiKeyId: string }): Promise<void> {
|
||||||
|
await api.post("/api-keys/revoke", data);
|
||||||
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
import { IUser } from "@/features/user/types/user.types.ts";
|
||||||
|
|
||||||
|
export interface IApiKey {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
token?: string;
|
||||||
|
creatorId: string;
|
||||||
|
workspaceId: string;
|
||||||
|
expiresAt: string | null;
|
||||||
|
lastUsedAt: string | null;
|
||||||
|
createdAt: string;
|
||||||
|
creator: Partial<IUser>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ICreateApiKeyRequest {
|
||||||
|
name: string;
|
||||||
|
expiresAt?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface IUpdateApiKeyRequest {
|
||||||
|
apiKeyId: string;
|
||||||
|
name: string;
|
||||||
|
}
|
||||||
@@ -0,0 +1,124 @@
|
|||||||
|
import React, { useState } from "react";
|
||||||
|
import { Modal, TextInput, PasswordInput, Button, Stack } from "@mantine/core";
|
||||||
|
import { useForm } from "@mantine/form";
|
||||||
|
import { zodResolver } from "mantine-form-zod-resolver";
|
||||||
|
import { z } from "zod";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useNavigate } from "react-router-dom";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { IAuthProvider } from "@/ee/security/types/security.types";
|
||||||
|
import APP_ROUTE from "@/lib/app-route";
|
||||||
|
import { ldapLogin } from "@/ee/security/services/ldap-auth-service";
|
||||||
|
|
||||||
|
const formSchema = z.object({
|
||||||
|
username: z.string().min(1, { message: "Username is required" }),
|
||||||
|
password: z.string().min(1, { message: "Password is required" }),
|
||||||
|
});
|
||||||
|
|
||||||
|
interface LdapLoginModalProps {
|
||||||
|
opened: boolean;
|
||||||
|
onClose: () => void;
|
||||||
|
provider: IAuthProvider;
|
||||||
|
workspaceId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function LdapLoginModal({
|
||||||
|
opened,
|
||||||
|
onClose,
|
||||||
|
provider,
|
||||||
|
workspaceId,
|
||||||
|
}: LdapLoginModalProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const navigate = useNavigate();
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
const [error, setError] = useState<string | null>(null);
|
||||||
|
|
||||||
|
const form = useForm({
|
||||||
|
validate: zodResolver(formSchema),
|
||||||
|
initialValues: {
|
||||||
|
username: "",
|
||||||
|
password: "",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const handleSubmit = async (values: {
|
||||||
|
username: string;
|
||||||
|
password: string;
|
||||||
|
}) => {
|
||||||
|
setIsLoading(true);
|
||||||
|
setError(null);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await ldapLogin({
|
||||||
|
username: values.username,
|
||||||
|
password: values.password,
|
||||||
|
providerId: provider.id,
|
||||||
|
workspaceId,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Handle MFA like the regular login
|
||||||
|
if (response?.userHasMfa) {
|
||||||
|
onClose();
|
||||||
|
navigate(APP_ROUTE.AUTH.MFA_CHALLENGE);
|
||||||
|
} else if (response?.requiresMfaSetup) {
|
||||||
|
onClose();
|
||||||
|
navigate(APP_ROUTE.AUTH.MFA_SETUP_REQUIRED);
|
||||||
|
} else {
|
||||||
|
onClose();
|
||||||
|
navigate(APP_ROUTE.HOME);
|
||||||
|
}
|
||||||
|
} catch (err: any) {
|
||||||
|
setIsLoading(false);
|
||||||
|
const errorMessage =
|
||||||
|
err.response?.data?.message || "Authentication failed";
|
||||||
|
setError(errorMessage);
|
||||||
|
|
||||||
|
notifications.show({
|
||||||
|
message: errorMessage,
|
||||||
|
color: "red",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleClose = () => {
|
||||||
|
form.reset();
|
||||||
|
setError(null);
|
||||||
|
onClose();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
opened={opened}
|
||||||
|
onClose={handleClose}
|
||||||
|
title={`LDAP Login - ${provider.name}`}
|
||||||
|
size="md"
|
||||||
|
>
|
||||||
|
<form onSubmit={form.onSubmit(handleSubmit)}>
|
||||||
|
<Stack>
|
||||||
|
<TextInput
|
||||||
|
id="ldap-username"
|
||||||
|
type="text"
|
||||||
|
label={t("LDAP username")}
|
||||||
|
placeholder="Enter your LDAP username"
|
||||||
|
variant="filled"
|
||||||
|
disabled={isLoading}
|
||||||
|
data-autofocus
|
||||||
|
{...form.getInputProps("username")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<PasswordInput
|
||||||
|
label={t("LDAP password")}
|
||||||
|
placeholder={t("Enter your LDAP password")}
|
||||||
|
variant="filled"
|
||||||
|
disabled={isLoading}
|
||||||
|
{...form.getInputProps("password")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Button type="submit" fullWidth mt="md" loading={isLoading}>
|
||||||
|
{t("Sign in with LDAP")}
|
||||||
|
</Button>
|
||||||
|
</Stack>
|
||||||
|
</form>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,29 +1,62 @@
|
|||||||
|
import { useState } from "react";
|
||||||
import { useWorkspacePublicDataQuery } from "@/features/workspace/queries/workspace-query.ts";
|
import { useWorkspacePublicDataQuery } from "@/features/workspace/queries/workspace-query.ts";
|
||||||
import { Button, Divider, Stack } from "@mantine/core";
|
import { Button, Divider, Stack } from "@mantine/core";
|
||||||
import { IconLock } from "@tabler/icons-react";
|
import { IconLock, IconServer } from "@tabler/icons-react";
|
||||||
import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
||||||
import { buildSsoLoginUrl } from "@/ee/security/sso.utils.ts";
|
import { buildSsoLoginUrl } from "@/ee/security/sso.utils.ts";
|
||||||
import { SSO_PROVIDER } from "@/ee/security/contants.ts";
|
import { SSO_PROVIDER } from "@/ee/security/contants.ts";
|
||||||
import { GoogleIcon } from "@/components/icons/google-icon.tsx";
|
import { GoogleIcon } from "@/components/icons/google-icon.tsx";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
import { isCloud } from "@/lib/config.ts";
|
||||||
|
import { LdapLoginModal } from "@/ee/components/ldap-login-modal.tsx";
|
||||||
|
|
||||||
export default function SsoLogin() {
|
export default function SsoLogin() {
|
||||||
const { data, isLoading } = useWorkspacePublicDataQuery();
|
const { data, isLoading } = useWorkspacePublicDataQuery();
|
||||||
|
const [ldapModalOpened, setLdapModalOpened] = useState(false);
|
||||||
|
const [selectedLdapProvider, setSelectedLdapProvider] = useState<IAuthProvider | null>(null);
|
||||||
|
|
||||||
if (!data?.authProviders || data?.authProviders?.length === 0) {
|
if (!data?.authProviders || data?.authProviders?.length === 0) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleSsoLogin = (provider: IAuthProvider) => {
|
const handleSsoLogin = (provider: IAuthProvider) => {
|
||||||
window.location.href = buildSsoLoginUrl({
|
if (provider.type === SSO_PROVIDER.LDAP) {
|
||||||
providerId: provider.id,
|
// Open modal for LDAP instead of redirecting
|
||||||
type: provider.type,
|
setSelectedLdapProvider(provider);
|
||||||
workspaceId: data.id,
|
setLdapModalOpened(true);
|
||||||
});
|
} else {
|
||||||
|
// Redirect for other SSO providers
|
||||||
|
window.location.href = buildSsoLoginUrl({
|
||||||
|
providerId: provider.id,
|
||||||
|
type: provider.type,
|
||||||
|
workspaceId: data.id,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getProviderIcon = (provider: IAuthProvider) => {
|
||||||
|
if (provider.type === SSO_PROVIDER.GOOGLE) {
|
||||||
|
return <GoogleIcon size={16} />;
|
||||||
|
} else if (provider.type === SSO_PROVIDER.LDAP) {
|
||||||
|
return <IconServer size={16} />;
|
||||||
|
} else {
|
||||||
|
return <IconLock size={16} />;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
{selectedLdapProvider && (
|
||||||
|
<LdapLoginModal
|
||||||
|
opened={ldapModalOpened}
|
||||||
|
onClose={() => {
|
||||||
|
setLdapModalOpened(false);
|
||||||
|
setSelectedLdapProvider(null);
|
||||||
|
}}
|
||||||
|
provider={selectedLdapProvider}
|
||||||
|
workspaceId={data.id}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
{(isCloud() || data.hasLicenseKey) && (
|
{(isCloud() || data.hasLicenseKey) && (
|
||||||
<>
|
<>
|
||||||
<Stack align="stretch" justify="center" gap="sm">
|
<Stack align="stretch" justify="center" gap="sm">
|
||||||
@@ -31,13 +64,7 @@ export default function SsoLogin() {
|
|||||||
<div key={provider.id}>
|
<div key={provider.id}>
|
||||||
<Button
|
<Button
|
||||||
onClick={() => handleSsoLogin(provider)}
|
onClick={() => handleSsoLogin(provider)}
|
||||||
leftSection={
|
leftSection={getProviderIcon(provider)}
|
||||||
provider.type === SSO_PROVIDER.GOOGLE ? (
|
|
||||||
<GoogleIcon size={16} />
|
|
||||||
) : (
|
|
||||||
<IconLock size={16} />
|
|
||||||
)
|
|
||||||
}
|
|
||||||
variant="default"
|
variant="default"
|
||||||
fullWidth
|
fullWidth
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -0,0 +1,12 @@
|
|||||||
|
import { isCloud } from "@/lib/config";
|
||||||
|
import useLicense from "@/ee/hooks/use-license";
|
||||||
|
import usePlan from "@/ee/hooks/use-plan";
|
||||||
|
|
||||||
|
const useEnterpriseAccess = () => {
|
||||||
|
const { hasLicenseKey } = useLicense();
|
||||||
|
const { isBusiness } = usePlan();
|
||||||
|
|
||||||
|
return (isCloud() && isBusiness) || (!isCloud() && hasLicenseKey);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default useEnterpriseAccess;
|
||||||
@@ -11,7 +11,7 @@ export default function OssDetails() {
|
|||||||
withTableBorder
|
withTableBorder
|
||||||
>
|
>
|
||||||
<Table.Caption>
|
<Table.Caption>
|
||||||
To unlock enterprise features like SSO, MFA, Resolve comments, contact sales@docmost.com.
|
To unlock enterprise features like AI, SSO, MFA, Resolve comments, contact sales@docmost.com.
|
||||||
</Table.Caption>
|
</Table.Caption>
|
||||||
<Table.Tbody>
|
<Table.Tbody>
|
||||||
<Table.Tr>
|
<Table.Tr>
|
||||||
|
|||||||
@@ -45,6 +45,7 @@ export function MfaBackupCodeInput({
|
|||||||
onChange={(e) => onChange(e.currentTarget.value.toUpperCase())}
|
onChange={(e) => onChange(e.currentTarget.value.toUpperCase())}
|
||||||
error={error}
|
error={error}
|
||||||
autoFocus
|
autoFocus
|
||||||
|
data-autofocus
|
||||||
maxLength={8}
|
maxLength={8}
|
||||||
styles={{
|
styles={{
|
||||||
input: {
|
input: {
|
||||||
|
|||||||
@@ -25,23 +25,30 @@ import { regenerateBackupCodes } from "@/ee/mfa";
|
|||||||
import { useForm } from "@mantine/form";
|
import { useForm } from "@mantine/form";
|
||||||
import { zodResolver } from "mantine-form-zod-resolver";
|
import { zodResolver } from "mantine-form-zod-resolver";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
import useCurrentUser from "@/features/user/hooks/use-current-user";
|
||||||
|
|
||||||
interface MfaBackupCodesModalProps {
|
interface MfaBackupCodesModalProps {
|
||||||
opened: boolean;
|
opened: boolean;
|
||||||
onClose: () => void;
|
onClose: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
const formSchema = z.object({
|
|
||||||
confirmPassword: z.string().min(1, { message: "Password is required" }),
|
|
||||||
});
|
|
||||||
|
|
||||||
export function MfaBackupCodesModal({
|
export function MfaBackupCodesModal({
|
||||||
opened,
|
opened,
|
||||||
onClose,
|
onClose,
|
||||||
}: MfaBackupCodesModalProps) {
|
}: MfaBackupCodesModalProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
const { data: currentUser } = useCurrentUser();
|
||||||
const [backupCodes, setBackupCodes] = useState<string[]>([]);
|
const [backupCodes, setBackupCodes] = useState<string[]>([]);
|
||||||
const [showNewCodes, setShowNewCodes] = useState(false);
|
const [showNewCodes, setShowNewCodes] = useState(false);
|
||||||
|
const requiresPassword = !currentUser?.user?.hasGeneratedPassword;
|
||||||
|
|
||||||
|
const formSchema = requiresPassword
|
||||||
|
? z.object({
|
||||||
|
confirmPassword: z.string().min(1, { message: "Password is required" }),
|
||||||
|
})
|
||||||
|
: z.object({
|
||||||
|
confirmPassword: z.string().optional(),
|
||||||
|
});
|
||||||
|
|
||||||
const form = useForm({
|
const form = useForm({
|
||||||
validate: zodResolver(formSchema),
|
validate: zodResolver(formSchema),
|
||||||
@@ -51,7 +58,7 @@ export function MfaBackupCodesModal({
|
|||||||
});
|
});
|
||||||
|
|
||||||
const regenerateMutation = useMutation({
|
const regenerateMutation = useMutation({
|
||||||
mutationFn: (data: { confirmPassword: string }) =>
|
mutationFn: (data: { confirmPassword?: string }) =>
|
||||||
regenerateBackupCodes(data),
|
regenerateBackupCodes(data),
|
||||||
onSuccess: (data) => {
|
onSuccess: (data) => {
|
||||||
setBackupCodes(data.backupCodes);
|
setBackupCodes(data.backupCodes);
|
||||||
@@ -73,8 +80,12 @@ export function MfaBackupCodesModal({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const handleRegenerate = (values: { confirmPassword: string }) => {
|
const handleRegenerate = (values: { confirmPassword?: string }) => {
|
||||||
regenerateMutation.mutate(values);
|
// Only send confirmPassword if it's required (non-SSO users)
|
||||||
|
const payload = requiresPassword
|
||||||
|
? { confirmPassword: values.confirmPassword }
|
||||||
|
: {};
|
||||||
|
regenerateMutation.mutate(payload);
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleClose = () => {
|
const handleClose = () => {
|
||||||
@@ -114,12 +125,16 @@ export function MfaBackupCodesModal({
|
|||||||
)}
|
)}
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
<PasswordInput
|
{requiresPassword && (
|
||||||
label={t("Confirm password")}
|
<PasswordInput
|
||||||
placeholder={t("Enter your password")}
|
label={t("Confirm password")}
|
||||||
variant="filled"
|
placeholder={t("Enter your password")}
|
||||||
{...form.getInputProps("confirmPassword")}
|
variant="filled"
|
||||||
/>
|
{...form.getInputProps("confirmPassword")}
|
||||||
|
autoFocus
|
||||||
|
data-autofocus
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
<Button
|
<Button
|
||||||
type="submit"
|
type="submit"
|
||||||
|
|||||||
@@ -97,6 +97,7 @@ export function MfaChallenge() {
|
|||||||
length={6}
|
length={6}
|
||||||
type="number"
|
type="number"
|
||||||
autoFocus
|
autoFocus
|
||||||
|
data-autofocus
|
||||||
oneTimeCode
|
oneTimeCode
|
||||||
{...form.getInputProps("code")}
|
{...form.getInputProps("code")}
|
||||||
error={!!form.errors.code}
|
error={!!form.errors.code}
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ import { notifications } from "@mantine/notifications";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
import { disableMfa } from "@/ee/mfa";
|
import { disableMfa } from "@/ee/mfa";
|
||||||
|
import useCurrentUser from "@/features/user/hooks/use-current-user";
|
||||||
|
|
||||||
interface MfaDisableModalProps {
|
interface MfaDisableModalProps {
|
||||||
opened: boolean;
|
opened: boolean;
|
||||||
@@ -22,16 +23,22 @@ interface MfaDisableModalProps {
|
|||||||
onComplete: () => void;
|
onComplete: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
const formSchema = z.object({
|
|
||||||
confirmPassword: z.string().min(1, { message: "Password is required" }),
|
|
||||||
});
|
|
||||||
|
|
||||||
export function MfaDisableModal({
|
export function MfaDisableModal({
|
||||||
opened,
|
opened,
|
||||||
onClose,
|
onClose,
|
||||||
onComplete,
|
onComplete,
|
||||||
}: MfaDisableModalProps) {
|
}: MfaDisableModalProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
const { data: currentUser } = useCurrentUser();
|
||||||
|
const requiresPassword = !currentUser?.user?.hasGeneratedPassword;
|
||||||
|
|
||||||
|
const formSchema = requiresPassword
|
||||||
|
? z.object({
|
||||||
|
confirmPassword: z.string().min(1, { message: "Password is required" }),
|
||||||
|
})
|
||||||
|
: z.object({
|
||||||
|
confirmPassword: z.string().optional(),
|
||||||
|
});
|
||||||
|
|
||||||
const form = useForm({
|
const form = useForm({
|
||||||
validate: zodResolver(formSchema),
|
validate: zodResolver(formSchema),
|
||||||
@@ -54,8 +61,12 @@ export function MfaDisableModal({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const handleSubmit = async (values: { confirmPassword: string }) => {
|
const handleSubmit = async (values: { confirmPassword?: string }) => {
|
||||||
await disableMutation.mutateAsync(values);
|
// Only send confirmPassword if it's required (non-SSO users)
|
||||||
|
const payload = requiresPassword
|
||||||
|
? { confirmPassword: values.confirmPassword }
|
||||||
|
: {};
|
||||||
|
await disableMutation.mutateAsync(payload);
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleClose = () => {
|
const handleClose = () => {
|
||||||
@@ -85,18 +96,23 @@ export function MfaDisableModal({
|
|||||||
</Text>
|
</Text>
|
||||||
</Alert>
|
</Alert>
|
||||||
|
|
||||||
<Text size="sm">
|
{requiresPassword && (
|
||||||
{t(
|
<>
|
||||||
"Please enter your password to disable two-factor authentication:",
|
<Text size="sm">
|
||||||
)}
|
{t(
|
||||||
</Text>
|
"Please enter your password to disable two-factor authentication:",
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
|
||||||
<PasswordInput
|
<PasswordInput
|
||||||
label={t("Password")}
|
label={t("Password")}
|
||||||
placeholder={t("Enter your password")}
|
placeholder={t("Enter your password")}
|
||||||
{...form.getInputProps("confirmPassword")}
|
{...form.getInputProps("confirmPassword")}
|
||||||
autoFocus
|
autoFocus
|
||||||
/>
|
data-autofocus
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
<Stack gap="sm">
|
<Stack gap="sm">
|
||||||
<Button
|
<Button
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import { MfaDisableModal } from "@/ee/mfa";
|
|||||||
import { MfaBackupCodesModal } from "@/ee/mfa";
|
import { MfaBackupCodesModal } from "@/ee/mfa";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
import { isCloud } from "@/lib/config.ts";
|
||||||
import useLicense from "@/ee/hooks/use-license.tsx";
|
import useLicense from "@/ee/hooks/use-license.tsx";
|
||||||
|
import { ResponsiveSettingsRow, ResponsiveSettingsContent, ResponsiveSettingsControl } from "@/components/ui/responsive-settings-row";
|
||||||
|
|
||||||
export function MfaSettings() {
|
export function MfaSettings() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -53,8 +54,8 @@ export function MfaSettings() {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Group justify="space-between" wrap="nowrap" gap="xl">
|
<ResponsiveSettingsRow>
|
||||||
<div style={{ minWidth: 0, flex: 1 }}>
|
<ResponsiveSettingsContent>
|
||||||
<Text size="md">{t("2-step verification")}</Text>
|
<Text size="md">{t("2-step verification")}</Text>
|
||||||
<Text size="sm" c="dimmed">
|
<Text size="sm" c="dimmed">
|
||||||
{!isMfaEnabled
|
{!isMfaEnabled
|
||||||
@@ -63,44 +64,46 @@ export function MfaSettings() {
|
|||||||
)
|
)
|
||||||
: t("Two-factor authentication is active on your account.")}
|
: t("Two-factor authentication is active on your account.")}
|
||||||
</Text>
|
</Text>
|
||||||
</div>
|
</ResponsiveSettingsContent>
|
||||||
|
|
||||||
{!isMfaEnabled ? (
|
<ResponsiveSettingsControl>
|
||||||
<Tooltip
|
{!isMfaEnabled ? (
|
||||||
label={t("Available in enterprise edition")}
|
<Tooltip
|
||||||
disabled={canUseMfa}
|
label={t("Available in enterprise edition")}
|
||||||
>
|
disabled={canUseMfa}
|
||||||
<Button
|
|
||||||
disabled={!canUseMfa}
|
|
||||||
variant="default"
|
|
||||||
onClick={() => setSetupModalOpen(true)}
|
|
||||||
style={{ whiteSpace: "nowrap" }}
|
|
||||||
>
|
>
|
||||||
{t("Add 2FA method")}
|
<Button
|
||||||
</Button>
|
disabled={!canUseMfa}
|
||||||
</Tooltip>
|
variant="default"
|
||||||
) : (
|
onClick={() => setSetupModalOpen(true)}
|
||||||
<Group gap="sm" wrap="nowrap">
|
style={{ whiteSpace: "nowrap" }}
|
||||||
<Button
|
>
|
||||||
variant="default"
|
{t("Add 2FA method")}
|
||||||
size="sm"
|
</Button>
|
||||||
onClick={() => setBackupCodesModalOpen(true)}
|
</Tooltip>
|
||||||
style={{ whiteSpace: "nowrap" }}
|
) : (
|
||||||
>
|
<Group gap="sm" wrap="nowrap">
|
||||||
{t("Backup codes")} ({mfaStatus?.backupCodesCount || 0})
|
<Button
|
||||||
</Button>
|
variant="default"
|
||||||
<Button
|
size="sm"
|
||||||
variant="default"
|
onClick={() => setBackupCodesModalOpen(true)}
|
||||||
size="sm"
|
style={{ whiteSpace: "nowrap" }}
|
||||||
color="red"
|
>
|
||||||
onClick={() => setDisableModalOpen(true)}
|
{t("Backup codes")} ({mfaStatus?.backupCodesCount || 0})
|
||||||
style={{ whiteSpace: "nowrap" }}
|
</Button>
|
||||||
>
|
<Button
|
||||||
{t("Disable")}
|
variant="default"
|
||||||
</Button>
|
size="sm"
|
||||||
</Group>
|
color="red"
|
||||||
)}
|
onClick={() => setDisableModalOpen(true)}
|
||||||
</Group>
|
style={{ whiteSpace: "nowrap" }}
|
||||||
|
>
|
||||||
|
{t("Disable")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
)}
|
||||||
|
</ResponsiveSettingsControl>
|
||||||
|
</ResponsiveSettingsRow>
|
||||||
|
|
||||||
<MfaSetupModal
|
<MfaSetupModal
|
||||||
opened={setupModalOpen}
|
opened={setupModalOpen}
|
||||||
|
|||||||
@@ -235,6 +235,7 @@ export function MfaSetupModal({
|
|||||||
length={6}
|
length={6}
|
||||||
type="number"
|
type="number"
|
||||||
autoFocus
|
autoFocus
|
||||||
|
data-autofocus
|
||||||
oneTimeCode
|
oneTimeCode
|
||||||
{...form.getInputProps("verificationCode")}
|
{...form.getInputProps("verificationCode")}
|
||||||
styles={{
|
styles={{
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ export async function disableMfa(
|
|||||||
}
|
}
|
||||||
|
|
||||||
export async function regenerateBackupCodes(data: {
|
export async function regenerateBackupCodes(data: {
|
||||||
confirmPassword: string;
|
confirmPassword?: string;
|
||||||
}): Promise<MfaBackupCodesResponse> {
|
}): Promise<MfaBackupCodesResponse> {
|
||||||
const req = await api.post<MfaBackupCodesResponse>(
|
const req = await api.post<MfaBackupCodesResponse>(
|
||||||
"/mfa/generate-backup-codes",
|
"/mfa/generate-backup-codes",
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ export interface MfaEnableResponse {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface MfaDisableRequest {
|
export interface MfaDisableRequest {
|
||||||
confirmPassword: string;
|
confirmPassword?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface MfaBackupCodesResponse {
|
export interface MfaBackupCodesResponse {
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import * as z from "zod";
|
import * as z from "zod";
|
||||||
import { useForm, zodResolver } from "@mantine/form";
|
import { useForm } from "@mantine/form";
|
||||||
|
import { zodResolver } from "mantine-form-zod-resolver";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { Button, Text, TagsInput } from "@mantine/core";
|
import { Button, Text, TagsInput } from "@mantine/core";
|
||||||
@@ -54,9 +55,11 @@ export default function AllowedDomains() {
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div>
|
<div>
|
||||||
<Text size="md">Allowed email domains</Text>
|
<Text size="md">{t("Allowed email domains")}</Text>
|
||||||
<Text size="sm" c="dimmed">
|
<Text size="sm" c="dimmed">
|
||||||
Only users with email addresses from these domains can signup via SSO.
|
{t(
|
||||||
|
"Only users with email addresses from these domains can signup via SSO.",
|
||||||
|
)}
|
||||||
</Text>
|
</Text>
|
||||||
</div>
|
</div>
|
||||||
<form onSubmit={form.onSubmit(handleSubmit)}>
|
<form onSubmit={form.onSubmit(handleSubmit)}>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { useDisclosure } from "@mantine/hooks";
|
import { useDisclosure } from "@mantine/hooks";
|
||||||
import { Button, Menu, Group } from "@mantine/core";
|
import { Button, Menu, Group } from "@mantine/core";
|
||||||
import { IconChevronDown, IconLock } from "@tabler/icons-react";
|
import { IconChevronDown, IconLock, IconServer } from "@tabler/icons-react";
|
||||||
import { useCreateSsoProviderMutation } from "@/ee/security/queries/security-query.ts";
|
import { useCreateSsoProviderMutation } from "@/ee/security/queries/security-query.ts";
|
||||||
import { SSO_PROVIDER } from "@/ee/security/contants.ts";
|
import { SSO_PROVIDER } from "@/ee/security/contants.ts";
|
||||||
import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
||||||
@@ -40,6 +40,19 @@ export default function CreateSsoProvider() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleCreateLDAP = async () => {
|
||||||
|
try {
|
||||||
|
const newProvider = await createSsoProviderMutation.mutateAsync({
|
||||||
|
type: SSO_PROVIDER.LDAP,
|
||||||
|
name: "LDAP",
|
||||||
|
});
|
||||||
|
setProvider(newProvider);
|
||||||
|
open();
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to create LDAP provider", error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<SsoProviderModal opened={opened} onClose={close} provider={provider} />
|
<SsoProviderModal opened={opened} onClose={close} provider={provider} />
|
||||||
@@ -71,6 +84,13 @@ export default function CreateSsoProvider() {
|
|||||||
>
|
>
|
||||||
OpenID (OIDC)
|
OpenID (OIDC)
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
|
|
||||||
|
<Menu.Item
|
||||||
|
onClick={handleCreateLDAP}
|
||||||
|
leftSection={<IconServer size={16} />}
|
||||||
|
>
|
||||||
|
LDAP / Active Directory
|
||||||
|
</Menu.Item>
|
||||||
</Menu.Dropdown>
|
</Menu.Dropdown>
|
||||||
</Menu>
|
</Menu>
|
||||||
</Group>
|
</Group>
|
||||||
|
|||||||
@@ -0,0 +1,88 @@
|
|||||||
|
import { Group, Text, Switch, Tooltip } from "@mantine/core";
|
||||||
|
import { modals } from "@mantine/modals";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
import React, { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import useEnterpriseAccess from "@/ee/hooks/use-enterprise-access.tsx";
|
||||||
|
|
||||||
|
export default function DisablePublicSharing() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
||||||
|
<div>
|
||||||
|
<Text size="md">{t("Disable public sharing")}</Text>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t("Prevent members from sharing pages publicly.")}
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<DisablePublicSharingToggle />
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function DisablePublicSharingToggle() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
|
const [checked, setChecked] = useState(
|
||||||
|
workspace?.settings?.sharing?.disabled === true,
|
||||||
|
);
|
||||||
|
const hasAccess = useEnterpriseAccess();
|
||||||
|
|
||||||
|
const applyChange = async (value: boolean) => {
|
||||||
|
try {
|
||||||
|
const updatedWorkspace = await updateWorkspace({
|
||||||
|
disablePublicSharing: value,
|
||||||
|
});
|
||||||
|
setChecked(value);
|
||||||
|
setWorkspace(updatedWorkspace);
|
||||||
|
} catch (err) {
|
||||||
|
notifications.show({
|
||||||
|
message: err?.response?.data?.message,
|
||||||
|
color: "red",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleChange = (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = event.currentTarget.checked;
|
||||||
|
|
||||||
|
modals.openConfirmModal({
|
||||||
|
title: value ? t("Disable public sharing") : t("Enable public sharing"),
|
||||||
|
children: (
|
||||||
|
<Text size="sm">
|
||||||
|
{value
|
||||||
|
? t(
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this workspace will be deleted.",
|
||||||
|
)
|
||||||
|
: t(
|
||||||
|
"Are you sure you want to enable public sharing? Members will be able to share pages publicly.",
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
),
|
||||||
|
centered: true,
|
||||||
|
labels: { confirm: t("Confirm"), cancel: t("Cancel") },
|
||||||
|
confirmProps: value ? { color: "red" } : {},
|
||||||
|
onConfirm: () => applyChange(value),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Tooltip
|
||||||
|
label={t("Requires an enterprise license")}
|
||||||
|
disabled={hasAccess}
|
||||||
|
refProp="rootRef"
|
||||||
|
>
|
||||||
|
<Switch
|
||||||
|
checked={checked}
|
||||||
|
onChange={handleChange}
|
||||||
|
disabled={!hasAccess}
|
||||||
|
aria-label={t("Toggle public sharing")}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -10,23 +10,18 @@ export default function EnforceMfa() {
|
|||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
||||||
<Title order={4} my="sm">
|
<div>
|
||||||
MFA
|
<Text size="md">{t("Enforce two-factor authentication")}</Text>
|
||||||
</Title>
|
<Text size="sm" c="dimmed">
|
||||||
<Group justify="space-between" wrap="nowrap" gap="xl">
|
{t(
|
||||||
<div>
|
"Once enforced, all members must enable two-factor authentication to access the workspace.",
|
||||||
<Text size="md">{t("Enforce two-factor authentication")}</Text>
|
)}
|
||||||
<Text size="sm" c="dimmed">
|
</Text>
|
||||||
{t(
|
</div>
|
||||||
"Once enforced, all members must enable two-factor authentication to access the workspace.",
|
|
||||||
)}
|
|
||||||
</Text>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<EnforceMfaToggle />
|
<EnforceMfaToggle />
|
||||||
</Group>
|
</Group>
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,84 @@
|
|||||||
|
import { Group, Text, Switch, Tooltip } from "@mantine/core";
|
||||||
|
import { modals } from "@mantine/modals";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
import React, { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { ISpace } from "@/features/space/types/space.types.ts";
|
||||||
|
import { useUpdateSpaceMutation } from "@/features/space/queries/space-query.ts";
|
||||||
|
|
||||||
|
type SpacePublicSharingToggleProps = {
|
||||||
|
space: ISpace;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function SpacePublicSharingToggle({
|
||||||
|
space,
|
||||||
|
}: SpacePublicSharingToggleProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [workspace] = useAtom(workspaceAtom);
|
||||||
|
const workspaceDisabled = workspace?.settings?.sharing?.disabled === true;
|
||||||
|
const [checked, setChecked] = useState(
|
||||||
|
space.settings?.sharing?.disabled === true,
|
||||||
|
);
|
||||||
|
const updateSpaceMutation = useUpdateSpaceMutation();
|
||||||
|
|
||||||
|
const applyChange = async (value: boolean) => {
|
||||||
|
try {
|
||||||
|
await updateSpaceMutation.mutateAsync({
|
||||||
|
spaceId: space.id,
|
||||||
|
disablePublicSharing: value,
|
||||||
|
});
|
||||||
|
setChecked(value);
|
||||||
|
} catch {
|
||||||
|
// error handled by mutation
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleChange = (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = event.currentTarget.checked;
|
||||||
|
|
||||||
|
modals.openConfirmModal({
|
||||||
|
title: value ? t("Disable public sharing") : t("Enable public sharing"),
|
||||||
|
children: (
|
||||||
|
<Text size="sm">
|
||||||
|
{value
|
||||||
|
? t(
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this space will be deleted.",
|
||||||
|
)
|
||||||
|
: t(
|
||||||
|
"Are you sure you want to enable public sharing for this space?",
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
),
|
||||||
|
centered: true,
|
||||||
|
labels: { confirm: t("Confirm"), cancel: t("Cancel") },
|
||||||
|
confirmProps: value ? { color: "red" } : {},
|
||||||
|
onConfirm: () => applyChange(value),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
||||||
|
<div>
|
||||||
|
<Text size="md">{t("Disable public sharing")}</Text>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{workspaceDisabled
|
||||||
|
? t("Public sharing is disabled at the workspace level")
|
||||||
|
: t("Prevent pages in this space from being shared publicly.")}
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
<Tooltip
|
||||||
|
label={t("Public sharing is disabled at the workspace level")}
|
||||||
|
disabled={!workspaceDisabled}
|
||||||
|
refProp="rootRef"
|
||||||
|
>
|
||||||
|
<Switch
|
||||||
|
checked={checked}
|
||||||
|
onChange={handleChange}
|
||||||
|
disabled={workspaceDisabled}
|
||||||
|
aria-label={t("Toggle space public sharing")}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
import { useForm, zodResolver } from "@mantine/form";
|
import { useForm } from "@mantine/form";
|
||||||
|
import { zodResolver } from "mantine-form-zod-resolver";
|
||||||
import { Box, Button, Group, Stack, Switch, TextInput } from "@mantine/core";
|
import { Box, Button, Group, Stack, Switch, TextInput } from "@mantine/core";
|
||||||
import classes from "@/ee/security/components/sso.module.css";
|
import classes from "@/ee/security/components/sso.module.css";
|
||||||
import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
||||||
|
|||||||
@@ -0,0 +1,228 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { z } from "zod";
|
||||||
|
import { useForm } from "@mantine/form";
|
||||||
|
import { zodResolver } from "mantine-form-zod-resolver";
|
||||||
|
import {
|
||||||
|
Box,
|
||||||
|
Button,
|
||||||
|
Group,
|
||||||
|
Stack,
|
||||||
|
Switch,
|
||||||
|
TextInput,
|
||||||
|
Textarea,
|
||||||
|
Text,
|
||||||
|
Accordion,
|
||||||
|
} from "@mantine/core";
|
||||||
|
import classes from "@/ee/security/components/sso.module.css";
|
||||||
|
import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { useUpdateSsoProviderMutation } from "@/ee/security/queries/security-query.ts";
|
||||||
|
import { IconInfoCircle } from "@tabler/icons-react";
|
||||||
|
|
||||||
|
const ssoSchema = z.object({
|
||||||
|
name: z.string().min(1, "Display name is required"),
|
||||||
|
ldapUrl: z.string().url().startsWith("ldap", "Must be an LDAP URL"),
|
||||||
|
ldapBindDn: z.string().min(1, "Bind DN is required"),
|
||||||
|
ldapBindPassword: z.string().min(1, "Bind password is required"),
|
||||||
|
ldapBaseDn: z.string().min(1, "Base DN is required"),
|
||||||
|
ldapUserSearchFilter: z.string().optional(),
|
||||||
|
ldapTlsEnabled: z.boolean(),
|
||||||
|
ldapTlsCaCert: z.string().optional(),
|
||||||
|
isEnabled: z.boolean(),
|
||||||
|
allowSignup: z.boolean(),
|
||||||
|
groupSync: z.boolean(),
|
||||||
|
});
|
||||||
|
|
||||||
|
type SSOFormValues = z.infer<typeof ssoSchema>;
|
||||||
|
|
||||||
|
interface SsoFormProps {
|
||||||
|
provider: IAuthProvider;
|
||||||
|
onClose?: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function SsoLDAPForm({ provider, onClose }: SsoFormProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const updateSsoProviderMutation = useUpdateSsoProviderMutation();
|
||||||
|
|
||||||
|
const form = useForm<SSOFormValues>({
|
||||||
|
initialValues: {
|
||||||
|
name: provider.name || "",
|
||||||
|
ldapUrl: provider.ldapUrl || "",
|
||||||
|
ldapBindDn: provider.ldapBindDn || "",
|
||||||
|
ldapBindPassword: provider.ldapBindPassword || "",
|
||||||
|
ldapBaseDn: provider.ldapBaseDn || "",
|
||||||
|
ldapUserSearchFilter:
|
||||||
|
provider.ldapUserSearchFilter || "(mail={{username}})",
|
||||||
|
ldapTlsEnabled: provider.ldapTlsEnabled || false,
|
||||||
|
ldapTlsCaCert: provider.ldapTlsCaCert || "",
|
||||||
|
isEnabled: provider.isEnabled,
|
||||||
|
allowSignup: provider.allowSignup,
|
||||||
|
groupSync: provider.groupSync || false,
|
||||||
|
},
|
||||||
|
validate: zodResolver(ssoSchema),
|
||||||
|
});
|
||||||
|
|
||||||
|
const handleSubmit = async (values: SSOFormValues) => {
|
||||||
|
const ssoData: Partial<IAuthProvider> = {
|
||||||
|
providerId: provider.id,
|
||||||
|
};
|
||||||
|
if (form.isDirty("name")) {
|
||||||
|
ssoData.name = values.name;
|
||||||
|
}
|
||||||
|
if (form.isDirty("ldapUrl")) {
|
||||||
|
ssoData.ldapUrl = values.ldapUrl;
|
||||||
|
}
|
||||||
|
if (form.isDirty("ldapBindDn")) {
|
||||||
|
ssoData.ldapBindDn = values.ldapBindDn;
|
||||||
|
}
|
||||||
|
if (form.isDirty("ldapBindPassword")) {
|
||||||
|
ssoData.ldapBindPassword = values.ldapBindPassword;
|
||||||
|
}
|
||||||
|
if (form.isDirty("ldapBaseDn")) {
|
||||||
|
ssoData.ldapBaseDn = values.ldapBaseDn;
|
||||||
|
}
|
||||||
|
if (form.isDirty("ldapUserSearchFilter")) {
|
||||||
|
ssoData.ldapUserSearchFilter = values.ldapUserSearchFilter;
|
||||||
|
}
|
||||||
|
if (form.isDirty("ldapTlsEnabled")) {
|
||||||
|
ssoData.ldapTlsEnabled = values.ldapTlsEnabled;
|
||||||
|
}
|
||||||
|
if (form.isDirty("ldapTlsCaCert")) {
|
||||||
|
ssoData.ldapTlsCaCert = values.ldapTlsCaCert;
|
||||||
|
}
|
||||||
|
if (form.isDirty("isEnabled")) {
|
||||||
|
ssoData.isEnabled = values.isEnabled;
|
||||||
|
}
|
||||||
|
if (form.isDirty("allowSignup")) {
|
||||||
|
ssoData.allowSignup = values.allowSignup;
|
||||||
|
}
|
||||||
|
if (form.isDirty("groupSync")) {
|
||||||
|
ssoData.groupSync = values.groupSync;
|
||||||
|
}
|
||||||
|
|
||||||
|
await updateSsoProviderMutation.mutateAsync(ssoData);
|
||||||
|
form.resetDirty();
|
||||||
|
onClose();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box maw={600} mx="auto">
|
||||||
|
<form onSubmit={form.onSubmit(handleSubmit)}>
|
||||||
|
<Stack>
|
||||||
|
<TextInput
|
||||||
|
label={t("Display name")}
|
||||||
|
placeholder="e.g Company LDAP"
|
||||||
|
data-autofocus
|
||||||
|
{...form.getInputProps("name")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<TextInput
|
||||||
|
label="LDAP Server URL"
|
||||||
|
description="URL of your LDAP server"
|
||||||
|
placeholder="ldap://ldap.example.com:389 or ldaps://ldap.example.com:636"
|
||||||
|
{...form.getInputProps("ldapUrl")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<TextInput
|
||||||
|
label="Bind DN"
|
||||||
|
description="Distinguished Name of the service account for searching"
|
||||||
|
placeholder="cn=admin,dc=example,dc=com"
|
||||||
|
{...form.getInputProps("ldapBindDn")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<TextInput
|
||||||
|
label="Bind Password"
|
||||||
|
description="Password for the service account"
|
||||||
|
type="password"
|
||||||
|
placeholder="••••••••"
|
||||||
|
{...form.getInputProps("ldapBindPassword")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<TextInput
|
||||||
|
label="Base DN"
|
||||||
|
description="Base DN where user searches will start"
|
||||||
|
placeholder="ou=users,dc=example,dc=com"
|
||||||
|
{...form.getInputProps("ldapBaseDn")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<TextInput
|
||||||
|
label="User Search Filter"
|
||||||
|
description="LDAP filter to find users. Use {{username}} as placeholder"
|
||||||
|
placeholder="(mail={{username}})"
|
||||||
|
{...form.getInputProps("ldapUserSearchFilter")}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Accordion variant="separated">
|
||||||
|
<Accordion.Item value="advanced">
|
||||||
|
<Accordion.Control icon={<IconInfoCircle size={20} />}>
|
||||||
|
{t("Advanced Settings")}
|
||||||
|
</Accordion.Control>
|
||||||
|
<Accordion.Panel>
|
||||||
|
<Stack>
|
||||||
|
<Group justify="space-between">
|
||||||
|
<div>
|
||||||
|
<Text size="sm">{t("Enable TLS/SSL")}</Text>
|
||||||
|
<Text size="xs" c="dimmed">
|
||||||
|
Use secure connection to LDAP server
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
<Switch
|
||||||
|
className={classes.switch}
|
||||||
|
checked={form.values.ldapTlsEnabled}
|
||||||
|
{...form.getInputProps("ldapTlsEnabled")}
|
||||||
|
/>
|
||||||
|
</Group>
|
||||||
|
|
||||||
|
{form.values.ldapTlsEnabled && (
|
||||||
|
<Textarea
|
||||||
|
label="CA Certificate"
|
||||||
|
description="PEM-encoded CA certificate for TLS verification (optional)"
|
||||||
|
placeholder="-----BEGIN CERTIFICATE-----
|
||||||
|
...
|
||||||
|
-----END CERTIFICATE-----"
|
||||||
|
minRows={4}
|
||||||
|
{...form.getInputProps("ldapTlsCaCert")}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Stack>
|
||||||
|
</Accordion.Panel>
|
||||||
|
</Accordion.Item>
|
||||||
|
</Accordion>
|
||||||
|
|
||||||
|
<Group justify="space-between">
|
||||||
|
<div>{t("Group sync")}</div>
|
||||||
|
<Switch
|
||||||
|
className={classes.switch}
|
||||||
|
checked={form.values.groupSync}
|
||||||
|
{...form.getInputProps("groupSync")}
|
||||||
|
/>
|
||||||
|
</Group>
|
||||||
|
|
||||||
|
<Group justify="space-between">
|
||||||
|
<div>{t("Allow signup")}</div>
|
||||||
|
<Switch
|
||||||
|
className={classes.switch}
|
||||||
|
checked={form.values.allowSignup}
|
||||||
|
{...form.getInputProps("allowSignup")}
|
||||||
|
/>
|
||||||
|
</Group>
|
||||||
|
|
||||||
|
<Group justify="space-between">
|
||||||
|
<div>{t("Enabled")}</div>
|
||||||
|
<Switch
|
||||||
|
className={classes.switch}
|
||||||
|
checked={form.values.isEnabled}
|
||||||
|
{...form.getInputProps("isEnabled")}
|
||||||
|
/>
|
||||||
|
</Group>
|
||||||
|
|
||||||
|
<Group mt="md" justify="flex-end">
|
||||||
|
<Button type="submit" disabled={!form.isDirty()}>
|
||||||
|
{t("Save")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
</Stack>
|
||||||
|
</form>
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -16,6 +16,7 @@ const ssoSchema = z.object({
|
|||||||
oidcClientSecret: z.string().min(1, "Client secret is required"),
|
oidcClientSecret: z.string().min(1, "Client secret is required"),
|
||||||
isEnabled: z.boolean(),
|
isEnabled: z.boolean(),
|
||||||
allowSignup: z.boolean(),
|
allowSignup: z.boolean(),
|
||||||
|
groupSync: z.boolean(),
|
||||||
});
|
});
|
||||||
|
|
||||||
type SSOFormValues = z.infer<typeof ssoSchema>;
|
type SSOFormValues = z.infer<typeof ssoSchema>;
|
||||||
@@ -36,6 +37,7 @@ export function SsoOIDCForm({ provider, onClose }: SsoFormProps) {
|
|||||||
oidcClientSecret: provider.oidcClientSecret || "",
|
oidcClientSecret: provider.oidcClientSecret || "",
|
||||||
isEnabled: provider.isEnabled,
|
isEnabled: provider.isEnabled,
|
||||||
allowSignup: provider.allowSignup,
|
allowSignup: provider.allowSignup,
|
||||||
|
groupSync: provider.groupSync || false,
|
||||||
},
|
},
|
||||||
validate: zodResolver(ssoSchema),
|
validate: zodResolver(ssoSchema),
|
||||||
});
|
});
|
||||||
@@ -67,6 +69,9 @@ export function SsoOIDCForm({ provider, onClose }: SsoFormProps) {
|
|||||||
if (form.isDirty("allowSignup")) {
|
if (form.isDirty("allowSignup")) {
|
||||||
ssoData.allowSignup = values.allowSignup;
|
ssoData.allowSignup = values.allowSignup;
|
||||||
}
|
}
|
||||||
|
if (form.isDirty("groupSync")) {
|
||||||
|
ssoData.groupSync = values.groupSync;
|
||||||
|
}
|
||||||
|
|
||||||
await updateSsoProviderMutation.mutateAsync(ssoData);
|
await updateSsoProviderMutation.mutateAsync(ssoData);
|
||||||
form.resetDirty();
|
form.resetDirty();
|
||||||
@@ -78,7 +83,7 @@ export function SsoOIDCForm({ provider, onClose }: SsoFormProps) {
|
|||||||
<form onSubmit={form.onSubmit(handleSubmit)}>
|
<form onSubmit={form.onSubmit(handleSubmit)}>
|
||||||
<Stack>
|
<Stack>
|
||||||
<TextInput
|
<TextInput
|
||||||
label="Display name"
|
label={t("Display name")}
|
||||||
placeholder="e.g Google SSO"
|
placeholder="e.g Google SSO"
|
||||||
data-autofocus
|
data-autofocus
|
||||||
{...form.getInputProps("name")}
|
{...form.getInputProps("name")}
|
||||||
@@ -110,6 +115,15 @@ export function SsoOIDCForm({ provider, onClose }: SsoFormProps) {
|
|||||||
{...form.getInputProps("oidcClientSecret")}
|
{...form.getInputProps("oidcClientSecret")}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<Group justify="space-between">
|
||||||
|
<div>{t("Group sync")}</div>
|
||||||
|
<Switch
|
||||||
|
className={classes.switch}
|
||||||
|
checked={form.values.groupSync}
|
||||||
|
{...form.getInputProps("groupSync")}
|
||||||
|
/>
|
||||||
|
</Group>
|
||||||
|
|
||||||
<Group justify="space-between">
|
<Group justify="space-between">
|
||||||
<div>{t("Allow signup")}</div>
|
<div>{t("Allow signup")}</div>
|
||||||
<Switch
|
<Switch
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ export default function SsoProviderList() {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data?.length === 0) {
|
if (data?.items.length === 0) {
|
||||||
return <Text c="dimmed">{t("No SSO providers found.")}</Text>;
|
return <Text c="dimmed">{t("No SSO providers found.")}</Text>;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -69,7 +69,7 @@ export default function SsoProviderList() {
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Card shadow="sm" radius="sm">
|
<Card shadow="sm" radius="sm">
|
||||||
<Table.ScrollContainer minWidth={500}>
|
<Table.ScrollContainer minWidth={600}>
|
||||||
<Table verticalSpacing="sm">
|
<Table verticalSpacing="sm">
|
||||||
<Table.Thead>
|
<Table.Thead>
|
||||||
<Table.Tr>
|
<Table.Tr>
|
||||||
@@ -81,7 +81,7 @@ export default function SsoProviderList() {
|
|||||||
</Table.Tr>
|
</Table.Tr>
|
||||||
</Table.Thead>
|
</Table.Thead>
|
||||||
<Table.Tbody>
|
<Table.Tbody>
|
||||||
{data
|
{data?.items
|
||||||
.sort((a, b) => {
|
.sort((a, b) => {
|
||||||
const enabledDiff = Number(b.isEnabled) - Number(a.isEnabled);
|
const enabledDiff = Number(b.isEnabled) - Number(a.isEnabled);
|
||||||
if (enabledDiff !== 0) return enabledDiff;
|
if (enabledDiff !== 0) return enabledDiff;
|
||||||
@@ -104,7 +104,11 @@ export default function SsoProviderList() {
|
|||||||
</Group>
|
</Group>
|
||||||
</Table.Td>
|
</Table.Td>
|
||||||
<Table.Td>
|
<Table.Td>
|
||||||
<Badge color={"gray"} variant="light">
|
<Badge
|
||||||
|
color={"gray"}
|
||||||
|
variant="light"
|
||||||
|
style={{ whiteSpace: "nowrap" }}
|
||||||
|
>
|
||||||
{provider.type.toUpperCase()}
|
{provider.type.toUpperCase()}
|
||||||
</Badge>
|
</Badge>
|
||||||
</Table.Td>
|
</Table.Td>
|
||||||
@@ -133,41 +137,43 @@ export default function SsoProviderList() {
|
|||||||
)}
|
)}
|
||||||
</Table.Td>
|
</Table.Td>
|
||||||
<Table.Td>
|
<Table.Td>
|
||||||
<ActionIcon
|
<Group gap="xs" wrap="nowrap">
|
||||||
variant="subtle"
|
<ActionIcon
|
||||||
color="gray"
|
variant="subtle"
|
||||||
onClick={() => handleEdit(provider)}
|
color="gray"
|
||||||
>
|
onClick={() => handleEdit(provider)}
|
||||||
<IconPencil size={16} />
|
>
|
||||||
</ActionIcon>
|
<IconPencil size={16} />
|
||||||
<Menu
|
</ActionIcon>
|
||||||
transitionProps={{ transition: "pop" }}
|
<Menu
|
||||||
withArrow
|
transitionProps={{ transition: "pop" }}
|
||||||
position="bottom-end"
|
withArrow
|
||||||
withinPortal
|
position="bottom-end"
|
||||||
>
|
withinPortal
|
||||||
<Menu.Target>
|
>
|
||||||
<ActionIcon variant="subtle" color="gray">
|
<Menu.Target>
|
||||||
<IconDots size={16} />
|
<ActionIcon variant="subtle" color="gray">
|
||||||
</ActionIcon>
|
<IconDots size={16} />
|
||||||
</Menu.Target>
|
</ActionIcon>
|
||||||
<Menu.Dropdown>
|
</Menu.Target>
|
||||||
<Menu.Item
|
<Menu.Dropdown>
|
||||||
onClick={() => handleEdit(provider)}
|
<Menu.Item
|
||||||
leftSection={<IconPencil size={16} />}
|
onClick={() => handleEdit(provider)}
|
||||||
>
|
leftSection={<IconPencil size={16} />}
|
||||||
{t("Edit")}
|
>
|
||||||
</Menu.Item>
|
{t("Edit")}
|
||||||
<Menu.Item
|
</Menu.Item>
|
||||||
onClick={() => openDeleteModal(provider.id)}
|
<Menu.Item
|
||||||
leftSection={<IconTrash size={16} />}
|
onClick={() => openDeleteModal(provider.id)}
|
||||||
color="red"
|
leftSection={<IconTrash size={16} />}
|
||||||
disabled={provider.type === SSO_PROVIDER.GOOGLE}
|
color="red"
|
||||||
>
|
disabled={provider.type === SSO_PROVIDER.GOOGLE}
|
||||||
{t("Delete")}
|
>
|
||||||
</Menu.Item>
|
{t("Delete")}
|
||||||
</Menu.Dropdown>
|
</Menu.Item>
|
||||||
</Menu>
|
</Menu.Dropdown>
|
||||||
|
</Menu>
|
||||||
|
</Group>
|
||||||
</Table.Td>
|
</Table.Td>
|
||||||
</Table.Tr>
|
</Table.Tr>
|
||||||
))}
|
))}
|
||||||
|
|||||||
@@ -5,6 +5,8 @@ import { SsoSamlForm } from "@/ee/security/components/sso-saml-form.tsx";
|
|||||||
import { SSO_PROVIDER } from "@/ee/security/contants.ts";
|
import { SSO_PROVIDER } from "@/ee/security/contants.ts";
|
||||||
import { SsoOIDCForm } from "@/ee/security/components/sso-oidc-form.tsx";
|
import { SsoOIDCForm } from "@/ee/security/components/sso-oidc-form.tsx";
|
||||||
import { SsoGoogleForm } from "@/ee/security/components/sso-google-form.tsx";
|
import { SsoGoogleForm } from "@/ee/security/components/sso-google-form.tsx";
|
||||||
|
import { SsoLDAPForm } from "@/ee/security/components/sso-ldap-form.tsx";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
interface SsoModalProps {
|
interface SsoModalProps {
|
||||||
opened: boolean;
|
opened: boolean;
|
||||||
@@ -17,6 +19,8 @@ export default function SsoProviderModal({
|
|||||||
onClose,
|
onClose,
|
||||||
provider,
|
provider,
|
||||||
}: SsoModalProps) {
|
}: SsoModalProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
if (!provider) {
|
if (!provider) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -24,7 +28,9 @@ export default function SsoProviderModal({
|
|||||||
return (
|
return (
|
||||||
<Modal
|
<Modal
|
||||||
opened={opened}
|
opened={opened}
|
||||||
title={`${provider.type.toUpperCase()} Configuration`}
|
title={t("{{ssoProviderType}} configuration", {
|
||||||
|
ssoProviderType: provider.type.toUpperCase(),
|
||||||
|
})}
|
||||||
onClose={onClose}
|
onClose={onClose}
|
||||||
>
|
>
|
||||||
{provider.type === SSO_PROVIDER.SAML && (
|
{provider.type === SSO_PROVIDER.SAML && (
|
||||||
@@ -38,6 +44,10 @@ export default function SsoProviderModal({
|
|||||||
{provider.type === SSO_PROVIDER.GOOGLE && (
|
{provider.type === SSO_PROVIDER.GOOGLE && (
|
||||||
<SsoGoogleForm provider={provider} onClose={onClose} />
|
<SsoGoogleForm provider={provider} onClose={onClose} />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{provider.type === SSO_PROVIDER.LDAP && (
|
||||||
|
<SsoLDAPForm provider={provider} onClose={onClose} />
|
||||||
|
)}
|
||||||
</Modal>
|
</Modal>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
import { useForm, zodResolver } from "@mantine/form";
|
import { useForm } from "@mantine/form";
|
||||||
|
import { zodResolver } from "mantine-form-zod-resolver";
|
||||||
import {
|
import {
|
||||||
Box,
|
Box,
|
||||||
Button,
|
Button,
|
||||||
@@ -26,6 +27,7 @@ const ssoSchema = z.object({
|
|||||||
samlCertificate: z.string().min(1, "SAML Idp Certificate is required"),
|
samlCertificate: z.string().min(1, "SAML Idp Certificate is required"),
|
||||||
isEnabled: z.boolean(),
|
isEnabled: z.boolean(),
|
||||||
allowSignup: z.boolean(),
|
allowSignup: z.boolean(),
|
||||||
|
groupSync: z.boolean(),
|
||||||
});
|
});
|
||||||
|
|
||||||
type SSOFormValues = z.infer<typeof ssoSchema>;
|
type SSOFormValues = z.infer<typeof ssoSchema>;
|
||||||
@@ -45,6 +47,7 @@ export function SsoSamlForm({ provider, onClose }: SsoFormProps) {
|
|||||||
samlCertificate: provider.samlCertificate || "",
|
samlCertificate: provider.samlCertificate || "",
|
||||||
isEnabled: provider.isEnabled,
|
isEnabled: provider.isEnabled,
|
||||||
allowSignup: provider.allowSignup,
|
allowSignup: provider.allowSignup,
|
||||||
|
groupSync: provider.groupSync || false,
|
||||||
},
|
},
|
||||||
validate: zodResolver(ssoSchema),
|
validate: zodResolver(ssoSchema),
|
||||||
});
|
});
|
||||||
@@ -75,6 +78,9 @@ export function SsoSamlForm({ provider, onClose }: SsoFormProps) {
|
|||||||
if (form.isDirty("allowSignup")) {
|
if (form.isDirty("allowSignup")) {
|
||||||
ssoData.allowSignup = values.allowSignup;
|
ssoData.allowSignup = values.allowSignup;
|
||||||
}
|
}
|
||||||
|
if (form.isDirty("groupSync")) {
|
||||||
|
ssoData.groupSync = values.groupSync;
|
||||||
|
}
|
||||||
|
|
||||||
await updateSsoProviderMutation.mutateAsync(ssoData);
|
await updateSsoProviderMutation.mutateAsync(ssoData);
|
||||||
form.resetDirty();
|
form.resetDirty();
|
||||||
@@ -86,7 +92,7 @@ export function SsoSamlForm({ provider, onClose }: SsoFormProps) {
|
|||||||
<form onSubmit={form.onSubmit(handleSubmit)}>
|
<form onSubmit={form.onSubmit(handleSubmit)}>
|
||||||
<Stack>
|
<Stack>
|
||||||
<TextInput
|
<TextInput
|
||||||
label="Display name"
|
label={t("Display name")}
|
||||||
placeholder="e.g Azure Entra"
|
placeholder="e.g Azure Entra"
|
||||||
data-autofocus
|
data-autofocus
|
||||||
{...form.getInputProps("name")}
|
{...form.getInputProps("name")}
|
||||||
@@ -123,6 +129,15 @@ export function SsoSamlForm({ provider, onClose }: SsoFormProps) {
|
|||||||
{...form.getInputProps("samlCertificate")}
|
{...form.getInputProps("samlCertificate")}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<Group justify="space-between">
|
||||||
|
<div>{t("Group sync")}</div>
|
||||||
|
<Switch
|
||||||
|
className={classes.switch}
|
||||||
|
checked={form.values.groupSync}
|
||||||
|
{...form.getInputProps("groupSync")}
|
||||||
|
/>
|
||||||
|
</Group>
|
||||||
|
|
||||||
<Group justify="space-between">
|
<Group justify="space-between">
|
||||||
<div>{t("Allow signup")}</div>
|
<div>{t("Allow signup")}</div>
|
||||||
<Switch
|
<Switch
|
||||||
|
|||||||
@@ -2,4 +2,5 @@ export enum SSO_PROVIDER {
|
|||||||
OIDC = 'oidc',
|
OIDC = 'oidc',
|
||||||
SAML = 'saml',
|
SAML = 'saml',
|
||||||
GOOGLE = 'google',
|
GOOGLE = 'google',
|
||||||
|
LDAP = 'ldap',
|
||||||
}
|
}
|
||||||
@@ -9,15 +9,16 @@ import CreateSsoProvider from "@/ee/security/components/create-sso-provider.tsx"
|
|||||||
import EnforceSso from "@/ee/security/components/enforce-sso.tsx";
|
import EnforceSso from "@/ee/security/components/enforce-sso.tsx";
|
||||||
import AllowedDomains from "@/ee/security/components/allowed-domains.tsx";
|
import AllowedDomains from "@/ee/security/components/allowed-domains.tsx";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import useLicense from "@/ee/hooks/use-license.tsx";
|
|
||||||
import usePlan from "@/ee/hooks/use-plan.tsx";
|
|
||||||
import EnforceMfa from "@/ee/security/components/enforce-mfa.tsx";
|
import EnforceMfa from "@/ee/security/components/enforce-mfa.tsx";
|
||||||
|
import DisablePublicSharing from "@/ee/security/components/disable-public-sharing.tsx";
|
||||||
|
import useEnterpriseAccess from "@/ee/hooks/use-enterprise-access.tsx";
|
||||||
|
import { useIsCloudEE } from "@/hooks/use-is-cloud-ee.tsx";
|
||||||
|
|
||||||
export default function Security() {
|
export default function Security() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { isAdmin } = useUserRole();
|
const { isAdmin } = useUserRole();
|
||||||
const { hasLicenseKey } = useLicense();
|
const hasEnterpriseAccess = useEnterpriseAccess();
|
||||||
const { isBusiness } = usePlan();
|
const isCloudEE = useIsCloudEE();
|
||||||
|
|
||||||
if (!isAdmin) {
|
if (!isAdmin) {
|
||||||
return null;
|
return null;
|
||||||
@@ -30,26 +31,41 @@ export default function Security() {
|
|||||||
</Helmet>
|
</Helmet>
|
||||||
<SettingsTitle title={t("Security")} />
|
<SettingsTitle title={t("Security")} />
|
||||||
|
|
||||||
<AllowedDomains />
|
|
||||||
|
|
||||||
<Divider my="lg" />
|
|
||||||
|
|
||||||
<EnforceMfa />
|
<EnforceMfa />
|
||||||
|
|
||||||
<Divider my="lg" />
|
<Divider my="lg" />
|
||||||
|
|
||||||
|
{(!isCloud() || hasEnterpriseAccess) && (
|
||||||
|
<>
|
||||||
|
<DisablePublicSharing />
|
||||||
|
<Divider my="lg" />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
<Title order={4} my="lg">
|
<Title order={4} my="lg">
|
||||||
Single sign-on (SSO)
|
Single sign-on (SSO)
|
||||||
</Title>
|
</Title>
|
||||||
|
|
||||||
{(isCloud() && isBusiness) || (!isCloud() && hasLicenseKey) ? (
|
{hasEnterpriseAccess && (
|
||||||
<>
|
<>
|
||||||
<EnforceSso />
|
<EnforceSso />
|
||||||
<Divider my="lg" />
|
<Divider my="lg" />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{isCloudEE && (
|
||||||
|
<>
|
||||||
|
<AllowedDomains />
|
||||||
|
<Divider my="lg" />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{hasEnterpriseAccess && (
|
||||||
|
<>
|
||||||
<CreateSsoProvider />
|
<CreateSsoProvider />
|
||||||
<Divider size={0} my="lg" />
|
<Divider size={0} my="lg" />
|
||||||
</>
|
</>
|
||||||
) : null}
|
)}
|
||||||
|
|
||||||
<SsoProviderList />
|
<SsoProviderList />
|
||||||
</>
|
</>
|
||||||
|
|||||||
@@ -13,8 +13,9 @@ import {
|
|||||||
} from "@/ee/security/services/security-service.ts";
|
} from "@/ee/security/services/security-service.ts";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
||||||
|
import { IPagination } from "@/lib/types.ts";
|
||||||
|
|
||||||
export function useGetSsoProviders(): UseQueryResult<IAuthProvider[], Error> {
|
export function useGetSsoProviders(): UseQueryResult<IPagination<IAuthProvider>, Error> {
|
||||||
return useQuery({
|
return useQuery({
|
||||||
queryKey: ["sso-providers"],
|
queryKey: ["sso-providers"],
|
||||||
queryFn: () => getSsoProviders(),
|
queryFn: () => getSsoProviders(),
|
||||||
|
|||||||
@@ -0,0 +1,23 @@
|
|||||||
|
import api from "@/lib/api-client.ts";
|
||||||
|
import { ILoginResponse } from "@/features/auth/types/auth.types.ts";
|
||||||
|
|
||||||
|
interface ILdapLogin {
|
||||||
|
username: string;
|
||||||
|
password: string;
|
||||||
|
providerId: string;
|
||||||
|
workspaceId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function ldapLogin(data: ILdapLogin): Promise<ILoginResponse> {
|
||||||
|
const requestData = {
|
||||||
|
username: data.username,
|
||||||
|
password: data.password,
|
||||||
|
};
|
||||||
|
|
||||||
|
const response = await api.post<ILoginResponse>(
|
||||||
|
`/sso/ldap/${data.providerId}/login`,
|
||||||
|
requestData
|
||||||
|
);
|
||||||
|
|
||||||
|
return response.data;
|
||||||
|
}
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
import api from "@/lib/api-client.ts";
|
import api from "@/lib/api-client.ts";
|
||||||
import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
import { IAuthProvider } from "@/ee/security/types/security.types.ts";
|
||||||
|
import { IPagination } from "@/lib/types.ts";
|
||||||
|
|
||||||
export async function getSsoProviderById(data: {
|
export async function getSsoProviderById(data: {
|
||||||
providerId: string;
|
providerId: string;
|
||||||
@@ -8,8 +9,8 @@ export async function getSsoProviderById(data: {
|
|||||||
return req.data;
|
return req.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function getSsoProviders(): Promise<IAuthProvider[]> {
|
export async function getSsoProviders(): Promise<IPagination<IAuthProvider>> {
|
||||||
const req = await api.post<IAuthProvider[]>("/sso/providers");
|
const req = await api.post<IPagination<IAuthProvider>>("/sso/providers");
|
||||||
return req.data;
|
return req.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -9,8 +9,17 @@ export interface IAuthProvider {
|
|||||||
oidcIssuer: string;
|
oidcIssuer: string;
|
||||||
oidcClientId: string;
|
oidcClientId: string;
|
||||||
oidcClientSecret: string;
|
oidcClientSecret: string;
|
||||||
|
ldapUrl: string;
|
||||||
|
ldapBindDn: string;
|
||||||
|
ldapBindPassword: string;
|
||||||
|
ldapBaseDn: string;
|
||||||
|
ldapUserSearchFilter: string;
|
||||||
|
ldapUserAttributes: any;
|
||||||
|
ldapTlsEnabled: boolean;
|
||||||
|
ldapTlsCaCert: string;
|
||||||
allowSignup: boolean;
|
allowSignup: boolean;
|
||||||
isEnabled: boolean;
|
isEnabled: boolean;
|
||||||
|
groupSync: boolean;
|
||||||
creatorId: string;
|
creatorId: string;
|
||||||
workspaceId: string;
|
workspaceId: string;
|
||||||
createdAt: Date;
|
createdAt: Date;
|
||||||
|
|||||||
@@ -0,0 +1,64 @@
|
|||||||
|
import api from "@/lib/api-client";
|
||||||
|
import {
|
||||||
|
AvatarIconType,
|
||||||
|
IAttachment,
|
||||||
|
} from "@/features/attachments/types/attachment.types.ts";
|
||||||
|
|
||||||
|
export async function uploadIcon(
|
||||||
|
file: File,
|
||||||
|
type: AvatarIconType,
|
||||||
|
spaceId?: string,
|
||||||
|
): Promise<IAttachment> {
|
||||||
|
const formData = new FormData();
|
||||||
|
formData.append("type", type);
|
||||||
|
if (spaceId) {
|
||||||
|
formData.append("spaceId", spaceId);
|
||||||
|
}
|
||||||
|
formData.append("image", file);
|
||||||
|
|
||||||
|
return await api.post("/attachments/upload-image", formData, {
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "multipart/form-data",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function uploadUserAvatar(file: File): Promise<IAttachment> {
|
||||||
|
return uploadIcon(file, AvatarIconType.AVATAR);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function uploadSpaceIcon(
|
||||||
|
file: File,
|
||||||
|
spaceId: string,
|
||||||
|
): Promise<IAttachment> {
|
||||||
|
return uploadIcon(file, AvatarIconType.SPACE_ICON, spaceId);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function uploadWorkspaceIcon(file: File): Promise<IAttachment> {
|
||||||
|
return uploadIcon(file, AvatarIconType.WORKSPACE_ICON);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function removeIcon(
|
||||||
|
type: AvatarIconType,
|
||||||
|
spaceId?: string,
|
||||||
|
): Promise<void> {
|
||||||
|
const payload: { spaceId?: string; type: string } = { type };
|
||||||
|
|
||||||
|
if (spaceId) {
|
||||||
|
payload.spaceId = spaceId;
|
||||||
|
}
|
||||||
|
|
||||||
|
await api.post("/attachments/remove-icon", payload);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function removeAvatar(): Promise<void> {
|
||||||
|
await removeIcon(AvatarIconType.AVATAR);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function removeSpaceIcon(spaceId: string): Promise<void> {
|
||||||
|
await removeIcon(AvatarIconType.SPACE_ICON, spaceId);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function removeWorkspaceIcon(): Promise<void> {
|
||||||
|
await removeIcon(AvatarIconType.WORKSPACE_ICON);
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
export {
|
||||||
|
uploadIcon,
|
||||||
|
uploadUserAvatar,
|
||||||
|
uploadSpaceIcon,
|
||||||
|
uploadWorkspaceIcon,
|
||||||
|
removeAvatar,
|
||||||
|
removeSpaceIcon,
|
||||||
|
removeWorkspaceIcon,
|
||||||
|
} from "./attachment-service.ts";
|
||||||
@@ -0,0 +1,29 @@
|
|||||||
|
export interface IAttachment {
|
||||||
|
id: string;
|
||||||
|
fileName: string;
|
||||||
|
filePath: string;
|
||||||
|
fileSize: number;
|
||||||
|
fileExt: string;
|
||||||
|
mimeType: string;
|
||||||
|
type: string;
|
||||||
|
creatorId: string;
|
||||||
|
pageId: string | null;
|
||||||
|
spaceId: string | null;
|
||||||
|
workspaceId: string;
|
||||||
|
createdAt: string;
|
||||||
|
updatedAt: string;
|
||||||
|
deletedAt: string | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum AvatarIconType {
|
||||||
|
AVATAR = "avatar",
|
||||||
|
SPACE_ICON = "space-icon",
|
||||||
|
WORKSPACE_ICON = "workspace-icon",
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum AttachmentType {
|
||||||
|
AVATAR = "avatar",
|
||||||
|
WORKSPACE_ICON = "workspace-icon",
|
||||||
|
SPACE_ICON = "space-icon",
|
||||||
|
FILE = "file",
|
||||||
|
}
|
||||||
@@ -1,11 +1,13 @@
|
|||||||
import { NodeViewProps, NodeViewWrapper } from "@tiptap/react";
|
import { NodeViewProps, NodeViewWrapper } from "@tiptap/react";
|
||||||
import { Group, Text, Paper, ActionIcon } from "@mantine/core";
|
import { Group, Text, Paper, ActionIcon, Loader } from "@mantine/core";
|
||||||
import { getFileUrl } from "@/lib/config.ts";
|
import { getFileUrl } from "@/lib/config.ts";
|
||||||
import { IconDownload, IconPaperclip } from "@tabler/icons-react";
|
import { IconDownload, 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";
|
||||||
|
|
||||||
export default function AttachmentView(props: NodeViewProps) {
|
export default function AttachmentView(props: NodeViewProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
const { node, selected } = props;
|
const { node, selected } = props;
|
||||||
const { url, name, size } = node.attrs;
|
const { url, name, size } = node.attrs;
|
||||||
const { hovered, ref } = useHover();
|
const { hovered, ref } = useHover();
|
||||||
@@ -20,26 +22,28 @@ export default function AttachmentView(props: NodeViewProps) {
|
|||||||
wrap="nowrap"
|
wrap="nowrap"
|
||||||
h={25}
|
h={25}
|
||||||
>
|
>
|
||||||
<Group justify="space-between" wrap="nowrap">
|
<Group wrap="nowrap" gap="sm" style={{ minWidth: 0, flex: 1 }}>
|
||||||
<IconPaperclip size={20} />
|
{url ? (
|
||||||
|
<IconPaperclip size={20} style={{ flexShrink: 0 }} />
|
||||||
|
) : (
|
||||||
|
<Loader size={20} style={{ flexShrink: 0 }} />
|
||||||
|
)}
|
||||||
|
|
||||||
<Text component="span" size="md" truncate="end">
|
<Text component="span" size="md" truncate="end" style={{ minWidth: 0 }}>
|
||||||
{name}
|
{url ? name : t("Uploading {{name}}", { name })}
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
<Text component="span" size="sm" c="dimmed" inline>
|
<Text component="span" size="sm" c="dimmed" style={{ flexShrink: 0 }}>
|
||||||
{formatBytes(size)}
|
{formatBytes(size)}
|
||||||
</Text>
|
</Text>
|
||||||
</Group>
|
</Group>
|
||||||
|
|
||||||
{selected || hovered ? (
|
{url && (selected || hovered) && (
|
||||||
<a href={getFileUrl(url)} target="_blank">
|
<a href={getFileUrl(url)} target="_blank">
|
||||||
<ActionIcon variant="default" aria-label="download file">
|
<ActionIcon variant="default" aria-label="download file">
|
||||||
<IconDownload size={18} />
|
<IconDownload size={18} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</a>
|
</a>
|
||||||
) : (
|
|
||||||
""
|
|
||||||
)}
|
)}
|
||||||
</Group>
|
</Group>
|
||||||
</Paper>
|
</Paper>
|
||||||
|
|||||||
@@ -1,9 +1,6 @@
|
|||||||
import {
|
import { BubbleMenu, BubbleMenuProps } from "@tiptap/react/menus";
|
||||||
BubbleMenu,
|
import { isNodeSelection, useEditorState } from "@tiptap/react";
|
||||||
BubbleMenuProps,
|
import type { Editor } from "@tiptap/react";
|
||||||
isNodeSelection,
|
|
||||||
useEditor,
|
|
||||||
} from "@tiptap/react";
|
|
||||||
import { FC, useEffect, useRef, useState } from "react";
|
import { FC, useEffect, useRef, useState } from "react";
|
||||||
import {
|
import {
|
||||||
IconBold,
|
IconBold,
|
||||||
@@ -37,7 +34,7 @@ export interface BubbleMenuItem {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type EditorBubbleMenuProps = Omit<BubbleMenuProps, "children" | "editor"> & {
|
type EditorBubbleMenuProps = Omit<BubbleMenuProps, "children" | "editor"> & {
|
||||||
editor: ReturnType<typeof useEditor>;
|
editor: Editor | null;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
||||||
@@ -50,34 +47,52 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
showCommentPopupRef.current = showCommentPopup;
|
showCommentPopupRef.current = showCommentPopup;
|
||||||
}, [showCommentPopup]);
|
}, [showCommentPopup]);
|
||||||
|
|
||||||
|
const editorState = useEditorState({
|
||||||
|
editor: props.editor,
|
||||||
|
selector: (ctx) => {
|
||||||
|
if (!props.editor) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
isBold: ctx.editor.isActive("bold"),
|
||||||
|
isItalic: ctx.editor.isActive("italic"),
|
||||||
|
isUnderline: ctx.editor.isActive("underline"),
|
||||||
|
isStrike: ctx.editor.isActive("strike"),
|
||||||
|
isCode: ctx.editor.isActive("code"),
|
||||||
|
isComment: ctx.editor.isActive("comment"),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const items: BubbleMenuItem[] = [
|
const items: BubbleMenuItem[] = [
|
||||||
{
|
{
|
||||||
name: "Bold",
|
name: "Bold",
|
||||||
isActive: () => props.editor.isActive("bold"),
|
isActive: () => editorState?.isBold,
|
||||||
command: () => props.editor.chain().focus().toggleBold().run(),
|
command: () => props.editor.chain().focus().toggleBold().run(),
|
||||||
icon: IconBold,
|
icon: IconBold,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Italic",
|
name: "Italic",
|
||||||
isActive: () => props.editor.isActive("italic"),
|
isActive: () => editorState?.isItalic,
|
||||||
command: () => props.editor.chain().focus().toggleItalic().run(),
|
command: () => props.editor.chain().focus().toggleItalic().run(),
|
||||||
icon: IconItalic,
|
icon: IconItalic,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Underline",
|
name: "Underline",
|
||||||
isActive: () => props.editor.isActive("underline"),
|
isActive: () => editorState?.isUnderline,
|
||||||
command: () => props.editor.chain().focus().toggleUnderline().run(),
|
command: () => props.editor.chain().focus().toggleUnderline().run(),
|
||||||
icon: IconUnderline,
|
icon: IconUnderline,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Strike",
|
name: "Strike",
|
||||||
isActive: () => props.editor.isActive("strike"),
|
isActive: () => editorState?.isStrike,
|
||||||
command: () => props.editor.chain().focus().toggleStrike().run(),
|
command: () => props.editor.chain().focus().toggleStrike().run(),
|
||||||
icon: IconStrikethrough,
|
icon: IconStrikethrough,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Code",
|
name: "Code",
|
||||||
isActive: () => props.editor.isActive("code"),
|
isActive: () => editorState?.isCode,
|
||||||
command: () => props.editor.chain().focus().toggleCode().run(),
|
command: () => props.editor.chain().focus().toggleCode().run(),
|
||||||
icon: IconCode,
|
icon: IconCode,
|
||||||
},
|
},
|
||||||
@@ -85,7 +100,7 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
|
|
||||||
const commentItem: BubbleMenuItem = {
|
const commentItem: BubbleMenuItem = {
|
||||||
name: "Comment",
|
name: "Comment",
|
||||||
isActive: () => props.editor.isActive("comment"),
|
isActive: () => editorState?.isComment,
|
||||||
command: () => {
|
command: () => {
|
||||||
const commentId = uuid7();
|
const commentId = uuid7();
|
||||||
|
|
||||||
@@ -114,30 +129,25 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
}
|
}
|
||||||
return isTextSelected(editor);
|
return isTextSelected(editor);
|
||||||
},
|
},
|
||||||
tippyOptions: {
|
options: {
|
||||||
moveTransition: "transform 0.15s ease-out",
|
placement: "top",
|
||||||
onCreate: (instance) => {
|
offset: 8,
|
||||||
instance.popper.firstChild?.addEventListener("blur", (event) => {
|
|
||||||
event.preventDefault();
|
|
||||||
event.stopImmediatePropagation();
|
|
||||||
});
|
|
||||||
},
|
|
||||||
onHide: () => {
|
onHide: () => {
|
||||||
setIsNodeSelectorOpen(false);
|
setIsNodeSelectorOpen(false);
|
||||||
setIsTextAlignmentOpen(false);
|
setIsTextAlignmentOpen(false);
|
||||||
setIsColorSelectorOpen(false);
|
|
||||||
setIsLinkSelectorOpen(false);
|
setIsLinkSelectorOpen(false);
|
||||||
|
setIsColorSelectorOpen(false);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const [isNodeSelectorOpen, setIsNodeSelectorOpen] = useState(false);
|
const [isNodeSelectorOpen, setIsNodeSelectorOpen] = useState(false);
|
||||||
const [isTextAlignmentSelectorOpen, setIsTextAlignmentOpen] = useState(false);
|
const [isTextAlignmentSelectorOpen, setIsTextAlignmentOpen] = useState(false);
|
||||||
const [isColorSelectorOpen, setIsColorSelectorOpen] = useState(false);
|
|
||||||
const [isLinkSelectorOpen, setIsLinkSelectorOpen] = useState(false);
|
const [isLinkSelectorOpen, setIsLinkSelectorOpen] = useState(false);
|
||||||
|
const [isColorSelectorOpen, setIsColorSelectorOpen] = useState(false);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<BubbleMenu {...bubbleMenuProps}>
|
<BubbleMenu {...bubbleMenuProps} style={{ zIndex: 200, position: "relative"}}>
|
||||||
<div className={classes.bubbleMenu}>
|
<div className={classes.bubbleMenu}>
|
||||||
<NodeSelector
|
<NodeSelector
|
||||||
editor={props.editor}
|
editor={props.editor}
|
||||||
@@ -145,8 +155,8 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
setIsOpen={() => {
|
setIsOpen={() => {
|
||||||
setIsNodeSelectorOpen(!isNodeSelectorOpen);
|
setIsNodeSelectorOpen(!isNodeSelectorOpen);
|
||||||
setIsTextAlignmentOpen(false);
|
setIsTextAlignmentOpen(false);
|
||||||
setIsColorSelectorOpen(false);
|
|
||||||
setIsLinkSelectorOpen(false);
|
setIsLinkSelectorOpen(false);
|
||||||
|
setIsColorSelectorOpen(false);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
@@ -156,8 +166,8 @@ export const EditorBubbleMenu: FC<EditorBubbleMenuProps> = (props) => {
|
|||||||
setIsOpen={() => {
|
setIsOpen={() => {
|
||||||
setIsTextAlignmentOpen(!isTextAlignmentSelectorOpen);
|
setIsTextAlignmentOpen(!isTextAlignmentSelectorOpen);
|
||||||
setIsNodeSelectorOpen(false);
|
setIsNodeSelectorOpen(false);
|
||||||
setIsColorSelectorOpen(false);
|
|
||||||
setIsLinkSelectorOpen(false);
|
setIsLinkSelectorOpen(false);
|
||||||
|
setIsColorSelectorOpen(false);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Dispatch, FC, SetStateAction } from "react";
|
import React, { Dispatch, FC, SetStateAction } from "react";
|
||||||
import { IconCheck, IconPalette } from "@tabler/icons-react";
|
import { IconCheck, IconChevronDown, IconPalette } from "@tabler/icons-react";
|
||||||
import {
|
import {
|
||||||
ActionIcon,
|
ActionIcon,
|
||||||
Button,
|
Button,
|
||||||
@@ -8,8 +8,12 @@ import {
|
|||||||
ScrollArea,
|
ScrollArea,
|
||||||
Text,
|
Text,
|
||||||
Tooltip,
|
Tooltip,
|
||||||
|
SimpleGrid,
|
||||||
|
Box,
|
||||||
|
Stack,
|
||||||
} from "@mantine/core";
|
} from "@mantine/core";
|
||||||
import { useEditor } from "@tiptap/react";
|
import type { Editor } from "@tiptap/react";
|
||||||
|
import { useEditorState } from "@tiptap/react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
export interface BubbleColorMenuItem {
|
export interface BubbleColorMenuItem {
|
||||||
@@ -18,7 +22,7 @@ export interface BubbleColorMenuItem {
|
|||||||
}
|
}
|
||||||
|
|
||||||
interface ColorSelectorProps {
|
interface ColorSelectorProps {
|
||||||
editor: ReturnType<typeof useEditor>;
|
editor: Editor | null;
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
setIsOpen: Dispatch<SetStateAction<boolean>>;
|
setIsOpen: Dispatch<SetStateAction<boolean>>;
|
||||||
}
|
}
|
||||||
@@ -60,9 +64,12 @@ const TEXT_COLORS: BubbleColorMenuItem[] = [
|
|||||||
name: "Gray",
|
name: "Gray",
|
||||||
color: "#A8A29E",
|
color: "#A8A29E",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
name: "Brown",
|
||||||
|
color: "#92400E",
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
// TODO: handle dark mode
|
|
||||||
const HIGHLIGHT_COLORS: BubbleColorMenuItem[] = [
|
const HIGHLIGHT_COLORS: BubbleColorMenuItem[] = [
|
||||||
{
|
{
|
||||||
name: "Default",
|
name: "Default",
|
||||||
@@ -70,35 +77,39 @@ const HIGHLIGHT_COLORS: BubbleColorMenuItem[] = [
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Blue",
|
name: "Blue",
|
||||||
color: "#c1ecf9",
|
color: "#98d8f2",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Green",
|
name: "Green",
|
||||||
color: "#acf79f",
|
color: "#7edb6c",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Purple",
|
name: "Purple",
|
||||||
color: "#f6f3f8",
|
color: "#e0d6ed",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Red",
|
name: "Red",
|
||||||
color: "#fdebeb",
|
color: "#ffc6c2",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Yellow",
|
name: "Yellow",
|
||||||
color: "#fbf4a2",
|
color: "#faf594",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Orange",
|
name: "Orange",
|
||||||
color: "#faebdd",
|
color: "#f5c8a9",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Pink",
|
name: "Pink",
|
||||||
color: "#faf1f5",
|
color: "#f5cfe0",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Gray",
|
name: "Gray",
|
||||||
color: "#f1f1ef",
|
color: "#dfdfd7",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Brown",
|
||||||
|
color: "#d7c4b7",
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -108,67 +119,180 @@ export const ColorSelector: FC<ColorSelectorProps> = ({
|
|||||||
setIsOpen,
|
setIsOpen,
|
||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const activeColorItem = TEXT_COLORS.find(({ color }) =>
|
|
||||||
editor.isActive("textStyle", { color }),
|
const editorState = useEditorState({
|
||||||
|
editor,
|
||||||
|
selector: (ctx) => {
|
||||||
|
if (!ctx.editor) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const activeColors: Record<string, boolean> = {};
|
||||||
|
TEXT_COLORS.forEach(({ color }) => {
|
||||||
|
activeColors[`text_${color}`] = ctx.editor.isActive("textStyle", {
|
||||||
|
color,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
HIGHLIGHT_COLORS.forEach(({ color }) => {
|
||||||
|
activeColors[`highlight_${color}`] = ctx.editor.isActive("highlight", {
|
||||||
|
color,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return activeColors;
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!editor || !editorState) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const activeColorItem = TEXT_COLORS.find(
|
||||||
|
({ color }) => editorState[`text_${color}`],
|
||||||
);
|
);
|
||||||
|
|
||||||
const activeHighlightItem = HIGHLIGHT_COLORS.find(({ color }) =>
|
const activeHighlightItem = HIGHLIGHT_COLORS.find(
|
||||||
editor.isActive("highlight", { color }),
|
({ color }) => editorState[`highlight_${color}`],
|
||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Popover width={200} opened={isOpen} withArrow>
|
<Popover width={220} opened={isOpen} withArrow>
|
||||||
<Popover.Target>
|
<Popover.Target>
|
||||||
<Tooltip label={t("Text color")} withArrow>
|
<Tooltip label={t("Text color")} withArrow>
|
||||||
<ActionIcon
|
<Button
|
||||||
variant="default"
|
variant="default"
|
||||||
size="lg"
|
|
||||||
radius="0"
|
radius="0"
|
||||||
style={{
|
rightSection={<IconChevronDown size={16} />}
|
||||||
border: "none",
|
|
||||||
color: activeColorItem?.color,
|
|
||||||
}}
|
|
||||||
onClick={() => setIsOpen(!isOpen)}
|
onClick={() => setIsOpen(!isOpen)}
|
||||||
|
data-text-color={activeColorItem?.color || ""}
|
||||||
|
data-highlight-color={activeHighlightItem?.color || ""}
|
||||||
|
className="color-selector-trigger"
|
||||||
|
style={{
|
||||||
|
height: "34px",
|
||||||
|
border: "none",
|
||||||
|
fontWeight: 500,
|
||||||
|
fontSize: rem(16),
|
||||||
|
paddingLeft: rem(8),
|
||||||
|
paddingRight: rem(4),
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
<IconPalette size={16} stroke={2} />
|
A
|
||||||
</ActionIcon>
|
</Button>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</Popover.Target>
|
</Popover.Target>
|
||||||
|
|
||||||
<Popover.Dropdown>
|
<Popover.Dropdown>
|
||||||
{/* make mah responsive */}
|
|
||||||
<ScrollArea.Autosize type="scroll" mah="400">
|
<ScrollArea.Autosize type="scroll" mah="400">
|
||||||
<Text span c="dimmed" tt="uppercase" inherit>
|
<Stack gap="md">
|
||||||
{t("Color")}
|
<Box>
|
||||||
</Text>
|
<Text size="sm" fw={600} mb="xs">
|
||||||
|
{t("Text color")}
|
||||||
|
</Text>
|
||||||
|
<SimpleGrid cols={5} spacing="xs">
|
||||||
|
{TEXT_COLORS.map(({ name, color }, index) => (
|
||||||
|
<Tooltip key={index} label={t(name)} withArrow>
|
||||||
|
<Box
|
||||||
|
onClick={() => {
|
||||||
|
if (name === "Default") {
|
||||||
|
editor.commands.unsetColor();
|
||||||
|
} else {
|
||||||
|
editor
|
||||||
|
.chain()
|
||||||
|
.focus()
|
||||||
|
.setColor(color || "")
|
||||||
|
.run();
|
||||||
|
}
|
||||||
|
setIsOpen(false);
|
||||||
|
}}
|
||||||
|
style={{
|
||||||
|
width: rem(28),
|
||||||
|
height: rem(28),
|
||||||
|
borderRadius: rem(6),
|
||||||
|
border: editorState[`text_${color}`]
|
||||||
|
? "2px solid var(--mantine-color-gray-8)"
|
||||||
|
: "1px solid var(--mantine-color-gray-4)",
|
||||||
|
cursor: "pointer",
|
||||||
|
position: "relative",
|
||||||
|
display: "flex",
|
||||||
|
alignItems: "center",
|
||||||
|
justifyContent: "center",
|
||||||
|
fontSize: rem(16),
|
||||||
|
fontWeight: 600,
|
||||||
|
color: color || "var(--mantine-color-gray-8)",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
A
|
||||||
|
</Box>
|
||||||
|
</Tooltip>
|
||||||
|
))}
|
||||||
|
</SimpleGrid>
|
||||||
|
</Box>
|
||||||
|
|
||||||
<Button.Group orientation="vertical">
|
<Box>
|
||||||
{TEXT_COLORS.map(({ name, color }, index) => (
|
<Text size="sm" fw={600} mb="xs">
|
||||||
<Button
|
{t("Highlight color")}
|
||||||
key={index}
|
</Text>
|
||||||
variant="default"
|
<SimpleGrid cols={5} spacing="xs">
|
||||||
leftSection={<span style={{ color }}>A</span>}
|
{HIGHLIGHT_COLORS.map(({ name, color }, index) => (
|
||||||
justify="left"
|
<Tooltip key={index} label={t(name)} withArrow>
|
||||||
fullWidth
|
<Box
|
||||||
rightSection={
|
onClick={() => {
|
||||||
editor.isActive("textStyle", { color }) && (
|
if (name === "Default") {
|
||||||
<IconCheck style={{ width: rem(16) }} />
|
editor.commands.unsetHighlight();
|
||||||
)
|
} else {
|
||||||
}
|
editor
|
||||||
onClick={() => {
|
.chain()
|
||||||
if (name === "Default") {
|
.focus()
|
||||||
editor.commands.unsetColor();
|
.toggleMark("highlight", {
|
||||||
} else {
|
color: color || "",
|
||||||
editor.chain().focus().setColor(color || "").run();
|
colorName: name.toLowerCase() || "",
|
||||||
}
|
})
|
||||||
setIsOpen(false);
|
.run();
|
||||||
}}
|
}
|
||||||
style={{ border: "none" }}
|
setIsOpen(false);
|
||||||
>
|
}}
|
||||||
{t(name)}
|
style={{
|
||||||
</Button>
|
width: rem(28),
|
||||||
))}
|
height: rem(28),
|
||||||
</Button.Group>
|
borderRadius: rem(4),
|
||||||
|
backgroundColor: color || "var(--mantine-color-gray-2)",
|
||||||
|
border: "1px solid var(--mantine-color-gray-4)",
|
||||||
|
cursor: "pointer",
|
||||||
|
position: "relative",
|
||||||
|
display: "flex",
|
||||||
|
alignItems: "center",
|
||||||
|
justifyContent: "center",
|
||||||
|
fontSize: rem(16),
|
||||||
|
fontWeight: 600,
|
||||||
|
color: "var(--mantine-color-gray-8)",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{editorState[`highlight_${color}`] ? (
|
||||||
|
<IconCheck
|
||||||
|
size={16}
|
||||||
|
color="var(--mantine-color-green-7)"
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
"A"
|
||||||
|
)}
|
||||||
|
</Box>
|
||||||
|
</Tooltip>
|
||||||
|
))}
|
||||||
|
</SimpleGrid>
|
||||||
|
</Box>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
variant="default"
|
||||||
|
fullWidth
|
||||||
|
onClick={() => {
|
||||||
|
editor.commands.unsetColor();
|
||||||
|
editor.commands.unsetHighlight();
|
||||||
|
setIsOpen(false);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{t("Remove color")}
|
||||||
|
</Button>
|
||||||
|
</Stack>
|
||||||
</ScrollArea.Autosize>
|
</ScrollArea.Autosize>
|
||||||
</Popover.Dropdown>
|
</Popover.Dropdown>
|
||||||
</Popover>
|
</Popover>
|
||||||
|
|||||||
@@ -13,11 +13,12 @@ import {
|
|||||||
IconTypography,
|
IconTypography,
|
||||||
} from "@tabler/icons-react";
|
} from "@tabler/icons-react";
|
||||||
import { Popover, Button, ScrollArea } from "@mantine/core";
|
import { Popover, Button, ScrollArea } from "@mantine/core";
|
||||||
import { useEditor } from "@tiptap/react";
|
import type { Editor } from "@tiptap/react";
|
||||||
|
import { useEditorState } from "@tiptap/react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
interface NodeSelectorProps {
|
interface NodeSelectorProps {
|
||||||
editor: ReturnType<typeof useEditor>;
|
editor: Editor | null;
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
setIsOpen: Dispatch<SetStateAction<boolean>>;
|
setIsOpen: Dispatch<SetStateAction<boolean>>;
|
||||||
}
|
}
|
||||||
@@ -36,6 +37,27 @@ export const NodeSelector: FC<NodeSelectorProps> = ({
|
|||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const editorState = useEditorState({
|
||||||
|
editor,
|
||||||
|
selector: (ctx) => {
|
||||||
|
if (!editor) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
isParagraph: ctx.editor.isActive("paragraph"),
|
||||||
|
isBulletList: ctx.editor.isActive("bulletList"),
|
||||||
|
isOrderedList: ctx.editor.isActive("orderedList"),
|
||||||
|
isHeading1: ctx.editor.isActive("heading", { level: 1 }),
|
||||||
|
isHeading2: ctx.editor.isActive("heading", { level: 2 }),
|
||||||
|
isHeading3: ctx.editor.isActive("heading", { level: 3 }),
|
||||||
|
isTaskItem: ctx.editor.isActive("taskItem"),
|
||||||
|
isBlockquote: ctx.editor.isActive("blockquote"),
|
||||||
|
isCodeBlock: ctx.editor.isActive("codeBlock"),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const items: BubbleMenuItem[] = [
|
const items: BubbleMenuItem[] = [
|
||||||
{
|
{
|
||||||
name: "Text",
|
name: "Text",
|
||||||
@@ -43,45 +65,45 @@ export const NodeSelector: FC<NodeSelectorProps> = ({
|
|||||||
command: () =>
|
command: () =>
|
||||||
editor.chain().focus().toggleNode("paragraph", "paragraph").run(),
|
editor.chain().focus().toggleNode("paragraph", "paragraph").run(),
|
||||||
isActive: () =>
|
isActive: () =>
|
||||||
editor.isActive("paragraph") &&
|
editorState?.isParagraph &&
|
||||||
!editor.isActive("bulletList") &&
|
!editorState?.isBulletList &&
|
||||||
!editor.isActive("orderedList"),
|
!editorState?.isOrderedList,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Heading 1",
|
name: "Heading 1",
|
||||||
icon: IconH1,
|
icon: IconH1,
|
||||||
command: () => editor.chain().focus().toggleHeading({ level: 1 }).run(),
|
command: () => editor.chain().focus().toggleHeading({ level: 1 }).run(),
|
||||||
isActive: () => editor.isActive("heading", { level: 1 }),
|
isActive: () => editorState?.isHeading1,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Heading 2",
|
name: "Heading 2",
|
||||||
icon: IconH2,
|
icon: IconH2,
|
||||||
command: () => editor.chain().focus().toggleHeading({ level: 2 }).run(),
|
command: () => editor.chain().focus().toggleHeading({ level: 2 }).run(),
|
||||||
isActive: () => editor.isActive("heading", { level: 2 }),
|
isActive: () => editorState?.isHeading2,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Heading 3",
|
name: "Heading 3",
|
||||||
icon: IconH3,
|
icon: IconH3,
|
||||||
command: () => editor.chain().focus().toggleHeading({ level: 3 }).run(),
|
command: () => editor.chain().focus().toggleHeading({ level: 3 }).run(),
|
||||||
isActive: () => editor.isActive("heading", { level: 3 }),
|
isActive: () => editorState?.isHeading3,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "To-do List",
|
name: "To-do List",
|
||||||
icon: IconCheckbox,
|
icon: IconCheckbox,
|
||||||
command: () => editor.chain().focus().toggleTaskList().run(),
|
command: () => editor.chain().focus().toggleTaskList().run(),
|
||||||
isActive: () => editor.isActive("taskItem"),
|
isActive: () => editorState?.isTaskItem,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Bullet List",
|
name: "Bullet List",
|
||||||
icon: IconList,
|
icon: IconList,
|
||||||
command: () => editor.chain().focus().toggleBulletList().run(),
|
command: () => editor.chain().focus().toggleBulletList().run(),
|
||||||
isActive: () => editor.isActive("bulletList"),
|
isActive: () => editorState?.isBulletList,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Numbered List",
|
name: "Numbered List",
|
||||||
icon: IconListNumbers,
|
icon: IconListNumbers,
|
||||||
command: () => editor.chain().focus().toggleOrderedList().run(),
|
command: () => editor.chain().focus().toggleOrderedList().run(),
|
||||||
isActive: () => editor.isActive("orderedList"),
|
isActive: () => editorState?.isOrderedList,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Blockquote",
|
name: "Blockquote",
|
||||||
@@ -93,13 +115,13 @@ export const NodeSelector: FC<NodeSelectorProps> = ({
|
|||||||
.toggleNode("paragraph", "paragraph")
|
.toggleNode("paragraph", "paragraph")
|
||||||
.toggleBlockquote()
|
.toggleBlockquote()
|
||||||
.run(),
|
.run(),
|
||||||
isActive: () => editor.isActive("blockquote"),
|
isActive: () => editorState?.isBlockquote,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Code",
|
name: "Code",
|
||||||
icon: IconCode,
|
icon: IconCode,
|
||||||
command: () => editor.chain().focus().toggleCodeBlock().run(),
|
command: () => editor.chain().focus().toggleCodeBlock().run(),
|
||||||
isActive: () => editor.isActive("codeBlock"),
|
isActive: () => editorState?.isCodeBlock,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|||||||
+29
-10
@@ -8,11 +8,12 @@ import {
|
|||||||
IconChevronDown,
|
IconChevronDown,
|
||||||
} from "@tabler/icons-react";
|
} from "@tabler/icons-react";
|
||||||
import { Popover, Button, ScrollArea, rem } from "@mantine/core";
|
import { Popover, Button, ScrollArea, rem } from "@mantine/core";
|
||||||
import { useEditor } from "@tiptap/react";
|
import type { Editor } from "@tiptap/react";
|
||||||
|
import { useEditorState } from "@tiptap/react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
interface TextAlignmentProps {
|
interface TextAlignmentProps {
|
||||||
editor: ReturnType<typeof useEditor>;
|
editor: Editor | null;
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
setIsOpen: Dispatch<SetStateAction<boolean>>;
|
setIsOpen: Dispatch<SetStateAction<boolean>>;
|
||||||
}
|
}
|
||||||
@@ -31,36 +32,54 @@ export const TextAlignmentSelector: FC<TextAlignmentProps> = ({
|
|||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const editorState = useEditorState({
|
||||||
|
editor,
|
||||||
|
selector: (ctx) => {
|
||||||
|
if (!ctx.editor) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
isAlignLeft: ctx.editor.isActive({ textAlign: "left" }),
|
||||||
|
isAlignCenter: ctx.editor.isActive({ textAlign: "center" }),
|
||||||
|
isAlignRight: ctx.editor.isActive({ textAlign: "right" }),
|
||||||
|
isAlignJustify: ctx.editor.isActive({ textAlign: "justify" }),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!editor || !editorState) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
const items: BubbleMenuItem[] = [
|
const items: BubbleMenuItem[] = [
|
||||||
{
|
{
|
||||||
name: "Align left",
|
name: "Align left",
|
||||||
isActive: () => editor.isActive({ textAlign: "left" }),
|
isActive: () => editorState?.isAlignLeft,
|
||||||
command: () => editor.chain().focus().setTextAlign("left").run(),
|
command: () => editor.chain().focus().setTextAlign("left").run(),
|
||||||
icon: IconAlignLeft,
|
icon: IconAlignLeft,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Align center",
|
name: "Align center",
|
||||||
isActive: () => editor.isActive({ textAlign: "center" }),
|
isActive: () => editorState?.isAlignCenter,
|
||||||
command: () => editor.chain().focus().setTextAlign("center").run(),
|
command: () => editor.chain().focus().setTextAlign("center").run(),
|
||||||
icon: IconAlignCenter,
|
icon: IconAlignCenter,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Align right",
|
name: "Align right",
|
||||||
isActive: () => editor.isActive({ textAlign: "right" }),
|
isActive: () => editorState?.isAlignRight,
|
||||||
command: () => editor.chain().focus().setTextAlign("right").run(),
|
command: () => editor.chain().focus().setTextAlign("right").run(),
|
||||||
icon: IconAlignRight,
|
icon: IconAlignRight,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Justify",
|
name: "Justify",
|
||||||
isActive: () => editor.isActive({ textAlign: "justify" }),
|
isActive: () => editorState?.isAlignJustify,
|
||||||
command: () => editor.chain().focus().setTextAlign("justify").run(),
|
command: () => editor.chain().focus().setTextAlign("justify").run(),
|
||||||
icon: IconAlignJustified,
|
icon: IconAlignJustified,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const activeItem = items.filter((item) => item.isActive()).pop() ?? {
|
const activeItem = items.filter((item) => item.isActive()).pop() ?? items[0];
|
||||||
name: "Multiple",
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Popover opened={isOpen} withArrow>
|
<Popover opened={isOpen} withArrow>
|
||||||
@@ -73,7 +92,7 @@ export const TextAlignmentSelector: FC<TextAlignmentProps> = ({
|
|||||||
rightSection={<IconChevronDown size={16} />}
|
rightSection={<IconChevronDown size={16} />}
|
||||||
onClick={() => setIsOpen(!isOpen)}
|
onClick={() => setIsOpen(!isOpen)}
|
||||||
>
|
>
|
||||||
<IconAlignLeft style={{ width: rem(16) }} stroke={2} />
|
<activeItem.icon style={{ width: rem(16) }} stroke={2} />
|
||||||
</Button>
|
</Button>
|
||||||
</Popover.Target>
|
</Popover.Target>
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,5 @@
|
|||||||
import {
|
import { BubbleMenu as BaseBubbleMenu } from "@tiptap/react/menus";
|
||||||
BubbleMenu as BaseBubbleMenu,
|
import { findParentNode, posToDOMRect, useEditorState } from "@tiptap/react";
|
||||||
findParentNode,
|
|
||||||
posToDOMRect,
|
|
||||||
} from "@tiptap/react";
|
|
||||||
import React, { useCallback } from "react";
|
import React, { useCallback } from "react";
|
||||||
import { Node as PMNode } from "prosemirror-model";
|
import { Node as PMNode } from "prosemirror-model";
|
||||||
import {
|
import {
|
||||||
@@ -15,12 +12,15 @@ import {
|
|||||||
IconCircleCheckFilled,
|
IconCircleCheckFilled,
|
||||||
IconCircleXFilled,
|
IconCircleXFilled,
|
||||||
IconInfoCircleFilled,
|
IconInfoCircleFilled,
|
||||||
|
IconMoodSmile,
|
||||||
} from "@tabler/icons-react";
|
} from "@tabler/icons-react";
|
||||||
import { CalloutType } from "@docmost/editor-ext";
|
import { CalloutType } from "@docmost/editor-ext";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
import EmojiPicker from "@/components/ui/emoji-picker.tsx";
|
||||||
|
|
||||||
export function CalloutMenu({ editor }: EditorMenuProps) {
|
export function CalloutMenu({ editor }: EditorMenuProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
const shouldShow = useCallback(
|
const shouldShow = useCallback(
|
||||||
({ state }: ShouldShowProps) => {
|
({ state }: ShouldShowProps) => {
|
||||||
if (!state) {
|
if (!state) {
|
||||||
@@ -32,17 +32,43 @@ export function CalloutMenu({ editor }: EditorMenuProps) {
|
|||||||
[editor],
|
[editor],
|
||||||
);
|
);
|
||||||
|
|
||||||
const getReferenceClientRect = useCallback(() => {
|
const editorState = useEditorState({
|
||||||
|
editor,
|
||||||
|
selector: (ctx) => {
|
||||||
|
if (!ctx.editor) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
isCallout: ctx.editor.isActive("callout"),
|
||||||
|
isInfo: ctx.editor.isActive("callout", { type: "info" }),
|
||||||
|
isSuccess: ctx.editor.isActive("callout", { type: "success" }),
|
||||||
|
isWarning: ctx.editor.isActive("callout", { type: "warning" }),
|
||||||
|
isDanger: ctx.editor.isActive("callout", { type: "danger" }),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const getReferencedVirtualElement = useCallback(() => {
|
||||||
|
if (!editor) return;
|
||||||
const { selection } = editor.state;
|
const { selection } = editor.state;
|
||||||
const predicate = (node: PMNode) => node.type.name === "callout";
|
const predicate = (node: PMNode) => node.type.name === "callout";
|
||||||
const parent = findParentNode(predicate)(selection);
|
const parent = findParentNode(predicate)(selection);
|
||||||
|
|
||||||
if (parent) {
|
if (parent) {
|
||||||
const dom = editor.view.nodeDOM(parent?.pos) as HTMLElement;
|
const dom = editor.view.nodeDOM(parent?.pos) as HTMLElement;
|
||||||
return dom.getBoundingClientRect();
|
const domRect = dom.getBoundingClientRect();
|
||||||
|
return {
|
||||||
|
getBoundingClientRect: () => domRect,
|
||||||
|
getClientRects: () => [domRect],
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
return posToDOMRect(editor.view, selection.from, selection.to);
|
const domRect = posToDOMRect(editor.view, selection.from, selection.to);
|
||||||
|
return {
|
||||||
|
getBoundingClientRect: () => domRect,
|
||||||
|
getClientRects: () => [domRect],
|
||||||
|
};
|
||||||
}, [editor]);
|
}, [editor]);
|
||||||
|
|
||||||
const setCalloutType = useCallback(
|
const setCalloutType = useCallback(
|
||||||
@@ -56,19 +82,47 @@ export function CalloutMenu({ editor }: EditorMenuProps) {
|
|||||||
[editor],
|
[editor],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const setCalloutIcon = useCallback(
|
||||||
|
(emoji: any) => {
|
||||||
|
const emojiChar = emoji?.native || emoji?.emoji || emoji;
|
||||||
|
editor
|
||||||
|
.chain()
|
||||||
|
.focus(undefined, { scrollIntoView: false })
|
||||||
|
.updateCalloutIcon(emojiChar)
|
||||||
|
.run();
|
||||||
|
},
|
||||||
|
[editor],
|
||||||
|
);
|
||||||
|
|
||||||
|
const removeCalloutIcon = useCallback(() => {
|
||||||
|
editor
|
||||||
|
.chain()
|
||||||
|
.focus(undefined, { scrollIntoView: false })
|
||||||
|
.updateCalloutIcon("")
|
||||||
|
.run();
|
||||||
|
}, [editor]);
|
||||||
|
|
||||||
|
const getCurrentIcon = () => {
|
||||||
|
const { selection } = editor.state;
|
||||||
|
const predicate = (node: PMNode) => node.type.name === "callout";
|
||||||
|
const parent = findParentNode(predicate)(selection);
|
||||||
|
const icon = parent?.node.attrs.icon;
|
||||||
|
return icon || null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const currentIcon = getCurrentIcon();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<BaseBubbleMenu
|
<BaseBubbleMenu
|
||||||
editor={editor}
|
editor={editor}
|
||||||
pluginKey={`callout-menu}`}
|
pluginKey={`callout-menu`}
|
||||||
updateDelay={0}
|
updateDelay={0}
|
||||||
tippyOptions={{
|
getReferencedVirtualElement={getReferencedVirtualElement}
|
||||||
getReferenceClientRect,
|
options={{
|
||||||
offset: [0, 10],
|
|
||||||
placement: "bottom",
|
placement: "bottom",
|
||||||
zIndex: 99,
|
// offset: 233, // // offset: [0, 10],
|
||||||
popperOptions: {
|
// zIndex: 99,
|
||||||
modifiers: [{ name: "flip", enabled: false }],
|
flip: false,
|
||||||
},
|
|
||||||
}}
|
}}
|
||||||
shouldShow={shouldShow}
|
shouldShow={shouldShow}
|
||||||
>
|
>
|
||||||
@@ -78,9 +132,7 @@ export function CalloutMenu({ editor }: EditorMenuProps) {
|
|||||||
onClick={() => setCalloutType("info")}
|
onClick={() => setCalloutType("info")}
|
||||||
size="lg"
|
size="lg"
|
||||||
aria-label={t("Info")}
|
aria-label={t("Info")}
|
||||||
variant={
|
variant={editorState?.isInfo ? "light" : "default"}
|
||||||
editor.isActive("callout", { type: "info" }) ? "light" : "default"
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
<IconInfoCircleFilled size={18} />
|
<IconInfoCircleFilled size={18} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
@@ -91,11 +143,7 @@ export function CalloutMenu({ editor }: EditorMenuProps) {
|
|||||||
onClick={() => setCalloutType("success")}
|
onClick={() => setCalloutType("success")}
|
||||||
size="lg"
|
size="lg"
|
||||||
aria-label={t("Success")}
|
aria-label={t("Success")}
|
||||||
variant={
|
variant={editorState?.isSuccess ? "light" : "default"}
|
||||||
editor.isActive("callout", { type: "success" })
|
|
||||||
? "light"
|
|
||||||
: "default"
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
<IconCircleCheckFilled size={18} />
|
<IconCircleCheckFilled size={18} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
@@ -106,11 +154,7 @@ export function CalloutMenu({ editor }: EditorMenuProps) {
|
|||||||
onClick={() => setCalloutType("warning")}
|
onClick={() => setCalloutType("warning")}
|
||||||
size="lg"
|
size="lg"
|
||||||
aria-label={t("Warning")}
|
aria-label={t("Warning")}
|
||||||
variant={
|
variant={editorState?.isWarning ? "light" : "default"}
|
||||||
editor.isActive("callout", { type: "warning" })
|
|
||||||
? "light"
|
|
||||||
: "default"
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
<IconAlertTriangleFilled size={18} />
|
<IconAlertTriangleFilled size={18} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
@@ -121,15 +165,23 @@ export function CalloutMenu({ editor }: EditorMenuProps) {
|
|||||||
onClick={() => setCalloutType("danger")}
|
onClick={() => setCalloutType("danger")}
|
||||||
size="lg"
|
size="lg"
|
||||||
aria-label={t("Danger")}
|
aria-label={t("Danger")}
|
||||||
variant={
|
variant={editorState?.isDanger ? "light" : "default"}
|
||||||
editor.isActive("callout", { type: "danger" })
|
|
||||||
? "light"
|
|
||||||
: "default"
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
<IconCircleXFilled size={18} />
|
<IconCircleXFilled size={18} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
|
<EmojiPicker
|
||||||
|
onEmojiSelect={setCalloutIcon}
|
||||||
|
removeEmojiAction={removeCalloutIcon}
|
||||||
|
readOnly={false}
|
||||||
|
icon={currentIcon || <IconMoodSmile size={18} />}
|
||||||
|
actionIconProps={{
|
||||||
|
size: "lg",
|
||||||
|
variant: "default",
|
||||||
|
c: undefined,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
</ActionIcon.Group>
|
</ActionIcon.Group>
|
||||||
</BaseBubbleMenu>
|
</BaseBubbleMenu>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ import { CalloutType } from "@docmost/editor-ext";
|
|||||||
|
|
||||||
export default function CalloutView(props: NodeViewProps) {
|
export default function CalloutView(props: NodeViewProps) {
|
||||||
const { node } = props;
|
const { node } = props;
|
||||||
const { type } = node.attrs;
|
const { type, icon } = node.attrs;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<NodeViewWrapper>
|
<NodeViewWrapper>
|
||||||
@@ -19,7 +19,7 @@ export default function CalloutView(props: NodeViewProps) {
|
|||||||
variant="light"
|
variant="light"
|
||||||
title=""
|
title=""
|
||||||
color={getCalloutColor(type)}
|
color={getCalloutColor(type)}
|
||||||
icon={getCalloutIcon(type)}
|
icon={getCalloutIcon(type, icon)}
|
||||||
p="xs"
|
p="xs"
|
||||||
classNames={{
|
classNames={{
|
||||||
message: classes.message,
|
message: classes.message,
|
||||||
@@ -32,7 +32,11 @@ export default function CalloutView(props: NodeViewProps) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function getCalloutIcon(type: CalloutType) {
|
function getCalloutIcon(type: CalloutType, customIcon?: string) {
|
||||||
|
if (customIcon && customIcon.trim() !== "") {
|
||||||
|
return <span style={{ fontSize: '18px' }}>{customIcon}</span>;
|
||||||
|
}
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case "info":
|
case "info":
|
||||||
return <IconInfoCircleFilled />;
|
return <IconInfoCircleFilled />;
|
||||||
|
|||||||
@@ -90,6 +90,7 @@ export default function CodeBlockView(props: NodeViewProps) {
|
|||||||
node.textContent.length > 0
|
node.textContent.length > 0
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
|
{/* @ts-ignore */}
|
||||||
<NodeViewContent as="code" className={`language-${language}`} />
|
<NodeViewContent as="code" className={`language-${language}`} />
|
||||||
</pre>
|
</pre>
|
||||||
|
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user