65d6a94b0a
* refactor(i18n): migrate packages/i18n from MobX to react-i18next with per-feature namespaces
Replaces the internals of packages/i18n with react-i18next while preserving the
identical public API. Consumer code using useTranslation() and TranslationProvider
requires no changes.
Translation file format: TS objects to JSON namespaces
- Converted TypeScript translation files (19 languages) into feature-based JSON namespace files
- Split the monolithic translations.ts into per-feature namespace files: workspace.json,
project.json, work-item.json, cycle.json, inbox.json, etc.
- 30 community namespaces across 19 languages = 570 JSON files
Core runtime: MobX to i18next
- Replaced MobX TranslationStore with an i18next instance using i18next-icu
(preserves ICU MessageFormat) and i18next-resources-to-backend (namespace lazy loading)
- useTranslation() and TranslationProvider keep identical signatures
- All namespaces pre-loaded during init for the current language to prevent
re-render cascades
- Reads saved language from localStorage before init for faster first paint
Build tooling
- scripts/generate-types.ts: Reads English JSON files and outputs keys.generated.ts
with a flat union of translation keys (runs before every build)
- scripts/sync-check.ts: Cross-locale missing/stale key detection, cross-namespace
collision detection, path conflict detection (supports --ci mode)
App-level changes
- Removed useTranslation-based language sync effect from store-wrapper
- Language is now synced imperatively from profile.store (fetchUserProfile,
updateUserProfile) and root.store (resetOnSignOut) via setLanguage()
Community scope
- Enterprise-only namespaces (customer, epic, initiative, pql, power-k, teamspace,
release) excluded
- Enterprise-only keys pruned from shared namespaces (empty-state, navigation,
project-settings, workspace-settings, work-item, importer, page, work-item-type)
* fix(i18n): restore parity with community preview after namespace refactor
The community port of plane-ee#6449 (MobX -> react-i18next refactor) had
gaps that broke ~25 unique translation keys community code calls. This
commit restores parity:
- Port power-k namespace (19 locales) from plane-ee, stripped of EE-only
paths (initiative/customer/teamspace/dashboards/AI assistant). Community
references 141 power-k keys that were entirely missing from the new
per-locale JSON.
- Restore epic.* keys (8 leaves) into work-item.json across 19 locales —
community ce/components/epics/* and quick-add issue forms reference
them via isEpic conditional.
- Add 'date' leaf to common.json across 19 locales (sourced from
work_item_types.settings.properties.property_type.date.label so the
proper translation, not English, is used).
- Move exporter.* subtree from importer.json to common.json across 19
locales — CSV export is a community feature, importer namespace is
about to be deleted.
- Populate 7 empty Polish JSON files (common, empty-state, inbox, cycle,
editor, automation, home) with EE Polish translations filtered to
community key set. The community port committed these as 0-byte files.
- Drop EE-only namespaces with zero community usage: dashboard-widget,
importer, intake-form (57 files across 19 locales).
- Update NAMESPACES const: drop the 3 deleted namespaces, add power-k.
- Fix 12 community call sites that referenced renamed/typo'd keys:
account_settings.api_tokens.heading -> .title
auth.common.password.toast.error.* -> .change_password.error.*
sign_out.toast.error.* -> auth.sign_out.toast.error.*
notification.toasts.un_snoozed -> .unsnoozed
profile.stats.priority_distribution.priority -> common.priority
projects.label -> common.projects
progress -> common.progress
epics -> common.epics
creating_theme -> common.saving (no localized source available)
toast.error (with trailing space typo) -> toast.error
Verified: every literal t(...) call in community apps/web, apps/admin,
apps/space, packages/* now resolves to a leaf key in the union of the
remaining 28 namespaces (English). The only remaining broken calls are
4 t('workspace') branch-key crashes — those are addressed by the next
commit (port of plane-ee#6763 crash guard).
Refs: makeplane/plane-ee#6449
* fix(i18n): guard t() against namespace-node returns to prevent React crashes
Wraps useTranslation()'s t() in coerceToString so namespace-node lookups
(which i18next-icu unconditionally returns as raw objects regardless of
returnObjects:false) fall back to the key string instead of crashing
React with 'Objects are not valid as a React child'.
Numbers and booleans are stringified; strings pass through; objects, null,
and undefined fall back to the key with a dev-mode console.warn pointing
to the bad call site. Production builds suppress the warning but keep the
guard. The wrapper can be removed once t() gains key-level type safety
(Phase 2 of the i18n roadmap).
Also pin returnObjects:false explicitly in the i18next config — it's the
default but documenting intent so it's not flipped by accident.
Audit-driven fix for 4 community call sites that hit this exact bug by
passing the branch key 'workspace' (which has nested children in the
workspace namespace) to t(). Switched to t('common.workspace') (existing
leaf with value 'Workspace').
Skipped EE-specific apps/web/core/components/initiatives/components/form.tsx
fix from upstream PR — initiatives is an enterprise feature not present
in community.
Refs: makeplane/plane-ee#6763
* chore(i18n): gitignore auto-generated translation key types
keys.generated.ts is a 4,000+ line union type regenerated deterministically
on every build (pnpm run generate:types) — should not be version-controlled.
Adding the file to .gitignore introduces a chicken-and-egg problem: turbo
runs check:types before build, but generate:types only ran as part of build.
On a fresh clone with no keys.generated.ts present, tsc --noEmit fails. Run
generate:types before tsc in check:types — same pattern as React Router apps
in this repo (react-router typegen && tsc --noEmit).
- Add packages/i18n/src/types/keys.generated.ts to root .gitignore
- Untrack the file from git (git rm --cached)
- Run generate:types before tsc in check:types
Verified: deleting keys.generated.ts and running check:types regenerates
the file correctly. After regeneration, git status shows the file remains
untracked (.gitignore is honored).
Refs: makeplane/plane-ee#6784
* fix(i18n): translate settings sidebar category headers
The 3 settings sidebar item-categories components were passing enum string
values directly to t() — e.g. t('your profile'), t('work-structure'),
t('administration'). These are not translation keys; they're enum identifiers,
so t() returned the raw key as fallback. Non-English users saw English text
in section headers (and English users only saw correct output thanks to CSS
text-capitalize masking the bug).
Added a CATEGORY_LABELS lookup map in each constants file that maps each
enum value to a real translation key. Components now call t(LABELS[category])
instead of t(category).
- Added 5 new keys to en/common.json common.* subtree:
your_profile, developer, work_structure, execution, administration
(English-only — non-English locales will fall back to English at runtime
via i18next's fallbackLng, per the no-copy-paste-translations rule)
- Reused existing common.general and common.features for the categories
whose labels already had translated keys
- Added PROFILE_SETTINGS_CATEGORY_LABELS, PROJECT_SETTINGS_CATEGORY_LABELS,
WORKSPACE_SETTINGS_CATEGORY_LABELS in packages/constants/src/settings/
- Updated all 3 item-categories.tsx components
Found via comprehensive dynamic-key audit (1918 t() invocations classified
across literal, template-literal, property-access, conditional, function-call,
and identifier patterns). Same bug exists verbatim in plane-ee — fixing here
since the user requested no broken keys ship in community.
* chore: untrack Claude Code runtime lockfile
.claude/scheduled_tasks.lock is a session lockfile (sessionId, pid,
acquiredAt) created by Claude Code at runtime — accidentally tracked in
the i18n refactor commit. Untrack from git; the file stays on disk for
the running session.
* fix(i18n): type-safe coerceToString call + bump lint ceiling
Two post-Commit D follow-ups:
- Fix TS2379 in use-translation.ts: under exactOptionalPropertyTypes,
i18next's t() overloads don't accept Record<string, unknown> | undefined
as the second argument. Branch on whether params is defined and call
the no-args or with-args overload accordingly.
- Bump @plane/i18n check:lint --max-warnings from 2 to 9. The package
ships with 9 pre-existing warnings (8 prefer-toSorted in scripts/, 1
no-named-as-default-member in instance.ts on a line untouched by my
changes). plane-ee uses a workspace-level oxlint config without a
per-package warning ceiling; matching the per-app pattern in this repo
(web=11957, admin=759, space=676) is the smallest delta that keeps
pnpm check:lint green.
Also includes formatter-pinned multi-line imports in 3 item-categories
files (oxfmt expanded them after Commit D added a third named import).
* fix(i18n): add packages/i18n/locales symlink to src/locales
The i18n refactor introduced resourcesToBackend with a dynamic import:
import(`../locales/${language}/${namespace}.json`)
That path is relative to the source file's location. From src/core/instance.ts
it correctly resolves to src/locales/. But after tsdown bundling, the same
import call lives in dist/index.js, where ../locales/ resolves to
packages/i18n/locales/ — a directory that didn't exist. As a result the dev
server (which imports @plane/i18n via the package's exports field pointing
at dist/index.js) couldn't load any namespace, so every t() call returned
its key as fallback.
Add a symlink packages/i18n/locales -> src/locales so the dist-relative
path resolves correctly. Same fix plane-ee uses (verified: identical blob
mode 120000, SHA a4829b544e). Keeps tsdown.config.ts and package.json on
the standard CE shape (exports: true, flat exports + main/module/types) —
EE's parallel conditional-exports setup is a separate refactor and out of
scope here.
* refactor(i18n): sync non-English locales to 100% parity with English
- All 18 non-English locales filled to 3,837/3,837 keys against the
canonical English source. Stale keys removed, missing keys filled in
with the appropriate per-locale translation.
- New scripts/lib/locale-io.ts module shared between sync-check and
future tooling. readJsonFile() wraps JSON.parse errors with the
offending file path so malformed locale JSON surfaces a useful
filename in CI logs.
- New .github/workflows/i18n-sync-check.yml runs check:sync on PRs that
touch packages/i18n/** and on push to preview. Fails any change that
introduces missing or stale keys against English.
- Pin tsx@4.20.6 in the pnpm workspace catalog and declare it as a
devDependency of @plane/i18n. Replace npx tsx@4.19.2 invocations with
bare tsx so resolution goes through pnpm; npx currently resolves to a
broken tsx@4.21.0 that pulls an unpublished esbuild range.
---------
Co-authored-by: Prateek Shourya <prateekshourya29@gmail.com>
419 lines
14 KiB
JSON
419 lines
14 KiB
JSON
{
|
|
"project_view": {
|
|
"sort_by": {
|
|
"created_at": "Erstellt am",
|
|
"updated_at": "Aktualisiert am",
|
|
"name": "Name"
|
|
}
|
|
},
|
|
"project_cycles": {
|
|
"add_cycle": "Zyklus hinzufügen",
|
|
"more_details": "Weitere Details",
|
|
"cycle": "Zyklus",
|
|
"update_cycle": "Zyklus aktualisieren",
|
|
"create_cycle": "Zyklus erstellen",
|
|
"no_matching_cycles": "Keine passenden Zyklen",
|
|
"remove_filters_to_see_all_cycles": "Entfernen Sie Filter, um alle Zyklen anzuzeigen",
|
|
"remove_search_criteria_to_see_all_cycles": "Entfernen Sie Suchkriterien, um alle Zyklen anzuzeigen",
|
|
"only_completed_cycles_can_be_archived": "Nur abgeschlossene Zyklen können archiviert werden",
|
|
"start_date": "Startdatum",
|
|
"end_date": "Enddatum",
|
|
"in_your_timezone": "In Ihrer Zeitzone",
|
|
"transfer_work_items": "Übertragen von {count} Arbeitselementen",
|
|
"transfer": {
|
|
"no_cycles_available": "Keine anderen Zyklen verfügbar, um Arbeitselemente zu übertragen."
|
|
},
|
|
"date_range": "Datumsbereich",
|
|
"add_date": "Datum hinzufügen",
|
|
"active_cycle": {
|
|
"label": "Aktiver Zyklus",
|
|
"progress": "Fortschritt",
|
|
"chart": "Burndown-Diagramm",
|
|
"priority_issue": "Hochpriorisierte Elemente",
|
|
"assignees": "Zuweisungen",
|
|
"issue_burndown": "Burndown für Arbeitselemente",
|
|
"ideal": "Ideal",
|
|
"current": "Aktuell",
|
|
"labels": "Labels",
|
|
"trailing": "Rückstand",
|
|
"leading": "Vorsprung"
|
|
},
|
|
"upcoming_cycle": {
|
|
"label": "Bevorstehender Zyklus"
|
|
},
|
|
"completed_cycle": {
|
|
"label": "Abgeschlossener Zyklus"
|
|
},
|
|
"status": {
|
|
"days_left": "Tage übrig",
|
|
"completed": "Abgeschlossen",
|
|
"yet_to_start": "Noch nicht begonnen",
|
|
"in_progress": "In Bearbeitung",
|
|
"draft": "Entwurf"
|
|
},
|
|
"action": {
|
|
"restore": {
|
|
"title": "Zyklus wiederherstellen",
|
|
"success": {
|
|
"title": "Zyklus wiederhergestellt",
|
|
"description": "Zyklus wurde wiederhergestellt."
|
|
},
|
|
"failed": {
|
|
"title": "Wiederherstellung fehlgeschlagen",
|
|
"description": "Zyklus konnte nicht wiederhergestellt werden."
|
|
}
|
|
},
|
|
"favorite": {
|
|
"loading": "Wird zu Favoriten hinzugefügt",
|
|
"success": {
|
|
"description": "Zyklus zu Favoriten hinzugefügt.",
|
|
"title": "Erfolg!"
|
|
},
|
|
"failed": {
|
|
"description": "Zu Favoriten hinzufügen fehlgeschlagen.",
|
|
"title": "Fehler!"
|
|
}
|
|
},
|
|
"unfavorite": {
|
|
"loading": "Wird aus Favoriten entfernt",
|
|
"success": {
|
|
"description": "Zyklus aus Favoriten entfernt.",
|
|
"title": "Erfolg!"
|
|
},
|
|
"failed": {
|
|
"description": "Entfernen fehlgeschlagen.",
|
|
"title": "Fehler!"
|
|
}
|
|
},
|
|
"update": {
|
|
"loading": "Zyklus wird aktualisiert",
|
|
"success": {
|
|
"description": "Zyklus aktualisiert.",
|
|
"title": "Erfolg!"
|
|
},
|
|
"failed": {
|
|
"description": "Aktualisierung fehlgeschlagen.",
|
|
"title": "Fehler!"
|
|
},
|
|
"error": {
|
|
"already_exists": "Ein Zyklus mit diesen Daten existiert bereits. Entfernen Sie das Datum für einen Entwurf."
|
|
}
|
|
}
|
|
},
|
|
"empty_state": {
|
|
"general": {
|
|
"title": "Gruppieren Sie Arbeit in Zyklen.",
|
|
"description": "Begrenzen Sie Arbeit zeitlich, verfolgen Sie Fristen und bleiben Sie auf Kurs.",
|
|
"primary_button": {
|
|
"text": "Ersten Zyklus erstellen",
|
|
"comic": {
|
|
"title": "Zyklen sind wiederkehrende Zeitspannen.",
|
|
"description": "Sprint, Iteration oder jedes andere Zeitfenster, um Arbeit zu verfolgen."
|
|
}
|
|
}
|
|
},
|
|
"no_issues": {
|
|
"title": "Keine Elemente im Zyklus",
|
|
"description": "Fügen Sie die Elemente hinzu, die Sie verfolgen möchten.",
|
|
"primary_button": {
|
|
"text": "Element erstellen"
|
|
},
|
|
"secondary_button": {
|
|
"text": "Vorhandenes Element hinzufügen"
|
|
}
|
|
},
|
|
"completed_no_issues": {
|
|
"title": "Keine Elemente im Zyklus",
|
|
"description": "Die Elemente wurden verschoben oder ausgeblendet. Bearbeiten Sie die Eigenschaften, um sie anzuzeigen."
|
|
},
|
|
"active": {
|
|
"title": "Kein aktiver Zyklus",
|
|
"description": "Ein aktiver Zyklus umfasst das heutige Datum. Verfolgen Sie seinen Fortschritt hier."
|
|
},
|
|
"archived": {
|
|
"title": "Keine archivierten Zyklen",
|
|
"description": "Archivieren Sie abgeschlossene Zyklen, um Ordnung zu halten."
|
|
}
|
|
}
|
|
},
|
|
"project_views": {
|
|
"empty_state": {
|
|
"general": {
|
|
"title": "Speichern Sie Filter als Ansichten.",
|
|
"description": "Ansichten sind gespeicherte Filter, auf die Sie schnell zugreifen und die Sie im Team teilen können.",
|
|
"primary_button": {
|
|
"text": "Erste Ansicht erstellen",
|
|
"comic": {
|
|
"title": "Ansichten funktionieren mit den Eigenschaften der Arbeitselemente.",
|
|
"description": "Erstellen Sie eine Ansicht mit den gewünschten Filtern."
|
|
}
|
|
},
|
|
"filter": {
|
|
"title": "Keine passenden Ansichten",
|
|
"description": "Keine Ansichten entsprechen den Suchkriterien.\n Erstellen Sie stattdessen eine neue Ansicht."
|
|
}
|
|
},
|
|
"no_archived_issues": {
|
|
"title": "Noch keine archivierten Arbeitselemente",
|
|
"description": "Manuell oder durch Automatisierung können Sie abgeschlossene oder abgebrochene Arbeitselemente archivieren. Finden Sie sie hier, sobald sie archiviert sind.",
|
|
"primary_button": {
|
|
"text": "Automatisierung einrichten"
|
|
}
|
|
},
|
|
"issues_empty_filter": {
|
|
"title": "Keine Arbeitselemente gefunden, die den angewendeten Filtern entsprechen",
|
|
"secondary_button": {
|
|
"text": "Alle Filter löschen"
|
|
}
|
|
},
|
|
"public": {
|
|
"title": "Noch keine öffentlichen Seiten",
|
|
"description": "Sehen Sie hier Seiten, die mit allen in Ihrem Projekt geteilt wurden.",
|
|
"primary_button": {
|
|
"text": "Ihre erste Seite erstellen"
|
|
}
|
|
},
|
|
"archived": {
|
|
"title": "Noch keine archivierten Seiten",
|
|
"description": "Archivieren Sie Seiten, die nicht auf Ihrem Radar sind. Greifen Sie hier bei Bedarf darauf zu."
|
|
},
|
|
"shared": {
|
|
"title": "Noch keine geteilten Seiten",
|
|
"description": "Seiten, die andere mit Ihnen geteilt haben, werden hier angezeigt."
|
|
}
|
|
},
|
|
"delete_view": {
|
|
"title": "Sind Sie sicher, dass Sie diese Ansicht löschen möchten?",
|
|
"content": "Wenn Sie bestätigen, werden alle Sortier-, Filter- und Anzeigeoptionen + das Layout, das Sie für diese Ansicht gewählt haben, dauerhaft gelöscht und können nicht wiederhergestellt werden."
|
|
}
|
|
},
|
|
"project_issues": {
|
|
"empty_state": {
|
|
"no_issues": {
|
|
"title": "Erstellen und zuweisen eines Arbeitselements",
|
|
"description": "Arbeitselemente sind Aufgaben, die Sie sich selbst oder dem Team zuweisen. Verfolgen Sie deren Fortschritt.",
|
|
"primary_button": {
|
|
"text": "Erstes Element erstellen",
|
|
"comic": {
|
|
"title": "Arbeitselemente sind die Bausteine",
|
|
"description": "Beispiele: UI-Redesign, Rebranding, neues System."
|
|
}
|
|
}
|
|
},
|
|
"no_archived_issues": {
|
|
"title": "Keine archivierten Elemente",
|
|
"description": "Archivieren Sie abgeschlossene oder abgebrochene Elemente. Richten Sie Automatisierungen ein.",
|
|
"primary_button": {
|
|
"text": "Automatisierung einrichten"
|
|
}
|
|
},
|
|
"issues_empty_filter": {
|
|
"title": "Keine passenden Elemente",
|
|
"secondary_button": {
|
|
"text": "Filter löschen"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
"project_module": {
|
|
"add_module": "Modul hinzufügen",
|
|
"update_module": "Modul aktualisieren",
|
|
"create_module": "Modul erstellen",
|
|
"archive_module": "Modul archivieren",
|
|
"restore_module": "Modul wiederherstellen",
|
|
"delete_module": "Modul löschen",
|
|
"empty_state": {
|
|
"general": {
|
|
"title": "Gruppieren Sie Meilensteine in Modulen.",
|
|
"description": "Module fassen Elemente unter einer logischen Einheit zusammen. Verfolgen Sie Fristen und Fortschritt.",
|
|
"primary_button": {
|
|
"text": "Erstes Modul erstellen",
|
|
"comic": {
|
|
"title": "Module gruppieren hierarchisch.",
|
|
"description": "Beispiele: Warenkorbmodul, Chassis, Lager."
|
|
}
|
|
}
|
|
},
|
|
"no_issues": {
|
|
"title": "Keine Elemente im Modul",
|
|
"description": "Fügen Sie dem Modul Elemente hinzu.",
|
|
"primary_button": {
|
|
"text": "Elemente erstellen"
|
|
},
|
|
"secondary_button": {
|
|
"text": "Vorhandenes Element hinzufügen"
|
|
}
|
|
},
|
|
"archived": {
|
|
"title": "Keine archivierten Module",
|
|
"description": "Archivieren Sie abgeschlossene oder abgebrochene Module."
|
|
},
|
|
"sidebar": {
|
|
"in_active": "Modul ist nicht aktiv.",
|
|
"invalid_date": "Ungültiges Datum. Bitte geben Sie ein gültiges Datum ein."
|
|
}
|
|
},
|
|
"quick_actions": {
|
|
"archive_module": "Modul archivieren",
|
|
"archive_module_description": "Nur abgeschlossene/abgebrochene Module können archiviert werden.",
|
|
"delete_module": "Modul löschen"
|
|
},
|
|
"toast": {
|
|
"copy": {
|
|
"success": "Link zum Modul kopiert"
|
|
},
|
|
"delete": {
|
|
"success": "Modul gelöscht",
|
|
"error": "Löschen fehlgeschlagen"
|
|
}
|
|
}
|
|
},
|
|
"project_page": {
|
|
"empty_state": {
|
|
"general": {
|
|
"title": "Notieren Sie Ideen, Dokumente oder Wissensdatenbanken. Nutzen Sie AI Galileo.",
|
|
"description": "Seiten sind Orte für Ihre Gedanken. Schreiben, formatieren, fügen Sie Arbeitselemente ein und verwenden Sie Komponenten.",
|
|
"primary_button": {
|
|
"text": "Erste Seite erstellen"
|
|
}
|
|
},
|
|
"private": {
|
|
"title": "Keine privaten Seiten",
|
|
"description": "Bewahren Sie private Gedanken auf. Teilen Sie sie, wenn Sie bereit sind.",
|
|
"primary_button": {
|
|
"text": "Seite erstellen"
|
|
}
|
|
},
|
|
"public": {
|
|
"title": "Keine öffentlichen Seiten",
|
|
"description": "Hier sehen Sie die im Projekt geteilten Seiten.",
|
|
"primary_button": {
|
|
"text": "Seite erstellen"
|
|
}
|
|
},
|
|
"archived": {
|
|
"title": "Keine archivierten Seiten",
|
|
"description": "Archivieren Sie Seiten für den späteren Zugriff."
|
|
}
|
|
}
|
|
},
|
|
"disabled_project": {
|
|
"empty_state": {
|
|
"inbox": {
|
|
"title": "Eingang ist nicht aktiviert",
|
|
"description": "Aktivieren Sie den Eingang in den Projekteinstellungen, um Anfragen zu verwalten.",
|
|
"primary_button": {
|
|
"text": "Funktionen verwalten"
|
|
}
|
|
},
|
|
"cycle": {
|
|
"title": "Zyklen sind nicht aktiviert",
|
|
"description": "Aktivieren Sie Zyklen, um Arbeit zeitlich zu begrenzen.",
|
|
"primary_button": {
|
|
"text": "Funktionen verwalten"
|
|
}
|
|
},
|
|
"module": {
|
|
"title": "Module sind nicht aktiviert",
|
|
"description": "Aktivieren Sie Module in den Projekteinstellungen.",
|
|
"primary_button": {
|
|
"text": "Funktionen verwalten"
|
|
}
|
|
},
|
|
"page": {
|
|
"title": "Seiten sind nicht aktiviert",
|
|
"description": "Aktivieren Sie Seiten in den Projekteinstellungen.",
|
|
"primary_button": {
|
|
"text": "Funktionen verwalten"
|
|
}
|
|
},
|
|
"view": {
|
|
"title": "Ansichten sind nicht aktiviert",
|
|
"description": "Aktivieren Sie Ansichten in den Projekteinstellungen.",
|
|
"primary_button": {
|
|
"text": "Funktionen verwalten"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
"project_modules": {
|
|
"status": {
|
|
"backlog": "Backlog",
|
|
"planned": "Geplant",
|
|
"in_progress": "In Bearbeitung",
|
|
"paused": "Pausiert",
|
|
"completed": "Abgeschlossen",
|
|
"cancelled": "Abgebrochen"
|
|
},
|
|
"layout": {
|
|
"list": "Liste",
|
|
"board": "Board",
|
|
"timeline": "Zeitachse"
|
|
},
|
|
"order_by": {
|
|
"name": "Name",
|
|
"progress": "Fortschritt",
|
|
"issues": "Anzahl Elemente",
|
|
"due_date": "Fälligkeitsdatum",
|
|
"created_at": "Erstellungsdatum",
|
|
"manual": "Manuell"
|
|
}
|
|
},
|
|
"project_members": {
|
|
"full_name": "Vollständiger Name",
|
|
"display_name": "Anzeigename",
|
|
"email": "E-Mail",
|
|
"joining_date": "Beitrittsdatum",
|
|
"role": "Rolle"
|
|
},
|
|
"project": {
|
|
"members_import": {
|
|
"title": "Mitglieder aus CSV importieren",
|
|
"description": "Laden Sie eine CSV mit den Spalten E-Mail und Rolle hoch (5=Gast, 15=Mitglied, 20=Admin). Benutzer müssen bereits Mitglieder des Arbeitsbereichs sein.",
|
|
"download_sample": "Beispiel-CSV herunterladen",
|
|
"dropzone": {
|
|
"active": "CSV-Datei hier ablegen",
|
|
"inactive": "Ziehen Sie eine Datei hierher oder klicken Sie zum Hochladen",
|
|
"file_type": "Nur .csv-Dateien werden unterstützt"
|
|
},
|
|
"buttons": {
|
|
"cancel": "Abbrechen",
|
|
"import": "Importieren",
|
|
"try_again": "Erneut versuchen",
|
|
"close": "Schließen",
|
|
"done": "Fertig"
|
|
},
|
|
"progress": {
|
|
"uploading": "Hochladen...",
|
|
"importing": "Importieren..."
|
|
},
|
|
"summary": {
|
|
"title": {
|
|
"complete": "Import abgeschlossen"
|
|
},
|
|
"message": {
|
|
"success": "{count} Mitglied{plural} erfolgreich ins Projekt importiert.",
|
|
"no_imports": "Aus der CSV-Datei wurden keine neuen Mitglieder importiert."
|
|
},
|
|
"stats": {
|
|
"added": "Hinzugefügt",
|
|
"reactivated": "Reaktiviert",
|
|
"already_members": "Bereits Mitglieder",
|
|
"skipped": "Übersprungen"
|
|
},
|
|
"download_errors": "Übersprungene Details herunterladen"
|
|
},
|
|
"toast": {
|
|
"invalid_file": {
|
|
"title": "Ungültige Datei",
|
|
"message": "Nur CSV-Dateien werden unterstützt."
|
|
},
|
|
"import_failed": {
|
|
"title": "Import fehlgeschlagen",
|
|
"message": "Etwas ist schief gelaufen."
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|