diff --git a/docs/src/pages/milestones.tsx b/docs/src/pages/milestones.tsx
index 38dc091a9..6c7e3c585 100644
--- a/docs/src/pages/milestones.tsx
+++ b/docs/src/pages/milestones.tsx
@@ -3,6 +3,7 @@ import {
mdiAndroid,
mdiAppleIos,
mdiArchiveOutline,
+ mdiBash,
mdiBookSearchOutline,
mdiCakeVariant,
mdiCheckAll,
@@ -15,6 +16,7 @@ import {
mdiFile,
mdiFileSearch,
mdiFolder,
+ mdiForum,
mdiHeart,
mdiImage,
mdiImageAlbum,
@@ -41,6 +43,7 @@ import {
mdiText,
mdiThemeLightDark,
mdiTrashCanOutline,
+ mdiVectorCombine,
mdiVideo,
mdiWeb,
} from '@mdi/js';
@@ -49,6 +52,34 @@ import React from 'react';
import Timeline, { DateType, Item } from '../components/timeline';
const items: Item[] = [
+ {
+ icon: mdiVectorCombine,
+ description:
+ 'The serving of the web app is merged into the server image, allowing us to remove two containers from the stack.',
+ title: 'Container consolidation',
+ release: 'v1.88.0',
+ tag: 'v1.88.0',
+ date: new Date(2023, 10, 20),
+ dateType: DateType.RELEASE,
+ },
+ {
+ icon: mdiBash,
+ description: 'Version 2 of the Immich CLI is released, replacing the legacy v1 CLI.',
+ title: 'CLI v2',
+ release: 'v1.88.0',
+ tag: 'v1.88.0',
+ date: new Date(2023, 10, 19),
+ dateType: DateType.RELEASE,
+ },
+ {
+ icon: mdiForum,
+ description: 'Comment a photo or a video in a shared album',
+ title: 'Activity',
+ release: 'v1.84.0',
+ tag: 'v1.84.0',
+ date: new Date(2023, 10, 1),
+ dateType: DateType.RELEASE,
+ },
{
icon: mdiStar,
description: 'Reach 20K Stars on GitHub!',
diff --git a/docs/src/theme/SearchBar/algolia.css b/docs/src/theme/SearchBar/algolia.css
index 8bea784b9..66e696e96 100644
--- a/docs/src/theme/SearchBar/algolia.css
+++ b/docs/src/theme/SearchBar/algolia.css
@@ -61,8 +61,12 @@
.searchbox__input {
display: inline-block;
box-sizing: border-box;
- -webkit-transition: box-shadow 0.4s ease, background 0.4s ease;
- transition: box-shadow 0.4s ease, background 0.4s ease;
+ -webkit-transition:
+ box-shadow 0.4s ease,
+ background 0.4s ease;
+ transition:
+ box-shadow 0.4s ease,
+ background 0.4s ease;
border: 0;
border-radius: 16px;
box-shadow: inset 0 0 0 1px #cccccc;
@@ -243,7 +247,9 @@
}
.algolia-autocomplete .ds-dropdown-menu {
- box-shadow: 0 1px 0 0 rgba(0, 0, 0, 0.2), 0 2px 3px 0 rgba(0, 0, 0, 0.1);
+ box-shadow:
+ 0 1px 0 0 rgba(0, 0, 0, 0.2),
+ 0 2px 3px 0 rgba(0, 0, 0, 0.1);
}
@media (min-width: 601px) {
diff --git a/docs/static/img/immich-screenshots.png b/docs/static/img/immich-screenshots.png
index 7ad188e5a..c8569528d 100644
Binary files a/docs/static/img/immich-screenshots.png and b/docs/static/img/immich-screenshots.png differ
diff --git a/docs/vercel.json b/docs/vercel.json
index f973f913d..d05820ebe 100644
--- a/docs/vercel.json
+++ b/docs/vercel.json
@@ -12,7 +12,8 @@
{ "source": "/docs/overview/logo-meaning", "destination": "/docs/overview/logo" },
{ "source": "/docs/overview/technology-stack", "destination": "/docs/developer/architecture" },
{ "source": "/docs/usage/automatic-backup", "destination": "/docs/features/automatic-backup" },
- { "source": "/docs/usage/bulk-upload", "destination": "/docs/features/bulk-upload" },
+ { "source": "/docs/usage/bulk-upload", "destination": "/docs/features/command-line-interface" },
+ { "source": "/docs/features/bulk-upload", "destination": "/docs/features/command-line-interface" },
{ "source": "/docs/usage/oauth", "destination": "/docs/administration/oauth" },
{ "source": "/docs/usage/post-installation", "destination": "/docs/install/post-install" },
{ "source": "/docs/usage/update", "destination": "/docs/install/docker-compose#step-4---upgrading" },
diff --git a/machine-learning/Dockerfile b/machine-learning/Dockerfile
index d43855320..62a7c86da 100644
--- a/machine-learning/Dockerfile
+++ b/machine-learning/Dockerfile
@@ -1,4 +1,4 @@
-FROM python:3.11-bookworm as builder
+FROM python:3.11-bookworm@sha256:ba7a7ac30c38e119c4304f98ef0e188f90f4f67a958bb6899da9defb99bfb471 as builder
ENV PYTHONDONTWRITEBYTECODE=1 \
PYTHONUNBUFFERED=1 \
@@ -13,7 +13,7 @@ ENV VIRTUAL_ENV="/opt/venv" PATH="/opt/venv/bin:${PATH}"
COPY poetry.lock pyproject.toml ./
RUN poetry install --sync --no-interaction --no-ansi --no-root --only main
-FROM python:3.11-slim-bookworm
+FROM python:3.11-slim-bookworm@sha256:8f82989e563d0dbad057a874a96438a360978c148e34f36c1db8d2d61b5fd6f0
RUN apt-get update && apt-get install -y --no-install-recommends tini libmimalloc2.0 && rm -rf /var/lib/apt/lists/*
diff --git a/machine-learning/app/config.py b/machine-learning/app/config.py
index f3b41d22d..fa4fefeb3 100644
--- a/machine-learning/app/config.py
+++ b/machine-learning/app/config.py
@@ -13,7 +13,8 @@ from .schemas import ModelType
class Settings(BaseSettings):
cache_folder: str = "/cache"
- model_ttl: int = 0
+ model_ttl: int = 300
+ model_ttl_poll_s: int = 10
host: str = "0.0.0.0"
port: int = 3003
workers: int = 1
@@ -38,8 +39,16 @@ class LogSettings(BaseSettings):
_clean_name = str.maketrans(":\\/", "___", ".")
+def clean_name(model_name: str) -> str:
+ return model_name.split("/")[-1].translate(_clean_name)
+
+
def get_cache_dir(model_name: str, model_type: ModelType) -> Path:
- return Path(settings.cache_folder) / model_type.value / model_name.translate(_clean_name)
+ return Path(settings.cache_folder) / model_type.value / clean_name(model_name)
+
+
+def get_hf_model_name(model_name: str) -> str:
+ return f"immich-app/{clean_name(model_name)}"
LOG_LEVELS: dict[str, int] = {
diff --git a/machine-learning/app/conftest.py b/machine-learning/app/conftest.py
index 3bbb89c52..5e2dc1e84 100644
--- a/machine-learning/app/conftest.py
+++ b/machine-learning/app/conftest.py
@@ -36,7 +36,8 @@ def deployed_app() -> TestClient:
@pytest.fixture(scope="session")
def responses() -> dict[str, Any]:
- return json.load(open("responses.json", "r"))
+ responses: dict[str, Any] = json.load(open("responses.json", "r"))
+ return responses
@pytest.fixture(scope="session")
diff --git a/machine-learning/app/main.py b/machine-learning/app/main.py
index 375c14a9e..bf232071b 100644
--- a/machine-learning/app/main.py
+++ b/machine-learning/app/main.py
@@ -1,5 +1,9 @@
import asyncio
+import gc
+import os
+import sys
import threading
+import time
from concurrent.futures import ThreadPoolExecutor
from typing import Any
from zipfile import BadZipFile
@@ -7,7 +11,7 @@ from zipfile import BadZipFile
import orjson
from fastapi import FastAPI, Form, HTTPException, UploadFile
from fastapi.responses import ORJSONResponse
-from onnxruntime.capi.onnxruntime_pybind11_state import InvalidProtobuf, NoSuchFile # type: ignore
+from onnxruntime.capi.onnxruntime_pybind11_state import InvalidProtobuf, NoSuchFile
from starlette.formparsers import MultiPartParser
from app.models.base import InferenceModel
@@ -20,7 +24,7 @@ from .schemas import (
TextResponse,
)
-MultiPartParser.max_file_size = 2**24 # spools to disk if payload is 16 MiB or larger
+MultiPartParser.max_file_size = 2**26 # spools to disk if payload is 64 MiB or larger
app = FastAPI()
@@ -34,7 +38,10 @@ def init_state() -> None:
)
# asyncio is a huge bottleneck for performance, so we use a thread pool to run blocking code
app.state.thread_pool = ThreadPoolExecutor(settings.request_threads) if settings.request_threads > 0 else None
- app.state.locks = {model_type: threading.Lock() for model_type in ModelType}
+ app.state.lock = threading.Lock()
+ app.state.last_called = None
+ if settings.model_ttl > 0 and settings.model_ttl_poll_s > 0:
+ asyncio.ensure_future(idle_shutdown_task())
log.info(f"Initialized request thread pool with {settings.request_threads} threads.")
@@ -79,9 +86,9 @@ async def predict(
async def run(model: InferenceModel, inputs: Any) -> Any:
+ app.state.last_called = time.time()
if app.state.thread_pool is None:
return model.predict(inputs)
-
return await asyncio.get_running_loop().run_in_executor(app.state.thread_pool, model.predict, inputs)
@@ -90,7 +97,7 @@ async def load(model: InferenceModel) -> InferenceModel:
return model
def _load() -> None:
- with app.state.locks[model.model_type]:
+ with app.state.lock:
model.load()
loop = asyncio.get_running_loop()
@@ -113,3 +120,27 @@ async def load(model: InferenceModel) -> InferenceModel:
else:
await loop.run_in_executor(app.state.thread_pool, _load)
return model
+
+
+async def idle_shutdown_task() -> None:
+ while True:
+ log.debug("Checking for inactivity...")
+ if app.state.last_called is not None and time.time() - app.state.last_called > settings.model_ttl:
+ log.info("Shutting down due to inactivity.")
+ loop = asyncio.get_running_loop()
+ for task in asyncio.all_tasks(loop):
+ if task is not asyncio.current_task():
+ try:
+ task.cancel()
+ except asyncio.CancelledError:
+ pass
+ sys.stderr.close()
+ sys.stdout.close()
+ sys.stdout = sys.stderr = open(os.devnull, "w")
+ try:
+ await app.state.model_cache.cache.clear()
+ gc.collect()
+ loop.stop()
+ except asyncio.CancelledError:
+ pass
+ await asyncio.sleep(settings.model_ttl_poll_s)
diff --git a/machine-learning/app/models/__init__.py b/machine-learning/app/models/__init__.py
index a8df0050d..fa00a8614 100644
--- a/machine-learning/app/models/__init__.py
+++ b/machine-learning/app/models/__init__.py
@@ -3,7 +3,8 @@ from typing import Any
from app.schemas import ModelType
from .base import InferenceModel
-from .clip import MCLIPEncoder, OpenCLIPEncoder, is_mclip, is_openclip
+from .clip import MCLIPEncoder, OpenCLIPEncoder
+from .constants import is_insightface, is_mclip, is_openclip
from .facial_recognition import FaceRecognizer
from .image_classification import ImageClassifier
@@ -15,11 +16,12 @@ def from_model_type(model_type: ModelType, model_name: str, **model_kwargs: Any)
return OpenCLIPEncoder(model_name, **model_kwargs)
elif is_mclip(model_name):
return MCLIPEncoder(model_name, **model_kwargs)
- else:
- raise ValueError(f"Unknown CLIP model {model_name}")
case ModelType.FACIAL_RECOGNITION:
- return FaceRecognizer(model_name, **model_kwargs)
+ if is_insightface(model_name):
+ return FaceRecognizer(model_name, **model_kwargs)
case ModelType.IMAGE_CLASSIFICATION:
return ImageClassifier(model_name, **model_kwargs)
case _:
raise ValueError(f"Unknown model type {model_type}")
+
+ raise ValueError(f"Unknown ${model_type} model {model_name}")
diff --git a/machine-learning/app/models/base.py b/machine-learning/app/models/base.py
index 4f597d876..d3252d000 100644
--- a/machine-learning/app/models/base.py
+++ b/machine-learning/app/models/base.py
@@ -7,8 +7,10 @@ from shutil import rmtree
from typing import Any
import onnxruntime as ort
+from huggingface_hub import snapshot_download
+from typing_extensions import Buffer
-from ..config import get_cache_dir, log, settings
+from ..config import get_cache_dir, get_hf_model_name, log, settings
from ..schemas import ModelType
@@ -78,9 +80,13 @@ class InferenceModel(ABC):
def configure(self, **model_kwargs: Any) -> None:
pass
- @abstractmethod
def _download(self) -> None:
- ...
+ snapshot_download(
+ get_hf_model_name(self.model_name),
+ cache_dir=self.cache_dir,
+ local_dir=self.cache_dir,
+ local_dir_use_symlinks=False,
+ )
@abstractmethod
def _load(self) -> None:
@@ -134,11 +140,12 @@ class InferenceModel(ABC):
# HF deep copies configs, so we need to make session options picklable
-class PicklableSessionOptions(ort.SessionOptions):
+class PicklableSessionOptions(ort.SessionOptions): # type: ignore[misc]
def __getstate__(self) -> bytes:
return pickle.dumps([(attr, getattr(self, attr)) for attr in dir(self) if not callable(getattr(self, attr))])
- def __setstate__(self, state: Any) -> None:
- self.__init__() # type: ignore
- for attr, val in pickle.loads(state):
+ def __setstate__(self, state: Buffer) -> None:
+ self.__init__() # type: ignore[misc]
+ attrs: list[tuple[str, Any]] = pickle.loads(state)
+ for attr, val in attrs:
setattr(self, attr, val)
diff --git a/machine-learning/app/models/cache.py b/machine-learning/app/models/cache.py
index bd8b59b3e..1d6a0fc76 100644
--- a/machine-learning/app/models/cache.py
+++ b/machine-learning/app/models/cache.py
@@ -6,7 +6,7 @@ from aiocache.plugins import BasePlugin, TimingPlugin
from app.models import from_model_type
-from ..schemas import ModelType
+from ..schemas import ModelType, has_profiling
from .base import InferenceModel
@@ -50,20 +50,20 @@ class ModelCache:
key = f"{model_name}{model_type.value}{model_kwargs.get('mode', '')}"
async with OptimisticLock(self.cache, key) as lock:
- model = await self.cache.get(key)
+ model: InferenceModel | None = await self.cache.get(key)
if model is None:
model = from_model_type(model_type, model_name, **model_kwargs)
await lock.cas(model, ttl=self.ttl)
return model
async def get_profiling(self) -> dict[str, float] | None:
- if not hasattr(self.cache, "profiling"):
+ if not has_profiling(self.cache):
return None
- return self.cache.profiling # type: ignore
+ return self.cache.profiling
-class RevalidationPlugin(BasePlugin):
+class RevalidationPlugin(BasePlugin): # type: ignore[misc]
"""Revalidates cache item's TTL after cache hit."""
async def post_get(
diff --git a/machine-learning/app/models/clip.py b/machine-learning/app/models/clip.py
index da0381d3a..1dee967de 100644
--- a/machine-learning/app/models/clip.py
+++ b/machine-learning/app/models/clip.py
@@ -7,11 +7,10 @@ from typing import Any, Literal
import numpy as np
import onnxruntime as ort
-from huggingface_hub import snapshot_download
from PIL import Image
from transformers import AutoTokenizer
-from app.config import log
+from app.config import clean_name, log
from app.models.transforms import crop, get_pil_resampling, normalize, resize, to_numpy
from app.schemas import ModelType, ndarray_f32, ndarray_i32, ndarray_i64
@@ -52,7 +51,7 @@ class BaseCLIPEncoder(InferenceModel):
provider_options=self.provider_options,
)
- def _predict(self, image_or_text: Image.Image | str) -> list[float]:
+ def _predict(self, image_or_text: Image.Image | str) -> ndarray_f32:
if isinstance(image_or_text, bytes):
image_or_text = Image.open(BytesIO(image_or_text))
@@ -61,16 +60,16 @@ class BaseCLIPEncoder(InferenceModel):
if self.mode == "text":
raise TypeError("Cannot encode image as text-only model")
- outputs = self.vision_model.run(None, self.transform(image_or_text))
+ outputs: ndarray_f32 = self.vision_model.run(None, self.transform(image_or_text))[0][0]
case str():
if self.mode == "vision":
raise TypeError("Cannot encode text as vision-only model")
- outputs = self.text_model.run(None, self.tokenize(image_or_text))
+ outputs = self.text_model.run(None, self.tokenize(image_or_text))[0][0]
case _:
raise TypeError(f"Expected Image or str, but got: {type(image_or_text)}")
- return outputs[0][0].tolist()
+ return outputs
@abstractmethod
def tokenize(self, text: str) -> dict[str, ndarray_i32]:
@@ -117,15 +116,7 @@ class OpenCLIPEncoder(BaseCLIPEncoder):
mode: Literal["text", "vision"] | None = None,
**model_kwargs: Any,
) -> None:
- super().__init__(_clean_model_name(model_name), cache_dir, mode, **model_kwargs)
-
- def _download(self) -> None:
- snapshot_download(
- f"immich-app/{self.model_name}",
- cache_dir=self.cache_dir,
- local_dir=self.cache_dir,
- local_dir_use_symlinks=False,
- )
+ super().__init__(clean_name(model_name), cache_dir, mode, **model_kwargs)
def _load(self) -> None:
super()._load()
@@ -160,63 +151,16 @@ class OpenCLIPEncoder(BaseCLIPEncoder):
@cached_property
def model_cfg(self) -> dict[str, Any]:
- return json.load(self.model_cfg_path.open())
+ model_cfg: dict[str, Any] = json.load(self.model_cfg_path.open())
+ return model_cfg
@cached_property
def preprocess_cfg(self) -> dict[str, Any]:
- return json.load(self.preprocess_cfg_path.open())
+ preprocess_cfg: dict[str, Any] = json.load(self.preprocess_cfg_path.open())
+ return preprocess_cfg
class MCLIPEncoder(OpenCLIPEncoder):
def tokenize(self, text: str) -> dict[str, ndarray_i32]:
tokens: dict[str, ndarray_i64] = self.tokenizer(text, return_tensors="np")
return {k: v.astype(np.int32) for k, v in tokens.items()}
-
-
-_OPENCLIP_MODELS = {
- "RN50__openai",
- "RN50__yfcc15m",
- "RN50__cc12m",
- "RN101__openai",
- "RN101__yfcc15m",
- "RN50x4__openai",
- "RN50x16__openai",
- "RN50x64__openai",
- "ViT-B-32__openai",
- "ViT-B-32__laion2b_e16",
- "ViT-B-32__laion400m_e31",
- "ViT-B-32__laion400m_e32",
- "ViT-B-32__laion2b-s34b-b79k",
- "ViT-B-16__openai",
- "ViT-B-16__laion400m_e31",
- "ViT-B-16__laion400m_e32",
- "ViT-B-16-plus-240__laion400m_e31",
- "ViT-B-16-plus-240__laion400m_e32",
- "ViT-L-14__openai",
- "ViT-L-14__laion400m_e31",
- "ViT-L-14__laion400m_e32",
- "ViT-L-14__laion2b-s32b-b82k",
- "ViT-L-14-336__openai",
- "ViT-H-14__laion2b-s32b-b79k",
- "ViT-g-14__laion2b-s12b-b42k",
-}
-
-
-_MCLIP_MODELS = {
- "LABSE-Vit-L-14",
- "XLM-Roberta-Large-Vit-B-32",
- "XLM-Roberta-Large-Vit-B-16Plus",
- "XLM-Roberta-Large-Vit-L-14",
-}
-
-
-def _clean_model_name(model_name: str) -> str:
- return model_name.split("/")[-1].replace("::", "__")
-
-
-def is_openclip(model_name: str) -> bool:
- return _clean_model_name(model_name) in _OPENCLIP_MODELS
-
-
-def is_mclip(model_name: str) -> bool:
- return _clean_model_name(model_name) in _MCLIP_MODELS
diff --git a/machine-learning/app/models/constants.py b/machine-learning/app/models/constants.py
new file mode 100644
index 000000000..53f3f3381
--- /dev/null
+++ b/machine-learning/app/models/constants.py
@@ -0,0 +1,57 @@
+from app.config import clean_name
+
+_OPENCLIP_MODELS = {
+ "RN50__openai",
+ "RN50__yfcc15m",
+ "RN50__cc12m",
+ "RN101__openai",
+ "RN101__yfcc15m",
+ "RN50x4__openai",
+ "RN50x16__openai",
+ "RN50x64__openai",
+ "ViT-B-32__openai",
+ "ViT-B-32__laion2b_e16",
+ "ViT-B-32__laion400m_e31",
+ "ViT-B-32__laion400m_e32",
+ "ViT-B-32__laion2b-s34b-b79k",
+ "ViT-B-16__openai",
+ "ViT-B-16__laion400m_e31",
+ "ViT-B-16__laion400m_e32",
+ "ViT-B-16-plus-240__laion400m_e31",
+ "ViT-B-16-plus-240__laion400m_e32",
+ "ViT-L-14__openai",
+ "ViT-L-14__laion400m_e31",
+ "ViT-L-14__laion400m_e32",
+ "ViT-L-14__laion2b-s32b-b82k",
+ "ViT-L-14-336__openai",
+ "ViT-H-14__laion2b-s32b-b79k",
+ "ViT-g-14__laion2b-s12b-b42k",
+}
+
+
+_MCLIP_MODELS = {
+ "LABSE-Vit-L-14",
+ "XLM-Roberta-Large-Vit-B-32",
+ "XLM-Roberta-Large-Vit-B-16Plus",
+ "XLM-Roberta-Large-Vit-L-14",
+}
+
+
+_INSIGHTFACE_MODELS = {
+ "antelopev2",
+ "buffalo_l",
+ "buffalo_m",
+ "buffalo_s",
+}
+
+
+def is_openclip(model_name: str) -> bool:
+ return clean_name(model_name) in _OPENCLIP_MODELS
+
+
+def is_mclip(model_name: str) -> bool:
+ return clean_name(model_name) in _MCLIP_MODELS
+
+
+def is_insightface(model_name: str) -> bool:
+ return clean_name(model_name) in _INSIGHTFACE_MODELS
diff --git a/machine-learning/app/models/facial_recognition.py b/machine-learning/app/models/facial_recognition.py
index 2ea7fdf67..24719eb83 100644
--- a/machine-learning/app/models/facial_recognition.py
+++ b/machine-learning/app/models/facial_recognition.py
@@ -1,4 +1,3 @@
-import zipfile
from pathlib import Path
from typing import Any
@@ -7,9 +6,9 @@ import numpy as np
import onnxruntime as ort
from insightface.model_zoo import ArcFaceONNX, RetinaFace
from insightface.utils.face_align import norm_crop
-from insightface.utils.storage import BASE_REPO_URL, download_file
-from app.schemas import ModelType, ndarray_f32
+from app.config import clean_name
+from app.schemas import BoundingBox, Face, ModelType, ndarray_f32
from .base import InferenceModel
@@ -25,37 +24,21 @@ class FaceRecognizer(InferenceModel):
**model_kwargs: Any,
) -> None:
self.min_score = model_kwargs.pop("minScore", min_score)
- super().__init__(model_name, cache_dir, **model_kwargs)
-
- def _download(self) -> None:
- zip_file = self.cache_dir / f"{self.model_name}.zip"
- download_file(f"{BASE_REPO_URL}/{self.model_name}.zip", zip_file)
- with zipfile.ZipFile(zip_file, "r") as zip:
- members = zip.namelist()
- det_file = next(model for model in members if model.startswith("det_"))
- rec_file = next(model for model in members if model.startswith("w600k_"))
- zip.extractall(self.cache_dir, members=[det_file, rec_file])
- zip_file.unlink()
+ super().__init__(clean_name(model_name), cache_dir, **model_kwargs)
def _load(self) -> None:
- try:
- det_file = next(self.cache_dir.glob("det_*.onnx"))
- rec_file = next(self.cache_dir.glob("w600k_*.onnx"))
- except StopIteration:
- raise FileNotFoundError("Facial recognition models not found in cache directory")
-
self.det_model = RetinaFace(
session=ort.InferenceSession(
- det_file.as_posix(),
+ self.det_file.as_posix(),
sess_options=self.sess_options,
providers=self.providers,
provider_options=self.provider_options,
),
)
self.rec_model = ArcFaceONNX(
- rec_file.as_posix(),
+ self.rec_file.as_posix(),
session=ort.InferenceSession(
- rec_file.as_posix(),
+ self.rec_file.as_posix(),
sess_options=self.sess_options,
providers=self.providers,
provider_options=self.provider_options,
@@ -69,7 +52,7 @@ class FaceRecognizer(InferenceModel):
)
self.rec_model.prepare(ctx_id=0)
- def _predict(self, image: ndarray_f32 | bytes) -> list[dict[str, Any]]:
+ def _predict(self, image: ndarray_f32 | bytes) -> list[Face]:
if isinstance(image, bytes):
image = cv2.imdecode(np.frombuffer(image, np.uint8), cv2.IMREAD_COLOR)
bboxes, kpss = self.det_model.detect(image)
@@ -84,26 +67,33 @@ class FaceRecognizer(InferenceModel):
height, width, _ = image.shape
for (x1, y1, x2, y2), score, kps in zip(bboxes, scores, kpss):
cropped_img = norm_crop(image, kps)
- embedding = self.rec_model.get_feat(cropped_img)[0].tolist()
- results.append(
- {
- "imageWidth": width,
- "imageHeight": height,
- "boundingBox": {
- "x1": x1,
- "y1": y1,
- "x2": x2,
- "y2": y2,
- },
- "score": score,
- "embedding": embedding,
- }
- )
+ embedding: ndarray_f32 = self.rec_model.get_feat(cropped_img)[0]
+ face: Face = {
+ "imageWidth": width,
+ "imageHeight": height,
+ "boundingBox": {
+ "x1": x1,
+ "y1": y1,
+ "x2": x2,
+ "y2": y2,
+ },
+ "score": score,
+ "embedding": embedding,
+ }
+ results.append(face)
return results
@property
def cached(self) -> bool:
- return self.cache_dir.is_dir() and any(self.cache_dir.glob("*.onnx"))
+ return self.det_file.is_file() and self.rec_file.is_file()
+
+ @property
+ def det_file(self) -> Path:
+ return self.cache_dir / "detection" / "model.onnx"
+
+ @property
+ def rec_file(self) -> Path:
+ return self.cache_dir / "recognition" / "model.onnx"
def configure(self, **model_kwargs: Any) -> None:
self.det_model.det_thresh = model_kwargs.pop("minScore", self.det_model.det_thresh)
diff --git a/machine-learning/app/models/image_classification.py b/machine-learning/app/models/image_classification.py
index cbf784e5a..b8c38327c 100644
--- a/machine-learning/app/models/image_classification.py
+++ b/machine-learning/app/models/image_classification.py
@@ -66,7 +66,7 @@ class ImageClassifier(InferenceModel):
def _predict(self, image: Image.Image | bytes) -> list[str]:
if isinstance(image, bytes):
image = Image.open(BytesIO(image))
- predictions: list[dict[str, Any]] = self.model(image) # type: ignore
+ predictions: list[dict[str, Any]] = self.model(image)
tags = [tag for pred in predictions for tag in pred["label"].split(", ") if pred["score"] >= self.min_score]
return tags
diff --git a/machine-learning/app/schemas.py b/machine-learning/app/schemas.py
index ad1faac8c..9e7f62fc8 100644
--- a/machine-learning/app/schemas.py
+++ b/machine-learning/app/schemas.py
@@ -1,17 +1,12 @@
from enum import StrEnum
-from typing import TypeAlias
+from typing import Any, Protocol, TypeAlias, TypedDict, TypeGuard
import numpy as np
from pydantic import BaseModel
-
-def to_lower_camel(string: str) -> str:
- tokens = [token.capitalize() if i > 0 else token for i, token in enumerate(string.split("_"))]
- return "".join(tokens)
-
-
-class TextModelRequest(BaseModel):
- text: str
+ndarray_f32: TypeAlias = np.ndarray[int, np.dtype[np.float32]]
+ndarray_i64: TypeAlias = np.ndarray[int, np.dtype[np.int64]]
+ndarray_i32: TypeAlias = np.ndarray[int, np.dtype[np.int32]]
class TextResponse(BaseModel):
@@ -22,7 +17,7 @@ class MessageResponse(BaseModel):
message: str
-class BoundingBox(BaseModel):
+class BoundingBox(TypedDict):
x1: int
y1: int
x2: int
@@ -35,6 +30,17 @@ class ModelType(StrEnum):
FACIAL_RECOGNITION = "facial-recognition"
-ndarray_f32: TypeAlias = np.ndarray[int, np.dtype[np.float32]]
-ndarray_i64: TypeAlias = np.ndarray[int, np.dtype[np.int64]]
-ndarray_i32: TypeAlias = np.ndarray[int, np.dtype[np.int32]]
+class HasProfiling(Protocol):
+ profiling: dict[str, float]
+
+
+class Face(TypedDict):
+ boundingBox: BoundingBox
+ embedding: ndarray_f32
+ imageWidth: int
+ imageHeight: int
+ score: float
+
+
+def has_profiling(obj: Any) -> TypeGuard[HasProfiling]:
+ return hasattr(obj, "profiling") and type(obj.profiling) == dict
diff --git a/machine-learning/app/test_main.py b/machine-learning/app/test_main.py
index 0b28f8234..318c3b045 100644
--- a/machine-learning/app/test_main.py
+++ b/machine-learning/app/test_main.py
@@ -75,9 +75,9 @@ class TestCLIP:
embedding = clip_encoder.predict(pil_image)
assert clip_encoder.mode == "vision"
- assert isinstance(embedding, list)
- assert len(embedding) == clip_model_cfg["embed_dim"]
- assert all([isinstance(num, float) for num in embedding])
+ assert isinstance(embedding, np.ndarray)
+ assert embedding.shape[0] == clip_model_cfg["embed_dim"]
+ assert embedding.dtype == np.float32
clip_encoder.vision_model.run.assert_called_once()
def test_basic_text(
@@ -97,22 +97,22 @@ class TestCLIP:
embedding = clip_encoder.predict("test search query")
assert clip_encoder.mode == "text"
- assert isinstance(embedding, list)
- assert len(embedding) == clip_model_cfg["embed_dim"]
- assert all([isinstance(num, float) for num in embedding])
+ assert isinstance(embedding, np.ndarray)
+ assert embedding.shape[0] == clip_model_cfg["embed_dim"]
+ assert embedding.dtype == np.float32
clip_encoder.text_model.run.assert_called_once()
class TestFaceRecognition:
def test_set_min_score(self, mocker: MockerFixture) -> None:
mocker.patch.object(FaceRecognizer, "load")
- face_recognizer = FaceRecognizer("test_model_name", cache_dir="test_cache", min_score=0.5)
+ face_recognizer = FaceRecognizer("buffalo_s", cache_dir="test_cache", min_score=0.5)
assert face_recognizer.min_score == 0.5
def test_basic(self, cv_image: cv2.Mat, mocker: MockerFixture) -> None:
mocker.patch.object(FaceRecognizer, "load")
- face_recognizer = FaceRecognizer("test_model_name", min_score=0.0, cache_dir="test_cache")
+ face_recognizer = FaceRecognizer("buffalo_s", min_score=0.0, cache_dir="test_cache")
det_model = mock.Mock()
num_faces = 2
@@ -133,9 +133,9 @@ class TestFaceRecognition:
for face in faces:
assert face["imageHeight"] == 800
assert face["imageWidth"] == 600
- assert isinstance(face["embedding"], list)
- assert len(face["embedding"]) == 512
- assert all([isinstance(num, float) for num in face["embedding"]])
+ assert isinstance(face["embedding"], np.ndarray)
+ assert face["embedding"].shape[0] == 512
+ assert face["embedding"].dtype == np.float32
det_model.detect.assert_called_once()
assert rec_model.get_feat.call_count == num_faces
diff --git a/machine-learning/export/Dockerfile b/machine-learning/export/Dockerfile
index 1c74546bc..7bd04e0c2 100644
--- a/machine-learning/export/Dockerfile
+++ b/machine-learning/export/Dockerfile
@@ -1,4 +1,4 @@
-FROM mambaorg/micromamba:bookworm-slim as builder
+FROM mambaorg/micromamba:bookworm-slim@sha256:d20c621f3ae42f50f380166b15b6c88b14fa62ab6ea188f2cef33451d64057c7 as builder
ENV NODE_ENV=production \
TRANSFORMERS_CACHE=/cache \
diff --git a/machine-learning/export/models/openclip.py b/machine-learning/export/models/openclip.py
index c29dafce7..46c11cb4e 100644
--- a/machine-learning/export/models/openclip.py
+++ b/machine-learning/export/models/openclip.py
@@ -1,6 +1,7 @@
import tempfile
import warnings
from dataclasses import dataclass, field
+from math import e
from pathlib import Path
import open_clip
@@ -69,10 +70,12 @@ def export_image_encoder(model: open_clip.CLIP, model_cfg: OpenCLIPModelConfig,
output_path = Path(output_path)
def encode_image(image: torch.Tensor) -> torch.Tensor:
- return model.encode_image(image, normalize=True)
+ output = model.encode_image(image, normalize=True)
+ assert isinstance(output, torch.Tensor)
+ return output
args = (torch.randn(1, 3, model_cfg.image_size, model_cfg.image_size),)
- traced = torch.jit.trace(encode_image, args)
+ traced = torch.jit.trace(encode_image, args) # type: ignore[no-untyped-call]
with warnings.catch_warnings():
warnings.simplefilter("ignore", UserWarning)
@@ -91,10 +94,12 @@ def export_text_encoder(model: open_clip.CLIP, model_cfg: OpenCLIPModelConfig, o
output_path = Path(output_path)
def encode_text(text: torch.Tensor) -> torch.Tensor:
- return model.encode_text(text, normalize=True)
+ output = model.encode_text(text, normalize=True)
+ assert isinstance(output, torch.Tensor)
+ return output
args = (torch.ones(1, model_cfg.sequence_length, dtype=torch.int32),)
- traced = torch.jit.trace(encode_text, args)
+ traced = torch.jit.trace(encode_text, args) # type: ignore[no-untyped-call]
with warnings.catch_warnings():
warnings.simplefilter("ignore", UserWarning)
diff --git a/machine-learning/poetry.lock b/machine-learning/poetry.lock
index 2b5a11e8b..e9ec397ff 100644
--- a/machine-learning/poetry.lock
+++ b/machine-learning/poetry.lock
@@ -585,68 +585,6 @@ files = [
test = ["PyYAML", "mock", "pytest"]
yaml = ["PyYAML"]
-[[package]]
-name = "contourpy"
-version = "1.1.0"
-description = "Python library for calculating contours of 2D quadrilateral grids"
-optional = false
-python-versions = ">=3.8"
-files = [
- {file = "contourpy-1.1.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:89f06eff3ce2f4b3eb24c1055a26981bffe4e7264acd86f15b97e40530b794bc"},
- {file = "contourpy-1.1.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:dffcc2ddec1782dd2f2ce1ef16f070861af4fb78c69862ce0aab801495dda6a3"},
- {file = "contourpy-1.1.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:25ae46595e22f93592d39a7eac3d638cda552c3e1160255258b695f7b58e5655"},
- {file = "contourpy-1.1.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:17cfaf5ec9862bc93af1ec1f302457371c34e688fbd381f4035a06cd47324f48"},
- {file = "contourpy-1.1.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:18a64814ae7bce73925131381603fff0116e2df25230dfc80d6d690aa6e20b37"},
- {file = "contourpy-1.1.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:90c81f22b4f572f8a2110b0b741bb64e5a6427e0a198b2cdc1fbaf85f352a3aa"},
- {file = "contourpy-1.1.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:53cc3a40635abedbec7f1bde60f8c189c49e84ac180c665f2cd7c162cc454baa"},
- {file = "contourpy-1.1.0-cp310-cp310-win32.whl", hash = "sha256:9b2dd2ca3ac561aceef4c7c13ba654aaa404cf885b187427760d7f7d4c57cff8"},
- {file = "contourpy-1.1.0-cp310-cp310-win_amd64.whl", hash = "sha256:1f795597073b09d631782e7245016a4323cf1cf0b4e06eef7ea6627e06a37ff2"},
- {file = "contourpy-1.1.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:0b7b04ed0961647691cfe5d82115dd072af7ce8846d31a5fac6c142dcce8b882"},
- {file = "contourpy-1.1.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:27bc79200c742f9746d7dd51a734ee326a292d77e7d94c8af6e08d1e6c15d545"},
- {file = "contourpy-1.1.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:052cc634bf903c604ef1a00a5aa093c54f81a2612faedaa43295809ffdde885e"},
- {file = "contourpy-1.1.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9382a1c0bc46230fb881c36229bfa23d8c303b889b788b939365578d762b5c18"},
- {file = "contourpy-1.1.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e5cec36c5090e75a9ac9dbd0ff4a8cf7cecd60f1b6dc23a374c7d980a1cd710e"},
- {file = "contourpy-1.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1f0cbd657e9bde94cd0e33aa7df94fb73c1ab7799378d3b3f902eb8eb2e04a3a"},
- {file = "contourpy-1.1.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:181cbace49874f4358e2929aaf7ba84006acb76694102e88dd15af861996c16e"},
- {file = "contourpy-1.1.0-cp311-cp311-win32.whl", hash = "sha256:edb989d31065b1acef3828a3688f88b2abb799a7db891c9e282df5ec7e46221b"},
- {file = "contourpy-1.1.0-cp311-cp311-win_amd64.whl", hash = "sha256:fb3b7d9e6243bfa1efb93ccfe64ec610d85cfe5aec2c25f97fbbd2e58b531256"},
- {file = "contourpy-1.1.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:bcb41692aa09aeb19c7c213411854402f29f6613845ad2453d30bf421fe68fed"},
- {file = "contourpy-1.1.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:5d123a5bc63cd34c27ff9c7ac1cd978909e9c71da12e05be0231c608048bb2ae"},
- {file = "contourpy-1.1.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:62013a2cf68abc80dadfd2307299bfa8f5aa0dcaec5b2954caeb5fa094171103"},
- {file = "contourpy-1.1.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:0b6616375d7de55797d7a66ee7d087efe27f03d336c27cf1f32c02b8c1a5ac70"},
- {file = "contourpy-1.1.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:317267d915490d1e84577924bd61ba71bf8681a30e0d6c545f577363157e5e94"},
- {file = "contourpy-1.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d551f3a442655f3dcc1285723f9acd646ca5858834efeab4598d706206b09c9f"},
- {file = "contourpy-1.1.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:e7a117ce7df5a938fe035cad481b0189049e8d92433b4b33aa7fc609344aafa1"},
- {file = "contourpy-1.1.0-cp38-cp38-win32.whl", hash = "sha256:108dfb5b3e731046a96c60bdc46a1a0ebee0760418951abecbe0fc07b5b93b27"},
- {file = "contourpy-1.1.0-cp38-cp38-win_amd64.whl", hash = "sha256:d4f26b25b4f86087e7d75e63212756c38546e70f2a92d2be44f80114826e1cd4"},
- {file = "contourpy-1.1.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:bc00bb4225d57bff7ebb634646c0ee2a1298402ec10a5fe7af79df9a51c1bfd9"},
- {file = "contourpy-1.1.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:189ceb1525eb0655ab8487a9a9c41f42a73ba52d6789754788d1883fb06b2d8a"},
- {file = "contourpy-1.1.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9f2931ed4741f98f74b410b16e5213f71dcccee67518970c42f64153ea9313b9"},
- {file = "contourpy-1.1.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:30f511c05fab7f12e0b1b7730ebdc2ec8deedcfb505bc27eb570ff47c51a8f15"},
- {file = "contourpy-1.1.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:143dde50520a9f90e4a2703f367cf8ec96a73042b72e68fcd184e1279962eb6f"},
- {file = "contourpy-1.1.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e94bef2580e25b5fdb183bf98a2faa2adc5b638736b2c0a4da98691da641316a"},
- {file = "contourpy-1.1.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:ed614aea8462735e7d70141374bd7650afd1c3f3cb0c2dbbcbe44e14331bf002"},
- {file = "contourpy-1.1.0-cp39-cp39-win32.whl", hash = "sha256:71551f9520f008b2950bef5f16b0e3587506ef4f23c734b71ffb7b89f8721999"},
- {file = "contourpy-1.1.0-cp39-cp39-win_amd64.whl", hash = "sha256:438ba416d02f82b692e371858143970ed2eb6337d9cdbbede0d8ad9f3d7dd17d"},
- {file = "contourpy-1.1.0-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:a698c6a7a432789e587168573a864a7ea374c6be8d4f31f9d87c001d5a843493"},
- {file = "contourpy-1.1.0-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:397b0ac8a12880412da3551a8cb5a187d3298a72802b45a3bd1805e204ad8439"},
- {file = "contourpy-1.1.0-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:a67259c2b493b00e5a4d0f7bfae51fb4b3371395e47d079a4446e9b0f4d70e76"},
- {file = "contourpy-1.1.0-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:2b836d22bd2c7bb2700348e4521b25e077255ebb6ab68e351ab5aa91ca27e027"},
- {file = "contourpy-1.1.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:084eaa568400cfaf7179b847ac871582199b1b44d5699198e9602ecbbb5f6104"},
- {file = "contourpy-1.1.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:911ff4fd53e26b019f898f32db0d4956c9d227d51338fb3b03ec72ff0084ee5f"},
- {file = "contourpy-1.1.0.tar.gz", hash = "sha256:e53046c3863828d21d531cc3b53786e6580eb1ba02477e8681009b6aa0870b21"},
-]
-
-[package.dependencies]
-numpy = ">=1.16"
-
-[package.extras]
-bokeh = ["bokeh", "selenium"]
-docs = ["furo", "sphinx-copybutton"]
-mypy = ["contourpy[bokeh,docs]", "docutils-stubs", "mypy (==1.2.0)", "types-Pillow"]
-test = ["Pillow", "contourpy[test-no-images]", "matplotlib"]
-test-no-images = ["pytest", "pytest-cov", "wurlitzer"]
-
[[package]]
name = "contourpy"
version = "1.1.1"
@@ -2408,35 +2346,35 @@ reference = ["Pillow", "google-re2"]
[[package]]
name = "onnxruntime"
-version = "1.16.1"
+version = "1.16.2"
description = "ONNX Runtime is a runtime accelerator for Machine Learning models"
optional = false
python-versions = "*"
files = [
- {file = "onnxruntime-1.16.1-cp310-cp310-macosx_10_15_x86_64.whl", hash = "sha256:28b2c7f444b4119950b69370801cd66067f403d19cbaf2a444735d7c269cce4a"},
- {file = "onnxruntime-1.16.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:c24e04f33e7899f6aebb03ed51e51d346c1f906b05c5569d58ac9a12d38a2f58"},
- {file = "onnxruntime-1.16.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9fa93b166f2d97063dc9f33c5118c5729a4a5dd5617296b6dbef42f9047b3e81"},
- {file = "onnxruntime-1.16.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:042dd9201b3016ee18f8f8bc4609baf11ff34ca1ff489c0a46bcd30919bf883d"},
- {file = "onnxruntime-1.16.1-cp310-cp310-win32.whl", hash = "sha256:c20aa0591f305012f1b21aad607ed96917c86ae7aede4a4dd95824b3d124ceb7"},
- {file = "onnxruntime-1.16.1-cp310-cp310-win_amd64.whl", hash = "sha256:5581873e578917bea76d6434ee7337e28195d03488dcf72d161d08e9398c6249"},
- {file = "onnxruntime-1.16.1-cp311-cp311-macosx_10_15_x86_64.whl", hash = "sha256:ef8c0c8abf5f309aa1caf35941380839dc5f7a2fa53da533be4a3f254993f120"},
- {file = "onnxruntime-1.16.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:e680380bea35a137cbc3efd67a17486e96972901192ad3026ee79c8d8fe264f7"},
- {file = "onnxruntime-1.16.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5e62cc38ce1a669013d0a596d984762dc9c67c56f60ecfeee0d5ad36da5863f6"},
- {file = "onnxruntime-1.16.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:025c7a4d57bd2e63b8a0f84ad3df53e419e3df1cc72d63184f2aae807b17c13c"},
- {file = "onnxruntime-1.16.1-cp311-cp311-win32.whl", hash = "sha256:9ad074057fa8d028df248b5668514088cb0937b6ac5954073b7fb9b2891ffc8c"},
- {file = "onnxruntime-1.16.1-cp311-cp311-win_amd64.whl", hash = "sha256:d5e43a3478bffc01f817ecf826de7b25a2ca1bca8547d70888594ab80a77ad24"},
- {file = "onnxruntime-1.16.1-cp38-cp38-macosx_10_15_x86_64.whl", hash = "sha256:3aef4d70b0930e29a8943eab248cd1565664458d3a62b2276bd11181f28fd0a3"},
- {file = "onnxruntime-1.16.1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:55a7b843a57c8ca0c8ff169428137958146081d5d76f1a6dd444c4ffcd37c3c2"},
- {file = "onnxruntime-1.16.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:62c631af1941bf3b5f7d063d24c04aacce8cff0794e157c497e315e89ac5ad7b"},
- {file = "onnxruntime-1.16.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5671f296c3d5c233f601e97a10ab5a1dd8e65ba35c7b7b0c253332aba9dff330"},
- {file = "onnxruntime-1.16.1-cp38-cp38-win32.whl", hash = "sha256:eb3802305023dd05e16848d4e22b41f8147247894309c0c27122aaa08793b3d2"},
- {file = "onnxruntime-1.16.1-cp38-cp38-win_amd64.whl", hash = "sha256:fecfb07443d09d271b1487f401fbdf1ba0c829af6fd4fe8f6af25f71190e7eb9"},
- {file = "onnxruntime-1.16.1-cp39-cp39-macosx_10_15_x86_64.whl", hash = "sha256:de3e12094234db6545c67adbf801874b4eb91e9f299bda34c62967ef0050960f"},
- {file = "onnxruntime-1.16.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:ff723c2a5621b5e7103f3be84d5aae1e03a20621e72219dddceae81f65f240af"},
- {file = "onnxruntime-1.16.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:14a7fb3073aaf6b462e3d7fb433320f7700558a8892e5021780522dc4574292a"},
- {file = "onnxruntime-1.16.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:963159f1f699b0454cd72fcef3276c8a1aab9389a7b301bcd8e320fb9d9e8597"},
- {file = "onnxruntime-1.16.1-cp39-cp39-win32.whl", hash = "sha256:85771adb75190db9364b25ddec353ebf07635b83eb94b64ed014f1f6d57a3857"},
- {file = "onnxruntime-1.16.1-cp39-cp39-win_amd64.whl", hash = "sha256:d32d2b30799c1f950123c60ae8390818381fd5f88bdf3627eeca10071c155dc5"},
+ {file = "onnxruntime-1.16.2-cp310-cp310-macosx_10_15_x86_64.whl", hash = "sha256:e19316bb15c29ca0397e78861ee7cdb4db763ac5c53eaa83169bcdcb1149878c"},
+ {file = "onnxruntime-1.16.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:773f6d99d1e6a58936a55a4933c66674241dace9ec4bab71664cdfa170a7cd87"},
+ {file = "onnxruntime-1.16.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e8b8df9583a6e874f1983b85a361d22c205c96e926626eb486d3e69d72642f79"},
+ {file = "onnxruntime-1.16.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ceef600de846997e3ef5f9af956ae87c88d84d6e925c3e9d435ce17ea223568f"},
+ {file = "onnxruntime-1.16.2-cp310-cp310-win32.whl", hash = "sha256:4fed41edb766c6adea6c34f1eb63a344d697fd4625133e5e48f23950bce60803"},
+ {file = "onnxruntime-1.16.2-cp310-cp310-win_amd64.whl", hash = "sha256:9fc410ec220804fb384e7cb4fd68c474d89da11a1b68184db2001d64ba1477a9"},
+ {file = "onnxruntime-1.16.2-cp311-cp311-macosx_10_15_x86_64.whl", hash = "sha256:aa09d8d9d9a4dc2f6647b5135bb540da36e2d78206aaf14140ba73e05928c4f8"},
+ {file = "onnxruntime-1.16.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:68f8d3347f11fcc6256266c562e4314b8c6da3e30fc275052a2ab693540b17fd"},
+ {file = "onnxruntime-1.16.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:16217fa87d3482300a91036f9b499c85215a3b495de1ef9a68cbcf3df1a7c548"},
+ {file = "onnxruntime-1.16.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ce6b7046005442fcd09b86647bdc9a85d60c1367cb36ce7f16b942744cf27fe4"},
+ {file = "onnxruntime-1.16.2-cp311-cp311-win32.whl", hash = "sha256:773c231e526f815b8a3f3549d216cd8fed4c9e226e9e16e86af1b69a4bd29b58"},
+ {file = "onnxruntime-1.16.2-cp311-cp311-win_amd64.whl", hash = "sha256:90e83a93b3d946c4a1d9dcbae286350accb0d80512d7c1b85953a444d19c0058"},
+ {file = "onnxruntime-1.16.2-cp38-cp38-macosx_10_15_x86_64.whl", hash = "sha256:8616f56905775dd8beeae11cf145542fff06c38cd97bfe9afe0c4a66142fc6d5"},
+ {file = "onnxruntime-1.16.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:9f5e1d5ca5560044896edb2ad79113f863dc7daa804a26787c7b21c2a96d41e7"},
+ {file = "onnxruntime-1.16.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b97ce538ffb668c4897e7500a586c150a045869876e0234e0611c4e4f428be63"},
+ {file = "onnxruntime-1.16.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7cadf175baa782599f36586c23f84fe12b02702ceb59be57dbd8eefc6cc13cc4"},
+ {file = "onnxruntime-1.16.2-cp38-cp38-win32.whl", hash = "sha256:0ffd3b8a3039be713476b8783d254564976664c9b51ec70e7fb5d3e2832bf0f0"},
+ {file = "onnxruntime-1.16.2-cp38-cp38-win_amd64.whl", hash = "sha256:e2211f336e83819edbf174dcf56de35b0dcbfc6c92d3b685c8d85fba19bdf97d"},
+ {file = "onnxruntime-1.16.2-cp39-cp39-macosx_10_15_x86_64.whl", hash = "sha256:98a49bda980bcf819f8d9be880e3e7ba8a1df66aa5ce4fc7bb68ba9acf1fc7ad"},
+ {file = "onnxruntime-1.16.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:1f1e90fa0f43e988cd043e5a4b1eb77eda6cbd7523f316d93d36b33ff1ceb91f"},
+ {file = "onnxruntime-1.16.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b0cbdb7df8078b2e8d9804de948963961eb8c6f417ef35ed243455162a9a065c"},
+ {file = "onnxruntime-1.16.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b93c1cbd885c5fe0018b982c9dabe3cc3531416a3b50d0958a291605b32fe3ce"},
+ {file = "onnxruntime-1.16.2-cp39-cp39-win32.whl", hash = "sha256:713101b65d74438f380f5ea2475ce4f6026171e6229100e5be2baa92519fca17"},
+ {file = "onnxruntime-1.16.2-cp39-cp39-win_amd64.whl", hash = "sha256:3382934f9d86060b6bacd3eb4633c5ff904be2c99d3a7fb7faf2828381b15928"},
]
[package.dependencies]
@@ -2578,64 +2516,6 @@ files = [
{file = "packaging-23.2.tar.gz", hash = "sha256:048fb0e9405036518eaaf48a55953c750c11e1a1b68e0dd1a9d62ed0c092cfc5"},
]
-[[package]]
-name = "pandas"
-version = "2.1.0"
-description = "Powerful data structures for data analysis, time series, and statistics"
-optional = false
-python-versions = ">=3.9"
-files = [
- {file = "pandas-2.1.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:40dd20439ff94f1b2ed55b393ecee9cb6f3b08104c2c40b0cb7186a2f0046242"},
- {file = "pandas-2.1.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:d4f38e4fedeba580285eaac7ede4f686c6701a9e618d8a857b138a126d067f2f"},
- {file = "pandas-2.1.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6e6a0fe052cf27ceb29be9429428b4918f3740e37ff185658f40d8702f0b3e09"},
- {file = "pandas-2.1.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9d81e1813191070440d4c7a413cb673052b3b4a984ffd86b8dd468c45742d3cc"},
- {file = "pandas-2.1.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:eb20252720b1cc1b7d0b2879ffc7e0542dd568f24d7c4b2347cb035206936421"},
- {file = "pandas-2.1.0-cp310-cp310-win_amd64.whl", hash = "sha256:38f74ef7ebc0ffb43b3d633e23d74882bce7e27bfa09607f3c5d3e03ffd9a4a5"},
- {file = "pandas-2.1.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:cda72cc8c4761c8f1d97b169661f23a86b16fdb240bdc341173aee17e4d6cedd"},
- {file = "pandas-2.1.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:d97daeac0db8c993420b10da4f5f5b39b01fc9ca689a17844e07c0a35ac96b4b"},
- {file = "pandas-2.1.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d8c58b1113892e0c8078f006a167cc210a92bdae23322bb4614f2f0b7a4b510f"},
- {file = "pandas-2.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:629124923bcf798965b054a540f9ccdfd60f71361255c81fa1ecd94a904b9dd3"},
- {file = "pandas-2.1.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:70cf866af3ab346a10debba8ea78077cf3a8cd14bd5e4bed3d41555a3280041c"},
- {file = "pandas-2.1.0-cp311-cp311-win_amd64.whl", hash = "sha256:d53c8c1001f6a192ff1de1efe03b31a423d0eee2e9e855e69d004308e046e694"},
- {file = "pandas-2.1.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:86f100b3876b8c6d1a2c66207288ead435dc71041ee4aea789e55ef0e06408cb"},
- {file = "pandas-2.1.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:28f330845ad21c11db51e02d8d69acc9035edfd1116926ff7245c7215db57957"},
- {file = "pandas-2.1.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b9a6ccf0963db88f9b12df6720e55f337447aea217f426a22d71f4213a3099a6"},
- {file = "pandas-2.1.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d99e678180bc59b0c9443314297bddce4ad35727a1a2656dbe585fd78710b3b9"},
- {file = "pandas-2.1.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:b31da36d376d50a1a492efb18097b9101bdbd8b3fbb3f49006e02d4495d4c644"},
- {file = "pandas-2.1.0-cp39-cp39-win_amd64.whl", hash = "sha256:0164b85937707ec7f70b34a6c3a578dbf0f50787f910f21ca3b26a7fd3363437"},
- {file = "pandas-2.1.0.tar.gz", hash = "sha256:62c24c7fc59e42b775ce0679cfa7b14a5f9bfb7643cfbe708c960699e05fb918"},
-]
-
-[package.dependencies]
-numpy = {version = ">=1.23.2", markers = "python_version >= \"3.11\""}
-python-dateutil = ">=2.8.2"
-pytz = ">=2020.1"
-tzdata = ">=2022.1"
-
-[package.extras]
-all = ["PyQt5 (>=5.15.6)", "SQLAlchemy (>=1.4.36)", "beautifulsoup4 (>=4.11.1)", "bottleneck (>=1.3.4)", "dataframe-api-compat (>=0.1.7)", "fastparquet (>=0.8.1)", "fsspec (>=2022.05.0)", "gcsfs (>=2022.05.0)", "html5lib (>=1.1)", "hypothesis (>=6.46.1)", "jinja2 (>=3.1.2)", "lxml (>=4.8.0)", "matplotlib (>=3.6.1)", "numba (>=0.55.2)", "numexpr (>=2.8.0)", "odfpy (>=1.4.1)", "openpyxl (>=3.0.10)", "pandas-gbq (>=0.17.5)", "psycopg2 (>=2.9.3)", "pyarrow (>=7.0.0)", "pymysql (>=1.0.2)", "pyreadstat (>=1.1.5)", "pytest (>=7.3.2)", "pytest-asyncio (>=0.17.0)", "pytest-xdist (>=2.2.0)", "pyxlsb (>=1.0.9)", "qtpy (>=2.2.0)", "s3fs (>=2022.05.0)", "scipy (>=1.8.1)", "tables (>=3.7.0)", "tabulate (>=0.8.10)", "xarray (>=2022.03.0)", "xlrd (>=2.0.1)", "xlsxwriter (>=3.0.3)", "zstandard (>=0.17.0)"]
-aws = ["s3fs (>=2022.05.0)"]
-clipboard = ["PyQt5 (>=5.15.6)", "qtpy (>=2.2.0)"]
-compression = ["zstandard (>=0.17.0)"]
-computation = ["scipy (>=1.8.1)", "xarray (>=2022.03.0)"]
-consortium-standard = ["dataframe-api-compat (>=0.1.7)"]
-excel = ["odfpy (>=1.4.1)", "openpyxl (>=3.0.10)", "pyxlsb (>=1.0.9)", "xlrd (>=2.0.1)", "xlsxwriter (>=3.0.3)"]
-feather = ["pyarrow (>=7.0.0)"]
-fss = ["fsspec (>=2022.05.0)"]
-gcp = ["gcsfs (>=2022.05.0)", "pandas-gbq (>=0.17.5)"]
-hdf5 = ["tables (>=3.7.0)"]
-html = ["beautifulsoup4 (>=4.11.1)", "html5lib (>=1.1)", "lxml (>=4.8.0)"]
-mysql = ["SQLAlchemy (>=1.4.36)", "pymysql (>=1.0.2)"]
-output-formatting = ["jinja2 (>=3.1.2)", "tabulate (>=0.8.10)"]
-parquet = ["pyarrow (>=7.0.0)"]
-performance = ["bottleneck (>=1.3.4)", "numba (>=0.55.2)", "numexpr (>=2.8.0)"]
-plot = ["matplotlib (>=3.6.1)"]
-postgresql = ["SQLAlchemy (>=1.4.36)", "psycopg2 (>=2.9.3)"]
-spss = ["pyreadstat (>=1.1.5)"]
-sql-other = ["SQLAlchemy (>=1.4.36)"]
-test = ["hypothesis (>=6.46.1)", "pytest (>=7.3.2)", "pytest-asyncio (>=0.17.0)", "pytest-xdist (>=2.2.0)"]
-xml = ["lxml (>=4.8.0)"]
-
[[package]]
name = "pandas"
version = "2.1.2"
@@ -4771,5 +4651,5 @@ testing = ["coverage (>=5.0.3)", "zope.event", "zope.testing"]
[metadata]
lock-version = "2.0"
-python-versions = "^3.11"
-content-hash = "bba5f87aa67bc1d2283a9f4b471ef78e572337f22413870d324e908014410d53"
+python-versions = "~3.11"
+content-hash = "a4c9b3550bb2a67a54b9ab70e700b24fb9eb0b652e90d7dd8ec92abd121ca6e3"
diff --git a/machine-learning/pyproject.toml b/machine-learning/pyproject.toml
index 6fea1e5d5..b7c831429 100644
--- a/machine-learning/pyproject.toml
+++ b/machine-learning/pyproject.toml
@@ -1,13 +1,13 @@
[tool.poetry]
name = "machine-learning"
-version = "1.84.0"
+version = "1.89.0"
description = ""
authors = ["Hau Tran
"]
readme = "README.md"
packages = [{include = "app"}]
[tool.poetry.dependencies]
-python = "^3.11"
+python = "~3.11"
torch = [
{markers = "platform_machine == 'arm64' or platform_machine == 'aarch64'", version = "=2.1.0", source = "pypi"},
{markers = "platform_machine == 'amd64' or platform_machine == 'x86_64'", version = "=2.1.0", source = "pytorch-cpu"}
diff --git a/mobile/analysis_options.yaml b/mobile/analysis_options.yaml
index b570c2b2e..230e2a0d7 100644
--- a/mobile/analysis_options.yaml
+++ b/mobile/analysis_options.yaml
@@ -36,3 +36,57 @@ analyzer:
- openapi/
- openapi/test/
- lib/generated_plugin_registrant.dart
+
+plugins:
+ - custom_lint
+
+dart_code_metrics:
+ metrics:
+ cyclomatic-complexity: 20
+ number-of-parameters: 4
+ maximum-nesting-level: 5
+ rules:
+ # Common
+ - avoid-accessing-collections-by-constant-index
+ - avoid-accessing-other-classes-private-members
+ - avoid-cascade-after-if-null
+ - avoid-collapsible-if
+ - avoid-collection-methods-with-unrelated-types
+ - avoid-declaring-call-method
+ - avoid-double-slash-imports
+ - avoid-duplicate-cascades
+ - avoid-duplicate-patterns
+ - avoid-generics-shadowing
+ - avoid-global-state
+ # Flutter
+ - add-copy-with:
+ file-name-pattern: '.model.dart'
+ - always-remove-listener
+ - avoid-border-all
+ - avoid-empty-setstate
+ - avoid-expanded-as-spacer
+ - avoid-incomplete-copy-with
+ - avoid-inherited-widget-in-initstate
+ - avoid-late-context
+ - avoid-recursive-widget-calls
+ - avoid-returning-widgets
+ - avoid-shrink-wrap-in-lists
+ - avoid-single-child-column-or-row
+ - avoid-state-constructors
+ - avoid-stateless-widget-initialized-fields
+ - avoid-unnecessary-overrides-in-state
+ - avoid-unnecessary-stateful-widgets
+ - avoid-wrapping-in-padding
+ - dispose-fields
+ - prefer-const-border-radius
+ - prefer-correct-edge-insets-constructor
+ - prefer-dedicated-media-query-methods
+ - prefer-define-hero-tag
+ - prefer-extracting-callbacks
+ - prefer-single-widget-per-file:
+ ignore-private-widgets: true
+ - prefer-sliver-prefix
+ - prefer-text-rich
+ - prefer-using-list-view
+ - proper-super-calls
+ - use-setstate-synchronously
diff --git a/mobile/android/fastlane/Fastfile b/mobile/android/fastlane/Fastfile
index 281fa52d2..d68edc784 100644
--- a/mobile/android/fastlane/Fastfile
+++ b/mobile/android/fastlane/Fastfile
@@ -35,8 +35,8 @@ platform :android do
task: 'bundle',
build_type: 'Release',
properties: {
- "android.injected.version.code" => 108,
- "android.injected.version.name" => "1.84.0",
+ "android.injected.version.code" => 113,
+ "android.injected.version.name" => "1.89.0",
}
)
upload_to_play_store(skip_upload_apk: true, skip_upload_images: true, skip_upload_screenshots: true, aab: '../build/app/outputs/bundle/release/app-release.aab')
diff --git a/mobile/android/fastlane/report.xml b/mobile/android/fastlane/report.xml
index a0530d9aa..3e8a519ff 100644
--- a/mobile/android/fastlane/report.xml
+++ b/mobile/android/fastlane/report.xml
@@ -5,17 +5,17 @@
-
+
-
+
-
+
diff --git a/mobile/assets/i18n/ca.json b/mobile/assets/i18n/ca.json
index 1ed97ed07..36aad1957 100644
--- a/mobile/assets/i18n/ca.json
+++ b/mobile/assets/i18n/ca.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Error al modificar el títol de l'àlbum",
"album_viewer_appbar_share_leave": "Surt de l'àlbum",
"album_viewer_appbar_share_remove": "Treu de l'àlbum",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Afegeix usuaris",
"all_people_page_title": "Persones",
"all_videos_page_title": "Vídeos",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "No s'ha trobat res arxivat",
"archive_page_title": "Arxiu({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamic layout",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Ús de memòria cau",
"cache_settings_subtitle": "Control the caching behaviour of the Immich mobile application",
"cache_settings_thumbnail_size": "Thumbnail cache size ({} assets)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Configuració de la memòria cau",
"change_password_form_confirm_password": "Confirma la contrasenya",
- "change_password_form_description": "Hi {firstName} {lastName},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
+ "change_password_form_description": "Hi {name},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
"change_password_form_new_password": "New Password",
"change_password_form_password_mismatch": "Passwords do not match",
"change_password_form_reenter_new_password": "Re-enter New Password",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Added {added} assets to album {album}. {failed} assets are already in the album.",
"home_page_add_to_album_err_local": "Can not add local assets to albums yet, skipping",
"home_page_add_to_album_success": "Added {added} assets to album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Can not archive local assets yet, skipping",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Building the timeline",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Can not favorite local assets yet, skipping",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "If this is your first time using the app, please make sure to choose a backup album(s) so that the timeline can populate photos and videos in the album(s).",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
"image_viewer_page_state_provider_download_error": "Download Error",
"image_viewer_page_state_provider_download_success": "Download Success",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
"partner_page_stop_sharing_title": "Stop sharing your photos?",
"partner_page_title": "Company",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continue anyway",
"permission_onboarding_get_started": "Get started",
"permission_onboarding_go_to_settings": "Go to settings",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich requires permission to view your photos and videos.",
"profile_drawer_app_logs": "Logs",
"profile_drawer_client_server_up_to_date": "Client and Server are up-to-date",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Settings",
"profile_drawer_sign_out": "Tanca la sessió",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Error al crear l'àlbum",
"select_user_for_sharing_page_share_suggestions": "Suggestions",
"server_info_box_app_version": "Versió de l'aplicació",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Versió del servidor",
"setting_image_viewer_help": "The detail viewer loads the small thumbnail first, then loads the medium-size preview (if enabled), finally loads the original (if enabled).",
"setting_image_viewer_original_subtitle": "Enable to load the original full-resolution image (large!). Disable to reduce data usage (both network and on device cache).",
@@ -300,6 +315,12 @@
"share_add_photos": "Afegeix fotografies",
"share_add_title": "Afegeix un títol",
"share_create_album": "Crea un àlbum",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Preparing...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/cs-CZ.json b/mobile/assets/i18n/cs-CZ.json
index d7151e9cf..8290c8f8c 100644
--- a/mobile/assets/i18n/cs-CZ.json
+++ b/mobile/assets/i18n/cs-CZ.json
@@ -13,7 +13,7 @@
"album_info_card_backup_album_included": "ZAHRNUTO",
"album_thumbnail_card_item": "1 položka",
"album_thumbnail_card_items": "{} položek",
- "album_thumbnail_card_shared": "Sdíleno",
+ "album_thumbnail_card_shared": " · Sdíleno",
"album_thumbnail_owned": "Vlastní",
"album_thumbnail_shared_by": "Sdílel(a) {}",
"album_viewer_appbar_share_delete": "Odstranit album",
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Nepodařilo se změnit název alba",
"album_viewer_appbar_share_leave": "Opustit album",
"album_viewer_appbar_share_remove": "Odstranit z alba",
+ "album_viewer_appbar_share_to": "Sdílet na",
"album_viewer_page_share_add_users": "Přidat uživatele",
"all_people_page_title": "Lidé",
"all_videos_page_title": "Videa",
+ "app_bar_signout_dialog_content": "Určitě se chcete odhlásit?",
+ "app_bar_signout_dialog_ok": "Ano",
+ "app_bar_signout_dialog_title": "Odhlásit se",
"archive_page_no_archived_assets": "Nebyla nalezena žádná archivovaná média",
"archive_page_title": "Archív ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamické rozložení",
@@ -44,11 +48,11 @@
"backup_all": "Vše",
"backup_background_service_backup_failed_message": "Zálohování médií selhalo. Zkouším to znovu...",
"backup_background_service_connection_failed_message": "Nepodařilo se připojit k serveru. Zkouším to znovu...",
- "backup_background_service_current_upload_notification": "Zálohování {}",
+ "backup_background_service_current_upload_notification": "Nahrávání {}",
"backup_background_service_default_notification": "Kontrola nových médií…",
"backup_background_service_error_title": "Chyba zálohování",
"backup_background_service_in_progress_notification": "Zálohování vašich médií...",
- "backup_background_service_upload_failure_notification": "Nepodařilo se zálohovat {}",
+ "backup_background_service_upload_failure_notification": "Nepodařilo se nahrát {}",
"backup_controller_page_albums": "Zálohovaná alba",
"backup_controller_page_background_app_refresh_disabled_content": "Povolte obnovení aplikace na pozadí v Nastavení > Obecné > Obnovení aplikace na pozadí, abyste mohli používat zálohování na pozadí.",
"backup_controller_page_background_app_refresh_disabled_title": " Obnovování aplikací na pozadí je vypnuté",
@@ -91,14 +95,14 @@
"backup_controller_page_total_sub": "Všechny jedinečné fotografie a videa z vybraných alb",
"backup_controller_page_turn_off": "Vypnout zálohování na popředí",
"backup_controller_page_turn_on": "Povolit zálohování na popředí",
- "backup_controller_page_uploading_file_info": "Informace o zálohovaném souboru",
+ "backup_controller_page_uploading_file_info": "Informace o nahraném souboru",
"backup_err_only_album": "Nelze odstranit jediné vybrané album",
"backup_info_card_assets": "položek",
"backup_manual_cancelled": "Zrušeno",
"backup_manual_failed": "Selhalo",
- "backup_manual_in_progress": "Zálohování již probíhá. Zkuste to znovu později",
+ "backup_manual_in_progress": "Nahrávání již probíhá. Zkuste to znovu později",
"backup_manual_success": "Úspěch",
- "backup_manual_title": "Stav zálohování",
+ "backup_manual_title": "Stav nahrávání",
"cache_settings_album_thumbnails": "Náhledy stránek knihovny (položek {})",
"cache_settings_clear_cache_button": "Vymazat vyrovnávací paměť",
"cache_settings_clear_cache_button_title": "Vymaže vyrovnávací paměť aplikace. To výrazně ovlivní výkon aplikace, dokud se vyrovnávací paměť neobnoví.",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Použití vyrovnávací paměti",
"cache_settings_subtitle": "Ovládání chování mobilní aplikace Immich v mezipaměti",
"cache_settings_thumbnail_size": "Velikost vyrovnávací paměti náhledů (položek {})",
+ "cache_settings_tile_subtitle": "Ovládání chování místního úložiště",
+ "cache_settings_tile_title": "Místní úložiště",
"cache_settings_title": "Nastavení vyrovnávací paměti",
"change_password_form_confirm_password": "Potvrďte heslo",
- "change_password_form_description": "Dobrý den, {firstName} {lastName},\n\nje to buď poprvé, co se přihlašujete do systému, nebo byl vytvořen požadavek na změnu hesla. Níže zadejte nové heslo.",
+ "change_password_form_description": "Dobrý den, {name},\n\nje to buď poprvé, co se přihlašujete do systému, nebo byl vytvořen požadavek na změnu hesla. Níže zadejte nové heslo.",
"change_password_form_new_password": "Nové heslo",
"change_password_form_password_mismatch": "Hesla se neshodují",
"change_password_form_reenter_new_password": "Znovu zadejte nové heslo",
@@ -130,10 +136,10 @@
"control_bottom_app_bar_delete": "Vymazat",
"control_bottom_app_bar_favorite": "Oblíbené",
"control_bottom_app_bar_share": "Sdílet",
- "control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_share_to": "Sdílet v",
+ "control_bottom_app_bar_stack": "Zásobník",
"control_bottom_app_bar_unarchive": "Odarchivovat",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Nahrát",
"create_album_page_untitled": "Bez názvu",
"create_shared_album_page_create": "Vytvořit",
"create_shared_album_page_share": "Sdílet",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "Zrušit",
"delete_dialog_ok": "Vymazat",
"delete_dialog_title": "Vymazat trvale",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Opravdu chcete tento odkaz ke sdílení odstranit?",
+ "delete_shared_link_dialog_title": "Odstranit sdílený odkaz",
"description_input_hint_text": "Přidat popis...",
"description_input_submit_error": "Chyba aktualizace popisu, další podrobnosti najdete v logu",
"exif_bottom_sheet_description": "Přidat popis...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "Přidáno {added} položek do alba {album}. {failed} položek již je v albu.",
"home_page_add_to_album_err_local": "Zatím není možné přidat lokální média do alb, přeskakuji",
"home_page_add_to_album_success": "Přidány položky {added} do alba {album}.",
+ "home_page_album_err_partner": "Položky partnera nelze zatím přidat do alba, přeskakuji",
"home_page_archive_err_local": "Zatím nemohu archivovat lokální média, přeskakuji",
+ "home_page_archive_err_partner": "Položky partnera nelze archivovat, přeskakuji",
"home_page_building_timeline": "Vytváření časové osy",
+ "home_page_delete_err_partner": "Položky partnera nelze odstranit, přeskakuji",
"home_page_favorite_err_local": "Zatím není možné zařadit lokální média mezi oblíbená, přeskakuji",
+ "home_page_favorite_err_partner": "Položky partnera nelze označit jako oblíbené, přeskakuji",
"home_page_first_time_notice": "Pokud aplikaci používáte poprvé, nezapomeňte si vybrat zálohovaná alba, aby se na časové ose mohly nacházet fotografie a videa z vybraných alb.",
- "home_page_upload_err_limit": "Lze zálohovat nejvýše 30 položek najednou, přeskakuji",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
+ "home_page_upload_err_limit": "Lze nahrát nejvýše 30 položek najednou, přeskakuji",
"image_viewer_page_state_provider_download_error": "Chyba stahování",
"image_viewer_page_state_provider_download_success": "Stahování bylo úspěšné",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Chyba sdílení",
"library_page_albums": "Alba",
"library_page_archive": "Archív",
"library_page_device_albums": "Alba v zařízení",
@@ -179,8 +190,8 @@
"library_page_new_album": "Nové album",
"library_page_sharing": "Sdílení",
"library_page_sort_created": "Naposledy vytvořené",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Naposledy upraveno",
+ "library_page_sort_most_recent_photo": "Nejnovější fotografie",
"library_page_sort_title": "Podle názvu alba",
"login_disabled": "Přihlášení bylo zakázáno",
"login_form_api_exception": "Výjimka API. Zkontrolujte URL serveru a zkuste to znovu.",
@@ -218,7 +229,7 @@
"map_settings_dialog_cancel": "Zrušit",
"map_settings_dialog_save": "Uložit",
"map_settings_dialog_title": "Nastavení map",
- "map_settings_include_show_archived": "Include Archived",
+ "map_settings_include_show_archived": "Zahrnout archivované",
"map_settings_only_relative_range": "Rozsah data",
"map_settings_only_show_favorites": "Zobrazit pouze oblíbené",
"map_zoom_to_see_photos": "Oddálit pro zobrazení fotografií",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} již nebude mít přístup k vašim fotografiím.",
"partner_page_stop_sharing_title": "Přestat sdílet vaše fotografie?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Přesto pokračovat",
"permission_onboarding_get_started": "Začít",
"permission_onboarding_go_to_settings": "Přejít do nastavení",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich potřebuje přístup k zobrazení vašich fotek a videí.",
"profile_drawer_app_logs": "Logy",
"profile_drawer_client_server_up_to_date": "Klient a server jsou aktuální",
+ "profile_drawer_documentation": "Dokumentace",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Nastavení",
"profile_drawer_sign_out": "Odhlásit se",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Vyhodit",
"recently_added_page_title": "Nedávno přidané",
"search_bar_hint": "Prohledejte své fotky",
"search_page_categories": "Kategorie",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Nepodařilo se vytvořit album",
"select_user_for_sharing_page_share_suggestions": "Návrhy",
"server_info_box_app_version": "Verze aplikace",
+ "server_info_box_server_url": "URL serveru",
"server_info_box_server_version": "Verze serveru",
"setting_image_viewer_help": "V prohlížeči detailů se nejprve načte malá miniatura, poté se načte náhled střední velikosti (je-li povolen) a nakonec se načte originál (je-li povolen).",
"setting_image_viewer_original_subtitle": "Umožňuje načíst původní obrázek v plném rozlišení (velký!). Zakázat pro snížení využití dat (v síti i v mezipaměti zařízení).",
@@ -288,11 +303,11 @@
"setting_notifications_notify_minutes": "{} minut",
"setting_notifications_notify_never": "nikdy",
"setting_notifications_notify_seconds": "{} sekundy",
- "setting_notifications_single_progress_subtitle": "Podrobné informace o průběhu zálohování položky",
+ "setting_notifications_single_progress_subtitle": "Podrobné informace o průběhu nahrávání položky",
"setting_notifications_single_progress_title": "Zobrazit průběh detailů zálohování na pozadí",
"setting_notifications_subtitle": "Přizpůsobení předvoleb oznámení",
"setting_notifications_title": "Oznámení",
- "setting_notifications_total_progress_subtitle": "Celkový průběh zálohování (hotovo/celkově)",
+ "setting_notifications_total_progress_subtitle": "Celkový průběh nahrání (hotovo/celkově)",
"setting_notifications_total_progress_title": "Zobrazit celkový průběh zálohování na pozadí",
"setting_pages_app_bar_settings": "Nastavení",
"settings_require_restart": "Pro použití tohoto nastavení restartujte Immich",
@@ -300,28 +315,37 @@
"share_add_photos": "Přidat fotografie",
"share_add_title": "Přidat název",
"share_create_album": "Vytvořit album",
+ "shared_album_activities_input_disable": "Komentář je vypnutý",
+ "shared_album_activities_input_hint": "Řekněte něco",
+ "shared_album_activity_remove_content": "Chcete odstranit tuto aktivitu?",
+ "shared_album_activity_remove_title": "Odstranit aktivitu",
+ "shared_album_activity_setting_subtitle": "Nechte ostatní reagovat",
+ "shared_album_activity_setting_title": "Komentáře a lajky",
"share_dialog_preparing": "Připravuji...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Sdílené odkazy",
+ "shared_link_create_app_bar_title": "Vytvořit odkaz pro sdílení",
+ "shared_link_create_info": "Umožnit každému, kdo má odkaz, zobrazit vybrané fotografie",
+ "shared_link_create_submit_button": "Vytvořit odkaz",
+ "shared_link_edit_allow_download": "Povolit veřejným uživatelům stahovat",
+ "shared_link_edit_allow_upload": "Povolit veřejným uživatelům nahrávat",
+ "shared_link_edit_app_bar_title": "Upravit odkaz",
+ "shared_link_edit_change_expiry": "Změnit dobu platnosti",
+ "shared_link_edit_description": "Popis",
+ "shared_link_edit_description_hint": "Zadejte popis sdílení",
+ "shared_link_edit_expire_after": "Platnost vyprší po",
+ "shared_link_edit_password": "Heslo",
+ "shared_link_edit_password_hint": "Zadejte heslo pro sdílení",
+ "shared_link_edit_show_meta": "Zobrazit metadata",
+ "shared_link_edit_submit_button": "Aktualizovat odkaz",
+ "shared_link_empty": "Nemáte žádné sdílené odkazy",
+ "shared_link_manage_links": "Spravovat sdílené odkazy",
+ "share_done": "Hotovo",
"share_invite": "Pozvat do alba",
"sharing_page_album": "Sdílená alba",
"sharing_page_description": "Vytvářejte sdílená alba a sdílejte fotografie a videa s lidmi ve vaší síti.",
"sharing_page_empty_list": "PRÁZDNÝ SEZNAM",
"sharing_silver_appbar_create_shared_album": "Vytvořit sdílené album",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Sdílené odkazy",
"sharing_silver_appbar_share_partner": "Sdílet s partnerem",
"tab_controller_nav_library": "Knihovna",
"tab_controller_nav_photos": "Fotografie",
@@ -338,29 +362,29 @@
"theme_setting_three_stage_loading_subtitle": "Třístupňové načítání může zvýšit výkonnost načítání, ale vede k výrazně vyššímu zatížení sítě.",
"theme_setting_three_stage_loading_title": "Povolení třístupňového načítání",
"translated_text_options": "Možnosti",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
+ "trash_page_delete": "Smazat",
+ "trash_page_delete_all": "Smazat všechny",
+ "trash_page_empty_trash_btn": "Vysypat koš",
+ "trash_page_empty_trash_dialog_content": "Chcete vyprázdnit svoje vyhozené položky? Tyto položky budou trvale odstraněny z aplikace",
"trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
+ "trash_page_info": "Vyhozené položky budou trvale odstraněny po {} dnech",
+ "trash_page_no_assets": "Žádné vyhozené položky",
+ "trash_page_restore": "Obnovit",
+ "trash_page_restore_all": "Obnovit všechny",
+ "trash_page_select_assets_btn": "Vybrat položky",
+ "trash_page_select_btn": "Vybrat",
+ "trash_page_title": "Koš ({})",
"upload_dialog_cancel": "Zrušit",
"upload_dialog_info": "Chcete zálohovat vybrané položky na server?",
- "upload_dialog_ok": "Zálohovat",
- "upload_dialog_title": "Zálohovat položku",
+ "upload_dialog_ok": "Nahrát",
+ "upload_dialog_title": "Nahrát položku",
"version_announcement_overlay_ack": "Potvrdit",
"version_announcement_overlay_release_notes": "poznámky k vydání",
"version_announcement_overlay_text_1": "Ahoj, k dispozici je nová verze",
"version_announcement_overlay_text_2": "najděte si čas na návštěvu ",
"version_announcement_overlay_text_3": " a ujistěte se, že vaše konfigurace docker-compose a .env je aktuální, abyste předešli nesprávné konfiguraci, zvláště pokud používáte WatchTower nebo jakýkoli mechanismus, který podporuje automatické aktualizace serverových aplikací.",
"version_announcement_overlay_title": "K dispozici je nová verze serveru \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Odstranit ze zásobníku",
+ "viewer_stack_use_as_main_asset": "Použít jako hlavní položku",
+ "viewer_unstack": "Rozbalit zásobník"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/da-DK.json b/mobile/assets/i18n/da-DK.json
index 41e2cc340..b5b7c7e45 100644
--- a/mobile/assets/i18n/da-DK.json
+++ b/mobile/assets/i18n/da-DK.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Fejlede i at ændre albumtitel",
"album_viewer_appbar_share_leave": "Forlad album",
"album_viewer_appbar_share_remove": "Fjern fra album",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Tilføj brugere",
"all_people_page_title": "Personer",
"all_videos_page_title": "Videoer",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "Ingen arkiverede elementer blev fundet",
"archive_page_title": "Arkivér ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamisk layout",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Cacheforbrug",
"cache_settings_subtitle": "Håndter cache-adfærden for Immich-appen.",
"cache_settings_thumbnail_size": "Størrelse af miniaturebillede cache ({} elementer)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Cache-indstillinger",
"change_password_form_confirm_password": "Bekræft kodeord",
- "change_password_form_description": "Hej {firstName} {lastName},\n\nDette er enten første gang du logger ind eller også er der lavet en anmodning om at ændre dit kodeord. Indtast venligst et nyt kodeord nedenfor.",
+ "change_password_form_description": "Hej {name},\n\nDette er enten første gang du logger ind eller også er der lavet en anmodning om at ændre dit kodeord. Indtast venligst et nyt kodeord nedenfor.",
"change_password_form_new_password": "Nyt kodeord",
"change_password_form_password_mismatch": "Kodeord er ikke ens",
"change_password_form_reenter_new_password": "Gentag nyt kodeord",
@@ -130,8 +136,8 @@
"control_bottom_app_bar_delete": "Slet",
"control_bottom_app_bar_favorite": "Favorit",
"control_bottom_app_bar_share": "Del",
- "control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_share_to": "Del til",
+ "control_bottom_app_bar_stack": "Stak",
"control_bottom_app_bar_unarchive": "Afakivér",
"control_bottom_app_bar_upload": "Upload",
"create_album_page_untitled": "Uden titel",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "Annuller",
"delete_dialog_ok": "Slet",
"delete_dialog_title": "Slet permanent",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Er du sikker på, du vil slette dette delte link?",
+ "delete_shared_link_dialog_title": "Slet delt link",
"description_input_hint_text": "Tilføj en beskrivelse...",
"description_input_submit_error": "Fejl med at opdatere beskrivelsen. Tjek loggen for flere detaljer",
"exif_bottom_sheet_description": "Tilføj beskrivelse...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "Tilføjede {added} elementer til album {album}. {failed} elementer er allerede i albummet.",
"home_page_add_to_album_err_local": "Kan endnu ikke tilføje lokale elementer til album. Springer over..",
"home_page_add_to_album_success": "Tilføjede {added} elementer til album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Kan ikke arkivere lokalt element endnu.. Springer over",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Bygger tidslinjen",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Kan endnu ikke gøre lokale elementer til favoritter. Springer over..",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Hvis det er din første gang i appen, bedes du vælge en sikkerhedskopi af albummer så tidlinjen kan blive fyldt med billeder og videoer fra albummerne.",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Det er kun muligt at lave sikkerhedskopi af 30 elementer ad gangen. Springer over",
"image_viewer_page_state_provider_download_error": "Fejl ved download",
"image_viewer_page_state_provider_download_success": "Download succesfuld",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Delingsfejl",
"library_page_albums": "Albummer",
"library_page_archive": "Arkiv",
"library_page_device_albums": "Albummer på enhed",
@@ -179,8 +190,8 @@
"library_page_new_album": "Nyt album",
"library_page_sharing": "Delte",
"library_page_sort_created": "Senest oprettet",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Sidst redigeret",
+ "library_page_sort_most_recent_photo": "Seneste billede",
"library_page_sort_title": "Albumtitel",
"login_disabled": "Login er blevet deaktiveret",
"login_form_api_exception": "API-undtagelse. Tjek serverens URL og prøv igen. ",
@@ -218,7 +229,7 @@
"map_settings_dialog_cancel": "Annuller",
"map_settings_dialog_save": "Gem",
"map_settings_dialog_title": "Kortindstillinger",
- "map_settings_include_show_archived": "Include Archived",
+ "map_settings_include_show_archived": "Inkluder arkiveret",
"map_settings_only_relative_range": "Datointerval",
"map_settings_only_show_favorites": "Vis kun favoritter",
"map_zoom_to_see_photos": "Zoom ud for at vise billeder",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} vil ikke længere have adgang til dine billeder.",
"partner_page_stop_sharing_title": "Stop med at dele dine billeder?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Fortsæt alligevel",
"permission_onboarding_get_started": "Kom i gang",
"permission_onboarding_go_to_settings": "Gå til indstillinger",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich kræver tilliadelse til at se dine billeder og videoer.",
"profile_drawer_app_logs": "Log",
"profile_drawer_client_server_up_to_date": "Klient og server er ajour",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Indstillinger",
"profile_drawer_sign_out": "Log ud",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Papirkurv",
"recently_added_page_title": "Nyligt tilføjet",
"search_bar_hint": "Søg i dine billeder",
"search_page_categories": "Kategorier",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Fejlede i at oprette et nyt album",
"select_user_for_sharing_page_share_suggestions": "Anbefalinger",
"server_info_box_app_version": "Applikationsversion",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Serverversion ",
"setting_image_viewer_help": "Detaljeret visning indlæser miniaturebilleder først. Herefter indlæses mediumstørrelse forhåndsvisning af billedet (hvis dette er slået til), for til sidst at vise originalen (hvis dette er slået til).",
"setting_image_viewer_original_subtitle": "Slå indlæsning af originalbillede i fuld størrelse til (stort!). Deaktiver for at reducere dataforbruget (både på netværket og for enhedscache).",
@@ -300,28 +315,37 @@
"share_add_photos": "Tilføj billeder",
"share_add_title": "Tilføj en titel",
"share_create_album": "Opret album",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Forbereder...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Delte links",
+ "shared_link_create_app_bar_title": "Opret link for at dele",
+ "shared_link_create_info": "Lad enhver med linket se de(t) valgte billede(r)",
+ "shared_link_create_submit_button": "Oprat link",
+ "shared_link_edit_allow_download": "Tillad at en offenlig bruger kan downloade",
+ "shared_link_edit_allow_upload": "Tillad at en offentlig bruger kan uploade",
+ "shared_link_edit_app_bar_title": "Rediger link",
+ "shared_link_edit_change_expiry": "Ændrer udløbstidspunkt",
+ "shared_link_edit_description": "Beskrivelse",
+ "shared_link_edit_description_hint": "Indtast beskrivelse",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
+ "shared_link_edit_show_meta": "Vis metadata",
+ "shared_link_edit_submit_button": "Opdater link",
+ "shared_link_empty": "Du har endnu ingen delte links",
+ "shared_link_manage_links": "Håndter delte links",
+ "share_done": "Færdig",
"share_invite": "Inviter til album",
"sharing_page_album": "Delt albums",
"sharing_page_description": "Opret delte albummer for at dele billeder og video med personer på dit netværk.",
"sharing_page_empty_list": "TOM LISTE",
"sharing_silver_appbar_create_shared_album": "Opret delt album",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Delte links",
"sharing_silver_appbar_share_partner": "Del med partner",
"tab_controller_nav_library": "Bibliotek",
"tab_controller_nav_photos": "Billeder",
@@ -338,18 +362,18 @@
"theme_setting_three_stage_loading_subtitle": "Tre-trins indlæsning kan øge ydeevnen, men kan ligeledes føre til højere netværksbelastning",
"theme_setting_three_stage_loading_title": "Slå tre-trins indlæsning til",
"translated_text_options": "Handlinger",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
+ "trash_page_delete": "Slet",
+ "trash_page_delete_all": "Slet alt",
+ "trash_page_empty_trash_btn": "Tøm papirkurv",
+ "trash_page_empty_trash_dialog_content": "Vil du tømme papirkurven? Disse elementer vil blive permanent fjernet fra Immich",
"trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
+ "trash_page_info": "Slettede elementer vil blive slettet permanent efter {} dage",
+ "trash_page_no_assets": "Ingen slettede elementer",
+ "trash_page_restore": "Gendan",
+ "trash_page_restore_all": "Gendan alt",
+ "trash_page_select_assets_btn": "Vælg elementer",
+ "trash_page_select_btn": "Vælg",
+ "trash_page_title": "Papirkurv ({})",
"upload_dialog_cancel": "Annuller",
"upload_dialog_info": "Vil du sikkerhedskopiere de(t) valgte element(er) til serveren?",
"upload_dialog_ok": "Upload",
@@ -360,7 +384,7 @@
"version_announcement_overlay_text_2": ". Besøg venligst ",
"version_announcement_overlay_text_3": " for at sikre dig, at din dockercompose- og .env-fil er opdateret, så der undgås fejlkonfiguration, specielt hvis du bruger WatchTower eller lignede.",
"version_announcement_overlay_title": "Ny serverversion er tilgængelig \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Fjern fra stak",
+ "viewer_stack_use_as_main_asset": "Brug som hovedelement",
+ "viewer_unstack": "Fjern fra stak"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/de-DE.json b/mobile/assets/i18n/de-DE.json
index 76085a61b..2af7aecc1 100644
--- a/mobile/assets/i18n/de-DE.json
+++ b/mobile/assets/i18n/de-DE.json
@@ -3,8 +3,8 @@
"add_to_album_bottom_sheet_already_exists": "Bereits in {album}",
"advanced_settings_prefer_remote_subtitle": "Manche Endgeräte laden Vorschaubilder lokaler Bilder sehr langsam. Durch diese Einstellung werden diese stattdessen direkt vom Server geladen.",
"advanced_settings_prefer_remote_title": "Server-Bilder bevorzugen",
- "advanced_settings_self_signed_ssl_subtitle": "Skips SSL certificate verification for the server endpoint. Required for self-signed certificates.",
- "advanced_settings_self_signed_ssl_title": "Allow self-signed SSL certificates",
+ "advanced_settings_self_signed_ssl_subtitle": "Verifizierung von SSL-Zertifikaten vom Server überspringen. Notwendig bei selbstsignierten Zertifikaten.",
+ "advanced_settings_self_signed_ssl_title": "Selbstsignierte SSL-Zertifikate erlauben",
"advanced_settings_tile_subtitle": "Erweiterte Benutzereinstellungen",
"advanced_settings_tile_title": "Sonstige",
"advanced_settings_troubleshooting_subtitle": "Aktiviere erweiterte Funktionen zur Fehlersuche",
@@ -14,7 +14,7 @@
"album_thumbnail_card_item": "1 Element",
"album_thumbnail_card_items": "{} Elemente",
"album_thumbnail_card_shared": " · Geteilt",
- "album_thumbnail_owned": "Owned",
+ "album_thumbnail_owned": "Eigene",
"album_thumbnail_shared_by": "Geteilt von {}",
"album_viewer_appbar_share_delete": "Album löschen",
"album_viewer_appbar_share_err_delete": "Album konnte nicht gelöscht werden",
@@ -22,12 +22,16 @@
"album_viewer_appbar_share_err_remove": "Beim Löschen von Elementen aus dem Album ist ein Problem aufgetreten",
"album_viewer_appbar_share_err_title": "Der Titel konnte nicht geändert werden",
"album_viewer_appbar_share_leave": "Album verlassen",
- "album_viewer_appbar_share_remove": "Entferne vom Album",
+ "album_viewer_appbar_share_remove": "Vom Album entfernen",
+ "album_viewer_appbar_share_to": "Teile über",
"album_viewer_page_share_add_users": "Nutzer hinzufügen",
"all_people_page_title": "Personen",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "Bist du sicher, dass du dich abmelden möchtest?",
+ "app_bar_signout_dialog_ok": "Ja",
+ "app_bar_signout_dialog_title": "Abmelden",
"archive_page_no_archived_assets": "Keine archivierten Inhalte gefunden",
- "archive_page_title": "Archive ({})",
+ "archive_page_title": "Archiv ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamisches Layout",
"asset_list_layout_settings_group_automatically": "Automatisch",
"asset_list_layout_settings_group_by": "Gruppiere Elemente nach",
@@ -54,13 +58,13 @@
"backup_controller_page_background_app_refresh_disabled_title": "Hintergrundaktualisierungen sind deaktiviert.",
"backup_controller_page_background_app_refresh_enable_button_text": "Gehe zu Einstellungen",
"backup_controller_page_background_battery_info_link": "Zeige mir wie",
- "backup_controller_page_background_battery_info_message": "For the best background backup experience, please disable any battery optimizations restricting background activity for Immich.\n\nSince this is device-specific, please lookup the required information for your device manufacturer.",
+ "backup_controller_page_background_battery_info_message": "Für die besten Ergebnisse für Sicherungen im Hintergrund, deaktiviere alle Batterieoptimierungen und Einschränkungen für die Hintergrundaktivitäten von Immich.\n\nDa dies gerätespezifisch ist, schlage diese Informationen für deinen Gerätehersteller nach.",
"backup_controller_page_background_battery_info_ok": "OK",
"backup_controller_page_background_battery_info_title": "Batterieoptimierungen",
"backup_controller_page_background_charging": "Nur während des Ladens",
"backup_controller_page_background_configure_error": "Konnte Hintergrundservice nicht konfigurieren",
- "backup_controller_page_background_delay": "Delay new assets backup: {}",
- "backup_controller_page_background_description": "Turn on the background service to automatically backup any new assets without needing to open the app",
+ "backup_controller_page_background_delay": "Sicherung neuer Elemente verzögern: {}",
+ "backup_controller_page_background_description": "Schalte den Hintergrundservice ein, um neue Elemente automatisch im Hintergrund zu sichern ohne die App zu öffnen",
"backup_controller_page_background_is_off": "Automatische Sicherung im Hintergrund ist deaktiviert",
"backup_controller_page_background_is_on": "Automatische Sicherung im Hintergrund ist aktiviert",
"backup_controller_page_background_turn_off": "Hintergrundservice ausschalten",
@@ -78,7 +82,7 @@
"backup_controller_page_id": "ID: {}",
"backup_controller_page_info": "Informationen zur Sicherung",
"backup_controller_page_none_selected": "Keine ausgewählt",
- "backup_controller_page_remainder": "Übrig",
+ "backup_controller_page_remainder": "Verbleibend",
"backup_controller_page_remainder_sub": "Noch zu sichernde Fotos und Videos",
"backup_controller_page_select": "Auswählen",
"backup_controller_page_server_storage": "Server Speicher",
@@ -94,46 +98,48 @@
"backup_controller_page_uploading_file_info": "Informationen",
"backup_err_only_album": "Das einzige Album kann nicht entfernt werden",
"backup_info_card_assets": "Elemente",
- "backup_manual_cancelled": "Cancelled",
- "backup_manual_failed": "Failed",
- "backup_manual_in_progress": "Upload already in progress. Try after sometime",
- "backup_manual_success": "Success",
- "backup_manual_title": "Upload status",
- "cache_settings_album_thumbnails": "Library page thumbnails ({} assets)",
+ "backup_manual_cancelled": "Abgebrochen",
+ "backup_manual_failed": "Fehlgeschlagen",
+ "backup_manual_in_progress": "Sicherung läuft bereits. Bitte später erneut versuchen",
+ "backup_manual_success": "Erfolgreich",
+ "backup_manual_title": "Sicherungsstatus",
+ "cache_settings_album_thumbnails": "Vorschaubilder der Bibliothek ({} Elemente)",
"cache_settings_clear_cache_button": "Zwischenspeicher löschen",
"cache_settings_clear_cache_button_title": "Löscht den Zwischenspeicher der App. Dies wird die Leistungsfähigkeit der App deutlich einschränken, bis der Zwischenspeicher wieder aufgebaut wurde.",
"cache_settings_image_cache_size": "{} Bilder im Zwischenspeicher",
- "cache_settings_statistics_album": "Library thumbnails",
- "cache_settings_statistics_assets": "{} assets ({})",
- "cache_settings_statistics_full": "Full images",
- "cache_settings_statistics_shared": "Shared album thumbnails",
+ "cache_settings_statistics_album": "Vorschaubilder der Bibliothek",
+ "cache_settings_statistics_assets": "{} Elemente ({})",
+ "cache_settings_statistics_full": "Originalbilder",
+ "cache_settings_statistics_shared": "Vorschaubilder geteilter Alben",
"cache_settings_statistics_thumbnail": "Vorschaubilder",
"cache_settings_statistics_title": "Zwischenspeicher Nutzung",
"cache_settings_subtitle": "Kontrolliere wie Immich den Zwischenspeicher nutzen soll",
"cache_settings_thumbnail_size": "{} Vorschaubilder im Zwischenspeicher",
+ "cache_settings_tile_subtitle": "Lokalen Speicher verwalten",
+ "cache_settings_tile_title": "Lokaler Speicher",
"cache_settings_title": "Zwischenspeicher Einstellungen",
"change_password_form_confirm_password": "Passwort bestätigen",
- "change_password_form_description": "Hallo {firstName} {lastName}\n\nDas ist entweder das erste Mal dass du dich einloggst oder eine Anfrage zur Änderung deines Passwortes wurde gestellt. Bitte gebe das neue Passwort ein.",
+ "change_password_form_description": "Hallo {name}\n\nDas ist entweder das erste Mal dass du dich einloggst oder eine Anfrage zur Änderung deines Passwortes wurde gestellt. Bitte gebe das neue Passwort ein.",
"change_password_form_new_password": "Neues Passwort",
"change_password_form_password_mismatch": "Passwörter stimmen nicht überein",
"change_password_form_reenter_new_password": "Passwort erneut eingeben",
"common_add_to_album": "Zu Album hinzufügen",
"common_change_password": "Passwort ändern",
- "common_create_new_album": "Erstelle ein neues Album",
+ "common_create_new_album": "Neues Album erstellen",
"common_server_error": "Bitte überprüfe Deine Netzwerkverbindung und stelle sicher, dass die App und Server Versionen kompatibel sind.",
"common_shared": "Geteilt",
"control_bottom_app_bar_add_to_album": "Zu Album hinzufügen",
"control_bottom_app_bar_album_info": "{} Elemente",
- "control_bottom_app_bar_album_info_shared": "{} Elemente · geteilt",
+ "control_bottom_app_bar_album_info_shared": "{} Elemente · Geteilt",
"control_bottom_app_bar_archive": "Archiv",
"control_bottom_app_bar_create_new_album": "Neues Album erstellen",
"control_bottom_app_bar_delete": "Löschen",
"control_bottom_app_bar_favorite": "Favorit",
"control_bottom_app_bar_share": "Teilen",
- "control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_share_to": "Teilen mit",
+ "control_bottom_app_bar_stack": "Stapeln",
"control_bottom_app_bar_unarchive": "Dearchivieren",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Hochladen",
"create_album_page_untitled": "Unbenannt",
"create_shared_album_page_create": "Erstellen",
"create_shared_album_page_share": "Teilen",
@@ -147,16 +153,16 @@
"delete_dialog_alert": "Diese Elemente werden unwiderruflich von Immich und dem Gerät entfernt",
"delete_dialog_cancel": "Abbrechen",
"delete_dialog_ok": "Löschen",
- "delete_dialog_title": "Für immer löschen",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_dialog_title": "Endgültig löschen",
+ "delete_shared_link_dialog_content": "Bist du sicher, dass du diesen geteilten Link löschen möchtest?",
+ "delete_shared_link_dialog_title": "Geteilten Link löschen",
"description_input_hint_text": "Beschreibung hinzufügen...",
"description_input_submit_error": "Beschreibung konnte nicht geändert werden, bitte im Log für mehr Details nachsehen.",
"exif_bottom_sheet_description": "Beschreibung hinzufügen...",
"exif_bottom_sheet_details": "DETAILS",
"exif_bottom_sheet_location": "STANDORT",
"experimental_settings_new_asset_list_subtitle": "In Arbeit",
- "experimental_settings_new_asset_list_title": "Experimentelle Fotogitter aktivieren",
+ "experimental_settings_new_asset_list_title": "Experimentelles Fotogitter aktivieren",
"experimental_settings_subtitle": "Benutzung auf eigene Gefahr!",
"experimental_settings_title": "Experimentell",
"favorites_page_no_favorites": "Keine favorisierten Inhalte gefunden",
@@ -164,39 +170,44 @@
"home_page_add_to_album_conflicts": "{added} Elemente zu {album} hinzugefügt. {failed} Elemente sind bereits vorhanden.",
"home_page_add_to_album_err_local": "Kann lokale Elemente noch nicht zu Alben hinzufügen, überspringe",
"home_page_add_to_album_success": "{added} Elemente zu {album} hinzugefügt.",
+ "home_page_album_err_partner": "Inhalte von Partnern können derzeit nicht zu Alben hinzugefügt werden",
"home_page_archive_err_local": "Kann lokale Elemente nicht archvieren, überspringe",
+ "home_page_archive_err_partner": "Inhalte von Partnern können nicht archiviert werden",
"home_page_building_timeline": "Zeitachse wird erstellt.",
+ "home_page_delete_err_partner": "Inhalte von Partnern können nicht gelöscht werden",
"home_page_favorite_err_local": "Kann lokale Elemente noch nicht favorisieren, überspringe",
+ "home_page_favorite_err_partner": "Inhalte von Partnern können nicht favorisiert werden",
"home_page_first_time_notice": "Wenn dies das erste Mal ist dass Du Immich nutzt, stelle bitte sicher, dass mindestens ein Album zur Sicherung ausgewählt ist, sodass die Zeitachse mit Fotos und Videos gefüllt werden kann.",
- "home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
+ "home_page_upload_err_limit": "Max. 30 Elemente können gleichzeitig hochgeladen werden, überspringe",
"image_viewer_page_state_provider_download_error": "Fehler beim Herunterladen",
"image_viewer_page_state_provider_download_success": "Erfolgreich heruntergeladen",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Fehler beim Teilen",
"library_page_albums": "Alben",
"library_page_archive": "Archiv",
- "library_page_device_albums": "Alben auf dem Gerät.",
+ "library_page_device_albums": "Alben auf dem Gerät",
"library_page_favorites": "Favoriten",
"library_page_new_album": "Neues Album",
"library_page_sharing": "Teilen",
"library_page_sort_created": "Zuletzt erstellt",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
- "library_page_sort_title": "Albumtitel",
- "login_disabled": "Login has been disabled",
+ "library_page_sort_last_modified": "Zuletzt bearbeitet",
+ "library_page_sort_most_recent_photo": "Neuestes Foto",
+ "library_page_sort_title": "Titel des Albums",
+ "login_disabled": "Login ist deaktiviert",
"login_form_api_exception": "API Fehler. Bitte die Serveradresse überprüfen und erneut versuchen.",
"login_form_button_text": "Anmelden",
"login_form_email_hint": "deine@email.de",
"login_form_endpoint_hint": "http://deine-server-ip:port/api",
- "login_form_endpoint_url": "Server URL",
+ "login_form_endpoint_url": "Server-URL",
"login_form_err_http": "Bitte gebe http:// oder https:// an",
"login_form_err_invalid_email": "Ungültige E-Mail",
"login_form_err_invalid_url": "Ungültige URL",
- "login_form_err_leading_whitespace": "Leerzichen am Anfang",
+ "login_form_err_leading_whitespace": "Leerzeichen am Anfang",
"login_form_err_trailing_whitespace": "Leerzeichen am Ende",
"login_form_failed_get_oauth_server_config": "Fehler beim Login per OAuth, Server-URL überprüfen",
"login_form_failed_get_oauth_server_disable": "OAuth-Funktion nicht verfügbar auf diesem Server.",
- "login_form_failed_login": "Error logging you in, check server url, email and password",
- "login_form_handshake_exception": "There was an Handshake Exception with the server. Enable self-signed certificate support in the settings if you are using a self-signed certificate.",
+ "login_form_failed_login": "Fehler beim Login, prüfe Server-URL, E-Mail und Passwort",
+ "login_form_handshake_exception": "Fehler beim Verbindungsaufbau mit dem Server. Falls du ein selbstsigniertes Zertifikat verwendest, aktiviere die Unterstützung in den Einstellungen.",
"login_form_label_email": "E-Mail",
"login_form_label_password": "Passwort",
"login_form_next_button": "Weiter",
@@ -204,26 +215,26 @@
"login_form_save_login": "Angemeldet bleiben",
"login_form_server_empty": "Serveradresse eingeben.",
"login_form_server_error": "Konnte nicht mit Server verbinden.",
- "login_password_changed_error": "There was an error updating your password",
- "login_password_changed_success": "Password updated successfully",
- "map_cannot_get_user_location": "Cannot get user's location",
- "map_location_dialog_cancel": "Cancel",
- "map_location_dialog_yes": "Yes",
- "map_location_service_disabled_content": "Location service needs to be enabled to display assets from your current location. Do you want to enable it now?",
- "map_location_service_disabled_title": "Location Service disabled",
- "map_no_assets_in_bounds": "No photos in this area",
- "map_no_location_permission_content": "Location permission is needed to display assets from your current location. Do you want to allow it now?",
- "map_no_location_permission_title": "Location Permission denied",
- "map_settings_dark_mode": "Dark mode",
- "map_settings_dialog_cancel": "Cancel",
- "map_settings_dialog_save": "Save",
- "map_settings_dialog_title": "Map Settings",
- "map_settings_include_show_archived": "Include Archived",
- "map_settings_only_relative_range": "Date range",
- "map_settings_only_show_favorites": "Show Favorite Only",
- "map_zoom_to_see_photos": "Zoom out to see photos",
+ "login_password_changed_error": "Fehler beim Passwort ändern",
+ "login_password_changed_success": "Passwort erfolgreich geändert",
+ "map_cannot_get_user_location": "Standort konnte nicht ermittelt werden",
+ "map_location_dialog_cancel": "Abbrechen",
+ "map_location_dialog_yes": "Ja",
+ "map_location_service_disabled_content": "Ortungsdienste müssen aktiviert sein, um Inhalte am aktuellen Standort anzuzeigen. Willst du die Ortungsdienste aktivieren?",
+ "map_location_service_disabled_title": "Ortungsdienste deaktiviert",
+ "map_no_assets_in_bounds": "Keine Fotos in dieser Gegend",
+ "map_no_location_permission_content": "Ortungsdienste müssen aktiviert sein, um Inhalte am aktuellen Standort anzuzeigen. Willst du die Ortungsdienste aktivieren?",
+ "map_no_location_permission_title": "Kein Zugriff auf den Standort",
+ "map_settings_dark_mode": "Dunkler Modus",
+ "map_settings_dialog_cancel": "Abbrechen",
+ "map_settings_dialog_save": "Speichern",
+ "map_settings_dialog_title": "Karteneinstellungen",
+ "map_settings_include_show_archived": "Archivierte anzeigen",
+ "map_settings_only_relative_range": "Datumsbereich",
+ "map_settings_only_show_favorites": "Nur Favoriten anzeigen",
+ "map_zoom_to_see_photos": "Ansicht verkleinern um Fotos zu sehen",
"monthly_title_text_date_format": "MMMM y",
- "motion_photos_page_title": "Live Photos",
+ "motion_photos_page_title": "Live-Fotos",
"notification_permission_dialog_cancel": "Abbrechen",
"notification_permission_dialog_content": "Um Benachrichtigungen zu aktivieren, navigiere zu Einstellungen und klicke \"Erlauben\"",
"notification_permission_dialog_settings": "Einstellungen",
@@ -231,33 +242,36 @@
"notification_permission_list_tile_enable_button": "Aktiviere Benachrichtigungen",
"notification_permission_list_tile_title": "Benachrichtigungs-Berechtigung",
"partner_page_add_partner": "Partner hinzufügen",
- "partner_page_empty_message": "Your photos are not yet shared with any partner.",
- "partner_page_no_more_users": "No more users to add",
- "partner_page_partner_add_failed": "Failed to add partner",
+ "partner_page_empty_message": "Deine Fotos sind noch nicht geteilt mit einem Partner",
+ "partner_page_no_more_users": "Keine weiteren Nutzer",
+ "partner_page_partner_add_failed": "Fehler beim Partner hinzufügen",
"partner_page_select_partner": "Partner auswählen",
"partner_page_shared_to_title": "Geteilt mit",
"partner_page_stop_sharing_content": "{} wird nicht mehr auf deine Fotos zugreifen können.",
- "partner_page_stop_sharing_title": "Stop sharing your photos?",
+ "partner_page_stop_sharing_title": "Deine Fotos nicht mehr teilen?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Trotzdem fortfahren",
- "permission_onboarding_get_started": "Get started",
+ "permission_onboarding_get_started": "Jetzt starten",
"permission_onboarding_go_to_settings": "Gehe zu Einstellungen",
"permission_onboarding_grant_permission": "Berechtigung erteilen",
"permission_onboarding_log_out": "Abmelden",
- "permission_onboarding_permission_denied": "Berechtigungen verweigert. Um Immich zu benutzen, Zugriff auf Fotos und Videos in Einstellungen erlauben.",
+ "permission_onboarding_permission_denied": "Berechtigung verweigert. Um Immich zu benutzen, muss Zugriff auf Fotos und Videos in Einstellungen erlaubt werden.",
"permission_onboarding_permission_granted": "Berechtigung erteilt! Du bist startklar.",
"permission_onboarding_permission_limited": "Berechtigungen unzureichend. Um Immich das Sichern von ganzen Sammlungen zu ermöglichen, muss der Zugriff auf alle Fotos und Videos in den Einstellungen erlaubt werden.",
"permission_onboarding_request": "Immich benötigt Berechtigung um auf deine Fotos und Videos zuzugreifen.",
"profile_drawer_app_logs": "Logs",
"profile_drawer_client_server_up_to_date": "App und Server sind aktuell",
+ "profile_drawer_documentation": "Dokumentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Einstellungen",
"profile_drawer_sign_out": "Abmelden",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Papierkorb",
"recently_added_page_title": "Zuletzt hinzugefügt",
"search_bar_hint": "Durchsuche deine Fotos",
"search_page_categories": "Kategorien",
"search_page_favorites": "Favoriten",
- "search_page_motion_photos": "Live Photos",
+ "search_page_motion_photos": "Live-Fotos",
"search_page_no_objects": "Keine Objektinformationen verfügbar",
"search_page_no_places": "Keine Informationen über Orte verfügbar",
"search_page_people": "Personen",
@@ -265,7 +279,7 @@
"search_page_recently_added": "Zuletzt hinzugefügt",
"search_page_screenshots": "Bildschirmfotos",
"search_page_selfies": "Selfies",
- "search_page_things": "Dinge",
+ "search_page_things": "Gegenstände und Tiere",
"search_page_videos": "Videos",
"search_page_view_all_button": "Alle anzeigen",
"search_page_your_activity": "Deine Aktivität",
@@ -274,55 +288,65 @@
"search_suggestion_list_smart_search_hint_2": "m:dein-suchbegriff",
"select_additional_user_for_sharing_page_suggestions": "Vorschläge",
"select_user_for_sharing_page_err_album": "Album konnte nicht erstellt werden",
- "select_user_for_sharing_page_share_suggestions": "Suggestions",
+ "select_user_for_sharing_page_share_suggestions": "Empfehlungen",
"server_info_box_app_version": "App Version",
+ "server_info_box_server_url": "Server-URL",
"server_info_box_server_version": "Server Version",
"setting_image_viewer_help": "Der Detailbildbetrachter lädt zuerst die kleine Miniaturansicht, dann die Vorschau in mittlerer Größe (falls aktiviert) und schließlich das Original (falls aktiviert).",
"setting_image_viewer_original_subtitle": "Aktivieren, um das Originalbild in voller Auflösung (groß!) zu laden. Deaktivieren, um den Datenverbrauch zu reduzieren (sowohl im Netzwerk als auch im Gerätespeicher).",
"setting_image_viewer_original_title": "Original laden",
"setting_image_viewer_preview_subtitle": "Aktivieren, um ein Bild mit mittlerer Auflösung zu laden. Deaktivieren, um entweder das Original direkt zu laden oder nur die Miniaturansicht zu verwenden.",
"setting_image_viewer_preview_title": "Vorschaubild laden",
- "setting_notifications_notify_failures_grace_period": "Notify background backup failures: {}",
+ "setting_notifications_notify_failures_grace_period": "Benachrichtigung über Fehler bei der Hintergrundsicherung: {}",
"setting_notifications_notify_hours": "{} Stunden",
"setting_notifications_notify_immediately": "sofort",
"setting_notifications_notify_minutes": "{} Minuten",
"setting_notifications_notify_never": "niemals",
"setting_notifications_notify_seconds": "{} Sekunden",
- "setting_notifications_single_progress_subtitle": "Detaillierte Upload Informationen für jedes Element.",
- "setting_notifications_single_progress_title": "Zeige Hintergrund-Sicherungs Detailfortschritt",
- "setting_notifications_subtitle": "Passe Deine Benachrichtigungen an",
+ "setting_notifications_single_progress_subtitle": "Detaillierter Upload-Fortschritt für jedes Element.",
+ "setting_notifications_single_progress_title": "Zeige detaillierten Fortschritt bei der Hintergrundsicherung",
+ "setting_notifications_subtitle": "Benachrichtigungen anpassen",
"setting_notifications_title": "Benachrichtigungen",
"setting_notifications_total_progress_subtitle": "Gesamter Upload-Fortschritt (abgeschlossen/Anzahl Elemente)",
- "setting_notifications_total_progress_title": "Zeige Hintergrundsicherungsfortschritt",
+ "setting_notifications_total_progress_title": "Zeige Gesamtfortschritt bei der Hintergrundsicherung",
"setting_pages_app_bar_settings": "Einstellungen",
"settings_require_restart": "Bitte starte Immich neu, um diese Einstellung anzuwenden.",
"share_add": "Hinzufügen",
"share_add_photos": "Fotos hinzufügen",
"share_add_title": "Titel hinzufügen",
"share_create_album": "Album erstellen",
+ "shared_album_activities_input_disable": "Kommentare sind deaktiviert.",
+ "shared_album_activities_input_hint": "Sag etwas",
+ "shared_album_activity_remove_content": "Möchtest du diese Aktivität entfernen?",
+ "shared_album_activity_remove_title": "Aktivität entfernen",
+ "shared_album_activity_setting_subtitle": "Lass andere reagieren.",
+ "shared_album_activity_setting_title": "Kommentare & Likes",
"share_dialog_preparing": "Vorbereiten...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Geteilte Links",
+ "shared_link_create_app_bar_title": "Link zum Teilen erstellen",
+ "shared_link_create_info": "Alle, die über den Link verfügen, können die Fotos sehen",
+ "shared_link_create_submit_button": "Link erstellen",
+ "shared_link_edit_allow_download": "Jeder darf herunterladen",
+ "shared_link_edit_allow_upload": "Jeder darf hochladen",
+ "shared_link_edit_app_bar_title": "Link bearbeiten",
+ "shared_link_edit_change_expiry": "Ablaufdatum bearbeiten",
+ "shared_link_edit_description": "Beschreibung",
+ "shared_link_edit_description_hint": "Beschreibung eingeben",
+ "shared_link_edit_expire_after": "Erlischt nach",
+ "shared_link_edit_password": "Passwort",
+ "shared_link_edit_password_hint": "Passwort eingeben",
+ "shared_link_edit_show_meta": "Metadaten anzeigen",
+ "shared_link_edit_submit_button": "Link aktualisieren",
+ "shared_link_empty": "Du hast keine geteilten Links",
+ "shared_link_manage_links": "Geteilte Links verwalten",
+ "share_done": "Fertig",
"share_invite": "Zum Album einladen",
"sharing_page_album": "Geteilte Alben",
"sharing_page_description": "Erstelle ein geteiltes Album um Fotos und Videos mit Personen in deinem Netzwerk zu teilen.",
"sharing_page_empty_list": "LEERE LISTE",
"sharing_silver_appbar_create_shared_album": "Neues geteiltes Album",
- "sharing_silver_appbar_shared_links": "Shared links",
- "sharing_silver_appbar_share_partner": "Teile mit Partner",
+ "sharing_silver_appbar_shared_links": "Geteilte Links",
+ "sharing_silver_appbar_share_partner": "Mit Partner teilen",
"tab_controller_nav_library": "Bibliothek",
"tab_controller_nav_photos": "Fotos",
"tab_controller_nav_search": "Suche",
@@ -337,30 +361,30 @@
"theme_setting_theme_title": "Theme",
"theme_setting_three_stage_loading_subtitle": "Das dreistufige Ladeverfahren kann die Performance beim Laden verbessern, erhöht allerdings den Datenverbrauch deutlich",
"theme_setting_three_stage_loading_title": "Dreistufiges Laden aktivieren",
- "translated_text_options": "Options",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
+ "translated_text_options": "Optionen",
+ "trash_page_delete": "Löschen",
+ "trash_page_delete_all": "Alle löschen",
+ "trash_page_empty_trash_btn": "Papierkorb leeren",
+ "trash_page_empty_trash_dialog_content": "Elemente im Papierkorb löschen? Diese Elemente werden dauerhaft von Immich entfernt",
"trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
- "upload_dialog_cancel": "Cancel",
- "upload_dialog_info": "Do you want to backup the selected Asset(s) to the server?",
- "upload_dialog_ok": "Upload",
- "upload_dialog_title": "Upload Asset",
+ "trash_page_info": "Elemente im Papierkorb werden nach {} Tagen endgültig gelöscht",
+ "trash_page_no_assets": "Keine Elemente im Papierkorb",
+ "trash_page_restore": "Wiederherstellen",
+ "trash_page_restore_all": "Alle wiederherstellen",
+ "trash_page_select_assets_btn": "Elemente auswählen",
+ "trash_page_select_btn": "Auswählen",
+ "trash_page_title": "Papierkorb ({})",
+ "upload_dialog_cancel": "Abbrechen",
+ "upload_dialog_info": "Willst du die ausgewählten Elemente auf dem Server sichern?",
+ "upload_dialog_ok": "Hochladen",
+ "upload_dialog_title": "Element hochladen",
"version_announcement_overlay_ack": "Ich habe verstanden",
"version_announcement_overlay_release_notes": "Änderungsprotokoll",
"version_announcement_overlay_text_1": "Hallo mein Freund! Es gibt eine neue Version von",
- "version_announcement_overlay_text_2": "Bitte nehm dir die Zeit und lese das ",
+ "version_announcement_overlay_text_2": "Bitte nehme dir die Zeit und lies das ",
"version_announcement_overlay_text_3": " und achte darauf, dass deine docker-compose und .env Dateien aktuell sind, vor allem wenn du ein System für automatische Updates benutzt (z.B. Watchtower).",
"version_announcement_overlay_title": "Neue Server-Version verfügbar \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Aus Stapel entfernen",
+ "viewer_stack_use_as_main_asset": "An Stapelanfang",
+ "viewer_unstack": "Stapel aufheben"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/en-US.json b/mobile/assets/i18n/en-US.json
index f710bec06..6d28890eb 100644
--- a/mobile/assets/i18n/en-US.json
+++ b/mobile/assets/i18n/en-US.json
@@ -1,6 +1,7 @@
{
"add_to_album_bottom_sheet_added": "Added to {album}",
"add_to_album_bottom_sheet_already_exists": "Already in {album}",
+ "advanced_settings_log_level_title": "Log level: {}",
"advanced_settings_prefer_remote_subtitle": "Some devices are painfully slow to load thumbnails from assets on the device. Activate this setting to load remote images instead.",
"advanced_settings_prefer_remote_title": "Prefer remote images",
"advanced_settings_self_signed_ssl_subtitle": "Skips SSL certificate verification for the server endpoint. Required for self-signed certificates.",
@@ -27,6 +28,9 @@
"album_viewer_page_share_add_users": "Add users",
"all_people_page_title": "People",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "No archived assets found",
"archive_page_title": "Archive ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamic layout",
@@ -103,6 +107,9 @@
"cache_settings_album_thumbnails": "Library page thumbnails ({} assets)",
"cache_settings_clear_cache_button": "Clear cache",
"cache_settings_clear_cache_button_title": "Clears the app's cache. This will significantly impact the app's performance until the cache has rebuilt.",
+ "cache_settings_duplicated_assets_clear_button": "CLEAR",
+ "cache_settings_duplicated_assets_subtitle": "Photos and videos that are black listed by the app",
+ "cache_settings_duplicated_assets_title": "Duplicated Assets ({})",
"cache_settings_image_cache_size": "Image cache size ({} assets)",
"cache_settings_statistics_album": "Library thumbnails",
"cache_settings_statistics_assets": "{} assets ({})",
@@ -112,9 +119,11 @@
"cache_settings_statistics_title": "Cache usage",
"cache_settings_subtitle": "Control the caching behaviour of the Immich mobile application",
"cache_settings_thumbnail_size": "Thumbnail cache size ({} assets)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Caching Settings",
"change_password_form_confirm_password": "Confirm Password",
- "change_password_form_description": "Hi {firstName} {lastName},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
+ "change_password_form_description": "Hi {name},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
"change_password_form_new_password": "New Password",
"change_password_form_password_mismatch": "Passwords do not match",
"change_password_form_reenter_new_password": "Re-enter New Password",
@@ -165,10 +174,15 @@
"home_page_add_to_album_conflicts": "Added {added} assets to album {album}. {failed} assets are already in the album.",
"home_page_add_to_album_err_local": "Can not add local assets to albums yet, skipping",
"home_page_add_to_album_success": "Added {added} assets to album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Can not archive local assets yet, skipping",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Building the timeline",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Can not favorite local assets yet, skipping",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "If this is your first time using the app, please make sure to choose a backup album(s) so that the timeline can populate photos and videos in the album(s).",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
"image_viewer_page_state_provider_download_error": "Download Error",
"image_viewer_page_state_provider_download_success": "Download Success",
@@ -185,6 +199,7 @@
"library_page_sort_title": "Album title",
"login_disabled": "Login has been disabled",
"login_form_api_exception": "API exception. Please check the server URL and try again.",
+ "login_form_back_button_text": "Back",
"login_form_button_text": "Login",
"login_form_email_hint": "youremail@email.com",
"login_form_endpoint_hint": "http://your-server-ip:port/api",
@@ -207,6 +222,10 @@
"login_form_server_error": "Could not connect to server.",
"login_password_changed_error": "There was an error updating your password",
"login_password_changed_success": "Password updated successfully",
+ "map_assets_in_bounds": {
+ "one": "{} photo",
+ "other": "{} photos"
+ },
"map_cannot_get_user_location": "Cannot get user's location",
"map_location_dialog_cancel": "Cancel",
"map_location_dialog_yes": "Yes",
@@ -216,6 +235,15 @@
"map_no_location_permission_content": "Location permission is needed to display assets from your current location. Do you want to allow it now?",
"map_no_location_permission_title": "Location Permission denied",
"map_settings_dark_mode": "Dark mode",
+ "map_settings_date_range_option_all": "All",
+ "map_settings_date_range_option_days": {
+ "one": "Past 24 hours",
+ "other": "Past {} days"
+ },
+ "map_settings_date_range_option_years": {
+ "one": "Past year",
+ "other": "Past {} years"
+ },
"map_settings_dialog_cancel": "Cancel",
"map_settings_dialog_save": "Save",
"map_settings_dialog_title": "Map Settings",
@@ -240,6 +268,7 @@
"partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
"partner_page_stop_sharing_title": "Stop sharing your photos?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continue anyway",
"permission_onboarding_get_started": "Get started",
"permission_onboarding_go_to_settings": "Go to settings",
@@ -250,12 +279,16 @@
"permission_onboarding_permission_limited": "Permission limited. To let Immich backup and manage your entire gallery collection, grant photo and video permissions in Settings.",
"permission_onboarding_request": "Immich requires permission to view your photos and videos.",
"profile_drawer_app_logs": "Logs",
+ "profile_drawer_client_out_of_date_major": "Mobile App is out of date. Please update to the latest major version.",
+ "profile_drawer_client_out_of_date_minor": "Mobile App is out of date. Please update to the latest minor version.",
"profile_drawer_client_server_up_to_date": "Client and Server are up-to-date",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
+ "profile_drawer_server_out_of_date_major": "Server is out of date. Please update to the latest major version.",
+ "profile_drawer_server_out_of_date_minor": "Server is out of date. Please update to the latest minor version.",
"profile_drawer_settings": "Settings",
"profile_drawer_sign_out": "Sign Out",
"profile_drawer_trash": "Trash",
- "profile_drawer_documentation": "Documentation",
- "profile_drawer_github": "GitHub",
"recently_added_page_title": "Recently Added",
"search_bar_hint": "Search your photos",
"search_page_categories": "Categories",
@@ -264,6 +297,13 @@
"search_page_no_objects": "No Objects Info Available",
"search_page_no_places": "No Places Info Available",
"search_page_people": "People",
+ "search_page_person_add_name_dialog_cancel": "Cancel",
+ "search_page_person_add_name_dialog_save": "Save",
+ "search_page_person_add_name_dialog_hint": "Name",
+ "search_page_person_add_name_dialog_title": "Add a name",
+ "search_page_person_add_name_subtitle": "Find them fast by name with search",
+ "search_page_person_add_name_title": "Add a name",
+ "search_page_person_edit_name": "Edit name",
"search_page_places": "Places",
"search_page_recently_added": "Recently added",
"search_page_screenshots": "Screenshots",
@@ -272,6 +312,7 @@
"search_page_videos": "Videos",
"search_page_view_all_button": "View all",
"search_page_your_activity": "Your activity",
+ "search_page_your_map": "Your Map",
"search_result_page_new_search_hint": "New Search",
"search_suggestion_list_smart_search_hint_1": "Smart search is enabled by default, to search for metadata use the syntax ",
"search_suggestion_list_smart_search_hint_2": "m:your-search-term",
@@ -279,8 +320,9 @@
"select_user_for_sharing_page_err_album": "Failed to create album",
"select_user_for_sharing_page_share_suggestions": "Suggestions",
"server_info_box_app_version": "App Version",
- "server_info_box_server_version": "Server Version",
"server_info_box_server_url": "Server URL",
+ "server_info_box_server_version": "Server Version",
+ "server_info_box_latest_release":"Latest Version",
"setting_image_viewer_help": "The detail viewer loads the small thumbnail first, then loads the medium-size preview (if enabled), finally loads the original (if enabled).",
"setting_image_viewer_original_subtitle": "Enable to load the original full-resolution image (large!). Disable to reduce data usage (both network and on device cache).",
"setting_image_viewer_original_title": "Load original image",
@@ -304,9 +346,23 @@
"share_add_photos": "Add photos",
"share_add_title": "Add a title",
"share_create_album": "Create album",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
+ "shared_album_section_people_action_error": "Error leaving/removing from album",
+ "shared_album_section_people_action_leave": "Remove user from album",
+ "shared_album_section_people_action_remove_user": "Remove user from album",
+ "shared_album_section_people_owner_label": "Owner",
+ "shared_album_section_people_title": "PEOPLE",
"share_dialog_preparing": "Preparing...",
"shared_link_app_bar_title": "Shared Links",
+ "shared_link_clipboard_copied_massage": "Copied to clipboard",
+ "shared_link_clipboard_text": "Link: {}\nPassword: {}",
"shared_link_create_app_bar_title": "Create link to share",
+ "shared_link_create_error": "Error while creating shared link",
"shared_link_create_info": "Let anyone with the link see the selected photo(s)",
"shared_link_create_submit_button": "Create link",
"shared_link_edit_allow_download": "Allow public user to download",
@@ -315,18 +371,54 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_expire_after_option_days": {
+ "one": "{} day",
+ "other": "{} days"
+ },
+ "shared_link_edit_expire_after_option_hours": {
+ "one": "{} hour",
+ "other": "{} hours"
+ },
+ "shared_link_edit_expire_after_option_minutes": {
+ "one": "{} minute",
+ "other": "{} minutes"
+ },
+ "shared_link_edit_expire_after_option_never": "Never",
"shared_link_edit_password": "Password",
"shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
+ "shared_link_error_server_url_fetch": "Cannot fetch the server url",
+ "shared_link_expired": "Expired",
+ "shared_link_expires_days": {
+ "one": "Expires in {} day",
+ "other": "Expires in {} days"
+ },
+ "shared_link_expires_hours": {
+ "one": "Expires in {} hour",
+ "other": "Expires in {} hours"
+ },
+ "shared_link_expires_minutes": {
+ "one": "Expires in {} minute",
+ "other": "Expires in {} minutes"
+ },
+ "shared_link_expires_seconds": {
+ "one": "Expires in {} second",
+ "other": "Expires in {} seconds"
+ },
+ "shared_link_expires_never": "Expires ∞",
+ "shared_link_info_chip_download": "Download",
+ "shared_link_info_chip_metadata": "EXIF",
+ "shared_link_info_chip_upload": "Upload",
"shared_link_manage_links": "Manage Shared links",
"share_done": "Done",
"share_invite": "Invite to album",
"sharing_page_album": "Shared albums",
"sharing_page_description": "Create shared albums to share photos and videos with people in your network.",
"sharing_page_empty_list": "EMPTY LIST",
- "sharing_silver_appbar_create_shared_album": "Create shared album",
+ "sharing_silver_appbar_create_shared_album": "New shared album",
"sharing_silver_appbar_shared_links": "Shared links",
"sharing_silver_appbar_share_partner": "Share with partner",
"tab_controller_nav_library": "Library",
@@ -367,14 +459,7 @@
"version_announcement_overlay_text_3": " and ensure your docker-compose and .env setup is up-to-date to prevent any misconfigurations, especially if you use WatchTower or any mechanism that handles updating your server application automatically.",
"version_announcement_overlay_title": "New Server Version Available \uD83C\uDF89",
"viewer_remove_from_stack": "Remove from Stack",
- "viewer_unstack": "Un-Stack",
- "cache_settings_tile_title": "Local Storage",
- "cache_settings_tile_subtitle": "Control the local storage behaviour",
"viewer_stack_use_as_main_asset": "Use as Main Asset",
- "app_bar_signout_dialog_title": "Sign out",
- "app_bar_signout_dialog_content": "Are you sure you wanna sign out?",
- "app_bar_signout_dialog_ok": "Yes",
- "shared_album_activities_input_hint": "Say something",
- "shared_album_activity_remove_title": "Delete Activity",
- "shared_album_activity_remove_content": "Do you want to delete this activity?"
+ "viewer_unstack": "Un-Stack",
+ "scaffold_body_error_occured": "Error occured"
}
diff --git a/mobile/assets/i18n/es-ES.json b/mobile/assets/i18n/es-ES.json
index 7e68fe759..f32a8b1f8 100644
--- a/mobile/assets/i18n/es-ES.json
+++ b/mobile/assets/i18n/es-ES.json
@@ -3,8 +3,8 @@
"add_to_album_bottom_sheet_already_exists": "Ya se encuentra en {album}",
"advanced_settings_prefer_remote_subtitle": "Algunos dispositivos tardan mucho en cargar las miniaturas de recursos encontrados el dispositivo. Activa esta opción para cargar imágenes remotas en su lugar.",
"advanced_settings_prefer_remote_title": "Preferir imágenes remotas",
- "advanced_settings_self_signed_ssl_subtitle": "Skips SSL certificate verification for the server endpoint. Required for self-signed certificates.",
- "advanced_settings_self_signed_ssl_title": "Allow self-signed SSL certificates",
+ "advanced_settings_self_signed_ssl_subtitle": "Omitir verificación del certificado SSL del servidor. Requerido para certificados autofirmados",
+ "advanced_settings_self_signed_ssl_title": "Permitir certificados autofirmados",
"advanced_settings_tile_subtitle": "Configuraciones avanzadas del usuario",
"advanced_settings_tile_title": "Avanzado",
"advanced_settings_troubleshooting_subtitle": "Habilitar funciones adicionales para solución de problemas",
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Error al cambiar el título del álbum ",
"album_viewer_appbar_share_leave": "Abandonar álbum ",
"album_viewer_appbar_share_remove": "Eliminar del álbum ",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Agregar usuarios",
"all_people_page_title": "Personas",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "¿Estás seguro que quieres cerrar sesión?",
+ "app_bar_signout_dialog_ok": "Sí",
+ "app_bar_signout_dialog_title": "Cerrar sesión",
"archive_page_no_archived_assets": "No se encontraron recursos archivados",
"archive_page_title": "Archivo ({})",
"asset_list_layout_settings_dynamic_layout_title": "Diseño dinámico",
@@ -59,7 +63,7 @@
"backup_controller_page_background_battery_info_title": "Optimizaciones de batería",
"backup_controller_page_background_charging": "Solo mientras se carga",
"backup_controller_page_background_configure_error": "Error al configurar el servicio en segundo plano",
- "backup_controller_page_background_delay": "Retraso en la copia de seguridad de nuevos activos: {}",
+ "backup_controller_page_background_delay": "Retraso en la copia de seguridad de nuevos elementos: {}",
"backup_controller_page_background_description": "Activa el servicio en segundo plano para copiar automáticamente cualquier nuevos archivos sin necesidad de abrir la aplicación.",
"backup_controller_page_background_is_off": "La copia de seguridad en segundo plano automática está desactivada",
"backup_controller_page_background_is_on": "La copia de seguridad en segundo plano automática está activada",
@@ -71,7 +75,7 @@
"backup_controller_page_backup_sub": "Fotos y videos respaldados",
"backup_controller_page_cancel": "Cancelar",
"backup_controller_page_created": "Creado el: {}",
- "backup_controller_page_desc_backup": "Active la copia de seguridad para cargar automáticamente los nuevos activos al servidor.",
+ "backup_controller_page_desc_backup": "Active la copia de seguridad para cargar automáticamente los nuevos elementos al servidor.",
"backup_controller_page_excluded": "Excluido:",
"backup_controller_page_failed": "Fallidos ({})",
"backup_controller_page_filename": "Nombre del archivo: {} [{}]",
@@ -94,11 +98,11 @@
"backup_controller_page_uploading_file_info": "Cargando información del archivo",
"backup_err_only_album": "No se puede eliminar el único álbum",
"backup_info_card_assets": "archivos",
- "backup_manual_cancelled": "Cancelled",
- "backup_manual_failed": "Failed",
- "backup_manual_in_progress": "Upload already in progress. Try after sometime",
- "backup_manual_success": "Success",
- "backup_manual_title": "Upload status",
+ "backup_manual_cancelled": "Cancelado",
+ "backup_manual_failed": "Fallido",
+ "backup_manual_in_progress": "Subida en progreso. Espere",
+ "backup_manual_success": "Éxito",
+ "backup_manual_title": "Estado de la subida",
"cache_settings_album_thumbnails": "Miniaturas de la página de la biblioteca ({} archivos)",
"cache_settings_clear_cache_button": "Borrar caché",
"cache_settings_clear_cache_button_title": "Borra la caché de la aplicación. Esto afectará significativamente el rendimiento de la aplicación hasta que se reconstruya la caché.",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Uso de caché",
"cache_settings_subtitle": "Controla el comportamiento del almacenamiento en caché de la aplicación móvil Immich",
"cache_settings_thumbnail_size": "Tamaño de la caché de miniaturas ({} archivos)",
+ "cache_settings_tile_subtitle": "Controla el comportamiento del almacenamiento local",
+ "cache_settings_tile_title": "Almacenamiento local",
"cache_settings_title": "Configuración de la caché",
"change_password_form_confirm_password": "Confirmar Contraseña",
- "change_password_form_description": "Hola {firstName} {lastName},\n\nEsta es la primera vez que inicias sesión en el sistema o se ha solicitado cambiar tu contraseña. Por favor, introduce la nueva contraseña a continuación.",
+ "change_password_form_description": "Hola {name},\n\nEsta es la primera vez que inicias sesión en el sistema o se ha solicitado cambiar tu contraseña. Por favor, introduce la nueva contraseña a continuación.",
"change_password_form_new_password": "Nueva Contraseña",
"change_password_form_password_mismatch": "Las contraseñas no coinciden",
"change_password_form_reenter_new_password": "Vuelve a ingresar la nueva contraseña",
@@ -130,10 +136,10 @@
"control_bottom_app_bar_delete": "Eliminar",
"control_bottom_app_bar_favorite": "Favorito",
"control_bottom_app_bar_share": "Compartir",
- "control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_share_to": "Enviar",
+ "control_bottom_app_bar_stack": "Apilar",
"control_bottom_app_bar_unarchive": "Desarchivar",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Subir",
"create_album_page_untitled": "Sin título",
"create_shared_album_page_create": "Crear",
"create_shared_album_page_share": "Compartir",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "Cancelar",
"delete_dialog_ok": "Eliminar",
"delete_dialog_title": "Eliminar Permanentemente",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Estás seguro que quieres eliminar este enlace compartido",
+ "delete_shared_link_dialog_title": "Eliminar enlace compartido",
"description_input_hint_text": "Agregar descripción...",
"description_input_submit_error": "Error al actualizar la descripción, verifica el registro para obtener más detalles",
"exif_bottom_sheet_description": "Agregar Descripción...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "{added} elementos agregados al álbum {album}.{failed} elementos ya existen en el álbum.",
"home_page_add_to_album_err_local": "Aún no se pueden agregar recursos locales a álbumes, omitiendo",
"home_page_add_to_album_success": "{added} elementos agregados al álbum {album}. ",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Los recursos locales no pueden ser archivados, omitiendo",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Construyendo la línea de tiempo",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Aún no se pueden archivar recursos locales, omitiendo",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Si esta es la primera vez que usas la app, por favor, asegúrate de elegir un álbum de respaldo para que la línea de tiempo pueda cargar fotos y videos en los álbumes.",
- "home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
+ "home_page_upload_err_limit": "Solo se pueden subir 30 elementos simultáneamente, omitiendo",
"image_viewer_page_state_provider_download_error": "Error de descarga",
"image_viewer_page_state_provider_download_success": "Descarga exitosa",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Error al compartir",
"library_page_albums": "Álbumes",
"library_page_archive": "Archivo",
"library_page_device_albums": "Álbumes en el dispositivo",
@@ -179,10 +190,10 @@
"library_page_new_album": "Nuevo álbum",
"library_page_sharing": "Compartiendo",
"library_page_sort_created": "Creado más recientemente",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Última modificación",
+ "library_page_sort_most_recent_photo": "Foto más reciente",
"library_page_sort_title": "Título del álbum",
- "login_disabled": "Login has been disabled",
+ "login_disabled": "El inicio de sesión ha sido desactivado",
"login_form_api_exception": "Excepción producida por API. Por favor, verifica el URL del servidor e inténtalo de nuevo.",
"login_form_button_text": "Iniciar Sesión",
"login_form_email_hint": "tucorreo@correo.com",
@@ -196,7 +207,7 @@
"login_form_failed_get_oauth_server_config": "Error al iniciar sesión con OAuth, verifica la URL del servidor",
"login_form_failed_get_oauth_server_disable": "La función de OAuth no está disponible en este servidor",
"login_form_failed_login": "Error al iniciar sesión, comprueba la URL del servidor, el correo electrónico y la contraseña",
- "login_form_handshake_exception": "There was an Handshake Exception with the server. Enable self-signed certificate support in the settings if you are using a self-signed certificate.",
+ "login_form_handshake_exception": "Hubo un error de verificación del certificado del servidor. Activa el soporte para certificados autofirmados en las preferencias si estás usando un certificado autofirmado",
"login_form_label_email": "Correo",
"login_form_label_password": "Contraseña",
"login_form_next_button": "Siguiente",
@@ -204,24 +215,24 @@
"login_form_save_login": "Mantener la sesión iniciada",
"login_form_server_empty": "Agrega la URL del servidor.",
"login_form_server_error": "No se pudo conectar al servidor.",
- "login_password_changed_error": "There was an error updating your password",
- "login_password_changed_success": "Password updated successfully",
- "map_cannot_get_user_location": "Cannot get user's location",
- "map_location_dialog_cancel": "Cancel",
- "map_location_dialog_yes": "Yes",
- "map_location_service_disabled_content": "Location service needs to be enabled to display assets from your current location. Do you want to enable it now?",
- "map_location_service_disabled_title": "Location Service disabled",
- "map_no_assets_in_bounds": "No photos in this area",
- "map_no_location_permission_content": "Location permission is needed to display assets from your current location. Do you want to allow it now?",
- "map_no_location_permission_title": "Location Permission denied",
- "map_settings_dark_mode": "Dark mode",
- "map_settings_dialog_cancel": "Cancel",
- "map_settings_dialog_save": "Save",
- "map_settings_dialog_title": "Map Settings",
- "map_settings_include_show_archived": "Include Archived",
- "map_settings_only_relative_range": "Date range",
- "map_settings_only_show_favorites": "Show Favorite Only",
- "map_zoom_to_see_photos": "Zoom out to see photos",
+ "login_password_changed_error": "Hubo un error actualizando la contraseña",
+ "login_password_changed_success": "Contraseña cambiado con éxito",
+ "map_cannot_get_user_location": "No se pudo obtener la posición del usuario",
+ "map_location_dialog_cancel": "Cancelar",
+ "map_location_dialog_yes": "Sí",
+ "map_location_service_disabled_content": "Los servicios de ubicación deben estar activados para mostrar elementos de tu ubicación actual. Deseas activarlos ahora?",
+ "map_location_service_disabled_title": "Servicios de ubicación desactivados",
+ "map_no_assets_in_bounds": "No hay fotos en esta zona",
+ "map_no_location_permission_content": "Se necesitan permisos de ubicación para mostrar elementos de tu ubicación actual. Deseas activarlos ahora?",
+ "map_no_location_permission_title": "Permisos de ubicación denegados",
+ "map_settings_dark_mode": "Modo oscuro",
+ "map_settings_dialog_cancel": "Cancelar",
+ "map_settings_dialog_save": "Guardar",
+ "map_settings_dialog_title": "Ajustes mapa",
+ "map_settings_include_show_archived": "Incluir archivados",
+ "map_settings_only_relative_range": "Rango de fechas",
+ "map_settings_only_show_favorites": "Mostrar solo favoritas",
+ "map_zoom_to_see_photos": "Alejar para ver fotos",
"monthly_title_text_date_format": "MMMM y",
"motion_photos_page_title": "Foto en Movimiento",
"notification_permission_dialog_cancel": "Cancelar",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} ya no podrá acceder a tus fotos",
"partner_page_stop_sharing_title": "¿Dejar de compartir tus fotos?",
"partner_page_title": "Compañero",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continuar de todos modos",
"permission_onboarding_get_started": "Empezar",
"permission_onboarding_go_to_settings": "Ir a configuración",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich requiere permiso para ver tus fotos y videos.",
"profile_drawer_app_logs": "Registros",
"profile_drawer_client_server_up_to_date": "El Cliente y el Servidor están actualizados",
+ "profile_drawer_documentation": "Documentación",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Configuración",
"profile_drawer_sign_out": "Cerrar Sesión",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Papelera",
"recently_added_page_title": "Recién Agregadas",
"search_bar_hint": "Busca tus fotos",
"search_page_categories": "Categorías",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Fallo al crear el álbum",
"select_user_for_sharing_page_share_suggestions": "Sugerencias",
"server_info_box_app_version": "Versión de la Aplicación",
+ "server_info_box_server_url": "URL del servidor",
"server_info_box_server_version": "Versión del Servidor",
"setting_image_viewer_help": "El visor de detalles carga primero la miniatura pequeña, luego carga la vista previa de tamaño mediano (si está habilitada), finalmente carga la original (si está habilitada).",
"setting_image_viewer_original_subtitle": "Activar para cargar la imagen en resolución original (¡muy grande!). Deshabilitar para reducir el consumo de datos (de red y caché).",
@@ -300,35 +315,44 @@
"share_add_photos": "Agregar fotos",
"share_add_title": "Agregar un título",
"share_create_album": "Crear álbum",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Preparando...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Enlaces compartidos",
+ "shared_link_create_app_bar_title": "Crear enlace compartido",
+ "shared_link_create_info": "Cualquier persona con el enlace puede ver las fotos seleccionadas",
+ "shared_link_create_submit_button": "Crear enlace",
+ "shared_link_edit_allow_download": "Permitir descargar a usuarios públicos",
+ "shared_link_edit_allow_upload": "Permitir subir a usuarios públicos",
+ "shared_link_edit_app_bar_title": "Editar enlace",
+ "shared_link_edit_change_expiry": "Cambiar fecha de caducidad",
+ "shared_link_edit_description": "Descripción",
+ "shared_link_edit_description_hint": "Introduce la descripción del enlace",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Contraseña",
+ "shared_link_edit_password_hint": "Introduce la contraseña del enlace",
+ "shared_link_edit_show_meta": "Mostrar metadatos",
+ "shared_link_edit_submit_button": "Actualizar enlace",
+ "shared_link_empty": "No tienes enlaces compartidos",
+ "shared_link_manage_links": "Administrar enlaces compartidos",
+ "share_done": "Hecho",
"share_invite": "Invitar al álbum",
"sharing_page_album": "Álbumes compartidos",
"sharing_page_description": "Crea álbumes compartidos para compartir fotos y vídeos con las personas de tu red.",
"sharing_page_empty_list": "LISTA VACIA",
"sharing_silver_appbar_create_shared_album": "Crear un álbum compartido",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Enlaces compartidos",
"sharing_silver_appbar_share_partner": "Compartir con el compañero",
"tab_controller_nav_library": "Biblioteca",
"tab_controller_nav_photos": "Fotos",
"tab_controller_nav_search": "Buscar",
"tab_controller_nav_sharing": "Compartiendo",
"theme_setting_asset_list_storage_indicator_title": "Mostrar indicador de almacenamiento en las miniaturas de los archivos",
- "theme_setting_asset_list_tiles_per_row_title": "Número de activos por fila ({})",
+ "theme_setting_asset_list_tiles_per_row_title": "Número de elementos por fila ({})",
"theme_setting_dark_mode_switch": "Modo oscuro",
"theme_setting_image_viewer_quality_subtitle": "Ajustar la calidad del visor de detalles de imágenes",
"theme_setting_image_viewer_quality_title": "Calidad del visor de imágenes",
@@ -337,30 +361,30 @@
"theme_setting_theme_title": "Tema",
"theme_setting_three_stage_loading_subtitle": "La carga en tres etapas puede aumentar el rendimiento de carga pero provoca un consumo de red significativamente mayor",
"theme_setting_three_stage_loading_title": "Activar carga en tres etapas",
- "translated_text_options": "Options",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
- "trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
- "upload_dialog_cancel": "Cancel",
- "upload_dialog_info": "Do you want to backup the selected Asset(s) to the server?",
- "upload_dialog_ok": "Upload",
- "upload_dialog_title": "Upload Asset",
+ "translated_text_options": "Opciones",
+ "trash_page_delete": "Eliminar",
+ "trash_page_delete_all": "Eliminar todos",
+ "trash_page_empty_trash_btn": "Vaciar papelera",
+ "trash_page_empty_trash_dialog_content": "Estás seguro que quieres eliminar los elementos? Estos elementos serán eliminados de Immich permanentemente",
+ "trash_page_empty_trash_dialog_ok": "Sí",
+ "trash_page_info": "Los archivos en la papelera serán eliminados automáticamente después de {} días",
+ "trash_page_no_assets": "No hay elementos en la papelera",
+ "trash_page_restore": "Restaurar",
+ "trash_page_restore_all": "Restaurar todos",
+ "trash_page_select_assets_btn": "Seleccionar elementos",
+ "trash_page_select_btn": "Seleccionar",
+ "trash_page_title": "Papelera ({})",
+ "upload_dialog_cancel": "Cancelar",
+ "upload_dialog_info": "Quieres hacer una copia de seguridad al servidor de los elementos seleccionados?",
+ "upload_dialog_ok": "Subir",
+ "upload_dialog_title": "Subir elementos",
"version_announcement_overlay_ack": "Aceptar",
"version_announcement_overlay_release_notes": "notas de versión",
"version_announcement_overlay_text_1": "Hola amigo, hay una nueva versión de",
"version_announcement_overlay_text_2": "por favor, tómate tu tiempo para visitar las ",
"version_announcement_overlay_text_3": " y asegúrate de que la configuración de docker-compose y .env estén actualizadas para evitar cualquier error de configuración, especialmente si utilizas WatchTower o cualquier mecanismo que actualice automáticamente la aplicación del servidor.",
"version_announcement_overlay_title": "Nueva versión del servidor disponible \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Quitar de la pila",
+ "viewer_stack_use_as_main_asset": "Usar como elemento principal",
+ "viewer_unstack": "Desapilar"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/es-MX.json b/mobile/assets/i18n/es-MX.json
index 55299a262..d140b60ee 100644
--- a/mobile/assets/i18n/es-MX.json
+++ b/mobile/assets/i18n/es-MX.json
@@ -3,8 +3,8 @@
"add_to_album_bottom_sheet_already_exists": "Ya se encuentra en {album}",
"advanced_settings_prefer_remote_subtitle": "Algunos dispositivos tardan mucho en cargar las miniaturas de recursos encontrados el dispositivo. Activa esta opción para cargar imágenes remotas en su lugar.",
"advanced_settings_prefer_remote_title": "Preferir imágenes remotas",
- "advanced_settings_self_signed_ssl_subtitle": "Skips SSL certificate verification for the server endpoint. Required for self-signed certificates.",
- "advanced_settings_self_signed_ssl_title": "Allow self-signed SSL certificates",
+ "advanced_settings_self_signed_ssl_subtitle": "Omitir verificación del certificado SSL del servidor. Requerido para certificados autofirmados",
+ "advanced_settings_self_signed_ssl_title": "Permitir certificados autofirmados",
"advanced_settings_tile_subtitle": "Configuraciones avanzadas del usuario",
"advanced_settings_tile_title": "Avanzado",
"advanced_settings_troubleshooting_subtitle": "Habilitar funciones adicionales para solución de problemas",
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Error al cambiar el título del álbum",
"album_viewer_appbar_share_leave": "Abandonar álbum ",
"album_viewer_appbar_share_remove": "Eliminar del álbum",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Agregar usuarios",
"all_people_page_title": "Personas",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "¿Estás seguro que quieres cerrar sesión?",
+ "app_bar_signout_dialog_ok": "Sí",
+ "app_bar_signout_dialog_title": "Cerrar sesión",
"archive_page_no_archived_assets": "No se encontraron recursos archivados",
"archive_page_title": "Archivo ({})",
"asset_list_layout_settings_dynamic_layout_title": "Diseño dinámico",
@@ -59,7 +63,7 @@
"backup_controller_page_background_battery_info_title": "Optimizaciones de batería",
"backup_controller_page_background_charging": "Solo mientras se carga",
"backup_controller_page_background_configure_error": "Error al configurar el servicio en segundo plano",
- "backup_controller_page_background_delay": "Retraso en la copia de seguridad de nuevos activos: {}",
+ "backup_controller_page_background_delay": "Retraso en la copia de seguridad de nuevos elementos: {}",
"backup_controller_page_background_description": "Activa el servicio en segundo plano para copiar automáticamente cualquier nuevos archivos sin necesidad de abrir la aplicación.",
"backup_controller_page_background_is_off": "La copia de seguridad en segundo plano automática está desactivada",
"backup_controller_page_background_is_on": "La copia de seguridad en segundo plano automática está desactivada",
@@ -94,11 +98,11 @@
"backup_controller_page_uploading_file_info": "Cargando información del archivo",
"backup_err_only_album": "No se puede eliminar el único álbum",
"backup_info_card_assets": "archivos",
- "backup_manual_cancelled": "Cancelled",
- "backup_manual_failed": "Failed",
- "backup_manual_in_progress": "Upload already in progress. Try after sometime",
- "backup_manual_success": "Success",
- "backup_manual_title": "Upload status",
+ "backup_manual_cancelled": "Cancelado",
+ "backup_manual_failed": "Fallido",
+ "backup_manual_in_progress": "Subida en progreso. Espere",
+ "backup_manual_success": "Éxito",
+ "backup_manual_title": "Estado de la subida",
"cache_settings_album_thumbnails": "Miniaturas de la página de la biblioteca ({} archivos)",
"cache_settings_clear_cache_button": "Borrar caché",
"cache_settings_clear_cache_button_title": "Borra la caché de la aplicación. Esto afectará significativamente el rendimiento de la aplicación hasta que se reconstruya la caché.",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Uso de caché",
"cache_settings_subtitle": "Controla el comportamiento del almacenamiento en caché de la aplicación móvil Immich",
"cache_settings_thumbnail_size": "Tamaño de la caché de miniaturas ({} archivos)",
+ "cache_settings_tile_subtitle": "Controla el comportamiento del almacenamiento local",
+ "cache_settings_tile_title": "Almacenamiento local",
"cache_settings_title": "Configuración de la caché",
"change_password_form_confirm_password": "Confirmar Contraseña",
- "change_password_form_description": "Hola {firstName} {lastName},\n\nEsta es la primera vez que inicias sesión en el sistema o se ha solicitado cambiar tu contraseña. Por favor, introduce la nueva contraseña a continuación.",
+ "change_password_form_description": "Hola {name},\n\nEsta es la primera vez que inicias sesión en el sistema o se ha solicitado cambiar tu contraseña. Por favor, introduce la nueva contraseña a continuación.",
"change_password_form_new_password": "Nueva Contraseña",
"change_password_form_password_mismatch": "Las contraseñas no coinciden",
"change_password_form_reenter_new_password": "Vuelve a ingresar la nueva contraseña",
@@ -130,10 +136,10 @@
"control_bottom_app_bar_delete": "Eliminar",
"control_bottom_app_bar_favorite": "Favorito",
"control_bottom_app_bar_share": "Compartir",
- "control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_share_to": "Enviar",
+ "control_bottom_app_bar_stack": "Apilar",
"control_bottom_app_bar_unarchive": "Desarchivar",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Subir",
"create_album_page_untitled": "Sin título",
"create_shared_album_page_create": "Crear",
"create_shared_album_page_share": "Compartir",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "Cancelar",
"delete_dialog_ok": "Eliminar",
"delete_dialog_title": "Eliminar permanentemente",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Estás seguro que quieres eliminar este enlace compartido",
+ "delete_shared_link_dialog_title": "Eliminar enlace compartido",
"description_input_hint_text": "Agregar descripción...",
"description_input_submit_error": "Error al actualizar la descripción, verifica el registro para obtener más detalles",
"exif_bottom_sheet_description": "Agregar Descripción...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "{added} elementos agregados al álbum {album}.\n{failed} elementos ya existen en el álbum.",
"home_page_add_to_album_err_local": "Aún no se pueden agregar recursos locales a álbumes, omitiendo",
"home_page_add_to_album_success": "{added} elementos agregados al álbum {album}. ",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Los recursos locales no pueden ser archivados, omitiendo",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Construyendo la línea de tiempo",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Aún no se pueden archivar recursos locales, omitiendo",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Si esta es la primera vez que usas la app, por favor, asegúrate de elegir un álbum de respaldo para que la línea de tiempo pueda cargar fotos y videos en los álbumes.",
- "home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
+ "home_page_upload_err_limit": "Solo se pueden subir 30 elementos simultáneamente, omitiendo",
"image_viewer_page_state_provider_download_error": "Error de descarga",
"image_viewer_page_state_provider_download_success": "Descarga exitosa",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Error al compartir",
"library_page_albums": "Álbumes",
"library_page_archive": "Archivo",
"library_page_device_albums": "Álbumes en el dispositivo",
@@ -179,10 +190,10 @@
"library_page_new_album": "Nuevo álbum",
"library_page_sharing": "Compartiendo",
"library_page_sort_created": "Creado más recientemente",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Última modificación",
+ "library_page_sort_most_recent_photo": "Foto más reciente",
"library_page_sort_title": "Título del álbum",
- "login_disabled": "Login has been disabled",
+ "login_disabled": "El inicio de sesión ha sido desactivado",
"login_form_api_exception": "Excepción producida por API. Por favor, verifica el URL del servidor e inténtalo de nuevo.",
"login_form_button_text": "Iniciar sesión",
"login_form_email_hint": "tucorreo@correo.com",
@@ -196,7 +207,7 @@
"login_form_failed_get_oauth_server_config": "Error al iniciar sesión con OAuth, verifica la URL del servidor",
"login_form_failed_get_oauth_server_disable": "La función de OAuth no está disponible en este servidor",
"login_form_failed_login": "Error al iniciar sesión, comprueba la URL del servidor, el correo electrónico y la contraseña",
- "login_form_handshake_exception": "There was an Handshake Exception with the server. Enable self-signed certificate support in the settings if you are using a self-signed certificate.",
+ "login_form_handshake_exception": "Hubo un error de verificación del certificado del servidor. Activa el soporte para certificados autofirmados en las preferencias si estás usando un certificado autofirmado",
"login_form_label_email": "Correo electrónico",
"login_form_label_password": "Contraseña",
"login_form_next_button": "Siguiente",
@@ -204,24 +215,24 @@
"login_form_save_login": "Permanecer conectado",
"login_form_server_empty": "Agrega la URL del servidor.",
"login_form_server_error": "No se pudo conectar al servidor.",
- "login_password_changed_error": "There was an error updating your password",
- "login_password_changed_success": "Password updated successfully",
- "map_cannot_get_user_location": "Cannot get user's location",
- "map_location_dialog_cancel": "Cancel",
- "map_location_dialog_yes": "Yes",
- "map_location_service_disabled_content": "Location service needs to be enabled to display assets from your current location. Do you want to enable it now?",
- "map_location_service_disabled_title": "Location Service disabled",
- "map_no_assets_in_bounds": "No photos in this area",
- "map_no_location_permission_content": "Location permission is needed to display assets from your current location. Do you want to allow it now?",
- "map_no_location_permission_title": "Location Permission denied",
- "map_settings_dark_mode": "Dark mode",
- "map_settings_dialog_cancel": "Cancel",
- "map_settings_dialog_save": "Save",
- "map_settings_dialog_title": "Map Settings",
- "map_settings_include_show_archived": "Include Archived",
- "map_settings_only_relative_range": "Date range",
- "map_settings_only_show_favorites": "Show Favorite Only",
- "map_zoom_to_see_photos": "Zoom out to see photos",
+ "login_password_changed_error": "Hubo un error actualizando la contraseña",
+ "login_password_changed_success": "Contraseña cambiado con éxito",
+ "map_cannot_get_user_location": "No se pudo obtener la posición del usuario",
+ "map_location_dialog_cancel": "Cancelar",
+ "map_location_dialog_yes": "Sí",
+ "map_location_service_disabled_content": "Los servicios de localización deben estar activados para mostrar elementos de tu ubicación actual. Deseas activarlos ahora?",
+ "map_location_service_disabled_title": "Servicios de localización desactivados",
+ "map_no_assets_in_bounds": "No hay fotos en esta zona",
+ "map_no_location_permission_content": "Se necesitan permisos de ubicación para mostrar elementos de tu ubicación actual. Deseas activarlos ahora?",
+ "map_no_location_permission_title": "Permisos de ubicación denegados",
+ "map_settings_dark_mode": "Modo oscuro",
+ "map_settings_dialog_cancel": "Cancelar",
+ "map_settings_dialog_save": "Guardar",
+ "map_settings_dialog_title": "Ajustes mapa",
+ "map_settings_include_show_archived": "Incluir archivados",
+ "map_settings_only_relative_range": "Rango de fechas",
+ "map_settings_only_show_favorites": "Mostrar solo favoritas",
+ "map_zoom_to_see_photos": "Alejar para ver fotos",
"monthly_title_text_date_format": "MMMM y",
"motion_photos_page_title": "Foto en Movimiento",
"notification_permission_dialog_cancel": "Cancelar",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} ya no podrá acceder a tus fotos",
"partner_page_stop_sharing_title": "¿Dejar de compartir tus fotos?",
"partner_page_title": "Compañero",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continuar de todos modos",
"permission_onboarding_get_started": "Empezar",
"permission_onboarding_go_to_settings": "Ir a configuración",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich requiere permiso para ver tus fotos y videos.",
"profile_drawer_app_logs": "Registros",
"profile_drawer_client_server_up_to_date": "El cliente y el servidor están actualizados",
+ "profile_drawer_documentation": "Documentación",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Configuración",
"profile_drawer_sign_out": "Cerrar sesión",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Papelera",
"recently_added_page_title": "Recién Agregadas",
"search_bar_hint": "Busca tus fotos",
"search_page_categories": "Categorías",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Error al crear álbum",
"select_user_for_sharing_page_share_suggestions": "Sugerencias",
"server_info_box_app_version": "Versión de la Aplicación",
+ "server_info_box_server_url": "URL del servidor",
"server_info_box_server_version": "Versión del Servidor",
"setting_image_viewer_help": "El visor de detalles carga primero la miniatura pequeña, luego carga la vista previa de tamaño mediano (si está habilitada), finalmente carga la original (si está habilitada).",
"setting_image_viewer_original_subtitle": "Activar para cargar la imagen en resolución original (¡muy grande!). Deshabilitar para reducir el consumo de datos (de red y caché).",
@@ -300,35 +315,44 @@
"share_add_photos": "Agregar fotos",
"share_add_title": "Agregar un título",
"share_create_album": "Crear álbum",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Preparando...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Enlaces compartidos",
+ "shared_link_create_app_bar_title": "Crear enlace compartido",
+ "shared_link_create_info": "Cualquier persona con el enlace puede ver las fotos seleccionadas",
+ "shared_link_create_submit_button": "Crear enlace",
+ "shared_link_edit_allow_download": "Permitir descargar a usuarios públicos",
+ "shared_link_edit_allow_upload": "Permitir subir a usuarios públicos",
+ "shared_link_edit_app_bar_title": "Editar enlace",
+ "shared_link_edit_change_expiry": "Cambiar fecha de caducidad",
+ "shared_link_edit_description": "Descripción",
+ "shared_link_edit_description_hint": "Introduce la descripción del enlace",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Contraseña",
+ "shared_link_edit_password_hint": "Introduce la contraseña del enlace",
+ "shared_link_edit_show_meta": "Mostrar metadatos",
+ "shared_link_edit_submit_button": "Actualizar enlace",
+ "shared_link_empty": "No tienes enlaces compartidos",
+ "shared_link_manage_links": "Administrar enlaces compartidos",
+ "share_done": "Hecho",
"share_invite": "Invitar al álbum",
"sharing_page_album": "Álbumes compartidos",
"sharing_page_description": "Crea álbumes compartidos para compartir fotos y videos con personas de tu red.",
"sharing_page_empty_list": "LISTA VACIA",
"sharing_silver_appbar_create_shared_album": "Crear álbum compartido",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Enlaces compartidos",
"sharing_silver_appbar_share_partner": "Compartir con compañero",
"tab_controller_nav_library": "Biblioteca",
"tab_controller_nav_photos": "Fotos",
"tab_controller_nav_search": "Buscar",
"tab_controller_nav_sharing": "Compartiendo",
"theme_setting_asset_list_storage_indicator_title": "Mostrar indicador de almacenamiento en las miniaturas de los archivos",
- "theme_setting_asset_list_tiles_per_row_title": "Número de activos por fila ({})",
+ "theme_setting_asset_list_tiles_per_row_title": "Número de elementos por fila ({})",
"theme_setting_dark_mode_switch": "Modo oscuro",
"theme_setting_image_viewer_quality_subtitle": "Ajustar la calidad del visor de detalles de imágenes",
"theme_setting_image_viewer_quality_title": "Calidad del visor de imágenes",
@@ -337,30 +361,30 @@
"theme_setting_theme_title": "Tema",
"theme_setting_three_stage_loading_subtitle": "La carga en tres etapas puede aumentar el rendimiento de carga pero provoca un consumo de red significativamente mayor",
"theme_setting_three_stage_loading_title": "Activar carga en tres etapas",
- "translated_text_options": "Options",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
- "trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
- "upload_dialog_cancel": "Cancel",
- "upload_dialog_info": "Do you want to backup the selected Asset(s) to the server?",
- "upload_dialog_ok": "Upload",
- "upload_dialog_title": "Upload Asset",
+ "translated_text_options": "Opciones",
+ "trash_page_delete": "Eliminar",
+ "trash_page_delete_all": "Eliminar todos",
+ "trash_page_empty_trash_btn": "Vaciar papelera",
+ "trash_page_empty_trash_dialog_content": "Estás seguro que quieres eliminar los elementos? Estos elementos serán eliminados de Immich permanentemente",
+ "trash_page_empty_trash_dialog_ok": "Sí",
+ "trash_page_info": "Los archivos en la papelera serán eliminados automáticamente después de {} días",
+ "trash_page_no_assets": "No hay elementos en la papelera",
+ "trash_page_restore": "Restaurar",
+ "trash_page_restore_all": "Restaurar todos",
+ "trash_page_select_assets_btn": "Seleccionar elementos",
+ "trash_page_select_btn": "Seleccionar",
+ "trash_page_title": "Papelera ({})",
+ "upload_dialog_cancel": "Cancelar",
+ "upload_dialog_info": "Quieres hacer una copia de seguridad al servidor de los elementos seleccionados?",
+ "upload_dialog_ok": "Subir",
+ "upload_dialog_title": "Subir elementos",
"version_announcement_overlay_ack": "Aceptar",
"version_announcement_overlay_release_notes": "notas de la versión",
"version_announcement_overlay_text_1": "Hola, amigo, hay una nueva versión de",
"version_announcement_overlay_text_2": "por favor, tómate tu tiempo para visitar las ",
"version_announcement_overlay_text_3": " y asegúrate de que la configuración de docker-compose y .env estén actualizadas para evitar cualquier error de configuración, especialmente si utilizas WatchTower o cualquier mecanismo que actualice automáticamente la aplicación del servidor.",
"version_announcement_overlay_title": "Nueva versión del servidor disponible \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Quitar de la pila",
+ "viewer_stack_use_as_main_asset": "Usar como elemento principal",
+ "viewer_unstack": "Desapilar"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/es-PE.json b/mobile/assets/i18n/es-PE.json
index 55299a262..0e03e2fd6 100644
--- a/mobile/assets/i18n/es-PE.json
+++ b/mobile/assets/i18n/es-PE.json
@@ -3,8 +3,8 @@
"add_to_album_bottom_sheet_already_exists": "Ya se encuentra en {album}",
"advanced_settings_prefer_remote_subtitle": "Algunos dispositivos tardan mucho en cargar las miniaturas de recursos encontrados el dispositivo. Activa esta opción para cargar imágenes remotas en su lugar.",
"advanced_settings_prefer_remote_title": "Preferir imágenes remotas",
- "advanced_settings_self_signed_ssl_subtitle": "Skips SSL certificate verification for the server endpoint. Required for self-signed certificates.",
- "advanced_settings_self_signed_ssl_title": "Allow self-signed SSL certificates",
+ "advanced_settings_self_signed_ssl_subtitle": "Omitir verificación del certificado SSL del servidor. Requerido para certificados autofirmados",
+ "advanced_settings_self_signed_ssl_title": "Permitir certificados autofirmados",
"advanced_settings_tile_subtitle": "Configuraciones avanzadas del usuario",
"advanced_settings_tile_title": "Avanzado",
"advanced_settings_troubleshooting_subtitle": "Habilitar funciones adicionales para solución de problemas",
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Error al cambiar el título del álbum",
"album_viewer_appbar_share_leave": "Abandonar álbum ",
"album_viewer_appbar_share_remove": "Eliminar del álbum",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Agregar usuarios",
"all_people_page_title": "Personas",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "¿Estás seguro que quieres cerrar sesión?",
+ "app_bar_signout_dialog_ok": "Sí",
+ "app_bar_signout_dialog_title": "Cerrar sesión",
"archive_page_no_archived_assets": "No se encontraron recursos archivados",
"archive_page_title": "Archivo ({})",
"asset_list_layout_settings_dynamic_layout_title": "Diseño dinámico",
@@ -59,7 +63,7 @@
"backup_controller_page_background_battery_info_title": "Optimizaciones de batería",
"backup_controller_page_background_charging": "Solo mientras se carga",
"backup_controller_page_background_configure_error": "Error al configurar el servicio en segundo plano",
- "backup_controller_page_background_delay": "Retraso en la copia de seguridad de nuevos activos: {}",
+ "backup_controller_page_background_delay": "Retraso en la copia de seguridad de nuevos elementos: {}",
"backup_controller_page_background_description": "Activa el servicio en segundo plano para copiar automáticamente cualquier nuevos archivos sin necesidad de abrir la aplicación.",
"backup_controller_page_background_is_off": "La copia de seguridad en segundo plano automática está desactivada",
"backup_controller_page_background_is_on": "La copia de seguridad en segundo plano automática está desactivada",
@@ -71,7 +75,7 @@
"backup_controller_page_backup_sub": "Fotos y videos respaldados",
"backup_controller_page_cancel": "Cancelar",
"backup_controller_page_created": "Creado el: {}",
- "backup_controller_page_desc_backup": "Activa la copia de seguridad en primer plano para cargar automáticamente nuevos recursos al servidor al abrir la aplicación.",
+ "backup_controller_page_desc_backup": "Active la copia de seguridad para cargar automáticamente los nuevos elementos al servidor.",
"backup_controller_page_excluded": "Excluido:",
"backup_controller_page_failed": "Fallidos ({})",
"backup_controller_page_filename": "Nombre del archivo: {} [{}]",
@@ -94,11 +98,11 @@
"backup_controller_page_uploading_file_info": "Cargando información del archivo",
"backup_err_only_album": "No se puede eliminar el único álbum",
"backup_info_card_assets": "archivos",
- "backup_manual_cancelled": "Cancelled",
- "backup_manual_failed": "Failed",
- "backup_manual_in_progress": "Upload already in progress. Try after sometime",
- "backup_manual_success": "Success",
- "backup_manual_title": "Upload status",
+ "backup_manual_cancelled": "Cancelado",
+ "backup_manual_failed": "Fallido",
+ "backup_manual_in_progress": "Subida en progreso. Espere",
+ "backup_manual_success": "Éxito",
+ "backup_manual_title": "Estado de la subida",
"cache_settings_album_thumbnails": "Miniaturas de la página de la biblioteca ({} archivos)",
"cache_settings_clear_cache_button": "Borrar caché",
"cache_settings_clear_cache_button_title": "Borra la caché de la aplicación. Esto afectará significativamente el rendimiento de la aplicación hasta que se reconstruya la caché.",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Uso de caché",
"cache_settings_subtitle": "Controla el comportamiento del almacenamiento en caché de la aplicación móvil Immich",
"cache_settings_thumbnail_size": "Tamaño de la caché de miniaturas ({} archivos)",
+ "cache_settings_tile_subtitle": "Controla el comportamiento del almacenamiento local",
+ "cache_settings_tile_title": "Almacenamiento local",
"cache_settings_title": "Configuración de la caché",
"change_password_form_confirm_password": "Confirmar Contraseña",
- "change_password_form_description": "Hola {firstName} {lastName},\n\nEsta es la primera vez que inicias sesión en el sistema o se ha solicitado cambiar tu contraseña. Por favor, introduce la nueva contraseña a continuación.",
+ "change_password_form_description": "Hola {name},\n\nEsta es la primera vez que inicias sesión en el sistema o se ha solicitado cambiar tu contraseña. Por favor, introduce la nueva contraseña a continuación.",
"change_password_form_new_password": "Nueva Contraseña",
"change_password_form_password_mismatch": "Las contraseñas no coinciden",
"change_password_form_reenter_new_password": "Vuelve a ingresar la nueva contraseña",
@@ -130,10 +136,10 @@
"control_bottom_app_bar_delete": "Eliminar",
"control_bottom_app_bar_favorite": "Favorito",
"control_bottom_app_bar_share": "Compartir",
- "control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_share_to": "Enviar",
+ "control_bottom_app_bar_stack": "Apilar",
"control_bottom_app_bar_unarchive": "Desarchivar",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Subir",
"create_album_page_untitled": "Sin título",
"create_shared_album_page_create": "Crear",
"create_shared_album_page_share": "Compartir",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "Cancelar",
"delete_dialog_ok": "Eliminar",
"delete_dialog_title": "Eliminar permanentemente",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Estás seguro que quieres eliminar este enlace compartido",
+ "delete_shared_link_dialog_title": "Eliminar enlace compartido",
"description_input_hint_text": "Agregar descripción...",
"description_input_submit_error": "Error al actualizar la descripción, verifica el registro para obtener más detalles",
"exif_bottom_sheet_description": "Agregar Descripción...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "{added} elementos agregados al álbum {album}.\n{failed} elementos ya existen en el álbum.",
"home_page_add_to_album_err_local": "Aún no se pueden agregar recursos locales a álbumes, omitiendo",
"home_page_add_to_album_success": "{added} elementos agregados al álbum {album}. ",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Los recursos locales no pueden ser archivados, omitiendo",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Construyendo la línea de tiempo",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Aún no se pueden archivar recursos locales, omitiendo",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Si esta es la primera vez que usas la app, por favor, asegúrate de elegir un álbum de respaldo para que la línea de tiempo pueda cargar fotos y videos en los álbumes.",
- "home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
+ "home_page_upload_err_limit": "Solo se pueden subir 30 elementos simultáneamente, omitiendo",
"image_viewer_page_state_provider_download_error": "Error de descarga",
"image_viewer_page_state_provider_download_success": "Descarga exitosa",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Error al compartir",
"library_page_albums": "Álbumes",
"library_page_archive": "Archivo",
"library_page_device_albums": "Álbumes en el dispositivo",
@@ -179,10 +190,10 @@
"library_page_new_album": "Nuevo álbum",
"library_page_sharing": "Compartiendo",
"library_page_sort_created": "Creado más recientemente",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Última modificación",
+ "library_page_sort_most_recent_photo": "Foto más reciente",
"library_page_sort_title": "Título del álbum",
- "login_disabled": "Login has been disabled",
+ "login_disabled": "El inicio de sesión ha sido desactivado",
"login_form_api_exception": "Excepción producida por API. Por favor, verifica el URL del servidor e inténtalo de nuevo.",
"login_form_button_text": "Iniciar sesión",
"login_form_email_hint": "tucorreo@correo.com",
@@ -196,7 +207,7 @@
"login_form_failed_get_oauth_server_config": "Error al iniciar sesión con OAuth, verifica la URL del servidor",
"login_form_failed_get_oauth_server_disable": "La función de OAuth no está disponible en este servidor",
"login_form_failed_login": "Error al iniciar sesión, comprueba la URL del servidor, el correo electrónico y la contraseña",
- "login_form_handshake_exception": "There was an Handshake Exception with the server. Enable self-signed certificate support in the settings if you are using a self-signed certificate.",
+ "login_form_handshake_exception": "Hubo un error de verificación del certificado del servidor. Activa el soporte para certificados autofirmados en las preferencias si estás usando un certificado autofirmado",
"login_form_label_email": "Correo electrónico",
"login_form_label_password": "Contraseña",
"login_form_next_button": "Siguiente",
@@ -204,24 +215,24 @@
"login_form_save_login": "Permanecer conectado",
"login_form_server_empty": "Agrega la URL del servidor.",
"login_form_server_error": "No se pudo conectar al servidor.",
- "login_password_changed_error": "There was an error updating your password",
- "login_password_changed_success": "Password updated successfully",
- "map_cannot_get_user_location": "Cannot get user's location",
- "map_location_dialog_cancel": "Cancel",
- "map_location_dialog_yes": "Yes",
- "map_location_service_disabled_content": "Location service needs to be enabled to display assets from your current location. Do you want to enable it now?",
- "map_location_service_disabled_title": "Location Service disabled",
- "map_no_assets_in_bounds": "No photos in this area",
- "map_no_location_permission_content": "Location permission is needed to display assets from your current location. Do you want to allow it now?",
- "map_no_location_permission_title": "Location Permission denied",
- "map_settings_dark_mode": "Dark mode",
- "map_settings_dialog_cancel": "Cancel",
- "map_settings_dialog_save": "Save",
- "map_settings_dialog_title": "Map Settings",
- "map_settings_include_show_archived": "Include Archived",
- "map_settings_only_relative_range": "Date range",
- "map_settings_only_show_favorites": "Show Favorite Only",
- "map_zoom_to_see_photos": "Zoom out to see photos",
+ "login_password_changed_error": "Hubo un error actualizando la contraseña",
+ "login_password_changed_success": "Contraseña cambiado con éxito",
+ "map_cannot_get_user_location": "No se pudo obtener la posición del usuario",
+ "map_location_dialog_cancel": "Cancelar",
+ "map_location_dialog_yes": "Sí",
+ "map_location_service_disabled_content": "Los servicios de localización deben estar activados para mostrar elementos de tu ubicación actual. Deseas activarlos ahora?",
+ "map_location_service_disabled_title": "Servicios de localización desactivados",
+ "map_no_assets_in_bounds": "No hay fotos en esta zona",
+ "map_no_location_permission_content": "Se necesitan permisos de ubicación para mostrar elementos de tu ubicación actual. Deseas activarlos ahora?",
+ "map_no_location_permission_title": "Permisos de ubicación denegados",
+ "map_settings_dark_mode": "Modo oscuro",
+ "map_settings_dialog_cancel": "Cancelar",
+ "map_settings_dialog_save": "Guardar",
+ "map_settings_dialog_title": "Ajustes mapa",
+ "map_settings_include_show_archived": "Incluir archivados",
+ "map_settings_only_relative_range": "Rango de fechas",
+ "map_settings_only_show_favorites": "Mostrar solo favoritas",
+ "map_zoom_to_see_photos": "Alejar para ver fotos",
"monthly_title_text_date_format": "MMMM y",
"motion_photos_page_title": "Foto en Movimiento",
"notification_permission_dialog_cancel": "Cancelar",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} ya no podrá acceder a tus fotos",
"partner_page_stop_sharing_title": "¿Dejar de compartir tus fotos?",
"partner_page_title": "Compañero",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continuar de todos modos",
"permission_onboarding_get_started": "Empezar",
"permission_onboarding_go_to_settings": "Ir a configuración",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich requiere permiso para ver tus fotos y videos.",
"profile_drawer_app_logs": "Registros",
"profile_drawer_client_server_up_to_date": "El cliente y el servidor están actualizados",
+ "profile_drawer_documentation": "Documentación",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Configuración",
"profile_drawer_sign_out": "Cerrar sesión",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Papelera",
"recently_added_page_title": "Recién Agregadas",
"search_bar_hint": "Busca tus fotos",
"search_page_categories": "Categorías",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Error al crear álbum",
"select_user_for_sharing_page_share_suggestions": "Sugerencias",
"server_info_box_app_version": "Versión de la Aplicación",
+ "server_info_box_server_url": "URL del servidor",
"server_info_box_server_version": "Versión del Servidor",
"setting_image_viewer_help": "El visor de detalles carga primero la miniatura pequeña, luego carga la vista previa de tamaño mediano (si está habilitada), finalmente carga la original (si está habilitada).",
"setting_image_viewer_original_subtitle": "Activar para cargar la imagen en resolución original (¡muy grande!). Deshabilitar para reducir el consumo de datos (de red y caché).",
@@ -300,35 +315,44 @@
"share_add_photos": "Agregar fotos",
"share_add_title": "Agregar un título",
"share_create_album": "Crear álbum",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Preparando...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Enlaces compartidos",
+ "shared_link_create_app_bar_title": "Crear enlace compartido",
+ "shared_link_create_info": "Cualquier persona con el enlace puede ver las fotos seleccionadas",
+ "shared_link_create_submit_button": "Crear enlace",
+ "shared_link_edit_allow_download": "Permitir descargar a usuarios públicos",
+ "shared_link_edit_allow_upload": "Permitir subir a usuarios públicos",
+ "shared_link_edit_app_bar_title": "Editar enlace",
+ "shared_link_edit_change_expiry": "Cambiar fecha de caducidad",
+ "shared_link_edit_description": "Descripción",
+ "shared_link_edit_description_hint": "Introduce la descripción del enlace",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Contraseña",
+ "shared_link_edit_password_hint": "Introduce la contraseña del enlace",
+ "shared_link_edit_show_meta": "Mostrar metadatos",
+ "shared_link_edit_submit_button": "Actualizar enlace",
+ "shared_link_empty": "No tienes enlaces compartidos",
+ "shared_link_manage_links": "Administrar enlaces compartidos",
+ "share_done": "Hecho",
"share_invite": "Invitar al álbum",
"sharing_page_album": "Álbumes compartidos",
"sharing_page_description": "Crea álbumes compartidos para compartir fotos y videos con personas de tu red.",
"sharing_page_empty_list": "LISTA VACIA",
"sharing_silver_appbar_create_shared_album": "Crear álbum compartido",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Enlaces compartidos",
"sharing_silver_appbar_share_partner": "Compartir con compañero",
"tab_controller_nav_library": "Biblioteca",
"tab_controller_nav_photos": "Fotos",
"tab_controller_nav_search": "Buscar",
"tab_controller_nav_sharing": "Compartiendo",
"theme_setting_asset_list_storage_indicator_title": "Mostrar indicador de almacenamiento en las miniaturas de los archivos",
- "theme_setting_asset_list_tiles_per_row_title": "Número de activos por fila ({})",
+ "theme_setting_asset_list_tiles_per_row_title": "Número de elementos por fila ({})",
"theme_setting_dark_mode_switch": "Modo oscuro",
"theme_setting_image_viewer_quality_subtitle": "Ajustar la calidad del visor de detalles de imágenes",
"theme_setting_image_viewer_quality_title": "Calidad del visor de imágenes",
@@ -337,30 +361,30 @@
"theme_setting_theme_title": "Tema",
"theme_setting_three_stage_loading_subtitle": "La carga en tres etapas puede aumentar el rendimiento de carga pero provoca un consumo de red significativamente mayor",
"theme_setting_three_stage_loading_title": "Activar carga en tres etapas",
- "translated_text_options": "Options",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
- "trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
- "upload_dialog_cancel": "Cancel",
- "upload_dialog_info": "Do you want to backup the selected Asset(s) to the server?",
- "upload_dialog_ok": "Upload",
- "upload_dialog_title": "Upload Asset",
+ "translated_text_options": "Opciones",
+ "trash_page_delete": "Eliminar",
+ "trash_page_delete_all": "Eliminar todos",
+ "trash_page_empty_trash_btn": "Vaciar papelera",
+ "trash_page_empty_trash_dialog_content": "Estás seguro que quieres eliminar los elementos? Estos elementos serán eliminados de Immich permanentemente",
+ "trash_page_empty_trash_dialog_ok": "Sí",
+ "trash_page_info": "Los archivos en la papelera serán eliminados automáticamente después de {} días",
+ "trash_page_no_assets": "No hay elementos en la papelera",
+ "trash_page_restore": "Restaurar",
+ "trash_page_restore_all": "Restaurar todos",
+ "trash_page_select_assets_btn": "Seleccionar elementos",
+ "trash_page_select_btn": "Seleccionar",
+ "trash_page_title": "Papelera ({})",
+ "upload_dialog_cancel": "Cancelar",
+ "upload_dialog_info": "Quieres hacer una copia de seguridad al servidor de los elementos seleccionados?",
+ "upload_dialog_ok": "Subir",
+ "upload_dialog_title": "Subir elementos",
"version_announcement_overlay_ack": "Aceptar",
"version_announcement_overlay_release_notes": "notas de la versión",
"version_announcement_overlay_text_1": "Hola, amigo, hay una nueva versión de",
"version_announcement_overlay_text_2": "por favor, tómate tu tiempo para visitar las ",
"version_announcement_overlay_text_3": " y asegúrate de que la configuración de docker-compose y .env estén actualizadas para evitar cualquier error de configuración, especialmente si utilizas WatchTower o cualquier mecanismo que actualice automáticamente la aplicación del servidor.",
"version_announcement_overlay_title": "Nueva versión del servidor disponible \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Quitar de la pila",
+ "viewer_stack_use_as_main_asset": "Usar como elemento principal",
+ "viewer_unstack": "Desapilar"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/es-US.json b/mobile/assets/i18n/es-US.json
new file mode 100644
index 000000000..1694f5a1f
--- /dev/null
+++ b/mobile/assets/i18n/es-US.json
@@ -0,0 +1,464 @@
+{
+ "add_to_album_bottom_sheet_added": "Agregado a {album}",
+ "add_to_album_bottom_sheet_already_exists": "Ya se encuentra en {album}",
+ "advanced_settings_log_level_title": "Nivel de registro: {}",
+ "advanced_settings_prefer_remote_subtitle": "Algunos dispositivos tardan mucho en cargar las miniaturas de recursos encontrados en el dispositivo. Activa esta opción para cargar imágenes remotas en su lugar.",
+ "advanced_settings_prefer_remote_title": "Preferir imágenes remotas",
+ "advanced_settings_self_signed_ssl_subtitle": "Omite la verificación del certificado SSL para la URL del servidor. Requerido para certificados autofirmados.",
+ "advanced_settings_self_signed_ssl_title": "Permitir certificados SSL autofirmados",
+ "advanced_settings_tile_subtitle": "Configuraciones avanzadas de usuario",
+ "advanced_settings_tile_title": "Avanzado",
+ "advanced_settings_troubleshooting_subtitle": "Habilitar funciones adicionales para solución de problemas",
+ "advanced_settings_troubleshooting_title": "Solución de problemas",
+ "album_info_card_backup_album_excluded": "EXCLUIDOS",
+ "album_info_card_backup_album_included": "INCLUIDOS",
+ "album_thumbnail_card_item": "1 elemento",
+ "album_thumbnail_card_items": "{} elementos",
+ "album_thumbnail_card_shared": " · Compartido",
+ "album_thumbnail_owned": "Propio",
+ "album_thumbnail_shared_by": "Compartido por {}",
+ "album_viewer_appbar_share_delete": "Eliminar álbum",
+ "album_viewer_appbar_share_err_delete": "No se ha podido eliminar el álbum",
+ "album_viewer_appbar_share_err_leave": "No se ha podido abandonar el álbum",
+ "album_viewer_appbar_share_err_remove": "Hay problemas para remover los archivos del álbum",
+ "album_viewer_appbar_share_err_title": "Error al cambiar el título del álbum",
+ "album_viewer_appbar_share_leave": "Abandonar álbum",
+ "album_viewer_appbar_share_remove": "Remover del álbum",
+ "album_viewer_appbar_share_to": "Compartir con",
+ "album_viewer_page_share_add_users": "Agregar usuarios",
+ "all_people_page_title": "Personas",
+ "all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "¿Estás seguro de que quieres cerrar sesión?",
+ "app_bar_signout_dialog_ok": "Sí",
+ "app_bar_signout_dialog_title": "Cerrar sesión",
+ "archive_page_no_archived_assets": "No se encontraron recursos archivados",
+ "archive_page_title": "Archivo ({})",
+ "asset_list_layout_settings_dynamic_layout_title": "Diseño dinámico",
+ "asset_list_layout_settings_group_automatically": "Automático",
+ "asset_list_layout_settings_group_by": "Agrupar recursos por",
+ "asset_list_layout_settings_group_by_month": "Mes",
+ "asset_list_layout_settings_group_by_month_day": "Mes + día",
+ "asset_list_settings_subtitle": "Configuraciones del diseño de la cuadrícula de fotos",
+ "asset_list_settings_title": "Cuadrícula de fotos",
+ "backup_album_selection_page_albums_device": "Álbumes en el dispositivo ({})",
+ "backup_album_selection_page_albums_tap": "Pulsar para incluir, pulsar dos veces para excluir",
+ "backup_album_selection_page_assets_scatter": "Los archivos pueden dispersarse en varios álbumes. De este modo, los álbumes pueden ser incluidos o excluidos durante el proceso de copia de seguridad.",
+ "backup_album_selection_page_select_albums": "Seleccionar álbumes",
+ "backup_album_selection_page_selection_info": "Información de la selección",
+ "backup_album_selection_page_total_assets": "Total de archivos únicos",
+ "backup_all": "Todos",
+ "backup_background_service_backup_failed_message": "Error al copiar archivos. Reintentando…",
+ "backup_background_service_connection_failed_message": "Error al conectar con el servidor. Reintentando…",
+ "backup_background_service_current_upload_notification": "Subiendo {}",
+ "backup_background_service_default_notification": "Verificando si hay nuevos archivos…",
+ "backup_background_service_error_title": "Error de copia de seguridad",
+ "backup_background_service_in_progress_notification": "Creando copia de seguridad de tus archivos…",
+ "backup_background_service_upload_failure_notification": "Error al subir {}",
+ "backup_controller_page_albums": "Álbumes de respaldo",
+ "backup_controller_page_background_app_refresh_disabled_content": "Activa la actualización en segundo plano de la aplicación en Configuración > General > Actualización en segundo plano para usar la copia de seguridad en segundo plano.",
+ "backup_controller_page_background_app_refresh_disabled_title": "Actualización en segundo plano desactivada",
+ "backup_controller_page_background_app_refresh_enable_button_text": "Ir a configuración",
+ "backup_controller_page_background_battery_info_link": "Muéstrame cómo",
+ "backup_controller_page_background_battery_info_message": "Para obtener la mejor experiencia de copia de seguridad en segundo plano, desactiva cualquier optimización de batería que restrinja la actividad en segundo plano para Immich.\n\nDado que esto es específico en cada dispositivo, busca la información necesaria del fabricante de tu dispositivo.",
+ "backup_controller_page_background_battery_info_ok": "Ok",
+ "backup_controller_page_background_battery_info_title": "Optimizaciones de batería",
+ "backup_controller_page_background_charging": "Sólo mientras se carga",
+ "backup_controller_page_background_configure_error": "Error al configurar el servicio en segundo plano",
+ "backup_controller_page_background_delay": "Retraso en la copia de seguridad de nuevos recursos: {}",
+ "backup_controller_page_background_description": "Activa el servicio en segundo plano para copiar automáticamente cualquier nuevo recurso sin necesidad de abrir la aplicación",
+ "backup_controller_page_background_is_off": "La copia de seguridad en segundo plano automática está desactivada",
+ "backup_controller_page_background_is_on": "La copia de seguridad en segundo plano automática está activada",
+ "backup_controller_page_background_turn_off": "Desactivar el servicio en segundo plano",
+ "backup_controller_page_background_turn_on": "Activar el servicio en segundo plano",
+ "backup_controller_page_background_wifi": "Sólo en WiFi",
+ "backup_controller_page_backup": "Respaldo",
+ "backup_controller_page_backup_selected": "Seleccionado: ",
+ "backup_controller_page_backup_sub": "Fotos y videos respaldados",
+ "backup_controller_page_cancel": "Cancelar",
+ "backup_controller_page_created": "Creado el: {}",
+ "backup_controller_page_desc_backup": "Activa la copia de seguridad en primer plano para subir automáticamente nuevos recursos al servidor al abrir la aplicación.",
+ "backup_controller_page_excluded": "Excluído: ",
+ "backup_controller_page_failed": "Fallidos ({})",
+ "backup_controller_page_filename": "Nombre del archivo: {} [{}]",
+ "backup_controller_page_id": "ID: {}",
+ "backup_controller_page_info": "Información del respaldo",
+ "backup_controller_page_none_selected": "Ninguno seleccionado",
+ "backup_controller_page_remainder": "Restante",
+ "backup_controller_page_remainder_sub": "Fotos y videos restantes para hacer una copia de seguridad de la selección",
+ "backup_controller_page_select": "Seleccionar",
+ "backup_controller_page_server_storage": "Almacenamiento del servidor",
+ "backup_controller_page_start_backup": "Iniciar respaldo",
+ "backup_controller_page_status_off": "La copia de seguridad automática en primer plano está desactivada",
+ "backup_controller_page_status_on": "La copia de seguridad automática en primer plano está activada",
+ "backup_controller_page_storage_format": "{} de {} usado",
+ "backup_controller_page_to_backup": "Álbumes a respaldar",
+ "backup_controller_page_total": "Total",
+ "backup_controller_page_total_sub": "Todas las fotos y videos únicos de los álbumes seleccionados",
+ "backup_controller_page_turn_off": "Desactivar la copia de seguridad en primer plano",
+ "backup_controller_page_turn_on": "Activar la copia de seguridad en primer plano",
+ "backup_controller_page_uploading_file_info": "Subiendo información del archivo",
+ "backup_err_only_album": "No se puede eliminar el único álbum",
+ "backup_info_card_assets": "recursos",
+ "backup_manual_cancelled": "Cancelado",
+ "backup_manual_failed": "Fallido",
+ "backup_manual_in_progress": "Subida ya en progreso. Inténtalo después de un tiempo",
+ "backup_manual_success": "Exitoso",
+ "backup_manual_title": "Estado de subida",
+ "cache_settings_album_thumbnails": "Miniaturas de la página de la biblioteca ({} recursos)",
+ "cache_settings_clear_cache_button": "Borrar caché",
+ "cache_settings_clear_cache_button_title": "Borra la caché de la aplicación. Esto afectará significativamente el rendimiento de la aplicación hasta que se reconstruya la caché.",
+ "cache_settings_duplicated_assets_clear_button": "BORRAR",
+ "cache_settings_duplicated_assets_subtitle": "Fotos y videos que son ignorados por la aplicación",
+ "cache_settings_duplicated_assets_title": "Recursos duplicados ({})",
+ "cache_settings_image_cache_size": "Tamaño de la caché de imágenes ({} recursos)",
+ "cache_settings_statistics_album": "Miniaturas de la biblioteca",
+ "cache_settings_statistics_assets": "{} recursos ({})",
+ "cache_settings_statistics_full": "Imágenes completas",
+ "cache_settings_statistics_shared": "Miniaturas de álbumes compartidos",
+ "cache_settings_statistics_thumbnail": "Miniaturas",
+ "cache_settings_statistics_title": "Uso de caché",
+ "cache_settings_subtitle": "Controla el comportamiento de la caché de la aplicación móvil Immich",
+ "cache_settings_thumbnail_size": "Tamaño de la caché de miniaturas ({} recursos)",
+ "cache_settings_tile_subtitle": "Controla el comportamiento del almacenamiento local",
+ "cache_settings_tile_title": "Almacenamiento local",
+ "cache_settings_title": "Configuración de la caché",
+ "change_password_form_confirm_password": "Confirmar Contraseña",
+ "change_password_form_description": "Hola {name},\n\nÉsta es la primera vez que inicias sesión en el sistema o se ha solicitado cambiar tu contraseña. Por favor, introduce la nueva contraseña a continuación.",
+ "change_password_form_new_password": "Nueva Contraseña",
+ "change_password_form_password_mismatch": "Las contraseñas no coinciden",
+ "change_password_form_reenter_new_password": "Vuelve a ingresar la nueva contraseña",
+ "common_add_to_album": "Agregar al álbum",
+ "common_change_password": "Cambiar Contraseña",
+ "common_create_new_album": "Crear nuevo álbum",
+ "common_server_error": "Por favor, verifica tu conexión de red, asegúrate de que el servidor esté accesible y las versiones de la aplicación y del servidor sean compatibles.",
+ "common_shared": "Compartido",
+ "control_bottom_app_bar_add_to_album": "Agregar al álbum",
+ "control_bottom_app_bar_album_info": "{} elementos",
+ "control_bottom_app_bar_album_info_shared": "{} elementos · Compartido",
+ "control_bottom_app_bar_archive": "Archivar",
+ "control_bottom_app_bar_create_new_album": "Crear nuevo álbum",
+ "control_bottom_app_bar_delete": "Eliminar",
+ "control_bottom_app_bar_favorite": "Favorito",
+ "control_bottom_app_bar_share": "Compartir",
+ "control_bottom_app_bar_share_to": "Compartir con",
+ "control_bottom_app_bar_stack": "Apilar",
+ "control_bottom_app_bar_unarchive": "Desarchivar",
+ "control_bottom_app_bar_upload": "Subir",
+ "create_album_page_untitled": "Sin título",
+ "create_shared_album_page_create": "Crear",
+ "create_shared_album_page_share": "Compartir",
+ "create_shared_album_page_share_add_assets": "AGREGAR RECURSOS",
+ "create_shared_album_page_share_select_photos": "Seleccionar fotos",
+ "curated_location_page_title": "Lugares",
+ "curated_object_page_title": "Objetos",
+ "daily_title_text_date": "E, dd MMM",
+ "daily_title_text_date_year": "E, dd de MMM, yyyy",
+ "date_format": "E d, LLL y • h:mm a",
+ "delete_dialog_alert": "Estos elementos se eliminarán permanentemente de Immich y de tu dispositivo",
+ "delete_dialog_cancel": "Cancelar",
+ "delete_dialog_ok": "Eliminar",
+ "delete_dialog_title": "Eliminar permanentemente",
+ "delete_shared_link_dialog_content": "¿Estás seguro de que quieres eliminar este enlace compartido?",
+ "delete_shared_link_dialog_title": "Eliminar enlace compartido",
+ "description_input_hint_text": "Agregar descripción...",
+ "description_input_submit_error": "Error al actualizar la descripción, verifica el registro para obtener más detalles",
+ "exif_bottom_sheet_description": "Agregar Descripción...",
+ "exif_bottom_sheet_details": "DETALLES",
+ "exif_bottom_sheet_location": "UBICACIÓN",
+ "experimental_settings_new_asset_list_subtitle": "Trabajo en progreso",
+ "experimental_settings_new_asset_list_title": "Habilitar cuadrícula fotográfica experimental",
+ "experimental_settings_subtitle": "¡Úsalo bajo tu propio riesgo!",
+ "experimental_settings_title": "Experimental",
+ "favorites_page_no_favorites": "No se encontraron recursos marcados como favoritos",
+ "favorites_page_title": "Favoritos",
+ "home_page_add_to_album_conflicts": "{added} recursos agregados al álbum {album}.\n{failed} recursos ya existen en el álbum.",
+ "home_page_add_to_album_err_local": "Aún no se pueden agregar recursos locales a álbumes, omitiendo",
+ "home_page_add_to_album_success": "{added} recursos agregados al álbum {album}.",
+ "home_page_archive_err_local": "Aún no se pueden archivar recursos locales, omitiendo",
+ "home_page_building_timeline": "Construyendo la línea de tiempo",
+ "home_page_favorite_err_local": "Aún no se pueden marcar recursos locales como favoritos, omitiendo",
+ "home_page_first_time_notice": "Si ésta es la primera vez que usas la app, por favor, asegúrate de elegir un álbum de respaldo para que la línea de tiempo pueda cargar fotos y videos en los álbumes.",
+ "home_page_share_err_local": "No se pueden compartir recursos locales a través de enlaces, omitiendo",
+ "home_page_upload_err_limit": "Sólo se pueden subir un máximo de 30 recursos a la vez, omitiendo",
+ "home_page_favorite_err_partner": "Aún no se pueden marcar recursos de compañeros como favoritos, omitiendo",
+ "home_page_album_err_partner": "Aún no se pueden agregar recursos de compañeros a un álbum, omitiendo",
+ "home_page_archive_err_partner": "Aún no se pueden archivar recursos de compañeros, omitiendo",
+ "home_page_delete_err_partner": "Aún no se pueden eliminar recursos de compañeros, omitiendo",
+ "image_viewer_page_state_provider_download_error": "Error de descarga",
+ "image_viewer_page_state_provider_download_success": "Descarga exitosa",
+ "image_viewer_page_state_provider_share_error": "Error al compartir",
+ "library_page_albums": "Álbumes",
+ "library_page_archive": "Archivo",
+ "library_page_device_albums": "Álbumes en el dispositivo",
+ "library_page_favorites": "Favoritos",
+ "library_page_new_album": "Nuevo álbum",
+ "library_page_sharing": "Compartiendo",
+ "library_page_sort_created": "Creado más recientemente",
+ "library_page_sort_last_modified": "Modificado más recientemente",
+ "library_page_sort_most_recent_photo": "Foto más reciente",
+ "library_page_sort_title": "Título del álbum",
+ "login_disabled": "El inicio de sesión ha sido deshabilitado",
+ "login_form_api_exception": "Excepción de API. Por favor, verifica la URL del servidor e inténtalo de nuevo.",
+ "login_form_back_button_text": "Volver",
+ "login_form_button_text": "Iniciar sesión",
+ "login_form_email_hint": "tucorreo@correo.com",
+ "login_form_endpoint_hint": "http://ip-de-tu-servidor:puerto/api",
+ "login_form_endpoint_url": "URL del servidor",
+ "login_form_err_http": "Por favor, especifique http:// o https://",
+ "login_form_err_invalid_email": "Correo electrónico inválido",
+ "login_form_err_invalid_url": "URL inválido",
+ "login_form_err_leading_whitespace": "Espacio en blanco inicial",
+ "login_form_err_trailing_whitespace": "Espacio en blanco al final",
+ "login_form_failed_get_oauth_server_config": "Error al iniciar sesión con OAuth, verifica la URL del servidor",
+ "login_form_failed_get_oauth_server_disable": "La función de OAuth no está disponible en este servidor",
+ "login_form_failed_login": "Error al iniciar sesión, comprueba la URL del servidor, el correo electrónico y la contraseña",
+ "login_form_handshake_exception": "Hubo una excepción de handshake con el servidor. Habilita el soporte de certificado autofirmado en la configuración si estás usando un certificado autofirmado.",
+ "login_form_label_email": "Correo electrónico",
+ "login_form_label_password": "Contraseña",
+ "login_form_next_button": "Siguiente",
+ "login_form_password_hint": "contraseña",
+ "login_form_save_login": "Permanecer conectado",
+ "login_form_server_empty": "Introduce la URL del servidor.",
+ "login_form_server_error": "No se pudo conectar al servidor.",
+ "login_password_changed_error": "Hubo un error al actualizar tu contraseña",
+ "login_password_changed_success": "Contraseña actualizada exitosamente",
+ "map_assets_in_bounds": {
+ "one": "{} foto",
+ "other": "{} fotos"
+ },
+ "map_cannot_get_user_location": "No se puede obtener la ubicación del usuario",
+ "map_location_dialog_cancel": "Cancelar",
+ "map_location_dialog_yes": "Sí",
+ "map_location_service_disabled_content": "El servicio de ubicación debe estar habilitado para mostrar recursos desde tu ubicación actual. ¿Quieres habilitarlo ahora?",
+ "map_location_service_disabled_title": "Servicio de ubicación deshabilitado",
+ "map_no_assets_in_bounds": "No hay fotos en esta área",
+ "map_no_location_permission_content": "Se necesita permiso de ubicación para mostrar recursos desde tu ubicación actual. ¿Quieres permitirlo ahora?",
+ "map_no_location_permission_title": "Permiso de ubicación denegado",
+ "map_settings_dark_mode": "Modo oscuro",
+ "map_settings_date_range_option_all": "Todo",
+ "map_settings_date_range_option_days": {
+ "one": "Últimas 24 horas",
+ "other": "Últimos {} días"
+ },
+ "map_settings_date_range_option_years": {
+ "one": "Último año",
+ "other": "Últimos {} años"
+ },
+ "map_settings_dialog_cancel": "Cancelar",
+ "map_settings_dialog_save": "Guardar",
+ "map_settings_dialog_title": "Configuración del mapa",
+ "map_settings_include_show_archived": "Incluir archivados",
+ "map_settings_only_relative_range": "Rango de fechas",
+ "map_settings_only_show_favorites": "Mostrar sólo favoritos",
+ "map_zoom_to_see_photos": "Aleja el mapa para ver las fotos",
+ "monthly_title_text_date_format": "MMMM y",
+ "motion_photos_page_title": "Fotos en movimiento",
+ "notification_permission_dialog_cancel": "Cancelar",
+ "notification_permission_dialog_content": "Para activar las notificaciones, ve a Configuración y selecciona permitir.",
+ "notification_permission_dialog_settings": "Configuración",
+ "notification_permission_list_tile_content": "Concede permiso para activar las notificaciones.",
+ "notification_permission_list_tile_enable_button": "Activar notificaciones",
+ "notification_permission_list_tile_title": "Permisos de notificación",
+ "partner_page_add_partner": "Agregar compañero",
+ "partner_page_empty_message": "Tus fotos aún no se han compartido con ningún compañero.",
+ "partner_page_no_more_users": "No hay más usuarios para agregar",
+ "partner_page_partner_add_failed": "Error al agregar compañero",
+ "partner_page_select_partner": "Seleccionar compañero",
+ "partner_page_shared_to_title": "Compartido con",
+ "partner_page_stop_sharing_content": "{} ya no podrá acceder a tus fotos",
+ "partner_page_stop_sharing_title": "¿Dejar de compartir tus fotos?",
+ "partner_page_title": "Compañero",
+ "permission_onboarding_back": "Volver",
+ "permission_onboarding_continue_anyway": "Continuar de todos modos",
+ "permission_onboarding_get_started": "Empezar",
+ "permission_onboarding_go_to_settings": "Ir a configuración",
+ "permission_onboarding_grant_permission": "Conceder permiso",
+ "permission_onboarding_log_out": "Cerrar sesión",
+ "permission_onboarding_permission_denied": "Permiso denegado. Para usar Immich, concede permisos de fotos y videos en Configuración.",
+ "permission_onboarding_permission_granted": "¡Permiso concedido! Todo listo.",
+ "permission_onboarding_permission_limited": "Permiso limitado. Para permitir que Immich haga copia de seguridad y gestione toda tu colección de galería, concede permisos de fotos y videos en Configuración.",
+ "permission_onboarding_request": "Immich requiere permiso para ver tus fotos y videos.",
+ "profile_drawer_app_logs": "Registros",
+ "profile_drawer_client_out_of_date_major": "La aplicación móvil está desactualizada. Actualiza a la última versión mayor.",
+ "profile_drawer_client_out_of_date_minor": "La aplicación móvil está desactualizada. Actualiza a la última versión menor.",
+ "profile_drawer_client_server_up_to_date": "El cliente y el servidor están actualizados",
+ "profile_drawer_documentation": "Documentación",
+ "profile_drawer_github": "GitHub",
+ "profile_drawer_server_out_of_date_major": "El servidor está desactualizado. Actualiza a la última versión mayor.",
+ "profile_drawer_server_out_of_date_minor": "El servidor está desactualizado. Actualiza a la última versión menor.",
+ "profile_drawer_settings": "Configuración",
+ "profile_drawer_sign_out": "Cerrar sesión",
+ "profile_drawer_trash": "Papelera",
+ "recently_added_page_title": "Recién Agregados",
+ "search_bar_hint": "Busca tus fotos",
+ "search_page_categories": "Categorías",
+ "search_page_favorites": "Favoritos",
+ "search_page_motion_photos": "Fotos en movimiento",
+ "search_page_no_objects": "No hay información de objetos disponible",
+ "search_page_no_places": "No hay información de lugares disponible",
+ "search_page_people": "Personas",
+ "search_page_person_add_name_dialog_cancel": "Cancelar",
+ "search_page_person_add_name_dialog_hint": "Nombre",
+ "search_page_person_add_name_dialog_save": "Guardar",
+ "search_page_person_add_name_dialog_title": "Agregar nombre",
+ "search_page_person_add_name_subtitle": "Encuéntralos rápidamente por nombre",
+ "search_page_person_add_name_title": "Agregar un nombre",
+ "search_page_person_edit_name": "Editar nombre",
+ "search_page_places": "Lugares",
+ "search_page_recently_added": "Recién agregados",
+ "search_page_screenshots": "Capturas de pantalla",
+ "search_page_selfies": "Selfies",
+ "search_page_things": "Cosas",
+ "search_page_videos": "Videos",
+ "search_page_view_all_button": "Ver todo",
+ "search_page_your_activity": "Tu actividad",
+ "search_page_your_map": "Tu mapa",
+ "search_result_page_new_search_hint": "Nueva búsqueda",
+ "search_suggestion_list_smart_search_hint_1": "La búsqueda inteligente está habilitada por defecto, para buscar metadatos utiliza la sintaxis ",
+ "search_suggestion_list_smart_search_hint_2": "m:tu-término-de-búsqueda",
+ "select_additional_user_for_sharing_page_suggestions": "Sugerencias",
+ "select_user_for_sharing_page_err_album": "Error al crear álbum",
+ "select_user_for_sharing_page_share_suggestions": "Sugerencias",
+ "server_info_box_app_version": "Versión de la Aplicación",
+ "server_info_box_server_url": "URL del Servidor",
+ "server_info_box_server_version": "Versión del Servidor",
+ "server_info_box_latest_release": "Última versión",
+ "setting_image_viewer_help": "El visor de detalles carga primero la miniatura pequeña, luego carga la vista previa de tamaño mediano (si está habilitada), finalmente carga la original (si está habilitada).",
+ "setting_image_viewer_original_subtitle": "Activar para cargar la imagen en resolución original (¡muy grande!). Deshabilitar para reducir el consumo de datos (de red y caché).",
+ "setting_image_viewer_original_title": "Cargar imagen original",
+ "setting_image_viewer_preview_subtitle": "Activar para cargar una imagen de resolución media. Deshabilitar para cargar directamente la imagen original o usar una miniatura.",
+ "setting_image_viewer_preview_title": "Cargar imagen de previsualización",
+ "setting_notifications_notify_failures_grace_period": "Notificar fallos de copia de seguridad en segundo plano: {}",
+ "setting_notifications_notify_hours": "{} horas",
+ "setting_notifications_notify_immediately": "inmediatamente",
+ "setting_notifications_notify_minutes": "{} minutos",
+ "setting_notifications_notify_never": "nunca",
+ "setting_notifications_notify_seconds": "{} segundos",
+ "setting_notifications_single_progress_subtitle": "Información detallada del progreso de subida de cada recurso",
+ "setting_notifications_single_progress_title": "Mostrar progreso detallado de copia de seguridad en segundo plano",
+ "setting_notifications_subtitle": "Ajusta tus preferencias de notificación",
+ "setting_notifications_title": "Notificaciones",
+ "setting_notifications_total_progress_subtitle": "Progreso general de subida (recursos completados/totales)",
+ "setting_notifications_total_progress_title": "Mostrar progreso total de copia de seguridad en segundo plano",
+ "setting_pages_app_bar_settings": "Configuración",
+ "settings_require_restart": "Por favor, reinicia Immich para aplicar este ajuste",
+ "share_add": "Agregar",
+ "share_add_photos": "Agregar fotos",
+ "share_add_title": "Agregar un título",
+ "share_create_album": "Crear álbum",
+ "shared_album_activities_input_disable": "Los comentarios están deshabilitados",
+ "shared_album_activities_input_hint": "Di algo",
+ "shared_album_activity_remove_content": "¿Quieres eliminar esta actividad?",
+ "shared_album_activity_remove_title": "Eliminar actividad",
+ "shared_album_activity_setting_subtitle": "Permitir que otros respondan",
+ "shared_album_activity_setting_title": "Comentarios y me gusta",
+ "shared_album_section_people_action_error": "Error al dejar/remover del álbum",
+ "shared_album_section_people_action_leave": "Dejar álbum",
+ "shared_album_section_people_action_remove_user": "Remover usuario del álbum",
+ "shared_album_section_people_owner_label": "Dueño",
+ "shared_album_section_people_title": "PERSONAS",
+ "share_dialog_preparing": "Preparando...",
+ "shared_link_app_bar_title": "Enlaces compartidos",
+ "shared_link_clipboard_copied_massage": "Copiado al portapapeles",
+ "shared_link_clipboard_text": "Enlace: {}\nContraseña: {}",
+ "shared_link_create_app_bar_title": "Crear enlace para compartir",
+ "shared_link_create_error": "Error al crear enlace compartido",
+ "shared_link_create_info": "Permitir que cualquiera con el enlace vea la(s) foto(s) seleccionada(s)",
+ "shared_link_create_submit_button": "Crear enlace",
+ "shared_link_edit_allow_download": "Permitir que el usuario público pueda descargar",
+ "shared_link_edit_allow_upload": "Permitir que el usuario público pueda subir",
+ "shared_link_edit_app_bar_title": "Editar enlace",
+ "shared_link_edit_change_expiry": "Cambiar tiempo de expiración",
+ "shared_link_edit_description": "Descripción",
+ "shared_link_edit_description_hint": "Introduce la descripción del enlace",
+ "shared_link_edit_expire_after": "Expirar después de",
+ "shared_link_edit_expire_after_option_days": {
+ "one": "{} día",
+ "other": "{} días"
+ },
+ "shared_link_edit_expire_after_option_hours": {
+ "one": "{} hora",
+ "other": "{} horas"
+ },
+ "shared_link_edit_expire_after_option_minutes": {
+ "one": "{} minuto",
+ "other": "{} minutos"
+ },
+ "shared_link_edit_expire_after_option_never": "Nunca",
+ "shared_link_edit_password": "Contraseña",
+ "shared_link_edit_password_hint": "Introduce la contraseña del enlace",
+ "shared_link_edit_show_meta": "Mostrar metadatos",
+ "shared_link_edit_submit_button": "Actualizar enlace",
+ "shared_link_empty": "No tienes ningún enlace compartido",
+ "shared_link_error_server_url_fetch": "No se puede obtener la URL del servidor",
+ "shared_link_expired": "Expirado",
+ "shared_link_expires_days": {
+ "one": "Expira en {} día",
+ "other": "Expira en {} días"
+ },
+ "shared_link_expires_hours": {
+ "one": "Expira en {} hora",
+ "other": "Expira en {} horas"
+ },
+ "shared_link_expires_minutes": {
+ "one": "Expira en {} minuto",
+ "other": "Expira en {} minutos"
+ },
+ "shared_link_expires_seconds": {
+ "one": "Expira en {} segundo",
+ "other": "Expira en {} segundos"
+ },
+ "shared_link_expires_never": "Sin expiración",
+ "shared_link_info_chip_download": "Descargar",
+ "shared_link_info_chip_metadata": "EXIF",
+ "shared_link_info_chip_upload": "Subir",
+ "shared_link_manage_links": "Administrar enlaces compartidos",
+ "share_done": "Hecho",
+ "share_invite": "Invitar al álbum",
+ "sharing_page_album": "Álbumes compartidos",
+ "sharing_page_description": "Crea álbumes compartidos para compartir fotos y videos con personas de tu red.",
+ "sharing_page_empty_list": "LISTA VACÍA",
+ "sharing_silver_appbar_create_shared_album": "Crear álbum compartido",
+ "sharing_silver_appbar_shared_links": "Enlaces compartidos",
+ "sharing_silver_appbar_share_partner": "Compartir con compañero",
+ "tab_controller_nav_library": "Biblioteca",
+ "tab_controller_nav_photos": "Fotos",
+ "tab_controller_nav_search": "Buscar",
+ "tab_controller_nav_sharing": "Compartidos",
+ "theme_setting_asset_list_storage_indicator_title": "Mostrar indicador de almacenamiento en las miniaturas de los recursos",
+ "theme_setting_asset_list_tiles_per_row_title": "Número de recursos por fila ({})",
+ "theme_setting_dark_mode_switch": "Modo oscuro",
+ "theme_setting_image_viewer_quality_subtitle": "Ajustar la calidad del visor de detalles de imágenes",
+ "theme_setting_image_viewer_quality_title": "Calidad del visor de imágenes",
+ "theme_setting_system_theme_switch": "Automático (seguir ajuste del sistema)",
+ "theme_setting_theme_subtitle": "Elige la configuración del tema de la aplicación",
+ "theme_setting_theme_title": "Tema",
+ "theme_setting_three_stage_loading_subtitle": "La carga en tres etapas puede aumentar el rendimiento de carga pero provoca un consumo de red significativamente mayor",
+ "theme_setting_three_stage_loading_title": "Activar carga en tres etapas",
+ "translated_text_options": "Opciones",
+ "trash_page_delete": "Eliminar",
+ "trash_page_delete_all": "Eliminar todos",
+ "trash_page_empty_trash_btn": "Vaciar papelera",
+ "trash_page_empty_trash_dialog_content": "¿Quieres vaciar los recursos de la papelera? Estos elementos serán eliminados permanentemente de Immich",
+ "trash_page_empty_trash_dialog_ok": "Ok",
+ "trash_page_info": "Los elementos en la papelera serán borrados permanentemente luego de {} días",
+ "trash_page_no_assets": "No hay recursos en la papelera",
+ "trash_page_restore": "Restaurar",
+ "trash_page_restore_all": "Restaurar todos",
+ "trash_page_select_assets_btn": "Seleccionar recursos",
+ "trash_page_select_btn": "Seleccionar",
+ "trash_page_title": "Papelera ({})",
+ "upload_dialog_cancel": "Cancelar",
+ "upload_dialog_info": "¿Quieres respaldar los recursos seleccionados en el servidor?",
+ "upload_dialog_ok": "Subir",
+ "upload_dialog_title": "Subir recurso",
+ "version_announcement_overlay_ack": "Aceptar",
+ "version_announcement_overlay_release_notes": "notas de la versión",
+ "version_announcement_overlay_text_1": "Hola, amigo, hay una nueva versión de",
+ "version_announcement_overlay_text_2": "por favor, tómate tu tiempo para visitar las ",
+ "version_announcement_overlay_text_3": " y asegúrate de que la configuración de docker-compose y .env estén actualizadas para evitar cualquier error de configuración, especialmente si utilizas WatchTower o cualquier mecanismo que actualice automáticamente la aplicación del servidor.",
+ "version_announcement_overlay_title": "Nueva versión del servidor disponible \uD83C\uDF89",
+ "viewer_remove_from_stack": "Eliminar de la pila",
+ "viewer_stack_use_as_main_asset": "Utilizar como recurso principal",
+ "viewer_unstack": "Desapilar"
+}
diff --git a/mobile/assets/i18n/fi-FI.json b/mobile/assets/i18n/fi-FI.json
index 1aa8a926a..d746f99c1 100644
--- a/mobile/assets/i18n/fi-FI.json
+++ b/mobile/assets/i18n/fi-FI.json
@@ -3,8 +3,8 @@
"add_to_album_bottom_sheet_already_exists": "Kohde on jo albumissa {album}",
"advanced_settings_prefer_remote_subtitle": "Jotkut laitteet ovat erittäin hitaita lataamaan esikatselukuvia laitteen kohteista. Aktivoi tämä asetus käyttääksesi etäkuvia.",
"advanced_settings_prefer_remote_title": "Suosi etäkuvia",
- "advanced_settings_self_signed_ssl_subtitle": "Skips SSL certificate verification for the server endpoint. Required for self-signed certificates.",
- "advanced_settings_self_signed_ssl_title": "Allow self-signed SSL certificates",
+ "advanced_settings_self_signed_ssl_subtitle": "Ohita SSL sertifikaattivarmennus palvelimen päätepisteellä. Vaaditaan self-signed -sertifikaateissa.",
+ "advanced_settings_self_signed_ssl_title": "Salli self-signed SSL -sertifikaatit",
"advanced_settings_tile_subtitle": "Edistyneen käyttäjän asetukset",
"advanced_settings_tile_title": "Edistyneet",
"advanced_settings_troubleshooting_subtitle": "Kytke vianetsinnän lisäominaisuudet päälle",
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Albumin nimen muuttaminen epäonnistui",
"album_viewer_appbar_share_leave": "Poistu albumista",
"album_viewer_appbar_share_remove": "Poista albumista",
+ "album_viewer_appbar_share_to": "Jaa",
"album_viewer_page_share_add_users": "Lisää käyttäjiä",
"all_people_page_title": "Ihmiset",
"all_videos_page_title": "Videot",
+ "app_bar_signout_dialog_content": "Haluatko varmasti kirjautua ulos?",
+ "app_bar_signout_dialog_ok": "Kyllä",
+ "app_bar_signout_dialog_title": "Kirjaudu ulos",
"archive_page_no_archived_assets": "Arkistoituja kohteita ei löytynyt",
"archive_page_title": "Arkisto ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynaaminen asetelma",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Välimuistin käyttö",
"cache_settings_subtitle": "Hallitse Immich-mobiilisovelluksen välimuistin käyttöä",
"cache_settings_thumbnail_size": "Esikatselukuvien välimuistin koko ({} kohdetta)",
+ "cache_settings_tile_subtitle": "Hallitse paikallista tallenustilaa",
+ "cache_settings_tile_title": "Paikallinen tallennustila",
"cache_settings_title": "Välimuistin asetukset",
"change_password_form_confirm_password": "Vahvista salasana",
- "change_password_form_description": "Hei {firstName} {lastName},\n\nTämä on joko ensimmäinen kirjautumisesi järjestelmään tai salasanan vaihtaminen vaihtaminen on pakotettu. Ole hyvä ja syötä uusi salasana alle.",
+ "change_password_form_description": "Hei {name},\n\nTämä on joko ensimmäinen kirjautumisesi järjestelmään tai salasanan vaihtaminen vaihtaminen on pakotettu. Ole hyvä ja syötä uusi salasana alle.",
"change_password_form_new_password": "Uusi salasana",
"change_password_form_password_mismatch": "Salasanat eivät täsmää",
"change_password_form_reenter_new_password": "Uusi salasana uudelleen",
@@ -130,10 +136,10 @@
"control_bottom_app_bar_delete": "Poista",
"control_bottom_app_bar_favorite": "Suosikki",
"control_bottom_app_bar_share": "Jaa",
- "control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_share_to": "Jaa",
+ "control_bottom_app_bar_stack": "Pinoa",
"control_bottom_app_bar_unarchive": "Palauta arkistosta",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Siirrä palvelimelle",
"create_album_page_untitled": "Nimetön",
"create_shared_album_page_create": "Luo",
"create_shared_album_page_share": "Jaa",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "Peruuta",
"delete_dialog_ok": "Poista",
"delete_dialog_title": "Poista pysyvästi",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Oletko varma, että haluat poistaa jaetun linkin?",
+ "delete_shared_link_dialog_title": "Poista jaettu linkki",
"description_input_hint_text": "Lisää kuvaus...",
"description_input_submit_error": "Virhe kuvauksen päivittämisessä, tarkista lisätiedot lokista",
"exif_bottom_sheet_description": "Lisää kuvaus…",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "Lisätty {added} kohdetta albumiin {album}. {failed} kohdetta on jo albumissa.",
"home_page_add_to_album_err_local": "Paikallisten kohteiden lisääminen albumeihin ei ole mahdollista, ohitetaan",
"home_page_add_to_album_success": "Lisätty {added} kohdetta albumiin {album}.",
+ "home_page_album_err_partner": "Kumppanin kohteita ei voi vielä lisätä albumiin. Hypätään yli",
"home_page_archive_err_local": "Paikallisten kohteiden arkistointi ei ole mahdollista, ohitetaan",
+ "home_page_archive_err_partner": "Kumppanin kohteita ei voi arkistoida. Hypätään yli",
"home_page_building_timeline": "Rakennetaan aikajanaa",
+ "home_page_delete_err_partner": "Kumppanin kohteita ei voi poistaa.Hypätään yli",
"home_page_favorite_err_local": "Paikallisten kohteiden lisääminen suosikkeihin ei ole mahdollista, ohitetaan",
+ "home_page_favorite_err_partner": "Kumppanin kohteita ei voi vielä merkitä suosikiksi. Hypätään yli",
"home_page_first_time_notice": "Jos käytät sovellusta ensimmäistä kertaa, muista valita varmuuskopioitavat albumi(t), jotta aikajanalla voi olla kuvia ja videoita.",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Voit lähettää palvelimelle enintään 30 kohdetta kerrallaan, ohitetaan",
"image_viewer_page_state_provider_download_error": "Lataus epäonnistui",
"image_viewer_page_state_provider_download_success": "Lataus onnistui",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Jakovirhe",
"library_page_albums": "Albumit",
"library_page_archive": "Arkisto",
"library_page_device_albums": "Laitteen albumit",
@@ -179,8 +190,8 @@
"library_page_new_album": "Uusi albumi",
"library_page_sharing": "Jakaminen",
"library_page_sort_created": "Viimeisin luotu",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Viimeksi muokattu",
+ "library_page_sort_most_recent_photo": "Viimeisin kuva",
"library_page_sort_title": "Albumin otsikko",
"login_disabled": "Kirjautuminen on poistettu käytöstä",
"login_form_api_exception": "API-virhe. Tarkista palvelimen URL-osoite ja yritä uudelleen.",
@@ -196,7 +207,7 @@
"login_form_failed_get_oauth_server_config": "Virhe kirjauduttaessa OAuth:lla, tarkista palvelimen URL",
"login_form_failed_get_oauth_server_disable": "OAuth-ominaisuus ei ole käytössä tällä palvelimella",
"login_form_failed_login": "Virhe kirjautumisessa. Tarkista palvelimen URL, sähköpostiosoite ja salasana.",
- "login_form_handshake_exception": "There was an Handshake Exception with the server. Enable self-signed certificate support in the settings if you are using a self-signed certificate.",
+ "login_form_handshake_exception": "Tapahtui poikkeus kättelyssä palvelimen kanssa. Kytke päälle self-signed -sertifikaattituki asetuksista, mikäli käytät self-signed -sertifikaatteja.",
"login_form_label_email": "Sähköposti",
"login_form_label_password": "Salasana",
"login_form_next_button": "Seuraava",
@@ -204,24 +215,24 @@
"login_form_save_login": "Pysy kirjautuneena",
"login_form_server_empty": "Syötä palvelimen URL-osoite.",
"login_form_server_error": "Palvelimeen ei saatu yhteyttä.",
- "login_password_changed_error": "There was an error updating your password",
- "login_password_changed_success": "Password updated successfully",
- "map_cannot_get_user_location": "Cannot get user's location",
- "map_location_dialog_cancel": "Cancel",
- "map_location_dialog_yes": "Yes",
- "map_location_service_disabled_content": "Location service needs to be enabled to display assets from your current location. Do you want to enable it now?",
- "map_location_service_disabled_title": "Location Service disabled",
- "map_no_assets_in_bounds": "No photos in this area",
- "map_no_location_permission_content": "Location permission is needed to display assets from your current location. Do you want to allow it now?",
- "map_no_location_permission_title": "Location Permission denied",
- "map_settings_dark_mode": "Dark mode",
- "map_settings_dialog_cancel": "Cancel",
- "map_settings_dialog_save": "Save",
- "map_settings_dialog_title": "Map Settings",
- "map_settings_include_show_archived": "Include Archived",
- "map_settings_only_relative_range": "Date range",
- "map_settings_only_show_favorites": "Show Favorite Only",
- "map_zoom_to_see_photos": "Zoom out to see photos",
+ "login_password_changed_error": "Salasanan päivityksessä tapahtui virhe",
+ "login_password_changed_success": "Salasan päivitetty onnistuneesti",
+ "map_cannot_get_user_location": "Käyttäjän sijaintia ei voitu määrittää",
+ "map_location_dialog_cancel": "Peruuta",
+ "map_location_dialog_yes": "Kyllä",
+ "map_location_service_disabled_content": "Paikannuspalvelun pitää olla päälle kytkettynä, jotta nykyisen sijaintisi kohteita voidaan näyttää. Haluatko kytkeä sen päälle?",
+ "map_location_service_disabled_title": "Paikannuspalvelu pois päältä",
+ "map_no_assets_in_bounds": "Ei kuvia tällä alueella",
+ "map_no_location_permission_content": "Paikannuslupa tarvitaan, jotta nykyisen sijainnin kohteita voidaan näyttää. Haluatko sallia pääsyn sijaintiin?",
+ "map_no_location_permission_title": "Paikannuslupa estetty",
+ "map_settings_dark_mode": "Tumma tila",
+ "map_settings_dialog_cancel": "Peruuta",
+ "map_settings_dialog_save": "Tallenna",
+ "map_settings_dialog_title": "Kartta-asetukset",
+ "map_settings_include_show_archived": "Sisällytä arkistoidut",
+ "map_settings_only_relative_range": "Päivämäärän rajaus",
+ "map_settings_only_show_favorites": "Näytä vain suosikit",
+ "map_zoom_to_see_photos": "Tarkenna nähdäksesi kuvat",
"monthly_title_text_date_format": "MMMM y",
"motion_photos_page_title": "Liikekuvat",
"notification_permission_dialog_cancel": "Peruuta",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} ei voi enää käyttää kuviasi.",
"partner_page_stop_sharing_title": "Lopetetaanko kuvien jakaminen?",
"partner_page_title": "Kumppani",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Jatka silti",
"permission_onboarding_get_started": "Aloittaminen",
"permission_onboarding_go_to_settings": "Siirry asetuksiin",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich vaatii käyttöoikeuden kuvien ja videoiden käyttämiseen.",
"profile_drawer_app_logs": "Lokit",
"profile_drawer_client_server_up_to_date": "Asiakassovellus ja palvelin ovat ajan tasalla",
+ "profile_drawer_documentation": "Dokumentaatio",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Asetukset",
"profile_drawer_sign_out": "Kirjaudu ulos",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Roskakori",
"recently_added_page_title": "Viimeksi lisätyt",
"search_bar_hint": "Etsi kuvia",
"search_page_categories": "Kategoriat",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Albumin luonti epäonnistui",
"select_user_for_sharing_page_share_suggestions": "Ehdotukset",
"server_info_box_app_version": "Sovelluksen versio",
+ "server_info_box_server_url": "Palvelimen URL-osoite",
"server_info_box_server_version": "Palvelimen versio",
"setting_image_viewer_help": "Sovellus lataa ensin pienen esikatselukuvan, toisena keskitarkkuuksisen kuvan (jos käytössä) ja kolmantena alkuperäisen täysitarkkuuksisen kuvan (jos käytössä)",
"setting_image_viewer_original_subtitle": "Ota käyttöön ladataksesi alkuperäinen täysitarkkuuksinen kuva (suuri!). Poista käytöstä vähentääksesi datan käyttöä (sekä verkossa että laitteen välimuistissa).",
@@ -300,28 +315,37 @@
"share_add_photos": "Lisää kuvia",
"share_add_title": "Lisää nimi",
"share_create_album": "Luo albumi",
+ "shared_album_activities_input_disable": "Kommentointi on kytketty pois päältä",
+ "shared_album_activities_input_hint": "Sano jotain",
+ "shared_album_activity_remove_content": "Haluatko poistaa tämän aktiviteetin?",
+ "shared_album_activity_remove_title": "Poista aktiviteetti",
+ "shared_album_activity_setting_subtitle": "Anna muiden vastata",
+ "shared_album_activity_setting_title": "Kommentit ja tykkäykset",
"share_dialog_preparing": "Valmistellaan...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Jaetut linkit",
+ "shared_link_create_app_bar_title": "Luo linkki jaettavaksi",
+ "shared_link_create_info": "Salli kaikkien linkinhaltijoiden nähdä valitut kuvat",
+ "shared_link_create_submit_button": "Luo linkki",
+ "shared_link_edit_allow_download": "Salli julkisen käyttäjän ladata palvelimelta",
+ "shared_link_edit_allow_upload": "Salli julkisen käyttäjän siirtää palvelimelle",
+ "shared_link_edit_app_bar_title": "Muokkaa linkkiä",
+ "shared_link_edit_change_expiry": "Muuta erääntymisaikaa",
+ "shared_link_edit_description": "Kuvaus",
+ "shared_link_edit_description_hint": "Lisää jaon kuvaus",
+ "shared_link_edit_expire_after": "Umpeutuu",
+ "shared_link_edit_password": "Salasana",
+ "shared_link_edit_password_hint": "Syötä jaon salasana",
+ "shared_link_edit_show_meta": "Näytä metadata",
+ "shared_link_edit_submit_button": "Päivitä linkki",
+ "shared_link_empty": "Sinulla ei ole jaettuja linkkejä",
+ "shared_link_manage_links": "Hallitse jaettuja linkkejä",
+ "share_done": "Valmis",
"share_invite": "Kutsu albumiin",
"sharing_page_album": "Jaetut albumit",
"sharing_page_description": "Luo jaettuja albumeja jakaaksesi kuvia ja videoita läheisillesi.",
"sharing_page_empty_list": "TYHJÄ LISTA",
"sharing_silver_appbar_create_shared_album": "Luo jaettu albumi",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Jaetut linkit",
"sharing_silver_appbar_share_partner": "Jaa kumppanille",
"tab_controller_nav_library": "Kirjasto",
"tab_controller_nav_photos": "Kuvat",
@@ -337,19 +361,19 @@
"theme_setting_theme_title": "Teema",
"theme_setting_three_stage_loading_subtitle": "Kolmivaiheinen lataaminen saattaa parantaa latauksen suorituskykyä, mutta lisää kaistankäyttöä huomattavasti.",
"theme_setting_three_stage_loading_title": "Ota kolmivaiheinen lataus käyttöön",
- "translated_text_options": "Options",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
+ "translated_text_options": "Vaihtoehdot",
+ "trash_page_delete": "Poista",
+ "trash_page_delete_all": "Poista kaikki",
+ "trash_page_empty_trash_btn": "Tyhjennä roskakori",
+ "trash_page_empty_trash_dialog_content": "Haluatko poistaa roskakoriin siirretyt kohteet? Kohteet poistetaan lopullisesti Immich:sta.",
"trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
+ "trash_page_info": "Roskakoriin siirretyt kohteet poistetaan lopullisesti {} päivän jälkeen",
+ "trash_page_no_assets": "Ei poistettuja kohteita",
+ "trash_page_restore": "Palauta",
+ "trash_page_restore_all": "Palauta kaikki",
+ "trash_page_select_assets_btn": "Valitse kohteet",
+ "trash_page_select_btn": "Valitse",
+ "trash_page_title": "Roskakori",
"upload_dialog_cancel": "Peruuta",
"upload_dialog_info": "Haluatko varmuuskopioida valitut kohteet palvelimelle?",
"upload_dialog_ok": "Lähetä",
@@ -360,7 +384,7 @@
"version_announcement_overlay_text_2": "Ota hetki aikaa vieraillaksesi",
"version_announcement_overlay_text_3": "ja varmista, että käyttämäsi docker-compose ja .env-asetukset ovat ajantasalla välttyäksesi asetusongelmilta. Varsinkin jos käytät WatchToweria tai jotain muuta mekanismia päivittääksesi palvelinsovellusta automaattisesti.",
"version_announcement_overlay_title": "Uusi palvelinversio saatavilla \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Poista pinosta",
+ "viewer_stack_use_as_main_asset": "Käytä pääkohteena",
+ "viewer_unstack": "Pura pino"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/fr-CA.json b/mobile/assets/i18n/fr-CA.json
new file mode 100644
index 000000000..25e2615d9
--- /dev/null
+++ b/mobile/assets/i18n/fr-CA.json
@@ -0,0 +1,384 @@
+{
+ "add_to_album_bottom_sheet_added": "Ajouté à {album}",
+ "add_to_album_bottom_sheet_already_exists": "Déjà dans {album}",
+ "advanced_settings_prefer_remote_subtitle": "Certains appareils sont très lents à charger des vignettes à partir de ressources présentes sur l'appareil. Activez ce paramètre pour charger des images externes à la place.",
+ "advanced_settings_prefer_remote_title": "Préférer les images externes",
+ "advanced_settings_self_signed_ssl_subtitle": "Permet d'ignorer la vérification du certificat SSL pour le point d'accès du serveur. Requis pour les certificats auto-signés.",
+ "advanced_settings_self_signed_ssl_title": "Autoriser les certificats SSL auto-signés",
+ "advanced_settings_tile_subtitle": "Paramètres d'utilisateur avancés",
+ "advanced_settings_tile_title": "Avancé",
+ "advanced_settings_troubleshooting_subtitle": "Activer des fonctions supplémentaires pour le dépannage",
+ "advanced_settings_troubleshooting_title": "Dépannage",
+ "album_info_card_backup_album_excluded": "EXCLUS",
+ "album_info_card_backup_album_included": "INCLUS",
+ "album_thumbnail_card_item": "1 élément",
+ "album_thumbnail_card_items": "{} éléments",
+ "album_thumbnail_card_shared": " · Partagé",
+ "album_thumbnail_owned": "Possédé",
+ "album_thumbnail_shared_by": "Partagé par {}",
+ "album_viewer_appbar_share_delete": "Supprimer l'album",
+ "album_viewer_appbar_share_err_delete": "Échec de la suppression de l'album",
+ "album_viewer_appbar_share_err_leave": "Impossible de quitter l'album",
+ "album_viewer_appbar_share_err_remove": "Il y a des problèmes lors de la suppression des éléments de l'album",
+ "album_viewer_appbar_share_err_title": "Échec de la modification du titre de l'album",
+ "album_viewer_appbar_share_leave": "Quitter l'album",
+ "album_viewer_appbar_share_remove": "Retirer de l'album",
+ "album_viewer_appbar_share_to": "Partager à",
+ "album_viewer_page_share_add_users": "Ajouter des utilisateurs",
+ "all_people_page_title": "Personnes",
+ "all_videos_page_title": "Vidéos",
+ "app_bar_signout_dialog_content": "Êtes-vous sûr de vouloir vous déconnecter?",
+ "app_bar_signout_dialog_ok": "Oui",
+ "app_bar_signout_dialog_title": "Se déconnecter",
+ "archive_page_no_archived_assets": "Aucun élément archivé n'a été trouvé",
+ "archive_page_title": "Archive ({})",
+ "asset_list_layout_settings_dynamic_layout_title": "Affichage dynamique",
+ "asset_list_layout_settings_group_automatically": "Automatique",
+ "asset_list_layout_settings_group_by": "Grouper les éléments par",
+ "asset_list_layout_settings_group_by_month": "Mois",
+ "asset_list_layout_settings_group_by_month_day": "Mois + jour",
+ "asset_list_settings_subtitle": "Paramètres de disposition de la grille de photos",
+ "asset_list_settings_title": "Grille de photos",
+ "backup_album_selection_page_albums_device": "Albums sur l'appareil ({})",
+ "backup_album_selection_page_albums_tap": "Tapez pour inclure, tapez deux fois pour exclure",
+ "backup_album_selection_page_assets_scatter": "Les éléments peuvent être répartis sur plusieurs albums. De ce fait, les albums peuvent être inclus ou exclus pendant le processus de sauvegarde.",
+ "backup_album_selection_page_select_albums": "Sélectionner les albums",
+ "backup_album_selection_page_selection_info": "Informations sur la sélection",
+ "backup_album_selection_page_total_assets": "Total des éléments uniques",
+ "backup_all": "Tout",
+ "backup_background_service_backup_failed_message": "Échec de la sauvegarde des éléments. Nouvelle tentative...",
+ "backup_background_service_connection_failed_message": "Impossible de se connecter au serveur. Nouvelle tentative...",
+ "backup_background_service_current_upload_notification": "Transfert {}",
+ "backup_background_service_default_notification": "Recherche de nouveaux éléments...",
+ "backup_background_service_error_title": "Erreur de sauvegarde",
+ "backup_background_service_in_progress_notification": "Sauvegarde de vos éléments...",
+ "backup_background_service_upload_failure_notification": "Impossible de transférer {}",
+ "backup_controller_page_albums": "Sauvegarder les albums",
+ "backup_controller_page_background_app_refresh_disabled_content": "Activez le rafraîchissement de l'application en arrière-plan dans Paramètres > Général > Rafraîchissement de l'application en arrière-plan afin d'utiliser la sauvegarde en arrière-plan.",
+ "backup_controller_page_background_app_refresh_disabled_title": "Rafraîchissement de l'application en arrière-plan désactivé",
+ "backup_controller_page_background_app_refresh_enable_button_text": "Aller aux paramètres",
+ "backup_controller_page_background_battery_info_link": "Montrez-moi comment",
+ "backup_controller_page_background_battery_info_message": "Pour une expérience optimale de la sauvegarde en arrière-plan, veuillez désactiver toute optimisation de la batterie limitant l'activité en arrière-plan pour Immich.\n\nÉtant donné que cela est spécifique à chaque appareil, veuillez consulter les informations requises pour le fabricant de votre appareil.",
+ "backup_controller_page_background_battery_info_ok": "OK",
+ "backup_controller_page_background_battery_info_title": "Optimisation de la batterie",
+ "backup_controller_page_background_charging": "Seulement pendant la charge",
+ "backup_controller_page_background_configure_error": "Échec de la configuration du service d'arrière-plan",
+ "backup_controller_page_background_delay": "Retarder la sauvegarde des nouveaux éléments d'actif: {}",
+ "backup_controller_page_background_description": "Activez le service d'arrière-plan pour sauvegarder automatiquement tous les nouveaux éléments sans avoir à ouvrir l'application.",
+ "backup_controller_page_background_is_off": "La sauvegarde automatique en arrière-plan est désactivée",
+ "backup_controller_page_background_is_on": "La sauvegarde automatique en arrière-plan est activée",
+ "backup_controller_page_background_turn_off": "Désactiver le service d'arrière-plan",
+ "backup_controller_page_background_turn_on": "Activer le service d'arrière-plan",
+ "backup_controller_page_background_wifi": "Uniquement sur WiFi",
+ "backup_controller_page_backup": "Sauvegardé",
+ "backup_controller_page_backup_selected": "Sélectionné: ",
+ "backup_controller_page_backup_sub": "Photos et vidéos sauvegardées",
+ "backup_controller_page_cancel": "Annuler",
+ "backup_controller_page_created": "Créé le: {}",
+ "backup_controller_page_desc_backup": "Activez la sauvegarde pour envoyer automatiquement les nouveaux éléments sur le serveur.",
+ "backup_controller_page_excluded": "Exclus: ",
+ "backup_controller_page_failed": "Échec de l'opération ({})",
+ "backup_controller_page_filename": "Nom du fichier: {} [{}]",
+ "backup_controller_page_id": "ID: {}",
+ "backup_controller_page_info": "Informations de sauvegarde",
+ "backup_controller_page_none_selected": "Aucune sélection",
+ "backup_controller_page_remainder": "Restant",
+ "backup_controller_page_remainder_sub": "Photos et albums restants à sauvegarder à partir de la sélection",
+ "backup_controller_page_select": "Sélectionner",
+ "backup_controller_page_server_storage": "Stockage du serveur",
+ "backup_controller_page_start_backup": "Démarrer la sauvegarde",
+ "backup_controller_page_status_off": "La sauvegarde est désactivée",
+ "backup_controller_page_status_on": "La sauvegarde est activée",
+ "backup_controller_page_storage_format": "{} de {} utilisé",
+ "backup_controller_page_to_backup": "Albums à sauvegarder",
+ "backup_controller_page_total": "Total",
+ "backup_controller_page_total_sub": "Toutes les photos et vidéos uniques des albums sélectionnés",
+ "backup_controller_page_turn_off": "Désactiver la sauvegarde",
+ "backup_controller_page_turn_on": "Activer la sauvegarde",
+ "backup_controller_page_uploading_file_info": "Transfert des informations du fichier",
+ "backup_err_only_album": "Impossible de retirer le seul album",
+ "backup_info_card_assets": "éléments",
+ "backup_manual_cancelled": "Annulé",
+ "backup_manual_failed": "Echec",
+ "backup_manual_in_progress": "Téléchargement déjà en cours. Essayez après un instant",
+ "backup_manual_success": "Succès ",
+ "backup_manual_title": "Statut du téléchargement ",
+ "cache_settings_album_thumbnails": "vignettes de la page bibliothèque ({} éléments)",
+ "cache_settings_clear_cache_button": "Effacer le cache",
+ "cache_settings_clear_cache_button_title": "Efface le cache de l'application. Cela aura un impact significatif sur les performances de l'application jusqu'à ce que le cache soit reconstruit.",
+ "cache_settings_image_cache_size": "Taille du cache des images ({} éléments)",
+ "cache_settings_statistics_album": "vignettes de la bibliothèque",
+ "cache_settings_statistics_assets": "{} éléments ({})",
+ "cache_settings_statistics_full": "Images complètes",
+ "cache_settings_statistics_shared": "vignettes d'albums partagés",
+ "cache_settings_statistics_thumbnail": "vignettes",
+ "cache_settings_statistics_title": "Utilisation du cache",
+ "cache_settings_subtitle": "Contrôler le comportement de mise en cache de l'application mobile Immich",
+ "cache_settings_thumbnail_size": "Taille du cache des vignettes ({} éléments)",
+ "cache_settings_tile_subtitle": "Contrôler le comportement du stockage local",
+ "cache_settings_tile_title": "Stockage local",
+ "cache_settings_title": "Paramètres de mise en cache",
+ "change_password_form_confirm_password": "Confirmez le mot de passe",
+ "change_password_form_description": "Bonjour {name},\n\nC'est la première fois que vous vous connectez au système ou vous avez demandé de changer votre mot de passe. Veuillez saisir le nouveau mot de passe ci-dessous.",
+ "change_password_form_new_password": "Nouveau mot de passe",
+ "change_password_form_password_mismatch": "Les mots de passe ne correspondent pas",
+ "change_password_form_reenter_new_password": "Saisissez à nouveau le nouveau mot de passe",
+ "common_add_to_album": "Ajouter à l'album",
+ "common_change_password": "Modifier le mot de passe",
+ "common_create_new_album": "Créer un nouvel album",
+ "common_server_error": "Veuillez vérifier votre connexion réseau, vous assurer que le serveur est accessible et que les versions de l'application et du serveur sont compatibles.",
+ "common_shared": "Partagé",
+ "control_bottom_app_bar_add_to_album": "Ajouter à l'album",
+ "control_bottom_app_bar_album_info": "{} éléments",
+ "control_bottom_app_bar_album_info_shared": "{} éléments - Partagés",
+ "control_bottom_app_bar_archive": "Archive",
+ "control_bottom_app_bar_create_new_album": "Créer un nouvel album",
+ "control_bottom_app_bar_delete": "Supprimer",
+ "control_bottom_app_bar_favorite": "Favoris",
+ "control_bottom_app_bar_share": "Partager",
+ "control_bottom_app_bar_share_to": "Partager à",
+ "control_bottom_app_bar_stack": "Empiler",
+ "control_bottom_app_bar_unarchive": "Désarchiver",
+ "control_bottom_app_bar_upload": "Téléverser",
+ "create_album_page_untitled": "Sans titre",
+ "create_shared_album_page_create": "Créer",
+ "create_shared_album_page_share": "Partager",
+ "create_shared_album_page_share_add_assets": "AJOUTER DES ÉLÉMENTS",
+ "create_shared_album_page_share_select_photos": "Sélectionner les photos",
+ "curated_location_page_title": "Places",
+ "curated_object_page_title": "Objets",
+ "daily_title_text_date": "E, dd MMM",
+ "daily_title_text_date_year": "E, dd MMM, yyyy",
+ "date_format": "E, LLL d, y • h:mm a",
+ "delete_dialog_alert": "Ces éléments seront définitivement supprimés de Immich et de votre appareil.",
+ "delete_dialog_cancel": "Annuler",
+ "delete_dialog_ok": "Supprimer",
+ "delete_dialog_title": "Supprimer définitivement",
+ "delete_shared_link_dialog_content": "Êtes-vous sûr de vouloir supprimer ce lien partagé?",
+ "delete_shared_link_dialog_title": "Supprimer le lien partagé",
+ "description_input_hint_text": "Ajouter une description...",
+ "description_input_submit_error": "Erreur de mise à jour de la description, vérifier le journal pour plus de détails",
+ "exif_bottom_sheet_description": "Ajouter une description...",
+ "exif_bottom_sheet_details": "DÉTAILS",
+ "exif_bottom_sheet_location": "LOCALISATION",
+ "experimental_settings_new_asset_list_subtitle": "En cours de développement",
+ "experimental_settings_new_asset_list_title": "Activer la grille de photos expérimentale",
+ "experimental_settings_subtitle": "Utilisez à vos dépends!",
+ "experimental_settings_title": "Expérimental",
+ "favorites_page_no_favorites": "Aucun élément favori n'a été trouvé",
+ "favorites_page_title": "Favoris",
+ "home_page_add_to_album_conflicts": "{added} éléments ajoutés à l'album {album}. Les éléments {failed} sont déjà dans l'album.",
+ "home_page_add_to_album_err_local": "Impossible d'ajouter des éléments locaux aux albums pour le moment, étape ignorée",
+ "home_page_add_to_album_success": "{added} éléments ajoutés à l'album {album}.",
+ "home_page_archive_err_local": "Impossible d'archiver les ressources locales pour l'instant, étape ignorée",
+ "home_page_building_timeline": "Construction de la chronologie",
+ "home_page_favorite_err_local": "Impossible d'ajouter des éléments locaux aux favoris pour le moment, étape ignorée",
+ "home_page_first_time_notice": "Si c'est la première fois que vous utilisez l'application, veillez à choisir un ou plusieurs albums de sauvegarde afin que la chronologie puisse alimenter les photos et les vidéos de cet ou ces albums.",
+ "home_page_upload_err_limit": "Limite de téléchargement de 30 éléments en même temps, demande ignorée",
+ "image_viewer_page_state_provider_download_error": "Erreur de téléchargement",
+ "image_viewer_page_state_provider_download_success": "Téléchargement réussi",
+ "image_viewer_page_state_provider_share_error": "Erreur de partage",
+ "library_page_albums": "Albums",
+ "library_page_archive": "Archive",
+ "library_page_device_albums": "Albums sur l'appareil",
+ "library_page_favorites": "Favoris",
+ "library_page_new_album": "Nouvel album",
+ "library_page_sharing": "Partage",
+ "library_page_sort_created": "Créations les plus récentes",
+ "library_page_sort_last_modified": "Dernière modification",
+ "library_page_sort_most_recent_photo": "Photo la plus récente",
+ "library_page_sort_title": "Titre de l'album",
+ "login_disabled": "La connexion a été désactivée ",
+ "login_form_api_exception": "Erreur de l'API. Veuillez vérifier l'URL du serveur et et réessayer.",
+ "login_form_button_text": "Connexion",
+ "login_form_email_hint": "votreemail@email.com",
+ "login_form_endpoint_hint": "http://adresse-ip-serveur:port/api",
+ "login_form_endpoint_url": "URL du point d'accès au serveur",
+ "login_form_err_http": "Veuillez préciser http:// ou https://",
+ "login_form_err_invalid_email": "E-mail invalide",
+ "login_form_err_invalid_url": "URL invalide",
+ "login_form_err_leading_whitespace": "Espace en début de ligne",
+ "login_form_err_trailing_whitespace": "Espace de fin de ligne",
+ "login_form_failed_get_oauth_server_config": "Erreur de connexion par OAuth, vérifiez l\"URL du serveur",
+ "login_form_failed_get_oauth_server_disable": "La fonctionnalité OAuth n'est pas disponible sur ce serveur",
+ "login_form_failed_login": "Erreur de connexion, vérifiez l'url du serveur, l'email et le mot de passe",
+ "login_form_handshake_exception": "Il y a eu une exception de liaison avec le serveur. Activez la prise en charge des certificats auto-signés dans les paramètres si vous utilisez un certificat auto-signé.",
+ "login_form_label_email": "E-mail",
+ "login_form_label_password": "Mot de passe",
+ "login_form_next_button": "Suivant",
+ "login_form_password_hint": "mot de passe",
+ "login_form_save_login": "Rester connecté",
+ "login_form_server_empty": "Saisissez l'URL du serveur.",
+ "login_form_server_error": "Impossible de se connecter au serveur.",
+ "login_password_changed_error": "Une erreur s'est produite lors de la mise à jour de votre mot de passe",
+ "login_password_changed_success": "Mot de passe mis à jour avec succès",
+ "map_cannot_get_user_location": "Impossible d'obtenir la localisation de l'utilisateur",
+ "map_location_dialog_cancel": "Annuler",
+ "map_location_dialog_yes": "Oui",
+ "map_location_service_disabled_content": "Le service de localisation doit être activé pour afficher les éléments de votre emplacement actuel. Souhaitez-vous l'activer maintenant?",
+ "map_location_service_disabled_title": "Service de localisation désactivé",
+ "map_no_assets_in_bounds": "Pas de photos dans cette zone",
+ "map_no_location_permission_content": "L'autorisation de localisation est nécessaire pour afficher les éléments de votre emplacement actuel. Souhaitez-vous l'autoriser maintenant?",
+ "map_no_location_permission_title": "Permission de localisation refusée",
+ "map_settings_dark_mode": "Mode sombre",
+ "map_settings_dialog_cancel": "Annuler",
+ "map_settings_dialog_save": "Sauvegarder",
+ "map_settings_dialog_title": "Paramètres de la carte",
+ "map_settings_include_show_archived": "Inclure les archives",
+ "map_settings_only_relative_range": "Plage de dates",
+ "map_settings_only_show_favorites": "Afficher uniquement les favoris",
+ "map_zoom_to_see_photos": "Dézoomer pour voir les photos",
+ "monthly_title_text_date_format": "MMMM y",
+ "motion_photos_page_title": "Photos avec mouvement",
+ "notification_permission_dialog_cancel": "Annuler",
+ "notification_permission_dialog_content": "Pour activer les notifications, allez dans Paramètres et sélectionnez Autoriser.",
+ "notification_permission_dialog_settings": "Paramètres",
+ "notification_permission_list_tile_content": "Accordez la permission d'activer les notifications.",
+ "notification_permission_list_tile_enable_button": "Activer les notifications",
+ "notification_permission_list_tile_title": "Permission de notification",
+ "partner_page_add_partner": "Ajouter un partenaire",
+ "partner_page_empty_message": "Vos photos ne sont pas encore partagées avec un partenaire.",
+ "partner_page_no_more_users": "Plus d'utilisateurs à ajouter",
+ "partner_page_partner_add_failed": "Échec de l'ajout d'un partenaire",
+ "partner_page_select_partner": "Sélectionner un partenaire",
+ "partner_page_shared_to_title": "Partagé avec",
+ "partner_page_stop_sharing_content": "{} ne pourra plus accéder à vos photos.",
+ "partner_page_stop_sharing_title": "Arrêter de partager vos photos?",
+ "partner_page_title": "Partenaire",
+ "permission_onboarding_continue_anyway": "Continuer quand même",
+ "permission_onboarding_get_started": "Commencer",
+ "permission_onboarding_go_to_settings": "Accéder aux paramètres",
+ "permission_onboarding_grant_permission": "Accorder l'autorisation",
+ "permission_onboarding_log_out": "Se déconnecter",
+ "permission_onboarding_permission_denied": "Permission refusée. Pour utiliser Immich, accordez lautorisation pour les photos et vidéos dans les Paramètres.",
+ "permission_onboarding_permission_granted": "Permission accordée! Vous êtes prêts.",
+ "permission_onboarding_permission_limited": "Permission limitée. Pour permettre à Immich de sauvegarder et de gérer l'ensemble de votre bibliothèque, accordez l'autorisation pour les photos et vidéos dans les Paramètres.",
+ "permission_onboarding_request": "Immich demande l'autorisation de visionner vos photos et vidéo",
+ "profile_drawer_app_logs": "Journaux",
+ "profile_drawer_client_server_up_to_date": "Le client et le serveur sont à jour",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
+ "profile_drawer_settings": "Paramètres",
+ "profile_drawer_sign_out": "Se déconnecter",
+ "profile_drawer_trash": "Corbeille",
+ "recently_added_page_title": "Récemment ajouté",
+ "search_bar_hint": "Rechercher vos photos",
+ "search_page_categories": "Catégories",
+ "search_page_favorites": "Favoris",
+ "search_page_motion_photos": "Photos avec mouvement",
+ "search_page_no_objects": "Aucune information disponible sur les objets",
+ "search_page_no_places": "Aucune information disponible sur la localisation",
+ "search_page_people": "Personnes",
+ "search_page_places": "Lieux",
+ "search_page_recently_added": "Récemment ajouté",
+ "search_page_screenshots": "Captures d'écran",
+ "search_page_selfies": "Selfies",
+ "search_page_things": "Objets",
+ "search_page_videos": "Vidéos",
+ "search_page_view_all_button": "Voir tout",
+ "search_page_your_activity": "Votre activité",
+ "search_result_page_new_search_hint": "Nouvelle recherche",
+ "search_suggestion_list_smart_search_hint_1": "La recherche intelligente est activée par défaut. Pour rechercher des métadonnées, utilisez la syntaxe suivante",
+ "search_suggestion_list_smart_search_hint_2": "m:votre-terme-de-recherche",
+ "select_additional_user_for_sharing_page_suggestions": "Suggestions",
+ "select_user_for_sharing_page_err_album": "Échec de la création de l'album",
+ "select_user_for_sharing_page_share_suggestions": "Suggestions",
+ "server_info_box_app_version": "Version de l'application",
+ "server_info_box_server_url": "URL du serveur",
+ "server_info_box_server_version": "Version du serveur",
+ "setting_image_viewer_help": "Le visualiseur de détails charge d'abord la petite vignette, puis l'aperçu de taille moyenne (s'il est activé), enfin l'original (s'il est activé).",
+ "setting_image_viewer_original_subtitle": "Activez cette option pour charger l'image en résolution originale (volumineux!). Désactiver pour réduire l'utilisation des données (réseau et cache de l'appareil).",
+ "setting_image_viewer_original_title": "Charger l'image originale",
+ "setting_image_viewer_preview_subtitle": "Activer pour charger une image de résolution moyenne. Désactiver pour charger directement l'original ou utiliser uniquement la vignette.",
+ "setting_image_viewer_preview_title": "Charger l'image d'aperçu",
+ "setting_notifications_notify_failures_grace_period": "Notifier les échecs de la sauvegarde en arrière-plan: {}",
+ "setting_notifications_notify_hours": "{} heures",
+ "setting_notifications_notify_immediately": "immédiatement",
+ "setting_notifications_notify_minutes": "{} minutes",
+ "setting_notifications_notify_never": "jamais",
+ "setting_notifications_notify_seconds": "{} secondes",
+ "setting_notifications_single_progress_subtitle": "Informations détaillées sur la progression du transfert par élément",
+ "setting_notifications_single_progress_title": "Afficher la progression du détail de la sauvegarde en arrière-plan",
+ "setting_notifications_subtitle": "Ajustez vos préférences de notification",
+ "setting_notifications_title": "Notifications",
+ "setting_notifications_total_progress_subtitle": "Progression globale du transfert (effectué/total des éléments)",
+ "setting_notifications_total_progress_title": "Afficher la progression totale de la sauvegarde en arrière-plan",
+ "setting_pages_app_bar_settings": "Paramètres",
+ "settings_require_restart": "Veuillez redémarrer Immich pour appliquer ce paramètre",
+ "share_add": "Ajouter",
+ "share_add_photos": "Ajouter des photos",
+ "share_add_title": "Ajouter un titre",
+ "share_create_album": "Créer un album",
+ "shared_album_activities_input_disable": "Les commentaires sont désactivés",
+ "shared_album_activities_input_hint": "Dire quelque chose",
+ "shared_album_activity_remove_content": "Souhaitez-vous supprimer cette activité?",
+ "shared_album_activity_remove_title": "Supprimer l'activité",
+ "shared_album_activity_setting_subtitle": "Laisser les autres réagir",
+ "shared_album_activity_setting_title": "Commentaires et likes",
+ "share_dialog_preparing": "Préparation...",
+ "shared_link_app_bar_title": "Liens partagés",
+ "shared_link_create_app_bar_title": "Créer un lien pour partager",
+ "shared_link_create_info": "Permettre à toute personne ayant le lien de voir la ou les photos sélectionnées",
+ "shared_link_create_submit_button": "Créer le lien",
+ "shared_link_edit_allow_download": "Autoriser les utilisateurs publics à télécharger",
+ "shared_link_edit_allow_upload": "Autoriser les utilisateurs publics à téléverser",
+ "shared_link_edit_app_bar_title": "Modifier le lien",
+ "shared_link_edit_change_expiry": "Modifier le délai d'expiration",
+ "shared_link_edit_description": "Description",
+ "shared_link_edit_description_hint": "Saisir la description du partage",
+ "shared_link_edit_expire_after": "Expire après",
+ "shared_link_edit_password": "Mot de passe",
+ "shared_link_edit_password_hint": "Saisir le mot de passe de partage",
+ "shared_link_edit_show_meta": "Afficher les métadonnées",
+ "shared_link_edit_submit_button": "Mettre à jour le lien",
+ "shared_link_empty": "Vous n'avez pas de liens partagés",
+ "shared_link_manage_links": "Gérer les liens partagés",
+ "share_done": "Fait",
+ "share_invite": "Inviter à l'album",
+ "sharing_page_album": "Albums partagés",
+ "sharing_page_description": "Créez des albums partagés pour partager des photos et des vidéos avec les personnes de votre réseau.",
+ "sharing_page_empty_list": "LISTE VIDE",
+ "sharing_silver_appbar_create_shared_album": "Créer un album partagé",
+ "sharing_silver_appbar_shared_links": "Liens partagés",
+ "sharing_silver_appbar_share_partner": "Partager avec un partenaire",
+ "tab_controller_nav_library": "Bibliothèque",
+ "tab_controller_nav_photos": "Photos",
+ "tab_controller_nav_search": "Recherche",
+ "tab_controller_nav_sharing": "Partage",
+ "theme_setting_asset_list_storage_indicator_title": "Afficher l'indicateur de stockage sur les tuiles des éléments",
+ "theme_setting_asset_list_tiles_per_row_title": "Nombre d'éléments par ligne ({})",
+ "theme_setting_dark_mode_switch": "Mode sombre",
+ "theme_setting_image_viewer_quality_subtitle": "Ajustez la qualité de la visionneuse d'images détaillées",
+ "theme_setting_image_viewer_quality_title": "Qualité de la visualisation des images",
+ "theme_setting_system_theme_switch": "Automatique (suivre les paramètres du système)",
+ "theme_setting_theme_subtitle": "Choisissez le thème de l'application",
+ "theme_setting_theme_title": "Thème",
+ "theme_setting_three_stage_loading_subtitle": "Le chargement en trois étapes peut améliorer les performances de chargement, mais entraîne une augmentation significative de la charge du réseau.",
+ "theme_setting_three_stage_loading_title": "Activer le chargement en trois étapes",
+ "translated_text_options": "Options",
+ "trash_page_delete": "Supprimer",
+ "trash_page_delete_all": "Tout supprimer",
+ "trash_page_empty_trash_btn": "Vider la corbeille",
+ "trash_page_empty_trash_dialog_content": "Voulez-vous vider les éléments de la corbeille? Ces objets seront définitivement retirés d'Immich",
+ "trash_page_empty_trash_dialog_ok": "Ok",
+ "trash_page_info": "Les éléments mis à la corbeille seront définitivement supprimés au bout de {} jours.",
+ "trash_page_no_assets": "Pas d'éléments dans la corbeille",
+ "trash_page_restore": "Restaurer",
+ "trash_page_restore_all": "Tout restaurer",
+ "trash_page_select_assets_btn": "Sélectionner les éléments",
+ "trash_page_select_btn": "Sélectionner",
+ "trash_page_title": "Corbeille ({})",
+ "upload_dialog_cancel": "Annuler",
+ "upload_dialog_info": "Voulez-vous sauvegarder la sélection vers le serveur?",
+ "upload_dialog_ok": "Télécharger ",
+ "upload_dialog_title": "Télécharger cet élément ",
+ "version_announcement_overlay_ack": "Confirmer",
+ "version_announcement_overlay_release_notes": "notes de mise à jour",
+ "version_announcement_overlay_text_1": "Bonjour, une nouvelle version de",
+ "version_announcement_overlay_text_2": "veuillez prendre le temps de visiter le ",
+ "version_announcement_overlay_text_3": " et assurez-vous que votre configuration docker-compose et .env est à jour pour éviter toute erreur de configuration, en particulier si vous utilisez WatchTower ou tout autre mécanisme qui gère la mise à jour automatique de votre application serveur.",
+ "version_announcement_overlay_title": "Nouvelle version serveur disponible \uD83C\uDF89",
+ "viewer_remove_from_stack": "Retirer de la pile",
+ "viewer_stack_use_as_main_asset": "Utiliser comme élément principal",
+ "viewer_unstack": "Désempiler"
+ }
\ No newline at end of file
diff --git a/mobile/assets/i18n/fr-FR.json b/mobile/assets/i18n/fr-FR.json
index 5eacd5b56..22050072d 100644
--- a/mobile/assets/i18n/fr-FR.json
+++ b/mobile/assets/i18n/fr-FR.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Échec de la modification du titre de l'album",
"album_viewer_appbar_share_leave": "Quitter l'album",
"album_viewer_appbar_share_remove": "Retirer de l'album",
+ "album_viewer_appbar_share_to": "Partager à",
"album_viewer_page_share_add_users": "Ajouter des utilisateurs",
"all_people_page_title": "Personnes",
"all_videos_page_title": "Vidéos",
+ "app_bar_signout_dialog_content": "Êtes-vous sûr de vouloir vous déconnecter ?",
+ "app_bar_signout_dialog_ok": "Oui",
+ "app_bar_signout_dialog_title": "Se déconnecter",
"archive_page_no_archived_assets": "Aucun élément archivé n'a été trouvé",
"archive_page_title": "Archive ({})",
"asset_list_layout_settings_dynamic_layout_title": "Affichage dynamique",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Utilisation du cache",
"cache_settings_subtitle": "Contrôler le comportement de mise en cache de l'application mobile Immich",
"cache_settings_thumbnail_size": "Taille du cache des miniatures ({} éléments)",
+ "cache_settings_tile_subtitle": "Contrôler le comportement du stockage local",
+ "cache_settings_tile_title": "Stockage local",
"cache_settings_title": "Paramètres de mise en cache",
"change_password_form_confirm_password": "Confirmez le mot de passe",
- "change_password_form_description": "Bonjour {firstName} {lastName},\n\nC'est la première fois que vous vous connectez au système ou vous avez demandé à changer votre mot de passe. Veuillez saisir le nouveau mot de passe ci-dessous.",
+ "change_password_form_description": "Bonjour {name},\n\nC'est la première fois que vous vous connectez au système ou vous avez demandé à changer votre mot de passe. Veuillez saisir le nouveau mot de passe ci-dessous.",
"change_password_form_new_password": "Nouveau mot de passe",
"change_password_form_password_mismatch": "Les mots de passe ne correspondent pas",
"change_password_form_reenter_new_password": "Saisissez à nouveau le nouveau mot de passe",
@@ -130,10 +136,10 @@
"control_bottom_app_bar_delete": "Supprimer",
"control_bottom_app_bar_favorite": "Favoris",
"control_bottom_app_bar_share": "Partager",
- "control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_share_to": "Partager à",
+ "control_bottom_app_bar_stack": "Empiler",
"control_bottom_app_bar_unarchive": "Désarchiver",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Téléverser",
"create_album_page_untitled": "Sans titre",
"create_shared_album_page_create": "Créer",
"create_shared_album_page_share": "Partager",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "Annuler",
"delete_dialog_ok": "Supprimer",
"delete_dialog_title": "Supprimer définitivement",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Êtes-vous sûr de vouloir supprimer ce lien partagé ?",
+ "delete_shared_link_dialog_title": "Supprimer le lien partagé",
"description_input_hint_text": "Ajouter une description...",
"description_input_submit_error": "Erreur de mise à jour de la description, vérifier le journal pour plus de détails",
"exif_bottom_sheet_description": "Ajouter une description...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "{added} éléments ajoutés à l'album {album}. Les éléments {failed} sont déjà dans l'album.",
"home_page_add_to_album_err_local": "Impossible d'ajouter des éléments locaux aux albums pour le moment, étape ignorée",
"home_page_add_to_album_success": "{added} éléments ajoutés à l'album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Impossible d'archiver les ressources locales pour l'instant, étape ignorée",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Construction de la chronologie",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Impossible d'ajouter des éléments locaux aux favoris pour le moment, étape ignorée",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Si c'est la première fois que vous utilisez l'application, veillez à choisir un ou plusieurs albums de sauvegarde afin que la chronologie puisse alimenter les photos et les vidéos de cet ou ces albums.",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Limite de téléchargement de 30 éléments en même temps, demande ignorée",
"image_viewer_page_state_provider_download_error": "Erreur de téléchargement",
"image_viewer_page_state_provider_download_success": "Téléchargement réussi",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Erreur de partage",
"library_page_albums": "Albums",
"library_page_archive": "Archive",
"library_page_device_albums": "Albums sur l'appareil",
@@ -179,8 +190,8 @@
"library_page_new_album": "Nouvel album",
"library_page_sharing": "Partage",
"library_page_sort_created": "Créations les plus récentes",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Dernière modification",
+ "library_page_sort_most_recent_photo": "Photo la plus récente",
"library_page_sort_title": "Titre de l'album",
"login_disabled": "La connexion a été désactivée ",
"login_form_api_exception": "Erreur de l'API. Veuillez vérifier l'URL du serveur et et réessayer.",
@@ -218,7 +229,7 @@
"map_settings_dialog_cancel": "Annuler",
"map_settings_dialog_save": "Sauvegarder",
"map_settings_dialog_title": "Paramètres de la carte",
- "map_settings_include_show_archived": "Include Archived",
+ "map_settings_include_show_archived": "Inclure les archives",
"map_settings_only_relative_range": "Plage de dates",
"map_settings_only_show_favorites": "Afficher uniquement les favoris",
"map_zoom_to_see_photos": "Dézoomer pour voir les photos",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} ne pourra plus accéder à vos photos.",
"partner_page_stop_sharing_title": "Arrêter de partager vos photos ?",
"partner_page_title": "Partenaire",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continuer quand même",
"permission_onboarding_get_started": "Commencer",
"permission_onboarding_go_to_settings": "Accéder aux paramètres",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich demande l'autorisation de visionner vos photos et vidéo",
"profile_drawer_app_logs": "Journaux",
"profile_drawer_client_server_up_to_date": "Le client et le serveur sont à jour",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Paramètres",
"profile_drawer_sign_out": "Se déconnecter",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Corbeille",
"recently_added_page_title": "Récemment ajouté",
"search_bar_hint": "Rechercher vos photos",
"search_page_categories": "Catégories",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Échec de la création de l'album",
"select_user_for_sharing_page_share_suggestions": "Suggestions",
"server_info_box_app_version": "Version de l'application",
+ "server_info_box_server_url": "URL du serveur",
"server_info_box_server_version": "Version du serveur",
"setting_image_viewer_help": "Le visualiseur de détails charge d'abord la petite miniature, puis l'aperçu de taille moyenne (s'il est activé), enfin l'original (s'il est activé).",
"setting_image_viewer_original_subtitle": "Activez cette option pour charger l'image en résolution originale (volumineux !). Désactiver pour réduire l'utilisation des données (réseau et cache de l'appareil).",
@@ -300,28 +315,37 @@
"share_add_photos": "Ajouter des photos",
"share_add_title": "Ajouter un titre",
"share_create_album": "Créer un album",
+ "shared_album_activities_input_disable": "Les commentaires sont désactivés",
+ "shared_album_activities_input_hint": "Dire quelque chose",
+ "shared_album_activity_remove_content": "Souhaitez-vous supprimer cette activité ?",
+ "shared_album_activity_remove_title": "Supprimer l'activité",
+ "shared_album_activity_setting_subtitle": "Laisser les autres réagir",
+ "shared_album_activity_setting_title": "Commentaires et likes",
"share_dialog_preparing": "Préparation...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
+ "shared_link_app_bar_title": "Liens partagés",
+ "shared_link_create_app_bar_title": "Créer un lien pour partager",
+ "shared_link_create_info": "Permettre à toute personne ayant le lien de voir la ou les photos sélectionnées",
+ "shared_link_create_submit_button": "Créer le lien",
+ "shared_link_edit_allow_download": "Autoriser les utilisateurs publics à télécharger",
+ "shared_link_edit_allow_upload": "Autoriser les utilisateurs publics à téléverser",
+ "shared_link_edit_app_bar_title": "Modifier le lien",
+ "shared_link_edit_change_expiry": "Modifier le délai d'expiration",
"shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_edit_description_hint": "Saisir la description du partage",
+ "shared_link_edit_expire_after": "Expire après",
+ "shared_link_edit_password": "Mot de passe",
+ "shared_link_edit_password_hint": "Saisir le mot de passe de partage",
+ "shared_link_edit_show_meta": "Afficher les métadonnées",
+ "shared_link_edit_submit_button": "Mettre à jour le lien",
+ "shared_link_empty": "Vous n'avez pas de liens partagés",
+ "shared_link_manage_links": "Gérer les liens partagés",
+ "share_done": "Fait",
"share_invite": "Inviter à l'album",
"sharing_page_album": "Albums partagés",
"sharing_page_description": "Créez des albums partagés pour partager des photos et des vidéos avec les personnes de votre réseau.",
"sharing_page_empty_list": "LISTE VIDE",
"sharing_silver_appbar_create_shared_album": "Créer un album partagé",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Liens partagés",
"sharing_silver_appbar_share_partner": "Partager avec un partenaire",
"tab_controller_nav_library": "Bibliothèque",
"tab_controller_nav_photos": "Photos",
@@ -338,18 +362,18 @@
"theme_setting_three_stage_loading_subtitle": "Le chargement en trois étapes peut améliorer les performances de chargement, mais entraîne une augmentation significative de la charge du réseau.",
"theme_setting_three_stage_loading_title": "Activer le chargement en trois étapes",
"translated_text_options": "Options",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
+ "trash_page_delete": "Supprimer",
+ "trash_page_delete_all": "Tout supprimer",
+ "trash_page_empty_trash_btn": "Vider la corbeille",
+ "trash_page_empty_trash_dialog_content": "Voulez-vous vider les éléments de la corbeille? Ces objets seront définitivement retirés d'Immich",
"trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
+ "trash_page_info": "Les éléments mis à la corbeille seront définitivement supprimés au bout de {} jours.",
+ "trash_page_no_assets": "Pas d'éléments dans la corbeille",
+ "trash_page_restore": "Restaurer",
+ "trash_page_restore_all": "Tout restaurer",
+ "trash_page_select_assets_btn": "Sélectionner les éléments",
+ "trash_page_select_btn": "Sélectionner",
+ "trash_page_title": "Corbeille ({})",
"upload_dialog_cancel": "Annuler",
"upload_dialog_info": "Voulez-vous sauvegarder la sélection vers le serveur ?",
"upload_dialog_ok": "Télécharger ",
@@ -360,7 +384,7 @@
"version_announcement_overlay_text_2": "veuillez prendre le temps de visiter le ",
"version_announcement_overlay_text_3": " et assurez-vous que votre configuration docker-compose et .env est à jour pour éviter toute erreur de configuration, en particulier si vous utilisez WatchTower ou tout autre mécanisme qui gère la mise à jour automatique de votre application serveur.",
"version_announcement_overlay_title": "Nouvelle version serveur disponible \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Retirer de la pile",
+ "viewer_stack_use_as_main_asset": "Utiliser comme élément principal",
+ "viewer_unstack": "Désempiler"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/hi-IN.json b/mobile/assets/i18n/hi-IN.json
index 31311535b..9c42afb7d 100644
--- a/mobile/assets/i18n/hi-IN.json
+++ b/mobile/assets/i18n/hi-IN.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Failed to change album title",
"album_viewer_appbar_share_leave": "Leave album",
"album_viewer_appbar_share_remove": "Remove from album",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Add users",
"all_people_page_title": "People",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "No archived assets found",
"archive_page_title": "Archive ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamic layout",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Cache usage",
"cache_settings_subtitle": "Control the caching behaviour of the Immich mobile application",
"cache_settings_thumbnail_size": "Thumbnail cache size ({} assets)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Caching Settings",
"change_password_form_confirm_password": "Confirm Password",
- "change_password_form_description": "Hi {firstName} {lastName},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
+ "change_password_form_description": "Hi {name},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
"change_password_form_new_password": "New Password",
"change_password_form_password_mismatch": "Passwords do not match",
"change_password_form_reenter_new_password": "Re-enter New Password",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Added {added} assets to album {album}. {failed} assets are already in the album.",
"home_page_add_to_album_err_local": "Can not add local assets to albums yet, skipping",
"home_page_add_to_album_success": "Added {added} assets to album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Can not archive local assets yet, skipping",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Building the timeline",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Can not favorite local assets yet, skipping",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "If this is your first time using the app, please make sure to choose a backup album(s) so that the timeline can populate photos and videos in the album(s).",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
"image_viewer_page_state_provider_download_error": "Download Error",
"image_viewer_page_state_provider_download_success": "Download Success",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
"partner_page_stop_sharing_title": "Stop sharing your photos?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continue anyway",
"permission_onboarding_get_started": "Get started",
"permission_onboarding_go_to_settings": "Go to settings",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich requires permission to view your photos and videos.",
"profile_drawer_app_logs": "Logs",
"profile_drawer_client_server_up_to_date": "Client and Server are up-to-date",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Settings",
"profile_drawer_sign_out": "Sign Out",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Failed to create album",
"select_user_for_sharing_page_share_suggestions": "Suggestions",
"server_info_box_app_version": "App Version",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Server Version",
"setting_image_viewer_help": "The detail viewer loads the small thumbnail first, then loads the medium-size preview (if enabled), finally loads the original (if enabled).",
"setting_image_viewer_original_subtitle": "Enable to load the original full-resolution image (large!). Disable to reduce data usage (both network and on device cache).",
@@ -300,6 +315,12 @@
"share_add_photos": "Add photos",
"share_add_title": "Add a title",
"share_create_album": "Create album",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Preparing...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/hu-HU.json b/mobile/assets/i18n/hu-HU.json
index cc177bffb..60bf00d99 100644
--- a/mobile/assets/i18n/hu-HU.json
+++ b/mobile/assets/i18n/hu-HU.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Hiba az album átnevezése közben",
"album_viewer_appbar_share_leave": "Kilépés az albumból",
"album_viewer_appbar_share_remove": "Törlés az albumból",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Felhasználók hozzáadása",
"all_people_page_title": "Emberek",
"all_videos_page_title": "Videók",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "Nem található archivált média",
"archive_page_title": "Archívum ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamic layout",
@@ -111,6 +115,8 @@
"cache_settings_statistics_title": "Gyorsítótár által használt terület",
"cache_settings_subtitle": "Control the caching behaviour of the Immich mobile application",
"cache_settings_thumbnail_size": "Thumbnail cache size ({} assets)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Gyorsítótár beállítások",
"change_password_form_confirm_password": "Jelszó Megerősítése",
"change_password_form_description": "Kedves {lastName} {firstName}!\n\nMost jelentkezel be először a rendszerbe vagy más okból szükséfes a jelszavad meváltoztatása. Kérjük, add meg új jelszavad.",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Added {added} assets to album {album}. {failed} assets are already in the album.",
"home_page_add_to_album_err_local": "Helyi médiát még nem lehet albumba tenni. Kihagyjuk.",
"home_page_add_to_album_success": "Added {added} assets to album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Helyi média archiválása még nem támogatott, úgyhogy kihagyjuk",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Building the timeline",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Helyi médiát még nem lehet a kedvencek közé tenni. Kihagyjuk.",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "If this is your first time using the app, please make sure to choose a backup album(s) so that the timeline can populate photos and videos in the album(s).",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Csak 30 elemet tudsz egyszerre feltölteni, átugrás",
"image_viewer_page_state_provider_download_error": "Letöltési Hiba",
"image_viewer_page_state_provider_download_success": "Letöltés Sikeres",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
"partner_page_stop_sharing_title": "Stop sharing your photos?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Folytatás mindenképp",
"permission_onboarding_get_started": "Kezdjük el",
"permission_onboarding_go_to_settings": "Beállítások megnyitása",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Engedélyezni kell, hogy az Immich hozzáférjen a képekhez és videókhoz",
"profile_drawer_app_logs": "Naplók",
"profile_drawer_client_server_up_to_date": "Kliens és a szerver is naprakész",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Beállítások",
"profile_drawer_sign_out": "Kijelentkezés",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Hiba az album létrehozása közben",
"select_user_for_sharing_page_share_suggestions": "Javaslatok",
"server_info_box_app_version": "Alkalmazás Verzió",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Szerver Verzió",
"setting_image_viewer_help": "The detail viewer loads the small thumbnail first, then loads the medium-size preview (if enabled), finally loads the original (if enabled).",
"setting_image_viewer_original_subtitle": "Enable to load the original full-resolution image (large!). Disable to reduce data usage (both network and on device cache).",
@@ -300,6 +315,12 @@
"share_add_photos": "Fotók hozzáadása",
"share_add_title": "Cím hozzáadása",
"share_create_album": "Album létrehozása",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Előkészítés...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/it-IT.json b/mobile/assets/i18n/it-IT.json
index 479a46016..d8a372527 100644
--- a/mobile/assets/i18n/it-IT.json
+++ b/mobile/assets/i18n/it-IT.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Impossibile cambiare il titolo dell'album ",
"album_viewer_appbar_share_leave": "Lascia album",
"album_viewer_appbar_share_remove": "Rimuovere dall'album ",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Aggiungi utenti",
"all_people_page_title": "Persone",
"all_videos_page_title": "Video",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "Nessuna oggetto archiviato",
"archive_page_title": "Archivia ({})",
"asset_list_layout_settings_dynamic_layout_title": "Layout dinamico",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Uso della cache",
"cache_settings_subtitle": "Controlla il comportamento della cache dell'applicazione mobile immich",
"cache_settings_thumbnail_size": "Dimensione cache dei thumbnail ({} assets)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Impostazioni della Cache",
"change_password_form_confirm_password": "Conferma Password ",
- "change_password_form_description": "Ciao {firstName} {lastName},\n\nQuesto è la prima volta che accedi al sistema oppure è stato fatto una richiesta di cambiare la password. Per favore inserisca la nuova password qui sotto",
+ "change_password_form_description": "Ciao {name},\n\nQuesto è la prima volta che accedi al sistema oppure è stato fatto una richiesta di cambiare la password. Per favore inserisca la nuova password qui sotto",
"change_password_form_new_password": "Nuova Password",
"change_password_form_password_mismatch": "Le password non coincidono",
"change_password_form_reenter_new_password": "Inserisci ancora la nuova password ",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Aggiunti {added} elementi all'album {album}. {failed} elementi erano già presenti nell'album.",
"home_page_add_to_album_err_local": "Non puoi aggiungere negli album foto ancora non caricate",
"home_page_add_to_album_success": "Aggiunti {added} elementi all'album {album}",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Non puoi archiviare immagini non ancora caricate",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Costruendo il Timeline",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Non puoi aggiungere tra i preferiti le foto ancora non caricate",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Se è la prima volta che usi l'app, assicurati di scegliere gli album per avere il Timeline con immagini e video",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Puoi caricare al massimo 30 file per volta, ignora quelli in eccesso",
"image_viewer_page_state_provider_download_error": "Errore nel Download",
"image_viewer_page_state_provider_download_success": "Download con successo",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} non sarà più in grado di accedere alle tue foto.",
"partner_page_stop_sharing_title": "Stoppare la condivisione delle tue foto?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continua lo stesso",
"permission_onboarding_get_started": "Inizia",
"permission_onboarding_go_to_settings": "Vai a Impostazioni",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich richiede i permessi per vedere le tue foto e video",
"profile_drawer_app_logs": "Logs",
"profile_drawer_client_server_up_to_date": "Client e server sono aggiornati",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Impostazioni ",
"profile_drawer_sign_out": "Logout",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Impossibile nel creare l'album ",
"select_user_for_sharing_page_share_suggestions": "Suggerimenti",
"server_info_box_app_version": "Versione App",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Versione Server",
"setting_image_viewer_help": "Il visualizzatore dettagliato carica una piccola thumbnail per prima, per poi caricare un immagine di media grandezza (se abilitato). Ed infine carica l'originale (se abilitato).",
"setting_image_viewer_original_subtitle": "Abilita per caricare l'immagine originale a risoluzione massima (grande!). Disabilita per ridurre l'utilizzo di banda (sia sul network che nella cache del dispositivo).",
@@ -300,6 +315,12 @@
"share_add_photos": "Aggiungi foto",
"share_add_title": "Aggiungi un titolo ",
"share_create_album": "Crea album",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Preparo…",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
@@ -355,10 +379,10 @@
"upload_dialog_ok": "Carica",
"upload_dialog_title": "Carica file",
"version_announcement_overlay_ack": "Presa visione",
- "version_announcement_overlay_release_notes": "note di rilascio ",
+ "version_announcement_overlay_release_notes": "note di rilascio",
"version_announcement_overlay_text_1": "Ciao, c'è una nuova versione di",
- "version_announcement_overlay_text_2": "per favore prenditi il tuo tempo per visitare il",
- "version_announcement_overlay_text_3": "e verifica che il tuo docker-compose e il file .env siano aggiornati per impedire qualsiasi errore di configurazione, specialmente se utilizzate WatchTower o altri strumenti per l'aggiornamento automatico dell'applicativo",
+ "version_announcement_overlay_text_2": "per favore prenditi il tuo tempo per visitare le ",
+ "version_announcement_overlay_text_3": " e verifica che il tuo docker-compose e il file .env siano aggiornati per impedire qualsiasi errore di configurazione, specialmente se utilizzate WatchTower o altri strumenti per l'aggiornamento automatico dell'applicativo",
"version_announcement_overlay_title": "Nuova versione del server disponibile \uD83C\uDF89",
"viewer_remove_from_stack": "Remove from Stack",
"viewer_stack_use_as_main_asset": "Use as Main Asset",
diff --git a/mobile/assets/i18n/ja-JP.json b/mobile/assets/i18n/ja-JP.json
index de2b23e06..07658e444 100644
--- a/mobile/assets/i18n/ja-JP.json
+++ b/mobile/assets/i18n/ja-JP.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "タイトル変更の失敗",
"album_viewer_appbar_share_leave": "アルバムから脱退",
"album_viewer_appbar_share_remove": "アルバムから削除",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "ユーザーを追加",
"all_people_page_title": "People",
"all_videos_page_title": "ビデオ",
+ "app_bar_signout_dialog_content": " サインアウトしますか?",
+ "app_bar_signout_dialog_ok": "はい",
+ "app_bar_signout_dialog_title": " サインアウト",
"archive_page_no_archived_assets": "アーカイブ済みの写真またはビデオがありません",
"archive_page_title": "アーカイブ({})",
"asset_list_layout_settings_dynamic_layout_title": "ダイナミックレイアウト",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "キャッシュ",
"cache_settings_subtitle": "キャッシュの動作を変更する",
"cache_settings_thumbnail_size": "サムネイルのキャッシュのサイズ ({}枚)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "キャッシュの設定",
"change_password_form_confirm_password": "確定",
- "change_password_form_description": "{lastaName} {firstName}さん こんにちは\n\nサーバーにアクセスするのが初めてか、パスワードリセットのリクエストがされました。新しいパスワードを入力してください",
+ "change_password_form_description": "{lastName} {firstName}さん こんにちは\n\nサーバーにアクセスするのが初めてか、パスワードリセットのリクエストがされました。新しいパスワードを入力してください",
"change_password_form_new_password": "新しいパスワード",
"change_password_form_password_mismatch": "パスワードが一致しません",
"change_password_form_reenter_new_password": "再度パスワードを入力してください",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "キャンセル",
"delete_dialog_ok": "削除",
"delete_dialog_title": "永久的に削除",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "本当にこの共有リンクを消しますか?",
+ "delete_shared_link_dialog_title": "共有リンクを消す",
"description_input_hint_text": "説明を追加",
"description_input_submit_error": "説明の編集に失敗、詳細の確認はログで行ってください",
"exif_bottom_sheet_description": "説明を追加",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "{album}に{added}枚写真を追加しました。追加済みの{failed}枚はスキップしました。",
"home_page_add_to_album_err_local": "まだアップロードされてない項目はアルバムに登録できません",
"home_page_add_to_album_success": "{album}に{added}枚写真を追加しました",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "まだアップロードされてない項目はアーカイブできません",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "タイムライン構築中",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "まだアップロードされてない項目はお気に入り登録できません",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "はじめてアプリを使う場合、タイムラインに写真を表示するためにアルバムを選択してください",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
"image_viewer_page_state_provider_download_error": "ダウンロード失敗",
"image_viewer_page_state_provider_download_success": "ダウンロード成功",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
"partner_page_stop_sharing_title": "Stop sharing your photos?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "無視して続行",
"permission_onboarding_get_started": "はじめる",
"permission_onboarding_go_to_settings": "システム設定",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immichは写真へのアクセス許可が必要です",
"profile_drawer_app_logs": "ログ",
"profile_drawer_client_server_up_to_date": "すべて最新です",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "設定",
"profile_drawer_sign_out": "サインアウト",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "アルバム作成に失敗",
"select_user_for_sharing_page_share_suggestions": "ユーザ一覧",
"server_info_box_app_version": "アプリVer.",
+ "server_info_box_server_url": " サーバのURL",
"server_info_box_server_version": "サーバーVer.",
"setting_image_viewer_help": "写真をタップするとサムネイル・中画質(要設定)・オリジナル(要設定)の順に読み込みます",
"setting_image_viewer_original_subtitle": "オリジナルの画像を表示したい時にオンにしてください(最大画質で表示されるのでモバイルデータとストレージの消費量が増えます)。",
@@ -300,20 +315,29 @@
"share_add_photos": "写真を追加",
"share_add_title": "タイトルを追加",
"share_create_album": "アルバムを作成",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "準備中",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
+ "shared_link_app_bar_title": "共有リンク",
+ "shared_link_create_app_bar_title": "共有リンクを作る",
"shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
+ "shared_link_create_submit_button": "リンクを作る",
"shared_link_edit_allow_download": "Allow public user to download",
"shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
+ "shared_link_edit_app_bar_title": " リンクを編集する",
"shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
+ "shared_link_edit_description": " デスクリプション ",
"shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": " パスワード",
+ "shared_link_edit_password_hint": "共有パスワードを入力する",
+ "shared_link_edit_show_meta": " メタデータを見る",
+ "shared_link_edit_submit_button": "リンクをアップデートする",
+ "shared_link_empty": "共有リンクはありません ",
"shared_link_manage_links": "Manage Shared links",
"share_done": "Done",
"share_invite": "アルバムに招待",
diff --git a/mobile/assets/i18n/ko-KR.json b/mobile/assets/i18n/ko-KR.json
index d2786e35c..4fa57d983 100644
--- a/mobile/assets/i18n/ko-KR.json
+++ b/mobile/assets/i18n/ko-KR.json
@@ -1,21 +1,21 @@
{
"add_to_album_bottom_sheet_added": "{album}에 추가",
"add_to_album_bottom_sheet_already_exists": "{album}에 이미 포함되어 있습니다",
- "advanced_settings_prefer_remote_subtitle": "Some devices are painfully slow to load thumbnails from assets on the device. Activate this setting to load remote images instead.",
- "advanced_settings_prefer_remote_title": "Prefer remote images",
- "advanced_settings_self_signed_ssl_subtitle": "Skips SSL certificate verification for the server endpoint. Required for self-signed certificates.",
- "advanced_settings_self_signed_ssl_title": "Allow self-signed SSL certificates",
- "advanced_settings_tile_subtitle": "Advanced user's settings",
- "advanced_settings_tile_title": "Advanced",
- "advanced_settings_troubleshooting_subtitle": "Enable additional features for troubleshooting",
- "advanced_settings_troubleshooting_title": "Troubleshooting",
+ "advanced_settings_prefer_remote_subtitle": "일부 디바이스에서는 디바이스에 있는 미디어의 썸네일을 로드하는 속도가 매우 느립니다. 대신 원격 이미지를 로드하려면 이 설정을 활성화하세요",
+ "advanced_settings_prefer_remote_title": "원격 이미지 선호",
+ "advanced_settings_self_signed_ssl_subtitle": "서버 엔드포인트에 대한 SSL 인증서 확인을 건너뜁니다. 자체 서명 인증서에 필요합니다",
+ "advanced_settings_self_signed_ssl_title": "자체 서명된 SSL 인증서 허용",
+ "advanced_settings_tile_subtitle": "고급 사용자 설정",
+ "advanced_settings_tile_title": "고급",
+ "advanced_settings_troubleshooting_subtitle": "문제 해결을 위한 추가 기능 사용",
+ "advanced_settings_troubleshooting_title": "문제 해결",
"album_info_card_backup_album_excluded": "제외됨",
"album_info_card_backup_album_included": "포함됨",
"album_thumbnail_card_item": "1개 항목",
"album_thumbnail_card_items": "{}개 항목",
"album_thumbnail_card_shared": " · 공유",
- "album_thumbnail_owned": "Owned",
- "album_thumbnail_shared_by": "Shared by {}",
+ "album_thumbnail_owned": "소유",
+ "album_thumbnail_shared_by": "공유자 {}",
"album_viewer_appbar_share_delete": "앨범 삭제",
"album_viewer_appbar_share_err_delete": "앨범 삭제 실패",
"album_viewer_appbar_share_err_leave": "앨범에서 나가지 못했습니다",
@@ -23,14 +23,18 @@
"album_viewer_appbar_share_err_title": "앨범 제목 변경 실패",
"album_viewer_appbar_share_leave": "앨범 나가기",
"album_viewer_appbar_share_remove": "앨범에서 제거",
+ "album_viewer_appbar_share_to": "공유 대상",
"album_viewer_page_share_add_users": "사용자 추가",
- "all_people_page_title": "People",
- "all_videos_page_title": "Videos",
- "archive_page_no_archived_assets": "No archived assets found",
- "archive_page_title": "Archive ({})",
+ "all_people_page_title": "사람",
+ "all_videos_page_title": "동영상",
+ "app_bar_signout_dialog_content": "정말 로그아웃하시겠습니까?",
+ "app_bar_signout_dialog_ok": "네",
+ "app_bar_signout_dialog_title": "로그 아웃",
+ "archive_page_no_archived_assets": "보관된 미디어를 찾을 수 없습니다",
+ "archive_page_title": "보관 ({})",
"asset_list_layout_settings_dynamic_layout_title": "다이나믹 레이아웃",
- "asset_list_layout_settings_group_automatically": "Automatic",
- "asset_list_layout_settings_group_by": "다음으로 그룹화",
+ "asset_list_layout_settings_group_automatically": "자동",
+ "asset_list_layout_settings_group_by": "다음으로 미디어 그룹화",
"asset_list_layout_settings_group_by_month": "월",
"asset_list_layout_settings_group_by_month_day": "월 + 일",
"asset_list_settings_subtitle": "사진 배열 레이아웃 설정",
@@ -50,9 +54,9 @@
"backup_background_service_in_progress_notification": "미디어파일 백업 중...",
"backup_background_service_upload_failure_notification": "{} 업로드 실패",
"backup_controller_page_albums": "백업대상",
- "backup_controller_page_background_app_refresh_disabled_content": "Enable background app refresh in Settings > General > Background App Refresh in order to use background backup.",
- "backup_controller_page_background_app_refresh_disabled_title": "Background app refresh disabled",
- "backup_controller_page_background_app_refresh_enable_button_text": "Go to settings",
+ "backup_controller_page_background_app_refresh_disabled_content": "백그라운드 백업을 사용하려면 설정 > 일반 > 백그라운드 앱 새로 고침에서 백그라운드 앱 새로 고침을 활성화합니다",
+ "backup_controller_page_background_app_refresh_disabled_title": "백그라운드 앱 새로 고침 비활성화",
+ "backup_controller_page_background_app_refresh_enable_button_text": "설정으로 이동",
"backup_controller_page_background_battery_info_link": "사용 가이드",
"backup_controller_page_background_battery_info_message": "최상의 백업 환경을 위해 Immich 앱의 백그라운드 활동을 제한하는 배터리 최적화기능을 꺼주세요.\n\n휴대폰마다 설정방법이 다르므로 제조업체별로 설정방법을 확인하세요.",
"backup_controller_page_background_battery_info_ok": "확인",
@@ -94,11 +98,11 @@
"backup_controller_page_uploading_file_info": "파일 정보 업로드 중",
"backup_err_only_album": "유일한 앨범은 제거할 수 없습니다",
"backup_info_card_assets": "미디어",
- "backup_manual_cancelled": "Cancelled",
- "backup_manual_failed": "Failed",
- "backup_manual_in_progress": "Upload already in progress. Try after sometime",
- "backup_manual_success": "Success",
- "backup_manual_title": "Upload status",
+ "backup_manual_cancelled": "취소됨",
+ "backup_manual_failed": "실패",
+ "backup_manual_in_progress": "업로드가 이미 진행 중입니다. 잠시 후 시도하세요",
+ "backup_manual_success": "성공",
+ "backup_manual_title": "업로드 상태",
"cache_settings_album_thumbnails": "라이브러리 페이지 썸네일 ({} 미디어)",
"cache_settings_clear_cache_button": "캐시 지우기",
"cache_settings_clear_cache_button_title": "앱의 캐시를 지웁니다. 이 작업은 캐시가 다시 빌드될 때까지 앱의 성능에 상당한 영향을 미칩니다.",
@@ -111,36 +115,38 @@
"cache_settings_statistics_title": "캐시 사용률",
"cache_settings_subtitle": "Immich 앱의 캐싱 동작 제어",
"cache_settings_thumbnail_size": "썸네일 캐시 크기 ({} 미디어)",
+ "cache_settings_tile_subtitle": "로컬 저장소 동작 제어",
+ "cache_settings_tile_title": "로컬 저장소",
"cache_settings_title": "캐시 설정",
"change_password_form_confirm_password": "비밀번호 확인",
- "change_password_form_description": "{firstName} {lastName} 님, 안녕하세요.\n\n시스템에 처음 로그인했거나 비밀번호 변경 요청이 있었습니다. 아래에 새 비밀번호를 입력하세요.",
+ "change_password_form_description": "{name} 님, 안녕하세요.\n\n시스템에 처음 로그인했거나 비밀번호 변경 요청이 있었습니다. 아래에 새 비밀번호를 입력하세요.",
"change_password_form_new_password": "새 비밀번호",
"change_password_form_password_mismatch": "비밀번호가 일치하지 않습니다",
"change_password_form_reenter_new_password": "새 비밀번호 재입력",
"common_add_to_album": "앨범에 추가",
"common_change_password": "비밀번호 변경",
"common_create_new_album": "새 앨범 만들기",
- "common_server_error": "Please check your network connection, make sure the server is reachable and app/server versions are compatible.",
+ "common_server_error": "네트워크 연결을 확인하고 서버에 연결할 수 있는지, 앱/서버 버전이 호환되는지 확인하세요",
"common_shared": "공유됨",
"control_bottom_app_bar_add_to_album": "앨범에 추가",
"control_bottom_app_bar_album_info": "{} 항목",
"control_bottom_app_bar_album_info_shared": "{} 항목 · 공유됨",
- "control_bottom_app_bar_archive": "Archive",
+ "control_bottom_app_bar_archive": "보관",
"control_bottom_app_bar_create_new_album": "앨범 생성",
"control_bottom_app_bar_delete": "삭제",
"control_bottom_app_bar_favorite": "즐겨찾기",
"control_bottom_app_bar_share": "공유",
- "control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
- "control_bottom_app_bar_unarchive": "Unarchive",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_share_to": "공유 대상",
+ "control_bottom_app_bar_stack": "스택",
+ "control_bottom_app_bar_unarchive": "보관 해제",
+ "control_bottom_app_bar_upload": "업로드",
"create_album_page_untitled": "제목없음",
"create_shared_album_page_create": "만들기",
"create_shared_album_page_share": "공유",
"create_shared_album_page_share_add_assets": "사진 추가",
"create_shared_album_page_share_select_photos": "사진 선택",
- "curated_location_page_title": "Places",
- "curated_object_page_title": "Things",
+ "curated_location_page_title": "장소",
+ "curated_object_page_title": "사물",
"daily_title_text_date": "E, M월 d일",
"daily_title_text_date_year": "E, M월 d일, yyyy",
"date_format": "yyyy년 M월 d일, EEEE • a h:mm",
@@ -148,10 +154,10 @@
"delete_dialog_cancel": "취소",
"delete_dialog_ok": "삭제",
"delete_dialog_title": "영구적으로 삭제",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
- "description_input_hint_text": "Add description...",
- "description_input_submit_error": "Error updating description, check the log for more details",
+ "delete_shared_link_dialog_content": "이 공유 링크를 삭제하시겠습니까?",
+ "delete_shared_link_dialog_title": "공유 링크 삭제",
+ "description_input_hint_text": "설명 추가",
+ "description_input_submit_error": "설명 업데이트 오류, 자세한 내용은 로그를 확인하세요",
"exif_bottom_sheet_description": "설명 추가...",
"exif_bottom_sheet_details": "상세정보",
"exif_bottom_sheet_location": "위치",
@@ -159,31 +165,36 @@
"experimental_settings_new_asset_list_title": "실험적 사진 그리드 적용",
"experimental_settings_subtitle": "문제시 책임지지 않습니다!",
"experimental_settings_title": "실험적기능",
- "favorites_page_no_favorites": "No favorite assets found",
+ "favorites_page_no_favorites": "즐겨찾기된 미디어를 찾을 수 없습니다",
"favorites_page_title": "즐겨찾기",
"home_page_add_to_album_conflicts": "{album} 앨범에 {added} 미디어를 추가했습니다. {failed} 이미 앨범에 있는 항목입니다.",
- "home_page_add_to_album_err_local": "앨범에 미디어파일을 추가할 수 없어, 건너뜁니다.",
+ "home_page_add_to_album_err_local": "아직 앨범에 로컬 미디어를 추가할 수 없으므로 건너뜁니다",
"home_page_add_to_album_success": "{album} 앨범에 {added} 미디어를 추가했습니다. ",
- "home_page_archive_err_local": "Can not archive local assets yet, skipping",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
+ "home_page_archive_err_local": "아직 로컬 미디어를 보관할 수 없습니다",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "타임라인 생성",
- "home_page_favorite_err_local": "미디어파일을 즐겨찾기에 추가할 수 없어, 건너뜁니다.",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
+ "home_page_favorite_err_local": "아직 로컬 미디어를 즐겨찾기에 추가할 수 없으므로 건너뜁니다",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "앱을 처음 사용하는 경우 타임라인이 앨범의 사진과 비디오를 채울 수 있도록 백업대상 앨범을 선택해야 합니다.",
- "home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
+ "home_page_upload_err_limit": "한번에 최대 30개의 미디어만 업로드할 수 있습니다",
"image_viewer_page_state_provider_download_error": "다운로드 에러",
"image_viewer_page_state_provider_download_success": "다운로드 완료",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "공유 오류",
"library_page_albums": "앨범",
- "library_page_archive": "Archive",
- "library_page_device_albums": "Albums on Device",
+ "library_page_archive": "보관",
+ "library_page_device_albums": "장치의 앨범",
"library_page_favorites": "즐겨찾기",
"library_page_new_album": "새 앨범",
"library_page_sharing": "공유",
"library_page_sort_created": "최근생성일",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "마지막 수정",
+ "library_page_sort_most_recent_photo": "가장 최근 사진",
"library_page_sort_title": "앨범 제목",
- "login_disabled": "Login has been disabled",
- "login_form_api_exception": "API exception. Please check the server URL and try again.",
+ "login_disabled": "로그인이 비활성화되었습니다",
+ "login_form_api_exception": "API 예외입니다. 서버 URL을 확인한 후 다시 시도하세요",
"login_form_button_text": "로그인",
"login_form_email_hint": "youremail@email.com",
"login_form_endpoint_hint": "https://your-server-ip:port/api",
@@ -196,86 +207,90 @@
"login_form_failed_get_oauth_server_config": "OAuth 로그인 오류, 서버 URL을 확인해주세요",
"login_form_failed_get_oauth_server_disable": "이 서버에서는 OAuth 기능을 사용할 수 없습니다.",
"login_form_failed_login": "로그인 오류, 서버 URL, 이메일 및 비밀번호를 확인하세요",
- "login_form_handshake_exception": "There was an Handshake Exception with the server. Enable self-signed certificate support in the settings if you are using a self-signed certificate.",
+ "login_form_handshake_exception": "서버에 핸드셰이크 예외가 발생했습니다. 자체 서명 인증서를 사용하는 경우 설정에서 자체 서명 인증서 지원을 사용 설정합니다",
"login_form_label_email": "이메일",
"login_form_label_password": "비밀번호",
- "login_form_next_button": "Next",
+ "login_form_next_button": "다음",
"login_form_password_hint": "비밀번호",
"login_form_save_login": "로그인상태 유지",
- "login_form_server_empty": "Enter a server URL.",
- "login_form_server_error": "Could not connect to server.",
- "login_password_changed_error": "There was an error updating your password",
- "login_password_changed_success": "Password updated successfully",
- "map_cannot_get_user_location": "Cannot get user's location",
- "map_location_dialog_cancel": "Cancel",
- "map_location_dialog_yes": "Yes",
- "map_location_service_disabled_content": "Location service needs to be enabled to display assets from your current location. Do you want to enable it now?",
- "map_location_service_disabled_title": "Location Service disabled",
- "map_no_assets_in_bounds": "No photos in this area",
- "map_no_location_permission_content": "Location permission is needed to display assets from your current location. Do you want to allow it now?",
- "map_no_location_permission_title": "Location Permission denied",
- "map_settings_dark_mode": "Dark mode",
- "map_settings_dialog_cancel": "Cancel",
- "map_settings_dialog_save": "Save",
- "map_settings_dialog_title": "Map Settings",
- "map_settings_include_show_archived": "Include Archived",
- "map_settings_only_relative_range": "Date range",
- "map_settings_only_show_favorites": "Show Favorite Only",
- "map_zoom_to_see_photos": "Zoom out to see photos",
+ "login_form_server_empty": "서버 URL 입력",
+ "login_form_server_error": "서버에 연결할 수 없습니다",
+ "login_password_changed_error": "비밀번호를 업데이트하는 동안 오류가 발생했습니다",
+ "login_password_changed_success": "비밀번호 업데이트 성공",
+ "map_cannot_get_user_location": "사용자 위치를 가져올 수 없습니다.",
+ "map_location_dialog_cancel": "아니오",
+ "map_location_dialog_yes": "예",
+ "map_location_service_disabled_content": "현재 위치의 미디어를 표시하려면 위치 서비스를 활성화해야 합니다. 지금 활성화하시겠습니까?",
+ "map_location_service_disabled_title": "위치 서비스 비활성화",
+ "map_no_assets_in_bounds": "이 영역에 사진이 없습니다",
+ "map_no_location_permission_content": "현재 위치의 미디어를 표시하려면 위치 권한이 필요합니다. 지금 허용하시겠습니까?",
+ "map_no_location_permission_title": "위치 권한 거부됨",
+ "map_settings_dark_mode": "다크 모드",
+ "map_settings_dialog_cancel": "취소",
+ "map_settings_dialog_save": "저장",
+ "map_settings_dialog_title": "지도 설정",
+ "map_settings_include_show_archived": "아카이브 포함",
+ "map_settings_only_relative_range": "날짜 범위",
+ "map_settings_only_show_favorites": "즐겨찾기에만 표시",
+ "map_zoom_to_see_photos": "축소하여 사진 보기",
"monthly_title_text_date_format": "y년 M월",
- "motion_photos_page_title": "Motion Photos",
+ "motion_photos_page_title": "모션 사진",
"notification_permission_dialog_cancel": "취소",
"notification_permission_dialog_content": "알림을 활성화하려면 설정으로 이동하여 허용을 선택해주세요.",
"notification_permission_dialog_settings": "설정",
"notification_permission_list_tile_content": "알림 활성화 권한허용",
"notification_permission_list_tile_enable_button": "알림 활성화",
"notification_permission_list_tile_title": "알림 권한",
- "partner_page_add_partner": "Add partner",
- "partner_page_empty_message": "Your photos are not yet shared with any partner.",
- "partner_page_no_more_users": "No more users to add",
- "partner_page_partner_add_failed": "Failed to add partner",
- "partner_page_select_partner": "Select partner",
- "partner_page_shared_to_title": "Shared to",
- "partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
- "partner_page_stop_sharing_title": "Stop sharing your photos?",
- "partner_page_title": "Partner",
- "permission_onboarding_continue_anyway": "Continue anyway",
- "permission_onboarding_get_started": "Get started",
- "permission_onboarding_go_to_settings": "Go to settings",
- "permission_onboarding_grant_permission": "Grant permission",
- "permission_onboarding_log_out": "Log out",
- "permission_onboarding_permission_denied": "Permission denied. To use Immich, grant photo and video permissions in Settings.",
- "permission_onboarding_permission_granted": "Permission granted! You are all set.",
- "permission_onboarding_permission_limited": "Permission limited. To let Immich backup and manage your entire gallery collection, grant photo and video permissions in Settings.",
- "permission_onboarding_request": "Immich requires permission to view your photos and videos.",
+ "partner_page_add_partner": "파트너 추가",
+ "partner_page_empty_message": "사진이 아직 어떤 파트너와도 공유되지 않았습니다",
+ "partner_page_no_more_users": "더 이상 추가할 사용자 없음",
+ "partner_page_partner_add_failed": "파트너 추가에 실패했습니다",
+ "partner_page_select_partner": "파트너 선택",
+ "partner_page_shared_to_title": "공유 대상",
+ "partner_page_stop_sharing_content": "더 이상 {}에서 사진에 액세스할 수 없습니다.",
+ "partner_page_stop_sharing_title": "사진 공유를 중단하시겠습니까?",
+ "partner_page_title": "파트너",
+ "permission_onboarding_back": "Back",
+ "permission_onboarding_continue_anyway": "어쨌든 계속하기",
+ "permission_onboarding_get_started": "시작하기",
+ "permission_onboarding_go_to_settings": "설정으로 이동",
+ "permission_onboarding_grant_permission": "권한 부여",
+ "permission_onboarding_log_out": "로그 아웃",
+ "permission_onboarding_permission_denied": "권한이 거부되었습니다. Immich를 사용하려면 설정에서 사진 및 동영상 권한을 부여하세요",
+ "permission_onboarding_permission_granted": "승인되었습니다! 모든 준비가 완료되었습니다",
+ "permission_onboarding_permission_limited": "권한 제한. Immich가 전체 갤러리 컬렉션을 백업하고 관리하도록 하려면 설정에서 사진 및 동영상 권한을 부여하세요",
+ "permission_onboarding_request": "Immich는 사진과 동영상을 볼 수 있는 권한을 요구합니다",
"profile_drawer_app_logs": "로그",
"profile_drawer_client_server_up_to_date": "클라이언트와 서버가 최신 상태입니다",
+ "profile_drawer_documentation": "문서",
+ "profile_drawer_github": "깃허브",
"profile_drawer_settings": "설정",
"profile_drawer_sign_out": "로그아웃",
- "profile_drawer_trash": "Trash",
- "recently_added_page_title": "Recently Added",
+ "profile_drawer_trash": "휴지통",
+ "recently_added_page_title": "최근 추가",
"search_bar_hint": "사진 검색",
- "search_page_categories": "Categories",
- "search_page_favorites": "Favorites",
- "search_page_motion_photos": "Motion Photos",
+ "search_page_categories": "카테고리",
+ "search_page_favorites": "즐겨찾기",
+ "search_page_motion_photos": "모션 사진",
"search_page_no_objects": "발견된 사물이\n없습니다",
"search_page_no_places": "발견된 장소가\n없습니다",
- "search_page_people": "People",
+ "search_page_people": "사람",
"search_page_places": "장소",
- "search_page_recently_added": "Recently added",
- "search_page_screenshots": "Screenshots",
- "search_page_selfies": "Selfies",
+ "search_page_recently_added": "최근 추가",
+ "search_page_screenshots": "스크린샷",
+ "search_page_selfies": "셀카",
"search_page_things": "사물",
- "search_page_videos": "Videos",
- "search_page_view_all_button": "View all",
- "search_page_your_activity": "Your activity",
+ "search_page_videos": "동영상",
+ "search_page_view_all_button": "모두 보기",
+ "search_page_your_activity": "내 활동",
"search_result_page_new_search_hint": "새 검색",
- "search_suggestion_list_smart_search_hint_1": "Smart search is enabled by default, to search for metadata use the syntax ",
+ "search_suggestion_list_smart_search_hint_1": "스마트 검색은 기본적으로 활성화되어 있으며, 메타데이터를 검색하려면 다음 구문을 사용합니다",
"search_suggestion_list_smart_search_hint_2": "m:your-search-term",
"select_additional_user_for_sharing_page_suggestions": "초대 가능한 사용자 제안",
"select_user_for_sharing_page_err_album": "앨범 생성 실패",
"select_user_for_sharing_page_share_suggestions": "초대 가능한 사용자 제안",
"server_info_box_app_version": "앱 버전",
+ "server_info_box_server_url": "서버 URL",
"server_info_box_server_version": "서버 버전",
"setting_image_viewer_help": "상세뷰어는 먼저 작은 썸네일을 불러온 다음 중간크기 미리보기를 불러오고(활성화된 경우) 마지막으로 원본을 불러옵니다(활성화된 경우).",
"setting_image_viewer_original_subtitle": "원본 해상도 이미지(고화질)를 로드하려면 활성화합니다. 데이터 사용량을 줄이려면 비활성화합니다.",
@@ -300,28 +315,37 @@
"share_add_photos": "사진 추가",
"share_add_title": "새 앨범제목",
"share_create_album": "앨범 만들기",
+ "shared_album_activities_input_disable": "댓글이 비활성화되었습니다.",
+ "shared_album_activities_input_hint": "말하기",
+ "shared_album_activity_remove_content": "이 활동을 삭제하시겠습니까?",
+ "shared_album_activity_remove_title": "활동 삭제",
+ "shared_album_activity_setting_subtitle": "다른 사람이 응답하도록 허용",
+ "shared_album_activity_setting_title": "댓글 및 좋아요",
"share_dialog_preparing": "준비중...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "공유 링크",
+ "shared_link_create_app_bar_title": "공유할 링크 만들기",
+ "shared_link_create_info": "링크를 가진 모든 사람이 선택한 사진을 볼 수 있도록 합니다",
+ "shared_link_create_submit_button": "링크 만들기",
+ "shared_link_edit_allow_download": "공용 사용자의 다운로드 허용",
+ "shared_link_edit_allow_upload": "공용 사용자의 업로드 허용",
+ "shared_link_edit_app_bar_title": "링크 수정",
+ "shared_link_edit_change_expiry": "만료 시간 변경",
+ "shared_link_edit_description": "설명",
+ "shared_link_edit_description_hint": "공유 설명 입력",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "비밀번호",
+ "shared_link_edit_password_hint": "공유 비밀번호 입력",
+ "shared_link_edit_show_meta": "메타데이터 표시",
+ "shared_link_edit_submit_button": "링크 업데이트",
+ "shared_link_empty": "공유 링크가 없습니다",
+ "shared_link_manage_links": "공유 링크 관리",
+ "share_done": "완료",
"share_invite": "앨범에 초대",
"sharing_page_album": "공유앨범",
"sharing_page_description": "공유앨범을 만들어 다른 사용자들과 사진 및 비디오를 공유합니다.",
"sharing_page_empty_list": "공유앨범 없음",
"sharing_silver_appbar_create_shared_album": "공유앨범 만들기",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "공유 링크",
"sharing_silver_appbar_share_partner": "파트너와 공유",
"tab_controller_nav_library": "라이브러리",
"tab_controller_nav_photos": "사진",
@@ -337,30 +361,30 @@
"theme_setting_theme_title": "테마",
"theme_setting_three_stage_loading_subtitle": "이 기능은 로딩 성능을 향상시킬 수 있지만 훨씬 더 많은 데이터를 사용합니다.",
"theme_setting_three_stage_loading_title": "3단계 로딩 활성화",
- "translated_text_options": "Options",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
- "trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
- "upload_dialog_cancel": "Cancel",
- "upload_dialog_info": "Do you want to backup the selected Asset(s) to the server?",
- "upload_dialog_ok": "Upload",
- "upload_dialog_title": "Upload Asset",
+ "translated_text_options": "옵션",
+ "trash_page_delete": "삭제",
+ "trash_page_delete_all": "모두 삭제",
+ "trash_page_empty_trash_btn": "휴지통 비우기",
+ "trash_page_empty_trash_dialog_content": "휴지통에 버린 미디어를 비우고 싶으신가요? 이 항목들은 Immich에서 영구적으로 삭제됩니다",
+ "trash_page_empty_trash_dialog_ok": "확인",
+ "trash_page_info": "휴지통에 버린 항목은 {}일 후에 영구 삭제됩니다",
+ "trash_page_no_assets": "휴지통에 버려진 미디어 없음",
+ "trash_page_restore": "복원",
+ "trash_page_restore_all": "모두 복원",
+ "trash_page_select_assets_btn": "미디어 선택",
+ "trash_page_select_btn": "선택",
+ "trash_page_title": "휴지통 ({})",
+ "upload_dialog_cancel": "취소",
+ "upload_dialog_info": "선택한 미디어를 서버에 백업하시겠습니까?",
+ "upload_dialog_ok": "업로드",
+ "upload_dialog_title": "미디어 업로드",
"version_announcement_overlay_ack": "승인",
"version_announcement_overlay_release_notes": "릴리스 정보",
"version_announcement_overlay_text_1": "안녕하세요!",
"version_announcement_overlay_text_2": "앱에 새로운 업데이트가 있습니다!",
"version_announcement_overlay_text_3": "특히 WatchTower 또는 서버 응용 프로그램 자동 업데이트를 처리하는 메커니즘을 사용하는 경우 잘못된 구성을 방지하기 위해 docker-compose 및 .env 설정이 최신 상태인지 확인하세요.",
"version_announcement_overlay_title": "새 서버 버전 사용 가능 \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "스택에서 제거",
+ "viewer_stack_use_as_main_asset": "메인 미디어로 사용",
+ "viewer_unstack": "스택 해제"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/lv-LV.json b/mobile/assets/i18n/lv-LV.json
index 966bc8db8..875044c29 100644
--- a/mobile/assets/i18n/lv-LV.json
+++ b/mobile/assets/i18n/lv-LV.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Neizdevās mainīt albuma nosaukumu",
"album_viewer_appbar_share_leave": "Pamest albumu",
"album_viewer_appbar_share_remove": "Noņemt no albuma",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Pievienot lietotājus",
"all_people_page_title": "Cilvēki",
"all_videos_page_title": "Videoklipi",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "Nav atrasts neviens arhivēts aktīvs",
"archive_page_title": "Arhīvs ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dinamiskais izkārtojums",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Kešatmiņas lietojums",
"cache_settings_subtitle": "Kontrolēt Immich mobilās lietotnes kešdarbi",
"cache_settings_thumbnail_size": "Sīktēlu keša lielums ({} aktīvi)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Kešdarbes iestatījumi",
"change_password_form_confirm_password": "Apstiprināt Paroli",
- "change_password_form_description": "Sveiki {FirstName} {LastName},\n\nŠī ir pirmā reize, kad pierakstāties sistēmā, vai arī ir iesniegts pieprasījums mainīt paroli. Lūdzu, zemāk ievadiet jauno paroli.",
+ "change_password_form_description": "Sveiki {name},\n\nŠī ir pirmā reize, kad pierakstāties sistēmā, vai arī ir iesniegts pieprasījums mainīt paroli. Lūdzu, zemāk ievadiet jauno paroli.",
"change_password_form_new_password": "Jauna Parole",
"change_password_form_password_mismatch": "Paroles nesakrīt",
"change_password_form_reenter_new_password": "Atkārtoti ievadīt jaunu paroli",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Pievienoja {added} aktīvus albumam {album}. {failed} aktīvi jau ir albumā.",
"home_page_add_to_album_err_local": "Albumiem vēl nevar pievienot lokālos aktīvus, notiek izlaišana",
"home_page_add_to_album_success": "Pievienoja {added} aktīvus albumam {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Vēl nevar arhivēt lokālos aktīvus, notiek izlaišana",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Tiek izveidota laika skala",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Vēl nevar pievienot izlaisei vietējos aktīvus, notiek izlaišana",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Ja šī ir pirmā reize, kad izmantojat aplikāciju, lūdzu, izvēlieties dublējuma albumu(s), lai laika skala varētu aizpildīt fotoattēlus un videoklipus albumā(os).",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
"image_viewer_page_state_provider_download_error": "Lejupielādes Kļūda",
"image_viewer_page_state_provider_download_success": "Lejupielāde Izdevās",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} vairs nevarēs piekļūt jūsu fotoattēliem.",
"partner_page_stop_sharing_title": "Beigt kopīgot jūsu fotogrāfijas?",
"partner_page_title": "Partneris",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Tomēr turpināt",
"permission_onboarding_get_started": "Darba sākšana",
"permission_onboarding_go_to_settings": "Doties uz iestatījumiem",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich nepieciešama atļauja skatīt jūsu fotoattēlus un videoklipus.",
"profile_drawer_app_logs": "Žurnāli",
"profile_drawer_client_server_up_to_date": "Klients un serveris ir atjaunināti",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Iestatījumi",
"profile_drawer_sign_out": "Izrakstīties",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Neizdevās izveidot albumu",
"select_user_for_sharing_page_share_suggestions": "Ieteikumi",
"server_info_box_app_version": "Aplikācijas Versija",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Servera Versija",
"setting_image_viewer_help": "Detaļu skatītājs vispirms ielādē mazo sīktēlu, pēc tam ielādē vidēja lieluma priekšskatījumu (ja iespējots), visbeidzot ielādē oriģinālu (ja iespējots).",
"setting_image_viewer_original_subtitle": "Iespējojiet sākotnējā pilnas izšķirtspējas attēla (liels!) ielādi. Atspējot lai samazinātu datu lietojumu (gan tīklā, gan ierīces kešatmiņā).",
@@ -300,6 +315,12 @@
"share_add_photos": "Pievienot fotoattēlus",
"share_add_title": "Pievienot virsrakstu",
"share_create_album": "Izveidot albumu",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Notiek sagatavošana...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/mn.json b/mobile/assets/i18n/mn.json
index b20d35a7d..dfbb54d89 100644
--- a/mobile/assets/i18n/mn.json
+++ b/mobile/assets/i18n/mn.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Failed to change album title",
"album_viewer_appbar_share_leave": "Leave album",
"album_viewer_appbar_share_remove": "Remove from album",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Add users",
"all_people_page_title": "People",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "No archived assets found",
"archive_page_title": "Archive ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamic layout",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Cache usage",
"cache_settings_subtitle": "Control the caching behaviour of the Immich mobile application",
"cache_settings_thumbnail_size": "Thumbnail cache size ({} assets)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Caching Settings",
"change_password_form_confirm_password": "Confirm Password",
- "change_password_form_description": "Hi {firstName} {lastName},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
+ "change_password_form_description": "Hi {name},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
"change_password_form_new_password": "New Password",
"change_password_form_password_mismatch": "Passwords do not match",
"change_password_form_reenter_new_password": "Re-enter New Password",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Added {added} assets to album {album}. {failed} assets are already in the album.",
"home_page_add_to_album_err_local": "Can not add local assets to albums yet, skipping",
"home_page_add_to_album_success": "Added {added} assets to album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Can not archive local assets yet, skipping",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Building the timeline",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Can not favorite local assets yet, skipping",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "If this is your first time using the app, please make sure to choose a backup album(s) so that the timeline can populate photos and videos in the album(s).",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
"image_viewer_page_state_provider_download_error": "Download Error",
"image_viewer_page_state_provider_download_success": "Download Success",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
"partner_page_stop_sharing_title": "Stop sharing your photos?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continue anyway",
"permission_onboarding_get_started": "Get started",
"permission_onboarding_go_to_settings": "Go to settings",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich requires permission to view your photos and videos.",
"profile_drawer_app_logs": "Logs",
"profile_drawer_client_server_up_to_date": "Client and Server are up-to-date",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Settings",
"profile_drawer_sign_out": "Sign Out",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Failed to create album",
"select_user_for_sharing_page_share_suggestions": "Suggestions",
"server_info_box_app_version": "App Version",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Server Version",
"setting_image_viewer_help": "The detail viewer loads the small thumbnail first, then loads the medium-size preview (if enabled), finally loads the original (if enabled).",
"setting_image_viewer_original_subtitle": "Enable to load the original full-resolution image (large!). Disable to reduce data usage (both network and on device cache).",
@@ -300,6 +315,12 @@
"share_add_photos": "Add photos",
"share_add_title": "Add a title",
"share_create_album": "Create album",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Preparing...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/nb-NO.json b/mobile/assets/i18n/nb-NO.json
index 18cf4475b..cd2e9be27 100644
--- a/mobile/assets/i18n/nb-NO.json
+++ b/mobile/assets/i18n/nb-NO.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Feilet ved endring av albumtittel",
"album_viewer_appbar_share_leave": "Forlat album",
"album_viewer_appbar_share_remove": "Fjern fra album",
+ "album_viewer_appbar_share_to": "Del til",
"album_viewer_page_share_add_users": "Legg til brukere",
"all_people_page_title": "Folk",
"all_videos_page_title": "Videoer",
+ "app_bar_signout_dialog_content": "Er du sikker på at du vil logge ut?",
+ "app_bar_signout_dialog_ok": "Ja",
+ "app_bar_signout_dialog_title": "Logg ut",
"archive_page_no_archived_assets": "Ingen arkiverte objekter funnet",
"archive_page_title": "Arkiv ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamisk bildeorganisering",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Bufferbruk",
"cache_settings_subtitle": "Kontroller bufringsadferden til Immich-appen",
"cache_settings_thumbnail_size": "Størrelse på miniatyrbildebuffer ({} objekter)",
+ "cache_settings_tile_subtitle": "Kontroller lokal lagring",
+ "cache_settings_tile_title": "Lokal lagring",
"cache_settings_title": "Bufringsinnstillinger",
"change_password_form_confirm_password": "Bekreft passord",
- "change_password_form_description": "Hei {firstName} {lastName}!\n\nDette er enten første gang du logger på systemet, eller det er sendt en forespørsel om å endre passordet ditt. Vennligst skriv inn det nye passordet nedenfor.",
+ "change_password_form_description": "Hei {name}!\n\nDette er enten første gang du logger på systemet, eller det er sendt en forespørsel om å endre passordet ditt. Vennligst skriv inn det nye passordet nedenfor.",
"change_password_form_new_password": "Nytt passord",
"change_password_form_password_mismatch": "Passordene stemmer ikke",
"change_password_form_reenter_new_password": "Skriv nytt passord igjen",
@@ -130,10 +136,10 @@
"control_bottom_app_bar_delete": "Slett",
"control_bottom_app_bar_favorite": "Favoritt",
"control_bottom_app_bar_share": "Del",
- "control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_share_to": "Del til",
+ "control_bottom_app_bar_stack": "Stable",
"control_bottom_app_bar_unarchive": "Fjern fra arkiv",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Last opp",
"create_album_page_untitled": "Uten navn",
"create_shared_album_page_create": "Opprett",
"create_shared_album_page_share": "Del",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "Avbryt",
"delete_dialog_ok": "Slett",
"delete_dialog_title": "Slett permanent",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Er du sikker på at du vil slette denne delte linken?",
+ "delete_shared_link_dialog_title": "Slett delt link",
"description_input_hint_text": "Legg til beskrivelse ...",
"description_input_submit_error": "Feil ved oppdatering av beskrivelse, sjekk loggen for flere detaljer",
"exif_bottom_sheet_description": "Legg til beskrivelse ...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "Lagt til {added} objekter til album {album}. {failed} objekter er allerede i albumet.",
"home_page_add_to_album_err_local": "Kan ikke legge til lokale objekter til album enda, hopper over",
"home_page_add_to_album_success": "Lagt til {added} objekter til album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Kan ikke arkivere lokale objekter enda, hopper over",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Genererer tidslinjen",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Kan ikke sette favoritt på lokale objekter enda, hopper over",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Hvis dette er første gangen du benytter appen, velg et album (eller flere) for sikkerhetskopiering, slik at tidslinjen kan fylles med dine bilder og videoer.",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Maksimalt 30 objekter kan lastes opp om gangen, hopper over",
"image_viewer_page_state_provider_download_error": "Nedlasting feilet",
"image_viewer_page_state_provider_download_success": "Nedlasting vellykket",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Delingsfeil",
"library_page_albums": "Albumer",
"library_page_archive": "Arkiv",
"library_page_device_albums": "Albumer på enheten",
@@ -179,8 +190,8 @@
"library_page_new_album": "Nytt album",
"library_page_sharing": "Deling",
"library_page_sort_created": "Nylig opplastet",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Sist endret",
+ "library_page_sort_most_recent_photo": "Siste bilde",
"library_page_sort_title": "Albumtittel",
"login_disabled": "Innlogging har blitt deaktivert",
"login_form_api_exception": "API-feil. Sjekk URL-en til serveren og prøv igjen.",
@@ -218,7 +229,7 @@
"map_settings_dialog_cancel": "Avbryt",
"map_settings_dialog_save": "Lagre",
"map_settings_dialog_title": "Kartinnstillinger",
- "map_settings_include_show_archived": "Include Archived",
+ "map_settings_include_show_archived": "Inkluder arkiverte",
"map_settings_only_relative_range": "Datoområde",
"map_settings_only_show_favorites": "Vis kun favoritter",
"map_zoom_to_see_photos": "Zoom ut for å se bilder",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} vil ikke lenger ha tilgang til dine bilder.",
"partner_page_stop_sharing_title": "Stopp deling av bildene dine?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Fortsett uansett",
"permission_onboarding_get_started": "Kom i gang",
"permission_onboarding_go_to_settings": "Gå til innstillinger",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich trenger tilgang til å se dine bilder og videoer",
"profile_drawer_app_logs": "Logg",
"profile_drawer_client_server_up_to_date": "Klient og server er oppdatert",
+ "profile_drawer_documentation": "Dokumentasjon",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Innstillinger",
"profile_drawer_sign_out": "Logg ut",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Søppelbøtte",
"recently_added_page_title": "Nylig lagt til",
"search_bar_hint": "Søk i dine bilder",
"search_page_categories": "Kategorier",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Feilet ved oppretting av album",
"select_user_for_sharing_page_share_suggestions": "Forslag",
"server_info_box_app_version": "App-versjon",
+ "server_info_box_server_url": "Server-adresse",
"server_info_box_server_version": "Server-versjon",
"setting_image_viewer_help": "Detaljvisningen laster først miniatyrbildet, deretter forhåndsvisningsbildet (hvis aktivert), og til slutt originalen (hvis aktivert).",
"setting_image_viewer_original_subtitle": "Aktiver for å laste originalbildet i full oppløsning (stort!). Deaktiver for å spare databruk (både nettverksbruk og bufferdata på enheten).",
@@ -300,28 +315,37 @@
"share_add_photos": "Legg til bilder",
"share_add_title": "Legg til tittel",
"share_create_album": "Opprett album",
+ "shared_album_activities_input_disable": "Kommenterer er deaktivert",
+ "shared_album_activities_input_hint": "Si noe",
+ "shared_album_activity_remove_content": "Vil du slette denne aktiviteten?",
+ "shared_album_activity_remove_title": "Slett aktivitet",
+ "shared_album_activity_setting_subtitle": "La andre respondere",
+ "shared_album_activity_setting_title": "Kommentarer og likes",
"share_dialog_preparing": "Forbereder ...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Delte linker",
+ "shared_link_create_app_bar_title": "Opprett delelink",
+ "shared_link_create_info": "La alle med linken se de(t) valgte bilde(ne)",
+ "shared_link_create_submit_button": "Opprett link",
+ "shared_link_edit_allow_download": "Tillat offentlig bruker å laste ned",
+ "shared_link_edit_allow_upload": "Tillat offentlig bruker å laste opp",
+ "shared_link_edit_app_bar_title": "Endre link",
+ "shared_link_edit_change_expiry": "Endre utløpstid",
+ "shared_link_edit_description": "Beskrivelse",
+ "shared_link_edit_description_hint": "Endre delebeskrivelse",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Passord",
+ "shared_link_edit_password_hint": "Skriv inn dele-passord",
+ "shared_link_edit_show_meta": "Vis metadata",
+ "shared_link_edit_submit_button": "Oppdater link",
+ "shared_link_empty": "Du har ingen delte linker",
+ "shared_link_manage_links": "Håndter delte linker",
+ "share_done": "Ferdig",
"share_invite": "Inviter til album",
"sharing_page_album": "Delte album",
"sharing_page_description": "Lag delte albumer for å dele bilder og videoer med folk i nettverket ditt.",
"sharing_page_empty_list": "TOM LISTE",
"sharing_silver_appbar_create_shared_album": "Lag delt album",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Delte linker",
"sharing_silver_appbar_share_partner": "Del med partner",
"tab_controller_nav_library": "Bibliotek",
"tab_controller_nav_photos": "Bilder",
@@ -338,18 +362,18 @@
"theme_setting_three_stage_loading_subtitle": "Tre-trinns innlasting kan øke lasteytelsen, men forårsaker betydelig høyere nettverksbelastning",
"theme_setting_three_stage_loading_title": "Aktiver tre-trinns innlasting",
"translated_text_options": "Valg",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
+ "trash_page_delete": "Slett",
+ "trash_page_delete_all": "Slett alt",
+ "trash_page_empty_trash_btn": "Tøm søppelbøtte",
+ "trash_page_empty_trash_dialog_content": "Vil du tømme søppelbøtten? Objektene vil bli permanent fjernet fra Immich",
"trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
+ "trash_page_info": "Objekter i søppelbøtten blir permanent fjernet etter {} dager",
+ "trash_page_no_assets": "Ingen forkastede objekter",
+ "trash_page_restore": "Gjenopprett",
+ "trash_page_restore_all": "Gjenopprett alt",
+ "trash_page_select_assets_btn": "Velg objekter",
+ "trash_page_select_btn": "Velg",
+ "trash_page_title": "Søppelbøtte ({})",
"upload_dialog_cancel": "Avbryt",
"upload_dialog_info": "Vil du utføre backup av valgte objekt(er) til serveren?",
"upload_dialog_ok": "Last opp",
@@ -360,7 +384,7 @@
"version_announcement_overlay_text_2": "vennligst ta deg tid til å besøke ",
"version_announcement_overlay_text_3": " og verifiser at docker-compose og .env-oppsettet ditt er oppdatert for å forhindre en eventuell feilkonfigurasjon, spesielt hvis du benytter WatchTower eller en annen tjeneste som håndterer oppdatering av server-applikasjonen automatisk.",
"version_announcement_overlay_title": "Ny serverversjon tilgjengelig",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Fjern fra stabling",
+ "viewer_stack_use_as_main_asset": "Bruk som hovedobjekt",
+ "viewer_unstack": "avstable"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/nl-NL.json b/mobile/assets/i18n/nl-NL.json
index 724b650f9..eece8bd40 100644
--- a/mobile/assets/i18n/nl-NL.json
+++ b/mobile/assets/i18n/nl-NL.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Albumtitel wijzigen mislukt",
"album_viewer_appbar_share_leave": "Verlaat album",
"album_viewer_appbar_share_remove": "Verwijder uit album",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Gebruikers toevoegen",
"all_people_page_title": "Personen",
"all_videos_page_title": "Video's",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "Geen gearchiveerde items gevonden",
"archive_page_title": "Archief ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamische layout",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Cachegebruik",
"cache_settings_subtitle": "Beheer het cachegedrag van de Immich app",
"cache_settings_thumbnail_size": "Thumbnail-cachegrootte ({} items)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Cache-instellingen",
"change_password_form_confirm_password": "Bevestig wachtwoord",
- "change_password_form_description": "Hallo {firstName} {lastName},\n\nDit is ofwel de eerste keer dat je inlogt, of er is een verzoek gedaan om je wachtwoord te wijzigen. Vul hieronder een nieuw wachtwoord in.",
+ "change_password_form_description": "Hallo {name},\n\nDit is ofwel de eerste keer dat je inlogt, of er is een verzoek gedaan om je wachtwoord te wijzigen. Vul hieronder een nieuw wachtwoord in.",
"change_password_form_new_password": "Nieuw wachtwoord",
"change_password_form_password_mismatch": "Wachtwoorden komen niet overeen",
"change_password_form_reenter_new_password": "Vul het wachtwoord opnieuw in",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "{added} items toegevoegd aan album {album}. {failed} items staan al in het album.",
"home_page_add_to_album_err_local": "Lokale items kunnen nog niet aan albums worden toegevoegd, overslaan",
"home_page_add_to_album_success": "{added} items toegevoegd aan album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Lokale items kunnen nog niet gearchiveerd worden, overslaan",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Tijdlijn opbouwen",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Lokale items kunnen nog niet als favoriet worden aangemerkt, overslaan",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Als dit de eerste keer is dat je de app gebruikt, zorg er dan voor dat je een back-up album kiest, zodat de tijdlijn gevuld kan worden met foto's en video's uit het album.",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Kan maximaal 30 assets tegelijk uploaden, overslaan",
"image_viewer_page_state_provider_download_error": "Download mislukt",
"image_viewer_page_state_provider_download_success": "Download succesvol",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} zal geen toegang meer hebben tot je fotos's.",
"partner_page_stop_sharing_title": "Stoppen met het delen van je foto's?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Toch doorgaan",
"permission_onboarding_get_started": "Aan de slag",
"permission_onboarding_go_to_settings": "Ga naar instellingen",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich heeft toestemming nodig om je foto's en video's te bekijken.",
"profile_drawer_app_logs": "Logboek",
"profile_drawer_client_server_up_to_date": "App en server zijn up-to-date",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Instellingen",
"profile_drawer_sign_out": "Uitloggen",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Album aanmaken mislukt",
"select_user_for_sharing_page_share_suggestions": "Suggesties",
"server_info_box_app_version": "Appversie",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Serverversie",
"setting_image_viewer_help": "De gedetailleerde weergave laadt eerst de kleine thumbnail, vervolgens het middelgrote voorbeeld (indien ingeschakeld) en ten slotte het origineel (indien ingeschakeld).",
"setting_image_viewer_original_subtitle": "Schakel in om de originele afbeelding met volledige resolutie (groot!) te laden. Schakel uit om datagebruik te verminderen (zowel netwerk als apparaatcache).",
@@ -300,6 +315,12 @@
"share_add_photos": "Foto's toevoegen",
"share_add_title": "Titel toevoegen",
"share_create_album": "Album aanmaken",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Voorbereiden...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/pl-PL.json b/mobile/assets/i18n/pl-PL.json
index e9a8e305f..0c7cebe24 100644
--- a/mobile/assets/i18n/pl-PL.json
+++ b/mobile/assets/i18n/pl-PL.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Nie udało się zmienić tytułu albumu",
"album_viewer_appbar_share_leave": "Opuść album",
"album_viewer_appbar_share_remove": "Usuń z albumu",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Dodaj użytkowników",
"all_people_page_title": "Ludzie",
"all_videos_page_title": "Filmy",
+ "app_bar_signout_dialog_content": "Czy na pewno chcesz się wylogować?",
+ "app_bar_signout_dialog_ok": "Tak",
+ "app_bar_signout_dialog_title": "Wyloguj się",
"archive_page_no_archived_assets": "Nie znaleziono zarchiwizowanych zasobów",
"archive_page_title": "Archiwum ({})",
"asset_list_layout_settings_dynamic_layout_title": "Układ dynamiczny",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Użycie Cache",
"cache_settings_subtitle": "Kontrolowanie zachowania buforowania aplikacji mobilnej Immich",
"cache_settings_thumbnail_size": "Rozmiar pamięci podręcznej miniatur ({} zasobów)",
+ "cache_settings_tile_subtitle": "Kontroluj zachowanie lokalnego magazynu",
+ "cache_settings_tile_title": "Lokalny magazyn",
"cache_settings_title": "Ustawienia Buforowania",
"change_password_form_confirm_password": "Potwierdź Hasło",
- "change_password_form_description": "Cześć {firstName} {lastName},\n\nPierwszy raz logujesz się do systemu, albo złożono prośbę o zmianę hasła. Wpisz poniżej nowe hasło.",
+ "change_password_form_description": "Cześć {name},\n\nPierwszy raz logujesz się do systemu, albo złożono prośbę o zmianę hasła. Wpisz poniżej nowe hasło.",
"change_password_form_new_password": "Nowe Hasło",
"change_password_form_password_mismatch": "Hasła nie są zgodne",
"change_password_form_reenter_new_password": "Wprowadź ponownie Nowe Hasło",
@@ -131,9 +137,9 @@
"control_bottom_app_bar_favorite": "Ulubione",
"control_bottom_app_bar_share": "Udostępnij",
"control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_stack": "Stos",
"control_bottom_app_bar_unarchive": "Cofnij archiwizację",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Wgraj",
"create_album_page_untitled": "Bez tytułu",
"create_shared_album_page_create": "Utwórz",
"create_shared_album_page_share": "Udostępnij",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "Anuluj",
"delete_dialog_ok": "Usuń",
"delete_dialog_title": "Usuń trwale",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Czy na pewno chcesz usunąć ten udostępniony link?",
+ "delete_shared_link_dialog_title": "Usuń udostępniony link",
"description_input_hint_text": "Dodaj opis...",
"description_input_submit_error": "Błąd aktualizacji opisu, sprawdź dziennik, aby uzyskać więcej szczegółów",
"exif_bottom_sheet_description": "Dodaj Opis...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "Dodano {added} zasoby do albumu {album}. {failed} zasobów jest już w albumie.",
"home_page_add_to_album_err_local": "Nie można dodawać zasobów lokalnych do albumów, pomijam",
"home_page_add_to_album_success": "Dodano {added} zasoby do albumu {album}.",
+ "home_page_album_err_partner": "Nie można jeszcze dodać zasobów partnera do albumu, pomijam",
"home_page_archive_err_local": "Nie można jeszcze zarchiwizować zasobów lokalnych, pomijanie",
+ "home_page_archive_err_partner": "Nie można zarchiwizować zasobów partnera, pomijam",
"home_page_building_timeline": "Budowanie osi czasu",
+ "home_page_delete_err_partner": "Nie można usunąć zasobów partnera, pomijam",
"home_page_favorite_err_local": "Nie można dodać do ulubionych lokalnych zasobów, pomijam",
+ "home_page_favorite_err_partner": "Nie można jeszcze dodać do ulubionych zasobów partnera, pomijam",
"home_page_first_time_notice": "Jeśli korzystasz z aplikacji po raz pierwszy, pamiętaj o wybraniu albumów zapasowych, aby oś czasu mogła zapełnić zdjęcia i filmy w albumach.",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Można przesłać maksymalnie 30 zasobów jednocześnie, pomijanie",
"image_viewer_page_state_provider_download_error": "Błąd pobierania",
"image_viewer_page_state_provider_download_success": "Pobieranie zakończone",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Udostępnij błąd",
"library_page_albums": "Albumy",
"library_page_archive": "Archiwum",
"library_page_device_albums": "Albumy na Urządzeniu",
@@ -179,8 +190,8 @@
"library_page_new_album": "Nowy album",
"library_page_sharing": "Udostępnianie",
"library_page_sort_created": "Ostatnio utworzone",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Ostatnio zmodyfikowany",
+ "library_page_sort_most_recent_photo": "Najnowsze zdjęcie",
"library_page_sort_title": "Tytuł albumu",
"login_disabled": "Logowanie zostało wyłączone",
"login_form_api_exception": "Wyjątek API. Sprawdź adres URL serwera i spróbuj ponownie.",
@@ -218,7 +229,7 @@
"map_settings_dialog_cancel": "Anuluj",
"map_settings_dialog_save": "Zapisz",
"map_settings_dialog_title": "Ustawienia mapy",
- "map_settings_include_show_archived": "Include Archived",
+ "map_settings_include_show_archived": "Uwzględnij zarchiwizowane",
"map_settings_only_relative_range": "Zakres dat",
"map_settings_only_show_favorites": "Pokaż tylko ulubione",
"map_zoom_to_see_photos": "Pomniejsz, aby zobaczyć zdjęcia",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} nie będziesz już mieć dostępu do swoich zdjęć.",
"partner_page_stop_sharing_title": "Przestać udostępniać swoje zdjęcia?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Kontynuuj mimo to",
"permission_onboarding_get_started": "Rozpocznij",
"permission_onboarding_go_to_settings": "Przejdź do ustawień",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich potrzebuje pozwolenia na przeglądanie Twoich zdjęć i filmów.",
"profile_drawer_app_logs": "Logi",
"profile_drawer_client_server_up_to_date": "Klient i serwer są aktualne",
+ "profile_drawer_documentation": "Dokumentacja",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Ustawienia",
"profile_drawer_sign_out": "Wyloguj się",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Kosz",
"recently_added_page_title": "Ostatnio Dodane",
"search_bar_hint": "Szukaj swoich zdjęć",
"search_page_categories": "Kategorie",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Nie udało się utworzyć albumu",
"select_user_for_sharing_page_share_suggestions": "Propozycje",
"server_info_box_app_version": "Wersja Aplikacji",
+ "server_info_box_server_url": "Adres URL",
"server_info_box_server_version": "Wersja Serwera",
"setting_image_viewer_help": "Przeglądarka szczegółów najpierw ładuje małą miniaturę, następnie ładuje podgląd średniej wielkości (jeśli jest włączony), a na koniec ładuje oryginał (jeśli jest włączony).",
"setting_image_viewer_original_subtitle": "Włącz ładowanie oryginalnego obrazu w pełnej rozdzielczości (dużego!). Wyłącz, aby zmniejszyć zużycie danych (zarówno w sieci, jak i w pamięci podręcznej urządzenia).",
@@ -300,28 +315,37 @@
"share_add_photos": "Dodaj zdjęcia",
"share_add_title": "Dodaj tytuł",
"share_create_album": "Utwórz album",
+ "shared_album_activities_input_disable": "Komentarz jest wyłączony",
+ "shared_album_activities_input_hint": "Powiedz coś",
+ "shared_album_activity_remove_content": "Czy chcesz usunąć tę aktywność?",
+ "shared_album_activity_remove_title": "Usuń aktywność",
+ "shared_album_activity_setting_subtitle": "Pozwól innym odpowiedzieć",
+ "shared_album_activity_setting_title": "Komentarze i polubienia",
"share_dialog_preparing": "Przygotowywanie...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Udostępnione linki",
+ "shared_link_create_app_bar_title": "Utwórz link do udostępnienia",
+ "shared_link_create_info": "Pozwól każdemu, kto ma link, zobaczyć wybrane zdjęcia",
+ "shared_link_create_submit_button": "Utwórz link",
+ "shared_link_edit_allow_download": "Zezwalaj użytkownikowi publicznemu na pobieranie",
+ "shared_link_edit_allow_upload": "Zezwalaj użytkownikowi publicznemu na przesyłanie",
+ "shared_link_edit_app_bar_title": "Edytuj link",
+ "shared_link_edit_change_expiry": "Zmień czas ważności",
+ "shared_link_edit_description": "Opis",
+ "shared_link_edit_description_hint": "Wprowadź opis udostępnienia",
+ "shared_link_edit_expire_after": "Wygasa po",
+ "shared_link_edit_password": "Hasło",
+ "shared_link_edit_password_hint": "Wprowadź hasło udostępniania",
+ "shared_link_edit_show_meta": "Pokaż metadane",
+ "shared_link_edit_submit_button": "Aktualizuj link",
+ "shared_link_empty": "Nie masz żadnych udostępnionych linków",
+ "shared_link_manage_links": "Zarządzaj udostępnionymi linkami",
+ "share_done": "Zrobione",
"share_invite": "Zaproś do albumu",
"sharing_page_album": "Udostępnione albumy",
"sharing_page_description": "Twórz wspóldzielone albumy, aby udostępniać zdjęcia i filmy osobom w sieci.",
"sharing_page_empty_list": "PUSTA LISTA",
"sharing_silver_appbar_create_shared_album": "Utwórz współdzielony album",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Udostępnione linki",
"sharing_silver_appbar_share_partner": "Udostępnij partnerce/partnerowi",
"tab_controller_nav_library": "Biblioteka",
"tab_controller_nav_photos": "Zdjęcia",
@@ -338,18 +362,18 @@
"theme_setting_three_stage_loading_subtitle": "Trójstopniowe ładowanie może zwiększyć wydajność ładowania, ale powoduje znacznie większe obciążenie sieci",
"theme_setting_three_stage_loading_title": "Włączenie trójstopniowego ładowania",
"translated_text_options": "Opcje",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
+ "trash_page_delete": "Usuń",
+ "trash_page_delete_all": "Usuń wszystko",
+ "trash_page_empty_trash_btn": "Opróżnij kosz",
+ "trash_page_empty_trash_dialog_content": "Czy chcesz opróżnić swoje usunięte zasoby? Przedmioty te zostaną trwale usunięte z Immich",
"trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
+ "trash_page_info": "Elementy przeniesione do kosza zostaną trwale usunięte po {} dniach",
+ "trash_page_no_assets": "Brak usuniętych zasobów",
+ "trash_page_restore": "Przywrócić",
+ "trash_page_restore_all": "Przywrócić wszystkie",
+ "trash_page_select_assets_btn": "Wybierz zasoby",
+ "trash_page_select_btn": "Wybierz",
+ "trash_page_title": "Kosz({})",
"upload_dialog_cancel": "Anuluj",
"upload_dialog_info": "Czy chcesz wykonać kopię zapasową wybranych zasobów na serwerze?",
"upload_dialog_ok": "Prześlij",
@@ -360,7 +384,7 @@
"version_announcement_overlay_text_2": "prosimy o poświęcenie czasu na odwiedzenie ",
"version_announcement_overlay_text_3": " i upewnij się, że twoja konfiguracja docker-compose i .env jest aktualna, aby zapobiec błędnym konfiguracjom, zwłaszcza jeśli używasz WatchTower lub dowolnego mechanizmu, który obsługuje automatyczną aktualizację aplikacji serwera.",
"version_announcement_overlay_title": "Nowa wersja serwera dostępna \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Usuń ze stosu",
+ "viewer_stack_use_as_main_asset": "Użyj jako głównego zasobu",
+ "viewer_unstack": "Usuń stos"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/ru-RU.json b/mobile/assets/i18n/ru-RU.json
index 542eab218..2767a8200 100644
--- a/mobile/assets/i18n/ru-RU.json
+++ b/mobile/assets/i18n/ru-RU.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Ошибка переименования альбома",
"album_viewer_appbar_share_leave": "Покинуть альбом",
"album_viewer_appbar_share_remove": "Удалить из альбома",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Добавить пользователей",
"all_people_page_title": "Люди",
"all_videos_page_title": "Видео",
+ "app_bar_signout_dialog_content": "Вы уверены, что хотите выйти из системы?",
+ "app_bar_signout_dialog_ok": "Да",
+ "app_bar_signout_dialog_title": "Выйти из системы",
"archive_page_no_archived_assets": "В архиве сейчас пусто",
"archive_page_title": "Архив ({})",
"asset_list_layout_settings_dynamic_layout_title": "Динамическое расположение",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Размер кэша",
"cache_settings_subtitle": "Управление кэшированием мобильного приложения Immich",
"cache_settings_thumbnail_size": "Размер кэша эскизов ({} объектов)",
+ "cache_settings_tile_subtitle": "Управление поведением локального хранилища",
+ "cache_settings_tile_title": "Локальное хранилище",
"cache_settings_title": "Настройки кэширования",
"change_password_form_confirm_password": "Подтвердите пароль",
- "change_password_form_description": "Привет {firstName} {lastName},\n\nЭто либо ваш первый вход в систему, либо был сделан запрос на смену пароля. Пожалуйста, введите новый пароль ниже.",
+ "change_password_form_description": "Привет {name},\n\nЭто либо ваш первый вход в систему, либо был сделан запрос на смену пароля. Пожалуйста, введите новый пароль ниже.",
"change_password_form_new_password": "Новый пароль",
"change_password_form_password_mismatch": "Пароли не совпадают",
"change_password_form_reenter_new_password": "Повторно введите новый пароль",
@@ -130,10 +136,10 @@
"control_bottom_app_bar_delete": "Удалить",
"control_bottom_app_bar_favorite": "Избранное",
"control_bottom_app_bar_share": "Поделиться",
- "control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_share_to": "Поделиться",
+ "control_bottom_app_bar_stack": "Стек",
"control_bottom_app_bar_unarchive": "Восстановить",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Загрузить",
"create_album_page_untitled": "Без названия",
"create_shared_album_page_create": "Создать",
"create_shared_album_page_share": "Поделиться",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "Отменить",
"delete_dialog_ok": "Удалить",
"delete_dialog_title": "Удалить навсегда",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Вы уверены, что хотите удалить эту общую ссылку?",
+ "delete_shared_link_dialog_title": "Удалить общую ссылку",
"description_input_hint_text": "Добавить описание...",
"description_input_submit_error": "Не удалось обновить описание, проверьте логи, чтобы узнать причину",
"exif_bottom_sheet_description": "Добавить описание...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "Добавлено {added} объектов в альбом {album}. Объекты {failed} уже есть в альбоме.",
"home_page_add_to_album_err_local": "Пока нельзя добавлять локальные объекты в альбомы, пропускаем",
"home_page_add_to_album_success": "Добавлено {added} объектов в альбом {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Пока невозможно добавить локальные объекты в архив, пропускаем",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Построение временной шкалы",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Пока не удается добавить в избранное локальные объекты, пропускаем",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Если вы используете приложение впервые, убедитесь, что вы выбрали резервный(е) альбом(ы), чтобы временная шкала могла заполнить фотографии и видео в альбоме(ах).",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Вы можете выгрузить максимум 30 файлов за раз",
"image_viewer_page_state_provider_download_error": "Ошибка загрузки",
"image_viewer_page_state_provider_download_success": "Успешно загружено",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Ошибка при публикации",
"library_page_albums": "Альбомы",
"library_page_archive": "Архив",
"library_page_device_albums": "Альбомы на устройстве",
@@ -179,8 +190,8 @@
"library_page_new_album": "Новый альбом",
"library_page_sharing": "Общие",
"library_page_sort_created": "По новизне",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Последнее изменение",
+ "library_page_sort_most_recent_photo": "Последняя фотография",
"library_page_sort_title": "По названию альбома",
"login_disabled": "Вход отключен",
"login_form_api_exception": "Ошибка при попытке взаимодействия с сервером. Проверьте URL-адрес до него и попробуйте еще раз.",
@@ -218,7 +229,7 @@
"map_settings_dialog_cancel": "Отмена",
"map_settings_dialog_save": "Сохранить",
"map_settings_dialog_title": "Настройки карты",
- "map_settings_include_show_archived": "Include Archived",
+ "map_settings_include_show_archived": "Включить архивные данные",
"map_settings_only_relative_range": "Период времени",
"map_settings_only_show_favorites": "Показать только избранное",
"map_zoom_to_see_photos": "Уменьшение масштаба для просмотра фотографий",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} больше не сможет получить доступ к вашим фотографиям",
"partner_page_stop_sharing_title": "Закрыть доступ партнёра к вашим фото?",
"partner_page_title": "Партнёр",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Все равно продолжить",
"permission_onboarding_get_started": "Давайте начнём",
"permission_onboarding_go_to_settings": "Перейти в настройки",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich просит вас предоставить разрешение на доступ к вашим фото и видео",
"profile_drawer_app_logs": "Журнал",
"profile_drawer_client_server_up_to_date": "Клиент и сервер обновлены",
+ "profile_drawer_documentation": "Документация",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Настройки",
"profile_drawer_sign_out": "Выйти",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Корзина",
"recently_added_page_title": "Недавно добавленные",
"search_bar_hint": "Поиск фотографий",
"search_page_categories": "Категории",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "\nНе удалось создать альбом",
"select_user_for_sharing_page_share_suggestions": "Предложения",
"server_info_box_app_version": "Версия приложения",
+ "server_info_box_server_url": "URL сервера",
"server_info_box_server_version": "Версия сервера",
"setting_image_viewer_help": "Средство просмотра деталей сначала загружает маленькую миниатюру, затем загружает предварительный просмотр среднего размера (если включено) и, наконец, загружает оригинал (если включено).",
"setting_image_viewer_original_subtitle": "Включите загрузку оригинального изображения в полном разрешении (большое!). Отключите, чтобы уменьшить объем данных (как в сети, так и в кеше устройства).",
@@ -300,28 +315,37 @@
"share_add_photos": "Добавить фото",
"share_add_title": "Добавить название",
"share_create_album": "Создать альбом",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Подготовка...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Общие ссылки",
+ "shared_link_create_app_bar_title": "Создать ссылку для совместного использования",
+ "shared_link_create_info": "Позволить любому человеку, имеющему ссылку, увидеть выбранную фотографию (фотографии)",
+ "shared_link_create_submit_button": "Создать ссылку",
+ "shared_link_edit_allow_download": "Разрешить публичному пользователю скачивать",
+ "shared_link_edit_allow_upload": "Разрешить публичному пользователю загружать файлы",
+ "shared_link_edit_app_bar_title": "Редактировать ссылку",
+ "shared_link_edit_change_expiry": "Изменить срок действия доступа",
+ "shared_link_edit_description": "Описание",
+ "shared_link_edit_description_hint": "Введите описание совместного доступа",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Пароль",
+ "shared_link_edit_password_hint": "Введите пароль общего доступа",
+ "shared_link_edit_show_meta": "Показать метаданные",
+ "shared_link_edit_submit_button": "Обновить ссылку",
+ "shared_link_empty": "У вас нет общих ссылок",
+ "shared_link_manage_links": "Управление общими ссылками",
+ "share_done": "Выполнено",
"share_invite": "\nПригласить в альбом",
"sharing_page_album": "Общие альбомы",
"sharing_page_description": "Создавайте общие альбомы, чтобы делиться фотографиями и видео с людьми в вашей сети.",
"sharing_page_empty_list": "ПУСТОЙ СПИСОК",
"sharing_silver_appbar_create_shared_album": "Создать общий альбом",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Общие ссылки",
"sharing_silver_appbar_share_partner": "Поделиться с партнёром",
"tab_controller_nav_library": "Библиотека",
"tab_controller_nav_photos": "Фото",
@@ -338,18 +362,18 @@
"theme_setting_three_stage_loading_subtitle": "Трехэтапная загрузка может повысить производительность загрузки, но вызывает значительно более высокую нагрузку на сеть",
"theme_setting_three_stage_loading_title": "Включить трехэтапную загрузку",
"translated_text_options": "Опции",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
- "trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
+ "trash_page_delete": "Удалить",
+ "trash_page_delete_all": "Удалить все",
+ "trash_page_empty_trash_btn": "Очистить корзину",
+ "trash_page_empty_trash_dialog_content": "Вы хотите очистить свою корзину? Эти объекты будут навсегда удалены из Immich",
+ "trash_page_empty_trash_dialog_ok": "ОК",
+ "trash_page_info": "Удаленные элементы будут окончательно удалены через {} дней",
+ "trash_page_no_assets": "Отсутствие удаленных объектов",
+ "trash_page_restore": "Восстановить",
+ "trash_page_restore_all": "Восстановить все",
+ "trash_page_select_assets_btn": "Выбранные объекты",
+ "trash_page_select_btn": "Выбрать",
+ "trash_page_title": "Корзина ({})",
"upload_dialog_cancel": "Отмена",
"upload_dialog_info": "Вы хотите загрузить выбранный объект(ы) на ваш сервер?",
"upload_dialog_ok": "Загрузить",
@@ -360,7 +384,7 @@
"version_announcement_overlay_text_2": "пожалуйста, найдите время, чтобы посетить",
"version_announcement_overlay_text_3": " и убедитесь, что ваши настройки docker-compose и .env обновлены, чтобы предотвратить любые неправильные настройки, особенно если вы используете WatchTower или любой другой механизм, который обрабатывает обновление вашего серверного приложения автоматически.",
"version_announcement_overlay_title": "Доступна новая версия сервера \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Удалить из стека",
+ "viewer_stack_use_as_main_asset": "Использование в качестве основного объекта",
+ "viewer_unstack": "Разобрать стек"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/sk-SK.json b/mobile/assets/i18n/sk-SK.json
index 42ff34680..3e4921ffe 100644
--- a/mobile/assets/i18n/sk-SK.json
+++ b/mobile/assets/i18n/sk-SK.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Nepodarilo sa zmeniť názov albumu",
"album_viewer_appbar_share_leave": "Opustiť album",
"album_viewer_appbar_share_remove": "Odstrániť z albumu",
+ "album_viewer_appbar_share_to": "Zdieľať s",
"album_viewer_page_share_add_users": "Pridať používateľov",
"all_people_page_title": "Ľudia",
"all_videos_page_title": "Videá",
+ "app_bar_signout_dialog_content": "Skutočne sa chcete odhlásiť?",
+ "app_bar_signout_dialog_ok": "Áno",
+ "app_bar_signout_dialog_title": "Odhlásiť sa",
"archive_page_no_archived_assets": "Žiadne archivované médiá",
"archive_page_title": "Archív ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamické rozloženie",
@@ -65,7 +69,7 @@
"backup_controller_page_background_is_on": "Automatické zálohovanie na pozadí je zapnuté",
"backup_controller_page_background_turn_off": "Vypnúť zálohovanie na pozadí",
"backup_controller_page_background_turn_on": "Povoliť zálohovanie na pozadí",
- "backup_controller_page_background_wifi": "Len na WiFi",
+ "backup_controller_page_background_wifi": "Len cez WiFi",
"backup_controller_page_backup": "Zálohovanie",
"backup_controller_page_backup_selected": "Vybrané: ",
"backup_controller_page_backup_sub": "Zálohované fotografie a videa",
@@ -86,7 +90,7 @@
"backup_controller_page_status_off": "Automatické zálohovanie na popredí je vypnuté",
"backup_controller_page_status_on": "Automatické zálohovanie na popredí je zapnuté",
"backup_controller_page_storage_format": "{} z {} použitých",
- "backup_controller_page_to_backup": "Albumy, ktoré sa majú zálohovať",
+ "backup_controller_page_to_backup": "Albumy ktoré budú zálohované",
"backup_controller_page_total": "Celkom",
"backup_controller_page_total_sub": "Všetky jedinečné fotografie a videá z vybraných albumov",
"backup_controller_page_turn_off": "Vypnúť zálohovanie na popredí",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Použitie vyrovnávacej pamäte",
"cache_settings_subtitle": "Ovládanie správania mobilnej aplikácie Immich v medzipamäti",
"cache_settings_thumbnail_size": "Veľkosť vyrovnávacej pamäte náhľadov (položiek {})",
+ "cache_settings_tile_subtitle": "Ovládanie správania lokálneho úložiska",
+ "cache_settings_tile_title": "Lokálne úložisko",
"cache_settings_title": "Nastavenia vyrovnávacej pamäte",
"change_password_form_confirm_password": "Potvrďte heslo",
- "change_password_form_description": "Dobrý deň, {firstName} {lastName},\n\nBuď sa do systému prihlasujete prvýkrát, alebo bola podaná žiadosť o zmenu hesla. Prosím, zadajte nové heslo nižšie.",
+ "change_password_form_description": "Dobrý deň, {name},\n\nBuď sa do systému prihlasujete prvýkrát, alebo bola podaná žiadosť o zmenu hesla. Prosím, zadajte nové heslo nižšie.",
"change_password_form_new_password": "Nové heslo",
"change_password_form_password_mismatch": "Heslá sa nezhodujú",
"change_password_form_reenter_new_password": "Znova zadajte nové heslo",
@@ -125,15 +131,15 @@
"control_bottom_app_bar_add_to_album": "Pridať do albumu",
"control_bottom_app_bar_album_info": "{} položiek",
"control_bottom_app_bar_album_info_shared": "{} položiek - zdieľané",
- "control_bottom_app_bar_archive": "Archív",
+ "control_bottom_app_bar_archive": "Archivovať",
"control_bottom_app_bar_create_new_album": "Vytvoriť nový album",
"control_bottom_app_bar_delete": "Vymazať",
"control_bottom_app_bar_favorite": "Obľúbené",
"control_bottom_app_bar_share": "Zdieľať",
"control_bottom_app_bar_share_to": "Share To",
- "control_bottom_app_bar_stack": "Stack",
+ "control_bottom_app_bar_stack": "Zoskupenie",
"control_bottom_app_bar_unarchive": "Odarchivovať",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Nahrať",
"create_album_page_untitled": "Bez názvu",
"create_shared_album_page_create": "Vytvoriť",
"create_shared_album_page_share": "Zdieľať",
@@ -148,8 +154,8 @@
"delete_dialog_cancel": "Zrušiť",
"delete_dialog_ok": "Vymazať",
"delete_dialog_title": "Vymazať natrvalo",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Ste si istí že chcete odstrániť tento zdieľaný odkaz?",
+ "delete_shared_link_dialog_title": "Odstrániť zdieľaný odkaz",
"description_input_hint_text": "Pridať popis...",
"description_input_submit_error": "Chyba pri aktualizovaní popisu, zobrazte log pre viac detailov",
"exif_bottom_sheet_description": "Pridať popis...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "Pridané {added} položiek do albumu {album}. {failed} položiek už je v albume.",
"home_page_add_to_album_err_local": "Zatiaľ nie je možné pridať lokálne média do albumov, preskakuje sa",
"home_page_add_to_album_success": "Pridané {added} položky do albumu {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Zatiaľ nemožno archivovať lokálne médiá, preskakuje sa",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Vytváranie časovej osi",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Zatiaľ nie je možné zaradiť lokálne média medzi obľúbené, preskakuje sa",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Ak aplikáciu používate prvý krát, nezabudnite si vybrať zálohované albumy, aby sa na časovej osi mohli nachádzať fotografie a videá z vybraných albumoch.",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Naraz môžete nahrať len 30 médií, preskakujem...",
"image_viewer_page_state_provider_download_error": "Chyba sťahovania",
"image_viewer_page_state_provider_download_success": "Sťahovanie bolo úspešné",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "Chyba zdieľania",
"library_page_albums": "Albumy",
"library_page_archive": "Archív",
"library_page_device_albums": "Albumy v zariadení",
@@ -179,8 +190,8 @@
"library_page_new_album": "Nový album",
"library_page_sharing": "Zdieľanie",
"library_page_sort_created": "Najnovšie vytvorené",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Naposledy upravené",
+ "library_page_sort_most_recent_photo": "Najnovšia fotka",
"library_page_sort_title": "Podľa názvu albumu",
"login_disabled": "Prihlasovanie bolo vypnuté",
"login_form_api_exception": "Chyba API. Skontrolujte adresu URL servera a skúste to znova.",
@@ -218,7 +229,7 @@
"map_settings_dialog_cancel": "Zrušiť",
"map_settings_dialog_save": "Uložiť",
"map_settings_dialog_title": "Nastavenia máp",
- "map_settings_include_show_archived": "Include Archived",
+ "map_settings_include_show_archived": "Zahrnúť archivované",
"map_settings_only_relative_range": "Rozsah dátumu",
"map_settings_only_show_favorites": "Zobraziť iba obľúbené",
"map_zoom_to_see_photos": "Oddiaľte priblíženie aby ste videli fotky",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} už nebude mať prístup ku vašim fotkám.",
"partner_page_stop_sharing_title": "Zastaviť zdieľanie vašich fotiek?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Pokračovať aj tak",
"permission_onboarding_get_started": "Začať",
"permission_onboarding_go_to_settings": "Prejsť do nastavení",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich vyžaduje povolenie na prezeranie vašich fotografií a videí.",
"profile_drawer_app_logs": "Logy",
"profile_drawer_client_server_up_to_date": "Klient a server sú aktuálne",
+ "profile_drawer_documentation": "Dokumentácia",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Nastavenia",
"profile_drawer_sign_out": "Odhlásiť sa",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Kôš",
"recently_added_page_title": "Nedávno pridané",
"search_bar_hint": "Prehľadajte svoje obrázky",
"search_page_categories": "Kategórie",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Nepodarilo sa vytvoriť album",
"select_user_for_sharing_page_share_suggestions": "Návrhy",
"server_info_box_app_version": "Verzia aplikácie",
+ "server_info_box_server_url": "URL Serveru",
"server_info_box_server_version": "Verzia servera",
"setting_image_viewer_help": "Prehliadač detailov najprv načíta malú miniatúru, potom načíta náhľad strednej veľkosti (ak je povolený) a nakoniec načíta originál (ak je povolený).",
"setting_image_viewer_original_subtitle": "Povolením umožníte načítať pôvodný obrázok v plnom rozlíšení (veľký!). Zakázaním znížite používania dát (v sieti, aj v dočasnej pamäte zariadenia).",
@@ -300,28 +315,37 @@
"share_add_photos": "Pridať fotografie",
"share_add_title": "Pridať názov",
"share_create_album": "Vytvoriť album",
+ "shared_album_activities_input_disable": "Komentár je zakázaný",
+ "shared_album_activities_input_hint": "Napíšte niečo",
+ "shared_album_activity_remove_content": "Chcete vymazať túto aktivitu?",
+ "shared_album_activity_remove_title": "Vymazať aktivitu",
+ "shared_album_activity_setting_subtitle": "Nechajte ostatných reagovať",
+ "shared_album_activity_setting_title": "Komentáre a lajky",
"share_dialog_preparing": "Pripravujem...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Zdieľané odkazy",
+ "shared_link_create_app_bar_title": "Vytvoriť odkaz na zdieľanie",
+ "shared_link_create_info": "Umožniť komukoľvek s odkazom zobraziť označené médiá",
+ "shared_link_create_submit_button": "Vytvoriť odkaz",
+ "shared_link_edit_allow_download": "Povoliť návštevníkom sťahovať médiá",
+ "shared_link_edit_allow_upload": "Povoliť návštevníkom pridávať médiá",
+ "shared_link_edit_app_bar_title": "Upraviť odkaz",
+ "shared_link_edit_change_expiry": "Zmeniť čas vypršania",
+ "shared_link_edit_description": "Popis",
+ "shared_link_edit_description_hint": "Zadajte popis zdieľania",
+ "shared_link_edit_expire_after": "Expiruje po",
+ "shared_link_edit_password": "Heslo",
+ "shared_link_edit_password_hint": "Zadajte heslo zdieľania",
+ "shared_link_edit_show_meta": "Zobraziť metadáta",
+ "shared_link_edit_submit_button": "Aktualizovať odkaz",
+ "shared_link_empty": "Zatiaľ nemáte žiadne zdieľané odkazy",
+ "shared_link_manage_links": "Spravovať zdieľané odkazy",
+ "share_done": "Hotovo",
"share_invite": "Pozvať do albumu",
"sharing_page_album": "Zdieľané albumy",
"sharing_page_description": "Vytvárajte zdieľané albumy a zdieľajte fotografie a videá s ľuďmi vo vašej sieti.",
"sharing_page_empty_list": "Prázdny list",
"sharing_silver_appbar_create_shared_album": "Vytvoriť zdieľaný album",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Zdieľané odkazy",
"sharing_silver_appbar_share_partner": "Zdieľať s partnerom",
"tab_controller_nav_library": "Knižnica",
"tab_controller_nav_photos": "Fotografie",
@@ -338,18 +362,18 @@
"theme_setting_three_stage_loading_subtitle": "Trojstupňové načítanie môže zvýšiť výkonnosť načítania, ale vedie k výrazne vyššiemu zaťaženiu siete.",
"theme_setting_three_stage_loading_title": "Povolenie trojstupňového načítavania",
"translated_text_options": "Nastavenia",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
+ "trash_page_delete": "Vymazať",
+ "trash_page_delete_all": "Vymazať všetky",
+ "trash_page_empty_trash_btn": "Vyprázdniť kôš",
+ "trash_page_empty_trash_dialog_content": "Skutočne chcete vyprázdniť kôš? Tieto položky budú permanentne odstránené z Immichu",
"trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
+ "trash_page_info": "Médiá v koši sa permanentne odstránia po {} dňoch",
+ "trash_page_no_assets": "Žiadne médiá v koši",
+ "trash_page_restore": "Obnoviť",
+ "trash_page_restore_all": "Obnoviť všetky",
+ "trash_page_select_assets_btn": "Označiť médiá",
+ "trash_page_select_btn": "Označiť",
+ "trash_page_title": "Kôš ({})",
"upload_dialog_cancel": "Zrušiť",
"upload_dialog_info": "Chcete zálohovať zvolené médiá na server?",
"upload_dialog_ok": "Nahrať",
@@ -360,7 +384,7 @@
"version_announcement_overlay_text_2": "nájdite si čas na návštevu ",
"version_announcement_overlay_text_3": " a uistite sa, že vaša konfigurácia docker-compose a .env je aktuálna, aby ste predišli nesprávnej konfigurácii, najmä ak používate WatchTower alebo akýkoľvek mechanizmus, ktorý podporuje automatické aktualizácie serverových aplikácií.",
"version_announcement_overlay_title": "K dispozícii je nová verzia servera \uD83C\uDF89",
- "viewer_remove_from_stack": "Remove from Stack",
- "viewer_stack_use_as_main_asset": "Use as Main Asset",
- "viewer_unstack": "Un-Stack"
+ "viewer_remove_from_stack": "Odstrániť zo zoskupenia",
+ "viewer_stack_use_as_main_asset": "Použiť ako hlavnú fotku",
+ "viewer_unstack": "Odskupiť"
}
\ No newline at end of file
diff --git a/mobile/assets/i18n/sr-Cyrl.json b/mobile/assets/i18n/sr-Cyrl.json
index 31311535b..9c42afb7d 100644
--- a/mobile/assets/i18n/sr-Cyrl.json
+++ b/mobile/assets/i18n/sr-Cyrl.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Failed to change album title",
"album_viewer_appbar_share_leave": "Leave album",
"album_viewer_appbar_share_remove": "Remove from album",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Add users",
"all_people_page_title": "People",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "No archived assets found",
"archive_page_title": "Archive ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamic layout",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Cache usage",
"cache_settings_subtitle": "Control the caching behaviour of the Immich mobile application",
"cache_settings_thumbnail_size": "Thumbnail cache size ({} assets)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Caching Settings",
"change_password_form_confirm_password": "Confirm Password",
- "change_password_form_description": "Hi {firstName} {lastName},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
+ "change_password_form_description": "Hi {name},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
"change_password_form_new_password": "New Password",
"change_password_form_password_mismatch": "Passwords do not match",
"change_password_form_reenter_new_password": "Re-enter New Password",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Added {added} assets to album {album}. {failed} assets are already in the album.",
"home_page_add_to_album_err_local": "Can not add local assets to albums yet, skipping",
"home_page_add_to_album_success": "Added {added} assets to album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Can not archive local assets yet, skipping",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Building the timeline",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Can not favorite local assets yet, skipping",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "If this is your first time using the app, please make sure to choose a backup album(s) so that the timeline can populate photos and videos in the album(s).",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
"image_viewer_page_state_provider_download_error": "Download Error",
"image_viewer_page_state_provider_download_success": "Download Success",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
"partner_page_stop_sharing_title": "Stop sharing your photos?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continue anyway",
"permission_onboarding_get_started": "Get started",
"permission_onboarding_go_to_settings": "Go to settings",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich requires permission to view your photos and videos.",
"profile_drawer_app_logs": "Logs",
"profile_drawer_client_server_up_to_date": "Client and Server are up-to-date",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Settings",
"profile_drawer_sign_out": "Sign Out",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Failed to create album",
"select_user_for_sharing_page_share_suggestions": "Suggestions",
"server_info_box_app_version": "App Version",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Server Version",
"setting_image_viewer_help": "The detail viewer loads the small thumbnail first, then loads the medium-size preview (if enabled), finally loads the original (if enabled).",
"setting_image_viewer_original_subtitle": "Enable to load the original full-resolution image (large!). Disable to reduce data usage (both network and on device cache).",
@@ -300,6 +315,12 @@
"share_add_photos": "Add photos",
"share_add_title": "Add a title",
"share_create_album": "Create album",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Preparing...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/sr-Latn.json b/mobile/assets/i18n/sr-Latn.json
index 6273f33fc..9588a2ede 100644
--- a/mobile/assets/i18n/sr-Latn.json
+++ b/mobile/assets/i18n/sr-Latn.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Neuspešno menjanje naziva albuma",
"album_viewer_appbar_share_leave": "Izađi iz albuma",
"album_viewer_appbar_share_remove": "Obriši iz albuma",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Dodaj korisnike",
"all_people_page_title": "People",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "No archived assets found",
"archive_page_title": "Archive ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dinamični raspored",
@@ -111,6 +115,8 @@
"cache_settings_statistics_title": "Iskorišćena keš memorija",
"cache_settings_subtitle": "Kontrole za keš memoriju mobilne aplikacije Immich",
"cache_settings_thumbnail_size": "Keš memorija koju zauzimaju minijature ({} stavki)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Opcije za keširanje",
"change_password_form_confirm_password": "Ponovo unesite šifru",
"change_password_form_description": "Ćao, {firstName}, {lastName}\n\nOvo je verovatno Vaše prvo pristupanje sistemu, ili je podnešen zahtev za promenu šifre. Molimo Vas, unesite novu šifru ispod",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Dodat {added} zapis u album {album}. {failed} zapisi su već u albumu ",
"home_page_add_to_album_err_local": "Trenutno nemoguće dodati lokalne zapise u albume, preskacu se",
"home_page_add_to_album_success": "Dodate {added} stavke u album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Can not archive local assets yet, skipping",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Kreiranje hronološke linije",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Trenutno nije moguce dodati lokalne zapise u favorite, preskacu se",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Ako je ovo prvi put da koristite aplikaciju, molimo Vas da odaberete albume koje želite da sačuvate",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
"image_viewer_page_state_provider_download_error": "Preuzimanje Neuspešno",
"image_viewer_page_state_provider_download_success": "Preuzimanje Uspešno",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
"partner_page_stop_sharing_title": "Stop sharing your photos?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continue anyway",
"permission_onboarding_get_started": "Get started",
"permission_onboarding_go_to_settings": "Go to settings",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich requires permission to view your photos and videos.",
"profile_drawer_app_logs": "Evidencija",
"profile_drawer_client_server_up_to_date": "Klijent i server su najnovije verzije",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Opcije",
"profile_drawer_sign_out": "Odjavi se",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Neuspešno kreiranje albuma",
"select_user_for_sharing_page_share_suggestions": "Sugestije",
"server_info_box_app_version": "Verzija Aplikacije",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Verzija Servera",
"setting_image_viewer_help": "Detaljno pregledanje prvo učitava minijaturu, pa srednju, pa original. (Ako te opcije uključene)",
"setting_image_viewer_original_subtitle": "Aktiviraj učitavanje slika u punoj rezoluciji (Velika!). Deaktivacijom ove stavke možeš da smanjiš potrošnju interneta i zauzetog prostora na uređaju.",
@@ -300,6 +315,12 @@
"share_add_photos": "Dodaj fotografije",
"share_add_title": "Dodaj naslov",
"share_create_album": "Napravi album",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Pripremanje...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/sv-FI.json b/mobile/assets/i18n/sv-FI.json
index 31311535b..9c42afb7d 100644
--- a/mobile/assets/i18n/sv-FI.json
+++ b/mobile/assets/i18n/sv-FI.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Failed to change album title",
"album_viewer_appbar_share_leave": "Leave album",
"album_viewer_appbar_share_remove": "Remove from album",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Add users",
"all_people_page_title": "People",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "No archived assets found",
"archive_page_title": "Archive ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamic layout",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Cache usage",
"cache_settings_subtitle": "Control the caching behaviour of the Immich mobile application",
"cache_settings_thumbnail_size": "Thumbnail cache size ({} assets)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Caching Settings",
"change_password_form_confirm_password": "Confirm Password",
- "change_password_form_description": "Hi {firstName} {lastName},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
+ "change_password_form_description": "Hi {name},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
"change_password_form_new_password": "New Password",
"change_password_form_password_mismatch": "Passwords do not match",
"change_password_form_reenter_new_password": "Re-enter New Password",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Added {added} assets to album {album}. {failed} assets are already in the album.",
"home_page_add_to_album_err_local": "Can not add local assets to albums yet, skipping",
"home_page_add_to_album_success": "Added {added} assets to album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Can not archive local assets yet, skipping",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Building the timeline",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Can not favorite local assets yet, skipping",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "If this is your first time using the app, please make sure to choose a backup album(s) so that the timeline can populate photos and videos in the album(s).",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
"image_viewer_page_state_provider_download_error": "Download Error",
"image_viewer_page_state_provider_download_success": "Download Success",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
"partner_page_stop_sharing_title": "Stop sharing your photos?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continue anyway",
"permission_onboarding_get_started": "Get started",
"permission_onboarding_go_to_settings": "Go to settings",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich requires permission to view your photos and videos.",
"profile_drawer_app_logs": "Logs",
"profile_drawer_client_server_up_to_date": "Client and Server are up-to-date",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Settings",
"profile_drawer_sign_out": "Sign Out",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Failed to create album",
"select_user_for_sharing_page_share_suggestions": "Suggestions",
"server_info_box_app_version": "App Version",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Server Version",
"setting_image_viewer_help": "The detail viewer loads the small thumbnail first, then loads the medium-size preview (if enabled), finally loads the original (if enabled).",
"setting_image_viewer_original_subtitle": "Enable to load the original full-resolution image (large!). Disable to reduce data usage (both network and on device cache).",
@@ -300,6 +315,12 @@
"share_add_photos": "Add photos",
"share_add_title": "Add a title",
"share_create_album": "Create album",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Preparing...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/sv-SE.json b/mobile/assets/i18n/sv-SE.json
index 04ae04992..026219fe5 100644
--- a/mobile/assets/i18n/sv-SE.json
+++ b/mobile/assets/i18n/sv-SE.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Kunde inte ändra albumtitel",
"album_viewer_appbar_share_leave": "Lämna album",
"album_viewer_appbar_share_remove": "Ta bort från album",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Lägg till användare",
"all_people_page_title": "People",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "No archived assets found",
"archive_page_title": "Arkivera ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamisk layout",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Cacheförbrukning",
"cache_settings_subtitle": "Hantera cachebeteendet för Immich-appen.",
"cache_settings_thumbnail_size": "Storlek på cacheminnet ({} bilder och videor)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Cache Inställningar",
"change_password_form_confirm_password": "Bekräfta lösenord",
- "change_password_form_description": "Hi {firstName} {lastName},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
+ "change_password_form_description": "Hi {name},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
"change_password_form_new_password": "Nytt lösenord",
"change_password_form_password_mismatch": "Passwords do not match",
"change_password_form_reenter_new_password": "Re-enter New Password",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Lade till {added} foton och videor i albumet {album}. {failed} foton och videor finns redan i albumet.",
"home_page_add_to_album_err_local": "Can not add local assets to albums yet, skipping",
"home_page_add_to_album_success": "Lade till {added} foton och videor i albumet {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Can not archive local assets yet, skipping",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Bygger tidslinjen",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Can not favorite local assets yet, skipping",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Om det här är första gången du använder appen, välj ett eller flera backup-album så att tidslinjen kan fyllas med foton och videor från albumen.",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
"image_viewer_page_state_provider_download_error": "Download Error",
"image_viewer_page_state_provider_download_success": "Download Success",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
"partner_page_stop_sharing_title": "Sluta dela dina foton?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continue anyway",
"permission_onboarding_get_started": "Kom igång",
"permission_onboarding_go_to_settings": "Gå till inställningar",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich kräver tillstånd för att se dina foton och videor.",
"profile_drawer_app_logs": "Loggar",
"profile_drawer_client_server_up_to_date": "Klient och server är uppdaterade",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Inställningar",
"profile_drawer_sign_out": "Logga ut",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Kunde inte skapa nytt album",
"select_user_for_sharing_page_share_suggestions": "Förslag",
"server_info_box_app_version": "App version",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Server version",
"setting_image_viewer_help": "Detaljerad vy laddar miniatyrer först. Efter detta laddas den medelstora förhandsgranskningen av bilden (om detta är aktiverat), och visar slutligen originalet (om detta är aktiverat).",
"setting_image_viewer_original_subtitle": "Aktivera för att ladda originalbilden i full storlek (stor!). Inaktivera för att minska dataanvändningen (både i nätverket och för enhetscache).",
@@ -300,6 +315,12 @@
"share_add_photos": "Lägg till foton",
"share_add_title": "Lägg till en titel",
"share_create_album": "Skapa album",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Förbereder...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/th-TH.json b/mobile/assets/i18n/th-TH.json
index 6f89ce173..0cfad7eff 100644
--- a/mobile/assets/i18n/th-TH.json
+++ b/mobile/assets/i18n/th-TH.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "เปลี่ยนชื่ออัลบั้มไม่สำเร็จ",
"album_viewer_appbar_share_leave": "ออกจากอัลบั้ม",
"album_viewer_appbar_share_remove": "ลบออกจากอัลบั้ม",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "เพิ่มผู้ใช้งาน",
"all_people_page_title": "ผู้คน",
"all_videos_page_title": "วิดีโอ",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "ไม่พบทรัพยากรในที่เก็บถาวร",
"archive_page_title": "เก็บถาวร ({})",
"asset_list_layout_settings_dynamic_layout_title": "แผนผังปรับตัว",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "การใช้งานแคช",
"cache_settings_subtitle": "ควบคุมพฤติกรรมการแคชของแอปพลิเคชัน Immich",
"cache_settings_thumbnail_size": "ขนาดแคชรูปย่อ ({} ทรัพยากร)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "ตั้งค่าแคช",
"change_password_form_confirm_password": "ยืนยันรหัสผ่าน",
- "change_password_form_description": "สวัสดี {firstName} {lastName},\n\nครั้งนี้อาจจะเป็นครั้งแรกที่คุณเข้าสู่ระบบ หรือมีคำขอเพื่อที่จะเปลี่ยนรหัสผ่านของคุI กรุณาเพิ่มรหัสผ่านใหม่ข้างล่าง",
+ "change_password_form_description": "สวัสดี {name},\n\nครั้งนี้อาจจะเป็นครั้งแรกที่คุณเข้าสู่ระบบ หรือมีคำขอเพื่อที่จะเปลี่ยนรหัสผ่านของคุI กรุณาเพิ่มรหัสผ่านใหม่ข้างล่าง",
"change_password_form_new_password": "รหัสผ่านใหม่",
"change_password_form_password_mismatch": "รหัสผ่านไม่ตรงกัน",
"change_password_form_reenter_new_password": "กรอกรหัสผ่านใหม่",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "เพิ่ม {added} ทรัพยากรเข้าอัลบั้ม {album}. {failed} ทรัพยากรอยู่ในอัลบั้มอยู่แล้ว",
"home_page_add_to_album_err_local": " ไม่สามารถเพิ่มทรัพยากรบนเครื่องเข้าอัลบั้ม กำลังข้าม",
"home_page_add_to_album_success": "Added {added} assets to album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "ไม่สามารถเก็บถาวรในขณะนี้ กำลังข้าม",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "กำลังสร้าง timeline",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": " ไม่สามารถตั้งทรัพยากรบนเครื่องเป็นรายการโปรด กำลังข้าม",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "ถ้าครั้งนี้เป็นครั้งแรกที่ใช้แอปนี้ กรุณาเลือกอัลบั้มที่จะสำรองข้อมูล ไทม์ไลน์จะได้เพิ่มรูปภาพและวิดีโอที่อยู่ในอัลบั้ม",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "สามารถอัพโหลดได้มากสุดครั้งละ 30 ทรัพยากร กำลังข้าม",
"image_viewer_page_state_provider_download_error": "ดาวน์โหลดผิดพลาด",
"image_viewer_page_state_provider_download_success": "ดาวน์โหลดสำเร็จ",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} จะไม่สามารถเข้าถึงรูปภาพของคุณ",
"partner_page_stop_sharing_title": "หยุดแชร์รูปภาพ?",
"partner_page_title": "พันธมิตร",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "ดำเนินการต่อ",
"permission_onboarding_get_started": "เริ่มต้น",
"permission_onboarding_go_to_settings": "ไปยังการตั้งค่า",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich จำเป็นจะต้องได้รับสิทธิ์ดูรูปภาพและวิดีโอ",
"profile_drawer_app_logs": "Log",
"profile_drawer_client_server_up_to_date": "ไคลเอนต์และเซิร์ฟเวอร์เป็นปัจจุบัน",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Settings",
"profile_drawer_sign_out": "ออกจากระบบ",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "สร้างอัลบั้มล้มเหลว",
"select_user_for_sharing_page_share_suggestions": "Suggestions",
"server_info_box_app_version": "เวอร์ชั่นแอพ",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "เวอร์ชั้นเซิร์ฟเวอร์",
"setting_image_viewer_help": "The detail viewer loads the small thumbnail first, then loads the medium-size preview (if enabled), finally loads the original (if enabled).",
"setting_image_viewer_original_subtitle": "Enable to load the original full-resolution image (large!). Disable to reduce data usage (both network and on device cache).",
@@ -300,6 +315,12 @@
"share_add_photos": "เพิ่มรูปภาพ",
"share_add_title": "เพิ่มชื่อ",
"share_create_album": "สร้างอัลบั้ม",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "กำลังเตรียม...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/uk-UA.json b/mobile/assets/i18n/uk-UA.json
index 6a733c4f1..d3df92692 100644
--- a/mobile/assets/i18n/uk-UA.json
+++ b/mobile/assets/i18n/uk-UA.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Не вдалося змінити назву альбому",
"album_viewer_appbar_share_leave": "Вийти з альбому",
"album_viewer_appbar_share_remove": "Видалити з альбому",
+ "album_viewer_appbar_share_to": "Share To",
"album_viewer_page_share_add_users": "Додати користувачів",
"all_people_page_title": "Люди",
"all_videos_page_title": "Відео",
+ "app_bar_signout_dialog_content": "Are you sure you want to sign out?",
+ "app_bar_signout_dialog_ok": "Yes",
+ "app_bar_signout_dialog_title": "Sign out",
"archive_page_no_archived_assets": "Немає архівних елементів",
"archive_page_title": "Архів ({})",
"asset_list_layout_settings_dynamic_layout_title": "Динамічне компонування",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Використання кешу",
"cache_settings_subtitle": "Контролює кешування у мобільному застосунку",
"cache_settings_thumbnail_size": "Розмір кешованих мініатюр ({} елементи)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "Local Storage",
"cache_settings_title": "Налаштування Кешування",
"change_password_form_confirm_password": "Підтвердити пароль",
- "change_password_form_description": "Привіт {firstName} {lastName},\n\nВи або або вперше входите у систему, або було зроблено запит на зміну вашого пароля. \nВведіть ваш новий пароль.",
+ "change_password_form_description": "Привіт {name},\n\nВи або або вперше входите у систему, або було зроблено запит на зміну вашого пароля. \nВведіть ваш новий пароль.",
"change_password_form_new_password": "Новий Пароль",
"change_password_form_password_mismatch": "Паролі не співпадають",
"change_password_form_reenter_new_password": "Повторіть Новий Пароль",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Додано {added} елементів у альбом {album}. {failed} елементів вже було в альбомі.",
"home_page_add_to_album_err_local": "Неможливо додати локальні елементи до альбомів, пропущено",
"home_page_add_to_album_success": "Додано {added} елементів у альбом {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Поки що неможливо заархівувати локальні елементи, пропущено",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Побудова хронології",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Неможливо отримати улюблені локальні елементи, пропущено",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "Якщо ви вперше користуєтеся програмою, переконайтеся, що ви вибрали альбоми для резервування, щоб могти заповнювати хронологію знімків та відео в альбомах.",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Можна вантажити не більше 30 елементів водночас, пропущено",
"image_viewer_page_state_provider_download_error": "Помилка завантаження",
"image_viewer_page_state_provider_download_success": "Усіпшно завантажено",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} втратить доступ до ваших знімків.",
"partner_page_stop_sharing_title": "Припинити надання ваших знімків?",
"partner_page_title": "Партнер",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Все одно продовжити",
"permission_onboarding_get_started": "Розпочати",
"permission_onboarding_go_to_settings": "Перейти до налаштувань",
@@ -250,6 +262,8 @@
"permission_onboarding_request": "Immich потребує доступу до ваших знімків та відео.",
"profile_drawer_app_logs": "Журнал",
"profile_drawer_client_server_up_to_date": "Клієнт та Сервер — актуальні",
+ "profile_drawer_documentation": "Documentation",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Налаштування",
"profile_drawer_sign_out": "Вийти",
"profile_drawer_trash": "Trash",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Не вдалося створити альбом",
"select_user_for_sharing_page_share_suggestions": "Suggestions",
"server_info_box_app_version": "App Version",
+ "server_info_box_server_url": "Server URL",
"server_info_box_server_version": "Server Version",
"setting_image_viewer_help": "The detail viewer loads the small thumbnail first, then loads the medium-size preview (if enabled), finally loads the original (if enabled).",
"setting_image_viewer_original_subtitle": "Enable to load the original full-resolution image (large!). Disable to reduce data usage (both network and on device cache).",
@@ -300,6 +315,12 @@
"share_add_photos": "Додати знімки",
"share_add_title": "Додати назву",
"share_create_album": "Створити альбом",
+ "shared_album_activities_input_disable": "Comment is disabled",
+ "shared_album_activities_input_hint": "Say something",
+ "shared_album_activity_remove_content": "Do you want to delete this activity?",
+ "shared_album_activity_remove_title": "Delete Activity",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "Comments & likes",
"share_dialog_preparing": "Підготовка...",
"shared_link_app_bar_title": "Shared Links",
"shared_link_create_app_bar_title": "Create link to share",
@@ -311,6 +332,9 @@
"shared_link_edit_change_expiry": "Change expiration time",
"shared_link_edit_description": "Description",
"shared_link_edit_description_hint": "Enter the share description",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "Password",
+ "shared_link_edit_password_hint": "Enter the share password",
"shared_link_edit_show_meta": "Show metadata",
"shared_link_edit_submit_button": "Update link",
"shared_link_empty": "You don't have any shared links",
diff --git a/mobile/assets/i18n/vi-VN.json b/mobile/assets/i18n/vi-VN.json
index 340efef7d..60edb10b4 100644
--- a/mobile/assets/i18n/vi-VN.json
+++ b/mobile/assets/i18n/vi-VN.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "Failed to change album title",
"album_viewer_appbar_share_leave": "Leave album",
"album_viewer_appbar_share_remove": "Remove from album",
+ "album_viewer_appbar_share_to": "Chia sẻ với",
"album_viewer_page_share_add_users": "Add users",
- "all_people_page_title": "People",
+ "all_people_page_title": "Mọi người",
"all_videos_page_title": "Videos",
+ "app_bar_signout_dialog_content": "Bạn có muốn đăng xuất?",
+ "app_bar_signout_dialog_ok": "Có",
+ "app_bar_signout_dialog_title": "Đăng xuất",
"archive_page_no_archived_assets": "No archived assets found",
"archive_page_title": "Archive ({})",
"asset_list_layout_settings_dynamic_layout_title": "Dynamic layout",
@@ -69,7 +73,7 @@
"backup_controller_page_backup": "Backup",
"backup_controller_page_backup_selected": "Selected: ",
"backup_controller_page_backup_sub": "Backed up photos and videos",
- "backup_controller_page_cancel": "Cancel",
+ "backup_controller_page_cancel": "Từ chối",
"backup_controller_page_created": "Created on: {}",
"backup_controller_page_desc_backup": "Turn on foreground backup to automatically upload new assets to the server when opening the app.",
"backup_controller_page_excluded": "Excluded: ",
@@ -95,10 +99,10 @@
"backup_err_only_album": "Cannot remove the only album",
"backup_info_card_assets": "assets",
"backup_manual_cancelled": "Cancelled",
- "backup_manual_failed": "Failed",
+ "backup_manual_failed": "Thất bại",
"backup_manual_in_progress": "Upload already in progress. Try after sometime",
- "backup_manual_success": "Success",
- "backup_manual_title": "Upload status",
+ "backup_manual_success": "Thành công",
+ "backup_manual_title": "Trạng thái tải lên",
"cache_settings_album_thumbnails": "Library page thumbnails ({} assets)",
"cache_settings_clear_cache_button": "Clear cache",
"cache_settings_clear_cache_button_title": "Clears the app's cache. This will significantly impact the app's performance until the cache has rebuilt.",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "Cache usage",
"cache_settings_subtitle": "Control the caching behaviour of the Immich mobile application",
"cache_settings_thumbnail_size": "Thumbnail cache size ({} assets)",
+ "cache_settings_tile_subtitle": "Kiểm soát cách xử lý lưu trữ cục bộ",
+ "cache_settings_tile_title": "Lưu trữ cục bộ",
"cache_settings_title": "Caching Settings",
"change_password_form_confirm_password": "Confirm Password",
- "change_password_form_description": "Hi {firstName} {lastName},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
+ "change_password_form_description": "Hi {name},\n\nThis is either the first time you are signing into the system or a request has been made to change your password. Please enter the new password below.",
"change_password_form_new_password": "New Password",
"change_password_form_password_mismatch": "Passwords do not match",
"change_password_form_reenter_new_password": "Re-enter New Password",
@@ -130,10 +136,10 @@
"control_bottom_app_bar_delete": "Delete",
"control_bottom_app_bar_favorite": "Favorite",
"control_bottom_app_bar_share": "Share",
- "control_bottom_app_bar_share_to": "Share To",
+ "control_bottom_app_bar_share_to": "Chia sẻ với",
"control_bottom_app_bar_stack": "Stack",
"control_bottom_app_bar_unarchive": "Unarchive",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "Tải lên",
"create_album_page_untitled": "Untitled",
"create_shared_album_page_create": "Create",
"create_shared_album_page_share": "Share",
@@ -145,11 +151,11 @@
"daily_title_text_date_year": "E, MMM dd, yyyy",
"date_format": "E, LLL d, y • h:mm a",
"delete_dialog_alert": "These items will be permanently deleted from Immich and from your device",
- "delete_dialog_cancel": "Cancel",
+ "delete_dialog_cancel": "Từ chối",
"delete_dialog_ok": "Delete",
"delete_dialog_title": "Delete Permanently",
- "delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_content": "Bạn có muốn xóa liên kết đã chia sẻ này không?",
+ "delete_shared_link_dialog_title": "Xoá liên kết đã chia sẻ",
"description_input_hint_text": "Add description...",
"description_input_submit_error": "Error updating description, check the log for more details",
"exif_bottom_sheet_description": "Add Description...",
@@ -164,10 +170,15 @@
"home_page_add_to_album_conflicts": "Added {added} assets to album {album}. {failed} assets are already in the album.",
"home_page_add_to_album_err_local": "Can not add local assets to albums yet, skipping",
"home_page_add_to_album_success": "Added {added} assets to album {album}.",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "Can not archive local assets yet, skipping",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "Building the timeline",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "Can not favorite local assets yet, skipping",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "If this is your first time using the app, please make sure to choose a backup album(s) so that the timeline can populate photos and videos in the album(s).",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "Can only upload a maximum of 30 assets at a time, skipping",
"image_viewer_page_state_provider_download_error": "Download Error",
"image_viewer_page_state_provider_download_success": "Download Success",
@@ -179,8 +190,8 @@
"library_page_new_album": "New album",
"library_page_sharing": "Sharing",
"library_page_sort_created": "Most recently created",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "Sửa đổi lần cuối",
+ "library_page_sort_most_recent_photo": "Ảnh gần đây nhất",
"library_page_sort_title": "Album title",
"login_disabled": "Login has been disabled",
"login_form_api_exception": "API exception. Please check the server URL and try again.",
@@ -199,32 +210,32 @@
"login_form_handshake_exception": "There was an Handshake Exception with the server. Enable self-signed certificate support in the settings if you are using a self-signed certificate.",
"login_form_label_email": "Email",
"login_form_label_password": "Password",
- "login_form_next_button": "Next",
+ "login_form_next_button": "Tiếp tục",
"login_form_password_hint": "password",
"login_form_save_login": "Stay logged in",
"login_form_server_empty": "Enter a server URL.",
"login_form_server_error": "Could not connect to server.",
"login_password_changed_error": "There was an error updating your password",
- "login_password_changed_success": "Password updated successfully",
+ "login_password_changed_success": "Cập nhật mật khẩu thành công",
"map_cannot_get_user_location": "Cannot get user's location",
- "map_location_dialog_cancel": "Cancel",
- "map_location_dialog_yes": "Yes",
+ "map_location_dialog_cancel": "Từ chối",
+ "map_location_dialog_yes": "Có",
"map_location_service_disabled_content": "Location service needs to be enabled to display assets from your current location. Do you want to enable it now?",
"map_location_service_disabled_title": "Location Service disabled",
"map_no_assets_in_bounds": "No photos in this area",
"map_no_location_permission_content": "Location permission is needed to display assets from your current location. Do you want to allow it now?",
"map_no_location_permission_title": "Location Permission denied",
- "map_settings_dark_mode": "Dark mode",
- "map_settings_dialog_cancel": "Cancel",
- "map_settings_dialog_save": "Save",
- "map_settings_dialog_title": "Map Settings",
+ "map_settings_dark_mode": "Chế độ tối",
+ "map_settings_dialog_cancel": "Từ chối",
+ "map_settings_dialog_save": "Lưu",
+ "map_settings_dialog_title": "Cài đặt bản đồ",
"map_settings_include_show_archived": "Include Archived",
"map_settings_only_relative_range": "Date range",
"map_settings_only_show_favorites": "Show Favorite Only",
"map_zoom_to_see_photos": "Zoom out to see photos",
"monthly_title_text_date_format": "MMMM y",
"motion_photos_page_title": "Motion Photos",
- "notification_permission_dialog_cancel": "Cancel",
+ "notification_permission_dialog_cancel": "Từ chối",
"notification_permission_dialog_content": "To enable notifications, go to Settings and select allow.",
"notification_permission_dialog_settings": "Settings",
"notification_permission_list_tile_content": "Grant permission to enable notifications.",
@@ -235,10 +246,11 @@
"partner_page_no_more_users": "No more users to add",
"partner_page_partner_add_failed": "Failed to add partner",
"partner_page_select_partner": "Select partner",
- "partner_page_shared_to_title": "Shared to",
+ "partner_page_shared_to_title": "Chia sẻ với",
"partner_page_stop_sharing_content": "{} will no longer be able to access your photos.",
"partner_page_stop_sharing_title": "Stop sharing your photos?",
"partner_page_title": "Partner",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "Continue anyway",
"permission_onboarding_get_started": "Get started",
"permission_onboarding_go_to_settings": "Go to settings",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich requires permission to view your photos and videos.",
"profile_drawer_app_logs": "Logs",
"profile_drawer_client_server_up_to_date": "Client and Server are up-to-date",
+ "profile_drawer_documentation": "Tài liệu",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "Settings",
"profile_drawer_sign_out": "Sign Out",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "Thùng rác",
"recently_added_page_title": "Recently Added",
"search_bar_hint": "Search your photos",
"search_page_categories": "Categories",
@@ -260,7 +274,7 @@
"search_page_motion_photos": "Motion Photos",
"search_page_no_objects": "Không có thông tin vật thể",
"search_page_no_places": "No Places Info Available",
- "search_page_people": "People",
+ "search_page_people": "Mọi người",
"search_page_places": "Places",
"search_page_recently_added": "Recently added",
"search_page_screenshots": "Screenshots",
@@ -276,6 +290,7 @@
"select_user_for_sharing_page_err_album": "Failed to create album",
"select_user_for_sharing_page_share_suggestions": "Gợi ý",
"server_info_box_app_version": "App Version",
+ "server_info_box_server_url": "Địa chỉ máy chủ",
"server_info_box_server_version": "Server Version",
"setting_image_viewer_help": "The detail viewer loads the small thumbnail first, then loads the medium-size preview (if enabled), finally loads the original (if enabled).",
"setting_image_viewer_original_subtitle": "Enable to load the original full-resolution image (large!). Disable to reduce data usage (both network and on device cache).",
@@ -300,28 +315,37 @@
"share_add_photos": "Add photos",
"share_add_title": "Add a title",
"share_create_album": "Create album",
+ "shared_album_activities_input_disable": "Nhận xét hiện đã tắt",
+ "shared_album_activities_input_hint": "Nói điều gì đó",
+ "shared_album_activity_remove_content": "Bạn có muốn xoá hoạt động này?",
+ "shared_album_activity_remove_title": "Xoá hoạt động",
+ "shared_album_activity_setting_subtitle": "Cho phép người khác phản hồi",
+ "shared_album_activity_setting_title": "Bình luận và lượt thích",
"share_dialog_preparing": "Preparing...",
- "shared_link_app_bar_title": "Shared Links",
- "shared_link_create_app_bar_title": "Create link to share",
- "shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
- "shared_link_edit_allow_download": "Allow public user to download",
- "shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
- "shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
- "shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_app_bar_title": "Đường liên kết chia sẻ",
+ "shared_link_create_app_bar_title": "Tạo liên kết để chia sẻ",
+ "shared_link_create_info": "Cho phép bất cứ ai có liên kết xem (các) ảnh đã chọn",
+ "shared_link_create_submit_button": "Tạo liên kết",
+ "shared_link_edit_allow_download": "Cho phép bất cứ ai đều có thể tải xuống",
+ "shared_link_edit_allow_upload": "Cho phép bất cứ ai đều có thể tải lên",
+ "shared_link_edit_app_bar_title": "Chỉnh sửa liên kết",
+ "shared_link_edit_change_expiry": "Thay đổi thời gian hết hạn",
+ "shared_link_edit_description": "Mô tả",
+ "shared_link_edit_description_hint": "Nhập mô tả chia sẻ",
+ "shared_link_edit_expire_after": "Hết hạn sau",
+ "shared_link_edit_password": "Mật khẩu",
+ "shared_link_edit_password_hint": "Nhập mật khẩu chia sẻ",
+ "shared_link_edit_show_meta": "Hiện thị siêu dữ liệu",
+ "shared_link_edit_submit_button": "Cập nhật liên kết",
+ "shared_link_empty": "Bạn không có liên kết được chia sẻ nào",
+ "shared_link_manage_links": "Quản lý liên kết được chia sẻ",
+ "share_done": "Hoàn tất",
"share_invite": "Invite to album",
"sharing_page_album": "Shared albums",
"sharing_page_description": "Create shared albums to share photos and videos with people in your network.",
"sharing_page_empty_list": "EMPTY LIST",
"sharing_silver_appbar_create_shared_album": "Create shared album",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "Các liên kết chia sẻ",
"sharing_silver_appbar_share_partner": "Share with partner",
"tab_controller_nav_library": "Library",
"tab_controller_nav_photos": "Photos",
@@ -337,22 +361,22 @@
"theme_setting_theme_title": "Theme",
"theme_setting_three_stage_loading_subtitle": "Three-stage loading might increase the loading performance but causes significantly higher network load",
"theme_setting_three_stage_loading_title": "Enable three-stage loading",
- "translated_text_options": "Options",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
- "trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
- "trash_page_empty_trash_dialog_ok": "Ok",
- "trash_page_info": "Trashed items will be permanently deleted after {} days",
- "trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
- "trash_page_title": "Trash ({})",
- "upload_dialog_cancel": "Cancel",
+ "translated_text_options": "Tuỳ chỉnh",
+ "trash_page_delete": "Xoá",
+ "trash_page_delete_all": "Xoá tất cả",
+ "trash_page_empty_trash_btn": "Dọn sạch thùng rác",
+ "trash_page_empty_trash_dialog_content": "Bạn có muốn dọn sạch thùng rác của mình không? Những mục này sẽ bị xoá vĩnh viễn khỏi Immich",
+ "trash_page_empty_trash_dialog_ok": "Đồng ý",
+ "trash_page_info": "Những mục này sẽ bị xoá sau {} ngày",
+ "trash_page_no_assets": "Không có ảnh hoặc video",
+ "trash_page_restore": "Khôi phục",
+ "trash_page_restore_all": "Khôi phục tất cả",
+ "trash_page_select_assets_btn": "Chọn mục",
+ "trash_page_select_btn": "Chọn",
+ "trash_page_title": "Thùng rác ({})",
+ "upload_dialog_cancel": "Từ chối",
"upload_dialog_info": "Do you want to backup the selected Asset(s) to the server?",
- "upload_dialog_ok": "Upload",
+ "upload_dialog_ok": "Tải lên",
"upload_dialog_title": "Upload Asset",
"version_announcement_overlay_ack": "Acknowledge",
"version_announcement_overlay_release_notes": "release notes",
diff --git a/mobile/assets/i18n/zh-CN.json b/mobile/assets/i18n/zh-CN.json
index 6ddf38545..28591e395 100644
--- a/mobile/assets/i18n/zh-CN.json
+++ b/mobile/assets/i18n/zh-CN.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "修改相册标题失败",
"album_viewer_appbar_share_leave": "退出共享",
"album_viewer_appbar_share_remove": "从相册中移除",
+ "album_viewer_appbar_share_to": "共享给",
"album_viewer_page_share_add_users": "创建用户",
"all_people_page_title": "人物",
"all_videos_page_title": "视频",
+ "app_bar_signout_dialog_content": "您确定要注消吗?",
+ "app_bar_signout_dialog_ok": "是",
+ "app_bar_signout_dialog_title": "注消",
"archive_page_no_archived_assets": "未找到归档项目",
"archive_page_title": "归档({})",
"asset_list_layout_settings_dynamic_layout_title": "动态布局",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "缓存使用情况",
"cache_settings_subtitle": "控制 Immich 的缓存行为",
"cache_settings_thumbnail_size": "缩略图缓存大小({} 项)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "本地存储",
"cache_settings_title": "缓存设置",
"change_password_form_confirm_password": "确认密码",
- "change_password_form_description": "{firstName} {lastName} 您好,\n\n这是您首次登录系统,或被管理员要求更改密码。\n请在下方输入新密码。",
+ "change_password_form_description": "{name} 您好,\n\n这是您首次登录系统,或被管理员要求更改密码。\n请在下方输入新密码。",
"change_password_form_new_password": "新密码",
"change_password_form_password_mismatch": "密码不匹配",
"change_password_form_reenter_new_password": "重新输入新的密码",
@@ -130,10 +136,10 @@
"control_bottom_app_bar_delete": "删除",
"control_bottom_app_bar_favorite": "收藏",
"control_bottom_app_bar_share": "共享",
- "control_bottom_app_bar_share_to": "Share To",
+ "control_bottom_app_bar_share_to": "共享给",
"control_bottom_app_bar_stack": "Stack",
"control_bottom_app_bar_unarchive": "取消归档",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "上传",
"create_album_page_untitled": "未命名",
"create_shared_album_page_create": "创建",
"create_shared_album_page_share": "共享",
@@ -149,7 +155,7 @@
"delete_dialog_ok": "删除",
"delete_dialog_title": "永久删除",
"delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_title": "删除共享链接",
"description_input_hint_text": "添加描述...",
"description_input_submit_error": "更新描述时出错,请检查日志以获取更多详细信息",
"exif_bottom_sheet_description": "添加描述...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "已向相册 {album} 中添加 {added} 项。\n其中 {failed} 项在相册中已存在。",
"home_page_add_to_album_err_local": "暂不能将本地项目添加到相册中,跳过",
"home_page_add_to_album_success": "已向相册 {album} 中添加 {added} 项。",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "暂无法归档本地项目,跳过",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "正在生成时间线",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "暂不能收藏本地项目,跳过",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "如果这是您第一次使用该应用程序,请确保选择一个要备份的本地相册,以便可以在时间线中预览该相册中的照片和视频。",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "一次最多只能上传 30 个项目,跳过",
"image_viewer_page_state_provider_download_error": "下载出现错误",
"image_viewer_page_state_provider_download_success": "下载成功",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "共享错误",
"library_page_albums": "相册",
"library_page_archive": "归档",
"library_page_device_albums": "设备上的相册",
@@ -179,14 +190,14 @@
"library_page_new_album": "新建相册",
"library_page_sharing": "共享",
"library_page_sort_created": "最近创建的",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "上次修改",
+ "library_page_sort_most_recent_photo": "最近的项目",
"library_page_sort_title": "相册标题",
"login_disabled": "登录已被禁用",
"login_form_api_exception": "API 异常,请检查服务器地址并重试。",
"login_form_button_text": "登录",
"login_form_email_hint": "youremail@email.com",
- "login_form_endpoint_hint": "http(s)://你的服务器地址:端口/api",
+ "login_form_endpoint_hint": "http(s)://您的服务器地址:端口/api",
"login_form_endpoint_url": "服务器链接地址",
"login_form_err_http": "请注明 http:// 或 https://",
"login_form_err_invalid_email": "无效的电子邮件",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} 将无法再访问您的照片。",
"partner_page_stop_sharing_title": "您确定要停止共享您的照片吗?",
"partner_page_title": "同伴",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "仍然继续",
"permission_onboarding_get_started": "开始使用",
"permission_onboarding_go_to_settings": "转到设置",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich 需要权限才能查看您的照片和视频。",
"profile_drawer_app_logs": "日志",
"profile_drawer_client_server_up_to_date": "客户端和服务端都是最新的",
+ "profile_drawer_documentation": "文档",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "设置",
"profile_drawer_sign_out": "退出登录",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "回收站",
"recently_added_page_title": "最近添加",
"search_bar_hint": "搜索照片",
"search_page_categories": "类别",
@@ -271,11 +285,12 @@
"search_page_your_activity": "您的活动",
"search_result_page_new_search_hint": "搜索新的",
"search_suggestion_list_smart_search_hint_1": "默认情况下启用智能搜索;要搜索元数据,请使用相关语法",
- "search_suggestion_list_smart_search_hint_2": "m:你的搜索关键词",
+ "search_suggestion_list_smart_search_hint_2": "m:您的搜索关键词",
"select_additional_user_for_sharing_page_suggestions": "建议",
"select_user_for_sharing_page_err_album": "创建相册失败",
"select_user_for_sharing_page_share_suggestions": "建议",
"server_info_box_app_version": "App 版本",
+ "server_info_box_server_url": "服务器地址",
"server_info_box_server_version": "服务器版本",
"setting_image_viewer_help": "详细信息查看器首先加载小缩略图,然后加载中等大小的预览图(若启用),最后加载原始图像。",
"setting_image_viewer_original_subtitle": "启用以加载原图,禁用以减少数据使用量(网络和设备缓存)。",
@@ -300,28 +315,37 @@
"share_add_photos": "添加项目",
"share_add_title": "添加标题",
"share_create_album": "创建相册",
+ "shared_album_activities_input_disable": "评论已禁用",
+ "shared_album_activities_input_hint": "说些什么",
+ "shared_album_activity_remove_content": "您确定要删除此活动吗?",
+ "shared_album_activity_remove_title": "删除活动",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "评论与喜欢",
"share_dialog_preparing": "正在准备...",
- "shared_link_app_bar_title": "Shared Links",
+ "shared_link_app_bar_title": "共享链接",
"shared_link_create_app_bar_title": "Create link to share",
"shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
+ "shared_link_create_submit_button": "创建链接",
"shared_link_edit_allow_download": "Allow public user to download",
"shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
+ "shared_link_edit_app_bar_title": "编辑链接",
+ "shared_link_edit_change_expiry": "修改过期时间",
+ "shared_link_edit_description": "描述",
"shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "密码",
+ "shared_link_edit_password_hint": "输入共享密码",
+ "shared_link_edit_show_meta": "显示元数据",
+ "shared_link_edit_submit_button": "更新链接",
"shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_manage_links": "管理共享链接",
+ "share_done": "完成",
"share_invite": "邀请相册共享",
"sharing_page_album": "共享相册",
"sharing_page_description": "创建共享相册以与网络中的人共享照片和视频。",
"sharing_page_empty_list": "空",
"sharing_silver_appbar_create_shared_album": "创建共享相册",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "共享链接",
"sharing_silver_appbar_share_partner": "共享给同伴",
"tab_controller_nav_library": "图库",
"tab_controller_nav_photos": "照片",
@@ -338,17 +362,17 @@
"theme_setting_three_stage_loading_subtitle": "三段式加载可能会提升加载性能,但可能会导致更高的网络负载",
"theme_setting_three_stage_loading_title": "启用三段式加载",
"translated_text_options": "选项",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
+ "trash_page_delete": "删除",
+ "trash_page_delete_all": "删除全部",
+ "trash_page_empty_trash_btn": "清空回收站",
"trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
- "trash_page_empty_trash_dialog_ok": "Ok",
+ "trash_page_empty_trash_dialog_ok": "好的",
"trash_page_info": "Trashed items will be permanently deleted after {} days",
"trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
+ "trash_page_restore": "恢复",
+ "trash_page_restore_all": "恢复全部",
+ "trash_page_select_assets_btn": "选择项目",
+ "trash_page_select_btn": "选择",
"trash_page_title": "Trash ({})",
"upload_dialog_cancel": "取消",
"upload_dialog_info": "是否要将所选项目备份到服务器?",
diff --git a/mobile/assets/i18n/zh-Hans.json b/mobile/assets/i18n/zh-Hans.json
index 59d67bd66..cc8b89a15 100644
--- a/mobile/assets/i18n/zh-Hans.json
+++ b/mobile/assets/i18n/zh-Hans.json
@@ -23,9 +23,13 @@
"album_viewer_appbar_share_err_title": "修改相册标题失败",
"album_viewer_appbar_share_leave": "退出共享",
"album_viewer_appbar_share_remove": "从相册中移除",
+ "album_viewer_appbar_share_to": "共享给",
"album_viewer_page_share_add_users": "创建用户",
"all_people_page_title": "人物",
"all_videos_page_title": "视频",
+ "app_bar_signout_dialog_content": "您确定要注消吗?",
+ "app_bar_signout_dialog_ok": "是",
+ "app_bar_signout_dialog_title": "注消",
"archive_page_no_archived_assets": "未找到归档项目",
"archive_page_title": "归档({})",
"asset_list_layout_settings_dynamic_layout_title": "动态布局",
@@ -111,9 +115,11 @@
"cache_settings_statistics_title": "缓存使用情况",
"cache_settings_subtitle": "控制 Immich 的缓存行为",
"cache_settings_thumbnail_size": "缩略图缓存大小({} 项)",
+ "cache_settings_tile_subtitle": "Control the local storage behaviour",
+ "cache_settings_tile_title": "本地存储",
"cache_settings_title": "缓存设置",
"change_password_form_confirm_password": "确认密码",
- "change_password_form_description": "{firstName} {lastName} 您好,\n\n这是您首次登录系统,或被管理员要求更改密码。\n请在下方输入新密码。",
+ "change_password_form_description": "{name} 您好,\n\n这是您首次登录系统,或被管理员要求更改密码。\n请在下方输入新密码。",
"change_password_form_new_password": "新密码",
"change_password_form_password_mismatch": "密码不匹配",
"change_password_form_reenter_new_password": "重新输入新的密码",
@@ -130,10 +136,10 @@
"control_bottom_app_bar_delete": "删除",
"control_bottom_app_bar_favorite": "收藏",
"control_bottom_app_bar_share": "共享",
- "control_bottom_app_bar_share_to": "Share To",
+ "control_bottom_app_bar_share_to": "共享给",
"control_bottom_app_bar_stack": "Stack",
"control_bottom_app_bar_unarchive": "取消归档",
- "control_bottom_app_bar_upload": "Upload",
+ "control_bottom_app_bar_upload": "上传",
"create_album_page_untitled": "未命名",
"create_shared_album_page_create": "创建",
"create_shared_album_page_share": "共享",
@@ -149,7 +155,7 @@
"delete_dialog_ok": "删除",
"delete_dialog_title": "永久删除",
"delete_shared_link_dialog_content": "Are you sure you want to delete this shared link?",
- "delete_shared_link_dialog_title": "Delete Shared Link",
+ "delete_shared_link_dialog_title": "删除共享链接",
"description_input_hint_text": "添加描述...",
"description_input_submit_error": "更新描述时出错,请检查日志以获取更多详细信息",
"exif_bottom_sheet_description": "添加描述...",
@@ -164,14 +170,19 @@
"home_page_add_to_album_conflicts": "已向相册 {album} 中添加 {added} 项。\n其中 {failed} 项在相册中已存在。",
"home_page_add_to_album_err_local": "暂不能将本地资项目添加到相册中,跳过",
"home_page_add_to_album_success": "已向相册 {album} 中添加 {added} 项。",
+ "home_page_album_err_partner": "Can not add partner assets to an album yet, skipping",
"home_page_archive_err_local": "暂无法归档本地项目,跳过",
+ "home_page_archive_err_partner": "Can not archive partner assets, skipping",
"home_page_building_timeline": "正在生成时间线",
+ "home_page_delete_err_partner": "Can not delete partner assets, skipping",
"home_page_favorite_err_local": "暂不能收藏本地项目,跳过",
+ "home_page_favorite_err_partner": "Can not favorite partner assets yet, skipping",
"home_page_first_time_notice": "如果这是您第一次使用该应用程序,请确保选择一个要备份的本地相册,以便可以在时间线中预览该相册中的照片和视频。",
+ "home_page_share_err_local": "Can not share local assets via link, skipping",
"home_page_upload_err_limit": "一次最多只能上传 30 个项目,跳过",
"image_viewer_page_state_provider_download_error": "下载出现错误",
"image_viewer_page_state_provider_download_success": "下载成功",
- "image_viewer_page_state_provider_share_error": "Share Error",
+ "image_viewer_page_state_provider_share_error": "共享错误",
"library_page_albums": "相册",
"library_page_archive": "归档",
"library_page_device_albums": "设备上的相册",
@@ -179,14 +190,14 @@
"library_page_new_album": "新建相册",
"library_page_sharing": "共享",
"library_page_sort_created": "最近创建的",
- "library_page_sort_last_modified": "Last modified",
- "library_page_sort_most_recent_photo": "Most recent photo",
+ "library_page_sort_last_modified": "上次修改",
+ "library_page_sort_most_recent_photo": "最近的项目",
"library_page_sort_title": "相册标题",
"login_disabled": "登录已被禁用",
"login_form_api_exception": "API 异常,请检查服务器地址并重试。",
"login_form_button_text": "登录",
"login_form_email_hint": "youremail@email.com",
- "login_form_endpoint_hint": "http(s)://你的服务器地址:端口/api",
+ "login_form_endpoint_hint": "http(s)://您的服务器地址:端口/api",
"login_form_endpoint_url": "服务器链接地址",
"login_form_err_http": "请注明 http:// 或 https://",
"login_form_err_invalid_email": "无效的电子邮件",
@@ -239,6 +250,7 @@
"partner_page_stop_sharing_content": "{} 将无法再访问您的照片。",
"partner_page_stop_sharing_title": "您确定要停止共享您的照片吗?",
"partner_page_title": "同伴",
+ "permission_onboarding_back": "Back",
"permission_onboarding_continue_anyway": "仍然继续",
"permission_onboarding_get_started": "开始使用",
"permission_onboarding_go_to_settings": "转到设置",
@@ -250,9 +262,11 @@
"permission_onboarding_request": "Immich 需要权限才能查看您的照片和视频。",
"profile_drawer_app_logs": "日志",
"profile_drawer_client_server_up_to_date": "客户端和服务端都是最新的",
+ "profile_drawer_documentation": "文档",
+ "profile_drawer_github": "GitHub",
"profile_drawer_settings": "设置",
"profile_drawer_sign_out": "退出登录",
- "profile_drawer_trash": "Trash",
+ "profile_drawer_trash": "回收站",
"recently_added_page_title": "最近添加",
"search_bar_hint": "搜索照片",
"search_page_categories": "类别",
@@ -271,11 +285,12 @@
"search_page_your_activity": "您的活动",
"search_result_page_new_search_hint": "搜索新的",
"search_suggestion_list_smart_search_hint_1": "默认情况下启用智能搜索;要搜索元数据,请使用相关语法",
- "search_suggestion_list_smart_search_hint_2": "m:你的搜索关键词",
+ "search_suggestion_list_smart_search_hint_2": "m:您的搜索关键词",
"select_additional_user_for_sharing_page_suggestions": "建议",
"select_user_for_sharing_page_err_album": "创建相册失败",
"select_user_for_sharing_page_share_suggestions": "建议",
"server_info_box_app_version": "App 版本",
+ "server_info_box_server_url": "服务器地址",
"server_info_box_server_version": "服务器版本",
"setting_image_viewer_help": "详细信息查看器首先加载小缩略图,然后加载中等大小的预览图(若启用),最后加载原始图像。",
"setting_image_viewer_original_subtitle": "启用以加载原图,禁用以减少数据使用量(网络和设备缓存)。",
@@ -300,28 +315,37 @@
"share_add_photos": "添加项目",
"share_add_title": "添加标题",
"share_create_album": "创建相册",
+ "shared_album_activities_input_disable": "评论已禁用",
+ "shared_album_activities_input_hint": "说些什么",
+ "shared_album_activity_remove_content": "您确定要删除此活动吗?",
+ "shared_album_activity_remove_title": "删除活动",
+ "shared_album_activity_setting_subtitle": "Let others respond",
+ "shared_album_activity_setting_title": "评论与喜欢",
"share_dialog_preparing": "正在准备...",
- "shared_link_app_bar_title": "Shared Links",
+ "shared_link_app_bar_title": "共享链接",
"shared_link_create_app_bar_title": "Create link to share",
"shared_link_create_info": "Let anyone with the link see the selected photo(s)",
- "shared_link_create_submit_button": "Create link",
+ "shared_link_create_submit_button": "创建链接",
"shared_link_edit_allow_download": "Allow public user to download",
"shared_link_edit_allow_upload": "Allow public user to upload",
- "shared_link_edit_app_bar_title": "Edit link",
- "shared_link_edit_change_expiry": "Change expiration time",
- "shared_link_edit_description": "Description",
+ "shared_link_edit_app_bar_title": "编辑链接",
+ "shared_link_edit_change_expiry": "修改过期时间",
+ "shared_link_edit_description": "描述",
"shared_link_edit_description_hint": "Enter the share description",
- "shared_link_edit_show_meta": "Show metadata",
- "shared_link_edit_submit_button": "Update link",
+ "shared_link_edit_expire_after": "Expire after",
+ "shared_link_edit_password": "密码",
+ "shared_link_edit_password_hint": "输入共享密码",
+ "shared_link_edit_show_meta": "显示元数据",
+ "shared_link_edit_submit_button": "更新链接",
"shared_link_empty": "You don't have any shared links",
- "shared_link_manage_links": "Manage Shared links",
- "share_done": "Done",
+ "shared_link_manage_links": "管理共享链接",
+ "share_done": "完成",
"share_invite": "邀请相册共享",
"sharing_page_album": "共享相册",
"sharing_page_description": "创建共享相册以与网络中的人共享照片和视频。",
"sharing_page_empty_list": "空",
"sharing_silver_appbar_create_shared_album": "创建共享相册",
- "sharing_silver_appbar_shared_links": "Shared links",
+ "sharing_silver_appbar_shared_links": "共享链接",
"sharing_silver_appbar_share_partner": "共享给同伴",
"tab_controller_nav_library": "图库",
"tab_controller_nav_photos": "照片",
@@ -338,17 +362,17 @@
"theme_setting_three_stage_loading_subtitle": "三段式加载可能会提升加载性能,但可能会导致更高的网络负载",
"theme_setting_three_stage_loading_title": "启用三段式加载",
"translated_text_options": "选项",
- "trash_page_delete": "Delete",
- "trash_page_delete_all": "Delete All",
- "trash_page_empty_trash_btn": "Empty trash",
+ "trash_page_delete": "删除",
+ "trash_page_delete_all": "删除全部",
+ "trash_page_empty_trash_btn": "清空回收站",
"trash_page_empty_trash_dialog_content": "Do you want to empty your trashed assets? These items will be permanently removed from Immich",
- "trash_page_empty_trash_dialog_ok": "Ok",
+ "trash_page_empty_trash_dialog_ok": "好的",
"trash_page_info": "Trashed items will be permanently deleted after {} days",
"trash_page_no_assets": "No trashed assets",
- "trash_page_restore": "Restore",
- "trash_page_restore_all": "Restore All",
- "trash_page_select_assets_btn": "Select assets",
- "trash_page_select_btn": "Select",
+ "trash_page_restore": "恢复",
+ "trash_page_restore_all": "恢复全部",
+ "trash_page_select_assets_btn": "选择项目",
+ "trash_page_select_btn": "选择",
"trash_page_title": "Trash ({})",
"upload_dialog_cancel": "取消",
"upload_dialog_info": "是否要将所选项目备份到服务器?",
diff --git a/mobile/fonts/WorkSans-Black.ttf b/mobile/fonts/WorkSans-Black.ttf
deleted file mode 100644
index f0f61fbde..000000000
Binary files a/mobile/fonts/WorkSans-Black.ttf and /dev/null differ
diff --git a/mobile/fonts/WorkSans-Bold.ttf b/mobile/fonts/WorkSans-Bold.ttf
deleted file mode 100644
index c30cb0708..000000000
Binary files a/mobile/fonts/WorkSans-Bold.ttf and /dev/null differ
diff --git a/mobile/fonts/WorkSans-ExtraBold.ttf b/mobile/fonts/WorkSans-ExtraBold.ttf
deleted file mode 100644
index 2d0d46a3a..000000000
Binary files a/mobile/fonts/WorkSans-ExtraBold.ttf and /dev/null differ
diff --git a/mobile/fonts/WorkSans-Italic.ttf b/mobile/fonts/WorkSans-Italic.ttf
deleted file mode 100644
index f2bf33102..000000000
Binary files a/mobile/fonts/WorkSans-Italic.ttf and /dev/null differ
diff --git a/mobile/fonts/WorkSans-Medium.ttf b/mobile/fonts/WorkSans-Medium.ttf
deleted file mode 100644
index 1800fe2d8..000000000
Binary files a/mobile/fonts/WorkSans-Medium.ttf and /dev/null differ
diff --git a/mobile/fonts/WorkSans-SemiBold.ttf b/mobile/fonts/WorkSans-SemiBold.ttf
deleted file mode 100644
index bce808c82..000000000
Binary files a/mobile/fonts/WorkSans-SemiBold.ttf and /dev/null differ
diff --git a/mobile/fonts/WorkSans.ttf b/mobile/fonts/WorkSans.ttf
deleted file mode 100644
index 09829a516..000000000
Binary files a/mobile/fonts/WorkSans.ttf and /dev/null differ
diff --git a/mobile/fonts/overpass/Overpass-Bold.ttf b/mobile/fonts/overpass/Overpass-Bold.ttf
new file mode 100644
index 000000000..0cd2fb107
Binary files /dev/null and b/mobile/fonts/overpass/Overpass-Bold.ttf differ
diff --git a/mobile/fonts/overpass/Overpass-Italic.ttf b/mobile/fonts/overpass/Overpass-Italic.ttf
new file mode 100644
index 000000000..1031a9e57
Binary files /dev/null and b/mobile/fonts/overpass/Overpass-Italic.ttf differ
diff --git a/mobile/fonts/overpass/Overpass-Medium.ttf b/mobile/fonts/overpass/Overpass-Medium.ttf
new file mode 100644
index 000000000..815276fe2
Binary files /dev/null and b/mobile/fonts/overpass/Overpass-Medium.ttf differ
diff --git a/mobile/fonts/overpass/Overpass-Regular.ttf b/mobile/fonts/overpass/Overpass-Regular.ttf
new file mode 100644
index 000000000..b89906141
Binary files /dev/null and b/mobile/fonts/overpass/Overpass-Regular.ttf differ
diff --git a/mobile/fonts/overpass/Overpass-SemiBold.ttf b/mobile/fonts/overpass/Overpass-SemiBold.ttf
new file mode 100644
index 000000000..b8faf9b99
Binary files /dev/null and b/mobile/fonts/overpass/Overpass-SemiBold.ttf differ
diff --git a/mobile/fonts/overpass/OverpassMono.ttf b/mobile/fonts/overpass/OverpassMono.ttf
new file mode 100644
index 000000000..4502d6123
Binary files /dev/null and b/mobile/fonts/overpass/OverpassMono.ttf differ
diff --git a/mobile/integration_test/test_utils/general_helper.dart b/mobile/integration_test/test_utils/general_helper.dart
index ac0b14ef4..8daa08d70 100644
--- a/mobile/integration_test/test_utils/general_helper.dart
+++ b/mobile/integration_test/test_utils/general_helper.dart
@@ -2,7 +2,9 @@ import 'dart:async';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter_test/flutter_test.dart';
+import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/shared/models/store.dart';
+import 'package:immich_mobile/shared/providers/db.provider.dart';
import 'package:integration_test/integration_test.dart';
import 'package:isar/isar.dart';
// ignore: depend_on_referenced_packages
@@ -40,7 +42,12 @@ class ImmichTestHelper {
await Store.clear();
await db.writeTxn(() => db.clear());
// Load main Widget
- await tester.pumpWidget(app.getMainWidget(db));
+ await tester.pumpWidget(
+ ProviderScope(
+ overrides: [dbProvider.overrideWithValue(db)],
+ child: const app.MainWidget(),
+ ),
+ );
// Post run tasks
await EasyLocalization.ensureInitialized();
}
diff --git a/mobile/ios/Podfile.lock b/mobile/ios/Podfile.lock
index c6c23d942..75168ce1c 100644
--- a/mobile/ios/Podfile.lock
+++ b/mobile/ios/Podfile.lock
@@ -169,4 +169,4 @@ SPEC CHECKSUMS:
PODFILE CHECKSUM: 599d8aeb73728400c15364e734525722250a5382
-COCOAPODS: 1.12.1
+COCOAPODS: 1.11.3
diff --git a/mobile/ios/Runner.xcodeproj/project.pbxproj b/mobile/ios/Runner.xcodeproj/project.pbxproj
index b15f480e2..c5c6ed57c 100644
--- a/mobile/ios/Runner.xcodeproj/project.pbxproj
+++ b/mobile/ios/Runner.xcodeproj/project.pbxproj
@@ -379,7 +379,7 @@
CODE_SIGN_ENTITLEMENTS = Runner/RunnerProfile.entitlements;
CODE_SIGN_IDENTITY = "Apple Development";
CODE_SIGN_STYLE = Automatic;
- CURRENT_PROJECT_VERSION = 124;
+ CURRENT_PROJECT_VERSION = 128;
DEVELOPMENT_TEAM = 2F67MQ8R79;
ENABLE_BITCODE = NO;
INFOPLIST_FILE = Runner/Info.plist;
@@ -515,7 +515,7 @@
CLANG_ENABLE_MODULES = YES;
CODE_SIGN_IDENTITY = "Apple Development";
CODE_SIGN_STYLE = Automatic;
- CURRENT_PROJECT_VERSION = 124;
+ CURRENT_PROJECT_VERSION = 128;
DEVELOPMENT_TEAM = 2F67MQ8R79;
ENABLE_BITCODE = NO;
INFOPLIST_FILE = Runner/Info.plist;
@@ -543,7 +543,7 @@
CLANG_ENABLE_MODULES = YES;
CODE_SIGN_IDENTITY = "Apple Development";
CODE_SIGN_STYLE = Automatic;
- CURRENT_PROJECT_VERSION = 124;
+ CURRENT_PROJECT_VERSION = 128;
DEVELOPMENT_TEAM = 2F67MQ8R79;
ENABLE_BITCODE = NO;
INFOPLIST_FILE = Runner/Info.plist;
diff --git a/mobile/ios/Runner/Info.plist b/mobile/ios/Runner/Info.plist
index a6fddc98a..2d3c9d1ee 100644
--- a/mobile/ios/Runner/Info.plist
+++ b/mobile/ios/Runner/Info.plist
@@ -43,15 +43,10 @@
mn
ko
sr
- sr
hi
- es
- es
- sv
ca
hu
lv
- zh
th
CFBundleName
@@ -59,11 +54,11 @@
CFBundlePackageType
APPL
CFBundleShortVersionString
- 1.84.0
+ 1.88.0
CFBundleSignature
????
CFBundleVersion
- 124
+ 128
FLTEnableImpeller
ITSAppUsesNonExemptEncryption
diff --git a/mobile/ios/fastlane/Fastfile b/mobile/ios/fastlane/Fastfile
index 19cefc12f..85b006ab4 100644
--- a/mobile/ios/fastlane/Fastfile
+++ b/mobile/ios/fastlane/Fastfile
@@ -19,7 +19,7 @@ platform :ios do
desc "iOS Beta"
lane :beta do
increment_version_number(
- version_number: "1.84.0"
+ version_number: "1.89.0"
)
increment_build_number(
build_number: latest_testflight_build_number + 1,
diff --git a/mobile/ios/fastlane/report.xml b/mobile/ios/fastlane/report.xml
index c61f1d5d2..947c79674 100644
--- a/mobile/ios/fastlane/report.xml
+++ b/mobile/ios/fastlane/report.xml
@@ -5,32 +5,32 @@
-
+
-
+
-
+
-
+
-
+
-
+
diff --git a/mobile/lib/constants/immich_colors.dart b/mobile/lib/constants/immich_colors.dart
index e3f601358..598f95661 100644
--- a/mobile/lib/constants/immich_colors.dart
+++ b/mobile/lib/constants/immich_colors.dart
@@ -1,5 +1,5 @@
import 'package:flutter/material.dart';
-Color immichBackgroundColor = const Color(0xFFf6f8fe);
-Color immichDarkBackgroundColor = const Color.fromARGB(255, 0, 0, 0);
-Color immichDarkThemePrimaryColor = const Color.fromARGB(255, 173, 203, 250);
+const Color immichBackgroundColor = Color(0xFFf6f8fe);
+const Color immichDarkBackgroundColor = Color.fromARGB(255, 0, 0, 0);
+const Color immichDarkThemePrimaryColor = Color.fromARGB(255, 173, 203, 250);
diff --git a/mobile/lib/constants/locales.dart b/mobile/lib/constants/locales.dart
index 452203d51..0913b1978 100644
--- a/mobile/lib/constants/locales.dart
+++ b/mobile/lib/constants/locales.dart
@@ -9,6 +9,7 @@ const List locales = [
Locale('it', 'IT'),
Locale('es', 'ES'),
Locale('vi', 'VN'),
+ Locale('fr', 'CA'),
Locale('fr', 'FR'),
Locale('ja', 'JP'),
Locale('pl', 'PL'),
@@ -29,6 +30,7 @@ const List locales = [
Locale('hi', 'IN'),
Locale('es', 'PE'),
Locale('es', 'MX'),
+ Locale('es', 'US'),
Locale('sv', 'FI'),
Locale('ca', 'CA'),
Locale('hu', 'HU'),
diff --git a/mobile/lib/extensions/asyncvalue_extensions.dart b/mobile/lib/extensions/asyncvalue_extensions.dart
new file mode 100644
index 000000000..036881f3c
--- /dev/null
+++ b/mobile/lib/extensions/asyncvalue_extensions.dart
@@ -0,0 +1,37 @@
+import 'package:flutter/material.dart';
+import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
+import 'package:immich_mobile/shared/ui/scaffold_error_body.dart';
+import 'package:logging/logging.dart';
+
+extension LogOnError on AsyncValue {
+ static final Logger _asyncErrorLogger = Logger("AsyncValue");
+
+ Widget widgetWhen({
+ bool skipLoadingOnRefresh = true,
+ Widget Function()? onLoading,
+ Widget Function(Object? error, StackTrace? stack)? onError,
+ required Widget Function(T data) onData,
+ }) {
+ if (isLoading) {
+ bool skip = false;
+ if (isRefreshing) {
+ skip = skipLoadingOnRefresh;
+ }
+
+ if (!skip) {
+ return onLoading?.call() ??
+ const Center(
+ child: ImmichLoadingIndicator(),
+ );
+ }
+ }
+
+ if (hasError && !hasValue) {
+ _asyncErrorLogger.severe("Error occured", error, stackTrace);
+ return onError?.call(error, stackTrace) ?? const ScaffoldErrorBody();
+ }
+
+ return onData(requireValue);
+ }
+}
diff --git a/mobile/lib/extensions/build_context_extensions.dart b/mobile/lib/extensions/build_context_extensions.dart
new file mode 100644
index 000000000..6151bd1a5
--- /dev/null
+++ b/mobile/lib/extensions/build_context_extensions.dart
@@ -0,0 +1,54 @@
+import 'package:auto_route/auto_route.dart';
+import 'package:flutter/material.dart';
+
+extension ContextHelper on BuildContext {
+ // Returns the current size from MediaQuery
+ Size get size => MediaQuery.sizeOf(this);
+
+ // Returns the current width from MediaQuery
+ double get width => size.width;
+
+ // Returns the current height from MediaQuery
+ double get height => size.height;
+
+ // Returns true if the app is running on a mobile device (!tablets)
+ bool get isMobile => width < 550;
+
+ // Returns the current ThemeData
+ ThemeData get themeData => Theme.of(this);
+
+ // Returns true if the app is using a dark theme
+ bool get isDarkTheme => themeData.brightness == Brightness.dark;
+
+ // Returns the current Primary color of the Theme
+ Color get primaryColor => themeData.primaryColor;
+
+ // Returns the Scaffold background color of the Theme
+ Color get scaffoldBackgroundColor => themeData.scaffoldBackgroundColor;
+
+ // Returns the current TextTheme
+ TextTheme get textTheme => themeData.textTheme;
+
+ // Current ColorScheme used
+ ColorScheme get colorScheme => themeData.colorScheme;
+
+ // Pop-out from the current context with optional result
+ void pop([T? result]) => Navigator.of(this).pop(result);
+
+ // Auto-Push new route from the current context
+ Future autoPush(PageRouteInfo route) =>
+ AutoRouter.of(this).push(route);
+
+ // Auto-Push navigate route from the current context
+ Future autoNavigate(
+ PageRouteInfo route,
+ ) =>
+ AutoRouter.of(this).navigate(route);
+
+ // Auto-Push replace route from the current context
+ Future autoReplace(PageRouteInfo route) =>
+ AutoRouter.of(this).replace(route);
+
+ // Auto-Pop from the current context
+ Future autoPop([T? result]) => AutoRouter.of(this).pop(result);
+}
diff --git a/mobile/lib/utils/builtin_extensions.dart b/mobile/lib/extensions/collection_extensions.dart
similarity index 70%
rename from mobile/lib/utils/builtin_extensions.dart
rename to mobile/lib/extensions/collection_extensions.dart
index 5b769f26f..283726ede 100644
--- a/mobile/lib/utils/builtin_extensions.dart
+++ b/mobile/lib/extensions/collection_extensions.dart
@@ -2,27 +2,6 @@ import 'dart:typed_data';
import 'package:collection/collection.dart';
-extension DurationExtension on String {
- Duration? toDuration() {
- try {
- final parts = split(':')
- .map((e) => double.parse(e).toInt())
- .toList(growable: false);
- return Duration(hours: parts[0], minutes: parts[1], seconds: parts[2]);
- } catch (e) {
- return null;
- }
- }
-
- double toDouble() {
- return double.parse(this);
- }
-
- int toInt() {
- return int.parse(this);
- }
-}
-
extension ListExtension on List {
List uniqueConsecutive({
int Function(E a, E b)? compare,
diff --git a/mobile/lib/utils/datetime_extensions.dart b/mobile/lib/extensions/datetime_extensions.dart
similarity index 100%
rename from mobile/lib/utils/datetime_extensions.dart
rename to mobile/lib/extensions/datetime_extensions.dart
diff --git a/mobile/lib/utils/flutter_map_extensions.dart b/mobile/lib/extensions/flutter_map_extensions.dart
similarity index 100%
rename from mobile/lib/utils/flutter_map_extensions.dart
rename to mobile/lib/extensions/flutter_map_extensions.dart
diff --git a/mobile/lib/extensions/string_extensions.dart b/mobile/lib/extensions/string_extensions.dart
new file mode 100644
index 000000000..a25ab4f50
--- /dev/null
+++ b/mobile/lib/extensions/string_extensions.dart
@@ -0,0 +1,30 @@
+extension StringExtension on String {
+ String capitalize() {
+ return split(" ")
+ .map(
+ (str) => str.isEmpty ? str : str[0].toUpperCase() + str.substring(1),
+ )
+ .join(" ");
+ }
+}
+
+extension DurationExtension on String {
+ Duration? toDuration() {
+ try {
+ final parts = split(':')
+ .map((e) => double.parse(e).toInt())
+ .toList(growable: false);
+ return Duration(hours: parts[0], minutes: parts[1], seconds: parts[2]);
+ } catch (e) {
+ return null;
+ }
+ }
+
+ double toDouble() {
+ return double.parse(this);
+ }
+
+ int toInt() {
+ return int.parse(this);
+ }
+}
diff --git a/mobile/lib/main.dart b/mobile/lib/main.dart
index 13eda9d6e..a12c43b6c 100644
--- a/mobile/lib/main.dart
+++ b/mobile/lib/main.dart
@@ -1,3 +1,4 @@
+import 'dart:async';
import 'dart:io';
import 'package:device_info_plus/device_info_plus.dart';
@@ -7,6 +8,7 @@ import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_displaymode/flutter_displaymode.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:timezone/data/latest.dart';
import 'package:immich_mobile/constants/locales.dart';
import 'package:immich_mobile/modules/backup/background_service/background.service.dart';
@@ -26,11 +28,8 @@ import 'package:immich_mobile/shared/models/store.dart';
import 'package:immich_mobile/shared/models/user.dart';
import 'package:immich_mobile/shared/providers/app_state.provider.dart';
import 'package:immich_mobile/shared/providers/db.provider.dart';
-import 'package:immich_mobile/shared/providers/release_info.provider.dart';
import 'package:immich_mobile/shared/services/immich_logger.service.dart';
import 'package:immich_mobile/shared/services/local_notification.service.dart';
-import 'package:immich_mobile/shared/views/immich_loading_overlay.dart';
-import 'package:immich_mobile/shared/views/version_announcement_overlay.dart';
import 'package:immich_mobile/utils/http_ssl_cert_override.dart';
import 'package:immich_mobile/utils/immich_app_theme.dart';
import 'package:immich_mobile/utils/migration.dart';
@@ -45,7 +44,13 @@ void main() async {
await initApp();
await migrateDatabaseIfNeeded(db);
HttpOverrides.global = HttpSSLCertOverride();
- runApp(getMainWidget(db));
+
+ runApp(
+ ProviderScope(
+ overrides: [dbProvider.overrideWithValue(db)],
+ child: const MainWidget(),
+ ),
+ );
}
Future initApp() async {
@@ -105,19 +110,6 @@ Future loadDb() async {
return db;
}
-Widget getMainWidget(Isar db) {
- return EasyLocalization(
- supportedLocales: locales,
- path: translationsPath,
- useFallbackTranslations: true,
- fallbackLocale: locales.first,
- child: ProviderScope(
- overrides: [dbProvider.overrideWithValue(db)],
- child: const ImmichApp(),
- ),
- );
-}
-
class ImmichApp extends ConsumerStatefulWidget {
const ImmichApp({super.key});
@@ -167,10 +159,9 @@ class ImmichAppState extends ConsumerState
// Android 8 does not support transparent app bars
final info = await DeviceInfoPlugin().androidInfo;
if (info.version.sdkInt <= 26) {
- overlayStyle =
- MediaQuery.of(context).platformBrightness == Brightness.light
- ? SystemUiOverlayStyle.light
- : SystemUiOverlayStyle.dark;
+ overlayStyle = context.isDarkTheme
+ ? SystemUiOverlayStyle.dark
+ : SystemUiOverlayStyle.light;
}
}
SystemChrome.setSystemUIOverlayStyle(overlayStyle);
@@ -196,30 +187,39 @@ class ImmichAppState extends ConsumerState
@override
Widget build(BuildContext context) {
var router = ref.watch(appRouterProvider);
- ref.watch(releaseInfoProvider.notifier).checkGithubReleaseInfo();
return MaterialApp(
localizationsDelegates: context.localizationDelegates,
supportedLocales: context.supportedLocales,
locale: context.locale,
debugShowCheckedModeBanner: false,
- home: Stack(
- children: [
- MaterialApp.router(
- title: 'Immich',
- debugShowCheckedModeBanner: false,
- themeMode: ref.watch(immichThemeProvider),
- darkTheme: immichDarkTheme,
- theme: immichLightTheme,
- routeInformationParser: router.defaultRouteParser(),
- routerDelegate: router.delegate(
- navigatorObservers: () => [TabNavigationObserver(ref: ref)],
- ),
- ),
- const ImmichLoadingOverlay(),
- const VersionAnnouncementOverlay(),
- ],
+ home: MaterialApp.router(
+ title: 'Immich',
+ debugShowCheckedModeBanner: false,
+ themeMode: ref.watch(immichThemeProvider),
+ darkTheme: immichDarkTheme,
+ theme: immichLightTheme,
+ routeInformationParser: router.defaultRouteParser(),
+ routerDelegate: router.delegate(
+ navigatorObservers: () => [TabNavigationObserver(ref: ref)],
+ ),
),
);
}
}
+
+// ignore: prefer-single-widget-per-file
+class MainWidget extends StatelessWidget {
+ const MainWidget({super.key});
+
+ @override
+ Widget build(BuildContext context) {
+ return EasyLocalization(
+ supportedLocales: locales,
+ path: translationsPath,
+ useFallbackTranslations: true,
+ fallbackLocale: locales.first,
+ child: const ImmichApp(),
+ );
+ }
+}
diff --git a/mobile/lib/modules/activities/models/activity.model.dart b/mobile/lib/modules/activities/models/activity.model.dart
index 417ba4a86..2db626f54 100644
--- a/mobile/lib/modules/activities/models/activity.model.dart
+++ b/mobile/lib/modules/activities/models/activity.model.dart
@@ -48,8 +48,7 @@ class Activity {
: ActivityType.like,
user = User(
email: dto.user.email,
- firstName: dto.user.firstName,
- lastName: dto.user.lastName,
+ name: dto.user.name,
profileImagePath: dto.user.profileImagePath,
id: dto.user.id,
// Placeholder values
diff --git a/mobile/lib/modules/activities/views/activities_page.dart b/mobile/lib/modules/activities/views/activities_page.dart
index 69afe2e5d..f0c68a349 100644
--- a/mobile/lib/modules/activities/views/activities_page.dart
+++ b/mobile/lib/modules/activities/views/activities_page.dart
@@ -4,13 +4,14 @@ import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart' hide Store;
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/activities/models/activity.model.dart';
import 'package:immich_mobile/modules/activities/providers/activity.provider.dart';
import 'package:immich_mobile/shared/models/store.dart';
import 'package:immich_mobile/shared/ui/confirm_dialog.dart';
-import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
import 'package:immich_mobile/shared/ui/user_circle_avatar.dart';
-import 'package:immich_mobile/utils/datetime_extensions.dart';
+import 'package:immich_mobile/extensions/datetime_extensions.dart';
import 'package:immich_mobile/utils/image_url_builder.dart';
class ActivitiesPage extends HookConsumerWidget {
@@ -19,12 +20,14 @@ class ActivitiesPage extends HookConsumerWidget {
final bool withAssetThumbs;
final String appBarTitle;
final bool isOwner;
+ final bool isReadOnly;
const ActivitiesPage(
this.albumId, {
this.appBarTitle = "",
this.assetId,
this.withAssetThumbs = true,
this.isOwner = false,
+ this.isReadOnly = false,
super.key,
});
@@ -45,13 +48,10 @@ class ActivitiesPage extends HookConsumerWidget {
},
[],
);
+
buildTitleWithTimestamp(Activity activity, {bool leftAlign = true}) {
- final textColor = Theme.of(context).brightness == Brightness.dark
- ? Colors.white
- : Colors.black;
- final textStyle = Theme.of(context)
- .textTheme
- .bodyMedium
+ final textColor = context.isDarkTheme ? Colors.white : Colors.black;
+ final textStyle = context.textTheme.bodyMedium
?.copyWith(color: textColor.withOpacity(0.6));
return Row(
@@ -61,7 +61,7 @@ class ActivitiesPage extends HookConsumerWidget {
mainAxisSize: leftAlign ? MainAxisSize.min : MainAxisSize.max,
children: [
Text(
- "${activity.user.firstName} ${activity.user.lastName}",
+ activity.user.name,
style: textStyle,
overflow: TextOverflow.ellipsis,
),
@@ -88,7 +88,7 @@ class ActivitiesPage extends HookConsumerWidget {
width: 40,
height: 30,
decoration: BoxDecoration(
- borderRadius: BorderRadius.circular(4),
+ borderRadius: const BorderRadius.all(Radius.circular(4)),
image: DecorationImage(
image: CachedNetworkImageProvider(
getThumbnailUrlForRemoteId(
@@ -116,6 +116,7 @@ class ActivitiesPage extends HookConsumerWidget {
padding: const EdgeInsets.only(bottom: 10),
child: TextField(
controller: inputController,
+ enabled: !isReadOnly,
focusNode: inputFocusNode,
textInputAction: TextInputAction.send,
autofocus: false,
@@ -150,7 +151,9 @@ class ActivitiesPage extends HookConsumerWidget {
),
),
suffixIconColor: liked ? Colors.red[700] : null,
- hintText: 'shared_album_activities_input_hint'.tr(),
+ hintText: isReadOnly
+ ? 'shared_album_activities_input_disable'.tr()
+ : 'shared_album_activities_input_hint'.tr(),
hintStyle: TextStyle(
fontWeight: FontWeight.normal,
fontSize: 14,
@@ -228,11 +231,8 @@ class ActivitiesPage extends HookConsumerWidget {
return Scaffold(
appBar: AppBar(title: Text(appBarTitle)),
- body: activities.maybeWhen(
- orElse: () {
- return const Center(child: ImmichLoadingIndicator());
- },
- data: (data) {
+ body: activities.widgetWhen(
+ onData: (data) {
final liked = data.firstWhereOrNull(
(a) =>
a.type == ActivityType.like &&
@@ -240,70 +240,72 @@ class ActivitiesPage extends HookConsumerWidget {
a.assetId == assetId,
);
- return Stack(
- children: [
- ListView.builder(
- controller: listViewScrollController,
- itemCount: data.length + 1,
- itemBuilder: (context, index) {
- // Vertical gap after the last element
- if (index == data.length) {
- return const SizedBox(
- height: 80,
- );
- }
+ return SafeArea(
+ child: Stack(
+ children: [
+ ListView.builder(
+ controller: listViewScrollController,
+ itemCount: data.length + 1,
+ itemBuilder: (context, index) {
+ // Vertical gap after the last element
+ if (index == data.length) {
+ return const SizedBox(
+ height: 80,
+ );
+ }
- final activity = data[index];
- final canDelete =
- activity.user.id == currentUser?.id || isOwner;
+ final activity = data[index];
+ final canDelete =
+ activity.user.id == currentUser?.id || isOwner;
- return Padding(
- padding: const EdgeInsets.all(5),
- child: activity.type == ActivityType.comment
- ? getDismissibleWidget(
- ListTile(
- minVerticalPadding: 15,
- leading: UserCircleAvatar(user: activity.user),
- title: buildTitleWithTimestamp(
- activity,
- leftAlign:
- withAssetThumbs && activity.assetId != null,
- ),
- titleAlignment: ListTileTitleAlignment.top,
- trailing: buildAssetThumbnail(activity),
- subtitle: Text(activity.comment!),
- ),
- activity,
- canDelete,
- )
- : getDismissibleWidget(
- ListTile(
- minVerticalPadding: 15,
- leading: Container(
- width: 44,
- alignment: Alignment.center,
- child: Icon(
- Icons.favorite_rounded,
- color: Colors.red[700],
+ return Padding(
+ padding: const EdgeInsets.all(5),
+ child: activity.type == ActivityType.comment
+ ? getDismissibleWidget(
+ ListTile(
+ minVerticalPadding: 15,
+ leading: UserCircleAvatar(user: activity.user),
+ title: buildTitleWithTimestamp(
+ activity,
+ leftAlign: withAssetThumbs &&
+ activity.assetId != null,
),
+ titleAlignment: ListTileTitleAlignment.top,
+ trailing: buildAssetThumbnail(activity),
+ subtitle: Text(activity.comment!),
),
- title: buildTitleWithTimestamp(activity),
- trailing: buildAssetThumbnail(activity),
+ activity,
+ canDelete,
+ )
+ : getDismissibleWidget(
+ ListTile(
+ minVerticalPadding: 15,
+ leading: Container(
+ width: 44,
+ alignment: Alignment.center,
+ child: Icon(
+ Icons.favorite_rounded,
+ color: Colors.red[700],
+ ),
+ ),
+ title: buildTitleWithTimestamp(activity),
+ trailing: buildAssetThumbnail(activity),
+ ),
+ activity,
+ canDelete,
),
- activity,
- canDelete,
- ),
- );
- },
- ),
- Align(
- alignment: Alignment.bottomCenter,
- child: Container(
- color: Theme.of(context).scaffoldBackgroundColor,
- child: buildTextField(liked?.id),
+ );
+ },
),
- ),
- ],
+ Align(
+ alignment: Alignment.bottomCenter,
+ child: Container(
+ color: context.scaffoldBackgroundColor,
+ child: buildTextField(liked?.id),
+ ),
+ ),
+ ],
+ ),
);
},
),
diff --git a/mobile/lib/modules/album/providers/album.provider.dart b/mobile/lib/modules/album/providers/album.provider.dart
index 24679c517..48a2e8f1f 100644
--- a/mobile/lib/modules/album/providers/album.provider.dart
+++ b/mobile/lib/modules/album/providers/album.provider.dart
@@ -25,6 +25,8 @@ class AlbumNotifier extends StateNotifier> {
_albumService.refreshRemoteAlbums(isShared: false),
]);
+ Future getDeviceAlbums() => _albumService.refreshDeviceAlbums();
+
Future deleteAlbum(Album album) => _albumService.deleteAlbum(album);
Future createAlbum(
diff --git a/mobile/lib/modules/album/providers/shared_album.provider.dart b/mobile/lib/modules/album/providers/shared_album.provider.dart
index 4f36c4633..f8084da00 100644
--- a/mobile/lib/modules/album/providers/shared_album.provider.dart
+++ b/mobile/lib/modules/album/providers/shared_album.provider.dart
@@ -2,6 +2,7 @@ import 'dart:async';
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/modules/album/providers/album_detail.provider.dart';
import 'package:immich_mobile/modules/album/services/album.service.dart';
import 'package:immich_mobile/shared/models/album.dart';
import 'package:immich_mobile/shared/models/asset.dart';
@@ -10,7 +11,7 @@ import 'package:immich_mobile/shared/providers/db.provider.dart';
import 'package:isar/isar.dart';
class SharedAlbumNotifier extends StateNotifier> {
- SharedAlbumNotifier(this._albumService, Isar db) : super([]) {
+ SharedAlbumNotifier(this._albumService, Isar db, this._ref) : super([]) {
final query = db.albums.filter().sharedEqualTo(true).sortByCreatedAtDesc();
query.findAll().then((value) => state = value);
_streamSub = query.watch().listen((data) => state = data);
@@ -18,6 +19,7 @@ class SharedAlbumNotifier extends StateNotifier> {
final AlbumService _albumService;
late final StreamSubscription> _streamSub;
+ final Ref _ref;
Future createSharedAlbum(
String albumName,
@@ -66,6 +68,17 @@ class SharedAlbumNotifier extends StateNotifier> {
return result;
}
+ Future setActivityEnabled(Album album, bool activityEnabled) async {
+ final result =
+ await _albumService.setActivityEnabled(album, activityEnabled);
+
+ if (result) {
+ _ref.invalidate(albumDetailProvider(album.id));
+ }
+
+ return result;
+ }
+
@override
void dispose() {
_streamSub.cancel();
@@ -78,5 +91,6 @@ final sharedAlbumProvider =
return SharedAlbumNotifier(
ref.watch(albumServiceProvider),
ref.watch(dbProvider),
+ ref,
);
});
diff --git a/mobile/lib/modules/album/services/album.service.dart b/mobile/lib/modules/album/services/album.service.dart
index 4488eca23..1f50a3667 100644
--- a/mobile/lib/modules/album/services/album.service.dart
+++ b/mobile/lib/modules/album/services/album.service.dart
@@ -67,6 +67,10 @@ class AlbumService {
final List selectedIds =
await _backupService.selectedAlbumsQuery().idProperty().findAll();
if (selectedIds.isEmpty) {
+ final numLocal = await _db.albums.where().localIdIsNotNull().count();
+ if (numLocal > 0) {
+ _syncService.removeAllLocalAlbumsAndAssets();
+ }
return false;
}
final List onDevice =
@@ -284,6 +288,23 @@ class AlbumService {
return false;
}
+ Future setActivityEnabled(Album album, bool enabled) async {
+ try {
+ final result = await _apiService.albumApi.updateAlbumInfo(
+ album.remoteId!,
+ UpdateAlbumDto(isActivityEnabled: enabled),
+ );
+ if (result != null) {
+ album.activityEnabled = enabled;
+ await _db.writeTxn(() => _db.albums.put(album));
+ return true;
+ }
+ } catch (e) {
+ debugPrint("Error setActivityEnabled ${e.toString()}");
+ }
+ return false;
+ }
+
Future deleteAlbum(Album album) async {
try {
final userId = Store.get(StoreKey.currentUser).isarId;
diff --git a/mobile/lib/modules/album/ui/add_to_album_bottom_sheet.dart b/mobile/lib/modules/album/ui/add_to_album_bottom_sheet.dart
index 257dbdbaa..25747177a 100644
--- a/mobile/lib/modules/album/ui/add_to_album_bottom_sheet.dart
+++ b/mobile/lib/modules/album/ui/add_to_album_bottom_sheet.dart
@@ -1,8 +1,8 @@
import 'package:easy_localization/easy_localization.dart';
-import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/providers/album.provider.dart';
import 'package:immich_mobile/modules/album/providers/album_detail.provider.dart';
import 'package:immich_mobile/modules/album/providers/shared_album.provider.dart';
@@ -65,7 +65,7 @@ class AddToAlbumBottomSheet extends HookConsumerWidget {
}
ref.invalidate(albumDetailProvider(album.id));
- Navigator.pop(context);
+ context.pop();
}
return Card(
@@ -95,20 +95,19 @@ class AddToAlbumBottomSheet extends HookConsumerWidget {
children: [
Text(
'common_add_to_album'.tr(),
- style: Theme.of(context).textTheme.displayMedium,
+ style: context.textTheme.displayMedium,
),
TextButton.icon(
icon: Icon(
Icons.add,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
label: Text(
'common_create_new_album'.tr(),
- style:
- TextStyle(color: Theme.of(context).primaryColor),
+ style: TextStyle(color: context.primaryColor),
),
onPressed: () {
- AutoRouter.of(context).push(
+ context.autoPush(
CreateAlbumRoute(
isSharedAlbum: false,
initialAssets: assets,
diff --git a/mobile/lib/modules/album/ui/album_action_outlined_button.dart b/mobile/lib/modules/album/ui/album_action_outlined_button.dart
index 928a50794..768be0e3c 100644
--- a/mobile/lib/modules/album/ui/album_action_outlined_button.dart
+++ b/mobile/lib/modules/album/ui/album_action_outlined_button.dart
@@ -1,4 +1,5 @@
import 'package:flutter/material.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
class AlbumActionOutlinedButton extends StatelessWidget {
final VoidCallback? onPressed;
@@ -14,10 +15,8 @@ class AlbumActionOutlinedButton extends StatelessWidget {
@override
Widget build(BuildContext context) {
- final isDarkTheme = Theme.of(context).brightness == Brightness.dark;
-
return Padding(
- padding: const EdgeInsets.only(right: 8.0),
+ padding: const EdgeInsets.only(right: 16.0),
child: OutlinedButton.icon(
style: OutlinedButton.styleFrom(
padding: const EdgeInsets.symmetric(vertical: 0, horizontal: 10),
@@ -26,21 +25,21 @@ class AlbumActionOutlinedButton extends StatelessWidget {
),
side: BorderSide(
width: 1,
- color: isDarkTheme
+ color: context.isDarkTheme
? const Color.fromARGB(255, 63, 63, 63)
: const Color.fromARGB(255, 206, 206, 206),
),
),
icon: Icon(
iconData,
- size: 15,
- color: Theme.of(context).primaryColor,
+ size: 18,
+ color: context.primaryColor,
),
label: Text(
labelText,
- style: Theme.of(context).textTheme.labelSmall?.copyWith(
- fontWeight: FontWeight.bold,
- ),
+ style: context.textTheme.labelMedium?.copyWith(
+ fontWeight: FontWeight.w600,
+ ),
),
onPressed: onPressed,
),
diff --git a/mobile/lib/modules/album/ui/album_thumbnail_card.dart b/mobile/lib/modules/album/ui/album_thumbnail_card.dart
index f37751e3e..b295deec5 100644
--- a/mobile/lib/modules/album/ui/album_thumbnail_card.dart
+++ b/mobile/lib/modules/album/ui/album_thumbnail_card.dart
@@ -1,5 +1,6 @@
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/shared/models/album.dart';
import 'package:immich_mobile/shared/models/store.dart';
import 'package:immich_mobile/shared/ui/immich_image.dart';
@@ -22,7 +23,8 @@ class AlbumThumbnailCard extends StatelessWidget {
@override
Widget build(BuildContext context) {
- var isDarkMode = Theme.of(context).brightness == Brightness.dark;
+ var isDarkTheme = context.isDarkTheme;
+
return LayoutBuilder(
builder: (context, constraints) {
var cardSize = constraints.maxWidth;
@@ -32,7 +34,7 @@ class AlbumThumbnailCard extends StatelessWidget {
height: cardSize,
width: cardSize,
decoration: BoxDecoration(
- color: isDarkMode ? Colors.grey[800] : Colors.grey[200],
+ color: isDarkTheme ? Colors.grey[800] : Colors.grey[200],
),
child: Center(
child: Icon(
@@ -70,17 +72,13 @@ class AlbumThumbnailCard extends StatelessWidget {
.tr(args: ['${album.assetCount}'])
: 'album_thumbnail_card_items'
.tr(args: ['${album.assetCount}']),
- style: TextStyle(
- fontFamily: 'WorkSans',
- fontSize: 12,
- color: isDarkMode ? Colors.white : Colors.black,
- ),
+ style: context.textTheme.bodyMedium,
),
if (owner != null) const TextSpan(text: ' · '),
if (owner != null)
TextSpan(
text: owner,
- style: Theme.of(context).textTheme.labelSmall,
+ style: context.textTheme.bodyMedium,
),
],
),
@@ -112,11 +110,9 @@ class AlbumThumbnailCard extends StatelessWidget {
width: cardSize,
child: Text(
album.name,
- style: TextStyle(
- fontWeight: FontWeight.bold,
- color: isDarkMode
- ? Theme.of(context).primaryColor
- : Colors.black,
+ style: context.textTheme.bodyMedium?.copyWith(
+ color: context.primaryColor,
+ fontWeight: FontWeight.w500,
),
),
),
diff --git a/mobile/lib/modules/album/ui/album_thumbnail_listtile.dart b/mobile/lib/modules/album/ui/album_thumbnail_listtile.dart
index c9237ea27..38208e88c 100644
--- a/mobile/lib/modules/album/ui/album_thumbnail_listtile.dart
+++ b/mobile/lib/modules/album/ui/album_thumbnail_listtile.dart
@@ -1,7 +1,7 @@
-import 'package:auto_route/auto_route.dart';
import 'package:cached_network_image/cached_network_image.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/routing/router.dart';
import 'package:immich_mobile/shared/models/album.dart';
import 'package:immich_mobile/shared/models/store.dart';
@@ -21,12 +21,11 @@ class AlbumThumbnailListTile extends StatelessWidget {
@override
Widget build(BuildContext context) {
var cardSize = 68.0;
- var isDarkMode = Theme.of(context).brightness == Brightness.dark;
buildEmptyThumbnail() {
return Container(
decoration: BoxDecoration(
- color: isDarkMode ? Colors.grey[800] : Colors.grey[200],
+ color: context.isDarkTheme ? Colors.grey[800] : Colors.grey[200],
),
child: SizedBox(
height: cardSize,
@@ -61,7 +60,7 @@ class AlbumThumbnailListTile extends StatelessWidget {
behavior: HitTestBehavior.opaque,
onTap: onTap ??
() {
- AutoRouter.of(context).push(AlbumViewerRoute(albumId: album.id));
+ context.autoPush(AlbumViewerRoute(albumId: album.id));
},
child: Padding(
padding: const EdgeInsets.only(bottom: 12.0),
diff --git a/mobile/lib/modules/album/ui/album_title_text_field.dart b/mobile/lib/modules/album/ui/album_title_text_field.dart
index 38c1f681a..57b82a80d 100644
--- a/mobile/lib/modules/album/ui/album_title_text_field.dart
+++ b/mobile/lib/modules/album/ui/album_title_text_field.dart
@@ -1,6 +1,7 @@
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/providers/album_title.provider.dart';
class AlbumTitleTextField extends ConsumerWidget {
@@ -19,7 +20,7 @@ class AlbumTitleTextField extends ConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
- final isDarkTheme = Theme.of(context).brightness == Brightness.dark;
+ final isDarkTheme = context.isDarkTheme;
return TextField(
onChanged: (v) {
@@ -55,7 +56,7 @@ class AlbumTitleTextField extends ConsumerWidget {
},
icon: Icon(
Icons.cancel_rounded,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
splashRadius: 10,
)
diff --git a/mobile/lib/modules/album/ui/album_viewer_appbar.dart b/mobile/lib/modules/album/ui/album_viewer_appbar.dart
index 05db82e10..0e2fc74fb 100644
--- a/mobile/lib/modules/album/ui/album_viewer_appbar.dart
+++ b/mobile/lib/modules/album/ui/album_viewer_appbar.dart
@@ -1,8 +1,8 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:fluttertoast/fluttertoast.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/activities/providers/activity.provider.dart';
import 'package:immich_mobile/modules/album/providers/album.provider.dart';
import 'package:immich_mobile/modules/album/providers/album_detail.provider.dart';
@@ -43,6 +43,7 @@ class AlbumViewerAppbar extends HookConsumerWidget
Widget build(BuildContext context, WidgetRef ref) {
final newAlbumTitle = ref.watch(albumViewerProvider).editTitleText;
final isEditAlbum = ref.watch(albumViewerProvider).isEditAlbum;
+ final isProcessing = useProcessingOverlay();
final comments = album.shared
? ref.watch(
activityStatisticsStateProvider(
@@ -52,18 +53,18 @@ class AlbumViewerAppbar extends HookConsumerWidget
: 0;
deleteAlbum() async {
- ImmichLoadingOverlayController.appLoader.show();
+ isProcessing.value = true;
final bool success;
if (album.shared) {
success =
await ref.watch(sharedAlbumProvider.notifier).deleteAlbum(album);
- AutoRouter.of(context)
- .navigate(const TabControllerRoute(children: [SharingRoute()]));
+ context
+ .autoNavigate(const TabControllerRoute(children: [SharingRoute()]));
} else {
success = await ref.watch(albumProvider.notifier).deleteAlbum(album);
- AutoRouter.of(context)
- .navigate(const TabControllerRoute(children: [LibraryRoute()]));
+ context
+ .autoNavigate(const TabControllerRoute(children: [LibraryRoute()]));
}
if (!success) {
ImmichToast.show(
@@ -74,7 +75,7 @@ class AlbumViewerAppbar extends HookConsumerWidget
);
}
- ImmichLoadingOverlayController.appLoader.hide();
+ isProcessing.value = false;
}
Future showConfirmationDialog() async {
@@ -89,27 +90,25 @@ class AlbumViewerAppbar extends HookConsumerWidget
),
actions: [
TextButton(
- onPressed: () => Navigator.pop(context, 'Cancel'),
+ onPressed: () => context.pop('Cancel'),
child: Text(
'Cancel',
style: TextStyle(
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
fontWeight: FontWeight.bold,
),
),
),
TextButton(
onPressed: () {
- Navigator.pop(context, 'Confirm');
+ context.pop('Confirm');
deleteAlbum();
},
child: Text(
'Confirm',
style: TextStyle(
fontWeight: FontWeight.bold,
- color: Theme.of(context).brightness == Brightness.light
- ? Colors.red
- : Colors.red[300],
+ color: !context.isDarkTheme ? Colors.red : Colors.red[300],
),
),
),
@@ -124,16 +123,16 @@ class AlbumViewerAppbar extends HookConsumerWidget
}
void onLeaveAlbumPressed() async {
- ImmichLoadingOverlayController.appLoader.show();
+ isProcessing.value = true;
bool isSuccess =
await ref.watch(sharedAlbumProvider.notifier).leaveAlbum(album);
if (isSuccess) {
- AutoRouter.of(context)
- .navigate(const TabControllerRoute(children: [SharingRoute()]));
+ context
+ .autoNavigate(const TabControllerRoute(children: [SharingRoute()]));
} else {
- Navigator.pop(context);
+ context.pop();
ImmichToast.show(
context: context,
msg: "album_viewer_appbar_share_err_leave".tr(),
@@ -142,11 +141,11 @@ class AlbumViewerAppbar extends HookConsumerWidget
);
}
- ImmichLoadingOverlayController.appLoader.hide();
+ isProcessing.value = false;
}
void onRemoveFromAlbumPressed() async {
- ImmichLoadingOverlayController.appLoader.show();
+ isProcessing.value = true;
bool isSuccess =
await ref.watch(sharedAlbumProvider.notifier).removeAssetFromAlbum(
@@ -155,12 +154,12 @@ class AlbumViewerAppbar extends HookConsumerWidget
);
if (isSuccess) {
- Navigator.pop(context);
+ context.pop();
selectionDisabled();
ref.watch(albumProvider.notifier).getAllAlbums();
ref.invalidate(albumDetailProvider(album.id));
} else {
- Navigator.pop(context);
+ context.pop();
ImmichToast.show(
context: context,
msg: "album_viewer_appbar_share_err_remove".tr(),
@@ -169,7 +168,7 @@ class AlbumViewerAppbar extends HookConsumerWidget
);
}
- ImmichLoadingOverlayController.appLoader.hide();
+ isProcessing.value = false;
}
void handleShareAssets(
@@ -190,7 +189,7 @@ class AlbumViewerAppbar extends HookConsumerWidget
gravity: ToastGravity.BOTTOM,
);
}
- Navigator.of(buildContext).pop();
+ buildContext.pop();
},
);
return const ShareDialog();
@@ -200,9 +199,9 @@ class AlbumViewerAppbar extends HookConsumerWidget
}
void onShareAssetsTo() async {
- ImmichLoadingOverlayController.appLoader.show();
+ isProcessing.value = true;
handleShareAssets(ref, context, selected);
- ImmichLoadingOverlayController.appLoader.hide();
+ isProcessing.value = false;
}
buildBottomSheetActions() {
@@ -212,36 +211,40 @@ class AlbumViewerAppbar extends HookConsumerWidget
leading: const Icon(Icons.ios_share_rounded),
title: const Text(
'album_viewer_appbar_share_to',
- style: TextStyle(fontWeight: FontWeight.bold),
+ style: TextStyle(fontWeight: FontWeight.w500),
).tr(),
onTap: () => onShareAssetsTo(),
),
- album.ownerId == userId ? ListTile(
- leading: const Icon(Icons.delete_sweep_rounded),
- title: const Text(
- 'album_viewer_appbar_share_remove',
- style: TextStyle(fontWeight: FontWeight.bold),
- ).tr(),
- onTap: () => onRemoveFromAlbumPressed(),
- ) : const SizedBox(),
+ album.ownerId == userId
+ ? ListTile(
+ leading: const Icon(Icons.delete_sweep_rounded),
+ title: const Text(
+ 'album_viewer_appbar_share_remove',
+ style: TextStyle(fontWeight: FontWeight.w500),
+ ).tr(),
+ onTap: () => onRemoveFromAlbumPressed(),
+ )
+ : const SizedBox(),
];
} else {
return [
- album.ownerId == userId ? ListTile(
- leading: const Icon(Icons.delete_forever_rounded),
- title: const Text(
- 'album_viewer_appbar_share_delete',
- style: TextStyle(fontWeight: FontWeight.bold),
- ).tr(),
- onTap: () => onDeleteAlbumPressed(),
- ) : ListTile(
- leading: const Icon(Icons.person_remove_rounded),
- title: const Text(
- 'album_viewer_appbar_share_leave',
- style: TextStyle(fontWeight: FontWeight.bold),
- ).tr(),
- onTap: () => onLeaveAlbumPressed(),
- ),
+ album.ownerId == userId
+ ? ListTile(
+ leading: const Icon(Icons.delete_forever_rounded),
+ title: const Text(
+ 'album_viewer_appbar_share_delete',
+ style: TextStyle(fontWeight: FontWeight.w500),
+ ).tr(),
+ onTap: () => onDeleteAlbumPressed(),
+ )
+ : ListTile(
+ leading: const Icon(Icons.person_remove_rounded),
+ title: const Text(
+ 'album_viewer_appbar_share_leave',
+ style: TextStyle(fontWeight: FontWeight.w500),
+ ).tr(),
+ onTap: () => onLeaveAlbumPressed(),
+ ),
];
}
}
@@ -251,33 +254,31 @@ class AlbumViewerAppbar extends HookConsumerWidget
ListTile(
leading: const Icon(Icons.person_add_alt_rounded),
onTap: () {
- Navigator.pop(context);
+ context.pop();
onAddUsers!(album);
},
title: const Text(
"album_viewer_page_share_add_users",
- style: TextStyle(fontWeight: FontWeight.bold),
+ style: TextStyle(fontWeight: FontWeight.w500),
).tr(),
),
ListTile(
leading: const Icon(Icons.share_rounded),
onTap: () {
- AutoRouter.of(context)
- .push(SharedLinkEditRoute(albumId: album.remoteId));
- Navigator.pop(context);
+ context.autoPush(SharedLinkEditRoute(albumId: album.remoteId));
+ context.pop();
},
title: const Text(
"control_bottom_app_bar_share",
- style: TextStyle(fontWeight: FontWeight.bold),
+ style: TextStyle(fontWeight: FontWeight.w500),
).tr(),
),
ListTile(
leading: const Icon(Icons.settings_rounded),
- onTap: () =>
- AutoRouter.of(context).navigate(AlbumOptionsRoute(album: album)),
+ onTap: () => context.autoNavigate(AlbumOptionsRoute(album: album)),
title: const Text(
"translated_text_options",
- style: TextStyle(fontWeight: FontWeight.bold),
+ style: TextStyle(fontWeight: FontWeight.w500),
).tr(),
),
];
@@ -286,17 +287,17 @@ class AlbumViewerAppbar extends HookConsumerWidget
ListTile(
leading: const Icon(Icons.add_photo_alternate_outlined),
onTap: () {
- Navigator.pop(context);
+ context.pop();
onAddPhotos!(album);
},
title: const Text(
"share_add_photos",
- style: TextStyle(fontWeight: FontWeight.bold),
+ style: TextStyle(fontWeight: FontWeight.w500),
).tr(),
),
];
showModalBottomSheet(
- backgroundColor: Theme.of(context).scaffoldBackgroundColor,
+ backgroundColor: context.scaffoldBackgroundColor,
isScrollControlled: false,
context: context,
builder: (context) {
@@ -338,7 +339,7 @@ class AlbumViewerAppbar extends HookConsumerWidget
comments.toString(),
style: TextStyle(
fontWeight: FontWeight.bold,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
),
),
@@ -377,7 +378,7 @@ class AlbumViewerAppbar extends HookConsumerWidget
);
} else {
return IconButton(
- onPressed: () async => await AutoRouter.of(context).pop(),
+ onPressed: () async => await context.autoPop(),
icon: const Icon(Icons.arrow_back_ios_rounded),
splashRadius: 25,
);
@@ -390,7 +391,8 @@ class AlbumViewerAppbar extends HookConsumerWidget
title: selected.isNotEmpty ? Text('${selected.length}') : null,
centerTitle: false,
actions: [
- if (album.shared) buildActivitiesButton(),
+ if (album.shared && (album.activityEnabled || comments != 0))
+ buildActivitiesButton(),
if (album.isRemote)
IconButton(
splashRadius: 25,
diff --git a/mobile/lib/modules/album/ui/album_viewer_editable_title.dart b/mobile/lib/modules/album/ui/album_viewer_editable_title.dart
index 8a7e46f8c..1ece28afb 100644
--- a/mobile/lib/modules/album/ui/album_viewer_editable_title.dart
+++ b/mobile/lib/modules/album/ui/album_viewer_editable_title.dart
@@ -2,6 +2,7 @@ import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/providers/album_viewer.provider.dart';
import 'package:immich_mobile/shared/models/album.dart';
@@ -17,7 +18,6 @@ class AlbumViewerEditableTitle extends HookConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
final titleTextEditController = useTextEditingController(text: album.name);
- final isDarkTheme = Theme.of(context).brightness == Brightness.dark;
void onFocusModeChange() {
if (!titleFocusNode.hasFocus && titleTextEditController.text.isEmpty) {
@@ -44,7 +44,7 @@ class AlbumViewerEditableTitle extends HookConsumerWidget {
}
},
focusNode: titleFocusNode,
- style: const TextStyle(fontSize: 24, fontWeight: FontWeight.bold),
+ style: context.textTheme.headlineMedium,
controller: titleTextEditController,
onTap: () {
FocusScope.of(context).requestFocus(titleFocusNode);
@@ -65,7 +65,7 @@ class AlbumViewerEditableTitle extends HookConsumerWidget {
},
icon: Icon(
Icons.cancel_rounded,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
splashRadius: 10,
)
@@ -79,14 +79,14 @@ class AlbumViewerEditableTitle extends HookConsumerWidget {
borderRadius: BorderRadius.circular(10),
),
focusColor: Colors.grey[300],
- fillColor: isDarkTheme
+ fillColor: context.isDarkTheme
? const Color.fromARGB(255, 32, 33, 35)
: Colors.grey[200],
filled: titleFocusNode.hasFocus,
hintText: 'share_add_title'.tr(),
hintStyle: TextStyle(
fontSize: 28,
- color: isDarkTheme ? Colors.grey[300] : Colors.grey[700],
+ color: context.isDarkTheme ? Colors.grey[300] : Colors.grey[700],
fontWeight: FontWeight.bold,
),
),
diff --git a/mobile/lib/modules/album/views/album_options_part.dart b/mobile/lib/modules/album/views/album_options_part.dart
index eb08b6bda..6ef773339 100644
--- a/mobile/lib/modules/album/views/album_options_part.dart
+++ b/mobile/lib/modules/album/views/album_options_part.dart
@@ -1,9 +1,9 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:fluttertoast/fluttertoast.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/providers/shared_album.provider.dart';
import 'package:immich_mobile/modules/login/providers/authentication.provider.dart';
import 'package:immich_mobile/routing/router.dart';
@@ -23,28 +23,31 @@ class AlbumOptionsPage extends HookConsumerWidget {
final sharedUsers = useState(album.sharedUsers.toList());
final owner = album.owner.value;
final userId = ref.watch(authenticationProvider).userId;
+ final activityEnabled = useState(album.activityEnabled);
+ final isProcessing = useProcessingOverlay();
final isOwner = owner?.id == userId;
void showErrorMessage() {
- Navigator.pop(context);
+ context.pop();
ImmichToast.show(
context: context,
- msg: "Error leaving/removing from album",
+ msg: "shared_album_section_people_action_error".tr(),
toastType: ToastType.error,
gravity: ToastGravity.BOTTOM,
);
}
void leaveAlbum() async {
- ImmichLoadingOverlayController.appLoader.show();
+ isProcessing.value = true;
try {
final isSuccess =
await ref.read(sharedAlbumProvider.notifier).leaveAlbum(album);
if (isSuccess) {
- AutoRouter.of(context)
- .navigate(const TabControllerRoute(children: [SharingRoute()]));
+ context.autoNavigate(
+ const TabControllerRoute(children: [SharingRoute()]),
+ );
} else {
showErrorMessage();
}
@@ -52,11 +55,11 @@ class AlbumOptionsPage extends HookConsumerWidget {
showErrorMessage();
}
- ImmichLoadingOverlayController.appLoader.hide();
+ isProcessing.value = false;
}
void removeUserFromAlbum(User user) async {
- ImmichLoadingOverlayController.appLoader.show();
+ isProcessing.value = true;
try {
await ref
@@ -68,8 +71,8 @@ class AlbumOptionsPage extends HookConsumerWidget {
showErrorMessage();
}
- Navigator.pop(context);
- ImmichLoadingOverlayController.appLoader.hide();
+ context.pop();
+ isProcessing.value = false;
}
void handleUserClick(User user) {
@@ -79,7 +82,7 @@ class AlbumOptionsPage extends HookConsumerWidget {
actions = [
ListTile(
leading: const Icon(Icons.exit_to_app_rounded),
- title: const Text("Leave album"),
+ title: const Text("shared_album_section_people_action_leave").tr(),
onTap: leaveAlbum,
),
];
@@ -89,14 +92,15 @@ class AlbumOptionsPage extends HookConsumerWidget {
actions = [
ListTile(
leading: const Icon(Icons.person_remove_rounded),
- title: const Text("Remove user from album"),
+ title: const Text("shared_album_section_people_action_remove_user")
+ .tr(),
onTap: () => removeUserFromAlbum(user),
),
];
}
showModalBottomSheet(
- backgroundColor: Theme.of(context).scaffoldBackgroundColor,
+ backgroundColor: context.scaffoldBackgroundColor,
isScrollControlled: false,
context: context,
builder: (context) {
@@ -115,28 +119,22 @@ class AlbumOptionsPage extends HookConsumerWidget {
buildOwnerInfo() {
return ListTile(
- leading: owner != null
- ? UserCircleAvatar(
- user: owner,
- useRandomBackgroundColor: true,
- )
- : const SizedBox(),
+ leading:
+ owner != null ? UserCircleAvatar(user: owner) : const SizedBox(),
title: Text(
- album.owner.value?.firstName ?? "",
+ album.owner.value?.name ?? "",
style: const TextStyle(
- fontWeight: FontWeight.bold,
+ fontWeight: FontWeight.w500,
),
),
subtitle: Text(
album.owner.value?.email ?? "",
- style: TextStyle(color: Colors.grey[500]),
- ),
- trailing: const Text(
- "Owner",
- style: TextStyle(
- fontWeight: FontWeight.bold,
- ),
+ style: TextStyle(color: Colors.grey[600]),
),
+ trailing: Text(
+ "shared_album_section_people_owner_label",
+ style: context.textTheme.labelLarge,
+ ).tr(),
);
}
@@ -149,18 +147,17 @@ class AlbumOptionsPage extends HookConsumerWidget {
return ListTile(
leading: UserCircleAvatar(
user: user,
- useRandomBackgroundColor: true,
radius: 22,
),
title: Text(
- user.firstName,
+ user.name,
style: const TextStyle(
- fontWeight: FontWeight.bold,
+ fontWeight: FontWeight.w500,
),
),
subtitle: Text(
user.email,
- style: TextStyle(color: Colors.grey[500]),
+ style: TextStyle(color: Colors.grey[600]),
),
trailing: userId == user.id || isOwner
? const Icon(Icons.more_horiz_rounded)
@@ -176,7 +173,7 @@ class AlbumOptionsPage extends HookConsumerWidget {
buildSectionTitle(String text) {
return Padding(
padding: const EdgeInsets.all(16.0),
- child: Text(text, style: Theme.of(context).textTheme.bodySmall),
+ child: Text(text, style: context.textTheme.bodySmall),
);
}
@@ -184,9 +181,7 @@ class AlbumOptionsPage extends HookConsumerWidget {
appBar: AppBar(
leading: IconButton(
icon: const Icon(Icons.arrow_back_ios_new_rounded),
- onPressed: () {
- AutoRouter.of(context).pop(null);
- },
+ onPressed: () => context.autoPop(null),
),
centerTitle: true,
title: Text("translated_text_options".tr()),
@@ -195,7 +190,34 @@ class AlbumOptionsPage extends HookConsumerWidget {
mainAxisAlignment: MainAxisAlignment.start,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
- buildSectionTitle("PEOPLE"),
+ if (isOwner && album.shared)
+ SwitchListTile.adaptive(
+ value: activityEnabled.value,
+ onChanged: (bool value) async {
+ activityEnabled.value = value;
+ if (await ref
+ .read(sharedAlbumProvider.notifier)
+ .setActivityEnabled(album, value)) {
+ album.activityEnabled = value;
+ }
+ },
+ activeColor: activityEnabled.value
+ ? context.primaryColor
+ : context.themeData.disabledColor,
+ dense: true,
+ title: Text(
+ "shared_album_activity_setting_title",
+ style: context.textTheme.titleMedium
+ ?.copyWith(fontWeight: FontWeight.w500),
+ ).tr(),
+ subtitle: Text(
+ "shared_album_activity_setting_subtitle",
+ style: context.textTheme.labelLarge?.copyWith(
+ color: context.textTheme.labelLarge?.color?.withAlpha(175),
+ ),
+ ).tr(),
+ ),
+ buildSectionTitle("shared_album_section_people_title".tr()),
buildOwnerInfo(),
buildSharedUsersList(),
],
diff --git a/mobile/lib/modules/album/views/album_viewer_page.dart b/mobile/lib/modules/album/views/album_viewer_page.dart
index dc30b3718..6d07c3b66 100644
--- a/mobile/lib/modules/album/views/album_viewer_page.dart
+++ b/mobile/lib/modules/album/views/album_viewer_page.dart
@@ -1,10 +1,11 @@
import 'dart:async';
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/models/asset_selection_page_result.model.dart';
import 'package:immich_mobile/modules/album/providers/album_detail.provider.dart';
import 'package:immich_mobile/modules/album/services/album.service.dart';
@@ -17,7 +18,6 @@ import 'package:immich_mobile/routing/router.dart';
import 'package:immich_mobile/shared/models/album.dart';
import 'package:immich_mobile/shared/models/asset.dart';
import 'package:immich_mobile/shared/providers/asset.provider.dart';
-import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
import 'package:immich_mobile/shared/ui/user_circle_avatar.dart';
import 'package:immich_mobile/shared/views/immich_loading_overlay.dart';
@@ -33,6 +33,7 @@ class AlbumViewerPage extends HookConsumerWidget {
final userId = ref.watch(authenticationProvider).userId;
final selection = useState>({});
final multiSelectEnabled = useState(false);
+ final isProcessing = useProcessingOverlay();
useEffect(
() {
@@ -67,7 +68,7 @@ class AlbumViewerPage extends HookConsumerWidget {
/// If they exist, add to selected asset state to show they are already selected.
void onAddPhotosPressed(Album albumInfo) async {
AssetSelectionPageResult? returnPayload =
- await AutoRouter.of(context).push(
+ await context.autoPush(
AssetSelectionRoute(
existingAssets: albumInfo.assets,
canDeselect: false,
@@ -75,35 +76,31 @@ class AlbumViewerPage extends HookConsumerWidget {
),
);
- if (returnPayload != null) {
+ if (returnPayload != null && returnPayload.selectedAssets.isNotEmpty) {
// Check if there is new assets add
- if (returnPayload.selectedAssets.isNotEmpty) {
- ImmichLoadingOverlayController.appLoader.show();
+ isProcessing.value = true;
- var addAssetsResult =
- await ref.watch(albumServiceProvider).addAdditionalAssetToAlbum(
- returnPayload.selectedAssets,
- albumInfo,
- );
+ var addAssetsResult =
+ await ref.watch(albumServiceProvider).addAdditionalAssetToAlbum(
+ returnPayload.selectedAssets,
+ albumInfo,
+ );
- if (addAssetsResult != null &&
- addAssetsResult.successfullyAdded > 0) {
- ref.invalidate(albumDetailProvider(albumId));
- }
-
- ImmichLoadingOverlayController.appLoader.hide();
+ if (addAssetsResult != null && addAssetsResult.successfullyAdded > 0) {
+ ref.invalidate(albumDetailProvider(albumId));
}
+
+ isProcessing.value = false;
}
}
void onAddUsersPressed(Album album) async {
- List? sharedUserIds =
- await AutoRouter.of(context).push?>(
+ List? sharedUserIds = await context.autoPush?>(
SelectAdditionalUserForSharingRoute(album: album),
);
if (sharedUserIds != null) {
- ImmichLoadingOverlayController.appLoader.show();
+ isProcessing.value = true;
var isSuccess = await ref
.watch(albumServiceProvider)
@@ -113,7 +110,7 @@ class AlbumViewerPage extends HookConsumerWidget {
ref.invalidate(albumDetailProvider(album.id));
}
- ImmichLoadingOverlayController.appLoader.hide();
+ isProcessing.value = false;
}
}
@@ -154,10 +151,7 @@ class AlbumViewerPage extends HookConsumerWidget {
padding: const EdgeInsets.only(left: 8.0),
child: Text(
album.name,
- style: const TextStyle(
- fontSize: 24,
- fontWeight: FontWeight.bold,
- ),
+ style: context.textTheme.headlineMedium,
),
),
);
@@ -171,11 +165,19 @@ class AlbumViewerPage extends HookConsumerWidget {
return const SizedBox();
}
- final String startDateText = (startDate.year == endDate.year
- ? DateFormat.MMMd()
- : DateFormat.yMMMd())
- .format(startDate);
- final String endDateText = DateFormat.yMMMd().format(endDate);
+ final String dateRangeText;
+ if (startDate.day == endDate.day &&
+ startDate.month == endDate.month &&
+ startDate.year == endDate.year) {
+ dateRangeText = DateFormat.yMMMd().format(startDate);
+ } else {
+ final String startDateText = (startDate.year == endDate.year
+ ? DateFormat.MMMd()
+ : DateFormat.yMMMd())
+ .format(startDate);
+ final String endDateText = DateFormat.yMMMd().format(endDate);
+ dateRangeText = "$startDateText - $endDateText";
+ }
return Padding(
padding: EdgeInsets.only(
@@ -183,11 +185,8 @@ class AlbumViewerPage extends HookConsumerWidget {
bottom: album.shared ? 0.0 : 8.0,
),
child: Text(
- "$startDateText - $endDateText",
- style: const TextStyle(
- fontSize: 14,
- fontWeight: FontWeight.bold,
- ),
+ dateRangeText,
+ style: context.textTheme.labelLarge,
),
);
}
@@ -195,7 +194,7 @@ class AlbumViewerPage extends HookConsumerWidget {
Widget buildSharedUserIconsRow(Album album) {
return GestureDetector(
onTap: () async {
- await AutoRouter.of(context).push(AlbumOptionsRoute(album: album));
+ await context.autoPush(AlbumOptionsRoute(album: album));
ref.invalidate(albumDetailProvider(album.id));
},
child: SizedBox(
@@ -210,7 +209,6 @@ class AlbumViewerPage extends HookConsumerWidget {
user: album.sharedUsers.toList()[index],
radius: 18,
size: 36,
- useRandomBackgroundColor: true,
),
);
}),
@@ -234,11 +232,12 @@ class AlbumViewerPage extends HookConsumerWidget {
onActivitiesPressed(Album album) {
if (album.remoteId != null) {
- AutoRouter.of(context).push(
+ context.autoPush(
ActivitiesRoute(
albumId: album.remoteId!,
appBarTitle: album.name,
isOwner: userId == album.ownerId,
+ isReadOnly: !album.activityEnabled,
),
);
}
@@ -259,13 +258,11 @@ class AlbumViewerPage extends HookConsumerWidget {
error: (error, stackTrace) => AppBar(title: const Text("Error")),
loading: () => AppBar(),
),
- body: album.when(
- data: (data) => WillPopScope(
+ body: album.widgetWhen(
+ onData: (data) => WillPopScope(
onWillPop: onWillPop,
child: GestureDetector(
- onTap: () {
- titleFocusNode.unfocus();
- },
+ onTap: () => titleFocusNode.unfocus(),
child: ImmichAssetGrid(
renderList: data.renderList,
listener: selectionListener,
@@ -279,14 +276,11 @@ class AlbumViewerPage extends HookConsumerWidget {
],
),
isOwner: userId == data.ownerId,
- sharedAlbumId: data.remoteId,
+ sharedAlbumId:
+ data.shared && data.activityEnabled ? data.remoteId : null,
),
),
),
- error: (e, _) => Center(child: Text("Error loading album info!\n$e")),
- loading: () => const Center(
- child: ImmichLoadingIndicator(),
- ),
),
);
}
diff --git a/mobile/lib/modules/album/views/asset_selection_page.dart b/mobile/lib/modules/album/views/asset_selection_page.dart
index 9c30ba80f..471a74ace 100644
--- a/mobile/lib/modules/album/views/asset_selection_page.dart
+++ b/mobile/lib/modules/album/views/asset_selection_page.dart
@@ -3,6 +3,8 @@ import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/models/asset_selection_page_result.model.dart';
import 'package:immich_mobile/modules/asset_viewer/providers/render_list.provider.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/asset_grid_data_structure.dart';
@@ -78,18 +80,14 @@ class AssetSelectionPage extends HookConsumerWidget {
canDeselect ? "share_done" : "share_add",
style: TextStyle(
fontWeight: FontWeight.bold,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
).tr(),
),
],
),
- body: renderList.when(
- data: (data) => buildBody(data),
- error: (error, stackTrace) => Center(
- child: Text(error.toString()),
- ),
- loading: () => const Center(child: CircularProgressIndicator()),
+ body: renderList.widgetWhen(
+ onData: (data) => buildBody(data),
),
);
}
diff --git a/mobile/lib/modules/album/views/create_album_page.dart b/mobile/lib/modules/album/views/create_album_page.dart
index 191ce1470..7e5fb8168 100644
--- a/mobile/lib/modules/album/views/create_album_page.dart
+++ b/mobile/lib/modules/album/views/create_album_page.dart
@@ -1,8 +1,8 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/models/asset_selection_page_result.model.dart';
import 'package:immich_mobile/modules/album/providers/album.provider.dart';
import 'package:immich_mobile/modules/album/providers/album_title.provider.dart';
@@ -34,11 +34,11 @@ class CreateAlbumPage extends HookConsumerWidget {
final selectedAssets = useState>(
initialAssets != null ? Set.from(initialAssets!) : const {},
);
- final isDarkTheme = Theme.of(context).brightness == Brightness.dark;
showSelectUserPage() async {
- final bool? ok = await AutoRouter.of(context)
- .push(SelectUserForSharingRoute(assets: selectedAssets.value));
+ final bool? ok = await context.autoPush(
+ SelectUserForSharingRoute(assets: selectedAssets.value),
+ );
if (ok == true) {
selectedAssets.value = {};
}
@@ -58,7 +58,7 @@ class CreateAlbumPage extends HookConsumerWidget {
onSelectPhotosButtonPressed() async {
AssetSelectionPageResult? selectedAsset =
- await AutoRouter.of(context).push(
+ await context.autoPush(
AssetSelectionRoute(
existingAssets: selectedAssets.value,
canDeselect: true,
@@ -94,10 +94,7 @@ class CreateAlbumPage extends HookConsumerWidget {
padding: const EdgeInsets.only(top: 200, left: 18),
child: Text(
'create_shared_album_page_share_add_assets',
- style: Theme.of(context).textTheme.displayMedium?.copyWith(
- fontSize: 12,
- fontWeight: FontWeight.normal,
- ),
+ style: context.textTheme.labelLarge,
).tr(),
),
);
@@ -117,9 +114,9 @@ class CreateAlbumPage extends HookConsumerWidget {
padding:
const EdgeInsets.symmetric(vertical: 22, horizontal: 16),
side: BorderSide(
- color: isDarkTheme
+ color: context.isDarkTheme
? const Color.fromARGB(255, 63, 63, 63)
- : const Color.fromARGB(255, 206, 206, 206),
+ : const Color.fromARGB(255, 129, 129, 129),
),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(5),
@@ -128,16 +125,15 @@ class CreateAlbumPage extends HookConsumerWidget {
onPressed: onSelectPhotosButtonPressed,
icon: Icon(
Icons.add_rounded,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
label: Padding(
padding: const EdgeInsets.only(left: 8.0),
child: Text(
'create_shared_album_page_share_select_photos',
- style: Theme.of(context).textTheme.labelLarge?.copyWith(
- fontSize: 16,
- fontWeight: FontWeight.bold,
- ),
+ style: context.textTheme.titleMedium?.copyWith(
+ color: context.primaryColor,
+ ),
).tr(),
),
),
@@ -206,7 +202,7 @@ class CreateAlbumPage extends HookConsumerWidget {
selectedAssets.value = {};
ref.watch(albumTitleProvider.notifier).clearAlbumTitle();
- AutoRouter.of(context).replace(AlbumViewerRoute(albumId: newAlbum.id));
+ context.autoReplace(AlbumViewerRoute(albumId: newAlbum.id));
}
}
@@ -214,19 +210,16 @@ class CreateAlbumPage extends HookConsumerWidget {
appBar: AppBar(
elevation: 0,
centerTitle: false,
- backgroundColor: Theme.of(context).scaffoldBackgroundColor,
+ backgroundColor: context.scaffoldBackgroundColor,
leading: IconButton(
onPressed: () {
selectedAssets.value = {};
- AutoRouter.of(context).pop();
+ context.autoPop();
},
icon: const Icon(Icons.close_rounded),
),
- title: Text(
+ title: const Text(
'share_create_album',
- style: Theme.of(context).textTheme.displayMedium?.copyWith(
- color: Theme.of(context).primaryColor,
- ),
).tr(),
actions: [
if (isSharedAlbum)
@@ -239,8 +232,8 @@ class CreateAlbumPage extends HookConsumerWidget {
style: TextStyle(
fontWeight: FontWeight.bold,
color: albumTitleController.text.isEmpty
- ? Theme.of(context).disabledColor
- : Theme.of(context).primaryColor,
+ ? context.themeData.disabledColor
+ : context.primaryColor,
),
),
),
@@ -254,7 +247,7 @@ class CreateAlbumPage extends HookConsumerWidget {
'create_shared_album_page_create'.tr(),
style: TextStyle(
fontWeight: FontWeight.bold,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
),
),
@@ -265,7 +258,7 @@ class CreateAlbumPage extends HookConsumerWidget {
child: CustomScrollView(
slivers: [
SliverAppBar(
- backgroundColor: Theme.of(context).scaffoldBackgroundColor,
+ backgroundColor: context.scaffoldBackgroundColor,
elevation: 5,
automaticallyImplyLeading: false,
pinned: true,
diff --git a/mobile/lib/modules/album/views/library_page.dart b/mobile/lib/modules/album/views/library_page.dart
index 90d095640..ce4d5e0cc 100644
--- a/mobile/lib/modules/album/views/library_page.dart
+++ b/mobile/lib/modules/album/views/library_page.dart
@@ -1,9 +1,9 @@
-import 'package:auto_route/auto_route.dart';
import 'package:collection/collection.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/providers/album.provider.dart';
import 'package:immich_mobile/modules/album/ui/album_thumbnail_card.dart';
import 'package:immich_mobile/routing/router.dart';
@@ -21,7 +21,7 @@ class LibraryPage extends HookConsumerWidget {
final trashEnabled =
ref.watch(serverInfoProvider.select((v) => v.serverFeatures.trash));
final albums = ref.watch(albumProvider);
- var isDarkMode = Theme.of(context).brightness == Brightness.dark;
+ var isDarkTheme = context.isDarkTheme;
var settings = ref.watch(appSettingsServiceProvider);
useEffect(
@@ -96,15 +96,14 @@ class LibraryPage extends HookConsumerWidget {
padding: const EdgeInsets.only(right: 12.0),
child: Icon(
Icons.check,
- color: selected
- ? Theme.of(context).primaryColor
- : Colors.transparent,
+ color:
+ selected ? context.primaryColor : Colors.transparent,
),
),
Text(
option,
style: TextStyle(
- color: selected ? Theme.of(context).primaryColor : null,
+ color: selected ? context.primaryColor : null,
fontSize: 12.0,
),
),
@@ -122,14 +121,12 @@ class LibraryPage extends HookConsumerWidget {
Icon(
Icons.swap_vert_rounded,
size: 18,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
Text(
options[selectedAlbumSortOrder.value],
- style: TextStyle(
- fontWeight: FontWeight.bold,
- color: Theme.of(context).primaryColor,
- fontSize: 12.0,
+ style: context.textTheme.labelLarge?.copyWith(
+ color: context.primaryColor,
),
),
],
@@ -140,7 +137,7 @@ class LibraryPage extends HookConsumerWidget {
Widget buildCreateAlbumButton() {
return GestureDetector(
onTap: () {
- AutoRouter.of(context).push(CreateAlbumRoute(isSharedAlbum: false));
+ context.autoPush(CreateAlbumRoute(isSharedAlbum: false));
},
child: Padding(
padding: const EdgeInsets.only(bottom: 32),
@@ -152,18 +149,18 @@ class LibraryPage extends HookConsumerWidget {
child: Container(
decoration: BoxDecoration(
border: Border.all(
- color: isDarkMode
+ color: isDarkTheme
? const Color.fromARGB(255, 53, 53, 53)
: const Color.fromARGB(255, 203, 203, 203),
),
- color: isDarkMode ? Colors.grey[900] : Colors.grey[50],
+ color: isDarkTheme ? Colors.grey[900] : Colors.grey[50],
borderRadius: BorderRadius.circular(20),
),
child: Center(
child: Icon(
Icons.add_rounded,
size: 28,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
),
),
@@ -173,11 +170,9 @@ class LibraryPage extends HookConsumerWidget {
top: 8.0,
bottom: 16,
),
- child: const Text(
+ child: Text(
'library_page_new_album',
- style: TextStyle(
- fontWeight: FontWeight.bold,
- ),
+ style: context.textTheme.labelLarge,
).tr(),
),
],
@@ -199,23 +194,23 @@ class LibraryPage extends HookConsumerWidget {
child: Text(
label,
style: TextStyle(
- fontWeight: FontWeight.bold,
- fontSize: 13.0,
- color: isDarkMode ? Colors.white : Colors.grey[800],
+ color: context.isDarkTheme
+ ? Colors.white
+ : Colors.black.withAlpha(200),
),
),
),
style: OutlinedButton.styleFrom(
padding: const EdgeInsets.symmetric(vertical: 12, horizontal: 16),
- backgroundColor: isDarkMode ? Colors.grey[900] : Colors.grey[50],
+ backgroundColor: isDarkTheme ? Colors.grey[900] : Colors.grey[50],
side: BorderSide(
- color: isDarkMode ? Colors.grey[800]! : Colors.grey[300]!,
+ color: isDarkTheme ? Colors.grey[800]! : Colors.grey[300]!,
),
alignment: Alignment.centerLeft,
),
icon: Icon(
icon,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
),
);
@@ -228,7 +223,7 @@ class LibraryPage extends HookConsumerWidget {
Widget? shareTrashButton() {
return trashEnabled
? InkWell(
- onTap: () => AutoRouter.of(context).push(const TrashRoute()),
+ onTap: () => context.autoPush(const TrashRoute()),
borderRadius: BorderRadius.circular(12),
child: const Icon(
Icons.delete_rounded,
@@ -257,12 +252,12 @@ class LibraryPage extends HookConsumerWidget {
children: [
buildLibraryNavButton(
"library_page_favorites".tr(), Icons.favorite_border, () {
- AutoRouter.of(context).navigate(const FavoritesRoute());
+ context.autoNavigate(const FavoritesRoute());
}),
const SizedBox(width: 12.0),
buildLibraryNavButton(
"library_page_archive".tr(), Icons.archive_outlined, () {
- AutoRouter.of(context).navigate(const ArchiveRoute());
+ context.autoNavigate(const ArchiveRoute());
}),
],
),
@@ -279,9 +274,11 @@ class LibraryPage extends HookConsumerWidget {
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
- const Text(
+ Text(
'library_page_albums',
- style: TextStyle(fontWeight: FontWeight.bold),
+ style: context.textTheme.bodyLarge?.copyWith(
+ fontWeight: FontWeight.w500,
+ ),
).tr(),
buildSortButton(),
],
@@ -306,7 +303,7 @@ class LibraryPage extends HookConsumerWidget {
return AlbumThumbnailCard(
album: sorted[index - 1],
- onTap: () => AutoRouter.of(context).push(
+ onTap: () => context.autoPush(
AlbumViewerRoute(
albumId: sorted[index - 1].id,
),
@@ -327,9 +324,11 @@ class LibraryPage extends HookConsumerWidget {
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
- const Text(
+ Text(
'library_page_device_albums',
- style: TextStyle(fontWeight: FontWeight.bold),
+ style: context.textTheme.bodyLarge?.copyWith(
+ fontWeight: FontWeight.w500,
+ ),
).tr(),
],
),
@@ -348,7 +347,7 @@ class LibraryPage extends HookConsumerWidget {
childCount: local.length,
(context, index) => AlbumThumbnailCard(
album: local[index],
- onTap: () => AutoRouter.of(context).push(
+ onTap: () => context.autoPush(
AlbumViewerRoute(
albumId: local[index].id,
),
diff --git a/mobile/lib/modules/album/views/select_additional_user_for_sharing_page.dart b/mobile/lib/modules/album/views/select_additional_user_for_sharing_page.dart
index e13637d23..2aad67ef5 100644
--- a/mobile/lib/modules/album/views/select_additional_user_for_sharing_page.dart
+++ b/mobile/lib/modules/album/views/select_additional_user_for_sharing_page.dart
@@ -1,12 +1,12 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/providers/suggested_shared_users.provider.dart';
import 'package:immich_mobile/shared/models/album.dart';
import 'package:immich_mobile/shared/models/user.dart';
-import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
import 'package:immich_mobile/shared/ui/user_circle_avatar.dart';
class SelectAdditionalUserForSharingPage extends HookConsumerWidget {
@@ -22,14 +22,13 @@ class SelectAdditionalUserForSharingPage extends HookConsumerWidget {
final sharedUsersList = useState>({});
addNewUsersHandler() {
- AutoRouter.of(context)
- .pop(sharedUsersList.value.map((e) => e.id).toList());
+ context.autoPop(sharedUsersList.value.map((e) => e.id).toList());
}
buildTileIcon(User user) {
if (sharedUsersList.value.contains(user)) {
return CircleAvatar(
- backgroundColor: Theme.of(context).primaryColor,
+ backgroundColor: context.primaryColor,
child: const Icon(
Icons.check_rounded,
size: 25,
@@ -50,7 +49,7 @@ class SelectAdditionalUserForSharingPage extends HookConsumerWidget {
Padding(
padding: const EdgeInsets.symmetric(horizontal: 8.0),
child: Chip(
- backgroundColor: Theme.of(context).primaryColor.withOpacity(0.15),
+ backgroundColor: context.primaryColor.withOpacity(0.15),
label: Text(
user.email,
style: const TextStyle(
@@ -124,7 +123,7 @@ class SelectAdditionalUserForSharingPage extends HookConsumerWidget {
leading: IconButton(
icon: const Icon(Icons.close_rounded),
onPressed: () {
- AutoRouter.of(context).pop(null);
+ context.autoPop(null);
},
),
actions: [
@@ -138,8 +137,8 @@ class SelectAdditionalUserForSharingPage extends HookConsumerWidget {
),
],
),
- body: suggestedShareUsers.when(
- data: (users) {
+ body: suggestedShareUsers.widgetWhen(
+ onData: (users) {
for (var sharedUsers in album.sharedUsers) {
users.removeWhere(
(u) => u.id == sharedUsers.id || u.id == album.ownerId,
@@ -148,10 +147,6 @@ class SelectAdditionalUserForSharingPage extends HookConsumerWidget {
return buildUserList(users);
},
- error: (e, _) => Text("Error loading suggested users $e"),
- loading: () => const Center(
- child: ImmichLoadingIndicator(),
- ),
),
);
}
diff --git a/mobile/lib/modules/album/views/select_user_for_sharing_page.dart b/mobile/lib/modules/album/views/select_user_for_sharing_page.dart
index fe6cef401..3d6dcf678 100644
--- a/mobile/lib/modules/album/views/select_user_for_sharing_page.dart
+++ b/mobile/lib/modules/album/views/select_user_for_sharing_page.dart
@@ -1,15 +1,15 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/providers/album_title.provider.dart';
import 'package:immich_mobile/modules/album/providers/shared_album.provider.dart';
import 'package:immich_mobile/modules/album/providers/suggested_shared_users.provider.dart';
import 'package:immich_mobile/routing/router.dart';
import 'package:immich_mobile/shared/models/asset.dart';
import 'package:immich_mobile/shared/models/user.dart';
-import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
import 'package:immich_mobile/shared/ui/user_circle_avatar.dart';
class SelectUserForSharingPage extends HookConsumerWidget {
@@ -35,14 +35,19 @@ class SelectUserForSharingPage extends HookConsumerWidget {
await ref.watch(sharedAlbumProvider.notifier).getAllSharedAlbums();
// ref.watch(assetSelectionProvider.notifier).removeAll();
ref.watch(albumTitleProvider.notifier).clearAlbumTitle();
- AutoRouter.of(context).pop(true);
- AutoRouter.of(context)
- .navigate(const TabControllerRoute(children: [SharingRoute()]));
+ context.autoPop(true);
+ context
+ .autoNavigate(const TabControllerRoute(children: [SharingRoute()]));
}
ScaffoldMessenger(
child: SnackBar(
- content: const Text('select_user_for_sharing_page_err_album').tr(),
+ content: Text(
+ 'select_user_for_sharing_page_err_album',
+ style: context.textTheme.bodyLarge?.copyWith(
+ color: context.primaryColor,
+ ),
+ ).tr(),
),
);
}
@@ -50,7 +55,7 @@ class SelectUserForSharingPage extends HookConsumerWidget {
buildTileIcon(User user) {
if (sharedUsersList.value.contains(user)) {
return CircleAvatar(
- backgroundColor: Theme.of(context).primaryColor,
+ backgroundColor: context.primaryColor,
child: const Icon(
Icons.check_rounded,
size: 25,
@@ -71,7 +76,7 @@ class SelectUserForSharingPage extends HookConsumerWidget {
Padding(
padding: const EdgeInsets.symmetric(horizontal: 8.0),
child: Chip(
- backgroundColor: Theme.of(context).primaryColor.withOpacity(0.15),
+ backgroundColor: context.primaryColor.withOpacity(0.15),
label: Text(
user.email,
style: const TextStyle(
@@ -139,20 +144,20 @@ class SelectUserForSharingPage extends HookConsumerWidget {
appBar: AppBar(
title: Text(
'share_invite',
- style: TextStyle(color: Theme.of(context).primaryColor),
+ style: TextStyle(color: context.primaryColor),
).tr(),
elevation: 0,
centerTitle: false,
leading: IconButton(
icon: const Icon(Icons.close_rounded),
onPressed: () async {
- AutoRouter.of(context).pop();
+ context.autoPop();
},
),
actions: [
TextButton(
style: TextButton.styleFrom(
- foregroundColor: Theme.of(context).primaryColor,
+ foregroundColor: context.primaryColor,
),
onPressed: sharedUsersList.value.isEmpty ? null : createSharedAlbum,
child: const Text(
@@ -160,20 +165,16 @@ class SelectUserForSharingPage extends HookConsumerWidget {
style: TextStyle(
fontSize: 14,
fontWeight: FontWeight.bold,
- // color: Theme.of(context).primaryColor,
+ // color: context.primaryColor,
),
).tr(),
),
],
),
- body: suggestedShareUsers.when(
- data: (users) {
+ body: suggestedShareUsers.widgetWhen(
+ onData: (users) {
return buildUserList(users);
},
- error: (e, _) => Text("Error loading suggested users $e"),
- loading: () => const Center(
- child: ImmichLoadingIndicator(),
- ),
),
);
}
diff --git a/mobile/lib/modules/album/views/sharing_page.dart b/mobile/lib/modules/album/views/sharing_page.dart
index 5e74aef67..2e2e44aca 100644
--- a/mobile/lib/modules/album/views/sharing_page.dart
+++ b/mobile/lib/modules/album/views/sharing_page.dart
@@ -1,8 +1,8 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/providers/shared_album.provider.dart';
import 'package:immich_mobile/modules/album/ui/album_thumbnail_card.dart';
import 'package:immich_mobile/modules/partner/providers/partner.provider.dart';
@@ -21,7 +21,6 @@ class SharingPage extends HookConsumerWidget {
final List sharedAlbums = ref.watch(sharedAlbumProvider);
final userId = ref.watch(currentUserProvider)?.id;
final partner = ref.watch(partnerSharedWithProvider);
- var isDarkMode = Theme.of(context).brightness == Brightness.dark;
useEffect(
() {
@@ -47,8 +46,9 @@ class SharingPage extends HookConsumerWidget {
album: sharedAlbums[index],
showOwner: true,
onTap: () {
- AutoRouter.of(context)
- .push(AlbumViewerRoute(albumId: sharedAlbums[index].id));
+ context.autoPush(
+ AlbumViewerRoute(albumId: sharedAlbums[index].id),
+ );
},
);
},
@@ -79,32 +79,27 @@ class SharingPage extends HookConsumerWidget {
album.name,
maxLines: 1,
overflow: TextOverflow.ellipsis,
- style: Theme.of(context).textTheme.bodyMedium?.copyWith(
- fontWeight: FontWeight.bold,
- color: isDarkMode
- ? Theme.of(context).primaryColor
- : Colors.black,
- ),
+ style: context.textTheme.bodyMedium?.copyWith(
+ color: context.primaryColor,
+ fontWeight: FontWeight.w500,
+ ),
),
subtitle: isOwner
? Text(
'album_thumbnail_owned'.tr(),
- style: const TextStyle(
- fontSize: 12.0,
- ),
+ style: context.textTheme.bodyMedium,
)
: album.ownerName != null
? Text(
'album_thumbnail_shared_by'
.tr(args: [album.ownerName!]),
- style: const TextStyle(
- fontSize: 12.0,
- ),
+ style: context.textTheme.bodyMedium,
)
: null,
onTap: () {
- AutoRouter.of(context)
- .push(AlbumViewerRoute(albumId: sharedAlbums[index].id));
+ context.autoPush(
+ AlbumViewerRoute(albumId: sharedAlbums[index].id),
+ );
},
);
},
@@ -127,8 +122,7 @@ class SharingPage extends HookConsumerWidget {
Expanded(
child: ElevatedButton.icon(
onPressed: () {
- AutoRouter.of(context)
- .push(CreateAlbumRoute(isSharedAlbum: true));
+ context.autoPush(CreateAlbumRoute(isSharedAlbum: true));
},
icon: const Icon(
Icons.photo_album_outlined,
@@ -138,8 +132,8 @@ class SharingPage extends HookConsumerWidget {
"sharing_silver_appbar_create_shared_album",
maxLines: 1,
style: TextStyle(
- fontWeight: FontWeight.bold,
- fontSize: 11,
+ fontWeight: FontWeight.w500,
+ fontSize: 12,
),
).tr(),
),
@@ -147,8 +141,7 @@ class SharingPage extends HookConsumerWidget {
const SizedBox(width: 12.0),
Expanded(
child: ElevatedButton.icon(
- onPressed: () =>
- AutoRouter.of(context).push(const SharedLinkRoute()),
+ onPressed: () => context.autoPush(const SharedLinkRoute()),
icon: const Icon(
Icons.link,
size: 20,
@@ -156,8 +149,8 @@ class SharingPage extends HookConsumerWidget {
label: const Text(
"sharing_silver_appbar_shared_links",
style: TextStyle(
- fontWeight: FontWeight.bold,
- fontSize: 11,
+ fontWeight: FontWeight.w500,
+ fontSize: 12,
),
maxLines: 1,
).tr(),
@@ -191,21 +184,21 @@ class SharingPage extends HookConsumerWidget {
child: Icon(
Icons.insert_photo_rounded,
size: 50,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
),
Padding(
padding: const EdgeInsets.all(8.0),
child: Text(
'sharing_page_empty_list',
- style: Theme.of(context).textTheme.displaySmall,
+ style: context.textTheme.displaySmall,
).tr(),
),
Padding(
padding: const EdgeInsets.all(8.0),
child: Text(
'sharing_page_description',
- style: Theme.of(context).textTheme.bodyMedium,
+ style: context.textTheme.bodyMedium,
).tr(),
),
],
@@ -218,7 +211,7 @@ class SharingPage extends HookConsumerWidget {
Widget sharePartnerButton() {
return InkWell(
- onTap: () => AutoRouter.of(context).push(const PartnerRoute()),
+ onTap: () => context.autoPush(const PartnerRoute()),
borderRadius: BorderRadius.circular(12),
child: const Icon(
Icons.swap_horizontal_circle_rounded,
@@ -238,9 +231,11 @@ class SharingPage extends HookConsumerWidget {
SliverPadding(
padding: const EdgeInsets.all(12),
sliver: SliverToBoxAdapter(
- child: const Text(
+ child: Text(
"partner_page_title",
- style: TextStyle(fontWeight: FontWeight.bold),
+ style: context.textTheme.bodyLarge?.copyWith(
+ fontWeight: FontWeight.w500,
+ ),
).tr(),
),
),
@@ -248,10 +243,10 @@ class SharingPage extends HookConsumerWidget {
SliverPadding(
padding: const EdgeInsets.all(12),
sliver: SliverToBoxAdapter(
- child: const Text(
+ child: Text(
"sharing_page_album",
- style: TextStyle(
- fontWeight: FontWeight.bold,
+ style: context.textTheme.bodyLarge?.copyWith(
+ fontWeight: FontWeight.w500,
),
).tr(),
),
diff --git a/mobile/lib/modules/archive/providers/archive_asset_provider.dart b/mobile/lib/modules/archive/providers/archive_asset_provider.dart
index 328b3c7b5..579ace7c5 100644
--- a/mobile/lib/modules/archive/providers/archive_asset_provider.dart
+++ b/mobile/lib/modules/archive/providers/archive_asset_provider.dart
@@ -1,15 +1,14 @@
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/asset_grid_data_structure.dart';
-import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart';
-import 'package:immich_mobile/modules/settings/services/app_settings.service.dart';
import 'package:immich_mobile/shared/models/asset.dart';
import 'package:immich_mobile/shared/providers/db.provider.dart';
import 'package:immich_mobile/shared/providers/user.provider.dart';
+import 'package:immich_mobile/utils/renderlist_generator.dart';
import 'package:isar/isar.dart';
-final archiveProvider = StreamProvider((ref) async* {
+final archiveProvider = StreamProvider((ref) {
final user = ref.watch(currentUserProvider);
- if (user == null) return;
+ if (user == null) return const Stream.empty();
final query = ref
.watch(dbProvider)
.assets
@@ -19,11 +18,5 @@ final archiveProvider = StreamProvider((ref) async* {
.isArchivedEqualTo(true)
.isTrashedEqualTo(false)
.sortByFileCreatedAt();
- final settings = ref.watch(appSettingsServiceProvider);
- final groupBy =
- GroupAssetsBy.values[settings.getSetting(AppSettingsEnum.groupAssetsBy)];
- yield await RenderList.fromQuery(query, groupBy);
- await for (final _ in query.watchLazy()) {
- yield await RenderList.fromQuery(query, groupBy);
- }
+ return renderListGenerator(query, ref);
});
diff --git a/mobile/lib/modules/archive/views/archive_page.dart b/mobile/lib/modules/archive/views/archive_page.dart
index 2e1c1cd4a..fb3cecc10 100644
--- a/mobile/lib/modules/archive/views/archive_page.dart
+++ b/mobile/lib/modules/archive/views/archive_page.dart
@@ -1,8 +1,9 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/archive/providers/archive_asset_provider.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/immich_asset_grid.dart';
import 'package:immich_mobile/shared/models/asset.dart';
@@ -30,7 +31,7 @@ class ArchivePage extends HookConsumerWidget {
AppBar buildAppBar(String count) {
return AppBar(
leading: IconButton(
- onPressed: () => AutoRouter.of(context).pop(),
+ onPressed: () => context.autoPop(),
icon: const Icon(Icons.arrow_back_ios_rounded),
),
centerTitle: true,
@@ -48,37 +49,33 @@ class ArchivePage extends HookConsumerWidget {
child: SizedBox(
height: 64,
child: Card(
- child: Column(
- children: [
- ListTile(
- shape: RoundedRectangleBorder(
- borderRadius: BorderRadius.circular(10),
- ),
- leading: const Icon(
- Icons.unarchive_rounded,
- ),
- title: Text(
- 'control_bottom_app_bar_unarchive'.tr(),
- style: const TextStyle(fontSize: 14),
- ),
- onTap: processing.value
- ? null
- : () async {
- processing.value = true;
- try {
- await handleArchiveAssets(
- ref,
- context,
- selection.value.toList(),
- shouldArchive: false,
- );
- } finally {
- processing.value = false;
- selectionEnabledHook.value = false;
- }
- },
- ),
- ],
+ child: ListTile(
+ shape: const RoundedRectangleBorder(
+ borderRadius: BorderRadius.all(Radius.circular(10)),
+ ),
+ leading: const Icon(
+ Icons.unarchive_rounded,
+ ),
+ title: Text(
+ 'control_bottom_app_bar_unarchive'.tr(),
+ style: const TextStyle(fontSize: 14),
+ ),
+ onTap: processing.value
+ ? null
+ : () async {
+ processing.value = true;
+ try {
+ await handleArchiveAssets(
+ ref,
+ context,
+ selection.value.toList(),
+ shouldArchive: false,
+ );
+ } finally {
+ processing.value = false;
+ selectionEnabledHook.value = false;
+ }
+ },
),
),
),
@@ -86,18 +83,13 @@ class ArchivePage extends HookConsumerWidget {
);
}
- return archivedAssets.when(
- loading: () => Scaffold(
- appBar: buildAppBar("?"),
- body: const Center(child: CircularProgressIndicator()),
+ return Scaffold(
+ appBar: archivedAssets.maybeWhen(
+ data: (data) => buildAppBar(data.totalAssets.toString()),
+ orElse: () => buildAppBar("?"),
),
- error: (error, stackTrace) => Scaffold(
- appBar: buildAppBar("Error"),
- body: Center(child: Text(error.toString())),
- ),
- data: (data) => Scaffold(
- appBar: buildAppBar(data.totalAssets.toString()),
- body: data.isEmpty
+ body: archivedAssets.widgetWhen(
+ onData: (data) => data.isEmpty
? Center(
child: Text('archive_page_no_archived_assets'.tr()),
)
diff --git a/mobile/lib/modules/asset_viewer/providers/image_viewer_page_state.provider.dart b/mobile/lib/modules/asset_viewer/providers/image_viewer_page_state.provider.dart
index 6df633f12..06cc128f7 100644
--- a/mobile/lib/modules/asset_viewer/providers/image_viewer_page_state.provider.dart
+++ b/mobile/lib/modules/asset_viewer/providers/image_viewer_page_state.provider.dart
@@ -2,6 +2,7 @@ import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:fluttertoast/fluttertoast.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/services/album.service.dart';
import 'package:immich_mobile/modules/asset_viewer/models/image_viewer_page_state.model.dart';
import 'package:immich_mobile/modules/asset_viewer/services/image_viewer.service.dart';
@@ -67,7 +68,7 @@ class ImageViewerStateNotifier extends StateNotifier {
gravity: ToastGravity.BOTTOM,
);
}
- Navigator.of(buildContext).pop();
+ buildContext.pop();
},
);
return const ShareDialog();
diff --git a/mobile/lib/modules/asset_viewer/providers/render_list.provider.dart b/mobile/lib/modules/asset_viewer/providers/render_list.provider.dart
index 04532ce1b..c2e8782bb 100644
--- a/mobile/lib/modules/asset_viewer/providers/render_list.provider.dart
+++ b/mobile/lib/modules/asset_viewer/providers/render_list.provider.dart
@@ -3,6 +3,7 @@ import 'package:immich_mobile/modules/home/ui/asset_grid/asset_grid_data_structu
import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart';
import 'package:immich_mobile/modules/settings/services/app_settings.service.dart';
import 'package:immich_mobile/shared/models/asset.dart';
+import 'package:immich_mobile/utils/renderlist_generator.dart';
import 'package:isar/isar.dart';
final renderListProvider =
@@ -17,16 +18,6 @@ final renderListProvider =
final renderListQueryProvider = StreamProvider.family?>(
- (ref, query) async* {
- if (query == null) {
- return;
- }
- final settings = ref.watch(appSettingsServiceProvider);
- final groupBy = GroupAssetsBy
- .values[settings.getSetting(AppSettingsEnum.groupAssetsBy)];
- yield await RenderList.fromQuery(query, groupBy);
- await for (final _ in query.watchLazy()) {
- yield await RenderList.fromQuery(query, groupBy);
- }
- },
+ (ref, query) =>
+ query == null ? const Stream.empty() : renderListGenerator(query, ref),
);
diff --git a/mobile/lib/modules/asset_viewer/services/image_viewer.service.dart b/mobile/lib/modules/asset_viewer/services/image_viewer.service.dart
index 3a356c840..27be7029d 100644
--- a/mobile/lib/modules/asset_viewer/services/image_viewer.service.dart
+++ b/mobile/lib/modules/asset_viewer/services/image_viewer.service.dart
@@ -1,6 +1,5 @@
import 'dart:io';
-import 'package:flutter/foundation.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/shared/models/asset.dart';
import 'package:immich_mobile/shared/providers/api.provider.dart';
@@ -84,8 +83,8 @@ class ImageViewerService {
}
return entity != null;
}
- } catch (e) {
- debugPrint("Error saving file $e");
+ } catch (error, stack) {
+ _log.severe("Error saving file ${error.toString()}", error, stack);
return false;
}
}
diff --git a/mobile/lib/modules/asset_viewer/ui/advanced_bottom_sheet.dart b/mobile/lib/modules/asset_viewer/ui/advanced_bottom_sheet.dart
index 441492c39..c265346b0 100644
--- a/mobile/lib/modules/asset_viewer/ui/advanced_bottom_sheet.dart
+++ b/mobile/lib/modules/asset_viewer/ui/advanced_bottom_sheet.dart
@@ -1,6 +1,7 @@
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/shared/models/asset.dart';
class AdvancedBottomSheet extends HookConsumerWidget {
@@ -11,8 +12,6 @@ class AdvancedBottomSheet extends HookConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
- var isDarkMode = Theme.of(context).brightness == Brightness.dark;
-
return SingleChildScrollView(
child: Card(
shape: const RoundedRectangleBorder(
@@ -40,7 +39,9 @@ class AdvancedBottomSheet extends HookConsumerWidget {
const SizedBox(height: 32.0),
Container(
decoration: BoxDecoration(
- color: isDarkMode ? Colors.grey[900] : Colors.grey[200],
+ color: context.isDarkTheme
+ ? Colors.grey[900]
+ : Colors.grey[200],
borderRadius: BorderRadius.circular(15.0),
),
child: Padding(
@@ -61,8 +62,14 @@ class AdvancedBottomSheet extends HookConsumerWidget {
ClipboardData(text: assetDetail.toString()),
).then((_) {
ScaffoldMessenger.of(context).showSnackBar(
- const SnackBar(
- content: Text("Copied to clipboard"),
+ SnackBar(
+ content: Text(
+ "Copied to clipboard",
+ style: context.textTheme.bodyLarge
+ ?.copyWith(
+ color: context.primaryColor,
+ ),
+ ),
),
);
});
@@ -70,7 +77,7 @@ class AdvancedBottomSheet extends HookConsumerWidget {
icon: Icon(
Icons.copy,
size: 16.0,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
),
),
diff --git a/mobile/lib/modules/asset_viewer/ui/description_input.dart b/mobile/lib/modules/asset_viewer/ui/description_input.dart
index e00739faf..c5972a822 100644
--- a/mobile/lib/modules/asset_viewer/ui/description_input.dart
+++ b/mobile/lib/modules/asset_viewer/ui/description_input.dart
@@ -2,6 +2,7 @@ import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/asset_viewer/providers/asset_description.provider.dart';
import 'package:immich_mobile/shared/models/asset.dart';
import 'package:immich_mobile/shared/providers/user.provider.dart';
@@ -19,8 +20,7 @@ class DescriptionInput extends HookConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
- final isDarkTheme = Theme.of(context).brightness == Brightness.dark;
- final textColor = isDarkTheme ? Colors.white : Colors.black;
+ final textColor = context.isDarkTheme ? Colors.white : Colors.black;
final controller = useTextEditingController();
final focusNode = useFocusNode();
final isFocus = useState(false);
@@ -93,15 +93,11 @@ class DescriptionInput extends HookConsumerWidget {
maxLines: null,
keyboardType: TextInputType.multiline,
controller: controller,
- style: const TextStyle(
- fontSize: 14,
- ),
+ style: context.textTheme.labelLarge,
decoration: InputDecoration(
hintText: 'description_input_hint_text'.tr(),
border: InputBorder.none,
- hintStyle: TextStyle(
- fontWeight: FontWeight.normal,
- fontSize: 12,
+ hintStyle: context.textTheme.labelLarge?.copyWith(
color: textColor.withOpacity(0.5),
),
suffixIcon: suffixIcon,
diff --git a/mobile/lib/modules/asset_viewer/ui/exif_bottom_sheet.dart b/mobile/lib/modules/asset_viewer/ui/exif_bottom_sheet.dart
index f194738a2..08a6a0515 100644
--- a/mobile/lib/modules/asset_viewer/ui/exif_bottom_sheet.dart
+++ b/mobile/lib/modules/asset_viewer/ui/exif_bottom_sheet.dart
@@ -4,6 +4,7 @@ import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_map/flutter_map.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:timezone/timezone.dart';
import 'package:immich_mobile/modules/asset_viewer/ui/description_input.dart';
import 'package:immich_mobile/modules/map/ui/map_thumbnail.dart';
@@ -28,7 +29,7 @@ class ExifBottomSheet extends HookConsumerWidget {
exifInfo.longitude != 0;
String formatTimeZone(Duration d) =>
- "GMT${d.isNegative ? '-': '+'}${d.inHours.abs().toString().padLeft(2, '0')}:${d.inMinutes.abs().remainder(60).toString().padLeft(2, '0')}";
+ "GMT${d.isNegative ? '-' : '+'}${d.inHours.abs().toString().padLeft(2, '0')}:${d.inMinutes.abs().remainder(60).toString().padLeft(2, '0')}";
String get formattedDateTime {
DateTime dt = asset.fileCreatedAt.toLocal();
@@ -41,10 +42,16 @@ class ExifBottomSheet extends HookConsumerWidget {
final location = getLocation(asset.exifInfo!.timeZone!);
dt = TZDateTime.from(dt, location);
} on LocationNotFoundException {
- RegExp re = RegExp(r'^utc(?:([+-]\d{1,2})(?::(\d{2}))?)?$', caseSensitive: false);
+ RegExp re = RegExp(
+ r'^utc(?:([+-]\d{1,2})(?::(\d{2}))?)?$',
+ caseSensitive: false,
+ );
final m = re.firstMatch(asset.exifInfo!.timeZone!);
if (m != null) {
- final duration = Duration(hours: int.parse(m.group(1) ?? '0'), minutes: int.parse(m.group(2) ?? '0'));
+ final duration = Duration(
+ hours: int.parse(m.group(1) ?? '0'),
+ minutes: int.parse(m.group(2) ?? '0'),
+ );
dt = dt.add(duration);
timeZone = formatTimeZone(duration);
}
@@ -105,8 +112,7 @@ class ExifBottomSheet extends HookConsumerWidget {
Widget build(BuildContext context, WidgetRef ref) {
final assetWithExif = ref.watch(assetDetailProvider(asset));
final exifInfo = (assetWithExif.value ?? asset).exifInfo;
- var isDarkTheme = Theme.of(context).brightness == Brightness.dark;
- var textColor = isDarkTheme ? Colors.white : Colors.black;
+ var textColor = context.isDarkTheme ? Colors.white : Colors.black;
buildMap() {
return Padding(
@@ -187,21 +193,15 @@ class ExifBottomSheet extends HookConsumerWidget {
children: [
Text(
"exif_bottom_sheet_location",
- style: TextStyle(
- fontSize: 11,
- color: textColor,
- fontWeight: FontWeight.bold,
+ style: context.textTheme.labelMedium?.copyWith(
+ color: context.textTheme.labelMedium?.color?.withAlpha(200),
+ fontWeight: FontWeight.w600,
),
).tr(),
buildMap(),
RichText(
text: TextSpan(
- style: TextStyle(
- fontSize: 12,
- fontWeight: FontWeight.bold,
- color: textColor,
- fontFamily: 'WorkSans',
- ),
+ style: context.textTheme.labelLarge,
children: [
if (exifInfo != null && exifInfo.city != null)
TextSpan(
@@ -222,7 +222,9 @@ class ExifBottomSheet extends HookConsumerWidget {
),
Text(
"${exifInfo!.latitude!.toStringAsFixed(4)}, ${exifInfo.longitude!.toStringAsFixed(4)}",
- style: const TextStyle(fontSize: 12),
+ style: context.textTheme.labelMedium?.copyWith(
+ color: context.textTheme.labelMedium?.color?.withAlpha(150),
+ ),
),
],
),
@@ -252,10 +254,7 @@ class ExifBottomSheet extends HookConsumerWidget {
titleAlignment: ListTileTitleAlignment.center,
title: Text(
title,
- style: TextStyle(
- fontWeight: FontWeight.bold,
- color: textColor,
- ),
+ style: context.textTheme.labelLarge,
),
subtitle: subtitle,
);
@@ -272,7 +271,7 @@ class ExifBottomSheet extends HookConsumerWidget {
// There is both filename and size information
return createImagePropertiesListStyle(
asset.fileName,
- Text(imgSizeString),
+ Text(imgSizeString, style: context.textTheme.bodySmall),
);
} else if (imgSizeString != null && asset.fileName.isEmpty) {
// There is only size information
@@ -299,10 +298,9 @@ class ExifBottomSheet extends HookConsumerWidget {
padding: const EdgeInsets.only(bottom: 8.0),
child: Text(
"exif_bottom_sheet_details",
- style: TextStyle(
- fontSize: 11,
- color: textColor,
- fontWeight: FontWeight.bold,
+ style: context.textTheme.labelMedium?.copyWith(
+ color: context.textTheme.labelMedium?.color?.withAlpha(200),
+ fontWeight: FontWeight.w600,
),
).tr(),
),
@@ -317,14 +315,17 @@ class ExifBottomSheet extends HookConsumerWidget {
),
title: Text(
"${exifInfo!.make} ${exifInfo.model}",
- style: TextStyle(
- color: textColor,
- fontWeight: FontWeight.bold,
- ),
+ style: context.textTheme.labelLarge,
),
- subtitle: exifInfo.f != null || exifInfo.exposureSeconds != null || exifInfo.mm != null || exifInfo.iso != null ? Text(
- "ƒ/${exifInfo.fNumber} ${exifInfo.exposureTime} ${exifInfo.focalLength} mm ISO ${exifInfo.iso ?? ''} ",
- ) : null,
+ subtitle: exifInfo.f != null ||
+ exifInfo.exposureSeconds != null ||
+ exifInfo.mm != null ||
+ exifInfo.iso != null
+ ? Text(
+ "ƒ/${exifInfo.fNumber} ${exifInfo.exposureTime} ${exifInfo.focalLength} mm ISO ${exifInfo.iso ?? ''} ",
+ style: context.textTheme.bodySmall,
+ )
+ : null,
),
],
);
@@ -393,7 +394,7 @@ class ExifBottomSheet extends HookConsumerWidget {
data: (data) => DescriptionInput(asset: data),
error: (error, stackTrace) => Icon(
Icons.image_not_supported_outlined,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
loading: () => const SizedBox(
width: 75,
diff --git a/mobile/lib/modules/asset_viewer/ui/top_control_app_bar.dart b/mobile/lib/modules/asset_viewer/ui/top_control_app_bar.dart
index 95965f6d8..3e6dfe2ee 100644
--- a/mobile/lib/modules/asset_viewer/ui/top_control_app_bar.dart
+++ b/mobile/lib/modules/asset_viewer/ui/top_control_app_bar.dart
@@ -1,6 +1,6 @@
-import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/activities/providers/activity.provider.dart';
import 'package:immich_mobile/shared/models/asset.dart';
import 'package:immich_mobile/shared/providers/asset.provider.dart';
@@ -147,7 +147,7 @@ class TopControlAppBar extends HookConsumerWidget {
Widget buildBackButton() {
return IconButton(
onPressed: () {
- AutoRouter.of(context).pop();
+ context.autoPop();
},
icon: Icon(
Icons.arrow_back_ios_new_rounded,
diff --git a/mobile/lib/modules/asset_viewer/views/gallery_viewer.dart b/mobile/lib/modules/asset_viewer/views/gallery_viewer.dart
index f8acef588..9948ab4cc 100644
--- a/mobile/lib/modules/asset_viewer/views/gallery_viewer.dart
+++ b/mobile/lib/modules/asset_viewer/views/gallery_viewer.dart
@@ -8,6 +8,7 @@ import 'package:flutter/services.dart';
import 'package:flutter_hooks/flutter_hooks.dart' hide Store;
import 'package:fluttertoast/fluttertoast.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/asset_viewer/providers/asset_stack.provider.dart';
import 'package:immich_mobile/modules/asset_viewer/providers/show_controls.provider.dart';
import 'package:immich_mobile/modules/asset_viewer/providers/video_player_controls_provider.dart';
@@ -209,7 +210,7 @@ class GalleryViewerPage extends HookConsumerWidget {
if (isDeleted && isParent) {
if (totalAssets == 1) {
// Handle only one asset
- AutoRouter.of(context).pop();
+ context.autoPop();
} else {
// Go to next page otherwise
controller.nextPage(
@@ -293,7 +294,7 @@ class GalleryViewerPage extends HookConsumerWidget {
final ratio = d.dy / max(d.dx.abs(), 1);
if (d.dy > sensitivity && ratio > ratioThreshold) {
- AutoRouter.of(context).pop();
+ context.autoPop();
} else if (d.dy < -sensitivity && ratio < -ratioThreshold) {
showInfo();
}
@@ -308,7 +309,7 @@ class GalleryViewerPage extends HookConsumerWidget {
.watch(assetProvider.notifier)
.toggleArchive([asset], !asset.isArchived);
if (isParent) {
- AutoRouter.of(context).pop();
+ context.autoPop();
return;
}
removeAssetFromStack();
@@ -331,7 +332,7 @@ class GalleryViewerPage extends HookConsumerWidget {
handleActivities() {
if (sharedAlbumId != null) {
- AutoRouter.of(context).push(
+ context.autoPush(
ActivitiesRoute(
albumId: sharedAlbumId!,
assetId: asset().remoteId,
@@ -513,8 +514,8 @@ class GalleryViewerPage extends HookConsumerWidget {
currentAsset,
stackElements.elementAt(stackIndex.value),
);
- Navigator.pop(ctx);
- AutoRouter.of(context).pop();
+ ctx.pop();
+ context.autoPop();
},
title: const Text(
"viewer_stack_use_as_main_asset",
@@ -540,8 +541,8 @@ class GalleryViewerPage extends HookConsumerWidget {
stackElements.elementAt(1),
childrenToRemove: [currentAsset],
);
- Navigator.pop(ctx);
- AutoRouter.of(context).pop();
+ ctx.pop();
+ context.autoPop();
} else {
await ref.read(assetStackServiceProvider).updateStack(
currentAsset,
@@ -550,7 +551,7 @@ class GalleryViewerPage extends HookConsumerWidget {
],
);
removeAssetFromStack();
- Navigator.pop(ctx);
+ ctx.pop();
}
},
title: const Text(
@@ -568,8 +569,8 @@ class GalleryViewerPage extends HookConsumerWidget {
currentAsset,
childrenToRemove: stack,
);
- Navigator.pop(ctx);
- AutoRouter.of(context).pop();
+ ctx.pop();
+ context.autoPop();
},
title: const Text(
"viewer_unstack",
@@ -829,8 +830,8 @@ class GalleryViewerPage extends HookConsumerWidget {
placeholder: Image(
image: provider,
fit: BoxFit.fitWidth,
- height: MediaQuery.of(context).size.height,
- width: MediaQuery.of(context).size.width,
+ height: context.height,
+ width: context.width,
alignment: Alignment.center,
),
onVideoEnded: () {
diff --git a/mobile/lib/modules/asset_viewer/views/video_viewer_page.dart b/mobile/lib/modules/asset_viewer/views/video_viewer_page.dart
index a56d65595..ad929a8c5 100644
--- a/mobile/lib/modules/asset_viewer/views/video_viewer_page.dart
+++ b/mobile/lib/modules/asset_viewer/views/video_viewer_page.dart
@@ -3,6 +3,7 @@ import 'dart:io';
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:chewie/chewie.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/asset_viewer/models/image_viewer_page_state.model.dart';
import 'package:immich_mobile/modules/asset_viewer/providers/image_viewer_page_state.provider.dart';
import 'package:immich_mobile/modules/asset_viewer/ui/video_player_controls.dart';
@@ -44,7 +45,7 @@ class VideoViewerPage extends HookConsumerWidget {
),
error: (error, stackTrace) => Icon(
Icons.image_not_supported_outlined,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
loading: () => const Center(
child: SizedBox(
@@ -74,8 +75,8 @@ class VideoViewerPage extends HookConsumerWidget {
),
if (downloadAssetStatus == DownloadAssetStatus.loading)
SizedBox(
- height: MediaQuery.of(context).size.height,
- width: MediaQuery.of(context).size.width,
+ height: context.height,
+ width: context.width,
child: const Center(
child: ImmichLoadingIndicator(),
),
@@ -205,8 +206,8 @@ class _VideoPlayerState extends State {
);
} else {
return SizedBox(
- height: MediaQuery.of(context).size.height,
- width: MediaQuery.of(context).size.width,
+ height: context.height,
+ width: context.width,
child: Center(
child: Stack(
children: [
diff --git a/mobile/lib/modules/backup/providers/backup.provider.dart b/mobile/lib/modules/backup/providers/backup.provider.dart
index 0df8bc90d..3d8d6414c 100644
--- a/mobile/lib/modules/backup/providers/backup.provider.dart
+++ b/mobile/lib/modules/backup/providers/backup.provider.dart
@@ -89,7 +89,6 @@ class BackupNotifier extends StateNotifier {
state = state
.copyWith(selectedBackupAlbums: {...state.selectedBackupAlbums, album});
- _updateBackupAssetCount();
}
void addExcludedAlbumForBackup(AvailableAlbum album) {
@@ -98,7 +97,6 @@ class BackupNotifier extends StateNotifier {
}
state = state
.copyWith(excludedBackupAlbums: {...state.excludedBackupAlbums, album});
- _updateBackupAssetCount();
}
void removeAlbumForBackup(AvailableAlbum album) {
@@ -107,7 +105,6 @@ class BackupNotifier extends StateNotifier {
currentSelectedAlbums.removeWhere((a) => a == album);
state = state.copyWith(selectedBackupAlbums: currentSelectedAlbums);
- _updateBackupAssetCount();
}
void removeExcludedAlbumForBackup(AvailableAlbum album) {
@@ -116,7 +113,20 @@ class BackupNotifier extends StateNotifier {
currentExcludedAlbums.removeWhere((a) => a == album);
state = state.copyWith(excludedBackupAlbums: currentExcludedAlbums);
- _updateBackupAssetCount();
+ }
+
+ Future backupAlbumSelectionDone() {
+ if (state.selectedBackupAlbums.isEmpty) {
+ // disable any backup
+ cancelBackup();
+ setAutoBackup(false);
+ configureBackgroundBackup(
+ enabled: false,
+ onError: (msg) {},
+ onBatteryInfo: () {},
+ );
+ }
+ return _updateBackupAssetCount();
}
void setAutoBackup(bool enabled) {
@@ -249,30 +259,6 @@ class BackupNotifier extends StateNotifier {
final List selectedBackupAlbums =
await _backupService.selectedAlbumsQuery().findAll();
- // First time backup - set isAll album is the default one for backup.
- if (selectedBackupAlbums.isEmpty) {
- log.info("First time backup; setup 'Recent(s)' album as default");
-
- // Get album that contains all assets
- final list = await PhotoManager.getAssetPathList(
- hasAll: true,
- onlyAll: true,
- type: RequestType.common,
- );
-
- if (list.isEmpty) {
- return;
- }
- AssetPathEntity albumHasAllAssets = list.first;
-
- final ba = BackupAlbum(
- albumHasAllAssets.id,
- DateTime.fromMillisecondsSinceEpoch(0),
- BackupSelection.select,
- );
- await _db.writeTxn(() => _db.backupAlbums.put(ba));
- }
-
// Generate AssetPathEntity from id to add to local state
final Set selectedAlbums = {};
for (final BackupAlbum ba in selectedBackupAlbums) {
@@ -362,7 +348,6 @@ class BackupNotifier extends StateNotifier {
allUniqueAssets: {},
selectedAlbumsBackupAssetsIds: selectedAlbumsBackupAssets,
);
- return;
} else {
state = state.copyWith(
allAssetsInDatabase: allAssetsInDatabase,
@@ -373,8 +358,6 @@ class BackupNotifier extends StateNotifier {
// Save to persistent storage
await _updatePersistentAlbumsSelection();
-
- return;
}
/// Get all necessary information for calculating the available albums,
diff --git a/mobile/lib/modules/backup/providers/manual_upload.provider.dart b/mobile/lib/modules/backup/providers/manual_upload.provider.dart
index 6d585e289..4d643b9a1 100644
--- a/mobile/lib/modules/backup/providers/manual_upload.provider.dart
+++ b/mobile/lib/modules/backup/providers/manual_upload.provider.dart
@@ -274,7 +274,7 @@ class ManualUploadNotifier extends StateNotifier {
// The app is currently in background. Perform the necessary cleanups which
// are on-hold for upload completion
if (appState != AppStateEnum.active && appState != AppStateEnum.resumed) {
- ref.read(appStateProvider.notifier).handleAppInactivity();
+ ref.read(backupProvider.notifier).cancelBackup();
}
}
diff --git a/mobile/lib/modules/backup/services/backup.service.dart b/mobile/lib/modules/backup/services/backup.service.dart
index 15cc0c349..f4ca5932a 100644
--- a/mobile/lib/modules/backup/services/backup.service.dart
+++ b/mobile/lib/modules/backup/services/backup.service.dart
@@ -42,6 +42,9 @@ class BackupService {
try {
return await _apiService.assetApi.getUserAssetsByDeviceId(deviceId);
+
+ // TODO! Start using this in 1.92.0
+ // return await _apiService.assetApi.getAllUserAssetsByDeviceId(deviceId);
} catch (e) {
debugPrint('Error [getDeviceBackupAsset] ${e.toString()}');
return null;
@@ -275,13 +278,6 @@ class BackupService {
req.files.add(assetRawUploadData);
- if (entity.isLivePhoto) {
- var livePhotoRawUploadData = await _getLivePhotoFile(entity);
- if (livePhotoRawUploadData != null) {
- req.files.add(livePhotoRawUploadData);
- }
- }
-
setCurrentUploadAssetCb(
CurrentUploadAsset(
id: entity.id,
@@ -296,6 +292,29 @@ class BackupService {
var response =
await httpClient.send(req, cancellationToken: cancelToken);
+ // Send live photo separately
+ if (entity.isLivePhoto) {
+ var livePhotoRawUploadData = await _getLivePhotoFile(entity);
+ if (livePhotoRawUploadData != null) {
+ var livePhotoReq = MultipartRequest(
+ req.method,
+ req.url,
+ onProgress: req.onProgress,
+ )
+ ..headers.addAll(req.headers)
+ ..fields.addAll(req.fields);
+
+ livePhotoReq.files.add(livePhotoRawUploadData);
+ // Send live photo only if the non-motion part is successful
+ if (response.statusCode == 200 || response.statusCode == 201) {
+ response = await httpClient.send(
+ livePhotoReq,
+ cancellationToken: cancelToken,
+ );
+ }
+ }
+ }
+
if (response.statusCode == 200) {
// asset is a duplicate (already exists on the server)
duplicatedAssetIds.add(entity.id);
@@ -353,7 +372,7 @@ class BackupService {
var fileStream = motionFile.openRead();
String fileName = p.basename(motionFile.path);
return http.MultipartFile(
- "livePhotoData",
+ "assetData",
fileStream,
motionFile.lengthSync(),
filename: fileName,
diff --git a/mobile/lib/modules/backup/ui/album_info_card.dart b/mobile/lib/modules/backup/ui/album_info_card.dart
index eaace503e..3e579a84c 100644
--- a/mobile/lib/modules/backup/ui/album_info_card.dart
+++ b/mobile/lib/modules/backup/ui/album_info_card.dart
@@ -1,9 +1,9 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:fluttertoast/fluttertoast.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/backup/models/available_album.model.dart';
import 'package:immich_mobile/modules/backup/providers/backup.provider.dart';
import 'package:immich_mobile/routing/router.dart';
@@ -22,10 +22,10 @@ class AlbumInfoCard extends HookConsumerWidget {
ref.watch(backupProvider).selectedBackupAlbums.contains(albumInfo);
final bool isExcluded =
ref.watch(backupProvider).excludedBackupAlbums.contains(albumInfo);
- final isDarkTheme = Theme.of(context).brightness == Brightness.dark;
+ final isDarkTheme = context.isDarkTheme;
ColorFilter selectedFilter = ColorFilter.mode(
- Theme.of(context).primaryColor.withAlpha(100),
+ context.primaryColor.withAlpha(100),
BlendMode.darken,
);
ColorFilter excludedFilter =
@@ -46,7 +46,7 @@ class AlbumInfoCard extends HookConsumerWidget {
fontWeight: FontWeight.bold,
),
).tr(),
- backgroundColor: Theme.of(context).primaryColor,
+ backgroundColor: context.primaryColor,
);
} else if (isExcluded) {
return Chip(
@@ -82,19 +82,9 @@ class AlbumInfoCard extends HookConsumerWidget {
HapticFeedback.selectionClick();
if (isSelected) {
- if (ref.watch(backupProvider).selectedBackupAlbums.length == 1) {
- ImmichToast.show(
- context: context,
- msg: "backup_err_only_album".tr(),
- toastType: ToastType.error,
- gravity: ToastGravity.BOTTOM,
- );
- return;
- }
-
- ref.watch(backupProvider.notifier).removeAlbumForBackup(albumInfo);
+ ref.read(backupProvider.notifier).removeAlbumForBackup(albumInfo);
} else {
- ref.watch(backupProvider.notifier).addAlbumForBackup(albumInfo);
+ ref.read(backupProvider.notifier).addAlbumForBackup(albumInfo);
}
},
onDoubleTap: () {
@@ -103,23 +93,10 @@ class AlbumInfoCard extends HookConsumerWidget {
if (isExcluded) {
// Remove from exclude album list
ref
- .watch(backupProvider.notifier)
+ .read(backupProvider.notifier)
.removeExcludedAlbumForBackup(albumInfo);
} else {
// Add to exclude album list
- if (ref.watch(backupProvider).selectedBackupAlbums.length == 1 &&
- ref
- .watch(backupProvider)
- .selectedBackupAlbums
- .contains(albumInfo)) {
- ImmichToast.show(
- context: context,
- msg: "backup_err_only_album".tr(),
- toastType: ToastType.error,
- gravity: ToastGravity.BOTTOM,
- );
- return;
- }
if (albumInfo.id == 'isAll' || albumInfo.name == 'Recents') {
ImmichToast.show(
@@ -132,7 +109,7 @@ class AlbumInfoCard extends HookConsumerWidget {
}
ref
- .watch(backupProvider.notifier)
+ .read(backupProvider.notifier)
.addExcludedAlbumForBackup(albumInfo);
}
},
@@ -194,7 +171,7 @@ class AlbumInfoCard extends HookConsumerWidget {
albumInfo.name,
style: TextStyle(
fontSize: 14,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
fontWeight: FontWeight.bold,
),
),
@@ -224,13 +201,13 @@ class AlbumInfoCard extends HookConsumerWidget {
),
IconButton(
onPressed: () {
- AutoRouter.of(context).push(
+ context.autoPush(
AlbumPreviewRoute(album: albumInfo.albumEntity),
);
},
icon: Icon(
Icons.image_outlined,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
size: 24,
),
splashRadius: 25,
diff --git a/mobile/lib/modules/backup/ui/album_info_list_tile.dart b/mobile/lib/modules/backup/ui/album_info_list_tile.dart
index 88141a015..0c27ca1ba 100644
--- a/mobile/lib/modules/backup/ui/album_info_list_tile.dart
+++ b/mobile/lib/modules/backup/ui/album_info_list_tile.dart
@@ -1,10 +1,9 @@
-import 'package:auto_route/auto_route.dart';
-import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:fluttertoast/fluttertoast.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/backup/models/available_album.model.dart';
import 'package:immich_mobile/modules/backup/providers/backup.provider.dart';
import 'package:immich_mobile/routing/router.dart';
@@ -25,14 +24,13 @@ class AlbumInfoListTile extends HookConsumerWidget {
ref.watch(backupProvider).excludedBackupAlbums.contains(albumInfo);
ColorFilter selectedFilter = ColorFilter.mode(
- Theme.of(context).primaryColor.withAlpha(100),
+ context.primaryColor.withAlpha(100),
BlendMode.darken,
);
ColorFilter excludedFilter =
ColorFilter.mode(Colors.red.withAlpha(75), BlendMode.darken);
ColorFilter unselectedFilter =
const ColorFilter.mode(Colors.black, BlendMode.color);
- var isDarkTheme = Theme.of(context).brightness == Brightness.dark;
var assetCount = useState(0);
@@ -56,11 +54,11 @@ class AlbumInfoListTile extends HookConsumerWidget {
buildTileColor() {
if (isSelected) {
- return isDarkTheme
- ? Theme.of(context).primaryColor.withAlpha(100)
- : Theme.of(context).primaryColor.withAlpha(25);
+ return context.isDarkTheme
+ ? context.primaryColor.withAlpha(100)
+ : context.primaryColor.withAlpha(25);
} else if (isExcluded) {
- return isDarkTheme
+ return context.isDarkTheme
? Colors.red[300]?.withAlpha(150)
: Colors.red[100]?.withAlpha(150);
} else {
@@ -75,23 +73,10 @@ class AlbumInfoListTile extends HookConsumerWidget {
if (isExcluded) {
// Remove from exclude album list
ref
- .watch(backupProvider.notifier)
+ .read(backupProvider.notifier)
.removeExcludedAlbumForBackup(albumInfo);
} else {
// Add to exclude album list
- if (ref.watch(backupProvider).selectedBackupAlbums.length == 1 &&
- ref
- .watch(backupProvider)
- .selectedBackupAlbums
- .contains(albumInfo)) {
- ImmichToast.show(
- context: context,
- msg: "backup_err_only_album".tr(),
- toastType: ToastType.error,
- gravity: ToastGravity.BOTTOM,
- );
- return;
- }
if (albumInfo.id == 'isAll' || albumInfo.name == 'Recents') {
ImmichToast.show(
@@ -104,7 +89,7 @@ class AlbumInfoListTile extends HookConsumerWidget {
}
ref
- .watch(backupProvider.notifier)
+ .read(backupProvider.notifier)
.addExcludedAlbumForBackup(albumInfo);
}
},
@@ -114,19 +99,9 @@ class AlbumInfoListTile extends HookConsumerWidget {
onTap: () {
HapticFeedback.selectionClick();
if (isSelected) {
- if (ref.watch(backupProvider).selectedBackupAlbums.length == 1) {
- ImmichToast.show(
- context: context,
- msg: "backup_err_only_album".tr(),
- toastType: ToastType.error,
- gravity: ToastGravity.BOTTOM,
- );
- return;
- }
-
- ref.watch(backupProvider.notifier).removeAlbumForBackup(albumInfo);
+ ref.read(backupProvider.notifier).removeAlbumForBackup(albumInfo);
} else {
- ref.watch(backupProvider.notifier).addAlbumForBackup(albumInfo);
+ ref.read(backupProvider.notifier).addAlbumForBackup(albumInfo);
}
},
leading: ClipRRect(
@@ -159,13 +134,13 @@ class AlbumInfoListTile extends HookConsumerWidget {
subtitle: Text(assetCount.value.toString()),
trailing: IconButton(
onPressed: () {
- AutoRouter.of(context).push(
+ context.autoPush(
AlbumPreviewRoute(album: albumInfo.albumEntity),
);
},
icon: Icon(
Icons.image_outlined,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
size: 24,
),
splashRadius: 25,
diff --git a/mobile/lib/modules/backup/ui/backup_info_card.dart b/mobile/lib/modules/backup/ui/backup_info_card.dart
index bf52c79e6..4feccd19b 100644
--- a/mobile/lib/modules/backup/ui/backup_info_card.dart
+++ b/mobile/lib/modules/backup/ui/backup_info_card.dart
@@ -1,5 +1,6 @@
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
class BackupInfoCard extends StatelessWidget {
final String title;
@@ -14,13 +15,11 @@ class BackupInfoCard extends StatelessWidget {
@override
Widget build(BuildContext context) {
- var isDarkMode = Theme.of(context).brightness == Brightness.dark;
-
return Card(
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(20), // if you need this
side: BorderSide(
- color: isDarkMode
+ color: context.isDarkTheme
? const Color.fromARGB(255, 56, 56, 56)
: Colors.black12,
width: 1,
@@ -29,17 +28,17 @@ class BackupInfoCard extends StatelessWidget {
elevation: 0,
borderOnForeground: false,
child: ListTile(
- minVerticalPadding: 15,
+ minVerticalPadding: 18,
isThreeLine: true,
title: Text(
title,
- style: const TextStyle(fontWeight: FontWeight.bold, fontSize: 20),
+ style: context.textTheme.titleMedium,
),
subtitle: Padding(
- padding: const EdgeInsets.only(top: 8.0),
+ padding: const EdgeInsets.only(top: 4.0, right: 18.0),
child: Text(
subtitle,
- style: const TextStyle(fontSize: 12),
+ style: context.textTheme.bodyMedium,
),
),
trailing: Column(
@@ -47,9 +46,12 @@ class BackupInfoCard extends StatelessWidget {
children: [
Text(
info,
- style: const TextStyle(fontSize: 24, fontWeight: FontWeight.bold),
+ style: context.textTheme.titleLarge,
),
- const Text("backup_info_card_assets").tr(),
+ Text(
+ "backup_info_card_assets",
+ style: context.textTheme.labelLarge,
+ ).tr(),
],
),
),
diff --git a/mobile/lib/modules/backup/ui/current_backup_asset_info_box.dart b/mobile/lib/modules/backup/ui/current_backup_asset_info_box.dart
index 23bdf11ed..926ccd25b 100644
--- a/mobile/lib/modules/backup/ui/current_backup_asset_info_box.dart
+++ b/mobile/lib/modules/backup/ui/current_backup_asset_info_box.dart
@@ -1,9 +1,9 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/backup/models/backup_state.model.dart';
import 'package:immich_mobile/modules/backup/providers/backup.provider.dart';
import 'package:immich_mobile/modules/backup/providers/error_backup_list.provider.dart';
@@ -53,7 +53,7 @@ class CurrentUploadingAssetInfoBox extends HookConsumerWidget {
),
backgroundColor: Colors.white,
onPressed: () {
- AutoRouter.of(context).push(const FailedBackupStatusRoute());
+ context.autoPush(const FailedBackupStatusRoute());
},
);
}
@@ -61,7 +61,7 @@ class CurrentUploadingAssetInfoBox extends HookConsumerWidget {
Widget buildAssetInfoTable() {
return Table(
border: TableBorder.all(
- color: Theme.of(context).primaryColorLight,
+ color: context.themeData.primaryColorLight,
width: 1,
),
children: [
@@ -176,7 +176,7 @@ class CurrentUploadingAssetInfoBox extends HookConsumerWidget {
onTap: () => isShowThumbnail.value = true,
child: Icon(
Icons.image_outlined,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
size: 30,
),
),
@@ -188,9 +188,9 @@ class CurrentUploadingAssetInfoBox extends HookConsumerWidget {
title: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
- const Text(
+ Text(
"backup_controller_page_uploading_file_info",
- style: TextStyle(fontWeight: FontWeight.bold, fontSize: 14),
+ style: context.textTheme.titleSmall,
).tr(),
if (ref.watch(errorBackupListProvider).isNotEmpty) buildErrorChip(),
],
@@ -206,7 +206,7 @@ class CurrentUploadingAssetInfoBox extends HookConsumerWidget {
minHeight: 10.0,
value: uploadProgress / 100.0,
backgroundColor: Colors.grey,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
),
Text(
diff --git a/mobile/lib/modules/backup/ui/ios_debug_info_tile.dart b/mobile/lib/modules/backup/ui/ios_debug_info_tile.dart
index 9ab92104e..283c329b6 100644
--- a/mobile/lib/modules/backup/ui/ios_debug_info_tile.dart
+++ b/mobile/lib/modules/backup/ui/ios_debug_info_tile.dart
@@ -1,5 +1,6 @@
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/backup/providers/ios_background_settings.provider.dart';
import 'package:intl/intl.dart';
@@ -43,7 +44,7 @@ class IosDebugInfoTile extends HookConsumerWidget {
style: TextStyle(
fontWeight: FontWeight.bold,
fontSize: 14,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
),
subtitle: Text(
@@ -54,7 +55,7 @@ class IosDebugInfoTile extends HookConsumerWidget {
),
leading: Icon(
Icons.bug_report,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
);
}
diff --git a/mobile/lib/modules/backup/views/album_preview_page.dart b/mobile/lib/modules/backup/views/album_preview_page.dart
index 27ca79082..cdb0204ec 100644
--- a/mobile/lib/modules/backup/views/album_preview_page.dart
+++ b/mobile/lib/modules/backup/views/album_preview_page.dart
@@ -1,9 +1,9 @@
import 'dart:typed_data';
-import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
import 'package:photo_manager/photo_manager.dart';
@@ -53,7 +53,7 @@ class AlbumPreviewPage extends HookConsumerWidget {
],
),
leading: IconButton(
- onPressed: () => AutoRouter.of(context).pop(),
+ onPressed: () => context.autoPop(),
icon: const Icon(Icons.arrow_back_ios_new_rounded),
),
),
diff --git a/mobile/lib/modules/backup/views/backup_album_selection_page.dart b/mobile/lib/modules/backup/views/backup_album_selection_page.dart
index 9e4118038..96ed879ac 100644
--- a/mobile/lib/modules/backup/views/backup_album_selection_page.dart
+++ b/mobile/lib/modules/backup/views/backup_album_selection_page.dart
@@ -1,15 +1,13 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
-import 'package:fluttertoast/fluttertoast.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/constants/immich_colors.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/backup/providers/backup.provider.dart';
import 'package:immich_mobile/modules/backup/ui/album_info_card.dart';
import 'package:immich_mobile/modules/backup/ui/album_info_list_tile.dart';
import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
-import 'package:immich_mobile/shared/ui/immich_toast.dart';
class BackupAlbumSelectionPage extends HookConsumerWidget {
const BackupAlbumSelectionPage({Key? key}) : super(key: key);
@@ -18,7 +16,7 @@ class BackupAlbumSelectionPage extends HookConsumerWidget {
// final availableAlbums = ref.watch(backupProvider).availableAlbums;
final selectedBackupAlbums = ref.watch(backupProvider).selectedBackupAlbums;
final excludedBackupAlbums = ref.watch(backupProvider).excludedBackupAlbums;
- final isDarkTheme = Theme.of(context).brightness == Brightness.dark;
+ final isDarkTheme = context.isDarkTheme;
final allAlbums = ref.watch(backupProvider).availableAlbums;
// Albums which are displayed to the user
@@ -91,19 +89,8 @@ class BackupAlbumSelectionPage extends HookConsumerWidget {
buildSelectedAlbumNameChip() {
return selectedBackupAlbums.map((album) {
- void removeSelection() {
- if (ref.watch(backupProvider).selectedBackupAlbums.length == 1) {
- ImmichToast.show(
- context: context,
- msg: "backup_err_only_album".tr(),
- toastType: ToastType.error,
- gravity: ToastGravity.BOTTOM,
- );
- return;
- }
-
- ref.watch(backupProvider.notifier).removeAlbumForBackup(album);
- }
+ void removeSelection() =>
+ ref.read(backupProvider.notifier).removeAlbumForBackup(album);
return Padding(
padding: const EdgeInsets.only(right: 8.0),
@@ -113,12 +100,12 @@ class BackupAlbumSelectionPage extends HookConsumerWidget {
label: Text(
album.name,
style: TextStyle(
- fontSize: 10,
+ fontSize: 12,
color: isDarkTheme ? Colors.black : Colors.white,
fontWeight: FontWeight.bold,
),
),
- backgroundColor: Theme.of(context).primaryColor,
+ backgroundColor: context.primaryColor,
deleteIconColor: isDarkTheme ? Colors.black : Colors.white,
deleteIcon: const Icon(
Icons.cancel_rounded,
@@ -147,7 +134,7 @@ class BackupAlbumSelectionPage extends HookConsumerWidget {
label: Text(
album.name,
style: TextStyle(
- fontSize: 10,
+ fontSize: 12,
color: isDarkTheme ? Colors.black : immichBackgroundColor,
fontWeight: FontWeight.bold,
),
@@ -211,12 +198,11 @@ class BackupAlbumSelectionPage extends HookConsumerWidget {
return Scaffold(
appBar: AppBar(
leading: IconButton(
- onPressed: () => AutoRouter.of(context).pop(),
+ onPressed: () => context.autoPop(),
icon: const Icon(Icons.arrow_back_ios_rounded),
),
title: const Text(
"backup_album_selection_page_select_albums",
- style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold),
).tr(),
elevation: 0,
),
@@ -232,12 +218,9 @@ class BackupAlbumSelectionPage extends HookConsumerWidget {
vertical: 8.0,
horizontal: 16.0,
),
- child: const Text(
+ child: Text(
"backup_album_selection_page_selection_info",
- style: TextStyle(
- fontWeight: FontWeight.bold,
- fontSize: 14,
- ),
+ style: context.textTheme.titleSmall,
).tr(),
),
// Selected Album Chips
@@ -252,47 +235,6 @@ class BackupAlbumSelectionPage extends HookConsumerWidget {
),
),
- Padding(
- padding:
- const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8),
- child: Card(
- margin: const EdgeInsets.all(0),
- shape: RoundedRectangleBorder(
- borderRadius: BorderRadius.circular(10),
- side: BorderSide(
- color: isDarkTheme
- ? const Color.fromARGB(255, 0, 0, 0)
- : const Color.fromARGB(255, 235, 235, 235),
- width: 1,
- ),
- ),
- elevation: 0,
- borderOnForeground: false,
- child: Column(
- children: [
- ListTile(
- visualDensity: VisualDensity.compact,
- title: const Text(
- "backup_album_selection_page_total_assets",
- style: TextStyle(
- fontWeight: FontWeight.bold,
- fontSize: 14,
- ),
- ).tr(),
- trailing: Text(
- ref
- .watch(backupProvider)
- .allUniqueAssets
- .length
- .toString(),
- style: const TextStyle(fontWeight: FontWeight.bold),
- ),
- ),
- ],
- ),
- ),
- ),
-
ListTile(
title: Text(
"backup_album_selection_page_albums_device".tr(
@@ -304,19 +246,14 @@ class BackupAlbumSelectionPage extends HookConsumerWidget {
.toString(),
],
),
- style: const TextStyle(
- fontWeight: FontWeight.bold,
- fontSize: 14,
- ),
+ style: context.textTheme.titleSmall,
),
subtitle: Padding(
padding: const EdgeInsets.symmetric(vertical: 8.0),
child: Text(
"backup_album_selection_page_albums_tap",
- style: TextStyle(
- fontSize: 12,
- color: Theme.of(context).primaryColor,
- fontWeight: FontWeight.bold,
+ style: context.textTheme.labelLarge?.copyWith(
+ color: context.primaryColor,
),
).tr(),
),
@@ -325,7 +262,7 @@ class BackupAlbumSelectionPage extends HookConsumerWidget {
icon: Icon(
Icons.info,
size: 20,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
onPressed: () {
// show the dialog
@@ -342,7 +279,7 @@ class BackupAlbumSelectionPage extends HookConsumerWidget {
style: TextStyle(
fontSize: 16,
fontWeight: FontWeight.bold,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
).tr(),
content: SingleChildScrollView(
diff --git a/mobile/lib/modules/backup/views/backup_controller_page.dart b/mobile/lib/modules/backup/views/backup_controller_page.dart
index 9dc40eced..2bdb3a5dd 100644
--- a/mobile/lib/modules/backup/views/backup_controller_page.dart
+++ b/mobile/lib/modules/backup/views/backup_controller_page.dart
@@ -1,11 +1,12 @@
import 'dart:io';
-import 'package:auto_route/auto_route.dart';
import 'package:connectivity_plus/connectivity_plus.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
+import 'package:immich_mobile/modules/album/providers/album.provider.dart';
import 'package:immich_mobile/modules/backup/background_service/background.service.dart';
import 'package:immich_mobile/modules/backup/providers/error_backup_list.provider.dart';
import 'package:immich_mobile/modules/backup/providers/ios_background_settings.provider.dart';
@@ -38,6 +39,7 @@ class BackupControllerPage extends HookConsumerWidget {
final settingsService = ref.watch(appSettingsServiceProvider);
final showBackupFix = Platform.isAndroid &&
settingsService.getSetting(AppSettingsEnum.advancedTroubleshooting);
+ final hasAnyAlbum = backupState.selectedBackupAlbums.isNotEmpty;
final appRefreshDisabled =
Platform.isIOS && settings?.appRefreshEnabled != true;
@@ -49,7 +51,6 @@ class BackupControllerPage extends HookConsumerWidget {
!hasExclusiveAccess
? false
: true;
- var isDarkMode = Theme.of(context).brightness == Brightness.dark;
final checkInProgress = useState(false);
useEffect(
@@ -151,7 +152,7 @@ class BackupControllerPage extends HookConsumerWidget {
return ListTile(
leading: Icon(
Icons.warning_rounded,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
title: const Text(
"Check for corrupt asset backups",
@@ -187,12 +188,12 @@ class BackupControllerPage extends HookConsumerWidget {
leading: isAutoBackup
? Icon(
Icons.cloud_done_rounded,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
)
: const Icon(Icons.cloud_off_rounded),
title: Text(
backUpOption,
- style: const TextStyle(fontWeight: FontWeight.bold, fontSize: 14),
+ style: context.textTheme.titleSmall,
),
subtitle: Padding(
padding: const EdgeInsets.symmetric(vertical: 8.0),
@@ -212,9 +213,8 @@ class BackupControllerPage extends HookConsumerWidget {
.setAutoBackup(!isAutoBackup),
child: Text(
backupBtnText,
- style: const TextStyle(
- fontWeight: FontWeight.bold,
- fontSize: 12,
+ style: context.textTheme.labelLarge?.copyWith(
+ color: context.isDarkTheme ? Colors.black : Colors.white,
),
),
),
@@ -229,6 +229,9 @@ class BackupControllerPage extends HookConsumerWidget {
final snackBar = SnackBar(
content: Text(
msg.tr(),
+ style: context.textTheme.bodyLarge?.copyWith(
+ color: context.primaryColor,
+ ),
),
backgroundColor: Colors.red,
);
@@ -266,7 +269,7 @@ class BackupControllerPage extends HookConsumerWidget {
style: TextStyle(fontWeight: FontWeight.bold, fontSize: 12),
).tr(),
onPressed: () {
- Navigator.of(context).pop();
+ context.pop();
},
),
],
@@ -279,7 +282,7 @@ class BackupControllerPage extends HookConsumerWidget {
final bool isBackgroundEnabled = backupState.backgroundBackup;
final bool isWifiRequired = backupState.backupRequireWifi;
final bool isChargingRequired = backupState.backupRequireCharging;
- final Color activeColor = Theme.of(context).primaryColor;
+ final Color activeColor = context.primaryColor;
String formatBackupDelaySliderValue(double v) {
if (v == 0.0) {
@@ -334,7 +337,7 @@ class BackupControllerPage extends HookConsumerWidget {
isBackgroundEnabled
? "backup_controller_page_background_is_on"
: "backup_controller_page_background_is_off",
- style: const TextStyle(fontWeight: FontWeight.bold, fontSize: 14),
+ style: context.textTheme.titleSmall,
).tr(),
subtitle: Column(
crossAxisAlignment: CrossAxisAlignment.start,
@@ -410,7 +413,7 @@ class BackupControllerPage extends HookConsumerWidget {
max: 3.0,
divisions: 3,
label: formatBackupDelaySliderValue(triggerDelay.value),
- activeColor: Theme.of(context).primaryColor,
+ activeColor: context.primaryColor,
),
),
ElevatedButton(
@@ -425,9 +428,8 @@ class BackupControllerPage extends HookConsumerWidget {
isBackgroundEnabled
? "backup_controller_page_background_turn_off"
: "backup_controller_page_background_turn_on",
- style: const TextStyle(
- fontWeight: FontWeight.bold,
- fontSize: 12,
+ style: context.textTheme.labelLarge?.copyWith(
+ color: context.isDarkTheme ? Colors.black : Colors.white,
),
).tr(),
),
@@ -510,10 +512,8 @@ class BackupControllerPage extends HookConsumerWidget {
padding: const EdgeInsets.only(top: 8.0),
child: Text(
text.trim().substring(0, text.length - 2),
- style: TextStyle(
- color: Theme.of(context).primaryColor,
- fontSize: 12,
- fontWeight: FontWeight.bold,
+ style: context.textTheme.labelLarge?.copyWith(
+ color: context.primaryColor,
),
),
);
@@ -522,10 +522,8 @@ class BackupControllerPage extends HookConsumerWidget {
padding: const EdgeInsets.only(top: 8.0),
child: Text(
"backup_controller_page_none_selected".tr(),
- style: TextStyle(
- color: Theme.of(context).primaryColor,
- fontSize: 12,
- fontWeight: FontWeight.bold,
+ style: context.textTheme.labelLarge?.copyWith(
+ color: context.primaryColor,
),
),
);
@@ -545,10 +543,8 @@ class BackupControllerPage extends HookConsumerWidget {
padding: const EdgeInsets.only(top: 8.0),
child: Text(
text.trim().substring(0, text.length - 2),
- style: TextStyle(
+ style: context.textTheme.labelLarge?.copyWith(
color: Colors.red[300],
- fontSize: 12,
- fontWeight: FontWeight.bold,
),
),
);
@@ -558,49 +554,57 @@ class BackupControllerPage extends HookConsumerWidget {
}
buildFolderSelectionTile() {
- return Card(
- shape: RoundedRectangleBorder(
- borderRadius: BorderRadius.circular(20),
- side: BorderSide(
- color: isDarkMode
- ? const Color.fromARGB(255, 56, 56, 56)
- : Colors.black12,
- width: 1,
- ),
- ),
- elevation: 0,
- borderOnForeground: false,
- child: ListTile(
- minVerticalPadding: 15,
- title: const Text(
- "backup_controller_page_albums",
- style: TextStyle(fontWeight: FontWeight.bold, fontSize: 20),
- ).tr(),
- subtitle: Padding(
- padding: const EdgeInsets.only(top: 8.0),
- child: Column(
- crossAxisAlignment: CrossAxisAlignment.start,
- children: [
- const Text(
- "backup_controller_page_to_backup",
- style: TextStyle(fontSize: 12),
- ).tr(),
- buildSelectedAlbumName(),
- buildExcludedAlbumName(),
- ],
+ return Padding(
+ padding: const EdgeInsets.only(top: 8.0),
+ child: Card(
+ shape: RoundedRectangleBorder(
+ borderRadius: BorderRadius.circular(20),
+ side: BorderSide(
+ color: context.isDarkTheme
+ ? const Color.fromARGB(255, 56, 56, 56)
+ : Colors.black12,
+ width: 1,
),
),
- trailing: ElevatedButton(
- onPressed: () {
- AutoRouter.of(context).push(const BackupAlbumSelectionRoute());
- },
- child: const Text(
- "backup_controller_page_select",
- style: TextStyle(
- fontWeight: FontWeight.bold,
- fontSize: 12,
- ),
+ elevation: 0,
+ borderOnForeground: false,
+ child: ListTile(
+ minVerticalPadding: 18,
+ title: Text(
+ "backup_controller_page_albums",
+ style: context.textTheme.titleMedium,
).tr(),
+ subtitle: Padding(
+ padding: const EdgeInsets.only(top: 8.0),
+ child: Column(
+ crossAxisAlignment: CrossAxisAlignment.start,
+ children: [
+ Text(
+ "backup_controller_page_to_backup",
+ style: context.textTheme.bodyMedium,
+ ).tr(),
+ buildSelectedAlbumName(),
+ buildExcludedAlbumName(),
+ ],
+ ),
+ ),
+ trailing: ElevatedButton(
+ onPressed: () async {
+ await context.autoPush(const BackupAlbumSelectionRoute());
+ // waited until returning from selection
+ await ref
+ .read(backupProvider.notifier)
+ .backupAlbumSelectionDone();
+ // waited until backup albums are stored in DB
+ ref.read(albumProvider.notifier).getDeviceAlbums();
+ },
+ child: const Text(
+ "backup_controller_page_select",
+ style: TextStyle(
+ fontWeight: FontWeight.bold,
+ ),
+ ).tr(),
+ ),
),
),
);
@@ -650,7 +654,7 @@ class BackupControllerPage extends HookConsumerWidget {
child: const Text(
"backup_controller_page_start_backup",
style: TextStyle(
- fontSize: 14,
+ fontSize: 16,
fontWeight: FontWeight.bold,
),
).tr(),
@@ -673,12 +677,11 @@ class BackupControllerPage extends HookConsumerWidget {
elevation: 0,
title: const Text(
"backup_controller_page_backup",
- style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold),
).tr(),
leading: IconButton(
onPressed: () {
ref.watch(websocketProvider.notifier).listenUploadEvent();
- AutoRouter.of(context).pop(true);
+ context.autoPop(true);
},
splashRadius: 24,
icon: const Icon(
@@ -690,55 +693,49 @@ class BackupControllerPage extends HookConsumerWidget {
padding: const EdgeInsets.only(left: 16.0, right: 16, bottom: 32),
child: ListView(
// crossAxisAlignment: CrossAxisAlignment.start,
- children: [
- Padding(
- padding: const EdgeInsets.all(8.0),
- child: const Text(
- "backup_controller_page_info",
- style: TextStyle(fontWeight: FontWeight.bold, fontSize: 16),
- ).tr(),
- ),
- buildFolderSelectionTile(),
- BackupInfoCard(
- title: "backup_controller_page_total".tr(),
- subtitle: "backup_controller_page_total_sub".tr(),
- info: ref.watch(backupProvider).availableAlbums.isEmpty
- ? "..."
- : "${backupState.allUniqueAssets.length}",
- ),
- BackupInfoCard(
- title: "backup_controller_page_backup".tr(),
- subtitle: "backup_controller_page_backup_sub".tr(),
- info: ref.watch(backupProvider).availableAlbums.isEmpty
- ? "..."
- : "${backupState.selectedAlbumsBackupAssetsIds.length}",
- ),
- BackupInfoCard(
- title: "backup_controller_page_remainder".tr(),
- subtitle: "backup_controller_page_remainder_sub".tr(),
- info: ref.watch(backupProvider).availableAlbums.isEmpty
- ? "..."
- : "${backupState.allUniqueAssets.length - backupState.selectedAlbumsBackupAssetsIds.length}",
- ),
- const Divider(),
- buildAutoBackupController(),
- const Divider(),
- AnimatedSwitcher(
- duration: const Duration(milliseconds: 500),
- child: Platform.isIOS
- ? (appRefreshDisabled
- ? buildBackgroundAppRefreshWarning()
- : buildBackgroundBackupController())
- : buildBackgroundBackupController(),
- ),
- if (showBackupFix) const Divider(),
- if (showBackupFix) buildCheckCorruptBackups(),
- const Divider(),
- const Divider(),
- const CurrentUploadingAssetInfoBox(),
- if (!hasExclusiveAccess) buildBackgroundBackupInfo(),
- buildBackupButton(),
- ],
+ children: hasAnyAlbum
+ ? [
+ buildFolderSelectionTile(),
+ BackupInfoCard(
+ title: "backup_controller_page_total".tr(),
+ subtitle: "backup_controller_page_total_sub".tr(),
+ info: ref.watch(backupProvider).availableAlbums.isEmpty
+ ? "..."
+ : "${backupState.allUniqueAssets.length}",
+ ),
+ BackupInfoCard(
+ title: "backup_controller_page_backup".tr(),
+ subtitle: "backup_controller_page_backup_sub".tr(),
+ info: ref.watch(backupProvider).availableAlbums.isEmpty
+ ? "..."
+ : "${backupState.selectedAlbumsBackupAssetsIds.length}",
+ ),
+ BackupInfoCard(
+ title: "backup_controller_page_remainder".tr(),
+ subtitle: "backup_controller_page_remainder_sub".tr(),
+ info: ref.watch(backupProvider).availableAlbums.isEmpty
+ ? "..."
+ : "${backupState.allUniqueAssets.length - backupState.selectedAlbumsBackupAssetsIds.length}",
+ ),
+ const Divider(),
+ buildAutoBackupController(),
+ const Divider(),
+ AnimatedSwitcher(
+ duration: const Duration(milliseconds: 500),
+ child: Platform.isIOS
+ ? (appRefreshDisabled
+ ? buildBackgroundAppRefreshWarning()
+ : buildBackgroundBackupController())
+ : buildBackgroundBackupController(),
+ ),
+ if (showBackupFix) const Divider(),
+ if (showBackupFix) buildCheckCorruptBackups(),
+ const Divider(),
+ const CurrentUploadingAssetInfoBox(),
+ if (!hasExclusiveAccess) buildBackgroundBackupInfo(),
+ buildBackupButton(),
+ ]
+ : [buildFolderSelectionTile()],
),
),
);
diff --git a/mobile/lib/modules/backup/views/failed_backup_status_page.dart b/mobile/lib/modules/backup/views/failed_backup_status_page.dart
index c55383cf3..433ed3420 100644
--- a/mobile/lib/modules/backup/views/failed_backup_status_page.dart
+++ b/mobile/lib/modules/backup/views/failed_backup_status_page.dart
@@ -1,6 +1,6 @@
-import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/backup/providers/error_backup_list.provider.dart';
import 'package:intl/intl.dart';
import 'package:photo_manager/photo_manager.dart';
@@ -20,7 +20,7 @@ class FailedBackupStatusPage extends HookConsumerWidget {
),
leading: IconButton(
onPressed: () {
- AutoRouter.of(context).pop(true);
+ context.autoPop(true);
},
splashRadius: 24,
icon: const Icon(
@@ -114,7 +114,7 @@ class FailedBackupStatusPage extends HookConsumerWidget {
style: TextStyle(
fontWeight: FontWeight.bold,
fontSize: 12,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
),
),
diff --git a/mobile/lib/modules/favorite/providers/favorite_provider.dart b/mobile/lib/modules/favorite/providers/favorite_provider.dart
index 427d2c88b..0da6b3f8a 100644
--- a/mobile/lib/modules/favorite/providers/favorite_provider.dart
+++ b/mobile/lib/modules/favorite/providers/favorite_provider.dart
@@ -1,15 +1,14 @@
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/asset_grid_data_structure.dart';
-import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart';
-import 'package:immich_mobile/modules/settings/services/app_settings.service.dart';
import 'package:immich_mobile/shared/models/asset.dart';
import 'package:immich_mobile/shared/providers/db.provider.dart';
import 'package:immich_mobile/shared/providers/user.provider.dart';
+import 'package:immich_mobile/utils/renderlist_generator.dart';
import 'package:isar/isar.dart';
-final favoriteAssetsProvider = StreamProvider((ref) async* {
+final favoriteAssetsProvider = StreamProvider((ref) {
final user = ref.watch(currentUserProvider);
- if (user == null) return;
+ if (user == null) return const Stream.empty();
final query = ref
.watch(dbProvider)
.assets
@@ -19,11 +18,5 @@ final favoriteAssetsProvider = StreamProvider((ref) async* {
.isFavoriteEqualTo(true)
.isTrashedEqualTo(false)
.sortByFileCreatedAt();
- final settings = ref.watch(appSettingsServiceProvider);
- final groupBy =
- GroupAssetsBy.values[settings.getSetting(AppSettingsEnum.groupAssetsBy)];
- yield await RenderList.fromQuery(query, groupBy);
- await for (final _ in query.watchLazy()) {
- yield await RenderList.fromQuery(query, groupBy);
- }
+ return renderListGenerator(query, ref);
});
diff --git a/mobile/lib/modules/favorite/views/favorites_page.dart b/mobile/lib/modules/favorite/views/favorites_page.dart
index 62f8763bb..095297507 100644
--- a/mobile/lib/modules/favorite/views/favorites_page.dart
+++ b/mobile/lib/modules/favorite/views/favorites_page.dart
@@ -1,8 +1,9 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/favorite/providers/favorite_provider.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/immich_asset_grid.dart';
import 'package:immich_mobile/shared/models/asset.dart';
@@ -28,7 +29,7 @@ class FavoritesPage extends HookConsumerWidget {
AppBar buildAppBar() {
return AppBar(
leading: IconButton(
- onPressed: () => AutoRouter.of(context).pop(),
+ onPressed: () => context.autoPop(),
icon: const Icon(Icons.arrow_back_ios_rounded),
),
centerTitle: true,
@@ -62,22 +63,18 @@ class FavoritesPage extends HookConsumerWidget {
child: SizedBox(
height: 64,
child: Card(
- child: Column(
- children: [
- ListTile(
- shape: RoundedRectangleBorder(
- borderRadius: BorderRadius.circular(10),
- ),
- leading: const Icon(
- Icons.star_border,
- ),
- title: const Text(
- "Unfavorite",
- style: TextStyle(fontSize: 14),
- ),
- onTap: processing.value ? null : unfavorite,
- ),
- ],
+ child: ListTile(
+ shape: const RoundedRectangleBorder(
+ borderRadius: BorderRadius.all(Radius.circular(10)),
+ ),
+ leading: const Icon(
+ Icons.star_border,
+ ),
+ title: const Text(
+ "Unfavorite",
+ style: TextStyle(fontSize: 14),
+ ),
+ onTap: processing.value ? null : unfavorite,
),
),
),
@@ -87,10 +84,8 @@ class FavoritesPage extends HookConsumerWidget {
return Scaffold(
appBar: buildAppBar(),
- body: ref.watch(favoriteAssetsProvider).when(
- loading: () => const Center(child: CircularProgressIndicator()),
- error: (error, stackTrace) => Center(child: Text(error.toString())),
- data: (data) => data.isEmpty
+ body: ref.watch(favoriteAssetsProvider).widgetWhen(
+ onData: (data) => data.isEmpty
? Center(
child: Text('favorites_page_no_favorites'.tr()),
)
diff --git a/mobile/lib/modules/home/ui/asset_grid/group_divider_title.dart b/mobile/lib/modules/home/ui/asset_grid/group_divider_title.dart
index d63b0631e..1455f53ef 100644
--- a/mobile/lib/modules/home/ui/asset_grid/group_divider_title.dart
+++ b/mobile/lib/modules/home/ui/asset_grid/group_divider_title.dart
@@ -1,8 +1,13 @@
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
+import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
+import 'package:immich_mobile/modules/home/ui/asset_grid/asset_grid_data_structure.dart';
+import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart';
+import 'package:immich_mobile/modules/settings/services/app_settings.service.dart';
-class GroupDividerTitle extends ConsumerWidget {
+class GroupDividerTitle extends HookConsumerWidget {
const GroupDividerTitle({
Key? key,
required this.text,
@@ -20,6 +25,18 @@ class GroupDividerTitle extends ConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
+ final appSettingService = ref.watch(appSettingsServiceProvider);
+ final groupBy = useState(GroupAssetsBy.day);
+
+ useEffect(
+ () {
+ groupBy.value = GroupAssetsBy.values[
+ appSettingService.getSetting(AppSettingsEnum.groupAssetsBy)];
+ return null;
+ },
+ [],
+ );
+
void handleTitleIconClick() {
HapticFeedback.heavyImpact();
if (selected) {
@@ -30,8 +47,8 @@ class GroupDividerTitle extends ConsumerWidget {
}
return Padding(
- padding: const EdgeInsets.only(
- top: 12.0,
+ padding: EdgeInsets.only(
+ top: groupBy.value == GroupAssetsBy.month ? 32.0 : 16.0,
bottom: 16.0,
left: 12.0,
right: 12.0,
@@ -40,10 +57,14 @@ class GroupDividerTitle extends ConsumerWidget {
children: [
Text(
text,
- style: const TextStyle(
- fontSize: 14,
- fontWeight: FontWeight.bold,
- ),
+ style: groupBy.value == GroupAssetsBy.month
+ ? context.textTheme.bodyLarge?.copyWith(
+ fontSize: 24.0,
+ )
+ : context.textTheme.labelLarge?.copyWith(
+ color: context.textTheme.labelLarge?.color?.withAlpha(250),
+ fontWeight: FontWeight.w500,
+ ),
),
const Spacer(),
GestureDetector(
@@ -51,7 +72,7 @@ class GroupDividerTitle extends ConsumerWidget {
child: multiselectEnabled && selected
? Icon(
Icons.check_circle_rounded,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
)
: const Icon(
Icons.check_circle_outline_rounded,
diff --git a/mobile/lib/modules/home/ui/asset_grid/immich_asset_grid.dart b/mobile/lib/modules/home/ui/asset_grid/immich_asset_grid.dart
index 2c0f63394..562b7892c 100644
--- a/mobile/lib/modules/home/ui/asset_grid/immich_asset_grid.dart
+++ b/mobile/lib/modules/home/ui/asset_grid/immich_asset_grid.dart
@@ -5,13 +5,13 @@ import 'package:flutter/gestures.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
import 'package:immich_mobile/modules/asset_viewer/providers/render_list.provider.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/asset_grid_data_structure.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/immich_asset_grid_view.dart';
import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart';
import 'package:immich_mobile/modules/settings/services/app_settings.service.dart';
import 'package:immich_mobile/shared/models/asset.dart';
-import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
import 'package:scrollable_positioned_list/scrollable_positioned_list.dart';
class ImmichAssetGrid extends HookConsumerWidget {
@@ -130,12 +130,8 @@ class ImmichAssetGrid extends HookConsumerWidget {
if (renderList != null) return buildAssetGridView(renderList!);
final renderListFuture = ref.watch(renderListProvider(assets!));
- return renderListFuture.when(
- data: (renderList) => buildAssetGridView(renderList),
- error: (err, stack) => Center(child: Text("$err")),
- loading: () => const Center(
- child: ImmichLoadingIndicator(),
- ),
+ return renderListFuture.widgetWhen(
+ onData: (renderList) => buildAssetGridView(renderList),
);
}
}
diff --git a/mobile/lib/modules/home/ui/asset_grid/immich_asset_grid_view.dart b/mobile/lib/modules/home/ui/asset_grid/immich_asset_grid_view.dart
index 4424f722e..e27292eaa 100644
--- a/mobile/lib/modules/home/ui/asset_grid/immich_asset_grid_view.dart
+++ b/mobile/lib/modules/home/ui/asset_grid/immich_asset_grid_view.dart
@@ -4,10 +4,11 @@ import 'dart:math';
import 'package:collection/collection.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/asset_viewer/providers/scroll_notifier.provider.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/thumbnail_image.dart';
import 'package:immich_mobile/shared/models/asset.dart';
-import 'package:immich_mobile/utils/builtin_extensions.dart';
+import 'package:immich_mobile/extensions/collection_extensions.dart';
import 'package:scrollable_positioned_list/scrollable_positioned_list.dart';
import 'asset_grid_data_structure.dart';
import 'group_divider_title.dart';
@@ -223,10 +224,9 @@ class ImmichAssetGridViewState extends State {
padding: const EdgeInsets.only(left: 12.0, top: 24.0),
child: Text(
title,
- style: TextStyle(
+ style: const TextStyle(
fontSize: 26,
- fontWeight: FontWeight.bold,
- color: Theme.of(context).textTheme.displayLarge?.color,
+ fontWeight: FontWeight.w500,
),
),
);
@@ -374,7 +374,7 @@ class ImmichAssetGridViewState extends State {
scrollStateListener: dragScrolling,
itemPositionsListener: _itemPositionsListener,
controller: _itemScrollController,
- backgroundColor: Theme.of(context).hintColor,
+ backgroundColor: context.themeData.hintColor,
labelTextBuilder: _labelBuilder,
labelConstraints: const BoxConstraints(maxHeight: 28),
scrollbarAnimationDuration: const Duration(milliseconds: 300),
diff --git a/mobile/lib/modules/home/ui/asset_grid/thumbnail_image.dart b/mobile/lib/modules/home/ui/asset_grid/thumbnail_image.dart
index 16423b3b4..694279c0d 100644
--- a/mobile/lib/modules/home/ui/asset_grid/thumbnail_image.dart
+++ b/mobile/lib/modules/home/ui/asset_grid/thumbnail_image.dart
@@ -1,6 +1,6 @@
-import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/routing/router.dart';
import 'package:immich_mobile/shared/models/asset.dart';
import 'package:immich_mobile/shared/ui/immich_image.dart';
@@ -43,9 +43,9 @@ class ThumbnailImage extends StatelessWidget {
@override
Widget build(BuildContext context) {
- final isDarkTheme = Theme.of(context).brightness == Brightness.dark;
- final assetContainerColor =
- isDarkTheme ? Colors.blueGrey : Theme.of(context).primaryColorLight;
+ final assetContainerColor = context.isDarkTheme
+ ? Colors.blueGrey
+ : context.themeData.primaryColorLight;
// Assets from response DTOs do not have an isar id, querying which would give us the default autoIncrement id
final isFromDto = asset.id == Isar.autoIncrement;
@@ -58,7 +58,7 @@ class ThumbnailImage extends StatelessWidget {
),
child: Icon(
Icons.check_circle_rounded,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
);
} else {
@@ -178,7 +178,7 @@ class ThumbnailImage extends StatelessWidget {
onSelect?.call();
}
} else {
- AutoRouter.of(context).push(
+ context.autoPush(
GalleryViewerRoute(
initialIndex: index,
loadAsset: loadAsset,
@@ -197,7 +197,9 @@ class ThumbnailImage extends StatelessWidget {
},
child: Stack(
children: [
- Container(
+ AnimatedContainer(
+ duration: const Duration(milliseconds: 300),
+ curve: Curves.decelerate,
decoration: BoxDecoration(
border: multiselectEnabled && isSelected
? Border.all(
diff --git a/mobile/lib/modules/home/ui/control_bottom_app_bar.dart b/mobile/lib/modules/home/ui/control_bottom_app_bar.dart
index 4f6e2706d..8ae7f98cd 100644
--- a/mobile/lib/modules/home/ui/control_bottom_app_bar.dart
+++ b/mobile/lib/modules/home/ui/control_bottom_app_bar.dart
@@ -1,6 +1,7 @@
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/ui/add_to_album_sliverlist.dart';
import 'package:immich_mobile/modules/home/models/selection_state.dart';
import 'package:immich_mobile/modules/home/ui/delete_dialog.dart';
@@ -42,7 +43,6 @@ class ControlBottomAppBar extends ConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
- var isDarkMode = Theme.of(context).brightness == Brightness.dark;
var hasRemote =
selectionAssetState.hasRemote || selectionAssetState.hasMerged;
var hasLocal = selectionAssetState.hasLocal;
@@ -128,7 +128,7 @@ class ControlBottomAppBar extends ConsumerWidget {
ScrollController scrollController,
) {
return Card(
- color: isDarkMode ? Colors.grey[900] : Colors.grey[100],
+ color: context.isDarkTheme ? Colors.grey[900] : Colors.grey[100],
surfaceTintColor: Colors.transparent,
elevation: 18.0,
shape: const RoundedRectangleBorder(
@@ -150,6 +150,7 @@ class ControlBottomAppBar extends ConsumerWidget {
SizedBox(
height: 70,
child: ListView(
+ shrinkWrap: true,
scrollDirection: Axis.horizontal,
children: renderActionButtons(),
),
@@ -211,12 +212,12 @@ class AddToAlbumTitleRow extends StatelessWidget {
onPressed: onCreateNewAlbum,
icon: Icon(
Icons.add,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
label: Text(
"common_create_new_album",
style: TextStyle(
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
fontWeight: FontWeight.bold,
fontSize: 14,
),
diff --git a/mobile/lib/modules/home/views/home_page.dart b/mobile/lib/modules/home/views/home_page.dart
index d41022a29..58770ed5c 100644
--- a/mobile/lib/modules/home/views/home_page.dart
+++ b/mobile/lib/modules/home/views/home_page.dart
@@ -1,12 +1,12 @@
import 'dart:async';
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:fluttertoast/fluttertoast.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/album/providers/album.provider.dart';
import 'package:immich_mobile/modules/album/providers/album_detail.provider.dart';
import 'package:immich_mobile/modules/album/providers/shared_album.provider.dart';
@@ -28,6 +28,7 @@ import 'package:immich_mobile/shared/providers/websocket.provider.dart';
import 'package:immich_mobile/shared/ui/immich_app_bar.dart';
import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
import 'package:immich_mobile/shared/ui/immich_toast.dart';
+import 'package:immich_mobile/shared/views/immich_loading_overlay.dart';
import 'package:immich_mobile/utils/selection_handlers.dart';
class HomePage extends HookConsumerWidget {
@@ -44,17 +45,19 @@ class HomePage extends HookConsumerWidget {
final sharedAlbums = ref.watch(sharedAlbumProvider);
final albumService = ref.watch(albumServiceProvider);
final currentUser = ref.watch(currentUserProvider);
+ final timelineUsers = ref.watch(timelineUsersIdsProvider);
final trashEnabled =
ref.watch(serverInfoProvider.select((v) => v.serverFeatures.trash));
final tipOneOpacity = useState(0.0);
final refreshCount = useState(0);
- final processing = useState(false);
+ final processing = useProcessingOverlay();
useEffect(
() {
ref.read(websocketProvider.notifier).connect();
Future(() => ref.read(assetProvider.notifier).getAllAsset());
+ ref.read(assetProvider.notifier).getPartnerAssets();
ref.read(albumProvider.notifier).getAllAlbums();
ref.read(sharedAlbumProvider.notifier).getAllSharedAlbums();
ref.read(serverInfoProvider.notifier).getServerInfo();
@@ -84,30 +87,65 @@ class HomePage extends HookConsumerWidget {
SelectionAssetState.fromSelection(selectedAssets);
}
- List remoteOnlySelection({String? localErrorMessage}) {
- final Set assets = selection.value;
+ errorBuilder(String? msg) => msg != null && msg.isNotEmpty
+ ? () => ImmichToast.show(
+ context: context,
+ msg: msg,
+ gravity: ToastGravity.BOTTOM,
+ )
+ : null;
+
+ Iterable remoteOnly(
+ Iterable assets, {
+ void Function()? errorCallback,
+ }) {
final bool onlyRemote = assets.every((e) => e.isRemote);
if (!onlyRemote) {
- if (localErrorMessage != null && localErrorMessage.isNotEmpty) {
- ImmichToast.show(
- context: context,
- msg: localErrorMessage,
- gravity: ToastGravity.BOTTOM,
- );
- }
- return assets.where((a) => a.isRemote).toList();
+ if (errorCallback != null) errorCallback();
+ return assets.where((a) => a.isRemote);
}
- return assets.toList();
+ return assets;
}
+ Iterable ownedOnly(
+ Iterable assets, {
+ void Function()? errorCallback,
+ }) {
+ if (currentUser == null) return [];
+ final userId = currentUser.isarId;
+ final bool onlyOwned = assets.every((e) => e.ownerId == userId);
+ if (!onlyOwned) {
+ if (errorCallback != null) errorCallback();
+ return assets.where((a) => a.ownerId == userId);
+ }
+ return assets;
+ }
+
+ Iterable ownedRemoteSelection({
+ String? localErrorMessage,
+ String? ownerErrorMessage,
+ }) {
+ final assets = selection.value;
+ return remoteOnly(
+ ownedOnly(assets, errorCallback: errorBuilder(ownerErrorMessage)),
+ errorCallback: errorBuilder(localErrorMessage),
+ );
+ }
+
+ Iterable remoteSelection({String? errorMessage}) => remoteOnly(
+ selection.value,
+ errorCallback: errorBuilder(errorMessage),
+ );
+
void onShareAssets(bool shareLocal) {
processing.value = true;
if (shareLocal) {
handleShareAssets(ref, context, selection.value.toList());
} else {
- final ids = remoteOnlySelection().map((e) => e.remoteId!);
- AutoRouter.of(context)
- .push(SharedLinkEditRoute(assetsList: ids.toList()));
+ final ids =
+ remoteSelection(errorMessage: "home_page_share_err_local".tr())
+ .map((e) => e.remoteId!);
+ context.autoPush(SharedLinkEditRoute(assetsList: ids.toList()));
}
processing.value = false;
selectionEnabledHook.value = false;
@@ -116,11 +154,12 @@ class HomePage extends HookConsumerWidget {
void onFavoriteAssets() async {
processing.value = true;
try {
- final remoteAssets = remoteOnlySelection(
+ final remoteAssets = ownedRemoteSelection(
localErrorMessage: 'home_page_favorite_err_local'.tr(),
+ ownerErrorMessage: 'home_page_favorite_err_partner'.tr(),
);
if (remoteAssets.isNotEmpty) {
- await handleFavoriteAssets(ref, context, remoteAssets);
+ await handleFavoriteAssets(ref, context, remoteAssets.toList());
}
} finally {
processing.value = false;
@@ -131,10 +170,11 @@ class HomePage extends HookConsumerWidget {
void onArchiveAsset() async {
processing.value = true;
try {
- final remoteAssets = remoteOnlySelection(
+ final remoteAssets = ownedRemoteSelection(
localErrorMessage: 'home_page_archive_err_local'.tr(),
+ ownerErrorMessage: 'home_page_archive_err_partner'.tr(),
);
- await handleArchiveAssets(ref, context, remoteAssets);
+ await handleArchiveAssets(ref, context, remoteAssets.toList());
} finally {
processing.value = false;
selectionEnabledHook.value = false;
@@ -144,12 +184,16 @@ class HomePage extends HookConsumerWidget {
void onDelete() async {
processing.value = true;
try {
+ final toDelete = ownedOnly(
+ selection.value,
+ errorCallback: errorBuilder('home_page_delete_err_partner'.tr()),
+ ).toList();
await ref
.read(assetProvider.notifier)
- .deleteAssets(selection.value, force: !trashEnabled);
+ .deleteAssets(toDelete, force: !trashEnabled);
- final hasRemote = selection.value.any((a) => a.isRemote);
- final assetOrAssets = selection.value.length > 1 ? 'assets' : 'asset';
+ final hasRemote = toDelete.any((a) => a.isRemote);
+ final assetOrAssets = toDelete.length > 1 ? 'assets' : 'asset';
final trashOrRemoved =
!trashEnabled ? 'deleted permanently' : 'trashed';
if (hasRemote) {
@@ -169,10 +213,10 @@ class HomePage extends HookConsumerWidget {
processing.value = true;
selectionEnabledHook.value = false;
try {
- ref.read(manualUploadProvider.notifier).uploadAssets(
- context,
- selection.value.where((a) => a.storage == AssetState.local),
- );
+ ref.read(manualUploadProvider.notifier).uploadAssets(
+ context,
+ selection.value.where((a) => a.storage == AssetState.local),
+ );
} finally {
processing.value = false;
}
@@ -181,8 +225,8 @@ class HomePage extends HookConsumerWidget {
void onAddToAlbum(Album album) async {
processing.value = true;
try {
- final Iterable assets = remoteOnlySelection(
- localErrorMessage: "home_page_add_to_album_err_local".tr(),
+ final Iterable assets = remoteSelection(
+ errorMessage: "home_page_add_to_album_err_local".tr(),
);
if (assets.isEmpty) {
return;
@@ -229,8 +273,8 @@ class HomePage extends HookConsumerWidget {
void onCreateNewAlbum() async {
processing.value = true;
try {
- final Iterable assets = remoteOnlySelection(
- localErrorMessage: "home_page_add_to_album_err_local".tr(),
+ final Iterable assets = remoteSelection(
+ errorMessage: "home_page_add_to_album_err_local".tr(),
);
if (assets.isEmpty) {
return;
@@ -243,7 +287,7 @@ class HomePage extends HookConsumerWidget {
ref.watch(sharedAlbumProvider.notifier).getAllSharedAlbums();
selectionEnabledHook.value = false;
- AutoRouter.of(context).push(AlbumViewerRoute(albumId: result.id));
+ context.autoPush(AlbumViewerRoute(albumId: result.id));
}
} finally {
processing.value = false;
@@ -271,22 +315,21 @@ class HomePage extends HookConsumerWidget {
Future refreshAssets() async {
final fullRefresh = refreshCount.value > 0;
await ref.read(assetProvider.notifier).getAllAsset(clear: fullRefresh);
+ if (timelineUsers.length > 1) {
+ await ref.read(assetProvider.notifier).getPartnerAssets();
+ }
if (fullRefresh) {
// refresh was forced: user requested another refresh within 2 seconds
refreshCount.value = 0;
} else {
refreshCount.value++;
// set counter back to 0 if user does not request refresh again
- Timer(const Duration(seconds: 4), () {
- refreshCount.value = 0;
- });
+ Timer(const Duration(seconds: 4), () => refreshCount.value = 0);
}
}
buildLoadingIndicator() {
- Timer(const Duration(seconds: 2), () {
- tipOneOpacity.value = 1;
- });
+ Timer(const Duration(seconds: 2), () => tipOneOpacity.value = 1);
return Center(
child: Column(
@@ -300,7 +343,7 @@ class HomePage extends HookConsumerWidget {
style: TextStyle(
fontWeight: FontWeight.w600,
fontSize: 16,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
).tr(),
),
@@ -331,7 +374,13 @@ class HomePage extends HookConsumerWidget {
bottom: false,
child: Stack(
children: [
- ref.watch(assetsProvider(currentUser?.isarId)).when(
+ ref
+ .watch(
+ timelineUsers.length > 1
+ ? multiUserAssetsProvider(timelineUsers)
+ : assetsProvider(currentUser?.isarId),
+ )
+ .when(
data: (data) => data.isEmpty
? buildLoadingIndicator()
: ImmichAssetGrid(
@@ -339,11 +388,10 @@ class HomePage extends HookConsumerWidget {
listener: selectionListener,
selectionActive: selectionEnabledHook.value,
onRefresh: refreshAssets,
- topWidget: (currentUser != null &&
- currentUser.memoryEnabled != null &&
- currentUser.memoryEnabled!)
- ? const MemoryLane()
- : const SizedBox(),
+ topWidget:
+ (currentUser != null && currentUser.memoryEnabled)
+ ? const MemoryLane()
+ : const SizedBox(),
showStack: true,
),
error: (error, _) => Center(child: Text(error.toString())),
@@ -364,7 +412,6 @@ class HomePage extends HookConsumerWidget {
selectionAssetState: selectionAssetState.value,
onStack: onStack,
),
- if (processing.value) const Center(child: ImmichLoadingIndicator()),
],
),
);
diff --git a/mobile/lib/modules/login/models/authentication_state.model.dart b/mobile/lib/modules/login/models/authentication_state.model.dart
index 648670ca6..9dcd320c8 100644
--- a/mobile/lib/modules/login/models/authentication_state.model.dart
+++ b/mobile/lib/modules/login/models/authentication_state.model.dart
@@ -3,8 +3,7 @@ class AuthenticationState {
final String userId;
final String userEmail;
final bool isAuthenticated;
- final String firstName;
- final String lastName;
+ final String name;
final bool isAdmin;
final bool shouldChangePassword;
final String profileImagePath;
@@ -13,8 +12,7 @@ class AuthenticationState {
required this.userId,
required this.userEmail,
required this.isAuthenticated,
- required this.firstName,
- required this.lastName,
+ required this.name,
required this.isAdmin,
required this.shouldChangePassword,
required this.profileImagePath,
@@ -25,8 +23,7 @@ class AuthenticationState {
String? userId,
String? userEmail,
bool? isAuthenticated,
- String? firstName,
- String? lastName,
+ String? name,
bool? isAdmin,
bool? shouldChangePassword,
String? profileImagePath,
@@ -36,8 +33,7 @@ class AuthenticationState {
userId: userId ?? this.userId,
userEmail: userEmail ?? this.userEmail,
isAuthenticated: isAuthenticated ?? this.isAuthenticated,
- firstName: firstName ?? this.firstName,
- lastName: lastName ?? this.lastName,
+ name: name ?? this.name,
isAdmin: isAdmin ?? this.isAdmin,
shouldChangePassword: shouldChangePassword ?? this.shouldChangePassword,
profileImagePath: profileImagePath ?? this.profileImagePath,
@@ -46,7 +42,7 @@ class AuthenticationState {
@override
String toString() {
- return 'AuthenticationState(deviceId: $deviceId, userId: $userId, userEmail: $userEmail, isAuthenticated: $isAuthenticated, firstName: $firstName, lastName: $lastName, isAdmin: $isAdmin, shouldChangePassword: $shouldChangePassword, profileImagePath: $profileImagePath)';
+ return 'AuthenticationState(deviceId: $deviceId, userId: $userId, userEmail: $userEmail, isAuthenticated: $isAuthenticated, name: $name, isAdmin: $isAdmin, shouldChangePassword: $shouldChangePassword, profileImagePath: $profileImagePath)';
}
@override
@@ -58,8 +54,7 @@ class AuthenticationState {
other.userId == userId &&
other.userEmail == userEmail &&
other.isAuthenticated == isAuthenticated &&
- other.firstName == firstName &&
- other.lastName == lastName &&
+ other.name == name &&
other.isAdmin == isAdmin &&
other.shouldChangePassword == shouldChangePassword &&
other.profileImagePath == profileImagePath;
@@ -71,8 +66,7 @@ class AuthenticationState {
userId.hashCode ^
userEmail.hashCode ^
isAuthenticated.hashCode ^
- firstName.hashCode ^
- lastName.hashCode ^
+ name.hashCode ^
isAdmin.hashCode ^
shouldChangePassword.hashCode ^
profileImagePath.hashCode;
diff --git a/mobile/lib/modules/login/providers/authentication.provider.dart b/mobile/lib/modules/login/providers/authentication.provider.dart
index 33d2c0f3e..14a094963 100644
--- a/mobile/lib/modules/login/providers/authentication.provider.dart
+++ b/mobile/lib/modules/login/providers/authentication.provider.dart
@@ -26,8 +26,7 @@ class AuthenticationNotifier extends StateNotifier {
deviceId: "",
userId: "",
userEmail: "",
- firstName: '',
- lastName: '',
+ name: '',
profileImagePath: '',
isAdmin: false,
shouldChangePassword: false,
@@ -117,8 +116,7 @@ class AuthenticationNotifier extends StateNotifier {
deviceId: "",
userId: "",
userEmail: "",
- firstName: '',
- lastName: '',
+ name: '',
profileImagePath: '',
isAdmin: false,
shouldChangePassword: false,
@@ -187,12 +185,15 @@ class AuthenticationNotifier extends StateNotifier {
if (userResponseDto != null) {
Store.put(StoreKey.deviceId, deviceId);
Store.put(StoreKey.deviceIdHash, fastHash(deviceId));
- Store.put(StoreKey.currentUser, User.fromDto(userResponseDto));
+ Store.put(
+ StoreKey.currentUser,
+ User.fromUserDto(userResponseDto),
+ );
Store.put(StoreKey.serverUrl, serverUrl);
Store.put(StoreKey.accessToken, accessToken);
shouldChangePassword = userResponseDto.shouldChangePassword;
- user = User.fromDto(userResponseDto);
+ user = User.fromUserDto(userResponseDto);
retResult = true;
} else {
@@ -205,8 +206,7 @@ class AuthenticationNotifier extends StateNotifier {
isAuthenticated: true,
userId: user.id,
userEmail: user.email,
- firstName: user.firstName,
- lastName: user.lastName,
+ name: user.name,
profileImagePath: user.profileImagePath,
isAdmin: user.isAdmin,
shouldChangePassword: shouldChangePassword,
diff --git a/mobile/lib/modules/login/ui/change_password_form.dart b/mobile/lib/modules/login/ui/change_password_form.dart
index b7db83b3f..560967821 100644
--- a/mobile/lib/modules/login/ui/change_password_form.dart
+++ b/mobile/lib/modules/login/ui/change_password_form.dart
@@ -4,6 +4,7 @@ import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:fluttertoast/fluttertoast.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/backup/providers/backup.provider.dart';
import 'package:immich_mobile/modules/backup/providers/manual_upload.provider.dart';
import 'package:immich_mobile/modules/login/providers/authentication.provider.dart';
@@ -37,7 +38,7 @@ class ChangePasswordForm extends HookConsumerWidget {
style: TextStyle(
fontSize: 24,
fontWeight: FontWeight.bold,
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
),
),
Padding(
@@ -45,8 +46,7 @@ class ChangePasswordForm extends HookConsumerWidget {
child: Text(
'change_password_form_description'.tr(
namedArgs: {
- 'firstName': authState.firstName,
- 'lastName': authState.lastName,
+ 'name': authState.name,
},
),
style: TextStyle(
@@ -191,7 +191,7 @@ class ChangePasswordButton extends ConsumerWidget {
return ElevatedButton(
style: ElevatedButton.styleFrom(
visualDensity: VisualDensity.standard,
- backgroundColor: Theme.of(context).primaryColor,
+ backgroundColor: context.primaryColor,
foregroundColor: Colors.grey[50],
elevation: 2,
padding: const EdgeInsets.symmetric(vertical: 10, horizontal: 25),
diff --git a/mobile/lib/modules/login/ui/login_form.dart b/mobile/lib/modules/login/ui/login_form.dart
index 58c7feec2..6c12742e4 100644
--- a/mobile/lib/modules/login/ui/login_form.dart
+++ b/mobile/lib/modules/login/ui/login_form.dart
@@ -1,9 +1,9 @@
import 'dart:io';
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart' hide Store;
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/login/providers/oauth.provider.dart';
import 'package:immich_mobile/modules/onboarding/providers/gallery_permission.provider.dart';
import 'package:immich_mobile/routing/router.dart';
@@ -48,7 +48,7 @@ class LoginForm extends HookConsumerWidget {
/// Fetch the server login credential and enables oAuth login if necessary
/// Returns true if successful, false otherwise
Future getServerLoginCredential() async {
- final serverUrl = serverEndpointController.text.trim();
+ final serverUrl = sanitizeUrl(serverEndpointController.text);
// Guard empty URL
if (serverUrl.isEmpty) {
@@ -127,6 +127,12 @@ class LoginForm extends HookConsumerWidget {
);
populateTestLoginInfo() {
+ usernameController.text = 'demo@immich.app';
+ passwordController.text = 'demo';
+ serverEndpointController.text = 'https://demo.immich.app';
+ }
+
+ populateTestLoginInfo1() {
usernameController.text = 'testuser@email.com';
passwordController.text = 'password';
serverEndpointController.text = 'http://10.1.15.216:2283/api';
@@ -144,13 +150,13 @@ class LoginForm extends HookConsumerWidget {
await ref.read(authenticationProvider.notifier).login(
usernameController.text,
passwordController.text,
- serverEndpointController.text.trim(),
+ sanitizeUrl(serverEndpointController.text),
);
if (isAuthenticated) {
// Resume backup (if enable) then navigate
if (ref.read(authenticationProvider).shouldChangePassword &&
!ref.read(authenticationProvider).isAdmin) {
- AutoRouter.of(context).push(const ChangePasswordRoute());
+ context.autoPush(const ChangePasswordRoute());
} else {
final hasPermission = await ref
.read(galleryPermissionNotifier.notifier)
@@ -159,7 +165,7 @@ class LoginForm extends HookConsumerWidget {
// Don't resume the backup until we have gallery permission
ref.read(backupProvider.notifier).resumeBackup();
}
- AutoRouter.of(context).replace(const TabControllerRoute());
+ context.autoReplace(const TabControllerRoute());
}
} else {
ImmichToast.show(
@@ -181,7 +187,7 @@ class LoginForm extends HookConsumerWidget {
try {
oAuthServerConfig = await oAuthService
- .getOAuthServerConfig(serverEndpointController.text);
+ .getOAuthServerConfig(sanitizeUrl(serverEndpointController.text));
isLoading.value = true;
} catch (e) {
@@ -203,7 +209,7 @@ class LoginForm extends HookConsumerWidget {
.watch(authenticationProvider.notifier)
.setSuccessLoginInfo(
accessToken: loginResponseDto.accessToken,
- serverUrl: serverEndpointController.text,
+ serverUrl: sanitizeUrl(serverEndpointController.text),
);
if (isSuccess) {
@@ -212,9 +218,7 @@ class LoginForm extends HookConsumerWidget {
if (permission.isGranted || permission.isLimited) {
ref.watch(backupProvider.notifier).resumeBackup();
}
- AutoRouter.of(context).replace(
- const TabControllerRoute(),
- );
+ context.autoReplace(const TabControllerRoute());
} else {
ImmichToast.show(
context: context,
@@ -260,8 +264,7 @@ class LoginForm extends HookConsumerWidget {
),
),
),
- onPressed: () =>
- AutoRouter.of(context).push(const SettingsRoute()),
+ onPressed: () => context.autoPush(const SettingsRoute()),
icon: const Icon(Icons.settings_rounded),
label: const SizedBox.shrink(),
),
@@ -302,8 +305,8 @@ class LoginForm extends HookConsumerWidget {
crossAxisAlignment: CrossAxisAlignment.stretch,
children: [
Text(
- serverEndpointController.text,
- style: Theme.of(context).textTheme.displaySmall,
+ sanitizeUrl(serverEndpointController.text),
+ style: context.textTheme.displaySmall,
textAlign: TextAlign.center,
),
if (isPasswordLoginEnable.value) ...[
@@ -339,8 +342,7 @@ class LoginForm extends HookConsumerWidget {
horizontal: 16.0,
),
child: Divider(
- color: Brightness.dark ==
- Theme.of(context).brightness
+ color: context.isDarkTheme
? Colors.white
: Colors.black,
),
@@ -362,7 +364,7 @@ class LoginForm extends HookConsumerWidget {
TextButton.icon(
icon: const Icon(Icons.arrow_back),
onPressed: () => serverEndpoint.value = null,
- label: const Text('Back'),
+ label: const Text('login_form_back_button_text').tr(),
),
],
),
@@ -391,6 +393,7 @@ class LoginForm extends HookConsumerWidget {
children: [
GestureDetector(
onDoubleTap: () => populateTestLoginInfo(),
+ onLongPress: () => populateTestLoginInfo1(),
child: RotationTransition(
turns: logoAnimationController,
child: const ImmichLogo(
@@ -588,7 +591,7 @@ class OAuthLoginButton extends ConsumerWidget {
Widget build(BuildContext context, WidgetRef ref) {
return ElevatedButton.icon(
style: ElevatedButton.styleFrom(
- backgroundColor: Theme.of(context).primaryColor.withAlpha(230),
+ backgroundColor: context.primaryColor.withAlpha(230),
padding: const EdgeInsets.symmetric(vertical: 12),
),
onPressed: onPressed,
diff --git a/mobile/lib/modules/login/views/login_page.dart b/mobile/lib/modules/login/views/login_page.dart
index 98778736e..4e1b9a6df 100644
--- a/mobile/lib/modules/login/views/login_page.dart
+++ b/mobile/lib/modules/login/views/login_page.dart
@@ -1,7 +1,7 @@
-import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/login/ui/login_form.dart';
import 'package:immich_mobile/routing/router.dart';
import 'package:package_info_plus/package_info_plus.dart';
@@ -47,13 +47,13 @@ class LoginPage extends HookConsumerWidget {
child: Text(
'Logs',
style: TextStyle(
- color: Theme.of(context).primaryColor,
+ color: context.primaryColor,
fontWeight: FontWeight.bold,
fontFamily: "Inconsolata",
),
),
onTap: () {
- AutoRouter.of(context).push(const AppLogRoute());
+ context.autoPush(const AppLogRoute());
},
),
],
diff --git a/mobile/lib/modules/map/models/map_state.model.dart b/mobile/lib/modules/map/models/map_state.model.dart
index 7aab8efab..d606f1005 100644
--- a/mobile/lib/modules/map/models/map_state.model.dart
+++ b/mobile/lib/modules/map/models/map_state.model.dart
@@ -1,14 +1,20 @@
+import 'package:vector_map_tiles/vector_map_tiles.dart';
+
class MapState {
final bool isDarkTheme;
final bool showFavoriteOnly;
final bool includeArchived;
final int relativeTime;
+ final Style? mapStyle;
+ final bool isLoading;
MapState({
this.isDarkTheme = false,
this.showFavoriteOnly = false,
this.includeArchived = false,
this.relativeTime = 0,
+ this.mapStyle,
+ this.isLoading = false,
});
MapState copyWith({
@@ -16,18 +22,22 @@ class MapState {
bool? showFavoriteOnly,
bool? includeArchived,
int? relativeTime,
+ Style? mapStyle,
+ bool? isLoading,
}) {
return MapState(
isDarkTheme: isDarkTheme ?? this.isDarkTheme,
showFavoriteOnly: showFavoriteOnly ?? this.showFavoriteOnly,
includeArchived: includeArchived ?? this.includeArchived,
relativeTime: relativeTime ?? this.relativeTime,
+ mapStyle: mapStyle ?? this.mapStyle,
+ isLoading: isLoading ?? this.isLoading,
);
}
@override
String toString() {
- return 'MapSettingsState(isDarkTheme: $isDarkTheme, showFavoriteOnly: $showFavoriteOnly, relativeTime: $relativeTime, includeArchived: $includeArchived)';
+ return 'MapSettingsState(isDarkTheme: $isDarkTheme, showFavoriteOnly: $showFavoriteOnly, relativeTime: $relativeTime, includeArchived: $includeArchived, mapStyle: $mapStyle, isLoading: $isLoading)';
}
@override
@@ -38,7 +48,9 @@ class MapState {
other.isDarkTheme == isDarkTheme &&
other.showFavoriteOnly == showFavoriteOnly &&
other.relativeTime == relativeTime &&
- other.includeArchived == includeArchived;
+ other.includeArchived == includeArchived &&
+ other.mapStyle == mapStyle &&
+ other.isLoading == isLoading;
}
@override
@@ -46,6 +58,8 @@ class MapState {
return isDarkTheme.hashCode ^
showFavoriteOnly.hashCode ^
relativeTime.hashCode ^
- includeArchived.hashCode;
+ includeArchived.hashCode ^
+ mapStyle.hashCode ^
+ isLoading.hashCode;
}
}
diff --git a/mobile/lib/modules/map/providers/map_state.provider.dart b/mobile/lib/modules/map/providers/map_state.provider.dart
index 8f4356198..3296b09c0 100644
--- a/mobile/lib/modules/map/providers/map_state.provider.dart
+++ b/mobile/lib/modules/map/providers/map_state.provider.dart
@@ -1,10 +1,23 @@
+import 'dart:convert';
+import 'dart:io';
+
+import 'package:flutter/foundation.dart';
+import 'package:flutter/material.dart';
+import 'package:flutter_map/flutter_map.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/modules/map/models/map_state.model.dart';
import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart';
import 'package:immich_mobile/modules/settings/services/app_settings.service.dart';
+import 'package:immich_mobile/shared/providers/api.provider.dart';
+import 'package:immich_mobile/shared/services/api.service.dart';
+import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
+import 'package:immich_mobile/utils/color_filter_generator.dart';
+import 'package:logging/logging.dart';
+import 'package:openapi/api.dart';
+import 'package:vector_map_tiles/vector_map_tiles.dart';
class MapStateNotifier extends StateNotifier {
- MapStateNotifier(this._appSettingsProvider)
+ MapStateNotifier(this._appSettingsProvider, this._apiService)
: super(
MapState(
isDarkTheme: _appSettingsProvider
@@ -15,17 +28,69 @@ class MapStateNotifier extends StateNotifier {
.getSetting(AppSettingsEnum.mapIncludeArchived),
relativeTime: _appSettingsProvider
.getSetting(AppSettingsEnum.mapRelativeDate),
+ isLoading: true,
),
- );
+ ) {
+ _fetchStyleFromServer(
+ _appSettingsProvider.getSetting(AppSettingsEnum.mapThemeMode),
+ );
+ }
final AppSettingsService _appSettingsProvider;
+ final ApiService _apiService;
+ final Logger _log = Logger("MapStateNotifier");
+
+ bool get isRaster =>
+ state.mapStyle != null && state.mapStyle!.rasterTileProvider != null;
+
+ double get maxZoom =>
+ (isRaster ? state.mapStyle!.rasterTileProvider!.maximumZoom : 14)
+ .toDouble();
void switchTheme(bool isDarkTheme) {
+ _updateThemeMode(isDarkTheme);
+ _fetchStyleFromServer(isDarkTheme);
+ }
+
+ void _updateThemeMode(bool isDarkTheme) {
_appSettingsProvider.setSetting(
AppSettingsEnum.mapThemeMode,
isDarkTheme,
);
- state = state.copyWith(isDarkTheme: isDarkTheme);
+ state = state.copyWith(isDarkTheme: isDarkTheme, isLoading: true);
+ }
+
+ void _fetchStyleFromServer(bool isDarkTheme) async {
+ final styleResponse = await _apiService.systemConfigApi
+ .getMapStyleWithHttpInfo(isDarkTheme ? MapTheme.dark : MapTheme.light);
+ if (styleResponse.statusCode >= HttpStatus.badRequest) {
+ throw ApiException(styleResponse.statusCode, styleResponse.body);
+ }
+ final styleJsonString = styleResponse.body.isNotEmpty &&
+ styleResponse.statusCode != HttpStatus.noContent
+ ? styleResponse.body
+ : null;
+
+ if (styleJsonString == null) {
+ _log.severe('Style JSON from server is empty');
+ return;
+ }
+ final styleJson = await compute(jsonDecode, styleJsonString);
+ if (styleJson is! Map) {
+ _log.severe('Style JSON from server is invalid');
+ return;
+ }
+ final styleReader = StyleReader(uri: '');
+ Style? style;
+ try {
+ style = await styleReader.readFromMap(styleJson);
+ } finally {
+ // Consume all error
+ }
+ state = state.copyWith(
+ mapStyle: style,
+ isLoading: false,
+ );
}
void switchFavoriteOnly(bool isFavoriteOnly) {
@@ -51,9 +116,44 @@ class MapStateNotifier extends StateNotifier {
);
state = state.copyWith(relativeTime: relativeTime);
}
+
+ Widget getTileLayer([bool forceDark = false]) {
+ if (isRaster) {
+ final rasterProvider = state.mapStyle!.rasterTileProvider;
+ final rasterLayer = TileLayer(
+ urlTemplate: rasterProvider!.url,
+ maxNativeZoom: rasterProvider.maximumZoom,
+ maxZoom: rasterProvider.maximumZoom.toDouble(),
+ );
+ return state.isDarkTheme || forceDark
+ ? InvertionFilter(
+ child: SaturationFilter(
+ saturation: -1,
+ child: BrightnessFilter(
+ brightness: -1,
+ child: rasterLayer,
+ ),
+ ),
+ )
+ : rasterLayer;
+ }
+ if (state.mapStyle != null && !isRaster) {
+ return VectorTileLayer(
+ // Tiles and themes will be set for vector providers
+ tileProviders: state.mapStyle!.providers!,
+ theme: state.mapStyle!.theme!,
+ sprites: state.mapStyle!.sprites,
+ concurrency: 6,
+ );
+ }
+ return const Center(child: ImmichLoadingIndicator());
+ }
}
final mapStateNotifier =
StateNotifierProvider((ref) {
- return MapStateNotifier(ref.watch(appSettingsServiceProvider));
+ return MapStateNotifier(
+ ref.watch(appSettingsServiceProvider),
+ ref.watch(apiServiceProvider),
+ );
});
diff --git a/mobile/lib/modules/map/ui/map_page_app_bar.dart b/mobile/lib/modules/map/ui/map_page_app_bar.dart
index e9ed75cb1..ce426cf03 100644
--- a/mobile/lib/modules/map/ui/map_page_app_bar.dart
+++ b/mobile/lib/modules/map/ui/map_page_app_bar.dart
@@ -1,8 +1,8 @@
import 'dart:io';
-import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/disable_multi_select_button.dart';
import 'package:immich_mobile/modules/map/ui/map_settings_dialog.dart';
@@ -30,7 +30,7 @@ class MapAppBar extends HookWidget implements PreferredSizeWidget {
Padding(
padding: const EdgeInsets.only(left: 15, top: 15),
child: ElevatedButton(
- onPressed: () => AutoRouter.of(context).pop(),
+ onPressed: () => context.autoPop(),
style: ElevatedButton.styleFrom(
shape: const CircleBorder(),
padding: const EdgeInsets.all(12),
diff --git a/mobile/lib/modules/map/ui/map_page_bottom_sheet.dart b/mobile/lib/modules/map/ui/map_page_bottom_sheet.dart
index c9c3cb8aa..7e8ff13d0 100644
--- a/mobile/lib/modules/map/ui/map_page_bottom_sheet.dart
+++ b/mobile/lib/modules/map/ui/map_page_bottom_sheet.dart
@@ -5,6 +5,7 @@ import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/asset_viewer/providers/render_list.provider.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/asset_grid_data_structure.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/immich_asset_grid.dart';
@@ -15,7 +16,6 @@ import 'package:immich_mobile/shared/ui/drag_sheet.dart';
import 'package:immich_mobile/utils/color_filter_generator.dart';
import 'package:immich_mobile/utils/debounce.dart';
import 'package:scrollable_positioned_list/scrollable_positioned_list.dart';
-import 'package:url_launcher/url_launcher.dart';
class MapPageBottomSheet extends StatefulHookConsumerWidget {
final Stream mapPageEventStream;
@@ -57,10 +57,10 @@ class AssetsInBoundBottomSheetState extends ConsumerState {
@override
Widget build(BuildContext context) {
- final isDarkMode = Theme.of(context).brightness == Brightness.dark;
+ final isDarkTheme = context.isDarkTheme;
final bottomPadding =
Platform.isAndroid ? MediaQuery.of(context).padding.bottom - 10 : 0.0;
- final maxHeight = MediaQuery.of(context).size.height - bottomPadding;
+ final maxHeight = context.height - bottomPadding;
final isSheetScrolled = useState(false);
final isSheetExpanded = useState(false);
final assetsInBound = useState([]);
@@ -137,7 +137,7 @@ class AssetsInBoundBottomSheetState extends ConsumerState {
SizedBox(
height: 150,
width: 150,
- child: isDarkMode
+ child: isDarkTheme
? const InvertionFilter(
child: SaturationFilter(
saturation: -1,
@@ -156,7 +156,7 @@ class AssetsInBoundBottomSheetState extends ConsumerState {
"map_zoom_to_see_photos".tr(),
style: TextStyle(
fontSize: 20,
- color: Theme.of(context).textTheme.displayLarge?.color,
+ color: context.textTheme.displayLarge?.color,
),
),
],
@@ -176,13 +176,13 @@ class AssetsInBoundBottomSheetState extends ConsumerState {
Widget buildDragHandle(ScrollController scrollController) {
final textToDisplay = assetsInBound.value.isNotEmpty
- ? "${assetsInBound.value.length} photo${assetsInBound.value.length > 1 ? "s" : ""}"
+ ? "map_assets_in_bounds".plural(assetsInBound.value.length)
: "map_no_assets_in_bounds".tr();
final dragHandle = Container(
- height: 60,
+ height: 70,
width: double.infinity,
decoration: BoxDecoration(
- color: isDarkMode ? Colors.grey[900] : Colors.grey[100],
+ color: isDarkTheme ? Colors.grey[900] : Colors.grey[100],
),
child: Stack(
children: [
@@ -195,19 +195,12 @@ class AssetsInBoundBottomSheetState extends ConsumerState {
const SizedBox(height: 15),
Text(
textToDisplay,
- style: TextStyle(
- fontSize: 16,
- color: Theme.of(context).textTheme.displayLarge?.color,
- fontWeight: FontWeight.bold,
- ),
+ style: context.textTheme.bodyLarge,
),
Divider(
height: 10,
- color: Theme.of(context)
- .textTheme
- .displayLarge
- ?.color
- ?.withOpacity(0.5),
+ color:
+ context.textTheme.displayLarge?.color?.withOpacity(0.5),
),
],
),
@@ -218,7 +211,7 @@ class AssetsInBoundBottomSheetState extends ConsumerState {
child: IconButton(
icon: Icon(
Icons.map_outlined,
- color: Theme.of(context).textTheme.displayLarge?.color,
+ color: context.textTheme.displayLarge?.color,
),
iconSize: 20,
tooltip: 'Zoom to bounds',
@@ -266,7 +259,7 @@ class AssetsInBoundBottomSheetState extends ConsumerState {
ScrollController scrollController,
) {
return Card(
- color: isDarkMode ? Colors.grey[900] : Colors.grey[100],
+ color: isDarkTheme ? Colors.grey[900] : Colors.grey[100],
surfaceTintColor: Colors.transparent,
elevation: 18.0,
margin: const EdgeInsets.all(0),
@@ -320,24 +313,18 @@ class AssetsInBoundBottomSheetState extends ConsumerState {
Positioned(
bottom: maxHeight * currentExtend.value,
left: 0,
- child: GestureDetector(
- onTap: () => launchUrl(
- Uri.parse('https://openstreetmap.org/copyright'),
- ),
- child: ColoredBox(
- color: (widget.isDarkTheme
- ? Colors.grey[900]
- : Colors.grey[100])!,
- child: Padding(
- padding: const EdgeInsets.all(3),
- child: Text(
- '© OpenStreetMap contributors',
- style: TextStyle(
- fontSize: 6,
- color: !widget.isDarkTheme
- ? Colors.grey[900]
- : Colors.grey[100],
- ),
+ child: ColoredBox(
+ color:
+ (widget.isDarkTheme ? Colors.grey[900] : Colors.grey[100])!,
+ child: Padding(
+ padding: const EdgeInsets.all(3),
+ child: Text(
+ 'OpenStreetMap contributors',
+ style: TextStyle(
+ fontSize: 6,
+ color: !widget.isDarkTheme
+ ? Colors.grey[900]
+ : Colors.grey[100],
),
),
),
diff --git a/mobile/lib/modules/map/ui/map_settings_dialog.dart b/mobile/lib/modules/map/ui/map_settings_dialog.dart
index f8a308b35..9619bd661 100644
--- a/mobile/lib/modules/map/ui/map_settings_dialog.dart
+++ b/mobile/lib/modules/map/ui/map_settings_dialog.dart
@@ -2,6 +2,7 @@ import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/map/providers/map_state.provider.dart';
class MapSettingsDialog extends HookConsumerWidget {
@@ -15,7 +16,7 @@ class MapSettingsDialog extends HookConsumerWidget {
final showFavoriteOnly = useState(mapSettings.showFavoriteOnly);
final showIncludeArchived = useState(mapSettings.includeArchived);
final showRelativeDate = useState(mapSettings.relativeTime);
- final ThemeData theme = Theme.of(context);
+ final ThemeData theme = context.themeData;
Widget buildMapThemeSetting() {
return SwitchListTile.adaptive(
@@ -75,18 +76,21 @@ class MapSettingsDialog extends HookConsumerWidget {
showRelativeDate.value = value!;
},
dropdownMenuEntries: [
- const DropdownMenuEntry(value: 0, label: "All"),
- const DropdownMenuEntry(
+ DropdownMenuEntry(
+ value: 0,
+ label: "map_settings_date_range_option_all".tr(),
+ ),
+ DropdownMenuEntry(
value: 1,
- label: "Past 24 hours",
+ label: "map_settings_date_range_option_days".plural(1),
),
- const DropdownMenuEntry(
+ DropdownMenuEntry(
value: 7,
- label: "Past 7 days",
+ label: "map_settings_date_range_option_days".plural(7),
),
- const DropdownMenuEntry(
+ DropdownMenuEntry(
value: 30,
- label: "Past 30 days",
+ label: "map_settings_date_range_option_days".plural(30),
),
DropdownMenuEntry(
value: now
@@ -101,7 +105,7 @@ class MapSettingsDialog extends HookConsumerWidget {
),
)
.inDays,
- label: "Past year",
+ label: "map_settings_date_range_option_years".plural(1),
),
DropdownMenuEntry(
value: now
@@ -116,7 +120,7 @@ class MapSettingsDialog extends HookConsumerWidget {
),
)
.inDays,
- label: "Past 3 years",
+ label: "map_settings_date_range_option_years".plural(3),
),
],
);
@@ -125,17 +129,21 @@ class MapSettingsDialog extends HookConsumerWidget {
List getDialogActions() {
return [
TextButton(
- onPressed: () => Navigator.of(context).pop(),
+ onPressed: () => context.pop(),
style: TextButton.styleFrom(
backgroundColor:
mapSettings.isDarkTheme ? Colors.grey[100] : Colors.grey[700],
),
- child: Text(
- "map_settings_dialog_cancel".tr(),
- style: theme.textTheme.labelSmall?.copyWith(
- fontWeight: FontWeight.bold,
- color:
- mapSettings.isDarkTheme ? Colors.grey[900] : Colors.grey[100],
+ child: Padding(
+ padding: const EdgeInsets.symmetric(horizontal: 16.0),
+ child: Text(
+ "map_settings_dialog_cancel".tr(),
+ style: theme.textTheme.labelLarge?.copyWith(
+ fontWeight: FontWeight.w500,
+ color: mapSettings.isDarkTheme
+ ? Colors.grey[900]
+ : Colors.grey[100],
+ ),
),
),
),
@@ -146,16 +154,19 @@ class MapSettingsDialog extends HookConsumerWidget {
mapSettingsNotifier.setRelativeTime(showRelativeDate.value);
mapSettingsNotifier
.switchIncludeArchived(showIncludeArchived.value);
- Navigator.of(context).pop();
+ context.pop();
},
style: TextButton.styleFrom(
backgroundColor: theme.primaryColor,
),
- child: Text(
- "map_settings_dialog_save".tr(),
- style: theme.textTheme.labelSmall?.copyWith(
- fontWeight: FontWeight.bold,
- color: theme.primaryTextTheme.labelLarge?.color,
+ child: Padding(
+ padding: const EdgeInsets.symmetric(horizontal: 16.0),
+ child: Text(
+ "map_settings_dialog_save".tr(),
+ style: theme.textTheme.labelLarge?.copyWith(
+ fontWeight: FontWeight.w500,
+ color: theme.primaryTextTheme.labelLarge?.color,
+ ),
),
),
),
@@ -178,7 +189,7 @@ class MapSettingsDialog extends HookConsumerWidget {
width: double.maxFinite,
child: ConstrainedBox(
constraints: BoxConstraints(
- maxHeight: MediaQuery.of(context).size.height * 0.6,
+ maxHeight: context.height * 0.6,
),
child: ListView(
shrinkWrap: true,
diff --git a/mobile/lib/modules/map/ui/map_thumbnail.dart b/mobile/lib/modules/map/ui/map_thumbnail.dart
index 14cc2a83b..d42d99de1 100644
--- a/mobile/lib/modules/map/ui/map_thumbnail.dart
+++ b/mobile/lib/modules/map/ui/map_thumbnail.dart
@@ -1,8 +1,7 @@
import 'package:flutter/material.dart';
import 'package:flutter_map/plugin_api.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:immich_mobile/shared/providers/server_info.provider.dart';
-import 'package:immich_mobile/utils/color_filter_generator.dart';
+import 'package:immich_mobile/modules/map/providers/map_state.provider.dart';
import 'package:latlong2/latlong.dart';
import 'package:url_launcher/url_launcher.dart';
@@ -29,11 +28,7 @@ class MapThumbnail extends HookConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
- final tileLayer = TileLayer(
- urlTemplate: ref.watch(
- serverInfoProvider.select((v) => v.serverConfig.mapTileUrl),
- ),
- );
+ ref.watch(mapStateNotifier.select((s) => s.mapStyle));
return SizedBox(
height: height,
@@ -55,20 +50,14 @@ class MapThumbnail extends HookConsumerWidget {
'OpenStreetMap contributors',
onTap: () => launchUrl(
Uri.parse('https://openstreetmap.org/copyright'),
+ mode: LaunchMode.externalApplication,
),
),
],
),
],
children: [
- isDarkTheme
- ? InvertionFilter(
- child: SaturationFilter(
- saturation: -1,
- child: tileLayer,
- ),
- )
- : tileLayer,
+ ref.read(mapStateNotifier.notifier).getTileLayer(isDarkTheme),
if (markers.isNotEmpty) MarkerLayer(markers: markers),
],
),
diff --git a/mobile/lib/modules/map/views/map_page.dart b/mobile/lib/modules/map/views/map_page.dart
index ffa18d37a..b03c13e36 100644
--- a/mobile/lib/modules/map/views/map_page.dart
+++ b/mobile/lib/modules/map/views/map_page.dart
@@ -1,6 +1,6 @@
import 'dart:async';
+import 'dart:math' as math;
-import 'package:auto_route/auto_route.dart';
import 'package:collection/collection.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
@@ -11,6 +11,7 @@ import 'package:flutter_map_heatmap/flutter_map_heatmap.dart';
import 'package:fluttertoast/fluttertoast.dart';
import 'package:geolocator/geolocator.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/map/models/map_page_event.model.dart';
import 'package:immich_mobile/modules/map/providers/map_marker.provider.dart';
import 'package:immich_mobile/modules/map/providers/map_state.provider.dart';
@@ -20,12 +21,10 @@ import 'package:immich_mobile/modules/map/ui/map_page_bottom_sheet.dart';
import 'package:immich_mobile/modules/map/ui/map_page_app_bar.dart';
import 'package:immich_mobile/routing/router.dart';
import 'package:immich_mobile/shared/models/asset.dart';
-import 'package:immich_mobile/shared/providers/server_info.provider.dart';
import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
import 'package:immich_mobile/shared/ui/immich_toast.dart';
-import 'package:immich_mobile/utils/color_filter_generator.dart';
import 'package:immich_mobile/utils/debounce.dart';
-import 'package:immich_mobile/utils/flutter_map_extensions.dart';
+import 'package:immich_mobile/extensions/flutter_map_extensions.dart';
import 'package:immich_mobile/utils/immich_app_theme.dart';
import 'package:immich_mobile/utils/selection_handlers.dart';
import 'package:latlong2/latlong.dart';
@@ -79,26 +78,30 @@ class MapPageState extends ConsumerState {
Set? assetMarkers, {
bool forceReload = false,
}) {
- final bounds = mapController.bounds;
- if (bounds != null) {
- final oldAssetsInBounds = assetsInBounds.toSet();
- assetsInBounds =
- assetMarkers?.where((e) => bounds.contains(e.point)).toSet() ?? {};
- final shouldReload = forceReload ||
- assetsInBounds.difference(oldAssetsInBounds).isNotEmpty ||
- assetsInBounds.length != oldAssetsInBounds.length;
- if (shouldReload) {
- mapPageEventSC.add(
- MapPageAssetsInBoundUpdated(
- assetsInBounds.map((e) => e.asset).toList(),
- ),
- );
+ try {
+ final bounds = mapController.bounds;
+ if (bounds != null) {
+ final oldAssetsInBounds = assetsInBounds.toSet();
+ assetsInBounds =
+ assetMarkers?.where((e) => bounds.contains(e.point)).toSet() ?? {};
+ final shouldReload = forceReload ||
+ assetsInBounds.difference(oldAssetsInBounds).isNotEmpty ||
+ assetsInBounds.length != oldAssetsInBounds.length;
+ if (shouldReload) {
+ mapPageEventSC.add(
+ MapPageAssetsInBoundUpdated(
+ assetsInBounds.map((e) => e.asset).toList(),
+ ),
+ );
+ }
}
+ } finally {
+ // Consume all error
}
}
void openAssetInViewer(Asset asset) {
- AutoRouter.of(context).push(
+ context.autoPush(
GalleryViewerRoute(
initialIndex: 0,
loadAsset: (index) => asset,
@@ -120,6 +123,10 @@ class MapPageState extends ConsumerState {
final selectedAssets = useState({});
final showLoadingIndicator = useState(false);
final refetchMarkers = useState(true);
+ final isLoading =
+ ref.watch(mapStateNotifier.select((state) => state.isLoading));
+ final maxZoom = ref.read(mapStateNotifier.notifier).maxZoom;
+ final zoomLevel = math.min(maxZoom, 14.0);
if (refetchMarkers.value) {
mapMarkerData.value = ref.watch(mapMarkersProvider).when(
@@ -168,7 +175,6 @@ class MapPageState extends ConsumerState {
final mapMarker = mapMarkerData.value
.firstWhereOrNull((e) => e.asset.id == assetInBottomSheet.id);
if (mapMarker != null) {
- const zoomLevel = 16.0;
LatLng? newCenter = mapController.centerBoundsWithPadding(
mapMarker.point,
const Offset(0, -120),
@@ -230,7 +236,7 @@ class MapPageState extends ConsumerState {
forceAssetUpdate = true;
mapController.move(
LatLng(currentUserLocation.latitude, currentUserLocation.longitude),
- 12,
+ zoomLevel,
);
} catch (error) {
log.severe(
@@ -359,24 +365,6 @@ class MapPageState extends ConsumerState {
selectedAssets.value = selection;
}
- final tileLayer = TileLayer(
- urlTemplate: ref.watch(
- serverInfoProvider.select((v) => v.serverConfig.mapTileUrl),
- ),
- maxNativeZoom: 19,
- maxZoom: 19,
- );
-
- final darkTileLayer = InvertionFilter(
- child: SaturationFilter(
- saturation: -1,
- child: BrightnessFilter(
- brightness: -1,
- child: tileLayer,
- ),
- ),
- );
-
final markerLayer = MarkerLayer(
markers: [
if (closestAssetMarker.value != null)
@@ -451,41 +439,43 @@ class MapPageState extends ConsumerState {
extendBodyBehindAppBar: true,
body: Stack(
children: [
- FlutterMap(
- mapController: mapController,
- options: MapOptions(
- maxBounds:
- LatLngBounds(LatLng(-90, -180.0), LatLng(90.0, 180.0)),
- interactiveFlags: InteractiveFlag.doubleTapZoom |
- InteractiveFlag.drag |
- InteractiveFlag.flingAnimation |
- InteractiveFlag.pinchMove |
- InteractiveFlag.pinchZoom,
- center: LatLng(20, 20),
- zoom: 2,
- minZoom: 1,
- maxZoom: 18, // max level supported by OSM,
- onMapReady: () {
- mapController.mapEventStream.listen(onMapEvent);
- },
+ if (!isLoading)
+ FlutterMap(
+ mapController: mapController,
+ options: MapOptions(
+ maxBounds:
+ LatLngBounds(LatLng(-90, -180.0), LatLng(90.0, 180.0)),
+ interactiveFlags: InteractiveFlag.doubleTapZoom |
+ InteractiveFlag.drag |
+ InteractiveFlag.flingAnimation |
+ InteractiveFlag.pinchMove |
+ InteractiveFlag.pinchZoom,
+ center: LatLng(20, 20),
+ zoom: 2,
+ minZoom: 1,
+ maxZoom: maxZoom,
+ onMapReady: () {
+ mapController.mapEventStream.listen(onMapEvent);
+ },
+ ),
+ children: [
+ ref.read(mapStateNotifier.notifier).getTileLayer(),
+ heatMapLayer,
+ markerLayer,
+ ],
),
- children: [
- isDarkTheme ? darkTileLayer : tileLayer,
- heatMapLayer,
- markerLayer,
- ],
- ),
- MapPageBottomSheet(
- mapPageEventStream: mapPageEventSC.stream,
- bottomSheetEventSC: bottomSheetEventSC,
- selectionEnabled: selectionEnabledHook.value,
- selectionlistener: selectionListener,
- isDarkTheme: isDarkTheme,
- ),
- if (showLoadingIndicator.value)
+ if (!isLoading)
+ MapPageBottomSheet(
+ mapPageEventStream: mapPageEventSC.stream,
+ bottomSheetEventSC: bottomSheetEventSC,
+ selectionEnabled: selectionEnabledHook.value,
+ selectionlistener: selectionListener,
+ isDarkTheme: isDarkTheme,
+ ),
+ if (showLoadingIndicator.value || isLoading)
Positioned(
- top: MediaQuery.of(context).size.height * 0.35,
- left: MediaQuery.of(context).size.width * 0.425,
+ top: context.height * 0.35,
+ left: context.width * 0.425,
child: const ImmichLoadingIndicator(),
),
],
diff --git a/mobile/lib/modules/memories/ui/memory_card.dart b/mobile/lib/modules/memories/ui/memory_card.dart
index 8ef06e0d1..6c27ea748 100644
--- a/mobile/lib/modules/memories/ui/memory_card.dart
+++ b/mobile/lib/modules/memories/ui/memory_card.dart
@@ -3,6 +3,7 @@ import 'dart:ui';
import 'package:cached_network_image/cached_network_image.dart';
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/shared/models/asset.dart';
import 'package:immich_mobile/shared/models/store.dart';
import 'package:immich_mobile/shared/ui/immich_image.dart';
@@ -34,10 +35,9 @@ class MemoryCard extends HookConsumerWidget {
buildTitle() {
return Text(
title,
- style: const TextStyle(
+ style: context.textTheme.headlineMedium?.copyWith(
color: Colors.white,
- fontWeight: FontWeight.bold,
- fontSize: 24.0,
+ fontWeight: FontWeight.w500,
),
);
}
diff --git a/mobile/lib/modules/memories/ui/memory_lane.dart b/mobile/lib/modules/memories/ui/memory_lane.dart
index dcd803651..0c709919b 100644
--- a/mobile/lib/modules/memories/ui/memory_lane.dart
+++ b/mobile/lib/modules/memories/ui/memory_lane.dart
@@ -1,7 +1,7 @@
-import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/memories/providers/memory.provider.dart';
import 'package:immich_mobile/routing/router.dart';
import 'package:immich_mobile/shared/ui/immich_image.dart';
@@ -17,7 +17,7 @@ class MemoryLane extends HookConsumerWidget {
.whenData(
(memories) => memories != null
? Container(
- margin: const EdgeInsets.only(top: 10),
+ margin: const EdgeInsets.only(top: 10, left: 10),
height: 200,
child: ListView.builder(
scrollDirection: Axis.horizontal,
@@ -31,7 +31,7 @@ class MemoryLane extends HookConsumerWidget {
child: GestureDetector(
onTap: () {
HapticFeedback.heavyImpact();
- AutoRouter.of(context).push(
+ context.autoPush(
MemoryRoute(
memories: memories,
memoryIndex: index,
@@ -48,7 +48,7 @@ class MemoryLane extends HookConsumerWidget {
clipBehavior: Clip.hardEdge,
child: ColorFiltered(
colorFilter: ColorFilter.mode(
- Colors.black.withOpacity(0.1),
+ Colors.black.withOpacity(0.2),
BlendMode.darken,
),
child: ImmichImage(
@@ -71,9 +71,9 @@ class MemoryLane extends HookConsumerWidget {
child: Text(
memory.title,
style: const TextStyle(
- fontWeight: FontWeight.bold,
+ fontWeight: FontWeight.w600,
color: Colors.white,
- fontSize: 14,
+ fontSize: 15,
),
),
),
diff --git a/mobile/lib/modules/memories/views/memory_page.dart b/mobile/lib/modules/memories/views/memory_page.dart
index a1aae5156..9c135961e 100644
--- a/mobile/lib/modules/memories/views/memory_page.dart
+++ b/mobile/lib/modules/memories/views/memory_page.dart
@@ -1,8 +1,8 @@
-import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/memories/models/memory.dart';
import 'package:immich_mobile/modules/memories/ui/memory_card.dart';
import 'package:immich_mobile/shared/models/asset.dart';
@@ -138,8 +138,8 @@ class MemoryPage extends HookConsumerWidget {
memory.title,
style: TextStyle(
color: Colors.grey[400],
- fontSize: 11.0,
- fontWeight: FontWeight.w600,
+ fontSize: 13.0,
+ fontWeight: FontWeight.w500,
),
),
Text(
@@ -148,7 +148,7 @@ class MemoryPage extends HookConsumerWidget {
),
style: const TextStyle(
color: Colors.white,
- fontSize: 14.0,
+ fontSize: 15.0,
fontWeight: FontWeight.w500,
),
),
@@ -182,14 +182,14 @@ class MemoryPage extends HookConsumerWidget {
currentMemory.value.assets.length;
if (isLastAsset &&
(offset > notification.metrics.maxScrollExtent + 150)) {
- AutoRouter.of(context).pop();
+ context.autoPop();
return true;
}
}
// Horizontal scroll handling
if (notification.depth == 1 &&
(offset > notification.metrics.maxScrollExtent + 100)) {
- AutoRouter.of(context).pop();
+ context.autoPop();
return true;
}
}
@@ -244,7 +244,7 @@ class MemoryPage extends HookConsumerWidget {
child: MemoryCard(
asset: asset,
onTap: () => toNextAsset(index),
- onClose: () => AutoRouter.of(context).pop(),
+ onClose: () => context.autoPop(),
rightCornerText: assetProgress.value,
title: memories[mIndex].title,
showTitle: index == 0,
diff --git a/mobile/lib/modules/onboarding/views/permission_onboarding_page.dart b/mobile/lib/modules/onboarding/views/permission_onboarding_page.dart
index efbbc78a3..771deefa3 100644
--- a/mobile/lib/modules/onboarding/views/permission_onboarding_page.dart
+++ b/mobile/lib/modules/onboarding/views/permission_onboarding_page.dart
@@ -1,9 +1,7 @@
-import 'package:auto_route/auto_route.dart';
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
-import 'package:immich_mobile/modules/backup/providers/backup.provider.dart';
-import 'package:immich_mobile/modules/login/providers/authentication.provider.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/onboarding/providers/gallery_permission.provider.dart';
import 'package:immich_mobile/routing/router.dart';
import 'package:immich_mobile/shared/ui/immich_logo.dart';
@@ -11,7 +9,6 @@ import 'package:immich_mobile/shared/ui/immich_title_text.dart';
import 'package:permission_handler/permission_handler.dart';
class PermissionOnboardingPage extends HookConsumerWidget {
-
const PermissionOnboardingPage({super.key});
@override
@@ -19,16 +16,7 @@ class PermissionOnboardingPage extends HookConsumerWidget {
final PermissionStatus permission = ref.watch(galleryPermissionNotifier);
// Navigate to the main Tab Controller when permission is granted
- void goToHome() {
- // Resume backup (if enable) then navigate
- ref.watch(backupProvider.notifier).resumeBackup()
- .catchError((error) {
- debugPrint('PermissionOnboardingPage error: $error');
- });
- AutoRouter.of(context).replace(
- const TabControllerRoute(),
- );
- }
+ void goToBackup() => context.autoReplace(const BackupControllerRoute());
// When the permission is denied, we show a request permission page
buildRequestPermission() {
@@ -38,21 +26,21 @@ class PermissionOnboardingPage extends HookConsumerWidget {
children: [
Text(
'permission_onboarding_request',
- style: Theme.of(context).textTheme.titleMedium,
+ style: context.textTheme.titleMedium,
textAlign: TextAlign.center,
).tr(),
const SizedBox(height: 18),
ElevatedButton(
onPressed: () => ref
- .read(galleryPermissionNotifier.notifier)
- .requestGalleryPermission()
- .then((permission) async {
- if (permission.isGranted) {
- // If permission is limited, we will show the limited
- // permission page
- goToHome();
- }
- }),
+ .read(galleryPermissionNotifier.notifier)
+ .requestGalleryPermission()
+ .then((permission) async {
+ if (permission.isGranted) {
+ // If permission is limited, we will show the limited
+ // permission page
+ goToBackup();
+ }
+ }),
child: const Text(
'permission_onboarding_grant_permission',
).tr(),
@@ -70,12 +58,12 @@ class PermissionOnboardingPage extends HookConsumerWidget {
children: [
Text(
'permission_onboarding_permission_granted',
- style: Theme.of(context).textTheme.titleMedium,
+ style: context.textTheme.titleMedium,
textAlign: TextAlign.center,
).tr(),
const SizedBox(height: 18),
ElevatedButton(
- onPressed: () => goToHome(),
+ onPressed: () => goToBackup(),
child: const Text('permission_onboarding_get_started').tr(),
),
],
@@ -90,14 +78,15 @@ class PermissionOnboardingPage extends HookConsumerWidget {
crossAxisAlignment: CrossAxisAlignment.center,
mainAxisAlignment: MainAxisAlignment.center,
children: [
- const Icon(Icons.warning_outlined,
+ const Icon(
+ Icons.warning_outlined,
color: Colors.yellow,
size: 48,
),
const SizedBox(height: 8),
Text(
'permission_onboarding_permission_limited',
- style: Theme.of(context).textTheme.titleMedium,
+ style: context.textTheme.titleMedium,
textAlign: TextAlign.center,
).tr(),
const SizedBox(height: 18),
@@ -109,7 +98,7 @@ class PermissionOnboardingPage extends HookConsumerWidget {
),
const SizedBox(height: 8.0),
TextButton(
- onPressed: () => goToHome(),
+ onPressed: () => goToBackup(),
child: const Text(
'permission_onboarding_continue_anyway',
).tr(),
@@ -123,14 +112,15 @@ class PermissionOnboardingPage extends HookConsumerWidget {
crossAxisAlignment: CrossAxisAlignment.center,
mainAxisAlignment: MainAxisAlignment.center,
children: [
- const Icon(Icons.warning_outlined,
+ const Icon(
+ Icons.warning_outlined,
color: Colors.red,
size: 48,
),
const SizedBox(height: 8),
Text(
'permission_onboarding_permission_denied',
- style: Theme.of(context).textTheme.titleMedium,
+ style: context.textTheme.titleMedium,
textAlign: TextAlign.center,
).tr(),
const SizedBox(height: 18),
@@ -183,16 +173,10 @@ class PermissionOnboardingPage extends HookConsumerWidget {
),
),
TextButton(
- child: const Text('permission_onboarding_log_out').tr(),
- onPressed: () {
- ref.read(authenticationProvider.notifier).logout();
- AutoRouter.of(context).replace(
- const LoginRoute(),
- );
- },
+ child: const Text('permission_onboarding_back').tr(),
+ onPressed: () => context.autoPop(),
),
],
-
),
),
),
diff --git a/mobile/lib/modules/partner/providers/partner.provider.dart b/mobile/lib/modules/partner/providers/partner.provider.dart
index d48435616..3123382bb 100644
--- a/mobile/lib/modules/partner/providers/partner.provider.dart
+++ b/mobile/lib/modules/partner/providers/partner.provider.dart
@@ -2,21 +2,31 @@ import 'dart:async';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/modules/album/providers/suggested_shared_users.provider.dart';
+import 'package:immich_mobile/modules/partner/services/partner.service.dart';
import 'package:immich_mobile/shared/models/user.dart';
import 'package:immich_mobile/shared/providers/db.provider.dart';
import 'package:isar/isar.dart';
class PartnerSharedWithNotifier extends StateNotifier> {
- PartnerSharedWithNotifier(Isar db) : super([]) {
+ PartnerSharedWithNotifier(Isar db, this._ps) : super([]) {
final query = db.users.filter().isPartnerSharedWithEqualTo(true);
query.findAll().then((partners) => state = partners);
query.watch().listen((partners) => state = partners);
}
+
+ Future updatePartner(User partner, {required bool inTimeline}) {
+ return _ps.updatePartner(partner, inTimeline: inTimeline);
+ }
+
+ final PartnerService _ps;
}
final partnerSharedWithProvider =
StateNotifierProvider>((ref) {
- return PartnerSharedWithNotifier(ref.watch(dbProvider));
+ return PartnerSharedWithNotifier(
+ ref.watch(dbProvider),
+ ref.watch(partnerServiceProvider),
+ );
});
class PartnerSharedByNotifier extends StateNotifier> {
diff --git a/mobile/lib/modules/partner/services/partner.service.dart b/mobile/lib/modules/partner/services/partner.service.dart
index 42fcdb438..32e500353 100644
--- a/mobile/lib/modules/partner/services/partner.service.dart
+++ b/mobile/lib/modules/partner/services/partner.service.dart
@@ -5,6 +5,7 @@ import 'package:immich_mobile/shared/providers/db.provider.dart';
import 'package:immich_mobile/shared/services/api.service.dart';
import 'package:isar/isar.dart';
import 'package:logging/logging.dart';
+import 'package:openapi/api.dart';
final partnerServiceProvider = Provider(
(ref) => PartnerService(
@@ -36,7 +37,7 @@ class PartnerService {
final userDtos =
await _apiService.partnerApi.getPartners(direction._value);
if (userDtos != null) {
- return userDtos.map((u) => User.fromDto(u)).toList();
+ return userDtos.map((u) => User.fromPartnerDto(u)).toList();
}
} catch (e) {
_log.warning("failed to get partners for direction $direction:\n$e");
@@ -69,4 +70,19 @@ class PartnerService {
}
return false;
}
+
+ Future updatePartner(User partner, {required bool inTimeline}) async {
+ try {
+ final dto = await _apiService.partnerApi
+ .updatePartner(partner.id, UpdatePartnerDto(inTimeline: inTimeline));
+ if (dto != null) {
+ partner.inTimeline = dto.inTimeline ?? partner.inTimeline;
+ await _db.writeTxn(() => _db.users.put(partner));
+ return true;
+ }
+ } catch (e) {
+ _log.warning("failed to update partner ${partner.id}:\n$e");
+ }
+ return false;
+ }
}
diff --git a/mobile/lib/modules/partner/ui/partner_list.dart b/mobile/lib/modules/partner/ui/partner_list.dart
index 64db045b1..6cf330509 100644
--- a/mobile/lib/modules/partner/ui/partner_list.dart
+++ b/mobile/lib/modules/partner/ui/partner_list.dart
@@ -1,6 +1,6 @@
-import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/routing/router.dart';
import 'package:immich_mobile/shared/models/user.dart';
import 'package:immich_mobile/shared/ui/user_avatar.dart';
@@ -21,17 +21,22 @@ class PartnerList extends HookConsumerWidget {
Widget listEntry(BuildContext context, int index) {
final User p = partner[index];
return ListTile(
- contentPadding: const EdgeInsets.symmetric(horizontal: 12.0),
- leading: userAvatar(context, p, radius: 30),
+ contentPadding: const EdgeInsets.only(
+ left: 12.0,
+ right: 18.0,
+ ),
+ leading: userAvatar(context, p, radius: 24),
title: Text(
- "${p.firstName} ${p.lastName}'s photos",
- style: TextStyle(
- fontWeight: FontWeight.bold,
- fontSize: 14,
- color: Theme.of(context).primaryColor,
+ "${p.name}'s photos",
+ style: context.textTheme.labelLarge,
+ ),
+ trailing: Text(
+ "View all",
+ style: context.textTheme.labelLarge?.copyWith(
+ color: context.primaryColor,
),
),
- onTap: () => AutoRouter.of(context).push(PartnerDetailRoute(partner: p)),
+ onTap: () => context.autoPush((PartnerDetailRoute(partner: p))),
);
}
}
diff --git a/mobile/lib/modules/partner/views/partner_detail_page.dart b/mobile/lib/modules/partner/views/partner_detail_page.dart
index 995eebd3f..28d53646d 100644
--- a/mobile/lib/modules/partner/views/partner_detail_page.dart
+++ b/mobile/lib/modules/partner/views/partner_detail_page.dart
@@ -1,10 +1,12 @@
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/immich_asset_grid.dart';
+import 'package:immich_mobile/modules/partner/providers/partner.provider.dart';
import 'package:immich_mobile/shared/models/user.dart';
import 'package:immich_mobile/shared/providers/asset.provider.dart';
-import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
+import 'package:immich_mobile/shared/ui/immich_toast.dart';
class PartnerDetailPage extends HookConsumerWidget {
const PartnerDetailPage({Key? key, required this.partner}) : super(key: key);
@@ -14,6 +16,8 @@ class PartnerDetailPage extends HookConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
final assets = ref.watch(assetsProvider(partner.isarId));
+ final inTimeline = useState(partner.inTimeline);
+ bool toggleInProcess = false;
useEffect(
() {
@@ -23,18 +27,56 @@ class PartnerDetailPage extends HookConsumerWidget {
[],
);
+ void toggleInTimeline() async {
+ if (toggleInProcess) return;
+ toggleInProcess = true;
+ try {
+ final ok = await ref
+ .read(partnerSharedWithProvider.notifier)
+ .updatePartner(partner, inTimeline: !inTimeline.value);
+ if (ok) {
+ inTimeline.value = !inTimeline.value;
+ final action = inTimeline.value ? "shown on" : "hidden from";
+ ImmichToast.show(
+ context: context,
+ toastType: ToastType.success,
+ durationInSecond: 1,
+ msg: "${partner.name}'s assets $action your timeline",
+ );
+ } else {
+ ImmichToast.show(
+ context: context,
+ toastType: ToastType.error,
+ durationInSecond: 1,
+ msg: "Failed to toggle the timeline setting",
+ );
+ }
+ } finally {
+ toggleInProcess = false;
+ }
+ }
+
return Scaffold(
appBar: AppBar(
- title: Text("${partner.firstName} ${partner.lastName}"),
+ title: Text(partner.name),
elevation: 0,
centerTitle: false,
+ actions: [
+ IconButton(
+ onPressed: toggleInTimeline,
+ icon: Icon(
+ inTimeline.value ? Icons.collections : Icons.collections_outlined,
+ ),
+ tooltip: "Show/hide photos on your main timeline",
+ ),
+ ],
),
- body: assets.when(
- data: (renderList) => renderList.isEmpty
+ body: assets.widgetWhen(
+ onData: (renderList) => renderList.isEmpty
? Padding(
padding: const EdgeInsets.all(16),
child: Text(
- "It seems ${partner.firstName} does not have any photos...\n"
+ "It seems ${partner.name} does not have any photos...\n"
"Or your server version does not match the app version."),
)
: ImmichAssetGrid(
@@ -42,8 +84,6 @@ class PartnerDetailPage extends HookConsumerWidget {
onRefresh: () =>
ref.read(assetProvider.notifier).getPartnerAssets(partner),
),
- error: (e, _) => Text("Error loading partners:\n$e"),
- loading: () => const Center(child: ImmichLoadingIndicator()),
),
);
}
diff --git a/mobile/lib/modules/partner/views/partner_page.dart b/mobile/lib/modules/partner/views/partner_page.dart
index 61c639746..e90250df5 100644
--- a/mobile/lib/modules/partner/views/partner_page.dart
+++ b/mobile/lib/modules/partner/views/partner_page.dart
@@ -1,6 +1,7 @@
import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/partner/providers/partner.provider.dart';
import 'package:immich_mobile/modules/partner/services/partner.service.dart';
import 'package:immich_mobile/shared/models/user.dart';
@@ -34,14 +35,14 @@ class PartnerPage extends HookConsumerWidget {
children: [
for (User u in users)
SimpleDialogOption(
- onPressed: () => Navigator.pop(context, u),
+ onPressed: () => context.pop(u),
child: Row(
children: [
Padding(
padding: const EdgeInsets.only(right: 8),
child: userAvatar(context, u),
),
- Text("${u.firstName} ${u.lastName}"),
+ Text(u.name),
],
),
),
@@ -70,8 +71,7 @@ class PartnerPage extends HookConsumerWidget {
builder: (BuildContext context) {
return ConfirmDialog(
title: "partner_page_stop_sharing_title",
- content:
- "partner_page_stop_sharing_content".tr(args: [u.firstName]),
+ content: "partner_page_stop_sharing_content".tr(args: [u.name]),
onOk: () => ref.read(partnerServiceProvider).removePartner(u),
);
},
@@ -118,6 +118,7 @@ class PartnerPage extends HookConsumerWidget {
Padding(
padding: const EdgeInsets.symmetric(horizontal: 16.0),
child: Column(
+ crossAxisAlignment: CrossAxisAlignment.start,
children: [
Padding(
padding: const EdgeInsets.symmetric(vertical: 8),
@@ -126,12 +127,15 @@ class PartnerPage extends HookConsumerWidget {
style: TextStyle(fontSize: 14),
).tr(),
),
- ElevatedButton.icon(
- onPressed: availableUsers.whenOrNull(
- data: (data) => addNewUsersHandler,
+ Align(
+ alignment: Alignment.center,
+ child: ElevatedButton.icon(
+ onPressed: availableUsers.whenOrNull(
+ data: (data) => addNewUsersHandler,
+ ),
+ icon: const Icon(Icons.person_add),
+ label: const Text("partner_page_add_partner").tr(),
),
- icon: const Icon(Icons.person_add),
- label: const Text("partner_page_add_partner").tr(),
),
],
),
diff --git a/mobile/lib/modules/search/providers/people.provider.dart b/mobile/lib/modules/search/providers/people.provider.dart
index e40ff3fc8..6009ee53a 100644
--- a/mobile/lib/modules/search/providers/people.provider.dart
+++ b/mobile/lib/modules/search/providers/people.provider.dart
@@ -1,44 +1,51 @@
-import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/modules/home/ui/asset_grid/asset_grid_data_structure.dart';
+import 'package:immich_mobile/modules/search/models/curated_content.dart';
import 'package:immich_mobile/modules/search/services/person.service.dart';
-import 'package:openapi/api.dart';
+import 'package:immich_mobile/modules/settings/providers/app_settings.provider.dart';
+import 'package:immich_mobile/modules/settings/services/app_settings.service.dart';
+import 'package:riverpod_annotation/riverpod_annotation.dart';
-final personAssetsProvider = FutureProvider.family
- .autoDispose((ref, personId) async {
- final PersonService personService = ref.watch(personServiceProvider);
+part 'people.provider.g.dart';
+@riverpod
+Future> getCuratedPeople(
+ GetCuratedPeopleRef ref,
+) async {
+ final PersonService personService = ref.read(personServiceProvider);
+
+ final curatedPeople = await personService.getCuratedPeople();
+
+ return curatedPeople
+ .map((p) => CuratedContent(id: p.id, label: p.name))
+ .toList();
+}
+
+@riverpod
+Future personAssets(PersonAssetsRef ref, String personId) async {
+ final PersonService personService = ref.read(personServiceProvider);
final assets = await personService.getPersonAssets(personId);
-
if (assets == null) {
return RenderList.empty();
}
- return RenderList.fromAssets(assets, GroupAssetsBy.auto);
-});
-
-final getCuratedPeopleProvider =
- FutureProvider.autoDispose>((ref) async {
- final PersonService personService = ref.watch(personServiceProvider);
-
- final curatedPeople = await personService.getCuratedPeople();
-
- return curatedPeople ?? [];
-});
-
-class UpdatePersonName {
- final String id;
- final String name;
-
- UpdatePersonName(this.id, this.name);
+ final settings = ref.read(appSettingsServiceProvider);
+ final groupBy =
+ GroupAssetsBy.values[settings.getSetting(AppSettingsEnum.groupAssetsBy)];
+ return await RenderList.fromAssets(assets, groupBy);
}
-final updatePersonNameProvider =
- StateProvider.family((ref, dto) async {
- final PersonService personService = ref.watch(personServiceProvider);
+@riverpod
+Future updatePersonName(
+ UpdatePersonNameRef ref,
+ String personId,
+ String updatedName,
+) async {
+ final PersonService personService = ref.read(personServiceProvider);
+ final person = await personService.updateName(personId, updatedName);
- final person = await personService.updateName(dto.id, dto.name);
-
- if (person != null && person.name == dto.name) {
+ if (person != null && person.name == updatedName) {
ref.invalidate(getCuratedPeopleProvider);
+ return true;
}
-});
+ return false;
+}
diff --git a/mobile/lib/modules/search/providers/people.provider.g.dart b/mobile/lib/modules/search/providers/people.provider.g.dart
new file mode 100644
index 000000000..c13c2c160
--- /dev/null
+++ b/mobile/lib/modules/search/providers/people.provider.g.dart
@@ -0,0 +1,320 @@
+// GENERATED CODE - DO NOT MODIFY BY HAND
+
+part of 'people.provider.dart';
+
+// **************************************************************************
+// RiverpodGenerator
+// **************************************************************************
+
+String _$getCuratedPeopleHash() => r'2a534553812abe69abce2c2e41aa62b8de16e9d0';
+
+/// See also [getCuratedPeople].
+@ProviderFor(getCuratedPeople)
+final getCuratedPeopleProvider =
+ AutoDisposeFutureProvider>.internal(
+ getCuratedPeople,
+ name: r'getCuratedPeopleProvider',
+ debugGetCreateSourceHash: const bool.fromEnvironment('dart.vm.product')
+ ? null
+ : _$getCuratedPeopleHash,
+ dependencies: null,
+ allTransitiveDependencies: null,
+);
+
+typedef GetCuratedPeopleRef
+ = AutoDisposeFutureProviderRef>;
+String _$personAssetsHash() => r'1d6eff5ca3aa630b58c4dad9516193b21896984d';
+
+/// Copied from Dart SDK
+class _SystemHash {
+ _SystemHash._();
+
+ static int combine(int hash, int value) {
+ // ignore: parameter_assignments
+ hash = 0x1fffffff & (hash + value);
+ // ignore: parameter_assignments
+ hash = 0x1fffffff & (hash + ((0x0007ffff & hash) << 10));
+ return hash ^ (hash >> 6);
+ }
+
+ static int finish(int hash) {
+ // ignore: parameter_assignments
+ hash = 0x1fffffff & (hash + ((0x03ffffff & hash) << 3));
+ // ignore: parameter_assignments
+ hash = hash ^ (hash >> 11);
+ return 0x1fffffff & (hash + ((0x00003fff & hash) << 15));
+ }
+}
+
+/// See also [personAssets].
+@ProviderFor(personAssets)
+const personAssetsProvider = PersonAssetsFamily();
+
+/// See also [personAssets].
+class PersonAssetsFamily extends Family> {
+ /// See also [personAssets].
+ const PersonAssetsFamily();
+
+ /// See also [personAssets].
+ PersonAssetsProvider call(
+ String personId,
+ ) {
+ return PersonAssetsProvider(
+ personId,
+ );
+ }
+
+ @override
+ PersonAssetsProvider getProviderOverride(
+ covariant PersonAssetsProvider provider,
+ ) {
+ return call(
+ provider.personId,
+ );
+ }
+
+ static const Iterable? _dependencies = null;
+
+ @override
+ Iterable? get dependencies => _dependencies;
+
+ static const Iterable? _allTransitiveDependencies = null;
+
+ @override
+ Iterable? get allTransitiveDependencies =>
+ _allTransitiveDependencies;
+
+ @override
+ String? get name => r'personAssetsProvider';
+}
+
+/// See also [personAssets].
+class PersonAssetsProvider extends AutoDisposeFutureProvider {
+ /// See also [personAssets].
+ PersonAssetsProvider(
+ String personId,
+ ) : this._internal(
+ (ref) => personAssets(
+ ref as PersonAssetsRef,
+ personId,
+ ),
+ from: personAssetsProvider,
+ name: r'personAssetsProvider',
+ debugGetCreateSourceHash:
+ const bool.fromEnvironment('dart.vm.product')
+ ? null
+ : _$personAssetsHash,
+ dependencies: PersonAssetsFamily._dependencies,
+ allTransitiveDependencies:
+ PersonAssetsFamily._allTransitiveDependencies,
+ personId: personId,
+ );
+
+ PersonAssetsProvider._internal(
+ super._createNotifier, {
+ required super.name,
+ required super.dependencies,
+ required super.allTransitiveDependencies,
+ required super.debugGetCreateSourceHash,
+ required super.from,
+ required this.personId,
+ }) : super.internal();
+
+ final String personId;
+
+ @override
+ Override overrideWith(
+ FutureOr Function(PersonAssetsRef provider) create,
+ ) {
+ return ProviderOverride(
+ origin: this,
+ override: PersonAssetsProvider._internal(
+ (ref) => create(ref as PersonAssetsRef),
+ from: from,
+ name: null,
+ dependencies: null,
+ allTransitiveDependencies: null,
+ debugGetCreateSourceHash: null,
+ personId: personId,
+ ),
+ );
+ }
+
+ @override
+ AutoDisposeFutureProviderElement createElement() {
+ return _PersonAssetsProviderElement(this);
+ }
+
+ @override
+ bool operator ==(Object other) {
+ return other is PersonAssetsProvider && other.personId == personId;
+ }
+
+ @override
+ int get hashCode {
+ var hash = _SystemHash.combine(0, runtimeType.hashCode);
+ hash = _SystemHash.combine(hash, personId.hashCode);
+
+ return _SystemHash.finish(hash);
+ }
+}
+
+mixin PersonAssetsRef on AutoDisposeFutureProviderRef {
+ /// The parameter `personId` of this provider.
+ String get personId;
+}
+
+class _PersonAssetsProviderElement
+ extends AutoDisposeFutureProviderElement with PersonAssetsRef {
+ _PersonAssetsProviderElement(super.provider);
+
+ @override
+ String get personId => (origin as PersonAssetsProvider).personId;
+}
+
+String _$updatePersonNameHash() => r'c7179a7cc558669c3b30b03fbca7782a42f2b6fd';
+
+/// See also [updatePersonName].
+@ProviderFor(updatePersonName)
+const updatePersonNameProvider = UpdatePersonNameFamily();
+
+/// See also [updatePersonName].
+class UpdatePersonNameFamily extends Family> {
+ /// See also [updatePersonName].
+ const UpdatePersonNameFamily();
+
+ /// See also [updatePersonName].
+ UpdatePersonNameProvider call(
+ String personId,
+ String updatedName,
+ ) {
+ return UpdatePersonNameProvider(
+ personId,
+ updatedName,
+ );
+ }
+
+ @override
+ UpdatePersonNameProvider getProviderOverride(
+ covariant UpdatePersonNameProvider provider,
+ ) {
+ return call(
+ provider.personId,
+ provider.updatedName,
+ );
+ }
+
+ static const Iterable? _dependencies = null;
+
+ @override
+ Iterable? get dependencies => _dependencies;
+
+ static const Iterable? _allTransitiveDependencies = null;
+
+ @override
+ Iterable? get allTransitiveDependencies =>
+ _allTransitiveDependencies;
+
+ @override
+ String? get name => r'updatePersonNameProvider';
+}
+
+/// See also [updatePersonName].
+class UpdatePersonNameProvider extends AutoDisposeFutureProvider {
+ /// See also [updatePersonName].
+ UpdatePersonNameProvider(
+ String personId,
+ String updatedName,
+ ) : this._internal(
+ (ref) => updatePersonName(
+ ref as UpdatePersonNameRef,
+ personId,
+ updatedName,
+ ),
+ from: updatePersonNameProvider,
+ name: r'updatePersonNameProvider',
+ debugGetCreateSourceHash:
+ const bool.fromEnvironment('dart.vm.product')
+ ? null
+ : _$updatePersonNameHash,
+ dependencies: UpdatePersonNameFamily._dependencies,
+ allTransitiveDependencies:
+ UpdatePersonNameFamily._allTransitiveDependencies,
+ personId: personId,
+ updatedName: updatedName,
+ );
+
+ UpdatePersonNameProvider._internal(
+ super._createNotifier, {
+ required super.name,
+ required super.dependencies,
+ required super.allTransitiveDependencies,
+ required super.debugGetCreateSourceHash,
+ required super.from,
+ required this.personId,
+ required this.updatedName,
+ }) : super.internal();
+
+ final String personId;
+ final String updatedName;
+
+ @override
+ Override overrideWith(
+ FutureOr Function(UpdatePersonNameRef provider) create,
+ ) {
+ return ProviderOverride(
+ origin: this,
+ override: UpdatePersonNameProvider._internal(
+ (ref) => create(ref as UpdatePersonNameRef),
+ from: from,
+ name: null,
+ dependencies: null,
+ allTransitiveDependencies: null,
+ debugGetCreateSourceHash: null,
+ personId: personId,
+ updatedName: updatedName,
+ ),
+ );
+ }
+
+ @override
+ AutoDisposeFutureProviderElement createElement() {
+ return _UpdatePersonNameProviderElement(this);
+ }
+
+ @override
+ bool operator ==(Object other) {
+ return other is UpdatePersonNameProvider &&
+ other.personId == personId &&
+ other.updatedName == updatedName;
+ }
+
+ @override
+ int get hashCode {
+ var hash = _SystemHash.combine(0, runtimeType.hashCode);
+ hash = _SystemHash.combine(hash, personId.hashCode);
+ hash = _SystemHash.combine(hash, updatedName.hashCode);
+
+ return _SystemHash.finish(hash);
+ }
+}
+
+mixin UpdatePersonNameRef on AutoDisposeFutureProviderRef {
+ /// The parameter `personId` of this provider.
+ String get personId;
+
+ /// The parameter `updatedName` of this provider.
+ String get updatedName;
+}
+
+class _UpdatePersonNameProviderElement
+ extends AutoDisposeFutureProviderElement with UpdatePersonNameRef {
+ _UpdatePersonNameProviderElement(super.provider);
+
+ @override
+ String get personId => (origin as UpdatePersonNameProvider).personId;
+ @override
+ String get updatedName => (origin as UpdatePersonNameProvider).updatedName;
+}
+// ignore_for_file: type=lint
+// ignore_for_file: subtype_of_sealed_class, invalid_use_of_internal_member, invalid_use_of_visible_for_testing_member
diff --git a/mobile/lib/modules/search/services/person.service.dart b/mobile/lib/modules/search/services/person.service.dart
index 8314ed109..d4cbe0de5 100644
--- a/mobile/lib/modules/search/services/person.service.dart
+++ b/mobile/lib/modules/search/services/person.service.dart
@@ -1,44 +1,40 @@
-import 'package:flutter/material.dart';
-import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:immich_mobile/shared/models/asset.dart';
import 'package:immich_mobile/shared/providers/api.provider.dart';
import 'package:immich_mobile/shared/services/api.service.dart';
+import 'package:logging/logging.dart';
import 'package:openapi/api.dart';
+import 'package:riverpod_annotation/riverpod_annotation.dart';
-final personServiceProvider = Provider(
- (ref) => PersonService(
- ref.watch(apiServiceProvider),
- ),
-);
+part 'person.service.g.dart';
+
+@riverpod
+PersonService personService(PersonServiceRef ref) =>
+ PersonService(ref.read(apiServiceProvider));
class PersonService {
+ final Logger _log = Logger("PersonService");
final ApiService _apiService;
PersonService(this._apiService);
- Future?> getCuratedPeople() async {
+ Future> getCuratedPeople() async {
try {
final peopleResponseDto = await _apiService.personApi.getAllPeople();
- return peopleResponseDto?.people;
- } catch (e) {
- debugPrint("Error [getCuratedPeople] ${e.toString()}");
- return null;
+ return peopleResponseDto?.people ?? [];
+ } catch (error, stack) {
+ _log.severe("Error while fetching curated people", error, stack);
+ return [];
}
}
Future?> getPersonAssets(String id) async {
try {
final assets = await _apiService.personApi.getPersonAssets(id);
-
- if (assets == null) {
- return null;
- }
-
- return assets.map((e) => Asset.remote(e)).toList();
- } catch (e) {
- debugPrint("Error [getPersonAssets] ${e.toString()}");
- return null;
+ return assets?.map((e) => Asset.remote(e)).toList();
+ } catch (error, stack) {
+ _log.severe("Error while fetching person assets", error, stack);
}
+ return null;
}
Future updateName(String id, String name) async {
@@ -49,9 +45,9 @@ class PersonService {
name: name,
),
);
- } catch (e) {
- debugPrint("Error [updateName] ${e.toString()}");
- return null;
+ } catch (error, stack) {
+ _log.severe("Error while updating person name", error, stack);
}
+ return null;
}
}
diff --git a/mobile/lib/modules/search/services/person.service.g.dart b/mobile/lib/modules/search/services/person.service.g.dart
new file mode 100644
index 000000000..e66c6c2aa
--- /dev/null
+++ b/mobile/lib/modules/search/services/person.service.g.dart
@@ -0,0 +1,25 @@
+// GENERATED CODE - DO NOT MODIFY BY HAND
+
+part of 'person.service.dart';
+
+// **************************************************************************
+// RiverpodGenerator
+// **************************************************************************
+
+String _$personServiceHash() => r'3fc3dcf4603c7b55c0deae65f39f6c212eea492b';
+
+/// See also [personService].
+@ProviderFor(personService)
+final personServiceProvider = AutoDisposeProvider.internal(
+ personService,
+ name: r'personServiceProvider',
+ debugGetCreateSourceHash: const bool.fromEnvironment('dart.vm.product')
+ ? null
+ : _$personServiceHash,
+ dependencies: null,
+ allTransitiveDependencies: null,
+);
+
+typedef PersonServiceRef = AutoDisposeProviderRef;
+// ignore_for_file: type=lint
+// ignore_for_file: subtype_of_sealed_class, invalid_use_of_internal_member, invalid_use_of_visible_for_testing_member
diff --git a/mobile/lib/modules/search/ui/curated_people_row.dart b/mobile/lib/modules/search/ui/curated_people_row.dart
index 8a65c25f7..e838c59e1 100644
--- a/mobile/lib/modules/search/ui/curated_people_row.dart
+++ b/mobile/lib/modules/search/ui/curated_people_row.dart
@@ -1,4 +1,5 @@
import 'package:flutter/material.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/search/models/curated_content.dart';
import 'package:immich_mobile/modules/search/ui/thumbnail_with_info.dart';
import 'package:immich_mobile/shared/models/store.dart';
@@ -83,9 +84,8 @@ class CuratedPeopleRow extends StatelessWidget {
padding: const EdgeInsets.only(top: 8.0),
child: Text(
"Add name",
- style: TextStyle(
- fontWeight: FontWeight.bold,
- color: Theme.of(context).primaryColor,
+ style: context.textTheme.labelLarge?.copyWith(
+ color: context.primaryColor,
),
),
),
@@ -97,10 +97,7 @@ class CuratedPeopleRow extends StatelessWidget {
person.label,
textAlign: TextAlign.center,
overflow: TextOverflow.ellipsis,
- style: const TextStyle(
- fontWeight: FontWeight.bold,
- fontSize: 13.0,
- ),
+ style: context.textTheme.labelLarge,
),
),
],
diff --git a/mobile/lib/modules/search/ui/curated_places_row.dart b/mobile/lib/modules/search/ui/curated_places_row.dart
index d62607a80..b0343f5ed 100644
--- a/mobile/lib/modules/search/ui/curated_places_row.dart
+++ b/mobile/lib/modules/search/ui/curated_places_row.dart
@@ -1,5 +1,6 @@
-import 'package:auto_route/auto_route.dart';
+import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/map/ui/map_thumbnail.dart';
import 'package:immich_mobile/modules/search/ui/curated_row.dart';
import 'package:immich_mobile/modules/search/ui/thumbnail_with_info.dart';
@@ -25,7 +26,7 @@ class CuratedPlacesRow extends CuratedRow {
final int actualContentIndex = isMapEnabled ? 1 : 0;
Widget buildMapThumbnail() {
return GestureDetector(
- onTap: () => AutoRouter.of(context).push(
+ onTap: () => context.autoPush(
const MapRoute(),
),
child: SizedBox(
@@ -43,36 +44,39 @@ class CuratedPlacesRow extends CuratedRow {
),
height: imageSize,
showAttribution: false,
- isDarkTheme: Theme.of(context).brightness == Brightness.dark,
+ isDarkTheme: context.isDarkTheme,
),
),
- Container(
- decoration: BoxDecoration(
- borderRadius: BorderRadius.circular(10),
- color: Colors.black,
- gradient: LinearGradient(
- begin: FractionalOffset.topCenter,
- end: FractionalOffset.bottomCenter,
- colors: [
- Colors.blueGrey.withOpacity(0.0),
- Colors.black.withOpacity(0.4),
- ],
- stops: const [0.0, 1.0],
+ Padding(
+ padding: const EdgeInsets.only(right: 10.0),
+ child: Container(
+ decoration: BoxDecoration(
+ borderRadius: BorderRadius.circular(10),
+ color: Colors.black,
+ gradient: LinearGradient(
+ begin: FractionalOffset.topCenter,
+ end: FractionalOffset.bottomCenter,
+ colors: [
+ Colors.blueGrey.withOpacity(0.0),
+ Colors.black.withOpacity(0.4),
+ ],
+ stops: const [0.0, 0.4],
+ ),
),
),
),
- const Align(
+ Align(
alignment: Alignment.bottomCenter,
child: Padding(
- padding: EdgeInsets.only(bottom: 10),
- child: Text(
- "Your Map",
+ padding: const EdgeInsets.only(bottom: 10),
+ child: const Text(
+ "search_page_your_map",
style: TextStyle(
color: Colors.white,
fontWeight: FontWeight.bold,
fontSize: 14,
),
- ),
+ ).tr(),
),
),
],
diff --git a/mobile/lib/modules/search/ui/explore_grid.dart b/mobile/lib/modules/search/ui/explore_grid.dart
index 12f8ec6a1..984f65a40 100644
--- a/mobile/lib/modules/search/ui/explore_grid.dart
+++ b/mobile/lib/modules/search/ui/explore_grid.dart
@@ -1,5 +1,5 @@
-import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/search/models/curated_content.dart';
import 'package:immich_mobile/modules/search/ui/thumbnail_with_info.dart';
import 'package:immich_mobile/routing/router.dart';
@@ -50,13 +50,13 @@ class ExploreGrid extends StatelessWidget {
borderRadius: 0,
onTap: () {
isPeople
- ? AutoRouter.of(context).push(
+ ? context.autoPush(
PersonResultRoute(
personId: content.id,
personName: content.label,
),
)
- : AutoRouter.of(context).push(
+ : context.autoPush(
SearchResultRoute(searchTerm: 'm:${content.label}'),
);
},
diff --git a/mobile/lib/modules/search/ui/immich_search_bar.dart b/mobile/lib/modules/search/ui/immich_search_bar.dart
index 5ab4bc964..b3275237f 100644
--- a/mobile/lib/modules/search/ui/immich_search_bar.dart
+++ b/mobile/lib/modules/search/ui/immich_search_bar.dart
@@ -2,6 +2,7 @@ import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/search/providers/search_page_state.provider.dart';
class ImmichSearchBar extends HookConsumerWidget
@@ -57,11 +58,9 @@ class ImmichSearchBar extends HookConsumerWidget
},
decoration: InputDecoration(
hintText: 'search_bar_hint'.tr(),
- hintStyle: Theme.of(context).textTheme.titleSmall?.copyWith(
- color: Theme.of(context).colorScheme.onSurface.withOpacity(0.5),
- fontWeight: FontWeight.w500,
- fontSize: 14,
- ),
+ hintStyle: context.textTheme.bodyLarge?.copyWith(
+ color: context.themeData.colorScheme.onSurface.withOpacity(0.75),
+ ),
enabledBorder: const UnderlineInputBorder(
borderSide: BorderSide(color: Colors.transparent),
),
diff --git a/mobile/lib/modules/search/ui/person_name_edit_form.dart b/mobile/lib/modules/search/ui/person_name_edit_form.dart
index f60824b5f..e32d4a9e0 100644
--- a/mobile/lib/modules/search/ui/person_name_edit_form.dart
+++ b/mobile/lib/modules/search/ui/person_name_edit_form.dart
@@ -1,6 +1,8 @@
+import 'package:easy_localization/easy_localization.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
+import 'package:immich_mobile/extensions/build_context_extensions.dart';
import 'package:immich_mobile/modules/search/providers/people.provider.dart';
class PersonNameEditFormResult {
@@ -23,58 +25,55 @@ class PersonNameEditForm extends HookConsumerWidget {
@override
Widget build(BuildContext context, WidgetRef ref) {
final controller = useTextEditingController(text: personName);
+ final isError = useState(false);
return AlertDialog(
title: const Text(
- "Add a name",
+ "search_page_person_add_name_dialog_title",
style: TextStyle(fontWeight: FontWeight.bold),
- ),
+ ).tr(),
content: SingleChildScrollView(
child: TextFormField(
controller: controller,
autofocus: true,
- decoration: const InputDecoration(
- hintText: 'Name',
+ decoration: InputDecoration(
+ hintText: 'search_page_person_add_name_dialog_hint'.tr(),
+ border: const OutlineInputBorder(),
+ errorText: isError.value ? 'Error occured' : null,
),
),
),
actions: [
TextButton(
- style: TextButton.styleFrom(),
- onPressed: () {
- Navigator.of(context, rootNavigator: true)
- .pop(
- PersonNameEditFormResult(false, ''),
- );
- },
+ onPressed: () => context.pop(
+ PersonNameEditFormResult(false, ''),
+ ),
child: Text(
- "Cancel",
+ "search_page_person_add_name_dialog_cancel",
style: TextStyle(
color: Colors.red[300],
fontWeight: FontWeight.bold,
),
- ),
+ ).tr(),
),
TextButton(
- onPressed: () {
- ref.read(
- updatePersonNameProvider(
- UpdatePersonName(personId, controller.text),
- ),
- );
-
- Navigator.of(context, rootNavigator: true)
- .pop