feat(phase2): localization EN/RU + additional widget types

- Add svelte-i18n with 224 translation keys (English + Russian)
- Language switcher in header (EN/RU toggle, persists to localStorage)
- Extract all hardcoded strings from 37 component/page files
- Add 4 new widget types: Bookmark, Note (markdown), Embed (iframe), Status
- WidgetRenderer dispatches by type, WidgetGrid supports full-width widgets
- Type-specific config forms in board editor
- Install marked for markdown rendering
This commit is contained in:
2026-03-24 23:18:05 +03:00
parent bf4e5089ee
commit 477c0e4d52
52 changed files with 1776 additions and 395 deletions
+245
View File
@@ -0,0 +1,245 @@
{
"app_name": "App Launcher",
"app_title": "Web App Launcher",
"nav.navigation": "Navigation",
"nav.boards": "Boards",
"nav.apps": "Apps",
"nav.admin": "Admin",
"nav.admin_panel": "Admin Panel",
"auth.login": "Sign In",
"auth.login_title": "Welcome back",
"auth.login_subtitle": "Sign in to your account",
"auth.login_submit": "Sign In",
"auth.login_submitting": "Signing in...",
"auth.register": "Register",
"auth.register_title": "Create Account",
"auth.register_subtitle": "Get started with App Launcher",
"auth.register_submit": "Create Account",
"auth.register_submitting": "Creating account...",
"auth.email": "Email",
"auth.email_placeholder": "you@example.com",
"auth.password": "Password",
"auth.password_placeholder": "Enter your password",
"auth.password_placeholder_register": "At least 6 characters",
"auth.display_name": "Display Name",
"auth.display_name_placeholder": "Your name",
"auth.logout": "Sign Out",
"auth.oauth_signin": "Sign in with OAuth",
"auth.or": "or",
"auth.no_account": "Don't have an account?",
"auth.have_account": "Already have an account?",
"auth.sign_in_link": "Sign in",
"board.title": "Boards",
"board.boards_available": "{count} board(s) available",
"board.new": "New Board",
"board.edit": "Edit",
"board.edit_board": "Edit Board",
"board.all_boards": "All Boards",
"board.back_to_boards": "Back to Boards",
"board.back_to_board": "Back to Board",
"board.no_boards": "No boards available.",
"board.sign_in_more": "Sign in to see more boards.",
"board.no_sections": "This board has no sections yet.",
"board.default": "Default",
"board.guest": "Guest",
"board.sections_count": "{count} section(s)",
"board.properties": "Board Properties",
"board.save": "Save Board",
"board.create": "Create Board",
"board.creating": "Creating...",
"board.default_board": "Default board",
"board.guest_accessible": "Guest accessible",
"section.title_label": "Title",
"section.icon_label": "Icon",
"section.icon_placeholder": "Optional",
"section.sections": "Sections",
"section.add": "Add Section",
"section.create": "Create Section",
"section.order": "Order: {order}",
"widget.add": "Add Widget",
"widget.select_app": "Select App",
"widget.choose_app": "Choose an app...",
"widget.no_widgets": "No widgets in this section.",
"widget.no_widgets_dnd": "No widgets. Drag widgets here or add one above.",
"widget.type": "{type} widget",
"widget.number": "Widget #{order}",
"widget.remove": "Remove",
"app.title": "App Registry",
"app.apps_registered": "{count} app(s) registered",
"app.add": "Add App",
"app.new": "New App",
"app.no_apps": "No apps registered yet.",
"app.no_apps_hint": "Click \"Add App\" to register your first application.",
"app.all_categories": "All",
"app.name": "Name",
"app.name_placeholder": "My Application",
"app.url": "URL",
"app.url_placeholder": "https://my-app.local:8080",
"app.description": "Description",
"app.description_placeholder": "Brief description of this app",
"app.category": "Category",
"app.category_placeholder": "e.g. Media, Monitoring, Storage",
"app.tags": "Tags",
"app.tags_placeholder": "Comma-separated tags",
"app.icon": "Icon",
"app.icon_lucide": "Lucide Icon",
"app.icon_simple": "Simple Icons",
"app.icon_url": "Image URL",
"app.icon_emoji": "Emoji",
"app.icon_lucide_placeholder": "e.g. globe, server, home",
"app.icon_simple_placeholder": "e.g. github, docker",
"app.icon_url_placeholder": "https://example.com/icon.png",
"app.icon_emoji_placeholder": "e.g. \ud83c\udf10",
"app.icon_preview": "Icon preview",
"app.save": "Save App",
"app.saving": "Saving...",
"app.healthcheck_toggle": "Healthcheck Settings",
"app.healthcheck_show": "Show",
"app.healthcheck_hide": "Hide",
"app.healthcheck_enabled": "Enable Healthcheck",
"app.healthcheck_method": "Method",
"app.healthcheck_expected_status": "Expected Status",
"app.healthcheck_timeout": "Timeout (ms)",
"app.healthcheck_interval": "Interval (seconds)",
"app.icon_board_label": "Icon (Lucide name)",
"admin.panel": "Admin Panel",
"admin.users": "Users",
"admin.groups": "Groups",
"admin.settings": "Settings",
"admin.user_management": "User Management",
"admin.create_user": "Create User",
"admin.new_user": "New User",
"admin.user_column": "User",
"admin.email_column": "Email",
"admin.role_column": "Role",
"admin.provider_column": "Provider",
"admin.groups_column": "Groups",
"admin.actions_column": "Actions",
"admin.role_user": "User",
"admin.role_admin": "Admin",
"admin.select_group": "Select group",
"admin.add_to_group": "+ Add",
"admin.remove_from_group": "Remove from group",
"admin.no_users": "No users found.",
"admin.group_management": "Group Management",
"admin.create_group": "Create Group",
"admin.new_group": "New Group",
"admin.name_column": "Name",
"admin.description_column": "Description",
"admin.members_column": "Members",
"admin.default_column": "Default",
"admin.default_group_hint": "Default group (auto-assign new users)",
"admin.no_groups": "No groups found.",
"admin.yes": "Yes",
"admin.no": "No",
"admin.system_settings": "System Settings",
"admin.settings_description": "Configure global application settings.",
"admin.authentication": "Authentication",
"admin.auth_mode": "Auth Mode",
"admin.auth_local": "Local",
"admin.auth_oauth": "OAuth",
"admin.auth_both": "Both",
"admin.registration_enabled": "Allow user registration",
"admin.oauth_config": "OAuth Configuration",
"admin.oauth_description": "Configure your OIDC provider (e.g. Authentik, Keycloak). Set Auth Mode to \"OAuth\" or \"Both\" above to enable OAuth login.",
"admin.oauth_client_id": "Client ID",
"admin.oauth_client_id_placeholder": "OAuth client ID",
"admin.oauth_client_secret": "Client Secret",
"admin.oauth_client_secret_placeholder": "OAuth client secret",
"admin.oauth_discovery_url": "Discovery URL",
"admin.oauth_discovery_url_placeholder": "https://example.com/.well-known/openid-configuration",
"admin.oauth_test": "Test Connection",
"admin.oauth_testing": "Testing...",
"admin.oauth_connected": "Connected to issuer: {issuer}",
"admin.oauth_network_error": "Network error \u2014 could not reach the server",
"admin.theme_defaults": "Theme Defaults",
"admin.default_theme": "Default Theme",
"admin.default_primary_color": "Default Primary Color",
"admin.healthcheck_defaults": "Healthcheck Defaults",
"admin.healthcheck_defaults_description": "JSON configuration for default healthcheck behavior (interval, timeout, method).",
"admin.healthcheck_defaults_label": "Defaults (JSON)",
"admin.save_settings": "Save Settings",
"admin.saving_settings": "Saving...",
"admin.perm_title": "Grant Permission",
"admin.perm_entity_type": "Entity Type",
"admin.perm_entity": "Entity",
"admin.perm_target_type": "Target Type",
"admin.perm_target": "Target",
"admin.perm_level": "Level",
"admin.perm_board": "Board",
"admin.perm_app": "App",
"admin.perm_user": "User",
"admin.perm_group": "Group",
"admin.perm_view": "View",
"admin.perm_edit": "Edit",
"admin.perm_admin": "Admin",
"admin.perm_grant": "Grant",
"admin.perm_revoke": "Revoke",
"admin.perm_select": "Select...",
"admin.perm_entity_column": "Entity",
"admin.perm_target_column": "Target",
"admin.perm_level_column": "Level",
"admin.perm_action_column": "Action",
"admin.perm_none": "No permissions configured.",
"search.placeholder": "Search apps and boards...",
"search.trigger": "Search...",
"search.min_chars": "Type at least 2 characters to search",
"search.no_results": "No results for \"{query}\"",
"search.apps": "Apps",
"search.boards": "Boards",
"common.save": "Save",
"common.cancel": "Cancel",
"common.delete": "Delete",
"common.create": "Create",
"common.back": "Back",
"common.edit": "Edit",
"common.add": "Add",
"common.confirm": "Confirm?",
"common.yes": "Yes",
"common.no": "No",
"common.name": "Name",
"common.description": "Description",
"common.required": "*",
"status.online": "Online",
"status.offline": "Offline",
"status.degraded": "Degraded",
"status.unknown": "Unknown",
"theme.dark": "Dark",
"theme.light": "Light",
"theme.system": "System",
"theme.toggle": "Toggle theme (current: {mode})",
"theme.title": "Theme: {mode}",
"bg.mesh": "Mesh Gradient",
"bg.particles": "Particles",
"bg.aurora": "Aurora",
"bg.none": "None",
"bg.title": "Background effect",
"bg.aria_label": "Change background effect",
"sidebar.expand": "Expand sidebar",
"sidebar.collapse": "Collapse sidebar",
"sidebar.toggle": "Toggle sidebar",
"sidebar.close": "Close sidebar",
"home.welcome": "Welcome, {name}. No default board is configured yet.",
"home.view_boards": "View Boards",
"home.browse_apps": "Browse Apps",
"language.label": "Language"
}
+34
View File
@@ -0,0 +1,34 @@
import { addMessages, init, getLocaleFromNavigator } from 'svelte-i18n';
import en from './en.json';
import ru from './ru.json';
const LOCALE_STORAGE_KEY = 'wal-locale';
addMessages('en', en);
addMessages('ru', ru);
function getStoredLocale(): string | null {
if (typeof localStorage === 'undefined') return null;
return localStorage.getItem(LOCALE_STORAGE_KEY);
}
function detectLocale(): string {
const stored = getStoredLocale();
if (stored && (stored === 'en' || stored === 'ru')) {
return stored;
}
const browserLocale = getLocaleFromNavigator() ?? 'en';
return browserLocale.startsWith('ru') ? 'ru' : 'en';
}
export function storeLocale(locale: string): void {
if (typeof localStorage !== 'undefined') {
localStorage.setItem(LOCALE_STORAGE_KEY, locale);
}
}
init({
fallbackLocale: 'en',
initialLocale: detectLocale()
});
+245
View File
@@ -0,0 +1,245 @@
{
"app_name": "App Launcher",
"app_title": "Web App Launcher",
"nav.navigation": "\u041d\u0430\u0432\u0438\u0433\u0430\u0446\u0438\u044f",
"nav.boards": "\u0414\u043e\u0441\u043a\u0438",
"nav.apps": "\u041f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f",
"nav.admin": "\u0410\u0434\u043c\u0438\u043d",
"nav.admin_panel": "\u041f\u0430\u043d\u0435\u043b\u044c \u0430\u0434\u043c\u0438\u043d\u0438\u0441\u0442\u0440\u0430\u0442\u043e\u0440\u0430",
"auth.login": "\u0412\u043e\u0439\u0442\u0438",
"auth.login_title": "\u0414\u043e\u0431\u0440\u043e \u043f\u043e\u0436\u0430\u043b\u043e\u0432\u0430\u0442\u044c",
"auth.login_subtitle": "\u0412\u043e\u0439\u0434\u0438\u0442\u0435 \u0432 \u0441\u0432\u043e\u0439 \u0430\u043a\u043a\u0430\u0443\u043d\u0442",
"auth.login_submit": "\u0412\u043e\u0439\u0442\u0438",
"auth.login_submitting": "\u0412\u0445\u043e\u0434...",
"auth.register": "\u0420\u0435\u0433\u0438\u0441\u0442\u0440\u0430\u0446\u0438\u044f",
"auth.register_title": "\u0421\u043e\u0437\u0434\u0430\u0442\u044c \u0430\u043a\u043a\u0430\u0443\u043d\u0442",
"auth.register_subtitle": "\u041d\u0430\u0447\u043d\u0438\u0442\u0435 \u0440\u0430\u0431\u043e\u0442\u0443 \u0441 App Launcher",
"auth.register_submit": "\u0421\u043e\u0437\u0434\u0430\u0442\u044c \u0430\u043a\u043a\u0430\u0443\u043d\u0442",
"auth.register_submitting": "\u0421\u043e\u0437\u0434\u0430\u043d\u0438\u0435 \u0430\u043a\u043a\u0430\u0443\u043d\u0442\u0430...",
"auth.email": "\u042d\u043b\u0435\u043a\u0442\u0440\u043e\u043d\u043d\u0430\u044f \u043f\u043e\u0447\u0442\u0430",
"auth.email_placeholder": "you@example.com",
"auth.password": "\u041f\u0430\u0440\u043e\u043b\u044c",
"auth.password_placeholder": "\u0412\u0432\u0435\u0434\u0438\u0442\u0435 \u043f\u0430\u0440\u043e\u043b\u044c",
"auth.password_placeholder_register": "\u041d\u0435 \u043c\u0435\u043d\u0435\u0435 6 \u0441\u0438\u043c\u0432\u043e\u043b\u043e\u0432",
"auth.display_name": "\u0418\u043c\u044f",
"auth.display_name_placeholder": "\u0412\u0430\u0448\u0435 \u0438\u043c\u044f",
"auth.logout": "\u0412\u044b\u0445\u043e\u0434",
"auth.oauth_signin": "\u0412\u043e\u0439\u0442\u0438 \u0447\u0435\u0440\u0435\u0437 OAuth",
"auth.or": "\u0438\u043b\u0438",
"auth.no_account": "\u041d\u0435\u0442 \u0430\u043a\u043a\u0430\u0443\u043d\u0442\u0430?",
"auth.have_account": "\u0423\u0436\u0435 \u0435\u0441\u0442\u044c \u0430\u043a\u043a\u0430\u0443\u043d\u0442?",
"auth.sign_in_link": "\u0412\u043e\u0439\u0442\u0438",
"board.title": "\u0414\u043e\u0441\u043a\u0438",
"board.boards_available": "\u0414\u043e\u0441\u0442\u0443\u043f\u043d\u043e \u0434\u043e\u0441\u043e\u043a: {count}",
"board.new": "\u041d\u043e\u0432\u0430\u044f \u0434\u043e\u0441\u043a\u0430",
"board.edit": "\u0420\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c",
"board.edit_board": "\u0420\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u0434\u043e\u0441\u043a\u0438",
"board.all_boards": "\u0412\u0441\u0435 \u0434\u043e\u0441\u043a\u0438",
"board.back_to_boards": "\u041d\u0430\u0437\u0430\u0434 \u043a \u0434\u043e\u0441\u043a\u0430\u043c",
"board.back_to_board": "\u041d\u0430\u0437\u0430\u0434 \u043a \u0434\u043e\u0441\u043a\u0435",
"board.no_boards": "\u0414\u043e\u0441\u043a\u0438 \u043d\u0435 \u043d\u0430\u0439\u0434\u0435\u043d\u044b.",
"board.sign_in_more": "\u0412\u043e\u0439\u0434\u0438\u0442\u0435, \u0447\u0442\u043e\u0431\u044b \u0443\u0432\u0438\u0434\u0435\u0442\u044c \u0431\u043e\u043b\u044c\u0448\u0435 \u0434\u043e\u0441\u043e\u043a.",
"board.no_sections": "\u041d\u0430 \u044d\u0442\u043e\u0439 \u0434\u043e\u0441\u043a\u0435 \u043f\u043e\u043a\u0430 \u043d\u0435\u0442 \u0440\u0430\u0437\u0434\u0435\u043b\u043e\u0432.",
"board.default": "\u041f\u043e \u0443\u043c\u043e\u043b\u0447\u0430\u043d\u0438\u044e",
"board.guest": "\u0413\u043e\u0441\u0442\u0435\u0432\u0430\u044f",
"board.sections_count": "\u0420\u0430\u0437\u0434\u0435\u043b\u043e\u0432: {count}",
"board.properties": "\u0421\u0432\u043e\u0439\u0441\u0442\u0432\u0430 \u0434\u043e\u0441\u043a\u0438",
"board.save": "\u0421\u043e\u0445\u0440\u0430\u043d\u0438\u0442\u044c \u0434\u043e\u0441\u043a\u0443",
"board.create": "\u0421\u043e\u0437\u0434\u0430\u0442\u044c \u0434\u043e\u0441\u043a\u0443",
"board.creating": "\u0421\u043e\u0437\u0434\u0430\u043d\u0438\u0435...",
"board.default_board": "\u0414\u043e\u0441\u043a\u0430 \u043f\u043e \u0443\u043c\u043e\u043b\u0447\u0430\u043d\u0438\u044e",
"board.guest_accessible": "\u0414\u043e\u0441\u0442\u0443\u043f\u043d\u0430 \u0433\u043e\u0441\u0442\u044f\u043c",
"section.title_label": "\u0417\u0430\u0433\u043e\u043b\u043e\u0432\u043e\u043a",
"section.icon_label": "\u0418\u043a\u043e\u043d\u043a\u0430",
"section.icon_placeholder": "\u041d\u0435\u043e\u0431\u044f\u0437\u0430\u0442\u0435\u043b\u044c\u043d\u043e",
"section.sections": "\u0420\u0430\u0437\u0434\u0435\u043b\u044b",
"section.add": "\u0414\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0440\u0430\u0437\u0434\u0435\u043b",
"section.create": "\u0421\u043e\u0437\u0434\u0430\u0442\u044c \u0440\u0430\u0437\u0434\u0435\u043b",
"section.order": "\u041f\u043e\u0440\u044f\u0434\u043e\u043a: {order}",
"widget.add": "\u0414\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0432\u0438\u0434\u0436\u0435\u0442",
"widget.select_app": "\u0412\u044b\u0431\u0435\u0440\u0438\u0442\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435",
"widget.choose_app": "\u0412\u044b\u0431\u0435\u0440\u0438\u0442\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435...",
"widget.no_widgets": "\u0412 \u044d\u0442\u043e\u043c \u0440\u0430\u0437\u0434\u0435\u043b\u0435 \u043d\u0435\u0442 \u0432\u0438\u0434\u0436\u0435\u0442\u043e\u0432.",
"widget.no_widgets_dnd": "\u041d\u0435\u0442 \u0432\u0438\u0434\u0436\u0435\u0442\u043e\u0432. \u041f\u0435\u0440\u0435\u0442\u0430\u0449\u0438\u0442\u0435 \u0441\u044e\u0434\u0430 \u0438\u043b\u0438 \u0434\u043e\u0431\u0430\u0432\u044c\u0442\u0435 \u0432\u044b\u0448\u0435.",
"widget.type": "\u0412\u0438\u0434\u0436\u0435\u0442 {type}",
"widget.number": "\u0412\u0438\u0434\u0436\u0435\u0442 #{order}",
"widget.remove": "\u0423\u0434\u0430\u043b\u0438\u0442\u044c",
"app.title": "\u0420\u0435\u0435\u0441\u0442\u0440 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439",
"app.apps_registered": "\u0417\u0430\u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043e \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439: {count}",
"app.add": "\u0414\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435",
"app.new": "\u041d\u043e\u0432\u043e\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435",
"app.no_apps": "\u041f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u0435\u0449\u0451 \u043d\u0435 \u0437\u0430\u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u044b.",
"app.no_apps_hint": "\u041d\u0430\u0436\u043c\u0438\u0442\u0435 \u00ab\u0414\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435\u00bb, \u0447\u0442\u043e\u0431\u044b \u0437\u0430\u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043f\u0435\u0440\u0432\u043e\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435.",
"app.all_categories": "\u0412\u0441\u0435",
"app.name": "\u041d\u0430\u0437\u0432\u0430\u043d\u0438\u0435",
"app.name_placeholder": "\u041c\u043e\u0451 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435",
"app.url": "URL",
"app.url_placeholder": "https://my-app.local:8080",
"app.description": "\u041e\u043f\u0438\u0441\u0430\u043d\u0438\u0435",
"app.description_placeholder": "\u041a\u0440\u0430\u0442\u043a\u043e\u0435 \u043e\u043f\u0438\u0441\u0430\u043d\u0438\u0435 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f",
"app.category": "\u041a\u0430\u0442\u0435\u0433\u043e\u0440\u0438\u044f",
"app.category_placeholder": "\u043d\u0430\u043f\u0440. \u041c\u0435\u0434\u0438\u0430, \u041c\u043e\u043d\u0438\u0442\u043e\u0440\u0438\u043d\u0433, \u0425\u0440\u0430\u043d\u0438\u043b\u0438\u0449\u0435",
"app.tags": "\u0422\u0435\u0433\u0438",
"app.tags_placeholder": "\u0422\u0435\u0433\u0438 \u0447\u0435\u0440\u0435\u0437 \u0437\u0430\u043f\u044f\u0442\u0443\u044e",
"app.icon": "\u0418\u043a\u043e\u043d\u043a\u0430",
"app.icon_lucide": "Lucide",
"app.icon_simple": "Simple Icons",
"app.icon_url": "URL \u0438\u0437\u043e\u0431\u0440\u0430\u0436\u0435\u043d\u0438\u044f",
"app.icon_emoji": "\u042d\u043c\u043e\u0434\u0437\u0438",
"app.icon_lucide_placeholder": "\u043d\u0430\u043f\u0440. globe, server, home",
"app.icon_simple_placeholder": "\u043d\u0430\u043f\u0440. github, docker",
"app.icon_url_placeholder": "https://example.com/icon.png",
"app.icon_emoji_placeholder": "\u043d\u0430\u043f\u0440. \ud83c\udf10",
"app.icon_preview": "\u041f\u0440\u0435\u0432\u044c\u044e \u0438\u043a\u043e\u043d\u043a\u0438",
"app.save": "\u0421\u043e\u0445\u0440\u0430\u043d\u0438\u0442\u044c",
"app.saving": "\u0421\u043e\u0445\u0440\u0430\u043d\u0435\u043d\u0438\u0435...",
"app.healthcheck_toggle": "\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u043f\u0440\u043e\u0432\u0435\u0440\u043a\u0438 \u0437\u0434\u043e\u0440\u043e\u0432\u044c\u044f",
"app.healthcheck_show": "\u041f\u043e\u043a\u0430\u0437\u0430\u0442\u044c",
"app.healthcheck_hide": "\u0421\u043a\u0440\u044b\u0442\u044c",
"app.healthcheck_enabled": "\u0412\u043a\u043b\u044e\u0447\u0438\u0442\u044c \u043f\u0440\u043e\u0432\u0435\u0440\u043a\u0443 \u0437\u0434\u043e\u0440\u043e\u0432\u044c\u044f",
"app.healthcheck_method": "\u041c\u0435\u0442\u043e\u0434",
"app.healthcheck_expected_status": "\u041e\u0436\u0438\u0434\u0430\u0435\u043c\u044b\u0439 \u0441\u0442\u0430\u0442\u0443\u0441",
"app.healthcheck_timeout": "\u0422\u0430\u0439\u043c\u0430\u0443\u0442 (\u043c\u0441)",
"app.healthcheck_interval": "\u0418\u043d\u0442\u0435\u0440\u0432\u0430\u043b (\u0441\u0435\u043a\u0443\u043d\u0434\u044b)",
"app.icon_board_label": "\u0418\u043a\u043e\u043d\u043a\u0430 (Lucide)",
"admin.panel": "\u041f\u0430\u043d\u0435\u043b\u044c \u0430\u0434\u043c\u0438\u043d\u0438\u0441\u0442\u0440\u0430\u0442\u043e\u0440\u0430",
"admin.users": "\u041f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0438",
"admin.groups": "\u0413\u0440\u0443\u043f\u043f\u044b",
"admin.settings": "\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438",
"admin.user_management": "\u0423\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f\u043c\u0438",
"admin.create_user": "\u0421\u043e\u0437\u0434\u0430\u0442\u044c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f",
"admin.new_user": "\u041d\u043e\u0432\u044b\u0439 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c",
"admin.user_column": "\u041f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c",
"admin.email_column": "\u042d\u043b\u0435\u043a\u0442\u0440\u043e\u043d\u043d\u0430\u044f \u043f\u043e\u0447\u0442\u0430",
"admin.role_column": "\u0420\u043e\u043b\u044c",
"admin.provider_column": "\u041f\u0440\u043e\u0432\u0430\u0439\u0434\u0435\u0440",
"admin.groups_column": "\u0413\u0440\u0443\u043f\u043f\u044b",
"admin.actions_column": "\u0414\u0435\u0439\u0441\u0442\u0432\u0438\u044f",
"admin.role_user": "\u041f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c",
"admin.role_admin": "\u0410\u0434\u043c\u0438\u043d\u0438\u0441\u0442\u0440\u0430\u0442\u043e\u0440",
"admin.select_group": "\u0412\u044b\u0431\u0440\u0430\u0442\u044c \u0433\u0440\u0443\u043f\u043f\u0443",
"admin.add_to_group": "+ \u0414\u043e\u0431\u0430\u0432\u0438\u0442\u044c",
"admin.remove_from_group": "\u0423\u0434\u0430\u043b\u0438\u0442\u044c \u0438\u0437 \u0433\u0440\u0443\u043f\u043f\u044b",
"admin.no_users": "\u041f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0438 \u043d\u0435 \u043d\u0430\u0439\u0434\u0435\u043d\u044b.",
"admin.group_management": "\u0423\u043f\u0440\u0430\u0432\u043b\u0435\u043d\u0438\u0435 \u0433\u0440\u0443\u043f\u043f\u0430\u043c\u0438",
"admin.create_group": "\u0421\u043e\u0437\u0434\u0430\u0442\u044c \u0433\u0440\u0443\u043f\u043f\u0443",
"admin.new_group": "\u041d\u043e\u0432\u0430\u044f \u0433\u0440\u0443\u043f\u043f\u0430",
"admin.name_column": "\u041d\u0430\u0437\u0432\u0430\u043d\u0438\u0435",
"admin.description_column": "\u041e\u043f\u0438\u0441\u0430\u043d\u0438\u0435",
"admin.members_column": "\u0423\u0447\u0430\u0441\u0442\u043d\u0438\u043a\u0438",
"admin.default_column": "\u041f\u043e \u0443\u043c\u043e\u043b\u0447\u0430\u043d\u0438\u044e",
"admin.default_group_hint": "\u0413\u0440\u0443\u043f\u043f\u0430 \u043f\u043e \u0443\u043c\u043e\u043b\u0447\u0430\u043d\u0438\u044e (\u0430\u0432\u0442\u043e-\u043d\u0430\u0437\u043d\u0430\u0447\u0435\u043d\u0438\u0435 \u043d\u043e\u0432\u044b\u043c \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f\u043c)",
"admin.no_groups": "\u0413\u0440\u0443\u043f\u043f\u044b \u043d\u0435 \u043d\u0430\u0439\u0434\u0435\u043d\u044b.",
"admin.yes": "\u0414\u0430",
"admin.no": "\u041d\u0435\u0442",
"admin.system_settings": "\u0421\u0438\u0441\u0442\u0435\u043c\u043d\u044b\u0435 \u043d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438",
"admin.settings_description": "\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0430 \u0433\u043b\u043e\u0431\u0430\u043b\u044c\u043d\u044b\u0445 \u043f\u0430\u0440\u0430\u043c\u0435\u0442\u0440\u043e\u0432 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f.",
"admin.authentication": "\u0410\u0443\u0442\u0435\u043d\u0442\u0438\u0444\u0438\u043a\u0430\u0446\u0438\u044f",
"admin.auth_mode": "\u0420\u0435\u0436\u0438\u043c \u0430\u0443\u0442\u0435\u043d\u0442\u0438\u0444\u0438\u043a\u0430\u0446\u0438\u0438",
"admin.auth_local": "\u041b\u043e\u043a\u0430\u043b\u044c\u043d\u044b\u0439",
"admin.auth_oauth": "OAuth",
"admin.auth_both": "\u041e\u0431\u0430",
"admin.registration_enabled": "\u0420\u0430\u0437\u0440\u0435\u0448\u0438\u0442\u044c \u0440\u0435\u0433\u0438\u0441\u0442\u0440\u0430\u0446\u0438\u044e \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0435\u0439",
"admin.oauth_config": "\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0430 OAuth",
"admin.oauth_description": "\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u0442\u0435 \u043f\u0440\u043e\u0432\u0430\u0439\u0434\u0435\u0440 OIDC (\u043d\u0430\u043f\u0440. Authentik, Keycloak). \u0423\u0441\u0442\u0430\u043d\u043e\u0432\u0438\u0442\u0435 \u0440\u0435\u0436\u0438\u043c \u0430\u0443\u0442\u0435\u043d\u0442\u0438\u0444\u0438\u043a\u0430\u0446\u0438\u0438 \u00abOAuth\u00bb \u0438\u043b\u0438 \u00ab\u041e\u0431\u0430\u00bb \u0432\u044b\u0448\u0435, \u0447\u0442\u043e\u0431\u044b \u0432\u043a\u043b\u044e\u0447\u0438\u0442\u044c \u0432\u0445\u043e\u0434 \u0447\u0435\u0440\u0435\u0437 OAuth.",
"admin.oauth_client_id": "Client ID",
"admin.oauth_client_id_placeholder": "OAuth client ID",
"admin.oauth_client_secret": "\u0421\u0435\u043a\u0440\u0435\u0442 \u043a\u043b\u0438\u0435\u043d\u0442\u0430",
"admin.oauth_client_secret_placeholder": "\u0421\u0435\u043a\u0440\u0435\u0442 OAuth \u043a\u043b\u0438\u0435\u043d\u0442\u0430",
"admin.oauth_discovery_url": "Discovery URL",
"admin.oauth_discovery_url_placeholder": "https://example.com/.well-known/openid-configuration",
"admin.oauth_test": "\u0422\u0435\u0441\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435",
"admin.oauth_testing": "\u0422\u0435\u0441\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435...",
"admin.oauth_connected": "\u041f\u043e\u0434\u043a\u043b\u044e\u0447\u0435\u043d\u043e \u043a: {issuer}",
"admin.oauth_network_error": "\u041e\u0448\u0438\u0431\u043a\u0430 \u0441\u0435\u0442\u0438 \u2014 \u043d\u0435 \u0443\u0434\u0430\u043b\u043e\u0441\u044c \u0441\u0432\u044f\u0437\u0430\u0442\u044c\u0441\u044f \u0441 \u0441\u0435\u0440\u0432\u0435\u0440\u043e\u043c",
"admin.theme_defaults": "\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u0442\u0435\u043c\u044b",
"admin.default_theme": "\u0422\u0435\u043c\u0430 \u043f\u043e \u0443\u043c\u043e\u043b\u0447\u0430\u043d\u0438\u044e",
"admin.default_primary_color": "\u041e\u0441\u043d\u043e\u0432\u043d\u043e\u0439 \u0446\u0432\u0435\u0442 \u043f\u043e \u0443\u043c\u043e\u043b\u0447\u0430\u043d\u0438\u044e",
"admin.healthcheck_defaults": "\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 \u043f\u0440\u043e\u0432\u0435\u0440\u043a\u0438 \u0437\u0434\u043e\u0440\u043e\u0432\u044c\u044f",
"admin.healthcheck_defaults_description": "JSON-\u043a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0430\u0446\u0438\u044f \u043f\u0440\u043e\u0432\u0435\u0440\u043a\u0438 \u0437\u0434\u043e\u0440\u043e\u0432\u044c\u044f \u043f\u043e \u0443\u043c\u043e\u043b\u0447\u0430\u043d\u0438\u044e (\u0438\u043d\u0442\u0435\u0440\u0432\u0430\u043b, \u0442\u0430\u0439\u043c\u0430\u0443\u0442, \u043c\u0435\u0442\u043e\u0434).",
"admin.healthcheck_defaults_label": "\u041d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438 (JSON)",
"admin.save_settings": "\u0421\u043e\u0445\u0440\u0430\u043d\u0438\u0442\u044c \u043d\u0430\u0441\u0442\u0440\u043e\u0439\u043a\u0438",
"admin.saving_settings": "\u0421\u043e\u0445\u0440\u0430\u043d\u0435\u043d\u0438\u0435...",
"admin.perm_title": "\u041d\u0430\u0437\u043d\u0430\u0447\u0438\u0442\u044c \u043f\u0440\u0430\u0432\u0430",
"admin.perm_entity_type": "\u0422\u0438\u043f \u043e\u0431\u044a\u0435\u043a\u0442\u0430",
"admin.perm_entity": "\u041e\u0431\u044a\u0435\u043a\u0442",
"admin.perm_target_type": "\u0422\u0438\u043f \u0446\u0435\u043b\u0438",
"admin.perm_target": "\u0426\u0435\u043b\u044c",
"admin.perm_level": "\u0423\u0440\u043e\u0432\u0435\u043d\u044c",
"admin.perm_board": "\u0414\u043e\u0441\u043a\u0430",
"admin.perm_app": "\u041f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0435",
"admin.perm_user": "\u041f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044c",
"admin.perm_group": "\u0413\u0440\u0443\u043f\u043f\u0430",
"admin.perm_view": "\u041f\u0440\u043e\u0441\u043c\u043e\u0442\u0440",
"admin.perm_edit": "\u0420\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435",
"admin.perm_admin": "\u0410\u0434\u043c\u0438\u043d\u0438\u0441\u0442\u0440\u0430\u0442\u043e\u0440",
"admin.perm_grant": "\u041d\u0430\u0437\u043d\u0430\u0447\u0438\u0442\u044c",
"admin.perm_revoke": "\u041e\u0442\u043e\u0437\u0432\u0430\u0442\u044c",
"admin.perm_select": "\u0412\u044b\u0431\u0440\u0430\u0442\u044c...",
"admin.perm_entity_column": "\u041e\u0431\u044a\u0435\u043a\u0442",
"admin.perm_target_column": "\u0426\u0435\u043b\u044c",
"admin.perm_level_column": "\u0423\u0440\u043e\u0432\u0435\u043d\u044c",
"admin.perm_action_column": "\u0414\u0435\u0439\u0441\u0442\u0432\u0438\u0435",
"admin.perm_none": "\u041f\u0440\u0430\u0432\u0430 \u043d\u0435 \u043d\u0430\u0441\u0442\u0440\u043e\u0435\u043d\u044b.",
"search.placeholder": "\u041f\u043e\u0438\u0441\u043a \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439 \u0438 \u0434\u043e\u0441\u043e\u043a...",
"search.trigger": "\u041f\u043e\u0438\u0441\u043a...",
"search.min_chars": "\u0412\u0432\u0435\u0434\u0438\u0442\u0435 \u043c\u0438\u043d\u0438\u043c\u0443\u043c 2 \u0441\u0438\u043c\u0432\u043e\u043b\u0430 \u0434\u043b\u044f \u043f\u043e\u0438\u0441\u043a\u0430",
"search.no_results": "\u041d\u0438\u0447\u0435\u0433\u043e \u043d\u0435 \u043d\u0430\u0439\u0434\u0435\u043d\u043e \u043f\u043e \u0437\u0430\u043f\u0440\u043e\u0441\u0443 \u00ab{query}\u00bb",
"search.apps": "\u041f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u044f",
"search.boards": "\u0414\u043e\u0441\u043a\u0438",
"common.save": "\u0421\u043e\u0445\u0440\u0430\u043d\u0438\u0442\u044c",
"common.cancel": "\u041e\u0442\u043c\u0435\u043d\u0430",
"common.delete": "\u0423\u0434\u0430\u043b\u0438\u0442\u044c",
"common.create": "\u0421\u043e\u0437\u0434\u0430\u0442\u044c",
"common.back": "\u041d\u0430\u0437\u0430\u0434",
"common.edit": "\u0420\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c",
"common.add": "\u0414\u043e\u0431\u0430\u0432\u0438\u0442\u044c",
"common.confirm": "\u041f\u043e\u0434\u0442\u0432\u0435\u0440\u0434\u0438\u0442\u044c?",
"common.yes": "\u0414\u0430",
"common.no": "\u041d\u0435\u0442",
"common.name": "\u041d\u0430\u0437\u0432\u0430\u043d\u0438\u0435",
"common.description": "\u041e\u043f\u0438\u0441\u0430\u043d\u0438\u0435",
"common.required": "*",
"status.online": "\u041e\u043d\u043b\u0430\u0439\u043d",
"status.offline": "\u041e\u0444\u0444\u043b\u0430\u0439\u043d",
"status.degraded": "\u041d\u0435\u0441\u0442\u0430\u0431\u0438\u043b\u044c\u043d\u043e",
"status.unknown": "\u041d\u0435\u0438\u0437\u0432\u0435\u0441\u0442\u043d\u043e",
"theme.dark": "\u0422\u0451\u043c\u043d\u0430\u044f",
"theme.light": "\u0421\u0432\u0435\u0442\u043b\u0430\u044f",
"theme.system": "\u0421\u0438\u0441\u0442\u0435\u043c\u043d\u0430\u044f",
"theme.toggle": "\u041f\u0435\u0440\u0435\u043a\u043b\u044e\u0447\u0438\u0442\u044c \u0442\u0435\u043c\u0443 (\u0442\u0435\u043a\u0443\u0449\u0430\u044f: {mode})",
"theme.title": "\u0422\u0435\u043c\u0430: {mode}",
"bg.mesh": "\u041c\u0435\u0448-\u0433\u0440\u0430\u0434\u0438\u0435\u043d\u0442",
"bg.particles": "\u0427\u0430\u0441\u0442\u0438\u0446\u044b",
"bg.aurora": "\u0421\u0438\u044f\u043d\u0438\u0435",
"bg.none": "\u041d\u0435\u0442",
"bg.title": "\u042d\u0444\u0444\u0435\u043a\u0442 \u0444\u043e\u043d\u0430",
"bg.aria_label": "\u0418\u0437\u043c\u0435\u043d\u0438\u0442\u044c \u044d\u0444\u0444\u0435\u043a\u0442 \u0444\u043e\u043d\u0430",
"sidebar.expand": "\u0420\u0430\u0437\u0432\u0435\u0440\u043d\u0443\u0442\u044c \u0431\u043e\u043a\u043e\u0432\u0443\u044e \u043f\u0430\u043d\u0435\u043b\u044c",
"sidebar.collapse": "\u0421\u0432\u0435\u0440\u043d\u0443\u0442\u044c \u0431\u043e\u043a\u043e\u0432\u0443\u044e \u043f\u0430\u043d\u0435\u043b\u044c",
"sidebar.toggle": "\u041f\u0435\u0440\u0435\u043a\u043b\u044e\u0447\u0438\u0442\u044c \u0431\u043e\u043a\u043e\u0432\u0443\u044e \u043f\u0430\u043d\u0435\u043b\u044c",
"sidebar.close": "\u0417\u0430\u043a\u0440\u044b\u0442\u044c \u0431\u043e\u043a\u043e\u0432\u0443\u044e \u043f\u0430\u043d\u0435\u043b\u044c",
"home.welcome": "\u0414\u043e\u0431\u0440\u043e \u043f\u043e\u0436\u0430\u043b\u043e\u0432\u0430\u0442\u044c, {name}. \u0414\u043e\u0441\u043a\u0430 \u043f\u043e \u0443\u043c\u043e\u043b\u0447\u0430\u043d\u0438\u044e \u0435\u0449\u0451 \u043d\u0435 \u043d\u0430\u0441\u0442\u0440\u043e\u0435\u043d\u0430.",
"home.view_boards": "\u041f\u043e\u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c \u0434\u043e\u0441\u043a\u0438",
"home.browse_apps": "\u041e\u0431\u0437\u043e\u0440 \u043f\u0440\u0438\u043b\u043e\u0436\u0435\u043d\u0438\u0439",
"language.label": "\u042f\u0437\u044b\u043a"
}