mirror of
https://github.com/docmost/docmost.git
synced 2026-05-07 14:43:06 +08:00
Compare commits
175 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| af6d9672f5 | |||
| 2e4e5ab47f | |||
| cd5fa445be | |||
| a8565e57fb | |||
| 0fb89f52f0 | |||
| 9226014fe9 | |||
| c9531acf04 | |||
| b3ab12d732 | |||
| 74118e11d7 | |||
| 57efb91bd3 | |||
| da9b43681e | |||
| 4966f9b152 | |||
| e1bbceb9a6 | |||
| 895c1817ae | |||
| 642024ba9d | |||
| 147d028036 | |||
| 992691e6e0 | |||
| 9aaa6c731c | |||
| fd91b11c6c | |||
| af8b0ddf3a | |||
| 879aa2c3d8 | |||
| c180d0e487 | |||
| a062f7a165 | |||
| cbd0dd4a0b | |||
| 2d6d829581 | |||
| 5cea30cc5c | |||
| bca85a49d6 | |||
| c9cdfa0f17 | |||
| 412962204c | |||
| a42ac3d450 | |||
| 642c92f779 | |||
| ccb35517bb | |||
| cbdb37ed0a | |||
| aa27d57624 | |||
| 3829b6cbef | |||
| 17da762984 | |||
| 859f16740b | |||
| 7981ef462e | |||
| 2d835da0e3 | |||
| a3559b7c33 | |||
| 803f1f0b81 | |||
| 4e8f533b91 | |||
| 7b0d8fe140 | |||
| 2f92278a9d | |||
| 53608eae35 | |||
| 0e4a1e7419 | |||
| 9125996e97 | |||
| fa4872e89e | |||
| 6d6f3a8a8e | |||
| 975b4dcaab | |||
| 6683c515cf | |||
| cc5c800238 | |||
| cfaee93af9 | |||
| 74eddb0638 | |||
| 7c83a9d4f0 | |||
| 2678c4e279 | |||
| b0bde4b375 | |||
| 724e37d5b7 | |||
| 33184e9d8d | |||
| 7520c329d0 | |||
| d7a5fda53c | |||
| 236a63dadc | |||
| 89b94e5d19 | |||
| 97c459be67 | |||
| d0ed6865cb | |||
| 65b89a1b24 | |||
| 1fdee33206 | |||
| 7b69727a30 | |||
| 66c26af34b | |||
| b4f009513e | |||
| fcffa3dfa0 | |||
| 1980b94825 | |||
| bea1637519 | |||
| 37355452e1 | |||
| 057360c6be | |||
| f12bfc1ff7 | |||
| f5d794220e | |||
| a3c1c6cccd | |||
| 4b105586a9 | |||
| d2641db895 | |||
| 1111df65cd | |||
| e455154b7d | |||
| ef24b3c07d | |||
| 2352f3c5d9 | |||
| 568dd4c321 | |||
| b6478fee84 | |||
| 5d2aad3668 | |||
| 9331ac2df8 | |||
| 9f4728e279 | |||
| 628b08339a | |||
| 68842dbea2 | |||
| b1510cd6d7 | |||
| af92224e10 | |||
| c24ff44e09 | |||
| 90c190df78 | |||
| 17ec2f4ac5 | |||
| 9881c53f00 | |||
| 721651e2e2 | |||
| a3fd79dae8 | |||
| 616d9297eb | |||
| ee6b98edaa | |||
| cf43e2b4fe | |||
| 614baf153b | |||
| 4f3577f009 | |||
| d5e4b8bb59 | |||
| 1a897faaa2 | |||
| 6f1a91cc05 | |||
| 60848ea903 | |||
| 2309d1434b | |||
| dcc2bacb22 | |||
| 69d7532c6c | |||
| 85ce0d32bf | |||
| fc0997fd90 | |||
| df64de5306 | |||
| ea44468fad | |||
| 59e945562d | |||
| 22f33bab7c | |||
| e0a8521566 | |||
| b5803f42da | |||
| 5de1c8e3ed | |||
| ef87210b3d | |||
| c172d3bd5e | |||
| 53132acb0a | |||
| d6472f0876 | |||
| 873c963043 | |||
| 03a70d768a | |||
| 0aeaa43112 | |||
| 92d5d0b237 | |||
| 0ce74d34de | |||
| 00b5328676 | |||
| 2ebdc2baea | |||
| 621ef4f0cf | |||
| 26b9338da5 | |||
| 618f56577d | |||
| 0a05ce6133 | |||
| cb9d6be3b9 | |||
| b76f5adaad | |||
| 41fa77b29d | |||
| 05b3c65b0f | |||
| e0ab9d9b5e | |||
| 55280db672 | |||
| 32bbc6911f | |||
| 5814542128 | |||
| 18b5781522 | |||
| 49ab9875ba | |||
| 25f4b8c2b4 | |||
| 4d43f86c51 | |||
| f170ede8da | |||
| 7861b5b186 | |||
| 3a9bdfbb06 | |||
| ab7999a946 | |||
| 0f02261ee6 | |||
| aff8dba2cb | |||
| f6a8247c48 | |||
| 7879e1f600 | |||
| 3cb70f0696 | |||
| fbb44df548 | |||
| bc3ce893c4 | |||
| ae96352189 | |||
| 1ad53c2581 | |||
| 2f97a3debc | |||
| 40b5346f9e | |||
| d6b4573b79 | |||
| 4878850b25 | |||
| 5c3942c159 | |||
| e0809e7104 | |||
| da6793ac87 | |||
| 08e94eb3c1 | |||
| 5a14186f1c | |||
| 6a0bb8d4cb | |||
| fba9f4cb2b | |||
| d8f7c4a822 | |||
| 202685b39f | |||
| fc4a428208 | |||
| 5506eb194b |
@@ -0,0 +1,154 @@
|
|||||||
|
name: Release
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
tags:
|
||||||
|
- 'v*'
|
||||||
|
workflow_dispatch:
|
||||||
|
inputs:
|
||||||
|
version:
|
||||||
|
description: 'Version tag (e.g. v0.25.3)'
|
||||||
|
required: true
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
env:
|
||||||
|
VERSION: ${{ inputs.version || github.ref_name }}
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
include:
|
||||||
|
- platform: linux/amd64
|
||||||
|
runner: ubuntu-latest
|
||||||
|
suffix: amd64
|
||||||
|
- platform: linux/arm64
|
||||||
|
runner: ubuntu-24.04-arm
|
||||||
|
suffix: arm64
|
||||||
|
runs-on: ${{ matrix.runner }}
|
||||||
|
steps:
|
||||||
|
- name: Generate token
|
||||||
|
id: app-token
|
||||||
|
uses: actions/create-github-app-token@v1
|
||||||
|
with:
|
||||||
|
app-id: ${{ secrets.BUILD_APP_ID }}
|
||||||
|
private-key: ${{ secrets.BUILD_APP_PRIVATE_KEY }}
|
||||||
|
owner: ${{ github.repository_owner }}
|
||||||
|
|
||||||
|
- name: Checkout with submodules
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
submodules: recursive
|
||||||
|
token: ${{ steps.app-token.outputs.token }}
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- name: Login to Docker Hub
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
||||||
|
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Build and push by digest
|
||||||
|
id: build
|
||||||
|
uses: docker/build-push-action@v6
|
||||||
|
with:
|
||||||
|
context: .
|
||||||
|
platforms: ${{ matrix.platform }}
|
||||||
|
outputs: type=image,name=docmost/docmost,push-by-digest=true,name-canonical=true,push=true
|
||||||
|
cache-from: type=gha,scope=${{ matrix.suffix }}
|
||||||
|
cache-to: type=gha,scope=${{ matrix.suffix }},mode=max
|
||||||
|
|
||||||
|
- name: Export digest
|
||||||
|
run: |
|
||||||
|
mkdir -p /tmp/digests
|
||||||
|
digest="${{ steps.build.outputs.digest }}"
|
||||||
|
touch "/tmp/digests/${digest#sha256:}"
|
||||||
|
|
||||||
|
- name: Upload digest
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: digest-${{ matrix.suffix }}
|
||||||
|
path: /tmp/digests/*
|
||||||
|
if-no-files-found: error
|
||||||
|
|
||||||
|
- name: Strip v prefix
|
||||||
|
id: strip-v
|
||||||
|
run: echo "version=${VERSION#v}" >> "$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
- name: Export Docker image
|
||||||
|
uses: docker/build-push-action@v6
|
||||||
|
with:
|
||||||
|
context: .
|
||||||
|
platforms: ${{ matrix.platform }}
|
||||||
|
push: false
|
||||||
|
tags: |
|
||||||
|
docmost/docmost:latest
|
||||||
|
docmost/docmost:${{ steps.strip-v.outputs.version }}
|
||||||
|
outputs: type=docker,dest=docmost-${{ matrix.suffix }}.docker.tar
|
||||||
|
cache-from: type=gha,scope=${{ matrix.suffix }}
|
||||||
|
|
||||||
|
- name: Compress image
|
||||||
|
run: gzip docmost-${{ matrix.suffix }}.docker.tar
|
||||||
|
|
||||||
|
- name: Upload image archive
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: docker-image-${{ matrix.suffix }}
|
||||||
|
path: docmost-${{ matrix.suffix }}.docker.tar.gz
|
||||||
|
if-no-files-found: error
|
||||||
|
|
||||||
|
release:
|
||||||
|
needs: build
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Download digests
|
||||||
|
uses: actions/download-artifact@v4
|
||||||
|
with:
|
||||||
|
pattern: digest-*
|
||||||
|
path: /tmp/digests
|
||||||
|
merge-multiple: true
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- name: Login to Docker Hub
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
||||||
|
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Extract metadata for tags
|
||||||
|
id: meta
|
||||||
|
uses: docker/metadata-action@v5
|
||||||
|
with:
|
||||||
|
images: docmost/docmost
|
||||||
|
tags: |
|
||||||
|
type=semver,pattern={{version}},value=${{ env.VERSION }}
|
||||||
|
type=semver,pattern={{major}}.{{minor}},value=${{ env.VERSION }},enable=${{ !contains(env.VERSION, '-') }}
|
||||||
|
type=raw,value=latest,enable=${{ !contains(env.VERSION, '-') }}
|
||||||
|
|
||||||
|
- name: Create manifest list and push
|
||||||
|
working-directory: /tmp/digests
|
||||||
|
run: |
|
||||||
|
docker buildx imagetools create $(jq -cr '.tags | map("-t " + .) | join(" ")' <<< "$DOCKER_METADATA_OUTPUT_JSON") \
|
||||||
|
$(printf 'docmost/docmost@sha256:%s ' *)
|
||||||
|
|
||||||
|
- name: Download image archives
|
||||||
|
uses: actions/download-artifact@v4
|
||||||
|
with:
|
||||||
|
pattern: docker-image-*
|
||||||
|
path: /tmp/images
|
||||||
|
merge-multiple: true
|
||||||
|
|
||||||
|
- name: Create GitHub Release
|
||||||
|
uses: softprops/action-gh-release@v2
|
||||||
|
with:
|
||||||
|
tag_name: ${{ env.VERSION }}
|
||||||
|
files: |
|
||||||
|
/tmp/images/docmost-amd64.docker.tar.gz
|
||||||
|
/tmp/images/docmost-arm64.docker.tar.gz
|
||||||
|
draft: true
|
||||||
+42
-40
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "client",
|
"name": "client",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "0.25.0-beta.1",
|
"version": "0.71.1",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
"build": "tsc && vite build",
|
"build": "tsc && vite build",
|
||||||
@@ -10,74 +10,76 @@
|
|||||||
"format": "prettier --write \"src/**/*.tsx\" \"src/**/*.ts\""
|
"format": "prettier --write \"src/**/*.tsx\" \"src/**/*.ts\""
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@casl/react": "^4.0.0",
|
"@casl/react": "^5.0.1",
|
||||||
"@docmost/editor-ext": "workspace:*",
|
"@docmost/editor-ext": "workspace:*",
|
||||||
"@emoji-mart/data": "^1.2.1",
|
"@emoji-mart/data": "^1.2.1",
|
||||||
"@emoji-mart/react": "^1.1.1",
|
"@emoji-mart/react": "^1.1.1",
|
||||||
"@excalidraw/excalidraw": "0.18.0-c158187",
|
"@excalidraw/excalidraw": "0.18.0-3a5ef40",
|
||||||
"@mantine/core": "^8.3.12",
|
"@mantine/core": "^8.3.18",
|
||||||
"@mantine/dates": "^8.3.12",
|
"@mantine/dates": "^8.3.18",
|
||||||
"@mantine/form": "^8.3.12",
|
"@mantine/form": "^8.3.18",
|
||||||
"@mantine/hooks": "^8.3.12",
|
"@mantine/hooks": "^8.3.18",
|
||||||
"@mantine/modals": "^8.3.12",
|
"@mantine/modals": "^8.3.18",
|
||||||
"@mantine/notifications": "^8.3.12",
|
"@mantine/notifications": "^8.3.18",
|
||||||
"@mantine/spotlight": "^8.3.12",
|
"@mantine/spotlight": "^8.3.18",
|
||||||
"@tabler/icons-react": "^3.36.1",
|
"@tabler/icons-react": "^3.40.0",
|
||||||
"@tanstack/react-query": "^5.90.17",
|
"@tanstack/react-query": "5.90.17",
|
||||||
"alfaaz": "^1.1.0",
|
"alfaaz": "^1.1.0",
|
||||||
"axios": "^1.13.2",
|
"axios": "1.13.6",
|
||||||
|
"blueimp-load-image": "^5.16.0",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
"emoji-mart": "^5.6.0",
|
"emoji-mart": "^5.6.0",
|
||||||
"file-saver": "^2.0.5",
|
"file-saver": "^2.0.5",
|
||||||
"highlightjs-sap-abap": "^0.3.0",
|
"highlightjs-sap-abap": "^0.3.0",
|
||||||
"i18next": "^23.16.8",
|
"i18next": "^25.10.1",
|
||||||
"i18next-http-backend": "^2.7.3",
|
"i18next-http-backend": "^3.0.2",
|
||||||
"jotai": "^2.16.2",
|
"jotai": "^2.18.1",
|
||||||
"jotai-optics": "^0.4.0",
|
"jotai-optics": "^0.4.0",
|
||||||
"js-cookie": "^3.0.5",
|
"js-cookie": "^3.0.5",
|
||||||
"jwt-decode": "^4.0.0",
|
"jwt-decode": "^4.0.0",
|
||||||
"katex": "0.16.27",
|
"katex": "0.16.40",
|
||||||
"lowlight": "^3.3.0",
|
"lowlight": "^3.3.0",
|
||||||
"mantine-form-zod-resolver": "^1.3.0",
|
"mantine-form-zod-resolver": "^1.3.0",
|
||||||
"mermaid": "^11.12.2",
|
"mermaid": "^11.13.0",
|
||||||
"mitt": "^3.0.1",
|
"mitt": "^3.0.1",
|
||||||
"posthog-js": "^1.255.1",
|
"posthog-js": "1.363.1",
|
||||||
"react": "^18.3.1",
|
"react": "^18.3.1",
|
||||||
"react-arborist": "3.4.0",
|
"react-arborist": "3.4.0",
|
||||||
"react-clear-modal": "^2.0.17",
|
"react-clear-modal": "^2.0.18",
|
||||||
"react-dom": "^18.3.1",
|
"react-dom": "^18.3.1",
|
||||||
"react-drawio": "^1.0.7",
|
"react-drawio": "^1.0.7",
|
||||||
"react-error-boundary": "^4.1.2",
|
"react-error-boundary": "^6.1.1",
|
||||||
"react-helmet-async": "^2.0.5",
|
"react-helmet-async": "^3.0.0",
|
||||||
"react-i18next": "^15.0.1",
|
"react-i18next": "^16.5.8",
|
||||||
"react-router-dom": "^7.12.0",
|
"react-router-dom": "^7.13.1",
|
||||||
"semver": "^7.7.3",
|
"semver": "^7.7.4",
|
||||||
"socket.io-client": "^4.8.3",
|
"socket.io-client": "^4.8.3",
|
||||||
"tiptap-extension-global-drag-handle": "^0.1.18",
|
"tiptap-extension-global-drag-handle": "^0.1.18",
|
||||||
"zod": "^3.25.76"
|
"zod": "^4.3.6"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@eslint/js": "^9.16.0",
|
"@eslint/js": "^9.28.0",
|
||||||
"@tanstack/eslint-plugin-query": "^5.62.1",
|
"@tanstack/eslint-plugin-query": "^5.94.4",
|
||||||
|
"@types/blueimp-load-image": "^5.16.6",
|
||||||
"@types/file-saver": "^2.0.7",
|
"@types/file-saver": "^2.0.7",
|
||||||
"@types/js-cookie": "^3.0.6",
|
"@types/js-cookie": "^3.0.6",
|
||||||
"@types/katex": "^0.16.7",
|
"@types/katex": "^0.16.8",
|
||||||
"@types/node": "22.19.1",
|
"@types/node": "22.19.1",
|
||||||
"@types/react": "^18.3.12",
|
"@types/react": "^18.3.12",
|
||||||
"@types/react-dom": "^18.3.1",
|
"@types/react-dom": "^18.3.1",
|
||||||
"@vitejs/plugin-react": "^5.1.1",
|
"@vitejs/plugin-react": "^6.0.1",
|
||||||
"eslint": "^9.15.0",
|
"eslint": "^9.28.0",
|
||||||
"eslint-plugin-react": "^7.37.2",
|
"eslint-plugin-react": "^7.37.5",
|
||||||
"eslint-plugin-react-hooks": "^5.1.0",
|
"eslint-plugin-react-hooks": "^7.0.1",
|
||||||
"eslint-plugin-react-refresh": "^0.4.16",
|
"eslint-plugin-react-refresh": "^0.5.2",
|
||||||
"globals": "^15.13.0",
|
"globals": "^15.13.0",
|
||||||
"optics-ts": "^2.4.1",
|
"optics-ts": "^2.4.1",
|
||||||
"postcss": "^8.4.49",
|
"postcss": "^8.5.8",
|
||||||
"postcss-preset-mantine": "^1.17.0",
|
"postcss-preset-mantine": "^1.18.0",
|
||||||
"postcss-simple-vars": "^7.0.1",
|
"postcss-simple-vars": "^7.0.1",
|
||||||
"prettier": "^3.4.1",
|
"prettier": "^3.8.1",
|
||||||
"typescript": "^5.7.2",
|
"typescript": "^5.9.3",
|
||||||
"typescript-eslint": "^8.17.0",
|
"typescript-eslint": "^8.57.1",
|
||||||
"vite": "^7.2.4"
|
"vite": "8.0.5"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -41,7 +41,7 @@
|
|||||||
"Date": "Datum",
|
"Date": "Datum",
|
||||||
"Delete": "Löschen",
|
"Delete": "Löschen",
|
||||||
"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? Dabei werden auch alle Unterseiten und der Seitenverlauf gelöscht. Diese Aktion kann nicht rückgängig gemacht werden.",
|
||||||
"Description": "Beschreibung",
|
"Description": "Beschreibung",
|
||||||
"Details": "Details",
|
"Details": "Details",
|
||||||
"e.g ACME": "z.B. ACME",
|
"e.g ACME": "z.B. ACME",
|
||||||
@@ -66,7 +66,7 @@
|
|||||||
"Enter your new preferred email": "Geben Sie Ihre neue bevorzugte E-Mail ein",
|
"Enter your new preferred email": "Geben Sie Ihre neue bevorzugte E-Mail ein",
|
||||||
"Enter your password": "Geben Sie Ihr Passwort ein",
|
"Enter your password": "Geben Sie Ihr Passwort ein",
|
||||||
"Error fetching page data.": "Fehler beim Abrufen der Seitendaten.",
|
"Error fetching page data.": "Fehler beim Abrufen der Seitendaten.",
|
||||||
"Error loading page history.": "Fehler beim Laden der Seitengeschichte.",
|
"Error loading page history.": "Fehler beim Laden des Seitenverlaufs.",
|
||||||
"Export": "Exportieren",
|
"Export": "Exportieren",
|
||||||
"Failed to create page": "Erstellung der Seite fehlgeschlagen",
|
"Failed to create page": "Erstellung der Seite fehlgeschlagen",
|
||||||
"Failed to delete page": "Löschen der Seite fehlgeschlagen",
|
"Failed to delete page": "Löschen der Seite fehlgeschlagen",
|
||||||
@@ -114,20 +114,24 @@
|
|||||||
"New page": "Neue Seite",
|
"New page": "Neue Seite",
|
||||||
"New password": "Neues Passwort",
|
"New password": "Neues Passwort",
|
||||||
"No group found": "Keine Gruppe gefunden",
|
"No group found": "Keine Gruppe gefunden",
|
||||||
"No page history saved yet.": "Es wurde noch keine Seitengeschichte gespeichert.",
|
"No page history saved yet.": "Es wurde noch kein Seitenverlauf gespeichert.",
|
||||||
"No pages yet": "Noch keine Seiten",
|
"No pages yet": "Noch keine Seiten",
|
||||||
|
"No shared pages": "Keine freigegebenen Seiten",
|
||||||
"No results found...": "Keine Ergebnisse gefunden...",
|
"No results found...": "Keine Ergebnisse gefunden...",
|
||||||
"No user found": "Kein Benutzer gefunden",
|
"No user found": "Kein Benutzer gefunden",
|
||||||
"Overview": "Überblick",
|
"Overview": "Überblick",
|
||||||
"Owner": "Besitzer",
|
"Owner": "Besitzer",
|
||||||
"page": "Seite",
|
"page": "Seite",
|
||||||
"Page deleted successfully": "Seite erfolgreich gelöscht",
|
"Page deleted successfully": "Seite erfolgreich gelöscht",
|
||||||
"Page history": "Seitengeschichte",
|
"Page history": "Seitenverlauf",
|
||||||
|
"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",
|
||||||
"Password": "Passwort",
|
"Password": "Passwort",
|
||||||
"Password changed successfully": "Passwort erfolgreich geändert",
|
"Password changed successfully": "Passwort erfolgreich geändert",
|
||||||
|
"People": "Personen",
|
||||||
"Pending": "Ausstehend",
|
"Pending": "Ausstehend",
|
||||||
"Please confirm your action": "Bitte bestätigen Sie Ihre Aktion",
|
"Please confirm your action": "Bitte bestätigen Sie Ihre Aktion",
|
||||||
"Preferences": "Vorlieben",
|
"Preferences": "Vorlieben",
|
||||||
@@ -205,6 +209,9 @@
|
|||||||
"Reply...": "Antworten...",
|
"Reply...": "Antworten...",
|
||||||
"Error loading comments.": "Fehler beim Laden der Kommentare.",
|
"Error loading comments.": "Fehler beim Laden der Kommentare.",
|
||||||
"No comments yet.": "Noch keine Kommentare.",
|
"No comments yet.": "Noch keine Kommentare.",
|
||||||
|
"No open comments.": "Keine offenen Kommentare.",
|
||||||
|
"No resolved comments.": "Keine gelösten Kommentare.",
|
||||||
|
"Add a comment...": "Kommentar hinzufügen...",
|
||||||
"Edit comment": "Kommentar bearbeiten",
|
"Edit comment": "Kommentar bearbeiten",
|
||||||
"Delete comment": "Kommentar löschen",
|
"Delete comment": "Kommentar löschen",
|
||||||
"Are you sure you want to delete this comment?": "Sind Sie sicher, dass Sie diesen Kommentar löschen möchten?",
|
"Are you sure you want to delete this comment?": "Sind Sie sicher, dass Sie diesen Kommentar löschen möchten?",
|
||||||
@@ -226,7 +233,6 @@
|
|||||||
"Are you sure you want to unresolve this comment thread?": "Sind Sie sicher, dass Sie diesen Kommentarthread nicht lösen möchten?",
|
"Are you sure you want to unresolve this comment thread?": "Sind Sie sicher, dass Sie diesen Kommentarthread nicht lösen möchten?",
|
||||||
"Resolved": "Gelöst",
|
"Resolved": "Gelöst",
|
||||||
"No active comments.": "Keine aktiven Kommentare.",
|
"No active comments.": "Keine aktiven Kommentare.",
|
||||||
"No resolved comments.": "Keine gelösten Kommentare.",
|
|
||||||
"Revoke invitation": "Einladung widerrufen",
|
"Revoke invitation": "Einladung widerrufen",
|
||||||
"Revoke": "Widerrufen",
|
"Revoke": "Widerrufen",
|
||||||
"Don't": "Nicht",
|
"Don't": "Nicht",
|
||||||
@@ -272,6 +278,7 @@
|
|||||||
"Add row below": "Zeile unten hinzufügen",
|
"Add row below": "Zeile unten hinzufügen",
|
||||||
"Delete table": "Tabelle löschen",
|
"Delete table": "Tabelle löschen",
|
||||||
"Info": "Info",
|
"Info": "Info",
|
||||||
|
"Note": "Hinweis",
|
||||||
"Success": "Erfolg",
|
"Success": "Erfolg",
|
||||||
"Warning": "Warnung",
|
"Warning": "Warnung",
|
||||||
"Danger": "Gefahr",
|
"Danger": "Gefahr",
|
||||||
@@ -282,6 +289,11 @@
|
|||||||
"Save & Exit": "Speichern & Beenden",
|
"Save & Exit": "Speichern & Beenden",
|
||||||
"Double-click to edit Excalidraw diagram": "Zum Bearbeiten des Excalidraw-Diagramms doppelklicken",
|
"Double-click to edit Excalidraw diagram": "Zum Bearbeiten des Excalidraw-Diagramms doppelklicken",
|
||||||
"Paste link": "Link einfügen",
|
"Paste link": "Link einfügen",
|
||||||
|
"Paste link or search pages": "Link einfügen oder Seiten durchsuchen",
|
||||||
|
"Link to web page": "Link zur Webseite",
|
||||||
|
"Recents": "Zuletzt verwendet",
|
||||||
|
"Page or URL": "Seite oder URL",
|
||||||
|
"Link title": "Linktitel",
|
||||||
"Edit link": "Link bearbeiten",
|
"Edit link": "Link bearbeiten",
|
||||||
"Remove link": "Link entfernen",
|
"Remove link": "Link entfernen",
|
||||||
"Add link": "Link hinzufügen",
|
"Add link": "Link hinzufügen",
|
||||||
@@ -329,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Horizontale Trennlinie einfügen",
|
"Insert horizontal rule divider": "Horizontale Trennlinie einfügen",
|
||||||
"Upload any image from your device.": "Laden Sie ein beliebiges Bild von Ihrem Gerät hoch.",
|
"Upload any image from your device.": "Laden Sie ein beliebiges Bild von Ihrem Gerät hoch.",
|
||||||
"Upload any video from your device.": "Laden Sie ein beliebiges Video von Ihrem Gerät hoch.",
|
"Upload any video from your device.": "Laden Sie ein beliebiges Video von Ihrem Gerät hoch.",
|
||||||
|
"Upload any audio from your device.": "Laden Sie beliebige Audiodateien von Ihrem Gerät hoch.",
|
||||||
"Upload any file from your device.": "Laden Sie eine beliebige Datei von Ihrem Gerät hoch.",
|
"Upload any file from your device.": "Laden Sie eine beliebige Datei von Ihrem Gerät hoch.",
|
||||||
"Uploading {{name}}": "Lade {{name}} hoch",
|
"Uploading {{name}}": "Lade {{name}} hoch",
|
||||||
"Uploading file": "Datei wird hochgeladen",
|
"Uploading file": "Datei wird hochgeladen",
|
||||||
@@ -339,6 +352,12 @@
|
|||||||
"Divider": "Trennlinie",
|
"Divider": "Trennlinie",
|
||||||
"Quote": "Zitat",
|
"Quote": "Zitat",
|
||||||
"Image": "Bild",
|
"Image": "Bild",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "PDF einbetten",
|
||||||
|
"Upload and embed a PDF file.": "Laden Sie eine PDF-Datei hoch und betten Sie sie ein.",
|
||||||
|
"Embed as PDF": "Als PDF einbetten",
|
||||||
|
"Failed to load PDF": "Fehler beim Laden der PDF",
|
||||||
|
"Convert to attachment": "In Anhang umwandeln",
|
||||||
"File attachment": "Dateianhang",
|
"File attachment": "Dateianhang",
|
||||||
"Toggle block": "Block umschalten",
|
"Toggle block": "Block umschalten",
|
||||||
"Callout": "Hinweisbox",
|
"Callout": "Hinweisbox",
|
||||||
@@ -353,9 +372,23 @@
|
|||||||
"Insert current date": "Aktuelles Datum einfügen",
|
"Insert current date": "Aktuelles Datum einfügen",
|
||||||
"Draw and sketch excalidraw diagrams": "Excalidraw-Diagramme zeichnen und skizzieren",
|
"Draw and sketch excalidraw diagrams": "Excalidraw-Diagramme zeichnen und skizzieren",
|
||||||
"Multiple": "Mehrere",
|
"Multiple": "Mehrere",
|
||||||
|
"Turn into": "In verwandeln",
|
||||||
|
"Text align": "Text ausrichten",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "\"Diese Seite wurde möglicherweise gelöscht, verschoben oder Sie haben keinen Zugriff darauf.\"",
|
||||||
|
"Go to homepage": "Zur Startseite",
|
||||||
|
"Pages you create will show up here.": "\"Die von Ihnen erstellten Seiten werden hier angezeigt.\"",
|
||||||
"Heading {{level}}": "Überschrift {{level}}",
|
"Heading {{level}}": "Überschrift {{level}}",
|
||||||
"Toggle title": "Titel umschalten",
|
"Toggle title": "Titel umschalten",
|
||||||
"Write anything. Enter \"/\" for commands": "Schreiben Sie irgendetwas. Geben Sie \"/\" für Befehle ein",
|
"Write anything. Enter \"/\" for commands": "Schreiben Sie irgendetwas. Geben Sie \"/\" für Befehle ein",
|
||||||
|
"Write...": "\"Schreiben...\"",
|
||||||
|
"Column count": "Spaltenanzahl",
|
||||||
|
"{{count}} Columns": "{count, plural, one {# Spalte} other {# Spalten}}",
|
||||||
|
"Equal columns": "Gleich breite Spalten",
|
||||||
|
"Left sidebar": "Linke Seitenleiste",
|
||||||
|
"Right sidebar": "Rechte Seitenleiste",
|
||||||
|
"Wide center": "Breiter Mittelbereich",
|
||||||
|
"Left wide": "Breiter linker Bereich",
|
||||||
|
"Right wide": "Breiter rechter Bereich",
|
||||||
"Names do not match": "Namen stimmen nicht überein",
|
"Names do not match": "Namen stimmen nicht überein",
|
||||||
"Today, {{time}}": "Heute, {{time}}",
|
"Today, {{time}}": "Heute, {{time}}",
|
||||||
"Yesterday, {{time}}": "Gestern, {{time}}",
|
"Yesterday, {{time}}": "Gestern, {{time}}",
|
||||||
@@ -378,6 +411,13 @@
|
|||||||
"Delete member": "Mitglied löschen",
|
"Delete member": "Mitglied löschen",
|
||||||
"Member deleted successfully": "Mitglied erfolgreich gelöscht",
|
"Member deleted successfully": "Mitglied erfolgreich gelöscht",
|
||||||
"Are you sure you want to delete this workspace member? This action is irreversible.": "Sind Sie sicher, dass Sie dieses Arbeitsbereichsmitglied löschen möchten? Diese Aktion ist unwiderruflich.",
|
"Are you sure you want to delete this workspace member? This action is irreversible.": "Sind Sie sicher, dass Sie dieses Arbeitsbereichsmitglied löschen möchten? Diese Aktion ist unwiderruflich.",
|
||||||
|
"Deactivate member": "Mitglied deaktivieren",
|
||||||
|
"Activate member": "Mitglied aktivieren",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "Sind Sie sicher, dass Sie dieses Mitglied des Arbeitsbereichs deaktivieren möchten? Dieses Mitglied kann danach nicht mehr auf diesen Arbeitsbereich zugreifen.",
|
||||||
|
"Are you sure you want to activate this workspace member?": "Sind Sie sicher, dass Sie dieses Mitglied des Arbeitsbereichs aktivieren möchten?",
|
||||||
|
"Deactivate": "Deaktivieren",
|
||||||
|
"Activate": "Aktivieren",
|
||||||
|
"Deactivated": "Deaktiviert",
|
||||||
"Move": "Verschieben",
|
"Move": "Verschieben",
|
||||||
"Move page": "Seite verschieben",
|
"Move page": "Seite verschieben",
|
||||||
"Move page to a different space.": "Seite in einen anderen Bereich verschieben.",
|
"Move page to a different space.": "Seite in einen anderen Bereich verschieben.",
|
||||||
@@ -405,6 +445,25 @@
|
|||||||
"Share deleted successfully": "Freigabe erfolgreich gelöscht",
|
"Share deleted successfully": "Freigabe erfolgreich gelöscht",
|
||||||
"Share not found": "Freigabe nicht gefunden",
|
"Share not found": "Freigabe nicht gefunden",
|
||||||
"Failed to share page": "Fehler beim Teilen der Seite",
|
"Failed to share page": "Fehler beim Teilen der Seite",
|
||||||
|
"Disable public sharing": "Öffentliches Teilen deaktivieren",
|
||||||
|
"Prevent members from sharing pages publicly.": "Verhindern Sie, dass Mitglieder Seiten öffentlich teilen.",
|
||||||
|
"Toggle public sharing": "Öffentliches Teilen umschalten",
|
||||||
|
"Toggle space public sharing": "Öffentliches Teilen im Bereich umschalten",
|
||||||
|
"Allow viewers to comment": "Zuschauern erlauben, Kommentare zu hinterlassen",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Erlauben Sie Zuschauern, Kommentare auf Seiten in diesem Bereich hinzuzufügen.",
|
||||||
|
"Toggle viewer comments": "Zuschauerkommentare umschalten",
|
||||||
|
"Public sharing is disabled at the workspace level": "Öffentliches Teilen ist auf der Arbeitsbereichsebene deaktiviert",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "Verhindern Sie, dass Seiten in diesem Bereich öffentlich geteilt werden.",
|
||||||
|
"Page permissions": "Seitenberechtigungen",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Steuern Sie, wer einzelne Seiten ansehen und bearbeiten kann. Verfügbar mit einer Enterprise-Lizenz.",
|
||||||
|
"Enable public sharing": "Öffentliches Teilen aktivieren",
|
||||||
|
"Are you sure you want to enable public sharing? Members will be able to share pages publicly.": "Sind Sie sicher, dass Sie das öffentliche Teilen aktivieren möchten? Mitglieder können Seiten öffentlich teilen.",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this workspace will be deleted.": "Sind Sie sicher, dass Sie das öffentliche Teilen deaktivieren möchten? Alle bestehenden Freigabelinks in diesem Arbeitsbereich werden gelöscht.",
|
||||||
|
"Are you sure you want to enable public sharing for this space?": "Sind Sie sicher, dass Sie das öffentliche Teilen für diesen Bereich aktivieren möchten?",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this space will be deleted.": "Sind Sie sicher, dass Sie das öffentliche Teilen deaktivieren möchten? Alle bestehenden Freigabelinks in diesem Bereich werden gelöscht.",
|
||||||
|
"Public sharing is disabled": "Öffentliches Teilen ist deaktiviert",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "Das öffentliche Teilen wurde auf der Arbeitsbereichsebene deaktiviert.",
|
||||||
|
"Public sharing has been disabled for this space.": "Das öffentliche Teilen wurde für diesen Bereich deaktiviert.",
|
||||||
"Copy page": "Seite kopieren",
|
"Copy page": "Seite kopieren",
|
||||||
"Copy page to a different space.": "Seite in einen anderen Bereich kopieren.",
|
"Copy page to a different space.": "Seite in einen anderen Bereich kopieren.",
|
||||||
"Page copied successfully": "Seite erfolgreich kopiert",
|
"Page copied successfully": "Seite erfolgreich kopiert",
|
||||||
@@ -487,7 +546,7 @@
|
|||||||
"Enter one of your backup codes. Each backup code can only be used once.": "Geben Sie einen Ihrer Sicherungscodes ein. Jeder Sicherungscode kann nur einmal verwendet werden.",
|
"Enter one of your backup codes. Each backup code can only be used once.": "Geben Sie einen Ihrer Sicherungscodes ein. Jeder Sicherungscode kann nur einmal verwendet werden.",
|
||||||
"Verify": "Überprüfen",
|
"Verify": "Überprüfen",
|
||||||
"Trash": "Papierkorb",
|
"Trash": "Papierkorb",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "Seiten im Papierkorb werden nach 30 Tagen endgültig gelöscht.",
|
"Pages in trash will be permanently deleted after {{count}} days.": "Seiten im Papierkorb werden nach {{count}} Tagen endgültig gelöscht.",
|
||||||
"Deleted": "Gelöscht",
|
"Deleted": "Gelöscht",
|
||||||
"No pages in trash": "Keine Seiten im Papierkorb",
|
"No pages in trash": "Keine Seiten im Papierkorb",
|
||||||
"Permanently delete page?": "Seite endgültig löschen?",
|
"Permanently delete page?": "Seite endgültig löschen?",
|
||||||
@@ -559,19 +618,138 @@
|
|||||||
"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.",
|
"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",
|
"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",
|
"Manage API keys for all users in the workspace": "Verwalten Sie API-Schlüssel für alle Benutzer im Arbeitsbereich",
|
||||||
|
"Restrict API key creation to admins": "API-Schlüsselerstellung auf Administratoren beschränken",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "Nur Administratoren und Eigentümer können neue API-Schlüssel erstellen. Bestehende Mitgliederschlüssel funktionieren weiterhin.",
|
||||||
|
"Toggle restrict API keys to admins": "Beschränkung der API-Schlüssel auf Administratoren umschalten",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "Die Erstellung von API-Schlüsseln ist durch Ihren Workspace-Administrator auf Administratoren beschränkt.",
|
||||||
"AI settings": "KI-Einstellungen",
|
"AI settings": "KI-Einstellungen",
|
||||||
"AI search": "KI-Suche",
|
"AI search": "KI-Suche",
|
||||||
"AI Answer": "KI-Antwort",
|
"AI Answer": "KI-Antwort",
|
||||||
"Ask AI": "KI fragen",
|
"Ask AI": "KI fragen",
|
||||||
"AI is thinking...": "Die KI überlegt...",
|
"AI is thinking...": "Die KI überlegt...",
|
||||||
"Ask a question...": "Fragen stellen...",
|
"Ask a question...": "Fragen stellen...",
|
||||||
"AI-powered search (Ask AI)": "KI-gestützte Suche (KI fragen)",
|
"AI Answers": "KI-Antworten",
|
||||||
|
"AI-powered search (AI Answers)": "KI-unterstützte Suche (KI-Antworten)",
|
||||||
"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.",
|
"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",
|
"Toggle AI search": "KI-Suche umschalten",
|
||||||
|
"Generative AI (Ask AI)": "Generative KI (KI fragen)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Aktivieren Sie die KI-unterstützte Inhaltserstellung im Editor. Ermöglicht Benutzern das Erzeugen, Verbessern, Übersetzen und Transformieren von Text.",
|
||||||
|
"Toggle generative AI": "Generative KI umschalten",
|
||||||
|
"Upgrade your plan": "Upgrade Ihres Plans",
|
||||||
|
"Available with a paid license": "Verfügbar mit einer kostenpflichtigen Lizenz",
|
||||||
|
"Upgrade your license tier.": "Stufen Sie Ihre Lizenz hoch.",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "KI ist nur in der Docmost Enterprise-Edition verfügbar. Kontaktieren Sie sales@docmost.com.",
|
||||||
|
"AI & MCP": "KI & MCP",
|
||||||
|
"AI": "KI",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Aktivieren Sie den MCP-Server, damit KI-Assistenten und -Tools mit den Inhalten Ihres Arbeitsbereichs interagieren können.",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP ist nur in der Docmost Enterprise-Edition verfügbar. Kontaktieren Sie sales@docmost.com.",
|
||||||
|
"MCP Server URL": "MCP-Server-URL",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Verwenden Sie Ihren API-Schlüssel zur Authentifizierung. API-Schlüssel können in Ihren Kontoeinstellungen verwaltet werden.",
|
||||||
|
"Supported tools": "Unterstützte Tools",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "In Ihrem Arbeitsbereich ist MCP aktiviert. Verwenden Sie Ihren API-Schlüssel, um KI-Assistenten anzubinden.",
|
||||||
|
"MCP server URL:": "MCP-Server-URL:",
|
||||||
|
"Learn more": "Mehr erfahren",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Verwalten Sie API-Schlüssel für alle Nutzer im Arbeitsbereich. Siehe die <anchor>API-Dokumentation</anchor> für Details zur Verwendung.",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "Siehe die <anchor>API-Dokumentation</anchor> für Details zur Verwendung.",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "Sehen Sie die <anchor>MCP-Dokumentation</anchor> ein.",
|
||||||
"Sources": "Quellen",
|
"Sources": "Quellen",
|
||||||
"Ask AI not available for attachments": "KI fragen nicht für Anhänge verfügbar",
|
"AI Answers not available for attachments": "KI-Antworten sind für Anhänge nicht verfügbar",
|
||||||
"No answer available": "Keine Antwort verfügbar",
|
"No answer available": "Keine Antwort verfügbar",
|
||||||
"Background color": "Hintergrundfarbe",
|
"Background color": "Hintergrundfarbe",
|
||||||
"Highlight color": "Hervorhebungsfarbe",
|
"Highlight color": "Hervorhebungsfarbe",
|
||||||
"Remove color": "Farbe entfernen"
|
"Remove color": "Farbe entfernen",
|
||||||
|
"Notifications": "Benachrichtigungen",
|
||||||
|
"No notifications": "Keine Benachrichtigungen",
|
||||||
|
"No unread notifications": "Keine ungelesenen Benachrichtigungen",
|
||||||
|
"All notifications": "Alle Benachrichtigungen",
|
||||||
|
"Unread only": "Nur ungelesen",
|
||||||
|
"Mark all as read": "Alle als gelesen markieren",
|
||||||
|
"Mark as read": "Als gelesen markieren",
|
||||||
|
"More options": "Weitere Optionen",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> hat Sie in einem Kommentar erwähnt",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> hat einen Kommentar auf einer Seite hinterlassen",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> hat einen Kommentar als erledigt markiert",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> hat Sie auf einer Seite erwähnt",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> hat Ihnen Bearbeitungszugriff auf eine Seite gegeben",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> hat Ihnen Ansichtsrechte für eine Seite gegeben",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> hat eine Seite aktualisiert.",
|
||||||
|
"Watch page": "Seite beobachten",
|
||||||
|
"Stop watching": "Beobachtung beenden",
|
||||||
|
"Email notifications": "E-Mail-Benachrichtigungen",
|
||||||
|
"Page updates": "Seitenaktualisierungen",
|
||||||
|
"Get notified when pages you watch are updated.": "Erhalten Sie eine Benachrichtigung, wenn Seiten, die Sie beobachten, aktualisiert werden.",
|
||||||
|
"Page mentions": "Seiten-Erwähnungen",
|
||||||
|
"Get notified when someone mentions you on a page.": "Erhalten Sie eine Benachrichtigung, wenn Sie jemand auf einer Seite erwähnt.",
|
||||||
|
"Comment mentions": "Kommentar-Erwähnungen",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Erhalten Sie eine Benachrichtigung, wenn Sie jemand in einem Kommentar erwähnt.",
|
||||||
|
"New comments": "Neue Kommentare",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Erhalten Sie eine Benachrichtigung über neue Kommentare in Threads, an denen Sie teilnehmen.",
|
||||||
|
"Resolved comments": "Erledigte Kommentare",
|
||||||
|
"Get notified when your comment is resolved.": "Erhalten Sie eine Benachrichtigung, wenn Ihr Kommentar erledigt wurde.",
|
||||||
|
"You are now watching this page": "Sie beobachten diese Seite jetzt",
|
||||||
|
"You are no longer watching this page": "Sie beobachten diese Seite nicht mehr",
|
||||||
|
"Direct": "Direkt",
|
||||||
|
"Updates": "Aktualisierungen",
|
||||||
|
"Today": "Heute",
|
||||||
|
"Yesterday": "Gestern",
|
||||||
|
"This week": "Diese Woche",
|
||||||
|
"Older": "Älter",
|
||||||
|
"Restricted page": "Eingeschränkte Seite",
|
||||||
|
"Restricted pages cannot be shared publicly.": "Eingeschränkte Seiten können nicht öffentlich geteilt werden.",
|
||||||
|
"Restricted by parent": "Eingeschränkt durch die übergeordnete Seite",
|
||||||
|
"Restricted": "Eingeschränkt",
|
||||||
|
"Open": "Offen",
|
||||||
|
"Inherits restrictions from ancestor page": "Erbt Einschränkungen von einer übergeordneten Seite",
|
||||||
|
"Only people listed below can access this page": "Nur die unten aufgeführten Personen können auf diese Seite zugreifen.",
|
||||||
|
"Everyone in this space can access": "Jeder in diesem Bereich kann darauf zugreifen.",
|
||||||
|
"No additional restrictions on this page": "Keine zusätzlichen Einschränkungen auf dieser Seite",
|
||||||
|
"Only specific people can access": "Nur bestimmte Personen können zugreifen",
|
||||||
|
"Use only inherited restrictions": "Nur geerbte Einschränkungen verwenden",
|
||||||
|
"Add restrictions on top of inherited": "Einschränkungen zusätzlich zu den geerbten hinzufügen",
|
||||||
|
"Inherited restriction": "Geerbte Einschränkung",
|
||||||
|
"Access limited by": "Zugriff beschränkt durch",
|
||||||
|
"Restrict access to control who can view and edit this page": "Beschränken Sie den Zugriff, um festzulegen, wer diese Seite ansehen und bearbeiten kann.",
|
||||||
|
"Add additional restrictions specific to this page": "Fügen Sie zusätzliche, für diese Seite spezifische Einschränkungen hinzu.",
|
||||||
|
"Access": "Zugriff",
|
||||||
|
"People with access": "Personen mit Zugriff",
|
||||||
|
"Remove all": "Alle entfernen",
|
||||||
|
"Remove access": "Zugriff entfernen",
|
||||||
|
"Remove all access": "Alle Zugriffsrechte entfernen",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "Sind Sie sicher, dass Sie den Zugriff dieses Mitglieds auf die Seite entfernen möchten?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "Sind Sie sicher, dass Sie alle spezifischen Zugriffsrechte entfernen möchten? Dadurch wird die Seite für alle in diesem Bereich zugänglich.",
|
||||||
|
"Trash retention": "Aufbewahrungsdauer des Papierkorbs",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "Seiten im Papierkorb werden nach Ablauf dieses Zeitraums endgültig gelöscht.",
|
||||||
|
"Trash retention updated": "Aufbewahrungsdauer des Papierkorbs aktualisiert",
|
||||||
|
"Failed to update trash retention": "Aktualisierung der Aufbewahrungsdauer des Papierkorbs fehlgeschlagen",
|
||||||
|
"Removed page restriction": "Seitenbeschränkung entfernt",
|
||||||
|
"Added page permission": "Seitenberechtigung hinzugefügt",
|
||||||
|
"Removed page permission": "Seitenberechtigung entfernt",
|
||||||
|
"Verifying your email": "E-Mail wird überprüft",
|
||||||
|
"Please wait...": "Bitte warten...",
|
||||||
|
"Verification failed. The link may have expired.": "Überprüfung fehlgeschlagen. Der Link ist möglicherweise abgelaufen.",
|
||||||
|
"Check your email": "Prüfen Sie Ihr E-Mail-Postfach",
|
||||||
|
"We sent a verification link to {{email}}.": "Wir haben einen Bestätigungslink an {{email}} gesendet.",
|
||||||
|
"We sent a verification link to your email.": "Wir haben einen Bestätigungslink an Ihre E-Mail-Adresse gesendet.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Klicken Sie auf den Link, um Ihre E-Mail zu bestätigen und auf Ihren Arbeitsbereich zuzugreifen.",
|
||||||
|
"Resend verification email": "Bestätigungs-E-Mail erneut senden",
|
||||||
|
"Verification email sent. Please check your inbox.": "Bestätigungs-E-Mail gesendet. Bitte überprüfen Sie Ihr Postfach.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Fehler beim erneuten Senden der Bestätigungs-E-Mail. Bitte versuchen Sie es erneut.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Wir haben Ihnen eine E-Mail mit Ihren zugehörigen Arbeitsbereichen gesendet.",
|
||||||
|
"Load more": "Mehr laden",
|
||||||
|
"Log out of all devices": "Von allen Geräten abmelden",
|
||||||
|
"Log out of all sessions except this device": "Von allen Sitzungen außer diesem Gerät abmelden",
|
||||||
|
"This Device": "Dieses Gerät",
|
||||||
|
"Unknown device": "Unbekanntes Gerät",
|
||||||
|
"No active sessions": "Keine aktiven Sitzungen",
|
||||||
|
"Session revoked": "Sitzung widerrufen",
|
||||||
|
"All other sessions revoked": "Alle anderen Sitzungen widerrufen",
|
||||||
|
"Last used": "Zuletzt verwendet",
|
||||||
|
"Created": "Erstellt",
|
||||||
|
"Rename": "Umbenennen",
|
||||||
|
"Publish": "Veröffentlichen",
|
||||||
|
"Security": "Sicherheit",
|
||||||
|
"Enforce SSO": "SSO erzwingen",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Nach dem Erzwingen können sich Mitglieder nicht mehr mit E-Mail und Passwort anmelden."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,6 +7,7 @@
|
|||||||
"Add members": "Add members",
|
"Add members": "Add members",
|
||||||
"Add to groups": "Add to groups",
|
"Add to groups": "Add to groups",
|
||||||
"Add space members": "Add space members",
|
"Add space members": "Add space members",
|
||||||
|
"Add to favorites": "Add to favorites",
|
||||||
"Admin": "Admin",
|
"Admin": "Admin",
|
||||||
"Are you sure you want to delete this group? Members will lose access to resources this group has access to.": "Are you sure you want to delete this group? Members will lose access to resources this group has access to.",
|
"Are you sure you want to delete this group? Members will lose access to resources this group has access to.": "Are you sure you want to delete this group? Members will lose access to resources this group has access to.",
|
||||||
"Are you sure you want to delete this page?": "Are you sure you want to delete this page?",
|
"Are you sure you want to delete this page?": "Are you sure you want to delete this page?",
|
||||||
@@ -74,6 +75,9 @@
|
|||||||
"Failed to import pages": "Failed to import pages",
|
"Failed to import pages": "Failed to import pages",
|
||||||
"Failed to load page. An error occurred.": "Failed to load page. An error occurred.",
|
"Failed to load page. An error occurred.": "Failed to load page. An error occurred.",
|
||||||
"Failed to update data": "Failed to update data",
|
"Failed to update data": "Failed to update data",
|
||||||
|
"Favorite spaces": "Favorite spaces",
|
||||||
|
"Favorite spaces appear here": "Favorite spaces appear here",
|
||||||
|
"Favorites": "Favorites",
|
||||||
"Full access": "Full access",
|
"Full access": "Full access",
|
||||||
"Full page width": "Full page width",
|
"Full page width": "Full page width",
|
||||||
"Full width": "Full width",
|
"Full width": "Full width",
|
||||||
@@ -92,6 +96,7 @@
|
|||||||
"Invite by email": "Invite by email",
|
"Invite by email": "Invite by email",
|
||||||
"Invite members": "Invite members",
|
"Invite members": "Invite members",
|
||||||
"Invite new members": "Invite new members",
|
"Invite new members": "Invite new members",
|
||||||
|
"Invite People": "Invite People",
|
||||||
"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 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",
|
"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",
|
"Join the workspace": "Join the workspace",
|
||||||
@@ -116,6 +121,7 @@
|
|||||||
"No group found": "No group found",
|
"No group found": "No group found",
|
||||||
"No page history saved yet.": "No page history saved yet.",
|
"No page history saved yet.": "No page history saved yet.",
|
||||||
"No pages yet": "No pages yet",
|
"No pages yet": "No pages yet",
|
||||||
|
"No shared pages": "No shared pages",
|
||||||
"No results found...": "No results found...",
|
"No results found...": "No results found...",
|
||||||
"No user found": "No user found",
|
"No user found": "No user found",
|
||||||
"Overview": "Overview",
|
"Overview": "Overview",
|
||||||
@@ -123,11 +129,14 @@
|
|||||||
"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",
|
||||||
"Password": "Password",
|
"Password": "Password",
|
||||||
"Password changed successfully": "Password changed successfully",
|
"Password changed successfully": "Password changed successfully",
|
||||||
|
"People": "People",
|
||||||
"Pending": "Pending",
|
"Pending": "Pending",
|
||||||
"Please confirm your action": "Please confirm your action",
|
"Please confirm your action": "Please confirm your action",
|
||||||
"Preferences": "Preferences",
|
"Preferences": "Preferences",
|
||||||
@@ -135,6 +144,7 @@
|
|||||||
"Profile": "Profile",
|
"Profile": "Profile",
|
||||||
"Recently updated": "Recently updated",
|
"Recently updated": "Recently updated",
|
||||||
"Remove": "Remove",
|
"Remove": "Remove",
|
||||||
|
"Remove from favorites": "Remove from favorites",
|
||||||
"Remove group member": "Remove group member",
|
"Remove group member": "Remove group member",
|
||||||
"Remove space member": "Remove space member",
|
"Remove space member": "Remove space member",
|
||||||
"Restore": "Restore",
|
"Restore": "Restore",
|
||||||
@@ -171,6 +181,7 @@
|
|||||||
"Successfully imported": "Successfully imported",
|
"Successfully imported": "Successfully imported",
|
||||||
"Successfully restored": "Successfully restored",
|
"Successfully restored": "Successfully restored",
|
||||||
"System settings": "System settings",
|
"System settings": "System settings",
|
||||||
|
"Templates": "Templates",
|
||||||
"Theme": "Theme",
|
"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.",
|
"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",
|
"Toggle full page width": "Toggle full page width",
|
||||||
@@ -205,6 +216,9 @@
|
|||||||
"Reply...": "Reply...",
|
"Reply...": "Reply...",
|
||||||
"Error loading comments.": "Error loading comments.",
|
"Error loading comments.": "Error loading comments.",
|
||||||
"No comments yet.": "No comments yet.",
|
"No comments yet.": "No comments yet.",
|
||||||
|
"No open comments.": "No open comments.",
|
||||||
|
"No resolved comments.": "No resolved comments.",
|
||||||
|
"Add a comment...": "Add a comment...",
|
||||||
"Edit comment": "Edit comment",
|
"Edit comment": "Edit comment",
|
||||||
"Delete comment": "Delete comment",
|
"Delete comment": "Delete comment",
|
||||||
"Are you sure you want to delete this comment?": "Are you sure you want to delete this comment?",
|
"Are you sure you want to delete this comment?": "Are you sure you want to delete this comment?",
|
||||||
@@ -226,7 +240,6 @@
|
|||||||
"Are you sure you want to unresolve this comment thread?": "Are you sure you want to unresolve this comment thread?",
|
"Are you sure you want to unresolve this comment thread?": "Are you sure you want to unresolve this comment thread?",
|
||||||
"Resolved": "Resolved",
|
"Resolved": "Resolved",
|
||||||
"No active comments.": "No active comments.",
|
"No active comments.": "No active comments.",
|
||||||
"No resolved comments.": "No resolved comments.",
|
|
||||||
"Revoke invitation": "Revoke invitation",
|
"Revoke invitation": "Revoke invitation",
|
||||||
"Revoke": "Revoke",
|
"Revoke": "Revoke",
|
||||||
"Don't": "Don't",
|
"Don't": "Don't",
|
||||||
@@ -272,6 +285,7 @@
|
|||||||
"Add row below": "Add row below",
|
"Add row below": "Add row below",
|
||||||
"Delete table": "Delete table",
|
"Delete table": "Delete table",
|
||||||
"Info": "Info",
|
"Info": "Info",
|
||||||
|
"Note": "Note",
|
||||||
"Success": "Success",
|
"Success": "Success",
|
||||||
"Warning": "Warning",
|
"Warning": "Warning",
|
||||||
"Danger": "Danger",
|
"Danger": "Danger",
|
||||||
@@ -282,6 +296,11 @@
|
|||||||
"Save & Exit": "Save & Exit",
|
"Save & Exit": "Save & Exit",
|
||||||
"Double-click to edit Excalidraw diagram": "Double-click to edit Excalidraw diagram",
|
"Double-click to edit Excalidraw diagram": "Double-click to edit Excalidraw diagram",
|
||||||
"Paste link": "Paste link",
|
"Paste link": "Paste link",
|
||||||
|
"Paste link or search pages": "Paste link or search pages",
|
||||||
|
"Link to web page": "Link to web page",
|
||||||
|
"Recents": "Recents",
|
||||||
|
"Page or URL": "Page or URL",
|
||||||
|
"Link title": "Link title",
|
||||||
"Edit link": "Edit link",
|
"Edit link": "Edit link",
|
||||||
"Remove link": "Remove link",
|
"Remove link": "Remove link",
|
||||||
"Add link": "Add link",
|
"Add link": "Add link",
|
||||||
@@ -329,6 +348,7 @@
|
|||||||
"Insert horizontal rule divider": "Insert horizontal rule divider",
|
"Insert horizontal rule divider": "Insert horizontal rule divider",
|
||||||
"Upload any image from your device.": "Upload any image from your device.",
|
"Upload any image from your device.": "Upload any image from your device.",
|
||||||
"Upload any video from your device.": "Upload any video from your device.",
|
"Upload any video from your device.": "Upload any video from your device.",
|
||||||
|
"Upload any audio from your device.": "Upload any audio from your device.",
|
||||||
"Upload any file from your device.": "Upload any file from your device.",
|
"Upload any file from your device.": "Upload any file from your device.",
|
||||||
"Uploading {{name}}": "Uploading {{name}}",
|
"Uploading {{name}}": "Uploading {{name}}",
|
||||||
"Uploading file": "Uploading file",
|
"Uploading file": "Uploading file",
|
||||||
@@ -339,6 +359,12 @@
|
|||||||
"Divider": "Divider",
|
"Divider": "Divider",
|
||||||
"Quote": "Quote",
|
"Quote": "Quote",
|
||||||
"Image": "Image",
|
"Image": "Image",
|
||||||
|
"Audio": "Audio",
|
||||||
|
"Embed PDF": "Embed PDF",
|
||||||
|
"Upload and embed a PDF file.": "Upload and embed a PDF file.",
|
||||||
|
"Embed as PDF": "Embed as PDF",
|
||||||
|
"Failed to load PDF": "Failed to load PDF",
|
||||||
|
"Convert to attachment": "Convert to attachment",
|
||||||
"File attachment": "File attachment",
|
"File attachment": "File attachment",
|
||||||
"Toggle block": "Toggle block",
|
"Toggle block": "Toggle block",
|
||||||
"Callout": "Callout",
|
"Callout": "Callout",
|
||||||
@@ -353,9 +379,23 @@
|
|||||||
"Insert current date": "Insert current date",
|
"Insert current date": "Insert current date",
|
||||||
"Draw and sketch excalidraw diagrams": "Draw and sketch excalidraw diagrams",
|
"Draw and sketch excalidraw diagrams": "Draw and sketch excalidraw diagrams",
|
||||||
"Multiple": "Multiple",
|
"Multiple": "Multiple",
|
||||||
|
"Turn into": "Turn into",
|
||||||
|
"Text align": "Text align",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "This page may have been deleted, moved, or you may not have access.",
|
||||||
|
"Go to homepage": "Go to homepage",
|
||||||
|
"Pages you create will show up here.": "Pages you create will show up here.",
|
||||||
"Heading {{level}}": "Heading {{level}}",
|
"Heading {{level}}": "Heading {{level}}",
|
||||||
"Toggle title": "Toggle title",
|
"Toggle title": "Toggle title",
|
||||||
"Write anything. Enter \"/\" for commands": "Write anything. Enter \"/\" for commands",
|
"Write anything. Enter \"/\" for commands": "Write anything. Enter \"/\" for commands",
|
||||||
|
"Write...": "Write...",
|
||||||
|
"Column count": "Column count",
|
||||||
|
"{{count}} Columns": "{{count}} Columns",
|
||||||
|
"Equal columns": "Equal columns",
|
||||||
|
"Left sidebar": "Left sidebar",
|
||||||
|
"Right sidebar": "Right sidebar",
|
||||||
|
"Wide center": "Wide center",
|
||||||
|
"Left wide": "Left wide",
|
||||||
|
"Right wide": "Right wide",
|
||||||
"Names do not match": "Names do not match",
|
"Names do not match": "Names do not match",
|
||||||
"Today, {{time}}": "Today, {{time}}",
|
"Today, {{time}}": "Today, {{time}}",
|
||||||
"Yesterday, {{time}}": "Yesterday, {{time}}",
|
"Yesterday, {{time}}": "Yesterday, {{time}}",
|
||||||
@@ -378,6 +418,13 @@
|
|||||||
"Delete member": "Delete member",
|
"Delete member": "Delete member",
|
||||||
"Member deleted successfully": "Member deleted successfully",
|
"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.",
|
"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.",
|
||||||
|
"Deactivate member": "Deactivate member",
|
||||||
|
"Activate member": "Activate member",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.",
|
||||||
|
"Are you sure you want to activate this workspace member?": "Are you sure you want to activate this workspace member?",
|
||||||
|
"Deactivate": "Deactivate",
|
||||||
|
"Activate": "Activate",
|
||||||
|
"Deactivated": "Deactivated",
|
||||||
"Move": "Move",
|
"Move": "Move",
|
||||||
"Move page": "Move page",
|
"Move page": "Move page",
|
||||||
"Move page to a different space.": "Move page to a different space.",
|
"Move page to a different space.": "Move page to a different space.",
|
||||||
@@ -405,6 +452,25 @@
|
|||||||
"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",
|
||||||
|
"Allow viewers to comment": "Allow viewers to comment",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Allow viewers to add comments on pages in this space.",
|
||||||
|
"Toggle viewer comments": "Toggle viewer comments",
|
||||||
|
"Public sharing is disabled at the workspace level": "Public sharing is disabled at the workspace level",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "Prevent pages in this space from being shared publicly.",
|
||||||
|
"Page permissions": "Page permissions",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Control who can view and edit individual pages. Available with an enterprise license.",
|
||||||
|
"Enable public sharing": "Enable public sharing",
|
||||||
|
"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",
|
||||||
@@ -419,6 +485,7 @@
|
|||||||
"Replace (Enter)": "Replace (Enter)",
|
"Replace (Enter)": "Replace (Enter)",
|
||||||
"Replace all (Ctrl+Alt+Enter)": "Replace all (Ctrl+Alt+Enter)",
|
"Replace all (Ctrl+Alt+Enter)": "Replace all (Ctrl+Alt+Enter)",
|
||||||
"Replace all": "Replace all",
|
"Replace all": "Replace all",
|
||||||
|
"View all": "View all",
|
||||||
"View all spaces": "View all spaces",
|
"View all spaces": "View all spaces",
|
||||||
"Error": "Error",
|
"Error": "Error",
|
||||||
"Failed to disable MFA": "Failed to disable MFA",
|
"Failed to disable MFA": "Failed to disable MFA",
|
||||||
@@ -487,7 +554,7 @@
|
|||||||
"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.",
|
"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.",
|
||||||
"Verify": "Verify",
|
"Verify": "Verify",
|
||||||
"Trash": "Trash",
|
"Trash": "Trash",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "Pages in trash will be permanently deleted after 30 days.",
|
"Pages in trash will be permanently deleted after {{count}} days.": "Pages in trash will be permanently deleted after {{count}} days.",
|
||||||
"Deleted": "Deleted",
|
"Deleted": "Deleted",
|
||||||
"No pages in trash": "No pages in trash",
|
"No pages in trash": "No pages in trash",
|
||||||
"Permanently delete page?": "Permanently delete page?",
|
"Permanently delete page?": "Permanently delete page?",
|
||||||
@@ -559,19 +626,170 @@
|
|||||||
"This action cannot be undone. Any applications using this API key will stop working.": "This action cannot be undone. Any applications using this API key will stop working.",
|
"This action cannot be undone. Any applications using this API key will stop working.": "This action cannot be undone. Any applications using this API key will stop working.",
|
||||||
"Update API key": "Update API key",
|
"Update API key": "Update API key",
|
||||||
"Manage API keys for all users in the workspace": "Manage API keys for all users in the workspace",
|
"Manage API keys for all users in the workspace": "Manage API keys for all users in the workspace",
|
||||||
|
"Restrict API key creation to admins": "Restrict API key creation to admins",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "Only admins and owners can create new API keys. Existing member keys will continue to work.",
|
||||||
|
"Toggle restrict API keys to admins": "Toggle restrict API keys to admins",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "API key creation is restricted to admins by your workspace administrator.",
|
||||||
"AI settings": "AI settings",
|
"AI settings": "AI settings",
|
||||||
"AI search": "AI search",
|
"AI search": "AI search",
|
||||||
"AI Answer": "AI Answer",
|
"AI Answer": "AI Answer",
|
||||||
"Ask AI": "Ask AI",
|
"Ask AI": "Ask AI",
|
||||||
"AI is thinking...": "AI is thinking...",
|
"AI is thinking...": "AI is thinking...",
|
||||||
|
"Thinking": "Thinking",
|
||||||
"Ask a question...": "Ask a question...",
|
"Ask a question...": "Ask a question...",
|
||||||
"AI-powered search (Ask AI)": "AI-powered search (Ask AI)",
|
"AI Answers": "AI Answers",
|
||||||
|
"AI-powered search (AI Answers)": "AI-powered search (AI Answers)",
|
||||||
"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.",
|
"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",
|
"Toggle AI search": "Toggle AI search",
|
||||||
|
"Generative AI (Ask AI)": "Generative AI (Ask AI)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.",
|
||||||
|
"Toggle generative AI": "Toggle generative AI",
|
||||||
|
"Upgrade your plan": "Upgrade your plan",
|
||||||
|
"Available with a paid license": "Available with a paid license",
|
||||||
|
"Upgrade your license tier.": "Upgrade your license tier.",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
||||||
|
"AI & MCP": "AI & MCP",
|
||||||
|
"AI": "AI",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Enable the MCP server to allow AI assistants and tools to interact with your workspace content.",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
||||||
|
"MCP Server URL": "MCP Server URL",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Use your API key for authentication. You can manage API keys in your account settings.",
|
||||||
|
"Supported tools": "Supported tools",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Your workspace has MCP enabled. Use your API key to connect AI assistants.",
|
||||||
|
"MCP server URL:": "MCP server URL:",
|
||||||
|
"Learn more": "Learn more",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "View the <anchor>API documentation</anchor> for usage details.",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "View the <anchor>MCP documentation</anchor>.",
|
||||||
"Sources": "Sources",
|
"Sources": "Sources",
|
||||||
"Ask AI not available for attachments": "Ask AI not available for attachments",
|
"AI Answers not available for attachments": "AI Answers not available for attachments",
|
||||||
"No answer available": "No answer available",
|
"No answer available": "No answer available",
|
||||||
"Background color": "Background color",
|
"Background color": "Background color",
|
||||||
"Highlight color": "Highlight color",
|
"Highlight color": "Highlight color",
|
||||||
"Remove color": "Remove color"
|
"Remove color": "Remove color",
|
||||||
|
"Notifications": "Notifications",
|
||||||
|
"No notifications": "No notifications",
|
||||||
|
"No unread notifications": "No unread notifications",
|
||||||
|
"All notifications": "All notifications",
|
||||||
|
"Unread only": "Unread only",
|
||||||
|
"Mark all as read": "Mark all as read",
|
||||||
|
"Mark as read": "Mark as read",
|
||||||
|
"More options": "More options",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> mentioned you in a comment",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> commented on a page",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> resolved a comment",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> mentioned you on a page",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> gave you edit access to a page",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> gave you view access to a page",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> updated a page",
|
||||||
|
"Watch page": "Watch page",
|
||||||
|
"Stop watching": "Stop watching",
|
||||||
|
"Watch space": "Watch space",
|
||||||
|
"Stop watching space": "Stop watching space",
|
||||||
|
"Email notifications": "Email notifications",
|
||||||
|
"Page updates": "Page updates",
|
||||||
|
"Get notified when pages you watch are updated.": "Receive notifications when the pages you watch are updated.",
|
||||||
|
"Page mentions": "Page mentions",
|
||||||
|
"Get notified when someone mentions you on a page.": "Receive notifications when someone mentions you on a page.",
|
||||||
|
"Comment mentions": "Comment mentions",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Receive notifications when someone mentions you in a comment.",
|
||||||
|
"New comments": "New comments",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Receive notifications about new comments in threads you are participating in.",
|
||||||
|
"Resolved comments": "Resolved comments",
|
||||||
|
"Get notified when your comment is resolved.": "Receive a notification when your comment is resolved.",
|
||||||
|
"You are now watching this page": "You’re now watching this page",
|
||||||
|
"You are no longer watching this page": "You’re no longer watching this page",
|
||||||
|
"You are now watching this space": "You’re now watching this space",
|
||||||
|
"You are no longer watching this space": "You’re no longer watching this space",
|
||||||
|
"Direct": "Direct",
|
||||||
|
"Updates": "Updates",
|
||||||
|
"Today": "Today",
|
||||||
|
"Yesterday": "Yesterday",
|
||||||
|
"This week": "This week",
|
||||||
|
"Older": "Older",
|
||||||
|
"Restricted page": "Restricted page",
|
||||||
|
"Restricted pages cannot be shared publicly.": "Restricted pages cannot be shared publicly.",
|
||||||
|
"Restricted by parent": "Restricted by parent",
|
||||||
|
"Restricted": "Restricted",
|
||||||
|
"Open": "Open",
|
||||||
|
"Inherits restrictions from ancestor page": "Inherits restrictions from ancestor page",
|
||||||
|
"Only people listed below can access this page": "Only people listed below can access this page",
|
||||||
|
"Everyone in this space can access": "Everyone in this space can access",
|
||||||
|
"No additional restrictions on this page": "No additional restrictions on this page",
|
||||||
|
"Only specific people can access": "Only specific people can access",
|
||||||
|
"Use only inherited restrictions": "Use only inherited restrictions",
|
||||||
|
"Add restrictions on top of inherited": "Add restrictions on top of inherited",
|
||||||
|
"Inherited restriction": "Inherited restriction",
|
||||||
|
"Access limited by": "Access limited by",
|
||||||
|
"Restrict access to control who can view and edit this page": "Restrict access to control who can view and edit this page",
|
||||||
|
"Add additional restrictions specific to this page": "Add additional restrictions specific to this page",
|
||||||
|
"Access": "Access",
|
||||||
|
"People with access": "People with access",
|
||||||
|
"Remove all": "Remove all",
|
||||||
|
"Remove access": "Remove access",
|
||||||
|
"Remove all access": "Remove all access",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "Are you sure you want to remove this member's access to the page?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "Are you sure you want to remove all specific access? This will make the page open to everyone in the space.",
|
||||||
|
"Trash retention": "Trash retention",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "Pages in trash will be permanently deleted after this period.",
|
||||||
|
"Trash retention updated": "Trash retention updated",
|
||||||
|
"Failed to update trash retention": "Failed to update trash retention",
|
||||||
|
"Removed page restriction": "Removed page restriction",
|
||||||
|
"Added page permission": "Added page permission",
|
||||||
|
"Removed page permission": "Removed page permission",
|
||||||
|
"Verifying your email": "Verifying your email",
|
||||||
|
"Please wait...": "Please wait...",
|
||||||
|
"Verification failed. The link may have expired.": "Verification failed. The link may have expired.",
|
||||||
|
"Check your email": "Check your email",
|
||||||
|
"We sent a verification link to {{email}}.": "We sent a verification link to {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "We sent a verification link to your email.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Click the link to verify your email and access your workspace.",
|
||||||
|
"Resend verification email": "Resend verification email",
|
||||||
|
"Verification email sent. Please check your inbox.": "Verification email sent. Please check your inbox.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Failed to resend verification email. Please try again.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "We've sent you an email with your associated workspaces.",
|
||||||
|
"Load more": "Load more",
|
||||||
|
"Log out of all devices": "Log out of all devices",
|
||||||
|
"Log out of all sessions except this device": "Log out of all sessions except this device",
|
||||||
|
"This Device": "This Device",
|
||||||
|
"Unknown device": "Unknown device",
|
||||||
|
"No active sessions": "No active sessions",
|
||||||
|
"Session revoked": "Session revoked",
|
||||||
|
"All other sessions revoked": "All other sessions revoked",
|
||||||
|
"Last used": "Last used",
|
||||||
|
"Created": "Created",
|
||||||
|
"Rename": "Rename",
|
||||||
|
"Publish": "Publish",
|
||||||
|
"Security": "Security",
|
||||||
|
"Enforce SSO": "Enforce SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Once enforced, members will not be able to login with email and password.",
|
||||||
|
"AI-generated content may not be accurate.": "AI-generated content may not be accurate.",
|
||||||
|
"AI Chat": "AI Chat",
|
||||||
|
"Analyze for insights": "Analyze for insights",
|
||||||
|
"Ask anything...": "Ask anything...",
|
||||||
|
"Chat history": "Chat history",
|
||||||
|
"Chat name": "Chat name",
|
||||||
|
"Close": "Close",
|
||||||
|
"Docmost AI": "Docmost AI",
|
||||||
|
"Failed to load chat. An error occurred.": "Failed to load chat. An error occurred.",
|
||||||
|
"Failed to render this message.": "Failed to render this message.",
|
||||||
|
"How can I help you today?": "How can I help you today?",
|
||||||
|
"New chat": "New chat",
|
||||||
|
"No chat history": "No chat history",
|
||||||
|
"No chats found": "No chats found",
|
||||||
|
"No conversations yet": "No conversations yet",
|
||||||
|
"Open full page": "Open full page",
|
||||||
|
"Previous 7 days": "Previous 7 days",
|
||||||
|
"Previous 30 days": "Previous 30 days",
|
||||||
|
"Search chats...": "Search chats...",
|
||||||
|
"Start a new chat to see it here.": "Start a new chat to see it here.",
|
||||||
|
"Summarize this page": "Summarize this page",
|
||||||
|
"Toggle AI Chat": "Toggle AI Chat",
|
||||||
|
"Translate this page": "Translate this page",
|
||||||
|
"Try a different search term.": "Try a different search term.",
|
||||||
|
"Try again": "Try again",
|
||||||
|
"Untitled chat": "Untitled chat",
|
||||||
|
"What can I help you with?": "What can I help you with?"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -116,6 +116,7 @@
|
|||||||
"No group found": "No se encontró grupo",
|
"No group found": "No se encontró grupo",
|
||||||
"No page history saved yet.": "No hay historial de la página guardado aún.",
|
"No page history saved yet.": "No hay historial de la página guardado aún.",
|
||||||
"No pages yet": "No hay páginas todavía",
|
"No pages yet": "No hay páginas todavía",
|
||||||
|
"No shared pages": "No hay páginas compartidas",
|
||||||
"No results found...": "No se encontraron resultados...",
|
"No results found...": "No se encontraron resultados...",
|
||||||
"No user found": "No se encontró usuario",
|
"No user found": "No se encontró usuario",
|
||||||
"Overview": "Visión general",
|
"Overview": "Visión general",
|
||||||
@@ -123,11 +124,14 @@
|
|||||||
"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",
|
||||||
"Password": "Contraseña",
|
"Password": "Contraseña",
|
||||||
"Password changed successfully": "Contraseña cambiada con éxito",
|
"Password changed successfully": "Contraseña cambiada con éxito",
|
||||||
|
"People": "Personas",
|
||||||
"Pending": "Pendiente",
|
"Pending": "Pendiente",
|
||||||
"Please confirm your action": "Por favor, confirme su acción",
|
"Please confirm your action": "Por favor, confirme su acción",
|
||||||
"Preferences": "Preferencias",
|
"Preferences": "Preferencias",
|
||||||
@@ -205,6 +209,9 @@
|
|||||||
"Reply...": "Responder...",
|
"Reply...": "Responder...",
|
||||||
"Error loading comments.": "Error al cargar comentarios.",
|
"Error loading comments.": "Error al cargar comentarios.",
|
||||||
"No comments yet.": "No hay comentarios todavía.",
|
"No comments yet.": "No hay comentarios todavía.",
|
||||||
|
"No open comments.": "No hay comentarios abiertos.",
|
||||||
|
"No resolved comments.": "No hay comentarios resueltos.",
|
||||||
|
"Add a comment...": "Agregar un comentario...",
|
||||||
"Edit comment": "Editar comentario",
|
"Edit comment": "Editar comentario",
|
||||||
"Delete comment": "Eliminar comentario",
|
"Delete comment": "Eliminar comentario",
|
||||||
"Are you sure you want to delete this comment?": "¿Está seguro de que desea eliminar este comentario?",
|
"Are you sure you want to delete this comment?": "¿Está seguro de que desea eliminar este comentario?",
|
||||||
@@ -226,7 +233,6 @@
|
|||||||
"Are you sure you want to unresolve this comment thread?": "¿Está seguro de que desea no resolver este hilo de comentarios?",
|
"Are you sure you want to unresolve this comment thread?": "¿Está seguro de que desea no resolver este hilo de comentarios?",
|
||||||
"Resolved": "Resuelto",
|
"Resolved": "Resuelto",
|
||||||
"No active comments.": "No hay comentarios activos.",
|
"No active comments.": "No hay comentarios activos.",
|
||||||
"No resolved comments.": "No hay comentarios resueltos.",
|
|
||||||
"Revoke invitation": "Revocar invitación",
|
"Revoke invitation": "Revocar invitación",
|
||||||
"Revoke": "Revocar",
|
"Revoke": "Revocar",
|
||||||
"Don't": "No",
|
"Don't": "No",
|
||||||
@@ -272,6 +278,7 @@
|
|||||||
"Add row below": "Agregar fila debajo",
|
"Add row below": "Agregar fila debajo",
|
||||||
"Delete table": "Eliminar tabla",
|
"Delete table": "Eliminar tabla",
|
||||||
"Info": "Información",
|
"Info": "Información",
|
||||||
|
"Note": "Nota",
|
||||||
"Success": "Satisfactorio",
|
"Success": "Satisfactorio",
|
||||||
"Warning": "Advertencia",
|
"Warning": "Advertencia",
|
||||||
"Danger": "Peligro",
|
"Danger": "Peligro",
|
||||||
@@ -282,6 +289,11 @@
|
|||||||
"Save & Exit": "Guardar y Salir",
|
"Save & Exit": "Guardar y Salir",
|
||||||
"Double-click to edit Excalidraw diagram": "Doble clic para editar el diagrama de Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Doble clic para editar el diagrama de Excalidraw",
|
||||||
"Paste link": "Pegar enlace",
|
"Paste link": "Pegar enlace",
|
||||||
|
"Paste link or search pages": "Pega un enlace o busca páginas",
|
||||||
|
"Link to web page": "Enlazar a una página web",
|
||||||
|
"Recents": "Recientes",
|
||||||
|
"Page or URL": "Página o URL",
|
||||||
|
"Link title": "Título del enlace",
|
||||||
"Edit link": "Editar enlace",
|
"Edit link": "Editar enlace",
|
||||||
"Remove link": "Eliminar enlace",
|
"Remove link": "Eliminar enlace",
|
||||||
"Add link": "Agregar enlace",
|
"Add link": "Agregar enlace",
|
||||||
@@ -329,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Insertar regla horizontal",
|
"Insert horizontal rule divider": "Insertar regla horizontal",
|
||||||
"Upload any image from your device.": "Sube cualquier imagen desde tu dispositivo.",
|
"Upload any image from your device.": "Sube cualquier imagen desde tu dispositivo.",
|
||||||
"Upload any video from your device.": "Sube cualquier video desde tu dispositivo.",
|
"Upload any video from your device.": "Sube cualquier video desde tu dispositivo.",
|
||||||
|
"Upload any audio from your device.": "Sube cualquier audio desde tu dispositivo.",
|
||||||
"Upload any file from your device.": "Sube cualquier archivo desde tu dispositivo.",
|
"Upload any file from your device.": "Sube cualquier archivo desde tu dispositivo.",
|
||||||
"Uploading {{name}}": "Subiendo {{name}}",
|
"Uploading {{name}}": "Subiendo {{name}}",
|
||||||
"Uploading file": "Subiendo archivo",
|
"Uploading file": "Subiendo archivo",
|
||||||
@@ -339,6 +352,12 @@
|
|||||||
"Divider": "Divisor",
|
"Divider": "Divisor",
|
||||||
"Quote": "Cita",
|
"Quote": "Cita",
|
||||||
"Image": "Imagen",
|
"Image": "Imagen",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "Adjuntar PDF",
|
||||||
|
"Upload and embed a PDF file.": "Sube y adjunta un archivo PDF.",
|
||||||
|
"Embed as PDF": "Adjuntar como PDF",
|
||||||
|
"Failed to load PDF": "Error al cargar el PDF",
|
||||||
|
"Convert to attachment": "Convertir en adjunto",
|
||||||
"File attachment": "Adjunto de archivo",
|
"File attachment": "Adjunto de archivo",
|
||||||
"Toggle block": "Alternar bloque",
|
"Toggle block": "Alternar bloque",
|
||||||
"Callout": "Aviso",
|
"Callout": "Aviso",
|
||||||
@@ -353,9 +372,23 @@
|
|||||||
"Insert current date": "Insertar fecha actual",
|
"Insert current date": "Insertar fecha actual",
|
||||||
"Draw and sketch excalidraw diagrams": "Dibujar y esbozar diagramas de Excalidraw",
|
"Draw and sketch excalidraw diagrams": "Dibujar y esbozar diagramas de Excalidraw",
|
||||||
"Multiple": "Múltiple",
|
"Multiple": "Múltiple",
|
||||||
|
"Turn into": "Convertir en",
|
||||||
|
"Text align": "Alineación del texto",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "Es posible que esta página haya sido eliminada, movida o que no tengas acceso.",
|
||||||
|
"Go to homepage": "Ir a la página principal",
|
||||||
|
"Pages you create will show up here.": "Las páginas que crees aparecerán aquí.",
|
||||||
"Heading {{level}}": "Encabezado {{level}}",
|
"Heading {{level}}": "Encabezado {{level}}",
|
||||||
"Toggle title": "Alternar título",
|
"Toggle title": "Alternar título",
|
||||||
"Write anything. Enter \"/\" for commands": "Escribe cualquier cosa. Ingresa \"/\" para comandos",
|
"Write anything. Enter \"/\" for commands": "Escribe cualquier cosa. Ingresa \"/\" para comandos",
|
||||||
|
"Write...": "Escribe...",
|
||||||
|
"Column count": "Número de columnas",
|
||||||
|
"{{count}} Columns": "{count, plural, one {# columna} other {# columnas}}",
|
||||||
|
"Equal columns": "Columnas iguales",
|
||||||
|
"Left sidebar": "Barra lateral izquierda",
|
||||||
|
"Right sidebar": "Barra lateral derecha",
|
||||||
|
"Wide center": "Centro ancho",
|
||||||
|
"Left wide": "Izquierda ancha",
|
||||||
|
"Right wide": "Derecha ancha",
|
||||||
"Names do not match": "Los nombres no coinciden",
|
"Names do not match": "Los nombres no coinciden",
|
||||||
"Today, {{time}}": "Hoy, {{time}}",
|
"Today, {{time}}": "Hoy, {{time}}",
|
||||||
"Yesterday, {{time}}": "Ayer, {{time}}",
|
"Yesterday, {{time}}": "Ayer, {{time}}",
|
||||||
@@ -378,6 +411,13 @@
|
|||||||
"Delete member": "Eliminar miembro",
|
"Delete member": "Eliminar miembro",
|
||||||
"Member deleted successfully": "Miembro eliminado con éxito",
|
"Member deleted successfully": "Miembro eliminado con éxito",
|
||||||
"Are you sure you want to delete this workspace member? This action is irreversible.": "¿Está seguro que desea eliminar este miembro del área de trabajo? Esta acción es irreversible.",
|
"Are you sure you want to delete this workspace member? This action is irreversible.": "¿Está seguro que desea eliminar este miembro del área de trabajo? Esta acción es irreversible.",
|
||||||
|
"Deactivate member": "Desactivar miembro",
|
||||||
|
"Activate member": "Activar miembro",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "¿Está seguro de que desea desactivar a este miembro del espacio de trabajo? Ya no podrá acceder a este espacio de trabajo.",
|
||||||
|
"Are you sure you want to activate this workspace member?": "¿Está seguro de que desea activar a este miembro del espacio de trabajo?",
|
||||||
|
"Deactivate": "Desactivar",
|
||||||
|
"Activate": "Activar",
|
||||||
|
"Deactivated": "Desactivado",
|
||||||
"Move": "Mover",
|
"Move": "Mover",
|
||||||
"Move page": "Mover página",
|
"Move page": "Mover página",
|
||||||
"Move page to a different space.": "Mover página a un espacio diferente.",
|
"Move page to a different space.": "Mover página a un espacio diferente.",
|
||||||
@@ -405,6 +445,25 @@
|
|||||||
"Share deleted successfully": "Compartición eliminada con éxito",
|
"Share deleted successfully": "Compartición eliminada con éxito",
|
||||||
"Share not found": "Compartición no encontrada",
|
"Share not found": "Compartición no encontrada",
|
||||||
"Failed to share page": "Error al compartir la página",
|
"Failed to share page": "Error al compartir la página",
|
||||||
|
"Disable public sharing": "Desactivar el uso compartido público",
|
||||||
|
"Prevent members from sharing pages publicly.": "Evitar que los miembros compartan páginas públicamente.",
|
||||||
|
"Toggle public sharing": "Alternar el uso compartido público",
|
||||||
|
"Toggle space public sharing": "Alternar el uso compartido público del espacio",
|
||||||
|
"Allow viewers to comment": "Permitir que los espectadores comenten",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Permitir que los espectadores agreguen comentarios en las páginas de este espacio.",
|
||||||
|
"Toggle viewer comments": "Activar/desactivar comentarios de los espectadores",
|
||||||
|
"Public sharing is disabled at the workspace level": "El uso compartido público está desactivado a nivel de espacio de trabajo",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "Evitar que las páginas en este espacio se compartan públicamente.",
|
||||||
|
"Page permissions": "Permisos de la página},{",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Controla quién puede ver y editar páginas individuales. Disponible con una licencia empresarial.",
|
||||||
|
"Enable public sharing": "Activar el uso compartido público",
|
||||||
|
"Are you sure you want to enable public sharing? Members will be able to share pages publicly.": "¿Está seguro de que desea activar el uso compartido público? Los miembros podrán compartir páginas públicamente.",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this workspace will be deleted.": "¿Está seguro de que desea desactivar el uso compartido público? Todos los enlaces compartidos existentes en este espacio de trabajo se eliminarán.",
|
||||||
|
"Are you sure you want to enable public sharing for this space?": "¿Está seguro de que desea activar el uso compartido público para este espacio?",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this space will be deleted.": "¿Está seguro de que desea desactivar el uso compartido público? Todos los enlaces compartidos existentes en este espacio se eliminarán.",
|
||||||
|
"Public sharing is disabled": "El uso compartido público está desactivado",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "El uso compartido público se ha desactivado a nivel de espacio de trabajo.",
|
||||||
|
"Public sharing has been disabled for this space.": "El uso compartido público se ha desactivado para este espacio.",
|
||||||
"Copy page": "Copiar página",
|
"Copy page": "Copiar página",
|
||||||
"Copy page to a different space.": "Copiar página en otro espacio",
|
"Copy page to a different space.": "Copiar página en otro espacio",
|
||||||
"Page copied successfully": "Página copiada exitosamente",
|
"Page copied successfully": "Página copiada exitosamente",
|
||||||
@@ -487,7 +546,7 @@
|
|||||||
"Enter one of your backup codes. Each backup code can only be used once.": "Introduce uno de tus códigos de seguridad. Cada código de seguridad solo puede ser usado una vez.",
|
"Enter one of your backup codes. Each backup code can only be used once.": "Introduce uno de tus códigos de seguridad. Cada código de seguridad solo puede ser usado una vez.",
|
||||||
"Verify": "Verificar",
|
"Verify": "Verificar",
|
||||||
"Trash": "Papelera",
|
"Trash": "Papelera",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "Las páginas en la papelera serán eliminadas permanentemente después de 30 días.",
|
"Pages in trash will be permanently deleted after {{count}} days.": "{count, plural, one{Las páginas en la papelera se eliminarán permanentemente después de # día.} other{Las páginas en la papelera se eliminarán permanentemente después de # días.}}",
|
||||||
"Deleted": "Eliminado",
|
"Deleted": "Eliminado",
|
||||||
"No pages in trash": "No hay páginas en la papelera",
|
"No pages in trash": "No hay páginas en la papelera",
|
||||||
"Permanently delete page?": "¿Eliminar página permanentemente?",
|
"Permanently delete page?": "¿Eliminar página permanentemente?",
|
||||||
@@ -559,19 +618,138 @@
|
|||||||
"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.",
|
"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",
|
"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",
|
"Manage API keys for all users in the workspace": "Gestionar claves API para todos los usuarios en el espacio de trabajo",
|
||||||
|
"Restrict API key creation to admins": "Restringir la creación de claves API a administradores",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "Solo los administradores y propietarios pueden crear nuevas claves API. Las claves de miembros existentes seguirán funcionando.",
|
||||||
|
"Toggle restrict API keys to admins": "Activar o desactivar la restricción de claves API solo a administradores",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "La creación de claves API está restringida a administradores por el administrador de tu espacio de trabajo.",
|
||||||
"AI settings": "Configuración de IA",
|
"AI settings": "Configuración de IA",
|
||||||
"AI search": "Búsqueda de IA",
|
"AI search": "Búsqueda de IA",
|
||||||
"AI Answer": "Respuesta de IA",
|
"AI Answer": "Respuesta de IA",
|
||||||
"Ask AI": "Preguntar a IA",
|
"Ask AI": "Preguntar a IA",
|
||||||
"AI is thinking...": "IA está pensando...",
|
"AI is thinking...": "IA está pensando...",
|
||||||
"Ask a question...": "Haz una pregunta...",
|
"Ask a question...": "Haz una pregunta...",
|
||||||
"AI-powered search (Ask AI)": "Búsqueda impulsada por IA (Preguntar a IA)",
|
"AI Answers": "Respuestas de IA",
|
||||||
|
"AI-powered search (AI Answers)": "Búsqueda impulsada por IA (Respuestas de 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.",
|
"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",
|
"Toggle AI search": "Alternar búsqueda de IA",
|
||||||
|
"Generative AI (Ask AI)": "IA generativa (Preguntar a la IA)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Habilitar la generación de contenido impulsada por IA en el editor. Permite a los usuarios generar, mejorar, traducir y transformar texto.",
|
||||||
|
"Toggle generative AI": "Activar IA generativa",
|
||||||
|
"Upgrade your plan": "Mejora tu plan",
|
||||||
|
"Available with a paid license": "Disponible con una licencia de pago",
|
||||||
|
"Upgrade your license tier.": "Mejora el nivel de tu licencia.",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "La IA solo está disponible en la edición empresarial de Docmost. Contacte con sales@docmost.com.",
|
||||||
|
"AI & MCP": "IA y MCP",
|
||||||
|
"AI": "IA",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "Protocolo de Contexto del Modelo (MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Habilite el servidor MCP para permitir que asistentes de IA y herramientas interactúen con el contenido de su espacio de trabajo.",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP solo está disponible en la edición empresarial de Docmost. Contacte con sales@docmost.com.",
|
||||||
|
"MCP Server URL": "URL del servidor MCP",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Use su clave API para la autenticación. Puede gestionar las claves API en la configuración de su cuenta.",
|
||||||
|
"Supported tools": "Herramientas compatibles",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Su espacio de trabajo tiene MCP habilitado. Use su clave API para conectar asistentes de IA.",
|
||||||
|
"MCP server URL:": "URL del servidor MCP:",
|
||||||
|
"Learn more": "Más información",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Gestiona las claves de API para todos los usuarios en el espacio de trabajo. Consulta la <anchor>documentación de la API</anchor> para detalles de uso.",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "Consulta la <anchor>documentación de la API</anchor> para detalles de uso.",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "Consulta la <anchor>documentación de MCP</anchor>.",
|
||||||
"Sources": "Fuentes",
|
"Sources": "Fuentes",
|
||||||
"Ask AI not available for attachments": "Preguntar a IA no está disponible para adjuntos",
|
"AI Answers not available for attachments": "Respuestas de IA no disponibles para archivos adjuntos",
|
||||||
"No answer available": "No hay respuesta disponible",
|
"No answer available": "No hay respuesta disponible",
|
||||||
"Background color": "Color de fondo",
|
"Background color": "Color de fondo",
|
||||||
"Highlight color": "Color de resaltado",
|
"Highlight color": "Color de resaltado",
|
||||||
"Remove color": "Eliminar color"
|
"Remove color": "Eliminar color",
|
||||||
|
"Notifications": "Notificaciones",
|
||||||
|
"No notifications": "Sin notificaciones",
|
||||||
|
"No unread notifications": "No hay notificaciones no leídas",
|
||||||
|
"All notifications": "Todas las notificaciones",
|
||||||
|
"Unread only": "Solo no leídas",
|
||||||
|
"Mark all as read": "Marcar todo como leído",
|
||||||
|
"Mark as read": "Marcar como leído",
|
||||||
|
"More options": "Más opciones",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> te mencionó en un comentario",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> comentó en una página",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> resolvió un comentario",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> te mencionó en una página",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> te dio acceso de edición a una página",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> te dio acceso de visualización a una página",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> actualizó una página.",
|
||||||
|
"Watch page": "Seguir página",
|
||||||
|
"Stop watching": "Dejar de seguir",
|
||||||
|
"Email notifications": "Notificaciones por correo electrónico",
|
||||||
|
"Page updates": "Actualizaciones de página",
|
||||||
|
"Get notified when pages you watch are updated.": "Recibe una notificación cuando se actualicen las páginas que sigues.",
|
||||||
|
"Page mentions": "Menciones en la página",
|
||||||
|
"Get notified when someone mentions you on a page.": "Recibe una notificación cuando alguien te mencione en una página.",
|
||||||
|
"Comment mentions": "Menciones en comentarios",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Recibe una notificación cuando alguien te mencione en un comentario.",
|
||||||
|
"New comments": "Nuevos comentarios",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Recibe una notificación sobre nuevos comentarios en los hilos donde participas.",
|
||||||
|
"Resolved comments": "Comentarios resueltos",
|
||||||
|
"Get notified when your comment is resolved.": "Recibe una notificación cuando tu comentario sea resuelto.",
|
||||||
|
"You are now watching this page": "Ahora sigues esta página",
|
||||||
|
"You are no longer watching this page": "Ya no sigues esta página",
|
||||||
|
"Direct": "Directo",
|
||||||
|
"Updates": "Actualizaciones",
|
||||||
|
"Today": "Hoy",
|
||||||
|
"Yesterday": "Ayer",
|
||||||
|
"This week": "Esta semana",
|
||||||
|
"Older": "Más antiguo",
|
||||||
|
"Restricted page": "Página restringida",
|
||||||
|
"Restricted pages cannot be shared publicly.": "Las páginas restringidas no pueden compartirse públicamente.",
|
||||||
|
"Restricted by parent": "Restringida por la página padre",
|
||||||
|
"Restricted": "Restringida",
|
||||||
|
"Open": "Abierta",
|
||||||
|
"Inherits restrictions from ancestor page": "Hereda las restricciones de una página superior.",
|
||||||
|
"Only people listed below can access this page": "Solo las personas que figuran a continuación pueden acceder a esta página.",
|
||||||
|
"Everyone in this space can access": "Todos en este espacio pueden acceder.",
|
||||||
|
"No additional restrictions on this page": "No hay restricciones adicionales en esta página.",
|
||||||
|
"Only specific people can access": "Solo determinadas personas pueden acceder.",
|
||||||
|
"Use only inherited restrictions": "Usar solo las restricciones heredadas.",
|
||||||
|
"Add restrictions on top of inherited": "Agregar restricciones además de las heredadas.",
|
||||||
|
"Inherited restriction": "Restricción heredada",
|
||||||
|
"Access limited by": "Acceso limitado por",
|
||||||
|
"Restrict access to control who can view and edit this page": "Restringir el acceso para controlar quién puede ver y editar esta página.",
|
||||||
|
"Add additional restrictions specific to this page": "Agregar restricciones adicionales específicas para esta página.",
|
||||||
|
"Access": "Acceso",
|
||||||
|
"People with access": "Personas con acceso",
|
||||||
|
"Remove all": "Eliminar todo",
|
||||||
|
"Remove access": "Eliminar acceso",
|
||||||
|
"Remove all access": "Eliminar todo el acceso",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "¿Está seguro de que desea eliminar el acceso de este miembro a la página?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "¿Está seguro de que desea eliminar todo el acceso específico? Esto hará que la página esté abierta a todos en el espacio.",
|
||||||
|
"Trash retention": "Retención de la papelera",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "Las páginas en la papelera se eliminarán permanentemente después de este período.",
|
||||||
|
"Trash retention updated": "Retención de la papelera actualizada",
|
||||||
|
"Failed to update trash retention": "No se pudo actualizar la retención de la papelera.",
|
||||||
|
"Removed page restriction": "Restricción de página eliminada",
|
||||||
|
"Added page permission": "Permiso de página añadido",
|
||||||
|
"Removed page permission": "Permiso de página eliminado",
|
||||||
|
"Verifying your email": "Verificando tu correo electrónico",
|
||||||
|
"Please wait...": "Por favor, espera...",
|
||||||
|
"Verification failed. The link may have expired.": "La verificación ha fallado. Es posible que el enlace haya expirado.",
|
||||||
|
"Check your email": "Revisa tu correo electrónico",
|
||||||
|
"We sent a verification link to {{email}}.": "Te enviamos un enlace de verificación a {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Te enviamos un enlace de verificación a tu correo.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Haz clic en el enlace para verificar tu correo electrónico y acceder a tu espacio de trabajo.",
|
||||||
|
"Resend verification email": "Reenviar correo de verificación",
|
||||||
|
"Verification email sent. Please check your inbox.": "Correo de verificación enviado. Por favor, revisa tu bandeja de entrada.",
|
||||||
|
"Failed to resend verification email. Please try again.": "No se pudo reenviar el correo de verificación. Por favor, intente de nuevo.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Te hemos enviado un correo electrónico con tus espacios de trabajo asociados.",
|
||||||
|
"Load more": "Cargar más",
|
||||||
|
"Log out of all devices": "Cerrar sesión en todos los dispositivos",
|
||||||
|
"Log out of all sessions except this device": "Cerrar sesión en todos los dispositivos excepto este",
|
||||||
|
"This Device": "Este dispositivo",
|
||||||
|
"Unknown device": "Dispositivo desconocido",
|
||||||
|
"No active sessions": "No hay sesiones activas",
|
||||||
|
"Session revoked": "Sesión revocada",
|
||||||
|
"All other sessions revoked": "Todas las demás sesiones revocadas",
|
||||||
|
"Last used": "Último uso",
|
||||||
|
"Created": "Creado",
|
||||||
|
"Rename": "Renombrar",
|
||||||
|
"Publish": "Publicar",
|
||||||
|
"Security": "Seguridad",
|
||||||
|
"Enforce SSO": "Forzar SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Una vez forzado, los miembros no podrán iniciar sesión con correo electrónico y contraseña."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -116,6 +116,7 @@
|
|||||||
"No group found": "Aucun groupe trouvé",
|
"No group found": "Aucun groupe trouvé",
|
||||||
"No page history saved yet.": "Aucun historique de la page enregistré pour l'instant.",
|
"No page history saved yet.": "Aucun historique de la page enregistré pour l'instant.",
|
||||||
"No pages yet": "Aucune page pour l'instant",
|
"No pages yet": "Aucune page pour l'instant",
|
||||||
|
"No shared pages": "Aucune page partagée",
|
||||||
"No results found...": "Aucun résultat trouvé...",
|
"No results found...": "Aucun résultat trouvé...",
|
||||||
"No user found": "Aucun utilisateur trouvé",
|
"No user found": "Aucun utilisateur trouvé",
|
||||||
"Overview": "Vue d'ensemble",
|
"Overview": "Vue d'ensemble",
|
||||||
@@ -123,11 +124,14 @@
|
|||||||
"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",
|
||||||
"Password": "Mot de passe",
|
"Password": "Mot de passe",
|
||||||
"Password changed successfully": "Mot de passe changé avec succès",
|
"Password changed successfully": "Mot de passe changé avec succès",
|
||||||
|
"People": "Personnes",
|
||||||
"Pending": "En attente",
|
"Pending": "En attente",
|
||||||
"Please confirm your action": "Veuillez confirmer votre action",
|
"Please confirm your action": "Veuillez confirmer votre action",
|
||||||
"Preferences": "Préférences",
|
"Preferences": "Préférences",
|
||||||
@@ -205,6 +209,9 @@
|
|||||||
"Reply...": "Répondre...",
|
"Reply...": "Répondre...",
|
||||||
"Error loading comments.": "Erreur lors du chargement des commentaires.",
|
"Error loading comments.": "Erreur lors du chargement des commentaires.",
|
||||||
"No comments yet.": "Pas de commentaires pour l'instant.",
|
"No comments yet.": "Pas de commentaires pour l'instant.",
|
||||||
|
"No open comments.": "Aucun commentaire ouvert.",
|
||||||
|
"No resolved comments.": "Aucun commentaire résolu.",
|
||||||
|
"Add a comment...": "Ajouter un commentaire...",
|
||||||
"Edit comment": "Modifier le commentaire",
|
"Edit comment": "Modifier le commentaire",
|
||||||
"Delete comment": "Supprimer le commentaire",
|
"Delete comment": "Supprimer le commentaire",
|
||||||
"Are you sure you want to delete this comment?": "Êtes-vous sûr de vouloir supprimer ce commentaire ?",
|
"Are you sure you want to delete this comment?": "Êtes-vous sûr de vouloir supprimer ce commentaire ?",
|
||||||
@@ -226,7 +233,6 @@
|
|||||||
"Are you sure you want to unresolve this comment thread?": "Êtes-vous sûr de vouloir désorganiser ce fil de commentaires ?",
|
"Are you sure you want to unresolve this comment thread?": "Êtes-vous sûr de vouloir désorganiser ce fil de commentaires ?",
|
||||||
"Resolved": "Résolu",
|
"Resolved": "Résolu",
|
||||||
"No active comments.": "Aucun commentaire actif.",
|
"No active comments.": "Aucun commentaire actif.",
|
||||||
"No resolved comments.": "Aucun commentaire résolu.",
|
|
||||||
"Revoke invitation": "Révoquer l'invitation",
|
"Revoke invitation": "Révoquer l'invitation",
|
||||||
"Revoke": "Révoquer",
|
"Revoke": "Révoquer",
|
||||||
"Don't": "Ne pas",
|
"Don't": "Ne pas",
|
||||||
@@ -272,6 +278,7 @@
|
|||||||
"Add row below": "Ajouter une ligne en dessous",
|
"Add row below": "Ajouter une ligne en dessous",
|
||||||
"Delete table": "Supprimer le tableau",
|
"Delete table": "Supprimer le tableau",
|
||||||
"Info": "Info",
|
"Info": "Info",
|
||||||
|
"Note": "Remarque",
|
||||||
"Success": "Succès",
|
"Success": "Succès",
|
||||||
"Warning": "Avertissement",
|
"Warning": "Avertissement",
|
||||||
"Danger": "Danger",
|
"Danger": "Danger",
|
||||||
@@ -282,6 +289,11 @@
|
|||||||
"Save & Exit": "Enregistrer & Quitter",
|
"Save & Exit": "Enregistrer & Quitter",
|
||||||
"Double-click to edit Excalidraw diagram": "Double-cliquez pour modifier le diagramme Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Double-cliquez pour modifier le diagramme Excalidraw",
|
||||||
"Paste link": "Coller le lien",
|
"Paste link": "Coller le lien",
|
||||||
|
"Paste link or search pages": "Coller le lien ou rechercher des pages",
|
||||||
|
"Link to web page": "Lien vers une page web",
|
||||||
|
"Recents": "Récents",
|
||||||
|
"Page or URL": "Page ou URL",
|
||||||
|
"Link title": "Titre du lien",
|
||||||
"Edit link": "Modifier le lien",
|
"Edit link": "Modifier le lien",
|
||||||
"Remove link": "Supprimer le lien",
|
"Remove link": "Supprimer le lien",
|
||||||
"Add link": "Ajouter un lien",
|
"Add link": "Ajouter un lien",
|
||||||
@@ -329,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Insérer un séparateur de règle horizontale",
|
"Insert horizontal rule divider": "Insérer un séparateur de règle horizontale",
|
||||||
"Upload any image from your device.": "Téléchargez n'importe quelle image depuis votre appareil.",
|
"Upload any image from your device.": "Téléchargez n'importe quelle image depuis votre appareil.",
|
||||||
"Upload any video from your device.": "Téléchargez n'importe quelle vidéo depuis votre appareil.",
|
"Upload any video from your device.": "Téléchargez n'importe quelle vidéo depuis votre appareil.",
|
||||||
|
"Upload any audio from your device.": "Téléchargez n'importe quel fichier audio depuis votre appareil.",
|
||||||
"Upload any file from your device.": "Téléchargez n'importe quel fichier depuis votre appareil.",
|
"Upload any file from your device.": "Téléchargez n'importe quel fichier depuis votre appareil.",
|
||||||
"Uploading {{name}}": "Téléchargement de {{name}}",
|
"Uploading {{name}}": "Téléchargement de {{name}}",
|
||||||
"Uploading file": "Téléchargement du fichier",
|
"Uploading file": "Téléchargement du fichier",
|
||||||
@@ -339,6 +352,12 @@
|
|||||||
"Divider": "Diviseur",
|
"Divider": "Diviseur",
|
||||||
"Quote": "Citation",
|
"Quote": "Citation",
|
||||||
"Image": "Image",
|
"Image": "Image",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "Intégrer un PDF",
|
||||||
|
"Upload and embed a PDF file.": "Téléchargez et intégrez un fichier PDF.",
|
||||||
|
"Embed as PDF": "Intégrer comme PDF",
|
||||||
|
"Failed to load PDF": "Échec du chargement du PDF",
|
||||||
|
"Convert to attachment": "Convertir en pièce jointe",
|
||||||
"File attachment": "Pièce jointe",
|
"File attachment": "Pièce jointe",
|
||||||
"Toggle block": "Basculer le bloc",
|
"Toggle block": "Basculer le bloc",
|
||||||
"Callout": "Appel",
|
"Callout": "Appel",
|
||||||
@@ -353,9 +372,23 @@
|
|||||||
"Insert current date": "Insérer la date actuelle",
|
"Insert current date": "Insérer la date actuelle",
|
||||||
"Draw and sketch excalidraw diagrams": "Dessiner et esquisser des diagrammes Excalidraw",
|
"Draw and sketch excalidraw diagrams": "Dessiner et esquisser des diagrammes Excalidraw",
|
||||||
"Multiple": "Multiple",
|
"Multiple": "Multiple",
|
||||||
|
"Turn into": "Transformer en",
|
||||||
|
"Text align": "Alignement du texte",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "Cette page a peut-être été supprimée, déplacée ou vous n'y avez peut-être pas accès.",
|
||||||
|
"Go to homepage": "Aller à l'accueil",
|
||||||
|
"Pages you create will show up here.": "Les pages que vous créez apparaîtront ici.",
|
||||||
"Heading {{level}}": "Titre {{level}}",
|
"Heading {{level}}": "Titre {{level}}",
|
||||||
"Toggle title": "Basculer le titre",
|
"Toggle title": "Basculer le titre",
|
||||||
"Write anything. Enter \"/\" for commands": "Écrivez n'importe quoi. Entrez \"/\" pour les commandes",
|
"Write anything. Enter \"/\" for commands": "Écrivez n'importe quoi. Entrez \"/\" pour les commandes",
|
||||||
|
"Write...": "Écrire...",
|
||||||
|
"Column count": "Nombre de colonnes",
|
||||||
|
"{{count}} Columns": "{count, plural, one {# colonne} other {# colonnes}}",
|
||||||
|
"Equal columns": "Colonnes égales",
|
||||||
|
"Left sidebar": "Barre latérale gauche",
|
||||||
|
"Right sidebar": "Barre latérale droite",
|
||||||
|
"Wide center": "Large au centre",
|
||||||
|
"Left wide": "Large à gauche",
|
||||||
|
"Right wide": "Large à droite",
|
||||||
"Names do not match": "Les noms ne correspondent pas",
|
"Names do not match": "Les noms ne correspondent pas",
|
||||||
"Today, {{time}}": "Aujourd'hui, {{time}}",
|
"Today, {{time}}": "Aujourd'hui, {{time}}",
|
||||||
"Yesterday, {{time}}": "Hier, {{time}}",
|
"Yesterday, {{time}}": "Hier, {{time}}",
|
||||||
@@ -378,11 +411,18 @@
|
|||||||
"Delete member": "Supprimer le membre",
|
"Delete member": "Supprimer le membre",
|
||||||
"Member deleted successfully": "Membre supprimé avec succès",
|
"Member deleted successfully": "Membre supprimé avec succès",
|
||||||
"Are you sure you want to delete this workspace member? This action is irreversible.": "Êtes-vous sûr de vouloir supprimer ce membre de l'espace de travail? Cette action est irréversible.",
|
"Are you sure you want to delete this workspace member? This action is irreversible.": "Êtes-vous sûr de vouloir supprimer ce membre de l'espace de travail? Cette action est irréversible.",
|
||||||
|
"Deactivate member": "Désactiver le membre",
|
||||||
|
"Activate member": "Activer le membre",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "Êtes-vous sûr de vouloir désactiver ce membre de l'espace de travail ? Cette personne ne pourra plus accéder à cet espace de travail.",
|
||||||
|
"Are you sure you want to activate this workspace member?": "Êtes-vous sûr de vouloir activer ce membre de l'espace de travail ?",
|
||||||
|
"Deactivate": "Désactiver",
|
||||||
|
"Activate": "Activer",
|
||||||
|
"Deactivated": "Désactivé",
|
||||||
"Move": "Déplacer",
|
"Move": "Déplacer",
|
||||||
"Move page": "Déplacer la page",
|
"Move page": "Déplacer la page",
|
||||||
"Move page to a different space.": "Déplacer la page vers un autre espace.",
|
"Move page to a different space.": "Déplacer la page vers un autre espace.",
|
||||||
"Real-time editor connection lost. Retrying...": "Connexion avec l'éditeur en temps réel perdue. Nouvelle tentative...",
|
"Real-time editor connection lost. Retrying...": "Connexion avec l'éditeur en temps réel perdue. Nouvelle tentative...",
|
||||||
"Table of contents": "",
|
"Table of contents": "Table des matières.",
|
||||||
"Add headings (H1, H2, H3) to generate a table of contents.": "Ajoutez des titres (H1, H2, H3) pour générer une table des matières.",
|
"Add headings (H1, H2, H3) to generate a table of contents.": "Ajoutez des titres (H1, H2, H3) pour générer une table des matières.",
|
||||||
"Share": "Partager",
|
"Share": "Partager",
|
||||||
"Public sharing": "Partage public",
|
"Public sharing": "Partage public",
|
||||||
@@ -405,6 +445,25 @@
|
|||||||
"Share deleted successfully": "Partage supprimé avec succès",
|
"Share deleted successfully": "Partage supprimé avec succès",
|
||||||
"Share not found": "Partage non trouvé",
|
"Share not found": "Partage non trouvé",
|
||||||
"Failed to share page": "Échec du partage de la page",
|
"Failed to share page": "Échec du partage de la page",
|
||||||
|
"Disable public sharing": "Désactiver le partage public",
|
||||||
|
"Prevent members from sharing pages publicly.": "Empêcher les membres de partager des pages publiquement.",
|
||||||
|
"Toggle public sharing": "Basculer le partage public",
|
||||||
|
"Toggle space public sharing": "Basculer le partage public de l'espace",
|
||||||
|
"Allow viewers to comment": "Autoriser les spectateurs à commenter",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Autoriser les spectateurs à ajouter des commentaires sur les pages de cet espace.",
|
||||||
|
"Toggle viewer comments": "Basculer les commentaires des spectateurs",
|
||||||
|
"Public sharing is disabled at the workspace level": "Le partage public est désactivé au niveau de l'espace de travail",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "Empêcher les pages de cet espace d'être partagées publiquement.",
|
||||||
|
"Page permissions": "Autorisations de la page",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Contrôlez qui peut consulter et modifier chaque page. Disponible avec une licence Entreprise.",
|
||||||
|
"Enable public sharing": "Activer le partage public",
|
||||||
|
"Are you sure you want to enable public sharing? Members will be able to share pages publicly.": "Êtes-vous sûr de vouloir activer le partage public ? Les membres pourront partager des pages publiquement.",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this workspace will be deleted.": "Êtes-vous sûr de vouloir désactiver le partage public ? Tous les liens partagés existants dans cet espace de travail seront supprimés.",
|
||||||
|
"Are you sure you want to enable public sharing for this space?": "Êtes-vous sûr de vouloir activer le partage public pour cet espace ?",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this space will be deleted.": "Êtes-vous sûr de vouloir désactiver le partage public ? Tous les liens partagés existants dans cet espace seront supprimés.",
|
||||||
|
"Public sharing is disabled": "Le partage public est désactivé",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "Le partage public a été désactivé au niveau de l'espace de travail.",
|
||||||
|
"Public sharing has been disabled for this space.": "Le partage public a été désactivé pour cet espace.",
|
||||||
"Copy page": "Copier la page",
|
"Copy page": "Copier la page",
|
||||||
"Copy page to a different space.": "Copier la page dans un autre espace.",
|
"Copy page to a different space.": "Copier la page dans un autre espace.",
|
||||||
"Page copied successfully": "Page copiée avec succès",
|
"Page copied successfully": "Page copiée avec succès",
|
||||||
@@ -487,7 +546,7 @@
|
|||||||
"Enter one of your backup codes. Each backup code can only be used once.": "Entrez un de vos codes de sauvegarde. Chaque code de sauvegarde ne peut être utilisé qu'une seule fois.",
|
"Enter one of your backup codes. Each backup code can only be used once.": "Entrez un de vos codes de sauvegarde. Chaque code de sauvegarde ne peut être utilisé qu'une seule fois.",
|
||||||
"Verify": "Vérifier",
|
"Verify": "Vérifier",
|
||||||
"Trash": "Corbeille",
|
"Trash": "Corbeille",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "Les pages dans la corbeille seront définitivement supprimées après 30 jours.",
|
"Pages in trash will be permanently deleted after {{count}} days.": "Les pages dans la corbeille seront définitivement supprimées après {{count}} jours.",
|
||||||
"Deleted": "Supprimé",
|
"Deleted": "Supprimé",
|
||||||
"No pages in trash": "Aucune page dans la corbeille",
|
"No pages in trash": "Aucune page dans la corbeille",
|
||||||
"Permanently delete page?": "Supprimer définitivement la page ?",
|
"Permanently delete page?": "Supprimer définitivement la page ?",
|
||||||
@@ -559,19 +618,138 @@
|
|||||||
"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.",
|
"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",
|
"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",
|
"Manage API keys for all users in the workspace": "Gérer les clés API pour tous les utilisateurs dans l'espace de travail",
|
||||||
|
"Restrict API key creation to admins": "Restreindre la création de clés API aux administrateurs",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "Seuls les administrateurs et les propriétaires peuvent créer de nouvelles clés API. Les clés des membres existants continueront de fonctionner.",
|
||||||
|
"Toggle restrict API keys to admins": "Activer ou désactiver la restriction des clés API aux administrateurs",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "La création de clés API est restreinte aux administrateurs par l’administrateur de votre espace de travail.",
|
||||||
"AI settings": "Paramètres de l'IA",
|
"AI settings": "Paramètres de l'IA",
|
||||||
"AI search": "Recherche IA",
|
"AI search": "Recherche IA",
|
||||||
"AI Answer": "Réponse IA",
|
"AI Answer": "Réponse IA",
|
||||||
"Ask AI": "Demander à l'IA",
|
"Ask AI": "Demander à l'IA",
|
||||||
"AI is thinking...": "L'IA réfléchit...",
|
"AI is thinking...": "L'IA réfléchit...",
|
||||||
"Ask a question...": "Posez une question...",
|
"Ask a question...": "Posez une question...",
|
||||||
"AI-powered search (Ask AI)": "Recherche assistée par l'IA (Demander à l'IA)",
|
"AI Answers": "Réponses IA",
|
||||||
|
"AI-powered search (AI Answers)": "Recherche propulsée par IA (Réponses 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.",
|
"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",
|
"Toggle AI search": "Basculer la recherche IA",
|
||||||
|
"Generative AI (Ask AI)": "IA générative (Demandez à l'IA)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Activer la génération de contenu assistée par IA dans l'éditeur. Permet aux utilisateurs de générer, améliorer, traduire et transformer du texte.",
|
||||||
|
"Toggle generative AI": "Activer/désactiver l'IA générative",
|
||||||
|
"Upgrade your plan": "Mettez à niveau votre forfait",
|
||||||
|
"Available with a paid license": "Disponible avec une licence payante",
|
||||||
|
"Upgrade your license tier.": "Mettez à niveau votre niveau de licence.",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "L'IA n'est disponible que dans l'édition Entreprise de Docmost. Contactez sales@docmost.com.",
|
||||||
|
"AI & MCP": "IA & MCP",
|
||||||
|
"AI": "IA",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "Protocole de contexte de modèle (MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Activez le serveur MCP pour permettre aux assistants et outils IA d'interagir avec le contenu de votre espace de travail.",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP n'est disponible que dans l'édition Entreprise de Docmost. Contactez sales@docmost.com.",
|
||||||
|
"MCP Server URL": "URL du serveur MCP",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Utilisez votre clé API pour l'authentification. Vous pouvez gérer les clés API dans les paramètres de votre compte.",
|
||||||
|
"Supported tools": "Outils pris en charge",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Votre espace de travail a MCP activé. Utilisez votre clé API pour connecter des assistants IA.",
|
||||||
|
"MCP server URL:": "URL du serveur MCP :",
|
||||||
|
"Learn more": "En savoir plus",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Gérez les clés API pour tous les utilisateurs de l'espace de travail. Consultez la <anchor>documentation API</anchor> pour plus de détails sur l'utilisation.",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "Consultez la <anchor>documentation API</anchor> pour plus de détails sur l'utilisation.",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "Consultez la <anchor>documentation MCP</anchor>.",
|
||||||
"Sources": "Sources",
|
"Sources": "Sources",
|
||||||
"Ask AI not available for attachments": "Demande à l'IA non disponible pour les pièces jointes",
|
"AI Answers not available for attachments": "Réponses IA non disponibles pour les pièces jointes",
|
||||||
"No answer available": "Pas de réponse disponible",
|
"No answer available": "Pas de réponse disponible",
|
||||||
"Background color": "Couleur de fond",
|
"Background color": "Couleur de fond",
|
||||||
"Highlight color": "Couleur de surbrillance",
|
"Highlight color": "Couleur de surbrillance",
|
||||||
"Remove color": "Supprimer la couleur"
|
"Remove color": "Supprimer la couleur",
|
||||||
|
"Notifications": "Notifications",
|
||||||
|
"No notifications": "Aucune notification",
|
||||||
|
"No unread notifications": "Aucune notification non lue",
|
||||||
|
"All notifications": "Toutes les notifications",
|
||||||
|
"Unread only": "Non lues uniquement",
|
||||||
|
"Mark all as read": "Tout marquer comme lu",
|
||||||
|
"Mark as read": "Marquer comme lu",
|
||||||
|
"More options": "Plus d'options",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> vous a mentionné dans un commentaire",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> a commenté une page",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> a résolu un commentaire",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> vous a mentionné sur une page",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> vous a donné l'accès en modification à une page",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> vous a donné l'accès en lecture à une page",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> a mis à jour une page.",
|
||||||
|
"Watch page": "Surveiller la page",
|
||||||
|
"Stop watching": "Ne plus surveiller",
|
||||||
|
"Email notifications": "Notifications par e-mail",
|
||||||
|
"Page updates": "Mises à jour de la page",
|
||||||
|
"Get notified when pages you watch are updated.": "Recevez une notification lorsque les pages que vous surveillez sont mises à jour.",
|
||||||
|
"Page mentions": "Mentions sur la page",
|
||||||
|
"Get notified when someone mentions you on a page.": "Recevez une notification lorsqu'une personne vous mentionne sur une page.",
|
||||||
|
"Comment mentions": "Mentions dans les commentaires",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Recevez une notification lorsqu'une personne vous mentionne dans un commentaire.",
|
||||||
|
"New comments": "Nouveaux commentaires",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Recevez une notification concernant les nouveaux commentaires dans les fils auxquels vous participez.",
|
||||||
|
"Resolved comments": "Commentaires résolus",
|
||||||
|
"Get notified when your comment is resolved.": "Recevez une notification lorsque votre commentaire est résolu.",
|
||||||
|
"You are now watching this page": "Vous surveillez désormais cette page",
|
||||||
|
"You are no longer watching this page": "Vous ne surveillez plus cette page",
|
||||||
|
"Direct": "Direct",
|
||||||
|
"Updates": "Mises à jour",
|
||||||
|
"Today": "Aujourd'hui",
|
||||||
|
"Yesterday": "Hier",
|
||||||
|
"This week": "Cette semaine",
|
||||||
|
"Older": "Plus ancien",
|
||||||
|
"Restricted page": "Page restreinte",
|
||||||
|
"Restricted pages cannot be shared publicly.": "Les pages restreintes ne peuvent pas être partagées publiquement.",
|
||||||
|
"Restricted by parent": "Restreint par la page parente",
|
||||||
|
"Restricted": "Restreint",
|
||||||
|
"Open": "Ouvert",
|
||||||
|
"Inherits restrictions from ancestor page": "Hérite des restrictions d'une page ancêtre",
|
||||||
|
"Only people listed below can access this page": "Seules les personnes listées ci-dessous peuvent accéder à cette page",
|
||||||
|
"Everyone in this space can access": "Tout le monde dans cet espace y a accès",
|
||||||
|
"No additional restrictions on this page": "Aucune restriction supplémentaire sur cette page",
|
||||||
|
"Only specific people can access": "Seules certaines personnes peuvent y accéder",
|
||||||
|
"Use only inherited restrictions": "Utiliser uniquement les restrictions héritées",
|
||||||
|
"Add restrictions on top of inherited": "Ajouter des restrictions en plus de celles héritées",
|
||||||
|
"Inherited restriction": "Restriction héritée",
|
||||||
|
"Access limited by": "Accès limité par",
|
||||||
|
"Restrict access to control who can view and edit this page": "Restreindre l'accès pour contrôler qui peut consulter et modifier cette page",
|
||||||
|
"Add additional restrictions specific to this page": "Ajouter des restrictions supplémentaires propres à cette page",
|
||||||
|
"Access": "Accès",
|
||||||
|
"People with access": "Personnes ayant accès",
|
||||||
|
"Remove all": "Tout retirer",
|
||||||
|
"Remove access": "Retirer l'accès",
|
||||||
|
"Remove all access": "Retirer tous les accès",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "Êtes-vous sûr de vouloir retirer l'accès de ce membre à la page ?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "Êtes-vous sûr de vouloir supprimer tous les accès spécifiques ? Cela rendra la page accessible à tous les membres de l'espace.",
|
||||||
|
"Trash retention": "Conservation de la corbeille",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "Les pages dans la corbeille seront définitivement supprimées après cette période.",
|
||||||
|
"Trash retention updated": "Durée de conservation de la corbeille mise à jour",
|
||||||
|
"Failed to update trash retention": "Échec de la mise à jour de la durée de conservation de la corbeille",
|
||||||
|
"Removed page restriction": "Restriction de la page supprimée",
|
||||||
|
"Added page permission": "Autorisation de la page ajoutée",
|
||||||
|
"Removed page permission": "Autorisation de la page supprimée",
|
||||||
|
"Verifying your email": "Vérification de votre e-mail",
|
||||||
|
"Please wait...": "Veuillez patienter...",
|
||||||
|
"Verification failed. The link may have expired.": "Échec de la vérification. Le lien a peut-être expiré.",
|
||||||
|
"Check your email": "Vérifiez votre e-mail",
|
||||||
|
"We sent a verification link to {{email}}.": "Nous avons envoyé un lien de vérification à {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Nous avons envoyé un lien de vérification à votre adresse e-mail.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Cliquez sur le lien pour vérifier votre adresse et accéder à votre espace de travail.",
|
||||||
|
"Resend verification email": "Renvoyer l'e-mail de vérification",
|
||||||
|
"Verification email sent. Please check your inbox.": "E-mail de vérification envoyé. Veuillez vérifier votre boîte de réception.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Échec de l'envoi du nouvel e-mail de vérification. Veuillez réessayer.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Nous vous avons envoyé un e-mail avec vos espaces de travail associés.",
|
||||||
|
"Load more": "Charger plus",
|
||||||
|
"Log out of all devices": "Déconnexion de tous les appareils",
|
||||||
|
"Log out of all sessions except this device": "Déconnexion de toutes les sessions sauf cet appareil",
|
||||||
|
"This Device": "Cet appareil",
|
||||||
|
"Unknown device": "Appareil inconnu",
|
||||||
|
"No active sessions": "Aucune session active",
|
||||||
|
"Session revoked": "Session révoquée",
|
||||||
|
"All other sessions revoked": "Toutes les autres sessions révoquées",
|
||||||
|
"Last used": "Dernière utilisation",
|
||||||
|
"Created": "Créé",
|
||||||
|
"Rename": "Renommer",
|
||||||
|
"Publish": "Publier",
|
||||||
|
"Security": "Sécurité",
|
||||||
|
"Enforce SSO": "Imposer SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Une fois imposé, les membres ne pourront plus se connecter par e-mail et mot de passe."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -116,6 +116,7 @@
|
|||||||
"No group found": "Nessun gruppo trovato",
|
"No group found": "Nessun gruppo trovato",
|
||||||
"No page history saved yet.": "La pagina non ha una cronologia per ora.",
|
"No page history saved yet.": "La pagina non ha una cronologia per ora.",
|
||||||
"No pages yet": "Nessuna pagina per ora",
|
"No pages yet": "Nessuna pagina per ora",
|
||||||
|
"No shared pages": "Nessuna pagina condivisa.",
|
||||||
"No results found...": "Nessun risultato trovato...",
|
"No results found...": "Nessun risultato trovato...",
|
||||||
"No user found": "Nessun utente trovato",
|
"No user found": "Nessun utente trovato",
|
||||||
"Overview": "Panoramica",
|
"Overview": "Panoramica",
|
||||||
@@ -123,11 +124,14 @@
|
|||||||
"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",
|
||||||
"Password": "Password",
|
"Password": "Password",
|
||||||
"Password changed successfully": "Password cambiata con successo",
|
"Password changed successfully": "Password cambiata con successo",
|
||||||
|
"People": "Persone",
|
||||||
"Pending": "In sospeso",
|
"Pending": "In sospeso",
|
||||||
"Please confirm your action": "Si prega di confermare la propria azione",
|
"Please confirm your action": "Si prega di confermare la propria azione",
|
||||||
"Preferences": "Preferenze",
|
"Preferences": "Preferenze",
|
||||||
@@ -205,6 +209,9 @@
|
|||||||
"Reply...": "Rispondi...",
|
"Reply...": "Rispondi...",
|
||||||
"Error loading comments.": "Si è verificato un errore durante il caricamento dei commenti.",
|
"Error loading comments.": "Si è verificato un errore durante il caricamento dei commenti.",
|
||||||
"No comments yet.": "Nessun commento per ora.",
|
"No comments yet.": "Nessun commento per ora.",
|
||||||
|
"No open comments.": "Nessun commento aperto.",
|
||||||
|
"No resolved comments.": "Nessun commento risolto.",
|
||||||
|
"Add a comment...": "Aggiungi un commento...",
|
||||||
"Edit comment": "Modifica commento",
|
"Edit comment": "Modifica commento",
|
||||||
"Delete comment": "Elimina commento",
|
"Delete comment": "Elimina commento",
|
||||||
"Are you sure you want to delete this comment?": "Sei sicuro di voler eliminare questo commento?",
|
"Are you sure you want to delete this comment?": "Sei sicuro di voler eliminare questo commento?",
|
||||||
@@ -226,7 +233,6 @@
|
|||||||
"Are you sure you want to unresolve this comment thread?": "Sei sicuro di voler annullare la risoluzione di questa discussione di commenti?",
|
"Are you sure you want to unresolve this comment thread?": "Sei sicuro di voler annullare la risoluzione di questa discussione di commenti?",
|
||||||
"Resolved": "Risolto",
|
"Resolved": "Risolto",
|
||||||
"No active comments.": "Nessun commento attivo.",
|
"No active comments.": "Nessun commento attivo.",
|
||||||
"No resolved comments.": "Nessun commento risolto.",
|
|
||||||
"Revoke invitation": "Revoca invito",
|
"Revoke invitation": "Revoca invito",
|
||||||
"Revoke": "Revoca",
|
"Revoke": "Revoca",
|
||||||
"Don't": "Non",
|
"Don't": "Non",
|
||||||
@@ -272,6 +278,7 @@
|
|||||||
"Add row below": "Aggiungi riga sotto",
|
"Add row below": "Aggiungi riga sotto",
|
||||||
"Delete table": "Elimina tabella",
|
"Delete table": "Elimina tabella",
|
||||||
"Info": "Informazioni",
|
"Info": "Informazioni",
|
||||||
|
"Note": "Nota",
|
||||||
"Success": "Successo",
|
"Success": "Successo",
|
||||||
"Warning": "Avviso",
|
"Warning": "Avviso",
|
||||||
"Danger": "Pericolo",
|
"Danger": "Pericolo",
|
||||||
@@ -282,6 +289,11 @@
|
|||||||
"Save & Exit": "Salva ed esci",
|
"Save & Exit": "Salva ed esci",
|
||||||
"Double-click to edit Excalidraw diagram": "Fai doppio clic per modificare il diagramma di Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Fai doppio clic per modificare il diagramma di Excalidraw",
|
||||||
"Paste link": "Incolla link",
|
"Paste link": "Incolla link",
|
||||||
|
"Paste link or search pages": "Incolla il link o cerca le pagine",
|
||||||
|
"Link to web page": "Collega a una pagina web",
|
||||||
|
"Recents": "Recenti",
|
||||||
|
"Page or URL": "Pagina o URL",
|
||||||
|
"Link title": "Titolo del link",
|
||||||
"Edit link": "Modifica link",
|
"Edit link": "Modifica link",
|
||||||
"Remove link": "Rimuovi link",
|
"Remove link": "Rimuovi link",
|
||||||
"Add link": "Aggiungi link",
|
"Add link": "Aggiungi link",
|
||||||
@@ -329,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Inserisci divisore di regola orizzontale",
|
"Insert horizontal rule divider": "Inserisci divisore di regola orizzontale",
|
||||||
"Upload any image from your device.": "Carica un'immagine dal tuo dispositivo.",
|
"Upload any image from your device.": "Carica un'immagine dal tuo dispositivo.",
|
||||||
"Upload any video from your device.": "Carica qualsiasi video dal tuo dispositivo.",
|
"Upload any video from your device.": "Carica qualsiasi video dal tuo dispositivo.",
|
||||||
|
"Upload any audio from your device.": "Carica qualsiasi audio dal tuo dispositivo.",
|
||||||
"Upload any file from your device.": "Carica qualsiasi file dal tuo dispositivo.",
|
"Upload any file from your device.": "Carica qualsiasi file dal tuo dispositivo.",
|
||||||
"Uploading {{name}}": "Caricamento di {{name}}",
|
"Uploading {{name}}": "Caricamento di {{name}}",
|
||||||
"Uploading file": "Caricamento file",
|
"Uploading file": "Caricamento file",
|
||||||
@@ -339,6 +352,12 @@
|
|||||||
"Divider": "Divisore",
|
"Divider": "Divisore",
|
||||||
"Quote": "Preventivo",
|
"Quote": "Preventivo",
|
||||||
"Image": "Immagine",
|
"Image": "Immagine",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "Incorpora PDF",
|
||||||
|
"Upload and embed a PDF file.": "Carica e incorpora un file PDF.",
|
||||||
|
"Embed as PDF": "Incorpora come PDF",
|
||||||
|
"Failed to load PDF": "Caricamento del PDF non riuscito",
|
||||||
|
"Convert to attachment": "Converti in allegato",
|
||||||
"File attachment": "Allegato file",
|
"File attachment": "Allegato file",
|
||||||
"Toggle block": "Attiva blocco",
|
"Toggle block": "Attiva blocco",
|
||||||
"Callout": "Avviso",
|
"Callout": "Avviso",
|
||||||
@@ -353,9 +372,23 @@
|
|||||||
"Insert current date": "Inserisci la data corrente",
|
"Insert current date": "Inserisci la data corrente",
|
||||||
"Draw and sketch excalidraw diagrams": "Disegna e schizza diagrammi excalidraw",
|
"Draw and sketch excalidraw diagrams": "Disegna e schizza diagrammi excalidraw",
|
||||||
"Multiple": "Multiplo",
|
"Multiple": "Multiplo",
|
||||||
|
"Turn into": "Trasforma in",
|
||||||
|
"Text align": "Allinea testo",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "Questa pagina potrebbe essere stata eliminata o spostata, oppure potresti non avere accesso.",
|
||||||
|
"Go to homepage": "Vai alla pagina principale",
|
||||||
|
"Pages you create will show up here.": "Le pagine che crei appariranno qui.",
|
||||||
"Heading {{level}}": "Intestazione {{level}}",
|
"Heading {{level}}": "Intestazione {{level}}",
|
||||||
"Toggle title": "Attiva/disattiva titolo",
|
"Toggle title": "Attiva/disattiva titolo",
|
||||||
"Write anything. Enter \"/\" for commands": "Scrivi qualcosa. Digita \"/\" per i comandi",
|
"Write anything. Enter \"/\" for commands": "Scrivi qualcosa. Digita \"/\" per i comandi",
|
||||||
|
"Write...": "Scrivi...",
|
||||||
|
"Column count": "Numero di colonne",
|
||||||
|
"{{count}} Columns": "{{count}} colonne",
|
||||||
|
"Equal columns": "Colonne uguali",
|
||||||
|
"Left sidebar": "Barra laterale sinistra",
|
||||||
|
"Right sidebar": "Barra laterale destra",
|
||||||
|
"Wide center": "Centro ampio",
|
||||||
|
"Left wide": "Ampia a sinistra",
|
||||||
|
"Right wide": "Ampia a destra",
|
||||||
"Names do not match": "I nomi non corrispondono",
|
"Names do not match": "I nomi non corrispondono",
|
||||||
"Today, {{time}}": "Oggi, {{time}}",
|
"Today, {{time}}": "Oggi, {{time}}",
|
||||||
"Yesterday, {{time}}": "Ieri, {{time}}",
|
"Yesterday, {{time}}": "Ieri, {{time}}",
|
||||||
@@ -378,6 +411,13 @@
|
|||||||
"Delete member": "Elimina membro",
|
"Delete member": "Elimina membro",
|
||||||
"Member deleted successfully": "Membro eliminato con successo",
|
"Member deleted successfully": "Membro eliminato con successo",
|
||||||
"Are you sure you want to delete this workspace member? This action is irreversible.": "Sei sicuro di voler eliminare questo membro del workspace? Questa azione è irreversibile.",
|
"Are you sure you want to delete this workspace member? This action is irreversible.": "Sei sicuro di voler eliminare questo membro del workspace? Questa azione è irreversibile.",
|
||||||
|
"Deactivate member": "Disattiva membro",
|
||||||
|
"Activate member": "Attiva membro",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "Sei sicuro di voler disattivare questo membro dello spazio di lavoro? Non potrà più accedere a questo spazio di lavoro.",
|
||||||
|
"Are you sure you want to activate this workspace member?": "Sei sicuro di voler attivare questo membro dello spazio di lavoro?",
|
||||||
|
"Deactivate": "Disattiva",
|
||||||
|
"Activate": "Attiva",
|
||||||
|
"Deactivated": "Disattivato",
|
||||||
"Move": "Sposta",
|
"Move": "Sposta",
|
||||||
"Move page": "Sposta pagina",
|
"Move page": "Sposta pagina",
|
||||||
"Move page to a different space.": "Sposta la pagina in un altro spazio.",
|
"Move page to a different space.": "Sposta la pagina in un altro spazio.",
|
||||||
@@ -405,6 +445,25 @@
|
|||||||
"Share deleted successfully": "Condivisione eliminata con successo",
|
"Share deleted successfully": "Condivisione eliminata con successo",
|
||||||
"Share not found": "Condivisione non trovata",
|
"Share not found": "Condivisione non trovata",
|
||||||
"Failed to share page": "Condivisione della pagina fallita",
|
"Failed to share page": "Condivisione della pagina fallita",
|
||||||
|
"Disable public sharing": "Disabilita la condivisione pubblica",
|
||||||
|
"Prevent members from sharing pages publicly.": "Impedisci ai membri di condividere pubblicamente le pagine.",
|
||||||
|
"Toggle public sharing": "Attiva/disattiva la condivisione pubblica",
|
||||||
|
"Toggle space public sharing": "Attiva/disattiva la condivisione pubblica nello spazio",
|
||||||
|
"Allow viewers to comment": "Consenti agli utenti di commentare",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Consenti agli utenti di aggiungere commenti alle pagine in questo spazio.",
|
||||||
|
"Toggle viewer comments": "Attiva/disattiva i commenti degli utenti",
|
||||||
|
"Public sharing is disabled at the workspace level": "La condivisione pubblica è disabilitata a livello di area di lavoro",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "Impedisci che le pagine in questo spazio vengano condivise pubblicamente.",
|
||||||
|
"Page permissions": "Autorizzazioni della pagina.",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Controlla chi può visualizzare e modificare le singole pagine. Disponibile con una licenza Enterprise.",
|
||||||
|
"Enable public sharing": "Abilita la condivisione pubblica",
|
||||||
|
"Are you sure you want to enable public sharing? Members will be able to share pages publicly.": "Sei sicuro di voler abilitare la condivisione pubblica? I membri potranno condividere le pagine pubblicamente.",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this workspace will be deleted.": "Sei sicuro di voler disabilitare la condivisione pubblica? Tutti i link condivisi esistenti in questa area di lavoro verranno eliminati.",
|
||||||
|
"Are you sure you want to enable public sharing for this space?": "Sei sicuro di voler abilitare la condivisione pubblica per questo spazio?",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this space will be deleted.": "Sei sicuro di voler disabilitare la condivisione pubblica? Tutti i link condivisi esistenti in questo spazio verranno eliminati.",
|
||||||
|
"Public sharing is disabled": "La condivisione pubblica è disabilitata",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "La condivisione pubblica è stata disabilitata a livello di area di lavoro.",
|
||||||
|
"Public sharing has been disabled for this space.": "La condivisione pubblica è stata disabilitata per questo spazio.",
|
||||||
"Copy page": "Copia pagina",
|
"Copy page": "Copia pagina",
|
||||||
"Copy page to a different space.": "Copia pagina in un altro spazio.",
|
"Copy page to a different space.": "Copia pagina in un altro spazio.",
|
||||||
"Page copied successfully": "Pagina copiata con successo",
|
"Page copied successfully": "Pagina copiata con successo",
|
||||||
@@ -487,7 +546,7 @@
|
|||||||
"Enter one of your backup codes. Each backup code can only be used once.": "Inserisci uno dei tuoi codici di backup. Ogni codice di backup può essere utilizzato solo una volta.",
|
"Enter one of your backup codes. Each backup code can only be used once.": "Inserisci uno dei tuoi codici di backup. Ogni codice di backup può essere utilizzato solo una volta.",
|
||||||
"Verify": "Verifica",
|
"Verify": "Verifica",
|
||||||
"Trash": "Cestino",
|
"Trash": "Cestino",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "Le pagine nel cestino verranno eliminate definitivamente dopo 30 giorni.",
|
"Pages in trash will be permanently deleted after {{count}} days.": "Le pagine nel cestino verranno eliminate definitivamente dopo {{count}} giorni.",
|
||||||
"Deleted": "Eliminato",
|
"Deleted": "Eliminato",
|
||||||
"No pages in trash": "Nessuna pagina nel cestino",
|
"No pages in trash": "Nessuna pagina nel cestino",
|
||||||
"Permanently delete page?": "Eliminare definitivamente la pagina?",
|
"Permanently delete page?": "Eliminare definitivamente la pagina?",
|
||||||
@@ -559,19 +618,138 @@
|
|||||||
"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.",
|
"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",
|
"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",
|
"Manage API keys for all users in the workspace": "Gestisci le chiavi API per tutti gli utenti nell'area di lavoro",
|
||||||
|
"Restrict API key creation to admins": "Limita la creazione delle chiavi API agli amministratori",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "Solo gli amministratori e i proprietari possono creare nuove chiavi API. Le chiavi dei membri esistenti continueranno a funzionare.",
|
||||||
|
"Toggle restrict API keys to admins": "Attiva/disattiva la limitazione delle chiavi API agli amministratori",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "La creazione delle chiavi API è limitata agli amministratori dal tuo amministratore dello spazio di lavoro.",
|
||||||
"AI settings": "Impostazioni AI",
|
"AI settings": "Impostazioni AI",
|
||||||
"AI search": "Ricerca AI",
|
"AI search": "Ricerca AI",
|
||||||
"AI Answer": "Risposta AI",
|
"AI Answer": "Risposta AI",
|
||||||
"Ask AI": "Chiedi all'AI",
|
"Ask AI": "Chiedi all'AI",
|
||||||
"AI is thinking...": "L'AI sta pensando...",
|
"AI is thinking...": "L'AI sta pensando...",
|
||||||
"Ask a question...": "Fai una domanda...",
|
"Ask a question...": "Fai una domanda...",
|
||||||
"AI-powered search (Ask AI)": "Ricerca potenziata dall'AI (Chiedi all'AI)",
|
"AI Answers": "Risposte AI",
|
||||||
|
"AI-powered search (AI Answers)": "Ricerca con AI (Risposte 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.",
|
"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",
|
"Toggle AI search": "Attiva/disattiva ricerca AI",
|
||||||
|
"Generative AI (Ask AI)": "AI generativa (Chiedi AI)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Abilita la generazione di contenuti con AI nell'editor. Consente agli utenti di generare, migliorare, tradurre e trasformare il testo.",
|
||||||
|
"Toggle generative AI": "Attiva/Disattiva AI generativa",
|
||||||
|
"Upgrade your plan": "Aggiorna il tuo piano",
|
||||||
|
"Available with a paid license": "Disponibile con una licenza a pagamento",
|
||||||
|
"Upgrade your license tier.": "Aggiorna il livello della tua licenza.",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "L'IA è disponibile solo nell'edizione Enterprise di Docmost. Contatta sales@docmost.com.",
|
||||||
|
"AI & MCP": "IA e MCP",
|
||||||
|
"AI": "IA",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Abilita il server MCP per consentire ad assistenti e strumenti IA di interagire con i contenuti del tuo spazio di lavoro.",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP è disponibile solo nell'edizione Enterprise di Docmost. Contatta sales@docmost.com.",
|
||||||
|
"MCP Server URL": "URL del server MCP",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Usa la tua chiave API per l'autenticazione. Puoi gestire le chiavi API nelle impostazioni del tuo account.",
|
||||||
|
"Supported tools": "Strumenti supportati",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Il tuo spazio di lavoro ha MCP abilitato. Usa la tua chiave API per collegare gli assistenti IA.",
|
||||||
|
"MCP server URL:": "URL del server MCP:",
|
||||||
|
"Learn more": "Scopri di più",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Gestisci le API key per tutti gli utenti nello spazio di lavoro. Consulta la <anchor>documentazione API</anchor> per i dettagli sull'utilizzo.",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "Consulta la <anchor>documentazione API</anchor> per i dettagli sull'utilizzo.",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "Consulta la <anchor>documentazione MCP</anchor>.",
|
||||||
"Sources": "Fonti",
|
"Sources": "Fonti",
|
||||||
"Ask AI not available for attachments": "Chiedere all'AI non è disponibile per gli allegati",
|
"AI Answers not available for attachments": "Risposte AI non disponibili per gli allegati",
|
||||||
"No answer available": "Nessuna risposta disponibile",
|
"No answer available": "Nessuna risposta disponibile",
|
||||||
"Background color": "Colore di sfondo",
|
"Background color": "Colore di sfondo",
|
||||||
"Highlight color": "Colore evidenziato",
|
"Highlight color": "Colore evidenziato",
|
||||||
"Remove color": "Rimuovi colore"
|
"Remove color": "Rimuovi colore",
|
||||||
|
"Notifications": "Notifiche",
|
||||||
|
"No notifications": "Nessuna notifica",
|
||||||
|
"No unread notifications": "Nessuna notifica non letta",
|
||||||
|
"All notifications": "Tutte le notifiche",
|
||||||
|
"Unread only": "Solo non lette",
|
||||||
|
"Mark all as read": "Segna tutto come letto",
|
||||||
|
"Mark as read": "Segna come letto",
|
||||||
|
"More options": "Altre opzioni",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> ti ha menzionato in un commento",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> ha commentato una pagina",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> ha risolto un commento",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> ti ha menzionato su una pagina",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> ti ha dato l'accesso di modifica a una pagina",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> ti ha dato l'accesso di visualizzazione a una pagina",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> ha aggiornato una pagina.",
|
||||||
|
"Watch page": "Segui pagina",
|
||||||
|
"Stop watching": "Smetti di seguire",
|
||||||
|
"Email notifications": "Notifiche email",
|
||||||
|
"Page updates": "Aggiornamenti pagina",
|
||||||
|
"Get notified when pages you watch are updated.": "Ricevi una notifica quando le pagine che segui vengono aggiornate.",
|
||||||
|
"Page mentions": "Menzioni nella pagina",
|
||||||
|
"Get notified when someone mentions you on a page.": "Ricevi una notifica quando qualcuno ti menziona su una pagina.",
|
||||||
|
"Comment mentions": "Menzioni nei commenti",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Ricevi una notifica quando qualcuno ti menziona in un commento.",
|
||||||
|
"New comments": "Nuovi commenti",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Ricevi una notifica sui nuovi commenti nelle discussioni a cui partecipi.",
|
||||||
|
"Resolved comments": "Commenti risolti",
|
||||||
|
"Get notified when your comment is resolved.": "Ricevi una notifica quando il tuo commento viene risolto.",
|
||||||
|
"You are now watching this page": "Ora stai seguendo questa pagina",
|
||||||
|
"You are no longer watching this page": "Non stai più seguendo questa pagina",
|
||||||
|
"Direct": "Diretto",
|
||||||
|
"Updates": "Aggiornamenti",
|
||||||
|
"Today": "Oggi",
|
||||||
|
"Yesterday": "Ieri",
|
||||||
|
"This week": "Questa settimana",
|
||||||
|
"Older": "Più vecchie",
|
||||||
|
"Restricted page": "Pagina con accesso ristretto",
|
||||||
|
"Restricted pages cannot be shared publicly.": "Le pagine con accesso ristretto non possono essere condivise pubblicamente.",
|
||||||
|
"Restricted by parent": "Limitata dalla pagina genitore",
|
||||||
|
"Restricted": "Limitata",
|
||||||
|
"Open": "Aperta",
|
||||||
|
"Inherits restrictions from ancestor page": "Eredita le restrizioni dalla pagina genitore",
|
||||||
|
"Only people listed below can access this page": "Solo le persone elencate di seguito possono accedere a questa pagina",
|
||||||
|
"Everyone in this space can access": "Chiunque in questo spazio può accedere",
|
||||||
|
"No additional restrictions on this page": "Nessuna restrizione aggiuntiva su questa pagina",
|
||||||
|
"Only specific people can access": "Solo persone specifiche possono accedere",
|
||||||
|
"Use only inherited restrictions": "Usa solo le restrizioni ereditate",
|
||||||
|
"Add restrictions on top of inherited": "Aggiungi restrizioni oltre a quelle ereditate",
|
||||||
|
"Inherited restriction": "Restrizione ereditata",
|
||||||
|
"Access limited by": "Accesso limitato da",
|
||||||
|
"Restrict access to control who can view and edit this page": "Limita l'accesso per controllare chi può visualizzare e modificare questa pagina",
|
||||||
|
"Add additional restrictions specific to this page": "Aggiungi restrizioni aggiuntive specifiche per questa pagina",
|
||||||
|
"Access": "Accesso",
|
||||||
|
"People with access": "Persone con accesso",
|
||||||
|
"Remove all": "Rimuovi tutto",
|
||||||
|
"Remove access": "Rimuovi accesso",
|
||||||
|
"Remove all access": "Rimuovi tutti gli accessi",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "Sei sicuro di voler rimuovere l'accesso di questo membro alla pagina?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "Sei sicuro di voler rimuovere tutti gli accessi specifici? Questo renderà la pagina accessibile a tutti nello spazio.",
|
||||||
|
"Trash retention": "Conservazione del cestino",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "Le pagine nel cestino verranno eliminate definitivamente dopo questo periodo.",
|
||||||
|
"Trash retention updated": "Conservazione del cestino aggiornata",
|
||||||
|
"Failed to update trash retention": "Impossibile aggiornare la conservazione del cestino",
|
||||||
|
"Removed page restriction": "Restrizione della pagina rimossa",
|
||||||
|
"Added page permission": "Permesso sulla pagina aggiunto",
|
||||||
|
"Removed page permission": "Permesso sulla pagina rimosso",
|
||||||
|
"Verifying your email": "Verifica della tua email",
|
||||||
|
"Please wait...": "Attendere...",
|
||||||
|
"Verification failed. The link may have expired.": "Verifica non riuscita. Il link potrebbe essere scaduto.",
|
||||||
|
"Check your email": "Controlla la tua email",
|
||||||
|
"We sent a verification link to {{email}}.": "Abbiamo inviato un link di verifica a {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Abbiamo inviato un link di verifica alla tua email.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Clicca sul link per verificare la tua email e accedere al tuo workspace.",
|
||||||
|
"Resend verification email": "Invia nuovamente l'email di verifica",
|
||||||
|
"Verification email sent. Please check your inbox.": "Email di verifica inviata. Controlla la tua casella di posta.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Invio dell'email di verifica non riuscito. Si prega di riprovare.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Ti abbiamo inviato un'email con i workspace associati.",
|
||||||
|
"Load more": "Carica altro",
|
||||||
|
"Log out of all devices": "Disconnetti da tutti i dispositivi",
|
||||||
|
"Log out of all sessions except this device": "Disconnetti da tutte le sessioni tranne questo dispositivo",
|
||||||
|
"This Device": "Questo dispositivo",
|
||||||
|
"Unknown device": "Dispositivo sconosciuto",
|
||||||
|
"No active sessions": "Nessuna sessione attiva",
|
||||||
|
"Session revoked": "Sessione revocata",
|
||||||
|
"All other sessions revoked": "Tutte le altre sessioni revocate",
|
||||||
|
"Last used": "Ultimo utilizzo",
|
||||||
|
"Created": "Creato",
|
||||||
|
"Rename": "Rinomina",
|
||||||
|
"Publish": "Pubblica",
|
||||||
|
"Security": "Sicurezza",
|
||||||
|
"Enforce SSO": "Forza SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Una volta attivata, i membri non potranno più accedere con email e password."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -116,6 +116,7 @@
|
|||||||
"No group found": "グループが見つかりません",
|
"No group found": "グループが見つかりません",
|
||||||
"No page history saved yet.": "ページ履歴がありません",
|
"No page history saved yet.": "ページ履歴がありません",
|
||||||
"No pages yet": "ページがありません",
|
"No pages yet": "ページがありません",
|
||||||
|
"No shared pages": "共有ページはありません。",
|
||||||
"No results found...": "結果が見つかりません",
|
"No results found...": "結果が見つかりません",
|
||||||
"No user found": "ユーザーが見つかりません",
|
"No user found": "ユーザーが見つかりません",
|
||||||
"Overview": "概要",
|
"Overview": "概要",
|
||||||
@@ -123,11 +124,14 @@
|
|||||||
"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": "ページ",
|
||||||
"Password": "パスワード",
|
"Password": "パスワード",
|
||||||
"Password changed successfully": "パスワードを変更しました",
|
"Password changed successfully": "パスワードを変更しました",
|
||||||
|
"People": "メンバー",
|
||||||
"Pending": "保留中",
|
"Pending": "保留中",
|
||||||
"Please confirm your action": "アクションを確認してください",
|
"Please confirm your action": "アクションを確認してください",
|
||||||
"Preferences": "設定",
|
"Preferences": "設定",
|
||||||
@@ -205,6 +209,9 @@
|
|||||||
"Reply...": "返信...",
|
"Reply...": "返信...",
|
||||||
"Error loading comments.": "コメントの読み込みに失敗しました",
|
"Error loading comments.": "コメントの読み込みに失敗しました",
|
||||||
"No comments yet.": "コメントがありません",
|
"No comments yet.": "コメントがありません",
|
||||||
|
"No open comments.": "未解決のコメントはありません。",
|
||||||
|
"No resolved comments.": "解決済みのコメントはありません",
|
||||||
|
"Add a comment...": "コメントを追加...",
|
||||||
"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?": "このコメントを削除してもよろしいですか?",
|
||||||
@@ -226,7 +233,6 @@
|
|||||||
"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.": "解決済みのコメントはありません",
|
|
||||||
"Revoke invitation": "招待を取り消す",
|
"Revoke invitation": "招待を取り消す",
|
||||||
"Revoke": "取り消す",
|
"Revoke": "取り消す",
|
||||||
"Don't": "取り消さない",
|
"Don't": "取り消さない",
|
||||||
@@ -272,6 +278,7 @@
|
|||||||
"Add row below": "下に行を追加",
|
"Add row below": "下に行を追加",
|
||||||
"Delete table": "テーブルを削除",
|
"Delete table": "テーブルを削除",
|
||||||
"Info": "情報",
|
"Info": "情報",
|
||||||
|
"Note": "ノート",
|
||||||
"Success": "成功",
|
"Success": "成功",
|
||||||
"Warning": "警告",
|
"Warning": "警告",
|
||||||
"Danger": "危険",
|
"Danger": "危険",
|
||||||
@@ -282,6 +289,11 @@
|
|||||||
"Save & Exit": "保存して終了",
|
"Save & Exit": "保存して終了",
|
||||||
"Double-click to edit Excalidraw diagram": "ダブルクリックして Excalidraw 図を編集",
|
"Double-click to edit Excalidraw diagram": "ダブルクリックして Excalidraw 図を編集",
|
||||||
"Paste link": "リンクを貼り付け",
|
"Paste link": "リンクを貼り付け",
|
||||||
|
"Paste link or search pages": "リンクを貼り付けるかページを検索してください。 ",
|
||||||
|
"Link to web page": "ウェブページへのリンク",
|
||||||
|
"Recents": "最近使用したもの",
|
||||||
|
"Page or URL": "ページまたはURL",
|
||||||
|
"Link title": "リンクタイトル",
|
||||||
"Edit link": "リンクを編集",
|
"Edit link": "リンクを編集",
|
||||||
"Remove link": "リンクを削除",
|
"Remove link": "リンクを削除",
|
||||||
"Add link": "リンクを追加",
|
"Add link": "リンクを追加",
|
||||||
@@ -329,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "区切り線を挿入します",
|
"Insert horizontal rule divider": "区切り線を挿入します",
|
||||||
"Upload any image from your device.": "デバイスから画像をアップロードします",
|
"Upload any image from your device.": "デバイスから画像をアップロードします",
|
||||||
"Upload any video from your device.": "デバイスから動画をアップロードします",
|
"Upload any video from your device.": "デバイスから動画をアップロードします",
|
||||||
|
"Upload any audio from your device.": "デバイスから音声ファイルをアップロードします。",
|
||||||
"Upload any file from your device.": "デバイスからファイルをアップロードします",
|
"Upload any file from your device.": "デバイスからファイルをアップロードします",
|
||||||
"Uploading {{name}}": "{{name}} をアップロード中",
|
"Uploading {{name}}": "{{name}} をアップロード中",
|
||||||
"Uploading file": "ファイルをアップロード中",
|
"Uploading file": "ファイルをアップロード中",
|
||||||
@@ -339,6 +352,12 @@
|
|||||||
"Divider": "区切り線",
|
"Divider": "区切り線",
|
||||||
"Quote": "引用",
|
"Quote": "引用",
|
||||||
"Image": "画像",
|
"Image": "画像",
|
||||||
|
"Audio": "音声。",
|
||||||
|
"Embed PDF": "PDFを埋め込む",
|
||||||
|
"Upload and embed a PDF file.": "PDFファイルをアップロードして埋め込みます。",
|
||||||
|
"Embed as PDF": "PDFとして埋め込む",
|
||||||
|
"Failed to load PDF": "PDFの読み込みに失敗しました",
|
||||||
|
"Convert to attachment": "添付ファイルに変換",
|
||||||
"File attachment": "ファイル添付",
|
"File attachment": "ファイル添付",
|
||||||
"Toggle block": "ブロックを切り替える",
|
"Toggle block": "ブロックを切り替える",
|
||||||
"Callout": "コールアウト",
|
"Callout": "コールアウト",
|
||||||
@@ -353,9 +372,23 @@
|
|||||||
"Insert current date": "現在の日付を挿入します",
|
"Insert current date": "現在の日付を挿入します",
|
||||||
"Draw and sketch excalidraw diagrams": "Excalidraw 図を挿入します",
|
"Draw and sketch excalidraw diagrams": "Excalidraw 図を挿入します",
|
||||||
"Multiple": "複数",
|
"Multiple": "複数",
|
||||||
|
"Turn into": "変換する",
|
||||||
|
"Text align": "テキストの配置",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "このページは削除されたか移動されたか、またはアクセス権がない可能性があります。},{",
|
||||||
|
"Go to homepage": "ホームページへ移動",
|
||||||
|
"Pages you create will show up here.": "ここに作成したページが表示されます。",
|
||||||
"Heading {{level}}": "見出し {{level}}",
|
"Heading {{level}}": "見出し {{level}}",
|
||||||
"Toggle title": "タイトルの表示/非表示を切り替える",
|
"Toggle title": "タイトルの表示/非表示を切り替える",
|
||||||
"Write anything. Enter \"/\" for commands": "文字を入力するか、「/」でコマンドを呼び出します",
|
"Write anything. Enter \"/\" for commands": "文字を入力するか、「/」でコマンドを呼び出します",
|
||||||
|
"Write...": "ここに入力...",
|
||||||
|
"Column count": "列数",
|
||||||
|
"{{count}} Columns": "{{count}}列",
|
||||||
|
"Equal columns": "均等な列",
|
||||||
|
"Left sidebar": "左サイドバー",
|
||||||
|
"Right sidebar": "右サイドバー",
|
||||||
|
"Wide center": "中央ワイド",
|
||||||
|
"Left wide": "左ワイド",
|
||||||
|
"Right wide": "右ワイド",
|
||||||
"Names do not match": "名前が一致しません",
|
"Names do not match": "名前が一致しません",
|
||||||
"Today, {{time}}": "今日、{{time}}",
|
"Today, {{time}}": "今日、{{time}}",
|
||||||
"Yesterday, {{time}}": "昨日、{{time}}",
|
"Yesterday, {{time}}": "昨日、{{time}}",
|
||||||
@@ -378,6 +411,13 @@
|
|||||||
"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.": "このメンバーを削除してもよろしいですか?この操作は取り消せません",
|
||||||
|
"Deactivate member": "メンバーを無効化",
|
||||||
|
"Activate member": "メンバーを有効化",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "本当にこのワークスペースのメンバーを無効化しますか?無効化すると、このワークスペースにアクセスできなくなります。",
|
||||||
|
"Are you sure you want to activate this workspace member?": "本当にこのワークスペースのメンバーを有効化しますか?",
|
||||||
|
"Deactivate": "無効化",
|
||||||
|
"Activate": "有効化",
|
||||||
|
"Deactivated": "無効化済み",
|
||||||
"Move": "移動",
|
"Move": "移動",
|
||||||
"Move page": "ページを移動",
|
"Move page": "ページを移動",
|
||||||
"Move page to a different space.": "ページを別のスペースに移動します",
|
"Move page to a different space.": "ページを別のスペースに移動します",
|
||||||
@@ -405,6 +445,25 @@
|
|||||||
"Share deleted successfully": "共有を削除しました",
|
"Share deleted successfully": "共有を削除しました",
|
||||||
"Share not found": "共有が見つかりません",
|
"Share not found": "共有が見つかりません",
|
||||||
"Failed to share page": "ページの共有に失敗しました",
|
"Failed to share page": "ページの共有に失敗しました",
|
||||||
|
"Disable public sharing": "公開共有を無効にする",
|
||||||
|
"Prevent members from sharing pages publicly.": "メンバーがページを公開で共有するのを防ぐ。",
|
||||||
|
"Toggle public sharing": "公開共有を切り替える",
|
||||||
|
"Toggle space public sharing": "スペースの公開共有を切り替える",
|
||||||
|
"Allow viewers to comment": "閲覧者によるコメントを許可",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "このスペース内のページに閲覧者がコメントを追加できるようにします。",
|
||||||
|
"Toggle viewer comments": "閲覧者コメントの切り替え",
|
||||||
|
"Public sharing is disabled at the workspace level": "ワークスペースレベルで公開共有が無効になっています",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "このスペース内のページが公開で共有されるのを防ぐ。",
|
||||||
|
"Page permissions": "ページのアクセス権",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "個々のページを誰が表示・編集できるかを制御します。エンタープライズライセンスで利用可能です。",
|
||||||
|
"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 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 disable public sharing? All existing shared links in this space will be deleted.": "本当に公開共有を無効にしますか?このスペースのすべての既存の共有リンクが削除されます。",
|
||||||
|
"Public sharing is disabled": "公開共有が無効になっています",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "ワークスペースレベルで公開共有が無効になりました。",
|
||||||
|
"Public sharing has been disabled for this space.": "このスペースで公開共有が無効になりました。",
|
||||||
"Copy page": "ページをコピー",
|
"Copy page": "ページをコピー",
|
||||||
"Copy page to a different space.": "ページを別のスペースにコピーします",
|
"Copy page to a different space.": "ページを別のスペースにコピーします",
|
||||||
"Page copied successfully": "ページをコピーしました",
|
"Page copied successfully": "ページをコピーしました",
|
||||||
@@ -487,7 +546,7 @@
|
|||||||
"Enter one of your backup codes. Each backup code can only be used once.": "バックアップコードを入力してください。各コードは1回のみ使用可能です",
|
"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 {{count}} days.": "{count, plural, other {ゴミ箱内のページは#日後に完全に削除されます。}}",
|
||||||
"Deleted": "削除",
|
"Deleted": "削除",
|
||||||
"No pages in trash": "ごみ箱にページがありません",
|
"No pages in trash": "ごみ箱にページがありません",
|
||||||
"Permanently delete page?": "ページを完全に削除しますか?",
|
"Permanently delete page?": "ページを完全に削除しますか?",
|
||||||
@@ -559,19 +618,138 @@
|
|||||||
"This action cannot be undone. Any applications using this API key will stop working.": "この操作は取り消せません。このAPIキーを使用しているアプリケーションは動作しなくなります",
|
"This action cannot be undone. Any applications using this API key will stop working.": "この操作は取り消せません。このAPIキーを使用しているアプリケーションは動作しなくなります",
|
||||||
"Update API key": "APIキーを更新",
|
"Update API key": "APIキーを更新",
|
||||||
"Manage API keys for all users in the workspace": "ワークスペース内のすべてのユーザーのAPIキーを管理",
|
"Manage API keys for all users in the workspace": "ワークスペース内のすべてのユーザーのAPIキーを管理",
|
||||||
|
"Restrict API key creation to admins": "APIキーの作成を管理者のみに制限する",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "新しいAPIキーを作成できるのは管理者とオーナーのみです。既存のメンバーキーは引き続き有効です。",
|
||||||
|
"Toggle restrict API keys to admins": "APIキーの作成制限(管理者のみ)を切り替える",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "ワークスペース管理者によってAPIキーの作成が管理者のみに制限されています。",
|
||||||
"AI settings": "AI設定",
|
"AI settings": "AI設定",
|
||||||
"AI search": "AI検索",
|
"AI search": "AI検索",
|
||||||
"AI Answer": "AI回答",
|
"AI Answer": "AI回答",
|
||||||
"Ask AI": "AIに質問する",
|
"Ask AI": "AIに質問する",
|
||||||
"AI is thinking...": "AIが考え中...",
|
"AI is thinking...": "AIが考え中...",
|
||||||
"Ask a question...": "質問を入力...",
|
"Ask a question...": "質問を入力...",
|
||||||
"AI-powered search (Ask AI)": "AIによる検索(AIに質問)",
|
"AI Answers": "AI回答",
|
||||||
|
"AI-powered search (AI Answers)": "AI搭載検索 (AI回答)",
|
||||||
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "AI検索はベクター埋め込みを使用してワークスペース全体の意味検索を実現します",
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "AI検索はベクター埋め込みを使用してワークスペース全体の意味検索を実現します",
|
||||||
"Toggle AI search": "AI検索を切り替え",
|
"Toggle AI search": "AI検索を切り替え",
|
||||||
|
"Generative AI (Ask AI)": "生成AI (Ask AI)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "エディターでAIを活用したコンテンツ生成を有効にします。ユーザーがテキストの生成、改善、翻訳、および変換を行うことができます。",
|
||||||
|
"Toggle generative AI": "生成AIを切り替える",
|
||||||
|
"Upgrade your plan": "プランをアップグレードする",
|
||||||
|
"Available with a paid license": "有料ライセンスで利用可能",
|
||||||
|
"Upgrade your license tier.": "ライセンスタイアをアップグレードしてください。",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI は Docmost のエンタープライズ版でのみ利用可能です。sales@docmost.com までお問い合わせください。",
|
||||||
|
"AI & MCP": "AI と MCP",
|
||||||
|
"AI": "AI",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "モデルコンテキストプロトコル(MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "MCP サーバーを有効にして、AI アシスタントやツールがワークスペースのコンテンツとやり取りできるようにします。",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP は Docmost のエンタープライズ版でのみ利用可能です。sales@docmost.com までお問い合わせください。",
|
||||||
|
"MCP Server URL": "MCP サーバーの URL",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "認証には API キーを使用してください。API キーはアカウント設定で管理できます。",
|
||||||
|
"Supported tools": "サポートされているツール",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "このワークスペースでは MCP が有効になっています。AI アシスタントを接続するには API キーを使用してください。",
|
||||||
|
"MCP server URL:": "MCP サーバーの URL:",
|
||||||
|
"Learn more": "詳細を見る",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "ワークスペース内のすべてのユーザーのAPIキーを管理します。利用方法の詳細は<anchor>APIドキュメント</anchor>をご覧ください。",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "利用方法の詳細は<anchor>APIドキュメント</anchor>をご覧ください。",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "<anchor>MCPドキュメント</anchor>をご覧ください。",
|
||||||
"Sources": "ソース",
|
"Sources": "ソース",
|
||||||
"Ask AI not available for attachments": "添付ファイルにはAI質問は利用できません",
|
"AI Answers not available for attachments": "添付ファイルにはAI回答を利用できません",
|
||||||
"No answer available": "回答がありません",
|
"No answer available": "回答がありません",
|
||||||
"Background color": "背景色",
|
"Background color": "背景色",
|
||||||
"Highlight color": "ハイライト色",
|
"Highlight color": "ハイライト色",
|
||||||
"Remove color": "色を削除"
|
"Remove color": "色を削除",
|
||||||
|
"Notifications": "通知",
|
||||||
|
"No notifications": "通知なし",
|
||||||
|
"No unread notifications": "未読の通知はありません",
|
||||||
|
"All notifications": "すべての通知",
|
||||||
|
"Unread only": "未読のみ",
|
||||||
|
"Mark all as read": "すべてを既読にする",
|
||||||
|
"Mark as read": "既読にする",
|
||||||
|
"More options": "その他のオプション",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold>さんがコメントであなたに言及しました",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold>さんがページにコメントしました",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold>さんがコメントを解決しました",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>さんがページであなたに言及しました",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>さんがページの編集権限をあなたに付与しました",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>さんがページの閲覧権限をあなたに付与しました",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold>さんがページを更新しました。",
|
||||||
|
"Watch page": "ページをウォッチ",
|
||||||
|
"Stop watching": "ウォッチを解除",
|
||||||
|
"Email notifications": "メール通知",
|
||||||
|
"Page updates": "ページの更新",
|
||||||
|
"Get notified when pages you watch are updated.": "ウォッチしているページが更新されたときに通知を受け取ります。",
|
||||||
|
"Page mentions": "ページでの言及",
|
||||||
|
"Get notified when someone mentions you on a page.": "誰かがページであなたに言及したとき通知を受け取ります。",
|
||||||
|
"Comment mentions": "コメントでの言及",
|
||||||
|
"Get notified when someone mentions you in a comment.": "誰かがコメントであなたに言及したとき通知を受け取ります。",
|
||||||
|
"New comments": "新しいコメント",
|
||||||
|
"Get notified about new comments on threads you participate in.": "参加しているスレッドに新しいコメントがあると通知されます。",
|
||||||
|
"Resolved comments": "解決済みコメント",
|
||||||
|
"Get notified when your comment is resolved.": "あなたのコメントが解決されたとき通知を受け取ります。",
|
||||||
|
"You are now watching this page": "このページをウォッチしています",
|
||||||
|
"You are no longer watching this page": "このページのウォッチを解除しました",
|
||||||
|
"Direct": "直接",
|
||||||
|
"Updates": "アップデート",
|
||||||
|
"Today": "今日",
|
||||||
|
"Yesterday": "昨日",
|
||||||
|
"This week": "今週",
|
||||||
|
"Older": "以前のもの",
|
||||||
|
"Restricted page": "アクセス制限されたページ",
|
||||||
|
"Restricted pages cannot be shared publicly.": "アクセス制限されたページは公開共有できません。",
|
||||||
|
"Restricted by parent": "親ページによって制限されています",
|
||||||
|
"Restricted": "制限あり",
|
||||||
|
"Open": "制限なし",
|
||||||
|
"Inherits restrictions from ancestor page": "上位ページから制限を継承しています",
|
||||||
|
"Only people listed below can access this page": "以下に記載されている人のみがこのページにアクセスできます",
|
||||||
|
"Everyone in this space can access": "このスペース内の全員がアクセスできます",
|
||||||
|
"No additional restrictions on this page": "このページに追加の制限はありません",
|
||||||
|
"Only specific people can access": "特定の人のみがアクセスできます",
|
||||||
|
"Use only inherited restrictions": "継承された制限のみを適用する",
|
||||||
|
"Add restrictions on top of inherited": "継承された制限に追加の制限を加える",
|
||||||
|
"Inherited restriction": "継承された制限",
|
||||||
|
"Access limited by": "アクセス制限元",
|
||||||
|
"Restrict access to control who can view and edit this page": "このページを誰が表示・編集できるかを制御するためにアクセスを制限します",
|
||||||
|
"Add additional restrictions specific to this page": "このページ固有の追加制限を設定する",
|
||||||
|
"Access": "アクセス",
|
||||||
|
"People with access": "アクセスできる人",
|
||||||
|
"Remove all": "すべてを削除",
|
||||||
|
"Remove access": "アクセス権を削除",
|
||||||
|
"Remove all access": "すべてのアクセス権を削除",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "このメンバーのページへのアクセス権を削除してもよろしいですか?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "すべての特定のアクセスを削除してもよろしいですか?これによりページはスペース内の全員に公開されます。",
|
||||||
|
"Trash retention": "ゴミ箱の保持期間",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "この期間を過ぎるとゴミ箱内のページは完全に削除されます。",
|
||||||
|
"Trash retention updated": "ゴミ箱保持期間が更新されました",
|
||||||
|
"Failed to update trash retention": "ゴミ箱保持期間の更新に失敗しました",
|
||||||
|
"Removed page restriction": "ページの制限を解除しました",
|
||||||
|
"Added page permission": "ページの権限を追加しました",
|
||||||
|
"Removed page permission": "ページの権限を削除しました",
|
||||||
|
"Verifying your email": "メールを確認中",
|
||||||
|
"Please wait...": "お待ちください…",
|
||||||
|
"Verification failed. The link may have expired.": "認証に失敗しました。リンクの有効期限が切れている可能性があります。",
|
||||||
|
"Check your email": "メールを確認してください",
|
||||||
|
"We sent a verification link to {{email}}.": "確認用リンクを{{email}}に送信しました。",
|
||||||
|
"We sent a verification link to your email.": "確認用リンクをあなたのメールアドレスに送信しました。",
|
||||||
|
"Click the link to verify your email and access your workspace.": "リンクをクリックしてメールを認証し、ワークスペースにアクセスしてください。",
|
||||||
|
"Resend verification email": "確認メールを再送信",
|
||||||
|
"Verification email sent. Please check your inbox.": "確認メールを送信しました。受信箱をご確認ください。",
|
||||||
|
"Failed to resend verification email. Please try again.": "確認メールの再送信に失敗しました。もう一度お試しください。",
|
||||||
|
"We've sent you an email with your associated workspaces.": "紐づいているワークスペース情報をメールでお送りしました。",
|
||||||
|
"Load more": "もっと見る",
|
||||||
|
"Log out of all devices": "すべての端末からログアウト",
|
||||||
|
"Log out of all sessions except this device": "この端末以外の全セッションからログアウト",
|
||||||
|
"This Device": "このデバイス",
|
||||||
|
"Unknown device": "不明な端末",
|
||||||
|
"No active sessions": "アクティブなセッションはありません",
|
||||||
|
"Session revoked": "セッションが取り消されました",
|
||||||
|
"All other sessions revoked": "他のすべてのセッションが取り消されました",
|
||||||
|
"Last used": "最終使用",
|
||||||
|
"Created": "作成日",
|
||||||
|
"Rename": "名前を変更",
|
||||||
|
"Publish": "公開する",
|
||||||
|
"Security": "セキュリティ",
|
||||||
|
"Enforce SSO": "SSOを強制する",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "一度SSOが強制されると、メールとパスワードでログインできなくなります。"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -116,6 +116,7 @@
|
|||||||
"No group found": "팀을 찾을 수 없음",
|
"No group found": "팀을 찾을 수 없음",
|
||||||
"No page history saved yet.": "아직 저장된 페이지 기록이 없습니다.",
|
"No page history saved yet.": "아직 저장된 페이지 기록이 없습니다.",
|
||||||
"No pages yet": "아직 페이지가 없습니다",
|
"No pages yet": "아직 페이지가 없습니다",
|
||||||
|
"No shared pages": "공유된 페이지가 없습니다.",
|
||||||
"No results found...": "결과를 찾을 수 없습니다...",
|
"No results found...": "결과를 찾을 수 없습니다...",
|
||||||
"No user found": "사용자를 찾을 수 없음",
|
"No user found": "사용자를 찾을 수 없음",
|
||||||
"Overview": "개요",
|
"Overview": "개요",
|
||||||
@@ -123,11 +124,14 @@
|
|||||||
"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": "페이지",
|
||||||
"Password": "비밀번호",
|
"Password": "비밀번호",
|
||||||
"Password changed successfully": "비밀번호 변경 완료",
|
"Password changed successfully": "비밀번호 변경 완료",
|
||||||
|
"People": "사용자",
|
||||||
"Pending": "대기 중",
|
"Pending": "대기 중",
|
||||||
"Please confirm your action": "작업을 확인해 주세요",
|
"Please confirm your action": "작업을 확인해 주세요",
|
||||||
"Preferences": "설정",
|
"Preferences": "설정",
|
||||||
@@ -205,6 +209,9 @@
|
|||||||
"Reply...": "답글...",
|
"Reply...": "답글...",
|
||||||
"Error loading comments.": "댓글 불러오기 오류.",
|
"Error loading comments.": "댓글 불러오기 오류.",
|
||||||
"No comments yet.": "아직 댓글이 없습니다.",
|
"No comments yet.": "아직 댓글이 없습니다.",
|
||||||
|
"No open comments.": "열린 댓글이 없습니다.",
|
||||||
|
"No resolved comments.": "해결된 댓글이 없습니다.",
|
||||||
|
"Add a comment...": "댓글 추가...",
|
||||||
"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?": "이 댓글을 삭제하시겠습니까?",
|
||||||
@@ -226,7 +233,6 @@
|
|||||||
"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.": "해결된 댓글이 없습니다.",
|
|
||||||
"Revoke invitation": "초대 취소",
|
"Revoke invitation": "초대 취소",
|
||||||
"Revoke": "취소",
|
"Revoke": "취소",
|
||||||
"Don't": "하지 않음",
|
"Don't": "하지 않음",
|
||||||
@@ -272,6 +278,7 @@
|
|||||||
"Add row below": "아래에 행 추가",
|
"Add row below": "아래에 행 추가",
|
||||||
"Delete table": "테이블 삭제",
|
"Delete table": "테이블 삭제",
|
||||||
"Info": "정보",
|
"Info": "정보",
|
||||||
|
"Note": "참고",
|
||||||
"Success": "완료",
|
"Success": "완료",
|
||||||
"Warning": "주의",
|
"Warning": "주의",
|
||||||
"Danger": "위험",
|
"Danger": "위험",
|
||||||
@@ -282,6 +289,11 @@
|
|||||||
"Save & Exit": "저장 후 나가기",
|
"Save & Exit": "저장 후 나가기",
|
||||||
"Double-click to edit Excalidraw diagram": "Excalidraw diagram을 편집하려면 더블 클릭하세요",
|
"Double-click to edit Excalidraw diagram": "Excalidraw diagram을 편집하려면 더블 클릭하세요",
|
||||||
"Paste link": "링크 붙여넣기",
|
"Paste link": "링크 붙여넣기",
|
||||||
|
"Paste link or search pages": "링크를 붙여넣거나 페이지를 검색",
|
||||||
|
"Link to web page": "웹페이지에 링크하기",
|
||||||
|
"Recents": "최근 항목",
|
||||||
|
"Page or URL": "페이지 또는 URL",
|
||||||
|
"Link title": "링크 제목",
|
||||||
"Edit link": "링크 수정",
|
"Edit link": "링크 수정",
|
||||||
"Remove link": "링크 제거",
|
"Remove link": "링크 제거",
|
||||||
"Add link": "링크 추가",
|
"Add link": "링크 추가",
|
||||||
@@ -329,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "가로 구분선 삽입",
|
"Insert horizontal rule divider": "가로 구분선 삽입",
|
||||||
"Upload any image from your device.": "기기에서 이미지를 업로드하세요.",
|
"Upload any image from your device.": "기기에서 이미지를 업로드하세요.",
|
||||||
"Upload any video from your device.": "기기에서 비디오를 업로드하세요.",
|
"Upload any video from your device.": "기기에서 비디오를 업로드하세요.",
|
||||||
|
"Upload any audio from your device.": "기기에서 오디오를 업로드하세요.",
|
||||||
"Upload any file from your device.": "기기에서 파일을 업로드하세요.",
|
"Upload any file from your device.": "기기에서 파일을 업로드하세요.",
|
||||||
"Uploading {{name}}": "{{name}} 업로드 중",
|
"Uploading {{name}}": "{{name}} 업로드 중",
|
||||||
"Uploading file": "파일 업로드 중",
|
"Uploading file": "파일 업로드 중",
|
||||||
@@ -339,6 +352,12 @@
|
|||||||
"Divider": "구분선",
|
"Divider": "구분선",
|
||||||
"Quote": "인용",
|
"Quote": "인용",
|
||||||
"Image": "이미지",
|
"Image": "이미지",
|
||||||
|
"Audio": "오디오.",
|
||||||
|
"Embed PDF": "PDF 임베드",
|
||||||
|
"Upload and embed a PDF file.": "PDF 파일을 업로드하고 임베드하세요.",
|
||||||
|
"Embed as PDF": "PDF로 임베드",
|
||||||
|
"Failed to load PDF": "PDF 로드 실패",
|
||||||
|
"Convert to attachment": "첨부 파일로 변환",
|
||||||
"File attachment": "파일 첨부",
|
"File attachment": "파일 첨부",
|
||||||
"Toggle block": "블록 토글",
|
"Toggle block": "블록 토글",
|
||||||
"Callout": "경고 상자",
|
"Callout": "경고 상자",
|
||||||
@@ -353,9 +372,23 @@
|
|||||||
"Insert current date": "현재 날짜 삽입",
|
"Insert current date": "현재 날짜 삽입",
|
||||||
"Draw and sketch excalidraw diagrams": "Excalidraw diagram 그리기 및 스케치",
|
"Draw and sketch excalidraw diagrams": "Excalidraw diagram 그리기 및 스케치",
|
||||||
"Multiple": "복제",
|
"Multiple": "복제",
|
||||||
|
"Turn into": "변경하기",
|
||||||
|
"Text align": "텍스트 정렬",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "이 페이지는 삭제되었거나 이동되었거나 접근 권한이 없을 수 있습니다.",
|
||||||
|
"Go to homepage": "홈으로 이동",
|
||||||
|
"Pages you create will show up here.": "여기에 생성한 페이지가 표시됩니다.",
|
||||||
"Heading {{level}}": "제목 {{level}}",
|
"Heading {{level}}": "제목 {{level}}",
|
||||||
"Toggle title": "제목 토글",
|
"Toggle title": "제목 토글",
|
||||||
"Write anything. Enter \"/\" for commands": "아무거나 입력하세요. 명령어를 사용하려면 \"/\"를 입력하세요",
|
"Write anything. Enter \"/\" for commands": "아무거나 입력하세요. 명령어를 사용하려면 \"/\"를 입력하세요",
|
||||||
|
"Write...": "작성...",
|
||||||
|
"Column count": "열 개수",
|
||||||
|
"{{count}} Columns": "{{count}}열",
|
||||||
|
"Equal columns": "열 너비 균등",
|
||||||
|
"Left sidebar": "왼쪽 사이드바",
|
||||||
|
"Right sidebar": "오른쪽 사이드바",
|
||||||
|
"Wide center": "가운데 넓게",
|
||||||
|
"Left wide": "왼쪽 넓게",
|
||||||
|
"Right wide": "오른쪽 넓게",
|
||||||
"Names do not match": "이름이 일치하지 않습니다",
|
"Names do not match": "이름이 일치하지 않습니다",
|
||||||
"Today, {{time}}": "오늘, {{time}}",
|
"Today, {{time}}": "오늘, {{time}}",
|
||||||
"Yesterday, {{time}}": "어제, {{time}}",
|
"Yesterday, {{time}}": "어제, {{time}}",
|
||||||
@@ -378,6 +411,13 @@
|
|||||||
"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.": "이 워크스페이스 멤버를 삭제하시겠습니까? 이 작업은 되돌릴 수 없습니다.",
|
||||||
|
"Deactivate member": "멤버 비활성화",
|
||||||
|
"Activate member": "멤버 활성화",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "이 워크스페이스 멤버를 비활성화하시겠습니까? 해당 사용자는 더 이상 이 워크스페이스에 접근할 수 없습니다.",
|
||||||
|
"Are you sure you want to activate this workspace member?": "이 워크스페이스 멤버를 활성화하시겠습니까?",
|
||||||
|
"Deactivate": "비활성화",
|
||||||
|
"Activate": "활성화",
|
||||||
|
"Deactivated": "비활성화됨",
|
||||||
"Move": "이동",
|
"Move": "이동",
|
||||||
"Move page": "페이지 이동",
|
"Move page": "페이지 이동",
|
||||||
"Move page to a different space.": "페이지를 다른 공간으로 이동합니다.",
|
"Move page to a different space.": "페이지를 다른 공간으로 이동합니다.",
|
||||||
@@ -405,6 +445,25 @@
|
|||||||
"Share deleted successfully": "공유가 성공적으로 삭제되었습니다",
|
"Share deleted successfully": "공유가 성공적으로 삭제되었습니다",
|
||||||
"Share not found": "공유를 찾을 수 없습니다",
|
"Share not found": "공유를 찾을 수 없습니다",
|
||||||
"Failed to share page": "페이지 공유에 실패했습니다",
|
"Failed to share page": "페이지 공유에 실패했습니다",
|
||||||
|
"Disable public sharing": "공유 비활성화",
|
||||||
|
"Prevent members from sharing pages publicly.": "멤버들이 페이지를 공개적으로 공유하지 못하도록 방지하십시오.",
|
||||||
|
"Toggle public sharing": "공유 전환",
|
||||||
|
"Toggle space public sharing": "공간 공유 전환",
|
||||||
|
"Allow viewers to comment": "뷰어가 댓글을 달 수 있도록 허용",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "이 공간 내 페이지에 뷰어가 댓글을 추가할 수 있도록 허용합니다.",
|
||||||
|
"Toggle viewer comments": "뷰어 댓글 전환",
|
||||||
|
"Public sharing is disabled at the workspace level": "워크스페이스 수준에서 공유가 비활성화되었습니다.",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "이 공간의 페이지가 공개적으로 공유되지 않도록 방지하십시오.",
|
||||||
|
"Page permissions": "페이지 권한},{",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "개별 페이지의 조회 및 편집 권한을 제어합니다. 엔터프라이즈 라이선스에서 이용 가능합니다.",
|
||||||
|
"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 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 disable public sharing? All existing shared links in this space will be deleted.": "정말로 공유를 비활성화하시겠습니까? 이 공간의 모든 기존 공유 링크가 삭제됩니다.",
|
||||||
|
"Public sharing is disabled": "공유가 비활성화되었습니다.",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "워크스페이스 수준에서 공유가 비활성화되었습니다.",
|
||||||
|
"Public sharing has been disabled for this space.": "이 공간의 공유가 비활성화되었습니다.",
|
||||||
"Copy page": "페이지 복사하기",
|
"Copy page": "페이지 복사하기",
|
||||||
"Copy page to a different space.": "다른 공간으로 페이지 복사하기.",
|
"Copy page to a different space.": "다른 공간으로 페이지 복사하기.",
|
||||||
"Page copied successfully": "페이지가 성공적으로 복사되었습니다",
|
"Page copied successfully": "페이지가 성공적으로 복사되었습니다",
|
||||||
@@ -487,7 +546,7 @@
|
|||||||
"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.": "백업 코드 중 하나를 입력하세요. 각 백업 코드는 한 번만 사용할 수 있습니다.",
|
||||||
"Verify": "확인",
|
"Verify": "확인",
|
||||||
"Trash": "휴지통",
|
"Trash": "휴지통",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "휴지통의 페이지는 30일 후에 영구적으로 삭제됩니다.",
|
"Pages in trash will be permanently deleted after {{count}} days.": "휴지통의 페이지는 {{count}}일 후 영구적으로 삭제됩니다.",
|
||||||
"Deleted": "삭제됨",
|
"Deleted": "삭제됨",
|
||||||
"No pages in trash": "휴지통에 페이지가 없습니다",
|
"No pages in trash": "휴지통에 페이지가 없습니다",
|
||||||
"Permanently delete page?": "페이지를 영구적으로 삭제하시겠습니까?",
|
"Permanently delete page?": "페이지를 영구적으로 삭제하시겠습니까?",
|
||||||
@@ -559,19 +618,138 @@
|
|||||||
"This action cannot be undone. Any applications using this API key will stop working.": "이 작업은 되돌릴 수 없습니다. 이 API 키를 사용하는 모든 응용 프로그램이 작동을 멈출 것입니다.",
|
"This action cannot be undone. Any applications using this API key will stop working.": "이 작업은 되돌릴 수 없습니다. 이 API 키를 사용하는 모든 응용 프로그램이 작동을 멈출 것입니다.",
|
||||||
"Update API key": "API 키 갱신",
|
"Update API key": "API 키 갱신",
|
||||||
"Manage API keys for all users in the workspace": "워크스페이스 내 모든 사용자의 API 키 관리",
|
"Manage API keys for all users in the workspace": "워크스페이스 내 모든 사용자의 API 키 관리",
|
||||||
|
"Restrict API key creation to admins": "API 키 생성 권한을 관리자에게만 제한합니다",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "새로운 API 키는 관리자와 소유자만 생성할 수 있습니다. 기존 멤버 키는 계속 사용할 수 있습니다.",
|
||||||
|
"Toggle restrict API keys to admins": "API 키 생성 제한(관리자 전용) 설정 전환",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "API 키 생성이 워크스페이스 관리자로 인해 관리자에게만 제한되었습니다.",
|
||||||
"AI settings": "AI 설정",
|
"AI settings": "AI 설정",
|
||||||
"AI search": "AI 검색",
|
"AI search": "AI 검색",
|
||||||
"AI Answer": "AI 답변",
|
"AI Answer": "AI 답변",
|
||||||
"Ask AI": "AI에게 묻기",
|
"Ask AI": "AI에게 묻기",
|
||||||
"AI is thinking...": "AI가 생각 중입니다...",
|
"AI is thinking...": "AI가 생각 중입니다...",
|
||||||
"Ask a question...": "질문하세요...",
|
"Ask a question...": "질문하세요...",
|
||||||
"AI-powered search (Ask AI)": "AI 지원 검색 (AI에게 묻기)",
|
"AI Answers": "AI 답변",
|
||||||
|
"AI-powered search (AI Answers)": "AI 구동 검색 (AI 답변)",
|
||||||
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "AI 검색은 벡터 임베딩을 사용하여 작업공간 콘텐츠에 대한 의미 검색 기능을 제공합니다.",
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "AI 검색은 벡터 임베딩을 사용하여 작업공간 콘텐츠에 대한 의미 검색 기능을 제공합니다.",
|
||||||
"Toggle AI search": "AI 검색 전환",
|
"Toggle AI search": "AI 검색 전환",
|
||||||
|
"Generative AI (Ask AI)": "생성 AI (Ask AI)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "편집기에서 AI 구동 콘텐츠 생성을 활성화합니다. 사용자가 텍스트를 생성, 개선, 번역 및 변환할 수 있습니다.",
|
||||||
|
"Toggle generative AI": "생성 AI 토글",
|
||||||
|
"Upgrade your plan": "요금제를 업그레이드하세요",
|
||||||
|
"Available with a paid license": "유료 라이선스에서만 사용 가능합니다",
|
||||||
|
"Upgrade your license tier.": "라이선스 등급을 업그레이드하세요.",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI는 Docmost 엔터프라이즈 에디션에서만 제공됩니다. sales@docmost.com으로 문의하세요.",
|
||||||
|
"AI & MCP": "AI 및 MCP",
|
||||||
|
"AI": "AI",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "모델 컨텍스트 프로토콜(MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "AI 어시스턴트와 도구가 워크스페이스 콘텐츠와 상호작용할 수 있도록 MCP 서버를 활성화하세요.",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP는 Docmost 엔터프라이즈 에디션에서만 제공됩니다. sales@docmost.com으로 문의하세요.",
|
||||||
|
"MCP Server URL": "MCP 서버 URL",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "인증을 위해 API 키를 사용하세요. API 키는 계정 설정에서 관리할 수 있습니다.",
|
||||||
|
"Supported tools": "지원되는 도구",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "워크스페이스에 MCP가 활성화되어 있습니다. AI 어시스턴트를 연결하려면 API 키를 사용하세요.",
|
||||||
|
"MCP server URL:": "MCP 서버 URL:",
|
||||||
|
"Learn more": "자세히 알아보기",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "워크스페이스의 모든 사용자를 위한 API 키를 관리하세요. 사용 방법은 <anchor>API 문서</anchor>를 참고하세요.",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "사용 방법은 <anchor>API 문서</anchor>를 참고하세요.",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "<anchor>MCP 문서</anchor>를 확인하세요.",
|
||||||
"Sources": "출처",
|
"Sources": "출처",
|
||||||
"Ask AI not available for attachments": "AI에게 묻기 기능은 첨부 파일에 대해 사용할 수 없습니다",
|
"AI Answers not available for attachments": "첨부 파일에 대해 AI 답변을 사용할 수 없습니다",
|
||||||
"No answer available": "답변을 제공할 수 없습니다",
|
"No answer available": "답변을 제공할 수 없습니다",
|
||||||
"Background color": "배경 색",
|
"Background color": "배경 색",
|
||||||
"Highlight color": "강조 색",
|
"Highlight color": "강조 색",
|
||||||
"Remove color": "색 제거"
|
"Remove color": "색 제거",
|
||||||
|
"Notifications": "알림",
|
||||||
|
"No notifications": "알림 없음",
|
||||||
|
"No unread notifications": "읽지 않은 알림 없음",
|
||||||
|
"All notifications": "모든 알림",
|
||||||
|
"Unread only": "읽지 않음만",
|
||||||
|
"Mark all as read": "모두 읽음으로 표시",
|
||||||
|
"Mark as read": "읽음으로 표시",
|
||||||
|
"More options": "추가 옵션",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold>님이 댓글에서 당신을 언급했습니다",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold>님이 페이지에 댓글을 남겼습니다",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold>님이 댓글을 해결했습니다",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>님이 페이지에서 당신을 언급했습니다",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>님이 페이지 편집 권한을 부여했습니다",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>님이 페이지 조회 권한을 부여했습니다",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold>님이 페이지를 업데이트했습니다.",
|
||||||
|
"Watch page": "페이지 구독",
|
||||||
|
"Stop watching": "구독 취소",
|
||||||
|
"Email notifications": "이메일 알림",
|
||||||
|
"Page updates": "페이지 업데이트",
|
||||||
|
"Get notified when pages you watch are updated.": "구독한 페이지가 업데이트될 때 알림을 받으세요.",
|
||||||
|
"Page mentions": "페이지 언급",
|
||||||
|
"Get notified when someone mentions you on a page.": "누군가가 페이지에서 당신을 언급하면 알림을 받으세요.",
|
||||||
|
"Comment mentions": "댓글 언급",
|
||||||
|
"Get notified when someone mentions you in a comment.": "누군가가 댓글에서 당신을 언급하면 알림을 받으세요.",
|
||||||
|
"New comments": "새 댓글",
|
||||||
|
"Get notified about new comments on threads you participate in.": "참여 중인 스레드에 새 댓글이 달리면 알림을 받으세요.",
|
||||||
|
"Resolved comments": "해결된 댓글",
|
||||||
|
"Get notified when your comment is resolved.": "내 댓글이 해결되었을 때 알림을 받으세요.",
|
||||||
|
"You are now watching this page": "이제 이 페이지를 주시합니다.",
|
||||||
|
"You are no longer watching this page": "더 이상 이 페이지를 주시하지 않습니다.",
|
||||||
|
"Direct": "직접",
|
||||||
|
"Updates": "업데이트",
|
||||||
|
"Today": "오늘",
|
||||||
|
"Yesterday": "어제",
|
||||||
|
"This week": "이번 주",
|
||||||
|
"Older": "이전",
|
||||||
|
"Restricted page": "제한된 페이지",
|
||||||
|
"Restricted pages cannot be shared publicly.": "제한된 페이지는 공개적으로 공유할 수 없습니다.",
|
||||||
|
"Restricted by parent": "상위 페이지에 의해 제한됨",
|
||||||
|
"Restricted": "제한됨",
|
||||||
|
"Open": "공개",
|
||||||
|
"Inherits restrictions from ancestor page": "상위 페이지로부터 제한을 상속함",
|
||||||
|
"Only people listed below can access this page": "아래에 나열된 사용자만 이 페이지에 접근할 수 있습니다.",
|
||||||
|
"Everyone in this space can access": "이 공간의 모든 사용자가 접근할 수 있습니다.",
|
||||||
|
"No additional restrictions on this page": "이 페이지에는 추가 제한이 없습니다.",
|
||||||
|
"Only specific people can access": "특정 사용자만 접근할 수 있습니다.",
|
||||||
|
"Use only inherited restrictions": "상속된 제한만 사용",
|
||||||
|
"Add restrictions on top of inherited": "상속된 제한 위에 추가 제한 적용",
|
||||||
|
"Inherited restriction": "상속된 제한",
|
||||||
|
"Access limited by": "접근 제한:",
|
||||||
|
"Restrict access to control who can view and edit this page": "이 페이지를 누가 조회하고 편집할 수 있는지 제어하려면 접근을 제한하세요.",
|
||||||
|
"Add additional restrictions specific to this page": "이 페이지에 대한 추가 제한을 적용하세요.",
|
||||||
|
"Access": "접근",
|
||||||
|
"People with access": "접근 권한이 있는 사용자",
|
||||||
|
"Remove all": "모두 제거",
|
||||||
|
"Remove access": "접근 권한 제거",
|
||||||
|
"Remove all access": "모든 접근 권한 제거",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "이 멤버의 페이지 접근 권한을 제거하시겠습니까?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "모든 특정 접근 권한을 제거하시겠습니까? 이렇게 하면 페이지가 공간의 모든 사용자에게 공개됩니다.",
|
||||||
|
"Trash retention": "휴지통 보관 기간",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "이 기간이 지나면 휴지통의 페이지는 영구적으로 삭제됩니다.",
|
||||||
|
"Trash retention updated": "휴지통 보관 기간이 업데이트되었습니다.",
|
||||||
|
"Failed to update trash retention": "휴지통 보관 기간 업데이트에 실패했습니다.",
|
||||||
|
"Removed page restriction": "페이지 제한이 제거됨",
|
||||||
|
"Added page permission": "페이지 권한이 추가됨",
|
||||||
|
"Removed page permission": "페이지 권한이 제거됨",
|
||||||
|
"Verifying your email": "이메일 인증 중",
|
||||||
|
"Please wait...": "잠시만 기다려 주세요...",
|
||||||
|
"Verification failed. The link may have expired.": "인증에 실패했습니다. 링크가 만료되었을 수 있습니다.",
|
||||||
|
"Check your email": "이메일을 확인하세요",
|
||||||
|
"We sent a verification link to {{email}}.": "{{email}} 주소로 인증 링크를 보냈습니다.",
|
||||||
|
"We sent a verification link to your email.": "이메일로 인증 링크를 보냈습니다.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "이메일의 링크를 클릭하여 인증하고 워크스페이스에 접속하세요.",
|
||||||
|
"Resend verification email": "인증 이메일 재전송",
|
||||||
|
"Verification email sent. Please check your inbox.": "인증 이메일이 전송되었습니다. 받은 편지함을 확인하세요.",
|
||||||
|
"Failed to resend verification email. Please try again.": "인증 이메일 재전송에 실패했습니다. 다시 시도해 주세요.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "연결된 워크스페이스 목록이 포함된 이메일을 보내드렸습니다.",
|
||||||
|
"Load more": "더 불러오기",
|
||||||
|
"Log out of all devices": "모든 기기에서 로그아웃",
|
||||||
|
"Log out of all sessions except this device": "이 기기를 제외한 모든 세션에서 로그아웃",
|
||||||
|
"This Device": "이 기기",
|
||||||
|
"Unknown device": "알 수 없는 기기",
|
||||||
|
"No active sessions": "활성 세션이 없습니다",
|
||||||
|
"Session revoked": "세션이 해제되었습니다",
|
||||||
|
"All other sessions revoked": "나머지 모든 세션이 해제되었습니다",
|
||||||
|
"Last used": "최근 사용",
|
||||||
|
"Created": "생성됨",
|
||||||
|
"Rename": "이름 바꾸기",
|
||||||
|
"Publish": "게시",
|
||||||
|
"Security": "보안",
|
||||||
|
"Enforce SSO": "SSO 강제 적용",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "강제 적용 시, 멤버는 이메일과 비밀번호로는 로그인할 수 없습니다."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -116,6 +116,7 @@
|
|||||||
"No group found": "Geen groep gevonden",
|
"No group found": "Geen groep gevonden",
|
||||||
"No page history saved yet.": "Er is nog geen pagina geschiedenis opgeslagen.",
|
"No page history saved yet.": "Er is nog geen pagina geschiedenis opgeslagen.",
|
||||||
"No pages yet": "Nog geen pagina's",
|
"No pages yet": "Nog geen pagina's",
|
||||||
|
"No shared pages": "Geen gedeelde pagina's",
|
||||||
"No results found...": "Geen resultaten gevonden...",
|
"No results found...": "Geen resultaten gevonden...",
|
||||||
"No user found": "Geen gebruiker gevonden",
|
"No user found": "Geen gebruiker gevonden",
|
||||||
"Overview": "Overzicht",
|
"Overview": "Overzicht",
|
||||||
@@ -123,11 +124,14 @@
|
|||||||
"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",
|
||||||
"Password": "Wachtwoord",
|
"Password": "Wachtwoord",
|
||||||
"Password changed successfully": "Wachtwoord met succes gewijzigd",
|
"Password changed successfully": "Wachtwoord met succes gewijzigd",
|
||||||
|
"People": "Personen",
|
||||||
"Pending": "Wachtende",
|
"Pending": "Wachtende",
|
||||||
"Please confirm your action": "Bevestig alstublieft uw actie",
|
"Please confirm your action": "Bevestig alstublieft uw actie",
|
||||||
"Preferences": "Voorkeuren",
|
"Preferences": "Voorkeuren",
|
||||||
@@ -205,6 +209,9 @@
|
|||||||
"Reply...": "Antwoord...",
|
"Reply...": "Antwoord...",
|
||||||
"Error loading comments.": "Fout bij het laden van reacties.",
|
"Error loading comments.": "Fout bij het laden van reacties.",
|
||||||
"No comments yet.": "Nog geen reacties.",
|
"No comments yet.": "Nog geen reacties.",
|
||||||
|
"No open comments.": "Geen openstaande opmerkingen.",
|
||||||
|
"No resolved comments.": "Geen opgeloste reacties.",
|
||||||
|
"Add a comment...": "Voeg een opmerking toe...",
|
||||||
"Edit comment": "Bewerk reactie",
|
"Edit comment": "Bewerk reactie",
|
||||||
"Delete comment": "Verwijder reactie",
|
"Delete comment": "Verwijder reactie",
|
||||||
"Are you sure you want to delete this comment?": "Weet je zeker dat je deze reactie wilt verwijderen?",
|
"Are you sure you want to delete this comment?": "Weet je zeker dat je deze reactie wilt verwijderen?",
|
||||||
@@ -226,7 +233,6 @@
|
|||||||
"Are you sure you want to unresolve this comment thread?": "Weet u zeker dat u deze reactiedraad niet wilt oplossen?",
|
"Are you sure you want to unresolve this comment thread?": "Weet u zeker dat u deze reactiedraad niet wilt oplossen?",
|
||||||
"Resolved": "Opgelost",
|
"Resolved": "Opgelost",
|
||||||
"No active comments.": "Geen actieve reacties.",
|
"No active comments.": "Geen actieve reacties.",
|
||||||
"No resolved comments.": "Geen opgeloste reacties.",
|
|
||||||
"Revoke invitation": "Uitnodiging intrekken",
|
"Revoke invitation": "Uitnodiging intrekken",
|
||||||
"Revoke": "Intrekken",
|
"Revoke": "Intrekken",
|
||||||
"Don't": "Niet doen",
|
"Don't": "Niet doen",
|
||||||
@@ -272,6 +278,7 @@
|
|||||||
"Add row below": "Rij hieronder toevoegen",
|
"Add row below": "Rij hieronder toevoegen",
|
||||||
"Delete table": "Verwijder tabel",
|
"Delete table": "Verwijder tabel",
|
||||||
"Info": "Info",
|
"Info": "Info",
|
||||||
|
"Note": "Opmerking",
|
||||||
"Success": "Geslaagd",
|
"Success": "Geslaagd",
|
||||||
"Warning": "Waarschuwing",
|
"Warning": "Waarschuwing",
|
||||||
"Danger": "Gevaar",
|
"Danger": "Gevaar",
|
||||||
@@ -282,6 +289,11 @@
|
|||||||
"Save & Exit": "Opslaan & Afsluiten",
|
"Save & Exit": "Opslaan & Afsluiten",
|
||||||
"Double-click to edit Excalidraw diagram": "Dubbelklik om Excalidraw diagram te bewerken",
|
"Double-click to edit Excalidraw diagram": "Dubbelklik om Excalidraw diagram te bewerken",
|
||||||
"Paste link": "Link plakken",
|
"Paste link": "Link plakken",
|
||||||
|
"Paste link or search pages": "Plak link of zoek pagina's",
|
||||||
|
"Link to web page": "Link naar webpagina",
|
||||||
|
"Recents": "Recent",
|
||||||
|
"Page or URL": "Pagina of URL",
|
||||||
|
"Link title": "Kop van de link",
|
||||||
"Edit link": "Link bewerken",
|
"Edit link": "Link bewerken",
|
||||||
"Remove link": "Link verwijderen",
|
"Remove link": "Link verwijderen",
|
||||||
"Add link": "Link toevoegen",
|
"Add link": "Link toevoegen",
|
||||||
@@ -329,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Horizontale lijn invoegen",
|
"Insert horizontal rule divider": "Horizontale lijn invoegen",
|
||||||
"Upload any image from your device.": "Upload een afbeelding vanaf uw apparaat.",
|
"Upload any image from your device.": "Upload een afbeelding vanaf uw apparaat.",
|
||||||
"Upload any video from your device.": "Upload een video vanaf uw apparaat.",
|
"Upload any video from your device.": "Upload een video vanaf uw apparaat.",
|
||||||
|
"Upload any audio from your device.": "Upload een audio vanaf uw apparaat.",
|
||||||
"Upload any file from your device.": "Upload een bestand vanaf uw apparaat.",
|
"Upload any file from your device.": "Upload een bestand vanaf uw apparaat.",
|
||||||
"Uploading {{name}}": "Uploaden {{name}}",
|
"Uploading {{name}}": "Uploaden {{name}}",
|
||||||
"Uploading file": "Bestand uploaden",
|
"Uploading file": "Bestand uploaden",
|
||||||
@@ -339,6 +352,12 @@
|
|||||||
"Divider": "Scheidingslijn",
|
"Divider": "Scheidingslijn",
|
||||||
"Quote": "Quote",
|
"Quote": "Quote",
|
||||||
"Image": "Afbeelding",
|
"Image": "Afbeelding",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "PDF insluiten",
|
||||||
|
"Upload and embed a PDF file.": "Upload en sluit een PDF-bestand in.",
|
||||||
|
"Embed as PDF": "Insluiten als PDF",
|
||||||
|
"Failed to load PDF": "Laden van PDF mislukt",
|
||||||
|
"Convert to attachment": "Converteren naar bijlage",
|
||||||
"File attachment": "Bestand bijlage",
|
"File attachment": "Bestand bijlage",
|
||||||
"Toggle block": "Schakel blok in/uit",
|
"Toggle block": "Schakel blok in/uit",
|
||||||
"Callout": "Opmerking",
|
"Callout": "Opmerking",
|
||||||
@@ -353,9 +372,23 @@
|
|||||||
"Insert current date": "Huidige datum invoeren",
|
"Insert current date": "Huidige datum invoeren",
|
||||||
"Draw and sketch excalidraw diagrams": "Teken en schets excalidraw diagrammen",
|
"Draw and sketch excalidraw diagrams": "Teken en schets excalidraw diagrammen",
|
||||||
"Multiple": "Meerdere",
|
"Multiple": "Meerdere",
|
||||||
|
"Turn into": "Omzetten naar",
|
||||||
|
"Text align": "Tekstuitlijning",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "Deze pagina is mogelijk verwijderd of verplaatst, of u heeft er geen toegang toe.",
|
||||||
|
"Go to homepage": "Ga naar de startpagina",
|
||||||
|
"Pages you create will show up here.": "Pagina's die u aanmaakt, verschijnen hier.",
|
||||||
"Heading {{level}}": "Kop {{level}}",
|
"Heading {{level}}": "Kop {{level}}",
|
||||||
"Toggle title": "Schakel titel in/uit",
|
"Toggle title": "Schakel titel in/uit",
|
||||||
"Write anything. Enter \"/\" for commands": "Schrijf iets. Voer \"/\" in voor commando's",
|
"Write anything. Enter \"/\" for commands": "Schrijf iets. Voer \"/\" in voor commando's",
|
||||||
|
"Write...": "Typ...",
|
||||||
|
"Column count": "Aantal kolommen",
|
||||||
|
"{{count}} Columns": "{{count}} kolommen",
|
||||||
|
"Equal columns": "Gelijke kolommen",
|
||||||
|
"Left sidebar": "Linker zijbalk",
|
||||||
|
"Right sidebar": "Rechter zijbalk",
|
||||||
|
"Wide center": "Brede middenkolom",
|
||||||
|
"Left wide": "Brede linkerkolom",
|
||||||
|
"Right wide": "Brede rechterkolom",
|
||||||
"Names do not match": "Namen komen niet overeen",
|
"Names do not match": "Namen komen niet overeen",
|
||||||
"Today, {{time}}": "Vandaag, {{time}}",
|
"Today, {{time}}": "Vandaag, {{time}}",
|
||||||
"Yesterday, {{time}}": "Gisteren, {{time}}",
|
"Yesterday, {{time}}": "Gisteren, {{time}}",
|
||||||
@@ -378,6 +411,13 @@
|
|||||||
"Delete member": "Verwijder lid",
|
"Delete member": "Verwijder lid",
|
||||||
"Member deleted successfully": "Lid succesvol verwijderd",
|
"Member deleted successfully": "Lid succesvol verwijderd",
|
||||||
"Are you sure you want to delete this workspace member? This action is irreversible.": "Weet u zeker dat u dit lid van de werkruimte wilt verwijderen? Deze actie kan niet ongedaan gemaakt worden.",
|
"Are you sure you want to delete this workspace member? This action is irreversible.": "Weet u zeker dat u dit lid van de werkruimte wilt verwijderen? Deze actie kan niet ongedaan gemaakt worden.",
|
||||||
|
"Deactivate member": "Lid deactiveren",
|
||||||
|
"Activate member": "Lid activeren",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "Weet u zeker dat u dit lid van de werkruimte wilt deactiveren? Deze persoon heeft daarna geen toegang meer tot deze werkruimte.",
|
||||||
|
"Are you sure you want to activate this workspace member?": "Weet u zeker dat u dit lid van de werkruimte wilt activeren?",
|
||||||
|
"Deactivate": "Deactiveren",
|
||||||
|
"Activate": "Activeren",
|
||||||
|
"Deactivated": "Gedeactiveerd",
|
||||||
"Move": "Verplaatsen",
|
"Move": "Verplaatsen",
|
||||||
"Move page": "Pagina verplaatsen",
|
"Move page": "Pagina verplaatsen",
|
||||||
"Move page to a different space.": "Verplaats pagina naar een andere ruimte.",
|
"Move page to a different space.": "Verplaats pagina naar een andere ruimte.",
|
||||||
@@ -405,6 +445,25 @@
|
|||||||
"Share deleted successfully": "Delen succesvol verwijderd",
|
"Share deleted successfully": "Delen succesvol verwijderd",
|
||||||
"Share not found": "Delen niet gevonden",
|
"Share not found": "Delen niet gevonden",
|
||||||
"Failed to share page": "Pagina delen mislukt",
|
"Failed to share page": "Pagina delen mislukt",
|
||||||
|
"Disable public sharing": "Openbaar delen uitschakelen",
|
||||||
|
"Prevent members from sharing pages publicly.": "Voorkom dat leden pagina's openbaar delen.",
|
||||||
|
"Toggle public sharing": "Wissel openbaar delen",
|
||||||
|
"Toggle space public sharing": "Wissel openbaar delen van ruimte",
|
||||||
|
"Allow viewers to comment": "Toestaan dat kijkers reageren",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Sta kijkers toe om reacties toe te voegen op pagina\u0019s in deze ruimte.",
|
||||||
|
"Toggle viewer comments": "Reacties van kijkers in- of uitschakelen",
|
||||||
|
"Public sharing is disabled at the workspace level": "Openbaar delen is uitgeschakeld op werkruimteniveau",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "Voorkom dat pagina's in deze ruimte openbaar worden gedeeld.",
|
||||||
|
"Page permissions": "Pagina rechten",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Beheer wie individuele pagina's kan bekijken en bewerken. Beschikbaar met een Enterprise-licentie.",
|
||||||
|
"Enable public sharing": "Openbaar delen inschakelen",
|
||||||
|
"Are you sure you want to enable public sharing? Members will be able to share pages publicly.": "Weet je zeker dat je openbaar delen wilt inschakelen? Leden kunnen pagina's openbaar delen.",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this workspace will be deleted.": "Weet je zeker dat je openbaar delen wilt uitschakelen? Alle bestaande gedeelde links in deze werkruimte zullen worden verwijderd.",
|
||||||
|
"Are you sure you want to enable public sharing for this space?": "Weet je zeker dat je openbaar delen voor deze ruimte wilt inschakelen?",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this space will be deleted.": "Weet je zeker dat je openbaar delen wilt uitschakelen? Alle bestaande gedeelde links in deze ruimte zullen worden verwijderd.",
|
||||||
|
"Public sharing is disabled": "Openbaar delen is uitgeschakeld",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "Openbaar delen is uitgeschakeld op werkruimteniveau.",
|
||||||
|
"Public sharing has been disabled for this space.": "Openbaar delen is uitgeschakeld voor deze ruimte.",
|
||||||
"Copy page": "Pagina kopiëren",
|
"Copy page": "Pagina kopiëren",
|
||||||
"Copy page to a different space.": "Kopieer pagina naar een andere ruimte.",
|
"Copy page to a different space.": "Kopieer pagina naar een andere ruimte.",
|
||||||
"Page copied successfully": "Pagina succesvol gekopieerd",
|
"Page copied successfully": "Pagina succesvol gekopieerd",
|
||||||
@@ -487,7 +546,7 @@
|
|||||||
"Enter one of your backup codes. Each backup code can only be used once.": "Voer een van uw back-up codes in. Elke back-up code kan slechts één keer worden gebruikt.",
|
"Enter one of your backup codes. Each backup code can only be used once.": "Voer een van uw back-up codes in. Elke back-up code kan slechts één keer worden gebruikt.",
|
||||||
"Verify": "Verifiëren",
|
"Verify": "Verifiëren",
|
||||||
"Trash": "Prullenbak",
|
"Trash": "Prullenbak",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "Pagina's in de prullenbak worden na 30 dagen permanent verwijderd.",
|
"Pages in trash will be permanently deleted after {{count}} days.": "Pagina's in de prullenbak worden na {{count}} dagen permanent verwijderd.",
|
||||||
"Deleted": "Verwijderd",
|
"Deleted": "Verwijderd",
|
||||||
"No pages in trash": "Geen pagina's in de prullenbak",
|
"No pages in trash": "Geen pagina's in de prullenbak",
|
||||||
"Permanently delete page?": "Pagina permanent verwijderen?",
|
"Permanently delete page?": "Pagina permanent verwijderen?",
|
||||||
@@ -559,19 +618,138 @@
|
|||||||
"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.",
|
"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",
|
"Update API key": "API-sleutel bijwerken",
|
||||||
"Manage API keys for all users in the workspace": "Beheer API-sleutels voor alle gebruikers in de werkruimte",
|
"Manage API keys for all users in the workspace": "Beheer API-sleutels voor alle gebruikers in de werkruimte",
|
||||||
|
"Restrict API key creation to admins": "Beperk het aanmaken van API-sleutels tot beheerders.",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "Alleen beheerders en eigenaren kunnen nieuwe API-sleutels aanmaken. Bestaande leden-sleutels blijven werken.",
|
||||||
|
"Toggle restrict API keys to admins": "Schakel het beperken van API-sleutels tot beheerders in/uit",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "Het aanmaken van API-sleutels is door je werkruimtebeheerder beperkt tot beheerders.",
|
||||||
"AI settings": "AI-instellingen",
|
"AI settings": "AI-instellingen",
|
||||||
"AI search": "AI-zoekopdracht",
|
"AI search": "AI-zoekopdracht",
|
||||||
"AI Answer": "AI Antwoord",
|
"AI Answer": "AI Antwoord",
|
||||||
"Ask AI": "Vraag AI",
|
"Ask AI": "Vraag AI",
|
||||||
"AI is thinking...": "AI is aan het nadenken...",
|
"AI is thinking...": "AI is aan het nadenken...",
|
||||||
"Ask a question...": "Stel een vraag...",
|
"Ask a question...": "Stel een vraag...",
|
||||||
"AI-powered search (Ask AI)": "AI-ondersteunde zoekopdracht (Vraag AI)",
|
"AI Answers": "AI Antwoorden",
|
||||||
|
"AI-powered search (AI Answers)": "AI-gestuurde zoekopdracht (AI Antwoorden)",
|
||||||
"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.",
|
"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",
|
"Toggle AI search": "Schakel AI-zoekopdracht in/uit",
|
||||||
|
"Generative AI (Ask AI)": "Generatieve AI (Vraag het AI)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Schakel AI-gestuurde inhoudsgeneratie in de editor in. Hiermee kunnen gebruikers tekst genereren, verbeteren, vertalen en transformeren.",
|
||||||
|
"Toggle generative AI": "Generatieve AI schakelen",
|
||||||
|
"Upgrade your plan": "Upgrade je abonnement",
|
||||||
|
"Available with a paid license": "Beschikbaar met een betaalde licentie",
|
||||||
|
"Upgrade your license tier.": "Upgrade je licentieniveau.",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI is alleen beschikbaar in de Docmost Enterprise-editie. Neem contact op met sales@docmost.com.",
|
||||||
|
"AI & MCP": "AI & MCP",
|
||||||
|
"AI": "AI",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "Model Context Protocol (MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Schakel de MCP-server in zodat AI-assistenten en tools kunnen interageren met de inhoud van uw werkruimte.",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP is alleen beschikbaar in de Docmost Enterprise-editie. Neem contact op met sales@docmost.com.",
|
||||||
|
"MCP Server URL": "MCP-server-URL",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Gebruik uw API-sleutel voor authenticatie. U kunt API-sleutels beheren in uw accountinstellingen.",
|
||||||
|
"Supported tools": "Ondersteunde tools",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "In uw werkruimte is MCP ingeschakeld. Gebruik uw API-sleutel om AI-assistenten te koppelen.",
|
||||||
|
"MCP server URL:": "MCP-server-URL:",
|
||||||
|
"Learn more": "Meer informatie",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Beheer API-sleutels voor alle gebruikers in de werkruimte. Bekijk de <anchor>API-documentatie</anchor> voor gebruiksdetails.",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "Bekijk de <anchor>API-documentatie</anchor> voor gebruiksdetails.",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "Bekijk de <anchor>MCP-documentatie</anchor>.",
|
||||||
"Sources": "Bronnen",
|
"Sources": "Bronnen",
|
||||||
"Ask AI not available for attachments": "Vraag AI is niet beschikbaar voor bijlages",
|
"AI Answers not available for attachments": "AI Antwoorden niet beschikbaar voor bijlagen",
|
||||||
"No answer available": "Geen antwoord beschikbaar",
|
"No answer available": "Geen antwoord beschikbaar",
|
||||||
"Background color": "Achtergrondkleur",
|
"Background color": "Achtergrondkleur",
|
||||||
"Highlight color": "Markeerkleur",
|
"Highlight color": "Markeerkleur",
|
||||||
"Remove color": "Kleur verwijderen"
|
"Remove color": "Kleur verwijderen",
|
||||||
|
"Notifications": "Meldingen",
|
||||||
|
"No notifications": "Geen meldingen",
|
||||||
|
"No unread notifications": "Geen ongelezen meldingen",
|
||||||
|
"All notifications": "Alle meldingen",
|
||||||
|
"Unread only": "Alleen ongelezen",
|
||||||
|
"Mark all as read": "Markeer alles als gelezen",
|
||||||
|
"Mark as read": "Markeer als gelezen",
|
||||||
|
"More options": "Meer opties",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> noemde je in een reactie",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> heeft een reactie geplaatst op een pagina",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> heeft een reactie opgelost",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> noemde je op een pagina",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> heeft je toegang gegeven om een pagina te bewerken",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> heeft je toegang gegeven om een pagina te bekijken",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> heeft een pagina bijgewerkt.",
|
||||||
|
"Watch page": "Pagina volgen",
|
||||||
|
"Stop watching": "Volgen stoppen",
|
||||||
|
"Email notifications": "E-mailmeldingen",
|
||||||
|
"Page updates": "Pagina-updates",
|
||||||
|
"Get notified when pages you watch are updated.": "Ontvang een melding wanneer pagina's die je volgt worden bijgewerkt.",
|
||||||
|
"Page mentions": "Pagina-vermeldingen",
|
||||||
|
"Get notified when someone mentions you on a page.": "Ontvang een melding wanneer iemand je noemt op een pagina.",
|
||||||
|
"Comment mentions": "Vermeldingen in opmerkingen",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Ontvang een melding wanneer iemand je noemt in een opmerking.",
|
||||||
|
"New comments": "Nieuwe opmerkingen",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Ontvang meldingen over nieuwe reacties in threads waaraan je deelneemt.",
|
||||||
|
"Resolved comments": "Opgeloste opmerkingen",
|
||||||
|
"Get notified when your comment is resolved.": "Ontvang een melding wanneer je reactie is opgelost.",
|
||||||
|
"You are now watching this page": "Je volgt nu deze pagina",
|
||||||
|
"You are no longer watching this page": "Je volgt deze pagina niet meer",
|
||||||
|
"Direct": "Direct",
|
||||||
|
"Updates": "Updates",
|
||||||
|
"Today": "Vandaag",
|
||||||
|
"Yesterday": "Gisteren",
|
||||||
|
"This week": "Deze week",
|
||||||
|
"Older": "Ouder",
|
||||||
|
"Restricted page": "Beperkte pagina",
|
||||||
|
"Restricted pages cannot be shared publicly.": "Beperkte pagina's kunnen niet openbaar worden gedeeld.",
|
||||||
|
"Restricted by parent": "Beperkt door bovenliggende",
|
||||||
|
"Restricted": "Beperkt",
|
||||||
|
"Open": "Open",
|
||||||
|
"Inherits restrictions from ancestor page": "Erft restricties van de bovenliggende pagina",
|
||||||
|
"Only people listed below can access this page": "Alleen onderstaande personen hebben toegang tot deze pagina",
|
||||||
|
"Everyone in this space can access": "Iedereen in deze ruimte heeft toegang",
|
||||||
|
"No additional restrictions on this page": "Geen aanvullende restricties op deze pagina",
|
||||||
|
"Only specific people can access": "Alleen specifieke personen hebben toegang",
|
||||||
|
"Use only inherited restrictions": "Gebruik alleen overgenomen restricties",
|
||||||
|
"Add restrictions on top of inherited": "Restricties toevoegen bovenop geërfd",
|
||||||
|
"Inherited restriction": "Overgenomen restrictie",
|
||||||
|
"Access limited by": "Toegang beperkt door",
|
||||||
|
"Restrict access to control who can view and edit this page": "Beperk de toegang om te bepalen wie deze pagina kan bekijken en bewerken",
|
||||||
|
"Add additional restrictions specific to this page": "Voeg extra beperkingen toe voor deze pagina",
|
||||||
|
"Access": "Toegang",
|
||||||
|
"People with access": "Personen die toegang",
|
||||||
|
"Remove all": "Alles verwijderen",
|
||||||
|
"Remove access": "Toegang verwijderen",
|
||||||
|
"Remove all access": "Alle toegang verwijderen",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "Weet u zeker dat u de toegang van dit lid tot de pagina wilt intrekken?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "Weet u zeker dat u alle specifieke toegang wilt verwijderen? Hiermee wordt de pagina voor iedereen in de ruimte beschikbaar.",
|
||||||
|
"Trash retention": "Bewaartermijn prullenbak",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "Pagina's in de prullenbak worden na deze periode permanent verwijderd.",
|
||||||
|
"Trash retention updated": "Bewaartermijn prullenbak bijgewerkt",
|
||||||
|
"Failed to update trash retention": "Bijwerken van de bewaartermijn voor de prullenbak is mislukt.",
|
||||||
|
"Removed page restriction": "Pagina-restrictie verwijderd",
|
||||||
|
"Added page permission": "Paginatoestemming toegevoegd",
|
||||||
|
"Removed page permission": "Paginatoestemming verwijderd",
|
||||||
|
"Verifying your email": "Je e-mailadres wordt geverifieerd",
|
||||||
|
"Please wait...": "Even geduld...",
|
||||||
|
"Verification failed. The link may have expired.": "Verificatie mislukt. De link is mogelijk verlopen.",
|
||||||
|
"Check your email": "Controleer je e-mail",
|
||||||
|
"We sent a verification link to {{email}}.": "We hebben een verificatielink naar {{email}} gestuurd.",
|
||||||
|
"We sent a verification link to your email.": "We hebben een verificatielink naar je e-mailadres gestuurd.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Klik op de link om je e-mailadres te verifiëren en toegang te krijgen tot je werkruimte.",
|
||||||
|
"Resend verification email": "Verificatie-e-mail opnieuw verzenden",
|
||||||
|
"Verification email sent. Please check your inbox.": "Verificatie-e-mail verzonden. Controleer je inbox.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Het verzenden van de verificatie-e-mail is mislukt. Probeer het opnieuw.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "We hebben je een e-mail gestuurd met je gekoppelde werkruimtes.",
|
||||||
|
"Load more": "Meer laden",
|
||||||
|
"Log out of all devices": "Log uit op alle apparaten",
|
||||||
|
"Log out of all sessions except this device": "Log uit op alle sessies behalve dit apparaat",
|
||||||
|
"This Device": "Dit apparaat",
|
||||||
|
"Unknown device": "Onbekend apparaat",
|
||||||
|
"No active sessions": "Geen actieve sessies",
|
||||||
|
"Session revoked": "Sessie ingetrokken",
|
||||||
|
"All other sessions revoked": "Alle andere sessies ingetrokken",
|
||||||
|
"Last used": "Laatst gebruikt",
|
||||||
|
"Created": "Aangemaakt",
|
||||||
|
"Rename": "Hernoemen",
|
||||||
|
"Publish": "Publiceren",
|
||||||
|
"Security": "Beveiliging",
|
||||||
|
"Enforce SSO": "SSO afdwingen",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Zodra dit is afgedwongen, kunnen leden niet meer inloggen met e-mail en wachtwoord."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -116,6 +116,7 @@
|
|||||||
"No group found": "Nenhum grupo encontrado",
|
"No group found": "Nenhum grupo encontrado",
|
||||||
"No page history saved yet.": "Nenhum histórico de página salvo ainda.",
|
"No page history saved yet.": "Nenhum histórico de página salvo ainda.",
|
||||||
"No pages yet": "Nenhuma página ainda",
|
"No pages yet": "Nenhuma página ainda",
|
||||||
|
"No shared pages": "Sem páginas compartilhadas",
|
||||||
"No results found...": "Nenhum resultado encontrado...",
|
"No results found...": "Nenhum resultado encontrado...",
|
||||||
"No user found": "Nenhum usuário encontrado",
|
"No user found": "Nenhum usuário encontrado",
|
||||||
"Overview": "Visão geral",
|
"Overview": "Visão geral",
|
||||||
@@ -123,11 +124,14 @@
|
|||||||
"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",
|
||||||
"Password": "Senha",
|
"Password": "Senha",
|
||||||
"Password changed successfully": "Senha alterada com sucesso",
|
"Password changed successfully": "Senha alterada com sucesso",
|
||||||
|
"People": "Pessoas",
|
||||||
"Pending": "Pendente",
|
"Pending": "Pendente",
|
||||||
"Please confirm your action": "Por favor, confirme sua ação",
|
"Please confirm your action": "Por favor, confirme sua ação",
|
||||||
"Preferences": "Preferências",
|
"Preferences": "Preferências",
|
||||||
@@ -205,6 +209,9 @@
|
|||||||
"Reply...": "Responder...",
|
"Reply...": "Responder...",
|
||||||
"Error loading comments.": "Erro ao carregar comentários.",
|
"Error loading comments.": "Erro ao carregar comentários.",
|
||||||
"No comments yet.": "Ainda sem comentários.",
|
"No comments yet.": "Ainda sem comentários.",
|
||||||
|
"No open comments.": "Sem comentários em aberto.",
|
||||||
|
"No resolved comments.": "Sem comentários resolvidos.",
|
||||||
|
"Add a comment...": "Adicione um comentário...",
|
||||||
"Edit comment": "Editar comentário",
|
"Edit comment": "Editar comentário",
|
||||||
"Delete comment": "Excluir comentário",
|
"Delete comment": "Excluir comentário",
|
||||||
"Are you sure you want to delete this comment?": "Você tem certeza de que deseja excluir este comentário?",
|
"Are you sure you want to delete this comment?": "Você tem certeza de que deseja excluir este comentário?",
|
||||||
@@ -226,7 +233,6 @@
|
|||||||
"Are you sure you want to unresolve this comment thread?": "Tem certeza de que deseja não resolver este fio de comentários?",
|
"Are you sure you want to unresolve this comment thread?": "Tem certeza de que deseja não resolver este fio de comentários?",
|
||||||
"Resolved": "Resolvido",
|
"Resolved": "Resolvido",
|
||||||
"No active comments.": "Sem comentários ativos.",
|
"No active comments.": "Sem comentários ativos.",
|
||||||
"No resolved comments.": "Sem comentários resolvidos.",
|
|
||||||
"Revoke invitation": "Cancelar o convite",
|
"Revoke invitation": "Cancelar o convite",
|
||||||
"Revoke": "Anular",
|
"Revoke": "Anular",
|
||||||
"Don't": "Não",
|
"Don't": "Não",
|
||||||
@@ -272,6 +278,7 @@
|
|||||||
"Add row below": "Adicionar linha abaixo",
|
"Add row below": "Adicionar linha abaixo",
|
||||||
"Delete table": "Excluir tabela",
|
"Delete table": "Excluir tabela",
|
||||||
"Info": "Informação",
|
"Info": "Informação",
|
||||||
|
"Note": "Observação",
|
||||||
"Success": "Sucesso",
|
"Success": "Sucesso",
|
||||||
"Warning": "Aviso",
|
"Warning": "Aviso",
|
||||||
"Danger": "Perigo",
|
"Danger": "Perigo",
|
||||||
@@ -282,6 +289,11 @@
|
|||||||
"Save & Exit": "Salvar e Sair",
|
"Save & Exit": "Salvar e Sair",
|
||||||
"Double-click to edit Excalidraw diagram": "Clique duas vezes para editar o diagrama Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Clique duas vezes para editar o diagrama Excalidraw",
|
||||||
"Paste link": "Colar link",
|
"Paste link": "Colar link",
|
||||||
|
"Paste link or search pages": "Cole o link ou pesquise páginas",
|
||||||
|
"Link to web page": "Link para página da web",
|
||||||
|
"Recents": "Recentes",
|
||||||
|
"Page or URL": "Página ou URL",
|
||||||
|
"Link title": "Título do link",
|
||||||
"Edit link": "Editar link",
|
"Edit link": "Editar link",
|
||||||
"Remove link": "Remover link",
|
"Remove link": "Remover link",
|
||||||
"Add link": "Adicionar link",
|
"Add link": "Adicionar link",
|
||||||
@@ -329,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Insira um divisor horizontal",
|
"Insert horizontal rule divider": "Insira um divisor horizontal",
|
||||||
"Upload any image from your device.": "Envie qualquer imagem do seu dispositivo.",
|
"Upload any image from your device.": "Envie qualquer imagem do seu dispositivo.",
|
||||||
"Upload any video from your device.": "Envie qualquer vídeo do seu dispositivo.",
|
"Upload any video from your device.": "Envie qualquer vídeo do seu dispositivo.",
|
||||||
|
"Upload any audio from your device.": "Envie qualquer áudio do seu dispositivo.",
|
||||||
"Upload any file from your device.": "Envie qualquer arquivo do seu dispositivo.",
|
"Upload any file from your device.": "Envie qualquer arquivo do seu dispositivo.",
|
||||||
"Uploading {{name}}": "Enviando {{name}}",
|
"Uploading {{name}}": "Enviando {{name}}",
|
||||||
"Uploading file": "Enviando arquivo",
|
"Uploading file": "Enviando arquivo",
|
||||||
@@ -339,6 +352,12 @@
|
|||||||
"Divider": "Divisor",
|
"Divider": "Divisor",
|
||||||
"Quote": "Citação",
|
"Quote": "Citação",
|
||||||
"Image": "Imagem",
|
"Image": "Imagem",
|
||||||
|
"Audio": "Áudio.",
|
||||||
|
"Embed PDF": "Incorporar PDF",
|
||||||
|
"Upload and embed a PDF file.": "Envie e incorpore um arquivo PDF.",
|
||||||
|
"Embed as PDF": "Incorporar como PDF",
|
||||||
|
"Failed to load PDF": "Falha ao carregar PDF",
|
||||||
|
"Convert to attachment": "Converter em anexo",
|
||||||
"File attachment": "Anexo de arquivo",
|
"File attachment": "Anexo de arquivo",
|
||||||
"Toggle block": "Bloco colapsável",
|
"Toggle block": "Bloco colapsável",
|
||||||
"Callout": "Aviso",
|
"Callout": "Aviso",
|
||||||
@@ -353,9 +372,23 @@
|
|||||||
"Insert current date": "Insira a data atual",
|
"Insert current date": "Insira a data atual",
|
||||||
"Draw and sketch excalidraw diagrams": "Desenhe e esboce diagramas Excalidraw",
|
"Draw and sketch excalidraw diagrams": "Desenhe e esboce diagramas Excalidraw",
|
||||||
"Multiple": "Múltiplo",
|
"Multiple": "Múltiplo",
|
||||||
|
"Turn into": "Transformar em",
|
||||||
|
"Text align": "Alinhar texto",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "Esta página pode ter sido excluída, movida ou você pode não ter acesso a ela.",
|
||||||
|
"Go to homepage": "Ir para a página inicial",
|
||||||
|
"Pages you create will show up here.": "As páginas que você criar aparecerão aqui.",
|
||||||
"Heading {{level}}": "Título {{level}}",
|
"Heading {{level}}": "Título {{level}}",
|
||||||
"Toggle title": "Alternar título",
|
"Toggle title": "Alternar título",
|
||||||
"Write anything. Enter \"/\" for commands": "Escreva qualquer coisa. Digite \"/\" para comandos",
|
"Write anything. Enter \"/\" for commands": "Escreva qualquer coisa. Digite \"/\" para comandos",
|
||||||
|
"Write...": "Escreva...",
|
||||||
|
"Column count": "Número de colunas",
|
||||||
|
"{{count}} Columns": "{{count}} colunas",
|
||||||
|
"Equal columns": "Colunas iguais",
|
||||||
|
"Left sidebar": "Barra lateral esquerda",
|
||||||
|
"Right sidebar": "Barra lateral direita",
|
||||||
|
"Wide center": "Centro largo",
|
||||||
|
"Left wide": "Largo à esquerda",
|
||||||
|
"Right wide": "Largo à direita",
|
||||||
"Names do not match": "Os nomes não coincidem",
|
"Names do not match": "Os nomes não coincidem",
|
||||||
"Today, {{time}}": "Hoje, {{time}}",
|
"Today, {{time}}": "Hoje, {{time}}",
|
||||||
"Yesterday, {{time}}": "Ontem, {{time}}",
|
"Yesterday, {{time}}": "Ontem, {{time}}",
|
||||||
@@ -378,6 +411,13 @@
|
|||||||
"Delete member": "Excluir membro",
|
"Delete member": "Excluir membro",
|
||||||
"Member deleted successfully": "Membro removido com sucesso",
|
"Member deleted successfully": "Membro removido com sucesso",
|
||||||
"Are you sure you want to delete this workspace member? This action is irreversible.": "Você tem certeza que deseja deletar este membro do workspace? Esta ação é irreversível.",
|
"Are you sure you want to delete this workspace member? This action is irreversible.": "Você tem certeza que deseja deletar este membro do workspace? Esta ação é irreversível.",
|
||||||
|
"Deactivate member": "Desativar membro",
|
||||||
|
"Activate member": "Ativar membro",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "Tem certeza de que deseja desativar este membro do espaço de trabalho? Ele não poderá mais acessar este espaço de trabalho.",
|
||||||
|
"Are you sure you want to activate this workspace member?": "Tem certeza de que deseja ativar este membro do espaço de trabalho?",
|
||||||
|
"Deactivate": "Desativar",
|
||||||
|
"Activate": "Ativar",
|
||||||
|
"Deactivated": "Desativado",
|
||||||
"Move": "Mover",
|
"Move": "Mover",
|
||||||
"Move page": "Mover página",
|
"Move page": "Mover página",
|
||||||
"Move page to a different space.": "Mover página para um espaço diferente.",
|
"Move page to a different space.": "Mover página para um espaço diferente.",
|
||||||
@@ -405,6 +445,25 @@
|
|||||||
"Share deleted successfully": "Compartilhamento excluído com sucesso",
|
"Share deleted successfully": "Compartilhamento excluído com sucesso",
|
||||||
"Share not found": "Compartilhamento não encontrado",
|
"Share not found": "Compartilhamento não encontrado",
|
||||||
"Failed to share page": "Falha ao compartilhar página",
|
"Failed to share page": "Falha ao compartilhar página",
|
||||||
|
"Disable public sharing": "Desativar compartilhamento público",
|
||||||
|
"Prevent members from sharing pages publicly.": "Impedir que os membros compartilhem páginas publicamente.",
|
||||||
|
"Toggle public sharing": "Alternar compartilhamento público",
|
||||||
|
"Toggle space public sharing": "Alternar compartilhamento público do espaço",
|
||||||
|
"Allow viewers to comment": "Permitir que os visualizadores comentem",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Permitir que os visualizadores adicionem comentários em páginas deste espaço.",
|
||||||
|
"Toggle viewer comments": "Ativar/desativar comentários de visualizadores",
|
||||||
|
"Public sharing is disabled at the workspace level": "O compartilhamento público está desativado no nível do espaço de trabalho",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "Impedir que as páginas neste espaço sejam compartilhadas publicamente.",
|
||||||
|
"Page permissions": "Permissões da página},{",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Controle quem pode visualizar e editar páginas individuais. Disponível com licença empresarial.",
|
||||||
|
"Enable public sharing": "Ativar compartilhamento público",
|
||||||
|
"Are you sure you want to enable public sharing? Members will be able to share pages publicly.": "Tem certeza de que deseja ativar o compartilhamento público? Os membros poderão compartilhar páginas publicamente.",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this workspace will be deleted.": "Tem certeza de que deseja desativar o compartilhamento público? Todos os links compartilhados existentes neste espaço de trabalho serão excluídos.",
|
||||||
|
"Are you sure you want to enable public sharing for this space?": "Tem certeza de que deseja ativar o compartilhamento público para este espaço?",
|
||||||
|
"Are you sure you want to disable public sharing? All existing shared links in this space will be deleted.": "Tem certeza de que deseja desativar o compartilhamento público? Todos os links compartilhados existentes neste espaço serão excluídos.",
|
||||||
|
"Public sharing is disabled": "Compartilhamento público está desativado",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "O compartilhamento público foi desativado no nível do espaço de trabalho.",
|
||||||
|
"Public sharing has been disabled for this space.": "O compartilhamento público foi desativado para este espaço.",
|
||||||
"Copy page": "Copiar página",
|
"Copy page": "Copiar página",
|
||||||
"Copy page to a different space.": "Copiar página para um espaço diferente.",
|
"Copy page to a different space.": "Copiar página para um espaço diferente.",
|
||||||
"Page copied successfully": "Página copiada com sucesso",
|
"Page copied successfully": "Página copiada com sucesso",
|
||||||
@@ -487,7 +546,7 @@
|
|||||||
"Enter one of your backup codes. Each backup code can only be used once.": "Digite um de seus códigos de backup. Cada código de backup só pode ser usado uma vez.",
|
"Enter one of your backup codes. Each backup code can only be used once.": "Digite um de seus códigos de backup. Cada código de backup só pode ser usado uma vez.",
|
||||||
"Verify": "Verificar",
|
"Verify": "Verificar",
|
||||||
"Trash": "Lixeira",
|
"Trash": "Lixeira",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "Páginas na lixeira serão excluídas permanentemente após 30 dias.",
|
"Pages in trash will be permanently deleted after {{count}} days.": "{count, plural, one {A página na lixeira será excluída permanentemente após # dia.} other {As páginas na lixeira serão excluídas permanentemente após # dias.}}",
|
||||||
"Deleted": "Excluído",
|
"Deleted": "Excluído",
|
||||||
"No pages in trash": "Sem páginas na lixeira",
|
"No pages in trash": "Sem páginas na lixeira",
|
||||||
"Permanently delete page?": "Excluir página permanentemente?",
|
"Permanently delete page?": "Excluir página permanentemente?",
|
||||||
@@ -559,19 +618,138 @@
|
|||||||
"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.",
|
"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",
|
"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",
|
"Manage API keys for all users in the workspace": "Gerenciar chaves API para todos os usuários no espaço de trabalho",
|
||||||
|
"Restrict API key creation to admins": "Restringir a criação de chave de API aos administradores",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "Somente administradores e proprietários podem criar novas chaves de API. As chaves de membros já existentes continuarão funcionando.",
|
||||||
|
"Toggle restrict API keys to admins": "Alternar restrição de chaves de API para administradores",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "A criação de chaves de API foi restringida aos administradores pelo administrador do seu workspace.",
|
||||||
"AI settings": "Configurações de IA",
|
"AI settings": "Configurações de IA",
|
||||||
"AI search": "Pesquisa IA",
|
"AI search": "Pesquisa IA",
|
||||||
"AI Answer": "Resposta de IA",
|
"AI Answer": "Resposta de IA",
|
||||||
"Ask AI": "Pergunte à IA",
|
"Ask AI": "Pergunte à IA",
|
||||||
"AI is thinking...": "IA está pensando...",
|
"AI is thinking...": "IA está pensando...",
|
||||||
"Ask a question...": "Faça uma pergunta...",
|
"Ask a question...": "Faça uma pergunta...",
|
||||||
"AI-powered search (Ask AI)": "Pesquisa com IA (Pergunte à IA)",
|
"AI Answers": "Respostas de IA",
|
||||||
|
"AI-powered search (AI Answers)": "Pesquisa com IA (Respostas de 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.",
|
"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",
|
"Toggle AI search": "Alternar pesquisa de IA",
|
||||||
|
"Generative AI (Ask AI)": "IA generativa (Perguntar à IA)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Habilitar geração de conteúdo com IA no editor. Permite aos usuários gerar, melhorar, traduzir e transformar texto.",
|
||||||
|
"Toggle generative AI": "Alternar IA generativa",
|
||||||
|
"Upgrade your plan": "Faça upgrade do seu plano",
|
||||||
|
"Available with a paid license": "Disponível com uma licença paga",
|
||||||
|
"Upgrade your license tier.": "Faça upgrade do seu nível de licença.",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "A IA está disponível apenas na edição empresarial do Docmost. Contate sales@docmost.com.",
|
||||||
|
"AI & MCP": "IA e MCP",
|
||||||
|
"AI": "IA",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "Protocolo de Contexto de Modelo (MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Ative o servidor MCP para permitir que assistentes de IA e ferramentas interajam com o conteúdo do seu espaço de trabalho.",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "O MCP está disponível apenas na edição empresarial do Docmost. Contate sales@docmost.com.",
|
||||||
|
"MCP Server URL": "URL do servidor MCP",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Use sua chave de API para autenticação. Você pode gerenciar chaves de API nas configurações da sua conta.",
|
||||||
|
"Supported tools": "Ferramentas compatíveis",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "Seu espaço de trabalho tem MCP habilitado. Use sua chave de API para conectar assistentes de IA.",
|
||||||
|
"MCP server URL:": "URL do servidor MCP:",
|
||||||
|
"Learn more": "Saiba mais",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Gerencie as chaves de API de todos os usuários do workspace. Veja a <anchor>documentação da API</anchor> para detalhes de uso.",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "Veja a <anchor>documentação da API</anchor> para detalhes de uso.",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "Veja a <anchor>documentação MCP</anchor>.",
|
||||||
"Sources": "Fontes",
|
"Sources": "Fontes",
|
||||||
"Ask AI not available for attachments": "Perguntar à IA não está disponível para anexos",
|
"AI Answers not available for attachments": "Respostas de IA não disponíveis para anexos",
|
||||||
"No answer available": "Nenhuma resposta disponível",
|
"No answer available": "Nenhuma resposta disponível",
|
||||||
"Background color": "Cor de fundo",
|
"Background color": "Cor de fundo",
|
||||||
"Highlight color": "Cor de destaque",
|
"Highlight color": "Cor de destaque",
|
||||||
"Remove color": "Remover cor"
|
"Remove color": "Remover cor",
|
||||||
|
"Notifications": "Notificações",
|
||||||
|
"No notifications": "Sem notificações",
|
||||||
|
"No unread notifications": "Sem notificações não lidas",
|
||||||
|
"All notifications": "Todas as notificações",
|
||||||
|
"Unread only": "Somente não lidas",
|
||||||
|
"Mark all as read": "Marcar todas como lidas",
|
||||||
|
"Mark as read": "Marcar como lida",
|
||||||
|
"More options": "Mais opções",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> mencionou você em um comentário",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> comentou em uma página",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> resolveu um comentário",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> mencionou você em uma página",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> concedeu acesso de edição a uma página",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> concedeu acesso de visualização a uma página",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> atualizou uma página.",
|
||||||
|
"Watch page": "Observar página",
|
||||||
|
"Stop watching": "Parar de observar",
|
||||||
|
"Email notifications": "Notificações por e-mail",
|
||||||
|
"Page updates": "Atualizações da página",
|
||||||
|
"Get notified when pages you watch are updated.": "Receba notificações quando as páginas que você observa forem atualizadas.",
|
||||||
|
"Page mentions": "Menções na página",
|
||||||
|
"Get notified when someone mentions you on a page.": "Receba notificações quando alguém mencionar você em uma página.",
|
||||||
|
"Comment mentions": "Menções em comentários",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Receba notificações quando alguém mencionar você em um comentário.",
|
||||||
|
"New comments": "Novos comentários",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Receba notificações sobre novos comentários nas discussões em que você participa.",
|
||||||
|
"Resolved comments": "Comentários resolvidos",
|
||||||
|
"Get notified when your comment is resolved.": "Receba notificações quando seu comentário for resolvido.",
|
||||||
|
"You are now watching this page": "Agora você está observando esta página",
|
||||||
|
"You are no longer watching this page": "Você não está mais observando esta página",
|
||||||
|
"Direct": "Direto",
|
||||||
|
"Updates": "Atualizações",
|
||||||
|
"Today": "Hoje",
|
||||||
|
"Yesterday": "Ontem",
|
||||||
|
"This week": "Esta semana",
|
||||||
|
"Older": "Mais antigo",
|
||||||
|
"Restricted page": "Página restrita",
|
||||||
|
"Restricted pages cannot be shared publicly.": "Páginas restritas não podem ser compartilhadas publicamente.",
|
||||||
|
"Restricted by parent": "Restrita pela página pai",
|
||||||
|
"Restricted": "Restrito",
|
||||||
|
"Open": "Aberto",
|
||||||
|
"Inherits restrictions from ancestor page": "Herda restrições da página ancestral",
|
||||||
|
"Only people listed below can access this page": "Apenas as pessoas listadas abaixo podem acessar esta página",
|
||||||
|
"Everyone in this space can access": "Todos neste espaço podem acessar",
|
||||||
|
"No additional restrictions on this page": "Sem restrições adicionais nesta página",
|
||||||
|
"Only specific people can access": "Apenas pessoas específicas podem acessar",
|
||||||
|
"Use only inherited restrictions": "Usar apenas restrições herdadas",
|
||||||
|
"Add restrictions on top of inherited": "Adicionar restrições além das herdadas",
|
||||||
|
"Inherited restriction": "Restrição herdada",
|
||||||
|
"Access limited by": "Acesso limitado por",
|
||||||
|
"Restrict access to control who can view and edit this page": "Restringir o acesso para controlar quem pode visualizar e editar esta página",
|
||||||
|
"Add additional restrictions specific to this page": "Adicionar restrições adicionais específicas para esta página",
|
||||||
|
"Access": "Acesso",
|
||||||
|
"People with access": "Pessoas com acesso",
|
||||||
|
"Remove all": "Remover tudo",
|
||||||
|
"Remove access": "Remover acesso",
|
||||||
|
"Remove all access": "Remover todo o acesso",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "Tem certeza de que deseja remover o acesso deste membro à página?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "Tem certeza de que deseja remover todo o acesso específico? Isso fará com que a página fique aberta para todos no espaço.",
|
||||||
|
"Trash retention": "Retenção da lixeira",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "As páginas na lixeira serão excluídas permanentemente após este período.",
|
||||||
|
"Trash retention updated": "Retenção da lixeira atualizada",
|
||||||
|
"Failed to update trash retention": "Falha ao atualizar a retenção da lixeira",
|
||||||
|
"Removed page restriction": "Restrição de página removida",
|
||||||
|
"Added page permission": "Permissão de página adicionada",
|
||||||
|
"Removed page permission": "Permissão de página removida",
|
||||||
|
"Verifying your email": "Verificando seu e-mail",
|
||||||
|
"Please wait...": "Por favor, aguarde...",
|
||||||
|
"Verification failed. The link may have expired.": "Falha na verificação. O link pode ter expirado.",
|
||||||
|
"Check your email": "Verifique seu e-mail",
|
||||||
|
"We sent a verification link to {{email}}.": "Enviamos um link de verificação para {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Enviamos um link de verificação para seu e-mail.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Clique no link para verificar seu e-mail e acessar seu workspace.",
|
||||||
|
"Resend verification email": "Reenviar e-mail de verificação",
|
||||||
|
"Verification email sent. Please check your inbox.": "E-mail de verificação enviado. Por favor, verifique sua caixa de entrada.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Falha ao reenviar o e-mail de verificação. Por favor, tente novamente.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Enviamos um e-mail para você com seus workspaces associados.",
|
||||||
|
"Load more": "Carregar mais",
|
||||||
|
"Log out of all devices": "Sair de todos os dispositivos",
|
||||||
|
"Log out of all sessions except this device": "Sair de todas as sessões, exceto neste dispositivo",
|
||||||
|
"This Device": "Este dispositivo",
|
||||||
|
"Unknown device": "Dispositivo desconhecido",
|
||||||
|
"No active sessions": "Sem sessões ativas",
|
||||||
|
"Session revoked": "Sessão revogada",
|
||||||
|
"All other sessions revoked": "Todas as outras sessões revogadas",
|
||||||
|
"Last used": "Último uso",
|
||||||
|
"Created": "Criado",
|
||||||
|
"Rename": "Renomear",
|
||||||
|
"Publish": "Publicar",
|
||||||
|
"Security": "Segurança",
|
||||||
|
"Enforce SSO": "Exigir SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Uma vez exigido, os membros não poderão entrar com e-mail e senha."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,7 +10,7 @@
|
|||||||
"Admin": "Администратор",
|
"Admin": "Администратор",
|
||||||
"Are you sure you want to delete this group? Members will lose access to resources this group has access to.": "Вы уверены, что хотите удалить эту группу? Участники потеряют доступ к материалам, к которым у этой группы есть доступ.",
|
"Are you sure you want to delete this group? Members will lose access to resources this group has access to.": "Вы уверены, что хотите удалить эту группу? Участники потеряют доступ к материалам, к которым у этой группы есть доступ.",
|
||||||
"Are you sure you want to delete this page?": "Вы уверены, что хотите удалить эту страницу?",
|
"Are you sure you want to delete this page?": "Вы уверены, что хотите удалить эту страницу?",
|
||||||
"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": "Могут становиться участниками групп и пространств в рабочей области",
|
||||||
@@ -116,6 +116,7 @@
|
|||||||
"No group found": "Группа не найдена",
|
"No group found": "Группа не найдена",
|
||||||
"No page history saved yet.": "История страниц ещё не сохранена.",
|
"No page history saved yet.": "История страниц ещё не сохранена.",
|
||||||
"No pages yet": "Страниц пока нет",
|
"No pages yet": "Страниц пока нет",
|
||||||
|
"No shared pages": "Нет общих страниц",
|
||||||
"No results found...": "Результаты не найдены...",
|
"No results found...": "Результаты не найдены...",
|
||||||
"No user found": "Пользователь не найден",
|
"No user found": "Пользователь не найден",
|
||||||
"Overview": "Обзор",
|
"Overview": "Обзор",
|
||||||
@@ -123,11 +124,14 @@
|
|||||||
"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": "страницы",
|
||||||
"Password": "Пароль",
|
"Password": "Пароль",
|
||||||
"Password changed successfully": "Пароль успешно изменён",
|
"Password changed successfully": "Пароль успешно изменён",
|
||||||
|
"People": "Люди",
|
||||||
"Pending": "В ожидании",
|
"Pending": "В ожидании",
|
||||||
"Please confirm your action": "Пожалуйста, подтвердите ваше действие",
|
"Please confirm your action": "Пожалуйста, подтвердите ваше действие",
|
||||||
"Preferences": "Настройки",
|
"Preferences": "Настройки",
|
||||||
@@ -205,6 +209,9 @@
|
|||||||
"Reply...": "Ответить...",
|
"Reply...": "Ответить...",
|
||||||
"Error loading comments.": "Ошибка при загрузке комментариев.",
|
"Error loading comments.": "Ошибка при загрузке комментариев.",
|
||||||
"No comments yet.": "Комментариев пока нет.",
|
"No comments yet.": "Комментариев пока нет.",
|
||||||
|
"No open comments.": "Нет открытых комментариев.",
|
||||||
|
"No resolved comments.": "Нет решённых комментариев.",
|
||||||
|
"Add a comment...": "Добавить комментарий...",
|
||||||
"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?": "Вы уверены, что хотите удалить этот комментарий?",
|
||||||
@@ -226,7 +233,6 @@
|
|||||||
"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.": "Нет решённых комментариев.",
|
|
||||||
"Revoke invitation": "Отозвать приглашение",
|
"Revoke invitation": "Отозвать приглашение",
|
||||||
"Revoke": "Отозвать",
|
"Revoke": "Отозвать",
|
||||||
"Don't": "Нет",
|
"Don't": "Нет",
|
||||||
@@ -272,6 +278,7 @@
|
|||||||
"Add row below": "Добавить строку ниже",
|
"Add row below": "Добавить строку ниже",
|
||||||
"Delete table": "Удалить таблицу",
|
"Delete table": "Удалить таблицу",
|
||||||
"Info": "Информация",
|
"Info": "Информация",
|
||||||
|
"Note": "Примечание",
|
||||||
"Success": "Успешно",
|
"Success": "Успешно",
|
||||||
"Warning": "Предупреждение",
|
"Warning": "Предупреждение",
|
||||||
"Danger": "Важно",
|
"Danger": "Важно",
|
||||||
@@ -282,6 +289,11 @@
|
|||||||
"Save & Exit": "Сохранить и выйти",
|
"Save & Exit": "Сохранить и выйти",
|
||||||
"Double-click to edit Excalidraw diagram": "Кликните дважды для редактирования диаграммы Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Кликните дважды для редактирования диаграммы Excalidraw",
|
||||||
"Paste link": "Вставить ссылку",
|
"Paste link": "Вставить ссылку",
|
||||||
|
"Paste link or search pages": "Вставьте ссылку или найдите страницы",
|
||||||
|
"Link to web page": "Ссылка на веб-страницу",
|
||||||
|
"Recents": "Недавние",
|
||||||
|
"Page or URL": "Страница или URL",
|
||||||
|
"Link title": "Заголовок ссылки",
|
||||||
"Edit link": "Редактировать ссылку",
|
"Edit link": "Редактировать ссылку",
|
||||||
"Remove link": "Удалить ссылку",
|
"Remove link": "Удалить ссылку",
|
||||||
"Add link": "Добавить ссылку",
|
"Add link": "Добавить ссылку",
|
||||||
@@ -329,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Вставить горизонтальный разделитель",
|
"Insert horizontal rule divider": "Вставить горизонтальный разделитель",
|
||||||
"Upload any image from your device.": "Загрузить любое изображение с вашего устройства.",
|
"Upload any image from your device.": "Загрузить любое изображение с вашего устройства.",
|
||||||
"Upload any video from your device.": "Загрузить любое видео с вашего устройства.",
|
"Upload any video from your device.": "Загрузить любое видео с вашего устройства.",
|
||||||
|
"Upload any audio from your device.": "Загрузите любой аудиофайл с вашего устройства.",
|
||||||
"Upload any file from your device.": "Загрузить любой файл с вашего устройства.",
|
"Upload any file from your device.": "Загрузить любой файл с вашего устройства.",
|
||||||
"Uploading {{name}}": "Загрузка {{name}}",
|
"Uploading {{name}}": "Загрузка {{name}}",
|
||||||
"Uploading file": "Загрузка файла",
|
"Uploading file": "Загрузка файла",
|
||||||
@@ -339,6 +352,12 @@
|
|||||||
"Divider": "Разделитель",
|
"Divider": "Разделитель",
|
||||||
"Quote": "Цитата",
|
"Quote": "Цитата",
|
||||||
"Image": "Изображение",
|
"Image": "Изображение",
|
||||||
|
"Audio": "Аудио.",
|
||||||
|
"Embed PDF": "Встроить PDF",
|
||||||
|
"Upload and embed a PDF file.": "Загрузите и встроите PDF-файл.",
|
||||||
|
"Embed as PDF": "Встроить как PDF",
|
||||||
|
"Failed to load PDF": "Не удалось загрузить PDF",
|
||||||
|
"Convert to attachment": "Преобразовать в вложение",
|
||||||
"File attachment": "Прикрепленный файл",
|
"File attachment": "Прикрепленный файл",
|
||||||
"Toggle block": "Сворачиваемый блок",
|
"Toggle block": "Сворачиваемый блок",
|
||||||
"Callout": "Выноска",
|
"Callout": "Выноска",
|
||||||
@@ -353,9 +372,23 @@
|
|||||||
"Insert current date": "Вставить текущую дату",
|
"Insert current date": "Вставить текущую дату",
|
||||||
"Draw and sketch excalidraw diagrams": "Вставить и рисовать диаграммы Excalidraw",
|
"Draw and sketch excalidraw diagrams": "Вставить и рисовать диаграммы Excalidraw",
|
||||||
"Multiple": "Несколько",
|
"Multiple": "Несколько",
|
||||||
|
"Turn into": "Преобразовать в",
|
||||||
|
"Text align": "Выравнивание текста",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "Эта страница могла быть удалена, перемещена, или у вас может отсутствовать доступ к ней.",
|
||||||
|
"Go to homepage": "Вернуться на главную",
|
||||||
|
"Pages you create will show up here.": "Созданные вами страницы появятся здесь.",
|
||||||
"Heading {{level}}": "Заголовок {{level}}",
|
"Heading {{level}}": "Заголовок {{level}}",
|
||||||
"Toggle title": "Переключить заголовок",
|
"Toggle title": "Переключить заголовок",
|
||||||
"Write anything. Enter \"/\" for commands": "Начните писать. Введите \"/\" для списка команд",
|
"Write anything. Enter \"/\" for commands": "Начните писать. Введите \"/\" для списка команд",
|
||||||
|
"Write...": "Напишите...",
|
||||||
|
"Column count": "Количество столбцов",
|
||||||
|
"{{count}} Columns": "{count, plural, one{# столбец} few{# столбца} many{# столбцов} other{# столбца}}",
|
||||||
|
"Equal columns": "Равные столбцы",
|
||||||
|
"Left sidebar": "Левая боковая панель",
|
||||||
|
"Right sidebar": "Правая боковая панель",
|
||||||
|
"Wide center": "Широкий по центру",
|
||||||
|
"Left wide": "Широкий слева",
|
||||||
|
"Right wide": "Широкий справа",
|
||||||
"Names do not match": "Названия не совпадают",
|
"Names do not match": "Названия не совпадают",
|
||||||
"Today, {{time}}": "Сегодня, {{time}}",
|
"Today, {{time}}": "Сегодня, {{time}}",
|
||||||
"Yesterday, {{time}}": "Вчера, {{time}}",
|
"Yesterday, {{time}}": "Вчера, {{time}}",
|
||||||
@@ -378,6 +411,13 @@
|
|||||||
"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.": "Вы уверены, что хотите удалить этого участника рабочей области? Это действие необратимо.",
|
||||||
|
"Deactivate member": "Деактивировать участника",
|
||||||
|
"Activate member": "Активировать участника",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "Вы уверены, что хотите деактивировать этого участника рабочего пространства? Они больше не смогут получить доступ к этому рабочему пространству.",
|
||||||
|
"Are you sure you want to activate this workspace member?": "Вы уверены, что хотите активировать этого участника рабочего пространства?",
|
||||||
|
"Deactivate": "Деактивировать",
|
||||||
|
"Activate": "Активировать",
|
||||||
|
"Deactivated": "Деактивирован",
|
||||||
"Move": "Переместить",
|
"Move": "Переместить",
|
||||||
"Move page": "Переместить страницу",
|
"Move page": "Переместить страницу",
|
||||||
"Move page to a different space.": "Переместите страницу в другое пространство.",
|
"Move page to a different space.": "Переместите страницу в другое пространство.",
|
||||||
@@ -405,6 +445,25 @@
|
|||||||
"Share deleted successfully": "Общий доступ успешно удален",
|
"Share deleted successfully": "Общий доступ успешно удален",
|
||||||
"Share not found": "Общий доступ не найден",
|
"Share not found": "Общий доступ не найден",
|
||||||
"Failed to share page": "Не удалось поделиться страницей",
|
"Failed to share page": "Не удалось поделиться страницей",
|
||||||
|
"Disable public sharing": "Отключить общий доступ",
|
||||||
|
"Prevent members from sharing pages publicly.": "Запретить участникам делиться страницами публично.",
|
||||||
|
"Toggle public sharing": "Переключить общий доступ",
|
||||||
|
"Toggle space public sharing": "Переключить общий доступ для пространства",
|
||||||
|
"Allow viewers to comment": "Разрешить зрителям комментировать",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Разрешить зрителям добавлять комментарии на страницах в этом пространстве.",
|
||||||
|
"Toggle viewer comments": "Переключить комментарии зрителей",
|
||||||
|
"Public sharing is disabled at the workspace level": "Общий доступ отключен на уровне рабочего пространства",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "Запретить делиться страницами в этом пространстве публично.",
|
||||||
|
"Page permissions": "Права доступа к странице},{",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Контролируйте, кто может просматривать и редактировать отдельные страницы. Доступно при наличии лицензии Enterprise.",
|
||||||
|
"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 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 disable public sharing? All existing shared links in this space will be deleted.": "Вы уверены, что хотите отключить общий доступ? Все существующие ссылки в этом пространстве будут удалены.",
|
||||||
|
"Public sharing is disabled": "Общий доступ отключен",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "Общий доступ был отключен на уровне рабочего пространства.",
|
||||||
|
"Public sharing has been disabled for this space.": "Общий доступ был отключен для этого пространства.",
|
||||||
"Copy page": "Копировать страницу",
|
"Copy page": "Копировать страницу",
|
||||||
"Copy page to a different space.": "Копировать страницу в другое пространство.",
|
"Copy page to a different space.": "Копировать страницу в другое пространство.",
|
||||||
"Page copied successfully": "Страница успешно скопирована",
|
"Page copied successfully": "Страница успешно скопирована",
|
||||||
@@ -487,7 +546,7 @@
|
|||||||
"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.": "Введите один из ваших резервных кодов. Каждый резервный код можно использовать только один раз.",
|
||||||
"Verify": "Проверить",
|
"Verify": "Проверить",
|
||||||
"Trash": "Корзина",
|
"Trash": "Корзина",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "Страницы в корзине будут окончательно удалены через 30 дней.",
|
"Pages in trash will be permanently deleted after {{count}} days.": "{count, plural, one {Страница в корзине будет окончательно удалена через # день.} few {Страницы в корзине будут окончательно удалены через # дня.} many {Страницы в корзине будут окончательно удалены через # дней.} other {Страницы в корзине будут окончательно удалены через # дней.}}",
|
||||||
"Deleted": "Удалено",
|
"Deleted": "Удалено",
|
||||||
"No pages in trash": "В корзине нет страниц",
|
"No pages in trash": "В корзине нет страниц",
|
||||||
"Permanently delete page?": "Удалить страницу окончательно?",
|
"Permanently delete page?": "Удалить страницу окончательно?",
|
||||||
@@ -559,19 +618,138 @@
|
|||||||
"This action cannot be undone. Any applications using this API key will stop working.": "Это действие необратимо. Любые приложения, использующие этот API ключ, перестанут работать.",
|
"This action cannot be undone. Any applications using this API key will stop working.": "Это действие необратимо. Любые приложения, использующие этот API ключ, перестанут работать.",
|
||||||
"Update API key": "Обновить API ключ",
|
"Update API key": "Обновить API ключ",
|
||||||
"Manage API keys for all users in the workspace": "Управлять API ключами для всех пользователей в рабочей области",
|
"Manage API keys for all users in the workspace": "Управлять API ключами для всех пользователей в рабочей области",
|
||||||
|
"Restrict API key creation to admins": "Ограничить создание API-ключей только администраторами.",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "Только администраторы и владельцы могут создавать новые API-ключи. Существующие ключи участников продолжат работать.",
|
||||||
|
"Toggle restrict API keys to admins": "Переключить ограничение создания API-ключей только для администраторов",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "Создание API-ключей ограничено администраторами вашего рабочего пространства.",
|
||||||
"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 Answers": "Ответы ИИ",
|
||||||
|
"AI-powered search (AI Answers)": "Поиск на базе ИИ (Ответы ИИ)",
|
||||||
"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": "Переключить поиск ИИ",
|
||||||
|
"Generative AI (Ask AI)": "Генеративный ИИ (Спросить ИИ)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Включите создание контента на базе ИИ в редакторе. Позволяет пользователям генерировать, улучшать, переводить и преобразовывать текст.",
|
||||||
|
"Toggle generative AI": "Переключить генеративный ИИ",
|
||||||
|
"Upgrade your plan": "Обновите свой тарифный план",
|
||||||
|
"Available with a paid license": "Доступно с платной лицензией",
|
||||||
|
"Upgrade your license tier.": "Обновите уровень вашей лицензии.",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "ИИ доступен только в корпоративной версии Docmost. Свяжитесь по адресу sales@docmost.com.",
|
||||||
|
"AI & MCP": "ИИ и MCP",
|
||||||
|
"AI": "ИИ",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "Протокол контекста модели (MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Включите сервер MCP, чтобы ИИ-ассистенты и инструменты могли взаимодействовать с содержимым вашего рабочего пространства.",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP доступен только в корпоративной версии Docmost. Свяжитесь по адресу sales@docmost.com.",
|
||||||
|
"MCP Server URL": "URL сервера MCP",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Используйте ваш API-ключ для аутентификации. Управлять API-ключами можно в настройках аккаунта.",
|
||||||
|
"Supported tools": "Поддерживаемые инструменты",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "В вашем рабочем пространстве включён MCP. Используйте свой API-ключ для подключения ИИ-ассистентов.",
|
||||||
|
"MCP server URL:": "URL сервера MCP:",
|
||||||
|
"Learn more": "Подробнее",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Управляйте API-ключами для всех пользователей в рабочем пространстве. Смотрите <anchor>документацию по API</anchor> для получения информации об использовании.",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "Смотрите <anchor>документацию по API</anchor> для получения информации об использовании.",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "Смотрите <anchor>документацию по MCP</anchor>.",
|
||||||
"Sources": "Источники",
|
"Sources": "Источники",
|
||||||
"Ask AI not available for attachments": "Функция \"Спросить ИИ\" недоступна для вложений",
|
"AI Answers not available for attachments": "Ответы ИИ недоступны для вложений",
|
||||||
"No answer available": "Ответ недоступен",
|
"No answer available": "Ответ недоступен",
|
||||||
"Background color": "Цвет фона",
|
"Background color": "Цвет фона",
|
||||||
"Highlight color": "Цвет выделения",
|
"Highlight color": "Цвет выделения",
|
||||||
"Remove color": "Удалить цвет"
|
"Remove color": "Удалить цвет",
|
||||||
|
"Notifications": "Уведомления",
|
||||||
|
"No notifications": "Нет уведомлений",
|
||||||
|
"No unread notifications": "Нет непрочитанных уведомлений",
|
||||||
|
"All notifications": "Все уведомления",
|
||||||
|
"Unread only": "Только непрочитанные",
|
||||||
|
"Mark all as read": "Отметить все как прочитанные",
|
||||||
|
"Mark as read": "Отметить как прочитанное",
|
||||||
|
"More options": "Больше возможностей",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> упомянул вас в комментарии",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> оставил комментарий на странице",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> решил комментарий",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> упомянул вас на странице",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> предоставил вам доступ для редактирования страницы",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> предоставил вам доступ к просмотру страницы",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> обновил страницу.",
|
||||||
|
"Watch page": "Следить за страницей",
|
||||||
|
"Stop watching": "Прекратить отслеживание",
|
||||||
|
"Email notifications": "Уведомления на email",
|
||||||
|
"Page updates": "Обновления страницы",
|
||||||
|
"Get notified when pages you watch are updated.": "Получайте уведомления, когда отслеживаемые вами страницы обновляются.",
|
||||||
|
"Page mentions": "Упоминания на странице",
|
||||||
|
"Get notified when someone mentions you on a page.": "Получайте уведомления, когда кто-то упоминает вас на странице.",
|
||||||
|
"Comment mentions": "Упоминания в комментариях",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Получайте уведомления, когда кто-то упоминает вас в комментарии.",
|
||||||
|
"New comments": "Новые комментарии",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Получайте уведомления о новых комментариях в цепочках, в которых вы участвуете.",
|
||||||
|
"Resolved comments": "Разрешённые комментарии",
|
||||||
|
"Get notified when your comment is resolved.": "Получайте уведомление, когда ваш комментарий разрешён.",
|
||||||
|
"You are now watching this page": "Вы теперь следите за этой страницей",
|
||||||
|
"You are no longer watching this page": "Вы больше не следите за этой страницей",
|
||||||
|
"Direct": "Прямые",
|
||||||
|
"Updates": "Обновления",
|
||||||
|
"Today": "Сегодня",
|
||||||
|
"Yesterday": "Вчера",
|
||||||
|
"This week": "На этой неделе",
|
||||||
|
"Older": "Старше",
|
||||||
|
"Restricted page": "Страница с ограниченным доступом",
|
||||||
|
"Restricted pages cannot be shared publicly.": "Страницы с ограниченным доступом нельзя сделать общедоступными.",
|
||||||
|
"Restricted by parent": "Ограничено родительской страницей",
|
||||||
|
"Restricted": "Ограничено",
|
||||||
|
"Open": "Открыто",
|
||||||
|
"Inherits restrictions from ancestor page": "Наследует ограничения от родительской страницы",
|
||||||
|
"Only people listed below can access this page": "Доступ к этой странице имеют только перечисленные ниже пользователи",
|
||||||
|
"Everyone in this space can access": "Доступ имеют все участники этого пространства",
|
||||||
|
"No additional restrictions on this page": "На этой странице нет дополнительных ограничений",
|
||||||
|
"Only specific people can access": "Доступ имеют только определённые пользователи",
|
||||||
|
"Use only inherited restrictions": "Использовать только унаследованные ограничения",
|
||||||
|
"Add restrictions on top of inherited": "Добавить ограничения поверх унаследованных",
|
||||||
|
"Inherited restriction": "Унаследованное ограничение",
|
||||||
|
"Access limited by": "Доступ ограничен",
|
||||||
|
"Restrict access to control who can view and edit this page": "Ограничьте доступ, чтобы контролировать, кто может просматривать и редактировать эту страницу",
|
||||||
|
"Add additional restrictions specific to this page": "Добавить дополнительные ограничения, применимые только к этой странице",
|
||||||
|
"Access": "Доступ",
|
||||||
|
"People with access": "Пользователи с доступом",
|
||||||
|
"Remove all": "Удалить всё",
|
||||||
|
"Remove access": "Удалить доступ",
|
||||||
|
"Remove all access": "Удалить весь доступ",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "Вы уверены, что хотите удалить доступ этого участника к странице?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "Вы уверены, что хотите удалить все специальные права доступа? Это сделает страницу доступной всем участникам пространства.",
|
||||||
|
"Trash retention": "Срок хранения корзины",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "Страницы в корзине будут окончательно удалены по истечении этого срока.",
|
||||||
|
"Trash retention updated": "Срок хранения корзины обновлён",
|
||||||
|
"Failed to update trash retention": "Не удалось обновить срок хранения корзины",
|
||||||
|
"Removed page restriction": "Ограничение доступа к странице удалено",
|
||||||
|
"Added page permission": "Добавлено разрешение доступа к странице",
|
||||||
|
"Removed page permission": "Удалено разрешение доступа к странице",
|
||||||
|
"Verifying your email": "Проверка вашей электронной почты",
|
||||||
|
"Please wait...": "Пожалуйста, подождите...",
|
||||||
|
"Verification failed. The link may have expired.": "Ошибка проверки. Ссылка могла устареть.",
|
||||||
|
"Check your email": "Проверьте вашу электронную почту",
|
||||||
|
"We sent a verification link to {{email}}.": "Мы отправили ссылку для подтверждения на {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Мы отправили ссылку для подтверждения на вашу электронную почту.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Перейдите по ссылке, чтобы подтвердить электронную почту и получить доступ к рабочему пространству.",
|
||||||
|
"Resend verification email": "Отправить письмо для подтверждения повторно",
|
||||||
|
"Verification email sent. Please check your inbox.": "Письмо для подтверждения отправлено. Пожалуйста, проверьте ваш почтовый ящик.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Не удалось отправить письмо для подтверждения. Пожалуйста, попробуйте снова.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Мы отправили вам электронное письмо с привязанными рабочими пространствами.",
|
||||||
|
"Load more": "Загрузить ещё",
|
||||||
|
"Log out of all devices": "Выйти со всех устройств",
|
||||||
|
"Log out of all sessions except this device": "Выйти из всех сессий, кроме этого устройства",
|
||||||
|
"This Device": "Это устройство",
|
||||||
|
"Unknown device": "Неизвестное устройство",
|
||||||
|
"No active sessions": "Нет активных сессий",
|
||||||
|
"Session revoked": "Сессия отозвана",
|
||||||
|
"All other sessions revoked": "Все другие сессии отозваны",
|
||||||
|
"Last used": "Последнее использование",
|
||||||
|
"Created": "Создано",
|
||||||
|
"Rename": "Переименовать",
|
||||||
|
"Publish": "Опубликовать",
|
||||||
|
"Security": "Безопасность",
|
||||||
|
"Enforce SSO": "Принудительно использовать SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "После включения участники не смогут войти с помощью электронной почты и пароля."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -116,6 +116,7 @@
|
|||||||
"No group found": "Групу не знайдено",
|
"No group found": "Групу не знайдено",
|
||||||
"No page history saved yet.": "Історія сторінок ще не збережена.",
|
"No page history saved yet.": "Історія сторінок ще не збережена.",
|
||||||
"No pages yet": "Сторінок поки немає",
|
"No pages yet": "Сторінок поки немає",
|
||||||
|
"No shared pages": "Немає спільних сторінок",
|
||||||
"No results found...": "Результати не знайдено...",
|
"No results found...": "Результати не знайдено...",
|
||||||
"No user found": "Користувача не знайдено",
|
"No user found": "Користувача не знайдено",
|
||||||
"Overview": "Огляд",
|
"Overview": "Огляд",
|
||||||
@@ -123,11 +124,14 @@
|
|||||||
"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": "сторінки",
|
||||||
"Password": "Пароль",
|
"Password": "Пароль",
|
||||||
"Password changed successfully": "Пароль успішно змінено",
|
"Password changed successfully": "Пароль успішно змінено",
|
||||||
|
"People": "Користувачі",
|
||||||
"Pending": "В очікуванні",
|
"Pending": "В очікуванні",
|
||||||
"Please confirm your action": "Будь ласка, підтвердіть вашу дію",
|
"Please confirm your action": "Будь ласка, підтвердіть вашу дію",
|
||||||
"Preferences": "Налаштування",
|
"Preferences": "Налаштування",
|
||||||
@@ -205,6 +209,9 @@
|
|||||||
"Reply...": "Відповісти...",
|
"Reply...": "Відповісти...",
|
||||||
"Error loading comments.": "Помилка при завантаженні коментарів.",
|
"Error loading comments.": "Помилка при завантаженні коментарів.",
|
||||||
"No comments yet.": "Коментарів поки немає.",
|
"No comments yet.": "Коментарів поки немає.",
|
||||||
|
"No open comments.": "Немає відкритих коментарів.",
|
||||||
|
"No resolved comments.": "Немає вирішених коментарів.",
|
||||||
|
"Add a comment...": "Додати коментар...",
|
||||||
"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?": "Ви впевнені, що хочете видалити цей коментар?",
|
||||||
@@ -226,7 +233,6 @@
|
|||||||
"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.": "Немає вирішених коментарів.",
|
|
||||||
"Revoke invitation": "Відкликати запрошення",
|
"Revoke invitation": "Відкликати запрошення",
|
||||||
"Revoke": "Відкликати",
|
"Revoke": "Відкликати",
|
||||||
"Don't": "Ні",
|
"Don't": "Ні",
|
||||||
@@ -272,6 +278,7 @@
|
|||||||
"Add row below": "Додати рядок нижче",
|
"Add row below": "Додати рядок нижче",
|
||||||
"Delete table": "Видалити таблицю",
|
"Delete table": "Видалити таблицю",
|
||||||
"Info": "Інформація",
|
"Info": "Інформація",
|
||||||
|
"Note": "Примітка",
|
||||||
"Success": "Успішно",
|
"Success": "Успішно",
|
||||||
"Warning": "Попередження",
|
"Warning": "Попередження",
|
||||||
"Danger": "Важливо",
|
"Danger": "Важливо",
|
||||||
@@ -282,6 +289,11 @@
|
|||||||
"Save & Exit": "Зберегти та вийти",
|
"Save & Exit": "Зберегти та вийти",
|
||||||
"Double-click to edit Excalidraw diagram": "Клацніть двічі для редагування діаграми Excalidraw",
|
"Double-click to edit Excalidraw diagram": "Клацніть двічі для редагування діаграми Excalidraw",
|
||||||
"Paste link": "Вставити посилання",
|
"Paste link": "Вставити посилання",
|
||||||
|
"Paste link or search pages": "Вставте посилання або знайдіть сторінки",
|
||||||
|
"Link to web page": "Посилання на веб-сторінку",
|
||||||
|
"Recents": "Нещодавні",
|
||||||
|
"Page or URL": "Сторінка або URL",
|
||||||
|
"Link title": "Назва посилання",
|
||||||
"Edit link": "Редагувати посилання",
|
"Edit link": "Редагувати посилання",
|
||||||
"Remove link": "Видалити посилання",
|
"Remove link": "Видалити посилання",
|
||||||
"Add link": "Додати посилання",
|
"Add link": "Додати посилання",
|
||||||
@@ -329,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Вставити горизонтальний роздільник",
|
"Insert horizontal rule divider": "Вставити горизонтальний роздільник",
|
||||||
"Upload any image from your device.": "Завантажити будь-яке зображення з вашого пристрою.",
|
"Upload any image from your device.": "Завантажити будь-яке зображення з вашого пристрою.",
|
||||||
"Upload any video from your device.": "Завантажити будь-яке відео з вашого пристрою.",
|
"Upload any video from your device.": "Завантажити будь-яке відео з вашого пристрою.",
|
||||||
|
"Upload any audio from your device.": "Завантажте будь-який аудіофайл зі свого пристрою.",
|
||||||
"Upload any file from your device.": "Завантажити будь-який файл з вашого пристрою.",
|
"Upload any file from your device.": "Завантажити будь-який файл з вашого пристрою.",
|
||||||
"Uploading {{name}}": "Завантаження {{name}}",
|
"Uploading {{name}}": "Завантаження {{name}}",
|
||||||
"Uploading file": "Завантаження файлу",
|
"Uploading file": "Завантаження файлу",
|
||||||
@@ -339,6 +352,12 @@
|
|||||||
"Divider": "Роздільник",
|
"Divider": "Роздільник",
|
||||||
"Quote": "Цитата",
|
"Quote": "Цитата",
|
||||||
"Image": "Зображення",
|
"Image": "Зображення",
|
||||||
|
"Audio": "Аудіо.",
|
||||||
|
"Embed PDF": "Вбудувати PDF",
|
||||||
|
"Upload and embed a PDF file.": "Завантажте та вбудуйте файл PDF.",
|
||||||
|
"Embed as PDF": "Вбудувати як PDF",
|
||||||
|
"Failed to load PDF": "Не вдалося завантажити PDF",
|
||||||
|
"Convert to attachment": "Перетворити на вкладення",
|
||||||
"File attachment": "Прикріплений файл",
|
"File attachment": "Прикріплений файл",
|
||||||
"Toggle block": "Блок, що згортається",
|
"Toggle block": "Блок, що згортається",
|
||||||
"Callout": "Виноска",
|
"Callout": "Виноска",
|
||||||
@@ -353,9 +372,23 @@
|
|||||||
"Insert current date": "Вставити поточну дату",
|
"Insert current date": "Вставити поточну дату",
|
||||||
"Draw and sketch excalidraw diagrams": "Вставити та малювати діаграми Excalidraw",
|
"Draw and sketch excalidraw diagrams": "Вставити та малювати діаграми Excalidraw",
|
||||||
"Multiple": "Декілька",
|
"Multiple": "Декілька",
|
||||||
|
"Turn into": "Перетворити",
|
||||||
|
"Text align": "Вирівнювання тексту",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "Цю сторінку могли видалити, перемістити або у вас може не бути до неї доступу.",
|
||||||
|
"Go to homepage": "Перейти на головну",
|
||||||
|
"Pages you create will show up here.": "Сторінки, які ви створите, з'являться тут.",
|
||||||
"Heading {{level}}": "Заголовок {{level}}",
|
"Heading {{level}}": "Заголовок {{level}}",
|
||||||
"Toggle title": "Перемкнути заголовок",
|
"Toggle title": "Перемкнути заголовок",
|
||||||
"Write anything. Enter \"/\" for commands": "Почніть писати. Введіть \"/\" для списку команд",
|
"Write anything. Enter \"/\" for commands": "Почніть писати. Введіть \"/\" для списку команд",
|
||||||
|
"Write...": "Напишіть...",
|
||||||
|
"Column count": "Кількість колонок",
|
||||||
|
"{{count}} Columns": "{count, plural, one{# колонка} few{# колонки} many{# колонок} other{# колонки}}",
|
||||||
|
"Equal columns": "Рівні колонки",
|
||||||
|
"Left sidebar": "Ліва бічна панель",
|
||||||
|
"Right sidebar": "Права бічна панель",
|
||||||
|
"Wide center": "Широка центральна колонка",
|
||||||
|
"Left wide": "Широка ліва колонка",
|
||||||
|
"Right wide": "Широка права колонка",
|
||||||
"Names do not match": "Назви не співпадають",
|
"Names do not match": "Назви не співпадають",
|
||||||
"Today, {{time}}": "Сьогодні, {{time}}",
|
"Today, {{time}}": "Сьогодні, {{time}}",
|
||||||
"Yesterday, {{time}}": "Вчора, {{time}}",
|
"Yesterday, {{time}}": "Вчора, {{time}}",
|
||||||
@@ -378,6 +411,13 @@
|
|||||||
"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.": "Ви впевнені, що хочете видалити цього учасника робочої області? Ця дія незворотна.",
|
||||||
|
"Deactivate member": "Деактивувати учасника",
|
||||||
|
"Activate member": "Активувати учасника",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "Ви впевнені, що хочете деактивувати цього учасника робочого простору? Вони більше не зможуть отримати доступ до цього робочого простору.",
|
||||||
|
"Are you sure you want to activate this workspace member?": "Ви впевнені, що хочете активувати цього учасника робочого простору?",
|
||||||
|
"Deactivate": "Деактивувати",
|
||||||
|
"Activate": "Активувати",
|
||||||
|
"Deactivated": "Деактивовано",
|
||||||
"Move": "Перемістити",
|
"Move": "Перемістити",
|
||||||
"Move page": "Перемістити сторінку",
|
"Move page": "Перемістити сторінку",
|
||||||
"Move page to a different space.": "Перемістити сторінку в інший простір.",
|
"Move page to a different space.": "Перемістити сторінку в інший простір.",
|
||||||
@@ -405,6 +445,25 @@
|
|||||||
"Share deleted successfully": "Спільний доступ успішно видалено",
|
"Share deleted successfully": "Спільний доступ успішно видалено",
|
||||||
"Share not found": "Спільний доступ не знайдено",
|
"Share not found": "Спільний доступ не знайдено",
|
||||||
"Failed to share page": "Не вдалося поділитися сторінкою",
|
"Failed to share page": "Не вдалося поділитися сторінкою",
|
||||||
|
"Disable public sharing": "Вимкнути публічний доступ",
|
||||||
|
"Prevent members from sharing pages publicly.": "Перешкодити учасникам публічно ділитися сторінками.",
|
||||||
|
"Toggle public sharing": "Перемикання публічного доступу",
|
||||||
|
"Toggle space public sharing": "Перемикання публічного доступу до просторів",
|
||||||
|
"Allow viewers to comment": "Дозволити глядачам коментувати",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Дозволити глядачам додавати коментарі на сторінках у цьому просторі.",
|
||||||
|
"Toggle viewer comments": "Увімкнути або вимкнути коментарі глядачів",
|
||||||
|
"Public sharing is disabled at the workspace level": "Публічний доступ вимкнуто на рівні робочого простору",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "Перешкодити публічному поширенню сторінок у цьому просторі.",
|
||||||
|
"Page permissions": "Права доступу до сторінки.",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "Керуйте тим, хто може переглядати та редагувати окремі сторінки. Доступно з корпоративною ліцензією.",
|
||||||
|
"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 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 disable public sharing? All existing shared links in this space will be deleted.": "Ви впевнені, що хочете вимкнути публічний доступ? Усі існуючі посилання для спільного доступу в цьому просторі будуть видалені.",
|
||||||
|
"Public sharing is disabled": "Публічний доступ вимкнуто",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "Публічний доступ було вимкнено на рівні робочого простору.",
|
||||||
|
"Public sharing has been disabled for this space.": "Публічний доступ було вимкнено для цього простору.",
|
||||||
"Copy page": "Копіювати сторінки",
|
"Copy page": "Копіювати сторінки",
|
||||||
"Copy page to a different space.": "Скопіювати сторінку в інший простір.",
|
"Copy page to a different space.": "Скопіювати сторінку в інший простір.",
|
||||||
"Page copied successfully": "Сторінку успішно скопійовано",
|
"Page copied successfully": "Сторінку успішно скопійовано",
|
||||||
@@ -487,7 +546,7 @@
|
|||||||
"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.": "Введіть один з ваших резервних кодів. Кожен резервний код можна використовувати лише один раз.",
|
||||||
"Verify": "Перевірити",
|
"Verify": "Перевірити",
|
||||||
"Trash": "Кошик",
|
"Trash": "Кошик",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "Сторінки у кошику будуть остаточно видалені через 30 днів.",
|
"Pages in trash will be permanently deleted after {{count}} days.": "Сторінки в кошику будуть остаточно видалені через {count, plural, one{# день} few{# дні} many{# днів} other{# дня}}.",
|
||||||
"Deleted": "Видалено",
|
"Deleted": "Видалено",
|
||||||
"No pages in trash": "Немає сторінок у кошику",
|
"No pages in trash": "Немає сторінок у кошику",
|
||||||
"Permanently delete page?": "Остаточно видалити сторінку?",
|
"Permanently delete page?": "Остаточно видалити сторінку?",
|
||||||
@@ -559,19 +618,138 @@
|
|||||||
"This action cannot be undone. Any applications using this API key will stop working.": "Цю дію не можна скасувати. Будь-які додатки, що використовують цей ключ API, перестануть працювати.",
|
"This action cannot be undone. Any applications using this API key will stop working.": "Цю дію не можна скасувати. Будь-які додатки, що використовують цей ключ API, перестануть працювати.",
|
||||||
"Update API key": "Оновити ключ API",
|
"Update API key": "Оновити ключ API",
|
||||||
"Manage API keys for all users in the workspace": "Керувати ключами API для всіх користувачів у робочій області",
|
"Manage API keys for all users in the workspace": "Керувати ключами API для всіх користувачів у робочій області",
|
||||||
|
"Restrict API key creation to admins": "Обмежити створення API-ключів лише для адміністраторів",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "Тільки адміністратори та власники можуть створювати нові API-ключі. Існуючі ключі учасників і надалі працюватимуть.",
|
||||||
|
"Toggle restrict API keys to admins": "Увімкнути або вимкнути обмеження створення API-ключів лише для адміністраторів",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "Створення API-ключів дозволено лише адміністраторам за налаштуванням адміністратора робочого простору.",
|
||||||
"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 Answers": "Відповіді ШІ",
|
||||||
|
"AI-powered search (AI Answers)": "Пошук на базі ШІ (Відповіді ШІ)",
|
||||||
"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": "Переключити пошук з ШІ",
|
||||||
|
"Generative AI (Ask AI)": "Генеративний ШІ (Запитати ШІ)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "Увімкнути генерацію контенту за допомогою ШІ в редакторі. Дозволяє користувачам генерувати, покращувати, перекладати та трансформувати текст.",
|
||||||
|
"Toggle generative AI": "Переключити генеративний ШІ",
|
||||||
|
"Upgrade your plan": "Оновіть свій тарифний план",
|
||||||
|
"Available with a paid license": "Доступно за платною ліцензією",
|
||||||
|
"Upgrade your license tier.": "Оновіть рівень своєї ліцензії.",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "ШІ доступний лише в корпоративній редакції Docmost. Зверніться до sales@docmost.com.",
|
||||||
|
"AI & MCP": "ШІ та MCP",
|
||||||
|
"AI": "ШІ",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "Протокол контексту моделі (MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "Увімкніть MCP‑сервер, щоб дозволити ШІ‑помічникам та інструментам взаємодіяти з вмістом вашого робочого простору.",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP доступний лише в корпоративній редакції Docmost. Зверніться до sales@docmost.com.",
|
||||||
|
"MCP Server URL": "URL сервера MCP",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "Використовуйте свій API‑ключ для аутентифікації. Ви можете керувати API‑ключами в налаштуваннях облікового запису.",
|
||||||
|
"Supported tools": "Підтримувані інструменти",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "У вашому робочому просторі MCP увімкнено. Використайте свій API‑ключ, щоб підключити ШІ‑помічників.",
|
||||||
|
"MCP server URL:": "URL сервера MCP:",
|
||||||
|
"Learn more": "Дізнатися більше",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "Керуйте ключами API для всіх користувачів у робочому просторі. Перегляньте <anchor>документацію API</anchor> для деталей використання.",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "Перегляньте <anchor>документацію API</anchor> для деталей використання.",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "Перегляньте <anchor>документацію MCP</anchor>.",
|
||||||
"Sources": "Джерела",
|
"Sources": "Джерела",
|
||||||
"Ask AI not available for attachments": "Запитати ШІ недоступно для вкладень",
|
"AI Answers not available for attachments": "Відповіді ШІ недоступні для вкладень",
|
||||||
"No answer available": "Відповідь недоступна",
|
"No answer available": "Відповідь недоступна",
|
||||||
"Background color": "Колір фону",
|
"Background color": "Колір фону",
|
||||||
"Highlight color": "Колір підсвічування",
|
"Highlight color": "Колір підсвічування",
|
||||||
"Remove color": "Видалити колір"
|
"Remove color": "Видалити колір",
|
||||||
|
"Notifications": "Сповіщення",
|
||||||
|
"No notifications": "Немає сповіщень",
|
||||||
|
"No unread notifications": "Немає непрочитаних сповіщень",
|
||||||
|
"All notifications": "Усі сповіщення",
|
||||||
|
"Unread only": "Тільки непрочитані",
|
||||||
|
"Mark all as read": "Позначити все як прочитане",
|
||||||
|
"Mark as read": "Позначити як прочитане",
|
||||||
|
"More options": "Більше опцій",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> згадав вас у коментарі",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> залишив коментар на сторінці",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> вирішив коментар",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> згадав вас на сторінці",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> надав вам доступ до редагування сторінки",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> надав вам доступ до перегляду сторінки",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> оновив сторінку.",
|
||||||
|
"Watch page": "Стежити за сторінкою",
|
||||||
|
"Stop watching": "Припинити стежити",
|
||||||
|
"Email notifications": "Сповіщення електронною поштою",
|
||||||
|
"Page updates": "Оновлення сторінки",
|
||||||
|
"Get notified when pages you watch are updated.": "Отримуйте сповіщення, коли сторінки, за якими ви стежите, оновлюються.",
|
||||||
|
"Page mentions": "Згадки на сторінці",
|
||||||
|
"Get notified when someone mentions you on a page.": "Отримуйте сповіщення, коли хтось згадує вас на сторінці.",
|
||||||
|
"Comment mentions": "Згадки у коментарях",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Отримуйте сповіщення, коли хтось згадує вас у коментарі.",
|
||||||
|
"New comments": "Нові коментарі",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Отримуйте сповіщення про нові коментарі у темах, у яких ви берете участь.",
|
||||||
|
"Resolved comments": "Вирішені коментарі",
|
||||||
|
"Get notified when your comment is resolved.": "Отримайте сповіщення, коли ваш коментар вирішено.",
|
||||||
|
"You are now watching this page": "Ви зараз стежите за цією сторінкою",
|
||||||
|
"You are no longer watching this page": "Ви більше не стежите за цією сторінкою",
|
||||||
|
"Direct": "Прямі",
|
||||||
|
"Updates": "Оновлення",
|
||||||
|
"Today": "Сьогодні",
|
||||||
|
"Yesterday": "Вчора",
|
||||||
|
"This week": "Цього тижня",
|
||||||
|
"Older": "Старіші",
|
||||||
|
"Restricted page": "Сторінка з обмеженим доступом",
|
||||||
|
"Restricted pages cannot be shared publicly.": "Сторінки з обмеженим доступом не можна робити загальнодоступними.",
|
||||||
|
"Restricted by parent": "Обмежено батьківською сторінкою",
|
||||||
|
"Restricted": "Обмежено",
|
||||||
|
"Open": "Відкрита",
|
||||||
|
"Inherits restrictions from ancestor page": "Наслідує обмеження від батьківської сторінки",
|
||||||
|
"Only people listed below can access this page": "Доступ до цієї сторінки мають лише люди, вказані нижче.",
|
||||||
|
"Everyone in this space can access": "Усі в цьому просторі мають доступ",
|
||||||
|
"No additional restrictions on this page": "Додаткових обмежень на цій сторінці немає.",
|
||||||
|
"Only specific people can access": "Доступ мають лише конкретні особи.",
|
||||||
|
"Use only inherited restrictions": "Використовувати лише успадковані обмеження",
|
||||||
|
"Add restrictions on top of inherited": "Додати обмеження поверх успадкованих",
|
||||||
|
"Inherited restriction": "Успадковане обмеження",
|
||||||
|
"Access limited by": "Доступ обмежено через",
|
||||||
|
"Restrict access to control who can view and edit this page": "Обмежте доступ, щоб контролювати, хто може переглядати та редагувати цю сторінку.",
|
||||||
|
"Add additional restrictions specific to this page": "Додати додаткові обмеження для цієї сторінки.",
|
||||||
|
"Access": "Доступ",
|
||||||
|
"People with access": "Особи з доступом",
|
||||||
|
"Remove all": "Видалити все",
|
||||||
|
"Remove access": "Видалити доступ",
|
||||||
|
"Remove all access": "Видалити весь доступ",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "Ви впевнені, що хочете видалити доступ цього учасника до сторінки?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "Ви впевнені, що хочете видалити всі індивідуальні дозволи доступу? Це зробить сторінку доступною для всіх у просторі.",
|
||||||
|
"Trash retention": "Термін зберігання у кошику",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "Сторінки в кошику будуть остаточно видалені після цього періоду.",
|
||||||
|
"Trash retention updated": "Термін зберігання у кошику оновлено",
|
||||||
|
"Failed to update trash retention": "Не вдалося оновити термін зберігання у кошику",
|
||||||
|
"Removed page restriction": "Обмеження сторінки видалено",
|
||||||
|
"Added page permission": "Додано дозвіл на сторінку",
|
||||||
|
"Removed page permission": "Дозвіл на сторінку видалено",
|
||||||
|
"Verifying your email": "Підтвердження вашої електронної пошти",
|
||||||
|
"Please wait...": "Будь ласка, зачекайте...",
|
||||||
|
"Verification failed. The link may have expired.": "Підтвердження не вдалося. Посилання могло втратити чинність.",
|
||||||
|
"Check your email": "Перевірте свою електронну пошту",
|
||||||
|
"We sent a verification link to {{email}}.": "Ми надіслали посилання для підтвердження на {{email}}.",
|
||||||
|
"We sent a verification link to your email.": "Ми надіслали посилання для підтвердження на вашу електронну пошту.",
|
||||||
|
"Click the link to verify your email and access your workspace.": "Клікніть на посилання, щоб підтвердити електронну пошту та отримати доступ до робочого простору.",
|
||||||
|
"Resend verification email": "Повторно надіслати лист для підтвердження",
|
||||||
|
"Verification email sent. Please check your inbox.": "Лист для підтвердження надіслано. Будь ласка, перевірте свою скриньку.",
|
||||||
|
"Failed to resend verification email. Please try again.": "Не вдалося повторно надіслати лист для підтвердження. Будь ласка, спробуйте ще раз.",
|
||||||
|
"We've sent you an email with your associated workspaces.": "Ми надіслали вам лист із переліком пов’язаних робочих просторів.",
|
||||||
|
"Load more": "Завантажити ще",
|
||||||
|
"Log out of all devices": "Вийти з усіх пристроїв",
|
||||||
|
"Log out of all sessions except this device": "Вийти з усіх сесій, окрім цього пристрою",
|
||||||
|
"This Device": "Цей пристрій",
|
||||||
|
"Unknown device": "Невідомий пристрій",
|
||||||
|
"No active sessions": "Немає активних сесій",
|
||||||
|
"Session revoked": "Сесію скасовано",
|
||||||
|
"All other sessions revoked": "Всі інші сесії скасовано",
|
||||||
|
"Last used": "Останнє використання",
|
||||||
|
"Created": "Створено",
|
||||||
|
"Rename": "Перейменувати",
|
||||||
|
"Publish": "Опублікувати",
|
||||||
|
"Security": "Безпека",
|
||||||
|
"Enforce SSO": "Вимагати SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "Після активування учасники не зможуть увійти за допомогою електронної пошти та паролю."
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -116,6 +116,7 @@
|
|||||||
"No group found": "未找到群组",
|
"No group found": "未找到群组",
|
||||||
"No page history saved yet.": "尚未保存页面历史。",
|
"No page history saved yet.": "尚未保存页面历史。",
|
||||||
"No pages yet": "暂无页面",
|
"No pages yet": "暂无页面",
|
||||||
|
"No shared pages": "没有共享页面",
|
||||||
"No results found...": "未找到结果...",
|
"No results found...": "未找到结果...",
|
||||||
"No user found": "未找到用户",
|
"No user found": "未找到用户",
|
||||||
"Overview": "概览",
|
"Overview": "概览",
|
||||||
@@ -123,11 +124,14 @@
|
|||||||
"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": "个页面",
|
||||||
"Password": "密码",
|
"Password": "密码",
|
||||||
"Password changed successfully": "密码更改成功",
|
"Password changed successfully": "密码更改成功",
|
||||||
|
"People": "人员",
|
||||||
"Pending": "待定",
|
"Pending": "待定",
|
||||||
"Please confirm your action": "请确认您的操作",
|
"Please confirm your action": "请确认您的操作",
|
||||||
"Preferences": "偏好设置",
|
"Preferences": "偏好设置",
|
||||||
@@ -205,6 +209,9 @@
|
|||||||
"Reply...": "回复...",
|
"Reply...": "回复...",
|
||||||
"Error loading comments.": "加载评论时出错",
|
"Error loading comments.": "加载评论时出错",
|
||||||
"No comments yet.": "目前还没有评论",
|
"No comments yet.": "目前还没有评论",
|
||||||
|
"No open comments.": "没有未解决的评论。",
|
||||||
|
"No resolved comments.": "没有已解决的评论。",
|
||||||
|
"Add a comment...": "添加评论...",
|
||||||
"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?": "你确定要删除这条评论吗?",
|
||||||
@@ -226,7 +233,6 @@
|
|||||||
"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.": "没有已解决的评论。",
|
|
||||||
"Revoke invitation": "撤回邀请",
|
"Revoke invitation": "撤回邀请",
|
||||||
"Revoke": "撤销",
|
"Revoke": "撤销",
|
||||||
"Don't": "不要",
|
"Don't": "不要",
|
||||||
@@ -272,6 +278,7 @@
|
|||||||
"Add row below": "在下方插入行",
|
"Add row below": "在下方插入行",
|
||||||
"Delete table": "删除表格",
|
"Delete table": "删除表格",
|
||||||
"Info": "信息",
|
"Info": "信息",
|
||||||
|
"Note": "注意",
|
||||||
"Success": "成功",
|
"Success": "成功",
|
||||||
"Warning": "警告",
|
"Warning": "警告",
|
||||||
"Danger": "危险",
|
"Danger": "危险",
|
||||||
@@ -282,6 +289,11 @@
|
|||||||
"Save & Exit": "保存并退出",
|
"Save & Exit": "保存并退出",
|
||||||
"Double-click to edit Excalidraw diagram": "双击以编辑 Excalidraw 图表",
|
"Double-click to edit Excalidraw diagram": "双击以编辑 Excalidraw 图表",
|
||||||
"Paste link": "粘贴链接",
|
"Paste link": "粘贴链接",
|
||||||
|
"Paste link or search pages": "粘贴链接或搜索页面",
|
||||||
|
"Link to web page": "链接到网页",
|
||||||
|
"Recents": "最近使用",
|
||||||
|
"Page or URL": "页面或网址",
|
||||||
|
"Link title": "链接标题",
|
||||||
"Edit link": "编辑链接",
|
"Edit link": "编辑链接",
|
||||||
"Remove link": "移除链接",
|
"Remove link": "移除链接",
|
||||||
"Add link": "添加链接",
|
"Add link": "添加链接",
|
||||||
@@ -329,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "插入水平分割线",
|
"Insert horizontal rule divider": "插入水平分割线",
|
||||||
"Upload any image from your device.": "从设备上传任何图像",
|
"Upload any image from your device.": "从设备上传任何图像",
|
||||||
"Upload any video from your device.": "从设备上传任何视频",
|
"Upload any video from your device.": "从设备上传任何视频",
|
||||||
|
"Upload any audio from your device.": "从您的设备上传任意音频文件。",
|
||||||
"Upload any file from your device.": "从设备上传任何文件",
|
"Upload any file from your device.": "从设备上传任何文件",
|
||||||
"Uploading {{name}}": "正在上传{{name}}",
|
"Uploading {{name}}": "正在上传{{name}}",
|
||||||
"Uploading file": "正在上传文件",
|
"Uploading file": "正在上传文件",
|
||||||
@@ -339,6 +352,12 @@
|
|||||||
"Divider": "分割线",
|
"Divider": "分割线",
|
||||||
"Quote": "引用",
|
"Quote": "引用",
|
||||||
"Image": "图像",
|
"Image": "图像",
|
||||||
|
"Audio": "音频。",
|
||||||
|
"Embed PDF": "嵌入 PDF",
|
||||||
|
"Upload and embed a PDF file.": "上传并嵌入 PDF 文件。",
|
||||||
|
"Embed as PDF": "作为 PDF 嵌入",
|
||||||
|
"Failed to load PDF": "加载 PDF 失败",
|
||||||
|
"Convert to attachment": "转换为附件",
|
||||||
"File attachment": "文件附件",
|
"File attachment": "文件附件",
|
||||||
"Toggle block": "切换块",
|
"Toggle block": "切换块",
|
||||||
"Callout": "标注块",
|
"Callout": "标注块",
|
||||||
@@ -353,9 +372,23 @@
|
|||||||
"Insert current date": "插入当前日期",
|
"Insert current date": "插入当前日期",
|
||||||
"Draw and sketch excalidraw diagrams": "绘制 Excalidraw 图表",
|
"Draw and sketch excalidraw diagrams": "绘制 Excalidraw 图表",
|
||||||
"Multiple": "多个",
|
"Multiple": "多个",
|
||||||
|
"Turn into": "变成",
|
||||||
|
"Text align": "文本对齐",
|
||||||
|
"This page may have been deleted, moved, or you may not have access.": "此页面可能已被删除、移动,或者您可能无权访问。{",
|
||||||
|
"Go to homepage": "前往首页",
|
||||||
|
"Pages you create will show up here.": "您创建的页面将显示在此处。",
|
||||||
"Heading {{level}}": "{{level}} 级标题",
|
"Heading {{level}}": "{{level}} 级标题",
|
||||||
"Toggle title": "切换标题",
|
"Toggle title": "切换标题",
|
||||||
"Write anything. Enter \"/\" for commands": "开始编写内容,输入 \"/\" 以使用指令",
|
"Write anything. Enter \"/\" for commands": "开始编写内容,输入 \"/\" 以使用指令",
|
||||||
|
"Write...": "写点内容...",
|
||||||
|
"Column count": "列数",
|
||||||
|
"{{count}} Columns": "{{count}} 列",
|
||||||
|
"Equal columns": "等宽列",
|
||||||
|
"Left sidebar": "左侧边栏",
|
||||||
|
"Right sidebar": "右侧边栏",
|
||||||
|
"Wide center": "中间加宽",
|
||||||
|
"Left wide": "左侧加宽",
|
||||||
|
"Right wide": "右侧加宽",
|
||||||
"Names do not match": "名称不匹配",
|
"Names do not match": "名称不匹配",
|
||||||
"Today, {{time}}": "今天,{{time}}",
|
"Today, {{time}}": "今天,{{time}}",
|
||||||
"Yesterday, {{time}}": "昨天,{{time}}",
|
"Yesterday, {{time}}": "昨天,{{time}}",
|
||||||
@@ -378,6 +411,13 @@
|
|||||||
"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.": "您确定要删除此工作区成员吗?此操作不可逆。",
|
||||||
|
"Deactivate member": "停用成员",
|
||||||
|
"Activate member": "激活成员",
|
||||||
|
"Are you sure you want to deactivate this workspace member? They will no longer be able to access this workspace.": "您确定要停用此工作区成员吗?该成员将无法再访问此工作区。",
|
||||||
|
"Are you sure you want to activate this workspace member?": "您确定要激活此工作区成员吗?",
|
||||||
|
"Deactivate": "停用",
|
||||||
|
"Activate": "激活",
|
||||||
|
"Deactivated": "已停用",
|
||||||
"Move": "移动",
|
"Move": "移动",
|
||||||
"Move page": "移动页面",
|
"Move page": "移动页面",
|
||||||
"Move page to a different space.": "将页面移动到不同的空间。",
|
"Move page to a different space.": "将页面移动到不同的空间。",
|
||||||
@@ -405,6 +445,25 @@
|
|||||||
"Share deleted successfully": "分享已成功删除",
|
"Share deleted successfully": "分享已成功删除",
|
||||||
"Share not found": "未找到分享",
|
"Share not found": "未找到分享",
|
||||||
"Failed to share page": "页面分享失败",
|
"Failed to share page": "页面分享失败",
|
||||||
|
"Disable public sharing": "禁用公开分享",
|
||||||
|
"Prevent members from sharing pages publicly.": "阻止成员公开分享页面。",
|
||||||
|
"Toggle public sharing": "切换公开分享",
|
||||||
|
"Toggle space public sharing": "切换空间公开分享",
|
||||||
|
"Allow viewers to comment": "允许观众评论",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "允许观众在此空间的页面上添加评论。",
|
||||||
|
"Toggle viewer comments": "切换观众评论",
|
||||||
|
"Public sharing is disabled at the workspace level": "公开分享在工作区级别被禁用",
|
||||||
|
"Prevent pages in this space from being shared publicly.": "阻止此空间中的页面被公开分享。",
|
||||||
|
"Page permissions": "页面权限},{",
|
||||||
|
"Control who can view and edit individual pages. Available with an enterprise license.": "控制谁可以查看和编辑单个页面。此功能在企业版许可下可用。",
|
||||||
|
"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 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 disable public sharing? All existing shared links in this space will be deleted.": "您确定要禁用公开分享吗?此空间中的所有现有共享链接都将被删除。",
|
||||||
|
"Public sharing is disabled": "公开分享已被禁用",
|
||||||
|
"Public sharing has been disabled at the workspace level.": "公开分享已在工作区级别被禁用。",
|
||||||
|
"Public sharing has been disabled for this space.": "此空间的公开分享已被禁用。",
|
||||||
"Copy page": "复制页面",
|
"Copy page": "复制页面",
|
||||||
"Copy page to a different space.": "将页面复制到不同的空间。",
|
"Copy page to a different space.": "将页面复制到不同的空间。",
|
||||||
"Page copied successfully": "页面复制成功",
|
"Page copied successfully": "页面复制成功",
|
||||||
@@ -487,7 +546,7 @@
|
|||||||
"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.": "输入您的一个备份代码。每个备份代码只能使用一次。",
|
||||||
"Verify": "验证",
|
"Verify": "验证",
|
||||||
"Trash": "垃圾箱",
|
"Trash": "垃圾箱",
|
||||||
"Pages in trash will be permanently deleted after 30 days.": "垃圾箱中的页面将在30天后被永久删除。",
|
"Pages in trash will be permanently deleted after {{count}} days.": "垃圾箱中的页面将在{{count}}天后被永久删除。",
|
||||||
"Deleted": "已删除",
|
"Deleted": "已删除",
|
||||||
"No pages in trash": "垃圾箱中没有页面",
|
"No pages in trash": "垃圾箱中没有页面",
|
||||||
"Permanently delete page?": "永久删除页面?",
|
"Permanently delete page?": "永久删除页面?",
|
||||||
@@ -559,19 +618,138 @@
|
|||||||
"This action cannot be undone. Any applications using this API key will stop working.": "此操作无法撤销。使用此API密钥的任何应用程序将停止工作。",
|
"This action cannot be undone. Any applications using this API key will stop working.": "此操作无法撤销。使用此API密钥的任何应用程序将停止工作。",
|
||||||
"Update API key": "更新API密钥",
|
"Update API key": "更新API密钥",
|
||||||
"Manage API keys for all users in the workspace": "管理工作空间中所有用户的API密钥",
|
"Manage API keys for all users in the workspace": "管理工作空间中所有用户的API密钥",
|
||||||
|
"Restrict API key creation to admins": "仅限管理员创建 API 密钥。",
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.": "只有管理员和所有者可以创建新的 API 密钥。现有成员密钥将继续有效。",
|
||||||
|
"Toggle restrict API keys to admins": "切换仅限管理员创建 API 密钥",
|
||||||
|
"API key creation is restricted to admins by your workspace administrator.": "API 密钥的创建已被您的工作区管理员限制为仅管理员可用。",
|
||||||
"AI settings": "AI设置",
|
"AI settings": "AI设置",
|
||||||
"AI search": "AI搜索",
|
"AI search": "AI搜索",
|
||||||
"AI Answer": "AI回答",
|
"AI Answer": "AI回答",
|
||||||
"Ask AI": "询问AI",
|
"Ask AI": "询问AI",
|
||||||
"AI is thinking...": "AI正在思考...",
|
"AI is thinking...": "AI正在思考...",
|
||||||
"Ask a question...": "提问...",
|
"Ask a question...": "提问...",
|
||||||
"AI-powered search (Ask AI)": "AI驱动的搜索(询问AI)",
|
"AI Answers": "AI答案",
|
||||||
|
"AI-powered search (AI Answers)": "AI驱动的搜索 (AI答案)",
|
||||||
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "AI搜索使用向量嵌入提供跨工作空间内容的语义搜索功能。",
|
"AI search uses vector embeddings to provide semantic search capabilities across your workspace content.": "AI搜索使用向量嵌入提供跨工作空间内容的语义搜索功能。",
|
||||||
"Toggle AI search": "切换AI搜索",
|
"Toggle AI search": "切换AI搜索",
|
||||||
|
"Generative AI (Ask AI)": "生成型AI (询问AI)",
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.": "在编辑器中启用AI驱动的内容生成。允许用户生成、改进、翻译和转换文本。",
|
||||||
|
"Toggle generative AI": "切换生成型AI",
|
||||||
|
"Upgrade your plan": "升级您的方案",
|
||||||
|
"Available with a paid license": "需付费许可才可用",
|
||||||
|
"Upgrade your license tier.": "升级您的许可等级。",
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "AI 仅在 Docmost 企业版中提供。请联系 sales@docmost.com。",
|
||||||
|
"AI & MCP": "AI 与 MCP",
|
||||||
|
"AI": "AI",
|
||||||
|
"MCP": "MCP",
|
||||||
|
"Model Context Protocol (MCP)": "模型上下文协议(MCP)",
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.": "启用 MCP 服务器以允许 AI 助手和工具与您的工作区内容交互。",
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.": "MCP 仅在 Docmost 企业版中提供。请联系 sales@docmost.com。",
|
||||||
|
"MCP Server URL": "MCP 服务器 URL",
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.": "使用您的 API 密钥进行身份验证。您可以在账户设置中管理 API 密钥。",
|
||||||
|
"Supported tools": "支持的工具",
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.": "您的工作区已启用 MCP。使用您的 API 密钥连接 AI 助手。",
|
||||||
|
"MCP server URL:": "MCP 服务器 URL:",
|
||||||
|
"Learn more": "了解更多",
|
||||||
|
"Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details.": "为工作区内所有用户管理 API 密钥。有关使用详情,请查阅<anchor>API 文档</anchor>。",
|
||||||
|
"View the <anchor>API documentation</anchor> for usage details.": "有关使用详情,请查阅<anchor>API 文档</anchor>。",
|
||||||
|
"View the <anchor>MCP documentation</anchor>.": "查看<anchor>MCP 文档</anchor>。",
|
||||||
"Sources": "来源",
|
"Sources": "来源",
|
||||||
"Ask AI not available for attachments": "附件不支持询问AI",
|
"AI Answers not available for attachments": "AI答案不适用于附件",
|
||||||
"No answer available": "无可用答案",
|
"No answer available": "无可用答案",
|
||||||
"Background color": "背景颜色",
|
"Background color": "背景颜色",
|
||||||
"Highlight color": "突出显示颜色",
|
"Highlight color": "突出显示颜色",
|
||||||
"Remove color": "移除颜色"
|
"Remove color": "移除颜色",
|
||||||
|
"Notifications": "通知",
|
||||||
|
"No notifications": "没有通知",
|
||||||
|
"No unread notifications": "没有未读通知",
|
||||||
|
"All notifications": "所有通知",
|
||||||
|
"Unread only": "仅未读",
|
||||||
|
"Mark all as read": "标记所有为已读",
|
||||||
|
"Mark as read": "标记为已读",
|
||||||
|
"More options": "更多选项",
|
||||||
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold>在评论中提到你",
|
||||||
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold>在页面上评论了",
|
||||||
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold>已解决一条评论",
|
||||||
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>在页面上提到你",
|
||||||
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>授予你页面编辑权限",
|
||||||
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>授予你页面查看权限",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold>更新了一个页面。",
|
||||||
|
"Watch page": "关注页面",
|
||||||
|
"Stop watching": "取消关注",
|
||||||
|
"Email notifications": "邮件通知",
|
||||||
|
"Page updates": "页面更新",
|
||||||
|
"Get notified when pages you watch are updated.": "当你关注的页面有更新时收到通知。",
|
||||||
|
"Page mentions": "页面提及",
|
||||||
|
"Get notified when someone mentions you on a page.": "当有人在页面上提到你时收到通知。",
|
||||||
|
"Comment mentions": "评论提及",
|
||||||
|
"Get notified when someone mentions you in a comment.": "当有人在评论中提到你时收到通知。",
|
||||||
|
"New comments": "新评论",
|
||||||
|
"Get notified about new comments on threads you participate in.": "当你参与的讨论有新评论时收到通知。",
|
||||||
|
"Resolved comments": "已解决的评论",
|
||||||
|
"Get notified when your comment is resolved.": "当你的评论被解决时收到通知。",
|
||||||
|
"You are now watching this page": "你现在正在关注此页面",
|
||||||
|
"You are no longer watching this page": "你已取消关注此页面",
|
||||||
|
"Direct": "直接",
|
||||||
|
"Updates": "更新",
|
||||||
|
"Today": "今天",
|
||||||
|
"Yesterday": "昨天",
|
||||||
|
"This week": "本周",
|
||||||
|
"Older": "较早",
|
||||||
|
"Restricted page": "受限页面",
|
||||||
|
"Restricted pages cannot be shared publicly.": "受限页面不能公开共享。",
|
||||||
|
"Restricted by parent": "受父页面限制",
|
||||||
|
"Restricted": "受限",
|
||||||
|
"Open": "公开",
|
||||||
|
"Inherits restrictions from ancestor page": "继承自上级页面的限制",
|
||||||
|
"Only people listed below can access this page": "只有下面列出的人可以访问此页面",
|
||||||
|
"Everyone in this space can access": "此空间中的所有人均可访问",
|
||||||
|
"No additional restrictions on this page": "此页面无额外限制",
|
||||||
|
"Only specific people can access": "仅特定人员可访问",
|
||||||
|
"Use only inherited restrictions": "仅使用继承的限制",
|
||||||
|
"Add restrictions on top of inherited": "在继承的限制之上添加限制",
|
||||||
|
"Inherited restriction": "继承的限制",
|
||||||
|
"Access limited by": "访问受限于",
|
||||||
|
"Restrict access to control who can view and edit this page": "限制访问以控制谁可以查看和编辑此页面",
|
||||||
|
"Add additional restrictions specific to this page": "为此页面添加额外的特定限制",
|
||||||
|
"Access": "访问",
|
||||||
|
"People with access": "有访问权限的人员",
|
||||||
|
"Remove all": "全部移除",
|
||||||
|
"Remove access": "移除访问权限",
|
||||||
|
"Remove all access": "移除所有访问权限",
|
||||||
|
"Are you sure you want to remove this member's access to the page?": "您确定要移除此成员对该页面的访问权限吗?",
|
||||||
|
"Are you sure you want to remove all specific access? This will make the page open to everyone in the space.": "您确定要删除所有特定访问权限吗?这将使该页面对该空间中的所有人开放。",
|
||||||
|
"Trash retention": "垃圾箱保留期",
|
||||||
|
"Pages in trash will be permanently deleted after this period.": "该期限结束后,垃圾箱中的页面将被永久删除。",
|
||||||
|
"Trash retention updated": "垃圾箱保留期已更新",
|
||||||
|
"Failed to update trash retention": "更新垃圾箱保留期失败",
|
||||||
|
"Removed page restriction": "已移除页面限制",
|
||||||
|
"Added page permission": "已添加页面权限",
|
||||||
|
"Removed page permission": "已移除页面权限",
|
||||||
|
"Verifying your email": "正在验证您的邮箱",
|
||||||
|
"Please wait...": "请稍候……",
|
||||||
|
"Verification failed. The link may have expired.": "验证失败。该链接可能已过期。",
|
||||||
|
"Check your email": "查看您的邮箱",
|
||||||
|
"We sent a verification link to {{email}}.": "我们已向{{email}}发送了一封验证邮件。",
|
||||||
|
"We sent a verification link to your email.": "我们已向您的邮箱发送了一封验证邮件。",
|
||||||
|
"Click the link to verify your email and access your workspace.": "请点击链接以验证邮箱并访问您的工作区。",
|
||||||
|
"Resend verification email": "重新发送验证邮件",
|
||||||
|
"Verification email sent. Please check your inbox.": "验证邮件已发送。请检查您的收件箱。",
|
||||||
|
"Failed to resend verification email. Please try again.": "重新发送验证邮件失败。请重试。",
|
||||||
|
"We've sent you an email with your associated workspaces.": "我们已向您发送包含关联工作区的邮件。",
|
||||||
|
"Load more": "加载更多",
|
||||||
|
"Log out of all devices": "退出所有设备登录",
|
||||||
|
"Log out of all sessions except this device": "除本设备外,退出所有会话",
|
||||||
|
"This Device": "本设备",
|
||||||
|
"Unknown device": "未知设备",
|
||||||
|
"No active sessions": "无活动会话",
|
||||||
|
"Session revoked": "会话已被撤销",
|
||||||
|
"All other sessions revoked": "所有其他会话已被撤销",
|
||||||
|
"Last used": "上次使用",
|
||||||
|
"Created": "创建时间",
|
||||||
|
"Rename": "重命名",
|
||||||
|
"Publish": "发布",
|
||||||
|
"Security": "安全性",
|
||||||
|
"Enforce SSO": "强制启用 SSO",
|
||||||
|
"Once enforced, members will not be able to login with email and password.": "一旦强制,成员将无法用邮箱和密码登录。"
|
||||||
}
|
}
|
||||||
|
|||||||
+18
-8
@@ -14,7 +14,6 @@ import AccountPreferences from "@/pages/settings/account/account-preferences.tsx
|
|||||||
import SpaceHome from "@/pages/space/space-home.tsx";
|
import SpaceHome from "@/pages/space/space-home.tsx";
|
||||||
import PageRedirect from "@/pages/page/page-redirect.tsx";
|
import PageRedirect from "@/pages/page/page-redirect.tsx";
|
||||||
import Layout from "@/components/layouts/global/layout.tsx";
|
import Layout from "@/components/layouts/global/layout.tsx";
|
||||||
import { ErrorBoundary } from "react-error-boundary";
|
|
||||||
import InviteSignup from "@/pages/auth/invite-signup.tsx";
|
import InviteSignup from "@/pages/auth/invite-signup.tsx";
|
||||||
import ForgotPassword from "@/pages/auth/forgot-password.tsx";
|
import ForgotPassword from "@/pages/auth/forgot-password.tsx";
|
||||||
import PasswordReset from "./pages/auth/password-reset";
|
import PasswordReset from "./pages/auth/password-reset";
|
||||||
@@ -38,6 +37,12 @@ import SpaceTrash from "@/pages/space/space-trash.tsx";
|
|||||||
import UserApiKeys from "@/ee/api-key/pages/user-api-keys";
|
import UserApiKeys from "@/ee/api-key/pages/user-api-keys";
|
||||||
import WorkspaceApiKeys from "@/ee/api-key/pages/workspace-api-keys";
|
import WorkspaceApiKeys from "@/ee/api-key/pages/workspace-api-keys";
|
||||||
import AiSettings from "@/ee/ai/pages/ai-settings.tsx";
|
import AiSettings from "@/ee/ai/pages/ai-settings.tsx";
|
||||||
|
import AuditLogs from "@/ee/audit/pages/audit-logs.tsx";
|
||||||
|
import TemplateList from "@/ee/template/pages/template-list";
|
||||||
|
import TemplateEditor from "@/ee/template/pages/template-editor";
|
||||||
|
import FavoritesPage from "@/pages/favorites/favorites-page";
|
||||||
|
import AiChat from "@/ee/ai-chat/pages/ai-chat.tsx";
|
||||||
|
import VerifyEmail from "@/ee/pages/verify-email.tsx";
|
||||||
|
|
||||||
export default function App() {
|
export default function App() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -63,6 +68,7 @@ export default function App() {
|
|||||||
<>
|
<>
|
||||||
<Route path={"/create"} element={<CreateWorkspace />} />
|
<Route path={"/create"} element={<CreateWorkspace />} />
|
||||||
<Route path={"/select"} element={<CloudLogin />} />
|
<Route path={"/select"} element={<CloudLogin />} />
|
||||||
|
<Route path={"/verify-email"} element={<VerifyEmail />} />
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
@@ -79,18 +85,20 @@ export default function App() {
|
|||||||
|
|
||||||
<Route element={<Layout />}>
|
<Route element={<Layout />}>
|
||||||
<Route path={"/home"} element={<Home />} />
|
<Route path={"/home"} element={<Home />} />
|
||||||
|
<Route path={"/ai"} element={<AiChat />} />
|
||||||
|
<Route path={"/ai/chat/:chatId"} element={<AiChat />} />
|
||||||
<Route path={"/spaces"} element={<SpacesPage />} />
|
<Route path={"/spaces"} element={<SpacesPage />} />
|
||||||
|
<Route path={"/favorites"} element={<FavoritesPage />} />
|
||||||
|
<Route path={"/templates"} element={<TemplateList />} />
|
||||||
|
<Route
|
||||||
|
path={"/templates/:templateId"}
|
||||||
|
element={<TemplateEditor />}
|
||||||
|
/>
|
||||||
<Route path={"/s/:spaceSlug"} element={<SpaceHome />} />
|
<Route path={"/s/:spaceSlug"} element={<SpaceHome />} />
|
||||||
<Route path={"/s/:spaceSlug/trash"} element={<SpaceTrash />} />
|
<Route path={"/s/:spaceSlug/trash"} element={<SpaceTrash />} />
|
||||||
<Route
|
<Route
|
||||||
path={"/s/:spaceSlug/p/:pageSlug"}
|
path={"/s/:spaceSlug/p/:pageSlug"}
|
||||||
element={
|
element={<Page />}
|
||||||
<ErrorBoundary
|
|
||||||
fallback={<>{t("Failed to load page. An error occurred.")}</>}
|
|
||||||
>
|
|
||||||
<Page />
|
|
||||||
</ErrorBoundary>
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Route path={"/settings"}>
|
<Route path={"/settings"}>
|
||||||
@@ -109,6 +117,8 @@ export default function App() {
|
|||||||
<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 />} />
|
<Route path={"ai"} element={<AiSettings />} />
|
||||||
|
<Route path={"ai/mcp"} element={<AiSettings />} />
|
||||||
|
<Route path={"audit"} element={<AuditLogs />} />
|
||||||
{!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>
|
||||||
|
|||||||
@@ -130,7 +130,7 @@ export default function AvatarUploader({
|
|||||||
top: "50%",
|
top: "50%",
|
||||||
left: "50%",
|
left: "50%",
|
||||||
transform: "translate(-50%, -50%)",
|
transform: "translate(-50%, -50%)",
|
||||||
zIndex: 1000,
|
zIndex: 200,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Loader size="sm" />
|
<Loader size="sm" />
|
||||||
|
|||||||
@@ -0,0 +1,33 @@
|
|||||||
|
// Source: https://github.com/mantinedev/mantine/blob/master/packages/@mantine/core/src/components/CopyButton/CopyButton.tsx - MIT
|
||||||
|
// modified to use the polyfilled clipboard api
|
||||||
|
import React from "react";
|
||||||
|
import { useClipboard } from "@/hooks/use-clipboard";
|
||||||
|
import { useProps } from "@mantine/core";
|
||||||
|
|
||||||
|
interface CopyButtonProps {
|
||||||
|
/** Children callback, provides current status and copy function as an argument */
|
||||||
|
children: (payload: { copied: boolean; copy: () => void }) => React.ReactNode;
|
||||||
|
|
||||||
|
/** Value that is copied to the clipboard when the button is clicked */
|
||||||
|
value: string;
|
||||||
|
|
||||||
|
/** Copied status timeout in ms @default `1000` */
|
||||||
|
timeout?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
const defaultProps = {
|
||||||
|
timeout: 1000,
|
||||||
|
} satisfies Partial<CopyButtonProps>;
|
||||||
|
|
||||||
|
export function CopyButton(props: CopyButtonProps) {
|
||||||
|
const { children, timeout, value, ...others } = useProps(
|
||||||
|
"CopyButton",
|
||||||
|
defaultProps,
|
||||||
|
props,
|
||||||
|
);
|
||||||
|
const clipboard = useClipboard({ timeout });
|
||||||
|
const copy = () => clipboard.copy(value);
|
||||||
|
return <>{children({ copy, copied: clipboard.copied, ...others })}</>;
|
||||||
|
}
|
||||||
|
|
||||||
|
CopyButton.displayName = "@mantine/core/CopyButton";
|
||||||
@@ -1,12 +1,15 @@
|
|||||||
import { ActionIcon, CopyButton, Tooltip } from "@mantine/core";
|
import { ActionIcon, MantineColor, MantineSize, Tooltip } from "@mantine/core";
|
||||||
|
import { CopyButton } from "@/components/common/copy-button";
|
||||||
import { IconCheck, IconCopy } from "@tabler/icons-react";
|
import { IconCheck, IconCopy } from "@tabler/icons-react";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
interface CopyProps {
|
interface CopyProps {
|
||||||
text: string;
|
text: string;
|
||||||
|
size?: MantineSize;
|
||||||
|
color?: MantineColor;
|
||||||
}
|
}
|
||||||
export default function CopyTextButton({ text }: CopyProps) {
|
export default function CopyTextButton({ text, size }: CopyProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -21,6 +24,7 @@ export default function CopyTextButton({ text }: CopyProps) {
|
|||||||
color={copied ? "teal" : "gray"}
|
color={copied ? "teal" : "gray"}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
onClick={copy}
|
onClick={copy}
|
||||||
|
size={size}
|
||||||
>
|
>
|
||||||
{copied ? <IconCheck size={16} /> : <IconCopy size={16} />}
|
{copied ? <IconCheck size={16} /> : <IconCopy size={16} />}
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
|
|||||||
@@ -5,13 +5,15 @@ import {
|
|||||||
Badge,
|
Badge,
|
||||||
Table,
|
Table,
|
||||||
ActionIcon,
|
ActionIcon,
|
||||||
|
Button,
|
||||||
} 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, IconFiles } from "@tabler/icons-react";
|
||||||
|
import { EmptyState } from "@/components/ui/empty-state.tsx";
|
||||||
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";
|
import { getInitialsColor } from "@/lib/get-initials-color.ts";
|
||||||
@@ -22,7 +24,8 @@ interface Props {
|
|||||||
|
|
||||||
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, isLoading, isError, hasNextPage, fetchNextPage, isFetchingNextPage } = useRecentChangesQuery(spaceId);
|
||||||
|
const pages = data?.pages.flatMap((p) => p.items) ?? [];
|
||||||
|
|
||||||
if (isLoading) {
|
if (isLoading) {
|
||||||
return <PageListSkeleton />;
|
return <PageListSkeleton />;
|
||||||
@@ -32,61 +35,77 @@ export default function RecentChanges({ spaceId }: Props) {
|
|||||||
return <Text>{t("Failed to fetch recent pages")}</Text>;
|
return <Text>{t("Failed to fetch recent pages")}</Text>;
|
||||||
}
|
}
|
||||||
|
|
||||||
return pages && pages.items.length > 0 ? (
|
return pages.length > 0 ? (
|
||||||
<Table.ScrollContainer minWidth={500}>
|
<>
|
||||||
<Table highlightOnHover verticalSpacing="sm">
|
<Table.ScrollContainer minWidth={500}>
|
||||||
<Table.Tbody>
|
<Table highlightOnHover verticalSpacing="sm">
|
||||||
{pages.items.map((page) => (
|
<Table.Tbody>
|
||||||
<Table.Tr key={page.id}>
|
{pages.map((page) => (
|
||||||
<Table.Td>
|
<Table.Tr key={page.id}>
|
||||||
<UnstyledButton
|
|
||||||
component={Link}
|
|
||||||
to={buildPageUrl(page?.space.slug, page.slugId, page.title)}
|
|
||||||
>
|
|
||||||
<Group wrap="nowrap">
|
|
||||||
{page.icon || (
|
|
||||||
<ActionIcon variant="transparent" color="gray" size={18}>
|
|
||||||
<IconFileDescription size={18} />
|
|
||||||
</ActionIcon>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<Text fw={500} size="md" lineClamp={1}>
|
|
||||||
{page.title || t("Untitled")}
|
|
||||||
</Text>
|
|
||||||
</Group>
|
|
||||||
</UnstyledButton>
|
|
||||||
</Table.Td>
|
|
||||||
{!spaceId && (
|
|
||||||
<Table.Td>
|
<Table.Td>
|
||||||
<Badge
|
<UnstyledButton
|
||||||
color={getInitialsColor(page?.space.name)}
|
|
||||||
variant="light"
|
|
||||||
component={Link}
|
component={Link}
|
||||||
to={getSpaceUrl(page?.space.slug)}
|
to={buildPageUrl(page?.space.slug, page.slugId, page.title)}
|
||||||
style={{ cursor: "pointer" }}
|
|
||||||
>
|
>
|
||||||
{page?.space.name}
|
<Group wrap="nowrap">
|
||||||
</Badge>
|
{page.icon || (
|
||||||
|
<ActionIcon variant="transparent" color="gray" size={18}>
|
||||||
|
<IconFileDescription size={18} />
|
||||||
|
</ActionIcon>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<Text fw={500} size="md" lineClamp={1}>
|
||||||
|
{page.title || t("Untitled")}
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
</UnstyledButton>
|
||||||
</Table.Td>
|
</Table.Td>
|
||||||
)}
|
{!spaceId && (
|
||||||
<Table.Td>
|
<Table.Td>
|
||||||
<Text
|
<Badge
|
||||||
c="dimmed"
|
color={getInitialsColor(page?.space.name)}
|
||||||
style={{ whiteSpace: "nowrap" }}
|
variant="light"
|
||||||
size="xs"
|
component={Link}
|
||||||
fw={500}
|
to={getSpaceUrl(page?.space.slug)}
|
||||||
>
|
style={{ cursor: "pointer" }}
|
||||||
{formattedDate(page.updatedAt)}
|
>
|
||||||
</Text>
|
{page?.space.name}
|
||||||
</Table.Td>
|
</Badge>
|
||||||
</Table.Tr>
|
</Table.Td>
|
||||||
))}
|
)}
|
||||||
</Table.Tbody>
|
<Table.Td>
|
||||||
</Table>
|
<Text
|
||||||
</Table.ScrollContainer>
|
c="dimmed"
|
||||||
|
style={{ whiteSpace: "nowrap" }}
|
||||||
|
size="xs"
|
||||||
|
fw={500}
|
||||||
|
>
|
||||||
|
{formattedDate(page.updatedAt)}
|
||||||
|
</Text>
|
||||||
|
</Table.Td>
|
||||||
|
</Table.Tr>
|
||||||
|
))}
|
||||||
|
</Table.Tbody>
|
||||||
|
</Table>
|
||||||
|
</Table.ScrollContainer>
|
||||||
|
{hasNextPage && (
|
||||||
|
<Button
|
||||||
|
variant="subtle"
|
||||||
|
fullWidth
|
||||||
|
mt="sm"
|
||||||
|
mb="xl"
|
||||||
|
onClick={() => fetchNextPage()}
|
||||||
|
loading={isFetchingNextPage}
|
||||||
|
>
|
||||||
|
{t("Load more")}
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
) : (
|
) : (
|
||||||
<Text size="md" ta="center">
|
<EmptyState
|
||||||
{t("No pages yet")}
|
icon={IconFiles}
|
||||||
</Text>
|
title={t("No pages yet")}
|
||||||
|
description={t("Pages you create will show up here.")}
|
||||||
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,27 @@
|
|||||||
|
import { rem } from "@mantine/core";
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
size?: number | string;
|
||||||
|
stroke?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function IconColumns4({ size = 24, stroke = 2 }: Props) {
|
||||||
|
return (
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width={rem(size)}
|
||||||
|
height={rem(size)}
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth={stroke}
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
>
|
||||||
|
<path d="M3 4a1 1 0 0 1 1 -1h16a1 1 0 0 1 1 1v16a1 1 0 0 1 -1 1h-16a1 1 0 0 1 -1 -1v-16" />
|
||||||
|
<path d="M7.5 3v18" />
|
||||||
|
<path d="M12 3v18" />
|
||||||
|
<path d="M16.5 3v18" />
|
||||||
|
</svg>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
import { rem } from "@mantine/core";
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
size?: number | string;
|
||||||
|
stroke?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function IconColumns5({ size = 24, stroke = 2 }: Props) {
|
||||||
|
return (
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width={rem(size)}
|
||||||
|
height={rem(size)}
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
strokeWidth={stroke}
|
||||||
|
strokeLinecap="round"
|
||||||
|
strokeLinejoin="round"
|
||||||
|
>
|
||||||
|
<path d="M3 4a1 1 0 0 1 1 -1h16a1 1 0 0 1 1 1v16a1 1 0 0 1 -1 1h-16a1 1 0 0 1 -1 -1v-16" />
|
||||||
|
<path d="M6.6 3v18" />
|
||||||
|
<path d="M10.2 3v18" />
|
||||||
|
<path d="M13.8 3v18" />
|
||||||
|
<path d="M17.4 3v18" />
|
||||||
|
</svg>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -7,6 +7,19 @@
|
|||||||
padding-right: var(--mantine-spacing-md);
|
padding-right: var(--mantine-spacing-md);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.brand {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
text-decoration: none;
|
||||||
|
color: inherit;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.brandIcon {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
.link {
|
.link {
|
||||||
display: block;
|
display: block;
|
||||||
line-height: 1;
|
line-height: 1;
|
||||||
@@ -16,6 +29,9 @@
|
|||||||
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-0));
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-0));
|
||||||
font-size: var(--mantine-font-size-sm);
|
font-size: var(--mantine-font-size-sm);
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
|
user-select: none;
|
||||||
|
white-space: nowrap;
|
||||||
|
flex-shrink: 0;
|
||||||
|
|
||||||
@mixin hover {
|
@mixin hover {
|
||||||
background-color: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-6));
|
background-color: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-6));
|
||||||
|
|||||||
@@ -1,8 +1,18 @@
|
|||||||
import { Badge, Group, Text, Tooltip } from "@mantine/core";
|
import {
|
||||||
|
ActionIcon,
|
||||||
|
Badge,
|
||||||
|
Box,
|
||||||
|
Group,
|
||||||
|
Text,
|
||||||
|
Tooltip,
|
||||||
|
UnstyledButton,
|
||||||
|
} from "@mantine/core";
|
||||||
import classes from "./app-header.module.css";
|
import classes from "./app-header.module.css";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import TopMenu from "@/components/layouts/global/top-menu.tsx";
|
import TopMenu from "@/components/layouts/global/top-menu.tsx";
|
||||||
import { Link } from "react-router-dom";
|
import { Link, useLocation } from "react-router-dom";
|
||||||
|
import { IconSparkles } from "@tabler/icons-react";
|
||||||
|
import useToggleAside from "@/hooks/use-toggle-aside.tsx";
|
||||||
import APP_ROUTE from "@/lib/app-route.ts";
|
import APP_ROUTE from "@/lib/app-route.ts";
|
||||||
import { useAtom } from "jotai";
|
import { useAtom } from "jotai";
|
||||||
import {
|
import {
|
||||||
@@ -22,8 +32,12 @@ import {
|
|||||||
searchSpotlight,
|
searchSpotlight,
|
||||||
shareSearchSpotlight,
|
shareSearchSpotlight,
|
||||||
} from "@/features/search/constants.ts";
|
} from "@/features/search/constants.ts";
|
||||||
|
import { NotificationPopover } from "@/features/notification/components/notification-popover.tsx";
|
||||||
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
|
||||||
const links = [{ link: APP_ROUTE.HOME, label: "Home" }];
|
const links = [
|
||||||
|
{ link: APP_ROUTE.HOME, label: "Home" },
|
||||||
|
];
|
||||||
|
|
||||||
export function AppHeader() {
|
export function AppHeader() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -33,10 +47,12 @@ export function AppHeader() {
|
|||||||
const [desktopOpened] = useAtom(desktopSidebarAtom);
|
const [desktopOpened] = useAtom(desktopSidebarAtom);
|
||||||
const toggleDesktop = useToggleSidebar(desktopSidebarAtom);
|
const toggleDesktop = useToggleSidebar(desktopSidebarAtom);
|
||||||
const { isTrial, trialDaysLeft } = useTrial();
|
const { isTrial, trialDaysLeft } = useTrial();
|
||||||
|
const location = useLocation();
|
||||||
|
const toggleAside = useToggleAside();
|
||||||
|
const [workspace] = useAtom(workspaceAtom);
|
||||||
|
const aiChatEnabled = workspace?.settings?.ai?.chat === true;
|
||||||
|
|
||||||
const isHomeRoute = location.pathname.startsWith("/home");
|
const isPageRoute = location.pathname.includes("/p/");
|
||||||
const isSpacesRoute = location.pathname === "/spaces";
|
|
||||||
const hideSidebar = isHomeRoute || isSpacesRoute;
|
|
||||||
|
|
||||||
const items = links.map((link) => (
|
const items = links.map((link) => (
|
||||||
<Link key={link.label} to={link.link} className={classes.link}>
|
<Link key={link.label} to={link.link} className={classes.link}>
|
||||||
@@ -48,39 +64,44 @@ export function AppHeader() {
|
|||||||
<>
|
<>
|
||||||
<Group h="100%" px="md" justify="space-between" wrap={"nowrap"}>
|
<Group h="100%" px="md" justify="space-between" wrap={"nowrap"}>
|
||||||
<Group wrap="nowrap">
|
<Group wrap="nowrap">
|
||||||
{!hideSidebar && (
|
<Tooltip label={t("Sidebar toggle")}>
|
||||||
<>
|
<SidebarToggle
|
||||||
<Tooltip label={t("Sidebar toggle")}>
|
aria-label={t("Sidebar toggle")}
|
||||||
<SidebarToggle
|
opened={mobileOpened}
|
||||||
aria-label={t("Sidebar toggle")}
|
onClick={toggleMobile}
|
||||||
opened={mobileOpened}
|
hiddenFrom="sm"
|
||||||
onClick={toggleMobile}
|
size="sm"
|
||||||
hiddenFrom="sm"
|
/>
|
||||||
size="sm"
|
</Tooltip>
|
||||||
/>
|
|
||||||
</Tooltip>
|
|
||||||
|
|
||||||
<Tooltip label={t("Sidebar toggle")}>
|
<Tooltip label={t("Sidebar toggle")}>
|
||||||
<SidebarToggle
|
<SidebarToggle
|
||||||
aria-label={t("Sidebar toggle")}
|
aria-label={t("Sidebar toggle")}
|
||||||
opened={desktopOpened}
|
opened={desktopOpened}
|
||||||
onClick={toggleDesktop}
|
onClick={toggleDesktop}
|
||||||
visibleFrom="sm"
|
visibleFrom="sm"
|
||||||
size="sm"
|
size="sm"
|
||||||
/>
|
/>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<Text
|
<Link to="/home" className={classes.brand} aria-label="Docmost">
|
||||||
size="lg"
|
<Box hiddenFrom="sm" className={classes.brandIcon}>
|
||||||
fw={600}
|
<img
|
||||||
style={{ cursor: "pointer", userSelect: "none" }}
|
src="/icons/favicon-32x32.png"
|
||||||
component={Link}
|
alt="Docmost"
|
||||||
to="/home"
|
width={22}
|
||||||
>
|
height={22}
|
||||||
Docmost
|
/>
|
||||||
</Text>
|
</Box>
|
||||||
|
<Text
|
||||||
|
size="lg"
|
||||||
|
fw={600}
|
||||||
|
style={{ userSelect: "none" }}
|
||||||
|
visibleFrom="sm"
|
||||||
|
>
|
||||||
|
Docmost
|
||||||
|
</Text>
|
||||||
|
</Link>
|
||||||
|
|
||||||
<Group ml={50} gap={5} className={classes.links} visibleFrom="sm">
|
<Group ml={50} gap={5} className={classes.links} visibleFrom="sm">
|
||||||
{items}
|
{items}
|
||||||
@@ -97,6 +118,50 @@ export function AppHeader() {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Group px={"xl"} wrap="nowrap">
|
<Group px={"xl"} wrap="nowrap">
|
||||||
|
{aiChatEnabled && (
|
||||||
|
<>
|
||||||
|
<UnstyledButton
|
||||||
|
component={Link}
|
||||||
|
to="/ai"
|
||||||
|
className={classes.link}
|
||||||
|
visibleFrom="sm"
|
||||||
|
onClick={(e: React.MouseEvent) => {
|
||||||
|
if (e.metaKey || e.ctrlKey || e.shiftKey || e.button === 1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (isPageRoute) {
|
||||||
|
e.preventDefault();
|
||||||
|
toggleAside("chat");
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{t("AI Chat")}
|
||||||
|
</UnstyledButton>
|
||||||
|
<Tooltip label={t("AI Chat")} openDelay={250} withArrow>
|
||||||
|
<ActionIcon
|
||||||
|
component={Link}
|
||||||
|
to="/ai"
|
||||||
|
variant="subtle"
|
||||||
|
color="dark"
|
||||||
|
size="sm"
|
||||||
|
hiddenFrom="sm"
|
||||||
|
aria-label={t("AI Chat")}
|
||||||
|
onClick={(e: React.MouseEvent) => {
|
||||||
|
if (e.metaKey || e.ctrlKey || e.shiftKey || e.button === 1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (isPageRoute) {
|
||||||
|
e.preventDefault();
|
||||||
|
toggleAside("chat");
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<IconSparkles size={20} stroke={2} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
<NotificationPopover />
|
||||||
{isCloud() && isTrial && trialDaysLeft !== 0 && (
|
{isCloud() && isTrial && trialDaysLeft !== 0 && (
|
||||||
<Badge
|
<Badge
|
||||||
variant="light"
|
variant="light"
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import { useTranslation } from "react-i18next";
|
|||||||
import { TableOfContents } from "@/features/editor/components/table-of-contents/table-of-contents.tsx";
|
import { TableOfContents } from "@/features/editor/components/table-of-contents/table-of-contents.tsx";
|
||||||
import { useAtomValue } from "jotai";
|
import { useAtomValue } from "jotai";
|
||||||
import { pageEditorAtom } from "@/features/editor/atoms/editor-atoms.ts";
|
import { pageEditorAtom } from "@/features/editor/atoms/editor-atoms.ts";
|
||||||
|
import AsideChatPanel from "@/ee/ai-chat/components/aside-chat-panel";
|
||||||
|
|
||||||
export default function Aside() {
|
export default function Aside() {
|
||||||
const [{ tab }] = useAtom(asideStateAtom);
|
const [{ tab }] = useAtom(asideStateAtom);
|
||||||
@@ -25,21 +26,27 @@ export default function Aside() {
|
|||||||
component = <TableOfContents editor={pageEditor} />;
|
component = <TableOfContents editor={pageEditor} />;
|
||||||
title = "Table of contents";
|
title = "Table of contents";
|
||||||
break;
|
break;
|
||||||
|
case "chat":
|
||||||
|
component = <AsideChatPanel />;
|
||||||
|
title = "AI Chat";
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
component = null;
|
component = null;
|
||||||
title = null;
|
title = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box p="md">
|
<Box p="md" style={{ height: "100%", display: "flex", flexDirection: "column" }}>
|
||||||
{component && (
|
{component && (
|
||||||
<>
|
<>
|
||||||
<Text mb="md" fw={500}>
|
{tab !== "chat" && (
|
||||||
{t(title)}
|
<Text mb="md" fw={500}>
|
||||||
</Text>
|
{t(title)}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
|
||||||
{tab === "comments" ? (
|
{tab === "comments" || tab === "chat" ? (
|
||||||
<CommentListWithTabs />
|
component
|
||||||
) : (
|
) : (
|
||||||
<ScrollArea
|
<ScrollArea
|
||||||
style={{ height: "85vh" }}
|
style={{ height: "85vh" }}
|
||||||
|
|||||||
@@ -10,11 +10,13 @@ import {
|
|||||||
sidebarWidthAtom,
|
sidebarWidthAtom,
|
||||||
} from "@/components/layouts/global/hooks/atoms/sidebar-atom.ts";
|
} from "@/components/layouts/global/hooks/atoms/sidebar-atom.ts";
|
||||||
import { SpaceSidebar } from "@/features/space/components/sidebar/space-sidebar.tsx";
|
import { SpaceSidebar } from "@/features/space/components/sidebar/space-sidebar.tsx";
|
||||||
|
import AiChatSidebar from "@/ee/ai-chat/components/ai-chat-sidebar.tsx";
|
||||||
import { AppHeader } from "@/components/layouts/global/app-header.tsx";
|
import { AppHeader } from "@/components/layouts/global/app-header.tsx";
|
||||||
import Aside from "@/components/layouts/global/aside.tsx";
|
import Aside from "@/components/layouts/global/aside.tsx";
|
||||||
import classes from "./app-shell.module.css";
|
import classes from "./app-shell.module.css";
|
||||||
import { useTrialEndAction } from "@/ee/hooks/use-trial-end-action.tsx";
|
import { useTrialEndAction } from "@/ee/hooks/use-trial-end-action.tsx";
|
||||||
import { useToggleSidebar } from "@/components/layouts/global/hooks/hooks/use-toggle-sidebar.ts";
|
import { useToggleSidebar } from "@/components/layouts/global/hooks/hooks/use-toggle-sidebar.ts";
|
||||||
|
import GlobalSidebar from "@/components/layouts/global/global-sidebar.tsx";
|
||||||
|
|
||||||
export default function GlobalAppShell({
|
export default function GlobalAppShell({
|
||||||
children,
|
children,
|
||||||
@@ -72,24 +74,21 @@ export default function GlobalAppShell({
|
|||||||
const location = useLocation();
|
const location = useLocation();
|
||||||
const isSettingsRoute = location.pathname.startsWith("/settings");
|
const isSettingsRoute = location.pathname.startsWith("/settings");
|
||||||
const isSpaceRoute = location.pathname.startsWith("/s/");
|
const isSpaceRoute = location.pathname.startsWith("/s/");
|
||||||
const isHomeRoute = location.pathname.startsWith("/home");
|
const isAiRoute = location.pathname.startsWith("/ai");
|
||||||
const isSpacesRoute = location.pathname === "/spaces";
|
|
||||||
const isPageRoute = location.pathname.includes("/p/");
|
const isPageRoute = location.pathname.includes("/p/");
|
||||||
const hideSidebar = isHomeRoute || isSpacesRoute;
|
const showGlobalSidebar = !isSpaceRoute && !isSettingsRoute && !isAiRoute;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<AppShell
|
<AppShell
|
||||||
header={{ height: 45 }}
|
header={{ height: 45 }}
|
||||||
navbar={
|
navbar={{
|
||||||
!hideSidebar && {
|
width: isSpaceRoute ? sidebarWidth : 300,
|
||||||
width: isSpaceRoute ? sidebarWidth : 300,
|
breakpoint: "sm",
|
||||||
breakpoint: "sm",
|
collapsed: {
|
||||||
collapsed: {
|
mobile: !mobileOpened,
|
||||||
mobile: !mobileOpened,
|
desktop: !desktopOpened,
|
||||||
desktop: !desktopOpened,
|
},
|
||||||
},
|
}}
|
||||||
}
|
|
||||||
}
|
|
||||||
aside={
|
aside={
|
||||||
isPageRoute && {
|
isPageRoute && {
|
||||||
width: 350,
|
width: 350,
|
||||||
@@ -102,20 +101,22 @@ export default function GlobalAppShell({
|
|||||||
<AppShell.Header px="md" className={classes.header}>
|
<AppShell.Header px="md" className={classes.header}>
|
||||||
<AppHeader />
|
<AppHeader />
|
||||||
</AppShell.Header>
|
</AppShell.Header>
|
||||||
{!hideSidebar && (
|
<AppShell.Navbar
|
||||||
<AppShell.Navbar
|
className={classes.navbar}
|
||||||
className={classes.navbar}
|
withBorder={false}
|
||||||
withBorder={false}
|
ref={sidebarRef}
|
||||||
ref={sidebarRef}
|
>
|
||||||
>
|
{isSpaceRoute && (
|
||||||
<div className={classes.resizeHandle} onMouseDown={startResizing} />
|
<div className={classes.resizeHandle} onMouseDown={startResizing} />
|
||||||
{isSpaceRoute && <SpaceSidebar />}
|
)}
|
||||||
{isSettingsRoute && <SettingsSidebar />}
|
{isSpaceRoute && <SpaceSidebar />}
|
||||||
</AppShell.Navbar>
|
{isSettingsRoute && <SettingsSidebar />}
|
||||||
)}
|
{isAiRoute && <AiChatSidebar />}
|
||||||
|
{showGlobalSidebar && <GlobalSidebar />}
|
||||||
|
</AppShell.Navbar>
|
||||||
<AppShell.Main>
|
<AppShell.Main>
|
||||||
{isSettingsRoute ? (
|
{isSettingsRoute ? (
|
||||||
<Container size={850}>{children}</Container>
|
<Container size={900}>{children}</Container>
|
||||||
) : (
|
) : (
|
||||||
children
|
children
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -0,0 +1,89 @@
|
|||||||
|
.navbar {
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
padding: var(--mantine-spacing-md);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section {
|
||||||
|
padding-bottom: var(--mantine-spacing-xs);
|
||||||
|
}
|
||||||
|
|
||||||
|
.link {
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
text-decoration: none;
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
padding-left: var(--mantine-spacing-xs);
|
||||||
|
min-height: 30px;
|
||||||
|
border-radius: var(--mantine-radius-sm);
|
||||||
|
font-weight: 500;
|
||||||
|
user-select: none;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
background-color: light-dark(
|
||||||
|
var(--mantine-color-gray-1),
|
||||||
|
var(--mantine-color-dark-6)
|
||||||
|
);
|
||||||
|
color: light-dark(var(--mantine-color-black), var(--mantine-color-white));
|
||||||
|
}
|
||||||
|
|
||||||
|
&[data-active] {
|
||||||
|
&,
|
||||||
|
& :hover {
|
||||||
|
background-color: light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-6));
|
||||||
|
color: light-dark(var(--mantine-color-black), var(--mantine-color-white));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.linkIcon {
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
margin-right: var(--mantine-spacing-sm);
|
||||||
|
width: rem(16px);
|
||||||
|
height: rem(16px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.sectionHeader {
|
||||||
|
padding: var(--mantine-spacing-xs) var(--mantine-spacing-sm);
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-3));
|
||||||
|
font-weight: 600;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.spacer {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bottomSection {
|
||||||
|
padding-top: var(--mantine-spacing-xs);
|
||||||
|
border-top: rem(1px) solid light-dark(var(--mantine-color-gray-3), var(--mantine-color-dark-4));
|
||||||
|
}
|
||||||
|
|
||||||
|
.spaceItem {
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: var(--mantine-spacing-sm);
|
||||||
|
text-decoration: none;
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
padding-left: var(--mantine-spacing-xs);
|
||||||
|
min-height: 30px;
|
||||||
|
border-radius: var(--mantine-radius-sm);
|
||||||
|
font-weight: 500;
|
||||||
|
user-select: none;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
background-color: light-dark(
|
||||||
|
var(--mantine-color-gray-1),
|
||||||
|
var(--mantine-color-dark-6)
|
||||||
|
);
|
||||||
|
color: light-dark(var(--mantine-color-black), var(--mantine-color-white));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,158 @@
|
|||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { ScrollArea, Text, Divider, Modal } from "@mantine/core";
|
||||||
|
import {
|
||||||
|
IconHome,
|
||||||
|
IconClock,
|
||||||
|
IconStar,
|
||||||
|
IconLayoutGrid,
|
||||||
|
IconSettings,
|
||||||
|
IconUserPlus,
|
||||||
|
} from "@tabler/icons-react";
|
||||||
|
import { Link, useLocation } from "react-router-dom";
|
||||||
|
import classes from "./global-sidebar.module.css";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { mobileSidebarAtom } from "@/components/layouts/global/hooks/atoms/sidebar-atom";
|
||||||
|
import { useToggleSidebar } from "@/components/layouts/global/hooks/hooks/use-toggle-sidebar";
|
||||||
|
import { useFavoritesQuery } from "@/features/favorite/queries/favorite-query";
|
||||||
|
import { getSpaceUrl } from "@/lib/config";
|
||||||
|
import { useDisclosure } from "@mantine/hooks";
|
||||||
|
import { WorkspaceInviteForm } from "@/features/workspace/components/members/components/workspace-invite-form";
|
||||||
|
import { CustomAvatar } from "@/components/ui/custom-avatar";
|
||||||
|
import { AvatarIconType } from "@/features/attachments/types/attachment.types";
|
||||||
|
|
||||||
|
const mainNavItems = [
|
||||||
|
{ label: "Home", icon: IconHome, path: "/home" },
|
||||||
|
{ label: "Favorites", icon: IconStar, path: "/favorites" },
|
||||||
|
{ label: "Spaces", icon: IconLayoutGrid, path: "/spaces" },
|
||||||
|
];
|
||||||
|
|
||||||
|
export default function GlobalSidebar() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const location = useLocation();
|
||||||
|
const [active, setActive] = useState(location.pathname);
|
||||||
|
const [mobileSidebarOpened] = useAtom(mobileSidebarAtom);
|
||||||
|
const toggleMobileSidebar = useToggleSidebar(mobileSidebarAtom);
|
||||||
|
const { data: favoriteSpacesData } = useFavoritesQuery("space");
|
||||||
|
const favoriteSpaces = favoriteSpacesData?.pages.flatMap((p) => p.items) ?? [];
|
||||||
|
const sortedFavoriteSpaces = [...favoriteSpaces]
|
||||||
|
.filter((fav) => fav.space)
|
||||||
|
.sort((a, b) => {
|
||||||
|
const cmp = (a.space!.name ?? "").localeCompare(b.space!.name ?? "", undefined, { sensitivity: "base" });
|
||||||
|
return cmp !== 0 ? cmp : a.id.localeCompare(b.id);
|
||||||
|
});
|
||||||
|
const [inviteOpened, { open: openInvite, close: closeInvite }] =
|
||||||
|
useDisclosure(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setActive(location.pathname);
|
||||||
|
}, [location.pathname]);
|
||||||
|
|
||||||
|
const handleNavClick = () => {
|
||||||
|
if (mobileSidebarOpened) {
|
||||||
|
toggleMobileSidebar();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.navbar}>
|
||||||
|
<ScrollArea w="100%" style={{ flex: 1 }}>
|
||||||
|
<div className={classes.section}>
|
||||||
|
{mainNavItems.map((item) => (
|
||||||
|
<Link
|
||||||
|
key={item.label}
|
||||||
|
className={classes.link}
|
||||||
|
data-active={active === item.path || undefined}
|
||||||
|
to={item.path}
|
||||||
|
onClick={handleNavClick}
|
||||||
|
>
|
||||||
|
<item.icon className={classes.linkIcon} stroke={2} />
|
||||||
|
<span>{t(item.label)}</span>
|
||||||
|
</Link>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Divider my="xs" />
|
||||||
|
<div className={classes.section}>
|
||||||
|
<Text className={classes.sectionHeader}>{t("Favorite spaces")}</Text>
|
||||||
|
{sortedFavoriteSpaces.length === 0 ? (
|
||||||
|
<Text size="xs" c="dimmed" pl="xs" py={4}>
|
||||||
|
{t("Favorite spaces appear here")}
|
||||||
|
</Text>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
{sortedFavoriteSpaces.slice(0, 10).map((fav) => (
|
||||||
|
<Link
|
||||||
|
key={fav.id}
|
||||||
|
className={classes.spaceItem}
|
||||||
|
to={getSpaceUrl(fav.space!.slug)}
|
||||||
|
onClick={handleNavClick}
|
||||||
|
>
|
||||||
|
<CustomAvatar
|
||||||
|
name={fav.space!.name}
|
||||||
|
avatarUrl={fav.space!.logo}
|
||||||
|
type={AvatarIconType.SPACE_ICON}
|
||||||
|
color="initials"
|
||||||
|
variant="filled"
|
||||||
|
size={20}
|
||||||
|
/>
|
||||||
|
<Text size="sm" fw={500} lineClamp={1}>
|
||||||
|
{fav.space!.name}
|
||||||
|
</Text>
|
||||||
|
</Link>
|
||||||
|
))}
|
||||||
|
{sortedFavoriteSpaces.length > 10 && (
|
||||||
|
<Link
|
||||||
|
className={classes.spaceItem}
|
||||||
|
to="/spaces"
|
||||||
|
onClick={handleNavClick}
|
||||||
|
>
|
||||||
|
<Text size="xs" c="dimmed">
|
||||||
|
{t("View all")}
|
||||||
|
</Text>
|
||||||
|
</Link>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</ScrollArea>
|
||||||
|
|
||||||
|
<div className={classes.bottomSection}>
|
||||||
|
<a
|
||||||
|
className={classes.link}
|
||||||
|
onClick={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
openInvite();
|
||||||
|
}}
|
||||||
|
href="#"
|
||||||
|
>
|
||||||
|
<IconUserPlus className={classes.linkIcon} stroke={2} />
|
||||||
|
<span>{t("Invite People")}</span>
|
||||||
|
</a>
|
||||||
|
<Link
|
||||||
|
className={classes.link}
|
||||||
|
data-active={active.startsWith("/settings") || undefined}
|
||||||
|
to="/settings/account/profile"
|
||||||
|
onClick={handleNavClick}
|
||||||
|
>
|
||||||
|
<IconSettings className={classes.linkIcon} stroke={2} />
|
||||||
|
<span>{t("Settings")}</span>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Modal
|
||||||
|
size="550"
|
||||||
|
opened={inviteOpened}
|
||||||
|
onClose={closeInvite}
|
||||||
|
title={t("Invite new members")}
|
||||||
|
centered
|
||||||
|
>
|
||||||
|
<Divider size="xs" mb="xs" />
|
||||||
|
<ScrollArea h="80%">
|
||||||
|
<WorkspaceInviteForm onClose={closeInvite} />
|
||||||
|
</ScrollArea>
|
||||||
|
</Modal>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -11,6 +11,7 @@ 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";
|
import { getApiKeys } from "@/ee/api-key";
|
||||||
|
import { getAuditLogs } from "@/ee/audit/services/audit-service";
|
||||||
|
|
||||||
export const prefetchWorkspaceMembers = () => {
|
export const prefetchWorkspaceMembers = () => {
|
||||||
const params: QueryParams = { limit: 100, query: "" };
|
const params: QueryParams = { limit: 100, query: "" };
|
||||||
@@ -80,3 +81,11 @@ export const prefetchApiKeyManagement = () => {
|
|||||||
queryFn: () => getApiKeys({ adminView: true }),
|
queryFn: () => getApiKeys({ adminView: true }),
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const prefetchAuditLogs = () => {
|
||||||
|
const params = { limit: 50 };
|
||||||
|
queryClient.prefetchQuery({
|
||||||
|
queryKey: ["audit-logs", params],
|
||||||
|
queryFn: () => getAuditLogs(params),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import {
|
|||||||
IconKey,
|
IconKey,
|
||||||
IconWorld,
|
IconWorld,
|
||||||
IconSparkles,
|
IconSparkles,
|
||||||
|
IconHistory,
|
||||||
} 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";
|
||||||
@@ -20,7 +21,9 @@ 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";
|
import { useAtom } from "jotai";
|
||||||
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
import { entitlementAtom } from "@/ee/entitlement/entitlement-atom";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
import {
|
import {
|
||||||
prefetchApiKeyManagement,
|
prefetchApiKeyManagement,
|
||||||
prefetchApiKeys,
|
prefetchApiKeys,
|
||||||
@@ -31,27 +34,26 @@ import {
|
|||||||
prefetchSpaces,
|
prefetchSpaces,
|
||||||
prefetchSsoProviders,
|
prefetchSsoProviders,
|
||||||
prefetchWorkspaceMembers,
|
prefetchWorkspaceMembers,
|
||||||
|
prefetchAuditLogs,
|
||||||
} from "@/components/settings/settings-queries.tsx";
|
} from "@/components/settings/settings-queries.tsx";
|
||||||
import AppVersion from "@/components/settings/app-version.tsx";
|
import AppVersion from "@/components/settings/app-version.tsx";
|
||||||
import { mobileSidebarAtom } from "@/components/layouts/global/hooks/atoms/sidebar-atom.ts";
|
import { mobileSidebarAtom } from "@/components/layouts/global/hooks/atoms/sidebar-atom.ts";
|
||||||
import { useToggleSidebar } from "@/components/layouts/global/hooks/hooks/use-toggle-sidebar.ts";
|
import { useToggleSidebar } from "@/components/layouts/global/hooks/hooks/use-toggle-sidebar.ts";
|
||||||
import { useSettingsNavigation } from "@/hooks/use-settings-navigation";
|
import { useSettingsNavigation } from "@/hooks/use-settings-navigation";
|
||||||
|
|
||||||
interface DataItem {
|
type DataItem = {
|
||||||
label: string;
|
label: string;
|
||||||
icon: React.ElementType;
|
icon: React.ElementType;
|
||||||
path: string;
|
path: string;
|
||||||
isCloud?: boolean;
|
feature?: string;
|
||||||
isEnterprise?: boolean;
|
role?: "admin" | "owner";
|
||||||
isAdmin?: boolean;
|
env?: "cloud" | "selfhosted";
|
||||||
isSelfhosted?: boolean;
|
};
|
||||||
showDisabledInNonEE?: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface DataGroup {
|
type DataGroup = {
|
||||||
heading: string;
|
heading: string;
|
||||||
items: DataItem[];
|
items: DataItem[];
|
||||||
}
|
};
|
||||||
|
|
||||||
const groupedData: DataGroup[] = [
|
const groupedData: DataGroup[] = [
|
||||||
{
|
{
|
||||||
@@ -67,9 +69,7 @@ const groupedData: DataGroup[] = [
|
|||||||
label: "API keys",
|
label: "API keys",
|
||||||
icon: IconKey,
|
icon: IconKey,
|
||||||
path: "/settings/account/api-keys",
|
path: "/settings/account/api-keys",
|
||||||
isCloud: true,
|
feature: Feature.API_KEYS,
|
||||||
isEnterprise: true,
|
|
||||||
showDisabledInNonEE: true,
|
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -77,26 +77,20 @@ const groupedData: DataGroup[] = [
|
|||||||
heading: "Workspace",
|
heading: "Workspace",
|
||||||
items: [
|
items: [
|
||||||
{ label: "General", icon: IconSettings, path: "/settings/workspace" },
|
{ label: "General", icon: IconSettings, path: "/settings/workspace" },
|
||||||
{
|
{ label: "Members", icon: IconUsers, path: "/settings/members" },
|
||||||
label: "Members",
|
|
||||||
icon: IconUsers,
|
|
||||||
path: "/settings/members",
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
label: "Billing",
|
label: "Billing",
|
||||||
icon: IconCoin,
|
icon: IconCoin,
|
||||||
path: "/settings/billing",
|
path: "/settings/billing",
|
||||||
isCloud: true,
|
role: "admin",
|
||||||
isAdmin: true,
|
env: "cloud",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Security & SSO",
|
label: "Security & SSO",
|
||||||
icon: IconLock,
|
icon: IconLock,
|
||||||
path: "/settings/security",
|
path: "/settings/security",
|
||||||
isCloud: true,
|
feature: Feature.SECURITY_SETTINGS,
|
||||||
isEnterprise: true,
|
role: "admin",
|
||||||
isAdmin: true,
|
|
||||||
showDisabledInNonEE: true,
|
|
||||||
},
|
},
|
||||||
{ 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" },
|
||||||
@@ -105,17 +99,22 @@ const groupedData: DataGroup[] = [
|
|||||||
label: "API management",
|
label: "API management",
|
||||||
icon: IconKey,
|
icon: IconKey,
|
||||||
path: "/settings/api-keys",
|
path: "/settings/api-keys",
|
||||||
isCloud: true,
|
feature: Feature.API_KEYS,
|
||||||
isEnterprise: true,
|
role: "admin",
|
||||||
isAdmin: true,
|
|
||||||
showDisabledInNonEE: true,
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "AI settings",
|
label: "AI settings",
|
||||||
icon: IconSparkles,
|
icon: IconSparkles,
|
||||||
path: "/settings/ai",
|
path: "/settings/ai",
|
||||||
isAdmin: true,
|
role: "admin",
|
||||||
isSelfhosted: true,
|
},
|
||||||
|
{
|
||||||
|
label: "Audit log",
|
||||||
|
icon: IconHistory,
|
||||||
|
path: "/settings/audit",
|
||||||
|
feature: Feature.AUDIT_LOGS,
|
||||||
|
role: "owner",
|
||||||
|
env: "selfhosted",
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@@ -136,8 +135,9 @@ export default function SettingsSidebar() {
|
|||||||
const location = useLocation();
|
const location = useLocation();
|
||||||
const [active, setActive] = useState(location.pathname);
|
const [active, setActive] = useState(location.pathname);
|
||||||
const { goBack } = useSettingsNavigation();
|
const { goBack } = useSettingsNavigation();
|
||||||
const { isAdmin } = useUserRole();
|
const { isAdmin, isOwner } = useUserRole();
|
||||||
const [workspace] = useAtom(workspaceAtom);
|
const [entitlements] = useAtom(entitlementAtom);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
const [mobileSidebarOpened] = useAtom(mobileSidebarAtom);
|
const [mobileSidebarOpened] = useAtom(mobileSidebarAtom);
|
||||||
const toggleMobileSidebar = useToggleSidebar(mobileSidebarAtom);
|
const toggleMobileSidebar = useToggleSidebar(mobileSidebarAtom);
|
||||||
|
|
||||||
@@ -145,41 +145,20 @@ export default function SettingsSidebar() {
|
|||||||
setActive(location.pathname);
|
setActive(location.pathname);
|
||||||
}, [location.pathname]);
|
}, [location.pathname]);
|
||||||
|
|
||||||
|
const hasFeature = (f: string) =>
|
||||||
|
entitlements?.features?.includes(f) ?? false;
|
||||||
|
|
||||||
const canShowItem = (item: DataItem) => {
|
const canShowItem = (item: DataItem) => {
|
||||||
if (item.showDisabledInNonEE && item.isEnterprise) {
|
if (item.env === "cloud" && !isCloud()) return false;
|
||||||
// Check admin permission regardless of license
|
if (item.env === "selfhosted" && isCloud()) return false;
|
||||||
return item.isAdmin ? isAdmin : true;
|
if (item.role === "admin" && !isAdmin) return false;
|
||||||
}
|
if (item.role === "owner" && !isOwner) return false;
|
||||||
|
|
||||||
if (item.isCloud && item.isEnterprise) {
|
|
||||||
if (!(isCloud() || workspace?.hasLicenseKey)) return false;
|
|
||||||
return item.isAdmin ? isAdmin : true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (item.isCloud) {
|
|
||||||
return isCloud() ? (item.isAdmin ? isAdmin : true) : false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (item.isSelfhosted) {
|
|
||||||
return !isCloud() ? (item.isAdmin ? isAdmin : true) : false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (item.isEnterprise) {
|
|
||||||
return workspace?.hasLicenseKey ? (item.isAdmin ? isAdmin : true) : false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (item.isAdmin) {
|
|
||||||
return isAdmin;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
const isItemDisabled = (item: DataItem) => {
|
const isItemDisabled = (item: DataItem) => {
|
||||||
if (item.showDisabledInNonEE && item.isEnterprise) {
|
if (!item.feature) return false;
|
||||||
return !(isCloud() || workspace?.hasLicenseKey);
|
return !hasFeature(item.feature);
|
||||||
}
|
|
||||||
return false;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const menuItems = groupedData.map((group) => {
|
const menuItems = groupedData.map((group) => {
|
||||||
@@ -212,7 +191,7 @@ export default function SettingsSidebar() {
|
|||||||
prefetchHandler = prefetchBilling;
|
prefetchHandler = prefetchBilling;
|
||||||
break;
|
break;
|
||||||
case "License & Edition":
|
case "License & Edition":
|
||||||
if (workspace?.hasLicenseKey) {
|
if (entitlements?.tier !== "free") {
|
||||||
prefetchHandler = prefetchLicense;
|
prefetchHandler = prefetchLicense;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -228,6 +207,9 @@ export default function SettingsSidebar() {
|
|||||||
case "API management":
|
case "API management":
|
||||||
prefetchHandler = prefetchApiKeyManagement;
|
prefetchHandler = prefetchApiKeyManagement;
|
||||||
break;
|
break;
|
||||||
|
case "Audit log":
|
||||||
|
prefetchHandler = prefetchAuditLogs;
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -264,7 +246,7 @@ export default function SettingsSidebar() {
|
|||||||
return (
|
return (
|
||||||
<Tooltip
|
<Tooltip
|
||||||
key={item.label}
|
key={item.label}
|
||||||
label={t("Available in enterprise edition")}
|
label={upgradeLabel}
|
||||||
position="right"
|
position="right"
|
||||||
withArrow
|
withArrow
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -34,6 +34,7 @@ export function AutoTooltipText({
|
|||||||
disabled={!isTruncated || !label}
|
disabled={!isTruncated || !label}
|
||||||
multiline
|
multiline
|
||||||
withArrow
|
withArrow
|
||||||
|
withinPortal={false}
|
||||||
{...tooltipProps}
|
{...tooltipProps}
|
||||||
>
|
>
|
||||||
<Text
|
<Text
|
||||||
|
|||||||
@@ -0,0 +1,68 @@
|
|||||||
|
.root {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.track {
|
||||||
|
display: flex;
|
||||||
|
gap: var(--mantine-spacing-md);
|
||||||
|
overflow-x: auto;
|
||||||
|
scroll-snap-type: x mandatory;
|
||||||
|
scroll-behavior: smooth;
|
||||||
|
scrollbar-width: none;
|
||||||
|
-ms-overflow-style: none;
|
||||||
|
padding: 2px;
|
||||||
|
margin: -2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.track::-webkit-scrollbar {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.track > * {
|
||||||
|
scroll-snap-align: start;
|
||||||
|
flex: 0 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.arrow {
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
width: 32px;
|
||||||
|
height: 32px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border-radius: 50%;
|
||||||
|
border: 1px solid light-dark(var(--mantine-color-gray-3), var(--mantine-color-dark-4));
|
||||||
|
background: light-dark(var(--mantine-color-white), var(--mantine-color-dark-6));
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 0;
|
||||||
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);
|
||||||
|
opacity: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
transition: opacity 120ms ease, background-color 120ms ease, transform 120ms ease;
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.root:hover .arrow.visible,
|
||||||
|
.arrow.visible:focus-visible {
|
||||||
|
opacity: 1;
|
||||||
|
pointer-events: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.arrow:hover {
|
||||||
|
background: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-5));
|
||||||
|
}
|
||||||
|
|
||||||
|
.arrow:active {
|
||||||
|
transform: translateY(-50%) scale(0.95);
|
||||||
|
}
|
||||||
|
|
||||||
|
.arrowLeft {
|
||||||
|
left: -14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.arrowRight {
|
||||||
|
right: -14px;
|
||||||
|
}
|
||||||
@@ -0,0 +1,77 @@
|
|||||||
|
import { useCallback, useEffect, useRef, useState, type ReactNode } from "react";
|
||||||
|
import { IconChevronLeft, IconChevronRight } from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import classes from "./card-carousel.module.css";
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
children: ReactNode;
|
||||||
|
ariaLabel?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function CardCarousel({ children, ariaLabel }: Props) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const trackRef = useRef<HTMLDivElement>(null);
|
||||||
|
const [canScrollLeft, setCanScrollLeft] = useState(false);
|
||||||
|
const [canScrollRight, setCanScrollRight] = useState(false);
|
||||||
|
|
||||||
|
const updateScrollState = useCallback(() => {
|
||||||
|
const el = trackRef.current;
|
||||||
|
if (!el) return;
|
||||||
|
const maxScroll = el.scrollWidth - el.clientWidth;
|
||||||
|
setCanScrollLeft(el.scrollLeft > 1);
|
||||||
|
setCanScrollRight(el.scrollLeft < maxScroll - 1);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
updateScrollState();
|
||||||
|
const el = trackRef.current;
|
||||||
|
if (!el) return;
|
||||||
|
|
||||||
|
const observer = new ResizeObserver(updateScrollState);
|
||||||
|
observer.observe(el);
|
||||||
|
for (const child of Array.from(el.children)) {
|
||||||
|
observer.observe(child);
|
||||||
|
}
|
||||||
|
|
||||||
|
return () => observer.disconnect();
|
||||||
|
}, [updateScrollState, children]);
|
||||||
|
|
||||||
|
const scrollBy = (direction: 1 | -1) => {
|
||||||
|
const el = trackRef.current;
|
||||||
|
if (!el) return;
|
||||||
|
el.scrollBy({ left: direction * el.clientWidth * 0.85, behavior: "smooth" });
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.root}>
|
||||||
|
<div
|
||||||
|
ref={trackRef}
|
||||||
|
className={classes.track}
|
||||||
|
onScroll={updateScrollState}
|
||||||
|
{...(ariaLabel ? { role: "region", "aria-label": ariaLabel } : {})}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={`${classes.arrow} ${classes.arrowLeft} ${canScrollLeft ? classes.visible : ""}`}
|
||||||
|
onClick={() => scrollBy(-1)}
|
||||||
|
aria-label={t("Scroll left")}
|
||||||
|
tabIndex={canScrollLeft ? 0 : -1}
|
||||||
|
>
|
||||||
|
<IconChevronLeft size={18} />
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={`${classes.arrow} ${classes.arrowRight} ${canScrollRight ? classes.visible : ""}`}
|
||||||
|
onClick={() => scrollBy(1)}
|
||||||
|
aria-label={t("Scroll right")}
|
||||||
|
tabIndex={canScrollRight ? 0 : -1}
|
||||||
|
>
|
||||||
|
<IconChevronRight size={18} />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -4,7 +4,7 @@ import { getAvatarUrl } from "@/lib/config.ts";
|
|||||||
import { AvatarIconType } from "@/features/attachments/types/attachment.types.ts";
|
import { AvatarIconType } from "@/features/attachments/types/attachment.types.ts";
|
||||||
|
|
||||||
interface CustomAvatarProps {
|
interface CustomAvatarProps {
|
||||||
avatarUrl: string;
|
avatarUrl?: string;
|
||||||
name: string;
|
name: string;
|
||||||
color?: string;
|
color?: string;
|
||||||
size?: string | number;
|
size?: string | number;
|
||||||
|
|||||||
@@ -0,0 +1,67 @@
|
|||||||
|
import { useState, useEffect } from "react";
|
||||||
|
import { Modal, Button, Group } from "@mantine/core";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { DestinationPicker } from "./destination-picker";
|
||||||
|
import {
|
||||||
|
DestinationPickerModalProps,
|
||||||
|
DestinationSelection,
|
||||||
|
} from "./destination-picker.types";
|
||||||
|
|
||||||
|
export function DestinationPickerModal({
|
||||||
|
opened,
|
||||||
|
onClose,
|
||||||
|
title,
|
||||||
|
actionLabel,
|
||||||
|
onSelect,
|
||||||
|
loading,
|
||||||
|
excludePageId,
|
||||||
|
pageLimit,
|
||||||
|
}: DestinationPickerModalProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [selection, setSelection] = useState<DestinationSelection | null>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!opened) {
|
||||||
|
setSelection(null);
|
||||||
|
}
|
||||||
|
}, [opened]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal.Root
|
||||||
|
opened={opened}
|
||||||
|
onClose={onClose}
|
||||||
|
size={550}
|
||||||
|
padding="lg"
|
||||||
|
yOffset="10vh"
|
||||||
|
onClick={(e) => e.stopPropagation()}
|
||||||
|
>
|
||||||
|
<Modal.Overlay />
|
||||||
|
<Modal.Content>
|
||||||
|
<Modal.Header py={0}>
|
||||||
|
<Modal.Title fw={500}>{title}</Modal.Title>
|
||||||
|
<Modal.CloseButton />
|
||||||
|
</Modal.Header>
|
||||||
|
<Modal.Body>
|
||||||
|
<DestinationPicker
|
||||||
|
onSelectionChange={setSelection}
|
||||||
|
excludePageId={excludePageId}
|
||||||
|
pageLimit={pageLimit}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Group justify="flex-end" mt="md">
|
||||||
|
<Button variant="default" onClick={onClose}>
|
||||||
|
{t("Close")}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
onClick={() => selection && onSelect(selection)}
|
||||||
|
disabled={!selection}
|
||||||
|
loading={loading}
|
||||||
|
>
|
||||||
|
{actionLabel}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
</Modal.Body>
|
||||||
|
</Modal.Content>
|
||||||
|
</Modal.Root>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,128 @@
|
|||||||
|
.searchInput {
|
||||||
|
margin-bottom: var(--mantine-spacing-sm);
|
||||||
|
}
|
||||||
|
|
||||||
|
.scrollArea {
|
||||||
|
max-height: 50vh;
|
||||||
|
}
|
||||||
|
|
||||||
|
.row {
|
||||||
|
padding: 6px 12px;
|
||||||
|
border-radius: var(--mantine-radius-sm);
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
transition: background-color 150ms ease;
|
||||||
|
user-select: none;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
background-color: light-dark(
|
||||||
|
var(--mantine-color-gray-0),
|
||||||
|
var(--mantine-color-dark-6)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.selected {
|
||||||
|
background-color: light-dark(
|
||||||
|
var(--mantine-color-blue-0),
|
||||||
|
var(--mantine-color-dark-5)
|
||||||
|
);
|
||||||
|
border-left: 2px solid var(--mantine-primary-color-filled);
|
||||||
|
}
|
||||||
|
|
||||||
|
.spaceRow {
|
||||||
|
composes: row;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pageRow {
|
||||||
|
composes: row;
|
||||||
|
font-weight: 400;
|
||||||
|
}
|
||||||
|
|
||||||
|
.disabled {
|
||||||
|
opacity: 0.5;
|
||||||
|
cursor: not-allowed;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chevron {
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border-radius: var(--mantine-radius-sm);
|
||||||
|
flex-shrink: 0;
|
||||||
|
transition: transform 150ms ease;
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
background-color: light-dark(
|
||||||
|
var(--mantine-color-gray-1),
|
||||||
|
var(--mantine-color-dark-5)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.chevronExpanded {
|
||||||
|
transform: rotate(90deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.loadMore {
|
||||||
|
text-align: center;
|
||||||
|
padding: 6px;
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.selectedIndicator {
|
||||||
|
padding: 8px 12px;
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
border-top: 1px solid light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-4));
|
||||||
|
margin-top: var(--mantine-spacing-xs);
|
||||||
|
}
|
||||||
|
|
||||||
|
.emptyState {
|
||||||
|
padding: 12px;
|
||||||
|
text-align: center;
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchResult {
|
||||||
|
composes: row;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pageTitle {
|
||||||
|
flex: 1;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
}
|
||||||
|
|
||||||
|
.spaceName {
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.iconWrapper {
|
||||||
|
width: 22px;
|
||||||
|
height: 22px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
flex-shrink: 0;
|
||||||
|
font-size: 16px;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
@@ -0,0 +1,168 @@
|
|||||||
|
import { useState, useCallback } from "react";
|
||||||
|
import { TextInput, ScrollArea, Loader } from "@mantine/core";
|
||||||
|
import { useDebouncedValue } from "@mantine/hooks";
|
||||||
|
import { IconSearch, IconFile } from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { useGetSpacesQuery } from "@/features/space/queries/space-query";
|
||||||
|
import { useSearchSuggestionsQuery } from "@/features/search/queries/search-query";
|
||||||
|
import { ISpace } from "@/features/space/types/space.types";
|
||||||
|
import { IPage } from "@/features/page/types/page.types";
|
||||||
|
import { DestinationSelection } from "./destination-picker.types";
|
||||||
|
import { SpaceRow } from "./space-row";
|
||||||
|
import classes from "./destination-picker.module.css";
|
||||||
|
|
||||||
|
type DestinationPickerProps = {
|
||||||
|
onSelectionChange: (selection: DestinationSelection | null) => void;
|
||||||
|
excludePageId?: string;
|
||||||
|
pageLimit?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function DestinationPicker({
|
||||||
|
onSelectionChange,
|
||||||
|
excludePageId,
|
||||||
|
pageLimit = 15,
|
||||||
|
}: DestinationPickerProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [searchQuery, setSearchQuery] = useState("");
|
||||||
|
const [selection, setSelection] = useState<DestinationSelection | null>(null);
|
||||||
|
const [debouncedQuery] = useDebouncedValue(searchQuery, 300);
|
||||||
|
|
||||||
|
const { data: spacesData, isLoading: spacesLoading } = useGetSpacesQuery({
|
||||||
|
limit: 100,
|
||||||
|
});
|
||||||
|
|
||||||
|
const searchEnabled = debouncedQuery && debouncedQuery.length >= 2;
|
||||||
|
|
||||||
|
const { data: searchData, isLoading: searchLoading } =
|
||||||
|
useSearchSuggestionsQuery({
|
||||||
|
query: searchEnabled ? debouncedQuery : "",
|
||||||
|
includePages: true,
|
||||||
|
limit: 20,
|
||||||
|
});
|
||||||
|
|
||||||
|
const isSearching = !!searchEnabled;
|
||||||
|
|
||||||
|
const selectedId =
|
||||||
|
selection?.type === "space" ? selection.spaceId : selection?.pageId ?? null;
|
||||||
|
|
||||||
|
const updateSelection = useCallback(
|
||||||
|
(next: DestinationSelection | null) => {
|
||||||
|
setSelection(next);
|
||||||
|
onSelectionChange(next);
|
||||||
|
},
|
||||||
|
[onSelectionChange],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleSearchResultClick = (page: Partial<IPage>) => {
|
||||||
|
if (!page.space || !page.id) return;
|
||||||
|
|
||||||
|
updateSelection({
|
||||||
|
type: "page",
|
||||||
|
spaceId: page.space.id,
|
||||||
|
pageId: page.id,
|
||||||
|
page,
|
||||||
|
space: page.space,
|
||||||
|
});
|
||||||
|
setSearchQuery("");
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleSelectSpace = useCallback(
|
||||||
|
(space: ISpace) => {
|
||||||
|
updateSelection({ type: "space", spaceId: space.id, space });
|
||||||
|
},
|
||||||
|
[updateSelection],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleSelectPage = useCallback(
|
||||||
|
(page: Partial<IPage>, space: ISpace) => {
|
||||||
|
if (!page.id) return;
|
||||||
|
updateSelection({
|
||||||
|
type: "page",
|
||||||
|
spaceId: page.spaceId ?? space.id,
|
||||||
|
pageId: page.id,
|
||||||
|
page,
|
||||||
|
space,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
[updateSelection],
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<TextInput
|
||||||
|
leftSection={<IconSearch size={16} />}
|
||||||
|
placeholder={t("Search pages and spaces...")}
|
||||||
|
variant="filled"
|
||||||
|
value={searchQuery}
|
||||||
|
onChange={(e) => setSearchQuery(e.currentTarget.value)}
|
||||||
|
className={classes.searchInput}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ScrollArea h="50vh" offsetScrollbars className={classes.scrollArea}>
|
||||||
|
{isSearching ? (
|
||||||
|
searchLoading ? (
|
||||||
|
<div className={classes.emptyState}>
|
||||||
|
<Loader size="xs" />
|
||||||
|
</div>
|
||||||
|
) : searchData?.pages && searchData.pages.length > 0 ? (
|
||||||
|
searchData.pages.map(
|
||||||
|
(page) =>
|
||||||
|
page && (
|
||||||
|
<div
|
||||||
|
key={page.id}
|
||||||
|
className={classes.searchResult}
|
||||||
|
onClick={() => handleSearchResultClick(page)}
|
||||||
|
>
|
||||||
|
<div className={classes.iconWrapper}>
|
||||||
|
{page.icon ? (
|
||||||
|
page.icon
|
||||||
|
) : (
|
||||||
|
<IconFile
|
||||||
|
size={16}
|
||||||
|
color="var(--mantine-color-gray-5)"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className={classes.pageTitle}>
|
||||||
|
{page.title || t("Untitled")}
|
||||||
|
</div>
|
||||||
|
{page.space && (
|
||||||
|
<div className={classes.spaceName}>
|
||||||
|
{page.space.name}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
)
|
||||||
|
) : (
|
||||||
|
<div className={classes.emptyState}>{t("No results found")}</div>
|
||||||
|
)
|
||||||
|
) : spacesLoading ? (
|
||||||
|
<div className={classes.emptyState}>
|
||||||
|
<Loader size="xs" />
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
spacesData?.items?.map((space) => (
|
||||||
|
<SpaceRow
|
||||||
|
key={space.id}
|
||||||
|
space={space}
|
||||||
|
limit={pageLimit}
|
||||||
|
selectedId={selectedId}
|
||||||
|
excludePageId={excludePageId}
|
||||||
|
onSelectSpace={handleSelectSpace}
|
||||||
|
onSelectPage={handleSelectPage}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
)}
|
||||||
|
</ScrollArea>
|
||||||
|
|
||||||
|
{selection && (
|
||||||
|
<div className={classes.selectedIndicator}>
|
||||||
|
{selection.type === "space"
|
||||||
|
? selection.space.name
|
||||||
|
: `${selection.space.name} / ${selection.page.title || t("Untitled")}`}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
import { ISpace } from "@/features/space/types/space.types";
|
||||||
|
import { IPage } from "@/features/page/types/page.types";
|
||||||
|
|
||||||
|
export type DestinationSelection =
|
||||||
|
| { type: "space"; spaceId: string; space: ISpace }
|
||||||
|
| {
|
||||||
|
type: "page";
|
||||||
|
spaceId: string;
|
||||||
|
pageId: string;
|
||||||
|
page: Partial<IPage>;
|
||||||
|
space: Partial<ISpace>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type DestinationPickerModalProps = {
|
||||||
|
opened: boolean;
|
||||||
|
onClose: () => void;
|
||||||
|
title: string;
|
||||||
|
actionLabel: string;
|
||||||
|
onSelect: (selection: DestinationSelection) => void | Promise<void>;
|
||||||
|
loading?: boolean;
|
||||||
|
excludePageId?: string;
|
||||||
|
pageLimit?: number;
|
||||||
|
};
|
||||||
@@ -0,0 +1,83 @@
|
|||||||
|
import { useInfiniteQuery } from "@tanstack/react-query";
|
||||||
|
import { Loader } from "@mantine/core";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { getSidebarPages } from "@/features/page/services/page-service";
|
||||||
|
import { IPage } from "@/features/page/types/page.types";
|
||||||
|
import { IPagination } from "@/lib/types";
|
||||||
|
import { PageRow } from "./page-row";
|
||||||
|
import classes from "./destination-picker.module.css";
|
||||||
|
|
||||||
|
type PageChildrenProps = {
|
||||||
|
spaceId: string;
|
||||||
|
pageId?: string;
|
||||||
|
depth: number;
|
||||||
|
limit: number;
|
||||||
|
selectedId: string | null;
|
||||||
|
excludePageId?: string;
|
||||||
|
onSelectPage: (page: Partial<IPage>) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function PageChildren({
|
||||||
|
spaceId,
|
||||||
|
pageId,
|
||||||
|
depth,
|
||||||
|
limit,
|
||||||
|
selectedId,
|
||||||
|
excludePageId,
|
||||||
|
onSelectPage,
|
||||||
|
}: PageChildrenProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const { data, isLoading, hasNextPage, fetchNextPage } = useInfiniteQuery({
|
||||||
|
queryKey: ["destination-pages", spaceId, pageId ?? "root"],
|
||||||
|
queryFn: ({ pageParam }) =>
|
||||||
|
getSidebarPages({
|
||||||
|
spaceId,
|
||||||
|
pageId,
|
||||||
|
limit,
|
||||||
|
cursor: pageParam,
|
||||||
|
}),
|
||||||
|
initialPageParam: undefined as string | undefined,
|
||||||
|
getNextPageParam: (lastPage: IPagination<IPage>) =>
|
||||||
|
lastPage.meta?.nextCursor ?? undefined,
|
||||||
|
});
|
||||||
|
|
||||||
|
const pages = data?.pages.flatMap((page) => page.items) ?? [];
|
||||||
|
|
||||||
|
if (isLoading) {
|
||||||
|
return (
|
||||||
|
<div className={classes.emptyState}>
|
||||||
|
<Loader size="xs" />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pages.length === 0) {
|
||||||
|
return (
|
||||||
|
<div className={classes.emptyState}>
|
||||||
|
{pageId ? t("No pages inside") : t("No pages in this space")}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{pages.map((page) => (
|
||||||
|
<PageRow
|
||||||
|
key={page.id}
|
||||||
|
page={page}
|
||||||
|
depth={depth}
|
||||||
|
limit={limit}
|
||||||
|
selectedId={selectedId}
|
||||||
|
excludePageId={excludePageId}
|
||||||
|
onSelect={onSelectPage}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
{hasNextPage && (
|
||||||
|
<div className={classes.loadMore} onClick={() => fetchNextPage()}>
|
||||||
|
{t("Load more")}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,89 @@
|
|||||||
|
import { useState } from "react";
|
||||||
|
import { IconChevronRight, IconFile } from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { IPage } from "@/features/page/types/page.types";
|
||||||
|
import { PageChildren } from "./page-children";
|
||||||
|
import classes from "./destination-picker.module.css";
|
||||||
|
|
||||||
|
type PageRowProps = {
|
||||||
|
page: Partial<IPage>;
|
||||||
|
depth: number;
|
||||||
|
limit: number;
|
||||||
|
selectedId: string | null;
|
||||||
|
excludePageId?: string;
|
||||||
|
onSelect: (page: Partial<IPage>) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function PageRow({
|
||||||
|
page,
|
||||||
|
depth,
|
||||||
|
limit,
|
||||||
|
selectedId,
|
||||||
|
excludePageId,
|
||||||
|
onSelect,
|
||||||
|
}: PageRowProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [expanded, setExpanded] = useState(false);
|
||||||
|
|
||||||
|
const isExcluded = page.id === excludePageId;
|
||||||
|
const isSelected = page.id === selectedId;
|
||||||
|
|
||||||
|
const rowClasses = [
|
||||||
|
classes.pageRow,
|
||||||
|
isSelected && classes.selected,
|
||||||
|
isExcluded && classes.disabled,
|
||||||
|
]
|
||||||
|
.filter(Boolean)
|
||||||
|
.join(" ");
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<div
|
||||||
|
className={rowClasses}
|
||||||
|
style={{ paddingLeft: depth * 20 + 12 }}
|
||||||
|
onClick={() => !isExcluded && onSelect(page)}
|
||||||
|
>
|
||||||
|
{page.hasChildren ? (
|
||||||
|
<div
|
||||||
|
className={`${classes.chevron} ${expanded ? classes.chevronExpanded : ""}`}
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
setExpanded(!expanded);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<IconChevronRight size={14} />
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div style={{ width: 20, flexShrink: 0 }} />
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div className={classes.iconWrapper}>
|
||||||
|
{page.icon ? (
|
||||||
|
page.icon
|
||||||
|
) : (
|
||||||
|
<IconFile
|
||||||
|
size={16}
|
||||||
|
color="var(--mantine-color-gray-5)"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className={classes.pageTitle}>
|
||||||
|
{page.title || t("Untitled")}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{expanded && page.hasChildren && (
|
||||||
|
<PageChildren
|
||||||
|
spaceId={page.spaceId}
|
||||||
|
pageId={page.id}
|
||||||
|
depth={depth + 1}
|
||||||
|
limit={limit}
|
||||||
|
selectedId={selectedId}
|
||||||
|
excludePageId={excludePageId}
|
||||||
|
onSelectPage={onSelect}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,108 @@
|
|||||||
|
import { useState } from "react";
|
||||||
|
import { Tooltip } from "@mantine/core";
|
||||||
|
import { IconChevronRight, IconLock } from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { ISpace } from "@/features/space/types/space.types";
|
||||||
|
import { IPage } from "@/features/page/types/page.types";
|
||||||
|
import { SpaceRole } from "@/lib/types";
|
||||||
|
import { CustomAvatar } from "@/components/ui/custom-avatar";
|
||||||
|
import { AvatarIconType } from "@/features/attachments/types/attachment.types";
|
||||||
|
import { PageChildren } from "./page-children";
|
||||||
|
import classes from "./destination-picker.module.css";
|
||||||
|
|
||||||
|
type SpaceRowProps = {
|
||||||
|
space: ISpace;
|
||||||
|
limit: number;
|
||||||
|
selectedId: string | null;
|
||||||
|
excludePageId?: string;
|
||||||
|
onSelectSpace: (space: ISpace) => void;
|
||||||
|
onSelectPage: (page: Partial<IPage>, space: ISpace) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function SpaceRow({
|
||||||
|
space,
|
||||||
|
limit,
|
||||||
|
selectedId,
|
||||||
|
excludePageId,
|
||||||
|
onSelectSpace,
|
||||||
|
onSelectPage,
|
||||||
|
}: SpaceRowProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [expanded, setExpanded] = useState(false);
|
||||||
|
|
||||||
|
const writable =
|
||||||
|
!!space.membership?.role && space.membership.role !== SpaceRole.READER;
|
||||||
|
const isSelected = space.id === selectedId;
|
||||||
|
|
||||||
|
const rowClasses = [
|
||||||
|
classes.spaceRow,
|
||||||
|
isSelected && classes.selected,
|
||||||
|
!writable && classes.disabled,
|
||||||
|
]
|
||||||
|
.filter(Boolean)
|
||||||
|
.join(" ");
|
||||||
|
|
||||||
|
const rowContent = (
|
||||||
|
<div
|
||||||
|
className={rowClasses}
|
||||||
|
onClick={() => writable && onSelectSpace(space)}
|
||||||
|
>
|
||||||
|
{writable ? (
|
||||||
|
<div
|
||||||
|
className={`${classes.chevron} ${expanded ? classes.chevronExpanded : ""}`}
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
setExpanded(!expanded);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<IconChevronRight size={14} />
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div style={{ width: 20, flexShrink: 0 }} />
|
||||||
|
)}
|
||||||
|
|
||||||
|
<CustomAvatar
|
||||||
|
name={space.name}
|
||||||
|
avatarUrl={space.logo}
|
||||||
|
type={AvatarIconType.SPACE_ICON}
|
||||||
|
size={22}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<div className={classes.pageTitle}>{space.name}</div>
|
||||||
|
|
||||||
|
{!writable && (
|
||||||
|
<IconLock
|
||||||
|
size={14}
|
||||||
|
color="var(--mantine-color-gray-5)"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{writable ? (
|
||||||
|
rowContent
|
||||||
|
) : (
|
||||||
|
<Tooltip
|
||||||
|
label={t("You don't have permission to create pages here")}
|
||||||
|
position="right"
|
||||||
|
withArrow
|
||||||
|
>
|
||||||
|
<div>{rowContent}</div>
|
||||||
|
</Tooltip>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{expanded && writable && (
|
||||||
|
<PageChildren
|
||||||
|
spaceId={space.id}
|
||||||
|
depth={1}
|
||||||
|
limit={limit}
|
||||||
|
selectedId={selectedId}
|
||||||
|
excludePageId={excludePageId}
|
||||||
|
onSelectPage={(page) => onSelectPage(page, space)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
.root {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 60px 20px;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
import { Stack, Text } from "@mantine/core";
|
||||||
|
import { type TablerIcon } from "@tabler/icons-react";
|
||||||
|
import { ReactNode } from "react";
|
||||||
|
import classes from "./empty-state.module.css";
|
||||||
|
|
||||||
|
type EmptyStateProps = {
|
||||||
|
icon: TablerIcon;
|
||||||
|
title: string;
|
||||||
|
description?: string;
|
||||||
|
action?: ReactNode;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function EmptyState({ icon: Icon, title, description, action }: EmptyStateProps) {
|
||||||
|
return (
|
||||||
|
<div className={classes.root}>
|
||||||
|
<Stack align="center" gap="xs">
|
||||||
|
<Icon size={40} stroke={1.5} color="var(--mantine-color-dimmed)" />
|
||||||
|
<Text size="lg" fw={500}>
|
||||||
|
{title}
|
||||||
|
</Text>
|
||||||
|
{description && (
|
||||||
|
<Text size="sm" c="dimmed" maw={350}>
|
||||||
|
{description}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
{action}
|
||||||
|
</Stack>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,106 @@
|
|||||||
|
import { useEffect, useRef } from "react";
|
||||||
|
import { useLocation, useNavigate, useParams } from "react-router-dom";
|
||||||
|
import { useChatInfoQuery } from "../queries/ai-chat-query";
|
||||||
|
import { useChatStream } from "../hooks/use-chat-stream";
|
||||||
|
import ChatMessageList from "./chat-message-list";
|
||||||
|
import ChatEmptyState from "./chat-empty-state";
|
||||||
|
import ChatInput from "./chat-input";
|
||||||
|
import type { HomeAiPromptInitialState } from "@/features/home/components/home-ai-prompt";
|
||||||
|
import classes from "../styles/ai-chat.module.css";
|
||||||
|
|
||||||
|
export default function AiChatLayout() {
|
||||||
|
const { chatId } = useParams<{ chatId: string }>();
|
||||||
|
const location = useLocation();
|
||||||
|
const navigate = useNavigate();
|
||||||
|
const chatInfoQuery = useChatInfoQuery(chatId);
|
||||||
|
|
||||||
|
// If the URL points at a chat the user does not own, the info fetch 404s.
|
||||||
|
// Bounce them back to /ai so they cannot interact with any chat UI (including
|
||||||
|
// kicking off orphan uploads) tied to a chat they have no access to.
|
||||||
|
useEffect(() => {
|
||||||
|
if (chatId && chatInfoQuery.isError) {
|
||||||
|
navigate("/ai", { replace: true });
|
||||||
|
}
|
||||||
|
}, [chatId, chatInfoQuery.isError, navigate]);
|
||||||
|
const {
|
||||||
|
messages,
|
||||||
|
streamingContent,
|
||||||
|
streamingToolCalls,
|
||||||
|
isStreaming,
|
||||||
|
error,
|
||||||
|
sendMessage,
|
||||||
|
stopGeneration,
|
||||||
|
hydrateFromServer,
|
||||||
|
} = useChatStream(chatId);
|
||||||
|
|
||||||
|
const autoSentRef = useRef(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (chatInfoQuery.data?.messages) {
|
||||||
|
hydrateFromServer(chatInfoQuery.data.messages);
|
||||||
|
}
|
||||||
|
}, [chatInfoQuery.data, hydrateFromServer]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (autoSentRef.current || chatId) return;
|
||||||
|
const state = location.state as HomeAiPromptInitialState | null;
|
||||||
|
if (!state?.initialContent && !state?.initialAttachments?.length) return;
|
||||||
|
|
||||||
|
autoSentRef.current = true;
|
||||||
|
sendMessage(
|
||||||
|
state.initialContent ?? "",
|
||||||
|
state.initialMentions ?? [],
|
||||||
|
state.initialAttachments ?? [],
|
||||||
|
);
|
||||||
|
navigate(location.pathname, { replace: true, state: null });
|
||||||
|
}, [chatId, location, navigate, sendMessage]);
|
||||||
|
|
||||||
|
const hasMessages = messages.length > 0 || isStreaming;
|
||||||
|
|
||||||
|
// While the redirect effect is running (or if the user is still on this
|
||||||
|
// component for any reason) never render the chat UI for a forbidden chat.
|
||||||
|
if (chatId && chatInfoQuery.isError) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.main}>
|
||||||
|
{error && (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
padding: "var(--mantine-spacing-sm) var(--mantine-spacing-lg)",
|
||||||
|
color: "var(--mantine-color-red-6)",
|
||||||
|
fontSize: "var(--mantine-font-size-sm)",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{error}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{hasMessages ? (
|
||||||
|
<>
|
||||||
|
<ChatMessageList
|
||||||
|
messages={messages}
|
||||||
|
isStreaming={isStreaming}
|
||||||
|
streamingContent={streamingContent}
|
||||||
|
streamingToolCalls={streamingToolCalls}
|
||||||
|
/>
|
||||||
|
<div className={classes.inputArea}>
|
||||||
|
<ChatInput
|
||||||
|
isStreaming={isStreaming}
|
||||||
|
onSend={sendMessage}
|
||||||
|
onStop={stopGeneration}
|
||||||
|
chatId={chatId}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<ChatEmptyState
|
||||||
|
isStreaming={isStreaming}
|
||||||
|
onSend={sendMessage}
|
||||||
|
onStop={stopGeneration}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,166 @@
|
|||||||
|
import { useState, useRef, useEffect, useMemo, useCallback } from "react";
|
||||||
|
import { ActionIcon, Menu, TextInput } from "@mantine/core";
|
||||||
|
import { IconDots, IconTrash, IconEdit } from "@tabler/icons-react";
|
||||||
|
import { Link } from "react-router-dom";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import type { AiChat } from "../types/ai-chat.types";
|
||||||
|
import classes from "../styles/chat-sidebar.module.css";
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
chat: AiChat;
|
||||||
|
isActive: boolean;
|
||||||
|
onDelete: (chatId: string) => void;
|
||||||
|
onRename: (chatId: string, title: string) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
function formatChatDate(
|
||||||
|
isoString: string | Date,
|
||||||
|
locale: string | undefined,
|
||||||
|
): string {
|
||||||
|
const date = new Date(isoString);
|
||||||
|
if (Number.isNaN(date.getTime())) return "";
|
||||||
|
|
||||||
|
const now = new Date();
|
||||||
|
const startOfToday = new Date(
|
||||||
|
now.getFullYear(),
|
||||||
|
now.getMonth(),
|
||||||
|
now.getDate(),
|
||||||
|
).getTime();
|
||||||
|
const ts = date.getTime();
|
||||||
|
const sameYear = date.getFullYear() === now.getFullYear();
|
||||||
|
|
||||||
|
if (ts >= startOfToday) {
|
||||||
|
return date.toLocaleTimeString(locale, {
|
||||||
|
hour: "numeric",
|
||||||
|
minute: "2-digit",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sameYear) {
|
||||||
|
return date.toLocaleDateString(locale, {
|
||||||
|
month: "short",
|
||||||
|
day: "numeric",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return date.toLocaleDateString(locale, {
|
||||||
|
month: "short",
|
||||||
|
day: "numeric",
|
||||||
|
year: "numeric",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function AiChatSidebarItem({
|
||||||
|
chat,
|
||||||
|
isActive,
|
||||||
|
onDelete,
|
||||||
|
onRename,
|
||||||
|
}: Props) {
|
||||||
|
const { t, i18n } = useTranslation();
|
||||||
|
const [renaming, setRenaming] = useState(false);
|
||||||
|
const [renameValue, setRenameValue] = useState("");
|
||||||
|
const inputRef = useRef<HTMLInputElement>(null);
|
||||||
|
|
||||||
|
const formattedDate = useMemo(
|
||||||
|
() => formatChatDate(chat.updatedAt, i18n.language),
|
||||||
|
[chat.updatedAt, i18n.language],
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (renaming) {
|
||||||
|
// Wait for the input to be mounted before selecting.
|
||||||
|
const id = window.setTimeout(() => inputRef.current?.select(), 0);
|
||||||
|
return () => window.clearTimeout(id);
|
||||||
|
}
|
||||||
|
}, [renaming]);
|
||||||
|
|
||||||
|
const startRename = useCallback(() => {
|
||||||
|
setRenameValue(chat.title || "");
|
||||||
|
setRenaming(true);
|
||||||
|
}, [chat.title]);
|
||||||
|
|
||||||
|
const submitRename = useCallback(() => {
|
||||||
|
const trimmed = renameValue.trim();
|
||||||
|
if (trimmed && trimmed !== chat.title) {
|
||||||
|
onRename(chat.id, trimmed);
|
||||||
|
}
|
||||||
|
setRenaming(false);
|
||||||
|
}, [renameValue, chat.id, chat.title, onRename]);
|
||||||
|
|
||||||
|
if (renaming) {
|
||||||
|
return (
|
||||||
|
<div className={classes.chatItem} data-active={isActive || undefined}>
|
||||||
|
<TextInput
|
||||||
|
ref={inputRef}
|
||||||
|
size="xs"
|
||||||
|
variant="unstyled"
|
||||||
|
placeholder={t("Chat name")}
|
||||||
|
value={renameValue}
|
||||||
|
onChange={(e) => setRenameValue(e.currentTarget.value)}
|
||||||
|
onKeyDown={(e) => {
|
||||||
|
if (e.key === "Enter") {
|
||||||
|
e.preventDefault();
|
||||||
|
submitRename();
|
||||||
|
} else if (e.key === "Escape") {
|
||||||
|
e.preventDefault();
|
||||||
|
setRenaming(false);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
onBlur={submitRename}
|
||||||
|
classNames={{ input: classes.chatItemRenameInput }}
|
||||||
|
style={{ flex: 1 }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Link
|
||||||
|
to={`/ai/chat/${chat.id}`}
|
||||||
|
className={classes.chatItem}
|
||||||
|
data-active={isActive || undefined}
|
||||||
|
>
|
||||||
|
<span className={classes.chatItemTitle}>
|
||||||
|
{chat.title || t("Untitled chat")}
|
||||||
|
</span>
|
||||||
|
<span className={classes.chatItemDate}>{formattedDate}</span>
|
||||||
|
<div className={classes.chatItemActions}>
|
||||||
|
<Menu position="bottom-end" withinPortal>
|
||||||
|
<Menu.Target>
|
||||||
|
<ActionIcon
|
||||||
|
variant="subtle"
|
||||||
|
size="xs"
|
||||||
|
color="gray"
|
||||||
|
onClick={(e) => e.preventDefault()}
|
||||||
|
>
|
||||||
|
<IconDots size={14} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Menu.Target>
|
||||||
|
<Menu.Dropdown>
|
||||||
|
<Menu.Item
|
||||||
|
leftSection={<IconEdit size={14} />}
|
||||||
|
onClick={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
startRename();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{t("Rename")}
|
||||||
|
</Menu.Item>
|
||||||
|
<Menu.Item
|
||||||
|
leftSection={<IconTrash size={14} />}
|
||||||
|
color="red"
|
||||||
|
onClick={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
onDelete(chat.id);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{t("Delete")}
|
||||||
|
</Menu.Item>
|
||||||
|
</Menu.Dropdown>
|
||||||
|
</Menu>
|
||||||
|
</div>
|
||||||
|
</Link>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,180 @@
|
|||||||
|
import { useState, useCallback, useEffect, useMemo, useRef } from "react";
|
||||||
|
import { Link, useNavigate, useParams } from "react-router-dom";
|
||||||
|
import { ActionIcon, Center, TextInput, Loader, Tooltip } from "@mantine/core";
|
||||||
|
import { useDebouncedValue } from "@mantine/hooks";
|
||||||
|
import { IconPlus, IconSearch, IconMessageCircle2 } from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import {
|
||||||
|
useChatsQuery,
|
||||||
|
useDeleteChatMutation,
|
||||||
|
useUpdateChatTitleMutation,
|
||||||
|
useSearchChatsQuery,
|
||||||
|
} from "../queries/ai-chat-query";
|
||||||
|
import AiChatSidebarItem from "./ai-chat-sidebar-item";
|
||||||
|
import { groupChatsByAge } from "../utils/group-chats-by-age";
|
||||||
|
import classes from "../styles/chat-sidebar.module.css";
|
||||||
|
|
||||||
|
export default function AiChatSidebar() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const navigate = useNavigate();
|
||||||
|
const { chatId } = useParams<{ chatId: string }>();
|
||||||
|
const [search, setSearch] = useState("");
|
||||||
|
const [debouncedSearch] = useDebouncedValue(search, 300);
|
||||||
|
const chatsQuery = useChatsQuery();
|
||||||
|
const searchQuery = useSearchChatsQuery(debouncedSearch);
|
||||||
|
const deleteMutation = useDeleteChatMutation();
|
||||||
|
const renameMutation = useUpdateChatTitleMutation();
|
||||||
|
|
||||||
|
const chats = useMemo(() => {
|
||||||
|
if (debouncedSearch) {
|
||||||
|
return searchQuery.data || [];
|
||||||
|
}
|
||||||
|
return chatsQuery.data?.pages.flatMap((p) => p.items) || [];
|
||||||
|
}, [debouncedSearch, searchQuery.data, chatsQuery.data]);
|
||||||
|
|
||||||
|
const groupedChats = useMemo(() => groupChatsByAge(chats, t), [chats, t]);
|
||||||
|
|
||||||
|
const sentinelRef = useRef<HTMLDivElement>(null);
|
||||||
|
const { hasNextPage, fetchNextPage, isFetchingNextPage } = chatsQuery;
|
||||||
|
const isSearching = Boolean(debouncedSearch);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (isSearching) return;
|
||||||
|
const sentinel = sentinelRef.current;
|
||||||
|
if (!sentinel) return;
|
||||||
|
|
||||||
|
const observer = new IntersectionObserver(
|
||||||
|
(entries) => {
|
||||||
|
if (entries[0].isIntersecting && hasNextPage && !isFetchingNextPage) {
|
||||||
|
fetchNextPage();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ threshold: 0.1 },
|
||||||
|
);
|
||||||
|
|
||||||
|
observer.observe(sentinel);
|
||||||
|
return () => observer.disconnect();
|
||||||
|
}, [isSearching, hasNextPage, isFetchingNextPage, fetchNextPage]);
|
||||||
|
|
||||||
|
const handleNewChat = useCallback(
|
||||||
|
(event: React.MouseEvent<HTMLAnchorElement>) => {
|
||||||
|
if (
|
||||||
|
event.button !== 0 ||
|
||||||
|
event.ctrlKey ||
|
||||||
|
event.metaKey ||
|
||||||
|
event.shiftKey
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
event.preventDefault();
|
||||||
|
navigate("/ai");
|
||||||
|
},
|
||||||
|
[navigate],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleDelete = useCallback(
|
||||||
|
(id: string) => {
|
||||||
|
deleteMutation.mutate(id, {
|
||||||
|
onSuccess: () => {
|
||||||
|
if (chatId === id) {
|
||||||
|
navigate("/ai");
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
},
|
||||||
|
[deleteMutation, chatId, navigate],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleRename = useCallback(
|
||||||
|
(chatId: string, title: string) => {
|
||||||
|
renameMutation.mutate({ chatId, title });
|
||||||
|
},
|
||||||
|
[renameMutation],
|
||||||
|
);
|
||||||
|
|
||||||
|
const isLoading = chatsQuery.isLoading || searchQuery.isLoading;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.sidebar}>
|
||||||
|
<div className={classes.header}>
|
||||||
|
<span className={classes.title}>{t("AI Chat")}</span>
|
||||||
|
<Tooltip label={t("New chat")} openDelay={250} withArrow>
|
||||||
|
<ActionIcon
|
||||||
|
component={Link}
|
||||||
|
to="/ai"
|
||||||
|
variant="subtle"
|
||||||
|
color="gray"
|
||||||
|
onClick={handleNewChat}
|
||||||
|
aria-label={t("New chat")}
|
||||||
|
>
|
||||||
|
<IconPlus size={18} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<TextInput
|
||||||
|
className={classes.searchInput}
|
||||||
|
placeholder="Search chats..."
|
||||||
|
leftSection={<IconSearch size={14} />}
|
||||||
|
size="xs"
|
||||||
|
value={search}
|
||||||
|
onChange={(e) => setSearch(e.currentTarget.value)}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<div className={classes.chatList}>
|
||||||
|
{isLoading && <Loader size="xs" mx="auto" mt="md" />}
|
||||||
|
{!isLoading && chats.length === 0 && (
|
||||||
|
<div className={classes.chatListEmpty}>
|
||||||
|
<IconMessageCircle2
|
||||||
|
size={28}
|
||||||
|
stroke={1.5}
|
||||||
|
className={classes.chatListEmptyIcon}
|
||||||
|
/>
|
||||||
|
<div className={classes.chatListEmptyTitle}>
|
||||||
|
{isSearching ? t("No chats found") : t("No conversations yet")}
|
||||||
|
</div>
|
||||||
|
<div className={classes.chatListEmptyHint}>
|
||||||
|
{isSearching
|
||||||
|
? t("Try a different search term.")
|
||||||
|
: t("Start a new chat to see it here.")}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{isSearching
|
||||||
|
? chats.map((chat) => (
|
||||||
|
<AiChatSidebarItem
|
||||||
|
key={chat.id}
|
||||||
|
chat={chat}
|
||||||
|
isActive={chat.id === chatId}
|
||||||
|
onDelete={handleDelete}
|
||||||
|
onRename={handleRename}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
: groupedChats.map((group) => (
|
||||||
|
<div key={group.key} className={classes.chatGroup}>
|
||||||
|
<div className={classes.chatGroupLabel}>{group.label}</div>
|
||||||
|
{group.chats.map((chat) => (
|
||||||
|
<AiChatSidebarItem
|
||||||
|
key={chat.id}
|
||||||
|
chat={chat}
|
||||||
|
isActive={chat.id === chatId}
|
||||||
|
onDelete={handleDelete}
|
||||||
|
onRename={handleRename}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
{!isSearching && (
|
||||||
|
<>
|
||||||
|
<div ref={sentinelRef} style={{ height: 1 }} />
|
||||||
|
{isFetchingNextPage && (
|
||||||
|
<Center py="xs">
|
||||||
|
<Loader size="xs" />
|
||||||
|
</Center>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,67 @@
|
|||||||
|
import { useState } from "react";
|
||||||
|
import { TextInput, Loader, Text, ScrollArea } from "@mantine/core";
|
||||||
|
import { IconSearch } from "@tabler/icons-react";
|
||||||
|
import { useChatsQuery, useSearchChatsQuery } from "../queries/ai-chat-query";
|
||||||
|
import { useDebouncedValue } from "@mantine/hooks";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import classes from "../styles/aside-chat-panel.module.css";
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
activeChatId: string | undefined;
|
||||||
|
onSelect: (chatId: string) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function AsideChatHistory({ activeChatId, onSelect }: Props) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [searchValue, setSearchValue] = useState("");
|
||||||
|
const [debouncedSearch] = useDebouncedValue(searchValue, 300);
|
||||||
|
|
||||||
|
const chatsQuery = useChatsQuery();
|
||||||
|
const searchQuery = useSearchChatsQuery(debouncedSearch);
|
||||||
|
|
||||||
|
const isSearching = debouncedSearch.length > 0;
|
||||||
|
const chats = isSearching
|
||||||
|
? (searchQuery.data ?? [])
|
||||||
|
: (chatsQuery.data?.pages.flatMap((p) => p.items) ?? []);
|
||||||
|
const isLoading = isSearching ? searchQuery.isLoading : chatsQuery.isLoading;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<TextInput
|
||||||
|
placeholder={t("Search chats...")}
|
||||||
|
leftSection={<IconSearch size={14} />}
|
||||||
|
size="xs"
|
||||||
|
mb="xs"
|
||||||
|
value={searchValue}
|
||||||
|
onChange={(e) => setSearchValue(e.currentTarget.value)}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{isLoading ? (
|
||||||
|
<div style={{ display: "flex", justifyContent: "center", padding: 16 }}>
|
||||||
|
<Loader size="sm" />
|
||||||
|
</div>
|
||||||
|
) : chats.length === 0 ? (
|
||||||
|
<Text size="sm" c="dimmed" ta="center" py="md">
|
||||||
|
{isSearching ? t("No chats found") : t("No chat history")}
|
||||||
|
</Text>
|
||||||
|
) : (
|
||||||
|
<ScrollArea.Autosize mah={300} scrollbars="y">
|
||||||
|
<div className={classes.historyList}>
|
||||||
|
{chats.map((chat) => (
|
||||||
|
<div
|
||||||
|
key={chat.id}
|
||||||
|
className={classes.historyItem}
|
||||||
|
data-active={chat.id === activeChatId || undefined}
|
||||||
|
onClick={() => onSelect(chat.id)}
|
||||||
|
>
|
||||||
|
<span className={classes.historyItemTitle}>
|
||||||
|
{chat.title || t("Untitled chat")}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</ScrollArea.Autosize>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,258 @@
|
|||||||
|
import { useState, useEffect, useCallback } from "react";
|
||||||
|
import { ActionIcon, Popover, Tooltip, UnstyledButton } from "@mantine/core";
|
||||||
|
import {
|
||||||
|
IconPlus,
|
||||||
|
IconChevronDown,
|
||||||
|
IconArrowsDiagonal,
|
||||||
|
IconX,
|
||||||
|
IconSparkles,
|
||||||
|
IconFileText,
|
||||||
|
IconLanguage,
|
||||||
|
IconSearch,
|
||||||
|
} from "@tabler/icons-react";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { useNavigate, useParams } from "react-router-dom";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { asideStateAtom } from "@/components/layouts/global/hooks/atoms/sidebar-atom";
|
||||||
|
import { usePageQuery } from "@/features/page/queries/page-query";
|
||||||
|
import { extractPageSlugId } from "@/lib";
|
||||||
|
import { useChatStream } from "../hooks/use-chat-stream";
|
||||||
|
import { useChatInfoQuery } from "../queries/ai-chat-query";
|
||||||
|
import ChatMessageList from "./chat-message-list";
|
||||||
|
import ChatInput from "./chat-input";
|
||||||
|
import AsideChatHistory from "./aside-chat-history";
|
||||||
|
import type { ChatAttachment, PageMention } from "../types/ai-chat.types";
|
||||||
|
import classes from "../styles/aside-chat-panel.module.css";
|
||||||
|
|
||||||
|
type QuickAction = {
|
||||||
|
icon: React.ReactNode;
|
||||||
|
label: string;
|
||||||
|
prompt: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function AsideChatPanel() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const navigate = useNavigate();
|
||||||
|
const [, setAsideState] = useAtom(asideStateAtom);
|
||||||
|
const [chatId, setChatId] = useState<string | undefined>(undefined);
|
||||||
|
const [historyOpen, setHistoryOpen] = useState(false);
|
||||||
|
const [contextPages, setContextPages] = useState<PageMention[]>([]);
|
||||||
|
const { pageSlug } = useParams();
|
||||||
|
const slugId = extractPageSlugId(pageSlug);
|
||||||
|
const { data: page } = usePageQuery({ pageId: slugId });
|
||||||
|
|
||||||
|
const chatInfoQuery = useChatInfoQuery(chatId);
|
||||||
|
const {
|
||||||
|
messages,
|
||||||
|
streamingContent,
|
||||||
|
streamingToolCalls,
|
||||||
|
isStreaming,
|
||||||
|
error,
|
||||||
|
sendMessage,
|
||||||
|
stopGeneration,
|
||||||
|
hydrateFromServer,
|
||||||
|
} = useChatStream(chatId, {
|
||||||
|
onChatCreated: (newChatId) => {
|
||||||
|
setChatId(newChatId);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (page && !chatId) {
|
||||||
|
setContextPages([{ id: page.id, title: page.title || "", slugId: page.slugId }]);
|
||||||
|
}
|
||||||
|
}, [page, chatId]);
|
||||||
|
|
||||||
|
const handleRemoveContextPage = useCallback((pageId: string) => {
|
||||||
|
setContextPages((prev) => prev.filter((p) => p.id !== pageId));
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (chatInfoQuery.data?.messages) {
|
||||||
|
hydrateFromServer(chatInfoQuery.data.messages);
|
||||||
|
}
|
||||||
|
}, [chatInfoQuery.data, hydrateFromServer]);
|
||||||
|
|
||||||
|
// Drop the open chatId if the current user lost access to it (404/403 on
|
||||||
|
// the info fetch). Reverts the panel to a fresh chat instead of presenting
|
||||||
|
// an input tied to a chat the user does not own.
|
||||||
|
useEffect(() => {
|
||||||
|
if (chatId && chatInfoQuery.isError) {
|
||||||
|
setChatId(undefined);
|
||||||
|
}
|
||||||
|
}, [chatId, chatInfoQuery.isError]);
|
||||||
|
|
||||||
|
const handleNewChat = useCallback(
|
||||||
|
(event: React.MouseEvent<HTMLAnchorElement>) => {
|
||||||
|
if (
|
||||||
|
event.button !== 0 ||
|
||||||
|
event.ctrlKey ||
|
||||||
|
event.metaKey ||
|
||||||
|
event.shiftKey
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
event.preventDefault();
|
||||||
|
setChatId(undefined);
|
||||||
|
if (page) {
|
||||||
|
setContextPages([
|
||||||
|
{ id: page.id, title: page.title || "", slugId: page.slugId },
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[page],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleSelectChat = useCallback((selectedChatId: string) => {
|
||||||
|
setChatId(selectedChatId);
|
||||||
|
setHistoryOpen(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleExpand = useCallback(() => {
|
||||||
|
if (chatId) {
|
||||||
|
navigate(`/ai/chat/${chatId}`);
|
||||||
|
} else {
|
||||||
|
navigate("/ai");
|
||||||
|
}
|
||||||
|
setAsideState({ tab: "", isAsideOpen: false });
|
||||||
|
}, [chatId, navigate, setAsideState]);
|
||||||
|
|
||||||
|
const handleClose = useCallback(() => {
|
||||||
|
setAsideState({ tab: "", isAsideOpen: false });
|
||||||
|
}, [setAsideState]);
|
||||||
|
|
||||||
|
const handleSend = useCallback(
|
||||||
|
(content: string, mentions: PageMention[], attachments: ChatAttachment[]) => {
|
||||||
|
const contextPageId = contextPages.length > 0 ? contextPages[0].id : undefined;
|
||||||
|
sendMessage(content, mentions, attachments, contextPageId);
|
||||||
|
},
|
||||||
|
[sendMessage, contextPages],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleQuickAction = useCallback(
|
||||||
|
(prompt: string) => {
|
||||||
|
handleSend(prompt, [], []);
|
||||||
|
},
|
||||||
|
[handleSend],
|
||||||
|
);
|
||||||
|
|
||||||
|
const hasMessages = messages.length > 0 || isStreaming;
|
||||||
|
|
||||||
|
const quickActions: QuickAction[] = [
|
||||||
|
{ icon: <IconFileText size={16} />, label: t("Summarize this page"), prompt: "Summarize this page" },
|
||||||
|
{ icon: <IconLanguage size={16} />, label: t("Translate this page"), prompt: "Translate this page" },
|
||||||
|
{ icon: <IconSearch size={16} />, label: t("Analyze for insights"), prompt: "Analyze this page for insights" },
|
||||||
|
];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.panel}>
|
||||||
|
<div className={classes.toolbar}>
|
||||||
|
<Popover
|
||||||
|
opened={historyOpen}
|
||||||
|
onChange={setHistoryOpen}
|
||||||
|
position="bottom-start"
|
||||||
|
width={280}
|
||||||
|
shadow="md"
|
||||||
|
>
|
||||||
|
<Popover.Target>
|
||||||
|
<UnstyledButton
|
||||||
|
className={classes.titleButton}
|
||||||
|
onClick={() => setHistoryOpen((o) => !o)}
|
||||||
|
>
|
||||||
|
<span className={classes.titleText}>
|
||||||
|
{chatInfoQuery.data?.chat?.title || t("New chat")}
|
||||||
|
</span>
|
||||||
|
<IconChevronDown size={16} stroke={1.75} />
|
||||||
|
</UnstyledButton>
|
||||||
|
</Popover.Target>
|
||||||
|
<Popover.Dropdown>
|
||||||
|
<AsideChatHistory activeChatId={chatId} onSelect={handleSelectChat} />
|
||||||
|
</Popover.Dropdown>
|
||||||
|
</Popover>
|
||||||
|
|
||||||
|
<div className={classes.toolbarSpacer} />
|
||||||
|
|
||||||
|
<Tooltip label={t("New chat")} openDelay={250}>
|
||||||
|
<ActionIcon
|
||||||
|
component="a"
|
||||||
|
href="/ai"
|
||||||
|
variant="subtle"
|
||||||
|
color="dark"
|
||||||
|
onClick={handleNewChat}
|
||||||
|
>
|
||||||
|
<IconPlus size={20} stroke={1.75} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
|
||||||
|
<Tooltip label={t("Open full page")} openDelay={250}>
|
||||||
|
<ActionIcon variant="subtle" color="dark" onClick={handleExpand}>
|
||||||
|
<IconArrowsDiagonal size={18} stroke={1.5} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
|
||||||
|
<Tooltip label={t("Close")} openDelay={250}>
|
||||||
|
<ActionIcon variant="subtle" color="dark" onClick={handleClose}>
|
||||||
|
<IconX size={20} stroke={1.75} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{error && (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
padding: "var(--mantine-spacing-xs) var(--mantine-spacing-sm)",
|
||||||
|
color: "var(--mantine-color-red-6)",
|
||||||
|
fontSize: "var(--mantine-font-size-xs)",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{error}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{hasMessages ? (
|
||||||
|
<>
|
||||||
|
<div className={classes.messages} data-aside-chat>
|
||||||
|
<ChatMessageList
|
||||||
|
messages={messages}
|
||||||
|
isStreaming={isStreaming}
|
||||||
|
streamingContent={streamingContent}
|
||||||
|
streamingToolCalls={streamingToolCalls}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<div className={classes.emptyState}>
|
||||||
|
<IconSparkles size={36} stroke={1.5} className={classes.emptyStateIcon} />
|
||||||
|
<div className={classes.emptyStateTitle}>{t("How can I help you today?")}</div>
|
||||||
|
<div className={classes.quickActions}>
|
||||||
|
{quickActions.map((action) => (
|
||||||
|
<button
|
||||||
|
key={action.label}
|
||||||
|
type="button"
|
||||||
|
className={classes.quickAction}
|
||||||
|
onClick={() => handleQuickAction(action.prompt)}
|
||||||
|
>
|
||||||
|
<span className={classes.quickActionIcon}>{action.icon}</span>
|
||||||
|
{action.label}
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div className={classes.inputArea}>
|
||||||
|
<ChatInput
|
||||||
|
isStreaming={isStreaming}
|
||||||
|
onSend={handleSend}
|
||||||
|
onStop={stopGeneration}
|
||||||
|
placeholder={t("Ask anything...")}
|
||||||
|
autofocus={false}
|
||||||
|
contextPages={contextPages}
|
||||||
|
onRemoveContextPage={handleRemoveContextPage}
|
||||||
|
variant="flat"
|
||||||
|
chatId={chatId}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,91 @@
|
|||||||
|
import {
|
||||||
|
IconSparkles,
|
||||||
|
IconSearch,
|
||||||
|
IconFilePlus,
|
||||||
|
IconEdit,
|
||||||
|
IconFileText,
|
||||||
|
} from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import ChatInput from "./chat-input";
|
||||||
|
import type { ChatAttachment, PageMention } from "../types/ai-chat.types";
|
||||||
|
import classes from "../styles/ai-chat.module.css";
|
||||||
|
|
||||||
|
type Suggestion = {
|
||||||
|
icon: React.ReactNode;
|
||||||
|
text: string;
|
||||||
|
prompt: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
const SUGGESTIONS: Suggestion[] = [
|
||||||
|
{
|
||||||
|
icon: <IconSearch size={16} />,
|
||||||
|
text: "Search across all pages",
|
||||||
|
prompt: "Search for pages about ",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: <IconFilePlus size={16} />,
|
||||||
|
text: "Create a new page",
|
||||||
|
prompt: "Create a new page titled ",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: <IconFileText size={16} />,
|
||||||
|
text: "Summarize a page",
|
||||||
|
prompt: "Summarize the page @",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: <IconEdit size={16} />,
|
||||||
|
text: "Update page content",
|
||||||
|
prompt: "Update the page @",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
isStreaming: boolean;
|
||||||
|
onSend: (content: string, mentions: PageMention[], attachments: ChatAttachment[]) => void;
|
||||||
|
onStop: () => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function ChatEmptyState({ isStreaming, onSend, onStop }: Props) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const handleSuggestionClick = (prompt: string) => {
|
||||||
|
onSend(prompt, [], []);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.emptyState}>
|
||||||
|
<IconSparkles size={48} stroke={1.5} className={classes.emptyStateIcon} />
|
||||||
|
<div className={classes.emptyStateBrand}>{t("Docmost AI")}</div>
|
||||||
|
<div className={classes.emptyStateTitle}>
|
||||||
|
{t("What can I help you with?")}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className={classes.emptyStateInput}>
|
||||||
|
<ChatInput
|
||||||
|
isStreaming={isStreaming}
|
||||||
|
onSend={onSend}
|
||||||
|
onStop={onStop}
|
||||||
|
placeholder="Ask anything... Use @ to mention pages"
|
||||||
|
autofocus
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className={classes.suggestionsSection}>
|
||||||
|
<div className={classes.suggestionsLabel}>Get started</div>
|
||||||
|
<div className={classes.suggestionsGrid}>
|
||||||
|
{SUGGESTIONS.map((s) => (
|
||||||
|
<button
|
||||||
|
key={s.text}
|
||||||
|
type="button"
|
||||||
|
className={classes.suggestionCard}
|
||||||
|
onClick={() => handleSuggestionClick(s.prompt)}
|
||||||
|
>
|
||||||
|
<span className={classes.suggestionIcon}>{s.icon}</span>
|
||||||
|
<span className={classes.suggestionText}>{s.text}</span>
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,409 @@
|
|||||||
|
import { useCallback, useRef, useEffect, useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { IconArrowUp, IconPaperclip, IconPlayerStopFilled, IconX, IconFile, IconPhoto, IconPlus, IconAt, IconFileText } from "@tabler/icons-react";
|
||||||
|
import { Popover } from "@mantine/core";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { EditorContent, ReactNodeViewRenderer, useEditor } from "@tiptap/react";
|
||||||
|
import { Placeholder } from "@tiptap/extension-placeholder";
|
||||||
|
import { CharacterCount } from "@tiptap/extensions";
|
||||||
|
import { StarterKit } from "@tiptap/starter-kit";
|
||||||
|
import { Mention, LinkExtension } from "@docmost/editor-ext";
|
||||||
|
import EmojiCommand from "@/features/editor/extensions/emoji-command";
|
||||||
|
import mentionRenderItems from "@/features/editor/components/mention/mention-suggestion";
|
||||||
|
import MentionView from "@/features/editor/components/mention/mention-view";
|
||||||
|
import { uploadChatFile } from "../services/ai-chat-service";
|
||||||
|
import type { ChatAttachment, PageMention } from "../types/ai-chat.types";
|
||||||
|
import classes from "../styles/chat-input.module.css";
|
||||||
|
|
||||||
|
type PendingAttachment = ChatAttachment & { uploading: boolean };
|
||||||
|
|
||||||
|
const IMAGE_EXTENSIONS = ["png", "jpg", "jpeg", "webp", "gif"];
|
||||||
|
const ACCEPTED_FILE_TYPES = ".pdf,.docx,.txt,.csv,.md,.png,.jpg,.jpeg,.webp";
|
||||||
|
// Kept in sync with MAX_ATTACHMENTS_PER_MESSAGE in apps/server/src/ee/ai-chat/ai-chat-limits.ts
|
||||||
|
const MAX_ATTACHMENTS_PER_MESSAGE = 5;
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
isStreaming: boolean;
|
||||||
|
onSend: (content: string, mentions: PageMention[], attachments: ChatAttachment[]) => void;
|
||||||
|
onStop: () => void;
|
||||||
|
placeholder?: string;
|
||||||
|
autofocus?: boolean;
|
||||||
|
contextPages?: PageMention[];
|
||||||
|
onRemoveContextPage?: (pageId: string) => void;
|
||||||
|
variant?: "card" | "flat";
|
||||||
|
showDisclaimer?: boolean;
|
||||||
|
chatId?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
function extractMentions(json: any): PageMention[] {
|
||||||
|
const mentions: PageMention[] = [];
|
||||||
|
const seen = new Set<string>();
|
||||||
|
|
||||||
|
function walk(node: any) {
|
||||||
|
if (node.type === "mention" && node.attrs?.entityType === "page" && node.attrs?.entityId) {
|
||||||
|
if (!seen.has(node.attrs.entityId)) {
|
||||||
|
seen.add(node.attrs.entityId);
|
||||||
|
mentions.push({
|
||||||
|
id: node.attrs.entityId,
|
||||||
|
title: node.attrs.label || "",
|
||||||
|
slugId: node.attrs.slugId || "",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (node.content) {
|
||||||
|
for (const child of node.content) {
|
||||||
|
walk(child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
walk(json);
|
||||||
|
return mentions;
|
||||||
|
}
|
||||||
|
|
||||||
|
function editorJsonToText(json: any): string {
|
||||||
|
let text = "";
|
||||||
|
|
||||||
|
function walk(node: any) {
|
||||||
|
if (node.type === "text") {
|
||||||
|
text += node.text || "";
|
||||||
|
} else if (node.type === "mention") {
|
||||||
|
text += `@${node.attrs?.label || ""}`;
|
||||||
|
} else if (node.type === "paragraph") {
|
||||||
|
if (text.length > 0) text += "\n";
|
||||||
|
if (node.content) {
|
||||||
|
for (const child of node.content) {
|
||||||
|
walk(child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (node.content) {
|
||||||
|
for (const child of node.content) {
|
||||||
|
walk(child);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
walk(json);
|
||||||
|
return text;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ChatInput({
|
||||||
|
isStreaming,
|
||||||
|
onSend,
|
||||||
|
onStop,
|
||||||
|
placeholder,
|
||||||
|
autofocus = true,
|
||||||
|
contextPages,
|
||||||
|
onRemoveContextPage,
|
||||||
|
variant = "card",
|
||||||
|
showDisclaimer = true,
|
||||||
|
chatId,
|
||||||
|
}: Props) {
|
||||||
|
const chatIdRef = useRef(chatId);
|
||||||
|
chatIdRef.current = chatId;
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [isEmpty, setIsEmpty] = useState(true);
|
||||||
|
const [pendingAttachments, setPendingAttachments] = useState<PendingAttachment[]>([]);
|
||||||
|
const [plusMenuOpen, setPlusMenuOpen] = useState(false);
|
||||||
|
const fileInputRef = useRef<HTMLInputElement>(null);
|
||||||
|
const onSendRef = useRef(onSend);
|
||||||
|
onSendRef.current = onSend;
|
||||||
|
|
||||||
|
const handleFileSelect = useCallback(async (files: FileList | null) => {
|
||||||
|
if (!files?.length) return;
|
||||||
|
|
||||||
|
const room = MAX_ATTACHMENTS_PER_MESSAGE - pendingAttachments.length;
|
||||||
|
if (room <= 0) {
|
||||||
|
notifications.show({
|
||||||
|
color: "yellow",
|
||||||
|
message: t("You can attach up to {{max}} files per message.", {
|
||||||
|
max: MAX_ATTACHMENTS_PER_MESSAGE,
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
if (fileInputRef.current) fileInputRef.current.value = "";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const incoming = Array.from(files);
|
||||||
|
const accepted = incoming.slice(0, room);
|
||||||
|
|
||||||
|
if (incoming.length > accepted.length) {
|
||||||
|
notifications.show({
|
||||||
|
color: "yellow",
|
||||||
|
message: t(
|
||||||
|
"Only the first {{n}} file(s) were added (max {{max}} per message).",
|
||||||
|
{ n: accepted.length, max: MAX_ATTACHMENTS_PER_MESSAGE },
|
||||||
|
),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const file of accepted) {
|
||||||
|
const tempId = `uploading-${Date.now()}-${Math.random()}`;
|
||||||
|
const ext = file.name.split(".").pop()?.toLowerCase() || "";
|
||||||
|
|
||||||
|
const placeholder: PendingAttachment = {
|
||||||
|
id: tempId,
|
||||||
|
fileName: file.name,
|
||||||
|
fileExt: ext,
|
||||||
|
fileSize: file.size,
|
||||||
|
mimeType: file.type,
|
||||||
|
uploading: true,
|
||||||
|
};
|
||||||
|
|
||||||
|
setPendingAttachments((prev) => [...prev, placeholder]);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const uploaded = await uploadChatFile(file, chatIdRef.current);
|
||||||
|
setPendingAttachments((prev) =>
|
||||||
|
prev.map((a) =>
|
||||||
|
a.id === tempId ? { ...uploaded, uploading: false } : a,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} catch {
|
||||||
|
setPendingAttachments((prev) => prev.filter((a) => a.id !== tempId));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fileInputRef.current) {
|
||||||
|
fileInputRef.current.value = "";
|
||||||
|
}
|
||||||
|
}, [pendingAttachments.length, t]);
|
||||||
|
|
||||||
|
const removeAttachment = useCallback((id: string) => {
|
||||||
|
setPendingAttachments((prev) => prev.filter((a) => a.id !== id));
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleSubmit = useCallback(() => {
|
||||||
|
if (!editor || isStreaming) return;
|
||||||
|
const json = editor.getJSON();
|
||||||
|
const text = editorJsonToText(json).trim();
|
||||||
|
const readyAttachments = pendingAttachments.filter((a) => !a.uploading);
|
||||||
|
if (!text && readyAttachments.length === 0) return;
|
||||||
|
|
||||||
|
const mentions = extractMentions(json);
|
||||||
|
onSendRef.current(text, mentions, readyAttachments);
|
||||||
|
editor.commands.clearContent();
|
||||||
|
editor.commands.focus();
|
||||||
|
setPendingAttachments([]);
|
||||||
|
}, [isStreaming, pendingAttachments]);
|
||||||
|
|
||||||
|
const handleSubmitRef = useRef(handleSubmit);
|
||||||
|
handleSubmitRef.current = handleSubmit;
|
||||||
|
|
||||||
|
const editor = useEditor({
|
||||||
|
extensions: [
|
||||||
|
StarterKit.configure({
|
||||||
|
gapcursor: false,
|
||||||
|
dropcursor: false,
|
||||||
|
link: false,
|
||||||
|
}),
|
||||||
|
Placeholder.configure({
|
||||||
|
placeholder: placeholder || "Ask anything... Use @ to mention pages",
|
||||||
|
}),
|
||||||
|
CharacterCount.configure({
|
||||||
|
limit: 50000,
|
||||||
|
}),
|
||||||
|
LinkExtension,
|
||||||
|
EmojiCommand,
|
||||||
|
Mention.configure({
|
||||||
|
suggestion: {
|
||||||
|
allowSpaces: true,
|
||||||
|
items: () => [],
|
||||||
|
// @ts-ignore
|
||||||
|
render: mentionRenderItems,
|
||||||
|
},
|
||||||
|
HTMLAttributes: {
|
||||||
|
class: "mention",
|
||||||
|
},
|
||||||
|
}).extend({
|
||||||
|
addNodeView() {
|
||||||
|
this.editor.isInitialized = true;
|
||||||
|
return ReactNodeViewRenderer(MentionView);
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
editorProps: {
|
||||||
|
handleDOMEvents: {
|
||||||
|
keydown: (_view, event) => {
|
||||||
|
if (
|
||||||
|
["ArrowUp", "ArrowDown", "ArrowLeft", "ArrowRight", "Enter"].includes(
|
||||||
|
event.key,
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
const emojiCommand = document.querySelector("#emoji-command");
|
||||||
|
const mentionPopup = document.querySelector("#mention");
|
||||||
|
if (emojiCommand || mentionPopup) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.key === "Enter" && !event.shiftKey) {
|
||||||
|
event.preventDefault();
|
||||||
|
handleSubmitRef.current();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
content: "",
|
||||||
|
editable: true,
|
||||||
|
immediatelyRender: true,
|
||||||
|
shouldRerenderOnTransaction: false,
|
||||||
|
autofocus: autofocus ? "end" : false,
|
||||||
|
onUpdate: ({ editor: e }) => {
|
||||||
|
setIsEmpty(!e.getText().trim());
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (editor && autofocus) {
|
||||||
|
editor.commands.focus();
|
||||||
|
}
|
||||||
|
}, [editor]);
|
||||||
|
|
||||||
|
const hasContent = !isEmpty || pendingAttachments.some((a) => !a.uploading) || (contextPages?.length ?? 0) > 0;
|
||||||
|
|
||||||
|
const wrapperClass = variant === "flat" ? classes.inputWrapperFlat : classes.inputWrapper;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<div className={wrapperClass} data-chat-input>
|
||||||
|
<input
|
||||||
|
ref={fileInputRef}
|
||||||
|
type="file"
|
||||||
|
accept={ACCEPTED_FILE_TYPES}
|
||||||
|
multiple
|
||||||
|
style={{ display: "none" }}
|
||||||
|
onChange={(e) => handleFileSelect(e.target.files)}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{((contextPages?.length ?? 0) > 0 || pendingAttachments.length > 0) && (
|
||||||
|
<div className={classes.attachmentChips}>
|
||||||
|
{contextPages?.map((page) => (
|
||||||
|
<div key={page.id} className={classes.attachmentChip}>
|
||||||
|
<IconFileText size={14} />
|
||||||
|
<span className={classes.attachmentChipName}>
|
||||||
|
{page.title || "Untitled"}
|
||||||
|
</span>
|
||||||
|
{onRemoveContextPage && (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={classes.attachmentChipRemove}
|
||||||
|
onClick={() => onRemoveContextPage(page.id)}
|
||||||
|
aria-label={`Remove ${page.title}`}
|
||||||
|
>
|
||||||
|
<IconX size={12} />
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
{pendingAttachments.map((attachment) => (
|
||||||
|
<div
|
||||||
|
key={attachment.id}
|
||||||
|
className={`${classes.attachmentChip} ${attachment.uploading ? classes.attachmentChipUploading : ""}`}
|
||||||
|
>
|
||||||
|
{IMAGE_EXTENSIONS.includes(attachment.fileExt) ? (
|
||||||
|
<IconPhoto size={14} />
|
||||||
|
) : (
|
||||||
|
<IconFile size={14} />
|
||||||
|
)}
|
||||||
|
<span className={classes.attachmentChipName}>
|
||||||
|
{attachment.fileName}
|
||||||
|
</span>
|
||||||
|
{!attachment.uploading && (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={classes.attachmentChipRemove}
|
||||||
|
onClick={() => removeAttachment(attachment.id)}
|
||||||
|
aria-label={`Remove ${attachment.fileName}`}
|
||||||
|
>
|
||||||
|
<IconX size={12} />
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<EditorContent editor={editor} className={classes.editorContent} />
|
||||||
|
<div className={classes.actions}>
|
||||||
|
<Popover opened={plusMenuOpen} onChange={setPlusMenuOpen} position="top-start" width={220} shadow="md">
|
||||||
|
<Popover.Target>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={classes.plusButton}
|
||||||
|
onClick={() => setPlusMenuOpen((o) => !o)}
|
||||||
|
aria-label="Add content"
|
||||||
|
>
|
||||||
|
<IconPlus size={14} />
|
||||||
|
</button>
|
||||||
|
</Popover.Target>
|
||||||
|
<Popover.Dropdown p={4}>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={classes.plusMenuItem}
|
||||||
|
onClick={() => {
|
||||||
|
fileInputRef.current?.click();
|
||||||
|
setPlusMenuOpen(false);
|
||||||
|
}}
|
||||||
|
disabled={pendingAttachments.length >= MAX_ATTACHMENTS_PER_MESSAGE}
|
||||||
|
title={
|
||||||
|
pendingAttachments.length >= MAX_ATTACHMENTS_PER_MESSAGE
|
||||||
|
? t("Max {{max}} files per message", {
|
||||||
|
max: MAX_ATTACHMENTS_PER_MESSAGE,
|
||||||
|
})
|
||||||
|
: undefined
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<IconPaperclip size={16} className={classes.plusMenuIcon} />
|
||||||
|
{t("Add files")}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={classes.plusMenuItem}
|
||||||
|
onClick={() => {
|
||||||
|
editor?.commands.insertContent("@");
|
||||||
|
editor?.commands.focus();
|
||||||
|
setPlusMenuOpen(false);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<IconAt size={16} className={classes.plusMenuIcon} />
|
||||||
|
Mention a page
|
||||||
|
</button>
|
||||||
|
</Popover.Dropdown>
|
||||||
|
</Popover>
|
||||||
|
|
||||||
|
<div style={{ flex: 1 }} />
|
||||||
|
|
||||||
|
{isStreaming ? (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={classes.stopButton}
|
||||||
|
onClick={onStop}
|
||||||
|
aria-label="Stop generation"
|
||||||
|
>
|
||||||
|
<IconPlayerStopFilled size={14} />
|
||||||
|
</button>
|
||||||
|
) : (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={classes.sendButton}
|
||||||
|
onClick={handleSubmit}
|
||||||
|
disabled={!hasContent}
|
||||||
|
aria-label="Send message"
|
||||||
|
>
|
||||||
|
<IconArrowUp size={16} stroke={2.5} />
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{showDisclaimer && (
|
||||||
|
<div className={classes.disclaimer}>
|
||||||
|
{t("AI-generated content may not be accurate.")}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,174 @@
|
|||||||
|
import { useEffect, useRef, useCallback, useState } from "react";
|
||||||
|
import { ErrorBoundary } from "react-error-boundary";
|
||||||
|
import { IconArrowDown, IconAlertTriangle } from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import type { AiChatMessage, AiChatToolCall } from "../types/ai-chat.types";
|
||||||
|
import ChatMessage from "./chat-message";
|
||||||
|
import classes from "../styles/ai-chat.module.css";
|
||||||
|
|
||||||
|
function ChatMessageErrorFallback() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
return (
|
||||||
|
<div className={classes.messageErrorFallback}>
|
||||||
|
<IconAlertTriangle size={14} />
|
||||||
|
<span>{t("Failed to render this message.")}</span>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
messages: AiChatMessage[];
|
||||||
|
isStreaming: boolean;
|
||||||
|
streamingContent: string;
|
||||||
|
streamingToolCalls: AiChatToolCall[];
|
||||||
|
};
|
||||||
|
|
||||||
|
const BOTTOM_THRESHOLD_PX = 32;
|
||||||
|
const SCROLL_UP_THRESHOLD_PX = 5;
|
||||||
|
const SMOOTH_SCROLL_SETTLE_MS = 600;
|
||||||
|
|
||||||
|
export default function ChatMessageList({
|
||||||
|
messages,
|
||||||
|
isStreaming,
|
||||||
|
streamingContent,
|
||||||
|
streamingToolCalls,
|
||||||
|
}: Props) {
|
||||||
|
const containerRef = useRef<HTMLDivElement>(null);
|
||||||
|
const bottomRef = useRef<HTMLDivElement>(null);
|
||||||
|
const isAtBottomRef = useRef(true);
|
||||||
|
const isAutoScrollingRef = useRef(false);
|
||||||
|
const prevScrollTopRef = useRef(0);
|
||||||
|
const [showScrollButton, setShowScrollButton] = useState(false);
|
||||||
|
|
||||||
|
const scrollToBottom = useCallback((behavior: ScrollBehavior = "smooth") => {
|
||||||
|
const container = containerRef.current;
|
||||||
|
if (!container) return;
|
||||||
|
|
||||||
|
isAutoScrollingRef.current = true;
|
||||||
|
const target = container.scrollHeight - container.clientHeight;
|
||||||
|
container.scrollTo({ top: target, behavior });
|
||||||
|
prevScrollTopRef.current = target;
|
||||||
|
isAtBottomRef.current = true;
|
||||||
|
setShowScrollButton(false);
|
||||||
|
|
||||||
|
if (behavior === "smooth") {
|
||||||
|
setTimeout(() => {
|
||||||
|
isAutoScrollingRef.current = false;
|
||||||
|
if (containerRef.current) {
|
||||||
|
prevScrollTopRef.current = containerRef.current.scrollTop;
|
||||||
|
}
|
||||||
|
}, SMOOTH_SCROLL_SETTLE_MS);
|
||||||
|
} else {
|
||||||
|
isAutoScrollingRef.current = false;
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleScroll = useCallback(() => {
|
||||||
|
if (isAutoScrollingRef.current) return;
|
||||||
|
|
||||||
|
const container = containerRef.current;
|
||||||
|
if (!container) return;
|
||||||
|
|
||||||
|
const currentScrollTop = container.scrollTop;
|
||||||
|
const scrolledUp =
|
||||||
|
currentScrollTop < prevScrollTopRef.current - SCROLL_UP_THRESHOLD_PX;
|
||||||
|
prevScrollTopRef.current = currentScrollTop;
|
||||||
|
|
||||||
|
const distanceFromBottom =
|
||||||
|
container.scrollHeight - currentScrollTop - container.clientHeight;
|
||||||
|
const atBottom = distanceFromBottom <= BOTTOM_THRESHOLD_PX;
|
||||||
|
|
||||||
|
if (scrolledUp) {
|
||||||
|
isAtBottomRef.current = atBottom;
|
||||||
|
} else if (atBottom) {
|
||||||
|
isAtBottomRef.current = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
setShowScrollButton(!atBottom);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const container = containerRef.current;
|
||||||
|
if (!container) return;
|
||||||
|
|
||||||
|
container.addEventListener("scroll", handleScroll, { passive: true });
|
||||||
|
return () => container.removeEventListener("scroll", handleScroll);
|
||||||
|
}, [handleScroll]);
|
||||||
|
|
||||||
|
// Instant scroll during streaming to keep up with rapid updates
|
||||||
|
useEffect(() => {
|
||||||
|
if (isAtBottomRef.current) {
|
||||||
|
scrollToBottom("instant");
|
||||||
|
}
|
||||||
|
}, [streamingContent, streamingToolCalls.length, scrollToBottom]);
|
||||||
|
|
||||||
|
// Smooth scroll for new messages. Always force-scroll when the latest
|
||||||
|
// message is from the user (they just sent it), even if they were reading
|
||||||
|
// scrollback.
|
||||||
|
useEffect(() => {
|
||||||
|
const lastMessage = messages[messages.length - 1];
|
||||||
|
const lastIsUser = lastMessage?.role === "user";
|
||||||
|
if (lastIsUser || isAtBottomRef.current) {
|
||||||
|
scrollToBottom("smooth");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// No auto-scroll: recompute from actual layout so that chat switches to
|
||||||
|
// content that doesn't overflow correctly hide the button even when no
|
||||||
|
// scroll event fires.
|
||||||
|
const container = containerRef.current;
|
||||||
|
if (!container) return;
|
||||||
|
const distanceFromBottom =
|
||||||
|
container.scrollHeight - container.scrollTop - container.clientHeight;
|
||||||
|
const atBottom = distanceFromBottom <= BOTTOM_THRESHOLD_PX;
|
||||||
|
isAtBottomRef.current = atBottom;
|
||||||
|
setShowScrollButton(!atBottom);
|
||||||
|
}, [messages, scrollToBottom]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.messageListWrapper}>
|
||||||
|
<div ref={containerRef} className={classes.messageList}>
|
||||||
|
{messages.map((msg) => (
|
||||||
|
<ErrorBoundary
|
||||||
|
key={msg.id}
|
||||||
|
fallback={<ChatMessageErrorFallback />}
|
||||||
|
>
|
||||||
|
<ChatMessage message={msg} />
|
||||||
|
</ErrorBoundary>
|
||||||
|
))}
|
||||||
|
{isStreaming && (
|
||||||
|
<ErrorBoundary
|
||||||
|
resetKeys={[streamingContent, streamingToolCalls.length]}
|
||||||
|
fallback={<ChatMessageErrorFallback />}
|
||||||
|
>
|
||||||
|
<ChatMessage
|
||||||
|
message={{
|
||||||
|
id: "streaming",
|
||||||
|
chatId: "",
|
||||||
|
role: "assistant",
|
||||||
|
content: null,
|
||||||
|
toolCalls: null,
|
||||||
|
metadata: null,
|
||||||
|
createdAt: new Date().toISOString(),
|
||||||
|
}}
|
||||||
|
isStreaming
|
||||||
|
streamingContent={streamingContent}
|
||||||
|
streamingToolCalls={streamingToolCalls}
|
||||||
|
/>
|
||||||
|
</ErrorBoundary>
|
||||||
|
)}
|
||||||
|
<div ref={bottomRef} />
|
||||||
|
</div>
|
||||||
|
{showScrollButton && (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
aria-label="Scroll to bottom"
|
||||||
|
className={classes.scrollToBottomButton}
|
||||||
|
onClick={() => scrollToBottom("smooth")}
|
||||||
|
>
|
||||||
|
<IconArrowDown size={16} stroke={2} />
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,139 @@
|
|||||||
|
import { useCallback } from "react";
|
||||||
|
import { useNavigate } from "react-router";
|
||||||
|
import DOMPurify from "dompurify";
|
||||||
|
import { ActionIcon, Tooltip } from "@mantine/core";
|
||||||
|
import {
|
||||||
|
IconCheck,
|
||||||
|
IconCopy,
|
||||||
|
IconFile,
|
||||||
|
IconLoader2,
|
||||||
|
IconPhoto,
|
||||||
|
} from "@tabler/icons-react";
|
||||||
|
import { markdownToHtml } from "@docmost/editor-ext";
|
||||||
|
import { CopyButton } from "@/components/common/copy-button";
|
||||||
|
import type { AiChatMessage, AiChatToolCall } from "../types/ai-chat.types";
|
||||||
|
import ChatToolGroup from "./chat-tool-group";
|
||||||
|
import classes from "../styles/chat-message.module.css";
|
||||||
|
import CopyTextButton from "@/components/common/copy.tsx";
|
||||||
|
|
||||||
|
const chatSanitizer = DOMPurify();
|
||||||
|
chatSanitizer.addHook("afterSanitizeAttributes", (node) => {
|
||||||
|
if (node.tagName === "A") {
|
||||||
|
const href = node.getAttribute("href") || "";
|
||||||
|
if (href.startsWith("http://") || href.startsWith("https://")) {
|
||||||
|
node.setAttribute("target", "_blank");
|
||||||
|
node.setAttribute("rel", "noopener noreferrer");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const IMAGE_EXTENSIONS = ["png", "jpg", "jpeg", "webp", "gif"];
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
message: AiChatMessage;
|
||||||
|
isStreaming?: boolean;
|
||||||
|
streamingContent?: string;
|
||||||
|
streamingToolCalls?: AiChatToolCall[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function ChatMessage({
|
||||||
|
message,
|
||||||
|
isStreaming,
|
||||||
|
streamingContent,
|
||||||
|
streamingToolCalls,
|
||||||
|
}: Props) {
|
||||||
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
const handleContentClick = useCallback(
|
||||||
|
(e: React.MouseEvent<HTMLDivElement>) => {
|
||||||
|
const target = e.target as HTMLElement;
|
||||||
|
const anchor = target.closest("a");
|
||||||
|
if (!anchor) return;
|
||||||
|
|
||||||
|
const href = anchor.getAttribute("href");
|
||||||
|
if (href && (href.startsWith("/s/") || href.startsWith("/p/"))) {
|
||||||
|
e.preventDefault();
|
||||||
|
navigate(href);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[navigate],
|
||||||
|
);
|
||||||
|
|
||||||
|
if (message.role === "tool") return null;
|
||||||
|
|
||||||
|
const isUser = message.role === "user";
|
||||||
|
const content = isStreaming ? streamingContent : message.content;
|
||||||
|
const toolCalls = isStreaming ? streamingToolCalls : message.toolCalls;
|
||||||
|
|
||||||
|
if (isUser) {
|
||||||
|
const displayContent = (content || "").replace(
|
||||||
|
/\n\n<referenced_pages>[\s\S]*<\/referenced_pages>$/,
|
||||||
|
"",
|
||||||
|
);
|
||||||
|
const attachments =
|
||||||
|
(message.metadata?.attachments as {
|
||||||
|
id: string;
|
||||||
|
fileName: string;
|
||||||
|
fileExt: string;
|
||||||
|
}[]) || [];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.userMessage}>
|
||||||
|
<div className={classes.userBubble}>
|
||||||
|
{attachments.length > 0 && (
|
||||||
|
<div className={classes.messageAttachments}>
|
||||||
|
{attachments.map((a) => (
|
||||||
|
<span key={a.id} className={classes.messageAttachmentChip}>
|
||||||
|
{IMAGE_EXTENSIONS.includes(a.fileExt) ? (
|
||||||
|
<IconPhoto size={13} />
|
||||||
|
) : (
|
||||||
|
<IconFile size={13} />
|
||||||
|
)}
|
||||||
|
{a.fileName}
|
||||||
|
</span>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{displayContent}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.assistantMessage}>
|
||||||
|
<div className={classes.messageContent}>
|
||||||
|
{toolCalls && toolCalls.length > 0 && (
|
||||||
|
<ChatToolGroup toolCalls={toolCalls} isStreaming={isStreaming} />
|
||||||
|
)}
|
||||||
|
{content && (
|
||||||
|
<div
|
||||||
|
onClick={handleContentClick}
|
||||||
|
dangerouslySetInnerHTML={{
|
||||||
|
__html: chatSanitizer.sanitize(
|
||||||
|
markdownToHtml(content) as string,
|
||||||
|
{ ADD_ATTR: ["target", "rel"] },
|
||||||
|
),
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{isStreaming && (
|
||||||
|
<>
|
||||||
|
{!content && (
|
||||||
|
<span className={classes.processingIndicator}>
|
||||||
|
<IconLoader2 size={16} className={classes.processingSpinner} />
|
||||||
|
Thinking
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
<span className={classes.streamingCursor} />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
{!isStreaming && message.content && (
|
||||||
|
<div className={classes.messageActions}>
|
||||||
|
<CopyTextButton text={message?.content} />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,56 @@
|
|||||||
|
import { useState } from "react";
|
||||||
|
import {
|
||||||
|
IconChevronRight,
|
||||||
|
IconChevronDown,
|
||||||
|
IconLoader2,
|
||||||
|
} from "@tabler/icons-react";
|
||||||
|
import type { AiChatToolCall } from "../types/ai-chat.types";
|
||||||
|
import ChatToolResult, { TOOL_LABELS } from "./chat-tool-result";
|
||||||
|
import classes from "../styles/chat-message.module.css";
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
toolCalls: AiChatToolCall[];
|
||||||
|
isStreaming?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function ChatToolGroup({ toolCalls, isStreaming }: Props) {
|
||||||
|
const [expanded, setExpanded] = useState(false);
|
||||||
|
|
||||||
|
if (!toolCalls || toolCalls.length === 0) return null;
|
||||||
|
|
||||||
|
const activeCall =
|
||||||
|
isStreaming && toolCalls.length > 0
|
||||||
|
? [...toolCalls].reverse().find((tc) => tc.result === undefined)
|
||||||
|
: undefined;
|
||||||
|
|
||||||
|
const activeLabel = activeCall
|
||||||
|
? TOOL_LABELS[activeCall.name] || activeCall.name
|
||||||
|
: null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.toolGroup}>
|
||||||
|
<div
|
||||||
|
className={classes.toolGroupHeader}
|
||||||
|
onClick={() => setExpanded((prev) => !prev)}
|
||||||
|
>
|
||||||
|
{activeLabel ? (
|
||||||
|
<IconLoader2 size={12} className={classes.processingSpinner} />
|
||||||
|
) : expanded ? (
|
||||||
|
<IconChevronDown size={12} />
|
||||||
|
) : (
|
||||||
|
<IconChevronRight size={12} />
|
||||||
|
)}
|
||||||
|
<span className={classes.toolGroupLabel}>
|
||||||
|
{activeLabel ? `${activeLabel}…` : `Steps ${toolCalls.length}`}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
{expanded && (
|
||||||
|
<div className={classes.toolGroupSteps}>
|
||||||
|
{toolCalls.map((tc) => (
|
||||||
|
<ChatToolResult key={tc.id} toolCall={tc} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
import { useState } from "react";
|
||||||
|
import { IconChevronRight, IconChevronDown } from "@tabler/icons-react";
|
||||||
|
import type { AiChatToolCall } from "../types/ai-chat.types";
|
||||||
|
import classes from "../styles/chat-message.module.css";
|
||||||
|
|
||||||
|
export const TOOL_LABELS: Record<string, string> = {
|
||||||
|
list_spaces: "Listed spaces",
|
||||||
|
search_pages: "Searched pages",
|
||||||
|
get_page: "Read page",
|
||||||
|
create_page: "Created page",
|
||||||
|
update_page: "Updated page",
|
||||||
|
};
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
toolCall: AiChatToolCall;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function ChatToolResult({ toolCall }: Props) {
|
||||||
|
const [expanded, setExpanded] = useState(false);
|
||||||
|
const label = TOOL_LABELS[toolCall.name] || toolCall.name;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.toolStep}>
|
||||||
|
<div
|
||||||
|
className={classes.toolStepRow}
|
||||||
|
onClick={() => setExpanded((prev) => !prev)}
|
||||||
|
>
|
||||||
|
<span className={classes.toolStepBullet}>·</span>
|
||||||
|
{expanded ? (
|
||||||
|
<IconChevronDown size={12} />
|
||||||
|
) : (
|
||||||
|
<IconChevronRight size={12} />
|
||||||
|
)}
|
||||||
|
<span>{label}</span>
|
||||||
|
</div>
|
||||||
|
{expanded && (
|
||||||
|
<div className={classes.toolStepDetails}>
|
||||||
|
<pre style={{ margin: 0, whiteSpace: "pre-wrap" }}>
|
||||||
|
{JSON.stringify(
|
||||||
|
{ args: toolCall.args, result: toolCall.result },
|
||||||
|
null,
|
||||||
|
2,
|
||||||
|
)}
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,67 @@
|
|||||||
|
import { Badge, Group, Text, Switch, Tooltip } from "@mantine/core";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
import { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
|
|
||||||
|
export default function EnableAiChat() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
||||||
|
<div>
|
||||||
|
<Group gap="xs" align="center">
|
||||||
|
<Text size="md">{t("AI Chat")}</Text>
|
||||||
|
<Badge color="gray" variant="light" size="sm" radius="sm">
|
||||||
|
{t("Beta")}
|
||||||
|
</Badge>
|
||||||
|
</Group>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t(
|
||||||
|
"Enable AI Chat to allow users to have multi-turn conversations with AI about your workspace content.",
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<AiChatToggle />
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function AiChatToggle() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
|
const [checked, setChecked] = useState(workspace?.settings?.ai?.chat);
|
||||||
|
const hasAccess = useHasFeature(Feature.AI);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = event.currentTarget.checked;
|
||||||
|
try {
|
||||||
|
const updatedWorkspace = await updateWorkspace({ aiChat: value } as any);
|
||||||
|
setChecked(value);
|
||||||
|
setWorkspace(updatedWorkspace);
|
||||||
|
} catch (err: any) {
|
||||||
|
notifications.show({
|
||||||
|
message: err?.response?.data?.message,
|
||||||
|
color: "red",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Tooltip label={upgradeLabel} disabled={hasAccess} refProp="rootRef">
|
||||||
|
<Switch
|
||||||
|
defaultChecked={checked}
|
||||||
|
onChange={handleChange}
|
||||||
|
disabled={!hasAccess}
|
||||||
|
aria-label={t("Toggle AI Chat")}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,227 @@
|
|||||||
|
import { useState, useCallback, useEffect, useRef } from "react";
|
||||||
|
import { useQueryClient } from "@tanstack/react-query";
|
||||||
|
import { useNavigate } from "react-router-dom";
|
||||||
|
import { sendChatMessage } from "../services/ai-chat-service";
|
||||||
|
import type {
|
||||||
|
AiChatMessage,
|
||||||
|
AiChatStreamEvent,
|
||||||
|
AiChatToolCall,
|
||||||
|
ChatAttachment,
|
||||||
|
PageMention,
|
||||||
|
} from "../types/ai-chat.types";
|
||||||
|
|
||||||
|
type ChatStreamOptions = {
|
||||||
|
onChatCreated?: (chatId: string) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function useChatStream(
|
||||||
|
chatId: string | undefined,
|
||||||
|
options?: ChatStreamOptions,
|
||||||
|
) {
|
||||||
|
const [messages, setMessages] = useState<AiChatMessage[]>([]);
|
||||||
|
const [streamingContent, setStreamingContent] = useState("");
|
||||||
|
const [streamingToolCalls, setStreamingToolCalls] = useState<AiChatToolCall[]>(
|
||||||
|
[],
|
||||||
|
);
|
||||||
|
const [isStreaming, setIsStreaming] = useState(false);
|
||||||
|
const [error, setError] = useState<string | null>(null);
|
||||||
|
const [errorCode, setErrorCode] = useState<string | null>(null);
|
||||||
|
const [isRetryable, setIsRetryable] = useState(false);
|
||||||
|
const abortRef = useRef<AbortController | null>(null);
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const navigate = useNavigate();
|
||||||
|
const currentChatIdRef = useRef(chatId);
|
||||||
|
currentChatIdRef.current = chatId;
|
||||||
|
// Tracks which chatId the local `messages` state currently represents.
|
||||||
|
// Set when we seed from a server fetch AND when we optimistically own a
|
||||||
|
// freshly-created chat after `chat_created`. This is the single authority
|
||||||
|
// marker that keeps server-state effects from clobbering in-flight streams.
|
||||||
|
const hydratedChatIdRef = useRef<string | undefined>(undefined);
|
||||||
|
|
||||||
|
// Reset local state when the consumer switches to a different chat.
|
||||||
|
// Skip the reset if the new chatId is one the hook itself already claimed
|
||||||
|
// during a new-chat flow — in that case our optimistic state is the truth.
|
||||||
|
useEffect(() => {
|
||||||
|
if (chatId && chatId === hydratedChatIdRef.current) return;
|
||||||
|
hydratedChatIdRef.current = undefined;
|
||||||
|
setMessages([]);
|
||||||
|
setError(null);
|
||||||
|
setErrorCode(null);
|
||||||
|
setIsRetryable(false);
|
||||||
|
}, [chatId]);
|
||||||
|
|
||||||
|
const hydrateFromServer = useCallback((msgs: AiChatMessage[]) => {
|
||||||
|
const forId = currentChatIdRef.current;
|
||||||
|
if (!forId) return;
|
||||||
|
if (hydratedChatIdRef.current === forId) return;
|
||||||
|
hydratedChatIdRef.current = forId;
|
||||||
|
setMessages(msgs);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const sendMessage = useCallback(
|
||||||
|
(content: string, mentions: PageMention[] = [], attachments: ChatAttachment[] = [], contextPageId?: string) => {
|
||||||
|
if (isStreaming || (!content.trim() && attachments.length === 0)) return;
|
||||||
|
|
||||||
|
setError(null);
|
||||||
|
setErrorCode(null);
|
||||||
|
setIsRetryable(false);
|
||||||
|
setIsStreaming(true);
|
||||||
|
setStreamingContent("");
|
||||||
|
setStreamingToolCalls([]);
|
||||||
|
|
||||||
|
const metadata: Record<string, unknown> = {};
|
||||||
|
if (mentions.length) {
|
||||||
|
metadata.mentionedPageIds = mentions.map((m) => m.id);
|
||||||
|
}
|
||||||
|
if (attachments.length) {
|
||||||
|
metadata.attachments = attachments.map((a) => ({
|
||||||
|
id: a.id,
|
||||||
|
fileName: a.fileName,
|
||||||
|
fileExt: a.fileExt,
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
const userMessage: AiChatMessage = {
|
||||||
|
id: `temp-${Date.now()}`,
|
||||||
|
chatId: currentChatIdRef.current || "",
|
||||||
|
role: "user",
|
||||||
|
content,
|
||||||
|
toolCalls: null,
|
||||||
|
metadata: Object.keys(metadata).length ? metadata : null,
|
||||||
|
createdAt: new Date().toISOString(),
|
||||||
|
};
|
||||||
|
|
||||||
|
setMessages((prev) => [...prev, userMessage]);
|
||||||
|
|
||||||
|
const attachmentIds = attachments.map((a) => a.id);
|
||||||
|
|
||||||
|
const abortController = sendChatMessage(
|
||||||
|
{
|
||||||
|
chatId: currentChatIdRef.current,
|
||||||
|
content,
|
||||||
|
mentionedPageIds: mentions.map((m) => m.id),
|
||||||
|
...(contextPageId && { contextPageId }),
|
||||||
|
...(attachmentIds.length && { attachmentIds }),
|
||||||
|
},
|
||||||
|
(event: AiChatStreamEvent) => {
|
||||||
|
switch (event.type) {
|
||||||
|
case "chat_created":
|
||||||
|
currentChatIdRef.current = event.chatId;
|
||||||
|
// Claim authority over this new chatId so when the consumer's
|
||||||
|
// prop catches up via navigation/onChatCreated, the reset effect
|
||||||
|
// sees a match and preserves our optimistic messages.
|
||||||
|
hydratedChatIdRef.current = event.chatId;
|
||||||
|
if (options?.onChatCreated) {
|
||||||
|
options.onChatCreated(event.chatId);
|
||||||
|
} else {
|
||||||
|
navigate(`/ai/chat/${event.chatId}`, { replace: true });
|
||||||
|
}
|
||||||
|
queryClient.invalidateQueries({ queryKey: ["ai-chats"] });
|
||||||
|
break;
|
||||||
|
case "content":
|
||||||
|
setStreamingContent((prev) => prev + event.text);
|
||||||
|
break;
|
||||||
|
case "tool_call":
|
||||||
|
setStreamingToolCalls((prev) => [
|
||||||
|
...prev,
|
||||||
|
{
|
||||||
|
id: event.id,
|
||||||
|
name: event.name,
|
||||||
|
args: event.args,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
break;
|
||||||
|
case "tool_result":
|
||||||
|
setStreamingToolCalls((prev) =>
|
||||||
|
prev.map((tc) =>
|
||||||
|
tc.id === event.id ? { ...tc, result: event.result } : tc,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case "done": {
|
||||||
|
setStreamingContent((currentContent) => {
|
||||||
|
setStreamingToolCalls((currentToolCalls) => {
|
||||||
|
const assistantMessage: AiChatMessage = {
|
||||||
|
id: event.messageId,
|
||||||
|
chatId: currentChatIdRef.current || "",
|
||||||
|
role: "assistant",
|
||||||
|
content: currentContent || null,
|
||||||
|
toolCalls: currentToolCalls.length
|
||||||
|
? currentToolCalls
|
||||||
|
: null,
|
||||||
|
metadata: event.usage ? { tokenUsage: event.usage } : null,
|
||||||
|
createdAt: new Date().toISOString(),
|
||||||
|
};
|
||||||
|
|
||||||
|
setMessages((prev) => [...prev, assistantMessage]);
|
||||||
|
return [];
|
||||||
|
});
|
||||||
|
return "";
|
||||||
|
});
|
||||||
|
setIsStreaming(false);
|
||||||
|
queryClient.invalidateQueries({
|
||||||
|
queryKey: ["ai-chat", currentChatIdRef.current],
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case "error":
|
||||||
|
setError(event.message);
|
||||||
|
setErrorCode(event.code || null);
|
||||||
|
setIsRetryable(event.retryable || false);
|
||||||
|
setIsStreaming(false);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
(errorMsg) => {
|
||||||
|
setError(errorMsg);
|
||||||
|
setIsStreaming(false);
|
||||||
|
},
|
||||||
|
() => {
|
||||||
|
setIsStreaming(false);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
abortRef.current = abortController;
|
||||||
|
},
|
||||||
|
[isStreaming, navigate, queryClient],
|
||||||
|
);
|
||||||
|
|
||||||
|
const stopGeneration = useCallback(() => {
|
||||||
|
abortRef.current?.abort();
|
||||||
|
abortRef.current = null;
|
||||||
|
|
||||||
|
setStreamingContent((currentContent) => {
|
||||||
|
setStreamingToolCalls((currentToolCalls) => {
|
||||||
|
if (currentContent || currentToolCalls.length > 0) {
|
||||||
|
const partialMessage: AiChatMessage = {
|
||||||
|
id: `stopped-${Date.now()}`,
|
||||||
|
chatId: currentChatIdRef.current || "",
|
||||||
|
role: "assistant",
|
||||||
|
content: currentContent || null,
|
||||||
|
toolCalls: currentToolCalls.length ? currentToolCalls : null,
|
||||||
|
metadata: null,
|
||||||
|
createdAt: new Date().toISOString(),
|
||||||
|
};
|
||||||
|
setMessages((prev) => [...prev, partialMessage]);
|
||||||
|
}
|
||||||
|
return [];
|
||||||
|
});
|
||||||
|
return "";
|
||||||
|
});
|
||||||
|
|
||||||
|
setIsStreaming(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return {
|
||||||
|
messages,
|
||||||
|
streamingContent,
|
||||||
|
streamingToolCalls,
|
||||||
|
isStreaming,
|
||||||
|
error,
|
||||||
|
errorCode,
|
||||||
|
isRetryable,
|
||||||
|
sendMessage,
|
||||||
|
stopGeneration,
|
||||||
|
hydrateFromServer,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
import { useParams } from "react-router-dom";
|
||||||
|
import { ErrorBoundary } from "react-error-boundary";
|
||||||
|
import { Button } from "@mantine/core";
|
||||||
|
import { IconAlertTriangle } from "@tabler/icons-react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import AiChatLayout from "../components/ai-chat-layout";
|
||||||
|
import { EmptyState } from "@/components/ui/empty-state.tsx";
|
||||||
|
import classes from "../styles/ai-chat.module.css";
|
||||||
|
|
||||||
|
export default function AiChat() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { chatId } = useParams<{ chatId: string }>();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes.layout}>
|
||||||
|
<ErrorBoundary
|
||||||
|
resetKeys={[chatId]}
|
||||||
|
fallbackRender={({ resetErrorBoundary }) => (
|
||||||
|
<EmptyState
|
||||||
|
icon={IconAlertTriangle}
|
||||||
|
title={t("Failed to load chat. An error occurred.")}
|
||||||
|
action={
|
||||||
|
<Button
|
||||||
|
variant="default"
|
||||||
|
size="sm"
|
||||||
|
mt="xs"
|
||||||
|
onClick={resetErrorBoundary}
|
||||||
|
>
|
||||||
|
{t("Try again")}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<AiChatLayout />
|
||||||
|
</ErrorBoundary>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
import {
|
||||||
|
useQuery,
|
||||||
|
useMutation,
|
||||||
|
useQueryClient,
|
||||||
|
useInfiniteQuery,
|
||||||
|
} from "@tanstack/react-query";
|
||||||
|
import {
|
||||||
|
listChats,
|
||||||
|
getChatInfo,
|
||||||
|
deleteChat,
|
||||||
|
updateChatTitle,
|
||||||
|
searchChats,
|
||||||
|
} from "../services/ai-chat-service";
|
||||||
|
|
||||||
|
export function useChatsQuery() {
|
||||||
|
return useInfiniteQuery({
|
||||||
|
queryKey: ["ai-chats"],
|
||||||
|
queryFn: ({ pageParam }) =>
|
||||||
|
listChats({ cursor: pageParam, limit: 30 }),
|
||||||
|
initialPageParam: undefined as string | undefined,
|
||||||
|
getNextPageParam: (lastPage) =>
|
||||||
|
lastPage.meta.hasNextPage ? lastPage.meta.nextCursor : undefined,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useChatInfoQuery(chatId: string | undefined) {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: ["ai-chat", chatId],
|
||||||
|
queryFn: () => getChatInfo(chatId!),
|
||||||
|
enabled: !!chatId,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useDeleteChatMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: (chatId: string) => deleteChat(chatId),
|
||||||
|
onSuccess: () => {
|
||||||
|
queryClient.invalidateQueries({ queryKey: ["ai-chats"] });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useUpdateChatTitleMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: ({ chatId, title }: { chatId: string; title: string }) =>
|
||||||
|
updateChatTitle(chatId, title),
|
||||||
|
onSuccess: () => {
|
||||||
|
queryClient.invalidateQueries({ queryKey: ["ai-chats"] });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useSearchChatsQuery(query: string) {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: ["ai-chats-search", query],
|
||||||
|
queryFn: () => searchChats(query),
|
||||||
|
enabled: query.length > 0,
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,144 @@
|
|||||||
|
import api from "@/lib/api-client.ts";
|
||||||
|
import type {
|
||||||
|
AiChat,
|
||||||
|
AiChatMessage,
|
||||||
|
AiChatStreamEvent,
|
||||||
|
ChatAttachment,
|
||||||
|
} from "../types/ai-chat.types";
|
||||||
|
import { IPagination } from "@/lib/types.ts";
|
||||||
|
|
||||||
|
export async function createChat(): Promise<AiChat> {
|
||||||
|
const req = await api.post<AiChat>("/ai/chats/create");
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function listChats(params?: {
|
||||||
|
limit?: number;
|
||||||
|
cursor?: string;
|
||||||
|
}): Promise<IPagination<AiChat>> {
|
||||||
|
const req = await api.post("/ai/chats", params);
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getChatInfo(
|
||||||
|
chatId: string,
|
||||||
|
): Promise<{ chat: AiChat; messages: AiChatMessage[] }> {
|
||||||
|
const req = await api.post("/ai/chats/info", { chatId });
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function deleteChat(chatId: string): Promise<void> {
|
||||||
|
await api.post("/ai/chats/delete", { chatId });
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function updateChatTitle(
|
||||||
|
chatId: string,
|
||||||
|
title: string,
|
||||||
|
): Promise<void> {
|
||||||
|
await api.post("/ai/chats/update", { chatId, title });
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function searchChats(query: string): Promise<AiChat[]> {
|
||||||
|
const req = await api.post("/ai/chats/search", { query });
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function uploadChatFile(
|
||||||
|
file: File,
|
||||||
|
chatId?: string,
|
||||||
|
): Promise<ChatAttachment> {
|
||||||
|
const formData = new FormData();
|
||||||
|
formData.append("file", file);
|
||||||
|
if (chatId) {
|
||||||
|
formData.append("chatId", chatId);
|
||||||
|
}
|
||||||
|
return await api.post("/ai/chats/upload", formData, {
|
||||||
|
headers: { "Content-Type": "multipart/form-data" },
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function sendChatMessage(
|
||||||
|
params: {
|
||||||
|
chatId?: string;
|
||||||
|
content: string;
|
||||||
|
mentionedPageIds?: string[];
|
||||||
|
contextPageId?: string;
|
||||||
|
attachmentIds?: string[];
|
||||||
|
},
|
||||||
|
onEvent: (event: AiChatStreamEvent) => void,
|
||||||
|
onError?: (error: string) => void,
|
||||||
|
onComplete?: () => void,
|
||||||
|
): AbortController {
|
||||||
|
const abortController = new AbortController();
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
try {
|
||||||
|
const response = await fetch("/api/ai/chats/send", {
|
||||||
|
method: "POST",
|
||||||
|
headers: { "Content-Type": "application/json" },
|
||||||
|
body: JSON.stringify(params),
|
||||||
|
signal: abortController.signal,
|
||||||
|
credentials: "include",
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
const errorBody = await response.text();
|
||||||
|
let errorMessage = `HTTP error ${response.status}`;
|
||||||
|
try {
|
||||||
|
const parsed = JSON.parse(errorBody);
|
||||||
|
errorMessage = parsed.message || errorMessage;
|
||||||
|
} catch {
|
||||||
|
// use default
|
||||||
|
}
|
||||||
|
onError?.(errorMessage);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const reader = response.body?.getReader();
|
||||||
|
const decoder = new TextDecoder();
|
||||||
|
|
||||||
|
if (!reader) {
|
||||||
|
onError?.("Response body is not readable");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let buffer = "";
|
||||||
|
try {
|
||||||
|
while (true) {
|
||||||
|
const { done, value } = await reader.read();
|
||||||
|
if (done) break;
|
||||||
|
|
||||||
|
buffer += decoder.decode(value, { stream: true });
|
||||||
|
const lines = buffer.split("\n");
|
||||||
|
buffer = lines.pop() || "";
|
||||||
|
|
||||||
|
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) as AiChatStreamEvent;
|
||||||
|
onEvent(parsed);
|
||||||
|
} catch {
|
||||||
|
// Skip invalid JSON
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
reader.releaseLock();
|
||||||
|
}
|
||||||
|
|
||||||
|
onComplete?.();
|
||||||
|
} catch (error: any) {
|
||||||
|
if (error.name !== "AbortError") {
|
||||||
|
onError?.(error.message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
|
||||||
|
return abortController;
|
||||||
|
}
|
||||||
@@ -0,0 +1,169 @@
|
|||||||
|
.layout {
|
||||||
|
display: flex;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.main {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: calc(100vh - 45px - 2 * var(--mantine-spacing-md));
|
||||||
|
max-width: 900px;
|
||||||
|
margin: 0 auto;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageListWrapper {
|
||||||
|
flex: 1;
|
||||||
|
position: relative;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
min-height: 0;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageList {
|
||||||
|
flex: 1;
|
||||||
|
overflow-y: auto;
|
||||||
|
padding: var(--mantine-spacing-md) var(--mantine-spacing-lg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageErrorFallback {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 6px;
|
||||||
|
padding: 8px 12px;
|
||||||
|
margin-bottom: var(--mantine-spacing-lg);
|
||||||
|
border-radius: var(--mantine-radius-sm);
|
||||||
|
background: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-6));
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
}
|
||||||
|
|
||||||
|
.scrollToBottomButton {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 12px;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
width: 32px;
|
||||||
|
height: 32px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border-radius: 50%;
|
||||||
|
border: 1px solid light-dark(var(--mantine-color-gray-3), var(--mantine-color-dark-4));
|
||||||
|
background: light-dark(var(--mantine-color-white), var(--mantine-color-dark-6));
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 0;
|
||||||
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);
|
||||||
|
transition: background-color 120ms ease, border-color 120ms ease, transform 120ms ease;
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.scrollToBottomButton:hover {
|
||||||
|
background: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-5));
|
||||||
|
border-color: light-dark(var(--mantine-color-gray-4), var(--mantine-color-dark-3));
|
||||||
|
}
|
||||||
|
|
||||||
|
.scrollToBottomButton:active {
|
||||||
|
transform: translateX(-50%) scale(0.95);
|
||||||
|
}
|
||||||
|
|
||||||
|
.inputArea {
|
||||||
|
padding: var(--mantine-spacing-xs) var(--mantine-spacing-lg) var(--mantine-spacing-lg);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Empty state - Notion AI style centered layout */
|
||||||
|
.emptyState {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
padding: var(--mantine-spacing-xl) var(--mantine-spacing-lg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.emptyStateIcon {
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
margin-bottom: var(--mantine-spacing-sm);
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
}
|
||||||
|
|
||||||
|
.emptyStateBrand {
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
font-weight: 600;
|
||||||
|
letter-spacing: 0.08em;
|
||||||
|
text-transform: uppercase;
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
margin-bottom: var(--mantine-spacing-xs);
|
||||||
|
}
|
||||||
|
|
||||||
|
.emptyStateTitle {
|
||||||
|
font-size: 1.5rem;
|
||||||
|
font-weight: 600;
|
||||||
|
color: light-dark(var(--mantine-color-gray-8), var(--mantine-color-dark-0));
|
||||||
|
margin-bottom: var(--mantine-spacing-xl);
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.emptyStateInput {
|
||||||
|
width: 100%;
|
||||||
|
max-width: 600px;
|
||||||
|
margin-bottom: var(--mantine-spacing-xl);
|
||||||
|
padding: 6px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestionsSection {
|
||||||
|
width: 100%;
|
||||||
|
max-width: 600px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestionsLabel {
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
font-weight: 500;
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.5px;
|
||||||
|
margin-bottom: var(--mantine-spacing-sm);
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestionsGrid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(2, 1fr);
|
||||||
|
gap: var(--mantine-spacing-sm);
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestionCard {
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-start;
|
||||||
|
gap: var(--mantine-spacing-sm);
|
||||||
|
padding: var(--mantine-spacing-sm) var(--mantine-spacing-md);
|
||||||
|
border: 1px solid light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
||||||
|
border-radius: var(--mantine-radius-md);
|
||||||
|
cursor: pointer;
|
||||||
|
background: transparent;
|
||||||
|
transition: background-color 150ms, border-color 150ms;
|
||||||
|
text-align: left;
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
background: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-6));
|
||||||
|
border-color: light-dark(var(--mantine-color-gray-3), var(--mantine-color-dark-4));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestionIcon {
|
||||||
|
flex-shrink: 0;
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
margin-top: 1px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.suggestionText {
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
line-height: 1.4;
|
||||||
|
}
|
||||||
@@ -0,0 +1,139 @@
|
|||||||
|
.panel {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: 100%;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolbar {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
padding: 0 0 var(--mantine-spacing-sm) 0;
|
||||||
|
border-bottom: 1px solid light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolbarSpacer {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.titleButton {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
padding: 4px 8px;
|
||||||
|
border-radius: var(--mantine-radius-sm);
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
font-weight: 500;
|
||||||
|
color: light-dark(var(--mantine-color-gray-8), var(--mantine-color-dark-0));
|
||||||
|
max-width: 60%;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.titleButton:hover {
|
||||||
|
background-color: light-dark(var(--mantine-color-gray-1), var(--mantine-color-dark-6));
|
||||||
|
}
|
||||||
|
|
||||||
|
.titleText {
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messages {
|
||||||
|
flex: 1;
|
||||||
|
overflow-y: auto;
|
||||||
|
padding: var(--mantine-spacing-sm) 0;
|
||||||
|
scroll-behavior: smooth;
|
||||||
|
}
|
||||||
|
|
||||||
|
.inputArea {
|
||||||
|
padding-top: var(--mantine-spacing-sm);
|
||||||
|
}
|
||||||
|
|
||||||
|
.emptyState {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: var(--mantine-spacing-md);
|
||||||
|
padding: var(--mantine-spacing-xl) var(--mantine-spacing-sm);
|
||||||
|
}
|
||||||
|
|
||||||
|
.emptyStateIcon {
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
}
|
||||||
|
|
||||||
|
.emptyStateTitle {
|
||||||
|
font-size: var(--mantine-font-size-lg);
|
||||||
|
font-weight: 600;
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.quickActions {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 6px;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.quickAction {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: var(--mantine-spacing-sm);
|
||||||
|
padding: var(--mantine-spacing-xs) var(--mantine-spacing-sm);
|
||||||
|
border: 1px solid light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
||||||
|
border-radius: var(--mantine-radius-md);
|
||||||
|
cursor: pointer;
|
||||||
|
background: transparent;
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
text-align: left;
|
||||||
|
width: 100%;
|
||||||
|
transition: background-color 150ms, border-color 150ms;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
background: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-6));
|
||||||
|
border-color: light-dark(var(--mantine-color-gray-3), var(--mantine-color-dark-4));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.quickActionIcon {
|
||||||
|
flex-shrink: 0;
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
}
|
||||||
|
|
||||||
|
.historyList {
|
||||||
|
max-height: 300px;
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.historyItem {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding: var(--mantine-spacing-xs) var(--mantine-spacing-sm);
|
||||||
|
cursor: pointer;
|
||||||
|
border-radius: var(--mantine-radius-sm);
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
transition: background-color 150ms;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
background: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-6));
|
||||||
|
}
|
||||||
|
|
||||||
|
&[data-active] {
|
||||||
|
background: light-dark(var(--mantine-color-gray-1), var(--mantine-color-dark-5));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.historyItemTitle {
|
||||||
|
flex: 1;
|
||||||
|
min-width: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
@@ -0,0 +1,242 @@
|
|||||||
|
.inputWrapper {
|
||||||
|
position: relative;
|
||||||
|
overflow: hidden;
|
||||||
|
border: 1px solid
|
||||||
|
light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
||||||
|
border-radius: 16px;
|
||||||
|
background: light-dark(var(--mantine-color-white), var(--mantine-color-dark-7));
|
||||||
|
box-shadow: light-dark(
|
||||||
|
0 2px 40px 4px rgba(0, 0, 0, 0.07),
|
||||||
|
0 2px 40px 4px rgba(0, 0, 0, 0.5)
|
||||||
|
);
|
||||||
|
transition:
|
||||||
|
border-color 150ms,
|
||||||
|
box-shadow 150ms;
|
||||||
|
|
||||||
|
&:focus-within {
|
||||||
|
border-color: light-dark(
|
||||||
|
var(--mantine-color-gray-3),
|
||||||
|
var(--mantine-color-dark-4)
|
||||||
|
);
|
||||||
|
box-shadow: light-dark(
|
||||||
|
0 4px 48px 6px rgba(0, 0, 0, 0.09),
|
||||||
|
0 4px 48px 6px rgba(0, 0, 0, 0.6)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.inputWrapperFlat {
|
||||||
|
position: relative;
|
||||||
|
overflow: hidden;
|
||||||
|
border: 1px solid light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
||||||
|
border-radius: 12px;
|
||||||
|
background: light-dark(var(--mantine-color-white), var(--mantine-color-dark-7));
|
||||||
|
box-shadow: none;
|
||||||
|
transition: border-color 150ms;
|
||||||
|
|
||||||
|
&:focus-within {
|
||||||
|
border-color: light-dark(var(--mantine-color-gray-3), var(--mantine-color-dark-4));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.disclaimer {
|
||||||
|
margin-top: 6px;
|
||||||
|
text-align: center;
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
}
|
||||||
|
|
||||||
|
.attachmentChips {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 6px;
|
||||||
|
padding: 10px 14px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.attachmentChip {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 5px;
|
||||||
|
padding: 4px 8px;
|
||||||
|
border-radius: 8px;
|
||||||
|
background: light-dark(var(--mantine-color-gray-1), var(--mantine-color-dark-6));
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
max-width: 200px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.attachmentChipUploading {
|
||||||
|
opacity: 0.55;
|
||||||
|
}
|
||||||
|
|
||||||
|
.attachmentChipName {
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.attachmentChipRemove {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 0;
|
||||||
|
margin-left: 2px;
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
border-radius: 50%;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
color: light-dark(var(--mantine-color-gray-8), var(--mantine-color-dark-0));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.editorContent {
|
||||||
|
overflow: hidden;
|
||||||
|
|
||||||
|
:global(.ProseMirror) {
|
||||||
|
outline: none;
|
||||||
|
border: none;
|
||||||
|
background-color: transparent;
|
||||||
|
padding: 14px 18px 8px;
|
||||||
|
font-size: 15px;
|
||||||
|
line-height: 1.6;
|
||||||
|
max-height: 200px;
|
||||||
|
overflow-y: auto;
|
||||||
|
min-height: 24px;
|
||||||
|
color: light-dark(var(--mantine-color-gray-9), var(--mantine-color-dark-0));
|
||||||
|
}
|
||||||
|
|
||||||
|
:global(.ProseMirror p) {
|
||||||
|
margin-block-start: 0;
|
||||||
|
margin-block-end: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
:global(.ProseMirror p.is-editor-empty:first-child::before) {
|
||||||
|
color: light-dark(var(--mantine-color-gray-4), var(--mantine-color-dark-3));
|
||||||
|
content: attr(data-placeholder);
|
||||||
|
float: left;
|
||||||
|
height: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.actions {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-end;
|
||||||
|
padding: 4px 12px 10px;
|
||||||
|
gap: var(--mantine-spacing-xs);
|
||||||
|
}
|
||||||
|
|
||||||
|
.sendButton {
|
||||||
|
width: 28px;
|
||||||
|
height: 28px;
|
||||||
|
min-width: 28px;
|
||||||
|
min-height: 28px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border: none;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background-color 150ms, opacity 150ms;
|
||||||
|
background: light-dark(var(--mantine-color-dark-9), var(--mantine-color-gray-0));
|
||||||
|
color: light-dark(var(--mantine-color-white), var(--mantine-color-dark-9));
|
||||||
|
|
||||||
|
&:disabled {
|
||||||
|
opacity: 0.25;
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
&:not(:disabled) {
|
||||||
|
opacity: 0.85;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.attachButton {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 2px;
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
transition: color 150ms;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.plusButton {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
border-radius: 50%;
|
||||||
|
border: 1px solid light-dark(var(--mantine-color-gray-3), var(--mantine-color-dark-4));
|
||||||
|
background: none;
|
||||||
|
cursor: pointer;
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
transition: color 150ms, background-color 150ms;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
background: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-6));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.plusMenuItem {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: var(--mantine-spacing-sm);
|
||||||
|
padding: var(--mantine-spacing-xs) var(--mantine-spacing-sm);
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
cursor: pointer;
|
||||||
|
width: 100%;
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
border-radius: var(--mantine-radius-sm);
|
||||||
|
transition: background-color 150ms;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
background: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-6));
|
||||||
|
}
|
||||||
|
|
||||||
|
&:disabled {
|
||||||
|
opacity: 0.45;
|
||||||
|
cursor: not-allowed;
|
||||||
|
background: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.plusMenuIcon {
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
}
|
||||||
|
|
||||||
|
.stopButton {
|
||||||
|
width: 28px;
|
||||||
|
height: 28px;
|
||||||
|
min-width: 28px;
|
||||||
|
min-height: 28px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border: 1px solid light-dark(var(--mantine-color-gray-3), var(--mantine-color-dark-4));
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background-color 150ms;
|
||||||
|
background: light-dark(var(--mantine-color-white), var(--mantine-color-dark-6));
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
background: light-dark(var(--mantine-color-gray-1), var(--mantine-color-dark-5));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,286 @@
|
|||||||
|
.message {
|
||||||
|
margin-bottom: var(--mantine-spacing-lg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.userMessage {
|
||||||
|
composes: message;
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
}
|
||||||
|
|
||||||
|
.userBubble {
|
||||||
|
max-width: 75%;
|
||||||
|
padding: 10px 16px;
|
||||||
|
border-radius: 18px;
|
||||||
|
background: light-dark(var(--mantine-color-gray-1), var(--mantine-color-dark-6));
|
||||||
|
color: light-dark(var(--mantine-color-gray-9), var(--mantine-color-dark-0));
|
||||||
|
font-size: 15px;
|
||||||
|
line-height: 1.6;
|
||||||
|
word-wrap: break-word;
|
||||||
|
overflow-wrap: break-word;
|
||||||
|
}
|
||||||
|
|
||||||
|
[data-aside-chat] .userBubble {
|
||||||
|
background: light-dark(var(--mantine-color-white), var(--mantine-color-dark-7));
|
||||||
|
border: 1px solid light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
||||||
|
}
|
||||||
|
|
||||||
|
.userBubble p {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageAttachments {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 4px;
|
||||||
|
margin-bottom: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageAttachmentChip {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
padding: 2px 8px;
|
||||||
|
border-radius: 6px;
|
||||||
|
background: light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-2));
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
max-width: 180px;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.assistantMessage {
|
||||||
|
composes: message;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent {
|
||||||
|
font-size: 15px;
|
||||||
|
line-height: 1.7;
|
||||||
|
color: light-dark(var(--mantine-color-gray-8), var(--mantine-color-dark-1));
|
||||||
|
word-wrap: break-word;
|
||||||
|
overflow-wrap: break-word;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent p {
|
||||||
|
margin: 0 0 0.75em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent p:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent ul,
|
||||||
|
.messageContent ol {
|
||||||
|
margin: 0.5em 0 0.75em 0;
|
||||||
|
padding-left: 1.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent li {
|
||||||
|
margin-bottom: 0.3em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent h1,
|
||||||
|
.messageContent h2,
|
||||||
|
.messageContent h3 {
|
||||||
|
margin: 1em 0 0.5em 0;
|
||||||
|
font-weight: 600;
|
||||||
|
color: light-dark(var(--mantine-color-gray-9), var(--mantine-color-dark-0));
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent h1 {
|
||||||
|
font-size: 1.4em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent h2 {
|
||||||
|
font-size: 1.2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent h3 {
|
||||||
|
font-size: 1.05em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent pre {
|
||||||
|
background-color: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-7));
|
||||||
|
padding: var(--mantine-spacing-sm) var(--mantine-spacing-md);
|
||||||
|
border-radius: var(--mantine-radius-md);
|
||||||
|
overflow-x: auto;
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
margin: 0.75em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent code {
|
||||||
|
background-color: light-dark(var(--mantine-color-gray-1), var(--mantine-color-dark-6));
|
||||||
|
padding: 2px 6px;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 0.88em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent pre code {
|
||||||
|
background: none;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent blockquote {
|
||||||
|
border-left: 3px solid light-dark(var(--mantine-color-gray-3), var(--mantine-color-dark-4));
|
||||||
|
padding-left: var(--mantine-spacing-md);
|
||||||
|
margin: 0.75em 0;
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-2));
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent a {
|
||||||
|
color: light-dark(var(--mantine-color-blue-7), var(--mantine-color-blue-4));
|
||||||
|
text-decoration: none;
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent a[href^="/s/"],
|
||||||
|
.messageContent a[href^="/p/"] {
|
||||||
|
color: light-dark(var(--mantine-color-dark-4), var(--mantine-color-dark-1));
|
||||||
|
font-weight: 500;
|
||||||
|
text-decoration: none;
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
@mixin light {
|
||||||
|
border-bottom: 0.05em solid var(--mantine-color-dark-0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@mixin dark {
|
||||||
|
border-bottom: 0.05em solid var(--mantine-color-dark-2);
|
||||||
|
}
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
text-decoration: none;
|
||||||
|
@mixin light {
|
||||||
|
border-bottom-color: var(--mantine-color-dark-2);
|
||||||
|
}
|
||||||
|
@mixin dark {
|
||||||
|
border-bottom-color: var(--mantine-color-dark-0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageContent hr {
|
||||||
|
border: none;
|
||||||
|
border-top: 1px solid light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
||||||
|
margin: 1em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolGroup {
|
||||||
|
margin: 6px 0;
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolGroupHeader {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 6px;
|
||||||
|
cursor: pointer;
|
||||||
|
user-select: none;
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
line-height: 1.4;
|
||||||
|
transition: color 120ms ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolGroupHeader:hover {
|
||||||
|
color: light-dark(var(--mantine-color-gray-8), var(--mantine-color-dark-0));
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolGroupLabel {
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolGroupSteps {
|
||||||
|
margin-top: 4px;
|
||||||
|
padding-left: 14px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolStep {
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolStepRow {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
cursor: pointer;
|
||||||
|
user-select: none;
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
line-height: 1.5;
|
||||||
|
transition: color 120ms ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolStepRow:hover {
|
||||||
|
color: light-dark(var(--mantine-color-gray-8), var(--mantine-color-dark-0));
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolStepBullet {
|
||||||
|
display: inline-block;
|
||||||
|
width: 8px;
|
||||||
|
text-align: center;
|
||||||
|
opacity: 0.6;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolStepDetails {
|
||||||
|
margin-top: 4px;
|
||||||
|
margin-left: 18px;
|
||||||
|
padding: 6px 10px;
|
||||||
|
border-radius: var(--mantine-radius-sm);
|
||||||
|
background: light-dark(var(--mantine-color-gray-0), var(--mantine-color-dark-7));
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
font-size: 11px;
|
||||||
|
line-height: 1.5;
|
||||||
|
overflow-x: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.messageActions {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
margin-top: 4px;
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
}
|
||||||
|
|
||||||
|
.processingIndicator {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 6px;
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
}
|
||||||
|
|
||||||
|
.processingSpinner {
|
||||||
|
animation: spin 1s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spin {
|
||||||
|
from {
|
||||||
|
transform: rotate(0deg);
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
transform: rotate(360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.streamingCursor {
|
||||||
|
display: inline-block;
|
||||||
|
width: 2px;
|
||||||
|
height: 1em;
|
||||||
|
background: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-0));
|
||||||
|
animation: blink 1s step-end infinite;
|
||||||
|
vertical-align: text-bottom;
|
||||||
|
margin-left: 1px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes blink {
|
||||||
|
50% {
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,138 @@
|
|||||||
|
.sidebar {
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
padding: var(--mantine-spacing-md);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: var(--mantine-spacing-xs);
|
||||||
|
}
|
||||||
|
|
||||||
|
.header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding-bottom: var(--mantine-spacing-xs);
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
font-weight: 600;
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchInput {
|
||||||
|
margin-bottom: var(--mantine-spacing-xs);
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatList {
|
||||||
|
flex: 1;
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatGroup + .chatGroup {
|
||||||
|
margin-top: var(--mantine-spacing-sm);
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatGroupLabel {
|
||||||
|
padding: 4px var(--mantine-spacing-xs);
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
font-weight: 600;
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-2));
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatListEmpty {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
padding: var(--mantine-spacing-xl) var(--mantine-spacing-md);
|
||||||
|
text-align: center;
|
||||||
|
gap: 4px;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatListEmptyIcon {
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
margin-bottom: var(--mantine-spacing-xs);
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatListEmptyTitle {
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
font-weight: 600;
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatListEmptyHint {
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
color: light-dark(var(--mantine-color-gray-6), var(--mantine-color-dark-3));
|
||||||
|
line-height: 1.4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatItem {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding: 8px var(--mantine-spacing-xs);
|
||||||
|
border-radius: var(--mantine-radius-sm);
|
||||||
|
cursor: pointer;
|
||||||
|
text-decoration: none;
|
||||||
|
color: light-dark(var(--mantine-color-gray-7), var(--mantine-color-dark-1));
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
user-select: none;
|
||||||
|
gap: var(--mantine-spacing-xs);
|
||||||
|
|
||||||
|
@mixin hover {
|
||||||
|
background-color: light-dark(
|
||||||
|
var(--mantine-color-gray-1),
|
||||||
|
var(--mantine-color-dark-6)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
&[data-active] {
|
||||||
|
background-color: light-dark(
|
||||||
|
var(--mantine-color-gray-2),
|
||||||
|
var(--mantine-color-dark-6)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatItemTitle {
|
||||||
|
flex: 1;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatItemDate {
|
||||||
|
font-size: var(--mantine-font-size-xs);
|
||||||
|
color: light-dark(var(--mantine-color-gray-5), var(--mantine-color-dark-3));
|
||||||
|
white-space: nowrap;
|
||||||
|
transition: opacity 150ms;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatItemRenameInput {
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
padding: 0;
|
||||||
|
height: auto;
|
||||||
|
min-height: 0;
|
||||||
|
background: transparent;
|
||||||
|
color: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatItem:hover .chatItemDate {
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatItemActions {
|
||||||
|
position: absolute;
|
||||||
|
right: var(--mantine-spacing-xs);
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 150ms;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatItem {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chatItem:hover .chatItemActions {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
export type AiChat = {
|
||||||
|
id: string;
|
||||||
|
workspaceId: string;
|
||||||
|
creatorId: string;
|
||||||
|
title: string | null;
|
||||||
|
createdAt: string;
|
||||||
|
updatedAt: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type AiChatToolCall = {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
args: Record<string, unknown>;
|
||||||
|
result?: unknown;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type AiChatMessage = {
|
||||||
|
id: string;
|
||||||
|
chatId: string;
|
||||||
|
role: 'user' | 'assistant' | 'tool';
|
||||||
|
content: string | null;
|
||||||
|
toolCalls: AiChatToolCall[] | null;
|
||||||
|
metadata: Record<string, unknown> | null;
|
||||||
|
createdAt: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type AiChatStreamEvent =
|
||||||
|
| { type: 'chat_created'; chatId: string }
|
||||||
|
| { type: 'content'; text: string }
|
||||||
|
| { type: 'tool_call'; id: string; name: string; args: Record<string, unknown> }
|
||||||
|
| { type: 'tool_result'; id: string; result: unknown }
|
||||||
|
| { type: 'done'; messageId: string; usage?: Record<string, number> }
|
||||||
|
| { type: 'error'; message: string; code?: string; retryable?: boolean };
|
||||||
|
|
||||||
|
export type PageMention = {
|
||||||
|
id: string;
|
||||||
|
title: string;
|
||||||
|
slugId: string;
|
||||||
|
spaceSlug?: string;
|
||||||
|
icon?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ChatAttachment = {
|
||||||
|
id: string;
|
||||||
|
fileName: string;
|
||||||
|
fileExt: string;
|
||||||
|
fileSize: number;
|
||||||
|
mimeType: string;
|
||||||
|
};
|
||||||
@@ -0,0 +1,45 @@
|
|||||||
|
import type { AiChat } from "../types/ai-chat.types";
|
||||||
|
|
||||||
|
export type ChatGroup = { key: string; label: string; chats: AiChat[] };
|
||||||
|
|
||||||
|
export function groupChatsByAge(
|
||||||
|
chats: AiChat[],
|
||||||
|
t: (key: string) => string,
|
||||||
|
): ChatGroup[] {
|
||||||
|
if (chats.length === 0) return [];
|
||||||
|
|
||||||
|
const now = new Date();
|
||||||
|
const startOfToday = new Date(
|
||||||
|
now.getFullYear(),
|
||||||
|
now.getMonth(),
|
||||||
|
now.getDate(),
|
||||||
|
).getTime();
|
||||||
|
const startOfYesterday = startOfToday - 24 * 60 * 60 * 1000;
|
||||||
|
const startOfLast7 = startOfToday - 7 * 24 * 60 * 60 * 1000;
|
||||||
|
const startOfLast30 = startOfToday - 30 * 24 * 60 * 60 * 1000;
|
||||||
|
|
||||||
|
const buckets: Record<string, ChatGroup> = {
|
||||||
|
today: { key: "today", label: t("Today"), chats: [] },
|
||||||
|
yesterday: { key: "yesterday", label: t("Yesterday"), chats: [] },
|
||||||
|
last7: { key: "last7", label: t("Previous 7 days"), chats: [] },
|
||||||
|
last30: { key: "last30", label: t("Previous 30 days"), chats: [] },
|
||||||
|
older: { key: "older", label: t("Older"), chats: [] },
|
||||||
|
};
|
||||||
|
|
||||||
|
for (const chat of chats) {
|
||||||
|
const ts = new Date(chat.updatedAt).getTime();
|
||||||
|
if (ts >= startOfToday) buckets.today.chats.push(chat);
|
||||||
|
else if (ts >= startOfYesterday) buckets.yesterday.chats.push(chat);
|
||||||
|
else if (ts >= startOfLast7) buckets.last7.chats.push(chat);
|
||||||
|
else if (ts >= startOfLast30) buckets.last30.chats.push(chat);
|
||||||
|
else buckets.older.chats.push(chat);
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
buckets.today,
|
||||||
|
buckets.yesterday,
|
||||||
|
buckets.last7,
|
||||||
|
buckets.last30,
|
||||||
|
buckets.older,
|
||||||
|
].filter((b) => b.chats.length > 0);
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
.aiMenu {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
width: 100%;
|
||||||
|
max-width: 600px;
|
||||||
|
min-height: 2.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.aiInput {
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
& input {
|
||||||
|
height: 44px;
|
||||||
|
border-radius: 22px;
|
||||||
|
padding-left: 20px;
|
||||||
|
padding-right: 40px;
|
||||||
|
border: 1px solid
|
||||||
|
light-dark(var(--mantine-color-gray-3), var(--mantine-color-dark-4));
|
||||||
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);
|
||||||
|
font-size: var(--mantine-font-size-sm);
|
||||||
|
|
||||||
|
&:focus {
|
||||||
|
border-color: light-dark(
|
||||||
|
var(--mantine-color-gray-4),
|
||||||
|
var(--mantine-color-dark-3)
|
||||||
|
);
|
||||||
|
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.12);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.menuItemSelected {
|
||||||
|
background-color: var(--mantine-color-gray-1);
|
||||||
|
|
||||||
|
@mixin dark {
|
||||||
|
background-color: var(--mantine-color-dark-5);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.resultPreview {
|
||||||
|
background-color: light-dark(
|
||||||
|
var(--mantine-color-white),
|
||||||
|
var(--mantine-color-dark-6)
|
||||||
|
);
|
||||||
|
border: 1px solid
|
||||||
|
light-dark(var(--mantine-color-gray-3), var(--mantine-color-dark-4));
|
||||||
|
box-shadow: 0 4px 16px rgba(0, 0, 0, 0.1);
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.resultPreviewWrapper {
|
||||||
|
font-size: var(--mantine-font-size-md);
|
||||||
|
line-height: 1.6;
|
||||||
|
padding: var(--mantine-spacing-md);
|
||||||
|
|
||||||
|
*:first-child {
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
*:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,349 @@
|
|||||||
|
import { Editor } from "@tiptap/react";
|
||||||
|
import { ActionIcon, TextInput } from "@mantine/core";
|
||||||
|
import { useDebouncedCallback, useMediaQuery } from "@mantine/hooks";
|
||||||
|
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
||||||
|
import { createPortal } from "react-dom";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { IconArrowUp } from "@tabler/icons-react";
|
||||||
|
import { showAiMenuAtom } from "@/features/editor/atoms/editor-atoms.ts";
|
||||||
|
import { useAiGenerateStreamMutation } from "@/ee/ai/queries/ai-query.ts";
|
||||||
|
import { AiAction } from "@/ee/ai/types/ai.types.ts";
|
||||||
|
import { CommandItem, commandItems, CommandSet } from "./command-items.ts";
|
||||||
|
import { CommandSelector } from "./command-selector.tsx";
|
||||||
|
import { ResultPreview } from "./result-preview.tsx";
|
||||||
|
import classes from "./ai-menu.module.css";
|
||||||
|
import { marked } from "marked";
|
||||||
|
import { DOMSerializer } from "@tiptap/pm/model";
|
||||||
|
import { copyToClipboard, htmlToMarkdown } from "@docmost/editor-ext";
|
||||||
|
import { useLocation } from "react-router-dom";
|
||||||
|
|
||||||
|
interface EditorAiMenuProps {
|
||||||
|
editor: Editor | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const EditorAiMenu = ({ editor }: EditorAiMenuProps): JSX.Element | null => {
|
||||||
|
const aiGenerateStreamMutation = useAiGenerateStreamMutation();
|
||||||
|
const location = useLocation();
|
||||||
|
const isSmBreakpoint = useMediaQuery("(max-width: 48em)");
|
||||||
|
const [showAiMenu, setShowAiMenu] = useAtom(showAiMenuAtom);
|
||||||
|
const containerRef = useRef<HTMLDivElement | null>(null);
|
||||||
|
const inputRef = useRef<HTMLInputElement | null>(null);
|
||||||
|
const [prompt, setPrompt] = useState("");
|
||||||
|
const [output, setOutput] = useState("");
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
const [selectedIndex, setSelectedIndex] = useState(-1);
|
||||||
|
const [activeCommandSet, setActiveCommandSet] = useState<CommandSet>("main");
|
||||||
|
const [lastAction, setLastAction] = useState<CommandItem | null>(null);
|
||||||
|
const [menuPlacement, setMenuPlacement] = useState<{
|
||||||
|
top: number;
|
||||||
|
left: number;
|
||||||
|
width: number;
|
||||||
|
}>({
|
||||||
|
top: 0,
|
||||||
|
left: 0,
|
||||||
|
width: 0,
|
||||||
|
});
|
||||||
|
const currentItems = useMemo(() => {
|
||||||
|
return commandItems[activeCommandSet].filter((item) => {
|
||||||
|
return item.name.toLowerCase().includes(prompt.toLowerCase());
|
||||||
|
});
|
||||||
|
}, [prompt, output, activeCommandSet]);
|
||||||
|
const updateMenuPlacement = useCallback(() => {
|
||||||
|
if (!editor || !showAiMenu) return;
|
||||||
|
|
||||||
|
const { view } = editor;
|
||||||
|
const { from, to } = editor.state.selection;
|
||||||
|
const editorRect = view.dom.getBoundingClientRect();
|
||||||
|
const fromCoords = view.coordsAtPos(from);
|
||||||
|
const toCoords = view.coordsAtPos(to);
|
||||||
|
const topOffset = 8;
|
||||||
|
const editorPadding = isSmBreakpoint ? 16 : 48;
|
||||||
|
|
||||||
|
const anchorBottom =
|
||||||
|
toCoords.bottom > 0 && toCoords.bottom < window.innerHeight
|
||||||
|
? toCoords.bottom
|
||||||
|
: fromCoords.bottom;
|
||||||
|
|
||||||
|
const menuMaxWidth = 600;
|
||||||
|
const editorLeft = editorRect.left + editorPadding;
|
||||||
|
const editorRight = editorRect.right - editorPadding;
|
||||||
|
const availableWidth = editorRight - editorLeft;
|
||||||
|
const menuWidth = Math.min(menuMaxWidth, availableWidth);
|
||||||
|
|
||||||
|
let menuLeft = Math.max(editorLeft, fromCoords.left);
|
||||||
|
if (menuLeft + menuWidth > editorRight) {
|
||||||
|
menuLeft = editorRight - menuWidth;
|
||||||
|
}
|
||||||
|
menuLeft = Math.max(editorLeft, menuLeft);
|
||||||
|
|
||||||
|
setMenuPlacement({
|
||||||
|
top: anchorBottom + topOffset + window.scrollY,
|
||||||
|
left: menuLeft + window.scrollX,
|
||||||
|
width: menuWidth,
|
||||||
|
});
|
||||||
|
}, [editor, showAiMenu, isSmBreakpoint]);
|
||||||
|
const resetMenu = useCallback(() => {
|
||||||
|
setPrompt("");
|
||||||
|
setOutput("");
|
||||||
|
setActiveCommandSet("main");
|
||||||
|
setLastAction(null);
|
||||||
|
aiGenerateStreamMutation.reset();
|
||||||
|
}, [aiGenerateStreamMutation.reset]);
|
||||||
|
const debouncedUpdateMenuPlacement = useDebouncedCallback(
|
||||||
|
updateMenuPlacement,
|
||||||
|
60,
|
||||||
|
);
|
||||||
|
const handleGenerate = useCallback(
|
||||||
|
(item?: CommandItem) => {
|
||||||
|
if (!editor || isLoading) return;
|
||||||
|
|
||||||
|
let command: CommandItem | null = item || null;
|
||||||
|
|
||||||
|
if (!command) {
|
||||||
|
if (!prompt) return;
|
||||||
|
|
||||||
|
command = {
|
||||||
|
id: "custom",
|
||||||
|
name: "Custom",
|
||||||
|
action: AiAction.CUSTOM,
|
||||||
|
prompt,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const { from, to } = editor.state.selection;
|
||||||
|
const slice = editor.state.doc.slice(from, to);
|
||||||
|
const serializer = DOMSerializer.fromSchema(editor.schema);
|
||||||
|
const fragment = serializer.serializeFragment(slice.content);
|
||||||
|
const wrapper = document.createElement("div");
|
||||||
|
wrapper.appendChild(fragment);
|
||||||
|
const content = htmlToMarkdown(wrapper.innerHTML);
|
||||||
|
|
||||||
|
setOutput("");
|
||||||
|
setIsLoading(true);
|
||||||
|
aiGenerateStreamMutation.mutate({
|
||||||
|
action: command.action,
|
||||||
|
prompt: command.prompt,
|
||||||
|
content,
|
||||||
|
onChunk: (chunk) => {
|
||||||
|
setOutput((output) => output + chunk.content);
|
||||||
|
},
|
||||||
|
onComplete: () => {
|
||||||
|
setPrompt("");
|
||||||
|
setIsLoading(false);
|
||||||
|
setActiveCommandSet("result");
|
||||||
|
},
|
||||||
|
onError: () => {
|
||||||
|
setIsLoading(false);
|
||||||
|
resetMenu();
|
||||||
|
},
|
||||||
|
});
|
||||||
|
setLastAction(command);
|
||||||
|
},
|
||||||
|
[
|
||||||
|
editor,
|
||||||
|
prompt,
|
||||||
|
isLoading,
|
||||||
|
aiGenerateStreamMutation.mutateAsync,
|
||||||
|
resetMenu,
|
||||||
|
],
|
||||||
|
);
|
||||||
|
const handleCommand = useCallback(
|
||||||
|
(item?: CommandItem) => {
|
||||||
|
setPrompt("");
|
||||||
|
|
||||||
|
if (!item) {
|
||||||
|
return handleGenerate();
|
||||||
|
}
|
||||||
|
if (item.id === "back") {
|
||||||
|
return setActiveCommandSet("main");
|
||||||
|
}
|
||||||
|
if (item.id === "result-replace") {
|
||||||
|
const chain = editor.chain().focus();
|
||||||
|
|
||||||
|
if (lastAction.action === AiAction.CONTINUE_WRITING) {
|
||||||
|
chain.setTextSelection(editor.state.selection.to);
|
||||||
|
}
|
||||||
|
|
||||||
|
const html = (marked.parse(output) as string).trim();
|
||||||
|
const isSingleParagraph =
|
||||||
|
html.startsWith("<p>") &&
|
||||||
|
html.endsWith("</p>") &&
|
||||||
|
html.lastIndexOf("<p>") === 0;
|
||||||
|
|
||||||
|
// Strip <p> wrapper for single-paragraph output to preserve inline context,
|
||||||
|
// then decode HTML entities via DOMParser since TipTap would otherwise
|
||||||
|
// treat the tagless string as plain text and insert entities literally.
|
||||||
|
const content = isSingleParagraph
|
||||||
|
? new DOMParser().parseFromString(html.slice(3, -4), "text/html")
|
||||||
|
.body.innerHTML
|
||||||
|
: html;
|
||||||
|
|
||||||
|
chain.insertContent(content).run();
|
||||||
|
|
||||||
|
return setShowAiMenu(false);
|
||||||
|
}
|
||||||
|
if (item.id === "result-insert-below") {
|
||||||
|
editor
|
||||||
|
.chain()
|
||||||
|
.focus()
|
||||||
|
.setTextSelection(editor.state.selection.to)
|
||||||
|
.insertContent(marked.parse(output))
|
||||||
|
.run();
|
||||||
|
|
||||||
|
return setShowAiMenu(false);
|
||||||
|
}
|
||||||
|
if (item.id === "result-copy") {
|
||||||
|
copyToClipboard(output);
|
||||||
|
|
||||||
|
return setShowAiMenu(false);
|
||||||
|
}
|
||||||
|
if (item.id === "result-discard") {
|
||||||
|
setOutput("");
|
||||||
|
|
||||||
|
return resetMenu();
|
||||||
|
}
|
||||||
|
if (item.id === "result-try-again" && lastAction) {
|
||||||
|
return handleGenerate(lastAction);
|
||||||
|
}
|
||||||
|
if (item.subCommandSet) {
|
||||||
|
return setActiveCommandSet(item.subCommandSet);
|
||||||
|
}
|
||||||
|
|
||||||
|
return handleGenerate(item);
|
||||||
|
},
|
||||||
|
[editor, output, lastAction, handleGenerate, resetMenu],
|
||||||
|
);
|
||||||
|
const handleKeyDown = useCallback(
|
||||||
|
(event: React.KeyboardEvent<HTMLInputElement>) => {
|
||||||
|
const totalItems = currentItems.length;
|
||||||
|
const cycleSize = totalItems + 1;
|
||||||
|
|
||||||
|
if (event.key === "Escape") {
|
||||||
|
return setShowAiMenu(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.key === "ArrowDown" || event.key === "ArrowUp") {
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
return setSelectedIndex((selectedIndex) => {
|
||||||
|
const direction = event.key === "ArrowDown" ? 1 : -1;
|
||||||
|
const newIndex = selectedIndex + direction;
|
||||||
|
|
||||||
|
if (newIndex < -1) return cycleSize - 1;
|
||||||
|
if (newIndex >= cycleSize) return 0;
|
||||||
|
|
||||||
|
return newIndex;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.key === "Enter") {
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
return handleCommand(currentItems[selectedIndex]);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[currentItems, selectedIndex],
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!editor) return;
|
||||||
|
|
||||||
|
const handleClose = () => setShowAiMenu(false);
|
||||||
|
const observer = new ResizeObserver(() => {
|
||||||
|
debouncedUpdateMenuPlacement();
|
||||||
|
});
|
||||||
|
|
||||||
|
updateMenuPlacement();
|
||||||
|
editor.on("focus", handleClose);
|
||||||
|
editor.on("blur", handleClose);
|
||||||
|
window.addEventListener("resize", debouncedUpdateMenuPlacement);
|
||||||
|
window.addEventListener("scroll", debouncedUpdateMenuPlacement, true);
|
||||||
|
observer.observe(editor.view.dom);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
editor.off("focus", handleClose);
|
||||||
|
editor.off("blur", handleClose);
|
||||||
|
window.removeEventListener("resize", debouncedUpdateMenuPlacement);
|
||||||
|
window.removeEventListener("scroll", debouncedUpdateMenuPlacement, true);
|
||||||
|
observer.disconnect();
|
||||||
|
};
|
||||||
|
}, [editor, updateMenuPlacement, debouncedUpdateMenuPlacement]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setShowAiMenu(false);
|
||||||
|
}, [location]);
|
||||||
|
useEffect(() => {
|
||||||
|
if (showAiMenu) {
|
||||||
|
resetMenu();
|
||||||
|
}
|
||||||
|
}, [showAiMenu, resetMenu]);
|
||||||
|
useEffect(() => {
|
||||||
|
// Focus input when menu opens or command set changes
|
||||||
|
requestAnimationFrame(() => {
|
||||||
|
inputRef.current?.focus({ preventScroll: true });
|
||||||
|
});
|
||||||
|
}, [showAiMenu, isLoading, currentItems]);
|
||||||
|
useEffect(() => {
|
||||||
|
if (!currentItems.length) {
|
||||||
|
setSelectedIndex(-1);
|
||||||
|
}
|
||||||
|
setSelectedIndex(prompt || activeCommandSet !== "main" ? 0 : -1);
|
||||||
|
}, [prompt, activeCommandSet, currentItems]);
|
||||||
|
|
||||||
|
if (!showAiMenu) return null;
|
||||||
|
|
||||||
|
return createPortal(
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
zIndex: 199,
|
||||||
|
position: "absolute",
|
||||||
|
top: menuPlacement.top,
|
||||||
|
left: menuPlacement.left,
|
||||||
|
width: menuPlacement.width,
|
||||||
|
pointerEvents: "none",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className={classes.aiMenu}
|
||||||
|
style={{ pointerEvents: "auto" }}
|
||||||
|
tabIndex={0}
|
||||||
|
ref={containerRef}
|
||||||
|
>
|
||||||
|
<ResultPreview output={output} isLoading={isLoading} />
|
||||||
|
<CommandSelector
|
||||||
|
selectedIndex={selectedIndex}
|
||||||
|
isLoading={isLoading}
|
||||||
|
output={output}
|
||||||
|
currentItems={currentItems}
|
||||||
|
handleCommand={handleCommand}
|
||||||
|
>
|
||||||
|
<TextInput
|
||||||
|
ref={inputRef}
|
||||||
|
className={classes.aiInput}
|
||||||
|
placeholder="Ask AI..."
|
||||||
|
data-autofocus
|
||||||
|
value={prompt}
|
||||||
|
disabled={isLoading}
|
||||||
|
onChange={(e) => setPrompt(e.currentTarget.value)}
|
||||||
|
rightSection={
|
||||||
|
<ActionIcon
|
||||||
|
disabled={!prompt || isLoading}
|
||||||
|
variant="filled"
|
||||||
|
color="blue"
|
||||||
|
radius="xl"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => handleGenerate()}
|
||||||
|
>
|
||||||
|
<IconArrowUp size={14} stroke={2.5} />
|
||||||
|
</ActionIcon>
|
||||||
|
}
|
||||||
|
onKeyDown={handleKeyDown}
|
||||||
|
/>
|
||||||
|
</CommandSelector>
|
||||||
|
</div>
|
||||||
|
</div>,
|
||||||
|
document.body,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export { EditorAiMenu };
|
||||||
@@ -0,0 +1,219 @@
|
|||||||
|
import { AiAction } from "@/ee/ai/types/ai.types.ts";
|
||||||
|
import {
|
||||||
|
IconSparkles,
|
||||||
|
IconArrowsMaximize,
|
||||||
|
IconArrowsMinimize,
|
||||||
|
IconWriting,
|
||||||
|
IconHelp,
|
||||||
|
IconList,
|
||||||
|
IconMoodSmile,
|
||||||
|
IconLanguage,
|
||||||
|
IconTrash,
|
||||||
|
IconRefresh,
|
||||||
|
IconChevronLeft,
|
||||||
|
IconCheck,
|
||||||
|
IconArrowDownLeft,
|
||||||
|
IconCopy,
|
||||||
|
IconTextPlus,
|
||||||
|
IconAlignJustified,
|
||||||
|
} from "@tabler/icons-react";
|
||||||
|
|
||||||
|
interface CommandItem {
|
||||||
|
name: string;
|
||||||
|
id: string;
|
||||||
|
icon?: typeof IconSparkles;
|
||||||
|
action?: AiAction;
|
||||||
|
prompt?: string;
|
||||||
|
subCommandSet?: CommandSet;
|
||||||
|
}
|
||||||
|
|
||||||
|
type CommandSet = "main" | "tone" | "translate" | "result";
|
||||||
|
|
||||||
|
const mainItems: CommandItem[] = [
|
||||||
|
{
|
||||||
|
id: "improve-writing",
|
||||||
|
name: "Improve writing",
|
||||||
|
icon: IconSparkles,
|
||||||
|
action: AiAction.IMPROVE_WRITING,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "fix-spelling-grammar",
|
||||||
|
name: "Fix spelling & grammar",
|
||||||
|
icon: IconCheck,
|
||||||
|
action: AiAction.FIX_SPELLING_GRAMMAR,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "make-longer",
|
||||||
|
name: "Make longer",
|
||||||
|
icon: IconTextPlus,
|
||||||
|
action: AiAction.MAKE_LONGER,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "make-shorter",
|
||||||
|
name: "Make shorter",
|
||||||
|
icon: IconAlignJustified,
|
||||||
|
action: AiAction.MAKE_SHORTER,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "continue-writing",
|
||||||
|
name: "Continue writing",
|
||||||
|
icon: IconWriting,
|
||||||
|
action: AiAction.CONTINUE_WRITING,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "explain",
|
||||||
|
name: "Explain",
|
||||||
|
icon: IconHelp,
|
||||||
|
action: AiAction.EXPLAIN,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "summarize",
|
||||||
|
name: "Summarize",
|
||||||
|
icon: IconList,
|
||||||
|
action: AiAction.SUMMARIZE,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "change-tone",
|
||||||
|
name: "Change tone",
|
||||||
|
icon: IconMoodSmile,
|
||||||
|
subCommandSet: "tone",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate",
|
||||||
|
name: "Translate",
|
||||||
|
icon: IconLanguage,
|
||||||
|
subCommandSet: "translate",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const toneItems: CommandItem[] = [
|
||||||
|
{
|
||||||
|
id: "back",
|
||||||
|
name: "Back",
|
||||||
|
icon: IconChevronLeft,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "tone-professional",
|
||||||
|
name: "Professional",
|
||||||
|
icon: IconMoodSmile,
|
||||||
|
action: AiAction.CHANGE_TONE,
|
||||||
|
prompt: "Professional",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "tone-casual",
|
||||||
|
name: "Casual",
|
||||||
|
icon: IconMoodSmile,
|
||||||
|
action: AiAction.CHANGE_TONE,
|
||||||
|
prompt: "Casual",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "tone-friendly",
|
||||||
|
name: "Friendly",
|
||||||
|
icon: IconMoodSmile,
|
||||||
|
action: AiAction.CHANGE_TONE,
|
||||||
|
prompt: "Friendly",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const translateItems: CommandItem[] = [
|
||||||
|
{
|
||||||
|
id: "back",
|
||||||
|
name: "Back",
|
||||||
|
icon: IconChevronLeft,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate-english",
|
||||||
|
name: "English",
|
||||||
|
icon: IconLanguage,
|
||||||
|
action: AiAction.TRANSLATE,
|
||||||
|
prompt: "English",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate-spanish",
|
||||||
|
name: "Spanish",
|
||||||
|
icon: IconLanguage,
|
||||||
|
action: AiAction.TRANSLATE,
|
||||||
|
prompt: "Spanish",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate-german",
|
||||||
|
name: "German",
|
||||||
|
icon: IconLanguage,
|
||||||
|
action: AiAction.TRANSLATE,
|
||||||
|
prompt: "German",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate-french",
|
||||||
|
name: "French",
|
||||||
|
icon: IconLanguage,
|
||||||
|
action: AiAction.TRANSLATE,
|
||||||
|
prompt: "French",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate-dutch",
|
||||||
|
name: "Dutch",
|
||||||
|
icon: IconLanguage,
|
||||||
|
action: AiAction.TRANSLATE,
|
||||||
|
prompt: "Dutch",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate-portuguese",
|
||||||
|
name: "Portuguese",
|
||||||
|
icon: IconLanguage,
|
||||||
|
action: AiAction.TRANSLATE,
|
||||||
|
prompt: "Portuguese",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate-italian",
|
||||||
|
name: "Italian",
|
||||||
|
icon: IconLanguage,
|
||||||
|
action: AiAction.TRANSLATE,
|
||||||
|
prompt: "Italian",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate-japanese",
|
||||||
|
name: "Japanese",
|
||||||
|
icon: IconLanguage,
|
||||||
|
action: AiAction.TRANSLATE,
|
||||||
|
prompt: "Japanese",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate-korean",
|
||||||
|
name: "Korean",
|
||||||
|
icon: IconLanguage,
|
||||||
|
action: AiAction.TRANSLATE,
|
||||||
|
prompt: "Korean",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate-swedish",
|
||||||
|
name: "Swedish",
|
||||||
|
icon: IconLanguage,
|
||||||
|
action: AiAction.TRANSLATE,
|
||||||
|
prompt: "Swedish",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "translate-chinese",
|
||||||
|
name: "Chinese (Simplified)",
|
||||||
|
icon: IconLanguage,
|
||||||
|
action: AiAction.TRANSLATE,
|
||||||
|
prompt: "Simplified Chinese",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const resultItems: CommandItem[] = [
|
||||||
|
{ id: "result-replace", name: "Replace", icon: IconCheck },
|
||||||
|
{ id: "result-insert-below", name: "Insert below", icon: IconArrowDownLeft },
|
||||||
|
{ id: "result-copy", name: "Copy", icon: IconCopy },
|
||||||
|
{ id: "result-discard", name: "Discard", icon: IconTrash },
|
||||||
|
{
|
||||||
|
id: "result-try-again",
|
||||||
|
name: "Try again",
|
||||||
|
icon: IconRefresh,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const commandItems: Record<CommandSet, CommandItem[]> = {
|
||||||
|
main: mainItems,
|
||||||
|
tone: toneItems,
|
||||||
|
translate: translateItems,
|
||||||
|
result: resultItems,
|
||||||
|
};
|
||||||
|
|
||||||
|
export type { CommandItem, CommandSet };
|
||||||
|
export { commandItems };
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
import { Loader, Menu, ScrollArea } from "@mantine/core";
|
||||||
|
import { IconChevronRight } from "@tabler/icons-react";
|
||||||
|
import { ReactNode } from "react";
|
||||||
|
import { CommandItem } from "./command-items.ts";
|
||||||
|
import classes from "./ai-menu.module.css";
|
||||||
|
|
||||||
|
interface CommandSelectorProps {
|
||||||
|
selectedIndex: number;
|
||||||
|
|
||||||
|
isLoading: boolean;
|
||||||
|
output: string;
|
||||||
|
currentItems: CommandItem[];
|
||||||
|
children: ReactNode;
|
||||||
|
handleCommand(item: CommandItem): void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const CommandSelector = ({
|
||||||
|
selectedIndex,
|
||||||
|
children,
|
||||||
|
isLoading,
|
||||||
|
output,
|
||||||
|
currentItems,
|
||||||
|
handleCommand,
|
||||||
|
}: CommandSelectorProps) => {
|
||||||
|
return (
|
||||||
|
<Menu
|
||||||
|
opened={!isLoading && currentItems.length > 0}
|
||||||
|
middlewares={{ flip: false }}
|
||||||
|
position="bottom-start"
|
||||||
|
offset={4}
|
||||||
|
width={250}
|
||||||
|
trapFocus={false}
|
||||||
|
shadow="lg"
|
||||||
|
>
|
||||||
|
<Menu.Target>{children}</Menu.Target>
|
||||||
|
<Menu.Dropdown>
|
||||||
|
<ScrollArea.Autosize type="scroll" scrollbarSize={5} mah={300}>
|
||||||
|
{currentItems.map((item, index) => {
|
||||||
|
const isSelected = selectedIndex === index;
|
||||||
|
const showLoader =
|
||||||
|
isLoading && output === "" && !item.subCommandSet;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Menu.Item
|
||||||
|
key={item.id}
|
||||||
|
className={isSelected ? classes.menuItemSelected : undefined}
|
||||||
|
leftSection={
|
||||||
|
showLoader ? (
|
||||||
|
<Loader size={14} />
|
||||||
|
) : item.icon ? (
|
||||||
|
<item.icon size={16} />
|
||||||
|
) : undefined
|
||||||
|
}
|
||||||
|
rightSection={
|
||||||
|
item.subCommandSet ? (
|
||||||
|
<IconChevronRight size={14} />
|
||||||
|
) : undefined
|
||||||
|
}
|
||||||
|
onClick={() => handleCommand(item)}
|
||||||
|
disabled={isLoading}
|
||||||
|
>
|
||||||
|
{item.name}
|
||||||
|
</Menu.Item>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</ScrollArea.Autosize>
|
||||||
|
</Menu.Dropdown>
|
||||||
|
</Menu>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export { CommandSelector };
|
||||||
@@ -0,0 +1,32 @@
|
|||||||
|
import { Loader, Paper, ScrollArea } from "@mantine/core";
|
||||||
|
import DOMPurify from "dompurify";
|
||||||
|
import { marked } from "marked";
|
||||||
|
import { memo } from "react";
|
||||||
|
import classes from "./ai-menu.module.css";
|
||||||
|
|
||||||
|
interface ResultPreviewProps {
|
||||||
|
output: string;
|
||||||
|
isLoading: boolean;
|
||||||
|
}
|
||||||
|
const ResultPreview = memo(({ output, isLoading }: ResultPreviewProps) => {
|
||||||
|
if (!output && !isLoading) return;
|
||||||
|
|
||||||
|
const parsedOutput = `${marked.parse(output)}`;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Paper mb={4} shadow="lg" radius="md" className={classes.resultPreview}>
|
||||||
|
<ScrollArea.Autosize mah={300} type="scroll" scrollbarSize={5}>
|
||||||
|
<div className={classes.resultPreviewWrapper}>
|
||||||
|
{parsedOutput && (
|
||||||
|
<div
|
||||||
|
dangerouslySetInnerHTML={{ __html: DOMPurify.sanitize(parsedOutput) }}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{isLoading && <Loader size={12} ml="xs" display="inline-block" />}
|
||||||
|
</div>
|
||||||
|
</ScrollArea.Autosize>
|
||||||
|
</Paper>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
export { ResultPreview };
|
||||||
@@ -1,12 +1,13 @@
|
|||||||
import { Group, Text, Switch, MantineSize, Title } from "@mantine/core";
|
import { Group, Text, Switch, MantineSize, Tooltip } from "@mantine/core";
|
||||||
import { useAtom } from "jotai";
|
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 React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import { isCloud } from "@/lib/config.ts";
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
import useLicense from "@/ee/hooks/use-license.tsx";
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
|
|
||||||
export default function EnableAiSearch() {
|
export default function EnableAiSearch() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -15,7 +16,7 @@ export default function EnableAiSearch() {
|
|||||||
<>
|
<>
|
||||||
<Group justify="space-between" wrap="nowrap" gap="xl">
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
||||||
<div>
|
<div>
|
||||||
<Text size="md">{t("AI-powered search (Ask AI)")}</Text>
|
<Text size="md">{t("AI-powered search (AI Answers)")}</Text>
|
||||||
<Text size="sm" c="dimmed">
|
<Text size="sm" c="dimmed">
|
||||||
{t(
|
{t(
|
||||||
"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.",
|
||||||
@@ -37,9 +38,8 @@ export function AiSearchToggle({ size, label }: AiSearchToggleProps) {
|
|||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
const [checked, setChecked] = useState(workspace?.settings?.ai?.search);
|
const [checked, setChecked] = useState(workspace?.settings?.ai?.search);
|
||||||
const { hasLicenseKey } = useLicense();
|
const hasAccess = useHasFeature(Feature.AI);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
const hasAccess = isCloud() || (!isCloud() && hasLicenseKey);
|
|
||||||
|
|
||||||
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
const value = event.currentTarget.checked;
|
const value = event.currentTarget.checked;
|
||||||
@@ -56,14 +56,16 @@ export function AiSearchToggle({ size, label }: AiSearchToggleProps) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Switch
|
<Tooltip label={upgradeLabel} disabled={hasAccess} refProp="rootRef">
|
||||||
size={size}
|
<Switch
|
||||||
label={label}
|
size={size}
|
||||||
labelPosition="left"
|
label={label}
|
||||||
defaultChecked={checked}
|
labelPosition="left"
|
||||||
onChange={handleChange}
|
defaultChecked={checked}
|
||||||
disabled={!hasAccess}
|
onChange={handleChange}
|
||||||
aria-label={t("Toggle AI search")}
|
disabled={!hasAccess}
|
||||||
/>
|
aria-label={t("Toggle AI search")}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,53 @@
|
|||||||
|
import { Group, Text, Switch, Tooltip } from "@mantine/core";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
import React, { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
|
|
||||||
|
export default function EnableGenerativeAi() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
|
const [checked, setChecked] = useState(workspace?.settings?.ai?.generative);
|
||||||
|
const hasAccess = useHasFeature(Feature.AI);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = event.currentTarget.checked;
|
||||||
|
try {
|
||||||
|
const updatedWorkspace = await updateWorkspace({ generativeAi: value });
|
||||||
|
setChecked(value);
|
||||||
|
setWorkspace(updatedWorkspace);
|
||||||
|
} catch (err) {
|
||||||
|
notifications.show({
|
||||||
|
message: err?.response?.data?.message,
|
||||||
|
color: "red",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
||||||
|
<div>
|
||||||
|
<Text size="md">{t("Generative AI (Ask AI)")}</Text>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t(
|
||||||
|
"Enable AI-powered content generation in the editor. Allows users to generate, improve, translate and transform text.",
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Tooltip label={upgradeLabel} disabled={hasAccess} refProp="rootRef">
|
||||||
|
<Switch
|
||||||
|
defaultChecked={checked}
|
||||||
|
onChange={handleChange}
|
||||||
|
disabled={!hasAccess}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,156 @@
|
|||||||
|
import {
|
||||||
|
Anchor,
|
||||||
|
Group,
|
||||||
|
List,
|
||||||
|
Text,
|
||||||
|
Switch,
|
||||||
|
TextInput,
|
||||||
|
ActionIcon,
|
||||||
|
Tooltip,
|
||||||
|
Stack,
|
||||||
|
Alert,
|
||||||
|
} from "@mantine/core";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
import React, { useState } from "react";
|
||||||
|
import { Trans, useTranslation } from "react-i18next";
|
||||||
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
|
import { getAppUrl } from "@/lib/config.ts";
|
||||||
|
import { IconCheck, IconCopy, IconInfoCircle } from "@tabler/icons-react";
|
||||||
|
import { CopyButton } from "@/components/common/copy-button.tsx";
|
||||||
|
|
||||||
|
export default function McpSettings() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
|
const [checked, setChecked] = useState(workspace?.settings?.ai?.mcp);
|
||||||
|
const hasAccess = useHasFeature(Feature.MCP);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
|
const mcpUrl = `${getAppUrl()}/mcp`;
|
||||||
|
|
||||||
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = event.currentTarget.checked;
|
||||||
|
try {
|
||||||
|
const updatedWorkspace = await updateWorkspace({ mcpEnabled: value });
|
||||||
|
setChecked(value);
|
||||||
|
setWorkspace(updatedWorkspace);
|
||||||
|
} catch (err) {
|
||||||
|
notifications.show({
|
||||||
|
message: err?.response?.data?.message,
|
||||||
|
color: "red",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack gap="lg">
|
||||||
|
{!hasAccess && (
|
||||||
|
<Alert icon={<IconInfoCircle />} title={upgradeLabel} color="blue">
|
||||||
|
{t(
|
||||||
|
"MCP is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
||||||
|
)}
|
||||||
|
</Alert>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
||||||
|
<div>
|
||||||
|
<Text size="md">{t("Model Context Protocol (MCP)")}</Text>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t(
|
||||||
|
"Enable the MCP server to allow AI assistants and tools to interact with your workspace content.",
|
||||||
|
)}{" "}
|
||||||
|
<Trans
|
||||||
|
i18nKey="View the <anchor>MCP documentation</anchor>."
|
||||||
|
components={{
|
||||||
|
anchor: <Anchor href="https://docmost.com/docs/user-guide/mcp" target="_blank" size="sm" />,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Tooltip label={upgradeLabel} disabled={hasAccess} refProp="rootRef">
|
||||||
|
<Switch
|
||||||
|
defaultChecked={checked}
|
||||||
|
onChange={handleChange}
|
||||||
|
disabled={!hasAccess}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
</Group>
|
||||||
|
|
||||||
|
{checked && (
|
||||||
|
<div>
|
||||||
|
<Text size="sm" fw={500} mb={4}>
|
||||||
|
{t("MCP Server URL")}
|
||||||
|
</Text>
|
||||||
|
<Group gap="xs">
|
||||||
|
<TextInput value={mcpUrl} readOnly style={{ flex: 1 }} />
|
||||||
|
<CopyButton value={mcpUrl} timeout={2000}>
|
||||||
|
{({ copied, copy }) => (
|
||||||
|
<Tooltip
|
||||||
|
label={copied ? t("Copied") : t("Copy")}
|
||||||
|
withArrow
|
||||||
|
position="right"
|
||||||
|
>
|
||||||
|
<ActionIcon
|
||||||
|
color={copied ? "teal" : "gray"}
|
||||||
|
variant="subtle"
|
||||||
|
onClick={copy}
|
||||||
|
>
|
||||||
|
{copied ? <IconCheck size={16} /> : <IconCopy size={16} />}
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
)}
|
||||||
|
</CopyButton>
|
||||||
|
</Group>
|
||||||
|
<Text size="sm" c="dimmed" mt="xs">
|
||||||
|
{t(
|
||||||
|
"Use your API key for authentication. You can manage API keys in your account settings.",
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<Text size="sm" fw={500} mt="md" mb={4}>
|
||||||
|
{t("Supported tools")}
|
||||||
|
</Text>
|
||||||
|
<List size="sm" spacing={2}>
|
||||||
|
<List.Item>
|
||||||
|
<Text size="sm" c="dimmed" span>
|
||||||
|
search_pages, get_page, create_page, update_page
|
||||||
|
</Text>
|
||||||
|
</List.Item>
|
||||||
|
<List.Item>
|
||||||
|
<Text size="sm" c="dimmed" span>
|
||||||
|
list_pages, list_child_pages, duplicate_page
|
||||||
|
</Text>
|
||||||
|
</List.Item>
|
||||||
|
<List.Item>
|
||||||
|
<Text size="sm" c="dimmed" span>
|
||||||
|
copy_page_to_space, move_page, move_page_to_space
|
||||||
|
</Text>
|
||||||
|
</List.Item>
|
||||||
|
<List.Item>
|
||||||
|
<Text size="sm" c="dimmed" span>
|
||||||
|
get_space, list_spaces, create_space, update_space
|
||||||
|
</Text>
|
||||||
|
</List.Item>
|
||||||
|
<List.Item>
|
||||||
|
<Text size="sm" c="dimmed" span>
|
||||||
|
get_comments, create_comment, update_comment
|
||||||
|
</Text>
|
||||||
|
</List.Item>
|
||||||
|
<List.Item>
|
||||||
|
<Text size="sm" c="dimmed" span>
|
||||||
|
search_attachments, list_workspace_members, get_current_user
|
||||||
|
</Text>
|
||||||
|
</List.Item>
|
||||||
|
</List>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import { useMutation, UseMutationResult } from "@tanstack/react-query";
|
import { useMutation, UseMutationResult } from "@tanstack/react-query";
|
||||||
import { useState, useCallback } from "react";
|
import { useState, useCallback } from "react";
|
||||||
import { askAi, IAiSearchResponse } from "@/ee/ai/services/ai-search-service.ts";
|
import { aiAnswers, IAiSearchResponse } from "@/ee/ai/services/ai-search-service.ts";
|
||||||
import { IPageSearchParams } from "@/features/search/types/search.types.ts";
|
import { IPageSearchParams } from "@/features/search/types/search.types.ts";
|
||||||
|
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
@@ -26,7 +26,7 @@ export function useAiSearch(): UseAiSearchResult {
|
|||||||
|
|
||||||
const { contentType, ...apiParams } = params;
|
const { contentType, ...apiParams } = params;
|
||||||
|
|
||||||
return await askAi(apiParams, (chunk) => {
|
return await aiAnswers(apiParams, (chunk) => {
|
||||||
if (chunk.content) {
|
if (chunk.content) {
|
||||||
setStreamingAnswer((prev) => prev + chunk.content);
|
setStreamingAnswer((prev) => prev + chunk.content);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,46 +1,85 @@
|
|||||||
import { Helmet } from "react-helmet-async";
|
import { Helmet } from "react-helmet-async";
|
||||||
import { getAppName, isCloud } from "@/lib/config.ts";
|
import { getAppName } from "@/lib/config.ts";
|
||||||
import SettingsTitle from "@/components/settings/settings-title.tsx";
|
import SettingsTitle from "@/components/settings/settings-title.tsx";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import useUserRole from "@/hooks/use-user-role.tsx";
|
import useUserRole from "@/hooks/use-user-role.tsx";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import useLicense from "@/ee/hooks/use-license.tsx";
|
|
||||||
import EnableAiSearch from "@/ee/ai/components/enable-ai-search.tsx";
|
import EnableAiSearch from "@/ee/ai/components/enable-ai-search.tsx";
|
||||||
import { Alert } from "@mantine/core";
|
import EnableGenerativeAi from "@/ee/ai/components/enable-generative-ai.tsx";
|
||||||
|
import EnableAiChat from "@/ee/ai-chat/components/enable-ai-chat.tsx";
|
||||||
|
import McpSettings from "@/ee/ai/components/mcp-settings.tsx";
|
||||||
|
import { Alert, Stack, Tabs } from "@mantine/core";
|
||||||
import { IconInfoCircle } from "@tabler/icons-react";
|
import { IconInfoCircle } from "@tabler/icons-react";
|
||||||
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label";
|
||||||
|
import { isCloud } from "@/lib/config.ts";
|
||||||
|
import { useLocation, useNavigate } from "react-router-dom";
|
||||||
|
|
||||||
export default function AiSettings() {
|
export default function AiSettings() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { isAdmin } = useUserRole();
|
const { isAdmin } = useUserRole();
|
||||||
const { hasLicenseKey } = useLicense();
|
const hasAccess = useHasFeature(Feature.AI);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
const location = useLocation();
|
||||||
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
const activeTab = location.pathname.endsWith("/mcp") ? "mcp" : "ai";
|
||||||
|
|
||||||
if (!isAdmin) {
|
if (!isAdmin) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const hasAccess = isCloud() || (!isCloud() && hasLicenseKey);
|
const handleTabChange = (value: string | null) => {
|
||||||
|
if (value === "mcp") {
|
||||||
|
navigate("/settings/ai/mcp");
|
||||||
|
} else {
|
||||||
|
navigate("/settings/ai");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Helmet>
|
<Helmet>
|
||||||
<title>AI - {getAppName()}</title>
|
<title>AI settings - {getAppName()}</title>
|
||||||
</Helmet>
|
</Helmet>
|
||||||
<SettingsTitle title={t("AI settings")} />
|
<SettingsTitle title={t("AI settings")} />
|
||||||
|
|
||||||
{!hasAccess && (
|
<Tabs color="dark" value={activeTab} onChange={handleTabChange}>
|
||||||
<Alert
|
<Tabs.List>
|
||||||
icon={<IconInfoCircle />}
|
<Tabs.Tab fw={500} value="ai">
|
||||||
title={t("Enterprise feature")}
|
{t("AI")}
|
||||||
color="blue"
|
</Tabs.Tab>
|
||||||
mb="lg"
|
<Tabs.Tab fw={500} value="mcp">
|
||||||
>
|
{t("MCP")}
|
||||||
{t(
|
</Tabs.Tab>
|
||||||
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
</Tabs.List>
|
||||||
)}
|
|
||||||
</Alert>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<EnableAiSearch />
|
<Tabs.Panel value="ai" pt="md">
|
||||||
|
{!hasAccess && (
|
||||||
|
<Alert
|
||||||
|
icon={<IconInfoCircle />}
|
||||||
|
title={upgradeLabel}
|
||||||
|
color="blue"
|
||||||
|
mb="lg"
|
||||||
|
>
|
||||||
|
{t(
|
||||||
|
"AI is only available in the Docmost enterprise edition. Contact sales@docmost.com.",
|
||||||
|
)}
|
||||||
|
</Alert>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<Stack gap="md">
|
||||||
|
{!isCloud() && <EnableAiSearch />}
|
||||||
|
<EnableGenerativeAi />
|
||||||
|
<EnableAiChat />
|
||||||
|
</Stack>
|
||||||
|
</Tabs.Panel>
|
||||||
|
|
||||||
|
<Tabs.Panel value="mcp" pt="md">
|
||||||
|
<McpSettings />
|
||||||
|
</Tabs.Panel>
|
||||||
|
</Tabs>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,11 +15,11 @@ export interface IAiSearchResponse {
|
|||||||
}>;
|
}>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function askAi(
|
export async function aiAnswers(
|
||||||
params: IPageSearchParams,
|
params: IPageSearchParams,
|
||||||
onChunk?: (chunk: { content?: string; sources?: any[] }) => void,
|
onChunk?: (chunk: { content?: string; sources?: any[] }) => void,
|
||||||
): Promise<IAiSearchResponse> {
|
): Promise<IAiSearchResponse> {
|
||||||
const response = await fetch("/api/ai/ask", {
|
const response = await fetch("/api/ai/answers", {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
|
|||||||
@@ -43,13 +43,16 @@ export async function generateAiContentStream(
|
|||||||
}
|
}
|
||||||
|
|
||||||
const processStream = async () => {
|
const processStream = async () => {
|
||||||
|
let buffer = "";
|
||||||
try {
|
try {
|
||||||
while (true) {
|
while (true) {
|
||||||
const { done, value } = await reader.read();
|
const { done, value } = await reader.read();
|
||||||
if (done) break;
|
if (done) break;
|
||||||
|
|
||||||
const chunk = decoder.decode(value, { stream: true });
|
buffer += decoder.decode(value, { stream: true });
|
||||||
const lines = chunk.split("\n");
|
const lines = buffer.split("\n");
|
||||||
|
|
||||||
|
buffer = lines.pop() || "";
|
||||||
|
|
||||||
for (const line of lines) {
|
for (const line of lines) {
|
||||||
if (line.startsWith("data: ")) {
|
if (line.startsWith("data: ")) {
|
||||||
@@ -66,7 +69,7 @@ export async function generateAiContentStream(
|
|||||||
onChunk(parsed);
|
onChunk(parsed);
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
// Ignore parse errors for incomplete chunks
|
// Skip invalid JSON
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ export enum AiAction {
|
|||||||
SIMPLIFY = "simplify",
|
SIMPLIFY = "simplify",
|
||||||
CHANGE_TONE = "change_tone",
|
CHANGE_TONE = "change_tone",
|
||||||
SUMMARIZE = "summarize",
|
SUMMARIZE = "summarize",
|
||||||
|
EXPLAIN = "explain",
|
||||||
CONTINUE_WRITING = "continue_writing",
|
CONTINUE_WRITING = "continue_writing",
|
||||||
TRANSLATE = "translate",
|
TRANSLATE = "translate",
|
||||||
CUSTOM = "custom",
|
CUSTOM = "custom",
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import { lazy, Suspense, useState } from "react";
|
import { lazy, Suspense, useState } from "react";
|
||||||
import { Modal, TextInput, Button, Group, Stack, Select } from "@mantine/core";
|
import { Modal, TextInput, Button, Group, Stack, Select } from "@mantine/core";
|
||||||
import { useForm } from "@mantine/form";
|
import { useForm } from "@mantine/form";
|
||||||
import { zodResolver } from "mantine-form-zod-resolver";
|
import { zod4Resolver } from "mantine-form-zod-resolver";
|
||||||
import { z } from "zod";
|
import { z } from "zod/v4";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { useCreateApiKeyMutation } from "@/ee/api-key/queries/api-key-query";
|
import { useCreateApiKeyMutation } from "@/ee/api-key/queries/api-key-query";
|
||||||
import { IconCalendar } from "@tabler/icons-react";
|
import { IconCalendar } from "@tabler/icons-react";
|
||||||
@@ -36,7 +36,7 @@ export function CreateApiKeyModal({
|
|||||||
const createApiKeyMutation = useCreateApiKeyMutation();
|
const createApiKeyMutation = useCreateApiKeyMutation();
|
||||||
|
|
||||||
const form = useForm<FormValues>({
|
const form = useForm<FormValues>({
|
||||||
validate: zodResolver(formSchema),
|
validate: zod4Resolver(formSchema),
|
||||||
initialValues: {
|
initialValues: {
|
||||||
name: "",
|
name: "",
|
||||||
expiresAt: "",
|
expiresAt: "",
|
||||||
|
|||||||
@@ -0,0 +1,71 @@
|
|||||||
|
import { Text, Switch, Tooltip } from "@mantine/core";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
import React, { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useHasFeature } from "@/ee/hooks/use-feature";
|
||||||
|
import { Feature } from "@/ee/features";
|
||||||
|
import {
|
||||||
|
ResponsiveSettingsRow,
|
||||||
|
ResponsiveSettingsContent,
|
||||||
|
ResponsiveSettingsControl,
|
||||||
|
} from "@/components/ui/responsive-settings-row";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label.ts";
|
||||||
|
|
||||||
|
export default function RestrictApiToAdmins() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [workspace, setWorkspace] = useAtom(workspaceAtom);
|
||||||
|
const [checked, setChecked] = useState(
|
||||||
|
workspace?.settings?.api?.restrictToAdmins === true,
|
||||||
|
);
|
||||||
|
const hasAccess = useHasFeature(Feature.API_KEYS);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
|
||||||
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = event.currentTarget.checked;
|
||||||
|
try {
|
||||||
|
const updatedWorkspace = await updateWorkspace({
|
||||||
|
restrictApiToAdmins: value,
|
||||||
|
});
|
||||||
|
setChecked(value);
|
||||||
|
setWorkspace(updatedWorkspace);
|
||||||
|
} catch (err) {
|
||||||
|
notifications.show({
|
||||||
|
message: err?.response?.data?.message,
|
||||||
|
color: "red",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ResponsiveSettingsRow>
|
||||||
|
<ResponsiveSettingsContent>
|
||||||
|
<Text size="md">
|
||||||
|
{t("Restrict API key creation to admins")}
|
||||||
|
</Text>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t(
|
||||||
|
"Only admins and owners can create new API keys. Existing member keys will continue to work.",
|
||||||
|
)}
|
||||||
|
</Text>
|
||||||
|
</ResponsiveSettingsContent>
|
||||||
|
|
||||||
|
<ResponsiveSettingsControl>
|
||||||
|
<Tooltip
|
||||||
|
label={upgradeLabel}
|
||||||
|
disabled={hasAccess}
|
||||||
|
refProp="rootRef"
|
||||||
|
>
|
||||||
|
<Switch
|
||||||
|
checked={checked}
|
||||||
|
onChange={handleChange}
|
||||||
|
disabled={!hasAccess}
|
||||||
|
aria-label={t("Toggle restrict API keys to admins")}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
</ResponsiveSettingsControl>
|
||||||
|
</ResponsiveSettingsRow>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
import { Modal, TextInput, Button, Group, Stack } from "@mantine/core";
|
import { Modal, TextInput, Button, Group, Stack } from "@mantine/core";
|
||||||
import { useForm } from "@mantine/form";
|
import { useForm } from "@mantine/form";
|
||||||
import { zodResolver } from "mantine-form-zod-resolver";
|
import { zod4Resolver } from "mantine-form-zod-resolver";
|
||||||
import { z } from "zod";
|
import { z } from "zod/v4";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { useUpdateApiKeyMutation } from "@/ee/api-key/queries/api-key-query";
|
import { useUpdateApiKeyMutation } from "@/ee/api-key/queries/api-key-query";
|
||||||
import { IApiKey } from "@/ee/api-key";
|
import { IApiKey } from "@/ee/api-key";
|
||||||
@@ -27,7 +27,7 @@ export function UpdateApiKeyModal({
|
|||||||
const updateApiKeyMutation = useUpdateApiKeyMutation();
|
const updateApiKeyMutation = useUpdateApiKeyMutation();
|
||||||
|
|
||||||
const form = useForm<FormValues>({
|
const form = useForm<FormValues>({
|
||||||
validate: zodResolver(formSchema),
|
validate: zod4Resolver(formSchema),
|
||||||
initialValues: {
|
initialValues: {
|
||||||
name: "",
|
name: "",
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { Button, Group, Space } from "@mantine/core";
|
import { Anchor, Alert, Button, Group, Space, Text } from "@mantine/core";
|
||||||
|
import { IconInfoCircle } from "@tabler/icons-react";
|
||||||
import { Helmet } from "react-helmet-async";
|
import { Helmet } from "react-helmet-async";
|
||||||
import { useTranslation } from "react-i18next";
|
import { Trans, useTranslation } from "react-i18next";
|
||||||
import SettingsTitle from "@/components/settings/settings-title";
|
import SettingsTitle from "@/components/settings/settings-title";
|
||||||
import { getAppName } from "@/lib/config";
|
import { getAppName, getAppUrl } from "@/lib/config";
|
||||||
import { ApiKeyTable } from "@/ee/api-key/components/api-key-table";
|
import { ApiKeyTable } from "@/ee/api-key/components/api-key-table";
|
||||||
import { CreateApiKeyModal } from "@/ee/api-key/components/create-api-key-modal";
|
import { CreateApiKeyModal } from "@/ee/api-key/components/create-api-key-modal";
|
||||||
import { ApiKeyCreatedModal } from "@/ee/api-key/components/api-key-created-modal";
|
import { ApiKeyCreatedModal } from "@/ee/api-key/components/api-key-created-modal";
|
||||||
@@ -13,6 +14,9 @@ import Paginate from "@/components/common/paginate";
|
|||||||
import { useCursorPaginate } from "@/hooks/use-cursor-paginate";
|
import { useCursorPaginate } from "@/hooks/use-cursor-paginate";
|
||||||
import { useGetApiKeysQuery } from "@/ee/api-key/queries/api-key-query.ts";
|
import { useGetApiKeysQuery } from "@/ee/api-key/queries/api-key-query.ts";
|
||||||
import { IApiKey } from "@/ee/api-key";
|
import { IApiKey } from "@/ee/api-key";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import { workspaceAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
import useUserRole from "@/hooks/use-user-role.tsx";
|
||||||
|
|
||||||
export default function UserApiKeys() {
|
export default function UserApiKeys() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -23,6 +27,11 @@ export default function UserApiKeys() {
|
|||||||
const [revokeModalOpened, setRevokeModalOpened] = useState(false);
|
const [revokeModalOpened, setRevokeModalOpened] = useState(false);
|
||||||
const [selectedApiKey, setSelectedApiKey] = useState<IApiKey | null>(null);
|
const [selectedApiKey, setSelectedApiKey] = useState<IApiKey | null>(null);
|
||||||
const { data, isLoading } = useGetApiKeysQuery({ cursor });
|
const { data, isLoading } = useGetApiKeysQuery({ cursor });
|
||||||
|
const [workspace] = useAtom(workspaceAtom);
|
||||||
|
const { isAdmin } = useUserRole();
|
||||||
|
const mcpEnabled = workspace?.settings?.ai?.mcp === true;
|
||||||
|
const restrictToAdmins = workspace?.settings?.api?.restrictToAdmins === true;
|
||||||
|
const canCreate = !restrictToAdmins || isAdmin;
|
||||||
|
|
||||||
const handleCreateSuccess = (response: IApiKey) => {
|
const handleCreateSuccess = (response: IApiKey) => {
|
||||||
setCreatedApiKey(response);
|
setCreatedApiKey(response);
|
||||||
@@ -48,11 +57,51 @@ export default function UserApiKeys() {
|
|||||||
|
|
||||||
<SettingsTitle title={t("API keys")} />
|
<SettingsTitle title={t("API keys")} />
|
||||||
|
|
||||||
<Group justify="flex-end" mb="md">
|
<Text size="sm" c="dimmed" mb="md">
|
||||||
<Button onClick={() => setCreateModalOpened(true)}>
|
<Trans
|
||||||
{t("Create API Key")}
|
i18nKey="View the <anchor>API documentation</anchor> for usage details."
|
||||||
</Button>
|
components={{
|
||||||
</Group>
|
anchor: <Anchor href="https://docmost.com/api-docs" target="_blank" size="sm" />,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
{mcpEnabled && canCreate && (
|
||||||
|
<Alert variant="light" color="blue" mb="md" p="sm" icon={<IconInfoCircle />}>
|
||||||
|
<Text size="sm">
|
||||||
|
{t(
|
||||||
|
"Your workspace has MCP enabled. Use your API key to connect AI assistants.",
|
||||||
|
)}{" "}
|
||||||
|
<Anchor
|
||||||
|
href="https://docmost.com/docs/user-guide/mcp"
|
||||||
|
target="_blank"
|
||||||
|
size="sm"
|
||||||
|
>
|
||||||
|
{t("Learn more")}
|
||||||
|
</Anchor>
|
||||||
|
</Text>
|
||||||
|
<Text size="sm" mt={4}>
|
||||||
|
{t("MCP server URL:")}{" "}
|
||||||
|
<Text size="sm" fw={500} span ff="monospace">
|
||||||
|
{`${getAppUrl()}/mcp`}
|
||||||
|
</Text>
|
||||||
|
</Text>
|
||||||
|
</Alert>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{canCreate ? (
|
||||||
|
<Group justify="flex-end" mb="md">
|
||||||
|
<Button onClick={() => setCreateModalOpened(true)}>
|
||||||
|
{t("Create API Key")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
) : restrictToAdmins ? (
|
||||||
|
<Alert variant="light" color="yellow" mb="md" p="sm" icon={<IconInfoCircle />}>
|
||||||
|
<Text size="sm">
|
||||||
|
{t("API key creation is restricted to admins by your workspace administrator.")}
|
||||||
|
</Text>
|
||||||
|
</Alert>
|
||||||
|
) : null}
|
||||||
|
|
||||||
<ApiKeyTable
|
<ApiKeyTable
|
||||||
apiKeys={data?.items || []}
|
apiKeys={data?.items || []}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { Button, Group, Space, Text } from "@mantine/core";
|
import { Anchor, Button, Divider, Group, Space, Text } from "@mantine/core";
|
||||||
import { Helmet } from "react-helmet-async";
|
import { Helmet } from "react-helmet-async";
|
||||||
import { useTranslation } from "react-i18next";
|
import { Trans, useTranslation } from "react-i18next";
|
||||||
import SettingsTitle from "@/components/settings/settings-title";
|
import SettingsTitle from "@/components/settings/settings-title";
|
||||||
import { getAppName } from "@/lib/config";
|
import { getAppName } from "@/lib/config";
|
||||||
import { ApiKeyTable } from "@/ee/api-key/components/api-key-table";
|
import { ApiKeyTable } from "@/ee/api-key/components/api-key-table";
|
||||||
@@ -14,6 +14,7 @@ import { useCursorPaginate } from "@/hooks/use-cursor-paginate";
|
|||||||
import { useGetApiKeysQuery } from "@/ee/api-key/queries/api-key-query.ts";
|
import { useGetApiKeysQuery } from "@/ee/api-key/queries/api-key-query.ts";
|
||||||
import { IApiKey } from "@/ee/api-key";
|
import { IApiKey } from "@/ee/api-key";
|
||||||
import useUserRole from '@/hooks/use-user-role.tsx';
|
import useUserRole from '@/hooks/use-user-role.tsx';
|
||||||
|
import RestrictApiToAdmins from "@/ee/api-key/components/restrict-api-to-admins";
|
||||||
|
|
||||||
export default function WorkspaceApiKeys() {
|
export default function WorkspaceApiKeys() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -54,10 +55,18 @@ export default function WorkspaceApiKeys() {
|
|||||||
|
|
||||||
<SettingsTitle title={t("API management")} />
|
<SettingsTitle title={t("API management")} />
|
||||||
|
|
||||||
<Text size="md" c="dimmed" mb="md">
|
<Text size="sm" c="dimmed" mb="md">
|
||||||
{t("Manage API keys for all users in the workspace")}
|
<Trans
|
||||||
|
i18nKey="Manage API keys for all users in the workspace. View the <anchor>API documentation</anchor> for usage details."
|
||||||
|
components={{
|
||||||
|
anchor: <Anchor href="https://docmost.com/api-docs" target="_blank" size="sm" />,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
|
<RestrictApiToAdmins />
|
||||||
|
<Divider my="lg" />
|
||||||
|
|
||||||
<Group justify="flex-end" mb="md">
|
<Group justify="flex-end" mb="md">
|
||||||
<Button onClick={() => setCreateModalOpened(true)}>
|
<Button onClick={() => setCreateModalOpened(true)}>
|
||||||
{t("Create API Key")}
|
{t("Create API Key")}
|
||||||
|
|||||||
@@ -0,0 +1,333 @@
|
|||||||
|
import { Fragment, useState } from "react";
|
||||||
|
import {
|
||||||
|
Table,
|
||||||
|
Text,
|
||||||
|
Group,
|
||||||
|
Skeleton,
|
||||||
|
Anchor,
|
||||||
|
Collapse,
|
||||||
|
Box,
|
||||||
|
} from "@mantine/core";
|
||||||
|
import { Link } from "react-router-dom";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import {
|
||||||
|
IconChevronRight,
|
||||||
|
IconChevronDown,
|
||||||
|
IconArrowRight,
|
||||||
|
} from "@tabler/icons-react";
|
||||||
|
import { IAuditLog } from "@/ee/audit/types/audit.types";
|
||||||
|
import { CustomAvatar } from "@/components/ui/custom-avatar";
|
||||||
|
import { getEventLabel } from "@/ee/audit/lib/audit-event-labels";
|
||||||
|
import { formattedDate } from "@/lib/time";
|
||||||
|
import NoTableResults from "@/components/common/no-table-results";
|
||||||
|
import classes from "./audit-logs.module.css";
|
||||||
|
|
||||||
|
type AuditLogsTableProps = {
|
||||||
|
items?: IAuditLog[];
|
||||||
|
isLoading: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
function hasDetails(entry: IAuditLog): boolean {
|
||||||
|
return !!(entry.changes?.before || entry.changes?.after || entry.metadata);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getResourceUrl(entry: IAuditLog): string | null {
|
||||||
|
if (!entry.resource) return null;
|
||||||
|
|
||||||
|
switch (entry.resourceType) {
|
||||||
|
case "group":
|
||||||
|
return `/settings/groups/${entry.resource.id}`;
|
||||||
|
case "space":
|
||||||
|
case "space_member":
|
||||||
|
return entry.resource.slug ? `/s/${entry.resource.slug}` : null;
|
||||||
|
default:
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatValue(value: unknown): string {
|
||||||
|
if (value === null || value === undefined) return "—";
|
||||||
|
if (typeof value === "boolean") return value ? "true" : "false";
|
||||||
|
if (typeof value === "object") return JSON.stringify(value);
|
||||||
|
return String(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ChangesDiff({ changes }: { changes: IAuditLog["changes"] }) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
if (!changes) return null;
|
||||||
|
|
||||||
|
const { before, after } = changes;
|
||||||
|
const allKeys = new Set([
|
||||||
|
...Object.keys(before ?? {}),
|
||||||
|
...Object.keys(after ?? {}),
|
||||||
|
]);
|
||||||
|
|
||||||
|
if (allKeys.size === 0) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box>
|
||||||
|
<Text fz="xs" fw={600} mb={4}>
|
||||||
|
{t("Changes")}
|
||||||
|
</Text>
|
||||||
|
{[...allKeys].map((key) => {
|
||||||
|
const hasBefore = before && key in before;
|
||||||
|
const hasAfter = after && key in after;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group key={key} gap={6} mb={2} wrap="nowrap" align="center">
|
||||||
|
<Text
|
||||||
|
fz="xs"
|
||||||
|
c="dimmed"
|
||||||
|
fw={500}
|
||||||
|
style={{ minWidth: "fit-content" }}
|
||||||
|
>
|
||||||
|
{key}:
|
||||||
|
</Text>
|
||||||
|
{hasBefore && (
|
||||||
|
<Text fz="xs" component="span">
|
||||||
|
{formatValue(before[key])}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
{hasBefore && hasAfter && (
|
||||||
|
<IconArrowRight size={10} color="var(--mantine-color-dimmed)" />
|
||||||
|
)}
|
||||||
|
{hasAfter && (
|
||||||
|
<Text fz="xs" component="span">
|
||||||
|
{formatValue(after[key])}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function MetadataDisplay({ metadata }: { metadata: Record<string, any> }) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const entries = Object.entries(metadata);
|
||||||
|
if (entries.length === 0) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box>
|
||||||
|
<Text fz="xs" fw={600} mb={4}>
|
||||||
|
{t("Metadata")}
|
||||||
|
</Text>
|
||||||
|
{entries.map(([key, value]) => (
|
||||||
|
<Group key={key} gap={6} mb={2} wrap="nowrap">
|
||||||
|
<Text fz="xs" c="dimmed" fw={500}>
|
||||||
|
{key}:
|
||||||
|
</Text>
|
||||||
|
<Text fz="xs">{formatValue(value)}</Text>
|
||||||
|
</Group>
|
||||||
|
))}
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function TableSkeleton() {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{Array.from({ length: 8 }).map((_, i) => (
|
||||||
|
<Table.Tr key={i}>
|
||||||
|
<Table.Td>
|
||||||
|
<Group gap="sm" wrap="nowrap">
|
||||||
|
<Skeleton circle height={36} />
|
||||||
|
<div>
|
||||||
|
<Skeleton height={14} width={120} mb={4} />
|
||||||
|
<Skeleton height={10} width={160} />
|
||||||
|
</div>
|
||||||
|
</Group>
|
||||||
|
</Table.Td>
|
||||||
|
<Table.Td>
|
||||||
|
<Skeleton height={14} width={140} />
|
||||||
|
</Table.Td>
|
||||||
|
<Table.Td>
|
||||||
|
<Skeleton height={14} width={120} />
|
||||||
|
</Table.Td>
|
||||||
|
<Table.Td>
|
||||||
|
<Skeleton height={14} width={120} />
|
||||||
|
</Table.Td>
|
||||||
|
</Table.Tr>
|
||||||
|
))}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ResourceCell({ entry }: { entry: IAuditLog }) {
|
||||||
|
if (!entry.resource?.name) {
|
||||||
|
return (
|
||||||
|
<Text fz="sm" c="dimmed">
|
||||||
|
—
|
||||||
|
</Text>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const url = getResourceUrl(entry);
|
||||||
|
|
||||||
|
if (url) {
|
||||||
|
return (
|
||||||
|
<Anchor
|
||||||
|
size="sm"
|
||||||
|
underline="never"
|
||||||
|
style={{
|
||||||
|
cursor: "pointer",
|
||||||
|
color: "var(--mantine-color-text)",
|
||||||
|
}}
|
||||||
|
component={Link}
|
||||||
|
to={url}
|
||||||
|
>
|
||||||
|
<div className={classes.resourceLinkText}>
|
||||||
|
<Text fz="sm" fw={500} lineClamp={1}>
|
||||||
|
{entry.resource.name}
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
</Anchor>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Text fz="sm" lineClamp={1}>
|
||||||
|
{entry.resource.name}
|
||||||
|
</Text>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function AuditLogsTable({
|
||||||
|
items,
|
||||||
|
isLoading,
|
||||||
|
}: AuditLogsTableProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [expanded, setExpanded] = useState<Set<string>>(new Set());
|
||||||
|
|
||||||
|
const toggleExpanded = (id: string) => {
|
||||||
|
setExpanded((prev) => {
|
||||||
|
const next = new Set(prev);
|
||||||
|
if (next.has(id)) {
|
||||||
|
next.delete(id);
|
||||||
|
} else {
|
||||||
|
next.add(id);
|
||||||
|
}
|
||||||
|
return next;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Table.ScrollContainer minWidth={700}>
|
||||||
|
<Table highlightOnHover verticalSpacing="xs" className={classes.table}>
|
||||||
|
<Table.Thead>
|
||||||
|
<Table.Tr>
|
||||||
|
<Table.Th>{t("Actor")}</Table.Th>
|
||||||
|
<Table.Th>{t("Event")}</Table.Th>
|
||||||
|
<Table.Th>{t("Resource")}</Table.Th>
|
||||||
|
<Table.Th>{t("Date")}</Table.Th>
|
||||||
|
</Table.Tr>
|
||||||
|
</Table.Thead>
|
||||||
|
|
||||||
|
<Table.Tbody>
|
||||||
|
{isLoading ? (
|
||||||
|
<TableSkeleton />
|
||||||
|
) : items && items.length > 0 ? (
|
||||||
|
items.map((entry) => {
|
||||||
|
const expandable = hasDetails(entry);
|
||||||
|
const isExpanded = expanded.has(entry.id);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Fragment key={entry.id}>
|
||||||
|
<Table.Tr
|
||||||
|
onClick={
|
||||||
|
expandable ? () => toggleExpanded(entry.id) : undefined
|
||||||
|
}
|
||||||
|
style={{ cursor: expandable ? "pointer" : undefined }}
|
||||||
|
>
|
||||||
|
<Table.Td>
|
||||||
|
<Group gap="sm" wrap="nowrap">
|
||||||
|
{expandable ? (
|
||||||
|
isExpanded ? (
|
||||||
|
<IconChevronDown
|
||||||
|
size={16}
|
||||||
|
color="var(--mantine-color-dimmed)"
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<IconChevronRight
|
||||||
|
size={16}
|
||||||
|
color="var(--mantine-color-dimmed)"
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
) : (
|
||||||
|
<Box w={16} />
|
||||||
|
)}
|
||||||
|
{entry.actor ? (
|
||||||
|
<Group gap="sm" wrap="nowrap">
|
||||||
|
<CustomAvatar
|
||||||
|
avatarUrl={entry.actor.avatarUrl}
|
||||||
|
name={entry.actor.name}
|
||||||
|
size={36}
|
||||||
|
/>
|
||||||
|
<div>
|
||||||
|
<Text fz="sm" fw={500} lineClamp={1}>
|
||||||
|
{entry.actor.name}
|
||||||
|
</Text>
|
||||||
|
<Text fz="xs" c="dimmed">
|
||||||
|
{entry.actor.email}
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
</Group>
|
||||||
|
) : (
|
||||||
|
<Text fz="sm" c="dimmed" fs="italic">
|
||||||
|
{entry.actorType === "system"
|
||||||
|
? t("System")
|
||||||
|
: t("System")}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
</Group>
|
||||||
|
</Table.Td>
|
||||||
|
|
||||||
|
<Table.Td>
|
||||||
|
<Text fz="sm">{t(getEventLabel(entry.event))}</Text>
|
||||||
|
</Table.Td>
|
||||||
|
|
||||||
|
<Table.Td>
|
||||||
|
<ResourceCell entry={entry} />
|
||||||
|
</Table.Td>
|
||||||
|
|
||||||
|
<Table.Td>
|
||||||
|
<Text fz="sm" style={{ whiteSpace: "nowrap" }}>
|
||||||
|
{formattedDate(new Date(entry.createdAt))}
|
||||||
|
</Text>
|
||||||
|
</Table.Td>
|
||||||
|
</Table.Tr>
|
||||||
|
|
||||||
|
{expandable && (
|
||||||
|
<Table.Tr className={classes.detailRow}>
|
||||||
|
<Table.Td colSpan={4} p={0}>
|
||||||
|
<Collapse in={isExpanded}>
|
||||||
|
<Box
|
||||||
|
px="md"
|
||||||
|
py="sm"
|
||||||
|
className={classes.detailContent}
|
||||||
|
>
|
||||||
|
<Group gap="xl" align="flex-start">
|
||||||
|
{entry.changes && (
|
||||||
|
<ChangesDiff changes={entry.changes} />
|
||||||
|
)}
|
||||||
|
{entry.metadata && (
|
||||||
|
<MetadataDisplay metadata={entry.metadata} />
|
||||||
|
)}
|
||||||
|
</Group>
|
||||||
|
</Box>
|
||||||
|
</Collapse>
|
||||||
|
</Table.Td>
|
||||||
|
</Table.Tr>
|
||||||
|
)}
|
||||||
|
</Fragment>
|
||||||
|
);
|
||||||
|
})
|
||||||
|
) : (
|
||||||
|
<NoTableResults colSpan={4} />
|
||||||
|
)}
|
||||||
|
</Table.Tbody>
|
||||||
|
</Table>
|
||||||
|
</Table.ScrollContainer>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
.table {
|
||||||
|
--table-border-color: var(--mantine-color-gray-2);
|
||||||
|
|
||||||
|
@mixin dark {
|
||||||
|
--table-border-color: var(--mantine-color-dark-5);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.resourceLinkText {
|
||||||
|
width: fit-content;
|
||||||
|
|
||||||
|
@mixin light {
|
||||||
|
border-bottom: 0.05em solid var(--mantine-color-dark-0);
|
||||||
|
}
|
||||||
|
@mixin dark {
|
||||||
|
border-bottom: 0.05em solid var(--mantine-color-dark-2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.detailRow {
|
||||||
|
&:hover {
|
||||||
|
background: none !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.detailContent {
|
||||||
|
@mixin light {
|
||||||
|
background: var(--mantine-color-gray-0);
|
||||||
|
}
|
||||||
|
@mixin dark {
|
||||||
|
background: var(--mantine-color-dark-7);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,170 @@
|
|||||||
|
type EventOption = {
|
||||||
|
value: string;
|
||||||
|
label: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
type EventGroup = {
|
||||||
|
group: string;
|
||||||
|
items: EventOption[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export const auditEventLabels: Record<string, string> = {
|
||||||
|
"workspace.created": "Created workspace",
|
||||||
|
"workspace.updated": "Updated workspace",
|
||||||
|
"workspace.invite_created": "Created invitation",
|
||||||
|
"workspace.invite_resent": "Resent invitation",
|
||||||
|
"workspace.invite_revoked": "Revoked invitation",
|
||||||
|
|
||||||
|
"user.created": "Created user",
|
||||||
|
"user.deleted": "Deleted user",
|
||||||
|
"user.login": "Logged in",
|
||||||
|
"user.logout": "Logged out",
|
||||||
|
"user.role_changed": "Changed user role",
|
||||||
|
"user.password_changed": "Changed password",
|
||||||
|
"user.password_reset": "Reset password",
|
||||||
|
"user.updated": "Updated user",
|
||||||
|
"user.deactivated": "Deactivated user",
|
||||||
|
"user.activated": "Activated user",
|
||||||
|
"user.mfa_enabled": "Enabled MFA",
|
||||||
|
"user.mfa_disabled": "Disabled MFA",
|
||||||
|
"user.mfa_backup_code_generated": "Generated MFA backup codes",
|
||||||
|
|
||||||
|
"api_key.created": "Created API key",
|
||||||
|
"api_key.updated": "Updated API key",
|
||||||
|
"api_key.deleted": "Deleted API key",
|
||||||
|
|
||||||
|
"space.created": "Created space",
|
||||||
|
"space.updated": "Updated space",
|
||||||
|
"space.deleted": "Deleted space",
|
||||||
|
"space.member_added": "Added space member",
|
||||||
|
"space.member_removed": "Removed space member",
|
||||||
|
"space.member_role_changed": "Changed space member role",
|
||||||
|
"space.exported": "Exported space",
|
||||||
|
|
||||||
|
"group.created": "Created group",
|
||||||
|
"group.updated": "Updated group",
|
||||||
|
"group.deleted": "Deleted group",
|
||||||
|
"group.member_added": "Added group member",
|
||||||
|
"group.member_removed": "Removed group member",
|
||||||
|
|
||||||
|
"comment.deleted": "Deleted comment",
|
||||||
|
|
||||||
|
"page.trashed": "Trashed page",
|
||||||
|
"page.deleted": "Deleted page",
|
||||||
|
"page.restored": "Restored page",
|
||||||
|
"page.imported": "Imported page",
|
||||||
|
"page.exported": "Exported page",
|
||||||
|
"page.restricted": "Restricted page",
|
||||||
|
"page.restriction_removed": "Removed page restriction",
|
||||||
|
"page.permission_added": "Added page permission",
|
||||||
|
"page.permission_removed": "Removed page permission",
|
||||||
|
|
||||||
|
"share.created": "Created share link",
|
||||||
|
"share.deleted": "Deleted share link",
|
||||||
|
|
||||||
|
"sso.provider_created": "Created SSO provider",
|
||||||
|
"sso.provider_updated": "Updated SSO provider",
|
||||||
|
"sso.provider_deleted": "Deleted SSO provider",
|
||||||
|
|
||||||
|
"license.activated": "Activated license",
|
||||||
|
"license.removed": "Removed license",
|
||||||
|
};
|
||||||
|
|
||||||
|
export function getEventLabel(event: string): string {
|
||||||
|
return auditEventLabels[event] ?? event;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const eventFilterOptions: EventGroup[] = [
|
||||||
|
{
|
||||||
|
group: "Workspace",
|
||||||
|
items: [
|
||||||
|
{ value: "workspace.updated", label: "Updated workspace" },
|
||||||
|
{ value: "workspace.invite_created", label: "Created invitation" },
|
||||||
|
{ value: "workspace.invite_revoked", label: "Revoked invitation" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
group: "User",
|
||||||
|
items: [
|
||||||
|
{ value: "user.login", label: "Logged in" },
|
||||||
|
{ value: "user.logout", label: "Logged out" },
|
||||||
|
{ value: "user.created", label: "Created user" },
|
||||||
|
{ value: "user.deleted", label: "Deleted user" },
|
||||||
|
{ value: "user.deactivated", label: "Deactivated user" },
|
||||||
|
{ value: "user.activated", label: "Activated user" },
|
||||||
|
{ value: "user.role_changed", label: "Changed user role" },
|
||||||
|
{ value: "user.password_changed", label: "Changed password" },
|
||||||
|
{ value: "user.mfa_enabled", label: "Enabled MFA" },
|
||||||
|
{ value: "user.mfa_disabled", label: "Disabled MFA" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
group: "Space",
|
||||||
|
items: [
|
||||||
|
{ value: "space.created", label: "Created space" },
|
||||||
|
{ value: "space.updated", label: "Updated space" },
|
||||||
|
{ value: "space.deleted", label: "Deleted space" },
|
||||||
|
{ value: "space.member_added", label: "Added space member" },
|
||||||
|
{ value: "space.member_removed", label: "Removed space member" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
group: "Group",
|
||||||
|
items: [
|
||||||
|
{ value: "group.created", label: "Created group" },
|
||||||
|
{ value: "group.updated", label: "Updated group" },
|
||||||
|
{ value: "group.deleted", label: "Deleted group" },
|
||||||
|
{ value: "group.member_added", label: "Added group member" },
|
||||||
|
{ value: "group.member_removed", label: "Removed group member" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
group: "Comment",
|
||||||
|
items: [
|
||||||
|
{ value: "comment.deleted", label: "Deleted comment" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
group: "Page",
|
||||||
|
items: [
|
||||||
|
{ value: "page.trashed", label: "Trashed page" },
|
||||||
|
{ value: "page.deleted", label: "Deleted page" },
|
||||||
|
{ value: "page.restored", label: "Restored page" },
|
||||||
|
{ value: "page.imported", label: "Imported page" },
|
||||||
|
{ value: "page.exported", label: "Exported page" },
|
||||||
|
{ value: "page.restricted", label: "Restricted page" },
|
||||||
|
{ value: "page.restriction_removed", label: "Removed page restriction" },
|
||||||
|
{ value: "page.permission_added", label: "Added page permission" },
|
||||||
|
{ value: "page.permission_removed", label: "Removed page permission" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
group: "Share",
|
||||||
|
items: [
|
||||||
|
{ value: "share.created", label: "Created share link" },
|
||||||
|
{ value: "share.deleted", label: "Deleted share link" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
group: "SSO",
|
||||||
|
items: [
|
||||||
|
{ value: "sso.provider_created", label: "Created SSO provider" },
|
||||||
|
{ value: "sso.provider_updated", label: "Updated SSO provider" },
|
||||||
|
{ value: "sso.provider_deleted", label: "Deleted SSO provider" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
group: "API key",
|
||||||
|
items: [
|
||||||
|
{ value: "api_key.created", label: "Created API key" },
|
||||||
|
{ value: "api_key.deleted", label: "Deleted API key" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
group: "License",
|
||||||
|
items: [
|
||||||
|
{ value: "license.activated", label: "Activated license" },
|
||||||
|
{ value: "license.removed", label: "Removed license" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
];
|
||||||
@@ -0,0 +1,223 @@
|
|||||||
|
import { useState, useMemo, useEffect } from "react";
|
||||||
|
import {
|
||||||
|
ActionIcon,
|
||||||
|
Button,
|
||||||
|
Group,
|
||||||
|
NumberInput,
|
||||||
|
Popover,
|
||||||
|
Select,
|
||||||
|
Space,
|
||||||
|
Text,
|
||||||
|
Tooltip,
|
||||||
|
} from "@mantine/core";
|
||||||
|
import { Helmet } from "react-helmet-async";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { IconSettings } from "@tabler/icons-react";
|
||||||
|
import SettingsTitle from "@/components/settings/settings-title";
|
||||||
|
import { getAppName } from "@/lib/config";
|
||||||
|
import Paginate from "@/components/common/paginate";
|
||||||
|
import { useCursorPaginate } from "@/hooks/use-cursor-paginate";
|
||||||
|
import {
|
||||||
|
useAuditLogsQuery,
|
||||||
|
useAuditRetentionQuery,
|
||||||
|
useUpdateAuditRetentionMutation,
|
||||||
|
} from "@/ee/audit/queries/audit-query";
|
||||||
|
import { IAuditLogParams } from "@/ee/audit/types/audit.types";
|
||||||
|
import { eventFilterOptions } from "@/ee/audit/lib/audit-event-labels";
|
||||||
|
import AuditLogsTable from "@/ee/audit/components/audit-logs-table";
|
||||||
|
import useUserRole from "@/hooks/use-user-role";
|
||||||
|
|
||||||
|
type RetentionUnit = "days" | "months" | "years";
|
||||||
|
|
||||||
|
function daysToRetention(days: number): { amount: number; unit: RetentionUnit } {
|
||||||
|
if (days >= 365 && days % 365 === 0) {
|
||||||
|
return { amount: days / 365, unit: "years" };
|
||||||
|
}
|
||||||
|
if (days >= 30 && days % 30 === 0) {
|
||||||
|
return { amount: days / 30, unit: "months" };
|
||||||
|
}
|
||||||
|
return { amount: days, unit: "days" };
|
||||||
|
}
|
||||||
|
|
||||||
|
function retentionToDays(amount: number, unit: RetentionUnit): number {
|
||||||
|
if (unit === "years") return amount * 365;
|
||||||
|
if (unit === "months") return amount * 30;
|
||||||
|
return amount;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function AuditLogs() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { isOwner } = useUserRole();
|
||||||
|
const { cursor, goNext, goPrev, resetCursor } = useCursorPaginate();
|
||||||
|
|
||||||
|
const [eventFilter, setEventFilter] = useState<string | null>(null);
|
||||||
|
const [settingsOpen, setSettingsOpen] = useState(false);
|
||||||
|
|
||||||
|
const { data: retentionData } = useAuditRetentionQuery();
|
||||||
|
const updateRetention = useUpdateAuditRetentionMutation();
|
||||||
|
|
||||||
|
const currentDays = retentionData?.retentionDays ?? 365;
|
||||||
|
const parsed = daysToRetention(currentDays);
|
||||||
|
const [retentionAmount, setRetentionAmount] = useState<number | string>(parsed.amount);
|
||||||
|
const [retentionUnit, setRetentionUnit] = useState<RetentionUnit>(parsed.unit);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (retentionData) {
|
||||||
|
const { amount, unit } = daysToRetention(retentionData.retentionDays);
|
||||||
|
setRetentionAmount(amount);
|
||||||
|
setRetentionUnit(unit);
|
||||||
|
}
|
||||||
|
}, [retentionData?.retentionDays]);
|
||||||
|
|
||||||
|
const resetRetentionForm = () => {
|
||||||
|
const { amount, unit } = daysToRetention(currentDays);
|
||||||
|
setRetentionAmount(amount);
|
||||||
|
setRetentionUnit(unit);
|
||||||
|
};
|
||||||
|
|
||||||
|
const params: IAuditLogParams = useMemo(
|
||||||
|
() => ({
|
||||||
|
cursor,
|
||||||
|
limit: 50,
|
||||||
|
event: eventFilter ?? undefined,
|
||||||
|
}),
|
||||||
|
[cursor, eventFilter],
|
||||||
|
);
|
||||||
|
|
||||||
|
const { data, isLoading } = useAuditLogsQuery(params);
|
||||||
|
|
||||||
|
if (!isOwner) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleEventChange = (value: string | null) => {
|
||||||
|
setEventFilter(value);
|
||||||
|
resetCursor();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Helmet>
|
||||||
|
<title>
|
||||||
|
{t("Audit log")} - {getAppName()}
|
||||||
|
</title>
|
||||||
|
</Helmet>
|
||||||
|
|
||||||
|
<SettingsTitle title={t("Audit log")} />
|
||||||
|
|
||||||
|
<Group mb="md" gap="sm">
|
||||||
|
<Select
|
||||||
|
placeholder={t("Filter by event")}
|
||||||
|
data={eventFilterOptions.map((group) => ({
|
||||||
|
group: t(group.group),
|
||||||
|
items: group.items.map((item) => ({
|
||||||
|
value: item.value,
|
||||||
|
label: t(item.label),
|
||||||
|
})),
|
||||||
|
}))}
|
||||||
|
value={eventFilter}
|
||||||
|
onChange={handleEventChange}
|
||||||
|
clearable
|
||||||
|
searchable
|
||||||
|
w={220}
|
||||||
|
size="sm"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Popover
|
||||||
|
position="bottom-end"
|
||||||
|
shadow="md"
|
||||||
|
width={260}
|
||||||
|
withArrow
|
||||||
|
opened={settingsOpen}
|
||||||
|
onChange={(opened) => {
|
||||||
|
if (!opened) resetRetentionForm();
|
||||||
|
setSettingsOpen(opened);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Popover.Target>
|
||||||
|
<Tooltip label={t("Audit settings")}>
|
||||||
|
<ActionIcon variant="default" size="input-sm" ml="auto" onClick={() => setSettingsOpen((o) => !o)}>
|
||||||
|
<IconSettings size={16} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
</Popover.Target>
|
||||||
|
<Popover.Dropdown>
|
||||||
|
<Text fz="sm" fw={500} mb={4}>
|
||||||
|
{t("Retention")}
|
||||||
|
</Text>
|
||||||
|
<Text fz="xs" c="dimmed" mb="sm">
|
||||||
|
{t("Logs older than this period are automatically deleted.")}
|
||||||
|
</Text>
|
||||||
|
<Group gap="xs" wrap="nowrap" mb="sm">
|
||||||
|
<NumberInput
|
||||||
|
value={retentionAmount}
|
||||||
|
onChange={(val) => setRetentionAmount(val)}
|
||||||
|
min={1}
|
||||||
|
hideControls
|
||||||
|
size="sm"
|
||||||
|
w={60}
|
||||||
|
/>
|
||||||
|
<Select
|
||||||
|
data={[
|
||||||
|
{ value: "days", label: t("days") },
|
||||||
|
{ value: "months", label: t("months") },
|
||||||
|
{ value: "years", label: t("years") },
|
||||||
|
]}
|
||||||
|
value={retentionUnit}
|
||||||
|
onChange={(value) => {
|
||||||
|
if (value === "days" || value === "months" || value === "years") {
|
||||||
|
setRetentionUnit(value);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
size="sm"
|
||||||
|
style={{ flex: 1 }}
|
||||||
|
comboboxProps={{ withinPortal: false }}
|
||||||
|
/>
|
||||||
|
</Group>
|
||||||
|
<Group gap="xs" grow>
|
||||||
|
<Button
|
||||||
|
size="xs"
|
||||||
|
variant="default"
|
||||||
|
onClick={() => {
|
||||||
|
resetRetentionForm();
|
||||||
|
setSettingsOpen(false);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{t("Cancel")}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
size="xs"
|
||||||
|
onClick={() => {
|
||||||
|
const num = typeof retentionAmount === "number" ? retentionAmount : 1;
|
||||||
|
const clamped = Math.max(1, num);
|
||||||
|
setRetentionAmount(clamped);
|
||||||
|
const days = retentionToDays(clamped, retentionUnit);
|
||||||
|
if (days !== currentDays) {
|
||||||
|
updateRetention.mutate({ auditRetentionDays: days });
|
||||||
|
}
|
||||||
|
setSettingsOpen(false);
|
||||||
|
}}
|
||||||
|
loading={updateRetention.isPending}
|
||||||
|
>
|
||||||
|
{t("Save")}
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
</Popover.Dropdown>
|
||||||
|
</Popover>
|
||||||
|
</Group>
|
||||||
|
|
||||||
|
<AuditLogsTable items={data?.items} isLoading={isLoading} />
|
||||||
|
|
||||||
|
<Space h="md" />
|
||||||
|
|
||||||
|
{data?.items && data.items.length > 0 && (
|
||||||
|
<Paginate
|
||||||
|
hasPrevPage={data?.meta?.hasPrevPage}
|
||||||
|
hasNextPage={data?.meta?.hasNextPage}
|
||||||
|
onNext={() => goNext(data?.meta?.nextCursor)}
|
||||||
|
onPrev={goPrev}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,51 @@
|
|||||||
|
import {
|
||||||
|
keepPreviousData,
|
||||||
|
useMutation,
|
||||||
|
useQuery,
|
||||||
|
useQueryClient,
|
||||||
|
UseQueryResult,
|
||||||
|
} from "@tanstack/react-query";
|
||||||
|
import {
|
||||||
|
getAuditLogs,
|
||||||
|
getAuditRetention,
|
||||||
|
updateAuditRetention,
|
||||||
|
} from "@/ee/audit/services/audit-service";
|
||||||
|
import { IAuditLog, IAuditLogParams } from "@/ee/audit/types/audit.types";
|
||||||
|
import { IPagination } from "@/lib/types";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
|
export function useAuditLogsQuery(
|
||||||
|
params?: IAuditLogParams,
|
||||||
|
): UseQueryResult<IPagination<IAuditLog>, Error> {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: ["audit-logs", params],
|
||||||
|
queryFn: () => getAuditLogs(params),
|
||||||
|
placeholderData: keepPreviousData,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useAuditRetentionQuery() {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: ["audit-retention"],
|
||||||
|
queryFn: () => getAuditRetention(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useUpdateAuditRetentionMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: (data: { auditRetentionDays: number }) =>
|
||||||
|
updateAuditRetention(data),
|
||||||
|
onSuccess: () => {
|
||||||
|
notifications.show({ message: t("Audit retention updated") });
|
||||||
|
queryClient.invalidateQueries({ queryKey: ["audit-retention"] });
|
||||||
|
},
|
||||||
|
onError: (error) => {
|
||||||
|
const errorMessage = error["response"]?.data?.message;
|
||||||
|
notifications.show({ message: errorMessage, color: "red" });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
import api from "@/lib/api-client";
|
||||||
|
import { IAuditLog, IAuditLogParams } from "@/ee/audit/types/audit.types";
|
||||||
|
import { IPagination } from "@/lib/types";
|
||||||
|
|
||||||
|
export async function getAuditLogs(
|
||||||
|
params?: IAuditLogParams,
|
||||||
|
): Promise<IPagination<IAuditLog>> {
|
||||||
|
const req = await api.post("/audit", { ...params });
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getAuditRetention(): Promise<{ retentionDays: number }> {
|
||||||
|
const req = await api.post("/audit/retention");
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function updateAuditRetention(data: {
|
||||||
|
auditRetentionDays: number;
|
||||||
|
}): Promise<{ retentionDays: number }> {
|
||||||
|
const req = await api.post("/audit/retention/update", data);
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
export type IAuditLog = {
|
||||||
|
id: string;
|
||||||
|
workspaceId: string;
|
||||||
|
actorId?: string;
|
||||||
|
actorType: string;
|
||||||
|
event: string;
|
||||||
|
resourceType: string;
|
||||||
|
resourceId?: string;
|
||||||
|
spaceId?: string;
|
||||||
|
changes?: {
|
||||||
|
before?: Record<string, any>;
|
||||||
|
after?: Record<string, any>;
|
||||||
|
};
|
||||||
|
metadata?: Record<string, any>;
|
||||||
|
ipAddress?: string;
|
||||||
|
createdAt: string;
|
||||||
|
actor?: {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
email: string;
|
||||||
|
avatarUrl?: string;
|
||||||
|
};
|
||||||
|
resource?: {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
slug?: string;
|
||||||
|
slugId?: string;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export type IAuditLogParams = {
|
||||||
|
event?: string;
|
||||||
|
resourceType?: string;
|
||||||
|
actorId?: string;
|
||||||
|
spaceId?: string;
|
||||||
|
startDate?: string;
|
||||||
|
endDate?: string;
|
||||||
|
cursor?: string;
|
||||||
|
limit?: number;
|
||||||
|
};
|
||||||
@@ -5,3 +5,15 @@ export async function getJoinedWorkspaces(): Promise<Partial<IWorkspace[]>> {
|
|||||||
const req = await api.post<Partial<IWorkspace[]>>("/workspace/joined");
|
const req = await api.post<Partial<IWorkspace[]>>("/workspace/joined");
|
||||||
return req.data;
|
return req.data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function findWorkspacesByEmail(email: string): Promise<void> {
|
||||||
|
await api.post("/workspace/find-by-email", { email });
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function verifyEmail(data: { token: string }): Promise<void> {
|
||||||
|
await api.post("/workspace/verify-email", data);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function resendVerificationEmail(data: { email: string; sig: string }): Promise<void> {
|
||||||
|
await api.post("/workspace/resend-verification", data);
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import {
|
import {
|
||||||
useMutation,
|
useMutation,
|
||||||
useQueryClient,
|
useQueryClient,
|
||||||
|
InfiniteData,
|
||||||
} from "@tanstack/react-query";
|
} from "@tanstack/react-query";
|
||||||
import { resolveComment } from "@/features/comment/services/comment-service";
|
import { resolveComment } from "@/features/comment/services/comment-service";
|
||||||
import {
|
import {
|
||||||
@@ -10,41 +11,54 @@ import {
|
|||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import { IPagination } from "@/lib/types.ts";
|
import { IPagination } from "@/lib/types.ts";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { useQueryEmit } from "@/features/websocket/use-query-emit";
|
|
||||||
import { RQ_KEY } from "@/features/comment/queries/comment-query";
|
import { RQ_KEY } from "@/features/comment/queries/comment-query";
|
||||||
|
|
||||||
|
function updateCommentInCache(
|
||||||
|
cache: InfiniteData<IPagination<IComment>>,
|
||||||
|
commentId: string,
|
||||||
|
updater: (comment: IComment) => IComment,
|
||||||
|
): InfiniteData<IPagination<IComment>> {
|
||||||
|
return {
|
||||||
|
...cache,
|
||||||
|
pages: cache.pages.map((page) => ({
|
||||||
|
...page,
|
||||||
|
items: page.items.map((comment) =>
|
||||||
|
comment.id === commentId ? updater(comment) : comment,
|
||||||
|
),
|
||||||
|
})),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
export function useResolveCommentMutation() {
|
export function useResolveCommentMutation() {
|
||||||
const queryClient = useQueryClient();
|
const queryClient = useQueryClient();
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const emit = useQueryEmit();
|
|
||||||
|
|
||||||
return useMutation({
|
return useMutation({
|
||||||
mutationFn: (data: IResolveComment) => resolveComment(data),
|
mutationFn: (data: IResolveComment) => resolveComment(data),
|
||||||
onMutate: async (variables) => {
|
onMutate: async (variables) => {
|
||||||
await queryClient.cancelQueries({ queryKey: RQ_KEY(variables.pageId) });
|
await queryClient.cancelQueries({ queryKey: RQ_KEY(variables.pageId) });
|
||||||
const previousComments = queryClient.getQueryData(RQ_KEY(variables.pageId));
|
const previousCache = queryClient.getQueryData(RQ_KEY(variables.pageId));
|
||||||
queryClient.setQueryData(RQ_KEY(variables.pageId), (old: IPagination<IComment>) => {
|
|
||||||
if (!old || !old.items) return old;
|
const cache = previousCache as InfiniteData<IPagination<IComment>> | undefined;
|
||||||
const updatedItems = old.items.map((comment) =>
|
if (cache) {
|
||||||
comment.id === variables.commentId
|
queryClient.setQueryData(
|
||||||
? {
|
RQ_KEY(variables.pageId),
|
||||||
...comment,
|
updateCommentInCache(cache, variables.commentId, (comment) => ({
|
||||||
resolvedAt: variables.resolved ? new Date() : null,
|
...comment,
|
||||||
resolvedById: variables.resolved ? 'optimistic-user' : null,
|
resolvedAt: variables.resolved ? new Date() : null,
|
||||||
resolvedBy: variables.resolved ? { id: 'optimistic-user', name: 'Resolving...', avatarUrl: null } : null
|
resolvedById: variables.resolved ? "optimistic" : null,
|
||||||
}
|
resolvedBy: variables.resolved
|
||||||
: comment,
|
? ({ id: "optimistic", name: "", avatarUrl: null } as IComment["resolvedBy"])
|
||||||
|
: null,
|
||||||
|
})),
|
||||||
);
|
);
|
||||||
return {
|
}
|
||||||
...old,
|
|
||||||
items: updatedItems,
|
return { previousCache };
|
||||||
};
|
|
||||||
});
|
|
||||||
return { previousComments };
|
|
||||||
},
|
},
|
||||||
onError: (err, variables, context) => {
|
onError: (_err, variables, context) => {
|
||||||
if (context?.previousComments) {
|
if (context?.previousCache) {
|
||||||
queryClient.setQueryData(RQ_KEY(variables.pageId), context.previousComments);
|
queryClient.setQueryData(RQ_KEY(variables.pageId), context.previousCache);
|
||||||
}
|
}
|
||||||
notifications.show({
|
notifications.show({
|
||||||
message: t("Failed to resolve comment"),
|
message: t("Failed to resolve comment"),
|
||||||
@@ -52,35 +66,26 @@ export function useResolveCommentMutation() {
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
onSuccess: (data: IComment, variables) => {
|
onSuccess: (data: IComment, variables) => {
|
||||||
const pageId = data.pageId;
|
const cache = queryClient.getQueryData(
|
||||||
const currentComments = queryClient.getQueryData(
|
RQ_KEY(data.pageId),
|
||||||
RQ_KEY(pageId),
|
) as InfiniteData<IPagination<IComment>> | undefined;
|
||||||
) as IPagination<IComment>;
|
|
||||||
if (currentComments && currentComments.items) {
|
if (cache) {
|
||||||
const updatedComments = currentComments.items.map((comment) =>
|
queryClient.setQueryData(
|
||||||
comment.id === variables.commentId
|
RQ_KEY(data.pageId),
|
||||||
? { ...comment, resolvedAt: data.resolvedAt, resolvedById: data.resolvedById, resolvedBy: data.resolvedBy }
|
updateCommentInCache(cache, variables.commentId, (comment) => ({
|
||||||
: comment,
|
...comment,
|
||||||
|
resolvedAt: data.resolvedAt,
|
||||||
|
resolvedById: data.resolvedById,
|
||||||
|
resolvedBy: data.resolvedBy,
|
||||||
|
})),
|
||||||
);
|
);
|
||||||
queryClient.setQueryData(RQ_KEY(pageId), {
|
|
||||||
...currentComments,
|
|
||||||
items: updatedComments,
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
emit({
|
|
||||||
operation: "resolveComment",
|
notifications.show({
|
||||||
pageId: pageId,
|
message: variables.resolved
|
||||||
commentId: variables.commentId,
|
? t("Comment resolved successfully")
|
||||||
resolved: variables.resolved,
|
: t("Comment re-opened successfully"),
|
||||||
resolvedAt: data.resolvedAt,
|
|
||||||
resolvedById: data.resolvedById,
|
|
||||||
resolvedBy: data.resolvedBy,
|
|
||||||
});
|
|
||||||
queryClient.invalidateQueries({ queryKey: RQ_KEY(pageId) });
|
|
||||||
notifications.show({
|
|
||||||
message: variables.resolved
|
|
||||||
? t("Comment resolved successfully")
|
|
||||||
: t("Comment re-opened successfully")
|
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import * as z from "zod";
|
import { z } from "zod/v4";
|
||||||
import { useForm, zodResolver } from "@mantine/form";
|
import { useForm } from "@mantine/form";
|
||||||
|
import { zod4Resolver } from "mantine-form-zod-resolver";
|
||||||
import {
|
import {
|
||||||
Container,
|
Container,
|
||||||
Title,
|
Title,
|
||||||
@@ -19,23 +20,38 @@ import APP_ROUTE from "@/lib/app-route.ts";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import JoinedWorkspaces from "@/ee/components/joined-workspaces.tsx";
|
import JoinedWorkspaces from "@/ee/components/joined-workspaces.tsx";
|
||||||
import { useJoinedWorkspacesQuery } from "@/ee/cloud/query/cloud-query.ts";
|
import { useJoinedWorkspacesQuery } from "@/ee/cloud/query/cloud-query.ts";
|
||||||
|
import { findWorkspacesByEmail } from "@/ee/cloud/service/cloud-service.ts";
|
||||||
|
import { AuthLayout } from "@/features/auth/components/auth-layout.tsx";
|
||||||
|
|
||||||
const formSchema = z.object({
|
const formSchema = z.object({
|
||||||
hostname: z.string().min(1, { message: "subdomain is required" }),
|
hostname: z.string().min(1, { message: "subdomain is required" }),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const findWorkspaceSchema = z.object({
|
||||||
|
email: z.string().email({ message: "Please enter a valid email" }),
|
||||||
|
});
|
||||||
|
|
||||||
export function CloudLoginForm() {
|
export function CloudLoginForm() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [isLoading, setIsLoading] = useState<boolean>(false);
|
const [isLoading, setIsLoading] = useState<boolean>(false);
|
||||||
|
const [isFindLoading, setIsFindLoading] = useState<boolean>(false);
|
||||||
|
const [findEmailSent, setFindEmailSent] = useState<boolean>(false);
|
||||||
const { data: joinedWorkspaces } = useJoinedWorkspacesQuery();
|
const { data: joinedWorkspaces } = useJoinedWorkspacesQuery();
|
||||||
|
|
||||||
const form = useForm<any>({
|
const form = useForm<any>({
|
||||||
validate: zodResolver(formSchema),
|
validate: zod4Resolver(formSchema),
|
||||||
initialValues: {
|
initialValues: {
|
||||||
hostname: "",
|
hostname: "",
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const findForm = useForm<any>({
|
||||||
|
validate: zod4Resolver(findWorkspaceSchema),
|
||||||
|
initialValues: {
|
||||||
|
email: "",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
async function onSubmit(data: { hostname: string }) {
|
async function onSubmit(data: { hostname: string }) {
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
|
|
||||||
@@ -53,8 +69,21 @@ export function CloudLoginForm() {
|
|||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function onFindSubmit(data: { email: string }) {
|
||||||
|
setIsFindLoading(true);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await findWorkspacesByEmail(data.email);
|
||||||
|
setFindEmailSent(true);
|
||||||
|
} catch {
|
||||||
|
findForm.setFieldError("email", "An error occurred. Please try again.");
|
||||||
|
}
|
||||||
|
|
||||||
|
setIsFindLoading(false);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<AuthLayout>
|
||||||
<Container size={420} className={classes.container}>
|
<Container size={420} className={classes.container}>
|
||||||
<Box p="xl" className={classes.containerBox}>
|
<Box p="xl" className={classes.containerBox}>
|
||||||
<Title order={2} ta="center" fw={500} mb="md">
|
<Title order={2} ta="center" fw={500} mb="md">
|
||||||
@@ -82,15 +111,47 @@ export function CloudLoginForm() {
|
|||||||
{t("Continue")}
|
{t("Continue")}
|
||||||
</Button>
|
</Button>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
|
<Divider my="lg" label="or" labelPosition="center" />
|
||||||
|
|
||||||
|
{findEmailSent ? (
|
||||||
|
<Text ta="center" size="sm" c="dimmed">
|
||||||
|
{t("We've sent you an email with your associated workspaces.")}
|
||||||
|
</Text>
|
||||||
|
) : (
|
||||||
|
<form onSubmit={findForm.onSubmit(onFindSubmit)}>
|
||||||
|
<Text fw={600} mb="xs">
|
||||||
|
{t("Find your workspaces")}
|
||||||
|
</Text>
|
||||||
|
<TextInput
|
||||||
|
type="email"
|
||||||
|
placeholder="name@company.com"
|
||||||
|
description={t(
|
||||||
|
"We'll send a list of your workspaces to this email.",
|
||||||
|
)}
|
||||||
|
withErrorStyles={false}
|
||||||
|
{...findForm.getInputProps("email")}
|
||||||
|
/>
|
||||||
|
<Button
|
||||||
|
type="submit"
|
||||||
|
fullWidth
|
||||||
|
mt="md"
|
||||||
|
variant="light"
|
||||||
|
loading={isFindLoading}
|
||||||
|
>
|
||||||
|
{t("Send")}
|
||||||
|
</Button>
|
||||||
|
</form>
|
||||||
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
</Container>
|
</Container>
|
||||||
|
|
||||||
<Text ta="center">
|
<Text ta="center" mb="xl">
|
||||||
{t("Don't have a workspace?")}{" "}
|
{t("Don't have a workspace?")}{" "}
|
||||||
<Anchor component={Link} to={APP_ROUTE.AUTH.CREATE_WORKSPACE} fw={500}>
|
<Anchor component={Link} to={APP_ROUTE.AUTH.CREATE_WORKSPACE} fw={500}>
|
||||||
{t("Create new workspace")}
|
{t("Create new workspace")}
|
||||||
</Anchor>
|
</Anchor>
|
||||||
</Text>
|
</Text>
|
||||||
</div>
|
</AuthLayout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,13 +1,13 @@
|
|||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { Modal, TextInput, PasswordInput, Button, Stack } from "@mantine/core";
|
import { Modal, TextInput, PasswordInput, Button, Stack } from "@mantine/core";
|
||||||
import { useForm } from "@mantine/form";
|
import { useForm } from "@mantine/form";
|
||||||
import { zodResolver } from "mantine-form-zod-resolver";
|
import { zod4Resolver } from "mantine-form-zod-resolver";
|
||||||
import { z } from "zod";
|
import { z } from "zod/v4";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { IAuthProvider } from "@/ee/security/types/security.types";
|
import { IAuthProvider } from "@/ee/security/types/security.types";
|
||||||
import APP_ROUTE from "@/lib/app-route";
|
import APP_ROUTE, { getPostLoginRedirect } from "@/lib/app-route";
|
||||||
import { ldapLogin } from "@/ee/security/services/ldap-auth-service";
|
import { ldapLogin } from "@/ee/security/services/ldap-auth-service";
|
||||||
|
|
||||||
const formSchema = z.object({
|
const formSchema = z.object({
|
||||||
@@ -34,7 +34,7 @@ export function LdapLoginModal({
|
|||||||
const [error, setError] = useState<string | null>(null);
|
const [error, setError] = useState<string | null>(null);
|
||||||
|
|
||||||
const form = useForm({
|
const form = useForm({
|
||||||
validate: zodResolver(formSchema),
|
validate: zod4Resolver(formSchema),
|
||||||
initialValues: {
|
initialValues: {
|
||||||
username: "",
|
username: "",
|
||||||
password: "",
|
password: "",
|
||||||
@@ -59,13 +59,13 @@ export function LdapLoginModal({
|
|||||||
// Handle MFA like the regular login
|
// Handle MFA like the regular login
|
||||||
if (response?.userHasMfa) {
|
if (response?.userHasMfa) {
|
||||||
onClose();
|
onClose();
|
||||||
navigate(APP_ROUTE.AUTH.MFA_CHALLENGE);
|
navigate(APP_ROUTE.AUTH.MFA_CHALLENGE + window.location.search);
|
||||||
} else if (response?.requiresMfaSetup) {
|
} else if (response?.requiresMfaSetup) {
|
||||||
onClose();
|
onClose();
|
||||||
navigate(APP_ROUTE.AUTH.MFA_SETUP_REQUIRED);
|
navigate(APP_ROUTE.AUTH.MFA_SETUP_REQUIRED + window.location.search);
|
||||||
} else {
|
} else {
|
||||||
onClose();
|
onClose();
|
||||||
navigate(APP_ROUTE.HOME);
|
navigate(getPostLoginRedirect());
|
||||||
}
|
}
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
|
|||||||
@@ -1,16 +1,17 @@
|
|||||||
import { Button, Group, Text, Modal, TextInput } from "@mantine/core";
|
import { Button, Group, Text, Modal, TextInput } from "@mantine/core";
|
||||||
import * as z from "zod";
|
import { z } from "zod/v4";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
import { useDisclosure } from "@mantine/hooks";
|
import { useDisclosure } from "@mantine/hooks";
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import { useForm, zodResolver } from "@mantine/form";
|
import { useForm } from "@mantine/form";
|
||||||
|
import { zod4Resolver } from "mantine-form-zod-resolver";
|
||||||
import { notifications } from "@mantine/notifications";
|
import { notifications } from "@mantine/notifications";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { getSubdomainHost } from "@/lib/config.ts";
|
import { getSubdomainHost } from "@/lib/config.ts";
|
||||||
import { IWorkspace } from "@/features/workspace/types/workspace.types.ts";
|
import { IWorkspace } from "@/features/workspace/types/workspace.types.ts";
|
||||||
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
import { updateWorkspace } from "@/features/workspace/services/workspace-service.ts";
|
||||||
import { getHostnameUrl } from "@/ee/utils.ts";
|
import { getHostnameUrl } from "@/ee/utils.ts";
|
||||||
import { useAtom } from "jotai/index";
|
import { useAtom } from "jotai";
|
||||||
import {
|
import {
|
||||||
currentUserAtom,
|
currentUserAtom,
|
||||||
workspaceAtom,
|
workspaceAtom,
|
||||||
@@ -66,7 +67,7 @@ function ChangeHostnameForm({ onClose }: ChangeHostnameFormProps) {
|
|||||||
const [currentUser, setCurrentUser] = useAtom(currentUserAtom);
|
const [currentUser, setCurrentUser] = useAtom(currentUserAtom);
|
||||||
|
|
||||||
const form = useForm<FormValues>({
|
const form = useForm<FormValues>({
|
||||||
validate: zodResolver(formSchema),
|
validate: zod4Resolver(formSchema),
|
||||||
initialValues: {
|
initialValues: {
|
||||||
hostname: currentUser?.workspace?.hostname,
|
hostname: currentUser?.workspace?.hostname,
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ 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 { LdapLoginModal } from "@/ee/components/ldap-login-modal.tsx";
|
import { LdapLoginModal } from "@/ee/components/ldap-login-modal.tsx";
|
||||||
|
|
||||||
export default function SsoLogin() {
|
export default function SsoLogin() {
|
||||||
@@ -57,7 +56,7 @@ export default function SsoLogin() {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{(isCloud() || data.hasLicenseKey) && (
|
{data.authProviders.length > 0 && (
|
||||||
<>
|
<>
|
||||||
<Stack align="stretch" justify="center" gap="sm">
|
<Stack align="stretch" justify="center" gap="sm">
|
||||||
{data.authProviders.map((provider) => (
|
{data.authProviders.map((provider) => (
|
||||||
|
|||||||
@@ -0,0 +1,7 @@
|
|||||||
|
import { atomWithStorage } from "jotai/utils";
|
||||||
|
import type { Entitlements } from "./entitlement.types";
|
||||||
|
|
||||||
|
export const entitlementAtom = atomWithStorage<Entitlements | null>(
|
||||||
|
"entitlements",
|
||||||
|
null,
|
||||||
|
);
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
import api from "@/lib/api-client";
|
||||||
|
import { Entitlements } from "./entitlement.types";
|
||||||
|
|
||||||
|
export async function getEntitlements(): Promise<Entitlements> {
|
||||||
|
const req = await api.post<Entitlements>("/workspace/entitlements");
|
||||||
|
return req.data as Entitlements;
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
export type Tier = "free" | "standard" | "business" | "enterprise";
|
||||||
|
|
||||||
|
export type Entitlements = {
|
||||||
|
cloud: boolean;
|
||||||
|
tier: Tier;
|
||||||
|
features: string[];
|
||||||
|
};
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
import { useQuery, UseQueryResult } from "@tanstack/react-query";
|
||||||
|
import { getEntitlements } from "./entitlement-service";
|
||||||
|
import { Entitlements } from "./entitlement.types";
|
||||||
|
|
||||||
|
export function useEntitlements(): UseQueryResult<Entitlements> {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: ["entitlements"],
|
||||||
|
queryFn: getEntitlements,
|
||||||
|
staleTime: 5 * 60 * 1000,
|
||||||
|
});
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user