diff --git a/.github/actions/flutter_build/action.yml b/.github/actions/flutter_build/action.yml
index 66bfce44a5..bfcb501327 100644
--- a/.github/actions/flutter_build/action.yml
+++ b/.github/actions/flutter_build/action.yml
@@ -58,19 +58,24 @@ runs:
- name: Install prerequisites
working-directory: frontend
- run: |
- 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 keybinder-3.0 libnotify-dev libmpv-dev mpv
- 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
+ run: |
+ case $RUNNER_OS in
+ Linux)
+ 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 keybinder-3.0 libnotify-dev
+ ;;
+ Windows)
+ vcpkg integrate install
+ vcpkg update
+ ;;
+ macOS)
+ # No additional prerequisites needed for macOS
+ ;;
+ esac
+ cargo make appflowy-flutter-deps-tools
- name: Build AppFlowy
working-directory: frontend
@@ -94,4 +99,4 @@ runs:
- uses: actions/upload-artifact@v4
with:
name: ${{ github.run_id }}-${{ matrix.os }}
- path: appflowy_flutter.tar.gz
\ No newline at end of file
+ path: appflowy_flutter.tar.gz
diff --git a/.github/actions/flutter_integration_test/action.yml b/.github/actions/flutter_integration_test/action.yml
index 63066e0f38..e0fa508ade 100644
--- a/.github/actions/flutter_integration_test/action.yml
+++ b/.github/actions/flutter_integration_test/action.yml
@@ -52,7 +52,7 @@ runs:
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 keybinder-3.0 libnotify-dev network-manager libmpv-dev mpv
+ sudo apt-get install -y dart curl build-essential libssl-dev clang cmake ninja-build pkg-config libgtk-3-dev keybinder-3.0 libnotify-dev network-manager
shell: bash
- name: Enable Flutter Desktop
@@ -75,4 +75,4 @@ runs:
sudo Xvfb -ac :99 -screen 0 1280x1024x24 > /dev/null 2>&1 &
sudo apt-get install network-manager
flutter test ${{ inputs.test_path }} -d Linux --coverage
- shell: bash
\ No newline at end of file
+ shell: bash
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 e91a82af16..51e8a2ac28 100644
--- a/.github/workflows/docker_ci.yml
+++ b/.github/workflows/docker_ci.yml
@@ -2,18 +2,10 @@ name: Docker-CI
on:
push:
- branches:
- - main
- - release/*
- paths:
- - frontend/**
+ branches: [ "main", "release/*" ]
pull_request:
- branches:
- - main
- - release/*
- paths:
- - frontend/**
- types: [ opened, synchronize, reopened, unlocked, ready_for_review ]
+ branches: [ "main", "release/*" ]
+ workflow_dispatch:
concurrency:
group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
@@ -27,25 +19,29 @@ jobs:
- name: Checkout source code
uses: actions/checkout@v4
- - name: Set up Docker Compose
- run: |
- docker-compose --version || {
- echo "Docker Compose not found, installing..."
- sudo curl -L "https://github.com/docker/compose/releases/latest/download/docker-compose-$(uname -s)-$(uname -m)" -o /usr/local/bin/docker-compose
- sudo chmod +x /usr/local/bin/docker-compose
- docker-compose --version
- }
+ - name: Set up Docker Buildx
+ uses: docker/setup-buildx-action@v3
+
+ # cache the docker layers
+ # don't cache anything temporarly, because it always triggers "no space left on device" error
+ # - name: Cache Docker layers
+ # uses: actions/cache@v3
+ # with:
+ # path: /tmp/.buildx-cache
+ # key: ${{ runner.os }}-buildx-${{ github.sha }}
+ # restore-keys: |
+ # ${{ runner.os }}-buildx-
- name: Build the app
- shell: bash
- run: |
- set -eu -o pipefail
- cd frontend/scripts/docker-buildfiles
- docker-compose build --no-cache --progress=plain \
- | while read line; do \
- if [[ "$line" =~ ^Step[[:space:]] ]]; then \
- echo "$(date -u '+%H:%M:%S') | $line"; \
- else \
- echo "$line"; \
- fi; \
- done
+ uses: docker/build-push-action@v5
+ with:
+ context: .
+ file: ./frontend/scripts/docker-buildfiles/Dockerfile
+ push: false
+ # cache-from: type=local,src=/tmp/.buildx-cache
+ # cache-to: type=local,dest=/tmp/.buildx-cache-new,mode=max
+
+ # - name: Move cache
+ # run: |
+ # rm -rf /tmp/.buildx-cache
+ # mv /tmp/.buildx-cache-new /tmp/.buildx-cache
diff --git a/.github/workflows/flutter_ci.yaml b/.github/workflows/flutter_ci.yaml
index 9b9725a09f..1fc1b0e052 100644
--- a/.github/workflows/flutter_ci.yaml
+++ b/.github/workflows/flutter_ci.yaml
@@ -25,9 +25,10 @@ on:
env:
CARGO_TERM_COLOR: always
- FLUTTER_VERSION: "3.22.0"
- RUST_TOOLCHAIN: "1.77.2"
- CARGO_MAKE_VERSION: "0.36.6"
+ 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 }}
@@ -39,7 +40,7 @@ jobs:
strategy:
fail-fast: true
matrix:
- os: [ ubuntu-latest ]
+ os: [ubuntu-latest]
include:
- os: ubuntu-latest
flutter_profile: development-linux-x86_64
@@ -73,7 +74,7 @@ jobs:
strategy:
fail-fast: true
matrix:
- os: [ windows-latest ]
+ os: [windows-latest]
include:
- os: windows-latest
flutter_profile: development-windows-x86
@@ -100,7 +101,7 @@ jobs:
strategy:
fail-fast: true
matrix:
- os: [ macos-latest ]
+ os: [macos-latest]
include:
- os: macos-latest
flutter_profile: development-mac-x86_64
@@ -122,12 +123,12 @@ jobs:
flutter_profile: ${{ matrix.flutter_profile }}
unit_test:
- needs: [ prepare-linux ]
+ needs: [prepare-linux]
if: github.event.pull_request.draft != true
strategy:
fail-fast: false
matrix:
- os: [ ubuntu-latest ]
+ os: [ubuntu-latest]
include:
- os: ubuntu-latest
flutter_profile: development-linux-x86_64
@@ -173,7 +174,7 @@ jobs:
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 keybinder-3.0 libnotify-dev libmpv-dev mpv
+ sudo apt-get install -y dart curl build-essential libssl-dev clang cmake ninja-build pkg-config libgtk-3-dev keybinder-3.0 libnotify-dev
fi
shell: bash
@@ -216,11 +217,11 @@ jobs:
shell: bash
cloud_integration_test:
- needs: [ prepare-linux ]
+ needs: [prepare-linux]
strategy:
fail-fast: false
matrix:
- os: [ ubuntu-latest ]
+ os: [ubuntu-latest]
include:
- os: ubuntu-latest
flutter_profile: development-linux-x86_64
@@ -241,12 +242,15 @@ jobs:
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: Run Docker-Compose
working-directory: AppFlowy-Cloud
env:
- APPFLOWY_CLOUD_VERSION: 0.6.4-amd64
+ 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
@@ -258,11 +262,26 @@ jobs:
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..."
+ 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
+
+ # 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
@@ -289,7 +308,7 @@ jobs:
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 keybinder-3.0 libnotify-dev libmpv-dev mpv
+ sudo apt-get install -y dart curl build-essential libssl-dev clang cmake ninja-build pkg-config libgtk-3-dev keybinder-3.0 libnotify-dev
shell: bash
- name: Enable Flutter Desktop
@@ -317,96 +336,30 @@ jobs:
sudo Xvfb -ac :99 -screen 0 1280x1024x24 > /dev/null 2>&1 &
sudo apt-get install network-manager
docker ps -a
- flutter test integration_test/cloud/cloud_runner.dart -d Linux --coverage
+ flutter test integration_test/desktop/cloud/cloud_runner.dart -d Linux --coverage
shell: bash
- # split the integration tests into different machines to minimize the time
- integration_test_1:
- needs: [ prepare-linux ]
+ integration_test:
+ needs: [prepare-linux]
if: github.event.pull_request.draft != true
strategy:
fail-fast: false
matrix:
- os: [ ubuntu-latest ]
+ os: [ubuntu-latest]
+ test_number: [1, 2, 3, 4, 5, 6, 7, 8, 9]
include:
- os: ubuntu-latest
- target: 'x86_64-unknown-linux-gnu'
+ target: "x86_64-unknown-linux-gnu"
runs-on: ${{ matrix.os }}
steps:
- name: Checkout source code
uses: actions/checkout@v4
- - name: Install video dependency
- run: |
- sudo apt-get update
- sudo apt-get -y install libmpv-dev mpv
- shell: bash
-
- - name: Flutter Integration Test 1
+ - name: Flutter Integration Test ${{ matrix.test_number }}
uses: ./.github/actions/flutter_integration_test
with:
- test_path: integration_test/desktop_runner_1.dart
+ test_path: integration_test/desktop_runner_${{ matrix.test_number }}.dart
flutter_version: ${{ env.FLUTTER_VERSION }}
rust_toolchain: ${{ env.RUST_TOOLCHAIN }}
cargo_make_version: ${{ env.CARGO_MAKE_VERSION }}
rust_target: ${{ matrix.target }}
-
- integration_test_2:
- needs: [ prepare-linux ]
- if: github.event.pull_request.draft != true
- strategy:
- fail-fast: false
- matrix:
- os: [ ubuntu-latest ]
- include:
- - os: ubuntu-latest
- target: 'x86_64-unknown-linux-gnu'
- runs-on: ${{ matrix.os }}
- steps:
- - name: Checkout source code
- uses: actions/checkout@v4
-
- - name: Install video dependency
- run: |
- sudo apt-get update
- sudo apt-get -y install libmpv-dev mpv
- shell: bash
-
- - name: Flutter Integration Test 2
- uses: ./.github/actions/flutter_integration_test
- with:
- test_path: integration_test/desktop_runner_2.dart
- flutter_version: ${{ env.FLUTTER_VERSION }}
- rust_toolchain: ${{ env.RUST_TOOLCHAIN }}
- cargo_make_version: ${{ env.CARGO_MAKE_VERSION }}
- rust_target: ${{ matrix.target }}
-
- integration_test_3:
- needs: [ prepare-linux ]
- if: github.event.pull_request.draft != true
- strategy:
- fail-fast: false
- matrix:
- os: [ ubuntu-latest ]
- include:
- - os: ubuntu-latest
- target: 'x86_64-unknown-linux-gnu'
- runs-on: ${{ matrix.os }}
- steps:
- - name: Checkout source code
- uses: actions/checkout@v4
-
- - name: Install video dependency
- run: |
- sudo apt-get update
- sudo apt-get -y install libmpv-dev mpv
- shell: bash
-
- - name: Flutter Integration Test 3
- uses: ./.github/actions/flutter_integration_test
- with:
- test_path: integration_test/desktop_runner_3.dart
- flutter_version: ${{ env.FLUTTER_VERSION }}
- rust_toolchain: ${{ env.RUST_TOOLCHAIN }}
- cargo_make_version: ${{ env.CARGO_MAKE_VERSION }}
- rust_target: ${{ matrix.target }}
\ No newline at end of file
diff --git a/.github/workflows/ios_ci.yaml b/.github/workflows/ios_ci.yaml
index e1be95894d..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,12 +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"
+ FLUTTER_VERSION: "3.27.4"
+ RUST_TOOLCHAIN: "1.81.0"
concurrency:
group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
@@ -48,13 +46,13 @@ jobs:
model: "iPhone 15"
shutdown_after_job: false
- build-macos:
+ integration-tests:
if: github.event.pull_request.head.repo.full_name != github.repository
- runs-on: macos-13
+ runs-on: macos-latest
steps:
- name: Checkout source code
- uses: actions/checkout@v2
+ uses: actions/checkout@v4
- name: Install Rust toolchain
uses: actions-rs/toolchain@v1
@@ -85,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
@@ -102,16 +100,20 @@ jobs:
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
- # enable it again if the 12 mins timeout is fixed
- # - 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 63d0432061..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
@@ -232,10 +232,10 @@ jobs:
matrix:
job:
- {
- targets: "aarch64-apple-darwin,x86_64-apple-darwin",
- os: macos-latest,
- extra-build-args: "",
- }
+ targets: "aarch64-apple-darwin,x86_64-apple-darwin",
+ os: macos-latest,
+ extra-build-args: "",
+ }
steps:
- name: Checkout source code
uses: actions/checkout@v4
@@ -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
@@ -336,12 +336,12 @@ jobs:
matrix:
job:
- {
- arch: x86_64,
- target: x86_64-unknown-linux-gnu,
- os: ubuntu-20.04,
- extra-build-args: "",
- flutter_profile: production-linux-x86_64,
- }
+ arch: x86_64,
+ target: x86_64-unknown-linux-gnu,
+ os: ubuntu-22.04,
+ extra-build-args: "",
+ flutter_profile: production-linux-x86_64,
+ }
steps:
- name: Checkout source code
uses: actions/checkout@v4
@@ -368,10 +368,10 @@ jobs:
sudo apt-get update
sudo apt-get install -y build-essential libsqlite3-dev libssl-dev clang cmake ninja-build pkg-config libgtk-3-dev
sudo apt-get install keybinder-3.0
- sudo apt-get install -y alien libnotify-dev libmpv-dev mpv
+ 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 566aef3b7b..36c2e82064 100644
--- a/.github/workflows/rust_ci.yaml
+++ b/.github/workflows/rust_ci.yaml
@@ -15,82 +15,21 @@ on:
- "main"
- "develop"
- "release/*"
- paths:
- - "frontend/rust-lib/**"
env:
CARGO_TERM_COLOR: always
- 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: 0.6.4-amd64
- 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
- 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
+ uses: szenius/set-timezone@v2.0
+ with:
+ timezoneLinux: "US/Pacific"
+
- name: Maximize build space
run: |
sudo rm -rf /usr/share/dotnet
@@ -127,33 +66,37 @@ 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|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: 0.6.4-amd64
+ 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
+ # Remove all containers if any exist
+ if [ "$(docker ps -aq)" ]; then
+ docker rm -f $(docker ps -aq)
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
- else
- echo "AppFlowy-Cloud is running with the correct version."
- fi
+ echo "No containers to remove."
fi
+ # 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
+
- name: Run rust-lib tests
working-directory: frontend/rust-lib
env:
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 a1567a2501..a5e7e268a5 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,4 +1,249 @@
# 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
+- Support toggling headings
+- Create a subpage by typing in the document
+- Turn selected blocks into a subpage
+- Add a manual date picker for the Date property
+
+### Bug Fixes
+- Fixed an issue where the workspace owner was unable to delete spaces created by others
+- Fixed cursor height inconsistencies with text height
+- Fixed editing issues in Kanban cards
+- Fixed an issue preventing images or files from being dropped into empty paragraphs
+
+## Version 0.7.2 - 22/10/2024
+### New Features
+- Copy link to block
+- Support turn into in document
+- Enable sharing links and publishing pages on mobile
+- Enable drag and drop in row documents
+- Right-click on page in sidebar to open more actions
+- Create new subpage in document using `+` character
+- Allow reordering checklist item
+
+### Bug Fixes
+- Fixed issue with inability to cancel inline code format in French IME
+- Fixed delete with Shift or Ctrl shortcuts not working in documents
+- Fixed the issues with incorrect time zone being used in filters.
+
+## Version 0.7.1 - 07/10/2024
+### New Features
+- Copy link to share and open it in a browser
+- Enable the ability to edit the page title within the body of the document
+- Filter by last modified, created at, or a date range
+- Allow customization of database property icons
+- Support CTRL/CMD+X to delete the current line when the selection is collapsed in the document
+- Support window tiling on macOS
+- Add filters to grid views on mobile
+- Create and manage workspaces on mobile
+- Automatically convert property types for imported CSV files
+
+### Bug Fixes
+- Fixed calculations with filters applied
+- Fixed issues with importing data folders into a cloud account
+- Fixed French IME backtick issues
+- Fixed selection gesture bugs on mobile
+
## Version 0.7.0 - 19/09/2024
### New Features
- Support reordering blocks in document with drag and drop
@@ -42,7 +287,7 @@
- Fixed the inability to edit group names on Kanban boards
- Made error codes more user-friendly
- Added leading zeros to day and month in date format
-
+
## Version 0.6.8 - 22/08/2024
### New Features
- Enabled viewing data inside a database record on mobile.
@@ -872,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 92930aad0e..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.
-## Acknowledgements
+## 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 7fc72f78dd..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.0"
+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 c8066c92a5..4579b2d8c5 100644
--- a/frontend/appflowy_flutter/analysis_options.yaml
+++ b/frontend/appflowy_flutter/analysis_options.yaml
@@ -1,32 +1,12 @@
-# This file configures the analyzer, which statically analyzes Dart code to
-# check for errors, warnings, and lints.
-#
-# The issues identified by the analyzer are surfaced in the UI of Dart-enabled
-# IDEs (https://dart.dev/tools#ides-and-editors). The analyzer can also be
-# invoked from the command line by running `flutter analyze`.
-
-# The following line activates a set of recommended lints for Flutter apps,
-# packages, and plugins designed to encourage good coding practices.
-
include: package:flutter_lints/flutter.yaml
analyzer:
exclude:
- "**/*.g.dart"
- "**/*.freezed.dart"
+ - "packages/**/*.dart"
linter:
- # The lint rules applied to this project can be customized in the
- # section below to disable rules from the `package:flutter_lints/flutter.yaml`
- # included above or to enable additional rules. A list of all available lints
- # and their documentation is published at
- # https://dart-lang.github.io/linter/lints/index.html.
- #
- # Instead of disabling a lint rule for the entire project in the
- # section below, it can also be suppressed for a single line of code
- # or a specific dart file by using the `// ignore: name_of_lint` and
- # `// ignore_for_file: name_of_lint` syntax on the line or in the file
- # producing the lint.
rules:
- require_trailing_commas
@@ -51,8 +31,5 @@ linter:
- sort_constructors_first
- unawaited_futures
-# Additional information about this file can be found at
-# https://dart.dev/guides/language/analysis-options
-
errors:
invalid_annotation_target: ignore
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/cloud/anon_user_continue_test.dart b/frontend/appflowy_flutter/integration_test/cloud/anon_user_continue_test.dart
deleted file mode 100644
index 0364eaab57..0000000000
--- a/frontend/appflowy_flutter/integration_test/cloud/anon_user_continue_test.dart
+++ /dev/null
@@ -1,80 +0,0 @@
-// 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 '../desktop/board/board_hide_groups_test.dart';
-import '../shared/dir.dart';
-import '../shared/mock/mock_file_picker.dart';
-import '../shared/util.dart';
-
-void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
-
- group('appflowy cloud', () {
- testWidgets('anon user and then sign in', (tester) async {
- await tester.initializeAppFlowy(
- cloudType: AuthenticatorType.appflowyCloudSelfHost,
- );
-
- await tester.tapContinousAnotherWay();
- await tester.tapAnonymousSignInButton();
- await tester.expectToSeeHomePageWithGetStartedPage();
-
- // rename the name of the anon user
- await tester.openSettings();
- await tester.openSettingsPage(SettingsPage.account);
- await tester.pumpAndSettle();
-
- 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
- await tester.tapGoogleLoginInButton();
-
- // sign out
- await tester.expectToSeeHomePage();
- await tester.openSettings();
- await tester.openSettingsPage(SettingsPage.account);
-
- // Scroll to sign-out
- await tester.scrollUntilVisible(
- find.byType(AccountSignInOutButton),
- 100,
- scrollable: find.findSettingsScrollable(),
- );
-
- await tester.logout();
- await tester.pumpAndSettle();
- });
- });
-}
diff --git a/frontend/appflowy_flutter/integration_test/cloud/cloud_runner.dart b/frontend/appflowy_flutter/integration_test/cloud/cloud_runner.dart
deleted file mode 100644
index 209889e577..0000000000
--- a/frontend/appflowy_flutter/integration_test/cloud/cloud_runner.dart
+++ /dev/null
@@ -1,29 +0,0 @@
-import 'anon_user_continue_test.dart' as anon_user_continue_test;
-import 'appflowy_cloud_auth_test.dart' as appflowy_cloud_auth_test;
-import 'document/document_drag_block_test.dart' as document_drag_block_test;
-import 'empty_test.dart' as preset_af_cloud_env_test;
-import 'sidebar/sidebar_move_page_test.dart' as sidebar_move_page_test;
-import 'user_setting_sync_test.dart' as user_sync_test;
-import 'workspace/change_name_and_icon_test.dart'
- as change_workspace_name_and_icon_test;
-import 'workspace/collaborative_workspace_test.dart'
- as collaboration_workspace_test;
-import 'workspace/workspace_settings_test.dart' as workspace_settings_test;
-
-Future main() async {
- preset_af_cloud_env_test.main();
- appflowy_cloud_auth_test.main();
- user_sync_test.main();
- anon_user_continue_test.main();
-
- // workspace
- collaboration_workspace_test.main();
- change_workspace_name_and_icon_test.main();
- workspace_settings_test.main();
-
- // document
- document_drag_block_test.main();
-
- // sidebar
- sidebar_move_page_test.main();
-}
diff --git a/frontend/appflowy_flutter/integration_test/cloud/document/document_delete_block_test.dart b/frontend/appflowy_flutter/integration_test/cloud/document/document_delete_block_test.dart
deleted file mode 100644
index 277f878fb5..0000000000
--- a/frontend/appflowy_flutter/integration_test/cloud/document/document_delete_block_test.dart
+++ /dev/null
@@ -1,60 +0,0 @@
-import 'package:appflowy/env/cloud_env.dart';
-import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/actions/drag_to_reorder/draggable_option_button.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/constants.dart';
-import '../../shared/util.dart';
-
-void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
-
- group('document delete block: ', () {
- testWidgets('hover on the block and delete it', (tester) async {
- await tester.initializeAppFlowy(
- cloudType: AuthenticatorType.appflowyCloudSelfHost,
- );
- await tester.tapGoogleLoginInButton();
- await tester.expectToSeeHomePageWithGetStartedPage();
-
- // open getting started page
- await tester.openPage(Constants.gettingStartedPageName);
-
- // before delete
- final path = [1];
- final beforeDeletedBlock = tester.editor.getNodeAtPath(path);
-
- // hover on the block and delete it
- final optionButton = find.byWidgetPredicate(
- (widget) =>
- widget is DraggableOptionButton &&
- widget.blockComponentContext.node.path.equals(path),
- );
-
- await tester.hoverOnWidget(
- optionButton,
- onHover: () async {
- // click the delete button
- await tester.tapButton(optionButton);
- },
- );
- await tester.pumpAndSettle(Durations.short1);
-
- // click the delete button
- final deleteButton =
- find.findTextInFlowyText(LocaleKeys.button_delete.tr());
- await tester.tapButton(deleteButton);
-
- // wait for the deletion
- await tester.pumpAndSettle(Durations.short1);
-
- // check if the block is deleted
- final afterDeletedBlock = tester.editor.getNodeAtPath([1]);
- expect(afterDeletedBlock.id, isNot(equals(beforeDeletedBlock.id)));
- });
- });
-}
diff --git a/frontend/appflowy_flutter/integration_test/cloud/document/document_drag_block_test.dart b/frontend/appflowy_flutter/integration_test/cloud/document/document_drag_block_test.dart
deleted file mode 100644
index e8d366060a..0000000000
--- a/frontend/appflowy_flutter/integration_test/cloud/document/document_drag_block_test.dart
+++ /dev/null
@@ -1,67 +0,0 @@
-import 'package:appflowy/env/cloud_env.dart';
-import 'package:flutter/material.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('document drag block: ', () {
- testWidgets('drag block to the top', (tester) async {
- await tester.initializeAppFlowy(
- cloudType: AuthenticatorType.appflowyCloudSelfHost,
- );
- await tester.tapGoogleLoginInButton();
- await tester.expectToSeeHomePageWithGetStartedPage();
-
- // open getting started page
- await tester.openPage(Constants.gettingStartedPageName);
-
- // before move
- final beforeMoveBlock = tester.editor.getNodeAtPath([1]);
-
- // move the desktop guide to the top, above the getting started
- await tester.editor.dragBlock(
- [1],
- const Offset(20, -80),
- );
-
- // wait for the move animation to complete
- await tester.pumpAndSettle(Durations.short1);
-
- // check if the block is moved to the top
- final afterMoveBlock = tester.editor.getNodeAtPath([0]);
- expect(afterMoveBlock.delta, beforeMoveBlock.delta);
- });
-
- testWidgets('drag block to other block\'s child', (tester) async {
- await tester.initializeAppFlowy(
- cloudType: AuthenticatorType.appflowyCloudSelfHost,
- );
- await tester.tapGoogleLoginInButton();
- await tester.expectToSeeHomePageWithGetStartedPage();
-
- // open getting started page
- await tester.openPage(Constants.gettingStartedPageName);
-
- // before move
- final beforeMoveBlock = tester.editor.getNodeAtPath([10]);
-
- // move the checkbox to the child of the block at path [9]
- await tester.editor.dragBlock(
- [10],
- const Offset(80, -30),
- );
-
- // wait for the move animation to complete
- await tester.pumpAndSettle(Durations.short1);
-
- // check if the block is moved to the child of the block at path [9]
- final afterMoveBlock = tester.editor.getNodeAtPath([9, 0]);
- expect(afterMoveBlock.delta, beforeMoveBlock.delta);
- });
- });
-}
diff --git a/frontend/appflowy_flutter/integration_test/cloud/document_sync_test.dart b/frontend/appflowy_flutter/integration_test/cloud/document_sync_test.dart
deleted file mode 100644
index f8fe5a8c9a..0000000000
--- a/frontend/appflowy_flutter/integration_test/cloud/document_sync_test.dart
+++ /dev/null
@@ -1,70 +0,0 @@
-// 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() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- final email = '${uuid()}@appflowy.io';
- const inputContent = 'Hello world, this is a test document';
-
-// The test will create a new document called Sample, and sync it to the server.
-// Then the test will logout the user, and login with the same user. The data will
-// be synced from the server.
- group('appflowy cloud document', () {
- testWidgets('sync local docuemnt to server', (tester) async {
- await tester.initializeAppFlowy(
- cloudType: AuthenticatorType.appflowyCloudSelfHost,
- email: email,
- );
- await tester.tapGoogleLoginInButton();
- await tester.expectToSeeHomePageWithGetStartedPage();
-
- // create a new document called Sample
- await tester.createNewPage();
-
- // focus on the editor
- await tester.editor.tapLineOfEditorAt(0);
- await tester.ime.insertText(inputContent);
- expect(find.text(inputContent, findRichText: true), findsOneWidget);
-
- // 6 seconds for data sync
- await tester.waitForSeconds(6);
-
- await tester.openSettings();
- await tester.openSettingsPage(SettingsPage.account);
- await tester.logout();
- });
-
- testWidgets('sync doc from server', (tester) async {
- await tester.initializeAppFlowy(
- cloudType: AuthenticatorType.appflowyCloudSelfHost,
- email: email,
- );
- await tester.tapGoogleLoginInButton();
- await tester.expectToSeeHomePage();
-
- // the latest document will be opened, so the content must be the inputContent
- await tester.pumpAndSettle();
- expect(find.text(inputContent, findRichText: true), findsOneWidget);
- });
- });
-}
diff --git a/frontend/appflowy_flutter/integration_test/cloud/empty_test.dart b/frontend/appflowy_flutter/integration_test/cloud/empty_test.dart
deleted file mode 100644
index 9f7d3ce9ed..0000000000
--- a/frontend/appflowy_flutter/integration_test/cloud/empty_test.dart
+++ /dev/null
@@ -1,18 +0,0 @@
-import 'package:appflowy/env/cloud_env.dart';
-import 'package:flutter_test/flutter_test.dart';
-import 'package:integration_test/integration_test.dart';
-
-import '../shared/util.dart';
-
-// This test is meaningless, just for preventing the CI from failing.
-void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
-
- group('Empty', () {
- testWidgets('set appflowy cloud', (tester) async {
- await tester.initializeAppFlowy(
- cloudType: AuthenticatorType.appflowyCloudSelfHost,
- );
- });
- });
-}
diff --git a/frontend/appflowy_flutter/integration_test/cloud/sidebar/sidebar_move_page_test.dart b/frontend/appflowy_flutter/integration_test/cloud/sidebar/sidebar_move_page_test.dart
deleted file mode 100644
index 975c653d15..0000000000
--- a/frontend/appflowy_flutter/integration_test/cloud/sidebar/sidebar_move_page_test.dart
+++ /dev/null
@@ -1,121 +0,0 @@
-// 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/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() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
-
- group('sidebar move page: ', () {
- testWidgets('create a new document and move it to Getting started',
- (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,
- );
-
- // click the ... button and move to Getting started
- await tester.hoverOnPageName(
- pageName,
- onHover: () async {
- await tester.tapPageOptionButton();
- await tester.tapButtonWithName(
- LocaleKeys.disclosureAction_moveTo.tr(),
- );
- },
- );
-
- // expect to see two pages
- // one is in the sidebar, the other is in the move to page list
- // 1. Getting started
- // 2. To-dos
- final gettingStarted = find.findTextInFlowyText(
- Constants.gettingStartedPageName,
- );
- final toDos = find.findTextInFlowyText(Constants.toDosPageName);
- await tester.pumpUntilFound(gettingStarted);
- await tester.pumpUntilFound(toDos);
- expect(gettingStarted, findsNWidgets(2));
-
- // skip the length check on Linux temporarily,
- // because it failed in expect check but the previous pumpUntilFound is successful
- if (!UniversalPlatform.isLinux) {
- expect(toDos, findsNWidgets(2));
-
- // hover on the todos page, and will see a forbidden icon
- await tester.hoverOnWidget(
- toDos.last,
- onHover: () async {
- final tooltips = find.byTooltip(
- LocaleKeys.space_cannotMovePageToDatabase.tr(),
- );
- expect(tooltips, findsOneWidget);
- },
- );
- await tester.pumpAndSettle();
- }
-
- // move the current page to Getting started
- await tester.tapButton(
- gettingStarted.last,
- );
-
- await tester.pumpAndSettle();
-
- // after moving, expect to not see the page name in the sidebar
- final page = tester.findPageName(pageName);
- expect(page, findsNothing);
-
- // click to expand the getting started page
- await tester.expandOrCollapsePage(
- pageName: Constants.gettingStartedPageName,
- layout: ViewLayoutPB.Document,
- );
- await tester.pumpAndSettle();
-
- // expect to see the page name in the getting started page
- final pageInGettingStarted = tester.findPageName(
- pageName,
- parentName: Constants.gettingStartedPageName,
- );
- expect(pageInGettingStarted, findsOneWidget);
- });
- });
-}
diff --git a/frontend/appflowy_flutter/integration_test/cloud/supabase_auth_test.dart b/frontend/appflowy_flutter/integration_test/cloud/supabase_auth_test.dart
deleted file mode 100644
index 71cbc11431..0000000000
--- a/frontend/appflowy_flutter/integration_test/cloud/supabase_auth_test.dart
+++ /dev/null
@@ -1,93 +0,0 @@
-// import 'package:appflowy/env/cloud_env.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_supabase_cloud.dart';
-// import 'package:flutter_test/flutter_test.dart';
-// import 'package:integration_test/integration_test.dart';
-
-// import '../shared/util.dart';
-
-// void main() {
-// IntegrationTestWidgetsFlutterBinding.ensureInitialized();
-
-// group('supabase auth', () {
-// testWidgets('sign in with supabase', (tester) async {
-// await tester.initializeAppFlowy(cloudType: AuthenticatorType.supabase);
-// await tester.tapGoogleLoginInButton();
-// await tester.expectToSeeHomePageWithGetStartedPage();
-// });
-
-// testWidgets('sign out with supabase', (tester) async {
-// await tester.initializeAppFlowy(cloudType: AuthenticatorType.supabase);
-// await tester.tapGoogleLoginInButton();
-
-// // Open the setting page and sign out
-// await tester.openSettings();
-// await tester.openSettingsPage(SettingsPage.account);
-// await tester.logout();
-
-// // Go to the sign in page again
-// await tester.pumpAndSettle(const Duration(seconds: 1));
-// tester.expectToSeeGoogleLoginButton();
-// });
-
-// testWidgets('sign in as anonymous', (tester) async {
-// await tester.initializeAppFlowy(cloudType: AuthenticatorType.supabase);
-// await tester.tapSignInAsGuest();
-
-// // should not see the sync setting page when sign in as anonymous
-// await tester.openSettings();
-// await tester.openSettingsPage(SettingsPage.account);
-
-// // Scroll to sign-out
-// await tester.scrollUntilVisible(
-// find.byType(SignInOutButton),
-// 100,
-// scrollable: find.findSettingsScrollable(),
-// );
-// await tester.tapButton(find.byType(SignInOutButton));
-
-// tester.expectToSeeGoogleLoginButton();
-// });
-
-// // testWidgets('enable encryption', (tester) async {
-// // await tester.initializeAppFlowy(cloudType: CloudType.supabase);
-// // await tester.tapGoogleLoginInButton();
-
-// // // Open the setting page and sign out
-// // await tester.openSettings();
-// // await tester.openSettingsPage(SettingsPage.cloud);
-
-// // // the switch should be off by default
-// // tester.assertEnableEncryptSwitchValue(false);
-// // await tester.toggleEnableEncrypt();
-
-// // // the switch should be on after toggling
-// // tester.assertEnableEncryptSwitchValue(true);
-
-// // // the switch can not be toggled back to off
-// // await tester.toggleEnableEncrypt();
-// // tester.assertEnableEncryptSwitchValue(true);
-// // });
-
-// testWidgets('enable sync', (tester) async {
-// await tester.initializeAppFlowy(cloudType: AuthenticatorType.supabase);
-// await tester.tapGoogleLoginInButton();
-
-// // Open the setting page and sign out
-// await tester.openSettings();
-// await tester.openSettingsPage(SettingsPage.cloud);
-
-// // the switch should be on by default
-// tester.assertSupabaseEnableSyncSwitchValue(true);
-// await tester.toggleEnableSync(SupabaseEnableSync);
-
-// // the switch should be off
-// tester.assertSupabaseEnableSyncSwitchValue(false);
-
-// // the switch should be on after toggling
-// await tester.toggleEnableSync(SupabaseEnableSync);
-// tester.assertSupabaseEnableSyncSwitchValue(true);
-// });
-// });
-// }
diff --git a/frontend/appflowy_flutter/integration_test/cloud/user_setting_sync_test.dart b/frontend/appflowy_flutter/integration_test/cloud/user_setting_sync_test.dart
deleted file mode 100644
index 253d533607..0000000000
--- a/frontend/appflowy_flutter/integration_test/cloud/user_setting_sync_test.dart
+++ /dev/null
@@ -1,74 +0,0 @@
-// 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 '../desktop/board/board_hide_groups_test.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() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- final email = '${uuid()}@appflowy.io';
- const name = 'nathan';
-
- group('appflowy cloud setting', () {
- testWidgets('sync user name and icon to server', (tester) async {
- await tester.initializeAppFlowy(
- cloudType: AuthenticatorType.appflowyCloudSelfHost,
- email: email,
- );
- await tester.tapGoogleLoginInButton();
- await tester.expectToSeeHomePageWithGetStartedPage();
-
- await tester.openSettings();
- await tester.openSettingsPage(SettingsPage.account);
-
- await tester.enterUserName(name);
- await tester.pumpAndSettle(const Duration(seconds: 6));
- await tester.logout();
-
- await tester.pumpAndSettle(const Duration(seconds: 2));
- });
- });
- testWidgets('get user icon and name from server', (tester) async {
- await tester.initializeAppFlowy(
- cloudType: AuthenticatorType.appflowyCloudSelfHost,
- email: email,
- );
- await tester.tapGoogleLoginInButton();
- await tester.expectToSeeHomePageWithGetStartedPage();
- await tester.pumpAndSettle();
-
- await tester.openSettings();
- await tester.openSettingsPage(SettingsPage.account);
-
- // Verify name
- final profileSetting =
- tester.widget(find.byType(AccountUserProfile)) as AccountUserProfile;
-
- expect(profileSetting.name, name);
- });
-}
diff --git a/frontend/appflowy_flutter/integration_test/cloud/workspace/collaborative_workspace_test.dart b/frontend/appflowy_flutter/integration_test/cloud/workspace/collaborative_workspace_test.dart
deleted file mode 100644
index c33c4b7424..0000000000
--- a/frontend/appflowy_flutter/integration_test/cloud/workspace/collaborative_workspace_test.dart
+++ /dev/null
@@ -1,97 +0,0 @@
-// 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/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';
-
-void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
-
- 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 {
- // 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);
-
- Finder success;
-
- final Finder items = find.byType(WorkspaceMenuItem);
-
- // delete the newly created workspace
- await tester.openCollaborativeWorkspaceMenu();
- await tester.pumpUntilFound(items);
-
- expect(items, findsNWidgets(2));
- expect(
- tester.widget(items.last).workspace.name,
- name,
- );
-
- final secondWorkspace = find.byType(WorkspaceMenuItem).last;
- await tester.hoverOnWidget(
- secondWorkspace,
- onHover: () async {
- // click the more button
- final moreButton = find.byType(WorkspaceMoreActionList);
- expect(moreButton, findsOneWidget);
- await tester.tapButton(moreButton);
- // click the delete button
- final deleteButton = find.text(LocaleKeys.button_delete.tr());
- expect(deleteButton, findsOneWidget);
- await tester.tapButton(deleteButton);
- // see the delete confirm dialog
- final confirm =
- find.text(LocaleKeys.workspace_deleteWorkspaceHintText.tr());
- expect(confirm, findsOneWidget);
- await tester.tapButton(find.text(LocaleKeys.button_ok.tr()));
- // delete success
- success = find.text(LocaleKeys.workspace_createSuccess.tr());
- await tester.pumpUntilFound(success);
- expect(success, findsOneWidget);
- await tester.pumpUntilNotFound(success);
- },
- );
- });
- });
-}
diff --git a/frontend/appflowy_flutter/integration_test/cloud/workspace/workspace_settings_test.dart b/frontend/appflowy_flutter/integration_test/cloud/workspace/workspace_settings_test.dart
deleted file mode 100644
index 7276b7995a..0000000000
--- a/frontend/appflowy_flutter/integration_test/cloud/workspace/workspace_settings_test.dart
+++ /dev/null
@@ -1,96 +0,0 @@
-// 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/plugins/document/presentation/editor_style.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/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/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('workspace settings: ', () {
- testWidgets(
- 'change document width',
- (tester) async {
- await tester.initializeAppFlowy(
- cloudType: AuthenticatorType.appflowyCloudSelfHost,
- );
- await tester.tapGoogleLoginInButton();
- await tester.expectToSeeHomePageWithGetStartedPage();
-
- await tester.openSettings();
- await tester.openSettingsPage(SettingsPage.workspace);
-
- final documentWidthSettings = find.findTextInFlowyText(
- LocaleKeys.settings_appearance_documentSettings_width.tr(),
- );
-
- final scrollable = find.ancestor(
- of: find.byType(SettingsWorkspaceView),
- matching: find.descendant(
- of: find.byType(SingleChildScrollView),
- matching: find.byType(Scrollable),
- ),
- );
-
- await tester.scrollUntilVisible(
- documentWidthSettings,
- 0,
- scrollable: scrollable,
- );
- await tester.pumpAndSettle();
-
- // change the document width
- final slider = find.byType(Slider);
- final oldValue = tester.widget(slider).value;
- await tester.drag(slider, const Offset(-100, 0));
- await tester.pumpAndSettle();
-
- // check the document width is changed
- expect(tester.widget(slider).value, lessThan(oldValue));
-
- // click the reset button
- final resetButton = find.descendant(
- of: find.byType(DocumentPaddingSetting),
- matching: find.byType(SettingsResetButton),
- );
- await tester.tap(resetButton);
- await tester.pumpAndSettle();
-
- // check the document width is reset
- expect(
- tester.widget(slider).value,
- EditorStyleCustomizer.maxDocumentWidth,
- );
- },
- );
- });
-}
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 a83372fcc6..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
@@ -4,7 +4,6 @@ import 'package:appflowy/plugins/database/board/presentation/widgets/board_colum
import 'package:appflowy/plugins/database/board/presentation/widgets/board_hidden_groups.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.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';
@@ -24,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 {
@@ -49,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(
@@ -121,22 +123,3 @@ void main() {
});
});
}
-
-extension FlowySvgFinder on CommonFinders {
- Finder byFlowySvg(FlowySvgData svg) => _FlowySvgFinder(svg);
-}
-
-class _FlowySvgFinder extends MatchFinder {
- _FlowySvgFinder(this.svg);
-
- final FlowySvgData svg;
-
- @override
- String get description => 'flowy_svg "$svg"';
-
- @override
- bool matches(Element candidate) {
- final Widget widget = candidate.widget;
- return widget is FlowySvg && widget.svg == svg;
- }
-}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/board/board_row_test.dart b/frontend/appflowy_flutter/integration_test/desktop/board/board_row_test.dart
index 068de0e279..868c27d302 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/board/board_row_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/board/board_row_test.dart
@@ -6,6 +6,7 @@ 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:time/time.dart';
import '../../shared/database_test_op.dart';
import '../../shared/util.dart';
@@ -14,6 +15,31 @@ void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
group('board row test', () {
+ testWidgets('edit item in ToDo card', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Board);
+ const name = 'Card 1';
+ final card1 = find.ancestor(
+ matching: find.byType(RowCard),
+ of: find.text(name),
+ );
+ await tester.hoverOnWidget(
+ card1,
+ onHover: () async {
+ final editCard = find.byType(EditCardAccessory);
+ await tester.tapButton(editCard);
+ },
+ );
+ await tester.showKeyboard(card1);
+ tester.testTextInput.enterText("");
+ await tester.pump(300.milliseconds);
+ tester.testTextInput.enterText("a");
+ await tester.pump(300.milliseconds);
+ expect(find.text('a'), findsOneWidget);
+ });
+
testWidgets('delete item in ToDo card', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/cloud_runner.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/cloud_runner.dart
new file mode 100644
index 0000000000..a8c05d5f80
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/cloud_runner.dart
@@ -0,0 +1,35 @@
+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;
+
+Future main() async {
+ preset_af_cloud_env_test.main();
+
+ data_migration_test_runner.main();
+
+ // uncategorized
+ uncategorized_test_runner.main();
+
+ // workspace
+ workspace_test_runner.main();
+
+ // document
+ document_test_runner.main();
+
+ // 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
new file mode 100644
index 0000000000..e34ac02aab
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/data_migration/anon_user_data_migration_test.dart
@@ -0,0 +1,50 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/workspace/application/settings/prelude.dart';
+import 'package:appflowy/workspace/presentation/settings/pages/account/account.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('appflowy cloud', () {
+ testWidgets('anon user -> sign in -> open imported space', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+
+ await tester.tapAnonymousSignInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ const pageName = 'Test Document';
+ await tester.createNewPageWithNameUnderParent(name: pageName);
+ tester.expectToSeePageName(pageName);
+
+ // rename the name of the anon user
+ await tester.openSettings();
+ await tester.openSettingsPage(SettingsPage.account);
+ await tester.pumpAndSettle();
+
+ await tester.enterUserName('local_user');
+
+ // Scroll to sign-in
+ await tester.tapButton(find.byType(AccountSignInOutButton));
+
+ // sign up with Google
+ await tester.tapGoogleLoginInButton();
+ // await tester.pumpAndSettle(const Duration(seconds: 16));
+
+ // open the imported space
+ await tester.expectToSeeHomePage();
+ await tester.clickSpaceHeader();
+
+ // After import the anon user data, we will create a new space for it
+ await tester.openSpace("Getting started");
+ await tester.openPage(pageName);
+
+ await tester.pumpAndSettle();
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/data_migration/data_migration_test_runner.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/data_migration/data_migration_test_runner.dart
new file mode 100644
index 0000000000..a69c0480ce
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/data_migration/data_migration_test_runner.dart
@@ -0,0 +1,5 @@
+import 'anon_user_data_migration_test.dart' as anon_user_test;
+
+void main() async {
+ anon_user_test.main();
+}
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_copy_link_to_block_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_copy_link_to_block_test.dart
new file mode 100644
index 0000000000..24106cf99a
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_copy_link_to_block_test.dart
@@ -0,0 +1,275 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/document/document_page.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/mention/mention_block.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/mention/mention_page_block.dart';
+import 'package:appflowy/shared/patterns/common_patterns.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/constants.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ // copy link to block
+ group('copy link to block:', () {
+ testWidgets('copy link to check if the clipboard has the correct content',
+ (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // open getting started page
+ await tester.openPage(Constants.gettingStartedPageName);
+ await tester.editor.copyLinkToBlock([0]);
+ await tester.pumpAndSettle(Durations.short1);
+
+ // check the clipboard
+ final content = await Clipboard.getData(Clipboard.kTextPlain);
+ expect(
+ content?.text,
+ matches(appflowySharePageLinkPattern),
+ );
+ });
+
+ testWidgets('copy link to block(another page) and paste it in doc',
+ (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // open getting started page
+ await tester.openPage(Constants.gettingStartedPageName);
+ await tester.editor.copyLinkToBlock([0]);
+
+ // create a new page and paste it
+ const pageName = 'copy link to block';
+ await tester.createNewPageInSpace(
+ spaceName: Constants.generalSpaceName,
+ layout: ViewLayoutPB.Document,
+ pageName: pageName,
+ );
+
+ // paste the link to the new page
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.editor.paste();
+ await tester.pumpAndSettle();
+
+ // check the content of the block
+ final node = tester.editor.getNodeAtPath([0]);
+ 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.page.name);
+ expect(mention[MentionBlockKeys.blockId], isNotNull);
+ expect(mention[MentionBlockKeys.pageId], isNotNull);
+ expect(
+ find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.textContaining(
+ Constants.gettingStartedPageName,
+ findRichText: true,
+ ),
+ ),
+ findsOneWidget,
+ );
+ expect(
+ find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.textContaining(
+ // the pasted block content is 'Welcome to AppFlowy'
+ 'Welcome to AppFlowy',
+ findRichText: true,
+ ),
+ ),
+ findsOneWidget,
+ );
+
+ // tap the mention block to jump to the page
+ await tester.tapButton(find.byType(MentionPageBlock));
+ await tester.pumpAndSettle();
+
+ // expect to go to the getting started page
+ final documentPage = find.byType(DocumentPage);
+ expect(documentPage, findsOneWidget);
+ expect(
+ tester.widget(documentPage).view.name,
+ Constants.gettingStartedPageName,
+ );
+ // and the block is selected
+ expect(
+ tester.widget(documentPage).initialBlockId,
+ mention[MentionBlockKeys.blockId],
+ );
+ expect(
+ tester.editor.getCurrentEditorState().selection,
+ Selection.collapsed(
+ Position(
+ path: [0],
+ ),
+ ),
+ );
+ });
+
+ testWidgets('copy link to block(same page) and paste it in doc',
+ (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // create a new page and paste it
+ const pageName = 'copy link to block';
+ await tester.createNewPageInSpace(
+ spaceName: Constants.generalSpaceName,
+ layout: ViewLayoutPB.Document,
+ pageName: pageName,
+ );
+
+ // copy the link to block from the first line
+ const inputText = 'Hello World';
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText(inputText);
+ await tester.ime.insertCharacter('\n');
+ await tester.pumpAndSettle();
+ await tester.editor.copyLinkToBlock([0]);
+
+ // paste the link to the second line
+ await tester.editor.tapLineOfEditorAt(1);
+ await tester.editor.paste();
+ await tester.pumpAndSettle();
+
+ // check the content of the block
+ final node = tester.editor.getNodeAtPath([1]);
+ 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.page.name);
+ expect(mention[MentionBlockKeys.blockId], isNotNull);
+ expect(mention[MentionBlockKeys.pageId], isNotNull);
+ expect(
+ find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.textContaining(
+ inputText,
+ findRichText: true,
+ ),
+ ),
+ findsNWidgets(2),
+ );
+
+ // edit the pasted block
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText('!');
+ await tester.pumpAndSettle();
+
+ // check the content of the block
+ expect(
+ find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.textContaining(
+ '$inputText!',
+ findRichText: true,
+ ),
+ ),
+ findsNWidgets(2),
+ );
+
+ // tap the mention block
+ await tester.tapButton(find.byType(MentionPageBlock));
+ expect(
+ tester.editor.getCurrentEditorState().selection,
+ Selection.collapsed(
+ Position(
+ path: [0],
+ ),
+ ),
+ );
+ });
+
+ testWidgets('''1. copy link to block from another page
+ 2. paste the link to the new page
+ 3. delete the original page
+ 4. check the content of the block, it should be no access to the page
+ ''', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // open getting started page
+ await tester.openPage(Constants.gettingStartedPageName);
+ await tester.editor.copyLinkToBlock([0]);
+
+ // create a new page and paste it
+ const pageName = 'copy link to block';
+ await tester.createNewPageInSpace(
+ spaceName: Constants.generalSpaceName,
+ layout: ViewLayoutPB.Document,
+ pageName: pageName,
+ );
+
+ // paste the link to the new page
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.editor.paste();
+ await tester.pumpAndSettle();
+
+ // tap the mention block to jump to the page
+ await tester.tapButton(find.byType(MentionPageBlock));
+ await tester.pumpAndSettle();
+
+ // expect to go to the getting started page
+ final documentPage = find.byType(DocumentPage);
+ expect(documentPage, findsOneWidget);
+ expect(
+ tester.widget(documentPage).view.name,
+ Constants.gettingStartedPageName,
+ );
+ // delete the getting started page
+ await tester.hoverOnPageName(
+ Constants.gettingStartedPageName,
+ onHover: () async => tester.tapDeletePageButton(),
+ );
+ tester.expectToSeeDocumentBanner();
+ tester.expectNotToSeePageName(gettingStarted);
+
+ // delete the page permanently
+ await tester.tapDeletePermanentlyButton();
+
+ // go back the page
+ await tester.openPage(pageName);
+ await tester.pumpAndSettle();
+
+ // check the content of the block
+ // it should be no access to the page
+ expect(
+ find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.findTextInFlowyText(
+ LocaleKeys.document_mention_noAccess.tr(),
+ ),
+ ),
+ findsOneWidget,
+ );
+ });
+ });
+}
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
new file mode 100644
index 0000000000..1bc9bd8f92
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_option_actions_test.dart
@@ -0,0 +1,114 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/actions/drag_to_reorder/draggable_option_button.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/constants.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('document option actions:', () {
+ testWidgets('drag block to the top', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // open getting started page
+ await tester.openPage(Constants.gettingStartedPageName);
+
+ // before move
+ final beforeMoveBlock = tester.editor.getNodeAtPath([1]);
+
+ // move the desktop guide to the top, above the getting started
+ await tester.editor.dragBlock(
+ [1],
+ const Offset(20, -80),
+ );
+
+ // wait for the move animation to complete
+ await tester.pumpAndSettle(Durations.short1);
+
+ // check if the block is moved to the top
+ final afterMoveBlock = tester.editor.getNodeAtPath([0]);
+ expect(afterMoveBlock.delta, beforeMoveBlock.delta);
+ });
+
+ testWidgets('drag block to other block\'s child', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // open getting started page
+ await tester.openPage(Constants.gettingStartedPageName);
+
+ // before move
+ final beforeMoveBlock = tester.editor.getNodeAtPath([10]);
+
+ // move the checkbox to the child of the block at path [9]
+ await tester.editor.dragBlock(
+ [10],
+ const Offset(120, -20),
+ );
+
+ // wait for the move animation to complete
+ await tester.pumpAndSettle(Durations.short1);
+
+ // check if the block is moved to the child of the block at path [9]
+ final afterMoveBlock = tester.editor.getNodeAtPath([9, 0]);
+ expect(afterMoveBlock.delta, beforeMoveBlock.delta);
+ });
+
+ testWidgets('hover on the block and delete it', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // open getting started page
+ await tester.openPage(Constants.gettingStartedPageName);
+
+ // before delete
+ final path = [1];
+ final beforeDeletedBlock = tester.editor.getNodeAtPath(path);
+
+ // hover on the block and delete it
+ final optionButton = find.byWidgetPredicate(
+ (widget) =>
+ widget is DraggableOptionButton &&
+ widget.blockComponentContext.node.path.equals(path),
+ );
+
+ await tester.hoverOnWidget(
+ optionButton,
+ onHover: () async {
+ // click the delete button
+ await tester.tapButton(optionButton);
+ },
+ );
+ await tester.pumpAndSettle(Durations.short1);
+
+ // click the delete button
+ final deleteButton =
+ find.findTextInFlowyText(LocaleKeys.button_delete.tr());
+ await tester.tapButton(deleteButton);
+
+ // wait for the deletion
+ await tester.pumpAndSettle(Durations.short1);
+
+ // check if the block is deleted
+ final afterDeletedBlock = tester.editor.getNodeAtPath([1]);
+ expect(afterDeletedBlock.id, isNot(equals(beforeDeletedBlock.id)));
+ });
+ });
+}
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
new file mode 100644
index 0000000000..7877143116
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_publish_test.dart
@@ -0,0 +1,220 @@
+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/shared/share/publish_tab.dart';
+import 'package:appflowy/plugins/shared/share/share_menu.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/view.pb.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/constants.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('Publish:', () {
+ testWidgets('publish 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);
+ final unpublishButton = find.byType(UnPublishButton);
+ await tester.tapButton(publishButton);
+
+ // expect to see unpublish, visit site and manage all sites button
+ expect(unpublishButton, findsOneWidget);
+ expect(find.text(LocaleKeys.shareAction_visitSite.tr()), findsOneWidget);
+
+ // unpublish the document
+ await tester.tapButton(unpublishButton);
+
+ // expect to see publish button
+ expect(publishButton, findsOneWidget);
+ });
+
+ testWidgets('rename path name', (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),
+ );
+
+ // rename with invalid 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 errorToast1 = find.text(
+ LocaleKeys.settings_sites_error_publishNameContainsInvalidCharacters
+ .tr(),
+ );
+ await tester.pumpUntilFound(errorToast1);
+ await tester.pumpUntilNotFound(errorToast1);
+
+ // rename with long name
+ await tester.tap(inputField);
+ await tester.enterText(inputField, 'long-path-name' * 200);
+ await tester.tapButton(find.text(LocaleKeys.button_save.tr()));
+ await tester.pumpAndSettle();
+
+ // expect to see the toast with error message
+ final errorToast2 = find.text(
+ LocaleKeys.settings_sites_error_publishNameTooLong.tr(),
+ );
+ 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()));
+ await tester.pumpAndSettle();
+
+ // expect to see the toast with success message
+ final successToast = find.text(
+ LocaleKeys.settings_sites_success_updatePathNameSuccess.tr(),
+ );
+ await tester.pumpUntilFound(successToast);
+ await tester.pumpUntilNotFound(successToast);
+
+ // click the copy link button
+ await tester.tapButton(
+ find.byWidgetPredicate(
+ (widget) =>
+ widget is FlowySvg &&
+ widget.svg.path == FlowySvgs.m_toolbar_link_m.path,
+ ),
+ );
+ await tester.pumpAndSettle();
+ // check the clipboard has the link
+ final content = await Clipboard.getData(Clipboard.kTextPlain);
+ expect(
+ content?.text?.contains('new-path-name'),
+ 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
new file mode 100644
index 0000000000..58a9d7398b
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/document/document_test_runner.dart
@@ -0,0 +1,16 @@
+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;
+import 'document_publish_test.dart' as document_publish_test;
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ 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/set_env.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/set_env.dart
new file mode 100644
index 0000000000..b24c0faf27
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/set_env.dart
@@ -0,0 +1,18 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+// This test is meaningless, just for preventing the CI from failing.
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('Empty', () {
+ testWidgets('set appflowy cloud', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ });
+ });
+}
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
new file mode 100644
index 0000000000..37abd19ebc
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/sidebar/sidebar_move_page_test.dart
@@ -0,0 +1,108 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/generated/locale_keys.g.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_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+import 'package:universal_platform/universal_platform.dart';
+
+import '../../../shared/constants.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('sidebar move page: ', () {
+ testWidgets('create a new document and move it to Getting started',
+ (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,
+ );
+
+ // click the ... button and move to Getting started
+ await tester.hoverOnPageName(
+ pageName,
+ onHover: () async {
+ await tester.tapPageOptionButton();
+ await tester.tapButtonWithName(
+ LocaleKeys.disclosureAction_moveTo.tr(),
+ );
+ },
+ );
+
+ // expect to see two pages
+ // one is in the sidebar, the other is in the move to page list
+ // 1. Getting started
+ // 2. To-dos
+ final gettingStarted = find.findTextInFlowyText(
+ Constants.gettingStartedPageName,
+ );
+ final toDos = find.findTextInFlowyText(Constants.toDosPageName);
+ await tester.pumpUntilFound(gettingStarted);
+ await tester.pumpUntilFound(toDos);
+ expect(gettingStarted, findsNWidgets(2));
+
+ // skip the length check on Linux temporarily,
+ // because it failed in expect check but the previous pumpUntilFound is successful
+ if (!UniversalPlatform.isLinux) {
+ expect(toDos, findsNWidgets(2));
+
+ // hover on the todos page, and will see a forbidden icon
+ await tester.hoverOnWidget(
+ toDos.last,
+ onHover: () async {
+ final tooltips = find.byTooltip(
+ LocaleKeys.space_cannotMovePageToDatabase.tr(),
+ );
+ expect(tooltips, findsOneWidget);
+ },
+ );
+ await tester.pumpAndSettle();
+ }
+
+ // Attempt right-click on the page name and expect not to see
+ await tester.tap(gettingStarted.last, buttons: kSecondaryButton);
+ await tester.pumpAndSettle();
+ expect(
+ find.text(LocaleKeys.disclosureAction_moveTo.tr()),
+ findsOneWidget,
+ );
+
+ // move the current page to Getting started
+ await tester.tapButton(
+ gettingStarted.last,
+ );
+
+ await tester.pumpAndSettle();
+
+ // after moving, expect to not see the page name in the sidebar
+ final page = tester.findPageName(pageName);
+ expect(page, findsNothing);
+
+ // click to expand the getting started page
+ await tester.expandOrCollapsePage(
+ pageName: Constants.gettingStartedPageName,
+ layout: ViewLayoutPB.Document,
+ );
+ await tester.pumpAndSettle();
+
+ // expect to see the page name in the getting started page
+ final pageInGettingStarted = tester.findPageName(
+ pageName,
+ parentName: Constants.gettingStartedPageName,
+ );
+ expect(pageInGettingStarted, findsOneWidget);
+ });
+ });
+}
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/cloud/appflowy_cloud_auth_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/appflowy_cloud_auth_test.dart
similarity index 82%
rename from frontend/appflowy_flutter/integration_test/cloud/appflowy_cloud_auth_test.dart
rename to frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/appflowy_cloud_auth_test.dart
index f0982e044c..fd65c29927 100644
--- a/frontend/appflowy_flutter/integration_test/cloud/appflowy_cloud_auth_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/appflowy_cloud_auth_test.dart
@@ -1,22 +1,13 @@
-// 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';
+import '../../../shared/util.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
@@ -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
new file mode 100644
index 0000000000..b6b4ecf025
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/document_sync_test.dart
@@ -0,0 +1,55 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/workspace/application/settings/prelude.dart';
+import 'package:flowy_infra/uuid.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ final email = '${uuid()}@appflowy.io';
+ const inputContent = 'Hello world, this is a test document';
+
+// The test will create a new document called Sample, and sync it to the server.
+// Then the test will logout the user, and login with the same user. The data will
+// be synced from the server.
+ group('appflowy cloud document', () {
+ testWidgets('sync local docuemnt to server', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ email: email,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // create a new document called Sample
+ await tester.createNewPage();
+
+ // focus on the editor
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText(inputContent);
+ expect(find.text(inputContent, findRichText: true), findsOneWidget);
+
+ // 6 seconds for data sync
+ await tester.waitForSeconds(6);
+
+ await tester.openSettings();
+ await tester.openSettingsPage(SettingsPage.account);
+ await tester.logout();
+ });
+
+ testWidgets('sync doc from server', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ email: email,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePage();
+
+ // the latest document will be opened, so the content must be the inputContent
+ await tester.pumpAndSettle();
+ expect(find.text(inputContent, findRichText: true), findsOneWidget);
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/uncategorized_test_runner.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/uncategorized_test_runner.dart
new file mode 100644
index 0000000000..278d880965
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/uncategorized_test_runner.dart
@@ -0,0 +1,7 @@
+import 'appflowy_cloud_auth_test.dart' as appflowy_cloud_auth_test;
+import 'user_setting_sync_test.dart' as user_sync_test;
+
+void main() async {
+ appflowy_cloud_auth_test.main();
+ user_sync_test.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
new file mode 100644
index 0000000000..e666289bf5
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/uncategorized/user_setting_sync_test.dart
@@ -0,0 +1,52 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/workspace/application/settings/prelude.dart';
+import 'package:appflowy/workspace/presentation/settings/pages/account/account_user_profile.dart';
+import 'package:flowy_infra/uuid.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ final email = '${uuid()}@appflowy.io';
+ const name = 'nathan';
+
+ group('appflowy cloud setting', () {
+ testWidgets('sync user name and icon to server', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ email: email,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ await tester.openSettings();
+ await tester.openSettingsPage(SettingsPage.account);
+
+ await tester.enterUserName(name);
+ await tester.pumpAndSettle(const Duration(seconds: 6));
+ await tester.logout();
+
+ await tester.pumpAndSettle(const Duration(seconds: 2));
+ });
+ });
+ testWidgets('get user icon and name from server', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ email: email,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+ await tester.pumpAndSettle();
+
+ await tester.openSettings();
+ await tester.openSettingsPage(SettingsPage.account);
+
+ // Verify name
+ final profileSetting =
+ tester.widget(find.byType(AccountUserProfile)) as AccountUserProfile;
+
+ expect(profileSetting.name, name);
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/cloud/workspace/change_name_and_icon_test.dart b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/change_name_and_icon_test.dart
similarity index 80%
rename from frontend/appflowy_flutter/integration_test/cloud/workspace/change_name_and_icon_test.dart
rename to frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/change_name_and_icon_test.dart
index 75e420baac..f205b35354 100644
--- a/frontend/appflowy_flutter/integration_test/cloud/workspace/change_name_and_icon_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/change_name_and_icon_test.dart
@@ -1,23 +1,14 @@
-// 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';
+import '../../../shared/util.dart';
+import '../../../shared/workspace.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
@@ -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
new file mode 100644
index 0000000000..4d2e027646
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/collaborative_workspace_test.dart
@@ -0,0 +1,212 @@
+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/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: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/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ 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 {
+ // 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);
+
+ Finder success;
+
+ final Finder items = find.byType(WorkspaceMenuItem);
+
+ // delete the newly created workspace
+ await tester.openCollaborativeWorkspaceMenu();
+ await tester.pumpUntilFound(items);
+
+ expect(items, findsNWidgets(2));
+ expect(
+ tester.widget(items.last).workspace.name,
+ name,
+ );
+
+ final secondWorkspace = find.byType(WorkspaceMenuItem).last;
+ await tester.hoverOnWidget(
+ secondWorkspace,
+ onHover: () async {
+ // click the more button
+ final moreButton = find.byType(WorkspaceMoreActionList);
+ expect(moreButton, findsOneWidget);
+ await tester.tapButton(moreButton);
+ // click the delete button
+ final deleteButton = find.text(LocaleKeys.button_delete.tr());
+ expect(deleteButton, findsOneWidget);
+ await tester.tapButton(deleteButton);
+ // see the delete confirm dialog
+ final confirm =
+ find.text(LocaleKeys.workspace_deleteWorkspaceHintText.tr());
+ expect(confirm, findsOneWidget);
+ await tester.tapButton(find.text(LocaleKeys.button_ok.tr()));
+ // delete success
+ success = find.text(LocaleKeys.workspace_createSuccess.tr());
+ await tester.pumpUntilFound(success);
+ expect(success, findsOneWidget);
+ await tester.pumpUntilNotFound(success);
+ },
+ );
+ });
+
+ 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
new file mode 100644
index 0000000000..70bb46279e
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/share_menu_test.dart
@@ -0,0 +1,65 @@
+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/patterns/common_patterns.dart';
+import 'package:appflowy/startup/startup.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('Share menu:', () {
+ testWidgets('share tab', (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,
+ );
+
+ // click the share button
+ await tester.tapShareButton();
+
+ // expect the share menu is shown
+ final shareMenu = find.byType(ShareMenu);
+ expect(shareMenu, findsOneWidget);
+
+ // click the copy link button
+ final copyLinkButton = find.textContaining(
+ LocaleKeys.button_copyLink.tr(),
+ );
+ await tester.tapButton(copyLinkButton);
+
+ // read the clipboard content
+ final clipboardContent = await getIt().getData();
+ final plainText = clipboardContent.plainText;
+ expect(
+ plainText,
+ matches(appflowySharePageLinkPattern),
+ );
+
+ final shareValues = plainText!
+ .replaceAll('${ShareConstants.defaultBaseWebDomain}/app/', '')
+ .split('/');
+ final workspaceId = shareValues[0];
+ expect(workspaceId, isNotEmpty);
+ final pageId = shareValues[1];
+ expect(pageId, 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
new file mode 100644
index 0000000000..e9ad06caee
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_icon_test.dart
@@ -0,0 +1,44 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/generated/locale_keys.g.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:easy_localization/easy_localization.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../../shared/util.dart';
+import '../../../shared/workspace.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('workspace icon:', () {
+ testWidgets('remove icon from workspace', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ await tester.openWorkspaceMenu();
+
+ // click the workspace icon
+ await tester.tapButton(
+ find.descendant(
+ of: find.byType(WorkspaceMenuItem),
+ matching: find.byType(WorkspaceIcon),
+ ),
+ );
+ // click the remove icon button
+ await tester.tapButton(
+ find.text(LocaleKeys.button_remove.tr()),
+ );
+
+ // nothing should happen
+ expect(
+ find.text(LocaleKeys.workspace_updateIconSuccess.tr()),
+ findsNothing,
+ );
+ });
+ });
+}
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
new file mode 100644
index 0000000000..a58fea25b8
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_settings_test.dart
@@ -0,0 +1,353 @@
+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_style.dart';
+import 'package:appflowy/plugins/shared/share/publish_tab.dart';
+import 'package:appflowy/startup/startup.dart';
+import 'package:appflowy/workspace/application/settings/prelude.dart';
+import 'package:appflowy/workspace/presentation/settings/pages/settings_workspace_view.dart';
+import 'package:appflowy/workspace/presentation/settings/pages/sites/domain/domain_more_action.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_backend/protobuf/flowy-folder/view.pb.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/constants.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('workspace settings: ', () {
+ testWidgets(
+ 'change document width',
+ (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ await tester.openSettings();
+ await tester.openSettingsPage(SettingsPage.workspace);
+
+ final documentWidthSettings = find.findTextInFlowyText(
+ LocaleKeys.settings_appearance_documentSettings_width.tr(),
+ );
+
+ final scrollable = find.ancestor(
+ of: find.byType(SettingsWorkspaceView),
+ matching: find.descendant(
+ of: find.byType(SingleChildScrollView),
+ matching: find.byType(Scrollable),
+ ),
+ );
+
+ await tester.scrollUntilVisible(
+ documentWidthSettings,
+ 0,
+ scrollable: scrollable,
+ );
+ await tester.pumpAndSettle();
+
+ // change the document width
+ final slider = find.byType(Slider);
+ final oldValue = tester.widget(slider).value;
+ await tester.drag(slider, const Offset(-100, 0));
+ await tester.pumpAndSettle();
+
+ // check the document width is changed
+ expect(tester.widget(slider).value, lessThan(oldValue));
+
+ // click the reset button
+ final resetButton = find.descendant(
+ of: find.byType(DocumentPaddingSetting),
+ matching: find.byType(SettingsResetButton),
+ );
+ await tester.tap(resetButton);
+ await tester.pumpAndSettle();
+
+ // check the document width is reset
+ expect(
+ tester.widget(slider).value,
+ EditorStyleCustomizer.maxDocumentWidth,
+ );
+ },
+ );
+ });
+
+ group('sites settings:', () {
+ testWidgets(
+ 'manage published page, set it as homepage, remove the homepage',
+ (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
+ await tester.tapButton(find.byType(PublishButton));
+
+ // click empty area to close the publish menu
+ await tester.tapAt(Offset.zero);
+ await tester.pumpAndSettle();
+ // check if the page is published in sites page
+ await tester.openSettings();
+ await tester.openSettingsPage(SettingsPage.sites);
+ // wait the backend return the sites data
+ await tester.wait(1000);
+
+ // check if the page is published in sites page
+ final pageItem = find.byWidgetPredicate(
+ (widget) =>
+ 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
+ // // set it to homepage
+ // await tester.tapButton(
+ // find.textContaining(
+ // LocaleKeys.settings_sites_selectHomePage.tr(),
+ // ),
+ // );
+ // await tester.tapButton(
+ // find.descendant(
+ // of: find.byType(SelectHomePageMenu),
+ // matching: find.text(pageName),
+ // ),
+ // );
+ // await tester.pumpAndSettle();
+
+ // // check if the page is set to homepage
+ // final homePageItem = find.descendant(
+ // of: find.byType(DomainItem),
+ // matching: find.text(pageName),
+ // );
+ // expect(homePageItem, findsOneWidget);
+
+ // // remove the homepage
+ // await tester.tapButton(find.byType(DomainMoreAction));
+ // await tester.tapButton(
+ // find.text(LocaleKeys.settings_sites_removeHomepage.tr()),
+ // );
+ // await tester.pumpAndSettle();
+
+ // // check if the page is removed from homepage
+ // expect(homePageItem, findsNothing);
+ });
+
+ testWidgets('update namespace', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // check if the page is published in sites page
+ await tester.openSettings();
+ await tester.openSettingsPage(SettingsPage.sites);
+ // wait the backend return the sites data
+ await tester.wait(1000);
+
+ // update the domain
+ final domainMoreAction = find.byType(DomainMoreAction);
+ await tester.tapButton(domainMoreAction);
+ final updateNamespaceButton = find.text(
+ LocaleKeys.settings_sites_updateNamespace.tr(),
+ );
+ await tester.pumpUntilFound(updateNamespaceButton);
+
+ // click the update namespace button
+
+ await tester.tapButton(updateNamespaceButton);
+
+ // comment it out because it's not allowed to update the namespace in free plan
+ // expect to see the dialog
+ // await tester.updateNamespace('&&&???');
+
+ // // need to upgrade to pro plan to update the namespace
+ // final errorToast = find.text(
+ // LocaleKeys.settings_sites_error_proPlanLimitation.tr(),
+ // );
+ // await tester.pumpUntilFound(errorToast);
+ // expect(errorToast, findsOneWidget);
+ // await tester.pumpUntilNotFound(errorToast);
+
+ // comment it out because it's not allowed to update the namespace in free plan
+ // // short namespace
+ // await tester.updateNamespace('a');
+
+ // // expect to see the toast with error message
+ // final errorToast2 = find.text(
+ // LocaleKeys.settings_sites_error_namespaceTooShort.tr(),
+ // );
+ // await tester.pumpUntilFound(errorToast2);
+ // expect(errorToast2, findsOneWidget);
+ // await tester.pumpUntilNotFound(errorToast2);
+ // // valid namespace
+ // await tester.updateNamespace('AppFlowy');
+
+ // // expect to see the toast with success message
+ // final successToast = find.text(
+ // LocaleKeys.settings_sites_success_namespaceUpdated.tr(),
+ // );
+ // await tester.pumpUntilFound(successToast);
+ // expect(successToast, findsOneWidget);
+ });
+
+ testWidgets('''
+More actions for published page:
+1. visit site
+2. copy link
+3. settings
+4. unpublish
+5. custom url
+''', (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
+ await tester.tapButton(find.byType(PublishButton));
+
+ // click empty area to close the publish menu
+ await tester.tapAt(Offset.zero);
+ await tester.pumpAndSettle();
+ // check if the page is published in sites page
+ await tester.openSettings();
+ await tester.openSettingsPage(SettingsPage.sites);
+ // wait the backend return the sites data
+ await tester.wait(2000);
+
+ // check if the page is published in sites page
+ final pageItem = find.byWidgetPredicate(
+ (widget) =>
+ widget is PublishedViewItem &&
+ widget.publishInfoView.view.name == pageName,
+ );
+ if (pageItem.evaluate().isEmpty) {
+ return;
+ }
+
+ expect(pageItem, findsOneWidget);
+
+ final copyLinkItem = find.text(LocaleKeys.shareAction_copyLink.tr());
+ final customUrlItem = find.text(LocaleKeys.settings_sites_customUrl.tr());
+ final unpublishItem = find.text(LocaleKeys.shareAction_unPublish.tr());
+
+ // custom url
+ final publishMoreAction = find.byType(PublishedViewMoreAction);
+
+ // click the copy link button
+ {
+ await tester.tapButton(publishMoreAction);
+ await tester.pumpAndSettle();
+ await tester.pumpUntilFound(copyLinkItem);
+ await tester.tapButton(copyLinkItem);
+ await tester.pumpAndSettle();
+ await tester.pumpUntilNotFound(copyLinkItem);
+
+ final clipboardContent = await getIt().getData();
+ final plainText = clipboardContent.plainText;
+ expect(
+ plainText,
+ contains(pageName),
+ );
+ }
+
+ // custom url
+ {
+ await tester.tapButton(publishMoreAction);
+ await tester.pumpAndSettle();
+ await tester.pumpUntilFound(customUrlItem);
+ await tester.tapButton(customUrlItem);
+ await tester.pumpAndSettle();
+ await tester.pumpUntilNotFound(customUrlItem);
+
+ // see the custom url dialog
+ final customUrlDialog = find.byType(PublishedViewSettingsDialog);
+ expect(customUrlDialog, findsOneWidget);
+
+ // rename the custom url
+ final textField = find.descendant(
+ of: customUrlDialog,
+ matching: find.byType(TextField),
+ );
+ await tester.enterText(textField, 'hello-world');
+ await tester.pumpAndSettle();
+
+ // click the save button
+ final saveButton = find.descendant(
+ of: customUrlDialog,
+ matching: find.text(LocaleKeys.button_save.tr()),
+ );
+ await tester.tapButton(saveButton);
+ await tester.pumpAndSettle();
+
+ // expect to see the toast with success message
+ final successToast = find.text(
+ LocaleKeys.settings_sites_success_updatePathNameSuccess.tr(),
+ );
+ await tester.pumpUntilFound(successToast);
+ expect(successToast, findsOneWidget);
+ }
+
+ // unpublish
+ {
+ await tester.tapButton(publishMoreAction);
+ await tester.pumpAndSettle();
+ await tester.pumpUntilFound(unpublishItem);
+ await tester.tapButton(unpublishItem);
+ await tester.pumpAndSettle();
+ await tester.pumpUntilNotFound(unpublishItem);
+
+ // expect to see the toast with success message
+ final successToast = find.text(
+ LocaleKeys.publish_unpublishSuccessfully.tr(),
+ );
+ await tester.pumpUntilFound(successToast);
+ expect(successToast, findsOneWidget);
+ await tester.pumpUntilNotFound(successToast);
+
+ // check if the page is unpublished in sites page
+ expect(pageItem, findsNothing);
+ }
+ });
+ });
+}
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
new file mode 100644
index 0000000000..4d2862038e
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/cloud/workspace/workspace_test_runner.dart
@@ -0,0 +1,19 @@
+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;
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ workspace_settings_test.main();
+ share_menu_test.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 a9912e3ef3..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,5 +1,6 @@
import 'package:appflowy/plugins/database/calendar/presentation/calendar_event_editor.dart';
-import 'package:appflowy_backend/protobuf/flowy-database2/setting_entities.pbenum.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';
import 'package:flutter_test/flutter_test.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 {
@@ -277,5 +285,74 @@ void main() {
tester.assertRowDetailPageOpened();
});
+
+ testWidgets('filter calendar events', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // Create the calendar view
+ await tester.createNewPageWithNameUnderParent(
+ layout: ViewLayoutPB.Calendar,
+ );
+
+ // Create a new event on the first of this month
+ final today = DateTime.now();
+ final firstOfThisMonth = DateTime(today.year, today.month);
+ await tester.doubleClickCalendarCell(firstOfThisMonth);
+ await tester.dismissEventEditor();
+
+ tester.assertNumberOfEventsInCalendar(1);
+
+ await tester.openCalendarEvent(index: 0, date: firstOfThisMonth);
+ await tester.tapButton(finderForFieldType(FieldType.MultiSelect));
+ await tester.createOption(name: "asdf");
+ 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");
+
+ await tester.tapFilterButtonInGrid('Tags');
+ await tester.tapOptionFilterWithName('asdf');
+ await tester.dismissCellEditor();
+
+ tester.assertNumberOfEventsInCalendar(0);
+
+ await tester.tapFilterButtonInGrid('Tags');
+ await tester.tapOptionFilterWithName('asdf');
+ await tester.dismissCellEditor();
+
+ tester.assertNumberOfEventsInCalendar(1);
+
+ await tester.tapFilterButtonInGrid('Tags');
+ await tester.tapOptionFilterWithName('asdf');
+ await tester.dismissCellEditor();
+
+ tester.assertNumberOfEventsInCalendar(0);
+
+ final secondOfThisMonth = DateTime(today.year, today.month, 2);
+ await tester.doubleClickCalendarCell(secondOfThisMonth);
+ await tester.dismissEventEditor();
+ tester.assertNumberOfEventsInCalendar(1);
+
+ await tester.openCalendarEvent(index: 0, date: secondOfThisMonth);
+ await tester.tapButton(finderForFieldType(FieldType.MultiSelect));
+ await tester.selectOption(name: "asdf");
+ await tester.dismissCellEditor();
+ await tester.dismissCellEditor();
+
+ tester.assertNumberOfEventsInCalendar(0);
+
+ await tester.tapFilterButtonInGrid('Tags');
+ await tester.changeSelectFilterCondition(
+ SelectOptionFilterConditionPB.OptionIsEmpty,
+ );
+ await tester.dismissCellEditor();
+
+ tester.assertNumberOfEventsInCalendar(1);
+ tester.assertNumberOfEventsOnSpecificDay(1, secondOfThisMonth);
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_cell_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_cell_test.dart
index 28f50bf817..ca565474ec 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/database/database_cell_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_cell_test.dart
@@ -211,21 +211,21 @@ void main() {
await tester.toggleIncludeTime();
// Select a date
- final today = DateTime.now();
- await tester.selectDay(content: today.day);
+ DateTime now = DateTime.now();
+ await tester.selectDay(content: now.day);
await tester.dismissCellEditor();
tester.assertCellContent(
rowIndex: 0,
fieldType: FieldType.DateTime,
- content: DateFormat('MMM dd, y').format(today),
+ content: DateFormat('MMM dd, y').format(now),
);
await tester.tapCellInGrid(rowIndex: 0, fieldType: fieldType);
// Toggle include time
- final now = DateTime.now();
+ now = DateTime.now();
await tester.toggleIncludeTime();
await tester.dismissCellEditor();
@@ -299,7 +299,7 @@ void main() {
await tester.dismissCellEditor();
// Make sure the option is created and displayed in the cell
- await tester.findSelectOptionWithNameInGrid(
+ tester.findSelectOptionWithNameInGrid(
rowIndex: 0,
name: 'tag 1',
);
@@ -311,12 +311,12 @@ void main() {
await tester.createOption(name: 'tag 2');
await tester.dismissCellEditor();
- await tester.findSelectOptionWithNameInGrid(
+ tester.findSelectOptionWithNameInGrid(
rowIndex: 0,
name: 'tag 2',
);
- await tester.assertNumberOfSelectedOptionsInGrid(
+ tester.assertNumberOfSelectedOptionsInGrid(
rowIndex: 0,
matcher: findsOneWidget,
);
@@ -328,12 +328,12 @@ void main() {
await tester.selectOption(name: 'tag 1');
await tester.dismissCellEditor();
- await tester.findSelectOptionWithNameInGrid(
+ tester.findSelectOptionWithNameInGrid(
rowIndex: 0,
name: 'tag 1',
);
- await tester.assertNumberOfSelectedOptionsInGrid(
+ tester.assertNumberOfSelectedOptionsInGrid(
rowIndex: 0,
matcher: findsOneWidget,
);
@@ -345,7 +345,7 @@ void main() {
await tester.selectOption(name: 'tag 1');
await tester.dismissCellEditor();
- await tester.assertNumberOfSelectedOptionsInGrid(
+ tester.assertNumberOfSelectedOptionsInGrid(
rowIndex: 0,
matcher: findsNothing,
);
@@ -378,7 +378,7 @@ void main() {
await tester.dismissCellEditor();
// Make sure the option is created and displayed in the cell
- await tester.findSelectOptionWithNameInGrid(
+ tester.findSelectOptionWithNameInGrid(
rowIndex: 0,
name: tags.first,
);
@@ -393,13 +393,13 @@ void main() {
await tester.dismissCellEditor();
for (final tag in tags) {
- await tester.findSelectOptionWithNameInGrid(
+ tester.findSelectOptionWithNameInGrid(
rowIndex: 0,
name: tag,
);
}
- await tester.assertNumberOfSelectedOptionsInGrid(
+ tester.assertNumberOfSelectedOptionsInGrid(
rowIndex: 0,
matcher: findsNWidgets(4),
);
@@ -413,7 +413,7 @@ void main() {
}
await tester.dismissCellEditor();
- await tester.assertNumberOfSelectedOptionsInGrid(
+ tester.assertNumberOfSelectedOptionsInGrid(
rowIndex: 0,
matcher: findsNothing,
);
@@ -426,16 +426,16 @@ void main() {
await tester.selectOption(name: tags[3]);
await tester.dismissCellEditor();
- await tester.findSelectOptionWithNameInGrid(
+ tester.findSelectOptionWithNameInGrid(
rowIndex: 0,
name: tags[1],
);
- await tester.findSelectOptionWithNameInGrid(
+ tester.findSelectOptionWithNameInGrid(
rowIndex: 0,
name: tags[3],
);
- await tester.assertNumberOfSelectedOptionsInGrid(
+ tester.assertNumberOfSelectedOptionsInGrid(
rowIndex: 0,
matcher: findsNWidgets(2),
);
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 865ea15479..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
@@ -1,5 +1,5 @@
import 'package:appflowy/plugins/database/grid/presentation/grid_page.dart';
-import 'package:appflowy_backend/protobuf/flowy-database2/setting_entities.pbenum.dart';
+import 'package:appflowy_backend/protobuf/flowy-database2/protobuf.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/view.pbenum.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -10,11 +10,12 @@ import '../../shared/util.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- group('database field settings', () {
+ group('grid field settings test:', () {
testWidgets('field visibility', (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);
@@ -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');
@@ -50,6 +56,50 @@ void main() {
await tester.tapHidePropertyButtonInFieldEditor();
await tester.dismissRowDetailPage();
tester.noFieldWithName('New field 1');
+
+ // the field should still be sort and filter-able
+ await tester.tapDatabaseFilterButton();
+ await tester.tapCreateFilterByFieldType(
+ FieldType.RichText,
+ "New field 1",
+ );
+ 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 45d05207ff..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,8 +1,9 @@
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/field_entities.pbenum.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';
@@ -13,9 +14,16 @@ import '../../shared/database_test_op.dart';
import '../../shared/util.dart';
void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
- group('grid field editor:', () {
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
+
+ group('grid edit field test:', () {
testWidgets('rename existing field', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
@@ -24,7 +32,6 @@ void main() {
// Invoke the field editor
await tester.tapGridFieldWithName('Name');
- await tester.tapEditFieldButton();
await tester.renameField('hello world');
await tester.dismissFieldEditor();
@@ -33,6 +40,32 @@ void main() {
await tester.pumpAndSettle();
});
+ testWidgets('edit field icon', (tester) async {
+ const icon = 'artificial_intelligence/ai-upscale-spark';
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
+
+ tester.assertFieldSvg('Name', FieldType.RichText);
+
+ // choose specific icon
+ await tester.tapGridFieldWithName('Name');
+ await tester.changeFieldIcon(icon);
+ await tester.dismissFieldEditor();
+
+ tester.assertFieldCustomSvg('Name', icon);
+
+ // remove icon
+ await tester.tapGridFieldWithName('Name');
+ await tester.changeFieldIcon('');
+ await tester.dismissFieldEditor();
+
+ tester.assertFieldSvg('Name', FieldType.RichText);
+
+ await tester.pumpAndSettle();
+ });
+
testWidgets('update field type of existing field', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
@@ -126,7 +159,7 @@ void main() {
await tester.dismissFieldEditor();
tester.findFieldWithName('Right');
- // insert new field to the right
+ // insert new field to the left
await tester.tapGridFieldWithName('Type');
await tester.tapInsertFieldButton(left: true, name: "Left");
await tester.dismissFieldEditor();
@@ -245,7 +278,6 @@ void main() {
matching: find.byType(TextField),
);
await tester.enterText(inputField, text);
- await tester.pumpAndSettle();
await tester.testTextInput.receiveAction(TextInputAction.done);
await tester.pumpAndSettle(const Duration(milliseconds: 500));
@@ -292,6 +324,30 @@ void main() {
);
});
+ testWidgets('text in viewport while typing', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
+
+ await tester.changeCalculateAtIndex(0, CalculationType.Count);
+
+ // add very large text with 200 lines
+ final largeText = List.generate(
+ 200,
+ (index) => 'Line ${index + 1}',
+ ).join('\n');
+
+ await tester.editCell(
+ rowIndex: 2,
+ fieldType: FieldType.RichText,
+ input: largeText,
+ );
+
+ // checks if last line is in view port
+ tester.expectToSeeText('Line 200');
+ });
+
// Disable this test because it fails on CI randomly
// testWidgets('last modified and created at field type options',
// (tester) async {
@@ -357,5 +413,188 @@ void main() {
// content: DateFormat('dd/MM/y hh:mm a').format(modified),
// );
// });
+
+ testWidgets('select option transform', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(
+ layout: ViewLayoutPB.Grid,
+ );
+
+ // invoke the field editor of existing Single-Select field Type
+ await tester.tapGridFieldWithName('Type');
+ await tester.tapEditFieldButton();
+
+ // add some select options
+ await tester.tapAddSelectOptionButton();
+ for (final optionName in ['A', 'B', 'C']) {
+ final inputField = find.descendant(
+ of: find.byType(CreateOptionTextField),
+ matching: find.byType(TextField),
+ );
+ await tester.enterText(inputField, optionName);
+ await tester.testTextInput.receiveAction(TextInputAction.done);
+ }
+ await tester.dismissFieldEditor();
+
+ // select A in first row's cell under the Type field
+ await tester.tapCellInGrid(
+ rowIndex: 0,
+ fieldType: FieldType.SingleSelect,
+ );
+ await tester.selectOption(name: 'A');
+ await tester.dismissCellEditor();
+ tester.findSelectOptionWithNameInGrid(name: 'A', rowIndex: 0);
+
+ await tester.changeFieldTypeOfFieldWithName('Type', FieldType.RichText);
+ tester.assertCellContent(
+ rowIndex: 0,
+ fieldType: FieldType.RichText,
+ content: "A",
+ cellIndex: 1,
+ );
+
+ // add some random text in the second row
+ await tester.editCell(
+ rowIndex: 1,
+ fieldType: FieldType.RichText,
+ input: "random",
+ cellIndex: 1,
+ );
+ tester.assertCellContent(
+ rowIndex: 1,
+ fieldType: FieldType.RichText,
+ content: "random",
+ cellIndex: 1,
+ );
+
+ await tester.changeFieldTypeOfFieldWithName(
+ 'Type',
+ FieldType.SingleSelect,
+ );
+ tester.findSelectOptionWithNameInGrid(name: 'A', rowIndex: 0);
+ tester.assertNumberOfSelectedOptionsInGrid(
+ rowIndex: 1,
+ matcher: findsNothing,
+ );
+
+ // create a new field for testing
+ await tester.createField(FieldType.RichText, name: 'Test');
+
+ // edit the first 2 rows
+ await tester.editCell(
+ rowIndex: 0,
+ fieldType: FieldType.RichText,
+ input: "E,F",
+ cellIndex: 1,
+ );
+ await tester.editCell(
+ rowIndex: 1,
+ fieldType: FieldType.RichText,
+ input: "G",
+ cellIndex: 1,
+ );
+
+ await tester.changeFieldTypeOfFieldWithName(
+ 'Test',
+ FieldType.MultiSelect,
+ );
+ tester.assertMultiSelectOption(contents: ['E', 'F'], rowIndex: 0);
+ tester.assertMultiSelectOption(contents: ['G'], rowIndex: 1);
+
+ await tester.tapCellInGrid(
+ rowIndex: 2,
+ fieldType: FieldType.MultiSelect,
+ );
+ await tester.selectOption(name: 'G');
+ await tester.createOption(name: 'H');
+ await tester.dismissCellEditor();
+ tester.findSelectOptionWithNameInGrid(name: 'A', rowIndex: 0);
+ tester.assertMultiSelectOption(contents: ['G', 'H'], rowIndex: 2);
+
+ await tester.changeFieldTypeOfFieldWithName(
+ 'Test',
+ FieldType.RichText,
+ );
+ tester.assertCellContent(
+ rowIndex: 2,
+ fieldType: FieldType.RichText,
+ content: "G,H",
+ cellIndex: 1,
+ );
+ await tester.changeFieldTypeOfFieldWithName(
+ 'Test',
+ FieldType.MultiSelect,
+ );
+
+ tester.assertMultiSelectOption(contents: ['E', 'F'], rowIndex: 0);
+ tester.assertMultiSelectOption(contents: ['G'], rowIndex: 1);
+ tester.assertMultiSelectOption(contents: ['G', 'H'], rowIndex: 2);
+ });
+
+ testWidgets('date time transform', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
+ await tester.scrollToRight(find.byType(GridPage));
+
+ // create a date field
+ await tester.createField(FieldType.DateTime);
+
+ // edit the first date cell
+ await tester.tapCellInGrid(rowIndex: 0, fieldType: FieldType.DateTime);
+ final now = DateTime.now();
+ await tester.toggleIncludeTime();
+ await tester.selectDay(content: now.day);
+
+ await tester.dismissCellEditor();
+
+ tester.assertCellContent(
+ rowIndex: 0,
+ fieldType: FieldType.DateTime,
+ content: DateFormat('MMM dd, y HH:mm').format(now),
+ );
+
+ await tester.changeFieldTypeOfFieldWithName(
+ 'Date',
+ FieldType.RichText,
+ );
+ tester.assertCellContent(
+ rowIndex: 0,
+ fieldType: FieldType.RichText,
+ content: DateFormat('MMM dd, y HH:mm').format(now),
+ cellIndex: 1,
+ );
+
+ await tester.editCell(
+ rowIndex: 1,
+ fieldType: FieldType.RichText,
+ input: "Oct 5, 2024",
+ cellIndex: 1,
+ );
+ tester.assertCellContent(
+ rowIndex: 1,
+ fieldType: FieldType.RichText,
+ content: "Oct 5, 2024",
+ cellIndex: 1,
+ );
+
+ await tester.changeFieldTypeOfFieldWithName(
+ 'Date',
+ FieldType.DateTime,
+ );
+ tester.assertCellContent(
+ rowIndex: 0,
+ fieldType: FieldType.DateTime,
+ content: DateFormat('MMM dd, y').format(now),
+ );
+ tester.assertCellContent(
+ rowIndex: 1,
+ fieldType: FieldType.DateTime,
+ content: "Oct 05, 2024",
+ );
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_filter_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_filter_test.dart
index 8acc80a0dd..8e79445503 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/database/database_filter_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_filter_test.dart
@@ -1,10 +1,14 @@
+import 'package:appflowy/plugins/database/application/field/filter_entities.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/checkbox.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/text.dart';
-import 'package:appflowy_backend/protobuf/flowy-database2/field_entities.pbenum.dart';
+import 'package:appflowy/plugins/database/widgets/row/row_detail.dart';
+import 'package:appflowy_backend/protobuf/flowy-database2/protobuf.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/protobuf.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
import '../../shared/database_test_op.dart';
+import '../../shared/util.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
@@ -104,7 +108,7 @@ void main() {
await tester.tapOptionFilterWithName('s4');
// The row with 's4' should be shown.
- tester.assertNumberOfRowsInGridPage(1);
+ tester.assertNumberOfRowsInGridPage(2);
await tester.pumpAndSettle();
});
@@ -138,5 +142,83 @@ void main() {
await tester.pumpAndSettle();
});
+
+ testWidgets('add date filter', (tester) async {
+ await tester.openTestDatabase(v020GridFileName);
+
+ // create a filter
+ await tester.tapDatabaseFilterButton();
+ await tester.tapCreateFilterByFieldType(FieldType.DateTime, 'date');
+
+ // By default, the condition of date filter is current day and time
+ tester.assertNumberOfRowsInGridPage(0);
+
+ await tester.tapFilterButtonInGrid('date');
+ await tester.changeDateFilterCondition(DateTimeFilterCondition.before);
+ tester.assertNumberOfRowsInGridPage(7);
+
+ await tester.changeDateFilterCondition(DateTimeFilterCondition.isEmpty);
+ tester.assertNumberOfRowsInGridPage(3);
+
+ await tester.pumpAndSettle();
+ });
+
+ testWidgets('add timestamp filter', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
+
+ await tester.createField(
+ FieldType.CreatedTime,
+ name: 'Created at',
+ );
+
+ // create a filter
+ await tester.tapDatabaseFilterButton();
+ await tester.tapCreateFilterByFieldType(
+ FieldType.CreatedTime,
+ 'Created at',
+ );
+ await tester.pumpAndSettle();
+
+ tester.assertNumberOfRowsInGridPage(3);
+
+ await tester.tapFilterButtonInGrid('Created at');
+ await tester.changeDateFilterCondition(DateTimeFilterCondition.before);
+ tester.assertNumberOfRowsInGridPage(0);
+
+ await tester.pumpAndSettle();
+ });
+
+ testWidgets('create new row when filters don\'t autofill', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
+
+ // create a filter
+ await tester.tapDatabaseFilterButton();
+ await tester.tapCreateFilterByFieldType(
+ FieldType.RichText,
+ 'Name',
+ );
+ tester.assertNumberOfRowsInGridPage(3);
+
+ await tester.tapCreateRowButtonInGrid();
+ tester.assertNumberOfRowsInGridPage(4);
+
+ await tester.tapFilterButtonInGrid('Name');
+ await tester
+ .changeTextFilterCondition(TextFilterConditionPB.TextIsNotEmpty);
+ await tester.dismissCellEditor();
+ tester.assertNumberOfRowsInGridPage(0);
+
+ await tester.tapCreateRowButtonInGrid();
+ tester.assertNumberOfRowsInGridPage(0);
+ expect(find.byType(RowDetailPage), findsOneWidget);
+
+ await tester.pumpAndSettle();
+ });
});
}
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_media_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_media_test.dart
index 43f6f1f4a5..cb24a949bb 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/database/database_media_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_media_test.dart
@@ -21,7 +21,6 @@ import 'package:path_provider/path_provider.dart';
import '../../shared/database_test_op.dart';
import '../../shared/mock/mock_file_picker.dart';
import '../../shared/util.dart';
-import '../board/board_hide_groups_test.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
@@ -69,10 +68,7 @@ void main() {
await tester.pumpAndSettle();
// Tap on the upload interaction
- await tester.tapButtonWithName(
- LocaleKeys.document_plugins_file_fileUploadHint.tr(),
- );
- await tester.pumpAndSettle();
+ await tester.tapFileUploadHint();
// Expect one file
expect(find.byType(RenderMedia), findsOneWidget);
@@ -85,9 +81,7 @@ void main() {
await tester.pumpAndSettle();
// Tap on the upload interaction
- await tester.tapButtonWithName(
- LocaleKeys.document_plugins_file_fileUploadHint.tr(),
- );
+ await tester.tapFileUploadHint();
await tester.pumpAndSettle();
// Expect two files
@@ -139,10 +133,7 @@ void main() {
await tester.pumpAndSettle();
// Tap on the upload interaction
- await tester.tapButtonWithName(
- LocaleKeys.document_plugins_file_fileUploadHint.tr(),
- );
- await tester.pumpAndSettle();
+ await tester.tapFileUploadHint();
// Expect two files
expect(find.byType(RenderMedia), findsNWidgets(2));
@@ -193,10 +184,7 @@ void main() {
await tester.pumpAndSettle();
// Tap on the upload interaction
- await tester.tapButtonWithName(
- LocaleKeys.document_plugins_file_fileUploadHint.tr(),
- );
- await tester.pumpAndSettle();
+ await tester.tapFileUploadHint();
// Expect two files
expect(find.byType(RenderMedia), findsNWidgets(2));
@@ -230,7 +218,7 @@ void main() {
await Future.wait([firstFile.delete(), secondFile.delete()]);
});
- testWidgets('hide file names', (tester) async {
+ testWidgets('show file names', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
@@ -272,10 +260,7 @@ void main() {
await tester.pumpAndSettle();
// Tap on the upload interaction
- await tester.tapButtonWithName(
- LocaleKeys.document_plugins_file_fileUploadHint.tr(),
- );
- await tester.pumpAndSettle();
+ await tester.tapFileUploadHint();
// Expect two files
expect(find.byType(RenderMedia), findsNWidgets(2));
@@ -283,28 +268,28 @@ void main() {
await tester.dismissCellEditor();
await tester.pumpAndSettle();
- // Open first row in row detail view then toggle hide file names
+ // Open first row in row detail view then toggle show file names
await tester.openFirstRowDetailPage();
await tester.pumpAndSettle();
+ // Expect file names to not be shown (hidden)
+ expect(find.text('sample.jpeg'), findsNothing);
+ expect(find.text('sample.gif'), findsNothing);
+
+ await tester.tapGridFieldWithNameInRowDetailPage('Type');
+ await tester.pumpAndSettle();
+
+ // Toggle show file names
+ await tester.tap(find.byType(Toggle));
+ await tester.pumpAndSettle();
+
// Expect file names to be shown
expect(find.text('sample.jpeg'), findsOneWidget);
expect(find.text('sample.gif'), findsOneWidget);
- await tester.tapGridFieldWithNameInRowDetailPage('Type');
- await tester.pumpAndSettle();
-
- // Toggle hide file names
- await tester.tap(find.byType(Toggle));
- await tester.pumpAndSettle();
-
await tester.dismissRowDetailPage();
await tester.pumpAndSettle();
- // Expect file names to be hidden
- expect(find.text('sample.jpeg'), findsNothing);
- expect(find.text('sample.gif'), findsNothing);
-
// Remove the temp files
await Future.wait([firstFile.delete(), secondFile.delete()]);
});
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 4aee8e1feb..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
@@ -1,19 +1,16 @@
import 'dart:io';
-import 'package:flutter/services.dart';
-
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/header/document_header_node_widget.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';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
import 'package:path/path.dart' as p;
@@ -27,63 +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.tapButtonWithName(
- LocaleKeys.document_plugins_file_fileUploadHint.tr(),
- );
-
- // 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 25a816ad9d..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,15 +1,19 @@
-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/style_widget/text.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';
@@ -18,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 {
@@ -73,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 {
@@ -335,5 +364,145 @@ void main() {
tester.assertNumberOfRowsInGridPage(4);
});
+
+ testWidgets('edit checklist cell', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
+
+ const fieldType = FieldType.Checklist;
+ await tester.createField(fieldType);
+
+ await tester.openFirstRowDetailPage();
+ await tester.hoverOnWidget(
+ find.byType(ChecklistRowDetailCell),
+ onHover: () async {
+ await tester.tapButton(find.byType(ChecklistItemControl));
+ },
+ );
+
+ tester.assertPhantomChecklistItemAtIndex(index: 0);
+ await tester.enterText(find.byType(PhantomChecklistItem), 'task 1');
+ await tester.pumpAndSettle();
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ await tester.pumpAndSettle(const Duration(milliseconds: 500));
+
+ tester.assertChecklistTaskInEditor(
+ index: 0,
+ name: "task 1",
+ isChecked: false,
+ );
+ tester.assertPhantomChecklistItemAtIndex(index: 1);
+ tester.assertPhantomChecklistItemContent("");
+
+ await tester.enterText(find.byType(PhantomChecklistItem), 'task 2');
+ await tester.pumpAndSettle();
+ await tester.hoverOnWidget(
+ find.byType(ChecklistRowDetailCell),
+ onHover: () async {
+ await tester.tapButton(find.byType(ChecklistItemControl));
+ },
+ );
+
+ tester.assertChecklistTaskInEditor(
+ index: 1,
+ name: "task 2",
+ isChecked: false,
+ );
+ tester.assertPhantomChecklistItemAtIndex(index: 2);
+ tester.assertPhantomChecklistItemContent("");
+
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ await tester.pumpAndSettle();
+ expect(find.byType(PhantomChecklistItem), findsNothing);
+
+ await tester.renameChecklistTask(index: 0, name: "task -1", enter: false);
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+
+ tester.assertChecklistTaskInEditor(
+ index: 0,
+ name: "task -1",
+ isChecked: false,
+ );
+ tester.assertPhantomChecklistItemAtIndex(index: 1);
+
+ await tester.enterText(find.byType(PhantomChecklistItem), 'task 0');
+ await tester.pumpAndSettle();
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ await tester.pumpAndSettle(const Duration(milliseconds: 500));
+
+ tester.assertPhantomChecklistItemAtIndex(index: 2);
+
+ await tester.checkChecklistTask(index: 1);
+ expect(find.byType(PhantomChecklistItem), findsNothing);
+ expect(find.byType(ChecklistItem), findsNWidgets(3));
+
+ tester.assertChecklistTaskInEditor(
+ index: 0,
+ name: "task -1",
+ isChecked: false,
+ );
+ tester.assertChecklistTaskInEditor(
+ index: 1,
+ name: "task 0",
+ isChecked: true,
+ );
+ tester.assertChecklistTaskInEditor(
+ index: 2,
+ name: "task 2",
+ isChecked: false,
+ );
+
+ await tester.tapButton(
+ find.descendant(
+ of: find.byType(ProgressAndHideCompleteButton),
+ matching: find.byType(FlowyIconButton),
+ ),
+ );
+ expect(find.byType(ChecklistItem), findsNWidgets(2));
+
+ tester.assertChecklistTaskInEditor(
+ index: 0,
+ name: "task -1",
+ isChecked: false,
+ );
+ tester.assertChecklistTaskInEditor(
+ index: 1,
+ name: "task 2",
+ isChecked: false,
+ );
+
+ await tester.renameChecklistTask(index: 1, name: "task 3", enter: false);
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+
+ await tester.renameChecklistTask(index: 0, name: "task 1", enter: false);
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+
+ await tester.enterText(find.byType(PhantomChecklistItem), 'task 2');
+ await tester.pumpAndSettle();
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ await tester.pumpAndSettle(const Duration(milliseconds: 500));
+
+ tester.assertChecklistTaskInEditor(
+ index: 0,
+ name: "task 1",
+ isChecked: false,
+ );
+ tester.assertChecklistTaskInEditor(
+ index: 1,
+ name: "task 2",
+ isChecked: false,
+ );
+ tester.assertChecklistTaskInEditor(
+ index: 2,
+ name: "task 3",
+ isChecked: false,
+ );
+ tester.assertPhantomChecklistItemAtIndex(index: 2);
+
+ await tester.checkChecklistTask(index: 1);
+ expect(find.byType(ChecklistItem), findsNWidgets(2));
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_share_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_share_test.dart
index a04948b35e..2beb74a5f2 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/database/database_share_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_share_test.dart
@@ -115,7 +115,7 @@ void main() {
[],
];
for (final (index, contents) in multiSelectCells.indexed) {
- await tester.assertMultiSelectOption(
+ tester.assertMultiSelectOption(
rowIndex: index,
contents: contents,
);
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_sort_test.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_sort_test.dart
index fcd71d1bc1..e09d8718be 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/database/database_sort_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_sort_test.dart
@@ -1,3 +1,4 @@
+import 'package:appflowy/plugins/database/grid/presentation/widgets/sort/sort_editor.dart';
import 'package:appflowy_backend/protobuf/flowy-database2/field_entities.pbenum.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -7,8 +8,8 @@ import '../../shared/database_test_op.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- group('grid', () {
- testWidgets('add text sort', (tester) async {
+ group('grid sort:', () {
+ testWidgets('text sort', (tester) async {
await tester.openTestDatabase(v020GridFileName);
// create a sort
await tester.tapDatabaseSortButton();
@@ -37,7 +38,7 @@ void main() {
// open the sort menu and select order by descending
await tester.tapSortMenuInSettingBar();
- await tester.tapSortButtonByName('Name');
+ await tester.tapEditSortConditionButtonByFieldName('Name');
await tester.tapSortByDescending();
for (final (index, content) in [
'E',
@@ -84,7 +85,7 @@ void main() {
await tester.pumpAndSettle();
});
- testWidgets('add checkbox sort', (tester) async {
+ testWidgets('checkbox', (tester) async {
await tester.openTestDatabase(v020GridFileName);
// create a sort
await tester.tapDatabaseSortButton();
@@ -111,7 +112,7 @@ void main() {
// open the sort menu and select order by descending
await tester.tapSortMenuInSettingBar();
- await tester.tapSortButtonByName('Done');
+ await tester.tapEditSortConditionButtonByFieldName('Done');
await tester.tapSortByDescending();
for (final (index, content) in [
true,
@@ -134,7 +135,7 @@ void main() {
await tester.pumpAndSettle();
});
- testWidgets('add number sort', (tester) async {
+ testWidgets('number', (tester) async {
await tester.openTestDatabase(v020GridFileName);
// create a sort
await tester.tapDatabaseSortButton();
@@ -162,7 +163,7 @@ void main() {
// open the sort menu and select order by descending
await tester.tapSortMenuInSettingBar();
- await tester.tapSortButtonByName('number');
+ await tester.tapEditSortConditionButtonByFieldName('number');
await tester.tapSortByDescending();
for (final (index, content) in [
'12',
@@ -186,7 +187,7 @@ void main() {
await tester.pumpAndSettle();
});
- testWidgets('add checkbox and number sort', (tester) async {
+ testWidgets('checkbox and number', (tester) async {
await tester.openTestDatabase(v020GridFileName);
// create a sort
await tester.tapDatabaseSortButton();
@@ -194,7 +195,7 @@ void main() {
// open the sort menu and sort checkbox by descending
await tester.tapSortMenuInSettingBar();
- await tester.tapSortButtonByName('Done');
+ await tester.tapEditSortConditionButtonByFieldName('Done');
await tester.tapSortByDescending();
for (final (index, content) in [
true,
@@ -220,7 +221,7 @@ void main() {
FieldType.Number,
'number',
);
- await tester.tapSortButtonByName('number');
+ await tester.tapEditSortConditionButtonByFieldName('number');
await tester.tapSortByDescending();
// check checkbox cell order
@@ -273,7 +274,7 @@ void main() {
// open the sort menu and sort checkbox by descending
await tester.tapSortMenuInSettingBar();
- await tester.tapSortButtonByName('Done');
+ await tester.tapEditSortConditionButtonByFieldName('Done');
await tester.tapSortByDescending();
// add another sort, this time by number descending
@@ -282,7 +283,7 @@ void main() {
FieldType.Number,
'number',
);
- await tester.tapSortButtonByName('number');
+ await tester.tapEditSortConditionButtonByFieldName('number');
await tester.tapSortByDescending();
// check checkbox cell order
@@ -370,5 +371,101 @@ void main() {
);
}
});
+
+ testWidgets('edit field', (tester) async {
+ await tester.openTestDatabase(v020GridFileName);
+
+ // create a number sort
+ await tester.tapDatabaseSortButton();
+ await tester.tapCreateSortByFieldType(FieldType.Number, 'number');
+
+ // check the number cell order
+ for (final (index, content) in [
+ '-2',
+ '-1',
+ '0.1',
+ '0.2',
+ '1',
+ '2',
+ '10',
+ '11',
+ '12',
+ '',
+ ].indexed) {
+ tester.assertCellContent(
+ rowIndex: index,
+ fieldType: FieldType.Number,
+ content: content,
+ );
+ }
+
+ final textCells = [
+ 'B',
+ 'A',
+ 'C',
+ 'D',
+ 'E',
+ '',
+ '',
+ '',
+ '',
+ '',
+ ];
+ for (final (index, content) in textCells.indexed) {
+ tester.assertCellContent(
+ rowIndex: index,
+ fieldType: FieldType.RichText,
+ content: content,
+ );
+ }
+
+ // edit the name of the number field
+ await tester.tapGridFieldWithName('number');
+
+ await tester.renameField('hello world');
+ await tester.dismissFieldEditor();
+
+ await tester.tapGridFieldWithName('hello world');
+ await tester.dismissFieldEditor();
+
+ // expect name to be changed as well
+ await tester.tapSortMenuInSettingBar();
+ final sortItem = find.ancestor(
+ of: find.text('hello world'),
+ matching: find.byType(DatabaseSortItem),
+ );
+ expect(sortItem, findsOneWidget);
+
+ // change the field type of the field to checkbox
+ await tester.tapGridFieldWithName('hello world');
+ await tester.changeFieldTypeOfFieldWithName(
+ 'hello world',
+ FieldType.Checkbox,
+ );
+
+ // expect name to be changed as well
+ await tester.tapSortMenuInSettingBar();
+ expect(sortItem, findsOneWidget);
+
+ final newTextCells = [
+ 'A',
+ 'B',
+ 'C',
+ 'D',
+ 'E',
+ '',
+ '',
+ '',
+ '',
+ '',
+ ];
+ for (final (index, content) in newTextCells.indexed) {
+ tester.assertCellContent(
+ rowIndex: index,
+ fieldType: FieldType.RichText,
+ content: content,
+ );
+ }
+ });
});
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_test_runner_1.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_test_runner_1.dart
new file mode 100644
index 0000000000..3a5854bc1b
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_test_runner_1.dart
@@ -0,0 +1,20 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'database_cell_test.dart' as database_cell_test;
+import 'database_field_settings_test.dart' as database_field_settings_test;
+import 'database_field_test.dart' as database_field_test;
+import 'database_row_page_test.dart' as database_row_page_test;
+import 'database_setting_test.dart' as database_setting_test;
+import 'database_share_test.dart' as database_share_test;
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ database_cell_test.main();
+ database_field_test.main();
+ database_field_settings_test.main();
+ database_share_test.main();
+ database_row_page_test.main();
+ database_setting_test.main();
+ // DON'T add more tests here.
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/database/database_test_runner_2.dart b/frontend/appflowy_flutter/integration_test/desktop/database/database_test_runner_2.dart
new file mode 100644
index 0000000000..26b64af495
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/database/database_test_runner_2.dart
@@ -0,0 +1,22 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'database_calendar_test.dart' as database_calendar_test;
+import 'database_filter_test.dart' as database_filter_test;
+import 'database_media_test.dart' as database_media_test;
+import 'database_row_cover_test.dart' as database_row_cover_test;
+import 'database_share_test.dart' as database_share_test;
+import 'database_sort_test.dart' as database_sort_test;
+import 'database_view_test.dart' as database_view_test;
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ database_filter_test.main();
+ database_sort_test.main();
+ database_view_test.main();
+ database_calendar_test.main();
+ database_media_test.main();
+ database_row_cover_test.main();
+ database_share_test.main();
+ // DON'T add more tests here.
+}
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_app_lifecycle_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_app_lifecycle_test.dart
new file mode 100644
index 0000000000..fdde8bbeb8
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_app_lifecycle_test.dart
@@ -0,0 +1,72 @@
+import 'dart:ui';
+
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ final binding = IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('Editor AppLifeCycle tests', () {
+ testWidgets(
+ 'Selection is added back after pausing AppFlowy',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ final selection = Selection.single(path: [4], startOffset: 0);
+ await tester.editor.updateSelection(selection);
+
+ binding.handleAppLifecycleStateChanged(AppLifecycleState.inactive);
+ expect(tester.editor.getCurrentEditorState().selection, null);
+
+ binding.handleAppLifecycleStateChanged(AppLifecycleState.resumed);
+ await tester.pumpAndSettle();
+
+ expect(tester.editor.getCurrentEditorState().selection, selection);
+ },
+ );
+
+ testWidgets(
+ 'Null selection is retained after pausing AppFlowy',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ final selection = Selection.single(path: [4], startOffset: 0);
+ await tester.editor.updateSelection(selection);
+ await tester.editor.updateSelection(null);
+
+ binding.handleAppLifecycleStateChanged(AppLifecycleState.inactive);
+ expect(tester.editor.getCurrentEditorState().selection, null);
+
+ binding.handleAppLifecycleStateChanged(AppLifecycleState.resumed);
+ await tester.pumpAndSettle();
+
+ expect(tester.editor.getCurrentEditorState().selection, null);
+ },
+ );
+
+ testWidgets(
+ 'Non-collapsed selection is retained after pausing AppFlowy',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ final selection = Selection(
+ start: Position(path: [3]),
+ end: Position(path: [3], offset: 8),
+ );
+ await tester.editor.updateSelection(selection);
+
+ binding.handleAppLifecycleStateChanged(AppLifecycleState.inactive);
+ binding.handleAppLifecycleStateChanged(AppLifecycleState.resumed);
+ await tester.pumpAndSettle();
+
+ expect(tester.editor.getCurrentEditorState().selection, selection);
+ },
+ );
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_block_option_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_block_option_test.dart
new file mode 100644
index 0000000000..76e5dfcb6c
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_block_option_test.dart
@@ -0,0 +1,47 @@
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('Block option interaction tests', () {
+ testWidgets('has correct block selection on tap option button',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // We edit the document by entering some characters, to ensure the document has focus
+ await tester.editor.updateSelection(
+ Selection.collapsed(Position(path: [2])),
+ );
+
+ // Insert character 'a' three times - easy to identify
+ await tester.ime.insertText('aaa');
+ await tester.pumpAndSettle();
+
+ final editorState = tester.editor.getCurrentEditorState();
+ final node = editorState.getNodeAtPath([2]);
+ expect(node?.delta?.toPlainText(), startsWith('aaa'));
+
+ final multiSelection = Selection(
+ start: Position(path: [2], offset: 3),
+ end: Position(path: [4], offset: 40),
+ );
+
+ // Select multiple items
+ await tester.editor.updateSelection(multiSelection);
+ await tester.pumpAndSettle();
+
+ // Press the block option menu
+ await tester.editor.hoverAndClickOptionMenuButton([2]);
+ await tester.pumpAndSettle();
+
+ // Expect the selection to be Block type and not have changed
+ expect(editorState.selectionType, SelectionType.block);
+ expect(editorState.selection, multiSelection);
+ });
+ });
+}
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_codeblock_paste_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_codeblock_paste_test.dart
index 0fb8cc90e8..a498086952 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_codeblock_paste_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_codeblock_paste_test.dart
@@ -13,13 +13,15 @@ import '../../shared/util.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- group('paste in codeblock', () {
+ group('paste in codeblock:', () {
testWidgets('paste multiple lines in codeblock', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
// create a new document
- await tester.createNewPageWithNameUnderParent();
+ await tester.createNewPageWithNameUnderParent(name: 'Test Document');
+ // focus on the editor
+ await tester.tapButton(find.byType(AppFlowyEditor));
// mock the clipboard
const lines = 3;
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 d178f584b0..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,29 +1,40 @@
+import 'dart:async';
import 'dart:io';
+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/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';
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',
);
}
@@ -163,173 +174,363 @@ 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 {
- // It's not supported yet.
- // 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, 2);
- // 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"},
+ ],
+ },
+ ),
+ ],
);
+
+ // 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.type, ParagraphBlockKeys.type);
- expect(node.delta!.toJson(), [
- {
- 'insert': text,
- 'attributes': {'href': url},
- }
- ]);
+ final node = editorState.getNodeAtPath([0]);
+ expect(node?.delta?.toPlainText(), 'bullet test');
+ expect(node?.type, BulletedListBlockKeys.type);
},
);
- },
- );
+ });
- // 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 =
- '''
''';
+ 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(
- html: html,
- image: ('png', bytes),
- (editorState) {
+ 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, ImageBlockKeys.type);
+ 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 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 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 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);
+ }
+ },
+ );
});
});
-
- 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) {
- // 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);
- });
- },
- );
}
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();
@@ -337,6 +538,8 @@ extension on WidgetTester {
// create a new document
await createNewPageWithNameUnderParent();
+ // tap the editor
+ await tapButton(find.byType(AppFlowyEditor));
await beforeTest?.call(editor.getCurrentEditorState());
@@ -354,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 cf45afc828..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
@@ -1,6 +1,4 @@
-import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy_editor/appflowy_editor.dart';
-import 'package:easy_localization/easy_localization.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -15,14 +13,15 @@ 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
- await tester.createNewPageWithNameUnderParent();
+ const pageName = 'Test Document';
+ await tester.createNewPageWithNameUnderParent(name: pageName);
// expect to see a new document
- tester.expectToSeePageName(
- LocaleKeys.menuAppHeader_defaultNewPageName.tr(),
- );
+ tester.expectToSeePageName(pageName);
// and with one paragraph block
expect(find.byType(ParagraphBlockComponentWidget), findsOneWidget);
});
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_customer_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_customer_test.dart
new file mode 100644
index 0000000000..5cbb133f9d
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_customer_test.dart
@@ -0,0 +1,61 @@
+import 'package:appflowy_editor/appflowy_editor.dart';
+import 'package:appflowy_editor_plugins/appflowy_editor_plugins.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('customer:', () {
+ testWidgets('backtick issue - inline code', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ const pageName = 'backtick issue';
+ await tester.createNewPageWithNameUnderParent(name: pageName);
+
+ // focus on the editor
+ await tester.tap(find.byType(AppFlowyEditor));
+ // input backtick
+ const text = '`Hello` AppFlowy';
+
+ for (var i = 0; i < text.length; i++) {
+ await tester.ime.insertCharacter(text[i]);
+ }
+
+ final node = tester.editor.getNodeAtPath([0]);
+ expect(
+ node.delta?.toJson(),
+ equals([
+ {
+ "insert": "Hello",
+ "attributes": {"code": true},
+ },
+ {"insert": " AppFlowy"},
+ ]),
+ );
+ });
+
+ testWidgets('backtick issue - inline code', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ const pageName = 'backtick issue';
+ await tester.createNewPageWithNameUnderParent(name: pageName);
+
+ // focus on the editor
+ await tester.tap(find.byType(AppFlowyEditor));
+ // input backtick
+ const text = '```';
+
+ for (var i = 0; i < text.length; i++) {
+ await tester.ime.insertCharacter(text[i]);
+ }
+
+ final node = tester.editor.getNodeAtPath([0]);
+ expect(node.type, equals(CodeBlockKeys.type));
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_deletion_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_deletion_test.dart
new file mode 100644
index 0000000000..f38138ce8a
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_deletion_test.dart
@@ -0,0 +1,84 @@
+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/widgets/inline_actions_handler.dart';
+import 'package:appflowy/workspace/presentation/widgets/view_title_bar.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+import 'document_inline_page_reference_test.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('Document deletion', () {
+ testWidgets('Trash breadcrumb', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // This test shares behavior with the inline page reference test, thus
+ // we utilize the same helper functions there.
+ final name = await createDocumentToReference(tester);
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.pumpAndSettle();
+
+ await triggerReferenceDocumentBySlashMenu(tester);
+
+ // Search for prefix of document
+ await enterDocumentText(tester);
+
+ // Select result
+ final optionFinder = find.descendant(
+ of: find.byType(InlineActionsHandler),
+ matching: find.text(name),
+ );
+
+ await tester.tap(optionFinder);
+ await tester.pumpAndSettle();
+
+ final mentionBlock = find.byType(MentionPageBlock);
+ expect(mentionBlock, findsOneWidget);
+
+ // Delete the page
+ await tester.hoverOnPageName(
+ name,
+ onHover: () async => tester.tapDeletePageButton(),
+ );
+ await tester.pumpAndSettle();
+
+ // Navigate to the deleted page from the inline mention
+ await tester.tap(mentionBlock);
+ await tester.pumpUntilFound(find.byType(TrashBreadcrumb));
+
+ expect(find.byType(TrashBreadcrumb), findsOneWidget);
+
+ // Navigate using the trash breadcrumb
+ await tester.tap(
+ find.descendant(
+ of: find.byType(TrashBreadcrumb),
+ matching: find.text(
+ LocaleKeys.trash_text.tr(),
+ ),
+ ),
+ );
+ await tester.pumpUntilFound(find.text(LocaleKeys.trash_restoreAll.tr()));
+
+ // Restore all
+ await tester.tap(find.text(LocaleKeys.trash_restoreAll.tr()));
+ await tester.pumpAndSettle();
+ await tester.tap(find.text(LocaleKeys.trash_restore.tr()));
+ await tester.pumpAndSettle();
+
+ // Navigate back to the document
+ await tester.openPage('Getting started');
+ await tester.pumpAndSettle();
+
+ await tester.tap(mentionBlock);
+ await tester.pumpAndSettle();
+
+ expect(find.byType(TrashBreadcrumb), findsNothing);
+ });
+ });
+}
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
new file mode 100644
index 0000000000..30e115774a
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_inline_sub_page_test.dart
@@ -0,0 +1,382 @@
+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:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/keyboard.dart';
+import '../../shared/util.dart';
+
+const _firstDocName = "Inline Sub Page Mention";
+const _createdPageName = "hi world";
+
+// Test cases that are covered in this file:
+// - [x] Insert sub page mention from action menu (+)
+// - [x] Delete sub page mention from editor
+// - [x] Delete page from sidebar
+// - [x] Delete page from sidebar and then trash
+// - [x] Undo delete sub page mention
+// - [x] Cut+paste in same document
+// - [x] Cut+paste in different document
+// - [x] Cut+paste in same document and then paste again in same document
+// - [x] Turn paragraph with sub page mention into a heading
+// - [x] Turn heading with sub page mention into a paragraph
+// - [x] Duplicate a Block containing two sub page mentions
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('document inline sub-page mention tests:', () {
+ testWidgets('Insert (& delete) a sub page mention from action menu',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createOpenRenameDocumentUnderParent(name: _firstDocName);
+
+ await tester.insertInlineSubPageFromPlusMenu();
+
+ await tester.expandOrCollapsePage(
+ pageName: _firstDocName,
+ layout: ViewLayoutPB.Document,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNWidgets(2));
+ expect(find.byType(MentionSubPageBlock), findsOneWidget);
+
+ // Delete from editor
+ await tester.editor.updateSelection(
+ Selection.collapsed(Position(path: [0], offset: 1)),
+ );
+
+ await tester.simulateKeyEvent(LogicalKeyboardKey.backspace);
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNothing);
+ expect(find.byType(MentionSubPageBlock), findsNothing);
+
+ // Undo
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyZ,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNWidgets(2));
+ expect(find.byType(MentionSubPageBlock), findsOneWidget);
+
+ // Move to trash (delete from sidebar)
+ await tester.rightClickOnPageName(_createdPageName);
+ await tester.tapButtonWithName(ViewMoreActionType.delete.name);
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsOneWidget);
+ expect(find.byType(MentionSubPageBlock), findsOneWidget);
+ expect(
+ find.text(LocaleKeys.document_mention_trashHint.tr()),
+ findsOneWidget,
+ );
+
+ // Delete from trash
+ await tester.tapTrashButton();
+ await tester.pumpAndSettle();
+
+ await tester.tap(find.text(LocaleKeys.trash_deleteAll.tr()));
+ await tester.pumpAndSettle();
+
+ await tester.tap(find.text(LocaleKeys.button_delete.tr()));
+ await tester.pumpAndSettle();
+
+ await tester.openPage(_firstDocName);
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNothing);
+ expect(find.byType(MentionSubPageBlock), findsOneWidget);
+ expect(
+ find.text(LocaleKeys.document_mention_deletedPage.tr()),
+ findsOneWidget,
+ );
+ });
+
+ testWidgets(
+ 'Cut+paste in same document and cut+paste in different document',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createOpenRenameDocumentUnderParent(name: _firstDocName);
+
+ await tester.insertInlineSubPageFromPlusMenu();
+
+ await tester.expandOrCollapsePage(
+ pageName: _firstDocName,
+ layout: ViewLayoutPB.Document,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNWidgets(2));
+ expect(find.byType(MentionSubPageBlock), findsOneWidget);
+
+ // Cut from editor
+ await tester.editor.updateSelection(
+ Selection.collapsed(Position(path: [0], offset: 1)),
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyX,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNothing);
+ expect(find.byType(MentionSubPageBlock), findsNothing);
+
+ // Paste in same document
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNWidgets(2));
+ expect(find.byType(MentionSubPageBlock), findsOneWidget);
+
+ // Cut again
+ await tester.editor.updateSelection(
+ Selection.collapsed(Position(path: [0], offset: 1)),
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyX,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ // Create another document
+ const anotherDocName = "Another Document";
+ await tester.createOpenRenameDocumentUnderParent(
+ name: anotherDocName,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNothing);
+ expect(find.byType(MentionSubPageBlock), findsNothing);
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.pumpAndSettle();
+
+ // Paste in document
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpUntilFound(find.byType(MentionSubPageBlock));
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsOneWidget);
+
+ await tester.expandOrCollapsePage(
+ pageName: anotherDocName,
+ layout: ViewLayoutPB.Document,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNWidgets(2));
+ });
+ testWidgets(
+ 'Cut+paste in same docuemnt and then paste again in same document',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createOpenRenameDocumentUnderParent(name: _firstDocName);
+
+ await tester.insertInlineSubPageFromPlusMenu();
+
+ await tester.expandOrCollapsePage(
+ pageName: _firstDocName,
+ layout: ViewLayoutPB.Document,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNWidgets(2));
+ expect(find.byType(MentionSubPageBlock), findsOneWidget);
+
+ // Cut from editor
+ await tester.editor.updateSelection(
+ Selection.collapsed(Position(path: [0], offset: 1)),
+ );
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyX,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNothing);
+ expect(find.byType(MentionSubPageBlock), findsNothing);
+
+ // Paste in same document
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNWidgets(2));
+ expect(find.byType(MentionSubPageBlock), findsOneWidget);
+
+ // Paste again
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle(const Duration(seconds: 2));
+
+ expect(find.text(_createdPageName), findsNWidgets(2));
+ expect(find.byType(MentionSubPageBlock), findsNWidgets(2));
+ expect(find.text('$_createdPageName (copy)'), findsNWidgets(2));
+ });
+
+ testWidgets('Turn into w/ sub page mentions', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createOpenRenameDocumentUnderParent(name: _firstDocName);
+
+ await tester.insertInlineSubPageFromPlusMenu();
+
+ await tester.expandOrCollapsePage(
+ pageName: _firstDocName,
+ layout: ViewLayoutPB.Document,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNWidgets(2));
+ expect(find.byType(MentionSubPageBlock), findsOneWidget);
+
+ final headingText = LocaleKeys.document_slashMenu_name_heading1.tr();
+ final paragraphText = LocaleKeys.document_slashMenu_name_text.tr();
+
+ // Turn into heading
+ await tester.editor.openTurnIntoMenu([0]);
+ await tester.tapButton(find.findTextInFlowyText(headingText));
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNWidgets(2));
+ expect(find.byType(MentionSubPageBlock), findsOneWidget);
+
+ // Turn into paragraph
+ await tester.editor.openTurnIntoMenu([0]);
+ await tester.tapButton(find.findTextInFlowyText(paragraphText));
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsNWidgets(2));
+ expect(find.byType(MentionSubPageBlock), findsOneWidget);
+ });
+
+ testWidgets('Duplicate a block containing two sub page mentions',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createOpenRenameDocumentUnderParent(name: _firstDocName);
+
+ await tester.insertInlineSubPageFromPlusMenu();
+
+ // Copy paste it
+ 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();
+
+ 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.text(_createdPageName), findsOneWidget);
+ expect(find.text("$_createdPageName (copy)"), findsOneWidget);
+ expect(find.byType(MentionSubPageBlock), findsNWidgets(2));
+
+ // Duplicate node from block action menu
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+ await tester.tapButtonWithName(LocaleKeys.button_duplicate.tr());
+ await tester.pumpAndSettle();
+
+ expect(find.text(_createdPageName), findsOneWidget);
+ 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);
+ });
+ });
+}
+
+extension _InlineSubPageTestHelper on WidgetTester {
+ Future insertInlineSubPageFromPlusMenu() async {
+ await editor.tapLineOfEditorAt(0);
+
+ await editor.showPlusMenu();
+
+ // Workaround to allow typing a document name
+ await FlowyTestKeyboard.simulateKeyDownEvent(
+ tester: this,
+ withKeyUp: true,
+ [
+ LogicalKeyboardKey.keyH,
+ LogicalKeyboardKey.keyI,
+ LogicalKeyboardKey.space,
+ LogicalKeyboardKey.keyW,
+ LogicalKeyboardKey.keyO,
+ LogicalKeyboardKey.keyR,
+ LogicalKeyboardKey.keyL,
+ LogicalKeyboardKey.keyD,
+ ],
+ );
+
+ await FlowyTestKeyboard.simulateKeyDownEvent(
+ tester: this,
+ withKeyUp: true,
+ [LogicalKeyboardKey.enter],
+ );
+ await pumpUntilFound(find.byType(MentionSubPageBlock));
+ }
+}
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 cfea4381e0..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
@@ -1,3 +1,8 @@
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/callout/callout_block_component.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';
@@ -7,9 +12,23 @@ void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
// +, ... button beside the block component.
- group('document with option action button', () {
- testWidgets(
- 'click + to add a block after current selection, and click + and option key to add a block before current selection',
+ group('block option action:', () {
+ Future turnIntoBlock(
+ WidgetTester tester,
+ Path path, {
+ required String menuText,
+ required String afterType,
+ }) async {
+ await tester.editor.openTurnIntoMenu(path);
+ await tester.tapButton(
+ find.findTextInFlowyText(menuText),
+ );
+ final node = tester.editor.getCurrentEditorState().getNodeAtPath(path);
+ expect(node?.type, afterType);
+ }
+
+ testWidgets('''click + to add a block after current selection,
+ and click + and option key to add a block before current selection''',
(tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
@@ -40,5 +59,120 @@ void main() {
expect(editorState.getNodeAtPath([0])?.delta?.toPlainText(), isEmpty);
expect(editorState.getNodeAtPath([2])?.delta?.toPlainText(), isEmpty);
});
+
+ testWidgets('turn into - single line', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ const name = 'Test Document';
+ await tester.createNewPageWithNameUnderParent(name: name);
+ await tester.openPage(name);
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText('turn into');
+
+ // click the block option button to convert it to another blocks
+ final values = {
+ 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.editor_bulletedListShortForm.tr():
+ BulletedListBlockKeys.type,
+ LocaleKeys.editor_numberedListShortForm.tr():
+ NumberedListBlockKeys.type,
+ LocaleKeys.document_slashMenu_name_quote.tr(): QuoteBlockKeys.type,
+ LocaleKeys.editor_checkbox.tr(): TodoListBlockKeys.type,
+ LocaleKeys.document_slashMenu_name_callout.tr(): CalloutBlockKeys.type,
+ LocaleKeys.document_slashMenu_name_text.tr(): ParagraphBlockKeys.type,
+ };
+
+ for (final value in values.entries) {
+ final menuText = value.key;
+ final afterType = value.value;
+ await turnIntoBlock(
+ tester,
+ [0],
+ menuText: menuText,
+ afterType: afterType,
+ );
+ }
+ });
+
+ testWidgets('turn into - multi lines', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ const name = 'Test Document';
+ await tester.createNewPageWithNameUnderParent(name: name);
+ await tester.openPage(name);
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText('turn into 1');
+ await tester.ime.insertCharacter('\n');
+ await tester.ime.insertText('turn into 2');
+
+ // click the block option button to convert it to another blocks
+ final values = {
+ 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.editor_bulletedListShortForm.tr():
+ BulletedListBlockKeys.type,
+ LocaleKeys.editor_numberedListShortForm.tr():
+ NumberedListBlockKeys.type,
+ LocaleKeys.document_slashMenu_name_quote.tr(): QuoteBlockKeys.type,
+ LocaleKeys.editor_checkbox.tr(): TodoListBlockKeys.type,
+ LocaleKeys.document_slashMenu_name_callout.tr(): CalloutBlockKeys.type,
+ LocaleKeys.document_slashMenu_name_text.tr(): ParagraphBlockKeys.type,
+ };
+
+ for (final value in values.entries) {
+ final editorState = tester.editor.getCurrentEditorState();
+ editorState.selection = Selection(
+ start: Position(path: [0]),
+ end: Position(path: [1], offset: 2),
+ );
+ final menuText = value.key;
+ final afterType = value.value;
+ await turnIntoBlock(
+ tester,
+ [0],
+ menuText: menuText,
+ afterType: afterType,
+ );
+ }
+ });
+
+ testWidgets(
+ 'selecting the parent should deselect all the child nodes as well',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ const name = 'Test Document';
+ await tester.createNewPageWithNameUnderParent(name: name);
+ await tester.openPage(name);
+
+ // create a nested list
+ // Item 1
+ // Nested Item 1
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText('Item 1');
+ await tester.ime.insertCharacter('\n');
+ await tester.simulateKeyEvent(LogicalKeyboardKey.tab);
+ await tester.ime.insertText('Nested Item 1');
+
+ // select the 'Nested Item 1' and then tap the option button of the 'Item 1'
+ final editorState = tester.editor.getCurrentEditorState();
+ final selection = Selection.collapsed(
+ Position(path: [0, 0], offset: 1),
+ );
+ editorState.selection = selection;
+ await tester.pumpAndSettle();
+ expect(editorState.selection, selection);
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+ expect(editorState.selection, Selection.collapsed(Position(path: [0])));
+ },
+ );
});
}
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
new file mode 100644
index 0000000000..de1cb880a5
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_selection_test.dart
@@ -0,0 +1,88 @@
+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 '../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('document selection:', () {
+ testWidgets('select text from start to end by pan gesture ',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new document
+ await tester.createNewPageWithNameUnderParent();
+
+ final editor = tester.editor;
+ final editorState = editor.getCurrentEditorState();
+ // insert a paragraph
+ final transaction = editorState.transaction;
+ transaction.insertNode(
+ [0],
+ paragraphNode(
+ text:
+ '''Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged. It was popularised in the 1960s with the release of Letraset sheets containing Lorem Ipsum passages, and more recently with desktop publishing software like Aldus PageMaker including versions of Lorem Ipsum.''',
+ ),
+ );
+ await editorState.apply(transaction);
+ await tester.pumpAndSettle(Durations.short1);
+
+ final textBlocks = find.byType(AppFlowyRichText);
+ final topLeft = tester.getTopLeft(textBlocks.at(0));
+
+ final gesture = await tester.startGesture(
+ topLeft,
+ pointer: 7,
+ );
+ await tester.pumpAndSettle();
+
+ for (var i = 0; i < 10; i++) {
+ await gesture.moveBy(const Offset(10, 0));
+ await tester.pump(Durations.short1);
+ }
+
+ 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_shortcuts_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_shortcuts_test.dart
new file mode 100644
index 0000000000..cf33a66947
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_shortcuts_test.dart
@@ -0,0 +1,140 @@
+import 'package:appflowy_editor/appflowy_editor.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();
+
+ group('document shortcuts:', () {
+ testWidgets('custom cut command', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ const pageName = 'Test Document Shortcuts';
+ await tester.createNewPageWithNameUnderParent(name: pageName);
+
+ // focus on the editor
+ await tester.tap(find.byType(AppFlowyEditor));
+
+ // mock the data
+ final editorState = tester.editor.getCurrentEditorState();
+ final transaction = editorState.transaction;
+ const text1 = '1. First line';
+ const text2 = '2. Second line';
+ transaction.insertNodes([
+ 0,
+ ], [
+ paragraphNode(text: text1),
+ paragraphNode(text: text2),
+ ]);
+ await editorState.apply(transaction);
+ await tester.pumpAndSettle();
+
+ // focus on the end of the first line
+ await tester.editor.updateSelection(
+ Selection.collapsed(
+ Position(path: [0], offset: text1.length),
+ ),
+ );
+ // press the keybinding
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyX,
+ isControlPressed: !UniversalPlatform.isMacOS,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ // check the clipboard
+ final clipboard = await Clipboard.getData(Clipboard.kTextPlain);
+ expect(
+ clipboard?.text,
+ equals(text1),
+ );
+
+ final node = tester.editor.getNodeAtPath([0]);
+ expect(
+ node.delta?.toPlainText(),
+ equals(text2),
+ );
+
+ // select the whole line
+ await tester.editor.updateSelection(
+ Selection.single(
+ path: [0],
+ startOffset: 0,
+ endOffset: text2.length,
+ ),
+ );
+
+ // press the keybinding
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyX,
+ isControlPressed: !UniversalPlatform.isMacOS,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ // all the text should be deleted
+ expect(
+ node.delta?.toPlainText(),
+ equals(''),
+ );
+
+ final clipboard2 = await Clipboard.getData(Clipboard.kTextPlain);
+ expect(
+ clipboard2?.text,
+ equals(text2),
+ );
+ });
+
+ testWidgets(
+ 'custom copy command - copy whole line when selection is collapsed',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ const pageName = 'Test Document Shortcuts';
+ await tester.createNewPageWithNameUnderParent(name: pageName);
+
+ // focus on the editor
+ await tester.tap(find.byType(AppFlowyEditor));
+
+ // mock the data
+ final editorState = tester.editor.getCurrentEditorState();
+ final transaction = editorState.transaction;
+ const text1 = '1. First line';
+ transaction.insertNodes([
+ 0,
+ ], [
+ paragraphNode(text: text1),
+ ]);
+ await editorState.apply(transaction);
+ await tester.pumpAndSettle();
+
+ // focus on the end of the first line
+ await tester.editor.updateSelection(
+ Selection.collapsed(
+ Position(path: [0], offset: text1.length),
+ ),
+ );
+ // press the keybinding
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyC,
+ isControlPressed: !UniversalPlatform.isMacOS,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ // check the clipboard
+ final clipboard = await Clipboard.getData(Clipboard.kTextPlain);
+ expect(
+ clipboard?.text,
+ equals(text1),
+ );
+ });
+ });
+}
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
new file mode 100644
index 0000000000..50f0f903bc
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_sub_page_test.dart
@@ -0,0 +1,528 @@
+import 'dart:io';
+
+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:
+// - [x] Insert a new SubPageBlock from Slash menu items (Expect it will create a child view under current view)
+// - [x] Delete a SubPageBlock from Block Action Menu (Expect the view is moved to trash / deleted)
+// - [x] Delete a SubPageBlock with backspace when selected (Expect the view is moved to trash / deleted)
+// - [x] Copy+paste a SubPageBlock in same Document (Expect a new view is created under current view with same content and name)
+// - [x] Copy+paste a SubPageBlock in different Document (Expect a new view is created under current view with same content and name)
+// - [x] Cut+paste a SubPageBlock in same Document (Expect the view to be deleted on Cut, and brought back on Paste)
+// - [x] Cut+paste a SubPageBlock in different Document (Expect the view to be deleted on Cut, and brought back on Paste)
+// - [x] Undo adding a SubPageBlock (Expect the view to be deleted)
+// - [x] Undo delete of a SubPageBlock (Expect the view to be brought back to original position)
+// - [x] Redo adding a SubPageBlock (Expect the view to be restored)
+// - [x] Redo delete of a SubPageBlock (Expect the view to be moved to trash again)
+// - [x] Renaming a child view (Expect the view name to be updated in the document)
+// - [x] Deleting a view (to trash) linked to a SubPageBlock deleted the SubPageBlock (Expect the SubPageBlock to be deleted)
+// - [x] Duplicating a SubPageBlock node from Action Menu (Expect a new view is created under current view with same content and name + (copy))
+// - [x] Dragging a SubPageBlock node to a new position in the document (Expect everything to be normal)
+
+/// The defaut page name is empty, if we're looking for a "text" we can look for
+/// [LocaleKeys.menuAppHeader_defaultNewPageName] but it won't work for eg. hoverOnPageName
+/// as it looks at the text provided instead of the actual displayed text.
+///
+const _defaultPageName = "";
+
+void main() {
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
+
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
+
+ group('Document SubPageBlock tests', () {
+ testWidgets('Insert a new SubPageBlock from Slash menu items',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ await tester.insertSubPageFromSlashMenu();
+
+ expect(
+ find.text(LocaleKeys.menuAppHeader_defaultNewPageName.tr()),
+ findsNWidgets(3),
+ );
+ });
+
+ testWidgets('Rename and then Delete a SubPageBlock from Block Action Menu',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ await tester.insertSubPageFromSlashMenu();
+
+ await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
+ expect(find.text('Child page'), findsNWidgets(2));
+
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+
+ await tester.tapButtonWithName(LocaleKeys.button_delete.tr());
+ await tester.pumpAndSettle();
+
+ expect(find.text('Child page'), findsNothing);
+ });
+
+ testWidgets('Copy+paste a SubPageBlock in same Document', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ await tester.insertSubPageFromSlashMenu();
+
+ await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
+ expect(find.text('Child page'), findsNWidgets(2));
+
+ await tester.editor.hoverAndClickOptionAddButton([0], false);
+ await tester.editor.tapLineOfEditorAt(1);
+
+ // This is a workaround to allow CTRL+A and CTRL+C to work to copy
+ // the SubPageBlock as well.
+ await tester.ime.insertText('ABC');
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyA,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyC,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ await tester.editor.hoverAndClickOptionAddButton([1], false);
+ await tester.editor.tapLineOfEditorAt(2);
+ await tester.pumpAndSettle();
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle(const Duration(seconds: 5));
+
+ expect(find.byType(SubPageBlockComponent), findsNWidgets(2));
+ expect(find.text('Child page'), findsNWidgets(2));
+ expect(find.text('Child page (copy)'), findsNWidgets(2));
+ });
+
+ testWidgets('Copy+paste a SubPageBlock in different Document',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ await tester.insertSubPageFromSlashMenu();
+
+ await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
+ expect(find.text('Child page'), findsNWidgets(2));
+
+ await tester.editor.hoverAndClickOptionAddButton([0], false);
+ await tester.editor.tapLineOfEditorAt(1);
+
+ // This is a workaround to allow CTRL+A and CTRL+C to work to copy
+ // the SubPageBlock as well.
+ await tester.ime.insertText('ABC');
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyA,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyC,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock-2');
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.pumpAndSettle();
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ await tester.expandOrCollapsePage(
+ pageName: 'SubPageBlock-2',
+ layout: ViewLayoutPB.Document,
+ );
+
+ expect(find.byType(SubPageBlockComponent), findsOneWidget);
+ expect(find.text('Child page'), findsOneWidget);
+ expect(find.text('Child page (copy)'), findsNWidgets(2));
+ });
+
+ testWidgets('Cut+paste a SubPageBlock in same Document', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ await tester.insertSubPageFromSlashMenu();
+
+ await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
+ expect(find.text('Child page'), findsNWidgets(2));
+
+ await tester.editor
+ .updateSelection(Selection.single(path: [0], startOffset: 0));
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyX,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(SubPageBlockComponent), findsNothing);
+ expect(find.text('Child page'), findsNothing);
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(SubPageBlockComponent), findsOneWidget);
+ expect(find.text('Child page'), findsNWidgets(2));
+ });
+
+ testWidgets('Cut+paste a SubPageBlock in different Document',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ await tester.insertSubPageFromSlashMenu();
+
+ await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
+ expect(find.text('Child page'), findsNWidgets(2));
+
+ await tester.editor
+ .updateSelection(Selection.single(path: [0], startOffset: 0));
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyX,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(SubPageBlockComponent), findsNothing);
+ expect(find.text('Child page'), findsNothing);
+
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock-2');
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.pumpAndSettle();
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ await tester.expandOrCollapsePage(
+ pageName: 'SubPageBlock-2',
+ layout: ViewLayoutPB.Document,
+ );
+
+ expect(find.byType(SubPageBlockComponent), findsOneWidget);
+ expect(find.text('Child page'), findsNWidgets(2));
+ expect(find.text('Child page (copy)'), findsNothing);
+ });
+
+ testWidgets('Undo delete of a SubPageBlock', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ await tester.insertSubPageFromSlashMenu();
+
+ await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
+ expect(find.text('Child page'), findsNWidgets(2));
+
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+ await tester.tapButtonWithName(LocaleKeys.button_delete.tr());
+ await tester.pumpAndSettle();
+
+ expect(find.text('Child page'), findsNothing);
+ expect(find.byType(SubPageBlockComponent), findsNothing);
+
+ // Since there is no selection active in editor before deleting Node,
+ // we need to give focus back to the editor
+ await tester.editor
+ .updateSelection(Selection.collapsed(Position(path: [0])));
+
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyZ,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.text('Child page'), findsNWidgets(2));
+ expect(find.byType(SubPageBlockComponent), findsOneWidget);
+ });
+
+ // Redo: undoing deleting a subpage block, then redoing to delete it again
+ // -> Add a subpage block
+ // -> Delete
+ // -> Undo
+ // -> Redo
+ testWidgets('Redo delete of a SubPageBlock', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ await tester.insertSubPageFromSlashMenu(true);
+
+ await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
+ expect(find.text('Child page'), findsNWidgets(2));
+
+ // Delete
+ await tester.editor.hoverAndClickOptionMenuButton([1]);
+ await tester.tapButtonWithName(LocaleKeys.button_delete.tr());
+ await tester.pumpAndSettle();
+
+ expect(find.text('Child page'), findsNothing);
+ expect(find.byType(SubPageBlockComponent), findsNothing);
+
+ await tester.editor.tapLineOfEditorAt(0);
+
+ // Undo
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyZ,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+ expect(find.byType(SubPageBlockComponent), findsOneWidget);
+ expect(find.text('Child page'), findsNWidgets(2));
+
+ // Redo
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyZ,
+ isShiftPressed: true,
+ isControlPressed: Platform.isLinux || Platform.isWindows,
+ isMetaPressed: Platform.isMacOS,
+ );
+ await tester.pumpAndSettle();
+
+ expect(find.byType(SubPageBlockComponent), findsNothing);
+ expect(find.text('Child page'), findsNothing);
+ });
+
+ testWidgets('Delete a view from sidebar', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ await tester.insertSubPageFromSlashMenu();
+
+ await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
+ expect(find.text('Child page'), findsNWidgets(2));
+ expect(find.byType(SubPageBlockComponent), findsOneWidget);
+
+ await tester.hoverOnPageName(
+ 'Child page',
+ onHover: () async {
+ await tester.tapDeletePageButton();
+ },
+ );
+ await tester.pumpAndSettle(const Duration(seconds: 1));
+
+ expect(find.text('Child page'), findsNothing);
+ expect(find.byType(SubPageBlockComponent), findsNothing);
+ });
+
+ testWidgets('Duplicate SubPageBlock from Block Menu', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ await tester.insertSubPageFromSlashMenu();
+ await tester.renamePageWithSecondary(_defaultPageName, 'Child page');
+ expect(find.text('Child page'), findsNWidgets(2));
+
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+
+ await tester.tapButtonWithName(LocaleKeys.button_duplicate.tr());
+ await tester.pumpAndSettle();
+
+ expect(find.text('Child page'), findsNWidgets(2));
+ expect(find.text('Child page (copy)'), findsNWidgets(2));
+ expect(find.byType(SubPageBlockComponent), findsNWidgets(2));
+ });
+
+ testWidgets('Drag SubPageBlock to top of Document', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ await tester.insertSubPageFromSlashMenu(true);
+
+ expect(find.byType(SubPageBlockComponent), findsOneWidget);
+
+ final beforeNode = tester.editor.getNodeAtPath([1]);
+
+ await tester.editor.dragBlock([1], const Offset(20, -45));
+ await tester.pumpAndSettle(Durations.long1);
+
+ final afterNode = tester.editor.getNodeAtPath([0]);
+
+ expect(afterNode.type, SubPageBlockKeys.type);
+ expect(afterNode.type, beforeNode.type);
+ expect(find.byType(SubPageBlockComponent), findsOneWidget);
+ });
+
+ testWidgets('turn into page', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.createNewPageWithNameUnderParent(name: 'SubPageBlock');
+
+ final editorState = tester.editor.getCurrentEditorState();
+
+ // Insert nested list
+ final transaction = editorState.transaction;
+ transaction.insertNode(
+ [0],
+ bulletedListNode(
+ text: 'Parent',
+ children: [
+ bulletedListNode(text: 'Child 1'),
+ bulletedListNode(text: 'Child 2'),
+ ],
+ ),
+ );
+ await editorState.apply(transaction);
+ await tester.pumpAndSettle();
+
+ expect(find.byType(SubPageBlockComponent), findsNothing);
+
+ await tester.editor.hoverAndClickOptionMenuButton([0]);
+ await tester.tapButtonWithName(
+ LocaleKeys.document_plugins_optionAction_turnInto.tr(),
+ );
+ await tester.pumpAndSettle();
+
+ await tester.tap(find.text(LocaleKeys.editor_page.tr()));
+ await tester.pumpAndSettle();
+
+ expect(find.byType(SubPageBlockComponent), findsOneWidget);
+
+ await tester.expandOrCollapsePage(
+ pageName: 'SubPageBlock',
+ layout: ViewLayoutPB.Document,
+ );
+
+ 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);
+ });
+ });
+}
+
+extension _SubPageTestHelper on WidgetTester {
+ Future insertSubPageFromSlashMenu([bool withTextNode = false]) async {
+ await editor.tapLineOfEditorAt(0);
+
+ if (withTextNode) {
+ await ime.insertText('ABC');
+ await editor.getCurrentEditorState().insertNewLine();
+ await pumpAndSettle();
+ }
+
+ await editor.showSlashMenu();
+ await editor.tapSlashMenuItemWithName(
+ LocaleKeys.document_slashMenu_subPage_name.tr(),
+ offset: 100,
+ );
+
+ // Navigate to the previous page to see the SubPageBlock
+ await openPage('SubPageBlock');
+ await pumpAndSettle();
+
+ await pumpUntilFound(find.byType(SubPageBlockComponent));
+ }
+
+ Future renamePageWithSecondary(
+ String currentName,
+ String newName,
+ ) async {
+ await hoverOnPageName(currentName, onHover: () async => pumpAndSettle());
+ await rightClickOnPageName(currentName);
+ await tapButtonWithName(ViewMoreActionType.rename.name);
+ await enterText(find.byType(TextFormField), newName);
+ await tapOKButton();
+ await pumpAndSettle();
+ }
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner.dart
deleted file mode 100644
index 018ed1b8d4..0000000000
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner.dart
+++ /dev/null
@@ -1,48 +0,0 @@
-import 'package:integration_test/integration_test.dart';
-
-import 'document_alignment_test.dart' as document_alignment_test;
-import 'document_codeblock_paste_test.dart' as document_codeblock_paste_test;
-import 'document_copy_and_paste_test.dart' as document_copy_and_paste_test;
-import 'document_create_and_delete_test.dart'
- as document_create_and_delete_test;
-import 'document_option_action_test.dart' as document_option_action_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_text_direction_test.dart' as document_text_direction_test;
-import 'document_with_cover_image_test.dart' as document_with_cover_image_test;
-import 'document_with_database_test.dart' as document_with_database_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_inline_math_equation_test.dart'
- as document_with_inline_math_equation_test;
-import 'document_with_inline_page_test.dart' as document_with_inline_page_test;
-import 'document_with_multi_image_block_test.dart'
- as document_with_multi_image_block_test;
-import 'document_with_outline_block_test.dart' as document_with_outline_block;
-import 'document_with_toggle_list_test.dart' as document_with_toggle_list_test;
-import 'edit_document_test.dart' as document_edit_test;
-
-void startTesting() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
-
- // Document integration tests
- document_create_and_delete_test.main();
- document_edit_test.main();
- document_with_database_test.main();
- document_with_inline_page_test.main();
- document_with_inline_math_equation_test.main();
- document_with_cover_image_test.main();
- document_with_outline_block.main();
- document_with_toggle_list_test.main();
- document_copy_and_paste_test.main();
- document_codeblock_paste_test.main();
- document_alignment_test.main();
- document_text_direction_test.main();
- document_option_action_test.main();
- document_with_image_block_test.main();
- document_with_multi_image_block_test.main();
- document_inline_page_reference_test.main();
- document_more_actions_test.main();
- document_with_file_test.main();
-}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_1.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_1.dart
new file mode 100644
index 0000000000..6a4ad5cb62
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_1.dart
@@ -0,0 +1,23 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'document_create_and_delete_test.dart'
+ as document_create_and_delete_test;
+import 'document_with_cover_image_test.dart' as document_with_cover_image_test;
+import 'document_with_database_test.dart' as document_with_database_test;
+import 'document_with_inline_math_equation_test.dart'
+ as document_with_inline_math_equation_test;
+import 'document_with_inline_page_test.dart' as document_with_inline_page_test;
+import 'edit_document_test.dart' as document_edit_test;
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ // Document integration tests
+ document_create_and_delete_test.main();
+ document_edit_test.main();
+ document_with_database_test.main();
+ document_with_inline_page_test.main();
+ document_with_inline_math_equation_test.main();
+ document_with_cover_image_test.main();
+ // Don't add new tests here.
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_2.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_2.dart
new file mode 100644
index 0000000000..f32db64aa7
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_2.dart
@@ -0,0 +1,26 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'document_app_lifecycle_test.dart' as document_app_lifecycle_test;
+import 'document_deletion_test.dart' as document_deletion_test;
+import 'document_inline_sub_page_test.dart' as document_inline_sub_page_test;
+import 'document_option_action_test.dart' as document_option_action_test;
+import 'document_title_test.dart' as document_title_test;
+import 'document_with_date_reminder_test.dart'
+ as document_with_date_reminder_test;
+import 'document_with_toggle_heading_block_test.dart'
+ as document_with_toggle_heading_block_test;
+import 'document_sub_page_test.dart' as document_sub_page_test;
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ // Document integration tests
+ document_title_test.main();
+ document_app_lifecycle_test.main();
+ document_with_date_reminder_test.main();
+ document_deletion_test.main();
+ document_option_action_test.main();
+ document_inline_sub_page_test.main();
+ document_with_toggle_heading_block_test.main();
+ document_sub_page_test.main();
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_3.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_3.dart
new file mode 100644
index 0000000000..cecdaca580
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_3.dart
@@ -0,0 +1,22 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'document_alignment_test.dart' as document_alignment_test;
+import 'document_codeblock_paste_test.dart' as document_codeblock_paste_test;
+import 'document_copy_and_paste_test.dart' as document_copy_and_paste_test;
+import 'document_text_direction_test.dart' as document_text_direction_test;
+import 'document_with_outline_block_test.dart' as document_with_outline_block;
+import 'document_with_toggle_list_test.dart' as document_with_toggle_list_test;
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ // Document integration tests
+ document_with_outline_block.main();
+ document_with_toggle_list_test.main();
+ document_copy_and_paste_test.main();
+ document_codeblock_paste_test.main();
+ document_alignment_test.main();
+ document_text_direction_test.main();
+
+ // Don't add new tests here.
+}
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
new file mode 100644
index 0000000000..bc0671834b
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_test_runner_4.dart
@@ -0,0 +1,33 @@
+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_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();
+
+ // Document integration tests
+ document_with_image_block_test.main();
+ document_with_multi_image_block_test.main();
+ document_inline_page_reference_test.main();
+ document_more_actions_test.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
new file mode 100644
index 0000000000..c694ba8d6b
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_title_test.dart
@@ -0,0 +1,373 @@
+import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.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/constants.dart';
+import '../../shared/util.dart';
+
+const _testDocumentName = 'Test Document';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('document title:', () {
+ testWidgets('create a new document and edit title', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new document
+ await tester.createNewPageWithNameUnderParent();
+
+ final title = tester.editor.findDocumentTitle('');
+ expect(title, findsOneWidget);
+
+ // input name
+ await tester.enterText(title, _testDocumentName);
+ await tester.pumpAndSettle();
+
+ final newTitle = tester.editor.findDocumentTitle(_testDocumentName);
+ expect(newTitle, findsOneWidget);
+
+ // press enter to create a new line
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ await tester.pumpAndSettle();
+
+ const firstLine = 'First line of text';
+ await tester.ime.insertText(firstLine);
+ await tester.pumpAndSettle();
+
+ final firstLineText = find.text(firstLine, findRichText: true);
+ expect(firstLineText, findsOneWidget);
+
+ // press cmd/ctrl+left to move the cursor to the start of the line
+ if (UniversalPlatform.isMacOS) {
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.arrowLeft,
+ isMetaPressed: true,
+ );
+ } else {
+ await tester.simulateKeyEvent(LogicalKeyboardKey.home);
+ }
+ await tester.pumpAndSettle();
+
+ // press arrow left to delete the first line
+ await tester.simulateKeyEvent(LogicalKeyboardKey.arrowLeft);
+ await tester.pumpAndSettle();
+
+ // check if the title is on focus
+ final titleOnFocus = tester.editor.findDocumentTitle(_testDocumentName);
+ final titleWidget = tester.widget(titleOnFocus);
+ expect(titleWidget.focusNode?.hasFocus, isTrue);
+
+ // press the right arrow key to move the cursor to the first line
+ await tester.simulateKeyEvent(LogicalKeyboardKey.arrowRight);
+
+ // check if the title is not on focus
+ expect(titleWidget.focusNode?.hasFocus, isFalse);
+
+ final editorState = tester.editor.getCurrentEditorState();
+ expect(editorState.selection, Selection.collapsed(Position(path: [0])));
+
+ // press the backspace key to go to the title
+ await tester.simulateKeyEvent(LogicalKeyboardKey.backspace);
+
+ expect(editorState.selection, null);
+ expect(titleWidget.focusNode?.hasFocus, isTrue);
+ });
+
+ testWidgets('check if the title is saved', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new document
+ await tester.createNewPageWithNameUnderParent();
+
+ final title = tester.editor.findDocumentTitle('');
+ expect(title, findsOneWidget);
+
+ // input name
+ await tester.enterText(title, _testDocumentName);
+ await tester.pumpAndSettle();
+
+ if (UniversalPlatform.isLinux) {
+ // wait for the name to be saved
+ await tester.wait(250);
+ }
+
+ // go to the get started page
+ await tester.tapButton(
+ tester.findPageName(Constants.gettingStartedPageName),
+ );
+
+ // go back to the page
+ await tester.tapButton(tester.findPageName(_testDocumentName));
+
+ // check if the title is saved
+ final testDocumentTitle = tester.editor.findDocumentTitle(
+ _testDocumentName,
+ );
+ expect(testDocumentTitle, findsOneWidget);
+ });
+
+ testWidgets('arrow up from first line moves focus to title',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent();
+
+ final title = tester.editor.findDocumentTitle('');
+ await tester.enterText(title, _testDocumentName);
+ await tester.pumpAndSettle();
+
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ await tester.ime.insertText('First line of text');
+ await tester.pumpAndSettle();
+
+ await tester.simulateKeyEvent(LogicalKeyboardKey.home);
+
+ // press the arrow upload
+ await tester.simulateKeyEvent(LogicalKeyboardKey.arrowUp);
+
+ final titleWidget = tester.widget(
+ tester.editor.findDocumentTitle(_testDocumentName),
+ );
+ expect(titleWidget.focusNode?.hasFocus, isTrue);
+
+ final editorState = tester.editor.getCurrentEditorState();
+ expect(editorState.selection, null);
+ });
+
+ testWidgets(
+ 'backspace at start of first line moves focus to title and deletes empty paragraph',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent();
+
+ final title = tester.editor.findDocumentTitle('');
+ await tester.enterText(title, _testDocumentName);
+ await tester.pumpAndSettle();
+
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+
+ final editorState = tester.editor.getCurrentEditorState();
+ expect(editorState.document.root.children.length, equals(2));
+
+ await tester.simulateKeyEvent(LogicalKeyboardKey.backspace);
+
+ final titleWidget = tester.widget(
+ tester.editor.findDocumentTitle(_testDocumentName),
+ );
+ expect(titleWidget.focusNode?.hasFocus, isTrue);
+
+ // at least one empty paragraph node is created
+ expect(editorState.document.root.children.length, equals(1));
+ });
+
+ testWidgets('arrow right from end of title moves focus to first line',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent();
+
+ final title = tester.editor.findDocumentTitle('');
+ await tester.enterText(title, _testDocumentName);
+ await tester.pumpAndSettle();
+
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ await tester.ime.insertText('First line of text');
+
+ await tester.tapButton(
+ tester.editor.findDocumentTitle(_testDocumentName),
+ );
+ await tester.simulateKeyEvent(LogicalKeyboardKey.end);
+ await tester.simulateKeyEvent(LogicalKeyboardKey.arrowRight);
+
+ final editorState = tester.editor.getCurrentEditorState();
+ expect(
+ editorState.selection,
+ Selection.collapsed(
+ Position(path: [0]),
+ ),
+ );
+ });
+
+ testWidgets('change the title via sidebar, check the title is updated',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent();
+
+ final title = tester.editor.findDocumentTitle('');
+ expect(title, findsOneWidget);
+
+ await tester.hoverOnPageName(
+ '',
+ onHover: () async {
+ await tester.renamePage(_testDocumentName);
+ await tester.pumpAndSettle();
+ },
+ );
+ await tester.pumpAndSettle();
+
+ final newTitle = tester.editor.findDocumentTitle(_testDocumentName);
+ expect(newTitle, findsOneWidget);
+ });
+
+ testWidgets('execute undo and redo in title', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent();
+
+ final title = tester.editor.findDocumentTitle('');
+ await tester.enterText(title, _testDocumentName);
+ // press a random key to make the undo stack not empty
+ await tester.simulateKeyEvent(LogicalKeyboardKey.keyA);
+ await tester.pumpAndSettle();
+
+ // undo
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyZ,
+ isControlPressed: !UniversalPlatform.isMacOS,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ );
+ // wait for the undo to be applied
+ await tester.pumpAndSettle(Durations.long1);
+
+ // expect the title is empty
+ expect(
+ tester
+ .widget(
+ tester.editor.findDocumentTitle(''),
+ )
+ .controller
+ ?.text,
+ '',
+ );
+
+ // redo
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyZ,
+ isControlPressed: !UniversalPlatform.isMacOS,
+ isMetaPressed: UniversalPlatform.isMacOS,
+ isShiftPressed: true,
+ );
+
+ await tester.pumpAndSettle(Durations.short1);
+
+ if (UniversalPlatform.isMacOS) {
+ expect(
+ tester
+ .widget(
+ tester.editor.findDocumentTitle(_testDocumentName),
+ )
+ .controller
+ ?.text,
+ _testDocumentName,
+ );
+ }
+ });
+
+ testWidgets('escape key should exit the editing mode', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent();
+
+ final title = tester.editor.findDocumentTitle('');
+ await tester.enterText(title, _testDocumentName);
+ await tester.pumpAndSettle();
+
+ await tester.simulateKeyEvent(LogicalKeyboardKey.escape);
+ await tester.pumpAndSettle();
+
+ expect(
+ tester
+ .widget(
+ tester.editor.findDocumentTitle(_testDocumentName),
+ )
+ .focusNode
+ ?.hasFocus,
+ isFalse,
+ );
+ });
+
+ testWidgets('press arrow down key in title, check if the cursor flashes',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent();
+
+ final title = tester.editor.findDocumentTitle('');
+ await tester.enterText(title, _testDocumentName);
+ await tester.pumpAndSettle();
+
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ const inputText = 'Hello World';
+ await tester.ime.insertText(inputText);
+
+ await tester.tapButton(
+ tester.editor.findDocumentTitle(_testDocumentName),
+ );
+ await tester.simulateKeyEvent(LogicalKeyboardKey.arrowDown);
+ final editorState = tester.editor.getCurrentEditorState();
+ expect(
+ editorState.selection,
+ Selection.collapsed(
+ Position(path: [0], offset: inputText.length),
+ ),
+ );
+ });
+
+ testWidgets(
+ 'hover on the cover title, check if the add icon & add cover button are shown',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent();
+
+ final title = tester.editor.findDocumentTitle('');
+ await tester.hoverOnWidget(
+ title,
+ onHover: () async {
+ expect(find.byType(DocumentCoverWidget), findsOneWidget);
+ },
+ );
+
+ 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 fd6e0dfa19..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,20 +1,36 @@
-import 'package:flutter/material.dart';
+import 'dart:io';
import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/header/document_header_node_widget.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;
+ });
- group('cover image', () {
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
+
+ group('cover image:', () {
testWidgets('document cover tests', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
@@ -52,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();
@@ -148,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 eb07a2e7a8..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';
@@ -60,6 +63,57 @@ void main() {
);
});
+ testWidgets('insert multiple referenced boards', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new grid
+ final id = uuid();
+ final name = '${ViewLayoutPB.Board.name}_$id';
+ await tester.createNewPageWithNameUnderParent(
+ name: name,
+ layout: ViewLayoutPB.Board,
+ openAfterCreated: false,
+ );
+ // create a new document
+ await tester.createNewPageWithNameUnderParent(
+ name: 'insert_a_reference_${ViewLayoutPB.Board.name}',
+ );
+ // tap the first line of the document
+ await tester.editor.tapLineOfEditorAt(0);
+ // insert a referenced view
+ await tester.editor.showSlashMenu();
+ await tester.editor.tapSlashMenuItemWithName(
+ ViewLayoutPB.Board.slashMenuLinkedName,
+ );
+ final referencedDatabase1 = find.descendant(
+ of: find.byType(InlineActionsHandler),
+ matching: find.findTextInFlowyText(name),
+ );
+ expect(referencedDatabase1, findsOneWidget);
+ await tester.tapButton(referencedDatabase1);
+
+ await tester.editor.tapLineOfEditorAt(1);
+ await tester.editor.showSlashMenu();
+ await tester.editor.tapSlashMenuItemWithName(
+ ViewLayoutPB.Board.slashMenuLinkedName,
+ );
+ final referencedDatabase2 = find.descendant(
+ of: find.byType(InlineActionsHandler),
+ matching: find.findTextInFlowyText(name),
+ );
+ expect(referencedDatabase2, findsOneWidget);
+ await tester.tapButton(referencedDatabase2);
+
+ expect(
+ find.descendant(
+ of: find.byType(AppFlowyEditor),
+ matching: find.byType(DesktopBoardPage),
+ ),
+ findsNWidgets(2),
+ );
+ });
+
testWidgets('insert a referenced calendar', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
@@ -123,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
new file mode 100644
index 0000000000..ccfdbae76e
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_date_reminder_test.dart
@@ -0,0 +1,466 @@
+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';
+
+void main() {
+ setUp(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ TestWidgetsFlutterBinding.ensureInitialized();
+ });
+
+ group('date or reminder block in document:', () {
+ testWidgets("insert date with time block", (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new document
+ await tester.createNewPageWithNameUnderParent(
+ name: 'Date with time test',
+ );
+
+ // 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);
+
+ // tap on date field
+ await tester.tap(find.byType(MentionDateBlock));
+ await tester.pumpAndSettle();
+
+ // tap the toggle of include time
+ await tester.tap(find.byType(Toggle));
+ await tester.pumpAndSettle();
+
+ // add time 11:12
+ final textField = find
+ .descendant(
+ of: find.byType(DesktopAppFlowyDatePicker),
+ matching: find.byType(TextField),
+ )
+ .last;
+ await tester.pumpUntilFound(textField);
+ await tester.enterText(textField, "11:12");
+ await tester.testTextInput.receiveAction(TextInputAction.done);
+ await tester.pumpAndSettle();
+
+ // we will get field with current date and 11:12 as time
+ expect(find.byType(MentionDateBlock), findsOneWidget);
+ expect(find.text('@$formattedDate 11:12'), findsOneWidget);
+ });
+
+ testWidgets("insert date with reminder block", (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // create a new document
+ await tester.createNewPageWithNameUnderParent(
+ name: 'Date with reminder test',
+ );
+
+ // 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);
+
+ // tap on date field
+ await tester.tap(find.byType(MentionDateBlock));
+ await tester.pumpAndSettle();
+
+ // tap reminder and set reminder to 1 day before
+ await tester.tap(find.text(LocaleKeys.datePicker_reminderLabel.tr()));
+ await tester.pumpAndSettle();
+ await tester.tap(
+ find.textContaining(
+ LocaleKeys.datePicker_reminderOptions_oneDayBefore.tr(),
+ ),
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.pumpAndSettle();
+
+ // we will get field with current date reminder_clock.svg icon
+ expect(find.byType(MentionDateBlock), findsOneWidget);
+ 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_file_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_file_test.dart
index 76ad7d612f..9d7a97e6a8 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_file_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_file_test.dart
@@ -1,5 +1,7 @@
import 'dart:io';
+import 'package:flutter/services.dart';
+
import 'package:appflowy/core/config/kv.dart';
import 'package:appflowy/core/config/kv_keys.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
@@ -8,7 +10,6 @@ import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.da
import 'package:appflowy/startup/startup.dart';
import 'package:easy_localization/easy_localization.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';
import 'package:path/path.dart' as p;
@@ -36,10 +37,6 @@ void main() {
LocaleKeys.document_slashMenu_name_file.tr(),
);
expect(find.byType(FileBlockComponent), findsOneWidget);
-
- await tester.tap(find.byType(FileBlockComponent));
- await tester.pumpAndSettle(const Duration(seconds: 1));
-
expect(find.byType(FileUploadMenu), findsOneWidget);
final image = await rootBundle.load('assets/test/images/sample.jpeg');
@@ -50,9 +47,7 @@ void main() {
mockPickFilePaths(paths: [filePath]);
await getIt().set(KVKeys.kCloudType, '0');
- await tester.tap(
- find.text(LocaleKeys.document_plugins_file_fileUploadHint.tr()),
- );
+ await tester.tapFileUploadHint();
await tester.pumpAndSettle();
expect(find.byType(FileUploadMenu), findsNothing);
@@ -116,9 +111,6 @@ void main() {
LocaleKeys.document_slashMenu_name_file.tr(),
);
expect(find.byType(FileBlockComponent), findsOneWidget);
-
- await tester.tap(find.byType(FileBlockComponent));
- await tester.pumpAndSettle(const Duration(seconds: 1));
expect(find.byType(FileUploadMenu), findsOneWidget);
// Navigate to integrate link tab
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 fb0c686824..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
@@ -6,21 +6,17 @@ import 'package:appflowy/generated/locale_keys.g.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/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/unsplash_image_widget.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';
@@ -80,94 +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);
-
- await tester.tapButtonWithName(
- 'Unsplash',
- );
- expect(find.byType(UnsplashImageWidget), 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 45613fe97f..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
@@ -1,5 +1,5 @@
+import 'package:appflowy/plugins/database/grid/presentation/grid_page.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/mention/mention_page_block.dart';
-import 'package:appflowy/shared/flowy_error_page.dart';
import 'package:appflowy_backend/protobuf/flowy-folder/protobuf.dart';
import 'package:flowy_infra/uuid.dart';
import 'package:flutter_test/flutter_test.dart';
@@ -92,7 +92,21 @@ void main() {
);
expect(finder, findsOneWidget);
await tester.tapButton(finder);
- expect(find.byType(AppFlowyErrorPage), findsOneWidget);
+ 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_multi_image_block_test.dart b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_multi_image_block_test.dart
index d85e6c631e..d8b0784a39 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/document/document_with_multi_image_block_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_multi_image_block_test.dart
@@ -6,7 +6,6 @@ import 'package:appflowy/generated/flowy_svgs.g.dart';
import 'package:appflowy/generated/locale_keys.g.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/image/multi_image_block_component/image_render.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/image/multi_image_block_component/layouts/image_browser_layout.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/image/multi_image_block_component/multi_image_block_component.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/image/multi_image_block_component/multi_image_placeholder.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';
@@ -26,7 +25,6 @@ import 'package:path_provider/path_provider.dart';
import '../../shared/mock/mock_file_picker.dart';
import '../../shared/util.dart';
-import '../board/board_hide_groups_test.dart';
void main() {
setUp(() {
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 fc6d0f86a6..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,9 +1,7 @@
import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/outline/outline_block_component.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/plugins.dart';
-import 'package:appflowy/workspace/presentation/widgets/pop_up_action.dart';
import 'package:easy_localization/easy_localization.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';
@@ -46,6 +44,9 @@ void main() {
* # Heading 1
* ## Heading 2
* ### Heading 3
+ * > # Heading 1
+ * > ## Heading 2
+ * > ### Heading 3
*/
await tester.editor.tapLineOfEditorAt(3);
@@ -56,7 +57,7 @@ void main() {
of: find.byType(OutlineBlockWidget),
matching: find.text(heading1),
),
- findsOneWidget,
+ findsNWidgets(2),
);
// Heading 2 is prefixed with a bullet
@@ -65,7 +66,7 @@ void main() {
of: find.byType(OutlineBlockWidget),
matching: find.text(heading2),
),
- findsOneWidget,
+ findsNWidgets(2),
);
// Heading 3 is prefixed with a dash
@@ -74,7 +75,7 @@ void main() {
of: find.byType(OutlineBlockWidget),
matching: find.text(heading3),
),
- findsOneWidget,
+ findsNWidgets(2),
);
// update the Heading 1 to Heading 1Hello world
@@ -102,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),
@@ -126,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),
@@ -137,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(
@@ -151,7 +155,7 @@ void main() {
of: find.byType(OutlineBlockWidget),
matching: find.text(heading2),
),
- findsOneWidget,
+ findsNWidgets(2),
);
expect(
@@ -159,7 +163,7 @@ void main() {
of: find.byType(OutlineBlockWidget),
matching: find.text(heading3),
),
- findsOneWidget,
+ findsNWidgets(2),
);
//////
});
@@ -172,7 +176,6 @@ Future insertOutlineInDocument(WidgetTester tester) async {
await tester.editor.showSlashMenu();
await tester.editor.tapSlashMenuItemWithName(
LocaleKeys.document_slashMenu_name_outline.tr(),
- offset: 100,
);
await tester.pumpAndSettle();
}
@@ -182,19 +185,25 @@ Future hoverAndClickDepthOptionAction(
List path,
int level,
) async {
- await tester.editor.hoverAndClickOptionMenuButton([3]);
- await tester.tap(find.byType(AppFlowyPopover).hitTestable().last);
- await tester.pumpAndSettle();
-
- // Find a total of 4 HoverButtons under the [BlockOptionButton],
- // in addition to 3 HoverButtons under the [DepthOptionAction] - (child of BlockOptionButton)
- await tester.tap(find.byType(HoverButton).hitTestable().at(3 + level));
+ await tester.editor.openDepthMenu(path);
+ final type = OptionDepthType.fromLevel(level);
+ await tester.tapButton(find.findTextInFlowyText(type.description));
await tester.pumpAndSettle();
}
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
new file mode 100644
index 0000000000..c4aa289855
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/document/document_with_toggle_heading_block_test.dart
@@ -0,0 +1,123 @@
+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:flutter/services.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.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('toggle heading block test:', () {
+ testWidgets('insert toggle heading 1 - 3 block', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(
+ name: 'toggle heading block test',
+ );
+
+ for (var i = 1; i <= 3; i++) {
+ await tester.editor.tapLineOfEditorAt(0);
+ await _insertToggleHeadingBlockInDocument(tester, i);
+ await tester.pumpAndSettle();
+ expect(
+ find.byWidgetPredicate(
+ (widget) =>
+ widget is ToggleListBlockComponentWidget &&
+ widget.node.attributes[ToggleListBlockKeys.level] == i,
+ ),
+ findsOneWidget,
+ );
+ }
+ });
+
+ testWidgets('insert toggle heading 1 - 3 block by shortcuts',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(
+ name: 'toggle heading block test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText('# > $_heading1\n');
+ await tester.ime.insertText('## > $_heading2\n');
+ await tester.ime.insertText('### > $_heading3\n');
+ await tester.ime.insertText('> # $_heading1\n');
+ await tester.ime.insertText('> ## $_heading2\n');
+ await tester.ime.insertText('> ### $_heading3\n');
+ await tester.pumpAndSettle();
+
+ expect(
+ find.byType(ToggleListBlockComponentWidget),
+ findsNWidgets(6),
+ );
+ });
+
+ testWidgets('insert toggle heading and convert it to heading',
+ (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(
+ name: 'toggle heading block test',
+ );
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText('# > $_heading1\n');
+ await tester.simulateKeyEvent(LogicalKeyboardKey.enter);
+ await tester.ime.insertText('item 1');
+ await tester.pumpAndSettle();
+
+ await tester.editor.updateSelection(
+ Selection(
+ start: Position(path: [0]),
+ end: Position(path: [0], offset: _heading1.length),
+ ),
+ );
+
+ await tester.tapButton(find.byFlowySvg(FlowySvgs.toolbar_text_format_m));
+
+ // tap the H1 button
+ await tester.tapButton(find.byFlowySvg(FlowySvgs.type_h1_m).at(0));
+ await tester.pumpAndSettle();
+
+ final editorState = tester.editor.getCurrentEditorState();
+ final node1 = editorState.document.nodeAtPath([0])!;
+ expect(node1.type, HeadingBlockKeys.type);
+ expect(node1.attributes[HeadingBlockKeys.level], 1);
+
+ final node2 = editorState.document.nodeAtPath([1])!;
+ expect(node2.type, ParagraphBlockKeys.type);
+ expect(node2.delta!.toPlainText(), 'item 1');
+ });
+ });
+}
+
+Future _insertToggleHeadingBlockInDocument(
+ WidgetTester tester,
+ int level,
+) async {
+ final name = switch (level) {
+ 1 => LocaleKeys.document_slashMenu_name_toggleHeading1.tr(),
+ 2 => LocaleKeys.document_slashMenu_name_toggleHeading2.tr(),
+ 3 => LocaleKeys.document_slashMenu_name_toggleHeading3.tr(),
+ _ => throw Exception('Invalid level: $level'),
+ };
+ await tester.editor.showSlashMenu();
+ await tester.editor.tapSlashMenuItemWithName(
+ name,
+ offset: 150,
+ );
+ await tester.pumpAndSettle();
+}
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 0f3bab2f8e..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';
@@ -214,5 +216,73 @@ void main() {
expectToggleListOpened();
});
+
+ Future prepareToggleHeadingBlock(
+ WidgetTester tester,
+ String text,
+ ) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent();
+
+ await tester.editor.tapLineOfEditorAt(0);
+ await tester.ime.insertText(text);
+ }
+
+ testWidgets('> + # to toggle heading 1 block', (tester) async {
+ await prepareToggleHeadingBlock(tester, '> # Hello');
+ final editorState = tester.editor.getCurrentEditorState();
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ToggleListBlockKeys.type);
+ expect(node.attributes[ToggleListBlockKeys.level], 1);
+ expect(node.delta!.toPlainText(), 'Hello');
+ });
+
+ testWidgets('> + ### to toggle heading 3 block', (tester) async {
+ await prepareToggleHeadingBlock(tester, '> ### Hello');
+ final editorState = tester.editor.getCurrentEditorState();
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ToggleListBlockKeys.type);
+ expect(node.attributes[ToggleListBlockKeys.level], 3);
+ expect(node.delta!.toPlainText(), 'Hello');
+ });
+
+ testWidgets('# + > to toggle heading 1 block', (tester) async {
+ await prepareToggleHeadingBlock(tester, '# > Hello');
+ final editorState = tester.editor.getCurrentEditorState();
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ToggleListBlockKeys.type);
+ expect(node.attributes[ToggleListBlockKeys.level], 1);
+ expect(node.delta!.toPlainText(), 'Hello');
+ });
+
+ testWidgets('### + > to toggle heading 3 block', (tester) async {
+ await prepareToggleHeadingBlock(tester, '### > Hello');
+ final editorState = tester.editor.getCurrentEditorState();
+ final node = editorState.getNodeAtPath([0])!;
+ expect(node.type, ToggleListBlockKeys.type);
+ 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/first_test/first_test.dart b/frontend/appflowy_flutter/integration_test/desktop/first_test/first_test.dart
new file mode 100644
index 0000000000..36c0e391fb
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/first_test/first_test.dart
@@ -0,0 +1,17 @@
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/util.dart';
+
+// This test is meaningless, just for preventing the CI from failing.
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('Empty', () {
+ testWidgets('empty test', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+ await tester.wait(500);
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/grid/grid_create_row_test.dart b/frontend/appflowy_flutter/integration_test/desktop/grid/grid_create_row_test.dart
deleted file mode 100644
index b8f2d9ca61..0000000000
--- a/frontend/appflowy_flutter/integration_test/desktop/grid/grid_create_row_test.dart
+++ /dev/null
@@ -1,203 +0,0 @@
-import 'package:appflowy/generated/locale_keys.g.dart';
-import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/checkbox.dart';
-import 'package:appflowy/workspace/presentation/home/menu/sidebar/space/shared_widget.dart';
-import 'package:appflowy_backend/protobuf/flowy-database2/protobuf.dart';
-import 'package:appflowy_backend/protobuf/flowy-folder/view.pbenum.dart';
-import 'package:collection/collection.dart';
-import 'package:easy_localization/easy_localization.dart';
-import 'package:flutter_test/flutter_test.dart';
-import 'package:integration_test/integration_test.dart';
-
-import '../../shared/database_test_op.dart';
-import '../../shared/util.dart';
-import 'grid_test_extensions.dart';
-
-void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
-
- group('grid create row test:', () {
- testWidgets('from the bottom', (tester) async {
- await tester.initializeAppFlowy();
- await tester.tapAnonymousSignInButton();
-
- await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
-
- final expected = tester.getGridRows();
-
- // create row
- await tester.tapCreateRowButtonInGrid();
-
- final actual = tester.getGridRows();
- expect(actual.slice(0, 3), orderedEquals(expected));
- expect(actual.length, equals(4));
- tester.assertNumberOfRowsInGridPage(4);
- });
-
- testWidgets('from a row\'s menu', (tester) async {
- await tester.initializeAppFlowy();
- await tester.tapAnonymousSignInButton();
-
- await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
-
- final expected = tester.getGridRows();
-
- // create row
- await tester.hoverOnFirstRowOfGrid();
- await tester.tapCreateRowButtonAfterHoveringOnGridRow();
-
- final actual = tester.getGridRows();
- expect([actual[0], actual[2], actual[3]], orderedEquals(expected));
- expect(actual.length, equals(4));
- tester.assertNumberOfRowsInGridPage(4);
- });
-
- testWidgets('with sort configured', (tester) async {
- await tester.openTestDatabase(v069GridFileName);
-
- // get grid data
- final unsorted = tester.getGridRows();
-
- // add a sort
- await tester.tapDatabaseSortButton();
- await tester.tapCreateSortByFieldType(FieldType.RichText, 'Name');
-
- final sorted = [
- unsorted[7],
- unsorted[8],
- unsorted[1],
- unsorted[9],
- unsorted[11],
- unsorted[10],
- unsorted[6],
- unsorted[12],
- unsorted[2],
- unsorted[0],
- unsorted[3],
- unsorted[5],
- unsorted[4],
- ];
-
- List actual = tester.getGridRows();
- expect(actual, orderedEquals(sorted));
-
- // create row
- await tester.hoverOnFirstRowOfGrid();
- await tester.tapCreateRowButtonAfterHoveringOnGridRow();
-
- // cancel
- expect(find.byType(ConfirmPopup), findsOneWidget);
- await tester.tapButtonWithName(LocaleKeys.button_cancel.tr());
-
- // verify grid data
- actual = tester.getGridRows();
- expect(actual, orderedEquals(sorted));
-
- // try again, but confirm this time
- await tester.hoverOnFirstRowOfGrid();
- await tester.tapCreateRowButtonAfterHoveringOnGridRow();
- expect(find.byType(ConfirmPopup), findsOneWidget);
- await tester.tapButtonWithName(LocaleKeys.button_remove.tr());
-
- // verify grid data
- actual = tester.getGridRows();
- expect(actual.length, equals(14));
- tester.assertNumberOfRowsInGridPage(14);
- });
-
- testWidgets('with filter configured', (tester) async {
- await tester.openTestDatabase(v069GridFileName);
-
- // get grid data
- final original = tester.getGridRows();
-
- // create a filter
- await tester.tapDatabaseFilterButton();
- await tester.tapCreateFilterByFieldType(
- FieldType.Checkbox,
- 'Registration Complete',
- );
-
- final filtered = [
- original[1],
- original[3],
- original[5],
- original[6],
- original[7],
- original[9],
- original[12],
- ];
-
- // verify grid data
- List actual = tester.getGridRows();
- expect(actual, orderedEquals(filtered));
-
- // create row (one before and after the first row, and one at the bottom)
- await tester.tapCreateRowButtonInGrid();
- await tester.hoverOnFirstRowOfGrid();
- await tester.tapCreateRowButtonAfterHoveringOnGridRow();
- await tester.hoverOnFirstRowOfGrid(() async {
- await tester.tapRowMenuButtonInGrid();
- await tester.tapCreateRowAboveButtonInRowMenu();
- });
-
- actual = tester.getGridRows();
- expect(actual.length, equals(10));
- tester.assertNumberOfRowsInGridPage(10);
- actual = [
- actual[1],
- actual[3],
- actual[4],
- actual[5],
- actual[6],
- actual[7],
- actual[8],
- ];
- expect(actual, orderedEquals(filtered));
-
- // delete the filter
- await tester.tapFilterButtonInGrid('Registration Complete');
- await tester
- .tapDisclosureButtonInFinder(find.byType(CheckboxFilterEditor));
- await tester.tapDeleteFilterButtonInGrid();
-
- // verify grid data
- actual = tester.getGridRows();
- expect(actual.length, equals(16));
- tester.assertNumberOfRowsInGridPage(16);
- actual = [
- actual[0],
- actual[2],
- actual[4],
- actual[5],
- actual[6],
- actual[7],
- actual[8],
- actual[9],
- actual[10],
- actual[11],
- actual[12],
- actual[13],
- actual[14],
- ];
- expect(actual, orderedEquals(original));
- });
-
- // TODO(RS): move to somewhere else
- testWidgets('delete row of the grid', (tester) async {
- await tester.initializeAppFlowy();
- await tester.tapAnonymousSignInButton();
-
- await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
- await tester.hoverOnFirstRowOfGrid(() async {
- // Open the row menu and then click the delete
- await tester.tapRowMenuButtonInGrid();
- await tester.pumpAndSettle();
- await tester.tapDeleteOnRowMenu();
- await tester.pumpAndSettle();
-
- // 3 initial rows - 1 deleted
- tester.assertNumberOfRowsInGridPage(2);
- });
- });
- });
-}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/grid/grid_edit_row_test.dart b/frontend/appflowy_flutter/integration_test/desktop/grid/grid_edit_row_test.dart
new file mode 100644
index 0000000000..64b7a40ad1
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/grid/grid_edit_row_test.dart
@@ -0,0 +1,137 @@
+import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/checkbox.dart';
+import 'package:appflowy_backend/protobuf/flowy-database2/protobuf.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+import 'package:time/time.dart';
+
+import '../../shared/database_test_op.dart';
+import 'grid_test_extensions.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('grid edit row test:', () {
+ testWidgets('with sort configured', (tester) async {
+ await tester.openTestDatabase(v069GridFileName);
+
+ // get grid data
+ final unsorted = tester.getGridRows();
+
+ // add a sort
+ await tester.tapDatabaseSortButton();
+ await tester.tapCreateSortByFieldType(FieldType.RichText, 'Name');
+
+ final sorted = [
+ unsorted[7],
+ unsorted[8],
+ unsorted[1],
+ unsorted[9],
+ unsorted[11],
+ unsorted[10],
+ unsorted[6],
+ unsorted[12],
+ unsorted[2],
+ unsorted[0],
+ unsorted[3],
+ unsorted[5],
+ unsorted[4],
+ ];
+
+ List actual = tester.getGridRows();
+ expect(actual, orderedEquals(sorted));
+
+ await tester.editCell(
+ rowIndex: 4,
+ fieldType: FieldType.RichText,
+ input: "x",
+ );
+ await tester.pumpAndSettle(200.milliseconds);
+
+ final reSorted = [
+ unsorted[7],
+ unsorted[8],
+ unsorted[1],
+ unsorted[9],
+ unsorted[10],
+ unsorted[6],
+ unsorted[12],
+ unsorted[2],
+ unsorted[0],
+ unsorted[3],
+ unsorted[5],
+ unsorted[11],
+ unsorted[4],
+ ];
+
+ // verify grid data
+ actual = tester.getGridRows();
+ expect(actual, orderedEquals(reSorted));
+
+ // delete the sort
+ await tester.tapSortMenuInSettingBar();
+ await tester.tapDeleteAllSortsButton();
+
+ // verify grid data
+ actual = tester.getGridRows();
+ expect(actual, orderedEquals(unsorted));
+ });
+
+ testWidgets('with filter configured', (tester) async {
+ await tester.openTestDatabase(v069GridFileName);
+
+ // get grid data
+ final original = tester.getGridRows();
+
+ // create a filter
+ await tester.tapDatabaseFilterButton();
+ await tester.tapCreateFilterByFieldType(
+ FieldType.Checkbox,
+ 'Registration Complete',
+ );
+
+ final filtered = [
+ original[1],
+ original[3],
+ original[5],
+ original[6],
+ original[7],
+ original[9],
+ original[12],
+ ];
+
+ // verify grid data
+ List actual = tester.getGridRows();
+ expect(actual, orderedEquals(filtered));
+ expect(actual.length, equals(7));
+ tester.assertNumberOfRowsInGridPage(7);
+
+ await tester.tapCheckboxCellInGrid(rowIndex: 0);
+ await tester.pumpAndSettle(200.milliseconds);
+
+ // verify grid data
+ actual = tester.getGridRows();
+ expect(actual.length, equals(6));
+ tester.assertNumberOfRowsInGridPage(6);
+ final edited = [
+ original[3],
+ original[5],
+ original[6],
+ original[7],
+ original[9],
+ original[12],
+ ];
+ expect(actual, orderedEquals(edited));
+
+ // delete the filter
+ await tester.tapFilterButtonInGrid('Registration Complete');
+ await tester
+ .tapDisclosureButtonInFinder(find.byType(CheckboxFilterEditor));
+ await tester.tapDeleteFilterButtonInGrid();
+
+ // verify grid data
+ actual = tester.getGridRows();
+ expect(actual.length, equals(13));
+ tester.assertNumberOfRowsInGridPage(13);
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/grid/grid_row_test.dart b/frontend/appflowy_flutter/integration_test/desktop/grid/grid_row_test.dart
new file mode 100644
index 0000000000..d7efb797f0
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/grid/grid_row_test.dart
@@ -0,0 +1,234 @@
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/checkbox.dart';
+import 'package:appflowy/workspace/presentation/home/menu/sidebar/space/shared_widget.dart';
+import 'package:appflowy_backend/protobuf/flowy-database2/protobuf.dart';
+import 'package:appflowy_backend/protobuf/flowy-folder/view.pbenum.dart';
+import 'package:collection/collection.dart';
+import 'package:easy_localization/easy_localization.dart';
+import 'package:flutter_test/flutter_test.dart';
+import 'package:integration_test/integration_test.dart';
+
+import '../../shared/database_test_op.dart';
+import '../../shared/util.dart';
+
+import 'grid_test_extensions.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('grid row test:', () {
+ testWidgets('create from the bottom', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
+
+ final expected = tester.getGridRows();
+
+ // create row
+ await tester.tapCreateRowButtonInGrid();
+
+ final actual = tester.getGridRows();
+ expect(actual.slice(0, 3), orderedEquals(expected));
+ expect(actual.length, equals(4));
+ tester.assertNumberOfRowsInGridPage(4);
+ });
+
+ testWidgets('create from a row\'s menu', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
+
+ final expected = tester.getGridRows();
+
+ // create row
+ await tester.hoverOnFirstRowOfGrid();
+ await tester.tapCreateRowButtonAfterHoveringOnGridRow();
+
+ final actual = tester.getGridRows();
+ expect([actual[0], actual[2], actual[3]], orderedEquals(expected));
+ expect(actual.length, equals(4));
+ tester.assertNumberOfRowsInGridPage(4);
+ });
+
+ testWidgets('create with sort configured', (tester) async {
+ await tester.openTestDatabase(v069GridFileName);
+
+ // get grid data
+ final unsorted = tester.getGridRows();
+
+ // add a sort
+ await tester.tapDatabaseSortButton();
+ await tester.tapCreateSortByFieldType(FieldType.RichText, 'Name');
+
+ final sorted = [
+ unsorted[7],
+ unsorted[8],
+ unsorted[1],
+ unsorted[9],
+ unsorted[11],
+ unsorted[10],
+ unsorted[6],
+ unsorted[12],
+ unsorted[2],
+ unsorted[0],
+ unsorted[3],
+ unsorted[5],
+ unsorted[4],
+ ];
+
+ List actual = tester.getGridRows();
+ expect(actual, orderedEquals(sorted));
+
+ // create row
+ await tester.hoverOnFirstRowOfGrid();
+ await tester.tapCreateRowButtonAfterHoveringOnGridRow();
+
+ // cancel
+ expect(find.byType(ConfirmPopup), findsOneWidget);
+ await tester.tapButtonWithName(LocaleKeys.button_cancel.tr());
+
+ // verify grid data
+ actual = tester.getGridRows();
+ expect(actual, orderedEquals(sorted));
+
+ // try again, but confirm this time
+ await tester.hoverOnFirstRowOfGrid();
+ await tester.tapCreateRowButtonAfterHoveringOnGridRow();
+ expect(find.byType(ConfirmPopup), findsOneWidget);
+ await tester.tapButtonWithName(LocaleKeys.button_remove.tr());
+
+ // verify grid data
+ actual = tester.getGridRows();
+ expect(actual.length, equals(14));
+ tester.assertNumberOfRowsInGridPage(14);
+ });
+
+ testWidgets('create with filter configured', (tester) async {
+ await tester.openTestDatabase(v069GridFileName);
+
+ // get grid data
+ final original = tester.getGridRows();
+
+ // create a filter
+ await tester.tapDatabaseFilterButton();
+ await tester.tapCreateFilterByFieldType(
+ FieldType.Checkbox,
+ 'Registration Complete',
+ );
+
+ final filtered = [
+ original[1],
+ original[3],
+ original[5],
+ original[6],
+ original[7],
+ original[9],
+ original[12],
+ ];
+
+ // verify grid data
+ List actual = tester.getGridRows();
+ expect(actual, orderedEquals(filtered));
+
+ // create row (one before and after the first row, and one at the bottom)
+ await tester.tapCreateRowButtonInGrid();
+ await tester.hoverOnFirstRowOfGrid();
+ await tester.tapCreateRowButtonAfterHoveringOnGridRow();
+ await tester.hoverOnFirstRowOfGrid(() async {
+ await tester.tapRowMenuButtonInGrid();
+ await tester.tapCreateRowAboveButtonInRowMenu();
+ });
+
+ actual = tester.getGridRows();
+ expect(actual.length, equals(10));
+ tester.assertNumberOfRowsInGridPage(10);
+ actual = [
+ actual[1],
+ actual[3],
+ actual[4],
+ actual[5],
+ actual[6],
+ actual[7],
+ actual[8],
+ ];
+ expect(actual, orderedEquals(filtered));
+
+ // delete the filter
+ await tester.tapFilterButtonInGrid('Registration Complete');
+ await tester
+ .tapDisclosureButtonInFinder(find.byType(CheckboxFilterEditor));
+ await tester.tapDeleteFilterButtonInGrid();
+
+ // verify grid data
+ actual = tester.getGridRows();
+ expect(actual.length, equals(16));
+ tester.assertNumberOfRowsInGridPage(16);
+ actual = [
+ actual[0],
+ actual[2],
+ actual[4],
+ actual[5],
+ actual[6],
+ actual[7],
+ actual[8],
+ actual[9],
+ actual[10],
+ actual[11],
+ actual[12],
+ actual[13],
+ actual[14],
+ ];
+ expect(actual, orderedEquals(original));
+ });
+
+ testWidgets('delete row of the grid', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
+ await tester.hoverOnFirstRowOfGrid(() async {
+ // Open the row menu and click the delete button
+ await tester.tapRowMenuButtonInGrid();
+ await tester.tapDeleteOnRowMenu();
+ });
+ expect(find.byType(ConfirmPopup), findsOneWidget);
+ await tester.tapButtonWithName(LocaleKeys.button_delete.tr());
+
+ tester.assertNumberOfRowsInGridPage(2);
+ });
+
+ testWidgets('delete row in two views', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ await tester.createNewPageWithNameUnderParent(layout: ViewLayoutPB.Grid);
+ await tester.renameLinkedView(
+ tester.findTabBarLinkViewByViewLayout(ViewLayoutPB.Grid),
+ 'grid 1',
+ );
+ tester.assertNumberOfRowsInGridPage(3);
+
+ await tester.tapCreateLinkedDatabaseViewButton(DatabaseLayoutPB.Grid);
+ await tester.renameLinkedView(
+ tester.findTabBarLinkViewByViewLayout(ViewLayoutPB.Grid).at(1),
+ 'grid 2',
+ );
+ tester.assertNumberOfRowsInGridPage(3);
+
+ await tester.hoverOnFirstRowOfGrid(() async {
+ // Open the row menu and click the delete button
+ await tester.tapRowMenuButtonInGrid();
+ await tester.tapDeleteOnRowMenu();
+ });
+ expect(find.byType(ConfirmPopup), findsOneWidget);
+ await tester.tapButtonWithName(LocaleKeys.button_delete.tr());
+ // 3 initial rows - 1 deleted
+ tester.assertNumberOfRowsInGridPage(2);
+
+ await tester.tapTabBarLinkedViewByViewName('grid 1');
+ tester.assertNumberOfRowsInGridPage(2);
+ });
+ });
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/grid/grid_test_runner_1.dart b/frontend/appflowy_flutter/integration_test/desktop/grid/grid_test_runner_1.dart
new file mode 100644
index 0000000000..ff42bb6cc2
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/grid/grid_test_runner_1.dart
@@ -0,0 +1,19 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'grid_edit_row_test.dart' as grid_edit_row_test_runner;
+import 'grid_filter_and_sort_test.dart' as grid_filter_and_sort_test_runner;
+import 'grid_reopen_test.dart' as grid_reopen_test_runner;
+import 'grid_reorder_row_test.dart' as grid_reorder_row_test_runner;
+import 'grid_row_test.dart' as grid_row_test_runner;
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ grid_reopen_test_runner.main();
+ grid_row_test_runner.main();
+ grid_reorder_row_test_runner.main();
+ grid_filter_and_sort_test_runner.main();
+ grid_edit_row_test_runner.main();
+ // grid_calculations_test_runner.main();
+ // DON'T add more tests here.
+}
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 fb3383a218..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
@@ -1,23 +1,21 @@
-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/workspace/application/settings/settings_dialog_bloc.dart';
import 'package:appflowy/workspace/presentation/settings/pages/settings_shortcuts_view.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/keyboard.dart';
import '../../shared/util.dart';
-import '../board/board_hide_groups_test.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- group('shortcuts test', () {
- testWidgets('can change and overwrite shortcut', (tester) async {
+ group('shortcuts:', () {
+ testWidgets('change and overwrite shortcut', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
@@ -29,14 +27,20 @@ void main() {
LocaleKeys.settings_shortcutsPage_keybindings_backspace.tr();
// Input "Delete" into the search field
- await tester.enterText(find.byType(TextField), backspaceCmd);
+ final inputField = find.descendant(
+ of: find.byType(SettingsShortcutsView),
+ matching: find.byType(TextField),
+ );
+ await tester.enterText(inputField, backspaceCmd);
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 4659c98b55..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,82 +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;
- }
- await tester.createNewPageWithNameUnderParent(
- name: value.name,
- parentName: gettingStarted,
- layout: value,
- );
+ tester.expectViewHasIcon(
+ value.name,
+ value,
+ emoji,
+ );
+ }
+ });
- // update its icon
- await tester.updatePageIconInTitleBarByName(
- name: value.name,
- layout: value,
- icon: emoji,
- );
+ testWidgets('Update page emoji in title bar', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
- 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.dart b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_test.dart
index 006d7ff0b6..304e8e2e35 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_test.dart
@@ -1,4 +1,3 @@
-import 'package:appflowy/generated/locale_keys.g.dart';
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';
@@ -8,7 +7,6 @@ import 'package:appflowy/workspace/presentation/home/menu/view/view_item.dart';
import 'package:appflowy/workspace/presentation/home/menu/view/view_more_action_button.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_test/flutter_test.dart';
import 'package:integration_test/integration_test.dart';
@@ -17,7 +15,7 @@ import '../../shared/util.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- group('sidebar test', () {
+ group('sidebar:', () {
testWidgets('create a new page', (tester) async {
await tester.initializeAppFlowy();
await tester.tapAnonymousSignInButton();
@@ -26,9 +24,7 @@ void main() {
await tester.tapNewPageButton();
// expect to see a new document
- tester.expectToSeePageName(
- LocaleKeys.menuAppHeader_defaultNewPageName.tr(),
- );
+ tester.expectToSeePageName('');
// and with one paragraph block
expect(find.byType(ParagraphBlockComponentWidget), findsOneWidget);
});
@@ -202,7 +198,7 @@ void main() {
layout: ViewLayoutPB.Grid,
onHover: () async {
expect(find.byType(ViewAddButton), findsNothing);
- expect(find.byType(ViewMoreActionButton), findsOneWidget);
+ expect(find.byType(ViewMoreActionPopover), findsOneWidget);
},
);
});
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 3bc41d78c0..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,7 +2,9 @@ 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;
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
@@ -12,4 +14,6 @@ void main() {
// sidebar_expanded_test.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
new file mode 100644
index 0000000000..f2b721e686
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/sidebar/sidebar_view_item_test.dart
@@ -0,0 +1,57 @@
+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';
+import 'package:flutter/gestures.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;
+ });
+
+ group('Sidebar view item tests', () {
+ testWidgets('Access view item context menu by right click', (tester) async {
+ await tester.initializeAppFlowy();
+ await tester.tapAnonymousSignInButton();
+
+ // Right click on the view item and change icon
+ await tester.hoverOnWidget(
+ find.byType(ViewItem),
+ onHover: () async {
+ await tester.tap(find.byType(ViewItem), buttons: kSecondaryButton);
+ await tester.pumpAndSettle();
+ },
+ );
+
+ // Change icon
+ final changeIconButton =
+ find.text(LocaleKeys.document_plugins_cover_changeIcon.tr());
+
+ await tester.tapButton(changeIconButton);
+ await tester.pumpUntilFound(find.byType(FlowyEmojiPicker));
+
+ const emoji = '😁';
+ await tester.tapEmoji(emoji);
+ await tester.pumpAndSettle();
+
+ 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/empty_test.dart b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/empty_test.dart
deleted file mode 100644
index 8b1756cde1..0000000000
--- a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/empty_test.dart
+++ /dev/null
@@ -1,17 +0,0 @@
-import 'package:flutter_test/flutter_test.dart';
-import 'package:integration_test/integration_test.dart';
-
-import '../../shared/util.dart';
-
-// This test is meaningless, just for preventing the CI from failing.
-void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
-
- group('Empty', () {
- testWidgets('empty test', (tester) async {
- await tester.initializeAppFlowy();
- await tester.tapAnonymousSignInButton();
- await tester.wait(1000);
- });
- });
-}
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
new file mode 100644
index 0000000000..836cfe4ccd
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/uncategorized_test_runner_1.dart
@@ -0,0 +1,20 @@
+import 'package:integration_test/integration_test.dart';
+
+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 'zoom_in_out_test.dart' as zoom_in_out_test;
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ // This test must be run first, otherwise the CI will fail.
+ hotkeys_test.main();
+ emoji_shortcut_test.main();
+ hotkeys_test.main();
+ share_markdown_test.main();
+ import_files_test.main();
+ zoom_in_out_test.main();
+ // DON'T add more tests here.
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/zoom_in_out_test.dart b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/zoom_in_out_test.dart
index e7e95edbe4..f0cddadf68 100644
--- a/frontend/appflowy_flutter/integration_test/desktop/uncategorized/zoom_in_out_test.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop/uncategorized/zoom_in_out_test.dart
@@ -12,7 +12,7 @@ import '../../shared/common_operations.dart';
void main() {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- group('Zoom in/out: ', () {
+ group('Zoom in/out:', () {
Future resetAppFlowyScaleFactor(
WindowSizeManager windowSizeManager,
) async {
diff --git a/frontend/appflowy_flutter/integration_test/desktop_runner_1.dart b/frontend/appflowy_flutter/integration_test/desktop_runner_1.dart
index e972f49fbb..c91ba21edb 100644
--- a/frontend/appflowy_flutter/integration_test/desktop_runner_1.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop_runner_1.dart
@@ -1,7 +1,6 @@
import 'package:integration_test/integration_test.dart';
-import 'desktop/document/document_test_runner.dart' as document_test_runner;
-import 'desktop/uncategorized/empty_test.dart' as first_test;
+import 'desktop/document/document_test_runner_1.dart' as document_test_runner_1;
import 'desktop/uncategorized/switch_folder_test.dart' as switch_folder_test;
Future main() async {
@@ -11,11 +10,7 @@ Future main() async {
Future runIntegration1OnDesktop() async {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- // This test must be run first, otherwise the CI will fail.
- first_test.main();
-
switch_folder_test.main();
- document_test_runner.startTesting();
-
- // DON'T add more tests here. This is the first test runner for desktop.
+ document_test_runner_1.main();
+ // DON'T add more tests here.
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop_runner_2.dart b/frontend/appflowy_flutter/integration_test/desktop_runner_2.dart
index 576cfc1e99..99d6f7d58f 100644
--- a/frontend/appflowy_flutter/integration_test/desktop_runner_2.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop_runner_2.dart
@@ -1,18 +1,7 @@
import 'package:integration_test/integration_test.dart';
-import 'desktop/database/database_calendar_test.dart' as database_calendar_test;
-import 'desktop/database/database_cell_test.dart' as database_cell_test;
-import 'desktop/database/database_field_settings_test.dart'
- as database_field_settings_test;
-import 'desktop/database/database_field_test.dart' as database_field_test;
-import 'desktop/database/database_filter_test.dart' as database_filter_test;
-import 'desktop/database/database_media_test.dart' as database_media_test;
-import 'desktop/database/database_row_page_test.dart' as database_row_page_test;
-import 'desktop/database/database_setting_test.dart' as database_setting_test;
-import 'desktop/database/database_share_test.dart' as database_share_test;
-import 'desktop/database/database_sort_test.dart' as database_sort_test;
-import 'desktop/database/database_view_test.dart' as database_view_test;
-import 'desktop/uncategorized/empty_test.dart' as first_test;
+import 'desktop/database/database_test_runner_1.dart' as database_test_runner_1;
+import 'desktop/first_test/first_test.dart' as first_test;
Future main() async {
await runIntegration2OnDesktop();
@@ -21,20 +10,8 @@ Future main() async {
Future runIntegration2OnDesktop() async {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- // This test must be run first, otherwise the CI will fail.
first_test.main();
- database_cell_test.main();
- database_field_test.main();
- database_field_settings_test.main();
- database_share_test.main();
- database_row_page_test.main();
- database_setting_test.main();
- database_filter_test.main();
- database_sort_test.main();
- database_view_test.main();
- database_calendar_test.main();
- database_media_test.main();
-
+ database_test_runner_1.main();
// DON'T add more tests here. This is the second test runner for desktop.
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop_runner_3.dart b/frontend/appflowy_flutter/integration_test/desktop_runner_3.dart
index 3ddfb0c4d0..a9d3783f1d 100644
--- a/frontend/appflowy_flutter/integration_test/desktop_runner_3.dart
+++ b/frontend/appflowy_flutter/integration_test/desktop_runner_3.dart
@@ -1,23 +1,8 @@
import 'package:integration_test/integration_test.dart';
import 'desktop/board/board_test_runner.dart' as board_test_runner;
-import 'desktop/database/database_row_cover_test.dart'
- as database_row_cover_test;
-import 'desktop/grid/grid_create_row_test.dart' as grid_create_row_test_runner;
-import 'desktop/grid/grid_filter_and_sort_test.dart'
- as grid_filter_and_sort_test_runner;
-import 'desktop/grid/grid_reopen_test.dart' as grid_reopen_test_runner;
-import 'desktop/grid/grid_reorder_row_test.dart'
- as grid_reorder_row_test_runner;
-import 'desktop/settings/settings_runner.dart' as settings_test_runner;
-import 'desktop/sidebar/sidebar_test_runner.dart' as sidebar_test_runner;
-import 'desktop/uncategorized/emoji_shortcut_test.dart' as emoji_shortcut_test;
-import 'desktop/uncategorized/empty_test.dart' as first_test;
-import 'desktop/uncategorized/hotkeys_test.dart' as hotkeys_test;
-import 'desktop/uncategorized/import_files_test.dart' as import_files_test;
-import 'desktop/uncategorized/share_markdown_test.dart' as share_markdown_test;
-import 'desktop/uncategorized/tabs_test.dart' as tabs_test;
-import 'desktop/uncategorized/zoom_in_out_test.dart' as zoom_in_out_test;
+import 'desktop/first_test/first_test.dart' as first_test;
+import 'desktop/grid/grid_test_runner_1.dart' as grid_test_runner_1;
Future main() async {
await runIntegration3OnDesktop();
@@ -26,24 +11,9 @@ Future main() async {
Future runIntegration3OnDesktop() async {
IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- // This test must be run first, otherwise the CI will fail.
first_test.main();
- hotkeys_test.main();
- emoji_shortcut_test.main();
- hotkeys_test.main();
- emoji_shortcut_test.main();
- settings_test_runner.main();
- share_markdown_test.main();
- import_files_test.main();
- sidebar_test_runner.main();
board_test_runner.main();
- tabs_test.main();
- database_row_cover_test.main();
- grid_reopen_test_runner.main();
- grid_create_row_test_runner.main();
- grid_reorder_row_test_runner.main();
- grid_filter_and_sort_test_runner.main();
-
- zoom_in_out_test.main();
+ grid_test_runner_1.main();
+ // DON'T add more tests here.
}
diff --git a/frontend/appflowy_flutter/integration_test/desktop_runner_4.dart b/frontend/appflowy_flutter/integration_test/desktop_runner_4.dart
new file mode 100644
index 0000000000..e51c711549
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop_runner_4.dart
@@ -0,0 +1,17 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'desktop/document/document_test_runner_2.dart' as document_test_runner_2;
+import 'desktop/first_test/first_test.dart' as first_test;
+
+Future main() async {
+ await runIntegration4OnDesktop();
+}
+
+Future runIntegration4OnDesktop() async {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ first_test.main();
+
+ document_test_runner_2.main();
+ // DON'T add more tests here.
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop_runner_5.dart b/frontend/appflowy_flutter/integration_test/desktop_runner_5.dart
new file mode 100644
index 0000000000..be393e90c7
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop_runner_5.dart
@@ -0,0 +1,17 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'desktop/database/database_test_runner_2.dart' as database_test_runner_2;
+import 'desktop/first_test/first_test.dart' as first_test;
+
+Future main() async {
+ await runIntegration5OnDesktop();
+}
+
+Future runIntegration5OnDesktop() async {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ first_test.main();
+
+ database_test_runner_2.main();
+ // DON'T add more tests here.
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop_runner_6.dart b/frontend/appflowy_flutter/integration_test/desktop_runner_6.dart
new file mode 100644
index 0000000000..a1c5627b20
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop_runner_6.dart
@@ -0,0 +1,22 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'desktop/first_test/first_test.dart' as first_test;
+import 'desktop/settings/settings_runner.dart' as settings_test_runner;
+import 'desktop/sidebar/sidebar_test_runner.dart' as sidebar_test_runner;
+import 'desktop/uncategorized/uncategorized_test_runner_1.dart'
+ as uncategorized_test_runner_1;
+
+Future main() async {
+ await runIntegration6OnDesktop();
+}
+
+Future runIntegration6OnDesktop() async {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ first_test.main();
+
+ settings_test_runner.main();
+ sidebar_test_runner.main();
+ uncategorized_test_runner_1.main();
+ // DON'T add more tests here.
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop_runner_7.dart b/frontend/appflowy_flutter/integration_test/desktop_runner_7.dart
new file mode 100644
index 0000000000..0200591c57
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop_runner_7.dart
@@ -0,0 +1,17 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'desktop/document/document_test_runner_3.dart' as document_test_runner_3;
+import 'desktop/first_test/first_test.dart' as first_test;
+
+Future main() async {
+ await runIntegration7OnDesktop();
+}
+
+Future runIntegration7OnDesktop() async {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ first_test.main();
+
+ document_test_runner_3.main();
+ // DON'T add more tests here.
+}
diff --git a/frontend/appflowy_flutter/integration_test/desktop_runner_8.dart b/frontend/appflowy_flutter/integration_test/desktop_runner_8.dart
new file mode 100644
index 0000000000..5a706e5dec
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/desktop_runner_8.dart
@@ -0,0 +1,17 @@
+import 'package:integration_test/integration_test.dart';
+
+import 'desktop/document/document_test_runner_4.dart' as document_test_runner_4;
+import 'desktop/first_test/first_test.dart' as first_test;
+
+Future main() async {
+ await runIntegration8OnDesktop();
+}
+
+Future runIntegration8OnDesktop() async {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ first_test.main();
+
+ document_test_runner_4.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
new file mode 100644
index 0000000000..c2f3d7103a
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/cloud/cloud_runner.dart
@@ -0,0 +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
new file mode 100644
index 0000000000..e6015d0896
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/cloud/document/publish_test.dart
@@ -0,0 +1,110 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/mobile/presentation/home/workspaces/create_workspace_menu.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/constants.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('publish:', () {
+ testWidgets('''
+1. publish document
+2. update path name
+3. unpublish document
+''', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ await tester.openPage(Constants.gettingStartedPageName);
+ await tester.editor.openMoreActionMenuOnMobile();
+
+ // click the publish button
+ await tester.editor.clickMoreActionItemOnMobile(
+ LocaleKeys.shareAction_publish.tr(),
+ );
+
+ // wait the notification dismiss
+ final publishSuccessText = find.findTextInFlowyText(
+ LocaleKeys.publish_publishSuccessfully.tr(),
+ );
+ expect(publishSuccessText, findsOneWidget);
+ await tester.pumpUntilNotFound(publishSuccessText);
+
+ // open the menu again, to check the publish status
+ await tester.editor.openMoreActionMenuOnMobile();
+ // expect to see the unpublish button and the visit site button
+ expect(
+ find.text(LocaleKeys.shareAction_unPublish.tr()),
+ findsOneWidget,
+ );
+ expect(
+ find.text(LocaleKeys.shareAction_visitSite.tr()),
+ 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(),
+ );
+ final unPublishSuccessText = find.findTextInFlowyText(
+ LocaleKeys.publish_unpublishSuccessfully.tr(),
+ );
+ expect(unPublishSuccessText, findsOneWidget);
+ await tester.pumpUntilNotFound(unPublishSuccessText);
+ });
+ });
+}
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
new file mode 100644
index 0000000000..bf0ddc8711
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/cloud/document/share_link_test.dart
@@ -0,0 +1,42 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/shared/patterns/common_patterns.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/constants.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('share link:', () {
+ testWidgets('copy share link and paste it on doc', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // open the getting started page and paste the link
+ await tester.openPage(Constants.gettingStartedPageName);
+
+ // open the more action menu
+ await tester.editor.openMoreActionMenuOnMobile();
+
+ // click the share link item
+ await tester.editor.clickMoreActionItemOnMobile(
+ LocaleKeys.shareAction_copyLink.tr(),
+ );
+
+ // check the clipboard
+ final content = await Clipboard.getData(Clipboard.kTextPlain);
+ expect(
+ content?.text,
+ matches(appflowySharePageLinkPattern),
+ );
+ });
+ });
+}
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
new file mode 100644
index 0000000000..210d1bcf0e
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/cloud/workspace/workspace_operations_test.dart
@@ -0,0 +1,41 @@
+import 'package:appflowy/env/cloud_env.dart';
+import 'package:appflowy/generated/locale_keys.g.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/constants.dart';
+import '../../../shared/util.dart';
+
+void main() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+
+ group('workspace operations:', () {
+ testWidgets('create a new workspace', (tester) async {
+ await tester.initializeAppFlowy(
+ cloudType: AuthenticatorType.appflowyCloudSelfHost,
+ );
+ await tester.tapGoogleLoginInButton();
+ await tester.expectToSeeHomePageWithGetStartedPage();
+
+ // click the create a new workspace button
+ await tester.tapButton(find.text(Constants.defaultWorkspaceName));
+ await tester.tapButton(find.text(LocaleKeys.workspace_create.tr()));
+
+ // input the new workspace name
+ final inputField = find.byType(TextFormField);
+ const newWorkspaceName = 'AppFlowy';
+ await tester.enterText(inputField, newWorkspaceName);
+ await tester.pumpAndSettle();
+
+ // wait for the workspace to be created
+ await tester.pumpUntilFound(
+ find.text(LocaleKeys.workspace_createSuccess.tr()),
+ );
+
+ // expect to see the new workspace
+ expect(find.text(newWorkspaceName), findsOneWidget);
+ });
+ });
+}
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
new file mode 100644
index 0000000000..90d5ca6d0d
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/document_test_runner.dart
@@ -0,0 +1,24 @@
+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();
+
+ // 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
new file mode 100644
index 0000000000..e3d3bc093f
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/page_style_test.dart
@@ -0,0 +1,163 @@
+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/view_page/app_bar_buttons.dart';
+import 'package:appflowy/mobile/presentation/bottom_sheet/bottom_sheet_buttons.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/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: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() {
+ setUpAll(() {
+ IntegrationTestWidgetsFlutterBinding.ensureInitialized();
+ RecentIcons.enable = false;
+ });
+
+ tearDownAll(() {
+ RecentIcons.enable = true;
+ });
+
+ group('document page style:', () {
+ double getCurrentEditorFontSize() {
+ final editorPage = find
+ .byType(AppFlowyEditorPage)
+ .evaluate()
+ .single
+ .widget as AppFlowyEditorPage;
+ return editorPage.styleCustomizer
+ .style()
+ .textStyleConfiguration
+ .text
+ .fontSize!;
+ }
+
+ double getCurrentEditorLineHeight() {
+ final editorPage = find
+ .byType(AppFlowyEditorPage)
+ .evaluate()
+ .single
+ .widget as AppFlowyEditorPage;
+ return editorPage.styleCustomizer
+ .style()
+ .textStyleConfiguration
+ .lineHeight;
+ }
+
+ testWidgets('change font size in page style settings', (tester) async {
+ await tester.launchInAnonymousMode();
+
+ // click the getting start page
+ await tester.openPage(gettingStarted);
+ // click the layout button
+ await tester.tapButton(find.byType(MobileViewPageLayoutButton));
+ expect(getCurrentEditorFontSize(), PageStyleFontLayout.normal.fontSize);
+ // change font size from normal to large
+ await tester.tapSvgButton(FlowySvgs.m_font_size_large_s);
+ expect(getCurrentEditorFontSize(), PageStyleFontLayout.large.fontSize);
+ // change font size from large to small
+ await tester.tapSvgButton(FlowySvgs.m_font_size_small_s);
+ expect(getCurrentEditorFontSize(), PageStyleFontLayout.small.fontSize);
+ });
+
+ testWidgets('change line height in page style settings', (tester) async {
+ await tester.launchInAnonymousMode();
+
+ // click the getting start page
+ await tester.openPage(gettingStarted);
+ // click the layout button
+ await tester.tapButton(find.byType(MobileViewPageLayoutButton));
+ var lineHeight = getCurrentEditorLineHeight();
+ expect(
+ 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(
+ lineHeight,
+ PageStyleLineHeightLayout.large.lineHeight,
+ );
+ // change line height from large to small
+ await tester.tapSvgButton(FlowySvgs.m_layout_small_s);
+ lineHeight = getCurrentEditorLineHeight();
+ expect(
+ lineHeight,
+ PageStyleLineHeightLayout.small.lineHeight,
+ );
+ });
+
+ testWidgets('use built-in image as cover', (tester) async {
+ await tester.launchInAnonymousMode();
+
+ // click the getting start page
+ await tester.openPage(gettingStarted);
+ // click the layout button
+ await tester.tapButton(find.byType(MobileViewPageLayoutButton));
+ // toggle the preset button
+ await tester.tapSvgButton(FlowySvgs.m_page_style_presets_m);
+
+ // select the first preset
+ final firstBuiltInImage = find.byWidgetPredicate(
+ (widget) =>
+ widget is Image &&
+ widget.image is AssetImage &&
+ (widget.image as AssetImage).assetName ==
+ PageStyleCoverImageType.builtInImagePath('1'),
+ );
+ await tester.tap(firstBuiltInImage);
+
+ // click done button to exit the page style settings
+ await tester.tapButton(find.byType(BottomSheetDoneButton).first);
+
+ // check the cover
+ final builtInCover = find.descendant(
+ of: find.byType(DocumentImmersiveCover),
+ matching: firstBuiltInImage,
+ );
+ 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
new file mode 100644
index 0000000000..01b1d574ce
--- /dev/null
+++ b/frontend/appflowy_flutter/integration_test/mobile/document/title_test.dart
@@ -0,0 +1,47 @@
+import 'package:appflowy/mobile/presentation/presentation.dart';
+import 'package:appflowy/plugins/document/presentation/editor_page.dart';
+import 'package:appflowy_editor/appflowy_editor.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 title:', () {
+ testWidgets('create a new page, the title should be empty', (tester) async {
+ await tester.launchInAnonymousMode();
+
+ final createPageButton = find.byKey(
+ BottomNavigationBarItemType.add.valueKey,
+ );
+ await tester.tapButton(createPageButton);
+ expect(find.byType(MobileDocumentScreen), findsOneWidget);
+
+ final title = tester.editor.findDocumentTitle('');
+ expect(title, findsOneWidget);
+ final textField = tester.widget(title);
+ expect(textField.focusNode!.hasFocus, isTrue);
+
+ // input new name and press done button
+ const name = 'test document';
+ await tester.enterText(title, name);
+ await tester.testTextInput.receiveAction(TextInputAction.done);
+ await tester.pumpAndSettle();
+ final newTitle = tester.editor.findDocumentTitle(name);
+ expect(newTitle, findsOneWidget);
+ expect(textField.controller!.text, name);
+
+ // the document should get focus
+ final editor = tester.widget(
+ find.byType(AppFlowyEditorPage),
+ );
+ expect(
+ editor.editorState.selection,
+ Selection.collapsed(Position(path: [0])),
+ );
+ });
+ });
+}
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 8e3724a583..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/locale_keys.g.dart';
-import 'package:appflowy/mobile/presentation/editor/mobile_editor_screen.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/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:appflowy/generated/flowy_svgs.g.dart';
+import 'package:appflowy/mobile/presentation/presentation.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,
- );
-
- // click the anonymousSignInButton
- final anonymousSignInButton = find.byType(SignInAnonymousButtonV2);
- expect(anonymousSignInButton, findsOneWidget);
- await tester.tapButton(anonymousSignInButton);
+ await tester.launchInAnonymousMode();
// tap the create page button
- final createPageButton = find.byKey(mobileCreateNewPageButtonKey);
+ final createPageButton = find.byWidgetPredicate(
+ (widget) =>
+ widget is FlowySvg &&
+ 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/page_style/document_page_style_test.dart b/frontend/appflowy_flutter/integration_test/mobile/page_style/document_page_style_test.dart
deleted file mode 100644
index c915ebadfd..0000000000
--- a/frontend/appflowy_flutter/integration_test/mobile/page_style/document_page_style_test.dart
+++ /dev/null
@@ -1,139 +0,0 @@
-// 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/dir.dart';
-import '../../shared/mock/mock_file_picker.dart';
-import '../../shared/util.dart';
-
-void main() {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
-
- group('document page style', () {
- double getCurrentEditorFontSize() {
- final editorPage = find
- .byType(AppFlowyEditorPage)
- .evaluate()
- .single
- .widget as AppFlowyEditorPage;
- return editorPage.styleCustomizer
- .style()
- .textStyleConfiguration
- .text
- .fontSize!;
- }
-
- double getCurrentEditorLineHeight() {
- final editorPage = find
- .byType(AppFlowyEditorPage)
- .evaluate()
- .single
- .widget as AppFlowyEditorPage;
- return editorPage.styleCustomizer
- .style()
- .textStyleConfiguration
- .text
- .height!;
- }
-
- testWidgets('change font size in page style settings', (tester) async {
- await tester.launchInAnonymousMode();
-
- // click the getting start page
- await tester.openPage(gettingStarted);
- // click the layout button
- await tester.tapButton(find.byType(MobileViewPageLayoutButton));
- expect(getCurrentEditorFontSize(), PageStyleFontLayout.normal.fontSize);
- // change font size from normal to large
- await tester.tapSvgButton(FlowySvgs.m_font_size_large_s);
- expect(getCurrentEditorFontSize(), PageStyleFontLayout.large.fontSize);
- // change font size from large to small
- await tester.tapSvgButton(FlowySvgs.m_font_size_small_s);
- expect(getCurrentEditorFontSize(), PageStyleFontLayout.small.fontSize);
- });
-
- testWidgets('change line height in page style settings', (tester) async {
- await tester.launchInAnonymousMode();
-
- // click the getting start page
- await tester.openPage(gettingStarted);
- // click the layout button
- await tester.tapButton(find.byType(MobileViewPageLayoutButton));
- expect(
- getCurrentEditorLineHeight(),
- PageStyleLineHeightLayout.normal.lineHeight,
- );
- // change line height from normal to large
- await tester.tapSvgButton(FlowySvgs.m_layout_large_s);
- expect(
- getCurrentEditorLineHeight(),
- PageStyleLineHeightLayout.large.lineHeight,
- );
- // change line height from large to small
- await tester.tapSvgButton(FlowySvgs.m_layout_small_s);
- expect(
- getCurrentEditorLineHeight(),
- PageStyleLineHeightLayout.small.lineHeight,
- );
- });
-
- testWidgets('use built-in image as cover', (tester) async {
- await tester.launchInAnonymousMode();
-
- // click the getting start page
- await tester.openPage(gettingStarted);
- // click the layout button
- await tester.tapButton(find.byType(MobileViewPageLayoutButton));
- // toggle the preset button
- await tester.tapSvgButton(FlowySvgs.m_page_style_presets_m);
-
- // select the first preset
- final firstBuiltInImage = find.byWidgetPredicate(
- (widget) =>
- widget is Image &&
- widget.image is AssetImage &&
- (widget.image as AssetImage).assetName ==
- PageStyleCoverImageType.builtInImagePath('1'),
- );
- await tester.tap(firstBuiltInImage);
-
- // click done button to exit the page style settings
- await tester.tapButton(find.byType(BottomSheetDoneButton).first);
- await tester.tapButton(find.byType(BottomSheetDoneButton).first);
-
- // check the cover
- final builtInCover = find.descendant(
- of: find.byType(DocumentImmersiveCover),
- matching: firstBuiltInImage,
- );
- expect(builtInCover, 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 65f48a87ff..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,38 +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();
-
- // click the anonymousSignInButton
- final anonymousSignInButton = find.byType(SignInAnonymousButtonV2);
- expect(anonymousSignInButton, findsOneWidget);
- await tester.tapButton(anonymousSignInButton);
+ 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 9ebc2dcd97..0000000000
--- a/frontend/appflowy_flutter/integration_test/mobile_runner.dart
+++ /dev/null
@@ -1,10 +0,0 @@
-import 'package:integration_test/integration_test.dart';
-
-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 {
- IntegrationTestWidgetsFlutterBinding.ensureInitialized();
- anonymous_sign_in_test.main();
- create_new_page_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 cb7d2d6e33..0fc3c5d826 100644
--- a/frontend/appflowy_flutter/integration_test/runner.dart
+++ b/frontend/appflowy_flutter/integration_test/runner.dart
@@ -3,7 +3,13 @@ import 'dart:io';
import 'desktop_runner_1.dart';
import 'desktop_runner_2.dart';
import 'desktop_runner_3.dart';
-import 'mobile_runner.dart';
+import 'desktop_runner_4.dart';
+import 'desktop_runner_5.dart';
+import 'desktop_runner_6.dart';
+import 'desktop_runner_7.dart';
+import 'desktop_runner_8.dart';
+import 'desktop_runner_9.dart';
+import 'mobile_runner_1.dart';
/// The main task runner for all integration tests in AppFlowy.
///
@@ -17,8 +23,14 @@ Future main() async {
await runIntegration1OnDesktop();
await runIntegration2OnDesktop();
await runIntegration3OnDesktop();
+ await runIntegration4OnDesktop();
+ await runIntegration5OnDesktop();
+ 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/auth_operation.dart b/frontend/appflowy_flutter/integration_test/shared/auth_operation.dart
index 1f2f23dc2c..88f9634afd 100644
--- a/frontend/appflowy_flutter/integration_test/shared/auth_operation.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/auth_operation.dart
@@ -10,9 +10,10 @@ import 'package:flutter_test/flutter_test.dart';
import 'util.dart';
extension AppFlowyAuthTest on WidgetTester {
- Future tapGoogleLoginInButton() async {
+ Future tapGoogleLoginInButton({bool pumpAndSettle = true}) async {
await tapButton(
find.byKey(signInWithGoogleButtonKey),
+ pumpAndSettle: pumpAndSettle,
);
}
diff --git a/frontend/appflowy_flutter/integration_test/shared/base.dart b/frontend/appflowy_flutter/integration_test/shared/base.dart
index 371cd9b839..493cb4c1f0 100644
--- a/frontend/appflowy_flutter/integration_test/shared/base.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/base.dart
@@ -13,10 +13,12 @@ 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;
import 'package:path_provider/path_provider.dart';
+import 'package:universal_platform/universal_platform.dart';
class FlowyTestContext {
FlowyTestContext({required this.applicationDataDirectory});
@@ -105,7 +107,7 @@ extension AppFlowyTestBase on WidgetTester {
}
Future waitUntilSignInPageShow() async {
- if (isAuthEnabled) {
+ if (isAuthEnabled || UniversalPlatform.isMobile) {
final finder = find.byType(SignInAnonymousButtonV2);
await pumpUntilFound(finder, timeout: const Duration(seconds: 30));
expect(finder, findsOneWidget);
@@ -158,23 +160,45 @@ extension AppFlowyTestBase on WidgetTester {
Future tapButton(
Finder finder, {
- int? pointer,
int buttons = kPrimaryButton,
bool warnIfMissed = false,
int milliseconds = 500,
bool pumpAndSettle = true,
}) async {
- await tap(
- finder,
- buttons: buttons,
- warnIfMissed: warnIfMissed,
- );
+ await tap(finder, buttons: buttons, warnIfMissed: warnIfMissed);
if (pumpAndSettle) {
await this.pumpAndSettle(
Duration(milliseconds: milliseconds),
EnginePhase.sendSemanticsUpdate,
- const Duration(seconds: 5),
+ const Duration(seconds: 15),
+ );
+ }
+ }
+
+ 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),
);
}
}
@@ -212,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 8c33468d3b..d7a505d152 100644
--- a/frontend/appflowy_flutter/integration_test/shared/common_operations.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/common_operations.dart
@@ -4,16 +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';
@@ -38,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';
@@ -52,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 {
@@ -182,6 +195,21 @@ extension CommonOperations on WidgetTester {
}
}
+ /// Right click on the page name.
+ Future rightClickOnPageName(
+ String name, {
+ ViewLayoutPB layout = ViewLayoutPB.Document,
+ }) async {
+ final page = findPageName(name, layout: layout);
+ await hoverOnPageName(
+ name,
+ onHover: () async {
+ await tap(page, buttons: kSecondaryMouseButton);
+ await pumpAndSettle();
+ },
+ );
+ }
+
/// open the page with given name.
Future openPage(
String name, {
@@ -196,7 +224,10 @@ extension CommonOperations on WidgetTester {
///
/// Must call [hoverOnPageName] first.
Future tapPageOptionButton() async {
- final optionButton = find.byType(ViewMoreActionButton);
+ final optionButton = find.descendant(
+ of: find.byType(ViewMoreActionPopover),
+ matching: find.byFlowySvg(FlowySvgs.workspace_three_dots_s),
+ );
await tapButton(optionButton);
}
@@ -237,6 +268,10 @@ extension CommonOperations on WidgetTester {
await tapOKButton();
}
+ Future tapTrashButton() async {
+ await tap(find.byType(SidebarTrashButton));
+ }
+
Future tapOKButton() async {
final okButton = find.byWidgetPredicate(
(widget) =>
@@ -253,7 +288,10 @@ extension CommonOperations on WidgetTester {
}) async {
final page = findPageName(pageName, layout: layout);
await hoverOnWidget(page);
- final expandButton = find.byType(ViewItemDefaultLeftIcon);
+ final expandButton = find.descendant(
+ of: page,
+ matching: find.byType(ViewItemDefaultLeftIcon),
+ );
await tapButton(expandButton.first);
}
@@ -269,12 +307,14 @@ extension CommonOperations on WidgetTester {
/// Tap the delete permanently button.
///
- /// the restore button will show after the current page is deleted.
+ /// the delete permanently button will show after the current page is deleted.
Future tapDeletePermanentlyButton() async {
- final restoreButton = find.textContaining(
+ final deleteButton = find.textContaining(
LocaleKeys.deletePagePrompt_deletePermanent.tr(),
);
- await tapButton(restoreButton);
+ await tapButton(deleteButton);
+ await tap(find.text(LocaleKeys.button_delete.tr()));
+ await pumpAndSettle();
}
/// Tap the share button above the document page.
@@ -285,6 +325,15 @@ extension CommonOperations on WidgetTester {
await tapButton(shareButton);
}
+ // open the share menu and then click the publish tab
+ Future openPublishMenu() async {
+ await tapShareButton();
+ final publishButton = find.textContaining(
+ LocaleKeys.shareAction_publishTab.tr(),
+ );
+ await tapButton(publishButton);
+ }
+
/// Tap the export markdown button
///
/// Must call [tapShareButton] first.
@@ -317,7 +366,7 @@ extension CommonOperations on WidgetTester {
// hover on it and change it's name
if (name != null) {
await hoverOnPageName(
- LocaleKeys.menuAppHeader_defaultNewPageName.tr(),
+ layout.defaultName,
layout: layout,
onHover: () async {
await renamePage(name);
@@ -331,13 +380,39 @@ extension CommonOperations on WidgetTester {
if (openAfterCreated) {
await openPage(
// if the name is null, use the default name
- name ?? LocaleKeys.menuAppHeader_defaultNewPageName.tr(),
+ name ?? layout.defaultName,
layout: layout,
);
await pumpAndSettle();
}
}
+ Future createOpenRenameDocumentUnderParent({
+ required String name,
+ String? parentName,
+ }) async {
+ // create a new page
+ await tapAddViewButton(name: parentName ?? gettingStarted);
+ await tapButtonWithName(ViewLayoutPB.Document.menuName);
+ final settingsOrFailure = await getIt().getWithFormat(
+ KVKeys.showRenameDialogWhenCreatingNewFile,
+ (value) => bool.parse(value),
+ );
+ final showRenameDialog = settingsOrFailure ?? false;
+ if (showRenameDialog) {
+ await tapOKButton();
+ }
+ await pumpAndSettle();
+
+ // open the page after created
+ await openPage(ViewLayoutPB.Document.defaultName);
+ await pumpAndSettle();
+
+ // Enter new name in the document title
+ await enterText(find.byType(TextFieldWithMetricLines), name);
+ await pumpAndSettle();
+ }
+
/// Create a new page in the space
Future createNewPageInSpace({
required String spaceName,
@@ -368,7 +443,7 @@ extension CommonOperations on WidgetTester {
// hover on new created page and change it's name
await hoverOnPageName(
- LocaleKeys.menuAppHeader_defaultNewPageName.tr(),
+ '',
layout: layout,
onHover: () async {
await renamePage(pageName);
@@ -380,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();
}
}
@@ -398,6 +470,19 @@ extension CommonOperations on WidgetTester {
await tapButton(addPageButton);
}
+ /// Click the + button in the space header
+ Future clickSpaceHeader() async {
+ await tapButton(find.byType(SidebarSpaceHeader));
+ }
+
+ Future openSpace(String spaceName) async {
+ final space = find.descendant(
+ of: find.byType(SidebarSpaceMenuItem),
+ matching: find.text(spaceName),
+ );
+ await tapButton(space);
+ }
+
/// Create a new page on the top level
Future createNewPage({
ViewLayoutPB layout = ViewLayoutPB.Document,
@@ -516,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(
@@ -527,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(
@@ -541,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();
}
@@ -549,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,
@@ -561,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();
}
@@ -629,7 +760,11 @@ extension CommonOperations on WidgetTester {
expect(createWorkspaceDialog, findsOneWidget);
// input the workspace name
- await enterText(find.byType(TextField), name);
+ final workspaceNameInput = find.descendant(
+ of: createWorkspaceDialog,
+ matching: find.byType(TextField),
+ );
+ await enterText(workspaceNameInput, name);
await tapButtonWithName(LocaleKeys.button_ok.tr(), pumpAndSettle: false);
await pump(const Duration(seconds: 5));
@@ -692,6 +827,172 @@ extension CommonOperations on WidgetTester {
await tap(button);
await pump();
}
+
+ Future tapFileUploadHint() async {
+ final finder = find.byWidgetPredicate(
+ (w) =>
+ w is RichText &&
+ w.text.toPlainText().contains(
+ LocaleKeys.document_plugins_file_fileUploadHint.tr(),
+ ),
+ );
+ 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 {
@@ -720,6 +1021,25 @@ extension SettingsFinder on CommonFinders {
.first;
}
+extension FlowySvgFinder on CommonFinders {
+ Finder byFlowySvg(FlowySvgData svg) => _FlowySvgFinder(svg);
+}
+
+class _FlowySvgFinder extends MatchFinder {
+ _FlowySvgFinder(this.svg);
+
+ final FlowySvgData svg;
+
+ @override
+ String get description => 'flowy_svg "$svg"';
+
+ @override
+ bool matches(Element candidate) {
+ final Widget widget = candidate.widget;
+ return widget is FlowySvg && widget.svg == svg;
+ }
+}
+
extension ViewLayoutPBTest on ViewLayoutPB {
String get menuName {
switch (this) {
diff --git a/frontend/appflowy_flutter/integration_test/shared/constants.dart b/frontend/appflowy_flutter/integration_test/shared/constants.dart
index ffb0109355..bfe3349b10 100644
--- a/frontend/appflowy_flutter/integration_test/shared/constants.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/constants.dart
@@ -3,4 +3,6 @@ class Constants {
static const gettingStartedPageName = 'Getting started';
static const toDosPageName = 'To-dos';
static const generalSpaceName = 'General';
+
+ static const defaultWorkspaceName = 'My Workspace';
}
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 c76c41f62a..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,12 +1,9 @@
import 'dart:io';
-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';
@@ -20,13 +17,15 @@ import 'package:appflowy/plugins/database/grid/presentation/widgets/calculations
import 'package:appflowy/plugins/database/grid/presentation/widgets/calculations/calculation_type_item.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/common/type_option_separator.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/checkbox.dart';
-import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/choicechip/checklist/checklist.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/create_filter_list.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/disclosure_button.dart';
-import 'package:appflowy/plugins/database/grid/presentation/widgets/filter/filter_menu_item.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/footer/grid_footer.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/header/desktop_field_cell.dart';
import 'package:appflowy/plugins/database/grid/presentation/widgets/row/row.dart';
@@ -38,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';
@@ -49,7 +49,7 @@ import 'package:appflowy/plugins/database/widgets/cell/editable_cell_skeleton/ti
import 'package:appflowy/plugins/database/widgets/cell/editable_cell_skeleton/url.dart';
import 'package:appflowy/plugins/database/widgets/cell_editor/checklist_cell_editor.dart';
import 'package:appflowy/plugins/database/widgets/cell_editor/checklist_progress_bar.dart';
-import 'package:appflowy/plugins/database/widgets/cell_editor/date_editor.dart';
+import 'package:appflowy/plugins/database/widgets/cell_editor/date_cell_editor.dart';
import 'package:appflowy/plugins/database/widgets/cell_editor/extension.dart';
import 'package:appflowy/plugins/database/widgets/cell_editor/media_cell_editor.dart';
import 'package:appflowy/plugins/database/widgets/cell_editor/select_option_cell_editor.dart';
@@ -70,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';
@@ -84,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
@@ -147,6 +152,7 @@ extension AppFlowyDatabaseTest on WidgetTester {
expect(cell, findsOneWidget);
await enterText(cell, input);
+ await testTextInput.receiveAction(TextInputAction.done);
await pumpAndSettle();
}
@@ -242,10 +248,10 @@ extension AppFlowyDatabaseTest on WidgetTester {
}
}
- Future assertMultiSelectOption({
+ void assertMultiSelectOption({
required int rowIndex,
required List contents,
- }) async {
+ }) {
final findCell = cellFinder(rowIndex, FieldType.MultiSelect);
for (final content in contents) {
if (content.isNotEmpty) {
@@ -406,17 +412,20 @@ extension AppFlowyDatabaseTest on WidgetTester {
}
Future selectOption({required String name}) async {
- final option = find.byWidgetPredicate(
- (widget) => widget is SelectOptionTagCell && widget.option.name == name,
+ final option = find.descendant(
+ of: find.byType(SelectOptionCellEditor),
+ matching: find.byWidgetPredicate(
+ (widget) => widget is SelectOptionTagCell && widget.option.name == name,
+ ),
);
await tapButton(option);
}
- Future findSelectOptionWithNameInGrid({
+ void findSelectOptionWithNameInGrid({
required int rowIndex,
required String name,
- }) async {
+ }) {
final findRow = find.byType(GridRow);
final option = find.byWidgetPredicate(
(widget) =>
@@ -428,10 +437,10 @@ extension AppFlowyDatabaseTest on WidgetTester {
expect(cell, findsOneWidget);
}
- Future assertNumberOfSelectedOptionsInGrid({
+ void assertNumberOfSelectedOptionsInGrid({
required int rowIndex,
required Matcher matcher,
- }) async {
+ }) {
final findRow = find.byType(GridRow);
final options = find.byWidgetPredicate(
@@ -499,6 +508,7 @@ extension AppFlowyDatabaseTest on WidgetTester {
Future renameChecklistTask({
required int index,
required String name,
+ bool enter = true,
}) async {
final textField = find
.descendant(
@@ -508,7 +518,9 @@ extension AppFlowyDatabaseTest on WidgetTester {
.at(index);
await enterText(textField, name);
- await testTextInput.receiveAction(TextInputAction.done);
+ if (enter) {
+ await testTextInput.receiveAction(TextInputAction.done);
+ }
await pumpAndSettle();
}
@@ -526,14 +538,38 @@ extension AppFlowyDatabaseTest on WidgetTester {
Future deleteChecklistTask({required int index}) async {
final task = find.byType(ChecklistItem).at(index);
- await startGesture(getCenter(task), kind: PointerDeviceKind.mouse);
- await pumpAndSettle();
-
- final button = find.byWidgetPredicate(
- (widget) => widget is FlowySvg && widget.svg == FlowySvgs.delete_s,
+ await hoverOnWidget(
+ task,
+ onHover: () async {
+ final button = find.byWidgetPredicate(
+ (widget) => widget is FlowySvg && widget.svg == FlowySvgs.delete_s,
+ );
+ await tapButton(button);
+ },
);
+ }
- await tapButton(button);
+ void assertPhantomChecklistItemAtIndex({required int index}) {
+ final paddings = find.descendant(
+ of: find.descendant(
+ of: find.byType(ChecklistRowDetailCell),
+ matching: find.byType(ReorderableListView),
+ ),
+ matching: find.byWidgetPredicate(
+ (widget) =>
+ widget is Padding &&
+ (widget.child is ChecklistItem ||
+ widget.child is PhantomChecklistItem),
+ ),
+ );
+ final phantom = widget(paddings.at(index)).child!;
+ 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 {
@@ -565,7 +601,10 @@ extension AppFlowyDatabaseTest on WidgetTester {
final banner = find.byType(RowBanner);
expect(banner, findsOneWidget);
- await startGesture(getCenter(banner), kind: PointerDeviceKind.mouse);
+ await startGesture(
+ getCenter(banner) + const Offset(0, -10),
+ kind: PointerDeviceKind.mouse,
+ );
await pumpAndSettle();
}
@@ -577,7 +616,6 @@ extension AppFlowyDatabaseTest on WidgetTester {
await tapButtonWithName(
LocaleKeys.document_plugins_cover_addCover.tr(),
);
- await pumpAndSettle();
}
Future openEmojiPicker() async =>
@@ -679,6 +717,53 @@ extension AppFlowyDatabaseTest on WidgetTester {
await dismissFieldEditor();
}
+ Future changeFieldIcon(String icon) async {
+ await tapButton(find.byType(FieldEditIconButton));
+ if (icon.isEmpty) {
+ final button = find.descendant(
+ of: find.byType(FlowyIconEmojiPicker),
+ matching: find.text(
+ LocaleKeys.button_remove.tr(),
+ ),
+ );
+ await tapButton(button);
+ } else {
+ final svgContent = kIconGroups?.findSvgContent(icon);
+ await tapButton(
+ find.byWidgetPredicate(
+ (widget) => widget is FlowySvg && widget.svgString == svgContent,
+ ),
+ );
+ }
+ }
+
+ void assertFieldSvg(String name, FieldType fieldType) {
+ final svgFinder = find.byWidgetPredicate(
+ (widget) => widget is FlowySvg && widget.svg == fieldType.svgData,
+ );
+ final fieldButton = find.byWidgetPredicate(
+ (widget) => widget is FieldCellButton && widget.field.name == name,
+ );
+ expect(
+ find.descendant(of: fieldButton, matching: svgFinder),
+ findsOneWidget,
+ );
+ }
+
+ void assertFieldCustomSvg(String name, String svg) {
+ final svgContent = kIconGroups?.findSvgContent(svg);
+ final svgFinder = find.byWidgetPredicate(
+ (widget) => widget is FlowySvg && widget.svgString == svgContent,
+ );
+ final fieldButton = find.byWidgetPredicate(
+ (widget) => widget is FieldCellButton && widget.field.name == name,
+ );
+ expect(
+ find.descendant(of: fieldButton, matching: svgFinder),
+ findsOneWidget,
+ );
+ }
+
Future changeCalculateAtIndex(int index, CalculationType type) async {
await tap(find.byType(CalculateCell).at(index));
await pumpAndSettle();
@@ -857,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);
@@ -957,7 +1067,7 @@ extension AppFlowyDatabaseTest on WidgetTester {
Future tapCreateFilterByFieldType(FieldType type, String title) async {
final findFilter = find.byWidgetPredicate(
(widget) =>
- widget is GridFilterPropertyCell &&
+ widget is FilterableFieldButton &&
widget.fieldInfo.fieldType == type &&
widget.fieldInfo.name == title,
);
@@ -965,8 +1075,9 @@ extension AppFlowyDatabaseTest on WidgetTester {
}
Future tapFilterButtonInGrid(String name) async {
- final findFilter = find.byType(FilterMenuItem);
- final button = find.descendant(of: findFilter, matching: find.text(name));
+ final button = find.byWidgetPredicate(
+ (widget) => widget is ChoiceChipButton && widget.fieldInfo.name == name,
+ );
await tapButton(button);
}
@@ -1004,12 +1115,15 @@ extension AppFlowyDatabaseTest on WidgetTester {
}
/// Must call [tapSortMenuInSettingBar] first.
- Future tapSortButtonByName(String name) async {
- final findSortItem = find.byWidgetPredicate(
- (widget) =>
- widget is DatabaseSortItem && widget.sortInfo.fieldInfo.name == name,
+ Future tapEditSortConditionButtonByFieldName(String name) async {
+ final sortItem = find.descendant(
+ of: find.ancestor(
+ of: find.text(name),
+ matching: find.byType(DatabaseSortItem),
+ ),
+ matching: find.byType(SortConditionButton),
);
- await tapButton(findSortItem);
+ await tapButton(sortItem);
}
/// Must call [tapSortMenuInSettingBar] first.
@@ -1017,18 +1131,26 @@ extension AppFlowyDatabaseTest on WidgetTester {
(FieldType, String) from,
(FieldType, String) to,
) async {
- final fromSortItem = find.byWidgetPredicate(
- (widget) =>
- widget is DatabaseSortItem &&
- widget.sortInfo.fieldInfo.fieldType == from.$1 &&
- widget.sortInfo.fieldInfo.name == from.$2,
+ final fromSortItem = find.ancestor(
+ of: find.text(from.$2),
+ matching: find.byType(DatabaseSortItem),
);
- final toSortItem = find.byWidgetPredicate(
- (widget) =>
- widget is DatabaseSortItem &&
- widget.sortInfo.fieldInfo.fieldType == to.$1 &&
- widget.sortInfo.fieldInfo.name == to.$2,
+ final toSortItem = find.ancestor(
+ of: find.text(to.$2),
+ matching: find.byType(DatabaseSortItem),
);
+ // final fromSortItem = find.byWidgetPredicate(
+ // (widget) =>
+ // widget is DatabaseSortItem &&
+ // widget.sort.fieldInfo.fieldType == from.$1 &&
+ // widget.sort.fieldInfo.name == from.$2,
+ // );
+ // final toSortItem = find.byWidgetPredicate(
+ // (widget) =>
+ // widget is DatabaseSortItem &&
+ // widget.sort.fieldInfo.fieldType == to.$1 &&
+ // widget.sort.fieldInfo.name == to.$2,
+ // );
final dragElement = find.descendant(
of: fromSortItem,
matching: find.byType(ReorderableDragStartListener),
@@ -1037,16 +1159,13 @@ extension AppFlowyDatabaseTest on WidgetTester {
await pumpAndSettle(const Duration(milliseconds: 200));
}
- /// Must call [tapSortButtonByName] first.
+ /// Must call [tapEditSortConditionButtonByFieldName] first.
Future tapSortByDescending() async {
await tapButton(
- find.descendant(
- of: find.byType(OrderPannelItem),
- matching: find.byWidgetPredicate(
- (widget) =>
- widget is FlowyText &&
- widget.text == LocaleKeys.grid_sort_descending.tr(),
- ),
+ find.byWidgetPredicate(
+ (widget) =>
+ widget is OrderPanelItem &&
+ widget.condition == SortConditionPB.Descending,
),
);
await sendKeyEvent(LogicalKeyboardKey.escape);
@@ -1129,6 +1248,44 @@ extension AppFlowyDatabaseTest on WidgetTester {
await tapButton(button);
}
+ Future changeTextFilterCondition(
+ TextFilterConditionPB condition,
+ ) async {
+ await tapButton(find.byType(TextFilterConditionList));
+ final button = find.descendant(
+ of: find.byType(HoverButton),
+ matching: find.text(
+ condition.filterName,
+ ),
+ );
+
+ await tapButton(button);
+ }
+
+ Future changeSelectFilterCondition(
+ SelectOptionFilterConditionPB condition,
+ ) async {
+ await tapButton(find.byType(SelectOptionFilterConditionList));
+ final button = find.descendant(
+ of: find.byType(HoverButton),
+ matching: find.text(condition.i18n),
+ );
+
+ await tapButton(button);
+ }
+
+ Future changeDateFilterCondition(
+ DateTimeFilterCondition condition,
+ ) async {
+ await tapButton(find.byType(DateFilterConditionList));
+ final button = find.descendant(
+ of: find.byType(HoverButton),
+ matching: find.text(condition.filterName),
+ );
+
+ await tapButton(button);
+ }
+
/// Should call [tapDatabaseSettingButton] first.
Future tapViewPropertiesButton() async {
final findSettingItem = find.byType(DatabaseSettingsList);
@@ -1331,6 +1488,7 @@ extension AppFlowyDatabaseTest on WidgetTester {
);
await tapButton(button);
+ await tapButtonWithName(LocaleKeys.button_delete.tr());
}
Future dragDropRescheduleCalendarEvent() async {
@@ -1438,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 d754519ffa..398a3f9657 100644
--- a/frontend/appflowy_flutter/integration_test/shared/document_test_operations.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/document_test_operations.dart
@@ -2,23 +2,30 @@ import 'dart:async';
import 'dart:ui';
import 'package:appflowy/generated/locale_keys.g.dart';
+import 'package:appflowy/mobile/presentation/base/view_page/app_bar_buttons.dart';
+import 'package:appflowy/mobile/presentation/widgets/flowy_mobile_quick_action_button.dart';
import 'package:appflowy/plugins/base/emoji/emoji_picker.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/actions/block_action_add_button.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/actions/block_action_option_button.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/actions/drag_to_reorder/draggable_option_button.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/actions/option/option_actions.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/cover/document_immersive_cover.dart';
import 'package:appflowy/plugins/document/presentation/editor_plugins/header/cover_editor.dart';
-import 'package:appflowy/plugins/document/presentation/editor_plugins/header/document_header_node_widget.dart';
+import 'package:appflowy/plugins/document/presentation/editor_plugins/header/cover_title.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/plugins/document/presentation/editor_plugins/image/upload_image_menu/widgets/embed_image_url_widget.dart';
import 'package:appflowy/plugins/inline_actions/widgets/inline_actions_handler.dart';
import 'package:appflowy/shared/icon_emoji_picker/emoji_skin_tone.dart';
import 'package:appflowy/shared/icon_emoji_picker/flowy_icon_emoji_picker.dart';
+import 'package:appflowy/workspace/presentation/widgets/pop_up_action.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_emoji_mart/flutter_emoji_mart.dart';
import 'package:flutter_test/flutter_test.dart';
+import 'package:universal_platform/universal_platform.dart';
import 'util.dart';
@@ -43,7 +50,10 @@ class EditorOperations {
Future tapLineOfEditorAt(int index) async {
final textBlocks = find.byType(AppFlowyRichText);
index = index.clamp(0, textBlocks.evaluate().length - 1);
- await tester.tapAt(tester.getTopRight(textBlocks.at(index)));
+ final center = tester.getCenter(textBlocks.at(index));
+ final right = tester.getTopRight(textBlocks.at(index));
+ final centerRight = Offset(right.dx, center.dy);
+ await tester.tapAt(centerRight);
await tester.pumpAndSettle();
}
@@ -65,6 +75,20 @@ class EditorOperations {
expect(find.byType(FlowyEmojiPicker), findsOneWidget);
}
+ Future paste() async {
+ if (UniversalPlatform.isMacOS) {
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isMetaPressed: true,
+ );
+ } else {
+ await tester.simulateKeyEvent(
+ LogicalKeyboardKey.keyV,
+ isControlPressed: true,
+ );
+ }
+ }
+
Future tapGettingStartedIcon() async {
await tester.tapButton(
find.descendant(
@@ -173,6 +197,11 @@ class EditorOperations {
await tester.ime.insertCharacter('@');
}
+ /// trigger the plus action menu (+) command
+ Future showPlusMenu() async {
+ await tester.ime.insertCharacter('+');
+ }
+
/// Tap the slash menu item with [name]
///
/// Must call [showSlashMenu] first.
@@ -211,7 +240,7 @@ class EditorOperations {
}
/// Update the editor's selection
- Future updateSelection(Selection selection) async {
+ Future updateSelection(Selection? selection) async {
final editorState = getCurrentEditorState();
unawaited(
editorState.updateSelectionWithReason(
@@ -269,10 +298,44 @@ class EditorOperations {
widget.blockComponentContext.node.path.equals(path),
),
);
+ await tester.pumpUntilFound(find.byType(PopoverActionList));
},
);
}
+ /// open the turn into menu
+ Future openTurnIntoMenu(Path path) async {
+ await hoverAndClickOptionMenuButton(path);
+ await tester.tapButton(
+ find
+ .findTextInFlowyText(
+ LocaleKeys.document_plugins_optionAction_turnInto.tr(),
+ )
+ .first,
+ );
+ await tester.pumpUntilFound(find.byType(TurnIntoOptionMenu));
+ }
+
+ /// copy link to block
+ Future copyLinkToBlock(Path path) async {
+ await hoverAndClickOptionMenuButton(path);
+ await tester.tapButton(
+ find.findTextInFlowyText(
+ LocaleKeys.document_plugins_optionAction_copyLinkToBlock.tr(),
+ ),
+ );
+ }
+
+ Future openDepthMenu(Path path) async {
+ await hoverAndClickOptionMenuButton(path);
+ await tester.tapButton(
+ find.findTextInFlowyText(
+ LocaleKeys.document_plugins_optionAction_depth.tr(),
+ ),
+ );
+ await tester.pumpUntilFound(find.byType(DepthOptionMenu));
+ }
+
/// Drag block
///
/// [offset] is the offset to move the block.
@@ -323,4 +386,54 @@ class EditorOperations {
);
await tester.pumpAndSettle(Durations.short1);
}
+
+ Finder findDocumentTitle(String? title) {
+ final parent = UniversalPlatform.isDesktop
+ ? find.byType(CoverTitle)
+ : find.byType(DocumentImmersiveCover);
+
+ return find.descendant(
+ of: parent,
+ matching: find.byWidgetPredicate(
+ (widget) {
+ if (widget is! TextField) {
+ return false;
+ }
+
+ if (widget.controller?.text == title) {
+ return true;
+ }
+
+ if (title == null) {
+ return true;
+ }
+
+ if (title.isEmpty) {
+ return widget.controller?.text.isEmpty ?? false;
+ }
+
+ return false;
+ },
+ ),
+ );
+ }
+
+ /// open the more action menu on mobile
+ Future openMoreActionMenuOnMobile() async {
+ final moreActionButton = find.byType(MobileViewPageMoreButton);
+ await tester.tapButton(moreActionButton);
+ await tester.pumpAndSettle();
+ }
+
+ /// click the more action item on mobile
+ ///
+ /// rename, add collaborator, publish, delete, etc.
+ Future clickMoreActionItemOnMobile(String name) async {
+ final moreActionItem = find.descendant(
+ of: find.byType(MobileQuickActionButton),
+ matching: find.findTextInFlowyText(name),
+ );
+ await tester.tapButton(moreActionItem);
+ await tester.pumpAndSettle();
+ }
}
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 dedb2136f1..3b9ef0d75c 100644
--- a/frontend/appflowy_flutter/integration_test/shared/expectation.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/expectation.dart
@@ -1,9 +1,16 @@
+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';
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_header_node_widget.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';
@@ -13,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';
@@ -25,9 +34,15 @@ const String gettingStarted = 'Getting started';
extension Expectation on WidgetTester {
/// Expect to see the home page and with a default read me page.
Future expectToSeeHomePageWithGetStartedPage() async {
- final finder = find.byType(HomeStack);
- await pumpUntilFound(finder);
- expect(finder, findsOneWidget);
+ if (UniversalPlatform.isDesktopOrWeb) {
+ final finder = find.byType(HomeStack);
+ await pumpUntilFound(finder);
+ expect(finder, findsOneWidget);
+ } else if (UniversalPlatform.isMobile) {
+ final finder = find.byType(MobileHomePage);
+ await pumpUntilFound(finder);
+ expect(finder, findsOneWidget);
+ }
final docFinder = find.textContaining(gettingStarted);
await pumpUntilFound(docFinder);
@@ -110,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);
}
@@ -216,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 e06634efef..bfc5efedde 100644
--- a/frontend/appflowy_flutter/integration_test/shared/settings.dart
+++ b/frontend/appflowy_flutter/integration_test/shared/settings.dart
@@ -4,14 +4,15 @@ import 'package:appflowy/workspace/application/settings/prelude.dart';
import 'package:appflowy/workspace/presentation/home/menu/sidebar/shared/sidebar_setting.dart';
import 'package:appflowy/workspace/presentation/settings/pages/account/account_user_profile.dart';
import 'package:appflowy/workspace/presentation/settings/pages/settings_workspace_view.dart';
+import 'package:appflowy/workspace/presentation/settings/pages/sites/domain/domain_settings_dialog.dart';
import 'package:appflowy/workspace/presentation/settings/settings_dialog.dart';
import 'package:appflowy/workspace/presentation/settings/widgets/settings_menu_element.dart';
import 'package:appflowy/workspace/presentation/widgets/toggle/toggle.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flowy_infra_ui/style_widget/text_field.dart';
+import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
-import '../desktop/board/board_hide_groups_test.dart';
import 'base.dart';
import 'common_operations.dart';
@@ -78,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();
@@ -117,4 +118,20 @@ extension AppFlowySettings on WidgetTester {
await tapAt(Offset.zero);
await pumpAndSettle();
}
+
+ Future updateNamespace(String namespace) async {
+ final dialog = find.byType(DomainSettingsDialog);
+ expect(dialog, findsOneWidget);
+
+ // input the new namespace
+ await enterText(
+ find.descendant(
+ of: dialog,
+ matching: find.byType(TextField),
+ ),
+ namespace,
+ );
+ await tapButton(find.text(LocaleKeys.button_save.tr()));
+ 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 d9c5905736..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
@@ -56,8 +56,6 @@ PODS:
- Flutter
- keyboard_height_plugin (0.0.1):
- Flutter
- - open_file_ios (0.0.1):
- - Flutter
- open_filex (0.0.2):
- Flutter
- package_info_plus (0.4.5):
@@ -68,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)
@@ -81,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):
@@ -90,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`)
@@ -104,17 +107,18 @@ DEPENDENCIES:
- integration_test (from `.symlinks/plugins/integration_test/ios`)
- irondash_engine_context (from `.symlinks/plugins/irondash_engine_context/ios`)
- keyboard_height_plugin (from `.symlinks/plugins/keyboard_height_plugin/ios`)
- - open_file_ios (from `.symlinks/plugins/open_file_ios/ios`)
- open_filex (from `.symlinks/plugins/open_filex/ios`)
- 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:
@@ -151,8 +155,6 @@ EXTERNAL SOURCES:
:path: ".symlinks/plugins/irondash_engine_context/ios"
keyboard_height_plugin:
:path: ".symlinks/plugins/keyboard_height_plugin/ios"
- open_file_ios:
- :path: ".symlinks/plugins/open_file_ios/ios"
open_filex:
:path: ".symlinks/plugins/open_filex/ios"
package_info_plus:
@@ -161,51 +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_file_ios: 461db5853723763573e140de3193656f91990d9e
- 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