Repository: ferraridamiano/ConverterNOW Branch: master Commit: 4cd9fdf01cb3 Files: 244 Total size: 780.7 KB Directory structure: gitextract_nvfewi80/ ├── .firebaserc ├── .github/ │ ├── FUNDING.yml │ ├── ISSUE_TEMPLATE/ │ │ ├── bug_report.md │ │ └── new_unit_request.md │ ├── dependabot.yml │ └── workflows/ │ ├── build_deploy.yml │ ├── scripts/ │ │ └── build_appimage.sh │ └── tests.yml ├── .gitignore ├── .vscode/ │ └── settings.json ├── CONTRIBUTING.md ├── Dockerfile ├── LICENSE ├── README.md ├── analysis_options.yaml ├── android/ │ ├── .gitignore │ ├── app/ │ │ ├── build.gradle.kts │ │ └── src/ │ │ ├── debug/ │ │ │ └── AndroidManifest.xml │ │ ├── main/ │ │ │ ├── AndroidManifest.xml │ │ │ ├── kotlin/ │ │ │ │ └── com/ │ │ │ │ └── ferrarid/ │ │ │ │ └── converterpro/ │ │ │ │ └── MainActivity.kt │ │ │ └── res/ │ │ │ ├── drawable/ │ │ │ │ ├── ic_launcher_background.xml │ │ │ │ ├── ic_launcher_foreground.xml │ │ │ │ └── launch_background.xml │ │ │ ├── drawable-v21/ │ │ │ │ └── launch_background.xml │ │ │ ├── mipmap-anydpi-v26/ │ │ │ │ ├── ic_launcher.xml │ │ │ │ └── ic_launcher_round.xml │ │ │ ├── values/ │ │ │ │ └── styles.xml │ │ │ └── values-night/ │ │ │ └── styles.xml │ │ └── profile/ │ │ └── AndroidManifest.xml │ ├── build.gradle.kts │ ├── gradle/ │ │ └── wrapper/ │ │ └── gradle-wrapper.properties │ ├── gradle.properties │ ├── gradlew │ ├── gradlew.bat │ └── settings.gradle.kts ├── assets/ │ └── fonts/ │ └── OFL.txt ├── cliff.toml ├── fastlane/ │ └── metadata/ │ └── android/ │ ├── ar/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── bn/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── de-DE/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── el/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── en-US/ │ │ ├── changelogs/ │ │ │ ├── 24.txt │ │ │ ├── 25.txt │ │ │ ├── 26.txt │ │ │ ├── 27.txt │ │ │ ├── 28.txt │ │ │ ├── 29.txt │ │ │ ├── 30.txt │ │ │ ├── 31.txt │ │ │ ├── 33.txt │ │ │ ├── 34.txt │ │ │ ├── 35.txt │ │ │ ├── 36.txt │ │ │ ├── 37.txt │ │ │ ├── 38.txt │ │ │ ├── 39.txt │ │ │ ├── 40.txt │ │ │ ├── 41.txt │ │ │ ├── 42.txt │ │ │ ├── 43.txt │ │ │ ├── 44.txt │ │ │ ├── 45.txt │ │ │ ├── 46.txt │ │ │ ├── 47.txt │ │ │ ├── 48.txt │ │ │ ├── 49.txt │ │ │ ├── 50.txt │ │ │ ├── 51.txt │ │ │ └── 52.txt │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── es-ES/ │ │ ├── changelogs/ │ │ │ └── 33.txt │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── fr-FR/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── hr/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── id/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── it-IT/ │ │ ├── changelogs/ │ │ │ ├── 24.txt │ │ │ ├── 25.txt │ │ │ ├── 26.txt │ │ │ ├── 27.txt │ │ │ ├── 28.txt │ │ │ ├── 29.txt │ │ │ ├── 30.txt │ │ │ ├── 31.txt │ │ │ ├── 33.txt │ │ │ ├── 34.txt │ │ │ ├── 35.txt │ │ │ ├── 36.txt │ │ │ ├── 37.txt │ │ │ ├── 38.txt │ │ │ ├── 39.txt │ │ │ ├── 40.txt │ │ │ ├── 41.txt │ │ │ ├── 42.txt │ │ │ ├── 43.txt │ │ │ ├── 44.txt │ │ │ ├── 45.txt │ │ │ ├── 46.txt │ │ │ ├── 47.txt │ │ │ ├── 48.txt │ │ │ ├── 49.txt │ │ │ ├── 50.txt │ │ │ ├── 51.txt │ │ │ └── 52.txt │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── ja/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── nl_NL/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── pl-PL/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── pt-BR/ │ │ ├── full_description.txt │ │ └── short_description.txt │ ├── ru-RU/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ ├── tr-TR/ │ │ ├── changelogs/ │ │ │ ├── 24.txt │ │ │ ├── 25.txt │ │ │ ├── 26.txt │ │ │ └── 27.txt │ │ ├── full_description.txt │ │ └── short_description.txt │ ├── zh/ │ │ ├── full_description.txt │ │ ├── short_description.txt │ │ └── title.txt │ └── zh-TW/ │ ├── full_description.txt │ └── short_description.txt ├── firebase.json ├── integration_test/ │ ├── large_display_test.dart │ ├── small_display_test.dart │ └── utils.dart ├── lib/ │ ├── app_router.dart │ ├── data/ │ │ ├── default_order.dart │ │ └── property_unit_maps.dart │ ├── helpers/ │ │ └── responsive_helper.dart │ ├── main.dart │ ├── models/ │ │ ├── conversions.dart │ │ ├── currencies.dart │ │ ├── hide_units.dart │ │ ├── import_export.dart │ │ ├── order.dart │ │ ├── properties_list.dart │ │ └── settings.dart │ ├── pages/ │ │ ├── app_scaffold.dart │ │ ├── choose_property_page.dart │ │ ├── conversion_page.dart │ │ ├── custom_drawer.dart │ │ ├── error_page.dart │ │ ├── hide_units_page.dart │ │ ├── initial_page.dart │ │ ├── reorder_page.dart │ │ ├── reorder_properties_page.dart │ │ ├── reorder_units_page.dart │ │ ├── search_page.dart │ │ ├── select_units_page.dart │ │ ├── settings_page.dart │ │ └── splash_screen.dart │ ├── styles/ │ │ └── consts.dart │ └── utils/ │ ├── navigator_utils.dart │ ├── palette.dart │ ├── utils.dart │ └── utils_widgets.dart ├── linux/ │ ├── .gitignore │ ├── CMakeLists.txt │ ├── flutter/ │ │ ├── CMakeLists.txt │ │ ├── generated_plugin_registrant.cc │ │ ├── generated_plugin_registrant.h │ │ └── generated_plugins.cmake │ ├── io.github.ferraridamiano.ConverterNOW.desktop │ ├── io.github.ferraridamiano.ConverterNOW.metainfo.xml │ └── runner/ │ ├── CMakeLists.txt │ ├── main.cc │ ├── my_application.cc │ └── my_application.h ├── packages/ │ ├── calculator_widget/ │ │ ├── analysis_options.yaml │ │ ├── lib/ │ │ │ ├── animated_button.dart │ │ │ ├── calculator_model.dart │ │ │ └── calculator_widget.dart │ │ └── pubspec.yaml │ └── translations/ │ ├── l10n.yaml │ ├── lib/ │ │ └── l10n/ │ │ ├── app_ar.arb │ │ ├── app_bn.arb │ │ ├── app_ca.arb │ │ ├── app_de.arb │ │ ├── app_el.arb │ │ ├── app_en.arb │ │ ├── app_es.arb │ │ ├── app_fr.arb │ │ ├── app_hr.arb │ │ ├── app_id.arb │ │ ├── app_it.arb │ │ ├── app_ja.arb │ │ ├── app_nb.arb │ │ ├── app_nl.arb │ │ ├── app_pl.arb │ │ ├── app_pt.arb │ │ ├── app_ru.arb │ │ ├── app_tr.arb │ │ ├── app_zh.arb │ │ └── app_zh_TW.arb │ └── pubspec.yaml ├── pubspec.yaml ├── web/ │ ├── index.html │ └── manifest.json └── windows/ ├── .gitignore ├── CMakeLists.txt ├── flutter/ │ ├── CMakeLists.txt │ ├── generated_plugin_registrant.cc │ ├── generated_plugin_registrant.h │ └── generated_plugins.cmake ├── innosetup.iss └── runner/ ├── CMakeLists.txt ├── Runner.rc ├── flutter_window.cpp ├── flutter_window.h ├── main.cpp ├── resource.h ├── runner.exe.manifest ├── utils.cpp ├── utils.h ├── win32_window.cpp └── win32_window.h ================================================ FILE CONTENTS ================================================ ================================================ FILE: .firebaserc ================================================ { "projects": { "default": "converter-now" } } ================================================ FILE: .github/FUNDING.yml ================================================ # These are supported funding model platforms github: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2] patreon: # Replace with a single Patreon username open_collective: # Replace with a single Open Collective username ko_fi: # Replace with a single Ko-fi username tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry liberapay: # Replace with a single Liberapay username issuehunt: # Replace with a single IssueHunt username otechie: # Replace with a single Otechie username custom: ['https://www.paypal.com/paypalme/demapps'] ================================================ FILE: .github/ISSUE_TEMPLATE/bug_report.md ================================================ --- name: Bug report about: Create a report to help us improve title: "[BUG] " labels: '' assignees: '' --- **Describe the bug** A clear and concise description of what the bug is. **To Reproduce** Steps to reproduce the behavior: 1. Go to '...' 2. Click on '....' 3. Scroll down to '....' 4. See error **Expected behavior** A clear and concise description of what you expected to happen. **Screenshots / gif** If applicable, add screenshots to help explain your problem. **Device** - Device type: [e.g. Android, Windows, Linux, Web] - OS version: [e.g. Android 11] - Device type: [e.g. Google Pixel 6, Galaxy S21] **Additional context** Add any other context about the problem here. ================================================ FILE: .github/ISSUE_TEMPLATE/new_unit_request.md ================================================ --- name: New unit request about: Suggest a unit of measurement that should be added title: "[new-unit]" labels: '' assignees: '' --- **Description** Add here a brief explanation to which units should be added and possibly why (e.g. they are often used in a specific job). We know that there are a ton of unit of measurement that are not yet added to this app. We want to have just the most used one. **Units to be added** | Unit name | Property | Reference to the conversion formula | | ----------| -------- | -------------------------------------------------- | | Angstrom | Length | [link](https://en.wikipedia.org/wiki/Angstrom) | | Pound | Mass | [link](https://en.wikipedia.org/wiki/Pound_(mass)) | | Yen | Currency | / | ================================================ FILE: .github/dependabot.yml ================================================ version: 2 updates: - package-ecosystem: "github-actions" directory: "/" schedule: interval: "weekly" - package-ecosystem: "pub" directories: - "/" - "/packages/calculator_widget" - "/packages/translations" schedule: interval: "monthly" ignore: - dependency-name: "*" update-types: - "version-update:semver-patch" - "version-update:semver-minor" ================================================ FILE: .github/workflows/build_deploy.yml ================================================ name: "Build & deploy for Android, Linux, Web and Windows" on: push: tags: ['v*'] jobs: build_linux_x86_64: name: Build and release Linux app (x86_64) # Use the previous ubuntu version for compatibility (https://github.com/ferraridamiano/ConverterNOW/issues/370) runs-on: ubuntu-22.04 timeout-minutes: 15 steps: - name: Checkout uses: actions/checkout@v6 - name: Install missing packages run: | sudo add-apt-repository universe sudo apt-get update sudo apt-get install -y clang cmake ninja-build pkg-config libgtk-3-dev liblzma-dev libstdc++-12-dev desktop-file-utils - name: Setup Flutter uses: subosito/flutter-action@v2 with: channel: stable cache: true - name: Flutter version run: flutter --version - name: Bootstrap run: | flutter pub global activate melos melos bootstrap - name: Build for linux run: flutter build linux --release - name: Tar linux package run: | arch=$(ls build/linux) tar -C build/linux/$arch/release/bundle -czf converternow-linux-x86_64.tar.gz . - name: Build appimage run: | chmod +x .github/workflows/scripts/build_appimage.sh ./.github/workflows/scripts/build_appimage.sh - name: Release to GitHub uses: softprops/action-gh-release@v2 with: files: | converternow-linux-x86_64.tar.gz converternow-x86_64.AppImage token: ${{ secrets.GH_TOKEN }} - name: Upload compiled icons uses: actions/upload-artifact@v7 with: name: assets path: assets build_linux_aarch64: name: Build and release Linux app (aarch64) # Use the previous ubuntu version for compatibility (https://github.com/ferraridamiano/ConverterNOW/issues/370) runs-on: ubuntu-22.04-arm timeout-minutes: 15 needs: build_linux_x86_64 steps: - name: Checkout uses: actions/checkout@v6 - name: Install missing packages run: | sudo add-apt-repository universe sudo apt-get update sudo apt-get install -y clang cmake ninja-build pkg-config libgtk-3-dev liblzma-dev libstdc++-12-dev desktop-file-utils # Use git because subosito/flutter-action does not support arm64 yet (https://github.com/subosito/flutter-action/issues/345) - name: Setup Flutter run: | git clone --branch stable https://github.com/flutter/flutter.git $RUNNER_TEMP/flutter cd $RUNNER_TEMP/flutter echo "$RUNNER_TEMP/flutter/bin" >> $GITHUB_PATH echo "$HOME/.pub-cache/bin" >> $GITHUB_PATH - name: Flutter version run: flutter --version # continue-on-error is set because we would receive an error on icon # compilation (but we import the compiled icons from previous step) - name: Bootstrap continue-on-error: true run: | flutter pub global activate melos melos bootstrap - name: Remove current assets run: rm -rf assets - name: Download compiled assets uses: actions/download-artifact@v8 with: name: assets path: assets - name: Build for linux run: flutter build linux --release - name: Tar linux package run: | arch=$(ls build/linux) tar -C build/linux/$arch/release/bundle -czf converternow-linux-aarch64.tar.gz . - name: Build appimage run: | chmod +x .github/workflows/scripts/build_appimage.sh ./.github/workflows/scripts/build_appimage.sh - name: Release to GitHub uses: softprops/action-gh-release@v2 with: files: | converternow-linux-aarch64.tar.gz converternow-aarch64.AppImage token: ${{ secrets.GH_TOKEN }} build_android: needs: build_linux_aarch64 name: Build and release Android app runs-on: ubuntu-latest timeout-minutes: 15 steps: - name: Checkout uses: actions/checkout@v6 - name: Setup Java uses: actions/setup-java@v5 with: distribution: 'temurin' java-version: '17' - name: Setup Flutter uses: subosito/flutter-action@v2 with: channel: stable cache: true - name: Flutter version run: flutter --version - name: Bootstrap run: | dart pub global activate melos melos bootstrap - name: Download Android keystore id: android_keystore uses: timheuer/base64-to-file@v1 with: fileName: key.jks encodedString: ${{ secrets.ANDROID_KEYSTORE_BASE64 }} - name: Create key.properties run: | echo "storeFile=${{ steps.android_keystore.outputs.filePath }}" > android/key.properties echo "storePassword=${{ secrets.ANDROID_KEYSTORE_PASSWORD }}" >> android/key.properties echo "keyPassword=${{ secrets.ANDROID_KEY_PASSWORD }}" >> android/key.properties echo "keyAlias=${{ secrets.ANDROID_KEY_ALIAS }}" >> android/key.properties - name: Build Android App Bundle run: flutter build appbundle --dart-define=IS_PLAYSTORE=true --obfuscate --split-debug-info=/tmp - name: Build Android APK files run: flutter build apk --release --split-per-abi --obfuscate --split-debug-info=/tmp - name: Release to GitHub uses: softprops/action-gh-release@v2 with: files: | build/app/outputs/flutter-apk/app-*-release.apk build/app/outputs/bundle/release/app-release.aab token: ${{ secrets.GH_TOKEN }} build_web: needs: build_android name: Build and release Web app runs-on: ubuntu-latest timeout-minutes: 15 steps: - name: Checkout uses: actions/checkout@v6 - name: Setup Flutter uses: subosito/flutter-action@v2 with: channel: stable cache: true - name: Flutter version run: flutter --version - name: Bootstrap run: | dart pub global activate melos melos bootstrap - name: Build for Web run: flutter build web --release --wasm - name: Upload to Firebase Hosting uses: FirebaseExtended/action-hosting-deploy@v0 with: repoToken: '${{ secrets.GITHUB_TOKEN }}' firebaseServiceAccount: '${{ secrets.FIREBASE_SERVICE_ACCOUNT_CONVERTER_NOW }}' channelId: live projectId: converter-now build_windows: needs: build_web name: Build and release Windows app runs-on: windows-latest timeout-minutes: 20 steps: - name: Checkout uses: actions/checkout@v6 - name: Setup Flutter uses: subosito/flutter-action@v2 with: channel: stable cache: true - name: Flutter version run: flutter --version - name: Bootstrap run: | dart pub global activate melos melos bootstrap - name: Build Windows run: flutter build windows --release - name: Build and create the .msix installer run: dart run msix:create --store - name: Create the .exe installer shell: pwsh run: | $version = (Select-String -Path pubspec.yaml -Pattern "^version:\s*(.+)").Matches[0].Groups[1].Value.Split("+")[0] iscc /DMyAppVersion=$version windows\innosetup.iss - name: Release to GitHub uses: softprops/action-gh-release@v2 with: files: | build/windows/x64/runner/Release/converternow-windows.msix windows/converternow-windows-setup.exe token: ${{ secrets.GH_TOKEN }} build_docker: needs: build_windows name: Build and release Docker image runs-on: ubuntu-latest timeout-minutes: 10 env: REGISTRY: ghcr.io IMAGE_NAME: ${{ github.repository }} permissions: contents: read packages: write attestations: write id-token: write steps: - name: Checkout code uses: actions/checkout@v6 - name: Log in to the Container registry uses: docker/login-action@v4 with: registry: ${{ env.REGISTRY }} username: ${{ github.actor }} password: ${{ secrets.GITHUB_TOKEN }} - name: Extract metadata (tags, labels) for Docker id: meta uses: docker/metadata-action@v6 with: images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }} - name: Set up QEMU uses: docker/setup-qemu-action@v4 - name: Set up Docker Buildx uses: docker/setup-buildx-action@v4 - name: Build and push Docker image id: push uses: docker/build-push-action@v7 with: context: . push: true platforms: linux/amd64,linux/arm64 tags: ${{ steps.meta.outputs.tags }} labels: ${{ steps.meta.outputs.labels }} cache-from: type=gha cache-to: type=gha,mode=max - name: Generate artifact attestation uses: actions/attest@v4 with: subject-name: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME}} subject-digest: ${{ steps.push.outputs.digest }} push-to-registry: true create_release: needs: build_docker name: Create GitHub Release runs-on: ubuntu-latest timeout-minutes: 5 permissions: contents: write steps: - name: Checkout repository uses: actions/checkout@v6 with: fetch-depth: 0 - name: Generate Changelog using git-cliff uses: orhun/git-cliff-action@v4 id: git-cliff with: args: --latest - name: Create GitHub Release uses: softprops/action-gh-release@v2 with: body: ${{ steps.git-cliff.outputs.content }} ================================================ FILE: .github/workflows/scripts/build_appimage.sh ================================================ #!/bin/bash # This script needs to be called once the flutter linux app build is finished arch=$(ls build/linux) arch_label="" if [ "$arch" == "x64" ]; then arch_label="x86_64" else arch_label="aarch64" fi appimagetool="appimagetool-$arch_label.AppImage" wget "https://github.com/AppImage/appimagetool/releases/latest/download/$appimagetool" chmod +x "$appimagetool" mkdir ConverterNOW.AppDir cp -r build/linux/$arch/release/bundle/* ConverterNOW.AppDir cp assets/app_icons/logo.svg ConverterNOW.AppDir echo -e '#!/bin/sh\ncd "$(dirname "$0")"\nexec ./converternow' > ConverterNOW.AppDir/AppRun chmod +x ConverterNOW.AppDir/AppRun cp linux/io.github.ferraridamiano.ConverterNOW.desktop ConverterNOW.AppDir desktop-file-edit --set-icon="logo" --set-key="Exec" --set-value="converternow %u" ConverterNOW.AppDir/io.github.ferraridamiano.ConverterNOW.desktop APPIMAGETOOL_APP_NAME=converternow ./"$appimagetool" ConverterNOW.AppDir rm -r ConverterNOW.AppDir "$appimagetool" ================================================ FILE: .github/workflows/tests.yml ================================================ name: Tests on: push: branches: - master pull_request: jobs: # JOB to run change detection changes: name: Detect changed files runs-on: ubuntu-latest # Set job outputs to values from filter step outputs: integration_testing: ${{ steps.filter.outputs.integration_testing }} steps: - uses: actions/checkout@v6 - uses: dorny/paths-filter@v4 id: filter with: filters: | integration_testing: - 'lib/**' - 'packages/**' - 'pubspec.yaml' integration_testing: needs: changes if: ${{ needs.changes.outputs.integration_testing == 'true' }} name: Integration testing runs-on: ubuntu-latest timeout-minutes: 15 steps: - name: Checkout uses: actions/checkout@v6 - name: Install missing packages run: | sudo apt-get update sudo apt-get install -y ninja-build libgtk-3-dev - name: Setup Flutter uses: subosito/flutter-action@v2 with: cache: true channel: stable - name: Flutter doctor run: flutter doctor --verbose - name: Install required dart tools run: | dart pub global activate melos - name: Bootstrap run: melos bootstrap - name: Run large display integration testing run: | export DISPLAY=:99 sudo Xvfb -ac :99 -screen 0 1280x1024x24 > /dev/null 2>&1 & flutter test integration_test/large_display_test.dart -d linux - name: Run small display integration testing run: | export DISPLAY=:99 sudo Xvfb -ac :99 -screen 0 1280x1024x24 > /dev/null 2>&1 & flutter test integration_test/small_display_test.dart -d linux ================================================ FILE: .gitignore ================================================ .DS_Store .dart_tool/ .packages .pub/ build/ .flutter-plugins *.jks .flutter-plugins-dependencies ios/Flutter/flutter_export_environment.sh android/key.properties *.iml .metadata .idea/ ios/Flutter/flutter_export_environment.sh debug.log .snap packages/translations/lib/app_localizations*.dart pubspec_overrides.yaml assets/*/*.svg.vec ================================================ FILE: .vscode/settings.json ================================================ { "files.exclude": { "packages/translations/lib/app_localizations*.dart": true, }, "cmake.ignoreCMakeListsMissing": true } ================================================ FILE: CONTRIBUTING.md ================================================ # Welcome We invite you to join our team! Everyone is welcome to contribute by submitting code, pull requests, or reporting issues on GitHub. There are many ways to get involved, and you don't need to be an expert to make a difference. Think you're not ready to contribute? Let us show you otherwise! Let's get started! ## Translating the app The app is currently translated into several languages, but if you know another language and would like to help with further internationalization, please check out the latest instructions [here](https://github.com/ferraridamiano/ConverterNOW/wiki/Translations). You can also contribute by translating the Play Store/F-Droid app page, which can be found [here](https://github.com/ferraridamiano/ConverterNOW/tree/master/fastlane/metadata/android/en-US). If you have any questions, feel free to contact me via email or open a new issue. I'll handle the rest. Thank you for your support! ## Design a new logo If you think you can create a better app icon, feel free to open a new issue or pull request with your proposal. I’ll guide you on how to integrate your idea into the project. Thank you! ## Submit a PR If you're confident with Flutter coding, you can help improve the app by submitting a pull request. But first, let me explain how the project is organized. ### Project structure The project uses [riverpod](https://github.com/rrousselGit/riverpod) for state management, with providers located in [`lib/models`](https://github.com/ferraridamiano/ConverterNOW/tree/master/lib/models). The app is structured as a monorepo and managed with [`melos`](https://github.com/invertase/melos). ### Add new units If you want to add new units I wrote a [wiki page](https://github.com/ferraridamiano/ConverterNOW/wiki/Add-a-new-unit-of-measurement), it is a step by step guide. ## Donations This project is a labor of love, managed by a single person. Your [donations](https://www.paypal.com/paypalme/DemApps) are invaluable — they help keep the development going and provide the motivation to continue improving the app. Every contribution, big or small, makes a difference. If you enjoy using the app and believe in its future, please consider supporting the project. Thank you! ================================================ FILE: Dockerfile ================================================ FROM --platform=linux/amd64 debian:13-slim AS builder USER root ENV HOME="/root" ARG flutter_version="stable" ENV FLUTTER_HOME=${HOME}/sdks/flutter \ FLUTTER_VERSION=$flutter_version ENV FLUTTER_ROOT=$FLUTTER_HOME ENV PATH="$FLUTTER_HOME/bin:$FLUTTER_HOME/bin/cache/dart-sdk/bin:$HOME/.pub-cache/bin:${PATH}" RUN DEBIAN_FRONTEND=noninteractive apt-get update && \ DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \ bash \ curl \ git \ wget \ unzip \ libstdc++6 \ ca-certificates \ xz-utils \ zip \ libglu1-mesa && \ DEBIAN_FRONTEND=noninteractive apt-get clean && \ rm -rf /var/lib/apt/lists/* RUN git clone --branch ${FLUTTER_VERSION} https://github.com/flutter/flutter.git ${FLUTTER_HOME} RUN flutter config --no-enable-windows-desktop --no-enable-macos-desktop --no-enable-linux-desktop --no-enable-android --no-enable-ios --enable-web && \ flutter channel stable && \ flutter doctor && \ flutter precache --web --no-ios --no-windows --no-macos --no-linux --no-android && \ chown -R root:root ${FLUTTER_HOME} WORKDIR /app COPY . . RUN flutter pub global activate melos && \ melos bootstrap && \ flutter build web --release --wasm --no-web-resources-cdn FROM nginx:alpine-slim COPY --from=builder /app/build/web /usr/share/nginx/html # Workaround, remove once https://github.com/nginx/nginx/pull/448 has been merged RUN sed -i -E 's|application/javascript\s+js;|application/javascript js mjs;|' /etc/nginx/mime.types EXPOSE 80 HEALTHCHECK --interval=30s --timeout=5s --start-period=10s --retries=3 \ CMD wget -O /dev/null http://localhost || exit 1 CMD ["nginx", "-g", "daemon off;"] ================================================ FILE: LICENSE ================================================ GNU GENERAL PUBLIC LICENSE Version 3, 29 June 2007 Copyright (C) 2007 Free Software Foundation, Inc. Everyone is permitted to copy and distribute verbatim copies of this license document, but changing it is not allowed. Preamble The GNU General Public License is a free, copyleft license for software and other kinds of works. The licenses for most software and other practical works are designed to take away your freedom to share and change the works. By contrast, the GNU General Public License is intended to guarantee your freedom to share and change all versions of a program--to make sure it remains free software for all its users. We, the Free Software Foundation, use the GNU General Public License for most of our software; it applies also to any other work released this way by its authors. You can apply it to your programs, too. When we speak of free software, we are referring to freedom, not price. Our General Public Licenses are designed to make sure that you have the freedom to distribute copies of free software (and charge for them if you wish), that you receive source code or can get it if you want it, that you can change the software or use pieces of it in new free programs, and that you know you can do these things. To protect your rights, we need to prevent others from denying you these rights or asking you to surrender the rights. Therefore, you have certain responsibilities if you distribute copies of the software, or if you modify it: responsibilities to respect the freedom of others. For example, if you distribute copies of such a program, whether gratis or for a fee, you must pass on to the recipients the same freedoms that you received. You must make sure that they, too, receive or can get the source code. And you must show them these terms so they know their rights. Developers that use the GNU GPL protect your rights with two steps: (1) assert copyright on the software, and (2) offer you this License giving you legal permission to copy, distribute and/or modify it. For the developers' and authors' protection, the GPL clearly explains that there is no warranty for this free software. For both users' and authors' sake, the GPL requires that modified versions be marked as changed, so that their problems will not be attributed erroneously to authors of previous versions. Some devices are designed to deny users access to install or run modified versions of the software inside them, although the manufacturer can do so. This is fundamentally incompatible with the aim of protecting users' freedom to change the software. The systematic pattern of such abuse occurs in the area of products for individuals to use, which is precisely where it is most unacceptable. Therefore, we have designed this version of the GPL to prohibit the practice for those products. If such problems arise substantially in other domains, we stand ready to extend this provision to those domains in future versions of the GPL, as needed to protect the freedom of users. Finally, every program is threatened constantly by software patents. States should not allow patents to restrict development and use of software on general-purpose computers, but in those that do, we wish to avoid the special danger that patents applied to a free program could make it effectively proprietary. To prevent this, the GPL assures that patents cannot be used to render the program non-free. The precise terms and conditions for copying, distribution and modification follow. TERMS AND CONDITIONS 0. Definitions. "This License" refers to version 3 of the GNU General Public License. "Copyright" also means copyright-like laws that apply to other kinds of works, such as semiconductor masks. "The Program" refers to any copyrightable work licensed under this License. Each licensee is addressed as "you". "Licensees" and "recipients" may be individuals or organizations. To "modify" a work means to copy from or adapt all or part of the work in a fashion requiring copyright permission, other than the making of an exact copy. The resulting work is called a "modified version" of the earlier work or a work "based on" the earlier work. A "covered work" means either the unmodified Program or a work based on the Program. To "propagate" a work means to do anything with it that, without permission, would make you directly or secondarily liable for infringement under applicable copyright law, except executing it on a computer or modifying a private copy. Propagation includes copying, distribution (with or without modification), making available to the public, and in some countries other activities as well. To "convey" a work means any kind of propagation that enables other parties to make or receive copies. Mere interaction with a user through a computer network, with no transfer of a copy, is not conveying. An interactive user interface displays "Appropriate Legal Notices" to the extent that it includes a convenient and prominently visible feature that (1) displays an appropriate copyright notice, and (2) tells the user that there is no warranty for the work (except to the extent that warranties are provided), that licensees may convey the work under this License, and how to view a copy of this License. If the interface presents a list of user commands or options, such as a menu, a prominent item in the list meets this criterion. 1. Source Code. The "source code" for a work means the preferred form of the work for making modifications to it. "Object code" means any non-source form of a work. A "Standard Interface" means an interface that either is an official standard defined by a recognized standards body, or, in the case of interfaces specified for a particular programming language, one that is widely used among developers working in that language. The "System Libraries" of an executable work include anything, other than the work as a whole, that (a) is included in the normal form of packaging a Major Component, but which is not part of that Major Component, and (b) serves only to enable use of the work with that Major Component, or to implement a Standard Interface for which an implementation is available to the public in source code form. A "Major Component", in this context, means a major essential component (kernel, window system, and so on) of the specific operating system (if any) on which the executable work runs, or a compiler used to produce the work, or an object code interpreter used to run it. The "Corresponding Source" for a work in object code form means all the source code needed to generate, install, and (for an executable work) run the object code and to modify the work, including scripts to control those activities. However, it does not include the work's System Libraries, or general-purpose tools or generally available free programs which are used unmodified in performing those activities but which are not part of the work. For example, Corresponding Source includes interface definition files associated with source files for the work, and the source code for shared libraries and dynamically linked subprograms that the work is specifically designed to require, such as by intimate data communication or control flow between those subprograms and other parts of the work. The Corresponding Source need not include anything that users can regenerate automatically from other parts of the Corresponding Source. The Corresponding Source for a work in source code form is that same work. 2. Basic Permissions. All rights granted under this License are granted for the term of copyright on the Program, and are irrevocable provided the stated conditions are met. This License explicitly affirms your unlimited permission to run the unmodified Program. The output from running a covered work is covered by this License only if the output, given its content, constitutes a covered work. This License acknowledges your rights of fair use or other equivalent, as provided by copyright law. You may make, run and propagate covered works that you do not convey, without conditions so long as your license otherwise remains in force. You may convey covered works to others for the sole purpose of having them make modifications exclusively for you, or provide you with facilities for running those works, provided that you comply with the terms of this License in conveying all material for which you do not control copyright. Those thus making or running the covered works for you must do so exclusively on your behalf, under your direction and control, on terms that prohibit them from making any copies of your copyrighted material outside their relationship with you. Conveying under any other circumstances is permitted solely under the conditions stated below. Sublicensing is not allowed; section 10 makes it unnecessary. 3. Protecting Users' Legal Rights From Anti-Circumvention Law. No covered work shall be deemed part of an effective technological measure under any applicable law fulfilling obligations under article 11 of the WIPO copyright treaty adopted on 20 December 1996, or similar laws prohibiting or restricting circumvention of such measures. When you convey a covered work, you waive any legal power to forbid circumvention of technological measures to the extent such circumvention is effected by exercising rights under this License with respect to the covered work, and you disclaim any intention to limit operation or modification of the work as a means of enforcing, against the work's users, your or third parties' legal rights to forbid circumvention of technological measures. 4. Conveying Verbatim Copies. You may convey verbatim copies of the Program's source code as you receive it, in any medium, provided that you conspicuously and appropriately publish on each copy an appropriate copyright notice; keep intact all notices stating that this License and any non-permissive terms added in accord with section 7 apply to the code; keep intact all notices of the absence of any warranty; and give all recipients a copy of this License along with the Program. You may charge any price or no price for each copy that you convey, and you may offer support or warranty protection for a fee. 5. Conveying Modified Source Versions. You may convey a work based on the Program, or the modifications to produce it from the Program, in the form of source code under the terms of section 4, provided that you also meet all of these conditions: a) The work must carry prominent notices stating that you modified it, and giving a relevant date. b) The work must carry prominent notices stating that it is released under this License and any conditions added under section 7. This requirement modifies the requirement in section 4 to "keep intact all notices". c) You must license the entire work, as a whole, under this License to anyone who comes into possession of a copy. This License will therefore apply, along with any applicable section 7 additional terms, to the whole of the work, and all its parts, regardless of how they are packaged. This License gives no permission to license the work in any other way, but it does not invalidate such permission if you have separately received it. d) If the work has interactive user interfaces, each must display Appropriate Legal Notices; however, if the Program has interactive interfaces that do not display Appropriate Legal Notices, your work need not make them do so. A compilation of a covered work with other separate and independent works, which are not by their nature extensions of the covered work, and which are not combined with it such as to form a larger program, in or on a volume of a storage or distribution medium, is called an "aggregate" if the compilation and its resulting copyright are not used to limit the access or legal rights of the compilation's users beyond what the individual works permit. Inclusion of a covered work in an aggregate does not cause this License to apply to the other parts of the aggregate. 6. Conveying Non-Source Forms. You may convey a covered work in object code form under the terms of sections 4 and 5, provided that you also convey the machine-readable Corresponding Source under the terms of this License, in one of these ways: a) Convey the object code in, or embodied in, a physical product (including a physical distribution medium), accompanied by the Corresponding Source fixed on a durable physical medium customarily used for software interchange. b) Convey the object code in, or embodied in, a physical product (including a physical distribution medium), accompanied by a written offer, valid for at least three years and valid for as long as you offer spare parts or customer support for that product model, to give anyone who possesses the object code either (1) a copy of the Corresponding Source for all the software in the product that is covered by this License, on a durable physical medium customarily used for software interchange, for a price no more than your reasonable cost of physically performing this conveying of source, or (2) access to copy the Corresponding Source from a network server at no charge. c) Convey individual copies of the object code with a copy of the written offer to provide the Corresponding Source. This alternative is allowed only occasionally and noncommercially, and only if you received the object code with such an offer, in accord with subsection 6b. d) Convey the object code by offering access from a designated place (gratis or for a charge), and offer equivalent access to the Corresponding Source in the same way through the same place at no further charge. You need not require recipients to copy the Corresponding Source along with the object code. If the place to copy the object code is a network server, the Corresponding Source may be on a different server (operated by you or a third party) that supports equivalent copying facilities, provided you maintain clear directions next to the object code saying where to find the Corresponding Source. Regardless of what server hosts the Corresponding Source, you remain obligated to ensure that it is available for as long as needed to satisfy these requirements. e) Convey the object code using peer-to-peer transmission, provided you inform other peers where the object code and Corresponding Source of the work are being offered to the general public at no charge under subsection 6d. A separable portion of the object code, whose source code is excluded from the Corresponding Source as a System Library, need not be included in conveying the object code work. A "User Product" is either (1) a "consumer product", which means any tangible personal property which is normally used for personal, family, or household purposes, or (2) anything designed or sold for incorporation into a dwelling. In determining whether a product is a consumer product, doubtful cases shall be resolved in favor of coverage. For a particular product received by a particular user, "normally used" refers to a typical or common use of that class of product, regardless of the status of the particular user or of the way in which the particular user actually uses, or expects or is expected to use, the product. A product is a consumer product regardless of whether the product has substantial commercial, industrial or non-consumer uses, unless such uses represent the only significant mode of use of the product. "Installation Information" for a User Product means any methods, procedures, authorization keys, or other information required to install and execute modified versions of a covered work in that User Product from a modified version of its Corresponding Source. The information must suffice to ensure that the continued functioning of the modified object code is in no case prevented or interfered with solely because modification has been made. If you convey an object code work under this section in, or with, or specifically for use in, a User Product, and the conveying occurs as part of a transaction in which the right of possession and use of the User Product is transferred to the recipient in perpetuity or for a fixed term (regardless of how the transaction is characterized), the Corresponding Source conveyed under this section must be accompanied by the Installation Information. But this requirement does not apply if neither you nor any third party retains the ability to install modified object code on the User Product (for example, the work has been installed in ROM). The requirement to provide Installation Information does not include a requirement to continue to provide support service, warranty, or updates for a work that has been modified or installed by the recipient, or for the User Product in which it has been modified or installed. Access to a network may be denied when the modification itself materially and adversely affects the operation of the network or violates the rules and protocols for communication across the network. Corresponding Source conveyed, and Installation Information provided, in accord with this section must be in a format that is publicly documented (and with an implementation available to the public in source code form), and must require no special password or key for unpacking, reading or copying. 7. Additional Terms. "Additional permissions" are terms that supplement the terms of this License by making exceptions from one or more of its conditions. Additional permissions that are applicable to the entire Program shall be treated as though they were included in this License, to the extent that they are valid under applicable law. If additional permissions apply only to part of the Program, that part may be used separately under those permissions, but the entire Program remains governed by this License without regard to the additional permissions. When you convey a copy of a covered work, you may at your option remove any additional permissions from that copy, or from any part of it. (Additional permissions may be written to require their own removal in certain cases when you modify the work.) You may place additional permissions on material, added by you to a covered work, for which you have or can give appropriate copyright permission. Notwithstanding any other provision of this License, for material you add to a covered work, you may (if authorized by the copyright holders of that material) supplement the terms of this License with terms: a) Disclaiming warranty or limiting liability differently from the terms of sections 15 and 16 of this License; or b) Requiring preservation of specified reasonable legal notices or author attributions in that material or in the Appropriate Legal Notices displayed by works containing it; or c) Prohibiting misrepresentation of the origin of that material, or requiring that modified versions of such material be marked in reasonable ways as different from the original version; or d) Limiting the use for publicity purposes of names of licensors or authors of the material; or e) Declining to grant rights under trademark law for use of some trade names, trademarks, or service marks; or f) Requiring indemnification of licensors and authors of that material by anyone who conveys the material (or modified versions of it) with contractual assumptions of liability to the recipient, for any liability that these contractual assumptions directly impose on those licensors and authors. All other non-permissive additional terms are considered "further restrictions" within the meaning of section 10. If the Program as you received it, or any part of it, contains a notice stating that it is governed by this License along with a term that is a further restriction, you may remove that term. If a license document contains a further restriction but permits relicensing or conveying under this License, you may add to a covered work material governed by the terms of that license document, provided that the further restriction does not survive such relicensing or conveying. If you add terms to a covered work in accord with this section, you must place, in the relevant source files, a statement of the additional terms that apply to those files, or a notice indicating where to find the applicable terms. Additional terms, permissive or non-permissive, may be stated in the form of a separately written license, or stated as exceptions; the above requirements apply either way. 8. Termination. You may not propagate or modify a covered work except as expressly provided under this License. Any attempt otherwise to propagate or modify it is void, and will automatically terminate your rights under this License (including any patent licenses granted under the third paragraph of section 11). However, if you cease all violation of this License, then your license from a particular copyright holder is reinstated (a) provisionally, unless and until the copyright holder explicitly and finally terminates your license, and (b) permanently, if the copyright holder fails to notify you of the violation by some reasonable means prior to 60 days after the cessation. Moreover, your license from a particular copyright holder is reinstated permanently if the copyright holder notifies you of the violation by some reasonable means, this is the first time you have received notice of violation of this License (for any work) from that copyright holder, and you cure the violation prior to 30 days after your receipt of the notice. Termination of your rights under this section does not terminate the licenses of parties who have received copies or rights from you under this License. If your rights have been terminated and not permanently reinstated, you do not qualify to receive new licenses for the same material under section 10. 9. Acceptance Not Required for Having Copies. You are not required to accept this License in order to receive or run a copy of the Program. Ancillary propagation of a covered work occurring solely as a consequence of using peer-to-peer transmission to receive a copy likewise does not require acceptance. However, nothing other than this License grants you permission to propagate or modify any covered work. These actions infringe copyright if you do not accept this License. Therefore, by modifying or propagating a covered work, you indicate your acceptance of this License to do so. 10. Automatic Licensing of Downstream Recipients. Each time you convey a covered work, the recipient automatically receives a license from the original licensors, to run, modify and propagate that work, subject to this License. You are not responsible for enforcing compliance by third parties with this License. An "entity transaction" is a transaction transferring control of an organization, or substantially all assets of one, or subdividing an organization, or merging organizations. If propagation of a covered work results from an entity transaction, each party to that transaction who receives a copy of the work also receives whatever licenses to the work the party's predecessor in interest had or could give under the previous paragraph, plus a right to possession of the Corresponding Source of the work from the predecessor in interest, if the predecessor has it or can get it with reasonable efforts. You may not impose any further restrictions on the exercise of the rights granted or affirmed under this License. For example, you may not impose a license fee, royalty, or other charge for exercise of rights granted under this License, and you may not initiate litigation (including a cross-claim or counterclaim in a lawsuit) alleging that any patent claim is infringed by making, using, selling, offering for sale, or importing the Program or any portion of it. 11. Patents. A "contributor" is a copyright holder who authorizes use under this License of the Program or a work on which the Program is based. The work thus licensed is called the contributor's "contributor version". A contributor's "essential patent claims" are all patent claims owned or controlled by the contributor, whether already acquired or hereafter acquired, that would be infringed by some manner, permitted by this License, of making, using, or selling its contributor version, but do not include claims that would be infringed only as a consequence of further modification of the contributor version. For purposes of this definition, "control" includes the right to grant patent sublicenses in a manner consistent with the requirements of this License. Each contributor grants you a non-exclusive, worldwide, royalty-free patent license under the contributor's essential patent claims, to make, use, sell, offer for sale, import and otherwise run, modify and propagate the contents of its contributor version. In the following three paragraphs, a "patent license" is any express agreement or commitment, however denominated, not to enforce a patent (such as an express permission to practice a patent or covenant not to sue for patent infringement). To "grant" such a patent license to a party means to make such an agreement or commitment not to enforce a patent against the party. If you convey a covered work, knowingly relying on a patent license, and the Corresponding Source of the work is not available for anyone to copy, free of charge and under the terms of this License, through a publicly available network server or other readily accessible means, then you must either (1) cause the Corresponding Source to be so available, or (2) arrange to deprive yourself of the benefit of the patent license for this particular work, or (3) arrange, in a manner consistent with the requirements of this License, to extend the patent license to downstream recipients. "Knowingly relying" means you have actual knowledge that, but for the patent license, your conveying the covered work in a country, or your recipient's use of the covered work in a country, would infringe one or more identifiable patents in that country that you have reason to believe are valid. If, pursuant to or in connection with a single transaction or arrangement, you convey, or propagate by procuring conveyance of, a covered work, and grant a patent license to some of the parties receiving the covered work authorizing them to use, propagate, modify or convey a specific copy of the covered work, then the patent license you grant is automatically extended to all recipients of the covered work and works based on it. A patent license is "discriminatory" if it does not include within the scope of its coverage, prohibits the exercise of, or is conditioned on the non-exercise of one or more of the rights that are specifically granted under this License. You may not convey a covered work if you are a party to an arrangement with a third party that is in the business of distributing software, under which you make payment to the third party based on the extent of your activity of conveying the work, and under which the third party grants, to any of the parties who would receive the covered work from you, a discriminatory patent license (a) in connection with copies of the covered work conveyed by you (or copies made from those copies), or (b) primarily for and in connection with specific products or compilations that contain the covered work, unless you entered into that arrangement, or that patent license was granted, prior to 28 March 2007. Nothing in this License shall be construed as excluding or limiting any implied license or other defenses to infringement that may otherwise be available to you under applicable patent law. 12. No Surrender of Others' Freedom. If conditions are imposed on you (whether by court order, agreement or otherwise) that contradict the conditions of this License, they do not excuse you from the conditions of this License. If you cannot convey a covered work so as to satisfy simultaneously your obligations under this License and any other pertinent obligations, then as a consequence you may not convey it at all. For example, if you agree to terms that obligate you to collect a royalty for further conveying from those to whom you convey the Program, the only way you could satisfy both those terms and this License would be to refrain entirely from conveying the Program. 13. Use with the GNU Affero General Public License. Notwithstanding any other provision of this License, you have permission to link or combine any covered work with a work licensed under version 3 of the GNU Affero General Public License into a single combined work, and to convey the resulting work. The terms of this License will continue to apply to the part which is the covered work, but the special requirements of the GNU Affero General Public License, section 13, concerning interaction through a network will apply to the combination as such. 14. Revised Versions of this License. The Free Software Foundation may publish revised and/or new versions of the GNU General Public License from time to time. Such new versions will be similar in spirit to the present version, but may differ in detail to address new problems or concerns. Each version is given a distinguishing version number. If the Program specifies that a certain numbered version of the GNU General Public License "or any later version" applies to it, you have the option of following the terms and conditions either of that numbered version or of any later version published by the Free Software Foundation. If the Program does not specify a version number of the GNU General Public License, you may choose any version ever published by the Free Software Foundation. If the Program specifies that a proxy can decide which future versions of the GNU General Public License can be used, that proxy's public statement of acceptance of a version permanently authorizes you to choose that version for the Program. Later license versions may give you additional or different permissions. However, no additional obligations are imposed on any author or copyright holder as a result of your choosing to follow a later version. 15. Disclaimer of Warranty. THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. 16. Limitation of Liability. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. 17. Interpretation of Sections 15 and 16. If the disclaimer of warranty and limitation of liability provided above cannot be given local legal effect according to their terms, reviewing courts shall apply local law that most closely approximates an absolute waiver of all civil liability in connection with the Program, unless a warranty or assumption of liability accompanies a copy of the Program in return for a fee. END OF TERMS AND CONDITIONS How to Apply These Terms to Your New Programs If you develop a new program, and you want it to be of the greatest possible use to the public, the best way to achieve this is to make it free software which everyone can redistribute and change under these terms. To do so, attach the following notices to the program. It is safest to attach them to the start of each source file to most effectively state the exclusion of warranty; and each file should have at least the "copyright" line and a pointer to where the full notice is found. Copyright (C) This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . Also add information on how to contact you by electronic and paper mail. If the program does terminal interaction, make it output a short notice like this when it starts in an interactive mode: Copyright (C) This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. This is free software, and you are welcome to redistribute it under certain conditions; type `show c' for details. The hypothetical commands `show w' and `show c' should show the appropriate parts of the General Public License. Of course, your program's commands might be different; for a GUI interface, you would use an "about box". You should also get your employer (if you work as a programmer) or school, if any, to sign a "copyright disclaimer" for the program, if necessary. For more information on this, and how to apply and follow the GNU GPL, see . The GNU General Public License does not permit incorporating your program into proprietary programs. If your program is a subroutine library, you may consider it more useful to permit linking proprietary applications with the library. If this is what you want to do, use the GNU Lesser General Public License instead of this License. But first, please read . ================================================ FILE: README.md ================================================
# Converter NOW
Icon


## Why Converter NOW Tired of complicated, cluttered, and ad-filled unit converters? Converter NOW is your perfect solution! A few years ago, we realized that most unit converters were not only visually unappealing but also lacked functionality, privacy, and ease of use. So, we created Converter NOW to change that! 🚀 **Fast & Intuitive**: Experience lightning-fast, real-time conversions as soon as you start typing. No more navigating through confusing menus – just instant results with all measurement units! 🛠️ **Fully Customizable**: Tailor your experience by reorganizing units based on your preferences and priorities. Make Converter NOW truly yours! 🔢 **Built-in Calculator**: Perform complex calculations effortlessly on any page. No need to switch between apps – it's all integrated seamlessly. 💱 **Daily Currency Exchange Updates**: Stay up-to-date with accurate, real-time currency conversions updated daily, perfect for travelers and global professionals. 🎨 **Personalize Your Style**: Choose your favorite accent color, matching your device's theme or your unique style preferences. ⚫⚪ **Light & Dark Modes**: Switch between dark and light themes for a comfortable experience, day or night. 📱🖥️ **Cross-Platform Support**: Enjoy Converter NOW on Android, Web, Linux, and Windows. Whether you're on your phone or desktop, we've got you covered! 💯 **Completely Free**: No ads, no data collection, and no intrusive permissions (just internet access for currency updates). Best of all, it's **100% open source!** ## Alternative installation sources ### Winget Run the following command: ```bash winget install -e 9P0Q79HWJH72 ``` ### Container self-hosted web app You can easily self-host the Converter NOW web app on your own machine using either Docker or Podman. Simply run one of the commands below, depending on your containerization tool of choice: ```bash docker run -d -p 8080:80 ghcr.io/ferraridamiano/converternow:latest podman run -d -p 8080:80 ghcr.io/ferraridamiano/converternow:latest ``` After running the command, the web app will be accessible in your web browser at `http://localhost:8080`. ## Build from source code First you need to [install flutter](https://docs.flutter.dev/get-started/install) and all the tooling for your target platform (e.g. Android Studio for Android, etc.). Then you have to install `melos` in order to do some pre-compiling task: ```shell dart pub global activate melos ``` Then, run `melos bootstrap` in the root of the project in order to get all the dependencies, generate the translation files and optimize the `svg` icons. Then follow the instructions for the platform you want to target. ### Android On Android you should first disable the signing option in [`android/app/build.gradle.kts`](https://github.com/ferraridamiano/ConverterNOW/blob/master/android/app/build.gradle.kts#L42) Then you can type `flutter build apk --split-per-abi` to compile the code. You can find the output in `build/app/outputs/flutter-apk` folder. ### Linux Type `flutter build linux` to build the Linux app. You can find the output in `build/linux/x64/release/bundle`. ### Windows Type `flutter build windows` to build the Windows app. You can find the output in `build/windows/runner/Release`. ### Web Type `flutter build web` to build the Web app. You can find the output in `build/web`. Note: if you deploy it not on the base path, add `--base-href=/path/` to the command above, change `/path/` with your path. ### Web app on Docker You can build the Docker image of Converter NOW in your environment with the following command: ```shell docker buildx build -t converternow . ``` Then, run the docker container: ```shell docker run -d -p 80:80 converternow ``` You can access it via browser at `localhost:80` ### iOS and MacOS The app is not tested against iOS and MacOS, but you should be able to compile it even for these platforms. You first need to generate the platform specific code and then compile them. ================================================ FILE: analysis_options.yaml ================================================ include: package:flutter_lints/flutter.yaml linter: rules: prefer_const_constructors: true prefer_const_declarations: true prefer_const_literals_to_create_immutables: true ================================================ FILE: android/.gitignore ================================================ gradle-wrapper.jar /.gradle /captures/ /gradlew /gradlew.bat /local.properties GeneratedPluginRegistrant.java .cxx/ # Remember to never publicly share your keystore. # See https://flutter.dev/to/reference-keystore key.properties **/*.keystore **/*.jks ================================================ FILE: android/app/build.gradle.kts ================================================ import java.util.Properties import java.io.FileInputStream plugins { id("com.android.application") id("kotlin-android") // The Flutter Gradle Plugin must be applied after the Android and Kotlin Gradle plugins. id("dev.flutter.flutter-gradle-plugin") } val keystoreProperties = Properties() val keystorePropertiesFile = rootProject.file("key.properties") if (keystorePropertiesFile.exists()) { keystoreProperties.load(FileInputStream(keystorePropertiesFile)) } android { namespace = "com.ferrarid.converterpro" compileSdk = flutter.compileSdkVersion ndkVersion = flutter.ndkVersion compileOptions { sourceCompatibility = JavaVersion.VERSION_17 targetCompatibility = JavaVersion.VERSION_17 } kotlinOptions { jvmTarget = JavaVersion.VERSION_17.toString() } defaultConfig { // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html). applicationId = "com.ferrarid.converterpro" // You can update the following values to match your application needs. // For more information, see: https://flutter.dev/to/review-gradle-config. minSdk = flutter.minSdkVersion targetSdk = flutter.targetSdkVersion versionCode = flutter.versionCode versionName = flutter.versionName } signingConfigs { create("release") { keyAlias = keystoreProperties["keyAlias"] as String keyPassword = keystoreProperties["keyPassword"] as String storeFile = keystoreProperties["storeFile"]?.let { file(it) } storePassword = keystoreProperties["storePassword"] as String } } buildTypes { release { signingConfig = signingConfigs.getByName("release") } } } flutter { source = "../.." } ================================================ FILE: android/app/src/debug/AndroidManifest.xml ================================================ ================================================ FILE: android/app/src/main/AndroidManifest.xml ================================================ ================================================ FILE: android/app/src/main/kotlin/com/ferrarid/converterpro/MainActivity.kt ================================================ package com.ferrarid.converterpro import io.flutter.embedding.android.FlutterActivity class MainActivity : FlutterActivity() ================================================ FILE: android/app/src/main/res/drawable/ic_launcher_background.xml ================================================ ================================================ FILE: android/app/src/main/res/drawable/ic_launcher_foreground.xml ================================================ ================================================ FILE: android/app/src/main/res/drawable/launch_background.xml ================================================ ================================================ FILE: android/app/src/main/res/drawable-v21/launch_background.xml ================================================ ================================================ FILE: android/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml ================================================ ================================================ FILE: android/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml ================================================ ================================================ FILE: android/app/src/main/res/values/styles.xml ================================================ ================================================ FILE: android/app/src/main/res/values-night/styles.xml ================================================ ================================================ FILE: android/app/src/profile/AndroidManifest.xml ================================================ ================================================ FILE: android/build.gradle.kts ================================================ allprojects { repositories { google() mavenCentral() } } val newBuildDir: Directory = rootProject.layout.buildDirectory .dir("../../build") .get() rootProject.layout.buildDirectory.value(newBuildDir) subprojects { val newSubprojectBuildDir: Directory = newBuildDir.dir(project.name) project.layout.buildDirectory.value(newSubprojectBuildDir) } subprojects { project.evaluationDependsOn(":app") } tasks.register("clean") { delete(rootProject.layout.buildDirectory) } ================================================ FILE: android/gradle/wrapper/gradle-wrapper.properties ================================================ distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists distributionUrl=https\://services.gradle.org/distributions/gradle-8.14-all.zip ================================================ FILE: android/gradle.properties ================================================ org.gradle.jvmargs=-Xmx8G -XX:MaxMetaspaceSize=4G -XX:ReservedCodeCacheSize=512m -XX:+HeapDumpOnOutOfMemoryError android.useAndroidX=true ================================================ FILE: android/gradlew ================================================ #!/usr/bin/env bash ############################################################################## ## ## Gradle start up script for UN*X ## ############################################################################## # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. DEFAULT_JVM_OPTS="" APP_NAME="Gradle" APP_BASE_NAME=`basename "$0"` # Use the maximum available, or set MAX_FD != -1 to use that value. MAX_FD="maximum" warn ( ) { echo "$*" } die ( ) { echo echo "$*" echo exit 1 } # OS specific support (must be 'true' or 'false'). cygwin=false msys=false darwin=false case "`uname`" in CYGWIN* ) cygwin=true ;; Darwin* ) darwin=true ;; MINGW* ) msys=true ;; esac # Attempt to set APP_HOME # Resolve links: $0 may be a link PRG="$0" # Need this for relative symlinks. while [ -h "$PRG" ] ; do ls=`ls -ld "$PRG"` link=`expr "$ls" : '.*-> \(.*\)$'` if expr "$link" : '/.*' > /dev/null; then PRG="$link" else PRG=`dirname "$PRG"`"/$link" fi done SAVED="`pwd`" cd "`dirname \"$PRG\"`/" >/dev/null APP_HOME="`pwd -P`" cd "$SAVED" >/dev/null CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar # Determine the Java command to use to start the JVM. if [ -n "$JAVA_HOME" ] ; then if [ -x "$JAVA_HOME/jre/sh/java" ] ; then # IBM's JDK on AIX uses strange locations for the executables JAVACMD="$JAVA_HOME/jre/sh/java" else JAVACMD="$JAVA_HOME/bin/java" fi if [ ! -x "$JAVACMD" ] ; then die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME Please set the JAVA_HOME variable in your environment to match the location of your Java installation." fi else JAVACMD="java" which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. Please set the JAVA_HOME variable in your environment to match the location of your Java installation." fi # Increase the maximum file descriptors if we can. if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then MAX_FD_LIMIT=`ulimit -H -n` if [ $? -eq 0 ] ; then if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then MAX_FD="$MAX_FD_LIMIT" fi ulimit -n $MAX_FD if [ $? -ne 0 ] ; then warn "Could not set maximum file descriptor limit: $MAX_FD" fi else warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" fi fi # For Darwin, add options to specify how the application appears in the dock if $darwin; then GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" fi # For Cygwin, switch paths to Windows format before running java if $cygwin ; then APP_HOME=`cygpath --path --mixed "$APP_HOME"` CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` JAVACMD=`cygpath --unix "$JAVACMD"` # We build the pattern for arguments to be converted via cygpath ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` SEP="" for dir in $ROOTDIRSRAW ; do ROOTDIRS="$ROOTDIRS$SEP$dir" SEP="|" done OURCYGPATTERN="(^($ROOTDIRS))" # Add a user-defined pattern to the cygpath arguments if [ "$GRADLE_CYGPATTERN" != "" ] ; then OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" fi # Now convert the arguments - kludge to limit ourselves to /bin/sh i=0 for arg in "$@" ; do CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` else eval `echo args$i`="\"$arg\"" fi i=$((i+1)) done case $i in (0) set -- ;; (1) set -- "$args0" ;; (2) set -- "$args0" "$args1" ;; (3) set -- "$args0" "$args1" "$args2" ;; (4) set -- "$args0" "$args1" "$args2" "$args3" ;; (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; esac fi # Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules function splitJvmOpts() { JVM_OPTS=("$@") } eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME" exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@" ================================================ FILE: android/gradlew.bat ================================================ @if "%DEBUG%" == "" @echo off @rem ########################################################################## @rem @rem Gradle startup script for Windows @rem @rem ########################################################################## @rem Set local scope for the variables with windows NT shell if "%OS%"=="Windows_NT" setlocal @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. set DEFAULT_JVM_OPTS= set DIRNAME=%~dp0 if "%DIRNAME%" == "" set DIRNAME=. set APP_BASE_NAME=%~n0 set APP_HOME=%DIRNAME% @rem Find java.exe if defined JAVA_HOME goto findJavaFromJavaHome set JAVA_EXE=java.exe %JAVA_EXE% -version >NUL 2>&1 if "%ERRORLEVEL%" == "0" goto init echo. echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. echo. echo Please set the JAVA_HOME variable in your environment to match the echo location of your Java installation. goto fail :findJavaFromJavaHome set JAVA_HOME=%JAVA_HOME:"=% set JAVA_EXE=%JAVA_HOME%/bin/java.exe if exist "%JAVA_EXE%" goto init echo. echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% echo. echo Please set the JAVA_HOME variable in your environment to match the echo location of your Java installation. goto fail :init @rem Get command-line arguments, handling Windowz variants if not "%OS%" == "Windows_NT" goto win9xME_args if "%@eval[2+2]" == "4" goto 4NT_args :win9xME_args @rem Slurp the command line arguments. set CMD_LINE_ARGS= set _SKIP=2 :win9xME_args_slurp if "x%~1" == "x" goto execute set CMD_LINE_ARGS=%* goto execute :4NT_args @rem Get arguments from the 4NT Shell from JP Software set CMD_LINE_ARGS=%$ :execute @rem Setup the command line set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar @rem Execute Gradle "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% :end @rem End local scope for the variables with windows NT shell if "%ERRORLEVEL%"=="0" goto mainEnd :fail rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of rem the _cmd.exe /c_ return code! if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 exit /b 1 :mainEnd if "%OS%"=="Windows_NT" endlocal :omega ================================================ FILE: android/settings.gradle.kts ================================================ pluginManagement { val flutterSdkPath = run { val properties = java.util.Properties() file("local.properties").inputStream().use { properties.load(it) } val flutterSdkPath = properties.getProperty("flutter.sdk") require(flutterSdkPath != null) { "flutter.sdk not set in local.properties" } flutterSdkPath } includeBuild("$flutterSdkPath/packages/flutter_tools/gradle") repositories { google() mavenCentral() gradlePluginPortal() } } plugins { id("dev.flutter.flutter-plugin-loader") version "1.0.0" id("com.android.application") version "8.11.1" apply false id("org.jetbrains.kotlin.android") version "2.2.20" apply false } include(":app") ================================================ FILE: assets/fonts/OFL.txt ================================================ Copyright 2010 The Josefin Sans Project Authors (https://github.com/ThomasJockin/JosefinSansFont-master), with Reserved Font Name "Josefin Sans". This Font Software is licensed under the SIL Open Font License, Version 1.1. This license is copied below, and is also available with a FAQ at: https://openfontlicense.org ----------------------------------------------------------- SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007 ----------------------------------------------------------- PREAMBLE The goals of the Open Font License (OFL) are to stimulate worldwide development of collaborative font projects, to support the font creation efforts of academic and linguistic communities, and to provide a free and open framework in which fonts may be shared and improved in partnership with others. The OFL allows the licensed fonts to be used, studied, modified and redistributed freely as long as they are not sold by themselves. The fonts, including any derivative works, can be bundled, embedded, redistributed and/or sold with any software provided that any reserved names are not used by derivative works. The fonts and derivatives, however, cannot be released under any other type of license. The requirement for fonts to remain under this license does not apply to any document created using the fonts or their derivatives. DEFINITIONS "Font Software" refers to the set of files released by the Copyright Holder(s) under this license and clearly marked as such. This may include source files, build scripts and documentation. "Reserved Font Name" refers to any names specified as such after the copyright statement(s). "Original Version" refers to the collection of Font Software components as distributed by the Copyright Holder(s). "Modified Version" refers to any derivative made by adding to, deleting, or substituting -- in part or in whole -- any of the components of the Original Version, by changing formats or by porting the Font Software to a new environment. "Author" refers to any designer, engineer, programmer, technical writer or other person who contributed to the Font Software. PERMISSION & CONDITIONS Permission is hereby granted, free of charge, to any person obtaining a copy of the Font Software, to use, study, copy, merge, embed, modify, redistribute, and sell modified and unmodified copies of the Font Software, subject to the following conditions: 1) Neither the Font Software nor any of its individual components, in Original or Modified Versions, may be sold by itself. 2) Original or Modified Versions of the Font Software may be bundled, redistributed and/or sold with any software, provided that each copy contains the above copyright notice and this license. These can be included either as stand-alone text files, human-readable headers or in the appropriate machine-readable metadata fields within text or binary files as long as those fields can be easily viewed by the user. 3) No Modified Version of the Font Software may use the Reserved Font Name(s) unless explicit written permission is granted by the corresponding Copyright Holder. This restriction only applies to the primary font name as presented to the users. 4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font Software shall not be used to promote, endorse or advertise any Modified Version, except to acknowledge the contribution(s) of the Copyright Holder(s) and the Author(s) or with their explicit written permission. 5) The Font Software, modified or unmodified, in part or in whole, must be distributed entirely under this license, and must not be distributed under any other license. The requirement for fonts to remain under this license does not apply to any document created using the Font Software. TERMINATION This license becomes null and void if any of the above conditions are not met. DISCLAIMER THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM OTHER DEALINGS IN THE FONT SOFTWARE. ================================================ FILE: cliff.toml ================================================ [remote.github] owner = "ferraridamiano" repo = "ConverterNOW" [changelog] header = "" body = """ {%- macro remote_url() -%} https://github.com/{{ remote.github.owner }}/{{ remote.github.repo }} {%- endmacro -%} {%- macro github_username(email) -%} {%- if email is containing("@users.noreply.github.com") -%} {{ email | split(pat="@") | first | split(pat="+") | last }} {%- elif email is containing("@") -%} {{ email | split(pat="@") | first }} {%- else -%} {{ email }} {%- endif -%} {%- endmacro -%} {% for group, commits in commits | group_by(attribute="group") %} ### {{ group | upper_first }} {% for commit in commits %} - {{ commit.message | split(pat="\n") | first | upper_first | trim }} ([`{{ commit.id | truncate(length=7, end="") }}`]({{ self::remote_url() }}/commit/{{ commit.id }}))\ {% for footer in commit.footers -%} , {{ footer.token }}{{ footer.separator }}{{ footer.value }}\ {% endfor %}\ {% endfor %} {% endfor %} ### 👥 Contributors {% set contributors = commits | unique(attribute="author.email") | sort(attribute="author.email") -%} {% for contributor in contributors -%} - @{{ self::github_username(email=contributor.author.email) }} {% endfor %}\n """ [git] conventional_commits = true filter_unconventional = false commit_parsers = [ { message = "^feat", group = "🚀 Features" }, { message = "^fix", group = "🐛 Bug Fixes" }, { message = "^doc", group = "📝 Documentation" }, { message = "^perf", group = "⚡ Performance" }, { message = "^refactor", group = "🧹 Refactor" }, { message = "^style", group = "🎨 Styling" }, { message = "^test", group = "🧪 Testing" }, { message = "^chore\\(deps.*\\)", skip = true }, { message = "^chore\\(pr\\)", skip = true }, { message = "^chore\\(pull\\)", skip = true }, { message = "^chore\\(release\\)", skip = true }, { message = "^chore|^ci", group = "🧰 Miscellaneous Tasks" }, { body = ".*security", group = "🔐 Security" }, ] filter_commits = false topo_order = false sort_commits = "oldest" ================================================ FILE: fastlane/metadata/android/ar/full_description.txt ================================================ Converter NOW هو محول فعال للوحدات والعُملات ✔️ يقوم بتبسيط عملية التحويل المُملة بين وحدات القياس إلى بِضع نقرات. 🚀 سريع وفوري: ما عليك سوى البدء في الكتابة وستحصل على الفور على تحويل في الوقت الفعلي مع جميع وحدات القياس الأخرى. 🖌️قابل للتخصيص: يُمكنك إعادة تنظيم الوحدات وفقًا لأولوياتك واستخدامك الخاصة. 🔢 يأتي مُتكامل مع آلة حاسبة تتيح لك إجراء العمليات الحسابية في كل صفحة. 💰 نحن نقوم بالتحديث بصورة يوميةلقيم العملات. ⚫⚪إختر مظهرك المُفضل: بين الوضع فاتح أو المُظلم 📱دعم كامل للهواتف الذكية و الأجهزة اللوحية. 💯مفتوح المصدر بشكل كامل! تفقد المستودع على منصة ال Github: https://github.com/ferraridamiano/ConverterNOW لدينا أيضًا تطبيق ويب! 🖥️ تفقده من خلال هذا الرابط: https://ferraridamiano.github.io/ConverterNOW/ 🌍 Converter NOW, بدون إعلانات ، لا يتطلب أي إذن ونحن لا نجمع بياناتك 🌍 Converter NOW يمكن تحويل أكثر من 200 وحدة قياس وأكثر من 30 عملة 🎉🎉 فيما يلي بعض الكميات المادية التي يستطيع Converter Now تحويلها: -العُمُلات: دولار, يورو, جنيه, روبيه, ين, وما إلى ذلك. -الطول: الأمتار ، البوصات ، الأميال ، الساحات ، السنوات الضوئية ، وما إلى ذلك. -المساحة: متر مربع ، هكتار ، فدان ، وما إلى ذلك. -الكمية: متر مكعب ، لترات ، جالون ، مكاييل ، ملاعق ، وما إلى ذلك. -الزمن: ثواني ، ساعات ، أيام ، سنوات ، أسابيع ، آلاف السنين ، وما إلى ذلك. -درجة الحرارة: مئوية ، فهرنهايت ، كيلفن -السرعة: متر في الثانية ، كيلومترات في الساعة ، عقدة ، وما إلى ذلك. -الكُتلة: جرامات ، رطل ، طن ، وحدات الكتلة الذرية ، وما إلى ذلك. -القوة: نيوتن ، داين ، قوة الجنيه ، باوندال ، وما إلى ذلك -الضغط: باسكال ، بار ، أتومسفير ، رطل ، وما إلى ذلك. -الطاقة: الجول ، السعرات الحرارية ، كيلو واط ساعة ، وما إلى ذلك. -القدرة: واط ، كيلووات ، قوة حصان ، وما إلى ذلك. -إستهلاك الوقود: ميل لكل جالون ، كيلومتر لكل لتر ، وما إلى ذلك. -الأنظمة العددية: عشري ، ثنائي ، سداسي عشري ، وما إلى ذلك. -عزم الدوران: نيوتن متر ، قدم قوة الجنيه ، متر باوند ، وما إلى ذلك. -البيانات الرقمية: نايبل ، بت ، بايت ، كيبيت ، كيبي بايت ، وما إلى ذلك. -مقياس الحذاء: المملكة المتحدة ، الهند ، أوروبا ، الولايات المتحدة الأمريكية ، اليابان ، وما إلى ذلك. -الزاوية: الدرجة والراديان والدقائق وما إلى ذلك. -سوابق SI: كيلو ، ميجا ، جيجا ، تيرا ، ملي ، ميكرو ، نانو ، وما إلى ذلك. ================================================ FILE: fastlane/metadata/android/ar/short_description.txt ================================================ محول وحدات بسيط ،فوري وسريع! ================================================ FILE: fastlane/metadata/android/ar/title.txt ================================================ Converter NOW: مُحول وحدات ================================================ FILE: fastlane/metadata/android/bn/full_description.txt ================================================ কনভার্টার NOW একটি কার্যকরী ইউনিট ও কারেন্সি কনভার্টার ✔️ সহজ মাত্র কয়েক ক্লিকেই পরিমাপের বিভিন্ন এককে রুপান্তর। 🚀 এটি দ্রুত ও তাৎক্ষণিক :টাইপ শুরু করার সাথে সাথেই পরিমাপের অন্যান্য এককের রূপান্তর দেখতে পাবেন। 🖌️ এটি কাস্টমাইজযোগ্য: পছন্দ এবং ব্যবহার অনুযায়ী ইউনিটগুলি সাজিয়ে নিতে পারেন। 🔢 বিল্ট ইনক্যালকুলেটর প্রতিটি পেইজ এ রয়েছে ক্যালকুলেটর ব্যাবহারের সুযোগ। 💰 কারেন্সি এক্সচেঞ্জ রেট প্রতিদিন আপডেট করা হচ্ছে 🎨 ডাইনামিক থীম আপনার ডিভাইস সেটিংস অনুযায়ী নির্বাচিত করা হয়। ⚫⚪ পছন্দের থীম টি ব্যাবহার করুন: ডার্ক এবং লাইট থীম। 📱🖥️ মাল্টিপ্ল্যাটফর্ম: Android, Web, Linux and Windows এ ব্যাবহার করা যাবে। 💯 এটি ফ্রী, বিজ্ঞাপন মুক্ত, ডাটা কালেকশন এবং অনুমতির প্রয়োজন নেই (শুধু মূদ্রার মান আপডেট করার জন্য ইন্টারনেট) এবং সম্পূর্ণ ওপেন সোর্স! কনভার্টার NOW ২০০ টির বেশি ইউনিট এবং ৩০ টিরও বেশি মূদ্রা রুপান্তর করতে পারে 🎉🎉 যেসব পরিমাপ কনভার্টার NOW রূপান্তর করতে সক্ষমঃ -মুদ্রা: ডলার, ইউরো, পাউন্ড, রুপি, ইয়েন, ইত্যাদি। -দৈর্ঘ্য: মিটার, ইঞ্চি, মাইল, গজ, আলোকবর্ষ, ইত্যাদি। -ক্ষেত্রফল: বর্গ মিটার, হেক্টর, একর, ইত্যাদি। -আয়তন: ঘন মিটার, লিটার, গ্যালন, পিন্ট, চামচ, ইত্যাদি। -সময়: সেকেন্ড, ঘণ্টা, দিন, বছর, সপ্তাহ, সহস্রাব্দ, ইত্যাদি। -তাপমাত্রা: সেলসিয়াস, ফারেনহাইট, কেলভিন। -গতিবেগ: মিটার প্রতি সেকেন্ড, কিলোমিটার প্রতি ঘণ্টা, নট, ইত্যাদি। -ভর: গ্রাম, পাউন্ড, টন, পারমাণবিক ভর একক, ইত্যাদি। -বল: নিউটন, ডাইন, পাউন্ড-ফোর্স, পাউন্ডাল, ইত্যাদি। -চাপ: প্যাসকেল, বার, বায়ুমণ্ডল, পিএসআই, ইত্যাদি। -শক্তি: জুল, ক্যালরি, কিলোওয়াট ঘণ্টা, ইত্যাদি। -ক্ষমতা: ওয়াট, কিলোওয়াট, হর্সপাওয়ার, ইত্যাদি। -জ্বালানি খরচ: মাইল প্রতি গ্যালন, কিলোমিটার প্রতি লিটার, ইত্যাদি। -সংখ্যা পদ্ধতি: দশমিক, বাইনারি, হেক্সাডেসিমাল, ইত্যাদি। -টর্ক: নিউটন মিটার, পাউন্ড-ফোর্স ফুট, পাউন্ডাল মিটার, ইত্যাদি। -ডিজিটাল ডাটা: নিবল, বিট, বাইট, কিবিবিট, কিবিবাইট, ইত্যাদি। -জুতার মাপ: ইউকে, ভারত, ইউরোপ, ইউএসএ, জাপান, ইত্যাদি। -কোণ: ডিগ্রি, রেডিয়ান, মিনিট, ইত্যাদি। -এসআই উপসর্গ: কিলো, মেগা, গিগা, টেরা, মিলি, মাইক্রো, ন্যানো, ইত্যাদি। ================================================ FILE: fastlane/metadata/android/bn/short_description.txt ================================================ একটি সহজ, তাৎক্ষণিক এবং দ্রুত ইউনিট কনভার্টার ================================================ FILE: fastlane/metadata/android/bn/title.txt ================================================ কনভার্টার NOW: ইউনিট কনভার্ট ================================================ FILE: fastlane/metadata/android/de-DE/full_description.txt ================================================ Converter NOW ist ein leistungsstarker Einheiten- und Währungsumrechner ✔️ Vereinfache den aufwendigen Umrechnungsprozess zwischen den Maßeinheiten mit wenigen Klicks. 🚀 Er ist schnell und unmittelbar: beginne zu tippen und du erhälst prompt die Umrechnung in die anderen Maßeinheiten in Echtzeit. 🖌️ Er ist anpassungsfähig: gestalte die Reihenfolge der Maßeinheiten nach deinen Vorlieben. 🔢 Er besitzt einen eingebauten Taschenrechner, der einfache Berechnungen jederzeit zulässt. 💰 Tägliche Aktualisierung der Währungskurse. ⚫⚪ Wähle dein bevorzugstes Schema: Dunkel, Hell oder Schwarz für AMOLED. 📱 Vollständige Smartphone- und Tablet-Unterstützung. 💯 Wir sind eine Open-Source-Software! Schau dir unser Repository auf GitHub an: https://github.com/ferraridamiano/ConverterNOW Wir haben auch eine Web-App! 🖥️ Schau vorbei: https://ferraridamiano.github.io/ConverterNOW/ 🌍 Converter NOW ist frei, ohne Werbung, setzt keinerlei Berechtigungen voraus und wir sammeln keine persönlichen Daten über dich 🌍 Converter NOW kann mehr als 200 Maßeinheiten und 30+ Währungen umwandeln. 🎉🎉 Hier sind einige Beispiele für physikale Größen und Kurse welche Converter NOW beherrscht: -Währungen: Dollar (USA & Kanada), Euro, Pfund, Rupee, Yen, etc. -Länge: Meter, Zoll, Meilen, Yards, Lichtjahre, etc. -Fläche: Quadratmeter, Hektar, Acre, etc. -Volumen: Kubikmeter, Liter, Gallonen und Pints (US & GB), Teelöfel (US & Australien), etc. -Zeit: Sekunden, Stunden, Tage, Jahre, Wochen, Jahrtausende, etc. -Temperatur: Celsius, Fahreneit, Kelvin, etc. -Geschwindigkeit: Meter pro Sekunde, Kilometer pro Stunde, Knoten, etc. -Masse: Gramm, Pfund, Tonen, Atomare Masseneinheiten, etc. -Kraft: Newton, Dyn, Pound-force (Kraftpfund), Poundal, etc -Druck: Pascal, Bar, Physikalische Atmosphäre (atm), Pfund pro Quadratzoll (PSI), etc. -Enrgie: Joule, Kalorien, Kilowattstunden (kWh), etc. -Leistung: Watt, Kilowatt, Pferdestärken (PS), etc. -Spritverbrauch: Meilen pro Gallone (US und UK), Kilometer pro Liter, etc. -Zahlensysteme: Dezimal, Binär, Hexadezimal, etc. -Drehmoment: Newtonmeter, Kilopondmeter, Poundalmeter, etc. -Digitale Daten: Nibble, Bit, Byte, Kibibit, Kibibyte, etc. -Schuhgröße: UK, Indien, Europa, USA, Japan, etc. -Winkel: Grad, Radiant, Minuten, etc. -SI Präfix: Kilo, Mega, Giga, Tera, Milli, Mikro, Nano, etc. ================================================ FILE: fastlane/metadata/android/de-DE/short_description.txt ================================================ Einfache, unmittelbare und schnelle Umrechnung von Einheiten! ================================================ FILE: fastlane/metadata/android/de-DE/title.txt ================================================ Einheitenumwandler ================================================ FILE: fastlane/metadata/android/el/full_description.txt ================================================ Converter NOW είναι ένα αποτελεσματικό εργαλείο για μετατροπή μονάδων και νομισμάτων ✔️ Απλοποιεί την κουραστική διαδικασία μετατροπής μεταξύ μονάδων μέτρησης με λίγα κλικ. 🚀 Είναι γρήγορο και άμεσο: απλώς αρχίστε να πληκτρολογείτε και έχετε αμέσως τη μετατροπή σε πραγματικό χρόνο με όλες τις άλλες μονάδες μέτρησης. 🖌️ Είναι προσαρμόσιμο: οι μονάδες μπορούν να οργανωθούν σύμφωνα με τις προτεραιότητες και τις ανάγκες σας. 🔢 Ενσωματώνει υπολογιστή που σας επιτρέπει να κάνετε υπολογισμούς σε κάθε σελίδα. 💰 Ισοτιμίες νομισμάτων ενημερώνονται καθημερινά 🎨 Δυναμικό θέμα βασισμένο στις ρυθμίσεις της συσκευής σας ⚫⚪ Επιλέξτε το αγαπημένο σας θέμα: σκοτεινό ή φωτεινό 📱🖥️ Πολλαπλές πλατφόρμες: διαθέσιμο για Android, Web, Linux και Windows 💯 Είναι δωρεάν, χωρίς διαφημίσεις, χωρίς συλλογή δεδομένων, χωρίς απαιτήσεις δικαιωμάτων (μόνο Ίντερνετ για ενημέρωση των ισοτιμιών) και κυρίως είναι ανοιχτού κώδικα! Το Converter NOW μπορεί να μετατρέψει 200+ μονάδες μέτρησης και 30+ νομίσματα 🎉🎉 Ακολουθούν ορισμένες από τις φυσικές ποσότητες που μπορεί να μετατρέψει το Converter NOW: -Νομίσματα: Δολάρια, Ευρώ, Λίρα, Ρουπία, Γιεν, κ.λπ. -Μήκος: μέτρα, ίντσες, μίλια, γιάρδες, έτη φωτός, κ.λπ. -Επιφάνεια: square meters, hectares, acres, etc. -Volume: τετραγωνικά μέτρα, εκτάρια, στρέμματα, κ.λπ. -Χρόνος: δευτερόλεπτα, ώρες, ημέρες, έτη, εβδομάδες, χιλιετίες, κ.λπ. -Θερμοκρασία: Κελσίου, Φαρενάιτ, Κέλβιν -Ταχύτητα: μέτρα ανά δευτερόλεπτο, χιλιόμετρα ανά ώρα, κόμβοι, κ.λπ. -Μάζα: γραμμάρια, λίβρες, τόνοι, ατομικές μάζες, κ.λπ. -Δύναμη: Νιούτον, dyne, pound-force, Πουντάλ, κ.λπ. -Πίεση: Πασκάλ, μπαρ, ατμόσφαιρα, psi, κ.λπ. -Ενέργεια: Τζάουλ, θερμίδες, κιλοβατώρες, κ.λπ. -Ισχύς: Βατ, κιλοβατ, ίπποι, κ.λπ. -Κατανάλωση καυσίμου: Μίλια ανά γαλόνι, Χιλιόμετρα ανά λίτρο, κ.λπ. -Συστήματα αριθμών: δεκαδικό, δυαδικό, εξαδικό, κ.λπ. -Ροπή: Νιούτον μέτρο, λίβρα-δύναμη πόδι, πουντάλ μέτρο, κ.λπ. -Ψηφιακά δεδομένα: Νίμπλ, μπίτ, μπάιτ, κιλομπάιτ, etc. -Μέγεθος παπουτσιών: Ηνωμένο Βασίλειο, Ινδία, Ευρώπη, ΗΠΑ, Ιαπωνία, κ.λπ. -Γωνίες: μοίρες, ακτίνια, λεπτά, κ.λπ. -SI προθέματα: κιλό, μέγα, γίγα, τέρα, μίλι, μικρο, νάνο, κ.λπ. ================================================ FILE: fastlane/metadata/android/el/short_description.txt ================================================ Ένας απλός, άμεσος και γρήγορος μετατροπέας μονάδων! ================================================ FILE: fastlane/metadata/android/el/title.txt ================================================ Μετατροπέας Μονάδων ================================================ FILE: fastlane/metadata/android/en-US/changelogs/24.txt ================================================ ★ 3 new languages: Portoguese, French and Norwegian 🌐 ★ Bugfix of some translation and shoe size conversion 🔧 Thanks to @mezysinc, @Torok42, @FTno, @dizietsma and Andreas 🔝 We also have a web app, check it out here: https://ferraridamiano.github.io/ConverterNOW/#/ What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/25.txt ================================================ ★ Bugfix of numerical systems conversion 🔧 We also have a web app, check it out here: https://ferraridamiano.github.io/ConverterNOW/#/ What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/26.txt ================================================ ★ You can now set dark theme on old devices ★ Dark theme get darker with AMOLED mode ⚫ ★ Added some units such as: pennyweights, troy ounces, stones, fluid ounces, gill, etc. ★ Added german and russian ★ Bug fix Thanks to Gena Haltmair and Vadim Young for your contributions! What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/27.txt ================================================ ★ Default language fix ★ You can now set dark theme on old devices ★ Dark theme get darker with AMOLED mode ⚫ ★ Added some units such as: pennyweights, troy ounces, stones, fluid ounces, gill, etc. ★ Added german and russian ★ Bug fix Thanks to Gena Haltmair and Vadim Young for your contributions! What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/28.txt ================================================ ★ Improved tablet support ★ Added some new units ★ Calculator improvements ★ Bugfix exchange rate update ★ Added Spanish Thanks to Gena Haltmair and n-berenice for your contributions! What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/29.txt ================================================ ★ Huge UI improvement for large screens ★ Support for adaptive icon ★ Added some new units ★ Added a splash screen ★ Change app language from settings ★ Added Croatian ★ Bugfix Thanks to Giovanni Spitti and @milotype for your contributions! What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/30.txt ================================================ ★ Fix the icon bug with Android version below 8.0 ★ Huge UI improvement for large screens ★ Support for adaptive icon ★ Added some new units ★ Added a splash screen ★ Change app language from settings ★ Added Croatian ★ Bugfix Thanks to Giovanni Spitti and @milotype for your contributions! What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/31.txt ================================================ ★ Improved the accessibility of the app ★ Added an option to undo a "Clear all" operation ★ Added indonesian and japanese ★ Added more units of measurement ★ Performance improvements ★ Bugfix Thanks to @the7thNightmare and 匿名 for your contributions! What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/33.txt ================================================ ★ Added app shortcut ★ Small graphical tweaks ★ Bugfix Thanks to @the7thNightmare and 匿名 for your contributions! What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/34.txt ================================================ ★ Now the app is available in arabic ★ Small tweaks ★ Bugfix Thanks to Omer Maki for your contributions! What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/35.txt ================================================ ★ Improved the calculator ★ Bugfix What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/36.txt ================================================ ★ Migration to Material You ★ Themed icons on Android 13+ ★ Added Polish translation ★ Bugfixes Thanks to @rehork for your contributions! What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/37.txt ================================================ ★ Unit symbols are now always visible Thanks to @rehork for your suggestion! What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/38.txt ================================================ Introducing Converter NOW 4.0! - Finished the migration to Material You - Dynamic theming support on Android 12+ - New icons - Translations updates - Bug fixes and general improvements What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/39.txt ================================================ - Fixed bug with currencies page after v4 update - Improved polish translations What do you think about Converter NOW? Rate us! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/40.txt ================================================ - Material you improvements - The calculator has a new look - Added dutch, chinese and traditional chinese translations - Improvements to other translations - Icons tweaks - General improvements Thanks to @David-git-code, @syu-pf-ssy, @plum7x, @HarshMakwana27 for your contributions! What do you think about Converter NOW? Rate the app! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/41.txt ================================================ - UI improvements - Improved french translation (credits: @Vsnmrn) - General improvements - Bugfix What do you think about Converter NOW? Rate the app! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/42.txt ================================================ - You can now block the internet access to the app - Improved the license page - Added haptic feedback - Improved the russian translation (@Atrafon) - Added catalan translation (@pereorga) - Bugfix and general improvements What do you think about Converter NOW? Rate the app! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/43.txt ================================================ 🎨 Custom Themes! Give your app a unique touch by choosing your favorite theme. ⚙️ Optimized Settings Page! Now even simpler and more intuitive to use. ✨ Graphic Enhancements! Small improvements that make the app even more beautiful and enjoyable to use. 🌍 Improved Translations! Thanks to @rehork and @pereorga, the Polish and Catalan versions are more accurate than ever. 🐞 Bug Fixes and General Improvements! We've worked hard to provide you with an even better experience. Do you like Converter NOW? Leave a review and let us know what you think! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/44.txt ================================================ 🎨 Graphic Enhancements for big screens 🌍 Updated translations Thanks to @rehork, @MajoranaOedipus, @FTno 🐞 Bug Fixes and General Improvements! We've worked hard to provide you with an even better experience Do you like Converter NOW? Leave a review and let us know what you think! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/45.txt ================================================ ⚙️ New features: use comma as decimal separator, paste number into the calculator, search with ctrl+K, Android 15 support and predictive back gesture 🌍 Updated translations Thanks to @solokot, @makishart, @HectorAE 🐞 Bug Fixes and General Improvements! We've worked hard to provide you with an even better experience Do you like Converter NOW? Leave a review and let us know what you think! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/46.txt ================================================ ⚙️ New features: hide unused units, improved default unit sorting, more units, improved conversion speed 🌍 Updated translations Thanks to @albanobattistella, @plum7x, @solokot 🐞 Bug Fixes and General Improvements! We've worked hard to provide you with an even better experience This version may break your unit sorting. We've made improvements so that this will no longer happen. We apologize for the inconvenience. Do you like Converter NOW? Leave a review and let us know what you think! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/47.txt ================================================ 🐞 Bug Fixes and General Improvements! We've worked hard to provide you with an even better experience Do you like Converter NOW? Leave a review and let us know what you think! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/48.txt ================================================ ⚙️ New units of measure: added density and new pressure units (Thanks to @Kheirlb) 🌍 Translation improvements: added Bengali and updated existing translations. Thanks to @aazmii, @rehork, @solokot 🐞 Bug fixes and general improvements! We've worked hard to give you an ever-better experience Do you like Converter NOW? Leave a review and let us know what you think! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/49.txt ================================================ 🐞 Fixed a bug related to the "Hide units" feature. Do you like Converter NOW? Leave a review and let us know what you think! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/50.txt ================================================ 🎨 New logo! Thanks to Giovanni Spitti for the suggestions 🖌️ Improved icons in the app 🌍 Translation improvements: thanks to @solokot 🐞 Bug fixes and general improvements! We've worked hard to give you an ever-better experience Do you like Converter NOW? Leave a review and let us know what you think! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/51.txt ================================================ 🔢 View the result preview in the calculator (Thanks to Bisher Asil) ⚙️ Choose the property to convert on startup (mobile only) 🌍 Translation improvements 🐞 Bug fixes and general improvements! We've worked hard to give you an even better experience Do you like Converter NOW? Leave a review and let us know what you think! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/changelogs/52.txt ================================================ 📤 Import / export / delete the app settings 🔢 Added several units of measurement 🌍 Translation improvements. Thanks to @rehork, @solokot, @milotype, @FTno 🐞 Bug fixes and general improvements! We've worked hard to give you an even better experience Do you like Converter NOW? Leave a review and let us know what you think! ✔️ ================================================ FILE: fastlane/metadata/android/en-US/full_description.txt ================================================ Converter NOW is an effective unit and currencies converter ✔️ Simplify the tedious conversion process between units of measurement in a few clicks. 🚀 It is fast and immediate: just start typing and immediately you have the real-time conversion with all the other units of measurement. 🖌️ It is customizable: the units can be reorganized according to your priorities and your use. 🔢 It integrates a calculator that let you do the calculations in every page. 💰 Currency exchange rates updated daily 🎨 Dynamic theming based on your device settings ⚫⚪ Choose your favorite theme: dark and white theme 📱🖥️ Multiplatform: available for Android, Web, Linux and Windows 💯 It is free, no ads, no data collection, no permissions (just Internet to update currency conversions) and first of all it is open source! Converter NOW can convert 200+ units of measurement and 30+ currencies 🎉🎉 Here are some of the physical quantity that Converter NOW is able to convert: -Currencies: Dollars, Euro, Pound, Rupee, Yen, etc -Length: meters, inches, miles, yards, light years, etc. -Area: square meters, hectares, acres, etc. -Volume: cubic meters, liters, gallons, pints, spoons, etc. -Time: seconds, hours, days, years, weeks, millennia, etc. -Temperature: Centigrade, Fahrenheit, Kelvin -Speed: meters per second, kilometers per hour, knots, etc. -Mass: grams, pounds, tons, atomic mass units, etc. -Force: Newton, dyne, pound-force, poundal, etc -Pressure: pascal, bar, atmosphere, psi, etc. -Energy: Joule, calories, kilowatt hours, etc. -Power: Watt, kilowatt, horsepower, etc. -Density:: grams per cubic meter, etc. -Fuel consumption: Miles per Gallon, Kilometers per liter, etc. -Numeral systems: decimal, binary, hexadecimal etc. -Torque: Newton meter, pound-force foot, poundal meter, etc. -Digital data: Nibble, bit, byte, kibibit, kibibyte, etc. -Shoes size: UK, India, Europe, USA, Japan, etc. -Angles: degree, radians, minutes, etc. -SI prefix: kilo, mega, giga, tera, milli, micro, nano, etc. ================================================ FILE: fastlane/metadata/android/en-US/short_description.txt ================================================ A simple, immediate and fast unit converter! ================================================ FILE: fastlane/metadata/android/en-US/title.txt ================================================ Converter NOW: Unit Converter ================================================ FILE: fastlane/metadata/android/es-ES/changelogs/33.txt ================================================ ★ Añadido atajo para aplicación ★ Pequeñas mejoras gráficas ★ Corrección de errores ¡Gracias a @the7thNightmare y 匿名 por sus contribuciones! ¿Qué piensa sobre Converter NOW? ¡Califíquenos! ✔️ ================================================ FILE: fastlane/metadata/android/es-ES/full_description.txt ================================================ Converter NOW es un efectivo conversor de unidades y divisas Simplifique el tedioso proceso de conversión entre unidades de medida con unos pocos clics. 🚀 Es rápido e inmediato: simplemente comience a tipear e inmediatamente tendrá una conversión en tiempo real con todas las unidades de medida. 🖌️ Es personalizable: puede reorganizar las unidades de acuerdo a sus prioridades y uso. 🔢 Contiene una calculadora integrada que le permite realizar operaciones en todas las páginas. 💰 El valor de las divisas es actualizado diariamente ⚫⚪ Escoja su tema favorito: oscuro o claro. 📱 Compatibilidad completa con teléfonos y tabletas. 💯 Es de código abierto Revise el repositorio en GitHub: https://github.com/ferraridamiano/ConverterNOW ¡También tenemos sitio web! 🖥️ Visite: https://ferraridamiano.github.io/ConverterNOW/ 🌍 Converter NOW es gratuito, sin publicidad, no requiere de ningún permiso y no recopila sus datos 🌍 Converter NOW puede convertir entre más de 200 unidades de medida y 30 divisas🎉🎉 Aquí hay algunas de las unidades que Converter NOW puede convertir: -Divisas: Dólar, Euro, Libra, Rupia, Yen, etc. -Longitud: Metros, pulgadas, millas, yardas, años luz, etc. -Área: Metros cuadrasdos, hectáreas, acres, etc. -Volumen: Metros cúbicos, litros, galones, pintas, cucharadas, etc. -Tiempo: Segundos, horas, días, semanas, años, milenios, etc. -Temperatura: Centígrado, Fahrenheit, Kelvin, y más. -Velocidad: Metros por segundo, kilómetros por hora, nudos, etc. -Masa: Gramos, libras, toneladas, unidades de masa atómica, etc. -Fuerza: Newton, dina, libra, poundal, etc. -Presión: Pascal, bar, atmósferas, psi, etc. -Energía: Julios, calorías, kilovatios-hora, etc. -Potencia: Vatios, kilovatios, caballos de potencia, etc. -Consumo de combustible: Millas por galón, kilómetros por litro, etc. -Sistemas de numeración: Decimal, binario, hexadecimal, etc. -Torsión mecánica: Newton-metro, pie-libra fuerza, poundal-metro, etc. -Sistema digital: Nibble, bit, byte, kibibit, kibibyte, etc. -Ángulos: grados, minutos, radianes, etc. -Prefijos del SI: kilo, mega, giga, tera, mili, micro, nano, etc. ================================================ FILE: fastlane/metadata/android/es-ES/short_description.txt ================================================ ¡Un conversor simple, inmediato y rápido! ================================================ FILE: fastlane/metadata/android/es-ES/title.txt ================================================ Conversor de unidades ================================================ FILE: fastlane/metadata/android/fr-FR/full_description.txt ================================================ Converter NOW est un convertisseur efficace d'unités et de devises ✔️ Simplifiez le fastidieux processus de conversion entre unités de mesure en quelques clics. 🚀 C'est rapide et immédiat : il suffit de commencer à taper et immédiatement vous avez la conversion en temps réel avec toutes les autres unités de mesure. 🖌️ Il est personnalisable : les unités peuvent être réorganisées en fonction de vos priorités et de votre utilisation. 🔢 Il intègre une calculatrice qui vous permet de faire les calculs dans chaque page. 💰 Les taux de change des devises sont mis à jour quotidiennement 🎨 Thème dynamique basé sur les paramètres de votre appareil ⚫⚪ Choisissez votre thème préféré : thème sombre et blanc 📱🖥️ Multiplateforme : disponible pour Android, Web, Linux et Windows 💯 C'est libre, sans publicités, sans collecte de données, sans autorisations (seulement l'accès à Internet pour mettre à jour la valeur des devises) et surtout c'est open-source ! Converter NOW peut convertir plus de 200 unités de mesure et plus de 30 devises 🎉🎉 Voici quelques-unes des quantités physiques que Converter NOW est capable de convertir: - Devises: Dollars, Euro, Livre, Roupie, Yen, etc. - Longueur: mètres, pouces, miles, yards, années-lumière, etc. - Superficie: mètres carrés, hectares, acres, etc. - Volume: mètres cubes, litres, gallons, pintes, cuillères, etc. - Heure: secondes, heures, jours, années, semaines, millénaires, etc. - Température: Centigrade, Fahrenheit, Kelvin - Vitesse: mètres par seconde, kilomètres par heure, nœuds, etc. - Masse: grammes, livres, tonnes, unités de masse atomique, etc. - Force: Newton, dyne, livre-force, poundal, etc. - Pression: pascal, bar, atmosphère, psi, etc. - Énergie: Joule, calories, kilowattheures, etc. - Puissance: Watt, kilowatt, puissance, etc. - Consommation de carburant: Miles par gallon, kilomètres par litre, etc. - Systèmes numériques: décimal, binaire, hexadécimal etc. - Couple: Newton mètre, pied livre-force, Poundal mètre, etc. - Données numériques: Nibble, bit, octet, kibibit, kibioctet, etc. - Pointure des chaussures: Royaume-Uni, Inde, Europe, USA, Japon, etc. - Angles: degrés, radians, minutes, etc. - Préfixe SI: kilo, méga, giga, téra, milli, micro, nano, etc. ================================================ FILE: fastlane/metadata/android/fr-FR/short_description.txt ================================================ Un convertisseur simple, immédiat et rapide! ================================================ FILE: fastlane/metadata/android/fr-FR/title.txt ================================================ Converter NOW : Convertisseur d'unités ================================================ FILE: fastlane/metadata/android/hr/full_description.txt ================================================ Converter NOW je učinkovit konverter mjernih jedinica i valuta ✔️ Pojednostavlja dosadan postupak izračunavanja mjernih jedinica kroz više koraka. 🚀 Računa brzo i trenutno: Jednostavno upiši vrijednost za željenu mjernu jedinicu. Sve ostale jedinice se izračunavaju trenutno. 🖌️ Prilagodljiv prikaz: redoslijed jedinica može se promijeniti prema vlastitim potrebama. 🔢 Integrira kalkulator koji omogućuje računanje na svakoj stranici. 💰 Dnevno aktualiziranje vrijednosti valuta. 🎨 Dinamičke teme na osnovi postavki tvog uređaja ⚫⚪ Odaberi svoju omiljenu temu: tamna i svijetla tema 📱🖥️ Višeplatformski: dostupno za Android, web, Linux i Windows 💯 Besplatan je, bez oglasa, bez prikupljanja podataka, bez dozvola (osim internetske veze za aktualiziranje tečaja valuta), a prije svega je otvorenog koda! Converter NOW pretvara više od 200 mjernih jedinica i više od 30 valuta 🎉🎉 Fizikalne veličine koje Converter NOW pretvara su: -Valute: dolar, euro, funta, rupija, jen itd -Duljina: metar, inč, milja, jard, svjetlosna godina itd. -Površina: kvadratni metar, hektar, aker itd. -Volumen: kubični metar, litra, galon, pinta, žlica itd. -Vrijeme: sekunda, sat, dan, godina, tjedan, tisućljeće itd. -Temperatura: Celzij, Fahreneit, Kelvin -Brzina: metara u sekundi, kilometara na sat, čvorovi itd. -Masa: gram, funta, tona, jedinica atomske mase itd. -Sila: Newton, din, funta-sila, poundal itd. -Pritisak: paskal, bar, atomfera, psi itd. -Energija: džul, kalorija, kilovat-sat itd. -Snaga: vat, kilovat, konjska snaga itd. -Gustoća: gram po kubičnom metru, itd. -Potrošnja goriva: milja na galon, kilometara na litru itd. -Numerički sustavi: decimalni, binarni, heksadecimalni itd. -Zakretni moment: njutn metar, funta-sila stopa, poundal metar itd. -Digitalni podaci: nibble, bit, bajt, kibibit, kibibajt itd. -Veličina obuće: UK, Indija, Europa, SAD, Japan itd. -Kutovi: stupanj, radijan, minute itd. -SI prefiksi: kilo, mega, giga, tera, milli, mikro, nano itd. ================================================ FILE: fastlane/metadata/android/hr/short_description.txt ================================================ Konverter mjernih jedinica ================================================ FILE: fastlane/metadata/android/hr/title.txt ================================================ Konverter mjernih jedinica ================================================ FILE: fastlane/metadata/android/id/full_description.txt ================================================ Converter NOW merupakan alat konversi satuan dan kurs yang efektif ✔️ Persingkat proses konversi yang berbelit antara satuan pengukuran dalam beberapa sentuhan. 🚀 Cepat dan segera: tinggal ketik dan Anda mendapatkan alat konversi waktu nyata dengan satuan pengukuran lainnya. 🖌️ Sangat bisa ditata ulang: satuannya bisa ditata tergantung penggunaan Anda. 🔢 Terdapat kalkulator yang terintegrasi yang bisa membuat Anda melakukan penghitungan. 💰 Kami memperbarui setiap hari valuasi kursnya. ⚫⚪ Pilih tema favorit Anda: tema gelap atau terang. 📱 Dukungan penuh untuk ponsel pintar and tablet. 💯 Sumber terbuka! Periksa repositorinya di GitHub: https://github.com/ferraridamiano/ConverterNOW Kami juga memiliki Aplikasi Web! 🖥️ Cek disini: https://ferraridamiano.github.io/ConverterNOW/ 🌍 Converter NOW gratis/bebas, tanpa iklan, dan tidak mengumpulkan data apapun 🌍 Converter NOW bisa mengonversi 200+ satuan pengukuran dan 30+ kurs 🎉🎉 Ini adalah beberapa satuan yang bisa Converter NOW konversikan: -Kurs: Dolar, Euro, Pound, Rupee, Yen, dll. -Panjang: meter, inci, mil, yard, tahun cahaya, dll. -Luas: meter persegi, hektar, ekar, dll. -Volume: meter kubik, liter, galon, pint, sendok, dll. -Waktu: detik, jam, hari, tahun, pekan, milenium, dll. -Temperatur: Centigrade, Fahreneit, Kelvin. -Kecepatan: meter per detik, kilometers per jam, knot, dll. -Massa: gram, pon, ton, satuan massa atom terpadu, dll. -Gaya: Newton, dyne, Pon-Gaya, pondal, dll. -Tekanan: Pascal, bar, atmosfer, psi, dll. -Eneri: Joule, kalori, kilowattjam, dll. -Daya: Watt, kilowatt, horse power, dll. -Konsumsi Bahan Bakar: mil per galon, kilometer per liter, dll. -Sistem Numeral: desimal, biner, heksadesimal dll. -Torsi: Newton meter, pon-gaya kaki, pondal meter, dll. -Data Digital: Nibel, bit, bita, kibibit, kibibita, dll. -Ukuran Sepatu: Britania, India, Eropa, AS, Jepang, dll. -Sudut: derajat, radian, menit, dll. -Awalan SI: kilo, mega, giga, tera, mili, mikro, nano, dll. ================================================ FILE: fastlane/metadata/android/id/short_description.txt ================================================ Sebuah alat konversi yang simpel, segera, dan cepat! ================================================ FILE: fastlane/metadata/android/id/title.txt ================================================ Alat Konversi Satuan dan Kurs ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/24.txt ================================================ ★ 3 nuove lingue: Portoghese, Francese e Norvegese 🌐 ★ Fix di traduzioni e della conversione della misura delle scarpe 🔧 Grazie a @mezysinc, @Torok42, @FTno, @dizietsma and Andreas 🔝 Abbiamo anche una web app, guarda qui: https://ferraridamiano.github.io/ConverterNOW/#/ Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/25.txt ================================================ ★ Fix della conversione delle basi numeriche 🔧 Abbiamo anche una web app, guarda qui: https://ferraridamiano.github.io/ConverterNOW/#/ Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/26.txt ================================================ ★ Aggiunta la possibilità di impostare il tema scuro anche nei dispositivi più datati ★ Il tema scuro si fa ancora più scuro con la modalità AMOLED ⚫ ★ Aggiunte alcune unità come: pennyweight, once troy, pietre, once fluide, gill, ecc. ★ Aggiunto il tedesco e il russo ★ Sistemati alcuni bug grafici Grazie a Gena Haltmair e a Vadim Young per i vostri contributi! Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/27.txt ================================================ ★ Fix lingua predefinita ★ Aggiunta la possibilità di impostare il tema scuro anche nei dispositivi più datati ★ Il tema scuro si fa ancora più scuro con la modalità AMOLED ⚫ ★ Aggiunte alcune unità come: pennyweight, once troy, pietre, once fluide, gill, ecc. ★ Aggiunto il tedesco e il russo ★ Sistemati alcuni bug grafici Grazie a Gena Haltmair e a Vadim Young per i vostri contributi! Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/28.txt ================================================ ★ Supporto tablet migliorato ★ Aggiunte alcune nuove unità ★ Miglioramenti alla calcolatrice ★ Bugfix aggiornamento del tasso di cambio ★ Aggiunto lo spagnolo Grazie a Gena Haltmair e n-berenice per i vostri contributi! Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/29.txt ================================================ ★ Migliorato il supporto per i grandi schermi ★ Supporto per l'icona adattiva ★ Aggiunte alcune unità di misura ★ Ora puoi cambiare la lingua dell'app dalle impostazioni ★ Aggiunta la splash screen ★ Aggiunto il croato ★ Bugfix Grazie a Giovanni Spitti e @milotype per i vostri contributi! Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/30.txt ================================================ ★ Fix bug icona con versione Android inferiore alla 8.0 ★ Migliorato il supporto per i grandi schermi ★ Supporto per l'icona adattiva ★ Aggiunte alcune unità di misura ★ Ora puoi cambiare la lingua dell'app dalle impostazioni ★ Aggiunta la splash screen ★ Aggiunto il croato ★ Bugfix Grazie a Giovanni Spitti e @milotype per i vostri contributi! Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/31.txt ================================================ ★ Migliorata l'accessibilità dell'app ★ Aggiunta la possibilità di annullare l'operazione "Cancella tutto" ★ Aggiunto il giapponese e l'indonesiano ★ Aggiunte altre unità di misura ★ Migliorate le performance ★ Bugfix Grazie a @the7thNightmare e 匿名 per i vostri contributi! Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/33.txt ================================================ ★ Aggiunte le app shortcut ★ Piccoli miglioramenti grafici ★ Bugfix Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/34.txt ================================================ ★ Ora l'app è disponibile in lingua araba ★ Piccoli ritocchi ★ Bugfix Grazie a Omer Maki per il suo contributo! Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/35.txt ================================================ ★ Calcolatrice migliorata ★ Bugfix Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/36.txt ================================================ ★ Migrazione a Material you ★ Themed icons su Android 13+ ★ Aggiunta traduzione in Polacco ★ Bugfixes Grazie a @rehork per il tuo contributo! Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/37.txt ================================================ ★ I simboli delle unità sono ora sempre visibili Grazie a @rehork per il tuo suggerimento! Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/38.txt ================================================ Ecco Converter NOW 4.0! - Terminata la migrazione al Material You - Supporto al tema dinamico su Android 12+ - Nuove icone - Aggiornamenti alle traduzioni - Bug fix e miglioramenti generali Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/39.txt ================================================ - Fixed bug with currencies page after v4 update - Improved polish translations Cosa ne pensi di Converter NOW? Recensiscici! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/40.txt ================================================ - Miglioramenti legati al material you - La calcolatrice ha un nuovo look - Aggiunte le seguenti traduzioni: olandese, cinese e cinese tradizionale - Miglioramenti alle altre traduzioni - Miglioramenti alle icone - Miglioramenti generali Grazie a @David-git-code, @syu-pf-ssy, @plum7x, @HarshMakwana27 per i vostri contributi! Cosa ne pensi di Converter NOW? Se ti va, recensisci l'app! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/41.txt ================================================ - Piccoli miglioramenti all'interfaccia utente - Migliorata la traduzione francese (crediti: @Vsnmrn) - Miglioramenti generali - Bugfix Cosa ne pensi di Converter NOW? Se ti va, recensisci l'app! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/42.txt ================================================ - Aggiunta la possibilità di bloccare l'accesso ad internet - Migliorata la pagina delle licenze - Aggiunto il feedback della vibrazione - Migliorata la traduzione in russo (@Atrafon) - Aggiunta la traduzione in catalano (@pereorga) - Bugfix e miglioramenti generali Cosa ne pensi di Converter NOW? Se ti va, recensisci l'app! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/43.txt ================================================ 🎨 Temi personalizzati! Dai un tocco unico all'app scegliendo il tema che più ti piace. ⚙️ Pagina delle impostazioni ottimizzata! Ora è ancora più semplice e intuitiva da usare. ✨ Ritocchi grafici! Piccoli miglioramenti che rendono l'app ancora più bella e piacevole da utilizzare. 🌍 Traduzioni migliorate! Grazie a @rehork e @pereorga, le versioni in Polacco e Catalano sono più accurate che mai. 🐞 Correzioni di bug e miglioramenti generali! Abbiamo lavorato sodo per offrirti un'esperienza sempre migliore. Ti piace Converter NOW? Lascia una recensione e facci sapere cosa ne pensi! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/44.txt ================================================ 🎨 Ritocchi grafici per i grandi schermi 🌍 Traduzioni aggiornate Grazie a @rehork, @MajoranaOedipus, @FTno 🐞 Correzioni di bug e miglioramenti generali! Abbiamo lavorato sodo per offrirti un'esperienza sempre migliore Ti piace Converter NOW? Lascia una recensione e facci sapere cosa ne pensi! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/45.txt ================================================ ⚙️ Nuove funzioni: usa la virgola come separatore decimale, incolla i numeri nella calcolatrice, cerca con ctrl+K, supporto ad android 15 e predictive back gesture 🌍 Traduzioni aggiornate Grazie a @solokot, @makishart, @HectorAE 🐞 Correzioni di bug e miglioramenti generali! Abbiamo lavorato sodo per offrirti un'esperienza sempre migliore Ti piace Converter NOW? Lascia una recensione e facci sapere cosa ne pensi! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/46.txt ================================================ ⚙️ Nuove funzioni: nascondi le unità di misura che non usi, ordinamento delle unità di default migliorato, più unità di misura, velocità di conversione migliorata 🌍 Traduzioni aggiornate Grazie a @albanobattistella, @plum7x, @solokot 🐞 Correzioni di bug e miglioramenti generali! Abbiamo lavorato sodo per offrirti un'esperienza sempre migliore Questa versione potrebbe rompere il tuo ordinamento delle unità di misura. Abbiamo fatto dei miglioramenti in modo che questo non accada più. Ci scusiamo per il diseguido. Ti piace Converter NOW? Lascia una recensione e facci sapere cosa ne pensi! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/47.txt ================================================ 🐞 Correzioni di bug e miglioramenti generali! Abbiamo lavorato sodo per offrirti un'esperienza sempre migliore Ti piace Converter NOW? Lascia una recensione e facci sapere cosa ne pensi! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/48.txt ================================================ ⚙️ Nuove unità di misura: aggiunta la densità e nuove unità di pressione (Grazie a @Kheirlb) 🌍 Miglioramento alle traduzioni: aggiunto bengalese e aggiornato le traduzioni esistenti. Grazie a @aazmii, @rehork, @solokot 🐞 Correzioni di bug e miglioramenti generali! Abbiamo lavorato sodo per offrirti un'esperienza sempre migliore Ti piace Converter NOW? Lascia una recensione e facci sapere cosa ne pensi! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/49.txt ================================================ 🐞 Correzione di un bug relativo alla funzione "Nascondi unità" Ti piace Converter NOW? Lascia una recensione e facci sapere cosa ne pensi! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/50.txt ================================================ 🎨 Nuovo logo! Gazie a Giovanni Spitti per i suggerimenti 🖌️ Migliorate le icone dentro l'app 🌍 Miglioramenti alle traduzioni: grazie a @solokot 🐞 Correzioni di bug e miglioramenti generali! Abbiamo lavorato sodo per offrirti un'esperienza sempre migliore Ti piace Converter NOW? Lascia una recensione e facci sapere cosa ne pensi! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/51.txt ================================================ 🔢 Visualizza l'anteprima del risultato nella calcolatrice (Grazie a Bisher Asil) ⚙️ Scegli la proprietà da convertire all'avvio (solo su mobile) 🌍 Miglioramenti alle traduzioni 🐞 Correzioni di bug e miglioramenti generali! Abbiamo lavorato sodo per offrirti un'esperienza sempre migliore Ti piace Converter NOW? Lascia una recensione e facci sapere cosa ne pensi! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/changelogs/52.txt ================================================ 📤 Importa / esporta / elimina le impostazioni dell'app 🔢 Aggiunte molte unità di misura 🌍 Miglioramenti alle traduzioni. Grazie a @rehork, @solokot, @milotype, @FTno 🐞 Correzioni di bug e miglioramenti generali! Abbiamo lavorato sodo per offrirti un'esperienza sempre migliore Ti piace Converter NOW? Lascia una recensione e facci sapere cosa ne pensi! ✔️ ================================================ FILE: fastlane/metadata/android/it-IT/full_description.txt ================================================ Converter NOW è un efficace convertitore di unità di misura e di valute made in Italy ✔️ Semplifica il noioso processo di conversione tra unità di misura in qualche click 🚀 È veloce e immediato: basta cominciare a digitare e subito si ha la conversione in real-time con tutte le altre unità di misura 🖌️ È personalizzabile: le unità di misura posso essere riorganizzate secondo le proprie priorità e il proprio utilizzo 🔢 Ha una calcolatrice integrata che permette di eseguire velocemente i calcoli in ogni schermata 💰 Tassi di cambio aggiornati quotidianamente 🎨 Tema dinamico in base alle impostazioni del tuo dispositivo ⚫⚪ Scegli il tuo tema preferito: tema chiaro e scuro 📱🖥️ Multipiattaforma: disponibile per Android, Web, Linux e Windows 💯 È gratuito, senza annuni, non raccogliamo dati, non richiede permessi (solo internet per aggiornare i tassi di cambio) e soprattutto è open source! Converter NOW può convertire 200+ unità di misura e 30+ valute 🎉🎉 Ecco alcune delle grandezze che Converter NOW è in grado di convertire: -Valute: Dollari, Euro, Sterline, Rupie, Yen, ecc -Lunghezza: metri, pollici, miglia, yard, anni luce, ecc -Superficie: metri quadrati, ettari, acri, ecc -Volume: metri cubi, litri, galloni, pinte, cucchiai, ecc -Tempo: secondi, ore, giorni, anni, settimane, millenni, ecc -Temperatura: Centigradi, Fahreneit, Kelvin -Velocità: metri al secondo, chilometri orari, nodi, ecc -Massa: grammi, libbre, tonnellate, unità di massa atomica, ecc -Forza: Newton, dyne, libbre-forza, poundal, ecc -Pressione: pascal, bar, atomsfere, psi, ecc -Energia: Joule, calorie, kilowattora, ecc -Potenza: Watt, kilowatt, cavalli vapore, ecc -Densità:: Grammi per metro cubo, ecc. -Consumo carburante: Chilometri con un litro, litro ogni 100 km, ecc -Basi numeriche: decimale, binario, esadecimale, ecc. -Momento: Newton metro, libbre-forza piede, poundal metro, ecc -Dati digitali: Nibble, bit, byte, kibibit, kibibyte, ecc -Taglia scarpe: UK, India, Europa, USA, Giappone, ecc -Angoli: gradi, primi, radianti, ecc -Prefissi SI: kilo, mega, giga,tera, milli, micro, nano, ecc ================================================ FILE: fastlane/metadata/android/it-IT/short_description.txt ================================================ Un convertitore di unità semplice, immediato e veloce! ================================================ FILE: fastlane/metadata/android/it-IT/title.txt ================================================ Convertitore di Unità ================================================ FILE: fastlane/metadata/android/ja/full_description.txt ================================================ Converter NOW は効果的な単位と通貨の変換器です ✔️ 簡単 数回クリックするだけで、測定単位間の面倒な変換プロセスが完了します。 🚀 迅速かつ即時: 入力を開始するだけで、すぐに他のすべての測定単位をリアルタイムで変換できます。 🖌️ カスタマイズ可能: ユニットは、優先順位と用途に応じて再編成できます。 🔢 電卓が統合されており、すべてのページで計算を行うことができます。 💰 毎日更新する為替レート 🎨 デバイス設定に基づく動的テーマ ⚫⚪ 好きなテーマをお選ぶ: ダークとホワイトのテーマ 📱🖥️ マルチプラットフォーム: Android、Web、Linux、および Windows で利用可能 💯 無料、広告なし、データ収集なし、権限なし (為替レートを更新するためのインターネット権限のみ) そしてオープンソース! Converter NOW は、200 以上の測定単位と 30 以上の通貨を変換できます 🎉🎉 Converter NOW が変換できる単位の一部: -通貨: ドル、ユーロ、ポンド、ルピー、円 など -長さ: メートル、インチ、マイル、ヤード、光年 など -面積: 平方メートル、ヘクタール、エーカー など -体積: 立方メートル、リットル、ガロン、パイント、スプーン など -時間: 秒、時間、日、年、週、ミレニアム など -温度: 摂氏、華氏、ケルビン -速度: メートル毎秒、キロメートル毎時、ノット など -質量: グラム、ポンド、トン、原子質量単位 など -力: ニュートン、ダイン、重量ポンド、パウンダル など -圧力: パスカル、バール、気圧、重量ポンド毎平方インチ など -エネルギー: ジュール、カロリー、キロワット時 など -仕事率: ワット、キロワット、馬力 など -燃費: マイル毎ガロン、キロメートル毎リットル など -記数法: 10進数、2進数、16進数 など -トルク: ニュートンメートル、フィート重量ポンド、パウンダルメートル など -デジタルデータ: ニブル、ビット、バイト、キビビット、キビバイト など -靴のサイズ: イギリス、中国、インド、ヨーロッパ、アメリカ、日本 など -角度: 度、ラジアン、分 など -国際単位変換: キロ、メガ、ギガ、テラ、ミリ、マイクロ、ナノ など ================================================ FILE: fastlane/metadata/android/ja/short_description.txt ================================================ シンプルで即時かつ高速な単位換算アプリ! ================================================ FILE: fastlane/metadata/android/ja/title.txt ================================================ Converter NOW: 単位換算 ================================================ FILE: fastlane/metadata/android/nl_NL/full_description.txt ================================================ Converter NOW is een effectieve eenheden en valuta omrekentool ✔️ Versimpel het omslachtige omrekenen van meeteenheden met een simpele klik op de knop! 🚀 It is Snel en onmiddelijk: begin gewoon met typen en je ziet direct de omgerekende eenheden verschijnen. 🖌️ It is Aanpasbaar: Eenheden kunnen anders worden geordend aan de hand van wat je vaak gebruikt. 🔢 Integreert een rekenmachine die je bij iedere meeteenheid helpt rekenen. 💰 Valuta wisselkoersen worden dagelijks geüpdate 🎨 Dynamische thema's gebaseerd op jouw apparaatinstellingen ⚫⚪ Kies je favoriete thema: donker en licht thema 📱🖥️ Multiplatform: beschikbaar op Android, Web, Linux and Windows 💯 Het is gratis, geen advertenties, geen dataverzameling, geen toestemmingen nodig (alleen internet om de wisselkoersen te updaten) en bovendien open source! Converter NOW kan 200+ meeteenheden en 30+ valuta omrekenen 🎉🎉 Dit zijn een aantal van de meeteenheden die Converter NOW kan omrekenen: -Valuta: Dollars, Euro, Pound, Rupee, Yen, etc. -Lengte: meters, inches, mijlen, yards, lichtjaren, etc. -Oppervlakte: vierkante meters, hectares, acres, etc. -Inhoud: kubieke meters, liters, gallons, pints, eetlepels, etc. -Tijd: seconden, uren, dagen, jaren, weken, millennia, etc. -Temperatuur: Celsius, Fahreneit, Kelvin -Snelheid: meters per seconde, kilometers per uur, knopen, etc. -Massa: grammen, pond, tonnen, atomic mass units, etc. -Kracht: Newton, dyne, pound-kracht, poundal, etc -Druk: pascal, bar, atomsfere, psi, etc. -Energie: Joule, calorieën, kilowattuur, etc. -Vermogen: Watt, kilowatt, paardenkracht, etc. -Brandstofverbruik: Mijl per Gallon, Kilometer per liter, etc. -Numerieke systemen: decimaal, binair, hexadecimaal, etc. -Koppel: Newtonmeter, pound-kracht voet, poundal meter, etc. -Digitale data: Nibble, bit, byte, kibibit, kibibyte, etc. -Schoenmaten: UK, India, Europe, USA, Japan, etc. -Hoeken: graden, straal, minuten, etc. -Decimale voorvoegsels: kilo, mega, giga, tera, milli, micro, nano, etc. ================================================ FILE: fastlane/metadata/android/nl_NL/short_description.txt ================================================ Een simpele, onmiddelijke en snelle manier om eenheden om te rekenen! ================================================ FILE: fastlane/metadata/android/nl_NL/title.txt ================================================ Converter NOW ================================================ FILE: fastlane/metadata/android/pl-PL/full_description.txt ================================================ Converter NOW to skuteczny przelicznik jednostek miar i walut ✔️ Uprość żmudny proces konwersji jednostek miar za pomocą kilku kliknięć.. 🚀 Jest szybki i natychmiastowy: po prostu zacznij pisać i natychmiast masz konwersję w czasie rzeczywistym na wszystkie inne jednostki miary. 🖌️ Jest konfigurowalny: jednostki mogą być reorganizowane zgodnie z twoimi upodobaniami i sposobem użytkowania. 🔢 Zawiera w sobie kalkulator, który pozwala wykonywać obliczenia na każdej stronie. 💰 Codziennie aktualizowane kursy walut ⚫⚪ b>Wybierz swój ulubiony motyw: ciemny lub jasny 📱 Pełne wsparcie dla tabletów i telefonów komórkowych 💯 Z otwartym kodem źródłowym! Sprawdź repozytorium na GitHubie: https://github.com/ferraridamiano/ConverterNOW Mamy również aplikację internetową! 🖥️ Sprawdź tutaj: https://ferraridamiano.github.io/ConverterNOW/ 🌍 Converter NOW jest darmowy, bez reklam, nie wymaga żadnej zgody i nie zbiera twoich danych 🌍 Converter NOW może przeliczać ponad 200 jednostek miary i ponad 30 walut. 🎉🎉 Oto przykładowe miary, które Converter NOW jest w stanie przeliczyć: -Waluty: dolary, euro, funty, rupie, jeny, itd. -Długość: metry, cale, mile, jardy, lata świetlne, itd. -Powierzchnia: metry kwadratowe, hektary, akry, itd. -Pojemność: metry sześcienne, litry, galony, pinty, łyżki, itd. -Czas: sekundy, godziny, dni, lata, tygodnie, tysiąclecia, itd. -Temperatura: stopnie Celsjusza, stopnie Fahrenheita, stopnie Kelwina, itd. -Prędkość: metry na sekundę, kilometry na godzinę, węzły, itd. -Masa: gramy, funty, tony, jednostki masy atomowej, itd. -Siła: niutony, dyny, funt-siła, itd. -Ciśnienie: paskal, bar, atmosfery, psi, itd. -Energia: dżule, kalorie, kilowatogodziny, itd. -Moc: wat, kilowat, konie mechaniczne, itd. -Zużycie paliwa: Mile z galona, kilometry z litra, itd. -Systemy liczbowe: dziesiętny, dwójkowy, szesnastkowy, itd. -Moment obrotowy: Niutonometr, funt na stopę, poundalometr, itd. -Dane cyfrowe: półjbajt, bit, bajt, kibibit, kibibajt, itd. -Rozmiar buta: angielski, indyjski, europejski, amerykański, japoński, itd. -Kąty: stopień, radian, minuta, itd. -Przedrostek SI: kilo, mega, giga, tera, mili, mikro, nano, itd. ================================================ FILE: fastlane/metadata/android/pl-PL/short_description.txt ================================================ Prosty i szybki przelicznik jednostek miar! ================================================ FILE: fastlane/metadata/android/pl-PL/title.txt ================================================ Przelicznik jednostek miar ================================================ FILE: fastlane/metadata/android/pt-BR/full_description.txt ================================================ Converter NOW é um conversor eficaz de unidades e moedas ✔️ Simplificar em poucos cliques o tedioso processo de conversão entre unidades de medida. 🚀 É rápido e imediato: basta começar a digitar e em tempo real você obtém a conversão com todas as outras unidades de medida. 🖌️ Facilmente personalizável: as unidades podem ser reorganizadas de acordo com suas prioridades e seu uso. 🔢 Integra uma calculadora que permite fazer os cálculos em cada seção. 💰 Nós atualizamos diariamente o valor das moedas. ⚫⚪ Escolha seu tema preferido: tema escuro ou branco. 📱 O suporte completo para Smartphone e Tablet. 💯 É de código livre! Visite nosso repositório no GitHub: https://github.com/ferraridamiano/ConverterNOW Também temos uma versão na Web! 🖥️ Confira aqui: https://ferraridamiano.github.io/ConverterNOW/ 🌍 Converter NOW é gratuito, sem anúncios, não requer nenhuma permissão e não coleta nenhum de seus dados 🌍 Converter NOW pode converter mais de 200 unidades de medida e mais de 30 moedas 🎉🎉 Aqui estão algumas unidades que o Converter NOW é capaz de converter: -Moedas: Dólar, Euro, Libra, Rúpia, Iene, etc -Comprimento: metros, polegadas, milhas, jardas, anos-luz, etc. -Área: metros quadrados, hectares, acres, etc. -Volume: metros cúbicos, litros, galões, pintas, colheres, etc. -Tempo: segundos, horas, dias, anos, semanas, milénios, etc. -Temperatura: centígrado, Fahreneit, Kelvin -Velocidade: metros por segundo, kilómetros por hora, nós, etc. -Massa: gramas, libras, toneladas, unidade de massa atômica, etc. -Força: Newton, dyne, libra-força, poundal, etc -Pressão: pascal, bar, atomsfere, psi, etc. -Energia: Joule, calorias, quilovate-hora, etc. -Potência: Watt, kilowatt, cavalo-vapor, etc. -Consumo de combustível: Milhas por galão, Kilómetros por litro, etc. -Sistemas numéricos: decimal, binário, hexadecimal etc. -Torque: Newton metro, libra-força pé, poundal metro, etc. -Dados digitais: Nibble, bit, byte, kibibit, kibibyte, etc. -Medida em Pés: UK, Índia, Europa, EUA, Japão, etc. -Ângulos: grau, radianos, minutos, etc. -Prefixo SI: kilo, mega, giga, tera, milli, micro, nano, etc. ================================================ FILE: fastlane/metadata/android/pt-BR/short_description.txt ================================================ Um conversor simples, imediato e rápido! ================================================ FILE: fastlane/metadata/android/ru-RU/full_description.txt ================================================ Converter NOW - это эффективный конвертер единиц измерения и валют ✔️ Простой: скучный процесс по переводу единиц измерения теперь доступен в пару нажатий. 🚀 Это очень быстро: просто начните печать, и в миг вы получите результат во всех единицах измерения. 🖌️ Это очень гибкая программа: единицы измерения могут быть определены вашими предпочтениями и сферами деятельности. 🔢 Есть встроенный удобный калькулятор, который поможет в ваших расчётах. 💰 Соотношение курса валют обновляется ежедневно. 🎨 Внешний вид подстраивающийся под настройки вашего устройства. ⚫⚪ Выберите между вашей любимой темой: тёмной или светлой. 📱🖥️ Кроссплатформенность: доступно на Android, в браузере, на системах Linux и Windows. 💯 Это абсолютно бесплатно, без рекламы, подозрительных разрешений и сбора личных данных (за исключением доступа в интернет для обновления курса валют), и конечно же - мы проект с открытым исходным кодом! Converter NOW может конвертировать больше 200 единиц измерения и более 30 валют 🎉🎉 Вот пример некоторых величин, которые Converter NOW может конвертировать: -Валюты: доллары, евро, фунты, рупии, иены и др. -Длина: метры, дюймы, мили, ярды, световые годы и др. -Площадь: квадратные метры, гектары, сотки и др. -Объем: кубические метры, литры, галлоны, пинты, ложки и др. -Время: секунды, часы, дни, годы, недели, тысячелетия и др. -Температура: градусы Цельсия, Фаренгейта, Кельвина -Скорость: метры в секунду, километры в час, узлы и др. -Масса: граммы, фунты, тонны, атомные единицы массы и др. -Сила: ньютоны, дины, фунт-силы, фунталы и др. -Давление: паскали, бары, атмосферы, пси и др. -Энергия: джоули, калории, киловатт-часы и др. -Мощность: ватт, киловатты, лошадиные силы и др. -Плотность: граммы на кубический метр и др. -Потребление топлива: мили на галлон, километры на литр и др. -Системы счисления: десятичная, двоичная, шестнадцатеричная и др. -Крутящий момент: ньютон-метры, фунт-сила-футы, фунт-метры и др. -Цифровые данные: полубайты, биты, байты, кибибиты, кибибайты и др. -Размер обуви: Великобритания, Индия, Европа, США, Япония и др. -Углы: градусы, радианы, минуты и др. -Префиксы Международной системы единиц (СИ): кило, мега, гига, тера, милли, микро, нано и др. ================================================ FILE: fastlane/metadata/android/ru-RU/short_description.txt ================================================ Простой, эффективный и быстрый конвертер! ================================================ FILE: fastlane/metadata/android/ru-RU/title.txt ================================================ Converter NOW: преобразование единиц ================================================ FILE: fastlane/metadata/android/tr-TR/changelogs/24.txt ================================================ ★ 3 Yeni Dil: Portekizce, Fransızca ve Norveççe 🌐 ★ Bazı çeviriler ve ayakkabı boyutu dönüşüm hatası düzeltildi 🔧 @Mezysinc, @Torok42, @FTno, @dizietsma ve Andreas'a teşekkürler 🔝 Ayrıca bir web uygulamamız da var! 🖥️ Şurada: https://ferraridamiano.github.io/converternow/ Converter NOW hakkında ne düşünüyorsun? Bizi değerlendir! ✔️ ================================================ FILE: fastlane/metadata/android/tr-TR/changelogs/25.txt ================================================ ★ Sayısal sistem dönüşümleri hatası düzeltildi 🔧 Ayrıca bir web uygulamamız da var! 🖥️ Şurada: https://ferraridamiano.github.io/converternow/ Converter NOW hakkında ne düşünüyorsun? Bizi değerlendir! ✔️ ================================================ FILE: fastlane/metadata/android/tr-TR/changelogs/26.txt ================================================ ★ Eski cihazlarda karanlık tema seçilebilir ★ Koyu tema, AMOLED moduyla daha da koyu ⚫ ★ Pennyweights, ons, taş, sıvı onsu, gill vb. bazı birimler eklendi. ★ Almanca ve Rusça çeviriler eklendi ★ Hata düzeltmesi yapıldı Katkılarınız için Gena Haltmair ve Vadim Young'a teşekkürler! Converter NOW hakkında ne düşünüyorsun? Bizi değerlendir! ✔️ ================================================ FILE: fastlane/metadata/android/tr-TR/changelogs/27.txt ================================================ ★ Varsayılan dil düzeltmesi ★ Eski cihazlarda karanlık tema seçilebilir ★ Koyu tema, AMOLED moduyla daha da koyu ⚫ ★ Pennyweights, ons, taş, sıvı onsu, gill vb. bazı birimler eklendi. ★ Almanca ve Rusça çeviriler eklendi ★ Hata düzeltmesi yapıldı Katkılarınız için Gena Haltmair ve Vadim Young'a teşekkürler! Converter NOW hakkında ne düşünüyorsun? Bizi değerlendir! ✔️ ================================================ FILE: fastlane/metadata/android/tr-TR/full_description.txt ================================================ Converter NOW efektif ölçü ve para birimleri dönüştürücüdür ✔️ Birkaç tıklamayla ölçü birimleri arasındaki sıkıcı dönüşüm işlemlerini basitleştirir. 🚀 Hızlı ve efektif: Yazmaya başladığınız anda diğer tüm ölçü birimlerine gerçek zamanlı çevirim. 🖌️ Özelleştirilebilir: Birimler önceliklerinize ve kullanımınıza göre yeniden düzenlenebilir. 🔢 Her sayfada hesaplama yapmanız için kullanabileceğiniz bir hesap makinesi içerir. 💰 Günlük, para birimi değerleri güncellemeleri ⚫⚪ Tema seçeneği: Koyu ve açık tema seçeneği 📱 Akıllı telefon ve tabletlere tam uyumlu 💯 Özgür yazılım! Github depomuz: https://github.com/ferraridamiano/converternow Ayrıca bir web uygulamamız da var! 🖥️ Şurada: https://ferraridamiano.github.io/converternow/ 🌍 Converter NOW ücretsiz, reklamsız, herhangi bir izin gerektirmez ve veri toplamaz 🌍 Converter NOW 200+ ölçü birimini ve 30+ para birimini dönüştürebilir 🎉🎉 Converter NOW uygulamasının şimdi dönüştürebileceği fiziksel niceliklerden bazıları: -Para Birimleri: Dolar, Euro, Pound, Rupi, Yen, vb. -Uzunluk: metre, inç, mil, yarda, ışık yılı vb. -Alan: metrekare, hektar, dönüm vb. -Hacim: metreküp, litre, galon, pint, kaşık vb. -Zaman: saniye, saat, gün, yıl, hafta, millennia, vb. -Sıcaklık: Santigrat, Fahreneit, Kelvin -Hız: saniyede metre, saatte kilometre, knot vb. -Kütle: gram, pound, ton, dalton vb. -Kuvvet: Newton, dyn, pound-kuvvet, poundal, vb -Basınç: pascal, bar, atmosfer, PSI, vb. -Enerji: Joule, kalori, kilovat saat, vb. -Güç: Watt, kilowatt, beygir gücü, vb. -Yakıt tüketimi: Galon başına mil, Litre başına kilometre vb. -Sayısal sistemler: ondalık, ikili, onaltılık vb. -Tork: Newton metre, pound-kuvvet fit, poundal metre, vb. -Dijital veriler: Nibble, bit, bayt, kibibit, kibibyte vb. -Ayakkabı ölçüleri: İngiltere, Hindistan, Avrupa, ABD, Japonya, vb. -Açılar: derece, radyan, dakika vb. -SI öneki: kilo, mega, giga, tera, milli, mikro, nano, vb. ================================================ FILE: fastlane/metadata/android/tr-TR/short_description.txt ================================================ Basit ve efektif bir dönüştürücü! ================================================ FILE: fastlane/metadata/android/zh/full_description.txt ================================================ Converter NOW 是一个有力的单位和货币转换器 ✔️ 简单 只需点击几下、即可完成计量单位之间繁琐的转换过程。 🚀 它快速且易于使用:只需开始输入、即可立即与所有其他测量单位进行实时转换。 🖌️ 它是可定制的:可以根据您的优先级和用途进行重新组织单元。 🔢 它集成了一个计算器、可让您在每个页面中进行计算。 💰 每日更新货币汇率 🎨 动态主题 基于您的设备配置 ⚫⚪ 选择你最喜欢的主题:黑暗或明亮主题 📱🖥️ 多平台:适用于 Android、Web、Linux 和 Windows 💯 它免费、无广告、无数据收集、无权限(只需互联网用来更新货币汇率)。首先它是开源的! Converter NOW 可以转换 200 以上的计量单位和 30 种以上的货币 🎉🎉 以下是 Converter NOW 能够转换的一些单位: -货币:美元、欧元、英镑、卢比、日元 等。 -长度:米、英寸、英里、码、光年 等。 -面积:平方米、公顷、英亩 等。 -体积:立方米、升、加仑、品脱、汤匙 等。 -时间:秒、小时、天、年、周、千年 等。 -温度:摄氏、华氏、开氏 -速度: 米/秒、公里/小时、节 等。 -质量:克、磅、吨、原子质量单位 等。 -:牛顿、达因、磅力、磅力 等。 -压力:帕斯卡、巴、原子力、psi 等。 -能量:焦耳、卡路里、千瓦时 等。 -功率:瓦特、千瓦、马力 等。 -油耗:每加仑英里数、每升公里数 等。 -数制:十进制、二进制、十六进制 等。 -力矩:牛顿米、磅力英尺、磅米 等。 -数据:比特、字节、kibibit、kibibyte、半字节 等。 -鞋码:英国、印度、欧洲、美国、日本 等。 -角度:度、弧度、分 等。 -SI 前缀:千、兆、千兆、万亿、毫、微米、纳米 等。 ================================================ FILE: fastlane/metadata/android/zh/short_description.txt ================================================ 一个简单、即时和快速的单位转换器! ================================================ FILE: fastlane/metadata/android/zh/title.txt ================================================ Converter NOW: 单位转换器 ================================================ FILE: fastlane/metadata/android/zh-TW/full_description.txt ================================================ Converter NOW 是一個有力的單位與貨幣轉換器 ✔️ 簡化繁瑣的的轉換處裡,只需點擊幾次即可轉換以下計量的單位。 🚀 快速且即時:只需開始輸入,您就會立即獲得所有計量單位即時轉換。 🖌️ 可自訂:這些單位可以依照您的優先順序與用途進行重新安排。 🔢 整合一個計算器以讓您在任何頁面都可以進行計算。 💰 貨幣匯率將會每日更新 🎨 動態主題基於您的裝置設定 ⚫⚪ 選擇您最愛的主題:深色、白色主題 📱🖥️ 多平台:可用於 Android、網頁、Linux 與 Windows 💯 它是免費、無廣告、不蒐集資料、無須權限 (就只需網際網路以更新貨幣匯率) 的,首先它是開放原始碼的! Converter NOW 可以轉換 200 種以上計量單位與 30 種以上貨幣 🎉🎉 Converter NOW 允許轉換以下這些物理量: -貨幣:美金、歐元、英鎊、盧比、元…等。 -長度:公尺、英吋、英里、碼、光年…等。 -面積:平方公尺、公頃、英畝…等。 -體積:立方公尺、公升、加侖、品脫、湯匙…等。 -時間:秒、時、日、週、年,千年紀…等。 -溫度:攝氏、華氏、克式…等。 -速度:米每秒、公里每小時、節…等。 -質量:公克、磅、噸、原子質量單位…等。 -力:牛頓、達因、磅力、磅達…等。 -壓力:百帕、巴、原子力、磅每平方英吋…等。 -能量:焦耳、卡路里、千瓦小時…等。 -功率:瓦特、千瓦、馬力…等。 -燃油消耗:英里每加侖、公里每加侖…等。 -數值系統:十進位制、二進位制、十六進位制 etc. -力矩:牛頓公尺、磅力英尺、磅達公尺…等。 -數位資料:半位元組、位元、位元組、Kibibit、KibiByte…等。 -鞋碼:英國、印度、歐洲、美國、日本…等。 -角度:度、弧度、分…等。 -國際單位制前置詞:千、百萬、吉、兆、毫、微、奈…等。 ================================================ FILE: fastlane/metadata/android/zh-TW/short_description.txt ================================================ 一個簡單、即時且快速的單位轉換器! ================================================ FILE: firebase.json ================================================ { "hosting": { "public": "build/web", "ignore": [ "firebase.json", "**/.*", "**/node_modules/**" ], "rewrites": [ { "source": "**", "destination": "/index.html" } ] } } ================================================ FILE: integration_test/large_display_test.dart ================================================ import 'package:flutter_test/flutter_test.dart'; import 'package:integration_test/integration_test.dart'; import 'package:flutter/material.dart'; import 'package:converterpro/main.dart' as app; import 'utils.dart'; void main() { IntegrationTestWidgetsFlutterBinding.ensureInitialized(); Future testInit(WidgetTester tester, {bool clearPrefs = true}) async { if (clearPrefs) { await clearPreferences(); } app.main(); await tester.pumpAndSettle(); setWindowSize(800, 700); await tester.pumpAndSettle(); } group('Common conversions tasks:', () { testWidgets('Change to a new property and perform conversion', ( WidgetTester tester, ) async { await testInit(tester); await tester.tap( find.byKey(const ValueKey('drawerItem_PROPERTYX.currencies')), ); await tester.pumpAndSettle(); expect( find.text('Currencies'), findsAtLeastNWidgets(2), reason: 'Expected the currencies page', ); await tester.tap(find.byKey(const ValueKey('drawerItem_PROPERTYX.area'))); await tester.pumpAndSettle(); expect( find.text('Area'), findsAtLeastNWidgets(2), reason: 'Expected the area page', ); final tffFeet = find.byKey(const ValueKey('AREA.squareFeet')).evaluate().single.widget as TextFormField; final tffHectares = find.byKey(const ValueKey('AREA.hectares')).evaluate().single.widget as TextFormField; final tffMeters = find .byKey(const ValueKey('AREA.squareMeters')) .evaluate() .single .widget as TextFormField; await tester.enterText( find.byKey(const ValueKey('AREA.squareFeet')), '1000', ); await tester.pumpAndSettle(); expect( tffHectares.controller!.text, '0.009290304', reason: 'Conversion error', ); expect( tffMeters.controller!.text, '92.90304', reason: 'Conversion error', ); await tester.tap(find.byKey(const ValueKey('clearAll'))); await tester.pumpAndSettle(); expect(tffFeet.controller!.text, '', reason: 'Text not cleared'); expect(tffHectares.controller!.text, '', reason: 'Text not cleared'); expect(tffMeters.controller!.text, '', reason: 'Text not cleared'); }); }); testWidgets('Perform conversion, clear and undo', ( WidgetTester tester, ) async { await testInit(tester); final tffMiles = find.byKey(const ValueKey('LENGTH.miles')).evaluate().single.widget as TextFormField; final tffFeet = find.byKey(const ValueKey('LENGTH.feet')).evaluate().single.widget as TextFormField; final tffMeters = find.byKey(const ValueKey('LENGTH.meters')).evaluate().single.widget as TextFormField; expect( find.text('Length'), findsAtLeastNWidgets(2), reason: 'Expected the length page', ); await tester.enterText(find.byKey(const ValueKey('LENGTH.miles')), '1'); await tester.pumpAndSettle(); expect(tffFeet.controller!.text, '5280', reason: 'Conversion error'); expect(tffMeters.controller!.text, '1609.344', reason: 'Conversion error'); await tester.tap(find.byKey(const ValueKey('clearAll'))); await tester.pumpAndSettle(); expect(tffMiles.controller!.text, '', reason: 'Text not cleared'); expect(tffFeet.controller!.text, '', reason: 'Text not cleared'); expect(tffMeters.controller!.text, '', reason: 'Text not cleared'); await tester.tap(find.byKey(const ValueKey('undoClearAll'))); await tester.pumpAndSettle(); expect(tffMiles.controller!.text, '1.0', reason: 'Text not restored'); expect(tffFeet.controller!.text, '5280.0', reason: 'Text not restored'); expect(tffMeters.controller!.text, '1609.344', reason: 'Text not restored'); }); group('Language tasks:', () { testWidgets('Change language', (WidgetTester tester) async { await testInit(tester); await tester.tap(find.byKey(const ValueKey('drawerItem_settings'))); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('language-dropdown'))); await tester.pumpAndSettle(); await tester.tap(find.text('Italiano').last); await tester.pumpAndSettle(); await tester.tap(find.text('Lunghezza')); await tester.pumpAndSettle(); expect( find.text('Lunghezza'), findsAtLeastNWidgets(2), reason: 'Expected translated string', ); }); testWidgets('Check if language has been saved', ( WidgetTester tester, ) async { await testInit(tester, clearPrefs: false); expect( find.text('Lunghezza'), findsAtLeastNWidgets(2), reason: 'Expected translated string', ); await clearPreferences(); }); }); group('Reordering tasks:', () { testWidgets('Reorder units', (WidgetTester tester) async { await testInit(tester); // At the beginning the ordering is Meters, Centimeters, Inches, ... expect( tester.getCenter(find.text('Meters')).dy < tester.getCenter(find.text('Feet')).dy && tester.getCenter(find.text('Yards')).dy < tester.getCenter(find.text('Kilometers')).dy, true, reason: 'Initial ordering of length units is not what expected', ); await tester.tap(find.byKey(const ValueKey('drawerItem_settings'))); await tester.pumpAndSettle(); await tester.scrollUntilVisible( find.byKey(const ValueKey('reorder-units')), 300, scrollable: find.byType(Scrollable).at(1), maxScrolls: 2, ); await tester.tap(find.byKey(const ValueKey('reorder-units'))); await tester.pumpAndSettle(); await tester.tap( find.byKey(const ValueKey('chooseProperty-PROPERTYX.length')), ); await tester.pumpAndSettle(); final xDragHandle = tester .getCenter(find.byIcon(Icons.drag_handle).first) .dx; await dragGesture( tester, Offset(xDragHandle, tester.getCenter(find.text('Meters')).dy), Offset(xDragHandle, tester.getCenter(find.text('Yards')).dy), ); await tester.pumpAndSettle(); await dragGesture( tester, Offset(xDragHandle, tester.getCenter(find.text('Kilometers')).dy), Offset(xDragHandle, tester.getCenter(find.text('Feet')).dy), ); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('confirm'))); await tester.pumpAndSettle(); await tester.tap( find.byKey(const ValueKey('drawerItem_PROPERTYX.length')), ); await tester.pumpAndSettle(); // Now the ordering should be Inches, Centimeters, Meters, ... expect( tester.getCenter(find.text('Kilometers')).dy < tester.getCenter(find.text('Feet')).dy && tester.getCenter(find.text('Feet')).dy < tester.getCenter(find.text('Meters')).dy, true, reason: 'Final ordering of length units is not what expected', ); }); testWidgets('Check if units ordering has been saved', ( WidgetTester tester, ) async { await testInit(tester, clearPrefs: false); expect( tester.getCenter(find.text('Kilometers')).dy < tester.getCenter(find.text('Feet')).dy && tester.getCenter(find.text('Feet')).dy < tester.getCenter(find.text('Meters')).dy, true, reason: 'Ordering of length units is not what expected', ); }); testWidgets('Reorder properties', (WidgetTester tester) async { await testInit(tester); // At the beginning the ordering is Length, Area, Volume, ... expect( tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.length')), ) .dy < tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy && tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy < tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.volume')), ) .dy, true, reason: 'Initial ordering of properties is not what expected', ); await tester.tap(find.byKey(const ValueKey('drawerItem_settings'))); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('reorder-properties'))); await tester.pumpAndSettle(); final xDragHandle = tester .getCenter(find.byIcon(Icons.drag_handle).first) .dx; await dragGesture( tester, Offset(xDragHandle, tester.getCenter(find.text('Length').last).dy), Offset(xDragHandle, tester.getCenter(find.text('Currencies').last).dy), ); await tester.pumpAndSettle(); await dragGesture( tester, Offset(xDragHandle, tester.getCenter(find.text('Volume').last).dy), Offset(xDragHandle, tester.getCenter(find.text('Area').last).dy), ); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('confirm'))); await tester.pumpAndSettle(); // Now the ordering should be Volume, Area, Length, ... expect( tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.length')), ) .dy > tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy && tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy > tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.volume')), ) .dy, true, reason: 'Final ordering the of properties is not what expected', ); }); testWidgets('Check if properties ordering has been saved', ( WidgetTester tester, ) async { await testInit(tester, clearPrefs: false); expect( tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.length')), ) .dy > tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy && tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy > tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.volume')), ) .dy, true, reason: 'Ordering of the properties is not what expected', ); }); }); group('Conversion after reorder:', () { testWidgets('Change the order of properties and units and convert', ( WidgetTester tester, ) async { await testInit(tester); await tester.tap(find.byKey(const ValueKey('drawerItem_settings'))); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('reorder-properties'))); await tester.pumpAndSettle(); final xDragHandleProperties = tester .getCenter(find.byIcon(Icons.drag_handle).first) .dx; await dragGesture( tester, Offset( xDragHandleProperties, tester.getCenter(find.text('Length').last).dy, ), Offset( xDragHandleProperties, tester.getCenter(find.text('Currencies').last).dy, ), ); await tester.pumpAndSettle(); await dragGesture( tester, Offset( xDragHandleProperties, tester.getCenter(find.text('Volume').last).dy, ), Offset( xDragHandleProperties, tester.getCenter(find.text('Area').last).dy, ), ); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('confirm'))); await tester.pumpAndSettle(); await tester.scrollUntilVisible( find.byKey(const ValueKey('reorder-units')), 300, scrollable: find.byType(Scrollable).at(1), maxScrolls: 2, ); await tester.tap(find.byKey(const ValueKey('reorder-units'))); await tester.pumpAndSettle(); await tester.tap( find.byKey(const ValueKey('chooseProperty-PROPERTYX.length')), ); await tester.pumpAndSettle(); final xDragHandleUnits = tester .getCenter(find.byIcon(Icons.drag_handle).first) .dx; await dragGesture( tester, Offset(xDragHandleUnits, tester.getCenter(find.text('Meters')).dy), Offset(xDragHandleUnits, tester.getCenter(find.text('Yards')).dy), ); await tester.pumpAndSettle(); await dragGesture( tester, Offset(xDragHandleUnits, tester.getCenter(find.text('Kilometers')).dy), Offset(xDragHandleUnits, tester.getCenter(find.text('Feet')).dy), ); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('confirm'))); await tester.pumpAndSettle(); await tester.tap( find.byKey(const ValueKey('drawerItem_PROPERTYX.length')), ); await tester.pumpAndSettle(); await tester.enterText(find.byKey(const ValueKey('LENGTH.miles')), '1'); await tester.pumpAndSettle(); final tffFeet = find.byKey(const ValueKey('LENGTH.feet')).evaluate().single.widget as TextFormField; final tffMeters = find.byKey(const ValueKey('LENGTH.meters')).evaluate().single.widget as TextFormField; expect(tffFeet.controller!.text, '5280', reason: 'Conversion error'); expect( tffMeters.controller!.text, '1609.344', reason: 'Conversion error', ); }); testWidgets('Check if it is capable of the same conversion after restart', ( WidgetTester tester, ) async { await testInit(tester, clearPrefs: false); await tester.tap(find.text('Length')); await tester.pumpAndSettle(); await tester.enterText(find.byKey(const ValueKey('LENGTH.miles')), '1'); await tester.pumpAndSettle(); final tffFeet = find.byKey(const ValueKey('LENGTH.feet')).evaluate().single.widget as TextFormField; final tffMeters = find.byKey(const ValueKey('LENGTH.meters')).evaluate().single.widget as TextFormField; expect(tffFeet.controller!.text, '5280', reason: 'Conversion error'); expect( tffMeters.controller!.text, '1609.344', reason: 'Conversion error', ); }); }); } ================================================ FILE: integration_test/small_display_test.dart ================================================ import 'package:flutter_test/flutter_test.dart'; import 'package:integration_test/integration_test.dart'; import 'package:flutter/material.dart'; import 'package:converterpro/main.dart' as app; import 'utils.dart'; void main() { IntegrationTestWidgetsFlutterBinding.ensureInitialized(); Future testInit( WidgetTester tester, { bool clearPrefs = true, bool openFirstProperty = true, }) async { if (clearPrefs) { await clearPreferences(); } app.main(); await tester.pumpAndSettle(); setWindowSize(400, 800); await tester.pumpAndSettle(); if (openFirstProperty) { await tester.tap(find.byKey(const ValueKey('gridtile-0'))); await tester.pumpAndSettle(); } } group('Common conversions tasks:', () { testWidgets('Change to a new property and perform conversion', ( WidgetTester tester, ) async { await testInit(tester, openFirstProperty: false); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); await tester.tap( find.byKey(const ValueKey('drawerItem_PROPERTYX.currencies')), ); await tester.pumpAndSettle(); expect( find.text('Currencies'), findsAtLeastNWidgets(1), reason: 'Expected the currencies page', ); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('drawerItem_PROPERTYX.area'))); await tester.pumpAndSettle(); expect( find.text('Area'), findsAtLeastNWidgets(1), reason: 'Expected the area page', ); final tffFeet = find.byKey(const ValueKey('AREA.squareFeet')).evaluate().single.widget as TextFormField; final tffHectares = find.byKey(const ValueKey('AREA.hectares')).evaluate().single.widget as TextFormField; final tffMeters = find .byKey(const ValueKey('AREA.squareMeters')) .evaluate() .single .widget as TextFormField; await tester.enterText( find.byKey(const ValueKey('AREA.squareFeet')), '1000', ); await tester.pumpAndSettle(); expect( tffHectares.controller!.text, '0.009290304', reason: 'Conversion error', ); expect( tffMeters.controller!.text, '92.90304', reason: 'Conversion error', ); await tester.tap(find.byKey(const ValueKey('clearAll'))); await tester.pumpAndSettle(); expect(tffFeet.controller!.text, '', reason: 'Text not cleared'); expect(tffHectares.controller!.text, '', reason: 'Text not cleared'); expect(tffMeters.controller!.text, '', reason: 'Text not cleared'); }); }); testWidgets('Perform conversion, clear and undo', ( WidgetTester tester, ) async { await testInit(tester); final tffMiles = find.byKey(const ValueKey('LENGTH.miles')).evaluate().single.widget as TextFormField; final tffFeet = find.byKey(const ValueKey('LENGTH.feet')).evaluate().single.widget as TextFormField; final tffMeters = find.byKey(const ValueKey('LENGTH.meters')).evaluate().single.widget as TextFormField; expect( find.text('Length'), findsAtLeastNWidgets(1), reason: 'Expected the length page', ); await tester.enterText(find.byKey(const ValueKey('LENGTH.miles')), '1'); await tester.pumpAndSettle(); expect(tffFeet.controller!.text, '5280', reason: 'Conversion error'); expect(tffMeters.controller!.text, '1609.344', reason: 'Conversion error'); await tester.tap(find.byKey(const ValueKey('clearAll'))); await tester.pumpAndSettle(); expect(tffMiles.controller!.text, '', reason: 'Text not cleared'); expect(tffFeet.controller!.text, '', reason: 'Text not cleared'); expect(tffMeters.controller!.text, '', reason: 'Text not cleared'); await tester.tap(find.byKey(const ValueKey('undoClearAll'))); await tester.pumpAndSettle(); expect(tffMiles.controller!.text, '1.0', reason: 'Text not restored'); expect(tffFeet.controller!.text, '5280.0', reason: 'Text not restored'); expect(tffMeters.controller!.text, '1609.344', reason: 'Text not restored'); }); group('Language tasks:', () { testWidgets('Change language', (WidgetTester tester) async { await testInit(tester); await tester.pumpAndSettle(); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('drawerItem_settings'))); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('language-dropdown'))); await tester.pumpAndSettle(); await tester.tap(find.text('Italiano').last); await tester.pumpAndSettle(); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); await tester.tap(find.text('Lunghezza')); await tester.pumpAndSettle(); expect( find.text('Lunghezza'), findsAtLeastNWidgets(1), reason: 'Expected translated string', ); }); testWidgets('Check if language has been saved', ( WidgetTester tester, ) async { await testInit(tester, clearPrefs: false); expect( find.text('Lunghezza'), findsAtLeastNWidgets(1), reason: 'Expected translated string', ); await clearPreferences(); }); }); group('Reordering tasks:', () { testWidgets('Reorder units', (WidgetTester tester) async { await testInit(tester); // At the beginning the ordering is Meters, Centimeters, Inches, ... expect( tester.getCenter(find.text('Meters')).dy < tester.getCenter(find.text('Feet')).dy && tester.getCenter(find.text('Yards')).dy < tester.getCenter(find.text('Kilometers')).dy, true, reason: 'Initial ordering of length units is not what expected', ); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('drawerItem_settings'))); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('reorder-units'))); await tester.pumpAndSettle(); await tester.tap( find.byKey(const ValueKey('chooseProperty-PROPERTYX.length')), ); await tester.pumpAndSettle(); final xDragHandle = tester .getCenter(find.byIcon(Icons.drag_handle).first) .dx; await dragGesture( tester, Offset(xDragHandle, tester.getCenter(find.text('Meters')).dy), Offset(xDragHandle, tester.getCenter(find.text('Yards')).dy), ); await tester.pumpAndSettle(); await dragGesture( tester, Offset(xDragHandle, tester.getCenter(find.text('Kilometers')).dy), Offset(xDragHandle, tester.getCenter(find.text('Feet')).dy), ); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('confirm'))); await tester.pumpAndSettle(); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); await tester.tap( find.byKey(const ValueKey('drawerItem_PROPERTYX.length')), ); await tester.pumpAndSettle(); // Now the ordering should be Inches, Centimeters, Meters, ... expect( tester.getCenter(find.text('Kilometers')).dy < tester.getCenter(find.text('Feet')).dy && tester.getCenter(find.text('Feet')).dy < tester.getCenter(find.text('Meters')).dy, true, reason: 'Final ordering of length units is not what expected', ); }); testWidgets('Check if units ordering has been saved', ( WidgetTester tester, ) async { await testInit(tester, clearPrefs: false); expect( tester.getCenter(find.text('Kilometers')).dy < tester.getCenter(find.text('Feet')).dy && tester.getCenter(find.text('Feet')).dy < tester.getCenter(find.text('Meters')).dy, true, reason: 'Ordering of length units is not what expected', ); }); testWidgets('Reorder properties', (WidgetTester tester) async { await testInit(tester); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); // At the beginning the ordering is Length, Area, Volume, ... expect( tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.length')), ) .dy < tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy && tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy < tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.volume')), ) .dy, true, reason: 'Initial ordering of properties is not what expected', ); await tester.tap(find.byKey(const ValueKey('drawerItem_settings'))); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('reorder-properties'))); await tester.pumpAndSettle(); final xDragHandle = tester .getCenter(find.byIcon(Icons.drag_handle).first) .dx; await dragGesture( tester, Offset(xDragHandle, tester.getCenter(find.text('Length').last).dy), Offset(xDragHandle, tester.getCenter(find.text('Currencies').last).dy), ); await tester.pumpAndSettle(); await dragGesture( tester, Offset(xDragHandle, tester.getCenter(find.text('Volume').last).dy), Offset(xDragHandle, tester.getCenter(find.text('Area').last).dy), ); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('confirm'))); await tester.pumpAndSettle(); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); // Now the ordering should be Volume, Area, Length, ... expect( tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.length')), ) .dy > tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy && tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy > tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.volume')), ) .dy, true, reason: 'Final ordering the of properties is not what expected', ); }); testWidgets('Check if properties ordering has been saved', ( WidgetTester tester, ) async { await testInit(tester, clearPrefs: false); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); expect( tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.length')), ) .dy > tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy && tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.area')), ) .dy > tester .getCenter( find.byKey(const ValueKey('drawerItem_PROPERTYX.volume')), ) .dy, true, reason: 'Ordering of the properties is not what expected', ); }); }); group('Conversion after reorder:', () { testWidgets('Change the order of properties and units and convert', ( WidgetTester tester, ) async { await testInit(tester); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('drawerItem_settings'))); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('reorder-properties'))); await tester.pumpAndSettle(); final xDragHandleProperties = tester .getCenter(find.byIcon(Icons.drag_handle).first) .dx; await dragGesture( tester, Offset( xDragHandleProperties, tester.getCenter(find.text('Length').last).dy, ), Offset( xDragHandleProperties, tester.getCenter(find.text('Currencies').last).dy, ), ); await tester.pumpAndSettle(); await dragGesture( tester, Offset( xDragHandleProperties, tester.getCenter(find.text('Volume').last).dy, ), Offset( xDragHandleProperties, tester.getCenter(find.text('Area').last).dy, ), ); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('confirm'))); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('reorder-units'))); await tester.pumpAndSettle(); await tester.tap( find.byKey(const ValueKey('chooseProperty-PROPERTYX.length')), ); await tester.pumpAndSettle(); final xDragHandleUnits = tester .getCenter(find.byIcon(Icons.drag_handle).first) .dx; await dragGesture( tester, Offset(xDragHandleUnits, tester.getCenter(find.text('Meters')).dy), Offset(xDragHandleUnits, tester.getCenter(find.text('Yards')).dy), ); await tester.pumpAndSettle(); await dragGesture( tester, Offset(xDragHandleUnits, tester.getCenter(find.text('Kilometers')).dy), Offset(xDragHandleUnits, tester.getCenter(find.text('Feet')).dy), ); await tester.pumpAndSettle(); await tester.tap(find.byKey(const ValueKey('confirm'))); await tester.pumpAndSettle(); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); await tester.tap( find.byKey(const ValueKey('drawerItem_PROPERTYX.length')), ); await tester.pumpAndSettle(); await tester.enterText(find.byKey(const ValueKey('LENGTH.miles')), '1'); await tester.pumpAndSettle(); final tffFeet = find.byKey(const ValueKey('LENGTH.feet')).evaluate().single.widget as TextFormField; final tffMeters = find.byKey(const ValueKey('LENGTH.meters')).evaluate().single.widget as TextFormField; expect(tffFeet.controller!.text, '5280', reason: 'Conversion error'); expect( tffMeters.controller!.text, '1609.344', reason: 'Conversion error', ); }); testWidgets('Check if it is capable of the same conversion after restart', ( WidgetTester tester, ) async { await testInit(tester, clearPrefs: false); await tester.tap(find.byIcon(Icons.menu)); // Open drawer await tester.pumpAndSettle(); await tester.tap(find.text('Length')); await tester.pumpAndSettle(); await tester.enterText(find.byKey(const ValueKey('LENGTH.miles')), '1'); await tester.pumpAndSettle(); final tffFeet = find.byKey(const ValueKey('LENGTH.feet')).evaluate().single.widget as TextFormField; final tffMeters = find.byKey(const ValueKey('LENGTH.meters')).evaluate().single.widget as TextFormField; expect(tffFeet.controller!.text, '5280', reason: 'Conversion error'); expect( tffMeters.controller!.text, '1609.344', reason: 'Conversion error', ); }); }); } ================================================ FILE: integration_test/utils.dart ================================================ import 'package:flutter/gestures.dart'; import 'package:flutter/material.dart'; import 'package:flutter_test/flutter_test.dart'; import 'package:shared_preferences/shared_preferences.dart'; import 'package:window_size/window_size.dart'; /// Clears the saved shared preferences Future clearPreferences() async { final pref = await SharedPreferencesWithCache.create( cacheOptions: const SharedPreferencesWithCacheOptions(), ); await pref.clear(); } /// Perform a drag from [start] to [end]. Useful for reorderable list Future dragGesture(WidgetTester tester, Offset start, Offset end) async { final TestGesture drag = await tester.startGesture(start); await tester.pump(kPressTimeout); await drag.moveTo(end); await tester.pump(kPressTimeout); await drag.up(); } /// Sets the window size void setWindowSize(double width, double height) { final size = Size(width, height); setWindowMinSize(size); setWindowMaxSize(size); } ================================================ FILE: lib/app_router.dart ================================================ import 'package:converterpro/models/conversions.dart'; import 'package:converterpro/models/hide_units.dart'; import 'package:converterpro/models/order.dart'; import 'package:converterpro/models/properties_list.dart'; import 'package:converterpro/models/settings.dart'; import 'package:converterpro/pages/conversion_page.dart'; import 'package:converterpro/pages/error_page.dart'; import 'package:converterpro/pages/hide_units_page.dart'; import 'package:converterpro/pages/initial_page.dart'; import 'package:converterpro/pages/reorder_properties_page.dart'; import 'package:converterpro/pages/reorder_units_page.dart'; import 'package:converterpro/pages/settings_page.dart'; import 'package:converterpro/pages/splash_screen.dart'; import 'package:converterpro/pages/app_scaffold.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_riverpod/legacy.dart'; import 'package:flutter_svg/flutter_svg.dart'; import 'package:go_router/go_router.dart'; import 'package:translations/app_localizations.dart'; import 'package:vector_graphics/vector_graphics.dart'; final isEverythingLoadedProvider = Provider( (ref) => ref.watch(significantFiguresProvider).hasValue && ref.watch(removeTrailingZerosProvider).hasValue && ref.watch(isPureDarkProvider).hasValue && ref.watch(propertySelectionOnStartupProvider).hasValue && ref.watch(useDeviceColorProvider).hasValue && ref.watch(colorThemeProvider).hasValue && ref.watch(revokeInternetProvider).hasValue && ref.watch(themeModeProvider).hasValue && ref.watch(languageTagProvider).hasValue && ref.watch(PropertiesOrderNotifier.provider).hasValue && ref.watch(UnitsOrderNotifier.provider).hasValue && ref.watch(ConversionsNotifier.provider).hasValue && ref.watch(HiddenUnitsNotifier.provider).hasValue && ref.watch(propertiesMapProvider).hasValue, ); final conversionPageHeroEnabledProvider = StateProvider((ref) => false); final routerProvider = Provider( (ref) => GoRouter( routes: [ GoRoute(path: '/', builder: (context, _) => const SplashScreen()), GoRoute( path: '/conversions', name: 'conversions', builder: (context, state) => const InitialPage(), redirect: (context, state) { ref.read(conversionPageHeroEnabledProvider.notifier).state = true; return null; }, ), ShellRoute( builder: (context, state, child) => AppScaffold(child: child), routes: [ GoRoute( path: '/conversions/:property', builder: (context, state) { final String property = state.pathParameters['property']!; final propertyx = kebabStringToPropertyX(property); return ConversionPage(propertyx); }, ), GoRoute( path: '/settings', name: 'settings', builder: (context, state) => const SettingsPage(), routes: [ GoRoute( path: 'reorder-properties', name: 'reorder-properties', builder: (context, state) => const ReorderPropertiesPage(), ), GoRoute( path: 'reorder-units', name: 'reorder-units', builder: (context, state) => const ReorderUnitsPage(), routes: [ GoRoute( path: ':property', builder: (context, state) { final String property = state.pathParameters['property']!; final propertyx = kebabStringToPropertyX(property); return ReorderUnitsPage( selectedProperty: propertyx, isPropertySelected: true, ); }, ), ], ), GoRoute( path: 'hide-units', name: 'hide-units', builder: (context, state) => const HideUnitsPage(), routes: [ GoRoute( path: ':property', builder: (context, state) { final String property = state.pathParameters['property']!; final propertyx = kebabStringToPropertyX(property); return HideUnitsPage( selectedProperty: propertyx, isPropertySelected: true, ); }, ), ], ), GoRoute( path: 'about', name: 'about', builder: (context, state) => LicensePage( applicationName: AppLocalizations.of(context)!.appName, applicationIcon: const SvgPicture( AssetBytesLoader('assets/app_icons_opti/logo.svg.vec'), width: 54, ), ), ), ], ), ], ), ], redirect: (context, state) { // Bypass splashscreen if variables are already loaded if (state.uri.toString() == '/') { if (ref.read(isEverythingLoadedProvider)) { final conversionsOrderDrawer = ref .read(PropertiesOrderNotifier.provider) .value!; return '/conversions/${conversionsOrderDrawer[0].toKebabCase()}'; } } return null; }, errorBuilder: (context, state) => const ErrorPage(), ), ); ================================================ FILE: lib/data/default_order.dart ================================================ import 'package:converterpro/utils/utils.dart'; import 'package:units_converter/units_converter.dart'; const defaultPropertiesOrder = [ PROPERTYX.length, PROPERTYX.area, PROPERTYX.volume, PROPERTYX.currencies, PROPERTYX.time, PROPERTYX.temperature, PROPERTYX.speed, PROPERTYX.mass, PROPERTYX.force, PROPERTYX.fuelConsumption, PROPERTYX.numeralSystems, PROPERTYX.pressure, PROPERTYX.energy, PROPERTYX.power, PROPERTYX.density, PROPERTYX.angle, PROPERTYX.shoeSize, PROPERTYX.digitalData, PROPERTYX.siPrefixes, PROPERTYX.torque, ]; const Map> defaultUnitsOrder = { PROPERTYX.length: [ LENGTH.meters, LENGTH.feet, LENGTH.yards, LENGTH.kilometers, LENGTH.miles, LENGTH.nauticalMiles, LENGTH.centimeters, LENGTH.inches, LENGTH.mils, LENGTH.millimeters, LENGTH.micrometers, LENGTH.nanometers, LENGTH.angstroms, LENGTH.picometers, LENGTH.feetUs, LENGTH.astronomicalUnits, LENGTH.lightYears, LENGTH.parsec, ], PROPERTYX.area: [ AREA.squareMeters, AREA.squareFeet, AREA.squareYard, AREA.hectares, AREA.acres, AREA.squareKilometers, AREA.squareMiles, AREA.squareCentimeters, AREA.squareMillimeters, AREA.squareInches, AREA.are, AREA.squareFeetUs, ], PROPERTYX.density: [ DENSITY.gramsPerCubicCentimeter, DENSITY.kilogramsPerCubicMeter, DENSITY.poundsPerCubicInches, DENSITY.poundsPerCubicFeet, DENSITY.gramsPerLiter, DENSITY.gramsPerMilliliter, DENSITY.gramsPerDeciliter, DENSITY.kilogramsPerLiter, DENSITY.milligramsPerLiter, DENSITY.milligramsPerDeciliter, DENSITY.milligramsPerMilliliter, DENSITY.milligramsPerCubicMeter, DENSITY.milligramsPerCubicCentimeter, DENSITY.microgramsPerLiter, DENSITY.microgramsPerDeciliter, DENSITY.microgramsPerMilliliter, ], PROPERTYX.volume: [ VOLUME.cubicMeters, VOLUME.liters, VOLUME.usGallons, VOLUME.imperialGallons, VOLUME.usPints, VOLUME.imperialPints, VOLUME.usQuarts, VOLUME.deciliters, VOLUME.centiliters, VOLUME.milliliters, VOLUME.microliters, VOLUME.tablespoonsUs, VOLUME.australianTablespoons, VOLUME.teaspoonsUs, VOLUME.teaspoonsMetric, VOLUME.cups, VOLUME.cubicMillimeters, VOLUME.cubicCentimeters, VOLUME.cubicInches, VOLUME.cubicFeet, VOLUME.usFluidOunces, VOLUME.imperialFluidOunces, VOLUME.usGill, VOLUME.imperialGill, ], PROPERTYX.currencies: [ 'USD', 'EUR', 'JPY', 'GBP', 'CNY', 'AUD', 'CAD', 'CHF', 'SEK', 'NOK', 'DKK', 'KRW', 'MXN', 'INR', 'BRL', 'ZAR', 'TRY', 'PLN', 'CZK', 'HUF', 'RON', 'IDR', 'THB', 'PHP', 'MYR', 'HKD', 'SGD', 'NZD', 'ILS', 'ISK', ], PROPERTYX.time: [ TIME.seconds, TIME.minutes, TIME.hours, TIME.days, TIME.weeks, TIME.years365, TIME.lustrum, TIME.decades, TIME.centuries, TIME.millennium, TIME.deciseconds, TIME.centiseconds, TIME.milliseconds, TIME.microseconds, TIME.nanoseconds, ], PROPERTYX.temperature: [ TEMPERATURE.celsius, TEMPERATURE.fahrenheit, TEMPERATURE.kelvin, TEMPERATURE.reamur, TEMPERATURE.romer, TEMPERATURE.delisle, TEMPERATURE.rankine, ], PROPERTYX.speed: [ SPEED.kilometersPerHour, SPEED.milesPerHour, SPEED.metersPerSecond, SPEED.feetsPerSecond, SPEED.knots, SPEED.minutesPerKilometer, SPEED.minutesPerMile, SPEED.speedOfLight, ], PROPERTYX.mass: [ MASS.kilograms, MASS.pounds, MASS.ounces, MASS.tonnes, MASS.grams, MASS.ettograms, MASS.centigrams, MASS.milligrams, MASS.carats, MASS.quintals, MASS.pennyweights, MASS.troyOunces, MASS.uma, MASS.stones, ], PROPERTYX.force: [ FORCE.newton, FORCE.kilogramForce, FORCE.poundForce, FORCE.dyne, FORCE.poundal, ], PROPERTYX.fuelConsumption: [ FUEL_CONSUMPTION.kilometersPerLiter, FUEL_CONSUMPTION.litersPer100km, FUEL_CONSUMPTION.milesPerUsGallon, FUEL_CONSUMPTION.milesPerImperialGallon, ], PROPERTYX.numeralSystems: [ NUMERAL_SYSTEMS.decimal, NUMERAL_SYSTEMS.hexadecimal, NUMERAL_SYSTEMS.octal, NUMERAL_SYSTEMS.binary, ], PROPERTYX.pressure: [ PRESSURE.atmosphere, PRESSURE.bar, PRESSURE.millibar, PRESSURE.psi, PRESSURE.pascal, PRESSURE.kiloPascal, PRESSURE.torr, PRESSURE.inchOfMercury, PRESSURE.hectoPascal, PRESSURE.ksi, PRESSURE.megaPascal, PRESSURE.gigaPascal, ], PROPERTYX.energy: [ ENERGY.kilowattHours, ENERGY.wattHours, ENERGY.kilocalories, ENERGY.calories, ENERGY.joules, ENERGY.kilojoules, ENERGY.electronvolts, ENERGY.energyFootPound, ENERGY.britishThermalUnit, ], PROPERTYX.power: [ POWER.kilowatt, POWER.europeanHorsePower, POWER.imperialHorsePower, POWER.watt, POWER.megawatt, POWER.gigawatt, POWER.milliwatt, ], PROPERTYX.angle: [ANGLE.degree, ANGLE.radians, ANGLE.minutes, ANGLE.seconds], PROPERTYX.shoeSize: [ SHOE_SIZE.centimeters, SHOE_SIZE.inches, SHOE_SIZE.euChina, SHOE_SIZE.usaCanadaChild, SHOE_SIZE.usaCanadaMan, SHOE_SIZE.usaCanadaWoman, SHOE_SIZE.ukIndiaChild, SHOE_SIZE.ukIndiaMan, SHOE_SIZE.ukIndiaWoman, SHOE_SIZE.japan, ], PROPERTYX.digitalData: [ DIGITAL_DATA.byte, DIGITAL_DATA.bit, DIGITAL_DATA.nibble, DIGITAL_DATA.kibibyte, DIGITAL_DATA.mebibyte, DIGITAL_DATA.gibibyte, DIGITAL_DATA.tebibyte, DIGITAL_DATA.pebibyte, DIGITAL_DATA.exbibyte, DIGITAL_DATA.kilobyte, DIGITAL_DATA.megabyte, DIGITAL_DATA.gigabyte, DIGITAL_DATA.terabyte, DIGITAL_DATA.petabyte, DIGITAL_DATA.exabyte, DIGITAL_DATA.kibibit, DIGITAL_DATA.mebibit, DIGITAL_DATA.gibibit, DIGITAL_DATA.tebibit, DIGITAL_DATA.pebibit, DIGITAL_DATA.exbibit, DIGITAL_DATA.kilobit, DIGITAL_DATA.megabit, DIGITAL_DATA.gigabit, DIGITAL_DATA.terabit, DIGITAL_DATA.petabit, DIGITAL_DATA.exabit, ], PROPERTYX.siPrefixes: [ SI_PREFIXES.base, SI_PREFIXES.deca, SI_PREFIXES.hecto, SI_PREFIXES.kilo, SI_PREFIXES.mega, SI_PREFIXES.giga, SI_PREFIXES.tera, SI_PREFIXES.peta, SI_PREFIXES.exa, SI_PREFIXES.zetta, SI_PREFIXES.yotta, SI_PREFIXES.deci, SI_PREFIXES.centi, SI_PREFIXES.milli, SI_PREFIXES.micro, SI_PREFIXES.nano, SI_PREFIXES.pico, SI_PREFIXES.femto, SI_PREFIXES.atto, SI_PREFIXES.zepto, SI_PREFIXES.yocto, ], PROPERTYX.torque: [ TORQUE.newtonMeter, TORQUE.kilogramForceMeter, TORQUE.dyneMeter, TORQUE.poundForceFeet, TORQUE.poundForceInch, TORQUE.poundalMeter, ], }; ================================================ FILE: lib/data/property_unit_maps.dart ================================================ import 'package:flutter/cupertino.dart'; import 'package:translations/app_localizations.dart'; import 'package:units_converter/units_converter.dart'; import 'package:converterpro/utils/utils.dart'; /// This will return the map of [PropertyUi], a record that contains all the /// data regarding the displaying of the property all over the app. Map getPropertyUiMap(BuildContext context) { const String basePath = 'assets/property_icons_opti'; final l10n = AppLocalizations.of(context)!; return { PROPERTYX.length: l10n.length, PROPERTYX.area: l10n.area, PROPERTYX.density: l10n.density, PROPERTYX.volume: l10n.volume, PROPERTYX.currencies: l10n.currencies, PROPERTYX.time: l10n.time, PROPERTYX.temperature: l10n.temperature, PROPERTYX.speed: l10n.speed, PROPERTYX.mass: l10n.mass, PROPERTYX.force: l10n.force, PROPERTYX.fuelConsumption: l10n.fuelConsumption, PROPERTYX.numeralSystems: l10n.numeralSystems, PROPERTYX.pressure: l10n.pressure, PROPERTYX.energy: l10n.energy, PROPERTYX.power: l10n.power, PROPERTYX.angle: l10n.angles, PROPERTYX.shoeSize: l10n.shoeSize, PROPERTYX.digitalData: l10n.digitalData, PROPERTYX.siPrefixes: l10n.siPrefixes, PROPERTYX.torque: l10n.torque, }.map( (key, value) => MapEntry(key, ( name: value, icon: '$basePath/${key.name}.svg.vec', selectedIcon: '$basePath/${key.name}_filled.svg.vec', )), ); } /// This will return a Map from a [PROPERTYX] to a Map from a unit key to the /// name that will be displayed on the app. Map> getUnitUiMap(BuildContext context) { final l10n = AppLocalizations.of(context)!; return { PROPERTYX.length: { LENGTH.meters: l10n.meters, LENGTH.feet: l10n.feet, LENGTH.yards: l10n.yards, LENGTH.kilometers: l10n.kilometers, LENGTH.miles: l10n.miles, LENGTH.nauticalMiles: l10n.nauticalMiles, LENGTH.centimeters: l10n.centimeters, LENGTH.inches: l10n.inches, LENGTH.mils: l10n.mils, LENGTH.millimeters: l10n.millimeters, LENGTH.micrometers: l10n.micrometers, LENGTH.nanometers: l10n.nanometers, LENGTH.angstroms: l10n.angstroms, LENGTH.picometers: l10n.picometers, LENGTH.feetUs: l10n.feetUsSurvey, LENGTH.astronomicalUnits: l10n.astronomicalUnits, LENGTH.lightYears: l10n.lightYears, LENGTH.parsec: l10n.parsec, }, PROPERTYX.area: { AREA.squareMeters: l10n.squareMeters, AREA.squareFeet: l10n.squareFeet, AREA.squareYard: l10n.squareYard, AREA.hectares: l10n.hectares, AREA.acres: l10n.acres, AREA.squareKilometers: l10n.squareKilometers, AREA.squareMiles: l10n.squareMiles, AREA.squareCentimeters: l10n.squareCentimeters, AREA.squareMillimeters: l10n.squareMillimeters, AREA.squareInches: l10n.squareInches, AREA.are: l10n.are, AREA.squareFeetUs: l10n.squareFeetUsSurvey, }, PROPERTYX.density: { DENSITY.gramsPerLiter: l10n.gramsPerLiter, DENSITY.gramsPerCubicCentimeter: l10n.gramsPerCubicCentimeter, DENSITY.gramsPerMilliliter: l10n.gramsPerMilliliter, DENSITY.gramsPerDeciliter: l10n.gramsPerDeciliter, DENSITY.kilogramsPerLiter: l10n.kilogramsPerLiter, DENSITY.kilogramsPerCubicMeter: l10n.kilogramsPerCubicMeter, DENSITY.milligramsPerLiter: l10n.milligramsPerLiter, DENSITY.milligramsPerDeciliter: l10n.milligramsPerDeciliter, DENSITY.milligramsPerMilliliter: l10n.milligramsPerMilliliter, DENSITY.milligramsPerCubicMeter: l10n.milligramsPerCubicMeter, DENSITY.milligramsPerCubicCentimeter: l10n.milligramsPerCubicCentimeter, DENSITY.microgramsPerLiter: l10n.microgramsPerLiter, DENSITY.microgramsPerDeciliter: l10n.microgramsPerDeciliter, DENSITY.microgramsPerMilliliter: l10n.microgramsPerMilliliter, DENSITY.poundsPerCubicInches: l10n.poundsPerCubicInches, DENSITY.poundsPerCubicFeet: l10n.poundsPerCubicFeet, }, PROPERTYX.volume: { VOLUME.cubicMeters: l10n.cubicMeters, VOLUME.liters: l10n.liters, VOLUME.usGallons: l10n.usGallons, VOLUME.imperialGallons: l10n.imperialGallons, VOLUME.usPints: l10n.usPints, VOLUME.imperialPints: l10n.imperialPints, VOLUME.usQuarts: l10n.usQuarts, VOLUME.deciliters: l10n.deciliters, VOLUME.centiliters: l10n.centiliters, VOLUME.milliliters: l10n.milliliters, VOLUME.microliters: l10n.microliters, VOLUME.tablespoonsUs: l10n.tablespoonUs, VOLUME.australianTablespoons: l10n.tablespoonAustralian, VOLUME.teaspoonsUs: l10n.teaspoonsUs, VOLUME.teaspoonsMetric: l10n.teaspoonsMetric, VOLUME.cups: l10n.cups, VOLUME.cubicMillimeters: l10n.cubicMillimeters, VOLUME.cubicCentimeters: l10n.cubicCentimeters, VOLUME.cubicInches: l10n.cubicInches, VOLUME.cubicFeet: l10n.cubicFeet, VOLUME.usFluidOunces: l10n.usFluidOunces, VOLUME.imperialFluidOunces: l10n.imperialFluidOunces, VOLUME.usGill: l10n.usGill, VOLUME.imperialGill: l10n.imperialGill, }, PROPERTYX.currencies: { 'USD': l10n.usd, 'EUR': l10n.eur, 'JPY': l10n.jpy, 'GBP': l10n.gbp, 'CNY': l10n.cny, 'AUD': l10n.aud, 'CAD': l10n.cad, 'CHF': l10n.chf, 'SEK': l10n.sek, 'NOK': l10n.nok, 'DKK': l10n.dkk, 'KRW': l10n.krw, 'MXN': l10n.mxn, 'INR': l10n.inr, 'BRL': l10n.brl, 'ZAR': l10n.zar, 'TRY': l10n.trY, 'PLN': l10n.pln, 'CZK': l10n.czk, 'HUF': l10n.huf, 'RON': l10n.ron, 'IDR': l10n.idr, 'THB': l10n.thb, 'PHP': l10n.php, 'MYR': l10n.myr, 'HKD': l10n.hkd, 'SGD': l10n.sgd, 'NZD': l10n.nzd, 'ILS': l10n.ils, 'ISK': l10n.isk, }, PROPERTYX.time: { TIME.seconds: l10n.seconds, TIME.minutes: l10n.minutes, TIME.hours: l10n.hours, TIME.days: l10n.days, TIME.weeks: l10n.weeks, TIME.years365: l10n.years, TIME.lustrum: l10n.lustrum, TIME.decades: l10n.decades, TIME.centuries: l10n.centuries, TIME.millennium: l10n.millennium, TIME.deciseconds: l10n.deciseconds, TIME.centiseconds: l10n.centiseconds, TIME.milliseconds: l10n.milliseconds, TIME.microseconds: l10n.microseconds, TIME.nanoseconds: l10n.nanoseconds, }, PROPERTYX.temperature: { TEMPERATURE.celsius: l10n.celsius, TEMPERATURE.fahrenheit: l10n.fahrenheit, TEMPERATURE.kelvin: l10n.kelvin, TEMPERATURE.reamur: l10n.reamur, TEMPERATURE.romer: l10n.romer, TEMPERATURE.delisle: l10n.delisle, TEMPERATURE.rankine: l10n.rankine, }, PROPERTYX.speed: { SPEED.kilometersPerHour: l10n.kilometersHour, SPEED.milesPerHour: l10n.milesHour, SPEED.metersPerSecond: l10n.metersSecond, SPEED.feetsPerSecond: l10n.feetSecond, SPEED.knots: l10n.knots, SPEED.minutesPerKilometer: l10n.minutesPerKilometer, SPEED.minutesPerMile: l10n.minutesPerMile, SPEED.speedOfLight: l10n.speedOfLight, }, PROPERTYX.mass: { MASS.kilograms: l10n.kilograms, MASS.pounds: l10n.pounds, MASS.ounces: l10n.ounces, MASS.tonnes: l10n.tonnes, MASS.grams: l10n.grams, MASS.ettograms: l10n.ettograms, MASS.centigrams: l10n.centigrams, MASS.milligrams: l10n.milligrams, MASS.carats: l10n.carats, MASS.quintals: l10n.quintals, MASS.pennyweights: l10n.pennyweights, MASS.troyOunces: l10n.troyOunces, MASS.uma: l10n.uma, MASS.stones: l10n.stones, }, PROPERTYX.force: { FORCE.newton: l10n.newton, FORCE.kilogramForce: l10n.kilogramForce, FORCE.poundForce: l10n.poundForce, FORCE.dyne: l10n.dyne, FORCE.poundal: l10n.poundal, }, PROPERTYX.fuelConsumption: { FUEL_CONSUMPTION.kilometersPerLiter: l10n.kilometersLiter, FUEL_CONSUMPTION.litersPer100km: l10n.liters100km, FUEL_CONSUMPTION.milesPerUsGallon: l10n.milesUsGallon, FUEL_CONSUMPTION.milesPerImperialGallon: l10n.milesImperialGallon, }, PROPERTYX.numeralSystems: { NUMERAL_SYSTEMS.decimal: l10n.decimal, NUMERAL_SYSTEMS.hexadecimal: l10n.hexadecimal, NUMERAL_SYSTEMS.octal: l10n.octal, NUMERAL_SYSTEMS.binary: l10n.binary, }, PROPERTYX.pressure: { PRESSURE.atmosphere: l10n.atmosphere, PRESSURE.bar: l10n.bar, PRESSURE.millibar: l10n.millibar, PRESSURE.psi: l10n.psi, PRESSURE.pascal: l10n.pascal, PRESSURE.kiloPascal: l10n.kiloPascal, PRESSURE.torr: l10n.torr, PRESSURE.inchOfMercury: l10n.inchesOfMercury, PRESSURE.hectoPascal: l10n.hectoPascal, PRESSURE.ksi: l10n.ksi, PRESSURE.megaPascal: l10n.megaPascal, PRESSURE.gigaPascal: l10n.gigaPascal, }, PROPERTYX.energy: { ENERGY.kilowattHours: l10n.kilowattHour, ENERGY.wattHours: l10n.watthour, ENERGY.kilocalories: l10n.kilocalories, ENERGY.calories: l10n.calories, ENERGY.joules: l10n.joule, ENERGY.kilojoules: l10n.kilojoules, ENERGY.electronvolts: l10n.electronvolt, ENERGY.energyFootPound: l10n.footPound, ENERGY.britishThermalUnit: l10n.britishThermalUnit, }, PROPERTYX.power: { POWER.kilowatt: l10n.kilowatt, POWER.europeanHorsePower: l10n.europeanHorsePower, POWER.imperialHorsePower: l10n.imperialHorsePower, POWER.watt: l10n.watt, POWER.megawatt: l10n.megawatt, POWER.gigawatt: l10n.gigawatt, POWER.milliwatt: l10n.milliwatt, }, PROPERTYX.angle: { ANGLE.degree: l10n.degree, ANGLE.radians: l10n.radiansDegree, ANGLE.minutes: l10n.minutes, ANGLE.seconds: l10n.seconds, }, PROPERTYX.shoeSize: { SHOE_SIZE.centimeters: l10n.centimeters, SHOE_SIZE.inches: l10n.inches, SHOE_SIZE.euChina: l10n.euChina, SHOE_SIZE.usaCanadaChild: l10n.usaCanadaChild, SHOE_SIZE.usaCanadaMan: l10n.usaCanadaMan, SHOE_SIZE.usaCanadaWoman: l10n.usaCanadaWoman, SHOE_SIZE.ukIndiaChild: l10n.ukIndiaChild, SHOE_SIZE.ukIndiaMan: l10n.ukIndiaMan, SHOE_SIZE.ukIndiaWoman: l10n.ukIndiaWoman, SHOE_SIZE.japan: l10n.japan, }, PROPERTYX.digitalData: { DIGITAL_DATA.byte: l10n.byte, DIGITAL_DATA.bit: l10n.bit, DIGITAL_DATA.nibble: l10n.nibble, DIGITAL_DATA.kilobyte: l10n.kilobyte, DIGITAL_DATA.megabyte: l10n.megabyte, DIGITAL_DATA.gigabyte: l10n.gigabyte, DIGITAL_DATA.terabyte: l10n.terabyte, DIGITAL_DATA.petabyte: l10n.petabyte, DIGITAL_DATA.exabyte: l10n.exabyte, DIGITAL_DATA.kibibyte: l10n.kibibyte, DIGITAL_DATA.mebibyte: l10n.mebibyte, DIGITAL_DATA.gibibyte: l10n.gibibyte, DIGITAL_DATA.tebibyte: l10n.tebibyte, DIGITAL_DATA.pebibyte: l10n.pebibyte, DIGITAL_DATA.exbibyte: l10n.exbibyte, DIGITAL_DATA.kilobit: l10n.kilobit, DIGITAL_DATA.megabit: l10n.megabit, DIGITAL_DATA.gigabit: l10n.gigabit, DIGITAL_DATA.terabit: l10n.terabit, DIGITAL_DATA.petabit: l10n.petabit, DIGITAL_DATA.exabit: l10n.exabit, DIGITAL_DATA.kibibit: l10n.kibibit, DIGITAL_DATA.mebibit: l10n.mebibit, DIGITAL_DATA.gibibit: l10n.gibibit, DIGITAL_DATA.tebibit: l10n.tebibit, DIGITAL_DATA.pebibit: l10n.pebibit, DIGITAL_DATA.exbibit: l10n.exbibit, }, PROPERTYX.siPrefixes: { SI_PREFIXES.base: l10n.base, SI_PREFIXES.deca: l10n.deca, SI_PREFIXES.hecto: l10n.hecto, SI_PREFIXES.kilo: l10n.kilo, SI_PREFIXES.mega: l10n.mega, SI_PREFIXES.giga: l10n.giga, SI_PREFIXES.tera: l10n.tera, SI_PREFIXES.peta: l10n.peta, SI_PREFIXES.exa: l10n.exa, SI_PREFIXES.zetta: l10n.zetta, SI_PREFIXES.yotta: l10n.yotta, SI_PREFIXES.deci: l10n.deci, SI_PREFIXES.centi: l10n.centi, SI_PREFIXES.milli: l10n.milli, SI_PREFIXES.micro: l10n.micro, SI_PREFIXES.nano: l10n.nano, SI_PREFIXES.pico: l10n.pico, SI_PREFIXES.femto: l10n.femto, SI_PREFIXES.atto: l10n.atto, SI_PREFIXES.zepto: l10n.zepto, SI_PREFIXES.yocto: l10n.yocto, }, PROPERTYX.torque: { TORQUE.newtonMeter: l10n.newtonMeter, TORQUE.kilogramForceMeter: l10n.kilogramForceMeter, TORQUE.dyneMeter: l10n.dyneMeter, TORQUE.poundForceFeet: l10n.poundForceFeet, TORQUE.poundForceInch: l10n.poundForceInch, TORQUE.poundalMeter: l10n.poundalMeter, }, }; } ================================================ FILE: lib/helpers/responsive_helper.dart ================================================ import 'package:converterpro/styles/consts.dart'; double responsivePadding(double displayWidth) => displayWidth * 0.03; int responsiveNumCols(double displayWidth) { if (displayWidth < pixelWidth1Column) { return 1; } else if (displayWidth < pixelWidth2Columns) { return 2; } else if (displayWidth < pixelWidth3Columns) { return 3; } return 4; } double responsiveChildAspectRatio(double width, int colNumber) => width / (colNumber * 90); bool isDrawerFixed(double displayWidth) => displayWidth > pixelFixedDrawer; ================================================ FILE: lib/main.dart ================================================ import 'package:converterpro/app_router.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/services.dart'; import 'package:converterpro/models/settings.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:translations/app_localizations.dart'; import 'package:dynamic_color/dynamic_color.dart'; void main() async { LicenseRegistry.addLicense(() async* { yield LicenseEntryWithLineBreaks([ 'Josefin Sans', ], await rootBundle.loadString('assets/fonts/OFL.txt')); }); WidgetsFlutterBinding.ensureInitialized(); runApp(const ProviderScope(child: MyApp())); } class MyApp extends ConsumerWidget { const MyApp({super.key}); @override Widget build(BuildContext context, WidgetRef ref) { return DynamicColorBuilder( builder: (ColorScheme? lightDynamic, ColorScheme? darkDynamic) { if (lightDynamic != null) { WidgetsBinding.instance.addPostFrameCallback( (_) => ref.read(deviceAccentColorProvider.notifier).state = lightDynamic.primary, ); } return Consumer( builder: (context, ref, child) { final colorTheme = ref.watch(actualColorThemeProvider); ThemeData lightTheme = ThemeData( colorScheme: ColorScheme.fromSeed(seedColor: colorTheme), ); ThemeData darkTheme = ThemeData( colorScheme: ColorScheme.fromSeed( seedColor: colorTheme, brightness: Brightness.dark, ), brightness: Brightness.dark, ); ThemeData amoledTheme = darkTheme.copyWith( scaffoldBackgroundColor: Colors.black, drawerTheme: const DrawerThemeData(backgroundColor: Colors.black), appBarTheme: const AppBarTheme(backgroundColor: Colors.black), bottomAppBarTheme: const BottomAppBarThemeData( color: Colors.black, ), ); final pageTransitionsTheme = PageTransitionsTheme( builders: Map.fromIterable( TargetPlatform.values, value: (e) => e == TargetPlatform.android ? const PredictiveBackFullscreenPageTransitionsBuilder() : const FadeForwardsPageTransitionsBuilder(), ), ); lightTheme = lightTheme.copyWith( pageTransitionsTheme: pageTransitionsTheme, ); darkTheme = darkTheme.copyWith( pageTransitionsTheme: pageTransitionsTheme, ); amoledTheme = amoledTheme.copyWith( pageTransitionsTheme: pageTransitionsTheme, ); final ThemeMode themeMode = ThemeMode.values[ref.watch(themeModeProvider).value ?? ThemeMode.system.index]; final appLocale = ref.watch(actualLocaleProvider); final appRouter = ref.read(routerProvider); return MaterialApp.router( routeInformationProvider: appRouter.routeInformationProvider, routeInformationParser: appRouter.routeInformationParser, routerDelegate: appRouter.routerDelegate, debugShowCheckedModeBanner: false, title: 'Converter NOW', themeMode: themeMode, theme: lightTheme, darkTheme: (ref.watch(isPureDarkProvider).value ?? false) ? amoledTheme : darkTheme, supportedLocales: mapLocale.keys, localizationsDelegates: AppLocalizations.localizationsDelegates, locale: appLocale, ); }, ); }, ); } } ================================================ FILE: lib/models/conversions.dart ================================================ import 'package:converterpro/models/order.dart'; import 'package:converterpro/models/properties_list.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:units_converter/units_converter.dart'; class ConversionsNotifier extends AsyncNotifier>> { static final provider = AsyncNotifierProvider< ConversionsNotifier, Map> >(ConversionsNotifier.new); @override Future>> build() async { final conversionsOrder = (await ref.watch( UnitsOrderNotifier.provider.future, )); final propertiesMap = await ref.watch(propertiesMapProvider.future); return conversionsOrder.map( (propertyx, orderedUnits) => MapEntry( propertyx, orderedUnits .map( (e) => UnitData( propertiesMap[propertyx]!.getUnit(e), tec: TextEditingController(), property: propertyx, textInputType: switch (e) { TEMPERATURE.celsius || TEMPERATURE.fahrenheit || TEMPERATURE.delisle || TEMPERATURE.reamur || TEMPERATURE.romer => const TextInputType.numberWithOptions( decimal: true, signed: true, ), NUMERAL_SYSTEMS.binary || NUMERAL_SYSTEMS.octal || NUMERAL_SYSTEMS.decimal => const TextInputType.numberWithOptions( decimal: false, signed: false, ), NUMERAL_SYSTEMS.hexadecimal => TextInputType.text, _ => const TextInputType.numberWithOptions( decimal: true, signed: false, ), }, validator: switch (e) { TEMPERATURE.celsius || TEMPERATURE.fahrenheit || TEMPERATURE.delisle || TEMPERATURE.reamur || TEMPERATURE.romer => VALIDATOR.rational, NUMERAL_SYSTEMS.binary => VALIDATOR.binary, NUMERAL_SYSTEMS.octal => VALIDATOR.octal, NUMERAL_SYSTEMS.decimal => VALIDATOR.decimal, NUMERAL_SYSTEMS.hexadecimal => VALIDATOR.hexadecimal, _ => VALIDATOR.rationalNonNegative, }, ), ) .toList(), ), ); } /// The list of values that has been just cleared out List? _savedUnitDataList; /// Which [PROPERTYX] has been just cleared PROPERTYX? _savedProperty; UnitData? _selectedUnit; //unit where the user is writing the value /// This function get the value of the unit from currentProperty and update /// the currentUnitDataList values. It is used when a conversion changes the /// values of the units Future _refreshCurrentUnitDataList(PROPERTYX property) async { final currentUnitDataList = state.value![property]!; final propertiesMap = await ref.read(propertiesMapProvider.future); for (UnitData currentUnitData in currentUnitDataList) { final currentProperty = propertiesMap[property]!; currentUnitData.unit = currentProperty.getUnit(currentUnitData.unit.name); if (currentUnitData != _selectedUnit) { if (currentUnitData.unit.stringValue == null) { currentUnitData.tec.value = TextEditingValue.empty; } else { currentUnitData.tec.value = TextEditingValue( text: currentUnitData.unit.stringValue!, ); } } } } /// This function is used to convert all the values from one that has been /// modified Future convert(UnitData unitData, var value, PROPERTYX property) async { final propertiesMap = await ref.read(propertiesMapProvider.future); propertiesMap[property]!.convert(unitData.unit.name, value); _selectedUnit = unitData; await _refreshCurrentUnitDataList(property); } /// Returns a UnitDataList at a certain page with the current ordering /// (usefult with reorder units) List getUnitDataListAtPage(PROPERTYX property) => state.value![property]!; ///Clears the values of the current page Future clearAllValues(PROPERTYX property) async { List currentUnitDataList = state.value![property]!; if (currentUnitDataList[0].property == PROPERTYX.numeralSystems) { _savedUnitDataList = [ ...currentUnitDataList.map((unitData) => unitData.unit.stringValue), ]; } else { _savedUnitDataList = [ ...currentUnitDataList.map((unitData) => unitData.unit.value), ]; } _savedProperty = property; await convert(currentUnitDataList[0], null, property); // convert doesn't clear a selected textfield currentUnitDataList[0].tec.value = TextEditingValue.empty; } /// Undo the last clear all operation performed Future undoClearOperation() async { if (_savedUnitDataList != null && _savedProperty != null) { List listToUndo = state.value![_savedProperty!]!; if (_savedUnitDataList![0] is double) { for (int i = 0; i < listToUndo.length; i++) { listToUndo[i] ..unit.value = _savedUnitDataList![i] ..tec.value = TextEditingValue( text: _savedUnitDataList![i].toString(), ); } } else if (_savedUnitDataList![0] is String) { for (int i = 0; i < listToUndo.length; i++) { listToUndo[i] ..unit.stringValue = _savedUnitDataList![i] ..tec.value = TextEditingValue(text: _savedUnitDataList![i]); } } _savedUnitDataList = _savedProperty = null; } } /// Returns true if we should show a snackbar when the user press on the clear /// all button (see [undoClearOperation]), false otherwise. bool shouldShowSnackbar(PROPERTYX property) => state.value![property]![0].tec.value.text != ''; } ================================================ FILE: lib/models/currencies.dart ================================================ import 'dart:convert'; import 'package:converterpro/models/settings.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:intl/intl.dart'; import 'package:http/http.dart' as http; import 'package:shared_preferences/shared_preferences.dart'; class Currencies { static const defaultExchangeRates = { 'EUR': 1.0, 'AUD': 1.6514, 'BRL': 6.0127, 'CAD': 1.4856, 'CHF': 0.9442, 'CNY': 7.6141, 'CZK': 25.043, 'DKK': 7.459, 'GBP': 0.83215, 'HKD': 8.1554, 'HUF': 402.95, 'IDR': 16980.33, 'ILS': 3.7341, 'INR': 90.81, 'ISK': 147.3, 'JPY': 160.09, 'KRW': 1509.5, 'MXN': 21.3146, 'MYR': 4.647, 'NOK': 11.6515, 'NZD': 1.8352, 'PHP': 60.487, 'PLN': 4.1653, 'RON': 4.977, 'SEK': 11.2445, 'SGD': 1.4052, 'THB': 35.238, 'TRY': 37.9486, 'USD': 1.0478, 'ZAR': 19.2555, }; /// The conversion rates with respect to EUR Map exchangeRates; /// The date of the last update encoded as 'yyyy-mm-dd' String lastUpdate; Currencies({ this.exchangeRates = defaultExchangeRates, this.lastUpdate = '2025-02-15', }); /// Transform the exchangeRates map into a json that can be stored String toJson() => jsonEncode(exchangeRates); /// It transforms a previous stored data (with the toJson method) into this /// object factory Currencies.fromJson(String jsonString) { var exchangeRates = Map.from(defaultExchangeRates); Map jsonData = json.decode(jsonString); for (String key in jsonData.keys) { exchangeRates[key] = jsonData[key]; } return Currencies(exchangeRates: exchangeRates); } Currencies copyWith({ Map? exchangeRates, String? lastUpdate, }) { return Currencies( exchangeRates: exchangeRates ?? this.exchangeRates, lastUpdate: lastUpdate ?? this.lastUpdate, ); } } class CurrenciesNotifier extends AsyncNotifier { static final provider = AsyncNotifierProvider( CurrenciesNotifier.new, ); late SharedPreferencesWithCache pref; @override Future build() async { pref = await ref.read(sharedPref.future); final String now = DateFormat("yyyy-MM-dd").format(DateTime.now()); // Let's search before if we already have downloaded the exchange rates String? lastUpdate = pref.getString("lastUpdateCurrencies"); // if I have never updated the conversions or if I have updated before today // I have to update if (!(ref.read(revokeInternetProvider).value ?? false) && (lastUpdate == null || lastUpdate != now)) { return _downloadCurrencies(); } // If I already have the data of today I just use it, no need of read them // from the web return _readSavedCurrencies(); } void forceCurrenciesDownload() async { state = AsyncData(await _downloadCurrencies()); } Currencies _readSavedCurrencies() { String? lastUpdate = pref.getString('lastUpdateCurrencies'); String? currenciesRead = pref.getString('currenciesRates'); if (currenciesRead != null) { return Currencies.fromJson( currenciesRead, ).copyWith(lastUpdate: lastUpdate); } return Currencies(); } /// Updates the currencies exchange rates with the latest values. It will also /// update the status at the end (updated or error) Future _downloadCurrencies() async { final stringRequest = Currencies.defaultExchangeRates.keys .where((e) => e != 'EUR') .join('+'); try { var response = await http.get( Uri.https( 'data-api.ecb.europa.eu', 'service/data/EXR/D.$stringRequest.EUR.SP00.A', {'lastNObservations': '1', 'detail': 'dataonly', 'format': 'csvdata'}, ), ); // if successful if (response.statusCode == 200) { var lastUpdate = DateFormat("yyyy-MM-dd").format(DateTime.now()); Map exchangeRates = {'EUR': 1}; final rows = const LineSplitter().convert(response.body); final tableHeader = rows[0].split(','); final valueIndex = tableHeader.indexOf('OBS_VALUE'); final currencyIndex = tableHeader.indexOf('CURRENCY'); rows.removeAt(0); for (var row in rows) { final elements = row.split(','); final currency = elements[currencyIndex]; final value = double.parse(elements[valueIndex]); exchangeRates[currency] = value; } pref.setString('currenciesRates', jsonEncode(exchangeRates)); pref.setString('lastUpdateCurrencies', lastUpdate); return Currencies(exchangeRates: exchangeRates, lastUpdate: lastUpdate); } } catch (e) { dPrint(e.toString); } return _readSavedCurrencies(); } } ================================================ FILE: lib/models/hide_units.dart ================================================ import 'package:collection/collection.dart'; import 'package:converterpro/data/default_order.dart'; import 'package:converterpro/models/settings.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; class HiddenUnitsNotifier extends AsyncNotifier> { static final provider = AsyncNotifierProvider>( HiddenUnitsNotifier.new, ); @override Future> build() async { final prefs = await ref.read(sharedPref.future); final Map newState = {}; for (final property in defaultPropertiesOrder) { final storedList = prefs.getStringList(storeKey(property)); final allUnits = defaultUnitsOrder[property]!; if (storedList == null) { newState[property] = []; // Default to no hidden units } else { final storedOrder = storedList .map( (storedString) => allUnits.firstWhereOrNull( (unit) => storedString == unit.toString(), ), ) .nonNulls .cast() .toList(); newState[property] = storedOrder; } } return newState; } bool set(List hiddenUnits, PROPERTYX property) { final allUnits = defaultUnitsOrder[property]; if (allUnits == null) return false; // Check if hiddenUnits are valid for this property final allUnitsStrings = allUnits.map((e) => e.toString()).toSet(); final hiddenUnitsStrings = hiddenUnits.map((e) => e.toString()).toSet(); if (!allUnitsStrings.containsAll(hiddenUnitsStrings)) { return false; } // Update the state final newState = {...state.value!}; newState[property] = hiddenUnits; state = AsyncData(newState); // Store the new values ref.read(sharedPref.future).then((prefs) { if (hiddenUnits.isEmpty) { // if there aren't hidden units (all visible), just delete the // corresponding value from storage prefs.remove(storeKey(property)); } else { prefs.setStringList( storeKey(property), hiddenUnits.map((e) => e.toString()).toList(), ); } }); return true; } String storeKey(PROPERTYX property) => 'hiddenUnits_${property.toString().substring('PROPERTYX.'.length)}'; } ================================================ FILE: lib/models/import_export.dart ================================================ import 'dart:convert'; import 'package:collection/collection.dart'; import 'package:converterpro/data/default_order.dart'; import 'package:converterpro/models/currencies.dart'; import 'package:converterpro/models/hide_units.dart'; import 'package:converterpro/models/order.dart'; import 'package:converterpro/models/settings.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; class ImportExportNotifier extends Notifier { static final provider = NotifierProvider( ImportExportNotifier.new, ); // List of simple settings providers to iterate over. // Using dynamic to allow mixed generic types. static final List _simpleSettingsProviders = [ significantFiguresProvider, removeTrailingZerosProvider, isPureDarkProvider, propertySelectionOnStartupProvider, revokeInternetProvider, useDeviceColorProvider, colorThemeProvider, themeModeProvider, languageTagProvider, ]; @override void build() {} /// Exports all settings to a JSON string Future exportSettings() async { final Map exportData = {}; // 1. Simple settings for (final provider in _simpleSettingsProviders) { final notifier = ref.read(provider.notifier); final key = notifier.prefKey as String; final value = await ref.read(provider.future); if (value != notifier.defaultValue) { exportData[key] = value; } } // 2. Properties Order final propertiesOrder = await ref.read( PropertiesOrderNotifier.provider.future, ); if (!listEquals(propertiesOrder, defaultPropertiesOrder)) { exportData[PropertiesOrderNotifier.storeKey] = propertiesOrder .map((e) => e.toString().substring('PROPERTYX.'.length)) .toList(); } // 3. Units Order & Hidden Units final unitsOrderMap = await ref.read(UnitsOrderNotifier.provider.future); final hiddenUnitsMap = await ref.read(HiddenUnitsNotifier.provider.future); for (final property in defaultPropertiesOrder) { // Units Order if (!listEquals(unitsOrderMap[property], defaultUnitsOrder[property])) { final key = ref .read(UnitsOrderNotifier.provider.notifier) .storeKey(property); exportData[key] = unitsOrderMap[property]! .map((e) => e.toString()) .toList(); } // Hidden Units if (hiddenUnitsMap[property]!.isNotEmpty) { final key = ref .read(HiddenUnitsNotifier.provider.notifier) .storeKey(property); exportData[key] = hiddenUnitsMap[property]! .map((e) => e.toString()) .toList(); } } return jsonEncode(exportData); } /// Imports settings from a JSON string /// Returns true if successful, false otherwise Future<(String?, List)> importSettings(String jsonString) async { String? importError; List keysError = []; try { final Map importData = jsonDecode(jsonString); // 1. Simple settings for (final provider in _simpleSettingsProviders) { final notifier = ref.read(provider.notifier); final key = notifier.prefKey as String; if (importData.containsKey(key)) { final value = importData[key]; try { if (!notifier.set(value)) { keysError.add(key); } } catch (e) { keysError.add(key); } } } // 2. Properties Order if (importData.containsKey(PropertiesOrderNotifier.storeKey)) { final List loadedList = List.from( importData[PropertiesOrderNotifier.storeKey], ); final currentOrder = await ref.read( PropertiesOrderNotifier.provider.future, ); if (currentOrder.length == loadedList.length) { final List newIndices = []; bool possible = true; for (var s in loadedList) { final index = currentOrder.indexWhere( (p) => p.toString().substring('PROPERTYX.'.length) == s, ); if (index == -1) { possible = false; break; } newIndices.add(index); } if (possible) { if (!ref .read(PropertiesOrderNotifier.provider.notifier) .set(newIndices)) { keysError.add(PropertiesOrderNotifier.storeKey); } } else { keysError.add(PropertiesOrderNotifier.storeKey); } } } // 3. Units Order & Hidden Units final unitsOrderMap = await ref.read(UnitsOrderNotifier.provider.future); for (final property in defaultPropertiesOrder) { // Hidden Units final hiddenUnitsKey = ref .read(HiddenUnitsNotifier.provider.notifier) .storeKey(property); if (importData.containsKey(hiddenUnitsKey)) { final List loadedList = List.from( importData[hiddenUnitsKey], ); final allUnits = defaultUnitsOrder[property]!; final List newHiddenList = loadedList .map((s) { return allUnits.firstWhereOrNull((u) => u.toString() == s); }) .nonNulls .toList(); if (!ref .read(HiddenUnitsNotifier.provider.notifier) .set(newHiddenList, property)) { keysError.add(hiddenUnitsKey); } } // Units Order final unitsOrderKey = ref .read(UnitsOrderNotifier.provider.notifier) .storeKey(property); if (importData.containsKey(unitsOrderKey)) { final List loadedList = List.from( importData[unitsOrderKey], ); final currentUnits = unitsOrderMap[property]; if (currentUnits != null && currentUnits.length == loadedList.length) { final List newIndices = []; bool possible = true; for (var s in loadedList) { final index = currentUnits.indexWhere((u) => u.toString() == s); if (index == -1) { possible = false; break; } newIndices.add(index); } if (possible) { if (!ref .read(UnitsOrderNotifier.provider.notifier) .set(newIndices, property)) { keysError.add(unitsOrderKey); } } else { keysError.add(unitsOrderKey); } } } } } catch (e) { importError = e.toString(); } return (importError, keysError); } void deleteSettings() { ref.read(sharedPref.future).then((pref) async { await pref.clear(); for (final provider in _simpleSettingsProviders) { ref.invalidate(provider); } ref.invalidate(PropertiesOrderNotifier.provider); ref.invalidate(UnitsOrderNotifier.provider); ref.invalidate(HiddenUnitsNotifier.provider); ref.invalidate(CurrenciesNotifier.provider); }); } } ================================================ FILE: lib/models/order.dart ================================================ import 'package:collection/collection.dart'; import 'package:converterpro/data/default_order.dart'; import 'package:converterpro/models/settings.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; class PropertiesOrderNotifier extends AsyncNotifier> { static final provider = AsyncNotifierProvider>( PropertiesOrderNotifier.new, ); static const storeKey = 'propertiesOrder'; @override Future> build() async { List? storedList = (await ref.read( sharedPref.future, )).getStringList(storeKey); if (storedList != null) { final newState = storedList .map( (storedString) => defaultPropertiesOrder.firstWhereOrNull( (property) => storedString == property.toString().substring('PROPERTYX.'.length), ), ) .nonNulls .toList(); // If there are different properties in the two lists if (newState.length != defaultPropertiesOrder.length) { newState.addAll( defaultPropertiesOrder.toSet().difference(newState.toSet()), ); (await ref.read( sharedPref.future, )).setStringList(storeKey, _toStorableString(newState)); } return newState; } return defaultPropertiesOrder; } bool set(List newOrder) { final currentOrdering = state.value; if (currentOrdering == null) { return false; } // Check if newOrder contains all numbers from 0 to length - 1 final Set expectedSet = List.generate( currentOrdering.length, (i) => i, ).toSet(); if (newOrder.length != currentOrdering.length || !newOrder.toSet().containsAll(expectedSet)) { return false; } final propertiesOrder = newOrder.map((e) => currentOrdering[e]).toList(); // Update the state state = AsyncData(propertiesOrder); // Store the new values ref.read(sharedPref.future).then((prefs) { // If it is the default order, do not save it if (listEquals(propertiesOrder, defaultPropertiesOrder)) { prefs.remove(storeKey); } else { prefs.setStringList(storeKey, _toStorableString(propertiesOrder)); } }); return true; } List _toStorableString(List listToConvert) => listToConvert .map((e) => e.toString().substring('PROPERTYX.'.length)) .toList(); } extension ReversedPropertiesOrdering on List { Map inverse() => Map.fromEntries(indexed.map((e) => MapEntry(e.$2, e.$1))); } class UnitsOrderNotifier extends AsyncNotifier> { static final provider = AsyncNotifierProvider>( UnitsOrderNotifier.new, ); @override Future> build() async { var prefs = await ref.read(sharedPref.future); final Map newState = {}; for (final property in defaultPropertiesOrder) { final storedList = prefs.getStringList(storeKey(property)); final defaultOrder = defaultUnitsOrder[property]!; if (storedList == null) { newState[property] = defaultOrder; } else { final storedOrder = storedList .map( (storedString) => defaultOrder.firstWhereOrNull( (unit) => storedString == unit.toString(), ), ) .nonNulls .cast() .toList(); // Add missing units if (storedOrder.length != defaultOrder.length) { storedOrder.addAll( defaultOrder.toSet().difference(storedOrder.toSet()), ); (await ref.read( sharedPref.future, )).setStringList(storeKey(property), _toStorableString(storedOrder)); } newState[property] = storedOrder; } } return newState; } bool set(List? newOrder, PROPERTYX property) { final currentState = state.value; if (newOrder == null || currentState == null) { return false; } final currentUnitsProperty = currentState[property]; if (currentUnitsProperty == null) return false; // Check if newOrder contains all numbers from 0 to length - 1 final Set expectedSet = List.generate( currentUnitsProperty.length, (i) => i, ).toSet(); if (newOrder.length != currentUnitsProperty.length || !newOrder.toSet().containsAll(expectedSet)) { return false; } final unitsOrder = newOrder.map((e) => currentUnitsProperty[e]).toList(); // Update the state final newState = {...currentState}; newState[property] = unitsOrder; state = AsyncData(newState); // Store the new values ref.read(sharedPref.future).then((prefs) { // If it is the default order, do not save it if (listEquals(unitsOrder, defaultUnitsOrder[property])) { prefs.remove(storeKey(property)); } else { prefs.setStringList(storeKey(property), _toStorableString(unitsOrder)); } }); return true; } String storeKey(PROPERTYX property) => 'unitsOrder_${property.toString().substring('PROPERTYX.'.length)}'; List _toStorableString(List listToConvert) => listToConvert.map((e) => e.toString()).toList(); } ================================================ FILE: lib/models/properties_list.dart ================================================ import 'package:converterpro/models/currencies.dart'; import 'package:converterpro/models/settings.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:units_converter/units_converter.dart'; const Map _currenciesSymbols = { 'EUR': '€ assets/flags/eu.png', 'CAD': '\$ assets/flags_opti/ca.svg.vec', 'HKD': 'HK\$ assets/flags_opti/hk.svg.vec', 'PHP': '₱ assets/flags_opti/ph.svg.vec', 'DKK': 'kr assets/flags_opti/dk.svg.vec', 'NZD': 'NZ\$ assets/flags/nz.png', 'CNY': '¥ assets/flags_opti/cn.svg.vec', 'AUD': 'A\$ assets/flags_opti/au.svg.vec', 'RON': 'L assets/flags_opti/ro.svg.vec', 'SEK': 'kr assets/flags_opti/se.svg.vec', 'IDR': 'Rp assets/flags_opti/id.svg.vec', 'INR': '₹ assets/flags/in.png', 'BRL': 'R\$ assets/flags/br.png', 'USD': '\$ assets/flags/us.png', 'ILS': '₪ assets/flags_opti/il.svg.vec', 'JPY': '¥ assets/flags_opti/jp.svg.vec', 'THB': '฿ assets/flags_opti/th.svg.vec', 'CHF': 'Fr. assets/flags_opti/ch.svg.vec', 'CZK': 'Kč assets/flags_opti/cz.svg.vec', 'MYR': 'RM assets/flags_opti/my.svg.vec', 'TRY': '₺ assets/flags_opti/tr.svg.vec', 'MXN': '\$ assets/flags/mx.png', 'NOK': 'kr assets/flags_opti/no.svg.vec', 'HUF': 'Ft assets/flags_opti/hu.svg.vec', 'ZAR': 'R assets/flags_opti/za.svg.vec', 'SGD': 'S\$ assets/flags_opti/sg.svg.vec', 'GBP': '£ assets/flags_opti/gb.svg.vec', 'KRW': '₩ assets/flags_opti/kr.svg.vec', 'PLN': 'zł assets/flags_opti/pl.svg.vec', 'ISK': 'kr assets/flags_opti/is.svg.vec', }; final propertiesMapProvider = FutureProvider>(( ref, ) async { final removeTrailingZeros = (await ref.watch( removeTrailingZerosProvider.future, ))!; final significantFigures = (await ref.watch( significantFiguresProvider.future, ))!; return { PROPERTYX.length: Length( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.length, ), PROPERTYX.area: Area( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.area, ), PROPERTYX.density: Density( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.density, ), PROPERTYX.volume: Volume( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.volume, ), PROPERTYX.currencies: SimpleCustomProperty( ref .watch(CurrenciesNotifier.provider) .when( data: (currencies) => currencies.exchangeRates, error: (_, trace) => Currencies.defaultExchangeRates, loading: () => Currencies.defaultExchangeRates, ), mapSymbols: _currenciesSymbols, significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.currencies, ), PROPERTYX.time: Time( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.time, ), PROPERTYX.temperature: Temperature( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.temperature, ), PROPERTYX.speed: Speed( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.speed, ), PROPERTYX.mass: Mass( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.mass, ), PROPERTYX.force: Force( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.force, ), PROPERTYX.fuelConsumption: FuelConsumption( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.fuelConsumption, ), PROPERTYX.numeralSystems: NumeralSystems(name: PROPERTYX.numeralSystems), PROPERTYX.pressure: Pressure( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.pressure, ), PROPERTYX.energy: Energy( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.energy, ), PROPERTYX.power: Power( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.power, ), PROPERTYX.angle: Angle( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.angle, ), PROPERTYX.shoeSize: ShoeSize( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.shoeSize, ), PROPERTYX.digitalData: DigitalData( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.digitalData, ), PROPERTYX.siPrefixes: SIPrefixes( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.siPrefixes, ), PROPERTYX.torque: Torque( significantFigures: significantFigures, removeTrailingZeros: removeTrailingZeros, name: PROPERTYX.torque, ), }; }); ================================================ FILE: lib/models/settings.dart ================================================ import 'dart:ui'; import 'package:converterpro/styles/consts.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_riverpod/legacy.dart'; import 'package:shared_preferences/shared_preferences.dart'; final Map mapLocale = { const Locale('en'): 'English', const Locale('bn'): 'বাংলা', const Locale('ca'): 'Català', const Locale('de'): 'Deutsch', const Locale('es'): 'Español', const Locale('fr'): 'Français', const Locale('el'): 'Ελληνικά', const Locale('hr'): 'Hrvatski', const Locale('id'): 'Bahasa Indonesia', const Locale('it'): 'Italiano', const Locale('ja'): '日本語', const Locale('nb'): 'Norsk', const Locale('nl'): 'Nederlands', const Locale('pl'): 'Polski', const Locale('pt'): 'Português', const Locale('ru'): 'Pусский', const Locale('tr'): 'Türkçe', const Locale('ar'): 'العربية', const Locale('zh'): '中文', const Locale('zh', 'TW'): '中文 (台灣)', }; final sharedPref = FutureProvider( (_) async => await SharedPreferencesWithCache.create( cacheOptions: const SharedPreferencesWithCacheOptions(), ), ); class SettingsNotifier extends AsyncNotifier { late final String prefKey; late final T? defaultValue; late final bool Function(T?) validate; // Returns true if the validation succeded, false otherwise bool set(T? value) { if (!validate(value)) { return false; } state = AsyncData(value); if (value == null || value == defaultValue) { ref.read(sharedPref.future).then((pref) => pref.remove(prefKey)); } else { ref.read(sharedPref.future).then((pref) { if (value is int) { pref.setInt(prefKey, value); } else if (value is bool) { pref.setBool(prefKey, value); } else if (value is String) { pref.setString(prefKey, value); } else if (value is double) { pref.setDouble(prefKey, value); } else { throw UnimplementedError('Type ${T.toString()} not supported'); } }); } return true; } @override Future build() async { var pref = await ref.watch(sharedPref.future); return pref.get(prefKey) as T? ?? defaultValue; } } final significantFiguresProvider = AsyncNotifierProvider, int?>(() { return SettingsNotifier() ..prefKey = 'significant_figures' ..defaultValue = 10 ..validate = (val) => val != null && val > 0 && val <= 16; }); final removeTrailingZerosProvider = AsyncNotifierProvider, bool?>(() { return SettingsNotifier() ..prefKey = 'remove_trailing_zeros' ..defaultValue = true ..validate = (val) => val != null; }); final isPureDarkProvider = AsyncNotifierProvider, bool?>(() { return SettingsNotifier() ..prefKey = 'isDarkAmoled' ..defaultValue = false ..validate = (val) => val != null; }); final propertySelectionOnStartupProvider = AsyncNotifierProvider, bool?>(() { return SettingsNotifier() ..prefKey = 'propertySelectionOnStartup' ..defaultValue = true ..validate = (val) => val != null; }); final revokeInternetProvider = AsyncNotifierProvider, bool?>(() { return SettingsNotifier() ..prefKey = 'revokeInternet' ..defaultValue = false ..validate = (val) => val != null; }); final useDeviceColorProvider = AsyncNotifierProvider, bool?>(() { return SettingsNotifier() ..prefKey = 'useDeviceColor' // Here we set default theme to fallbackColorTheme (it is easier to support // device that does not have a color accent) ..defaultValue = false ..validate = (val) => val != null; }); final colorThemeProvider = AsyncNotifierProvider, int?>( () { return SettingsNotifier() ..prefKey = 'colorTheme' ..defaultValue = fallbackColorTheme.toARGB32() ..validate = (val) => val != null && val > 0 && val <= 0xFFFFFFFF; }, ); /// `null` means no accent color final deviceAccentColorProvider = StateProvider((ref) => null); final actualColorThemeProvider = Provider((ref) { final useDeviceColor = ref.watch(useDeviceColorProvider).value ?? false; final colorTheme = ref.watch(colorThemeProvider).value; final deviceAccentColor = ref.watch(deviceAccentColorProvider); return useDeviceColor ? deviceAccentColor ?? fallbackColorTheme : colorTheme != null ? Color(colorTheme) : fallbackColorTheme; }); final themeModeProvider = AsyncNotifierProvider, int?>( () { return SettingsNotifier() ..prefKey = 'currentThemeMode' ..defaultValue = ThemeMode.system.index ..validate = (val) => val != null && val >= 0 && val < ThemeMode.values.length; }, ); final languageTagProvider = AsyncNotifierProvider, String?>(() { return SettingsNotifier() ..prefKey = 'locale' ..defaultValue = null ..validate = (val) => val == null || mapLocale.keys.any((e) => e.toLanguageTag() == val); }); final actualLocaleProvider = Provider((ref) { final languageTag = ref.watch(languageTagProvider).value; if (languageTag == null) { final deviceLocale = PlatformDispatcher.instance.locale; final isSupported = mapLocale.keys.any( (l) => l.languageCode == deviceLocale.languageCode, ); return isSupported ? deviceLocale : fallbackLocale; } return languageTagToLocale(languageTag); }); Locale languageTagToLocale(String languageTag) { return mapLocale.keys.firstWhere( (e) => e.toLanguageTag() == languageTag, orElse: () => fallbackLocale, ); } ================================================ FILE: lib/pages/app_scaffold.dart ================================================ import 'package:converterpro/app_router.dart'; import 'package:converterpro/helpers/responsive_helper.dart'; import 'package:calculator_widget/calculator_widget.dart'; import 'package:converterpro/models/conversions.dart'; import 'package:converterpro/models/order.dart'; import 'package:converterpro/pages/custom_drawer.dart'; import 'package:converterpro/pages/search_page.dart'; import 'package:converterpro/utils/navigator_utils.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:translations/app_localizations.dart'; import 'package:go_router/go_router.dart'; class AppScaffold extends ConsumerWidget { const AppScaffold({required this.child, super.key}); final Widget child; @override Widget build(BuildContext context, WidgetRef ref) { final l10n = AppLocalizations.of(context)!; void openCalculator() { showModalBottomSheet( context: context, isScrollControlled: true, builder: (BuildContext context) { return const CalculatorWidget(); }, ); } void clearAll(bool isDrawerFixed) { final currentProperty = kebabStringToPropertyX( GoRouterState.of( context, ).uri.toString().substring('/conversions/'.length), ); if (ref .read(ConversionsNotifier.provider.notifier) .shouldShowSnackbar(currentProperty)) { ref .read(ConversionsNotifier.provider.notifier) .clearAllValues(currentProperty); //Snackbar undo request final SnackBar snackBar = SnackBar( content: Text(l10n.undoClearAllMessage), behavior: SnackBarBehavior.floating, width: isDrawerFixed ? 400 : null, action: SnackBarAction( key: const ValueKey('undoClearAll'), label: l10n.undo, onPressed: () { ref .read(ConversionsNotifier.provider.notifier) .undoClearOperation(); }, ), ); ScaffoldMessenger.of(context).showSnackBar(snackBar); } } void openSearch() { ref.read(PropertiesOrderNotifier.provider).whenData((orderList) async { Future.delayed(const Duration(milliseconds: 300), () { ref.read(conversionPageHeroEnabledProvider.notifier).state = true; }); final selectedProperty = await showSearch( context: context, delegate: CustomSearchDelegate(orderList), ); if (selectedProperty != null) { Future.delayed(const Duration(milliseconds: 300), () { ref.read(conversionPageHeroEnabledProvider.notifier).state = false; }); final String targetPath = '/conversions/${selectedProperty.toKebabCase()}'; if (context.mounted && GoRouterState.of(context).uri.toString() != targetPath) { context.go(targetPath); } } else { return ref.read(conversionPageHeroEnabledProvider.notifier).state = false; } }); } return LayoutBuilder( builder: (context, constraints) { // ignore: no_leading_underscores_for_local_identifiers final bool _isDrawerFixed = isDrawerFixed(constraints.maxWidth); AppPage selectedSection = computeSelectedSection(context); Widget drawer = CustomDrawer( isDrawerFixed: _isDrawerFixed, openCalculator: openCalculator, openSearch: openSearch, ); final ret = _isDrawerFixed ? Scaffold( body: Row( children: [ drawer, Expanded(child: child), ], ), floatingActionButton: (selectedSection == AppPage.conversions && MediaQuery.viewInsetsOf(context).bottom == 0) ? FloatingActionButton( key: const ValueKey('clearAll'), onPressed: () => clearAll(_isDrawerFixed), tooltip: l10n.clearAll, child: const Icon(Icons.clear_outlined), ) : null, ) : Scaffold( drawer: drawer, body: child, floatingActionButtonLocation: FloatingActionButtonLocation.endContained, bottomNavigationBar: selectedSection == AppPage.conversions ? BottomAppBar( child: Row( mainAxisSize: MainAxisSize.max, children: [ IconButton( tooltip: l10n.search, icon: const Icon(Icons.search), onPressed: openSearch, ), IconButton( tooltip: l10n.calculator, icon: const Icon(Icons.calculate_outlined), onPressed: openCalculator, ), ], ), ) : null, floatingActionButton: (selectedSection == AppPage.conversions && MediaQuery.viewInsetsOf(context).bottom == 0) ? FloatingActionButton( key: const ValueKey('clearAll'), onPressed: () => clearAll(_isDrawerFixed), tooltip: l10n.clearAll, child: const Icon(Icons.clear_outlined), ) : null, ); return Shortcuts( shortcuts: { LogicalKeySet(LogicalKeyboardKey.control, LogicalKeyboardKey.keyK): const ActivateIntent(), // Workaround for linux platform. See github.com/flutter/flutter/issues/156806 LogicalKeySet( LogicalKeyboardKey.control, LogicalKeyboardKey.keyK, LogicalKeyboardKey.numLock, ): const ActivateIntent(), }, child: Actions( actions: >{ ActivateIntent: CallbackAction( onInvoke: (ActivateIntent intent) { openSearch(); return null; }, ), }, child: ret, ), ); }, ); } } ================================================ FILE: lib/pages/choose_property_page.dart ================================================ import 'package:converterpro/data/property_unit_maps.dart'; import 'package:converterpro/models/order.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:translations/app_localizations.dart'; class ChoosePropertyPage extends ConsumerWidget { const ChoosePropertyPage({ super.key, this.selectedProperty, this.onSelectedProperty, }); final PROPERTYX? selectedProperty; final void Function(PROPERTYX property)? onSelectedProperty; static const BorderRadius borderRadius = BorderRadius.all( Radius.circular(30), ); @override Widget build(BuildContext context, WidgetRef ref) { final propertiesOrder = ref.watch(PropertiesOrderNotifier.provider).value!; final propertyUiMap = getPropertyUiMap(context); final propertiesStringOrdered = propertiesOrder .map((e) => propertyUiMap[e]!.name) .toList(); Color selectedListTileColor = Theme.of(context).colorScheme.primaryContainer .withValues( alpha: Theme.brightnessOf(context) == Brightness.light ? 0.5 : 0.8, ); return CustomScrollView( slivers: [ SliverAppBar.large( title: Text(AppLocalizations.of(context)!.chooseProperty), ), SliverPadding( // Space for the navigation bar (android) padding: EdgeInsets.only( bottom: MediaQuery.paddingOf(context).bottom, ), sliver: SliverList( delegate: SliverChildBuilderDelegate( childCount: propertiesStringOrdered.length, (context, index) { final isSelectedProperty = propertiesOrder[index] == selectedProperty; return Stack( children: [ AnimatedSwitcher( duration: const Duration(milliseconds: 300), transitionBuilder: (Widget child, Animation animation) => SlideTransition( position: Tween( begin: const Offset(-1.0, 0.0), end: const Offset(0.0, 0.0), ).animate(animation), child: child, ), child: Padding( key: Key( '${propertiesStringOrdered[index]}-$isSelectedProperty', ), padding: const EdgeInsets.symmetric(horizontal: 15), child: Container( constraints: const BoxConstraints(maxWidth: 400), decoration: isSelectedProperty ? BoxDecoration( color: selectedListTileColor, borderRadius: borderRadius, ) : null, child: const ListTile(), ), ), ), Padding( padding: const EdgeInsets.symmetric(horizontal: 15), child: Container( constraints: const BoxConstraints(maxWidth: 400), child: ListTile( key: ValueKey( 'chooseProperty-${propertiesOrder[index]}', ), title: Text( propertiesStringOrdered[index], style: TextStyle( color: isSelectedProperty ? Theme.of( context, ).colorScheme.onPrimaryContainer : null, ), ), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), onTap: onSelectedProperty == null ? null : () => onSelectedProperty!(propertiesOrder[index]), ), ), ), ], ); }, ), ), ), ], ); } } ================================================ FILE: lib/pages/conversion_page.dart ================================================ import 'package:converterpro/app_router.dart'; import 'package:converterpro/helpers/responsive_helper.dart'; import 'package:converterpro/models/conversions.dart'; import 'package:converterpro/models/currencies.dart'; import 'package:converterpro/models/hide_units.dart'; import 'package:converterpro/utils/utils_widgets.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_svg/flutter_svg.dart'; import 'package:translations/app_localizations.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter/material.dart'; import 'package:converterpro/data/property_unit_maps.dart'; import 'package:intl/intl.dart'; import 'package:vector_graphics/vector_graphics.dart'; class ConversionPage extends ConsumerWidget { final PROPERTYX property; const ConversionPage(this.property, {super.key}); @override Widget build(BuildContext context, WidgetRef ref) { // if we remove the following check, if you enter the site directly to // '/conversions/:property' an error will occur if (!ref.watch(isEverythingLoadedProvider)) { return const SplashScreenWidget(); } final l10n = AppLocalizations.of(context)!; final heroEnabled = ref.watch(conversionPageHeroEnabledProvider); final unitDataList = ref .watch(ConversionsNotifier.provider) .value![property]!; final propertyUiMap = getPropertyUiMap(context); final unitMap = getUnitUiMap(context)[property]!; final hiddenUnits = ref .watch(HiddenUnitsNotifier.provider) .value![property]!; final hiddenUnitData = unitDataList.where( (e) => hiddenUnits.contains(e.unit.name), ); final unhiddenUnitData = unitDataList.where( (e) => !hiddenUnits.contains(e.unit.name), ); Widget? subtitleWidget; if (property == PROPERTYX.currencies) { Currencies? currencies = ref.watch(CurrenciesNotifier.provider).value; if (currencies == null) { subtitleWidget = const SizedBox( height: 30, child: Center( child: SizedBox( width: 25, height: 25, child: CircularProgressIndicator(), ), ), ); } else { subtitleWidget = Text( _getLastUpdateString(context, currencies.lastUpdate), style: Theme.of(context).textTheme.titleSmall, ); } } UnitWidget unitWidgetBuilder(UnitData unitData) => UnitWidget( tffKey: unitData.unit.name.toString(), unitName: unitMap[unitData.unit.name]!, unitSymbol: unitData.unit.symbol, symbolContainsIcon: unitData.property == PROPERTYX.currencies, keyboardType: unitData.textInputType, controller: unitData.tec, validator: (String? input) { if (input != null) { if (input != '' && !unitData.getValidator().hasMatch(input)) { return l10n.invalidCharacters; } } return null; }, onChanged: (String txt) { String newTxt = txt; bool changed = false; if (newTxt.contains(',')) { newTxt = newTxt.replaceAll(',', '.'); changed = true; } if (newTxt.startsWith('.')) { newTxt = '0$newTxt'; changed = true; } if (changed) { unitData.tec.value = TextEditingValue( text: newTxt, selection: TextSelection.collapsed(offset: newTxt.length), ); } if (txt == '' || unitData.getValidator().hasMatch(txt)) { var conversions = ref.read(ConversionsNotifier.provider.notifier); //just numeral system uses a string for conversion if (unitData.property == PROPERTYX.numeralSystems) { conversions.convert(unitData, txt == "" ? null : txt, property); } else { conversions.convert( unitData, txt == "" ? null : double.parse(txt), property, ); } } }, ); final unhiddenGridTiles = unhiddenUnitData.map(unitWidgetBuilder).toList(); final hiddenGridTiles = hiddenUnitData.map(unitWidgetBuilder).toList(); return LayoutBuilder( builder: (BuildContext context, BoxConstraints constraint) { final int numCols = responsiveNumCols(constraint.maxWidth); return CustomScrollView( slivers: [ SliverAppBar.large( title: Builder( builder: (context) { final isExpanded = (DefaultTextStyle.of(context).style.fontSize ?? 22.0) > 24.0; final iconWidget = SvgPicture( AssetBytesLoader(propertyUiMap[property]!.selectedIcon), width: 24, colorFilter: ColorFilter.mode( Theme.of(context).textTheme.titleLarge!.color!, BlendMode.srcIn, ), ); final textWidget = Material( type: MaterialType.transparency, child: Text( propertyUiMap[property]!.name, style: Theme.of(context).textTheme.headlineMedium, ), ); return Row( spacing: 12, children: [ isExpanded ? HeroMode( enabled: heroEnabled, child: Hero( tag: 'icon-${property.toString()}', child: iconWidget, ), ) : iconWidget, isExpanded ? HeroMode( enabled: heroEnabled, child: Hero( tag: 'text-${property.toString()}', child: textWidget, ), ) : textWidget, ], ); }, ), ), if (subtitleWidget != null) SliverToBoxAdapter( child: Padding( padding: const EdgeInsets.symmetric(horizontal: 15), child: Row( mainAxisAlignment: MainAxisAlignment.end, children: [subtitleWidget], ), ), ), SliverPadding( padding: const EdgeInsets.only(top: 10), sliver: SliverGrid.count( crossAxisCount: numCols, childAspectRatio: responsiveChildAspectRatio( constraint.maxWidth, numCols, ), children: unhiddenGridTiles, ), ), if (hiddenUnitData.isNotEmpty) SliverToBoxAdapter( child: ExpansionTile( leading: const Icon(Icons.visibility_off_outlined), title: Text( l10n.hiddenUnits, style: Theme.of(context).textTheme.titleLarge, ), children: [ GridView.count( shrinkWrap: true, physics: const NeverScrollableScrollPhysics(), crossAxisCount: numCols, childAspectRatio: responsiveChildAspectRatio( constraint.maxWidth, numCols, ), children: hiddenGridTiles, ), ], ), ), if (isDrawerFixed(MediaQuery.sizeOf(context).width)) // Space for FAB + navigation bar (android) SliverToBoxAdapter( child: SizedBox( height: 60 + MediaQuery.paddingOf(context).bottom, ), ), ], ); }, ); } } String _getLastUpdateString(BuildContext context, String lastUpdate) { final l10n = AppLocalizations.of(context)!; DateTime lastUpdateCurrencies = DateTime.parse(lastUpdate); DateTime dateNow = DateTime.now(); if (lastUpdateCurrencies.day == dateNow.day && lastUpdateCurrencies.month == dateNow.month && lastUpdateCurrencies.year == dateNow.year) { return l10n.lastCurrenciesUpdate + l10n.today; } return l10n.lastCurrenciesUpdate + DateFormat.yMd( Localizations.localeOf(context).languageCode, ).format(lastUpdateCurrencies); } ================================================ FILE: lib/pages/custom_drawer.dart ================================================ import 'package:converterpro/models/order.dart'; import 'package:converterpro/utils/navigator_utils.dart'; import 'package:converterpro/data/property_unit_maps.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_svg/flutter_svg.dart'; import 'package:translations/app_localizations.dart'; import 'package:go_router/go_router.dart'; import 'package:vector_graphics/vector_graphics.dart'; class CustomDrawer extends ConsumerWidget { final bool isDrawerFixed; final void Function() openCalculator; final void Function() openSearch; const CustomDrawer({ required this.isDrawerFixed, required this.openCalculator, required this.openSearch, super.key, }); @override Widget build(BuildContext context, WidgetRef ref) { final l10n = AppLocalizations.of(context)!; List headerDrawer = []; Color iconColor = getIconColor(Theme.of(context)); const title = Padding( padding: EdgeInsets.symmetric(vertical: 30), child: Center( child: Row( mainAxisAlignment: MainAxisAlignment.spaceEvenly, children: [ SvgPicture( AssetBytesLoader('assets/app_icons_opti/logo.svg.vec'), width: 50, height: 50, ), Text( 'Converter NOW', // Fixed independently of the accessibility settings. Already as // large as possible textScaler: TextScaler.noScaling, maxLines: 1, style: TextStyle( fontFamily: 'JosefinSans', fontWeight: FontWeight.w300, fontSize: 29, ), ), ], ), ), ); headerDrawer.add( ExcludeSemantics( child: isDrawerFixed ? InkWell(onTap: () => context.go('/'), child: title) : title, ), ); if (isDrawerFixed) { final keyDecoration = BoxDecoration( border: Border.all(color: Theme.of(context).colorScheme.onSurface), borderRadius: BorderRadius.circular(6), ); const keyPadding = EdgeInsets.symmetric(horizontal: 4); headerDrawer.add( NavigationDrawerDestination( key: const ValueKey('drawerItem_search'), icon: Icon(Icons.search_outlined, color: iconColor), label: Row( mainAxisAlignment: MainAxisAlignment.end, children: [ Text(l10n.search), const SizedBox(width: 16), Container( decoration: keyDecoration, child: const Padding(padding: keyPadding, child: Text('Ctrl')), ), const Padding( padding: EdgeInsets.symmetric(horizontal: 2), child: Text('+'), ), Container( decoration: keyDecoration, child: const Padding(padding: keyPadding, child: Text('K')), ), ], ), ), ); headerDrawer.add( NavigationDrawerDestination( key: const ValueKey('drawerItem_calculator'), icon: Icon(Icons.calculate_outlined, color: iconColor), label: Text(l10n.calculator), ), ); } headerDrawer.add( NavigationDrawerDestination( key: const ValueKey('drawerItem_settings'), icon: Icon(Icons.settings_outlined, color: iconColor), selectedIcon: Icon(Icons.settings, color: iconColor), label: Text(l10n.settings), ), ); headerDrawer.add( const Padding( padding: EdgeInsets.symmetric(horizontal: 25.0), child: Divider(), ), ); List? propertiesOrdering = ref .watch(PropertiesOrderNotifier.provider) .value; if (propertiesOrdering == null) { return const SizedBox(); } final propertyUiMap = getPropertyUiMap(context); final propertiesDrawer = propertiesOrdering.map((e) { final propertyUi = propertyUiMap[e]!; return NavigationDrawerDestination( key: ValueKey('drawerItem_$e'), icon: SvgPicture( AssetBytesLoader(propertyUi.icon), width: 25, height: 25, colorFilter: ColorFilter.mode(iconColor, BlendMode.srcIn), ), selectedIcon: SvgPicture( AssetBytesLoader(propertyUi.selectedIcon), width: 25, height: 25, colorFilter: ColorFilter.mode(iconColor, BlendMode.srcIn), ), label: Text(propertyUi.name), ); }); // How many NavigationDrawerDestination elements are there in the drawer int headerElements = headerDrawer .whereType() .toList() .length; return NavigationDrawer( selectedIndex: pathToNavigationIndex( context, isDrawerFixed, propertiesOrdering.inverse(), ), onDestinationSelected: (int selectedPage) { if (selectedPage >= headerElements) { context.go( '/conversions/${propertiesOrdering[selectedPage - headerElements].toKebabCase()}', ); if (!isDrawerFixed) { Navigator.of(context).pop(); } } else if (headerElements == 3) { switch (selectedPage) { case 0: openSearch(); case 1: openCalculator(); case 2: if (!isDrawerFixed) { Navigator.of(context).pop(); } context.goNamed('settings'); } } else if (headerElements == 1) { if (!isDrawerFixed) { Navigator.of(context).pop(); } context.goNamed('settings'); } }, children: [...headerDrawer, ...propertiesDrawer], ); } } int pathToNavigationIndex( BuildContext context, bool isDrawerFixed, Map inversePropertiesOrdering, ) { final String location = GoRouterState.of(context).uri.toString(); // 3 elements in the header if (isDrawerFixed) { if (location.startsWith('/conversions/')) { return computeSelectedConversionPage( context, inversePropertiesOrdering, )! + 3; } else { return 2; // Settings } } // 1 element in the header else { if (location.startsWith('/conversions/')) { return computeSelectedConversionPage( context, inversePropertiesOrdering, )! + 1; } else { return 0; // Settings } } } ================================================ FILE: lib/pages/error_page.dart ================================================ import 'package:flutter/material.dart'; import 'package:go_router/go_router.dart'; import 'package:translations/app_localizations.dart'; class ErrorPage extends StatelessWidget { const ErrorPage({super.key}); @override Widget build(BuildContext context) { final l10n = AppLocalizations.of(context)!; return Scaffold( body: Center( child: Column( mainAxisAlignment: MainAxisAlignment.center, children: [ Text(l10n.routeError1, style: const TextStyle(fontSize: 30)), const SizedBox(height: 40), ElevatedButton.icon( icon: const Icon(Icons.home_outlined), label: Padding( padding: const EdgeInsets.all(8.0), child: Text( l10n.routeError2, style: const TextStyle(fontSize: 20), ), ), onPressed: () => context.go('/'), style: ButtonStyle( backgroundColor: WidgetStateProperty.all( Theme.of(context).primaryColor, ), ), ), ], ), ), ); } } ================================================ FILE: lib/pages/hide_units_page.dart ================================================ import 'package:converterpro/app_router.dart'; import 'package:converterpro/models/order.dart'; import 'package:converterpro/pages/choose_property_page.dart'; import 'package:converterpro/pages/select_units_page.dart'; import 'package:converterpro/pages/splash_screen.dart'; import 'package:converterpro/styles/consts.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:converterpro/utils/utils_widgets.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:go_router/go_router.dart'; class HideUnitsPage extends ConsumerWidget { /// The index of the property the user tap. null means not yet selected. final PROPERTYX? selectedProperty; /// If `isDrawerFixed=false` then this variable is used to transition from the /// "Choose property page" to the "Reorder units" page final bool isPropertySelected; const HideUnitsPage({ super.key, this.selectedProperty, this.isPropertySelected = false, }); @override Widget build(BuildContext context, WidgetRef ref) { // Read the order of the properties in the drawer final conversionsOrderDrawer = ref .watch(PropertiesOrderNotifier.provider) .value; if (conversionsOrderDrawer == null) { return const SplashScreen(); } Widget? hideUnitsPage; if (selectedProperty != null) { // if we remove the following check, if you enter the site directly to // '/conversions/:property' an error will occur if (!ref.watch(isEverythingLoadedProvider)) { return const SplashScreenWidget(); } hideUnitsPage = SelectUnitsPage(selectedProperty: selectedProperty!); } final choosePropertyPage = ChoosePropertyPage( selectedProperty: selectedProperty, onSelectedProperty: (property) => context.go('/settings/hide-units/${property.toKebabCase()}'), ); return LayoutBuilder( builder: (BuildContext context, BoxConstraints constraints) { // Enough space for two sided pages if (constraints.maxWidth > twoSidedReorderScreen) { return Row( children: [ Expanded(child: choosePropertyPage), if (selectedProperty != null) Expanded( child: AnimatedSwitcher( duration: const Duration(milliseconds: 300), transitionBuilder: (Widget child, Animation animation) { final offsetAnimation = Tween( begin: const Offset(1.0, 0.0), end: const Offset(0.0, 0.0), ).animate(animation); return SlideTransition( position: offsetAnimation, child: child, ); }, child: hideUnitsPage, ), ), ], ); } // One page at a time if (!isPropertySelected) { return choosePropertyPage; } return hideUnitsPage!; }, ); } } ================================================ FILE: lib/pages/initial_page.dart ================================================ import 'package:converterpro/app_router.dart'; import 'package:converterpro/models/order.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:converterpro/utils/utils_widgets.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_svg/svg.dart'; import 'package:go_router/go_router.dart'; import 'package:translations/app_localizations.dart'; import 'package:vector_graphics/vector_graphics.dart'; class InitialPage extends ConsumerWidget { const InitialPage({super.key}); @override Widget build(BuildContext context, WidgetRef ref) { return Scaffold( body: CustomScrollView( slivers: [ SliverAppBar.large( title: Row( spacing: 16, children: [ const SvgPicture( AssetBytesLoader('assets/app_icons_opti/logo.svg.vec'), width: 39, height: 39, ), Text(AppLocalizations.of(context)!.appName), ], ), ), SliverGrid.extent( maxCrossAxisExtent: 180, children: getPropertyGridTiles( (PROPERTYX e) { Future.delayed(const Duration(milliseconds: 300), () { ref.read(conversionPageHeroEnabledProvider.notifier).state = false; }); HapticFeedback.selectionClick(); context.go('/conversions/${e.toKebabCase()}'); }, context, ref.read(PropertiesOrderNotifier.provider).value!, ), ), ], ), ); } } ================================================ FILE: lib/pages/reorder_page.dart ================================================ import 'package:translations/app_localizations.dart'; import 'package:flutter/material.dart'; typedef Item = ({int id, String title}); class ReorderPage extends StatefulWidget { final String title; final List itemsList; final void Function(List? orderList) onSave; const ReorderPage({ required this.itemsList, required this.onSave, required this.title, super.key, }); @override State createState() => _ReorderPageState(); } class _ReorderPageState extends State { late List _itemsList; final GlobalKey _scaffoldKey = GlobalKey(); @override Widget build(BuildContext context) { _itemsList = widget.itemsList.indexed .map((e) => (id: e.$1, title: e.$2)) .toList(); return Scaffold( key: _scaffoldKey, floatingActionButton: FloatingActionButton( key: const ValueKey('confirm'), tooltip: AppLocalizations.of(context)!.save, onPressed: () { List orderList = []; bool hasSomethingchanged = false; for (int i = 0; i < _itemsList.length; i++) { int currentIndex = _itemsList[i].id; orderList.add(currentIndex); if (i != currentIndex) hasSomethingchanged = true; } // if some modification has been done returns them, otherwise it will // return null widget.onSave(hasSomethingchanged ? orderList : null); }, child: const Icon(Icons.check), ), body: StatefulBuilder( builder: (context, setState) => CustomScrollView( slivers: [ SliverAppBar.large(title: Text(widget.title)), SliverPadding( // Space for FAB + navigation bar (android) padding: EdgeInsets.only( bottom: 60 + MediaQuery.paddingOf(context).bottom, ), sliver: SliverReorderableList( onReorder: (int oldIndex, int newIndex) => setState(() => _updateItemsOrder(oldIndex, newIndex)), itemCount: widget.itemsList.length, itemBuilder: (context, index) => Stack( key: Key('reaorderableListItem_$index'), children: [ Padding( padding: const EdgeInsetsDirectional.only(start: 34), child: ListTile( key: ValueKey(_itemsList[index].id), title: Text(_itemsList[index].title), ), ), Positioned.directional( textDirection: Directionality.of(context), top: 0, bottom: 0, start: 16, child: Align( alignment: AlignmentDirectional.centerStart, child: ReorderableDragStartListener( index: index, child: const Icon(Icons.drag_handle), ), ), ), ], ), ), ), ], ), ), ); } void _updateItemsOrder(int oldIndex, int newIndex) { if (newIndex > oldIndex) { newIndex -= 1; } final item = _itemsList.removeAt(oldIndex); _itemsList.insert(newIndex, item); } } ================================================ FILE: lib/pages/reorder_properties_page.dart ================================================ import 'package:converterpro/models/order.dart'; import 'package:converterpro/data/default_order.dart'; import 'package:converterpro/pages/reorder_page.dart'; import 'package:converterpro/pages/splash_screen.dart'; import 'package:converterpro/data/property_unit_maps.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:translations/app_localizations.dart'; import 'package:go_router/go_router.dart'; class ReorderPropertiesPage extends ConsumerWidget { const ReorderPropertiesPage({super.key}); @override Widget build(BuildContext context, WidgetRef ref) { // Read the order of the properties in the drawer final conversionsOrderDrawer = ref .watch(PropertiesOrderNotifier.provider) .value; final propertyUiMap = getPropertyUiMap(context); if (conversionsOrderDrawer == null) { return const SplashScreen(); } final orderedProperties = conversionsOrderDrawer .map((e) => propertyUiMap[e]!) .toList(); return ReorderPage( title: AppLocalizations.of(context)!.reorderProperties, itemsList: orderedProperties.map((e) => e.name).toList(), onSave: (List? orderList) { if (orderList != null) { // Save the order ref.read(PropertiesOrderNotifier.provider.notifier).set(orderList); // Update the quick actions initializeQuickAction( conversionsOrderDrawer: ref .read(PropertiesOrderNotifier.provider) .value!, propertyUiMap: propertyUiMap, onActionSelection: (String shortcutType) { final int index = int.parse(shortcutType); context.go( '/conversions/${defaultPropertiesOrder[index].toKebabCase()}', ); }, ); } context.pop(); }, ); } } ================================================ FILE: lib/pages/reorder_units_page.dart ================================================ import 'package:converterpro/app_router.dart'; import 'package:converterpro/models/order.dart'; import 'package:converterpro/pages/choose_property_page.dart'; import 'package:converterpro/pages/reorder_page.dart'; import 'package:converterpro/pages/splash_screen.dart'; import 'package:converterpro/styles/consts.dart'; import 'package:converterpro/data/property_unit_maps.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:converterpro/utils/utils_widgets.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:translations/app_localizations.dart'; import 'package:go_router/go_router.dart'; class ReorderUnitsPage extends ConsumerWidget { /// The index of the property the user tap. null means not yet selected. final PROPERTYX? selectedProperty; /// If `isDrawerFixed=false` then this variable is used to transition from the /// "Choose property page" to the "Reorder units" page final bool isPropertySelected; const ReorderUnitsPage({ this.selectedProperty, this.isPropertySelected = false, super.key, }); @override Widget build(BuildContext context, WidgetRef ref) { // Read the order of the properties in the drawer final conversionsOrderDrawer = ref .watch(PropertiesOrderNotifier.provider) .value; if (conversionsOrderDrawer == null) { return const SplashScreen(); } final propertyUiMap = getPropertyUiMap(context); Widget? reorderPage; if (selectedProperty != null) { final unitUiMap = getUnitUiMap(context); final conversionOrderUnits = ref .watch(UnitsOrderNotifier.provider) .value![selectedProperty]!; // if we remove the following check, if you enter the site directly to // '/conversions/:property' an error will occur if (!ref.watch(isEverythingLoadedProvider)) { return const SplashScreenWidget(); } reorderPage = ReorderPage( key: Key(selectedProperty.toString()), itemsList: conversionOrderUnits .map((e) => unitUiMap[selectedProperty]![e]!) .toList(), onSave: (List? orderList) { ref .read(UnitsOrderNotifier.provider.notifier) .set(orderList, selectedProperty!); context.goNamed('reorder-units'); }, title: AppLocalizations.of( context, )!.reorderProperty(propertyUiMap[selectedProperty]!.name), ); } final choosePropertyPage = ChoosePropertyPage( selectedProperty: selectedProperty, onSelectedProperty: (property) => context.go('/settings/reorder-units/${property.toKebabCase()}'), ); return LayoutBuilder( builder: (BuildContext context, BoxConstraints constraints) { // Enough space for two sided pages if (constraints.maxWidth > twoSidedReorderScreen) { return Row( children: [ Expanded(child: choosePropertyPage), if (selectedProperty != null) Expanded( child: AnimatedSwitcher( duration: const Duration(milliseconds: 300), transitionBuilder: (Widget child, Animation animation) { final offsetAnimation = Tween( begin: const Offset(1.0, 0.0), end: const Offset(0.0, 0.0), ).animate(animation); return SlideTransition( position: offsetAnimation, child: child, ); }, child: reorderPage, ), ), ], ); } // One page at a time if (!isPropertySelected) { return choosePropertyPage; } return reorderPage!; }, ); } } ================================================ FILE: lib/pages/search_page.dart ================================================ import 'package:converterpro/data/property_unit_maps.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:converterpro/utils/utils_widgets.dart'; import 'package:flutter/material.dart'; import 'package:translations/app_localizations.dart'; class CustomSearchDelegate extends SearchDelegate { final List orderList; CustomSearchDelegate(this.orderList); @override Widget buildLeading(BuildContext context) { return IconButton( tooltip: AppLocalizations.of(context)!.back, icon: AnimatedIcon( icon: AnimatedIcons.menu_arrow, progress: transitionAnimation, ), onPressed: () { close(context, null); }, ); } @override Widget buildSuggestions(BuildContext context) { final List dataSearch = getSearchUnitsList((PROPERTYX result) { close(context, result); }, context); final List allConversions = getPropertyGridTiles( (PROPERTYX result) { close(context, result); }, context, orderList, ); final Iterable suggestions = dataSearch.where( (searchUnit) => searchUnit.unitName.toLowerCase().contains(query.toLowerCase()), ); //.toLowercase in order to be case insesitive return query.isNotEmpty ? SuggestionList(suggestions: suggestions.toList()) : GridView( gridDelegate: const SliverGridDelegateWithMaxCrossAxisExtent( maxCrossAxisExtent: 180.0, ), children: allConversions, ); } @override Widget buildResults(BuildContext context) { return Container(); } @override List buildActions(BuildContext context) { return [ if (query.isNotEmpty) IconButton( tooltip: AppLocalizations.of(context)!.clearAll, icon: const Icon(Icons.clear), onPressed: () { query = ''; showSuggestions(context); }, ), ]; } } /// This method will return a List of [SearchUnit], needed in order to display the tiles in the search List getSearchUnitsList( void Function(PROPERTYX) onTap, BuildContext context, ) { List searchUnitsList = []; final propertyUiMap = getPropertyUiMap(context); final unitUiMap = getUnitUiMap(context); for (final property in propertyUiMap.entries) { final propertyx = property.key; final propertyUi = property.value; final propertyImagePath = property.value.icon; // Add properties in search searchUnitsList.add( SearchUnit( iconAsset: propertyImagePath, unitName: propertyUi.name, onTap: () => onTap(property.key), ), ); // Add units in search searchUnitsList.addAll( unitUiMap[propertyx]!.values.map( (e) => SearchUnit( iconAsset: propertyImagePath, unitName: e, onTap: () => onTap(propertyx), ), ), ); } return searchUnitsList; } ================================================ FILE: lib/pages/select_units_page.dart ================================================ import 'package:converterpro/data/default_order.dart'; import 'package:converterpro/data/property_unit_maps.dart'; import 'package:converterpro/models/hide_units.dart'; import 'package:converterpro/models/order.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:go_router/go_router.dart'; import 'package:translations/app_localizations.dart'; class SelectUnitsPage extends ConsumerStatefulWidget { const SelectUnitsPage({super.key, required this.selectedProperty}); final PROPERTYX selectedProperty; @override ConsumerState createState() => _SelectUnitsPageState(); } class _SelectUnitsPageState extends ConsumerState { List unselectedUnits = []; @override void initState() { super.initState(); initProvider(); } @override void didUpdateWidget(covariant SelectUnitsPage oldWidget) { super.didUpdateWidget(oldWidget); initProvider(); } void initProvider() { unselectedUnits = ref .read(HiddenUnitsNotifier.provider) .value![widget.selectedProperty]!; } @override Widget build(BuildContext context) { final l10n = AppLocalizations.of(context)!; final unitsNames = getUnitUiMap(context)[widget.selectedProperty]!; final conversionOrderUnits = ref .watch(UnitsOrderNotifier.provider) .value![widget.selectedProperty]!; final areAllSelected = unselectedUnits.isEmpty; return Scaffold( floatingActionButton: FloatingActionButton( key: const ValueKey('confirm'), tooltip: l10n.save, child: const Icon(Icons.check), onPressed: () { ref .read(HiddenUnitsNotifier.provider.notifier) .set(unselectedUnits, widget.selectedProperty); context.goNamed('hide-units'); }, ), body: CustomScrollView( slivers: [ SliverAppBar.large( title: Text( l10n.visibleUnits( getPropertyUiMap(context)[widget.selectedProperty]!.name, ), ), actions: [ TextButton.icon( label: Text(areAllSelected ? l10n.unselectAll : l10n.selectAll), onPressed: () { setState(() { unselectedUnits = areAllSelected ? defaultUnitsOrder[widget.selectedProperty]!.toList( growable: true, ) : []; }); }, icon: Icon( areAllSelected ? Icons.check_box_outline_blank : Icons.check_box, ), ), ], ), SliverPadding( // Space for FAB + navigation bar (android) padding: EdgeInsets.only( bottom: 60 + MediaQuery.paddingOf(context).bottom, ), sliver: SliverList( delegate: SliverChildBuilderDelegate( childCount: conversionOrderUnits.length, (context, index) { final unitCodeName = conversionOrderUnits[index]; return CheckboxListTile( value: !unselectedUnits.contains(unitCodeName), controlAffinity: ListTileControlAffinity.leading, shape: const RoundedRectangleBorder( borderRadius: BorderRadius.all(Radius.circular(30)), ), onChanged: (selected) { if (selected == null) { return; } setState(() { if (selected) { unselectedUnits.remove(unitCodeName); } else { unselectedUnits.add(unitCodeName); } }); }, title: Text(unitsNames[unitCodeName]!), ); }, ), ), ), ], ), ); } } ================================================ FILE: lib/pages/settings_page.dart ================================================ import 'dart:convert'; import 'dart:io'; import 'package:converterpro/models/currencies.dart'; import 'package:converterpro/models/import_export.dart'; import 'package:converterpro/models/settings.dart'; import 'package:converterpro/utils/palette.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:converterpro/utils/utils_widgets.dart'; import 'package:file_picker/file_picker.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_svg/flutter_svg.dart'; import 'package:go_router/go_router.dart'; import 'package:intl/intl.dart'; import 'package:path_provider/path_provider.dart'; import 'package:share_plus/share_plus.dart'; import 'package:translations/app_localizations.dart'; import 'package:url_launcher/url_launcher.dart'; import 'package:vector_graphics/vector_graphics.dart'; class SettingsPage extends ConsumerWidget { const SettingsPage({super.key}); static const List significantFiguresList = [6, 8, 10, 12, 14]; static const BorderRadiusGeometry borderRadius = BorderRadius.all( Radius.circular(30), ); @override Widget build(BuildContext context, WidgetRef ref) { final l10n = AppLocalizations.of(context)!; final mapTheme = { ThemeMode.system.index: ( title: l10n.system, icon: Icons.brightness_auto_outlined, ), ThemeMode.dark.index: (title: l10n.dark, icon: Icons.dark_mode_outlined), ThemeMode.light.index: ( title: l10n.light, icon: Icons.light_mode_outlined, ), }; final languageTag = ref.watch(languageTagProvider).value; final iconColor = getIconColor(Theme.of(context)); final titlesStyle = Theme.of(context).textTheme.titleSmall?.copyWith( color: switch (Theme.brightnessOf(context)) { Brightness.light => Theme.of(context).primaryColor, Brightness.dark => HSLColor.fromColor( Theme.of(context).primaryColor, ).withLightness(0.7).toColor(), }, ); return PopScope( canPop: false, onPopInvokedWithResult: (didPop, result) { if (!didPop) { context.go('/'); } }, child: CustomScrollView( slivers: [ SliverAppBar.large(title: Text(l10n.settings)), SliverList( delegate: SliverChildListDelegate( [ Padding( padding: const EdgeInsetsDirectional.only(start: 16), child: Text(l10n.appearance, style: titlesStyle), ), DropdownListTile( key: const ValueKey('language'), leading: Icon(Icons.language, color: iconColor), title: l10n.language, items: [l10n.system, ...mapLocale.values], value: languageTag == null ? l10n.system : mapLocale[languageTagToLocale(languageTag)]!, onChanged: (String? string) { if (string == null) return; ref .read(languageTagProvider.notifier) .set( string == l10n.system ? null : mapLocale.keys .firstWhere( (element) => mapLocale[element] == string, ) .toLanguageTag(), ); }, ), ListTile( title: Text(l10n.themeColor), leading: Icon(Icons.palette_outlined, color: iconColor), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), trailing: Padding( padding: const EdgeInsets.symmetric(horizontal: 18), child: Container( width: 24, height: 24, decoration: BoxDecoration( borderRadius: BorderRadius.circular(24 / 2), color: ref.watch(actualColorThemeProvider), ), ), ), onTap: () => showDialog( context: context, builder: (context) => const ColorPickerDialog(), ), ), SegmentedButtonListTile( leading: Icon(Icons.contrast, color: iconColor), title: l10n.theme, items: mapTheme.values.toList(), value: mapTheme[ref.watch(themeModeProvider).value ?? 0]!.title, onChanged: (String? string) { if (string != null) { ref .read(themeModeProvider.notifier) .set( mapTheme.keys .where((key) => mapTheme[key]?.title == string) .single, ); } }, ), SwitchListTile( secondary: Icon(Icons.dark_mode_outlined, color: iconColor), title: Text(l10n.pureBlackTheme), value: ref.watch(isPureDarkProvider).value ?? false, onChanged: (bool val) { ref.read(isPureDarkProvider.notifier).set(val); }, shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), ), SwitchListTile( secondary: Icon(Icons.apps_rounded, color: iconColor), title: Text(l10n.propertySelectionOnStartup), subtitle: Text(l10n.propertySelectionOnStartupSubtitle), value: ref.watch(propertySelectionOnStartupProvider).value ?? true, onChanged: (bool val) { ref .read(propertySelectionOnStartupProvider.notifier) .set(val); }, shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), ), Padding( padding: const EdgeInsetsDirectional.only(start: 16, top: 16), child: Text(l10n.conversions, style: titlesStyle), ), if (!kIsWeb) SwitchListTile( secondary: Icon(Icons.public_off, color: iconColor), title: Text(l10n.revokeInternetAccess), value: ref.watch(revokeInternetProvider).value ?? false, onChanged: (bool val) { if (val) { showDialog( context: context, builder: (context) { return AlertDialog( title: Text(l10n.revokeInternetAccess), content: SizedBox( width: 500, child: Text( l10n.revokeInternetExplanation, style: Theme.of(context).textTheme.bodyLarge, ), ), actions: [ TextButton( onPressed: () { Navigator.of(context).pop(); // Introduce a tiny delay to let the user see the // switch to turn on Future.delayed( const Duration(milliseconds: 200), () => ref .read(revokeInternetProvider.notifier) .set(val), ); }, child: Text(l10n.ok), ), ], ); }, ); } else { ref.read(revokeInternetProvider.notifier).set(val); ref .read(CurrenciesNotifier.provider.notifier) .forceCurrenciesDownload(); } }, shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), ), SwitchListTile( secondary: SvgPicture( const AssetBytesLoader( 'assets/app_icons_opti/remove_trailing_zeros.svg.vec', ), width: 25, colorFilter: ColorFilter.mode(iconColor, BlendMode.srcIn), ), title: Text(l10n.removeTrailingZeros), value: ref.watch(removeTrailingZerosProvider).value ?? true, onChanged: (bool val) { ref.read(removeTrailingZerosProvider.notifier).set(val); }, shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), ), DropdownListTile( leading: SvgPicture( const AssetBytesLoader( 'assets/app_icons_opti/significant_figures.svg.vec', ), width: 25, colorFilter: ColorFilter.mode(iconColor, BlendMode.srcIn), ), title: l10n.significantFigures, items: significantFiguresList .map((e) => e.toString()) .toList(), value: (ref.watch(significantFiguresProvider).value ?? 10) .toString(), onChanged: (String? string) { if (string != null) { ref .read(significantFiguresProvider.notifier) .set(int.parse(string)); } }, ), ListTile( key: const ValueKey('reorder-properties'), leading: SvgPicture( const AssetBytesLoader( 'assets/app_icons_opti/reorder_properties.svg.vec', ), width: 25, colorFilter: ColorFilter.mode(iconColor, BlendMode.srcIn), ), title: Text(l10n.reorderProperties), onTap: () => context.goNamed('reorder-properties'), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), ), ListTile( key: const ValueKey('reorder-units'), leading: SvgPicture( const AssetBytesLoader( 'assets/app_icons_opti/reorder_units.svg.vec', ), width: 25, colorFilter: ColorFilter.mode(iconColor, BlendMode.srcIn), ), title: Text(l10n.reorderUnits), onTap: () => context.goNamed('reorder-units'), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), ), ListTile( key: const ValueKey('hide-units'), leading: Icon( Icons.visibility_off_outlined, color: iconColor, ), title: Text(l10n.hideUnits), onTap: () => context.goNamed('hide-units'), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), ), Padding( padding: const EdgeInsetsDirectional.only(start: 16, top: 16), child: Text(l10n.backupAndRestore, style: titlesStyle), ), ListTile( leading: Icon(Icons.file_upload_outlined, color: iconColor), title: Text(l10n.exportSettings), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), onTap: () async { final jsonBytes = utf8.encode( await ref .read(ImportExportNotifier.provider.notifier) .exportSettings(), ); final filename = '${DateFormat('yyyyMMdd').format(DateTime.now())}_converternow.json'; if (kIsWeb) { await FilePicker.platform.saveFile( dialogTitle: l10n.exportSettings, fileName: filename, bytes: jsonBytes, ); } else if (Platform.isAndroid || Platform.isIOS) { // Mobile: Share final file = File( '${(await getTemporaryDirectory()).path}/$filename', ); await file.writeAsBytes(jsonBytes); await SharePlus.instance.share( ShareParams( files: [XFile(file.path)], subject: 'Converter NOW Backup', ), ); await file.delete(); // Clean up } else { // Desktop: Save file final outputFile = await FilePicker.platform.saveFile( dialogTitle: l10n.exportSettings, fileName: filename, ); if (outputFile != null) { final file = File(outputFile); await file.writeAsBytes(jsonBytes); if (context.mounted) { ScaffoldMessenger.of(context).showSnackBar( SnackBar(content: Text(l10n.exportSuccess)), ); } } } }, ), ListTile( leading: Icon(Icons.file_download_outlined, color: iconColor), title: Text(l10n.importSettings), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), onTap: () async { try { FilePickerResult? result = await FilePicker.platform .pickFiles( type: FileType.custom, allowedExtensions: ['json'], ); if (result != null) { String content; if (kIsWeb) { final bytes = result.files.first.bytes; if (bytes == null) return; content = utf8.decode(bytes); } else { final path = result.files.single.path; if (path == null) return; content = await File(path).readAsString(); } final (importError, keysError) = await ref .read(ImportExportNotifier.provider.notifier) .importSettings(content); if (context.mounted) { if (importError == null && keysError.isEmpty) { ScaffoldMessenger.of(context).showSnackBar( SnackBar(content: Text(l10n.importSuccess)), ); } else { showDialog( context: context, builder: (context) { String errorString = '${l10n.problemImportFile}.\n'; if (keysError.isNotEmpty) { errorString += "${l10n.relatedSettings}:\n• ${keysError.join('\n• ')}\n"; } if (importError != null) { errorString += '${l10n.reason}:\n $importError'; } return AlertDialog( title: Text(l10n.importError), content: Text(errorString), actions: [ TextButton( child: Text(l10n.ok), onPressed: () => Navigator.of(context).pop(), ), ], ); }, ); } } } } catch (e) { dPrint(() => e.toString()); } }, ), ListTile( leading: Icon( Icons.delete_forever_outlined, color: iconColor, ), title: Text(l10n.clearSettings), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), onTap: () async { showDialog( context: context, builder: (context) => AlertDialog( title: Text(l10n.clearSettings), content: Text(l10n.confirmationClear), actions: [ TextButton( child: Text(l10n.cancel), onPressed: () => Navigator.of(context).pop(), ), TextButton( child: Text( l10n.clearAll, style: TextStyle( color: Theme.of(context).colorScheme.error, fontWeight: FontWeight.bold, ), ), onPressed: () { ref .read(ImportExportNotifier.provider.notifier) .deleteSettings(); Navigator.of(context).pop(); }, ), ], ), ); }, ), Padding( padding: const EdgeInsetsDirectional.only(start: 16, top: 16), child: Text(l10n.findOutMore, style: titlesStyle), ), ListTile( leading: Icon(Icons.computer, color: iconColor), title: Text(l10n.otherPlatforms), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), onTap: () => showDialog( context: context, builder: (BuildContext context) { return SimpleDialog( title: Text(l10n.otherPlatforms), children: [ if (!kIsWeb) ListTile( title: const Text('Web'), leading: const Icon(Icons.public_outlined), onTap: () => launchURL( Uri( scheme: 'https', host: 'converter-now.web.app', ), mode: LaunchMode.externalApplication, ), ), if (kIsWeb || Platform.isWindows || Platform.isLinux) ListTile( title: const Text('Android'), leading: const Icon(Icons.android_outlined), onTap: () => launchURL( Uri( scheme: 'https', host: 'play.google.com', path: '/store/apps/details', queryParameters: { 'id': 'com.ferrarid.converterpro', }, ), ), ), if (kIsWeb || Platform.isAndroid || Platform.isLinux) ListTile( title: const Text('Windows'), leading: const Icon(Icons.laptop), onTap: () => launchURL( Uri( scheme: 'https', host: 'apps.microsoft.com', path: '/detail/9p0q79hwjh72', ), mode: LaunchMode.externalApplication, ), ), ListTile( title: const Text('Linux (Flatpak)'), leading: const Icon(Icons.desktop_windows_outlined), onTap: () => launchURL( Uri( scheme: 'https', host: 'flathub.org', path: '/apps/details/io.github.ferraridamiano.ConverterNOW', ), mode: LaunchMode.externalApplication, ), ), ListTile( title: const Text('Linux (AppImage)'), leading: const Icon(Icons.desktop_windows_outlined), onTap: () => launchURL( Uri( scheme: 'https', host: 'github.com', path: '/ferraridamiano/ConverterNOW/releases/latest', ), mode: LaunchMode.externalApplication, ), ), ListTile( title: Text(l10n.sourceCode), leading: const Icon(Icons.code), onTap: () => launchURL( Uri( scheme: 'https', host: 'github.com', path: '/ferraridamiano/ConverterNOW', ), mode: LaunchMode.externalApplication, ), ), ], ); }, ), ), ListTile( leading: Icon(Icons.translate, color: iconColor), title: Text(l10n.contributeTranslating), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), onTap: () { launchURL( Uri( scheme: 'https', host: 'github.com', path: '/ferraridamiano/ConverterNOW/issues/2', ), mode: LaunchMode.externalApplication, ); }, ), if (!const bool.fromEnvironment( 'IS_PLAYSTORE', defaultValue: false, )) ListTile( leading: Icon(Icons.coffee_outlined, color: iconColor), title: Text(l10n.buyMeACoffee), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), onTap: () { showDialog( context: context, barrierDismissible: true, builder: (BuildContext context) { return AlertDialog( title: Text(l10n.buyMeACoffee), content: SizedBox( width: 500, child: Text( l10n.donationDialog, style: Theme.of(context).textTheme.bodyLarge, ), ), actions: [ TextButton( child: Text( l10n.buyMeACoffee, style: TextStyle( color: Theme.of( context, ).colorScheme.secondary, ), ), onPressed: () { Navigator.of(context).pop(); launchURL( Uri( scheme: 'https', host: 'paypal.me', path: '/DemApps', ), ); }, ), ], ); }, ); }, ), ListTile( leading: Icon(Icons.info_outline, color: iconColor), title: Text(l10n.about), shape: const RoundedRectangleBorder( borderRadius: borderRadius, ), onTap: () => context.goNamed('about'), ), // Space for the navigation bar (android) SizedBox(height: MediaQuery.paddingOf(context).bottom), ].map(ConstrainedContainer.new).toList(), ), ), ], ), ); } } class ColorPickerDialog extends ConsumerWidget { const ColorPickerDialog({super.key}); @override Widget build(BuildContext context, WidgetRef ref) { final l10n = AppLocalizations.of(context)!; final deviceAccentColor = ref.watch(deviceAccentColorProvider); final colorTheme = Color(ref.watch(colorThemeProvider).value!); final useDeviceColor = ref.watch(useDeviceColorProvider).value!; return AlertDialog( title: Text(l10n.themeColor), content: SizedBox( width: 300, child: Column( mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.start, children: [ if (deviceAccentColor != null) ...[ SwitchListTile( value: useDeviceColor, onChanged: (val) => ref.read(useDeviceColorProvider.notifier).set(val), title: Text(l10n.useDeviceColor), ), const SizedBox(height: 8), ], Text( !useDeviceColor ? l10n.pickColor : '', style: Theme.of(context).textTheme.titleMedium, ), const SizedBox(height: 4), Center( child: Palette( initial: colorTheme, enabled: !useDeviceColor, onSelected: (color) => ref.read(colorThemeProvider.notifier).set(color.toARGB32()), ), ), ], ), ), ); } } ================================================ FILE: lib/pages/splash_screen.dart ================================================ import 'package:collection/collection.dart'; import 'package:converterpro/app_router.dart'; import 'package:converterpro/models/order.dart'; import 'package:converterpro/data/default_order.dart'; import 'package:converterpro/data/property_unit_maps.dart'; import 'package:converterpro/models/settings.dart'; import 'package:converterpro/styles/consts.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:converterpro/utils/utils_widgets.dart'; import 'package:flutter/material.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:go_router/go_router.dart'; class SplashScreen extends ConsumerWidget { const SplashScreen({super.key}); @override Widget build(BuildContext context, WidgetRef ref) { if (ref.watch(isEverythingLoadedProvider)) { final conversionsOrderDrawer = ref .read(PropertiesOrderNotifier.provider) .value!; initializeQuickAction( conversionsOrderDrawer: conversionsOrderDrawer, propertyUiMap: getPropertyUiMap(context), onActionSelection: (String shortcut) { final selectedProperty = defaultPropertiesOrder.firstWhereOrNull( (e) => e.toString() == shortcut, ); if (selectedProperty != null) { context.go('/conversions/${selectedProperty.toKebabCase()}'); } }, ); WidgetsBinding.instance.addPostFrameCallback( (_) => GoRouter.of(context).go( MediaQuery.sizeOf(context).width > pixelFixedDrawer || !ref.read(propertySelectionOnStartupProvider).value! ? '/conversions/${conversionsOrderDrawer[0].toKebabCase()}' : '/conversions', ), ); } return const SplashScreenWidget(); } } ================================================ FILE: lib/styles/consts.dart ================================================ import 'package:flutter/material.dart'; /// trigger from classic drawer to drawer fixed open const double pixelFixedDrawer = 650; /// trigger from 1 column to 2 columns (just inner view). Should be greater /// than `pixelFixedDrawer` const double pixelWidth1Column = 700; /// trigger from 2 columns to 3 columns (just inner view). Should be greater /// than `pixelWidth1Column` const double pixelWidth2Columns = 1000; /// trigger from 3 columns to 4 columns (just inner view). Should be greater /// than `pixelWidth2Columns` const double pixelWidth3Columns = 1300; /// trigger from 1 page reorder screen to 2 pages reorder screen (just inner /// view). Should be greater than `pixelFixedDrawer` const double twoSidedReorderScreen = 700; const fallbackColorTheme = Colors.blue; const fallbackLocale = Locale('en'); ================================================ FILE: lib/utils/navigator_utils.dart ================================================ import 'package:converterpro/utils/utils.dart'; import 'package:flutter/material.dart'; import 'package:go_router/go_router.dart'; enum AppPage { conversions, settings, reorder, reorderDetails } AppPage computeSelectedSection(BuildContext context) { final String location = GoRouterState.of(context).uri.toString(); if (location.startsWith('/conversions/')) { return AppPage.conversions; } if (location.startsWith('/settings/reorder-properties')) { return AppPage.reorder; } if (location.startsWith('/settings/reorder-units/') && location.split('/')[3] != '') { return AppPage.reorderDetails; } if (location.startsWith('/settings/reorder-units')) { return AppPage.reorder; } if (location.startsWith('/settings')) { return AppPage.settings; } return AppPage.conversions; } int? computeSelectedConversionPage( BuildContext context, Map inversePropertiesOrdering, ) { final location = GoRouterState.of(context).uri.toString(); if (location.startsWith('/conversions')) { return inversePropertiesOrdering[kebabStringToPropertyX( location.split('/').last, )]; } return null; } ================================================ FILE: lib/utils/palette.dart ================================================ import 'package:converterpro/utils/utils.dart'; import 'package:flutter/material.dart'; class Palette extends StatefulWidget { const Palette({ super.key, required this.onSelected, required this.initial, this.enabled = true, }); final Function(Color color) onSelected; final Color initial; final bool enabled; @override State createState() => _PaletteState(); } class _PaletteState extends State { Color? hoveredColor; late Color selectedColor; static const double squareSize = 43; static const double checkSize = 24; @override void initState() { super.initState(); selectedColor = widget.initial; } @override Widget build(BuildContext context) { final selectedColorValue = color2Int(selectedColor); final palette = Wrap( spacing: 4, runSpacing: 4, children: [ Colors.red, Colors.pink, Colors.purple, Colors.deepPurple, Colors.indigo, Colors.blue, Colors.lightBlue, Colors.cyan, Colors.teal, Colors.green, Colors.lightGreen, Colors.lime, Colors.yellow, Colors.amber, Colors.orange, Colors.deepOrange, Colors.brown, Colors.blueGrey, ].map((e) { final isHovered = hoveredColor == e; return MouseRegion( onEnter: (_) { setState(() => hoveredColor = e); }, onExit: (_) { setState(() => hoveredColor = null); }, child: InkWell( onTap: () { setState(() => selectedColor = e); widget.onSelected(e); }, borderRadius: BorderRadius.circular(squareSize / 2), child: Stack( children: [ AnimatedContainer( duration: const Duration(milliseconds: 300), width: squareSize, height: squareSize, decoration: BoxDecoration( borderRadius: BorderRadius.circular( isHovered ? squareSize / 2 : squareSize / 4, ), color: Color.lerp( widget.enabled ? e : HSVColor.fromColor( e, ).withSaturation(0.02).toColor(), Colors.white, isHovered ? 0.5 : 0, ), ), ), if (widget.enabled && selectedColorValue == color2Int(e)) Positioned( top: (squareSize - checkSize) / 2, left: (squareSize - checkSize) / 2, child: Icon( Icons.check, size: checkSize, color: e.computeLuminance() < 0.5 ? Colors.white : Colors.black, ), ), ], ), ), ); }).toList(), ); return widget.enabled ? palette : AbsorbPointer(absorbing: true, child: palette); } } ================================================ FILE: lib/utils/utils.dart ================================================ // ignore_for_file: constant_identifier_names import 'dart:io'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:quick_actions/quick_actions.dart'; import 'package:shared_preferences/shared_preferences.dart'; import 'package:units_converter/models/unit.dart'; import 'package:url_launcher/url_launcher.dart'; Future launchURL( Uri url, { LaunchMode mode = LaunchMode.platformDefault, }) async { if (!await launchUrl(url, mode: mode)) throw 'Could not launch $url'; } ///Saves the key value with SharedPreferences void saveSettings(String key, dynamic value) async { SharedPreferences prefs = await SharedPreferences.getInstance(); if (value is bool) { prefs.setBool(key, value); } else if (value is int) { prefs.setInt(key, value); } else if (value is String) { prefs.setString(key, value); } else if (value is double) { prefs.setDouble(key, value); } else if (value is List) { prefs.setStringList(key, value); } } enum VALIDATOR { binary, decimal, octal, hexadecimal, rational, rationalNonNegative, } class UnitData { Unit unit; TextEditingController tec; TextInputType textInputType; VALIDATOR validator; PROPERTYX? property; UnitData( this.unit, { required this.tec, this.property, this.validator = VALIDATOR.rationalNonNegative, this.textInputType = const TextInputType.numberWithOptions( decimal: true, signed: false, ), }); RegExp getValidator() => switch (validator) { VALIDATOR.binary => RegExp(r'^[0-1]+$'), VALIDATOR.octal => RegExp(r'^[0-7]+$'), VALIDATOR.decimal => RegExp(r'^[0-9]+$'), VALIDATOR.hexadecimal => RegExp(r'^[0-9A-Fa-f]+$'), VALIDATOR.rational => RegExp(r'^([+-]?\d+)\.?(\d*)(e[+-]?\d+)?$'), _ => RegExp(r'^(\+?\d+)\.?(\d*)(e[+-]?\d+)?$'), }; } /// PROPERTYX stands for PROPERTY extended and want to extends the PROPERTY enum /// defined in units_converter package enum PROPERTYX { angle, area, density, currencies, digitalData, energy, force, fuelConsumption, length, mass, numeralSystems, power, pressure, shoeSize, siPrefixes, speed, temperature, time, torque, volume, } typedef PropertyUi = ({String name, String icon, String selectedIcon}); void initializeQuickAction({ required void Function(String index) onActionSelection, required List conversionsOrderDrawer, required Map propertyUiMap, }) { // If it is not on a mobile device, return, otherwise set the quick actions final bool isMobileDevice = !kIsWeb && (Platform.isIOS || Platform.isAndroid); if (!isMobileDevice) return; const QuickActions() ..initialize(onActionSelection) ..setShortcutItems( conversionsOrderDrawer .take(3) .map( (e) => ShortcutItem( type: e.toString(), localizedTitle: propertyUiMap[e]!.name, icon: 'launch_image', ), ) .toList(), ); } Color getIconColor(ThemeData theme) => theme.brightness == Brightness.light ? Colors.black45 : Colors.white70; int _floatToInt8(double x) => (x * 255.0).round() & 0xff; int color2Int(Color color) => _floatToInt8(color.a) << 24 | _floatToInt8(color.r) << 16 | _floatToInt8(color.g) << 8 | _floatToInt8(color.b) << 0; /// Converts PROPERTYX.digitalData to a kebab String like 'digital-data' extension KebabCaseExtension on PROPERTYX { String toKebabCase() => toString() .split('.') .last .replaceAllMapped( RegExp(r'([A-Z])'), (match) => '-${match[0]!.toLowerCase()}', ); } PROPERTYX kebabStringToPropertyX(String string) { final lowerCaseString = string.replaceAll('-', '').toLowerCase(); return PROPERTYX.values.firstWhere( (e) => e.toString().toLowerCase() == 'propertyx.$lowerCaseString', ); } @pragma('vm:prefer-inline') void dPrint(String Function() message) { if (kDebugMode) { debugPrint(message()); } } ================================================ FILE: lib/utils/utils_widgets.dart ================================================ import 'package:converterpro/data/property_unit_maps.dart'; import 'package:converterpro/utils/utils.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_svg/flutter_svg.dart'; import 'package:translations/app_localizations.dart'; import 'package:vector_graphics/vector_graphics.dart'; class UnitWidget extends StatefulWidget { final String tffKey; final TextInputType? keyboardType; final TextEditingController controller; final String? Function(String?)? validator; final String unitName; final String? unitSymbol; final bool symbolContainsIcon; final void Function(String) onChanged; const UnitWidget({ super.key, required this.tffKey, this.keyboardType, required this.controller, this.validator, required this.unitName, this.unitSymbol, required this.symbolContainsIcon, required this.onChanged, }); @override State createState() => _UnitWidgetState(); } class _UnitWidgetState extends State { FocusNode focusNode = FocusNode(); @override void dispose() { super.dispose(); focusNode.dispose(); } @override Widget build(BuildContext context) { focusNode.addListener(() => setState(() {})); return Padding( padding: const EdgeInsets.symmetric(vertical: 8, horizontal: 15), child: TextFormField( key: ValueKey(widget.tffKey), focusNode: focusNode, style: const TextStyle(fontSize: 16.0), keyboardType: widget.keyboardType, controller: widget.controller, autovalidateMode: AutovalidateMode.onUserInteraction, validator: widget.validator, decoration: InputDecoration( labelText: widget.unitName, suffixIcon: focusNode.hasFocus && widget.controller.text.isNotEmpty ? IconButton( iconSize: 20, icon: const Icon(Icons.copy), tooltip: AppLocalizations.of(context)?.copy, onPressed: () { Clipboard.setData( ClipboardData(text: widget.controller.text), ); HapticFeedback.heavyImpact(); }, ) : widget.unitSymbol == null ? null : Padding( padding: const EdgeInsetsDirectional.only(end: 10), child: widget.symbolContainsIcon ? () { final symbolSplitted = widget.unitSymbol!.split(' '); final iconPath = symbolSplitted.removeLast(); final symbol = symbolSplitted.join(' '); return Row( mainAxisSize: MainAxisSize.min, spacing: 8, children: [ Text(symbol), iconPath.endsWith('.svg.vec') ? SvgPicture( AssetBytesLoader(iconPath), height: 16, ) : Image.asset(iconPath, height: 16), ], ); }() : Text(widget.unitSymbol!), ), // Workaround to make suffixIcon always visible // See: https://stackoverflow.com/questions/58819979 suffixIconConstraints: const BoxConstraints( minWidth: 0, minHeight: 0, ), suffixStyle: TextStyle( color: Theme.brightnessOf(context) == Brightness.light ? Colors.black : Colors.white, ), border: const OutlineInputBorder(), focusedBorder: OutlineInputBorder( borderSide: BorderSide( color: Theme.of(context).colorScheme.primary, width: 2, ), ), floatingLabelStyle: TextStyle( fontSize: 20, color: focusNode.hasFocus ? Theme.of(context).colorScheme.secondary : null, ), ), onChanged: (text) { widget.onChanged(text); setState(() {}); }, ), ); } } class SearchUnit { String iconAsset; String unitName; GestureTapCallback onTap; SearchUnit({ required this.iconAsset, required this.unitName, required this.onTap, }); } class SearchUnitTile extends StatelessWidget { final SearchUnit searchUnit; const SearchUnitTile(this.searchUnit, {super.key}); @override Widget build(BuildContext context) { return ListTile( leading: SvgPicture( AssetBytesLoader(searchUnit.iconAsset), height: 26.0, colorFilter: ColorFilter.mode( getIconColor(Theme.of(context)), BlendMode.srcIn, ), ), title: Text(searchUnit.unitName), onTap: searchUnit.onTap, ); } } class SuggestionList extends StatelessWidget { final List suggestions; const SuggestionList({required this.suggestions, super.key}); @override Widget build(BuildContext context) { return ListView( children: [ for (int i = 0; i < suggestions.length; i++) SearchUnitTile(suggestions[i]), ], ); } } class PropertyGridTile extends StatelessWidget { final String iconAsset; final String footer; final GestureTapCallback onTap; final String heroTag; const PropertyGridTile({ required this.iconAsset, required this.footer, required this.onTap, required this.heroTag, super.key, }); @override Widget build(BuildContext context) { return InkWell( onTap: onTap, customBorder: RoundedRectangleBorder( borderRadius: BorderRadius.circular(30), ), child: Column( spacing: 12, children: [ Expanded( flex: 6, child: Align( alignment: Alignment.bottomCenter, child: Hero( tag: 'icon-$heroTag', child: SvgPicture( AssetBytesLoader(iconAsset), height: 55, colorFilter: ColorFilter.mode( Theme.of(context).colorScheme.secondary, BlendMode.srcIn, ), ), ), ), ), Expanded( flex: 4, child: Hero( tag: 'text-$heroTag', child: Text( footer, textAlign: TextAlign.center, style: const TextStyle(fontSize: 18, height: 1.1), maxLines: 2, overflow: TextOverflow.ellipsis, ), ), ), ], ), ); } } class DropdownListTile extends StatelessWidget { final String title; final List items; final String value; final ValueChanged onChanged; final Widget? leading; /// This widget will return a [ListTile] with a dialog on mobile device and a /// [ListTile] with a [DropdownMenu] for desktop device. const DropdownListTile({ required this.title, required this.items, required this.value, required this.onChanged, this.leading, ValueKey? key, }) : super(key: key); static const BorderRadiusGeometry borderRadius = BorderRadius.all( Radius.circular(30), ); @override Widget build(BuildContext context) { return switch (Theme.of(context).platform) { TargetPlatform.android || TargetPlatform.iOS || TargetPlatform.fuchsia => ListTile( leading: leading, title: Text(title), subtitle: Text(value), shape: const RoundedRectangleBorder(borderRadius: borderRadius), onTap: () async => onChanged( await showModalBottomRadioList( context: context, title: title, items: items, value: value, ), ), ), TargetPlatform.linux || TargetPlatform.windows || TargetPlatform.macOS => ListTile( leading: leading, title: Text(title), shape: const RoundedRectangleBorder(borderRadius: borderRadius), trailing: DropdownMenu( key: key != null ? ValueKey('${(key as ValueKey).value}-dropdown') : null, initialSelection: value, onSelected: onChanged, requestFocusOnTap: false, width: 170, inputDecorationTheme: const InputDecorationTheme( outlineBorder: BorderSide.none, ), dropdownMenuEntries: items.map((String item) { return DropdownMenuEntry( value: item, label: item.toString(), ); }).toList(), ), ), }; } } class SegmentedButtonListTile extends StatelessWidget { final String title; final List<({IconData icon, String title})> items; final String value; final ValueChanged onChanged; final Widget? leading; /// This widget will return a [ListTile] with a dialog on mobile device and a /// [ListTile] with a [DropdownMenu] for desktop device. const SegmentedButtonListTile({ required this.title, required this.items, required this.value, required this.onChanged, this.leading, ValueKey? key, }) : super(key: key); static const BorderRadiusGeometry borderRadius = BorderRadius.all( Radius.circular(30), ); @override Widget build(BuildContext context) { return LayoutBuilder( builder: (context, constraints) => constraints.maxWidth > 450 ? ListTile( leading: leading, title: Text(title), shape: const RoundedRectangleBorder(borderRadius: borderRadius), trailing: SegmentedButton( segments: items .map( (e) => ButtonSegment( icon: Icon(e.icon), value: e.title, tooltip: e.title, ), ) .toList(), selected: {value}, onSelectionChanged: (val) => onChanged(val.first), ), ) : ListTile( leading: leading, title: Text(title), subtitle: Text(value), shape: const RoundedRectangleBorder(borderRadius: borderRadius), onTap: () async => onChanged( await showModalBottomRadioList( context: context, title: title, items: items.map((e) => e.title).toList(), value: value, ), ), ), ); } } Future showModalBottomRadioList({ required BuildContext context, required String title, required List items, required String value, }) { return showModalBottomSheet( context: context, showDragHandle: true, builder: (context) { return ListView( shrinkWrap: true, children: [ Padding( padding: const EdgeInsets.symmetric(horizontal: 16), child: Text(title, style: Theme.of(context).textTheme.titleLarge), ), const SizedBox(height: 15), RadioGroup( groupValue: value, onChanged: (value) => Navigator.pop(context, value), child: Column( children: items .map((item) => RadioListTile(value: item, title: Text(item))) .toList(), ), ), ], ); }, ); } class SplashScreenWidget extends StatelessWidget { const SplashScreenWidget({super.key}); @override Widget build(BuildContext context) { return const Scaffold( body: Center( child: SvgPicture( AssetBytesLoader('assets/app_icons_opti/logo.svg.vec'), width: 160, ), ), ); } } /// This widget limits the size of the [child] (e.g. a ListTile) to [maxWidth] /// and centers the content class ConstrainedContainer extends StatelessWidget { final Widget child; final double maxWidth; const ConstrainedContainer(this.child, {this.maxWidth = 600, super.key}); @override Widget build(BuildContext context) { return Align( alignment: AlignmentDirectional.centerStart, child: Container( constraints: BoxConstraints(maxWidth: maxWidth), child: child, ), ); } } /// This method will return a List of [PropertyGridTile], needed in order to /// display the gridtiles in the search List getPropertyGridTiles( void Function(PROPERTYX) onTap, BuildContext context, List orderList, ) { final propertyUiMap = getPropertyUiMap(context); return orderList.indexed.map((e) { final propertyUi = propertyUiMap[e.$2]!; return PropertyGridTile( key: ValueKey('gridtile-${e.$1}'), heroTag: e.$2.toString(), iconAsset: propertyUi.icon, footer: propertyUi.name, onTap: () => onTap(e.$2), ); }).toList(); } ================================================ FILE: linux/.gitignore ================================================ flutter/ephemeral ================================================ FILE: linux/CMakeLists.txt ================================================ # Project-level configuration. cmake_minimum_required(VERSION 3.13) project(runner LANGUAGES CXX) # The name of the executable created for the application. Change this to change # the on-disk name of your application. set(BINARY_NAME "converternow") # The unique GTK application identifier for this application. See: # https://wiki.gnome.org/HowDoI/ChooseApplicationID set(APPLICATION_ID "io.github.ferraridamiano.ConverterNOW") # Explicitly opt in to modern CMake behaviors to avoid warnings with recent # versions of CMake. cmake_policy(SET CMP0063 NEW) # Load bundled libraries from the lib/ directory relative to the binary. set(CMAKE_INSTALL_RPATH "$ORIGIN/lib") # Root filesystem for cross-building. if(FLUTTER_TARGET_PLATFORM_SYSROOT) set(CMAKE_SYSROOT ${FLUTTER_TARGET_PLATFORM_SYSROOT}) set(CMAKE_FIND_ROOT_PATH ${CMAKE_SYSROOT}) set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER) set(CMAKE_FIND_ROOT_PATH_MODE_PACKAGE ONLY) set(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY) set(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY) endif() # Define build configuration options. if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) set(CMAKE_BUILD_TYPE "Debug" CACHE STRING "Flutter build mode" FORCE) set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS "Debug" "Profile" "Release") endif() # Compilation settings that should be applied to most targets. # # Be cautious about adding new options here, as plugins use this function by # default. In most cases, you should add new options to specific targets instead # of modifying this function. function(APPLY_STANDARD_SETTINGS TARGET) target_compile_features(${TARGET} PUBLIC cxx_std_14) target_compile_options(${TARGET} PRIVATE -Wall -Werror) target_compile_options(${TARGET} PRIVATE "$<$>:-O3>") target_compile_definitions(${TARGET} PRIVATE "$<$>:NDEBUG>") endfunction() # Flutter library and tool build rules. set(FLUTTER_MANAGED_DIR "${CMAKE_CURRENT_SOURCE_DIR}/flutter") add_subdirectory(${FLUTTER_MANAGED_DIR}) # System-level dependencies. find_package(PkgConfig REQUIRED) pkg_check_modules(GTK REQUIRED IMPORTED_TARGET gtk+-3.0) # Application build; see runner/CMakeLists.txt. add_subdirectory("runner") # Run the Flutter tool portions of the build. This must not be removed. add_dependencies(${BINARY_NAME} flutter_assemble) # Only the install-generated bundle's copy of the executable will launch # correctly, since the resources must in the right relative locations. To avoid # people trying to run the unbundled copy, put it in a subdirectory instead of # the default top-level location. set_target_properties(${BINARY_NAME} PROPERTIES RUNTIME_OUTPUT_DIRECTORY "${CMAKE_BINARY_DIR}/intermediates_do_not_run" ) # Generated plugin build rules, which manage building the plugins and adding # them to the application. include(flutter/generated_plugins.cmake) # === Installation === # By default, "installing" just makes a relocatable bundle in the build # directory. set(BUILD_BUNDLE_DIR "${PROJECT_BINARY_DIR}/bundle") if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) set(CMAKE_INSTALL_PREFIX "${BUILD_BUNDLE_DIR}" CACHE PATH "..." FORCE) endif() # Start with a clean build bundle directory every time. install(CODE " file(REMOVE_RECURSE \"${BUILD_BUNDLE_DIR}/\") " COMPONENT Runtime) set(INSTALL_BUNDLE_DATA_DIR "${CMAKE_INSTALL_PREFIX}/data") set(INSTALL_BUNDLE_LIB_DIR "${CMAKE_INSTALL_PREFIX}/lib") install(TARGETS ${BINARY_NAME} RUNTIME DESTINATION "${CMAKE_INSTALL_PREFIX}" COMPONENT Runtime) install(FILES "${FLUTTER_ICU_DATA_FILE}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" COMPONENT Runtime) install(FILES "${FLUTTER_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" COMPONENT Runtime) foreach(bundled_library ${PLUGIN_BUNDLED_LIBRARIES}) install(FILES "${bundled_library}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" COMPONENT Runtime) endforeach(bundled_library) # Copy the native assets provided by the build.dart from all packages. set(NATIVE_ASSETS_DIR "${PROJECT_BUILD_DIR}native_assets/linux/") install(DIRECTORY "${NATIVE_ASSETS_DIR}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" COMPONENT Runtime) # Fully re-copy the assets directory on each build to avoid having stale files # from a previous install. set(FLUTTER_ASSET_DIR_NAME "flutter_assets") install(CODE " file(REMOVE_RECURSE \"${INSTALL_BUNDLE_DATA_DIR}/${FLUTTER_ASSET_DIR_NAME}\") " COMPONENT Runtime) install(DIRECTORY "${PROJECT_BUILD_DIR}/${FLUTTER_ASSET_DIR_NAME}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" COMPONENT Runtime) # Install the AOT library on non-Debug builds only. if(NOT CMAKE_BUILD_TYPE MATCHES "Debug") install(FILES "${AOT_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" COMPONENT Runtime) endif() ================================================ FILE: linux/flutter/CMakeLists.txt ================================================ # This file controls Flutter-level build steps. It should not be edited. cmake_minimum_required(VERSION 3.10) set(EPHEMERAL_DIR "${CMAKE_CURRENT_SOURCE_DIR}/ephemeral") # Configuration provided via flutter tool. include(${EPHEMERAL_DIR}/generated_config.cmake) # TODO: Move the rest of this into files in ephemeral. See # https://github.com/flutter/flutter/issues/57146. # Serves the same purpose as list(TRANSFORM ... PREPEND ...), # which isn't available in 3.10. function(list_prepend LIST_NAME PREFIX) set(NEW_LIST "") foreach(element ${${LIST_NAME}}) list(APPEND NEW_LIST "${PREFIX}${element}") endforeach(element) set(${LIST_NAME} "${NEW_LIST}" PARENT_SCOPE) endfunction() # === Flutter Library === # System-level dependencies. find_package(PkgConfig REQUIRED) pkg_check_modules(GTK REQUIRED IMPORTED_TARGET gtk+-3.0) pkg_check_modules(GLIB REQUIRED IMPORTED_TARGET glib-2.0) pkg_check_modules(GIO REQUIRED IMPORTED_TARGET gio-2.0) set(FLUTTER_LIBRARY "${EPHEMERAL_DIR}/libflutter_linux_gtk.so") # Published to parent scope for install step. set(FLUTTER_LIBRARY ${FLUTTER_LIBRARY} PARENT_SCOPE) set(FLUTTER_ICU_DATA_FILE "${EPHEMERAL_DIR}/icudtl.dat" PARENT_SCOPE) set(PROJECT_BUILD_DIR "${PROJECT_DIR}/build/" PARENT_SCOPE) set(AOT_LIBRARY "${PROJECT_DIR}/build/lib/libapp.so" PARENT_SCOPE) list(APPEND FLUTTER_LIBRARY_HEADERS "fl_basic_message_channel.h" "fl_binary_codec.h" "fl_binary_messenger.h" "fl_dart_project.h" "fl_engine.h" "fl_json_message_codec.h" "fl_json_method_codec.h" "fl_message_codec.h" "fl_method_call.h" "fl_method_channel.h" "fl_method_codec.h" "fl_method_response.h" "fl_plugin_registrar.h" "fl_plugin_registry.h" "fl_standard_message_codec.h" "fl_standard_method_codec.h" "fl_string_codec.h" "fl_value.h" "fl_view.h" "flutter_linux.h" ) list_prepend(FLUTTER_LIBRARY_HEADERS "${EPHEMERAL_DIR}/flutter_linux/") add_library(flutter INTERFACE) target_include_directories(flutter INTERFACE "${EPHEMERAL_DIR}" ) target_link_libraries(flutter INTERFACE "${FLUTTER_LIBRARY}") target_link_libraries(flutter INTERFACE PkgConfig::GTK PkgConfig::GLIB PkgConfig::GIO ) add_dependencies(flutter flutter_assemble) # === Flutter tool backend === # _phony_ is a non-existent file to force this command to run every time, # since currently there's no way to get a full input/output list from the # flutter tool. add_custom_command( OUTPUT ${FLUTTER_LIBRARY} ${FLUTTER_LIBRARY_HEADERS} ${CMAKE_CURRENT_BINARY_DIR}/_phony_ COMMAND ${CMAKE_COMMAND} -E env ${FLUTTER_TOOL_ENVIRONMENT} "${FLUTTER_ROOT}/packages/flutter_tools/bin/tool_backend.sh" ${FLUTTER_TARGET_PLATFORM} ${CMAKE_BUILD_TYPE} VERBATIM ) add_custom_target(flutter_assemble DEPENDS "${FLUTTER_LIBRARY}" ${FLUTTER_LIBRARY_HEADERS} ) ================================================ FILE: linux/flutter/generated_plugin_registrant.cc ================================================ // // Generated file. Do not edit. // // clang-format off #include "generated_plugin_registrant.h" #include #include #include #include void fl_register_plugins(FlPluginRegistry* registry) { g_autoptr(FlPluginRegistrar) dynamic_color_registrar = fl_plugin_registry_get_registrar_for_plugin(registry, "DynamicColorPlugin"); dynamic_color_plugin_register_with_registrar(dynamic_color_registrar); g_autoptr(FlPluginRegistrar) handy_window_registrar = fl_plugin_registry_get_registrar_for_plugin(registry, "HandyWindowPlugin"); handy_window_plugin_register_with_registrar(handy_window_registrar); g_autoptr(FlPluginRegistrar) url_launcher_linux_registrar = fl_plugin_registry_get_registrar_for_plugin(registry, "UrlLauncherPlugin"); url_launcher_plugin_register_with_registrar(url_launcher_linux_registrar); g_autoptr(FlPluginRegistrar) window_size_registrar = fl_plugin_registry_get_registrar_for_plugin(registry, "WindowSizePlugin"); window_size_plugin_register_with_registrar(window_size_registrar); } ================================================ FILE: linux/flutter/generated_plugin_registrant.h ================================================ // // Generated file. Do not edit. // // clang-format off #ifndef GENERATED_PLUGIN_REGISTRANT_ #define GENERATED_PLUGIN_REGISTRANT_ #include // Registers Flutter plugins. void fl_register_plugins(FlPluginRegistry* registry); #endif // GENERATED_PLUGIN_REGISTRANT_ ================================================ FILE: linux/flutter/generated_plugins.cmake ================================================ # # Generated file, do not edit. # list(APPEND FLUTTER_PLUGIN_LIST dynamic_color handy_window url_launcher_linux window_size ) list(APPEND FLUTTER_FFI_PLUGIN_LIST ) set(PLUGIN_BUNDLED_LIBRARIES) foreach(plugin ${FLUTTER_PLUGIN_LIST}) add_subdirectory(flutter/ephemeral/.plugin_symlinks/${plugin}/linux plugins/${plugin}) target_link_libraries(${BINARY_NAME} PRIVATE ${plugin}_plugin) list(APPEND PLUGIN_BUNDLED_LIBRARIES $) list(APPEND PLUGIN_BUNDLED_LIBRARIES ${${plugin}_bundled_libraries}) endforeach(plugin) foreach(ffi_plugin ${FLUTTER_FFI_PLUGIN_LIST}) add_subdirectory(flutter/ephemeral/.plugin_symlinks/${ffi_plugin}/linux plugins/${ffi_plugin}) list(APPEND PLUGIN_BUNDLED_LIBRARIES ${${ffi_plugin}_bundled_libraries}) endforeach(ffi_plugin) ================================================ FILE: linux/io.github.ferraridamiano.ConverterNOW.desktop ================================================ [Desktop Entry] Name=Converter NOW Comment=A Unit and Currencies Converter. It is immediate, fast and easy to use! Exec=converternow Icon=io.github.ferraridamiano.ConverterNOW.svg Terminal=false Type=Application Categories=Utility; Keywords=Converter;Unit;Currencies;Tools; StartupNotify=false ================================================ FILE: linux/io.github.ferraridamiano.ConverterNOW.metainfo.xml ================================================ io.github.ferraridamiano.ConverterNOW Converter NOW Simple unit and currency converter https://github.com/ferraridamiano/ConverterNOW https://github.com/ferraridamiano/ConverterNOW https://github.com/ferraridamiano/ConverterNOW/issues https://www.paypal.me/DemApps https://github.com/ferraridamiano/ConverterNOW/wiki/Translations MIT GPL-3.0-or-later Damiano Ferrari pointing keyboard touch 360 #95ef53 #4e7c2b

Converter NOW is an effective unit and currencies converter

  • ✔️ Simplify the tedious conversion process between units of measurement in a few clicks
  • 🚀 It is fast and immediate: just start typing and immediately you have the real-time conversion with all the other units of measurement
  • 🖌️ It is customizable: the units can be reorganized according to your priorities and your use
  • 🔢 It integrates a calculator that let you do the calculations in every page
  • 💰 Currency exchange rates updated daily
  • 🎨 Dynamic theming based on your device settings
  • ⚫⚪ Choose your favorite theme: dark and white theme
  • 📱🖥️ Multiplatform: available for Android, Web, Linux and Windows
  • 💯 It is free, no ads, no data collection, no permissions (just Internet to update currency conversions) and first of all it is open source!

Converter NOW can convert 200+ units of measurement and 30+ currencies 🎉🎉

Here are some of the physical quantity that Converter NOW is able to convert:

  • Currencies: Dollars, Euro, Pound, Rupee, Yen, etc.
  • Length: meters, inches, miles, yards, light years, etc.
  • Area: square meters, hectares, acres, etc.
  • Volume: cubic meters, liters, gallons, pints, spoons, etc.
  • Time: seconds, hours, days, years, weeks, millennia, etc.
  • Temperature: Centigrade, Fahrenheit, Kelvin, etc.
  • Speed: meters per second, kilometers per hour, knots, etc.
  • Mass: grams, pounds, tons, atomic mass units, etc.
  • Force: Newton, dyne, pound-force, poundal, etc
  • Pressure: pascal, bar, atmosphere, psi, etc.
  • Energy: Joule, calories, kilowatt hours, etc.
  • Power: Watt, kilowatt, horsepower, etc.
  • Density: grams per cubic meter, etc.
  • Fuel consumption: Miles per Gallon, Kilometers per liter, etc.
  • Numeral systems: decimal, binary, hexadecimal etc.
  • Torque: Newton meter, pound-force foot, poundal meter, etc.
  • Digital data: Nibble, bit, byte, kibibit, kibibyte, etc.
  • Shoes size: UK, India, Europe, USA, Japan, etc.
  • Angles: degree, radians, minutes, etc.
  • SI prefix: kilo, mega, giga, tera, milli, micro, nano, etc.
io.github.ferraridamiano.ConverterNOW.desktop Simply start typing and it will convert to all the other units in real time https://raw.githubusercontent.com/ferraridamiano/ConverterNOW/master/promotional/screenshots/1.png A settings page that allows you to customize every aspect of the app https://raw.githubusercontent.com/ferraridamiano/ConverterNOW/master/promotional/screenshots/2.png The app features an integrated calculator https://raw.githubusercontent.com/ferraridamiano/ConverterNOW/master/promotional/screenshots/3.png The app features an integrated search bar https://raw.githubusercontent.com/ferraridamiano/ConverterNOW/master/promotional/screenshots/4.png
  • 📤 Import / export / delete the app settings
  • 🔢 Added several units of measurement
  • 🌍 Translation improvements. Thanks to @rehork, @solokot, @milotype, @FTno
  • 🐞 Bug fixes and general improvements! We've worked hard to give you an even better experience

Do you like Converter NOW? Leave a review and let us know what you think! ✔️

  • 🔢 View the result preview in the calculator (Thanks to Bisher Asil)
  • ⚙️ Choose the property to convert on startup (mobile only)
  • 🌍 Translation improvements

Do you like Converter NOW? Leave a review and let us know what you think! ✔️

  • 🎨 New logo! Thanks to Giovanni Spitti for the suggestions
  • 🖌️ Improved icons in the app
  • 🌍 Translation improvements: thanks to @solokot

Do you like Converter NOW? Leave a review and let us know what you think! ✔️

  • 🐞 Fixed a bug related to the "Hide units" feature

Do you like Converter NOW? Leave a review and let us know what you think! ✔️

  • ⚙️ New units of measure: added density and new pressure units. Thanks to @Kheirlb
  • 🌍 Translation improvements: added Bengali and updated existing translations. Thanks to @aazmii, @rehork, @solokot

Do you like Converter NOW? Leave a review and let us know what you think! ✔️

  • 🐞 Bug Fixes and General Improvements! We've worked hard to provide you with an even better experience

Do you like Converter NOW? Leave a review and let us know what you think! ✔️

  • ⚙️ New features: hide unused units, improved default unit sorting, more units, improved conversion speed
  • 🌍 Updated translations. Thanks to @albanobattistella, @plum7x, @solokot

This version may break your unit sorting. We've made improvements so that this will no longer happen. We apologize for the inconvenience.

Do you like Converter NOW? Leave a review and let us know what you think! ✔️

  • ⚙️ New features: use comma as decimal separator, paste number into the calculator, search with ctrl+K
  • 🌍 Updated translations. Thanks to @solokot, @makishart, @HectorAE

Do you like Converter NOW? Leave a review and let us know what you think! ✔️

  • 🎨 Graphic Enhancements for big screens
  • 🌍 Updated translations Thanks to @rehork, @MajoranaOedipus, @FTno

Do you like Converter NOW? Leave a review and let us know what you think! ✔️

  • 🎨 Custom Themes! Give your app a unique touch by choosing your favorite theme.
  • ⚙️ Optimized Settings Page! Now even simpler and more intuitive to use.
  • ✨ Graphic Enhancements! Small improvements that make the app even more beautiful and enjoyable to use.
  • 🌍 Improved Translations! Thanks to @rehork and @pereorga, the Polish and Catalan versions are more accurate than ever.

Do you like Converter NOW? Leave a review and let us know what you think! ✔️

  • You can now block the internet access to the app
  • Improved the license page
  • Improved the russian translation (@Atrafon)
  • Added catalan translation (@pereorga)

What do you think about Converter NOW? Rate the app! ✔️

  • UI improvements
  • Improved french translation (credits: @Vsnmrn)

What do you think about Converter NOW? Rate the app! ✔️

  • Material you improvements
  • The calculator has a new look
  • Added dutch, chinese and traditional chinese translations
  • Improvements to other translations
  • Icons tweaks

Thanks to @David-git-code, @syu-pf-ssy, @plum7x, @HarshMakwana27 for your contributions!

What do you think about Converter NOW? Rate the app! ✔️

  • Fixed bug with currencies page after v4 update
  • Improved polish translations

Introducing Converter NOW 4.0!

  • Finished the migration to Material You
  • Dynamic theming support on supported desktop environments
  • New icons
  • Translations updates
  • Unit symbols are now always visible

Thanks to @rehork for your suggestion!

  • Migration to Material You
  • Added window icon
  • Added Polish translation

Thanks to @rehork for your contributions!

================================================ FILE: linux/runner/CMakeLists.txt ================================================ cmake_minimum_required(VERSION 3.13) project(runner LANGUAGES CXX) # Define the application target. To change its name, change BINARY_NAME in the # top-level CMakeLists.txt, not the value here, or `flutter run` will no longer # work. # # Any new source files that you add to the application should be added here. add_executable(${BINARY_NAME} "main.cc" "my_application.cc" "${FLUTTER_MANAGED_DIR}/generated_plugin_registrant.cc" ) # Apply the standard set of build settings. This can be removed for applications # that need different build settings. apply_standard_settings(${BINARY_NAME}) # Add preprocessor definitions for the application ID. add_definitions(-DAPPLICATION_ID="${APPLICATION_ID}") # Add dependency libraries. Add any application-specific dependencies here. target_link_libraries(${BINARY_NAME} PRIVATE flutter) target_link_libraries(${BINARY_NAME} PRIVATE PkgConfig::GTK) target_include_directories(${BINARY_NAME} PRIVATE "${CMAKE_SOURCE_DIR}") ================================================ FILE: linux/runner/main.cc ================================================ #include "my_application.h" int main(int argc, char** argv) { g_autoptr(MyApplication) app = my_application_new(); return g_application_run(G_APPLICATION(app), argc, argv); } ================================================ FILE: linux/runner/my_application.cc ================================================ #include "my_application.h" #include #ifdef GDK_WINDOWING_X11 #include #endif #include "flutter/generated_plugin_registrant.h" struct _MyApplication { GtkApplication parent_instance; char** dart_entrypoint_arguments; }; G_DEFINE_TYPE(MyApplication, my_application, GTK_TYPE_APPLICATION) // Called when first Flutter frame received. static void first_frame_cb(MyApplication* self, FlView* view) { gtk_widget_show(gtk_widget_get_toplevel(GTK_WIDGET(view))); } // Implements GApplication::activate. static void my_application_activate(GApplication* application) { MyApplication* self = MY_APPLICATION(application); GtkWindow* window = GTK_WINDOW(gtk_application_window_new(GTK_APPLICATION(application))); // Use a header bar when running in GNOME as this is the common style used // by applications and is the setup most users will be using (e.g. Ubuntu // desktop). // If running on X and not using GNOME then just use a traditional title bar // in case the window manager does more exotic layout, e.g. tiling. // If running on Wayland assume the header bar will work (may need changing // if future cases occur). gboolean use_header_bar = TRUE; #ifdef GDK_WINDOWING_X11 GdkScreen* screen = gtk_window_get_screen(window); if (GDK_IS_X11_SCREEN(screen)) { const gchar* wm_name = gdk_x11_screen_get_window_manager_name(screen); if (g_strcmp0(wm_name, "GNOME Shell") != 0) { use_header_bar = FALSE; } } #endif if (use_header_bar) { GtkHeaderBar* header_bar = GTK_HEADER_BAR(gtk_header_bar_new()); gtk_widget_show(GTK_WIDGET(header_bar)); gtk_header_bar_set_title(header_bar, "Converter NOW"); gtk_header_bar_set_show_close_button(header_bar, TRUE); gtk_window_set_titlebar(window, GTK_WIDGET(header_bar)); } else { gtk_window_set_title(window, "Converter NOW"); } gtk_window_set_default_size(window, 1280, 720); g_autoptr(FlDartProject) project = fl_dart_project_new(); fl_dart_project_set_dart_entrypoint_arguments( project, self->dart_entrypoint_arguments); FlView* view = fl_view_new(project); GdkRGBA background_color; // Background defaults to black, override it here if necessary, e.g. #00000000 // for transparent. gdk_rgba_parse(&background_color, "#000000"); fl_view_set_background_color(view, &background_color); gtk_container_add(GTK_CONTAINER(window), GTK_WIDGET(view)); // Moved these 2 lines as requested by the `handy_window` package fl_register_plugins(FL_PLUGIN_REGISTRY(view)); gtk_widget_show(GTK_WIDGET(view)); // Show the window when Flutter renders. // Requires the view to be realized so we can start rendering. g_signal_connect_swapped(view, "first-frame", G_CALLBACK(first_frame_cb), self); gtk_widget_realize(GTK_WIDGET(view)); gtk_widget_grab_focus(GTK_WIDGET(view)); } // Implements GApplication::local_command_line. static gboolean my_application_local_command_line(GApplication* application, gchar*** arguments, int* exit_status) { MyApplication* self = MY_APPLICATION(application); // Strip out the first argument as it is the binary name. self->dart_entrypoint_arguments = g_strdupv(*arguments + 1); g_autoptr(GError) error = nullptr; if (!g_application_register(application, nullptr, &error)) { g_warning("Failed to register: %s", error->message); *exit_status = 1; return TRUE; } g_application_activate(application); *exit_status = 0; return TRUE; } // Implements GApplication::startup. static void my_application_startup(GApplication* application) { // MyApplication* self = MY_APPLICATION(object); // Perform any actions required at application startup. G_APPLICATION_CLASS(my_application_parent_class)->startup(application); } // Implements GApplication::shutdown. static void my_application_shutdown(GApplication* application) { // MyApplication* self = MY_APPLICATION(object); // Perform any actions required at application shutdown. G_APPLICATION_CLASS(my_application_parent_class)->shutdown(application); } // Implements GObject::dispose. static void my_application_dispose(GObject* object) { MyApplication* self = MY_APPLICATION(object); g_clear_pointer(&self->dart_entrypoint_arguments, g_strfreev); G_OBJECT_CLASS(my_application_parent_class)->dispose(object); } static void my_application_class_init(MyApplicationClass* klass) { G_APPLICATION_CLASS(klass)->activate = my_application_activate; G_APPLICATION_CLASS(klass)->local_command_line = my_application_local_command_line; G_APPLICATION_CLASS(klass)->startup = my_application_startup; G_APPLICATION_CLASS(klass)->shutdown = my_application_shutdown; G_OBJECT_CLASS(klass)->dispose = my_application_dispose; } static void my_application_init(MyApplication* self) {} MyApplication* my_application_new() { // Set the program name to the application ID, which helps various systems // like GTK and desktop environments map this running application to its // corresponding .desktop file. This ensures better integration by allowing // the application to be recognized beyond its binary name. g_set_prgname(APPLICATION_ID); return MY_APPLICATION(g_object_new(my_application_get_type(), "application-id", APPLICATION_ID, "flags", G_APPLICATION_NON_UNIQUE, nullptr)); } ================================================ FILE: linux/runner/my_application.h ================================================ #ifndef FLUTTER_MY_APPLICATION_H_ #define FLUTTER_MY_APPLICATION_H_ #include G_DECLARE_FINAL_TYPE(MyApplication, my_application, MY, APPLICATION, GtkApplication) /** * my_application_new: * * Creates a new Flutter-based application. * * Returns: a new #MyApplication. */ MyApplication* my_application_new(); #endif // FLUTTER_MY_APPLICATION_H_ ================================================ FILE: packages/calculator_widget/analysis_options.yaml ================================================ include: package:flutter_lints/flutter.yaml linter: rules: ================================================ FILE: packages/calculator_widget/lib/animated_button.dart ================================================ import 'dart:io'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; class AnimatedButton extends StatefulWidget { final Widget child; final double initialRadius; final double finalRadius; final Duration animationDuration; final Color? foregroundColor; final Color? backgroundColor; final void Function()? onPressed; final void Function()? onLongPress; const AnimatedButton({ super.key, required this.child, required this.initialRadius, required this.finalRadius, required this.onPressed, this.onLongPress, this.foregroundColor, this.backgroundColor, this.animationDuration = const Duration(milliseconds: 100), }); @override State createState() => _AnimatedButtonState(); } class _AnimatedButtonState extends State with SingleTickerProviderStateMixin { late Animation _animation; late AnimationController _animationController; late final _materialStatesController = WidgetStatesController(); late Color backgroundColor; late Color foregroundColor; @override void initState() { super.initState(); if (!kIsWeb && (Platform.isAndroid || Platform.isIOS || Platform.isFuchsia)) { _materialStatesController.addListener(() { if (_materialStatesController.value.contains(WidgetState.pressed)) { _animationController.forward(); } else { _animationController.reverse(); } }); } _animationController = AnimationController( duration: widget.animationDuration, reverseDuration: widget.animationDuration, vsync: this, ); _animation = Tween( begin: widget.initialRadius, end: widget.finalRadius, ).animate( CurvedAnimation(parent: _animationController, curve: Curves.linear), ); } @override Widget build(BuildContext context) { backgroundColor = widget.backgroundColor ?? Theme.of(context).colorScheme.secondaryContainer; foregroundColor = widget.foregroundColor ?? Theme.of(context).colorScheme.onSecondaryContainer; return _ElevatedButtonTransition( radius: _animation, materialStatesController: _materialStatesController, foregroundColor: foregroundColor, backgroundColor: backgroundColor, onPressed: widget.onPressed, onLongPress: widget.onLongPress, onHover: (isHovered) => isHovered ? _animationController.forward() : _animationController.reverse(), child: widget.child, ); } @override void dispose() { _animationController.dispose(); _materialStatesController.dispose(); super.dispose(); } } class _ElevatedButtonTransition extends AnimatedWidget { final void Function()? onPressed; final void Function()? onLongPress; final void Function(bool)? onHover; final WidgetStatesController? materialStatesController; final Widget child; final Color foregroundColor; final Color backgroundColor; const _ElevatedButtonTransition({ this.onPressed, this.onLongPress, this.onHover, this.materialStatesController, required this.foregroundColor, required this.backgroundColor, required Animation radius, required this.child, }) : super(listenable: radius); Animation get radius => listenable as Animation; @override Widget build(BuildContext context) { final ButtonStyle style = FilledButton.styleFrom( foregroundColor: foregroundColor, backgroundColor: backgroundColor, shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(radius.value), ), ).copyWith( overlayColor: WidgetStateProperty.resolveWith( (Set states) => Colors.transparent, ), backgroundColor: WidgetStateProperty.resolveWith( (Set states) => states.contains(WidgetState.pressed) ? Color.alphaBlend( Theme.brightnessOf(context) == Brightness.light ? Colors.black.withValues(alpha: 0.18) : Colors.white30, backgroundColor, ) : backgroundColor, ), ); return FilledButton( onPressed: onPressed, onLongPress: onLongPress, statesController: materialStatesController, onHover: onHover, style: style, child: child, ); } } ================================================ FILE: packages/calculator_widget/lib/calculator_model.dart ================================================ import 'package:decimal/decimal.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:flutter_riverpod/legacy.dart'; import 'dart:math' as math; import 'package:rational/rational.dart'; enum OPERATION { product, division, addition, subtraction; @override String toString() => switch (this) { product => '×', division => '÷', addition => '+', subtraction => '−', }; } class Calculator extends Notifier { final RegExp _regExpValidatingChar = RegExp(r'^[0-9πe,.+-/=*×÷−]+$'); final RegExp _regExpNumber = RegExp(r'^[0-9πe]+$'); static final provider = NotifierProvider(Calculator.new); Decimal? _firstNumber; late final Map mapOperation; @override String build() { mapOperation = ref.read(selectedOperationProvider.notifier).mapOperation; return ''; } void submitString(String string) { for (final char in string.split('')) { submitChar(char); } } /// With this function you can subit any char from 0-9, [, . + - * / = /// backspace canc] void submitChar(String char) { //if string is not valid then return if (!(char.length == 1 && _regExpValidatingChar.hasMatch(char))) { return; } // if it is a number if (_regExpNumber.hasMatch(char)) { if (ref.read(isResultProvider)) { ref.read(isResultProvider.notifier).state = false; ref.read(selectedOperationProvider.notifier).state = null; _firstNumber = null; ref.read(previewResultProvider.notifier).state = ''; } if (ref.read(endNumberProvider)) { state = ''; ref.read(endNumberProvider.notifier).state = false; } if (char == 'π') { state = math.pi.toString(); } else if (char == 'e') { state = math.e.toString(); } else { // is a number state += char; } // Update preview result if an operation is selected _updatePreviewResult(); } //if char is a comma or a dot else if (char == '.' || char == ',') { // if it is already submitted a decimal value ignore it if (state.contains('.') || state.contains(',')) { return; } // otherwise state += '.'; //append the point at the end } // If it is the leading minus before the first number else if (char == '-' && state == '' && _firstNumber == null) { state += '-'; } // if it is an operator else if (mapOperation.containsKey(char)) { if (ref.read(isResultProvider)) { ref.read(isResultProvider.notifier).state = false; ref.read(selectedOperationProvider.notifier).state = null; _firstNumber = null; } // if it is the first operation submitted if (state.isNotEmpty && _firstNumber == null && ref.read(selectedOperationProvider) == null) { _firstNumber = Decimal.parse(state); ref.read(selectedOperationProvider.notifier).state = mapOperation[char]; ref.read(endNumberProvider.notifier).state = true; } else if (state.isNotEmpty && _firstNumber != null && ref.read(selectedOperationProvider) != null && !ref.read(endNumberProvider)) { // chained operation // Compute the result with the previous operator _computeResult(); ref.read(endNumberProvider.notifier).state = true; ref.read(selectedOperationProvider.notifier).state = mapOperation[char]; } else if (state.isNotEmpty && _firstNumber != null && ref.read(selectedOperationProvider) != null) { // change of operation ref.read(selectedOperationProvider.notifier).state = mapOperation[char]; } } // if it is equal symbol else if (char == '=') { if (_firstNumber != null && state.isNotEmpty && ref.read(selectedOperationProvider) != null && !ref.read(isResultProvider)) { _computeResult(); ref.read(isResultProvider.notifier).state = true; ref.read(previewResultProvider.notifier).state = ''; } else if (_firstNumber != null && state.isNotEmpty && ref.read(selectedOperationProvider) != null && ref.read(isResultProvider)) { _computeResult(); ref.read(previewResultProvider.notifier).state = ''; } } } /// Given firstNumber, secondNumber and selectedOperation in computes the /// result and put it in currentNumber string void _computeResult() { assert( _firstNumber != null && state.isNotEmpty, 'firstNumber is null or state is empty', ); final previewResult = ref.read(previewResultProvider); if (previewResult.isNotEmpty) { final result = Decimal.parse(previewResult); _firstNumber = result; state = previewResult; ref.read(endNumberProvider.notifier).state = true; ref.read(previewResultProvider.notifier).state = ''; } } /// Computes the preview result without updating the main state. /// This is used to show a preview of the result before the user clicks '=' void _updatePreviewResult() { if (_firstNumber != null && state.isNotEmpty && ref.read(selectedOperationProvider) != null) { try { final secondNumber = Decimal.parse(state); late Decimal result; result = switch (ref.read(selectedOperationProvider)) { OPERATION.addition => _firstNumber! + secondNumber, OPERATION.subtraction => _firstNumber! - secondNumber, OPERATION.product => _firstNumber! * secondNumber, OPERATION.division => (_firstNumber! / secondNumber).toDecimal( scaleOnInfinitePrecision: 15, ), null => throw Exception('selectedOperation is null'), }; ref.read(previewResultProvider.notifier).state = _getStringFromDecimal( result, ); } catch (_) { // If parsing fails, clear the preview ref.read(previewResultProvider.notifier).state = ''; } } else { ref.read(previewResultProvider.notifier).state = ''; } } /// This method bring the calculator to the initial state (nothing submitted, /// no selected operation) void clearAll() { state = ''; _firstNumber = null; ref.read(selectedOperationProvider.notifier).state = null; ref.read(endNumberProvider.notifier).state = false; ref.read(isResultProvider.notifier).state = false; ref.read(previewResultProvider.notifier).state = ''; } /// This method delete the last character of currentNumber void deleteLastChar() { if (state.isNotEmpty) { state = state.substring(0, state.length - 1); } } /// This method either clearAll() or deleteLastChar() depending on the state /// of the calculator. void adaptiveDeleteClear() => ref.read(endNumberProvider) ? clearAll() : deleteLastChar(); void percentage() => _unaryOperation( (Decimal x) => _getStringFromRational(x / Decimal.fromInt(100)), ); /// Computes the square root of currentNumber void squareRoot() => _unaryOperation( (Decimal x) => _getStringFromNum(math.sqrt(x.toDouble())), ); /// Computes the base-10 logarithm of currentNumber void log10() => _unaryOperation( (Decimal x) => _getStringFromNum((math.log(x.toDouble()) / math.log(10))), ); /// Computes the square of currentNumber void square() => _unaryOperation((Decimal x) => _getStringFromDecimal((x * x))); /// Computes the natural logarithm (base e) of currentNumber void ln() => _unaryOperation((Decimal x) => _getStringFromNum(math.log(x.toDouble()))); /// Computes the reciprocal (multiplicative inverse) of currentNumber void reciprocal() => _unaryOperation( (Decimal x) => x.inverse.toDecimal(scaleOnInfinitePrecision: 15).toString(), ); /// Computes the factorial of currentNumber void factorial() => _unaryOperation( (Decimal x) => _getStringFromNum(_myFactorialFunction(x.toBigInt().toInt())), ); /// General function that applies to all the unary operations, just pass the /// function void _unaryOperation(String Function(Decimal) operation) { //if it is the first operation submitted if (state.isNotEmpty) { if (_firstNumber != null) { // If the previous operation is not completed, compute it before // computing the new unary operation submitChar('='); } state = operation(Decimal.parse(state)); ref.read(endNumberProvider.notifier).state = true; ref.read(isResultProvider.notifier).state = true; } } } String _getStringFromRational(Rational rational) { String stringValue = (rational.numerator / rational.denominator).toString(); if (stringValue.endsWith('.0')) { stringValue = stringValue.substring(0, stringValue.length - 2); } return stringValue; } String _getStringFromDecimal(Decimal value) => _getStringFromRational(value.toRational()); String _getStringFromNum(num value) { String stringValue = value.toString(); if (stringValue.endsWith('.0')) { stringValue = stringValue.substring(0, stringValue.length - 2); } return stringValue; } int _myFactorialFunction(int x) => x == 0 ? 1 : x * _myFactorialFunction(x - 1); class SelectedOperationNotifier extends Notifier { static const Map _mapOperation = { '*': OPERATION.product, '/': OPERATION.division, '+': OPERATION.addition, '-': OPERATION.subtraction, '×': OPERATION.product, '÷': OPERATION.division, '−': OPERATION.subtraction, }; @override OPERATION? build() => null; Map get mapOperation => _mapOperation; } final selectedOperationProvider = NotifierProvider( () => SelectedOperationNotifier(), ); final endNumberProvider = StateProvider((ref) => false); final isResultProvider = StateProvider((ref) => false); final previewResultProvider = StateProvider((ref) => ''); ================================================ FILE: packages/calculator_widget/lib/calculator_widget.dart ================================================ import 'package:calculator_widget/animated_button.dart'; import 'package:calculator_widget/calculator_model.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:translations/app_localizations.dart'; enum ButtonType { number, operation, clear } const double _buttonsSpacing = 5; class CalculatorWidget extends StatelessWidget { const CalculatorWidget({super.key}); @override Widget build(BuildContext context) => ProviderScope(child: _CalculatorWidget()); } class _CalculatorWidget extends ConsumerWidget { final FocusNode focusKeyboard = FocusNode(); _CalculatorWidget(); @override Widget build(BuildContext context, WidgetRef ref) { return Shortcuts( shortcuts: { LogicalKeySet(LogicalKeyboardKey.control, LogicalKeyboardKey.keyV): const ActivateIntent(), }, child: Actions( actions: >{ ActivateIntent: CallbackAction( onInvoke: (ActivateIntent intent) { Clipboard.getData(Clipboard.kTextPlain).then((value) { if (value != null) { final clipboardContent = value.text; if (clipboardContent != null && clipboardContent.length < 30) { ref .read(Calculator.provider.notifier) .submitString(clipboardContent); } } }); return null; }, ), }, child: SafeArea( child: SizedBox( height: 480, child: Builder( builder: (context) { focusKeyboard.requestFocus(); return KeyboardListener( focusNode: focusKeyboard, onKeyEvent: (KeyEvent event) { if (event.runtimeType.toString() == 'KeyDownEvent') { switch (event.logicalKey) { case LogicalKeyboardKey.backspace: ref .read(Calculator.provider.notifier) .adaptiveDeleteClear(); case LogicalKeyboardKey.delete: ref.read(Calculator.provider.notifier).clearAll(); case LogicalKeyboardKey.enter: ref .read(Calculator.provider.notifier) .submitChar('='); default: ref .read(Calculator.provider.notifier) .submitChar(event.character ?? ''); } } }, child: Column( children: [ Expanded( flex: 3, child: Container( decoration: BoxDecoration( color: Theme.of(context).colorScheme.surface, borderRadius: const BorderRadius.all( Radius.circular(30), ), ), child: Column( children: [ // handle Center( child: Padding( padding: const EdgeInsets.symmetric( vertical: 8, ), child: Container( width: 32, height: 4, decoration: BoxDecoration( color: Theme.of(context) .colorScheme .onSurfaceVariant .withValues(alpha: 0.4), borderRadius: BorderRadius.circular(2), ), ), ), ), const Expanded(child: CalculatorHeader()), ], ), ), ), const Expanded(flex: 7, child: CalculatorNumpad()), ], ), ); }, ), ), ), ), ); } } class CalculatorHeader extends ConsumerWidget { const CalculatorHeader({super.key}); @override Widget build(BuildContext context, WidgetRef ref) { final text = ref.watch(Calculator.provider); final previewText = ref.watch(previewResultProvider); final operation = ref.watch(selectedOperationProvider); return Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [ Expanded( child: Padding( padding: const EdgeInsets.symmetric(horizontal: 10), child: Column( mainAxisAlignment: MainAxisAlignment.center, crossAxisAlignment: CrossAxisAlignment.start, children: [ SelectableText( text, style: TextStyle( fontSize: 45, fontWeight: FontWeight.bold, color: Theme.of(context).colorScheme.onSurface, ), maxLines: 1, textScaler: TextScaler.noScaling, scrollPhysics: const ClampingScrollPhysics(), ), if (previewText.isNotEmpty) SelectableText( previewText, style: TextStyle( fontSize: 20, fontWeight: FontWeight.normal, color: Theme.of( context, ).colorScheme.onSurface.withValues(alpha: 0.5), ), maxLines: 1, textScaler: TextScaler.noScaling, scrollPhysics: const ClampingScrollPhysics(), ), ], ), ), ), SizedBox( width: 50, child: Center( child: ref.watch(isResultProvider) ? IconButton( tooltip: AppLocalizations.of(context)?.copy, icon: Icon( Icons.content_copy_outlined, color: Theme.of(context).colorScheme.onSurface, ), onPressed: () { Clipboard.setData(ClipboardData(text: text)); HapticFeedback.heavyImpact(); }, ) : text.isEmpty ? IconButton( tooltip: AppLocalizations.of(context)?.paste, icon: Icon( Icons.content_paste_outlined, color: Theme.of(context).colorScheme.onSurface, ), onPressed: () { Clipboard.getData(Clipboard.kTextPlain).then((value) { final cliboard = value?.text; if (cliboard != null) { ref .watch(Calculator.provider.notifier) .submitString(cliboard); } }); HapticFeedback.heavyImpact(); }, ) : Text( operation != null ? operation.toString() : '', style: TextStyle( fontSize: 45.0, fontWeight: FontWeight.bold, color: Theme.of(context).colorScheme.onSurface, ), textScaler: TextScaler.noScaling, maxLines: 1, ), ), ), ], ); } } class CalculatorNumpad extends ConsumerWidget { const CalculatorNumpad({super.key}); static const double breakPoint1 = 500; static const double breakPoint2 = 610; static const decimalSeparator = '.'; @override Widget build(BuildContext context, WidgetRef ref) { final calcWidth = MediaQuery.sizeOf(context).width; return Padding( padding: const EdgeInsets.all(_buttonsSpacing), child: Row( children: [ if (calcWidth > breakPoint2) Column( children: [ CalculatorButton( text: 'x²', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).square(); }, ), CalculatorButton( text: 'ln', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).ln(); }, ), CalculatorButton( text: 'n!', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).factorial(); }, ), CalculatorButton( text: '1/x', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).reciprocal(); }, ), ].map((e) => Expanded(child: e)).toList(), ), if (calcWidth > breakPoint1) Column( children: [ CalculatorButton( text: '%', buttonType: ButtonType.operation, onPressed: ref.read(Calculator.provider.notifier).percentage, ), CalculatorButton( text: '√', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).squareRoot(); }, ), CalculatorButton( text: 'log', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).log10(); }, ), CalculatorButton( text: 'π', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).submitChar('π'); }, ), ].map((e) => Expanded(child: e)).toList(), ), ...List.generate( 3, (columnIndex) => Column( children: [ ...List.generate(3, (rowIndex) { final char = (7 - 3 * rowIndex + columnIndex).toString(); return CalculatorButton( text: char, buttonType: ButtonType.number, onPressed: () { ref.read(Calculator.provider.notifier).submitChar(char); }, ); }), if (columnIndex == 0) CalculatorButton( text: decimalSeparator, buttonType: ButtonType.operation, onPressed: () { ref .read(Calculator.provider.notifier) .submitChar(decimalSeparator); }, ) else if (columnIndex == 1) CalculatorButton( text: '0', buttonType: ButtonType.number, onPressed: () { ref.read(Calculator.provider.notifier).submitChar('0'); }, ) else if (columnIndex == 2) CalculatorButton( text: '=', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).submitChar('='); }, ), ].map((e) => Expanded(child: e)).toList(), ), ), Column( children: [ CalculatorButton( text: ref.read(endNumberProvider) ? 'AC' : '←', buttonType: ButtonType.clear, onPressed: () { ref.read(Calculator.provider.notifier).adaptiveDeleteClear(); }, onLongPress: () { ref.read(Calculator.provider.notifier).clearAll(); }, ), CalculatorButton( text: '÷', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).submitChar('/'); }, ), CalculatorButton( text: '×', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).submitChar('*'); }, ), CalculatorButton( text: '−', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).submitChar('-'); }, ), CalculatorButton( text: '+', buttonType: ButtonType.operation, onPressed: () { ref.read(Calculator.provider.notifier).submitChar('+'); }, ), ].map((e) => Expanded(child: e)).toList(), ), ].map((e) => Expanded(child: e)).toList(), ), ); } } class CalculatorButton extends StatelessWidget { final String? text; final void Function()? onPressed; final void Function()? onLongPress; final ButtonType buttonType; const CalculatorButton({ super.key, this.text, this.onLongPress, this.onPressed, required this.buttonType, }); @override Widget build(BuildContext context) { final foregroundColor = switch (buttonType) { ButtonType.number => Theme.of(context).colorScheme.onPrimaryContainer, ButtonType.operation => Theme.of( context, ).colorScheme.onSecondaryContainer, ButtonType.clear => Theme.of(context).colorScheme.onTertiaryContainer, }; final backgroundColor = switch (buttonType) { ButtonType.number => Theme.of(context).colorScheme.primaryContainer, ButtonType.operation => Theme.of(context).colorScheme.secondaryContainer, ButtonType.clear => Theme.of(context).colorScheme.tertiaryContainer, }; return Padding( padding: const EdgeInsets.all(_buttonsSpacing), child: AnimatedButton( initialRadius: 60, finalRadius: 20, foregroundColor: foregroundColor, backgroundColor: backgroundColor, onPressed: () { if (onPressed != null) { HapticFeedback.heavyImpact(); onPressed!(); } }, onLongPress: () { if (onLongPress != null) { HapticFeedback.heavyImpact(); onLongPress!(); } }, child: SizedBox.expand( child: Center( child: text == "←" ? const Icon(Icons.backspace_outlined) : Text( text ?? '', style: const TextStyle(fontSize: 27), maxLines: 1, textScaler: TextScaler.noScaling, ), ), ), ), ); } } ================================================ FILE: packages/calculator_widget/pubspec.yaml ================================================ name: calculator_widget description: Calculator widget for Converter NOW publish_to: 'none' resolution: workspace version: 1.0.0+1 environment: sdk: ">=3.9.0 <4.0.0" dependencies: flutter: sdk: flutter # Calculator computation decimal: ^3.2.4 # State management flutter_riverpod: ^3.2.1 # Calculator computation rational: ^2.2.3 # App translation translations: path: ../translations dev_dependencies: flutter_test: sdk: flutter flutter_lints: ^6.0.0 flutter: uses-material-design: true ================================================ FILE: packages/translations/l10n.yaml ================================================ arb-dir: lib/l10n template-arb-file: app_en.arb output-dir: lib output-localization-file: app_localizations.dart ================================================ FILE: packages/translations/lib/l10n/app_ar.arb ================================================ { "@@locale": "ar", "appName": "Converter NOW", "reorder": "الترتيب", "myOrdering": "ترتيبي الخاص", "about": "حول", "lastCurrenciesUpdate": "أخر تحديث: ", "today": "اليوم", "enableDarkTheme": "تمكين المظهر الداكن", "settings": "الإعدادت", "menu": "القائمة", "donation": "قم بالتبرع", "buyMeACoffee": "قم بشراء كوب قهوة من أجلي", "donationDialog": "مرحبا، كما تعلم هذا التطبيق مفتوح المصدر ومجاني. هذا يعني انه يُمكنك نسخ و تعديل على الكود المصدر. لن يكسب المطور شيء من ذلك. و لكن إذا كنت تؤمن في هذا المشروع و تود أن ترى ميزات جديدة، إعتبر شراء كوب قهوة من أجلي.", "drawerLogo": "Drawer logo", "rateApp": "قيم التطبيق", "repoGithub": "تصفح مستودع الكود على منصة ال Github", "contributeTranslating": "ساهم في ترجمة التطبيق", "search": "بحث", "clearAll": "مسح الكل", "calculator": "الآلة الحاسبة", "significantFigures": "عدد الخانات", "removeTrailingZeros": "إزالة الأصفار اللاحقة", "longPressAdvice": "إضغط بشكل مطول لنقل العناصر", "back": "رجوع", "save": "حفظ", "invalidCharacters": "خطأ، حروف غير صالحة!", "theme": "المظهر", "dark": "داكن", "light": "فاتح", "system": "إعتماداً على النظام", "pureBlackTheme": "مظهر AMOLED المظلم", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "نسخ", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "المزيد", "language": "اللغة", "reorderProperties": "ترتيب الكميات", "reorderUnits": "ترتيب الوحدات", "chooseProperty": "إختر كمية", "reorderProperty": "ترتيب {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW على منصات أخرى", "sourceCode": "الكود المصدر", "undoClearAllMessage": "تم مسح الكل. هل تريد التراجُع", "undo": "تراجُع", "routeError1": "هذه الصفحة غير موجودة", "routeError2": "الروجوع إلى الصفحة الأولى", "ok": "Ok", "@ok": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetAccess": "Revoke internet access", "@revokeInternetAccess": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetExplanation": "This app uses the internet to update currency exchange rates once a day upon opening. This download only requires a few kilobytes. However, if you don't need this feature, you can prevent the app from accessing the internet by enabling this option.", "@revokeInternetExplanation": { "description": "Not yet translated. Once done, remove this comment" }, "useDeviceColor": "Use device color", "@useDeviceColor": { "description": "Not yet translated. Once done, remove this comment" }, "pickColor": "Pick a color", "@pickColor": { "description": "Not yet translated. Once done, remove this comment" }, "themeColor": "Theme color", "@themeColor": { "description": "Not yet translated. Once done, remove this comment" }, "appearance": "Appearance", "@appearance": { "description": "Not yet translated. Once done, remove this comment" }, "conversions": "Conversions", "@conversions": { "description": "Not yet translated. Once done, remove this comment" }, "findOutMore": "Find out more", "@findOutMore": { "description": "Not yet translated. Once done, remove this comment" }, "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "الطول", "area": "المساحة", "volume": "الحجم", "time": "الزمن", "temperature": "درجة الحرارة", "speed": "السرعة", "siPrefixes": "SI سوابق", "mass": "الكتلة", "pressure": "الضغط", "energy": "الطاقة", "angles": "الزاوية", "currencies": "العُمُلات", "shoeSize": "مِقياس الحذاء", "digitalData": "البيانات الرقمية", "power": "القُدرة", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "عزم الدوران", "force": "القوة", "fuelConsumption": "إستهلاك الوقود", "numeralSystems": "الأنظمة العددية", "meters": "متر", "centimeters": "سنتيمتر", "inches": "بوصة", "feet": "قدم", "nauticalMiles": "ميل بحري", "yards": "ياردة", "miles": "ميل", "millimeters": "مليمتر", "micrometers": "ميكرومتر", "nanometers": "نانومتر", "angstroms": "أنغستروم", "picometers": "بيكومتر", "kilometers": "كيلومتر", "astronomicalUnits": "الوحدات الفلكية", "lightYears": "سنة ضوئية", "parsec": "فرسخ فلكي", "mils": "جزء من ألف من البوصة", "feetUsSurvey": "قدم (المقياس أمريكي)", "@feetUsSurvey": { "description": "Not yet translated. Once done, remove this comment" }, "squareMeters": "متر مربع", "squareCentimeters": "سنتيمتر مربع", "squareInches": "بوصة مربعة", "squareFeet": "قدم مربع", "squareMiles": "ميل مربع", "squareYard": "ياردة مربعة", "squareMillimeters": "مليمتر مربع", "squareKilometers": "كيلومتر مربع", "hectares": "هكتار", "acres": "فدان", "are": "آر", "squareFeetUsSurvey": "Piedi Quadrati (US survey)", "@squareFeetUsSurvey": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "متر مكعب", "liters": "لتر", "imperialGallons": "غالون إمبراطوري", "usGallons": "غالون أمريكي", "imperialPints": "باينت إمبراطوري", "usPints": "باينت أمريكي", "milliliters": "ملليلتر", "tablespoonUs": "ملعقة طعام أمريكية", "tablespoonAustralian": "ملعقة طعام أُسترالية", "cups": "كوب", "cubicCentimeters": "سنتيمتر مكعب", "cubicFeet": "قدم مكعب", "cubicInches": "بوصة مكعبة", "cubicMillimeters": "ملليمتر مكعب", "imperialFluidOunces": "أونصة سائلة", "usFluidOunces": "أونصة سائلة أمريكية", "imperialGill": "جل", "usGill": "جل أمريكي", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "ثانية", "deciseconds": "ديسي ثانية", "centiseconds": "سنتي ثانية", "milliseconds": "ملي ثانية", "microseconds": "ميكرو ثانية", "nanoseconds": "نانو ثانية", "minutes": "دقيقة", "hours": "ساعة", "days": "يوم", "weeks": "أسبوع", "years": "سنة (365)", "lustrum": "لوسترم", "decades": "عقد", "centuries": "قرن", "millennium": "ألفية", "fahrenheit": "فهرنهايت", "celsius": "سِلِسْيُوس (درجة حرارة مئوية)", "kelvin": "كلفن", "reamur": "ريومور", "romer": "رومر", "delisle": "ديليسل", "rankine": "رانكين", "metersSecond": "متر في الثانية", "kilometersHour": "كيلومتر في الساعة", "milesHour": "ميل في الساعة", "knots": "عقدة", "feetSecond": "قدم في الثانية", "minutesPerKilometer": "دقيقة لكل كيلومتر", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "غرام", "ettograms": "إيتو غرام", "kilograms": "كيلوغرام", "pounds": "رطل", "ounces": "أونصة", "quintals": "كنتال", "tonnes": "طن", "milligrams": "مليغرام", "uma": "وحدة كتل ذرية", "carats": "قيراط", "centigrams": "سنتي غرام", "pennyweights": "وزن البنس", "troyOunces": "أونصة تروي", "stones": "حجر", "pascal": "باسكال", "atmosphere": "مقياس الغلاف الجوي", "bar": "بار", "millibar": "ملي بار", "psi": "رطل لكل بوصة مربعة", "torr": "تور", "hectoPascal": "هكتو باسكال", "inchesOfMercury": "بوصة زئبق", "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "جول", "calories": "سعرة الحرارية)كالوري)", "kilowattHour": "كيلوواط ساعة", "electronvolt": "إلكترون فولت", "footPound": "قدم - رطل", "kilocalories": "Kilocalories", "@kilocalories": { "description": "Not yet translated. Once done, remove this comment" }, "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "درجة", "minutesDegree": "دقيقة", "secondsDegree": "ثانية", "radiansDegree": "راديان", "usd": "دولار أمريكي", "eur": "يورو", "gbp": "جنيه إسترليني", "inr": "روبية هندية", "cny": "يوان صيني", "jpy": "ين ياباني", "chf": "فرنك سويسري", "sek": "كرونا سويدي", "rub": "روبل روسي", "cad": "دورلار كندي", "krw": "وون كوري جنوبي", "brl": "ريال برازيلي", "hkd": "دولار هونغ كونغ", "aud": "دولار أسترالي", "nzd": "دولار نيوزيلندي", "mxn": "بيزو مكسيكي", "sgd": "دولار سنغافوري", "nok": "كرونة نروجية", "trY": "ليرة تركية", "zar": "راند جنوب أفريقي", "dkk": "كرونة دنماركية", "pln": "زلوتي بولندي", "thb": "بات تايلاندي", "myr": "رينغيت ماليزي", "huf": "فورنت مجري", "czk": "كرونة تشيكية", "ils": "شيكل إسرائيلي", "idr": "روبية إندونيسية", "php": "بيسو فلبيني", "ron": "ليو روماني", "isk": "كرونة آيسلندية", "twd": "دولار تايواني", "mad": "درهم مغربي", "euChina": "الإتحاد الأوروبي و الصين", "ukIndiaChild": "المملكة المُتحدة و الهند - أطفال", "ukIndiaMan": "المملكة المُتحدة و الهند - رجال", "ukIndiaWoman": "المملكة المُتحدة و الهند - نساء", "usaCanadaChild": "الولايات المُتحدة و كند - أطفال", "usaCanadaMan": "الولايات المُتحدة و كندا - رجال", "usaCanadaWoman": "الولايات المُتحدة و كندا - نساء", "japan": "اليابان", "watt": "واط", "milliwatt": "ملي واط", "kilowatt": "كيلو واط", "megawatt": "ميغا واط", "gigawatt": "جيجا واط", "europeanHorsePower": "حِصان أوروبي", "imperialHorsePower": "حِصان", "newton": "نيوتن", "dyne": "داين", "poundForce": "رطل- قوة", "kilogramForce": "كيلوغرام ثقلي", "poundal": "باوندال", "newtonMeter": "نيوتن متر", "dyneMeter": "داين متر", "poundForceFeet": "رطل قدم", "kilogramForceMeter": "قوة كيلوغرام متر", "poundalMeter": "Poundal meter", "@poundalMeter": { "description": "Not yet translated. Once done, remove this comment." }, "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "كليومتر لكل لتر", "liters100km": "لتر لكل 100 كلم", "milesUsGallon": "ميل لكل غالون أمريكي", "milesImperialGallon": "ميل لكل غالون الامبراطوري", "decimal": "عشري", "hexadecimal": "سادس عشر", "octal": "ثُماني", "binary": "ثُنائي", "bit": "بت", "nibble": "نايبل", "kilobit": "كيلوبت", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "ميغابت", "@megabit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "جيجابت", "@gigabit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "تيرابت", "@terabit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "بيتابت", "@petabit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "إكسابت", "@exabit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "كيبيبت", "mebibit": "ميبيبت", "gibibit": "جيبيبت", "tebibit": "تيبيبت", "pebibit": "بيبيبت", "exbibit": "إكسبيبت", "byte": "بايت", "kilobyte": "كيلوبايت", "@kilobyte": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "ميغابابت", "@megabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "جيجابابت", "@gigabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "تيرابابت", "@terabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "بيتابابت", "@petabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "إكسابابت", "@exabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "كيبي بايت", "mebibyte": "مبي بايت", "gibibyte": "جيبي بايت", "tebibyte": "تيبي بايت", "pebibyte": "بيبي بايت", "exbibyte": "اكسبي بايت", "base": "أساس", "deca": "ديكا", "hecto": "هكتو-", "kilo": "كيلو-", "mega": "ميغا-", "giga": "جيجا-", "tera": "تيرا", "peta": "بيتا-", "exa": "إكسا-", "zetta": "زيتا-", "yotta": "يوتا-", "deci": "ديسي-", "centi": "سنتي-", "milli": "ميللي-", "micro": "ميكرو-", "nano": "نانو-", "pico": "بيكو-", "femto": "فيمتو-", "atto": "أتو-", "zepto": "زيبتو-", "yocto": "يوكتو-" } ================================================ FILE: packages/translations/lib/l10n/app_bn.arb ================================================ { "@@locale": "bn", "appName": "Converter NOW", "reorder": "পুনর্বিন্যাস করুন", "myOrdering": "আমার ক্রম", "about": "সম্পর্কিত", "lastCurrenciesUpdate": "সর্বশেষ আপডেট: ", "today": "আজ", "enableDarkTheme": "ডার্ক থিম", "settings": "সেটিংস", "menu": "মেনু", "donation": "অনুদান দিন", "buyMeACoffee": "আমার জন্য একটা কফি ?", "donationDialog": "হাই, আপনি হয়তো যানেন যে এই অ্যাপটি ফ্রী এবং ওপেন সোর্স। এর মানে আপনি এটি বিনামূল্যে ব্যবহার এবং পরিবর্তন করতে পারেন। ডেভেলপার এর জন্য কিছু পাবেন না। তবে আপনার যদি এটি ভাল লাগে এবং নতুন আরও কোনও বৈশিষ্ট্য দেখতে চান, তাহলে আমাকে একটি কফি কিনে সহায়তা করার কথা ভাবতে পারেন", "drawerLogo": "ড্রয়ার লোগো", "rateApp": "অ্যাপ রেট করুন", "repoGithub": "গিটহাবে রিপো খুলুন", "contributeTranslating": "অ্যাপ অনুবাদে সহায়তা করুন", "search": "অনুসন্ধান", "clearAll": "সব মুছে ফেলুন", "calculator": "ক্যালকুলেটর", "significantFigures": "গুরুত্বপূর্ণ অঙ্ক", "removeTrailingZeros": "শেষের শূন্যগুলি সরান", "longPressAdvice": "আইটেম সরাতে চেপে ধরে রাখুন", "back": "পিছনে", "save": "সংরক্ষণ করুন", "invalidCharacters": "অগ্রহণযোগ্য অক্ষর", "theme": "থিম", "dark": "ডার্ক", "light": "লাইট", "system": "সিস্টেম ডিফল্ট", "pureBlackTheme": "AMOLED ডার্ক থিম", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "কপি", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "আরও", "language": "ভাষা", "reorderProperties": "প্রোপার্টিগুলি পুনর্বিন্যাস করুন", "reorderUnits": "একক পুনর্বিন্যাস করুন", "chooseProperty": "একটি প্রোপার্টি নির্বাচন করুন", "reorderProperty": "{property} পুনর্বিন্যাস করুন", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "অন্যান্য প্ল্যাটফর্মের জন্য ConverterNOW", "sourceCode": "সোর্স কোড", "undoClearAllMessage": "সব মুছে ফেলা হয়েছে। আপনি কি পূর্বাবস্থায় ফিরিয়ে আনতে চান ?", "undo": "পূর্বাবস্থায় ফিরিয়ে আনুন", "routeError1": "এই পৃষ্ঠাটি পাওয়া যায়নি", "routeError2": "প্রথম পৃষ্ঠায় ফিরে যান", "ok": "ঠিক আছে", "revokeInternetAccess": "ইন্টারনেট অ্যাক্সেস বাতিল করুন", "revokeInternetExplanation": "এই অ্যাপটি প্রতিদিন একবার মানি এক্সচেঞ্জ রেট আপডেট করতে ইন্টারনেট ব্যবহার করে। এটি মাত্র কয়েক কিলোবাইট ডাউনলোড করে। তবে যদি আপনি এটি না চান, তাহলে এই অপশনটি ব্যাবহার করে অ্যাপের ইন্টারনেট অ্যাক্সেস বন্ধ করতে পারেন।", "useDeviceColor": "ডিভাইসের রঙ ব্যবহার করুন", "pickColor": "একটি রঙ নির্বাচন করুন", "themeColor": "থিম কালার", "appearance": "চেহারা", "conversions": "কনভার্শন", "findOutMore": "আরও জানুন", "hideUnits": "একক গোপন করুন", "visibleUnits": "{property}, দৃশ্যমান একক", "@visibleUnits": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "গোপন একক", "selectAll": "সব নির্বাচন করুন", "unselectAll": "সব আনচেক করুন", "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "দৈর্ঘ্য", "area": "ক্ষেত্রফল", "volume": "আয়তন", "time": "সময়", "temperature": "তাপমাত্রা", "speed": "গতি", "siPrefixes": "এসআই উপসর্গ", "mass": "ভর", "pressure": "চাপ", "energy": "শক্তি", "angles": "কোণ", "currencies": "মুদ্রা", "shoeSize": "জুতার মাপ", "digitalData": "ডিজিটাল ডেটা", "power": "শক্তি", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "টর্ক", "force": "বল", "fuelConsumption": "জ্বালানি খরচ", "numeralSystems": "সংখ্যা পদ্ধতি", "meters": "মিটার", "centimeters": "সেন্টিমিটার", "inches": "ইঞ্চি", "feet": "ফুট", "nauticalMiles": "নটিক্যাল মাইল", "yards": "গজ", "miles": "মাইল", "millimeters": "মিলিমিটার", "micrometers": "মাইক্রোমিটার", "nanometers": "ন্যানোমিটার", "angstroms": "অ্যাংস্ট্রোম", "picometers": "পিকোমিটার", "kilometers": "কিলোমিটার", "astronomicalUnits": "মহাকাশীয় একক", "lightYears": "আলোকবর্ষ", "parsec": "পারসেক", "mils": "মিল", "feetUsSurvey": "ফুট (মার্কিন জরিপ)", "squareMeters": "বর্গ মিটার", "squareCentimeters": "বর্গ সেন্টিমিটার", "squareInches": "বর্গ ইঞ্চি", "squareFeet": "বর্গ ফুট", "squareMiles": "বর্গ মাইল", "squareYard": "বর্গ গজ", "squareMillimeters": "বর্গ মিলিমিটার", "squareKilometers": "বর্গ কিলোমিটার", "hectares": "হেক্টর", "acres": "একর", "are": "আর", "squareFeetUsSurvey": "বর্গ ফুট (মার্কিন জরিপ)", "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "ঘন মিটার", "liters": "লিটার", "imperialGallons": "ইম্পেরিয়াল গ্যালন", "usGallons": "মার্কিন গ্যালন", "imperialPints": "ইম্পেরিয়াল পাইন্ট", "usPints": "মার্কিন পাইন্ট", "milliliters": "মিলিলিটার", "tablespoonUs": "মার্কিন টেবিল চামচ", "tablespoonAustralian": "অস্ট্রেলিয়ান টেবিল চামচ", "cups": "কাপ", "cubicCentimeters": "ঘন সেন্টিমিটার", "cubicFeet": "ঘন ফুট", "cubicInches": "ঘন ইঞ্চি", "cubicMillimeters": "ঘন মিলিমিটার", "imperialFluidOunces": "ইম্পেরিয়াল তরল আউন্স", "usFluidOunces": "মার্কিন তরল আউন্স", "imperialGill": "ইম্পেরিয়াল গিল", "usGill": "মার্কিন গিল", "usQuarts": "মার্কিন কোয়ার্ট", "microliters": "মাইক্রোলিটার", "deciliters": "ডেসিলিটার", "centiliters": "সেন্টিলিটার", "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "সেকেন্ড", "deciseconds": "ডেসিসেকেন্ড", "centiseconds": "সেন্টিসেকেন্ড", "milliseconds": "মিলিসেকেন্ড", "microseconds": "মাইক্রোসেকেন্ড", "nanoseconds": "ন্যানোসেকেন্ড", "minutes": "মিনিট", "hours": "ঘণ্টা", "days": "দিন", "weeks": "সপ্তাহ", "years": "বছর", "lustrum": "লাস্ট্রাম (৫ বছর)", "decades": "দশক", "centuries": "শতাব্দী", "millennium": "সহস্রাব্দ", "fahrenheit": "ফারেনহাইট", "celsius": "সেলসিয়াস", "kelvin": "কেলভিন", "reamur": "রিয়োমুর", "romer": "রোমার", "delisle": "ডেলিসল", "rankine": "রাঙ্কিন", "metersSecond": "মিটার/সেকেন্ড", "kilometersHour": "কিলোমিটার/ঘণ্টা", "milesHour": "মাইল/ঘণ্টা", "knots": "নট", "feetSecond": "ফুট/সেকেন্ড", "minutesPerKilometer": "মিনিট/কিলোমিটার", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "গ্রাম", "ettograms": "এটোগ্রাম", "kilograms": "কিলোগ্রাম", "pounds": "পাউন্ড", "ounces": "আউন্স", "quintals": "কুইন্টাল", "tonnes": "টন", "milligrams": "মিলিগ্রাম", "uma": "একীভূত পারমাণবিক ভর একক", "carats": "ক্যারেট", "centigrams": "সেন্টিগ্রাম", "pennyweights": "পেনিওয়েট", "troyOunces": "ট্রয় আউন্স", "stones": "স্টোন", "pascal": "পাসক্যাল", "atmosphere": "বায়ুমণ্ডল", "bar": "বার", "millibar": "মিলিবার", "psi": "পাউন্ড প্রতি বর্গ ইঞ্চি", "torr": "পারদ মিলিমিটার (টর)", "hectoPascal": "হেক্টোপাসক্যাল", "inchesOfMercury": "পারদের ইঞ্চি", "kiloPascal": "কিলোপাসক্যাল", "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "জুল", "calories": "ক্যালোরি", "kilowattHour": "কিলোওয়াট ঘণ্টা", "electronvolt": "ইলেকট্রন ভোল্ট", "footPound": "ফুট-পাউন্ড", "kilocalories": "কিলোক্যালোরি", "kilojoules": "কিলোজুল", "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "ডিগ্রি", "minutesDegree": "মিনিট", "secondsDegree": "সেকেন্ড", "radiansDegree": "রেডিয়ান", "usd": "মার্কিন ডলার", "eur": "ইউরো", "gbp": "ব্রিটিশ পাউন্ড", "inr": "ভারতীয় রুপি", "cny": "চীনা ইউয়ান", "jpy": "জাপানি ইয়েন", "chf": "সুইস ফ্রাঙ্ক", "sek": "সুইডিশ ক্রোনা", "rub": "রাশিয়ান রুবল", "cad": "কানাডিয়ান ডলার", "krw": "দক্ষিণ কোরিয়ান ওন", "brl": "ব্রাজিলিয়ান রিয়াল", "hkd": "হংকং ডলার", "aud": "অস্ট্রেলিয়ান ডলার", "nzd": "নিউজিল্যান্ড ডলার", "mxn": "মেক্সিকান পেসো", "sgd": "সিঙ্গাপুর ডলার", "nok": "নরওয়েজিয়ান ক্রোন", "trY": "তুর্কি লিরা", "zar": "দক্ষিণ আফ্রিকান র্যান্ড", "dkk": "ড্যানিশ ক্রোন", "pln": "পোলিশ জ্লোটি", "thb": "থাই বাহত", "myr": "মালয়েশিয়ান রিঙ্গিত", "huf": "হাঙ্গেরিয়ান ফরিন্ট", "czk": "চেক কোরুনা", "ils": "ইসরায়েলি শেকেল", "idr": "ইন্দোনেশিয়ান রুপিয়াহ", "php": "ফিলিপিন পেসো", "ron": "রোমানিয়ান লেউ", "isk": "আইসল্যান্ডিক ক্রোনা", "twd": "নিউ তাইওয়ান ডলার", "mad": "মরোক্কান দিরহাম", "euChina": "ইইউ ও চীন", "ukIndiaChild": "যুক্তরাজ্য ও ভারত - শিশু", "ukIndiaMan": "যুক্তরাজ্য ও ভারত - পুরুষ", "ukIndiaWoman": "যুক্তরাজ্য ও ভারত - নারী", "usaCanadaChild": "যুক্তরাষ্ট্র ও কানাডা - শিশু", "usaCanadaMan": "যুক্তরাষ্ট্র ও কানাডা - পুরুষ", "usaCanadaWoman": "যুক্তরাষ্ট্র ও কানাডা - নারী", "japan": "জাপান", "watt": "ওয়াট", "milliwatt": "মিলিওয়াট", "kilowatt": "কিলোওয়াট", "megawatt": "মেগাওয়াট", "gigawatt": "গিগাওয়াট", "europeanHorsePower": "ইউরোপীয় হর্সপাওয়ার", "imperialHorsePower": "ইম্পেরিয়াল হর্সপাওয়ার", "newton": "নিউটন", "dyne": "ডাইন", "poundForce": "পাউন্ড-ফোর্স", "kilogramForce": "কিলোগ্রাম-ফোর্স", "poundal": "পাউন্ডাল", "newtonMeter": "নিউটন মিটার", "dyneMeter": "ডাইন মিটার", "poundForceFeet": "পাউন্ড-ফোর্স ফুট", "kilogramForceMeter": "কিলোগ্রাম-ফোর্স মিটার", "poundalMeter": "পাউন্ডাল মিটার", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "কিলোমিটার পার লিটার", "liters100km": "শতকিলোমিটার পার লিটার", "milesUsGallon": "মাইল পার মার্কিন গ্যালন", "milesImperialGallon": "মাইল পার ইম্পেরিয়াল গ্যালন", "decimal": "দশমিক", "hexadecimal": "হেক্সাডেসিমাল", "octal": "অক্টাল", "binary": "বাইনারি", "bit": "বিট", "nibble": "নিবল", "kilobit": "কিলোবিট", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "মেগাবিট", "@megabit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "গিগাবিট", "@gigabit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "টেরাবিট", "@terabit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "পেটাবিট", "@petabit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "এক্সাবিট", "@exabit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "কিবিবিট", "mebibit": "মেবিবিট", "gibibit": "গিবিবিট", "tebibit": "টেবিবিট", "pebibit": "পেবিবিট", "exbibit": "এক্সবিবিট", "byte": "বাইট", "kilobyte": "কিলোবাইট", "@kilobyte": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "মেগাবাইট", "@megabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "গিগাবাইট", "@gigabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "টেরাবাইট", "@terabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "পেটাবাইট", "@petabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "এক্সাবাইট", "@exabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "কিবিবাইট", "mebibyte": "মেবিবাইট", "gibibyte": "গিবিবাইট", "tebibyte": "টেবিবাইট", "pebibyte": "পেবিবাইট", "exbibyte": "এক্সবিবাইট", "base": "বেস", "deca": "ডেকা-", "hecto": "হেক্টো-", "kilo": "কিলো-", "mega": "মেগা-", "giga": "গিগা-", "tera": "টেরা-", "peta": "পেটা-", "exa": "এক্সা-", "zetta": "জেটা-", "yotta": "যোগ্টা-", "deci": "ডেসি-", "centi": "সেন্টি-", "milli": "মিলি-", "micro": "মাইক্রো-", "nano": "ন্যানো-", "pico": "পিকো-", "femto": "ফেমটো-", "atto": "অ্যাটো-", "zepto": "জেপ্টো-", "yocto": "ইওক্টো-" } ================================================ FILE: packages/translations/lib/l10n/app_ca.arb ================================================ { "@@locale": "ca", "appName": "Converter NOW", "reorder": "Reordena", "myOrdering": "Ordenació personal", "about": "Quant a", "lastCurrenciesUpdate": "Última actualització: ", "today": "avui", "enableDarkTheme": "Activa el tema fosc", "settings": "Configuració", "menu": "Menú", "donation": "Feu una donació", "buyMeACoffee": "Convideu-me a un cafè", "donationDialog": "Déu lo guard. Com deu saber aquesta aplicació és gratuïta i de codi obert. Això significa que podeu utilitzar-la gratuïtament i copiar-ne i editar-ne el codi font. El desenvolupador no hi guanya res. Però si creieu en aquest projecte i us agradaria veure funcionalitats noves, em podeu pagar un cafè.", "drawerLogo": "Logo del menú", "rateApp": "Valoreu l'aplicació", "repoGithub": "Obre el repositori a GitHub", "contributeTranslating": "Contribuïu traduint l'aplicació", "search": "Cerca", "clearAll": "Neteja-ho tot", "calculator": "Calculadora", "significantFigures": "Xifres significatives", "removeTrailingZeros": "Elimina els zeros del final", "longPressAdvice": "Prem llarg per moure ítems", "back": "Enrere", "save": "Desa", "invalidCharacters": "Error, caràcters no vàlids", "theme": "Tema", "dark": "Fosc", "light": "Clar", "system": "Predeterminat del sistema", "pureBlackTheme": "Tema fosc AMOLED", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "Copia", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "Més", "language": "Idioma", "reorderProperties": "Reordena les propietats", "reorderUnits": "Reordena les unitats", "chooseProperty": "Tria una propietat", "reorderProperty": "Reordena {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW per altres plataformes", "sourceCode": "Codi font", "undoClearAllMessage": "S'ha netejat tot. Voleu desfer-ho?", "undo": "Desfés", "routeError1": "Aquesta pàgina no existeix", "routeError2": "Torna a la primera pàgina", "ok": "Accepta", "revokeInternetAccess": "Revoca l'accés a internet", "revokeInternetExplanation": "Aquesta aplicació utilitza internet per actualitzar els tipus de canvi de divises una vegada al dia (quan està oberta). Són només uns quants kilobytes de descàrrega, però si no necessiteu aquesta funció podeu impedir que l'aplicació accedeixi a internet activant aquesta opció.", "useDeviceColor": "Use device color", "@useDeviceColor": { "description": "Not yet translated. Once done, remove this comment" }, "pickColor": "Pick a color", "@pickColor": { "description": "Not yet translated. Once done, remove this comment" }, "themeColor": "Theme color", "@themeColor": { "description": "Not yet translated. Once done, remove this comment" }, "appearance": "Appearance", "@appearance": { "description": "Not yet translated. Once done, remove this comment" }, "conversions": "Conversions", "@conversions": { "description": "Not yet translated. Once done, remove this comment" }, "findOutMore": "Find out more", "@findOutMore": { "description": "Not yet translated. Once done, remove this comment" }, "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "description": "Not yet translated. Once done, remove this comment", "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "Longitud", "area": "Àrea", "volume": "Volum", "time": "Temps", "temperature": "Temperatura", "speed": "Velocitat", "siPrefixes": "Prefixos SI", "mass": "Massa", "pressure": "Pressió", "energy": "Energia", "angles": "Angles", "currencies": "Monedes", "shoeSize": "Talla de sabates", "digitalData": "Dades digitals", "power": "Potència", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "Torsió", "force": "Força", "fuelConsumption": "Consum de combustible", "numeralSystems": "Sistemes numèrics", "meters": "Metres", "centimeters": "Centímetres", "inches": "Polzades", "feet": "Peus", "nauticalMiles": "Milles nàutiques", "yards": "Iardes", "miles": "Milles", "millimeters": "Mil·límetres", "micrometers": "Micròmetres", "nanometers": "Nanòmetres", "angstroms": "Ångström", "picometers": "Picòmetres", "kilometers": "Quilòmetres", "astronomicalUnits": "Unitats astronòmiques", "lightYears": "Anys llum", "parsec": "Parsec", "mils": "Mils", "feetUsSurvey": "Peus (mesura dels Estats Units)", "squareMeters": "Metres quadrats", "squareCentimeters": "Centímetres quadrats", "squareInches": "Polzades quadrades", "squareFeet": "Peus quadrats", "squareMiles": "Milles quadrades", "squareYard": "Iardes quadrades", "squareMillimeters": "Mil·límetres quadrats", "squareKilometers": "Quilòmetres quadrats", "hectares": "Hectàrees", "acres": "Acres", "are": "Àrees", "squareFeetUsSurvey": "Peus quadrats (mesura dels Estats Units)", "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "Metres cúbics", "liters": "Litres", "imperialGallons": "Galons imperials", "usGallons": "Galons dels Estats Units", "imperialPints": "Pintes imperials", "usPints": "Pintes dels Estats Units", "milliliters": "Mil·lilitres", "tablespoonUs": "Cullerades dels Estats Units", "tablespoonAustralian": "Cullerades australianes", "cups": "Tasses", "cubicCentimeters": "Centímetres cúbics", "cubicFeet": "Peus cúbics", "cubicInches": "Polzades cúbiques", "cubicMillimeters": "Mil·límetres cúbics", "imperialFluidOunces": "Unces líquides imperials", "usFluidOunces": "Unces líquides dels Estats Units", "imperialGill": "Gill imperial", "usGill": "Gill dels Estats Units", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "Segons", "deciseconds": "Decisegons", "centiseconds": "Centisegons", "milliseconds": "Mil·lisegons", "microseconds": "Microsegons", "nanoseconds": "Nanosegons", "minutes": "Minuts", "hours": "Hores", "days": "Dies", "weeks": "Setmanes", "years": "Anys (365)", "lustrum": "Llustres", "decades": "Dècades", "centuries": "Centúries", "millennium": "Mil·lennis", "fahrenheit": "Fahrenheit", "celsius": "Celsius (centígrads)", "kelvin": "Kelvin", "reamur": "Réaumur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Metres per segon", "kilometersHour": "Quilòmetres per hora", "milesHour": "Milles per hora", "knots": "Nusos", "feetSecond": "Peus per segon", "minutesPerKilometer": "Minuts per quilòmetre", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "Grams", "ettograms": "Ettograms", "kilograms": "Quilograms", "pounds": "Lliures", "ounces": "Unces", "quintals": "Quintals", "tonnes": "Tones", "milligrams": "Mil·ligrams", "uma": "Unitat de massa atòmica unificada", "carats": "Quirats", "centigrams": "Centigrams", "pennyweights": "Pennyweights", "troyOunces": "Unces troy", "stones": "Pedres", "pascal": "Pascal", "atmosphere": "Atmosfera", "bar": "Bar", "millibar": "Mil·libar", "psi": "Lliures per polzada quadrada", "torr": "Torricelli", "hectoPascal": "Hectopascal", "inchesOfMercury": "Polzades de mercuri", "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "Joule", "calories": "Calories", "kilowattHour": "Quilowatt hora", "electronvolt": "Electró-volt", "footPound": "Peu lliura", "kilocalories": "Quilocalories", "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "Graus", "minutesDegree": "Minuts", "secondsDegree": "Segons", "radiansDegree": "Radians", "usd": "Dòlar dels Estats Units", "eur": "Euro", "gbp": "Lliura esterlina", "inr": "Rupia índia", "cny": "Iuan xinès", "jpy": "Ien japonès", "chf": "Franc suís", "sek": "Corona sueca", "rub": "Ruble rus", "cad": "Dòlar canadenc", "krw": "Won sud-coreà", "brl": "Real brasiler", "hkd": "Dòlar de Hong Kong", "aud": "Dòlar australià", "nzd": "Dòlar neozelandès", "mxn": "Peso mexicà", "sgd": "Dòlar de Singapur", "nok": "Corona noruega", "trY": "Lira turca", "zar": "Rand sud-africà", "dkk": "Corona danesa", "pln": "Złoty polonès", "thb": "Baht tailandès", "myr": "Ringgit malai", "huf": "Fòrint hongarès", "czk": "Corona txeca", "ils": "Xéquel israelià", "idr": "Rupia indonèsia", "php": "Peso filipí", "ron": "Leu romanès", "isk": "Króna islandesa", "twd": "Nou dòlar taiwanès", "mad": "Dírham marroquí", "euChina": "UE i Xina", "ukIndiaChild": "Regne Unit i Índia - Infantil", "ukIndiaMan": "Regne Unit i Índia - Home", "ukIndiaWoman": "Regne Unit i Índia - Dona", "usaCanadaChild": "Estats Units i Canadà - Infantil", "usaCanadaMan": "Estats Units i Canadà - Home", "usaCanadaWoman": "Estats Units i Canadà - Dona", "japan": "Japó", "watt": "Watt", "milliwatt": "Mil·liwatt", "kilowatt": "Quilowatt", "megawatt": "Megawatt", "gigawatt": "Gigawatt", "europeanHorsePower": "Cavalls de força europeus", "imperialHorsePower": "Cavalls de força imperials", "newton": "Newton", "dyne": "Dyne", "poundForce": "Lliura-força", "kilogramForce": "Quilogram-força", "poundal": "Poundal", "newtonMeter": "Newton metre", "dyneMeter": "Dyne metre", "poundForceFeet": "Peu-lliura força", "kilogramForceMeter": "Quilogram-força metre", "poundalMeter": "Poundal metre", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "Quilòmetres per litre", "liters100km": "Litres per 100 km", "milesUsGallon": "Milles per galó dels Estats Units", "milesImperialGallon": "Milles per galó imperial", "decimal": "Decimal", "hexadecimal": "Hexadecimal", "octal": "Octal", "binary": "Binari", "bit": "Bit", "nibble": "Nibble", "kilobit": "Kilobit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "Megabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "Gigabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "Terabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "Petabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "Exabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "Gibibit", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Exbibit", "byte": "Byte", "kilobyte": "Kilobyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "Megabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "Gigabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "Terabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "Petabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "Exabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "Kibibyte", "mebibyte": "Mebibyte", "gibibyte": "Gibibyte", "tebibyte": "Tebibyte", "pebibyte": "Pebibyte", "exbibyte": "Exbibyte", "base": "Base", "deca": "Deca-", "hecto": "Hecto-", "kilo": "Kilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Exa-", "zetta": "Zetta-", "yotta": "Yotta-", "deci": "Deci-", "centi": "Centi-", "milli": "Milli-", "micro": "Micro-", "nano": "Nano-", "pico": "Pico-", "femto": "Femto-", "atto": "Atto-", "zepto": "Zepto-", "yocto": "Yocto-" } ================================================ FILE: packages/translations/lib/l10n/app_de.arb ================================================ { "@@locale": "de", "appName": "Converter NOW", "reorder": "Anordnen", "myOrdering": "Meine Anordnung", "about": "Über die App", "lastCurrenciesUpdate": "Letzte Aktualisierung: ", "today": "Heute", "enableDarkTheme": "Dunkles Thema aktivieren", "settings": "Einstellungen", "menu": "Menü", "donation": "Unterstüze die App mit einer Spende.", "buyMeACoffee": "Kaufe dem Entwickler einen Kaffee.", "donationDialog": "Hallo, wie Du vielleicht weißt, ist diese App frei und open-source. Das bedeutet, dass die App kostenlos verwendet und der Quellcode kopiert/bearbeitet werden darf. Der Entwickler wird dafür nicht entlohnt. Wenn Du aber an dieses Projekt glaubst und gerne neue Features sehen möchtest, kaufe dem Entwickler einen Kaffee.", "drawerLogo": "Logo anzeigen", "rateApp": "App bewerten", "repoGithub": "Öffne das Repo auf GitHub.", "contributeTranslating": "Unterstütze uns bei der Übersetzung der App.", "search": "Suche", "clearAll": "Leere alles", "calculator": "Rechner", "significantFigures": "Signifikante Stellen", "removeTrailingZeros": "Entferne nachgestellte Nullen.", "longPressAdvice": "Langes Drücken zum Verschieben.", "back": "Zurück", "save": "Speichern", "invalidCharacters": "Fehler, unzulässige Zeichen.", "theme": "Thema", "dark": "Dunkel", "light": "Hell", "system": "System", "pureBlackTheme": "schwarzes AMOLED-Thema", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "Kopieren", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "Mehr", "language": "Sprache", "reorderProperties": "Reihenfolge der Kategorien ändern", "reorderUnits": "Reihenfolge innerhalb der Kategorien ändern", "chooseProperty": "Eigenschaft wählen", "reorderProperty": "{property} anordnen", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW für andere Plattformen", "sourceCode": "Quellcode", "undoClearAllMessage": "Alles geleert. Rückgängig machen?", "undo": "Rückgängig machen", "routeError1": "Diese Seite existiert nicht.", "routeError2": "Zurück zur ersten Seite gehen", "ok": "Ok", "@ok": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetAccess": "Revoke internet access", "@revokeInternetAccess": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetExplanation": "This app uses the internet to update currency exchange rates once a day upon opening. This download only requires a few kilobytes. However, if you don't need this feature, you can prevent the app from accessing the internet by enabling this option.", "@revokeInternetExplanation": { "description": "Not yet translated. Once done, remove this comment" }, "useDeviceColor": "Use device color", "@useDeviceColor": { "description": "Not yet translated. Once done, remove this comment" }, "pickColor": "Pick a color", "@pickColor": { "description": "Not yet translated. Once done, remove this comment" }, "themeColor": "Theme color", "@themeColor": { "description": "Not yet translated. Once done, remove this comment" }, "appearance": "Appearance", "@appearance": { "description": "Not yet translated. Once done, remove this comment" }, "conversions": "Conversions", "@conversions": { "description": "Not yet translated. Once done, remove this comment" }, "findOutMore": "Find out more", "@findOutMore": { "description": "Not yet translated. Once done, remove this comment" }, "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "description": "Not yet translated. Once done, remove this comment", "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "Länge", "area": "Fläche", "volume": "Volumen", "time": "Zeit", "temperature": "Temperatur", "speed": "Geschwindigkeit", "siPrefixes": "SI-Präfix", "mass": "Masse", "pressure": "Druck", "energy": "Energie", "angles": "Winkel", "currencies": "Währungen", "shoeSize": "Schuhgröße", "digitalData": "Digitale Daten", "power": "Leistung", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "Drehmoment", "force": "Kraft", "fuelConsumption": "Spritverbrauch", "numeralSystems": "Zahlensysteme", "meters": "Meter", "centimeters": "Zentimeter", "inches": "Zoll", "feet": "Fuß", "nauticalMiles": "Nautische Meilen (Seemeile)", "yards": "Yard", "miles": "Meilen", "millimeters": "Millimeter", "micrometers": "Mikrometer", "nanometers": "Nanometer", "angstroms": "Ångström", "picometers": "Pikometer", "kilometers": "Kilometer", "astronomicalUnits": "Astronomische Einheiten", "lightYears": "Lichtjahre", "parsec": "Parsec", "mils": "Milli-Zoll (Thou)", "feetUsSurvey": "Fuß (US survey)", "squareMeters": "Quadratmeter", "squareCentimeters": "Quadratzentimeter", "squareInches": "Quadratzoll", "squareFeet": "Quadratfuß", "squareMiles": "Quadratmeilen", "squareYard": "Quadratyard", "squareMillimeters": "Quadratmillimeter", "squareKilometers": "Quadratkilometer", "hectares": "Hektar", "acres": "Acre", "are": "Ar / Are", "squareFeetUsSurvey": "Quadratfuß (US survey)", "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "Kubikmeter", "liters": "Liter", "imperialGallons": "Gallonen (Imperial)", "usGallons": "Gallonen US", "imperialPints": "Pints (GB und Irland)", "usPints": "Pints US", "milliliters": "Milliliter", "tablespoonUs": "Teelöffel US", "tablespoonAustralian": "Teelöffel Australien", "cups": "Cups US", "cubicCentimeters": "Kubikzentimeter", "cubicFeet": "Kubikfuß", "cubicInches": "Kubikzoll", "cubicMillimeters": "Kubikmillimeter", "imperialFluidOunces": "Flüssigunzen (Imperial)", "usFluidOunces": "US-Flüssigunzen", "imperialGill": "Gill (Imperial)", "usGill": "US-Gill", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "Sekunden", "deciseconds": "Dezisekunden", "centiseconds": "Zentisekunden", "milliseconds": "Millisekunden", "microseconds": "Mikrosekunden", "nanoseconds": "Nanosekunden", "minutes": "Minuten", "hours": "Stunden", "days": "Tage", "weeks": "Wochen", "years": "Jahre (365)", "lustrum": "Jahrfünft", "decades": "Jahrzehnte", "centuries": "Jahrhunderte", "millennium": "Jahrtausende", "fahrenheit": "Fahrenheit", "celsius": "Grad Celsius", "kelvin": "Kelvin", "reamur": "Grad Réaumur", "romer": "Grad Rømer", "delisle": "Grad Delisle", "rankine": "Grad Rankine", "metersSecond": "Meter pro Sekunde", "kilometersHour": "Kilometer pro Stunde", "milesHour": "Meilen pro Stunde", "knots": "Knoten", "feetSecond": "Fuß pro Sekunde", "minutesPerKilometer": "Minuten pro Kilometer", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "Gramm", "ettograms": "Hektogramm", "kilograms": "Kilogramm", "pounds": "Pfund (imperial)", "ounces": "Unzen (imperial)", "quintals": "Zentner / Quintal (100 kg)", "tonnes": "Tonnen", "milligrams": "Milligramm", "uma": "Atomare Masseneinheiten", "carats": "Metrisches Karat", "centigrams": "Zentigramm", "pennyweights": "Pennyweights", "troyOunces": "Feinunze", "stones": "Stone", "pascal": "Pascal", "atmosphere": "Physikalische Atmosphäre", "bar": "Bar", "millibar": "Millibar", "psi": "Pfund pro Quadratzoll", "torr": "Millimeter-Quecksilbersäule (Torr)", "hectoPascal": "Hektopascal", "inchesOfMercury": "Zoll Quecksilber", "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "Joule", "calories": "Kalorien", "kilowattHour": "Kilowattstunden", "electronvolt": "Elektronenvolt", "footPound": "Foot-Pound", "kilocalories": "Kilokalorien", "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "Grad", "minutesDegree": "Minuten", "secondsDegree": "Sekunden", "radiansDegree": "Radiant", "usd": "US-Dollar", "eur": "Euro", "gbp": "Britisches Pfund", "inr": "Indische Rupie", "cny": "Renminbi Yuan", "jpy": "Japanischer Yen", "chf": "Schweizer Franken", "sek": "Schwedische Krone", "rub": "Russischer Rubel", "cad": "Kanadischer Dollar", "krw": "Südkoreanischer Won", "brl": "Brasilianischer Real", "hkd": "Hongkong-Dollar", "aud": "Australischer Dollar", "nzd": "Neuseeland-Dollar", "mxn": "Mexikanischer Peso", "sgd": "Singapur-Dollar", "nok": "Norwegische Krone", "trY": "Türkische Lira", "zar": "Südafrikanischer Rand", "dkk": "Dänische Krone", "pln": "Polnischer Złoty", "thb": "Thailändischer Baht", "myr": "Malaysischer Ringgit", "huf": "Ungarischer Forint", "czk": "Tschechische Krone", "ils": "Israelischer Neuer Schekel", "idr": "Indonesische Rupiah", "php": "Philippinischer Peso", "ron": "Rumänischer Leu", "isk": "Isländische Krone", "twd": "Neuer Taiwan-Dollar", "mad": "Marokkanischer Dirham", "euChina": "EU & China", "ukIndiaChild": "UK & Indien – Kind", "ukIndiaMan": "UK & Indien – Mann", "ukIndiaWoman": "UK & Indien – Frau", "usaCanadaChild": "USA & Kanada – Kind", "usaCanadaMan": "USA & Kanada – Mann", "usaCanadaWoman": "USA & Kanada – Frau", "japan": "Japan", "watt": "Watt", "milliwatt": "Milliwatt", "kilowatt": "Kilowatt", "megawatt": "Megawatt", "gigawatt": "Gigawatt", "europeanHorsePower": "Europäische Pferdestärken", "imperialHorsePower": "Imperiale Pferdestärken", "newton": "Newton", "dyne": "Dyn", "poundForce": "Kraftpfund", "kilogramForce": "Kilopond", "poundal": "Poundal", "newtonMeter": "Newtonmeter", "dyneMeter": "Dynmeter", "poundForceFeet": "Pound-Force-Feet", "kilogramForceMeter": "Kilopondmeter", "poundalMeter": "Poundalmeter", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "Kilometer pro Liter", "liters100km": "Liter pro 100 km", "milesUsGallon": "Meilen pro US-Gallone", "milesImperialGallon": "Meilen pro Gallone (Imperial)", "decimal": "Dezimal", "hexadecimal": "Hexadezimal", "octal": "Oktal", "binary": "Binär", "bit": "Bit", "nibble": "Nibble", "kilobit": "Kilobit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "Megabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "Gigabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "Terabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "Petabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "Exabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "Gibibit", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Exbibit", "byte": "Byte", "kilobyte": "Kilobyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "Megabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "Gigabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "Terabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "Petabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "Exabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "Kibibyte", "mebibyte": "Mebibyte", "gibibyte": "Gibibyte", "tebibyte": "Tebibyte", "pebibyte": "Pebibyte", "exbibyte": "Exbibyte", "base": "Basis", "deca": "Deka-", "hecto": "Hekto-", "kilo": "Kilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Exa-", "zetta": "Zetta-", "yotta": "Yotta-", "deci": "Dezi-", "centi": "Zenti-", "milli": "Milli-", "micro": "Mikro-", "nano": "Nano-", "pico": "Piko-", "femto": "Femto-", "atto": "Atto-", "zepto": "Zepto-", "yocto": "Yokto-" } ================================================ FILE: packages/translations/lib/l10n/app_el.arb ================================================ { "@@locale": "el", "appName": "Converter NOW", "reorder": "Αναδιάταξη", "myOrdering": "Η διάταξή μου", "about": "Σχετικά", "lastCurrenciesUpdate": "Τελευταία ενημέρωση: ", "today": "σήμερα", "enableDarkTheme": "Ενεργοποίηση σκοτεινού θέματος", "settings": "Ρυθμίσεις", "menu": "Μενού", "donation": "Κάντε μια δωρεά", "buyMeACoffee": "Αγόρασέ μου έναν καφέ", "donationDialog": "Γεια σου! Όπως ίσως γνωρίζεις, αυτή η εφαρμογή είναι δωρεάν και ανοιχτού κώδικα. Αυτό σημαίνει ότι μπορείς να τη χρησιμοποιήσεις δωρεάν και να αντιγράψεις ή να επεξεργαστείς τον κώδικα. Ο προγραμματιστής δεν θα κερδίσει τίποτα από αυτό. Αλλά αν πιστεύεις επίσης σε αυτό το έργο και θα ήθελες να δεις νέες λειτουργίες, σε παρακαλώ σκέψου το ενδεχόμενο να μου αγοράσεις έναν καφέ.", "drawerLogo": "Λογότυπο του μενού", "rateApp": "Αξιολόγησε την εφαρμογή", "repoGithub": "Άνοιξε το αποθετήριο στο GitHub", "contributeTranslating": "Συνεισφέρετε μεταφράζοντας την εφαρμογή", "search": "Αναζήτηση", "clearAll": "Καθαρισμός όλων", "calculator": "Υπολογιστής", "significantFigures": "Δεκαδικά ψηφία", "removeTrailingZeros": "Αφαίρεση μη σημαντικών μηδενικών", "longPressAdvice": "Πατήστε παρατεταμένα για να μετακινήσετε στοιχεία", "back": "Πίσω", "save": "Αποθήκευση", "invalidCharacters": "Σφάλμα, μη έγκυροι χαρακτήρες", "theme": "Θέμα", "dark": "Σκοτεινό", "light": "Φωτεινό", "system": "Προεπιλογή συστήματος", "pureBlackTheme": "AMOLED σκοτεινό θέμα", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "Αντιγραφή", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "Περισσότερα", "language": "Γλώσσα", "reorderProperties": "Αναδιάταξη ιδιοτήτων", "reorderUnits": "Αναδιάταξη μονάδων", "chooseProperty": "Επίλεξε μια ιδιότητα", "reorderProperty": "Αναδιάταξη {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW for other platforms", "sourceCode": "Πηγαίος κώδικας", "undoClearAllMessage": "Καθαρίστηκαν όλα. Θέλεις να το αναιρέσεις;", "undo": "Αναίρεση", "routeError1": "Αυτή η σελίδα δεν υπάρχει", "routeError2": "Επιστροφή στην πρώτη σελίδα", "ok": "Ok", "revokeInternetAccess": "Αναίρεση πρόσβασης στο διαδίκτυο", "revokeInternetExplanation": "Αυτή η εφαρμογή χρησιμοποιεί το διαδίκτυο για να ενημερώνει τις τιμές συναλλάγματος μία φορά την ημέρα κατά την εκκίνηση. Αυτή η λήψη απαιτεί μόνο μερικά kilobytes. Ωστόσο, αν δεν χρειάζεστε αυτή τη δυνατότητα, μπορείτε να αποτρέψετε την εφαρμογή από το να έχει πρόσβαση στο διαδίκτυο ενεργοποιώντας αυτή την επιλογή.", "useDeviceColor": "Χρήση χρώματος συσκευής", "pickColor": "Διάλεξε ένα χρώμα", "themeColor": "Χρώμα θέματος", "appearance": "Εμφάνιση", "conversions": "Μετατροπές", "findOutMore": "Μάθε περισσότερα", "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "description": "Not yet translated. Once done, remove this comment", "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "Μήκος", "area": "Εμβαδόν", "volume": "Όγκος", "time": "Χρόνος", "temperature": "Θερμοκρασία", "speed": "Ταχύτητα", "siPrefixes": "Πρόθεμα SI", "mass": "Μάζα", "pressure": "Πίεση", "energy": "Ενέργεια", "angles": "Γωνίες", "currencies": "Νομίσματα", "shoeSize": "Μέγεθος παπουτσιού", "digitalData": "Ψηφιακά δεδομένα", "power": "Ισχύς", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "Ροπή", "force": "Δύναμη", "fuelConsumption": "Κατανάλωση καυσίμου", "numeralSystems": "Αριθμητικά συστήματα", "meters": "Μέτρα", "centimeters": "Εκατοστά", "inches": "Ίντσες", "feet": "Πόδια", "nauticalMiles": "Ναυτικά μίλια", "yards": "Γιάρδες", "miles": "Μίλια", "millimeters": "Χιλιοστά", "micrometers": "Μικρόμετρα", "nanometers": "Νανόμετρα", "angstroms": "Άνγκστρεμ", "picometers": "Πικόμετρα", "kilometers": "Χιλιόμετρα", "astronomicalUnits": "Αστρονομικές Μονάδες", "lightYears": "Έτη φωτός", "parsec": "Πάρσεκ", "mils": "Μιλς", "feetUsSurvey": "Πόδια (τοπογραφικά ΗΠΑ)", "squareMeters": "Τετραγωνικά Μέτρα", "squareCentimeters": "Τετραγωνικά Εκατοστά", "squareInches": "Τετραγωνικές Ίντσες", "squareFeet": "Τετραγωνικά Πόδια", "squareMiles": "Τετραγωνικά Μίλια", "squareYard": "Τετραγωνική Γιάρδα", "squareMillimeters": "Τετραγωνικά Χιλιοστά", "squareKilometers": "Τετραγωνικά Χιλιόμετρα", "hectares": "Εκτάρια", "acres": "Acres", "are": "Άρι", "squareFeetUsSurvey": "Τετραγωνικά Πόδια (τοπογραφικά ΗΠΑ)", "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "Κυβικά Μέτρα", "liters": "Λίτρα", "imperialGallons": "Imperial gallons", "usGallons": "US Gallons", "imperialPints": "Imperial Pints", "usPints": "US Pints", "milliliters": "Χιλιοστόλιτρα", "tablespoonUs": "Κουταλιές της σούπας US", "tablespoonAustralian": "Αυστραλιανές κουταλιές τραπεζής", "cups": "Φλιτζάνια", "cubicCentimeters": "Κυβικά Εκατοστά", "cubicFeet": "Κυβικά Πόδια", "cubicInches": "Κυβικές Ίντσες", "cubicMillimeters": "Κυβικά Χιλιοστά", "imperialFluidOunces": "Imperial Oυγγιά Yγρών", "usFluidOunces": "US Oυγγιά Yγρών", "imperialGill": "Imperial Gill", "usGill": "US Gill", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "Δευτερόλεπτα", "deciseconds": "Δεκατοστά του δευτερολέπτου", "centiseconds": "Εκατοστά του δευτερολέπτου", "milliseconds": "Χιλιοστά του δευτερολέπτου", "microseconds": "Μικροδευτερόλεπτα", "nanoseconds": "Νανοδευτερόλεπτα", "minutes": "Λεπτά", "hours": "Ώρες", "days": "Μέρες", "weeks": "Εβδομάδες", "years": "Χρόνια (365)", "lustrum": "Λούστρον", "decades": "Δεκαετίες", "centuries": "Αιώνες", "millennium": "Χιλιετία", "fahrenheit": "Φαρενάιτ", "celsius": "Κελσίου (Centigrade)", "kelvin": "Κέλβιν", "reamur": "Reamur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Μέτρα ανά Δευτερόλεπτο", "kilometersHour": "Χιλιόμετρα ανά Ώρα", "milesHour": "Μίλια ανά Ώρα", "knots": "Κόμβοι", "feetSecond": "Πόδια ανά Δευτερόλεπτο", "minutesPerKilometer": "Λεπτά ανά Χιλιόμετρο", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "Γραμμάρια", "ettograms": "Ettograms", "kilograms": "Κιλά", "pounds": "Λίβρες", "ounces": "Ουγκιές", "quintals": "Κουιντάλια", "tonnes": "Τόνοι", "milligrams": "Χιλιοστόγραμμα", "uma": "Ενιαία ατομική μονάδα μάζας", "carats": "Καράτια", "centigrams": "Εκατοστόγραμμα", "pennyweights": "Pennyweights", "troyOunces": "Troy ounces", "stones": "Λίθοι", "pascal": "Πασκάλ", "atmosphere": "Ατμόσφαιρα", "bar": "Μπάρ", "millibar": "Μιλιμπάρ", "psi": "Λίβρες ανά τετραγωνική ίντσα", "torr": "Torricelli", "hectoPascal": "Hectopascal", "inchesOfMercury": "Ίντσες υδραργύρου", "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "Τζάουλ", "calories": "Θερμίδες", "kilowattHour": "Κιλοβατώρα", "electronvolt": "Ηλεκτρονιοβόλτ", "footPound": "Ποδόλιβρα", "kilocalories": "Κιλοθερμίδες", "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "Βαθμός", "minutesDegree": "Λεπτά", "secondsDegree": "Δευτερόλεπτα", "radiansDegree": "Ακτίνια", "usd": "Δολάριο ΗΠΑ", "eur": "Ευρώ", "gbp": "Βρετανική Λίρα", "inr": "Ινδική Ρουπία", "cny": "Κινεζικό Γιουάν", "jpy": "Ιαπωνικό Γιεν", "chf": "Ελβετικό Φράγκο", "sek": "Σουηδική Κορόνα", "rub": "Ρωσικό Ρούβλι", "cad": "Καναδικό Δολάριο", "krw": "Νότιο Κορεατικό Γουόν", "brl": "Βραζιλιάνικο Ρεάλ", "hkd": "Δολάριο Χονγκ Κονγκ", "aud": "Αυστραλιανό Δολάριο", "nzd": "Νεοζηλανδικό Δολάριο", "mxn": "Μεξικάνικο Πέσο", "sgd": "Δολάριο Σιγκαπούρης", "nok": "Νορβηγική Κορόνα", "trY": "Τουρκική Λίρα", "zar": "Νότιο Αφρικανικό Ραντ", "dkk": "Δανέζικη Κορόνα", "pln": "Πολωνικό Ζλότι", "thb": "Ταϊλανδέζικο Μπάτ", "myr": "Μαλαισιανό Ρινγκίτ", "huf": "Ουγγρικό Φορίντ", "czk": "Τσέχικη Κορόνα", "ils": "Ισραηλινό Σέκελ", "idr": "Ινδονησιακή Ρουπία", "php": "Φιλιππινέζικο Πέσο", "ron": "Ρουμανικό Λέι", "isk": "Ισλανδική Κρόνα", "twd": "Νέο Δολάριο Ταϊβάν", "mad": "Μαροκινό Ντιρχάμ", "euChina": "ΕΕ & Κίνα", "ukIndiaChild": "Ηνωμένο Βασίλειο & Ινδία - Παιδί", "ukIndiaMan": "Ηνωμένο Βασίλειο & Ινδία - Άνδρας", "ukIndiaWoman": "Ηνωμένο Βασίλειο & Ινδία - Γυναίκα", "usaCanadaChild": "ΗΠΑ & Καναδάς - Παιδί", "usaCanadaMan": "ΗΠΑ & Καναδάς - Άνδρας", "usaCanadaWoman": "ΗΠΑ & Καναδάς - Γυναίκα", "japan": "Ιαπωνία", "watt": "Βατ", "milliwatt": "Μιλιβάτ", "kilowatt": "Κιλοβάτ", "megawatt": "Μεγαβάτ", "gigawatt": "Γιγαβάτ", "europeanHorsePower": "Ευρωπαϊκή ιπποδύναμη", "imperialHorsePower": "Αυτοκρατορική ιπποδύναμη", "newton": "Νιούτον", "dyne": "Dyne", "poundForce": "Λίβρα-δύναμη", "kilogramForce": "Κιλόγραμμα-δύναμη", "poundal": "Πουντάλ", "newtonMeter": "Νιούτον μέτρο", "dyneMeter": "Δύνα μέτρο", "poundForceFeet": "Λίβρα-δύναμη πόδια", "kilogramForceMeter": "Κιλόγραμμα-δύναμη μέτρο", "poundalMeter": "Πουντάλ μέτρο", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "Χιλιόμετρα ανά λίτρο", "liters100km": "Λίτρα ανά 100 χιλιόμετρα", "milesUsGallon": "Μίλια ανά Γαλόνι ΗΠΑ", "milesImperialGallon": "Μίλια ανά Αυτοκρατορικό Γαλόνι", "decimal": "Δεκαδικός", "hexadecimal": "Εξαδικός", "octal": "Οκταδικός", "binary": "Δυαδικός", "bit": "Μπιτ", "nibble": "Νίμπλ", "kilobit": "Κιλομπίτ", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "Μεγαμπίτ", "@megabit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "Γιγαμπίτ", "@gigabit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "Τεραμπίτ", "@terabit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "Πεταμπίτ", "@petabit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "Έξαμπίτ", "@exabit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "Κιμπιμπίτ", "mebibit": "Μεμπιμπίτ", "gibibit": "Γκιμπιμπίτ", "tebibit": "Τεμπιμπίτ", "pebibit": "Πεμπιμπίτ", "exbibit": "Εξμπιμπίτ", "byte": "Μπάιτ", "kilobyte": "Κιλομπάιτ", "@kilobyte": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "Μεγκαμπάιτ", "@megabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "Γκιγκαμπάιτ", "@gigabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "Τεραμπάιτ", "@terabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "Πεταμπάιτ", "@petabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "Εξαμπάιτ", "@exabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "Κιμπιμπάιτ", "mebibyte": "Μεμπιμπάιτ", "gibibyte": "Γκιμπιμπάιτ", "tebibyte": "Τεμπιμπάιτ", "pebibyte": "Πεμπιμπάιτ", "exbibyte": "Εξμπιμπάιτ", "base": "Βάση", "deca": "Δέκα-", "hecto": "Εκτό-", "kilo": "Κιλο-", "mega": "Μέγα-", "giga": "Γίγα-", "tera": "Τέρα-", "peta": "Πέτα-", "exa": "Έξα-", "zetta": "Ζέτα-", "yotta": "Γιότα-", "deci": "Δέκατο-", "centi": "Εκατοστό-", "milli": "Χιλιοστό-", "micro": "Μίκρο-", "nano": "Νάνο-", "pico": "Πίκο-", "femto": "Φέμτο-", "atto": "Άτο-", "zepto": "Ζέπτο-", "yocto": "Γιόκτο-" } ================================================ FILE: packages/translations/lib/l10n/app_en.arb ================================================ { "@@locale": "en", "appName": "Converter NOW", "reorder": "Reorder", "myOrdering": "My ordering", "about": "About", "lastCurrenciesUpdate": "Last update: ", "today": "today", "enableDarkTheme": "Enable dark theme", "settings": "Settings", "menu": "Menu", "donation": "Make a donation", "buyMeACoffee": "Buy me a coffee", "donationDialog": "Hi, as you might know this app is free and open source. It means that you can use it for free and copy/edit the source code. The developer will not gain nothing for it. But if you also believe in this project and would like to see new features, please consider buying me a coffee.", "drawerLogo": "Drawer logo", "rateApp": "Rate the app", "repoGithub": "Open repo on GitHub", "contributeTranslating": "Contribute translating the app", "search": "Search", "clearAll": "Clear all", "calculator": "Calculator", "significantFigures": "Significant figures", "removeTrailingZeros": "Remove trailing zeros", "longPressAdvice": "Long press to move items", "back": "Back", "save": "Save", "invalidCharacters": "Error, invalid characters", "theme": "Theme", "dark": "Dark", "light": "Light", "system": "System default", "pureBlackTheme": "Pure black theme", "propertySelectionOnStartup": "Property selection on startup", "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "copy": "Copy", "paste": "Paste", "more": "More", "language": "Language", "reorderProperties": "Reorder properties", "reorderUnits": "Reorder units", "chooseProperty": "Choose a property", "reorderProperty": "Reorder {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW for other platforms", "sourceCode": "Source code", "undoClearAllMessage": "Cleared all. Do you want to undo it?", "undo": "Undo", "routeError1": "This page doesn't exist", "routeError2": "Go back to the first page", "ok": "Ok", "revokeInternetAccess": "Revoke internet access", "revokeInternetExplanation": "This app uses the internet to update currency exchange rates once a day upon opening. This download only requires a few kilobytes. However, if you don't need this feature, you can prevent the app from accessing the internet by enabling this option.", "useDeviceColor": "Use device color", "pickColor": "Pick a color", "themeColor": "Theme color", "appearance": "Appearance", "conversions": "Conversions", "findOutMore": "Find out more", "hideUnits": "Hide units", "visibleUnits": "{property}, visible units", "@visibleUnits": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "selectAll": "Select all", "unselectAll": "Unselect all", "backupAndRestore": "Backup & Restore", "exportSettings": "Export settings", "importSettings": "Import settings", "clearSettings": "Clear all settings", "exportSuccess": "Settings exported successfully", "importSuccess": "Settings imported successfully", "problemImportFile": "There was a problem importing the file", "relatedSettings": "Related settings", "reason": "Reason", "importError": "Import Error", "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "cancel": "Cancel", "length": "Length", "area": "Area", "volume": "Volume", "time": "Time", "temperature": "Temperature", "speed": "Speed", "siPrefixes": "SI Prefix", "mass": "Mass", "pressure": "Pressure", "energy": "Energy", "angles": "Angles", "currencies": "Currencies", "shoeSize": "Shoe size", "digitalData": "Digital data", "power": "Power", "density": "Density", "torque": "Torque", "force": "Force", "fuelConsumption": "Fuel consumption", "numeralSystems": "Numeral systems", "meters": "Meters", "centimeters": "Centimeters", "inches": "Inches", "feet": "Feet", "nauticalMiles": "Nautical miles", "yards": "Yards", "miles": "Miles", "millimeters": "Millimeters", "micrometers": "Micrometers", "nanometers": "Nanometers", "angstroms": "Ångström", "picometers": "Picometers", "kilometers": "Kilometers", "astronomicalUnits": "Astronomical Units", "lightYears": "Light-years", "parsec": "Parsec", "mils": "Mils", "feetUsSurvey": "Feet (US survey)", "squareMeters": "Square Meters", "squareCentimeters": "Square Centimeters", "squareInches": "Square Inches", "squareFeet": "Square Feet", "squareMiles": "Square Miles", "squareYard": "Square Yard", "squareMillimeters": "Square Millimeters", "squareKilometers": "Square Kilometers", "hectares": "Hectares", "acres": "Acres", "are": "Are", "squareFeetUsSurvey": "Square Feet (US survey)", "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "gramsPerLiter": "Grams per Liter", "gramsPerMilliliter": "Grams per Milliliter", "gramsPerDeciliter": "Grams per Deciliter", "kilogramsPerLiter": "Kilograms per Liter", "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "milligramsPerLiter": "Milligrams per Liter", "milligramsPerDeciliter": "Milligrams per Deciliter", "milligramsPerMilliliter": "Milligrams per Milliliter", "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "microgramsPerLiter": "Micrograms per Liter", "microgramsPerDeciliter": "Micrograms per Deciliter", "microgramsPerMilliliter": "Micrograms per Milliliter", "poundsPerCubicInches": "Pounds per Cubic Inches", "poundsPerCubicFeet": "Pounds per Cubic Feet", "cubicMeters": "Cubic Meters", "liters": "Liters", "imperialGallons": "Imperial gallons", "usGallons": "US Gallons", "imperialPints": "Imperial Pints", "usPints": "US Pints", "milliliters": "Milliliters", "tablespoonUs": "Tablespoons US", "tablespoonAustralian": "Australian Tablespoons", "cups": "Cups", "cubicCentimeters": "Cubic Centimeters", "cubicFeet": "Cubic Feet", "cubicInches": "Cubic Inches", "cubicMillimeters": "Cubic Millimeters", "imperialFluidOunces": "Imperial Fluid Ounces", "usFluidOunces": "US Fluid Ounces", "imperialGill": "Imperial Gill", "usGill": "US Gill", "usQuarts": "US Quarts", "microliters": "Microliters", "deciliters": "Deciliters", "centiliters": "Centiliters", "teaspoonsUs": "Teaspoons US", "teaspoonsMetric": "Metric teaspoons", "seconds": "Seconds", "deciseconds": "Deciseconds", "centiseconds": "Centiseconds", "milliseconds": "Milliseconds", "microseconds": "Microseconds", "nanoseconds": "Nanoseconds", "minutes": "Minutes", "hours": "Hours", "days": "Days", "weeks": "Weeks", "years": "Years (365)", "lustrum": "Lustrum", "decades": "Decades", "centuries": "Centuries", "millennium": "Millennium", "fahrenheit": "Fahrenheit", "celsius": "Celsius (Centigrade)", "kelvin": "Kelvin", "reamur": "Reamur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Meters per Second", "kilometersHour": "Kilometers per Hour", "milesHour": "Miles per Hour", "knots": "Knots", "feetSecond": "Feet per Second", "minutesPerKilometer": "Minutes per Kilometer", "minutesPerMile": "Minutes per mile", "speedOfLight": "Speed of light", "grams": "Grams", "ettograms": "Ettograms", "kilograms": "Kilograms", "pounds": "Pounds", "ounces": "Ounces", "quintals": "Quintals", "tonnes": "Tonnes", "milligrams": "Milligrams", "uma": "Unified atomic mass unit", "carats": "Carats", "centigrams": "Centigrams", "pennyweights": "Pennyweights", "troyOunces": "Troy ounces", "stones": "Stones", "pascal": "Pascal", "atmosphere": "Atmosphere", "bar": "Bar", "millibar": "Millibar", "psi": "Pounds per square inch", "torr": "Millimeter of mercury (Torr)", "hectoPascal": "Hectopascal", "inchesOfMercury": "Inches of mercury", "kiloPascal": "Kilopascal", "ksi": "Kilopounds per square inch", "megaPascal": "Megapascal", "gigaPascal": "Gigapascal", "joule": "Joule", "calories": "Calories", "kilowattHour": "Kilowatt hour", "electronvolt": "Electron volt", "footPound": "Foot pound", "kilocalories": "Kilocalories", "kilojoules": "Kilojoules", "watthour": "Watt hour", "britishThermalUnit": "British thermal unit", "degree": "Degree", "minutesDegree": "Minutes", "secondsDegree": "Seconds", "radiansDegree": "Radians", "usd": "US Dollar", "eur": "Euro", "gbp": "British Pound", "inr": "Indian Rupee", "cny": "Chinese Yuan", "jpy": "Japanese Yen", "chf": "Swiss Franc", "sek": "Swedish Krona", "rub": "Russian Ruble", "cad": "Canadian Dollar", "krw": "South Korean Won", "brl": "Brazilian Real", "hkd": "Hong Kong Dollar", "aud": "Australian Dollar", "nzd": "New Zealand Dollar", "mxn": "Mexican Peso", "sgd": "Singapore Dollar", "nok": "Norwegian Krone", "trY": "Turkish Lira", "zar": "South African Rand", "dkk": "Danish Krone", "pln": "Polish Złoty", "thb": "Thai Baht", "myr": "Malaysian Ringgit", "huf": "Hungarian Forint", "czk": "Czech Koruna", "ils": "Israeli Shekel", "idr": "Indonesian Rupiah", "php": "Philippine Peso", "ron": "Romanian Leu", "isk": "Icelandic Króna", "twd": "New Taiwan Dollar", "mad": "Moroccan Dirham", "euChina": "EU & China", "ukIndiaChild": "UK & India - Child", "ukIndiaMan": "UK & India - Man", "ukIndiaWoman": "UK & India - Woman", "usaCanadaChild": "USA & Canada - Child", "usaCanadaMan": "USA & Canada - Man", "usaCanadaWoman": "USA & Canada - Woman", "japan": "Japan", "watt": "Watt", "milliwatt": "Milliwatt", "kilowatt": "Kilowatt", "megawatt": "Megawatt", "gigawatt": "GigaWatt", "europeanHorsePower": "European horsepower", "imperialHorsePower": "Imperial horsepower", "newton": "Newton", "dyne": "Dyne", "poundForce": "Pound-force", "kilogramForce": "Kilogram-force", "poundal": "Poundal", "newtonMeter": "Newton meter", "dyneMeter": "Dyne meter", "poundForceFeet": "Pound-force feet", "kilogramForceMeter": "Kilogram-force meter", "poundalMeter": "Poundal meter", "poundForceInch": "Pound-force inch", "kilometersLiter": "Kilometers per liter", "liters100km": "Liters per 100 km", "milesUsGallon": "Miles per US Gallon", "milesImperialGallon": "Miles per imperial Gallon", "decimal": "Decimal", "hexadecimal": "Hexadecimal", "octal": "Octal", "binary": "Binary", "bit": "Bit", "nibble": "Nibble", "kilobit": "Kilobit (decimal)", "megabit": "Megabit (decimal)", "gigabit": "Gigabit (decimal)", "terabit": "Terabit (decimal)", "petabit": "Petabit (decimal)", "exabit": "Exabit (decimal)", "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "Gibibit", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Exbibit", "byte": "Byte", "kilobyte": "Kilobyte (decimal)", "megabyte": "Megabyte (decimal)", "gigabyte": "Gigabyte (decimal)", "terabyte": "Terabyte (decimal)", "petabyte": "Petabyte (decimal)", "exabyte": "Exabyte (decimal)", "kibibyte": "Kibibyte", "mebibyte": "Mebibyte", "gibibyte": "Gibibyte", "tebibyte": "Tebibyte", "pebibyte": "Pebibyte", "exbibyte": "Exbibyte", "base": "Base", "deca": "Deca-", "hecto": "Hecto-", "kilo": "Kilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Exa-", "zetta": "Zetta-", "yotta": "Yotta-", "deci": "Deci-", "centi": "Centi-", "milli": "Milli-", "micro": "Micro-", "nano": "Nano-", "pico": "Pico-", "femto": "Femto-", "atto": "Atto-", "zepto": "Zepto-", "yocto": "Yocto-" } ================================================ FILE: packages/translations/lib/l10n/app_es.arb ================================================ { "@@locale": "es", "appName": "Converter NOW", "reorder": "Reordenar", "myOrdering": "Mi ordenamiento", "about": "Acerca de", "lastCurrenciesUpdate": "Última actualización: ", "today": "hoy", "enableDarkTheme": "Activar tema oscuro", "settings": "Ajustes", "menu": "Menú", "donation": "Haga una donación", "buyMeACoffee": "Cómpreme un café", "donationDialog": "Hola, como usted sabrá esta aplicación es gratuita y de código abierto. Eso significa que puede usarla sin costo y copiar o editar su código fuente. El desarrollador no ganará nada por ella. Pero si cree en este proyecto y le gustaría ver nuevas características, por favor considere comprarme un café.", "drawerLogo": "Logo en el cajón", "rateApp": "Calificar la aplicación", "repoGithub": "Abrir repositorio en GitHub", "contributeTranslating": "Contribuya traduciendo la aplicación", "search": "Buscar", "clearAll": "Limpiar todo", "calculator": "Calculadora", "significantFigures": "Cifras significativas", "removeTrailingZeros": "Remover decimales periódicos", "longPressAdvice": "Toque largo para remover items", "back": "Atrás", "save": "Guardar", "invalidCharacters": "Error, caracteres inválidos", "theme": "Tema", "dark": "Oscuro", "light": "Claro", "system": "Sistema", "pureBlackTheme": "Tema oscuro AMOLED", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "Copiar", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "Más", "language": "Idioma", "reorderProperties": "Reordenar propiedades", "reorderUnits": "Reordenar unidades de medida", "chooseProperty": "Elija una propiedad", "reorderProperty": "Reordenar {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW para otras plataformas", "sourceCode": "Código fuente", "undoClearAllMessage": "Todo despejado. ¿Desea deshacer?", "undo": "Deshacer", "routeError1": "This page doesn't exist", "@routeError1": { "description": "Not yet translated. Once done, remove this comment" }, "routeError2": "Go back to the first page", "@routeError2": { "description": "Not yet translated. Once done, remove this comment" }, "ok": "Ok", "@ok": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetAccess": "Revoke internet access", "@revokeInternetAccess": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetExplanation": "This app uses the internet to update currency exchange rates once a day upon opening. This download only requires a few kilobytes. However, if you don't need this feature, you can prevent the app from accessing the internet by enabling this option.", "@revokeInternetExplanation": { "description": "Not yet translated. Once done, remove this comment" }, "useDeviceColor": "Use device color", "@useDeviceColor": { "description": "Not yet translated. Once done, remove this comment" }, "pickColor": "Pick a color", "@pickColor": { "description": "Not yet translated. Once done, remove this comment" }, "themeColor": "Theme color", "@themeColor": { "description": "Not yet translated. Once done, remove this comment" }, "appearance": "Appearance", "@appearance": { "description": "Not yet translated. Once done, remove this comment" }, "conversions": "Conversions", "@conversions": { "description": "Not yet translated. Once done, remove this comment" }, "findOutMore": "Find out more", "@findOutMore": { "description": "Not yet translated. Once done, remove this comment" }, "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "Longitud", "area": "Área", "volume": "Volumen", "time": "Tiempo", "temperature": "Temperatura", "speed": "Velocidad", "siPrefixes": "Prefijos SI", "mass": "Masa", "pressure": "Presión", "energy": "Energía", "angles": "Ángulos", "currencies": "Divisas", "shoeSize": "Talla de calzado", "digitalData": "Sistema digital", "power": "Potencia", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "Torque", "force": "Fuerza", "fuelConsumption": "Consumo de combustible", "numeralSystems": "Sistemas de numeración", "meters": "Metros", "centimeters": "Centímetros", "inches": "Pulgadas", "feet": "Pies", "nauticalMiles": "Millas náuticas", "yards": "Yardas", "miles": "Millas", "millimeters": "Milímetros", "micrometers": "Micrómetros", "nanometers": "Nanómetros", "angstroms": "Ángstrom", "picometers": "Picómetros", "kilometers": "Kilómetros", "astronomicalUnits": "Unidades astronómicas", "lightYears": "Años luz", "parsec": "Pársec", "mils": "Mils", "feetUsSurvey": "Feet (US survey)", "@feetUsSurvey": { "description": "Not yet translated. Once done, remove this comment" }, "squareMeters": "Metros cuadrados", "squareCentimeters": "Centímetros cuadrados", "squareInches": "Pulgadas cuadradas", "squareFeet": "Pies cuadrados", "squareMiles": "Millas cuadradas", "squareYard": "Yardas cuadradas", "squareMillimeters": "Milímetros cuadrados", "squareKilometers": "Kilómetros cuadrados", "hectares": "Hectáreas", "acres": "Acres", "are": "Decámetros cuadrados", "squareFeetUsSurvey": "Square Feet (US survey)", "@squareFeetUsSurvey": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "Metros cúbicos", "liters": "Litros", "imperialGallons": "Galones imperiales", "usGallons": "Galones estadounidenses", "imperialPints": "Pintas imperiales", "usPints": "Pintas estadounidenses", "milliliters": "Mililitros", "tablespoonUs": "Cucharada estadounidense", "tablespoonAustralian": "Cucharada australiana", "cups": "Tazas", "cubicCentimeters": "Centímetros cúbicos", "cubicFeet": "Pies cúbicos", "cubicInches": "Pulgadas cúbicas", "cubicMillimeters": "Milímetros cúbicos", "imperialFluidOunces": "Onzas líquidas imperiales", "usFluidOunces": "Onzas líquidas estadounidenses", "imperialGill": "Gill imperial", "usGill": "Gill estadounidense", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "Segundos", "deciseconds": "Decisegundos", "centiseconds": "Centisegundos", "milliseconds": "Milisegundos", "microseconds": "Microsegundos", "nanoseconds": "Nanosegundos", "minutes": "Minutos", "hours": "Horas", "days": "Días", "weeks": "Semanas", "years": "Años (gregoriano)", "lustrum": "Quinquenios", "decades": "Décadas", "centuries": "Siglos", "millennium": "Milenios", "fahrenheit": "Fahrenheit", "celsius": "Celsius (Centígrados)", "kelvin": "Kelvin", "reamur": "Réaumur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Metros por segundo", "kilometersHour": "Kilómetros por hora", "milesHour": "Millas por hora", "knots": "Nudos", "feetSecond": "Pies por segundo", "minutesPerKilometer": "Minutos por kilómetro", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "Gramos", "ettograms": "Hectogramos", "kilograms": "Kilogramos", "pounds": "Libras", "ounces": "Onzas", "quintals": "Quintales", "tonnes": "Toneladas", "milligrams": "Miligramos", "uma": "Unidad de masa atómica", "carats": "Quilates", "centigrams": "Centigramos", "pennyweights": "Pennyweights", "troyOunces": "Onzas troy", "stones": "Stones", "pascal": "Pascal", "atmosphere": "Atmósfera", "bar": "Bar", "millibar": "Milibar", "psi": "Libras por pulgada cuadrada", "torr": "Milímetro de mercurio (Torr)", "hectoPascal": "Hectopascal", "inchesOfMercury": "Pulgadas de mercurio", "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "Julio", "calories": "Caloría", "kilowattHour": "Kilovatio-hora", "electronvolt": "Electronvoltio", "footPound": "Libra de pie", "kilocalories": "Kilocalories", "@kilocalories": { "description": "Not yet translated. Once done, remove this comment" }, "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "Grados", "minutesDegree": "Minutos", "secondsDegree": "Segundos", "radiansDegree": "Radianes", "usd": "Dólar estadounidense", "eur": "Euro", "gbp": "Libra esterlina", "inr": "Rupia india", "cny": "Yuan chino", "jpy": "Yen japonés", "chf": "Franco suizo", "sek": "Corona sueca", "rub": "Rublo ruso", "cad": "Dólar canadiense", "krw": "Won surcoreano", "brl": "Real brasileño", "hkd": "Dólar hongkonés", "aud": "Dólar australiano", "nzd": "Dólar neozelandés", "mxn": "Peso mexicano", "sgd": "Dólar singapurense", "nok": "Corona noruega", "trY": "Lira turca", "zar": "Rand sudafricano", "dkk": "Corona danesa", "pln": "Esloti polaco", "thb": "Baht tailandés", "myr": "Ringgit malasio", "huf": "Forint húngaro", "czk": "Corona checa", "ils": "Shekel israelí", "idr": "Rupia indonesia", "php": "Peso filipino", "ron": "Leu rumano", "isk": "Corona islandesa", "twd": "Nuevo dólar taiwanés", "mad": "Dírham marroquí", "euChina": "EE. UU. y China", "ukIndiaChild": "RU e India - Niño", "ukIndiaMan": "RU e India - Varón", "ukIndiaWoman": "RU e India - Mujer", "usaCanadaChild": "EE. UU. y Canadá - Niño", "usaCanadaMan": "EE. UU. y Canadá - Varón", "usaCanadaWoman": "EE. UU. y Canadá - Mujer", "japan": "Japón", "watt": "Vatio", "milliwatt": "Milivatio", "kilowatt": "Kilovatio", "megawatt": "Megavatio", "gigawatt": "Gigavatio", "europeanHorsePower": "Caballo de potencia europeo", "imperialHorsePower": "Caballo de potencia imperial", "newton": "Newton-metro", "dyne": "Dina", "poundForce": "Libra-fuerza", "kilogramForce": "Kilopondio", "poundal": "Poundal", "newtonMeter": "Newton meter", "dyneMeter": "Dinámetro", "poundForceFeet": "Pie-libra fuerza", "kilogramForceMeter": "Kilopondímetro", "poundalMeter": "Poundal metro", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "Kilómetros por litro", "liters100km": "Litros por cada 100 km", "milesUsGallon": "Millas por galones estadounidenses", "milesImperialGallon": "Millas por galones imperiales", "decimal": "Decimal", "hexadecimal": "Hexadecimal", "octal": "Octal", "binary": "Binario", "bit": "Bit", "nibble": "Nibble", "kilobit": "Kilobit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "Megabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "Gigabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "Terabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "Petabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "Exabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "Gibibit", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Exbibit", "byte": "Byte", "kilobyte": "Kilobyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "Megabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "Gigabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "Terabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "Petabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "Exabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "Kibibyte", "mebibyte": "Mebibyte", "gibibyte": "Gibibyte", "tebibyte": "Tebibyte", "pebibyte": "Pebibyte", "exbibyte": "Exbibyte", "base": "Base", "deca": "Deca-", "hecto": "Hecto-", "kilo": "Kilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Exa-", "zetta": "Zetta-", "yotta": "Yotta-", "deci": "Deci-", "centi": "Centi-", "milli": "Mili-", "micro": "Micro-", "nano": "Nano-", "pico": "Pico-", "femto": "Femto-", "atto": "Atto-", "zepto": "Zepto-", "yocto": "Yocto-" } ================================================ FILE: packages/translations/lib/l10n/app_fr.arb ================================================ { "@@locale": "fr", "appName": "Converter NOW", "reorder": "Réorganiser", "myOrdering": "Ma commande", "about": "À propos", "lastCurrenciesUpdate": "Dernière mise à jour: ", "today": "aujourd'hui", "enableDarkTheme": "Activer le thème sombre", "settings": "Paramètres", "menu": "Menu", "donation": "Faire un don", "buyMeACoffee": "Offrez-moi un café", "donationDialog": "Bonjour, comme vous le savez peut-être, cette application est libre et open-source. Cela signifie que vous pouvez l'utiliser gratuitement et copier/modifier le code source. Le développeur ne gagnera rien pour cela. Mais si vous croyez aussi en ce projet et que vous souhaitez voir de nouvelles fonctionnalités, veuillez envisager de m'offrir un café.", "drawerLogo": "Logo dans le tiroir", "rateApp": "Évaluer l'application", "repoGithub": "Ouvrir le dépôt sur GitHub", "contributeTranslating": "Contribuer à la traduction de l'application", "search": "Rechercher", "clearAll": "Tout effacer", "calculator": "Calculatrice", "significantFigures": "Chiffres significatifs", "removeTrailingZeros": "Supprimer les zéros inutiles", "longPressAdvice": "Appuyez longuement pour déplacer des éléments", "back": "Retour", "save": "Enregistrer", "invalidCharacters": "Erreur, caractères non valides", "theme": "Thème", "dark": "Sombre", "light": "Clair", "system": "Système", "pureBlackTheme": "Thème sombre AMOLED", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "Copie", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "Suite", "language": "Langue", "reorderProperties": "Réorganiser les propriétés", "reorderUnits": "Réorganiser les unités de mesure", "chooseProperty": "Choisissez une propriété", "reorderProperty": "Réorganiser {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW pour les autres plateformes", "sourceCode": "Code source", "undoClearAllMessage": "Tout a été effacé. Voulez-vous annuler ?", "undo": "Annuler", "routeError1": "Cette page n'existe pas", "routeError2": "Revenir à la première page", "ok": "Ok", "@ok": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetAccess": "Revoke internet access", "@revokeInternetAccess": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetExplanation": "This app uses the internet to update currency exchange rates once a day upon opening. This download only requires a few kilobytes. However, if you don't need this feature, you can prevent the app from accessing the internet by enabling this option.", "@revokeInternetExplanation": { "description": "Not yet translated. Once done, remove this comment" }, "useDeviceColor": "Use device color", "@useDeviceColor": { "description": "Not yet translated. Once done, remove this comment" }, "pickColor": "Pick a color", "@pickColor": { "description": "Not yet translated. Once done, remove this comment" }, "themeColor": "Theme color", "@themeColor": { "description": "Not yet translated. Once done, remove this comment" }, "appearance": "Appearance", "@appearance": { "description": "Not yet translated. Once done, remove this comment" }, "conversions": "Conversions", "@conversions": { "description": "Not yet translated. Once done, remove this comment" }, "findOutMore": "Find out more", "@findOutMore": { "description": "Not yet translated. Once done, remove this comment" }, "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "description": "Not yet translated. Once done, remove this comment", "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "Distance", "area": "Superficie", "volume": "Volume", "time": "Temps", "temperature": "Température", "speed": "Vitesse", "siPrefixes": "Préfixes SI", "mass": "Masse", "pressure": "Pression", "energy": "Energie", "angles": "Angles", "currencies": "Devises", "shoeSize": "Pointure de chaussure", "digitalData": "Données numériques", "power": "Puissance", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "Couple", "force": "Force", "fuelConsumption": "Consommation de carburant", "numeralSystems": "Systèmes numériques", "meters": "Mètres", "centimeters": "Centimètres", "inches": "Pouces", "feet": "Pieds", "nauticalMiles": "Miles nautiques", "yards": "Yards", "miles": "Miles", "millimeters": "Millimètres", "micrometers": "Micromètres", "nanometers": "Nanomètres", "angstroms": "Ångström", "picometers": "Picomètres", "kilometers": "Kilomètres", "astronomicalUnits": "Unités astronomiques", "lightYears": "Années-lumières", "parsec": "Parsec", "mils": "Mils", "feetUsSurvey": "Pieds (US survey)", "squareMeters": "Mètres carrés", "squareCentimeters": "Centimètres carrés", "squareInches": "Pouces carrés", "squareFeet": "Pieds carrés", "squareMiles": "Miles carrés", "squareYard": "yards carrés", "squareMillimeters": "Millimètres carrés", "squareKilometers": "Kilomètres carrés", "hectares": "Hectares", "acres": "Acres", "are": "Are", "squareFeetUsSurvey": "Pieds carrés (US survey)", "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "Mètres cubes", "liters": "Litres", "imperialGallons": "Gallons impériaux", "usGallons": "Gallons américains", "imperialPints": "Pintes impériales", "usPints": "Pintes américaines", "milliliters": "Millilitres", "tablespoonUs": "Cuillères à soupe US", "tablespoonAustralian": "Cuillères à soupe australiennes", "cups": "Tasses US", "cubicCentimeters": "Centimètres cubes", "cubicFeet": "Pieds cubes", "cubicInches": "Pouces cubes", "cubicMillimeters": "Millimètres cubes", "imperialFluidOunces": "Once liquide impériales", "usFluidOunces": "Once liquide US", "imperialGill": "Gill impériales", "usGill": "Gill US", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "Secondes", "deciseconds": "Décisecondes", "centiseconds": "Centisecondes", "milliseconds": "Millisecondes", "microseconds": "Microsecondes", "nanoseconds": "Nanosecondes", "minutes": "Minutes", "hours": "Heures", "days": "Jours", "weeks": "Semaines", "years": "Années (365)", "lustrum": "Lustre", "decades": "Décennies", "centuries": "Siècles", "millennium": "Millénaires", "fahrenheit": "Fahrenheit", "celsius": "Celsius (Centigrade)", "kelvin": "Kelvin", "reamur": "Reamur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Mètres par seconde", "kilometersHour": "Kilomètres par heure", "milesHour": "Miles par heure", "knots": "Noeuds", "feetSecond": "Pieds par seconde", "minutesPerKilometer": "Minutes par kilomètre", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "Grammes", "ettograms": "Hectogramme", "kilograms": "Kilogrammes", "pounds": "Livres", "ounces": "Onces", "quintals": "Quintaux", "tonnes": "Tonnes", "milligrams": "Milligrammes", "uma": "Unité de masse atomique unifiée", "carats": "Carats", "centigrams": "Centigrammes", "pennyweights": "Pennyweight", "troyOunces": "Once troy", "stones": "Stone", "pascal": "Pascal", "atmosphere": "Atmosphère", "bar": "Bar", "millibar": "Millibar", "psi": "Livres par pouce carré", "torr": "Millimètre de mercure (Torr)", "hectoPascal": "Hectopascal", "inchesOfMercury": "Pouces de mercure", "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "Joule", "calories": "Calorie", "kilowattHour": "Kilowatt heure", "electronvolt": "Électronvolt", "footPound": "Pied-livre", "kilocalories": "Kilocalorie", "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "Degré", "minutesDegree": "Minutes", "secondsDegree": "Secondes", "radiansDegree": "Radians", "usd": "Dollar des États-Unis", "eur": "Euro", "gbp": "Livre sterling", "inr": "Roupie indienne", "cny": "Yuan renminbi chinois", "jpy": "Yen japonais", "chf": "Franc suisse", "sek": "Couronne suédoise", "rub": "Rouble russe", "cad": "Dollar canadien", "krw": "Won sud-coréen", "brl": "Réal brésilien", "hkd": "Dollar de Hong Kong", "aud": "Dollar australien", "nzd": "Dollar néo-zélandais", "mxn": "Peso mexicain", "sgd": "Dollar de Singapour", "nok": "Couronne norvégienne", "trY": "Livre turque", "zar": "Rand sud-africain", "dkk": "Couronne danoise", "pln": "Zloty polonais", "thb": "Baht thaïlandais", "myr": "Ringgit malais", "huf": "Forint hongrois", "czk": "Couronne tchèque", "ils": "Nouveau shekel israélien", "idr": "Roupie indonésienne", "php": "Peso philippin", "ron": "Leu roumain", "isk": "Couronne islandaise", "twd": "Nouveau dollar taïwanais", "mad": "Dirham marocain", "euChina": "EU et Chine", "ukIndiaChild": "UK & Inde - Enfant", "ukIndiaMan": "UK et Inde - Homme", "ukIndiaWoman": "UK et Inde - Femme", "usaCanadaChild": "USA et Canada - Enfant", "usaCanadaMan": "USA et Canada - Homme", "usaCanadaWoman": "USA et Canada - Femme", "japan": "Japon", "watt": "Watt", "milliwatt": "Milliwatt", "kilowatt": "Kilowatt", "megawatt": "Mégawatt", "gigawatt": "GigaWatt", "europeanHorsePower": "Cheval vapeur européen", "imperialHorsePower": "Cheval vapeur imperial", "newton": "Newton", "dyne": "Dyne", "poundForce": "Livre-force", "kilogramForce": "Kilogramme-force", "poundal": "Poundal", "newtonMeter": "Newton mètre", "dyneMeter": "Dyne mètre", "poundForceFeet": "Pied-livre-force", "kilogramForceMeter": "Kilogramme-force mètre", "poundalMeter": "Poundal mètre", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "Kilomètres par litre", "liters100km": "Litres par 100 km", "milesUsGallon": "Miles par gallon américain", "milesImperialGallon": "Miles par gallon impérial", "decimal": "Décimal", "hexadecimal": "Hexadécimal", "octal": "Octal", "binary": "Binaire", "bit": "Bit", "nibble": "Nibble", "kilobit": "Kilobit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "Mégabit", "@megabit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "Gigabit", "@gigabit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "Térabit", "@terabit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "Pétabit", "@petabit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "Exabit", "@exabit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "Kibibit", "mebibit": "Mébibit", "gibibit": "Gibibit", "tebibit": "Tébibit", "pebibit": "Pébibit", "exbibit": "Exbibit", "byte": "Octet", "kilobyte": "Kilooctet", "@kilobyte": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "Mégaoctet", "@megabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "Gigaoctet", "@gigabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "Téraoctet", "@terabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "Pétaoctet", "@petabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "Exaoctet", "@exabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "Kibioctet", "mebibyte": "Mébioctet", "gibibyte": "Gibioctet", "tebibyte": "Tébioctet", "pebibyte": "Pébioctet", "exbibyte": "Exbioctet", "base": "Base", "deca": "Déca-", "hecto": "Hecto-", "kilo": "Kilo-", "mega": "Méga-", "giga": "Giga-", "tera": "Téra-", "peta": "Péta-", "exa": "Exa-", "zetta": "Zetta-", "yotta": "Yotta-", "deci": "Déci-", "centi": "Centi-", "milli": "Milli-", "micro": "Micro-", "nano": "Nano-", "pico": "Pico-", "femto": "Femto-", "atto": "Atto-", "zepto": "Zepto-", "yocto": "Yocto-" } ================================================ FILE: packages/translations/lib/l10n/app_hr.arb ================================================ { "@@locale": "hr", "appName": "Converter NOW", "reorder": "Promijeni redoslijed", "myOrdering": "Moj redoslijed", "about": "Informacije", "lastCurrenciesUpdate": "Zadnje aktualiziranje: ", "today": "danas", "enableDarkTheme": "Aktiviraj tamnu temu", "settings": "Postavke", "menu": "Izbornik", "donation": "Doniraj", "buyMeACoffee": "Kupi mi kavu", "donationDialog": "Kao što možda znaš, ovo je slobodan softver otvorenog izvornog koda. To znači da se program smije besplatno koristiti a izvorni kod kopirati i mijenjati. Programer na tome ne zarađuje ništa. No, ako vjeruješ u ovaj projekt i ako želiš da se ugrade nove funkcije, možeš mi kupiti kavu.", "drawerLogo": "Logotip u izborniku", "rateApp": "Ocijeni program", "repoGithub": "Otvori repozitorij na GitHub-u", "contributeTranslating": "Pomogni prevoditi program", "search": "Pretraga", "clearAll": "Isprazni sva polja", "calculator": "Kalkulator", "significantFigures": "Broj značajnih znamenki", "removeTrailingZeros": "Ukloni nepotrebne nule", "longPressAdvice": "Za mijenjanje redoslijeda elemenata pritisni ih dugo", "back": "Natrag", "save": "Spremi", "invalidCharacters": "Greška, neispravni znakovi", "theme": "Tema", "dark": "Tamna", "light": "Svijetla", "system": "Sustav", "pureBlackTheme": "Potpuno crna tema", "propertySelectionOnStartup": "Odabir svojstva pri pokretanju", "propertySelectionOnStartupSubtitle": "Prikaži stranicu za odabir svojstva pri pokretanju na uređajima s malim ekranom.", "copy": "Kopiraj", "paste": "Umetni", "more": "Više", "language": "Jezik", "reorderProperties": "Promijeni redoslijed svojstava", "reorderUnits": "Promijeni redoslijed jedinica", "chooseProperty": "Odaberi svojstvo", "reorderProperty": "Promijeni redoslijed za {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW za druge platforme", "sourceCode": "Izvorni kod", "undoClearAllMessage": "Sva polja su ispražnjena. Želiš li poništiti tu radnju?", "undo": "Poništi", "routeError1": "Ova stranica ne postoji", "routeError2": "Vrati se na prvu stranicu", "ok": "U redu", "revokeInternetAccess": "Isključi pristup internetu", "revokeInternetExplanation": "Ova aplikacija koristi internet za aktualiziranje tečajeva valuta jednom dnevno prilikom otvaranja. To preuzimanje zahtijeva samo nekoliko kilobajta. Međutim, ako ti ova funkcija nije potrebna, možeš isključiti pristup internetu za aplikaciju uključivanjem ove opcije.", "useDeviceColor": "Koristi boje uređaja", "pickColor": "Odaberi boju", "themeColor": "Boja teme", "appearance": "Izgled", "conversions": "Konverzije", "findOutMore": "Saznaj više", "hideUnits": "Sakrij jedinice", "visibleUnits": "{property}, vidljive jedinice", "@visibleUnits": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Skrivene jedinice", "selectAll": "Označi sve", "unselectAll": "Odznači sve", "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "Duljina", "area": "Površina", "volume": "Volumen", "time": "Vrijeme", "temperature": "Temperatura", "speed": "Brzina", "siPrefixes": "SI prefiksi", "mass": "Masa", "pressure": "Pritisak", "energy": "Energija", "angles": "Kut", "currencies": "Valuta", "shoeSize": "Veličina obuće", "digitalData": "Digitalni podaci", "power": "Snaga", "density": "Gustoća", "torque": "Zakretni moment", "force": "Sila", "fuelConsumption": "Potrošnja goriva", "numeralSystems": "Numerički sustavi", "meters": "Metar", "centimeters": "Centimetar", "inches": "Inč", "feet": "Stopa", "nauticalMiles": "Nautička milja", "yards": "Jard", "miles": "Milja", "millimeters": "Milimetar", "micrometers": "Mikrometar", "nanometers": "Nanometar", "angstroms": "Angstrem", "picometers": "Pikometar", "kilometers": "Kilometar", "astronomicalUnits": "Astronomska jedinica", "lightYears": "Svjetlosna godina", "parsec": "Parsek", "mils": "Tisućinka inča", "feetUsSurvey": "Stopa (US survey)", "squareMeters": "Kvadratni metar", "squareCentimeters": "Kvadratni centimetar", "squareInches": "Kvadratni inč", "squareFeet": "Kvadratna stopa", "squareMiles": "Kvadratna milja", "squareYard": "Kvadratni jard", "squareMillimeters": "Kvadratni milimetar", "squareKilometers": "Kvadratni kilometar", "hectares": "Hektar", "acres": "Aker", "are": "Are", "squareFeetUsSurvey": "Kvadratna stopa (US survey)", "gramsPerCubicCentimeter": "Gram po kubičnom centimetru", "gramsPerLiter": "Gram po litri", "gramsPerMilliliter": "Gram po mililitru", "gramsPerDeciliter": "Gram po decilitru", "kilogramsPerLiter": "Kilogram po litri", "kilogramsPerCubicMeter": "Kilogram po kubičnom metru", "milligramsPerLiter": "Miligram po litri", "milligramsPerDeciliter": "Miligram po decilitru", "milligramsPerMilliliter": "Miligram po milimetru", "milligramsPerCubicMeter": "Miligram po kubičnom metru", "milligramsPerCubicCentimeter": "Miligram po kubičnom centimetru", "microgramsPerLiter": "Mikrogram po litri", "microgramsPerDeciliter": "Mikrogram po decilitru", "microgramsPerMilliliter": "Mikrogram po milimetru", "poundsPerCubicInches": "Funta po kubičnom inču", "poundsPerCubicFeet": "Funta po kubičnoj stopi", "cubicMeters": "Kubični metar", "liters": "Litra", "imperialGallons": "Imperijalni galon", "usGallons": "Američki galon", "imperialPints": "Imperijalna pinta", "usPints": "Američka pinta", "milliliters": "Mililitar", "tablespoonUs": "Američka žlica", "tablespoonAustralian": "Australska žlica", "cups": "Šalica", "cubicCentimeters": "Kubični centimetar", "cubicFeet": "Kubična stopa", "cubicInches": "Kubični inč", "cubicMillimeters": "Kubični milimetar", "imperialFluidOunces": "Imperijalna unca tekućine", "usFluidOunces": "Američka unca tekućine", "imperialGill": "Imperijalni gill", "usGill": "Američki gill", "usQuarts": "Četvrtina američkog galona", "microliters": "Mikrolitar", "deciliters": "Decilitar", "centiliters": "Centilitar", "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "Sekunda", "deciseconds": "Desetinka sekunde", "centiseconds": "Stotinka sekunde", "milliseconds": "Tisućinka sekunde", "microseconds": "Mikrosekunda", "nanoseconds": "Nanosekunda", "minutes": "Minuta", "hours": "Sat", "days": "Dan", "weeks": "Tjedan", "years": "Godina (365)", "lustrum": "Lustrum", "decades": "Desetljeće", "centuries": "Stoljeće", "millennium": "Tisućljeće", "fahrenheit": "Fahrenheit", "celsius": "Celzij", "kelvin": "Kelvin", "reamur": "Reamur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Metar u sekundi", "kilometersHour": "Kilometar na sat", "milesHour": "Milja na sat", "knots": "Čvor", "feetSecond": "Stopa u sekundi", "minutesPerKilometer": "Minuta po kilometru", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "Gram", "ettograms": "Etogram", "kilograms": "Kilogram", "pounds": "Funta", "ounces": "Unca", "quintals": "Kvintal", "tonnes": "Tona", "milligrams": "Miligram", "uma": "Atomska jedinica mase", "carats": "Karat", "centigrams": "Centigram", "pennyweights": "Pennyweights", "troyOunces": "Troy unca", "stones": "Stones", "pascal": "Paskal", "atmosphere": "Atmosfera", "bar": "Bar", "millibar": "Milibar", "psi": "Funta po kvatdratnom inču", "torr": "Tor", "hectoPascal": "Hektopaskal", "inchesOfMercury": "Inč živina stupca", "kiloPascal": "Kilopascal", "ksi": "Kilofunta po kvadratnom inču", "megaPascal": "Megapaskal", "gigaPascal": "Gigapaskal", "joule": "Džul", "calories": "Kalorija", "kilowattHour": "Kilovat-sat", "electronvolt": "Elektronvolt", "footPound": "Funta po stopi", "kilocalories": "Kilokalorija", "kilojoules": "Kilodžul", "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "Stupanj", "minutesDegree": "Minuta", "secondsDegree": "Sekunda", "radiansDegree": "Radijan", "usd": "Američki dolar", "eur": "Euro", "gbp": "Britanska funta", "inr": "Indonezijska rupija", "cny": "Kineski juan", "jpy": "Japanski jen", "chf": "Švicarski franak", "sek": "Švedska kruna", "rub": "Ruski rubalj", "cad": "Kanadski dolar", "krw": "Južnokorejski von", "brl": "Brazilski real", "hkd": "Hongkonški dolar", "aud": "Australski dolar", "nzd": "Novozelandski dolar", "mxn": "Meksički pezo", "sgd": "Singapurski dolar", "nok": "Norveška kruna", "trY": "Turska lira", "zar": "Južnoafrički rand", "dkk": "Danska kruna", "pln": "Poljski zloti", "thb": "Tajlandski baht", "myr": "Malezijski ringit", "huf": "Mađarska forinta", "czk": "Češka kruna", "ils": "Izraelski šekel", "idr": "Indonezijska rupija", "php": "Filipinski pezo", "ron": "Rumunjski lej", "isk": "Islandska kruna", "twd": "Novotajvanski dolar", "mad": "Marokanski dirham", "euChina": "EU i Kina", "ukIndiaChild": "UK i Indija – djeca", "ukIndiaMan": "UK i Indija – muškarci", "ukIndiaWoman": "UK i Indija – žene", "usaCanadaChild": "SAD i Kanada – djeca", "usaCanadaMan": "SAD i Kanada – muškarci", "usaCanadaWoman": "SAD i Kanada – žene", "japan": "Japan", "watt": "Vat", "milliwatt": "Milivat", "kilowatt": "Kilovat", "megawatt": "Megavat", "gigawatt": "Gigavat", "europeanHorsePower": "Metrična konjska snaga", "imperialHorsePower": "Imperijalna konjska snaga", "newton": "Njutn", "dyne": "Din", "poundForce": "Funta-sila", "kilogramForce": "Kilogram-sila", "poundal": "Poundal", "newtonMeter": "Njutn metar", "dyneMeter": "Din metar", "poundForceFeet": "Funta-sila stopa", "kilogramForceMeter": "Kilogram-sila metar", "poundalMeter": "Poundal metar", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "Kilometar na litru", "liters100km": "Litra na 100 km", "milesUsGallon": "Milja na američki galon", "milesImperialGallon": "Milja na imperijalni galon", "decimal": "Decimalni", "hexadecimal": "Heksadecimalni", "octal": "Oktalni", "binary": "Binarni", "bit": "Bit", "nibble": "Nibble", "kilobit": "Kilobit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "Megabit", "@megabit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "Gigabit", "@gigabit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "Terabit", "@terabit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "Petabit", "@petabit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "Eksabit", "@exabit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "Gibibit", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Eksbibit", "byte": "Bajt", "kilobyte": "Kilobajt", "@kilobyte": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "Megabajt", "@megabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "Gigabajt", "@gigabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "Terabajt", "@terabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "Petabajt", "@petabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "Eksabajt", "@exabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "Kibibajt", "mebibyte": "Mebibajt", "gibibyte": "Gibibajt", "tebibyte": "Tebibajt", "pebibyte": "Pebibajt", "exbibyte": "Eksbibajt", "base": "Osnova", "deca": "Deka-", "hecto": "Hekto-", "kilo": "Kilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Eksa-", "zetta": "Zeta-", "yotta": "Jota-", "deci": "Deci-", "centi": "Centi-", "milli": "Mili-", "micro": "Mikro-", "nano": "Nano-", "pico": "Piko-", "femto": "Femto-", "atto": "Ato-", "zepto": "Zepto-", "yocto": "Jokto-" } ================================================ FILE: packages/translations/lib/l10n/app_id.arb ================================================ { "@@locale": "id", "appName": "Converter NOW", "reorder": "Tata ulang", "myOrdering": "Penataan saya", "about": "Tentang", "lastCurrenciesUpdate": "Terakhir diperbarui: ", "today": "hari ini", "enableDarkTheme": "Aktifkan tema gelap", "settings": "Pengaturan", "menu": "Menu", "donation": "Donasi", "buyMeACoffee": "Belikan saya sebuah kopi", "donationDialog": "Hi, seperti yang Anda tahu, aplikasi ini gratis/bebas dan sumber terbuka. Itu berarti Anda bisa menggunakan aplikasi ini secara gratis/bebas dan menyalin/edit kode sumbernya. Sang pengembang tidak akan mendapatkan apa pun dari itu. Tapi, jika Anda mau melihat fitur baru, mohon untuk mempertimbangkan membelikan saya sebuah kopi.", "drawerLogo": "Logo pada menu", "rateApp": "Ulas aplikasi", "repoGithub": "Buka repo di GitHub", "contributeTranslating": "Berkontribusi menerjemahkan aplikasi", "search": "Cari", "clearAll": "Bersihkan semua", "calculator": "Kalkulator", "significantFigures": "Angka Penting", "removeTrailingZeros": "Hapus angka nol pengikut", "longPressAdvice": "Tekan lama untuk memindahkan komponen", "back": "Kembali", "save": "Simpan", "invalidCharacters": "Galat, karakter tidak valid", "theme": "Tema", "dark": "Gelap", "light": "Terang", "system": "Sistem", "pureBlackTheme": "Tema gelap AMOLED", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "Salin", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "Lebih banyak", "language": "Bahasa", "reorderProperties": "Tata ulang properti", "reorderUnits": "Tata ulang satuan", "chooseProperty": "Pilih sebuah properti", "reorderProperty": "Tata ulang {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW untuk sistem operasi lain", "sourceCode": "Kode sumber", "undoClearAllMessage": "Dihapus semua. Apakah Anda ingin mengurungkannya?", "undo": "Urung", "routeError1": "Halaman ini tidak ada", "routeError2": "Kembali ke beranda", "ok": "Ok", "@ok": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetAccess": "Revoke internet access", "@revokeInternetAccess": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetExplanation": "This app uses the internet to update currency exchange rates once a day upon opening. This download only requires a few kilobytes. However, if you don't need this feature, you can prevent the app from accessing the internet by enabling this option.", "@revokeInternetExplanation": { "description": "Not yet translated. Once done, remove this comment" }, "useDeviceColor": "Use device color", "@useDeviceColor": { "description": "Not yet translated. Once done, remove this comment" }, "pickColor": "Pick a color", "@pickColor": { "description": "Not yet translated. Once done, remove this comment" }, "themeColor": "Theme color", "@themeColor": { "description": "Not yet translated. Once done, remove this comment" }, "appearance": "Appearance", "@appearance": { "description": "Not yet translated. Once done, remove this comment" }, "conversions": "Conversions", "@conversions": { "description": "Not yet translated. Once done, remove this comment" }, "findOutMore": "Find out more", "@findOutMore": { "description": "Not yet translated. Once done, remove this comment" }, "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "description": "Not yet translated. Once done, remove this comment", "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "Panjang", "area": "Luas", "volume": "Volume", "time": "Waktu", "temperature": "Temperatur", "speed": "Kecepatan", "siPrefixes": "Awalan SI", "mass": "Massa", "pressure": "Tekanan", "energy": "Energi", "angles": "Sudut", "currencies": "Kurs", "shoeSize": "Ukuran Sepatu", "digitalData": "Data Digital", "power": "Daya", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "Torsi", "force": "Gaya", "fuelConsumption": "Konsumsi Bahan Bakar", "numeralSystems": "Sistem Numeral", "meters": "Meter", "centimeters": "Sentimeter", "inches": "Inci", "feet": "Kaki", "nauticalMiles": "Mil laut", "yards": "Yard", "miles": "Mil", "millimeters": "Milimeter", "micrometers": "Mikrometer", "nanometers": "Nanometer", "angstroms": "Ångström", "picometers": "Pikometer", "kilometers": "Kilometer", "astronomicalUnits": "Satuan Astronomi", "lightYears": "Tahun Cahaya", "parsec": "Parsek", "mils": "Seperseribu Inci (Mil/Thou)", "feetUsSurvey": "Kaki (Survei AS)", "squareMeters": "Meter Persegi", "squareCentimeters": "Sentimeter Persegi", "squareInches": "Inci Persegi", "squareFeet": "Kaki Persegi", "squareMiles": "Mil Persegi", "squareYard": "Yard Persegi", "squareMillimeters": "Milimeter Persegi", "squareKilometers": "Kilometer Persegi", "hectares": "Hektare", "acres": "Ekar", "are": "Are", "squareFeetUsSurvey": "Square Feet (US survey)", "@squareFeetUsSurvey": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "Meter Kubik", "liters": "Liter", "imperialGallons": "Galon Imperial", "usGallons": "Galon AS", "imperialPints": "Pint Imperial", "usPints": "Pint AS", "milliliters": "Mililiter", "tablespoonUs": "Sendok Makan AS", "tablespoonAustralian": "Sendok Makan Australia", "cups": "Gelas", "cubicCentimeters": "Sentimeter Kubik", "cubicFeet": "Kaki Kubik", "cubicInches": "Inci Kubik", "cubicMillimeters": "Milimeter Kubik", "imperialFluidOunces": "Ons Cairan Imperial", "usFluidOunces": "Ons Cairan AS", "imperialGill": "Gil Imperial", "usGill": "Gil AS", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "Detik", "deciseconds": "Desidetik", "centiseconds": "Sentidetik", "milliseconds": "Milidetik", "microseconds": "Mikrodetik", "nanoseconds": "Nanodetik", "minutes": "Menit", "hours": "Jam", "days": "Hari", "weeks": "Pekan", "years": "Tahun (365)", "lustrum": "Lustrum", "decades": "Dasawarsa", "centuries": "Abad", "millennium": "Milenium", "fahrenheit": "Fahrenheit", "celsius": "Celsius (Centigrade)", "kelvin": "Kelvin", "reamur": "Reamur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Meter per Detik", "kilometersHour": "Kilometer per Jam", "milesHour": "Mil per Jam", "knots": "Knot", "feetSecond": "Kaki per Detik", "minutesPerKilometer": "Menit per Kilometer", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "Gram", "ettograms": "Ettogram", "kilograms": "Kilogram", "pounds": "Pon", "ounces": "Ons", "quintals": "Kuintal", "tonnes": "Ton", "milligrams": "Miligram", "uma": "Satuan Massa Atom Terpadu", "carats": "Karat", "centigrams": "Sentigram", "pennyweights": "Kelas Penny", "troyOunces": "Ons Troy", "stones": "Batu", "pascal": "Pascal", "atmosphere": "Atmosfer", "bar": "Bar", "millibar": "Milibar", "psi": "Pon per Inci Persegi", "torr": "Torr", "hectoPascal": "Hektopascal", "inchesOfMercury": "Inci Merkurium", "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "Joule", "calories": "Kalori", "kilowattHour": "Kilowattjam", "electronvolt": "Elektron Volt", "footPound": "Pon Kaki", "kilocalories": "Kilokalori", "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "Derajat", "minutesDegree": "Menit", "secondsDegree": "Detik", "radiansDegree": "Radian", "usd": "Dolar Amerika", "eur": "Euro", "gbp": "Pound Britania", "inr": "Rupee India", "cny": "Yuan China", "jpy": "Yen Jepang", "chf": "Franc Swiss", "sek": "Krona Swedia", "rub": "Rubel Rusia", "cad": "Dolar Kanada", "krw": "Won Korea Selatan", "brl": "Real Brasil", "hkd": "Dolar Hong Kong", "aud": "Dolar Australia", "nzd": "Dolar Selandia Baru", "mxn": "Peso Meksiko", "sgd": "Dolar Singapura", "nok": "Krona Norwegia", "trY": "Lira Turki", "zar": "Rand Afrika Selatan", "dkk": "Krona Denmark", "pln": "Złoty Polandia", "thb": "Baht Thailand", "myr": "Ringgit Malaysia", "huf": "Forint Hongaria", "czk": "Koruna Ceko", "ils": "Shekel Israel", "idr": "Rupiah Indonesia", "php": "Peso Filipina", "ron": "Leu Rumania", "isk": "Króna Islandia", "twd": "Dolar Baru Taiwan", "mad": "Dirham Maroko", "euChina": "Eropa dan Cina", "ukIndiaChild": "Britania dan India - Anak", "ukIndiaMan": "Britania dan India - Laki-laki", "ukIndiaWoman": "Britania dan India - Perempuan", "usaCanadaChild": "AS dan Kanada - Anak", "usaCanadaMan": "AS dan Kanada - Laki-laki", "usaCanadaWoman": "AS dan Kanada - Perempuan", "japan": "Jepang", "watt": "Watt", "milliwatt": "Milliwatt", "kilowatt": "Kilowatt", "megawatt": "Megawatt", "gigawatt": "Gigawatt", "europeanHorsePower": "Tenaga Kuda Eropa", "imperialHorsePower": "Tenaga Kuda Imperial", "newton": "Newton", "dyne": "Dyne", "poundForce": "Pon-Gaya", "kilogramForce": "Kilogram-Gaya", "poundal": "Pondal", "newtonMeter": "Newton meter", "dyneMeter": "Dyne meter", "poundForceFeet": "Pon-Gaya Kaki", "kilogramForceMeter": "Kilogram-Gaya Meter", "poundalMeter": "Pondal Meter", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "Kilometer per Liter", "liters100km": "Liter per 100 km", "milesUsGallon": "Mil per Galon AS", "milesImperialGallon": "Mil per Galon Imperial", "decimal": "Desimal", "hexadecimal": "Heksadesimal", "octal": "Oktal", "binary": "Biner", "bit": "Bit", "nibble": "Nibel", "kilobit": "Kilobit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "Megabit", "@megabit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "Gigabit", "@gigabit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "Terabit", "@terabit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "Petabit", "@petabit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "Eksabit", "@exabit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "Gibibit", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Eksbibit", "byte": "Bita", "kilobyte": "Kilobita", "@kilobyte": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "Megabita", "@megabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "Gigabita", "@gigabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "Terabita", "@terabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "Petabita", "@petabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "Eksabita", "@exabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "Kibibita", "mebibyte": "Mebibita", "gibibyte": "Gibibita", "tebibyte": "Tebibita", "pebibyte": "Pebibita", "exbibyte": "Eksbibita", "base": "Dasar", "deca": "Deka-", "hecto": "Hekto-", "kilo": "Kilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Eksa-", "zetta": "Zeta-", "yotta": "Yota-", "deci": "Desi-", "centi": "Senti-", "milli": "Mili-", "micro": "Mikro-", "nano": "Nano-", "pico": "Piko-", "femto": "Femto-", "atto": "Ato-", "zepto": "Zepto-", "yocto": "Yokto-" } ================================================ FILE: packages/translations/lib/l10n/app_it.arb ================================================ { "@@locale": "it", "appName": "Converter NOW", "reorder": "Riordina", "myOrdering": "Il mio ordinamento", "about": "Informazioni", "lastCurrenciesUpdate": "Aggiornato a: ", "today": "oggi", "enableDarkTheme": "Abilita tema scuro", "settings": "Impostazioni", "menu": "Menu", "donation": "Fai una donazione", "buyMeACoffee": "Offrimi un caffè", "donationDialog": "Ciao, come forse saprai questa app è gratuita e open source. Significa che puoi usarla gratuitamente e copiare / modificare il codice sorgente. Lo sviluppatore non ci guadagnerà nulla. Ma se anche tu credi in questo progetto e vorresti vedere nuove funzionalità, per favore considera di offrirmi un caffè.", "drawerLogo": "Logo nel drawer", "rateApp": "Valuta l'app", "repoGithub": "Apri repository su GitHub", "contributeTranslating": "Contribuisci a tradurre l'app", "search": "Cerca", "clearAll": "Elimina tutto", "calculator": "Calcolatrice", "significantFigures": "Cifre significative", "removeTrailingZeros": "Rimuovi gli zero finali", "longPressAdvice": "Tieni premuto a lungo per spostare gli elementi", "back": "Indietro", "save": "Salva", "invalidCharacters": "Errore, caratteri non validi", "theme": "Tema", "dark": "Scuro", "light": "Chiaro", "system": "Sistema", "pureBlackTheme": "Tema nero puro", "propertySelectionOnStartup": "Selezione delle proprietà all'avvio", "propertySelectionOnStartupSubtitle": "Mostra la pagina di selezione delle proprietà all'avvio sui dispositivi con piccolo schermo", "copy": "Copia", "paste": "Incolla", "more": "Altro", "language": "Lingua", "reorderProperties": "Riordina le grandezze", "reorderUnits": "Riordina le unità di misura", "chooseProperty": "Scegli una grandezza", "reorderProperty": "Riordina {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW per le altre piattaforme", "sourceCode": "Codice sorgente", "undoClearAllMessage": "Eliminato tutto. Annullare l'operazione eseguita?", "undo": "Annulla", "routeError1": "Questa pagina non esiste", "routeError2": "Ritorna alla prima pagina", "ok": "Ok", "revokeInternetAccess": "Revoca l'accesso a internet", "revokeInternetExplanation": "Questa app utilizza internet per aggiornare i tassi di cambio delle valute una volta al giorno (quando viene aperta). Si tratta di pochi chilobyte di download, ma se non hai bisogno di questa funzionalità puoi prevenire che la app acceda ad intenet abilitando questa opzione.", "useDeviceColor": "Usa il colore del dispositivo", "pickColor": "Scegli un colore", "themeColor": "Colore del tema", "appearance": "Aspetto", "conversions": "Conversioni", "findOutMore": "Scopri di più", "hideUnits": "Nascondi le unità", "visibleUnits": "{property}, unità visibili:", "@visibleUnits": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Unità nascoste", "selectAll": "Seleziona tutto", "unselectAll": "Deseleziona tutto", "backupAndRestore": "Backup e ripristino", "exportSettings": "Esporta impostazioni", "importSettings": "Importa impostazioni", "clearSettings": "Cancella tutte le impostazioni", "exportSuccess": "Impostazioni esportate con successo", "importSuccess": "Impostazioni importate con successo", "problemImportFile": "Si è verificato un problema durante l'importazione del file", "relatedSettings": "Impostazioni correlate", "reason": "Motivo", "importError": "Errore di importazione", "confirmationClear": "Sei sicuro di voler cancellare tutte le impostazioni? Questa azione non può essere annullata.", "cancel": "Annulla", "length": "Lunghezza", "area": "Superficie", "volume": "Volume", "time": "Tempo", "temperature": "Temperatura", "speed": "Velocità", "siPrefixes": "Prefissi SI", "mass": "Massa", "pressure": "Pressione", "energy": "Energia", "angles": "Angoli", "currencies": "Valute", "shoeSize": "Taglia scarpe", "digitalData": "Dati digitali", "power": "Potenza", "density": "Densità", "torque": "Momento", "force": "Forza", "fuelConsumption": "Consumo carburante", "numeralSystems": "Basi numeriche", "meters": "Metri", "centimeters": "Centimetri", "inches": "Pollici", "feet": "Piedi", "nauticalMiles": "Miglia Marine", "yards": "Yard", "miles": "Miglia", "millimeters": "Millimetri", "micrometers": "Micrometri", "nanometers": "Nanometri", "angstroms": "Ångström", "picometers": "Picometri", "kilometers": "Chilometri", "astronomicalUnits": "Unità Astronomiche", "lightYears": "Anni luce", "parsec": "Parsec", "mils": "Mils", "feetUsSurvey": "Piedi (US survey)", "squareMeters": "Metri Quadrati", "squareCentimeters": "Centimetri Quadrati", "squareInches": "Pollici Quadrati", "squareFeet": "Piedi Quadrati", "squareMiles": "Miglia Quadrate", "squareYard": "Yard Quadrati", "squareMillimeters": "Millimetri Quadrati", "squareKilometers": "Chilometri Quadrati", "hectares": "Ettaro", "acres": "Acri", "are": "Are", "squareFeetUsSurvey": "Piedi Quadrati (US survey)", "gramsPerCubicCentimeter": "Grammi per centimetro cubo", "gramsPerLiter": "Grammi per litro", "gramsPerMilliliter": "Grammi per millilitro", "gramsPerDeciliter": "Grammi per decilitro", "kilogramsPerLiter": "Chilogrammi per litro", "kilogramsPerCubicMeter": "Chilogrammi per metro cubo", "milligramsPerLiter": "Milligrammi per litro", "milligramsPerDeciliter": "Milligrammi per decilitro", "milligramsPerMilliliter": "Milligrammi per millilitro", "milligramsPerCubicMeter": "Milligrammi per metro cubo", "milligramsPerCubicCentimeter": "Milligrammi per centimetro cubo", "microgramsPerLiter": "Microgrammi per litro", "microgramsPerDeciliter": "Microgrammi per decilitro", "microgramsPerMilliliter": "Microgrammi per millilitro", "poundsPerCubicInches": "Libbre per pollice cubo", "poundsPerCubicFeet": "Libbre per piede cubo", "cubicMeters": "Metri Cubi", "liters": "Litri", "imperialGallons": "Galloni Imperiali", "usGallons": "Galloni US", "imperialPints": "Pinte imperiali", "usPints": "Pinte US", "milliliters": "Millilitri", "tablespoonUs": "Cucchiai US", "tablespoonAustralian": "Cucchiai Australiani", "cups": "Tazze", "cubicCentimeters": "Centimetri Cubi", "cubicFeet": "Piedi Cubi", "cubicInches": "Pollici Cubi", "cubicMillimeters": "Millimetri Cubi", "imperialFluidOunces": "Once liquide Imperiali", "usFluidOunces": "Once liquide US", "imperialGill": "Gill Imperiali", "usGill": "Gill US", "usQuarts": "Quarti USA", "microliters": "Microlitri", "deciliters": "Decilitri", "centiliters": "Centilitri", "teaspoonsUs": "Cucchiaini US", "teaspoonsMetric": "Cucchiaini metrici", "seconds": "Secondi", "deciseconds": "Decimi di Secondo", "centiseconds": "Centesimi di Secondo", "milliseconds": "Millisecondi", "microseconds": "Microsecondi", "nanoseconds": "Nanosecondi", "minutes": "Minuti", "hours": "Ore", "days": "Giorni", "weeks": "Settimane", "years": "Anni (365 d.)", "lustrum": "Lustri", "decades": "Decadi", "centuries": "Secoli", "millennium": "Millenni", "fahrenheit": "Fahrenheit", "celsius": "Celsius (Centigradi)", "kelvin": "Kelvin", "reamur": "Reamur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Metri al Secondo", "kilometersHour": "Chilometri Orari", "milesHour": "Miglia orarie", "knots": "Nodi", "feetSecond": "Piedi al secondo", "minutesPerKilometer": "Minuti al chilometro", "minutesPerMile": "Minuti al miglio", "speedOfLight": "Velocità della luce", "grams": "Grammi", "ettograms": "Ettogrammi", "kilograms": "Chilogrammi", "pounds": "Libbre", "ounces": "Once", "quintals": "Quintali", "tonnes": "Tonnellate", "milligrams": "Milligrammi", "uma": "Unità di Massa Atomica", "carats": "Carati", "centigrams": "Centigrammo", "pennyweights": "Pennyweights", "troyOunces": "Once troy", "stones": "Pietre", "pascal": "Pascal", "atmosphere": "Atmosfere", "bar": "Bar", "millibar": "Millibar", "psi": "Libbra su Pollice Quadrato", "torr": "Millimetri di mercurio (Torr)", "hectoPascal": "Ettopascal", "inchesOfMercury": "Pollici di mercurio", "kiloPascal": "Chilopascal", "ksi": "Chilolibbre per pollice quadrato", "megaPascal": "Megapascal", "gigaPascal": "Gigapascal", "joule": "Joule", "calories": "Calorie", "kilowattHour": "Chilowattora", "electronvolt": "Elettronvolt", "footPound": "Piede per libbra forza", "kilocalories": "Chilocalorie", "kilojoules": "Chilojoules", "watthour": "Wattora", "britishThermalUnit": "British thermal unit", "degree": "Gradi", "minutesDegree": "Primi", "secondsDegree": "Secondi", "radiansDegree": "Radianti", "usd": "Dollaro statunitense", "eur": "Euro", "gbp": "Sterlina britannica", "inr": "Rupia indiana", "cny": "Renminbi cinese", "jpy": "Yen giapponese", "chf": "Franco svizzero", "sek": "Corona svedese", "rub": "Rublo russo", "cad": "Dollaro canadese", "krw": "Won sudcoreano", "brl": "Real brasiliano", "hkd": "Dollaro di Hong Kong", "aud": "Dollaro australiano", "nzd": "Dollaro neozelandese", "mxn": "Peso messicano", "sgd": "Dollaro di Singapore", "nok": "Corona norvegese", "trY": "Lira turca", "zar": "Rand sudafricano", "dkk": "Corona danese", "pln": "Złoty polacco", "thb": "Baht thailandese", "myr": "Ringgit malese", "huf": "Fiorino ungherese", "czk": "Corona ceca", "ils": "Nuovo siclo israeliano", "idr": "Rupia indonesiana", "php": "Peso filippino", "ron": "Leu rumeno", "isk": "Corona islandese", "twd": "Nuovo dollaro taiwanese", "mad": "Dirham marocchino", "euChina": "EU e Cina", "ukIndiaChild": "UK e India - Bambino", "ukIndiaMan": "UK e India - Uomo", "ukIndiaWoman": "UK e India - Donna", "usaCanadaChild": "USA e Canada - Bambino", "usaCanadaMan": "USA e Canada - Uomo", "usaCanadaWoman": "USA e Canada - Donna", "japan": "Giappone", "watt": "Watt", "milliwatt": "Milliwatt", "kilowatt": "Chilowatt", "megawatt": "Megawatt", "gigawatt": "GigaWatt", "europeanHorsePower": "Cavalli vapore Europeo", "imperialHorsePower": "Cavalli vapore Imperiale", "newton": "Newton", "dyne": "Dyne", "poundForce": "Libbre forza", "kilogramForce": "Chilogrammi forza", "poundal": "Poundal", "newtonMeter": "Newton metro", "dyneMeter": "Dyne metro", "poundForceFeet": "Libbre-forza piede", "kilogramForceMeter": "Chilogrammmi-forza metro", "poundalMeter": "Poundal metro", "poundForceInch": "Libbre-forza pollice", "kilometersLiter": "Chilometri con un litro", "liters100km": "Litri ogni 100 km", "milesUsGallon": "Miglia per Gallone US", "milesImperialGallon": "Miglia per Gallone imperiale", "decimal": "Decimale", "hexadecimal": "Esadecimale", "octal": "Ottale", "binary": "Binario", "bit": "Bit", "nibble": "Nibble", "kilobit": "Kilobit decimale", "megabit": "Megabit decimale", "gigabit": "Gigabit decimale", "terabit": "Terait decimale", "petabit": "Petabit decimale", "exabit": "Exabit decimale", "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "Gibibit", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Exbibit", "byte": "Byte", "kilobyte": "Kilobyte decimale", "megabyte": "Megabyte decimale", "gigabyte": "Gigabyte decimale", "terabyte": "Terabyte decimale", "petabyte": "Petabyte decimale", "exabyte": "Exabyte decimale", "kibibyte": "Kibibyte", "mebibyte": "Mebibyte", "gibibyte": "Gibibyte", "tebibyte": "Tebibyte", "pebibyte": "Pebibyte", "exbibyte": "Exbibyte", "base": "Base", "deca": "Deca-", "hecto": "Etto-", "kilo": "Chilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Exa-", "zetta": "Zetta-", "yotta": "Yotta-", "deci": "Deci-", "centi": "Centi-", "milli": "Milli-", "micro": "Micro-", "nano": "Nano-", "pico": "Pico-", "femto": "Femto-", "atto": "Atto-", "zepto": "Zepto-", "yocto": "Yocto-" } ================================================ FILE: packages/translations/lib/l10n/app_ja.arb ================================================ { "@@locale": "ja", "appName": "Converter NOW", "reorder": "並び替え", "myOrdering": "マイ並び替え", "about": "このアプリについて", "lastCurrenciesUpdate": "最終更新: ", "today": "今日", "enableDarkTheme": "ダークテーマを有効化", "settings": "設定", "menu": "メニュー", "donation": "寄付をする", "buyMeACoffee": "開発者にコーヒーを購入", "donationDialog": "こんにちは、ご存知かもしれませんが、このアプリは無料でオープンソースです。あなたがこのアプリを無料で使用してソースコードをコピー、編集できることを意味します。開発者はそれに対して何も得られません。しかし、このプロジェクトに賛同し、新しい機能を見たい場合は、私にコーヒーを買うことを検討してください。", "drawerLogo": "ロゴドロワー", "rateApp": "アプリを評価する", "repoGithub": "GitHubでリポジトリを開く", "contributeTranslating": "アプリの翻訳に参加する", "search": "検索", "clearAll": "すべて削除", "calculator": "電卓", "significantFigures": "有効数字", "removeTrailingZeros": "末尾の0を削除する", "longPressAdvice": "長押しでアイテムを移動できます", "back": "戻る", "save": "保存", "invalidCharacters": "エラー、無効な文字", "theme": "テーマ", "dark": "ダーク", "light": "ライト", "system": "システムの設定", "pureBlackTheme": "AMOLEDダークテーマ", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "コピー", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "さらに", "language": "言語", "reorderProperties": "グループを並び替える", "reorderUnits": "項目を並び替える", "chooseProperty": "グループを選択してください", "reorderProperty": "{property}を並び替える", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "その他のプラットフォームで使用する", "sourceCode": "ソースコード", "undoClearAllMessage": "すべて削除しました。もとに戻しますか?", "undo": "元に戻す", "routeError1": "このページは存在しません。", "routeError2": "先頭ページに戻る", "ok": "Ok", "@ok": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetAccess": "Revoke internet access", "@revokeInternetAccess": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetExplanation": "This app uses the internet to update currency exchange rates once a day upon opening. This download only requires a few kilobytes. However, if you don't need this feature, you can prevent the app from accessing the internet by enabling this option.", "@revokeInternetExplanation": { "description": "Not yet translated. Once done, remove this comment" }, "useDeviceColor": "Use device color", "@useDeviceColor": { "description": "Not yet translated. Once done, remove this comment" }, "pickColor": "Pick a color", "@pickColor": { "description": "Not yet translated. Once done, remove this comment" }, "themeColor": "Theme color", "@themeColor": { "description": "Not yet translated. Once done, remove this comment" }, "appearance": "Appearance", "@appearance": { "description": "Not yet translated. Once done, remove this comment" }, "conversions": "Conversions", "@conversions": { "description": "Not yet translated. Once done, remove this comment" }, "findOutMore": "Find out more", "@findOutMore": { "description": "Not yet translated. Once done, remove this comment" }, "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "description": "Not yet translated. Once done, remove this comment", "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "長さ", "area": "面積", "volume": "体積", "time": "時間", "temperature": "温度", "speed": "速度", "siPrefixes": "国際単位変換", "mass": "質量", "pressure": "圧力", "energy": "エネルギー", "angles": "角度", "currencies": "通貨", "shoeSize": "靴のサイズ", "digitalData": "デジタルデータ", "power": "仕事率", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "トルク", "force": "力", "fuelConsumption": "燃費", "numeralSystems": "記数法", "meters": "メートル", "centimeters": "センチメートル", "inches": "インチ", "feet": "フィート", "nauticalMiles": "海里", "yards": "ヤード", "miles": "マイル", "millimeters": "ミリメートル", "micrometers": "マイクロメートル", "nanometers": "ナノメートル", "angstroms": "オングストローム", "picometers": "ピコメートル", "kilometers": "キロメートル", "astronomicalUnits": "天文単位", "lightYears": "光年", "parsec": "パーセク", "mils": "ミル", "feetUsSurvey": "測量フィート", "squareMeters": "平方メートル", "squareCentimeters": "平方センチメートル", "squareInches": "平方インチ", "squareFeet": "平方フィート", "squareMiles": "平方マイル", "squareYard": "平方ヤード", "squareMillimeters": "平方ミリメートル", "squareKilometers": "平方キロメートル", "hectares": "ヘクタール", "acres": "エーカー", "are": "アール", "squareFeetUsSurvey": "平方フィート(アメリカ)", "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "立方メートル", "liters": "リットル", "imperialGallons": "英ガロン", "usGallons": "米国乾量ガロン", "imperialPints": "英パイント", "usPints": "米乾量パイント", "milliliters": "ミリリットル", "tablespoonUs": "大さじ(アメリカ)", "tablespoonAustralian": "大さじ(オーストラリア)", "cups": "カップ", "cubicCentimeters": "立方センチメートル", "cubicFeet": "立方フィート", "cubicInches": "立方インチ", "cubicMillimeters": "立方ミリメートル", "imperialFluidOunces": "英液用オンス", "usFluidOunces": "米液用オンス", "imperialGill": "ジル(イギリス)", "usGill": "ジル(アメリカ)", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "秒", "deciseconds": "デシ秒", "centiseconds": "センチ秒", "milliseconds": "ミリ秒", "microseconds": "マイクロ秒", "nanoseconds": "ナノ秒", "minutes": "分", "hours": "時間", "days": "日", "weeks": "週", "years": "年 (365)", "lustrum": "5年紀", "decades": "10年紀", "centuries": "世紀", "millennium": "ミレニアム", "fahrenheit": "華氏", "celsius": "摂氏 (センティグレイド)", "kelvin": "ケルビン", "reamur": "レオミュール", "romer": "レーマー", "delisle": "ドリール", "rankine": "ランキン", "metersSecond": "メートル毎秒", "kilometersHour": "キロ毎時", "milesHour": "マイル毎時", "knots": "ノット", "feetSecond": "フィート毎秒", "minutesPerKilometer": "分毎キロ", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "グラム", "ettograms": "エット", "kilograms": "キログラム", "pounds": "ポンド", "ounces": "オンス", "quintals": "キンタル", "tonnes": "トン", "milligrams": "ミリグラム", "uma": "原子質量単位", "carats": "カラット", "centigrams": "センチグラム", "pennyweights": "ペニーウェイト", "troyOunces": "トロイオンス", "stones": "ストーン", "pascal": "パスカル", "atmosphere": "気圧", "bar": "バール", "millibar": "ミリバール", "psi": "重量ポンド毎平方インチ", "torr": "トル", "hectoPascal": "ヘクトパスカル", "inchesOfMercury": "水銀柱インチ", "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "ジュール", "calories": "カロリー", "kilowattHour": "キロワット時", "electronvolt": "電子ボルト", "footPound": "フィート重量ポンド", "kilocalories": "キロカロリー", "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "角度", "minutesDegree": "分", "secondsDegree": "秒", "radiansDegree": "ラジアン", "usd": "米ドル", "eur": "ユーロ", "gbp": "スターリングポンド", "inr": "インドルピー", "cny": "人民元", "jpy": "日本円", "chf": "スイスフラン", "sek": "スウェーデンクローナ", "rub": "ロシアルーブル", "cad": "カナダドル", "krw": "大韓民国ウォン", "brl": "レアル", "hkd": "香港ドル", "aud": "豪ドル", "nzd": "NZドル", "mxn": "メキシコペソ", "sgd": "シンガポールドル", "nok": "ノルウェークローネ", "trY": "トルコリラ", "zar": "ランド", "dkk": "デンマーククローネ", "pln": "ズウォティ", "thb": "バーツ", "myr": "リンギット", "huf": "フォリント", "czk": "チェココルナ", "ils": "新シェケル", "idr": "ルピア", "php": "フィリピンペソ", "ron": "ルーマニアレウ", "isk": "アイスランドクローナ", "twd": "新台湾ドル", "mad": "モロッコディルハム", "euChina": "EU & 中国", "ukIndiaChild": "イギリス & インド - 子供", "ukIndiaMan": "イギリス & インド - 男性", "ukIndiaWoman": "イギリス & インド - 女性", "usaCanadaChild": "アメリカ & ・カナダ - 子供", "usaCanadaMan": "アメリカ & カナダ - 男性", "usaCanadaWoman": "アメリカ & カナダ - 女性", "japan": "日本", "watt": "ワット", "milliwatt": "ミリワット", "kilowatt": "キロワット", "megawatt": "メガワット", "gigawatt": "ギガワット", "europeanHorsePower": "ヨーロッパ馬力", "imperialHorsePower": "英馬力", "newton": "ニュートン", "dyne": "ダイン", "poundForce": "重量ポンド", "kilogramForce": "重量キログラム", "poundal": "パウンダル", "newtonMeter": "ニュートンメートル", "dyneMeter": "ダインメートル", "poundForceFeet": "フィート重量ポンド", "kilogramForceMeter": "重量キログラムメートル", "poundalMeter": "パウンダルメートル", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "キロメートル毎リットル", "liters100km": "100キロメートル毎リットル", "milesUsGallon": "マイル毎米国液量ガロン", "milesImperialGallon": "マイル毎英ガロン", "decimal": "10進数", "hexadecimal": "16進数", "octal": "8進数", "binary": "2進数", "bit": "ビット", "nibble": "ビブル", "kilobit": "キロビット", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "メガビット", "@megabit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "ギガビット", "@gigabit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "テラビット", "@terabit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "ペタビット", "@petabit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "エクサビット", "@exabit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "キビビット", "mebibit": "メビビット", "gibibit": "ギビビット", "tebibit": "テビビット", "pebibit": "ペビビット", "exbibit": "エクスビビット", "byte": "バイト", "kilobyte": "キロバイト", "@kilobyte": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "メガバイト", "@megabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "ギガバイト", "@gigabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "テラバイト", "@terabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "ペタバイト", "@petabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "エクサバイト", "@exabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "キビバイト", "mebibyte": "メビバイト", "gibibyte": "ギビバイト", "tebibyte": "テビバイト", "pebibyte": "ペビバイト", "exbibyte": "エクサビバイト", "base": "基数", "deca": "デカ-", "hecto": "ヘクト-", "kilo": "キロ-", "mega": "メガ-", "giga": "ギガ-", "tera": "テラ-", "peta": "ペタ-", "exa": "エクサ-", "zetta": "ゼタ-", "yotta": "ヨタ-", "deci": "デシ-", "centi": "センチ-", "milli": "ミリ-", "micro": "マイクロ-", "nano": "ナノ-", "pico": "ピコ-", "femto": "フェムト-", "atto": "アト-", "zepto": "ゼプト-", "yocto": "ヨクト-" } ================================================ FILE: packages/translations/lib/l10n/app_nb.arb ================================================ { "@@locale": "nb", "appName": "Converter NOW", "reorder": "Endre rekkefølge", "myOrdering": "Min rekkefølge", "about": "Om", "lastCurrenciesUpdate": "Siste oppdatering: ", "today": "i dag", "enableDarkTheme": "Aktiver mørkt tema", "settings": "Innstillinger", "menu": "Meny", "donation": "Gi et bidrag", "buyMeACoffee": "Buy me a coffee", "donationDialog": "Hi, as you might know this app is free and open source. It means that you can use it for free and copy/edit the source code. The developer will not gain nothing for it. But if you also believe in this project and would like to see new features, please consider buying me a coffee.", "drawerLogo": "Menylogo", "rateApp": "Bedøm appen", "repoGithub": "Åpne repo på GitHub", "contributeTranslating": "Bidra med oversettelse av appen", "search": "Søk", "clearAll": "Fjern alle", "calculator": "Kalkulator", "significantFigures": "Gjeldende sifre", "removeTrailingZeros": "Fjern etterfølgende nuller", "longPressAdvice": "Lang-trykk for å flytte elementer", "back": "Tilbake", "save": "Lagre", "invalidCharacters": "Feil, ugyldige tegn", "theme": "Tema", "dark": "Mørkt", "light": "Lyst", "system": "System", "pureBlackTheme": "Rent svart tema", "propertySelectionOnStartup": "Kategorivalg ved oppstart", "propertySelectionOnStartupSubtitle": "Vis siden med kategorivalg ved oppstart på enheter med liten skjerm", "copy": "Kopier", "paste": "Lim inn", "more": "Mer", "language": "Språk", "reorderProperties": "Endre rekkefølge på kategorier", "reorderUnits": "Endre rekkefølge på måleenheter", "chooseProperty": "Velg en kategori", "reorderProperty": "Endre rekkefølge for {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW for andre plattformer", "sourceCode": "Kildekode", "undoClearAllMessage": "Fjernet alle. Vil du angre det?", "undo": "Angre", "routeError1": "Denne siden finnes ikke", "routeError2": "Gå tilbake til den første siden", "ok": "OK", "revokeInternetAccess": "Tilbakekall internettilgang", "revokeInternetExplanation": "Denne appen bruker internett til å oppdatere valutakurser en gang om dagen når den åpnes. Denne nedlastingen krever bare noen få kilobyte. Hvis du ikke trenger denne funksjonen, kan du imidlertid forhindre at appen får tilgang til internett ved å aktivere dette alternativet.", "useDeviceColor": "Bruk enhetens farger", "pickColor": "Velg en farge", "themeColor": "Temafarge", "appearance": "Utseende", "conversions": "Omregninger", "findOutMore": "Finn ut mer", "hideUnits": "Skjul enheter", "visibleUnits": "{property}, synlige enheter", "@visibleUnits": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Skjulte enheter", "selectAll": "Velg alle", "unselectAll": "Fravelg alle", "backupAndRestore": "Sikkerhetskopier & Gjenopprett", "exportSettings": "Eksporter innstillinger", "importSettings": "Importer innstillinger", "clearSettings": "Tøm alle innstillinger", "exportSuccess": "Innstillinger eksportert", "importSuccess": "Innstillinger importert", "problemImportFile": "Det var et problem med importering av filen", "relatedSettings": "Relaterte innstillinger", "reason": "Årsak", "importError": "Importfeil", "confirmationClear": "Er du sikker på at du vil tømme alle innstillinger? Denne handlingen kan ikke angres.", "cancel": "Avbryt", "length": "Lengde", "area": "Areal", "volume": "Volum", "time": "Tid", "temperature": "Temperatur", "speed": "Hastighet", "siPrefixes": "SI-prefiks", "mass": "Masse", "pressure": "Trykk", "energy": "Energi", "angles": "Vinkel", "currencies": "Valuta", "shoeSize": "Skostørrelse", "digitalData": "Digitale data", "power": "Effekt", "density": "Tetthet", "torque": "Dreiemoment", "force": "Kraft", "fuelConsumption": "Brenselsforbruk", "numeralSystems": "Tallsystem", "meters": "Meter", "centimeters": "Centimeter", "inches": "Tomme", "feet": "Fot", "nauticalMiles": "Nautisk mil", "yards": "Yard", "miles": "Engelsk mil", "millimeters": "Millimeter", "micrometers": "Mikrometer", "nanometers": "Nanometer", "angstroms": "Ångström", "picometers": "Pikometer", "kilometers": "Kilometer", "astronomicalUnits": "Astronomisk enhet", "lightYears": "Lysår", "parsec": "Parsec", "mils": "Mils", "feetUsSurvey": "Fot (US survey)", "squareMeters": "Kvadratmeter", "squareCentimeters": "Kvadratcentimeter", "squareInches": "Kvadrattomme", "squareFeet": "Kvadratfot", "squareMiles": "Kvadratmil (eng.)", "squareYard": "Kvadratyard", "squareMillimeters": "Kvadratmillimeter", "squareKilometers": "Kvadratkilometer", "hectares": "Hektar", "acres": "Acre", "are": "Are", "squareFeetUsSurvey": "Kvadratfot (US survey)", "gramsPerCubicCentimeter": "Gram per kubikkcentimeter", "gramsPerLiter": "Gram per liter", "gramsPerMilliliter": "Gram per milliliter", "gramsPerDeciliter": "Gram per desiliter", "kilogramsPerLiter": "Kilogram per liter", "kilogramsPerCubicMeter": "Kilogram per kubikkmeter", "milligramsPerLiter": "Milligram per liter", "milligramsPerDeciliter": "Milligram per desiliter", "milligramsPerMilliliter": "Milligram per milliliter", "milligramsPerCubicMeter": "Milligram per kubikkmeter", "milligramsPerCubicCentimeter": "Milligram per kubikkcentimeter", "microgramsPerLiter": "Mikrogram per liter", "microgramsPerDeciliter": "Mikrogram per desiliter", "microgramsPerMilliliter": "Mikrogram per milliliter", "poundsPerCubicInches": "Pund per kubikktomme", "poundsPerCubicFeet": "Pund per kubikkfot", "cubicMeters": "Kubikkmeter", "liters": "Liter", "imperialGallons": "Britisk gallon", "usGallons": "US gallon", "imperialPints": "Britisk pint", "usPints": "US pint", "milliliters": "Milliliter", "tablespoonUs": "Spiseskje US", "tablespoonAustralian": "Spiseskje australsk", "cups": "Cup", "cubicCentimeters": "Kubikkcentimeter", "cubicFeet": "Kubikkfot", "cubicInches": "Kubikktomme", "cubicMillimeters": "Kubikkmillimeter", "imperialFluidOunces": "Imperial Fluid Ounce", "usFluidOunces": "US Fluid Ounce", "imperialGill": "Britisk Gill", "usGill": "US Gill", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Mikroliter", "deciliters": "Desiliter", "centiliters": "Centiliter", "teaspoonsUs": "Teskje US", "teaspoonsMetric": "Teskje metrisk", "seconds": "Sekund", "deciseconds": "Desisekund", "centiseconds": "Centisekund", "milliseconds": "Millisekund", "microseconds": "Mikrosekund", "nanoseconds": "Nanosekund", "minutes": "Minutt", "hours": "Time", "days": "Dag", "weeks": "Uke", "years": "År (365)", "lustrum": "Lustrum", "decades": "Tiår", "centuries": "Århundre", "millennium": "Millennium", "fahrenheit": "Fahrenheit", "celsius": "Celsius", "kelvin": "Kelvin", "reamur": "Reamur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Meter per sekund", "kilometersHour": "Kilometer per time", "milesHour": "Engelsk mil per time", "knots": "Knop", "feetSecond": "Fot per sekund", "minutesPerKilometer": "Minutter per kilometer", "minutesPerMile": "Minutter per eng. mil", "speedOfLight": "Lysets hastighet", "grams": "Gram", "ettograms": "Ettogram", "kilograms": "Kilogram", "pounds": "Pund", "ounces": "Ounce", "quintals": "Centner", "tonnes": "Tonn", "milligrams": "Milligram", "uma": "Atommasseenhet", "carats": "Karat", "centigrams": "Centigram", "pennyweights": "Pennyweight", "troyOunces": "Troy ounce", "stones": "Stone", "pascal": "Pascal", "atmosphere": "Atmosfære", "bar": "Bar", "millibar": "Millibar", "psi": "Pund per kvadrattomme", "torr": "Millimeter-kvikksølv (Torr)", "hectoPascal": "Hektopascal", "inchesOfMercury": "Tommer kvikksølv", "kiloPascal": "Kilopascal", "ksi": "Kilopund per kvadrattomme", "megaPascal": "Megapascal", "gigaPascal": "Gigapascal", "joule": "Joule", "calories": "Kalori", "kilowattHour": "Kilowatt-time", "electronvolt": "Elektronvolt", "footPound": "Foot-pound", "kilocalories": "Kilokalori", "kilojoules": "Kilojoule", "watthour": "Watt-time", "britishThermalUnit": "Britisk varmeenhet", "degree": "Grad", "minutesDegree": "Bueminutt", "secondsDegree": "Buesekund", "radiansDegree": "Radian", "usd": "Amerikanske dollar", "eur": "Euro", "gbp": "Britiske pund", "inr": "Indiske rupier", "cny": "Kinesiske yuan", "jpy": "Japanske yen", "chf": "Sveitsiske franc", "sek": "Svenske kroner", "rub": "Russiske rubler", "cad": "Kanadiske dollar", "krw": "Sørkoreanske won", "brl": "Brasilianske real", "hkd": "Hongkong-dollar", "aud": "Australske dollar", "nzd": "Newzealandske dollar", "mxn": "Meksikanske pesos", "sgd": "Singaporske dollar", "nok": "Norske kroner", "trY": "Tyrkiske lira", "zar": "Sørafrikanske rand", "dkk": "Danske kroner", "pln": "Polske zloty", "thb": "Thailandske baht", "myr": "Malaysiske ringgit", "huf": "Ungarske forinter", "czk": "Tsjekkiske koruna", "ils": "Nye israelske shekler", "idr": "Indonesiske rupiah", "php": "Filippinske pesos", "ron": "Rumenske leu", "isk": "Islandske kroner", "twd": "Nye taiwanske dollar", "mad": "Marokkanske dirham", "euChina": "EU & Kina", "ukIndiaChild": "UK & India - Barn", "ukIndiaMan": "UK & India - Herre", "ukIndiaWoman": "UK & India - Dame", "usaCanadaChild": "USA & Canada - Barn", "usaCanadaMan": "USA & Canada - Herre", "usaCanadaWoman": "USA & Canada - Dame", "japan": "Japan", "watt": "Watt", "milliwatt": "Milliwatt", "kilowatt": "Kilowatt", "megawatt": "Megawatt", "gigawatt": "Gigawatt", "europeanHorsePower": "Europeisk hestekraft", "imperialHorsePower": "Britisk hestekraft", "newton": "Newton", "dyne": "Dyn", "poundForce": "Pound-force", "kilogramForce": "Kilogramkraft", "poundal": "Poundal", "newtonMeter": "Newtonmeter", "dyneMeter": "Dynmeter", "poundForceFeet": "Pound-foot", "kilogramForceMeter": "Kilogramkraftmeter", "poundalMeter": "Poundal meter", "poundForceInch": "Pund-kraft tomme", "kilometersLiter": "Kilometer per liter", "liters100km": "Liter per 100 km", "milesUsGallon": "Mile per gallon (US)", "milesImperialGallon": "Engelsk mil per britisk gallon", "decimal": "Desimal", "hexadecimal": "Heksadesimal", "octal": "Oktal", "binary": "Binær", "bit": "Bit", "nibble": "Nibble", "kilobit": "Kilobit (desimal)", "megabit": "Megabit (desimal)", "gigabit": "Gigabit (desimal)", "terabit": "Terabit (desimal)", "petabit": "Petabit (desimal)", "exabit": "Exabit (desimal)", "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "Gibibit", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Exbibit", "byte": "Byte", "kilobyte": "Kilobyte (desimal)", "megabyte": "Megabyte (desimal)", "gigabyte": "Gigabyte (desimal)", "terabyte": "Terabyte (desimal)", "petabyte": "Petabyte (desimal)", "exabyte": "Exabyte (desimal)", "kibibyte": "Kibibyte", "mebibyte": "Mebibyte", "gibibyte": "Gibibyte", "tebibyte": "Tebibyte", "pebibyte": "Pebibyte", "exbibyte": "Exbibyte", "base": "Base", "deca": "Deka-", "hecto": "Hekto-", "kilo": "Kilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Exa-", "zetta": "Zetta-", "yotta": "Yotta-", "deci": "Desi-", "centi": "Centi-", "milli": "Milli-", "micro": "Mikro-", "nano": "Nano-", "pico": "Pico-", "femto": "Femto-", "atto": "Atto-", "zepto": "Zepto-", "yocto": "Yocto-" } ================================================ FILE: packages/translations/lib/l10n/app_nl.arb ================================================ { "@@locale": "nl", "appName": "Converter NOW", "reorder": "Opnieuw ordenen", "myOrdering": "Mijn ordening", "about": "Over", "lastCurrenciesUpdate": "Laatste update: ", "today": "vandaag", "enableDarkTheme": "Schakel naar donker thema", "settings": "Instellingen", "menu": "Menu", "donation": "Doe een donatie", "buyMeACoffee": "Trakteer me op een koffie'tje", "donationDialog": "Hoi, zoals je misschien weet is deze app gratis en open source. Dat betekent dat je de app gratis kunt gebruiken en de broncode kunt kopiëren/bewerken. De ontwikkelaar maakt hier geen winst op, maar als je gelooft in dit project en nieuwe functies zou willen zien, overweeg me dan op een koffie'tje te trakteren.", "drawerLogo": "Lade logo", "rateApp": "Beoordeel de app", "repoGithub": "Open de repo op GitHub", "contributeTranslating": "Draag bij aan het vertalen van de app", "search": "Zoekn", "clearAll": "Wis alles", "calculator": "Rekenmachine", "significantFigures": "Significante getallen", "removeTrailingZeros": "Verwijder de laatste nullen", "longPressAdvice": "Houd ingedrukt om te verplaatsen", "back": "Terug", "save": "Opslaan", "invalidCharacters": "Fout, ongeldige karakters", "theme": "Thema", "dark": "Donker", "light": "Licht", "system": "Systeem standaard", "pureBlackTheme": "AMOLED donker thema", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "Kopiëren", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "Meer", "language": "Taal", "reorderProperties": "Meetwaarde opnieuw ordenen", "reorderUnits": "Eenheden opnieuw ordenen", "chooseProperty": "Kies een property", "reorderProperty": "Herorden {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW voor andere platforms", "sourceCode": "Broncode", "undoClearAllMessage": "Alles gewist. Ongedaan maken?", "undo": "Ongedaan maken", "routeError1": "Deze pagina lijkt niet te bestaan", "routeError2": "Ga terug naar de eerste pagina", "ok": "Ok", "@ok": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetAccess": "Revoke internet access", "@revokeInternetAccess": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetExplanation": "This app uses the internet to update currency exchange rates once a day upon opening. This download only requires a few kilobytes. However, if you don't need this feature, you can prevent the app from accessing the internet by enabling this option.", "@revokeInternetExplanation": { "description": "Not yet translated. Once done, remove this comment" }, "useDeviceColor": "Use device color", "@useDeviceColor": { "description": "Not yet translated. Once done, remove this comment" }, "pickColor": "Pick a color", "@pickColor": { "description": "Not yet translated. Once done, remove this comment" }, "themeColor": "Theme color", "@themeColor": { "description": "Not yet translated. Once done, remove this comment" }, "appearance": "Appearance", "@appearance": { "description": "Not yet translated. Once done, remove this comment" }, "conversions": "Conversions", "@conversions": { "description": "Not yet translated. Once done, remove this comment" }, "findOutMore": "Find out more", "@findOutMore": { "description": "Not yet translated. Once done, remove this comment" }, "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "description": "Not yet translated. Once done, remove this comment", "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "Lengte", "area": "Oppervlak", "volume": "Inhoud", "time": "Tijd", "temperature": "Temperatuur", "speed": "Snelheid", "siPrefixes": "SI Eenheden", "mass": "Massa", "pressure": "Druk", "energy": "Energie", "angles": "Hoeken", "currencies": "Valuta", "shoeSize": "Shoenmaten", "digitalData": "Digitale data", "power": "Vermogen", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "Koppel", "force": "Kracht", "fuelConsumption": "Brandstof verbruik", "numeralSystems": "Numerieke systemen", "meters": "Meter", "centimeters": "Centimeter", "inches": "Inches", "feet": "Voet", "nauticalMiles": "Britse zeemijl", "yards": "Yards", "miles": "Mijl", "millimeters": "Millimeter", "micrometers": "Micrometer", "nanometers": "Nanometer", "angstroms": "Ångström", "picometers": "Picometer", "kilometers": "Kilometer", "astronomicalUnits": "Astronomische Eenheden", "lightYears": "Lichtjaar", "parsec": "Parsec", "mils": "Mils", "feetUsSurvey": "Voet (Amerikaans)", "squareMeters": "Vierkante Meter", "squareCentimeters": "Vierkante Centimeter", "squareInches": "Vierkante Inch", "squareFeet": "Vierkante Voet", "squareMiles": "Vierkante Mijl", "squareYard": "Vierkante Yard", "squareMillimeters": "Vierkante Millimeter", "squareKilometers": "Vierkante Kilometer", "hectares": "Hectare", "acres": "Acre", "are": "Are", "squareFeetUsSurvey": "Vierkante Voet (Amerikaans)", "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "Kubieke Meter", "liters": "Liters", "imperialGallons": "Britse Gallon", "usGallons": "Amerikaanse Gallon", "imperialPints": "Britse Pint", "usPints": "Amerikaanse Pint", "milliliters": "Milliliter", "tablespoonUs": "Eetlepel", "tablespoonAustralian": "Australische Eetlepel", "cups": "Kopjes (Cups)", "cubicCentimeters": "Kubieke Centimeter", "cubicFeet": "Kubieke Voet", "cubicInches": "Kubieke Inch", "cubicMillimeters": "Kubieke Millimeter", "imperialFluidOunces": "Britse Vloeibare Ounce", "usFluidOunces": "Amerikaanse Vloeibare Ounce", "imperialGill": "Britse Gill (kwart pint)", "usGill": "Amerikaanse Gill (kwart pint)", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "Seconden", "deciseconds": "Deciseconden", "centiseconds": "Centiseconden", "milliseconds": "Milliseconden", "microseconds": "Microseconden", "nanoseconds": "Nanoseconden", "minutes": "Minuten", "hours": "Uren", "days": "Dagen", "weeks": "Weken", "years": "Jaren (365)", "lustrum": "Lustrum", "decades": "Decennia", "centuries": "Eeuwen", "millennium": "Millennia", "fahrenheit": "Fahrenheit", "celsius": "Celsius (Centigrade)", "kelvin": "Kelvin", "reamur": "Reamur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Meters per Seconde", "kilometersHour": "Kilometers per Uur", "milesHour": "Mijl per Uur", "knots": "Knopen", "feetSecond": "Voet per Seconde", "minutesPerKilometer": "Minuten per Kilometer", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "Gram", "ettograms": "Ettogram", "kilograms": "Kilogram", "pounds": "Pond", "ounces": "Ons", "quintals": "Quintals", "tonnes": "Ton", "milligrams": "Milligram", "uma": "Unified atomic mass unit", "carats": "Karaat", "centigrams": "Centigram", "pennyweights": "Pennyweights", "troyOunces": "Troy ounce", "stones": "Stones", "pascal": "Pascal", "atmosphere": "Atmosfeer", "bar": "Bar", "millibar": "Millibar", "psi": "Pond per Vierkante Inch", "torr": "Torricelli", "hectoPascal": "Hectopascal", "inchesOfMercury": "Inch-Kwikdruk", "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "Joule", "calories": "Calorieën", "kilowattHour": "Kilowattuur", "electronvolt": "Electron volt", "footPound": "Voet pond", "kilocalories": "Kilocalorieën", "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "Graden", "minutesDegree": "Minuten", "secondsDegree": "Seconden", "radiansDegree": "Radialen", "usd": "Amerikaanse Dollar", "eur": "Euro", "gbp": "Britse Pound", "inr": "Indiase Rupee", "cny": "Chinese Yuan", "jpy": "Japanese Yen", "chf": "Zwitserse Franc", "sek": "Zweedse Krona", "rub": "Russische Roebel", "cad": "Canadese Dollar", "krw": "Zuid-Koreaanse Won", "brl": "Braziliaanse Real", "hkd": "Hong Kong Dollar", "aud": "Australische Dollar", "nzd": "Nieuw-Zeelandse Dollar", "mxn": "Mexicaanse Peso", "sgd": "Singaporese Dollar", "nok": "Norweegse Krone", "trY": "Turkse Lira", "zar": "Zuid-Afrikaanse Rand", "dkk": "Deense Krone", "pln": "Poolse Złoty", "thb": "Thaise Baht", "myr": "Malaysische Ringgit", "huf": "Hongaarse Forint", "czk": "Tjechische Koruna", "ils": "Israelische Shekel", "idr": "Indonesische Rupiah", "php": "Filipijnse Peso", "ron": "Roemeense Leu", "isk": "Icelandse Króna", "twd": "Nieuw-Taiwanese Dollar", "mad": "Morokaanse Dirham", "euChina": "EU & China", "ukIndiaChild": "Groot Brittannië & India - Kind", "ukIndiaMan": "Groot Brittanië & India - Man", "ukIndiaWoman": "Groot Brittannië & India - Vrouw", "usaCanadaChild": "Verenigde Staten & Canada - Kind", "usaCanadaMan": "Verenigde Staten & Canada - Man", "usaCanadaWoman": "Verenigde Staten & Canada - Vrouw", "japan": "Japan", "watt": "Watt", "milliwatt": "Milliwatt", "kilowatt": "Kilowatt", "megawatt": "Megawatt", "gigawatt": "GigaWatt", "europeanHorsePower": "Europese Paardenkracht", "imperialHorsePower": "Metrische Paardenkracht", "newton": "Newton", "dyne": "Dyne", "poundForce": "Pound-kracht", "kilogramForce": "Kilogram-kracht", "poundal": "Poundal", "newtonMeter": "Newtonmeter", "dyneMeter": "Dynemeter", "poundForceFeet": "Pound-kracht voet", "kilogramForceMeter": "Kilogram-kracht meter", "poundalMeter": "Poundal meter", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "Kilometer per liter", "liters100km": "Liter per 100 km", "milesUsGallon": "Mijl per Amerikaanse Gallon", "milesImperialGallon": "Mijl per Britse Gallon", "decimal": "Decimaal", "hexadecimal": "Hexadecimaal", "octal": "Octaal", "binary": "Binair", "bit": "Bit", "nibble": "Nibble", "kilobit": "Kilobit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "Megabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "Gigabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "Terabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "Petabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "Exabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "Gibibit", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Exbibit", "byte": "Byte", "kilobyte": "Kilobyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "Megabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "Gigabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "Terabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "Petabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "Exabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "Kibibyte", "mebibyte": "Mebibyte", "gibibyte": "Gibibyte", "tebibyte": "Tebibyte", "pebibyte": "Pebibyte", "exbibyte": "Exbibyte", "base": "Base", "deca": "Deca-", "hecto": "Hecto-", "kilo": "Kilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Exa-", "zetta": "Zetta-", "yotta": "Yotta-", "deci": "Deci-", "centi": "Centi-", "milli": "Milli-", "micro": "Micro-", "nano": "Nano-", "pico": "Pico-", "femto": "Femto-", "atto": "Atto-", "zepto": "Zepto-", "yocto": "Yocto-" } ================================================ FILE: packages/translations/lib/l10n/app_pl.arb ================================================ { "@@locale": "pl", "appName": "Converter NOW", "reorder": "Zmień kolejność", "myOrdering": "Moja kolejność", "about": "O aplikacji", "lastCurrenciesUpdate": "Zaktualizowano: ", "today": "dziś", "enableDarkTheme": "Włącz ciemny motyw", "settings": "Ustawienia", "menu": "Menu", "donation": "Przekaż darowiznę", "buyMeACoffee": "Kup mi kawę", "donationDialog": "Cześć, jak być może wiesz, ta aplikacja jest darmowa i posiada otwarty kod źródłowy. Oznacza to, że możesz używać jej za darmo, kopiować/edytować kod źródłowy. Programista nic z tego nie ma. Ale jeśli również wierzysz w ten projekt i chcesz zobaczyć nowe funkcje, proszę rozważ kupienie mi kawy.", "drawerLogo": "Logo szuflady", "rateApp": "Oceń aplikację", "repoGithub": "Otwórz repozytorium na GitHubie", "contributeTranslating": "Przetłumacz tę aplikację", "search": "Szukaj", "clearAll": "Wyczyść wszystko", "calculator": "Kalkulator", "significantFigures": "Liczby znaczące", "removeTrailingZeros": "Usuń zera po przecinku", "longPressAdvice": "Długie przytrzymanie pozwala na przenoszenie elementów", "back": "Wstecz", "save": "Zapisz", "invalidCharacters": "Błąd, nieprawidłowe znaki", "theme": "Motyw", "dark": "Ciemny", "light": "Jasny", "system": "Domyślne ustawienie systemu", "pureBlackTheme": "Motyw czystej czerni", "propertySelectionOnStartup": "Wybór miar podczas uruchamiania", "propertySelectionOnStartupSubtitle": "Pokazuj stronę wyboru miar podczas uruchamiania na urządzeniach z małym ekranem", "copy": "Kopiuj", "paste": "Wklej", "more": "Więcej", "language": "Język", "reorderProperties": "Zmień kolejność miar", "reorderUnits": "Zmień kolejność jednostek", "chooseProperty": "Wybierz miarę", "reorderProperty": "Zmień kolejność {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW dla innych platform", "sourceCode": "Kod źródłowy", "undoClearAllMessage": "Wszystko wyczyszczone. Chcesz to cofnąć?", "undo": "Cofnij", "routeError1": "Ta strona nie istnieje", "routeError2": "Wróć do pierwszej strony", "ok": "Ok", "revokeInternetAccess": "Odwołanie dostępu do Internetu", "revokeInternetExplanation": "Ta aplikacja korzysta z Internetu, aby aktualizować kursy wymiany walut raz dziennie, po uruchomieniu aplikacji. Pobierane jest wtedy tylko kilka kilobajtów danych. Jeśli jednak nie potrzebujesz tej funkcji, możesz uniemożliwić aplikacji dostęp do Internetu, włączając tę opcję.", "useDeviceColor": "Używaj kolorów urządzenia", "pickColor": "Wybierz kolor", "themeColor": "Kolor motywu", "appearance": "Wygląd", "conversions": "Konwersje", "findOutMore": "Więcej informacji", "hideUnits": "Ukrywaj jednostki", "visibleUnits": "{property}, widoczne jednostki", "@visibleUnits": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Ukryte jednostki", "selectAll": "Zaznacz wszystkie", "unselectAll": "Odznacz wszystkie", "backupAndRestore": "Kopie zapasowe oraz przywracanie danych", "exportSettings": "Wyeksportuj ustawienia", "importSettings": "Zaimportuj ustawienia", "clearSettings": "Wyczyść wszystkie ustawienia", "exportSuccess": "Ustawienia zostały pomyślnie wyeksportowane", "importSuccess": "Ustawienia zostały pomyślnie zaimportowane", "problemImportFile": "Wystąpił problem podczas importowania pliku", "relatedSettings": "Powiązane ustawienia", "reason": "Powód", "importError": "Błąd podczas importowania", "confirmationClear": "Czy na pewno chcesz skasować wszystkie ustawienia? Tego działania nie można cofnąć.", "cancel": "Anuluj", "length": "Długość", "area": "Powierzchnia", "volume": "Objętość", "time": "Czas", "temperature": "Temperatura", "speed": "Prędkość", "siPrefixes": "Przedrostek SI", "mass": "Masa", "pressure": "Ciśnienie", "energy": "Energia", "angles": "Stopnie", "currencies": "Waluty", "shoeSize": "Rozmiar buta", "digitalData": "Dane cyfrowe", "power": "Moc", "density": "Gęstość", "torque": "Moment obrotowy", "force": "Siła", "fuelConsumption": "Zużycie paliwa", "numeralSystems": "Systemy liczbowe", "meters": "Metry", "centimeters": "Centymetry", "inches": "Cale", "feet": "Stopy", "nauticalMiles": "Mile morskie", "yards": "Jardy", "miles": "Mile", "millimeters": "Milimetry", "micrometers": "Mikrometry", "nanometers": "Nanometry", "angstroms": "Angstremy", "picometers": "Pikometry", "kilometers": "Kilometry", "astronomicalUnits": "Jednostki astronomiczne", "lightYears": "Lata świetlne", "parsec": "Parsek", "mils": "Mil", "feetUsSurvey": "Stopy (US survey)", "squareMeters": "Metry kwadratowe", "squareCentimeters": "Centymetry kwadratowe", "squareInches": "Cale kwadratowe", "squareFeet": "Stopy kwadratowe", "squareMiles": "Mile kwadratowe", "squareYard": "Jardy kwadratowe", "squareMillimeters": "Milimetry kwadratowe", "squareKilometers": "Kilometry kwadratowe", "hectares": "Hektary", "acres": "Akry", "are": "Ary", "squareFeetUsSurvey": "Stopy kwadratowe (US survey)", "gramsPerCubicCentimeter": "Gram na centymetr sześcienny", "gramsPerLiter": "Gram na litr", "gramsPerMilliliter": "Gram na mililitr", "gramsPerDeciliter": "Gram na decylitr", "kilogramsPerLiter": "Kilogram na litr", "kilogramsPerCubicMeter": "Kilogram na meter sześcienny", "milligramsPerLiter": "Miligram na litr", "milligramsPerDeciliter": "Miligram na decylitr", "milligramsPerMilliliter": "Miligram na mililitr", "milligramsPerCubicMeter": "Miligram na metr sześcienny", "milligramsPerCubicCentimeter": "Miligram na centymetr sześcienny", "microgramsPerLiter": "Mikrogram na litr", "microgramsPerDeciliter": "Mikrogram na decylitr", "microgramsPerMilliliter": "Mikrogram na mililitr", "poundsPerCubicInches": "Funt na cal sześcienny", "poundsPerCubicFeet": "Funt na stopę sześcienną", "cubicMeters": "Metry sześcienne", "liters": "Litry", "imperialGallons": "Galony angielskie", "usGallons": "Galony amerykańskie", "imperialPints": "Pinty angielskie", "usPints": "Pinty amerykańskie", "milliliters": "Mililitry", "tablespoonUs": "Łyżki stołowe amerykańskie", "tablespoonAustralian": "Łyżki stołowe australijskie", "cups": "Cupy", "cubicCentimeters": "Centymetry sześcienne", "cubicFeet": "Stopy sześcienne", "cubicInches": "Cale sześcienne", "cubicMillimeters": "Milimetry sześcienne", "imperialFluidOunces": "Angielskie uncje płynne", "usFluidOunces": "Amerykańskie uncje płynne", "imperialGill": "Angielski gill", "usGill": "Amerykański gill", "usQuarts": "Kwarta amerykańska", "microliters": "Mikrolitry", "deciliters": "Decylitry", "centiliters": "Centylitry", "teaspoonsUs": "Łyżeczki (amerykańskie)", "teaspoonsMetric": "Łyżeczki (miara metryczna)", "seconds": "Sekundy", "deciseconds": "Decysekundy", "centiseconds": "Centysekundy", "milliseconds": "Milisekundy", "microseconds": "Mikrosekundy", "nanoseconds": "Nanosekundy", "minutes": "Minuty", "hours": "Godziny", "days": "Dni", "weeks": "Tygodnie", "years": "Lata (365)", "lustrum": "Lustrum", "decades": "Dekady", "centuries": "Stulecia", "millennium": "Tysiąclecia", "fahrenheit": "Stopnie Fahrenheita", "celsius": "Stopnie Celcjusza", "kelvin": "Stopnie Kelwina", "reamur": "Stopnie Réaumura", "romer": "Stopnie Rømera", "delisle": "Stopnie Delisle’a", "rankine": "Stopnie Rankine’a", "metersSecond": "Metry na sekundę", "kilometersHour": "Kilometry na godzinę", "milesHour": "Mile na godzinę", "knots": "Węzły", "feetSecond": "Stopy na sekundę", "minutesPerKilometer": "Minuty na kilometr", "minutesPerMile": "Minuty na milę", "speedOfLight": "Prędkość światła", "grams": "Gramy", "ettograms": "Ettogramy", "kilograms": "Kilogramy", "pounds": "Funty", "ounces": "Uncje", "quintals": "Kwintale", "tonnes": "Tony", "milligrams": "Miligramy", "uma": "Zunifikowana jednostka masy atomowej", "carats": "Karaty", "centigrams": "Centygramy", "pennyweights": "Pennyweighty", "troyOunces": "Uncje trojańskie", "stones": "Kamienie", "pascal": "Paskal", "atmosphere": "Atmosfera", "bar": "Bar", "millibar": "Milibar", "psi": "Funt na cal kwadratowy", "torr": "Tor", "hectoPascal": "Hektopaskal", "inchesOfMercury": "Cal rtęci", "kiloPascal": "Kilopaskal", "ksi": "Kilofunt na cal kwadratowy", "megaPascal": "Megapaskal", "gigaPascal": "Gigapaskal", "joule": "Dżule", "calories": "Kalorie", "kilowattHour": "Kilowatogodzina", "electronvolt": "Elektronowolt", "footPound": "Stopa funta", "kilocalories": "Kilokalorie", "kilojoules": "Kilodżule", "watthour": "Watogodzina", "britishThermalUnit": "Brytyjska jednostka cieplna", "degree": "Stopnie", "minutesDegree": "Minuty", "secondsDegree": "Sekundy", "radiansDegree": "Radiany", "usd": "Amerykański dolar", "eur": "Euro", "gbp": "Brytyjski funt", "inr": "Indyjska rupia", "cny": "Chiński juan", "jpy": "Japoński jen", "chf": "Szwajcarski frank", "sek": "Szwedzka korona", "rub": "Rosyjski rubel", "cad": "Kanadyjski dolar", "krw": "Południowo-koreański won", "brl": "Brazylijski real", "hkd": "Hongkoński dolar", "aud": "Australijski dolar", "nzd": "Nowozelandzki dolar", "mxn": "Meksykańskie peso", "sgd": "Singapurski dolar", "nok": "Norweska korona", "trY": "Turecka lira", "zar": "Południowo-afrykański rand", "dkk": "Duńska korona", "pln": "Polski złoty", "thb": "Tajski baht", "myr": "Malezyjski ringgit", "huf": "Węgierski forint", "czk": "Czeska korona", "ils": "Izraelski nowy szekel", "idr": "Indonezyjska rupia", "php": "Filipińskie peso", "ron": "Rumuński lej", "isk": "Islandzka korona", "twd": "Tajwański nowy dolar", "mad": "Marokański dirham", "euChina": "UE i Chiny", "ukIndiaChild": "Wielka Brytania i Indie - Dziecko", "ukIndiaMan": "Wielka Brytania i Indie - Mężczyzna", "ukIndiaWoman": "Wielka Brytania i Indie - Kobieta", "usaCanadaChild": "USA i Kanada - Dziecko", "usaCanadaMan": "USA i Kanada - Mężczyzna", "usaCanadaWoman": "USA i Kanada - Kobieta", "japan": "Japonia", "watt": "Wat", "milliwatt": "Miliwat", "kilowatt": "Kilowat", "megawatt": "Megawat", "gigawatt": "Gigawat", "europeanHorsePower": "Europejskie konie mechaniczne", "imperialHorsePower": "Angielskie konie mechaniczne", "newton": "Niuton", "dyne": "Dyna", "poundForce": "Funt-siła", "kilogramForce": "Kilogram-siła", "poundal": "Poundal", "newtonMeter": "Niutonometr", "dyneMeter": "Dynametr", "poundForceFeet": "Funt na stopę", "kilogramForceMeter": "Kilogramometr", "poundalMeter": "Poundalometr", "poundForceInch": "Funt-siła na cal", "kilometersLiter": "Kilometry z litra", "liters100km": "Litry na 100 km", "milesUsGallon": "Kilometry z galona amerykańskiego", "milesImperialGallon": "Mile z galona angielskiego", "decimal": "Dziesiętny", "hexadecimal": "Szesnastkowy", "octal": "Ósemkowy", "binary": "Dwójkowy", "bit": "Bit", "nibble": "Półbajt", "kilobit": "Kilobit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "Megabit", "@megabit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "Gigabit", "@gigabit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "Terabit", "@terabit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "Petabit", "@petabit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "Eksabit", "@exabit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "Gibibit", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Exbibit", "byte": "Bajt", "kilobyte": "Kilobajt", "@kilobyte": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "Megabajt", "@megabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "Gigabajt", "@gigabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "Terabajt", "@terabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "Petabajt", "@petabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "Eksabajt", "@exabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "Kibibajt", "mebibyte": "Mebibajt", "gibibyte": "Gibibajt", "tebibyte": "Tebibajt", "pebibyte": "Pebibajt", "exbibyte": "Exbibajt", "base": "Podstawa", "deca": "Deko-", "hecto": "Hekto-", "kilo": "Kilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Eksa-", "zetta": "Zetta-", "yotta": "Jotta-", "deci": "Decy-", "centi": "Centy-", "milli": "Mili-", "micro": "Mikro-", "nano": "Nano-", "pico": "Piko-", "femto": "Femto-", "atto": "Atto-", "zepto": "Zepto-", "yocto": "Jokto-" } ================================================ FILE: packages/translations/lib/l10n/app_pt.arb ================================================ { "@@locale": "pt", "appName": "Converter NOW", "reorder": "Reorganizar", "myOrdering": "Minha classificação", "about": "Sobre", "lastCurrenciesUpdate": "Última atualização: ", "today": "hoje", "enableDarkTheme": "Usar tema escuro", "settings": "Configurações", "menu": "Menu", "donation": "Faça uma doação", "buyMeACoffee": "Buy me a coffee", "donationDialog": "Hi, as you might know this app is free and open source. It means that you can use it for free and copy/edit the source code. The developer will not gain nothing for it. But if you also believe in this project and would like to see new features, please consider buying me a coffee.", "drawerLogo": "Logotipo do menu", "rateApp": "Avaliar o App", "repoGithub": "Abrir repo no GitHub", "contributeTranslating": "Contribua traduzindo o App", "search": "Procurar", "clearAll": "Excluir tudo", "calculator": "Calculadora", "significantFigures": "Números significativos", "removeTrailingZeros": "Excluir os zeros seguidos", "longPressAdvice": "Pressione e segure para mover os itens", "back": "Voltar", "save": "Salvar", "invalidCharacters": "Erro, caracteres inválidos", "theme": "Tema", "dark": "Escuro", "light": "Claro", "system": "Sistema", "pureBlackTheme": "Tema escuro AMOLED", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "Cópia de", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "Mais", "language": "Língua", "reorderProperties": "Reordenar propriedades", "reorderUnits": "Reordenar unidades de medida", "chooseProperty": "Escolha uma propriedade", "reorderProperty": "Reorganizar {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW para outras plataformas", "sourceCode": "Código fonte", "undoClearAllMessage": "Cleared all. Do you want to undo it?", "@undoClearAllMessage": { "description": "Not yet translated. Once done, remove this comment" }, "undo": "Undo", "@undo": { "description": "Not yet translated. Once done, remove this comment" }, "routeError1": "This page doesn't exist", "@routeError1": { "description": "Not yet translated. Once done, remove this comment" }, "routeError2": "Go back to the first page", "@routeError2": { "description": "Not yet translated. Once done, remove this comment" }, "ok": "Ok", "@ok": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetAccess": "Revoke internet access", "@revokeInternetAccess": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetExplanation": "This app uses the internet to update currency exchange rates once a day upon opening. This download only requires a few kilobytes. However, if you don't need this feature, you can prevent the app from accessing the internet by enabling this option.", "@revokeInternetExplanation": { "description": "Not yet translated. Once done, remove this comment" }, "useDeviceColor": "Use device color", "@useDeviceColor": { "description": "Not yet translated. Once done, remove this comment" }, "pickColor": "Pick a color", "@pickColor": { "description": "Not yet translated. Once done, remove this comment" }, "themeColor": "Theme color", "@themeColor": { "description": "Not yet translated. Once done, remove this comment" }, "appearance": "Appearance", "@appearance": { "description": "Not yet translated. Once done, remove this comment" }, "conversions": "Conversions", "@conversions": { "description": "Not yet translated. Once done, remove this comment" }, "findOutMore": "Find out more", "@findOutMore": { "description": "Not yet translated. Once done, remove this comment" }, "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "Comprimento", "area": "Área", "volume": "Volume", "time": "Tempo", "temperature": "Temperatura", "speed": "Velocidade", "siPrefixes": "Prefixo SI", "mass": "Massa", "pressure": "Pressão", "energy": "Energia", "angles": "Ângulos", "currencies": "Moedas", "shoeSize": "Medida em Pés", "digitalData": "Dados digitais", "power": "Potência", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "Torque", "force": "Força", "fuelConsumption": "Consumo de combustível", "numeralSystems": "Sistemas numéricos", "meters": "Metros", "centimeters": "Centímetros", "inches": "Polegadas", "feet": "Pés", "nauticalMiles": "Milhas náuticas", "yards": "Jardas", "miles": "Milhas", "millimeters": "Milímetros", "micrometers": "Micrómetros", "nanometers": "Nanómetros", "angstroms": "Ångström", "picometers": "Picômetros", "kilometers": "Quilômetro", "astronomicalUnits": "Unidades Astronômicas", "lightYears": "Anos-luz", "parsec": "Parsec", "mils": "Mils", "feetUsSurvey": "Feet (US survey)", "@feetUsSurvey": { "description": "Not yet translated. Once done, remove this comment" }, "squareMeters": "Metros Quadrados", "squareCentimeters": "Centimetros Quadrados", "squareInches": "Polegadas Quadradas", "squareFeet": "Pés Quadrados", "squareMiles": "Milhas Quadradas", "squareYard": "Jardas Quadradas", "squareMillimeters": "Milímetros Quadrados", "squareKilometers": "Quilômetros Quadrados", "hectares": "Hectares", "acres": "Acres", "are": "Are", "squareFeetUsSurvey": "Square Feet (US survey)", "@squareFeetUsSurvey": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "Metros cúbicos", "liters": "Litros", "imperialGallons": "Galões Imperiais", "usGallons": "Galões Americanos", "imperialPints": "Pintas Imperiais", "usPints": "Pintas Americanas", "milliliters": "Mililitros", "tablespoonUs": "Colheres de mesa Americanas", "tablespoonAustralian": "Colheres de mesa Australianas", "cups": "Copos", "cubicCentimeters": "Centimetros Cúbicos", "cubicFeet": "Pés Cúbicos", "cubicInches": "Polegadas Cúbicas", "cubicMillimeters": "Milímetros Cúbicos", "imperialFluidOunces": "Onça líquida imperial", "usFluidOunces": "Onça líquida americana", "imperialGill": "Gill imperial", "usGill": "Gill estadunidense", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "Segundos", "deciseconds": "Decisecondos", "centiseconds": "Centisecondes", "milliseconds": "Milissegundos", "microseconds": "Microssegundos", "nanoseconds": "Nanossegundos", "minutes": "Minutos", "hours": "Horas", "days": "Dias", "weeks": "Semanas", "years": "Anos (365)", "lustrum": "Lustra", "decades": "Décadas", "centuries": "Séculos", "millennium": "Milénio", "fahrenheit": "Fahrenheit", "celsius": "Celsius (Centígrado)", "kelvin": "Kelvin", "reamur": "Reamur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Metros por Segundo", "kilometersHour": "Kilómetros por Hora", "milesHour": "Milhas por Hora", "knots": "Nós", "feetSecond": "Pés por Segundo", "minutesPerKilometer": "Minutos por quilômetro", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "Gramas", "ettograms": "Ettogramas", "kilograms": "Kilogramas", "pounds": "Libras", "ounces": "Onças", "quintals": "Quintals", "tonnes": "Toneladas", "milligrams": "Milligramas", "uma": "Unidade de massa atômica unificada", "carats": "Quilate", "centigrams": "Centigramas", "pennyweights": "Pennyweight", "troyOunces": "Onças troy", "stones": "Stone", "pascal": "Pascal", "atmosphere": "Atmosfera", "bar": "Bar", "millibar": "Millibar", "psi": "Libras por polegada quadrada", "torr": "Milímetro de mercúrio (Torr)", "hectoPascal": "Hectopascal", "@hectoPascal": { "description": "Not yet translated. Once done, remove this comment" }, "inchesOfMercury": "Inches of mercury", "@inchesOfMercury": { "description": "Not yet translated. Once done, remove this comment" }, "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "Joule", "calories": "Caloria", "kilowattHour": "Quilovate-Hora", "electronvolt": "Eletrão-Volt", "footPound": "Libra do pé", "kilocalories": "Kilocalories", "@kilocalories": { "description": "Not yet translated. Once done, remove this comment" }, "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "Grau", "minutesDegree": "Minutos", "secondsDegree": "Segundos", "radiansDegree": "Radianos", "usd": "Dólar americano", "eur": "Euro", "gbp": "Libra britânica", "inr": "Rupia indiana", "cny": "Yuan chinês", "jpy": "Iene japonês", "chf": "Franco suíço", "sek": "Coroa sueca", "rub": "Rublo russo", "cad": "Dólar canadense", "krw": "Won sul-coreano", "brl": "Real brasileiro", "hkd": "Dólar de Hong Kong", "aud": "Dólar australiano", "nzd": "Dólar neozelandês", "mxn": "Peso mexicano", "sgd": "Dólar singapuriano", "nok": "Coroa norueguesa", "trY": "Lira turca", "zar": "Rand sul-africano", "dkk": "Coroa dinamarquesa", "pln": "Zloti polonês", "thb": "Baht tailandês", "myr": "Ringgit malaio", "huf": "Forint húngaro", "czk": "Coroa tcheca", "ils": "Sheqel novo israelita", "idr": "Rupia indonésia", "php": "Peso filipino", "ron": "Leu romeno", "isk": "Coroa islandesa", "twd": "Novo dólar taiwanês", "mad": "Dirrã marroquino", "euChina": "EU & China", "ukIndiaChild": "UK & Índia - Criança", "ukIndiaMan": "UK & Índia - Homem", "ukIndiaWoman": "UK & Índia - Mulher", "usaCanadaChild": "EUA & Canada - Criança", "usaCanadaMan": "EUA & Canada - Homem", "usaCanadaWoman": "EUA & Canada - Mulher", "japan": "Japão", "watt": "Watt", "milliwatt": "Milliwatt", "kilowatt": "Kilowatt", "megawatt": "Megawatt", "gigawatt": "GigaWatt", "europeanHorsePower": "Cavalo-vapor europeu", "imperialHorsePower": "Cavalo-vapor Imperial", "newton": "Newton", "dyne": "Dyne", "poundForce": "Libra-força", "kilogramForce": "Quilograma-força", "poundal": "Poundal", "newtonMeter": "Newton metros", "dyneMeter": "Dyne metros", "poundForceFeet": "Libra-força metros", "kilogramForceMeter": "Quilograma-força metros", "poundalMeter": "Poundal metros", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "Kilómetros por litro", "liters100km": "Litros por 100 km", "milesUsGallon": "Milhas por Galão Americano", "milesImperialGallon": "Milhas por Galão Imperial", "decimal": "Decimal", "hexadecimal": "Hexadecimal", "octal": "Octal", "binary": "Binário", "bit": "Bit", "nibble": "Nibble", "kilobit": "Kilobit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "Megabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "Gigabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "Terabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "Petabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "Exabit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "Gibibit", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Exbibit", "byte": "Byte", "kilobyte": "Kilobyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "Megabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "Gigabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "Terabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "Petabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "Exabyte", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "Kibibyte", "mebibyte": "Mebibyte", "gibibyte": "Gibibyte", "tebibyte": "Tebibyte", "pebibyte": "Pebibyte", "exbibyte": "Exbibyte", "base": "Base", "deca": "Deca-", "hecto": "Hecto-", "kilo": "Kilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Exa-", "zetta": "Zetta-", "yotta": "Yotta-", "deci": "Deci-", "centi": "Centi-", "milli": "Milli-", "micro": "Micro-", "nano": "Nano-", "pico": "Pico-", "femto": "Femto-", "atto": "Atto-", "zepto": "Zepto-", "yocto": "Yocto-" } ================================================ FILE: packages/translations/lib/l10n/app_ru.arb ================================================ { "@@locale": "ru", "appName": "Converter NOW", "reorder": "Изменить порядок", "myOrdering": "Мой порядок", "about": "О приложении", "lastCurrenciesUpdate": "Последнее обновление: ", "today": "Сегодня", "enableDarkTheme": "Включить тёмную тему", "settings": "Настройки", "menu": "Меню", "donation": "Пожертвовать", "buyMeACoffee": "Купить мне кофе", "donationDialog": "Привет, как вы, возможно, знаете, это бесплатное приложение с открытым исходным кодом. Это означает, что вы можете использовать его бесплатно и копировать/редактировать исходный код. Разработчик ничего за это не получит. Но если вы также верите в этот проект и хотели бы увидеть новые функции, пожалуйста, подумайте о покупке мне кофе.", "drawerLogo": "Логотип в меню", "rateApp": "Оценить приложение", "repoGithub": "Открыть репозиторий GitHub", "contributeTranslating": "Помочь в переводе приложения", "search": "Поиск", "clearAll": "Очистить всё", "calculator": "Калькулятор", "significantFigures": "Значимые цифры", "removeTrailingZeros": "Удалять конечные нули", "longPressAdvice": "Длительное нажатие для перемещения элементов", "back": "Назад", "save": "Сохранить", "invalidCharacters": "Ошибка: недопустимые символы", "theme": "Тема", "dark": "Тёмная", "light": "Светлая", "system": "Как в системе", "pureBlackTheme": "Чёрная тема", "propertySelectionOnStartup": "Выбор категории при запуске", "propertySelectionOnStartupSubtitle": "Открывать страницу выбора категории при запуске на устройствах с маленьким экраном", "copy": "Копировать", "paste": "Вставить", "more": "Ещё", "language": "Язык", "reorderProperties": "Изменить порядок категорий", "reorderUnits": "Изменить порядок единиц измерения", "chooseProperty": "Выберите категорию", "reorderProperty": "Изменить порядок в «{property}»", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Converter NOW для других платформ", "sourceCode": "Исходный код", "undoClearAllMessage": "Всё очищено. Отменить это действие?", "undo": "Отмена", "routeError1": "Этой страницы не существует", "routeError2": "Вернуться на первую страницу", "ok": "OK", "revokeInternetAccess": "Запретить доступ в интернет", "revokeInternetExplanation": "Это приложение использует интернет для обновления курсов валют один раз в день при открытии. Для загрузки требуется всего несколько килобайт. Однако, если вам не нужна эта функция, вы можете запретить приложению доступ к интернету, включив данную настройку.", "useDeviceColor": "Использовать цвет устройства", "pickColor": "Выбрать цвет", "themeColor": "Цвет темы", "appearance": "Внешний вид", "conversions": "Преобразования", "findOutMore": "Узнать больше", "hideUnits": "Скрыть единицы измерения", "visibleUnits": "{property}, отображаемые единицы", "@visibleUnits": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Скрытые единицы измерения", "selectAll": "Выбрать все", "unselectAll": "Снять выбор", "backupAndRestore": "Резервное копирование", "exportSettings": "Экспорт настроек", "importSettings": "Импорт настроек", "clearSettings": "Очистить все настройки", "exportSuccess": "Настройки успешно экспортированы", "importSuccess": "Настройки успешно импортированы", "problemImportFile": "При импорте файла возникла проблема", "relatedSettings": "Связанные настройки", "reason": "Причина", "importError": "Ошибка импорта", "confirmationClear": "Очистить все настройки? Это действие невозможно отменить.", "cancel": "Отмена", "length": "Длина", "area": "Площадь", "volume": "Объём", "time": "Время", "temperature": "Температура", "speed": "Скорость", "siPrefixes": "Префикс СИ", "mass": "Масса", "pressure": "Давление", "energy": "Энергия", "angles": "Углы", "currencies": "Валюта", "shoeSize": "Размер обуви", "digitalData": "Цифровые данные", "power": "Мощность", "density": "Плотность", "torque": "Крутящий момент", "force": "Сила", "fuelConsumption": "Расход топлива", "numeralSystems": "Системы счисления", "meters": "Метры", "centimeters": "Сантиметры", "inches": "Дюймы", "feet": "Футы", "nauticalMiles": "Морские мили", "yards": "Ярды", "miles": "Мили", "millimeters": "Миллиметры", "micrometers": "Микрометры", "nanometers": "Нанометры", "angstroms": "Ангстремы", "picometers": "Пикометры", "kilometers": "Километры", "astronomicalUnits": "Астрономические единицы", "lightYears": "Световые года", "parsec": "Парсеки", "mils": "Тысячные доли дюйма", "feetUsSurvey": "Футы США", "squareMeters": "Квадратные метры", "squareCentimeters": "Квадратные сантиметры", "squareInches": "Квадратные дюймы", "squareFeet": "Квадратные футы", "squareMiles": "Квадратные мили", "squareYard": "Квадратные ярды", "squareMillimeters": "Квадратные миллиметры", "squareKilometers": "Квадратные километры", "hectares": "Гектары", "acres": "Акры", "are": "Ары", "squareFeetUsSurvey": "Квадратные футы США", "gramsPerCubicCentimeter": "Граммы на кубический сантиметр", "gramsPerLiter": "Граммы на литр", "gramsPerMilliliter": "Граммы на миллилитр", "gramsPerDeciliter": "Граммы на децилитр", "kilogramsPerLiter": "Килограммы на литр", "kilogramsPerCubicMeter": "Килограммы на кубический метр", "milligramsPerLiter": "Миллиграммы на литр", "milligramsPerDeciliter": "Миллиграммы на децилитр", "milligramsPerMilliliter": "Миллиграммы на миллилитр", "milligramsPerCubicMeter": "Миллиграммы на кубический метр", "milligramsPerCubicCentimeter": "Миллиграммы на кубический сантиметр", "microgramsPerLiter": "Микрограммы на литр", "microgramsPerDeciliter": "Микрограммы на децилитр", "microgramsPerMilliliter": "Микрограммы на миллилитр", "poundsPerCubicInches": "Фунты на кубический дюйм", "poundsPerCubicFeet": "Фунты на кубический фут", "cubicMeters": "Кубические метры", "liters": "Литры", "imperialGallons": "Британские галлоны", "usGallons": "Галлоны США", "imperialPints": "Британские пинты", "usPints": "Пинты США", "milliliters": "Миллилитры", "tablespoonUs": "Столовые ложки США", "tablespoonAustralian": "Австралийские столовые ложки", "cups": "Стаканы", "cubicCentimeters": "Кубические сантиметры", "cubicFeet": "Кубические футы", "cubicInches": "Кубические дюймы", "cubicMillimeters": "Кубические миллилитры", "imperialFluidOunces": "Британские жидкие унции", "usFluidOunces": "Жидкие унции США", "imperialGill": "Британские гиллы", "usGill": "Гиллы США", "usQuarts": "Кварты США", "microliters": "Микролитры", "deciliters": "Децилитры", "centiliters": "Сантилитры", "teaspoonsUs": "Чайные ложки США", "teaspoonsMetric": "Метрические чайные ложки", "seconds": "Секунды", "deciseconds": "Десятые доли секунды", "centiseconds": "Сотые доли секунды", "milliseconds": "Миллисекунды", "microseconds": "Микросекунды", "nanoseconds": "Наносекунды", "minutes": "Минуты", "hours": "Часы", "days": "Дни", "weeks": "Недели", "years": "Годы (365)", "lustrum": "Пятилетия", "decades": "Десятилетия", "centuries": "Века", "millennium": "Тысячелетия", "fahrenheit": "Фаренгейты", "celsius": "Градусы Цельсия (стоградусные)", "kelvin": "Кельвины", "reamur": "Градусы Реомюра", "romer": "Градусы Рёмера", "delisle": "Градусы Делиля", "rankine": "Градусы Ранкина", "metersSecond": "Метры в секунду", "kilometersHour": "Километры в час", "milesHour": "Мили в час", "knots": "Узлы", "feetSecond": "Футы в секунду", "minutesPerKilometer": "Минуты на километр", "minutesPerMile": "Минуты на милю", "speedOfLight": "Скорость света", "grams": "Граммы", "ettograms": "Этограммы", "kilograms": "Килограммы", "pounds": "Фунты", "ounces": "Унции", "quintals": "Центнеры", "tonnes": "Тонны", "milligrams": "Миллиграммы", "uma": "Единая атомная единица массы", "carats": "Караты", "centigrams": "Сантиграммы", "pennyweights": "Пеннивейты", "troyOunces": "Тройские унции", "stones": "Стоуны", "pascal": "Паскали", "atmosphere": "Атмосферы", "bar": "Бары", "millibar": "Миллибары", "psi": "Фунты на квадратный дюйм", "torr": "Торричелли", "hectoPascal": "Гектопаскали", "inchesOfMercury": "Дюймы ртутного столба", "kiloPascal": "Килопаскали", "ksi": "Килофунты на квадратный дюйм", "megaPascal": "Мегапаскали", "gigaPascal": "Гигапаскали", "joule": "Джоули", "calories": "Калории", "kilowattHour": "Киловатт-часы", "electronvolt": "Электрон-вольты", "footPound": "Фут-фунты", "kilocalories": "Килокалории", "kilojoules": "Килоджоули", "watthour": "Ватт-часы", "britishThermalUnit": "Британские тепловые единицы", "degree": "Градусы", "minutesDegree": "Минуты", "secondsDegree": "Секунды", "radiansDegree": "Радианы", "usd": "Доллар США", "eur": "Евро", "gbp": "Британский фунт стерлингов", "inr": "Индийская рупия", "cny": "Китайский юань", "jpy": "Японская иена", "chf": "Швейцарский франк", "sek": "Шведская крона", "rub": "Российский рубль", "cad": "Канадский доллар", "krw": "Южнокорейская вона", "brl": "Бразильский реал", "hkd": "Гонконгский доллар", "aud": "Австралийский доллар", "nzd": "Новозеландский доллар", "mxn": "Мексиканское песо", "sgd": "Сингапурский доллар", "nok": "Норвежская крона", "trY": "Турецкая лира", "zar": "Южноафриканский рэнд", "dkk": "Датская крона", "pln": "Польский злотый", "thb": "Таиландский бат", "myr": "Малайзийский ринггит", "huf": "Венгерский форинт", "czk": "Чешская крона", "ils": "Новый израильский шекель", "idr": "Индонезийская рупия", "php": "Филиппинское песо", "ron": "Румынский лей", "isk": "Исландская крона", "twd": "Новый тайваньский доллар", "mad": "Марокканский дирхам", "euChina": "ЕС и Китай", "ukIndiaChild": "Великобритания и Индия - ребёнок", "ukIndiaMan": "Великобритания и Индия - мужчина", "ukIndiaWoman": "Великобритания и Индия - женщина", "usaCanadaChild": "США и Канада - ребёнок", "usaCanadaMan": "США и Канада - мужчина", "usaCanadaWoman": "США и Канада - женщина", "japan": "Япония", "watt": "Ватты", "milliwatt": "Милливатты", "kilowatt": "Киловатты", "megawatt": "Мегаватты", "gigawatt": "Гигаватты", "europeanHorsePower": "Европейские лошадиные силы", "imperialHorsePower": "Британские лошадиные силы", "newton": "Ньютоны", "dyne": "Дины", "poundForce": "Фунт-силы", "kilogramForce": "Килограмм-силы", "poundal": "Фунталы", "newtonMeter": "Ньютон-метры", "dyneMeter": "Динамометры", "poundForceFeet": "Фунт-сила-футы", "kilogramForceMeter": "Килограмм-сила-метры", "poundalMeter": "Фунтал-метры", "poundForceInch": "Фунт-сила-дюймы", "kilometersLiter": "Километры на литр", "liters100km": "Литры на 100 км", "milesUsGallon": "Мили на галлон США", "milesImperialGallon": "Мили на британский галлон", "decimal": "Десятичная", "hexadecimal": "Шестнадцатеричная", "octal": "Восьмеричная", "binary": "Двоичная", "bit": "Биты", "nibble": "Полубайты", "kilobit": "Килобиты (десятичные)", "megabit": "Мегабиты (десятичные)", "gigabit": "Гигабиты (десятичные)", "terabit": "Терабиты (десятичные)", "petabit": "Петабиты (десятичные)", "exabit": "Эксабиты (десятичные)", "kibibit": "Кибибиты", "mebibit": "Мебибиты", "gibibit": "Гибибиты", "tebibit": "Тебибиты", "pebibit": "Пебибиты", "exbibit": "Эксабибиты", "byte": "Байты", "kilobyte": "Килобайты (десятичные)", "megabyte": "Мегабайты (десятичные)", "gigabyte": "Гигабайты (десятичные)", "terabyte": "Терабайты (десятичные)", "petabyte": "Петабайты (десятичные)", "exabyte": "Эксабайты (десятичные)", "kibibyte": "Кибибайты", "mebibyte": "Мебибайты", "gibibyte": "Гибибайты", "tebibyte": "Тебибайты", "pebibyte": "Пебибайты", "exbibyte": "Эксабибайты", "base": "Основа", "deca": "Дека-", "hecto": "Гекто-", "kilo": "Кило-", "mega": "Мега-", "giga": "Гига-", "tera": "Тера-", "peta": "Пета-", "exa": "Экса-", "zetta": "Зетта-", "yotta": "Йотта-", "deci": "Деци-", "centi": "Санти-", "milli": "Милли-", "micro": "Микро-", "nano": "Нано-", "pico": "Пико-", "femto": "Фемто-", "atto": "Атто-", "zepto": "Зепто-", "yocto": "Йокто-" } ================================================ FILE: packages/translations/lib/l10n/app_tr.arb ================================================ { "@@locale": "tr", "appName": "Converter NOW", "reorder": "Sırala", "myOrdering": "Düzenim", "about": "Hakkında", "lastCurrenciesUpdate": "Son güncelleme: ", "today": "bugün", "enableDarkTheme": "Koyu temayı etkinleştir", "settings": "Ayarlar", "menu": "Menü", "donation": "Bağış yap", "buyMeACoffee": "Bana bir kahve ısmarla", "donationDialog": "Merhaba, farkettiğin üzere bu uygulama özgür bir yazılım, ayrıca uygulamayı ücretsiz olarak dağıtıyorum. Bu, kaynak kodunu kopyalayabileceğin/düzenleyebileceğin anlamına gelir. Ben (geliştirici) bundan hiç gelir elde etmiyorum. Projeyi beğeniyor ve yeni özellikler görmek istiyorsan, bana bir kahve ısmarlamayı düşününebilirsin.", "drawerLogo": "Çekmece logosu", "rateApp": "Uygulamayı oyla", "repoGithub": "Github deposunu aç", "contributeTranslating": "Uygulamanın çevrilmesine katkıda bulun", "search": "Ara", "clearAll": "Tümünü temizle", "calculator": "Hesap makinesi", "significantFigures": "Anlamlı rakamlar", "removeTrailingZeros": "Sondaki sıfırları kaldır", "longPressAdvice": "Öğeleri taşımak için basılı tut", "back": "Geri", "save": "Kaydet", "invalidCharacters": "Hata, geçersiz karakter", "theme": "Tema", "dark": "Koyu", "light": "Açık", "system": "Sistem varsayılanı", "pureBlackTheme": "AMOLED koyu teması", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "Kopyala", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "Daha fazlası", "language": "Dil", "reorderProperties": "Özellikleri yeniden sırala", "reorderUnits": "Ölçü birimlerini yeniden sıralama", "chooseProperty": "Bir özellik seçin", "reorderProperty": "Sırala {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "Diğer platformlar için Converter NOW", "sourceCode": "Kaynak kodu", "undoClearAllMessage": "Hepsi temizlendi. Geri almak istiyor musunuz?", "undo": "Geri al", "routeError1": "Bu sayfa mevcut değil", "routeError2": "İlk sayfaya geri dön", "ok": "Ok", "@ok": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetAccess": "Revoke internet access", "@revokeInternetAccess": { "description": "Not yet translated. Once done, remove this comment" }, "revokeInternetExplanation": "This app uses the internet to update currency exchange rates once a day upon opening. This download only requires a few kilobytes. However, if you don't need this feature, you can prevent the app from accessing the internet by enabling this option.", "@revokeInternetExplanation": { "description": "Not yet translated. Once done, remove this comment" }, "useDeviceColor": "Use device color", "@useDeviceColor": { "description": "Not yet translated. Once done, remove this comment" }, "pickColor": "Pick a color", "@pickColor": { "description": "Not yet translated. Once done, remove this comment" }, "themeColor": "Theme color", "@themeColor": { "description": "Not yet translated. Once done, remove this comment" }, "appearance": "Appearance", "@appearance": { "description": "Not yet translated. Once done, remove this comment" }, "conversions": "Conversions", "@conversions": { "description": "Not yet translated. Once done, remove this comment" }, "findOutMore": "Find out more", "@findOutMore": { "description": "Not yet translated. Once done, remove this comment" }, "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "description": "Not yet translated. Once done, remove this comment", "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "Uzunluk", "area": "Alan", "volume": "Hacim", "time": "Zaman", "temperature": "Sıcaklık", "speed": "Hız", "siPrefixes": "SI Ön Ekleri", "mass": "Kütle", "pressure": "Basınç", "energy": "Enerji", "angles": "Açı", "currencies": "Para birimi", "shoeSize": "Ayakkabı", "digitalData": "Dijital veri", "power": "Güç", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "Tork", "force": "Kuvvet", "fuelConsumption": "Yakıt tüketimi", "numeralSystems": "Sayı sistemi", "meters": "Metre", "centimeters": "Santimetre", "inches": "İnç", "feet": "Fit", "nauticalMiles": "Deniz mili", "yards": "Yarda", "miles": "Mil", "millimeters": "Milimetre", "micrometers": "Mikrometre", "nanometers": "Nanometre", "angstroms": "Ångström", "picometers": "Pikometre", "kilometers": "Kilometre", "astronomicalUnits": "Astronomik birimler", "lightYears": "Işık yılı", "parsec": "Parsek", "mils": "Bir inçin binde biri", "feetUsSurvey": "Feet (US survey)", "@feetUsSurvey": { "description": "Not yet translated. Once done, remove this comment" }, "squareMeters": "Metrekare", "squareCentimeters": "Santimetrekare", "squareInches": "İnçkare", "squareFeet": "Fitkare", "squareMiles": "Milkare", "squareYard": "Yardakare", "squareMillimeters": "Milimetrekare", "squareKilometers": "Kilometrekare", "hectares": "Hektar", "acres": "Dönüm", "are": "Ar", "squareFeetUsSurvey": "Square Feet (US survey)", "@squareFeetUsSurvey": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "Metreküp", "liters": "Litre", "imperialGallons": "Galon (İngiliz)", "usGallons": "Galon (Amerikan)", "imperialPints": "Pint (İngiliz)", "usPints": "Pint (Amerikan)", "milliliters": "Mililitre", "tablespoonUs": "Yemek kaşığı (Amerikan)", "tablespoonAustralian": "Yemek kaşığı (Avustralya)", "cups": "Fincan", "cubicCentimeters": "Santimetreküp", "cubicFeet": "Fitküp", "cubicInches": "İnçküp", "cubicMillimeters": "Milimetreküp", "imperialFluidOunces": "Sıvı ons (İngiliz)", "usFluidOunces": "Sıvı ons (Amerikan)", "imperialGill": "Çay fincanı (İngiliz)", "usGill": "Çay fincanı (Amerikan)", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "Saniye", "deciseconds": "Desisaniye", "centiseconds": "Santisaniye", "milliseconds": "Milisaniye", "microseconds": "Mikrosaniye", "nanoseconds": "Nanosaniye", "minutes": "Dakika", "hours": "Saat", "days": "Gün", "weeks": "Hafta", "years": "Yıl (365)", "lustrum": "Lustrum", "decades": "Onyıl", "centuries": "Yüzyıl", "millennium": "Binyıl", "fahrenheit": "Fahrenhayt", "celsius": "Santigrat", "kelvin": "Kelvin", "reamur": "Reamur", "romer": "Rømer", "delisle": "Delisle", "rankine": "Rankine", "metersSecond": "Saniyede metre", "kilometersHour": "Saatte kilometre", "milesHour": "Saatte mil", "knots": "Knot", "feetSecond": "Saniyede fit", "minutesPerKilometer": "Kilometre başına dakika", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "Gram", "ettograms": "Ettogram", "kilograms": "Kilogram", "pounds": "Pound", "ounces": "Ons", "quintals": "Kental", "tonnes": "Ton", "milligrams": "Miligram", "uma": "Dalton", "carats": "Karat", "centigrams": "Santigram", "pennyweights": "Pennyweight", "troyOunces": "Troy ons", "stones": "Taş", "pascal": "Paskal", "atmosphere": "Atmosfer", "bar": "Bar", "millibar": "Milibar", "psi": "İnçkare başına libre", "torr": "Milimetre-cıva", "hectoPascal": "Hektopaskal", "inchesOfMercury": "İnç-cıva", "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "Joule", "calories": "Kalori", "kilowattHour": "Kilovat saat", "electronvolt": "Elektronvolt", "footPound": "Ayak pound", "kilocalories": "Kilocalories", "@kilocalories": { "description": "Not yet translated. Once done, remove this comment" }, "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "Derece", "minutesDegree": "Dakika", "secondsDegree": "Saniye", "radiansDegree": "Radyan", "usd": "Amerikan doları", "eur": "Euro", "gbp": "İngiliz sterlini", "inr": "Hint rupisi", "cny": "Çin yuanı", "jpy": "Japon yeni", "chf": "İsviçre frangı", "sek": "İsveç kronu", "rub": "Rus rublesi", "cad": "Kanada doları", "krw": "Güney Kore wonu", "brl": "Brezilya reali", "hkd": "Hong Kong doları", "aud": "Avustralya doları", "nzd": "Yeni Zelanda doları", "mxn": "Meksika pesosu", "sgd": "Singapur doları", "nok": "Norveç kronu", "trY": "Türk lirası", "zar": "Güney Afrika randı", "dkk": "Danimarka kronu", "pln": "Polonya zlotisi", "thb": "Tayland bahtı", "myr": "Malezya ringgiti", "huf": "Macar forinti", "czk": "Çek korunası", "ils": "İsrail şekeli", "idr": "Endonezya rupisi", "php": "Filipin pezosu", "ron": "Rumen leyi", "isk": "İzlanda kronu", "twd": "Yeni Tayvan doları", "mad": "Fas dirhemi", "euChina": "AB ve Çin", "ukIndiaChild": "İngiltere ve Hindistan - Çocuk", "ukIndiaMan": "İngiltere ve Hindistan - Erkek", "ukIndiaWoman": "İngiltere ve Hindistan - Kadın", "usaCanadaChild": "ABD ve Kanada - Çocuk", "usaCanadaMan": "ABD ve Kanada - Erkek", "usaCanadaWoman": "ABD ve Kanada - Kadın", "japan": "Japonya", "watt": "Watt", "milliwatt": "Miliwatt", "kilowatt": "Kilowatt", "megawatt": "Megawatt", "gigawatt": "Gigawatt", "europeanHorsePower": "Avrupa beygir gücü", "imperialHorsePower": "İngiliz beygir gücü", "newton": "Newton", "dyne": "Dyn", "poundForce": "Pound-kuvvet", "kilogramForce": "Kilogram-kuvvet", "poundal": "Poundal", "newtonMeter": "Newton metre", "dyneMeter": "Dyn metre", "poundForceFeet": "Pound-kuvvet fit", "kilogramForceMeter": "Kilogram-kuvvet metre", "poundalMeter": "Poundal metre", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "Kilometre bölü litre", "liters100km": "100 km bölü litre", "milesUsGallon": "ABD galonu bölü mil", "milesImperialGallon": "İngiliz galonu bölü mil", "decimal": "Onluk", "hexadecimal": "Onaltılık", "octal": "Sekizlik", "binary": "İkilik", "bit": "Bit", "nibble": "Nibble", "kilobit": "Kilobit", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "Megabit", "@megabit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "Gigabit", "@gigabit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "Terabit", "@terabit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "Petabit", "@petabit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "Eksabit", "@exabit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "Kibibit", "mebibit": "Mebibit", "gibibit": "GibiBit.", "tebibit": "Tebibit", "pebibit": "Pebibit", "exbibit": "Eksbibit", "byte": "Bayt", "kilobyte": "Kilobayt", "@kilobyte": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "MegaBayt", "@megabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "GigaBayt", "@gigabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "TeraBayt", "@terabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "Petabayt", "@petabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "Eksabayt", "@exabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "Kibibayt", "mebibyte": "Mebibayt", "gibibyte": "Gibibayt", "tebibyte": "Tebibayt", "pebibyte": "Pebibayt", "exbibyte": "Eksbibayt", "base": "Temel birim", "deca": "Deka-", "hecto": "Hekto-", "kilo": "Kilo-", "mega": "Mega-", "giga": "Giga-", "tera": "Tera-", "peta": "Peta-", "exa": "Eksa-", "zetta": "Zetta-", "yotta": "Yotta-", "deci": "Desi-", "centi": "Santi-", "milli": "Mili-", "micro": "Mikro-", "nano": "Nano-", "pico": "Piko-", "femto": "Femto-", "atto": "Atto-", "zepto": "Zepto-", "yocto": "Yokto-" } ================================================ FILE: packages/translations/lib/l10n/app_zh.arb ================================================ { "@@locale": "zh", "appName": "Converter NOW", "reorder": "排序", "myOrdering": "我的排序", "about": "关于此程序", "lastCurrenciesUpdate": "最终更新:", "today": "今天", "enableDarkTheme": "打开夜间模式", "settings": "设置", "menu": "菜单", "donation": "捐赠", "buyMeACoffee": "给开发者买一杯咖啡", "donationDialog": "你好!有可能你知道,这个程序开源且免费。这意味着你可以免费使用此软件,复制和编辑源代码。而开发者得不到任何利益。如果你赞同此项目,或者想看到一些新功能,请考虑给我买一杯咖啡。", "drawerLogo": "Drawer logo", "rateApp": "评价此程序", "repoGithub": "打开GitHub仓库", "contributeTranslating": "参加此程序的翻译", "search": "搜索", "clearAll": "全部删除", "calculator": "计算器", "significantFigures": "有效数字", "removeTrailingZeros": "删除末尾的0", "longPressAdvice": "长按以移动", "back": "返回", "save": "保存", "invalidCharacters": "错误,无效的文字", "theme": "主题", "dark": "深色", "light": "浅色", "system": "系统设置", "pureBlackTheme": "AMOLED黑暗模式", "@pureBlackTheme": { "description": "Revise this translation" }, "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "复制", "paste": "Paste", "@paste": { "description": "Not yet translated. Once done, remove this comment" }, "more": "更多", "language": "语言", "reorderProperties": "排序组", "reorderUnits": "排序项目", "chooseProperty": "请选择组", "reorderProperty": "排序{property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "在其他平台上使用此程序", "sourceCode": "源代码", "undoClearAllMessage": "已全部清除。 你想撤消它吗?", "undo": "撤消", "routeError1": "不存在此页面", "routeError2": "返回上一级", "ok": "OK", "revokeInternetAccess": "禁用网络连接", "revokeInternetExplanation": "该应用在每次启动时通过网络更新汇率。所需下载的数据只有几 kiB。不过,您若不需此功能,启用该选项可阻止该应用连接互联网。", "useDeviceColor": "使用系统设置的强调色", "pickColor": "选择颜色", "themeColor": "主题色", "appearance": "外观", "conversions": "转换", "findOutMore": "更多", "hideUnits": "Hide units", "@hideUnits": { "description": "Not yet translated. Once done, remove this comment" }, "visibleUnits": "{property}, visible units", "@visibleUnits": { "description": "Not yet translated. Once done, remove this comment", "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "Hidden units", "@hiddenUnits": { "description": "Not yet translated. Once done, remove this comment" }, "selectAll": "Select all", "@selectAll": { "description": "Not yet translated. Once done, remove this comment" }, "unselectAll": "Unselect all", "@unselectAll": { "description": "Not yet translated. Once done, remove this comment" }, "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "长度", "area": "面积", "volume": "体积", "time": "时间", "temperature": "温度", "speed": "速度", "siPrefixes": "国际单位前缀", "mass": "质量", "pressure": "压强", "energy": "能量", "angles": "角度", "currencies": "货币", "shoeSize": "鞋子尺寸", "digitalData": "电子数据", "power": "功率", "density": "Density", "@density": { "description": "Not yet translated. Once done, remove this comment" }, "torque": "力矩", "force": "力", "fuelConsumption": "燃费", "numeralSystems": "数制", "meters": "米", "centimeters": "厘米", "inches": "英寸", "feet": "英尺", "nauticalMiles": "海里", "yards": "码数", "miles": "英里", "millimeters": "毫米", "micrometers": "微米", "nanometers": "纳米", "angstroms": "埃米", "picometers": "皮米", "kilometers": "公里", "astronomicalUnits": "天文单位", "lightYears": "光年", "parsec": "秒差距", "mils": "Mils", "feetUsSurvey": "英尺(美国)", "squareMeters": "平方米", "squareCentimeters": "平方厘米", "squareInches": "平方英寸", "squareFeet": "平方英尺", "squareMiles": "平方英里", "squareYard": "平方码", "squareMillimeters": "平方毫米", "squareKilometers": "平方公里", "hectares": "公顷", "acres": "英亩", "are": "公亩", "squareFeetUsSurvey": "平方英尺(美国)", "gramsPerCubicCentimeter": "Grams per Cubic Centimeter", "@gramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerLiter": "Grams per Liter", "@gramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerMilliliter": "Grams per Milliliter", "@gramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "gramsPerDeciliter": "Grams per Deciliter", "@gramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerLiter": "Kilograms per Liter", "@kilogramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "kilogramsPerCubicMeter": "Kilograms per Cubic Meter", "@kilogramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerLiter": "Milligrams per Liter", "@milligramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerDeciliter": "Milligrams per Deciliter", "@milligramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerMilliliter": "Milligrams per Milliliter", "@milligramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicMeter": "Milligrams per Cubic Meter", "@milligramsPerCubicMeter": { "description": "Not yet translated. Once done, remove this comment" }, "milligramsPerCubicCentimeter": "Milligrams per Cubic Centimeter", "@milligramsPerCubicCentimeter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerLiter": "Micrograms per Liter", "@microgramsPerLiter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerDeciliter": "Micrograms per Deciliter", "@microgramsPerDeciliter": { "description": "Not yet translated. Once done, remove this comment" }, "microgramsPerMilliliter": "Micrograms per Milliliter", "@microgramsPerMilliliter": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicInches": "Pounds per Cubic Inches", "@poundsPerCubicInches": { "description": "Not yet translated. Once done, remove this comment" }, "poundsPerCubicFeet": "Pounds per Cubic Feet", "@poundsPerCubicFeet": { "description": "Not yet translated. Once done, remove this comment" }, "cubicMeters": "立方米", "liters": "公升", "imperialGallons": "英制加仑", "usGallons": "美制加仑", "imperialPints": "英制品脱", "usPints": "美制品脱", "milliliters": "毫升", "tablespoonUs": "一汤匙(美国)", "tablespoonAustralian": "一汤匙(澳洲)", "cups": "杯", "cubicCentimeters": "立方厘米", "cubicFeet": "立方英尺", "cubicInches": "立方英寸", "cubicMillimeters": "立方毫米", "imperialFluidOunces": "英液盎司", "usFluidOunces": "美液盎司", "imperialGill": "英制及耳", "usGill": "美制及耳", "usQuarts": "US Quarts", "@usQuarts": { "description": "Not yet translated. Once done, remove this comment" }, "microliters": "Microliters", "@microliters": { "description": "Not yet translated. Once done, remove this comment" }, "deciliters": "Deciliters", "@deciliters": { "description": "Not yet translated. Once done, remove this comment" }, "centiliters": "Centiliters", "@centiliters": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "秒", "deciseconds": "分秒", "centiseconds": "厘秒", "milliseconds": "毫秒", "microseconds": "微秒", "nanoseconds": "纳秒", "minutes": "分", "hours": "小时", "days": "日", "weeks": "周", "years": "年 (365)", "lustrum": "五年纪", "decades": "十年纪", "centuries": "世纪", "millennium": "千年纪", "fahrenheit": "华氏度", "celsius": "摄氏度", "kelvin": "开尔文", "reamur": "列氏度", "romer": "罗氏温标", "delisle": "德利尔温标", "rankine": "兰金温标", "metersSecond": "米/秒", "kilometersHour": "公里/小时", "milesHour": "英里/小时", "knots": "节(海里/小时)", "feetSecond": "英尺/秒", "minutesPerKilometer": "公里/分", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "克", "ettograms": "百克", "kilograms": "千克", "pounds": "磅", "ounces": "盎司", "quintals": "公担", "tonnes": "吨", "milligrams": "毫克", "uma": "原子质量单位", "carats": "克拉", "centigrams": "厘克", "pennyweights": "英钱", "troyOunces": "金衡盎司", "stones": "英石", "pascal": "帕斯卡", "atmosphere": "标准大气压", "bar": "巴", "millibar": "毫巴", "psi": "磅/平方英寸", "torr": "托", "hectoPascal": "百帕", "inchesOfMercury": "英寸汞柱", "kiloPascal": "Kilopascal", "@kiloPascal": { "description": "Not yet translated. Once done, remove this comment" }, "ksi": "Kilopounds per square inch", "@ksi": { "description": "Not yet translated. Once done, remove this comment" }, "megaPascal": "Megapascal", "@megaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "gigaPascal": "Gigapascal", "@gigaPascal": { "description": "Not yet translated. Once done, remove this comment" }, "joule": "焦耳", "calories": "卡", "kilowattHour": "千瓦时", "electronvolt": "电子伏特", "footPound": "英尺磅", "kilocalories": "大卡", "kilojoules": "Kilojoules", "@kilojoules": { "description": "Not yet translated. Once done, remove this comment" }, "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "度", "minutesDegree": "分", "secondsDegree": "秒", "radiansDegree": "弧度", "usd": "美元", "eur": "欧元", "gbp": "英镑", "inr": "印度卢比", "cny": "人民币", "jpy": "日元", "chf": "瑞士法郎", "sek": "瑞典克朗", "rub": "俄罗斯卢布", "cad": "加元", "krw": "韩元", "brl": "巴西雷亚尔", "hkd": "港元", "aud": "澳元", "nzd": "新西兰元", "mxn": "墨西哥比索", "sgd": "新加坡元", "nok": "挪威克朗", "trY": "土耳其里拉", "zar": "南非兰特", "dkk": "丹麦克朗", "pln": "波兰兹罗提", "thb": "泰铢", "myr": "马来西亚林吉特", "huf": "匈牙利福林", "czk": "捷克克朗", "ils": "以色列新锡克尔", "idr": "印尼卢比", "php": "菲律宾比索", "ron": "罗马尼亚列伊", "isk": "冰岛克朗", "twd": "新台币", "mad": "摩洛哥道拉姆", "euChina": "欧盟 & 中国", "ukIndiaChild": "英国 & 印度 - 小孩", "ukIndiaMan": "英国 & 印度 - 男性", "ukIndiaWoman": "英国 & 印度 - 女性", "usaCanadaChild": "美国 & 加拿大 - 小孩", "usaCanadaMan": "美国 & 加拿大 - 男性", "usaCanadaWoman": "美国 & 加拿大 - 女性", "japan": "日本", "watt": "瓦特", "milliwatt": "毫瓦", "kilowatt": "千瓦", "megawatt": "兆瓦", "gigawatt": "千兆瓦", "europeanHorsePower": "公制马力", "imperialHorsePower": "英制马力", "newton": "牛顿", "dyne": "达因", "poundForce": "磅力", "kilogramForce": "千克力", "poundal": "磅达", "newtonMeter": "牛顿米", "dyneMeter": "达因计", "poundForceFeet": "磅力英尺", "kilogramForceMeter": "千克力米", "poundalMeter": "磅达米", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "公里/升", "liters100km": "百公里/升", "milesUsGallon": "英里/美制加仑", "milesImperialGallon": "英里/英制加仑", "decimal": "十进制", "hexadecimal": "十六进制", "octal": "八进制", "binary": "二进制", "bit": "比特", "nibble": "半字节", "kilobit": "千比特", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "兆比特", "@megabit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "吉比特", "@gigabit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "太比特", "@terabit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "拍比特", "@petabit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "艾比特", "@exabit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "千比特(二进)", "mebibit": "兆比特(二进)", "gibibit": "吉比特(二进)", "tebibit": "太比特(二进)", "pebibit": "拍比特(二进)", "exbibit": "艾比特(二进)", "byte": "字节", "kilobyte": "千字节", "@kilobyte": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "兆字节", "@megabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "吉字节", "@gigabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "太字节", "@terabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "拍字节", "@petabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "艾字节", "@exabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "千字节(二进)", "mebibyte": "兆字节(二进)", "gibibyte": "吉字节(二进)", "tebibyte": "太字节(二进)", "pebibyte": "拍字节(二进)", "exbibyte": "艾字节(二进)", "base": "基数", "deca": "十-", "hecto": "百-", "kilo": "千-", "mega": "兆-", "giga": "吉-", "tera": "太-", "peta": "拍-", "exa": "艾-", "zetta": "泽-", "yotta": "尧-", "deci": "分-", "centi": "厘-", "milli": "毫-", "micro": "微-", "nano": "纳-", "pico": "皮-", "femto": "飞-", "atto": "阿托-", "zepto": "仄普托-", "yocto": "幺科托-" } ================================================ FILE: packages/translations/lib/l10n/app_zh_TW.arb ================================================ { "@@locale": "zh_TW", "appName": "Converter NOW", "reorder": "排序", "myOrdering": "我的排序", "about": "關於", "lastCurrenciesUpdate": "最後更新:", "today": "今日", "enableDarkTheme": "啟用深色主題", "settings": "設定", "menu": "選單", "donation": "捐贈", "buyMeACoffee": "給我買一杯咖啡", "donationDialog": "嗨!您也許知道此應用程式是免費與開放原始碼的。這意味著您可以免費使用它與複製、編輯原始碼。開發者並不會因此獲得任何利益。但假如您認同此專案,或者想要看到新功能,請考慮給我買一杯咖啡。", "drawerLogo": "抽屜標誌", "rateApp": "為此應用程式評分", "repoGithub": "在 GitHub 開啟儲存庫", "contributeTranslating": "為此應用程式貢獻翻譯", "search": "搜尋", "clearAll": "清除全部", "calculator": "計算器", "significantFigures": "有效數字", "removeTrailingZeros": "移除尾隨零", "longPressAdvice": "長按以移動項目", "back": "返回", "save": "儲存", "invalidCharacters": "錯誤,無效的字元", "theme": "主題", "dark": "深色", "light": "淺色", "system": "系統預設", "pureBlackTheme": "純黑主題", "propertySelectionOnStartup": "Property selection on startup", "@propertySelectionOnStartup": { "description": "Not yet translated. Once done, remove this comment" }, "propertySelectionOnStartupSubtitle": "Show the property selection page on startup on small screen devices", "@propertySelectionOnStartupSubtitle": { "description": "Not yet translated. Once done, remove this comment" }, "copy": "複製", "paste": "貼上", "more": "更多", "language": "語言", "reorderProperties": "排序屬性", "reorderUnits": "排序單位", "chooseProperty": "選擇屬性", "reorderProperty": "排序 {property}", "@reorderProperty": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "otherPlatforms": "其他平台的 Converter NOW", "sourceCode": "原始碼", "undoClearAllMessage": "已清除全部。您是否想取消?", "undo": "取消", "routeError1": "此頁面不存在", "routeError2": "返回至第一頁", "ok": "OK", "revokeInternetAccess": "禁用網路連線", "revokeInternetExplanation": "該應用在每次啟動時透過網路更新匯率。所需下載的數據只有幾 kiB。不過,您若不需此功能,啟用該選項可阻止該應用連線網際網路。", "useDeviceColor": "使用系統設定的強調色", "pickColor": "選擇顏色", "themeColor": "主題色", "appearance": "外觀", "conversions": "轉換", "findOutMore": "進一步了解", "hideUnits": "隱藏單位", "visibleUnits": "{property}, 可見單位", "@visibleUnits": { "placeholders": { "property": { "type": "String", "example": "Length" } } }, "hiddenUnits": "隱藏的單位", "selectAll": "選擇所有", "unselectAll": "取消選擇", "backupAndRestore": "Backup & Restore", "@backupAndRestore": { "description": "Not yet translated. Once done, remove this comment" }, "exportSettings": "Export settings", "@exportSettings": { "description": "Not yet translated. Once done, remove this comment" }, "importSettings": "Import settings", "@importSettings": { "description": "Not yet translated. Once done, remove this comment" }, "clearSettings": "Clear all settings", "@clearSettings": { "description": "Not yet translated. Once done, remove this comment" }, "exportSuccess": "Settings exported successfully", "@exportSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "importSuccess": "Settings imported successfully", "@importSuccess": { "description": "Not yet translated. Once done, remove this comment" }, "problemImportFile": "There was a problem importing the file", "@problemImportFile": { "description": "Not yet translated. Once done, remove this comment" }, "relatedSettings": "Related settings", "@relatedSettings": { "description": "Not yet translated. Once done, remove this comment" }, "reason": "Reason", "@reason": { "description": "Not yet translated. Once done, remove this comment" }, "importError": "Import Error", "@importError": { "description": "Not yet translated. Once done, remove this comment" }, "confirmationClear": "Are you sure you want to clear all settings? This action cannot be undone.", "@confirmationClear": { "description": "Not yet translated. Once done, remove this comment" }, "cancel": "Cancel", "@cancel": { "description": "Not yet translated. Once done, remove this comment" }, "length": "長度", "area": "面積", "volume": "體積", "time": "時間", "temperature": "溫度", "speed": "速度", "siPrefixes": "國際單位制前置詞", "mass": "質量", "pressure": "壓力", "energy": "能量", "angles": "角度", "currencies": "貨幣", "shoeSize": "鞋號", "digitalData": "數位資料", "power": "功率", "density": "密度", "torque": "力矩", "force": "力", "fuelConsumption": "燃料消耗量", "numeralSystems": "數值系統", "meters": "公尺;米", "centimeters": "公分", "inches": "英寸", "feet": "英尺", "nauticalMiles": "海里", "yards": "碼", "miles": "英里", "millimeters": "公厘;毫米", "micrometers": "微米", "nanometers": "奈米", "angstroms": "埃", "picometers": "皮米", "kilometers": "公里", "astronomicalUnits": "天文單位", "lightYears": "光年", "parsec": "秒差距", "mils": "密耳", "feetUsSurvey": "英尺 (美國)", "squareMeters": "平方公尺;平方米", "squareCentimeters": "平方公分", "squareInches": "平方英寸", "squareFeet": "平方英尺", "squareMiles": "平方英里", "squareYard": "平方碼", "squareMillimeters": "平方公厘;平方毫米", "squareKilometers": "平方公里", "hectares": "公頃", "acres": "英畝", "are": "公畝", "squareFeetUsSurvey": "平方英尺 (美國)", "gramsPerCubicCentimeter": "公克每立方公分", "gramsPerLiter": "公克每公升", "gramsPerMilliliter": "公克每毫升;公克每公撮", "gramsPerDeciliter": "公克每分升;公克每公合", "kilogramsPerLiter": "公斤每公升", "kilogramsPerCubicMeter": "公斤每立方公尺;公斤每立方米", "milligramsPerLiter": "毫克每公升", "milligramsPerDeciliter": "毫克每分升;毫克每公合", "milligramsPerMilliliter": "毫克每毫升;毫克每公撮", "milligramsPerCubicMeter": "毫克每立方公尺;毫克每立方米", "milligramsPerCubicCentimeter": "毫克每立方公分", "microgramsPerLiter": "微克每公升", "microgramsPerDeciliter": "微克每分升;微克每公合", "microgramsPerMilliliter": "微克每毫升;微克每公撮", "poundsPerCubicInches": "磅每立方英寸", "poundsPerCubicFeet": "磅每立方英尺", "cubicMeters": "立方公尺;立方米", "liters": "公升", "imperialGallons": "英制加侖", "usGallons": "美制加侖", "imperialPints": "英制品脫", "usPints": "美制品脫", "milliliters": "毫升;公撮", "tablespoonUs": "一湯匙 (美國)", "tablespoonAustralian": "一湯匙 (澳大利亞)", "cups": "杯", "cubicCentimeters": "立方公分", "cubicFeet": "立方英尺", "cubicInches": "立方英寸", "cubicMillimeters": "立方公厘;立方毫米", "imperialFluidOunces": "英制液盎司", "usFluidOunces": "美制液盎司", "imperialGill": "英制及耳", "usGill": "美制及耳", "usQuarts": "美制夸脫", "microliters": "微升", "deciliters": "公合;分升", "centiliters": "公勺;厘升", "teaspoonsUs": "Teaspoons US", "@teaspoonsUs": { "description": "Not yet translated. Once done, remove this comment" }, "teaspoonsMetric": "Metric teaspoons", "@teaspoonsMetric": { "description": "Not yet translated. Once done, remove this comment" }, "seconds": "秒", "deciseconds": "分秒", "centiseconds": "厘秒", "milliseconds": "毫秒", "microseconds": "微秒", "nanoseconds": "奈秒", "minutes": "分[鐘]", "hours": "[小]時", "days": "日", "weeks": "週", "years": "年 (365)", "lustrum": "五年紀", "decades": "十年紀", "centuries": "世紀", "millennium": "千年紀", "fahrenheit": "華氏", "celsius": "攝氏", "kelvin": "克氏;克耳文", "reamur": "列氏", "romer": "羅氏", "delisle": "德利爾", "rankine": "蘭氏;蘭金", "metersSecond": "公尺每秒;米每秒", "kilometersHour": "公里每小時", "milesHour": "英里每小時", "knots": "節", "feetSecond": "英尺每秒", "minutesPerKilometer": "分鐘每公里", "minutesPerMile": "Minutes per mile", "@minutesPerMile": { "description": "Not yet translated. Once done, remove this comment" }, "speedOfLight": "Speed of light", "@speedOfLight": { "description": "Not yet translated. Once done, remove this comment" }, "grams": "公克", "ettograms": "百克", "kilograms": "公斤", "pounds": "磅", "ounces": "盎司", "quintals": "公擔", "tonnes": "噸", "milligrams": "毫克", "uma": "統一原子質量單位", "carats": "克拉", "centigrams": "公毫", "pennyweights": "便士法碼", "troyOunces": "金衡盎司", "stones": "英石", "pascal": "帕[斯卡]", "atmosphere": "大氣壓", "bar": "巴", "millibar": "毫巴", "psi": "磅每平方英寸", "torr": "托", "hectoPascal": "百帕[斯卡]", "inchesOfMercury": "英寸汞柱", "kiloPascal": "千帕[斯卡]", "ksi": "千磅每平方英寸", "megaPascal": "百萬帕[斯卡]", "gigaPascal": "吉帕[斯卡]", "joule": "焦耳", "calories": "卡[路里]", "kilowattHour": "千瓦小時", "electronvolt": "電子伏特", "footPound": "英尺磅", "kilocalories": "大卡", "kilojoules": "千焦耳", "watthour": "Watt hour", "@watthour": { "description": "Not yet translated. Once done, remove this comment" }, "britishThermalUnit": "British thermal unit", "@britishThermalUnit": { "description": "Not yet translated. Once done, remove this comment" }, "degree": "度", "minutesDegree": "分", "secondsDegree": "秒", "radiansDegree": "弧度", "usd": "美金", "eur": "歐元", "gbp": "英鎊", "inr": "印度盧比", "cny": "人民幣", "jpy": "日圓", "chf": "瑞士法郎", "sek": "瑞典幣", "rub": "俄羅斯盧布", "cad": "加拿大幣", "krw": "韓元", "brl": "巴西雷亞爾", "hkd": "港幣", "aud": "澳幣", "nzd": "紐元", "mxn": "墨西哥披索", "sgd": "新加坡幣", "nok": "挪威克朗", "trY": "土耳其里拉", "zar": "南非幣", "dkk": "丹麥克朗", "pln": "波蘭茲羅提", "thb": "泰幣", "myr": "馬來幣", "huf": "匈牙利福林", "czk": "捷克克朗", "ils": "以色列新謝克爾", "idr": "印尼幣", "php": "菲國比索", "ron": "羅馬尼亞列伊", "isk": "冰島克朗", "twd": "新台幣", "mad": "摩洛哥迪拉姆", "euChina": "歐盟 & 中國", "ukIndiaChild": "英國 & 印度 - 孩童", "ukIndiaMan": "英國 & 印度 - 男性", "ukIndiaWoman": "英國 & 印度 - 女性", "usaCanadaChild": "美國 & 加拿大 - 孩童", "usaCanadaMan": "美國 & 加拿大 - 男性", "usaCanadaWoman": "美國 & 加拿大 - 女性", "japan": "日本", "watt": "瓦特", "milliwatt": "毫瓦", "kilowatt": "千瓦", "megawatt": "百萬瓦", "gigawatt": "吉瓦", "europeanHorsePower": "公制馬力", "imperialHorsePower": "英制馬力", "newton": "牛頓", "dyne": "達因", "poundForce": "磅力", "kilogramForce": "公斤力", "poundal": "磅達", "newtonMeter": "牛頓公尺;牛頓米", "dyneMeter": "達因公尺;達因米", "poundForceFeet": "磅力英尺", "kilogramForceMeter": "公斤力公尺", "poundalMeter": "磅達公尺;磅達米", "poundForceInch": "Pound-force inch", "@poundForceInch": { "description": "Not yet translated. Once done, remove this comment." }, "kilometersLiter": "公里每公升", "liters100km": "公升每百公里", "milesUsGallon": "英里每美制加侖", "milesImperialGallon": "英里每英制加侖", "decimal": "十進位制", "hexadecimal": "十六進位制", "octal": "八進位制", "binary": "二進位制", "bit": "位元", "nibble": "半位元組", "kilobit": "千位元", "@kilobit": { "description": "Revise this translation to also include the 'decimal' part" }, "megabit": "百萬位元", "@megabit": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabit": "吉位元", "@gigabit": { "description": "Revise this translation to also include the 'decimal' part" }, "terabit": "兆位元", "@terabit": { "description": "Revise this translation to also include the 'decimal' part" }, "petabit": "拍位元", "@petabit": { "description": "Revise this translation to also include the 'decimal' part" }, "exabit": "艾位元", "@exabit": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibit": "千位元(二進)", "mebibit": "百萬位元(二進)", "gibibit": "吉位元(二進)", "tebibit": "兆位元(二進)", "pebibit": "拍位元(二進)", "exbibit": "艾位元(二進)", "byte": "位元組", "kilobyte": "千位元組", "@kilobyte": { "description": "Revise this translation to also include the 'decimal' part" }, "megabyte": "百萬位元組", "@megabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "gigabyte": "吉位元組", "@gigabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "terabyte": "兆位元組", "@terabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "petabyte": "拍位元組", "@petabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "exabyte": "艾位元組", "@exabyte": { "description": "Revise this translation to also include the 'decimal' part" }, "kibibyte": "千位元組", "mebibyte": "百萬位元組", "gibibyte": "吉位元組", "tebibyte": "兆位元組", "pebibyte": "拍位元組", "exbibyte": "艾位元組", "base": "基數", "deca": "十-", "hecto": "百-", "kilo": "千-", "mega": "百萬-", "giga": "吉-", "tera": "兆-", "peta": "拍-", "exa": "艾-", "zetta": "皆-", "yotta": "佑-", "deci": "分-", "centi": "厘-", "milli": "毫-", "micro": "微-", "nano": "奈-", "pico": "皮-", "femto": "飛-", "atto": "阿-", "zepto": "介-", "yocto": "攸-" } ================================================ FILE: packages/translations/pubspec.yaml ================================================ name: translations description: All the translations for Converter NOW. publish_to: 'none' resolution: workspace version: 1.0.0+1 environment: sdk: ">=3.9.0 <4.0.0" dependencies: flutter: sdk: flutter flutter_localizations: sdk: flutter intl: ^0.20.2 dev_dependencies: flutter_test: sdk: flutter flutter: generate: true ================================================ FILE: pubspec.yaml ================================================ name: converterpro description: Unit and currencies converter publish_to: none version: 4.6.0+52 environment: sdk: ">=3.11.0 <4.0.0" flutter: ">=3.35.1 <4.0.0" workspace: - packages/calculator_widget - packages/translations dependencies: flutter: sdk: flutter flutter_localizations: sdk: flutter # Various collection utilities collection: ^1.19.1 # App dynamic color theme dynamic_color: ^1.8.1 # Required for import/export file_picker: ^10.3.10 # State management flutter_riverpod: ^3.2.1 # For svg icons flutter_svg: ^2.2.3 # Navigation go_router: ^17.1.0 # Rounded corner window on linux handy_window: ^0.4.2 # Downloading the exchange rates http: ^1.6.0 # Translations intl: ^0.20.2 # Required for import/export path_provider: ^2.1.5 # Android app shortcuts quick_actions: ^1.1.0 # Required for import/export share_plus: ^12.0.1 # Store the customizations and the latest currencies conversions shared_preferences: ^2.5.3 # Unit conversion units_converter: ^3.1.0 # Open external pages such as GitHub repo, Play Store, etc url_launcher: ^6.3.2 # Load optimized svg icons vector_graphics: ^1.1.19 # Resize window during testing window_size: git: url: https://github.com/google/flutter-desktop-embedding ref: eb3964990cf19629c89ff8cb4a37640c7b3d5601 path: plugins/window_size # Sub packages --------------------------------------------------------------- # Calculator widget and logic calculator_widget: path: packages/calculator_widget # App translations translations: path: packages/translations dev_dependencies: flutter_lints: ^6.0.0 flutter_test: sdk: flutter integration_test: sdk: flutter melos: ^7.4.0 msix: ^3.16.13 test: ^1.26.2 melos: command: bootstrap: hooks: post: melos run generate_translations && melos run compile_icons scripts: analyze: run: melos exec -- "flutter analyze" description: Run `flutter analyze` in all packages format: run: melos exec -- "flutter format . --set-exit-if-changed" description: Run `flutter format .` in all packages compile_icons: steps: - dart run vector_graphics_compiler --input-dir assets/app_icons --out-dir assets/app_icons_opti - dart run vector_graphics_compiler --input-dir assets/property_icons --out-dir assets/property_icons_opti - dart run vector_graphics_compiler --input-dir assets/flags --out-dir assets/flags_opti description: Optimizes the svg icons. generate_translations: run: cd packages/translations && flutter gen-l10n description: Generates translation files. test: run: melos exec --dir-exists=test -- "flutter test" description: Run `flutter test` in all packages msix_config: display_name: Converter NOW publisher_display_name: Damiano Ferrari identity_name: 39826DamianoFerrari.ConverterNOW logo_path: assets\logo.png output_name: converternow-windows publisher: CN=E8FE6044-F04C-422E-AC68-EDA83F6AAFB2 capabilities: internetClient flutter: uses-material-design: true generate: true fonts: - family: JosefinSans fonts: - asset: assets/fonts/JosefinSans-Light.ttf weight: 300 assets: - assets/app_icons_opti/ - assets/fonts/OFL.txt - assets/property_icons_opti/ - assets/flags_opti/ - assets/flags/br.png - assets/flags/eu.png - assets/flags/in.png - assets/flags/mx.png - assets/flags/nz.png - assets/flags/us.png - path: assets/logo.png platforms: [windows] ================================================ FILE: web/index.html ================================================ Converter NOW
Loading indicator...
================================================ FILE: web/manifest.json ================================================ { "name": "Converter NOW", "short_name": "Converter NOW", "start_url": ".", "display": "standalone", "background_color": "#000000", "theme_color": "#000000", "description": "A Unit and Currencies Converter. Converter NOW is immediate, fast and easy to use!", "orientation": "portrait-primary", "prefer_related_applications": false, "icons": [ { "src": "icons/Icon-192.png", "sizes": "192x192", "type": "image/png" }, { "src": "icons/Icon-512.png", "sizes": "512x512", "type": "image/png" }, { "src": "icons/Maskable-Icon-192.png", "sizes": "192x192", "type": "image/png", "purpose": "maskable" }, { "src": "icons/Maskable-Icon-512.png", "sizes": "512x512", "type": "image/png", "purpose": "maskable" } ] } ================================================ FILE: windows/.gitignore ================================================ flutter/ephemeral/ # Visual Studio user-specific files. *.suo *.user *.userosscache *.sln.docstates # Visual Studio build-related files. x64/ x86/ # Visual Studio cache files # files ending in .cache can be ignored *.[Cc]ache # but keep track of directories ending in .cache !*.[Cc]ache/ ================================================ FILE: windows/CMakeLists.txt ================================================ # Project-level configuration. cmake_minimum_required(VERSION 3.14) project(converternow LANGUAGES CXX) # The name of the executable created for the application. Change this to change # the on-disk name of your application. set(BINARY_NAME "converternow") # Explicitly opt in to modern CMake behaviors to avoid warnings with recent # versions of CMake. cmake_policy(VERSION 3.14...3.25) # Define build configuration option. get_property(IS_MULTICONFIG GLOBAL PROPERTY GENERATOR_IS_MULTI_CONFIG) if(IS_MULTICONFIG) set(CMAKE_CONFIGURATION_TYPES "Debug;Profile;Release" CACHE STRING "" FORCE) else() if(NOT CMAKE_BUILD_TYPE AND NOT CMAKE_CONFIGURATION_TYPES) set(CMAKE_BUILD_TYPE "Debug" CACHE STRING "Flutter build mode" FORCE) set_property(CACHE CMAKE_BUILD_TYPE PROPERTY STRINGS "Debug" "Profile" "Release") endif() endif() # Define settings for the Profile build mode. set(CMAKE_EXE_LINKER_FLAGS_PROFILE "${CMAKE_EXE_LINKER_FLAGS_RELEASE}") set(CMAKE_SHARED_LINKER_FLAGS_PROFILE "${CMAKE_SHARED_LINKER_FLAGS_RELEASE}") set(CMAKE_C_FLAGS_PROFILE "${CMAKE_C_FLAGS_RELEASE}") set(CMAKE_CXX_FLAGS_PROFILE "${CMAKE_CXX_FLAGS_RELEASE}") # Use Unicode for all projects. add_definitions(-DUNICODE -D_UNICODE) # Compilation settings that should be applied to most targets. # # Be cautious about adding new options here, as plugins use this function by # default. In most cases, you should add new options to specific targets instead # of modifying this function. function(APPLY_STANDARD_SETTINGS TARGET) target_compile_features(${TARGET} PUBLIC cxx_std_17) target_compile_options(${TARGET} PRIVATE /W4 /WX /wd"4100") target_compile_options(${TARGET} PRIVATE /EHsc) target_compile_definitions(${TARGET} PRIVATE "_HAS_EXCEPTIONS=0") target_compile_definitions(${TARGET} PRIVATE "$<$:_DEBUG>") endfunction() # Flutter library and tool build rules. set(FLUTTER_MANAGED_DIR "${CMAKE_CURRENT_SOURCE_DIR}/flutter") add_subdirectory(${FLUTTER_MANAGED_DIR}) # Application build; see runner/CMakeLists.txt. add_subdirectory("runner") # Generated plugin build rules, which manage building the plugins and adding # them to the application. include(flutter/generated_plugins.cmake) # === Installation === # Support files are copied into place next to the executable, so that it can # run in place. This is done instead of making a separate bundle (as on Linux) # so that building and running from within Visual Studio will work. set(BUILD_BUNDLE_DIR "$") # Make the "install" step default, as it's required to run. set(CMAKE_VS_INCLUDE_INSTALL_TO_DEFAULT_BUILD 1) if(CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) set(CMAKE_INSTALL_PREFIX "${BUILD_BUNDLE_DIR}" CACHE PATH "..." FORCE) endif() set(INSTALL_BUNDLE_DATA_DIR "${CMAKE_INSTALL_PREFIX}/data") set(INSTALL_BUNDLE_LIB_DIR "${CMAKE_INSTALL_PREFIX}") install(TARGETS ${BINARY_NAME} RUNTIME DESTINATION "${CMAKE_INSTALL_PREFIX}" COMPONENT Runtime) install(FILES "${FLUTTER_ICU_DATA_FILE}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" COMPONENT Runtime) install(FILES "${FLUTTER_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" COMPONENT Runtime) if(PLUGIN_BUNDLED_LIBRARIES) install(FILES "${PLUGIN_BUNDLED_LIBRARIES}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" COMPONENT Runtime) endif() # Copy the native assets provided by the build.dart from all packages. set(NATIVE_ASSETS_DIR "${PROJECT_BUILD_DIR}native_assets/windows/") install(DIRECTORY "${NATIVE_ASSETS_DIR}" DESTINATION "${INSTALL_BUNDLE_LIB_DIR}" COMPONENT Runtime) # Fully re-copy the assets directory on each build to avoid having stale files # from a previous install. set(FLUTTER_ASSET_DIR_NAME "flutter_assets") install(CODE " file(REMOVE_RECURSE \"${INSTALL_BUNDLE_DATA_DIR}/${FLUTTER_ASSET_DIR_NAME}\") " COMPONENT Runtime) install(DIRECTORY "${PROJECT_BUILD_DIR}/${FLUTTER_ASSET_DIR_NAME}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" COMPONENT Runtime) # Install the AOT library on non-Debug builds only. install(FILES "${AOT_LIBRARY}" DESTINATION "${INSTALL_BUNDLE_DATA_DIR}" CONFIGURATIONS Profile;Release COMPONENT Runtime) ================================================ FILE: windows/flutter/CMakeLists.txt ================================================ # This file controls Flutter-level build steps. It should not be edited. cmake_minimum_required(VERSION 3.14) set(EPHEMERAL_DIR "${CMAKE_CURRENT_SOURCE_DIR}/ephemeral") # Configuration provided via flutter tool. include(${EPHEMERAL_DIR}/generated_config.cmake) # TODO: Move the rest of this into files in ephemeral. See # https://github.com/flutter/flutter/issues/57146. set(WRAPPER_ROOT "${EPHEMERAL_DIR}/cpp_client_wrapper") # Set fallback configurations for older versions of the flutter tool. if (NOT DEFINED FLUTTER_TARGET_PLATFORM) set(FLUTTER_TARGET_PLATFORM "windows-x64") endif() # === Flutter Library === set(FLUTTER_LIBRARY "${EPHEMERAL_DIR}/flutter_windows.dll") # Published to parent scope for install step. set(FLUTTER_LIBRARY ${FLUTTER_LIBRARY} PARENT_SCOPE) set(FLUTTER_ICU_DATA_FILE "${EPHEMERAL_DIR}/icudtl.dat" PARENT_SCOPE) set(PROJECT_BUILD_DIR "${PROJECT_DIR}/build/" PARENT_SCOPE) set(AOT_LIBRARY "${PROJECT_DIR}/build/windows/app.so" PARENT_SCOPE) list(APPEND FLUTTER_LIBRARY_HEADERS "flutter_export.h" "flutter_windows.h" "flutter_messenger.h" "flutter_plugin_registrar.h" "flutter_texture_registrar.h" ) list(TRANSFORM FLUTTER_LIBRARY_HEADERS PREPEND "${EPHEMERAL_DIR}/") add_library(flutter INTERFACE) target_include_directories(flutter INTERFACE "${EPHEMERAL_DIR}" ) target_link_libraries(flutter INTERFACE "${FLUTTER_LIBRARY}.lib") add_dependencies(flutter flutter_assemble) # === Wrapper === list(APPEND CPP_WRAPPER_SOURCES_CORE "core_implementations.cc" "standard_codec.cc" ) list(TRANSFORM CPP_WRAPPER_SOURCES_CORE PREPEND "${WRAPPER_ROOT}/") list(APPEND CPP_WRAPPER_SOURCES_PLUGIN "plugin_registrar.cc" ) list(TRANSFORM CPP_WRAPPER_SOURCES_PLUGIN PREPEND "${WRAPPER_ROOT}/") list(APPEND CPP_WRAPPER_SOURCES_APP "flutter_engine.cc" "flutter_view_controller.cc" ) list(TRANSFORM CPP_WRAPPER_SOURCES_APP PREPEND "${WRAPPER_ROOT}/") # Wrapper sources needed for a plugin. add_library(flutter_wrapper_plugin STATIC ${CPP_WRAPPER_SOURCES_CORE} ${CPP_WRAPPER_SOURCES_PLUGIN} ) apply_standard_settings(flutter_wrapper_plugin) set_target_properties(flutter_wrapper_plugin PROPERTIES POSITION_INDEPENDENT_CODE ON) set_target_properties(flutter_wrapper_plugin PROPERTIES CXX_VISIBILITY_PRESET hidden) target_link_libraries(flutter_wrapper_plugin PUBLIC flutter) target_include_directories(flutter_wrapper_plugin PUBLIC "${WRAPPER_ROOT}/include" ) add_dependencies(flutter_wrapper_plugin flutter_assemble) # Wrapper sources needed for the runner. add_library(flutter_wrapper_app STATIC ${CPP_WRAPPER_SOURCES_CORE} ${CPP_WRAPPER_SOURCES_APP} ) apply_standard_settings(flutter_wrapper_app) target_link_libraries(flutter_wrapper_app PUBLIC flutter) target_include_directories(flutter_wrapper_app PUBLIC "${WRAPPER_ROOT}/include" ) add_dependencies(flutter_wrapper_app flutter_assemble) # === Flutter tool backend === # _phony_ is a non-existent file to force this command to run every time, # since currently there's no way to get a full input/output list from the # flutter tool. set(PHONY_OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/_phony_") set_source_files_properties("${PHONY_OUTPUT}" PROPERTIES SYMBOLIC TRUE) add_custom_command( OUTPUT ${FLUTTER_LIBRARY} ${FLUTTER_LIBRARY_HEADERS} ${CPP_WRAPPER_SOURCES_CORE} ${CPP_WRAPPER_SOURCES_PLUGIN} ${CPP_WRAPPER_SOURCES_APP} ${PHONY_OUTPUT} COMMAND ${CMAKE_COMMAND} -E env ${FLUTTER_TOOL_ENVIRONMENT} "${FLUTTER_ROOT}/packages/flutter_tools/bin/tool_backend.bat" ${FLUTTER_TARGET_PLATFORM} $ VERBATIM ) add_custom_target(flutter_assemble DEPENDS "${FLUTTER_LIBRARY}" ${FLUTTER_LIBRARY_HEADERS} ${CPP_WRAPPER_SOURCES_CORE} ${CPP_WRAPPER_SOURCES_PLUGIN} ${CPP_WRAPPER_SOURCES_APP} ) ================================================ FILE: windows/flutter/generated_plugin_registrant.cc ================================================ // // Generated file. Do not edit. // // clang-format off #include "generated_plugin_registrant.h" #include #include #include #include void RegisterPlugins(flutter::PluginRegistry* registry) { DynamicColorPluginCApiRegisterWithRegistrar( registry->GetRegistrarForPlugin("DynamicColorPluginCApi")); SharePlusWindowsPluginCApiRegisterWithRegistrar( registry->GetRegistrarForPlugin("SharePlusWindowsPluginCApi")); UrlLauncherWindowsRegisterWithRegistrar( registry->GetRegistrarForPlugin("UrlLauncherWindows")); WindowSizePluginRegisterWithRegistrar( registry->GetRegistrarForPlugin("WindowSizePlugin")); } ================================================ FILE: windows/flutter/generated_plugin_registrant.h ================================================ // // Generated file. Do not edit. // // clang-format off #ifndef GENERATED_PLUGIN_REGISTRANT_ #define GENERATED_PLUGIN_REGISTRANT_ #include // Registers Flutter plugins. void RegisterPlugins(flutter::PluginRegistry* registry); #endif // GENERATED_PLUGIN_REGISTRANT_ ================================================ FILE: windows/flutter/generated_plugins.cmake ================================================ # # Generated file, do not edit. # list(APPEND FLUTTER_PLUGIN_LIST dynamic_color share_plus url_launcher_windows window_size ) list(APPEND FLUTTER_FFI_PLUGIN_LIST ) set(PLUGIN_BUNDLED_LIBRARIES) foreach(plugin ${FLUTTER_PLUGIN_LIST}) add_subdirectory(flutter/ephemeral/.plugin_symlinks/${plugin}/windows plugins/${plugin}) target_link_libraries(${BINARY_NAME} PRIVATE ${plugin}_plugin) list(APPEND PLUGIN_BUNDLED_LIBRARIES $) list(APPEND PLUGIN_BUNDLED_LIBRARIES ${${plugin}_bundled_libraries}) endforeach(plugin) foreach(ffi_plugin ${FLUTTER_FFI_PLUGIN_LIST}) add_subdirectory(flutter/ephemeral/.plugin_symlinks/${ffi_plugin}/windows plugins/${ffi_plugin}) list(APPEND PLUGIN_BUNDLED_LIBRARIES ${${ffi_plugin}_bundled_libraries}) endforeach(ffi_plugin) ================================================ FILE: windows/innosetup.iss ================================================ ; Script generated by the Inno Setup Script Wizard. ; SEE THE DOCUMENTATION FOR DETAILS ON CREATING INNO SETUP SCRIPT FILES! #define MyAppName "Converter NOW" #define MyAppVersion "4.6.0" #define MyAppPublisher "Damiano Ferrari" #define MyAppURL "https://github.com/ferraridamiano/ConverterNOW" #define MyAppExeName "converternow.exe" #define MyAppDescription "Unit and currencies converter" [Setup] ; NOTE: The value of AppId uniquely identifies this application. Do not use the same AppId value in installers for other applications. ; (To generate a new GUID, click Tools | Generate GUID inside the IDE.) AppId={{B569A7ED-8CA0-402C-A04D-0BF6F634FF17} AppName={#MyAppName} AppVersion={#MyAppVersion} ;AppVerName={#MyAppName} {#MyAppVersion} AppPublisher={#MyAppPublisher} AppPublisherURL={#MyAppURL} AppSupportURL={#MyAppURL} AppUpdatesURL={#MyAppURL} DefaultDirName={autopf}\{#MyAppName} UninstallDisplayIcon={app}\{#MyAppExeName} ; "ArchitecturesAllowed=x64compatible" specifies that Setup cannot run ; on anything but x64 and Windows 11 on Arm. ArchitecturesAllowed=x64compatible ; "ArchitecturesInstallIn64BitMode=x64compatible" requests that the ; install be done in "64-bit mode" on x64 or Windows 11 on Arm, ; meaning it should use the native 64-bit Program Files directory and ; the 64-bit view of the registry. ArchitecturesInstallIn64BitMode=x64compatible DisableProgramGroupPage=yes ; Remove the following line to run in administrative install mode (install for all users). PrivilegesRequired=lowest OutputDir=. OutputBaseFilename=converternow-windows-setup SetupIconFile=runner\resources\app_icon.ico PrivilegesRequiredOverridesAllowed=dialog SolidCompression=yes WizardStyle=modern [Languages] Name: "english"; MessagesFile: "compiler:Default.isl" Name: "arabic"; MessagesFile: "compiler:Languages\Arabic.isl" Name: "brazilianportuguese"; MessagesFile: "compiler:Languages\BrazilianPortuguese.isl" Name: "catalan"; MessagesFile: "compiler:Languages\Catalan.isl" Name: "danish"; MessagesFile: "compiler:Languages\Danish.isl" Name: "dutch"; MessagesFile: "compiler:Languages\Dutch.isl" Name: "french"; MessagesFile: "compiler:Languages\French.isl" Name: "german"; MessagesFile: "compiler:Languages\German.isl" Name: "hungarian"; MessagesFile: "compiler:Languages\Hungarian.isl" Name: "italian"; MessagesFile: "compiler:Languages\Italian.isl" Name: "japanese"; MessagesFile: "compiler:Languages\Japanese.isl" Name: "norwegian"; MessagesFile: "compiler:Languages\Norwegian.isl" Name: "polish"; MessagesFile: "compiler:Languages\Polish.isl" Name: "portuguese"; MessagesFile: "compiler:Languages\Portuguese.isl" Name: "russian"; MessagesFile: "compiler:Languages\Russian.isl" Name: "spanish"; MessagesFile: "compiler:Languages\Spanish.isl" Name: "thai"; MessagesFile: "compiler:Languages\Thai.isl" [Tasks] Name: "desktopicon"; Description: "{cm:CreateDesktopIcon}"; GroupDescription: "{cm:AdditionalIcons}"; Flags: unchecked [Files] Source: "..\build\windows\x64\runner\Release\*"; DestDir: "{app}"; Flags: ignoreversion recursesubdirs createallsubdirs; Excludes: "*.msix" ; NOTE: Don't use "Flags: ignoreversion" on any shared system files [Icons] Name: "{autoprograms}\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}" Name: "{autodesktop}\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"; Tasks: desktopicon [Run] Filename: "{app}\{#MyAppExeName}"; Description: "{cm:LaunchProgram,{#StringChange(MyAppName, '&', '&&')}}"; Flags: nowait postinstall skipifsilent ================================================ FILE: windows/runner/CMakeLists.txt ================================================ cmake_minimum_required(VERSION 3.14) project(runner LANGUAGES CXX) # Define the application target. To change its name, change BINARY_NAME in the # top-level CMakeLists.txt, not the value here, or `flutter run` will no longer # work. # # Any new source files that you add to the application should be added here. add_executable(${BINARY_NAME} WIN32 "flutter_window.cpp" "main.cpp" "utils.cpp" "win32_window.cpp" "${FLUTTER_MANAGED_DIR}/generated_plugin_registrant.cc" "Runner.rc" "runner.exe.manifest" ) # Apply the standard set of build settings. This can be removed for applications # that need different build settings. apply_standard_settings(${BINARY_NAME}) # Add preprocessor definitions for the build version. target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION=\"${FLUTTER_VERSION}\"") target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_MAJOR=${FLUTTER_VERSION_MAJOR}") target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_MINOR=${FLUTTER_VERSION_MINOR}") target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_PATCH=${FLUTTER_VERSION_PATCH}") target_compile_definitions(${BINARY_NAME} PRIVATE "FLUTTER_VERSION_BUILD=${FLUTTER_VERSION_BUILD}") # Disable Windows macros that collide with C++ standard library functions. target_compile_definitions(${BINARY_NAME} PRIVATE "NOMINMAX") # Add dependency libraries and include directories. Add any application-specific # dependencies here. target_link_libraries(${BINARY_NAME} PRIVATE flutter flutter_wrapper_app) target_link_libraries(${BINARY_NAME} PRIVATE "dwmapi.lib") target_include_directories(${BINARY_NAME} PRIVATE "${CMAKE_SOURCE_DIR}") # Run the Flutter tool portions of the build. This must not be removed. add_dependencies(${BINARY_NAME} flutter_assemble) ================================================ FILE: windows/runner/Runner.rc ================================================ // Microsoft Visual C++ generated resource script. // #pragma code_page(65001) #include "resource.h" #define APSTUDIO_READONLY_SYMBOLS ///////////////////////////////////////////////////////////////////////////// // // Generated from the TEXTINCLUDE 2 resource. // #include "winres.h" ///////////////////////////////////////////////////////////////////////////// #undef APSTUDIO_READONLY_SYMBOLS ///////////////////////////////////////////////////////////////////////////// // English (United States) resources #if !defined(AFX_RESOURCE_DLL) || defined(AFX_TARG_ENU) LANGUAGE LANG_ENGLISH, SUBLANG_ENGLISH_US #ifdef APSTUDIO_INVOKED ///////////////////////////////////////////////////////////////////////////// // // TEXTINCLUDE // 1 TEXTINCLUDE BEGIN "resource.h\0" END 2 TEXTINCLUDE BEGIN "#include ""winres.h""\r\n" "\0" END 3 TEXTINCLUDE BEGIN "\r\n" "\0" END #endif // APSTUDIO_INVOKED ///////////////////////////////////////////////////////////////////////////// // // Icon // // Icon with lowest ID value placed first to ensure application icon // remains consistent on all systems. IDI_APP_ICON ICON "resources\\app_icon.ico" ///////////////////////////////////////////////////////////////////////////// // // Version // #if defined(FLUTTER_VERSION_MAJOR) && defined(FLUTTER_VERSION_MINOR) && defined(FLUTTER_VERSION_PATCH) && defined(FLUTTER_VERSION_BUILD) #define VERSION_AS_NUMBER FLUTTER_VERSION_MAJOR,FLUTTER_VERSION_MINOR,FLUTTER_VERSION_PATCH,FLUTTER_VERSION_BUILD #else #define VERSION_AS_NUMBER 1,0,0,0 #endif #if defined(FLUTTER_VERSION) #define VERSION_AS_STRING FLUTTER_VERSION #else #define VERSION_AS_STRING "1.0.0" #endif VS_VERSION_INFO VERSIONINFO FILEVERSION VERSION_AS_NUMBER PRODUCTVERSION VERSION_AS_NUMBER FILEFLAGSMASK VS_FFI_FILEFLAGSMASK #ifdef _DEBUG FILEFLAGS VS_FF_DEBUG #else FILEFLAGS 0x0L #endif FILEOS VOS__WINDOWS32 FILETYPE VFT_APP FILESUBTYPE 0x0L BEGIN BLOCK "StringFileInfo" BEGIN BLOCK "040904e4" BEGIN VALUE "CompanyName", "com.ferrarid" "\0" VALUE "FileDescription", "Converter NOW" "\0" VALUE "FileVersion", VERSION_AS_STRING "\0" VALUE "InternalName", "converternow" "\0" VALUE "LegalCopyright", "Copyright (C) 2024 Damiano Ferrari. All rights reserved." "\0" VALUE "OriginalFilename", "converternow.exe" "\0" VALUE "ProductName", "converternow" "\0" VALUE "ProductVersion", VERSION_AS_STRING "\0" END END BLOCK "VarFileInfo" BEGIN VALUE "Translation", 0x409, 1252 END END #endif // English (United States) resources ///////////////////////////////////////////////////////////////////////////// #ifndef APSTUDIO_INVOKED ///////////////////////////////////////////////////////////////////////////// // // Generated from the TEXTINCLUDE 3 resource. // ///////////////////////////////////////////////////////////////////////////// #endif // not APSTUDIO_INVOKED ================================================ FILE: windows/runner/flutter_window.cpp ================================================ #include "flutter_window.h" #include #include "flutter/generated_plugin_registrant.h" FlutterWindow::FlutterWindow(const flutter::DartProject& project) : project_(project) {} FlutterWindow::~FlutterWindow() {} bool FlutterWindow::OnCreate() { if (!Win32Window::OnCreate()) { return false; } RECT frame = GetClientArea(); // The size here must match the window dimensions to avoid unnecessary surface // creation / destruction in the startup path. flutter_controller_ = std::make_unique( frame.right - frame.left, frame.bottom - frame.top, project_); // Ensure that basic setup of the controller was successful. if (!flutter_controller_->engine() || !flutter_controller_->view()) { return false; } RegisterPlugins(flutter_controller_->engine()); SetChildContent(flutter_controller_->view()->GetNativeWindow()); flutter_controller_->engine()->SetNextFrameCallback([&]() { this->Show(); }); // Flutter can complete the first frame before the "show window" callback is // registered. The following call ensures a frame is pending to ensure the // window is shown. It is a no-op if the first frame hasn't completed yet. flutter_controller_->ForceRedraw(); return true; } void FlutterWindow::OnDestroy() { if (flutter_controller_) { flutter_controller_ = nullptr; } Win32Window::OnDestroy(); } LRESULT FlutterWindow::MessageHandler(HWND hwnd, UINT const message, WPARAM const wparam, LPARAM const lparam) noexcept { // Give Flutter, including plugins, an opportunity to handle window messages. if (flutter_controller_) { std::optional result = flutter_controller_->HandleTopLevelWindowProc(hwnd, message, wparam, lparam); if (result) { return *result; } } switch (message) { case WM_FONTCHANGE: flutter_controller_->engine()->ReloadSystemFonts(); break; } return Win32Window::MessageHandler(hwnd, message, wparam, lparam); } ================================================ FILE: windows/runner/flutter_window.h ================================================ #ifndef RUNNER_FLUTTER_WINDOW_H_ #define RUNNER_FLUTTER_WINDOW_H_ #include #include #include #include "win32_window.h" // A window that does nothing but host a Flutter view. class FlutterWindow : public Win32Window { public: // Creates a new FlutterWindow hosting a Flutter view running |project|. explicit FlutterWindow(const flutter::DartProject& project); virtual ~FlutterWindow(); protected: // Win32Window: bool OnCreate() override; void OnDestroy() override; LRESULT MessageHandler(HWND window, UINT const message, WPARAM const wparam, LPARAM const lparam) noexcept override; private: // The project to run. flutter::DartProject project_; // The Flutter instance hosted by this window. std::unique_ptr flutter_controller_; }; #endif // RUNNER_FLUTTER_WINDOW_H_ ================================================ FILE: windows/runner/main.cpp ================================================ #include #include #include #include "flutter_window.h" #include "utils.h" int APIENTRY wWinMain(_In_ HINSTANCE instance, _In_opt_ HINSTANCE prev, _In_ wchar_t *command_line, _In_ int show_command) { // Attach to console when present (e.g., 'flutter run') or create a // new console when running with a debugger. if (!::AttachConsole(ATTACH_PARENT_PROCESS) && ::IsDebuggerPresent()) { CreateAndAttachConsole(); } // Initialize COM, so that it is available for use in the library and/or // plugins. ::CoInitializeEx(nullptr, COINIT_APARTMENTTHREADED); flutter::DartProject project(L"data"); std::vector command_line_arguments = GetCommandLineArguments(); project.set_dart_entrypoint_arguments(std::move(command_line_arguments)); FlutterWindow window(project); Win32Window::Point origin(10, 10); Win32Window::Size size(1280, 720); if (!window.Create(L"Converter NOW", origin, size)) { return EXIT_FAILURE; } window.SetQuitOnClose(true); ::MSG msg; while (::GetMessage(&msg, nullptr, 0, 0)) { ::TranslateMessage(&msg); ::DispatchMessage(&msg); } ::CoUninitialize(); return EXIT_SUCCESS; } ================================================ FILE: windows/runner/resource.h ================================================ //{{NO_DEPENDENCIES}} // Microsoft Visual C++ generated include file. // Used by Runner.rc // #define IDI_APP_ICON 101 // Next default values for new objects // #ifdef APSTUDIO_INVOKED #ifndef APSTUDIO_READONLY_SYMBOLS #define _APS_NEXT_RESOURCE_VALUE 102 #define _APS_NEXT_COMMAND_VALUE 40001 #define _APS_NEXT_CONTROL_VALUE 1001 #define _APS_NEXT_SYMED_VALUE 101 #endif #endif ================================================ FILE: windows/runner/runner.exe.manifest ================================================ PerMonitorV2 ================================================ FILE: windows/runner/utils.cpp ================================================ #include "utils.h" #include #include #include #include #include void CreateAndAttachConsole() { if (::AllocConsole()) { FILE *unused; if (freopen_s(&unused, "CONOUT$", "w", stdout)) { _dup2(_fileno(stdout), 1); } if (freopen_s(&unused, "CONOUT$", "w", stderr)) { _dup2(_fileno(stdout), 2); } std::ios::sync_with_stdio(); FlutterDesktopResyncOutputStreams(); } } std::vector GetCommandLineArguments() { // Convert the UTF-16 command line arguments to UTF-8 for the Engine to use. int argc; wchar_t** argv = ::CommandLineToArgvW(::GetCommandLineW(), &argc); if (argv == nullptr) { return std::vector(); } std::vector command_line_arguments; // Skip the first argument as it's the binary name. for (int i = 1; i < argc; i++) { command_line_arguments.push_back(Utf8FromUtf16(argv[i])); } ::LocalFree(argv); return command_line_arguments; } std::string Utf8FromUtf16(const wchar_t* utf16_string) { if (utf16_string == nullptr) { return std::string(); } unsigned int target_length = ::WideCharToMultiByte( CP_UTF8, WC_ERR_INVALID_CHARS, utf16_string, -1, nullptr, 0, nullptr, nullptr) -1; // remove the trailing null character int input_length = (int)wcslen(utf16_string); std::string utf8_string; if (target_length == 0 || target_length > utf8_string.max_size()) { return utf8_string; } utf8_string.resize(target_length); int converted_length = ::WideCharToMultiByte( CP_UTF8, WC_ERR_INVALID_CHARS, utf16_string, input_length, utf8_string.data(), target_length, nullptr, nullptr); if (converted_length == 0) { return std::string(); } return utf8_string; } ================================================ FILE: windows/runner/utils.h ================================================ #ifndef RUNNER_UTILS_H_ #define RUNNER_UTILS_H_ #include #include // Creates a console for the process, and redirects stdout and stderr to // it for both the runner and the Flutter library. void CreateAndAttachConsole(); // Takes a null-terminated wchar_t* encoded in UTF-16 and returns a std::string // encoded in UTF-8. Returns an empty std::string on failure. std::string Utf8FromUtf16(const wchar_t* utf16_string); // Gets the command line arguments passed in as a std::vector, // encoded in UTF-8. Returns an empty std::vector on failure. std::vector GetCommandLineArguments(); #endif // RUNNER_UTILS_H_ ================================================ FILE: windows/runner/win32_window.cpp ================================================ #include "win32_window.h" #include #include #include "resource.h" namespace { /// Window attribute that enables dark mode window decorations. /// /// Redefined in case the developer's machine has a Windows SDK older than /// version 10.0.22000.0. /// See: https://docs.microsoft.com/windows/win32/api/dwmapi/ne-dwmapi-dwmwindowattribute #ifndef DWMWA_USE_IMMERSIVE_DARK_MODE #define DWMWA_USE_IMMERSIVE_DARK_MODE 20 #endif constexpr const wchar_t kWindowClassName[] = L"FLUTTER_RUNNER_WIN32_WINDOW"; /// Registry key for app theme preference. /// /// A value of 0 indicates apps should use dark mode. A non-zero or missing /// value indicates apps should use light mode. constexpr const wchar_t kGetPreferredBrightnessRegKey[] = L"Software\\Microsoft\\Windows\\CurrentVersion\\Themes\\Personalize"; constexpr const wchar_t kGetPreferredBrightnessRegValue[] = L"AppsUseLightTheme"; // The number of Win32Window objects that currently exist. static int g_active_window_count = 0; using EnableNonClientDpiScaling = BOOL __stdcall(HWND hwnd); // Scale helper to convert logical scaler values to physical using passed in // scale factor int Scale(int source, double scale_factor) { return static_cast(source * scale_factor); } // Dynamically loads the |EnableNonClientDpiScaling| from the User32 module. // This API is only needed for PerMonitor V1 awareness mode. void EnableFullDpiSupportIfAvailable(HWND hwnd) { HMODULE user32_module = LoadLibraryA("User32.dll"); if (!user32_module) { return; } auto enable_non_client_dpi_scaling = reinterpret_cast( GetProcAddress(user32_module, "EnableNonClientDpiScaling")); if (enable_non_client_dpi_scaling != nullptr) { enable_non_client_dpi_scaling(hwnd); } FreeLibrary(user32_module); } } // namespace // Manages the Win32Window's window class registration. class WindowClassRegistrar { public: ~WindowClassRegistrar() = default; // Returns the singleton registrar instance. static WindowClassRegistrar* GetInstance() { if (!instance_) { instance_ = new WindowClassRegistrar(); } return instance_; } // Returns the name of the window class, registering the class if it hasn't // previously been registered. const wchar_t* GetWindowClass(); // Unregisters the window class. Should only be called if there are no // instances of the window. void UnregisterWindowClass(); private: WindowClassRegistrar() = default; static WindowClassRegistrar* instance_; bool class_registered_ = false; }; WindowClassRegistrar* WindowClassRegistrar::instance_ = nullptr; const wchar_t* WindowClassRegistrar::GetWindowClass() { if (!class_registered_) { WNDCLASS window_class{}; window_class.hCursor = LoadCursor(nullptr, IDC_ARROW); window_class.lpszClassName = kWindowClassName; window_class.style = CS_HREDRAW | CS_VREDRAW; window_class.cbClsExtra = 0; window_class.cbWndExtra = 0; window_class.hInstance = GetModuleHandle(nullptr); window_class.hIcon = LoadIcon(window_class.hInstance, MAKEINTRESOURCE(IDI_APP_ICON)); window_class.hbrBackground = 0; window_class.lpszMenuName = nullptr; window_class.lpfnWndProc = Win32Window::WndProc; RegisterClass(&window_class); class_registered_ = true; } return kWindowClassName; } void WindowClassRegistrar::UnregisterWindowClass() { UnregisterClass(kWindowClassName, nullptr); class_registered_ = false; } Win32Window::Win32Window() { ++g_active_window_count; } Win32Window::~Win32Window() { --g_active_window_count; Destroy(); } bool Win32Window::Create(const std::wstring& title, const Point& origin, const Size& size) { Destroy(); const wchar_t* window_class = WindowClassRegistrar::GetInstance()->GetWindowClass(); const POINT target_point = {static_cast(origin.x), static_cast(origin.y)}; HMONITOR monitor = MonitorFromPoint(target_point, MONITOR_DEFAULTTONEAREST); UINT dpi = FlutterDesktopGetDpiForMonitor(monitor); double scale_factor = dpi / 96.0; HWND window = CreateWindow( window_class, title.c_str(), WS_OVERLAPPEDWINDOW, Scale(origin.x, scale_factor), Scale(origin.y, scale_factor), Scale(size.width, scale_factor), Scale(size.height, scale_factor), nullptr, nullptr, GetModuleHandle(nullptr), this); if (!window) { return false; } UpdateTheme(window); return OnCreate(); } bool Win32Window::Show() { return ShowWindow(window_handle_, SW_SHOWNORMAL); } // static LRESULT CALLBACK Win32Window::WndProc(HWND const window, UINT const message, WPARAM const wparam, LPARAM const lparam) noexcept { if (message == WM_NCCREATE) { auto window_struct = reinterpret_cast(lparam); SetWindowLongPtr(window, GWLP_USERDATA, reinterpret_cast(window_struct->lpCreateParams)); auto that = static_cast(window_struct->lpCreateParams); EnableFullDpiSupportIfAvailable(window); that->window_handle_ = window; } else if (Win32Window* that = GetThisFromHandle(window)) { return that->MessageHandler(window, message, wparam, lparam); } return DefWindowProc(window, message, wparam, lparam); } LRESULT Win32Window::MessageHandler(HWND hwnd, UINT const message, WPARAM const wparam, LPARAM const lparam) noexcept { switch (message) { case WM_DESTROY: window_handle_ = nullptr; Destroy(); if (quit_on_close_) { PostQuitMessage(0); } return 0; case WM_DPICHANGED: { auto newRectSize = reinterpret_cast(lparam); LONG newWidth = newRectSize->right - newRectSize->left; LONG newHeight = newRectSize->bottom - newRectSize->top; SetWindowPos(hwnd, nullptr, newRectSize->left, newRectSize->top, newWidth, newHeight, SWP_NOZORDER | SWP_NOACTIVATE); return 0; } case WM_SIZE: { RECT rect = GetClientArea(); if (child_content_ != nullptr) { // Size and position the child window. MoveWindow(child_content_, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, TRUE); } return 0; } case WM_ACTIVATE: if (child_content_ != nullptr) { SetFocus(child_content_); } return 0; case WM_DWMCOLORIZATIONCOLORCHANGED: UpdateTheme(hwnd); return 0; } return DefWindowProc(window_handle_, message, wparam, lparam); } void Win32Window::Destroy() { OnDestroy(); if (window_handle_) { DestroyWindow(window_handle_); window_handle_ = nullptr; } if (g_active_window_count == 0) { WindowClassRegistrar::GetInstance()->UnregisterWindowClass(); } } Win32Window* Win32Window::GetThisFromHandle(HWND const window) noexcept { return reinterpret_cast( GetWindowLongPtr(window, GWLP_USERDATA)); } void Win32Window::SetChildContent(HWND content) { child_content_ = content; SetParent(content, window_handle_); RECT frame = GetClientArea(); MoveWindow(content, frame.left, frame.top, frame.right - frame.left, frame.bottom - frame.top, true); SetFocus(child_content_); } RECT Win32Window::GetClientArea() { RECT frame; GetClientRect(window_handle_, &frame); return frame; } HWND Win32Window::GetHandle() { return window_handle_; } void Win32Window::SetQuitOnClose(bool quit_on_close) { quit_on_close_ = quit_on_close; } bool Win32Window::OnCreate() { // No-op; provided for subclasses. return true; } void Win32Window::OnDestroy() { // No-op; provided for subclasses. } void Win32Window::UpdateTheme(HWND const window) { DWORD light_mode; DWORD light_mode_size = sizeof(light_mode); LSTATUS result = RegGetValue(HKEY_CURRENT_USER, kGetPreferredBrightnessRegKey, kGetPreferredBrightnessRegValue, RRF_RT_REG_DWORD, nullptr, &light_mode, &light_mode_size); if (result == ERROR_SUCCESS) { BOOL enable_dark_mode = light_mode == 0; DwmSetWindowAttribute(window, DWMWA_USE_IMMERSIVE_DARK_MODE, &enable_dark_mode, sizeof(enable_dark_mode)); } } ================================================ FILE: windows/runner/win32_window.h ================================================ #ifndef RUNNER_WIN32_WINDOW_H_ #define RUNNER_WIN32_WINDOW_H_ #include #include #include #include // A class abstraction for a high DPI-aware Win32 Window. Intended to be // inherited from by classes that wish to specialize with custom // rendering and input handling class Win32Window { public: struct Point { unsigned int x; unsigned int y; Point(unsigned int x, unsigned int y) : x(x), y(y) {} }; struct Size { unsigned int width; unsigned int height; Size(unsigned int width, unsigned int height) : width(width), height(height) {} }; Win32Window(); virtual ~Win32Window(); // Creates a win32 window with |title| that is positioned and sized using // |origin| and |size|. New windows are created on the default monitor. Window // sizes are specified to the OS in physical pixels, hence to ensure a // consistent size this function will scale the inputted width and height as // as appropriate for the default monitor. The window is invisible until // |Show| is called. Returns true if the window was created successfully. bool Create(const std::wstring& title, const Point& origin, const Size& size); // Show the current window. Returns true if the window was successfully shown. bool Show(); // Release OS resources associated with window. void Destroy(); // Inserts |content| into the window tree. void SetChildContent(HWND content); // Returns the backing Window handle to enable clients to set icon and other // window properties. Returns nullptr if the window has been destroyed. HWND GetHandle(); // If true, closing this window will quit the application. void SetQuitOnClose(bool quit_on_close); // Return a RECT representing the bounds of the current client area. RECT GetClientArea(); protected: // Processes and route salient window messages for mouse handling, // size change and DPI. Delegates handling of these to member overloads that // inheriting classes can handle. virtual LRESULT MessageHandler(HWND window, UINT const message, WPARAM const wparam, LPARAM const lparam) noexcept; // Called when CreateAndShow is called, allowing subclass window-related // setup. Subclasses should return false if setup fails. virtual bool OnCreate(); // Called when Destroy is called. virtual void OnDestroy(); private: friend class WindowClassRegistrar; // OS callback called by message pump. Handles the WM_NCCREATE message which // is passed when the non-client area is being created and enables automatic // non-client DPI scaling so that the non-client area automatically // responds to changes in DPI. All other messages are handled by // MessageHandler. static LRESULT CALLBACK WndProc(HWND const window, UINT const message, WPARAM const wparam, LPARAM const lparam) noexcept; // Retrieves a class instance pointer for |window| static Win32Window* GetThisFromHandle(HWND const window) noexcept; // Update the window frame's theme to match the system theme. static void UpdateTheme(HWND const window); bool quit_on_close_ = false; // window handle for top level window. HWND window_handle_ = nullptr; // window handle for hosted content. HWND child_content_ = nullptr; }; #endif // RUNNER_WIN32_WINDOW_H_