diff --git a/.github/workflows/android_ci.yaml.bak b/.github/workflows/android_ci.yaml.bak
index 0cb110bae4..81e132cbf8 100644
--- a/.github/workflows/android_ci.yaml.bak
+++ b/.github/workflows/android_ci.yaml.bak
@@ -1,126 +1,196 @@
-# name: Android CI
+name: Android CI
-# on:
-# push:
-# branches:
-# - "main"
-# paths:
-# - ".github/workflows/mobile_ci.yaml"
-# - "frontend/**"
-# - "!frontend/appflowy_tauri/**"
+on:
+ push:
+ branches:
+ - "main"
+ paths:
+ - ".github/workflows/mobile_ci.yaml"
+ - "frontend/**"
-# pull_request:
-# branches:
-# - "main"
-# paths:
-# - ".github/workflows/mobile_ci.yaml"
-# - "frontend/**"
-# - "!frontend/appflowy_tauri/**"
+ pull_request:
+ branches:
+ - "main"
+ paths:
+ - ".github/workflows/mobile_ci.yaml"
+ - "frontend/**"
+ - "!frontend/appflowy_tauri/**"
-# env:
-# CARGO_TERM_COLOR: always
-# FLUTTER_VERSION: "3.22.0"
-# RUST_TOOLCHAIN: "1.77.2"
-# CARGO_MAKE_VERSION: "0.36.6"
+env:
+ CARGO_TERM_COLOR: always
+ FLUTTER_VERSION: "3.27.4"
+ RUST_TOOLCHAIN: "1.81.0"
+ CARGO_MAKE_VERSION: "0.37.18"
+ CLOUD_VERSION: 0.6.54-amd64
-# concurrency:
-# group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
-# cancel-in-progress: true
+concurrency:
+ group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
+ cancel-in-progress: true
-# jobs:
-# build:
-# if: github.event.pull_request.draft != true
-# strategy:
-# fail-fast: true
-# matrix:
-# os: [macos-14]
-# runs-on: ${{ matrix.os }}
+jobs:
+ build:
+ if: github.event.pull_request.draft != true
+ strategy:
+ fail-fast: true
+ matrix:
+ os: [ubuntu-latest]
+ runs-on: ${{ matrix.os }}
-# steps:
-# - name: Check storage space
-# run: df -h
+ steps:
+ - name: Check storage space
+ run:
+ df -h
-# # the following step is required to avoid running out of space
-# - name: Maximize build space
-# if: matrix.os == 'ubuntu-latest'
-# run: |
-# sudo rm -rf /usr/share/dotnet
-# sudo rm -rf /opt/ghc
-# sudo rm -rf "/usr/local/share/boost"
-# sudo rm -rf "$AGENT_TOOLSDIRECTORY"
-# sudo docker image prune --all --force
-# sudo rm -rf /opt/hostedtoolcache/codeQL
-# sudo rm -rf ${GITHUB_WORKSPACE}/.git
-# sudo rm -rf $ANDROID_HOME/ndk
+ # the following step is required to avoid running out of space
+ - name: Maximize build space
+ if: matrix.os == 'ubuntu-latest'
+ run: |
+ sudo rm -rf /usr/share/dotnet
+ sudo rm -rf /opt/ghc
+ sudo rm -rf "/usr/local/share/boost"
+ sudo rm -rf "$AGENT_TOOLSDIRECTORY"
+ sudo docker image prune --all --force
+ sudo rm -rf /opt/hostedtoolcache/codeQL
+ sudo rm -rf ${GITHUB_WORKSPACE}/.git
-# - name: Check storage space
-# run: df -h
+ - name: Check storage space
+ run: df -h
-# - name: Checkout source code
-# uses: actions/checkout@v4
+ - name: Checkout appflowy cloud code
+ uses: actions/checkout@v4
+ with:
+ repository: AppFlowy-IO/AppFlowy-Cloud
+ path: AppFlowy-Cloud
-# - uses: actions/setup-java@v4
-# with:
-# distribution: temurin
-# java-version: 11
+ - name: Prepare appflowy cloud env
+ working-directory: AppFlowy-Cloud
+ run: |
+ # log level
+ cp deploy.env .env
+ sed -i 's|RUST_LOG=.*|RUST_LOG=trace|' .env
+ sed -i 's/GOTRUE_EXTERNAL_GOOGLE_ENABLED=.*/GOTRUE_EXTERNAL_GOOGLE_ENABLED=true/' .env
+ sed -i 's|GOTRUE_MAILER_AUTOCONFIRM=.*|GOTRUE_MAILER_AUTOCONFIRM=true|' .env
+ sed -i 's|API_EXTERNAL_URL=.*|API_EXTERNAL_URL=http://localhost|' .env
-# - name: Install Rust toolchain
-# id: rust_toolchain
-# uses: actions-rs/toolchain@v1
-# with:
-# toolchain: ${{ env.RUST_TOOLCHAIN }}
-# override: true
-# profile: minimal
+ - name: Run Docker-Compose
+ working-directory: AppFlowy-Cloud
+ env:
+ APPFLOWY_CLOUD_VERSION: ${{ env.CLOUD_VERSION }}
+ APPFLOWY_HISTORY_VERSION: ${{ env.CLOUD_VERSION }}
+ APPFLOWY_WORKER_VERSION: ${{ env.CLOUD_VERSION }}
+ run: |
+ container_id=$(docker ps --filter name=appflowy-cloud-appflowy_cloud-1 -q)
+ if [ -z "$container_id" ]; then
+ echo "AppFlowy-Cloud container is not running. Pulling and starting the container..."
+ docker compose pull
+ docker compose up -d
+ echo "Waiting for the container to be ready..."
+ sleep 10
+ else
+ running_image=$(docker inspect --format='{{index .Config.Image}}' "$container_id")
+ if [ "$running_image" != "appflowy-cloud:$APPFLOWY_CLOUD_VERSION" ]; then
+ echo "AppFlowy-Cloud is running with an incorrect version. Restarting with the correct version..."
+ # Remove all containers if any exist
+ if [ "$(docker ps -aq)" ]; then
+ docker rm -f $(docker ps -aq)
+ else
+ echo "No containers to remove."
+ fi
-# - name: Install flutter
-# id: flutter
-# uses: subosito/flutter-action@v2
-# with:
-# channel: "stable"
-# flutter-version: ${{ env.FLUTTER_VERSION }}
+ # Remove all volumes if any exist
+ if [ "$(docker volume ls -q)" ]; then
+ docker volume rm $(docker volume ls -q)
+ else
+ echo "No volumes to remove."
+ fi
+ docker compose pull
+ docker compose up -d
+ echo "Waiting for the container to be ready..."
+ sleep 10
+ docker ps -a
+ docker compose logs
+ else
+ echo "AppFlowy-Cloud is running with the correct version."
+ fi
+ fi
-# - uses: gradle/gradle-build-action@v3
-# with:
-# gradle-version: 7.4.2
+ - name: Checkout source code
+ uses: actions/checkout@v4
-# - uses: davidB/rust-cargo-make@v1
-# with:
-# version: "0.36.6"
+ - uses: actions/setup-java@v4
+ with:
+ distribution: temurin
+ java-version: 11
-# - name: Install prerequisites
-# working-directory: frontend
-# run: |
-# rustup target install aarch64-linux-android
-# rustup target install x86_64-linux-android
-# cargo install --force duckscript_cli
-# cargo install cargo-ndk
-# if [ "$RUNNER_OS" == "Linux" ]; then
-# sudo wget -qO /etc/apt/trusted.gpg.d/dart_linux_signing_key.asc https://dl-ssl.google.com/linux/linux_signing_key.pub
-# sudo wget -qO /etc/apt/sources.list.d/dart_stable.list https://storage.googleapis.com/download.dartlang.org/linux/debian/dart_stable.list
-# sudo apt-get update
-# sudo apt-get install -y dart curl build-essential libssl-dev clang cmake ninja-build pkg-config libgtk-3-dev
-# sudo apt-get install keybinder-3.0 libnotify-dev
-# sudo apt-get install gcc-multilib
-# elif [ "$RUNNER_OS" == "Windows" ]; then
-# vcpkg integrate install
-# elif [ "$RUNNER_OS" == "macOS" ]; then
-# echo 'do nothing'
-# fi
-# cargo make appflowy-flutter-deps-tools
-# shell: bash
+ - name: Install Rust toolchain
+ id: rust_toolchain
+ uses: actions-rs/toolchain@v1
+ with:
+ toolchain: ${{ env.RUST_TOOLCHAIN }}
+ override: true
+ profile: minimal
-# - name: Build AppFlowy
-# working-directory: frontend
-# run: |
-# cargo make --profile development-android appflowy-android-dev-ci
+ - name: Install flutter
+ id: flutter
+ uses: subosito/flutter-action@v2
+ with:
+ channel: "stable"
+ flutter-version: ${{ env.FLUTTER_VERSION }}
+ - uses: gradle/gradle-build-action@v3
+ with:
+ gradle-version: 8.10
-# - name: Run integration tests
-# # https://github.com/ReactiveCircus/android-emulator-runner
-# uses: reactivecircus/android-emulator-runner@v2
-# with:
-# api-level: 32
-# arch: arm64-v8a
-# disk-size: 2048M
-# working-directory: frontend/appflowy_flutter
-# script: flutter test integration_test/runner.dart
\ No newline at end of file
+ - uses: davidB/rust-cargo-make@v1
+ with:
+ version: ${{ env.CARGO_MAKE_VERSION }}
+
+ - name: Install prerequisites
+ working-directory: frontend
+ run: |
+ rustup target install aarch64-linux-android
+ rustup target install x86_64-linux-android
+ rustup target add armv7-linux-androideabi
+ cargo install --force --locked duckscript_cli
+ cargo install cargo-ndk
+ if [ "$RUNNER_OS" == "Linux" ]; then
+ sudo wget -qO /etc/apt/trusted.gpg.d/dart_linux_signing_key.asc https://dl-ssl.google.com/linux/linux_signing_key.pub
+ sudo wget -qO /etc/apt/sources.list.d/dart_stable.list https://storage.googleapis.com/download.dartlang.org/linux/debian/dart_stable.list
+ sudo apt-get update
+ sudo apt-get install -y dart curl build-essential libssl-dev clang cmake ninja-build pkg-config libgtk-3-dev
+ sudo apt-get install keybinder-3.0 libnotify-dev
+ sudo apt-get install gcc-multilib
+ elif [ "$RUNNER_OS" == "Windows" ]; then
+ vcpkg integrate install
+ elif [ "$RUNNER_OS" == "macOS" ]; then
+ echo 'do nothing'
+ fi
+ cargo make appflowy-flutter-deps-tools
+ shell: bash
+
+ - name: Build AppFlowy
+ working-directory: frontend
+ run: |
+ cargo make --profile development-android appflowy-core-dev-android
+ cargo make --profile development-android code_generation
+ cd rust-lib
+ cargo clean
+
+ - name: Enable KVM group perms
+ run: |
+ echo 'KERNEL=="kvm", GROUP="kvm", MODE="0666", OPTIONS+="static_node=kvm"' | sudo tee /etc/udev/rules.d/99-kvm4all.rules
+ sudo udevadm control --reload-rules
+ sudo udevadm trigger --name-match=kvm
+
+ - name: Run integration tests
+ # https://github.com/ReactiveCircus/android-emulator-runner
+ uses: reactivecircus/android-emulator-runner@v2
+ with:
+ api-level: 33
+ arch: x86_64
+ disk-size: 2048M
+ working-directory: frontend/appflowy_flutter
+ disable-animations: true
+ force-avd-creation: false
+ target: google_apis
+ script: flutter test integration_test/mobile/cloud/cloud_runner.dart
diff --git a/.github/workflows/docker_ci.yml b/.github/workflows/docker_ci.yml
index e38ac4e671..51e8a2ac28 100644
--- a/.github/workflows/docker_ci.yml
+++ b/.github/workflows/docker_ci.yml
@@ -2,9 +2,9 @@ name: Docker-CI
on:
push:
- branches: ["main", "release/*"]
+ branches: [ "main", "release/*" ]
pull_request:
- branches: ["main", "release/*"]
+ branches: [ "main", "release/*" ]
workflow_dispatch:
concurrency:
diff --git a/.github/workflows/flutter_ci.yaml b/.github/workflows/flutter_ci.yaml
index 2dc45f879a..1fc1b0e052 100644
--- a/.github/workflows/flutter_ci.yaml
+++ b/.github/workflows/flutter_ci.yaml
@@ -25,8 +25,8 @@ on:
env:
CARGO_TERM_COLOR: always
- FLUTTER_VERSION: "3.22.2"
- RUST_TOOLCHAIN: "1.80.1"
+ FLUTTER_VERSION: "3.27.4"
+ RUST_TOOLCHAIN: "1.81.0"
CARGO_MAKE_VERSION: "0.37.18"
CLOUD_VERSION: 0.6.54-amd64
@@ -346,7 +346,7 @@ jobs:
fail-fast: false
matrix:
os: [ubuntu-latest]
- test_number: [1, 2, 3, 4, 5, 6, 7, 8]
+ test_number: [1, 2, 3, 4, 5, 6, 7, 8, 9]
include:
- os: ubuntu-latest
target: "x86_64-unknown-linux-gnu"
diff --git a/.github/workflows/ios_ci.yaml b/.github/workflows/ios_ci.yaml
index a39a2704c8..e13863f4a7 100644
--- a/.github/workflows/ios_ci.yaml
+++ b/.github/workflows/ios_ci.yaml
@@ -7,7 +7,6 @@ on:
paths:
- ".github/workflows/mobile_ci.yaml"
- "frontend/**"
- - "!frontend/appflowy_tauri/**"
- "!frontend/appflowy_web_app/**"
pull_request:
@@ -16,13 +15,11 @@ on:
paths:
- ".github/workflows/mobile_ci.yaml"
- "frontend/**"
- - "!frontend/appflowy_tauri/**"
- "!frontend/appflowy_web_app/**"
env:
- FLUTTER_VERSION: "3.22.0"
- RUST_TOOLCHAIN: "1.80.1"
- CLOUD_VERSION: 0.6.51
+ FLUTTER_VERSION: "3.27.4"
+ RUST_TOOLCHAIN: "1.81.0"
concurrency:
group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
@@ -86,7 +83,7 @@ jobs:
working-directory: frontend
run: |
rustup target install aarch64-apple-ios-sim
- cargo install --force duckscript_cli
+ cargo install --force --locked duckscript_cli
cargo install cargo-lipo
cargo make appflowy-flutter-deps-tools
shell: bash
@@ -97,21 +94,26 @@ jobs:
cargo make --profile development-ios-arm64-sim appflowy-core-dev-ios
cargo make --profile development-ios-arm64-sim code_generation
- # - uses: futureware-tech/simulator-action@v3
- # id: simulator-action
- # with:
- # model: "iPhone 15"
- # shutdown_after_job: false
+ - uses: futureware-tech/simulator-action@v3
+ id: simulator-action
+ with:
+ model: "iPhone 15"
+ shutdown_after_job: false
- # - name: Run AppFlowy on simulator
- # working-directory: frontend/appflowy_flutter
- # run: |
- # flutter run -d ${{ steps.simulator-action.outputs.udid }} &
- # pid=$!
- # sleep 500
- # kill $pid
- # continue-on-error: true
+ - name: Run AppFlowy on simulator
+ working-directory: frontend/appflowy_flutter
+ run: |
+ flutter run -d ${{ steps.simulator-action.outputs.udid }} &
+ pid=$!
+ sleep 500
+ kill $pid
+ continue-on-error: true
- # - name: Run integration tests
- # working-directory: frontend/appflowy_flutter
- # run: flutter test integration_test/runner.dart -d ${{ steps.simulator-action.outputs.udid }}
+ # Integration tests
+ - name: Run integration tests
+ working-directory: frontend/appflowy_flutter
+ # The integration tests are flaky and sometimes fail with "Connection timed out":
+ # Don't block the CI. If the tests fail, the CI will still pass.
+ # Instead, we're using Code Magic to re-run the tests to check if they pass.
+ continue-on-error: true
+ run: flutter test integration_test/runner.dart -d ${{ steps.simulator-action.outputs.udid }}
diff --git a/.github/workflows/mobile_ci.yml b/.github/workflows/mobile_ci.yml
new file mode 100644
index 0000000000..4606a67799
--- /dev/null
+++ b/.github/workflows/mobile_ci.yml
@@ -0,0 +1,83 @@
+name: Mobile-CI
+
+on:
+ workflow_dispatch:
+ inputs:
+ branch:
+ description: "Branch to build"
+ required: true
+ default: "main"
+ workflow_id:
+ description: "Codemagic workflow ID"
+ required: true
+ default: "ios-workflow"
+ type: choice
+ options:
+ - ios-workflow
+ - android-workflow
+
+env:
+ CODEMAGIC_API_TOKEN: ${{ secrets.CODEMAGIC_API_TOKEN }}
+ APP_ID: "6731d2f427e7c816080c3674"
+
+jobs:
+ trigger-mobile-build:
+ runs-on: ubuntu-latest
+ steps:
+ - name: Trigger Codemagic Build
+ id: trigger_build
+ run: |
+ RESPONSE=$(curl -X POST \
+ --header "Content-Type: application/json" \
+ --header "x-auth-token: $CODEMAGIC_API_TOKEN" \
+ --data '{
+ "appId": "${{ env.APP_ID }}",
+ "workflowId": "${{ github.event.inputs.workflow_id }}",
+ "branch": "${{ github.event.inputs.branch }}"
+ }' \
+ https://api.codemagic.io/builds)
+
+ BUILD_ID=$(echo $RESPONSE | jq -r '.buildId')
+ echo "build_id=$BUILD_ID" >> $GITHUB_OUTPUT
+ echo "build_id=$BUILD_ID"
+
+ - name: Wait for build and check status
+ id: check_status
+ run: |
+ while true; do
+ curl -X GET \
+ --header "Content-Type: application/json" \
+ --header "x-auth-token: $CODEMAGIC_API_TOKEN" \
+ https://api.codemagic.io/builds/${{ steps.trigger_build.outputs.build_id }} > /tmp/response.json
+
+ RESPONSE_WITHOUT_COMMAND=$(cat /tmp/response.json | jq 'walk(if type == "object" and has("subactions") then .subactions |= map(del(.command)) else . end)')
+ STATUS=$(echo $RESPONSE_WITHOUT_COMMAND | jq -r '.build.status')
+
+ if [ "$STATUS" = "finished" ]; then
+ SUCCESS=$(echo $RESPONSE_WITHOUT_COMMAND | jq -r '.success')
+ BUILD_URL=$(echo $RESPONSE_WITHOUT_COMMAND | jq -r '.buildUrl')
+ echo "status=$STATUS" >> $GITHUB_OUTPUT
+ echo "success=$SUCCESS" >> $GITHUB_OUTPUT
+ echo "build_url=$BUILD_URL" >> $GITHUB_OUTPUT
+ break
+ elif [ "$STATUS" = "failed" ]; then
+ echo "status=failed" >> $GITHUB_OUTPUT
+ break
+ fi
+
+ sleep 60
+ done
+
+ - name: Slack Notification
+ uses: 8398a7/action-slack@v3
+ if: always()
+ with:
+ status: ${{ steps.check_status.outputs.success == 'true' && 'success' || 'failure' }}
+ fields: repo,message,commit,author,action,eventName,ref,workflow,job,took
+ text: |
+ Mobile CI Build Result
+ Branch: ${{ github.event.inputs.branch }}
+ Workflow: ${{ github.event.inputs.workflow_id }}
+ Build URL: ${{ steps.check_status.outputs.build_url }}
+ env:
+ SLACK_WEBHOOK_URL: ${{ secrets.RELEASE_SLACK_WEBHOOK }}
diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml
index 1f2dde57e5..a4582ffa74 100644
--- a/.github/workflows/release.yml
+++ b/.github/workflows/release.yml
@@ -6,8 +6,8 @@ on:
- "*"
env:
- FLUTTER_VERSION: "3.22.0"
- RUST_TOOLCHAIN: "1.77.2"
+ FLUTTER_VERSION: "3.27.4"
+ RUST_TOOLCHAIN: "1.81.0"
jobs:
create-release:
@@ -73,8 +73,8 @@ jobs:
working-directory: frontend
run: |
vcpkg integrate install
- cargo install --force cargo-make
- cargo install --force duckscript_cli
+ cargo install --force --locked cargo-make
+ cargo install --force --locked duckscript_cli
- name: Build Windows app
working-directory: frontend
@@ -135,7 +135,7 @@ jobs:
fail-fast: false
matrix:
job:
- - { target: x86_64-apple-darwin, os: macos-12, extra-build-args: "" }
+ - { target: x86_64-apple-darwin, os: macos-13, extra-build-args: "" }
steps:
- name: Checkout source code
uses: actions/checkout@v4
@@ -158,8 +158,8 @@ jobs:
- name: Install prerequisites
working-directory: frontend
run: |
- cargo install --force cargo-make
- cargo install --force duckscript_cli
+ cargo install --force --locked cargo-make
+ cargo install --force --locked duckscript_cli
- name: Build AppFlowy
working-directory: frontend
@@ -256,8 +256,8 @@ jobs:
- name: Install prerequisites
working-directory: frontend
run: |
- cargo install --force cargo-make
- cargo install --force duckscript_cli
+ cargo install --force --locked cargo-make
+ cargo install --force --locked duckscript_cli
- name: Build AppFlowy
working-directory: frontend
@@ -338,7 +338,7 @@ jobs:
- {
arch: x86_64,
target: x86_64-unknown-linux-gnu,
- os: ubuntu-20.04,
+ os: ubuntu-22.04,
extra-build-args: "",
flutter_profile: production-linux-x86_64,
}
@@ -370,8 +370,8 @@ jobs:
sudo apt-get install keybinder-3.0
sudo apt-get install -y alien libnotify-dev
source $HOME/.cargo/env
- cargo install --force cargo-make
- cargo install --force duckscript_cli
+ cargo install --force --locked cargo-make
+ cargo install --force --locked duckscript_cli
rustup target add ${{ matrix.job.target }}
- name: Install gcc-aarch64-linux-gnu
diff --git a/.github/workflows/rust_ci.yaml b/.github/workflows/rust_ci.yaml
index 2702cbd365..36c2e82064 100644
--- a/.github/workflows/rust_ci.yaml
+++ b/.github/workflows/rust_ci.yaml
@@ -15,85 +15,14 @@ on:
- "main"
- "develop"
- "release/*"
- paths:
- - "frontend/rust-lib/**"
env:
CARGO_TERM_COLOR: always
- CLOUD_VERSION: 0.6.51-amd64
- RUST_TOOLCHAIN: "1.77.2"
+ CLOUD_VERSION: 0.8.3-amd64
+ RUST_TOOLCHAIN: "1.81.0"
jobs:
- self-hosted-job:
- if: github.event.pull_request.head.repo.full_name == github.repository
- runs-on: self-hosted
- steps:
- - name: Checkout source code
- uses: actions/checkout@v4
-
- - name: Checkout Appflowy Cloud
- uses: actions/checkout@v4
- with:
- repository: AppFlowy-IO/AppFlowy-Cloud
- path: AppFlowy-Cloud
-
- - name: Prepare Appflowy Cloud env
- working-directory: AppFlowy-Cloud
- run: |
- cp deploy.env .env
- sed -i '' 's|RUST_LOG=.*|RUST_LOG=trace|' .env
- sed -i '' 's|API_EXTERNAL_URL=.*|API_EXTERNAL_URL=http://localhost|' .env
-
- - name: Ensure AppFlowy-Cloud is Running with Correct Version
- working-directory: AppFlowy-Cloud
- env:
- APPFLOWY_CLOUD_VERSION: ${{ env.CLOUD_VERSION }}
- APPFLOWY_HISTORY_VERSION: ${{ env.CLOUD_VERSION }}
- APPFLOWY_WORKER_VERSION: ${{ env.CLOUD_VERSION }}
- run: |
- container_id=$(docker ps --filter name=appflowy-cloud-appflowy_cloud-1 -q)
- if [ -z "$container_id" ]; then
- echo "AppFlowy-Cloud container is not running. Pulling and starting the container..."
- docker compose pull
- docker compose up -d
- echo "Waiting for the container to be ready..."
- sleep 10
- else
- running_image=$(docker inspect --format='{{index .Config.Image}}' "$container_id")
- if [ "$running_image" != "appflowy-cloud:$APPFLOWY_CLOUD_VERSION" ]; then
- echo "AppFlowy-Cloud is running with an incorrect version. Pulling the correct version..."
- docker compose pull
- docker compose up -d
- echo "Waiting for the container to be ready..."
- sleep 10
- docker ps -a
- docker compose logs
- else
- echo "AppFlowy-Cloud is running with the correct version."
- fi
- fi
-
- - name: Run rust-lib tests
- working-directory: frontend/rust-lib
- env:
- RUST_LOG: info
- RUST_BACKTRACE: 1
- af_cloud_test_base_url: http://localhost
- af_cloud_test_ws_url: ws://localhost/ws/v1
- af_cloud_test_gotrue_url: http://localhost/gotrue
- run: |
- DISABLE_CI_TEST_LOG="true" cargo test --no-default-features --features="dart"
-
- - name: rustfmt rust-lib
- run: cargo fmt --all -- --check
- working-directory: frontend/rust-lib/
-
- - name: clippy rust-lib
- run: cargo clippy --all-targets -- -D warnings
- working-directory: frontend/rust-lib
-
ubuntu-job:
- if: github.event.pull_request.head.repo.full_name != github.repository
runs-on: ubuntu-latest
steps:
- name: Set timezone for action
@@ -137,7 +66,7 @@ jobs:
run: |
cp deploy.env .env
sed -i 's|RUST_LOG=.*|RUST_LOG=trace|' .env
- sed -i 's|GOTRUE_MAILER_AUTOCONFIRM=.*|GOTRUE_MAILER_AUTOCONFIRM=true|' .env
+ sed -i 's|GOTRUE_MAILER_AUTOCONFIRM=.*|GOTRUE_MAILER_AUTOCONFIRM=true|' .env
sed -i 's|API_EXTERNAL_URL=.*|API_EXTERNAL_URL=http://localhost|' .env
- name: Ensure AppFlowy-Cloud is Running with Correct Version
@@ -160,7 +89,7 @@ jobs:
else
echo "No volumes to remove."
fi
-
+
docker compose pull
docker compose up -d
echo "Waiting for the container to be ready..."
diff --git a/.github/workflows/rust_coverage.yml b/.github/workflows/rust_coverage.yml
index 12e728698f..53a5f66748 100644
--- a/.github/workflows/rust_coverage.yml
+++ b/.github/workflows/rust_coverage.yml
@@ -10,8 +10,8 @@ on:
env:
CARGO_TERM_COLOR: always
- FLUTTER_VERSION: "3.22.0"
- RUST_TOOLCHAIN: "1.77.2"
+ FLUTTER_VERSION: "3.27.4"
+ RUST_TOOLCHAIN: "1.81.0"
jobs:
tests:
@@ -40,8 +40,8 @@ jobs:
- name: Install prerequisites
working-directory: frontend
run: |
- cargo install --force cargo-make
- cargo install --force duckscript_cli
+ cargo install --force --locked cargo-make
+ cargo install --force --locked duckscript_cli
- uses: Swatinem/rust-cache@v2
with:
diff --git a/.github/workflows/tauri2_ci.yaml b/.github/workflows/tauri2_ci.yaml
deleted file mode 100644
index 6bbb7928ee..0000000000
--- a/.github/workflows/tauri2_ci.yaml
+++ /dev/null
@@ -1,124 +0,0 @@
-name: Tauri-CI
-
-on:
- pull_request:
- paths:
- - ".github/workflows/tauri2_ci.yaml"
- - "frontend/rust-lib/**"
- - "frontend/appflowy_web_app/**"
- - "frontend/resources/**"
-
-env:
- NODE_VERSION: "18.16.0"
- PNPM_VERSION: "8.5.0"
- RUST_TOOLCHAIN: "1.77.2"
- CARGO_MAKE_VERSION: "0.36.6"
- CI: true
-
-concurrency:
- group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
- cancel-in-progress: true
-
-jobs:
- # tauri-build-self-hosted:
- # if: github.event.pull_request.head.repo.full_name == github.repository
- # runs-on: self-hosted
- #
- # steps:
- # - uses: actions/checkout@v4
- # - name: install frontend dependencies
- # working-directory: frontend/appflowy_web_app
- # run: |
- # mkdir dist
- # pnpm install
- # cd src-tauri && cargo build
- #
- # - name: test and lint
- # working-directory: frontend/appflowy_web_app
- # run: |
- # pnpm run lint:tauri
- #
- # - uses: tauri-apps/tauri-action@v0
- # env:
- # GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
- # with:
- # tauriScript: pnpm tauri
- # projectPath: frontend/appflowy_web_app
- # args: "--debug"
-
- tauri-build-ubuntu:
- #if: github.event.pull_request.head.repo.full_name != github.repository
- runs-on: ubuntu-20.04
-
- steps:
- - uses: actions/checkout@v4
- - name: Maximize build space
- run: |
- sudo rm -rf /usr/share/dotnet
- sudo rm -rf /opt/ghc
- sudo rm -rf "/usr/local/share/boost"
- sudo rm -rf "$AGENT_TOOLSDIRECTORY"
- sudo docker image prune --all --force
- sudo rm -rf /opt/hostedtoolcache/codeQL
- sudo rm -rf ${GITHUB_WORKSPACE}/.git
- sudo rm -rf $ANDROID_HOME/ndk
-
- - name: setup node
- uses: actions/setup-node@v3
- with:
- node-version: ${{ env.NODE_VERSION }}
-
- - name: setup pnpm
- uses: pnpm/action-setup@v2
- with:
- version: ${{ env.PNPM_VERSION }}
-
- - name: Install Rust toolchain
- id: rust_toolchain
- uses: actions-rs/toolchain@v1
- with:
- toolchain: ${{ env.RUST_TOOLCHAIN }}
- override: true
- profile: minimal
-
- - name: Node_modules cache
- uses: actions/cache@v2
- with:
- path: frontend/appflowy_web_app/node_modules
- key: node-modules-${{ runner.os }}
-
- - name: install dependencies
- working-directory: frontend
- run: |
- sudo apt-get update
- sudo apt-get install -y libgtk-3-dev libwebkit2gtk-4.0-dev libappindicator3-dev librsvg2-dev patchelf
-
- - uses: taiki-e/install-action@v2
- with:
- tool: cargo-make@${{ env.CARGO_MAKE_VERSION }}
-
- - name: install tauri deps tools
- working-directory: frontend
- run: |
- cargo make appflowy-tauri-deps-tools
- shell: bash
-
- - name: install frontend dependencies
- working-directory: frontend/appflowy_web_app
- run: |
- mkdir dist
- pnpm install
- cd src-tauri && cargo build
-
- - name: test and lint
- working-directory: frontend/appflowy_web_app
- run: |
- pnpm run lint:tauri
-
- - uses: tauri-apps/tauri-action@v0
- env:
- GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
- with:
- tauriScript: pnpm tauri
- projectPath: frontend/appflowy_web_app
- args: "--debug"
\ No newline at end of file
diff --git a/.github/workflows/tauri_ci.yaml b/.github/workflows/tauri_ci.yaml
deleted file mode 100644
index 70ad621451..0000000000
--- a/.github/workflows/tauri_ci.yaml
+++ /dev/null
@@ -1,111 +0,0 @@
-name: Tauri-CI
-on:
- push:
- branches:
- - build/tauri
-
-env:
- NODE_VERSION: "18.16.0"
- PNPM_VERSION: "8.5.0"
- RUST_TOOLCHAIN: "1.77.2"
-
-concurrency:
- group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
- cancel-in-progress: true
-
-jobs:
- tauri-build:
- if: github.event.pull_request.draft != true
- strategy:
- fail-fast: false
- matrix:
- platform: [ ubuntu-20.04 ]
-
- runs-on: ${{ matrix.platform }}
-
- env:
- CI: true
- steps:
- - uses: actions/checkout@v4
-
- - name: Maximize build space (ubuntu only)
- if: matrix.platform == 'ubuntu-20.04'
- run: |
- sudo rm -rf /usr/share/dotnet
- sudo rm -rf /opt/ghc
- sudo rm -rf "/usr/local/share/boost"
- sudo rm -rf "$AGENT_TOOLSDIRECTORY"
- sudo docker image prune --all --force
- sudo rm -rf /opt/hostedtoolcache/codeQL
- sudo rm -rf ${GITHUB_WORKSPACE}/.git
- sudo rm -rf $ANDROID_HOME/ndk
-
- - name: setup node
- uses: actions/setup-node@v3
- with:
- node-version: ${{ env.NODE_VERSION }}
-
- - name: setup pnpm
- uses: pnpm/action-setup@v2
- with:
- version: ${{ env.PNPM_VERSION }}
-
- - name: Install Rust toolchain
- id: rust_toolchain
- uses: actions-rs/toolchain@v1
- with:
- toolchain: ${{ env.RUST_TOOLCHAIN }}
- override: true
- profile: minimal
-
- - name: Rust cache
- uses: swatinem/rust-cache@v2
- with:
- workspaces: "./frontend/appflowy_tauri/src-tauri -> target"
-
- - name: Node_modules cache
- uses: actions/cache@v2
- with:
- path: frontend/appflowy_tauri/node_modules
- key: node-modules-${{ runner.os }}
-
- - name: install dependencies (windows only)
- if: matrix.platform == 'windows-latest'
- working-directory: frontend
- run: |
- cargo install --force duckscript_cli
- vcpkg integrate install
-
- - name: install dependencies (ubuntu only)
- if: matrix.platform == 'ubuntu-20.04'
- working-directory: frontend
- run: |
- sudo apt-get update
- sudo apt-get install -y libgtk-3-dev libwebkit2gtk-4.0-dev libappindicator3-dev librsvg2-dev patchelf
-
- - name: install cargo-make
- working-directory: frontend
- run: |
- cargo install --force cargo-make
- cargo make appflowy-tauri-deps-tools
-
- - name: install frontend dependencies
- working-directory: frontend/appflowy_tauri
- run: |
- mkdir dist
- pnpm install
- cargo make --cwd .. tauri_build
-
- - name: frontend tests and linting
- working-directory: frontend/appflowy_tauri
- run: |
- pnpm test
- pnpm test:errors
-
- - uses: tauri-apps/tauri-action@v0
- env:
- GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
- with:
- tauriScript: pnpm tauri
- projectPath: frontend/appflowy_tauri
- args: "--debug"
\ No newline at end of file
diff --git a/.github/workflows/tauri_release.yml b/.github/workflows/tauri_release.yml
deleted file mode 100644
index 7de80b017e..0000000000
--- a/.github/workflows/tauri_release.yml
+++ /dev/null
@@ -1,153 +0,0 @@
-name: Publish Tauri Release
-
-on:
- workflow_dispatch:
- inputs:
- branch:
- description: 'The branch to release'
- required: true
- default: 'main'
- version:
- description: 'The version to release'
- required: true
- default: '0.0.0'
-env:
- NODE_VERSION: "18.16.0"
- PNPM_VERSION: "8.5.0"
- RUST_TOOLCHAIN: "1.77.2"
-
-jobs:
-
- publish-tauri:
- permissions:
- contents: write
- strategy:
- fail-fast: false
- matrix:
- settings:
- - platform: windows-latest
- args: "--verbose"
- target: "windows-x86_64"
- - platform: macos-latest
- args: "--target x86_64-apple-darwin"
- target: "macos-x86_64"
- - platform: ubuntu-20.04
- args: "--target x86_64-unknown-linux-gnu"
- target: "linux-x86_64"
-
- runs-on: ${{ matrix.settings.platform }}
-
- env:
- CI: true
- PACKAGE_PREFIX: AppFlowy_Tauri-${{ github.event.inputs.version }}-${{ matrix.settings.target }}
- steps:
- - uses: actions/checkout@v4
- with:
- ref: ${{ github.event.inputs.branch }}
-
- - name: Maximize build space (ubuntu only)
- if: matrix.settings.platform == 'ubuntu-20.04'
- run: |
- sudo rm -rf /usr/share/dotnet
- sudo rm -rf /opt/ghc
- sudo rm -rf "/usr/local/share/boost"
- sudo rm -rf "$AGENT_TOOLSDIRECTORY"
- sudo docker image prune --all --force
- sudo rm -rf /opt/hostedtoolcache/codeQL
- sudo rm -rf ${GITHUB_WORKSPACE}/.git
- sudo rm -rf $ANDROID_HOME/ndk
-
- - name: setup node
- uses: actions/setup-node@v3
- with:
- node-version: ${{ env.NODE_VERSION }}
-
- - name: setup pnpm
- uses: pnpm/action-setup@v2
- with:
- version: ${{ env.PNPM_VERSION }}
-
- - name: Install Rust toolchain
- id: rust_toolchain
- uses: actions-rs/toolchain@v1
- with:
- toolchain: ${{ env.RUST_TOOLCHAIN }}
- override: true
- profile: minimal
-
- - name: Rust cache
- uses: swatinem/rust-cache@v2
- with:
- workspaces: "./frontend/appflowy_tauri/src-tauri -> target"
-
- - name: install dependencies (windows only)
- if: matrix.settings.platform == 'windows-latest'
- working-directory: frontend
- run: |
- cargo install --force duckscript_cli
- vcpkg integrate install
-
- - name: install dependencies (ubuntu only)
- if: matrix.settings.platform == 'ubuntu-20.04'
- working-directory: frontend
- run: |
- sudo apt-get update
- sudo apt-get install -y libgtk-3-dev libwebkit2gtk-4.0-dev libappindicator3-dev librsvg2-dev patchelf
-
- - name: install cargo-make
- working-directory: frontend
- run: |
- cargo install --force cargo-make
- cargo make appflowy-tauri-deps-tools
-
- - name: install frontend dependencies
- working-directory: frontend/appflowy_tauri
- run: |
- mkdir dist
- pnpm install
- pnpm exec node scripts/update_version.cjs ${{ github.event.inputs.version }}
- cargo make --cwd .. tauri_build
-
- - uses: tauri-apps/tauri-action@dev
- env:
- GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
- APPLE_CERTIFICATE: ${{ secrets.MACOS_CERTIFICATE }}
- APPLE_CERTIFICATE_PASSWORD: ${{ secrets.MACOS_CERTIFICATE_PWD }}
- APPLE_SIGNING_IDENTITY: ${{ secrets.MACOS_TEAM_ID }}
- APPLE_ID: ${{ secrets.MACOS_NOTARY_USER }}
- APPLE_TEAM_ID: ${{ secrets.MACOS_TEAM_ID }}
- APPLE_PASSWORD: ${{ secrets.MACOS_NOTARY_PWD }}
- CI: true
- with:
- args: ${{ matrix.settings.args }}
- appVersion: ${{ github.event.inputs.version }}
- tauriScript: pnpm tauri
- projectPath: frontend/appflowy_tauri
-
- - name: Upload EXE package(windows only)
- uses: actions/upload-artifact@v4
- if: matrix.settings.platform == 'windows-latest'
- with:
- name: ${{ env.PACKAGE_PREFIX }}.exe
- path: frontend/appflowy_tauri/src-tauri/target/release/bundle/nsis/AppFlowy_${{ github.event.inputs.version }}_x64-setup.exe
-
- - name: Upload DMG package(macos only)
- uses: actions/upload-artifact@v4
- if: matrix.settings.platform == 'macos-latest'
- with:
- name: ${{ env.PACKAGE_PREFIX }}.dmg
- path: frontend/appflowy_tauri/src-tauri/target/x86_64-apple-darwin/release/bundle/dmg/AppFlowy_${{ github.event.inputs.version }}_x64.dmg
-
- - name: Upload Deb package(ubuntu only)
- uses: actions/upload-artifact@v4
- if: matrix.settings.platform == 'ubuntu-20.04'
- with:
- name: ${{ env.PACKAGE_PREFIX }}.deb
- path: frontend/appflowy_tauri/src-tauri/target/x86_64-unknown-linux-gnu/release/bundle/deb/app-flowy_${{ github.event.inputs.version }}_amd64.deb
-
- - name: Upload AppImage package(ubuntu only)
- uses: actions/upload-artifact@v4
- if: matrix.settings.platform == 'ubuntu-20.04'
- with:
- name: ${{ env.PACKAGE_PREFIX }}.AppImage
- path: frontend/appflowy_tauri/src-tauri/target/x86_64-unknown-linux-gnu/release/bundle/appimage/app-flowy_${{ github.event.inputs.version }}_amd64.AppImage
diff --git a/.github/workflows/web2_ci.yaml b/.github/workflows/web2_ci.yaml
deleted file mode 100644
index c52f71dd84..0000000000
--- a/.github/workflows/web2_ci.yaml
+++ /dev/null
@@ -1,75 +0,0 @@
-name: Web-CI
-on:
- pull_request:
- paths:
- - ".github/workflows/web2_ci.yaml"
- - "frontend/appflowy_web_app/**"
- - "frontend/resources/**"
-env:
- NODE_VERSION: "18.16.0"
- PNPM_VERSION: "8.5.0"
-concurrency:
- group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
- cancel-in-progress: true
-jobs:
- web-build:
- if: github.event.pull_request.draft != true
- strategy:
- fail-fast: false
- matrix:
- platform: [ ubuntu-20.04 ]
-
- runs-on: ${{ matrix.platform }}
-
- steps:
- - uses: actions/checkout@v4
- - name: Maximize build space (ubuntu only)
- if: matrix.platform == 'ubuntu-20.04'
- run: |
- sudo rm -rf /usr/share/dotnet
- sudo rm -rf /opt/ghc
- sudo rm -rf "/usr/local/share/boost"
- sudo rm -rf "$AGENT_TOOLSDIRECTORY"
- sudo docker image prune --all --force
- sudo rm -rf /opt/hostedtoolcache/codeQL
- sudo rm -rf ${GITHUB_WORKSPACE}/.git
- sudo rm -rf $ANDROID_HOME/ndk
- - name: setup node
- uses: actions/setup-node@v3
- with:
- node-version: ${{ env.NODE_VERSION }}
- - name: setup pnpm
- uses: pnpm/action-setup@v2
- with:
- version: ${{ env.PNPM_VERSION }}
- - name: Node_modules cache
- uses: actions/cache@v2
- with:
- path: frontend/appflowy_web_app/node_modules
- key: node-modules-${{ runner.os }}
-
- - name: install frontend dependencies
- working-directory: frontend/appflowy_web_app
- run: |
- pnpm install
- - name: Run lint check
- working-directory: frontend/appflowy_web_app
- run: |
- pnpm run lint
-
- - name: build and analyze
- working-directory: frontend/appflowy_web_app
- run: |
- pnpm run analyze >> analyze-size.txt
- - name: Upload analyze-size.txt
- uses: actions/upload-artifact@v4
- with:
- name: analyze-size.txt
- path: frontend/appflowy_web_app/analyze-size.txt
- retention-days: 30
- - name: Upload stats.html
- uses: actions/upload-artifact@v4
- with:
- name: stats.html
- path: frontend/appflowy_web_app/dist/stats.html
- retention-days: 30
diff --git a/.github/workflows/web_coverage.yaml b/.github/workflows/web_coverage.yaml
deleted file mode 100644
index 7803f719c9..0000000000
--- a/.github/workflows/web_coverage.yaml
+++ /dev/null
@@ -1,65 +0,0 @@
-name: Web Code Coverage
-
-on:
- pull_request:
- paths:
- - ".github/workflows/web2_ci.yaml"
- - "frontend/appflowy_web_app/**"
- - "frontend/resources/**"
-
-env:
- NODE_VERSION: "18.16.0"
- PNPM_VERSION: "8.5.0"
-concurrency:
- group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
- cancel-in-progress: true
-jobs:
- test:
- if: github.event.pull_request.draft != true
- runs-on: ubuntu-22.04
- steps:
- - uses: actions/checkout@v4
- - name: Maximize build space (ubuntu only)
- run: |
- sudo rm -rf /usr/share/dotnet
- sudo rm -rf /opt/ghc
- sudo rm -rf "/usr/local/share/boost"
- sudo rm -rf "$AGENT_TOOLSDIRECTORY"
- sudo docker image prune --all --force
- sudo rm -rf /opt/hostedtoolcache/codeQL
- sudo rm -rf ${GITHUB_WORKSPACE}/.git
- sudo rm -rf $ANDROID_HOME/ndk
- - name: setup node
- uses: actions/setup-node@v3
- with:
- node-version: ${{ env.NODE_VERSION }}
- - name: setup pnpm
- uses: pnpm/action-setup@v2
- with:
- version: ${{ env.PNPM_VERSION }}
- # Install pnpm dependencies, cache them correctly
- # and run all Cypress tests
- - name: Cypress run
- uses: cypress-io/github-action@v6
- with:
- working-directory: frontend/appflowy_web_app
- component: true
- build: pnpm run build
- start: pnpm run start
- browser: chrome
-
- - name: Jest run
- working-directory: frontend/appflowy_web_app
- run: |
- pnpm run test:unit
-
- - name: Upload coverage to Codecov
- uses: codecov/codecov-action@v2
- with:
- token: cf9245e0-e136-4e21-b0ee-35755fa0c493
- files: frontend/appflowy_web_app/coverage/jest/lcov.info,frontend/appflowy_web_app/coverage/cypress/lcov.info
- flags: appflowy_web_app
- name: frontend/appflowy_web_app
- fail_ci_if_error: true
- verbose: true
-
diff --git a/CHANGELOG.md b/CHANGELOG.md
index dc9e7bc897..a5e7e268a5 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,4 +1,202 @@
# Release Notes
+## Version 0.8.9 - 16/04/2025
+### Desktop
+#### New Features
+- Supported pasting a link as a mention, providing a more condensed visualization of linked content
+- Supported converting between link formats (e.g. transforming a mention into a bookmark)
+- Improved the link editing experience with enhanced UX
+- Added OTP (One-Time Password) support for sign-in authentication
+- Added latest AI models: GPT-4.1, GPT-4.1-mini, and Claude 3.7 Sonnet
+#### Bug Fixes
+- Fixed an issue where properties were not displaying in the row detail page
+- Fixed a bug where Undo didn't work in the row detail page
+- Fixed an issue where blocks didn't grow when the grid got bigger
+- Fixed several bugs related to AI writers
+### Mobile
+#### New Features
+- Added sign-in with OTP (One-Time Password)
+#### Bug Fixes
+- Fixed an issue where the slash menu sometimes failed to display
+- Updated the mention page block to handle page selection with more context.
+
+## Version 0.8.8 - 01/04/2025
+### New Features
+- Added support for selecting AI models in AI writer
+- Revamped link menu in toolbar
+- Added support for using ":" to add emojis in documents
+- Passed the history of past AI prompts and responses to AI writer
+### Bug Fixes
+- Improved AI writer scrolling user experience
+- Fixed issue where checklist items would disappear during reordering
+- Fixed numbered lists generated by AI to maintain the same index as the input
+
+## Version 0.8.7 - 18/03/2025
+### New Features
+- Made local AI free and integrated with Ollama
+- Supported nested lists within callout and quote blocks
+- Revamped the document's floating toolbar and added Turn Into
+- Enabled custom icons in callout blocks
+### Bug Fixes
+- Fixed occasional incorrect positioning of the slash menu
+- Improved AI Chat and AI Writers with various bug fixes
+- Adjusted the columns block to match the width of the editor
+- Fixed a potential segfault caused by infinite recursion in the trash view
+- Resolved an issue where the first added cover might be invisible
+- Fixed adding cover images via Unsplash
+
+## Version 0.8.6 - 06/03/2025
+### Bug Fixes
+- Fix the incorrect title positioning when adjusting the document width setting
+- Enhance the user experience of the icon color picker for smoother interactions
+- Add missing icons to the database to ensure completeness and consistency
+- Resolve the issue with links not functioning correctly on Linux systems
+- Improve the outline feature to work seamlessly within columns
+- Center the bulleted list icon within columns for better visual alignment
+- Enable dragging blocks under tables in the second column to enhance flexibility
+- Disable the AI writer feature within tables to prevent conflicts and improve usability
+- Automatically enable the header row when converting content from Markdown to ensure proper formatting
+- Use the "Undo" function to revert the auto-formatting
+
+## Version 0.8.5 - 04/03/2025
+### New Features
+- Columns in Documents: Arrange content side by side using drag-and-drop or the slash menu
+- AI Writers: New AI assistants in documents with response formatting options (list, table, text with images, image-only), follow-up questions, contextual memory, and more
+- Compact Mode for Databases: Enable compact mode for grid and kanban views (full-page and inline) to increase information density, displaying more data per screen
+### Bug Fixes
+- Fixed an issue where callout blocks couldn’t be deleted when appearing as the first line in a document
+- Fixed a bug preventing the relation field in databases from opening
+- Fixed an issue where links in documents were unclickable on Linux
+
+## Version 0.8.4 - 18/02/2025
+### New Features
+- Switch AI mode on mobile
+- Support locking page
+- Support uploading svg file as icon
+- Support the slash, at, and plus menus on mobile
+### Bug Fixes
+- Gallery not rendering in row page
+- Save image should not copy the image (mobile)
+- Support exporting more content to markdown
+
+## Version 0.8.2 - 23/01/2025
+### New Features
+- Customized database view icons
+- Support for uploading images as custom icons
+- Enabled selecting multiple AI messages to save into a document
+- Added the ability to scale the app's display size on mobile
+- Support for pasting image links without file extensions
+### Bug Fixes
+- Fixed an issue where pasting tables from other apps wasn't working
+- Fixed homepage URL issues in Settings
+- Fixed an issue where the 'Cancel' button was not visible on the Shortcuts page
+
+## Version 0.8.1 - 14/01/2025
+### New Features
+- AI Chat Layout Options: Customize how AI responses appear with new layouts—List, Table, Image with Text, and Media Only
+- DALL-E Integration: Generate stunning AI images from text prompts, now available in AI Chat
+- Improved Desktop Search: Find what you need faster using keywords or by asking questions in natural language
+- Self-Hosting: Configure web server URLs directly in Settings to enable features like Publish, Copy Link to Share, Custom URLs, and more
+- Sidebar Enhancement: Drag to reorder your favorited pages in the Sidebar
+- Mobile Table Resizing: Adjust column widths in Simple Tables by long pressing the column borders on mobile
+### Bug Fixes
+- Resolved an icon rendering issue in callout blocks, tab bars, and search results
+- Enhanced image reliability: Retry functionality ensures images load successfully if the first attempt fails
+
+## Version 0.8.0 - 06/01/2025
+### Bug Fixes
+- Fixed error displaying in the page style menu
+- Fixed filter logic in the icon picker
+- Fixed error displaying in the Favorite/Recent page
+- Fixed the color picker displaying when tapping down
+- Fixed icons not being supported in subpage blocks
+- Fixed recent icon functionality in the space icon menu
+- Fixed "Insert Below" not auto-scrolling the table
+- Fixed a to-do item with an emoji automatically creating a soft break
+- Fixed header row/column tap areas being too small
+- Fixed simple table alignment not working for items that wrap
+- Fixed web content reverting after removing the inline code format on desktop
+- Fixed inability to make changes to a row or column in the table when opening a new tab
+- Fixed changing the language to CKB-KU causing a gray screen on mobile
+
+## Version 0.7.9 - 30/12/2024
+### New Features
+- Meet AppFlowy Web (Lite): Use AppFlowy directly in your browser.
+ - Create beautiful documents with 22 content types and markdown support
+ - Use Quick Note to save anything you want to remember—like meeting notes, a grocery list, or to-dos
+ - Invite members to your workspace for seamless collaboration
+ - Create multiple public/private spaces to better organize your content
+- Simple Table is now available on Mobile, designed specifically for mobile devices.
+ - Create and manage Simple Table blocks on Mobile with easy-to-use action menus.
+ - Use the '+' button in the fixed toolbar to easily add a content block into a table cell on Mobile
+ - Use '/' to insert a content block into a table cell on Desktop
+- Add pages as AI sources in AI chat, enabling you to ask questions about the selected sources
+- Add messages to an editable document while chatting with AI side by side
+- The new Emoji menu now includes Icons with a Recent section for quickly reusing emojis/icons
+- Drag a page from the sidebar into a document to easily mention the page without typing its title
+- Paste as plain text, a new option in the right-click paste menu
+### Bug Fixes
+- Fixed misalignment in numbered lists
+- Resolved several bugs in the emoji menu
+- Fixed a bug with checklist items
+
+## Version 0.7.8 - 18/12/2024
+### New Features
+
+
+- Meet Simple Table 2.0:
+ - Insert a list into a table cell
+ - Insert images, quotes, callouts, and code blocks into a table cell
+ - Drag to move rows or columns
+ - Toggle header rows or columns on/off
+ - Distribute columns evenly
+ - Adjust to page width
+- Enjoy a new UI/UX for a seamless experience
+- Revamped mention page interactions in AI Chat
+- Improved AppFlowy AI service
+
+### Bug Fixes
+- Fixed an error when opening files in the database in local mode
+- Fixed arrow up/down navigation not working for selecting a language in Code Block
+- Fixed an issue where deleting multiple blocks using the drag button on the document page didn’t work
+
+## Version 0.7.7 - 09/12/2024
+### Bug Fixes
+- Fixed sidebar menu resize regression
+- Fixed AI chat loading issues
+- Fixed inability to open local files in database
+- Fixed mentions remaining in notifications after removal from document
+- Fixed event card closing when clicking on empty space
+- Fixed keyboard shortcut issues
+
+## Version 0.7.6 - 03/12/2024
+### New Features
+- Revamped the simple table UI
+- Added support for capturing images from camera on mobile
+### Bug Fixes
+- Improved markdown rendering capabilities in AI writer
+- Fixed an issue where pressing Enter on a collapsed toggle list would add an unnecessary new line
+- Fixed an issue where creating a document from slash menu could insert content at incorrect position
+
+## Version 0.7.5 - 25/11/2024
+### Bug Fixes
+- Improved chat response parsing
+- Fixed toggle list icon direction for RTL mode
+- Fixed cross blocks formatting not reflecting in float toolbar
+- Fixed unable to click inside the toggle list to create a new paragraph
+- Fixed open file error 50 on macOS
+- Fixed upload file exceed limit error
+
+## Version 0.7.4 - 19/11/2024
+### New Features
+- Support uploading WebP and BMP images
+- Support managing workspaces on mobile
+- Support adding toggle headings on mobile
+- Improve the AI chat page UI
+### Bug Fixes
+- Optimized the workspace menu loading performance
+- Optimized tab switching performance
+- Fixed searching issues in Document page
+
## Version 0.7.3 - 07/11/2024
### New Features
- Enable custom URLs for published pages
@@ -919,4 +1117,4 @@ Bug fixes and improvements
- Increased height of action
- CPU performance issue
- Fix potential data parser error
-- More foundation work for online collaboration
+- More foundation work for online collaboration
\ No newline at end of file
diff --git a/README.md b/README.md
index 6f62079bb5..565908e756 100644
--- a/README.md
+++ b/README.md
@@ -1,6 +1,6 @@
- AppFlowy.IO
+ AppFlowy
⭐️ The Open Source Alternative To Notion ⭐️
@@ -18,18 +18,18 @@ AppFlowy is the AI workspace where you achieve more without losing control of yo
- Website •
+ Website •
Forum •
Discord •
Reddit •
Twitter
-
-
-
-
-
+
+
+
+
+
@@ -42,11 +42,13 @@ AppFlowy is the AI workspace where you achieve more without losing control of yo
## User Installation
- [Download AppFlowy Desktop (macOS, Windows, and Linux)](https://github.com/AppFlowy-IO/AppFlowy/releases)
-- Other channels: [FlatHub](https://flathub.org/apps/io.appflowy.AppFlowy), [Snapcraft](https://snapcraft.io/appflowy), [Sourceforge](https://sourceforge.net/projects/appflowy/)
+- Other
+ channels: [FlatHub](https://flathub.org/apps/io.appflowy.AppFlowy), [Snapcraft](https://snapcraft.io/appflowy), [Sourceforge](https://sourceforge.net/projects/appflowy/)
- Available on
- - [App Store](https://apps.apple.com/app/appflowy/id6457261352): iPhone
- - [Play Store](https://play.google.com/store/apps/details?id=io.appflowy.appflowy): Android 10 or above; ARMv7 is not supported
-- [Self-hosting AppFlowy](https://docs.appflowy.io/docs/guides/appflowy/self-hosting-appflowy)
+ - [App Store](https://apps.apple.com/app/appflowy/id6457261352): iPhone
+ - [Play Store](https://play.google.com/store/apps/details?id=io.appflowy.appflowy): Android 10 or above; ARMv7 is
+ not supported
+- [Self-hosting AppFlowy](https://appflowy.com/docs/self-host-appflowy-overview)
- [Source](https://docs.appflowy.io/docs/documentation/appflowy/from-source)
## Built With
@@ -61,32 +63,41 @@ AppFlowy is the AI workspace where you achieve more without losing control of yo
## Getting Started with development
-Please view the [documentation](https://docs.appflowy.io/docs/documentation/appflowy/from-source) for OS specific development instructions
+Please view the [documentation](https://docs.appflowy.io/docs/documentation/appflowy/from-source) for OS specific
+development instructions
## Roadmap
- [AppFlowy Roadmap ReadMe](https://docs.appflowy.io/docs/appflowy/roadmap)
- [AppFlowy Public Roadmap](https://github.com/orgs/AppFlowy-IO/projects/5/views/12)
-If you'd like to propose a feature, submit a feature request [here](https://github.com/AppFlowy-IO/AppFlowy/issues/new?assignees=&labels=&template=feature_request.yaml&title=%5BFR%5D+)
-If you'd like to report a bug, submit a bug report [here](https://github.com/AppFlowy-IO/AppFlowy/issues/new?assignees=&labels=&template=bug_report.yaml&title=%5BBug%5D+)
+If you'd like to propose a feature, submit a feature
+request [here](https://github.com/AppFlowy-IO/AppFlowy/issues/new?assignees=&labels=&template=feature_request.yaml&title=%5BFR%5D+)
+If you'd like to report a bug, submit a bug
+report [here](https://github.com/AppFlowy-IO/AppFlowy/issues/new?assignees=&labels=&template=bug_report.yaml&title=%5BBug%5D+)
## **Releases**
-Please see the [changelog](https://www.appflowy.io/whatsnew) for more details about a given release.
+Please see the [changelog](https://appflowy.com/what-is-new) for more details about a given release.
## Contributing
-Contributions make the open-source community a fantastic place to learn, inspire, and create. Any contributions you make are **greatly appreciated**. Please look at [Contributing to AppFlowy](https://docs.appflowy.io/docs/documentation/software-contributions/contributing-to-appflowy) for details.
+Contributions make the open-source community a fantastic place to learn, inspire, and create. Any contributions you make
+are **greatly appreciated**. Please look
+at [Contributing to AppFlowy](https://docs.appflowy.io/docs/documentation/software-contributions/contributing-to-appflowy)
+for details.
-If your Pull Request is accepted as it fixes a bug, adds functionality, or makes AppFlowy's codebase significantly easier to use or understand, **Congratulations!** If your administrative and managerial work behind the scenes sustains the community, **Congratulations!** You are now an official contributor to AppFlowy. Get in touch with us ([link](https://tally.so/r/mKP5z3)) to receive the very special Contributor T-shirt!
-Proudly wear your T-shirt and show it to us by tagging [@appflowy](https://twitter.com/appflowy) on Twitter.
+If your Pull Request is accepted as it fixes a bug, adds functionality, or makes AppFlowy's codebase significantly
+easier to use or understand, **Congratulations!** If your administrative and managerial work behind the scenes sustains
+the community, **Congratulations!** You are now an official contributor to AppFlowy.
## Translations 🌎🗺
[](https://inlang.com/editor/github.com/AppFlowy-IO/AppFlowy?ref=badge)
-To add translations, you can manually edit the JSON translation files in `/frontend/resources/translations`, use the [inlang online editor](https://inlang.com/editor/github.com/AppFlowy-IO/AppFlowy), or run `npx inlang machine translate` to add missing translations.
+To add translations, you can manually edit the JSON translation files in `/frontend/resources/translations`, use
+the [inlang online editor](https://inlang.com/editor/github.com/AppFlowy-IO/AppFlowy), or
+run `npx inlang machine translate` to add missing translations.
## Join the community to build AppFlowy together
@@ -96,16 +107,30 @@ To add translations, you can manually edit the JSON translation files in `/front
## Why Are We Building This?
-Notion has been our favourite project and knowledge management tool in recent years because of its aesthetic appeal and functionality. Our team uses it daily, and we are on its paid plan. However, as we all know, Notion has its limitations. These include weak data security and poor compatibility with mobile devices. Likewise, alternative collaborative workplace management tools also have their constraints.
+Notion has been our favourite project and knowledge management tool in recent years because of its aesthetic appeal and
+functionality. Our team uses it daily, and we are on its paid plan. However, as we all know, Notion has its limitations.
+These include weak data security and poor compatibility with mobile devices. Likewise, alternative collaborative
+workplace management tools also have their constraints.
-The limitations we encountered using these tools and our past work experience with collaborative productivity tools have led to our firm belief that there is a glass ceiling on what's possible for these tools in the future. This emanates from the fact that these tools will probably struggle to scale horizontally at some point and be forced to prioritize a proportion of customers whose needs differ from the rest. While decision-makers want a workplace OS, it is impossible to come up with a one-size fits all solution in such a fragmented market.
+The limitations we encountered using these tools and our past work experience with collaborative productivity tools have
+led to our firm belief that there is a glass ceiling on what's possible for these tools in the future. This emanates
+from the fact that these tools will probably struggle to scale horizontally at some point and be forced to prioritize a
+proportion of customers whose needs differ from the rest. While decision-makers want a workplace OS, it is impossible to
+come up with a one-size fits all solution in such a fragmented market.
-When a customer's evolving core needs are not satisfied, they either switch to another or build one from the ground up, in-house. Consequently, they either go under another ceiling or buy an expensive ticket to learn a hard lesson. This is a requirement for many resources and expertise, building a reliable and easy-to-use collaborative tool, not to mention the speed and native experience. The same may apply to individual users as well.
+When a customer's evolving core needs are not satisfied, they either switch to another or build one from the ground up,
+in-house. Consequently, they either go under another ceiling or buy an expensive ticket to learn a hard lesson. This is
+a requirement for many resources and expertise, building a reliable and easy-to-use collaborative tool, not to mention
+the speed and native experience. The same may apply to individual users as well.
-All these restrictions necessitate our mission - to make it possible for anyone to create apps that suit their needs well.
+All these restrictions necessitate our mission - to make it possible for anyone to create apps that suit their needs
+well.
- To individuals, we would like to offer Notion's functionality, data security, and cross-platform native experience.
-- To enterprises and hackers, AppFlowy is dedicated to offering building blocks and collaboration infra services to enable you to make apps on your own. Moreover, you have 100% control of your data. You can design and modify AppFlowy your way, with a single codebase written in Flutter and Rust supporting multiple platforms armed with long-term maintainability.
+- To enterprises and hackers, AppFlowy is dedicated to offering building blocks and collaboration infra services to
+ enable you to make apps on your own. Moreover, you have 100% control of your data. You can design and modify AppFlowy
+ your way, with a single codebase written in Flutter and Rust supporting multiple platforms armed with long-term
+ maintainability.
We decided to achieve this mission by upholding the three most fundamental values:
@@ -113,16 +138,20 @@ We decided to achieve this mission by upholding the three most fundamental value
- Reliable native experience
- Community-driven extensibility
-We do not claim to outperform Notion in terms of functionality and design, at least for now. Besides, our priority doesn't lie in more functionality at the moment. Instead, we would like to cultivate a community to democratize the knowledge and wheels of making complex workplace management tools while enabling people and businesses to create beautiful things on their own by equipping them with a versatile toolbox of building blocks.
+We do not claim to outperform Notion in terms of functionality and design, at least for now. Besides, our priority
+doesn't lie in more functionality at the moment. Instead, we would like to cultivate a community to democratize the
+knowledge and wheels of making complex workplace management tools while enabling people and businesses to create
+beautiful things on their own by equipping them with a versatile toolbox of building blocks.
## License
-Distributed under the AGPLv3 License. See [`LICENSE.md`](https://github.com/AppFlowy-IO/AppFlowy/blob/main/LICENSE) for more information.
+Distributed under the AGPLv3 License. See [`LICENSE.md`](https://github.com/AppFlowy-IO/AppFlowy/blob/main/LICENSE) for
+more information.
## Acknowledgments
-Special thanks to these amazing projects which help power AppFlowy.IO:
+Special thanks to these amazing projects which help power AppFlowy:
-- [flutter-quill](https://github.com/singerdmx/flutter-quill)
- [cargo-make](https://github.com/sagiegurari/cargo-make)
- [contrib.rocks](https://contrib.rocks)
+- [flutter_chat_ui](https://pub.dev/packages/flutter_chat_ui)
diff --git a/codemagic.yaml b/codemagic.yaml
new file mode 100644
index 0000000000..9ba2a1a562
--- /dev/null
+++ b/codemagic.yaml
@@ -0,0 +1,47 @@
+workflows:
+ ios-workflow:
+ name: iOS Workflow
+ instance_type: mac_mini_m2
+ max_build_duration: 30
+ environment:
+ flutter: 3.27.4
+ xcode: latest
+ cocoapods: default
+
+ scripts:
+ - name: Build Flutter
+ script: |
+ curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y
+ source "$HOME/.cargo/env"
+ rustc --version
+ cargo --version
+
+ cd frontend
+
+ rustup target install aarch64-apple-ios-sim
+ cargo install --force cargo-make
+ cargo install --force --locked duckscript_cli
+ cargo install --force cargo-lipo
+
+ cargo make appflowy-flutter-deps-tools
+ cargo make --profile development-ios-arm64-sim appflowy-core-dev-ios
+ cargo make --profile development-ios-arm64-sim code_generation
+
+ - name: iOS integration tests
+ script: |
+ cd frontend/appflowy_flutter
+ flutter emulators --launch apple_ios_simulator
+ flutter -d iPhone test integration_test/runner.dart
+
+ artifacts:
+ - build/ios/ipa/*.ipa
+ - /tmp/xcodebuild_logs/*.log
+ - flutter_drive.log
+
+ publishing:
+ email:
+ recipients:
+ - lucas.xu@appflowy.io
+ notify:
+ success: true
+ failure: true
diff --git a/frontend/.vscode/launch.json b/frontend/.vscode/launch.json
index 09965baee1..d4ff85a2dd 100644
--- a/frontend/.vscode/launch.json
+++ b/frontend/.vscode/launch.json
@@ -1,140 +1,125 @@
{
- // Use IntelliSense to learn about possible attributes.
- // Hover to view descriptions of existing attributes.
- // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
- "version": "0.2.0",
- "configurations": [
- {
- // This task only builds the Dart code of AppFlowy.
- // It supports both the desktop and mobile version.
- "name": "AF: Build Dart Only",
- "request": "launch",
- "program": "./lib/main.dart",
- "type": "dart",
- "env": {
- "RUST_LOG": "debug",
- },
- // uncomment the following line to testing performance.
- // "flutterMode": "profile",
- "cwd": "${workspaceRoot}/appflowy_flutter"
- },
- {
- // This task builds the Rust and Dart code of AppFlowy.
- "name": "AF-desktop: Build All",
- "request": "launch",
- "program": "./lib/main.dart",
- "type": "dart",
- "preLaunchTask": "AF: Build Appflowy Core",
- "env": {
- "RUST_LOG": "trace",
- "RUST_BACKTRACE": "1"
- },
- "cwd": "${workspaceRoot}/appflowy_flutter"
- },
- {
- // This task builds will:
- // - call the clean task,
- // - rebuild all the generated Files (including freeze and language files)
- // - rebuild the the Rust and Dart code of AppFlowy.
- "name": "AF-desktop: Clean + Rebuild All",
- "request": "launch",
- "program": "./lib/main.dart",
- "type": "dart",
- "preLaunchTask": "AF: Clean + Rebuild All",
- "env": {
- "RUST_LOG": "trace"
- },
- "cwd": "${workspaceRoot}/appflowy_flutter"
- },
- {
- "name": "AF-iOS: Build All",
- "request": "launch",
- "program": "./lib/main.dart",
- "type": "dart",
- "preLaunchTask": "AF: Build Appflowy Core For iOS",
- "env": {
- "RUST_LOG": "trace"
- },
- "cwd": "${workspaceRoot}/appflowy_flutter"
- },
- {
- "name": "AF-iOS: Clean + Rebuild All",
- "request": "launch",
- "program": "./lib/main.dart",
- "type": "dart",
- "preLaunchTask": "AF: Clean + Rebuild All (iOS)",
- "env": {
- "RUST_LOG": "trace"
- },
- "cwd": "${workspaceRoot}/appflowy_flutter"
- },
- {
- "name": "AF-iOS-Simulator: Build All",
- "request": "launch",
- "program": "./lib/main.dart",
- "type": "dart",
- "preLaunchTask": "AF: Build Appflowy Core For iOS Simulator",
- "env": {
- "RUST_LOG": "trace"
- },
- "cwd": "${workspaceRoot}/appflowy_flutter"
- },
- {
- "name": "AF-iOS-Simulator: Clean + Rebuild All",
- "request": "launch",
- "program": "./lib/main.dart",
- "type": "dart",
- "preLaunchTask": "AF: Clean + Rebuild All (iOS Simulator)",
- "env": {
- "RUST_LOG": "trace"
- },
- "cwd": "${workspaceRoot}/appflowy_flutter"
- },
- {
- "name": "AF-Android: Build All",
- "request": "launch",
- "program": "./lib/main.dart",
- "type": "dart",
- "preLaunchTask": "AF: Build Appflowy Core For Android",
- "env": {
- "RUST_LOG": "trace"
- },
- "cwd": "${workspaceRoot}/appflowy_flutter"
- },
- {
- "name": "AF-Android: Clean + Rebuild All",
- "request": "launch",
- "program": "./lib/main.dart",
- "type": "dart",
- "preLaunchTask": "AF: Clean + Rebuild All (Android)",
- "env": {
- "RUST_LOG": "trace"
- },
- "cwd": "${workspaceRoot}/appflowy_flutter"
- },
- {
- "name": "AF-desktop: Debug Rust",
- "type": "lldb",
- "request": "attach",
- "pid": "${command:pickMyProcess}"
- // To launch the application directly, use the following configuration:
- // "request": "launch",
- // "program": "[YOUR_APPLICATION_PATH]",
- },
- {
- // https://tauri.app/v1/guides/debugging/vs-code
- "type": "lldb",
- "request": "launch",
- "name": "AF-tauri: Debug backend",
- "cargo": {
- "args": [
- "build",
- "--manifest-path=./appflowy_tauri/src-tauri/Cargo.toml",
- "--no-default-features"
- ]
- },
- "preLaunchTask": "AF: Tauri UI Dev",
- "cwd": "${workspaceRoot}/appflowy_tauri/"
- },
- ]
+ // Use IntelliSense to learn about possible attributes.
+ // Hover to view descriptions of existing attributes.
+ // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
+ "version": "0.2.0",
+ "configurations": [
+ {
+ // This task only builds the Dart code of AppFlowy.
+ // It supports both the desktop and mobile version.
+ "name": "AF: Build Dart Only",
+ "request": "launch",
+ "program": "./lib/main.dart",
+ "type": "dart",
+ "env": {
+ "RUST_LOG": "debug",
+ },
+ // uncomment the following line to testing performance.
+ // "flutterMode": "profile",
+ "cwd": "${workspaceRoot}/appflowy_flutter"
+ },
+ {
+ // This task builds the Rust and Dart code of AppFlowy.
+ "name": "AF-desktop: Build All",
+ "request": "launch",
+ "program": "./lib/main.dart",
+ "type": "dart",
+ "preLaunchTask": "AF: Build Appflowy Core",
+ "env": {
+ "RUST_LOG": "trace",
+ "RUST_BACKTRACE": "1"
+ },
+ "cwd": "${workspaceRoot}/appflowy_flutter"
+ },
+ {
+ // This task builds will:
+ // - call the clean task,
+ // - rebuild all the generated Files (including freeze and language files)
+ // - rebuild the the Rust and Dart code of AppFlowy.
+ "name": "AF-desktop: Clean + Rebuild All",
+ "request": "launch",
+ "program": "./lib/main.dart",
+ "type": "dart",
+ "preLaunchTask": "AF: Clean + Rebuild All",
+ "env": {
+ "RUST_LOG": "trace"
+ },
+ "cwd": "${workspaceRoot}/appflowy_flutter"
+ },
+ {
+ "name": "AF-iOS: Build All",
+ "request": "launch",
+ "program": "./lib/main.dart",
+ "type": "dart",
+ "preLaunchTask": "AF: Build Appflowy Core For iOS",
+ "env": {
+ "RUST_LOG": "trace"
+ },
+ "cwd": "${workspaceRoot}/appflowy_flutter"
+ },
+ {
+ "name": "AF-iOS: Clean + Rebuild All",
+ "request": "launch",
+ "program": "./lib/main.dart",
+ "type": "dart",
+ "preLaunchTask": "AF: Clean + Rebuild All (iOS)",
+ "env": {
+ "RUST_LOG": "trace"
+ },
+ "cwd": "${workspaceRoot}/appflowy_flutter"
+ },
+ {
+ "name": "AF-iOS-Simulator: Build All",
+ "request": "launch",
+ "program": "./lib/main.dart",
+ "type": "dart",
+ "preLaunchTask": "AF: Build Appflowy Core For iOS Simulator",
+ "env": {
+ "RUST_LOG": "trace"
+ },
+ "cwd": "${workspaceRoot}/appflowy_flutter"
+ },
+ {
+ "name": "AF-iOS-Simulator: Clean + Rebuild All",
+ "request": "launch",
+ "program": "./lib/main.dart",
+ "type": "dart",
+ "preLaunchTask": "AF: Clean + Rebuild All (iOS Simulator)",
+ "env": {
+ "RUST_LOG": "trace"
+ },
+ "cwd": "${workspaceRoot}/appflowy_flutter"
+ },
+ {
+ "name": "AF-Android: Build All",
+ "request": "launch",
+ "program": "./lib/main.dart",
+ "type": "dart",
+ "preLaunchTask": "AF: Build Appflowy Core For Android",
+ "env": {
+ "RUST_LOG": "trace"
+ },
+ "cwd": "${workspaceRoot}/appflowy_flutter"
+ },
+ {
+ "name": "AF-Android: Clean + Rebuild All",
+ "request": "launch",
+ "program": "./lib/main.dart",
+ "type": "dart",
+ "preLaunchTask": "AF: Clean + Rebuild All (Android)",
+ "env": {
+ "RUST_LOG": "trace"
+ },
+ "cwd": "${workspaceRoot}/appflowy_flutter"
+ },
+ {
+ "name": "AF-desktop: Debug Rust",
+ "type": "lldb",
+ "request": "attach",
+ "pid": "${command:pickMyProcess}"
+ // To launch the application directly, use the following configuration:
+ // "request": "launch",
+ // "program": "[YOUR_APPLICATION_PATH]",
+ },
+ ]
}
diff --git a/frontend/.vscode/tasks.json b/frontend/.vscode/tasks.json
index d940eef0a8..0be167fb12 100644
--- a/frontend/.vscode/tasks.json
+++ b/frontend/.vscode/tasks.json
@@ -245,51 +245,6 @@
"problemMatcher": [],
"detail": "appflowy_flutter"
},
- {
- "label": "AF: Tauri UI Build",
- "type": "shell",
- "command": "pnpm run build",
- "options": {
- "cwd": "${workspaceFolder}/appflowy_tauri"
- }
- },
- {
- "label": "AF: Tauri UI Dev",
- "type": "shell",
- "isBackground": true,
- "command": "pnpm sync:i18n && pnpm run dev",
- "options": {
- "cwd": "${workspaceFolder}/appflowy_tauri"
- }
- },
- {
- "label": "AF: Tauri Clean",
- "type": "shell",
- "command": "cargo make tauri_clean",
- "options": {
- "cwd": "${workspaceFolder}"
- }
- },
- {
- "label": "AF: Tauri Clean + Dev",
- "type": "shell",
- "dependsOrder": "sequence",
- "dependsOn": [
- "AF: Tauri Clean",
- "AF: Tauri UI Dev"
- ],
- "options": {
- "cwd": "${workspaceFolder}"
- }
- },
- {
- "label": "AF: Tauri ESLint",
- "type": "shell",
- "command": "npx eslint --fix src",
- "options": {
- "cwd": "${workspaceFolder}/appflowy_tauri"
- }
- },
{
"label": "AF: Generate Env File",
"type": "shell",
diff --git a/frontend/Makefile.toml b/frontend/Makefile.toml
index d2d915f5b9..41fdffb1af 100644
--- a/frontend/Makefile.toml
+++ b/frontend/Makefile.toml
@@ -26,7 +26,7 @@ CARGO_MAKE_EXTEND_WORKSPACE_MAKEFILE = true
CARGO_MAKE_CRATE_FS_NAME = "dart_ffi"
CARGO_MAKE_CRATE_NAME = "dart-ffi"
LIB_NAME = "dart_ffi"
-APPFLOWY_VERSION = "0.7.3"
+APPFLOWY_VERSION = "0.8.9"
FLUTTER_DESKTOP_FEATURES = "dart"
PRODUCT_NAME = "AppFlowy"
MACOSX_DEPLOYMENT_TARGET = "11.0"
diff --git a/frontend/appflowy_flutter/analysis_options.yaml b/frontend/appflowy_flutter/analysis_options.yaml
index 8da401ef26..4579b2d8c5 100644
--- a/frontend/appflowy_flutter/analysis_options.yaml
+++ b/frontend/appflowy_flutter/analysis_options.yaml
@@ -4,6 +4,7 @@ analyzer:
exclude:
- "**/*.g.dart"
- "**/*.freezed.dart"
+ - "packages/**/*.dart"
linter:
rules:
diff --git a/frontend/appflowy_flutter/android/app/build.gradle b/frontend/appflowy_flutter/android/app/build.gradle
index 3110b5b8ff..0b96e32472 100644
--- a/frontend/appflowy_flutter/android/app/build.gradle
+++ b/frontend/appflowy_flutter/android/app/build.gradle
@@ -53,7 +53,7 @@ android {
// TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html).
applicationId "io.appflowy.appflowy"
minSdkVersion 29
- targetSdkVersion 34
+ targetSdkVersion 35
versionCode flutterVersionCode.toInteger()
versionName flutterVersionName
multiDexEnabled true
diff --git a/frontend/appflowy_flutter/android/app/src/main/AndroidManifest.xml b/frontend/appflowy_flutter/android/app/src/main/AndroidManifest.xml
index 279b17320c..f746eeb610 100644
--- a/frontend/appflowy_flutter/android/app/src/main/AndroidManifest.xml
+++ b/frontend/appflowy_flutter/android/app/src/main/AndroidManifest.xml
@@ -36,7 +36,6 @@
-
+
-
+
@@ -65,4 +67,5 @@
-->
+
\ No newline at end of file
diff --git a/frontend/appflowy_flutter/assets/fonts/.gitkeep b/frontend/appflowy_flutter/assets/fonts/.gitkeep
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/frontend/appflowy_flutter/assets/fonts/FlowyIconData.ttf b/frontend/appflowy_flutter/assets/fonts/FlowyIconData.ttf
deleted file mode 100644
index 8f03a5c8f9..0000000000
Binary files a/frontend/appflowy_flutter/assets/fonts/FlowyIconData.ttf and /dev/null differ
diff --git a/frontend/appflowy_flutter/assets/test/images/sample.svg b/frontend/appflowy_flutter/assets/test/images/sample.svg
new file mode 100644
index 0000000000..7dcd6907d8
--- /dev/null
+++ b/frontend/appflowy_flutter/assets/test/images/sample.svg
@@ -0,0 +1,4 @@
+
+
\ No newline at end of file
diff --git a/frontend/appflowy_flutter/assets/translations/mr-IN.json b/frontend/appflowy_flutter/assets/translations/mr-IN.json
new file mode 100644
index 0000000000..f86a1e0081
--- /dev/null
+++ b/frontend/appflowy_flutter/assets/translations/mr-IN.json
@@ -0,0 +1,3210 @@
+{
+ "appName": "AppFlowy",
+ "defaultUsername": "मी",
+ "welcomeText": "@:appName मध्ये आ पले स्वागत आहे.",
+ "welcomeTo": "मध्ये आ पले स्वागत आ हे",
+ "githubStarText": "GitHub वर स्टार करा",
+ "subscribeNewsletterText": "वृत्तपत्राची सदस्यता घ्या",
+ "letsGoButtonText": "क्विक स्टार्ट",
+ "title": "Title",
+ "youCanAlso": "तुम्ही देखील",
+ "and": "आ णि",
+ "failedToOpenUrl": "URL उघडण्यात अयशस्वी: {}",
+ "blockActions": {
+ "addBelowTooltip": "खाली जोडण्यासाठी क्लिक करा",
+ "addAboveCmd": "Alt+click",
+ "addAboveMacCmd": "Option+click",
+ "addAboveTooltip": "वर जोडण्यासाठी",
+ "dragTooltip": "Drag to move",
+ "openMenuTooltip": "मेनू उघडण्यासाठी क्लिक करा"
+ },
+ "signUp": {
+ "buttonText": "साइन अप",
+ "title": "साइन अप to @:appName",
+ "getStartedText": "सुरुवात करा",
+ "emptyPasswordError": "पासवर्ड रिकामा असू शकत नाही",
+ "repeatPasswordEmptyError": "Repeat पासवर्ड रिकामा असू शकत नाही",
+ "unmatchedPasswordError": "पुन्हा लिहिलेला पासवर्ड मूळ पासवर्डशी जुळत नाही",
+ "alreadyHaveAnAccount": "आधीच खाते आहे?",
+ "emailHint": "Email",
+ "passwordHint": "Password",
+ "repeatPasswordHint": "पासवर्ड पुन्हा लिहा",
+ "signUpWith": "यामध्ये साइन अप करा:"
+ },
+ "signIn": {
+ "loginTitle": "@:appName मध्ये लॉगिन करा",
+ "loginButtonText": "लॉगिन",
+ "loginStartWithAnonymous": "अनामिक सत्रासह पुढे जा",
+ "continueAnonymousUser": "अनामिक सत्रासह पुढे जा",
+ "anonymous": "अनामिक",
+ "buttonText": "साइन इन",
+ "signingInText": "साइन इन होत आहे...",
+ "forgotPassword": "पासवर्ड विसरलात?",
+ "emailHint": "ईमेल",
+ "passwordHint": "पासवर्ड",
+ "dontHaveAnAccount": "तुमचं खाते नाही?",
+ "createAccount": "खाते तयार करा",
+ "repeatPasswordEmptyError": "पुन्हा पासवर्ड रिकामा असू शकत नाही",
+ "unmatchedPasswordError": "पुन्हा लिहिलेला पासवर्ड मूळ पासवर्डशी जुळत नाही",
+ "syncPromptMessage": "डेटा सिंक होण्यास थोडा वेळ लागू शकतो. कृपया हे पृष्ठ बंद करू नका",
+ "or": "किंवा",
+ "signInWithGoogle": "Google सह पुढे जा",
+ "signInWithGithub": "GitHub सह पुढे जा",
+ "signInWithDiscord": "Discord सह पुढे जा",
+ "signInWithApple": "Apple सह पुढे जा",
+ "continueAnotherWay": "इतर पर्यायांनी पुढे जा",
+ "signUpWithGoogle": "Google सह साइन अप करा",
+ "signUpWithGithub": "GitHub सह साइन अप करा",
+ "signUpWithDiscord": "Discord सह साइन अप करा",
+ "signInWith": "यासह पुढे जा:",
+ "signInWithEmail": "ईमेलसह पुढे जा",
+ "signInWithMagicLink": "पुढे जा",
+ "signUpWithMagicLink": "Magic Link सह साइन अप करा",
+ "pleaseInputYourEmail": "कृपया तुमचा ईमेल पत्ता टाका",
+ "settings": "सेटिंग्ज",
+ "magicLinkSent": "Magic Link पाठवण्यात आली आहे!",
+ "invalidEmail": "कृपया वैध ईमेल पत्ता टाका",
+ "alreadyHaveAnAccount": "आधीच खाते आहे?",
+ "logIn": "लॉगिन",
+ "generalError": "काहीतरी चुकलं. कृपया नंतर प्रयत्न करा",
+ "limitRateError": "सुरक्षेच्या कारणास्तव, तुम्ही दर ६० सेकंदांतून एकदाच Magic Link मागवू शकता",
+ "magicLinkSentDescription": "तुमच्या ईमेलवर Magic Link पाठवण्यात आली आहे. लॉगिन पूर्ण करण्यासाठी लिंकवर क्लिक करा. ही लिंक ५ मिनिटांत कालबाह्य होईल."
+ },
+ "workspace": {
+ "chooseWorkspace": "तुमचे workspace निवडा",
+ "defaultName": "माझे Workspace",
+ "create": "नवीन workspace तयार करा",
+ "new": "नवीन workspace",
+ "importFromNotion": "Notion मधून आयात करा",
+ "learnMore": "अधिक जाणून घ्या",
+ "reset": "workspace रीसेट करा",
+ "renameWorkspace": "workspace चे नाव बदला",
+ "workspaceNameCannotBeEmpty": "workspace चे नाव रिकामे असू शकत नाही",
+ "resetWorkspacePrompt": "workspace रीसेट केल्यास त्यातील सर्व पृष्ठे आणि डेटा हटवले जातील. तुम्हाला workspace रीसेट करायचे आहे का? पर्यायी म्हणून तुम्ही सपोर्ट टीमशी संपर्क करू शकता.",
+ "hint": "workspace",
+ "notFoundError": "workspace सापडले नाही",
+ "failedToLoad": "काहीतरी चूक झाली! workspace लोड होण्यात अयशस्वी. कृपया @:appName चे कोणतेही उघडे instance बंद करा आणि पुन्हा प्रयत्न करा.",
+ "errorActions": {
+ "reportIssue": "समस्या नोंदवा",
+ "reportIssueOnGithub": "Github वर समस्या नोंदवा",
+ "exportLogFiles": "लॉग फाइल्स निर्यात करा",
+ "reachOut": "Discord वर संपर्क करा"
+ },
+ "menuTitle": "कार्यक्षेत्रे",
+ "deleteWorkspaceHintText": "तुम्हाला हे कार्यक्षेत्र हटवायचे आहे का? ही कृती पूर्ववत केली जाऊ शकत नाही आणि तुम्ही प्रकाशित केलेली कोणतीही पृष्ठे अप्रकाशित होतील.",
+ "createSuccess": "कार्यक्षेत्र यशस्वीरित्या तयार झाले",
+ "createFailed": "कार्यक्षेत्र तयार करण्यात अयशस्वी",
+ "createLimitExceeded": "तुम्ही तुमच्या खात्यासाठी परवानगी दिलेल्या कार्यक्षेत्र मर्यादेपर्यंत पोहोचलात. अधिक कार्यक्षेत्रासाठी GitHub वर विनंती करा.",
+ "deleteSuccess": "कार्यक्षेत्र यशस्वीरित्या हटवले गेले",
+ "deleteFailed": "कार्यक्षेत्र हटवण्यात अयशस्वी",
+ "openSuccess": "कार्यक्षेत्र यशस्वीरित्या उघडले",
+ "openFailed": "कार्यक्षेत्र उघडण्यात अयशस्वी",
+ "renameSuccess": "कार्यक्षेत्राचे नाव यशस्वीरित्या बदलले",
+ "renameFailed": "कार्यक्षेत्राचे नाव बदलण्यात अयशस्वी",
+ "updateIconSuccess": "कार्यक्षेत्राचे चिन्ह यशस्वीरित्या अद्यतनित केले",
+ "updateIconFailed": "कार्यक्षेत्राचे चिन्ह अद्यतनित करण्यात अयशस्वी",
+ "cannotDeleteTheOnlyWorkspace": "फक्त एकच कार्यक्षेत्र असल्यास ते हटवता येत नाही",
+ "fetchWorkspacesFailed": "कार्यक्षेत्रे मिळवण्यात अयशस्वी",
+ "leaveCurrentWorkspace": "कार्यक्षेत्र सोडा",
+ "leaveCurrentWorkspacePrompt": "तुम्हाला हे कार्यक्षेत्र सोडायचे आहे का?"
+ },
+ "shareAction": {
+ "buttonText": "शेअर करा",
+ "workInProgress": "लवकरच येत आहे",
+ "markdown": "Markdown",
+ "html": "HTML",
+ "clipboard": "क्लिपबोर्डवर कॉपी करा",
+ "csv": "CSV",
+ "copyLink": "लिंक कॉपी करा",
+ "publishToTheWeb": "वेबवर प्रकाशित करा",
+ "publishToTheWebHint": "AppFlowy सह वेबसाइट तयार करा",
+ "publish": "प्रकाशित करा",
+ "unPublish": "अप्रकाशित करा",
+ "visitSite": "साइटला भेट द्या",
+ "exportAsTab": "या स्वरूपात निर्यात करा",
+ "publishTab": "प्रकाशित करा",
+ "shareTab": "शेअर करा",
+ "publishOnAppFlowy": "AppFlowy वर प्रकाशित करा",
+ "shareTabTitle": "सहकार्य करण्यासाठी आमंत्रित करा",
+ "shareTabDescription": "कोणासही सहज सहकार्य करण्यासाठी",
+ "copyLinkSuccess": "लिंक क्लिपबोर्डवर कॉपी केली",
+ "copyShareLink": "शेअर लिंक कॉपी करा",
+ "copyLinkFailed": "लिंक क्लिपबोर्डवर कॉपी करण्यात अयशस्वी",
+ "copyLinkToBlockSuccess": "ब्लॉकची लिंक क्लिपबोर्डवर कॉपी केली",
+ "copyLinkToBlockFailed": "ब्लॉकची लिंक क्लिपबोर्डवर कॉपी करण्यात अयशस्वी",
+ "manageAllSites": "सर्व साइट्स व्यवस्थापित करा",
+ "updatePathName": "पथाचे नाव अपडेट करा"
+ },
+ "moreAction": {
+ "small": "लहान",
+ "medium": "मध्यम",
+ "large": "मोठा",
+ "fontSize": "फॉन्ट आकार",
+ "import": "Import",
+ "moreOptions": "अधिक पर्याय",
+ "wordCount": "शब्द संख्या: {}",
+ "charCount": "अक्षर संख्या: {}",
+ "createdAt": "निर्मिती: {}",
+ "deleteView": "हटवा",
+ "duplicateView": "प्रत बनवा",
+ "wordCountLabel": "शब्द संख्या: ",
+ "charCountLabel": "अक्षर संख्या: ",
+ "createdAtLabel": "निर्मिती: ",
+ "syncedAtLabel": "सिंक केले: ",
+ "saveAsNewPage": "संदेश पृष्ठात जोडा",
+ "saveAsNewPageDisabled": "उपलब्ध संदेश नाहीत"
+ },
+ "importPanel": {
+ "textAndMarkdown": "मजकूर आणि Markdown",
+ "documentFromV010": "v0.1.0 पासून दस्तऐवज",
+ "databaseFromV010": "v0.1.0 पासून डेटाबेस",
+ "notionZip": "Notion निर्यात केलेली Zip फाईल",
+ "csv": "CSV",
+ "database": "डेटाबेस"
+ },
+ "emojiIconPicker": {
+ "iconUploader": {
+ "placeholderLeft": "फाईल ड्रॅग आणि ड्रॉप करा, क्लिक करा ",
+ "placeholderUpload": "अपलोड",
+ "placeholderRight": ", किंवा इमेज लिंक पेस्ट करा.",
+ "dropToUpload": "अपलोडसाठी फाईल ड्रॉप करा",
+ "change": "बदला"
+ }
+ },
+ "disclosureAction": {
+ "rename": "नाव बदला",
+ "delete": "हटवा",
+ "duplicate": "प्रत बनवा",
+ "unfavorite": "आवडतीतून काढा",
+ "favorite": "आवडतीत जोडा",
+ "openNewTab": "नवीन टॅबमध्ये उघडा",
+ "moveTo": "या ठिकाणी हलवा",
+ "addToFavorites": "आवडतीत जोडा",
+ "copyLink": "लिंक कॉपी करा",
+ "changeIcon": "आयकॉन बदला",
+ "collapseAllPages": "सर्व उपपृष्ठे संकुचित करा",
+ "movePageTo": "पृष्ठ हलवा",
+ "move": "हलवा",
+ "lockPage": "पृष्ठ लॉक करा"
+ },
+ "blankPageTitle": "रिक्त पृष्ठ",
+ "newPageText": "नवीन पृष्ठ",
+ "newDocumentText": "नवीन दस्तऐवज",
+ "newGridText": "नवीन ग्रिड",
+ "newCalendarText": "नवीन कॅलेंडर",
+ "newBoardText": "नवीन बोर्ड",
+ "chat": {
+ "newChat": "AI गप्पा",
+ "inputMessageHint": "@:appName AI ला विचार करा",
+ "inputLocalAIMessageHint": "@:appName लोकल AI ला विचार करा",
+ "unsupportedCloudPrompt": "ही सुविधा फक्त @:appName Cloud वापरताना उपलब्ध आहे",
+ "relatedQuestion": "सूचवलेले",
+ "serverUnavailable": "कनेक्शन गमावले. कृपया तुमचे इंटरनेट तपासा आणि पुन्हा प्रयत्न करा",
+ "aiServerUnavailable": "AI सेवा सध्या अनुपलब्ध आहे. कृपया नंतर पुन्हा प्रयत्न करा.",
+ "retry": "पुन्हा प्रयत्न करा",
+ "clickToRetry": "पुन्हा प्रयत्न करण्यासाठी क्लिक करा",
+ "regenerateAnswer": "उत्तर पुन्हा तयार करा",
+ "question1": "Kanban वापरून कामे कशी व्यवस्थापित करायची",
+ "question2": "GTD पद्धत समजावून सांगा",
+ "question3": "Rust का वापरावा",
+ "question4": "माझ्या स्वयंपाकघरात असलेल्या वस्तूंनी रेसिपी",
+ "question5": "माझ्या पृष्ठासाठी एक चित्र तयार करा",
+ "question6": "या आठवड्याची माझी कामांची यादी तयार करा",
+ "aiMistakePrompt": "AI चुकू शकतो. महत्त्वाची माहिती तपासा.",
+ "chatWithFilePrompt": "तुम्हाला फाइलसोबत गप्पा मारायच्या आहेत का?",
+ "indexFileSuccess": "फाईल यशस्वीरित्या अनुक्रमित केली गेली",
+ "inputActionNoPages": "काहीही पृष्ठे सापडली नाहीत",
+ "referenceSource": {
+ "zero": "0 स्रोत सापडले",
+ "one": "{count} स्रोत सापडला",
+ "other": "{count} स्रोत सापडले"
+ }
+ },
+ "clickToMention": "पृष्ठाचा उल्लेख करा",
+ "uploadFile": "PDFs, मजकूर किंवा markdown फाइल्स जोडा",
+ "questionDetail": "नमस्कार {}! मी तुम्हाला कशी मदत करू शकतो?",
+ "indexingFile": "{} अनुक्रमित करत आहे",
+ "generatingResponse": "उत्तर तयार होत आहे",
+ "selectSources": "स्रोत निवडा",
+ "currentPage": "सध्याचे पृष्ठ",
+ "sourcesLimitReached": "तुम्ही फक्त ३ मुख्य दस्तऐवज आणि त्यांची उपदस्तऐवज निवडू शकता",
+ "sourceUnsupported": "सध्या डेटाबेससह चॅटिंगसाठी आम्ही समर्थन करत नाही",
+ "regenerate": "पुन्हा प्रयत्न करा",
+ "addToPageButton": "संदेश पृष्ठावर जोडा",
+ "addToPageTitle": "या पृष्ठात संदेश जोडा...",
+ "addToNewPage": "नवीन पृष्ठ तयार करा",
+ "addToNewPageName": "\"{}\" मधून काढलेले संदेश",
+ "addToNewPageSuccessToast": "संदेश जोडण्यात आला",
+ "openPagePreviewFailedToast": "पृष्ठ उघडण्यात अयशस्वी",
+ "changeFormat": {
+ "actionButton": "फॉरमॅट बदला",
+ "confirmButton": "या फॉरमॅटसह पुन्हा तयार करा",
+ "textOnly": "मजकूर",
+ "imageOnly": "फक्त प्रतिमा",
+ "textAndImage": "मजकूर आणि प्रतिमा",
+ "text": "परिच्छेद",
+ "bullet": "बुलेट यादी",
+ "number": "क्रमांकित यादी",
+ "table": "सारणी",
+ "blankDescription": "उत्तराचे फॉरमॅट ठरवा",
+ "defaultDescription": "स्वयंचलित उत्तर फॉरमॅट",
+ "textWithImageDescription": "@:chat.changeFormat.text प्रतिमेसह",
+ "numberWithImageDescription": "@:chat.changeFormat.number प्रतिमेसह",
+ "bulletWithImageDescription": "@:chat.changeFormat.bullet प्रतिमेसह",
+ " tableWithImageDescription": "@:chat.changeFormat.table प्रतिमेसह"
+ },
+ "switchModel": {
+ "label": "मॉडेल बदला",
+ "localModel": "स्थानिक मॉडेल",
+ "cloudModel": "क्लाऊड मॉडेल",
+ "autoModel": "स्वयंचलित"
+ },
+ "selectBanner": {
+ "saveButton": "… मध्ये जोडा",
+ "selectMessages": "संदेश निवडा",
+ "nSelected": "{} निवडले गेले",
+ "allSelected": "सर्व निवडले गेले"
+ },
+ "stopTooltip": "उत्पन्न करणे थांबवा",
+ "trash": {
+ "text": "कचरा",
+ "restoreAll": "सर्व पुनर्संचयित करा",
+ "restore": "पुनर्संचयित करा",
+ "deleteAll": "सर्व हटवा",
+ "pageHeader": {
+ "fileName": "फाईलचे नाव",
+ "lastModified": "शेवटचा बदल",
+ "created": "निर्मिती"
+ }
+ },
+ "confirmDeleteAll": {
+ "title": "कचरापेटीतील सर्व पृष्ठे",
+ "caption": "तुम्हाला कचरापेटीतील सर्व काही हटवायचे आहे का? ही कृती पूर्ववत केली जाऊ शकत नाही."
+ },
+ "confirmRestoreAll": {
+ "title": "कचरापेटीतील सर्व पृष्ठे पुनर्संचयित करा",
+ "caption": "ही कृती पूर्ववत केली जाऊ शकत नाही."
+ },
+ "restorePage": {
+ "title": "पुनर्संचयित करा: {}",
+ "caption": "तुम्हाला हे पृष्ठ पुनर्संचयित करायचे आहे का?"
+ },
+ "mobile": {
+ "actions": "कचरा क्रिया",
+ "empty": "कचरापेटीत कोणतीही पृष्ठे किंवा जागा नाहीत",
+ "emptyDescription": "जे आवश्यक नाही ते कचरापेटीत हलवा.",
+ "isDeleted": "हटवले गेले आहे",
+ "isRestored": "पुनर्संचयित केले गेले आहे"
+ },
+ "confirmDeleteTitle": "तुम्हाला हे पृष्ठ कायमचे हटवायचे आहे का?",
+ "deletePagePrompt": {
+ "text": "हे पृष्ठ कचरापेटीत आहे",
+ "restore": "पृष्ठ पुनर्संचयित करा",
+ "deletePermanent": "कायमचे हटवा",
+ "deletePermanentDescription": "तुम्हाला हे पृष्ठ कायमचे हटवायचे आहे का? ही कृती पूर्ववत केली जाऊ शकत नाही."
+ },
+ "dialogCreatePageNameHint": "पृष्ठाचे नाव",
+ "questionBubble": {
+ "shortcuts": "शॉर्टकट्स",
+ "whatsNew": "नवीन काय आहे?",
+ "help": "मदत आणि समर्थन",
+ "markdown": "Markdown",
+ "debug": {
+ "name": "डीबग माहिती",
+ "success": "डीबग माहिती क्लिपबोर्डवर कॉपी केली!",
+ "fail": "डीबग माहिती कॉपी करता आली नाही"
+ },
+ "feedback": "अभिप्राय"
+ },
+ "menuAppHeader": {
+ "moreButtonToolTip": "हटवा, नाव बदला आणि अधिक...",
+ "addPageTooltip": "तत्काळ एक पृष्ठ जोडा",
+ "defaultNewPageName": "शीर्षक नसलेले",
+ "renameDialog": "नाव बदला",
+ "pageNameSuffix": "प्रत"
+ },
+ "noPagesInside": "अंदर कोणतीही पृष्ठे नाहीत",
+ "toolbar": {
+ "undo": "पूर्ववत करा",
+ "redo": "पुन्हा करा",
+ "bold": "ठळक",
+ "italic": "तिरकस",
+ "underline": "अधोरेखित",
+ "strike": "मागे ओढलेले",
+ "numList": "क्रमांकित यादी",
+ "bulletList": "बुलेट यादी",
+ "checkList": "चेक यादी",
+ "inlineCode": "इनलाइन कोड",
+ "quote": "उद्धरण ब्लॉक",
+ "header": "शीर्षक",
+ "highlight": "हायलाइट",
+ "color": "रंग",
+ "addLink": "लिंक जोडा"
+ },
+ "tooltip": {
+ "lightMode": "लाइट मोडमध्ये स्विच करा",
+ "darkMode": "डार्क मोडमध्ये स्विच करा",
+ "openAsPage": "पृष्ठ म्हणून उघडा",
+ "addNewRow": "नवीन पंक्ती जोडा",
+ "openMenu": "मेनू उघडण्यासाठी क्लिक करा",
+ "dragRow": "पंक्तीचे स्थान बदलण्यासाठी ड्रॅग करा",
+ "viewDataBase": "डेटाबेस पहा",
+ "referencePage": "हे {name} संदर्भित आहे",
+ "addBlockBelow": "खाली एक ब्लॉक जोडा",
+ "aiGenerate": "निर्मिती करा"
+ },
+ "sideBar": {
+ "closeSidebar": "साइडबार बंद करा",
+ "openSidebar": "साइडबार उघडा",
+ "expandSidebar": "पूर्ण पृष्ठावर विस्तारित करा",
+ "personal": "वैयक्तिक",
+ "private": "खाजगी",
+ "workspace": "कार्यक्षेत्र",
+ "favorites": "आवडती",
+ "clickToHidePrivate": "खाजगी जागा लपवण्यासाठी क्लिक करा\nयेथे तयार केलेली पाने फक्त तुम्हाला दिसतील",
+ "clickToHideWorkspace": "कार्यक्षेत्र लपवण्यासाठी क्लिक करा\nयेथे तयार केलेली पाने सर्व सदस्यांना दिसतील",
+ "clickToHidePersonal": "वैयक्तिक जागा लपवण्यासाठी क्लिक करा",
+ "clickToHideFavorites": "आवडती जागा लपवण्यासाठी क्लिक करा",
+ "addAPage": "नवीन पृष्ठ जोडा",
+ "addAPageToPrivate": "खाजगी जागेत पृष्ठ जोडा",
+ "addAPageToWorkspace": "कार्यक्षेत्रात पृष्ठ जोडा",
+ "recent": "अलीकडील",
+ "today": "आज",
+ "thisWeek": "या आठवड्यात",
+ "others": "पूर्वीच्या आवडती",
+ "earlier": "पूर्वीचे",
+ "justNow": "आत्ताच",
+ "minutesAgo": "{count} मिनिटांपूर्वी",
+ "lastViewed": "शेवटी पाहिलेले",
+ "favoriteAt": "आवडते म्हणून चिन्हांकित",
+ "emptyRecent": "अलीकडील पृष्ठे नाहीत",
+ "emptyRecentDescription": "तुम्ही पाहिलेली पृष्ठे येथे सहज पुन्हा मिळवण्यासाठी दिसतील.",
+ "emptyFavorite": "आवडती पृष्ठे नाहीत",
+ "emptyFavoriteDescription": "पृष्ठांना आवडते म्हणून चिन्हांकित करा—ते येथे झपाट्याने प्रवेशासाठी दिसतील!",
+ "removePageFromRecent": "हे पृष्ठ अलीकडील यादीतून काढायचे?",
+ "removeSuccess": "यशस्वीरित्या काढले गेले",
+ "favoriteSpace": "आवडती",
+ "RecentSpace": "अलीकडील",
+ "Spaces": "जागा",
+ "upgradeToPro": "Pro मध्ये अपग्रेड करा",
+ "upgradeToAIMax": "अमर्यादित AI अनलॉक करा",
+ "storageLimitDialogTitle": "तुमचा मोफत स्टोरेज संपला आहे. अमर्यादित स्टोरेजसाठी अपग्रेड करा",
+ "storageLimitDialogTitleIOS": "तुमचा मोफत स्टोरेज संपला आहे.",
+ "aiResponseLimitTitle": "तुमचे मोफत AI प्रतिसाद संपले आहेत. कृपया Pro प्लॅनला अपग्रेड करा किंवा AI add-on खरेदी करा",
+ "aiResponseLimitDialogTitle": "AI प्रतिसाद मर्यादा गाठली आहे",
+ "aiResponseLimit": "तुमचे मोफत AI प्रतिसाद संपले आहेत.\n\nसेटिंग्ज -> प्लॅन -> AI Max किंवा Pro प्लॅन क्लिक करा",
+ "askOwnerToUpgradeToPro": "तुमच्या कार्यक्षेत्राचे मोफत स्टोरेज संपत चालले आहे. कृपया कार्यक्षेत्र मालकाला Pro प्लॅनमध्ये अपग्रेड करण्यास सांगा",
+ "askOwnerToUpgradeToProIOS": "तुमच्या कार्यक्षेत्राचे मोफत स्टोरेज संपत चालले आहे.",
+ "askOwnerToUpgradeToAIMax": "तुमच्या कार्यक्षेत्राचे मोफत AI प्रतिसाद संपले आहेत. कृपया कार्यक्षेत्र मालकाला प्लॅन अपग्रेड किंवा AI add-ons खरेदी करण्यास सांगा",
+ "askOwnerToUpgradeToAIMaxIOS": "तुमच्या कार्यक्षेत्राचे मोफत AI प्रतिसाद संपले आहेत.",
+ "purchaseAIMax": "तुमच्या कार्यक्षेत्राचे AI प्रतिमा प्रतिसाद संपले आहेत. कृपया कार्यक्षेत्र मालकाला AI Max खरेदी करण्यास सांगा",
+ "aiImageResponseLimit": "तुमचे AI प्रतिमा प्रतिसाद संपले आहेत.\n\nसेटिंग्ज -> प्लॅन -> AI Max क्लिक करा",
+ "purchaseStorageSpace": "स्टोरेज स्पेस खरेदी करा",
+ "singleFileProPlanLimitationDescription": "तुम्ही मोफत प्लॅनमध्ये परवानगी दिलेल्या फाइल अपलोड आकाराची मर्यादा ओलांडली आहे. कृपया Pro प्लॅनमध्ये अपग्रेड करा",
+ "purchaseAIResponse": "AI प्रतिसाद खरेदी करा",
+ "askOwnerToUpgradeToLocalAI": "कार्यक्षेत्र मालकाला ऑन-डिव्हाइस AI सक्षम करण्यास सांगा",
+ "upgradeToAILocal": "अत्याधिक गोपनीयतेसाठी तुमच्या डिव्हाइसवर लोकल मॉडेल चालवा",
+ "upgradeToAILocalDesc": "PDFs सोबत गप्पा मारा, तुमचे लेखन सुधारा आणि लोकल AI वापरून टेबल्स आपोआप भरा"
+},
+ "notifications": {
+ "export": {
+ "markdown": "टीप Markdown मध्ये निर्यात केली",
+ "path": "Documents/flowy"
+ }
+ },
+ "contactsPage": {
+ "title": "संपर्क",
+ "whatsHappening": "या आठवड्यात काय घडत आहे?",
+ "addContact": "संपर्क जोडा",
+ "editContact": "संपर्क संपादित करा"
+ },
+ "button": {
+ "ok": "ठीक आहे",
+ "confirm": "खात्री करा",
+ "done": "पूर्ण",
+ "cancel": "रद्द करा",
+ "signIn": "साइन इन",
+ "signOut": "साइन आउट",
+ "complete": "पूर्ण करा",
+ "save": "जतन करा",
+ "generate": "निर्माण करा",
+ "esc": "ESC",
+ "keep": "ठेवा",
+ "tryAgain": "पुन्हा प्रयत्न करा",
+ "discard": "टाका",
+ "replace": "बदला",
+ "insertBelow": "खाली घाला",
+ "insertAbove": "वर घाला",
+ "upload": "अपलोड करा",
+ "edit": "संपादित करा",
+ "delete": "हटवा",
+ "copy": "कॉपी करा",
+ "duplicate": "प्रत बनवा",
+ "putback": "परत ठेवा",
+ "update": "अद्यतनित करा",
+ "share": "शेअर करा",
+ "removeFromFavorites": "आवडतीतून काढा",
+ "removeFromRecent": "अलीकडील यादीतून काढा",
+ "addToFavorites": "आवडतीत जोडा",
+ "favoriteSuccessfully": "आवडतीत यशस्वीरित्या जोडले",
+ "unfavoriteSuccessfully": "आवडतीतून यशस्वीरित्या काढले",
+ "duplicateSuccessfully": "प्रत यशस्वीरित्या तयार झाली",
+ "rename": "नाव बदला",
+ "helpCenter": "मदत केंद्र",
+ "add": "जोड़ा",
+ "yes": "होय",
+ "no": "नाही",
+ "clear": "साफ करा",
+ "remove": "काढा",
+ "dontRemove": "काढू नका",
+ "copyLink": "लिंक कॉपी करा",
+ "align": "जुळवा",
+ "login": "लॉगिन",
+ "logout": "लॉगआउट",
+ "deleteAccount": "खाते हटवा",
+ "back": "मागे",
+ "signInGoogle": "Google सह पुढे जा",
+ "signInGithub": "GitHub सह पुढे जा",
+ "signInDiscord": "Discord सह पुढे जा",
+ "more": "अधिक",
+ "create": "तयार करा",
+ "close": "बंद करा",
+ "next": "पुढे",
+ "previous": "मागील",
+ "submit": "सबमिट करा",
+ "download": "डाउनलोड करा",
+ "backToHome": "मुख्यपृष्ठावर परत जा",
+ "viewing": "पाहत आहात",
+ "editing": "संपादन करत आहात",
+ "gotIt": "समजले",
+ "retry": "पुन्हा प्रयत्न करा",
+ "uploadFailed": "अपलोड अयशस्वी.",
+ "copyLinkOriginal": "मूळ दुव्याची कॉपी करा"
+ },
+ "label": {
+ "welcome": "स्वागत आहे!",
+ "firstName": "पहिले नाव",
+ "middleName": "मधले नाव",
+ "lastName": "आडनाव",
+ "stepX": "पायरी {X}"
+ },
+ "oAuth": {
+ "err": {
+ "failedTitle": "तुमच्या खात्याशी कनेक्ट होता आले नाही.",
+ "failedMsg": "कृपया खात्री करा की तुम्ही ब्राउझरमध्ये साइन-इन प्रक्रिया पूर्ण केली आहे."
+ },
+ "google": {
+ "title": "GOOGLE साइन-इन",
+ "instruction1": "तुमचे Google Contacts आयात करण्यासाठी, तुम्हाला तुमच्या वेब ब्राउझरचा वापर करून या अॅप्लिकेशनला अधिकृत करणे आवश्यक आहे.",
+ "instruction2": "ही कोड आयकॉनवर क्लिक करून किंवा मजकूर निवडून क्लिपबोर्डवर कॉपी करा:",
+ "instruction3": "तुमच्या वेब ब्राउझरमध्ये खालील दुव्यावर जा आणि वरील कोड टाका:",
+ "instruction4": "साइनअप पूर्ण झाल्यावर खालील बटण क्लिक करा:"
+ }
+ },
+ "settings": {
+ "title": "सेटिंग्ज",
+ "popupMenuItem": {
+ "settings": "सेटिंग्ज",
+ "members": "सदस्य",
+ "trash": "कचरा",
+ "helpAndSupport": "मदत आणि समर्थन"
+ },
+ "sites": {
+ "title": "साइट्स",
+ "namespaceTitle": "नेमस्पेस",
+ "namespaceDescription": "तुमचा नेमस्पेस आणि मुख्यपृष्ठ व्यवस्थापित करा",
+ "namespaceHeader": "नेमस्पेस",
+ "homepageHeader": "मुख्यपृष्ठ",
+ "updateNamespace": "नेमस्पेस अद्यतनित करा",
+ "removeHomepage": "मुख्यपृष्ठ हटवा",
+ "selectHomePage": "एक पृष्ठ निवडा",
+ "clearHomePage": "या नेमस्पेससाठी मुख्यपृष्ठ साफ करा",
+ "customUrl": "स्वतःची URL",
+ "namespace": {
+ "description": "हे बदल सर्व प्रकाशित पृष्ठांवर लागू होतील जे या नेमस्पेसवर चालू आहेत",
+ "tooltip": "कोणताही अनुचित नेमस्पेस आम्ही काढून टाकण्याचा अधिकार राखून ठेवतो",
+ "updateExistingNamespace": "विद्यमान नेमस्पेस अद्यतनित करा",
+ "upgradeToPro": "मुख्यपृष्ठ सेट करण्यासाठी Pro प्लॅनमध्ये अपग्रेड करा",
+ "redirectToPayment": "पेमेंट पृष्ठावर वळवत आहे...",
+ "onlyWorkspaceOwnerCanSetHomePage": "फक्त कार्यक्षेत्र मालकच मुख्यपृष्ठ सेट करू शकतो",
+ "pleaseAskOwnerToSetHomePage": "कृपया कार्यक्षेत्र मालकाला Pro प्लॅनमध्ये अपग्रेड करण्यास सांगा"
+ },
+ "publishedPage": {
+ "title": "सर्व प्रकाशित पृष्ठे",
+ "description": "तुमची प्रकाशित पृष्ठे व्यवस्थापित करा",
+ "page": "पृष्ठ",
+ "pathName": "पथाचे नाव",
+ "date": "प्रकाशन तारीख",
+ "emptyHinText": "या कार्यक्षेत्रात तुमच्याकडे कोणतीही प्रकाशित पृष्ठे नाहीत",
+ "noPublishedPages": "प्रकाशित पृष्ठे नाहीत",
+ "settings": "प्रकाशन सेटिंग्ज",
+ "clickToOpenPageInApp": "पृष्ठ अॅपमध्ये उघडा",
+ "clickToOpenPageInBrowser": "पृष्ठ ब्राउझरमध्ये उघडा"
+ }
+ }
+ },
+ "error": {
+ "failedToGeneratePaymentLink": "Pro प्लॅनसाठी पेमेंट लिंक तयार करण्यात अयशस्वी",
+ "failedToUpdateNamespace": "नेमस्पेस अद्यतनित करण्यात अयशस्वी",
+ "proPlanLimitation": "नेमस्पेस अद्यतनित करण्यासाठी तुम्हाला Pro प्लॅनमध्ये अपग्रेड करणे आवश्यक आहे",
+ "namespaceAlreadyInUse": "नेमस्पेस आधीच वापरात आहे, कृपया दुसरे प्रयत्न करा",
+ "invalidNamespace": "अवैध नेमस्पेस, कृपया दुसरे प्रयत्न करा",
+ "namespaceLengthAtLeast2Characters": "नेमस्पेस किमान २ अक्षरे लांब असावे",
+ "onlyWorkspaceOwnerCanUpdateNamespace": "फक्त कार्यक्षेत्र मालकच नेमस्पेस अद्यतनित करू शकतो",
+ "onlyWorkspaceOwnerCanRemoveHomepage": "फक्त कार्यक्षेत्र मालकच मुख्यपृष्ठ हटवू शकतो",
+ "setHomepageFailed": "मुख्यपृष्ठ सेट करण्यात अयशस्वी",
+ "namespaceTooLong": "नेमस्पेस खूप लांब आहे, कृपया दुसरे प्रयत्न करा",
+ "namespaceTooShort": "नेमस्पेस खूप लहान आहे, कृपया दुसरे प्रयत्न करा",
+ "namespaceIsReserved": "हा नेमस्पेस राखीव आहे, कृपया दुसरे प्रयत्न करा",
+ "updatePathNameFailed": "पथाचे नाव अद्यतनित करण्यात अयशस्वी",
+ "removeHomePageFailed": "मुख्यपृष्ठ हटवण्यात अयशस्वी",
+ "publishNameContainsInvalidCharacters": "पथाच्या नावामध्ये अवैध अक्षरे आहेत, कृपया दुसरे प्रयत्न करा",
+ "publishNameTooShort": "पथाचे नाव खूप लहान आहे, कृपया दुसरे प्रयत्न करा",
+ "publishNameTooLong": "पथाचे नाव खूप लांब आहे, कृपया दुसरे प्रयत्न करा",
+ "publishNameAlreadyInUse": "हे पथाचे नाव आधीच वापरले गेले आहे, कृपया दुसरे प्रयत्न करा",
+ "namespaceContainsInvalidCharacters": "नेमस्पेसमध्ये अवैध अक्षरे आहेत, कृपया दुसरे प्रयत्न करा",
+ "publishPermissionDenied": "फक्त कार्यक्षेत्र मालक किंवा पृष्ठ प्रकाशकच प्रकाशन सेटिंग्ज व्यवस्थापित करू शकतो",
+ "publishNameCannotBeEmpty": "पथाचे नाव रिकामे असू शकत नाही, कृपया दुसरे प्रयत्न करा"
+ },
+ "success": {
+ "namespaceUpdated": "नेमस्पेस यशस्वीरित्या अद्यतनित केला",
+ "setHomepageSuccess": "मुख्यपृष्ठ यशस्वीरित्या सेट केले",
+ "updatePathNameSuccess": "पथाचे नाव यशस्वीरित्या अद्यतनित केले",
+ "removeHomePageSuccess": "मुख्यपृष्ठ यशस्वीरित्या हटवले"
+ },
+ "accountPage": {
+ "menuLabel": "खाते आणि अॅप",
+ "title": "माझे खाते",
+ "general": {
+ "title": "खात्याचे नाव आणि प्रोफाइल प्रतिमा",
+ "changeProfilePicture": "प्रोफाइल प्रतिमा बदला"
+ },
+ "email": {
+ "title": "ईमेल",
+ "actions": {
+ "change": "ईमेल बदला"
+ }
+ },
+ "login": {
+ "title": "खाते लॉगिन",
+ "loginLabel": "लॉगिन",
+ "logoutLabel": "लॉगआउट"
+ },
+ "isUpToDate": "@:appName अद्ययावत आहे!",
+ "officialVersion": "आवृत्ती {version} (अधिकृत बिल्ड)"
+},
+ "workspacePage": {
+ "menuLabel": "कार्यक्षेत्र",
+ "title": "कार्यक्षेत्र",
+ "description": "तुमचे कार्यक्षेत्र स्वरूप, थीम, फॉन्ट, मजकूर रचना, दिनांक/वेळ फॉरमॅट आणि भाषा सानुकूलित करा.",
+ "workspaceName": {
+ "title": "कार्यक्षेत्राचे नाव"
+ },
+ "workspaceIcon": {
+ "title": "कार्यक्षेत्राचे चिन्ह",
+ "description": "तुमच्या कार्यक्षेत्रासाठी प्रतिमा अपलोड करा किंवा इमोजी वापरा. हे चिन्ह साइडबार आणि सूचना मध्ये दिसेल."
+ },
+ "appearance": {
+ "title": "दृश्यरूप",
+ "description": "कार्यक्षेत्राचे दृश्यरूप, थीम, फॉन्ट, मजकूर रचना, दिनांक, वेळ आणि भाषा सानुकूलित करा.",
+ "options": {
+ "system": "स्वयंचलित",
+ "light": "लाइट",
+ "dark": "डार्क"
+ }
+ }
+ },
+ "resetCursorColor": {
+ "title": "दस्तऐवज कर्सरचा रंग रीसेट करा",
+ "description": "तुम्हाला कर्सरचा रंग रीसेट करायचा आहे का?"
+ },
+ "resetSelectionColor": {
+ "title": "दस्तऐवज निवडीचा रंग रीसेट करा",
+ "description": "तुम्हाला निवडीचा रंग रीसेट करायचा आहे का?"
+ },
+ "resetWidth": {
+ "resetSuccess": "दस्तऐवजाची रुंदी यशस्वीरित्या रीसेट केली"
+ },
+ "theme": {
+ "title": "थीम",
+ "description": "पूर्व-निर्धारित थीम निवडा किंवा तुमची स्वतःची थीम अपलोड करा.",
+ "uploadCustomThemeTooltip": "स्वतःची थीम अपलोड करा"
+ },
+ "workspaceFont": {
+ "title": "कार्यक्षेत्र फॉन्ट",
+ "noFontHint": "कोणताही फॉन्ट सापडला नाही, कृपया दुसरा शब्द वापरून पहा."
+ },
+ "textDirection": {
+ "title": "मजकूर दिशा",
+ "leftToRight": "डावीकडून उजवीकडे",
+ "rightToLeft": "उजवीकडून डावीकडे",
+ "auto": "स्वयंचलित",
+ "enableRTLItems": "RTL टूलबार घटक सक्षम करा"
+ },
+ "layoutDirection": {
+ "title": "लेआउट दिशा",
+ "leftToRight": "डावीकडून उजवीकडे",
+ "rightToLeft": "उजवीकडून डावीकडे"
+ },
+ "dateTime": {
+ "title": "दिनांक आणि वेळ",
+ "example": "{} वाजता {} ({})",
+ "24HourTime": "२४-तास वेळ",
+ "dateFormat": {
+ "label": "दिनांक फॉरमॅट",
+ "local": "स्थानिक",
+ "us": "US",
+ "iso": "ISO",
+ "friendly": "सुलभ",
+ "dmy": "D/M/Y"
+ }
+ },
+ "language": {
+ "title": "भाषा"
+ },
+ "deleteWorkspacePrompt": {
+ "title": "कार्यक्षेत्र हटवा",
+ "content": "तुम्हाला हे कार्यक्षेत्र हटवायचे आहे का? ही कृती पूर्ववत केली जाऊ शकत नाही, आणि तुम्ही प्रकाशित केलेली कोणतीही पृष्ठे अप्रकाशित होतील."
+ },
+ "leaveWorkspacePrompt": {
+ "title": "कार्यक्षेत्र सोडा",
+ "content": "तुम्हाला हे कार्यक्षेत्र सोडायचे आहे का? यानंतर तुम्हाला यामधील सर्व पृष्ठे आणि डेटावर प्रवेश मिळणार नाही.",
+ "success": "तुम्ही कार्यक्षेत्र यशस्वीरित्या सोडले.",
+ "fail": "कार्यक्षेत्र सोडण्यात अयशस्वी."
+ },
+ "manageWorkspace": {
+ "title": "कार्यक्षेत्र व्यवस्थापित करा",
+ "leaveWorkspace": "कार्यक्षेत्र सोडा",
+ "deleteWorkspace": "कार्यक्षेत्र हटवा"
+ },
+ "manageDataPage": {
+ "menuLabel": "डेटा व्यवस्थापित करा",
+ "title": "डेटा व्यवस्थापन",
+ "description": "@:appName मध्ये स्थानिक डेटा संचयन व्यवस्थापित करा किंवा तुमचा विद्यमान डेटा आयात करा.",
+ "dataStorage": {
+ "title": "फाइल संचयन स्थान",
+ "tooltip": "जिथे तुमच्या फाइल्स संग्रहित आहेत ते स्थान",
+ "actions": {
+ "change": "मार्ग बदला",
+ "open": "फोल्डर उघडा",
+ "openTooltip": "सध्याच्या डेटा फोल्डरचे स्थान उघडा",
+ "copy": "मार्ग कॉपी करा",
+ "copiedHint": "मार्ग कॉपी केला!",
+ "resetTooltip": "मूलभूत स्थानावर रीसेट करा"
+ },
+ "resetDialog": {
+ "title": "तुम्हाला खात्री आहे का?",
+ "description": "पथ मूलभूत डेटा स्थानावर रीसेट केल्याने तुमचा डेटा हटवला जाणार नाही. तुम्हाला सध्याचा डेटा पुन्हा आयात करायचा असल्यास, कृपया आधी त्याचा पथ कॉपी करा."
+ }
+ },
+ "importData": {
+ "title": "डेटा आयात करा",
+ "tooltip": "@:appName बॅकअप/डेटा फोल्डरमधून डेटा आयात करा",
+ "description": "बाह्य @:appName डेटा फोल्डरमधून डेटा कॉपी करा",
+ "action": "फाइल निवडा"
+ },
+ "encryption": {
+ "title": "एनक्रिप्शन",
+ "tooltip": "तुमचा डेटा कसा संग्रहित आणि एनक्रिप्ट केला जातो ते व्यवस्थापित करा",
+ "descriptionNoEncryption": "एनक्रिप्शन चालू केल्याने सर्व डेटा एनक्रिप्ट केला जाईल. ही क्रिया पूर्ववत केली जाऊ शकत नाही.",
+ "descriptionEncrypted": "तुमचा डेटा एनक्रिप्टेड आहे.",
+ "action": "डेटा एनक्रिप्ट करा",
+ "dialog": {
+ "title": "संपूर्ण डेटा एनक्रिप्ट करायचा?",
+ "description": "तुमचा संपूर्ण डेटा एनक्रिप्ट केल्याने तो सुरक्षित राहील. ही क्रिया पूर्ववत केली जाऊ शकत नाही. तुम्हाला पुढे जायचे आहे का?"
+ }
+ },
+ "cache": {
+ "title": "कॅशे साफ करा",
+ "description": "प्रतिमा न दिसणे, पृष्ठे हरवणे, फॉन्ट लोड न होणे अशा समस्यांचे निराकरण करण्यासाठी मदत करा. याचा तुमच्या डेटावर परिणाम होणार नाही.",
+ "dialog": {
+ "title": "कॅशे साफ करा",
+ "description": "प्रतिमा न दिसणे, पृष्ठे हरवणे, फॉन्ट लोड न होणे अशा समस्यांचे निराकरण करण्यासाठी मदत करा. याचा तुमच्या डेटावर परिणाम होणार नाही.",
+ "successHint": "कॅशे साफ झाली!"
+ }
+ },
+ "data": {
+ "fixYourData": "तुमचा डेटा सुधारा",
+ "fixButton": "सुधारा",
+ "fixYourDataDescription": "तुमच्या डेटामध्ये काही अडचण येत असल्यास, तुम्ही येथे ती सुधारू शकता."
+ }
+ },
+ "shortcutsPage": {
+ "menuLabel": "शॉर्टकट्स",
+ "title": "शॉर्टकट्स",
+ "editBindingHint": "नवीन बाइंडिंग टाका",
+ "searchHint": "शोधा",
+ "actions": {
+ "resetDefault": "मूलभूत रीसेट करा"
+ },
+ "errorPage": {
+ "message": "शॉर्टकट्स लोड करण्यात अयशस्वी: {}",
+ "howToFix": "कृपया पुन्हा प्रयत्न करा. समस्या कायम राहिल्यास GitHub वर संपर्क साधा."
+ },
+ "resetDialog": {
+ "title": "शॉर्टकट्स रीसेट करा",
+ "description": "हे सर्व कीबाइंडिंग्जना मूळ स्थितीत रीसेट करेल. ही क्रिया पूर्ववत करता येणार नाही. तुम्हाला खात्री आहे का?",
+ "buttonLabel": "रीसेट करा"
+ },
+ "conflictDialog": {
+ "title": "{} आधीच वापरले जात आहे",
+ "descriptionPrefix": "हे कीबाइंडिंग सध्या ",
+ "descriptionSuffix": " यामध्ये वापरले जात आहे. तुम्ही हे कीबाइंडिंग बदलल्यास, ते {} मधून काढले जाईल.",
+ "confirmLabel": "पुढे जा"
+ },
+ "editTooltip": "कीबाइंडिंग संपादित करण्यासाठी क्लिक करा",
+ "keybindings": {
+ "toggleToDoList": "टू-डू सूची चालू/बंद करा",
+ "insertNewParagraphInCodeblock": "नवीन परिच्छेद टाका",
+ "pasteInCodeblock": "कोडब्लॉकमध्ये पेस्ट करा",
+ "selectAllCodeblock": "सर्व निवडा",
+ "indentLineCodeblock": "ओळीच्या सुरुवातीला दोन स्पेस टाका",
+ "outdentLineCodeblock": "ओळीच्या सुरुवातीची दोन स्पेस काढा",
+ "twoSpacesCursorCodeblock": "कर्सरवर दोन स्पेस टाका",
+ "copy": "निवड कॉपी करा",
+ "paste": "मजकुरात पेस्ट करा",
+ "cut": "निवड कट करा",
+ "alignLeft": "मजकूर डावीकडे संरेखित करा",
+ "alignCenter": "मजकूर मधोमध संरेखित करा",
+ "alignRight": "मजकूर उजवीकडे संरेखित करा",
+ "insertInlineMathEquation": "इनलाइन गणितीय सूत्र टाका",
+ "undo": "पूर्ववत करा",
+ "redo": "पुन्हा करा",
+ "convertToParagraph": "ब्लॉक परिच्छेदात रूपांतरित करा",
+ "backspace": "हटवा",
+ "deleteLeftWord": "डावीकडील शब्द हटवा",
+ "deleteLeftSentence": "डावीकडील वाक्य हटवा",
+ "delete": "उजवीकडील अक्षर हटवा",
+ "deleteMacOS": "डावीकडील अक्षर हटवा",
+ "deleteRightWord": "उजवीकडील शब्द हटवा",
+ "moveCursorLeft": "कर्सर डावीकडे हलवा",
+ "moveCursorBeginning": "कर्सर सुरुवातीला हलवा",
+ "moveCursorLeftWord": "कर्सर एक शब्द डावीकडे हलवा",
+ "moveCursorLeftSelect": "निवडा आणि कर्सर डावीकडे हलवा",
+ "moveCursorBeginSelect": "निवडा आणि कर्सर सुरुवातीला हलवा",
+ "moveCursorLeftWordSelect": "निवडा आणि कर्सर एक शब्द डावीकडे हलवा",
+ "moveCursorRight": "कर्सर उजवीकडे हलवा",
+ "moveCursorEnd": "कर्सर शेवटी हलवा",
+ "moveCursorRightWord": "कर्सर एक शब्द उजवीकडे हलवा",
+ "moveCursorRightSelect": "निवडा आणि कर्सर उजवीकडे हलवा",
+ "moveCursorEndSelect": "निवडा आणि कर्सर शेवटी हलवा",
+ "moveCursorRightWordSelect": "निवडा आणि कर्सर एक शब्द उजवीकडे हलवा",
+ "moveCursorUp": "कर्सर वर हलवा",
+ "moveCursorTopSelect": "निवडा आणि कर्सर वर हलवा",
+ "moveCursorTop": "कर्सर वर हलवा",
+ "moveCursorUpSelect": "निवडा आणि कर्सर वर हलवा",
+ "moveCursorBottomSelect": "निवडा आणि कर्सर खाली हलवा",
+ "moveCursorBottom": "कर्सर खाली हलवा",
+ "moveCursorDown": "कर्सर खाली हलवा",
+ "moveCursorDownSelect": "निवडा आणि कर्सर खाली हलवा",
+ "home": "वर स्क्रोल करा",
+ "end": "खाली स्क्रोल करा",
+ "toggleBold": "बोल्ड चालू/बंद करा",
+ "toggleItalic": "इटालिक चालू/बंद करा",
+ "toggleUnderline": "अधोरेखित चालू/बंद करा",
+ "toggleStrikethrough": "स्ट्राईकथ्रू चालू/बंद करा",
+ "toggleCode": "इनलाइन कोड चालू/बंद करा",
+ "toggleHighlight": "हायलाईट चालू/बंद करा",
+ "showLinkMenu": "लिंक मेनू दाखवा",
+ "openInlineLink": "इनलाइन लिंक उघडा",
+ "openLinks": "सर्व निवडलेले लिंक उघडा",
+ "indent": "इंडेंट",
+ "outdent": "आउटडेंट",
+ "exit": "संपादनातून बाहेर पडा",
+ "pageUp": "एक पृष्ठ वर स्क्रोल करा",
+ "pageDown": "एक पृष्ठ खाली स्क्रोल करा",
+ "selectAll": "सर्व निवडा",
+ "pasteWithoutFormatting": "फॉरमॅटिंगशिवाय पेस्ट करा",
+ "showEmojiPicker": "इमोजी निवडकर्ता दाखवा",
+ "enterInTableCell": "टेबलमध्ये नवीन ओळ जोडा",
+ "leftInTableCell": "टेबलमध्ये डावीकडील सेलमध्ये जा",
+ "rightInTableCell": "टेबलमध्ये उजवीकडील सेलमध्ये जा",
+ "upInTableCell": "टेबलमध्ये वरील सेलमध्ये जा",
+ "downInTableCell": "टेबलमध्ये खालील सेलमध्ये जा",
+ "tabInTableCell": "टेबलमध्ये पुढील सेलमध्ये जा",
+ "shiftTabInTableCell": "टेबलमध्ये मागील सेलमध्ये जा",
+ "backSpaceInTableCell": "सेलच्या सुरुवातीला थांबा"
+ },
+ "commands": {
+ "codeBlockNewParagraph": "कोडब्लॉकच्या शेजारी नवीन परिच्छेद टाका",
+ "codeBlockIndentLines": "कोडब्लॉकमध्ये ओळीच्या सुरुवातीला दोन स्पेस टाका",
+ "codeBlockOutdentLines": "कोडब्लॉकमध्ये ओळीच्या सुरुवातीची दोन स्पेस काढा",
+ "codeBlockAddTwoSpaces": "कोडब्लॉकमध्ये कर्सरच्या ठिकाणी दोन स्पेस टाका",
+ "codeBlockSelectAll": "कोडब्लॉकमधील सर्व मजकूर निवडा",
+ "codeBlockPasteText": "कोडब्लॉकमध्ये मजकूर पेस्ट करा",
+ "textAlignLeft": "मजकूर डावीकडे संरेखित करा",
+ "textAlignCenter": "मजकूर मधोमध संरेखित करा",
+ "textAlignRight": "मजकूर उजवीकडे संरेखित करा"
+ },
+ "couldNotLoadErrorMsg": "शॉर्टकट लोड करता आले नाहीत. पुन्हा प्रयत्न करा",
+ "couldNotSaveErrorMsg": "शॉर्टकट सेव्ह करता आले नाहीत. पुन्हा प्रयत्न करा"
+},
+ "aiPage": {
+ "title": "AI सेटिंग्ज",
+ "menuLabel": "AI सेटिंग्ज",
+ "keys": {
+ "enableAISearchTitle": "AI शोध",
+ "aiSettingsDescription": "AppFlowy AI चालवण्यासाठी तुमचा पसंतीचा मॉडेल निवडा. आता GPT-4o, GPT-o3-mini, DeepSeek R1, Claude 3.5 Sonnet आणि Ollama मधील मॉडेल्सचा समावेश आहे.",
+ "loginToEnableAIFeature": "AI वैशिष्ट्ये फक्त @:appName Cloud मध्ये लॉगिन केल्यानंतरच सक्षम होतात. तुमच्याकडे @:appName खाते नसेल तर 'माय अकाउंट' मध्ये जाऊन साइन अप करा.",
+ "llmModel": "भाषा मॉडेल",
+ "llmModelType": "भाषा मॉडेल प्रकार",
+ "downloadLLMPrompt": "{} डाउनलोड करा",
+ "downloadAppFlowyOfflineAI": "AI ऑफलाइन पॅकेज डाउनलोड केल्याने तुमच्या डिव्हाइसवर AI चालवता येईल. तुम्हाला पुढे जायचे आहे का?",
+ "downloadLLMPromptDetail": "{} स्थानिक मॉडेल डाउनलोड करण्यासाठी सुमारे {} संचयन लागेल. तुम्हाला पुढे जायचे आहे का?",
+ "downloadBigFilePrompt": "डाउनलोड पूर्ण होण्यासाठी सुमारे १० मिनिटे लागू शकतात",
+ "downloadAIModelButton": "डाउनलोड करा",
+ "downloadingModel": "डाउनलोड करत आहे",
+ "localAILoaded": "स्थानिक AI मॉडेल यशस्वीरित्या जोडले गेले आणि वापरण्यास सज्ज आहे",
+ "localAIStart": "स्थानिक AI सुरू होत आहे. जर हळू वाटत असेल, तर ते बंद करून पुन्हा सुरू करून पहा",
+ "localAILoading": "स्थानिक AI Chat मॉडेल लोड होत आहे...",
+ "localAIStopped": "स्थानिक AI थांबले आहे",
+ "localAIRunning": "स्थानिक AI चालू आहे",
+ "localAINotReadyRetryLater": "स्थानिक AI सुरू होत आहे, कृपया नंतर पुन्हा प्रयत्न करा",
+ "localAIDisabled": "तुम्ही स्थानिक AI वापरत आहात, पण ते अकार्यक्षम आहे. कृपया सेटिंग्जमध्ये जाऊन ते सक्षम करा किंवा दुसरे मॉडेल वापरून पहा",
+ "localAIInitializing": "स्थानिक AI लोड होत आहे. तुमच्या डिव्हाइसवर अवलंबून याला काही मिनिटे लागू शकतात",
+ "localAINotReadyTextFieldPrompt": "स्थानिक AI लोड होत असताना संपादन करता येणार नाही",
+ "failToLoadLocalAI": "स्थानिक AI सुरू करण्यात अयशस्वी.",
+ "restartLocalAI": "पुन्हा सुरू करा",
+ "disableLocalAITitle": "स्थानिक AI अकार्यक्षम करा",
+ "disableLocalAIDescription": "तुम्हाला स्थानिक AI अकार्यक्षम करायचा आहे का?",
+ "localAIToggleTitle": "AppFlowy स्थानिक AI (LAI)",
+ "localAIToggleSubTitle": "AppFlowy मध्ये अत्याधुनिक स्थानिक AI मॉडेल्स वापरून गोपनीयता आणि सुरक्षा सुनिश्चित करा",
+ "offlineAIInstruction1": "हे अनुसरा",
+ "offlineAIInstruction2": "सूचना",
+ "offlineAIInstruction3": "ऑफलाइन AI सक्षम करण्यासाठी.",
+ "offlineAIDownload1": "जर तुम्ही AppFlowy AI डाउनलोड केले नसेल, तर कृपया",
+ "offlineAIDownload2": "डाउनलोड",
+ "offlineAIDownload3": "करा",
+ "activeOfflineAI": "सक्रिय",
+ "downloadOfflineAI": "डाउनलोड करा",
+ "openModelDirectory": "फोल्डर उघडा",
+ "laiNotReady": "स्थानिक AI अॅप योग्य प्रकारे इन्स्टॉल झालेले नाही.",
+ "ollamaNotReady": "Ollama सर्व्हर तयार नाही.",
+ "pleaseFollowThese": "कृपया हे अनुसरा",
+ "instructions": "सूचना",
+ "installOllamaLai": "Ollama आणि AppFlowy स्थानिक AI सेटअप करण्यासाठी.",
+ "modelsMissing": "आवश्यक मॉडेल्स सापडत नाहीत.",
+ "downloadModel": "त्यांना डाउनलोड करण्यासाठी."
+ }
+},
+ "planPage": {
+ "menuLabel": "योजना",
+ "title": "दर योजना",
+ "planUsage": {
+ "title": "योजनेचा वापर सारांश",
+ "storageLabel": "स्टोरेज",
+ "storageUsage": "{} पैकी {} GB",
+ "unlimitedStorageLabel": "अमर्यादित स्टोरेज",
+ "collaboratorsLabel": "सदस्य",
+ "collaboratorsUsage": "{} पैकी {}",
+ "aiResponseLabel": "AI प्रतिसाद",
+ "aiResponseUsage": "{} पैकी {}",
+ "unlimitedAILabel": "अमर्यादित AI प्रतिसाद",
+ "proBadge": "प्रो",
+ "aiMaxBadge": "AI Max",
+ "aiOnDeviceBadge": "मॅकसाठी ऑन-डिव्हाइस AI",
+ "memberProToggle": "अधिक सदस्य आणि अमर्यादित AI",
+ "aiMaxToggle": "अमर्यादित AI आणि प्रगत मॉडेल्सचा प्रवेश",
+ "aiOnDeviceToggle": "जास्त गोपनीयतेसाठी स्थानिक AI",
+ "aiCredit": {
+ "title": "@:appName AI क्रेडिट जोडा",
+ "price": "{}",
+ "priceDescription": "1,000 क्रेडिट्ससाठी",
+ "purchase": "AI खरेदी करा",
+ "info": "प्रत्येक कार्यक्षेत्रासाठी 1,000 AI क्रेडिट्स जोडा आणि आपल्या कामाच्या प्रवाहात सानुकूलनीय AI सहजपणे एकत्र करा — अधिक स्मार्ट आणि जलद निकालांसाठी:",
+ "infoItemOne": "प्रत्येक डेटाबेससाठी 10,000 प्रतिसाद",
+ "infoItemTwo": "प्रत्येक कार्यक्षेत्रासाठी 1,000 प्रतिसाद"
+ },
+ "currentPlan": {
+ "bannerLabel": "सद्य योजना",
+ "freeTitle": "फ्री",
+ "proTitle": "प्रो",
+ "teamTitle": "टीम",
+ "freeInfo": "2 सदस्यांपर्यंत वैयक्तिक वापरासाठी उत्तम",
+ "proInfo": "10 सदस्यांपर्यंत लहान आणि मध्यम टीमसाठी योग्य",
+ "teamInfo": "सर्व उत्पादनक्षम आणि सुव्यवस्थित टीमसाठी योग्य",
+ "upgrade": "योजना बदला",
+ "canceledInfo": "तुमची योजना रद्द करण्यात आली आहे, {} रोजी तुम्हाला फ्री योजनेवर डाऊनग्रेड केले जाईल."
+ },
+ "addons": {
+ "title": "ऍड-ऑन्स",
+ "addLabel": "जोडा",
+ "activeLabel": "जोडले गेले",
+ "aiMax": {
+ "title": "AI Max",
+ "description": "प्रगत AI मॉडेल्ससह अमर्यादित AI प्रतिसाद आणि दरमहा 50 AI प्रतिमा",
+ "price": "{}",
+ "priceInfo": "प्रति वापरकर्ता प्रति महिना (वार्षिक बिलिंग)"
+ },
+ "aiOnDevice": {
+ "title": "मॅकसाठी ऑन-डिव्हाइस AI",
+ "description": "तुमच्या डिव्हाइसवर Mistral 7B, LLAMA 3 आणि अधिक स्थानिक मॉडेल्स चालवा",
+ "price": "{}",
+ "priceInfo": "प्रति वापरकर्ता प्रति महिना (वार्षिक बिलिंग)",
+ "recommend": "M1 किंवा नवीनतम शिफारस केली जाते"
+ }
+ },
+ "deal": {
+ "bannerLabel": "नववर्षाचे विशेष ऑफर!",
+ "title": "तुमची टीम वाढवा!",
+ "info": "Pro आणि Team योजनांवर 10% सूट मिळवा! @:appName AI सह शक्तिशाली नवीन वैशिष्ट्यांसह तुमचे कार्यक्षेत्र अधिक कार्यक्षम बनवा.",
+ "viewPlans": "योजना पहा"
+ }
+ }
+},
+ "billingPage": {
+ "menuLabel": "बिलिंग",
+ "title": "बिलिंग",
+ "plan": {
+ "title": "योजना",
+ "freeLabel": "फ्री",
+ "proLabel": "प्रो",
+ "planButtonLabel": "योजना बदला",
+ "billingPeriod": "बिलिंग कालावधी",
+ "periodButtonLabel": "कालावधी संपादित करा"
+ },
+ "paymentDetails": {
+ "title": "पेमेंट तपशील",
+ "methodLabel": "पेमेंट पद्धत",
+ "methodButtonLabel": "पद्धत संपादित करा"
+ },
+ "addons": {
+ "title": "ऍड-ऑन्स",
+ "addLabel": "जोडा",
+ "removeLabel": "काढा",
+ "renewLabel": "नवीन करा",
+ "aiMax": {
+ "label": "AI Max",
+ "description": "अमर्यादित AI आणि प्रगत मॉडेल्स अनलॉक करा",
+ "activeDescription": "पुढील बिलिंग तारीख {} आहे",
+ "canceledDescription": "AI Max {} पर्यंत उपलब्ध असेल"
+ },
+ "aiOnDevice": {
+ "label": "मॅकसाठी ऑन-डिव्हाइस AI",
+ "description": "तुमच्या डिव्हाइसवर अमर्यादित ऑन-डिव्हाइस AI अनलॉक करा",
+ "activeDescription": "पुढील बिलिंग तारीख {} आहे",
+ "canceledDescription": "मॅकसाठी ऑन-डिव्हाइस AI {} पर्यंत उपलब्ध असेल"
+ },
+ "removeDialog": {
+ "title": "{} काढा",
+ "description": "तुम्हाला {plan} काढायचे आहे का? तुम्हाला तत्काळ {plan} चे सर्व फिचर्स आणि फायदे वापरण्याचा अधिकार गमावावा लागेल."
+ }
+ },
+ "currentPeriodBadge": "सद्य कालावधी",
+ "changePeriod": "कालावधी बदला",
+ "planPeriod": "{} कालावधी",
+ "monthlyInterval": "मासिक",
+ "monthlyPriceInfo": "प्रति सदस्य, मासिक बिलिंग",
+ "annualInterval": "वार्षिक",
+ "annualPriceInfo": "प्रति सदस्य, वार्षिक बिलिंग"
+},
+ "comparePlanDialog": {
+ "title": "योजना तुलना आणि निवड",
+ "planFeatures": "योजनेची\nवैशिष्ट्ये",
+ "current": "सध्याची",
+ "actions": {
+ "upgrade": "अपग्रेड करा",
+ "downgrade": "डाऊनग्रेड करा",
+ "current": "सध्याची"
+ },
+ "freePlan": {
+ "title": "फ्री",
+ "description": "२ सदस्यांपर्यंत व्यक्तींसाठी सर्व काही व्यवस्थित करण्यासाठी",
+ "price": "{}",
+ "priceInfo": "सदैव फ्री"
+ },
+ "proPlan": {
+ "title": "प्रो",
+ "description": "लहान टीम्ससाठी प्रोजेक्ट्स व ज्ञान व्यवस्थापनासाठी",
+ "price": "{}",
+ "priceInfo": "प्रति सदस्य प्रति महिना\nवार्षिक बिलिंग\n\n{} मासिक बिलिंगसाठी"
+ },
+ "planLabels": {
+ "itemOne": "वर्कस्पेसेस",
+ "itemTwo": "सदस्य",
+ "itemThree": "स्टोरेज",
+ "itemFour": "रिअल-टाइम सहकार्य",
+ "itemFive": "मोबाईल अॅप",
+ "itemSix": "AI प्रतिसाद",
+ "itemSeven": "AI प्रतिमा",
+ "itemFileUpload": "फाइल अपलोड",
+ "customNamespace": "सानुकूल नेमस्पेस",
+ "tooltipSix": "‘Lifetime’ म्हणजे या प्रतिसादांची मर्यादा कधीही रीसेट केली जात नाही",
+ "intelligentSearch": "स्मार्ट शोध",
+ "tooltipSeven": "तुमच्या कार्यक्षेत्रासाठी URL चा भाग सानुकूलित करू देते",
+ "customNamespaceTooltip": "सानुकूल प्रकाशित साइट URL"
+ },
+ "freeLabels": {
+ "itemOne": "प्रत्येक वर्कस्पेसवर शुल्क",
+ "itemTwo": "२ पर्यंत",
+ "itemThree": "५ GB",
+ "itemFour": "होय",
+ "itemFive": "होय",
+ "itemSix": "१० कायमस्वरूपी",
+ "itemSeven": "२ कायमस्वरूपी",
+ "itemFileUpload": "७ MB पर्यंत",
+ "intelligentSearch": "स्मार्ट शोध"
+ },
+ "proLabels": {
+ "itemOne": "प्रत्येक वर्कस्पेसवर शुल्क",
+ "itemTwo": "१० पर्यंत",
+ "itemThree": "अमर्यादित",
+ "itemFour": "होय",
+ "itemFive": "होय",
+ "itemSix": "अमर्यादित",
+ "itemSeven": "दर महिन्याला १० प्रतिमा",
+ "itemFileUpload": "अमर्यादित",
+ "intelligentSearch": "स्मार्ट शोध"
+ },
+ "paymentSuccess": {
+ "title": "तुम्ही आता {} योजनेवर आहात!",
+ "description": "तुमचे पेमेंट यशस्वीरित्या पूर्ण झाले असून तुमची योजना @:appName {} मध्ये अपग्रेड झाली आहे. तुम्ही 'योजना' पृष्ठावर तपशील पाहू शकता."
+ },
+ "downgradeDialog": {
+ "title": "तुम्हाला योजना डाऊनग्रेड करायची आहे का?",
+ "description": "तुमची योजना डाऊनग्रेड केल्यास तुम्ही फ्री योजनेवर परत जाल. काही सदस्यांना या कार्यक्षेत्राचा प्रवेश मिळणार नाही आणि तुम्हाला स्टोरेज मर्यादेप्रमाणे जागा रिकामी करावी लागू शकते.",
+ "downgradeLabel": "योजना डाऊनग्रेड करा"
+ }
+},
+ "cancelSurveyDialog": {
+ "title": "तुम्ही जात आहात याचे दुःख आहे",
+ "description": "तुम्ही जात आहात याचे आम्हाला खरोखरच दुःख वाटते. @:appName सुधारण्यासाठी तुमचा अभिप्राय आम्हाला महत्त्वाचा वाटतो. कृपया काही क्षण घेऊन काही प्रश्नांची उत्तरे द्या.",
+ "commonOther": "इतर",
+ "otherHint": "तुमचे उत्तर येथे लिहा",
+ "questionOne": {
+ "question": "तुम्ही तुमची @:appName Pro सदस्यता का रद्द केली?",
+ "answerOne": "खर्च खूप जास्त आहे",
+ "answerTwo": "वैशिष्ट्ये अपेक्षांनुसार नव्हती",
+ "answerThree": "यापेक्षा चांगला पर्याय सापडला",
+ "answerFour": "वापर फारसा केला नाही, त्यामुळे खर्च योग्य वाटला नाही",
+ "answerFive": "सेवा समस्या किंवा तांत्रिक अडचणी"
+ },
+ "questionTwo": {
+ "question": "भविष्यात @:appName Pro सदस्यता पुन्हा घेण्याची शक्यता किती आहे?",
+ "answerOne": "खूप शक्यता आहे",
+ "answerTwo": "काहीशी शक्यता आहे",
+ "answerThree": "निश्चित नाही",
+ "answerFour": "अल्प शक्यता",
+ "answerFive": "एकदम कमी शक्यता"
+ },
+ "questionThree": {
+ "question": "तुमच्या सदस्यत्वादरम्यान कोणते Pro वैशिष्ट्य सर्वात उपयुक्त वाटले?",
+ "answerOne": "अनेक वापरकर्त्यांशी सहकार्य",
+ "answerTwo": "लांब कालावधीची आवृत्ती इतिहास",
+ "answerThree": "अमर्यादित AI प्रतिसाद",
+ "answerFour": "स्थानिक AI मॉडेल्सचा प्रवेश"
+ },
+ "questionFour": {
+ "question": "@:appName वापरण्याचा तुमचा एकूण अनुभव कसा होता?",
+ "answerOne": "खूप छान",
+ "answerTwo": "चांगला",
+ "answerThree": "सरासरी",
+ "answerFour": "सरासरीपेक्षा कमी",
+ "answerFive": "असंतोषजनक"
+ }
+},
+ "common": {
+ "uploadingFile": "फाईल अपलोड होत आहे. कृपया अॅप बंद करू नका",
+ "uploadNotionSuccess": "तुमची Notion zip फाईल यशस्वीरित्या अपलोड झाली आहे. आयात पूर्ण झाल्यानंतर तुम्हाला पुष्टीकरण ईमेल मिळेल",
+ "reset": "रीसेट करा"
+},
+ "menu": {
+ "appearance": "दृश्यरूप",
+ "language": "भाषा",
+ "user": "वापरकर्ता",
+ "files": "फाईल्स",
+ "notifications": "सूचना",
+ "open": "सेटिंग्ज उघडा",
+ "logout": "लॉगआउट",
+ "logoutPrompt": "तुम्हाला नक्की लॉगआउट करायचे आहे का?",
+ "selfEncryptionLogoutPrompt": "तुम्हाला खात्रीने लॉगआउट करायचे आहे का? कृपया खात्री करा की तुम्ही एनक्रिप्शन गुप्तकी कॉपी केली आहे",
+ "syncSetting": "सिंक्रोनायझेशन सेटिंग",
+ "cloudSettings": "क्लाऊड सेटिंग्ज",
+ "enableSync": "सिंक्रोनायझेशन सक्षम करा",
+ "enableSyncLog": "सिंक लॉगिंग सक्षम करा",
+ "enableSyncLogWarning": "सिंक समस्यांचे निदान करण्यात मदतीसाठी धन्यवाद. हे तुमचे डॉक्युमेंट संपादन स्थानिक फाईलमध्ये लॉग करेल. कृपया सक्षम केल्यानंतर अॅप बंद करून पुन्हा उघडा",
+ "enableEncrypt": "डेटा एन्क्रिप्ट करा",
+ "cloudURL": "बेस URL",
+ "webURL": "वेब URL",
+ "invalidCloudURLScheme": "अवैध स्कीम",
+ "cloudServerType": "क्लाऊड सर्व्हर",
+ "cloudServerTypeTip": "कृपया लक्षात घ्या की क्लाऊड सर्व्हर बदलल्यास तुम्ही सध्या लॉगिन केलेले खाते लॉगआउट होऊ शकते",
+ "cloudLocal": "स्थानिक",
+ "cloudAppFlowy": "@:appName Cloud",
+ "cloudAppFlowySelfHost": "@:appName क्लाऊड सेल्फ-होस्टेड",
+ "appFlowyCloudUrlCanNotBeEmpty": "क्लाऊड URL रिकामा असू शकत नाही",
+ "clickToCopy": "क्लिपबोर्डवर कॉपी करा",
+ "selfHostStart": "जर तुमच्याकडे सर्व्हर नसेल, तर कृपया हे पाहा",
+ "selfHostContent": "दस्तऐवज",
+ "selfHostEnd": "तुमचा स्वतःचा सर्व्हर कसा होस्ट करावा यासाठी मार्गदर्शनासाठी",
+ "pleaseInputValidURL": "कृपया वैध URL टाका",
+ "changeUrl": "सेल्फ-होस्टेड URL {} मध्ये बदला",
+ "cloudURLHint": "तुमच्या सर्व्हरचा बेस URL टाका",
+ "webURLHint": "तुमच्या वेब सर्व्हरचा बेस URL टाका",
+ "cloudWSURL": "वेबसॉकेट URL",
+ "cloudWSURLHint": "तुमच्या सर्व्हरचा वेबसॉकेट पत्ता टाका",
+ "restartApp": "अॅप रीस्टार्ट करा",
+ "restartAppTip": "बदल प्रभावी होण्यासाठी अॅप रीस्टार्ट करा. कृपया लक्षात घ्या की यामुळे सध्याचे खाते लॉगआउट होऊ शकते.",
+ "changeServerTip": "सर्व्हर बदलल्यानंतर, बदल लागू करण्यासाठी तुम्हाला 'रीस्टार्ट' बटणावर क्लिक करणे आवश्यक आहे",
+ "enableEncryptPrompt": "तुमचा डेटा सुरक्षित करण्यासाठी एन्क्रिप्शन सक्रिय करा. ही गुप्तकी सुरक्षित ठेवा; एकदा सक्षम केल्यावर ती बंद करता येणार नाही. जर हरवली तर तुमचा डेटा पुन्हा मिळवता येणार नाही. कॉपी करण्यासाठी क्लिक करा",
+ "inputEncryptPrompt": "कृपया खालीलसाठी तुमची एनक्रिप्शन गुप्तकी टाका:",
+ "clickToCopySecret": "गुप्तकी कॉपी करण्यासाठी क्लिक करा",
+ "configServerSetting": "तुमच्या सर्व्हर सेटिंग्ज कॉन्फिगर करा",
+ "configServerGuide": "`Quick Start` निवडल्यानंतर, `Settings` → \"Cloud Settings\" मध्ये जा आणि तुमचा सेल्फ-होस्टेड सर्व्हर कॉन्फिगर करा.",
+ "inputTextFieldHint": "तुमची गुप्तकी",
+ "historicalUserList": "वापरकर्ता लॉगिन इतिहास",
+ "historicalUserListTooltip": "ही यादी तुमची अनामिक खाती दर्शवते. तपशील पाहण्यासाठी खात्यावर क्लिक करा. अनामिक खाती 'सुरुवात करा' बटणावर क्लिक करून तयार केली जातात",
+ "openHistoricalUser": "अनामिक खाते उघडण्यासाठी क्लिक करा",
+ "customPathPrompt": "@:appName डेटा फोल्डर Google Drive सारख्या क्लाऊड-सिंक फोल्डरमध्ये साठवणे धोकादायक ठरू शकते. फोल्डरमधील डेटाबेस अनेक ठिकाणांवरून एकाच वेळी ऍक्सेस/संपादित केल्यास डेटा सिंक समस्या किंवा भ्रष्ट होण्याची शक्यता असते.",
+ "importAppFlowyData": "बाह्य @:appName फोल्डरमधून डेटा आयात करा",
+ "importingAppFlowyDataTip": "डेटा आयात सुरू आहे. कृपया अॅप बंद करू नका",
+ "importAppFlowyDataDescription": "बाह्य @:appName फोल्डरमधून डेटा कॉपी करून सध्याच्या AppFlowy डेटा फोल्डरमध्ये आयात करा",
+ "importSuccess": "@:appName डेटा फोल्डर यशस्वीरित्या आयात झाला",
+ "importFailed": "@:appName डेटा फोल्डर आयात करण्यात अयशस्वी",
+ "importGuide": "अधिक माहितीसाठी, कृपया संदर्भित दस्तऐवज पहा"
+},
+ "notifications": {
+ "enableNotifications": {
+ "label": "सूचना सक्षम करा",
+ "hint": "स्थानिक सूचना दिसू नयेत यासाठी बंद करा."
+ },
+ "showNotificationsIcon": {
+ "label": "सूचना चिन्ह दाखवा",
+ "hint": "साइडबारमध्ये सूचना चिन्ह लपवण्यासाठी बंद करा."
+ },
+ "archiveNotifications": {
+ "allSuccess": "सर्व सूचना यशस्वीरित्या संग्रहित केल्या",
+ "success": "सूचना यशस्वीरित्या संग्रहित केली"
+ },
+ "markAsReadNotifications": {
+ "allSuccess": "सर्व वाचलेल्या म्हणून चिन्हांकित केल्या",
+ "success": "वाचलेले म्हणून चिन्हांकित केले"
+ },
+ "action": {
+ "markAsRead": "वाचलेले म्हणून चिन्हांकित करा",
+ "multipleChoice": "अधिक निवडा",
+ "archive": "संग्रहित करा"
+ },
+ "settings": {
+ "settings": "सेटिंग्ज",
+ "markAllAsRead": "सर्व वाचलेले म्हणून चिन्हांकित करा",
+ "archiveAll": "सर्व संग्रहित करा"
+ },
+ "emptyInbox": {
+ "title": "इनबॉक्स झिरो!",
+ "description": "इथे सूचना मिळवण्यासाठी रिमाइंडर सेट करा."
+ },
+ "emptyUnread": {
+ "title": "कोणतीही न वाचलेली सूचना नाही",
+ "description": "तुम्ही सर्व वाचले आहे!"
+ },
+ "emptyArchived": {
+ "title": "कोणतीही संग्रहित सूचना नाही",
+ "description": "संग्रहित सूचना इथे दिसतील."
+ },
+ "tabs": {
+ "inbox": "इनबॉक्स",
+ "unread": "न वाचलेले",
+ "archived": "संग्रहित"
+ },
+ "refreshSuccess": "सूचना यशस्वीरित्या रीफ्रेश केल्या",
+ "titles": {
+ "notifications": "सूचना",
+ "reminder": "रिमाइंडर"
+ }
+},
+ "appearance": {
+ "resetSetting": "रीसेट",
+ "fontFamily": {
+ "label": "फॉन्ट फॅमिली",
+ "search": "शोध",
+ "defaultFont": "सिस्टम"
+ },
+ "themeMode": {
+ "label": "थीम मोड",
+ "light": "लाइट मोड",
+ "dark": "डार्क मोड",
+ "system": "सिस्टमशी जुळवा"
+ },
+ "fontScaleFactor": "फॉन्ट स्केल घटक",
+ "displaySize": "डिस्प्ले आकार",
+ "documentSettings": {
+ "cursorColor": "डॉक्युमेंट कर्सरचा रंग",
+ "selectionColor": "डॉक्युमेंट निवडीचा रंग",
+ "width": "डॉक्युमेंटची रुंदी",
+ "changeWidth": "बदला",
+ "pickColor": "रंग निवडा",
+ "colorShade": "रंगाची छटा",
+ "opacity": "अपारदर्शकता",
+ "hexEmptyError": "Hex रंग रिकामा असू शकत नाही",
+ "hexLengthError": "Hex व्हॅल्यू 6 अंकांची असावी",
+ "hexInvalidError": "अवैध Hex व्हॅल्यू",
+ "opacityEmptyError": "अपारदर्शकता रिकामी असू शकत नाही",
+ "opacityRangeError": "अपारदर्शकता 1 ते 100 दरम्यान असावी",
+ "app": "अॅप",
+ "flowy": "Flowy",
+ "apply": "लागू करा"
+ },
+ "layoutDirection": {
+ "label": "लेआउट दिशा",
+ "hint": "तुमच्या स्क्रीनवरील कंटेंटचा प्रवाह नियंत्रित करा — डावीकडून उजवीकडे किंवा उजवीकडून डावीकडे.",
+ "ltr": "LTR",
+ "rtl": "RTL"
+ },
+ "textDirection": {
+ "label": "मूलभूत मजकूर दिशा",
+ "hint": "मजकूर डावीकडून सुरू व्हावा की उजवीकडून याचे पूर्वनिर्धारित निर्धारण करा.",
+ "ltr": "LTR",
+ "rtl": "RTL",
+ "auto": "स्वयं",
+ "fallback": "लेआउट दिशेशी जुळवा"
+ },
+ "themeUpload": {
+ "button": "अपलोड",
+ "uploadTheme": "थीम अपलोड करा",
+ "description": "खालील बटण वापरून तुमची स्वतःची @:appName थीम अपलोड करा.",
+ "loading": "कृपया प्रतीक्षा करा. आम्ही तुमची थीम पडताळत आहोत आणि अपलोड करत आहोत...",
+ "uploadSuccess": "तुमची थीम यशस्वीरित्या अपलोड झाली आहे",
+ "deletionFailure": "थीम हटवण्यात अयशस्वी. कृपया ती मॅन्युअली हटवून पाहा.",
+ "filePickerDialogTitle": ".flowy_plugin फाईल निवडा",
+ "urlUploadFailure": "URL उघडण्यात अयशस्वी: {}"
+ },
+ "theme": "थीम",
+ "builtInsLabel": "अंतर्गत थीम्स",
+ "pluginsLabel": "प्लगइन्स",
+ "dateFormat": {
+ "label": "दिनांक फॉरमॅट",
+ "local": "स्थानिक",
+ "us": "US",
+ "iso": "ISO",
+ "friendly": "अनौपचारिक",
+ "dmy": "D/M/Y"
+ },
+ "timeFormat": {
+ "label": "वेळ फॉरमॅट",
+ "twelveHour": "१२ तास",
+ "twentyFourHour": "२४ तास"
+ },
+ "showNamingDialogWhenCreatingPage": "पृष्ठ तयार करताना नाव विचारणारा डायलॉग दाखवा",
+ "enableRTLToolbarItems": "RTL टूलबार आयटम्स सक्षम करा",
+ "members": {
+ "title": "सदस्य सेटिंग्ज",
+ "inviteMembers": "सदस्यांना आमंत्रण द्या",
+ "inviteHint": "ईमेलद्वारे आमंत्रण द्या",
+ "sendInvite": "आमंत्रण पाठवा",
+ "copyInviteLink": "आमंत्रण दुवा कॉपी करा",
+ "label": "सदस्य",
+ "user": "वापरकर्ता",
+ "role": "भूमिका",
+ "removeFromWorkspace": "वर्कस्पेसमधून काढा",
+ "removeFromWorkspaceSuccess": "वर्कस्पेसमधून यशस्वीरित्या काढले",
+ "removeFromWorkspaceFailed": "वर्कस्पेसमधून काढण्यात अयशस्वी",
+ "owner": "मालक",
+ "guest": "अतिथी",
+ "member": "सदस्य",
+ "memberHintText": "सदस्य पृष्ठे वाचू व संपादित करू शकतो",
+ "guestHintText": "अतिथी पृष्ठे वाचू शकतो, प्रतिक्रिया देऊ शकतो, टिप्पणी करू शकतो, आणि परवानगी असल्यास काही पृष्ठे संपादित करू शकतो.",
+ "emailInvalidError": "अवैध ईमेल, कृपया तपासा व पुन्हा प्रयत्न करा",
+ "emailSent": "ईमेल पाठवला गेला आहे, कृपया इनबॉक्स तपासा",
+ "members": "सदस्य",
+ "membersCount": {
+ "zero": "{} सदस्य",
+ "one": "{} सदस्य",
+ "other": "{} सदस्य"
+ },
+ "inviteFailedDialogTitle": "आमंत्रण पाठवण्यात अयशस्वी",
+ "inviteFailedMemberLimit": "सदस्य मर्यादा गाठली आहे, अधिक सदस्यांसाठी कृपया अपग्रेड करा.",
+ "inviteFailedMemberLimitMobile": "तुमच्या वर्कस्पेसने सदस्य मर्यादा गाठली आहे.",
+ "memberLimitExceeded": "सदस्य मर्यादा गाठली आहे, अधिक सदस्य आमंत्रित करण्यासाठी कृपया ",
+ "memberLimitExceededUpgrade": "अपग्रेड करा",
+ "memberLimitExceededPro": "सदस्य मर्यादा गाठली आहे, अधिक सदस्य आवश्यक असल्यास संपर्क करा",
+ "memberLimitExceededProContact": "support@appflowy.io",
+ "failedToAddMember": "सदस्य जोडण्यात अयशस्वी",
+ "addMemberSuccess": "सदस्य यशस्वीरित्या जोडला गेला",
+ "removeMember": "सदस्य काढा",
+ "areYouSureToRemoveMember": "तुम्हाला हा सदस्य काढायचा आहे का?",
+ "inviteMemberSuccess": "आमंत्रण यशस्वीरित्या पाठवले गेले",
+ "failedToInviteMember": "सदस्य आमंत्रित करण्यात अयशस्वी",
+ "workspaceMembersError": "अरे! काहीतरी चूक झाली आहे",
+ "workspaceMembersErrorDescription": "आम्ही सध्या सदस्यांची यादी लोड करू शकत नाही. कृपया नंतर पुन्हा प्रयत्न करा"
+ }
+},
+ "files": {
+ "copy": "कॉपी करा",
+ "defaultLocation": "फाईल्स आणि डेटाचा संचय स्थान",
+ "exportData": "तुमचा डेटा निर्यात करा",
+ "doubleTapToCopy": "पथ कॉपी करण्यासाठी दोनदा टॅप करा",
+ "restoreLocation": "@:appName चे मूळ स्थान पुनर्संचयित करा",
+ "customizeLocation": "इतर फोल्डर उघडा",
+ "restartApp": "बदल लागू करण्यासाठी कृपया अॅप रीस्टार्ट करा.",
+ "exportDatabase": "डेटाबेस निर्यात करा",
+ "selectFiles": "निर्यात करण्यासाठी फाईल्स निवडा",
+ "selectAll": "सर्व निवडा",
+ "deselectAll": "सर्व निवड रद्द करा",
+ "createNewFolder": "नवीन फोल्डर तयार करा",
+ "createNewFolderDesc": "तुमचा डेटा कुठे साठवायचा हे सांगा",
+ "defineWhereYourDataIsStored": "तुमचा डेटा कुठे साठवला जातो हे ठरवा",
+ "open": "उघडा",
+ "openFolder": "आधीक फोल्डर उघडा",
+ "openFolderDesc": "तुमच्या विद्यमान @:appName फोल्डरमध्ये वाचन व लेखन करा",
+ "folderHintText": "फोल्डरचे नाव",
+ "location": "नवीन फोल्डर तयार करत आहे",
+ "locationDesc": "तुमच्या @:appName डेटासाठी नाव निवडा",
+ "browser": "ब्राउझ करा",
+ "create": "तयार करा",
+ "set": "सेट करा",
+ "folderPath": "फोल्डर साठवण्याचा मार्ग",
+ "locationCannotBeEmpty": "मार्ग रिकामा असू शकत नाही",
+ "pathCopiedSnackbar": "फाईल संचय मार्ग क्लिपबोर्डवर कॉपी केला!",
+ "changeLocationTooltips": "डेटा डिरेक्टरी बदला",
+ "change": "बदला",
+ "openLocationTooltips": "इतर डेटा डिरेक्टरी उघडा",
+ "openCurrentDataFolder": "सध्याची डेटा डिरेक्टरी उघडा",
+ "recoverLocationTooltips": "@:appName च्या मूळ डेटा डिरेक्टरीवर रीसेट करा",
+ "exportFileSuccess": "फाईल यशस्वीरित्या निर्यात झाली!",
+ "exportFileFail": "फाईल निर्यात करण्यात अयशस्वी!",
+ "export": "निर्यात करा",
+ "clearCache": "कॅशे साफ करा",
+ "clearCacheDesc": "प्रतिमा लोड होत नाहीत किंवा फॉन्ट अयोग्यरित्या दिसत असल्यास, कॅशे साफ करून पाहा. ही क्रिया तुमचा वापरकर्ता डेटा हटवणार नाही.",
+ "areYouSureToClearCache": "तुम्हाला नक्की कॅशे साफ करायची आहे का?",
+ "clearCacheSuccess": "कॅशे यशस्वीरित्या साफ झाली!"
+},
+ "user": {
+ "name": "नाव",
+ "email": "ईमेल",
+ "tooltipSelectIcon": "चिन्ह निवडा",
+ "selectAnIcon": "चिन्ह निवडा",
+ "pleaseInputYourOpenAIKey": "कृपया तुमची AI की टाका",
+ "clickToLogout": "सध्याचा वापरकर्ता लॉगआउट करण्यासाठी क्लिक करा"
+},
+ "mobile": {
+ "personalInfo": "वैयक्तिक माहिती",
+ "username": "वापरकर्तानाव",
+ "usernameEmptyError": "वापरकर्तानाव रिकामे असू शकत नाही",
+ "about": "विषयी",
+ "pushNotifications": "पुश सूचना",
+ "support": "सपोर्ट",
+ "joinDiscord": "Discord मध्ये सहभागी व्हा",
+ "privacyPolicy": "गोपनीयता धोरण",
+ "userAgreement": "वापरकर्ता करार",
+ "termsAndConditions": "अटी व शर्ती",
+ "userprofileError": "वापरकर्ता प्रोफाइल लोड करण्यात अयशस्वी",
+ "userprofileErrorDescription": "कृपया लॉगआउट करून पुन्हा लॉगिन करा आणि त्रुटी अजूनही येते का ते पहा.",
+ "selectLayout": "लेआउट निवडा",
+ "selectStartingDay": "सप्ताहाचा प्रारंभ दिवस निवडा",
+ "version": "आवृत्ती"
+},
+ "grid": {
+ "deleteView": "तुम्हाला हे दृश्य हटवायचे आहे का?",
+ "createView": "नवीन",
+ "title": {
+ "placeholder": "नाव नाही"
+ },
+ "settings": {
+ "filter": "फिल्टर",
+ "sort": "क्रमवारी",
+ "sortBy": "यावरून क्रमवारी लावा",
+ "properties": "गुणधर्म",
+ "reorderPropertiesTooltip": "गुणधर्मांचे स्थान बदला",
+ "group": "समूह",
+ "addFilter": "फिल्टर जोडा",
+ "deleteFilter": "फिल्टर हटवा",
+ "filterBy": "यावरून फिल्टर करा",
+ "typeAValue": "मूल्य लिहा...",
+ "layout": "लेआउट",
+ "compactMode": "कॉम्पॅक्ट मोड",
+ "databaseLayout": "लेआउट",
+ "viewList": {
+ "zero": "० दृश्ये",
+ "one": "{count} दृश्य",
+ "other": "{count} दृश्ये"
+ },
+ "editView": "दृश्य संपादित करा",
+ "boardSettings": "बोर्ड सेटिंग",
+ "calendarSettings": "कॅलेंडर सेटिंग",
+ "createView": "नवीन दृश्य",
+ "duplicateView": "दृश्याची प्रत बनवा",
+ "deleteView": "दृश्य हटवा",
+ "numberOfVisibleFields": "{} दर्शविले"
+ },
+ "filter": {
+ "empty": "कोणतेही सक्रिय फिल्टर नाहीत",
+ "addFilter": "फिल्टर जोडा",
+ "cannotFindCreatableField": "फिल्टर करण्यासाठी योग्य फील्ड सापडले नाही",
+ "conditon": "अट",
+ "where": "जिथे"
+ },
+ "textFilter": {
+ "contains": "अंतर्भूत आहे",
+ "doesNotContain": "अंतर्भूत नाही",
+ "endsWith": "याने समाप्त होते",
+ "startWith": "याने सुरू होते",
+ "is": "आहे",
+ "isNot": "नाही",
+ "isEmpty": "रिकामे आहे",
+ "isNotEmpty": "रिकामे नाही",
+ "choicechipPrefix": {
+ "isNot": "नाही",
+ "startWith": "याने सुरू होते",
+ "endWith": "याने समाप्त होते",
+ "isEmpty": "रिकामे आहे",
+ "isNotEmpty": "रिकामे नाही"
+ }
+ },
+ "checkboxFilter": {
+ "isChecked": "निवडलेले आहे",
+ "isUnchecked": "निवडलेले नाही",
+ "choicechipPrefix": {
+ "is": "आहे"
+ }
+ },
+ "checklistFilter": {
+ "isComplete": "पूर्ण झाले आहे",
+ "isIncomplted": "अपूर्ण आहे"
+ },
+ "selectOptionFilter": {
+ "is": "आहे",
+ "isNot": "नाही",
+ "contains": "अंतर्भूत आहे",
+ "doesNotContain": "अंतर्भूत नाही",
+ "isEmpty": "रिकामे आहे",
+ "isNotEmpty": "रिकामे नाही"
+},
+"dateFilter": {
+ "is": "या दिवशी आहे",
+ "before": "पूर्वी आहे",
+ "after": "नंतर आहे",
+ "onOrBefore": "या दिवशी किंवा त्याआधी आहे",
+ "onOrAfter": "या दिवशी किंवा त्यानंतर आहे",
+ "between": "दरम्यान आहे",
+ "empty": "रिकामे आहे",
+ "notEmpty": "रिकामे नाही",
+ "startDate": "सुरुवातीची तारीख",
+ "endDate": "शेवटची तारीख",
+ "choicechipPrefix": {
+ "before": "पूर्वी",
+ "after": "नंतर",
+ "between": "दरम्यान",
+ "onOrBefore": "या दिवशी किंवा त्याआधी",
+ "onOrAfter": "या दिवशी किंवा त्यानंतर",
+ "isEmpty": "रिकामे आहे",
+ "isNotEmpty": "रिकामे नाही"
+ }
+},
+"numberFilter": {
+ "equal": "बरोबर आहे",
+ "notEqual": "बरोबर नाही",
+ "lessThan": "पेक्षा कमी आहे",
+ "greaterThan": "पेक्षा जास्त आहे",
+ "lessThanOrEqualTo": "किंवा कमी आहे",
+ "greaterThanOrEqualTo": "किंवा जास्त आहे",
+ "isEmpty": "रिकामे आहे",
+ "isNotEmpty": "रिकामे नाही"
+},
+"field": {
+ "label": "गुणधर्म",
+ "hide": "गुणधर्म लपवा",
+ "show": "गुणधर्म दर्शवा",
+ "insertLeft": "डावीकडे जोडा",
+ "insertRight": "उजवीकडे जोडा",
+ "duplicate": "प्रत बनवा",
+ "delete": "हटवा",
+ "wrapCellContent": "पाठ लपेटा",
+ "clear": "सेल्स रिकामे करा",
+ "switchPrimaryFieldTooltip": "प्राथमिक फील्डचा प्रकार बदलू शकत नाही",
+ "textFieldName": "मजकूर",
+ "checkboxFieldName": "चेकबॉक्स",
+ "dateFieldName": "तारीख",
+ "updatedAtFieldName": "शेवटचे अपडेट",
+ "createdAtFieldName": "तयार झाले",
+ "numberFieldName": "संख्या",
+ "singleSelectFieldName": "सिंगल सिलेक्ट",
+ "multiSelectFieldName": "मल्टीसिलेक्ट",
+ "urlFieldName": "URL",
+ "checklistFieldName": "चेकलिस्ट",
+ "relationFieldName": "संबंध",
+ "summaryFieldName": "AI सारांश",
+ "timeFieldName": "वेळ",
+ "mediaFieldName": "फाईल्स आणि मीडिया",
+ "translateFieldName": "AI भाषांतर",
+ "translateTo": "मध्ये भाषांतर करा",
+ "numberFormat": "संख्या स्वरूप",
+ "dateFormat": "तारीख स्वरूप",
+ "includeTime": "वेळ जोडा",
+ "isRange": "शेवटची तारीख",
+ "dateFormatFriendly": "महिना दिवस, वर्ष",
+ "dateFormatISO": "वर्ष-महिना-दिनांक",
+ "dateFormatLocal": "महिना/दिवस/वर्ष",
+ "dateFormatUS": "वर्ष/महिना/दिवस",
+ "dateFormatDayMonthYear": "दिवस/महिना/वर्ष",
+ "timeFormat": "वेळ स्वरूप",
+ "invalidTimeFormat": "अवैध स्वरूप",
+ "timeFormatTwelveHour": "१२ तास",
+ "timeFormatTwentyFourHour": "२४ तास",
+ "clearDate": "तारीख हटवा",
+ "dateTime": "तारीख व वेळ",
+ "startDateTime": "सुरुवातीची तारीख व वेळ",
+ "endDateTime": "शेवटची तारीख व वेळ",
+ "failedToLoadDate": "तारीख मूल्य लोड करण्यात अयशस्वी",
+ "selectTime": "वेळ निवडा",
+ "selectDate": "तारीख निवडा",
+ "visibility": "दृश्यता",
+ "propertyType": "गुणधर्माचा प्रकार",
+ "addSelectOption": "पर्याय जोडा",
+ "typeANewOption": "नवीन पर्याय लिहा",
+ "optionTitle": "पर्याय",
+ "addOption": "पर्याय जोडा",
+ "editProperty": "गुणधर्म संपादित करा",
+ "newProperty": "नवीन गुणधर्म",
+ "openRowDocument": "पृष्ठ म्हणून उघडा",
+ "deleteFieldPromptMessage": "तुम्हाला खात्री आहे का? हा गुणधर्म आणि त्याचा डेटा हटवला जाईल",
+ "clearFieldPromptMessage": "तुम्हाला खात्री आहे का? या कॉलममधील सर्व सेल्स रिकामे होतील",
+ "newColumn": "नवीन कॉलम",
+ "format": "स्वरूप",
+ "reminderOnDateTooltip": "या सेलमध्ये अनुस्मारक शेड्यूल केले आहे",
+ "optionAlreadyExist": "पर्याय आधीच अस्तित्वात आहे"
+},
+ "rowPage": {
+ "newField": "नवीन फील्ड जोडा",
+ "fieldDragElementTooltip": "मेनू उघडण्यासाठी क्लिक करा",
+ "showHiddenFields": {
+ "one": "{count} लपलेले फील्ड दाखवा",
+ "many": "{count} लपलेली फील्ड दाखवा",
+ "other": "{count} लपलेली फील्ड दाखवा"
+ },
+ "hideHiddenFields": {
+ "one": "{count} लपलेले फील्ड लपवा",
+ "many": "{count} लपलेली फील्ड लपवा",
+ "other": "{count} लपलेली फील्ड लपवा"
+ },
+ "openAsFullPage": "पूर्ण पृष्ठ म्हणून उघडा",
+ "moreRowActions": "अधिक पंक्ती क्रिया"
+},
+"sort": {
+ "ascending": "चढत्या क्रमाने",
+ "descending": "उतरत्या क्रमाने",
+ "by": "द्वारे",
+ "empty": "सक्रिय सॉर्ट्स नाहीत",
+ "cannotFindCreatableField": "सॉर्टसाठी योग्य फील्ड सापडले नाही",
+ "deleteAllSorts": "सर्व सॉर्ट्स हटवा",
+ "addSort": "सॉर्ट जोडा",
+ "sortsActive": "सॉर्टिंग करत असताना {intention} करू शकत नाही",
+ "removeSorting": "या दृश्यातील सर्व सॉर्ट्स हटवायचे आहेत का?",
+ "fieldInUse": "या फील्डवर आधीच सॉर्टिंग चालू आहे"
+},
+"row": {
+ "label": "पंक्ती",
+ "duplicate": "प्रत बनवा",
+ "delete": "हटवा",
+ "titlePlaceholder": "शीर्षक नाही",
+ "textPlaceholder": "रिक्त",
+ "copyProperty": "गुणधर्म क्लिपबोर्डवर कॉपी केला",
+ "count": "संख्या",
+ "newRow": "नवीन पंक्ती",
+ "loadMore": "अधिक लोड करा",
+ "action": "क्रिया",
+ "add": "खाली जोडा वर क्लिक करा",
+ "drag": "हलवण्यासाठी ड्रॅग करा",
+ "deleteRowPrompt": "तुम्हाला खात्री आहे का की ही पंक्ती हटवायची आहे? ही क्रिया पूर्ववत करता येणार नाही.",
+ "deleteCardPrompt": "तुम्हाला खात्री आहे का की हे कार्ड हटवायचे आहे? ही क्रिया पूर्ववत करता येणार नाही.",
+ "dragAndClick": "हलवण्यासाठी ड्रॅग करा, मेनू उघडण्यासाठी क्लिक करा",
+ "insertRecordAbove": "वर रेकॉर्ड जोडा",
+ "insertRecordBelow": "खाली रेकॉर्ड जोडा",
+ "noContent": "माहिती नाही",
+ "reorderRowDescription": "पंक्तीचे पुन्हा क्रमांकन",
+ "createRowAboveDescription": "वर पंक्ती तयार करा",
+ "createRowBelowDescription": "खाली पंक्ती जोडा"
+},
+"selectOption": {
+ "create": "तयार करा",
+ "purpleColor": "जांभळा",
+ "pinkColor": "गुलाबी",
+ "lightPinkColor": "फिकट गुलाबी",
+ "orangeColor": "नारंगी",
+ "yellowColor": "पिवळा",
+ "limeColor": "लिंबू",
+ "greenColor": "हिरवा",
+ "aquaColor": "आक्वा",
+ "blueColor": "निळा",
+ "deleteTag": "टॅग हटवा",
+ "colorPanelTitle": "रंग",
+ "panelTitle": "पर्याय निवडा किंवा नवीन तयार करा",
+ "searchOption": "पर्याय शोधा",
+ "searchOrCreateOption": "पर्याय शोधा किंवा तयार करा",
+ "createNew": "नवीन तयार करा",
+ "orSelectOne": "किंवा पर्याय निवडा",
+ "typeANewOption": "नवीन पर्याय टाइप करा",
+ "tagName": "टॅग नाव"
+},
+"checklist": {
+ "taskHint": "कार्याचे वर्णन",
+ "addNew": "नवीन कार्य जोडा",
+ "submitNewTask": "तयार करा",
+ "hideComplete": "पूर्ण कार्ये लपवा",
+ "showComplete": "सर्व कार्ये दाखवा"
+},
+"url": {
+ "launch": "बाह्य ब्राउझरमध्ये लिंक उघडा",
+ "copy": "लिंक क्लिपबोर्डवर कॉपी करा",
+ "textFieldHint": "URL टाका",
+ "copiedNotification": "क्लिपबोर्डवर कॉपी केले!"
+},
+"relation": {
+ "relatedDatabasePlaceLabel": "संबंधित डेटाबेस",
+ "relatedDatabasePlaceholder": "काही नाही",
+ "inRelatedDatabase": "या मध्ये",
+ "rowSearchTextFieldPlaceholder": "शोध",
+ "noDatabaseSelected": "कोणताही डेटाबेस निवडलेला नाही, कृपया खालील यादीतून एक निवडा:",
+ "emptySearchResult": "कोणतीही नोंद सापडली नाही",
+ "linkedRowListLabel": "{count} लिंक केलेल्या पंक्ती",
+ "unlinkedRowListLabel": "आणखी एक पंक्ती लिंक करा"
+},
+"menuName": "ग्रिड",
+"referencedGridPrefix": "दृश्य",
+"calculate": "गणना करा",
+"calculationTypeLabel": {
+ "none": "काही नाही",
+ "average": "सरासरी",
+ "max": "कमाल",
+ "median": "मध्यम",
+ "min": "किमान",
+ "sum": "बेरीज",
+ "count": "मोजणी",
+ "countEmpty": "रिकाम्यांची मोजणी",
+ "countEmptyShort": "रिक्त",
+ "countNonEmpty": "रिक्त नसलेल्यांची मोजणी",
+ "countNonEmptyShort": "भरलेले"
+},
+"media": {
+ "rename": "पुन्हा नाव द्या",
+ "download": "डाउनलोड करा",
+ "expand": "मोठे करा",
+ "delete": "हटवा",
+ "moreFilesHint": "+{}",
+ "addFileOrImage": "फाईल किंवा लिंक जोडा",
+ "attachmentsHint": "{}",
+ "addFileMobile": "फाईल जोडा",
+ "extraCount": "+{}",
+ "deleteFileDescription": "तुम्हाला खात्री आहे का की ही फाईल हटवायची आहे? ही क्रिया पूर्ववत करता येणार नाही.",
+ "showFileNames": "फाईलचे नाव दाखवा",
+ "downloadSuccess": "फाईल डाउनलोड झाली",
+ "downloadFailedToken": "फाईल डाउनलोड अयशस्वी, वापरकर्ता टोकन अनुपलब्ध",
+ "setAsCover": "कव्हर म्हणून सेट करा",
+ "openInBrowser": "ब्राउझरमध्ये उघडा",
+ "embedLink": "फाईल लिंक एम्बेड करा"
+ }
+},
+ "document": {
+ "menuName": "दस्तऐवज",
+ "date": {
+ "timeHintTextInTwelveHour": "01:00 PM",
+ "timeHintTextInTwentyFourHour": "13:00"
+ },
+ "creating": "तयार करत आहे...",
+ "slashMenu": {
+ "board": {
+ "selectABoardToLinkTo": "लिंक करण्यासाठी बोर्ड निवडा",
+ "createANewBoard": "नवीन बोर्ड तयार करा"
+ },
+ "grid": {
+ "selectAGridToLinkTo": "लिंक करण्यासाठी ग्रिड निवडा",
+ "createANewGrid": "नवीन ग्रिड तयार करा"
+ },
+ "calendar": {
+ "selectACalendarToLinkTo": "लिंक करण्यासाठी दिनदर्शिका निवडा",
+ "createANewCalendar": "नवीन दिनदर्शिका तयार करा"
+ },
+ "document": {
+ "selectADocumentToLinkTo": "लिंक करण्यासाठी दस्तऐवज निवडा"
+ },
+ "name": {
+ "textStyle": "मजकुराची शैली",
+ "list": "यादी",
+ "toggle": "टॉगल",
+ "fileAndMedia": "फाईल व मीडिया",
+ "simpleTable": "सोपे टेबल",
+ "visuals": "दृश्य घटक",
+ "document": "दस्तऐवज",
+ "advanced": "प्रगत",
+ "text": "मजकूर",
+ "heading1": "शीर्षक 1",
+ "heading2": "शीर्षक 2",
+ "heading3": "शीर्षक 3",
+ "image": "प्रतिमा",
+ "bulletedList": "बुलेट यादी",
+ "numberedList": "क्रमांकित यादी",
+ "todoList": "करण्याची यादी",
+ "doc": "दस्तऐवज",
+ "linkedDoc": "पृष्ठाशी लिंक करा",
+ "grid": "ग्रिड",
+ "linkedGrid": "लिंक केलेला ग्रिड",
+ "kanban": "कानबन",
+ "linkedKanban": "लिंक केलेला कानबन",
+ "calendar": "दिनदर्शिका",
+ "linkedCalendar": "लिंक केलेली दिनदर्शिका",
+ "quote": "उद्धरण",
+ "divider": "विभाजक",
+ "table": "टेबल",
+ "callout": "महत्त्वाचा मजकूर",
+ "outline": "रूपरेषा",
+ "mathEquation": "गणिती समीकरण",
+ "code": "कोड",
+ "toggleList": "टॉगल यादी",
+ "toggleHeading1": "टॉगल शीर्षक 1",
+ "toggleHeading2": "टॉगल शीर्षक 2",
+ "toggleHeading3": "टॉगल शीर्षक 3",
+ "emoji": "इमोजी",
+ "aiWriter": "AI ला काहीही विचारा",
+ "dateOrReminder": "दिनांक किंवा स्मरणपत्र",
+ "photoGallery": "फोटो गॅलरी",
+ "file": "फाईल",
+ "twoColumns": "२ स्तंभ",
+ "threeColumns": "३ स्तंभ",
+ "fourColumns": "४ स्तंभ"
+ },
+ "subPage": {
+ "name": "दस्तऐवज",
+ "keyword1": "उपपृष्ठ",
+ "keyword2": "पृष्ठ",
+ "keyword3": "चाइल्ड पृष्ठ",
+ "keyword4": "पृष्ठ जोडा",
+ "keyword5": "एम्बेड पृष्ठ",
+ "keyword6": "नवीन पृष्ठ",
+ "keyword7": "पृष्ठ तयार करा",
+ "keyword8": "दस्तऐवज"
+ }
+ },
+ "selectionMenu": {
+ "outline": "रूपरेषा",
+ "codeBlock": "कोड ब्लॉक"
+ },
+ "plugins": {
+ "referencedBoard": "संदर्भित बोर्ड",
+ "referencedGrid": "संदर्भित ग्रिड",
+ "referencedCalendar": "संदर्भित दिनदर्शिका",
+ "referencedDocument": "संदर्भित दस्तऐवज",
+ "aiWriter": {
+ "userQuestion": "AI ला काहीही विचारा",
+ "continueWriting": "लेखन सुरू ठेवा",
+ "fixSpelling": "स्पेलिंग व व्याकरण सुधारणा",
+ "improveWriting": "लेखन सुधारित करा",
+ "summarize": "सारांश द्या",
+ "explain": "स्पष्टीकरण द्या",
+ "makeShorter": "लहान करा",
+ "makeLonger": "मोठे करा"
+ },
+ "autoGeneratorMenuItemName": "AI लेखक",
+"autoGeneratorTitleName": "AI: काहीही लिहिण्यासाठी AI ला विचारा...",
+"autoGeneratorLearnMore": "अधिक जाणून घ्या",
+"autoGeneratorGenerate": "उत्पन्न करा",
+"autoGeneratorHintText": "AI ला विचारा...",
+"autoGeneratorCantGetOpenAIKey": "AI की मिळवू शकलो नाही",
+"autoGeneratorRewrite": "पुन्हा लिहा",
+"smartEdit": "AI ला विचारा",
+"aI": "AI",
+"smartEditFixSpelling": "स्पेलिंग आणि व्याकरण सुधारा",
+"warning": "⚠️ AI उत्तरं चुकीची किंवा दिशाभूल करणारी असू शकतात.",
+"smartEditSummarize": "सारांश द्या",
+"smartEditImproveWriting": "लेखन सुधारित करा",
+"smartEditMakeLonger": "लांब करा",
+"smartEditCouldNotFetchResult": "AI कडून उत्तर मिळवता आले नाही",
+"smartEditCouldNotFetchKey": "AI की मिळवता आली नाही",
+"smartEditDisabled": "सेटिंग्जमध्ये AI कनेक्ट करा",
+"appflowyAIEditDisabled": "AI वैशिष्ट्ये सक्षम करण्यासाठी साइन इन करा",
+"discardResponse": "AI उत्तर फेकून द्यायचं आहे का?",
+"createInlineMathEquation": "समीकरण तयार करा",
+"fonts": "फॉन्ट्स",
+"insertDate": "तारीख जोडा",
+"emoji": "इमोजी",
+"toggleList": "टॉगल यादी",
+"emptyToggleHeading": "रिकामे टॉगल h{}. मजकूर जोडण्यासाठी क्लिक करा.",
+"emptyToggleList": "रिकामी टॉगल यादी. मजकूर जोडण्यासाठी क्लिक करा.",
+"emptyToggleHeadingWeb": "रिकामे टॉगल h{level}. मजकूर जोडण्यासाठी क्लिक करा",
+"quoteList": "उद्धरण यादी",
+"numberedList": "क्रमांकित यादी",
+"bulletedList": "बुलेट यादी",
+"todoList": "करण्याची यादी",
+"callout": "ठळक मजकूर",
+"simpleTable": {
+ "moreActions": {
+ "color": "रंग",
+ "align": "पंक्तिबद्ध करा",
+ "delete": "हटा",
+ "duplicate": "डुप्लिकेट करा",
+ "insertLeft": "डावीकडे घाला",
+ "insertRight": "उजवीकडे घाला",
+ "insertAbove": "वर घाला",
+ "insertBelow": "खाली घाला",
+ "headerColumn": "हेडर स्तंभ",
+ "headerRow": "हेडर ओळ",
+ "clearContents": "सामग्री साफ करा",
+ "setToPageWidth": "पृष्ठाच्या रुंदीप्रमाणे सेट करा",
+ "distributeColumnsWidth": "स्तंभ समान करा",
+ "duplicateRow": "ओळ डुप्लिकेट करा",
+ "duplicateColumn": "स्तंभ डुप्लिकेट करा",
+ "textColor": "मजकूराचा रंग",
+ "cellBackgroundColor": "सेलचा पार्श्वभूमी रंग",
+ "duplicateTable": "टेबल डुप्लिकेट करा"
+ },
+ "clickToAddNewRow": "नवीन ओळ जोडण्यासाठी क्लिक करा",
+ "clickToAddNewColumn": "नवीन स्तंभ जोडण्यासाठी क्लिक करा",
+ "clickToAddNewRowAndColumn": "नवीन ओळ आणि स्तंभ जोडण्यासाठी क्लिक करा",
+ "headerName": {
+ "table": "टेबल",
+ "alignText": "मजकूर पंक्तिबद्ध करा"
+ }
+},
+"cover": {
+ "changeCover": "कव्हर बदला",
+ "colors": "रंग",
+ "images": "प्रतिमा",
+ "clearAll": "सर्व साफ करा",
+ "abstract": "ऍबस्ट्रॅक्ट",
+ "addCover": "कव्हर जोडा",
+ "addLocalImage": "स्थानिक प्रतिमा जोडा",
+ "invalidImageUrl": "अवैध प्रतिमा URL",
+ "failedToAddImageToGallery": "प्रतिमा गॅलरीत जोडता आली नाही",
+ "enterImageUrl": "प्रतिमा URL लिहा",
+ "add": "जोडा",
+ "back": "मागे",
+ "saveToGallery": "गॅलरीत जतन करा",
+ "removeIcon": "आयकॉन काढा",
+ "removeCover": "कव्हर काढा",
+ "pasteImageUrl": "प्रतिमा URL पेस्ट करा",
+ "or": "किंवा",
+ "pickFromFiles": "फाईल्समधून निवडा",
+ "couldNotFetchImage": "प्रतिमा मिळवता आली नाही",
+ "imageSavingFailed": "प्रतिमा जतन करणे अयशस्वी",
+ "addIcon": "आयकॉन जोडा",
+ "changeIcon": "आयकॉन बदला",
+ "coverRemoveAlert": "हे हटवल्यानंतर कव्हरमधून काढले जाईल.",
+ "alertDialogConfirmation": "तुम्हाला खात्री आहे का? तुम्हाला पुढे जायचे आहे?"
+},
+"mathEquation": {
+ "name": "गणिती समीकरण",
+ "addMathEquation": "TeX समीकरण जोडा",
+ "editMathEquation": "गणिती समीकरण संपादित करा"
+},
+"optionAction": {
+ "click": "क्लिक",
+ "toOpenMenu": "मेनू उघडण्यासाठी",
+ "drag": "ओढा",
+ "toMove": "हलवण्यासाठी",
+ "delete": "हटा",
+ "duplicate": "डुप्लिकेट करा",
+ "turnInto": "मध्ये बदला",
+ "moveUp": "वर हलवा",
+ "moveDown": "खाली हलवा",
+ "color": "रंग",
+ "align": "पंक्तिबद्ध करा",
+ "left": "डावीकडे",
+ "center": "मध्यभागी",
+ "right": "उजवीकडे",
+ "defaultColor": "डिफॉल्ट",
+ "depth": "खोली",
+ "copyLinkToBlock": "ब्लॉकसाठी लिंक कॉपी करा"
+},
+ "image": {
+ "addAnImage": "प्रतिमा जोडा",
+ "copiedToPasteBoard": "प्रतिमेची लिंक क्लिपबोर्डवर कॉपी केली गेली आहे",
+ "addAnImageDesktop": "प्रतिमा जोडा",
+ "addAnImageMobile": "एक किंवा अधिक प्रतिमा जोडण्यासाठी क्लिक करा",
+ "dropImageToInsert": "प्रतिमा ड्रॉप करून जोडा",
+ "imageUploadFailed": "प्रतिमा अपलोड करण्यात अयशस्वी",
+ "imageDownloadFailed": "प्रतिमा डाउनलोड करण्यात अयशस्वी, कृपया पुन्हा प्रयत्न करा",
+ "imageDownloadFailedToken": "युजर टोकन नसल्यामुळे प्रतिमा डाउनलोड करण्यात अयशस्वी, कृपया पुन्हा प्रयत्न करा",
+ "errorCode": "त्रुटी कोड"
+},
+"photoGallery": {
+ "name": "फोटो गॅलरी",
+ "imageKeyword": "प्रतिमा",
+ "imageGalleryKeyword": "प्रतिमा गॅलरी",
+ "photoKeyword": "फोटो",
+ "photoBrowserKeyword": "फोटो ब्राउझर",
+ "galleryKeyword": "गॅलरी",
+ "addImageTooltip": "प्रतिमा जोडा",
+ "changeLayoutTooltip": "लेआउट बदला",
+ "browserLayout": "ब्राउझर",
+ "gridLayout": "ग्रिड",
+ "deleteBlockTooltip": "संपूर्ण गॅलरी हटवा"
+},
+"math": {
+ "copiedToPasteBoard": "समीकरण क्लिपबोर्डवर कॉपी केले गेले आहे"
+},
+"urlPreview": {
+ "copiedToPasteBoard": "लिंक क्लिपबोर्डवर कॉपी केली गेली आहे",
+ "convertToLink": "एंबेड लिंकमध्ये रूपांतर करा"
+},
+"outline": {
+ "addHeadingToCreateOutline": "सामग्री यादी तयार करण्यासाठी शीर्षके जोडा.",
+ "noMatchHeadings": "जुळणारी शीर्षके आढळली नाहीत."
+},
+"table": {
+ "addAfter": "नंतर जोडा",
+ "addBefore": "आधी जोडा",
+ "delete": "हटा",
+ "clear": "सामग्री साफ करा",
+ "duplicate": "डुप्लिकेट करा",
+ "bgColor": "पार्श्वभूमीचा रंग"
+},
+"contextMenu": {
+ "copy": "कॉपी करा",
+ "cut": "कापा",
+ "paste": "पेस्ट करा",
+ "pasteAsPlainText": "साध्या मजकूराच्या स्वरूपात पेस्ट करा"
+},
+"action": "कृती",
+"database": {
+ "selectDataSource": "डेटा स्रोत निवडा",
+ "noDataSource": "डेटा स्रोत नाही",
+ "selectADataSource": "डेटा स्रोत निवडा",
+ "toContinue": "पुढे जाण्यासाठी",
+ "newDatabase": "नवीन डेटाबेस",
+ "linkToDatabase": "डेटाबेसशी लिंक करा"
+},
+"date": "तारीख",
+"video": {
+ "label": "व्हिडिओ",
+ "emptyLabel": "व्हिडिओ जोडा",
+ "placeholder": "व्हिडिओ लिंक पेस्ट करा",
+ "copiedToPasteBoard": "व्हिडिओ लिंक क्लिपबोर्डवर कॉपी केली गेली आहे",
+ "insertVideo": "व्हिडिओ जोडा",
+ "invalidVideoUrl": "ही URL सध्या समर्थित नाही.",
+ "invalidVideoUrlYouTube": "YouTube सध्या समर्थित नाही.",
+ "supportedFormats": "समर्थित स्वरूप: MP4, WebM, MOV, AVI, FLV, MPEG/M4V, H.264"
+},
+"file": {
+ "name": "फाईल",
+ "uploadTab": "अपलोड",
+ "uploadMobile": "फाईल निवडा",
+ "uploadMobileGallery": "फोटो गॅलरीमधून",
+ "networkTab": "लिंक एम्बेड करा",
+ "placeholderText": "फाईल अपलोड किंवा एम्बेड करा",
+ "placeholderDragging": "फाईल ड्रॉप करून अपलोड करा",
+ "dropFileToUpload": "फाईल ड्रॉप करून अपलोड करा",
+ "fileUploadHint": "फाईल ड्रॅग आणि ड्रॉप करा किंवा क्लिक करा ",
+ "fileUploadHintSuffix": "ब्राउझ करा",
+ "networkHint": "फाईल लिंक पेस्ट करा",
+ "networkUrlInvalid": "अवैध URL. कृपया तपासा आणि पुन्हा प्रयत्न करा.",
+ "networkAction": "एम्बेड",
+ "fileTooBigError": "फाईल खूप मोठी आहे, कृपया 10MB पेक्षा कमी फाईल अपलोड करा",
+ "renameFile": {
+ "title": "फाईलचे नाव बदला",
+ "description": "या फाईलसाठी नवीन नाव लिहा",
+ "nameEmptyError": "फाईलचे नाव रिकामे असू शकत नाही."
+ },
+ "uploadedAt": "{} रोजी अपलोड केले",
+ "linkedAt": "{} रोजी लिंक जोडली",
+ "failedToOpenMsg": "उघडण्यात अयशस्वी, फाईल सापडली नाही"
+},
+"subPage": {
+ "handlingPasteHint": " - (पेस्ट प्रक्रिया करत आहे)",
+ "errors": {
+ "failedDeletePage": "पृष्ठ हटवण्यात अयशस्वी",
+ "failedCreatePage": "पृष्ठ तयार करण्यात अयशस्वी",
+ "failedMovePage": "हे पृष्ठ हलवण्यात अयशस्वी",
+ "failedDuplicatePage": "पृष्ठ डुप्लिकेट करण्यात अयशस्वी",
+ "failedDuplicateFindView": "पृष्ठ डुप्लिकेट करण्यात अयशस्वी - मूळ दृश्य सापडले नाही"
+ }
+},
+ "cannotMoveToItsChildren": "मुलांमध्ये हलवू शकत नाही"
+},
+"outlineBlock": {
+ "placeholder": "सामग्री सूची"
+},
+"textBlock": {
+ "placeholder": "कमांडसाठी '/' टाइप करा"
+},
+"title": {
+ "placeholder": "शीर्षक नाही"
+},
+"imageBlock": {
+ "placeholder": "प्रतिमा जोडण्यासाठी क्लिक करा",
+ "upload": {
+ "label": "अपलोड",
+ "placeholder": "प्रतिमा अपलोड करण्यासाठी क्लिक करा"
+ },
+ "url": {
+ "label": "प्रतिमेची URL",
+ "placeholder": "प्रतिमेची URL टाका"
+ },
+ "ai": {
+ "label": "AI द्वारे प्रतिमा तयार करा",
+ "placeholder": "AI द्वारे प्रतिमा तयार करण्यासाठी कृपया संकेत द्या"
+ },
+ "stability_ai": {
+ "label": "Stability AI द्वारे प्रतिमा तयार करा",
+ "placeholder": "Stability AI द्वारे प्रतिमा तयार करण्यासाठी कृपया संकेत द्या"
+ },
+ "support": "प्रतिमेचा कमाल आकार 5MB आहे. समर्थित स्वरूप: JPEG, PNG, GIF, SVG",
+ "error": {
+ "invalidImage": "अवैध प्रतिमा",
+ "invalidImageSize": "प्रतिमेचा आकार 5MB पेक्षा कमी असावा",
+ "invalidImageFormat": "प्रतिमेचे स्वरूप समर्थित नाही. समर्थित स्वरूप: JPEG, PNG, JPG, GIF, SVG, WEBP",
+ "invalidImageUrl": "अवैध प्रतिमेची URL",
+ "noImage": "अशी फाईल किंवा निर्देशिका नाही",
+ "multipleImagesFailed": "एक किंवा अधिक प्रतिमा अपलोड करण्यात अयशस्वी, कृपया पुन्हा प्रयत्न करा"
+ },
+ "embedLink": {
+ "label": "लिंक एम्बेड करा",
+ "placeholder": "प्रतिमेची लिंक पेस्ट करा किंवा टाका"
+ },
+ "unsplash": {
+ "label": "Unsplash"
+ },
+ "searchForAnImage": "प्रतिमा शोधा",
+ "pleaseInputYourOpenAIKey": "कृपया सेटिंग्ज पृष्ठात आपला AI की प्रविष्ट करा",
+ "saveImageToGallery": "प्रतिमा जतन करा",
+ "failedToAddImageToGallery": "प्रतिमा जतन करण्यात अयशस्वी",
+ "successToAddImageToGallery": "प्रतिमा 'Photos' मध्ये जतन केली",
+ "unableToLoadImage": "प्रतिमा लोड करण्यात अयशस्वी",
+ "maximumImageSize": "कमाल प्रतिमा अपलोड आकार 10MB आहे",
+ "uploadImageErrorImageSizeTooBig": "प्रतिमेचा आकार 10MB पेक्षा कमी असावा",
+ "imageIsUploading": "प्रतिमा अपलोड होत आहे",
+ "openFullScreen": "पूर्ण स्क्रीनमध्ये उघडा",
+ "interactiveViewer": {
+ "toolbar": {
+ "previousImageTooltip": "मागील प्रतिमा",
+ "nextImageTooltip": "पुढील प्रतिमा",
+ "zoomOutTooltip": "लहान करा",
+ "zoomInTooltip": "मोठी करा",
+ "changeZoomLevelTooltip": "झूम पातळी बदला",
+ "openLocalImage": "प्रतिमा उघडा",
+ "downloadImage": "प्रतिमा डाउनलोड करा",
+ "closeViewer": "इंटरअॅक्टिव्ह व्ह्युअर बंद करा",
+ "scalePercentage": "{}%",
+ "deleteImageTooltip": "प्रतिमा हटवा"
+ }
+ }
+},
+ "codeBlock": {
+ "language": {
+ "label": "भाषा",
+ "placeholder": "भाषा निवडा",
+ "auto": "स्वयंचलित"
+ },
+ "copyTooltip": "कॉपी करा",
+ "searchLanguageHint": "भाषा शोधा",
+ "codeCopiedSnackbar": "कोड क्लिपबोर्डवर कॉपी झाला!"
+},
+"inlineLink": {
+ "placeholder": "लिंक पेस्ट करा किंवा टाका",
+ "openInNewTab": "नवीन टॅबमध्ये उघडा",
+ "copyLink": "लिंक कॉपी करा",
+ "removeLink": "लिंक काढा",
+ "url": {
+ "label": "लिंक URL",
+ "placeholder": "लिंक URL टाका"
+ },
+ "title": {
+ "label": "लिंक शीर्षक",
+ "placeholder": "लिंक शीर्षक टाका"
+ }
+},
+"mention": {
+ "placeholder": "कोणीतरी, पृष्ठ किंवा तारीख नमूद करा...",
+ "page": {
+ "label": "पृष्ठाला लिंक करा",
+ "tooltip": "पृष्ठ उघडण्यासाठी क्लिक करा"
+ },
+ "deleted": "हटवले गेले",
+ "deletedContent": "ही सामग्री अस्तित्वात नाही किंवा हटवण्यात आली आहे",
+ "noAccess": "प्रवेश नाही",
+ "deletedPage": "हटवलेले पृष्ठ",
+ "trashHint": " - ट्रॅशमध्ये",
+ "morePages": "अजून पृष्ठे"
+},
+"toolbar": {
+ "resetToDefaultFont": "डीफॉल्ट फॉन्टवर परत जा",
+ "textSize": "मजकूराचा आकार",
+ "textColor": "मजकूराचा रंग",
+ "h1": "मथळा 1",
+ "h2": "मथळा 2",
+ "h3": "मथळा 3",
+ "alignLeft": "डावीकडे संरेखित करा",
+ "alignRight": "उजवीकडे संरेखित करा",
+ "alignCenter": "मध्यभागी संरेखित करा",
+ "link": "लिंक",
+ "textAlign": "मजकूर संरेखन",
+ "moreOptions": "अधिक पर्याय",
+ "font": "फॉन्ट",
+ "inlineCode": "इनलाइन कोड",
+ "suggestions": "सूचना",
+ "turnInto": "मध्ये रूपांतरित करा",
+ "equation": "समीकरण",
+ "insert": "घाला",
+ "linkInputHint": "लिंक पेस्ट करा किंवा पृष्ठे शोधा",
+ "pageOrURL": "पृष्ठ किंवा URL",
+ "linkName": "लिंकचे नाव",
+ "linkNameHint": "लिंकचे नाव प्रविष्ट करा"
+},
+"errorBlock": {
+ "theBlockIsNotSupported": "ब्लॉक सामग्री पार्स करण्यात अक्षम",
+ "clickToCopyTheBlockContent": "ब्लॉक सामग्री कॉपी करण्यासाठी क्लिक करा",
+ "blockContentHasBeenCopied": "ब्लॉक सामग्री कॉपी केली आहे.",
+ "parseError": "{} ब्लॉक पार्स करताना त्रुटी आली.",
+ "copyBlockContent": "ब्लॉक सामग्री कॉपी करा"
+},
+"mobilePageSelector": {
+ "title": "पृष्ठ निवडा",
+ "failedToLoad": "पृष्ठ यादी लोड करण्यात अयशस्वी",
+ "noPagesFound": "कोणतीही पृष्ठे सापडली नाहीत"
+},
+"attachmentMenu": {
+ "choosePhoto": "फोटो निवडा",
+ "takePicture": "फोटो काढा",
+ "chooseFile": "फाईल निवडा"
+ }
+ },
+ "board": {
+ "column": {
+ "label": "स्तंभ",
+ "createNewCard": "नवीन",
+ "renameGroupTooltip": "गटाचे नाव बदलण्यासाठी क्लिक करा",
+ "createNewColumn": "नवीन गट जोडा",
+ "addToColumnTopTooltip": "वर नवीन कार्ड जोडा",
+ "addToColumnBottomTooltip": "खाली नवीन कार्ड जोडा",
+ "renameColumn": "स्तंभाचे नाव बदला",
+ "hideColumn": "लपवा",
+ "newGroup": "नवीन गट",
+ "deleteColumn": "हटवा",
+ "deleteColumnConfirmation": "हा गट आणि त्यामधील सर्व कार्ड्स हटवले जातील. तुम्हाला खात्री आहे का?"
+ },
+ "hiddenGroupSection": {
+ "sectionTitle": "लपवलेले गट",
+ "collapseTooltip": "लपवलेले गट लपवा",
+ "expandTooltip": "लपवलेले गट पाहा"
+ },
+ "cardDetail": "कार्ड तपशील",
+ "cardActions": "कार्ड क्रिया",
+ "cardDuplicated": "कार्डची प्रत तयार झाली",
+ "cardDeleted": "कार्ड हटवले गेले",
+ "showOnCard": "कार्ड तपशिलावर दाखवा",
+ "setting": "सेटिंग",
+ "propertyName": "गुणधर्माचे नाव",
+ "menuName": "बोर्ड",
+ "showUngrouped": "गटात नसलेली कार्ड्स दाखवा",
+ "ungroupedButtonText": "गट नसलेली",
+ "ungroupedButtonTooltip": "ज्या कार्ड्स कोणत्याही गटात नाहीत",
+ "ungroupedItemsTitle": "बोर्डमध्ये जोडण्यासाठी क्लिक करा",
+ "groupBy": "या आधारावर गट करा",
+ "groupCondition": "गट स्थिती",
+ "referencedBoardPrefix": "याचे दृश्य",
+ "notesTooltip": "नोट्स आहेत",
+ "mobile": {
+ "editURL": "URL संपादित करा",
+ "showGroup": "गट दाखवा",
+ "showGroupContent": "हा गट बोर्डवर दाखवायचा आहे का?",
+ "failedToLoad": "बोर्ड दृश्य लोड होण्यात अयशस्वी"
+ },
+ "dateCondition": {
+ "weekOf": "{} - {} ची आठवडा",
+ "today": "आज",
+ "yesterday": "काल",
+ "tomorrow": "उद्या",
+ "lastSevenDays": "शेवटचे ७ दिवस",
+ "nextSevenDays": "पुढील ७ दिवस",
+ "lastThirtyDays": "शेवटचे ३० दिवस",
+ "nextThirtyDays": "पुढील ३० दिवस"
+ },
+ "noGroup": "गटासाठी कोणताही गुणधर्म निवडलेला नाही",
+ "noGroupDesc": "बोर्ड दृश्य दाखवण्यासाठी गट करण्याचा एक गुणधर्म आवश्यक आहे",
+ "media": {
+ "cardText": "{} {}",
+ "fallbackName": "फायली"
+ }
+},
+ "calendar": {
+ "menuName": "कॅलेंडर",
+ "defaultNewCalendarTitle": "नाव नाही",
+ "newEventButtonTooltip": "नवीन इव्हेंट जोडा",
+ "navigation": {
+ "today": "आज",
+ "jumpToday": "आजवर जा",
+ "previousMonth": "मागील महिना",
+ "nextMonth": "पुढील महिना",
+ "views": {
+ "day": "दिवस",
+ "week": "आठवडा",
+ "month": "महिना",
+ "year": "वर्ष"
+ }
+ },
+ "mobileEventScreen": {
+ "emptyTitle": "सध्या कोणतेही इव्हेंट नाहीत",
+ "emptyBody": "या दिवशी इव्हेंट तयार करण्यासाठी प्लस बटणावर क्लिक करा."
+ },
+ "settings": {
+ "showWeekNumbers": "आठवड्याचे क्रमांक दाखवा",
+ "showWeekends": "सप्ताहांत दाखवा",
+ "firstDayOfWeek": "आठवड्याची सुरुवात",
+ "layoutDateField": "कॅलेंडर मांडणी दिनांकानुसार",
+ "changeLayoutDateField": "मांडणी फील्ड बदला",
+ "noDateTitle": "तारीख नाही",
+ "noDateHint": {
+ "zero": "नियोजित नसलेली इव्हेंट्स येथे दिसतील",
+ "one": "{count} नियोजित नसलेली इव्हेंट",
+ "other": "{count} नियोजित नसलेल्या इव्हेंट्स"
+ },
+ "unscheduledEventsTitle": "नियोजित नसलेल्या इव्हेंट्स",
+ "clickToAdd": "कॅलेंडरमध्ये जोडण्यासाठी क्लिक करा",
+ "name": "कॅलेंडर सेटिंग्ज",
+ "clickToOpen": "रेकॉर्ड उघडण्यासाठी क्लिक करा"
+ },
+ "referencedCalendarPrefix": "याचे दृश्य",
+ "quickJumpYear": "या वर्षावर जा",
+ "duplicateEvent": "इव्हेंट डुप्लिकेट करा"
+},
+ "errorDialog": {
+ "title": "@:appName त्रुटी",
+ "howToFixFallback": "या गैरसोयीबद्दल आम्ही दिलगीर आहोत! कृपया GitHub पेजवर त्रुटीबद्दल माहिती देणारे एक issue सबमिट करा.",
+ "howToFixFallbackHint1": "या गैरसोयीबद्दल आम्ही दिलगीर आहोत! कृपया ",
+ "howToFixFallbackHint2": " पेजवर त्रुटीचे वर्णन करणारे issue सबमिट करा.",
+ "github": "GitHub वर पहा"
+},
+"search": {
+ "label": "शोध",
+ "sidebarSearchIcon": "पृष्ठ शोधा आणि पटकन जा",
+ "placeholder": {
+ "actions": "कृती शोधा..."
+ }
+},
+"message": {
+ "copy": {
+ "success": "कॉपी झाले!",
+ "fail": "कॉपी करू शकत नाही"
+ }
+},
+"unSupportBlock": "सध्याचा व्हर्जन या ब्लॉकला समर्थन देत नाही.",
+"views": {
+ "deleteContentTitle": "तुम्हाला हे {pageType} हटवायचे आहे का?",
+ "deleteContentCaption": "हे {pageType} हटवल्यास, तुम्ही ते trash मधून पुनर्संचयित करू शकता."
+},
+ "colors": {
+ "custom": "सानुकूल",
+ "default": "डीफॉल्ट",
+ "red": "लाल",
+ "orange": "संत्रा",
+ "yellow": "पिवळा",
+ "green": "हिरवा",
+ "blue": "निळा",
+ "purple": "जांभळा",
+ "pink": "गुलाबी",
+ "brown": "तपकिरी",
+ "gray": "करड्या रंगाचा"
+},
+ "emoji": {
+ "emojiTab": "इमोजी",
+ "search": "इमोजी शोधा",
+ "noRecent": "अलीकडील कोणतेही इमोजी नाहीत",
+ "noEmojiFound": "कोणतेही इमोजी सापडले नाहीत",
+ "filter": "फिल्टर",
+ "random": "योगायोगाने",
+ "selectSkinTone": "त्वचेचा टोन निवडा",
+ "remove": "इमोजी काढा",
+ "categories": {
+ "smileys": "स्मायली आणि भावना",
+ "people": "लोक",
+ "animals": "प्राणी आणि निसर्ग",
+ "food": "अन्न",
+ "activities": "क्रिया",
+ "places": "स्थळे",
+ "objects": "वस्तू",
+ "symbols": "चिन्हे",
+ "flags": "ध्वज",
+ "nature": "निसर्ग",
+ "frequentlyUsed": "नेहमी वापरलेले"
+ },
+ "skinTone": {
+ "default": "डीफॉल्ट",
+ "light": "हलका",
+ "mediumLight": "मध्यम-हलका",
+ "medium": "मध्यम",
+ "mediumDark": "मध्यम-गडद",
+ "dark": "गडद"
+ },
+ "openSourceIconsFrom": "खुल्या स्रोताचे आयकॉन्स"
+},
+ "inlineActions": {
+ "noResults": "निकाल नाही",
+ "recentPages": "अलीकडील पृष्ठे",
+ "pageReference": "पृष्ठ संदर्भ",
+ "docReference": "दस्तऐवज संदर्भ",
+ "boardReference": "बोर्ड संदर्भ",
+ "calReference": "कॅलेंडर संदर्भ",
+ "gridReference": "ग्रिड संदर्भ",
+ "date": "तारीख",
+ "reminder": {
+ "groupTitle": "स्मरणपत्र",
+ "shortKeyword": "remind"
+ },
+ "createPage": "\"{}\" उप-पृष्ठ तयार करा"
+},
+ "datePicker": {
+ "dateTimeFormatTooltip": "सेटिंग्जमध्ये तारीख आणि वेळ फॉरमॅट बदला",
+ "dateFormat": "तारीख फॉरमॅट",
+ "includeTime": "वेळ समाविष्ट करा",
+ "isRange": "शेवटची तारीख",
+ "timeFormat": "वेळ फॉरमॅट",
+ "clearDate": "तारीख साफ करा",
+ "reminderLabel": "स्मरणपत्र",
+ "selectReminder": "स्मरणपत्र निवडा",
+ "reminderOptions": {
+ "none": "काहीही नाही",
+ "atTimeOfEvent": "इव्हेंटच्या वेळी",
+ "fiveMinsBefore": "५ मिनिटे आधी",
+ "tenMinsBefore": "१० मिनिटे आधी",
+ "fifteenMinsBefore": "१५ मिनिटे आधी",
+ "thirtyMinsBefore": "३० मिनिटे आधी",
+ "oneHourBefore": "१ तास आधी",
+ "twoHoursBefore": "२ तास आधी",
+ "onDayOfEvent": "इव्हेंटच्या दिवशी",
+ "oneDayBefore": "१ दिवस आधी",
+ "twoDaysBefore": "२ दिवस आधी",
+ "oneWeekBefore": "१ आठवडा आधी",
+ "custom": "सानुकूल"
+ }
+},
+ "relativeDates": {
+ "yesterday": "काल",
+ "today": "आज",
+ "tomorrow": "उद्या",
+ "oneWeek": "१ आठवडा"
+},
+ "notificationHub": {
+ "title": "सूचना",
+ "mobile": {
+ "title": "अपडेट्स"
+ },
+ "emptyTitle": "सर्व पूर्ण झाले!",
+ "emptyBody": "कोणतीही प्रलंबित सूचना किंवा कृती नाहीत. शांततेचा आनंद घ्या.",
+ "tabs": {
+ "inbox": "इनबॉक्स",
+ "upcoming": "आगामी"
+ },
+ "actions": {
+ "markAllRead": "सर्व वाचलेल्या म्हणून चिन्हित करा",
+ "showAll": "सर्व",
+ "showUnreads": "न वाचलेल्या"
+ },
+ "filters": {
+ "ascending": "आरोही",
+ "descending": "अवरोही",
+ "groupByDate": "तारीखेनुसार गटबद्ध करा",
+ "showUnreadsOnly": "फक्त न वाचलेल्या दाखवा",
+ "resetToDefault": "डीफॉल्टवर रीसेट करा"
+ }
+},
+ "reminderNotification": {
+ "title": "स्मरणपत्र",
+ "message": "तुम्ही विसरण्याआधी हे तपासण्याचे लक्षात ठेवा!",
+ "tooltipDelete": "हटवा",
+ "tooltipMarkRead": "वाचले म्हणून चिन्हित करा",
+ "tooltipMarkUnread": "न वाचले म्हणून चिन्हित करा"
+},
+ "findAndReplace": {
+ "find": "शोधा",
+ "previousMatch": "मागील जुळणारे",
+ "nextMatch": "पुढील जुळणारे",
+ "close": "बंद करा",
+ "replace": "बदला",
+ "replaceAll": "सर्व बदला",
+ "noResult": "कोणतेही निकाल नाहीत",
+ "caseSensitive": "केस सेंसिटिव्ह",
+ "searchMore": "अधिक निकालांसाठी शोधा"
+},
+ "error": {
+ "weAreSorry": "आम्ही क्षमस्व आहोत",
+ "loadingViewError": "हे दृश्य लोड करण्यात अडचण येत आहे. कृपया तुमचे इंटरनेट कनेक्शन तपासा, अॅप रीफ्रेश करा, आणि समस्या कायम असल्यास आमच्याशी संपर्क साधा.",
+ "syncError": "इतर डिव्हाइसमधून डेटा सिंक झाला नाही",
+ "syncErrorHint": "कृपया हे पृष्ठ शेवटचे जिथे संपादित केले होते त्या डिव्हाइसवर उघडा, आणि मग सध्याच्या डिव्हाइसवर पुन्हा उघडा.",
+ "clickToCopy": "एरर कोड कॉपी करण्यासाठी क्लिक करा"
+},
+ "editor": {
+ "bold": "जाड",
+ "bulletedList": "बुलेट यादी",
+ "bulletedListShortForm": "बुलेट",
+ "checkbox": "चेकबॉक्स",
+ "embedCode": "कोड एम्बेड करा",
+ "heading1": "H1",
+ "heading2": "H2",
+ "heading3": "H3",
+ "highlight": "हायलाइट",
+ "color": "रंग",
+ "image": "प्रतिमा",
+ "date": "तारीख",
+ "page": "पृष्ठ",
+ "italic": "तिरका",
+ "link": "लिंक",
+ "numberedList": "क्रमांकित यादी",
+ "numberedListShortForm": "क्रमांकित",
+ "toggleHeading1ShortForm": "Toggle H1",
+ "toggleHeading2ShortForm": "Toggle H2",
+ "toggleHeading3ShortForm": "Toggle H3",
+ "quote": "कोट",
+ "strikethrough": "ओढून टाका",
+ "text": "मजकूर",
+ "underline": "अधोरेखित",
+ "fontColorDefault": "डीफॉल्ट",
+ "fontColorGray": "धूसर",
+ "fontColorBrown": "तपकिरी",
+ "fontColorOrange": "केशरी",
+ "fontColorYellow": "पिवळा",
+ "fontColorGreen": "हिरवा",
+ "fontColorBlue": "निळा",
+ "fontColorPurple": "जांभळा",
+ "fontColorPink": "पिंग",
+ "fontColorRed": "लाल",
+ "backgroundColorDefault": "डीफॉल्ट पार्श्वभूमी",
+ "backgroundColorGray": "धूसर पार्श्वभूमी",
+ "backgroundColorBrown": "तपकिरी पार्श्वभूमी",
+ "backgroundColorOrange": "केशरी पार्श्वभूमी",
+ "backgroundColorYellow": "पिवळी पार्श्वभूमी",
+ "backgroundColorGreen": "हिरवी पार्श्वभूमी",
+ "backgroundColorBlue": "निळी पार्श्वभूमी",
+ "backgroundColorPurple": "जांभळी पार्श्वभूमी",
+ "backgroundColorPink": "पिंग पार्श्वभूमी",
+ "backgroundColorRed": "लाल पार्श्वभूमी",
+ "backgroundColorLime": "लिंबू पार्श्वभूमी",
+ "backgroundColorAqua": "पाण्याचा पार्श्वभूमी",
+ "done": "पूर्ण",
+ "cancel": "रद्द करा",
+ "tint1": "टिंट 1",
+ "tint2": "टिंट 2",
+ "tint3": "टिंट 3",
+ "tint4": "टिंट 4",
+ "tint5": "टिंट 5",
+ "tint6": "टिंट 6",
+ "tint7": "टिंट 7",
+ "tint8": "टिंट 8",
+ "tint9": "टिंट 9",
+ "lightLightTint1": "जांभळा",
+ "lightLightTint2": "पिंग",
+ "lightLightTint3": "फिकट पिंग",
+ "lightLightTint4": "केशरी",
+ "lightLightTint5": "पिवळा",
+ "lightLightTint6": "लिंबू",
+ "lightLightTint7": "हिरवा",
+ "lightLightTint8": "पाणी",
+ "lightLightTint9": "निळा",
+ "urlHint": "URL",
+ "mobileHeading1": "Heading 1",
+ "mobileHeading2": "Heading 2",
+ "mobileHeading3": "Heading 3",
+ "mobileHeading4": "Heading 4",
+ "mobileHeading5": "Heading 5",
+ "mobileHeading6": "Heading 6",
+ "textColor": "मजकूराचा रंग",
+ "backgroundColor": "पार्श्वभूमीचा रंग",
+ "addYourLink": "तुमची लिंक जोडा",
+ "openLink": "लिंक उघडा",
+ "copyLink": "लिंक कॉपी करा",
+ "removeLink": "लिंक काढा",
+ "editLink": "लिंक संपादित करा",
+ "linkText": "मजकूर",
+ "linkTextHint": "कृपया मजकूर प्रविष्ट करा",
+ "linkAddressHint": "कृपया URL प्रविष्ट करा",
+ "highlightColor": "हायलाइट रंग",
+ "clearHighlightColor": "हायलाइट काढा",
+ "customColor": "स्वतःचा रंग",
+ "hexValue": "Hex मूल्य",
+ "opacity": "अपारदर्शकता",
+ "resetToDefaultColor": "डीफॉल्ट रंगावर रीसेट करा",
+ "ltr": "LTR",
+ "rtl": "RTL",
+ "auto": "स्वयंचलित",
+ "cut": "कट",
+ "copy": "कॉपी",
+ "paste": "पेस्ट",
+ "find": "शोधा",
+ "select": "निवडा",
+ "selectAll": "सर्व निवडा",
+ "previousMatch": "मागील जुळणारे",
+ "nextMatch": "पुढील जुळणारे",
+ "closeFind": "बंद करा",
+ "replace": "बदला",
+ "replaceAll": "सर्व बदला",
+ "regex": "Regex",
+ "caseSensitive": "केस सेंसिटिव्ह",
+ "uploadImage": "प्रतिमा अपलोड करा",
+ "urlImage": "URL प्रतिमा",
+ "incorrectLink": "चुकीची लिंक",
+ "upload": "अपलोड",
+ "chooseImage": "प्रतिमा निवडा",
+ "loading": "लोड करत आहे",
+ "imageLoadFailed": "प्रतिमा लोड करण्यात अयशस्वी",
+ "divider": "विभाजक",
+ "table": "तक्त्याचे स्वरूप",
+ "colAddBefore": "यापूर्वी स्तंभ जोडा",
+ "rowAddBefore": "यापूर्वी पंक्ती जोडा",
+ "colAddAfter": "यानंतर स्तंभ जोडा",
+ "rowAddAfter": "यानंतर पंक्ती जोडा",
+ "colRemove": "स्तंभ काढा",
+ "rowRemove": "पंक्ती काढा",
+ "colDuplicate": "स्तंभ डुप्लिकेट",
+ "rowDuplicate": "पंक्ती डुप्लिकेट",
+ "colClear": "सामग्री साफ करा",
+ "rowClear": "सामग्री साफ करा",
+ "slashPlaceHolder": "'/' टाइप करा आणि घटक जोडा किंवा टाइप सुरू करा",
+ "typeSomething": "काहीतरी लिहा...",
+ "toggleListShortForm": "टॉगल",
+ "quoteListShortForm": "कोट",
+ "mathEquationShortForm": "सूत्र",
+ "codeBlockShortForm": "कोड"
+},
+ "favorite": {
+ "noFavorite": "कोणतेही आवडते पृष्ठ नाही",
+ "noFavoriteHintText": "पृष्ठाला डावीकडे स्वाइप करा आणि ते आवडत्या यादीत जोडा",
+ "removeFromSidebar": "साइडबारमधून काढा",
+ "addToSidebar": "साइडबारमध्ये पिन करा"
+},
+"cardDetails": {
+ "notesPlaceholder": "/ टाइप करा ब्लॉक घालण्यासाठी, किंवा टाइप करायला सुरुवात करा"
+},
+"blockPlaceholders": {
+ "todoList": "करण्याची यादी",
+ "bulletList": "यादी",
+ "numberList": "क्रमांकित यादी",
+ "quote": "कोट",
+ "heading": "मथळा {}"
+},
+"titleBar": {
+ "pageIcon": "पृष्ठ चिन्ह",
+ "language": "भाषा",
+ "font": "फॉन्ट",
+ "actions": "क्रिया",
+ "date": "तारीख",
+ "addField": "फील्ड जोडा",
+ "userIcon": "वापरकर्त्याचे चिन्ह"
+},
+"noLogFiles": "कोणतीही लॉग फाइल्स नाहीत",
+"newSettings": {
+ "myAccount": {
+ "title": "माझे खाते",
+ "subtitle": "तुमचा प्रोफाइल सानुकूल करा, खाते सुरक्षा व्यवस्थापित करा, AI कीज पहा किंवा लॉगिन करा.",
+ "profileLabel": "खाते नाव आणि प्रोफाइल चित्र",
+ "profileNamePlaceholder": "तुमचे नाव प्रविष्ट करा",
+ "accountSecurity": "खाते सुरक्षा",
+ "2FA": "2-स्टेप प्रमाणीकरण",
+ "aiKeys": "AI कीज",
+ "accountLogin": "खाते लॉगिन",
+ "updateNameError": "नाव अपडेट करण्यात अयशस्वी",
+ "updateIconError": "चिन्ह अपडेट करण्यात अयशस्वी",
+ "aboutAppFlowy": "@:appName विषयी",
+ "deleteAccount": {
+ "title": "खाते हटवा",
+ "subtitle": "तुमचे खाते आणि सर्व डेटा कायमचे हटवा.",
+ "description": "तुमचे खाते कायमचे हटवले जाईल आणि सर्व वर्कस्पेसमधून प्रवेश काढून टाकला जाईल.",
+ "deleteMyAccount": "माझे खाते हटवा",
+ "dialogTitle": "खाते हटवा",
+ "dialogContent1": "तुम्हाला खात्री आहे की तुम्ही तुमचे खाते कायमचे हटवू इच्छिता?",
+ "dialogContent2": "ही क्रिया पूर्ववत केली जाऊ शकत नाही. हे सर्व वर्कस्पेसमधून प्रवेश हटवेल, खाजगी वर्कस्पेस मिटवेल, आणि सर्व शेअर्ड वर्कस्पेसमधून काढून टाकेल.",
+ "confirmHint1": "कृपया पुष्टीसाठी \"@:newSettings.myAccount.deleteAccount.confirmHint3\" टाइप करा.",
+ "confirmHint2": "मला समजले आहे की ही क्रिया अपरिवर्तनीय आहे आणि माझे खाते व सर्व संबंधित डेटा कायमचा हटवला जाईल.",
+ "confirmHint3": "DELETE MY ACCOUNT",
+ "checkToConfirmError": "हटवण्यासाठी पुष्टी बॉक्स निवडणे आवश्यक आहे",
+ "failedToGetCurrentUser": "वर्तमान वापरकर्त्याचा ईमेल मिळवण्यात अयशस्वी",
+ "confirmTextValidationFailed": "तुमचा पुष्टी मजकूर \"@:newSettings.myAccount.deleteAccount.confirmHint3\" शी जुळत नाही",
+ "deleteAccountSuccess": "खाते यशस्वीरित्या हटवले गेले"
+ }
+ },
+ "workplace": {
+ "name": "वर्कस्पेस",
+ "title": "वर्कस्पेस सेटिंग्स",
+ "subtitle": "तुमचा वर्कस्पेस लुक, थीम, फॉन्ट, मजकूर लेआउट, तारीख, वेळ आणि भाषा सानुकूल करा.",
+ "workplaceName": "वर्कस्पेसचे नाव",
+ "workplaceNamePlaceholder": "वर्कस्पेसचे नाव टाका",
+ "workplaceIcon": "वर्कस्पेस चिन्ह",
+ "workplaceIconSubtitle": "एक प्रतिमा अपलोड करा किंवा इमोजी वापरा. हे साइडबार आणि सूचना मध्ये दर्शवले जाईल.",
+ "renameError": "वर्कस्पेसचे नाव बदलण्यात अयशस्वी",
+ "updateIconError": "चिन्ह अपडेट करण्यात अयशस्वी",
+ "chooseAnIcon": "चिन्ह निवडा",
+ "appearance": {
+ "name": "दृश्यरूप",
+ "themeMode": {
+ "auto": "स्वयंचलित",
+ "light": "प्रकाश मोड",
+ "dark": "गडद मोड"
+ },
+ "language": "भाषा"
+ }
+ },
+ "syncState": {
+ "syncing": "सिंक्रोनायझ करत आहे",
+ "synced": "सिंक्रोनायझ झाले",
+ "noNetworkConnected": "नेटवर्क कनेक्ट केलेले नाही"
+ }
+},
+ "pageStyle": {
+ "title": "पृष्ठ शैली",
+ "layout": "लेआउट",
+ "coverImage": "मुखपृष्ठ प्रतिमा",
+ "pageIcon": "पृष्ठ चिन्ह",
+ "colors": "रंग",
+ "gradient": "ग्रेडियंट",
+ "backgroundImage": "पार्श्वभूमी प्रतिमा",
+ "presets": "पूर्वनियोजित",
+ "photo": "फोटो",
+ "unsplash": "Unsplash",
+ "pageCover": "पृष्ठ कव्हर",
+ "none": "काही नाही",
+ "openSettings": "सेटिंग्स उघडा",
+ "photoPermissionTitle": "@:appName तुमच्या फोटो लायब्ररीमध्ये प्रवेश करू इच्छित आहे",
+ "photoPermissionDescription": "तुमच्या कागदपत्रांमध्ये प्रतिमा जोडण्यासाठी @:appName ला तुमच्या फोटोंमध्ये प्रवेश आवश्यक आहे",
+ "cameraPermissionTitle": "@:appName तुमच्या कॅमेऱ्याला प्रवेश करू इच्छित आहे",
+ "cameraPermissionDescription": "कॅमेऱ्यातून प्रतिमा जोडण्यासाठी @:appName ला तुमच्या कॅमेऱ्याचा प्रवेश आवश्यक आहे",
+ "doNotAllow": "परवानगी देऊ नका",
+ "image": "प्रतिमा"
+},
+"commandPalette": {
+ "placeholder": "शोधा किंवा प्रश्न विचारा...",
+ "bestMatches": "सर्वोत्तम जुळवणी",
+ "recentHistory": "अलीकडील इतिहास",
+ "navigateHint": "नेव्हिगेट करण्यासाठी",
+ "loadingTooltip": "आम्ही निकाल शोधत आहोत...",
+ "betaLabel": "बेटा",
+ "betaTooltip": "सध्या आम्ही फक्त दस्तऐवज आणि पृष्ठ शोध समर्थन करतो",
+ "fromTrashHint": "कचरापेटीतून",
+ "noResultsHint": "आपण जे शोधत आहात ते सापडले नाही, कृपया दुसरा शब्द वापरून शोधा.",
+ "clearSearchTooltip": "शोध फील्ड साफ करा"
+},
+"space": {
+ "delete": "हटवा",
+ "deleteConfirmation": "हटवा: ",
+ "deleteConfirmationDescription": "या स्पेसमधील सर्व पृष्ठे हटवली जातील आणि कचरापेटीत टाकली जातील, आणि प्रकाशित पृष्ठे अनपब्लिश केली जातील.",
+ "rename": "स्पेसचे नाव बदला",
+ "changeIcon": "चिन्ह बदला",
+ "manage": "स्पेस व्यवस्थापित करा",
+ "addNewSpace": "स्पेस तयार करा",
+ "collapseAllSubPages": "सर्व उपपृष्ठे संकुचित करा",
+ "createNewSpace": "नवीन स्पेस तयार करा",
+ "createSpaceDescription": "तुमचे कार्य अधिक चांगल्या प्रकारे आयोजित करण्यासाठी अनेक सार्वजनिक व खाजगी स्पेस तयार करा.",
+ "spaceName": "स्पेसचे नाव",
+ "spaceNamePlaceholder": "उदा. मार्केटिंग, अभियांत्रिकी, HR",
+ "permission": "स्पेस परवानगी",
+ "publicPermission": "सार्वजनिक",
+ "publicPermissionDescription": "पूर्ण प्रवेशासह सर्व वर्कस्पेस सदस्य",
+ "privatePermission": "खाजगी",
+ "privatePermissionDescription": "फक्त तुम्हाला या स्पेसमध्ये प्रवेश आहे",
+ "spaceIconBackground": "पार्श्वभूमीचा रंग",
+ "spaceIcon": "चिन्ह",
+ "dangerZone": "धोकादायक क्षेत्र",
+ "unableToDeleteLastSpace": "शेवटची स्पेस हटवता येणार नाही",
+ "unableToDeleteSpaceNotCreatedByYou": "तुमच्याद्वारे तयार न केलेली स्पेस हटवता येणार नाही",
+ "enableSpacesForYourWorkspace": "तुमच्या वर्कस्पेससाठी स्पेस सक्षम करा",
+ "title": "स्पेसेस",
+ "defaultSpaceName": "सामान्य",
+ "upgradeSpaceTitle": "स्पेस सक्षम करा",
+ "upgradeSpaceDescription": "तुमच्या वर्कस्पेसचे अधिक चांगल्या प्रकारे व्यवस्थापन करण्यासाठी अनेक सार्वजनिक आणि खाजगी स्पेस तयार करा.",
+ "upgrade": "अपग्रेड",
+ "upgradeYourSpace": "अनेक स्पेस तयार करा",
+ "quicklySwitch": "पुढील स्पेसवर पटकन स्विच करा",
+ "duplicate": "स्पेस डुप्लिकेट करा",
+ "movePageToSpace": "पृष्ठ स्पेसमध्ये हलवा",
+ "cannotMovePageToDatabase": "पृष्ठ डेटाबेसमध्ये हलवता येणार नाही",
+ "switchSpace": "स्पेस स्विच करा",
+ "spaceNameCannotBeEmpty": "स्पेसचे नाव रिकामे असू शकत नाही",
+ "success": {
+ "deleteSpace": "स्पेस यशस्वीरित्या हटवली",
+ "renameSpace": "स्पेसचे नाव यशस्वीरित्या बदलले",
+ "duplicateSpace": "स्पेस यशस्वीरित्या डुप्लिकेट केली",
+ "updateSpace": "स्पेस यशस्वीरित्या अपडेट केली"
+ },
+ "error": {
+ "deleteSpace": "स्पेस हटवण्यात अयशस्वी",
+ "renameSpace": "स्पेसचे नाव बदलण्यात अयशस्वी",
+ "duplicateSpace": "स्पेस डुप्लिकेट करण्यात अयशस्वी",
+ "updateSpace": "स्पेस अपडेट करण्यात अयशस्वी"
+ },
+ "createSpace": "स्पेस तयार करा",
+ "manageSpace": "स्पेस व्यवस्थापित करा",
+ "renameSpace": "स्पेसचे नाव बदला",
+ "mSpaceIconColor": "स्पेस चिन्हाचा रंग",
+ "mSpaceIcon": "स्पेस चिन्ह"
+},
+ "publish": {
+ "hasNotBeenPublished": "हे पृष्ठ अजून प्रकाशित केलेले नाही",
+ "spaceHasNotBeenPublished": "स्पेस प्रकाशित करण्यासाठी समर्थन नाही",
+ "reportPage": "पृष्ठाची तक्रार करा",
+ "databaseHasNotBeenPublished": "डेटाबेस प्रकाशित करण्यास समर्थन नाही.",
+ "createdWith": "यांनी तयार केले",
+ "downloadApp": "AppFlowy डाउनलोड करा",
+ "copy": {
+ "codeBlock": "कोड ब्लॉकची सामग्री क्लिपबोर्डवर कॉपी केली गेली आहे",
+ "imageBlock": "प्रतिमा लिंक क्लिपबोर्डवर कॉपी केली गेली आहे",
+ "mathBlock": "गणितीय समीकरण क्लिपबोर्डवर कॉपी केले गेले आहे",
+ "fileBlock": "फाइल लिंक क्लिपबोर्डवर कॉपी केली गेली आहे"
+ },
+ "containsPublishedPage": "या पृष्ठात एक किंवा अधिक प्रकाशित पृष्ठे आहेत. तुम्ही पुढे गेल्यास ती अनपब्लिश होतील. तुम्हाला हटवणे चालू ठेवायचे आहे का?",
+ "publishSuccessfully": "यशस्वीरित्या प्रकाशित झाले",
+ "unpublishSuccessfully": "यशस्वीरित्या अनपब्लिश झाले",
+ "publishFailed": "प्रकाशित करण्यात अयशस्वी",
+ "unpublishFailed": "अनपब्लिश करण्यात अयशस्वी",
+ "noAccessToVisit": "या पृष्ठावर प्रवेश नाही...",
+ "createWithAppFlowy": "AppFlowy ने वेबसाइट तयार करा",
+ "fastWithAI": "AI सह जलद आणि सोपे.",
+ "tryItNow": "आत्ताच वापरून पहा",
+ "onlyGridViewCanBePublished": "फक्त Grid view प्रकाशित केला जाऊ शकतो",
+ "database": {
+ "zero": "{} निवडलेले दृश्य प्रकाशित करा",
+ "one": "{} निवडलेली दृश्ये प्रकाशित करा",
+ "many": "{} निवडलेली दृश्ये प्रकाशित करा",
+ "other": "{} निवडलेली दृश्ये प्रकाशित करा"
+ },
+ "mustSelectPrimaryDatabase": "प्राथमिक दृश्य निवडणे आवश्यक आहे",
+ "noDatabaseSelected": "कोणताही डेटाबेस निवडलेला नाही, कृपया किमान एक डेटाबेस निवडा.",
+ "unableToDeselectPrimaryDatabase": "प्राथमिक डेटाबेस अननिवड करता येणार नाही",
+ "saveThisPage": "या टेम्पलेटपासून सुरू करा",
+ "duplicateTitle": "तुम्हाला हे कुठे जोडायचे आहे",
+ "selectWorkspace": "वर्कस्पेस निवडा",
+ "addTo": "मध्ये जोडा",
+ "duplicateSuccessfully": "तुमच्या वर्कस्पेसमध्ये जोडले गेले",
+ "duplicateSuccessfullyDescription": "AppFlowy स्थापित केले नाही? तुम्ही 'डाउनलोड' वर क्लिक केल्यावर डाउनलोड आपोआप सुरू होईल.",
+ "downloadIt": "डाउनलोड करा",
+ "openApp": "अॅपमध्ये उघडा",
+ "duplicateFailed": "डुप्लिकेट करण्यात अयशस्वी",
+ "membersCount": {
+ "zero": "सदस्य नाहीत",
+ "one": "1 सदस्य",
+ "many": "{count} सदस्य",
+ "other": "{count} सदस्य"
+ },
+ "useThisTemplate": "हा टेम्पलेट वापरा"
+},
+"web": {
+ "continue": "पुढे जा",
+ "or": "किंवा",
+ "continueWithGoogle": "Google सह पुढे जा",
+ "continueWithGithub": "GitHub सह पुढे जा",
+ "continueWithDiscord": "Discord सह पुढे जा",
+ "continueWithApple": "Apple सह पुढे जा",
+ "moreOptions": "अधिक पर्याय",
+ "collapse": "आकुंचन",
+ "signInAgreement": "\"पुढे जा\" क्लिक करून, तुम्ही AppFlowy च्या अटींना सहमती दिली आहे",
+ "signInLocalAgreement": "\"सुरुवात करा\" क्लिक करून, तुम्ही AppFlowy च्या अटींना सहमती दिली आहे",
+ "and": "आणि",
+ "termOfUse": "वापर अटी",
+ "privacyPolicy": "गोपनीयता धोरण",
+ "signInError": "साइन इन त्रुटी",
+ "login": "साइन अप किंवा लॉग इन करा",
+ "fileBlock": {
+ "uploadedAt": "{time} रोजी अपलोड केले",
+ "linkedAt": "{time} रोजी लिंक जोडली",
+ "empty": "फाईल अपलोड करा किंवा एम्बेड करा",
+ "uploadFailed": "अपलोड अयशस्वी, कृपया पुन्हा प्रयत्न करा",
+ "retry": "पुन्हा प्रयत्न करा"
+ },
+ "importNotion": "Notion वरून आयात करा",
+ "import": "आयात करा",
+ "importSuccess": "यशस्वीरित्या अपलोड केले",
+ "importSuccessMessage": "आम्ही तुम्हाला आयात पूर्ण झाल्यावर सूचित करू. त्यानंतर, तुम्ही साइडबारमध्ये तुमची आयात केलेली पृष्ठे पाहू शकता.",
+ "importFailed": "आयात अयशस्वी, कृपया फाईल फॉरमॅट तपासा",
+ "dropNotionFile": "तुमची Notion zip फाईल येथे ड्रॉप करा किंवा ब्राउझ करा",
+ "error": {
+ "pageNameIsEmpty": "पृष्ठाचे नाव रिकामे आहे, कृपया दुसरे नाव वापरून पहा"
+ }
+},
+ "globalComment": {
+ "comments": "टिप्पण्या",
+ "addComment": "टिप्पणी जोडा",
+ "reactedBy": "यांनी प्रतिक्रिया दिली",
+ "addReaction": "प्रतिक्रिया जोडा",
+ "reactedByMore": "आणि {count} इतर",
+ "showSeconds": {
+ "one": "1 सेकंदापूर्वी",
+ "other": "{count} सेकंदांपूर्वी",
+ "zero": "आत्ताच",
+ "many": "{count} सेकंदांपूर्वी"
+ },
+ "showMinutes": {
+ "one": "1 मिनिटापूर्वी",
+ "other": "{count} मिनिटांपूर्वी",
+ "many": "{count} मिनिटांपूर्वी"
+ },
+ "showHours": {
+ "one": "1 तासापूर्वी",
+ "other": "{count} तासांपूर्वी",
+ "many": "{count} तासांपूर्वी"
+ },
+ "showDays": {
+ "one": "1 दिवसापूर्वी",
+ "other": "{count} दिवसांपूर्वी",
+ "many": "{count} दिवसांपूर्वी"
+ },
+ "showMonths": {
+ "one": "1 महिन्यापूर्वी",
+ "other": "{count} महिन्यांपूर्वी",
+ "many": "{count} महिन्यांपूर्वी"
+ },
+ "showYears": {
+ "one": "1 वर्षापूर्वी",
+ "other": "{count} वर्षांपूर्वी",
+ "many": "{count} वर्षांपूर्वी"
+ },
+ "reply": "उत्तर द्या",
+ "deleteComment": "टिप्पणी हटवा",
+ "youAreNotOwner": "तुम्ही या टिप्पणीचे मालक नाही",
+ "confirmDeleteDescription": "तुम्हाला ही टिप्पणी हटवायची आहे याची खात्री आहे का?",
+ "hasBeenDeleted": "हटवले गेले",
+ "replyingTo": "याला उत्तर देत आहे",
+ "noAccessDeleteComment": "तुम्हाला ही टिप्पणी हटवण्याची परवानगी नाही",
+ "collapse": "संकुचित करा",
+ "readMore": "अधिक वाचा",
+ "failedToAddComment": "टिप्पणी जोडण्यात अयशस्वी",
+ "commentAddedSuccessfully": "टिप्पणी यशस्वीरित्या जोडली गेली.",
+ "commentAddedSuccessTip": "तुम्ही नुकतीच एक टिप्पणी जोडली किंवा उत्तर दिले आहे. वर जाऊन ताजी टिप्पण्या पाहायच्या का?"
+},
+ "template": {
+ "asTemplate": "टेम्पलेट म्हणून जतन करा",
+ "name": "टेम्पलेट नाव",
+ "description": "टेम्पलेट वर्णन",
+ "about": "टेम्पलेट माहिती",
+ "deleteFromTemplate": "टेम्पलेटमधून हटवा",
+ "preview": "टेम्पलेट पूर्वदृश्य",
+ "categories": "टेम्पलेट श्रेणी",
+ "isNewTemplate": "नवीन टेम्पलेटमध्ये पिन करा",
+ "featured": "वैशिष्ट्यीकृतमध्ये पिन करा",
+ "relatedTemplates": "संबंधित टेम्पलेट्स",
+ "requiredField": "{field} आवश्यक आहे",
+ "addCategory": "\"{category}\" जोडा",
+ "addNewCategory": "नवीन श्रेणी जोडा",
+ "addNewCreator": "नवीन निर्माता जोडा",
+ "deleteCategory": "श्रेणी हटवा",
+ "editCategory": "श्रेणी संपादित करा",
+ "editCreator": "निर्माता संपादित करा",
+ "category": {
+ "name": "श्रेणीचे नाव",
+ "icon": "श्रेणी चिन्ह",
+ "bgColor": "श्रेणी पार्श्वभूमीचा रंग",
+ "priority": "श्रेणी प्राधान्य",
+ "desc": "श्रेणीचे वर्णन",
+ "type": "श्रेणी प्रकार",
+ "icons": "श्रेणी चिन्हे",
+ "colors": "श्रेणी रंग",
+ "byUseCase": "वापराच्या आधारे",
+ "byFeature": "वैशिष्ट्यांनुसार",
+ "deleteCategory": "श्रेणी हटवा",
+ "deleteCategoryDescription": "तुम्हाला ही श्रेणी हटवायची आहे का?",
+ "typeToSearch": "श्रेणी शोधण्यासाठी टाइप करा..."
+ },
+ "creator": {
+ "label": "टेम्पलेट निर्माता",
+ "name": "निर्मात्याचे नाव",
+ "avatar": "निर्मात्याचा अवतार",
+ "accountLinks": "निर्मात्याचे खाते दुवे",
+ "uploadAvatar": "अवतार अपलोड करण्यासाठी क्लिक करा",
+ "deleteCreator": "निर्माता हटवा",
+ "deleteCreatorDescription": "तुम्हाला हा निर्माता हटवायचा आहे का?",
+ "typeToSearch": "निर्माते शोधण्यासाठी टाइप करा..."
+ },
+ "uploadSuccess": "टेम्पलेट यशस्वीरित्या अपलोड झाले",
+ "uploadSuccessDescription": "तुमचे टेम्पलेट यशस्वीरित्या अपलोड झाले आहे. आता तुम्ही ते टेम्पलेट गॅलरीमध्ये पाहू शकता.",
+ "viewTemplate": "टेम्पलेट पहा",
+ "deleteTemplate": "टेम्पलेट हटवा",
+ "deleteSuccess": "टेम्पलेट यशस्वीरित्या हटवले गेले",
+ "deleteTemplateDescription": "याचा वर्तमान पृष्ठ किंवा प्रकाशित स्थितीवर परिणाम होणार नाही. तुम्हाला हे टेम्पलेट हटवायचे आहे का?",
+ "addRelatedTemplate": "संबंधित टेम्पलेट जोडा",
+ "removeRelatedTemplate": "संबंधित टेम्पलेट हटवा",
+ "uploadAvatar": "अवतार अपलोड करा",
+ "searchInCategory": "{category} मध्ये शोधा",
+ "label": "टेम्पलेट्स"
+},
+ "fileDropzone": {
+ "dropFile": "फाइल अपलोड करण्यासाठी येथे क्लिक करा किंवा ड्रॅग करा",
+ "uploading": "अपलोड करत आहे...",
+ "uploadFailed": "अपलोड अयशस्वी",
+ "uploadSuccess": "अपलोड यशस्वी",
+ "uploadSuccessDescription": "फाइल यशस्वीरित्या अपलोड झाली आहे",
+ "uploadFailedDescription": "फाइल अपलोड अयशस्वी झाली आहे",
+ "uploadingDescription": "फाइल अपलोड होत आहे"
+},
+ "gallery": {
+ "preview": "पूर्ण स्क्रीनमध्ये उघडा",
+ "copy": "कॉपी करा",
+ "download": "डाउनलोड",
+ "prev": "मागील",
+ "next": "पुढील",
+ "resetZoom": "झूम रिसेट करा",
+ "zoomIn": "झूम इन",
+ "zoomOut": "झूम आउट"
+},
+ "invitation": {
+ "join": "सामील व्हा",
+ "on": "वर",
+ "invitedBy": "यांनी आमंत्रित केले",
+ "membersCount": {
+ "zero": "{count} सदस्य",
+ "one": "{count} सदस्य",
+ "many": "{count} सदस्य",
+ "other": "{count} सदस्य"
+ },
+ "tip": "तुम्हाला खालील माहितीच्या आधारे या कार्यक्षेत्रात सामील होण्यासाठी आमंत्रित करण्यात आले आहे. ही माहिती चुकीची असल्यास, कृपया प्रशासकाशी संपर्क साधा.",
+ "joinWorkspace": "वर्कस्पेसमध्ये सामील व्हा",
+ "success": "तुम्ही यशस्वीरित्या वर्कस्पेसमध्ये सामील झाला आहात",
+ "successMessage": "आता तुम्ही सर्व पृष्ठे आणि कार्यक्षेत्रे वापरू शकता.",
+ "openWorkspace": "AppFlowy उघडा",
+ "alreadyAccepted": "तुम्ही आधीच आमंत्रण स्वीकारले आहे",
+ "errorModal": {
+ "title": "काहीतरी चुकले आहे",
+ "description": "तुमचे सध्याचे खाते {email} कदाचित या वर्कस्पेसमध्ये प्रवेशासाठी पात्र नाही. कृपया योग्य खात्याने लॉग इन करा किंवा वर्कस्पेस मालकाशी संपर्क साधा.",
+ "contactOwner": "मालकाशी संपर्क करा",
+ "close": "मुख्यपृष्ठावर परत जा",
+ "changeAccount": "खाते बदला"
+ }
+},
+ "requestAccess": {
+ "title": "या पृष्ठासाठी प्रवेश नाही",
+ "subtitle": "तुम्ही या पृष्ठाच्या मालकाकडून प्रवेशासाठी विनंती करू शकता. मंजुरीनंतर तुम्हाला हे पृष्ठ पाहता येईल.",
+ "requestAccess": "प्रवेशाची विनंती करा",
+ "backToHome": "मुख्यपृष्ठावर परत जा",
+ "tip": "तुम्ही सध्या म्हणून लॉग इन आहात.",
+ "mightBe": "कदाचित तुम्हाला दुसऱ्या खात्याने लॉग इन करणे आवश्यक आहे.",
+ "successful": "विनंती यशस्वीपणे पाठवली गेली",
+ "successfulMessage": "मालकाने मंजुरी दिल्यावर तुम्हाला सूचित केले जाईल.",
+ "requestError": "प्रवेशाची विनंती अयशस्वी",
+ "repeatRequestError": "तुम्ही यासाठी आधीच विनंती केली आहे"
+},
+ "approveAccess": {
+ "title": "वर्कस्पेसमध्ये सामील होण्यासाठी विनंती मंजूर करा",
+ "requestSummary": " यांनी मध्ये सामील होण्यासाठी आणि पाहण्यासाठी विनंती केली आहे",
+ "upgrade": "अपग्रेड",
+ "downloadApp": "AppFlowy डाउनलोड करा",
+ "approveButton": "मंजूर करा",
+ "approveSuccess": "मंजूर यशस्वी",
+ "approveError": "मंजुरी अयशस्वी. कृपया वर्कस्पेस मर्यादा ओलांडलेली नाही याची खात्री करा",
+ "getRequestInfoError": "विनंतीची माहिती मिळवण्यात अयशस्वी",
+ "memberCount": {
+ "zero": "कोणतेही सदस्य नाहीत",
+ "one": "1 सदस्य",
+ "many": "{count} सदस्य",
+ "other": "{count} सदस्य"
+ },
+ "alreadyProTitle": "वर्कस्पेस योजना मर्यादा गाठली आहे",
+ "alreadyProMessage": "अधिक सदस्यांसाठी शी संपर्क साधा",
+ "repeatApproveError": "तुम्ही ही विनंती आधीच मंजूर केली आहे",
+ "ensurePlanLimit": "कृपया खात्री करा की योजना मर्यादा ओलांडलेली नाही. जर ओलांडली असेल तर वर्कस्पेस योजना करा किंवा करा.",
+ "requestToJoin": "मध्ये सामील होण्यासाठी विनंती केली",
+ "asMember": "सदस्य म्हणून"
+},
+ "upgradePlanModal": {
+ "title": "Pro प्लॅनवर अपग्रेड करा",
+ "message": "{name} ने फ्री सदस्य मर्यादा गाठली आहे. अधिक सदस्य आमंत्रित करण्यासाठी Pro प्लॅनवर अपग्रेड करा.",
+ "upgradeSteps": "AppFlowy वर तुमची योजना कशी अपग्रेड करावी:",
+ "step1": "1. सेटिंग्जमध्ये जा",
+ "step2": "2. 'योजना' वर क्लिक करा",
+ "step3": "3. 'योजना बदला' निवडा",
+ "appNote": "नोंद:",
+ "actionButton": "अपग्रेड करा",
+ "downloadLink": "अॅप डाउनलोड करा",
+ "laterButton": "नंतर",
+ "refreshNote": "यशस्वी अपग्रेडनंतर, तुमची नवीन वैशिष्ट्ये सक्रिय करण्यासाठी वर क्लिक करा.",
+ "refresh": "येथे"
+},
+ "breadcrumbs": {
+ "label": "ब्रेडक्रम्स"
+},
+ "time": {
+ "justNow": "आत्ताच",
+ "seconds": {
+ "one": "1 सेकंद",
+ "other": "{count} सेकंद"
+ },
+ "minutes": {
+ "one": "1 मिनिट",
+ "other": "{count} मिनिटे"
+ },
+ "hours": {
+ "one": "1 तास",
+ "other": "{count} तास"
+ },
+ "days": {
+ "one": "1 दिवस",
+ "other": "{count} दिवस"
+ },
+ "weeks": {
+ "one": "1 आठवडा",
+ "other": "{count} आठवडे"
+ },
+ "months": {
+ "one": "1 महिना",
+ "other": "{count} महिने"
+ },
+ "years": {
+ "one": "1 वर्ष",
+ "other": "{count} वर्षे"
+ },
+ "ago": "पूर्वी",
+ "yesterday": "काल",
+ "today": "आज"
+},
+ "members": {
+ "zero": "सदस्य नाहीत",
+ "one": "1 सदस्य",
+ "many": "{count} सदस्य",
+ "other": "{count} सदस्य"
+},
+ "tabMenu": {
+ "close": "बंद करा",
+ "closeDisabledHint": "पिन केलेले टॅब बंद करता येत नाही, कृपया आधी अनपिन करा",
+ "closeOthers": "इतर टॅब बंद करा",
+ "closeOthersHint": "हे सर्व अनपिन केलेले टॅब्स बंद करेल या टॅब वगळता",
+ "closeOthersDisabledHint": "सर्व टॅब पिन केलेले आहेत, बंद करण्यासाठी कोणतेही टॅब सापडले नाहीत",
+ "favorite": "आवडते",
+ "unfavorite": "आवडते काढा",
+ "favoriteDisabledHint": "हे दृश्य आवडते म्हणून जतन करता येत नाही",
+ "pinTab": "पिन करा",
+ "unpinTab": "अनपिन करा"
+},
+ "openFileMessage": {
+ "success": "फाइल यशस्वीरित्या उघडली",
+ "fileNotFound": "फाइल सापडली नाही",
+ "noAppToOpenFile": "ही फाइल उघडण्यासाठी कोणतेही अॅप उपलब्ध नाही",
+ "permissionDenied": "ही फाइल उघडण्यासाठी परवानगी नाही",
+ "unknownError": "फाइल उघडण्यात अयशस्वी"
+},
+ "inviteMember": {
+ "requestInviteMembers": "तुमच्या वर्कस्पेसमध्ये आमंत्रित करा",
+ "inviteFailedMemberLimit": "सदस्य मर्यादा गाठली आहे, कृपया ",
+ "upgrade": "अपग्रेड करा",
+ "addEmail": "email@example.com, email2@example.com...",
+ "requestInvites": "आमंत्रण पाठवा",
+ "inviteAlready": "तुम्ही या ईमेलला आधीच आमंत्रित केले आहे: {email}",
+ "inviteSuccess": "आमंत्रण यशस्वीपणे पाठवले",
+ "description": "खाली ईमेल कॉमा वापरून टाका. शुल्क सदस्य संख्येवर आधारित असते.",
+ "emails": "ईमेल"
+},
+ "quickNote": {
+ "label": "झटपट नोंद",
+ "quickNotes": "झटपट नोंदी",
+ "search": "झटपट नोंदी शोधा",
+ "collapseFullView": "पूर्ण दृश्य लपवा",
+ "expandFullView": "पूर्ण दृश्य उघडा",
+ "createFailed": "झटपट नोंद तयार करण्यात अयशस्वी",
+ "quickNotesEmpty": "कोणत्याही झटपट नोंदी नाहीत",
+ "emptyNote": "रिकामी नोंद",
+ "deleteNotePrompt": "निवडलेली नोंद कायमची हटवली जाईल. तुम्हाला नक्की ती हटवायची आहे का?",
+ "addNote": "नवीन नोंद",
+ "noAdditionalText": "अधिक माहिती नाही"
+},
+ "subscribe": {
+ "upgradePlanTitle": "योजना तुलना करा आणि निवडा",
+ "yearly": "वार्षिक",
+ "save": "{discount}% बचत",
+ "monthly": "मासिक",
+ "priceIn": "किंमत येथे: ",
+ "free": "फ्री",
+ "pro": "प्रो",
+ "freeDescription": "2 सदस्यांपर्यंत वैयक्तिक वापरकर्त्यांसाठी सर्व काही आयोजित करण्यासाठी",
+ "proDescription": "प्रकल्प आणि टीम नॉलेज व्यवस्थापित करण्यासाठी लहान टीमसाठी",
+ "proDuration": {
+ "monthly": "प्रति सदस्य प्रति महिना\nमासिक बिलिंग",
+ "yearly": "प्रति सदस्य प्रति महिना\nवार्षिक बिलिंग"
+ },
+ "cancel": "खालच्या योजनेवर जा",
+ "changePlan": "प्रो योजनेवर अपग्रेड करा",
+ "everythingInFree": "फ्री योजनेतील सर्व काही +",
+ "currentPlan": "सध्याची योजना",
+ "freeDuration": "कायम",
+ "freePoints": {
+ "first": "1 सहकार्यात्मक वर्कस्पेस (2 सदस्यांपर्यंत)",
+ "second": "अमर्यादित पृष्ठे आणि ब्लॉक्स",
+ "three": "5 GB संचयन",
+ "four": "बुद्धिमान शोध",
+ "five": "20 AI प्रतिसाद",
+ "six": "मोबाईल अॅप",
+ "seven": "रिअल-टाइम सहकार्य"
+ },
+ "proPoints": {
+ "first": "अमर्यादित संचयन",
+ "second": "10 वर्कस्पेस सदस्यांपर्यंत",
+ "three": "अमर्यादित AI प्रतिसाद",
+ "four": "अमर्यादित फाइल अपलोड्स",
+ "five": "कस्टम नेमस्पेस"
+ },
+ "cancelPlan": {
+ "title": "आपल्याला जाताना पाहून वाईट वाटते",
+ "success": "आपली सदस्यता यशस्वीरित्या रद्द झाली आहे",
+ "description": "आपल्याला जाताना वाईट वाटते. कृपया आम्हाला सुधारण्यासाठी तुमचे अभिप्राय कळवा. खालील काही प्रश्नांना उत्तर द्या.",
+ "commonOther": "इतर",
+ "otherHint": "आपले उत्तर येथे लिहा",
+ "questionOne": {
+ "question": "तुम्ही AppFlowy Pro सदस्यता का रद्द केली?",
+ "answerOne": "खर्च खूप जास्त आहे",
+ "answerTwo": "वैशिष्ट्ये अपेक्षेनुसार नव्हती",
+ "answerThree": "चांगला पर्याय सापडला",
+ "answerFour": "खर्च योग्य ठरण्यासाठी वापर पुरेसा नव्हता",
+ "answerFive": "सेवा समस्या किंवा तांत्रिक अडचणी"
+ },
+ "questionTwo": {
+ "question": "तुम्ही भविष्यात AppFlowy Pro पुन्हा घेण्याची शक्यता किती आहे?",
+ "answerOne": "खूप शक्यता आहे",
+ "answerTwo": "काहीशी शक्यता आहे",
+ "answerThree": "निश्चित नाही",
+ "answerFour": "अल्प शक्यता आहे",
+ "answerFive": "शक्यता नाही"
+ },
+ "questionThree": {
+ "question": "सदस्यतेदरम्यान कोणते Pro फिचर तुम्हाला सर्वात जास्त उपयोगी वाटले?",
+ "answerOne": "मल्टी-यूजर सहकार्य",
+ "answerTwo": "लांब कालावधीसाठी आवृत्ती इतिहास",
+ "answerThree": "अमर्यादित AI प्रतिसाद",
+ "answerFour": "स्थानिक AI मॉडेल्सचा प्रवेश"
+ },
+ "questionFour": {
+ "question": "AppFlowy बाबत तुमचा एकूण अनुभव कसा होता?",
+ "answerOne": "छान",
+ "answerTwo": "चांगला",
+ "answerThree": "सामान्य",
+ "answerFour": "थोडासा वाईट",
+ "answerFive": "असंतोषजनक"
+ }
+ }
+},
+ "ai": {
+ "contentPolicyViolation": "संवेदनशील सामग्रीमुळे प्रतिमा निर्मिती अयशस्वी झाली. कृपया तुमचे इनपुट पुन्हा लिहा आणि पुन्हा प्रयत्न करा.",
+ "textLimitReachedDescription": "तुमच्या वर्कस्पेसमध्ये मोफत AI प्रतिसाद संपले आहेत. अनलिमिटेड प्रतिसादांसाठी प्रो योजना घेण्याचा किंवा AI अॅड-ऑन खरेदी करण्याचा विचार करा.",
+ "imageLimitReachedDescription": "तुमचे मोफत AI प्रतिमा कोटा संपले आहे. कृपया प्रो योजना घ्या किंवा AI अॅड-ऑन खरेदी करा.",
+ "limitReachedAction": {
+ "textDescription": "तुमच्या वर्कस्पेसमध्ये मोफत AI प्रतिसाद संपले आहेत. अधिक प्रतिसाद मिळवण्यासाठी कृपया",
+ "imageDescription": "तुमचे मोफत AI प्रतिमा कोटा संपले आहे. कृपया",
+ "upgrade": "अपग्रेड करा",
+ "toThe": "या योजनेवर",
+ "proPlan": "प्रो योजना",
+ "orPurchaseAn": "किंवा खरेदी करा",
+ "aiAddon": "AI अॅड-ऑन"
+ },
+ "editing": "संपादन करत आहे",
+ "analyzing": "विश्लेषण करत आहे",
+ "continueWritingEmptyDocumentTitle": "लेखन सुरू ठेवता आले नाही",
+ "continueWritingEmptyDocumentDescription": "तुमच्या दस्तऐवजातील मजकूर वाढवण्यात अडचण येत आहे. कृपया एक छोटं परिचय लिहा, मग आम्ही पुढे नेऊ!",
+ "more": "अधिक"
+},
+ "autoUpdate": {
+ "criticalUpdateTitle": "अद्यतन आवश्यक आहे",
+ "criticalUpdateDescription": "तुमचा अनुभव सुधारण्यासाठी आम्ही सुधारणा केल्या आहेत! कृपया {currentVersion} वरून {newVersion} वर अद्यतन करा.",
+ "criticalUpdateButton": "अद्यतन करा",
+ "bannerUpdateTitle": "नवीन आवृत्ती उपलब्ध!",
+ "bannerUpdateDescription": "नवीन वैशिष्ट्ये आणि सुधारणांसाठी. अद्यतनासाठी 'Update' क्लिक करा.",
+ "bannerUpdateButton": "अद्यतन करा",
+ "settingsUpdateTitle": "नवीन आवृत्ती ({newVersion}) उपलब्ध!",
+ "settingsUpdateDescription": "सध्याची आवृत्ती: {currentVersion} (अधिकृत बिल्ड) → {newVersion}",
+ "settingsUpdateButton": "अद्यतन करा",
+ "settingsUpdateWhatsNew": "काय नवीन आहे"
+},
+ "lockPage": {
+ "lockPage": "लॉक केलेले",
+ "reLockPage": "पुन्हा लॉक करा",
+ "lockTooltip": "अनवधानाने संपादन टाळण्यासाठी पृष्ठ लॉक केले आहे. अनलॉक करण्यासाठी क्लिक करा.",
+ "pageLockedToast": "पृष्ठ लॉक केले आहे. कोणी अनलॉक करेपर्यंत संपादन अक्षम आहे.",
+ "lockedOperationTooltip": "पृष्ठ अनवधानाने संपादनापासून वाचवण्यासाठी लॉक केले आहे."
+},
+ "suggestion": {
+ "accept": "स्वीकारा",
+ "keep": "जसे आहे तसे ठेवा",
+ "discard": "रद्द करा",
+ "close": "बंद करा",
+ "tryAgain": "पुन्हा प्रयत्न करा",
+ "rewrite": "पुन्हा लिहा",
+ "insertBelow": "खाली टाका"
+}
+}
diff --git a/frontend/appflowy_flutter/distribute_options.yaml b/frontend/appflowy_flutter/distribute_options.yaml
new file mode 100644
index 0000000000..60f603a938
--- /dev/null
+++ b/frontend/appflowy_flutter/distribute_options.yaml
@@ -0,0 +1,12 @@
+output: dist/
+releases:
+ - name: dev
+ jobs:
+ - name: release-dev-linux-deb
+ package:
+ platform: linux
+ target: deb
+ - name: release-dev-linux-rpm
+ package:
+ platform: linux
+ target: rpm
diff --git a/frontend/appflowy_flutter/dsa_pub.pem b/frontend/appflowy_flutter/dsa_pub.pem
new file mode 100644
index 0000000000..6a9d213b8a
--- /dev/null
+++ b/frontend/appflowy_flutter/dsa_pub.pem
@@ -0,0 +1,36 @@
+-----BEGIN PUBLIC KEY-----
+MIIGQzCCBDUGByqGSM44BAEwggQoAoICAQDlkozRmUnVH1MJFqOamAmUYu0YruaT
+rrt6rCIZ0LFrfNnmHA4LOQEcXwBTTyn5sBmkPq+lb/rjmERKhmvl1rfo6q7tJ8mG
+4TWqSu0tOJQ6QxexnNW4yhzK/r9MS5MQus4Al+y2hQLaAMOUIOnaWIrC9OHy7xyw
++sVipECVKyQqipS4shGUSqbcN+ocQuTB+I0MtIjBii0DGSEY3pxQrfNWjHFL7iTV
+KiTn3YOWPJQvv3FvEDrN+5xU5JZpD97ZhXaJpLUyOQaNvcPaOELPWcOSJwqHOpf5
+b5N/VZ8SGbHNdxy9d5sSChBgtuAOihEhSp6SjFQ9eVHOf4NyJwSEMmi0gpdpqm4Z
+QRJUnM2zIi0p9twR9FRYXzrxOs6yGCQEY+xFG93ShTLTj3zMrIyFqBsqEwFyJiJW
+YWe/zp0V7UlLP+jXO9u9eghNmly7QVqD2P0qs/1V0jZFRuLWpsv4inau/qMZ5EhG
+G4xCJZXfN1pkehy6e05/h+vs5anK3Wa/H8AtY6cK4CpzAanELvn3AH7VLbAhLswu
+6d5CV+DoFgxCWMzGBSdmCYU+2wRLaL8Q9TZHDR+pvQlunEFdfFoGES9WjBPhAsVA
+6Mq22U8XSje9yHI3X9Eqe/7a+ajSgcGmB7oQ11+4xf5h2PtubRW/JL0KMjxCxMTp
+q1md6Ndx/ptBUwIdAIOyiKb2YcTLWAOt+LAlRXMsY1+W4pTXJfV6RcMCggIAPxbd
+0HNj2O/aQhJxNZDMBIcx6+cZ+LKch7qLcaEpVqWHvDSnR2eOJJzWn0RoKK+Vuix/
+4T8texSQkWxAeFFdo6kyrR9XNL7hqEFFq8o9VpmvRzvG6h/bBgh3AHAQE3p/8Wrb
+K13IhnlWqd0MjFufSphm63o0gaWl95j+6KeUoKQnioetu9HiMtFKx0d/KYqTQJg7
+hvR6VNCU2oShfXR3ce7RnUYwD37+djrUjUkoAZkZq2KoxBiKyeoSIeqAme19tKcO
+s6b17mhALELuJ+NtDwlDunyiCDUYX9lTPijHwKeIFtBs38+OtRk3aIqmWTQdbsCz
+Axp+kUMA5ESBME/RBNCSPHuDvtA3wfWvNbA5DXfZLwCgNSxhekq8XntIsRzfJ4v4
+uPzKFcVM3+sUUfSF04HHC9ol+PpLqXUyMnskiizqxFPq7H+6tyFZ7X2HiG6TjcfV
+Wthmv+JyfcABjVnk2qFH7GagENbdtYmfUox13LhE59Sh5chaJnCFtCDp8NClWgZn
+ixCOFQ9EgTLaH6MovTvWpEgG2MfBCu5SMUHi2qSflorqpRFH+rA7NZSnyz3wm7NB
++fJSOP0IjEkOh7MafU6Z61oK9WY/Fc+F1zIENVv8PUc3p75y/4RAp4xzyKcTilaN
+C9U/3MRr3QmWwY7ejtZx6xdOxsvWBRDRSNbDdIkDggIGAAKCAgEAt1DHYZoeXY0r
+vYXmxdNO6zfnbz1GGZHXpakzm9h4BrxPDP5J8DQ9ZeVVKg5+cU9AyMO3cZHp7wkx
+k6IB+ZDUpqO1D3lWriRl2fI8cS4edI0fzpnW1nyhhFD4MbKmP+v27aH+DhZ4Up3y
+GMmJTLmKiYx1EgZp7Sx77PBYDVMsKKd3h9+Hjp2YtUTfD2lleAmC+wcQGZiNtGw/
+eKpsmUVnWrepOdntWTtCQi1OvfcHaF2QmgktCq+68hbDNYWaXmzVIiQqrdv/zzOG
+hCFIrRGWemrxL0iFG4Pzc4UfOINsISQLcUxRuF6pQWPxF8O/mWKfzAeqWxmIujUM
+EoSEuI3yQ8VjlYpW/8FSK7UhgnHBHOpCJWPWs/vQXAnaUR2PYyzuIzhVEhFs8YA8
+iBIKnixIC2hu0YbEk3TBr/TRcbd7mDw9Mq7NT88xzdU13+Wh+4zhdX3rtBHYzBtI
+7GaONGUNyY4h0duoyLpH6dxevaeKN6/bEdzYESjoE58QA88CpnAZGhJVphAba4cb
+w6GTDhK3RlPWh6hRqJwLDILGtnJS3UKeBDRmKMqNuqmHqPjyAAvt9JBO8lzjoLgf
+1cDsXHNWBVwA2jsX2CukNJPlY1Fa3MWhdaUXmy6QGMSisr1sptvBt1Phry8T2u+P
+Y29SB4jvwqls268rP0cWqy4WXwlVwuc=
+-----END PUBLIC KEY-----
diff --git a/frontend/appflowy_flutter/integration_test/desktop/board/board_hide_groups_test.dart b/frontend/appflowy_flutter/integration_test/desktop/board/board_hide_groups_test.dart
index 15da47f0f1..6a012ac763 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/board/board_hide_groups_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/board/board_hide_groups_test.dart
@@ -23,24 +23,24 @@ void main() {
final expandFinder = find.byFlowySvg(FlowySvgs.hamburger_s_s);
// Is expanded by default
- expect(collapseFinder, findsOneWidget);
- expect(expandFinder, findsNothing);
-
- // Collapse hidden groups
- await tester.tap(collapseFinder);
- await tester.pumpAndSettle();
-
- // Is collapsed
expect(collapseFinder, findsNothing);
expect(expandFinder, findsOneWidget);
- // Expand hidden groups
+ // Collapse hidden groups
await tester.tap(expandFinder);
await tester.pumpAndSettle();
- // Is expanded
+ // Is collapsed
expect(collapseFinder, findsOneWidget);
expect(expandFinder, findsNothing);
+
+ // Expand hidden groups
+ await tester.tap(collapseFinder);
+ await tester.pumpAndSettle();
+
+ // Is expanded
+ expect(collapseFinder, findsNothing);
+ expect(expandFinder, findsOneWidget);
});
testWidgets('hide first group, and show it again', (tester) async {
@@ -48,6 +48,9 @@ void main() {
await tester.tapAnonymousSignInButton();
await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Board);
+ final expandFinder = find.byFlowySvg(FlowySvgs.hamburger_s_s);
+ await tester.tapButton(expandFinder);
+
// Tap the options of the first group
final optionsFinder = find
.descendant(
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/cloud_runner.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/cloud_runner.dart
index 621baff5d0..a8c05d5f80 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/cloud_runner.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/cloud_runner.dart
@@ -1,16 +1,21 @@
+import 'data_migration/data_migration_test_runner.dart'
+ as data_migration_test_runner;
+import 'database/database_test_runner.dart' as database_test_runner;
import 'document/document_test_runner.dart' as document_test_runner;
+import 'set_env.dart' as preset_af_cloud_env_test;
+import 'sidebar/sidebar_icon_test.dart' as sidebar_icon_test;
import 'sidebar/sidebar_move_page_test.dart' as sidebar_move_page_test;
+import 'sidebar/sidebar_rename_untitled_test.dart'
+ as sidebar_rename_untitled_test;
import 'uncategorized/uncategorized_test_runner.dart'
as uncategorized_test_runner;
import 'workspace/workspace_test_runner.dart' as workspace_test_runner;
-import 'data_migration/data_migration_test_runner.dart'
- as data_migration_test_runner;
-import 'set_env.dart' as preset_af_cloud_env_test;
Future main() async {
preset_af_cloud_env_test.main();
data_migration_test_runner.main();
+
// uncategorized
uncategorized_test_runner.main();
@@ -22,4 +27,9 @@ Future main() async {
// sidebar
sidebar_move_page_test.main();
+ sidebar_rename_untitled_test.main();
+ sidebar_icon_test.main();
+
+ // database
+ database_test_runner.main();
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/data_migration/anon_user_data_migration_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/data_migration/anon_user_data_migration_test.dart
index 054e895b4a..e34ac02aab 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/data_migration/anon_user_data_migration_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/data_migration/anon_user_data_migration_test.dart
@@ -1,33 +1,10 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-import 'dart:ui';
-
import 'package:appflowy/env/cloud_env.dart';
-import 'package:appflowy/generated/flowy_svgs.g.dart';
-import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
import 'package:appflowy/workspace/application/settings/prelude.dart';
import 'package:appflowy/workspace/presentation/settings/pages/account/account.dart';
-import 'package:appflowy/workspace/presentation/settings/pages/settings_account_view.dart';
-import 'package:appflowy/workspace/presentation/settings/shared/settings_body.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
-import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
-import 'package:flowy_infra_ui/style_widget/text_field.dart';
-import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:intl/intl.dart';
-import 'package:path/path.dart' as p;
-import '../../../shared/dir.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
-import '../../board/board_hide_groups_test.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
@@ -38,7 +15,6 @@ void main() {
cloudType: AuthenticatorType.appflowyCloudSelfHost,
);
- await tester.tapContinousAnotherWay();
await tester.tapAnonymousSignInButton();
await tester.expectToSeeHomePageWithGetStartedPage();
@@ -54,12 +30,6 @@ void main() {
await tester.enterUserName('local_user');
// Scroll to sign-in
- await tester.scrollUntilVisible(
- find.byType(AccountSignInOutButton),
- 100,
- scrollable: find.findSettingsScrollable(),
- );
-
await tester.tapButton(find.byType(AccountSignInOutButton));
// sign up with Google
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/database/database_image_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/database/database_image_test.dart
new file mode 100644
index 0000000000..5561d40033
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/database/database_image_test.dart
@@ -0,0 +1,80 @@
+import 'dart:io';
+
+import 'package:appflowy/core/config/kv.dart';
+import 'package:appflowy/core/config/kv_keys.dart';
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/image/resizeable_image.dart';
+import 'package:appflowy/startup/startup.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/view.pbenum.dart';
+import 'package:appflowy_editor/appflowy_editor.dart'
+ hide UploadImageMenu, ResizableImage;
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/services.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+import 'package:path/path.dart' as p;
+import 'package:path_provider/path_provider.dart';
+
+import '../../../shared/constants.dart';
+import '../../../shared/database_test_op.dart';
+import '../../../shared/mock/mock_file_picker.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ // copy link to block
+ group('database image:', () {
+ testWidgets('insert image', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // open the first row detail page and upload an image
+ await tester.createNewPageInSpace(
+ spaceName: Constants.generalSpaceName,
+ layout: ViewLayoutPB.Grid,
+ pageName: 'database image',
+ );
+ await tester.openFirstRowDetailPage();
+
+ // insert an image block
+ {
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.editor.showSlashMenu();
+ await tester.editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_name_image.tr(),
+ );
+ }
+
+ // upload an image
+ {
+ final image = await rootBundle.load('assets/test/images/sample.jpeg');
+ final tempDirectory = await getTemporaryDirectory();
+ final imagePath = p.join(tempDirectory.path, 'sample.jpeg');
+ final file = File(imagePath)
+ ..writeAsBytesSync(image.buffer.asUint8List());
+
+ mockPickFilePaths(
+ paths: [imagePath],
+ );
+
+ await getIt().set(KVKeys.kCloudType, '0');
+ await tester.tapButtonWithName(
+ LocaleKeys.document_imageBlock_upload_placeholder.tr(),
+ );
+ await tester.pumpAndSettle();
+ expect(find.byType(ResizableImage), findsOneWidget);
+ final node = tester.editor.getCurrentEditorState().getNodeAtPath([0])!;
+ expect(node.type, ImageBlockKeys.type);
+ expect(node.attributes[ImageBlockKeys.url], isNotEmpty);
+
+ // remove the temp file
+ file.deleteSync();
+ }
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/database/database_test_runner.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/database/database_test_runner.dart
new file mode 100644
index 0000000000..4d1a623f07
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/database/database_test_runner.dart
@@ -0,0 +1,9 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'database_image_test.dart' as database_image_test;
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ database_image_test.main();
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_ai_writer_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_ai_writer_test.dart
new file mode 100644
index 0000000000..f163608ccb
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_ai_writer_test.dart
@@ -0,0 +1,47 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../../shared/constants.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('AI Writer:', () {
+ testWidgets('the ai writer transaction should only apply in memory',
+ (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ const pageName = 'Document';
+ await tester.createNewPageInSpace(
+ spaceName: Constants.generalSpaceName,
+ layout: ViewLayoutPB.Document,
+ pageName: pageName,
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.editor.showSlashMenu();
+ await tester.editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_name_aiWriter.tr(),
+ );
+ expect(find.byType(AiWriterBlockComponent), findsOneWidget);
+
+ // switch to another page
+ await tester.openPage(Constants.gettingStartedPageName);
+ // switch back to the page
+ await tester.openPage(pageName);
+
+ // expect the ai writer block is not in the document
+ expect(find.byType(AiWriterBlockComponent), findsNothing);
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_option_actions_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_option_actions_test.dart
index 0289fbe176..1bc9bd8f92 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_option_actions_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_option_actions_test.dart
@@ -57,7 +57,7 @@ void main() {
// move the checkbox to the child of the block at path [9]
await tester.editor.dragBlock(
[10],
- const Offset(80, -30),
+ const Offset(120, -20),
);
// wait for the move animation to complete
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_publish_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_publish_test.dart
index 0aa308ea53..7877143116 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_publish_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_publish_test.dart
@@ -1,33 +1,17 @@
-// ignore_for_file: unused_import
-
import 'package:appflowy/env/cloud_env.dart';
import 'package:appflowy/generated/flowy_svgs.g.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/copy_and_paste/clipboard_service.dart';
-import 'package:appflowy/plugins/shared/share/constants.dart';
import 'package:appflowy/plugins/shared/share/publish_tab.dart';
import 'package:appflowy/plugins/shared/share/share_menu.dart';
-import 'package:appflowy/shared/feature_flags.dart';
-import 'package:appflowy/shared/patterns/common_patterns.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_icon.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
import '../../../shared/constants.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
-import '../../../shared/workspace.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
@@ -122,9 +106,21 @@ void main() {
await tester.pumpUntilFound(errorToast2);
await tester.pumpUntilNotFound(errorToast2);
+ // rename with empty name
await tester.tap(inputField);
+ await tester.enterText(inputField, '');
+ await tester.tapButton(find.text(LocaleKeys.button_save.tr()));
+ await tester.pumpAndSettle();
+
+ // expect to see the toast with error message
+ final errorToast3 = find.text(
+ LocaleKeys.settings_sites_error_publishNameCannotBeEmpty.tr(),
+ );
+ await tester.pumpUntilFound(errorToast3);
+ await tester.pumpUntilNotFound(errorToast3);
// input the new path name
+ await tester.tap(inputField);
await tester.enterText(inputField, 'new-path-name');
// click save button
await tester.tapButton(find.text(LocaleKeys.button_save.tr()));
@@ -153,5 +149,72 @@ void main() {
isTrue,
);
});
+
+ testWidgets('re-publish the document', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ const pageName = 'Document';
+
+ await tester.createNewPageInSpace(
+ spaceName: Constants.generalSpaceName,
+ layout: ViewLayoutPB.Document,
+ pageName: pageName,
+ );
+
+ // open the publish menu
+ await tester.openPublishMenu();
+
+ // publish the document
+ final publishButton = find.byType(PublishButton);
+ await tester.tapButton(publishButton);
+
+ // rename the path name
+ final inputField = find.descendant(
+ of: find.byType(ShareMenu),
+ matching: find.byType(TextField),
+ );
+
+ // input the new path name
+ const newName = 'new-path-name';
+ await tester.enterText(inputField, newName);
+ // click save button
+ await tester.tapButton(find.text(LocaleKeys.button_save.tr()));
+ await tester.pumpAndSettle();
+
+ // expect to see the toast with success message
+ final successToast = find.text(
+ LocaleKeys.settings_sites_success_updatePathNameSuccess.tr(),
+ );
+ await tester.pumpUntilNotFound(successToast);
+
+ // unpublish the document
+ final unpublishButton = find.byType(UnPublishButton);
+ await tester.tapButton(unpublishButton);
+
+ final unpublishSuccessToast = find.text(
+ LocaleKeys.publish_unpublishSuccessfully.tr(),
+ );
+ await tester.pumpUntilNotFound(unpublishSuccessToast);
+
+ // re-publish the document
+ await tester.tapButton(publishButton);
+
+ // expect to see the toast with success message
+ final rePublishSuccessToast = find.text(
+ LocaleKeys.publish_publishSuccessfully.tr(),
+ );
+ await tester.pumpUntilNotFound(rePublishSuccessToast);
+
+ // check the clipboard has the link
+ final content = await Clipboard.getData(Clipboard.kTextPlain);
+ expect(
+ content?.text?.contains(newName),
+ isTrue,
+ );
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_test_runner.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_test_runner.dart
index 88d7db8a71..58a9d7398b 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_test_runner.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_test_runner.dart
@@ -1,5 +1,6 @@
import 'package:integration_test/integration_test.dart';
+import 'document_ai_writer_test.dart' as document_ai_writer_test;
import 'document_copy_link_to_block_test.dart'
as document_copy_link_to_block_test;
import 'document_option_actions_test.dart' as document_option_actions_test;
@@ -11,4 +12,5 @@ void main() {
document_option_actions_test.main();
document_copy_link_to_block_test.main();
document_publish_test.main();
+ document_ai_writer_test.main();
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/sidebar/sidebar_icon_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/sidebar/sidebar_icon_test.dart
new file mode 100644
index 0000000000..5bcca50153
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/sidebar/sidebar_icon_test.dart
@@ -0,0 +1,62 @@
+import 'dart:convert';
+
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
+import 'package:appflowy/workspace/presentation/home/menu/sidebar/space/sidebar_space_header.dart';
+import 'package:appflowy/workspace/presentation/home/menu/sidebar/space/space_action_type.dart';
+import 'package:appflowy/workspace/presentation/home/menu/sidebar/space/space_more_popup.dart';
+import 'package:flowy_svg/flowy_svg.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../../shared/emoji.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
+
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
+
+ testWidgets('Change slide bar space icon', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+ final emojiIconData = await tester.loadIcon();
+ final firstIcon = IconsData.fromJson(jsonDecode(emojiIconData.emoji));
+
+ await tester.hoverOnWidget(
+ find.byType(SidebarSpaceHeader),
+ onHover: () async {
+ final moreOption = find.byType(SpaceMorePopup);
+ await tester.tapButton(moreOption);
+ expect(find.byType(FlowyIconEmojiPicker), findsNothing);
+ await tester.tapSvgButton(SpaceMoreActionType.changeIcon.leftIconSvg);
+ expect(find.byType(FlowyIconEmojiPicker), findsOneWidget);
+ },
+ );
+
+ final icons = find.byWidgetPredicate(
+ (w) => w is FlowySvg && w.svgString == firstIcon.svgString,
+ );
+ expect(icons, findsOneWidget);
+ await tester.tapIcon(EmojiIconData.icon(firstIcon));
+
+ final spaceHeader = find.byType(SidebarSpaceHeader);
+ final spaceIcon = find.descendant(
+ of: spaceHeader,
+ matching: find.byWidgetPredicate(
+ (w) => w is FlowySvg && w.svgString == firstIcon.svgString,
+ ),
+ );
+ expect(spaceIcon, findsOneWidget);
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/sidebar/sidebar_move_page_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/sidebar/sidebar_move_page_test.dart
index 4179c2afb7..37abd19ebc 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/sidebar/sidebar_move_page_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/sidebar/sidebar_move_page_test.dart
@@ -1,35 +1,13 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
import 'package:appflowy/env/cloud_env.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/openai/widgets/loading.dart';
-import 'package:appflowy/shared/feature_flags.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_actions.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_menu.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/sidebar_workspace.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy/workspace/presentation/widgets/user_avatar.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
import 'package:flutter/gestures.dart';
-import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
import 'package:universal_platform/universal_platform.dart';
import '../../../shared/constants.dart';
-import '../../../shared/database_test_op.dart';
-import '../../../shared/dir.dart';
-import '../../../shared/emoji.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
void main() {
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/sidebar/sidebar_rename_untitled_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/sidebar/sidebar_rename_untitled_test.dart
new file mode 100644
index 0000000000..8226b68b26
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/sidebar/sidebar_rename_untitled_test.dart
@@ -0,0 +1,55 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/workspace/application/view/view_ext.dart';
+import 'package:appflowy/workspace/presentation/widgets/dialogs.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/view.pbenum.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flowy_infra_ui/style_widget/text_input.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../../shared/constants.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ testWidgets('Rename empty name view (untitled)', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ await tester.createNewPageInSpace(
+ spaceName: Constants.generalSpaceName,
+ layout: ViewLayoutPB.Document,
+ );
+
+ // click the ... button and open rename dialog
+ await tester.hoverOnPageName(
+ ViewLayoutPB.Document.defaultName,
+ onHover: () async {
+ await tester.tapPageOptionButton();
+ await tester.tapButtonWithName(
+ LocaleKeys.disclosureAction_rename.tr(),
+ );
+ },
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(NavigatorTextFieldDialog), findsOneWidget);
+
+ final textField = tester.widget(
+ find.descendant(
+ of: find.byType(NavigatorTextFieldDialog),
+ matching: find.byType(FlowyFormTextInput),
+ ),
+ );
+
+ expect(
+ textField.controller!.text,
+ LocaleKeys.menuAppHeader_defaultNewPageName.tr(),
+ );
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/appflowy_cloud_auth_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/appflowy_cloud_auth_test.dart
index 36475e7c28..fd65c29927 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/appflowy_cloud_auth_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/appflowy_cloud_auth_test.dart
@@ -1,21 +1,12 @@
-// ignore_for_file: unused_import
-
import 'package:appflowy/env/cloud_env.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
import 'package:appflowy/workspace/application/settings/prelude.dart';
import 'package:appflowy/workspace/presentation/settings/pages/account/account.dart';
-import 'package:appflowy/workspace/presentation/settings/pages/settings_account_view.dart';
import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
void main() {
@@ -66,12 +57,6 @@ void main() {
await tester.openSettings();
await tester.openSettingsPage(SettingsPage.account);
- // Scroll to sign-in
- await tester.scrollUntilVisible(
- find.byType(AccountSignInOutButton),
- 100,
- scrollable: find.findSettingsScrollable(),
- );
await tester.tapButton(find.byType(AccountSignInOutButton));
tester.expectToSeeGoogleLoginButton();
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/document_sync_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/document_sync_test.dart
index b2c525c384..b6b4ecf025 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/document_sync_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/document_sync_test.dart
@@ -1,24 +1,9 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
import 'package:appflowy/env/cloud_env.dart';
-import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/presentation/settings/pages/settings_account_view.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
-import 'package:easy_localization/easy_localization.dart';
import 'package:flowy_infra/uuid.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
-import '../../../shared/dir.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
void main() {
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/user_setting_sync_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/user_setting_sync_test.dart
index 780748e6fa..e666289bf5 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/user_setting_sync_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/user_setting_sync_test.dart
@@ -1,33 +1,11 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
import 'package:appflowy/env/cloud_env.dart';
-import 'package:appflowy/generated/flowy_svgs.g.dart';
-import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
import 'package:appflowy/workspace/application/settings/prelude.dart';
import 'package:appflowy/workspace/presentation/settings/pages/account/account_user_profile.dart';
-import 'package:appflowy/workspace/presentation/settings/pages/settings_account_view.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy/workspace/presentation/widgets/user_avatar.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
-import 'package:easy_localization/easy_localization.dart';
import 'package:flowy_infra/uuid.dart';
-import 'package:flowy_infra_ui/style_widget/text_field.dart';
-import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
-import '../../../shared/database_test_op.dart';
-import '../../../shared/dir.dart';
-import '../../../shared/emoji.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
-import '../../board/board_hide_groups_test.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/change_name_and_icon_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/change_name_and_icon_test.dart
index 30bc9e3830..f205b35354 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/change_name_and_icon_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/change_name_and_icon_test.dart
@@ -1,21 +1,12 @@
-// ignore_for_file: unused_import
-
import 'package:appflowy/env/cloud_env.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/shared/feature_flags.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_icon.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flowy_infra/uuid.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
import '../../../shared/workspace.dart';
@@ -49,6 +40,10 @@ void main() {
await tester.changeWorkspaceIcon(icon);
await tester.changeWorkspaceName(name);
+ await tester.pumpUntilNotFound(
+ find.text(LocaleKeys.workspace_renameSuccess.tr()),
+ );
+
workspaceIcon = tester.widget(
find.byType(WorkspaceIcon),
);
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/collaborative_workspace_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/collaborative_workspace_test.dart
index 903dc95fcc..4d2e027646 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/collaborative_workspace_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/collaborative_workspace_test.dart
@@ -1,38 +1,20 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
import 'package:appflowy/env/cloud_env.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/openai/widgets/loading.dart';
import 'package:appflowy/shared/feature_flags.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
+import 'package:appflowy/shared/loading.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_actions.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_menu.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/sidebar_workspace.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy/workspace/presentation/widgets/user_avatar.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
-import 'package:flutter/material.dart';
+import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
-import '../../../shared/database_test_op.dart';
-import '../../../shared/dir.dart';
-import '../../../shared/emoji.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- group('collaborative workspace: ', () {
+ group('collaborative workspace:', () {
// combine the create and delete workspace test to reduce the time
testWidgets('create a new workspace, open it and then delete it',
(tester) async {
@@ -93,5 +75,138 @@ void main() {
},
);
});
+
+ testWidgets('check the member count immediately after creating a workspace',
+ (tester) async {
+ // only run the test when the feature flag is on
+ if (!FeatureFlag.collaborativeWorkspace.isOn) {
+ return;
+ }
+
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ const name = 'AppFlowy.IO';
+ // the workspace will be opened after created
+ await tester.createCollaborativeWorkspace(name);
+
+ final loading = find.byType(Loading);
+ await tester.pumpUntilNotFound(loading);
+
+ await tester.openCollaborativeWorkspaceMenu();
+
+ // expect to see the member count
+ final memberCount = find.text('1 member');
+ expect(memberCount, findsNWidgets(2));
+ });
+
+ testWidgets('workspace menu popover behavior test', (tester) async {
+ // only run the test when the feature flag is on
+ if (!FeatureFlag.collaborativeWorkspace.isOn) {
+ return;
+ }
+
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ const name = 'AppFlowy.IO';
+ // the workspace will be opened after created
+ await tester.createCollaborativeWorkspace(name);
+
+ final loading = find.byType(Loading);
+ await tester.pumpUntilNotFound(loading);
+
+ await tester.openCollaborativeWorkspaceMenu();
+
+ // hover on the workspace and click the more button
+ final workspaceItem = find.byWidgetPredicate(
+ (w) => w is WorkspaceMenuItem && w.workspace.name == name,
+ );
+
+ // the workspace menu shouldn't conflict with logout
+ await tester.hoverOnWidget(
+ workspaceItem,
+ onHover: () async {
+ final moreButton = find.byWidgetPredicate(
+ (w) => w is WorkspaceMoreActionList && w.workspace.name == name,
+ );
+ expect(moreButton, findsOneWidget);
+ await tester.tapButton(moreButton);
+ expect(find.text(LocaleKeys.button_rename.tr()), findsOneWidget);
+
+ final logoutButton = find.byType(WorkspaceMoreButton);
+ await tester.tapButton(logoutButton);
+ expect(find.text(LocaleKeys.button_logout.tr()), findsOneWidget);
+ expect(moreButton, findsNothing);
+
+ await tester.tapButton(moreButton);
+ expect(find.text(LocaleKeys.button_logout.tr()), findsNothing);
+ expect(moreButton, findsOneWidget);
+ },
+ );
+ await tester.sendKeyEvent(LogicalKeyboardKey.escape);
+ await tester.pumpAndSettle();
+
+ // clicking on the more action button for the same workspace shouldn't do
+ // anything
+ await tester.openCollaborativeWorkspaceMenu();
+ await tester.hoverOnWidget(
+ workspaceItem,
+ onHover: () async {
+ final moreButton = find.byWidgetPredicate(
+ (w) => w is WorkspaceMoreActionList && w.workspace.name == name,
+ );
+ expect(moreButton, findsOneWidget);
+ await tester.tapButton(moreButton);
+ expect(find.text(LocaleKeys.button_rename.tr()), findsOneWidget);
+
+ // click it again
+ await tester.tapButton(moreButton);
+
+ // nothing should happen
+ expect(find.text(LocaleKeys.button_rename.tr()), findsOneWidget);
+ },
+ );
+ await tester.sendKeyEvent(LogicalKeyboardKey.escape);
+ await tester.pumpAndSettle();
+
+ // clicking on the more button of another workspace should close the menu
+ // for this one
+ await tester.openCollaborativeWorkspaceMenu();
+ final moreButton = find.byWidgetPredicate(
+ (w) => w is WorkspaceMoreActionList && w.workspace.name == name,
+ );
+ await tester.hoverOnWidget(
+ workspaceItem,
+ onHover: () async {
+ expect(moreButton, findsOneWidget);
+ await tester.tapButton(moreButton);
+ expect(find.text(LocaleKeys.button_rename.tr()), findsOneWidget);
+ },
+ );
+
+ final otherWorspaceItem = find.byWidgetPredicate(
+ (w) => w is WorkspaceMenuItem && w.workspace.name != name,
+ );
+ final otherMoreButton = find.byWidgetPredicate(
+ (w) => w is WorkspaceMoreActionList && w.workspace.name != name,
+ );
+ await tester.hoverOnWidget(
+ otherWorspaceItem,
+ onHover: () async {
+ expect(otherMoreButton, findsOneWidget);
+ await tester.tapButton(otherMoreButton);
+ expect(find.text(LocaleKeys.button_rename.tr()), findsOneWidget);
+
+ expect(moreButton, findsNothing);
+ },
+ );
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/share_menu_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/share_menu_test.dart
index 8512222f15..70bb46279e 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/share_menu_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/share_menu_test.dart
@@ -1,29 +1,17 @@
-// ignore_for_file: unused_import
-
import 'package:appflowy/env/cloud_env.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/copy_and_paste/clipboard_service.dart';
import 'package:appflowy/plugins/shared/share/constants.dart';
import 'package:appflowy/plugins/shared/share/share_menu.dart';
-import 'package:appflowy/shared/feature_flags.dart';
import 'package:appflowy/shared/patterns/common_patterns.dart';
import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_icon.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
import '../../../shared/constants.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
-import '../../../shared/workspace.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
@@ -66,7 +54,7 @@ void main() {
);
final shareValues = plainText!
- .replaceAll('https://${ShareConstants.shareBaseUrl}/', '')
+ .replaceAll('${ShareConstants.defaultBaseWebDomain}/app/', '')
.split('/');
final workspaceId = shareValues[0];
expect(workspaceId, isNotEmpty);
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/tabs_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/tabs_test.dart
new file mode 100644
index 0000000000..5c07d99afa
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/tabs_test.dart
@@ -0,0 +1,87 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/plugins/document/presentation/editor_page.dart';
+import 'package:appflowy/shared/loading.dart';
+import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_menu.dart';
+import 'package:appflowy/workspace/presentation/home/tabs/flowy_tab.dart';
+import 'package:appflowy/workspace/presentation/home/tabs/tabs_manager.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../../shared/constants.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('Tabs', () {
+ testWidgets('close other tabs before opening a new workspace',
+ (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ const name = 'AppFlowy.IO';
+ // the workspace will be opened after created
+ await tester.createCollaborativeWorkspace(name);
+
+ final loading = find.byType(Loading);
+ await tester.pumpUntilNotFound(loading);
+
+ // create new tabs in the workspace
+ expect(find.byType(FlowyTab), findsNothing);
+
+ const documentOneName = 'document one';
+ const documentTwoName = 'document two';
+ await tester.createNewPageInSpace(
+ spaceName: Constants.generalSpaceName,
+ layout: ViewLayoutPB.Document,
+ pageName: documentOneName,
+ );
+ await tester.createNewPageInSpace(
+ spaceName: Constants.generalSpaceName,
+ layout: ViewLayoutPB.Document,
+ pageName: documentTwoName,
+ );
+
+ /// Open second menu item in a new tab
+ await tester.openAppInNewTab(documentOneName, ViewLayoutPB.Document);
+
+ /// Open third menu item in a new tab
+ await tester.openAppInNewTab(documentTwoName, ViewLayoutPB.Document);
+
+ expect(
+ find.descendant(
+ of: find.byType(TabsManager),
+ matching: find.byType(FlowyTab),
+ ),
+ findsNWidgets(2),
+ );
+
+ // switch to the another workspace
+ final Finder items = find.byType(WorkspaceMenuItem);
+ await tester.openCollaborativeWorkspaceMenu();
+ await tester.pumpUntilFound(items);
+ expect(items, findsNWidgets(2));
+
+ // open the first workspace
+ await tester.tap(items.first);
+ await tester.pumpUntilNotFound(loading);
+
+ expect(find.byType(FlowyTab), findsNothing);
+ });
+
+ testWidgets('the space view should not be opened', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ expect(find.byType(AppFlowyEditorPage), findsNothing);
+ expect(find.text('Blank page'), findsOne);
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_icon_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_icon_test.dart
index 71d7c8c2eb..e9ad06caee 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_icon_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_icon_test.dart
@@ -1,35 +1,11 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
import 'package:appflowy/env/cloud_env.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/openai/widgets/loading.dart';
-import 'package:appflowy/shared/feature_flags.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_actions.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_icon.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_menu.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/sidebar_workspace.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy/workspace/presentation/widgets/user_avatar.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
-import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
-import 'package:universal_platform/universal_platform.dart';
-import '../../../shared/constants.dart';
-import '../../../shared/database_test_op.dart';
-import '../../../shared/dir.dart';
-import '../../../shared/emoji.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
import '../../../shared/workspace.dart';
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_settings_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_settings_test.dart
index c63a934427..a58fea25b8 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_settings_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_settings_test.dart
@@ -1,46 +1,23 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
import 'package:appflowy/env/cloud_env.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/copy_and_paste/clipboard_service.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/openai/widgets/loading.dart';
import 'package:appflowy/plugins/document/presentation/editor_style.dart';
import 'package:appflowy/plugins/shared/share/publish_tab.dart';
-import 'package:appflowy/shared/feature_flags.dart';
import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_actions.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_menu.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/sidebar_workspace.dart';
import 'package:appflowy/workspace/presentation/settings/pages/settings_workspace_view.dart';
-import 'package:appflowy/workspace/presentation/settings/pages/sites/domain/domain_item.dart';
import 'package:appflowy/workspace/presentation/settings/pages/sites/domain/domain_more_action.dart';
-import 'package:appflowy/workspace/presentation/settings/pages/sites/domain/domain_settings_dialog.dart';
-import 'package:appflowy/workspace/presentation/settings/pages/sites/domain/home_page_menu.dart';
import 'package:appflowy/workspace/presentation/settings/pages/sites/published_page/published_view_item.dart';
import 'package:appflowy/workspace/presentation/settings/pages/sites/published_page/published_view_more_action.dart';
import 'package:appflowy/workspace/presentation/settings/pages/sites/published_page/published_view_settings_dialog.dart';
import 'package:appflowy/workspace/presentation/settings/shared/setting_list_tile.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy/workspace/presentation/widgets/user_avatar.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
-import 'package:flowy_infra_ui/flowy_infra_ui.dart';
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
import '../../../shared/constants.dart';
-import '../../../shared/database_test_op.dart';
-import '../../../shared/dir.dart';
-import '../../../shared/emoji.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
void main() {
@@ -143,6 +120,10 @@ void main() {
widget is PublishedViewItem &&
widget.publishInfoView.view.name == pageName,
);
+ if (pageItem.evaluate().isEmpty) {
+ return;
+ }
+
expect(pageItem, findsOneWidget);
// comment it out because it's not allowed to update the namespace in free plan
@@ -272,7 +253,7 @@ More actions for published page:
await tester.openSettings();
await tester.openSettingsPage(SettingsPage.sites);
// wait the backend return the sites data
- await tester.wait(1000);
+ await tester.wait(2000);
// check if the page is published in sites page
final pageItem = find.byWidgetPredicate(
@@ -280,6 +261,10 @@ More actions for published page:
widget is PublishedViewItem &&
widget.publishInfoView.view.name == pageName,
);
+ if (pageItem.evaluate().isEmpty) {
+ return;
+ }
+
expect(pageItem, findsOneWidget);
final copyLinkItem = find.text(LocaleKeys.shareAction_copyLink.tr());
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_test_runner.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_test_runner.dart
index 8ed85e467d..4d2862038e 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_test_runner.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_test_runner.dart
@@ -3,6 +3,7 @@ import 'package:integration_test/integration_test.dart';
import 'change_name_and_icon_test.dart' as change_name_and_icon_test;
import 'collaborative_workspace_test.dart' as collaborative_workspace_test;
import 'share_menu_test.dart' as share_menu_test;
+import 'tabs_test.dart' as tabs_test;
import 'workspace_icon_test.dart' as workspace_icon_test;
import 'workspace_settings_test.dart' as workspace_settings_test;
@@ -14,4 +15,5 @@ void main() {
collaborative_workspace_test.main();
change_name_and_icon_test.main();
workspace_icon_test.main();
+ tabs_test.main();
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/command_palette/folder_search_test.dart b/frontend/appflowy_flutter/integration_test/desktop/command_palette/folder_search_test.dart
index 80c907b9e9..0b77a0167b 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/command_palette/folder_search_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/command_palette/folder_search_test.dart
@@ -1,6 +1,15 @@
+import 'dart:convert';
+import 'dart:math';
+
+import 'package:appflowy/generated/flowy_svgs.g.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
import 'package:appflowy/workspace/presentation/command_palette/command_palette.dart';
import 'package:appflowy/workspace/presentation/command_palette/widgets/search_field.dart';
-import 'package:appflowy/workspace/presentation/command_palette/widgets/search_result_tile.dart';
+import 'package:appflowy/workspace/presentation/command_palette/widgets/search_result_cell.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/view.pbenum.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:easy_localization/easy_localization.dart';
import 'package:flowy_infra_ui/flowy_infra_ui.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -8,7 +17,9 @@ import 'package:integration_test/integration_test.dart';
import '../../shared/util.dart';
void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ });
group('Folder Search', () {
testWidgets('Search for views', (tester) async {
@@ -33,21 +44,106 @@ void main() {
await tester.pumpAndSettle(const Duration(milliseconds: 200));
// Expect two search results "ViewOna" and "ViewOne" (Distance 1 to ViewOna)
- expect(find.byType(SearchResultTile), findsNWidgets(2));
+ expect(find.byType(SearchResultCell), findsNWidgets(2));
// The score should be higher for "ViewOna" thus it should be shown first
final secondDocumentWidget = tester
- .widget(find.byType(SearchResultTile).first) as SearchResultTile;
- expect(secondDocumentWidget.result.data, secondDocument);
+ .widget(find.byType(SearchResultCell).first) as SearchResultCell;
+ expect(secondDocumentWidget.item.displayName, secondDocument);
// Change search to "ViewOne"
await tester.enterText(searchFieldFinder, firstDocument);
await tester.pumpAndSettle(const Duration(seconds: 1));
// The score should be higher for "ViewOne" thus it should be shown first
- final firstDocumentWidget = tester
- .widget(find.byType(SearchResultTile).first) as SearchResultTile;
- expect(firstDocumentWidget.result.data, firstDocument);
+ final firstDocumentWidget = tester.widget(
+ find.byType(SearchResultCell).first,
+ ) as SearchResultCell;
+ expect(firstDocumentWidget.item.displayName, firstDocument);
+ });
+
+ testWidgets('Displaying icons in search results', (tester) async {
+ final randomValue = Random().nextInt(10000) + 10000;
+ final pageNames = ['First Page-$randomValue', 'Second Page-$randomValue'];
+
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ final emojiIconData = await tester.loadIcon();
+
+ /// create two pages
+ for (final pageName in pageNames) {
+ await tester.createNewPageWithNameUnderParent(name: pageName);
+ await tester.updatePageIconInTitleBarByName(
+ name: pageName,
+ layout: ViewLayoutPB.Document,
+ icon: emojiIconData,
+ );
+ }
+
+ await tester.toggleCommandPalette();
+
+ /// search for `Page`
+ final searchFieldFinder = find.descendant(
+ of: find.byType(SearchField),
+ matching: find.byType(FlowyTextField),
+ );
+ await tester.enterText(searchFieldFinder, 'Page-$randomValue');
+ await tester.pumpAndSettle(const Duration(milliseconds: 200));
+ expect(find.byType(SearchResultCell), findsNWidgets(2));
+
+ /// check results
+ final svgs = find.descendant(
+ of: find.byType(SearchResultCell),
+ matching: find.byType(FlowySvg),
+ );
+ expect(svgs, findsNWidgets(2));
+
+ final firstSvg = svgs.first.evaluate().first.widget as FlowySvg,
+ lastSvg = svgs.last.evaluate().first.widget as FlowySvg;
+ final iconData = IconsData.fromJson(jsonDecode(emojiIconData.emoji));
+
+ /// icon displayed correctly
+ expect(firstSvg.svgString, iconData.svgString);
+ expect(lastSvg.svgString, iconData.svgString);
+
+ testWidgets('select the content in document and search', (tester) async {
+ const firstDocument = ''; // empty document
+
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(name: firstDocument);
+ await tester.editor.updateSelection(
+ Selection(
+ start: Position(
+ path: [0],
+ ),
+ end: Position(
+ path: [0],
+ offset: 10,
+ ),
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ expect(
+ find.byType(FloatingToolbar),
+ findsOneWidget,
+ );
+
+ await tester.toggleCommandPalette();
+ expect(find.byType(CommandPaletteModal), findsOneWidget);
+
+ expect(
+ find.text(LocaleKeys.menuAppHeader_defaultNewPageName.tr()),
+ findsOneWidget,
+ );
+
+ expect(
+ find.text(firstDocument),
+ findsOneWidget,
+ );
+ });
});
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/command_palette/recent_history_test.dart b/frontend/appflowy_flutter/integration_test/desktop/command_palette/recent_history_test.dart
index 277ae8f21e..b9495ae0e7 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/command_palette/recent_history_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/command_palette/recent_history_test.dart
@@ -1,5 +1,5 @@
import 'package:appflowy/workspace/presentation/command_palette/command_palette.dart';
-import 'package:appflowy/workspace/presentation/command_palette/widgets/recent_view_tile.dart';
+import 'package:appflowy/workspace/presentation/command_palette/widgets/search_recent_view_cell.dart';
import 'package:appflowy/workspace/presentation/command_palette/widgets/recent_views_list.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -27,11 +27,12 @@ void main() {
expect(find.byType(RecentViewsList), findsOneWidget);
// Expect three recent history items
- expect(find.byType(RecentViewTile), findsNWidgets(3));
+ expect(find.byType(SearchRecentViewCell), findsNWidgets(3));
// Expect the first item to be the last viewed document
final firstDocumentWidget =
- tester.widget(find.byType(RecentViewTile).first) as RecentViewTile;
+ tester.widget(find.byType(SearchRecentViewCell).first)
+ as SearchRecentViewCell;
expect(firstDocumentWidget.view.name, secondDocument);
});
});
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_calendar_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_calendar_test.dart
index eb1d67ffcd..3a565cbee9 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/database/database_calendar_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_calendar_test.dart
@@ -1,4 +1,5 @@
import 'package:appflowy/plugins/database/calendar/presentation/calendar_event_editor.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
import 'package:appflowy_backend/protobuf/flowy-database2/protobuf.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pbenum.dart';
import 'package:flowy_infra_ui/style_widget/icon_button.dart';
@@ -9,7 +10,14 @@ import '../../shared/database_test_op.dart';
import '../../shared/util.dart';
void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
+
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
group('calendar', () {
testWidgets('update calendar layout', (tester) async {
@@ -301,6 +309,7 @@ void main() {
await tester.createOption(name: "qwer");
await tester.selectOption(name: "asdf");
await tester.dismissCellEditor();
+ await tester.dismissCellEditor();
await tester.tapDatabaseFilterButton();
await tester.tapCreateFilterByFieldType(FieldType.MultiSelect, "Tags");
@@ -332,6 +341,7 @@ void main() {
await tester.tapButton(finderForFieldType(FieldType.MultiSelect));
await tester.selectOption(name: "asdf");
await tester.dismissCellEditor();
+ await tester.dismissCellEditor();
tester.assertNumberOfEventsInCalendar(0);
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_field_settings_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_field_settings_test.dart
index 9b9434d3d7..a71110f1e0 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/database/database_field_settings_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_field_settings_test.dart
@@ -15,6 +15,7 @@ void main() {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
+ // create a database and add a linked database view
await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
await tester.tapCreateLinkedDatabaseViewButton(DatabaseLayoutPB.Grid);
@@ -29,6 +30,11 @@ void main() {
await tester.tapHidePropertyButton();
tester.noFieldWithName('New field 1');
+ // create another field, New field 1 to be hidden still
+ await tester.tapNewPropertyButton();
+ await tester.dismissFieldEditor();
+ tester.noFieldWithName('New field 1');
+
// go back to inline database view, expect field to be shown
await tester.tapTabBarLinkedViewByViewName('Untitled');
tester.findFieldWithName('New field 1');
@@ -60,5 +66,40 @@ void main() {
await tester.tapDatabaseSortButton();
await tester.tapCreateSortByFieldType(FieldType.RichText, "New field 1");
});
+
+ testWidgets('field cell width', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a database and add a linked database view
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
+ await tester.tapCreateLinkedDatabaseViewButton(DatabaseLayoutPB.Grid);
+
+ // create a field
+ await tester.scrollToRight(find.byType(GridPage));
+ await tester.tapNewPropertyButton();
+ await tester.renameField('New field 1');
+ await tester.dismissFieldEditor();
+
+ // check the width of the field
+ expect(tester.getFieldWidth('New field 1'), 150);
+
+ // change the width of the field
+ await tester.changeFieldWidth('New field 1', 200);
+ expect(tester.getFieldWidth('New field 1'), 205);
+
+ // create another field, New field 1 to be same width
+ await tester.tapNewPropertyButton();
+ await tester.dismissFieldEditor();
+ expect(tester.getFieldWidth('New field 1'), 205);
+
+ // go back to inline database view, expect New field 1 to be 150px
+ await tester.tapTabBarLinkedViewByViewName('Untitled');
+ expect(tester.getFieldWidth('New field 1'), 150);
+
+ // go back to linked database view, expect New field 1 to be 205px
+ await tester.tapTabBarLinkedViewByViewName('Grid');
+ expect(tester.getFieldWidth('New field 1'), 205);
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_field_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_field_test.dart
index 9ca6a524a6..6ce248a8a1 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/database/database_field_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_field_test.dart
@@ -1,12 +1,12 @@
-import 'package:flutter/material.dart';
-
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/database/grid/presentation/grid_page.dart';
import 'package:appflowy/plugins/database/widgets/field/type_option_editor/select/select_option.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
import 'package:appflowy/util/field_type_extension.dart';
import 'package:appflowy_backend/protobuf/flowy-database2/protobuf.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/protobuf.dart';
import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -14,7 +14,14 @@ import '../../shared/database_test_op.dart';
import '../../shared/util.dart';
void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
+
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
group('grid edit field test:', () {
testWidgets('rename existing field', (tester) async {
@@ -538,8 +545,8 @@ void main() {
// edit the first date cell
await tester.tapCellInGrid(rowIndex: 0, fieldType: FieldType.DateTime);
- await tester.toggleIncludeTime();
final now = DateTime.now();
+ await tester.toggleIncludeTime();
await tester.selectDay(content: now.day);
await tester.dismissCellEditor();
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_icon_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_icon_test.dart
new file mode 100644
index 0000000000..e6a629ded5
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_icon_test.dart
@@ -0,0 +1,190 @@
+import 'dart:convert';
+
+import 'package:appflowy/generated/flowy_svgs.g.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/database/tab_bar/desktop/tab_bar_add_button.dart';
+import 'package:appflowy/plugins/database/tab_bar/desktop/tab_bar_header.dart';
+import 'package:appflowy/plugins/database/widgets/database_layout_ext.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/header/emoji_icon_widget.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
+import 'package:appflowy/workspace/presentation/home/menu/sidebar/shared/sidebar_folder.dart';
+import 'package:appflowy/workspace/presentation/home/menu/view/view_item.dart';
+import 'package:appflowy_backend/protobuf/flowy-database2/setting_entities.pbenum.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/emoji.dart';
+import '../../shared/util.dart';
+
+void main() {
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
+
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
+
+ testWidgets('change icon', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ final iconData = await tester.loadIcon();
+
+ const pageName = 'Database';
+ await tester.createNewPageWithNameUnderParent(
+ layout: ViewLayoutPB.Grid,
+ name: pageName,
+ );
+
+ /// create board
+ final addButton = find.byType(AddDatabaseViewButton);
+ await tester.tapButton(addButton);
+ await tester.tapButton(
+ find.text(
+ '${LocaleKeys.grid_createView.tr()} ${DatabaseLayoutPB.Board.layoutName}',
+ findRichText: true,
+ ),
+ );
+
+ /// create calendar
+ await tester.tapButton(addButton);
+ await tester.tapButton(
+ find.text(
+ '${LocaleKeys.grid_createView.tr()} ${DatabaseLayoutPB.Calendar.layoutName}',
+ findRichText: true,
+ ),
+ );
+
+ final databaseTabBarItem = find.byType(DatabaseTabBarItem);
+ expect(databaseTabBarItem, findsNWidgets(3));
+ final gridItem = databaseTabBarItem.first,
+ boardItem = databaseTabBarItem.at(1),
+ calendarItem = databaseTabBarItem.last;
+
+ /// change the icon of grid
+ /// the first tapping is to select specific item
+ /// the second tapping is to show the menu
+ await tester.tapButton(gridItem);
+ await tester.tapButton(gridItem);
+
+ /// change icon
+ await tester
+ .tapButton(find.text(LocaleKeys.disclosureAction_changeIcon.tr()));
+ await tester.tapIcon(iconData, enableColor: false);
+ final gridIcon = find.descendant(
+ of: gridItem,
+ matching: find.byType(RawEmojiIconWidget),
+ );
+ final gridIconWidget =
+ gridIcon.evaluate().first.widget as RawEmojiIconWidget;
+ final iconsData = IconsData.fromJson(jsonDecode(iconData.emoji));
+ final gridIconsData =
+ IconsData.fromJson(jsonDecode(gridIconWidget.emoji.emoji));
+ expect(gridIconsData.iconName, iconsData.iconName);
+
+ /// change the icon of board
+ await tester.tapButton(boardItem);
+ await tester.tapButton(boardItem);
+ await tester
+ .tapButton(find.text(LocaleKeys.disclosureAction_changeIcon.tr()));
+ await tester.tapIcon(iconData, enableColor: false);
+ final boardIcon = find.descendant(
+ of: boardItem,
+ matching: find.byType(RawEmojiIconWidget),
+ );
+ final boardIconWidget =
+ boardIcon.evaluate().first.widget as RawEmojiIconWidget;
+ final boardIconsData =
+ IconsData.fromJson(jsonDecode(boardIconWidget.emoji.emoji));
+ expect(boardIconsData.iconName, iconsData.iconName);
+
+ /// change the icon of calendar
+ await tester.tapButton(calendarItem);
+ await tester.tapButton(calendarItem);
+ await tester
+ .tapButton(find.text(LocaleKeys.disclosureAction_changeIcon.tr()));
+ await tester.tapIcon(iconData, enableColor: false);
+ final calendarIcon = find.descendant(
+ of: calendarItem,
+ matching: find.byType(RawEmojiIconWidget),
+ );
+ final calendarIconWidget =
+ calendarIcon.evaluate().first.widget as RawEmojiIconWidget;
+ final calendarIconsData =
+ IconsData.fromJson(jsonDecode(calendarIconWidget.emoji.emoji));
+ expect(calendarIconsData.iconName, iconsData.iconName);
+ });
+
+ testWidgets('change database icon from sidebar', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ final iconData = await tester.loadIcon();
+ final icon = IconsData.fromJson(jsonDecode(iconData.emoji)), emoji = '😄';
+
+ const pageName = 'Database';
+ await tester.createNewPageWithNameUnderParent(
+ layout: ViewLayoutPB.Grid,
+ name: pageName,
+ );
+ final viewItem = find.descendant(
+ of: find.byType(SidebarFolder),
+ matching: find.byWidgetPredicate(
+ (w) => w is ViewItem && w.view.name == pageName,
+ ),
+ );
+
+ /// change icon to emoji
+ await tester.tapButton(
+ find.descendant(
+ of: viewItem,
+ matching: find.byType(FlowySvg),
+ ),
+ );
+ await tester.tapEmoji(emoji);
+ final iconWidget = find.descendant(
+ of: viewItem,
+ matching: find.byType(RawEmojiIconWidget),
+ );
+ expect(
+ (iconWidget.evaluate().first.widget as RawEmojiIconWidget).emoji.emoji,
+ emoji,
+ );
+
+ /// the icon will not be displayed in database item
+ Finder databaseIcon = find.descendant(
+ of: find.byType(DatabaseTabBarItem),
+ matching: find.byType(FlowySvg),
+ );
+ expect(
+ (databaseIcon.evaluate().first.widget as FlowySvg).svg,
+ FlowySvgs.icon_grid_s,
+ );
+
+ /// change emoji to icon
+ await tester.tapButton(iconWidget);
+ await tester.tapIcon(iconData);
+ expect(
+ (iconWidget.evaluate().first.widget as RawEmojiIconWidget).emoji.emoji,
+ iconData.emoji,
+ );
+
+ databaseIcon = find.descendant(
+ of: find.byType(DatabaseTabBarItem),
+ matching: find.byType(RawEmojiIconWidget),
+ );
+ final databaseIconWidget =
+ databaseIcon.evaluate().first.widget as RawEmojiIconWidget;
+ final databaseIconsData =
+ IconsData.fromJson(jsonDecode(databaseIconWidget.emoji.emoji));
+ expect(icon.svgString, databaseIconsData.svgString);
+ expect(icon.color, isNotEmpty);
+ expect(icon.color, databaseIconsData.color);
+
+ /// the icon in database item should not show the color
+ expect(databaseIconWidget.enableColor, false);
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_row_cover_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_row_cover_test.dart
index 6c96a4825e..8741dcd75f 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/database/database_row_cover_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_row_cover_test.dart
@@ -4,12 +4,10 @@ import 'package:appflowy/core/config/kv.dart';
import 'package:appflowy/core/config/kv_keys.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/database/widgets/card/card.dart';
-import 'package:appflowy/plugins/database/widgets/cell_editor/media_cell_editor.dart';
import 'package:appflowy/plugins/database/widgets/row/row_banner.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
import 'package:appflowy/shared/af_image.dart';
import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy_backend/protobuf/flowy-database2/field_entities.pbenum.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pbenum.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/services.dart';
@@ -26,61 +24,6 @@ void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
group('database row cover', () {
- testWidgets('add image to media field and check if cover is set (grid)',
- (tester) async {
- await tester.initializeAppFlowy();
- await tester.tapAnonymousSignInButton();
-
- await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
-
- // Invoke the field editor
- await tester.tapGridFieldWithName('Type');
- await tester.tapEditFieldButton();
-
- // Change to media type
- await tester.tapSwitchFieldTypeButton();
- await tester.selectFieldType(FieldType.Media);
- await tester.dismissFieldEditor();
-
- // Prepare file for upload from local
- final image = await rootBundle.load('assets/test/images/sample.jpeg');
- final tempDirectory = await getTemporaryDirectory();
-
- final imagePath = p.join(tempDirectory.path, 'sample.jpeg');
- final file = File(imagePath)
- ..writeAsBytesSync(image.buffer.asUint8List());
-
- mockPickFilePaths(paths: [imagePath]);
- await getIt().set(KVKeys.kCloudType, '0');
-
- // Open media cell editor
- await tester.tapCellInGrid(rowIndex: 0, fieldType: FieldType.Media);
- await tester.findMediaCellEditor(findsOneWidget);
-
- // Click on add file button in the Media Cell Editor
- await tester.tap(find.text(LocaleKeys.grid_media_addFileOrImage.tr()));
- await tester.pumpAndSettle();
-
- // Tap on the upload interaction
- await tester.tapFileUploadHint();
-
- // Expect one file
- expect(find.byType(RenderMedia), findsOneWidget);
-
- // Close cell editor
- await tester.dismissCellEditor();
-
- // Open first row in row detail view
- await tester.openFirstRowDetailPage();
- await tester.pumpAndSettle();
-
- // Expect a cover to be shown
- expect(find.byType(RowCover), findsOneWidget);
-
- // Remove the temp file
- await Future.wait([file.delete()]);
- });
-
testWidgets('add and remove cover from Row Detail Card', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_row_page_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_row_page_test.dart
index 26688ac0cf..22f059d199 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/database/database_row_page_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_row_page_test.dart
@@ -1,17 +1,18 @@
-import 'package:appflowy/plugins/database/widgets/cell/desktop_row_detail/desktop_row_detail_checklist_cell.dart';
-import 'package:appflowy/plugins/database/widgets/cell_editor/checklist_cell_editor.dart';
-import 'package:flowy_infra_ui/flowy_infra_ui.dart';
-import 'package:flutter/material.dart';
-
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/header/desktop_field_cell.dart';
+import 'package:appflowy/plugins/database/widgets/cell/desktop_row_detail/desktop_row_detail_checklist_cell.dart';
+import 'package:appflowy/plugins/database/widgets/cell_editor/checklist_cell_editor.dart';
import 'package:appflowy/plugins/database/widgets/row/row_detail.dart';
+import 'package:appflowy/plugins/document/document_page.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/header/emoji_icon_widget.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
import 'package:appflowy/util/field_type_extension.dart';
import 'package:appflowy_backend/protobuf/flowy-database2/field_entities.pbenum.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
import 'package:easy_localization/easy_localization.dart';
+import 'package:flowy_infra_ui/flowy_infra_ui.dart';
+import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -21,7 +22,14 @@ import '../../shared/emoji.dart';
import '../../shared/util.dart';
void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
+
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
group('grid row detail page:', () {
testWidgets('opens', (tester) async {
@@ -76,6 +84,24 @@ void main() {
// The number of emoji should be two. One in the row displayed in the grid
// one in the row detail page.
expect(emojiText, findsNWidgets(2));
+
+ // insert a sub page in database
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.editor.showSlashMenu();
+ await tester.pumpAndSettle();
+ await tester.editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_subPage_name.tr(),
+ offset: 100,
+ );
+ await tester.pumpAndSettle();
+
+ // the row detail page should be closed
+ final rowDetailPage = find.byType(RowDetailPage);
+ await tester.pumpUntilNotFound(rowDetailPage);
+
+ // expect to see a document page
+ final documentPage = find.byType(DocumentPage);
+ expect(documentPage, findsOneWidget);
});
testWidgets('remove emoji', (tester) async {
@@ -368,11 +394,16 @@ void main() {
isChecked: false,
);
tester.assertPhantomChecklistItemAtIndex(index: 1);
+ tester.assertPhantomChecklistItemContent("");
await tester.enterText(find.byType(PhantomChecklistItem), 'task 2');
await tester.pumpAndSettle();
- await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
- await tester.pumpAndSettle(const Duration(milliseconds: 500));
+ await tester.hoverOnWidget(
+ find.byType(ChecklistRowDetailCell),
+ onHover: () async {
+ await tester.tapButton(find.byType(ChecklistItemControl));
+ },
+ );
tester.assertChecklistTaskInEditor(
index: 1,
@@ -380,6 +411,7 @@ void main() {
isChecked: false,
);
tester.assertPhantomChecklistItemAtIndex(index: 2);
+ tester.assertPhantomChecklistItemContent("");
await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
await tester.pumpAndSettle();
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_view_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_view_test.dart
index e35c9cc9d8..71656c1ea6 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/database/database_view_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_view_test.dart
@@ -1,5 +1,10 @@
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/database/grid/presentation/grid_page.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
import 'package:appflowy_backend/protobuf/flowy-database2/setting_entities.pbenum.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter_animate/flutter_animate.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -73,5 +78,37 @@ void main() {
await tester.pumpAndSettle();
});
+
+ testWidgets('insert grid in column', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ /// create page and show slash menu
+ await tester.createNewPageWithNameUnderParent(name: 'test page');
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.editor.showSlashMenu();
+ await tester.pumpAndSettle();
+
+ /// create a column
+ await tester.editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_name_twoColumns.tr(),
+ );
+ final actionList = find.byType(BlockActionList);
+ expect(actionList, findsNWidgets(2));
+ final position = tester.getCenter(actionList.last);
+
+ /// tap the second child of column
+ await tester.tapAt(position.copyWith(dx: position.dx + 50));
+
+ /// create a grid
+ await tester.editor.showSlashMenu();
+ await tester.pumpAndSettle();
+ await tester.editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_name_grid.tr(),
+ );
+
+ final grid = find.byType(GridPageContent);
+ expect(grid, findsOneWidget);
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_alignment_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_alignment_test.dart
index d95d907881..1a8a3fcda8 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_alignment_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_alignment_test.dart
@@ -27,8 +27,9 @@ void main() {
await tester.pumpAndSettle();
// click the align center
- await tester.tapButtonWithFlowySvgData(FlowySvgs.toolbar_align_left_s);
- await tester.tapButtonWithFlowySvgData(FlowySvgs.toolbar_align_center_s);
+ await tester.tapButtonWithFlowySvgData(FlowySvgs.toolbar_alignment_m);
+ await tester
+ .tapButtonWithFlowySvgData(FlowySvgs.toolbar_text_align_center_m);
// expect to see the align center
final editorState = tester.editor.getCurrentEditorState();
@@ -36,13 +37,15 @@ void main() {
expect(first.attributes[blockComponentAlign], 'center');
// click the align right
- await tester.tapButtonWithFlowySvgData(FlowySvgs.toolbar_align_center_s);
- await tester.tapButtonWithFlowySvgData(FlowySvgs.toolbar_align_right_s);
+ await tester.tapButtonWithFlowySvgData(FlowySvgs.toolbar_alignment_m);
+ await tester
+ .tapButtonWithFlowySvgData(FlowySvgs.toolbar_text_align_right_m);
expect(first.attributes[blockComponentAlign], 'right');
// click the align left
- await tester.tapButtonWithFlowySvgData(FlowySvgs.toolbar_align_right_s);
- await tester.tapButtonWithFlowySvgData(FlowySvgs.toolbar_align_left_s);
+ await tester.tapButtonWithFlowySvgData(FlowySvgs.toolbar_alignment_m);
+ await tester
+ .tapButtonWithFlowySvgData(FlowySvgs.toolbar_text_align_left_m);
expect(first.attributes[blockComponentAlign], 'left');
});
@@ -75,7 +78,7 @@ void main() {
[
LogicalKeyboardKey.control,
LogicalKeyboardKey.shift,
- LogicalKeyboardKey.keyE,
+ LogicalKeyboardKey.keyC,
],
tester: tester,
withKeyUp: true,
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_callout_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_callout_test.dart
new file mode 100644
index 0000000000..b5449ec622
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_callout_test.dart
@@ -0,0 +1,67 @@
+import 'dart:convert';
+
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/base/icon/icon_widget.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/base/emoji_picker_button.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/callout/callout_block_component.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/emoji.dart';
+import '../../shared/util.dart';
+
+void main() {
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
+
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
+
+ testWidgets('callout with emoji icon picker', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ final emojiIconData = await tester.loadIcon();
+
+ /// create a new document
+ await tester.createNewPageWithNameUnderParent();
+
+ /// tap the first line of the document
+ await tester.editor.tapLineOfEditorAt(0);
+
+ /// create callout
+ await tester.editor.showSlashMenu();
+ await tester.pumpAndSettle();
+ await tester.editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_name_callout.tr(),
+ );
+
+ /// select an icon
+ final emojiPickerButton = find.descendant(
+ of: find.byType(CalloutBlockComponentWidget),
+ matching: find.byType(EmojiPickerButton),
+ );
+ await tester.tapButton(emojiPickerButton);
+ await tester.tapIcon(emojiIconData);
+
+ /// verification results
+ final iconData = IconsData.fromJson(jsonDecode(emojiIconData.emoji));
+ final iconWidget = find
+ .descendant(
+ of: emojiPickerButton,
+ matching: find.byType(IconWidget),
+ )
+ .evaluate()
+ .first
+ .widget as IconWidget;
+ final iconWidgetData = iconWidget.iconsData;
+ expect(iconWidgetData.svgString, iconData.svgString);
+ expect(iconWidgetData.iconName, iconData.iconName);
+ expect(iconWidgetData.groupName, iconData.groupName);
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_copy_and_paste_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_copy_and_paste_test.dart
index 5e03490113..d1e34edcb5 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_copy_and_paste_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_copy_and_paste_test.dart
@@ -1,16 +1,17 @@
+import 'dart:async';
import 'dart:io';
-import 'package:flutter/services.dart';
-
import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/block_menu/block_menu_button.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/copy_and_paste/clipboard_service.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/image/custom_image_block_component/custom_image_block_component.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/link_preview/custom_link_preview.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/link_preview/link_preview_menu.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/link_preview/paste_as/paste_as_menu.dart';
import 'package:appflowy/startup/startup.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
import 'package:appflowy_editor_plugins/appflowy_editor_plugins.dart';
import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
import 'package:universal_platform/universal_platform.dart';
@@ -20,17 +21,20 @@ import '../../shared/util.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- group('copy and paste in document', () {
+ group('copy and paste in document:', () {
testWidgets('paste multiple lines at the first line', (tester) async {
// mock the clipboard
const lines = 3;
await tester.pasteContent(
plainText: List.generate(lines, (index) => 'line $index').join('\n'),
(editorState) {
- expect(editorState.document.root.children.length, 3);
+ expect(editorState.document.root.children.length, 1);
+ final text =
+ editorState.document.root.children.first.delta!.toPlainText();
+ final textLines = text.split('\n');
for (var i = 0; i < lines; i++) {
expect(
- editorState.getNodeAtPath([i])!.delta!.toPlainText(),
+ textLines[i],
'line $i',
);
}
@@ -170,319 +174,370 @@ void main() {
},
);
});
- });
- testWidgets('paste text on part of bullet list', (tester) async {
- const plainText = 'test';
+ testWidgets('paste text on part of bullet list', (tester) async {
+ const plainText = 'test';
- await tester.pasteContent(
- plainText: plainText,
- beforeTest: (editorState) async {
- final transaction = editorState.transaction;
- transaction.insertNodes(
- [0],
- [
- Node(
- type: BulletedListBlockKeys.type,
- attributes: {
- 'delta': [
- {"insert": "bullet list"},
- ],
- },
- ),
- ],
- );
-
- // Set the selection to the second numbered list node (which has empty delta)
- transaction.afterSelection = Selection(
- start: Position(path: [0], offset: 7),
- end: Position(path: [0], offset: 11),
- );
-
- await editorState.apply(transaction);
- await tester.pumpAndSettle();
- },
- (editorState) {
- final node = editorState.getNodeAtPath([0]);
- expect(node?.delta?.toPlainText(), 'bullet test');
- expect(node?.type, BulletedListBlockKeys.type);
- },
- );
- });
-
- testWidgets('paste image(png) from memory', (tester) async {
- final image = await rootBundle.load('assets/test/images/sample.png');
- final bytes = image.buffer.asUint8List();
- await tester.pasteContent(image: ('png', bytes), (editorState) {
- expect(editorState.document.root.children.length, 1);
- final node = editorState.getNodeAtPath([0])!;
- expect(node.type, ImageBlockKeys.type);
- expect(node.attributes[ImageBlockKeys.url], isNotNull);
- });
- });
-
- testWidgets('paste image(jpeg) from memory', (tester) async {
- final image = await rootBundle.load('assets/test/images/sample.jpeg');
- final bytes = image.buffer.asUint8List();
- await tester.pasteContent(image: ('jpeg', bytes), (editorState) {
- expect(editorState.document.root.children.length, 1);
- final node = editorState.getNodeAtPath([0])!;
- expect(node.type, ImageBlockKeys.type);
- expect(node.attributes[ImageBlockKeys.url], isNotNull);
- });
- });
-
- testWidgets('paste image(gif) from memory', (tester) async {
- final image = await rootBundle.load('assets/test/images/sample.gif');
- final bytes = image.buffer.asUint8List();
- await tester.pasteContent(image: ('gif', bytes), (editorState) {
- expect(editorState.document.root.children.length, 1);
- final node = editorState.getNodeAtPath([0])!;
- expect(node.type, ImageBlockKeys.type);
- expect(node.attributes[ImageBlockKeys.url], isNotNull);
- });
- });
-
- testWidgets(
- 'format the selected text to href when pasting url if available',
- (tester) async {
- const text = 'appflowy';
- const url = 'https://appflowy.io';
await tester.pasteContent(
- plainText: url,
+ plainText: plainText,
beforeTest: (editorState) async {
- await tester.ime.insertText(text);
- await tester.editor.updateSelection(
- Selection.single(
- path: [0],
- startOffset: 0,
- endOffset: text.length,
- ),
+ final transaction = editorState.transaction;
+ transaction.insertNodes(
+ [0],
+ [
+ Node(
+ type: BulletedListBlockKeys.type,
+ attributes: {
+ 'delta': [
+ {"insert": "bullet list"},
+ ],
+ },
+ ),
+ ],
);
- },
- (editorState) {
- final node = editorState.getNodeAtPath([0])!;
- expect(node.type, ParagraphBlockKeys.type);
- expect(node.delta!.toJson(), [
- {
- 'insert': text,
- 'attributes': {'href': url},
- }
- ]);
- },
- );
- },
- );
- // https://github.com/AppFlowy-IO/AppFlowy/issues/3263
- testWidgets(
- 'paste the image from clipboard when html and image are both available',
- (tester) async {
- const html =
- '''
''';
- final image = await rootBundle.load('assets/test/images/sample.png');
- final bytes = image.buffer.asUint8List();
- await tester.pasteContent(
- html: html,
- image: ('png', bytes),
- (editorState) {
- expect(editorState.document.root.children.length, 1);
- final node = editorState.getNodeAtPath([0])!;
- expect(node.type, ImageBlockKeys.type);
- },
- );
- },
- );
+ // Set the selection to the second numbered list node (which has empty delta)
+ transaction.afterSelection = Selection(
+ start: Position(path: [0], offset: 7),
+ end: Position(path: [0], offset: 11),
+ );
- testWidgets('paste the html content contains section', (tester) async {
- const html =
- '''''';
- await tester.pasteContent(html: html, (editorState) {
- expect(editorState.document.root.children.length, 2);
- final node1 = editorState.getNodeAtPath([0])!;
- final node2 = editorState.getNodeAtPath([1])!;
- expect(node1.type, ParagraphBlockKeys.type);
- expect(node2.type, ParagraphBlockKeys.type);
- });
- });
-
- testWidgets('paste the html from google translation', (tester) async {
- const html =
- '''Assessment focus: potential motivations, empathy➢Personality characteristics and potential motivations:-Reflection of self-worth-Have a unique definition of success-Be true to your own lifestyle''';
- await tester.pasteContent(html: html, (editorState) {
- expect(editorState.document.root.children.length, 8);
- });
- });
-
- testWidgets(
- 'auto convert url to link preview block',
- (tester) async {
- const url = 'https://appflowy.io';
- await tester.pasteContent(plainText: url, (editorState) async {
- // the second one is the paragraph node
- expect(editorState.document.root.children.length, 2);
- final node = editorState.getNodeAtPath([0])!;
- expect(node.type, LinkPreviewBlockKeys.type);
- expect(node.attributes[LinkPreviewBlockKeys.url], url);
- });
-
- // hover on the link preview block
- // click the more button
- // and select convert to link
- await tester.hoverOnWidget(
- find.byType(CustomLinkPreviewWidget),
- onHover: () async {
- final convertToLinkButton = find.byWidgetPredicate((widget) {
- return widget is MenuBlockButton &&
- widget.tooltip ==
- LocaleKeys.document_plugins_urlPreview_convertToLink.tr();
- });
- expect(convertToLinkButton, findsOneWidget);
- await tester.tap(convertToLinkButton);
+ await editorState.apply(transaction);
await tester.pumpAndSettle();
},
- );
-
- await tester.pumpAndSettle();
-
- final editorState = tester.editor.getCurrentEditorState();
- final textNode = editorState.getNodeAtPath([0])!;
- expect(textNode.type, ParagraphBlockKeys.type);
- expect(textNode.delta!.toJson(), [
- {
- 'insert': url,
- 'attributes': {'href': url},
- }
- ]);
- },
- );
-
- testWidgets(
- 'ctrl/cmd+z to undo the auto convert url to link preview block',
- (tester) async {
- const url = 'https://appflowy.io';
- await tester.pasteContent(plainText: url, (editorState) async {
- // the second one is the paragraph node
- expect(editorState.document.root.children.length, 2);
- final node = editorState.getNodeAtPath([0])!;
- expect(node.type, LinkPreviewBlockKeys.type);
- expect(node.attributes[LinkPreviewBlockKeys.url], url);
- });
-
- await tester.editor.tapLineOfEditorAt(0);
- await tester.simulateKeyEvent(
- LogicalKeyboardKey.keyZ,
- isControlPressed:
- UniversalPlatform.isLinux || UniversalPlatform.isWindows,
- isMetaPressed: UniversalPlatform.isMacOS,
- );
- await tester.pumpAndSettle();
-
- final editorState = tester.editor.getCurrentEditorState();
- final node = editorState.getNodeAtPath([0])!;
- expect(node.type, ParagraphBlockKeys.type);
- expect(node.delta!.toJson(), [
- {
- 'insert': url,
- 'attributes': {'href': url},
- }
- ]);
- },
- );
-
- testWidgets(
- 'paste the nodes start with non-delta node',
- (tester) async {
- await tester.pasteContent((_) {});
- const text = 'Hello World';
- final editorState = tester.editor.getCurrentEditorState();
- final transaction = editorState.transaction;
- // [image_block]
- // [paragraph_block]
- transaction.insertNodes([
- 0,
- ], [
- customImageNode(url: ''),
- paragraphNode(text: text),
- ]);
- await editorState.apply(transaction);
- await tester.pumpAndSettle();
-
- await tester.editor.tapLineOfEditorAt(0);
- // select all and copy
- await tester.simulateKeyEvent(
- LogicalKeyboardKey.keyA,
- isControlPressed:
- UniversalPlatform.isLinux || UniversalPlatform.isWindows,
- isMetaPressed: UniversalPlatform.isMacOS,
- );
- await tester.simulateKeyEvent(
- LogicalKeyboardKey.keyC,
- isControlPressed:
- UniversalPlatform.isLinux || UniversalPlatform.isWindows,
- isMetaPressed: UniversalPlatform.isMacOS,
- );
-
- // put the cursor to the end of the paragraph block
- await tester.editor.tapLineOfEditorAt(0);
-
- // paste the content
- await tester.simulateKeyEvent(
- LogicalKeyboardKey.keyV,
- isControlPressed:
- UniversalPlatform.isLinux || UniversalPlatform.isWindows,
- isMetaPressed: UniversalPlatform.isMacOS,
- );
- await tester.pumpAndSettle();
-
- // expect the image and the paragraph block are inserted below the cursor
- expect(editorState.getNodeAtPath([0])!.type, CustomImageBlockKeys.type);
- expect(editorState.getNodeAtPath([1])!.type, ParagraphBlockKeys.type);
- expect(editorState.getNodeAtPath([2])!.type, CustomImageBlockKeys.type);
- expect(editorState.getNodeAtPath([3])!.type, ParagraphBlockKeys.type);
- },
- );
-
- testWidgets('paste the url without protocol', (tester) async {
- // paste the image that from local file
- const plainText = '1.jpg';
- final image = await rootBundle.load('assets/test/images/sample.jpeg');
- final bytes = image.buffer.asUint8List();
- await tester.pasteContent(plainText: plainText, image: ('jpeg', bytes),
(editorState) {
- final node = editorState.getNodeAtPath([0])!;
- expect(node.type, ImageBlockKeys.type);
- expect(node.attributes[ImageBlockKeys.url], isNotEmpty);
+ final node = editorState.getNodeAtPath([0]);
+ expect(node?.delta?.toPlainText(), 'bullet test');
+ expect(node?.type, BulletedListBlockKeys.type);
+ },
+ );
});
- });
- testWidgets('paste the image url', (tester) async {
- const plainText = 'https://appflowy.io/1.jpg';
- final image = await rootBundle.load('assets/test/images/sample.jpeg');
- final bytes = image.buffer.asUint8List();
- await tester.pasteContent(plainText: plainText, image: ('jpeg', bytes),
+ testWidgets('paste image(png) from memory', (tester) async {
+ final image = await rootBundle.load('assets/test/images/sample.png');
+ final bytes = image.buffer.asUint8List();
+ await tester.pasteContent(image: ('png', bytes), (editorState) {
+ expect(editorState.document.root.children.length, 1);
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ImageBlockKeys.type);
+ expect(node.attributes[ImageBlockKeys.url], isNotNull);
+ });
+ });
+
+ testWidgets('paste image(jpeg) from memory', (tester) async {
+ final image = await rootBundle.load('assets/test/images/sample.jpeg');
+ final bytes = image.buffer.asUint8List();
+ await tester.pasteContent(image: ('jpeg', bytes), (editorState) {
+ expect(editorState.document.root.children.length, 1);
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ImageBlockKeys.type);
+ expect(node.attributes[ImageBlockKeys.url], isNotNull);
+ });
+ });
+
+ testWidgets('paste image(gif) from memory', (tester) async {
+ final image = await rootBundle.load('assets/test/images/sample.gif');
+ final bytes = image.buffer.asUint8List();
+ await tester.pasteContent(image: ('gif', bytes), (editorState) {
+ expect(editorState.document.root.children.length, 1);
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ImageBlockKeys.type);
+ expect(node.attributes[ImageBlockKeys.url], isNotNull);
+ });
+ });
+
+ testWidgets(
+ 'format the selected text to href when pasting url if available',
+ (tester) async {
+ const text = 'appflowy';
+ const url = 'https://appflowy.io';
+ await tester.pasteContent(
+ plainText: url,
+ beforeTest: (editorState) async {
+ await tester.ime.insertText(text);
+ await tester.editor.updateSelection(
+ Selection.single(
+ path: [0],
+ startOffset: 0,
+ endOffset: text.length,
+ ),
+ );
+ },
+ (editorState) {
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ParagraphBlockKeys.type);
+ expect(node.delta!.toJson(), [
+ {
+ 'insert': text,
+ 'attributes': {'href': url},
+ }
+ ]);
+ },
+ );
+ },
+ );
+
+ // https://github.com/AppFlowy-IO/AppFlowy/issues/3263
+ testWidgets(
+ 'paste the image from clipboard when html and image are both available',
+ (tester) async {
+ const html =
+ '''
''';
+ final image = await rootBundle.load('assets/test/images/sample.png');
+ final bytes = image.buffer.asUint8List();
+ await tester.pasteContent(
+ html: html,
+ image: ('png', bytes),
+ (editorState) {
+ expect(editorState.document.root.children.length, 1);
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ImageBlockKeys.type);
+ },
+ );
+ },
+ );
+
+ testWidgets('paste the html content contains section', (tester) async {
+ const html =
+ '''''';
+ await tester.pasteContent(html: html, (editorState) {
+ expect(editorState.document.root.children.length, 2);
+ final node1 = editorState.getNodeAtPath([0])!;
+ final node2 = editorState.getNodeAtPath([1])!;
+ expect(node1.type, ParagraphBlockKeys.type);
+ expect(node2.type, ParagraphBlockKeys.type);
+ });
+ });
+
+ testWidgets('paste the html from google translation', (tester) async {
+ const html =
+ '''Assessment focus: potential motivations, empathy➢Personality characteristics and potential motivations:-Reflection of self-worth-Have a unique definition of success-Be true to your own lifestyle''';
+ await tester.pasteContent(html: html, (editorState) {
+ expect(editorState.document.root.children.length, 8);
+ });
+ });
+
+ testWidgets(
+ 'auto convert url to link preview block',
+ (tester) async {
+ const url = 'https://appflowy.io';
+ await tester.pasteContent(plainText: url, (editorState) async {
+ final pasteAsMenu = find.byType(PasteAsMenu);
+ expect(pasteAsMenu, findsOneWidget);
+ final bookmarkButton = find.text(
+ LocaleKeys.document_plugins_linkPreview_typeSelection_bookmark.tr(),
+ );
+ await tester.tapButton(bookmarkButton);
+ // the second one is the paragraph node
+ expect(editorState.document.root.children.length, 1);
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, LinkPreviewBlockKeys.type);
+ expect(node.attributes[LinkPreviewBlockKeys.url], url);
+ });
+
+ // hover on the link preview block
+ // click the more button
+ // and select convert to link
+ await tester.hoverOnWidget(
+ find.byType(CustomLinkPreviewWidget),
+ onHover: () async {
+ /// show menu
+ final menu = find.byType(CustomLinkPreviewMenu);
+ expect(menu, findsOneWidget);
+ await tester.tapButton(menu);
+
+ final convertToLinkButton = find.text(
+ LocaleKeys.document_plugins_linkPreview_linkPreviewMenu_toUrl
+ .tr(),
+ );
+ expect(convertToLinkButton, findsOneWidget);
+ await tester.tapButton(convertToLinkButton);
+ },
+ );
+
+ final editorState = tester.editor.getCurrentEditorState();
+ final textNode = editorState.getNodeAtPath([0])!;
+ expect(textNode.type, ParagraphBlockKeys.type);
+ expect(textNode.delta!.toJson(), [
+ {
+ 'insert': url,
+ 'attributes': {'href': url},
+ }
+ ]);
+ },
+ );
+
+ testWidgets(
+ 'ctrl/cmd+z to undo the auto convert url to link preview block',
+ (tester) async {
+ const url = 'https://appflowy.io';
+ await tester.pasteContent(plainText: url, (editorState) async {
+ final pasteAsMenu = find.byType(PasteAsMenu);
+ expect(pasteAsMenu, findsOneWidget);
+ final bookmarkButton = find.text(
+ LocaleKeys.document_plugins_linkPreview_typeSelection_bookmark.tr(),
+ );
+ await tester.tapButton(bookmarkButton);
+ // the second one is the paragraph node
+ expect(editorState.document.root.children.length, 1);
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, LinkPreviewBlockKeys.type);
+ expect(node.attributes[LinkPreviewBlockKeys.url], url);
+ });
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyZ,
+ isControlPressed:
+ UniversalPlatform.isLinux || UniversalPlatform.isWindows,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ final editorState = tester.editor.getCurrentEditorState();
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ParagraphBlockKeys.type);
+ expect(node.delta!.toJson(), [
+ {
+ 'insert': url,
+ 'attributes': {'href': url},
+ }
+ ]);
+ },
+ );
+
+ testWidgets(
+ 'paste the nodes start with non-delta node',
+ (tester) async {
+ await tester.pasteContent((_) {});
+ const text = 'Hello World';
+ final editorState = tester.editor.getCurrentEditorState();
+ final transaction = editorState.transaction;
+ // [image_block]
+ // [paragraph_block]
+ transaction.insertNodes([
+ 0,
+ ], [
+ customImageNode(url: ''),
+ paragraphNode(text: text),
+ ]);
+ await editorState.apply(transaction);
+ await tester.pumpAndSettle();
+
+ await tester.editor.tapLineOfEditorAt(0);
+ // select all and copy
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyA,
+ isControlPressed:
+ UniversalPlatform.isLinux || UniversalPlatform.isWindows,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyC,
+ isControlPressed:
+ UniversalPlatform.isLinux || UniversalPlatform.isWindows,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+
+ // put the cursor to the end of the paragraph block
+ await tester.editor.tapLineOfEditorAt(0);
+
+ // paste the content
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed:
+ UniversalPlatform.isLinux || UniversalPlatform.isWindows,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ // expect the image and the paragraph block are inserted below the cursor
+ expect(editorState.getNodeAtPath([0])!.type, CustomImageBlockKeys.type);
+ expect(editorState.getNodeAtPath([1])!.type, ParagraphBlockKeys.type);
+ expect(editorState.getNodeAtPath([2])!.type, CustomImageBlockKeys.type);
+ expect(editorState.getNodeAtPath([3])!.type, ParagraphBlockKeys.type);
+ },
+ );
+
+ testWidgets('paste the url without protocol', (tester) async {
+ // paste the image that from local file
+ const plainText = '1.jpg';
+ final image = await rootBundle.load('assets/test/images/sample.jpeg');
+ final bytes = image.buffer.asUint8List();
+ await tester.pasteContent(plainText: plainText, image: ('jpeg', bytes),
+ (editorState) {
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ImageBlockKeys.type);
+ expect(node.attributes[ImageBlockKeys.url], isNotEmpty);
+ });
+ });
+
+ testWidgets('paste the image url', (tester) async {
+ const plainText = 'http://example.com/1.jpg';
+ final image = await rootBundle.load('assets/test/images/sample.jpeg');
+ final bytes = image.buffer.asUint8List();
+ await tester.pasteContent(plainText: plainText, image: ('jpeg', bytes),
+ (editorState) {
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ImageBlockKeys.type);
+ expect(node.attributes[ImageBlockKeys.url], isNotEmpty);
+ });
+ });
+
+ const testMarkdownText = '''
+# I'm h1
+## I'm h2
+### I'm h3
+#### I'm h4
+##### I'm h5
+###### I'm h6''';
+
+ testWidgets('paste markdowns', (tester) async {
+ await tester.pasteContent(
+ plainText: testMarkdownText,
(editorState) {
- final node = editorState.getNodeAtPath([0])!;
- expect(node.type, ImageBlockKeys.type);
- expect(node.attributes[ImageBlockKeys.url], isNotEmpty);
+ final children = editorState.document.root.children;
+ expect(children.length, 6);
+ for (int i = 1; i <= children.length; i++) {
+ final text = children[i - 1].delta!.toPlainText();
+ expect(text, 'I\'m h$i');
+ }
+ },
+ );
+ });
+
+ testWidgets('paste markdowns as plain', (tester) async {
+ await tester.pasteContent(
+ plainText: testMarkdownText,
+ pasteAsPlain: true,
+ (editorState) {
+ final children = editorState.document.root.children;
+ expect(children.length, 6);
+ for (int i = 1; i <= children.length; i++) {
+ final text = children[i - 1].delta!.toPlainText();
+ final expectText = '${'#' * i} I\'m h$i';
+ expect(text, expectText);
+ }
+ },
+ );
});
});
}
extension on WidgetTester {
Future pasteContent(
- void Function(EditorState editorState) test, {
+ FutureOr Function(EditorState editorState) test, {
Future Function(EditorState editorState)? beforeTest,
String? plainText,
String? html,
String? inAppJson,
+ bool pasteAsPlain = false,
(String, Uint8List?)? image,
}) async {
await initializeAppFlowy();
await tapAnonymousSignInButton();
// create a new document
- await createNewPageWithNameUnderParent(name: 'Test Document');
+ await createNewPageWithNameUnderParent();
// tap the editor
await tapButton(find.byType(AppFlowyEditor));
@@ -502,10 +557,11 @@ extension on WidgetTester {
await simulateKeyEvent(
LogicalKeyboardKey.keyV,
isControlPressed: Platform.isLinux || Platform.isWindows,
+ isShiftPressed: pasteAsPlain,
isMetaPressed: Platform.isMacOS,
);
- await pumpAndSettle();
+ await pumpAndSettle(const Duration(milliseconds: 1000));
- test(editor.getCurrentEditorState());
+ await test(editor.getCurrentEditorState());
}
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_create_and_delete_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_create_and_delete_test.dart
index 43320509ce..c2e00a4b48 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_create_and_delete_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_create_and_delete_test.dart
@@ -13,6 +13,8 @@ void main() {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
+ final finder = find.text(gettingStarted, findRichText: true);
+ await tester.pumpUntilFound(finder, timeout: const Duration(seconds: 2));
// create a new document
const pageName = 'Test Document';
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_find_menu_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_find_menu_test.dart
new file mode 100644
index 0000000000..6212e7d9cf
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_find_menu_test.dart
@@ -0,0 +1,160 @@
+import 'dart:math';
+
+import 'package:appflowy/generated/flowy_svgs.g.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/copy_and_paste/clipboard_service.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
+import 'package:appflowy/startup/startup.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter/services.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+import 'package:universal_platform/universal_platform.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ String generateRandomString(int len) {
+ final r = Random();
+ return String.fromCharCodes(
+ List.generate(len, (index) => r.nextInt(33) + 89),
+ );
+ }
+
+ testWidgets(
+ 'document find menu test',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new document
+ await tester.createNewPageWithNameUnderParent();
+
+ // tap editor to get focus
+ await tester.tapButton(find.byType(AppFlowyEditor));
+
+ // set clipboard data
+ final data = [
+ "123456\n\n",
+ ...List.generate(100, (_) => "${generateRandomString(50)}\n\n"),
+ "1234567\n\n",
+ ...List.generate(100, (_) => "${generateRandomString(50)}\n\n"),
+ "12345678\n\n",
+ ...List.generate(100, (_) => "${generateRandomString(50)}\n\n"),
+ ].join();
+ await getIt().setData(
+ ClipboardServiceData(
+ plainText: data,
+ ),
+ );
+
+ // paste
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed:
+ UniversalPlatform.isLinux || UniversalPlatform.isWindows,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ // go back to beginning of document
+ // FIXME: Cannot run Ctrl+F unless selection is on screen
+ await tester.editor
+ .updateSelection(Selection.collapsed(Position(path: [0])));
+ await tester.pumpAndSettle();
+
+ expect(find.byType(FindAndReplaceMenuWidget), findsNothing);
+
+ // press cmd/ctrl+F to display the find menu
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyF,
+ isControlPressed:
+ UniversalPlatform.isLinux || UniversalPlatform.isWindows,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(FindAndReplaceMenuWidget), findsOneWidget);
+
+ final textField = find.descendant(
+ of: find.byType(FindAndReplaceMenuWidget),
+ matching: find.byType(TextField),
+ );
+
+ await tester.enterText(
+ textField,
+ "123456",
+ );
+ await tester.pumpAndSettle();
+ await tester.pumpAndSettle();
+
+ expect(
+ find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.text("123456", findRichText: true),
+ ),
+ findsOneWidget,
+ );
+
+ await tester.testTextInput.receiveAction(TextInputAction.done);
+ await tester.pumpAndSettle();
+ await tester.pumpAndSettle();
+
+ expect(
+ find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.text("1234567", findRichText: true),
+ ),
+ findsOneWidget,
+ );
+
+ await tester.showKeyboard(textField);
+ await tester.idle();
+ await tester.testTextInput.receiveAction(TextInputAction.done);
+ await tester.pumpAndSettle();
+ await tester.pumpAndSettle();
+
+ expect(
+ find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.text("12345678", findRichText: true),
+ ),
+ findsOneWidget,
+ );
+
+ // tap next button, go back to beginning of document
+ await tester.tapButton(
+ find.descendant(
+ of: find.byType(FindMenu),
+ matching: find.byFlowySvg(FlowySvgs.arrow_down_s),
+ ),
+ );
+
+ expect(
+ find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.text("123456", findRichText: true),
+ ),
+ findsOneWidget,
+ );
+
+ /// press cmd/ctrl+F to display the find menu
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyF,
+ isControlPressed:
+ UniversalPlatform.isLinux || UniversalPlatform.isWindows,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(FindAndReplaceMenuWidget), findsOneWidget);
+
+ /// press esc to dismiss the find menu
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ await tester.pumpAndSettle();
+ expect(find.byType(FindAndReplaceMenuWidget), findsNothing);
+ },
+ );
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_inline_sub_page_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_inline_sub_page_test.dart
index ef9ef73b3c..30e115774a 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_inline_sub_page_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_inline_sub_page_test.dart
@@ -2,12 +2,12 @@ import 'dart:io';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/mention/mention_page_block.dart';
+import 'package:appflowy/plugins/inline_actions/inline_actions_menu.dart';
import 'package:appflowy/workspace/presentation/home/menu/view/view_action_type.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/services.dart';
-
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -330,6 +330,23 @@ void main() {
expect(find.text("$_createdPageName (copy)"), findsNWidgets(2));
expect(find.text("$_createdPageName (copy) (copy)"), findsOneWidget);
});
+
+ testWidgets('Cancel inline page reference menu by space', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createOpenRenameDocumentUnderParent(name: _firstDocName);
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.editor.showPlusMenu();
+
+ // Cancel by space
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.space,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(InlineActionsMenu), findsNothing);
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_link_preview_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_link_preview_test.dart
new file mode 100644
index 0000000000..39f8bfd4f6
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_link_preview_test.dart
@@ -0,0 +1,453 @@
+import 'dart:io';
+
+import 'package:appflowy/generated/flowy_svgs.g.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/copy_and_paste/clipboard_service.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/desktop_toolbar/link/link_hover_menu.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/link_embed/link_embed_block_component.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/link_embed/link_embed_menu.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/link_preview/custom_link_preview_block_component.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/link_preview/link_preview_menu.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/link_preview/paste_as/paste_as_menu.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/mention/mention_block.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/mention/mention_link_block.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/mention/mention_link_error_preview.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/mention/mention_link_preview.dart';
+import 'package:appflowy/startup/startup.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:appflowy_editor_plugins/appflowy_editor_plugins.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter/services.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ const avaliableLink = 'https://appflowy.io/',
+ unavailableLink = 'www.thereIsNoting.com';
+
+ Future preparePage(WidgetTester tester, {String? pageName}) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: pageName);
+ await tester.editor.tapLineOfEditorAt(0);
+ }
+
+ Future pasteLink(WidgetTester tester, String link) async {
+ await getIt()
+ .setData(ClipboardServiceData(plainText: link));
+
+ /// paste the link
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle(Duration(seconds: 1));
+ }
+
+ Future pasteAs(
+ WidgetTester tester,
+ String link,
+ PasteMenuType type, {
+ Duration waitTime = const Duration(milliseconds: 500),
+ }) async {
+ await pasteLink(tester, link);
+ final convertToMentionButton = find.text(type.title);
+ await tester.tapButton(convertToMentionButton);
+ await tester.pumpAndSettle(waitTime);
+ }
+
+ void checkUrl(Node node, String link) {
+ expect(node.type, ParagraphBlockKeys.type);
+ expect(node.delta!.toJson(), [
+ {
+ 'insert': link,
+ 'attributes': {'href': link},
+ }
+ ]);
+ }
+
+ void checkMention(Node node, String link) {
+ final delta = node.delta!;
+ final insert = (delta.first as TextInsert).text;
+ final attributes = delta.first.attributes;
+ expect(insert, MentionBlockKeys.mentionChar);
+ final mention =
+ attributes?[MentionBlockKeys.mention] as Map;
+ expect(mention[MentionBlockKeys.type], MentionType.externalLink.name);
+ expect(mention[MentionBlockKeys.url], avaliableLink);
+ }
+
+ void checkBookmark(Node node, String link) {
+ expect(node.type, LinkPreviewBlockKeys.type);
+ expect(node.attributes[LinkPreviewBlockKeys.url], link);
+ }
+
+ void checkEmbed(Node node, String link) {
+ expect(node.type, LinkPreviewBlockKeys.type);
+ expect(node.attributes[LinkEmbedKeys.previewType], LinkEmbedKeys.embed);
+ expect(node.attributes[LinkPreviewBlockKeys.url], link);
+ }
+
+ group('Paste as URL', () {
+ Future pasteAndTurnInto(
+ WidgetTester tester,
+ String link,
+ String title,
+ ) async {
+ await pasteLink(tester, link);
+ final convertToLinkButton = find
+ .text(LocaleKeys.document_plugins_linkPreview_typeSelection_URL.tr());
+ await tester.tapButton(convertToLinkButton);
+
+ /// hover link and turn into mention
+ await tester.hoverOnWidget(
+ find.byType(LinkHoverTrigger),
+ onHover: () async {
+ final turnintoButton = find.byFlowySvg(FlowySvgs.turninto_m);
+ await tester.tapButton(turnintoButton);
+ final convertToButton = find.text(title);
+ await tester.tapButton(convertToButton);
+ await tester.pumpAndSettle(Duration(seconds: 1));
+ },
+ );
+ }
+
+ testWidgets('paste a link', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteLink(tester, link);
+ final convertToLinkButton = find
+ .text(LocaleKeys.document_plugins_linkPreview_typeSelection_URL.tr());
+ await tester.tapButton(convertToLinkButton);
+ final node = tester.editor.getNodeAtPath([0]);
+ checkUrl(node, link);
+ });
+
+ testWidgets('paste a link and turn into mention', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAndTurnInto(
+ tester,
+ link,
+ LinkConvertMenuCommand.toMention.title,
+ );
+
+ /// check metion values
+ final node = tester.editor.getNodeAtPath([0]);
+ checkMention(node, link);
+ });
+
+ testWidgets('paste a link and turn into bookmark', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAndTurnInto(
+ tester,
+ link,
+ LinkConvertMenuCommand.toBookmark.title,
+ );
+
+ /// check metion values
+ final node = tester.editor.getNodeAtPath([0]);
+ checkBookmark(node, link);
+ });
+
+ testWidgets('paste a link and turn into embed', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAndTurnInto(
+ tester,
+ link,
+ LinkConvertMenuCommand.toEmbed.title,
+ );
+
+ /// check metion values
+ final node = tester.editor.getNodeAtPath([0]);
+ checkEmbed(node, link);
+ });
+ });
+
+ group('Paste as Mention', () {
+ Future pasteAsMention(WidgetTester tester, String link) =>
+ pasteAs(tester, link, PasteMenuType.mention);
+
+ String getMentionLink(Node node) {
+ final insert = node.delta?.first as TextInsert?;
+ final mention = insert?.attributes?[MentionBlockKeys.mention]
+ as Map?;
+ return mention?[MentionBlockKeys.url] ?? '';
+ }
+
+ Future hoverMentionAndClick(
+ WidgetTester tester,
+ String command,
+ ) async {
+ final mentionLink = find.byType(MentionLinkBlock);
+ expect(mentionLink, findsOneWidget);
+ await tester.hoverOnWidget(
+ mentionLink,
+ onHover: () async {
+ final errorPreview = find.byType(MentionLinkErrorPreview);
+ expect(errorPreview, findsOneWidget);
+ final convertButton = find.byFlowySvg(FlowySvgs.turninto_m);
+ await tester.tapButton(convertButton);
+ final menuButton = find.text(command);
+ await tester.tapButton(menuButton);
+ },
+ );
+ }
+
+ testWidgets('paste a link as mention', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsMention(tester, link);
+ final node = tester.editor.getNodeAtPath([0]);
+ checkMention(node, link);
+ });
+
+ testWidgets('paste as mention and copy link', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsMention(tester, link);
+ final mentionLink = find.byType(MentionLinkBlock);
+ expect(mentionLink, findsOneWidget);
+ await tester.hoverOnWidget(
+ mentionLink,
+ onHover: () async {
+ final preview = find.byType(MentionLinkPreview);
+ if (!preview.hasFound) {
+ final copyButton = find.byFlowySvg(FlowySvgs.toolbar_link_m);
+ await tester.tapButton(copyButton);
+ } else {
+ final moreOptionButton = find.byFlowySvg(FlowySvgs.toolbar_more_m);
+ await tester.tapButton(moreOptionButton);
+ final copyButton =
+ find.text(MentionLinktMenuCommand.copyLink.title);
+ await tester.tapButton(copyButton);
+ }
+ },
+ );
+ final clipboardContent = await getIt().getData();
+ expect(clipboardContent.plainText, link);
+ });
+
+ testWidgets('paste as error mention and turninto url', (tester) async {
+ String link = unavailableLink;
+ await preparePage(tester);
+ await pasteAsMention(tester, link);
+ Node node = tester.editor.getNodeAtPath([0]);
+ link = getMentionLink(node);
+ await hoverMentionAndClick(
+ tester,
+ MentionLinktErrorMenuCommand.toURL.title,
+ );
+ node = tester.editor.getNodeAtPath([0]);
+ checkUrl(node, link);
+ });
+
+ testWidgets('paste as error mention and turninto embed', (tester) async {
+ String link = unavailableLink;
+ await preparePage(tester);
+ await pasteAsMention(tester, link);
+ Node node = tester.editor.getNodeAtPath([0]);
+ link = getMentionLink(node);
+ await hoverMentionAndClick(
+ tester,
+ MentionLinktErrorMenuCommand.toEmbed.title,
+ );
+ node = tester.editor.getNodeAtPath([0]);
+ checkEmbed(node, link);
+ });
+
+ testWidgets('paste as error mention and turninto bookmark', (tester) async {
+ String link = unavailableLink;
+ await preparePage(tester);
+ await pasteAsMention(tester, link);
+ Node node = tester.editor.getNodeAtPath([0]);
+ link = getMentionLink(node);
+ await hoverMentionAndClick(
+ tester,
+ MentionLinktErrorMenuCommand.toBookmark.title,
+ );
+ node = tester.editor.getNodeAtPath([0]);
+ checkBookmark(node, link);
+ });
+
+ testWidgets('paste as error mention and remove link', (tester) async {
+ String link = unavailableLink;
+ await preparePage(tester);
+ await pasteAsMention(tester, link);
+ Node node = tester.editor.getNodeAtPath([0]);
+ link = getMentionLink(node);
+ await hoverMentionAndClick(
+ tester,
+ MentionLinktErrorMenuCommand.removeLink.title,
+ );
+ node = tester.editor.getNodeAtPath([0]);
+ expect(node.type, ParagraphBlockKeys.type);
+ expect(node.delta!.toJson(), [
+ {'insert': link},
+ ]);
+ });
+ });
+
+ group('Paste as Bookmark', () {
+ Future pasteAsBookmark(WidgetTester tester, String link) =>
+ pasteAs(tester, link, PasteMenuType.bookmark);
+
+ Future hoverAndClick(
+ WidgetTester tester,
+ LinkPreviewMenuCommand command,
+ ) async {
+ final bookmark = find.byType(CustomLinkPreviewBlockComponent);
+ expect(bookmark, findsOneWidget);
+ await tester.hoverOnWidget(
+ bookmark,
+ onHover: () async {
+ final menuButton = find.byFlowySvg(FlowySvgs.toolbar_more_m);
+ await tester.tapButton(menuButton);
+ final commandButton = find.text(command.title);
+ await tester.tapButton(commandButton);
+ },
+ );
+ }
+
+ testWidgets('paste a link as bookmark', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsBookmark(tester, link);
+ final node = tester.editor.getNodeAtPath([0]);
+ checkBookmark(node, link);
+ });
+
+ testWidgets('paste a link as bookmark and convert to mention',
+ (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsBookmark(tester, link);
+ await hoverAndClick(tester, LinkPreviewMenuCommand.convertToMention);
+ final node = tester.editor.getNodeAtPath([0]);
+ checkMention(node, link);
+ });
+
+ testWidgets('paste a link as bookmark and convert to url', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsBookmark(tester, link);
+ await hoverAndClick(tester, LinkPreviewMenuCommand.convertToUrl);
+ final node = tester.editor.getNodeAtPath([0]);
+ checkUrl(node, link);
+ });
+
+ testWidgets('paste a link as bookmark and convert to embed',
+ (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsBookmark(tester, link);
+ await hoverAndClick(tester, LinkPreviewMenuCommand.convertToEmbed);
+ final node = tester.editor.getNodeAtPath([0]);
+ checkEmbed(node, link);
+ });
+
+ testWidgets('paste a link as bookmark and copy link', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsBookmark(tester, link);
+ await hoverAndClick(tester, LinkPreviewMenuCommand.copyLink);
+ final clipboardContent = await getIt().getData();
+ expect(clipboardContent.plainText, link);
+ });
+
+ testWidgets('paste a link as bookmark and replace link', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsBookmark(tester, link);
+ await hoverAndClick(tester, LinkPreviewMenuCommand.replace);
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyA,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.simulateKeyEvent(LogicalKeyboardKey.delete);
+ await tester.enterText(find.byType(TextFormField), unavailableLink);
+ await tester.tapButton(find.text(LocaleKeys.button_replace.tr()));
+ final node = tester.editor.getNodeAtPath([0]);
+ checkBookmark(node, unavailableLink);
+ });
+
+ testWidgets('paste a link as bookmark and remove link', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsBookmark(tester, link);
+ await hoverAndClick(tester, LinkPreviewMenuCommand.removeLink);
+ final node = tester.editor.getNodeAtPath([0]);
+ expect(node.type, ParagraphBlockKeys.type);
+ expect(node.delta!.toJson(), [
+ {'insert': link},
+ ]);
+ });
+ });
+ group('Paste as Embed', () {
+ Future pasteAsEmbed(WidgetTester tester, String link) =>
+ pasteAs(tester, link, PasteMenuType.embed);
+
+ Future hoverAndConvert(
+ WidgetTester tester,
+ LinkEmbedConvertCommand command,
+ ) async {
+ final embed = find.byType(LinkEmbedBlockComponent);
+ expect(embed, findsOneWidget);
+ await tester.hoverOnWidget(
+ embed,
+ onHover: () async {
+ final menuButton = find.byFlowySvg(FlowySvgs.turninto_m);
+ await tester.tapButton(menuButton);
+ final commandButton = find.text(command.title);
+ await tester.tapButton(commandButton);
+ },
+ );
+ }
+
+ testWidgets('paste a link as embed', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsEmbed(tester, link);
+ final node = tester.editor.getNodeAtPath([0]);
+ checkEmbed(node, link);
+ });
+
+ testWidgets('paste a link as bookmark and convert to mention',
+ (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsEmbed(tester, link);
+ await hoverAndConvert(tester, LinkEmbedConvertCommand.toMention);
+ final node = tester.editor.getNodeAtPath([0]);
+ checkMention(node, link);
+ });
+
+ testWidgets('paste a link as bookmark and convert to url', (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsEmbed(tester, link);
+ await hoverAndConvert(tester, LinkEmbedConvertCommand.toURL);
+ final node = tester.editor.getNodeAtPath([0]);
+ checkUrl(node, link);
+ });
+
+ testWidgets('paste a link as bookmark and convert to bookmark',
+ (tester) async {
+ final link = avaliableLink;
+ await preparePage(tester);
+ await pasteAsEmbed(tester, link);
+ await hoverAndConvert(tester, LinkEmbedConvertCommand.toBookmark);
+ final node = tester.editor.getNodeAtPath([0]);
+ checkBookmark(node, link);
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_more_actions_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_more_actions_test.dart
index d4cc11d7f0..eeb2ea3925 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_more_actions_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_more_actions_test.dart
@@ -1,4 +1,10 @@
+import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/workspace/presentation/home/menu/view/view_item.dart';
+import 'package:appflowy/workspace/presentation/widgets/more_view_actions/widgets/view_meta_info.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flowy_infra_ui/style_widget/text.dart';
+import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -31,4 +37,104 @@ void main() {
expect(pageFinder, findsNWidgets(1));
});
});
+
+ testWidgets('count title towards word count', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent();
+
+ Finder title = tester.editor.findDocumentTitle('');
+
+ await tester.openMoreViewActions();
+ final viewMetaInfo = find.byType(ViewMetaInfo);
+ expect(viewMetaInfo, findsOneWidget);
+
+ ViewMetaInfo viewMetaInfoWidget =
+ viewMetaInfo.evaluate().first.widget as ViewMetaInfo;
+ Counters titleCounter = viewMetaInfoWidget.titleCounters!;
+
+ expect(titleCounter.charCount, 0);
+ expect(titleCounter.wordCount, 0);
+
+ /// input [str1] within title
+ const str1 = 'Hello',
+ str2 = '$str1 AppFlowy',
+ str3 = '$str2!',
+ str4 = 'Hello world';
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ await tester.tapButton(title);
+ await tester.enterText(title, str1);
+ await tester.pumpAndSettle(const Duration(seconds: 1));
+ await tester.openMoreViewActions();
+ viewMetaInfoWidget = viewMetaInfo.evaluate().first.widget as ViewMetaInfo;
+ titleCounter = viewMetaInfoWidget.titleCounters!;
+ expect(titleCounter.charCount, str1.length);
+ expect(titleCounter.wordCount, 1);
+
+ /// input [str2] within title
+ title = tester.editor.findDocumentTitle(str1);
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ await tester.tapButton(title);
+ await tester.enterText(title, str2);
+ await tester.pumpAndSettle(const Duration(seconds: 1));
+ await tester.openMoreViewActions();
+ viewMetaInfoWidget = viewMetaInfo.evaluate().first.widget as ViewMetaInfo;
+ titleCounter = viewMetaInfoWidget.titleCounters!;
+ expect(titleCounter.charCount, str2.length);
+ expect(titleCounter.wordCount, 2);
+
+ /// input [str3] within title
+ title = tester.editor.findDocumentTitle(str2);
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ await tester.tapButton(title);
+ await tester.enterText(title, str3);
+ await tester.pumpAndSettle(const Duration(seconds: 1));
+ await tester.openMoreViewActions();
+ viewMetaInfoWidget = viewMetaInfo.evaluate().first.widget as ViewMetaInfo;
+ titleCounter = viewMetaInfoWidget.titleCounters!;
+ expect(titleCounter.charCount, str3.length);
+ expect(titleCounter.wordCount, 2);
+
+ /// input [str4] within document
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ await tester.editor
+ .updateSelection(Selection.collapsed(Position(path: [0])));
+ await tester.pumpAndSettle();
+ await tester.editor
+ .getCurrentEditorState()
+ .insertTextAtCurrentSelection(str4);
+ await tester.pumpAndSettle(const Duration(seconds: 1));
+ await tester.openMoreViewActions();
+ final texts =
+ find.descendant(of: viewMetaInfo, matching: find.byType(FlowyText));
+ expect(texts, findsNWidgets(3));
+ viewMetaInfoWidget = viewMetaInfo.evaluate().first.widget as ViewMetaInfo;
+ titleCounter = viewMetaInfoWidget.titleCounters!;
+ final Counters documentCounters = viewMetaInfoWidget.documentCounters!;
+ final wordCounter = texts.evaluate().elementAt(0).widget as FlowyText,
+ charCounter = texts.evaluate().elementAt(1).widget as FlowyText;
+ final numberFormat = NumberFormat();
+ expect(
+ wordCounter.text,
+ LocaleKeys.moreAction_wordCount.tr(
+ args: [
+ numberFormat
+ .format(titleCounter.wordCount + documentCounters.wordCount)
+ .toString(),
+ ],
+ ),
+ );
+ expect(
+ charCounter.text,
+ LocaleKeys.moreAction_charCount.tr(
+ args: [
+ numberFormat
+ .format(
+ titleCounter.charCount + documentCounters.charCount,
+ )
+ .toString(),
+ ],
+ ),
+ );
+ });
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_option_action_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_option_action_test.dart
index cbc634cf02..6ec12287a8 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_option_action_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_option_action_test.dart
@@ -76,13 +76,12 @@ void main() {
LocaleKeys.document_slashMenu_name_heading1.tr(): HeadingBlockKeys.type,
LocaleKeys.document_slashMenu_name_heading2.tr(): HeadingBlockKeys.type,
LocaleKeys.document_slashMenu_name_heading3.tr(): HeadingBlockKeys.type,
- LocaleKeys.document_slashMenu_name_bulletedList.tr():
+ LocaleKeys.editor_bulletedListShortForm.tr():
BulletedListBlockKeys.type,
- LocaleKeys.document_slashMenu_name_numberedList.tr():
+ LocaleKeys.editor_numberedListShortForm.tr():
NumberedListBlockKeys.type,
LocaleKeys.document_slashMenu_name_quote.tr(): QuoteBlockKeys.type,
- LocaleKeys.document_slashMenu_name_todoList.tr():
- TodoListBlockKeys.type,
+ LocaleKeys.editor_checkbox.tr(): TodoListBlockKeys.type,
LocaleKeys.document_slashMenu_name_callout.tr(): CalloutBlockKeys.type,
LocaleKeys.document_slashMenu_name_text.tr(): ParagraphBlockKeys.type,
};
@@ -117,13 +116,12 @@ void main() {
LocaleKeys.document_slashMenu_name_heading1.tr(): HeadingBlockKeys.type,
LocaleKeys.document_slashMenu_name_heading2.tr(): HeadingBlockKeys.type,
LocaleKeys.document_slashMenu_name_heading3.tr(): HeadingBlockKeys.type,
- LocaleKeys.document_slashMenu_name_bulletedList.tr():
+ LocaleKeys.editor_bulletedListShortForm.tr():
BulletedListBlockKeys.type,
- LocaleKeys.document_slashMenu_name_numberedList.tr():
+ LocaleKeys.editor_numberedListShortForm.tr():
NumberedListBlockKeys.type,
LocaleKeys.document_slashMenu_name_quote.tr(): QuoteBlockKeys.type,
- LocaleKeys.document_slashMenu_name_todoList.tr():
- TodoListBlockKeys.type,
+ LocaleKeys.editor_checkbox.tr(): TodoListBlockKeys.type,
LocaleKeys.document_slashMenu_name_callout.tr(): CalloutBlockKeys.type,
LocaleKeys.document_slashMenu_name_text.tr(): ParagraphBlockKeys.type,
};
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_selection_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_selection_test.dart
index bd0fd18c50..de1cb880a5 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_selection_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_selection_test.dart
@@ -1,5 +1,6 @@
import 'package:appflowy_editor/appflowy_editor.dart';
import 'package:flutter/material.dart';
+import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -47,5 +48,41 @@ void main() {
expect(editorState.selection!.start.offset, 0);
});
+
+ testWidgets('select and delete text', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ /// create a new document
+ await tester.createNewPageWithNameUnderParent();
+
+ /// input text
+ final editor = tester.editor;
+ final editorState = editor.getCurrentEditorState();
+
+ const inputText = 'Test for text selection and deletion';
+ final texts = inputText.split(' ');
+ await editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText(inputText);
+
+ /// selecte and delete
+ int index = 0;
+ while (texts.isNotEmpty) {
+ final text = texts.removeAt(0);
+ await tester.editor.updateSelection(
+ Selection(
+ start: Position(path: [0], offset: index),
+ end: Position(path: [0], offset: index + text.length),
+ ),
+ );
+ await tester.simulateKeyEvent(LogicalKeyboardKey.delete);
+ index++;
+ }
+
+ /// excpete the text value is correct
+ final node = editorState.getNodeAtPath([0])!;
+ final nodeText = node.delta?.toPlainText() ?? '';
+ expect(nodeText, ' ' * (index - 1));
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_sub_page_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_sub_page_test.dart
index efcb915468..50f0f903bc 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_sub_page_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_sub_page_test.dart
@@ -1,17 +1,19 @@
import 'dart:io';
-import 'package:appflowy/workspace/presentation/home/menu/view/view_action_type.dart';
-import 'package:flutter/material.dart';
-import 'package:flutter/services.dart';
-
import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/header/emoji_icon_widget.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/sub_page/sub_page_block_component.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
+import 'package:appflowy/workspace/presentation/home/menu/view/view_action_type.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
+import '../../shared/emoji.dart';
import '../../shared/util.dart';
// Test cases for the Document SubPageBlock that needs to be covered:
@@ -38,7 +40,14 @@ import '../../shared/util.dart';
const _defaultPageName = "";
void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
+
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
group('Document SubPageBlock tests', () {
testWidgets('Insert a new SubPageBlock from Slash menu items',
@@ -49,11 +58,6 @@ void main() {
await tester.insertSubPageFromSlashMenu();
- await tester.expandOrCollapsePage(
- pageName: 'SubPageBlock',
- layout: ViewLayoutPB.Document,
- );
-
expect(
find.text(LocaleKeys.menuAppHeader_defaultNewPageName.tr()),
findsNWidgets(3),
@@ -68,12 +72,6 @@ void main() {
await tester.insertSubPageFromSlashMenu();
- await tester.expandOrCollapsePage(
- pageName: 'SubPageBlock',
- layout: ViewLayoutPB.Document,
- );
- await tester.pumpAndSettle();
-
await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
expect(find.text('Child page'), findsNWidgets(2));
@@ -92,11 +90,6 @@ void main() {
await tester.insertSubPageFromSlashMenu();
- await tester.expandOrCollapsePage(
- pageName: 'SubPageBlock',
- layout: ViewLayoutPB.Document,
- );
-
await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
expect(find.text('Child page'), findsNWidgets(2));
@@ -145,11 +138,6 @@ void main() {
await tester.insertSubPageFromSlashMenu();
- await tester.expandOrCollapsePage(
- pageName: 'SubPageBlock',
- layout: ViewLayoutPB.Document,
- );
-
await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
expect(find.text('Child page'), findsNWidgets(2));
@@ -203,11 +191,6 @@ void main() {
await tester.insertSubPageFromSlashMenu();
- await tester.expandOrCollapsePage(
- pageName: 'SubPageBlock',
- layout: ViewLayoutPB.Document,
- );
-
await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
expect(find.text('Child page'), findsNWidgets(2));
@@ -244,11 +227,6 @@ void main() {
await tester.insertSubPageFromSlashMenu();
- await tester.expandOrCollapsePage(
- pageName: 'SubPageBlock',
- layout: ViewLayoutPB.Document,
- );
-
await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
expect(find.text('Child page'), findsNWidgets(2));
@@ -294,11 +272,6 @@ void main() {
await tester.insertSubPageFromSlashMenu();
- await tester.expandOrCollapsePage(
- pageName: 'SubPageBlock',
- layout: ViewLayoutPB.Document,
- );
-
await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
expect(find.text('Child page'), findsNWidgets(2));
@@ -337,11 +310,6 @@ void main() {
await tester.insertSubPageFromSlashMenu(true);
- await tester.expandOrCollapsePage(
- pageName: 'SubPageBlock',
- layout: ViewLayoutPB.Document,
- );
-
await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
expect(find.text('Child page'), findsNWidgets(2));
@@ -385,11 +353,6 @@ void main() {
await tester.insertSubPageFromSlashMenu();
- await tester.expandOrCollapsePage(
- pageName: 'SubPageBlock',
- layout: ViewLayoutPB.Document,
- );
-
await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
expect(find.text('Child page'), findsNWidgets(2));
expect(find.byType(SubPageBlockComponent), findsOneWidget);
@@ -412,12 +375,6 @@ void main() {
await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
await tester.insertSubPageFromSlashMenu();
-
- await tester.expandOrCollapsePage(
- pageName: 'SubPageBlock',
- layout: ViewLayoutPB.Document,
- );
-
await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
expect(find.text('Child page'), findsNWidgets(2));
@@ -438,11 +395,6 @@ void main() {
await tester.insertSubPageFromSlashMenu(true);
- await tester.expandOrCollapsePage(
- pageName: 'SubPageBlock',
- layout: ViewLayoutPB.Document,
- );
-
expect(find.byType(SubPageBlockComponent), findsOneWidget);
final beforeNode = tester.editor.getNodeAtPath([1]);
@@ -499,6 +451,43 @@ void main() {
expect(find.text('Parent'), findsNWidgets(2));
});
+
+ testWidgets('Displaying icon of subpage', (tester) async {
+ const firstPage = 'FirstPage';
+
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: firstPage);
+ final icon = await tester.loadIcon();
+
+ /// create subpage
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.editor.showSlashMenu();
+ await tester.editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_subPage_name.tr(),
+ offset: 100,
+ );
+
+ /// add icon
+ await tester.editor.hoverOnCoverToolbar();
+ await tester.editor.tapAddIconButton();
+ await tester.tapIcon(icon);
+ await tester.pumpAndSettle();
+ await tester.openPage(firstPage);
+
+ await tester.expandOrCollapsePage(
+ pageName: firstPage,
+ layout: ViewLayoutPB.Document,
+ );
+
+ /// check if there is a icon in document
+ final iconWidget = find.byWidgetPredicate((w) {
+ if (w is! RawEmojiIconWidget) return false;
+ final iconData = w.emoji.emoji;
+ return iconData == icon.emoji;
+ });
+ expect(iconWidget, findsOneWidget);
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_4.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_4.dart
index cfe2e640eb..bc0671834b 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_4.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_4.dart
@@ -1,14 +1,19 @@
import 'package:integration_test/integration_test.dart';
+import 'document_block_option_test.dart' as document_block_option_test;
+import 'document_find_menu_test.dart' as document_find_menu_test;
import 'document_inline_page_reference_test.dart'
as document_inline_page_reference_test;
import 'document_more_actions_test.dart' as document_more_actions_test;
import 'document_shortcuts_test.dart' as document_shortcuts_test;
+import 'document_toolbar_test.dart' as document_toolbar_test;
import 'document_with_file_test.dart' as document_with_file_test;
import 'document_with_image_block_test.dart' as document_with_image_block_test;
import 'document_with_multi_image_block_test.dart'
as document_with_multi_image_block_test;
-import 'document_block_option_test.dart' as document_block_option_test;
+import 'document_with_simple_table_test.dart'
+ as document_with_simple_table_test;
+import 'document_link_preview_test.dart' as document_link_preview_test;
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
@@ -21,4 +26,8 @@ void main() {
document_with_file_test.main();
document_shortcuts_test.main();
document_block_option_test.main();
+ document_find_menu_test.main();
+ document_toolbar_test.main();
+ document_with_simple_table_test.main();
+ document_link_preview_test.main();
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_title_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_title_test.dart
index 9b4b5e9ef7..c694ba8d6b 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_title_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_title_test.dart
@@ -347,5 +347,27 @@ void main() {
await tester.pumpAndSettle();
});
+
+ testWidgets('paste text in title, check if the text is updated',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent();
+
+ await Clipboard.setData(const ClipboardData(text: _testDocumentName));
+
+ final title = tester.editor.findDocumentTitle('');
+ await tester.tapButton(title);
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ isControlPressed: !UniversalPlatform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ final newTitle = tester.editor.findDocumentTitle(_testDocumentName);
+ expect(newTitle, findsOneWidget);
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_toolbar_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_toolbar_test.dart
new file mode 100644
index 0000000000..f455cd479d
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_toolbar_test.dart
@@ -0,0 +1,370 @@
+import 'package:appflowy/generated/flowy_svgs.g.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/copy_and_paste/clipboard_service.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/desktop_toolbar/desktop_floating_toolbar.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/desktop_toolbar/link/link_create_menu.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/desktop_toolbar/link/link_edit_menu.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/desktop_toolbar/link/link_hover_menu.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/toolbar_item/custom_link_toolbar_item.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/toolbar_item/more_option_toolbar_item.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/toolbar_item/text_suggestions_toolbar_item.dart';
+import 'package:appflowy/startup/startup.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter/services.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ Future selectText(WidgetTester tester, String text) async {
+ await tester.editor.updateSelection(
+ Selection.single(
+ path: [0],
+ startOffset: 0,
+ endOffset: text.length,
+ ),
+ );
+ }
+
+ Future prepareForToolbar(WidgetTester tester, String text) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent();
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText(text);
+ await selectText(tester, text);
+ }
+
+ group('document toolbar:', () {
+ testWidgets('font family', (tester) async {
+ await prepareForToolbar(tester, 'font family');
+
+ // tap more options button
+ await tester.tapButtonWithFlowySvgData(FlowySvgs.toolbar_more_m);
+ // tap the font family button
+ final fontFamilyButton = find.byKey(kFontFamilyToolbarItemKey);
+ await tester.tapButton(fontFamilyButton);
+
+ // expect to see the font family dropdown immediately
+ expect(find.byType(FontFamilyDropDown), findsOneWidget);
+
+ // click the font family 'Abel'
+ const abel = 'Abel';
+ await tester.tapButton(find.text(abel));
+
+ // check the text is updated to 'Abel'
+ final editorState = tester.editor.getCurrentEditorState();
+ expect(
+ editorState.getDeltaAttributeValueInSelection(
+ AppFlowyRichTextKeys.fontFamily,
+ ),
+ abel,
+ );
+ });
+
+ testWidgets('heading 1~3', (tester) async {
+ const text = 'heading';
+ await prepareForToolbar(tester, text);
+
+ Future testChangeHeading(
+ FlowySvgData svg,
+ String title,
+ int level,
+ ) async {
+ /// tap suggestions item
+ final suggestionsButton = find.byKey(kSuggestionsItemKey);
+ await tester.tapButton(suggestionsButton);
+
+ /// tap item
+ await tester.ensureVisible(find.byFlowySvg(svg));
+ await tester.tapButton(find.byFlowySvg(svg));
+
+ /// check the type of node is [HeadingBlockKeys.type]
+ await selectText(tester, text);
+ final editorState = tester.editor.getCurrentEditorState();
+ final selection = editorState.selection!;
+ final node = editorState.getNodeAtPath(selection.start.path)!,
+ nodeLevel = node.attributes[HeadingBlockKeys.level]!;
+ expect(node.type, HeadingBlockKeys.type);
+ expect(nodeLevel, level);
+
+ /// show toolbar again
+ await selectText(tester, text);
+
+ /// the text of suggestions item should be changed
+ expect(
+ find.descendant(of: suggestionsButton, matching: find.text(title)),
+ findsOneWidget,
+ );
+ }
+
+ await testChangeHeading(
+ FlowySvgs.type_h1_m,
+ LocaleKeys.document_toolbar_h1.tr(),
+ 1,
+ );
+
+ await testChangeHeading(
+ FlowySvgs.type_h2_m,
+ LocaleKeys.document_toolbar_h2.tr(),
+ 2,
+ );
+ await testChangeHeading(
+ FlowySvgs.type_h3_m,
+ LocaleKeys.document_toolbar_h3.tr(),
+ 3,
+ );
+ });
+
+ testWidgets('toggle 1~3', (tester) async {
+ const text = 'toggle';
+ await prepareForToolbar(tester, text);
+
+ Future testChangeToggle(
+ FlowySvgData svg,
+ String title,
+ int? level,
+ ) async {
+ /// tap suggestions item
+ final suggestionsButton = find.byKey(kSuggestionsItemKey);
+ await tester.tapButton(suggestionsButton);
+
+ /// tap item
+ await tester.ensureVisible(find.byFlowySvg(svg));
+ await tester.tapButton(find.byFlowySvg(svg));
+
+ /// check the type of node is [HeadingBlockKeys.type]
+ await selectText(tester, text);
+ final editorState = tester.editor.getCurrentEditorState();
+ final selection = editorState.selection!;
+ final node = editorState.getNodeAtPath(selection.start.path)!,
+ nodeLevel = node.attributes[ToggleListBlockKeys.level];
+ expect(node.type, ToggleListBlockKeys.type);
+ expect(nodeLevel, level);
+
+ /// show toolbar again
+ await selectText(tester, text);
+
+ /// the text of suggestions item should be changed
+ expect(
+ find.descendant(of: suggestionsButton, matching: find.text(title)),
+ findsOneWidget,
+ );
+ }
+
+ await testChangeToggle(
+ FlowySvgs.type_toggle_list_m,
+ LocaleKeys.editor_toggleListShortForm.tr(),
+ null,
+ );
+
+ await testChangeToggle(
+ FlowySvgs.type_toggle_h1_m,
+ LocaleKeys.editor_toggleHeading1ShortForm.tr(),
+ 1,
+ );
+
+ await testChangeToggle(
+ FlowySvgs.type_toggle_h2_m,
+ LocaleKeys.editor_toggleHeading2ShortForm.tr(),
+ 2,
+ );
+
+ await testChangeToggle(
+ FlowySvgs.type_toggle_h3_m,
+ LocaleKeys.editor_toggleHeading3ShortForm.tr(),
+ 3,
+ );
+ });
+
+ testWidgets('toolbar will not rebuild after click item', (tester) async {
+ const text = 'Test rebuilding';
+ await prepareForToolbar(tester, text);
+ Finder toolbar = find.byType(DesktopFloatingToolbar);
+ Element toolbarElement = toolbar.evaluate().first;
+ final elementHashcode = toolbarElement.hashCode;
+ final boldButton = find.byFlowySvg(FlowySvgs.toolbar_bold_m),
+ underlineButton = find.byFlowySvg(FlowySvgs.toolbar_underline_m),
+ italicButton = find.byFlowySvg(FlowySvgs.toolbar_inline_italic_m);
+
+ /// tap format buttons
+ await tester.tapButton(boldButton);
+ await tester.tapButton(underlineButton);
+ await tester.tapButton(italicButton);
+ toolbar = find.byType(DesktopFloatingToolbar);
+ toolbarElement = toolbar.evaluate().first;
+
+ /// check if the toolbar is not rebuilt
+ expect(elementHashcode, toolbarElement.hashCode);
+ final editorState = tester.editor.getCurrentEditorState();
+
+ /// check text formats
+ expect(
+ editorState
+ .getDeltaAttributeValueInSelection(AppFlowyRichTextKeys.bold),
+ true,
+ );
+ expect(
+ editorState
+ .getDeltaAttributeValueInSelection(AppFlowyRichTextKeys.italic),
+ true,
+ );
+ expect(
+ editorState
+ .getDeltaAttributeValueInSelection(AppFlowyRichTextKeys.underline),
+ true,
+ );
+ });
+ });
+
+ group('document toolbar: link', () {
+ String? getLinkFromNode(Node node) {
+ for (final insert in node.delta!) {
+ final link = insert.attributes?.href;
+ if (link != null) return link;
+ }
+ return null;
+ }
+
+ bool isPageLink(Node node) {
+ for (final insert in node.delta!) {
+ final isPage = insert.attributes?.isPage;
+ if (isPage == true) return true;
+ }
+ return false;
+ }
+
+ String getNodeText(Node node) {
+ for (final insert in node.delta!) {
+ if (insert is TextInsert) return insert.text;
+ }
+ return '';
+ }
+
+ testWidgets('insert link and remove link', (tester) async {
+ const text = 'insert link', link = 'https://test.appflowy.cloud';
+ await prepareForToolbar(tester, text);
+
+ final toolbar = find.byType(DesktopFloatingToolbar);
+ expect(toolbar, findsOneWidget);
+
+ /// tap link button to show CreateLinkMenu
+ final linkButton = find.byFlowySvg(FlowySvgs.toolbar_link_m);
+ await tester.tapButton(linkButton);
+ final createLinkMenu = find.byType(LinkCreateMenu);
+ expect(createLinkMenu, findsOneWidget);
+
+ /// test esc to close
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ expect(toolbar, findsNothing);
+
+ /// show toolbar again
+ await tester.editor.tapLineOfEditorAt(0);
+ await selectText(tester, text);
+ await tester.tapButton(linkButton);
+
+ /// insert link
+ final textField = find.descendant(
+ of: createLinkMenu,
+ matching: find.byType(TextFormField),
+ );
+ await tester.enterText(textField, link);
+ await tester.pumpAndSettle();
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ Node node = tester.editor.getNodeAtPath([0]);
+ expect(getLinkFromNode(node), link);
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+
+ /// hover link
+ await tester.hoverOnWidget(find.byType(LinkHoverTrigger));
+ final hoverMenu = find.byType(LinkHoverMenu);
+ expect(hoverMenu, findsOneWidget);
+
+ /// copy link
+ final copyButton = find.descendant(
+ of: hoverMenu,
+ matching: find.byFlowySvg(FlowySvgs.toolbar_link_m),
+ );
+ await tester.tapButton(copyButton);
+ final clipboardContent = await getIt().getData();
+ final plainText = clipboardContent.plainText;
+ expect(plainText, link);
+
+ /// remove link
+ await tester.hoverOnWidget(find.byType(LinkHoverTrigger));
+ await tester.tapButton(find.byFlowySvg(FlowySvgs.toolbar_link_unlink_m));
+ node = tester.editor.getNodeAtPath([0]);
+ expect(getLinkFromNode(node), null);
+ });
+
+ testWidgets('insert link and edit link', (tester) async {
+ const text = 'edit link',
+ link = 'https://test.appflowy.cloud',
+ afterText = '$text after';
+ await prepareForToolbar(tester, text);
+
+ /// tap link button to show CreateLinkMenu
+ final linkButton = find.byFlowySvg(FlowySvgs.toolbar_link_m);
+ await tester.tapButton(linkButton);
+
+ /// search for page and select it
+ final textField = find.descendant(
+ of: find.byType(LinkCreateMenu),
+ matching: find.byType(TextFormField),
+ );
+ await tester.enterText(textField, gettingStarted);
+ await tester.pumpAndSettle();
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+
+ Node node = tester.editor.getNodeAtPath([0]);
+ expect(isPageLink(node), true);
+ expect(getLinkFromNode(node) == link, false);
+
+ /// hover link
+ await tester.hoverOnWidget(find.byType(LinkHoverTrigger));
+
+ /// click edit button to show LinkEditMenu
+ final editButton = find.byFlowySvg(FlowySvgs.toolbar_link_edit_m);
+ await tester.tapButton(editButton);
+ final linkEditMenu = find.byType(LinkEditMenu);
+ expect(linkEditMenu, findsOneWidget);
+
+ /// change the link text
+ final titleField = find.descendant(
+ of: linkEditMenu,
+ matching: find.byType(TextFormField),
+ );
+ await tester.enterText(titleField, afterText);
+ await tester.pumpAndSettle();
+ await tester.tapButton(
+ find.descendant(of: linkEditMenu, matching: find.text(gettingStarted)),
+ );
+ final linkField = find.ancestor(
+ of: find.text(LocaleKeys.document_toolbar_linkInputHint.tr()),
+ matching: find.byType(TextFormField),
+ );
+ await tester.enterText(linkField, link);
+ await tester.pumpAndSettle();
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+
+ /// apply the change
+ final applyButton =
+ find.text(LocaleKeys.settings_appearance_documentSettings_apply.tr());
+ await tester.tapButton(applyButton);
+
+ node = tester.editor.getNodeAtPath([0]);
+ expect(isPageLink(node), false);
+ expect(getLinkFromNode(node), link);
+ expect(getNodeText(node), afterText);
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_cover_image_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_cover_image_test.dart
index c365c1bad5..84b6790403 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_cover_image_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_cover_image_test.dart
@@ -1,17 +1,34 @@
+import 'dart:io';
+
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/header/document_cover_widget.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/image/image_util.dart';
+import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/gestures.dart';
import 'package:flutter/material.dart';
+import 'package:flutter/services.dart';
import 'package:flutter_emoji_mart/flutter_emoji_mart.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
+import 'package:path/path.dart' as p;
+import 'package:path_provider/path_provider.dart';
import '../../shared/emoji.dart';
+import '../../shared/mock/mock_file_picker.dart';
import '../../shared/util.dart';
void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
+
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
group('cover image:', () {
testWidgets('document cover tests', (tester) async {
@@ -51,6 +68,59 @@ void main() {
tester.expectToSeeNoDocumentCover();
});
+ testWidgets('document cover local image tests', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ tester.expectToSeeNoDocumentCover();
+
+ // Hover over cover toolbar to show 'Add Cover' and 'Add Icon' buttons
+ await tester.editor.hoverOnCoverToolbar();
+
+ // Insert a document cover
+ await tester.editor.tapOnAddCover();
+ tester.expectToSeeDocumentCover(CoverType.asset);
+
+ // Hover over the cover to show the 'Change Cover' and delete buttons
+ await tester.editor.hoverOnCover();
+ tester.expectChangeCoverAndDeleteButton();
+
+ // Change cover to a local image image
+ final imagePath = await rootBundle.load('assets/test/images/sample.jpeg');
+ final tempDirectory = await getTemporaryDirectory();
+ final localImagePath = p.join(tempDirectory.path, 'sample.jpeg');
+ final imageFile = File(localImagePath)
+ ..writeAsBytesSync(imagePath.buffer.asUint8List());
+
+ await tester.editor.hoverOnCover();
+ await tester.editor.tapOnChangeCover();
+
+ final uploadButton = find.findTextInFlowyText(
+ LocaleKeys.document_imageBlock_upload_label.tr(),
+ );
+ await tester.tapButton(uploadButton);
+
+ mockPickFilePaths(paths: [localImagePath]);
+ await tester.tapButtonWithName(
+ LocaleKeys.document_imageBlock_upload_placeholder.tr(),
+ );
+
+ await tester.pumpAndSettle();
+ tester.expectToSeeDocumentCover(CoverType.file);
+
+ // Remove the cover
+ await tester.editor.hoverOnCover();
+ await tester.editor.tapOnRemoveCover();
+ tester.expectToSeeNoDocumentCover();
+
+ // Test if deleteImageFromLocalStorage(localImagePath) function is called once
+ await tester.pump(kDoubleTapTimeout);
+ expect(deleteImageTestCounter, 1);
+
+ // delete temp files
+ await imageFile.delete();
+ });
+
testWidgets('document icon tests', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
@@ -147,7 +217,7 @@ void main() {
tester.expectViewHasIcon(
gettingStarted,
ViewLayoutPB.Document,
- punch,
+ EmojiIconData.emoji(punch),
);
});
});
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_database_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_database_test.dart
index ede3627598..158eb501e3 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_database_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_database_test.dart
@@ -1,12 +1,15 @@
import 'package:appflowy/plugins/database/board/presentation/board_page.dart';
import 'package:appflowy/plugins/database/calendar/presentation/calendar_page.dart';
import 'package:appflowy/plugins/database/grid/presentation/grid_page.dart';
+import 'package:appflowy/plugins/database/grid/presentation/widgets/footer/grid_footer.dart';
+import 'package:appflowy/plugins/database/grid/presentation/widgets/row/row.dart';
import 'package:appflowy/plugins/database/widgets/cell/editable_cell_skeleton/text.dart';
import 'package:appflowy/plugins/inline_actions/widgets/inline_actions_handler.dart';
import 'package:appflowy/workspace/presentation/home/menu/view/view_item.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/protobuf.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
import 'package:flowy_infra/uuid.dart';
+import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -174,9 +177,110 @@ void main() {
findsOneWidget,
);
});
+
+ testWidgets('insert a referenced grid with many rows (load more option)',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await insertLinkedDatabase(tester, ViewLayoutPB.Grid);
+
+ // validate the referenced grid is inserted
+ expect(
+ find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.byType(GridPage),
+ ),
+ findsOneWidget,
+ );
+
+ // https://github.com/AppFlowy-IO/AppFlowy/issues/3533
+ // test: the selection of editor should be clear when editing the grid
+ await tester.editor.updateSelection(
+ Selection.collapsed(
+ Position(path: [1]),
+ ),
+ );
+ final gridTextCell = find.byType(EditableTextCell).first;
+ await tester.tapButton(gridTextCell);
+
+ expect(tester.editor.getCurrentEditorState().selection, isNull);
+
+ final editorScrollable = find
+ .descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.byWidgetPredicate(
+ (w) => w is Scrollable && w.axis == Axis.vertical,
+ ),
+ )
+ .first;
+
+ // Add 100 Rows to the linked database
+ final addRowFinder = find.byType(GridAddRowButton);
+ for (var i = 0; i < 100; i++) {
+ await tester.scrollUntilVisible(
+ addRowFinder,
+ 100,
+ scrollable: editorScrollable,
+ );
+ await tester.tapButton(addRowFinder);
+ await tester.pumpAndSettle();
+ }
+
+ // Since all rows visible are those we added, we should see all of them
+ expect(find.byType(GridRow), findsNWidgets(103));
+
+ // Navigate to getting started
+ await tester.openPage(gettingStarted);
+
+ // Navigate back to the document
+ await tester.openPage('insert_a_reference_${ViewLayoutPB.Grid.name}');
+
+ // We see only 25 Grid Rows
+ expect(find.byType(GridRow), findsNWidgets(25));
+
+ // We see Add row and load more button
+ expect(find.byType(GridAddRowButton), findsOneWidget);
+ expect(find.byType(GridRowLoadMoreButton), findsOneWidget);
+
+ // Load more rows, expect 50 visible
+ await _loadMoreRows(tester, editorScrollable, 50);
+
+ // Load more rows, expect 75 visible
+ await _loadMoreRows(tester, editorScrollable, 75);
+
+ // Load more rows, expect 100 visible
+ await _loadMoreRows(tester, editorScrollable, 100);
+
+ // Load more rows, expect 103 visible
+ await _loadMoreRows(tester, editorScrollable, 103);
+
+ // We no longer see load more option
+ expect(find.byType(GridRowLoadMoreButton), findsNothing);
+ });
});
}
+Future _loadMoreRows(
+ WidgetTester tester,
+ Finder scrollable, [
+ int? expectedRows,
+]) async {
+ await tester.scrollUntilVisible(
+ find.byType(GridRowLoadMoreButton),
+ 100,
+ scrollable: scrollable,
+ );
+ await tester.pumpAndSettle();
+
+ await tester.tap(find.byType(GridRowLoadMoreButton));
+ await tester.pumpAndSettle();
+
+ if (expectedRows != null) {
+ expect(find.byType(GridRow), findsNWidgets(expectedRows));
+ }
+}
+
/// Insert a referenced database of [layout] into the document
Future insertLinkedDatabase(
WidgetTester tester,
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_date_reminder_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_date_reminder_test.dart
index c7fea448f7..ccfdbae76e 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_date_reminder_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_date_reminder_test.dart
@@ -1,14 +1,21 @@
+import 'dart:io';
+
import 'package:appflowy/generated/flowy_svgs.g.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/mention/mention_date_block.dart';
+import 'package:appflowy/startup/startup.dart';
+import 'package:appflowy/user/application/reminder/reminder_bloc.dart';
import 'package:appflowy/workspace/application/settings/date_time/date_format_ext.dart';
import 'package:appflowy/workspace/presentation/widgets/date_picker/desktop_date_picker.dart';
import 'package:appflowy/workspace/presentation/widgets/toggle/toggle.dart';
import 'package:appflowy_backend/protobuf/flowy-user/protobuf.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
+import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
+import 'package:table_calendar/table_calendar.dart';
import '../../shared/util.dart';
@@ -18,7 +25,7 @@ void main() {
TestWidgetsFlutterBinding.ensureInitialized();
});
- group('date or reminder block in document', () {
+ group('date or reminder block in document:', () {
testWidgets("insert date with time block", (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
@@ -121,5 +128,339 @@ void main() {
expect(find.text('@$formattedDate'), findsOneWidget);
expect(find.byFlowySvg(FlowySvgs.reminder_clock_s), findsOneWidget);
});
+
+ testWidgets("copy, cut and paste a date mention", (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new document
+ await tester.createNewPageWithNameUnderParent(
+ name: 'copy, cut and paste a date mention',
+ );
+
+ // tap the first line of the document
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.editor.showSlashMenu();
+ await tester.editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_name_dateOrReminder.tr(),
+ );
+
+ final dateTimeSettings = DateTimeSettingsPB(
+ dateFormat: UserDateFormatPB.Friendly,
+ timeFormat: UserTimeFormatPB.TwentyFourHour,
+ );
+ final DateTime currentDateTime = DateTime.now();
+ final String formattedDate =
+ dateTimeSettings.dateFormat.formatDate(currentDateTime, false);
+
+ // get current date in editor
+ expect(find.byType(MentionDateBlock), findsOneWidget);
+ expect(find.text('@$formattedDate'), findsOneWidget);
+
+ // update selection and copy
+ await tester.editor.updateSelection(
+ Selection(
+ start: Position(path: [0]),
+ end: Position(path: [0], offset: 1),
+ ),
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyC,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ // update selection and paste
+ await tester.editor.updateSelection(
+ Selection.collapsed(Position(path: [0], offset: 1)),
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(MentionDateBlock), findsNWidgets(2));
+ expect(find.text('@$formattedDate'), findsNWidgets(2));
+
+ // update selection and cut
+ await tester.editor.updateSelection(
+ Selection(
+ start: Position(path: [0], offset: 1),
+ end: Position(path: [0], offset: 2),
+ ),
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyX,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(MentionDateBlock), findsOneWidget);
+ expect(find.text('@$formattedDate'), findsOneWidget);
+
+ // update selection and paste
+ await tester.editor.updateSelection(
+ Selection.collapsed(Position(path: [0], offset: 1)),
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(MentionDateBlock), findsNWidgets(2));
+ expect(find.text('@$formattedDate'), findsNWidgets(2));
+ });
+
+ testWidgets("copy, cut and paste a reminder mention", (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new document
+ await tester.createNewPageWithNameUnderParent(
+ name: 'copy, cut and paste a reminder mention',
+ );
+
+ // tap the first line of the document
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.editor.showSlashMenu();
+ await tester.editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_name_dateOrReminder.tr(),
+ );
+
+ // trigger popup
+ await tester.tapButton(find.byType(MentionDateBlock));
+ await tester.pumpAndSettle();
+
+ // set date to be fifteenth of the next month
+ await tester.tap(
+ find.descendant(
+ of: find.byType(DesktopAppFlowyDatePicker),
+ matching: find.byFlowySvg(FlowySvgs.arrow_right_s),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.tap(
+ find.descendant(
+ of: find.byType(TableCalendar),
+ matching: find.text(15.toString()),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ await tester.pumpAndSettle();
+
+ // add a reminder
+ await tester.tap(find.byType(MentionDateBlock));
+ await tester.pumpAndSettle();
+ await tester.tap(find.text(LocaleKeys.datePicker_reminderLabel.tr()));
+ await tester.pumpAndSettle();
+ await tester.tap(
+ find.textContaining(
+ LocaleKeys.datePicker_reminderOptions_oneDayBefore.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ await tester.pumpAndSettle();
+
+ // verify
+ final dateTimeSettings = DateTimeSettingsPB(
+ dateFormat: UserDateFormatPB.Friendly,
+ timeFormat: UserTimeFormatPB.TwentyFourHour,
+ );
+ final now = DateTime.now();
+ final fifteenthOfNextMonth = DateTime(now.year, now.month + 1, 15);
+ final formattedDate =
+ dateTimeSettings.dateFormat.formatDate(fifteenthOfNextMonth, false);
+
+ expect(find.byType(MentionDateBlock), findsOneWidget);
+ expect(find.text('@$formattedDate'), findsOneWidget);
+ expect(find.byFlowySvg(FlowySvgs.reminder_clock_s), findsOneWidget);
+ expect(getIt().state.reminders.map((e) => e.id).length, 1);
+
+ // update selection and copy
+ await tester.editor.updateSelection(
+ Selection(
+ start: Position(path: [0]),
+ end: Position(path: [0], offset: 1),
+ ),
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyC,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ // update selection and paste
+ await tester.editor.updateSelection(
+ Selection.collapsed(Position(path: [0], offset: 1)),
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(MentionDateBlock), findsNWidgets(2));
+ expect(find.text('@$formattedDate'), findsNWidgets(2));
+ expect(find.byFlowySvg(FlowySvgs.reminder_clock_s), findsNWidgets(2));
+ expect(
+ getIt().state.reminders.map((e) => e.id).toSet().length,
+ 2,
+ );
+
+ // update selection and cut
+ await tester.editor.updateSelection(
+ Selection(
+ start: Position(path: [0], offset: 1),
+ end: Position(path: [0], offset: 2),
+ ),
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyX,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(MentionDateBlock), findsOneWidget);
+ expect(find.text('@$formattedDate'), findsOneWidget);
+ expect(find.byFlowySvg(FlowySvgs.reminder_clock_s), findsOneWidget);
+ expect(getIt().state.reminders.map((e) => e.id).length, 1);
+
+ // update selection and paste
+ await tester.editor.updateSelection(
+ Selection.collapsed(Position(path: [0], offset: 1)),
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(MentionDateBlock), findsNWidgets(2));
+ expect(find.text('@$formattedDate'), findsNWidgets(2));
+ expect(find.byType(MentionDateBlock), findsNWidgets(2));
+ expect(find.text('@$formattedDate'), findsNWidgets(2));
+ expect(find.byFlowySvg(FlowySvgs.reminder_clock_s), findsNWidgets(2));
+ expect(
+ getIt().state.reminders.map((e) => e.id).toSet().length,
+ 2,
+ );
+ });
+
+ testWidgets("delete, undo and redo a reminder mention", (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new document
+ await tester.createNewPageWithNameUnderParent(
+ name: 'delete, undo and redo a reminder mention',
+ );
+
+ // tap the first line of the document
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.editor.showSlashMenu();
+ await tester.editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_name_dateOrReminder.tr(),
+ );
+
+ // trigger popup
+ await tester.tapButton(find.byType(MentionDateBlock));
+ await tester.pumpAndSettle();
+
+ // set date to be fifteenth of the next month
+ await tester.tap(
+ find.descendant(
+ of: find.byType(DesktopAppFlowyDatePicker),
+ matching: find.byFlowySvg(FlowySvgs.arrow_right_s),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.tap(
+ find.descendant(
+ of: find.byType(TableCalendar),
+ matching: find.text(15.toString()),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ await tester.pumpAndSettle();
+
+ // add a reminder
+ await tester.tap(find.byType(MentionDateBlock));
+ await tester.pumpAndSettle();
+ await tester.tap(find.text(LocaleKeys.datePicker_reminderLabel.tr()));
+ await tester.pumpAndSettle();
+ await tester.tap(
+ find.textContaining(
+ LocaleKeys.datePicker_reminderOptions_oneDayBefore.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ await tester.pumpAndSettle();
+
+ // verify
+ final dateTimeSettings = DateTimeSettingsPB(
+ dateFormat: UserDateFormatPB.Friendly,
+ timeFormat: UserTimeFormatPB.TwentyFourHour,
+ );
+ final now = DateTime.now();
+ final fifteenthOfNextMonth = DateTime(now.year, now.month + 1, 15);
+ final formattedDate =
+ dateTimeSettings.dateFormat.formatDate(fifteenthOfNextMonth, false);
+
+ expect(find.byType(MentionDateBlock), findsOneWidget);
+ expect(find.text('@$formattedDate'), findsOneWidget);
+ expect(find.byFlowySvg(FlowySvgs.reminder_clock_s), findsOneWidget);
+ expect(getIt().state.reminders.map((e) => e.id).length, 1);
+
+ // update selection and backspace to delete the mention
+ await tester.editor.updateSelection(
+ Selection.collapsed(Position(path: [0], offset: 1)),
+ );
+ await tester.simulateKeyEvent(LogicalKeyboardKey.backspace);
+ await tester.pumpAndSettle();
+
+ expect(find.byType(MentionDateBlock), findsNothing);
+ expect(find.text('@$formattedDate'), findsNothing);
+ expect(find.byFlowySvg(FlowySvgs.reminder_clock_s), findsNothing);
+ expect(getIt().state.reminders.isEmpty, isTrue);
+
+ // undo
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyZ,
+ isControlPressed: Platform.isWindows || Platform.isLinux,
+ isMetaPressed: Platform.isMacOS,
+ );
+
+ expect(find.byType(MentionDateBlock), findsOneWidget);
+ expect(find.text('@$formattedDate'), findsOneWidget);
+ expect(find.byFlowySvg(FlowySvgs.reminder_clock_s), findsOneWidget);
+ expect(getIt().state.reminders.map((e) => e.id).length, 1);
+
+ // redo
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyZ,
+ isControlPressed: Platform.isWindows || Platform.isLinux,
+ isMetaPressed: Platform.isMacOS,
+ isShiftPressed: true,
+ );
+
+ expect(find.byType(MentionDateBlock), findsNothing);
+ expect(find.text('@$formattedDate'), findsNothing);
+ expect(find.byFlowySvg(FlowySvgs.reminder_clock_s), findsNothing);
+ expect(getIt().state.reminders.isEmpty, isTrue);
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_image_block_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_image_block_test.dart
index 7d500b600f..3dcd6be8ae 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_image_block_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_image_block_test.dart
@@ -7,19 +7,16 @@ import 'package:appflowy/plugins/document/presentation/editor_plugins/image/cust
import 'package:appflowy/plugins/document/presentation/editor_plugins/image/image_placeholder.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/image/resizeable_image.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/image/upload_image_menu/upload_image_menu.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/image/upload_image_menu/widgets/embed_image_url_widget.dart';
import 'package:appflowy/startup/startup.dart';
import 'package:appflowy_editor/appflowy_editor.dart'
hide UploadImageMenu, ResizableImage;
import 'package:easy_localization/easy_localization.dart';
import 'package:flowy_infra_ui/flowy_infra_ui.dart';
-import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
import 'package:path/path.dart' as p;
import 'package:path_provider/path_provider.dart';
-import 'package:run_with_network_images/run_with_network_images.dart';
import '../../shared/mock/mock_file_picker.dart';
import '../../shared/util.dart';
@@ -79,90 +76,6 @@ void main() {
file.deleteSync();
});
- testWidgets('insert an image from network', (tester) async {
- await tester.initializeAppFlowy();
- await tester.tapAnonymousSignInButton();
-
- // create a new document
- await tester.createNewPageWithNameUnderParent(
- name: LocaleKeys.document_plugins_image_addAnImageDesktop.tr(),
- );
-
- // tap the first line of the document
- await tester.editor.tapLineOfEditorAt(0);
- await tester.editor.showSlashMenu();
- await tester.editor.tapSlashMenuItemWithName(
- LocaleKeys.document_slashMenu_name_image.tr(),
- );
- expect(find.byType(CustomImageBlockComponent), findsOneWidget);
- expect(find.byType(ImagePlaceholder), findsOneWidget);
- expect(
- find.descendant(
- of: find.byType(ImagePlaceholder),
- matching: find.byType(AppFlowyPopover),
- ),
- findsOneWidget,
- );
- expect(find.byType(UploadImageMenu), findsOneWidget);
-
- await tester.tapButtonWithName(
- LocaleKeys.document_imageBlock_embedLink_label.tr(),
- );
- const url =
- 'https://images.unsplash.com/photo-1469474968028-56623f02e42e?ixlib=rb-4.0.3&q=85&fm=jpg&crop=entropy&cs=srgb&dl=david-marcu-78A265wPiO4-unsplash.jpg&w=640';
- await tester.enterText(
- find.descendant(
- of: find.byType(EmbedImageUrlWidget),
- matching: find.byType(TextField),
- ),
- url,
- );
- await tester.tapButton(
- find.descendant(
- of: find.byType(EmbedImageUrlWidget),
- matching: find.text(
- LocaleKeys.document_imageBlock_embedLink_label.tr(),
- findRichText: true,
- ),
- ),
- );
- await tester.pumpAndSettle();
- expect(find.byType(ResizableImage), findsOneWidget);
- final node = tester.editor.getCurrentEditorState().getNodeAtPath([0])!;
- expect(node.type, ImageBlockKeys.type);
- expect(node.attributes[ImageBlockKeys.url], url);
- });
-
- testWidgets('insert an image from unsplash', (tester) async {
- await runWithNetworkImages(() async {
- await tester.initializeAppFlowy();
- await tester.tapAnonymousSignInButton();
-
- // create a new document
- await tester.createNewPageWithNameUnderParent(
- name: LocaleKeys.document_plugins_image_addAnImageDesktop.tr(),
- );
-
- // tap the first line of the document
- await tester.editor.tapLineOfEditorAt(0);
- await tester.editor.showSlashMenu();
- await tester.editor.tapSlashMenuItemWithName(
- LocaleKeys.document_slashMenu_name_image.tr(),
- );
- expect(find.byType(CustomImageBlockComponent), findsOneWidget);
- expect(find.byType(ImagePlaceholder), findsOneWidget);
- expect(
- find.descendant(
- of: find.byType(ImagePlaceholder),
- matching: find.byType(AppFlowyPopover),
- ),
- findsOneWidget,
- );
- expect(find.byType(UploadImageMenu), findsOneWidget);
- expect(find.text('Unsplash'), findsOneWidget);
- });
- });
-
testWidgets('insert two images from local file at once', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_inline_math_equation_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_inline_math_equation_test.dart
index c4a8e71a02..67e0149cd1 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_inline_math_equation_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_inline_math_equation_test.dart
@@ -1,9 +1,11 @@
+import 'package:appflowy/generated/flowy_svgs.g.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flowy_infra_ui/style_widget/button.dart';
import 'package:flutter/material.dart';
+import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -32,9 +34,15 @@ void main() {
Selection.single(path: [0], startOffset: 0, endOffset: formula.length),
);
+ // tap the more options button
+ final moreOptionButton = find.findFlowyTooltip(
+ LocaleKeys.document_toolbar_moreOptions.tr(),
+ );
+ await tester.tapButton(moreOptionButton);
+
// tap the inline math equation button
- final inlineMathEquationButton = find.findFlowyTooltip(
- LocaleKeys.document_plugins_createInlineMathEquation.tr(),
+ final inlineMathEquationButton = find.text(
+ LocaleKeys.document_toolbar_equation.tr(),
);
await tester.tapButton(inlineMathEquationButton);
@@ -77,10 +85,15 @@ void main() {
Selection.single(path: [0], startOffset: 0, endOffset: formula.length),
);
- // tap the inline math equation button
- var inlineMathEquationButton = find.findFlowyTooltip(
- LocaleKeys.document_plugins_createInlineMathEquation.tr(),
+ // tap the more options button
+ final moreOptionButton = find.findFlowyTooltip(
+ LocaleKeys.document_toolbar_moreOptions.tr(),
);
+ await tester.tapButton(moreOptionButton);
+
+ // tap the inline math equation button
+ final inlineMathEquationButton =
+ find.byFlowySvg(FlowySvgs.type_formula_m);
await tester.tapButton(inlineMathEquationButton);
// expect to see the math equation block
@@ -92,17 +105,7 @@ void main() {
Selection.single(path: [0], startOffset: 0, endOffset: 1),
);
- // expect to the see the inline math equation button is highlighted
- inlineMathEquationButton = find.descendant(
- of: find.findFlowyTooltip(
- LocaleKeys.document_plugins_createInlineMathEquation.tr(),
- ),
- matching: find.byType(SVGIconItemWidget),
- );
- expect(
- tester.widget(inlineMathEquationButton).isHighlight,
- isTrue,
- );
+ await tester.tapButton(moreOptionButton);
// cancel the format
await tester.tapButton(inlineMathEquationButton);
@@ -113,5 +116,110 @@ void main() {
tester.expectToSeeText(formula);
});
+
+ testWidgets('insert a inline math equation and type something after it',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new document
+ await tester.createNewPageWithNameUnderParent(
+ name: 'math equation',
+ );
+
+ // tap the first line of the document
+ await tester.editor.tapLineOfEditorAt(0);
+ // insert a inline page
+ const formula = 'E = MC ^ 2';
+ await tester.ime.insertText(formula);
+ await tester.editor.updateSelection(
+ Selection.single(path: [0], startOffset: 0, endOffset: formula.length),
+ );
+
+ // tap the more options button
+ final moreOptionButton = find.findFlowyTooltip(
+ LocaleKeys.document_toolbar_moreOptions.tr(),
+ );
+ await tester.tapButton(moreOptionButton);
+
+ // tap the inline math equation button
+ final inlineMathEquationButton =
+ find.byFlowySvg(FlowySvgs.type_formula_m);
+ await tester.tapButton(inlineMathEquationButton);
+
+ // expect to see the math equation block
+ final inlineMathEquation = find.byType(InlineMathEquation);
+ expect(inlineMathEquation, findsOneWidget);
+
+ await tester.editor.tapLineOfEditorAt(0);
+ const text = 'Hello World';
+ await tester.ime.insertText(text);
+
+ final inlineText = find.textContaining(text, findRichText: true);
+ expect(inlineText, findsOneWidget);
+
+ // the text should be in the same line with the math equation
+ final inlineMathEquationPosition = tester.getRect(inlineMathEquation);
+ final textPosition = tester.getRect(inlineText);
+ // allow 5px difference
+ expect(
+ (textPosition.top - inlineMathEquationPosition.top).abs(),
+ lessThan(5),
+ );
+ expect(
+ (textPosition.bottom - inlineMathEquationPosition.bottom).abs(),
+ lessThan(5),
+ );
+ });
+
+ testWidgets('insert inline math equation by shortcut', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new document
+ await tester.createNewPageWithNameUnderParent(
+ name: 'insert inline math equation by shortcut',
+ );
+
+ // tap the first line of the document
+ await tester.editor.tapLineOfEditorAt(0);
+ // insert a inline page
+ const formula = 'E = MC ^ 2';
+ await tester.ime.insertText(formula);
+ await tester.editor.updateSelection(
+ Selection.single(path: [0], startOffset: 0, endOffset: formula.length),
+ );
+
+ // mock key event
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyE,
+ isShiftPressed: true,
+ isControlPressed: true,
+ );
+
+ // expect to see the math equation block
+ final inlineMathEquation = find.byType(InlineMathEquation);
+ expect(inlineMathEquation, findsOneWidget);
+
+ await tester.editor.tapLineOfEditorAt(0);
+ const text = 'Hello World';
+ await tester.ime.insertText(text);
+
+ final inlineText = find.textContaining(text, findRichText: true);
+ expect(inlineText, findsOneWidget);
+
+ // the text should be in the same line with the math equation
+ final inlineMathEquationPosition = tester.getRect(inlineMathEquation);
+ final textPosition = tester.getRect(inlineText);
+ // allow 5px difference
+ expect(
+ (textPosition.top - inlineMathEquationPosition.top).abs(),
+ lessThan(5),
+ );
+ expect(
+ (textPosition.bottom - inlineMathEquationPosition.bottom).abs(),
+ lessThan(5),
+ );
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_inline_page_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_inline_page_test.dart
index 2a552c0d22..12047bd37f 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_inline_page_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_inline_page_test.dart
@@ -94,6 +94,20 @@ void main() {
await tester.tapButton(finder);
expect(find.byType(GridPage), findsOneWidget);
});
+
+ testWidgets('insert a inline page and type something after the page',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await insertInlinePage(tester, ViewLayoutPB.Grid);
+
+ await tester.editor.tapLineOfEditorAt(0);
+ const text = 'Hello World';
+ await tester.ime.insertText(text);
+
+ expect(find.textContaining(text, findRichText: true), findsOneWidget);
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_outline_block_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_outline_block_test.dart
index 3053bd748a..2f3f8c80b9 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_outline_block_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_outline_block_test.dart
@@ -1,6 +1,7 @@
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -43,6 +44,9 @@ void main() {
* # Heading 1
* ## Heading 2
* ### Heading 3
+ * > # Heading 1
+ * > ## Heading 2
+ * > ### Heading 3
*/
await tester.editor.tapLineOfEditorAt(3);
@@ -53,7 +57,7 @@ void main() {
of: find.byType(OutlineBlockWidget),
matching: find.text(heading1),
),
- findsOneWidget,
+ findsNWidgets(2),
);
// Heading 2 is prefixed with a bullet
@@ -62,7 +66,7 @@ void main() {
of: find.byType(OutlineBlockWidget),
matching: find.text(heading2),
),
- findsOneWidget,
+ findsNWidgets(2),
);
// Heading 3 is prefixed with a dash
@@ -71,7 +75,7 @@ void main() {
of: find.byType(OutlineBlockWidget),
matching: find.text(heading3),
),
- findsOneWidget,
+ findsNWidgets(2),
);
// update the Heading 1 to Heading 1Hello world
@@ -99,13 +103,16 @@ void main() {
* # Heading 1
* ## Heading 2
* ### Heading 3
+ * > # Heading 1
+ * > ## Heading 2
+ * > ### Heading 3
*/
- await tester.editor.tapLineOfEditorAt(3);
+ await tester.editor.tapLineOfEditorAt(7);
await insertOutlineInDocument(tester);
// expect to find only the `heading1` widget under the [OutlineBlockWidget]
- await hoverAndClickDepthOptionAction(tester, [3], 1);
+ await hoverAndClickDepthOptionAction(tester, [6], 1);
expect(
find.descendant(
of: find.byType(OutlineBlockWidget),
@@ -123,7 +130,7 @@ void main() {
//////
/// expect to find only the 'heading1' and 'heading2' under the [OutlineBlockWidget]
- await hoverAndClickDepthOptionAction(tester, [3], 2);
+ await hoverAndClickDepthOptionAction(tester, [6], 2);
expect(
find.descendant(
of: find.byType(OutlineBlockWidget),
@@ -134,13 +141,13 @@ void main() {
//////
// expect to find all the headings under the [OutlineBlockWidget]
- await hoverAndClickDepthOptionAction(tester, [3], 3);
+ await hoverAndClickDepthOptionAction(tester, [6], 3);
expect(
find.descendant(
of: find.byType(OutlineBlockWidget),
matching: find.text(heading1),
),
- findsOneWidget,
+ findsNWidgets(2),
);
expect(
@@ -148,7 +155,7 @@ void main() {
of: find.byType(OutlineBlockWidget),
matching: find.text(heading2),
),
- findsOneWidget,
+ findsNWidgets(2),
);
expect(
@@ -156,7 +163,7 @@ void main() {
of: find.byType(OutlineBlockWidget),
matching: find.text(heading3),
),
- findsOneWidget,
+ findsNWidgets(2),
);
//////
});
@@ -186,7 +193,17 @@ Future hoverAndClickDepthOptionAction(
Future insertHeadingComponent(WidgetTester tester) async {
await tester.editor.tapLineOfEditorAt(0);
+
+ // # heading 1-3
await tester.ime.insertText('# $heading1\n');
await tester.ime.insertText('## $heading2\n');
await tester.ime.insertText('### $heading3\n');
+
+ // > # toggle heading 1-3
+ await tester.ime.insertText('> # $heading1\n');
+ await tester.simulateKeyEvent(LogicalKeyboardKey.backspace);
+ await tester.ime.insertText('> ## $heading2\n');
+ await tester.simulateKeyEvent(LogicalKeyboardKey.backspace);
+ await tester.ime.insertText('> ### $heading3\n');
+ await tester.simulateKeyEvent(LogicalKeyboardKey.backspace);
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_simple_table_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_simple_table_test.dart
new file mode 100644
index 0000000000..bcf3fde24f
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_simple_table_test.dart
@@ -0,0 +1,783 @@
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/simple_table/simple_table.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/services.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+import 'package:universal_platform/universal_platform.dart';
+
+import '../../shared/util.dart';
+
+const String heading1 = "Heading 1";
+const String heading2 = "Heading 2";
+const String heading3 = "Heading 3";
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('simple table block test:', () {
+ testWidgets('insert a simple table block', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ // validate the table is inserted
+ expect(find.byType(SimpleTableBlockWidget), findsOneWidget);
+
+ final editorState = tester.editor.getCurrentEditorState();
+ expect(
+ editorState.selection,
+ // table -> row -> cell -> paragraph
+ Selection.collapsed(Position(path: [0, 0, 0, 0])),
+ );
+
+ final firstCell = find.byType(SimpleTableCellBlockWidget).first;
+ expect(
+ tester
+ .state(firstCell)
+ .isEditingCellNotifier
+ .value,
+ isTrue,
+ );
+ });
+
+ testWidgets('select all in table cell', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ const cell1Content = 'Cell 1';
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText('New Table');
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ await tester.pumpAndSettle();
+ await tester.editor.tapLineOfEditorAt(1);
+ await tester.insertTableInDocument();
+ await tester.ime.insertText(cell1Content);
+ await tester.pumpAndSettle();
+ // Select all in the cell
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyA,
+ isControlPressed: !UniversalPlatform.isMacOS,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+
+ expect(
+ tester.editor.getCurrentEditorState().selection,
+ Selection(
+ start: Position(path: [1, 0, 0, 0]),
+ end: Position(path: [1, 0, 0, 0], offset: cell1Content.length),
+ ),
+ );
+
+ // Press select all again, the selection should be the entire document
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyA,
+ isControlPressed: !UniversalPlatform.isMacOS,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+
+ expect(
+ tester.editor.getCurrentEditorState().selection,
+ Selection(
+ start: Position(path: [0]),
+ end: Position(path: [1, 1, 1, 0]),
+ ),
+ );
+ });
+
+ testWidgets('''
+1. hover on the table
+ 1.1 click the add row button
+ 1.2 click the add column button
+ 1.3 click the add row and column button
+2. validate the table is updated
+3. delete the last column
+4. delete the last row
+5. validate the table is updated
+''', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ // add a new row
+ final row = find.byWidgetPredicate((w) {
+ return w is SimpleTableRowBlockWidget && w.node.rowIndex == 1;
+ });
+ await tester.hoverOnWidget(
+ row,
+ onHover: () async {
+ final addRowButton = find.byType(SimpleTableAddRowButton).first;
+ await tester.tap(addRowButton);
+ },
+ );
+ await tester.pumpAndSettle();
+
+ // add a new column
+ final column = find.byWidgetPredicate((w) {
+ return w is SimpleTableCellBlockWidget && w.node.columnIndex == 1;
+ }).first;
+ await tester.hoverOnWidget(
+ column,
+ onHover: () async {
+ final addColumnButton = find.byType(SimpleTableAddColumnButton).first;
+ await tester.tap(addColumnButton);
+ },
+ );
+ await tester.pumpAndSettle();
+
+ // add a new row and a new column
+ final row2 = find.byWidgetPredicate((w) {
+ return w is SimpleTableCellBlockWidget &&
+ w.node.rowIndex == 2 &&
+ w.node.columnIndex == 2;
+ }).first;
+ await tester.hoverOnWidget(
+ row2,
+ onHover: () async {
+ // click the add row and column button
+ final addRowAndColumnButton =
+ find.byType(SimpleTableAddColumnAndRowButton).first;
+ await tester.tap(addRowAndColumnButton);
+ },
+ );
+ await tester.pumpAndSettle();
+
+ final tableNode =
+ tester.editor.getCurrentEditorState().document.nodeAtPath([0])!;
+ expect(tableNode.columnLength, 4);
+ expect(tableNode.rowLength, 4);
+
+ // delete the last row
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.row,
+ index: tableNode.rowLength - 1,
+ action: SimpleTableMoreAction.delete,
+ );
+ await tester.pumpAndSettle();
+ expect(tableNode.rowLength, 3);
+ expect(tableNode.columnLength, 4);
+
+ // delete the last column
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.column,
+ index: tableNode.columnLength - 1,
+ action: SimpleTableMoreAction.delete,
+ );
+ await tester.pumpAndSettle();
+
+ expect(tableNode.columnLength, 3);
+ expect(tableNode.rowLength, 3);
+ });
+
+ testWidgets('enable header column and header row', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ // enable the header row
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.row,
+ index: 0,
+ action: SimpleTableMoreAction.enableHeaderRow,
+ );
+ await tester.pumpAndSettle();
+ // enable the header column
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.column,
+ index: 0,
+ action: SimpleTableMoreAction.enableHeaderColumn,
+ );
+ await tester.pumpAndSettle();
+
+ final tableNode =
+ tester.editor.getCurrentEditorState().document.nodeAtPath([0])!;
+
+ expect(tableNode.isHeaderColumnEnabled, isTrue);
+ expect(tableNode.isHeaderRowEnabled, isTrue);
+
+ // disable the header row
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.row,
+ index: 0,
+ action: SimpleTableMoreAction.enableHeaderRow,
+ );
+ await tester.pumpAndSettle();
+ expect(tableNode.isHeaderColumnEnabled, isTrue);
+ expect(tableNode.isHeaderRowEnabled, isFalse);
+
+ // disable the header column
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.column,
+ index: 0,
+ action: SimpleTableMoreAction.enableHeaderColumn,
+ );
+ await tester.pumpAndSettle();
+ expect(tableNode.isHeaderColumnEnabled, isFalse);
+ expect(tableNode.isHeaderRowEnabled, isFalse);
+ });
+
+ testWidgets('duplicate a column / row', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ // duplicate the row
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.row,
+ index: 0,
+ action: SimpleTableMoreAction.duplicate,
+ );
+ await tester.pumpAndSettle();
+
+ // duplicate the column
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.column,
+ index: 0,
+ action: SimpleTableMoreAction.duplicate,
+ );
+ await tester.pumpAndSettle();
+
+ final tableNode =
+ tester.editor.getCurrentEditorState().document.nodeAtPath([0])!;
+ expect(tableNode.columnLength, 3);
+ expect(tableNode.rowLength, 3);
+ });
+
+ testWidgets('insert left / insert right', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ // insert left
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.column,
+ index: 0,
+ action: SimpleTableMoreAction.insertLeft,
+ );
+ await tester.pumpAndSettle();
+
+ // insert right
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.column,
+ index: 0,
+ action: SimpleTableMoreAction.insertRight,
+ );
+ await tester.pumpAndSettle();
+
+ final tableNode =
+ tester.editor.getCurrentEditorState().document.nodeAtPath([0])!;
+ expect(tableNode.columnLength, 4);
+ expect(tableNode.rowLength, 2);
+ });
+
+ testWidgets('insert above / insert below', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ // insert above
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.row,
+ index: 0,
+ action: SimpleTableMoreAction.insertAbove,
+ );
+ await tester.pumpAndSettle();
+
+ // insert below
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.row,
+ index: 0,
+ action: SimpleTableMoreAction.insertBelow,
+ );
+ await tester.pumpAndSettle();
+
+ final tableNode =
+ tester.editor.getCurrentEditorState().document.nodeAtPath([0])!;
+ expect(tableNode.rowLength, 4);
+ expect(tableNode.columnLength, 2);
+ });
+ });
+
+ testWidgets('set column width to page width (1)', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ final tableNode = tester.editor.getNodeAtPath([0]);
+ final beforeWidth = tableNode.width;
+
+ // set the column width to page width
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.column,
+ index: 0,
+ action: SimpleTableMoreAction.setToPageWidth,
+ );
+ await tester.pumpAndSettle();
+
+ final afterWidth = tableNode.width;
+ expect(afterWidth, greaterThan(beforeWidth));
+ });
+
+ testWidgets('set column width to page width (2)', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ final tableNode = tester.editor.getNodeAtPath([0]);
+ final beforeWidth = tableNode.width;
+
+ // set the column width to page width
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.row,
+ index: 0,
+ action: SimpleTableMoreAction.setToPageWidth,
+ );
+ await tester.pumpAndSettle();
+
+ final afterWidth = tableNode.width;
+ expect(afterWidth, greaterThan(beforeWidth));
+ });
+
+ testWidgets('distribute columns evenly (1)', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ final tableNode = tester.editor.getNodeAtPath([0]);
+ final beforeWidth = tableNode.width;
+
+ // set the column width to page width
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.row,
+ index: 0,
+ action: SimpleTableMoreAction.distributeColumnsEvenly,
+ );
+ await tester.pumpAndSettle();
+
+ final afterWidth = tableNode.width;
+ expect(afterWidth, equals(beforeWidth));
+
+ final distributeColumnWidthsEvenly =
+ tableNode.attributes[SimpleTableBlockKeys.distributeColumnWidthsEvenly];
+ expect(distributeColumnWidthsEvenly, isTrue);
+ });
+
+ testWidgets('distribute columns evenly (2)', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ final tableNode = tester.editor.getNodeAtPath([0]);
+ final beforeWidth = tableNode.width;
+
+ // set the column width to page width
+ await tester.clickMoreActionItemInTableMenu(
+ type: SimpleTableMoreActionType.column,
+ index: 0,
+ action: SimpleTableMoreAction.distributeColumnsEvenly,
+ );
+ await tester.pumpAndSettle();
+
+ final afterWidth = tableNode.width;
+ expect(afterWidth, equals(beforeWidth));
+
+ final distributeColumnWidthsEvenly =
+ tableNode.attributes[SimpleTableBlockKeys.distributeColumnWidthsEvenly];
+ expect(distributeColumnWidthsEvenly, isTrue);
+ });
+
+ testWidgets('using option menu to set column width', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+
+ final editorState = tester.editor.getCurrentEditorState();
+ final beforeWidth = editorState.document.nodeAtPath([0])!.width;
+
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_simpleTable_moreActions_setToPageWidth.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final afterWidth = editorState.document.nodeAtPath([0])!.width;
+ expect(afterWidth, greaterThan(beforeWidth));
+
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+ await tester.tapButton(
+ find.text(
+ LocaleKeys
+ .document_plugins_simpleTable_moreActions_distributeColumnsWidth
+ .tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final afterWidth2 = editorState.document.nodeAtPath([0])!.width;
+ expect(afterWidth2, equals(afterWidth));
+ });
+
+ testWidgets('insert a table and use select all the delete it',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ await tester.editor.tapLineOfEditorAt(1);
+ await tester.ime.insertText('Hello World');
+
+ // select all
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyA,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ isControlPressed: !UniversalPlatform.isMacOS,
+ );
+
+ await tester.simulateKeyEvent(LogicalKeyboardKey.backspace);
+ await tester.pumpAndSettle();
+
+ final editorState = tester.editor.getCurrentEditorState();
+ // only one paragraph left
+ expect(editorState.document.root.children.length, 1);
+ final paragraphNode = editorState.document.nodeAtPath([0])!;
+ expect(paragraphNode.delta, isNull);
+ });
+
+ testWidgets('use tab or shift+tab to navigate in table', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ await tester.simulateKeyEvent(LogicalKeyboardKey.tab);
+ await tester.pumpAndSettle();
+
+ final editorState = tester.editor.getCurrentEditorState();
+ final selection = editorState.selection;
+ expect(selection, isNotNull);
+ expect(selection!.start.path, [0, 0, 1, 0]);
+ expect(selection.end.path, [0, 0, 1, 0]);
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.tab,
+ isShiftPressed: true,
+ );
+ await tester.pumpAndSettle();
+
+ final selection2 = editorState.selection;
+ expect(selection2, isNotNull);
+ expect(selection2!.start.path, [0, 0, 0, 0]);
+ expect(selection2.end.path, [0, 0, 0, 0]);
+ });
+
+ testWidgets('shift+enter to insert a new line in table', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.enter,
+ isShiftPressed: true,
+ );
+ await tester.pumpAndSettle();
+
+ final editorState = tester.editor.getCurrentEditorState();
+ final node = editorState.document.nodeAtPath([0, 0, 0])!;
+ expect(node.children.length, 1);
+ });
+
+ testWidgets('using option menu to set table align', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+
+ final editorState = tester.editor.getCurrentEditorState();
+ final beforeAlign = editorState.document.nodeAtPath([0])!.tableAlign;
+ expect(beforeAlign, TableAlign.left);
+
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_simpleTable_moreActions_align.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_optionAction_center.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final afterAlign = editorState.document.nodeAtPath([0])!.tableAlign;
+ expect(afterAlign, TableAlign.center);
+
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_simpleTable_moreActions_align.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_optionAction_right.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final afterAlign2 = editorState.document.nodeAtPath([0])!.tableAlign;
+ expect(afterAlign2, TableAlign.right);
+
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_simpleTable_moreActions_align.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_optionAction_left.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final afterAlign3 = editorState.document.nodeAtPath([0])!.tableAlign;
+ expect(afterAlign3, TableAlign.left);
+ });
+
+ testWidgets('using option menu to set table align', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+
+ final editorState = tester.editor.getCurrentEditorState();
+ final beforeAlign = editorState.document.nodeAtPath([0])!.tableAlign;
+ expect(beforeAlign, TableAlign.left);
+
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_simpleTable_moreActions_align.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_optionAction_center.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final afterAlign = editorState.document.nodeAtPath([0])!.tableAlign;
+ expect(afterAlign, TableAlign.center);
+
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_simpleTable_moreActions_align.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_optionAction_right.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final afterAlign2 = editorState.document.nodeAtPath([0])!.tableAlign;
+ expect(afterAlign2, TableAlign.right);
+
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_simpleTable_moreActions_align.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+ await tester.tapButton(
+ find.text(
+ LocaleKeys.document_plugins_optionAction_left.tr(),
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final afterAlign3 = editorState.document.nodeAtPath([0])!.tableAlign;
+ expect(afterAlign3, TableAlign.left);
+ });
+
+ testWidgets('support slash menu in table', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(
+ name: 'simple_table_test',
+ );
+
+ final editorState = tester.editor.getCurrentEditorState();
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.insertTableInDocument();
+
+ final path = [0, 0, 0, 0];
+ final selection = Selection.collapsed(Position(path: path));
+ editorState.selection = selection;
+ await tester.editor.showSlashMenu();
+ await tester.pumpAndSettle();
+
+ final paragraphItem = find.byWidgetPredicate((w) {
+ return w is SelectionMenuItemWidget &&
+ w.item.name == LocaleKeys.document_slashMenu_name_text.tr();
+ });
+ expect(paragraphItem, findsOneWidget);
+
+ await tester.tap(paragraphItem);
+ await tester.pumpAndSettle();
+
+ final paragraphNode = editorState.document.nodeAtPath(path)!;
+ expect(paragraphNode.type, equals(ParagraphBlockKeys.type));
+ });
+}
+
+extension on WidgetTester {
+ /// Insert a table in the document
+ Future insertTableInDocument() async {
+ // open the actions menu and insert the outline block
+ await editor.showSlashMenu();
+ await editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_name_table.tr(),
+ );
+ await pumpAndSettle();
+ }
+
+ Future clickMoreActionItemInTableMenu({
+ required SimpleTableMoreActionType type,
+ required int index,
+ required SimpleTableMoreAction action,
+ }) async {
+ if (type == SimpleTableMoreActionType.row) {
+ final row = find.byWidgetPredicate((w) {
+ return w is SimpleTableRowBlockWidget && w.node.rowIndex == index;
+ });
+ await hoverOnWidget(
+ row,
+ onHover: () async {
+ final moreActionButton = find.byWidgetPredicate((w) {
+ return w is SimpleTableMoreActionMenu &&
+ w.type == SimpleTableMoreActionType.row &&
+ w.index == index;
+ });
+ await tapButton(moreActionButton);
+ await tapButton(find.text(action.name));
+ },
+ );
+ await pumpAndSettle();
+ } else if (type == SimpleTableMoreActionType.column) {
+ final column = find.byWidgetPredicate((w) {
+ return w is SimpleTableCellBlockWidget && w.node.columnIndex == index;
+ }).first;
+ await hoverOnWidget(
+ column,
+ onHover: () async {
+ final moreActionButton = find.byWidgetPredicate((w) {
+ return w is SimpleTableMoreActionMenu &&
+ w.type == SimpleTableMoreActionType.column &&
+ w.index == index;
+ });
+ await tapButton(moreActionButton);
+ await tapButton(find.text(action.name));
+ },
+ );
+ await pumpAndSettle();
+ }
+
+ await tapAt(Offset.zero);
+ }
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_toggle_heading_block_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_toggle_heading_block_test.dart
index 8eb47fc15f..c4aa289855 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_toggle_heading_block_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_toggle_heading_block_test.dart
@@ -1,3 +1,4 @@
+import 'package:appflowy/generated/flowy_svgs.g.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
@@ -85,16 +86,10 @@ void main() {
),
);
- await tester.tapButton(find.byType(HeadingPopup));
- await tester.pumpAndSettle();
-
- expect(
- find.byType(HeadingButton),
- findsNWidgets(3),
- );
+ await tester.tapButton(find.byFlowySvg(FlowySvgs.toolbar_text_format_m));
// tap the H1 button
- await tester.tapButton(find.byType(HeadingButton).at(0));
+ await tester.tapButton(find.byFlowySvg(FlowySvgs.type_h1_m).at(0));
await tester.pumpAndSettle();
final editorState = tester.editor.getCurrentEditorState();
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_toggle_list_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_toggle_list_test.dart
index f4f1be8c78..a4d011dccb 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_toggle_list_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_toggle_list_test.dart
@@ -1,7 +1,9 @@
import 'dart:io';
+import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
@@ -263,5 +265,24 @@ void main() {
expect(node.attributes[ToggleListBlockKeys.level], 3);
expect(node.delta!.toPlainText(), 'Hello');
});
+
+ testWidgets('click the toggle list to create a new paragraph',
+ (tester) async {
+ await prepareToggleHeadingBlock(tester, '> # Hello');
+ final emptyHintText = find.text(
+ LocaleKeys.document_plugins_emptyToggleHeading.tr(
+ args: ['1'],
+ ),
+ );
+ expect(emptyHintText, findsOneWidget);
+
+ await tester.tapButton(emptyHintText);
+ await tester.pumpAndSettle();
+
+ // check the new paragraph is created
+ final editorState = tester.editor.getCurrentEditorState();
+ final node = editorState.getNodeAtPath([0, 0])!;
+ expect(node.type, ParagraphBlockKeys.type);
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/settings/shortcuts_settings_test.dart b/frontend/appflowy_flutter/integration_test/desktop/settings/shortcuts_settings_test.dart
index 7913a88294..fe91becba6 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/settings/shortcuts_settings_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/settings/shortcuts_settings_test.dart
@@ -35,10 +35,12 @@ void main() {
await tester.pumpAndSettle();
await tester.hoverOnWidget(
- find.descendant(
- of: find.byType(ShortcutSettingTile),
- matching: find.text(backspaceCmd),
- ),
+ find
+ .descendant(
+ of: find.byType(ShortcutSettingTile),
+ matching: find.text(backspaceCmd),
+ )
+ .first,
onHover: () async {
await tester.tap(find.byFlowySvg(FlowySvgs.edit_s));
await tester.pumpAndSettle();
diff --git a/frontend/appflowy_flutter/integration_test/desktop/settings/sign_in_page_settings_test.dart b/frontend/appflowy_flutter/integration_test/desktop/settings/sign_in_page_settings_test.dart
index 7caf439b66..047e02da36 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/settings/sign_in_page_settings_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/settings/sign_in_page_settings_test.dart
@@ -2,10 +2,10 @@ import 'package:appflowy/env/cloud_env.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/user/presentation/screens/sign_in_screen/desktop_sign_in_screen.dart';
import 'package:appflowy/workspace/presentation/settings/settings_dialog.dart';
+import 'package:appflowy/workspace/presentation/widgets/dialogs.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:toastification/toastification.dart';
import '../../shared/util.dart';
@@ -23,7 +23,7 @@ void main() {
.last;
}
- group('sign-in page settings: ', () {
+ group('sign-in page settings:', () {
testWidgets('change server type', (tester) async {
await tester.initializeAppFlowy();
@@ -45,28 +45,36 @@ void main() {
// change the server type to self-host
await tester.tapButton(appflowyCloudType);
- final selfhostedButton = findServerType(
+ final selfHostedButton = findServerType(
AuthenticatorType.appflowyCloudSelfHost,
);
- await tester.tapButton(selfhostedButton);
+ await tester.tapButton(selfHostedButton);
// update server url
- const serverUrl = 'https://test.appflowy.cloud';
+ const serverUrl = 'https://self-hosted.appflowy.cloud';
await tester.enterText(
find.byKey(kSelfHostedTextInputFieldKey),
serverUrl,
);
await tester.pumpAndSettle();
+ // update the web url
+ const webUrl = 'https://self-hosted.appflowy.com';
+ await tester.enterText(
+ find.byKey(kSelfHostedWebTextInputFieldKey),
+ webUrl,
+ );
+ await tester.pumpAndSettle();
await tester.tapButton(
find.findTextInFlowyText(LocaleKeys.button_save.tr()),
);
// wait the app to restart, and the tooltip to disappear
- await tester.pumpUntilNotFound(find.byType(BuiltInToastBuilder));
+ await tester.pumpUntilNotFound(find.byType(DesktopToast));
await tester.pumpAndSettle(const Duration(milliseconds: 250));
// open settings page to check the result
await tester.tapButton(settingsButton);
+ await tester.pumpAndSettle(const Duration(milliseconds: 250));
// check the server type
expect(
@@ -78,18 +86,23 @@ void main() {
find.text(serverUrl),
findsOneWidget,
);
+ // check the web url
+ expect(
+ find.text(webUrl),
+ findsOneWidget,
+ );
// reset to appflowy cloud
await tester.tapButton(
findServerType(AuthenticatorType.appflowyCloudSelfHost),
- );
+ );
// change the server type to appflowy cloud
await tester.tapButton(
findServerType(AuthenticatorType.appflowyCloud),
);
// wait the app to restart, and the tooltip to disappear
- await tester.pumpUntilNotFound(find.byType(BuiltInToastBuilder));
+ await tester.pumpUntilNotFound(find.byType(DesktopToast));
await tester.pumpAndSettle(const Duration(milliseconds: 250));
// check the server type
diff --git a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_expand_test.dart b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_expand_test.dart
index f2a1fae8ae..ad18cf3de6 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_expand_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_expand_test.dart
@@ -1,8 +1,12 @@
import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/inline_actions/widgets/inline_actions_handler.dart';
import 'package:appflowy/workspace/application/sidebar/folder/folder_bloc.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/shared/sidebar_folder.dart';
import 'package:appflowy/workspace/presentation/home/menu/view/view_item.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/view.pbenum.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/cupertino.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -44,5 +48,82 @@ void main() {
);
expect(isExpanded(type: FolderSpaceType.private), true);
});
+
+ testWidgets('Expanding with subpage', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ const page1 = 'SubPageBloc', page2 = '$page1 2';
+ await tester.createNewPageWithNameUnderParent(name: page1);
+ await tester.createNewPageWithNameUnderParent(
+ name: page2,
+ parentName: page1,
+ );
+
+ await tester.expandOrCollapsePage(
+ pageName: gettingStarted,
+ layout: ViewLayoutPB.Document,
+ );
+
+ await tester.tapNewPageButton();
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.pumpAndSettle();
+ await tester.editor.showSlashMenu();
+ await tester.pumpAndSettle();
+
+ final slashMenu = find
+ .ancestor(
+ of: find.byType(SelectionMenuItemWidget),
+ matching: find.byWidgetPredicate(
+ (widget) => widget is Scrollable,
+ ),
+ )
+ .first;
+ final slashMenuItem = find.text(
+ LocaleKeys.document_slashMenu_name_linkedDoc.tr(),
+ );
+ await tester.scrollUntilVisible(
+ slashMenuItem,
+ 100,
+ scrollable: slashMenu,
+ duration: const Duration(milliseconds: 250),
+ );
+
+ final menuItemFinder = find.byWidgetPredicate(
+ (w) =>
+ w is SelectionMenuItemWidget &&
+ w.item.name == LocaleKeys.document_slashMenu_name_linkedDoc.tr(),
+ );
+
+ final menuItem =
+ menuItemFinder.evaluate().first.widget as SelectionMenuItemWidget;
+
+ /// tapSlashMenuItemWithName is not working, so invoke this function directly
+ menuItem.item.handler(
+ menuItem.editorState,
+ menuItem.menuService,
+ menuItemFinder.evaluate().first,
+ );
+
+ await tester.pumpAndSettle();
+ final actionHandler = find.byType(InlineActionsHandler);
+ final subPage = find.descendant(
+ of: actionHandler,
+ matching: find.text(page2, findRichText: true),
+ );
+ await tester.tapButton(subPage);
+
+ final subpageBlock = find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.text(page2, findRichText: true),
+ );
+
+ expect(find.text(page2, findRichText: true), findsOneWidget);
+ await tester.tapButton(subpageBlock);
+
+ /// one is in SectionFolder, another one is in CoverTitle
+ /// the last one is in FlowyNavigation
+ expect(find.text(page2, findRichText: true), findsNWidgets(3));
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_favorites_test.dart b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_favorites_test.dart
index 729ee62a3e..3345ed30ab 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_favorites_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_favorites_test.dart
@@ -196,5 +196,58 @@ void main() {
await tester.pumpAndSettle();
},
);
+
+ testWidgets(
+ 'reorder favorites',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ /// there are no favorite views
+ final favorites = find.descendant(
+ of: find.byType(FavoriteFolder),
+ matching: find.byType(ViewItem),
+ );
+ expect(favorites, findsNothing);
+
+ /// create views and then favorite them
+ const pageNames = ['001', '002', '003'];
+ for (final name in pageNames) {
+ await tester.createNewPageWithNameUnderParent(name: name);
+ }
+ for (final name in pageNames) {
+ await tester.favoriteViewByName(name);
+ }
+ expect(favorites, findsNWidgets(pageNames.length));
+
+ final oldNames = favorites
+ .evaluate()
+ .map((e) => (e.widget as ViewItem).view.name)
+ .toList();
+ expect(oldNames, pageNames);
+
+ /// drag first to last
+ await tester.reorderFavorite(
+ fromName: '001',
+ toName: '003',
+ );
+ List newNames = favorites
+ .evaluate()
+ .map((e) => (e.widget as ViewItem).view.name)
+ .toList();
+ expect(newNames, ['002', '003', '001']);
+
+ /// drag first to second
+ await tester.reorderFavorite(
+ fromName: '002',
+ toName: '003',
+ );
+ newNames = favorites
+ .evaluate()
+ .map((e) => (e.widget as ViewItem).view.name)
+ .toList();
+ expect(newNames, ['003', '002', '001']);
+ },
+ );
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_icon_test.dart b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_icon_test.dart
index bba172c27e..2236f03960 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_icon_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_icon_test.dart
@@ -1,83 +1,346 @@
+import 'package:appflowy/plugins/base/emoji/emoji_picker.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/base/emoji_picker_button.dart';
+import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
+import 'package:appflowy/workspace/presentation/widgets/view_title_bar.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
+import 'package:flowy_infra_ui/style_widget/text_field.dart';
+import 'package:flowy_svg/flowy_svg.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
import '../../shared/base.dart';
import '../../shared/common_operations.dart';
+import '../../shared/emoji.dart';
import '../../shared/expectation.dart';
void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ final emoji = EmojiIconData.emoji('😁');
- const emoji = '😁';
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
- group('Icon:', () {
- testWidgets('Update page icon in sidebar', (tester) async {
- await tester.initializeAppFlowy();
- await tester.tapAnonymousSignInButton();
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
- // create document, board, grid and calendar views
- for (final value in ViewLayoutPB.values) {
- if (value == ViewLayoutPB.Chat) {
- continue;
- }
- await tester.createNewPageWithNameUnderParent(
- name: value.name,
- parentName: gettingStarted,
- layout: value,
- );
+ testWidgets('Update page emoji in sidebar', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
- // update its icon
- await tester.updatePageIconInSidebarByName(
- name: value.name,
- parentName: gettingStarted,
- layout: value,
- icon: emoji,
- );
-
- tester.expectViewHasIcon(
- value.name,
- value,
- emoji,
- );
+ // create document, board, grid and calendar views
+ for (final value in ViewLayoutPB.values) {
+ if (value == ViewLayoutPB.Chat) {
+ continue;
}
- });
+ await tester.createNewPageWithNameUnderParent(
+ name: value.name,
+ parentName: gettingStarted,
+ layout: value,
+ );
- testWidgets('Update page icon in title bar', (tester) async {
- await tester.initializeAppFlowy();
- await tester.tapAnonymousSignInButton();
+ // update its emoji
+ await tester.updatePageIconInSidebarByName(
+ name: value.name,
+ parentName: gettingStarted,
+ layout: value,
+ icon: emoji,
+ );
- // create document, board, grid and calendar views
- for (final value in ViewLayoutPB.values) {
- if (value == ViewLayoutPB.Chat) {
- continue;
- }
+ tester.expectViewHasIcon(
+ value.name,
+ value,
+ emoji,
+ );
+ }
+ });
- await tester.createNewPageWithNameUnderParent(
- name: value.name,
- parentName: gettingStarted,
- layout: value,
- );
+ testWidgets('Update page emoji in title bar', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
- // update its icon
- await tester.updatePageIconInTitleBarByName(
- name: value.name,
- layout: value,
- icon: emoji,
- );
-
- tester.expectViewHasIcon(
- value.name,
- value,
- emoji,
- );
-
- tester.expectViewTitleHasIcon(
- value.name,
- value,
- emoji,
- );
+ // create document, board, grid and calendar views
+ for (final value in ViewLayoutPB.values) {
+ if (value == ViewLayoutPB.Chat) {
+ continue;
}
- });
+
+ await tester.createNewPageWithNameUnderParent(
+ name: value.name,
+ parentName: gettingStarted,
+ layout: value,
+ );
+
+ // update its emoji
+ await tester.updatePageIconInTitleBarByName(
+ name: value.name,
+ layout: value,
+ icon: emoji,
+ );
+
+ tester.expectViewHasIcon(
+ value.name,
+ value,
+ emoji,
+ );
+
+ tester.expectViewTitleHasIcon(
+ value.name,
+ value,
+ emoji,
+ );
+ }
+ });
+
+ testWidgets('Emoji Search Bar Get Focus', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create document, board, grid and calendar views
+ for (final value in ViewLayoutPB.values) {
+ if (value == ViewLayoutPB.Chat) {
+ continue;
+ }
+
+ await tester.createNewPageWithNameUnderParent(
+ name: value.name,
+ parentName: gettingStarted,
+ layout: value,
+ );
+
+ await tester.openPage(
+ value.name,
+ layout: value,
+ );
+ final title = find.descendant(
+ of: find.byType(ViewTitleBar),
+ matching: find.text(value.name),
+ );
+ await tester.tapButton(title);
+ await tester.tapButton(find.byType(EmojiPickerButton));
+
+ final emojiPicker = find.byType(FlowyEmojiPicker);
+ expect(emojiPicker, findsOneWidget);
+ final textField = find.descendant(
+ of: emojiPicker,
+ matching: find.byType(FlowyTextField),
+ );
+ expect(textField, findsOneWidget);
+ final textFieldWidget =
+ textField.evaluate().first.widget as FlowyTextField;
+ assert(textFieldWidget.focusNode!.hasFocus);
+ await tester.tapEmoji(emoji.emoji);
+ await tester.pumpAndSettle();
+ tester.expectViewHasIcon(
+ value.name,
+ value,
+ emoji,
+ );
+ }
+ });
+
+ testWidgets('Update page icon in sidebar', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ final iconData = await tester.loadIcon();
+
+ // create document, board, grid and calendar views
+ for (final value in ViewLayoutPB.values) {
+ if (value == ViewLayoutPB.Chat) {
+ continue;
+ }
+ await tester.createNewPageWithNameUnderParent(
+ name: value.name,
+ parentName: gettingStarted,
+ layout: value,
+ );
+
+ // update its icon
+ await tester.updatePageIconInSidebarByName(
+ name: value.name,
+ parentName: gettingStarted,
+ layout: value,
+ icon: iconData,
+ );
+
+ tester.expectViewHasIcon(
+ value.name,
+ value,
+ iconData,
+ );
+ }
+ });
+
+ testWidgets('Update page icon in title bar', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ final iconData = await tester.loadIcon();
+
+ // create document, board, grid and calendar views
+ for (final value in ViewLayoutPB.values) {
+ if (value == ViewLayoutPB.Chat) {
+ continue;
+ }
+
+ await tester.createNewPageWithNameUnderParent(
+ name: value.name,
+ parentName: gettingStarted,
+ layout: value,
+ );
+
+ // update its icon
+ await tester.updatePageIconInTitleBarByName(
+ name: value.name,
+ layout: value,
+ icon: iconData,
+ );
+
+ tester.expectViewHasIcon(
+ value.name,
+ value,
+ iconData,
+ );
+
+ tester.expectViewTitleHasIcon(
+ value.name,
+ value,
+ iconData,
+ );
+ }
+ });
+
+ testWidgets('Update page custom image icon in title bar', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ /// prepare local image
+ final iconData = await tester.prepareImageIcon();
+
+ // create document, board, grid and calendar views
+ for (final value in ViewLayoutPB.values) {
+ if (value == ViewLayoutPB.Chat) {
+ continue;
+ }
+
+ await tester.createNewPageWithNameUnderParent(
+ name: value.name,
+ parentName: gettingStarted,
+ layout: value,
+ );
+
+ // update its icon
+ await tester.updatePageIconInTitleBarByName(
+ name: value.name,
+ layout: value,
+ icon: iconData,
+ );
+
+ tester.expectViewHasIcon(
+ value.name,
+ value,
+ iconData,
+ );
+
+ tester.expectViewTitleHasIcon(
+ value.name,
+ value,
+ iconData,
+ );
+ }
+ });
+
+ testWidgets('Update page custom svg icon in title bar', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ /// prepare local image
+ final iconData = await tester.prepareSvgIcon();
+
+ // create document, board, grid and calendar views
+ for (final value in ViewLayoutPB.values) {
+ if (value == ViewLayoutPB.Chat) {
+ continue;
+ }
+
+ await tester.createNewPageWithNameUnderParent(
+ name: value.name,
+ parentName: gettingStarted,
+ layout: value,
+ );
+
+ // update its icon
+ await tester.updatePageIconInTitleBarByName(
+ name: value.name,
+ layout: value,
+ icon: iconData,
+ );
+
+ tester.expectViewHasIcon(
+ value.name,
+ value,
+ iconData,
+ );
+
+ tester.expectViewTitleHasIcon(
+ value.name,
+ value,
+ iconData,
+ );
+ }
+ });
+
+ testWidgets('Update page custom svg icon in title bar by pasting a link',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ /// prepare local image
+ const testIconLink =
+ 'https://beta.appflowy.cloud/api/file_storage/008e6f23-516b-4d8d-b1fe-2b75c51eee26/v1/blob/6bdf8dff%2D0e54%2D4d35%2D9981%2Dcde68bef1141/BGpLnRtb3AGBNgSJsceu70j83zevYKrMLzqsTIJcBeI=.svg';
+
+ /// create document, board, grid and calendar views
+ for (final value in ViewLayoutPB.values) {
+ if (value == ViewLayoutPB.Chat) {
+ continue;
+ }
+
+ await tester.createNewPageWithNameUnderParent(
+ name: value.name,
+ parentName: gettingStarted,
+ layout: value,
+ );
+
+ /// update its icon
+ await tester.updatePageIconInTitleBarByPasteALink(
+ name: value.name,
+ layout: value,
+ iconLink: testIconLink,
+ );
+
+ /// check if there is a svg in page
+ final pageName = tester.findPageName(
+ value.name,
+ layout: value,
+ );
+ final imageInPage = find.descendant(
+ of: pageName,
+ matching: find.byType(SvgPicture),
+ );
+ expect(imageInPage, findsOneWidget);
+
+ /// check if there is a svg in title
+ final imageInTitle = find.descendant(
+ of: find.byType(ViewTitleBar),
+ matching: find.byWidgetPredicate((w) {
+ if (w is! SvgPicture) return false;
+ final loader = w.bytesLoader;
+ if (loader is! SvgFileLoader) return false;
+ return loader.file.path.endsWith('.svg');
+ }),
+ );
+ expect(imageInTitle, findsOneWidget);
+ }
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_recent_icon_test.dart b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_recent_icon_test.dart
new file mode 100644
index 0000000000..2b724ffac1
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_recent_icon_test.dart
@@ -0,0 +1,56 @@
+import 'package:appflowy/plugins/document/presentation/editor_plugins/base/emoji_picker_button.dart';
+import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
+import 'package:appflowy/shared/icon_emoji_picker/tab.dart';
+import 'package:appflowy/workspace/presentation/widgets/view_title_bar.dart';
+import 'package:flutter_test/flutter_test.dart';
+
+import '../../shared/base.dart';
+import '../../shared/common_operations.dart';
+import '../../shared/expectation.dart';
+
+void main() {
+ testWidgets('Skip the empty group name icon in recent icons', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ /// clear local data
+ RecentIcons.clear();
+ await loadIconGroups();
+ final groups = kIconGroups!;
+ final List localIcons = [];
+ for (final e in groups) {
+ localIcons.addAll(e.icons.map((e) => RecentIcon(e, e.name)).toList());
+ }
+ await RecentIcons.putIcon(RecentIcon(localIcons.first.icon, ''));
+ await tester.openPage(gettingStarted);
+ final title = find.descendant(
+ of: find.byType(ViewTitleBar),
+ matching: find.text(gettingStarted),
+ );
+ await tester.tapButton(title);
+
+ /// tap emoji picker button
+ await tester.tapButton(find.byType(EmojiPickerButton));
+ expect(find.byType(FlowyIconEmojiPicker), findsOneWidget);
+
+ /// tap icon tab
+ final pickTab = find.byType(PickerTab);
+ final iconTab = find.descendant(
+ of: pickTab,
+ matching: find.text(PickerTabType.icon.tr),
+ );
+ await tester.tapButton(iconTab);
+
+ expect(find.byType(FlowyIconPicker), findsOneWidget);
+
+ /// no recent icons
+ final recentText = find.descendant(
+ of: find.byType(FlowyIconPicker),
+ matching: find.text('Recent'),
+ );
+ expect(recentText, findsNothing);
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_test_runner.dart b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_test_runner.dart
index 9e4212f955..ef7d3dbc8b 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_test_runner.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_test_runner.dart
@@ -2,6 +2,7 @@ import 'package:integration_test/integration_test.dart';
import 'sidebar_favorites_test.dart' as sidebar_favorite_test;
import 'sidebar_icon_test.dart' as sidebar_icon_test;
+import 'sidebar_recent_icon_test.dart' as sidebar_recent_icon_test;
import 'sidebar_test.dart' as sidebar_test;
import 'sidebar_view_item_test.dart' as sidebar_view_item_test;
@@ -14,4 +15,5 @@ void main() {
sidebar_favorite_test.main();
sidebar_icon_test.main();
sidebar_view_item_test.main();
+ sidebar_recent_icon_test.main();
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_view_item_test.dart b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_view_item_test.dart
index 6bc71b23e8..f2b721e686 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_view_item_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_view_item_test.dart
@@ -1,5 +1,7 @@
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/base/emoji/emoji_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
import 'package:appflowy/workspace/presentation/home/menu/view/view_item.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/protobuf.dart';
import 'package:easy_localization/easy_localization.dart';
@@ -11,7 +13,14 @@ import '../../shared/emoji.dart';
import '../../shared/util.dart';
void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
+
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
group('Sidebar view item tests', () {
testWidgets('Access view item context menu by right click', (tester) async {
@@ -38,7 +47,11 @@ void main() {
await tester.tapEmoji(emoji);
await tester.pumpAndSettle();
- tester.expectViewHasIcon(gettingStarted, ViewLayoutPB.Document, emoji);
+ tester.expectViewHasIcon(
+ gettingStarted,
+ ViewLayoutPB.Document,
+ EmojiIconData.emoji(emoji),
+ );
});
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/code_block_language_selector_test.dart b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/code_block_language_selector_test.dart
new file mode 100644
index 0000000000..e522e2fc73
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/code_block_language_selector_test.dart
@@ -0,0 +1,91 @@
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/code_block/code_block_language_selector.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:appflowy_editor_plugins/appflowy_editor_plugins.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/services.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/base.dart';
+import '../../shared/common_operations.dart';
+import '../../shared/document_test_operations.dart';
+import '../document/document_codeblock_paste_test.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ testWidgets('Code Block Language Selector Test', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ /// create a new document
+ await tester.createNewPageWithNameUnderParent();
+
+ /// tap editor to get focus
+ await tester.tapButton(find.byType(AppFlowyEditor));
+
+ expect(find.byType(CodeBlockLanguageSelector), findsNothing);
+ await insertCodeBlockInDocument(tester);
+
+ ///tap button
+ await tester.hoverOnWidget(find.byType(CodeBlockComponentWidget));
+ await tester
+ .tapButtonWithName(LocaleKeys.document_codeBlock_language_auto.tr());
+ expect(find.byType(CodeBlockLanguageSelector), findsOneWidget);
+
+ for (var i = 0; i < 3; ++i) {
+ await onKey(tester, LogicalKeyboardKey.arrowDown);
+ }
+ for (var i = 0; i < 2; ++i) {
+ await onKey(tester, LogicalKeyboardKey.arrowUp);
+ }
+
+ await onKey(tester, LogicalKeyboardKey.enter);
+
+ final editorState = tester.editor.getCurrentEditorState();
+ String language = editorState
+ .getNodeAtPath([0])!
+ .attributes[CodeBlockKeys.language]
+ .toString();
+ expect(
+ language.toLowerCase(),
+ defaultCodeBlockSupportedLanguages.first.toLowerCase(),
+ );
+
+ await tester.hoverOnWidget(find.byType(CodeBlockComponentWidget));
+ await tester.tapButtonWithName(language);
+
+ await onKey(tester, LogicalKeyboardKey.arrowUp);
+ await onKey(tester, LogicalKeyboardKey.enter);
+
+ language = editorState
+ .getNodeAtPath([0])!
+ .attributes[CodeBlockKeys.language]
+ .toString();
+ expect(
+ language.toLowerCase(),
+ defaultCodeBlockSupportedLanguages.last.toLowerCase(),
+ );
+
+ await tester.hoverOnWidget(find.byType(CodeBlockComponentWidget));
+ await tester.tapButtonWithName(language);
+ tester.testTextInput.enterText("rust");
+ await onKey(tester, LogicalKeyboardKey.delete);
+ await onKey(tester, LogicalKeyboardKey.delete);
+ await onKey(tester, LogicalKeyboardKey.arrowDown);
+ tester.testTextInput.enterText("st");
+ await onKey(tester, LogicalKeyboardKey.arrowDown);
+ await onKey(tester, LogicalKeyboardKey.enter);
+ language = editorState
+ .getNodeAtPath([0])!
+ .attributes[CodeBlockKeys.language]
+ .toString();
+ expect(language.toLowerCase(), 'rust');
+ });
+}
+
+Future onKey(WidgetTester tester, LogicalKeyboardKey key) async {
+ await tester.simulateKeyEvent(key);
+ await tester.pumpAndSettle();
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/emoji_shortcut_test.dart b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/emoji_shortcut_test.dart
index 554a6eecbf..1a4e57078f 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/emoji_shortcut_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/emoji_shortcut_test.dart
@@ -1,8 +1,10 @@
import 'dart:io';
+import 'package:appflowy/plugins/emoji/emoji_handler.dart';
import 'package:appflowy/workspace/presentation/settings/widgets/emoji_picker/emoji_picker.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
import 'package:appflowy_editor/src/editor/editor_component/service/editor.dart';
+import 'package:flowy_infra_ui/flowy_infra_ui.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -39,4 +41,110 @@ void main() {
expect(find.byType(EmojiSelectionMenu), findsOneWidget);
});
});
+
+ group('insert emoji by colon', () {
+ Future createNewDocumentAndShowEmojiList(
+ WidgetTester tester, {
+ String? search,
+ }) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent();
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText(':${search ?? 'a'}');
+ await tester.pumpAndSettle(Duration(seconds: 1));
+ }
+
+ testWidgets('insert with click', (tester) async {
+ await createNewDocumentAndShowEmojiList(tester);
+
+ /// emoji list is showing
+ final emojiHandler = find.byType(EmojiHandler);
+ expect(emojiHandler, findsOneWidget);
+ final emojiButtons =
+ find.descendant(of: emojiHandler, matching: find.byType(FlowyButton));
+ final firstTextFinder = find.descendant(
+ of: emojiButtons.first,
+ matching: find.byType(FlowyText),
+ );
+ final emojiText =
+ (firstTextFinder.evaluate().first.widget as FlowyText).text;
+
+ /// click first emoji item
+ await tester.tapButton(emojiButtons.first);
+ final firstNode =
+ tester.editor.getCurrentEditorState().getNodeAtPath([0])!;
+
+ /// except the emoji is in document
+ expect(emojiText.contains(firstNode.delta!.toPlainText()), true);
+ });
+
+ testWidgets('insert with arrow and enter', (tester) async {
+ await createNewDocumentAndShowEmojiList(tester);
+
+ /// emoji list is showing
+ final emojiHandler = find.byType(EmojiHandler);
+ expect(emojiHandler, findsOneWidget);
+ final emojiButtons =
+ find.descendant(of: emojiHandler, matching: find.byType(FlowyButton));
+
+ /// tap arrow down and arrow up
+ await tester.simulateKeyEvent(LogicalKeyboardKey.arrowUp);
+ await tester.simulateKeyEvent(LogicalKeyboardKey.arrowDown);
+
+ final firstTextFinder = find.descendant(
+ of: emojiButtons.first,
+ matching: find.byType(FlowyText),
+ );
+ final emojiText =
+ (firstTextFinder.evaluate().first.widget as FlowyText).text;
+
+ /// tap enter
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ final firstNode =
+ tester.editor.getCurrentEditorState().getNodeAtPath([0])!;
+
+ /// except the emoji is in document
+ expect(emojiText.contains(firstNode.delta!.toPlainText()), true);
+ });
+
+ testWidgets('insert with searching', (tester) async {
+ await createNewDocumentAndShowEmojiList(tester, search: 's');
+
+ /// search for `smiling eyes`, IME is not working, use keyboard input
+ final searchText = [
+ LogicalKeyboardKey.keyM,
+ LogicalKeyboardKey.keyI,
+ LogicalKeyboardKey.keyL,
+ LogicalKeyboardKey.keyI,
+ LogicalKeyboardKey.keyN,
+ LogicalKeyboardKey.keyG,
+ LogicalKeyboardKey.space,
+ LogicalKeyboardKey.keyE,
+ LogicalKeyboardKey.keyY,
+ LogicalKeyboardKey.keyE,
+ LogicalKeyboardKey.keyS,
+ ];
+
+ for (final key in searchText) {
+ await tester.simulateKeyEvent(key);
+ }
+
+ /// tap enter
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ final firstNode =
+ tester.editor.getCurrentEditorState().getNodeAtPath([0])!;
+
+ /// except the emoji is in document
+ expect(firstNode.delta!.toPlainText().contains('😄'), true);
+ });
+
+ testWidgets('start searching with sapce', (tester) async {
+ await createNewDocumentAndShowEmojiList(tester, search: ' ');
+
+ /// emoji list is showing
+ final emojiHandler = find.byType(EmojiHandler);
+ expect(emojiHandler, findsNothing);
+ });
+ });
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/import_files_test.dart b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/import_files_test.dart
index b4179777c9..84da89f6b7 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/import_files_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/import_files_test.dart
@@ -1,5 +1,6 @@
import 'dart:io';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
@@ -87,7 +88,7 @@ void main() {
);
expect(
importedPageEditorState.getNodeAtPath([2])!.type,
- TableBlockKeys.type,
+ SimpleTableBlockKeys.type,
);
});
});
diff --git a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/share_markdown_test.dart b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/share_markdown_test.dart
index 9a4fe30815..8c3c29ab77 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/share_markdown_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/share_markdown_test.dart
@@ -1,12 +1,16 @@
+import 'dart:convert';
import 'dart:io';
import 'package:appflowy/plugins/shared/share/share_button.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/view.pbenum.dart';
+import 'package:archive/archive.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
import 'package:path/path.dart' as p;
import '../../shared/mock/mock_file_picker.dart';
import '../../shared/util.dart';
+import '../document/document_with_database_test.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
@@ -18,7 +22,7 @@ void main() {
// mock the file picker
final path = await mockSaveFilePath(
- p.join(context.applicationDataDirectory, 'test.md'),
+ p.join(context.applicationDataDirectory, 'test.zip'),
);
// click the share button and select markdown
await tester.tapShareButton();
@@ -28,10 +32,14 @@ void main() {
tester.expectToExportSuccess();
final file = File(path);
- final isExist = file.existsSync();
- expect(isExist, true);
- final markdown = file.readAsStringSync();
- expect(markdown, expectedMarkdown);
+ expect(file.existsSync(), true);
+ final archive = ZipDecoder().decodeBytes(file.readAsBytesSync());
+ for (final entry in archive) {
+ if (entry.isFile && entry.name.endsWith('.md')) {
+ final markdown = utf8.decode(entry.content);
+ expect(markdown, expectedMarkdown);
+ }
+ }
});
testWidgets(
@@ -57,7 +65,7 @@ void main() {
final path = await mockSaveFilePath(
p.join(
context.applicationDataDirectory,
- '${shareButtonState.view.name}.md',
+ '${shareButtonState.view.name}.zip',
),
);
@@ -69,10 +77,44 @@ void main() {
tester.expectToExportSuccess();
final file = File(path);
- final isExist = file.existsSync();
- expect(isExist, true);
+ expect(file.existsSync(), true);
+ final archive = ZipDecoder().decodeBytes(file.readAsBytesSync());
+ for (final entry in archive) {
+ if (entry.isFile && entry.name.endsWith('.md')) {
+ final markdown = utf8.decode(entry.content);
+ expect(markdown, expectedMarkdown);
+ }
+ }
},
);
+
+ testWidgets('share the markdown with database', (tester) async {
+ final context = await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await insertLinkedDatabase(tester, ViewLayoutPB.Grid);
+
+ // mock the file picker
+ final path = await mockSaveFilePath(
+ p.join(context.applicationDataDirectory, 'test.zip'),
+ );
+ // click the share button and select markdown
+ await tester.tapShareButton();
+ await tester.tapMarkdownButton();
+
+ // expect to see the success dialog
+ tester.expectToExportSuccess();
+
+ final file = File(path);
+ expect(file.existsSync(), true);
+ final archive = ZipDecoder().decodeBytes(file.readAsBytesSync());
+ bool hasCsvFile = false;
+ for (final entry in archive) {
+ if (entry.isFile && entry.name.endsWith('.csv')) {
+ hasCsvFile = true;
+ }
+ }
+ expect(hasCsvFile, true);
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/tabs_test.dart b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/tabs_test.dart
index 7deea4aae4..63ec958c54 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/tabs_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/tabs_test.dart
@@ -1,17 +1,23 @@
+import 'dart:convert';
import 'dart:io';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
import 'package:appflowy/workspace/presentation/home/tabs/flowy_tab.dart';
import 'package:appflowy/workspace/presentation/home/tabs/tabs_manager.dart';
+import 'package:appflowy/workspace/presentation/widgets/tab_bar_item.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flowy_svg/flowy_svg.dart';
+import 'package:flutter/gestures.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import '../../shared/base.dart';
-import '../../shared/common_operations.dart';
-import '../../shared/expectation.dart';
import '../../shared/keyboard.dart';
+import '../../shared/util.dart';
const _documentName = 'First Doc';
const _documentTwoName = 'Second Doc';
@@ -20,17 +26,12 @@ void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
group('Tabs', () {
- testWidgets('Open AppFlowy and open/navigate/close tabs', (tester) async {
+ testWidgets('open/navigate/close tabs', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
- expect(
- find.descendant(
- of: find.byType(TabsManager),
- matching: find.byType(TabBar),
- ),
- findsNothing,
- );
+ // No tabs rendered yet
+ expect(find.byType(FlowyTab), findsNothing);
await tester.createNewPageWithNameUnderParent(name: _documentName);
@@ -44,7 +45,7 @@ void main() {
expect(
find.descendant(
- of: find.byType(TabBar),
+ of: find.byType(TabsManager),
matching: find.byType(FlowyTab),
),
findsNWidgets(3),
@@ -71,11 +72,300 @@ void main() {
expect(
find.descendant(
- of: find.byType(TabBar),
+ of: find.byType(TabsManager),
matching: find.byType(FlowyTab),
),
findsNWidgets(2),
);
});
+
+ testWidgets('right click show tab menu, close others', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ expect(
+ find.descendant(
+ of: find.byType(TabsManager),
+ matching: find.byType(TabBar),
+ ),
+ findsNothing,
+ );
+
+ await tester.createNewPageWithNameUnderParent(name: _documentName);
+ await tester.createNewPageWithNameUnderParent(name: _documentTwoName);
+
+ /// Open second menu item in a new tab
+ await tester.openAppInNewTab(gettingStarted, ViewLayoutPB.Document);
+
+ /// Open third menu item in a new tab
+ await tester.openAppInNewTab(_documentName, ViewLayoutPB.Document);
+
+ expect(
+ find.descendant(
+ of: find.byType(TabsManager),
+ matching: find.byType(FlowyTab),
+ ),
+ findsNWidgets(3),
+ );
+
+ /// Right click on second tab
+ await tester.tap(
+ buttons: kSecondaryButton,
+ find.descendant(
+ of: find.byType(FlowyTab),
+ matching: find.text(gettingStarted),
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(TabMenu), findsOneWidget);
+
+ final firstTabFinder = find.descendant(
+ of: find.byType(FlowyTab),
+ matching: find.text(_documentTwoName),
+ );
+ final secondTabFinder = find.descendant(
+ of: find.byType(FlowyTab),
+ matching: find.text(gettingStarted),
+ );
+ final thirdTabFinder = find.descendant(
+ of: find.byType(FlowyTab),
+ matching: find.text(_documentName),
+ );
+
+ expect(firstTabFinder, findsOneWidget);
+ expect(secondTabFinder, findsOneWidget);
+ expect(thirdTabFinder, findsOneWidget);
+
+ // Close other tabs than the second item
+ await tester.tap(find.text(LocaleKeys.tabMenu_closeOthers.tr()));
+ await tester.pumpAndSettle();
+
+ // We expect to not find any tabs
+ expect(firstTabFinder, findsNothing);
+ expect(secondTabFinder, findsNothing);
+ expect(thirdTabFinder, findsNothing);
+
+ // Expect second tab to be current page (current page has breadcrumb, cover title,
+ // and in this case view name in sidebar)
+ expect(find.text(gettingStarted), findsNWidgets(3));
+ });
+
+ testWidgets('cannot close pinned tabs', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ expect(
+ find.descendant(
+ of: find.byType(TabsManager),
+ matching: find.byType(TabBar),
+ ),
+ findsNothing,
+ );
+
+ await tester.createNewPageWithNameUnderParent(name: _documentName);
+ await tester.createNewPageWithNameUnderParent(name: _documentTwoName);
+
+ // Open second menu item in a new tab
+ await tester.openAppInNewTab(gettingStarted, ViewLayoutPB.Document);
+
+ // Open third menu item in a new tab
+ await tester.openAppInNewTab(_documentName, ViewLayoutPB.Document);
+
+ expect(
+ find.descendant(
+ of: find.byType(TabsManager),
+ matching: find.byType(FlowyTab),
+ ),
+ findsNWidgets(3),
+ );
+
+ const firstTab = _documentTwoName;
+ const secondTab = gettingStarted;
+ const thirdTab = _documentName;
+
+ expect(tester.isTabAtIndex(firstTab, 0), isTrue);
+ expect(tester.isTabAtIndex(secondTab, 1), isTrue);
+ expect(tester.isTabAtIndex(thirdTab, 2), isTrue);
+
+ expect(tester.isTabPinned(gettingStarted), isFalse);
+
+ // Right click on second tab
+ await tester.openTabMenu(gettingStarted);
+ expect(find.byType(TabMenu), findsOneWidget);
+
+ // Pin second tab
+ await tester.tap(find.text(LocaleKeys.tabMenu_pinTab.tr()));
+ await tester.pumpAndSettle();
+
+ expect(tester.isTabPinned(gettingStarted), isTrue);
+
+ /// Right click on first unpinned tab (second tab)
+ await tester.openTabMenu(_documentTwoName);
+
+ // Close others
+ await tester.tap(find.text(LocaleKeys.tabMenu_closeOthers.tr()));
+ await tester.pumpAndSettle();
+
+ // We expect to find 2 tabs, the first pinned tab and the second tab
+ expect(find.byType(FlowyTab), findsNWidgets(2));
+ expect(tester.isTabAtIndex(gettingStarted, 0), isTrue);
+ expect(tester.isTabAtIndex(_documentTwoName, 1), isTrue);
+ });
+
+ testWidgets('pin/unpin tabs proper order', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ expect(
+ find.descendant(
+ of: find.byType(TabsManager),
+ matching: find.byType(TabBar),
+ ),
+ findsNothing,
+ );
+
+ await tester.createNewPageWithNameUnderParent(name: _documentName);
+ await tester.createNewPageWithNameUnderParent(name: _documentTwoName);
+
+ // Open second menu item in a new tab
+ await tester.openAppInNewTab(gettingStarted, ViewLayoutPB.Document);
+
+ // Open third menu item in a new tab
+ await tester.openAppInNewTab(_documentName, ViewLayoutPB.Document);
+
+ expect(
+ find.descendant(
+ of: find.byType(TabsManager),
+ matching: find.byType(FlowyTab),
+ ),
+ findsNWidgets(3),
+ );
+
+ const firstTabName = _documentTwoName;
+ const secondTabName = gettingStarted;
+ const thirdTabName = _documentName;
+
+ // Expect correct order
+ expect(tester.isTabAtIndex(firstTabName, 0), isTrue);
+ expect(tester.isTabAtIndex(secondTabName, 1), isTrue);
+ expect(tester.isTabAtIndex(thirdTabName, 2), isTrue);
+
+ // Pin second tab
+ await tester.openTabMenu(secondTabName);
+ await tester.tap(find.text(LocaleKeys.tabMenu_pinTab.tr()));
+ await tester.pumpAndSettle();
+
+ expect(tester.isTabPinned(secondTabName), isTrue);
+
+ // Expect correct order
+ expect(tester.isTabAtIndex(secondTabName, 0), isTrue);
+ expect(tester.isTabAtIndex(firstTabName, 1), isTrue);
+ expect(tester.isTabAtIndex(thirdTabName, 2), isTrue);
+
+ // Pin new second tab (first tab)
+ await tester.openTabMenu(firstTabName);
+ await tester.tap(find.text(LocaleKeys.tabMenu_pinTab.tr()));
+ await tester.pumpAndSettle();
+
+ expect(tester.isTabPinned(firstTabName), isTrue);
+ expect(tester.isTabPinned(secondTabName), isTrue);
+ expect(tester.isTabPinned(thirdTabName), isFalse);
+
+ expect(tester.isTabAtIndex(secondTabName, 0), isTrue);
+ expect(tester.isTabAtIndex(firstTabName, 1), isTrue);
+ expect(tester.isTabAtIndex(thirdTabName, 2), isTrue);
+
+ // Unpin second tab
+ await tester.openTabMenu(secondTabName);
+ await tester.tap(find.text(LocaleKeys.tabMenu_unpinTab.tr()));
+ await tester.pumpAndSettle();
+
+ expect(tester.isTabPinned(firstTabName), isTrue);
+ expect(tester.isTabPinned(secondTabName), isFalse);
+ expect(tester.isTabPinned(thirdTabName), isFalse);
+
+ expect(tester.isTabAtIndex(firstTabName, 0), isTrue);
+ expect(tester.isTabAtIndex(secondTabName, 1), isTrue);
+ expect(tester.isTabAtIndex(thirdTabName, 2), isTrue);
+ });
+
+ testWidgets('displaying icons in tab', (tester) async {
+ RecentIcons.enable = false;
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ final icon = await tester.loadIcon();
+ // update emoji
+ await tester.updatePageIconInSidebarByName(
+ name: gettingStarted,
+ parentName: gettingStarted,
+ layout: ViewLayoutPB.Document,
+ icon: icon,
+ );
+
+ /// create new page
+ await tester.createNewPageWithNameUnderParent(name: _documentName);
+
+ /// open new tab for [gettingStarted]
+ await tester.openAppInNewTab(gettingStarted, ViewLayoutPB.Document);
+
+ final tabs = find.descendant(
+ of: find.byType(TabsManager),
+ matching: find.byType(FlowyTab),
+ );
+ expect(tabs, findsNWidgets(2));
+
+ final svgInTab =
+ find.descendant(of: tabs.last, matching: find.byType(FlowySvg));
+ final svgWidget = svgInTab.evaluate().first.widget as FlowySvg;
+ final iconsData = IconsData.fromJson(jsonDecode(icon.emoji));
+ expect(svgWidget.svgString, iconsData.svgString);
+ });
});
}
+
+extension _TabsTester on WidgetTester {
+ bool isTabPinned(String tabName) {
+ final tabFinder = find.ancestor(
+ of: find.byWidgetPredicate(
+ (w) => w is ViewTabBarItem && w.view.name == tabName,
+ ),
+ matching: find.byType(FlowyTab),
+ );
+
+ final FlowyTab tabWidget = widget(tabFinder);
+ return tabWidget.pageManager.isPinned;
+ }
+
+ bool isTabAtIndex(String tabName, int index) {
+ final tabFinder = find.ancestor(
+ of: find.byWidgetPredicate(
+ (w) => w is ViewTabBarItem && w.view.name == tabName,
+ ),
+ matching: find.byType(FlowyTab),
+ );
+
+ final pluginId = (widget(tabFinder) as FlowyTab).pageManager.plugin.id;
+
+ final pluginIds = find
+ .byType(FlowyTab)
+ .evaluate()
+ .map((e) => (e.widget as FlowyTab).pageManager.plugin.id);
+
+ return pluginIds.elementAt(index) == pluginId;
+ }
+
+ Future openTabMenu(String tabName) async {
+ await tap(
+ buttons: kSecondaryButton,
+ find.ancestor(
+ of: find.byWidgetPredicate(
+ (w) => w is ViewTabBarItem && w.view.name == tabName,
+ ),
+ matching: find.byType(FlowyTab),
+ ),
+ );
+ await pumpAndSettle();
+ }
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/uncategorized_test_runner_1.dart b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/uncategorized_test_runner_1.dart
index 4f43652c2e..836cfe4ccd 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/uncategorized_test_runner_1.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/uncategorized_test_runner_1.dart
@@ -4,7 +4,6 @@ import 'emoji_shortcut_test.dart' as emoji_shortcut_test;
import 'hotkeys_test.dart' as hotkeys_test;
import 'import_files_test.dart' as import_files_test;
import 'share_markdown_test.dart' as share_markdown_test;
-import 'tabs_test.dart' as tabs_test;
import 'zoom_in_out_test.dart' as zoom_in_out_test;
void main() {
@@ -14,10 +13,8 @@ void main() {
hotkeys_test.main();
emoji_shortcut_test.main();
hotkeys_test.main();
- emoji_shortcut_test.main();
share_markdown_test.main();
import_files_test.main();
- tabs_test.main();
zoom_in_out_test.main();
// DON'T add more tests here.
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop_runner_9.dart b/frontend/appflowy_flutter/integration_test/desktop_runner_9.dart
new file mode 100644
index 0000000000..451e24cdc1
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop_runner_9.dart
@@ -0,0 +1,20 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'desktop/database/database_icon_test.dart' as database_icon_test;
+import 'desktop/first_test/first_test.dart' as first_test;
+import 'desktop/uncategorized/code_block_language_selector_test.dart'
+ as code_language_selector;
+import 'desktop/uncategorized/tabs_test.dart' as tabs_test;
+
+Future main() async {
+ await runIntegration9OnDesktop();
+}
+
+Future runIntegration9OnDesktop() async {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ first_test.main();
+ tabs_test.main();
+ code_language_selector.main();
+ database_icon_test.main();
+}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/cloud/cloud_runner.dart b/frontend/appflowy_flutter/integration_test/mobile/cloud/cloud_runner.dart
index 9cea0ff57f..c2f3d7103a 100644
--- a/frontend/appflowy_flutter/integration_test/mobile/cloud/cloud_runner.dart
+++ b/frontend/appflowy_flutter/integration_test/mobile/cloud/cloud_runner.dart
@@ -1,9 +1,11 @@
import 'document/publish_test.dart' as publish_test;
import 'document/share_link_test.dart' as share_link_test;
+import 'space/space_test.dart' as space_test;
import 'workspace/workspace_operations_test.dart' as workspace_operations_test;
Future main() async {
workspace_operations_test.main();
share_link_test.main();
publish_test.main();
+ space_test.main();
}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/cloud/document/publish_test.dart b/frontend/appflowy_flutter/integration_test/mobile/cloud/document/publish_test.dart
index dfde191284..e6015d0896 100644
--- a/frontend/appflowy_flutter/integration_test/mobile/cloud/document/publish_test.dart
+++ b/frontend/appflowy_flutter/integration_test/mobile/cloud/document/publish_test.dart
@@ -1,47 +1,23 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-import 'dart:math';
-
import 'package:appflowy/env/cloud_env.dart';
-import 'package:appflowy/generated/flowy_svgs.g.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/mobile/application/page_style/document_page_style_bloc.dart';
-import 'package:appflowy/mobile/presentation/base/app_bar/app_bar_actions.dart';
-import 'package:appflowy/mobile/presentation/base/view_page/app_bar_buttons.dart';
-import 'package:appflowy/mobile/presentation/bottom_sheet/bottom_sheet_buttons.dart';
-import 'package:appflowy/mobile/presentation/home/home.dart';
-import 'package:appflowy/plugins/document/presentation/editor_page.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/cover/document_immersive_cover.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/cover/document_immersive_cover_bloc.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/page_style/_page_style_layout.dart';
-import 'package:appflowy/shared/patterns/common_patterns.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/user/presentation/screens/sign_in_screen/widgets/widgets.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/application/view/view_ext.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
+import 'package:appflowy/mobile/presentation/home/workspaces/create_workspace_menu.dart';
import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
import 'package:flutter/material.dart';
-import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
import '../../../shared/constants.dart';
-import '../../../shared/dir.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
group('publish:', () {
- testWidgets('publish document', (tester) async {
+ testWidgets('''
+1. publish document
+2. update path name
+3. unpublish document
+''', (tester) async {
await tester.initializeAppFlowy(
cloudType: AuthenticatorType.appflowyCloudSelfHost,
);
@@ -75,6 +51,51 @@ void main() {
findsOneWidget,
);
+ // update the path name
+ await tester.editor.clickMoreActionItemOnMobile(
+ LocaleKeys.shareAction_updatePathName.tr(),
+ );
+
+ const pathName1 = '???????????????';
+ const pathName2 = 'AppFlowy';
+
+ final textField = find.descendant(
+ of: find.byType(EditWorkspaceNameBottomSheet),
+ matching: find.byType(TextFormField),
+ );
+ await tester.enterText(textField, pathName1);
+ await tester.pumpAndSettle();
+
+ // wait 50ms to ensure the error message is shown
+ await tester.wait(50);
+
+ // click the confirm button
+ final confirmButton = find.text(LocaleKeys.button_confirm.tr());
+ await tester.tapButton(confirmButton);
+
+ // expect to see the update path name failed toast
+ final updatePathFailedText = find.text(
+ LocaleKeys.settings_sites_error_publishNameContainsInvalidCharacters
+ .tr(),
+ );
+ expect(updatePathFailedText, findsOneWidget);
+
+ // input the valid path name
+ await tester.enterText(textField, pathName2);
+ await tester.pumpAndSettle();
+ // click the confirm button
+ await tester.tapButton(confirmButton);
+
+ // wait 50ms to ensure the error message is shown
+ await tester.wait(50);
+
+ // expect to see the update path name success toast
+ final updatePathSuccessText = find.findTextInFlowyText(
+ LocaleKeys.settings_sites_success_updatePathNameSuccess.tr(),
+ );
+ expect(updatePathSuccessText, findsOneWidget);
+ await tester.pumpUntilNotFound(updatePathSuccessText);
+
// unpublish the document
await tester.editor.clickMoreActionItemOnMobile(
LocaleKeys.shareAction_unPublish.tr(),
diff --git a/frontend/appflowy_flutter/integration_test/mobile/cloud/document/share_link_test.dart b/frontend/appflowy_flutter/integration_test/mobile/cloud/document/share_link_test.dart
index 3228e785a6..bf0ddc8711 100644
--- a/frontend/appflowy_flutter/integration_test/mobile/cloud/document/share_link_test.dart
+++ b/frontend/appflowy_flutter/integration_test/mobile/cloud/document/share_link_test.dart
@@ -1,39 +1,12 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
import 'package:appflowy/env/cloud_env.dart';
-import 'package:appflowy/generated/flowy_svgs.g.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/mobile/application/page_style/document_page_style_bloc.dart';
-import 'package:appflowy/mobile/presentation/base/app_bar/app_bar_actions.dart';
-import 'package:appflowy/mobile/presentation/base/view_page/app_bar_buttons.dart';
-import 'package:appflowy/mobile/presentation/bottom_sheet/bottom_sheet_buttons.dart';
-import 'package:appflowy/mobile/presentation/home/home.dart';
-import 'package:appflowy/plugins/document/presentation/editor_page.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/cover/document_immersive_cover.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/cover/document_immersive_cover_bloc.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/page_style/_page_style_layout.dart';
import 'package:appflowy/shared/patterns/common_patterns.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/user/presentation/screens/sign_in_screen/widgets/widgets.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/application/view/view_ext.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
-import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
import '../../../shared/constants.dart';
-import '../../../shared/dir.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
void main() {
diff --git a/frontend/appflowy_flutter/integration_test/mobile/cloud/space/space_test.dart b/frontend/appflowy_flutter/integration_test/mobile/cloud/space/space_test.dart
new file mode 100644
index 0000000000..e7bf3afcc7
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/cloud/space/space_test.dart
@@ -0,0 +1,287 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/generated/flowy_svgs.g.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/mobile/presentation/bottom_sheet/bottom_sheet.dart';
+import 'package:appflowy/mobile/presentation/home/space/manage_space_widget.dart';
+import 'package:appflowy/mobile/presentation/home/space/mobile_space_header.dart';
+import 'package:appflowy/mobile/presentation/home/space/mobile_space_menu.dart';
+import 'package:appflowy/mobile/presentation/home/space/space_menu_bottom_sheet.dart';
+import 'package:appflowy/mobile/presentation/home/space/widgets.dart';
+import 'package:appflowy/mobile/presentation/home/workspaces/create_workspace_menu.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
+import 'package:appflowy/workspace/application/sidebar/space/space_bloc.dart';
+import 'package:appflowy/workspace/application/view/view_ext.dart';
+import 'package:appflowy/workspace/presentation/home/menu/sidebar/space/space_icon_popup.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/protobuf.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/cupertino.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('space operations:', () {
+ Future openSpaceMenu(WidgetTester tester) async {
+ final spaceHeader = find.byType(MobileSpaceHeader);
+ await tester.tapButton(spaceHeader);
+ await tester.pumpUntilFound(find.byType(MobileSpaceMenu));
+ }
+
+ Future openSpaceMenuMoreOptions(
+ WidgetTester tester,
+ ViewPB space,
+ ) async {
+ final spaceMenuItemTrailing = find.byWidgetPredicate(
+ (w) => w is SpaceMenuItemTrailing && w.space.id == space.id,
+ );
+ final moreOptions = find.descendant(
+ of: spaceMenuItemTrailing,
+ matching: find.byWidgetPredicate(
+ (w) =>
+ w is FlowySvg &&
+ w.svg.path == FlowySvgs.workspace_three_dots_s.path,
+ ),
+ );
+ await tester.tapButton(moreOptions);
+ await tester.pumpUntilFound(find.byType(SpaceMenuMoreOptions));
+ }
+
+ // combine the tests together to reduce the CI time
+ testWidgets('''
+1. create a new space
+2. update the space name
+3. update the space permission
+4. update the space icon
+5. delete the space
+''', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // 1. create a new space
+ // click the space menu
+ await openSpaceMenu(tester);
+
+ // click the create a new space button
+ final createNewSpaceButton = find.text(
+ LocaleKeys.space_createNewSpace.tr(),
+ );
+ await tester.pumpUntilFound(createNewSpaceButton);
+ await tester.tapButton(createNewSpaceButton);
+
+ // input the new space name
+ final inputField = find.descendant(
+ of: find.byType(ManageSpaceWidget),
+ matching: find.byType(TextField),
+ );
+ const newSpaceName = 'AppFlowy';
+ await tester.enterText(inputField, newSpaceName);
+ await tester.pumpAndSettle();
+
+ // change the space permission to private
+ final permissionOption = find.byType(ManageSpacePermissionOption);
+ await tester.tapButton(permissionOption);
+ await tester.pumpAndSettle();
+
+ final privateOption = find.text(LocaleKeys.space_privatePermission.tr());
+ await tester.tapButton(privateOption);
+ await tester.pumpAndSettle();
+
+ // change the space icon color
+ final color = builtInSpaceColors[1];
+ final iconOption = find.descendant(
+ of: find.byType(ManageSpaceIconOption),
+ matching: find.byWidgetPredicate(
+ (w) => w is SpaceColorItem && w.color == color,
+ ),
+ );
+ await tester.tapButton(iconOption);
+ await tester.pumpAndSettle();
+
+ // change the space icon
+ final icon = kIconGroups![0].icons[1];
+ final iconItem = find.descendant(
+ of: find.byType(ManageSpaceIconOption),
+ matching: find.byWidgetPredicate(
+ (w) => w is SpaceIconItem && w.icon == icon,
+ ),
+ );
+ await tester.tapButton(iconItem);
+ await tester.pumpAndSettle();
+
+ // click the done button
+ final doneButton = find.descendant(
+ of: find.byWidgetPredicate(
+ (w) =>
+ w is BottomSheetHeader &&
+ w.title == LocaleKeys.space_createSpace.tr(),
+ ),
+ matching: find.text(LocaleKeys.button_done.tr()),
+ );
+ await tester.tapButton(doneButton);
+ await tester.pumpAndSettle();
+
+ // wait 100ms for the space to be created
+ await tester.wait(100);
+
+ // verify the space is created
+ await openSpaceMenu(tester);
+ final spaceItems = find.byType(MobileSpaceMenuItem);
+ // expect to see 3 space items, 2 are built-in, 1 is the new space
+ expect(spaceItems, findsNWidgets(3));
+ // convert the space item to a widget
+ final spaceWidget =
+ tester.widgetList(spaceItems).last;
+ final space = spaceWidget.space;
+ expect(space.name, newSpaceName);
+ expect(space.spacePermission, SpacePermission.private);
+ expect(space.spaceIcon, icon.iconPath);
+ expect(space.spaceIconColor, color);
+
+ // open the SpaceMenuMoreOptions menu
+ await openSpaceMenuMoreOptions(tester, space);
+
+ // 2. rename the space name
+ final renameOption = find.text(LocaleKeys.button_rename.tr());
+ await tester.tapButton(renameOption);
+ await tester.pumpUntilFound(find.byType(EditWorkspaceNameBottomSheet));
+
+ // input the new space name
+ final renameInputField = find.descendant(
+ of: find.byType(EditWorkspaceNameBottomSheet),
+ matching: find.byType(TextField),
+ );
+ const renameSpaceName = 'HelloWorld';
+ await tester.enterText(renameInputField, renameSpaceName);
+ await tester.pumpAndSettle();
+ await tester.tapButton(find.text(LocaleKeys.button_confirm.tr()));
+
+ // click the done button
+ await tester.pumpAndSettle();
+
+ final renameSuccess = find.text(
+ LocaleKeys.space_success_renameSpace.tr(),
+ );
+ await tester.pumpUntilNotFound(renameSuccess);
+
+ // check the space name is updated
+ await openSpaceMenu(tester);
+ final renameSpaceItem = find.descendant(
+ of: find.byType(MobileSpaceMenuItem),
+ matching: find.text(renameSpaceName),
+ );
+ expect(renameSpaceItem, findsOneWidget);
+
+ // 3. manage the space
+ await openSpaceMenuMoreOptions(tester, space);
+
+ final manageOption = find.text(LocaleKeys.space_manage.tr());
+ await tester.tapButton(manageOption);
+ await tester.pumpUntilFound(find.byType(ManageSpaceWidget));
+
+ // 3.1 rename the space
+ final textField = find.descendant(
+ of: find.byType(ManageSpaceWidget),
+ matching: find.byType(TextField),
+ );
+ await tester.enterText(textField, 'AppFlowy');
+ await tester.pumpAndSettle();
+
+ // 3.2 change the permission
+ final permissionOption2 = find.byType(ManageSpacePermissionOption);
+ await tester.tapButton(permissionOption2);
+ await tester.pumpAndSettle();
+
+ final publicOption = find.text(LocaleKeys.space_publicPermission.tr());
+ await tester.tapButton(publicOption);
+ await tester.pumpAndSettle();
+
+ // 3.3 change the icon
+ // change the space icon color
+ final color2 = builtInSpaceColors[2];
+ final iconOption2 = find.descendant(
+ of: find.byType(ManageSpaceIconOption),
+ matching: find.byWidgetPredicate(
+ (w) => w is SpaceColorItem && w.color == color2,
+ ),
+ );
+ await tester.tapButton(iconOption2);
+ await tester.pumpAndSettle();
+
+ // change the space icon
+ final icon2 = kIconGroups![0].icons[2];
+ final iconItem2 = find.descendant(
+ of: find.byType(ManageSpaceIconOption),
+ matching: find.byWidgetPredicate(
+ (w) => w is SpaceIconItem && w.icon == icon2,
+ ),
+ );
+ await tester.tapButton(iconItem2);
+ await tester.pumpAndSettle();
+
+ // click the done button
+ final doneButton2 = find.descendant(
+ of: find.byWidgetPredicate(
+ (w) =>
+ w is BottomSheetHeader &&
+ w.title == LocaleKeys.space_manageSpace.tr(),
+ ),
+ matching: find.text(LocaleKeys.button_done.tr()),
+ );
+ await tester.tapButton(doneButton2);
+ await tester.pumpAndSettle();
+
+ // check the space is updated
+ final spaceItems2 = find.byType(MobileSpaceMenuItem);
+ final spaceWidget2 =
+ tester.widgetList(spaceItems2).last;
+ final space2 = spaceWidget2.space;
+ expect(space2.name, 'AppFlowy');
+ expect(space2.spacePermission, SpacePermission.publicToAll);
+ expect(space2.spaceIcon, icon2.iconPath);
+ expect(space2.spaceIconColor, color2);
+ final manageSuccess = find.text(
+ LocaleKeys.space_success_updateSpace.tr(),
+ );
+ await tester.pumpUntilNotFound(manageSuccess);
+
+ // 4. duplicate the space
+ await openSpaceMenuMoreOptions(tester, space);
+ final duplicateOption = find.text(LocaleKeys.space_duplicate.tr());
+ await tester.tapButton(duplicateOption);
+ final duplicateSuccess = find.text(
+ LocaleKeys.space_success_duplicateSpace.tr(),
+ );
+ await tester.pumpUntilNotFound(duplicateSuccess);
+
+ // check the space is duplicated
+ await openSpaceMenu(tester);
+ final spaceItems3 = find.byType(MobileSpaceMenuItem);
+ expect(spaceItems3, findsNWidgets(4));
+
+ // 5. delete the space
+ await openSpaceMenuMoreOptions(tester, space);
+ final deleteOption = find.text(LocaleKeys.button_delete.tr());
+ await tester.tapButton(deleteOption);
+ final confirmDeleteButton = find.descendant(
+ of: find.byType(CupertinoDialogAction),
+ matching: find.text(LocaleKeys.button_delete.tr()),
+ );
+ await tester.tapButton(confirmDeleteButton);
+ final deleteSuccess = find.text(
+ LocaleKeys.space_success_deleteSpace.tr(),
+ );
+ await tester.pumpUntilNotFound(deleteSuccess);
+
+ // check the space is deleted
+ final spaceItems4 = find.byType(MobileSpaceMenuItem);
+ expect(spaceItems4, findsNWidgets(3));
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/cloud/workspace/workspace_operations_test.dart b/frontend/appflowy_flutter/integration_test/mobile/cloud/workspace/workspace_operations_test.dart
index cb156121b9..210d1bcf0e 100644
--- a/frontend/appflowy_flutter/integration_test/mobile/cloud/workspace/workspace_operations_test.dart
+++ b/frontend/appflowy_flutter/integration_test/mobile/cloud/workspace/workspace_operations_test.dart
@@ -1,37 +1,11 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
import 'package:appflowy/env/cloud_env.dart';
-import 'package:appflowy/generated/flowy_svgs.g.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/mobile/application/page_style/document_page_style_bloc.dart';
-import 'package:appflowy/mobile/presentation/base/app_bar/app_bar_actions.dart';
-import 'package:appflowy/mobile/presentation/base/view_page/app_bar_buttons.dart';
-import 'package:appflowy/mobile/presentation/bottom_sheet/bottom_sheet_buttons.dart';
-import 'package:appflowy/mobile/presentation/home/home.dart';
-import 'package:appflowy/plugins/document/presentation/editor_page.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/cover/document_immersive_cover.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/cover/document_immersive_cover_bloc.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/page_style/_page_style_layout.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/user/presentation/screens/sign_in_screen/widgets/widgets.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/application/view/view_ext.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
import '../../../shared/constants.dart';
-import '../../../shared/dir.dart';
-import '../../../shared/mock/mock_file_picker.dart';
import '../../../shared/util.dart';
void main() {
diff --git a/frontend/appflowy_flutter/integration_test/mobile/document/at_menu_test.dart b/frontend/appflowy_flutter/integration_test/mobile/document/at_menu_test.dart
new file mode 100644
index 0000000000..2b348d3a2e
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/at_menu_test.dart
@@ -0,0 +1,56 @@
+import 'package:appflowy/mobile/presentation/inline_actions/mobile_inline_actions_menu.dart';
+import 'package:appflowy/mobile/presentation/inline_actions/mobile_inline_actions_menu_group.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/mention/mention_page_block.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ const title = 'Test At Menu';
+
+ group('at menu', () {
+ testWidgets('show at menu', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createPageAndShowAtMenu(title);
+ final menuWidget = find.byType(MobileInlineActionsMenu);
+ expect(menuWidget, findsOneWidget);
+ });
+
+ testWidgets('search by at menu', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createPageAndShowAtMenu(title);
+ const searchText = gettingStarted;
+ await tester.ime.insertText(searchText);
+ final actionWidgets = find.byType(MobileInlineActionsWidget);
+ expect(actionWidgets, findsNWidgets(2));
+ });
+
+ testWidgets('tap at menu', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createPageAndShowAtMenu(title);
+ const searchText = gettingStarted;
+ await tester.ime.insertText(searchText);
+ final actionWidgets = find.byType(MobileInlineActionsWidget);
+ await tester.tap(actionWidgets.last);
+ expect(find.byType(MentionPageBlock), findsOneWidget);
+ });
+
+ testWidgets('create subpage with at menu', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createNewDocumentOnMobile(title);
+ await tester.editor.tapLineOfEditorAt(0);
+ const subpageName = 'Subpage';
+ await tester.ime.insertText('[[$subpageName');
+ await tester.pumpAndSettle();
+ final actionWidgets = find.byType(MobileInlineActionsWidget);
+ await tester.tapButton(actionWidgets.first);
+ final firstNode =
+ tester.editor.getCurrentEditorState().getNodeAtPath([0]);
+ assert(firstNode != null);
+ expect(firstNode!.delta?.toPlainText().contains('['), false);
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/document/document_test_runner.dart b/frontend/appflowy_flutter/integration_test/mobile/document/document_test_runner.dart
index c719051174..90d5ca6d0d 100644
--- a/frontend/appflowy_flutter/integration_test/mobile/document/document_test_runner.dart
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/document_test_runner.dart
@@ -1,7 +1,13 @@
import 'package:integration_test/integration_test.dart';
+import 'at_menu_test.dart' as at_menu;
+import 'at_menu_test.dart' as at_menu_test;
import 'page_style_test.dart' as page_style_test;
+import 'plus_menu_test.dart' as plus_menu_test;
+import 'simple_table_test.dart' as simple_table_test;
+import 'slash_menu_test.dart' as slash_menu;
import 'title_test.dart' as title_test;
+import 'toolbar_test.dart' as toolbar_test;
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
@@ -9,4 +15,10 @@ void main() {
// Document integration tests
title_test.main();
page_style_test.main();
+ plus_menu_test.main();
+ at_menu_test.main();
+ simple_table_test.main();
+ toolbar_test.main();
+ slash_menu.main();
+ at_menu.main();
}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/document/icon_test.dart b/frontend/appflowy_flutter/integration_test/mobile/document/icon_test.dart
new file mode 100644
index 0000000000..da7c7e92e7
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/icon_test.dart
@@ -0,0 +1,104 @@
+import 'package:appflowy/generated/flowy_svgs.g.dart';
+import 'package:appflowy/mobile/presentation/base/view_page/app_bar_buttons.dart';
+import 'package:appflowy/mobile/presentation/presentation.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/header/emoji_icon_widget.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/page_style/_page_style_icon.dart';
+import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/emoji.dart';
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('document title:', () {
+ testWidgets('update page custom image icon in title bar', (tester) async {
+ await tester.launchInAnonymousMode();
+
+ /// prepare local image
+ final iconData = await tester.prepareImageIcon();
+
+ /// create an empty page
+ await tester
+ .tapButton(find.byKey(BottomNavigationBarItemType.add.valueKey));
+
+ /// show Page style page
+ await tester.tapButton(find.byType(MobileViewPageLayoutButton));
+ final pageStyleIcon = find.byType(PageStyleIcon);
+ final iconInPageStyleIcon = find.descendant(
+ of: pageStyleIcon,
+ matching: find.byType(RawEmojiIconWidget),
+ );
+ expect(iconInPageStyleIcon, findsNothing);
+
+ /// show icon picker
+ await tester.tapButton(pageStyleIcon);
+
+ /// upload custom icon
+ await tester.pickImage(iconData);
+
+ /// check result
+ final documentPage = find.byType(MobileDocumentScreen);
+ final rawEmojiIconFinder = find
+ .descendant(
+ of: documentPage,
+ matching: find.byType(RawEmojiIconWidget),
+ )
+ .last;
+ final rawEmojiIconWidget =
+ rawEmojiIconFinder.evaluate().first.widget as RawEmojiIconWidget;
+ final iconDataInWidget = rawEmojiIconWidget.emoji;
+ expect(iconDataInWidget.type, FlowyIconType.custom);
+ final imageFinder =
+ find.descendant(of: rawEmojiIconFinder, matching: find.byType(Image));
+ expect(imageFinder, findsOneWidget);
+ });
+
+ testWidgets('update page custom svg icon in title bar', (tester) async {
+ await tester.launchInAnonymousMode();
+
+ /// prepare local image
+ final iconData = await tester.prepareSvgIcon();
+
+ /// create an empty page
+ await tester
+ .tapButton(find.byKey(BottomNavigationBarItemType.add.valueKey));
+
+ /// show Page style page
+ await tester.tapButton(find.byType(MobileViewPageLayoutButton));
+ final pageStyleIcon = find.byType(PageStyleIcon);
+ final iconInPageStyleIcon = find.descendant(
+ of: pageStyleIcon,
+ matching: find.byType(RawEmojiIconWidget),
+ );
+ expect(iconInPageStyleIcon, findsNothing);
+
+ /// show icon picker
+ await tester.tapButton(pageStyleIcon);
+
+ /// upload custom icon
+ await tester.pickImage(iconData);
+
+ /// check result
+ final documentPage = find.byType(MobileDocumentScreen);
+ final rawEmojiIconFinder = find
+ .descendant(
+ of: documentPage,
+ matching: find.byType(RawEmojiIconWidget),
+ )
+ .last;
+ final rawEmojiIconWidget =
+ rawEmojiIconFinder.evaluate().first.widget as RawEmojiIconWidget;
+ final iconDataInWidget = rawEmojiIconWidget.emoji;
+ expect(iconDataInWidget.type, FlowyIconType.custom);
+ final svgFinder = find.descendant(
+ of: rawEmojiIconFinder,
+ matching: find.byType(SvgPicture),
+ );
+ expect(svgFinder, findsOneWidget);
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/document/page_style_test.dart b/frontend/appflowy_flutter/integration_test/mobile/document/page_style_test.dart
index 6e94eed1b8..e3d3bc093f 100644
--- a/frontend/appflowy_flutter/integration_test/mobile/document/page_style_test.dart
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/page_style_test.dart
@@ -1,42 +1,32 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
-import 'package:appflowy/env/cloud_env.dart';
import 'package:appflowy/generated/flowy_svgs.g.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/mobile/application/page_style/document_page_style_bloc.dart';
-import 'package:appflowy/mobile/presentation/base/app_bar/app_bar_actions.dart';
import 'package:appflowy/mobile/presentation/base/view_page/app_bar_buttons.dart';
import 'package:appflowy/mobile/presentation/bottom_sheet/bottom_sheet_buttons.dart';
-import 'package:appflowy/mobile/presentation/home/home.dart';
+import 'package:appflowy/mobile/presentation/mobile_bottom_navigation_bar.dart';
import 'package:appflowy/plugins/document/presentation/editor_page.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/cover/document_immersive_cover.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/cover/document_immersive_cover_bloc.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/page_style/_page_style_layout.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/user/presentation/screens/sign_in_screen/widgets/widgets.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/application/view/view_ext.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/page_style/_page_style_icon.dart';
+import 'package:appflowy/shared/icon_emoji_picker/recent_icons.dart';
import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
-import '../../shared/dir.dart';
-import '../../shared/mock/mock_file_picker.dart';
+import '../../shared/emoji.dart';
import '../../shared/util.dart';
void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
- group('document page style', () {
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
+
+ group('document page style:', () {
double getCurrentEditorFontSize() {
final editorPage = find
.byType(AppFlowyEditorPage)
@@ -57,11 +47,9 @@ void main() {
.single
.widget as AppFlowyEditorPage;
return editorPage.styleCustomizer
- .style()
- .textStyleConfiguration
- .text
- .height ??
- PageStyleLineHeightLayout.normal.lineHeight;
+ .style()
+ .textStyleConfiguration
+ .lineHeight;
}
testWidgets('change font size in page style settings', (tester) async {
@@ -87,20 +75,24 @@ void main() {
await tester.openPage(gettingStarted);
// click the layout button
await tester.tapButton(find.byType(MobileViewPageLayoutButton));
+ var lineHeight = getCurrentEditorLineHeight();
expect(
- getCurrentEditorLineHeight(),
+ lineHeight,
PageStyleLineHeightLayout.normal.lineHeight,
);
// change line height from normal to large
await tester.tapSvgButton(FlowySvgs.m_layout_large_s);
+ await tester.pumpAndSettle();
+ lineHeight = getCurrentEditorLineHeight();
expect(
- getCurrentEditorLineHeight(),
+ lineHeight,
PageStyleLineHeightLayout.large.lineHeight,
);
// change line height from large to small
await tester.tapSvgButton(FlowySvgs.m_layout_small_s);
+ lineHeight = getCurrentEditorLineHeight();
expect(
- getCurrentEditorLineHeight(),
+ lineHeight,
PageStyleLineHeightLayout.small.lineHeight,
);
});
@@ -135,5 +127,37 @@ void main() {
);
expect(builtInCover, findsOneWidget);
});
+
+ testWidgets('page style icon', (tester) async {
+ await tester.launchInAnonymousMode();
+
+ final createPageButton =
+ find.byKey(BottomNavigationBarItemType.add.valueKey);
+ await tester.tapButton(createPageButton);
+
+ /// toggle the preset button
+ await tester.tapSvgButton(FlowySvgs.m_layout_s);
+
+ /// select document plugins emoji
+ final pageStyleIcon = find.byType(PageStyleIcon);
+
+ /// there should be none of emoji
+ final noneText = find.text(LocaleKeys.pageStyle_none.tr());
+ expect(noneText, findsOneWidget);
+ await tester.tapButton(pageStyleIcon);
+
+ /// select an emoji
+ const emoji = '😄';
+ await tester.tapEmoji(emoji);
+ await tester.tapSvgButton(FlowySvgs.m_layout_s);
+ expect(noneText, findsNothing);
+ expect(
+ find.descendant(
+ of: pageStyleIcon,
+ matching: find.text(emoji),
+ ),
+ findsOneWidget,
+ );
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/document/plus_menu_test.dart b/frontend/appflowy_flutter/integration_test/mobile/document/plus_menu_test.dart
new file mode 100644
index 0000000000..b54c543f7e
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/plus_menu_test.dart
@@ -0,0 +1,119 @@
+import 'dart:async';
+
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/mobile/presentation/inline_actions/mobile_inline_actions_menu.dart';
+import 'package:appflowy/mobile/presentation/inline_actions/mobile_inline_actions_menu_group.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/mention/mention_page_block.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('document plus menu:', () {
+ testWidgets('add the toggle heading blocks via plus menu', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createNewDocumentOnMobile('toggle heading blocks');
+
+ final editorState = tester.editor.getCurrentEditorState();
+ // focus on the editor
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: [0])),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ // open the plus menu and select the toggle heading block
+ await tester.openPlusMenuAndClickButton(
+ LocaleKeys.document_slashMenu_name_toggleHeading1.tr(),
+ );
+
+ // check the block is inserted
+ final block1 = editorState.getNodeAtPath([0])!;
+ expect(block1.type, equals(ToggleListBlockKeys.type));
+ expect(block1.attributes[ToggleListBlockKeys.level], equals(1));
+
+ // click the expand button won't cancel the selection
+ await tester.tapButton(find.byIcon(Icons.arrow_right));
+ expect(
+ editorState.selection,
+ equals(Selection.collapsed(Position(path: [0]))),
+ );
+
+ // focus on the next line
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: [1])),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ // open the plus menu and select the toggle heading block
+ await tester.openPlusMenuAndClickButton(
+ LocaleKeys.document_slashMenu_name_toggleHeading2.tr(),
+ );
+
+ // check the block is inserted
+ final block2 = editorState.getNodeAtPath([1])!;
+ expect(block2.type, equals(ToggleListBlockKeys.type));
+ expect(block2.attributes[ToggleListBlockKeys.level], equals(2));
+
+ // focus on the next line
+ await tester.pumpAndSettle();
+
+ // open the plus menu and select the toggle heading block
+ await tester.openPlusMenuAndClickButton(
+ LocaleKeys.document_slashMenu_name_toggleHeading3.tr(),
+ );
+
+ // check the block is inserted
+ final block3 = editorState.getNodeAtPath([2])!;
+ expect(block3.type, equals(ToggleListBlockKeys.type));
+ expect(block3.attributes[ToggleListBlockKeys.level], equals(3));
+
+ // wait a few milliseconds to ensure the selection is updated
+ await Future.delayed(const Duration(milliseconds: 100));
+ // check the selection is collapsed
+ expect(
+ editorState.selection,
+ equals(Selection.collapsed(Position(path: [2]))),
+ );
+ });
+
+ const title = 'Test Plus Menu';
+ testWidgets('show plus menu', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createPageAndShowPlusMenu(title);
+ final menuWidget = find.byType(MobileInlineActionsMenu);
+ expect(menuWidget, findsOneWidget);
+ });
+
+ testWidgets('search by plus menu', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createPageAndShowPlusMenu(title);
+ const searchText = gettingStarted;
+ await tester.ime.insertText(searchText);
+ final actionWidgets = find.byType(MobileInlineActionsWidget);
+ expect(actionWidgets, findsNWidgets(2));
+ });
+
+ testWidgets('tap plus menu', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createPageAndShowPlusMenu(title);
+ const searchText = gettingStarted;
+ await tester.ime.insertText(searchText);
+ final actionWidgets = find.byType(MobileInlineActionsWidget);
+ await tester.tap(actionWidgets.last);
+ expect(find.byType(MentionPageBlock), findsOneWidget);
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/document/simple_table_test.dart b/frontend/appflowy_flutter/integration_test/mobile/document/simple_table_test.dart
new file mode 100644
index 0000000000..546baebb31
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/simple_table_test.dart
@@ -0,0 +1,554 @@
+import 'dart:async';
+
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/simple_table/simple_table.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/simple_table/simple_table_widgets/_simple_table_bottom_sheet_actions.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/cupertino.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('simple table:', () {
+ testWidgets('''
+1. insert a simple table via + menu
+2. insert a row above the table
+3. insert a row below the table
+4. insert a column left to the table
+5. insert a column right to the table
+6. delete the first row
+7. delete the first column
+''', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createNewDocumentOnMobile('simple table');
+
+ final editorState = tester.editor.getCurrentEditorState();
+ // focus on the editor
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: [0])),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final firstParagraphPath = [0, 0, 0, 0];
+
+ // open the plus menu and select the table block
+ {
+ await tester.openPlusMenuAndClickButton(
+ LocaleKeys.document_slashMenu_name_table.tr(),
+ );
+
+ // check the block is inserted
+ final table = editorState.getNodeAtPath([0])!;
+ expect(table.type, equals(SimpleTableBlockKeys.type));
+ expect(table.rowLength, equals(2));
+ expect(table.columnLength, equals(2));
+
+ // focus on the first cell
+
+ final selection = editorState.selection!;
+ expect(selection.isCollapsed, isTrue);
+ expect(selection.start.path, equals(firstParagraphPath));
+ }
+
+ // insert left and insert right
+ {
+ // click the column menu button
+ await tester.clickColumnMenuButton(0);
+
+ // insert left, insert right
+ await tester.tapButton(
+ find.findTextInFlowyText(
+ LocaleKeys.document_plugins_simpleTable_moreActions_insertLeft.tr(),
+ ),
+ );
+ await tester.tapButton(
+ find.findTextInFlowyText(
+ LocaleKeys.document_plugins_simpleTable_moreActions_insertRight
+ .tr(),
+ ),
+ );
+
+ await tester.cancelTableActionMenu();
+
+ // check the table is updated
+ final table = editorState.getNodeAtPath([0])!;
+ expect(table.type, equals(SimpleTableBlockKeys.type));
+ expect(table.rowLength, equals(2));
+ expect(table.columnLength, equals(4));
+ }
+
+ // insert above and insert below
+ {
+ // focus on the first cell
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: firstParagraphPath)),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ // click the row menu button
+ await tester.clickRowMenuButton(0);
+
+ await tester.tapButton(
+ find.findTextInFlowyText(
+ LocaleKeys.document_plugins_simpleTable_moreActions_insertAbove
+ .tr(),
+ ),
+ );
+ await tester.tapButton(
+ find.findTextInFlowyText(
+ LocaleKeys.document_plugins_simpleTable_moreActions_insertBelow
+ .tr(),
+ ),
+ );
+ await tester.cancelTableActionMenu();
+
+ // check the table is updated
+ final table = editorState.getNodeAtPath([0])!;
+ expect(table.rowLength, equals(4));
+ expect(table.columnLength, equals(4));
+ }
+
+ // delete the first row
+ {
+ // focus on the first cell
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: firstParagraphPath)),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ // delete the first row
+ await tester.clickRowMenuButton(0);
+ await tester.clickSimpleTableQuickAction(SimpleTableMoreAction.delete);
+ await tester.cancelTableActionMenu();
+
+ // check the table is updated
+ final table = editorState.getNodeAtPath([0])!;
+ expect(table.rowLength, equals(3));
+ expect(table.columnLength, equals(4));
+ }
+
+ // delete the first column
+ {
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: firstParagraphPath)),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ await tester.clickColumnMenuButton(0);
+ await tester.clickSimpleTableQuickAction(SimpleTableMoreAction.delete);
+ await tester.cancelTableActionMenu();
+
+ // check the table is updated
+ final table = editorState.getNodeAtPath([0])!;
+ expect(table.rowLength, equals(3));
+ expect(table.columnLength, equals(3));
+ }
+ });
+
+ testWidgets('''
+1. insert a simple table via + menu
+2. enable header column
+3. enable header row
+4. set to page width
+5. distribute columns evenly
+''', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createNewDocumentOnMobile('simple table');
+
+ final editorState = tester.editor.getCurrentEditorState();
+ // focus on the editor
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: [0])),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final firstParagraphPath = [0, 0, 0, 0];
+
+ // open the plus menu and select the table block
+ {
+ await tester.openPlusMenuAndClickButton(
+ LocaleKeys.document_slashMenu_name_table.tr(),
+ );
+
+ // check the block is inserted
+ final table = editorState.getNodeAtPath([0])!;
+ expect(table.type, equals(SimpleTableBlockKeys.type));
+ expect(table.rowLength, equals(2));
+ expect(table.columnLength, equals(2));
+
+ // focus on the first cell
+
+ final selection = editorState.selection!;
+ expect(selection.isCollapsed, isTrue);
+ expect(selection.start.path, equals(firstParagraphPath));
+ }
+
+ // enable header column
+ {
+ // click the column menu button
+ await tester.clickColumnMenuButton(0);
+
+ // enable header column
+ await tester.tapButton(
+ find.findTextInFlowyText(
+ LocaleKeys.document_plugins_simpleTable_moreActions_headerColumn
+ .tr(),
+ ),
+ );
+ }
+
+ // enable header row
+ {
+ // focus on the first cell
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: firstParagraphPath)),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ // click the row menu button
+ await tester.clickRowMenuButton(0);
+
+ // enable header column
+ await tester.tapButton(
+ find.findTextInFlowyText(
+ LocaleKeys.document_plugins_simpleTable_moreActions_headerRow.tr(),
+ ),
+ );
+ }
+
+ // check the table is updated
+ final table = editorState.getNodeAtPath([0])!;
+ expect(table.type, equals(SimpleTableBlockKeys.type));
+ expect(table.isHeaderColumnEnabled, isTrue);
+ expect(table.isHeaderRowEnabled, isTrue);
+
+ // disable header column
+ {
+ // focus on the first cell
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: firstParagraphPath)),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ // click the row menu button
+ await tester.clickColumnMenuButton(0);
+
+ final toggleButton = find.descendant(
+ of: find.byType(SimpleTableHeaderActionButton),
+ matching: find.byType(CupertinoSwitch),
+ );
+ await tester.tapButton(toggleButton);
+ }
+
+ // enable header row
+ {
+ // focus on the first cell
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: firstParagraphPath)),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ // click the row menu button
+ await tester.clickRowMenuButton(0);
+
+ // enable header column
+ final toggleButton = find.descendant(
+ of: find.byType(SimpleTableHeaderActionButton),
+ matching: find.byType(CupertinoSwitch),
+ );
+ await tester.tapButton(toggleButton);
+ }
+
+ // check the table is updated
+ expect(table.isHeaderColumnEnabled, isFalse);
+ expect(table.isHeaderRowEnabled, isFalse);
+
+ // set to page width
+ {
+ final table = editorState.getNodeAtPath([0])!;
+ final beforeWidth = table.width;
+ // focus on the first cell
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: firstParagraphPath)),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ // click the row menu button
+ await tester.clickRowMenuButton(0);
+
+ // enable header column
+ await tester.tapButton(
+ find.findTextInFlowyText(
+ LocaleKeys.document_plugins_simpleTable_moreActions_setToPageWidth
+ .tr(),
+ ),
+ );
+
+ // check the table is updated
+ expect(table.width, greaterThan(beforeWidth));
+ }
+
+ // distribute columns evenly
+ {
+ final table = editorState.getNodeAtPath([0])!;
+ final beforeWidth = table.width;
+
+ // focus on the first cell
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: firstParagraphPath)),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ // click the column menu button
+ await tester.clickColumnMenuButton(0);
+
+ // distribute columns evenly
+ await tester.tapButton(
+ find.findTextInFlowyText(
+ LocaleKeys
+ .document_plugins_simpleTable_moreActions_distributeColumnsWidth
+ .tr(),
+ ),
+ );
+
+ // check the table is updated
+ expect(table.width, equals(beforeWidth));
+ }
+ });
+
+ testWidgets('''
+1. insert a simple table via + menu
+2. bold
+3. clear content
+''', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createNewDocumentOnMobile('simple table');
+
+ final editorState = tester.editor.getCurrentEditorState();
+ // focus on the editor
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: [0])),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final firstParagraphPath = [0, 0, 0, 0];
+
+ // open the plus menu and select the table block
+ {
+ await tester.openPlusMenuAndClickButton(
+ LocaleKeys.document_slashMenu_name_table.tr(),
+ );
+
+ // check the block is inserted
+ final table = editorState.getNodeAtPath([0])!;
+ expect(table.type, equals(SimpleTableBlockKeys.type));
+ expect(table.rowLength, equals(2));
+ expect(table.columnLength, equals(2));
+
+ // focus on the first cell
+
+ final selection = editorState.selection!;
+ expect(selection.isCollapsed, isTrue);
+ expect(selection.start.path, equals(firstParagraphPath));
+ }
+
+ await tester.ime.insertText('Hello');
+
+ // enable bold
+ {
+ // click the column menu button
+ await tester.clickColumnMenuButton(0);
+
+ // enable bold
+ await tester.clickSimpleTableBoldContentAction();
+ await tester.cancelTableActionMenu();
+
+ // check the first cell is bold
+ final paragraph = editorState.getNodeAtPath(firstParagraphPath)!;
+ expect(paragraph.isInBoldColumn, isTrue);
+ }
+
+ // clear content
+ {
+ // focus on the first cell
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: firstParagraphPath)),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ // click the column menu button
+ await tester.clickColumnMenuButton(0);
+
+ final clearContents = find.findTextInFlowyText(
+ LocaleKeys.document_plugins_simpleTable_moreActions_clearContents
+ .tr(),
+ );
+
+ // clear content
+ final scrollable = find.descendant(
+ of: find.byType(SimpleTableBottomSheet),
+ matching: find.byType(Scrollable),
+ );
+ await tester.scrollUntilVisible(
+ clearContents,
+ 100,
+ scrollable: scrollable,
+ );
+ await tester.tapButton(clearContents);
+ await tester.cancelTableActionMenu();
+
+ // check the first cell is empty
+ final paragraph = editorState.getNodeAtPath(firstParagraphPath)!;
+ expect(paragraph.delta!, isEmpty);
+ }
+ });
+
+ testWidgets('''
+1. insert a simple table via + menu
+2. insert a heading block in table cell
+''', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createNewDocumentOnMobile('simple table');
+
+ final editorState = tester.editor.getCurrentEditorState();
+ // focus on the editor
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: [0])),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final firstParagraphPath = [0, 0, 0, 0];
+
+ // open the plus menu and select the table block
+ {
+ await tester.openPlusMenuAndClickButton(
+ LocaleKeys.document_slashMenu_name_table.tr(),
+ );
+
+ // check the block is inserted
+ final table = editorState.getNodeAtPath([0])!;
+ expect(table.type, equals(SimpleTableBlockKeys.type));
+ expect(table.rowLength, equals(2));
+ expect(table.columnLength, equals(2));
+
+ // focus on the first cell
+
+ final selection = editorState.selection!;
+ expect(selection.isCollapsed, isTrue);
+ expect(selection.start.path, equals(firstParagraphPath));
+ }
+
+ // open the plus menu and select the heading block
+ {
+ await tester.openPlusMenuAndClickButton(
+ LocaleKeys.editor_heading1.tr(),
+ );
+
+ // check the heading block is inserted
+ final heading = editorState.getNodeAtPath([0, 0, 0, 0])!;
+ expect(heading.type, equals(HeadingBlockKeys.type));
+ expect(heading.level, equals(1));
+ }
+ });
+
+ testWidgets('''
+1. insert a simple table via + menu
+2. resize column
+''', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createNewDocumentOnMobile('simple table');
+
+ final editorState = tester.editor.getCurrentEditorState();
+ // focus on the editor
+ unawaited(
+ editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: [0])),
+ reason: SelectionUpdateReason.uiEvent,
+ ),
+ );
+ await tester.pumpAndSettle();
+
+ final beforeWidth = editorState.getNodeAtPath([0, 0, 0])!.columnWidth;
+
+ // find the first cell
+ {
+ final resizeHandle = find.byType(SimpleTableColumnResizeHandle).first;
+ final offset = tester.getCenter(resizeHandle);
+ final gesture = await tester.startGesture(offset, pointer: 7);
+ await tester.pumpAndSettle();
+
+ await gesture.moveBy(const Offset(100, 0));
+ await tester.pumpAndSettle();
+
+ await gesture.up();
+ await tester.pumpAndSettle();
+ }
+
+ // check the table is updated
+ final afterWidth1 = editorState.getNodeAtPath([0, 0, 0])!.columnWidth;
+ expect(afterWidth1, greaterThan(beforeWidth));
+
+ // resize back to the original width
+ {
+ final resizeHandle = find.byType(SimpleTableColumnResizeHandle).first;
+ final offset = tester.getCenter(resizeHandle);
+ final gesture = await tester.startGesture(offset, pointer: 7);
+ await tester.pumpAndSettle();
+
+ await gesture.moveBy(const Offset(-100, 0));
+ await tester.pumpAndSettle();
+
+ await gesture.up();
+ await tester.pumpAndSettle();
+ }
+
+ // check the table is updated
+ final afterWidth2 = editorState.getNodeAtPath([0, 0, 0])!.columnWidth;
+ expect(afterWidth2, equals(beforeWidth));
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/document/slash_menu_test.dart b/frontend/appflowy_flutter/integration_test/mobile/document/slash_menu_test.dart
new file mode 100644
index 0000000000..11031d2b71
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/slash_menu_test.dart
@@ -0,0 +1,84 @@
+import 'package:appflowy/mobile/presentation/selection_menu/mobile_selection_menu_item.dart';
+import 'package:appflowy/mobile/presentation/selection_menu/mobile_selection_menu_item_widget.dart';
+import 'package:appflowy/mobile/presentation/selection_menu/mobile_selection_menu_widget.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/slash_menu/slash_menu_items/mobile_items.dart';
+import 'package:flutter/cupertino.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ const title = 'Test Slash Menu';
+
+ group('slash menu', () {
+ testWidgets('show slash menu', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createPageAndShowSlashMenu(title);
+ final menuWidget = find.byType(MobileSelectionMenuWidget);
+ expect(menuWidget, findsOneWidget);
+ final items =
+ (menuWidget.evaluate().first.widget as MobileSelectionMenuWidget)
+ .items;
+ int i = 0;
+ for (final item in items) {
+ final localItem = mobileItems[i];
+ expect(item.name, localItem.name);
+ i++;
+ }
+ });
+
+ testWidgets('search by slash menu', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createPageAndShowSlashMenu(title);
+ const searchText = 'Heading';
+ await tester.ime.insertText(searchText);
+ final itemWidgets = find.byType(MobileSelectionMenuItemWidget);
+ int number = 0;
+ for (final item in mobileItems) {
+ if (item is MobileSelectionMenuItem) {
+ for (final childItem in item.children) {
+ if (childItem.name
+ .toLowerCase()
+ .contains(searchText.toLowerCase())) {
+ number++;
+ }
+ }
+ } else {
+ if (item.name.toLowerCase().contains(searchText.toLowerCase())) {
+ number++;
+ }
+ }
+ }
+ expect(itemWidgets, findsNWidgets(number));
+ });
+
+ testWidgets('tap to show submenu', (tester) async {
+ await tester.launchInAnonymousMode();
+ await tester.createNewDocumentOnMobile(title);
+ await tester.editor.tapLineOfEditorAt(0);
+ final listview = find.descendant(
+ of: find.byType(MobileSelectionMenuWidget),
+ matching: find.byType(ListView),
+ );
+ for (final item in mobileItems) {
+ if (item is! MobileSelectionMenuItem) continue;
+ await tester.editor.showSlashMenu();
+ await tester.scrollUntilVisible(
+ find.text(item.name),
+ 50,
+ scrollable: listview,
+ duration: const Duration(milliseconds: 250),
+ );
+ await tester.tap(find.text(item.name));
+ final childrenLength = ((listview.evaluate().first.widget as ListView)
+ .childrenDelegate as SliverChildListDelegate)
+ .children
+ .length;
+ expect(childrenLength, item.children.length);
+ }
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/document/title_test.dart b/frontend/appflowy_flutter/integration_test/mobile/document/title_test.dart
index d9af265aa9..01b1d574ce 100644
--- a/frontend/appflowy_flutter/integration_test/mobile/document/title_test.dart
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/title_test.dart
@@ -1,39 +1,10 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
-import 'package:appflowy/env/cloud_env.dart';
-import 'package:appflowy/generated/flowy_svgs.g.dart';
-import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/mobile/application/page_style/document_page_style_bloc.dart';
-import 'package:appflowy/mobile/presentation/base/app_bar/app_bar_actions.dart';
-import 'package:appflowy/mobile/presentation/base/view_page/app_bar_buttons.dart';
-import 'package:appflowy/mobile/presentation/bottom_sheet/bottom_sheet_buttons.dart';
-import 'package:appflowy/mobile/presentation/home/home.dart';
-import 'package:appflowy/mobile/presentation/home/section_folder/mobile_home_section_folder_header.dart';
import 'package:appflowy/mobile/presentation/presentation.dart';
import 'package:appflowy/plugins/document/presentation/editor_page.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/cover/document_immersive_cover.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/cover/document_immersive_cover_bloc.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/page_style/_page_style_layout.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/user/presentation/screens/sign_in_screen/widgets/widgets.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/application/view/view_ext.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
-import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
-import '../../shared/dir.dart';
-import '../../shared/mock/mock_file_picker.dart';
import '../../shared/util.dart';
void main() {
diff --git a/frontend/appflowy_flutter/integration_test/mobile/document/toolbar_test.dart b/frontend/appflowy_flutter/integration_test/mobile/document/toolbar_test.dart
new file mode 100644
index 0000000000..72da283cd6
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/toolbar_test.dart
@@ -0,0 +1,117 @@
+import 'package:appflowy/generated/flowy_svgs.g.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/mobile/presentation/editor/mobile_editor_screen.dart';
+import 'package:appflowy/mobile/presentation/mobile_bottom_navigation_bar.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/mobile_toolbar_v3/appflowy_mobile_toolbar_item.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/mobile_toolbar_v3/util.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flowy_infra_ui/style_widget/text_field.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('toolbar menu:', () {
+ testWidgets('insert links', (tester) async {
+ await tester.launchInAnonymousMode();
+
+ final createPageButton = find.byKey(
+ BottomNavigationBarItemType.add.valueKey,
+ );
+ await tester.tapButton(createPageButton);
+ expect(find.byType(MobileDocumentScreen), findsOneWidget);
+
+ final editor = find.byType(AppFlowyEditor);
+ expect(editor, findsOneWidget);
+ final editorState = tester.editor.getCurrentEditorState();
+
+ /// move cursor to content
+ final root = editorState.document.root;
+ final lastNode = root.children.lastOrNull;
+ await editorState.updateSelectionWithReason(
+ Selection.collapsed(Position(path: lastNode!.path)),
+ );
+ await tester.pumpAndSettle();
+
+ /// insert two lines of text
+ const strFirst = 'FirstLine',
+ strSecond = 'SecondLine',
+ link = 'google.com';
+ await editorState.insertTextAtCurrentSelection(strFirst);
+ await tester.pumpAndSettle();
+ await editorState.insertNewLine();
+ await tester.pumpAndSettle();
+ await editorState.insertTextAtCurrentSelection(strSecond);
+ await tester.pumpAndSettle();
+ final firstLine = find.text(strFirst, findRichText: true),
+ secondLine = find.text(strSecond, findRichText: true);
+ expect(firstLine, findsOneWidget);
+ expect(secondLine, findsOneWidget);
+
+ /// select the first line
+ await tester.longPress(firstLine);
+ await tester.pumpAndSettle();
+
+ /// find aa item and tap it
+ final aaItem = find.byWidgetPredicate(
+ (widget) =>
+ widget is AppFlowyMobileToolbarIconItem &&
+ widget.icon == FlowySvgs.m_toolbar_aa_m,
+ );
+ expect(aaItem, findsOneWidget);
+ await tester.tapButton(aaItem);
+
+ /// find link button and tap it
+ final linkButton = find.byWidgetPredicate(
+ (widget) =>
+ widget is MobileToolbarMenuItemWrapper &&
+ widget.icon == FlowySvgs.m_toolbar_link_m,
+ );
+ expect(linkButton, findsOneWidget);
+ await tester.tapButton(linkButton);
+
+ /// input the link
+ final linkField = find.byWidgetPredicate(
+ (w) =>
+ w is FlowyTextField &&
+ w.hintText == LocaleKeys.document_inlineLink_url_placeholder.tr(),
+ );
+ await tester.enterText(linkField, link);
+ await tester.pumpAndSettle();
+
+ /// complete inputting
+ await tester.tapButton(find.text(LocaleKeys.button_done.tr()));
+
+ /// do it again
+ /// select the second line
+ await tester.longPress(secondLine);
+ await tester.pumpAndSettle();
+ await tester.tapButton(aaItem);
+ await tester.tapButton(linkButton);
+ await tester.enterText(linkField, link);
+ await tester.pumpAndSettle();
+ await tester.tapButton(find.text(LocaleKeys.button_done.tr()));
+
+ final firstNode = editorState.getNodeAtPath([0]);
+ final secondNode = editorState.getNodeAtPath([1]);
+
+ Map commonDeltaJson(String insert) => {
+ "insert": insert,
+ "attributes": {"href": link},
+ };
+
+ expect(
+ firstNode?.delta?.toJson(),
+ commonDeltaJson(strFirst),
+ );
+ expect(
+ secondNode?.delta?.toJson(),
+ commonDeltaJson(strSecond),
+ );
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/home_page/create_new_page_test.dart b/frontend/appflowy_flutter/integration_test/mobile/home_page/create_new_page_test.dart
index ae4e5ddea5..d64ab094de 100644
--- a/frontend/appflowy_flutter/integration_test/mobile/home_page/create_new_page_test.dart
+++ b/frontend/appflowy_flutter/integration_test/mobile/home_page/create_new_page_test.dart
@@ -1,47 +1,25 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
-import 'package:appflowy/env/cloud_env.dart';
import 'package:appflowy/generated/flowy_svgs.g.dart';
-import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/mobile/presentation/home/home.dart';
-import 'package:appflowy/mobile/presentation/home/section_folder/mobile_home_section_folder_header.dart';
import 'package:appflowy/mobile/presentation/presentation.dart';
-import 'package:appflowy/plugins/document/document_page.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/user/presentation/screens/sign_in_screen/widgets/widgets.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
-import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
-import '../../shared/dir.dart';
-import '../../shared/mock/mock_file_picker.dart';
import '../../shared/util.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- group('create new page', () {
+ group('create new page in home page:', () {
testWidgets('create document', (tester) async {
- await tester.initializeAppFlowy(
- cloudType: AuthenticatorType.local,
- );
+ await tester.launchInAnonymousMode();
// tap the create page button
final createPageButton = find.byWidgetPredicate(
(widget) =>
widget is FlowySvg &&
- widget.svg.path == FlowySvgs.m_home_unselected_m.path,
+ widget.svg.path == FlowySvgs.m_home_add_m.path,
);
await tester.tapButton(createPageButton);
+ await tester.pumpAndSettle();
expect(find.byType(MobileDocumentScreen), findsOneWidget);
});
});
diff --git a/frontend/appflowy_flutter/integration_test/mobile/settings/default_text_direction_test.dart b/frontend/appflowy_flutter/integration_test/mobile/settings/default_text_direction_test.dart
new file mode 100644
index 0000000000..158264cad1
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/settings/default_text_direction_test.dart
@@ -0,0 +1,81 @@
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/mobile/presentation/base/app_bar/app_bar_actions.dart';
+import 'package:appflowy/mobile/presentation/home/setting/settings_popup_menu.dart';
+import 'package:appflowy/mobile/presentation/mobile_bottom_navigation_bar.dart';
+import 'package:appflowy/mobile/presentation/widgets/flowy_option_tile.dart';
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/cupertino.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ testWidgets('Change default text direction', (tester) async {
+ await tester.launchInAnonymousMode();
+
+ /// tap [Setting] button
+ await tester.tapButton(find.byType(HomePageSettingsPopupMenu));
+ await tester
+ .tapButton(find.text(LocaleKeys.settings_popupMenuItem_settings.tr()));
+
+ /// tap [Default Text Direction]
+ await tester.tapButton(
+ find.text(LocaleKeys.settings_appearance_textDirection_label.tr()),
+ );
+
+ /// there are 3 items: LTR-RTL-AUTO
+ final bottomSheet = find.ancestor(
+ of: find.byType(FlowyOptionTile),
+ matching: find.byType(SafeArea),
+ );
+ final items = find.descendant(
+ of: bottomSheet,
+ matching: find.byType(FlowyOptionTile),
+ );
+ expect(items, findsNWidgets(3));
+
+ /// select [Auto]
+ await tester.tapButton(items.last);
+ expect(
+ find.text(LocaleKeys.settings_appearance_textDirection_auto.tr()),
+ findsOneWidget,
+ );
+
+ /// go back home
+ await tester.tapButton(find.byType(AppBarImmersiveBackButton));
+
+ /// create new page
+ final createPageButton =
+ find.byKey(BottomNavigationBarItemType.add.valueKey);
+ await tester.tapButton(createPageButton);
+
+ final editorState = tester.editor.getCurrentEditorState();
+ // focus on the editor
+ await tester.editor.tapLineOfEditorAt(0);
+
+ const testEnglish = 'English', testArabic = 'إنجليزي';
+
+ /// insert [testEnglish]
+ await editorState.insertTextAtCurrentSelection(testEnglish);
+ await tester.pumpAndSettle();
+ await editorState.insertNewLine(position: editorState.selection!.end);
+ await tester.pumpAndSettle();
+
+ /// insert [testArabic]
+ await editorState.insertTextAtCurrentSelection(testArabic);
+ await tester.pumpAndSettle();
+ final testEnglishFinder = find.text(testEnglish, findRichText: true),
+ testArabicFinder = find.text(testArabic, findRichText: true);
+ final testEnglishRenderBox =
+ testEnglishFinder.evaluate().first.renderObject as RenderBox,
+ testArabicRenderBox =
+ testArabicFinder.evaluate().first.renderObject as RenderBox;
+ final englishPosition = testEnglishRenderBox.localToGlobal(Offset.zero),
+ arabicPosition = testArabicRenderBox.localToGlobal(Offset.zero);
+ expect(englishPosition.dx > arabicPosition.dx, true);
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/settings/scale_factor_test.dart b/frontend/appflowy_flutter/integration_test/mobile/settings/scale_factor_test.dart
new file mode 100644
index 0000000000..908caa89d5
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/settings/scale_factor_test.dart
@@ -0,0 +1,48 @@
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/mobile/presentation/home/setting/settings_popup_menu.dart';
+import 'package:appflowy/workspace/presentation/home/hotkeys.dart';
+import 'package:appflowy/workspace/presentation/widgets/more_view_actions/widgets/font_size_stepper.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ testWidgets('test for change scale factor', (tester) async {
+ await tester.launchInAnonymousMode();
+
+ /// tap [Setting] button
+ await tester.tapButton(find.byType(HomePageSettingsPopupMenu));
+ await tester
+ .tapButton(find.text(LocaleKeys.settings_popupMenuItem_settings.tr()));
+
+ /// tap [Font Scale Factor]
+ await tester.tapButton(
+ find.text(LocaleKeys.settings_appearance_fontScaleFactor.tr()),
+ );
+
+ /// drag slider
+ final slider = find.descendant(
+ of: find.byType(FontSizeStepper),
+ matching: find.byType(Slider),
+ );
+ await tester.slideToValue(slider, 0.8);
+ expect(appflowyScaleFactor, 0.8);
+
+ await tester.slideToValue(slider, 0.9);
+ expect(appflowyScaleFactor, 0.9);
+
+ await tester.slideToValue(slider, 1.0);
+ expect(appflowyScaleFactor, 1.0);
+
+ await tester.slideToValue(slider, 1.1);
+ expect(appflowyScaleFactor, 1.1);
+
+ await tester.slideToValue(slider, 1.2);
+ expect(appflowyScaleFactor, 1.2);
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/mobile/sign_in/anonymous_sign_in_test.dart b/frontend/appflowy_flutter/integration_test/mobile/sign_in/anonymous_sign_in_test.dart
index 427e554733..ab98ca190a 100644
--- a/frontend/appflowy_flutter/integration_test/mobile/sign_in/anonymous_sign_in_test.dart
+++ b/frontend/appflowy_flutter/integration_test/mobile/sign_in/anonymous_sign_in_test.dart
@@ -1,33 +1,15 @@
-// ignore_for_file: unused_import
-
-import 'dart:io';
-
-import 'package:appflowy/env/cloud_env.dart';
-import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/mobile/presentation/home/home.dart';
-import 'package:appflowy/startup/startup.dart';
-import 'package:appflowy/user/application/auth/af_cloud_mock_auth_service.dart';
-import 'package:appflowy/user/application/auth/auth_service.dart';
-import 'package:appflowy/user/presentation/screens/sign_in_screen/widgets/widgets.dart';
-import 'package:appflowy/workspace/application/settings/prelude.dart';
-import 'package:appflowy/workspace/presentation/settings/widgets/setting_appflowy_cloud.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
-import 'package:easy_localization/easy_localization.dart';
-import 'package:flowy_infra/uuid.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
-import 'package:path/path.dart' as p;
-import '../../shared/dir.dart';
-import '../../shared/mock/mock_file_picker.dart';
import '../../shared/util.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- group('anonymous sign in on mobile', () {
+ group('anonymous sign in on mobile:', () {
testWidgets('anon user and then sign in', (tester) async {
- await tester.initializeAppFlowy();
+ await tester.launchInAnonymousMode();
// expect to see the home page
expect(find.byType(MobileHomeScreen), findsOneWidget);
diff --git a/frontend/appflowy_flutter/integration_test/mobile_runner.dart b/frontend/appflowy_flutter/integration_test/mobile_runner.dart
deleted file mode 100644
index 3f47f83997..0000000000
--- a/frontend/appflowy_flutter/integration_test/mobile_runner.dart
+++ /dev/null
@@ -1,15 +0,0 @@
-import 'package:appflowy_backend/log.dart';
-import 'package:integration_test/integration_test.dart';
-
-import 'mobile/document/page_style_test.dart' as page_style_test;
-import 'mobile/home_page/create_new_page_test.dart' as create_new_page_test;
-import 'mobile/sign_in/anonymous_sign_in_test.dart' as anonymous_sign_in_test;
-
-Future runIntegrationOnMobile() async {
- Log.shared.disableLog = true;
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
-
- anonymous_sign_in_test.main();
- create_new_page_test.main();
- page_style_test.main();
-}
diff --git a/frontend/appflowy_flutter/integration_test/mobile_runner_1.dart b/frontend/appflowy_flutter/integration_test/mobile_runner_1.dart
new file mode 100644
index 0000000000..4d92db7d25
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile_runner_1.dart
@@ -0,0 +1,23 @@
+import 'package:appflowy_backend/log.dart';
+import 'package:integration_test/integration_test.dart';
+
+import 'mobile/document/document_test_runner.dart' as document_test_runner;
+import 'mobile/home_page/create_new_page_test.dart' as create_new_page_test;
+import 'mobile/settings/default_text_direction_test.dart'
+ as default_text_direction_test;
+import 'mobile/sign_in/anonymous_sign_in_test.dart' as anonymous_sign_in_test;
+
+Future main() async {
+ Log.shared.disableLog = true;
+
+ await runIntegration1OnMobile();
+}
+
+Future runIntegration1OnMobile() async {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ anonymous_sign_in_test.main();
+ create_new_page_test.main();
+ document_test_runner.main();
+ default_text_direction_test.main();
+}
diff --git a/frontend/appflowy_flutter/integration_test/runner.dart b/frontend/appflowy_flutter/integration_test/runner.dart
index d995f81f6d..0fc3c5d826 100644
--- a/frontend/appflowy_flutter/integration_test/runner.dart
+++ b/frontend/appflowy_flutter/integration_test/runner.dart
@@ -8,7 +8,8 @@ import 'desktop_runner_5.dart';
import 'desktop_runner_6.dart';
import 'desktop_runner_7.dart';
import 'desktop_runner_8.dart';
-import 'mobile_runner.dart';
+import 'desktop_runner_9.dart';
+import 'mobile_runner_1.dart';
/// The main task runner for all integration tests in AppFlowy.
///
@@ -27,8 +28,9 @@ Future main() async {
await runIntegration6OnDesktop();
await runIntegration7OnDesktop();
await runIntegration8OnDesktop();
+ await runIntegration9OnDesktop();
} else if (Platform.isIOS || Platform.isAndroid) {
- await runIntegrationOnMobile();
+ await runIntegration1OnMobile();
} else {
throw Exception('Unsupported platform');
}
diff --git a/frontend/appflowy_flutter/integration_test/shared/base.dart b/frontend/appflowy_flutter/integration_test/shared/base.dart
index f6baa52721..493cb4c1f0 100644
--- a/frontend/appflowy_flutter/integration_test/shared/base.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/base.dart
@@ -13,6 +13,7 @@ import 'package:appflowy/workspace/application/settings/prelude.dart';
import 'package:flowy_infra/uuid.dart';
import 'package:flowy_infra_ui/flowy_infra_ui.dart';
import 'package:flutter/gestures.dart';
+import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:path/path.dart' as p;
@@ -175,6 +176,33 @@ extension AppFlowyTestBase on WidgetTester {
}
}
+ Future tapDown(
+ Finder finder, {
+ int? pointer,
+ int buttons = kPrimaryButton,
+ PointerDeviceKind kind = PointerDeviceKind.touch,
+ bool pumpAndSettle = true,
+ int milliseconds = 500,
+ }) async {
+ final location = getCenter(finder);
+ final TestGesture gesture = await startGesture(
+ location,
+ pointer: pointer,
+ buttons: buttons,
+ kind: kind,
+ );
+ await gesture.cancel();
+ await gesture.down(location);
+ await gesture.cancel();
+ if (pumpAndSettle) {
+ await this.pumpAndSettle(
+ Duration(milliseconds: milliseconds),
+ EnginePhase.sendSemanticsUpdate,
+ const Duration(seconds: 15),
+ );
+ }
+ }
+
Future tapButtonWithName(
String tr, {
int milliseconds = 500,
@@ -208,6 +236,25 @@ extension AppFlowyTestBase on WidgetTester {
Future wait(int milliseconds) async {
await pumpAndSettle(Duration(milliseconds: milliseconds));
}
+
+ Future slideToValue(
+ Finder slider,
+ double value, {
+ double paddingOffset = 24.0,
+ }) async {
+ final sliderWidget = slider.evaluate().first.widget as Slider;
+ final range = sliderWidget.max - sliderWidget.min;
+ final initialRate = (value - sliderWidget.min) / range;
+ final totalWidth = getSize(slider).width - (2 * paddingOffset);
+ final zeroPoint = getTopLeft(slider) +
+ Offset(
+ paddingOffset + initialRate * totalWidth,
+ getSize(slider).height / 2,
+ );
+ final calculatedOffset = value * (totalWidth / 100);
+ await dragFrom(zeroPoint, Offset(calculatedOffset, 0));
+ await pumpAndSettle();
+ }
}
extension AppFlowyFinderTestBase on CommonFinders {
diff --git a/frontend/appflowy_flutter/integration_test/shared/common_operations.dart b/frontend/appflowy_flutter/integration_test/shared/common_operations.dart
index 4649413717..d7a505d152 100644
--- a/frontend/appflowy_flutter/integration_test/shared/common_operations.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/common_operations.dart
@@ -4,20 +4,28 @@ import 'package:appflowy/core/config/kv.dart';
import 'package:appflowy/core/config/kv_keys.dart';
import 'package:appflowy/generated/flowy_svgs.g.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/mobile/presentation/base/type_option_menu_item.dart';
import 'package:appflowy/mobile/presentation/presentation.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/base/emoji_picker_button.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/mobile_toolbar_v3/add_block_toolbar_item.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/simple_table/simple_table.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/simple_table/simple_table_widgets/_simple_table_bottom_sheet_actions.dart';
import 'package:appflowy/plugins/shared/share/share_button.dart';
import 'package:appflowy/shared/feature_flags.dart';
+import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
import 'package:appflowy/shared/text_field/text_filed_with_metric_lines.dart';
import 'package:appflowy/startup/startup.dart';
import 'package:appflowy/user/presentation/screens/screens.dart';
import 'package:appflowy/user/presentation/screens/sign_in_screen/widgets/widgets.dart';
import 'package:appflowy/workspace/application/view/view_ext.dart';
+import 'package:appflowy/workspace/presentation/home/menu/sidebar/favorites/favorite_folder.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/footer/sidebar_footer.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/shared/sidebar_new_page_button.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/space/shared_widget.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/space/sidebar_space_header.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/space/sidebar_space_menu.dart';
+import 'package:appflowy/workspace/presentation/home/menu/sidebar/space/space_icon_popup.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/_sidebar_workspace_menu.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/workspace/sidebar_workspace.dart';
import 'package:appflowy/workspace/presentation/home/menu/view/draggable_view_item.dart';
@@ -42,6 +50,9 @@ import 'package:flutter/gestures.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
+import 'package:path/path.dart' as p;
+import 'package:path_provider/path_provider.dart';
+import 'package:universal_platform/universal_platform.dart';
import 'emoji.dart';
import 'util.dart';
@@ -56,12 +67,10 @@ extension CommonOperations on WidgetTester {
} else {
// cloud version
final anonymousButton = find.byType(SignInAnonymousButtonV2);
- await tapButton(anonymousButton);
+ await tapButton(anonymousButton, warnIfMissed: true);
}
- if (Platform.isWindows) {
- await pumpAndSettle(const Duration(milliseconds: 200));
- }
+ await pumpAndSettle(const Duration(milliseconds: 200));
}
Future tapContinousAnotherWay() async {
@@ -446,11 +455,8 @@ extension CommonOperations on WidgetTester {
// open the page after created
if (openAfterCreated) {
- await openPage(
- // if the name is null, use the default name
- pageName ?? LocaleKeys.menuAppHeader_defaultNewPageName.tr(),
- layout: layout,
- );
+ // if the name is null, use empty string
+ await openPage(pageName ?? '', layout: layout);
await pumpAndSettle();
}
}
@@ -595,6 +601,23 @@ extension CommonOperations on WidgetTester {
await pumpAndSettle();
}
+ Future reorderFavorite({
+ required String fromName,
+ required String toName,
+ }) async {
+ final from = find.descendant(
+ of: find.byType(FavoriteFolder),
+ matching: find.text(fromName),
+ ),
+ to = find.descendant(
+ of: find.byType(FavoriteFolder),
+ matching: find.text(toName),
+ );
+ final distanceY = getCenter(to).dy - getCenter(from).dx;
+ await drag(from, Offset(0, distanceY));
+ await pumpAndSettle(const Duration(seconds: 1));
+ }
+
// tap the button with [FlowySvgData]
Future tapButtonWithFlowySvgData(FlowySvgData svg) async {
final button = find.byWidgetPredicate(
@@ -606,9 +629,9 @@ extension CommonOperations on WidgetTester {
// update the page icon in the sidebar
Future updatePageIconInSidebarByName({
required String name,
- required String parentName,
+ String? parentName,
required ViewLayoutPB layout,
- required String icon,
+ required EmojiIconData icon,
}) async {
final iconButton = find.descendant(
of: findPageName(
@@ -620,7 +643,11 @@ extension CommonOperations on WidgetTester {
find.byTooltip(LocaleKeys.document_plugins_cover_changeIcon.tr()),
);
await tapButton(iconButton);
- await tapEmoji(icon);
+ if (icon.type == FlowyIconType.emoji) {
+ await tapEmoji(icon.emoji);
+ } else if (icon.type == FlowyIconType.icon) {
+ await tapIcon(icon);
+ }
await pumpAndSettle();
}
@@ -628,7 +655,7 @@ extension CommonOperations on WidgetTester {
Future updatePageIconInTitleBarByName({
required String name,
required ViewLayoutPB layout,
- required String icon,
+ required EmojiIconData icon,
}) async {
await openPage(
name,
@@ -640,7 +667,32 @@ extension CommonOperations on WidgetTester {
);
await tapButton(title);
await tapButton(find.byType(EmojiPickerButton));
- await tapEmoji(icon);
+ if (icon.type == FlowyIconType.emoji) {
+ await tapEmoji(icon.emoji);
+ } else if (icon.type == FlowyIconType.icon) {
+ await tapIcon(icon);
+ } else if (icon.type == FlowyIconType.custom) {
+ await pickImage(icon);
+ }
+ await pumpAndSettle();
+ }
+
+ Future updatePageIconInTitleBarByPasteALink({
+ required String name,
+ required ViewLayoutPB layout,
+ required String iconLink,
+ }) async {
+ await openPage(
+ name,
+ layout: layout,
+ );
+ final title = find.descendant(
+ of: find.byType(ViewTitleBar),
+ matching: find.text(name),
+ );
+ await tapButton(title);
+ await tapButton(find.byType(EmojiPickerButton));
+ await pasteImageLinkAsIcon(iconLink);
await pumpAndSettle();
}
@@ -787,6 +839,160 @@ extension CommonOperations on WidgetTester {
await tap(finder);
await pumpAndSettle(const Duration(seconds: 2));
}
+
+ /// Create a new document on mobile
+ Future createNewDocumentOnMobile(String name) async {
+ final createPageButton = find.byKey(
+ BottomNavigationBarItemType.add.valueKey,
+ );
+ await tapButton(createPageButton);
+ expect(find.byType(MobileDocumentScreen), findsOneWidget);
+
+ final title = editor.findDocumentTitle('');
+ expect(title, findsOneWidget);
+ final textField = widget(title);
+ expect(textField.focusNode!.hasFocus, isTrue);
+
+ // input new name and press done button
+ await enterText(title, name);
+ await testTextInput.receiveAction(TextInputAction.done);
+ await pumpAndSettle();
+ final newTitle = editor.findDocumentTitle(name);
+ expect(newTitle, findsOneWidget);
+ expect(textField.controller!.text, name);
+ }
+
+ /// Open the plus menu
+ Future openPlusMenuAndClickButton(String buttonName) async {
+ assert(
+ UniversalPlatform.isMobile,
+ 'This method is only supported on mobile platforms',
+ );
+
+ final plusMenuButton = find.byKey(addBlockToolbarItemKey);
+ final addMenuItem = find.byType(AddBlockMenu);
+ await tapButton(plusMenuButton);
+ await pumpUntilFound(addMenuItem);
+
+ final toggleHeading1 = find.byWidgetPredicate(
+ (widget) =>
+ widget is TypeOptionMenuItem && widget.value.text == buttonName,
+ );
+ final scrollable = find.ancestor(
+ of: find.byType(TypeOptionGridView),
+ matching: find.byType(Scrollable),
+ );
+ await scrollUntilVisible(
+ toggleHeading1,
+ 100,
+ scrollable: scrollable,
+ );
+ await tapButton(toggleHeading1);
+ await pumpUntilNotFound(addMenuItem);
+ }
+
+ /// Click the column menu button in the simple table
+ Future clickColumnMenuButton(int index) async {
+ final columnMenuButton = find.byWidgetPredicate(
+ (w) =>
+ w is SimpleTableMobileReorderButton &&
+ w.index == index &&
+ w.type == SimpleTableMoreActionType.column,
+ );
+ await tapButton(columnMenuButton);
+ await pumpUntilFound(find.byType(SimpleTableCellBottomSheet));
+ }
+
+ /// Click the row menu button in the simple table
+ Future clickRowMenuButton(int index) async {
+ final rowMenuButton = find.byWidgetPredicate(
+ (w) =>
+ w is SimpleTableMobileReorderButton &&
+ w.index == index &&
+ w.type == SimpleTableMoreActionType.row,
+ );
+ await tapButton(rowMenuButton);
+ await pumpUntilFound(find.byType(SimpleTableCellBottomSheet));
+ }
+
+ /// Click the SimpleTableQuickAction
+ Future clickSimpleTableQuickAction(SimpleTableMoreAction action) async {
+ final button = find.byWidgetPredicate(
+ (widget) => widget is SimpleTableQuickAction && widget.type == action,
+ );
+ await tapButton(button);
+ }
+
+ /// Click the SimpleTableContentAction
+ Future clickSimpleTableBoldContentAction() async {
+ final button = find.byType(SimpleTableContentBoldAction);
+ await tapButton(button);
+ }
+
+ /// Cancel the table action menu
+ Future cancelTableActionMenu() async {
+ final finder = find.byType(SimpleTableCellBottomSheet);
+ if (finder.evaluate().isEmpty) {
+ return;
+ }
+
+ await tapAt(Offset.zero);
+ await pumpUntilNotFound(finder);
+ }
+
+ /// load icon list and return the first one
+ Future loadIcon() async {
+ await loadIconGroups();
+ final groups = kIconGroups!;
+ final firstGroup = groups.first;
+ final firstIcon = firstGroup.icons.first;
+ return EmojiIconData.icon(
+ IconsData(
+ firstGroup.name,
+ firstIcon.name,
+ builtInSpaceColors.first,
+ ),
+ );
+ }
+
+ Future prepareImageIcon() async {
+ final imagePath = await rootBundle.load('assets/test/images/sample.jpeg');
+ final tempDirectory = await getTemporaryDirectory();
+ final localImagePath = p.join(tempDirectory.path, 'sample.jpeg');
+ final imageFile = File(localImagePath)
+ ..writeAsBytesSync(imagePath.buffer.asUint8List());
+ return EmojiIconData.custom(imageFile.path);
+ }
+
+ Future prepareSvgIcon() async {
+ final imagePath = await rootBundle.load('assets/test/images/sample.svg');
+ final tempDirectory = await getTemporaryDirectory();
+ final localImagePath = p.join(tempDirectory.path, 'sample.svg');
+ final imageFile = File(localImagePath)
+ ..writeAsBytesSync(imagePath.buffer.asUint8List());
+ return EmojiIconData.custom(imageFile.path);
+ }
+
+ /// create new page and show slash menu
+ Future createPageAndShowSlashMenu(String title) async {
+ await createNewDocumentOnMobile(title);
+ await editor.tapLineOfEditorAt(0);
+ await editor.showSlashMenu();
+ }
+
+ /// create new page and show at menu
+ Future createPageAndShowAtMenu(String title) async {
+ await createNewDocumentOnMobile(title);
+ await editor.tapLineOfEditorAt(0);
+ await editor.showAtMenu();
+ }
+
+ /// create new page and show plus menu
+ Future createPageAndShowPlusMenu(String title) async {
+ await createNewDocumentOnMobile(title);
+ await editor.tapLineOfEditorAt(0);
+ await editor.showPlusMenu();
+ }
}
extension SettingsFinder on CommonFinders {
diff --git a/frontend/appflowy_flutter/integration_test/shared/database_test_op.dart b/frontend/appflowy_flutter/integration_test/shared/database_test_op.dart
index ff7e33ebdf..970965f294 100644
--- a/frontend/appflowy_flutter/integration_test/shared/database_test_op.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/database_test_op.dart
@@ -1,17 +1,9 @@
import 'dart:io';
-import 'package:appflowy/plugins/database/application/field/filter_entities.dart';
-import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/select_option/condition_list.dart';
-import 'package:appflowy/plugins/database/widgets/cell/desktop_row_detail/desktop_row_detail_checklist_cell.dart';
-import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
-import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
-import 'package:flutter/gestures.dart';
-import 'package:flutter/material.dart';
-import 'package:flutter/services.dart';
-
import 'package:appflowy/generated/flowy_svgs.g.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/database/application/calculations/calculation_type_ext.dart';
+import 'package:appflowy/plugins/database/application/field/filter_entities.dart';
import 'package:appflowy/plugins/database/board/presentation/board_page.dart';
import 'package:appflowy/plugins/database/board/presentation/widgets/board_column_header.dart';
import 'package:appflowy/plugins/database/calendar/application/calendar_bloc.dart';
@@ -27,10 +19,11 @@ import 'package:appflowy/plugins/database/grid/presentation/widgets/common/type_
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/checkbox.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/checklist.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/choicechip.dart';
+import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/date.dart';
+import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/select_option/condition_list.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/select_option/option_list.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/select_option/select_option.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/text.dart';
-import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/date.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/create_filter_list.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/disclosure_button.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/footer/grid_footer.dart';
@@ -44,6 +37,7 @@ import 'package:appflowy/plugins/database/grid/presentation/widgets/toolbar/filt
import 'package:appflowy/plugins/database/grid/presentation/widgets/toolbar/sort_button.dart';
import 'package:appflowy/plugins/database/tab_bar/desktop/tab_bar_add_button.dart';
import 'package:appflowy/plugins/database/tab_bar/desktop/tab_bar_header.dart';
+import 'package:appflowy/plugins/database/widgets/cell/desktop_row_detail/desktop_row_detail_checklist_cell.dart';
import 'package:appflowy/plugins/database/widgets/cell/editable_cell_skeleton/checkbox.dart';
import 'package:appflowy/plugins/database/widgets/cell/editable_cell_skeleton/checklist.dart';
import 'package:appflowy/plugins/database/widgets/cell/editable_cell_skeleton/date.dart';
@@ -76,6 +70,8 @@ import 'package:appflowy/plugins/database/widgets/setting/database_setting_actio
import 'package:appflowy/plugins/database/widgets/setting/database_settings_list.dart';
import 'package:appflowy/plugins/database/widgets/setting/setting_button.dart';
import 'package:appflowy/plugins/database/widgets/setting/setting_property_list.dart';
+import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
import 'package:appflowy/util/field_type_extension.dart';
import 'package:appflowy/workspace/presentation/widgets/date_picker/widgets/clear_date_button.dart';
import 'package:appflowy/workspace/presentation/widgets/date_picker/widgets/date_type_option_button.dart';
@@ -90,6 +86,9 @@ import 'package:easy_localization/easy_localization.dart';
import 'package:flowy_infra_ui/flowy_infra_ui.dart';
import 'package:flowy_infra_ui/style_widget/text_input.dart';
import 'package:flowy_infra_ui/widget/buttons/primary_button.dart';
+import 'package:flutter/gestures.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter/services.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:path/path.dart' as p;
// Non-exported member of the table_calendar library
@@ -567,6 +566,12 @@ extension AppFlowyDatabaseTest on WidgetTester {
expect(phantom is PhantomChecklistItem, true);
}
+ void assertPhantomChecklistItemContent(String content) {
+ final phantom = find.byType(PhantomChecklistItem);
+ final text = find.text(content);
+ expect(find.descendant(of: phantom, matching: text), findsOneWidget);
+ }
+
Future openFirstRowDetailPage() async {
await hoverOnFirstRowOfGrid();
@@ -937,6 +942,31 @@ extension AppFlowyDatabaseTest on WidgetTester {
await pumpAndSettle(const Duration(milliseconds: 200));
}
+ Future changeFieldWidth(String fieldName, double width) async {
+ final field = find.byWidgetPredicate(
+ (widget) => widget is GridFieldCell && widget.fieldInfo.name == fieldName,
+ );
+ await hoverOnWidget(
+ field,
+ onHover: () async {
+ final dragHandle = find.descendant(
+ of: field,
+ matching: find.byType(DragToExpandLine),
+ );
+ await drag(dragHandle, Offset(width - getSize(field).width, 0));
+ await pumpAndSettle(const Duration(milliseconds: 200));
+ },
+ );
+ }
+
+ double getFieldWidth(String fieldName) {
+ final field = find.byWidgetPredicate(
+ (widget) => widget is GridFieldCell && widget.fieldInfo.name == fieldName,
+ );
+
+ return getSize(field).width;
+ }
+
Future findDateEditor(dynamic matcher) async {
final finder = find.byType(DateCellEditor);
expect(finder, matcher);
@@ -1458,6 +1488,7 @@ extension AppFlowyDatabaseTest on WidgetTester {
);
await tapButton(button);
+ await tapButtonWithName(LocaleKeys.button_delete.tr());
}
Future dragDropRescheduleCalendarEvent() async {
@@ -1565,7 +1596,7 @@ extension AppFlowyDatabaseTest on WidgetTester {
of: textField,
matching: find.byWidgetPredicate(
(widget) =>
- widget is FlowySvg && widget.svg == FlowySvgs.close_filled_m,
+ widget is FlowySvg && widget.svg == FlowySvgs.close_filled_s,
),
),
);
diff --git a/frontend/appflowy_flutter/integration_test/shared/document_test_operations.dart b/frontend/appflowy_flutter/integration_test/shared/document_test_operations.dart
index 491ac9432c..398a3f9657 100644
--- a/frontend/appflowy_flutter/integration_test/shared/document_test_operations.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/document_test_operations.dart
@@ -307,9 +307,11 @@ class EditorOperations {
Future openTurnIntoMenu(Path path) async {
await hoverAndClickOptionMenuButton(path);
await tester.tapButton(
- find.findTextInFlowyText(
- LocaleKeys.document_plugins_optionAction_turnInto.tr(),
- ),
+ find
+ .findTextInFlowyText(
+ LocaleKeys.document_plugins_optionAction_turnInto.tr(),
+ )
+ .first,
);
await tester.pumpUntilFound(find.byType(TurnIntoOptionMenu));
}
diff --git a/frontend/appflowy_flutter/integration_test/shared/emoji.dart b/frontend/appflowy_flutter/integration_test/shared/emoji.dart
index d439a9b3f7..cccd00a3f6 100644
--- a/frontend/appflowy_flutter/integration_test/shared/emoji.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/emoji.dart
@@ -1,7 +1,24 @@
+import 'dart:convert';
+import 'dart:io';
+
+import 'package:appflowy/plugins/document/presentation/editor_plugins/copy_and_paste/clipboard_service.dart';
+import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_color_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_uploader.dart';
+import 'package:appflowy/shared/icon_emoji_picker/tab.dart';
+import 'package:appflowy/startup/startup.dart';
+import 'package:appflowy/workspace/presentation/home/menu/sidebar/space/space_icon_popup.dart';
+import 'package:desktop_drop/desktop_drop.dart';
+import 'package:flowy_infra_ui/style_widget/primary_rounded_button.dart';
+import 'package:flowy_svg/flowy_svg.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter/services.dart';
import 'package:flutter_emoji_mart/flutter_emoji_mart.dart';
import 'package:flutter_test/flutter_test.dart';
import 'base.dart';
+import 'common_operations.dart';
extension EmojiTestExtension on WidgetTester {
Future tapEmoji(String emoji) async {
@@ -11,4 +28,117 @@ extension EmojiTestExtension on WidgetTester {
);
await tapButton(emojiWidget);
}
+
+ Future tapIcon(EmojiIconData icon, {bool enableColor = true}) async {
+ final iconsData = IconsData.fromJson(jsonDecode(icon.emoji));
+ final pickTab = find.byType(PickerTab);
+ expect(pickTab, findsOneWidget);
+ await pumpAndSettle();
+ final iconTab = find.descendant(
+ of: pickTab,
+ matching: find.text(PickerTabType.icon.tr),
+ );
+ expect(iconTab, findsOneWidget);
+ await tapButton(iconTab);
+ final selectedSvg = find.descendant(
+ of: find.byType(FlowyIconPicker),
+ matching: find.byWidgetPredicate(
+ (w) => w is FlowySvg && w.svgString == iconsData.svgString,
+ ),
+ );
+
+ await tapButton(selectedSvg.first);
+ if (enableColor) {
+ final colorPicker = find.byType(IconColorPicker);
+ expect(colorPicker, findsOneWidget);
+ final selectedColor = find.descendant(
+ of: colorPicker,
+ matching: find.byWidgetPredicate((w) {
+ if (w is Container) {
+ final d = w.decoration;
+ if (d is ShapeDecoration) {
+ if (d.color ==
+ Color(
+ int.parse(iconsData.color ?? builtInSpaceColors.first),
+ )) {
+ return true;
+ }
+ }
+ }
+ return false;
+ }),
+ );
+ await tapButton(selectedColor);
+ }
+ }
+
+ Future pickImage(EmojiIconData icon) async {
+ final pickTab = find.byType(PickerTab);
+ expect(pickTab, findsOneWidget);
+ await pumpAndSettle();
+
+ /// switch to custom tab
+ final iconTab = find.descendant(
+ of: pickTab,
+ matching: find.text(PickerTabType.custom.tr),
+ );
+ expect(iconTab, findsOneWidget);
+ await tapButton(iconTab);
+
+ /// mock for dragging image
+ final dropTarget = find.descendant(
+ of: find.byType(IconUploader),
+ matching: find.byType(DropTarget),
+ );
+ expect(dropTarget, findsOneWidget);
+ final dropTargetWidget = dropTarget.evaluate().first.widget as DropTarget;
+ dropTargetWidget.onDragDone?.call(
+ DropDoneDetails(
+ files: [DropItemFile(icon.emoji)],
+ localPosition: Offset.zero,
+ globalPosition: Offset.zero,
+ ),
+ );
+ await pumpAndSettle(const Duration(seconds: 3));
+
+ /// confirm to upload
+ final confirmButton = find.descendant(
+ of: find.byType(IconUploader),
+ matching: find.byType(PrimaryRoundedButton),
+ );
+ await tapButton(confirmButton);
+ }
+
+ Future pasteImageLinkAsIcon(String link) async {
+ final pickTab = find.byType(PickerTab);
+ expect(pickTab, findsOneWidget);
+ await pumpAndSettle();
+
+ /// switch to custom tab
+ final iconTab = find.descendant(
+ of: pickTab,
+ matching: find.text(PickerTabType.custom.tr),
+ );
+ expect(iconTab, findsOneWidget);
+ await tapButton(iconTab);
+
+ // mock the clipboard
+ await getIt()
+ .setData(ClipboardServiceData(plainText: link));
+
+ // paste the link
+ await simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await pumpAndSettle(const Duration(seconds: 5));
+
+ /// confirm to upload
+ final confirmButton = find.descendant(
+ of: find.byType(IconUploader),
+ matching: find.byType(PrimaryRoundedButton),
+ );
+ await tapButton(confirmButton);
+ }
}
diff --git a/frontend/appflowy_flutter/integration_test/shared/expectation.dart b/frontend/appflowy_flutter/integration_test/shared/expectation.dart
index e7625331ee..3b9ef0d75c 100644
--- a/frontend/appflowy_flutter/integration_test/shared/expectation.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/expectation.dart
@@ -1,3 +1,5 @@
+import 'dart:convert';
+
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/mobile/presentation/page_item/mobile_view_item.dart';
import 'package:appflowy/mobile/presentation/presentation.dart';
@@ -5,6 +7,10 @@ import 'package:appflowy/plugins/database/widgets/row/row_detail.dart';
import 'package:appflowy/plugins/document/presentation/banner.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/header/document_cover_widget.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/header/emoji_icon_widget.dart';
+import 'package:appflowy/shared/appflowy_network_image.dart';
+import 'package:appflowy/shared/appflowy_network_svg.dart';
+import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
+import 'package:appflowy/shared/icon_emoji_picker/icon_picker.dart';
import 'package:appflowy/workspace/application/sidebar/folder/folder_bloc.dart';
import 'package:appflowy/workspace/presentation/home/home_stack.dart';
import 'package:appflowy/workspace/presentation/home/menu/view/view_item.dart';
@@ -14,8 +20,10 @@ import 'package:appflowy/workspace/presentation/widgets/view_title_bar.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flowy_infra_ui/flowy_infra_ui.dart';
+import 'package:flowy_svg/flowy_svg.dart';
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
+import 'package:string_validator/string_validator.dart';
import 'package:universal_platform/universal_platform.dart';
import 'util.dart';
@@ -117,7 +125,7 @@ extension Expectation on WidgetTester {
return;
}
final iconWidget = find.byWidgetPredicate(
- (widget) => widget is EmojiIconWidget && widget.emoji == emoji,
+ (widget) => widget is EmojiIconWidget && widget.emoji.emoji == emoji,
);
expect(iconWidget, findsOneWidget);
}
@@ -223,24 +231,93 @@ extension Expectation on WidgetTester {
);
}
- void expectViewHasIcon(String name, ViewLayoutPB layout, String emoji) {
+ void expectViewHasIcon(String name, ViewLayoutPB layout, EmojiIconData data) {
final pageName = findPageName(
name,
layout: layout,
);
- final icon = find.descendant(
- of: pageName,
- matching: find.text(emoji),
- );
- expect(icon, findsOneWidget);
+ final type = data.type;
+ if (type == FlowyIconType.emoji) {
+ final icon = find.descendant(
+ of: pageName,
+ matching: find.text(data.emoji),
+ );
+ expect(icon, findsOneWidget);
+ } else if (type == FlowyIconType.icon) {
+ final iconsData = IconsData.fromJson(jsonDecode(data.emoji));
+ final icon = find.descendant(
+ of: pageName,
+ matching: find.byWidgetPredicate(
+ (w) => w is FlowySvg && w.svgString == iconsData.svgString,
+ ),
+ );
+ expect(icon, findsOneWidget);
+ } else if (type == FlowyIconType.custom) {
+ final isSvg = data.emoji.endsWith('.svg');
+ if (isURL(data.emoji)) {
+ final image = find.descendant(
+ of: pageName,
+ matching: isSvg
+ ? find.byType(FlowyNetworkSvg)
+ : find.byType(FlowyNetworkImage),
+ );
+ expect(image, findsOneWidget);
+ } else {
+ final image = find.descendant(
+ of: pageName,
+ matching: isSvg ? find.byType(SvgPicture) : find.byType(Image),
+ );
+ expect(image, findsOneWidget);
+ }
+ }
}
- void expectViewTitleHasIcon(String name, ViewLayoutPB layout, String emoji) {
- final icon = find.descendant(
- of: find.byType(ViewTitleBar),
- matching: find.text(emoji),
- );
- expect(icon, findsOneWidget);
+ void expectViewTitleHasIcon(
+ String name,
+ ViewLayoutPB layout,
+ EmojiIconData data,
+ ) {
+ final type = data.type;
+ if (type == FlowyIconType.emoji) {
+ final icon = find.descendant(
+ of: find.byType(ViewTitleBar),
+ matching: find.text(data.emoji),
+ );
+ expect(icon, findsOneWidget);
+ } else if (type == FlowyIconType.icon) {
+ final iconsData = IconsData.fromJson(jsonDecode(data.emoji));
+ final icon = find.descendant(
+ of: find.byType(ViewTitleBar),
+ matching: find.byWidgetPredicate(
+ (w) => w is FlowySvg && w.svgString == iconsData.svgString,
+ ),
+ );
+ expect(icon, findsOneWidget);
+ } else if (type == FlowyIconType.custom) {
+ final isSvg = data.emoji.endsWith('.svg');
+ if (isURL(data.emoji)) {
+ final image = find.descendant(
+ of: find.byType(ViewTitleBar),
+ matching: isSvg
+ ? find.byType(FlowyNetworkSvg)
+ : find.byType(FlowyNetworkImage),
+ );
+ expect(image, findsOneWidget);
+ } else {
+ final image = find.descendant(
+ of: find.byType(ViewTitleBar),
+ matching: isSvg
+ ? find.byWidgetPredicate((w) {
+ if (w is! SvgPicture) return false;
+ final loader = w.bytesLoader;
+ if (loader is! SvgFileLoader) return false;
+ return loader.file.path.endsWith('.svg');
+ })
+ : find.byType(Image),
+ );
+ expect(image, findsOneWidget);
+ }
+ }
}
void expectSelectedReminder(ReminderOption option) {
diff --git a/frontend/appflowy_flutter/integration_test/shared/mock/mock_openai_repository.dart b/frontend/appflowy_flutter/integration_test/shared/mock/mock_openai_repository.dart
deleted file mode 100644
index 7201bd89ca..0000000000
--- a/frontend/appflowy_flutter/integration_test/shared/mock/mock_openai_repository.dart
+++ /dev/null
@@ -1,81 +0,0 @@
-import 'dart:async';
-import 'dart:convert';
-
-import 'package:appflowy/plugins/document/presentation/editor_plugins/openai/service/error.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/openai/service/openai_client.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/openai/service/text_completion.dart';
-import 'package:http/http.dart' as http;
-import 'package:mocktail/mocktail.dart';
-
-class MyMockClient extends Mock implements http.Client {
- @override
- Future send(http.BaseRequest request) async {
- final requestType = request.method;
- final requestUri = request.url;
-
- if (requestType == 'POST' &&
- requestUri == OpenAIRequestType.textCompletion.uri) {
- final responseHeaders = {
- 'content-type': 'text/event-stream',
- };
- final responseBody = Stream.fromIterable([
- utf8.encode(
- '{ "choices": [{"text": "Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Aenean commodo ligula ", "index": 0, "logprobs": null, "finish_reason": null}]}',
- ),
- utf8.encode('\n'),
- utf8.encode('[DONE]'),
- ]);
-
- // Return a mocked response with the expected data
- return http.StreamedResponse(responseBody, 200, headers: responseHeaders);
- }
-
- // Return an error response for any other request
- return http.StreamedResponse(const Stream.empty(), 404);
- }
-}
-
-class MockOpenAIRepository extends HttpOpenAIRepository {
- MockOpenAIRepository() : super(apiKey: 'dummyKey', client: MyMockClient());
-
- @override
- Future getStreamedCompletions({
- required String prompt,
- required Future Function() onStart,
- required Future Function(TextCompletionResponse response) onProcess,
- required Future Function() onEnd,
- required void Function(AIError error) onError,
- String? suffix,
- int maxTokens = 2048,
- double temperature = 0.3,
- bool useAction = false,
- }) async {
- final request = http.Request('POST', OpenAIRequestType.textCompletion.uri);
- final response = await client.send(request);
-
- String previousSyntax = '';
- if (response.statusCode == 200) {
- await for (final chunk in response.stream
- .transform(const Utf8Decoder())
- .transform(const LineSplitter())) {
- await onStart();
- final data = chunk.trim().split('data: ');
- if (data[0] != '[DONE]') {
- final response = TextCompletionResponse.fromJson(
- json.decode(data[0]),
- );
- if (response.choices.isNotEmpty) {
- final text = response.choices.first.text;
- if (text == previousSyntax && text == '\n') {
- continue;
- }
- await onProcess(response);
- previousSyntax = response.choices.first.text;
- }
- } else {
- await onEnd();
- }
- }
- }
- }
-}
diff --git a/frontend/appflowy_flutter/integration_test/shared/settings.dart b/frontend/appflowy_flutter/integration_test/shared/settings.dart
index aade7bb4c9..bfc5efedde 100644
--- a/frontend/appflowy_flutter/integration_test/shared/settings.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/settings.dart
@@ -79,7 +79,7 @@ extension AppFlowySettings on WidgetTester {
// Enable editing username
final editUsernameFinder = find.descendant(
of: find.byType(AccountUserProfile),
- matching: find.byFlowySvg(FlowySvgs.edit_s),
+ matching: find.byFlowySvg(FlowySvgs.toolbar_link_edit_m),
);
await tap(editUsernameFinder, warnIfMissed: false);
await pumpAndSettle();
diff --git a/frontend/appflowy_flutter/integration_test/shared/workspace.dart b/frontend/appflowy_flutter/integration_test/shared/workspace.dart
index 67506879d5..1b2f22b944 100644
--- a/frontend/appflowy_flutter/integration_test/shared/workspace.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/workspace.dart
@@ -40,9 +40,13 @@ extension AppFlowyWorkspace on WidgetTester {
moreButton,
onHover: () async {
await tapButton(moreButton);
- await tapButton(
- find.findTextInFlowyText(LocaleKeys.button_rename.tr()),
+ // wait for the menu to open
+ final renameButton = find.findTextInFlowyText(
+ LocaleKeys.button_rename.tr(),
);
+ await pumpUntilFound(renameButton);
+ expect(renameButton, findsOneWidget);
+ await tapButton(renameButton);
final input = find.byType(TextFormField);
expect(input, findsOneWidget);
await enterText(input, name);
diff --git a/frontend/appflowy_flutter/ios/Podfile.lock b/frontend/appflowy_flutter/ios/Podfile.lock
index 3089e3f56e..4b7ed5d639 100644
--- a/frontend/appflowy_flutter/ios/Podfile.lock
+++ b/frontend/appflowy_flutter/ios/Podfile.lock
@@ -1,5 +1,5 @@
PODS:
- - app_links (0.0.1):
+ - app_links (0.0.2):
- Flutter
- appflowy_backend (0.0.1):
- Flutter
@@ -66,6 +66,8 @@ PODS:
- permission_handler_apple (9.3.0):
- Flutter
- ReachabilitySwift (5.0.0)
+ - saver_gallery (0.0.1):
+ - Flutter
- SDWebImage (5.14.2):
- SDWebImage/Core (= 5.14.2)
- SDWebImage/Core (5.14.2)
@@ -79,7 +81,7 @@ PODS:
- shared_preferences_foundation (0.0.1):
- Flutter
- FlutterMacOS
- - sqflite (0.0.3):
+ - sqflite_darwin (0.0.4):
- Flutter
- FlutterMacOS
- super_native_extensions (0.0.1):
@@ -88,6 +90,9 @@ PODS:
- Toast (4.0.0)
- url_launcher_ios (0.0.1):
- Flutter
+ - webview_flutter_wkwebview (0.0.1):
+ - Flutter
+ - FlutterMacOS
DEPENDENCIES:
- app_links (from `.symlinks/plugins/app_links/ios`)
@@ -106,12 +111,14 @@ DEPENDENCIES:
- package_info_plus (from `.symlinks/plugins/package_info_plus/ios`)
- path_provider_foundation (from `.symlinks/plugins/path_provider_foundation/darwin`)
- permission_handler_apple (from `.symlinks/plugins/permission_handler_apple/ios`)
+ - saver_gallery (from `.symlinks/plugins/saver_gallery/ios`)
- sentry_flutter (from `.symlinks/plugins/sentry_flutter/ios`)
- share_plus (from `.symlinks/plugins/share_plus/ios`)
- shared_preferences_foundation (from `.symlinks/plugins/shared_preferences_foundation/darwin`)
- - sqflite (from `.symlinks/plugins/sqflite/darwin`)
+ - sqflite_darwin (from `.symlinks/plugins/sqflite_darwin/darwin`)
- super_native_extensions (from `.symlinks/plugins/super_native_extensions/ios`)
- url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`)
+ - webview_flutter_wkwebview (from `.symlinks/plugins/webview_flutter_wkwebview/darwin`)
SPEC REPOS:
trunk:
@@ -156,50 +163,56 @@ EXTERNAL SOURCES:
:path: ".symlinks/plugins/path_provider_foundation/darwin"
permission_handler_apple:
:path: ".symlinks/plugins/permission_handler_apple/ios"
+ saver_gallery:
+ :path: ".symlinks/plugins/saver_gallery/ios"
sentry_flutter:
:path: ".symlinks/plugins/sentry_flutter/ios"
share_plus:
:path: ".symlinks/plugins/share_plus/ios"
shared_preferences_foundation:
:path: ".symlinks/plugins/shared_preferences_foundation/darwin"
- sqflite:
- :path: ".symlinks/plugins/sqflite/darwin"
+ sqflite_darwin:
+ :path: ".symlinks/plugins/sqflite_darwin/darwin"
super_native_extensions:
:path: ".symlinks/plugins/super_native_extensions/ios"
url_launcher_ios:
:path: ".symlinks/plugins/url_launcher_ios/ios"
+ webview_flutter_wkwebview:
+ :path: ".symlinks/plugins/webview_flutter_wkwebview/darwin"
SPEC CHECKSUMS:
- app_links: e70ca16b4b0f88253b3b3660200d4a10b4ea9795
- appflowy_backend: 144c20d8bfb298c4e10fa3fa6701a9f41bf98b88
- connectivity_plus: bf0076dd84a130856aa636df1c71ccaff908fa1d
- device_info_plus: 97af1d7e84681a90d0693e63169a5d50e0839a0d
+ app_links: 3da4c36b46cac3bf24eb897f1a6ce80bda109874
+ appflowy_backend: 78f6a053f756e6bc29bcc5a2106cbe77b756e97a
+ connectivity_plus: 481668c94744c30c53b8895afb39159d1e619bdf
+ device_info_plus: 71ffc6ab7634ade6267c7a93088ed7e4f74e5896
DKImagePickerController: b512c28220a2b8ac7419f21c491fc8534b7601ac
DKPhotoGallery: fdfad5125a9fdda9cc57df834d49df790dbb4179
- file_picker: 09aa5ec1ab24135ccd7a1621c46c84134bfd6655
- flowy_infra_ui: 0455e1fa8c51885aa1437848e361e99419f34ebc
+ file_picker: 9b3292d7c8bc68c8a7bf8eb78f730e49c8efc517
+ flowy_infra_ui: 931b73a18b54a392ab6152eebe29a63a30751f53
Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7
- fluttertoast: e9a18c7be5413da53898f660530c56f35edfba9c
- image_picker_ios: c560581cceedb403a6ff17f2f816d7fea1421fc1
- integration_test: ce0a3ffa1de96d1a89ca0ac26fca7ea18a749ef4
- irondash_engine_context: 3458bf979b90d616ffb8ae03a150bafe2e860cc9
- keyboard_height_plugin: 43fa8bba20fd5c4fdeed5076466b8b9d43cc6b86
- open_filex: 6e26e659846ec990262224a12ef1c528bb4edbe4
- package_info_plus: 58f0028419748fad15bf008b270aaa8e54380b1c
- path_provider_foundation: 2b6b4c569c0fb62ec74538f866245ac84301af46
- permission_handler_apple: 9878588469a2b0d0fc1e048d9f43605f92e6cec2
+ fluttertoast: 76fea30fcf04176325f6864c87306927bd7d2038
+ image_picker_ios: 7fe1ff8e34c1790d6fff70a32484959f563a928a
+ integration_test: 4a889634ef21a45d28d50d622cf412dc6d9f586e
+ irondash_engine_context: 8e58ca8e0212ee9d1c7dc6a42121849986c88486
+ keyboard_height_plugin: ef70a8181b29f27670e9e2450814ca6b6dc05b05
+ open_filex: 432f3cd11432da3e39f47fcc0df2b1603854eff1
+ package_info_plus: af8e2ca6888548050f16fa2f1938db7b5a5df499
+ path_provider_foundation: 080d55be775b7414fd5a5ef3ac137b97b097e564
+ permission_handler_apple: 4ed2196e43d0651e8ff7ca3483a069d469701f2d
ReachabilitySwift: 985039c6f7b23a1da463388634119492ff86c825
+ saver_gallery: af2d0c762dafda254e0ad025ef0dabd6506cd490
SDWebImage: b9a731e1d6307f44ca703b3976d18c24ca561e84
Sentry: 1fe34e9c2cbba1e347623610d26db121dcb569f1
- sentry_flutter: a39c2a2d67d5e5b9cb0b94a4985c76dd5b3fc737
- share_plus: 8875f4f2500512ea181eef553c3e27dba5135aad
- shared_preferences_foundation: fcdcbc04712aee1108ac7fda236f363274528f78
- sqflite: 673a0e54cc04b7d6dba8d24fb8095b31c3a99eec
- super_native_extensions: 4916b3c627a9c7fffdc48a23a9eca0b1ac228fa7
+ sentry_flutter: e24b397f9a61fa5bbefd8279c3b2242ca86faa90
+ share_plus: 50da8cb520a8f0f65671c6c6a99b3617ed10a58a
+ shared_preferences_foundation: 9e1978ff2562383bd5676f64ec4e9aa8fa06a6f7
+ sqflite_darwin: 20b2a3a3b70e43edae938624ce550a3cbf66a3d0
+ super_native_extensions: b763c02dc3a8fd078389f410bf15149179020cb4
SwiftyGif: 6c3eafd0ce693cad58bb63d2b2fb9bacb8552780
Toast: 91b396c56ee72a5790816f40d3a94dd357abc196
- url_launcher_ios: 5334b05cef931de560670eeae103fd3e431ac3fe
+ url_launcher_ios: 694010445543906933d732453a59da0a173ae33d
+ webview_flutter_wkwebview: 44d4dee7d7056d5ad185d25b38404436d56c547c
PODFILE CHECKSUM: d0d9b4ff572d8695c38eb3f9b490f55cdfc57eca
-COCOAPODS: 1.15.2
+COCOAPODS: 1.16.2
diff --git a/frontend/appflowy_flutter/ios/Runner/AppDelegate.swift b/frontend/appflowy_flutter/ios/Runner/AppDelegate.swift
index 70693e4a8c..b636303481 100644
--- a/frontend/appflowy_flutter/ios/Runner/AppDelegate.swift
+++ b/frontend/appflowy_flutter/ios/Runner/AppDelegate.swift
@@ -1,7 +1,7 @@
import UIKit
import Flutter
-@UIApplicationMain
+@main
@objc class AppDelegate: FlutterAppDelegate {
override func application(
_ application: UIApplication,
diff --git a/frontend/appflowy_flutter/ios/Runner/Info.plist b/frontend/appflowy_flutter/ios/Runner/Info.plist
index 0c8c1eff43..5d6a52bd2e 100644
--- a/frontend/appflowy_flutter/ios/Runner/Info.plist
+++ b/frontend/appflowy_flutter/ios/Runner/Info.plist
@@ -1,73 +1,78 @@
-
- CADisableMinimumFrameDurationOnPhone
-
- CFBundleDevelopmentRegion
- $(DEVELOPMENT_LANGUAGE)
- CFBundleExecutable
- $(EXECUTABLE_NAME)
- CFBundleIdentifier
- $(PRODUCT_BUNDLE_IDENTIFIER)
- CFBundleInfoDictionaryVersion
- 6.0
- CFBundleLocalizations
-
- en
-
- CFBundleName
- AppFlowy
- CFBundlePackageType
- APPL
- CFBundleShortVersionString
- $(FLUTTER_BUILD_NAME)
- CFBundleSignature
- ????
- CFBundleURLTypes
-
-
- CFBundleURLName
-
- CFBundleURLSchemes
-
- appflowy-flutter
-
-
-
- CFBundleVersion
- $(FLUTTER_BUILD_NUMBER)
- FLTEnableImpeller
-
- LSRequiresIPhoneOS
-
- NSAppTransportSecurity
- NSAllowsArbitraryLoads
-
+ CADisableMinimumFrameDurationOnPhone
+
+ CFBundleDevelopmentRegion
+ $(DEVELOPMENT_LANGUAGE)
+ CFBundleExecutable
+ $(EXECUTABLE_NAME)
+ CFBundleIdentifier
+ $(PRODUCT_BUNDLE_IDENTIFIER)
+ CFBundleInfoDictionaryVersion
+ 6.0
+ CFBundleLocalizations
+
+ en
+
+ CFBundleName
+ AppFlowy
+ CFBundlePackageType
+ APPL
+ CFBundleShortVersionString
+ $(FLUTTER_BUILD_NAME)
+ CFBundleSignature
+ ????
+ CFBundleURLTypes
+
+
+ CFBundleURLName
+
+ CFBundleURLSchemes
+
+ appflowy-flutter
+
+
+
+ CFBundleVersion
+ $(FLUTTER_BUILD_NUMBER)
+ FLTEnableImpeller
+
+ LSRequiresIPhoneOS
+
+ NSAppTransportSecurity
+
+ NSAllowsArbitraryLoads
+
+
+ NSPhotoLibraryUsageDescription
+ AppFlowy needs access to your photos to let you add images to your documents
+ NSPhotoLibraryAddUsageDescription
+ AppFlowy needs access to your photos to let you add images to your photo library
+ UIApplicationSupportsIndirectInputEvents
+
+ NSCameraUsageDescription
+ AppFlowy needs access to your camera to let you add images to your documents from
+ camera
+ UILaunchStoryboardName
+ LaunchScreen
+ UIMainStoryboardFile
+ Main
+ UISupportedInterfaceOrientations
+
+ UIInterfaceOrientationPortrait
+
+ UISupportedInterfaceOrientations~ipad
+
+ UIInterfaceOrientationLandscapeLeft
+ UIInterfaceOrientationLandscapeRight
+ UIInterfaceOrientationPortrait
+ UIInterfaceOrientationPortraitUpsideDown
+
+ UISupportsDocumentBrowser
+
+ UIViewControllerBasedStatusBarAppearance
+
- NSPhotoLibraryUsageDescription
- AppFlowy needs access to your photos to let you add images to your documents
- UIApplicationSupportsIndirectInputEvents
-
- UILaunchStoryboardName
- LaunchScreen
- UIMainStoryboardFile
- Main
- UISupportedInterfaceOrientations
-
- UIInterfaceOrientationPortrait
-
- UISupportedInterfaceOrientations~ipad
-
- UIInterfaceOrientationLandscapeLeft
- UIInterfaceOrientationLandscapeRight
- UIInterfaceOrientationPortrait
- UIInterfaceOrientationPortraitUpsideDown
-
- UIViewControllerBasedStatusBarAppearance
-
- UISupportsDocumentBrowser
-
-
-
+
\ No newline at end of file
diff --git a/frontend/appflowy_flutter/ios/Runner/Runner.entitlements b/frontend/appflowy_flutter/ios/Runner/Runner.entitlements
index 80b5221de7..e3bc137465 100644
--- a/frontend/appflowy_flutter/ios/Runner/Runner.entitlements
+++ b/frontend/appflowy_flutter/ios/Runner/Runner.entitlements
@@ -8,5 +8,12 @@
Default
+ com.apple.developer.associated-domains
+
+ applinks:appflowy.com
+ applinks:appflowy.io
+ applinks:test.appflowy.com
+ applinks:test.appflowy.io
+
diff --git a/frontend/appflowy_flutter/lib/ai/ai.dart b/frontend/appflowy_flutter/lib/ai/ai.dart
new file mode 100644
index 0000000000..9bfeeb4e00
--- /dev/null
+++ b/frontend/appflowy_flutter/lib/ai/ai.dart
@@ -0,0 +1,19 @@
+export 'service/ai_entities.dart';
+export 'service/ai_prompt_input_bloc.dart';
+export 'service/appflowy_ai_service.dart';
+export 'service/error.dart';
+export 'service/ai_model_state_notifier.dart';
+export 'service/select_model_bloc.dart';
+export 'widgets/loading_indicator.dart';
+export 'widgets/prompt_input/action_buttons.dart';
+export 'widgets/prompt_input/desktop_prompt_text_field.dart';
+export 'widgets/prompt_input/file_attachment_list.dart';
+export 'widgets/prompt_input/layout_define.dart';
+export 'widgets/prompt_input/mention_page_bottom_sheet.dart';
+export 'widgets/prompt_input/mention_page_menu.dart';
+export 'widgets/prompt_input/mentioned_page_text_span.dart';
+export 'widgets/prompt_input/predefined_format_buttons.dart';
+export 'widgets/prompt_input/select_sources_bottom_sheet.dart';
+export 'widgets/prompt_input/select_sources_menu.dart';
+export 'widgets/prompt_input/select_model_menu.dart';
+export 'widgets/prompt_input/send_button.dart';
diff --git a/frontend/appflowy_flutter/lib/ai/service/ai_entities.dart b/frontend/appflowy_flutter/lib/ai/service/ai_entities.dart
new file mode 100644
index 0000000000..b08fadb7f8
--- /dev/null
+++ b/frontend/appflowy_flutter/lib/ai/service/ai_entities.dart
@@ -0,0 +1,107 @@
+import 'package:appflowy/generated/flowy_svgs.g.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy_backend/protobuf/flowy-ai/protobuf.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:equatable/equatable.dart';
+
+class AIStreamEventPrefix {
+ static const data = 'data:';
+ static const error = 'error:';
+ static const metadata = 'metadata:';
+ static const start = 'start:';
+ static const finish = 'finish:';
+ static const comment = 'comment:';
+ static const aiResponseLimit = 'AI_RESPONSE_LIMIT';
+ static const aiImageResponseLimit = 'AI_IMAGE_RESPONSE_LIMIT';
+ static const aiMaxRequired = 'AI_MAX_REQUIRED:';
+ static const localAINotReady = 'LOCAL_AI_NOT_READY';
+ static const localAIDisabled = 'LOCAL_AI_DISABLED';
+}
+
+enum AiType {
+ cloud,
+ local;
+
+ bool get isCloud => this == cloud;
+ bool get isLocal => this == local;
+}
+
+class PredefinedFormat extends Equatable {
+ const PredefinedFormat({
+ required this.imageFormat,
+ required this.textFormat,
+ });
+
+ final ImageFormat imageFormat;
+ final TextFormat? textFormat;
+
+ PredefinedFormatPB toPB() {
+ return PredefinedFormatPB(
+ imageFormat: switch (imageFormat) {
+ ImageFormat.text => ResponseImageFormatPB.TextOnly,
+ ImageFormat.image => ResponseImageFormatPB.ImageOnly,
+ ImageFormat.textAndImage => ResponseImageFormatPB.TextAndImage,
+ },
+ textFormat: switch (textFormat) {
+ TextFormat.paragraph => ResponseTextFormatPB.Paragraph,
+ TextFormat.bulletList => ResponseTextFormatPB.BulletedList,
+ TextFormat.numberedList => ResponseTextFormatPB.NumberedList,
+ TextFormat.table => ResponseTextFormatPB.Table,
+ _ => null,
+ },
+ );
+ }
+
+ @override
+ List