From 2443935830d79e5542240adb0562478ad9d45694 Mon Sep 17 00:00:00 2001 From: liuyi Date: Tue, 24 Dec 2024 15:29:48 +0800 Subject: [PATCH] chore: add monorepo tools (#9196) --- .devcontainer/build.sh | 4 +- .devcontainer/devcontainer.json | 3 +- .env.template | 7 - .github/actions/copilot-test/action.yml | 2 +- .github/actions/server-test-env/action.yml | 6 +- .github/workflows/build-images.yml | 10 +- .github/workflows/build-test.yml | 39 +- .github/workflows/copilot-test.yml | 4 +- .github/workflows/release-desktop.yml | 14 +- .github/workflows/release-mobile.yml | 6 +- .gitignore | 4 + .prettierignore | 51 +- .vscode/launch.template.json | 18 +- docs/BUILDING.md | 4 +- docs/building-desktop-client-app.md | 10 +- docs/developing-server.md | 2 +- eslint.config.mjs | 34 +- oxlint.json | 35 +- package.json | 17 +- packages/backend/native/index.d.ts | 1 - packages/backend/server/README.md | 2 +- packages/backend/server/package.json | 21 +- packages/common/infra/tsconfig.json | 3 - packages/common/infra/tsconfig.node.json | 12 - packages/frontend/admin/package.json | 3 +- packages/frontend/admin/tailwind.config.js | 3 +- packages/frontend/apps/android/README.md | 6 +- packages/frontend/apps/android/package.json | 5 +- packages/frontend/apps/electron/README.md | 2 +- packages/frontend/apps/electron/package.json | 22 +- .../frontend/apps/electron/scripts/common.ts | 10 +- .../frontend/apps/electron/scripts/dev.ts | 1 + packages/frontend/apps/electron/tsconfig.json | 3 + .../frontend/apps/electron/tsconfig.node.json | 9 +- .../frontend/apps/electron/vitest.config.ts | 1 + .../frontend/apps/electron/webpack.config.js | 12 + packages/frontend/apps/ios/README.md | 10 +- packages/frontend/apps/ios/package.json | 7 +- packages/frontend/apps/mobile/package.json | 5 +- packages/frontend/apps/web/package.json | 5 +- .../frontend/component/.storybook/main.ts | 2 +- packages/frontend/component/package.json | 1 - packages/frontend/component/tsconfig.json | 7 +- packages/frontend/core/public/robots.txt | 1 + packages/frontend/core/tsconfig.json | 6 - packages/frontend/core/tsconfig.node.json | 23 - packages/frontend/core/tsconfig.server.json | 13 - packages/frontend/graphql/codegen.yml | 2 - packages/frontend/graphql/package.json | 2 +- packages/frontend/graphql/src/schema.ts | 2 +- packages/frontend/i18n/.i18n-codegen.json | 2 +- packages/frontend/i18n/build.mjs | 2 +- .../i18n/src/i18n-completenesses.json | 2 +- packages/frontend/i18n/src/i18n.gen.ts | 6888 +++++++++++++++++ packages/frontend/i18n/src/i18next.ts | 2 +- packages/frontend/templates/package.json | 3 +- scripts/bump-blocksuite.js | 3 - scripts/bump-octobase.sh | 3 - scripts/check-version.mjs | 15 - scripts/detect-blocksuite-update.mjs | 105 - scripts/setup/global.ts | 7 +- .../affine-cloud-copilot/playwright.config.ts | 5 +- tests/affine-cloud/playwright.config.ts | 11 +- .../affine-desktop-cloud/playwright.config.ts | 4 +- tests/affine-desktop/playwright.config.ts | 2 +- tests/affine-local/playwright.config.ts | 2 +- tests/affine-mobile/playwright.config.ts | 2 +- tests/kit/utils/cloud.ts | 14 +- tools/bump-blocksuite/README.md | 13 - tools/bump-blocksuite/index.js | 176 - tools/bump-blocksuite/package.json | 14 - tools/cli/README.md | 113 + tools/cli/bin/runner.js | 68 + tools/cli/hooks.js | 14 + tools/cli/package.json | 38 +- tools/cli/register.js | 3 + tools/cli/src/affine.ts | 32 + tools/cli/src/bin/build.ts | 66 - tools/cli/src/bin/dev.ts | 146 - tools/cli/src/build.ts | 15 + tools/cli/src/bundle.ts | 93 + tools/cli/src/clean.ts | 71 + tools/cli/src/codegen.ts | 67 + tools/cli/src/command.ts | 71 + tools/cli/src/config/cwd.cjs | 38 - tools/cli/src/config/index.ts | 9 - tools/cli/src/context.ts | 6 + tools/cli/src/dev.ts | 15 + tools/cli/src/run.ts | 135 + tools/cli/src/util/infra.ts | 26 - tools/cli/src/webpack/html-plugin.ts | 197 + tools/cli/src/webpack/{config.ts => index.ts} | 168 +- tools/cli/src/webpack/postcss.config.cjs | 47 - tools/cli/src/webpack/s3-plugin.ts | 2 - tools/cli/src/webpack/types.ts | 4 + tools/cli/src/webpack/webpack.config.ts | 183 - tools/cli/tsconfig.json | 13 +- tools/commitlint/.commitlintrc.json | 6 +- tools/playstore-auto-bump/index.ts | 14 +- tools/playstore-auto-bump/package.json | 1 + tools/playstore-auto-bump/tsconfig.json | 3 +- tools/utils/package.json | 21 + .../src/build-config.ts} | 46 +- tools/utils/src/distribution.ts | 30 + tools/utils/src/logger.ts | 46 + tools/utils/src/package.ts | 81 + tools/utils/src/path.ts | 49 + tools/utils/src/process.ts | 99 + tools/utils/src/types.ts | 20 + tools/utils/src/workspace.gen.ts | 512 ++ tools/utils/src/workspace.ts | 204 + tools/utils/tsconfig.json | 16 + tsconfig.json | 17 +- tsconfig.node.json | 3 - tsconfig.project.json | 58 + yarn.lock | 239 +- 116 files changed, 9388 insertions(+), 1464 deletions(-) delete mode 100644 .env.template delete mode 100644 packages/common/infra/tsconfig.node.json create mode 100644 packages/frontend/apps/electron/webpack.config.js delete mode 100644 packages/frontend/core/tsconfig.node.json delete mode 100644 packages/frontend/core/tsconfig.server.json create mode 100644 packages/frontend/i18n/src/i18n.gen.ts delete mode 100755 scripts/bump-blocksuite.js delete mode 100755 scripts/bump-octobase.sh delete mode 100644 scripts/check-version.mjs delete mode 100644 scripts/detect-blocksuite-update.mjs delete mode 100644 tools/bump-blocksuite/README.md delete mode 100644 tools/bump-blocksuite/index.js delete mode 100644 tools/bump-blocksuite/package.json create mode 100644 tools/cli/README.md create mode 100755 tools/cli/bin/runner.js create mode 100644 tools/cli/hooks.js create mode 100644 tools/cli/register.js create mode 100644 tools/cli/src/affine.ts delete mode 100644 tools/cli/src/bin/build.ts delete mode 100644 tools/cli/src/bin/dev.ts create mode 100644 tools/cli/src/build.ts create mode 100644 tools/cli/src/bundle.ts create mode 100644 tools/cli/src/clean.ts create mode 100755 tools/cli/src/codegen.ts create mode 100644 tools/cli/src/command.ts delete mode 100644 tools/cli/src/config/cwd.cjs delete mode 100644 tools/cli/src/config/index.ts create mode 100644 tools/cli/src/context.ts create mode 100644 tools/cli/src/dev.ts create mode 100644 tools/cli/src/run.ts delete mode 100644 tools/cli/src/util/infra.ts create mode 100644 tools/cli/src/webpack/html-plugin.ts rename tools/cli/src/webpack/{config.ts => index.ts} (77%) delete mode 100644 tools/cli/src/webpack/postcss.config.cjs create mode 100644 tools/cli/src/webpack/types.ts delete mode 100644 tools/cli/src/webpack/webpack.config.ts create mode 100644 tools/utils/package.json rename tools/{cli/src/webpack/runtime-config.ts => utils/src/build-config.ts} (67%) create mode 100644 tools/utils/src/distribution.ts create mode 100644 tools/utils/src/logger.ts create mode 100644 tools/utils/src/package.ts create mode 100644 tools/utils/src/path.ts create mode 100644 tools/utils/src/process.ts create mode 100644 tools/utils/src/types.ts create mode 100644 tools/utils/src/workspace.gen.ts create mode 100644 tools/utils/src/workspace.ts create mode 100644 tools/utils/tsconfig.json create mode 100644 tsconfig.project.json diff --git a/.devcontainer/build.sh b/.devcontainer/build.sh index 097273e020d13..47940963b8748 100644 --- a/.devcontainer/build.sh +++ b/.devcontainer/build.sh @@ -9,7 +9,7 @@ corepack prepare yarn@stable --activate yarn install # Build Server Dependencies -yarn workspace @affine/server-native build +yarn affine @affine/server-native build # Create database -yarn workspace @affine/server prisma db push +yarn affine @affine/server prisma db push diff --git a/.devcontainer/devcontainer.json b/.devcontainer/devcontainer.json index 40d84f828a074..191a02c0e95e5 100644 --- a/.devcontainer/devcontainer.json +++ b/.devcontainer/devcontainer.json @@ -21,6 +21,5 @@ } }, "updateContentCommand": "bash ./.devcontainer/build.sh", - "postCreateCommand": "bash ./.devcontainer/setup-user.sh", - "postStartCommand": ["yarn dev", "yarn workspace @affine/server dev"] + "postCreateCommand": "bash ./.devcontainer/setup-user.sh" } diff --git a/.env.template b/.env.template deleted file mode 100644 index 4c8ede67baf33..0000000000000 --- a/.env.template +++ /dev/null @@ -1,7 +0,0 @@ -CHANGELOG_URL= -ENABLE_NEW_SETTING_UNSTABLE_API= -ENABLE_CAPTCHA= -CAPTCHA_SITE_KEY= -ENABLE_ENHANCE_SHARE_MODE= -ALLOW_LOCAL_WORKSPACE= -DEBUG_JOTAI= \ No newline at end of file diff --git a/.github/actions/copilot-test/action.yml b/.github/actions/copilot-test/action.yml index 6c89bbde3971f..a9c95250621f4 100644 --- a/.github/actions/copilot-test/action.yml +++ b/.github/actions/copilot-test/action.yml @@ -3,7 +3,7 @@ description: 'Run Copilot E2E Test' inputs: script: description: 'Script to run' - default: 'yarn workspace @affine-test/affine-cloud-copilot e2e --forbid-only' + default: 'yarn affine @affine-test/affine-cloud-copilot e2e --forbid-only' required: false openai-key: description: 'OpenAI secret key' diff --git a/.github/actions/server-test-env/action.yml b/.github/actions/server-test-env/action.yml index 22bfe52172c37..5e8959fc4a8d6 100644 --- a/.github/actions/server-test-env/action.yml +++ b/.github/actions/server-test-env/action.yml @@ -18,6 +18,6 @@ runs: env: NODE_ENV: test run: | - yarn workspace @affine/server exec prisma generate - yarn workspace @affine/server exec prisma db push - yarn workspace @affine/server data-migration run + yarn affine @affine/server prisma generate + yarn affine @affine/server prisma db push + yarn affine @affine/server data-migration run diff --git a/.github/workflows/build-images.yml b/.github/workflows/build-images.yml index 326fc85803d14..25a70f9ab8f86 100644 --- a/.github/workflows/build-images.yml +++ b/.github/workflows/build-images.yml @@ -27,7 +27,7 @@ jobs: electron-install: false extra-flags: workspaces focus @affine/server - name: Build Server - run: yarn workspace @affine/server build + run: yarn affine @affine/server build - name: Upload server dist uses: actions/upload-artifact@v4 with: @@ -47,7 +47,7 @@ jobs: - name: Setup Node.js uses: ./.github/actions/setup-node - name: Build Core - run: yarn workspace @affine/web build + run: yarn affine @affine/web build env: R2_ACCOUNT_ID: ${{ secrets.R2_ACCOUNT_ID }} R2_ACCESS_KEY_ID: ${{ secrets.R2_ACCESS_KEY_ID }} @@ -80,7 +80,7 @@ jobs: - name: Setup Node.js uses: ./.github/actions/setup-node - name: Build Admin - run: yarn workspace @affine/admin build + run: yarn affine @affine/admin build env: R2_ACCOUNT_ID: ${{ secrets.R2_ACCOUNT_ID }} R2_ACCESS_KEY_ID: ${{ secrets.R2_ACCESS_KEY_ID }} @@ -112,7 +112,7 @@ jobs: - name: Setup Node.js uses: ./.github/actions/setup-node - name: Build Mobile - run: yarn workspace @affine/mobile build + run: yarn affine @affine/mobile build env: R2_ACCOUNT_ID: ${{ secrets.R2_ACCOUNT_ID }} R2_ACCESS_KEY_ID: ${{ secrets.R2_ACCESS_KEY_ID }} @@ -257,7 +257,7 @@ jobs: yarn workspaces focus @affine/server --production - name: Generate Prisma client - run: yarn workspace @affine/server prisma generate + run: yarn affine @affine/server prisma generate - name: Setup Version id: version diff --git a/.github/workflows/build-test.yml b/.github/workflows/build-test.yml index 4fb05573fef45..30ca8bc526e0c 100644 --- a/.github/workflows/build-test.yml +++ b/.github/workflows/build-test.yml @@ -89,7 +89,7 @@ jobs: electron-install: false full-cache: true - name: Run i18n codegen - run: yarn workspace @affine/i18n build + run: yarn affine @affine/i18n build - name: Run ESLint run: yarn lint:eslint --max-warnings=0 - name: Run Prettier @@ -162,7 +162,7 @@ jobs: full-cache: true - name: Run playwright tests - run: yarn workspace @affine-test/affine-local e2e --forbid-only --shard=${{ matrix.shard }}/${{ strategy.job-total }} + run: yarn affine @affine-test/affine-local e2e --forbid-only --shard=${{ matrix.shard }}/${{ strategy.job-total }} - name: Upload test results if: ${{ failure() }} @@ -192,7 +192,7 @@ jobs: full-cache: true - name: Run playwright tests - run: yarn workspace @affine-test/affine-mobile e2e --forbid-only --shard=${{ matrix.shard }}/${{ strategy.job-total }} + run: yarn affine @affine-test/affine-mobile e2e --forbid-only --shard=${{ matrix.shard }}/${{ strategy.job-total }} - name: Upload test results if: ${{ failure() }} @@ -315,8 +315,7 @@ jobs: electron-install: false full-cache: true - name: Build Electron renderer - # always skip cache because its fast, and cache configuration is always changing - run: yarn build + run: yarn affine @affine/electron bundle env: DISTRIBUTION: desktop - name: zip web @@ -377,7 +376,7 @@ jobs: uses: ./.github/actions/server-test-env - name: Run server tests - run: yarn workspace @affine/server test:coverage + run: yarn affine @affine/server test:coverage env: CARGO_TARGET_DIR: '${{ github.workspace }}/target' COPILOT_OPENAI_API_KEY: 'use_fake_openai_api_key' @@ -480,7 +479,7 @@ jobs: - name: Run server tests if: ${{ steps.check-blocksuite-update.outputs.skip != 'true' || steps.apifilter.outputs.changed == 'true' }} - run: yarn workspace @affine/server test:copilot:coverage --forbid-only + run: yarn affine @affine/server test:copilot:coverage --forbid-only env: CARGO_TARGET_DIR: '${{ github.workspace }}/target' COPILOT_OPENAI_API_KEY: ${{ secrets.COPILOT_OPENAI_API_KEY }} @@ -570,7 +569,7 @@ jobs: if: ${{ steps.check-blocksuite-update.outputs.skip != 'true' || steps.e2efilter.outputs.changed == 'true' }} uses: ./.github/actions/copilot-test with: - script: yarn workspace @affine-test/affine-cloud-copilot e2e --forbid-only --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} + script: yarn affine @affine-test/affine-cloud-copilot e2e --forbid-only --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} openai-key: ${{ secrets.COPILOT_OPENAI_API_KEY }} fal-key: ${{ secrets.COPILOT_FAL_API_KEY }} @@ -587,19 +586,19 @@ jobs: matrix: tests: - name: 'Server E2E Test 1/3' - script: yarn workspace @affine-test/affine-cloud e2e --forbid-only --shard=1/3 + script: yarn affine @affine-test/affine-cloud e2e --forbid-only --shard=1/3 - name: 'Server E2E Test 2/3' - script: yarn workspace @affine-test/affine-cloud e2e --forbid-only --shard=2/3 + script: yarn affine @affine-test/affine-cloud e2e --forbid-only --shard=2/3 - name: 'Server E2E Test 3/3' - script: yarn workspace @affine-test/affine-cloud e2e --forbid-only --shard=3/3 + script: yarn affine @affine-test/affine-cloud e2e --forbid-only --shard=3/3 - name: 'Server Desktop E2E Test' script: | - yarn workspace @affine/electron build:dev + yarn affine @affine/electron build:dev # Workaround for Electron apps failing to initialize on Ubuntu 24.04 due to AppArmor restrictions # Disables unprivileged user namespaces restriction to allow Electron apps to run # Reference: https://github.com/electron/electron/issues/42510 sudo sysctl -w kernel.apparmor_restrict_unprivileged_userns=0 - xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn workspace @affine-test/affine-desktop-cloud e2e + xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn affine @affine-test/affine-desktop-cloud e2e needs: - build-server-native - build-native @@ -729,7 +728,7 @@ jobs: - name: Run unit tests if: ${{ matrix.spec.test }} shell: bash - run: yarn workspace @affine/electron vitest + run: yarn affine @affine/electron vitest - name: Download web artifact uses: ./.github/actions/download-web @@ -737,7 +736,7 @@ jobs: path: packages/frontend/apps/electron/resources/web-static - name: Build Desktop Layers - run: yarn workspace @affine/electron build + run: yarn affine @affine/electron build - name: Run desktop tests if: ${{ matrix.spec.os == 'ubuntu-latest' }} @@ -746,11 +745,11 @@ jobs: # Disables unprivileged user namespaces restriction to allow Electron apps to run # Reference: https://github.com/electron/electron/issues/42510 sudo sysctl -w kernel.apparmor_restrict_unprivileged_userns=0 - xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn workspace @affine-test/affine-desktop e2e + xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn affine @affine-test/affine-desktop e2e - name: Run desktop tests if: ${{ matrix.spec.test && matrix.spec.os != 'ubuntu-latest' }} - run: yarn workspace @affine-test/affine-desktop e2e + run: yarn affine @affine-test/affine-desktop e2e - name: Make bundle (macOS) if: ${{ matrix.spec.target == 'aarch64-apple-darwin' }} @@ -758,7 +757,7 @@ jobs: SKIP_BUNDLE: true SKIP_WEB_BUILD: true HOIST_NODE_MODULES: 1 - run: yarn workspace @affine/electron package --platform=darwin --arch=arm64 + run: yarn affine @affine/electron package --platform=darwin --arch=arm64 - name: Make Bundle (Linux) run: | @@ -768,7 +767,7 @@ jobs: flatpak update # some flatpak deps need git protocol.file.allow git config --global protocol.file.allow always - yarn workspace @affine/electron make --platform=linux --arch=x64 + yarn affine @affine/electron make --platform=linux --arch=x64 if: ${{ matrix.spec.target == 'x86_64-unknown-linux-gnu' }} env: SKIP_WEB_BUILD: 1 @@ -777,7 +776,7 @@ jobs: - name: Output check if: ${{ matrix.spec.os == 'macos-14' && matrix.spec.arch == 'arm64' }} run: | - yarn workspace @affine/electron exec node --loader ts-node/esm/transpile-only ./scripts/macos-arm64-output-check.ts + yarn affine @affine/electron node ./scripts/macos-arm64-output-check.ts - name: Upload test results if: ${{ failure() }} diff --git a/.github/workflows/copilot-test.yml b/.github/workflows/copilot-test.yml index 6f3cc8156fc6f..e330ca49f615a 100644 --- a/.github/workflows/copilot-test.yml +++ b/.github/workflows/copilot-test.yml @@ -82,7 +82,7 @@ jobs: uses: ./.github/actions/server-test-env - name: Run server tests - run: yarn workspace @affine/server test:copilot:coverage --forbid-only + run: yarn affine @affine/server test:copilot:coverage --forbid-only env: CARGO_TARGET_DIR: '${{ github.workspace }}/target' COPILOT_OPENAI_API_KEY: ${{ secrets.COPILOT_OPENAI_API_KEY }} @@ -147,7 +147,7 @@ jobs: - name: Run Copilot E2E Test ${{ matrix.shardIndex }}/${{ matrix.shardTotal }} uses: ./.github/actions/copilot-test with: - script: yarn workspace @affine-test/affine-cloud-copilot e2e --forbid-only --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} + script: yarn affine @affine-test/affine-cloud-copilot e2e --forbid-only --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} openai-key: ${{ secrets.COPILOT_OPENAI_API_KEY }} fal-key: ${{ secrets.COPILOT_FAL_API_KEY }} diff --git a/.github/workflows/release-desktop.yml b/.github/workflows/release-desktop.yml index 9810136d41665..82516bf70c9b4 100644 --- a/.github/workflows/release-desktop.yml +++ b/.github/workflows/release-desktop.yml @@ -52,7 +52,7 @@ jobs: - name: Setup @sentry/cli uses: ./.github/actions/setup-sentry - name: generate-assets - run: yarn workspace @affine/electron generate-assets + run: yarn affine @affine/electron generate-assets env: SENTRY_ORG: ${{ secrets.SENTRY_ORG }} SENTRY_PROJECT: 'affine' @@ -122,7 +122,7 @@ jobs: path: packages/frontend/apps/electron/resources/web-static - name: Build Desktop Layers - run: yarn workspace @affine/electron build + run: yarn affine @affine/electron build - name: Signing By Apple Developer ID if: ${{ matrix.spec.platform == 'darwin' }} @@ -142,7 +142,7 @@ jobs: git config --global protocol.file.allow always - name: make - run: yarn workspace @affine/electron make --platform=${{ matrix.spec.platform }} --arch=${{ matrix.spec.arch }} + run: yarn affine @affine/electron make --platform=${{ matrix.spec.platform }} --arch=${{ matrix.spec.arch }} env: SKIP_WEB_BUILD: 1 HOIST_NODE_MODULES: 1 @@ -236,10 +236,10 @@ jobs: path: packages/frontend/apps/electron/resources/web-static - name: Build Desktop Layers - run: yarn workspace @affine/electron build + run: yarn affine @affine/electron build - name: package - run: yarn workspace @affine/electron package --platform=${{ matrix.spec.platform }} --arch=${{ matrix.spec.arch }} + run: yarn affine @affine/electron package --platform=${{ matrix.spec.platform }} --arch=${{ matrix.spec.arch }} env: SKIP_WEB_BUILD: 1 HOIST_NODE_MODULES: 1 @@ -314,10 +314,10 @@ jobs: run: Expand-Archive -Path signed.zip -DestinationPath packages/frontend/apps/electron/out - name: Make squirrel.windows installer - run: yarn workspace @affine/electron make-squirrel --platform=${{ matrix.spec.platform }} --arch=${{ matrix.spec.arch }} + run: yarn affine @affine/electron make-squirrel --platform=${{ matrix.spec.platform }} --arch=${{ matrix.spec.arch }} - name: Make nsis.windows installer - run: yarn workspace @affine/electron make-nsis --platform=${{ matrix.spec.platform }} --arch=${{ matrix.spec.arch }} + run: yarn affine @affine/electron make-nsis --platform=${{ matrix.spec.platform }} --arch=${{ matrix.spec.arch }} - name: Zip artifacts for faster upload run: Compress-Archive -CompressionLevel Fastest -Path packages/frontend/apps/electron/out/${{ env.BUILD_TYPE }}/make/* -DestinationPath archive.zip diff --git a/.github/workflows/release-mobile.yml b/.github/workflows/release-mobile.yml index 19a6ce1ccb829..5327806192944 100644 --- a/.github/workflows/release-mobile.yml +++ b/.github/workflows/release-mobile.yml @@ -68,7 +68,7 @@ jobs: - name: Setup @sentry/cli uses: ./.github/actions/setup-sentry - name: Build Mobile - run: yarn workspace @affine/ios build + run: yarn affine @affine/ios build env: PUBLIC_PATH: '/' MIXPANEL_TOKEN: ${{ secrets.MIXPANEL_TOKEN }} @@ -101,7 +101,7 @@ jobs: - name: Setup @sentry/cli uses: ./.github/actions/setup-sentry - name: Build Mobile - run: yarn workspace @affine/android build + run: yarn affine @affine/android build env: PUBLIC_PATH: '/' MIXPANEL_TOKEN: ${{ secrets.MIXPANEL_TOKEN }} @@ -216,7 +216,7 @@ jobs: - name: Auto increment version code id: bump if: ${{ env.BUILD_TARGET == 'distribution' }} - run: yarn workspace @affine/playstore-auto-bump bump + run: yarn affine @affine/playstore-auto-bump bump env: GOOGLE_APPLICATION_CREDENTIALS: ${{ steps.auth.outputs.credentials_file_path }} - name: Build diff --git a/.gitignore b/.gitignore index dfd06deca8975..73e51296a22d4 100644 --- a/.gitignore +++ b/.gitignore @@ -80,3 +80,7 @@ apps/web/next-routes.conf packages/frontend/templates/edgeless packages/frontend/core/public/static/templates + +# script +af +af.cmd diff --git a/.prettierignore b/.prettierignore index 489bc0a065375..abcb9775354db 100644 --- a/.prettierignore +++ b/.prettierignore @@ -1,30 +1,35 @@ -yarn.lock -target -lib -test-results -.next -out -dist +# we will make this file shared by prettier|eslint|oxlint +**/node_modules .yarn -.github/helm -_next -storybook-static -web-static -public -packages/backend/server/src/schema.gql -packages/backend/server/src/base/error/errors.gen.ts -packages/frontend/i18n/src/i18n-generated.ts -packages/frontend/i18n/src/i18n-completenesses.json -packages/frontend/graphql/src/graphql/index.ts +.github +.vscode .yarnrc.yml -packages/frontend/templates/*.gen.ts -packages/frontend/templates/onboarding -# auto-generated by NAPI-RS -# fixme(@joooye34): need script to check and generate ignore list here +# compiled output +.coverage +.nx/** +target +test-results +**/dist +**/lib +**/storybook-static +**/web-static +**/public +**/e2e-dist-* +**/static + +# generated files +**/*.gen.ts +**/*.gql +**/*.d.ts + +# per files +tools/cli/src/webpack/error-handler.js packages/backend/native/index.d.ts packages/frontend/native/index.d.ts packages/frontend/native/index.js -compose.yaml - +packages/frontend/graphql/src/graphql/index.ts +packages/frontend/graphql/src/schema.ts +packages/frontend/apps/android/App/app/build/** blocksuite/tests-legacy/snapshots +**/.storybook diff --git a/.vscode/launch.template.json b/.vscode/launch.template.json index 77681d7a18c9c..689cdc8e3ab75 100644 --- a/.vscode/launch.template.json +++ b/.vscode/launch.template.json @@ -2,24 +2,20 @@ "version": "0.2.0", "configurations": [ { - "name": "Run Dev", - "type": "node-terminal", - "request": "launch", - "command": "yarn run dev" - }, - { - "name": "Run Dev Locally", - "type": "node-terminal", + "name": "Launch AFFiNE Cloud", + "type": "node", "request": "launch", - "command": "yarn run dev:local" + "runtimeExecutable": "yarn", + "cwd": "${workspaceFolder}", + "runtimeArgs": ["affine", "@affine/server", "dev"] }, { - "name": "Launch AFFiNE Cloud", + "name": "Lanuch AFFiNE Web", "type": "node", "request": "launch", "runtimeExecutable": "yarn", "cwd": "${workspaceFolder}", - "runtimeArgs": ["workspace", "@affine/server", "dev"] + "runtimeArgs": ["affine", "@affine/web", "dev"] } ] } diff --git a/docs/BUILDING.md b/docs/BUILDING.md index 110bba4d6bd99..59dfc6c73e6c3 100644 --- a/docs/BUILDING.md +++ b/docs/BUILDING.md @@ -87,13 +87,13 @@ This could take a while if you build it for the first time. Note: use `strip` from system instead of `binutils` if you are running MacOS. [see problem here](https://github.com/toeverything/AFFiNE/discussions/2840) ``` -yarn workspace @affine/native build +yarn affine @affine/native build ``` ### Build Server Dependencies ```sh -yarn workspace @affine/server-native build +yarn affine @affine/server-native build ``` ## Testing diff --git a/docs/building-desktop-client-app.md b/docs/building-desktop-client-app.md index f7d631e1a6e8e..c73f3ec3f7b8f 100644 --- a/docs/building-desktop-client-app.md +++ b/docs/building-desktop-client-app.md @@ -46,7 +46,7 @@ Please refer to `Build Native Dependencies` section in [BUILDING.md](./BUILDING. On Mac & Linux ```shell -BUILD_TYPE=canary yarn workspace @affine/electron generate-assets +BUILD_TYPE=canary yarn affine @affine/electron generate-assets ``` On Windows (powershell) @@ -90,7 +90,7 @@ yarn install Note: you need to comment out `osxSign` and `osxNotarize` in `forge.config.js` to skip signing and notarizing the app. ```shell -BUILD_TYPE=canary SKIP_WEB_BUILD=1 HOIST_NODE_MODULES=1 yarn workspace @affine/electron make +BUILD_TYPE=canary SKIP_WEB_BUILD=1 HOIST_NODE_MODULES=1 yarn affine @affine/electron make ``` #### Windows @@ -101,9 +101,9 @@ Making the windows installer is a bit different. Right now we provide two instal $env:BUILD_TYPE="canary" $env:SKIP_WEB_BUILD=1 $env:HOIST_NODE_MODULES=1 -yarn workspace @affine/electron package -yarn workspace @affine/electron make-squirrel -yarn workspace @affine/electron make-nsis +yarn affine @affine/electron package +yarn affine @affine/electron make-squirrel +yarn affine @affine/electron make-nsis ``` Once the build is complete, you can find the paths to the binaries in the terminal output. diff --git a/docs/developing-server.md b/docs/developing-server.md index c85491017067d..19f85cbba68d0 100644 --- a/docs/developing-server.md +++ b/docs/developing-server.md @@ -18,7 +18,7 @@ docker compose -f ./.docker/dev/compose.yml up -d ```sh # build native -yarn workspace @affine/server-native build +yarn affine @affine/server-native build ``` ## Prepare dev environment diff --git a/eslint.config.mjs b/eslint.config.mjs index 78fa2593bc24c..47b14599f25cf 100644 --- a/eslint.config.mjs +++ b/eslint.config.mjs @@ -1,3 +1,5 @@ +import { readFileSync } from 'node:fs'; + import eslint from '@eslint/js'; import rxjs from '@smarttools/eslint-plugin-rxjs'; import tsParser from '@typescript-eslint/parser'; @@ -10,33 +12,13 @@ import sonarjs from 'eslint-plugin-sonarjs'; import unicorn from 'eslint-plugin-unicorn'; import tseslint from 'typescript-eslint'; +const ignoreList = readFileSync('.prettierignore', 'utf-8') + .split('\n') + .filter(line => line.trim() && !line.startsWith('#')); + export default tseslint.config( { - ignores: [ - '**/node_modules', - '**/dist', - '**/.next', - '**/out', - '**/storybook-static', - '**/affine-out', - '**/_next', - '**/lib', - '**/.eslintrc.js', - '**/e2e-dist-*', - '**/static', - '**/web-static', - '**/public', - '**/.coverage', - '.nx/**', - '.yarn/**', - '**/*.d.ts', - '.github/**/*', - 'packages/frontend/component/.storybook/**/*', - 'packages/frontend/i18n/src/i18n-generated.ts', - 'packages/frontend/i18n/src/i18n-completenesses.json', - 'packages/frontend/templates/*.gen.ts', - 'packages/frontend/apps/android/App/app/build/**', - ], + ignores: ignoreList, }, { settings: { @@ -218,7 +200,7 @@ export default tseslint.config( { files: [ 'packages/**/*.{ts,tsx}', - 'tools/cli/**/*.{ts,tsx}', + 'tools/**/*.{ts,tsx}', 'blocksuite/**/*.{ts,tsx}', ], rules: { diff --git a/oxlint.json b/oxlint.json index 4b02f12885d80..100755d8bd05e 100644 --- a/oxlint.json +++ b/oxlint.json @@ -5,13 +5,41 @@ "correctness": "error", "perf": "error" }, - "ignorePatterns": ["tools/cli/src/webpack/error-handler.js"], + "ignorePatterns": [ + "**/node_modules", + ".yarn", + ".github", + ".vscode", + ".yarnrc.yml", + ".coverage", + ".nx/**", + "target", + "test-results", + "**/dist", + "**/lib", + "**/storybook-static", + "**/web-static", + "**/public", + "**/e2e-dist-*", + "**/static", + "**/*.gen.ts", + "**/*.gql", + "**/*.d.ts", + "tools/cli/src/webpack/error-handler.js", + "packages/backend/native/index.d.ts", + "packages/frontend/native/index.d.ts", + "packages/frontend/native/index.js", + "packages/frontend/graphql/src/graphql/index.ts", + "packages/frontend/graphql/src/schema.ts", + "packages/frontend/apps/android/App/app/build/**", + "blocksuite/tests-legacy/snapshots", + "**/.storybook" + ], "rules": { "import/named": "allow", "no-await-in-loop": "allow", "promise/no-callback-in-promise": "allow", "typescript/ban-types": "allow", - "array-callback-return": "error", "constructor-super": "error", "eqeqeq": ["error", "smart"], @@ -172,7 +200,8 @@ "files": [ "*.{spec,test,e2e,stories}.{ts,tsx}", "tests/**/*.ts", - "packages/backend/server/tests/**/*.ts" + "packages/backend/server/tests/**/*.ts", + "tools/**.*" ], "rules": { "typescript/no-non-null-assertion": "off", diff --git a/package.json b/package.json index c6f40d7e38b6a..deb61ffbc2beb 100644 --- a/package.json +++ b/package.json @@ -18,12 +18,10 @@ "node": "<21.0.0" }, "scripts": { - "dev": "yarn workspace @affine/cli dev", - "build": "yarn workspace @affine/cli bundle", - "dev:electron": "yarn workspace @affine/electron dev", - "build:electron": "yarn workspace @affine/electron build", - "build:server-native": "yarn workspace @affine/server-native build", - "start:web-static": "yarn workspace @affine/web static-server", + "affine": "yarn workspace @affine-tools/cli affine", + "af": "yarn workspace @affine-tools/cli affine", + "dev": "yarn affine dev", + "build": "yarn affine build", "lint:eslint": "cross-env NODE_OPTIONS=\"--max-old-space-size=8192\" eslint --report-unused-disable-directives-severity=off . --cache", "lint:eslint:fix": "yarn lint:eslint --fix --fix-type problem,suggestion,layout", "lint:prettier": "prettier --ignore-unknown --cache --check .", @@ -34,9 +32,8 @@ "test": "vitest --run", "test:ui": "vitest --ui", "test:coverage": "vitest run --coverage", - "typecheck": "tsc -b tsconfig.json", - "postinstall": "node ./scripts/check-version.mjs && yarn workspace @affine/i18n i18n-codegen gen && yarn husky install", - "prepare": "husky" + "typecheck": "tsc -b tsconfig.json --verbose", + "postinstall": "yarn affine init && yarn husky" }, "lint-staged": { "*": "prettier --write --ignore-unknown --cache", @@ -52,7 +49,7 @@ ] }, "devDependencies": { - "@affine/cli": "workspace:*", + "@affine-tools/cli": "workspace:*", "@capacitor/cli": "^6.2.0", "@eslint/js": "^9.16.0", "@faker-js/faker": "^9.3.0", diff --git a/packages/backend/native/index.d.ts b/packages/backend/native/index.d.ts index de691213ec201..c5d2e4e1d0cb1 100644 --- a/packages/backend/native/index.d.ts +++ b/packages/backend/native/index.d.ts @@ -19,4 +19,3 @@ export declare function mergeUpdatesInApplyWay(updates: Array): Buffer export declare function mintChallengeResponse(resource: string, bits?: number | undefined | null): Promise export declare function verifyChallengeResponse(response: string, bits: number, resource: string): Promise - diff --git a/packages/backend/server/README.md b/packages/backend/server/README.md index e2aafea95b3e1..57ae0868afeb0 100644 --- a/packages/backend/server/README.md +++ b/packages/backend/server/README.md @@ -11,7 +11,7 @@ yarn ### Build Native binding ```bash -yarn workspace @affine/server-native build +yarn affine @affine/server-native build ``` ### Run server diff --git a/packages/backend/server/package.json b/packages/backend/server/package.json index 53d8055427f15..7a9f01e913c00 100644 --- a/packages/backend/server/package.json +++ b/packages/backend/server/package.json @@ -9,15 +9,14 @@ }, "scripts": { "build": "tsc", - "start": "node --loader ts-node/esm/transpile-only.mjs ./src/index.ts", "dev": "nodemon ./src/index.ts", "test": "ava --concurrency 1 --serial", "test:copilot": "ava \"tests/**/copilot-*.spec.ts\"", "test:coverage": "c8 ava --concurrency 1 --serial", "test:copilot:coverage": "c8 ava --timeout=5m \"tests/**/copilot-*.spec.ts\"", - "postinstall": "prisma generate", - "data-migration": "NODE_ENV=script node --loader ts-node/esm/transpile-only.mjs ./src/data/index.ts", - "predeploy": "yarn prisma migrate deploy && node --import ./scripts/register.js ./dist/data/index.js run" + "data-migration": "NODE_ENV=script node ./src/data/index.ts", + "predeploy": "yarn prisma migrate deploy && node --import ./scripts/register.js ./dist/data/index.js run", + "postinstall": "prisma generate" }, "dependencies": { "@apollo/server": "^4.11.2", @@ -119,10 +118,7 @@ }, "workerThreads": false, "nodeArguments": [ - "--trace-sigint", - "--loader", - "ts-node/esm/transpile-only.mjs", - "--es-module-specifier-resolution=node" + "--trace-sigint" ], "watchMode": { "ignoreChanges": [ @@ -139,7 +135,6 @@ "./src/prelude.ts" ], "environmentVariables": { - "TS_NODE_PROJECT": "./tests/tsconfig.json", "NODE_ENV": "test", "MAILER_HOST": "0.0.0.0", "MAILER_PORT": "1025", @@ -152,12 +147,6 @@ }, "nodemonConfig": { "exec": "node", - "script": "./src/index.ts", - "nodeArgs": [ - "--loader", - "ts-node/esm.mjs", - "--es-module-specifier-resolution=node" - ], "ignore": [ "**/__tests__/**", "**/dist/**", @@ -166,8 +155,6 @@ "env": { "NODE_ENV": "development", "AFFINE_SERVER_EXTERNAL_URL": "http://localhost:8080", - "TS_NODE_TRANSPILE_ONLY": true, - "TS_NODE_PROJECT": "./tsconfig.json", "DEBUG": "affine:*", "FORCE_COLOR": true, "DEBUG_COLORS": true diff --git a/packages/common/infra/tsconfig.json b/packages/common/infra/tsconfig.json index acf0872dae27d..ee7b751d96034 100644 --- a/packages/common/infra/tsconfig.json +++ b/packages/common/infra/tsconfig.json @@ -15,9 +15,6 @@ }, { "path": "../../../blocksuite/affine/all" - }, - { - "path": "./tsconfig.node.json" } ] } diff --git a/packages/common/infra/tsconfig.node.json b/packages/common/infra/tsconfig.node.json deleted file mode 100644 index 097adbc4e390e..0000000000000 --- a/packages/common/infra/tsconfig.node.json +++ /dev/null @@ -1,12 +0,0 @@ -{ - "extends": "../../../tsconfig.json", - "compilerOptions": { - "composite": true, - "module": "ESNext", - "moduleResolution": "Node", - "allowSyntheticDefaultImports": true, - "outDir": "lib", - "noEmit": false - }, - "include": ["vite.config.ts"] -} diff --git a/packages/frontend/admin/package.json b/packages/frontend/admin/package.json index 832d162a6d2c6..73e7d29afb9e3 100644 --- a/packages/frontend/admin/package.json +++ b/packages/frontend/admin/package.json @@ -62,7 +62,8 @@ "tailwindcss-animate": "^1.0.7" }, "scripts": { - "build": "cross-env DISTRIBUTION=admin yarn workspace @affine/cli bundle", + "build": "affine bundle", + "dev": "affine bundle --dev", "update-shadcn": "shadcn-ui add -p src/components/ui" }, "exports": { diff --git a/packages/frontend/admin/tailwind.config.js b/packages/frontend/admin/tailwind.config.js index ee3b0653cb636..f433890cff0f8 100644 --- a/packages/frontend/admin/tailwind.config.js +++ b/packages/frontend/admin/tailwind.config.js @@ -1,7 +1,8 @@ /** @type {import('tailwindcss').Config} */ module.exports = { darkMode: ['class'], - content: ['./src/**/*.{ts,tsx}'], + // TODO(@forehalo): we are not running webpack in admin dir + content: ['./packages/frontend/admin/src/**/*.{ts,tsx}'], prefix: '', theme: { container: { diff --git a/packages/frontend/apps/android/README.md b/packages/frontend/apps/android/README.md index 4ff2a00916d50..6d64273332825 100644 --- a/packages/frontend/apps/android/README.md +++ b/packages/frontend/apps/android/README.md @@ -5,6 +5,6 @@ AFFiNE Android app. ## Build - yarn install -- BUILD_TYPE=canary PUBLIC_PATH="/" yarn workspace @affine/android build -- yarn workspace @affine/android cap sync -- yarn workspace @affine/android cap open android +- BUILD_TYPE=canary PUBLIC_PATH="/" yarn affine @affine/android build +- yarn affine @affine/android cap sync +- yarn affine @affine/android cap open android diff --git a/packages/frontend/apps/android/package.json b/packages/frontend/apps/android/package.json index fffe61ad84744..32cd662b75586 100644 --- a/packages/frontend/apps/android/package.json +++ b/packages/frontend/apps/android/package.json @@ -5,9 +5,8 @@ "private": true, "browser": "src/index.tsx", "scripts": { - "build": "cross-env DISTRIBUTION=android yarn workspace @affine/cli bundle", - "dev": "yarn workspace @affine/cli dev", - "static-server": "cross-env DISTRIBUTION=android yarn workspace @affine/cli dev --static" + "build": "affine bundle", + "dev": "affine bundle --dev" }, "dependencies": { "@affine/component": "workspace:*", diff --git a/packages/frontend/apps/electron/README.md b/packages/frontend/apps/electron/README.md index 0ac1c7a9e6f0e..22d848dafa769 100644 --- a/packages/frontend/apps/electron/README.md +++ b/packages/frontend/apps/electron/README.md @@ -7,7 +7,7 @@ To run AFFiNE Desktop Client Application locally, run the following commands: ```sh # in repo root yarn install -yarn workspace @affine/native build +yarn affine @affine/native build yarn dev # in packages/frontend/apps/electron diff --git a/packages/frontend/apps/electron/package.json b/packages/frontend/apps/electron/package.json index 3596818e5f626..525dd44dd9a25 100644 --- a/packages/frontend/apps/electron/package.json +++ b/packages/frontend/apps/electron/package.json @@ -2,6 +2,7 @@ "name": "@affine/electron", "private": true, "version": "0.18.0", + "main": "./dist/main.js", "author": "toeverything", "repository": { "url": "https://github.com/toeverything/AFFiNE", @@ -11,19 +12,20 @@ "homepage": "https://github.com/toeverything/AFFiNE", "scripts": { "start": "electron .", - "dev": "DEV_SERVER_URL=http://localhost:8080 node --loader ts-node/esm/transpile-only ./scripts/dev.ts", - "dev:prod": "yarn node --loader ts-node/esm/transpile-only scripts/dev.ts", - "build": "NODE_ENV=production node --loader ts-node/esm/transpile-only scripts/build-layers.ts", - "build:dev": "NODE_ENV=development node --loader ts-node/esm/transpile-only scripts/build-layers.ts", - "generate-assets": "node --loader ts-node/esm/transpile-only scripts/generate-assets.ts", - "package": "cross-env NODE_OPTIONS=\"--loader ts-node/esm/transpile-only\" electron-forge package", - "make": "cross-env NODE_OPTIONS=\"--loader ts-node/esm/transpile-only\" electron-forge make", - "make-squirrel": "node --loader ts-node/esm/transpile-only scripts/make-squirrel.ts", - "make-nsis": "node --loader ts-node/esm/transpile-only scripts/make-nsis.ts" + "dev": "cross-env DEV_SERVER_URL=http://localhost:8080 node ./scripts/dev.ts", + "dev:prod": "node ./scripts/dev.ts", + "build": "cross-env NODE_ENV=production node ./scripts/build-layers.ts", + "build:dev": "node ./scripts/build-layers.ts", + "bundle": "affine bundle", + "generate-assets": "node ./scripts/generate-assets.ts", + "package": "electron-forge package", + "make": "electron-forge make", + "make-squirrel": "node ./scripts/make-squirrel.ts", + "make-nsis": "node ./scripts/make-nsis.ts" }, - "main": "./dist/main.js", "devDependencies": { "@affine-test/kit": "workspace:*", + "@affine-tools/utils": "workspace:*", "@affine/component": "workspace:*", "@affine/core": "workspace:*", "@affine/electron-api": "workspace:*", diff --git a/packages/frontend/apps/electron/scripts/common.ts b/packages/frontend/apps/electron/scripts/common.ts index ddc10995088f0..b96fd2856db8f 100644 --- a/packages/frontend/apps/electron/scripts/common.ts +++ b/packages/frontend/apps/electron/scripts/common.ts @@ -1,8 +1,8 @@ import { resolve } from 'node:path'; import { fileURLToPath } from 'node:url'; -// eslint-disable-next-line @typescript-eslint/no-restricted-imports -import { getBuildConfig } from '@affine/cli/src/webpack/runtime-config'; +import { getBuildConfig } from '@affine-tools/utils/build-config'; +import { Package } from '@affine-tools/utils/workspace'; import { sentryEsbuildPlugin } from '@sentry/esbuild-plugin'; import type { BuildOptions, Plugin } from 'esbuild'; @@ -24,12 +24,10 @@ export const config = (): BuildOptions => { 'process.env.NODE_ENV': process.env.NODE_ENV, REPLACE_ME_BUILD_ENV: process.env.BUILD_TYPE ?? 'stable', ...Object.entries( - getBuildConfig({ - channel: (process.env.BUILD_TYPE as any) ?? 'canary', - distribution: 'desktop', + getBuildConfig(new Package('@affine/electron'), { mode: process.env.NODE_ENV === 'production' ? 'production' : 'development', - static: false, + channel: (process.env.BUILD_TYPE as any) ?? 'canary', }) ).reduce( (def, [key, val]) => { diff --git a/packages/frontend/apps/electron/scripts/dev.ts b/packages/frontend/apps/electron/scripts/dev.ts index a2de51bd01d8d..718f22413579c 100644 --- a/packages/frontend/apps/electron/scripts/dev.ts +++ b/packages/frontend/apps/electron/scripts/dev.ts @@ -34,6 +34,7 @@ function spawnOrReloadElectron() { const ext = process.platform === 'win32' ? '.cmd' : ''; const exe = resolve(rootDir, 'node_modules', '.bin', `electron${ext}`); + delete process.env['NODE_OPTIONS']; spawnProcess = spawn(exe, ['.'], { cwd: electronDir, env: process.env, diff --git a/packages/frontend/apps/electron/tsconfig.json b/packages/frontend/apps/electron/tsconfig.json index df1043bd00f99..2b455fb9eaf05 100644 --- a/packages/frontend/apps/electron/tsconfig.json +++ b/packages/frontend/apps/electron/tsconfig.json @@ -33,6 +33,9 @@ }, { "path": "../../../../tests/kit" + }, + { + "path": "../../../../tools/utils" } ], "ts-node": { diff --git a/packages/frontend/apps/electron/tsconfig.node.json b/packages/frontend/apps/electron/tsconfig.node.json index 2beac5166efc1..2fcb612fd0800 100644 --- a/packages/frontend/apps/electron/tsconfig.node.json +++ b/packages/frontend/apps/electron/tsconfig.node.json @@ -5,7 +5,7 @@ "target": "ESNext", "module": "ESNext", "resolveJsonModule": true, - "moduleResolution": "Node", + "moduleResolution": "bundler", "allowSyntheticDefaultImports": true, "noEmit": false, "outDir": "./lib/scripts", @@ -17,5 +17,10 @@ "ts-node": { "esm": true, "experimentalSpecifierResolution": "node" - } + }, + "references": [ + { + "path": "../../../../tools/utils" + } + ] } diff --git a/packages/frontend/apps/electron/vitest.config.ts b/packages/frontend/apps/electron/vitest.config.ts index c3d33d8d0c94a..ced6065dae027 100644 --- a/packages/frontend/apps/electron/vitest.config.ts +++ b/packages/frontend/apps/electron/vitest.config.ts @@ -1,3 +1,4 @@ +// TODO(@forehalo): reuse '@affine-tools/utils' once it's ready to switch to esm module import { resolve } from 'node:path'; import { fileURLToPath } from 'node:url'; diff --git a/packages/frontend/apps/electron/webpack.config.js b/packages/frontend/apps/electron/webpack.config.js new file mode 100644 index 0000000000000..6af6e23233412 --- /dev/null +++ b/packages/frontend/apps/electron/webpack.config.js @@ -0,0 +1,12 @@ +// TODO(@forehalo): all packages would become 'module' +const path = require('node:path'); + +module.exports.config = { + entry: { + app: './renderer/index.tsx', + shell: './renderer/shell/index.tsx', + }, + output: { + path: path.resolve(__dirname, './renderer/dist'), + }, +}; diff --git a/packages/frontend/apps/ios/README.md b/packages/frontend/apps/ios/README.md index b8fcf843a26be..0fd8b5c5475c5 100644 --- a/packages/frontend/apps/ios/README.md +++ b/packages/frontend/apps/ios/README.md @@ -5,9 +5,9 @@ AFFiNE iOS app. ## Build - `yarn install` -- `BUILD_TYPE=canary PUBLIC_PATH="/" yarn workspace @affine/ios build` -- `yarn workspace @affine/ios cap sync` -- `yarn workspace @affine/ios cap open ios` +- `BUILD_TYPE=canary PUBLIC_PATH="/" yarn affine @affine/ios build` +- `yarn affine @affine/ios cap sync` +- `yarn affine @affine/ios cap open ios` ## Live Reload @@ -16,5 +16,5 @@ AFFiNE iOS app. - `yarn install` - `yarn dev` - select `ios` for the "Distribution" option -- `yarn workspace @affine/ios sync:dev` -- `yarn workspace @affine/ios cap open ios` +- `yarn affine @affine/ios sync:dev` +- `yarn affine @affine/ios cap open ios` diff --git a/packages/frontend/apps/ios/package.json b/packages/frontend/apps/ios/package.json index 192ca564b860e..b38796097f4cc 100644 --- a/packages/frontend/apps/ios/package.json +++ b/packages/frontend/apps/ios/package.json @@ -5,11 +5,10 @@ "private": true, "browser": "src/index.tsx", "scripts": { - "build": "cross-env DISTRIBUTION=ios yarn workspace @affine/cli bundle", - "dev": "yarn workspace @affine/cli dev", + "build": "affine bundle", + "dev": "affine bundle --dev", "sync": "yarn cap sync", - "sync:dev": "CAP_SERVER_URL=http://localhost:8080 yarn cap sync", - "static-server": "cross-env DISTRIBUTION=ios yarn workspace @affine/cli dev --static" + "sync:dev": "CAP_SERVER_URL=http://localhost:8080 yarn cap sync" }, "dependencies": { "@affine/component": "workspace:*", diff --git a/packages/frontend/apps/mobile/package.json b/packages/frontend/apps/mobile/package.json index 75cf6da660171..d60e8e58602fa 100644 --- a/packages/frontend/apps/mobile/package.json +++ b/packages/frontend/apps/mobile/package.json @@ -5,9 +5,8 @@ "private": true, "browser": "src/index.tsx", "scripts": { - "build": "cross-env DISTRIBUTION=mobile yarn workspace @affine/cli bundle", - "dev": "yarn workspace @affine/cli dev", - "static-server": "cross-env DISTRIBUTION=mobile yarn workspace @affine/cli dev --static" + "build": "affine bundle", + "dev": "affine bundle --dev" }, "dependencies": { "@affine/component": "workspace:*", diff --git a/packages/frontend/apps/web/package.json b/packages/frontend/apps/web/package.json index b3a995920f089..d60092b257f6e 100644 --- a/packages/frontend/apps/web/package.json +++ b/packages/frontend/apps/web/package.json @@ -5,9 +5,8 @@ "private": true, "browser": "src/index.tsx", "scripts": { - "build": "cross-env DISTRIBUTION=web yarn workspace @affine/cli bundle", - "dev": "yarn workspace @affine/cli dev", - "static-server": "yarn workspace @affine/cli dev --static" + "build": "affine bundle", + "dev": "affine bundle --dev" }, "dependencies": { "@affine/component": "workspace:*", diff --git a/packages/frontend/component/.storybook/main.ts b/packages/frontend/component/.storybook/main.ts index ac46638d1d16a..fc7474abde101 100644 --- a/packages/frontend/component/.storybook/main.ts +++ b/packages/frontend/component/.storybook/main.ts @@ -3,7 +3,7 @@ import { StorybookConfig } from '@storybook/react-vite'; import { vanillaExtractPlugin } from '@vanilla-extract/vite-plugin'; import swc from 'unplugin-swc'; import { mergeConfig } from 'vite'; -import { getBuildConfig } from '@affine/cli/src/webpack/runtime-config'; +import { getBuildConfig } from '@affine-tools/utils/build-config'; export default { stories: ['../src/ui/**/*.@(mdx|stories.@(js|jsx|ts|tsx))'], diff --git a/packages/frontend/component/package.json b/packages/frontend/component/package.json index 51680b150dcbb..ba665eab0b170 100644 --- a/packages/frontend/component/package.json +++ b/packages/frontend/component/package.json @@ -20,7 +20,6 @@ "react-dom": "^19.0.0" }, "dependencies": { - "@affine/cli": "workspace:*", "@affine/debug": "workspace:*", "@affine/electron-api": "workspace:*", "@affine/graphql": "workspace:*", diff --git a/packages/frontend/component/tsconfig.json b/packages/frontend/component/tsconfig.json index 3480a70c05b4f..d22b9c1cfdba1 100644 --- a/packages/frontend/component/tsconfig.json +++ b/packages/frontend/component/tsconfig.json @@ -1,7 +1,12 @@ { "extends": "../../../tsconfig.json", "exclude": ["lib"], - "include": ["./src/**/*", "./src/**/*.json", "./src/type.d.ts"], + "include": [ + "./src/**/*", + "./src/**/*.json", + "./src/type.d.ts", + "./.storybook" + ], "compilerOptions": { "composite": true, "noEmit": false, diff --git a/packages/frontend/core/public/robots.txt b/packages/frontend/core/public/robots.txt index 7b537c62f904c..66864824ccdb4 100644 --- a/packages/frontend/core/public/robots.txt +++ b/packages/frontend/core/public/robots.txt @@ -2,6 +2,7 @@ User-agent: * Allow: / +Allow: /api/workspaces/*/blobs/* Disallow: /oauth/* Disallow: /workspace/* Disallow: /public-workspace/* diff --git a/packages/frontend/core/tsconfig.json b/packages/frontend/core/tsconfig.json index 732b8e71fc7e4..5d766cd108328 100644 --- a/packages/frontend/core/tsconfig.json +++ b/packages/frontend/core/tsconfig.json @@ -28,12 +28,6 @@ }, { "path": "../../../blocksuite/affine/all" - }, - { - "path": "./tsconfig.node.json" - }, - { - "path": "./tsconfig.server.json" } ] } diff --git a/packages/frontend/core/tsconfig.node.json b/packages/frontend/core/tsconfig.node.json deleted file mode 100644 index acbcfbefb94c4..0000000000000 --- a/packages/frontend/core/tsconfig.node.json +++ /dev/null @@ -1,23 +0,0 @@ -{ - "extends": "../../../tsconfig.json", - "compilerOptions": { - "composite": true, - "target": "ESNext", - "module": "ESNext", - "verbatimModuleSyntax": false, - "resolveJsonModule": true, - "moduleResolution": "bundler", - "allowSyntheticDefaultImports": true, - "outDir": "./lib/.webpack", - "rootDir": "." - }, - "include": [".webpack/*.ts"], - "references": [ - { - "path": "../../../tools/cli" - }, - { - "path": "../../common/env" - } - ] -} diff --git a/packages/frontend/core/tsconfig.server.json b/packages/frontend/core/tsconfig.server.json deleted file mode 100644 index 6f4ae23fdf7a8..0000000000000 --- a/packages/frontend/core/tsconfig.server.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "extends": "../../../tsconfig.json", - "compilerOptions": { - "composite": true, - "target": "ESNext", - "module": "ESNext", - "moduleResolution": "Node", - "esModuleInterop": true, - "allowSyntheticDefaultImports": true, - "outDir": "./lib/server" - }, - "include": ["server.mts"] -} diff --git a/packages/frontend/graphql/codegen.yml b/packages/frontend/graphql/codegen.yml index 9ae652f4e6be8..c0ac93aa8e230 100644 --- a/packages/frontend/graphql/codegen.yml +++ b/packages/frontend/graphql/codegen.yml @@ -30,7 +30,5 @@ generates: plugins: - typescript - typescript-operations - - add: - content: '/* eslint-disable */' - ./export-gql-plugin.cjs: output: ./src/graphql/index.ts diff --git a/packages/frontend/graphql/package.json b/packages/frontend/graphql/package.json index 1aad8f3c8f8ee..706ceb2418375 100644 --- a/packages/frontend/graphql/package.json +++ b/packages/frontend/graphql/package.json @@ -21,7 +21,7 @@ "vitest": "2.1.8" }, "scripts": { - "postinstall": "gql-gen --errors-only" + "build": "gql-gen --errors-only" }, "dependencies": { "@affine/env": "workspace:*", diff --git a/packages/frontend/graphql/src/schema.ts b/packages/frontend/graphql/src/schema.ts index cc2b381f7ddee..3ca4e6af24ddc 100644 --- a/packages/frontend/graphql/src/schema.ts +++ b/packages/frontend/graphql/src/schema.ts @@ -1,4 +1,4 @@ - +/* oxlint-disable */ export type Maybe = T | null; export type InputMaybe = T | null; export type Exact = { diff --git a/packages/frontend/i18n/.i18n-codegen.json b/packages/frontend/i18n/.i18n-codegen.json index e92e4f8588c1f..140f1ef814f30 100644 --- a/packages/frontend/i18n/.i18n-codegen.json +++ b/packages/frontend/i18n/.i18n-codegen.json @@ -4,7 +4,7 @@ "list": [ { "input": "./src/resources/en.json", - "output": "./src/i18n-generated", + "output": "./src/i18n.gen", "parser": { "type": "i18next", "contextSeparator": "$", diff --git a/packages/frontend/i18n/build.mjs b/packages/frontend/i18n/build.mjs index a2bba9d0155ea..d63f59a6af0f5 100644 --- a/packages/frontend/i18n/build.mjs +++ b/packages/frontend/i18n/build.mjs @@ -39,7 +39,7 @@ function calcCompletenesses() { writeFileSync( join(pkgRoot, 'src', 'i18n-completenesses.json'), - JSON.stringify(completenesses, null, 2) + JSON.stringify(completenesses, null, 2) + '\n' ); } diff --git a/packages/frontend/i18n/src/i18n-completenesses.json b/packages/frontend/i18n/src/i18n-completenesses.json index 1c81c85c44084..0e369437bc3dd 100644 --- a/packages/frontend/i18n/src/i18n-completenesses.json +++ b/packages/frontend/i18n/src/i18n-completenesses.json @@ -21,4 +21,4 @@ "ur": 2, "zh-Hans": 91, "zh-Hant": 90 -} \ No newline at end of file +} diff --git a/packages/frontend/i18n/src/i18n.gen.ts b/packages/frontend/i18n/src/i18n.gen.ts new file mode 100644 index 0000000000000..d147baff700ad --- /dev/null +++ b/packages/frontend/i18n/src/i18n.gen.ts @@ -0,0 +1,6888 @@ +// @ts-nocheck +/* eslint-disable */ +import { createElement, useMemo, type ComponentType, type JSX } from "react"; +import { useTranslation, Trans, type TransProps } from "react-i18next"; +type TypedTransProps = Omit, "values" | "ns" | "i18nKey"> & ({} extends Value ? {} : { + values: Value; +}) & { + components: Components; +}; +function createProxy(initValue: (key: string) => any) { + function define(key: string) { + const value = initValue(key); + Object.defineProperty(container, key, { value, configurable: true }); + return value; + } + const container = { + __proto__: new Proxy({ __proto__: null }, { + get(_, key) { + if (typeof key === "symbol") + return undefined; + return define(key); + }, + }), + }; + return new Proxy(container, { + getPrototypeOf: () => null, + setPrototypeOf: (_, v) => v === null, + getOwnPropertyDescriptor: (_, key) => { + if (typeof key === "symbol") + return undefined; + if (!(key in container)) + define(key); + return Object.getOwnPropertyDescriptor(container, key); + }, + }); +} +export function useAFFiNEI18N(): { + /** + * `Back to my Content` + */ + ["404.back"](): string; + /** + * `Sorry, you do not have access or this content does not exist...` + */ + ["404.hint"](): string; + /** + * `Sign in to another account` + */ + ["404.signOut"](): string; + /** + * `AFFiNE Cloud` + */ + ["AFFiNE Cloud"](): string; + /** + * `All docs` + */ + ["All pages"](): string; + /** + * `App version` + */ + ["App Version"](): string; + /** + * `Available offline` + */ + ["Available Offline"](): string; + /** + * `Bold` + */ + Bold(): string; + /** + * `Cancel` + */ + Cancel(): string; + /** + * `Click to replace photo` + */ + ["Click to replace photo"](): string; + /** + * `Collections` + */ + Collections(): string; + /** + * `Complete` + */ + Complete(): string; + /** + * `Confirm` + */ + Confirm(): string; + /** + * `Continue` + */ + Continue(): string; + /** + * `Convert to ` + */ + ["Convert to "](): string; + /** + * `Copied link to clipboard` + */ + ["Copied link to clipboard"](): string; + /** + * `Copy` + */ + Copy(): string; + /** + * `Create` + */ + Create(): string; + /** + * `Created` + */ + Created(): string; + /** + * `Customise` + */ + Customize(): string; + /** + * `Colors` + */ + Colors(): string; + /** + * `Database file already loaded` + */ + DB_FILE_ALREADY_LOADED(): string; + /** + * `Invalid database file` + */ + DB_FILE_INVALID(): string; + /** + * `Database file migration failed` + */ + DB_FILE_MIGRATION_FAILED(): string; + /** + * `Database file path invalid` + */ + DB_FILE_PATH_INVALID(): string; + /** + * `Date` + */ + Date(): string; + /** + * `Delete` + */ + Delete(): string; + /** + * `Disable` + */ + Disable(): string; + /** + * `Disable public sharing` + */ + ["Disable Public Sharing"](): string; + /** + * `Disable snapshot` + */ + ["Disable Snapshot"](): string; + /** + * `Divider` + */ + Divider(): string; + /** + * `Edgeless` + */ + Edgeless(): string; + /** + * `Edit` + */ + Edit(): string; + /** + * `Editor version` + */ + ["Editor Version"](): string; + /** + * `Enable` + */ + Enable(): string; + /** + * `Enable AFFiNE Cloud` + */ + ["Enable AFFiNE Cloud"](): string; + /** + * `If enabled, the data in this workspace will be backed up and synchronised via AFFiNE Cloud.` + */ + ["Enable AFFiNE Cloud Description"](): string; + /** + * `The following functions rely on AFFiNE Cloud. All data is stored on the current device. You can enable AFFiNE Cloud for this workspace to keep data in sync with the cloud.` + */ + ["Enable cloud hint"](): string; + /** + * `Export` + */ + Export(): string; + /** + * `You can export the entire Workspace data for backup, and the exported data can be re-imported.` + */ + ["Export Description"](): string; + /** + * `Export failed` + */ + ["Export failed"](): string; + /** + * `Export success` + */ + ["Export success"](): string; + /** + * `Export to HTML` + */ + ["Export to HTML"](): string; + /** + * `Export to Markdown` + */ + ["Export to Markdown"](): string; + /** + * `Export to PNG` + */ + ["Export to PNG"](): string; + /** + * `File already exists` + */ + FILE_ALREADY_EXISTS(): string; + /** + * `Favourite` + */ + Favorite(): string; + /** + * `Favourited` + */ + Favorited(): string; + /** + * `Favourites` + */ + Favorites(): string; + /** + * `Feedback` + */ + Feedback(): string; + /** + * `Found 0 results` + */ + ["Find 0 result"](): string; + /** + * `Go back` + */ + ["Go Back"](): string; + /** + * `Go forward` + */ + ["Go Forward"](): string; + /** + * `Got it` + */ + ["Got it"](): string; + /** + * `Heading {{number}}` + */ + Heading(options: { + readonly number: string; + }): string; + /** + * `Image` + */ + Image(): string; + /** + * `Import` + */ + Import(): string; + /** + * `Info` + */ + Info(): string; + /** + * `Invitation sent` + */ + ["Invitation sent"](): string; + /** + * `Invited members have been notified with email to join this Workspace.` + */ + ["Invitation sent hint"](): string; + /** + * `Invite` + */ + Invite(): string; + /** + * `Invite members` + */ + ["Invite Members"](): string; + /** + * `Invited members will collaborate with you in current workspace` + */ + ["Invite Members Message"](): string; + /** + * `Joined workspace` + */ + ["Joined Workspace"](): string; + /** + * `Leave` + */ + Leave(): string; + /** + * `Hyperlink (with selected text)` + */ + Link(): string; + /** + * `Loading...` + */ + Loading(): string; + /** + * `Local` + */ + Local(): string; + /** + * `Member` + */ + Member(): string; + /** + * `Members` + */ + Members(): string; + /** + * `Manage members here, invite new member by email.` + */ + ["Members hint"](): string; + /** + * `New doc` + */ + ["New Page"](): string; + /** + * `Owner` + */ + Owner(): string; + /** + * `Page` + */ + Page(): string; + /** + * `Pen` + */ + Pen(): string; + /** + * `Pending` + */ + Pending(): string; + /** + * `Collaborator` + */ + Collaborator(): string; + /** + * `Under Review` + */ + ["Under-Review"](): string; + /** + * `Need More Seats` + */ + ["Need-More-Seats"](): string; + /** + * `Admin` + */ + Admin(): string; + /** + * `Publish` + */ + Publish(): string; + /** + * `Published to web` + */ + ["Published to Web"](): string; + /** + * `Quick search` + */ + ["Quick Search"](): string; + /** + * `Search` + */ + ["Quick search"](): string; + /** + * `Recent` + */ + Recent(): string; + /** + * `Remove from workspace` + */ + ["Remove from workspace"](): string; + /** + * `Remove photo` + */ + ["Remove photo"](): string; + /** + * `Remove special filter` + */ + ["Remove special filter"](): string; + /** + * `Removed successfully` + */ + ["Removed successfully"](): string; + /** + * `Rename` + */ + Rename(): string; + /** + * `Retry` + */ + Retry(): string; + /** + * `Save` + */ + Save(): string; + /** + * `Select` + */ + Select(): string; + /** + * `Sign in AFFiNE Cloud` + */ + ["Sign in"](): string; + /** + * `Sign in and enable` + */ + ["Sign in and Enable"](): string; + /** + * `Sign out` + */ + ["Sign out"](): string; + /** + * `Snapshot` + */ + Snapshot(): string; + /** + * `Storage` + */ + Storage(): string; + /** + * `Storage and export` + */ + ["Storage and Export"](): string; + /** + * `Successfully deleted` + */ + ["Successfully deleted"](): string; + /** + * `Successfully joined!` + */ + ["Successfully joined!"](): string; + /** + * `Switch` + */ + Switch(): string; + /** + * `Switch view` + */ + switchView(): string; + /** + * `Sync` + */ + Sync(): string; + /** + * `Synced with AFFiNE Cloud` + */ + ["Synced with AFFiNE Cloud"](): string; + /** + * `Tags` + */ + Tags(): string; + /** + * `Text` + */ + Text(): string; + /** + * `Theme` + */ + Theme(): string; + /** + * `Title` + */ + Title(): string; + /** + * `Trash` + */ + Trash(): string; + /** + * `Unknown error` + */ + UNKNOWN_ERROR(): string; + /** + * `Undo` + */ + Undo(): string; + /** + * `Unpin` + */ + Unpin(): string; + /** + * `Untitled` + */ + Untitled(): string; + /** + * `Update workspace name success` + */ + ["Update workspace name success"](): string; + /** + * `Updated` + */ + Updated(): string; + /** + * `Upload` + */ + Upload(): string; + /** + * `Users` + */ + Users(): string; + /** + * `Version` + */ + Version(): string; + /** + * `Visit workspace` + */ + ["Visit Workspace"](): string; + /** + * `Workspace name` + */ + ["Workspace Name"](): string; + /** + * `Workspace Owner` + */ + ["Workspace Owner"](): string; + /** + * `Workspace profile` + */ + ["Workspace Profile"](): string; + /** + * `Workspace settings` + */ + ["Workspace Settings"](): string; + /** + * `{{name}}'s settings` + */ + ["Workspace Settings with name"](options: { + readonly name: string; + }): string; + /** + * `{{name}} is saved locally` + */ + ["Workspace saved locally"](options: { + readonly name: string; + }): string; + /** + * `Zoom in` + */ + ["Zoom in"](): string; + /** + * `Zoom out` + */ + ["Zoom out"](): string; + /** + * `all` + */ + all(): string; + /** + * `Automatically check for new updates periodically.` + */ + ["com.affine.aboutAFFiNE.autoCheckUpdate.description"](): string; + /** + * `Check for updates automatically` + */ + ["com.affine.aboutAFFiNE.autoCheckUpdate.title"](): string; + /** + * `Automatically download updates (to this device).` + */ + ["com.affine.aboutAFFiNE.autoDownloadUpdate.description"](): string; + /** + * `Download updates automatically` + */ + ["com.affine.aboutAFFiNE.autoDownloadUpdate.title"](): string; + /** + * `View the AFFiNE Changelog.` + */ + ["com.affine.aboutAFFiNE.changelog.description"](): string; + /** + * `Discover what's new` + */ + ["com.affine.aboutAFFiNE.changelog.title"](): string; + /** + * `Check for update` + */ + ["com.affine.aboutAFFiNE.checkUpdate.button.check"](): string; + /** + * `Download update` + */ + ["com.affine.aboutAFFiNE.checkUpdate.button.download"](): string; + /** + * `Restart to update` + */ + ["com.affine.aboutAFFiNE.checkUpdate.button.restart"](): string; + /** + * `Retry` + */ + ["com.affine.aboutAFFiNE.checkUpdate.button.retry"](): string; + /** + * `New version is ready` + */ + ["com.affine.aboutAFFiNE.checkUpdate.description"](): string; + /** + * `Manually check for updates.` + */ + ["com.affine.aboutAFFiNE.checkUpdate.subtitle.check"](): string; + /** + * `Checking for updates...` + */ + ["com.affine.aboutAFFiNE.checkUpdate.subtitle.checking"](): string; + /** + * `Downloading the latest version...` + */ + ["com.affine.aboutAFFiNE.checkUpdate.subtitle.downloading"](): string; + /** + * `Unable to connect to the update server.` + */ + ["com.affine.aboutAFFiNE.checkUpdate.subtitle.error"](): string; + /** + * `You've got the latest version of AFFiNE.` + */ + ["com.affine.aboutAFFiNE.checkUpdate.subtitle.latest"](): string; + /** + * `Restart to apply update.` + */ + ["com.affine.aboutAFFiNE.checkUpdate.subtitle.restart"](): string; + /** + * `New update available ({{version}})` + */ + ["com.affine.aboutAFFiNE.checkUpdate.subtitle.update-available"](options: { + readonly version: string; + }): string; + /** + * `Check for updates` + */ + ["com.affine.aboutAFFiNE.checkUpdate.title"](): string; + /** + * `Communities` + */ + ["com.affine.aboutAFFiNE.community.title"](): string; + /** + * `AFFiNE community` + */ + ["com.affine.aboutAFFiNE.contact.community"](): string; + /** + * `Contact us` + */ + ["com.affine.aboutAFFiNE.contact.title"](): string; + /** + * `Official website` + */ + ["com.affine.aboutAFFiNE.contact.website"](): string; + /** + * `Privacy` + */ + ["com.affine.aboutAFFiNE.legal.privacy"](): string; + /** + * `Legal Info` + */ + ["com.affine.aboutAFFiNE.legal.title"](): string; + /** + * `Terms of use` + */ + ["com.affine.aboutAFFiNE.legal.tos"](): string; + /** + * `Information about AFFiNE` + */ + ["com.affine.aboutAFFiNE.subtitle"](): string; + /** + * `About AFFiNE` + */ + ["com.affine.aboutAFFiNE.title"](): string; + /** + * `App version` + */ + ["com.affine.aboutAFFiNE.version.app"](): string; + /** + * `Editor version` + */ + ["com.affine.aboutAFFiNE.version.editor.title"](): string; + /** + * `Version` + */ + ["com.affine.aboutAFFiNE.version.title"](): string; + /** + * `Get started` + */ + ["com.affine.ai-onboarding.edgeless.get-started"](): string; + /** + * `Lets you think bigger, create faster, work smarter and save time for every project.` + */ + ["com.affine.ai-onboarding.edgeless.message"](): string; + /** + * `Upgrade to unlimited usage` + */ + ["com.affine.ai-onboarding.edgeless.purchase"](): string; + /** + * `Right-clicking to select content AI` + */ + ["com.affine.ai-onboarding.edgeless.title"](): string; + /** + * `Lets you think bigger, create faster, work smarter and save time for every project.` + */ + ["com.affine.ai-onboarding.general.1.description"](): string; + /** + * `Meet AFFiNE AI` + */ + ["com.affine.ai-onboarding.general.1.title"](): string; + /** + * `Answer questions, draft docs, visualize ideas - AFFiNE AI can save you time at every possible step. Powered by GPT's most powerful model.` + */ + ["com.affine.ai-onboarding.general.2.description"](): string; + /** + * `Chat with AFFiNE AI` + */ + ["com.affine.ai-onboarding.general.2.title"](): string; + /** + * `Get insightful answer to any question, instantly.` + */ + ["com.affine.ai-onboarding.general.3.description"](): string; + /** + * `Edit inline with AFFiNE AI` + */ + ["com.affine.ai-onboarding.general.3.title"](): string; + /** + * `Expand thinking. Untangle complexity. Breakdown and visualise your content with crafted mindmap and presentable slides with one click.` + */ + ["com.affine.ai-onboarding.general.4.description"](): string; + /** + * `Make mind-map and presents with AI` + */ + ["com.affine.ai-onboarding.general.4.title"](): string; + /** + * `AFFiNE AI is ready` + */ + ["com.affine.ai-onboarding.general.5.title"](): string; + /** + * `Get started` + */ + ["com.affine.ai-onboarding.general.get-started"](): string; + /** + * `Next` + */ + ["com.affine.ai-onboarding.general.next"](): string; + /** + * `Back` + */ + ["com.affine.ai-onboarding.general.prev"](): string; + /** + * `Get unlimited usage` + */ + ["com.affine.ai-onboarding.general.purchase"](): string; + /** + * `Remind me later` + */ + ["com.affine.ai-onboarding.general.skip"](): string; + /** + * `Try for free` + */ + ["com.affine.ai-onboarding.general.try-for-free"](): string; + /** + * `Dismiss` + */ + ["com.affine.ai-onboarding.local.action-dismiss"](): string; + /** + * `Get started` + */ + ["com.affine.ai-onboarding.local.action-get-started"](): string; + /** + * `Learn more` + */ + ["com.affine.ai-onboarding.local.action-learn-more"](): string; + /** + * `Lets you think bigger, create faster, work smarter and save time for every project.` + */ + ["com.affine.ai-onboarding.local.message"](): string; + /** + * `Meet AFFiNE AI` + */ + ["com.affine.ai-onboarding.local.title"](): string; + /** + * `New` + */ + ["com.affine.ai-scroll-tip.tag"](): string; + /** + * `Meet AFFiNE AI` + */ + ["com.affine.ai-scroll-tip.title"](): string; + /** + * `View` + */ + ["com.affine.ai-scroll-tip.view"](): string; + /** + * `Please switch to edgeless mode` + */ + ["com.affine.ai.action.edgeless-only.dialog-title"](): string; + /** + * `Cancel` + */ + ["com.affine.ai.login-required.dialog-cancel"](): string; + /** + * `Sign in` + */ + ["com.affine.ai.login-required.dialog-confirm"](): string; + /** + * `To use AFFiNE AI, please sign in to your AFFiNE Cloud account.` + */ + ["com.affine.ai.login-required.dialog-content"](): string; + /** + * `Sign in to continue` + */ + ["com.affine.ai.login-required.dialog-title"](): string; + /** + * `Failed to insert template, please try again.` + */ + ["com.affine.ai.template-insert.failed"](): string; + /** + * `All docs` + */ + ["com.affine.all-pages.header"](): string; + /** + * `Learn more` + */ + ["com.affine.app-sidebar.learn-more"](): string; + /** + * `Star us` + */ + ["com.affine.app-sidebar.star-us"](): string; + /** + * `Download update` + */ + ["com.affine.appUpdater.downloadUpdate"](): string; + /** + * `Downloading` + */ + ["com.affine.appUpdater.downloading"](): string; + /** + * `Restart to install update` + */ + ["com.affine.appUpdater.installUpdate"](): string; + /** + * `Open download page` + */ + ["com.affine.appUpdater.openDownloadPage"](): string; + /** + * `Update available` + */ + ["com.affine.appUpdater.updateAvailable"](): string; + /** + * `Discover what's new!` + */ + ["com.affine.appUpdater.whatsNew"](): string; + /** + * `Customise the appearance of the client.` + */ + ["com.affine.appearanceSettings.clientBorder.description"](): string; + /** + * `Client border style` + */ + ["com.affine.appearanceSettings.clientBorder.title"](): string; + /** + * `Choose your colour mode` + */ + ["com.affine.appearanceSettings.color.description"](): string; + /** + * `Colour mode` + */ + ["com.affine.appearanceSettings.color.title"](): string; + /** + * `Edit all AFFiNE theme variables here` + */ + ["com.affine.appearanceSettings.customize-theme.description"](): string; + /** + * `Customize Theme` + */ + ["com.affine.appearanceSettings.customize-theme.title"](): string; + /** + * `Reset all` + */ + ["com.affine.appearanceSettings.customize-theme.reset"](): string; + /** + * `Open Theme Editor` + */ + ["com.affine.appearanceSettings.customize-theme.open"](): string; + /** + * `Choose your font style` + */ + ["com.affine.appearanceSettings.font.description"](): string; + /** + * `Font style` + */ + ["com.affine.appearanceSettings.font.title"](): string; + /** + * `Mono` + */ + ["com.affine.appearanceSettings.fontStyle.mono"](): string; + /** + * `Sans` + */ + ["com.affine.appearanceSettings.fontStyle.sans"](): string; + /** + * `Serif` + */ + ["com.affine.appearanceSettings.fontStyle.serif"](): string; + /** + * `Select the language for the interface.` + */ + ["com.affine.appearanceSettings.language.description"](): string; + /** + * `Display language` + */ + ["com.affine.appearanceSettings.language.title"](): string; + /** + * `Use background noise effect on the sidebar.` + */ + ["com.affine.appearanceSettings.noisyBackground.description"](): string; + /** + * `Noise background on the sidebar` + */ + ["com.affine.appearanceSettings.noisyBackground.title"](): string; + /** + * `Sidebar` + */ + ["com.affine.appearanceSettings.sidebar.title"](): string; + /** + * `Customise your AFFiNE appearance` + */ + ["com.affine.appearanceSettings.subtitle"](): string; + /** + * `Theme` + */ + ["com.affine.appearanceSettings.theme.title"](): string; + /** + * `Appearance settings` + */ + ["com.affine.appearanceSettings.title"](): string; + /** + * `Use transparency effect on the sidebar.` + */ + ["com.affine.appearanceSettings.translucentUI.description"](): string; + /** + * `Translucent UI on the sidebar` + */ + ["com.affine.appearanceSettings.translucentUI.title"](): string; + /** + * `Your current email is {{email}}. We'll send a temporary verification link to this email.` + */ + ["com.affine.auth.change.email.message"](options: { + readonly email: string; + }): string; + /** + * `Please enter your new email address below. We will send a verification link to this email address to complete the process.` + */ + ["com.affine.auth.change.email.page.subtitle"](): string; + /** + * `Congratulations! You have successfully updated the email address associated with your AFFiNE Cloud account.` + */ + ["com.affine.auth.change.email.page.success.subtitle"](): string; + /** + * `Email address updated!` + */ + ["com.affine.auth.change.email.page.success.title"](): string; + /** + * `Change email address` + */ + ["com.affine.auth.change.email.page.title"](): string; + /** + * `Forgot password` + */ + ["com.affine.auth.forget"](): string; + /** + * `Later` + */ + ["com.affine.auth.later"](): string; + /** + * `Open AFFiNE` + */ + ["com.affine.auth.open.affine"](): string; + /** + * `Download app` + */ + ["com.affine.auth.open.affine.download-app"](): string; + /** + * `Try again` + */ + ["com.affine.auth.open.affine.try-again"](): string; + /** + * `Still have problems?` + */ + ["com.affine.auth.open.affine.still-have-problems"](): string; + /** + * `Continue with Browser` + */ + ["com.affine.auth.open.affine.continue-with-browser"](): string; + /** + * `Download Latest Client` + */ + ["com.affine.auth.open.affine.download-latest-client"](): string; + /** + * `Open here instead` + */ + ["com.affine.auth.open.affine.doc.open-here"](): string; + /** + * `Edit settings` + */ + ["com.affine.auth.open.affine.doc.edit-settings"](): string; + /** + * `Requires AFFiNE desktop app version 0.18 or later.` + */ + ["com.affine.auth.open.affine.doc.footer-text"](): string; + /** + * `Please set a password of {{min}}-{{max}} characters with both letters and numbers to continue signing up with ` + */ + ["com.affine.auth.page.sent.email.subtitle"](options: Readonly<{ + min: string; + max: string; + }>): string; + /** + * `Welcome to AFFiNE Cloud, you are almost there!` + */ + ["com.affine.auth.page.sent.email.title"](): string; + /** + * `Password` + */ + ["com.affine.auth.password"](): string; + /** + * `Invalid password` + */ + ["com.affine.auth.password.error"](): string; + /** + * `Set password failed` + */ + ["com.affine.auth.password.set-failed"](): string; + /** + * `Reset password` + */ + ["com.affine.auth.reset.password"](): string; + /** + * `You will receive an email with a link to reset your password. Please check your inbox.` + */ + ["com.affine.auth.reset.password.message"](): string; + /** + * `Password reset successful` + */ + ["com.affine.auth.reset.password.page.success"](): string; + /** + * `Reset your AFFiNE Cloud password` + */ + ["com.affine.auth.reset.password.page.title"](): string; + /** + * `Send reset link` + */ + ["com.affine.auth.send.reset.password.link"](): string; + /** + * `Send set link` + */ + ["com.affine.auth.send.set.password.link"](): string; + /** + * `Send verification link` + */ + ["com.affine.auth.send.verify.email.hint"](): string; + /** + * `Sent` + */ + ["com.affine.auth.sent"](): string; + /** + * `The verification link failed to be sent, please try again later.` + */ + ["com.affine.auth.sent.change.email.fail"](): string; + /** + * `Verification link has been sent.` + */ + ["com.affine.auth.sent.change.email.hint"](): string; + /** + * `Reset password link has been sent.` + */ + ["com.affine.auth.sent.change.password.hint"](): string; + /** + * `Your password has upgraded! You can sign in AFFiNE Cloud with new password!` + */ + ["com.affine.auth.sent.reset.password.success.message"](): string; + /** + * `Set password link has been sent.` + */ + ["com.affine.auth.sent.set.password.hint"](): string; + /** + * `Your password has saved! You can sign in AFFiNE Cloud with email and password!` + */ + ["com.affine.auth.sent.set.password.success.message"](): string; + /** + * `Verification link has been sent.` + */ + ["com.affine.auth.sent.verify.email.hint"](): string; + /** + * `Save Email` + */ + ["com.affine.auth.set.email.save"](): string; + /** + * `Set password` + */ + ["com.affine.auth.set.password"](): string; + /** + * `Please set a password of {{min}}-{{max}} characters with both letters and numbers to continue signing up with ` + */ + ["com.affine.auth.set.password.message"](options: Readonly<{ + min: string; + max: string; + }>): string; + /** + * `Maximum {{max}} characters` + */ + ["com.affine.auth.set.password.message.maxlength"](options: { + readonly max: string; + }): string; + /** + * `Minimum {{min}} characters` + */ + ["com.affine.auth.set.password.message.minlength"](options: { + readonly min: string; + }): string; + /** + * `Password set successful` + */ + ["com.affine.auth.set.password.page.success"](): string; + /** + * `Set your AFFiNE Cloud password` + */ + ["com.affine.auth.set.password.page.title"](): string; + /** + * `Set a password at least {{min}} letters long` + */ + ["com.affine.auth.set.password.placeholder"](options: { + readonly min: string; + }): string; + /** + * `Confirm password` + */ + ["com.affine.auth.set.password.placeholder.confirm"](): string; + /** + * `Save password` + */ + ["com.affine.auth.set.password.save"](): string; + /** + * `Cancel` + */ + ["com.affine.auth.sign-out.confirm-modal.cancel"](): string; + /** + * `Sign Out` + */ + ["com.affine.auth.sign-out.confirm-modal.confirm"](): string; + /** + * `After signing out, the Cloud Workspaces associated with this account will be removed from the current device, and signing in again will add them back.` + */ + ["com.affine.auth.sign-out.confirm-modal.description"](): string; + /** + * `Sign out?` + */ + ["com.affine.auth.sign-out.confirm-modal.title"](): string; + /** + * `If you haven't received the email, please check your spam folder.` + */ + ["com.affine.auth.sign.auth.code.message"](): string; + /** + * `Resend link` + */ + ["com.affine.auth.sign.auth.code.resend.hint"](): string; + /** + * `Sign in with magic link` + */ + ["com.affine.auth.sign.auth.code.send-email.sign-in"](): string; + /** + * `Terms of conditions` + */ + ["com.affine.auth.sign.condition"](): string; + /** + * `Continue with email` + */ + ["com.affine.auth.sign.email.continue"](): string; + /** + * `Invalid email` + */ + ["com.affine.auth.sign.email.error"](): string; + /** + * `Enter your email address` + */ + ["com.affine.auth.sign.email.placeholder"](): string; + /** + * `Sign in` + */ + ["com.affine.auth.sign.in"](): string; + /** + * `Confirm your email` + */ + ["com.affine.auth.sign.in.sent.email.subtitle"](): string; + /** + * `Connect to a Self-Hosted Instance` + */ + ["com.affine.auth.sign.add-selfhosted"](): string; + /** + * `Server URL` + */ + ["com.affine.auth.sign.add-selfhosted.baseurl"](): string; + /** + * `Connect` + */ + ["com.affine.auth.sign.add-selfhosted.connect-button"](): string; + /** + * `Unable to connect to the server.` + */ + ["com.affine.auth.sign.add-selfhosted.error"](): string; + /** + * `Privacy policy` + */ + ["com.affine.auth.sign.policy"](): string; + /** + * ` You can click the link to create an account automatically.` + */ + ["com.affine.auth.sign.sent.email.message.end"](): string; + /** + * `You can click the link to sign in automatically.` + */ + ["com.affine.auth.sign.sent.email.message.sent-tips.sign-in"](): string; + /** + * `You can click the link to create an account automatically.` + */ + ["com.affine.auth.sign.sent.email.message.sent-tips.sign-up"](): string; + /** + * `Sign up` + */ + ["com.affine.auth.sign.up"](): string; + /** + * `Create your account` + */ + ["com.affine.auth.sign.up.sent.email.subtitle"](): string; + /** + * `The app will automatically open or redirect to the web version. If you encounter any issues, you can also click the button below to manually open the AFFiNE app.` + */ + ["com.affine.auth.sign.up.success.subtitle"](): string; + /** + * `Your account has been created and you're now signed in!` + */ + ["com.affine.auth.sign.up.success.title"](): string; + /** + * `You have successfully signed in. The app will automatically open or redirect to the web version. if you encounter any issues, you can also click the button below to manually open the AFFiNE app.` + */ + ["com.affine.auth.signed.success.subtitle"](): string; + /** + * `You're almost there!` + */ + ["com.affine.auth.signed.success.title"](): string; + /** + * `Server error, please try again later.` + */ + ["com.affine.auth.toast.message.failed"](): string; + /** + * `You have been signed in, start to sync your data with AFFiNE Cloud!` + */ + ["com.affine.auth.toast.message.signed-in"](): string; + /** + * `Unable to sign in` + */ + ["com.affine.auth.toast.title.failed"](): string; + /** + * `Signed in` + */ + ["com.affine.auth.toast.title.signed-in"](): string; + /** + * `Your current email is {{email}}. We'll send a temporary verification link to this email.` + */ + ["com.affine.auth.verify.email.message"](options: { + readonly email: string; + }): string; + /** + * `Back` + */ + ["com.affine.backButton"](): string; + /** + * `Your local data is stored in the browser and may be lost. Don't risk it - enable cloud now!` + */ + ["com.affine.banner.local-warning"](): string; + /** + * `AFFiNE Cloud` + */ + ["com.affine.brand.affineCloud"](): string; + /** + * `Jan,Feb,Mar,Apr,May,Jun,Jul,Aug,Sep,Oct,Nov,Dec` + */ + ["com.affine.calendar-date-picker.month-names"](): string; + /** + * `Today` + */ + ["com.affine.calendar-date-picker.today"](): string; + /** + * `Su,Mo,Tu,We,Th,Fr,Sa` + */ + ["com.affine.calendar-date-picker.week-days"](): string; + /** + * `Host by AFFiNE.Pro, Save, sync, and backup all your data.` + */ + ["com.affine.cloud-scroll-tip.caption"](): string; + /** + * `AFFiNE Cloud` + */ + ["com.affine.cloud-scroll-tip.title"](): string; + /** + * `Collections` + */ + ["com.affine.cmdk.affine.category.affine.collections"](): string; + /** + * `Create` + */ + ["com.affine.cmdk.affine.category.affine.creation"](): string; + /** + * `Edgeless` + */ + ["com.affine.cmdk.affine.category.affine.edgeless"](): string; + /** + * `General` + */ + ["com.affine.cmdk.affine.category.affine.general"](): string; + /** + * `Help` + */ + ["com.affine.cmdk.affine.category.affine.help"](): string; + /** + * `Layout controls` + */ + ["com.affine.cmdk.affine.category.affine.layout"](): string; + /** + * `Navigation` + */ + ["com.affine.cmdk.affine.category.affine.navigation"](): string; + /** + * `Docs` + */ + ["com.affine.cmdk.affine.category.affine.pages"](): string; + /** + * `Recent` + */ + ["com.affine.cmdk.affine.category.affine.recent"](): string; + /** + * `Settings` + */ + ["com.affine.cmdk.affine.category.affine.settings"](): string; + /** + * `Tags` + */ + ["com.affine.cmdk.affine.category.affine.tags"](): string; + /** + * `Updates` + */ + ["com.affine.cmdk.affine.category.affine.updates"](): string; + /** + * `Edgeless commands` + */ + ["com.affine.cmdk.affine.category.editor.edgeless"](): string; + /** + * `Insert object` + */ + ["com.affine.cmdk.affine.category.editor.insert-object"](): string; + /** + * `Doc Commands` + */ + ["com.affine.cmdk.affine.category.editor.page"](): string; + /** + * `Results` + */ + ["com.affine.cmdk.affine.category.results"](): string; + /** + * `Change client border style to` + */ + ["com.affine.cmdk.affine.client-border-style.to"](): string; + /** + * `Change colour mode to` + */ + ["com.affine.cmdk.affine.color-mode.to"](): string; + /** + * `Contact us` + */ + ["com.affine.cmdk.affine.contact-us"](): string; + /** + * `Create "{{keyWord}}" doc and insert` + */ + ["com.affine.cmdk.affine.create-new-doc-and-insert"](options: { + readonly keyWord: string; + }): string; + /** + * `New "{{keyWord}}" edgeless` + */ + ["com.affine.cmdk.affine.create-new-edgeless-as"](options: { + readonly keyWord: string; + }): string; + /** + * `New "{{keyWord}}" page` + */ + ["com.affine.cmdk.affine.create-new-page-as"](options: { + readonly keyWord: string; + }): string; + /** + * `Change display language to` + */ + ["com.affine.cmdk.affine.display-language.to"](): string; + /** + * `Add to favourites` + */ + ["com.affine.cmdk.affine.editor.add-to-favourites"](): string; + /** + * `Start presentation` + */ + ["com.affine.cmdk.affine.editor.edgeless.presentation-start"](): string; + /** + * `Remove from favourites` + */ + ["com.affine.cmdk.affine.editor.remove-from-favourites"](): string; + /** + * `Restore from trash` + */ + ["com.affine.cmdk.affine.editor.restore-from-trash"](): string; + /** + * `Reveal doc history modal` + */ + ["com.affine.cmdk.affine.editor.reveal-page-history-modal"](): string; + /** + * `This doc has been moved to the trash, you can either restore or permanently delete it.` + */ + ["com.affine.cmdk.affine.editor.trash-footer-hint"](): string; + /** + * `Change font style to` + */ + ["com.affine.cmdk.affine.font-style.to"](): string; + /** + * `Change full width layout to` + */ + ["com.affine.cmdk.affine.full-width-layout.to"](): string; + /** + * `Change default width for new pages in to standard` + */ + ["com.affine.cmdk.affine.default-page-width-layout.standard"](): string; + /** + * `Change default width for new pages in to full width` + */ + ["com.affine.cmdk.affine.default-page-width-layout.full-width"](): string; + /** + * `Change current page width to standard` + */ + ["com.affine.cmdk.affine.current-page-width-layout.standard"](): string; + /** + * `Change current page width to full width` + */ + ["com.affine.cmdk.affine.current-page-width-layout.full-width"](): string; + /** + * `Getting started` + */ + ["com.affine.cmdk.affine.getting-started"](): string; + /** + * `Import workspace` + */ + ["com.affine.cmdk.affine.import-workspace"](): string; + /** + * `Insert this link to the current doc` + */ + ["com.affine.cmdk.affine.insert-link"](): string; + /** + * `Collapse left sidebar` + */ + ["com.affine.cmdk.affine.left-sidebar.collapse"](): string; + /** + * `Expand left sidebar` + */ + ["com.affine.cmdk.affine.left-sidebar.expand"](): string; + /** + * `Go to all docs` + */ + ["com.affine.cmdk.affine.navigation.goto-all-pages"](): string; + /** + * `Go to edgeless list` + */ + ["com.affine.cmdk.affine.navigation.goto-edgeless-list"](): string; + /** + * `Go to page list` + */ + ["com.affine.cmdk.affine.navigation.goto-page-list"](): string; + /** + * `Go to trash` + */ + ["com.affine.cmdk.affine.navigation.goto-trash"](): string; + /** + * `Go to workspace` + */ + ["com.affine.cmdk.affine.navigation.goto-workspace"](): string; + /** + * `Go to account settings` + */ + ["com.affine.cmdk.affine.navigation.open-account-settings"](): string; + /** + * `Go to Settings` + */ + ["com.affine.cmdk.affine.navigation.open-settings"](): string; + /** + * `New edgeless` + */ + ["com.affine.cmdk.affine.new-edgeless-page"](): string; + /** + * `New page` + */ + ["com.affine.cmdk.affine.new-page"](): string; + /** + * `New workspace` + */ + ["com.affine.cmdk.affine.new-workspace"](): string; + /** + * `Change noise background on the sidebar to` + */ + ["com.affine.cmdk.affine.noise-background-on-the-sidebar.to"](): string; + /** + * `Restart to upgrade` + */ + ["com.affine.cmdk.affine.restart-to-upgrade"](): string; + /** + * `OFF` + */ + ["com.affine.cmdk.affine.switch-state.off"](): string; + /** + * `ON` + */ + ["com.affine.cmdk.affine.switch-state.on"](): string; + /** + * `Change translucent UI on the sidebar to` + */ + ["com.affine.cmdk.affine.translucent-ui-on-the-sidebar.to"](): string; + /** + * `What's new` + */ + ["com.affine.cmdk.affine.whats-new"](): string; + /** + * `Search docs or paste link...` + */ + ["com.affine.cmdk.docs.placeholder"](): string; + /** + * `Insert links` + */ + ["com.affine.cmdk.insert-links"](): string; + /** + * `No results found` + */ + ["com.affine.cmdk.no-results"](): string; + /** + * `No results found for` + */ + ["com.affine.cmdk.no-results-for"](): string; + /** + * `Type a command or search anything...` + */ + ["com.affine.cmdk.placeholder"](): string; + /** + * `Switch to $t(com.affine.edgelessMode)` + */ + ["com.affine.cmdk.switch-to-edgeless"](): string; + /** + * `Switch to $t(com.affine.pageMode)` + */ + ["com.affine.cmdk.switch-to-page"](): string; + /** + * `Delete` + */ + ["com.affine.collection-bar.action.tooltip.delete"](): string; + /** + * `Edit` + */ + ["com.affine.collection-bar.action.tooltip.edit"](): string; + /** + * `Pin to sidebar` + */ + ["com.affine.collection-bar.action.tooltip.pin"](): string; + /** + * `Unpin` + */ + ["com.affine.collection-bar.action.tooltip.unpin"](): string; + /** + * `Do you want to add a document to the current collection? If it is filtered based on rules, this will add a set of included rules.` + */ + ["com.affine.collection.add-doc.confirm.description"](): string; + /** + * `Add new doc to this collection` + */ + ["com.affine.collection.add-doc.confirm.title"](): string; + /** + * `Doc already exists` + */ + ["com.affine.collection.addPage.alreadyExists"](): string; + /** + * `Added successfully` + */ + ["com.affine.collection.addPage.success"](): string; + /** + * `Add docs` + */ + ["com.affine.collection.addPages"](): string; + /** + * `Add rules` + */ + ["com.affine.collection.addRules"](): string; + /** + * `All collections` + */ + ["com.affine.collection.allCollections"](): string; + /** + * `Empty collection` + */ + ["com.affine.collection.emptyCollection"](): string; + /** + * `Collection is a smart folder where you can manually add docs or automatically add docs through rules.` + */ + ["com.affine.collection.emptyCollectionDescription"](): string; + /** + * `HELP INFO` + */ + ["com.affine.collection.helpInfo"](): string; + /** + * `Edit collection` + */ + ["com.affine.collection.menu.edit"](): string; + /** + * `Rename` + */ + ["com.affine.collection.menu.rename"](): string; + /** + * `Removed successfully` + */ + ["com.affine.collection.removePage.success"](): string; + /** + * `No collections` + */ + ["com.affine.collections.empty.message"](): string; + /** + * `New collection` + */ + ["com.affine.collections.empty.new-collection-button"](): string; + /** + * `Collections` + */ + ["com.affine.collections.header"](): string; + /** + * `Couldn't copy image` + */ + ["com.affine.copy.asImage.notAvailable.title"](): string; + /** + * `The 'Copy as image' feature is only available on our desktop app. Please download and install the client to access this feature.` + */ + ["com.affine.copy.asImage.notAvailable.message"](): string; + /** + * `Download Client` + */ + ["com.affine.copy.asImage.notAvailable.action"](): string; + /** + * `Image copied` + */ + ["com.affine.copy.asImage.success"](): string; + /** + * `Image copy failed` + */ + ["com.affine.copy.asImage.failed"](): string; + /** + * `Cancel` + */ + ["com.affine.confirmModal.button.cancel"](): string; + /** + * `Current year` + */ + ["com.affine.currentYear"](): string; + /** + * `Deleting {{count}} tags cannot be undone, please proceed with caution.` + */ + ["com.affine.delete-tags.confirm.multi-tag-description"](options: { + readonly count: string; + }): string; + /** + * `Delete tag?` + */ + ["com.affine.delete-tags.confirm.title"](): string; + /** + * `{{count}} tag deleted` + + * - com.affine.delete-tags.count_one: `{{count}} tag deleted` + + * - com.affine.delete-tags.count_other: `{{count}} tags deleted` + */ + ["com.affine.delete-tags.count"](options: { + readonly count: string | number | bigint; + }): string; + /** + * `{{count}} tag deleted` + */ + ["com.affine.delete-tags.count_one"](options: { + readonly count: string | number | bigint; + }): string; + /** + * `{{count}} tags deleted` + */ + ["com.affine.delete-tags.count_other"](options: { + readonly count: string | number | bigint; + }): string; + /** + * `Delete workspace from this device and optionally delete all data.` + */ + ["com.affine.deleteLeaveWorkspace.description"](): string; + /** + * `Leave workspace` + */ + ["com.affine.deleteLeaveWorkspace.leave"](): string; + /** + * `After you leave, you will not be able to access content within this workspace.` + */ + ["com.affine.deleteLeaveWorkspace.leaveDescription"](): string; + /** + * `Docs` + */ + ["com.affine.docs.header"](): string; + /** + * `Draw with a blank whiteboard` + */ + ["com.affine.draw_with_a_blank_whiteboard"](): string; + /** + * `Earlier` + */ + ["com.affine.earlier"](): string; + /** + * `Edgeless mode` + */ + ["com.affine.edgelessMode"](): string; + /** + * `Cancel` + */ + ["com.affine.editCollection.button.cancel"](): string; + /** + * `Create` + */ + ["com.affine.editCollection.button.create"](): string; + /** + * `Create collection` + */ + ["com.affine.editCollection.createCollection"](): string; + /** + * `Filters` + */ + ["com.affine.editCollection.filters"](): string; + /** + * `Docs` + */ + ["com.affine.editCollection.pages"](): string; + /** + * `Clear selected` + */ + ["com.affine.editCollection.pages.clear"](): string; + /** + * `Rename collection` + */ + ["com.affine.editCollection.renameCollection"](): string; + /** + * `Rules` + */ + ["com.affine.editCollection.rules"](): string; + /** + * `No results` + */ + ["com.affine.editCollection.rules.empty.noResults"](): string; + /** + * `No docs meet the filtering rules` + */ + ["com.affine.editCollection.rules.empty.noResults.tips"](): string; + /** + * `No rules` + */ + ["com.affine.editCollection.rules.empty.noRules"](): string; + /** + * `Add selected doc` + */ + ["com.affine.editCollection.rules.include.add"](): string; + /** + * `is` + */ + ["com.affine.editCollection.rules.include.is"](): string; + /** + * `Doc` + */ + ["com.affine.editCollection.rules.include.page"](): string; + /** + * `“Selected docs” refers to manually adding docs rather than automatically adding them through rule matching. You can manually add docs through the “Add selected docs” option or by dragging and dropping.` + */ + ["com.affine.editCollection.rules.include.tips"](): string; + /** + * `What is "Selected docs"?` + */ + ["com.affine.editCollection.rules.include.tipsTitle"](): string; + /** + * `Selected docs` + */ + ["com.affine.editCollection.rules.include.title"](): string; + /** + * `Preview` + */ + ["com.affine.editCollection.rules.preview"](): string; + /** + * `Reset` + */ + ["com.affine.editCollection.rules.reset"](): string; + /** + * `automatically` + */ + ["com.affine.editCollection.rules.tips.highlight"](): string; + /** + * `Save` + */ + ["com.affine.editCollection.save"](): string; + /** + * `Save as new collection` + */ + ["com.affine.editCollection.saveCollection"](): string; + /** + * `Search doc...` + */ + ["com.affine.editCollection.search.placeholder"](): string; + /** + * `Untitled collection` + */ + ["com.affine.editCollection.untitledCollection"](): string; + /** + * `Update collection` + */ + ["com.affine.editCollection.updateCollection"](): string; + /** + * `Collection is a smart folder where you can manually add docs or automatically add docs through rules.` + */ + ["com.affine.editCollectionName.createTips"](): string; + /** + * `Name` + */ + ["com.affine.editCollectionName.name"](): string; + /** + * `Collection name` + */ + ["com.affine.editCollectionName.name.placeholder"](): string; + /** + * `Default to Edgeless mode` + */ + ["com.affine.editorDefaultMode.edgeless"](): string; + /** + * `Default to Page mode` + */ + ["com.affine.editorDefaultMode.page"](): string; + /** + * `Add docs` + */ + ["com.affine.empty.collection-detail.action.add-doc"](): string; + /** + * `Add rules` + */ + ["com.affine.empty.collection-detail.action.add-rule"](): string; + /** + * `Collection is a smart folder where you can manually add docs or automatically add docs through rules.` + */ + ["com.affine.empty.collection-detail.description"](): string; + /** + * `Empty collection` + */ + ["com.affine.empty.collection-detail.title"](): string; + /** + * `Add collection` + */ + ["com.affine.empty.collections.action.new-collection"](): string; + /** + * `Create your first collection here.` + */ + ["com.affine.empty.collections.description"](): string; + /** + * `Collection management` + */ + ["com.affine.empty.collections.title"](): string; + /** + * `New doc` + */ + ["com.affine.empty.docs.action.new-doc"](): string; + /** + * `Create your first doc here.` + */ + ["com.affine.empty.docs.all-description"](): string; + /** + * `Docs management` + */ + ["com.affine.empty.docs.title"](): string; + /** + * `Deleted docs will appear here.` + */ + ["com.affine.empty.docs.trash-description"](): string; + /** + * `Create a new tag for your documents.` + */ + ["com.affine.empty.tags.description"](): string; + /** + * `Tag management` + */ + ["com.affine.empty.tags.title"](): string; + /** + * `There's no doc here yet` + */ + ["com.affine.emptyDesc"](): string; + /** + * `Cancel` + */ + ["com.affine.enableAffineCloudModal.button.cancel"](): string; + /** + * `Enable Cloud for {{workspaceName}}` + */ + ["com.affine.enableAffineCloudModal.custom-server.title"](options: { + readonly workspaceName: string; + }): string; + /** + * `Choose an instance.` + */ + ["com.affine.enableAffineCloudModal.custom-server.description"](): string; + /** + * `Enable Cloud` + */ + ["com.affine.enableAffineCloudModal.custom-server.enable"](): string; + /** + * `Hide error` + */ + ["com.affine.error.hide-error"](): string; + /** + * `Doc content is missing` + */ + ["com.affine.error.no-page-root.title"](): string; + /** + * `Refetch` + */ + ["com.affine.error.refetch"](): string; + /** + * `Reload AFFiNE` + */ + ["com.affine.error.reload"](): string; + /** + * `Refresh` + */ + ["com.affine.error.retry"](): string; + /** + * `Something is wrong...` + */ + ["com.affine.error.unexpected-error.title"](): string; + /** + * `Please request a new reset password link.` + */ + ["com.affine.expired.page.subtitle"](): string; + /** + * `Please request a new link.` + */ + ["com.affine.expired.page.new-subtitle"](): string; + /** + * `This link has expired...` + */ + ["com.affine.expired.page.title"](): string; + /** + * `Please try it again later.` + */ + ["com.affine.export.error.message"](): string; + /** + * `Export failed due to an unexpected error` + */ + ["com.affine.export.error.title"](): string; + /** + * `Print` + */ + ["com.affine.export.print"](): string; + /** + * `Please open the download folder to check.` + */ + ["com.affine.export.success.message"](): string; + /** + * `Exported successfully` + */ + ["com.affine.export.success.title"](): string; + /** + * `Add to favourites` + */ + ["com.affine.favoritePageOperation.add"](): string; + /** + * `Remove from favourites` + */ + ["com.affine.favoritePageOperation.remove"](): string; + /** + * `Filter` + */ + ["com.affine.filter"](): string; + /** + * `after` + */ + ["com.affine.filter.after"](): string; + /** + * `before` + */ + ["com.affine.filter.before"](): string; + /** + * `contains all` + */ + ["com.affine.filter.contains all"](): string; + /** + * `contains one of` + */ + ["com.affine.filter.contains one of"](): string; + /** + * `does not contains all` + */ + ["com.affine.filter.does not contains all"](): string; + /** + * `does not contains one of` + */ + ["com.affine.filter.does not contains one of"](): string; + /** + * `Empty` + */ + ["com.affine.filter.empty-tag"](): string; + /** + * `false` + */ + ["com.affine.filter.false"](): string; + /** + * `is` + */ + ["com.affine.filter.is"](): string; + /** + * `is empty` + */ + ["com.affine.filter.is empty"](): string; + /** + * `is not empty` + */ + ["com.affine.filter.is not empty"](): string; + /** + * `Favourited` + */ + ["com.affine.filter.is-favourited"](): string; + /** + * `Shared` + */ + ["com.affine.filter.is-public"](): string; + /** + * `last` + */ + ["com.affine.filter.last"](): string; + /** + * `Save view` + */ + ["com.affine.filter.save-view"](): string; + /** + * `true` + */ + ["com.affine.filter.true"](): string; + /** + * `Add filter` + */ + ["com.affine.filterList.button.add"](): string; + /** + * `Add tag` + */ + ["com.affine.header.option.add-tag"](): string; + /** + * `Duplicate` + */ + ["com.affine.header.option.duplicate"](): string; + /** + * `Open in desktop app` + */ + ["com.affine.header.option.open-in-desktop"](): string; + /** + * `View all frames` + */ + ["com.affine.header.option.view-frame"](): string; + /** + * `View table of contents` + */ + ["com.affine.header.option.view-toc"](): string; + /** + * `Contact us` + */ + ["com.affine.helpIsland.contactUs"](): string; + /** + * `Getting started` + */ + ["com.affine.helpIsland.gettingStarted"](): string; + /** + * `Help and feedback` + */ + ["com.affine.helpIsland.helpAndFeedback"](): string; + /** + * `Cancel` + */ + ["com.affine.history-vision.tips-modal.cancel"](): string; + /** + * `Enable AFFiNE Cloud` + */ + ["com.affine.history-vision.tips-modal.confirm"](): string; + /** + * `The current workspace is a local workspace, and we do not support version history for it at the moment. You can enable AFFiNE Cloud. This will sync the workspace with the Cloud, allowing you to use this feature.` + */ + ["com.affine.history-vision.tips-modal.description"](): string; + /** + * `History vision needs AFFiNE Cloud` + */ + ["com.affine.history-vision.tips-modal.title"](): string; + /** + * `Back to doc` + */ + ["com.affine.history.back-to-page"](): string; + /** + * `You are about to restore the current version of the doc to the latest version available. This action will overwrite any changes made prior to the latest version.` + */ + ["com.affine.history.confirm-restore-modal.hint"](): string; + /** + * `Load more` + */ + ["com.affine.history.confirm-restore-modal.load-more"](): string; + /** + * `LIMITED DOC HISTORY` + */ + ["com.affine.history.confirm-restore-modal.plan-prompt.limited-title"](): string; + /** + * `HELP INFO` + */ + ["com.affine.history.confirm-restore-modal.plan-prompt.title"](): string; + /** + * `Upgrade` + */ + ["com.affine.history.confirm-restore-modal.pro-plan-prompt.upgrade"](): string; + /** + * `Restore` + */ + ["com.affine.history.confirm-restore-modal.restore"](): string; + /** + * `This document is such a spring chicken, it hasn't sprouted a single historical sprig yet!` + */ + ["com.affine.history.empty-prompt.description"](): string; + /** + * `Empty` + */ + ["com.affine.history.empty-prompt.title"](): string; + /** + * `Restore current version` + */ + ["com.affine.history.restore-current-version"](): string; + /** + * `Version history` + */ + ["com.affine.history.version-history"](): string; + /** + * `View history version` + */ + ["com.affine.history.view-history-version"](): string; + /** + * `Create into a New Workspace` + */ + ["com.affine.import-template.dialog.createDocToNewWorkspace"](): string; + /** + * `Create doc to "{{workspace}}"` + */ + ["com.affine.import-template.dialog.createDocToWorkspace"](options: { + readonly workspace: string; + }): string; + /** + * `Create doc with "{{templateName}}" template` + */ + ["com.affine.import-template.dialog.createDocWithTemplate"](options: { + readonly templateName: string; + }): string; + /** + * `Failed to import template, please try again.` + */ + ["com.affine.import-template.dialog.errorImport"](): string; + /** + * `Failed to load template, please try again.` + */ + ["com.affine.import-template.dialog.errorLoad"](): string; + /** + * `Support Markdown/Notion` + */ + ["com.affine.import_file"](): string; + /** + * `AFFiNE workspace data` + */ + ["com.affine.import.affine-workspace-data"](): string; + /** + * `HTML` + */ + ["com.affine.import.html-files"](): string; + /** + * `This is an experimental feature that is not perfect and may cause your data to be missing after import.` + */ + ["com.affine.import.html-files.tooltip"](): string; + /** + * `Markdown files (.md)` + */ + ["com.affine.import.markdown-files"](): string; + /** + * `Markdown with media files (.zip)` + */ + ["com.affine.import.markdown-with-media-files"](): string; + /** + * `Please upload a markdown zip file with attachments, experimental function, there may be data loss.` + */ + ["com.affine.import.markdown-with-media-files.tooltip"](): string; + /** + * `If you'd like to request support for additional file types, feel free to let us know on` + */ + ["com.affine.import.modal.tip"](): string; + /** + * `Notion` + */ + ["com.affine.import.notion"](): string; + /** + * `Import your Notion data. Supported import formats: HTML with subpages.` + */ + ["com.affine.import.notion.tooltip"](): string; + /** + * `Snapshot` + */ + ["com.affine.import.snapshot"](): string; + /** + * `Import failed, please try again.` + */ + ["com.affine.import.status.failed.message"](): string; + /** + * `No file selected` + */ + ["com.affine.import.status.failed.message.no-file-selected"](): string; + /** + * `Import failure` + */ + ["com.affine.import.status.failed.title"](): string; + /** + * `Importing your workspace data, please wait patiently.` + */ + ["com.affine.import.status.importing.message"](): string; + /** + * `Importing...` + */ + ["com.affine.import.status.importing.title"](): string; + /** + * `Your document has been imported successfully, thank you for choosing AFFiNE. Any questions please feel free to feedback to us` + */ + ["com.affine.import.status.success.message"](): string; + /** + * `Import completed` + */ + ["com.affine.import.status.success.title"](): string; + /** + * `Cancel` + */ + ["com.affine.inviteModal.button.cancel"](): string; + /** + * `Maybe later` + */ + ["com.affine.issue-feedback.cancel"](): string; + /** + * `Create issue on GitHub` + */ + ["com.affine.issue-feedback.confirm"](): string; + /** + * `Got feedback? We're all ears! Create an issue on GitHub to let us know your thoughts and suggestions` + */ + ["com.affine.issue-feedback.description"](): string; + /** + * `Share your feedback on GitHub` + */ + ["com.affine.issue-feedback.title"](): string; + /** + * `Journals` + */ + ["com.affine.journal.app-sidebar-title"](): string; + /** + * `{{count}} more articles` + */ + ["com.affine.journal.conflict-show-more"](options: { + readonly count: string; + }): string; + /** + * `Created` + */ + ["com.affine.journal.created-today"](): string; + /** + * `You haven't created anything yet` + */ + ["com.affine.journal.daily-count-created-empty-tips"](): string; + /** + * `You haven't updated anything yet` + */ + ["com.affine.journal.daily-count-updated-empty-tips"](): string; + /** + * `Updated` + */ + ["com.affine.journal.updated-today"](): string; + /** + * `Just now` + */ + ["com.affine.just-now"](): string; + /** + * `Append to daily note` + */ + ["com.affine.keyboardShortcuts.appendDailyNote"](): string; + /** + * `Body text` + */ + ["com.affine.keyboardShortcuts.bodyText"](): string; + /** + * `Bold` + */ + ["com.affine.keyboardShortcuts.bold"](): string; + /** + * `Cancel` + */ + ["com.affine.keyboardShortcuts.cancel"](): string; + /** + * `Code block` + */ + ["com.affine.keyboardShortcuts.codeBlock"](): string; + /** + * `Copy private link` + */ + ["com.affine.keyboardShortcuts.copy-private-link"](): string; + /** + * `Connector` + */ + ["com.affine.keyboardShortcuts.connector"](): string; + /** + * `Divider` + */ + ["com.affine.keyboardShortcuts.divider"](): string; + /** + * `Expand/collapse sidebar` + */ + ["com.affine.keyboardShortcuts.expandOrCollapseSidebar"](): string; + /** + * `Go back` + */ + ["com.affine.keyboardShortcuts.goBack"](): string; + /** + * `Go forward` + */ + ["com.affine.keyboardShortcuts.goForward"](): string; + /** + * `Group` + */ + ["com.affine.keyboardShortcuts.group"](): string; + /** + * `Group as database` + */ + ["com.affine.keyboardShortcuts.groupDatabase"](): string; + /** + * `Hand` + */ + ["com.affine.keyboardShortcuts.hand"](): string; + /** + * `Heading {{number}}` + */ + ["com.affine.keyboardShortcuts.heading"](options: { + readonly number: string; + }): string; + /** + * `Image` + */ + ["com.affine.keyboardShortcuts.image"](): string; + /** + * `Increase indent` + */ + ["com.affine.keyboardShortcuts.increaseIndent"](): string; + /** + * `Inline code` + */ + ["com.affine.keyboardShortcuts.inlineCode"](): string; + /** + * `Italic` + */ + ["com.affine.keyboardShortcuts.italic"](): string; + /** + * `Hyperlink (with selected text)` + */ + ["com.affine.keyboardShortcuts.link"](): string; + /** + * `Move down` + */ + ["com.affine.keyboardShortcuts.moveDown"](): string; + /** + * `Move up` + */ + ["com.affine.keyboardShortcuts.moveUp"](): string; + /** + * `New doc` + */ + ["com.affine.keyboardShortcuts.newPage"](): string; + /** + * `Note` + */ + ["com.affine.keyboardShortcuts.note"](): string; + /** + * `Pen` + */ + ["com.affine.keyboardShortcuts.pen"](): string; + /** + * `Quick search` + */ + ["com.affine.keyboardShortcuts.quickSearch"](): string; + /** + * `Redo` + */ + ["com.affine.keyboardShortcuts.redo"](): string; + /** + * `Reduce indent` + */ + ["com.affine.keyboardShortcuts.reduceIndent"](): string; + /** + * `Select` + */ + ["com.affine.keyboardShortcuts.select"](): string; + /** + * `Select all` + */ + ["com.affine.keyboardShortcuts.selectAll"](): string; + /** + * `Shape` + */ + ["com.affine.keyboardShortcuts.shape"](): string; + /** + * `Strikethrough` + */ + ["com.affine.keyboardShortcuts.strikethrough"](): string; + /** + * `Check keyboard shortcuts quickly` + */ + ["com.affine.keyboardShortcuts.subtitle"](): string; + /** + * `Switch view` + */ + ["com.affine.keyboardShortcuts.switch"](): string; + /** + * `Text` + */ + ["com.affine.keyboardShortcuts.text"](): string; + /** + * `Keyboard shortcuts` + */ + ["com.affine.keyboardShortcuts.title"](): string; + /** + * `Ungroup` + */ + ["com.affine.keyboardShortcuts.unGroup"](): string; + /** + * `Underline` + */ + ["com.affine.keyboardShortcuts.underline"](): string; + /** + * `Undo` + */ + ["com.affine.keyboardShortcuts.undo"](): string; + /** + * `Zoom in` + */ + ["com.affine.keyboardShortcuts.zoomIn"](): string; + /** + * `Zoom out` + */ + ["com.affine.keyboardShortcuts.zoomOut"](): string; + /** + * `Zoom to 100%` + */ + ["com.affine.keyboardShortcuts.zoomTo100"](): string; + /** + * `Zoom to fit` + */ + ["com.affine.keyboardShortcuts.zoomToFit"](): string; + /** + * `Last 30 days` + */ + ["com.affine.last30Days"](): string; + /** + * `Last 7 days` + */ + ["com.affine.last7Days"](): string; + /** + * `Last month` + */ + ["com.affine.lastMonth"](): string; + /** + * `Last week` + */ + ["com.affine.lastWeek"](): string; + /** + * `Last year` + */ + ["com.affine.lastYear"](): string; + /** + * `Loading...` + */ + ["com.affine.loading"](): string; + /** + * `Rename` + */ + ["com.affine.menu.rename"](): string; + /** + * `No results found` + */ + ["com.affine.mobile.search.empty"](): string; + /** + * `App version` + */ + ["com.affine.mobile.setting.about.appVersion"](): string; + /** + * `Editor version` + */ + ["com.affine.mobile.setting.about.editorVersion"](): string; + /** + * `About` + */ + ["com.affine.mobile.setting.about.title"](): string; + /** + * `Font style` + */ + ["com.affine.mobile.setting.appearance.font"](): string; + /** + * `Display language` + */ + ["com.affine.mobile.setting.appearance.language"](): string; + /** + * `Color mode` + */ + ["com.affine.mobile.setting.appearance.theme"](): string; + /** + * `Appearance` + */ + ["com.affine.mobile.setting.appearance.title"](): string; + /** + * `Settings` + */ + ["com.affine.mobile.setting.header-title"](): string; + /** + * `Star us on GitHub` + */ + ["com.affine.mobile.setting.others.github"](): string; + /** + * `Privacy` + */ + ["com.affine.mobile.setting.others.privacy"](): string; + /** + * `Terms of use` + */ + ["com.affine.mobile.setting.others.terms"](): string; + /** + * `Privacy & others` + */ + ["com.affine.mobile.setting.others.title"](): string; + /** + * `Official website` + */ + ["com.affine.mobile.setting.others.website"](): string; + /** + * `Want to keep data local?` + */ + ["com.affine.mobile.sign-in.skip.hint"](): string; + /** + * `Start AFFiNE without an account` + */ + ["com.affine.mobile.sign-in.skip.link"](): string; + /** + * `Older than a month` + */ + ["com.affine.moreThan30Days"](): string; + /** + * `Cancel` + */ + ["com.affine.moveToTrash.confirmModal.cancel"](): string; + /** + * `Delete` + */ + ["com.affine.moveToTrash.confirmModal.confirm"](): string; + /** + * `{{title}} will be moved to trash` + */ + ["com.affine.moveToTrash.confirmModal.description"](options: { + readonly title: string; + }): string; + /** + * `{{ number }} docs will be moved to Trash` + */ + ["com.affine.moveToTrash.confirmModal.description.multiple"](options: { + readonly number: string; + }): string; + /** + * `Delete doc?` + */ + ["com.affine.moveToTrash.confirmModal.title"](): string; + /** + * `Delete {{ number }} docs?` + */ + ["com.affine.moveToTrash.confirmModal.title.multiple"](options: { + readonly number: string; + }): string; + /** + * `Move to trash` + */ + ["com.affine.moveToTrash.title"](): string; + /** + * `New tab` + */ + ["com.affine.multi-tab.new-tab"](): string; + /** + * `Enabling AFFiNE Cloud allows you to synchronise and backup data, as well as support multi-user collaboration and content publishing.` + */ + ["com.affine.nameWorkspace.affine-cloud.description"](): string; + /** + * `Sync across devices with AFFiNE Cloud` + */ + ["com.affine.nameWorkspace.affine-cloud.title"](): string; + /** + * `If you want the workspace to be stored locally, you can download the desktop client.` + */ + ["com.affine.nameWorkspace.affine-cloud.web-tips"](): string; + /** + * `Cancel` + */ + ["com.affine.nameWorkspace.button.cancel"](): string; + /** + * `Create` + */ + ["com.affine.nameWorkspace.button.create"](): string; + /** + * `A workspace is your virtual space to capture, create and plan as just one person or together as a team.` + */ + ["com.affine.nameWorkspace.description"](): string; + /** + * `Set a workspace name` + */ + ["com.affine.nameWorkspace.placeholder"](): string; + /** + * `Workspace name` + */ + ["com.affine.nameWorkspace.subtitle.workspace-name"](): string; + /** + * `Name your workspace` + */ + ["com.affine.nameWorkspace.title"](): string; + /** + * `New page` + */ + ["com.affine.new.page-mode"](): string; + /** + * `New edgeless` + */ + ["com.affine.new_edgeless"](): string; + /** + * `Import` + */ + ["com.affine.new_import"](): string; + /** + * `Next week` + */ + ["com.affine.nextWeek"](): string; + /** + * `Back home` + */ + ["com.affine.notFoundPage.backButton"](): string; + /** + * `Page not found` + */ + ["com.affine.notFoundPage.title"](): string; + /** + * `AFFiNE Community` + */ + ["com.affine.other-page.nav.affine-community"](): string; + /** + * `Blog` + */ + ["com.affine.other-page.nav.blog"](): string; + /** + * `Contact us` + */ + ["com.affine.other-page.nav.contact-us"](): string; + /** + * `Download app` + */ + ["com.affine.other-page.nav.download-app"](): string; + /** + * `Official website` + */ + ["com.affine.other-page.nav.official-website"](): string; + /** + * `Open AFFiNE` + */ + ["com.affine.other-page.nav.open-affine"](): string; + /** + * `Add linked doc` + */ + ["com.affine.page-operation.add-linked-page"](): string; + /** + * `{{ count }} more properties` + */ + ["com.affine.page-properties.more-property.more"](options: { + readonly count: string; + }): string; + /** + * `{{ count }} more property` + */ + ["com.affine.page-properties.more-property.one"](options: { + readonly count: string; + }): string; + /** + * `hide {{ count }} property` + */ + ["com.affine.page-properties.hide-property.one"](options: { + readonly count: string; + }): string; + /** + * `hide {{ count }} properties` + */ + ["com.affine.page-properties.hide-property.more"](options: { + readonly count: string; + }): string; + /** + * `Add property` + */ + ["com.affine.page-properties.add-property"](): string; + /** + * `Create property` + */ + ["com.affine.page-properties.add-property.menu.create"](): string; + /** + * `Properties` + */ + ["com.affine.page-properties.add-property.menu.header"](): string; + /** + * `Config properties` + */ + ["com.affine.page-properties.config-properties"](): string; + /** + * `Backlinks` + */ + ["com.affine.page-properties.backlinks"](): string; + /** + * `Type` + */ + ["com.affine.page-properties.create-property.menu.header"](): string; + /** + * `Added` + */ + ["com.affine.page-properties.create-property.added"](): string; + /** + * `Icons` + */ + ["com.affine.page-properties.icons"](): string; + /** + * `Local user` + */ + ["com.affine.page-properties.local-user"](): string; + /** + * `Outgoing links` + */ + ["com.affine.page-properties.outgoing-links"](): string; + /** + * `Info` + */ + ["com.affine.page-properties.page-info"](): string; + /** + * `View Info` + */ + ["com.affine.page-properties.page-info.view"](): string; + /** + * `No Record` + */ + ["com.affine.page-properties.property-user-avatar-no-record"](): string; + /** + * `Local User` + */ + ["com.affine.page-properties.property-user-local"](): string; + /** + * `Empty` + */ + ["com.affine.page-properties.property-value-placeholder"](): string; + /** + * `Always hide` + */ + ["com.affine.page-properties.property.always-hide"](): string; + /** + * `Always show` + */ + ["com.affine.page-properties.property.always-show"](): string; + /** + * `Checkbox` + */ + ["com.affine.page-properties.property.checkbox"](): string; + /** + * `Created by` + */ + ["com.affine.page-properties.property.createdBy"](): string; + /** + * `Date` + */ + ["com.affine.page-properties.property.date"](): string; + /** + * `Hide in view` + */ + ["com.affine.page-properties.property.hide-in-view"](): string; + /** + * `Hide in view when empty` + */ + ["com.affine.page-properties.property.hide-in-view-when-empty"](): string; + /** + * `Hide when empty` + */ + ["com.affine.page-properties.property.hide-when-empty"](): string; + /** + * `Number` + */ + ["com.affine.page-properties.property.number"](): string; + /** + * `Progress` + */ + ["com.affine.page-properties.property.progress"](): string; + /** + * `Remove property` + */ + ["com.affine.page-properties.property.remove-property"](): string; + /** + * `Required` + */ + ["com.affine.page-properties.property.required"](): string; + /** + * `Show in view` + */ + ["com.affine.page-properties.property.show-in-view"](): string; + /** + * `Tags` + */ + ["com.affine.page-properties.property.tags"](): string; + /** + * `Doc mode` + */ + ["com.affine.page-properties.property.docPrimaryMode"](): string; + /** + * `Text` + */ + ["com.affine.page-properties.property.text"](): string; + /** + * `Journal` + */ + ["com.affine.page-properties.property.journal"](): string; + /** + * `Duplicated` + */ + ["com.affine.page-properties.property.journal-duplicated"](): string; + /** + * `Remove journal mark` + */ + ["com.affine.page-properties.property.journal-remove"](): string; + /** + * `Last edited by` + */ + ["com.affine.page-properties.property.updatedBy"](): string; + /** + * `Created` + */ + ["com.affine.page-properties.property.createdAt"](): string; + /** + * `Updated` + */ + ["com.affine.page-properties.property.updatedAt"](): string; + /** + * `Edgeless theme` + */ + ["com.affine.page-properties.property.edgelessTheme"](): string; + /** + * `Page width` + */ + ["com.affine.page-properties.property.pageWidth"](): string; + /** + * `Add relevant identifiers or categories to the doc. Useful for organizing content, improving searchability, and grouping related docs together.` + */ + ["com.affine.page-properties.property.tags.tooltips"](): string; + /** + * `Indicates that this doc is a journal entry or daily note. Facilitates easy capture of ideas, quick logging of thoughts, and ongoing personal reflection.` + */ + ["com.affine.page-properties.property.journal.tooltips"](): string; + /** + * `Use a checkbox to indicate whether a condition is true or false. Useful for confirming options, toggling features, or tracking task states.` + */ + ["com.affine.page-properties.property.checkbox.tooltips"](): string; + /** + * `Use a date field to select or display a specific date. Useful for scheduling, setting deadlines, or recording important events.` + */ + ["com.affine.page-properties.property.date.tooltips"](): string; + /** + * `Upload images to display or manage them. Useful for showcasing visual content, adding illustrations, or organizing a gallery.` + */ + ["com.affine.page-properties.property.image.tooltips"](): string; + /** + * `Select one or more options. Useful for categorizing items, filtering data, or managing tags.` + */ + ["com.affine.page-properties.property.multiSelect.tooltips"](): string; + /** + * `Enter a numeric value. Useful for quantities, measurements, or ranking items.` + */ + ["com.affine.page-properties.property.number.tooltips"](): string; + /** + * `Set a progress value between 0 and 100. Useful for tracking completion status, visualizing progress, or managing goals.` + */ + ["com.affine.page-properties.property.progress.tooltips"](): string; + /** + * `Choose one option. Useful for selecting a single preference, categorizing items, or making decisions.` + */ + ["com.affine.page-properties.property.select.tooltips"](): string; + /** + * `Enter a link to websites or AFFiNE docs. Useful for connecting to external resources and referencing internal docs.` + */ + ["com.affine.page-properties.property.link.tooltips"](): string; + /** + * `Enter text. Useful for descriptions, comments, notes, or any other free-form text input.` + */ + ["com.affine.page-properties.property.text.tooltips"](): string; + /** + * `Displays the author of the current doc. Useful for tracking doc ownership, accountability, and collaboration.` + */ + ["com.affine.page-properties.property.createdBy.tooltips"](): string; + /** + * `Displays the last editor of the current doc. Useful for tracking recent changes.` + */ + ["com.affine.page-properties.property.updatedBy.tooltips"](): string; + /** + * `Record the last modification timestamp. Useful for tracking changes, identifying recent updates, or monitoring content freshness.` + */ + ["com.affine.page-properties.property.updatedAt.tooltips"](): string; + /** + * `Track when a doc was first created. Useful for maintaining record history, sorting by creation date, or auditing content chronologically.` + */ + ["com.affine.page-properties.property.createdAt.tooltips"](): string; + /** + * `Select the doc mode from Page Mode, Edgeless Mode, or Auto. Useful for choosing the best display for your content.` + */ + ["com.affine.page-properties.property.docPrimaryMode.tooltips"](): string; + /** + * `Select the doc theme from Light, Dark, or System. Useful for precise control over content viewing style.` + */ + ["com.affine.page-properties.property.edgelessTheme.tooltips"](): string; + /** + * `Control the width of this page to fit content display needs.` + */ + ["com.affine.page-properties.property.pageWidth.tooltips"](): string; + /** + * `Properties` + */ + ["com.affine.propertySidebar.property-list.section"](): string; + /** + * `Add more properties` + */ + ["com.affine.propertySidebar.add-more.section"](): string; + /** + * `customize properties` + */ + ["com.affine.page-properties.settings.title"](): string; + /** + * `Open tag page` + */ + ["com.affine.page-properties.tags.open-tags-page"](): string; + /** + * `Select tag or create one` + */ + ["com.affine.page-properties.tags.selector-header-title"](): string; + /** + * `Display` + */ + ["com.affine.page.display"](): string; + /** + * `Display properties` + */ + ["com.affine.page.display.display-properties"](): string; + /** + * `Body notes` + */ + ["com.affine.page.display.display-properties.body-notes"](): string; + /** + * `Grouping` + */ + ["com.affine.page.display.grouping"](): string; + /** + * `Favourites` + */ + ["com.affine.page.display.grouping.group-by-favourites"](): string; + /** + * `Tag` + */ + ["com.affine.page.display.grouping.group-by-tag"](): string; + /** + * `Untagged` + */ + ["com.affine.page.display.grouping.group-by-tag.untagged"](): string; + /** + * `No grouping` + */ + ["com.affine.page.display.grouping.no-grouping"](): string; + /** + * `List option` + */ + ["com.affine.page.display.list-option"](): string; + /** + * `Clear selection` + */ + ["com.affine.page.group-header.clear"](): string; + /** + * `Favourited` + */ + ["com.affine.page.group-header.favourited"](): string; + /** + * `Not favourited` + */ + ["com.affine.page.group-header.not-favourited"](): string; + /** + * `Select all` + */ + ["com.affine.page.group-header.select-all"](): string; + /** + * `Doc mode` + */ + ["com.affine.pageMode"](): string; + /** + * `all` + */ + ["com.affine.pageMode.all"](): string; + /** + * `Edgeless` + */ + ["com.affine.pageMode.edgeless"](): string; + /** + * `Page` + */ + ["com.affine.pageMode.page"](): string; + /** + * `Congratulations on your successful purchase of AFFiNE AI! You're now empowered to refine your content, generate images, and craft comprehensive mindmaps directly within AFFiNE AI, dramatically enhancing your productivity.` + */ + ["com.affine.payment.ai-upgrade-success-page.text"](): string; + /** + * `Purchase successful!` + */ + ["com.affine.payment.ai-upgrade-success-page.title"](): string; + /** + * `Cancel subscription` + */ + ["com.affine.payment.ai.action.cancel.button-label"](): string; + /** + * `Keep AFFiNE AI` + */ + ["com.affine.payment.ai.action.cancel.confirm.cancel-text"](): string; + /** + * `Cancel subscription` + */ + ["com.affine.payment.ai.action.cancel.confirm.confirm-text"](): string; + /** + * `If you end your subscription now, you can still use AFFiNE AI until the end of this billing period.` + */ + ["com.affine.payment.ai.action.cancel.confirm.description"](): string; + /** + * `Cancel subscription` + */ + ["com.affine.payment.ai.action.cancel.confirm.title"](): string; + /** + * `Login` + */ + ["com.affine.payment.ai.action.login.button-label"](): string; + /** + * `Resume` + */ + ["com.affine.payment.ai.action.resume.button-label"](): string; + /** + * `Cancel` + */ + ["com.affine.payment.ai.action.resume.confirm.cancel-text"](): string; + /** + * `Confirm` + */ + ["com.affine.payment.ai.action.resume.confirm.confirm-text"](): string; + /** + * `Are you sure you want to resume the subscription for AFFiNE AI? This means your payment method will be charged automatically at the end of each billing cycle, starting from the next billing cycle.` + */ + ["com.affine.payment.ai.action.resume.confirm.description"](): string; + /** + * `You will be charged in the next billing cycle.` + */ + ["com.affine.payment.ai.action.resume.confirm.notify.msg"](): string; + /** + * `Subscription updated` + */ + ["com.affine.payment.ai.action.resume.confirm.notify.title"](): string; + /** + * `Resume auto-renewal?` + */ + ["com.affine.payment.ai.action.resume.confirm.title"](): string; + /** + * `Write with you` + */ + ["com.affine.payment.ai.benefit.g1"](): string; + /** + * `Create quality content from sentences to articles on topics you need` + */ + ["com.affine.payment.ai.benefit.g1-1"](): string; + /** + * `Rewrite like the professionals` + */ + ["com.affine.payment.ai.benefit.g1-2"](): string; + /** + * `Change the tones / fix spelling & grammar` + */ + ["com.affine.payment.ai.benefit.g1-3"](): string; + /** + * `Draw with you` + */ + ["com.affine.payment.ai.benefit.g2"](): string; + /** + * `Visualize your mind, magically` + */ + ["com.affine.payment.ai.benefit.g2-1"](): string; + /** + * `Turn your outline into beautiful, engaging presentations` + */ + ["com.affine.payment.ai.benefit.g2-2"](): string; + /** + * `Summarize your content into structured mind-map` + */ + ["com.affine.payment.ai.benefit.g2-3"](): string; + /** + * `Plan with you` + */ + ["com.affine.payment.ai.benefit.g3"](): string; + /** + * `Memorize and tidy up your knowledge` + */ + ["com.affine.payment.ai.benefit.g3-1"](): string; + /** + * `Auto-sorting and auto-tagging` + */ + ["com.affine.payment.ai.benefit.g3-2"](): string; + /** + * `Open source & Privacy ensured` + */ + ["com.affine.payment.ai.benefit.g3-3"](): string; + /** + * `You have purchased AFFiNE AI. The expiration date is {{end}}.` + */ + ["com.affine.payment.ai.billing-tip.end-at"](options: { + readonly end: string; + }): string; + /** + * `You have purchased AFFiNE AI. The next payment date is {{due}}.` + */ + ["com.affine.payment.ai.billing-tip.next-bill-at"](options: { + readonly due: string; + }): string; + /** + * `You are currently on the Free plan.` + */ + ["com.affine.payment.ai.pricing-plan.caption-free"](): string; + /** + * `You have purchased AFFiNE AI` + */ + ["com.affine.payment.ai.pricing-plan.caption-purchased"](): string; + /** + * `Learn about AFFiNE AI` + */ + ["com.affine.payment.ai.pricing-plan.learn"](): string; + /** + * `AFFiNE AI` + */ + ["com.affine.payment.ai.pricing-plan.title"](): string; + /** + * `Turn all your ideas into reality` + */ + ["com.affine.payment.ai.pricing-plan.title-caption-1"](): string; + /** + * `A true multimodal AI copilot.` + */ + ["com.affine.payment.ai.pricing-plan.title-caption-2"](): string; + /** + * `Billed annually` + */ + ["com.affine.payment.ai.subscribe.billed-annually"](): string; + /** + * `You have purchased AFFiNE AI.` + */ + ["com.affine.payment.ai.usage-description-purchased"](): string; + /** + * `AFFiNE AI usage` + */ + ["com.affine.payment.ai.usage-title"](): string; + /** + * `Change plan` + */ + ["com.affine.payment.ai.usage.change-button-label"](): string; + /** + * `Purchase` + */ + ["com.affine.payment.ai.usage.purchase-button-label"](): string; + /** + * `Times used` + */ + ["com.affine.payment.ai.usage.used-caption"](): string; + /** + * `{{used}}/{{limit}} times` + */ + ["com.affine.payment.ai.usage.used-detail"](options: Readonly<{ + used: string; + limit: string; + }>): string; + /** + * `Unlimited local workspaces` + */ + ["com.affine.payment.benefit-1"](): string; + /** + * `Unlimited login devices` + */ + ["com.affine.payment.benefit-2"](): string; + /** + * `Unlimited blocks` + */ + ["com.affine.payment.benefit-3"](): string; + /** + * `{{capacity}} of cloud storage` + */ + ["com.affine.payment.benefit-4"](options: { + readonly capacity: string; + }): string; + /** + * `{{capacity}} of maximum file size` + */ + ["com.affine.payment.benefit-5"](options: { + readonly capacity: string; + }): string; + /** + * `Number of members per workspace ≤ {{capacity}}` + */ + ["com.affine.payment.benefit-6"](options: { + readonly capacity: string; + }): string; + /** + * `{{capacity}}-days version history` + */ + ["com.affine.payment.benefit-7"](options: { + readonly capacity: string; + }): string; + /** + * `AFFiNE AI` + */ + ["com.affine.payment.billing-setting.ai-plan"](): string; + /** + * `Purchase` + */ + ["com.affine.payment.billing-setting.ai.purchase"](): string; + /** + * `One-time payment` + */ + ["com.affine.payment.billing-setting.believer.price-caption"](): string; + /** + * `AFFiNE Cloud` + */ + ["com.affine.payment.billing-setting.believer.title"](): string; + /** + * `Cancel subscription` + */ + ["com.affine.payment.billing-setting.cancel-subscription"](): string; + /** + * `Once you canceled subscription you will no longer enjoy the plan benefits.` + */ + ["com.affine.payment.billing-setting.cancel-subscription.description"](): string; + /** + * `Change plan` + */ + ["com.affine.payment.billing-setting.change-plan"](): string; + /** + * `AFFiNE Cloud` + */ + ["com.affine.payment.billing-setting.current-plan"](): string; + /** + * `Expiration date` + */ + ["com.affine.payment.billing-setting.expiration-date"](): string; + /** + * `Your subscription is valid until {{expirationDate}}` + */ + ["com.affine.payment.billing-setting.expiration-date.description"](options: { + readonly expirationDate: string; + }): string; + /** + * `Billing history` + */ + ["com.affine.payment.billing-setting.history"](): string; + /** + * `Information` + */ + ["com.affine.payment.billing-setting.information"](): string; + /** + * `month` + */ + ["com.affine.payment.billing-setting.month"](): string; + /** + * `There are no invoices to display.` + */ + ["com.affine.payment.billing-setting.no-invoice"](): string; + /** + * `Paid` + */ + ["com.affine.payment.billing-setting.paid"](): string; + /** + * `Manage payment details` + */ + ["com.affine.payment.billing-setting.payment-method"](): string; + /** + * `View future and past invoices, update billing information, and change payment methods. Provided by Stripe.` + */ + ["com.affine.payment.billing-setting.payment-method.description"](): string; + /** + * `Go` + */ + ["com.affine.payment.billing-setting.payment-method.go"](): string; + /** + * `Renew date` + */ + ["com.affine.payment.billing-setting.renew-date"](): string; + /** + * `Next billing date: {{renewDate}}` + */ + ["com.affine.payment.billing-setting.renew-date.description"](options: { + readonly renewDate: string; + }): string; + /** + * `Due date` + */ + ["com.affine.payment.billing-setting.due-date"](): string; + /** + * `Your subscription will end on {{dueDate}}` + */ + ["com.affine.payment.billing-setting.due-date.description"](options: { + readonly dueDate: string; + }): string; + /** + * `Resume` + */ + ["com.affine.payment.billing-setting.resume-subscription"](): string; + /** + * `Manage your billing information and invoices.` + */ + ["com.affine.payment.billing-setting.subtitle"](): string; + /** + * `Billing` + */ + ["com.affine.payment.billing-setting.title"](): string; + /** + * `Update` + */ + ["com.affine.payment.billing-setting.update"](): string; + /** + * `Upgrade` + */ + ["com.affine.payment.billing-setting.upgrade"](): string; + /** + * `View invoice` + */ + ["com.affine.payment.billing-setting.view-invoice"](): string; + /** + * `year` + */ + ["com.affine.payment.billing-setting.year"](): string; + /** + * `Please tell us more about your use case, to make AFFiNE better.` + */ + ["com.affine.payment.billing-type-form.description"](): string; + /** + * `Go` + */ + ["com.affine.payment.billing-type-form.go"](): string; + /** + * `Tell us your use case` + */ + ["com.affine.payment.billing-type-form.title"](): string; + /** + * `The maximum file upload size for local workspaces is {{quota}}.` + */ + ["com.affine.payment.blob-limit.description.local"](options: { + readonly quota: string; + }): string; + /** + * `The maximum file upload size for this joined workspace is {{quota}}. You can contact the owner of this workspace.` + */ + ["com.affine.payment.blob-limit.description.member"](options: { + readonly quota: string; + }): string; + /** + * `The maximum file upload size for this workspace is {{quota}}. To proceed, you can:` + */ + ["com.affine.payment.blob-limit.description.owner"](options: { + readonly quota: string; + }): string; + /** + * `Upgrade your account for larger file upload limits` + */ + ["com.affine.payment.blob-limit.description.owner.tips-1"](): string; + /** + * `Upgrade the workspace plan to increase storage for all member` + */ + ["com.affine.payment.blob-limit.description.owner.tips-2"](): string; + /** + * `Compress your file and upload again` + */ + ["com.affine.payment.blob-limit.description.owner.tips-3"](): string; + /** + * `You have reached the limit` + */ + ["com.affine.payment.blob-limit.title"](): string; + /** + * `Book a demo` + */ + ["com.affine.payment.book-a-demo"](): string; + /** + * `Buy Pro` + */ + ["com.affine.payment.buy-pro"](): string; + /** + * `Change to {{to}} Billing` + */ + ["com.affine.payment.change-to"](options: { + readonly to: string; + }): string; + /** + * `Include in FOSS` + */ + ["com.affine.payment.cloud.free.benefit.g1"](): string; + /** + * `Unlimited local workspaces` + */ + ["com.affine.payment.cloud.free.benefit.g1-1"](): string; + /** + * `Unlimited use and customization` + */ + ["com.affine.payment.cloud.free.benefit.g1-2"](): string; + /** + * `Unlimited doc and edgeless editing` + */ + ["com.affine.payment.cloud.free.benefit.g1-3"](): string; + /** + * `Include in Basic` + */ + ["com.affine.payment.cloud.free.benefit.g2"](): string; + /** + * `10 GB of cloud storage.` + */ + ["com.affine.payment.cloud.free.benefit.g2-1"](): string; + /** + * `10 MB of maximum file size.` + */ + ["com.affine.payment.cloud.free.benefit.g2-2"](): string; + /** + * `Up to 3 members per workspace.` + */ + ["com.affine.payment.cloud.free.benefit.g2-3"](): string; + /** + * `7-days cloud time machine file version history.` + */ + ["com.affine.payment.cloud.free.benefit.g2-4"](): string; + /** + * `Up to 3 login devices.` + */ + ["com.affine.payment.cloud.free.benefit.g2-5"](): string; + /** + * `Open-source under MIT license.` + */ + ["com.affine.payment.cloud.free.description"](): string; + /** + * `FOSS + Basic` + */ + ["com.affine.payment.cloud.free.name"](): string; + /** + * `Free forever` + */ + ["com.affine.payment.cloud.free.title"](): string; + /** + * `Included in Pro plan` + */ + ["com.affine.payment.cloud.onetime.included"](): string; + /** + * `Included in Believer plan` + */ + ["com.affine.payment.cloud.lifetime.included"](): string; + /** + * `We host, no technical setup required.` + */ + ["com.affine.payment.cloud.pricing-plan.select.caption"](): string; + /** + * `Hosted by AFFiNE.Pro` + */ + ["com.affine.payment.cloud.pricing-plan.select.title"](): string; + /** + * `Billed annually` + */ + ["com.affine.payment.cloud.pricing-plan.toggle-billed-yearly"](): string; + /** + * `Saving {{discount}}%` + */ + ["com.affine.payment.cloud.pricing-plan.toggle-discount"](options: { + readonly discount: string; + }): string; + /** + * `Annually` + */ + ["com.affine.payment.cloud.pricing-plan.toggle-yearly"](): string; + /** + * `Include in Pro` + */ + ["com.affine.payment.cloud.pro.benefit.g1"](): string; + /** + * `Everything in AFFiNE FOSS & Basic.` + */ + ["com.affine.payment.cloud.pro.benefit.g1-1"](): string; + /** + * `100 GB of cloud storage.` + */ + ["com.affine.payment.cloud.pro.benefit.g1-2"](): string; + /** + * `100 MB of maximum file size.` + */ + ["com.affine.payment.cloud.pro.benefit.g1-3"](): string; + /** + * `Up to 10 members per workspace.` + */ + ["com.affine.payment.cloud.pro.benefit.g1-4"](): string; + /** + * `30-days cloud time machine file version history.` + */ + ["com.affine.payment.cloud.pro.benefit.g1-5"](): string; + /** + * `Add comments on Doc and Edgeless.` + */ + ["com.affine.payment.cloud.pro.benefit.g1-6"](): string; + /** + * `Community support.` + */ + ["com.affine.payment.cloud.pro.benefit.g1-7"](): string; + /** + * `Real-time syncing & collaboration for more people.` + */ + ["com.affine.payment.cloud.pro.benefit.g1-8"](): string; + /** + * `For family and small teams.` + */ + ["com.affine.payment.cloud.pro.description"](): string; + /** + * `Pro` + */ + ["com.affine.payment.cloud.pro.name"](): string; + /** + * `annually` + */ + ["com.affine.payment.cloud.pro.title.billed-yearly"](): string; + /** + * `{{price}} per month` + */ + ["com.affine.payment.cloud.pro.title.price-monthly"](options: { + readonly price: string; + }): string; + /** + * `Include in Team Workspace` + */ + ["com.affine.payment.cloud.team-workspace.benefit.g1"](): string; + /** + * `Everything in AFFiNE Pro.` + */ + ["com.affine.payment.cloud.team-workspace.benefit.g1-1"](): string; + /** + * `100 GB initial storage + 20 GB per seat.` + */ + ["com.affine.payment.cloud.team-workspace.benefit.g1-2"](): string; + /** + * `500 MB of maximum file size.` + */ + ["com.affine.payment.cloud.team-workspace.benefit.g1-3"](): string; + /** + * `Unlimited team members (10+ seats).` + */ + ["com.affine.payment.cloud.team-workspace.benefit.g1-4"](): string; + /** + * `Multiple admin roles.` + */ + ["com.affine.payment.cloud.team-workspace.benefit.g1-5"](): string; + /** + * `Priority customer support.` + */ + ["com.affine.payment.cloud.team-workspace.benefit.g1-6"](): string; + /** + * `Best for scalable teams.` + */ + ["com.affine.payment.cloud.team-workspace.description"](): string; + /** + * `Team` + */ + ["com.affine.payment.cloud.team-workspace.name"](): string; + /** + * `annually` + */ + ["com.affine.payment.cloud.team-workspace.title.billed-yearly"](): string; + /** + * `{{price}} per seat/month` + */ + ["com.affine.payment.cloud.team-workspace.title.price-monthly"](options: { + readonly price: string; + }): string; + /** + * `Contact sales` + */ + ["com.affine.payment.contact-sales"](): string; + /** + * `Current plan` + */ + ["com.affine.payment.current-plan"](): string; + /** + * `Start 14-day free trial` + */ + ["com.affine.payment.start-free-trial"](): string; + /** + * `{{amount}}% off` + */ + ["com.affine.payment.discount-amount"](options: { + readonly amount: string; + }): string; + /** + * `Downgrade` + */ + ["com.affine.payment.downgrade"](): string; + /** + * `We'd like to hear more about where we fall short, so that we can make AFFiNE better.` + */ + ["com.affine.payment.downgraded-notify.content"](): string; + /** + * `Later` + */ + ["com.affine.payment.downgraded-notify.later"](): string; + /** + * `Sure, Open in browser` + */ + ["com.affine.payment.downgraded-notify.ok-client"](): string; + /** + * `Sure, Open in new tab` + */ + ["com.affine.payment.downgraded-notify.ok-web"](): string; + /** + * `Sorry to see you go` + */ + ["com.affine.payment.downgraded-notify.title"](): string; + /** + * `You have successfully downgraded. After the current billing period ends, your account will automatically switch to the Free plan.` + */ + ["com.affine.payment.downgraded-tooltip"](): string; + /** + * `Best team workspace for collaboration and knowledge distilling.` + */ + ["com.affine.payment.dynamic-benefit-1"](): string; + /** + * `Focusing on what really matters with team project management and automation.` + */ + ["com.affine.payment.dynamic-benefit-2"](): string; + /** + * `Pay for seats, fits all team size.` + */ + ["com.affine.payment.dynamic-benefit-3"](): string; + /** + * `Solutions & best practices for dedicated needs.` + */ + ["com.affine.payment.dynamic-benefit-4"](): string; + /** + * `Embedable & interrogations with IT support.` + */ + ["com.affine.payment.dynamic-benefit-5"](): string; + /** + * `Everything in AFFiNE Pro` + */ + ["com.affine.payment.lifetime.benefit-1"](): string; + /** + * `Life-time personal usage` + */ + ["com.affine.payment.lifetime.benefit-2"](): string; + /** + * `{{capacity}} Cloud Storage` + */ + ["com.affine.payment.lifetime.benefit-3"](options: { + readonly capacity: string; + }): string; + /** + * `Dedicated Discord support with AFFiNE makers` + */ + ["com.affine.payment.lifetime.benefit-4"](): string; + /** + * `Become a Life-time supporter?` + */ + ["com.affine.payment.lifetime.caption-1"](): string; + /** + * `Purchase` + */ + ["com.affine.payment.lifetime.purchase"](): string; + /** + * `Purchased` + */ + ["com.affine.payment.lifetime.purchased"](): string; + /** + * `Believer Plan` + */ + ["com.affine.payment.lifetime.title"](): string; + /** + * `Upgrade` + */ + ["com.affine.payment.member-limit.free.confirm"](): string; + /** + * `Workspaces created by {{planName}} users are limited to {{quota}} members. To add more collaborators, you can:` + */ + ["com.affine.payment.member-limit.description"](options: Readonly<{ + planName: string; + quota: string; + }>): string; + /** + * `Upgrade to AFFiNE Pro for expanded member capacity` + */ + ["com.affine.payment.member-limit.description.tips-for-free-plan"](): string; + /** + * `Convert to a Team Workspace for unlimited collaboration` + */ + ["com.affine.payment.member-limit.description.tips-1"](): string; + /** + * `Or create a new workspace` + */ + ["com.affine.payment.member-limit.description.tips-2"](): string; + /** + * `Got it` + */ + ["com.affine.payment.member-limit.pro.confirm"](): string; + /** + * `You have reached the limit` + */ + ["com.affine.payment.member-limit.title"](): string; + /** + * `Manage members here. {{planName}} users can invite up to {{memberLimit}}` + */ + ["com.affine.payment.member.description"](options: Readonly<{ + planName: string; + memberLimit: string; + }>): string; + /** + * `Choose your plan` + */ + ["com.affine.payment.member.description.choose-plan"](): string; + /** + * `go upgrade` + */ + ["com.affine.payment.member.description.go-upgrade"](): string; + /** + * `Looking to collaborate with more people?` + */ + ["com.affine.payment.member.description2"](): string; + /** + * `Work together with unlimited team members.` + */ + ["com.affine.payment.member.team.description"](): string; + /** + * `Invite team members` + */ + ["com.affine.payment.member.team.invite.title"](): string; + /** + * `Invite new members to join your workspace via email or share an invite link` + */ + ["com.affine.payment.member.team.invite.description"](): string; + /** + * `Email Invite` + */ + ["com.affine.payment.member.team.invite.email-invite"](): string; + /** + * `Invite Link` + */ + ["com.affine.payment.member.team.invite.invite-link"](): string; + /** + * `Email addresses` + */ + ["com.affine.payment.member.team.invite.email-addresses"](): string; + /** + * `Enter email addresses (separated by commas)` + */ + ["com.affine.payment.member.team.invite.email-placeholder"](): string; + /** + * `Import CSV` + */ + ["com.affine.payment.member.team.invite.import-csv"](): string; + /** + * `Send Invites` + */ + ["com.affine.payment.member.team.invite.send-invites"](): string; + /** + * `Link expiration` + */ + ["com.affine.payment.member.team.invite.link-expiration"](): string; + /** + * `{{number}} days` + */ + ["com.affine.payment.member.team.invite.expiration-date"](options: { + readonly number: string; + }): string; + /** + * `Invitation link` + */ + ["com.affine.payment.member.team.invite.invitation-link"](): string; + /** + * `Generate a link to invite members to your workspace` + */ + ["com.affine.payment.member.team.invite.invitation-link.description"](): string; + /** + * `Generate` + */ + ["com.affine.payment.member.team.invite.generate"](): string; + /** + * `Copy` + */ + ["com.affine.payment.member.team.invite.copy"](): string; + /** + * `Done` + */ + ["com.affine.payment.member.team.invite.done"](): string; + /** + * `Invitation sent,{{successCount}} successful, {{failedCount}} failed` + */ + ["com.affine.payment.member.team.invite.notify.title"](options: Readonly<{ + successCount: string; + failedCount: string; + }>): string; + /** + * `These email addresses have already been invited:` + */ + ["com.affine.payment.member.team.invite.notify.fail-message"](): string; + /** + * `Revoke invitation` + */ + ["com.affine.payment.member.team.revoke"](): string; + /** + * `Approve` + */ + ["com.affine.payment.member.team.approve"](): string; + /** + * `Decline` + */ + ["com.affine.payment.member.team.decline"](): string; + /** + * `Remove member` + */ + ["com.affine.payment.member.team.remove"](): string; + /** + * `Change role to admin` + */ + ["com.affine.payment.member.team.change.admin"](): string; + /** + * `Change role to collaborator` + */ + ["com.affine.payment.member.team.change.collaborator"](): string; + /** + * `Assign as owner` + */ + ["com.affine.payment.member.team.assign"](): string; + /** + * `Invitation Revoked` + */ + ["com.affine.payment.member.team.revoke.notify.title"](): string; + /** + * `You have canceled the invitation for {{name}}` + */ + ["com.affine.payment.member.team.revoke.notify.message"](options: { + readonly name: string; + }): string; + /** + * `Request approved` + */ + ["com.affine.payment.member.team.approve.notify.title"](): string; + /** + * `You have approved the {{name}}’s request to join this workspace` + */ + ["com.affine.payment.member.team.approve.notify.message"](options: { + readonly name: string; + }): string; + /** + * `Request declined` + */ + ["com.affine.payment.member.team.decline.notify.title"](): string; + /** + * `You have declined the {{name}}’s request to join this workspace` + */ + ["com.affine.payment.member.team.decline.notify.message"](options: { + readonly name: string; + }): string; + /** + * `Member removed` + */ + ["com.affine.payment.member.team.remove.notify.title"](): string; + /** + * `You have removed {{name}} from this workspace` + */ + ["com.affine.payment.member.team.remove.notify.message"](options: { + readonly name: string; + }): string; + /** + * `Role Updated` + */ + ["com.affine.payment.member.team.change.notify.title"](): string; + /** + * `You have successfully promoted {{name}} to Admin.` + */ + ["com.affine.payment.member.team.change.admin.notify.message"](options: { + readonly name: string; + }): string; + /** + * `You have successfully changed {{name}} s role to collaborator.` + */ + ["com.affine.payment.member.team.change.collaborator.notify.message"](options: { + readonly name: string; + }): string; + /** + * `Owner assigned` + */ + ["com.affine.payment.member.team.assign.notify.title"](): string; + /** + * `You have successfully assigned {{name}} as the owner of this workspace.` + */ + ["com.affine.payment.member.team.assign.notify.message"](options: { + readonly name: string; + }): string; + /** + * `Confirm new workspace owner` + */ + ["com.affine.payment.member.team.assign.confirm.title"](): string; + /** + * `You are about to transfer workspace ownership to {{name}}. Please review the following changes carefully:` + */ + ["com.affine.payment.member.team.assign.confirm.description"](options: { + readonly name: string; + }): string; + /** + * `This action cannot be undone` + */ + ["com.affine.payment.member.team.assign.confirm.description-1"](): string; + /** + * `Your role will be changed to Admin` + */ + ["com.affine.payment.member.team.assign.confirm.description-2"](): string; + /** + * `You will lose ownership rights to the entire workspace` + */ + ["com.affine.payment.member.team.assign.confirm.description-3"](): string; + /** + * `To confirm this transfer, please type the workspace name` + */ + ["com.affine.payment.member.team.assign.confirm.description-4"](): string; + /** + * `Type workspace name to confirm` + */ + ["com.affine.payment.member.team.assign.confirm.placeholder"](): string; + /** + * `Transfer Ownership` + */ + ["com.affine.payment.member.team.assign.confirm.button"](): string; + /** + * `Remove member from workspace?` + */ + ["com.affine.payment.member.team.remove.confirm.title"](): string; + /** + * `This action will revoke their access to all workspace resources immediately.` + */ + ["com.affine.payment.member.team.remove.confirm.description"](): string; + /** + * `Remove Member` + */ + ["com.affine.payment.member.team.remove.confirm.confirm-button"](): string; + /** + * `Cancel` + */ + ["com.affine.payment.member.team.remove.confirm.cancel"](): string; + /** + * `Cancel` + */ + ["com.affine.payment.modal.change.cancel"](): string; + /** + * `Change` + */ + ["com.affine.payment.modal.change.confirm"](): string; + /** + * `Change your subscription` + */ + ["com.affine.payment.modal.change.title"](): string; + /** + * `Cancel subscription` + */ + ["com.affine.payment.modal.downgrade.cancel"](): string; + /** + * `You can still use AFFiNE Cloud Pro until the end of this billing period :)` + */ + ["com.affine.payment.modal.downgrade.caption"](): string; + /** + * `Keep AFFiNE Cloud Pro` + */ + ["com.affine.payment.modal.downgrade.confirm"](): string; + /** + * `Keep Team plan` + */ + ["com.affine.payment.modal.downgrade.team-confirm"](): string; + /** + * `We're sorry to see you go, but we're always working to improve, and your feedback is welcome. We hope to see you return in the future.` + */ + ["com.affine.payment.modal.downgrade.content"](): string; + /** + * `Are you sure?` + */ + ["com.affine.payment.modal.downgrade.title"](): string; + /** + * `Cancel` + */ + ["com.affine.payment.modal.resume.cancel"](): string; + /** + * `Confirm` + */ + ["com.affine.payment.modal.resume.confirm"](): string; + /** + * `Are you sure you want to resume the subscription for your pro account? This means your payment method will be charged automatically at the end of each billing cycle, starting from the next billing cycle.` + */ + ["com.affine.payment.modal.resume.content"](): string; + /** + * `Resume auto-renewal?` + */ + ["com.affine.payment.modal.resume.title"](): string; + /** + * `Refresh` + */ + ["com.affine.payment.plans-error-retry"](): string; + /** + * `Unable to load pricing plans, please check your network. ` + */ + ["com.affine.payment.plans-error-tip"](): string; + /** + * `monthly` + */ + ["com.affine.payment.recurring-monthly"](): string; + /** + * `annually` + */ + ["com.affine.payment.recurring-yearly"](): string; + /** + * `Resume` + */ + ["com.affine.payment.resume"](): string; + /** + * `Resume auto-renewal` + */ + ["com.affine.payment.resume-renewal"](): string; + /** + * `See all plans` + */ + ["com.affine.payment.see-all-plans"](): string; + /** + * `Sign up free` + */ + ["com.affine.payment.sign-up-free"](): string; + /** + * `Cloud storage is insufficient. Please contact the owner of that workspace.` + */ + ["com.affine.payment.storage-limit.description.member"](): string; + /** + * `Cloud storage is insufficient. You can upgrade your account to unlock more cloud storage.` + */ + ["com.affine.payment.storage-limit.description.owner"](): string; + /** + * `Unable to sync due to insufficient storage space. You can remove excess content, upgrade your account, or increase your workspace storage to resolve this issue.` + */ + ["com.affine.payment.storage-limit.new-description.owner"](): string; + /** + * `Sync failed due to storage space limit` + */ + ["com.affine.payment.storage-limit.new-title"](): string; + /** + * `View` + */ + ["com.affine.payment.storage-limit.view"](): string; + /** + * `You are currently on the {{plan}} plan. After the current billing period ends, your account will automatically switch to the Free plan.` + */ + ["com.affine.payment.subtitle-canceled"](options: { + readonly plan: string; + }): string; + /** + * `This is the pricing plans of AFFiNE Cloud. You can sign up or sign in to your account first.` + */ + ["com.affine.payment.subtitle-not-signed-in"](): string; + /** + * `See all plans` + */ + ["com.affine.payment.tag-tooltips"](): string; + /** + * `Tell us your use case` + */ + ["com.affine.payment.tell-us-use-case"](): string; + /** + * `Pricing plans` + */ + ["com.affine.payment.title"](): string; + /** + * `You have changed your plan to {{plan}} billing.` + */ + ["com.affine.payment.updated-notify-msg"](options: { + readonly plan: string; + }): string; + /** + * `Subscription updated` + */ + ["com.affine.payment.updated-notify-title"](): string; + /** + * `Upgrade` + */ + ["com.affine.payment.upgrade"](): string; + /** + * `Redeem code` + */ + ["com.affine.payment.redeem-code"](): string; + /** + * `We'd like to hear more about your use case, so that we can make AFFiNE better.` + */ + ["com.affine.payment.upgrade-success-notify.content"](): string; + /** + * `Later` + */ + ["com.affine.payment.upgrade-success-notify.later"](): string; + /** + * `Sure, open in browser` + */ + ["com.affine.payment.upgrade-success-notify.ok-client"](): string; + /** + * `Sure, open in new tab` + */ + ["com.affine.payment.upgrade-success-notify.ok-web"](): string; + /** + * `Thanks for subscribing!` + */ + ["com.affine.payment.upgrade-success-notify.title"](): string; + /** + * `Congratulations! Your AFFiNE account has been successfully upgraded to a Pro account.` + */ + ["com.affine.payment.upgrade-success-page.text"](): string; + /** + * `Upgrade successful!` + */ + ["com.affine.payment.upgrade-success-page.title"](): string; + /** + * `Congratulations! Your workspace has been successfully upgraded to a Team Workspace. Now you can invite unlimited members to collaborate in this workspace.` + */ + ["com.affine.payment.upgrade-success-page.team.text-1"](): string; + /** + * `Close` + */ + ["com.affine.peek-view-controls.close"](): string; + /** + * `Open this doc` + */ + ["com.affine.peek-view-controls.open-doc"](): string; + /** + * `Open in new tab` + */ + ["com.affine.peek-view-controls.open-doc-in-new-tab"](): string; + /** + * `Open in split view` + */ + ["com.affine.peek-view-controls.open-doc-in-split-view"](): string; + /** + * `Open doc info` + */ + ["com.affine.peek-view-controls.open-info"](): string; + /** + * `Open this attachment` + */ + ["com.affine.peek-view-controls.open-attachment"](): string; + /** + * `Open in new tab` + */ + ["com.affine.peek-view-controls.open-attachment-in-new-tab"](): string; + /** + * `Open in split view` + */ + ["com.affine.peek-view-controls.open-attachment-in-split-view"](): string; + /** + * `New` + */ + ["com.affine.quicksearch.group.creation"](): string; + /** + * `Search for "{{query}}"` + */ + ["com.affine.quicksearch.group.searchfor"](options: { + readonly query: string; + }): string; + /** + * `Reset sync` + */ + ["com.affine.resetSyncStatus.button"](): string; + /** + * `This operation may fix some synchronization issues.` + */ + ["com.affine.resetSyncStatus.description"](): string; + /** + * `Collections` + */ + ["com.affine.rootAppSidebar.collections"](): string; + /** + * `Only doc can be placed on here` + */ + ["com.affine.rootAppSidebar.doc.link-doc-only"](): string; + /** + * `No linked docs` + */ + ["com.affine.rootAppSidebar.docs.no-subdoc"](): string; + /** + * `Loading linked docs...` + */ + ["com.affine.rootAppSidebar.docs.references-loading"](): string; + /** + * `New doc` + */ + ["com.affine.rootAppSidebar.explorer.collection-add-tooltip"](): string; + /** + * `New collection` + */ + ["com.affine.rootAppSidebar.explorer.collection-section-add-tooltip"](): string; + /** + * `New linked doc` + */ + ["com.affine.rootAppSidebar.explorer.doc-add-tooltip"](): string; + /** + * `Copy` + */ + ["com.affine.rootAppSidebar.explorer.drop-effect.copy"](): string; + /** + * `Link` + */ + ["com.affine.rootAppSidebar.explorer.drop-effect.link"](): string; + /** + * `Move` + */ + ["com.affine.rootAppSidebar.explorer.drop-effect.move"](): string; + /** + * `New doc` + */ + ["com.affine.rootAppSidebar.explorer.fav-section-add-tooltip"](): string; + /** + * `New doc` + */ + ["com.affine.rootAppSidebar.explorer.organize-add-tooltip"](): string; + /** + * `New folder` + */ + ["com.affine.rootAppSidebar.explorer.organize-section-add-tooltip"](): string; + /** + * `New doc` + */ + ["com.affine.rootAppSidebar.explorer.tag-add-tooltip"](): string; + /** + * `New tag` + */ + ["com.affine.rootAppSidebar.explorer.tag-section-add-tooltip"](): string; + /** + * `Favorites` + */ + ["com.affine.rootAppSidebar.favorites"](): string; + /** + * `No favorites` + */ + ["com.affine.rootAppSidebar.favorites.empty"](): string; + /** + * `Migration data` + */ + ["com.affine.rootAppSidebar.migration-data"](): string; + /** + * `Empty the old favorites` + */ + ["com.affine.rootAppSidebar.migration-data.clean-all"](): string; + /** + * `Cancel` + */ + ["com.affine.rootAppSidebar.migration-data.clean-all.cancel"](): string; + /** + * `OK` + */ + ["com.affine.rootAppSidebar.migration-data.clean-all.confirm"](): string; + /** + * `The old "Favorites" will be replaced` + */ + ["com.affine.rootAppSidebar.migration-data.help"](): string; + /** + * `Empty the old favorites` + */ + ["com.affine.rootAppSidebar.migration-data.help.clean-all"](): string; + /** + * `OK` + */ + ["com.affine.rootAppSidebar.migration-data.help.confirm"](): string; + /** + * `Organize` + */ + ["com.affine.rootAppSidebar.organize"](): string; + /** + * `Delete` + */ + ["com.affine.rootAppSidebar.organize.delete"](): string; + /** + * `Remove from folder` + */ + ["com.affine.rootAppSidebar.organize.delete-from-folder"](): string; + /** + * `Delete the folder will not delete any docs, tags, or collections.` + */ + ["com.affine.rootAppSidebar.organize.delete.notify-message"](): string; + /** + * `Delete {{name}}` + */ + ["com.affine.rootAppSidebar.organize.delete.notify-title"](options: { + readonly name: string; + }): string; + /** + * `No folders` + */ + ["com.affine.rootAppSidebar.organize.empty"](): string; + /** + * `Empty folder` + */ + ["com.affine.rootAppSidebar.organize.empty-folder"](): string; + /** + * `Add pages` + */ + ["com.affine.rootAppSidebar.organize.empty-folder.add-pages"](): string; + /** + * `New folder` + */ + ["com.affine.rootAppSidebar.organize.empty.new-folders-button"](): string; + /** + * `Add to favorites` + */ + ["com.affine.rootAppSidebar.organize.folder-add-favorite"](): string; + /** + * `Remove from favorites` + */ + ["com.affine.rootAppSidebar.organize.folder-rm-favorite"](): string; + /** + * `Add Collections` + */ + ["com.affine.rootAppSidebar.organize.folder.add-collections"](): string; + /** + * `Add docs` + */ + ["com.affine.rootAppSidebar.organize.folder.add-docs"](): string; + /** + * `Add others` + */ + ["com.affine.rootAppSidebar.organize.folder.add-others"](): string; + /** + * `Add tags` + */ + ["com.affine.rootAppSidebar.organize.folder.add-tags"](): string; + /** + * `Create a subfolder` + */ + ["com.affine.rootAppSidebar.organize.folder.create-subfolder"](): string; + /** + * `New folder` + */ + ["com.affine.rootAppSidebar.organize.new-folders"](): string; + /** + * `Only folder can be placed on here` + */ + ["com.affine.rootAppSidebar.organize.root-folder-only"](): string; + /** + * `Add More` + */ + ["com.affine.rootAppSidebar.organize.add-more"](): string; + /** + * `Add Folder` + */ + ["com.affine.rootAppSidebar.organize.add-folder"](): string; + /** + * `New Collection` + */ + ["com.affine.rootAppSidebar.collection.new"](): string; + /** + * `Others` + */ + ["com.affine.rootAppSidebar.others"](): string; + /** + * `Only doc can be placed on here` + */ + ["com.affine.rootAppSidebar.tag.doc-only"](): string; + /** + * `Tags` + */ + ["com.affine.rootAppSidebar.tags"](): string; + /** + * `No tags` + */ + ["com.affine.rootAppSidebar.tags.empty"](): string; + /** + * `New tag` + */ + ["com.affine.rootAppSidebar.tags.empty.new-tag-button"](): string; + /** + * `New tag` + */ + ["com.affine.rootAppSidebar.tags.new-tag"](): string; + /** + * `No docs` + */ + ["com.affine.rootAppSidebar.tags.no-doc"](): string; + /** + * `Drag to resize` + */ + ["com.affine.rootAppSidebar.resize-handle.tooltip.drag"](): string; + /** + * `Click to collapse` + */ + ["com.affine.rootAppSidebar.resize-handle.tooltip.click"](): string; + /** + * `Type here ...` + */ + ["com.affine.search-tags.placeholder"](): string; + /** + * `Empty` + */ + ["com.affine.selectPage.empty"](): string; + /** + * `Selected` + */ + ["com.affine.selectPage.selected"](): string; + /** + * `Add include doc` + */ + ["com.affine.selectPage.title"](): string; + /** + * `Search collections...` + */ + ["com.affine.selector-collection.search.placeholder"](): string; + /** + * `Search tags...` + */ + ["com.affine.selector-tag.search.placeholder"](): string; + /** + * `Account settings` + */ + ["com.affine.setting.account"](): string; + /** + * `Delete account` + */ + ["com.affine.setting.account.delete"](): string; + /** + * `Permanently delete this account and the Workspace data backup in AFFiNE Cloud. This action can not be undone.` + */ + ["com.affine.setting.account.delete.message"](): string; + /** + * `Your personal information` + */ + ["com.affine.setting.account.message"](): string; + /** + * `Sync with AFFiNE Cloud` + */ + ["com.affine.setting.sign.message"](): string; + /** + * `Securely sign out of your account.` + */ + ["com.affine.setting.sign.out.message"](): string; + /** + * `General` + */ + ["com.affine.settingSidebar.settings.general"](): string; + /** + * `Workspace` + */ + ["com.affine.settingSidebar.settings.workspace"](): string; + /** + * `Settings` + */ + ["com.affine.settingSidebar.title"](): string; + /** + * `Appearance` + */ + ["com.affine.settings.appearance"](): string; + /** + * `Customise the appearance of the client.` + */ + ["com.affine.settings.appearance.border-style-description"](): string; + /** + * `Customise your date style.` + */ + ["com.affine.settings.appearance.date-format-description"](): string; + /** + * `Maximum display of content within a doc.` + */ + ["com.affine.settings.appearance.full-width-description"](): string; + /** + * `Select the language for the interface.` + */ + ["com.affine.settings.appearance.language-description"](): string; + /** + * `By default, the week starts on Sunday.` + */ + ["com.affine.settings.appearance.start-week-description"](): string; + /** + * `Customise appearance of Windows Client.` + */ + ["com.affine.settings.appearance.window-frame-description"](): string; + /** + * `Links` + */ + ["com.affine.setting.appearance.links"](): string; + /** + * `Open AFFiNE links` + */ + ["com.affine.setting.appearance.open-in-app"](): string; + /** + * `You can choose to open the link in the desktop app or directly in the browser.` + */ + ["com.affine.setting.appearance.open-in-app.hint"](): string; + /** + * `Ask me each time` + */ + ["com.affine.setting.appearance.open-in-app.always-ask"](): string; + /** + * `Open links in desktop app` + */ + ["com.affine.setting.appearance.open-in-app.open-in-desktop-app"](): string; + /** + * `Open links in browser` + */ + ["com.affine.setting.appearance.open-in-app.open-in-web"](): string; + /** + * `Open AFFiNE links` + */ + ["com.affine.setting.appearance.open-in-app.title"](): string; + /** + * `Open this doc in AFFiNE app` + */ + ["com.affine.open-in-app.card.title"](): string; + /** + * `Open in app` + */ + ["com.affine.open-in-app.card.button.open"](): string; + /** + * `Dismiss` + */ + ["com.affine.open-in-app.card.button.dismiss"](): string; + /** + * `Remember choice` + */ + ["com.affine.open-in-app.card.remember"](): string; + /** + * `Download desktop app` + */ + ["com.affine.open-in-app.card.download"](): string; + /** + * `If enabled, it will automatically check for new versions at regular intervals.` + */ + ["com.affine.settings.auto-check-description"](): string; + /** + * `If enabled, new versions will be automatically downloaded to the current device.` + */ + ["com.affine.settings.auto-download-description"](): string; + /** + * `Editor` + */ + ["com.affine.settings.editorSettings"](): string; + /** + * `Edgeless` + */ + ["com.affine.settings.editorSettings.edgeless"](): string; + /** + * `Connector` + */ + ["com.affine.settings.editorSettings.edgeless.connecter"](): string; + /** + * `Border style` + */ + ["com.affine.settings.editorSettings.edgeless.connecter.border-style"](): string; + /** + * `Border thickness` + */ + ["com.affine.settings.editorSettings.edgeless.connecter.border-thickness"](): string; + /** + * `Color` + */ + ["com.affine.settings.editorSettings.edgeless.connecter.color"](): string; + /** + * `Connector shape` + */ + ["com.affine.settings.editorSettings.edgeless.connecter.connector-shape"](): string; + /** + * `Curve` + */ + ["com.affine.settings.editorSettings.edgeless.connecter.connector-shape.curve"](): string; + /** + * `Elbowed` + */ + ["com.affine.settings.editorSettings.edgeless.connecter.connector-shape.elbowed"](): string; + /** + * `Straight` + */ + ["com.affine.settings.editorSettings.edgeless.connecter.connector-shape.straight"](): string; + /** + * `End endpoint` + */ + ["com.affine.settings.editorSettings.edgeless.connecter.end-endpoint"](): string; + /** + * `Start endpoint` + */ + ["com.affine.settings.editorSettings.edgeless.connecter.start-endpoint"](): string; + /** + * `Custom` + */ + ["com.affine.settings.editorSettings.edgeless.custom"](): string; + /** + * `Mind Map` + */ + ["com.affine.settings.editorSettings.edgeless.mind-map"](): string; + /** + * `Layout` + */ + ["com.affine.settings.editorSettings.edgeless.mind-map.layout"](): string; + /** + * `Left` + */ + ["com.affine.settings.editorSettings.edgeless.mind-map.layout.left"](): string; + /** + * `Radial` + */ + ["com.affine.settings.editorSettings.edgeless.mind-map.layout.radial"](): string; + /** + * `Right` + */ + ["com.affine.settings.editorSettings.edgeless.mind-map.layout.right"](): string; + /** + * `Note` + */ + ["com.affine.settings.editorSettings.edgeless.note"](): string; + /** + * `Background` + */ + ["com.affine.settings.editorSettings.edgeless.note.background"](): string; + /** + * `Border style` + */ + ["com.affine.settings.editorSettings.edgeless.note.border"](): string; + /** + * `Border thickness` + */ + ["com.affine.settings.editorSettings.edgeless.note.border-thickness"](): string; + /** + * `Dash` + */ + ["com.affine.settings.editorSettings.edgeless.note.border.dash"](): string; + /** + * `None` + */ + ["com.affine.settings.editorSettings.edgeless.note.border.none"](): string; + /** + * `Solid` + */ + ["com.affine.settings.editorSettings.edgeless.note.border.solid"](): string; + /** + * `Corners` + */ + ["com.affine.settings.editorSettings.edgeless.note.corners"](): string; + /** + * `Shadow style` + */ + ["com.affine.settings.editorSettings.edgeless.note.shadow"](): string; + /** + * `Pen` + */ + ["com.affine.settings.editorSettings.edgeless.pen"](): string; + /** + * `Color` + */ + ["com.affine.settings.editorSettings.edgeless.pen.color"](): string; + /** + * `Thickness` + */ + ["com.affine.settings.editorSettings.edgeless.pen.thickness"](): string; + /** + * `Shape` + */ + ["com.affine.settings.editorSettings.edgeless.shape"](): string; + /** + * `Border color` + */ + ["com.affine.settings.editorSettings.edgeless.shape.border-color"](): string; + /** + * `Border style` + */ + ["com.affine.settings.editorSettings.edgeless.shape.border-style"](): string; + /** + * `Border thickness` + */ + ["com.affine.settings.editorSettings.edgeless.shape.border-thickness"](): string; + /** + * `Diamond` + */ + ["com.affine.settings.editorSettings.edgeless.shape.diamond"](): string; + /** + * `Ellipse` + */ + ["com.affine.settings.editorSettings.edgeless.shape.ellipse"](): string; + /** + * `Fill color` + */ + ["com.affine.settings.editorSettings.edgeless.shape.fill-color"](): string; + /** + * `Flow` + */ + ["com.affine.settings.editorSettings.edgeless.shape.flow"](): string; + /** + * `Font` + */ + ["com.affine.settings.editorSettings.edgeless.shape.font"](): string; + /** + * `Font size` + */ + ["com.affine.settings.editorSettings.edgeless.shape.font-size"](): string; + /** + * `Font style` + */ + ["com.affine.settings.editorSettings.edgeless.shape.font-style"](): string; + /** + * `List` + */ + ["com.affine.settings.editorSettings.edgeless.shape.list"](): string; + /** + * `Rounded Rectangle` + */ + ["com.affine.settings.editorSettings.edgeless.shape.rounded-rectangle"](): string; + /** + * `Square` + */ + ["com.affine.settings.editorSettings.edgeless.shape.square"](): string; + /** + * `Text alignment` + */ + ["com.affine.settings.editorSettings.edgeless.shape.text-alignment"](): string; + /** + * `Text color` + */ + ["com.affine.settings.editorSettings.edgeless.shape.text-color"](): string; + /** + * `Triangle` + */ + ["com.affine.settings.editorSettings.edgeless.shape.triangle"](): string; + /** + * `Style` + */ + ["com.affine.settings.editorSettings.edgeless.style"](): string; + /** + * `General` + */ + ["com.affine.settings.editorSettings.edgeless.style.general"](): string; + /** + * `Scribbled` + */ + ["com.affine.settings.editorSettings.edgeless.style.scribbled"](): string; + /** + * `Text` + */ + ["com.affine.settings.editorSettings.edgeless.text"](): string; + /** + * `Alignment` + */ + ["com.affine.settings.editorSettings.edgeless.text.alignment"](): string; + /** + * `Center` + */ + ["com.affine.settings.editorSettings.edgeless.text.alignment.center"](): string; + /** + * `Left` + */ + ["com.affine.settings.editorSettings.edgeless.text.alignment.left"](): string; + /** + * `Right` + */ + ["com.affine.settings.editorSettings.edgeless.text.alignment.right"](): string; + /** + * `Text color` + */ + ["com.affine.settings.editorSettings.edgeless.text.color"](): string; + /** + * `Font` + */ + ["com.affine.settings.editorSettings.edgeless.text.font"](): string; + /** + * `Font family` + */ + ["com.affine.settings.editorSettings.edgeless.text.font-family"](): string; + /** + * `Font size` + */ + ["com.affine.settings.editorSettings.edgeless.text.font-size"](): string; + /** + * `Font style` + */ + ["com.affine.settings.editorSettings.edgeless.text.font-style"](): string; + /** + * `Font weight` + */ + ["com.affine.settings.editorSettings.edgeless.text.font-weight"](): string; + /** + * `General` + */ + ["com.affine.settings.editorSettings.general"](): string; + /** + * `Enable the powerful AI assistant, AFFiNE AI.` + */ + ["com.affine.settings.editorSettings.general.ai.description"](): string; + /** + * `Disable AI and Reload` + */ + ["com.affine.settings.editorSettings.general.ai.disable.confirm"](): string; + /** + * `Are you sure you want to disable AI? We value your productivity and our AI can enhance it. Please think again!` + */ + ["com.affine.settings.editorSettings.general.ai.disable.description"](): string; + /** + * `Disable AI?` + */ + ["com.affine.settings.editorSettings.general.ai.disable.title"](): string; + /** + * `Enable AI and Reload` + */ + ["com.affine.settings.editorSettings.general.ai.enable.confirm"](): string; + /** + * `Do you want to enable AI? Our AI assistant is ready to enhance your productivity and provide smart assistance. Let's get started! We need reload page to make this change.` + */ + ["com.affine.settings.editorSettings.general.ai.enable.description"](): string; + /** + * `Enable AI?` + */ + ["com.affine.settings.editorSettings.general.ai.enable.title"](): string; + /** + * `AFFiNE AI` + */ + ["com.affine.settings.editorSettings.general.ai.title"](): string; + /** + * `Set a default programming language.` + */ + ["com.affine.settings.editorSettings.general.default-code-block.language.description"](): string; + /** + * `Code blocks default language` + */ + ["com.affine.settings.editorSettings.general.default-code-block.language.title"](): string; + /** + * `Encapsulate code snippets for better readability.` + */ + ["com.affine.settings.editorSettings.general.default-code-block.wrap.description"](): string; + /** + * `Wrap code in code blocks` + */ + ["com.affine.settings.editorSettings.general.default-code-block.wrap.title"](): string; + /** + * `Default mode for new doc.` + */ + ["com.affine.settings.editorSettings.general.default-new-doc.description"](): string; + /** + * `New doc default mode` + */ + ["com.affine.settings.editorSettings.general.default-new-doc.title"](): string; + /** + * `Customize your text experience.` + */ + ["com.affine.settings.editorSettings.general.font-family.custom.description"](): string; + /** + * `Custom font family` + */ + ["com.affine.settings.editorSettings.general.font-family.custom.title"](): string; + /** + * `Choose your editor's font family.` + */ + ["com.affine.settings.editorSettings.general.font-family.description"](): string; + /** + * `Font family` + */ + ["com.affine.settings.editorSettings.general.font-family.title"](): string; + /** + * `Automatically detect and correct spelling errors.` + */ + ["com.affine.settings.editorSettings.general.spell-check.description"](): string; + /** + * `Spell check` + */ + ["com.affine.settings.editorSettings.general.spell-check.title"](): string; + /** + * `Page` + */ + ["com.affine.settings.editorSettings.page"](): string; + /** + * `Display bi-directional links on the doc.` + */ + ["com.affine.settings.editorSettings.page.display-bi-link.description"](): string; + /** + * `Display bi-directional links` + */ + ["com.affine.settings.editorSettings.page.display-bi-link.title"](): string; + /** + * `Display document information on the doc.` + */ + ["com.affine.settings.editorSettings.page.display-doc-info.description"](): string; + /** + * `Display doc info` + */ + ["com.affine.settings.editorSettings.page.display-doc-info.title"](): string; + /** + * `Maximise display of content within a page.` + */ + ["com.affine.settings.editorSettings.page.full-width.description"](): string; + /** + * `Full width layout` + */ + ["com.affine.settings.editorSettings.page.full-width.title"](): string; + /** + * `Default page width` + */ + ["com.affine.settings.editorSettings.page.default-page-width.title"](): string; + /** + * `Set default width for new pages, individual pages can override.` + */ + ["com.affine.settings.editorSettings.page.default-page-width.description"](): string; + /** + * `Standard` + */ + ["com.affine.settings.editorSettings.page.default-page-width.standard"](): string; + /** + * `Full width` + */ + ["com.affine.settings.editorSettings.page.default-page-width.full-width"](): string; + /** + * `Set edgeless default color scheme.` + */ + ["com.affine.settings.editorSettings.page.edgeless-default-theme.description"](): string; + /** + * `Edgeless default theme` + */ + ["com.affine.settings.editorSettings.page.edgeless-default-theme.title"](): string; + /** + * `Specified by current color mode` + */ + ["com.affine.settings.editorSettings.page.edgeless-default-theme.specified"](): string; + /** + * `Preferences` + */ + ["com.affine.settings.editorSettings.preferences"](): string; + /** + * `You can export the entire preferences data for backup, and the exported data can be re-imported.` + */ + ["com.affine.settings.editorSettings.preferences.export.description"](): string; + /** + * `Export Settings` + */ + ["com.affine.settings.editorSettings.preferences.export.title"](): string; + /** + * `You can import previously exported preferences data for restoration.` + */ + ["com.affine.settings.editorSettings.preferences.import.description"](): string; + /** + * `Import Settings` + */ + ["com.affine.settings.editorSettings.preferences.import.title"](): string; + /** + * `Configure your own editor.` + */ + ["com.affine.settings.editorSettings.subtitle"](): string; + /** + * `Editor settings` + */ + ["com.affine.settings.editorSettings.title"](): string; + /** + * `Email` + */ + ["com.affine.settings.email"](): string; + /** + * `Change email` + */ + ["com.affine.settings.email.action"](): string; + /** + * `Change email` + */ + ["com.affine.settings.email.action.change"](): string; + /** + * `Verify email` + */ + ["com.affine.settings.email.action.verify"](): string; + /** + * `Enable AFFiNE Cloud to collaborate with others` + */ + ["com.affine.settings.member-tooltip"](): string; + /** + * `Loading member list...` + */ + ["com.affine.settings.member.loading"](): string; + /** + * `Noise background on the sidebar` + */ + ["com.affine.settings.noise-style"](): string; + /** + * `Use background noise effect on the sidebar.` + */ + ["com.affine.settings.noise-style-description"](): string; + /** + * `Password` + */ + ["com.affine.settings.password"](): string; + /** + * `Change password` + */ + ["com.affine.settings.password.action.change"](): string; + /** + * `Set password` + */ + ["com.affine.settings.password.action.set"](): string; + /** + * `Set a password to sign in to your account` + */ + ["com.affine.settings.password.message"](): string; + /** + * `My profile` + */ + ["com.affine.settings.profile"](): string; + /** + * `Your account profile will be displayed to everyone.` + */ + ["com.affine.settings.profile.message"](): string; + /** + * `Display name` + */ + ["com.affine.settings.profile.name"](): string; + /** + * `Input account name` + */ + ["com.affine.settings.profile.placeholder"](): string; + /** + * `Remove workspace` + */ + ["com.affine.settings.remove-workspace"](): string; + /** + * `Remove workspace from this device and optionally delete all data.` + */ + ["com.affine.settings.remove-workspace-description"](): string; + /** + * `Sign in / Sign up` + */ + ["com.affine.settings.sign"](): string; + /** + * `Need more customization options? Tell us in the community.` + */ + ["com.affine.settings.suggestion"](): string; + /** + * `Translucent UI on the sidebar` + */ + ["com.affine.settings.translucent-style"](): string; + /** + * `Use transparency effect on the sidebar.` + */ + ["com.affine.settings.translucent-style-description"](): string; + /** + * `Workspace` + */ + ["com.affine.settings.workspace"](): string; + /** + * `You can view current workspace's information here.` + */ + ["com.affine.settings.workspace.description"](): string; + /** + * `Experimental features` + */ + ["com.affine.settings.workspace.experimental-features"](): string; + /** + * `Get started` + */ + ["com.affine.settings.workspace.experimental-features.get-started"](): string; + /** + * `Experimental features` + */ + ["com.affine.settings.workspace.experimental-features.header.plugins"](): string; + /** + * `Some features available for early access` + */ + ["com.affine.settings.workspace.experimental-features.header.subtitle"](): string; + /** + * `I am aware of the risks, and I am willing to continue to use it.` + */ + ["com.affine.settings.workspace.experimental-features.prompt-disclaimer"](): string; + /** + * `Do you want to use the plugin system that is in an experimental stage?` + */ + ["com.affine.settings.workspace.experimental-features.prompt-header"](): string; + /** + * `You are about to enable an experimental feature. This feature is still in development and may contain errors or behave unpredictably. Please proceed with caution and at your own risk.` + */ + ["com.affine.settings.workspace.experimental-features.prompt-warning"](): string; + /** + * `WARNING MESSAGE` + */ + ["com.affine.settings.workspace.experimental-features.prompt-warning-title"](): string; + /** + * `Enable AI` + */ + ["com.affine.settings.workspace.experimental-features.enable-ai.name"](): string; + /** + * `Enable or disable ALL AI features.` + */ + ["com.affine.settings.workspace.experimental-features.enable-ai.description"](): string; + /** + * `Enable New DND` + */ + ["com.affine.settings.workspace.experimental-features.enable-new-dnd.name"](): string; + /** + * `Enable new drag and drop features.` + */ + ["com.affine.settings.workspace.experimental-features.enable-new-dnd.description"](): string; + /** + * `Database Full Width` + */ + ["com.affine.settings.workspace.experimental-features.enable-database-full-width.name"](): string; + /** + * `The database will be displayed in full-width mode.` + */ + ["com.affine.settings.workspace.experimental-features.enable-database-full-width.description"](): string; + /** + * `Database Attachment Note` + */ + ["com.affine.settings.workspace.experimental-features.enable-database-attachment-note.name"](): string; + /** + * `Allows adding notes to database attachments.` + */ + ["com.affine.settings.workspace.experimental-features.enable-database-attachment-note.description"](): string; + /** + * `Todo Block Query` + */ + ["com.affine.settings.workspace.experimental-features.enable-block-query.name"](): string; + /** + * `Enables querying of todo blocks.` + */ + ["com.affine.settings.workspace.experimental-features.enable-block-query.description"](): string; + /** + * `Synced Doc Block` + */ + ["com.affine.settings.workspace.experimental-features.enable-synced-doc-block.name"](): string; + /** + * `Enables syncing of doc blocks.` + */ + ["com.affine.settings.workspace.experimental-features.enable-synced-doc-block.description"](): string; + /** + * `Edgeless Text` + */ + ["com.affine.settings.workspace.experimental-features.enable-edgeless-text.name"](): string; + /** + * `Enables edgeless text blocks.` + */ + ["com.affine.settings.workspace.experimental-features.enable-edgeless-text.description"](): string; + /** + * `Color Picker` + */ + ["com.affine.settings.workspace.experimental-features.enable-color-picker.name"](): string; + /** + * `Enables color picker blocks.` + */ + ["com.affine.settings.workspace.experimental-features.enable-color-picker.description"](): string; + /** + * `AI Chat Block` + */ + ["com.affine.settings.workspace.experimental-features.enable-ai-chat-block.name"](): string; + /** + * `Enables AI chat blocks.` + */ + ["com.affine.settings.workspace.experimental-features.enable-ai-chat-block.description"](): string; + /** + * `AI Onboarding` + */ + ["com.affine.settings.workspace.experimental-features.enable-ai-onboarding.name"](): string; + /** + * `Enables AI onboarding.` + */ + ["com.affine.settings.workspace.experimental-features.enable-ai-onboarding.description"](): string; + /** + * `Mind Map Import` + */ + ["com.affine.settings.workspace.experimental-features.enable-mind-map-import.name"](): string; + /** + * `Enables mind map import.` + */ + ["com.affine.settings.workspace.experimental-features.enable-mind-map-import.description"](): string; + /** + * `Split View` + */ + ["com.affine.settings.workspace.experimental-features.enable-multi-view.name"](): string; + /** + * `The Split View feature enables you to divide your tab into multiple sections for simultaneous viewing and editing of different documents.` + */ + ["com.affine.settings.workspace.experimental-features.enable-multi-view.description"](): string; + /** + * `Emoji Folder Icon` + */ + ["com.affine.settings.workspace.experimental-features.enable-emoji-folder-icon.name"](): string; + /** + * `Once enabled, you can use an emoji as the folder icon. When the first character of the folder name is an emoji, it will be extracted and used as its icon.` + */ + ["com.affine.settings.workspace.experimental-features.enable-emoji-folder-icon.description"](): string; + /** + * `Emoji Doc Icon` + */ + ["com.affine.settings.workspace.experimental-features.enable-emoji-doc-icon.name"](): string; + /** + * `Once enabled, you can use an emoji as the doc icon. When the first character of the doc name is an emoji, it will be extracted and used as its icon.` + */ + ["com.affine.settings.workspace.experimental-features.enable-emoji-doc-icon.description"](): string; + /** + * `Editor Settings` + */ + ["com.affine.settings.workspace.experimental-features.enable-editor-settings.name"](): string; + /** + * `Enables editor settings.` + */ + ["com.affine.settings.workspace.experimental-features.enable-editor-settings.description"](): string; + /** + * `Offline Mode` + */ + ["com.affine.settings.workspace.experimental-features.enable-offline-mode.name"](): string; + /** + * `Stop Connecting to the Internet. Even with AFFiNE Cloud, enabling this toggle stops internet connection and keeps everything local, but syncing will be disabled.` + */ + ["com.affine.settings.workspace.experimental-features.enable-offline-mode.description"](): string; + /** + * `Theme Editor` + */ + ["com.affine.settings.workspace.experimental-features.enable-theme-editor.name"](): string; + /** + * `Enables theme editor.` + */ + ["com.affine.settings.workspace.experimental-features.enable-theme-editor.description"](): string; + /** + * `Allow create local workspace` + */ + ["com.affine.settings.workspace.experimental-features.enable-local-workspace.name"](): string; + /** + * `Allow create local workspace` + */ + ["com.affine.settings.workspace.experimental-features.enable-local-workspace.description"](): string; + /** + * `Advanced block visibility control` + */ + ["com.affine.settings.workspace.experimental-features.enable-advanced-block-visibility.name"](): string; + /** + * `To provide detailed control over which edgeless blocks are visible in page mode.` + */ + ["com.affine.settings.workspace.experimental-features.enable-advanced-block-visibility.description"](): string; + /** + * `Mobile Keyboard Toolbar` + */ + ["com.affine.settings.workspace.experimental-features.enable-mobile-keyboard-toolbar.name"](): string; + /** + * `Enables the mobile keyboard toolbar.` + */ + ["com.affine.settings.workspace.experimental-features.enable-mobile-keyboard-toolbar.description"](): string; + /** + * `Mobile Linked Doc Widget` + */ + ["com.affine.settings.workspace.experimental-features.enable-mobile-linked-doc-menu.name"](): string; + /** + * `Enables the mobile linked doc menu.` + */ + ["com.affine.settings.workspace.experimental-features.enable-mobile-linked-doc-menu.description"](): string; + /** + * `Enable Snapshot Import Export` + */ + ["com.affine.settings.workspace.experimental-features.enable-snapshot-import-export.name"](): string; + /** + * `Once enabled, users can import and export blocksuite snapshots.` + */ + ["com.affine.settings.workspace.experimental-features.enable-snapshot-import-export.description"](): string; + /** + * `Multiple Cloud Servers` + */ + ["com.affine.settings.workspace.experimental-features.enable-multiple-cloud-servers.name"](): string; + /** + * `Once enabled, users can connect to selfhosted cloud servers.` + */ + ["com.affine.settings.workspace.experimental-features.enable-multiple-cloud-servers.description"](): string; + /** + * `Enable Edgeless Editing` + */ + ["com.affine.settings.workspace.experimental-features.enable-mobile-edgeless-editing.name"](): string; + /** + * `Once enabled, users can edit edgeless canvas.` + */ + ["com.affine.settings.workspace.experimental-features.enable-mobile-edgeless-editing.description"](): string; + /** + * `Only an owner can edit the workspace avatar and name. Changes will be shown for everyone.` + */ + ["com.affine.settings.workspace.not-owner"](): string; + /** + * `Preference` + */ + ["com.affine.settings.workspace.preferences"](): string; + /** + * `Billing` + */ + ["com.affine.settings.workspace.billing"](): string; + /** + * `Team Workspace` + */ + ["com.affine.settings.workspace.billing.team-workspace"](): string; + /** + * `Your workspace is in a free trail period.` + */ + ["com.affine.settings.workspace.billing.team-workspace.description.free-trail"](): string; + /** + * `Your workspace is billed annually.` + */ + ["com.affine.settings.workspace.billing.team-workspace.description.billed.annually"](): string; + /** + * `Your workspace is billed monthly.` + */ + ["com.affine.settings.workspace.billing.team-workspace.description.billed.monthly"](): string; + /** + * `Your subscription will end on {{date}}` + */ + ["com.affine.settings.workspace.billing.team-workspace.not-renewed"](options: { + readonly date: string; + }): string; + /** + * `Next billing date: {{date}}` + */ + ["com.affine.settings.workspace.billing.team-workspace.next-billing-date"](options: { + readonly date: string; + }): string; + /** + * `Cancel Plan` + */ + ["com.affine.settings.workspace.billing.team-workspace.cancel-plan"](): string; + /** + * `Local` + */ + ["com.affine.settings.workspace.state.local"](): string; + /** + * `Sync with AFFiNE Cloud` + */ + ["com.affine.settings.workspace.state.sync-affine-cloud"](): string; + /** + * `Self-Hosted Server` + */ + ["com.affine.settings.workspace.state.self-hosted"](): string; + /** + * `Joined Workspace` + */ + ["com.affine.settings.workspace.state.joined"](): string; + /** + * `Available Offline` + */ + ["com.affine.settings.workspace.state.available-offline"](): string; + /** + * `Published to Web` + */ + ["com.affine.settings.workspace.state.published"](): string; + /** + * `Team Workspace` + */ + ["com.affine.settings.workspace.state.team"](): string; + /** + * `Properties` + */ + ["com.affine.settings.workspace.properties"](): string; + /** + * `Add property` + */ + ["com.affine.settings.workspace.properties.add_property"](): string; + /** + * `All` + */ + ["com.affine.settings.workspace.properties.all"](): string; + /** + * `Delete property` + */ + ["com.affine.settings.workspace.properties.delete-property"](): string; + /** + * `Edit property` + */ + ["com.affine.settings.workspace.properties.edit-property"](): string; + /** + * `General properties` + */ + ["com.affine.settings.workspace.properties.general-properties"](): string; + /** + * `Properties` + */ + ["com.affine.settings.workspace.properties.header.title"](): string; + /** + * `In use` + */ + ["com.affine.settings.workspace.properties.in-use"](): string; + /** + * `Readonly properties` + */ + ["com.affine.settings.workspace.properties.readonly-properties"](): string; + /** + * `Required properties` + */ + ["com.affine.settings.workspace.properties.required-properties"](): string; + /** + * `Set as required property` + */ + ["com.affine.settings.workspace.properties.set-as-required"](): string; + /** + * `Unused` + */ + ["com.affine.settings.workspace.properties.unused"](): string; + /** + * `Enable AFFiNE Cloud to publish this workspace` + */ + ["com.affine.settings.workspace.publish-tooltip"](): string; + /** + * `Sharing` + */ + ["com.affine.settings.workspace.sharing.title"](): string; + /** + * `Allow URL unfurling by Slack & other social apps, even if a doc is only accessible by workspace members.` + */ + ["com.affine.settings.workspace.sharing.url-preview.description"](): string; + /** + * `Always enable url preview` + */ + ["com.affine.settings.workspace.sharing.url-preview.title"](): string; + /** + * `AFFiNE AI` + */ + ["com.affine.settings.workspace.affine-ai.title"](): string; + /** + * `Allow AFFiNE AI Assistant` + */ + ["com.affine.settings.workspace.affine-ai.label"](): string; + /** + * `Allow workspace members to use AFFiNE AI features. This setting doesn't affect billing. Workspace members use AFFiNE AI through their personal accounts.` + */ + ["com.affine.settings.workspace.affine-ai.description"](): string; + /** + * `Sharing doc requires AFFiNE Cloud.` + */ + ["com.affine.share-menu.EnableCloudDescription"](): string; + /** + * `Share mode` + */ + ["com.affine.share-menu.ShareMode"](): string; + /** + * `Share doc` + */ + ["com.affine.share-menu.SharePage"](): string; + /** + * `Share via export` + */ + ["com.affine.share-menu.ShareViaExport"](): string; + /** + * `Download a static copy of your doc to share with others` + */ + ["com.affine.share-menu.ShareViaExportDescription"](): string; + /** + * `Print a paper copy` + */ + ["com.affine.share-menu.ShareViaPrintDescription"](): string; + /** + * `Share with link` + */ + ["com.affine.share-menu.ShareWithLink"](): string; + /** + * `Create a link you can easily share with anyone. The visitors will open your doc in the form od a document` + */ + ["com.affine.share-menu.ShareWithLinkDescription"](): string; + /** + * `Shared doc` + */ + ["com.affine.share-menu.SharedPage"](): string; + /** + * `Please try again later.` + */ + ["com.affine.share-menu.confirm-modify-mode.notification.fail.message"](): string; + /** + * `Failed to modify` + */ + ["com.affine.share-menu.confirm-modify-mode.notification.fail.title"](): string; + /** + * `Copy Link` + */ + ["com.affine.share-menu.copy"](): string; + /** + * `Copy private link` + */ + ["com.affine.share-menu.copy-private-link"](): string; + /** + * `Copy Link to Selected Block` + */ + ["com.affine.share-menu.copy.block"](): string; + /** + * `Copy Link to Edgeless Mode` + */ + ["com.affine.share-menu.copy.edgeless"](): string; + /** + * `Copy Link to Selected Frame` + */ + ["com.affine.share-menu.copy.frame"](): string; + /** + * `Copy Link to Page Mode` + */ + ["com.affine.share-menu.copy.page"](): string; + /** + * `You can share this document with link.` + */ + ["com.affine.share-menu.create-public-link.notification.success.message"](): string; + /** + * `Public link created` + */ + ["com.affine.share-menu.create-public-link.notification.success.title"](): string; + /** + * `Please try again later.` + */ + ["com.affine.share-menu.disable-publish-link.notification.fail.message"](): string; + /** + * `Failed to disable public link` + */ + ["com.affine.share-menu.disable-publish-link.notification.fail.title"](): string; + /** + * `This doc is no longer shared publicly.` + */ + ["com.affine.share-menu.disable-publish-link.notification.success.message"](): string; + /** + * `Public link disabled` + */ + ["com.affine.share-menu.disable-publish-link.notification.success.title"](): string; + /** + * `Manage workspace members` + */ + ["com.affine.share-menu.navigate.workspace"](): string; + /** + * `Anyone with the link` + */ + ["com.affine.share-menu.option.link.label"](): string; + /** + * `No Access` + */ + ["com.affine.share-menu.option.link.no-access"](): string; + /** + * `Only workspace members can access this link` + */ + ["com.affine.share-menu.option.link.no-access.description"](): string; + /** + * `Read Only` + */ + ["com.affine.share-menu.option.link.readonly"](): string; + /** + * `Anyone can access this link` + */ + ["com.affine.share-menu.option.link.readonly.description"](): string; + /** + * `Can Edit` + */ + ["com.affine.share-menu.option.permission.can-edit"](): string; + /** + * `Members in workspace` + */ + ["com.affine.share-menu.option.permission.label"](): string; + /** + * `Publish to web` + */ + ["com.affine.share-menu.publish-to-web"](): string; + /** + * `Share privately` + */ + ["com.affine.share-menu.share-privately"](): string; + /** + * `Share` + */ + ["com.affine.share-menu.shareButton"](): string; + /** + * `Shared` + */ + ["com.affine.share-menu.sharedButton"](): string; + /** + * `Built with` + */ + ["com.affine.share-page.footer.built-with"](): string; + /** + * `Create with` + */ + ["com.affine.share-page.footer.create-with"](): string; + /** + * `Empower your sharing with AFFiNE Cloud: One-click doc sharing` + */ + ["com.affine.share-page.footer.description"](): string; + /** + * `Get started for free` + */ + ["com.affine.share-page.footer.get-started"](): string; + /** + * `Use This Template` + */ + ["com.affine.share-page.header.import-template"](): string; + /** + * `Login or Sign Up` + */ + ["com.affine.share-page.header.login"](): string; + /** + * `Present` + */ + ["com.affine.share-page.header.present"](): string; + /** + * `Edgeless` + */ + ["com.affine.shortcutsTitle.edgeless"](): string; + /** + * `General` + */ + ["com.affine.shortcutsTitle.general"](): string; + /** + * `Markdown syntax` + */ + ["com.affine.shortcutsTitle.markdownSyntax"](): string; + /** + * `Page` + */ + ["com.affine.shortcutsTitle.page"](): string; + /** + * `Collapse sidebar` + */ + ["com.affine.sidebarSwitch.collapse"](): string; + /** + * `Expand sidebar` + */ + ["com.affine.sidebarSwitch.expand"](): string; + /** + * `Snapshot Imp. & Exp.` + */ + ["com.affine.snapshot.import-export.enable"](): string; + /** + * `Once enabled you can find the Snapshot Export Import option in the document's More menu.` + */ + ["com.affine.snapshot.import-export.enable.desc"](): string; + /** + * `Maybe later` + */ + ["com.affine.star-affine.cancel"](): string; + /** + * `Star on GitHub` + */ + ["com.affine.star-affine.confirm"](): string; + /** + * `Are you finding our app useful and enjoyable? We'd love your support to keep improving! A great way to help us out is by giving us a star on GitHub. This simple action can make a big difference and helps us continue to deliver the best experience for you.` + */ + ["com.affine.star-affine.description"](): string; + /** + * `Star us on GitHub` + */ + ["com.affine.star-affine.title"](): string; + /** + * `Change plan` + */ + ["com.affine.storage.change-plan"](): string; + /** + * `You have reached the maximum capacity limit for your current account` + */ + ["com.affine.storage.maximum-tips"](): string; + /** + * `Pro users will have unlimited storage capacity during the alpha test period of the team version` + */ + ["com.affine.storage.maximum-tips.pro"](): string; + /** + * `Plan` + */ + ["com.affine.storage.plan"](): string; + /** + * `AFFiNE Cloud storage` + */ + ["com.affine.storage.title"](): string; + /** + * `Upgrade` + */ + ["com.affine.storage.upgrade"](): string; + /** + * `Space used` + */ + ["com.affine.storage.used.hint"](): string; + /** + * `Syncing` + */ + ["com.affine.syncing"](): string; + /** + * `{{count}} doc` + + * - com.affine.tags.count_one: `{{count}} doc` + + * - com.affine.tags.count_other: `{{count}} docs` + + * - com.affine.tags.count_zero: `{{count}} doc` + */ + ["com.affine.tags.count"](options: { + readonly count: string | number | bigint; + }): string; + /** + * `{{count}} doc` + */ + ["com.affine.tags.count_one"](options: { + readonly count: string | number | bigint; + }): string; + /** + * `{{count}} docs` + */ + ["com.affine.tags.count_other"](options: { + readonly count: string | number | bigint; + }): string; + /** + * `{{count}} doc` + */ + ["com.affine.tags.count_zero"](options: { + readonly count: string | number | bigint; + }): string; + /** + * `Type tag name here...` + */ + ["com.affine.tags.create-tag.placeholder"](): string; + /** + * `Tag already exists` + */ + ["com.affine.tags.create-tag.toast.exist"](): string; + /** + * `Tag created` + */ + ["com.affine.tags.create-tag.toast.success"](): string; + /** + * `Tag deleted` + */ + ["com.affine.tags.delete-tags.toast"](): string; + /** + * `Tag updated` + */ + ["com.affine.tags.edit-tag.toast.success"](): string; + /** + * `New tag` + */ + ["com.affine.tags.empty.new-tag-button"](): string; + /** + * `Enable telemetry` + */ + ["com.affine.telemetry.enable"](): string; + /** + * `Telemetry is a feature that allows us to collect data on how you use the app. This data helps us improve the app and provide better features.` + */ + ["com.affine.telemetry.enable.desc"](): string; + /** + * `Dark` + */ + ["com.affine.themeSettings.dark"](): string; + /** + * `Light` + */ + ["com.affine.themeSettings.light"](): string; + /** + * `System` + */ + ["com.affine.themeSettings.system"](): string; + /** + * `Auto` + */ + ["com.affine.themeSettings.auto"](): string; + /** + * `now` + */ + ["com.affine.time.now"](): string; + /** + * `this month` + */ + ["com.affine.time.this-mouth"](): string; + /** + * `this week` + */ + ["com.affine.time.this-week"](): string; + /** + * `this year` + */ + ["com.affine.time.this-year"](): string; + /** + * `today` + */ + ["com.affine.time.today"](): string; + /** + * `Successfully added linked doc` + */ + ["com.affine.toastMessage.addLinkedPage"](): string; + /** + * `Added to favorites` + */ + ["com.affine.toastMessage.addedFavorites"](): string; + /** + * `Edgeless mode` + */ + ["com.affine.toastMessage.edgelessMode"](): string; + /** + * `Moved to trash` + */ + ["com.affine.toastMessage.movedTrash"](): string; + /** + * `Page Mode` + */ + ["com.affine.toastMessage.pageMode"](): string; + /** + * `Default mode has changed` + */ + ["com.affine.toastMessage.defaultMode.page.title"](): string; + /** + * `The default mode for this document has been changed to Page mode` + */ + ["com.affine.toastMessage.defaultMode.page.message"](): string; + /** + * `Default mode has changed` + */ + ["com.affine.toastMessage.defaultMode.edgeless.title"](): string; + /** + * `The default mode for this document has been changed to Edgeless mode` + */ + ["com.affine.toastMessage.defaultMode.edgeless.message"](): string; + /** + * `Permanently deleted` + */ + ["com.affine.toastMessage.permanentlyDeleted"](): string; + /** + * `Removed from favourites` + */ + ["com.affine.toastMessage.removedFavorites"](): string; + /** + * `Successfully renamed` + */ + ["com.affine.toastMessage.rename"](): string; + /** + * `{{title}} restored` + */ + ["com.affine.toastMessage.restored"](options: { + readonly title: string; + }): string; + /** + * `Successfully deleted` + */ + ["com.affine.toastMessage.successfullyDeleted"](): string; + /** + * `Today` + */ + ["com.affine.today"](): string; + /** + * `Tomorrow` + */ + ["com.affine.tomorrow"](): string; + /** + * `Last {{weekday}}` + */ + ["com.affine.last-week"](options: { + readonly weekday: string; + }): string; + /** + * `Next {{weekday}}` + */ + ["com.affine.next-week"](options: { + readonly weekday: string; + }): string; + /** + * `Limited to view-only on mobile.` + */ + ["com.affine.top-tip.mobile"](): string; + /** + * `Delete` + */ + ["com.affine.trashOperation.delete"](): string; + /** + * `Once deleted, you can't undo this action. Do you confirm?` + */ + ["com.affine.trashOperation.delete.description"](): string; + /** + * `Permanently delete` + */ + ["com.affine.trashOperation.delete.title"](): string; + /** + * `Once deleted, you can't undo this action. Do you confirm?` + */ + ["com.affine.trashOperation.deleteDescription"](): string; + /** + * `Delete permanently` + */ + ["com.affine.trashOperation.deletePermanently"](): string; + /** + * `Restore it` + */ + ["com.affine.trashOperation.restoreIt"](): string; + /** + * `Refresh current page` + */ + ["com.affine.upgrade.button-text.done"](): string; + /** + * `Data upgrade error` + */ + ["com.affine.upgrade.button-text.error"](): string; + /** + * `Upgrade workspace data` + */ + ["com.affine.upgrade.button-text.pending"](): string; + /** + * `Upgrading` + */ + ["com.affine.upgrade.button-text.upgrading"](): string; + /** + * `After upgrading the workspace data, please refresh the page to see the changes.` + */ + ["com.affine.upgrade.tips.done"](): string; + /** + * `We encountered some errors while upgrading the workspace data.` + */ + ["com.affine.upgrade.tips.error"](): string; + /** + * `To ensure compatibility with the updated AFFiNE client, please upgrade your data by clicking the "Upgrade workspace data" button below.` + */ + ["com.affine.upgrade.tips.normal"](): string; + /** + * `AI usage` + */ + ["com.affine.user-info.usage.ai"](): string; + /** + * `Cloud storage` + */ + ["com.affine.user-info.usage.cloud"](): string; + /** + * `Close` + */ + ["com.affine.workbench.split-view-menu.close"](): string; + /** + * `Full screen` + */ + ["com.affine.workbench.split-view-menu.full-screen"](): string; + /** + * `Solo view` + */ + ["com.affine.workbench.split-view-menu.keep-this-one"](): string; + /** + * `Move left` + */ + ["com.affine.workbench.split-view-menu.move-left"](): string; + /** + * `Move right` + */ + ["com.affine.workbench.split-view-menu.move-right"](): string; + /** + * `Open in split view` + */ + ["com.affine.workbench.split-view.page-menu-open"](): string; + /** + * `Open in new tab` + */ + ["com.affine.workbench.tab.page-menu-open"](): string; + /** + * `You cannot delete the last workspace` + */ + ["com.affine.workspace.cannot-delete"](): string; + /** + * `Cloud workspaces` + */ + ["com.affine.workspace.cloud"](): string; + /** + * `Sign out` + */ + ["com.affine.workspace.cloud.account.logout"](): string; + /** + * `Account settings` + */ + ["com.affine.workspace.cloud.account.settings"](): string; + /** + * `Sign up/ Sign in` + */ + ["com.affine.workspace.cloud.auth"](): string; + /** + * `Sync with AFFiNE Cloud` + */ + ["com.affine.workspace.cloud.description"](): string; + /** + * `Join workspace` + */ + ["com.affine.workspace.cloud.join"](): string; + /** + * `Cloud sync` + */ + ["com.affine.workspace.cloud.sync"](): string; + /** + * `Failed to enable Cloud, please try again.` + */ + ["com.affine.workspace.enable-cloud.failed"](): string; + /** + * `Local workspaces` + */ + ["com.affine.workspace.local"](): string; + /** + * `Import workspace` + */ + ["com.affine.workspace.local.import"](): string; + /** + * `Cancel` + */ + ["com.affine.workspaceDelete.button.cancel"](): string; + /** + * `Delete` + */ + ["com.affine.workspaceDelete.button.delete"](): string; + /** + * `Please type workspace name to confirm` + */ + ["com.affine.workspaceDelete.placeholder"](): string; + /** + * `Delete workspace` + */ + ["com.affine.workspaceDelete.title"](): string; + /** + * `Create workspace` + */ + ["com.affine.workspaceList.addWorkspace.create"](): string; + /** + * `Create cloud workspace` + */ + ["com.affine.workspaceList.addWorkspace.create-cloud"](): string; + /** + * `Cloud sync` + */ + ["com.affine.workspaceList.workspaceListType.cloud"](): string; + /** + * `Local storage` + */ + ["com.affine.workspaceList.workspaceListType.local"](): string; + /** + * `Add Server` + */ + ["com.affine.workspaceList.addServer"](): string; + /** + * `All docs` + */ + ["com.affine.workspaceSubPath.all"](): string; + /** + * `Trash` + */ + ["com.affine.workspaceSubPath.trash"](): string; + /** + * `Deleted docs will appear here.` + */ + ["com.affine.workspaceSubPath.trash.empty-description"](): string; + /** + * `Write with a blank page` + */ + ["com.affine.write_with_a_blank_page"](): string; + /** + * `Yesterday` + */ + ["com.affine.yesterday"](): string; + /** + * `core` + */ + core(): string; + /** + * `Dark` + */ + dark(): string; + /** + * `invited you to join` + */ + ["invited you to join"](): string; + /** + * `Light` + */ + light(): string; + /** + * `Others` + */ + others(): string; + /** + * `System` + */ + system(): string; + /** + * `unnamed` + */ + unnamed(): string; + /** + * `Please upgrade to the latest version of Chrome for the best experience.` + */ + upgradeBrowser(): string; + /** + * `Workspace properties` + */ + ["com.affine.workspace.properties"](): string; + /** + * `Rename to "{{name}}"` + */ + ["com.affine.m.rename-to"](options: { + readonly name: string; + }): string; + /** + * `Rename` + */ + ["com.affine.m.explorer.folder.rename"](): string; + /** + * `Create Folder` + */ + ["com.affine.m.explorer.folder.new-dialog-title"](): string; + /** + * `Organize` + */ + ["com.affine.m.explorer.folder.root"](): string; + /** + * `Create a folder in the {{parent}}.` + */ + ["com.affine.m.explorer.folder.new-tip-empty"](options: { + readonly parent: string; + }): string; + /** + * `Create "{{value}}" in the {{parent}}.` + */ + ["com.affine.m.explorer.folder.new-tip-not-empty"](options: Readonly<{ + value: string; + parent: string; + }>): string; + /** + * `Done` + */ + ["com.affine.m.explorer.folder.rename-confirm"](): string; + /** + * `Rename` + */ + ["com.affine.m.explorer.tag.rename"](): string; + /** + * `Rename Tag` + */ + ["com.affine.m.explorer.tag.rename-menu-title"](): string; + /** + * `Create Tag` + */ + ["com.affine.m.explorer.tag.new-dialog-title"](): string; + /** + * `Done` + */ + ["com.affine.m.explorer.tag.rename-confirm"](): string; + /** + * `Create a tag in this workspace.` + */ + ["com.affine.m.explorer.tag.new-tip-empty"](): string; + /** + * `Create "{{value}}" tag in this workspace.` + */ + ["com.affine.m.explorer.tag.new-tip-not-empty"](options: { + readonly value: string; + }): string; + /** + * `Manage Doc(s)` + */ + ["com.affine.m.explorer.tag.manage-docs"](): string; + /** + * `Rename` + */ + ["com.affine.m.explorer.collection.rename"](): string; + /** + * `Rename Collection` + */ + ["com.affine.m.explorer.collection.rename-menu-title"](): string; + /** + * `Create Collection` + */ + ["com.affine.m.explorer.collection.new-dialog-title"](): string; + /** + * `Rename` + */ + ["com.affine.m.explorer.doc.rename"](): string; + /** + * `Doc` + */ + ["com.affine.m.selector.type-doc"](): string; + /** + * `Tag` + */ + ["com.affine.m.selector.type-tag"](): string; + /** + * `Collection` + */ + ["com.affine.m.selector.type-collection"](): string; + /** + * `Folder` + */ + ["com.affine.m.selector.where-folder"](): string; + /** + * `Tag` + */ + ["com.affine.m.selector.where-tag"](): string; + /** + * `Collection` + */ + ["com.affine.m.selector.where-collection"](): string; + /** + * `Apply` + */ + ["com.affine.m.selector.confirm-default"](): string; + /** + * `Manage {{type}}(s)` + */ + ["com.affine.m.selector.title"](options: { + readonly type: string; + }): string; + /** + * `{{total}} item(s)` + */ + ["com.affine.m.selector.info-total"](options: { + readonly total: string; + }): string; + /** + * `Add {{count}} {{type}}(s)` + */ + ["com.affine.m.selector.info-added"](options: Readonly<{ + count: string; + type: string; + }>): string; + /** + * `Remove {{count}} {{type}}(s)` + */ + ["com.affine.m.selector.info-removed"](options: Readonly<{ + count: string; + type: string; + }>): string; + /** + * `Remove items` + */ + ["com.affine.m.selector.remove-warning.title"](): string; + /** + * `You unchecked {{type}} that already exist in the current {{where}}, which means you will remove them from this {{where}}. The item will not be deleted.` + */ + ["com.affine.m.selector.remove-warning.message"](options: Readonly<{ + type: string; + where: string; + }>): string; + /** + * `Do not ask again` + */ + ["com.affine.m.selector.remove-warning.confirm"](): string; + /** + * `Cancel` + */ + ["com.affine.m.selector.remove-warning.cancel"](): string; + /** + * `tag` + */ + ["com.affine.m.selector.remove-warning.where-tag"](): string; + /** + * `folder` + */ + ["com.affine.m.selector.remove-warning.where-folder"](): string; + /** + * `Today's activity` + */ + ["com.affine.m.selector.journal-menu.today-activity"](): string; + /** + * `Duplicate Entries in Today's Journal` + */ + ["com.affine.m.selector.journal-menu.conflicts"](): string; + /** + * `Unable to preview this file` + */ + ["com.affine.attachment.preview.error.title"](): string; + /** + * `file type not supported.` + */ + ["com.affine.attachment.preview.error.subtitle"](): string; + /** + * `Failed to render page.` + */ + ["com.affine.pdf.page.render.error"](): string; + /** + * `Duplicate Entries in Today's Journal` + */ + ["com.affine.editor.journal-conflict.title"](): string; + /** + * `Search for "{{query}}"` + */ + ["com.affine.editor.at-menu.link-to-doc"](options: { + readonly query: string; + }): string; + /** + * `Recent` + */ + ["com.affine.editor.at-menu.recent-docs"](): string; + /** + * `New` + */ + ["com.affine.editor.at-menu.new-doc"](): string; + /** + * `New "{{name}}" page` + */ + ["com.affine.editor.at-menu.create-page"](options: { + readonly name: string; + }): string; + /** + * `New "{{name}}" edgeless` + */ + ["com.affine.editor.at-menu.create-edgeless"](options: { + readonly name: string; + }): string; + /** + * `Import` + */ + ["com.affine.editor.at-menu.import"](): string; + /** + * `{{count}} more docs` + */ + ["com.affine.editor.at-menu.more-docs-hint"](options: { + readonly count: string; + }): string; + /** + * `Journal` + */ + ["com.affine.editor.at-menu.journal"](): string; + /** + * `Select a specific date` + */ + ["com.affine.editor.at-menu.date-picker"](): string; + /** + * `Show` + */ + ["com.affine.editor.bi-directional-link-panel.show"](): string; + /** + * `Hide` + */ + ["com.affine.editor.bi-directional-link-panel.hide"](): string; + /** + * `Empower Your Team with Seamless Collaboration` + */ + ["com.affine.upgrade-to-team-page.title"](): string; + /** + * `Select an existing workspace or create a new one` + */ + ["com.affine.upgrade-to-team-page.workspace-selector.placeholder"](): string; + /** + * `Create Workspace` + */ + ["com.affine.upgrade-to-team-page.workspace-selector.create-workspace"](): string; + /** + * `Upgrade to Team Workspace` + */ + ["com.affine.upgrade-to-team-page.upgrade-button"](): string; + /** + * `Team Workspace gives you everything you need for seamless team collaboration:` + */ + ["com.affine.upgrade-to-team-page.benefit.title"](): string; + /** + * `Invite unlimited members to your workspace` + */ + ["com.affine.upgrade-to-team-page.benefit.g1"](): string; + /** + * `Set custom roles and permissions for better control` + */ + ["com.affine.upgrade-to-team-page.benefit.g2"](): string; + /** + * `Access advanced team management features` + */ + ["com.affine.upgrade-to-team-page.benefit.g3"](): string; + /** + * `Get priority customer support` + */ + ["com.affine.upgrade-to-team-page.benefit.g4"](): string; + /** + * `Perfect for growing teams and organizations that need professional collaboration tools.` + */ + ["com.affine.upgrade-to-team-page.benefit.description"](): string; + /** + * `Upgrade to Team Workspace` + */ + ["com.affine.upgrade-to-team-page.upgrade-confirm.title"](): string; + /** + * `Name Your Workspace` + */ + ["com.affine.upgrade-to-team-page.create-and-upgrade-confirm.title"](): string; + /** + * `A workspace is your virtual space to capture, create and plan as just one person or together as a team.` + */ + ["com.affine.upgrade-to-team-page.create-and-upgrade-confirm.description"](): string; + /** + * `Set a workspace name` + */ + ["com.affine.upgrade-to-team-page.create-and-upgrade-confirm.placeholder"](): string; + /** + * `Continue to Pricing` + */ + ["com.affine.upgrade-to-team-page.create-and-upgrade-confirm.confirm"](): string; + /** + * `No workspace available` + */ + ["com.affine.upgrade-to-team-page.no-workspace-available"](): string; + /** + * `Workspace storage` + */ + ["com.affine.workspace.storage"](): string; + /** + * `Journal` + */ + ["com.affine.cmdk.affine.category.affine.journal"](): string; + /** + * `Select a specific date` + */ + ["com.affine.cmdk.affine.category.affine.date-picker"](): string; + /** + * `Workspace sync paused` + */ + ["com.affine.payment.sync-paused.title"](): string; + /** + * `Your workspace has exceeded both storage and member limits, causing synchronization to pause. To resume syncing, please either:` + */ + ["com.affine.payment.sync-paused.owner.both.description"](): string; + /** + * `Reduce storage usage and remove some team members` + */ + ["com.affine.payment.sync-paused.owner.both.tips-1"](): string; + /** + * `Upgrade your plan for increased capacity` + */ + ["com.affine.payment.sync-paused.owner.both.tips-2"](): string; + /** + * `Your workspace has exceeded its storage limit and synchronization has been paused. To resume syncing, please either:` + */ + ["com.affine.payment.sync-paused.owner.storage.description"](): string; + /** + * `Remove unnecessary files or content to reduce storage usage` + */ + ["com.affine.payment.sync-paused.owner.storage.tips-1"](): string; + /** + * `Upgrade your plan for increased storage capacity` + */ + ["com.affine.payment.sync-paused.owner.storage.tips-2"](): string; + /** + * `Your workspace has reached its maximum member capacity and synchronization has been paused. To resume syncing, you can either` + */ + ["com.affine.payment.sync-paused.owner.member.description"](): string; + /** + * `Remove some team members from the workspace` + */ + ["com.affine.payment.sync-paused.owner.member.tips-1"](): string; + /** + * `Upgrade your plan to accommodate more members` + */ + ["com.affine.payment.sync-paused.owner.member.tips-2"](): string; + /** + * `This workspace has exceeded both storage and member limits, causing synchronization to pause. Please contact your workspace owner to address these limits and resume syncing.` + */ + ["com.affine.payment.sync-paused.member.both.description"](): string; + /** + * `This workspace has exceeded its storage limit and synchronization has been paused. Please contact your workspace owner to either reduce storage usage or upgrade the plan to resume syncing.` + */ + ["com.affine.payment.sync-paused.member.storage.description"](): string; + /** + * `This workspace has reached its maximum member capacity and synchronization has been paused. Please contact your workspace owner to either adjust team membership or upgrade the plan to resume syncing.` + */ + ["com.affine.payment.sync-paused.member.member.description"](): string; + /** + * `Got It` + */ + ["com.affine.payment.sync-paused.member.member.confirm"](): string; +} { const { t } = useTranslation(); return useMemo(() => createProxy((key) => t.bind(null, key)), [t]); } +function createComponent(i18nKey: string) { + return (props) => createElement(Trans, { i18nKey, shouldUnescape: true, ...props }); +} +export const TypedTrans: { + /** + * `Go to {{link}} for learn more details about AFFiNE AI.` + */ + ["com.affine.ai-onboarding.general.5.description"]: ComponentType>; + /** + * `By continuing, you are agreeing to our AI Terms.` + */ + ["com.affine.ai-onboarding.general.privacy"]: ComponentType, { + a: JSX.Element; + }>>; + /** + * `Opening <1>AFFiNE app now` + */ + ["com.affine.auth.open.affine.prompt"]: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `This doc is now opened in <1>AFFiNE app` + */ + ["com.affine.auth.open.affine.open-doc-prompt"]: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `Or <1>sign in with password instead.` + */ + ["com.affine.auth.sign.auth.code.message.password"]: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `The Self-Hosted instance is not hosted or deployed by AFFiNE. Your data will be stored on these instances. <1>Learn more about Self-Host details.` + */ + ["com.affine.auth.sign.add-selfhosted.description"]: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `By clicking “Continue with Google/Email” above, you acknowledge that you agree to AFFiNE's <1>Terms of Conditions and <3>Privacy Policy.` + */ + ["com.affine.auth.sign.message"]: ComponentType, { + ["1"]: JSX.Element; + ["3"]: JSX.Element; + }>>; + /** + * `An email with a magic link has been sent to {{email}}.` + */ + ["com.affine.auth.sign.sent.email.message.sent-tips"]: ComponentType>; + /** + * `This demo is limited. <1>Download the AFFiNE Client for the latest features and Performance.` + */ + ["com.affine.banner.content"]: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `<0>{{count}} selected` + + * - com.affine.collection.toolbar.selected_one: `<0>{{count}} collection selected` + + * - com.affine.collection.toolbar.selected_other: `<0>{{count}} collection(s) selected` + */ + ["com.affine.collection.toolbar.selected"]: ComponentType>; + /** + * `<0>{{count}} collection selected` + */ + ["com.affine.collection.toolbar.selected_one"]: ComponentType>; + /** + * `<0>{{count}} collection(s) selected` + */ + ["com.affine.collection.toolbar.selected_other"]: ComponentType>; + /** + * `<0>{{count}} collection(s) selected` + */ + ["com.affine.collection.toolbar.selected_others"]: ComponentType>; + /** + * `Deleting <1>{{tag}} cannot be undone, please proceed with caution.` + */ + ["com.affine.delete-tags.confirm.description"]: ComponentType>; + /** + * `Selected <1>{{selectedCount}}, filtered <3>{{filteredCount}}` + */ + ["com.affine.editCollection.rules.countTips"]: ComponentType, { + ["1"]: JSX.Element; + ["3"]: JSX.Element; + }>>; + /** + * `Showing <1>{{count}} docs.` + */ + ["com.affine.editCollection.rules.countTips.more"]: ComponentType>; + /** + * `Showing <1>{{count}} doc.` + */ + ["com.affine.editCollection.rules.countTips.one"]: ComponentType>; + /** + * `Showing <1>{{count}} docs.` + */ + ["com.affine.editCollection.rules.countTips.zero"]: ComponentType>; + /** + * `Please <1>add rules to save this collection or switch to <3>Docs, use manual selection mode` + */ + ["com.affine.editCollection.rules.empty.noRules.tips"]: ComponentType, { + ["1"]: JSX.Element; + ["3"]: JSX.Element; + }>>; + /** + * `Docs that meet the rules will be added to the current collection <2>{{highlight}}` + */ + ["com.affine.editCollection.rules.tips"]: ComponentType>; + /** + * `With the workspace creator's free account, every member can access up to <1>7 days<1> of version history.` + */ + ["com.affine.history.confirm-restore-modal.free-plan-prompt.description"]: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `With the workspace creator's Pro account, every member enjoys the privilege of accessing up to <1>30 days<1> of version history.` + */ + ["com.affine.history.confirm-restore-modal.pro-plan-prompt.description"]: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `<0>{{count}} selected` + + * - com.affine.page.toolbar.selected_one: `<0>{{count}} doc selected` + + * - com.affine.page.toolbar.selected_other: `<0>{{count}} doc(s) selected` + */ + ["com.affine.page.toolbar.selected"]: ComponentType>; + /** + * `<0>{{count}} doc selected` + */ + ["com.affine.page.toolbar.selected_one"]: ComponentType>; + /** + * `<0>{{count}} doc(s) selected` + */ + ["com.affine.page.toolbar.selected_other"]: ComponentType>; + /** + * `<0>{{count}} doc(s) selected` + */ + ["com.affine.page.toolbar.selected_others"]: ComponentType>; + /** + * `You are currently on the free plan.` + */ + ["com.affine.payment.billing-setting.ai.free-desc"]: ComponentType, { + a: JSX.Element; + }>>; + /** + * `You have purchased Believer plan. Enjoy with your benefits!` + */ + ["com.affine.payment.billing-setting.believer.description"]: ComponentType, { + a: JSX.Element; + }>>; + /** + * `You are currently on the <1>{{planName}} plan.` + */ + ["com.affine.payment.billing-setting.current-plan.description"]: ComponentType>; + /** + * `You are currently on the believer <1>{{planName}} plan.` + */ + ["com.affine.payment.billing-setting.current-plan.description.lifetime"]: ComponentType>; + /** + * `You are currently on the monthly <1>{{planName}} plan.` + */ + ["com.affine.payment.billing-setting.current-plan.description.monthly"]: ComponentType>; + /** + * `You are currently on the annually <1>{{planName}} plan.` + */ + ["com.affine.payment.billing-setting.current-plan.description.yearly"]: ComponentType>; + /** + * `One-time Purchase. Personal use rights for up to 150 years. Fair Usage Policies may apply.` + */ + ["com.affine.payment.lifetime.caption-2"]: ComponentType, { + a: JSX.Element; + }>>; + /** + * `You are currently on the {{currentPlan}} plan. If you have any questions, please contact our <3>customer support.` + */ + ["com.affine.payment.subtitle-active"]: ComponentType>; + /** + * `If you have any questions, please contact our <1> customer support.` + */ + ["com.affine.payment.upgrade-success-page.support"]: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `If you have any questions, please contact our <1>customer support.` + */ + ["com.affine.payment.upgrade-success-page.team.text-2"]: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `This action deletes the old Favorites section. Your documents are safe, ensure you've moved your frequently accessed documents to the new personal Favorites section.` + */ + ["com.affine.rootAppSidebar.migration-data.clean-all.description"]: ComponentType, { + b: JSX.Element; + }>>; + /** + * `Your documents are safe, but you'll need to re-pin your most-used ones. "Favorites" are now personal. Move items from the old shared section to your new personal section or remove the old one by clicking "Empty the old favorites" now.` + */ + ["com.affine.rootAppSidebar.migration-data.help.description"]: ComponentType, { + b: JSX.Element; + }>>; + /** + * `No doc titles contain <1>{{search}}` + */ + ["com.affine.selectPage.empty.tips"]: ComponentType>; + /** + * `Don't have the app? <1>Click to download.` + */ + ["com.affine.open-in-app.card.subtitle"]: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `Settings changed; please restart the app. <1>Restart` + */ + ["com.affine.settings.editorSettings.general.spell-check.restart-hint"]: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `Love our app? <1>Star us on GitHub and <2>create issues for your valuable feedback!` + */ + ["com.affine.settings.suggestion-2"]: ComponentType, { + ["1"]: JSX.Element; + ["2"]: JSX.Element; + }>>; + /** + * `The "<1>{{ name }}" property will be removed. This action cannot be undone.` + */ + ["com.affine.settings.workspace.properties.delete-property-desc"]: ComponentType>; + /** + * `<0>{{count}} doc` + */ + ["com.affine.settings.workspace.properties.doc"]: ComponentType>; + /** + * `<0>{{count}} docs` + */ + ["com.affine.settings.workspace.properties.doc_others"]: ComponentType>; + /** + * `Manage workspace <1>{{name}} properties` + */ + ["com.affine.settings.workspace.properties.header.subtitle"]: ComponentType>; + /** + * `<0>{{count}} selected` + + * - com.affine.tag.toolbar.selected_one: `<0>{{count}} tag selected` + + * - com.affine.tag.toolbar.selected_other: `<0>{{count}} tag(s) selected` + */ + ["com.affine.tag.toolbar.selected"]: ComponentType>; + /** + * `<0>{{count}} tag selected` + */ + ["com.affine.tag.toolbar.selected_one"]: ComponentType>; + /** + * `<0>{{count}} tag(s) selected` + */ + ["com.affine.tag.toolbar.selected_other"]: ComponentType>; + /** + * `<0>{{count}} tag(s) selected` + */ + ["com.affine.tag.toolbar.selected_others"]: ComponentType>; + /** + * `Deleting <1>{{workspace}} cannot be undone, please proceed with caution. All contents will be lost.` + */ + ["com.affine.workspaceDelete.description"]: ComponentType>; + /** + * `Deleting <1>{{workspace}} will delete both local and cloud data, this operation cannot be undone, please proceed with caution.` + */ + ["com.affine.workspaceDelete.description2"]: ComponentType>; + /** + * ` We recommend the <1>Chrome browser for optimal experience.` + */ + recommendBrowser: ComponentType, { + ["1"]: JSX.Element; + }>>; + /** + * `Are you sure you want to upgrade <1>{{workspaceName}} to a Team Workspace? This will allow unlimited members to collaborate in this workspace.` + */ + ["com.affine.upgrade-to-team-page.upgrade-confirm.description"]: ComponentType>; +} = /*#__PURE__*/ createProxy(createComponent); diff --git a/packages/frontend/i18n/src/i18next.ts b/packages/frontend/i18n/src/i18next.ts index 3505567050085..6eac01e621b31 100644 --- a/packages/frontend/i18n/src/i18next.ts +++ b/packages/frontend/i18n/src/i18next.ts @@ -3,7 +3,7 @@ import type { BackendModule, i18n } from 'i18next'; import i18next from 'i18next'; import { initReactI18next } from 'react-i18next'; -import type { useAFFiNEI18N } from './i18n-generated'; +import type { useAFFiNEI18N } from './i18n.gen'; import type { Language } from './resources'; import { SUPPORTED_LANGUAGES } from './resources'; diff --git a/packages/frontend/templates/package.json b/packages/frontend/templates/package.json index 6646cb336f90a..698643e1259e2 100644 --- a/packages/frontend/templates/package.json +++ b/packages/frontend/templates/package.json @@ -4,7 +4,8 @@ "sideEffect": false, "version": "0.18.0", "scripts": { - "postinstall": "node ./build-edgeless.mjs && node ./build-stickers.mjs" + "build": "node ./build-edgeless.mjs && node ./build-stickers.mjs", + "postinstall": "yarn build" }, "type": "module", "exports": { diff --git a/scripts/bump-blocksuite.js b/scripts/bump-blocksuite.js deleted file mode 100755 index 1a665bff4b103..0000000000000 --- a/scripts/bump-blocksuite.js +++ /dev/null @@ -1,3 +0,0 @@ -#!/usr/bin/env node - -import('@affine/bump-blocksuite'); diff --git a/scripts/bump-octobase.sh b/scripts/bump-octobase.sh deleted file mode 100755 index 487e3702e75e2..0000000000000 --- a/scripts/bump-octobase.sh +++ /dev/null @@ -1,3 +0,0 @@ -#!/bin/bash - -cargo update -p jwst-codec -p jwst-core -p jwst-storage diff --git a/scripts/check-version.mjs b/scripts/check-version.mjs deleted file mode 100644 index 364ebc32c3dc4..0000000000000 --- a/scripts/check-version.mjs +++ /dev/null @@ -1,15 +0,0 @@ -const semver = await import('semver').catch( - () => import('../packages/backend/server/node_modules/semver/index.js') -); - -import packageJson from '../package.json' with { type: 'json' }; - -const { engines } = packageJson; - -const version = engines.node; -if (!semver.satisfies(process.version, version)) { - console.log( - `Required node version ${version} not satisfied with current version ${process.version}.` - ); - process.exit(1); -} diff --git a/scripts/detect-blocksuite-update.mjs b/scripts/detect-blocksuite-update.mjs deleted file mode 100644 index b6af17118abf2..0000000000000 --- a/scripts/detect-blocksuite-update.mjs +++ /dev/null @@ -1,105 +0,0 @@ -const { exec } = await import('node:child_process'); -const { fileURLToPath } = await import('node:url'); -const { readdir } = await import('node:fs/promises'); -const { join } = await import('node:path'); - -async function readJsonFromCommit(commit, file) { - function readFile(commit, file) { - return new Promise((resolve, reject) => { - exec(`git show ${commit}:${file}`, (error, stdout, stderr) => { - if (error) { - reject(stderr); - } else { - resolve(stdout); - } - }); - }); - } - - try { - const content = await readFile(commit, file); - return JSON.parse(content); - } catch {} -} - -function checkBlocksuiteChanged(oldPkg, newPkg) { - const changed = new Set(); - const keys = ['dependencies', 'devDependencies']; - - keys.forEach(key => { - const oldDeps = oldPkg[key] || {}; - const newDeps = newPkg[key] || {}; - - Object.keys(newDeps).forEach(dep => { - if (newDeps[dep] !== oldDeps[dep] && dep.startsWith('@blocksuite/')) { - changed.add(dep); - } - }); - }); - - return changed; -} - -async function findPackageJson(root) { - const packages = new Set(); - - async function walk(dir) { - const files = await readdir(dir, { withFileTypes: true }); - - for (const file of files) { - if (file.isDirectory() && file.name !== 'node_modules') { - await walk(join(dir, file.name)); - } else if (file.name === 'package.json') { - let path = join(dir.replace(root, ''), file.name); - if (path.startsWith('/')) { - path = path.slice(1); - } - packages.add(path); - } - } - } - - await walk(root); - - return packages; -} - -async function main() { - const commitHash = process.argv[2] || process.env.GITHUB_BASE_REF; - const currentHead = process.argv[3] || 'HEAD'; - if (!commitHash) { - console.error('Missing base ref commit hash, skipping check.'); - process.exit(1); - } - - const changedPackages = new Set(); - const folders = await findPackageJson( - join(fileURLToPath(import.meta.url), '..', '..') - ); - - for (const packagePath of folders) { - const old = await readJsonFromCommit(commitHash, packagePath); - const current = await readJsonFromCommit(currentHead, packagePath); - console.log('checking:', packagePath); - if ( - old && - current && - typeof old === 'object' && - typeof current === 'object' - ) { - for (const p of checkBlocksuiteChanged(old, current)) { - changedPackages.add(p); - } - } - } - - if (changedPackages.size > 0) { - console.log('Blocksuite packages have been updated.', changedPackages); - process.exit(0); - } else { - console.log('No changes to Blocksuite packages.'); - process.exit(1); - } -} - -main().catch(console.error); diff --git a/scripts/setup/global.ts b/scripts/setup/global.ts index cedc216dd425e..088114dcc3063 100644 --- a/scripts/setup/global.ts +++ b/scripts/setup/global.ts @@ -1,11 +1,10 @@ -import { getBuildConfig } from '@affine/cli/src/webpack/runtime-config'; import { setupGlobal } from '@affine/env/global'; +import { getBuildConfig } from '@affine-tools/utils/build-config'; +import { Package } from '@affine-tools/utils/workspace'; -globalThis.BUILD_CONFIG = getBuildConfig({ - distribution: 'web', +globalThis.BUILD_CONFIG = getBuildConfig(new Package('@affine/web'), { mode: 'development', channel: 'canary', - static: false, }); if (typeof window !== 'undefined') { diff --git a/tests/affine-cloud-copilot/playwright.config.ts b/tests/affine-cloud-copilot/playwright.config.ts index 4b6736354bba5..faff4e14eb90f 100644 --- a/tests/affine-cloud-copilot/playwright.config.ts +++ b/tests/affine-cloud-copilot/playwright.config.ts @@ -26,9 +26,8 @@ const config: PlaywrightTestConfig = { retries: 3, reporter: process.env.CI ? 'github' : 'list', webServer: [ - // Intentionally not building the web, reminds you to run it by yourself. { - command: 'yarn -T run start:web-static', + command: 'yarn run -T affine dev -p @affine/web', port: 8080, timeout: 120 * 1000, reuseExistingServer: !process.env.CI, @@ -37,7 +36,7 @@ const config: PlaywrightTestConfig = { }, }, { - command: 'yarn workspace @affine/server start', + command: 'yarn run -T affine dev -p @affine/server', port: 3010, timeout: 120 * 1000, reuseExistingServer: !process.env.CI, diff --git a/tests/affine-cloud/playwright.config.ts b/tests/affine-cloud/playwright.config.ts index 68e8d852e5e3f..59d59729b62c6 100644 --- a/tests/affine-cloud/playwright.config.ts +++ b/tests/affine-cloud/playwright.config.ts @@ -26,9 +26,12 @@ const config: PlaywrightTestConfig = { retries: process.env.COPILOT ? 1 : 3, reporter: process.env.CI ? 'github' : 'list', webServer: [ - // Intentionally not building the web, reminds you to run it by yourself. { - command: 'yarn -T run start:web-static', + // TODO(@forehalo): + // in ci, all the target will be built, + // we could download the builds from archives + // and then run the web with simple http serve, it's will be faster + command: 'yarn run -T affine dev -p @affine/web', port: 8080, timeout: 120 * 1000, reuseExistingServer: !process.env.CI, @@ -37,12 +40,10 @@ const config: PlaywrightTestConfig = { }, }, { - command: 'yarn workspace @affine/server start', + command: 'yarn run -T affine dev -p @affine/server', port: 3010, timeout: 120 * 1000, reuseExistingServer: !process.env.CI, - stdout: 'pipe', - stderr: 'pipe', env: { DATABASE_URL: process.env.DATABASE_URL ?? diff --git a/tests/affine-desktop-cloud/playwright.config.ts b/tests/affine-desktop-cloud/playwright.config.ts index 3432035187049..0ec2e72e6f029 100644 --- a/tests/affine-desktop-cloud/playwright.config.ts +++ b/tests/affine-desktop-cloud/playwright.config.ts @@ -23,7 +23,7 @@ const config: PlaywrightTestConfig = { webServer: [ // Intentionally not building the web, reminds you to run it by yourself. { - command: 'yarn -T run start:web-static', + command: 'yarn run -T affine bundle -p @affine/electron --dev', port: 8080, timeout: 120 * 1000, reuseExistingServer: !process.env.CI, @@ -34,7 +34,7 @@ const config: PlaywrightTestConfig = { }, }, { - command: 'yarn workspace @affine/server start', + command: 'yarn run -T affine dev -p @affine/server', port: 3010, timeout: 120 * 1000, reuseExistingServer: !process.env.CI, diff --git a/tests/affine-desktop/playwright.config.ts b/tests/affine-desktop/playwright.config.ts index f6cd39740d55d..fa6315de87aa4 100644 --- a/tests/affine-desktop/playwright.config.ts +++ b/tests/affine-desktop/playwright.config.ts @@ -41,7 +41,7 @@ if (process.env.DEV_SERVER_URL) { ); config.webServer = [ { - command: 'yarn run start:web-static', + command: 'yarn run -T affine bundle -p @affine/electron --dev', port: 8080, timeout: 120 * 1000, reuseExistingServer: !process.env.CI, diff --git a/tests/affine-local/playwright.config.ts b/tests/affine-local/playwright.config.ts index 8337cb53b111a..7f68d6a403089 100644 --- a/tests/affine-local/playwright.config.ts +++ b/tests/affine-local/playwright.config.ts @@ -46,7 +46,7 @@ const config: PlaywrightTestConfig = { webServer: [ // Intentionally not building the web, reminds you to run it by yourself. { - command: 'yarn run start:web-static', + command: 'yarn run -T affine dev -p @affine/web', port: 8080, timeout: 120 * 1000, reuseExistingServer: !process.env.CI, diff --git a/tests/affine-mobile/playwright.config.ts b/tests/affine-mobile/playwright.config.ts index 6804256482284..76b381bc588f9 100644 --- a/tests/affine-mobile/playwright.config.ts +++ b/tests/affine-mobile/playwright.config.ts @@ -49,7 +49,7 @@ const config: PlaywrightTestConfig = { webServer: [ // Intentionally not building the web, reminds you to run it by yourself. { - command: 'yarn workspace @affine/mobile static-server', + command: 'yarn run -T affine dev -p @affine/mobile', port: 8080, timeout: 120 * 1000, reuseExistingServer: !process.env.CI, diff --git a/tests/kit/utils/cloud.ts b/tests/kit/utils/cloud.ts index e110b5280d128..923ba1cf085a8 100644 --- a/tests/kit/utils/cloud.ts +++ b/tests/kit/utils/cloud.ts @@ -53,16 +53,22 @@ const cloudUserSchema = z.object({ export const runPrisma = async ( cb: ( prisma: InstanceType< - // eslint-disable-next-line @typescript-eslint/consistent-type-imports + // oxlint-disable-next-line @typescript-eslint/consistent-type-imports typeof import('../../../packages/backend/server/node_modules/@prisma/client').PrismaClient > ) => Promise ): Promise => { const { PrismaClient, - // eslint-disable-next-line @typescript-eslint/no-var-requires - } = require('../../../packages/backend/server/node_modules/@prisma/client'); - const client = new PrismaClient(); + // oxlint-disable-next-line @typescript-eslint/consistent-type-imports + } = await import( + '../../../packages/backend/server/node_modules/@prisma/client' + ); + const client = new PrismaClient({ + datasourceUrl: + process.env.DATABASE_URL || + 'postgresql://affine:affine@localhost:5432/affine', + }); await client.$connect(); try { return await cb(client); diff --git a/tools/bump-blocksuite/README.md b/tools/bump-blocksuite/README.md deleted file mode 100644 index be81a1fab2a50..0000000000000 --- a/tools/bump-blocksuite/README.md +++ /dev/null @@ -1,13 +0,0 @@ -# Bump BlockSuite - -To update BlockSuite, run following command in project root: - -```sh -node scripts/bump-blocksuite.js -``` - -For network issue, try setting proxy environment variables: - -```sh -export http_proxy=http://127.0.0.1:7890 -``` diff --git a/tools/bump-blocksuite/index.js b/tools/bump-blocksuite/index.js deleted file mode 100644 index c30bfd1009347..0000000000000 --- a/tools/bump-blocksuite/index.js +++ /dev/null @@ -1,176 +0,0 @@ -import { execSync } from 'node:child_process'; -import { join } from 'node:path'; -import { fileURLToPath } from 'node:url'; - -import { Clipboard } from '@napi-rs/clipboard'; -import { - FetchOptions, - ProxyOptions, - RemoteCallbacks, - Repository, - Sort, -} from '@napi-rs/simple-git'; -import chalk from 'chalk'; -import { ProxyAgent, setGlobalDispatcher } from 'undici'; - -import corePackage from '../../packages/frontend/core/package.json' with { type: 'json' }; - -const clipboard = new Clipboard(); - -const oldHash = corePackage.dependencies['@blocksuite/affine'].split('-').pop(); - -const info = await fetch('https://registry.npmjs.org/@blocksuite/affine').then( - res => res.json() -); - -const latestVersion = info['dist-tags'].latest; -const latestHash = latestVersion.split('-').pop(); -const latestGitHead = info.versions[latestHash].gitHead; -const oldGitHead = info.versions[oldHash].gitHead; - -if (oldHash === latestHash) { - console.info(chalk.greenBright('Already updated')); - process.exit(0); -} - -if (process.env.http_proxy) { - setGlobalDispatcher(new ProxyAgent(process.env.http_proxy)); -} - -console.info(`Upgrade blocksuite from ${oldHash} -> ${latestHash}`); - -const blockSuiteDeps = execSync(`yarn info -A --name-only --json`, { - encoding: 'utf8', -}); - -const blocksuiteDepsList = blockSuiteDeps - .split('\n') - .map(s => s.trim()) - .filter(Boolean) - .map(s => s.substring(1, s.length - 1)) - .filter( - s => s.startsWith('@blocksuite') && !s.startsWith('@blocksuite/icons') - ) - .map(s => s.split('@npm').at(0)); - -for (const pkg of blocksuiteDepsList) { - const command = `yarn up ${pkg}@${latestVersion}`; - console.info(chalk.bgCyan(`Executing ${command}`)); - execSync(command, { - stdio: 'inherit', - }); -} - -console.info(`Upgrade complete`); - -const repo = new Repository( - join(fileURLToPath(import.meta.url), '..', '..', '..', '..', 'BlockSuite') -); - -const remote = repo.remoteAnonymous( - 'https://github.com/toeverything/BlockSuite.git' -); - -remote.fetch( - ['master'], - new FetchOptions().proxyOptions(new ProxyOptions().auto()).remoteCallback( - new RemoteCallbacks().transferProgress(progress => { - if (progress.totalDeltas && progress.totalObjects) { - console.log( - `${( - (progress.receivedObjects / progress.totalObjects) * 50 + - (progress.indexedDeltas / progress.totalDeltas) * 50 - ).toFixed(2)}%` - ); - } - }) - ) -); - -const commits = { - Features: [], - Bugfix: [], - Refactor: [], - Misc: [], -}; - -if (!latestGitHead) { - console.info('latestGitHead is not found'); - console.info('Skip generating changelog'); - process.exit(0); -} - -for (const oid of repo - .revWalk() - .push(latestGitHead) - .setSorting(Sort.Time & Sort.Topological)) { - if (oid.startsWith(oldGitHead)) { - break; - } - const commit = repo.findCommit(oid); - const summary = commit.summary(); - if (summary.startsWith('feat')) { - commits.Features.push(commit); - } else if (summary.startsWith('fix')) { - commits.Bugfix.push(commit); - } else if (summary.startsWith('refactor')) { - commits.Refactor.push(commit); - } else { - commits.Misc.push(commit); - } -} - -clipboard.setText(await formatCommits(commits)); - -console.info(`Changelog copied to clipboard`); - -async function formatCommits(commits) { - return `## Features -${await Promise.all(commits.Features.map(format)).then(commits => - commits.join('\n') -)} - -## Bugfix -${await Promise.all(commits.Bugfix.map(format)).then(commits => - commits.join('\n') -)} - -## Refactor -${await Promise.all(commits.Refactor.map(format)).then(commits => - commits.join('\n') -)} - -## Misc -${await Promise.all(commits.Misc.map(format)).then(commits => - commits.join('\n') -)} -`; - /** - * @param {import('./index').Commit} commit - * @returns string - */ - async function format(commit) { - const summary = commit.summary(); - const match = summary.match(/\(#(\d+)\)/); - if (match) { - const [_, pull] = match; - const pullInfo = await fetch( - `https://api.github.com/repos/toeverything/BlockSuite/pulls/${pull}`, - { - headers: { - Accept: 'application/vnd.github+json', - Authorization: `Bearer ${process.env.GITHUB_TOKEN}`, - 'X-GitHub-Api-Version': '2022-11-28', - }, - } - ) - .then(res => res.json()) - .catch(() => ({ user: {} })); - const { - user: { login }, - } = pullInfo; - return `- https://github.com/toeverything/BlockSuite/pull/${pull} @${login}`; - } - return `- ${summary}`; - } -} diff --git a/tools/bump-blocksuite/package.json b/tools/bump-blocksuite/package.json deleted file mode 100644 index 541d2b2ee1d86..0000000000000 --- a/tools/bump-blocksuite/package.json +++ /dev/null @@ -1,14 +0,0 @@ -{ - "name": "@affine/bump-blocksuite", - "version": "0.18.0", - "type": "module", - "main": "index.js", - "private": true, - "description": "Generate changelog from blocksuite version change", - "dependencies": { - "@napi-rs/clipboard": "^1.1.2", - "@napi-rs/simple-git": "^0.1.19", - "chalk": "^5.3.0", - "undici": "^7.1.0" - } -} diff --git a/tools/cli/README.md b/tools/cli/README.md new file mode 100644 index 0000000000000..a2da1bd8e0841 --- /dev/null +++ b/tools/cli/README.md @@ -0,0 +1,113 @@ +# AFFiNE Monorepo Cli + +## Start + +```bash +yarn affine -h +``` + +### Run build command defined in package.json + +```bash +yarn affine i18n build +# or +yarn build -p i18n +``` + +### Run dev command defined in package.json + +```bash +yarn affine web dev +# or +yarn dev -p i18n +``` + +### Clean + +```bash +yarn affine clean --dist --rust +# clean node_modules +yarn affine clean --node-modules +``` + +### Init + +> Generate files that make the monorepo work properly, the per project codegen will not be included anymore + +```bash +yarn affine init +``` + +## Tricks + +### Define scripts to run a .ts files without `--loader ts-node/esm/transpile-only` + +`affine run` will automatically inject `ts-node`'s transpile service(swc used) for your scripts + +```json +{ + "name": "@affine/demo" + "scripts": { + "dev": "node ./dev.ts" + } +} +``` + +```bash +affine @affine/demo dev +``` + +### Short your key presses + +```bash +# af is also available for running the scripts +yarn af web build +``` + +#### by custom shell script + +> personally, I use 'af' + +create file `af` in the root of AFFiNE project with the following content + +```bash +#!/usr/bin/env sh +./tools/scripts/bin/runner.js affine.ts $@ +``` + +or on windows: + +```cmd +node "./tools/cli/bin/runner.js" affine.ts %* +``` + +and give it executable permission + +```bash +chmod a+x ./af + +# now you can run scripts with simply +./af web build +``` + +if you want to go further, but for vscode(or other forks) only, add the following to your `.vscode/settings.json` + +```json +{ + "terminal.integrated.env.osx": { + "PATH": "${env:PATH}:${cwd}" + }, + "terminal.integrated.env.linux": { + "PATH": "${env:PATH}:${cwd}" + }, + "terminal.integrated.env.windows": { + "PATH": "${env:PATH};${cwd}" + } +} +``` + +restart all the integrated terminals and now you get: + +```bash +af web build +``` diff --git a/tools/cli/bin/runner.js b/tools/cli/bin/runner.js new file mode 100755 index 0000000000000..2c7cd20e1cce8 --- /dev/null +++ b/tools/cli/bin/runner.js @@ -0,0 +1,68 @@ +#!/usr/bin/env node +import { spawn } from 'node:child_process'; +import { existsSync } from 'node:fs'; +import { join } from 'node:path'; +import { fileURLToPath, pathToFileURL } from 'node:url'; + +const scriptsFolder = join(fileURLToPath(import.meta.url), '..', '..'); +const scriptsSrcFolder = join(scriptsFolder, 'src'); +const projectRoot = join(scriptsFolder, '..', '..'); +const loader = join(scriptsFolder, 'register.js'); + +const [node, _self, file, ...options] = process.argv; + +if (!file) { + console.error(`Please provide a file to run, e.g. 'run src/index.{js/ts}'`); + process.exit(1); +} + +const fileLocationCandidates = new Set([ + process.cwd(), + scriptsSrcFolder, + projectRoot, +]); +const lookups = []; + +/** + * @type {string | undefined} + */ +let scriptLocation; +for (const location of fileLocationCandidates) { + const fileCandidates = [file, `${file}.js`, `${file}.ts`]; + for (const candidate of fileCandidates) { + const candidateLocation = join(location, candidate); + if (existsSync(candidateLocation)) { + scriptLocation = candidateLocation; + break; + } + lookups.push(candidateLocation); + } +} + +if (!scriptLocation) { + console.error( + `File ${file} not found, please make sure the first parameter passed to 'run' script is a valid js or ts file.` + ); + console.error(`Searched locations: `); + lookups.forEach(location => { + console.error(` - ${location}`); + }); + process.exit(1); +} + +const nodeOptions = []; + +if ( + scriptLocation.endsWith('.ts') || + scriptLocation.startsWith(scriptsFolder) +) { + nodeOptions.unshift(`--import=${pathToFileURL(loader)}`); +} else { + nodeOptions.unshift('--experimental-specifier-resolution=node'); +} + +spawn(node, [...nodeOptions, scriptLocation, ...options], { + stdio: 'inherit', +}).on('exit', code => { + process.exit(code); +}); diff --git a/tools/cli/hooks.js b/tools/cli/hooks.js new file mode 100644 index 0000000000000..9cf75089de481 --- /dev/null +++ b/tools/cli/hooks.js @@ -0,0 +1,14 @@ +import { create, createEsmHooks, register } from 'ts-node'; + +const service = create({ + experimentalSpecifierResolution: 'node', + esm: true, + transpileOnly: true, + swc: true, +}); + +register(service); +const hooks = createEsmHooks(service); + +export const resolve = hooks.resolve; +export const load = hooks.load; diff --git a/tools/cli/package.json b/tools/cli/package.json index bd6302f8b42ac..2b8d6be02380f 100644 --- a/tools/cli/package.json +++ b/tools/cli/package.json @@ -1,33 +1,39 @@ { - "name": "@affine/cli", + "name": "@affine-tools/cli", + "version": "0.0.1", "type": "module", "private": true, - "devDependencies": { - "@affine/env": "workspace:*", - "@affine/templates": "workspace:*", + "bin": { + "r": "./bin/runner.js" + }, + "exports": { + "./loader": "./loader.js" + }, + "scripts": { + "affine": "r ./src/affine.ts" + }, + "dependencies": { + "@affine-tools/utils": "workspace:*", "@aws-sdk/client-s3": "^3.709.0", - "@blocksuite/affine": "workspace:*", "@clack/core": "^0.4.0", "@clack/prompts": "^0.9.0", - "@magic-works/i18n-codegen": "^0.6.1", "@napi-rs/simple-git": "^0.1.19", "@perfsee/webpack": "^1.13.0", "@pmmmwh/react-refresh-webpack-plugin": "^0.5.15", "@sentry/webpack-plugin": "^2.22.7", - "@types/mime-types": "^2.1.4", - "@types/webpack-env": "^1.18.5", "@vanilla-extract/webpack-plugin": "^2.3.15", "autoprefixer": "^10.4.20", + "clipanion": "^3.2.1", "copy-webpack-plugin": "^12.0.2", "css-loader": "^7.1.2", "cssnano": "^7.0.6", - "dotenv": "^16.4.7", "html-webpack-plugin": "^5.6.3", "lodash-es": "^4.17.21", "mime-types": "^2.1.35", "mini-css-extract-plugin": "^2.9.2", "postcss": "^8.4.49", "postcss-loader": "^8.1.1", + "prettier": "^3.4.2", "react-refresh": "^0.16.0", "source-map-loader": "^5.0.0", "style-loader": "^4.0.0", @@ -35,14 +41,16 @@ "tailwindcss": "^3.4.16", "terser-webpack-plugin": "^5.3.10", "ts-node": "^10.9.2", - "vite": "^6.0.3", + "typanion": "^3.14.0", + "typescript": "^5.5.4", "webpack": "^5.97.1", "webpack-dev-server": "^5.2.0", "webpack-merge": "^6.0.1" }, - "scripts": { - "bundle": "node --loader ts-node/esm/transpile-only.mjs ./src/bin/build.ts", - "dev": "node --loader ts-node/esm/transpile-only.mjs ./src/bin/dev.ts" - }, - "version": "0.18.0" + "devDependencies": { + "@types/lodash-es": "^4.17.12", + "@types/mime-types": "^2.1.4", + "@types/node": "^20.17.10", + "@types/webpack-env": "^1.18.5" + } } diff --git a/tools/cli/register.js b/tools/cli/register.js new file mode 100644 index 0000000000000..212f6a2b8065e --- /dev/null +++ b/tools/cli/register.js @@ -0,0 +1,3 @@ +import { register } from 'node:module'; + +register('./hooks.js', import.meta.url); diff --git a/tools/cli/src/affine.ts b/tools/cli/src/affine.ts new file mode 100644 index 0000000000000..d69acfaabe6eb --- /dev/null +++ b/tools/cli/src/affine.ts @@ -0,0 +1,32 @@ +import { Workspace } from '@affine-tools/utils/workspace'; +import { Cli } from 'clipanion'; + +import { BuildCommand } from './build'; +import { BundleCommand } from './bundle'; +import { CleanCommand } from './clean'; +import { CodegenCommand } from './codegen'; +import type { CliContext } from './context'; +import { DevCommand } from './dev'; +import { RunCommand } from './run'; + +const cli = new Cli({ + binaryName: 'affine', + binaryVersion: '0.0.0', + binaryLabel: 'AFFiNE Monorepo Tools', + enableColors: true, + enableCapture: true, +}); + +cli.register(RunCommand); +cli.register(CodegenCommand); +cli.register(CleanCommand); +cli.register(BuildCommand); +cli.register(DevCommand); +cli.register(BundleCommand); + +await cli.runExit(process.argv.slice(2), { + workspace: new Workspace(), + stdin: process.stdin, + stdout: process.stdout, + stderr: process.stderr, +}); diff --git a/tools/cli/src/bin/build.ts b/tools/cli/src/bin/build.ts deleted file mode 100644 index 64113974a61dd..0000000000000 --- a/tools/cli/src/bin/build.ts +++ /dev/null @@ -1,66 +0,0 @@ -import { spawn } from 'node:child_process'; - -import webpack from 'webpack'; - -import { getCwdFromDistribution } from '../config/cwd.cjs'; -import type { BuildFlags } from '../config/index.js'; -import { createWebpackConfig } from '../webpack/webpack.config.js'; - -// eslint-disable-next-line @typescript-eslint/no-non-null-assertion -const buildType = process.env.BUILD_TYPE_OVERRIDE || process.env.BUILD_TYPE; - -if (process.env.BUILD_TYPE_OVERRIDE) { - process.env.BUILD_TYPE = process.env.BUILD_TYPE_OVERRIDE; -} - -const getChannel = () => { - switch (buildType) { - case 'canary': - case 'beta': - case 'stable': - case 'internal': - return buildType; - case '': - throw new Error('BUILD_TYPE is not set'); - default: { - throw new Error( - `BUILD_TYPE must be one of canary, beta, stable, internal, received [${buildType}]` - ); - } - } -}; - -let entry: BuildFlags['entry']; - -const { DISTRIBUTION = 'web' } = process.env; - -const cwd = getCwdFromDistribution(DISTRIBUTION); - -if (DISTRIBUTION === 'desktop') { - entry = { app: './index.tsx', shell: './shell/index.tsx' }; -} - -const flags = { - distribution: DISTRIBUTION as BuildFlags['distribution'], - mode: 'production', - channel: getChannel(), - coverage: process.env.COVERAGE === 'true', - entry, - static: false, -} satisfies BuildFlags; - -spawn('yarn', ['workspace', '@affine/i18n', 'build'], { - stdio: 'inherit', -}); - -// eslint-disable-next-line @typescript-eslint/no-non-null-assertion -webpack(createWebpackConfig(cwd!, flags), (err, stats) => { - if (err) { - console.error(err); - process.exit(1); - } - if (stats?.hasErrors()) { - console.error(stats.toString('errors-only')); - process.exit(1); - } -}); diff --git a/tools/cli/src/bin/dev.ts b/tools/cli/src/bin/dev.ts deleted file mode 100644 index 8962d797e263b..0000000000000 --- a/tools/cli/src/bin/dev.ts +++ /dev/null @@ -1,146 +0,0 @@ -import { spawn } from 'node:child_process'; -import { existsSync } from 'node:fs'; -import { join } from 'node:path'; - -import * as p from '@clack/prompts'; -import { config } from 'dotenv'; -import webpack from 'webpack'; -import WebpackDevServer from 'webpack-dev-server'; - -import { getCwdFromDistribution, projectRoot } from '../config/cwd.cjs'; -import type { BuildFlags } from '../config/index.js'; -import { createWebpackConfig } from '../webpack/webpack.config.js'; - -const flags: BuildFlags = { - distribution: - (process.env.DISTRIBUTION as BuildFlags['distribution']) ?? 'web', - mode: 'development', - static: false, - channel: 'canary', - coverage: process.env.COVERAGE === 'true', -}; - -const files = ['.env', '.env.local']; - -for (const file of files) { - if (existsSync(join(projectRoot, file))) { - config({ - path: join(projectRoot, file), - }); - console.log(`${file} loaded`); - break; - } -} - -const buildFlags = process.argv.includes('--static') - ? { ...flags, static: true } - : ((await p.group( - { - distribution: () => - p.select({ - message: 'Distribution', - options: [ - { - value: 'web', - }, - { - value: 'desktop', - }, - { - value: 'admin', - }, - { - value: 'mobile', - }, - { - value: 'ios', - }, - ], - initialValue: 'web', - }), - mode: () => - p.select({ - message: 'Mode', - options: [ - { - value: 'development', - }, - { - value: 'production', - }, - ], - initialValue: 'development', - }), - channel: () => - p.select({ - message: 'Channel', - options: [ - { - value: 'canary', - }, - { - value: 'beta', - }, - { - value: 'stable', - }, - ], - initialValue: 'canary', - }), - coverage: () => - p.confirm({ - message: 'Enable coverage', - initialValue: process.env.COVERAGE === 'true', - }), - }, - { - onCancel: () => { - p.cancel('Operation cancelled.'); - process.exit(0); - }, - } - )) as BuildFlags); - -flags.distribution = buildFlags.distribution; -flags.mode = buildFlags.mode; -flags.channel = buildFlags.channel; -flags.coverage = buildFlags.coverage; -flags.static = buildFlags.static; -flags.entry = undefined; - -const cwd = getCwdFromDistribution(flags.distribution); - -process.env.DISTRIBUTION = flags.distribution; - -if (flags.distribution === 'desktop') { - flags.entry = { - app: join(cwd, 'index.tsx'), - shell: join(cwd, 'shell/index.tsx'), - }; -} - -console.info(flags); - -if (!flags.static) { - spawn('yarn', ['workspace', '@affine/i18n', 'dev'], { - stdio: 'inherit', - shell: true, - }); -} - -try { - // @ts-expect-error no types - await import('@affine/templates/build-edgeless'); - const config = createWebpackConfig(cwd, flags); - if (flags.static) { - config.watch = false; - } - const compiler = webpack(config); - // Start webpack - const devServer = new WebpackDevServer(config.devServer, compiler); - - await devServer.start(); -} catch (error) { - console.error('Error during build:', error); - process.exit(1); -} diff --git a/tools/cli/src/build.ts b/tools/cli/src/build.ts new file mode 100644 index 0000000000000..94337b55d3ad3 --- /dev/null +++ b/tools/cli/src/build.ts @@ -0,0 +1,15 @@ +import { PackageCommand } from './command'; + +export class BuildCommand extends PackageCommand { + static override paths = [['build'], ['b']]; + + async execute() { + const args = ['affine build', this.package]; + + if (this.deps) { + args.push('--deps'); + } + + await this.cli.run(args); + } +} diff --git a/tools/cli/src/bundle.ts b/tools/cli/src/bundle.ts new file mode 100644 index 0000000000000..014ceb4043e85 --- /dev/null +++ b/tools/cli/src/bundle.ts @@ -0,0 +1,93 @@ +import webpack, { type Compiler, type Configuration } from 'webpack'; +import WebpackDevServer from 'webpack-dev-server'; +import { merge } from 'webpack-merge'; + +import { Option, PackageCommand } from './command'; +import { createWebpackConfig } from './webpack'; + +function getChannel() { + const channel = process.env.BUILD_TYPE ?? 'canary'; + switch (channel) { + case 'canary': + case 'beta': + case 'stable': + case 'internal': + return channel; + default: { + throw new Error( + `BUILD_TYPE must be one of canary, beta, stable, internal, received [${channel}]` + ); + } + } +} + +export class BundleCommand extends PackageCommand { + static override paths = [['bundle'], ['webpack'], ['pack'], ['bun']]; + + // bundle is not able to run with deps + override deps = false; + + dev = Option.Boolean('--dev,-d', false, { + description: 'Run in Development mode', + }); + + async execute() { + this.logger.info(`Packing package ${this.package}...`); + + const config = await this.getConfig(); + + const compiler = webpack(config); + + if (this.dev) { + await this.start(compiler, config.devServer); + } else { + await this.build(compiler); + } + } + + async getConfig() { + let config = createWebpackConfig(this.workspace.getPackage(this.package), { + mode: this.dev ? 'development' : 'production', + channel: getChannel(), + }); + + let configOverride: Configuration | undefined; + const overrideConfigPath = this.workspace + .getPackage(this.package) + .join('webpack.config.js'); + + if (overrideConfigPath.isFile()) { + const override = await import(overrideConfigPath.value); + configOverride = override.config ?? override.default; + } + + if (configOverride) { + config = merge(config, configOverride); + } + + return config; + } + + async start(compiler: Compiler, config: Configuration['devServer']) { + const devServer = new WebpackDevServer(config, compiler); + + await devServer.start(); + } + + async build(compiler: Compiler) { + compiler.run((error, stats) => { + if (error) { + console.error(error); + process.exit(1); + } + if (stats) { + if (stats.hasErrors()) { + console.error(stats.toString('errors-only')); + process.exit(1); + } else { + console.log(stats.toString('minimal')); + } + } + }); + } +} diff --git a/tools/cli/src/clean.ts b/tools/cli/src/clean.ts new file mode 100644 index 0000000000000..431ce63dd5473 --- /dev/null +++ b/tools/cli/src/clean.ts @@ -0,0 +1,71 @@ +import { rmSync } from 'node:fs'; + +import { exec } from '@affine-tools/utils/process'; + +import { Command, Option } from './command'; + +export class CleanCommand extends Command { + static override paths = [['clean']]; + + cleanDist = Option.Boolean('--dist', false); + cleanRustTarget = Option.Boolean('--rust', false); + cleanNodeModules = Option.Boolean('--node-modules', false); + all = Option.Boolean('--all,-a', false); + + async execute() { + this.logger.info('Cleaning Workspace...'); + if (this.all || this.cleanNodeModules) { + this.doCleanNodeModules(); + } + + if (this.all || this.cleanDist) { + this.doCleanDist(); + } + + if (this.all || this.cleanRustTarget) { + this.doCleanRust(); + } + } + + doCleanNodeModules() { + this.logger.info('Cleaning node_modules...'); + + const rootNodeModules = this.workspace.join('node_modules'); + if (rootNodeModules.isDirectory()) { + this.logger.info(`Cleaning ${rootNodeModules}`); + rmSync(rootNodeModules.value, { recursive: true }); + } + + this.workspace.forEach(pkg => { + const nodeModules = pkg.nodeModulesPath; + if (nodeModules.isDirectory()) { + this.logger.info(`Cleaning ${nodeModules}`); + rmSync(nodeModules.value, { recursive: true }); + } + }); + + this.logger.info('node_modules cleaned'); + } + + doCleanDist() { + this.logger.info('Cleaning dist...'); + + this.workspace.forEach(pkg => { + if (pkg.distPath.isDirectory()) { + this.logger.info(`Cleaning ${pkg.distPath}`); + rmSync(pkg.distPath.value, { recursive: true }); + } + + if (pkg.libPath.isDirectory()) { + this.logger.info(`Cleaning ${pkg.libPath}`); + rmSync(pkg.libPath.value, { recursive: true }); + } + }); + + this.logger.info('dist cleaned'); + } + + doCleanRust() { + exec('', 'cargo clean'); + } +} diff --git a/tools/cli/src/codegen.ts b/tools/cli/src/codegen.ts new file mode 100755 index 0000000000000..04f5dde410508 --- /dev/null +++ b/tools/cli/src/codegen.ts @@ -0,0 +1,67 @@ +import { readFileSync, writeFileSync } from 'node:fs'; + +import type { Path } from '@affine-tools/utils/path'; +import { type BuiltInParserName, format } from 'prettier'; + +import { Command } from './command'; + +export class CodegenCommand extends Command { + static override paths = [['init'], ['i'], ['codegen']]; + + async execute() { + this.logger.info('Generating Workspace configs'); + await this.generateWorkspaceFiles(); + this.logger.info('Workspace configs generated'); + } + + async generateWorkspaceFiles() { + const filesToGenerate: [Path, () => string, BuiltInParserName?][] = [ + [ + this.workspace.join('tsconfig.project.json'), + this.workspace.genProjectTsConfig.bind(this.workspace), + 'json', + ], + [ + this.workspace + .getPackage('@affine-tools/utils') + .join('src/workspace.gen.ts'), + this.workspace.genWorkspaceInfo.bind(this.workspace), + 'typescript', + ], + [this.workspace.join('oxlint.json'), this.genOxlintConfig, 'json'], + ]; + + for (const [path, content, formatter] of filesToGenerate) { + this.logger.info(`Output: ${path}`); + let file = content(); + if (formatter) { + file = await this.format(file, formatter); + } + writeFileSync(path.value, file); + } + } + + format(content: string, parser: BuiltInParserName) { + const config = JSON.parse( + readFileSync(this.workspace.join('.prettierrc').value, 'utf-8') + ); + return format(content, { parser, ...config }); + } + + genOxlintConfig = () => { + const json = JSON.parse( + readFileSync(this.workspace.join('oxlint.json').value, 'utf-8') + ); + + const ignoreList = readFileSync( + this.workspace.join('.prettierignore').value, + 'utf-8' + ) + .split('\n') + .filter(line => line.trim() && !line.startsWith('#')); + + json['ignorePatterns'] = ignoreList; + + return JSON.stringify(json, null, 2); + }; +} diff --git a/tools/cli/src/command.ts b/tools/cli/src/command.ts new file mode 100644 index 0000000000000..ab5b54da14c06 --- /dev/null +++ b/tools/cli/src/command.ts @@ -0,0 +1,71 @@ +import { AliasToPackage } from '@affine-tools/utils/distribution'; +import { Logger } from '@affine-tools/utils/logger'; +import { type PackageName, Workspace } from '@affine-tools/utils/workspace'; +import { Command as BaseCommand, Option } from 'clipanion'; +import * as t from 'typanion'; + +import type { CliContext } from './context'; + +export abstract class Command extends BaseCommand { + get logger() { + // @ts-expect-error hack: Get the command name + return new Logger(this.constructor.paths[0][0]); + } + + get workspace() { + return this.context.workspace; + } +} + +export abstract class PackageCommand extends Command { + protected availablePackageNameArgs = ( + Workspace.PackageNames as string[] + ).concat(Array.from(AliasToPackage.keys())); + protected packageNameValidator = t.isOneOf( + this.availablePackageNameArgs.map(k => t.isLiteral(k)) + ); + + protected packageNameOrAlias = Option.String('--package,-p', { + required: true, + validator: this.packageNameValidator, + description: 'The package name or alias to be run with', + }); + + get package(): PackageName { + return ( + AliasToPackage.get(this.packageNameOrAlias as any) ?? + (this.packageNameOrAlias as PackageName) + ); + } + + deps = Option.Boolean('--deps', false, { + description: + 'Execute the same command in workspace dependencies, if defined.', + }); +} + +export abstract class PackagesCommand extends Command { + protected availablePackageNameArgs = ( + Workspace.PackageNames as string[] + ).concat(Array.from(AliasToPackage.keys())); + protected packageNameValidator = t.isOneOf( + this.availablePackageNameArgs.map(k => t.isLiteral(k)) + ); + + protected packageNamesOrAliases = Option.Array('--package,-p', { + required: true, + validator: t.isArray(this.packageNameValidator), + }); + get packages() { + return this.packageNamesOrAliases.map( + name => AliasToPackage.get(name as any) ?? name + ); + } + + deps = Option.Boolean('--deps', false, { + description: + 'Execute the same command in workspace dependencies, if defined.', + }); +} + +export { Option }; diff --git a/tools/cli/src/config/cwd.cjs b/tools/cli/src/config/cwd.cjs deleted file mode 100644 index b01b774455bcb..0000000000000 --- a/tools/cli/src/config/cwd.cjs +++ /dev/null @@ -1,38 +0,0 @@ -// @ts-check - -const { join } = require('node:path'); - -const projectRoot = join(__dirname, '../../../..'); - -module.exports.projectRoot = projectRoot; - -/** - * - * @param {string | undefined} distribution - * @returns string - */ -module.exports.getCwdFromDistribution = function getCwdFromDistribution( - distribution -) { - switch (distribution) { - case 'web': - case undefined: - case null: - return join(projectRoot, 'packages/frontend/apps/web'); - case 'desktop': - return join(projectRoot, 'packages/frontend/apps/electron/renderer'); - case 'admin': - return join(projectRoot, 'packages/frontend/admin'); - case 'mobile': - return join(projectRoot, 'packages/frontend/apps/mobile'); - case 'ios': - return join(projectRoot, 'packages/frontend/apps/ios'); - case 'android': - return join(projectRoot, 'packages/frontend/apps/android'); - default: { - throw new Error( - 'DISTRIBUTION must be one of web, desktop, admin, mobile' - ); - } - } -}; diff --git a/tools/cli/src/config/index.ts b/tools/cli/src/config/index.ts deleted file mode 100644 index f4452c5da070a..0000000000000 --- a/tools/cli/src/config/index.ts +++ /dev/null @@ -1,9 +0,0 @@ -export type BuildFlags = { - distribution: 'web' | 'desktop' | 'admin' | 'mobile' | 'ios' | 'android'; - mode: 'development' | 'production'; - channel: 'stable' | 'beta' | 'canary' | 'internal'; - static: boolean; - coverage?: boolean; - localBlockSuite?: string; - entry?: string | { [key: string]: string }; -}; diff --git a/tools/cli/src/context.ts b/tools/cli/src/context.ts new file mode 100644 index 0000000000000..bc97d15a67328 --- /dev/null +++ b/tools/cli/src/context.ts @@ -0,0 +1,6 @@ +import type { Workspace } from '@affine-tools/utils/workspace'; +import type { BaseContext } from 'clipanion'; + +export interface CliContext extends BaseContext { + workspace: Workspace; +} diff --git a/tools/cli/src/dev.ts b/tools/cli/src/dev.ts new file mode 100644 index 0000000000000..efa3c3028c937 --- /dev/null +++ b/tools/cli/src/dev.ts @@ -0,0 +1,15 @@ +import { PackageCommand } from './command'; + +export class DevCommand extends PackageCommand { + static override paths = [['dev'], ['d']]; + + async execute() { + const args = [this.package, 'dev']; + + if (this.deps) { + args.push('--deps'); + } + + await this.cli.run(args); + } +} diff --git a/tools/cli/src/run.ts b/tools/cli/src/run.ts new file mode 100644 index 0000000000000..5149d9e50c3e8 --- /dev/null +++ b/tools/cli/src/run.ts @@ -0,0 +1,135 @@ +import { Path } from '@affine-tools/utils/path'; +import { execAsync } from '@affine-tools/utils/process'; +import type { PackageName } from '@affine-tools/utils/workspace'; + +import { Option, PackageCommand } from './command'; + +interface RunScriptOptions { + includeDependencies?: boolean; + waitDependencies?: boolean; +} + +const currentDir = Path.dir(import.meta.url); + +const ignoreLoaderScripts = [ + 'vitest', + 'vite', + 'ts-node', + 'prisma', + 'cap', + 'tsc', + /electron(?!-)/, +]; + +export class RunCommand extends PackageCommand { + static override paths = [[], ['run'], ['r']]; + + static override usage = PackageCommand.Usage({ + description: 'AFFiNE Monorepo scripts', + details: ` + \`affine web