Compare commits

..

57 commits

Author SHA1 Message Date
Thibault Deckers
30f96334eb Merge branch 'develop' 2025-03-25 23:38:23 +01:00
Thibault Deckers
160544926c Merge branch 'develop' 2025-03-16 19:44:30 +01:00
Thibault Deckers
12a89782cd Merge branch 'develop' 2025-03-11 00:33:19 +01:00
Thibault Deckers
ea496e5f52 Merge branch 'develop' 2025-03-07 20:11:42 +01:00
Thibault Deckers
d4d6b40be5 Merge branch 'develop' 2025-03-05 23:30:01 +01:00
Thibault Deckers
7bf59106f9 Merge branch 'develop' 2025-02-06 17:19:03 +01:00
Thibault Deckers
94ad4c01f4 Merge branch 'develop' 2025-01-13 10:45:04 +01:00
Thibault Deckers
6803dae438 Merge branch 'develop' 2025-01-05 16:35:42 +01:00
Thibault Deckers
3ebb6e8cbf Merge branch 'develop' 2024-12-19 21:56:08 +01:00
Thibault Deckers
c5f06baef1 Merge branch 'develop' 2024-12-11 21:51:37 +01:00
Thibault Deckers
1d7deac84d Merge branch 'develop' 2024-11-24 23:16:41 +01:00
Thibault Deckers
c62e68399c Merge branch 'develop' 2024-11-18 23:46:35 +01:00
Thibault Deckers
530bd1a8b0 Merge branch 'develop' 2024-10-30 23:14:23 +01:00
Thibault Deckers
1c1be8ba1c Merge branch 'develop' 2024-10-10 19:10:20 +02:00
Thibault Deckers
e6afda2bc9 Merge branch 'develop' 2024-10-09 01:04:19 +02:00
Thibault Deckers
35aeefe34b Merge branch 'develop' 2024-10-09 00:10:04 +02:00
Thibault Deckers
53b8e5b37b Merge branch 'develop' 2024-09-17 20:22:40 +02:00
Thibault Deckers
4688ba01f2 Merge branch 'develop' 2024-09-16 20:02:52 +02:00
Thibault Deckers
50416ad59a Merge branch 'develop' 2024-09-16 00:17:24 +02:00
Thibault Deckers
ce3afd87a0 Merge branch 'develop' 2024-09-16 00:14:08 +02:00
Thibault Deckers
5565ac08f7 Merge branch 'develop' 2024-09-01 01:32:32 +02:00
Thibault Deckers
00681cbaee Merge branch 'develop' 2024-08-07 22:01:05 +02:00
Thibault Deckers
9ea43d951c Merge branch 'develop' 2024-07-19 08:31:52 +02:00
Thibault Deckers
86b0d16ad1 Merge branch 'develop' 2024-07-18 23:59:02 +02:00
Thibault Deckers
d813a61b9b Merge branch 'develop' 2024-07-17 22:39:17 +02:00
Thibault Deckers
6d4c765613 Merge branch 'develop' 2024-07-11 19:03:40 +02:00
Thibault Deckers
a396635639 Merge branch 'develop' 2024-07-09 00:18:12 +02:00
Thibault Deckers
90fa60df69 Merge branch 'develop' 2024-06-17 21:40:55 +02:00
Thibault Deckers
ce11587482 Merge branch 'develop' 2024-06-11 23:11:21 +02:00
Thibault Deckers
07ac7b3fda Merge branch 'develop' 2024-05-03 00:43:58 +02:00
Thibault Deckers
645c199b33 Merge branch 'develop' 2024-05-01 18:03:08 +02:00
Thibault Deckers
ed75dba228 Merge branch 'develop' 2024-04-14 23:25:04 +02:00
Thibault Deckers
2619613ee5 Merge branch 'develop' 2024-04-02 00:01:32 +02:00
Thibault Deckers
82f070f8a1 Merge branch 'develop' 2024-04-01 23:10:42 +02:00
Thibault Deckers
42a869908c Merge branch 'develop' 2024-03-12 19:54:44 +01:00
Thibault Deckers
a02131c7c8 Merge branch 'develop' 2024-03-11 23:33:40 +01:00
Thibault Deckers
579e7a2db0 Merge branch 'develop' 2024-02-22 20:24:59 +01:00
Thibault Deckers
283a3eba60 Merge branch 'develop' 2024-02-07 22:30:36 +01:00
Thibault Deckers
6f7f70babe Merge branch 'develop' 2024-01-29 23:49:00 +01:00
Thibault Deckers
7cd170baf9 Merge branch 'develop' 2023-12-24 16:51:18 +01:00
Thibault Deckers
32fff626d2 Merge branch 'develop' 2023-12-21 18:57:11 +01:00
Thibault Deckers
5b6d7af0ac Merge branch 'develop' 2023-12-02 17:33:59 +01:00
Thibault Deckers
aa2b4c14e0 Merge branch 'develop' 2023-10-17 00:21:24 +02:00
Thibault Deckers
bf322c0aa9 Merge branch 'develop' 2023-09-25 00:38:11 +02:00
Thibault Deckers
1b87fb896d Merge branch 'develop' 2023-09-17 22:55:45 +02:00
Thibault Deckers
20f6d3f2a7 Merge branch 'develop' 2023-09-13 23:07:00 +02:00
Thibault Deckers
ffc6201e28 Merge branch 'develop' 2023-08-28 23:05:34 +02:00
Thibault Deckers
c4e06113b4 Merge branch 'develop' 2023-08-24 19:59:21 +02:00
Thibault Deckers
271809e189 Merge branch 'develop' 2023-08-22 23:43:28 +02:00
Thibault Deckers
fb8a97c5c6 Merge branch 'develop' 2023-08-21 00:20:29 +02:00
Thibault Deckers
cf64527c4b Merge branch 'develop' 2023-06-04 23:04:22 +02:00
Thibault Deckers
055d341a84 Merge branch 'develop' 2023-05-28 13:39:12 +02:00
Thibault Deckers
dc34fc6bc2 Merge branch 'develop' 2023-05-28 12:51:03 +02:00
Thibault Deckers
fa16430063 Merge branch 'develop' 2023-05-27 00:45:35 +02:00
Thibault Deckers
58e7adecde Merge branch 'develop' 2023-05-26 23:45:09 +02:00
Thibault Deckers
b8e8e3bfba Merge branch 'develop' 2023-05-26 23:16:42 +02:00
Thibault Deckers
79c5f5777b Merge branch 'develop' 2023-05-26 23:08:12 +02:00
395 changed files with 129404 additions and 7134 deletions

@ -1 +1 @@
Subproject commit d8a9f9a52e5af486f80d932e838ee93861ffd863
Subproject commit c23637390482d4cf9598c3ce3f2be31aa7332daf

View file

@ -17,11 +17,11 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Harden Runner
uses: step-security/harden-runner@0634a2670c59f64b4a01f0f96f84700a4088b9f0 # v2.12.0
uses: step-security/harden-runner@4d991eb9b905ef189e4c376166672c3f2f230481 # v2.11.0
with:
egress-policy: audit
- name: 'Checkout Repository'
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
- name: 'Dependency Review'
uses: actions/dependency-review-action@da24556b548a50705dd671f47852072ea4c105d9 # v4.7.1
uses: actions/dependency-review-action@3b139cfc5fae8b618d3eae3675e383bb1769c019 # v4.5.0

View file

@ -18,7 +18,7 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Harden Runner
uses: step-security/harden-runner@0634a2670c59f64b4a01f0f96f84700a4088b9f0 # v2.12.0
uses: step-security/harden-runner@4d991eb9b905ef189e4c376166672c3f2f230481 # v2.11.0
with:
egress-policy: audit
@ -28,9 +28,6 @@ jobs:
- name: Get Flutter packages
run: ./flutterw pub get
- name: Generate app localizations
run: ./flutterw gen-l10n
- name: Static analysis.
run: ./flutterw analyze
@ -55,14 +52,14 @@ jobs:
build-mode: manual
steps:
- name: Harden Runner
uses: step-security/harden-runner@0634a2670c59f64b4a01f0f96f84700a4088b9f0 # v2.12.0
uses: step-security/harden-runner@4d991eb9b905ef189e4c376166672c3f2f230481 # v2.11.0
with:
egress-policy: audit
# Building relies on the Android Gradle plugin,
# which requires a modern Java version (not the default one).
- name: Set up JDK for Android Gradle plugin
uses: actions/setup-java@c5195efecf7bdfc987ee8bae7a71cb8b11521c00 # v4.7.1
uses: actions/setup-java@3a4f6e1af504cf6a31855fa899c6aa5355ba6c12 # v4.7.0
with:
distribution: 'temurin'
java-version: '21'
@ -72,7 +69,7 @@ jobs:
# Initializes the CodeQL tools for scanning.
- name: Initialize CodeQL
uses: github/codeql-action/init@fca7ace96b7d713c7035871441bd52efbe39e27e # v3.28.19
uses: github/codeql-action/init@1b549b9259bda1cb5ddde3b41741a82a2d15a841 # v3.28.13
with:
languages: ${{ matrix.language }}
build-mode: ${{ matrix.build-mode }}
@ -86,6 +83,6 @@ jobs:
./flutterw build apk --profile -t lib/main_play.dart --flavor play
- name: Perform CodeQL Analysis
uses: github/codeql-action/analyze@fca7ace96b7d713c7035871441bd52efbe39e27e # v3.28.19
uses: github/codeql-action/analyze@1b549b9259bda1cb5ddde3b41741a82a2d15a841 # v3.28.13
with:
category: "/language:${{matrix.language}}"

View file

@ -18,14 +18,14 @@ jobs:
id-token: write
steps:
- name: Harden Runner
uses: step-security/harden-runner@0634a2670c59f64b4a01f0f96f84700a4088b9f0 # v2.12.0
uses: step-security/harden-runner@4d991eb9b905ef189e4c376166672c3f2f230481 # v2.11.0
with:
egress-policy: audit
# Building relies on the Android Gradle plugin,
# which requires a modern Java version (not the default one).
- name: Set up JDK for Android Gradle plugin
uses: actions/setup-java@c5195efecf7bdfc987ee8bae7a71cb8b11521c00 # v4.7.1
uses: actions/setup-java@3a4f6e1af504cf6a31855fa899c6aa5355ba6c12 # v4.7.0
with:
distribution: 'temurin'
java-version: '21'
@ -36,9 +36,6 @@ jobs:
- name: Get Flutter packages
run: ./flutterw pub get
- name: Generate app localizations
run: ./flutterw gen-l10n
- name: Update Flutter version file
run: scripts/update_flutter_version.sh
@ -78,7 +75,7 @@ jobs:
AVES_GOOGLE_API_KEY: ${{ secrets.AVES_GOOGLE_API_KEY }}
- name: Generate artifact attestation
uses: actions/attest-build-provenance@db473fddc028af60658334401dc6fa3ffd8669fd # v2.3.0
uses: actions/attest-build-provenance@c074443f1aee8d4aeeae555aebba3282517141b2 # v2.2.3
with:
subject-path: 'outputs/*'
@ -101,7 +98,7 @@ jobs:
runs-on: ubuntu-latest
steps:
- name: Harden Runner
uses: step-security/harden-runner@0634a2670c59f64b4a01f0f96f84700a4088b9f0 # v2.12.0
uses: step-security/harden-runner@4d991eb9b905ef189e4c376166672c3f2f230481 # v2.11.0
with:
egress-policy: audit
@ -109,7 +106,7 @@ jobs:
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
- name: Get appbundle from artifacts
uses: actions/download-artifact@d3f86a106a0bac45b974a628896c90dbdf5c8093 # v4.3.0
uses: actions/download-artifact@95815c38cf2ff2164869cbab79da8d1f422bc89e # v4.2.1
with:
name: appbundle

View file

@ -31,7 +31,7 @@ jobs:
steps:
- name: Harden Runner
uses: step-security/harden-runner@0634a2670c59f64b4a01f0f96f84700a4088b9f0 # v2.12.0
uses: step-security/harden-runner@4d991eb9b905ef189e4c376166672c3f2f230481 # v2.11.0
with:
egress-policy: audit
@ -41,7 +41,7 @@ jobs:
persist-credentials: false
- name: "Run analysis"
uses: ossf/scorecard-action@05b42c624433fc40578a4040d5cf5e36ddca8cde # v2.4.2
uses: ossf/scorecard-action@f49aabe0b5af0936a0987cfb85d86b75731b0186 # v2.4.1
with:
results_file: results.sarif
results_format: sarif
@ -71,6 +71,6 @@ jobs:
# Upload the results to GitHub's code scanning dashboard.
- name: "Upload to code-scanning"
uses: github/codeql-action/upload-sarif@fca7ace96b7d713c7035871441bd52efbe39e27e # v3.28.19
uses: github/codeql-action/upload-sarif@1b549b9259bda1cb5ddde3b41741a82a2d15a841 # v3.28.13
with:
sarif_file: results.sarif

3
.gitignore vendored
View file

@ -47,6 +47,3 @@ app.*.map.json
# screenshot generation
/test_driver/assets/screenshots/
/screenshots/
# generated files
/lib/l10ngen/app_localizations*

View file

@ -4,77 +4,6 @@ All notable changes to this project will be documented in this file.
## <a id="unreleased"></a>[Unreleased]
### Added
- Info: show matching dynamic albums
### Fixed
- crash when decoding some large thumbnails
## <a id="v1.13.2"></a>[v1.13.2] - 2025-06-02
### Changed
- downgraded Flutter to stable v3.27.4
- prevent display orientation flip when device rotation is locked
### Fixed
- moved file losing its extension and no longer being detected as media in some cases
- opening home when launching app as media picker
- removing groups with obsolete albums
- loading group custom covers
- crash when parsing some large media with trailing thumbnail
## <a id="v1.13.1"></a>[v1.13.1] - 2025-05-14
### Fixed
- albums: show groups to move/copy/export items
- albums: hide grouped albums containing hidden items only
## <a id="v1.13.0"></a>[v1.13.0] - 2025-05-12
### Added
- Albums: groups
- Collection: sort by storage path
- Search: week day filters
### Changed
- revert to Skia rendering engine
## <a id="v1.12.10"></a>[v1.12.10] - 2025-04-16
### Added
- Search: format filters
- Albums: sort by path
### Changed
- upgraded Flutter to stable v3.29.3
### Fixed
- region decoding failing to access decoder pool
## <a id="v1.12.9"></a>[v1.12.9] - 2025-04-06
### Added
- Kannada translation (thanks Chethan, Prasannakumar T Bhat)
### Changed
- enable Impeller rendering engine
### Fixed
- memory pressure during browsing
## <a id="v1.12.8"></a>[v1.12.8] - 2025-03-25
### Fixed

View file

@ -111,96 +111,17 @@ Some users have expressed the wish to financially support the project. Thanks!
## Project Setup
### Install dependencies
Before running or building the app, update the dependencies for the desired flavor:
```
scripts/apply_flavor_play.sh
# scripts/apply_flavor_play.sh
```
To build the project, create a file named `<app dir>/android/key.properties`. It should contain a reference to a keystore for app signing, and other necessary credentials. See [key_template.properties](https://github.com/deckerst/aves/blob/develop/android/key_template.properties) for the expected keys.
### To run the app:
To run the app:
```
./flutterw run -t lib/main_play.dart --flavor play
```
### To build the app:
creare file con le tue credenziali file.keystore
dove YOUR_ALIAS_NAME è il tuo unico alias name
e YOUR_ALIAS_PWD è la password del tuo alias
```sh
keytool -genkey -v -keystore file.keystore -alias YOUR_ALIAS_NAME -storepass YOUR_ALIAS_PWD -keypass YOUR_ALIAS_PWD -keyalg RSA -validity 36500
```
in questo caso ho inserito
```sh
cd android
keytool -genkey -v -keystore file.keystore -alias FabioMich66 -storepass Master66 -keypass Master66 -keyalg RSA -validity 36500
```
se non puoi eseguire keytool perchè non è nel path di sistema cercalo usando
```sh
cd /
sudo find -name keytool
```
compilare il file `<app dir>/android/key.properties`
```
nano android/key.properties
```
questi i miei dati utilizzando il format key_template.properties
```
storeFile=/Users/fabio/flutter_apps/aves/android/file.keystore
storePassword=Master66
keyAlias=FabioMich66
keyPassword=Master66
googleApiKey=<GOOGLE_API_KEY>
```
infine compilare l'apk
```
./flutterw build apk -t lib/main_play.dart --flavor play
# ./flutterw run -t lib/main_play.dart --flavor play
```
[Version badge]: https://img.shields.io/github/v/release/deckerst/aves?include_prereleases&sort=semver
[Build badge]: https://img.shields.io/github/actions/workflow/status/deckerst/aves/quality-check.yml?branch=develop
## Android studio
caricare il file da github selezionando le mnù a tendina File-New-project from Version Control
selezionare version control tipo: git
inserire URL di aves
https://github.com/deckerst/aves
flaggare shallow clone with history troncated 1 commits
aprire la console sulla dir aves appena creata e caricare le dipendenze
```
scripts/apply_flavor_izzy.sh
```
in settings - Languages and Framework - Dart inserire il path
```
/home/fabio/flutter/bin/cache/
```
e spuntare project aves
Edit configurations e aggiungere shell script con un nome x es izzi
poi flaggare script text e inserire
./flutterw run -t lib/main_izzy.dart --flavor izzy
la working directory sarà una cosa così
/home/fabio/StudioProjects/aves

View file

@ -9,11 +9,6 @@ analyzer:
# implicit-casts: false
# implicit-dynamic: false
# cf https://github.com/dart-lang/dart_style/wiki/Configuration
formatter:
page_width: 240
trailing_commas: preserve
linter:
rules:
# from 'flutter_lints', excluded

1
android/.gitignore vendored
View file

@ -7,7 +7,6 @@ gradle-wrapper.jar
GeneratedPluginRegistrant.java
.cxx/
.kotlin/
/build/
# Remember to never publicly share your keystore.
# See https://flutter.dev/to/reference-keystore

View file

@ -33,13 +33,13 @@ kotlin {
}
android {
namespace = 'deckers.thibault.aves'
compileSdk = 36
namespace 'deckers.thibault.aves'
compileSdk 35
defaultConfig {
applicationId packageName
minSdk flutter.minSdkVersion
targetSdk 36
targetSdk 35
versionCode flutter.versionCode
versionName flutter.versionName
manifestPlaceholders = [googleApiKey: keystoreProperties["googleApiKey"] ?: "<NONE>"]
@ -134,14 +134,14 @@ flutter {
repositories {
maven {
url = 'https://jitpack.io'
url 'https://jitpack.io'
content {
includeGroup "com.github.deckerst"
includeGroup "com.github.deckerst.mp4parser"
}
}
maven {
url = 'https://s3.amazonaws.com/repo.commonsware.com'
url 'https://s3.amazonaws.com/repo.commonsware.com'
content {
excludeGroupByRegex "com\\.github\\.deckerst.*"
}
@ -149,36 +149,36 @@ repositories {
}
dependencies {
implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.10.2'
implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.10.1'
implementation "androidx.appcompat:appcompat:1.7.1"
implementation 'androidx.core:core-ktx:1.16.0'
implementation 'androidx.lifecycle:lifecycle-process:2.9.1'
implementation "androidx.appcompat:appcompat:1.7.0"
implementation 'androidx.core:core-ktx:1.15.0'
implementation 'androidx.lifecycle:lifecycle-process:2.8.7'
implementation 'androidx.media:media:1.7.0'
implementation 'androidx.multidex:multidex:2.0.1'
implementation 'androidx.security:security-crypto:1.1.0-beta01'
implementation 'androidx.work:work-runtime-ktx:2.10.1'
implementation 'androidx.security:security-crypto:1.1.0-alpha06'
implementation 'androidx.work:work-runtime-ktx:2.10.0'
implementation 'com.commonsware.cwac:document:0.5.0'
implementation 'com.drewnoakes:metadata-extractor:2.19.0'
implementation "com.github.bumptech.glide:glide:$glide_version"
implementation 'com.google.android.material:material:1.12.0'
// SLF4J implementation for `mp4parser`
implementation 'org.slf4j:slf4j-simple:2.0.17'
implementation 'org.slf4j:slf4j-simple:2.0.16'
// forked, built by JitPack:
// - https://jitpack.io/p/deckerst/Android-TiffBitmapFactory
// - https://jitpack.io/p/deckerst/androidsvg
// - https://jitpack.io/p/deckerst/mp4parser
// - https://jitpack.io/p/deckerst/pixymeta-android
implementation 'com.github.deckerst:Android-TiffBitmapFactory:d6b2b0aa4f'
implementation 'com.github.deckerst:androidsvg:67db933051'
implementation 'com.github.deckerst.mp4parser:isoparser:c2898f1832'
implementation 'com.github.deckerst.mp4parser:muxer:c2898f1832'
implementation 'com.github.deckerst:pixymeta-android:cb1cdc932e'
implementation 'com.github.deckerst:Android-TiffBitmapFactory:3ed067f021'
implementation 'com.github.deckerst:androidsvg:cc9d59a88f'
implementation 'com.github.deckerst.mp4parser:isoparser:d5caf7a3dd'
implementation 'com.github.deckerst.mp4parser:muxer:d5caf7a3dd'
implementation 'com.github.deckerst:pixymeta-android:71eee77dc4'
implementation project(':exifinterface')
testImplementation 'org.junit.jupiter:junit-jupiter-engine:5.13.1'
testImplementation 'org.junit.jupiter:junit-jupiter-engine:5.11.4'
kapt 'androidx.annotation:annotation:1.9.1'
ksp "com.github.bumptech.glide:ksp:$glide_version"

View file

@ -329,6 +329,10 @@
<meta-data
android:name="flutterEmbedding"
android:value="2" />
<!--
Screenshot driver scenario is not supported by Impeller: "Compressed screenshots not supported for Impeller".
As of Flutter v3.29.2, switching pages with alpha transition yields artifacts when Impeller is enabled.
-->
<meta-data
android:name="io.flutter.embedding.android.EnableImpeller"
android:value="false" />

View file

@ -14,7 +14,6 @@ import androidx.work.ForegroundInfo
import androidx.work.WorkManager
import androidx.work.WorkerParameters
import app.loup.streams_channel.StreamsChannel
import deckers.thibault.aves.channel.calls.Coresult.Companion.safeSuspend
import deckers.thibault.aves.channel.calls.DeviceHandler
import deckers.thibault.aves.channel.calls.GeocodingHandler
import deckers.thibault.aves.channel.calls.MediaFetchObjectHandler
@ -45,12 +44,11 @@ class AnalysisWorker(context: Context, parameters: WorkerParameters) : Coroutine
private var backgroundChannel: MethodChannel? = null
override suspend fun doWork(): Result {
Log.i(LOG_TAG, "Start analysis worker $id")
defaultScope.launch {
// prevent ANR triggered by slow operations in main thread
createNotificationChannel()
setForeground(createForegroundInfo())
}.join()
}
suspendCoroutine { cont ->
workCont = cont
onStart()
@ -70,6 +68,7 @@ class AnalysisWorker(context: Context, parameters: WorkerParameters) : Coroutine
}
private fun onStart() {
Log.i(LOG_TAG, "Start analysis worker $id")
runBlocking {
FlutterUtils.initFlutterEngine(applicationContext, SHARED_PREFERENCES_KEY, PREF_CALLBACK_HANDLE_KEY) {
flutterEngine = it
@ -133,7 +132,12 @@ class AnalysisWorker(context: Context, parameters: WorkerParameters) : Coroutine
result.success(null)
}
"updateNotification" -> defaultScope.launch { safeSuspend(call, result, ::updateNotification) }
"updateNotification" -> {
val title = call.argument<String>("title")
val message = call.argument<String>("message")
setForegroundAsync(createForegroundInfo(title, message))
result.success(null)
}
"stop" -> {
workCont?.resume(null)
@ -176,22 +180,17 @@ class AnalysisWorker(context: Context, parameters: WorkerParameters) : Coroutine
.setContentIntent(openAppIntent)
.addAction(stopAction)
.build()
// from Android 14 (API 34), foreground service type is mandatory for long-running workers:
// https://developer.android.com/guide/background/persistent/how-to/long-running
return when {
Build.VERSION.SDK_INT >= 35 -> ForegroundInfo(NOTIFICATION_ID, notification, ServiceInfo.FOREGROUND_SERVICE_TYPE_MEDIA_PROCESSING)
Build.VERSION.SDK_INT == 34 -> ForegroundInfo(NOTIFICATION_ID, notification, ServiceInfo.FOREGROUND_SERVICE_TYPE_DATA_SYNC)
else -> ForegroundInfo(NOTIFICATION_ID, notification)
return if (Build.VERSION.SDK_INT == 34) {
// from Android 14 (API 34), foreground service type is mandatory for long-running workers:
// https://developer.android.com/guide/background/persistent/how-to/long-running
ForegroundInfo(NOTIFICATION_ID, notification, ServiceInfo.FOREGROUND_SERVICE_TYPE_DATA_SYNC)
} else if (Build.VERSION.SDK_INT >= 35) {
ForegroundInfo(NOTIFICATION_ID, notification, ServiceInfo.FOREGROUND_SERVICE_TYPE_MEDIA_PROCESSING)
} else {
ForegroundInfo(NOTIFICATION_ID, notification)
}
}
private suspend fun updateNotification(call: MethodCall, result: MethodChannel.Result) {
val title = call.argument<String>("title")
val message = call.argument<String>("message")
setForeground(createForegroundInfo(title, message))
result.success(null)
}
companion object {
private val LOG_TAG = LogUtils.createTag<AnalysisWorker>()
private const val BACKGROUND_CHANNEL = "deckers.thibault/aves/analysis_service_background"

View file

@ -2,7 +2,6 @@ package deckers.thibault.aves.channel.calls
import android.app.ActivityManager
import android.content.Context
import androidx.core.content.edit
import androidx.work.ExistingWorkPolicy
import androidx.work.OneTimeWorkRequestBuilder
import androidx.work.WorkInfo
@ -19,6 +18,7 @@ import kotlinx.coroutines.SupervisorJob
import kotlinx.coroutines.launch
import kotlinx.coroutines.runBlocking
class AnalysisHandler(private val activity: FlutterFragmentActivity, private val onAnalysisCompleted: () -> Unit) : MethodChannel.MethodCallHandler {
private val ioScope = CoroutineScope(SupervisorJob() + Dispatchers.IO)
@ -38,8 +38,9 @@ class AnalysisHandler(private val activity: FlutterFragmentActivity, private val
}
val preferences = activity.getSharedPreferences(AnalysisWorker.SHARED_PREFERENCES_KEY, Context.MODE_PRIVATE)
preferences.edit {
with(preferences.edit()) {
putLong(AnalysisWorker.PREF_CALLBACK_HANDLE_KEY, callbackHandle)
apply()
}
result.success(true)
}
@ -68,8 +69,9 @@ class AnalysisHandler(private val activity: FlutterFragmentActivity, private val
// work `Data` cannot occupy more than 10240 bytes when serialized
// so we save the possibly long list of entry IDs to shared preferences
val preferences = activity.getSharedPreferences(AnalysisWorker.SHARED_PREFERENCES_KEY, Context.MODE_PRIVATE)
preferences.edit {
with(preferences.edit()) {
putStringSet(AnalysisWorker.PREF_ENTRY_IDS_KEY, allEntryIds?.map { it.toString() }?.toSet())
apply()
}
val workData = workDataOf(

View file

@ -1,6 +1,5 @@
package deckers.thibault.aves.channel.calls
import android.annotation.SuppressLint
import android.app.LocaleConfig
import android.app.LocaleManager
import android.content.Context
@ -103,7 +102,6 @@ class DeviceHandler(private val context: Context) : MethodCallHandler {
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.UPSIDE_DOWN_CAKE) {
@SuppressLint("WrongConstant")
val lm = context.getSystemService(Context.LOCALE_SERVICE) as? LocaleManager
lm?.overrideLocaleConfig = LocaleConfig(LocaleList.forLanguageTags(locales.joinToString(",")))
}

View file

@ -311,7 +311,7 @@ class EmbeddedDataHandler(private val context: Context) : MethodCallHandler {
embeddedByteStream: InputStream,
embeddedByteLength: Long,
) {
val extension = extensionFor(mimeType, defaultExtension = null)
val extension = extensionFor(mimeType)
val targetFile = StorageUtils.createTempFile(context, extension).apply {
transferFrom(embeddedByteStream, embeddedByteLength)
}
@ -319,7 +319,7 @@ class EmbeddedDataHandler(private val context: Context) : MethodCallHandler {
val authority = "${context.applicationContext.packageName}.file_provider"
val uri = if (displayName != null) {
// add extension to ease type identification when sharing this content
val displayNameWithExtension = if (displayName.endsWith(extension, ignoreCase = true)) {
val displayNameWithExtension = if (extension == null || displayName.endsWith(extension, ignoreCase = true)) {
displayName
} else {
"$displayName$extension"

View file

@ -31,7 +31,7 @@ class GeocodingHandler(private val context: Context) : MethodCallHandler {
private fun getAddress(call: MethodCall, result: MethodChannel.Result) {
val latitude = call.argument<Number>("latitude")?.toDouble()
val longitude = call.argument<Number>("longitude")?.toDouble()
val localeLanguageTag = call.argument<String>("localeLanguageTag")
val localeString = call.argument<String>("locale")
val maxResults = call.argument<Int>("maxResults") ?: 1
if (latitude == null || longitude == null) {
result.error("getAddress-args", "missing arguments", null)
@ -43,8 +43,11 @@ class GeocodingHandler(private val context: Context) : MethodCallHandler {
return
}
geocoder = geocoder ?: if (localeLanguageTag != null) {
Geocoder(context, Locale.forLanguageTag(localeLanguageTag))
geocoder = geocoder ?: if (localeString != null) {
val split = localeString.split("_")
val language = split[0]
val country = if (split.size > 1) split[1] else ""
Geocoder(context, Locale(language, country))
} else {
Geocoder(context)
}

View file

@ -1,7 +1,6 @@
package deckers.thibault.aves.channel.calls
import android.content.Context
import androidx.core.content.edit
import deckers.thibault.aves.SearchSuggestionsProvider
import deckers.thibault.aves.channel.calls.Coresult.Companion.safe
import io.flutter.plugin.common.MethodCall
@ -30,8 +29,9 @@ class GlobalSearchHandler(private val context: Context) : MethodCallHandler {
}
val preferences = context.getSharedPreferences(SearchSuggestionsProvider.SHARED_PREFERENCES_KEY, Context.MODE_PRIVATE)
preferences.edit {
with(preferences.edit()) {
putLong(SearchSuggestionsProvider.CALLBACK_HANDLE_KEY, callbackHandle)
apply()
}
result.success(true)
}

View file

@ -1,8 +1,6 @@
package deckers.thibault.aves.channel.calls
import android.content.Context
import android.os.Handler
import android.os.Looper
import androidx.core.net.toUri
import com.bumptech.glide.Glide
import deckers.thibault.aves.channel.calls.Coresult.Companion.safe
@ -23,8 +21,7 @@ class MediaFetchObjectHandler(private val context: Context) : MethodCallHandler
override fun onMethodCall(call: MethodCall, result: MethodChannel.Result) {
when (call.method) {
"getEntry" -> ioScope.launch { safe(call, result, ::getEntry) }
"clearImageDiskCache" -> ioScope.launch { safe(call, result, ::clearImageDiskCache) }
"clearImageMemoryCache" -> ioScope.launch { safe(call, result, ::clearImageMemoryCache) }
"clearSizedThumbnailDiskCache" -> ioScope.launch { safe(call, result, ::clearSizedThumbnailDiskCache) }
else -> result.notImplemented()
}
}
@ -50,18 +47,11 @@ class MediaFetchObjectHandler(private val context: Context) : MethodCallHandler
})
}
private fun clearImageDiskCache(@Suppress("unused_parameter") call: MethodCall, result: MethodChannel.Result) {
private fun clearSizedThumbnailDiskCache(@Suppress("unused_parameter") call: MethodCall, result: MethodChannel.Result) {
Glide.get(context).clearDiskCache()
result.success(null)
}
private fun clearImageMemoryCache(@Suppress("unused_parameter") call: MethodCall, result: MethodChannel.Result) {
Handler(Looper.getMainLooper()).post {
Glide.get(context).clearMemory()
}
result.success(null)
}
companion object {
const val CHANNEL = "deckers.thibault/aves/media_fetch_object"
}

View file

@ -1007,7 +1007,7 @@ class MetadataFetchHandler(private val context: Context) : MethodCallHandler {
result.success(metadataMap)
}
// returns description from these fields (by precedence):
// return description from these fields (by precedence):
// - XMP / dc:description
// - IPTC / caption-abstract
// - Exif / UserComment
@ -1192,8 +1192,8 @@ class MetadataFetchHandler(private val context: Context) : MethodCallHandler {
result.success(null)
}
// returns XMP components
// returns an empty list if there is no XMP
// return XMP components
// return an empty list if there is no XMP
private fun getXmp(call: MethodCall, result: MethodChannel.Result) {
val mimeType = call.argument<String>("mimeType")
val uri = call.argument<String>("uri")?.toUri()

View file

@ -2,7 +2,6 @@ package deckers.thibault.aves.channel.calls
import android.content.Context
import android.content.SharedPreferences
import androidx.core.content.edit
import androidx.security.crypto.EncryptedSharedPreferences
import androidx.security.crypto.MasterKey
import deckers.thibault.aves.channel.calls.Coresult.Companion.safe
@ -46,7 +45,7 @@ class SecurityHandler(private val context: Context) : MethodCallHandler {
}
val preferences = getStore()
preferences.edit {
with(preferences.edit()) {
when (value) {
is Boolean -> putBoolean(key, value)
is Float -> putFloat(key, value)
@ -59,6 +58,7 @@ class SecurityHandler(private val context: Context) : MethodCallHandler {
return
}
}
apply()
}
result.success(true)
}

View file

@ -20,8 +20,6 @@ import deckers.thibault.aves.utils.MemoryUtils
import deckers.thibault.aves.utils.MimeTypes
import deckers.thibault.aves.utils.StorageUtils
import io.flutter.plugin.common.MethodChannel
import java.util.concurrent.locks.ReentrantLock
import kotlin.concurrent.withLock
import kotlin.math.max
import kotlin.math.roundToInt
@ -31,7 +29,11 @@ import kotlin.math.roundToInt
class RegionFetcher internal constructor(
private val context: Context,
) {
// returns decoded bytes in ARGB_8888, with trailer bytes:
private var lastDecoderRef: LastDecoderRef? = null
private val exportUris = HashMap<Pair<Uri, Int?>, Uri>()
// return decoded bytes in ARGB_8888, with trailer bytes:
// - width (int32)
// - height (int32)
fun fetch(
@ -61,12 +63,24 @@ class RegionFetcher internal constructor(
return
}
var currentDecoderRef = lastDecoderRef
if (currentDecoderRef != null && currentDecoderRef.requestKey != requestKey) {
currentDecoderRef = null
}
try {
val decoder = getOrCreateDecoder(context, uri, requestKey)
if (decoder == null) {
result.error("fetch-read-null", "failed to open file for mimeType=$mimeType uri=$uri regionRect=$regionRect", null)
return
if (currentDecoderRef == null) {
val newDecoder = StorageUtils.openInputStream(context, uri)?.use { input ->
BitmapRegionDecoderCompat.newInstance(input)
}
if (newDecoder == null) {
result.error("fetch-read-null", "failed to open file for mimeType=$mimeType uri=$uri regionRect=$regionRect", null)
return
}
currentDecoderRef = LastDecoderRef(requestKey, newDecoder)
}
val decoder = currentDecoderRef.decoder
lastDecoderRef = currentDecoderRef
// with raw images, the known image size may not match the decoded image size
// so we scale the requested region accordingly
@ -166,7 +180,7 @@ class RegionFetcher internal constructor(
}
}
private data class DecoderRef(
private data class LastDecoderRef(
val requestKey: Pair<Uri, Int?>,
val decoder: BitmapRegionDecoder,
)
@ -174,32 +188,5 @@ class RegionFetcher internal constructor(
companion object {
private val LOG_TAG = LogUtils.createTag<RegionFetcher>()
private val PREFERRED_CONFIG = Bitmap.Config.ARGB_8888
private const val DECODER_POOL_SIZE = 3
private val decoderPool = ArrayList<DecoderRef>()
private val exportUris = HashMap<Pair<Uri, Int?>, Uri>()
private val poolLock = ReentrantLock()
private fun getOrCreateDecoder(context: Context, uri: Uri, requestKey: Pair<Uri, Int?>): BitmapRegionDecoder? {
poolLock.withLock {
var decoderRef = decoderPool.firstOrNull { it.requestKey == requestKey }
if (decoderRef == null) {
val newDecoder = StorageUtils.openInputStream(context, uri)?.use { input ->
BitmapRegionDecoderCompat.newInstance(input)
}
if (newDecoder == null) {
return null
}
decoderRef = DecoderRef(requestKey, newDecoder)
} else {
decoderPool.remove(decoderRef)
}
decoderPool.add(0, decoderRef)
while (decoderPool.size > DECODER_POOL_SIZE) {
decoderPool.removeAt(decoderPool.size - 1)
}
return decoderRef.decoder
}
}
}
}

View file

@ -17,13 +17,13 @@ import deckers.thibault.aves.utils.BitmapUtils
import deckers.thibault.aves.utils.MemoryUtils
import deckers.thibault.aves.utils.StorageUtils
import io.flutter.plugin.common.MethodChannel
import java.util.concurrent.locks.ReentrantLock
import kotlin.concurrent.withLock
import kotlin.math.ceil
class SvgRegionFetcher internal constructor(
private val context: Context,
) {
private var lastSvgRef: LastSvgRef? = null
fun fetch(
uri: Uri,
sizeBytes: Long?,
@ -39,12 +39,32 @@ class SvgRegionFetcher internal constructor(
return
}
var currentSvgRef = lastSvgRef
if (currentSvgRef != null && currentSvgRef.uri != uri) {
currentSvgRef = null
}
try {
val svg = getOrCreateDecoder(context, uri)
if (svg == null) {
result.error("fetch-read-null", "failed to open file for uri=$uri regionRect=$regionRect", null)
return
if (currentSvgRef == null) {
val newSvg = StorageUtils.openInputStream(context, uri)?.use { input ->
try {
SVG.getFromInputStream(SVGParserBufferedInputStream(input))
} catch (ex: SVGParseException) {
result.error("fetch-parse", "failed to parse SVG for uri=$uri regionRect=$regionRect", null)
return
}
}
if (newSvg == null) {
result.error("fetch-read-null", "failed to open file for uri=$uri regionRect=$regionRect", null)
return
}
newSvg.normalizeSize()
currentSvgRef = LastSvgRef(uri, newSvg)
}
val svg = currentSvgRef.svg
lastSvgRef = currentSvgRef
// we scale the requested region accordingly to the viewbox size
val viewBox = svg.documentViewBox
@ -90,46 +110,17 @@ class SvgRegionFetcher internal constructor(
bitmap = Bitmap.createBitmap(bitmap, bleedX, bleedY, targetBitmapWidth, targetBitmapHeight)
val bytes = BitmapUtils.getRawBytes(bitmap, recycle = true)
result.success(bytes)
} catch (e: SVGParseException) {
result.error("fetch-parse", "failed to parse SVG for uri=$uri regionRect=$regionRect", null)
} catch (e: Exception) {
result.error("fetch-read-exception", "failed to initialize region decoder for uri=$uri regionRect=$regionRect", e.message)
}
}
private data class DecoderRef(
private data class LastSvgRef(
val uri: Uri,
val decoder: SVG,
val svg: SVG,
)
companion object {
private val PREFERRED_CONFIG = Bitmap.Config.ARGB_8888
private const val DECODER_POOL_SIZE = 3
private val decoderPool = ArrayList<DecoderRef>()
private val poolLock = ReentrantLock()
private fun getOrCreateDecoder(context: Context, uri: Uri): SVG? {
poolLock.withLock {
var decoderRef = decoderPool.firstOrNull { it.uri == uri }
if (decoderRef == null) {
val newDecoder = StorageUtils.openInputStream(context, uri)?.use { input ->
SVG.getFromInputStream(SVGParserBufferedInputStream(input))
}
if (newDecoder == null) {
return null
}
newDecoder.normalizeSize()
decoderRef = DecoderRef(uri, newDecoder)
} else {
decoderPool.remove(decoderRef)
}
decoderPool.add(0, decoderRef)
while (decoderPool.size > DECODER_POOL_SIZE) {
decoderPool.removeAt(decoderPool.size - 1)
}
return decoderRef.decoder
}
}
}
}

View file

@ -5,10 +5,8 @@ import android.graphics.Bitmap
import android.net.Uri
import android.os.Build
import android.provider.MediaStore
import android.util.Log
import android.util.Size
import androidx.annotation.RequiresApi
import androidx.core.graphics.scale
import androidx.core.net.toUri
import com.bumptech.glide.Glide
import com.bumptech.glide.load.DecodeFormat
@ -19,7 +17,6 @@ import deckers.thibault.aves.decoder.AvesAppGlideModule
import deckers.thibault.aves.decoder.MultiPageImage
import deckers.thibault.aves.utils.BitmapUtils
import deckers.thibault.aves.utils.BitmapUtils.applyExifOrientation
import deckers.thibault.aves.utils.LogUtils
import deckers.thibault.aves.utils.MimeTypes
import deckers.thibault.aves.utils.MimeTypes.SVG
import deckers.thibault.aves.utils.MimeTypes.isVideo
@ -28,8 +25,6 @@ import deckers.thibault.aves.utils.MimeTypes.needRotationAfterGlide
import deckers.thibault.aves.utils.StorageUtils
import deckers.thibault.aves.utils.UriUtils.tryParseId
import io.flutter.plugin.common.MethodChannel
import kotlin.math.min
import kotlin.math.roundToInt
class ThumbnailFetcher internal constructor(
private val context: Context,
@ -82,29 +77,6 @@ class ThumbnailFetcher internal constructor(
}
}
if (bitmap != null) {
if (bitmap.width > width && bitmap.height > height) {
val scalingFactor: Double = min(bitmap.width.toDouble() / width, bitmap.height.toDouble() / height)
val dstWidth = (bitmap.width / scalingFactor).roundToInt()
val dstHeight = (bitmap.height / scalingFactor).roundToInt()
Log.d(
LOG_TAG, "rescale thumbnail for mimeType=$mimeType uri=$uri width=$width height=$height" +
", with bitmap byteCount=${bitmap.byteCount} size=${bitmap.width}x${bitmap.height}" +
", to target=${dstWidth}x${dstHeight}"
)
bitmap = bitmap.scale(dstWidth, dstHeight)
}
if (bitmap.byteCount > BITMAP_SIZE_DANGER_THRESHOLD) {
result.error(
"getThumbnail-large", "thumbnail bitmap dangerously large" +
" for mimeType=$mimeType uri=$uri pageId=$pageId width=$width height=$height" +
", with bitmap byteCount=${bitmap.byteCount} size=${bitmap.width}x${bitmap.height} config=${bitmap.config?.name}", null
)
return
}
}
// do not recycle bitmaps fetched from `ContentResolver` or Glide as their lifecycle is unknown
val recycle = false
val bytes = BitmapUtils.getRawBytes(bitmap, recycle = recycle)
@ -172,9 +144,4 @@ class ThumbnailFetcher internal constructor(
Glide.with(context).clear(target)
}
}
companion object {
private val LOG_TAG = LogUtils.createTag<ThumbnailFetcher>()
private const val BITMAP_SIZE_DANGER_THRESHOLD = 20 * (1 shl 20) // MB
}
}

View file

@ -31,15 +31,9 @@ class MediaStoreChangeStreamHandler(private val context: Context) : EventChannel
init {
Log.i(LOG_TAG, "start listening to Media Store")
try {
context.contentResolver.apply {
registerContentObserver(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, true, contentObserver)
registerContentObserver(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, true, contentObserver)
}
} catch (e: SecurityException) {
// Trying to register an observer may yield a security exception with this message:
// "Failed to find provider media for user 0; expected to find a valid ContentProvider for this authority"
Log.w(LOG_TAG, "failed to register content observer", e)
context.contentResolver.apply {
registerContentObserver(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, true, contentObserver)
registerContentObserver(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, true, contentObserver)
}
}

View file

@ -2,7 +2,6 @@ package deckers.thibault.aves.decoder
import android.content.Context
import android.net.Uri
import android.text.format.Formatter
import android.util.Log
import com.bumptech.glide.Glide
import com.bumptech.glide.GlideBuilder
@ -11,17 +10,9 @@ import com.bumptech.glide.annotation.GlideModule
import com.bumptech.glide.load.DecodeFormat
import com.bumptech.glide.load.ImageHeaderParser
import com.bumptech.glide.load.engine.DiskCacheStrategy
import com.bumptech.glide.load.engine.bitmap_recycle.BitmapPoolAdapter
import com.bumptech.glide.load.engine.bitmap_recycle.LruArrayPool
import com.bumptech.glide.load.engine.bitmap_recycle.LruBitmapPool
import com.bumptech.glide.load.engine.cache.DiskCache
import com.bumptech.glide.load.engine.cache.InternalCacheDiskCacheFactory
import com.bumptech.glide.load.engine.cache.LruResourceCache
import com.bumptech.glide.load.engine.cache.MemorySizeCalculator
import com.bumptech.glide.load.resource.bitmap.ExifInterfaceImageHeaderParser
import com.bumptech.glide.module.AppGlideModule
import com.bumptech.glide.request.RequestOptions
import deckers.thibault.aves.utils.LogUtils
import deckers.thibault.aves.utils.MimeTypes
import deckers.thibault.aves.utils.MimeTypes.isVideo
import deckers.thibault.aves.utils.StorageUtils
@ -32,30 +23,6 @@ class AvesAppGlideModule : AppGlideModule() {
override fun applyOptions(context: Context, builder: GlideBuilder) {
// hide noisy warning (e.g. for images that can't be decoded)
builder.setLogLevel(Log.ERROR)
// sizing
val memorySizeCalculator = MemorySizeCalculator.Builder(context).build()
builder.setMemorySizeCalculator(memorySizeCalculator)
val size: Int = memorySizeCalculator.bitmapPoolSize
if (size > 0) {
builder.setBitmapPool(LruBitmapPool(size.toLong()))
} else {
builder.setBitmapPool(BitmapPoolAdapter())
}
builder.setArrayPool(LruArrayPool(memorySizeCalculator.arrayPoolSizeInBytes))
builder.setMemoryCache(LruResourceCache(memorySizeCalculator.memoryCacheSize.toLong()))
val diskCacheSize = DiskCache.Factory.DEFAULT_DISK_CACHE_SIZE
val internalCacheDiskCacheFactory = InternalCacheDiskCacheFactory(context, DiskCache.Factory.DEFAULT_DISK_CACHE_DIR, diskCacheSize.toLong())
builder.setDiskCache(internalCacheDiskCacheFactory)
fun toMb(bytes: Int) = Formatter.formatFileSize(context, bytes.toLong())
Log.d(
LOG_TAG, "Glide disk cache size=${toMb(diskCacheSize)}" +
", memory cache size=${toMb(memorySizeCalculator.memoryCacheSize)}" +
", bitmap pool size=${toMb(memorySizeCalculator.bitmapPoolSize)}" +
", array pool size=${toMb(memorySizeCalculator.arrayPoolSizeInBytes)}"
)
}
override fun registerComponents(context: Context, glide: Glide, registry: Registry) {
@ -67,8 +34,6 @@ class AvesAppGlideModule : AppGlideModule() {
override fun isManifestParsingEnabled(): Boolean = false
companion object {
private val LOG_TAG = LogUtils.createTag<AvesAppGlideModule>()
// request a fresh image with the highest quality format
val uncachedFullImageOptions = RequestOptions()
.format(DecodeFormat.PREFER_ARGB_8888)

View file

@ -81,12 +81,12 @@ object PixyMetaHelper {
output: OutputStream,
iptcDataList: List<FieldMap>?,
) {
val iptc: List<IPTCDataSet> = iptcDataList?.flatMap {
val iptc = iptcDataList?.flatMap {
val record = it["record"] as Int
val tag = it["tag"] as Int
val values = it["values"] as List<*>
values.map { data -> IPTCDataSet(IPTCRecord.fromRecordNumber(record), tag, data as ByteArray) }
} ?: ArrayList()
} ?: ArrayList<IPTCDataSet>()
Metadata.insertIPTC(input, output, iptc)
}

View file

@ -142,18 +142,16 @@ abstract class ImageProvider {
val oldFile = File(sourcePath)
if (oldFile.nameWithoutExtension != desiredNameWithoutExtension) {
val defaultExtension = oldFile.extension
oldFile.parent?.let { dir ->
val resolution = resolveTargetFileNameWithoutExtension(
contextWrapper = activity,
dir = dir,
desiredNameWithoutExtension = desiredNameWithoutExtension,
mimeType = mimeType,
defaultExtension = defaultExtension,
conflictStrategy = NameConflictStrategy.RENAME,
)
resolution.nameWithoutExtension?.let { targetNameWithoutExtension ->
val targetFileName = "$targetNameWithoutExtension${extensionFor(mimeType, defaultExtension)}"
val targetFileName = "$targetNameWithoutExtension${extensionFor(mimeType)}"
val newFile = File(dir, targetFileName)
if (oldFile != newFile) {
newFields = renameSingle(
@ -279,17 +277,11 @@ abstract class ImageProvider {
val page = if (sourceMimeType == MimeTypes.TIFF) pageId + 1 else pageId
desiredNameWithoutExtension += "_${page.toString().padStart(3, '0')}"
}
// there is no benefit providing input extension
// for known output MIME type
val defaultExtension = null
val resolution = resolveTargetFileNameWithoutExtension(
contextWrapper = activity,
dir = targetDir,
desiredNameWithoutExtension = desiredNameWithoutExtension,
mimeType = exportMimeType,
defaultExtension = defaultExtension,
conflictStrategy = nameConflictStrategy,
)
val targetNameWithoutExtension = resolution.nameWithoutExtension ?: return skippedFieldMap
@ -366,7 +358,6 @@ abstract class ImageProvider {
targetDir = targetDir,
targetDirDocFile = targetDirDocFile,
targetNameWithoutExtension = targetNameWithoutExtension,
defaultExtension = defaultExtension,
write = write,
)
@ -474,7 +465,6 @@ abstract class ImageProvider {
dir = targetDir,
desiredNameWithoutExtension = desiredNameWithoutExtension,
mimeType = captureMimeType,
defaultExtension = null,
conflictStrategy = nameConflictStrategy,
)
} catch (e: Exception) {
@ -581,14 +571,13 @@ abstract class ImageProvider {
dir: String,
desiredNameWithoutExtension: String,
mimeType: String,
defaultExtension: String?,
conflictStrategy: NameConflictStrategy,
): NameConflictResolution {
val sanitizedNameWithoutExtension = sanitizeDesiredFileName(desiredNameWithoutExtension)
var resolvedName: String? = sanitizedNameWithoutExtension
var replacementFile: File? = null
val extension = extensionFor(mimeType, defaultExtension)
val extension = extensionFor(mimeType)
val targetFile = File(dir, "$sanitizedNameWithoutExtension$extension")
when (conflictStrategy) {
NameConflictStrategy.RENAME -> {

View file

@ -557,7 +557,6 @@ class MediaStoreImageProvider : ImageProvider() {
toBin: Boolean,
): FieldMap {
val sourcePath = sourceFile?.path
val sourceExtension = sourceFile?.extension
val sourceDir = sourceFile?.parent?.let { ensureTrailingSeparator(it) }
if (sourceDir == targetDir && !(copy && nameConflictStrategy == NameConflictStrategy.RENAME)) {
// nothing to do unless it's a renamed copy
@ -570,7 +569,6 @@ class MediaStoreImageProvider : ImageProvider() {
dir = targetDir,
desiredNameWithoutExtension = desiredNameWithoutExtension,
mimeType = mimeType,
defaultExtension = sourceExtension,
conflictStrategy = nameConflictStrategy,
)
val targetNameWithoutExtension = resolution.nameWithoutExtension ?: return skippedFieldMap
@ -582,7 +580,6 @@ class MediaStoreImageProvider : ImageProvider() {
targetDir = targetDir,
targetDirDocFile = targetDirDocFile,
targetNameWithoutExtension = targetNameWithoutExtension,
defaultExtension = sourceExtension,
) { output: OutputStream ->
try {
sourceDocFile.copyTo(output)
@ -618,13 +615,12 @@ class MediaStoreImageProvider : ImageProvider() {
targetDir: String,
targetDirDocFile: DocumentFileCompat?,
targetNameWithoutExtension: String,
defaultExtension: String?,
write: (OutputStream) -> Unit,
): String {
if (StorageUtils.isInVault(activity, targetDir)) {
return insertByFile(
targetDir = targetDir,
targetFileName = "$targetNameWithoutExtension${extensionFor(mimeType, defaultExtension)}",
targetFileName = "$targetNameWithoutExtension${extensionFor(mimeType)}",
write = write,
)
}
@ -634,7 +630,7 @@ class MediaStoreImageProvider : ImageProvider() {
return insertByMediaStore(
activity = activity,
targetDir = targetDir,
targetFileName = "$targetNameWithoutExtension${extensionFor(mimeType, defaultExtension)}",
targetFileName = "$targetNameWithoutExtension${extensionFor(mimeType)}",
write = write,
)
}
@ -646,7 +642,6 @@ class MediaStoreImageProvider : ImageProvider() {
targetDir = targetDir,
targetDirDocFile = targetDirDocFile,
targetNameWithoutExtension = targetNameWithoutExtension,
defaultExtension = defaultExtension,
write = write,
)
}
@ -705,7 +700,6 @@ class MediaStoreImageProvider : ImageProvider() {
targetDir: String,
targetDirDocFile: DocumentFileCompat?,
targetNameWithoutExtension: String,
defaultExtension: String?,
write: (OutputStream) -> Unit,
): String {
targetDirDocFile ?: throw Exception("failed to get tree doc for directory at path=$targetDir")
@ -714,22 +708,8 @@ class MediaStoreImageProvider : ImageProvider() {
// but in order to open an output stream to it, we need to use a `SingleDocumentFile`
// through a document URI, not a tree URI
// note that `DocumentFile.getParentFile()` returns null if we did not pick a tree first
var targetTreeFile = targetDirDocFile.createFile(mimeType, targetNameWithoutExtension)
var targetDocFile = DocumentFileCompat.fromSingleUri(activity, targetTreeFile.uri)
// providing a display name and a MIME type does not guarantee
// that the created document will be backed by a file with a valid media extension,
// but having an extension is essential for media detection by Android,
// so we retry with a display name that includes the extension
if ((targetDocFile.extension == null || targetDocFile.extension.isEmpty() || targetDocFile.extension == "bin") && defaultExtension != null) {
if (targetDocFile.exists()) {
targetDocFile.delete()
}
val extension = if (defaultExtension.startsWith(".")) defaultExtension else ".$defaultExtension"
targetTreeFile = targetDirDocFile.createFile(mimeType, "$targetNameWithoutExtension$extension")
targetDocFile = DocumentFileCompat.fromSingleUri(activity, targetTreeFile.uri)
}
val targetTreeFile = targetDirDocFile.createFile(mimeType, targetNameWithoutExtension)
val targetDocFile = DocumentFileCompat.fromSingleUri(activity, targetTreeFile.uri)
try {
targetDocFile.openOutputStream().use(write)

View file

@ -5,5 +5,5 @@ import kotlin.math.pow
object MathUtils {
fun highestPowerOf2(x: Int): Int = highestPowerOf2(x.toDouble())
fun highestPowerOf2(x: Double): Int = if (x < 1) 0 else 2.toDouble().pow(log2(x).toInt()).toInt()
private fun highestPowerOf2(x: Double): Int = if (x < 1) 0 else 2.toDouble().pow(log2(x).toInt()).toInt()
}

View file

@ -163,24 +163,12 @@ object MimeTypes {
// among other refs:
// - https://android.googlesource.com/platform/external/mime-support/+/refs/heads/master/mime.types
fun extensionFor(mimeType: String, defaultExtension: String?): String = when (mimeType) {
fun extensionFor(mimeType: String): String? = when (mimeType) {
AVI, AVI_VND -> ".avi"
DNG, DNG_ADOBE -> ".dng"
HEIC, HEIF -> ".heif"
MP2T, MP2TS -> ".m2ts"
PSD_VND, PSD_X -> ".psd"
else -> {
val ext = MimeTypeMap.getSingleton().getExtensionFromMimeType(mimeType) ?: defaultExtension
if (ext != null) {
// fallback to provided extension when available,
// typically the original file extension when moving/renaming
if (ext.startsWith(".")) ext else ".$ext"
} else {
// fallback to generic extensions,
// as incorrect file extensions are better than none for media detection
if (isVideo(mimeType)) ".mp4" else ".jpg"
}
}
else -> MimeTypeMap.getSingleton().getExtensionFromMimeType(mimeType)?.let { ".$it" }
}
val TIFF_EXTENSION_PATTERN = Regex(".*\\.tiff?", RegexOption.IGNORE_CASE)

View file

@ -8,5 +8,4 @@
<string name="analysis_channel_name">Σάρωση πολυμέσων</string>
<string name="analysis_notification_default_title">Σάρωση στοιχείων</string>
<string name="analysis_notification_action_stop">Διακοπή</string>
<string name="map_shortcut_short_label">Χάρτης</string>
</resources>
</resources>

View file

@ -8,5 +8,4 @@
<string name="analysis_notification_action_stop">توقف کردن</string>
<string name="app_widget_label">قاب عکس</string>
<string name="app_name">Aves</string>
<string name="map_shortcut_short_label">نقشه</string>
</resources>
</resources>

View file

@ -1,6 +1,6 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<string name="app_name">Aves</string>
<string name="app_name">אייבז</string>
<string name="app_widget_label">מסגרת תמונה</string>
<string name="wallpaper">טפט</string>
<string name="search_shortcut_short_label">חיפוש</string>
@ -9,4 +9,4 @@
<string name="analysis_notification_default_title">סורק מדיה</string>
<string name="analysis_notification_action_stop">הפסק</string>
<string name="map_shortcut_short_label">מפה</string>
</resources>
</resources>

View file

@ -1,6 +1,6 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<string name="app_name">ಎವೀಸ್</string>
<string name="app_name">Aves</string>
<string name="app_widget_label">ಫೋಟೋ ಫ್ರೇಮ್</string>
<string name="wallpaper">ವಾಲ್ಪೇಪರ್</string>
<string name="videos_shortcut_short_label">ವೀಡಿಯೊಗಳು</string>
@ -8,5 +8,4 @@
<string name="analysis_notification_default_title">ಮೀಡಿಯಾ ಸ್ಕ್ಯಾನ್ ಮಾಡಲಾಗುತ್ತಿದೆ</string>
<string name="analysis_notification_action_stop">ನಿಲ್ಲಿಸಿ</string>
<string name="search_shortcut_short_label">ಹುಡುಕಿ</string>
<string name="map_shortcut_short_label">ನಕ್ಷೆ</string>
</resources>
</resources>

View file

@ -8,5 +8,4 @@
<string name="videos_shortcut_short_label">ဗီဒီယိုများ</string>
<string name="analysis_notification_default_title">မီဒီယာ ကိုစကင်ဖတ်နေသည်</string>
<string name="analysis_notification_action_stop">ရပ်ရန်</string>
<string name="map_shortcut_short_label">မြေပုံ</string>
</resources>
</resources>

View file

@ -1,4 +0,0 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<string name="app_name">एभस</string>
</resources>

View file

@ -1,4 +1,4 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<string name="search_shortcut_short_label">ସନ୍ଧାନ</string>
</resources>
</resources>

View file

@ -1,4 +0,0 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<string name="app_name">Aves</string>
</resources>

View file

@ -22,6 +22,7 @@ import static androidx.exifinterface.media.ExifInterfaceUtilsFork.convertToLongA
import static androidx.exifinterface.media.ExifInterfaceUtilsFork.copy;
import static androidx.exifinterface.media.ExifInterfaceUtilsFork.parseSubSeconds;
import static androidx.exifinterface.media.ExifInterfaceUtilsFork.startsWith;
import static java.lang.annotation.ElementType.TYPE_USE;
import static java.nio.ByteOrder.BIG_ENDIAN;
import static java.nio.ByteOrder.LITTLE_ENDIAN;
@ -90,7 +91,7 @@ import java.util.regex.Pattern;
import java.util.zip.CRC32;
/*
* Forked from 'androidx.exifinterface:exifinterface:1.4.1'
* Forked from 'androidx.exifinterface:exifinterface:1.4.0'
* Named differently to let ExifInterface be loaded as subdependency.
* cf https://maven.google.com/web/index.html?q=exifinterface#androidx.exifinterface:exifinterface
* cf https://github.com/androidx/androidx/tree/androidx-main/exifinterface/exifinterface/src/main/java/androidx/exifinterface/media
@ -138,12 +139,6 @@ public class ExifInterfaceFork {
// TLAD threshold for safer Exif attribute parsing
private static final int ATTRIBUTE_SIZE_DANGER_THRESHOLD = 3 * (1 << 20); // MB
// TLAD available heap size, to check allocations
private long getAvailableHeapSize() {
final Runtime runtime = Runtime.getRuntime();
return runtime.maxMemory() - (runtime.totalMemory() - runtime.freeMemory());
}
private static final String TAG = "ExifInterface";
private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG);
@ -4558,7 +4553,7 @@ public class ExifInterfaceFork {
&& (mXmpFromSeparateMarker != null || !containsTiff700Xmp))
|| (xmpHandling == XMP_HANDLING_PREFER_TIFF_700_IF_PRESENT
&& !containsTiff700Xmp)) {
mXmpFromSeparateMarker = value != null ? ExifAttribute.createByte(value) : null;
mXmpFromSeparateMarker = ExifAttribute.createByte(value);
return;
}
}
@ -6563,9 +6558,8 @@ public class ExifInterfaceFork {
// Exif data in WebP images (e.g.
// https://github.com/ImageMagick/ImageMagick/issues/3140)
if (startsWith(payload, IDENTIFIER_EXIF_APP1)) {
payload =
Arrays.copyOfRange(
payload, IDENTIFIER_EXIF_APP1.length, payload.length);
payload = Arrays.copyOfRange(payload, IDENTIFIER_EXIF_APP1.length,
payload.length);
}
// Save offset to EXIF data for handling thumbnail and attribute offsets.
@ -6728,11 +6722,8 @@ public class ExifInterfaceFork {
copy(dataInputStream, dataOutputStream, PNG_SIGNATURE.length);
boolean needToWriteExif = true;
// Either there's some XMP data to write, or it has been cleared locally but was present in
// the file when it was read (and so needs to be removed).
boolean needToHandleXmpChunk =
mXmpFromSeparateMarker != null || mFileOnDiskContainsSeparateXmpMarker;
while (needToWriteExif || needToHandleXmpChunk) {
boolean needToWriteXmp = mXmpFromSeparateMarker != null;
while (needToWriteExif || needToWriteXmp) {
int chunkLength = dataInputStream.readInt();
int chunkType = dataInputStream.readInt();
if (chunkType == PNG_CHUNK_TYPE_IHDR) {
@ -6747,7 +6738,7 @@ public class ExifInterfaceFork {
}
if (mXmpFromSeparateMarker != null && !mFileOnDiskContainsSeparateXmpMarker) {
writePngXmpItxtChunk(dataOutputStream);
needToHandleXmpChunk = false;
needToWriteXmp = false;
}
continue;
} else if (chunkType == PNG_CHUNK_TYPE_EXIF && needToWriteExif) {
@ -6755,25 +6746,10 @@ public class ExifInterfaceFork {
dataInputStream.skipFully(chunkLength + PNG_CHUNK_CRC_BYTE_LENGTH);
needToWriteExif = false;
continue;
} else if (chunkType == PNG_CHUNK_TYPE_ITXT
&& chunkLength >= PNG_ITXT_XMP_KEYWORD.length) {
// Read the 17 byte keyword and 5 expected null bytes.
byte[] keyword = new byte[PNG_ITXT_XMP_KEYWORD.length];
dataInputStream.readFully(keyword);
int remainingChunkBytes = chunkLength - keyword.length + PNG_CHUNK_CRC_BYTE_LENGTH;
if (Arrays.equals(keyword, PNG_ITXT_XMP_KEYWORD)) {
if (mXmpFromSeparateMarker != null) {
writePngXmpItxtChunk(dataOutputStream);
}
dataInputStream.skipFully(remainingChunkBytes);
needToHandleXmpChunk = false;
} else {
// This is a non-XMP iTXt chunk, so just copy it to the output and continue.
dataOutputStream.writeInt(chunkLength);
dataOutputStream.writeInt(chunkType);
dataOutputStream.write(keyword);
copy(dataInputStream, dataOutputStream, remainingChunkBytes);
}
} else if (chunkType == PNG_CHUNK_TYPE_ITXT && needToWriteXmp) {
writePngXmpItxtChunk(dataOutputStream);
dataInputStream.skipFully(chunkLength + PNG_CHUNK_CRC_BYTE_LENGTH);
needToWriteXmp = false;
continue;
}
dataOutputStream.writeInt(chunkLength);
@ -7560,13 +7536,6 @@ public class ExifInterfaceFork {
Log.d(TAG, "Invalid strip offset value");
return;
}
// TLAD start
if (bytesToSkip > getAvailableHeapSize()) {
throw new IOException("cannot allocate " + bytesToSkip + " bytes to skip to retrieve thumbnail");
}
// TLAD end
try {
in.skipFully(bytesToSkip);
} catch (EOFException e) {

View file

@ -31,7 +31,7 @@ import java.io.InputStream;
import java.io.OutputStream;
/*
* Forked from 'androidx.exifinterface:exifinterface:1.4.1'
* Forked from 'androidx.exifinterface:exifinterface:1.4.0-alpha01' on 2024/11/17
* Named differently to let ExifInterface be loaded as subdependency.
* cf https://github.com/androidx/androidx/tree/androidx-main/exifinterface/exifinterface/src/main/java/androidx/exifinterface/media
*/

View file

@ -18,10 +18,10 @@ pluginManagement {
plugins {
id("dev.flutter.flutter-plugin-loader") version "1.0.0"
id("com.android.application") version "8.10.1" apply false
id("org.jetbrains.kotlin.android") version "2.1.21" apply false
id("com.google.devtools.ksp") version "2.1.21-2.0.1" apply false
id("org.gradle.toolchains.foojay-resolver-convention") version "1.0.0"
id("com.android.application") version "8.8.1" apply false
id("org.jetbrains.kotlin.android") version "2.1.10" apply false
id("com.google.devtools.ksp") version "2.1.10-1.0.29" apply false
id("org.gradle.toolchains.foojay-resolver-convention") version "0.8.0"
}
include(":app")

View file

@ -1,35 +0,0 @@
Terms of Service
================
“Aves Gallery” is an open-source gallery and metadata explorer app allowing you to access and manage your local photos and videos.
The app is designed for legal, authorized and acceptable purposes.
Disclaimer
==========
The app is released “as-is”, without any warranty, responsibility or liability. Use of the app is at your own risk.
Privacy Policy
==============
The app does not collect any personal data. We never have access to your photos and videos. This also means that we cannot get them back for you if you delete them without backing them up.
Optionally, with your consent, the app accesses the inventory of installed apps to improve album display.
Optionally, with your consent, the app collects anonymous error and diagnostic data to improve the app quality. We use Firebase Crashlytics, and the anonymous data are stored on their servers. Please note that those are anonymous data, there is absolutely nothing personal about those data.
Contact
=======
Developer: Thibault Deckers
Email: gallery.aves@gmail.com
Website: https://github.com/deckerst/aves

View file

@ -2,4 +2,4 @@
<b>Navigation und Suche</b> ist ein wichtiger Bestandteil von <i>Aves</i>. Das Ziel besteht darin, dass Benutzer problemlos von Alben zu Fotos zu Tags zu Karten usw. wechseln können.
<i>Aves</i> integriert sich in Android (einschließlich Android TV) mit Funktionen wie <b>Widgets</b>, <b>App-Shortcuts</b>, <b>Bildschirmschoner</b> und der <b>globalen Suche</b> integrieren. Sie funktioniert auch als <b>Medienbetrachter und -Picker</b>.
<i>Aves</i> lässt sich mit Android mit Funktionen wie <b>App-Verknüpfungen</b> und <b>globaler Suche</b> integrieren. Es funktioniert auch als <b>Medienbetrachter und -auswahl</b>.

View file

@ -0,0 +1,4 @@
In v1.12.0:
- save your filtered collection as dynamic albums
- enjoy the app in Tamil, Bulgarian and Estonian
Full changelog available on GitHub

View file

@ -0,0 +1,4 @@
In v1.12.0:
- save your filtered collection as dynamic albums
- enjoy the app in Tamil, Bulgarian and Estonian
Full changelog available on GitHub

View file

@ -0,0 +1,3 @@
In v1.12.1:
- enjoy the app in Danish
Full changelog available on GitHub

View file

@ -0,0 +1,3 @@
In v1.12.1:
- enjoy the app in Danish
Full changelog available on GitHub

View file

@ -0,0 +1,3 @@
In v1.12.2:
- enjoy the app in Danish
Full changelog available on GitHub

View file

@ -0,0 +1,3 @@
In v1.12.2:
- enjoy the app in Danish
Full changelog available on GitHub

View file

@ -0,0 +1,3 @@
In v1.12.3:
- edit locations via GPX tracks
Full changelog available on GitHub

View file

@ -0,0 +1,3 @@
In v1.12.3:
- edit locations via GPX tracks
Full changelog available on GitHub

View file

@ -1,4 +1,4 @@
In v1.12.9:
In v1.12.4:
- play more kinds of motion photos
- enjoy the app in Galician and Kannada
- enjoy the app in Galician
Full changelog available on GitHub

View file

@ -1,4 +1,4 @@
In v1.12.9:
In v1.12.4:
- play more kinds of motion photos
- enjoy the app in Galician and Kannada
- enjoy the app in Galician
Full changelog available on GitHub

View file

@ -1,4 +1,4 @@
In v1.12.10:
In v1.12.5:
- play more kinds of motion photos
- enjoy the app in Galician and Kannada
- enjoy the app in Galician
Full changelog available on GitHub

View file

@ -1,4 +1,4 @@
In v1.12.10:
In v1.12.5:
- play more kinds of motion photos
- enjoy the app in Galician and Kannada
- enjoy the app in Galician
Full changelog available on GitHub

View file

@ -1,4 +0,0 @@
In v1.13.0:
- group albums
- filter by day of the week
Full changelog available on GitHub

View file

@ -1,4 +0,0 @@
In v1.13.0:
- group albums
- filter by day of the week
Full changelog available on GitHub

View file

@ -1,4 +0,0 @@
In v1.13.1:
- group albums
- filter by day of the week
Full changelog available on GitHub

View file

@ -1,4 +0,0 @@
In v1.13.1:
- group albums
- filter by day of the week
Full changelog available on GitHub

View file

@ -1,4 +0,0 @@
In v1.13.2:
- group albums
- filter by day of the week
Full changelog available on GitHub

View file

@ -1,4 +0,0 @@
In v1.13.2:
- group albums
- filter by day of the week
Full changelog available on GitHub

View file

@ -1,5 +1,5 @@
<i>Aves</i> יכול להתמודד עם כל מיני תמונות וסרטונים, כולל קובצי JPEG ו-MP4 הטיפוסיים שלך, אבל גם דברים אקזוטיים יותר כמו <b>TIFF מרובי עמודים, SVGs, AVI ישנים ועוד</b>! הוא סורק את אוסף המדיה שלך כדי לזהות <b>תמונות תנועה</b>, <b>פנורמות</b> (הידוע גם בתמונות פנורמיות), <b>סרטוני 360°</b>, וכן קבצי <b>GeoTIFF</b>.
<i>Aves</i> can handle all sorts of images and videos, including your typical JPEGs and MP4s, but also more exotic things like <b>multi-page TIFFs, SVGs, old AVIs and more</b>! It scans your media collection to identify <b>motion photos</b>, <b>panoramas</b> (aka photo spheres), <b>360° videos</b>, as well as <b>GeoTIFF</b> files.
<b>ניווט וחיפוש</b> הם חלק חשוב ב-<i>Aves</i>. המטרה היא שהמשתמשים יזרמו בקלות מאלבומים לתמונות לתגים למפות וכו'.
<b>Navigation and search</b> is an important part of <i>Aves</i>. The goal is for users to easily flow from albums to photos to tags to maps, etc.
<i>Aves</i> משתלב עם Android (כולל Android TV) עם תכונות כגון <b>ווידג'טים</b>, <b>קיצורי אפליקציות</b>, <b>שומר מסך</b> וטיפול ב<b>חיפוש גלובלי</b>. הוא פועל גם כ<b>מציג ובוחר מדיה</b>.
<i>Aves</i> integrates with Android (including Android TV) with features such as <b>widgets</b>, <b>app shortcuts</b>, <b>screen saver</b> and <b>global search</b> handling. It also works as a <b>media viewer and picker</b>.

View file

@ -1 +1 @@
סייר גלריה ומטא נתונים
Gallery and metadata explorer

View file

@ -1,5 +1,5 @@
<i>ಏವೀಸ್</i> ನಿಮ್ಮ JPEG ಗಳು ಮತ್ತು MP4 ಗಳನ್ನು ಒಳಗೊಂಡಂತೆ ಎಲ್ಲಾ ರೀತಿಯ ಚಿತ್ರಗಳು ಮತ್ತು ವೀಡಿಯೊಗಳನ್ನು ನಿಭಾಯಿಸಬಲ್ಲದು, ಅಲ್ಲದೆ ವಿಶಿಷ್ಟವಾದ <b>ಬಹು-ಪುಟ TIFFಗಳು, SVGಗಳು, ಹಳೆಯ AVIಗಳು ಮತ್ತು ಹಲವು ಪ್ರಕಾರಗಳನ್ನು ಕೂಡ ಬೆಂಬಲಿಸುತ್ತದೆ</b> ಇದು <b>ಚಲನೆಯ ಫೋಟೋಗಳು</b>, <b>ಪನೋರಮಾಗಳು</b> (ಫೋಟೋ ಗೋಳಗಳು) <b>360° ವೀಡಿಯೊಗಳು</b>, ಹಾಗೆಯೇ <b>GeoTIFF</b> ಕಡತಗಳನ್ನು ಗುರುತಿಸಲು ನಿಮ್ಮ ಮಾಧ್ಯಮ ಸಂಗ್ರಹವನ್ನು ಸ್ಕ್ಯಾನ್ ಮಾಡುತ್ತದೆ.
<i>Aves</i> can handle all sorts of images and videos, including your typical JPEGs and MP4s, but also more exotic things like <b>multi-page TIFFs, SVGs, old AVIs and more</b>! It scans your media collection to identify <b>motion photos</b>, <b>panoramas</b> (aka photo spheres), <b>360° videos</b>, as well as <b>GeoTIFF</b> files.
<b>ನ್ಯಾವಿಗೇಷನ್ ಮತ್ತು ಹುಡುಕಾಟ</b> <i>ಏವೀಸ್</i>ನ ಒಂದು ಪ್ರಮುಖ ಭಾಗವಾಗಿದೆ. ಬಳಕೆದಾರರು ಆಲ್ಬಮ್‌ಗಳಿಂದ ಫೋಟೋಗಳಿಂದ ಟ್ಯಾಗ್‌ಗಳಿಗೆ ನಕ್ಷೆಗಳಿಗೆ ಸುಲಭವಾಗಿ ಹರಿಯುವುದು ಗುರಿಯಾಗಿದೆ.
<b>Navigation and search</b> is an important part of <i>Aves</i>. The goal is for users to easily flow from albums to photos to tags to maps, etc.
<i>ಎವೀಸ್</i> ಆಂಡ್ರಾಯ್ಡ್ (ಟಿವಿ ಸೇರಿದಂತೆ) ನೊಂದಿಗೆ ಸಂಯೋಜಿಸುತ್ತದೆ, ಉದಾಹರಣೆಗೆ <b>ವಿಜೆಟ್‌ಗಳು</b>, <b>ಆ್ಯಪ್ ಶಾರ್ಟ್‌ಕಟ್‌ಗಳು</b>, <b>ಸ್ಕ್ರೀನ್ ಸೇವರ್</b> ಮತ್ತು <b>ಜಾಗತಿಕ ಹುಡುಕಾಟ</b> ನಿರ್ವಹಣೆ. ಇದು <b>ಮೀಡಿಯಾ ವೀಕ್ಷಕ ಮತ್ತು ಪಿಕ್ಕರ್</b> ಆಗಿಯೂ ಕಾರ್ಯನಿರ್ವಹಿಸುತ್ತದೆ.
<i>Aves</i> integrates with Android (including Android TV) with features such as <b>widgets</b>, <b>app shortcuts</b>, <b>screen saver</b> and <b>global search</b> handling. It also works as a <b>media viewer and picker</b>.

Binary file not shown.

Before

Width:  |  Height:  |  Size: 23 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 308 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 556 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 150 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 113 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 93 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 376 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 381 KiB

View file

@ -1 +1 @@
ಗ್ಯಾಲರಿ ಮತ್ತು ಮೆಟಾಡೇಟಾ ಎಕ್ಸ್‌ಪ್ಲೋರರ್
Gallery and metadata explorer

View file

@ -1,5 +0,0 @@
<i>Aves</i> can handle all sorts of images and videos, including your typical JPEGs and MP4s, but also more exotic things like <b>multi-page TIFFs, SVGs, old AVIs and more</b>! It scans your media collection to identify <b>motion photos</b>, <b>panoramas</b> (aka photo spheres), <b>360° videos</b>, as well as <b>GeoTIFF</b> files.
<b>Navigation and search</b> is an important part of <i>Aves</i>. The goal is for users to easily flow from albums to photos to tags to maps, etc.
<i>Aves</i> integrates with Android (including Android TV) with features such as <b>widgets</b>, <b>app shortcuts</b>, <b>screen saver</b> and <b>global search</b> handling. It also works as a <b>media viewer and picker</b>.

View file

@ -1 +0,0 @@
Gallery and metadata explorer

View file

@ -1,5 +0,0 @@
<i>Aves</i> can handle all sorts of images and videos, including your typical JPEGs and MP4s, but also more exotic things like <b>multi-page TIFFs, SVGs, old AVIs and more</b>! It scans your media collection to identify <b>motion photos</b>, <b>panoramas</b> (aka photo spheres), <b>360° videos</b>, as well as <b>GeoTIFF</b> files.
<b>Navigation and search</b> is an important part of <i>Aves</i>. The goal is for users to easily flow from albums to photos to tags to maps, etc.
<i>Aves</i> integrates with Android (including Android TV) with features such as <b>widgets</b>, <b>app shortcuts</b>, <b>screen saver</b> and <b>global search</b> handling. It also works as a <b>media viewer and picker</b>.

View file

@ -1 +0,0 @@
Gallery and metadata explorer

View file

@ -10,7 +10,7 @@ import 'package:flutter/foundation.dart';
import 'package:flutter/widgets.dart';
@immutable
class FullImage extends ImageProvider<FullImage> with EquatableMixin {
class UriImage extends ImageProvider<UriImage> with EquatableMixin {
final String uri, mimeType;
final int? pageId, rotationDegrees, sizeBytes;
final bool isFlipped, isAnimated;
@ -19,7 +19,7 @@ class FullImage extends ImageProvider<FullImage> with EquatableMixin {
@override
List<Object?> get props => [uri, pageId, rotationDegrees, isFlipped, isAnimated, scale];
const FullImage({
const UriImage({
required this.uri,
required this.mimeType,
required this.pageId,
@ -31,12 +31,12 @@ class FullImage extends ImageProvider<FullImage> with EquatableMixin {
});
@override
Future<FullImage> obtainKey(ImageConfiguration configuration) {
return SynchronousFuture<FullImage>(this);
Future<UriImage> obtainKey(ImageConfiguration configuration) {
return SynchronousFuture<UriImage>(this);
}
@override
ImageStreamCompleter loadImage(FullImage key, ImageDecoderCallback decode) {
ImageStreamCompleter loadImage(UriImage key, ImageDecoderCallback decode) {
final chunkEvents = StreamController<ImageChunkEvent>();
return MultiFrameImageStreamCompleter(
@ -59,11 +59,11 @@ class FullImage extends ImageProvider<FullImage> with EquatableMixin {
case MimeTypes.svg:
return false;
default:
return !isAnimated && !MimeTypes.isVideo(mimeType);
return !isAnimated;
}
}
Future<ui.Codec> _loadAsync(FullImage key, ImageDecoderCallback decode, StreamController<ImageChunkEvent> chunkEvents) async {
Future<ui.Codec> _loadAsync(UriImage key, ImageDecoderCallback decode, StreamController<ImageChunkEvent> chunkEvents) async {
assert(key == this);
final request = ImageRequest(

View file

@ -1,4 +1,6 @@
{
"filePickerDoNotShowHiddenFiles": "عدم إظهار الملفات المخفية",
"@filePickerDoNotShowHiddenFiles": {},
"tagPlaceholderPlace": "المكان",
"@tagPlaceholderPlace": {},
"tagPlaceholderCountry": "البلد",
@ -7,6 +9,12 @@
"@sourceViewerPageTitle": {},
"panoramaDisableSensorControl": "تعطيل التحكم في المستشعر",
"@panoramaDisableSensorControl": {},
"filePickerNoItems": "لا توجد عناصر",
"@filePickerNoItems": {},
"filePickerOpenFrom": "فتح من",
"@filePickerOpenFrom": {},
"filePickerShowHiddenFiles": "إظهار الملفات المخفية",
"@filePickerShowHiddenFiles": {},
"panoramaEnableSensorControl": "تمكين التحكم في المستشعر",
"@panoramaEnableSensorControl": {},
"saveTooltip": "حفظ",
@ -55,6 +63,8 @@
"@tagEditorSectionRecent": {},
"tagEditorSectionPlaceholders": "العناصر النائبة",
"@tagEditorSectionPlaceholders": {},
"filePickerUseThisFolder": "إستخدام هذا المجلد",
"@filePickerUseThisFolder": {},
"hideTooltip": "إخفاء",
"@hideTooltip": {},
"tagEditorPageAddTagTooltip": "إضافة علامة",
@ -493,6 +503,8 @@
"@viewerSetWallpaperButtonLabel": {},
"settingsVideoResumptionModeTile": "استئناف التشغيل",
"@settingsVideoResumptionModeTile": {},
"collectionGroupNone": "لا تجمع",
"@collectionGroupNone": {},
"searchRatingSectionTitle": "التقييمات",
"@searchRatingSectionTitle": {},
"vaultBinUsageDialogMessage": "تستخدم بعض الخزائن سلة المحذوفات.",
@ -599,7 +611,7 @@
"@settingsLanguagePageTitle": {},
"rootDirectoryDescription": "دليل الجذر",
"@rootDirectoryDescription": {},
"viewDialogGroupSectionTitle": "الأقسام",
"viewDialogGroupSectionTitle": "مجموعة",
"@viewDialogGroupSectionTitle": {},
"maxBrightnessAlways": "دائماً",
"@maxBrightnessAlways": {},
@ -1019,6 +1031,8 @@
"@entryActionSetAs": {},
"sortOrderLowestFirst": "الأدنى أولاً",
"@sortOrderLowestFirst": {},
"albumGroupNone": "لا تجمع",
"@albumGroupNone": {},
"statsTopStatesSectionTitle": "أهم الولايات",
"@statsTopStatesSectionTitle": {},
"settingsViewerQuickActionEditorDisplayedButtonsSectionTitle": "الأزرار المعروضة",
@ -1449,7 +1463,7 @@
"@binPageTitle": {},
"tagPlaceholderState": "الولاية",
"@tagPlaceholderState": {},
"sortByAlbumFileName": "حسب عنوان الألبوم والعنصر",
"sortByAlbumFileName": "حسب الألبوم واسم الملف",
"@sortByAlbumFileName": {},
"deleteMultiAlbumConfirmationDialogMessage": "{count, plural, =1{هل تريد حذف هذه الألبومات والعنصر الموجود فيها؟} other{احذف هذه الألبومات و {count} العناصر فيها؟}}",
"@deleteMultiAlbumConfirmationDialogMessage": {
@ -1594,33 +1608,5 @@
"editEntryLocationDialogTimeShift": "التحول الزمني",
"@editEntryLocationDialogTimeShift": {},
"removeEntryMetadataDialogAll": "الكل",
"@removeEntryMetadataDialogAll": {},
"sortByPath": "حسب المسار",
"@sortByPath": {},
"searchFormatSectionTitle": "التنسيقات",
"@searchFormatSectionTitle": {},
"chipActionGroup": "تغيير التجميع",
"@chipActionGroup": {},
"createButtonLabel": "خلق",
"@createButtonLabel": {},
"sectionNone": "لا يوجد أقسام",
"@sectionNone": {},
"chipActionCreateGroup": "إنشاء مجموعة",
"@chipActionCreateGroup": {},
"albumTierGroups": "المجموعات",
"@albumTierGroups": {},
"newGroupDialogTitle": "مجموعة جديدة",
"@newGroupDialogTitle": {},
"newGroupDialogNameLabel": "اسم المجموعة",
"@newGroupDialogNameLabel": {},
"groupAlreadyExists": "المجموعة موجودة بالفعل",
"@groupAlreadyExists": {},
"groupEmpty": "لا توجد مجموعات",
"@groupEmpty": {},
"ungrouped": "غير مجمعة",
"@ungrouped": {},
"groupPickerTitle": "اختر المجموعة",
"@groupPickerTitle": {},
"groupPickerUseThisGroupButton": "استخدم هذه المجموعة",
"@groupPickerUseThisGroupButton": {}
"@removeEntryMetadataDialogAll": {}
}

View file

@ -142,15 +142,5 @@
"chipActionUnpin": "Sabitləməyin",
"@chipActionUnpin": {},
"chipActionRename": "Bir də adlandır",
"@chipActionRename": {},
"chipActionDecompose": "Böl",
"@chipActionDecompose": {},
"chipActionCreateAlbum": "Albom yarat",
"@chipActionCreateAlbum": {},
"createButtonLabel": "YARAT",
"@createButtonLabel": {},
"chipActionGroup": "Qruplandırmanı dəyişdir",
"@chipActionGroup": {},
"chipActionCreateGroup": "Qrup yarat",
"@chipActionCreateGroup": {}
"@chipActionRename": {}
}

View file

@ -561,6 +561,14 @@
"@viewerInfoPageTitle": {},
"viewerErrorDoesNotExist": "Файл больш не існуе.",
"@viewerErrorDoesNotExist": {},
"filePickerUseThisFolder": "Выкарыстоўваць гэтую тэчку",
"@filePickerUseThisFolder": {},
"filePickerNoItems": "Няма элементаў",
"@filePickerNoItems": {},
"filePickerOpenFrom": "Адкрыць з",
"@filePickerOpenFrom": {},
"filePickerShowHiddenFiles": "Паказаць схаваныя файлы",
"@filePickerShowHiddenFiles": {},
"sourceViewerPageTitle": "Крыніца",
"@sourceViewerPageTitle": {},
"panoramaDisableSensorControl": "Адключыць сэнсарнае кіраванне",
@ -623,6 +631,8 @@
"@tagEditorDiscardDialogMessage": {},
"tagPlaceholderCountry": "Краіна",
"@tagPlaceholderCountry": {},
"filePickerDoNotShowHiddenFiles": "Не паказваць схаваныя файлы",
"@filePickerDoNotShowHiddenFiles": {},
"viewerInfoOpenEmbeddedFailureFeedback": "Не ўдалося атрымаць убудаваныя даныя",
"@viewerInfoOpenEmbeddedFailureFeedback": {},
"mapAttributionOsmHot": "Пліткі ад [HOT](https://www.hotosm.org/) • Арганізаваны [OSM France](https://openstreetmap.fr/)",
@ -917,6 +927,8 @@
"@settingsActionImport": {},
"locationPickerUseThisLocationButton": "Выкарыстоўваць гэтае месцазнаходжанне",
"@locationPickerUseThisLocationButton": {},
"collectionGroupNone": "Не групаваць",
"@collectionGroupNone": {},
"searchRatingSectionTitle": "Рэйтынгі",
"@searchRatingSectionTitle": {},
"settingsDisabled": "Адкл.",
@ -1031,6 +1043,8 @@
"@aboutLinkPolicy": {},
"sortOrderLowestFirst": "Спачатку з нізкім",
"@sortOrderLowestFirst": {},
"albumGroupNone": "Не групаваць",
"@albumGroupNone": {},
"countryPageTitle": "Краіны",
"@countryPageTitle": {},
"albumGroupType": "Па тыпу",

View file

@ -266,7 +266,7 @@
"@welcomeMessage": {},
"welcomeOptional": "Опционално",
"@welcomeOptional": {},
"itemCount": "{count, plural, =1{{count} елемент} few{{count} елемента} other{{count} елемента}}",
"itemCount": "{count, plural, =1{{count} обект} few{{count} обекта} other{{count} обекта}}",
"@itemCount": {
"placeholders": {
"count": {
@ -663,7 +663,7 @@
"@videoStartOverButtonLabel": {},
"videoResumeButtonLabel": "ПРОДЪЛЖИ",
"@videoResumeButtonLabel": {},
"setCoverDialogLatest": "Последен елемент",
"setCoverDialogLatest": "Последен обект",
"@setCoverDialogLatest": {},
"setCoverDialogCustom": "Персонален",
"@setCoverDialogCustom": {},
@ -725,7 +725,7 @@
"@exportEntryDialogWriteMetadata": {},
"renameEntryDialogLabel": "Ново име",
"@renameEntryDialogLabel": {},
"editEntryDialogCopyFromItem": "Копиране от друг елемент",
"editEntryDialogCopyFromItem": "Копиране от друг обект",
"@editEntryDialogCopyFromItem": {},
"editEntryDialogTargetFieldsHeader": "Полета за промяна",
"@editEntryDialogTargetFieldsHeader": {},
@ -873,6 +873,8 @@
"@collectionGroupMonth": {},
"collectionGroupDay": "По дни",
"@collectionGroupDay": {},
"collectionGroupNone": "Не групирай",
"@collectionGroupNone": {},
"sectionUnknown": "Неизвестно",
"@sectionUnknown": {},
"dateToday": "Днес",
@ -976,6 +978,8 @@
"@albumGroupType": {},
"albumGroupVolume": "По обем на съхранение",
"@albumGroupVolume": {},
"albumGroupNone": "Без групиране",
"@albumGroupNone": {},
"albumMimeTypeMixed": "Разни",
"@albumMimeTypeMixed": {},
"albumPickPageTitleCopy": "Копирай в албум",
@ -1318,7 +1322,7 @@
"@settingsStorageAccessTile": {},
"settingsAccessibilityShowPinchGestureAlternatives": "Показване на алтернативи за жестове с мултитъч",
"@settingsAccessibilityShowPinchGestureAlternatives": {},
"settingsDisplaySectionTitle": "Изглед",
"settingsDisplaySectionTitle": "Изобразяване",
"@settingsDisplaySectionTitle": {},
"settingsThemeBrightnessTile": "Тема",
"@settingsThemeBrightnessTile": {},
@ -1380,6 +1384,8 @@
"@tagPlaceholderState": {},
"tagPlaceholderPlace": "Локация",
"@tagPlaceholderPlace": {},
"filePickerShowHiddenFiles": "Показване на скритите файлове",
"@filePickerShowHiddenFiles": {},
"chipActionRemove": "Премахване",
"@chipActionRemove": {},
"albumTierDynamic": "Динамични",
@ -1448,7 +1454,7 @@
"@settingsAllowErrorReporting": {},
"settingsEnableBin": "Използвайте кошчето",
"@settingsEnableBin": {},
"settingsEnableBinSubtitle": "Съхранявайте изтритите елементи за 30 дни",
"settingsEnableBinSubtitle": "Съхранявайте изтритите обекти за 30 дни",
"@settingsEnableBinSubtitle": {},
"settingsAllowMediaManagement": "Разрешаване на управление на медиите",
"@settingsAllowMediaManagement": {},
@ -1565,6 +1571,10 @@
"@tagEditorSectionPlaceholders": {},
"tagPlaceholderCountry": "Държава",
"@tagPlaceholderCountry": {},
"filePickerDoNotShowHiddenFiles": "Не показвай скритите файлове",
"@filePickerDoNotShowHiddenFiles": {},
"filePickerOpenFrom": "Отворете от",
"@filePickerOpenFrom": {},
"settingsVideoGestureSideDoubleTapSeek": "Докоснете два пъти краищата на екрана, за превъртане назад/напред",
"@settingsVideoGestureSideDoubleTapSeek": {},
"settingsSaveSearchHistory": "Запазване на историята на търсенето",
@ -1607,10 +1617,14 @@
"@viewerErrorDoesNotExist": {},
"mapAttributionOsmData": "Данни карта © [OpenStreetMap](https://www.openstreetmap.org/copyright) участници",
"@mapAttributionOsmData": {},
"filePickerNoItems": "Не откривам нищо",
"@filePickerNoItems": {},
"newDynamicAlbumDialogTitle": "Нов динамичен албум",
"@newDynamicAlbumDialogTitle": {},
"tagEditorDiscardDialogMessage": "Искате ли да отхвърлите промените?",
"@tagEditorDiscardDialogMessage": {},
"filePickerUseThisFolder": "Използвай тази папка",
"@filePickerUseThisFolder": {},
"chipActionDecompose": "Раздели",
"@chipActionDecompose": {},
"coordinateFormatDdm": "Градуси, десетични минути",
@ -1620,23 +1634,5 @@
"editEntryLocationDialogTimeShift": "Изместване на времето",
"@editEntryLocationDialogTimeShift": {},
"removeEntryMetadataDialogAll": "Всички",
"@removeEntryMetadataDialogAll": {},
"sortByPath": "Според пътя",
"@sortByPath": {},
"searchFormatSectionTitle": "Формати",
"@searchFormatSectionTitle": {},
"chipActionCreateGroup": "Създайте група",
"@chipActionCreateGroup": {},
"chipActionGroup": "Групиране",
"@chipActionGroup": {},
"newGroupDialogTitle": "Нова Група",
"@newGroupDialogTitle": {},
"groupAlreadyExists": "Групата вече съществува",
"@groupAlreadyExists": {},
"albumTierGroups": "Групи",
"@albumTierGroups": {},
"groupPickerUseThisGroupButton": "Използвайте тази група",
"@groupPickerUseThisGroupButton": {},
"newGroupDialogNameLabel": "Име на групата",
"@newGroupDialogNameLabel": {}
"@removeEntryMetadataDialogAll": {}
}

View file

@ -839,6 +839,8 @@
"@collectionGroupAlbum": {},
"collectionGroupMonth": "Per mes",
"@collectionGroupMonth": {},
"collectionGroupNone": "No per grup",
"@collectionGroupNone": {},
"collectionGroupDay": "Per dia",
"@collectionGroupDay": {},
"dateToday": "Avui",
@ -947,6 +949,8 @@
"@albumGroupType": {},
"albumGroupVolume": "Per volum demmagatzematge",
"@albumGroupVolume": {},
"albumGroupNone": "No agrupar",
"@albumGroupNone": {},
"albumMimeTypeMixed": "Barrejat",
"@albumMimeTypeMixed": {},
"albumPickPageTitleCopy": "Copiar a Àlbum",
@ -1341,6 +1345,16 @@
"@panoramaDisableSensorControl": {},
"sourceViewerPageTitle": "Font",
"@sourceViewerPageTitle": {},
"filePickerShowHiddenFiles": "Mostra arxius amagats",
"@filePickerShowHiddenFiles": {},
"filePickerDoNotShowHiddenFiles": "No mostris arxius amagats",
"@filePickerDoNotShowHiddenFiles": {},
"filePickerOpenFrom": "Obrir des de",
"@filePickerOpenFrom": {},
"filePickerNoItems": "Sense element",
"@filePickerNoItems": {},
"filePickerUseThisFolder": "Utilitza aquesta carpeta",
"@filePickerUseThisFolder": {},
"settingsVideoControlsPageTitle": "Controls",
"@settingsVideoControlsPageTitle": {},
"settingsSubtitleThemeTextAlignmentDialogTitle": "Ajustament de Text",
@ -1594,9 +1608,5 @@
"dynamicAlbumAlreadyExists": "Làlbum dinàmic ja existeix",
"@dynamicAlbumAlreadyExists": {},
"sortOrderShortestFirst": "El més curt primer",
"@sortOrderShortestFirst": {},
"sortByPath": "Per ruta",
"@sortByPath": {},
"searchFormatSectionTitle": "Formats",
"@searchFormatSectionTitle": {}
"@sortOrderShortestFirst": {}
}

View file

@ -923,6 +923,16 @@
"@panoramaEnableSensorControl": {},
"sourceViewerPageTitle": "Zdroj",
"@sourceViewerPageTitle": {},
"filePickerShowHiddenFiles": "Zobrazit skryté soubory",
"@filePickerShowHiddenFiles": {},
"filePickerDoNotShowHiddenFiles": "Nezobrazovat skryté soubory",
"@filePickerDoNotShowHiddenFiles": {},
"filePickerOpenFrom": "Otevřít z",
"@filePickerOpenFrom": {},
"filePickerNoItems": "Žádné položky",
"@filePickerNoItems": {},
"filePickerUseThisFolder": "Použít tuto složku",
"@filePickerUseThisFolder": {},
"pickTooltip": "Vybrat",
"@pickTooltip": {},
"doubleBackExitMessage": "Pro ukončení klepněte znovu na „zpět“.",
@ -1031,6 +1041,8 @@
"@drawerCollectionSphericalVideos": {},
"aboutLicensesBanner": "Tato aplikace využívá tyto open-source baličky a knihovny.",
"@aboutLicensesBanner": {},
"collectionGroupNone": "Neseskupovat",
"@collectionGroupNone": {},
"aboutLicensesSectionTitle": "Licence open-source",
"@aboutLicensesSectionTitle": {},
"collectionActionHideTitleSearch": "Skrýt filtr dle názvu",
@ -1207,6 +1219,8 @@
"@sortOrderLargestFirst": {},
"sortOrderSmallestFirst": "Od nejužšího",
"@sortOrderSmallestFirst": {},
"albumGroupNone": "Neseskupovat",
"@albumGroupNone": {},
"albumVideoCaptures": "Snímky videa",
"@albumVideoCaptures": {},
"countryPageTitle": "Země",

View file

@ -555,6 +555,12 @@
"@tagEditorDiscardDialogMessage": {},
"panoramaEnableSensorControl": "Aktivér sensorstyring",
"@panoramaEnableSensorControl": {},
"filePickerOpenFrom": "Åbn fra",
"@filePickerOpenFrom": {},
"filePickerNoItems": "Ingen elementer",
"@filePickerNoItems": {},
"filePickerUseThisFolder": "Brug denne mappe",
"@filePickerUseThisFolder": {},
"authenticateToUnlockVault": "Godkend for at oplåse boks",
"@authenticateToUnlockVault": {},
"exportEntryDialogFormat": "Format:",
@ -835,6 +841,8 @@
"@collectionGroupMonth": {},
"collectionGroupDay": "Efter dag",
"@collectionGroupDay": {},
"collectionGroupNone": "Gruppér ikke",
"@collectionGroupNone": {},
"collectionDeleteFailureFeedback": "{count, plural, =1{Kunne ikke slette 1 element} other{Kunne ikke slette {count} elementer}}",
"@collectionDeleteFailureFeedback": {
"placeholders": {
@ -850,7 +858,7 @@
"@drawerCollectionRaws": {},
"sortByRating": "Efter bedømmelse",
"@sortByRating": {},
"sortByAlbumFileName": "Efter album og elementtitel",
"sortByAlbumFileName": "Efter album og filnavn",
"@sortByAlbumFileName": {},
"albumGroupVolume": "Efter lagervolume",
"@albumGroupVolume": {},
@ -1280,7 +1288,7 @@
"@tileLayoutGrid": {},
"removeEntryMetadataMotionPhotoXmpWarningDialogMessage": "XMP er påkrævet for at afspille videoen i et bevægelsesfoto.\n\nEr du sikker på, at du vil fjerne den?",
"@removeEntryMetadataMotionPhotoXmpWarningDialogMessage": {},
"viewDialogGroupSectionTitle": "Sektioner",
"viewDialogGroupSectionTitle": "Gruppér",
"@viewDialogGroupSectionTitle": {},
"hideFilterConfirmationDialogMessage": "Matchende fotos og videoer skjules fra din samling. Du kan vise dem igen i indstillingerne “Privatliv”.\n\nEr du sikker på, at du vil skjule dem?",
"@hideFilterConfirmationDialogMessage": {},
@ -1415,14 +1423,20 @@
"@mapAttributionOsmData": {},
"viewerInfoSearchSuggestionRights": "Rettigheder",
"@viewerInfoSearchSuggestionRights": {},
"filePickerShowHiddenFiles": "Vis skjulte filer",
"@filePickerShowHiddenFiles": {},
"viewerInfoOpenEmbeddedFailureFeedback": "Kunne ikke udtrække indlejrede data",
"@viewerInfoOpenEmbeddedFailureFeedback": {},
"tagPlaceholderPlace": "Sted",
"@tagPlaceholderPlace": {},
"filePickerDoNotShowHiddenFiles": "Vis ikke skjulte filer",
"@filePickerDoNotShowHiddenFiles": {},
"exportEntryDialogWriteMetadata": "Skriv metadata",
"@exportEntryDialogWriteMetadata": {},
"editEntryDateDialogCopyField": "Kopiér fra en anden dato",
"@editEntryDateDialogCopyField": {},
"albumGroupNone": "Gruppér ikke",
"@albumGroupNone": {},
"albumEmpty": "Ingen album",
"@albumEmpty": {},
"albumPageTitle": "Album",
@ -1538,115 +1552,5 @@
"settingsViewerQuickActionEmpty": "Ingen knapper",
"@settingsViewerQuickActionEmpty": {},
"chipActionFilterOut": "Filtrer ud",
"@chipActionFilterOut": {},
"mapStyleOsmHot": "Humanitært OSM",
"@mapStyleOsmHot": {},
"collectionDeselectSectionTooltip": "Fravælg sektion",
"@collectionDeselectSectionTooltip": {},
"editEntryLocationDialogImportGpx": "Importér GPX",
"@editEntryLocationDialogImportGpx": {},
"editEntryLocationDialogTimeShift": "Tidsskift",
"@editEntryLocationDialogTimeShift": {},
"videoStreamSelectionDialogTrack": "Spor",
"@videoStreamSelectionDialogTrack": {},
"albumGroupTier": "Efter kategori",
"@albumGroupTier": {},
"settingsVideoEnableHardwareAcceleration": "Hardwareacceleration",
"@settingsVideoEnableHardwareAcceleration": {},
"settingsViewerSectionTitle": "Fremviser",
"@settingsViewerSectionTitle": {},
"openMapPageTooltip": "Se på kortside",
"@openMapPageTooltip": {},
"settingsCollectionBurstPatternsTile": "Filnavnmønstre",
"@settingsCollectionBurstPatternsTile": {},
"wallpaperUseScrollEffect": "Brug rulleeffekt på startside",
"@wallpaperUseScrollEffect": {},
"editEntryDateDialogSourceFileModifiedDate": "Filens ændringsdato",
"@editEntryDateDialogSourceFileModifiedDate": {},
"editEntryDateDialogShift": "Skift",
"@editEntryDateDialogShift": {},
"chipActionDecompose": "Split",
"@chipActionDecompose": {},
"coordinateFormatDdm": "DDM",
"@coordinateFormatDdm": {},
"videoActionShowNextFrame": "Vis næste frame",
"@videoActionShowNextFrame": {},
"mapStyleStamenWatercolor": "Stamen Watercolor",
"@mapStyleStamenWatercolor": {},
"drawerCollectionAll": "Alle samlinger",
"@drawerCollectionAll": {},
"settingsThumbnailShowVideoDuration": "Vis videovarighed",
"@settingsThumbnailShowVideoDuration": {},
"settingsThemeColorHighlights": "Farvemarkeringer",
"@settingsThemeColorHighlights": {},
"viewerInfoSearchEmpty": "Ingen matchende nøgler",
"@viewerInfoSearchEmpty": {},
"removeEntryMetadataDialogAll": "Alle",
"@removeEntryMetadataDialogAll": {},
"aboutCreditsSectionTitle": "Kreditering",
"@aboutCreditsSectionTitle": {},
"settingsCollectionQuickActionTabSelecting": "Valg",
"@settingsCollectionQuickActionTabSelecting": {},
"settingsCollectionQuickActionTabBrowsing": "Browsing",
"@settingsCollectionQuickActionTabBrowsing": {},
"settingsViewerQuickActionEditorBanner": "Tryk og hold for at flytte knapper og vælge, hvilke handlinger der vises i fremviseren.",
"@settingsViewerQuickActionEditorBanner": {},
"settingsAllowInstalledAppAccess": "Tillad adgang til app-lager",
"@settingsAllowInstalledAppAccess": {},
"viewerInfoBackToViewerTooltip": "Tilbage til fremviser",
"@viewerInfoBackToViewerTooltip": {},
"videoActionShowPreviousFrame": "Vis forrige frame",
"@videoActionShowPreviousFrame": {},
"collectionSelectSectionTooltip": "Vælg sektion",
"@collectionSelectSectionTooltip": {},
"videoStreamSelectionDialogNoSelection": "Der er ingen andre spor.",
"@videoStreamSelectionDialogNoSelection": {},
"addShortcutDialogLabel": "Genvejsetiket",
"@addShortcutDialogLabel": {},
"moveUndatedConfirmationDialogMessage": "Gem elementdatoer, før du fortsætter?",
"@moveUndatedConfirmationDialogMessage": {},
"albumMimeTypeMixed": "Blandet",
"@albumMimeTypeMixed": {},
"videoActionCaptureFrame": "Tag billede af frame",
"@videoActionCaptureFrame": {},
"videoActionSelectStreams": "Vælg spor",
"@videoActionSelectStreams": {},
"videoActionABRepeat": "A-B gentagelse",
"@videoActionABRepeat": {},
"viewerActionLock": "Lås fremviser",
"@viewerActionLock": {},
"viewerActionUnlock": "Oplås fremviser",
"@viewerActionUnlock": {},
"keepScreenOnViewerOnly": "Kun fremvisningsside",
"@keepScreenOnViewerOnly": {},
"widgetOpenPageViewer": "Åbn fremviser",
"@widgetOpenPageViewer": {},
"sortByPath": "Efter sti",
"@sortByPath": {},
"searchFormatSectionTitle": "Formater",
"@searchFormatSectionTitle": {},
"createButtonLabel": "OPRET",
"@createButtonLabel": {},
"chipActionGroup": "Ændr gruppering",
"@chipActionGroup": {},
"chipActionCreateGroup": "Opret gruppe",
"@chipActionCreateGroup": {},
"albumTierGroups": "Grupper",
"@albumTierGroups": {},
"newGroupDialogTitle": "Ny gruppe",
"@newGroupDialogTitle": {},
"newGroupDialogNameLabel": "Gruppenavn",
"@newGroupDialogNameLabel": {},
"groupAlreadyExists": "Gruppen findes allerede",
"@groupAlreadyExists": {},
"groupEmpty": "Ingen grupper",
"@groupEmpty": {},
"groupPickerTitle": "Vælg gruppe",
"@groupPickerTitle": {},
"groupPickerUseThisGroupButton": "Brug denne gruppe",
"@groupPickerUseThisGroupButton": {},
"ungrouped": "Gruppe opløst",
"@ungrouped": {},
"sectionNone": "Ingen sektioner",
"@sectionNone": {}
"@chipActionFilterOut": {}
}

View file

@ -463,7 +463,7 @@
"@menuActionStats": {},
"viewDialogSortSectionTitle": "Sortieren",
"@viewDialogSortSectionTitle": {},
"viewDialogGroupSectionTitle": "Abschnitte",
"viewDialogGroupSectionTitle": "Gruppe",
"@viewDialogGroupSectionTitle": {},
"viewDialogLayoutSectionTitle": "Layout",
"@viewDialogLayoutSectionTitle": {},
@ -557,6 +557,8 @@
"@collectionGroupMonth": {},
"collectionGroupDay": "Nach Tag",
"@collectionGroupDay": {},
"collectionGroupNone": "Nicht gruppieren",
"@collectionGroupNone": {},
"sectionUnknown": "Unbekannt",
"@sectionUnknown": {},
"dateToday": "Heute",
@ -659,6 +661,8 @@
"@albumGroupType": {},
"albumGroupVolume": "Nach Speichervolumen",
"@albumGroupVolume": {},
"albumGroupNone": "Nicht gruppieren",
"@albumGroupNone": {},
"albumMimeTypeMixed": "Gemischt",
"@albumMimeTypeMixed": {},
"albumPickPageTitleCopy": "In Album kopieren",
@ -1107,6 +1111,16 @@
"@panoramaDisableSensorControl": {},
"sourceViewerPageTitle": "Quelle",
"@sourceViewerPageTitle": {},
"filePickerShowHiddenFiles": "Versteckte Dateien anzeigen",
"@filePickerShowHiddenFiles": {},
"filePickerDoNotShowHiddenFiles": "Versteckte Dateien nicht anzeigen",
"@filePickerDoNotShowHiddenFiles": {},
"filePickerOpenFrom": "Öffnen von",
"@filePickerOpenFrom": {},
"filePickerNoItems": "Keine Elemente",
"@filePickerNoItems": {},
"filePickerUseThisFolder": "Diesen Ordner verwenden",
"@filePickerUseThisFolder": {},
"widgetOpenPageCollection": "Sammlung öffnen",
"@widgetOpenPageCollection": {},
"durationDialogSeconds": "Sekunden",
@ -1396,39 +1410,5 @@
"chipActionDecompose": "Aufschlüsseln",
"@chipActionDecompose": {},
"editEntryLocationDialogImportGpx": "GPX importieren",
"@editEntryLocationDialogImportGpx": {},
"editEntryLocationDialogTimeShift": "Zeitverschiebung",
"@editEntryLocationDialogTimeShift": {},
"removeEntryMetadataDialogAll": "Alle",
"@removeEntryMetadataDialogAll": {},
"searchFormatSectionTitle": "Formate",
"@searchFormatSectionTitle": {},
"sortByPath": "Nach Pfad",
"@sortByPath": {},
"groupEmpty": "Keine Gruppen",
"@groupEmpty": {},
"newGroupDialogTitle": "Neue Gruppe",
"@newGroupDialogTitle": {},
"coordinateFormatDdm": "DDM",
"@coordinateFormatDdm": {},
"createButtonLabel": "ERSTELLEN",
"@createButtonLabel": {},
"chipActionGroup": "Gruppe",
"@chipActionGroup": {},
"chipActionCreateGroup": "Gruppe erstellen",
"@chipActionCreateGroup": {},
"albumTierGroups": "Gruppen",
"@albumTierGroups": {},
"sectionNone": "Keine Abschnitte",
"@sectionNone": {},
"newGroupDialogNameLabel": "Gruppenname",
"@newGroupDialogNameLabel": {},
"groupAlreadyExists": "Gruppe existiert bereits",
"@groupAlreadyExists": {},
"ungrouped": "Nicht gruppiert",
"@ungrouped": {},
"groupPickerTitle": "Gruppe auswählen",
"@groupPickerTitle": {},
"groupPickerUseThisGroupButton": "Diese Gruppe verwenden",
"@groupPickerUseThisGroupButton": {}
"@editEntryLocationDialogImportGpx": {}
}

View file

@ -377,9 +377,9 @@
"@renameProcessorCounter": {},
"renameProcessorName": "Όνομα",
"@renameProcessorName": {},
"deleteSingleAlbumConfirmationDialogMessage": "{count, plural, =1{Διαγράψτε αυτό το άλμπουμ και το αντικείμενο σε αυτό;} other{Διαγράψτε αυτό το άλμπουμ και τα {count} αντικείμενα σε αυτό;}}",
"deleteSingleAlbumConfirmationDialogMessage": "{count, plural, =1{Διαγραφή αυτού του άλμπουμ και του περιεχομένου του;} other{Διαγράψτε αυτό το άλμπουμ και όλα τα {count} στοιχεία του;}}",
"@deleteSingleAlbumConfirmationDialogMessage": {},
"deleteMultiAlbumConfirmationDialogMessage": "{count, plural, =1{Διαγράψτε αυτά τα άλμπουμ και τα αντικείμενα σε αυτά;} other{Διαγράψτε αυτά τα άλμπουμ και τα {count} αντικείμενα σε αυτά;}}",
"deleteMultiAlbumConfirmationDialogMessage": "{count, plural, =1{Διαγράψτε αυτά τα άλμπουμ και το περιεχόμενό τους;} other{Διαγράψτε αυτά τα άλμπουμ και όλα τα {count} στοιχεία τους;}}",
"@deleteMultiAlbumConfirmationDialogMessage": {},
"exportEntryDialogFormat": "Μορφή:",
"@exportEntryDialogFormat": {},
@ -463,7 +463,7 @@
"@menuActionStats": {},
"viewDialogSortSectionTitle": "Ταξινομηση",
"@viewDialogSortSectionTitle": {},
"viewDialogGroupSectionTitle": "Τμήματα",
"viewDialogGroupSectionTitle": "Ομαδοποιηση",
"@viewDialogGroupSectionTitle": {},
"viewDialogLayoutSectionTitle": "Διαταξη",
"@viewDialogLayoutSectionTitle": {},
@ -557,6 +557,8 @@
"@collectionGroupMonth": {},
"collectionGroupDay": "Ανά ημέρα",
"@collectionGroupDay": {},
"collectionGroupNone": "Να μην γίνει ομαδοποίηση",
"@collectionGroupNone": {},
"sectionUnknown": "Χωρίς λεπτομέρειες",
"@sectionUnknown": {},
"dateToday": "Σήμερα",
@ -659,6 +661,8 @@
"@albumGroupType": {},
"albumGroupVolume": "Ανά αποθηκευτική μονάδα",
"@albumGroupVolume": {},
"albumGroupNone": "Να μην γίνει ομαδοποίηση",
"@albumGroupNone": {},
"albumMimeTypeMixed": "Μικτα",
"@albumMimeTypeMixed": {},
"albumPickPageTitleCopy": "Αντιγραφή στο άλμπουμ",
@ -869,7 +873,7 @@
"@settingsSlideshowAnimatedZoomEffect": {},
"settingsSlideshowTransitionTile": "Μετάβαση",
"@settingsSlideshowTransitionTile": {},
"settingsSlideshowIntervalTile": "Διάστημα",
"settingsSlideshowIntervalTile": "Διάρκεια",
"@settingsSlideshowIntervalTile": {},
"settingsSlideshowVideoPlaybackTile": "Αναπαραγωγή βίντεο",
"@settingsSlideshowVideoPlaybackTile": {},
@ -1103,6 +1107,16 @@
"@panoramaDisableSensorControl": {},
"sourceViewerPageTitle": "Πηγη",
"@sourceViewerPageTitle": {},
"filePickerShowHiddenFiles": "Εμφάνιση κρυφών αρχείων",
"@filePickerShowHiddenFiles": {},
"filePickerDoNotShowHiddenFiles": "Να μην εμφανίζονται τα κρυφά αρχεία",
"@filePickerDoNotShowHiddenFiles": {},
"filePickerOpenFrom": "Άνοιγμα από",
"@filePickerOpenFrom": {},
"filePickerNoItems": "Κανένα στοιχείο",
"@filePickerNoItems": {},
"filePickerUseThisFolder": "Χρησιμοποιήστε αυτόν τον φάκελο",
"@filePickerUseThisFolder": {},
"widgetOpenPageCollection": "Ανοιχτή συλλογή",
"@widgetOpenPageCollection": {},
"durationDialogSeconds": "Δευτερόλεπτα",
@ -1306,129 +1320,5 @@
"overlayHistogramLuminance": "Φωτεινότητα",
"@overlayHistogramLuminance": {},
"chipActionShowCollection": "Εμφάνιση στη Συλλογή",
"@chipActionShowCollection": {},
"mapAttributionOsmHot": "Πλακάκια από [HOT](https://www.hotosm.org/) • Φιλοξενείται από [OSM France](https://openstreetmap.fr/)",
"@mapAttributionOsmHot": {},
"mapAttributionStamen": "Πλακάκια από [Stamen Design](https://stamen.com), [CC BY 3.0](https://creativecommons.org/licenses/by/3.0)",
"@mapAttributionStamen": {},
"editEntryLocationDialogImportGpx": "Εισαγωγή GPX",
"@editEntryLocationDialogImportGpx": {},
"editEntryLocationDialogTimeShift": "Χρονική μετατόπιση",
"@editEntryLocationDialogTimeShift": {},
"aboutDataUsageData": "Δεδομένα",
"@aboutDataUsageData": {},
"collectionActionAddDynamicAlbum": "Προσθήκη δυναμικού άλμπουμ",
"@collectionActionAddDynamicAlbum": {},
"collectionActionSetHome": "Ορίστε ως σπίτι",
"@collectionActionSetHome": {},
"sortByPath": "Από τη διαδρομή",
"@sortByPath": {},
"explorerPageTitle": "Εξερεύνηση",
"@explorerPageTitle": {},
"searchFormatSectionTitle": "Μορφές",
"@searchFormatSectionTitle": {},
"settingsViewerShowHistogram": "Εμφάνιση ιστογράμματος",
"@settingsViewerShowHistogram": {},
"settingsForceWesternArabicNumeralsTile": "Δύναμη αραβικών αριθμών",
"@settingsForceWesternArabicNumeralsTile": {},
"mapAttributionOsmData": "Δεδομένα χάρτη © [OpenStreetMap](https://www.openstreetmap.org/copyright) συνεισφέροντες",
"@mapAttributionOsmData": {},
"mapAttributionOsmLiberty": "Πλακάκια από [OpenMapTiles](https://www.openmaptiles.org/), [CC BY](http://creativecommons.org/licenses/by/4.0) • Φιλοξενείται από [OSM Americana](https://tile.ourmap.us)",
"@mapAttributionOsmLiberty": {},
"aboutDataUsageMisc": "Διάφορα στοιχεία",
"@aboutDataUsageMisc": {},
"selectStorageVolumeDialogTitle": "Επιλέξτε Αποθήκευση",
"@selectStorageVolumeDialogTitle": {},
"renameProcessorHash": "Κατακερματισμός",
"@renameProcessorHash": {},
"aboutDataUsageSectionTitle": "Χρήση δεδομένων",
"@aboutDataUsageSectionTitle": {},
"chipActionDecompose": "Διάσπαση",
"@chipActionDecompose": {},
"entryActionCast": "Εκτέλεση",
"@entryActionCast": {},
"videoActionShowPreviousFrame": "Εμφάνιση προηγούμενου καρέ",
"@videoActionShowPreviousFrame": {},
"videoActionABRepeat": "Επανάληψη A-B",
"@videoActionABRepeat": {},
"videoRepeatActionSetStart": "Ορισμός έναρξης",
"@videoRepeatActionSetStart": {},
"videoRepeatActionSetEnd": "Ορισμός τέλους",
"@videoRepeatActionSetEnd": {},
"albumTierDynamic": "Δυναμικό",
"@albumTierDynamic": {},
"newDynamicAlbumDialogTitle": "Νέο δυναμικό άλμπουμ",
"@newDynamicAlbumDialogTitle": {},
"dynamicAlbumAlreadyExists": "Το δυναμικό άλμπουμ υπάρχει ήδη",
"@dynamicAlbumAlreadyExists": {},
"setHomeCustom": "Προσαρμοσμένο",
"@setHomeCustom": {},
"settingsThumbnailShowHdrIcon": "Εμφάνιση εικονιδίου HDR",
"@settingsThumbnailShowHdrIcon": {},
"coordinateFormatDdm": "DDM",
"@coordinateFormatDdm": {},
"chipActionRemove": "Αφαίρεση",
"@chipActionRemove": {},
"newAlbumDialogAlbumAlreadyExistsHelper": "Το άλμπουμ υπάρχει ήδη",
"@newAlbumDialogAlbumAlreadyExistsHelper": {},
"explorerActionSelectStorageVolume": "Επιλέξτε αποθηκευτικό χώρο",
"@explorerActionSelectStorageVolume": {},
"removeEntryMetadataDialogAll": "Όλα",
"@removeEntryMetadataDialogAll": {},
"aboutDataUsageCache": "Κρυφή μνήμη",
"@aboutDataUsageCache": {},
"castDialogTitle": "Συσκευές Απεικόνισης",
"@castDialogTitle": {},
"aboutDataUsageDatabase": "Βάση δεδομένων",
"@aboutDataUsageDatabase": {},
"sortByDuration": "Με διάρκεια",
"@sortByDuration": {},
"appExportDynamicAlbums": "Δυναμικά άλμπουμ",
"@appExportDynamicAlbums": {},
"mapStyleOsmLiberty": "OSM Ελευθερία",
"@mapStyleOsmLiberty": {},
"mapStyleOpenTopoMap": "OpenTopoMap",
"@mapStyleOpenTopoMap": {},
"aboutDataUsageInternal": "Εσωτερικά",
"@aboutDataUsageInternal": {},
"aboutDataUsageExternal": "Εξωτερικά",
"@aboutDataUsageExternal": {},
"aboutDataUsageClearCache": "Εκκαθάριση μνήμης",
"@aboutDataUsageClearCache": {},
"videoActionShowNextFrame": "Εμφάνιση επόμενου καρέ",
"@videoActionShowNextFrame": {},
"mapAttributionOpenTopoMap": "[SRTM](https://www.earthdata.nasa.gov/sensors/srtm) | Πλακάκια από [OpenTopoMap](https://opentopomap.org/), [CC BY-SA](https://creativecommons.org/licenses/by-sa/3.0/)",
"@mapAttributionOpenTopoMap": {},
"sortOrderShortestFirst": "Ο συντομότερος πρώτος",
"@sortOrderShortestFirst": {},
"sortOrderLongestFirst": "Μακρύτερο πρώτο",
"@sortOrderLongestFirst": {},
"stopTooltip": "Σταμάτησε",
"@stopTooltip": {},
"chipActionGoToExplorerPage": "Εμφάνιση στην Εξερεύνηση",
"@chipActionGoToExplorerPage": {},
"chipActionCreateGroup": "Δημιουργία ομάδας",
"@chipActionCreateGroup": {},
"newGroupDialogNameLabel": "Όνομα ομάδας",
"@newGroupDialogNameLabel": {},
"groupPickerUseThisGroupButton": "Χρησιμοποιήστε αυτή την ομάδα",
"@groupPickerUseThisGroupButton": {},
"sectionNone": "Όχι τμήματα",
"@sectionNone": {},
"createButtonLabel": "ΔΗΜΙΟΥΡΓΙΑ",
"@createButtonLabel": {},
"chipActionGroup": "Ομάδα",
"@chipActionGroup": {},
"albumTierGroups": "Ομάδες",
"@albumTierGroups": {},
"newGroupDialogTitle": "Νέα Ομάδα",
"@newGroupDialogTitle": {},
"groupAlreadyExists": "Η ομάδα υπάρχει ήδη",
"@groupAlreadyExists": {},
"groupEmpty": "Όχι ομάδες",
"@groupEmpty": {},
"ungrouped": "Μη ομαδοποιημένο",
"@ungrouped": {},
"groupPickerTitle": "Επιλογή ομάδας",
"@groupPickerTitle": {}
"@chipActionShowCollection": {}
}

View file

@ -60,7 +60,6 @@
},
"applyButtonLabel": "APPLY",
"createButtonLabel": "CREATE",
"deleteButtonLabel": "DELETE",
"nextButtonLabel": "NEXT",
"showButtonLabel": "SHOW",
@ -105,11 +104,9 @@
"chipActionLock": "Lock",
"chipActionPin": "Pin to top",
"chipActionUnpin": "Unpin from top",
"chipActionGroup": "Change grouping",
"chipActionRename": "Rename",
"chipActionSetCover": "Set cover",
"chipActionShowCountryStates": "Show states",
"chipActionCreateGroup": "Create group",
"chipActionCreateAlbum": "Create album",
"chipActionCreateVault": "Create vault",
"chipActionConfigureVault": "Configure vault",
@ -211,7 +208,6 @@
"albumTierNew": "New",
"albumTierPinned": "Pinned",
"albumTierGroups": "Groups",
"albumTierSpecial": "Common",
"albumTierApps": "Apps",
"albumTierVaults": "Vaults",
@ -447,14 +443,6 @@
"newDynamicAlbumDialogTitle": "New Dynamic Album",
"dynamicAlbumAlreadyExists": "Dynamic album already exists",
"newGroupDialogTitle": "New Group",
"newGroupDialogNameLabel": "Group name",
"groupAlreadyExists": "Group already exists",
"groupEmpty": "No groups",
"ungrouped": "Ungrouped",
"groupPickerTitle": "Pick Group",
"groupPickerUseThisGroupButton": "Use this group",
"newVaultWarningDialogMessage": "Items in vaults are only available to this app and no others.\n\nIf you uninstall this app, or clear this app data, you will lose all these items.",
"newVaultDialogTitle": "New Vault",
"configureVaultDialogTitle": "Configure Vault",
@ -570,7 +558,7 @@
"menuActionStats": "Stats",
"viewDialogSortSectionTitle": "Sort",
"viewDialogGroupSectionTitle": "Sections",
"viewDialogGroupSectionTitle": "Group",
"viewDialogLayoutSectionTitle": "Layout",
"viewDialogReverseSortOrder": "Reverse sort order",
@ -642,8 +630,8 @@
"collectionGroupAlbum": "By album",
"collectionGroupMonth": "By month",
"collectionGroupDay": "By day",
"collectionGroupNone": "Do not group",
"sectionNone": "No sections",
"sectionUnknown": "Unknown",
"dateToday": "Today",
"dateYesterday": "Yesterday",
@ -767,10 +755,9 @@
"sortByName": "By name",
"sortByItemCount": "By item count",
"sortBySize": "By size",
"sortByAlbumFileName": "By album & item title",
"sortByAlbumFileName": "By album & file name",
"sortByRating": "By rating",
"sortByDuration": "By duration",
"sortByPath": "By path",
"sortOrderNewestFirst": "Newest first",
"sortOrderOldestFirst": "Oldest first",
@ -786,6 +773,7 @@
"albumGroupTier": "By tier",
"albumGroupType": "By type",
"albumGroupVolume": "By storage volume",
"albumGroupNone": "Do not group",
"albumMimeTypeMixed": "Mixed",
@ -827,7 +815,6 @@
"searchCollectionFieldHint": "Search collection",
"searchRecentSectionTitle": "Recent",
"searchDateSectionTitle": "Date",
"searchFormatSectionTitle": "Formats",
"searchAlbumsSectionTitle": "Albums",
"searchCountriesSectionTitle": "Countries",
"searchStatesSectionTitle": "States",
@ -1112,5 +1099,11 @@
"panoramaEnableSensorControl": "Enable sensor control",
"panoramaDisableSensorControl": "Disable sensor control",
"sourceViewerPageTitle": "Source"
"sourceViewerPageTitle": "Source",
"filePickerShowHiddenFiles": "Show hidden files",
"filePickerDoNotShowHiddenFiles": "Dont show hidden files",
"filePickerOpenFrom": "Open from",
"filePickerNoItems": "No items",
"filePickerUseThisFolder": "Use this folder"
}

View file

@ -921,6 +921,8 @@
"@collectionGroupMonth": {},
"collectionGroupDay": "𐑚𐑲 𐑛𐑱",
"@collectionGroupDay": {},
"collectionGroupNone": "𐑛𐑵 𐑯𐑪𐑑 𐑜𐑮𐑵𐑐",
"@collectionGroupNone": {},
"sectionUnknown": "𐑳𐑯𐑯𐑴𐑯",
"@sectionUnknown": {},
"dateToday": "𐑑𐑫𐑛𐑱",
@ -1083,6 +1085,8 @@
"@albumGroupType": {},
"albumGroupVolume": "𐑚𐑲 𐑕𐑑𐑹𐑦𐑡 𐑝𐑪𐑤𐑿𐑥",
"@albumGroupVolume": {},
"albumGroupNone": "𐑛𐑵 𐑯𐑪𐑑 𐑜𐑮𐑵𐑐",
"@albumGroupNone": {},
"albumMimeTypeMixed": "𐑥𐑦𐑒𐑕𐑑",
"@albumMimeTypeMixed": {},
"albumPickPageTitleCopy": "𐑒𐑪𐑐𐑦 𐑑 𐑨𐑤𐑚𐑩𐑥",
@ -1275,6 +1279,8 @@
"@settingsViewerShowMinimap": {},
"settingsViewerShowInformation": "𐑖𐑴 𐑦𐑯𐑓𐑼𐑥𐑱𐑖𐑩𐑯",
"@settingsViewerShowInformation": {},
"filePickerUseThisFolder": "𐑿𐑟 𐑞𐑦𐑕 𐑓𐑴𐑤𐑛𐑼",
"@filePickerUseThisFolder": {},
"settingsViewerShowInformationSubtitle": "𐑖𐑴 𐑑𐑲𐑑𐑩𐑤, 𐑛𐑱𐑑, 𐑤𐑴𐑒𐑱𐑖𐑩𐑯, 𐑯𐑯𐑯",
"@settingsViewerShowInformationSubtitle": {},
"settingsViewerShowOverlayThumbnails": "𐑖𐑴 𐑔𐑳𐑥𐑯𐑱𐑤𐑟",
@ -1594,6 +1600,14 @@
"@panoramaDisableSensorControl": {},
"sourceViewerPageTitle": "𐑕𐑹𐑕",
"@sourceViewerPageTitle": {},
"filePickerShowHiddenFiles": "𐑖𐑴 𐑣𐑦𐑛𐑩𐑯 𐑓𐑲𐑤𐑟",
"@filePickerShowHiddenFiles": {},
"filePickerDoNotShowHiddenFiles": "𐑛𐑴𐑯'𐑑 𐑖𐑴 𐑣𐑦𐑛𐑩𐑯 𐑓𐑲𐑤𐑟",
"@filePickerDoNotShowHiddenFiles": {},
"filePickerOpenFrom": "𐑴𐑐𐑩𐑯 𐑓𐑮𐑪𐑥",
"@filePickerOpenFrom": {},
"filePickerNoItems": "𐑯𐑴 𐑲𐑑𐑩𐑥𐑟",
"@filePickerNoItems": {},
"videoActionShowPreviousFrame": "𐑖𐑴 𐑐𐑮𐑰𐑝𐑾𐑕 𐑓𐑮𐑱𐑥",
"@videoActionShowPreviousFrame": {},
"videoActionShowNextFrame": "𐑖𐑴 𐑯𐑧𐑒𐑕𐑑 𐑓𐑮𐑱𐑥",

View file

@ -445,7 +445,7 @@
"@menuActionStats": {},
"viewDialogSortSectionTitle": "Ordenar",
"@viewDialogSortSectionTitle": {},
"viewDialogGroupSectionTitle": "Secciones",
"viewDialogGroupSectionTitle": "Grupo",
"@viewDialogGroupSectionTitle": {},
"viewDialogLayoutSectionTitle": "Disposición",
"@viewDialogLayoutSectionTitle": {},
@ -535,6 +535,8 @@
"@collectionGroupMonth": {},
"collectionGroupDay": "Por día",
"@collectionGroupDay": {},
"collectionGroupNone": "No agrupar",
"@collectionGroupNone": {},
"sectionUnknown": "Desconocido",
"@sectionUnknown": {},
"dateToday": "Hoy",
@ -619,6 +621,8 @@
"@albumGroupTier": {},
"albumGroupVolume": "Por volumen de almacenamiento",
"@albumGroupVolume": {},
"albumGroupNone": "No agrupar",
"@albumGroupNone": {},
"albumPickPageTitleCopy": "Copiar a álbum",
"@albumPickPageTitleCopy": {},
"albumPickPageTitleExport": "Exportar a álbum",
@ -1049,6 +1053,16 @@
"@panoramaDisableSensorControl": {},
"sourceViewerPageTitle": "Fuente",
"@sourceViewerPageTitle": {},
"filePickerShowHiddenFiles": "Mostrar archivos ocultos",
"@filePickerShowHiddenFiles": {},
"filePickerDoNotShowHiddenFiles": "No mostrar archivos ocultos",
"@filePickerDoNotShowHiddenFiles": {},
"filePickerOpenFrom": "Abrir desde",
"@filePickerOpenFrom": {},
"filePickerNoItems": "Sin elementos",
"@filePickerNoItems": {},
"filePickerUseThisFolder": "Usar esta carpeta",
"@filePickerUseThisFolder": {},
"chipActionFilterOut": "Filtrar",
"@chipActionFilterOut": {},
"chipActionFilterIn": "Filtrar en",
@ -1402,33 +1416,5 @@
"editEntryLocationDialogTimeShift": "Desplazamiento de tiempo",
"@editEntryLocationDialogTimeShift": {},
"coordinateFormatDdm": "DDM",
"@coordinateFormatDdm": {},
"sortByPath": "Por ruta",
"@sortByPath": {},
"searchFormatSectionTitle": "Formatos",
"@searchFormatSectionTitle": {},
"newGroupDialogTitle": "Nuevo grupo",
"@newGroupDialogTitle": {},
"ungrouped": "No agrupado",
"@ungrouped": {},
"albumTierGroups": "Grupos",
"@albumTierGroups": {},
"groupEmpty": "Sin grupos",
"@groupEmpty": {},
"newGroupDialogNameLabel": "Nombre del grupo",
"@newGroupDialogNameLabel": {},
"createButtonLabel": "CREAR",
"@createButtonLabel": {},
"chipActionGroup": "Agrupar",
"@chipActionGroup": {},
"chipActionCreateGroup": "Crear grupo",
"@chipActionCreateGroup": {},
"groupAlreadyExists": "Ya existe el grupo",
"@groupAlreadyExists": {},
"groupPickerTitle": "Seleccionar grupo",
"@groupPickerTitle": {},
"groupPickerUseThisGroupButton": "Usar este grupo",
"@groupPickerUseThisGroupButton": {},
"sectionNone": "Sin secciones",
"@sectionNone": {}
"@coordinateFormatDdm": {}
}

View file

@ -748,7 +748,7 @@
"@videoStreamSelectionDialogNoSelection": {},
"viewDialogSortSectionTitle": "Järjesta",
"@viewDialogSortSectionTitle": {},
"viewDialogGroupSectionTitle": "Rubriigid",
"viewDialogGroupSectionTitle": "Rühmita",
"@viewDialogGroupSectionTitle": {},
"viewDialogLayoutSectionTitle": "Paiguta",
"@viewDialogLayoutSectionTitle": {},
@ -790,7 +790,7 @@
"@aboutLicensesDartPackagesSectionTitle": {},
"aboutLicensesShowAllButtonLabel": "Näita kõiki litsentse",
"@aboutLicensesShowAllButtonLabel": {},
"policyPageTitle": "Andmekaitsepõhimõtted",
"policyPageTitle": "Privaatsuspoliitika",
"@policyPageTitle": {},
"collectionPageTitle": "Meediakogu",
"@collectionPageTitle": {},
@ -866,6 +866,8 @@
"@collectionGroupMonth": {},
"collectionGroupDay": "Päevade kaupa",
"@collectionGroupDay": {},
"collectionGroupNone": "Ära rühmita",
"@collectionGroupNone": {},
"sectionUnknown": "Teadmata",
"@sectionUnknown": {},
"dateToday": "Täna",
@ -1036,7 +1038,7 @@
"@sortBySize": {},
"sortByName": "Nime alusel",
"@sortByName": {},
"sortByAlbumFileName": "Albumi ja objekti nime alusel",
"sortByAlbumFileName": "Albumi ja failinime alusel",
"@sortByAlbumFileName": {},
"sortByRating": "Hinnangu alusel",
"@sortByRating": {},
@ -1054,6 +1056,8 @@
"@albumGroupType": {},
"albumGroupVolume": "Andmemahu alusel",
"@albumGroupVolume": {},
"albumGroupNone": "Ära rühmita",
"@albumGroupNone": {},
"albumMimeTypeMixed": "Erinev sisu",
"@albumMimeTypeMixed": {},
"albumPickPageTitleCopy": "Kopeeri albumisse",
@ -1561,6 +1565,14 @@
"@panoramaEnableSensorControl": {},
"sourceViewerPageTitle": "Allikas",
"@sourceViewerPageTitle": {},
"filePickerDoNotShowHiddenFiles": "Ära näita peidetud faile",
"@filePickerDoNotShowHiddenFiles": {},
"filePickerOpenFrom": "Ava asukohast",
"@filePickerOpenFrom": {},
"filePickerNoItems": "Meediafaile pole",
"@filePickerNoItems": {},
"filePickerUseThisFolder": "Kasuta seda kausta",
"@filePickerUseThisFolder": {},
"settingsSlideshowRepeat": "Korda",
"@settingsSlideshowRepeat": {},
"settingsVideoAutoPlay": "Automaatne taasesitus",
@ -1597,6 +1609,8 @@
"@viewerInfoLabelTitle": {},
"mapAttributionOsmData": "Kaardiandmed © [OpenStreetMap](https://www.openstreetmap.org/copyright) kaasautorid",
"@mapAttributionOsmData": {},
"filePickerShowHiddenFiles": "Näita peidetud faile",
"@filePickerShowHiddenFiles": {},
"settingsViewerQuickActionEditorBanner": "Nuppude/ikoonide valimiseks ja kahe vaate vahel teisaldamiseks puuduta ja all hoides lohista uude kohta.",
"@settingsViewerQuickActionEditorBanner": {},
"settingsViewerOverlayPageTitle": "Ülekate",
@ -1620,33 +1634,5 @@
"editEntryLocationDialogImportGpx": "Impordi GPX-fail",
"@editEntryLocationDialogImportGpx": {},
"removeEntryMetadataDialogAll": "Kõik",
"@removeEntryMetadataDialogAll": {},
"sortByPath": "Asukoha alusel",
"@sortByPath": {},
"searchFormatSectionTitle": "Vormingud",
"@searchFormatSectionTitle": {},
"chipActionCreateGroup": "Loo grupp",
"@chipActionCreateGroup": {},
"albumTierGroups": "Grupid",
"@albumTierGroups": {},
"newGroupDialogTitle": "Uus grupp",
"@newGroupDialogTitle": {},
"newGroupDialogNameLabel": "Grupi nimi",
"@newGroupDialogNameLabel": {},
"groupAlreadyExists": "Selline grupp on juba olemas",
"@groupAlreadyExists": {},
"groupEmpty": "Gruppe pole",
"@groupEmpty": {},
"ungrouped": "Grupeerimata",
"@ungrouped": {},
"groupPickerTitle": "Vali grupp",
"@groupPickerTitle": {},
"groupPickerUseThisGroupButton": "Kasuta seda gruppi",
"@groupPickerUseThisGroupButton": {},
"createButtonLabel": "LOO",
"@createButtonLabel": {},
"chipActionGroup": "Muuda grupeerimist",
"@chipActionGroup": {},
"sectionNone": "Rubriike pole",
"@sectionNone": {}
"@removeEntryMetadataDialogAll": {}
}

View file

@ -709,6 +709,8 @@
"@collectionSearchTitlesHintText": {},
"collectionGroupMonth": "Hilabetearen arabera",
"@collectionGroupMonth": {},
"collectionGroupNone": "Ez taldekatu",
"@collectionGroupNone": {},
"sectionUnknown": "Ezezaguna",
"@sectionUnknown": {},
"dateThisMonth": "Hilabete honetan",
@ -877,6 +879,8 @@
"@albumGroupTier": {},
"albumGroupType": "Motaren arabera",
"@albumGroupType": {},
"albumGroupNone": "Ez taldekatu",
"@albumGroupNone": {},
"albumPickPageTitleCopy": "Kopiatu albumera",
"@albumPickPageTitleCopy": {},
"albumPickPageTitleExport": "Albumera esportatu",
@ -1301,6 +1305,10 @@
"@panoramaDisableSensorControl": {},
"sourceViewerPageTitle": "Iturria",
"@sourceViewerPageTitle": {},
"filePickerShowHiddenFiles": "Erakutsi ezkutuko fitxategiak",
"@filePickerShowHiddenFiles": {},
"filePickerDoNotShowHiddenFiles": "Ez erakutsi ezkutuko fitxategiak",
"@filePickerDoNotShowHiddenFiles": {},
"settingsThemeBrightnessTile": "Gaia",
"@settingsThemeBrightnessTile": {},
"settingsDisplayUseTvInterface": "Android TV interfazea",
@ -1343,6 +1351,8 @@
"@settingsWidgetDisplayedItem": {},
"tagEditorPageNewTagFieldLabel": "Etiketa berria",
"@tagEditorPageNewTagFieldLabel": {},
"filePickerNoItems": "Elementurik ez",
"@filePickerNoItems": {},
"viewerInfoSearchFieldLabel": "Bilatu metadatuak",
"@viewerInfoSearchFieldLabel": {},
"viewerInfoViewXmlLinkText": "Ikusi XML",
@ -1353,6 +1363,10 @@
"@tagEditorPageAddTagTooltip": {},
"tagEditorSectionRecent": "Berrienak",
"@tagEditorSectionRecent": {},
"filePickerOpenFrom": "Ireki hemendik",
"@filePickerOpenFrom": {},
"filePickerUseThisFolder": "Erabili karpeta hau",
"@filePickerUseThisFolder": {},
"settingsWidgetOpenPage": "Widgetan sakatzean",
"@settingsWidgetOpenPage": {},
"exportEntryDialogWriteMetadata": "Idatzi metadatuak",

View file

@ -29,7 +29,7 @@
"@nextButtonLabel": {},
"pickTooltip": "انتخاب",
"@pickTooltip": {},
"actionRemove": "برداشتن",
"actionRemove": "پاک‌کردن",
"@actionRemove": {},
"chipActionGoToTagPage": "نمایش در برچسب‌ها",
"@chipActionGoToTagPage": {},
@ -63,7 +63,7 @@
"@hideButtonLabel": {},
"hideTooltip": "پنهان",
"@hideTooltip": {},
"chipActionCreateAlbum": "ساخت البوم",
"chipActionCreateAlbum": "ایجاد البوم",
"@chipActionCreateAlbum": {},
"filterNoRatingLabel": "بدون امتیاز",
"@filterNoRatingLabel": {},
@ -205,7 +205,7 @@
"@videoLoopModeAlways": {},
"menuActionSlideshow": "نمایش اسلایدی",
"@menuActionSlideshow": {},
"entryActionRemoveFavourite": "پاک‌کردن از برگزیده‌ها",
"entryActionRemoveFavourite": "پاک‌کردن از مورد علاقه ها",
"@entryActionRemoveFavourite": {},
"videoLoopModeNever": "هیچ وقت",
"@videoLoopModeNever": {},
@ -242,9 +242,9 @@
"@videoLoopModeShortOnly": {},
"entryActionEdit": "ویرایش",
"@entryActionEdit": {},
"entryActionAddFavourite": "افزودن به برگزیده‌ها",
"entryActionAddFavourite": "اضافه کردن به مورد علاقه ها",
"@entryActionAddFavourite": {},
"filterFavouriteLabel": "برگزیده",
"filterFavouriteLabel": "مورد علاقه",
"@filterFavouriteLabel": {},
"mapZoomOutTooltip": "کوچک نمایی",
"@mapZoomOutTooltip": {},
@ -260,7 +260,7 @@
"@chipActionFilterOut": {},
"entryActionRotateScreen": "چرخش صفحه",
"@entryActionRotateScreen": {},
"drawerCollectionFavourites": "برگزیده‌ها",
"drawerCollectionFavourites": "مورد علاقه ها",
"@drawerCollectionFavourites": {},
"filterMimeImageLabel": "عکس",
"@filterMimeImageLabel": {},
@ -272,11 +272,11 @@
"@entryInfoActionExportMetadata": {},
"exportEntryDialogFormat": "فرمت:",
"@exportEntryDialogFormat": {},
"collectionEmptyFavourites": "هیچ برگزیده‌ای نیست",
"collectionEmptyFavourites": "هیچ مورد علاقه ای وجود ندارد",
"@collectionEmptyFavourites": {},
"appExportSettings": "تنظیمات",
"@appExportSettings": {},
"appExportFavourites": "برگزیده‌ها",
"appExportFavourites": "مورد علاقه ها",
"@appExportFavourites": {},
"exportEntryDialogHeight": "طول",
"@exportEntryDialogHeight": {},
@ -397,7 +397,7 @@
"@widgetOpenPageHome": {},
"keepScreenOnAlways": "همیشه",
"@keepScreenOnAlways": {},
"albumTierRegular": "دیگری‌ها",
"albumTierRegular": "سایر",
"@albumTierRegular": {},
"accessibilityAnimationsKeep": "نمایش از جلوه‌های نمایشگر",
"@accessibilityAnimationsKeep": {},
@ -439,7 +439,7 @@
"@entryActionCast": {},
"viewerActionUnlock": "باز کردن قفل پخش‌کننده",
"@viewerActionUnlock": {},
"videoActionPause": "ایست",
"videoActionPause": "مکث",
"@videoActionPause": {},
"widgetOpenPageCollection": "باز کردن مجموعه",
"@widgetOpenPageCollection": {},
@ -471,7 +471,7 @@
},
"albumTierPinned": "سنجاق شده",
"@albumTierPinned": {},
"chipActionCreateVault": "ساخت گاوصندوق",
"chipActionCreateVault": "ایجاد گاوصندوق",
"@chipActionCreateVault": {},
"chipActionGoToPlacePage": "نمایش در مکان‌ها",
"@chipActionGoToPlacePage": {},
@ -517,7 +517,7 @@
"@keepScreenOnViewerOnly": {},
"wallpaperTargetHome": "صفحهٔ خانه",
"@wallpaperTargetHome": {},
"videoActionSelectStreams": "انتخاب صوت‌ها",
"videoActionSelectStreams": "انتخاب قطعه‌ٔ صوتی",
"@videoActionSelectStreams": {},
"widgetDisplayedItemRandom": "تصادفی",
"@widgetDisplayedItemRandom": {},
@ -678,6 +678,10 @@
"@mapStyleTooltip": {},
"viewerInfoSearchFieldLabel": "جستجو فراداده",
"@viewerInfoSearchFieldLabel": {},
"filePickerDoNotShowHiddenFiles": "پرونده‌های پنهان را نمایش نده",
"@filePickerDoNotShowHiddenFiles": {},
"filePickerUseThisFolder": "استفاده از این پوشه",
"@filePickerUseThisFolder": {},
"tagEmpty": "بدون برچسب ها",
"@tagEmpty": {},
"binPageTitle": "سطل زباله",
@ -698,6 +702,8 @@
"@sourceViewerPageTitle": {},
"panoramaDisableSensorControl": "خاموش کردن هدایت حسگر",
"@panoramaDisableSensorControl": {},
"filePickerShowHiddenFiles": "نمایش پرونده‌های پنهان",
"@filePickerShowHiddenFiles": {},
"setCoverDialogLatest": "آخرین مورد",
"@setCoverDialogLatest": {},
"configureVaultDialogTitle": "پیکربندی گاوصندوق",
@ -708,6 +714,8 @@
"@sortOrderSmallestFirst": {},
"albumMimeTypeMixed": "ترکیبی",
"@albumMimeTypeMixed": {},
"albumGroupNone": "گروه نکن",
"@albumGroupNone": {},
"newFilterBanner": "جدید",
"@newFilterBanner": {},
"albumScreenshots": "تصاویر از صفحه",
@ -871,6 +879,8 @@
"@renameEntryDialogLabel": {},
"settingsStorageAccessTile": "دسترسی حافظه",
"@settingsStorageAccessTile": {},
"filePickerNoItems": "چیزی نیست",
"@filePickerNoItems": {},
"settingsHomeDialogTitle": "خانه",
"@settingsHomeDialogTitle": {},
"settingsThumbnailShowRating": "نمایش امتیازبندی",
@ -1013,7 +1023,7 @@
"@aboutDataUsageCache": {},
"aboutDataUsageInternal": "داخلی",
"@aboutDataUsageInternal": {},
"aboutCreditsSectionTitle": "سپاس‌نامه",
"aboutCreditsSectionTitle": "اعتبار",
"@aboutCreditsSectionTitle": {},
"aboutDataUsageExternal": "خارجی",
"@aboutDataUsageExternal": {},
@ -1023,7 +1033,7 @@
"@aboutDataUsageClearCache": {},
"aboutCreditsWorldAtlas2": "زیر مجوز ISC.",
"@aboutCreditsWorldAtlas2": {},
"aboutLicensesBanner": "این برنامه بسته‌ها و کتابخانه‌های آزاد زیر به‌کار می‌برد.",
"aboutLicensesBanner": "این برنامه از بسته‌ها و کتابخانه‌های منبع باز زیر استفاده می کند.",
"@aboutLicensesBanner": {},
"collectionSelectPageTitle": "انتخاب موارد",
"@collectionSelectPageTitle": {},
@ -1043,6 +1053,8 @@
"@collectionGroupMonth": {},
"dateThisMonth": "این ماه",
"@dateThisMonth": {},
"collectionGroupNone": "گروه نکن",
"@collectionGroupNone": {},
"collectionEmptyVideos": "بدون ویدیو",
"@collectionEmptyVideos": {},
"drawerCollectionImages": "تصاویر",
@ -1105,13 +1117,13 @@
"@settingsNavigationDrawerEditorPageTitle": {},
"settingsConfirmationBeforeMoveUndatedItems": "پیش از جابجایی موارد بدون تاریخ بپرسید",
"@settingsConfirmationBeforeMoveUndatedItems": {},
"settingsNavigationDrawerBanner": "برای جابجایی و مرتب کردن دوباره موارد، لمس کنید و نگه دارید.",
"settingsNavigationDrawerBanner": "برای جابجایی و مرتب کردن مجدد موارد، لمس کنید و نگه دارید.",
"@settingsNavigationDrawerBanner": {},
"settingsConfirmationVaultDataLoss": "نمایش هشدار از دست دادن داده‌های گاوصندوق",
"@settingsConfirmationVaultDataLoss": {},
"settingsNavigationDrawerTabAlbums": "آلبوم ها",
"@settingsNavigationDrawerTabAlbums": {},
"settingsThumbnailShowFavouriteIcon": "نمایش نماد برگزیدن",
"settingsThumbnailShowFavouriteIcon": "نمایش نماد علاقه‌مندی",
"@settingsThumbnailShowFavouriteIcon": {},
"settingsThumbnailShowLocationIcon": "نمایش نماد مکان",
"@settingsThumbnailShowLocationIcon": {},
@ -1337,6 +1349,8 @@
"@tagEditorPageAddTagTooltip": {},
"tagEditorPageNewTagFieldLabel": "برسب جدید",
"@tagEditorPageNewTagFieldLabel": {},
"filePickerOpenFrom": "بازکردن از",
"@filePickerOpenFrom": {},
"searchRatingSectionTitle": "امتیازات",
"@searchRatingSectionTitle": {},
"searchMetadataSectionTitle": "فراداده",
@ -1385,7 +1399,7 @@
"@tooManyItemsErrorDialogMessage": {},
"viewDialogLayoutSectionTitle": "چیدمان",
"@viewDialogLayoutSectionTitle": {},
"viewDialogReverseSortOrder": "ترتیب مرتبسازی معکوس",
"viewDialogReverseSortOrder": "ترتیب مرتب سازی معکوس",
"@viewDialogReverseSortOrder": {},
"aboutBugCopyInfoInstruction": "رونوشت اطلاعات سامانه",
"@aboutBugCopyInfoInstruction": {},
@ -1542,61 +1556,5 @@
"chipActionShowCollection": "نمایش در مجموعه",
"@chipActionShowCollection": {},
"mapAttributionOsmData": "داده‌های نقشه © [OpenStreetMap](https:www.openstreetmap.org/copyright) مشارکت‌کنندگان",
"@mapAttributionOsmData": {},
"chipActionRemove": "برداشتن",
"@chipActionRemove": {},
"mapStyleOpenTopoMap": "اوپن‌توپومپ",
"@mapStyleOpenTopoMap": {},
"editEntryLocationDialogImportGpx": "وارد کردن GPX",
"@editEntryLocationDialogImportGpx": {},
"sortByPath": "بر پایه مسیر",
"@sortByPath": {},
"setHomeCustom": "سفارشی",
"@setHomeCustom": {},
"videoActionShowPreviousFrame": "نمایش قاب پیشین",
"@videoActionShowPreviousFrame": {},
"videoActionShowNextFrame": "نمایش قاب بعدی",
"@videoActionShowNextFrame": {},
"sortOrderShortestFirst": "اول کوتاه‌ترین",
"@sortOrderShortestFirst": {},
"sortOrderLongestFirst": "اول بلندترین",
"@sortOrderLongestFirst": {},
"editEntryLocationDialogTimeShift": "تغییر زمان",
"@editEntryLocationDialogTimeShift": {},
"searchFormatSectionTitle": "قالب‌ها",
"@searchFormatSectionTitle": {},
"appExportDynamicAlbums": "آلبوم‌های پویا",
"@appExportDynamicAlbums": {},
"mapAttributionOsmLiberty": "کاشی‌ها به‌دست [OpenMapTiles](https://www.openmaptiles.org/)، [CC BY](http://creativecommons.org/licenses/by/4.0) • میزبانی‌شده به‌دست [OSM Americana](https://tile.ourmap.us)",
"@mapAttributionOsmLiberty": {},
"chipActionDecompose": "جداکردن",
"@chipActionDecompose": {},
"albumTierDynamic": "پویا",
"@albumTierDynamic": {},
"newDynamicAlbumDialogTitle": "آلبوم پویای جدید",
"@newDynamicAlbumDialogTitle": {},
"dynamicAlbumAlreadyExists": "آلبوم پویا از پیش موجود است",
"@dynamicAlbumAlreadyExists": {},
"sortByDuration": "بر پایه مدت",
"@sortByDuration": {},
"coordinateFormatDdm": "DDM",
"@coordinateFormatDdm": {},
"newAlbumDialogAlbumAlreadyExistsHelper": "آلبوم از پیش موجود است",
"@newAlbumDialogAlbumAlreadyExistsHelper": {},
"collectionActionAddDynamicAlbum": "افزودن آلبوم پویا",
"@collectionActionAddDynamicAlbum": {},
"selectStorageVolumeDialogTitle": "گزینش حافظه",
"@selectStorageVolumeDialogTitle": {},
"removeEntryMetadataDialogAll": "همه",
"@removeEntryMetadataDialogAll": {},
"explorerPageTitle": "کاوشگر",
"@explorerPageTitle": {},
"explorerActionSelectStorageVolume": "گزینش حافظه",
"@explorerActionSelectStorageVolume": {},
"mapStyleOsmLiberty": "اواس‌ام لیبریتی",
"@mapStyleOsmLiberty": {},
"mapAttributionOpenTopoMap": "[SRTM](https://www.earthdata.nasa.gov/sensors/srtm) | کاشی‌ها به‌دست [OpenTopoMap](https://opentopomap.org/)، [CC BY-SA](https://creativecommons.org/licenses/by-sa/3.0/)",
"@mapAttributionOpenTopoMap": {},
"chipActionGoToExplorerPage": "نمایش در کاوشگر",
"@chipActionGoToExplorerPage": {}
"@mapAttributionOsmData": {}
}

View file

@ -467,7 +467,7 @@
"@menuActionStats": {},
"viewDialogSortSectionTitle": "Tri",
"@viewDialogSortSectionTitle": {},
"viewDialogGroupSectionTitle": "Sections",
"viewDialogGroupSectionTitle": "Groupes",
"@viewDialogGroupSectionTitle": {},
"viewDialogLayoutSectionTitle": "Vue",
"@viewDialogLayoutSectionTitle": {},
@ -561,6 +561,8 @@
"@collectionGroupMonth": {},
"collectionGroupDay": "par jour",
"@collectionGroupDay": {},
"collectionGroupNone": "ne pas grouper",
"@collectionGroupNone": {},
"sectionUnknown": "Inconnu",
"@sectionUnknown": {},
"dateToday": "Aujourdhui",
@ -637,7 +639,7 @@
"@sortByItemCount": {},
"sortBySize": "par taille",
"@sortBySize": {},
"sortByAlbumFileName": "par titre dalbum et élément",
"sortByAlbumFileName": "alphabétique",
"@sortByAlbumFileName": {},
"sortByRating": "par notation",
"@sortByRating": {},
@ -663,6 +665,8 @@
"@albumGroupType": {},
"albumGroupVolume": "par volume de stockage",
"@albumGroupVolume": {},
"albumGroupNone": "ne pas grouper",
"@albumGroupNone": {},
"albumMimeTypeMixed": "Mixte",
"@albumMimeTypeMixed": {},
"albumPickPageTitleCopy": "Copie",
@ -1111,6 +1115,16 @@
"@panoramaDisableSensorControl": {},
"sourceViewerPageTitle": "Code source",
"@sourceViewerPageTitle": {},
"filePickerShowHiddenFiles": "Afficher les fichiers masqués",
"@filePickerShowHiddenFiles": {},
"filePickerDoNotShowHiddenFiles": "Ne pas afficher les fichiers masqués",
"@filePickerDoNotShowHiddenFiles": {},
"filePickerOpenFrom": "Ouvrir à partir de",
"@filePickerOpenFrom": {},
"filePickerNoItems": "Aucun élément",
"@filePickerNoItems": {},
"filePickerUseThisFolder": "Utiliser ce dossier",
"@filePickerUseThisFolder": {},
"editEntryLocationDialogSetCustom": "Définir un lieu personnalisé",
"@editEntryLocationDialogSetCustom": {},
"tagEditorSectionPlaceholders": "Étiquettes de substitution",
@ -1402,33 +1416,5 @@
"editEntryLocationDialogImportGpx": "Importer un fichier GPX",
"@editEntryLocationDialogImportGpx": {},
"removeEntryMetadataDialogAll": "Tout",
"@removeEntryMetadataDialogAll": {},
"sortByPath": "par chemin",
"@sortByPath": {},
"searchFormatSectionTitle": "Formats",
"@searchFormatSectionTitle": {},
"chipActionGroup": "Modifier groupement",
"@chipActionGroup": {},
"createButtonLabel": "CRÉER",
"@createButtonLabel": {},
"chipActionCreateGroup": "Créer un groupe",
"@chipActionCreateGroup": {},
"newGroupDialogTitle": "Nouveau groupe",
"@newGroupDialogTitle": {},
"groupAlreadyExists": "Le groupe existe déjà",
"@groupAlreadyExists": {},
"groupEmpty": "Aucun groupe",
"@groupEmpty": {},
"groupPickerUseThisGroupButton": "Utiliser ce groupe",
"@groupPickerUseThisGroupButton": {},
"groupPickerTitle": "Sélection",
"@groupPickerTitle": {},
"ungrouped": "Non groupé",
"@ungrouped": {},
"sectionNone": "Aucune section",
"@sectionNone": {},
"newGroupDialogNameLabel": "Nom du groupe",
"@newGroupDialogNameLabel": {},
"albumTierGroups": "Groupes",
"@albumTierGroups": {}
"@removeEntryMetadataDialogAll": {}
}

View file

@ -819,11 +819,13 @@
"@searchCountriesSectionTitle": {},
"collectionActionEmptyBin": "Baleirar lixo",
"@collectionActionEmptyBin": {},
"albumGroupNone": "Non agrupar",
"@albumGroupNone": {},
"menuActionSlideshow": "Presentación",
"@menuActionSlideshow": {},
"menuActionStats": "Estatísticas",
"@menuActionStats": {},
"viewDialogGroupSectionTitle": "Seccións",
"viewDialogGroupSectionTitle": "Agrupar",
"@viewDialogGroupSectionTitle": {},
"castDialogTitle": "Dispositivos de emisión",
"@castDialogTitle": {},
@ -975,6 +977,8 @@
"@collectionGroupMonth": {},
"collectionGroupDay": "Por día",
"@collectionGroupDay": {},
"collectionGroupNone": "Non agrupar",
"@collectionGroupNone": {},
"drawerSettingsButton": "Axustes",
"@drawerSettingsButton": {},
"sortOrderNewestFirst": "Novos primeiro",
@ -1258,6 +1262,8 @@
"@mapPointNorthUpTooltip": {},
"viewerInfoSearchSuggestionDate": "Data e hora",
"@viewerInfoSearchSuggestionDate": {},
"filePickerOpenFrom": "Abrir dende",
"@filePickerOpenFrom": {},
"settingsImageBackground": "Fondo de imaxe",
"@settingsImageBackground": {},
"settingsViewerQuickActionEditorPageTitle": "Accións rápidas",
@ -1310,6 +1316,8 @@
"@viewerInfoLabelPath": {},
"viewerInfoSearchFieldLabel": "Procurar metadatos",
"@viewerInfoSearchFieldLabel": {},
"filePickerUseThisFolder": "Usar este cartafol",
"@filePickerUseThisFolder": {},
"settingsLanguageSectionTitle": "Idioma e formatos",
"@settingsLanguageSectionTitle": {},
"settingsDisablingBinWarningDialogMessage": "Os elementos no lixo borraranse para sempre.",
@ -1372,6 +1380,8 @@
"@tagEditorSectionRecent": {},
"tagPlaceholderPlace": "Lugar",
"@tagPlaceholderPlace": {},
"filePickerDoNotShowHiddenFiles": "Non amosar arquivos agochados",
"@filePickerDoNotShowHiddenFiles": {},
"viewerInfoLabelAddress": "Enderezo",
"@viewerInfoLabelAddress": {},
"settingsVideoPageTitle": "Axustes de vídeo",
@ -1464,6 +1474,10 @@
"@tagPlaceholderCountry": {},
"tagPlaceholderState": "Estado",
"@tagPlaceholderState": {},
"filePickerShowHiddenFiles": "Amosar arquivos agochados",
"@filePickerShowHiddenFiles": {},
"filePickerNoItems": "Sen elementos",
"@filePickerNoItems": {},
"settingsTimeToTakeActionTile": "Retardo para executar unha acción",
"@settingsTimeToTakeActionTile": {},
"settingsSubtitleThemeBackgroundColor": "Cor de fondo",
@ -1610,33 +1624,5 @@
"panoramaDisableSensorControl": "Desactivar control do sensor",
"@panoramaDisableSensorControl": {},
"settingsHiddenFiltersBanner": "As fotos e vídeos que cadren cos filtros ocultos non se amosarán na súa colección.",
"@settingsHiddenFiltersBanner": {},
"createButtonLabel": "CREAR",
"@createButtonLabel": {},
"chipActionGroup": "Agrupar",
"@chipActionGroup": {},
"chipActionCreateGroup": "Crear grupo",
"@chipActionCreateGroup": {},
"albumTierGroups": "Grupos",
"@albumTierGroups": {},
"newGroupDialogTitle": "Novo grupo",
"@newGroupDialogTitle": {},
"newGroupDialogNameLabel": "Nome do grupo",
"@newGroupDialogNameLabel": {},
"groupAlreadyExists": "Xa existe o grupo",
"@groupAlreadyExists": {},
"groupEmpty": "Sen grupos",
"@groupEmpty": {},
"ungrouped": "Non agrupado",
"@ungrouped": {},
"groupPickerTitle": "Escolmar grupo",
"@groupPickerTitle": {},
"groupPickerUseThisGroupButton": "Usar este grupo",
"@groupPickerUseThisGroupButton": {},
"sectionNone": "Sen seccións",
"@sectionNone": {},
"sortByPath": "Por ruta",
"@sortByPath": {},
"searchFormatSectionTitle": "Formatos",
"@searchFormatSectionTitle": {}
"@settingsHiddenFiltersBanner": {}
}

View file

@ -6,201 +6,5 @@
"welcomeOptional": "אופציונלי",
"@welcomeOptional": {},
"welcomeTermsToggle": "אני מסכימ/ה לתנאים",
"@welcomeTermsToggle": {},
"chipActionShowCollection": "הצג באוסף",
"@chipActionShowCollection": {},
"chipActionGoToAlbumPage": "הצג באלבום",
"@chipActionGoToAlbumPage": {},
"chipActionFilterIn": "סנן לבתוך",
"@chipActionFilterIn": {},
"chipActionFilterOut": "סנן החוצה",
"@chipActionFilterOut": {},
"chipActionPin": "הצמד למעלה",
"@chipActionPin": {},
"chipActionUnpin": "בטל הצמד למעלה",
"@chipActionUnpin": {},
"chipActionRename": "שנה שם",
"@chipActionRename": {},
"applyTooltip": "החל",
"@applyTooltip": {},
"changeTooltip": "שינוי",
"@changeTooltip": {},
"clearTooltip": "ניקוי",
"@clearTooltip": {},
"previousTooltip": "הקודם",
"@previousTooltip": {},
"nextTooltip": "הבא",
"@nextTooltip": {},
"resetTooltip": "אפס",
"@resetTooltip": {},
"saveTooltip": "שמור",
"@saveTooltip": {},
"pickTooltip": "בחר",
"@pickTooltip": {},
"chipActionGoToPlacePage": "הצג במקומות",
"@chipActionGoToPlacePage": {},
"chipActionGoToTagPage": "הצג בתגיות",
"@chipActionGoToTagPage": {},
"chipActionDecompose": "פיצול",
"@chipActionDecompose": {},
"chipActionHide": "הסתר",
"@chipActionHide": {},
"chipActionLock": "נעל",
"@chipActionLock": {},
"applyButtonLabel": "החל",
"@applyButtonLabel": {},
"nextButtonLabel": "הבא",
"@nextButtonLabel": {},
"showButtonLabel": "הצג",
"@showButtonLabel": {},
"hideButtonLabel": "הסתר",
"@hideButtonLabel": {},
"continueButtonLabel": "המשך",
"@continueButtonLabel": {},
"cancelTooltip": "ביטול",
"@cancelTooltip": {},
"columnCount": "{count, plural, =1{{count} עמודה} other{{count} עמודות}}",
"@columnCount": {
"placeholders": {
"count": {
"type": "int",
"format": "decimalPattern"
}
}
},
"itemCount": "{count, plural, =1{{count} פריט} other{{count} פריטים}}",
"@itemCount": {
"placeholders": {
"count": {
"type": "int",
"format": "decimalPattern"
}
}
},
"focalLength": "{length} mm",
"@focalLength": {
"placeholders": {
"length": {
"type": "String",
"example": "5.4"
}
}
},
"deleteButtonLabel": "מחק",
"@deleteButtonLabel": {},
"doubleBackExitMessage": "כדי לצאת הקש שוב על \"חזרה\".",
"@doubleBackExitMessage": {},
"doNotAskAgain": "אל תשאל שוב",
"@doNotAskAgain": {},
"sourceStateLocatingPlaces": "טוען מקומות",
"@sourceStateLocatingPlaces": {},
"chipActionRemove": "הסר",
"@chipActionRemove": {},
"timeSeconds": "{count, plural, =1{{count} שניה} other{{count} שניות}}",
"@timeSeconds": {
"placeholders": {
"count": {
"type": "int",
"format": "decimalPattern"
}
}
},
"timeMinutes": "{count, plural, =1{{count} דקה} other{{count} דקות}}",
"@timeMinutes": {
"placeholders": {
"count": {
"type": "int",
"format": "decimalPattern"
}
}
},
"timeDays": "{count, plural, =1{{count} יום} other{{count} ימים}}",
"@timeDays": {
"placeholders": {
"count": {
"type": "int",
"format": "decimalPattern"
}
}
},
"saveCopyButtonLabel": "שמור עותק",
"@saveCopyButtonLabel": {},
"chipActionGoToCountryPage": "הצג במדינות",
"@chipActionGoToCountryPage": {},
"chipActionDelete": "מחק",
"@chipActionDelete": {},
"hideTooltip": "הסתר",
"@hideTooltip": {},
"showTooltip": "הצג",
"@showTooltip": {},
"sourceStateLoading": "טוען",
"@sourceStateLoading": {},
"sourceStateCataloguing": "מקטלג",
"@sourceStateCataloguing": {},
"actionRemove": "הסרה",
"@actionRemove": {},
"sourceStateLocatingCountries": "טוען מדינות",
"@sourceStateLocatingCountries": {},
"stopTooltip": "עצור",
"@stopTooltip": {},
"chipActionGoToExplorerPage": "הצג בסייר",
"@chipActionGoToExplorerPage": {},
"chipActionSetCover": "הגדר עטיפה",
"@chipActionSetCover": {},
"chipActionCreateAlbum": "צור אלבום",
"@chipActionCreateAlbum": {},
"chipActionShowCountryStates": "הצג סטטיסטיקות",
"@chipActionShowCountryStates": {},
"createButtonLabel": "צור",
"@createButtonLabel": {},
"chipActionGroup": "קבוצה",
"@chipActionGroup": {},
"chipActionCreateGroup": "צור קבוצה",
"@chipActionCreateGroup": {},
"chipActionCreateVault": "צור כספת",
"@chipActionCreateVault": {},
"newGroupDialogTitle": "קבוצה חדשה",
"@newGroupDialogTitle": {},
"groupAlreadyExists": "הקבוצה כבר קיימת",
"@groupAlreadyExists": {},
"entryActionDelete": "מחיקה",
"@entryActionDelete": {},
"entryActionConvert": "המרה",
"@entryActionConvert": {},
"entryActionRotateCCW": "סובב נגד כיוון השעון",
"@entryActionRotateCCW": {},
"entryActionShare": "שיתוף",
"@entryActionShare": {},
"entryActionShareVideoOnly": "שיתוף וידיאו בלבד‍",
"@entryActionShareVideoOnly": {},
"videoActionSelectStreams": "בחר מסלולים",
"@videoActionSelectStreams": {},
"videoActionShowPreviousFrame": "הצג פריים קודם",
"@videoActionShowPreviousFrame": {},
"videoActionShowNextFrame": "הצג פריים הבא",
"@videoActionShowNextFrame": {},
"chipActionConfigureVault": "הגדרת כספת",
"@chipActionConfigureVault": {},
"entryActionCopyToClipboard": "הועתק ללוח",
"@entryActionCopyToClipboard": {},
"entryActionShareImageOnly": "שיתוף תמונה בלבד",
"@entryActionShareImageOnly": {},
"entryActionRotateCW": "סובב עם כיוון השעון",
"@entryActionRotateCW": {},
"entryActionFlip": "הפוך אופקית",
"@entryActionFlip": {},
"entryActionPrint": "הדפסה",
"@entryActionPrint": {},
"entryActionViewSource": "מקור וידאו",
"@entryActionViewSource": {},
"entryActionShowGeoTiffOnMap": "הצג כשכבת מפה",
"@entryActionShowGeoTiffOnMap": {},
"entryActionInfo": "מידע",
"@entryActionInfo": {},
"entryActionExport": "ייצוא",
"@entryActionExport": {},
"entryActionRename": "שינוי שם",
"@entryActionRename": {},
"entryActionRestore": "שחזור",
"@entryActionRestore": {}
"@welcomeTermsToggle": {}
}

View file

@ -864,6 +864,8 @@
"@collectionPageTitle": {},
"collectionGroupMonth": "महीने के अनुसार",
"@collectionGroupMonth": {},
"collectionGroupNone": "समूह न बनाएं",
"@collectionGroupNone": {},
"sectionUnknown": "अज्ञात",
"@sectionUnknown": {},
"dateYesterday": "कल",
@ -1087,6 +1089,8 @@
"@sortByRating": {},
"sortByDuration": "समय के अनुसार",
"@sortByDuration": {},
"albumGroupNone": "ग्रुप न बनाए",
"@albumGroupNone": {},
"albumPickPageTitleCopy": "एल्बम में कॉपी करे",
"@albumPickPageTitleCopy": {},
"albumPickPageTitleExport": "एल्बम में एक्सपोर्ट करे",

View file

@ -291,7 +291,7 @@
"@tileLayoutMosaic": {},
"tileLayoutGrid": "Rács",
"@tileLayoutGrid": {},
"viewDialogGroupSectionTitle": "Szekciók",
"viewDialogGroupSectionTitle": "Csoport",
"@viewDialogGroupSectionTitle": {},
"menuActionStats": "Statisztikák",
"@menuActionStats": {},
@ -433,6 +433,8 @@
"@viewerInfoLabelDate": {},
"viewerInfoUnknown": "ismeretlen",
"@viewerInfoUnknown": {},
"filePickerShowHiddenFiles": "Rejtett fájlok mutatása",
"@filePickerShowHiddenFiles": {},
"sourceViewerPageTitle": "Forrás",
"@sourceViewerPageTitle": {},
"tagPlaceholderPlace": "Hely",
@ -443,6 +445,8 @@
"@tagEditorSectionRecent": {},
"viewerInfoSearchSuggestionResolution": "Felbontás",
"@viewerInfoSearchSuggestionResolution": {},
"filePickerUseThisFolder": "Mappa használata",
"@filePickerUseThisFolder": {},
"tagPlaceholderState": "Megye",
"@tagPlaceholderState": {},
"tagEditorPageAddTagTooltip": "Címke hozzáadása",
@ -451,8 +455,12 @@
"@tagEditorPageNewTagFieldLabel": {},
"tagEditorPageTitle": "Címkék szerkesztése",
"@tagEditorPageTitle": {},
"filePickerNoItems": "Nincsenek elemek",
"@filePickerNoItems": {},
"settingsThumbnailSectionTitle": "Miniatűrök",
"@settingsThumbnailSectionTitle": {},
"filePickerDoNotShowHiddenFiles": "Ne mutassa a rejtett fájlokat",
"@filePickerDoNotShowHiddenFiles": {},
"settingsVideoControlsPageTitle": "Vezérlők",
"@settingsVideoControlsPageTitle": {},
"settingsVideoControlsTile": "Vezérlők",
@ -1054,6 +1062,8 @@
"@collectionGroupMonth": {},
"collectionGroupDay": "Napok szerint",
"@collectionGroupDay": {},
"collectionGroupNone": "Nincs csoportositás",
"@collectionGroupNone": {},
"collectionDeleteFailureFeedback": "{count, plural, =1{Elem törlése sikertelen} other{{count} elem törlése sikertelen}}",
"@collectionDeleteFailureFeedback": {
"placeholders": {
@ -1130,6 +1140,8 @@
"@sortOrderHighestFirst": {},
"sortOrderLowestFirst": "Legalacsonyabb legelöl",
"@sortOrderLowestFirst": {},
"albumGroupNone": "Nincs csoportositás",
"@albumGroupNone": {},
"albumPickPageTitlePick": "Album választása",
"@albumPickPageTitlePick": {},
"searchCollectionFieldHint": "Gyűjtemény keresése",
@ -1330,6 +1342,8 @@
"@mapPointNorthUpTooltip": {},
"viewerInfoSearchSuggestionDimensions": "Méretek",
"@viewerInfoSearchSuggestionDimensions": {},
"filePickerOpenFrom": "Megnyitás innen",
"@filePickerOpenFrom": {},
"panoramaDisableSensorControl": "Szenzoros vezérlés letiltása",
"@panoramaDisableSensorControl": {},
"panoramaEnableSensorControl": "Szenzoros vezérlés engedélyezése",
@ -1594,33 +1608,5 @@
"editEntryLocationDialogTimeShift": "Időeltolódás",
"@editEntryLocationDialogTimeShift": {},
"removeEntryMetadataDialogAll": "Összes",
"@removeEntryMetadataDialogAll": {},
"sortByPath": "Útvonal szerint",
"@sortByPath": {},
"chipActionCreateGroup": "Csoport létrehozása",
"@chipActionCreateGroup": {},
"albumTierGroups": "Csoportok",
"@albumTierGroups": {},
"chipActionGroup": "Csoportosítás",
"@chipActionGroup": {},
"createButtonLabel": "LÉTREHOZÁS",
"@createButtonLabel": {},
"newGroupDialogTitle": "Új csoport",
"@newGroupDialogTitle": {},
"newGroupDialogNameLabel": "Csoport neve",
"@newGroupDialogNameLabel": {},
"groupAlreadyExists": "Csoport már létezik",
"@groupAlreadyExists": {},
"groupEmpty": "Nincsenek csoportok",
"@groupEmpty": {},
"ungrouped": "Csoportosítatlan",
"@ungrouped": {},
"groupPickerTitle": "Válassza ki a csoportot",
"@groupPickerTitle": {},
"groupPickerUseThisGroupButton": "Használja ezt a csoportot",
"@groupPickerUseThisGroupButton": {},
"searchFormatSectionTitle": "Formátumok",
"@searchFormatSectionTitle": {},
"sectionNone": "Semmi szerint",
"@sectionNone": {}
"@removeEntryMetadataDialogAll": {}
}

View file

@ -453,7 +453,7 @@
"@menuActionStats": {},
"viewDialogSortSectionTitle": "Sortir",
"@viewDialogSortSectionTitle": {},
"viewDialogGroupSectionTitle": "Bagian",
"viewDialogGroupSectionTitle": "Grup",
"@viewDialogGroupSectionTitle": {},
"viewDialogLayoutSectionTitle": "Tata letak",
"@viewDialogLayoutSectionTitle": {},
@ -545,6 +545,8 @@
"@collectionGroupMonth": {},
"collectionGroupDay": "Lewat hari",
"@collectionGroupDay": {},
"collectionGroupNone": "Jangan kelompokkan",
"@collectionGroupNone": {},
"sectionUnknown": "Tidak dikenal",
"@sectionUnknown": {},
"dateToday": "Hari ini",
@ -645,6 +647,8 @@
"@albumGroupTier": {},
"albumGroupVolume": "Lewat volume penyimpanan",
"@albumGroupVolume": {},
"albumGroupNone": "Jangan kelompokkan",
"@albumGroupNone": {},
"albumPickPageTitleCopy": "Salin ke Album",
"@albumPickPageTitleCopy": {},
"albumPickPageTitleExport": "Ekspor ke Album",
@ -1081,6 +1085,16 @@
"@panoramaDisableSensorControl": {},
"sourceViewerPageTitle": "Sumber",
"@sourceViewerPageTitle": {},
"filePickerShowHiddenFiles": "Tampilkan file tersembunyi",
"@filePickerShowHiddenFiles": {},
"filePickerDoNotShowHiddenFiles": "Jangan tampilkan file tersembunyi",
"@filePickerDoNotShowHiddenFiles": {},
"filePickerOpenFrom": "Buka dari",
"@filePickerOpenFrom": {},
"filePickerNoItems": "Tidak ada benda",
"@filePickerNoItems": {},
"filePickerUseThisFolder": "Gunakan folder ini",
"@filePickerUseThisFolder": {},
"viewerTransitionNone": "Tidak ada",
"@viewerTransitionNone": {},
"widgetOpenPageHome": "Buka beranda",
@ -1402,33 +1416,5 @@
"editEntryLocationDialogTimeShift": "Pergeseran waktu",
"@editEntryLocationDialogTimeShift": {},
"removeEntryMetadataDialogAll": "Semua",
"@removeEntryMetadataDialogAll": {},
"sortByPath": "Melalui lokasi",
"@sortByPath": {},
"searchFormatSectionTitle": "Format",
"@searchFormatSectionTitle": {},
"sectionNone": "Tidak ada bagian",
"@sectionNone": {},
"albumTierGroups": "Kelompok",
"@albumTierGroups": {},
"createButtonLabel": "BUAT",
"@createButtonLabel": {},
"chipActionGroup": "Kelompok",
"@chipActionGroup": {},
"chipActionCreateGroup": "Buat kelompok",
"@chipActionCreateGroup": {},
"ungrouped": "Tidak dikelompokkan",
"@ungrouped": {},
"newGroupDialogTitle": "Kelompok Baru",
"@newGroupDialogTitle": {},
"newGroupDialogNameLabel": "Nama kelompok",
"@newGroupDialogNameLabel": {},
"groupAlreadyExists": "Kelompok sudah ada",
"@groupAlreadyExists": {},
"groupEmpty": "Tidak ada kelompok",
"@groupEmpty": {},
"groupPickerTitle": "Pilih Kelompok",
"@groupPickerTitle": {},
"groupPickerUseThisGroupButton": "Gunakan kelompok ini",
"@groupPickerUseThisGroupButton": {}
"@removeEntryMetadataDialogAll": {}
}

Some files were not shown because too many files have changed in this diff Show more