mirror of
				https://github.com/zadam/trilium.git
				synced 2025-10-26 15:56:29 +01:00 
			
		
		
		
	Compare commits
	
		
			90 Commits
		
	
	
		
			feat/add-w
			...
			feat/add-c
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 5332f015ef | ||
|  | 18f0f3ecac | ||
|  | e7d745ac94 | ||
|  | 24abf7f0ed | ||
|  | 9a08f6534b | ||
|  | c97c66ed8a | ||
|  | b581025bbe | ||
|  | 7bc5331747 | ||
|  | 2415976475 | ||
|  | 8d0d0f0449 | ||
|  | 16b00ed160 | ||
|  | df73a420f9 | ||
|  | 1e4d57f275 | ||
|  | 19a238c8d3 | ||
|  | 5ffd8a79eb | ||
|  | 58e58c192f | ||
|  | 5939344378 | ||
|  | 349f19fef7 | ||
|  | d5777a024e | ||
|  | b7f4ee6171 | ||
|  | a83c4e3970 | ||
|  | 5a767dae34 | ||
|  | 9f93d30b99 | ||
|  | dff525edc6 | ||
|  | 26da431320 | ||
|  | cde4622693 | ||
|  | a3d77421fd | ||
|  | 5ede7ecc69 | ||
|  | 4e755dc537 | ||
|  | 5351310a38 | ||
|  | 211ca43a82 | ||
|  | e5235e7f22 | ||
|  | e72298f0b4 | ||
|  | 3abf5c65c6 | ||
|  | 268acb0b88 | ||
|  | 196b3b873f | ||
|  | 4d9801a372 | ||
|  | bd710ba665 | ||
|  | afe369c876 | ||
|  | 206007bbce | ||
|  | 8ad05b92c0 | ||
|  | 735da2a855 | ||
|  | 980077f559 | ||
|  | 5daca270e4 | ||
|  | e18813a4bf | ||
|  | 4aa7e211f3 | ||
|  | 419dc7edfb | ||
|  | 1d0503d0e4 | ||
|  | f7f98aa9a3 | ||
|  | 575d14261a | ||
|  | 9aab606deb | ||
|  | 2e11681b52 | ||
|  | 8cca6637f7 | ||
|  | 82e076378c | ||
|  | 94ddad3c49 | ||
|  | d35dbca18b | ||
|  | 7468d6147a | ||
|  | 7c78d749de | ||
|  | 85dd99a3c4 | ||
|  | 0a9c0234e2 | ||
|  | fad77ba5a0 | ||
|  | 12723f3216 | ||
|  | a43140515f | ||
|  | 3e3cc8c541 | ||
|  | d1538508e8 | ||
|  | 9b1da8c311 | ||
|  | e4a8258acf | ||
|  | 5e88043c7b | ||
|  | bedf9112fb | ||
|  | 03681d23c5 | ||
|  | aa191e110c | ||
|  | dd09907925 | ||
|  | 35e9508bde | ||
|  | 4c8da70ef3 | ||
|  | ed5da5cd4a | ||
|  | dc5fccdbcd | ||
|  | 91aea333c7 | ||
|  | a0de01cff1 | ||
|  | a41ed34193 | ||
|  | 49e8811c18 | ||
|  | 488563a82e | ||
|  | a1b18c7f97 | ||
|  | 9958a6e1bf | ||
|  | 1fc6d8aca7 | ||
|  | 3e9ec2d943 | ||
|  | 1420def1c3 | ||
|  | 3b4184e765 | ||
|  | b70e25d348 | ||
|  | 772c0bbe1a | ||
|  | 144021c053 | 
| @@ -35,13 +35,13 @@ | ||||
|     "chore:generate-openapi": "tsx bin/generate-openapi.js" | ||||
|   }, | ||||
|   "devDependencies": {     | ||||
|     "@playwright/test": "1.54.2", | ||||
|     "@playwright/test": "1.55.0", | ||||
|     "@stylistic/eslint-plugin": "5.2.3",         | ||||
|     "@types/express": "5.0.3",     | ||||
|     "@types/node": "22.17.2",     | ||||
|     "@types/yargs": "17.0.33", | ||||
|     "@vitest/coverage-v8": "3.2.4", | ||||
|     "eslint": "9.33.0", | ||||
|     "eslint": "9.34.0", | ||||
|     "eslint-plugin-simple-import-sort": "12.1.1", | ||||
|     "esm": "3.2.25", | ||||
|     "jsdoc": "4.0.4", | ||||
|   | ||||
| @@ -10,7 +10,7 @@ | ||||
|     "url": "https://github.com/TriliumNext/Notes" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@eslint/js": "9.33.0", | ||||
|     "@eslint/js": "9.34.0", | ||||
|     "@excalidraw/excalidraw": "0.18.0", | ||||
|     "@fullcalendar/core": "6.1.19", | ||||
|     "@fullcalendar/daygrid": "6.1.19", | ||||
| @@ -36,7 +36,7 @@ | ||||
|     "draggabilly": "3.0.0", | ||||
|     "force-graph": "1.50.1", | ||||
|     "globals": "16.3.0", | ||||
|     "i18next": "25.3.6", | ||||
|     "i18next": "25.4.1", | ||||
|     "i18next-http-backend": "3.0.2", | ||||
|     "jquery": "3.7.1", | ||||
|     "jquery.fancytree": "2.38.5", | ||||
| @@ -47,12 +47,12 @@ | ||||
|     "leaflet-gpx": "2.2.0", | ||||
|     "mark.js": "8.11.1", | ||||
|     "marked": "16.2.0", | ||||
|     "mermaid": "11.10.0", | ||||
|     "mermaid": "11.10.1", | ||||
|     "mind-elixir": "5.0.6", | ||||
|     "normalize.css": "8.0.1", | ||||
|     "panzoom": "9.4.3", | ||||
|     "preact": "10.27.1", | ||||
|     "react-i18next": "15.6.1", | ||||
|     "react-i18next": "15.7.2", | ||||
|     "split.js": "1.6.5", | ||||
|     "svg-pan-zoom": "3.6.2", | ||||
|     "tabulator-tables": "6.3.1", | ||||
| @@ -70,7 +70,7 @@ | ||||
|     "copy-webpack-plugin": "13.0.1", | ||||
|     "happy-dom": "18.0.1", | ||||
|     "script-loader": "0.7.2", | ||||
|     "vite-plugin-static-copy": "3.1.1" | ||||
|     "vite-plugin-static-copy": "3.1.2" | ||||
|   }, | ||||
|   "nx": { | ||||
|     "name": "client", | ||||
|   | ||||
							
								
								
									
										223
									
								
								apps/client/src/services/ckeditor_plugin_config.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										223
									
								
								apps/client/src/services/ckeditor_plugin_config.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,223 @@ | ||||
| /** | ||||
|  * @module CKEditor Plugin Configuration Service | ||||
|  *  | ||||
|  * This service manages the dynamic configuration of CKEditor plugins based on user preferences. | ||||
|  * It handles plugin enablement, dependency resolution, and toolbar configuration. | ||||
|  */ | ||||
|  | ||||
| import server from "./server.js"; | ||||
| import type {  | ||||
|     PluginConfiguration,  | ||||
|     PluginMetadata,  | ||||
|     PluginRegistry, | ||||
|     PluginValidationResult | ||||
| } from "@triliumnext/commons"; | ||||
|  | ||||
| /** | ||||
|  * Cache for plugin registry and user configuration | ||||
|  */ | ||||
| let pluginRegistryCache: PluginRegistry | null = null; | ||||
| let userConfigCache: PluginConfiguration[] | null = null; | ||||
| let cacheTimestamp = 0; | ||||
| const CACHE_DURATION = 5 * 60 * 1000; // 5 minutes | ||||
|  | ||||
| /** | ||||
|  * Get the plugin registry from server | ||||
|  */ | ||||
| export async function getPluginRegistry(): Promise<PluginRegistry> { | ||||
|     const now = Date.now(); | ||||
|     if (pluginRegistryCache && (now - cacheTimestamp) < CACHE_DURATION) { | ||||
|         return pluginRegistryCache; | ||||
|     } | ||||
|      | ||||
|     try { | ||||
|         pluginRegistryCache = await server.get<PluginRegistry>('ckeditor-plugins/registry'); | ||||
|         cacheTimestamp = now; | ||||
|         return pluginRegistryCache; | ||||
|     } catch (error) { | ||||
|         console.error('Failed to load CKEditor plugin registry:', error); | ||||
|         throw error; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get the user's plugin configuration from server | ||||
|  */ | ||||
| export async function getUserPluginConfig(): Promise<PluginConfiguration[]> { | ||||
|     const now = Date.now(); | ||||
|     if (userConfigCache && (now - cacheTimestamp) < CACHE_DURATION) { | ||||
|         return userConfigCache; | ||||
|     } | ||||
|      | ||||
|     try { | ||||
|         userConfigCache = await server.get<PluginConfiguration[]>('ckeditor-plugins/config'); | ||||
|         cacheTimestamp = now; | ||||
|         return userConfigCache; | ||||
|     } catch (error) { | ||||
|         console.error('Failed to load user plugin configuration:', error); | ||||
|         throw error; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Clear the cache (call when configuration is updated) | ||||
|  */ | ||||
| export function clearCache(): void { | ||||
|     pluginRegistryCache = null; | ||||
|     userConfigCache = null; | ||||
|     cacheTimestamp = 0; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get the enabled plugins for the current user | ||||
|  */ | ||||
| export async function getEnabledPlugins(): Promise<Set<string>> { | ||||
|     const userConfig = await getUserPluginConfig(); | ||||
|     const enabledPlugins = new Set<string>(); | ||||
|      | ||||
|     // Add all enabled user plugins | ||||
|     userConfig.forEach(config => { | ||||
|         if (config.enabled) { | ||||
|             enabledPlugins.add(config.id); | ||||
|         } | ||||
|     }); | ||||
|      | ||||
|     // Always include core plugins | ||||
|     const registry = await getPluginRegistry(); | ||||
|     Object.values(registry.plugins).forEach(plugin => { | ||||
|         if (plugin.isCore) { | ||||
|             enabledPlugins.add(plugin.id); | ||||
|         } | ||||
|     }); | ||||
|      | ||||
|     return enabledPlugins; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get disabled plugin names for CKEditor config | ||||
|  */ | ||||
| export async function getDisabledPlugins(): Promise<string[]> { | ||||
|     try { | ||||
|         const registry = await getPluginRegistry(); | ||||
|         const enabledPlugins = await getEnabledPlugins(); | ||||
|         const disabledPlugins: string[] = []; | ||||
|          | ||||
|         // Find plugins that are disabled | ||||
|         Object.values(registry.plugins).forEach(plugin => { | ||||
|             if (!plugin.isCore && !enabledPlugins.has(plugin.id)) { | ||||
|                 // Map plugin ID to actual CKEditor plugin names if needed | ||||
|                 const pluginNames = getPluginNames(plugin.id); | ||||
|                 disabledPlugins.push(...pluginNames); | ||||
|             } | ||||
|         }); | ||||
|          | ||||
|         return disabledPlugins; | ||||
|     } catch (error) { | ||||
|         console.warn("Failed to get disabled plugins, returning empty list:", error); | ||||
|         return []; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Map plugin ID to actual CKEditor plugin names | ||||
|  * Some plugins might have multiple names or different names than their ID | ||||
|  */ | ||||
| function getPluginNames(pluginId: string): string[] { | ||||
|     const nameMap: Record<string, string[]> = { | ||||
|         "emoji": ["EmojiMention", "EmojiPicker"], | ||||
|         "math": ["Math", "AutoformatMath"], | ||||
|         "image": ["Image", "ImageCaption", "ImageInline", "ImageResize", "ImageStyle", "ImageToolbar", "ImageUpload"], | ||||
|         "table": ["Table", "TableToolbar", "TableProperties", "TableCellProperties", "TableSelection", "TableCaption", "TableColumnResize"], | ||||
|         "font": ["Font", "FontColor", "FontBackgroundColor"], | ||||
|         "list": ["List", "ListProperties"], | ||||
|         "specialcharacters": ["SpecialCharacters", "SpecialCharactersEssentials"], | ||||
|         "findandreplace": ["FindAndReplace"], | ||||
|         "horizontalline": ["HorizontalLine"], | ||||
|         "pagebreak": ["PageBreak"], | ||||
|         "removeformat": ["RemoveFormat"], | ||||
|         "alignment": ["Alignment"], | ||||
|         "indent": ["Indent", "IndentBlock"], | ||||
|         "codeblock": ["CodeBlock"], | ||||
|         "blockquote": ["BlockQuote"], | ||||
|         "todolist": ["TodoList"], | ||||
|         "heading": ["Heading", "HeadingButtonsUI"], | ||||
|         "paragraph": ["ParagraphButtonUI"], | ||||
|         // Add more mappings as needed | ||||
|     }; | ||||
|      | ||||
|     return nameMap[pluginId] || [pluginId.charAt(0).toUpperCase() + pluginId.slice(1)]; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Validate the current plugin configuration | ||||
|  */ | ||||
| export async function validatePluginConfiguration(): Promise<PluginValidationResult> { | ||||
|     try { | ||||
|         const userConfig = await getUserPluginConfig(); | ||||
|         return await server.post<PluginValidationResult>('ckeditor-plugins/validate', { | ||||
|             plugins: userConfig | ||||
|         }); | ||||
|     } catch (error) { | ||||
|         console.error('Failed to validate plugin configuration:', error); | ||||
|         return { | ||||
|             valid: false, | ||||
|             errors: [{ | ||||
|                 type: "missing_dependency", | ||||
|                 pluginId: "unknown", | ||||
|                 message: `Validation failed: ${error}` | ||||
|             }], | ||||
|             warnings: [], | ||||
|             resolvedPlugins: [] | ||||
|         }; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get toolbar items that should be hidden based on disabled plugins | ||||
|  */ | ||||
| export async function getHiddenToolbarItems(): Promise<string[]> { | ||||
|     const registry = await getPluginRegistry(); | ||||
|     const enabledPlugins = await getEnabledPlugins(); | ||||
|     const hiddenItems: string[] = []; | ||||
|      | ||||
|     Object.values(registry.plugins).forEach(plugin => { | ||||
|         if (!enabledPlugins.has(plugin.id) && plugin.toolbarItems) { | ||||
|             hiddenItems.push(...plugin.toolbarItems); | ||||
|         } | ||||
|     }); | ||||
|      | ||||
|     return hiddenItems; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Update user plugin configuration | ||||
|  */ | ||||
| export async function updatePluginConfiguration(plugins: PluginConfiguration[]): Promise<void> { | ||||
|     try { | ||||
|         const response = await server.put('ckeditor-plugins/config', { | ||||
|             plugins, | ||||
|             validate: true | ||||
|         }); | ||||
|          | ||||
|         if (!response.success) { | ||||
|             throw new Error(response.errors?.join(", ") || "Update failed"); | ||||
|         } | ||||
|          | ||||
|         // Clear cache so next requests fetch fresh data | ||||
|         clearCache(); | ||||
|     } catch (error) { | ||||
|         console.error('Failed to update plugin configuration:', error); | ||||
|         throw error; | ||||
|     } | ||||
| } | ||||
|  | ||||
| export default { | ||||
|     getPluginRegistry, | ||||
|     getUserPluginConfig, | ||||
|     getEnabledPlugins, | ||||
|     getDisabledPlugins, | ||||
|     getHiddenToolbarItems, | ||||
|     validatePluginConfiguration, | ||||
|     updatePluginConfiguration, | ||||
|     clearCache | ||||
| }; | ||||
| @@ -28,6 +28,28 @@ | ||||
|     --ck-mention-list-max-height: 500px; | ||||
| } | ||||
|  | ||||
| body#trilium-app.motion-disabled *, | ||||
| body#trilium-app.motion-disabled *::before, | ||||
| body#trilium-app.motion-disabled *::after { | ||||
|     /* Disable transitions and animations */ | ||||
|     transition: none !important; | ||||
|     animation: none !important; | ||||
| } | ||||
|  | ||||
| body#trilium-app.shadows-disabled *, | ||||
| body#trilium-app.shadows-disabled *::before, | ||||
| body#trilium-app.shadows-disabled *::after { | ||||
|     /* Disable shadows */ | ||||
|     box-shadow: none !important; | ||||
| } | ||||
|  | ||||
| body#trilium-app.backdrop-effects-disabled *, | ||||
| body#trilium-app.backdrop-effects-disabled *::before, | ||||
| body#trilium-app.backdrop-effects-disabled *::after { | ||||
|     /* Disable backdrop effects */ | ||||
|     backdrop-filter: none !important; | ||||
| } | ||||
|  | ||||
| .table { | ||||
|     --bs-table-bg: transparent !important; | ||||
| } | ||||
| @@ -355,7 +377,7 @@ body.desktop .tabulator-popup-container { | ||||
|  | ||||
| @supports (animation-fill-mode: forwards) { | ||||
|     /* Delay the opening of submenus */ | ||||
|     body.desktop .dropdown-submenu .dropdown-menu { | ||||
|     body.desktop:not(.motion-disabled) .dropdown-submenu .dropdown-menu { | ||||
|         opacity: 0; | ||||
|         animation-fill-mode: forwards; | ||||
|         animation-delay: var(--submenu-opening-delay); | ||||
|   | ||||
| @@ -89,6 +89,7 @@ | ||||
|  | ||||
|     --menu-text-color: #e3e3e3; | ||||
|     --menu-background-color: #222222d9; | ||||
|     --menu-background-color-no-backdrop: #1b1b1b; | ||||
|     --menu-item-icon-color: #8c8c8c; | ||||
|     --menu-item-disabled-opacity: 0.5; | ||||
|     --menu-item-keyboard-shortcut-color: #ffffff8f; | ||||
|   | ||||
| @@ -83,6 +83,7 @@ | ||||
|  | ||||
|     --menu-text-color: #272727; | ||||
|     --menu-background-color: #ffffffd9; | ||||
|     --menu-background-color-no-backdrop: #fdfdfd; | ||||
|     --menu-item-icon-color: #727272; | ||||
|     --menu-item-disabled-opacity: 0.6; | ||||
|     --menu-item-keyboard-shortcut-color: #666666a8; | ||||
|   | ||||
| @@ -83,6 +83,12 @@ | ||||
|     --tab-note-icons: true; | ||||
| } | ||||
|  | ||||
| body.backdrop-effects-disabled { | ||||
|     /* Backdrop effects are disabled, replace the menu background color with the | ||||
|      * no-backdrop fallback color */ | ||||
|     --menu-background-color: var(--menu-background-color-no-backdrop); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * MENUS | ||||
|  * | ||||
|   | ||||
| @@ -1871,7 +1871,12 @@ | ||||
|     "selected_provider": "已选提供商", | ||||
|     "selected_provider_description": "选择用于聊天和补全功能的AI提供商", | ||||
|     "select_model": "选择模型...", | ||||
|     "select_provider": "选择提供商..." | ||||
|     "select_provider": "选择提供商...", | ||||
|     "ai_enabled": "已启用 AI 功能", | ||||
|     "ai_disabled": "已禁用 AI 功能", | ||||
|     "no_models_found_online": "找不到模型。请检查您的 API 密钥及设置。", | ||||
|     "no_models_found_ollama": "找不到 Ollama 模型。请确认 Ollama 是否正在运行。", | ||||
|     "error_fetching": "获取模型失败:{{error}}" | ||||
|   }, | ||||
|   "code-editor-options": { | ||||
|     "title": "编辑器" | ||||
| @@ -1999,5 +2004,21 @@ | ||||
|     "next_theme_message": "当前使用旧版主题,要试用新主题吗?", | ||||
|     "next_theme_button": "试用新主题", | ||||
|     "dismiss": "关闭" | ||||
|   }, | ||||
|   "settings": { | ||||
|     "related_settings": "相关设置" | ||||
|   }, | ||||
|   "settings_appearance": { | ||||
|     "related_code_blocks": "文本笔记中代码块的色彩方案", | ||||
|     "related_code_notes": "代码笔记的色彩方案" | ||||
|   }, | ||||
|   "units": { | ||||
|     "percentage": "%" | ||||
|   }, | ||||
|   "ui-performance": { | ||||
|     "title": "性能", | ||||
|     "enable-motion": "启用过渡和动画", | ||||
|     "enable-shadows": "启用阴影", | ||||
|     "enable-backdrop-effects": "启用菜单、弹窗和面板的背景效果" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1113,6 +1113,12 @@ | ||||
|     "layout-vertical-description": "launcher bar is on the left (default)", | ||||
|     "layout-horizontal-description": "launcher bar is underneath the tab bar, the tab bar is now full width." | ||||
|   }, | ||||
|   "ui-performance": { | ||||
|     "title": "Performance", | ||||
|     "enable-motion": "Enable transitions and animations", | ||||
|     "enable-shadows": "Enable shadows", | ||||
|     "enable-backdrop-effects": "Enable background effects for menus, popups and panels" | ||||
|   }, | ||||
|   "ai_llm": { | ||||
|     "not_started": "Not started", | ||||
|     "title": "AI Settings", | ||||
| @@ -1808,6 +1814,43 @@ | ||||
|       "multiline-toolbar": "Display the toolbar on multiple lines if it doesn't fit." | ||||
|     } | ||||
|   }, | ||||
|   "ckeditor_plugins": { | ||||
|     "title": "Editor Plugins", | ||||
|     "description": "Configure which CKEditor plugins are enabled. Changes take effect when the editor is reloaded.", | ||||
|     "loading": "Loading plugin configuration...", | ||||
|     "load_failed": "Failed to load plugin configuration.", | ||||
|     "load_error": "Error loading plugins", | ||||
|     "retry": "Retry", | ||||
|     "category_formatting": "Text Formatting", | ||||
|     "category_structure": "Document Structure", | ||||
|     "category_media": "Media & Files", | ||||
|     "category_tables": "Tables", | ||||
|     "category_advanced": "Advanced Features", | ||||
|     "category_trilium": "Trilium Features", | ||||
|     "category_external": "External Plugins", | ||||
|     "stats_enabled": "Enabled", | ||||
|     "stats_total": "Total", | ||||
|     "stats_core": "Core", | ||||
|     "stats_premium": "Premium", | ||||
|     "no_license": "no license", | ||||
|     "premium": "Premium", | ||||
|     "premium_required": "Requires premium CKEditor license", | ||||
|     "has_dependencies": "Dependencies", | ||||
|     "depends_on": "Depends on", | ||||
|     "toolbar_items": "Toolbar items", | ||||
|     "validate": "Validate", | ||||
|     "validation_error": "Validation failed", | ||||
|     "validation_errors": "Configuration Errors:", | ||||
|     "validation_warnings": "Configuration Warnings:", | ||||
|     "save": "Save Changes", | ||||
|     "save_success": "Plugin configuration saved successfully", | ||||
|     "save_error": "Failed to save configuration", | ||||
|     "reload_editor_notice": "Please reload any open text notes to apply changes", | ||||
|     "reset_defaults": "Reset to Defaults", | ||||
|     "reset_confirm": "Are you sure you want to reset all plugin settings to their default values?", | ||||
|     "reset_success": "Plugin configuration reset to defaults", | ||||
|     "reset_error": "Failed to reset configuration" | ||||
|   }, | ||||
|   "electron_context_menu": { | ||||
|     "add-term-to-dictionary": "Add \"{{term}}\" to dictionary", | ||||
|     "cut": "Cut", | ||||
|   | ||||
| @@ -1680,6 +1680,16 @@ | ||||
|     "n_notes_queued_2": "", | ||||
|     "notes_indexed_0": "{{ count }} note indexée", | ||||
|     "notes_indexed_1": "{{ count }} notes indexées", | ||||
|     "notes_indexed_2": "" | ||||
|     "notes_indexed_2": "", | ||||
|     "anthropic_url_description": "URL de base pour l'API Anthropic (par défaut : https ://api.anthropic.com)", | ||||
|     "anthropic_model_description": "Modèles Anthropic Claude pour la complétion", | ||||
|     "voyage_settings": "Réglages d'IA Voyage", | ||||
|     "ollama_settings": "Réglages Ollama", | ||||
|     "ollama_url_description": "URL pour l'API Ollama (par défaut: http://localhost:11434)", | ||||
|     "ollama_model_description": "Model Ollama utilisé pour la complétion", | ||||
|     "anthropic_configuration": "Configuration Anthropic", | ||||
|     "voyage_configuration": "Configuration IA Voyage", | ||||
|     "voyage_url_description": "Défaut: https://api.voyageai.com/v1", | ||||
|     "ollama_configuration": "Configuration Ollama" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -692,7 +692,8 @@ | ||||
|     "placeholder_search": "ノート名で検索", | ||||
|     "dialog_title": "埋め込みノート", | ||||
|     "box_size_prompt": "埋め込みノート枠のサイズ:", | ||||
|     "button_include": "埋め込みノート" | ||||
|     "button_include": "埋め込みノート", | ||||
|     "label_note": "ノート" | ||||
|   }, | ||||
|   "ancestor": { | ||||
|     "placeholder": "ノート名で検索" | ||||
|   | ||||
							
								
								
									
										29
									
								
								apps/client/src/translations/ko/translation.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								apps/client/src/translations/ko/translation.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| { | ||||
|   "about": { | ||||
|     "title": "Trilium Notes에 대해서", | ||||
|     "homepage": "홈페이지:", | ||||
|     "app_version": "앱 버전:", | ||||
|     "db_version": "DB 버전:", | ||||
|     "sync_version": "동기화 버전:", | ||||
|     "build_date": "빌드 날짜:", | ||||
|     "build_revision": "빌드 리비전:", | ||||
|     "data_directory": "데이터 경로:" | ||||
|   }, | ||||
|   "toast": { | ||||
|     "critical-error": { | ||||
|       "title": "심각한 오류", | ||||
|       "message": "클라이언트 애플리케이션 시작 도중 심각한 오류가 발생했습니다:\n\n{{message}}\n\n이는 스크립트가 예기치 않게 실패하면서 발생한 것일 수 있습니다. 애플리케이션을 안전 모드로 시작한 뒤 문제를 해결해 보세요." | ||||
|     }, | ||||
|     "widget-error": { | ||||
|       "title": "위젯 초기화 실패" | ||||
|     } | ||||
|   }, | ||||
|   "add_link": { | ||||
|     "add_link": "링크 추가", | ||||
|     "note": "노트", | ||||
|     "search_note": "이름으로 노트 검색하기" | ||||
|   }, | ||||
|   "branch_prefix": { | ||||
|     "save": "저장" | ||||
|   } | ||||
| } | ||||
| @@ -78,7 +78,121 @@ | ||||
|     "n_notes_queued_2": "{{ count }} notas enfileiradas para indexação", | ||||
|     "notes_indexed_0": "{{ count }} nota indexada", | ||||
|     "notes_indexed_1": "{{ count }} notas indexadas", | ||||
|     "notes_indexed_2": "{{ count }} notas indexadas" | ||||
|     "notes_indexed_2": "{{ count }} notas indexadas", | ||||
|     "temperature": "Temperatura", | ||||
|     "retry_queued": "Nota enfileirada para nova tentativa", | ||||
|     "queued_notes": "Notas Enfileiradas", | ||||
|     "empty_key_warning": { | ||||
|       "voyage": "A chave de API da Voyage API está vazia. Por favor, digite uma chave de API válida.", | ||||
|       "ollama": "A chave de API da Ollama API está vazia. Por favor, digite uma chave de API válida.", | ||||
|       "anthropic": "A chave de API Anthropic está vazia. Por favor, digite uma chave de API válida.", | ||||
|       "openai": "A chave de API OpenAI está vazia. Por favor, digite uma chave de API válida." | ||||
|     }, | ||||
|     "not_started": "Não iniciado", | ||||
|     "title": "Configurações de IA", | ||||
|     "processed_notes": "Notas Processadas", | ||||
|     "total_notes": "Total de Notas", | ||||
|     "progress": "Andamento", | ||||
|     "failed_notes": "Notas com Falha", | ||||
|     "last_processed": "Últimas Processadas", | ||||
|     "refresh_stats": "Atualizar Estatísticas", | ||||
|     "enable_ai_features": "Ativar recurso IA/LLM", | ||||
|     "enable_ai_description": "Ativar recursos IA como sumarização de notas, geração de conteúdo, e outras capacidades de LLM", | ||||
|     "openai_tab": "OpenAI", | ||||
|     "anthropic_tab": "Anthropic", | ||||
|     "voyage_tab": "Voyage AI", | ||||
|     "enable_ai": "Ativar recursos IA/LLM", | ||||
|     "provider_configuration": "Configuração de Provedor de IA", | ||||
|     "system_prompt": "Prompt de Sistema", | ||||
|     "system_prompt_description": "Prompt padrão de sistema usado para todas as interações de IA", | ||||
|     "openai_configuration": "Configuração OpenAI", | ||||
|     "openai_settings": "Opções OpenAI", | ||||
|     "api_key": "Chave de API", | ||||
|     "url": "URL Base", | ||||
|     "model": "Modelo", | ||||
|     "openai_api_key_description": "Sua API Key da OpenAI para acessar os serviços de IA", | ||||
|     "anthropic_api_key_description": "Sua API Key da Anthropic para acessar os modelos Claude", | ||||
|     "default_model": "Modelo Padrão", | ||||
|     "openai_model_description": "Exemplos: gpt-4o, gpt-4-turbo, gpt-3.5-turbo", | ||||
|     "base_url": "URL Base", | ||||
|     "openai_url_description": "Padrão: https://api.openai.com/v1", | ||||
|     "anthropic_settings": "Configurações Anthropic", | ||||
|     "anthropic_url_description": "URL Base da API Anthropic (padrão: https://api.anthropic.com)", | ||||
|     "anthropic_model_description": "Modelos Claude da Anthropic para completar conversas", | ||||
|     "voyage_settings": "Configurações Voyage AI", | ||||
|     "retry": "Tentar novamente", | ||||
|     "retry_failed": "Falha ao enfileirar nota para nova tentativa", | ||||
|     "max_notes_per_llm_query": "Máximo de Notas por Consulta", | ||||
|     "max_notes_per_llm_query_description": "Número máximo de notas similares para incluir no contexto da IA", | ||||
|     "active_providers": "Provedores Ativos", | ||||
|     "disabled_providers": "Provedores Desativados", | ||||
|     "remove_provider": "Remover provedor da pesquisa", | ||||
|     "restore_provider": "Restaurar provedor na pesquisa", | ||||
|     "similarity_threshold": "Tolerância de Similaridade", | ||||
|     "similarity_threshold_description": "Pontuação máxima de similaridade (0-1) para notas a serem incluídas no contexto das consultas de LLM", | ||||
|     "reprocess_index": "Reconstruir Índice de Pesquisa", | ||||
|     "reprocessing_index": "Reconstruindo…", | ||||
|     "reprocess_index_started": "Otimiação do índice de pesquisa iniciado em plano de fundo", | ||||
|     "reprocess_index_error": "Erro ao reconstruir índice de pesquisa", | ||||
|     "index_rebuild_progress": "Andamento da Reconstrução do Índice", | ||||
|     "index_rebuilding": "Otimizando índice ({{percentage}}%)", | ||||
|     "index_rebuild_complete": "Otimização de índice finalizada", | ||||
|     "index_rebuild_status_error": "Erro ao verificar o estado da reconstrução do índice", | ||||
|     "never": "Nunca", | ||||
|     "processing": "Processando ({{percentage}}%)", | ||||
|     "incomplete": "Incompleto ({{percentage}}%)", | ||||
|     "complete": "Completo (100%)", | ||||
|     "refreshing": "Atualizando…", | ||||
|     "auto_refresh_notice": "Atualizando automaticamente a cada {{seconds}} segundos", | ||||
|     "note_queued_for_retry": "Nota enfileirada para nova tentativa", | ||||
|     "failed_to_retry_note": "Falha ao retentar nota", | ||||
|     "all_notes_queued_for_retry": "Todas as notas com falha enfileiradas para nova tentativa", | ||||
|     "failed_to_retry_all": "Falha ao retentar notas", | ||||
|     "ai_settings": "Configurações IA", | ||||
|     "api_key_tooltip": "Chave de API para acessar o serviço", | ||||
|     "agent": { | ||||
|       "processing": "Processando…", | ||||
|       "thinking": "Pensando…", | ||||
|       "loading": "Carregando…", | ||||
|       "generating": "Gerando…" | ||||
|     }, | ||||
|     "name": "IA", | ||||
|     "openai": "OpenAI", | ||||
|     "use_enhanced_context": "Usar contexto aprimorado", | ||||
|     "enhanced_context_description": "Alimentar IA com mais contexto sobre a nota e suas notas relacionadas para melhores respostas", | ||||
|     "show_thinking": "Exibir pensamento", | ||||
|     "enter_message": "Digite sua mensagem…", | ||||
|     "error_contacting_provider": "Erro ao contatar o provedor de IA. Por favor, verifique suas configurações e sua conexão de internet.", | ||||
|     "error_generating_response": "Erro ao gerar resposta da IA", | ||||
|     "index_all_notes": "Indexar Todas as Notas", | ||||
|     "index_status": "Estado do Índice", | ||||
|     "indexed_notes": "Notas Indexadas", | ||||
|     "indexing_stopped": "Indexação interrompida", | ||||
|     "indexing_in_progress": "Indexação em andamento…", | ||||
|     "last_indexed": "Última Indexada", | ||||
|     "note_chat": "Conversa de Nota", | ||||
|     "sources": "Origens", | ||||
|     "start_indexing": "Iniciar Indexação", | ||||
|     "use_advanced_context": "Usar Contexto Avançado", | ||||
|     "ollama_no_url": "Ollama não está configurado. Por favor, digite uma URL válida.", | ||||
|     "chat": { | ||||
|       "root_note_title": "Conversas IA", | ||||
|       "root_note_content": "Esta nota contém suas conversas com IA salvas.", | ||||
|       "new_chat_title": "Nova Conversa", | ||||
|       "create_new_ai_chat": "Criar nova Conversa IA" | ||||
|     }, | ||||
|     "create_new_ai_chat": "Criar nova Conversa IA", | ||||
|     "configuration_warnings": "Existem alguns problemas com sua configuração de IA. Por fovor, verifique suas configurações.", | ||||
|     "experimental_warning": "O recurso de LLM atualmente é experimental - você foi avisado.", | ||||
|     "selected_provider": "Provedor Selecionado", | ||||
|     "selected_provider_description": "Escolha o provedor de IA para conversas e recursos de completar", | ||||
|     "select_model": "Selecionar modelo…", | ||||
|     "select_provider": "Selecionar provedor…", | ||||
|     "ai_enabled": "Recursos de IA habilitados", | ||||
|     "ai_disabled": "Recursos de IA desabilitados", | ||||
|     "no_models_found_online": "Nenhum modelo encontrado. Por favor, verifique sua chave de API e as configurações.", | ||||
|     "no_models_found_ollama": "Nenhum modelo Ollama encontrado. Por favor, verifique se o Ollama está em execução.", | ||||
|     "error_fetching": "Erro ao obter modelos: {{error}}" | ||||
|   }, | ||||
|   "confirm": { | ||||
|     "confirmation": "Confirmação", | ||||
| @@ -405,10 +519,688 @@ | ||||
|     "share_index": "notas com este rótulo irão listar todas as raízes das notas compartilhadas", | ||||
|     "display_relations": "nomes das relações separados por vírgula que devem ser exibidos. Todas as outras serão ocultadas.", | ||||
|     "hide_relations": "nomes das relações separados por vírgula que devem ser ocultados. Todas as outras serão exibidas.", | ||||
|     "title_template": "Título padrão das notas criadas como filhas desta nota. O valor é avaliado como uma string JavaScript e pode ser enriquecido com conteúdo dinâmico usando as variáveis injetadas <code>now</code> e <code>parentNote</code>. Exemplos:\n\n<ul>\n    <li><code>${parentNote.getLabelValue('authorName')}'s literary works</code></li>\n    <li><code>Log for ${now.format('YYYY-MM-DD HH:mm:ss')}</code></li>\n</ul>\n\nVeja a <a href=\"https://triliumnext.github.io/Docs/Wiki/default-note-title.html\">wiki com detalhes</a>, a documentação da API para <a href=\"https://zadam.github.io/trilium/backend_api/Note.html\">parentNote</a> e para <a href=\"https://day.js.org/docs/en/display/format\">now</a> para mais informações.", | ||||
|     "title_template": "título padrão das notas criadas como filhas desta nota. O valor é avaliado como uma string JavaScript \n                        e pode ser enriquecido com conteúdo dinâmico usando as variáveis injetadas <code>now</code> e <code>parentNote</code>. Exemplos:\n                       \n                        <ul>\n                            <li><code>${parentNote.getLabelValue('authorName')}'s literary works</code></li>\n                            <li><code>Log for ${now.format('YYYY-MM-DD HH:mm:ss')}</code></li>\n                        </ul>\n                        \n                        Veja a <a href=\"https://triliumnext.github.io/Docs/Wiki/default-note-title.html\">wiki com detalhes</a>, a documentação da API para <a href=\"https://zadam.github.io/trilium/backend_api/Note.html\">parentNote</a> e para <a href=\"https://day.js.org/docs/en/display/format\">now</a> para mais informações.", | ||||
|     "template": "Esta nota aparecerá na seleção de modelos disponíveis ao criar uma nova nota", | ||||
|     "toc": "<code>#toc</code> ou <code>#toc=show</code> irá forçar a exibição do Sumário, <code>#toc=hide</code> irá forçar que ele fique oculto. Se o rótulo não existir, será considerado o ajuste global", | ||||
|     "color": "define a cor da nota na árvore de notas, links etc. Use qualquer valor de cor CSS válido, como 'red' ou #a13d5f", | ||||
|     "keyboard_shortcut": "Define um atalho de teclado que irá pular imediatamente para esta nota. Exemplo: 'ctrl+alt+e'. É necessário recarregar o frontend para que a alteração tenha efeito." | ||||
|     "keyboard_shortcut": "Define um atalho de teclado que irá pular imediatamente para esta nota. Exemplo: 'ctrl+alt+e'. É necessário recarregar o frontend para que a alteração tenha efeito.", | ||||
|     "hide_highlight_widget": "Ocultar o widget da lista de destaques", | ||||
|     "keep_current_hoisting": "Abrir este link não alterará o destaque, mesmo que a nota não seja exibível na subárvore destacada atual.", | ||||
|     "execute_button": "Titulo do botão que executará a nota de código atual", | ||||
|     "exclude_from_note_map": "Notas com este rótulo ficarão ocultas no Mapa de Notas", | ||||
|     "new_notes_on_top": "Novas notas serão criadas no topo da nota raiz, não na parte inferior.", | ||||
|     "execute_description": "Descrição longa da nota de código atualmente exibida junto ao botão executar", | ||||
|     "print_page_size": "Quando exportando para PDF, altera o tamanho da página. Valores suportados: <code>A0</code>, <code>A1</code>, <code>A2</code>, <code>A3</code>, <code>A4</code>, <code>A5</code>, <code>A6</code>, <code>Legal</code>, <code>Letter</code>, <code>Tabloid</code>, <code>Ledger</code>.", | ||||
|     "and_more": "... e {{count}} mais.", | ||||
|     "other_notes_with_name": "Outras notas com {{attributeType}} igual a \"{{attributeName}}\"", | ||||
|     "color_type": "Cor", | ||||
|     "run_on_note_creation": "executa quando a nota é criada no backend. Use esta relação se quiser executar o script para todas as notas criadas em uma subárvore específica. Neste caso, crie-a na nota raiz da subárvore e torne-a herdável. Uma nova nota criada dentro da subárvore (qualquer profundidade) irá acionar o script.", | ||||
|     "run_on_child_note_creation": "executa quando uma nova nota é criada sob a nota onde esta relação está definida", | ||||
|     "run_on_note_title_change": "executa quando o título da nota é alterado (inclusive na criação de nota)", | ||||
|     "run_on_note_content_change": "executa quando o conteúdo da nota é alterado (inclusive na criação de nota).", | ||||
|     "run_on_note_change": "executa quando a nota é alterada (inclusive na criação de nota). Não incluí alterações no conteúdo", | ||||
|     "run_on_note_deletion": "executa quando a nota está sendo excluída", | ||||
|     "run_on_branch_creation": "executa quando uma ramificação é criada. Ramificação é uma ligação entre nota pai e nota filha e é criado, por exemplo, ao clonar ou mover uma nota.", | ||||
|     "run_on_branch_change": "executa quando uma remificação é atualizada.", | ||||
|     "run_on_attribute_creation": "executa quando um novo atributo é criado para a nota que define esta relação", | ||||
|     "run_on_attribute_change": " executa quando o atributo é alterado na nota que define esta relação. Também é disparado quando o atributo é excluído", | ||||
|     "widget_relation": "o destino desta relação será executado e renderizado como um widget na barra lateral" | ||||
|   }, | ||||
|   "attachments_actions": { | ||||
|     "delete_attachment": "Excluir anexo", | ||||
|     "open_externally": "Abrir externamente", | ||||
|     "open_custom": "Abrir customizado", | ||||
|     "download": "Baixar", | ||||
|     "rename_attachment": "Renomear anexo", | ||||
|     "upload_new_revision": "Enviar nova revisão", | ||||
|     "copy_link_to_clipboard": "Copiar link para a área de transferência", | ||||
|     "convert_attachment_into_note": "Converter anexo para nota", | ||||
|     "upload_success": "Uma nova revisão de anexo foi enviada.", | ||||
|     "upload_failed": "O envio de uma nova revisão de anexo falhou.", | ||||
|     "delete_success": "O anexo '{{title}}' foi excluído.", | ||||
|     "convert_success": "O anexo '{{title}}' foi convertido para uma nota.", | ||||
|     "enter_new_name": "Por favor, digite o novo nome do anexo", | ||||
|     "delete_confirm": "Tem certeza que deseja excluir o anexo '{{title}}'?", | ||||
|     "convert_confirm": "Tem certeza que deseja converter o anexo '{{title}}' em uma nota separada?", | ||||
|     "open_externally_title": "O arquivo será aberto em uma aplicação externa e monitorado por alterações. Você então poderá enviar a versão modificada de volta para o Trilium.", | ||||
|     "open_custom_title": "O arquivo será aberto em uma aplicação externa e monitorado por alterações. Você então poderá enviar a versão modificada de volta para o Trilium.", | ||||
|     "open_externally_detail_page": "A abertura de anexo externamente só está disponível através da página de detalhes. Por favor, primeiro clique nos detalhes do anexo e repita a ação.", | ||||
|     "open_custom_client_only": "A abertura customizada de anexos só pode ser feita usando o cliente de desktop." | ||||
|   }, | ||||
|   "attachment_detail": { | ||||
|     "you_can_also_open": ", você também pode abrir o(a) ", | ||||
|     "open_help_page": "Abrir página de ajuda nos anexos", | ||||
|     "list_of_all_attachments": "Lista de todos os anexos", | ||||
|     "attachment_deleted": "Este anexo foi excluído." | ||||
|   }, | ||||
|   "ancestor": { | ||||
|     "depth_gt": "é maior que {{count}}", | ||||
|     "label": "Ancestral", | ||||
|     "placeholder": "buscar notas pelo nome", | ||||
|     "depth_label": "profundidade", | ||||
|     "depth_doesnt_matter": "não importa", | ||||
|     "depth_eq": "é exatamente {{count}}", | ||||
|     "direct_children": "filho direto", | ||||
|     "depth_lt": "é menor que {{count}}" | ||||
|   }, | ||||
|   "add_relation": { | ||||
|     "add_relation": "Adicionar relação", | ||||
|     "allowed_characters": "Caracteres alfanuméricos, underscore e vírgula são permitidos.", | ||||
|     "relation_name": "nome da relação", | ||||
|     "to": "para", | ||||
|     "target_note": "nota destino", | ||||
|     "create_relation_on_all_matched_notes": "Crie a relação informada em todas as notas correspondentes." | ||||
|   }, | ||||
|   "delete_label": { | ||||
|     "label_name_placeholder": "nome da etiqueta", | ||||
|     "label_name_title": "Caracteres alfanuméricos, underscore e vírgula são permitidos.", | ||||
|     "delete_label": "Excluir etiqueta" | ||||
|   }, | ||||
|   "rename_label": { | ||||
|     "rename_label": "Renomear etiqueta", | ||||
|     "rename_label_from": "Renomear etiqueta de", | ||||
|     "old_name_placeholder": "nome antigo", | ||||
|     "to": "Para", | ||||
|     "new_name_placeholder": "novo nome", | ||||
|     "name_title": "Caracteres alfanuméricos, underscore e vírgula são permitidos." | ||||
|   }, | ||||
|   "execute_script": { | ||||
|     "example_1": "Por exemplo para anexar um texto ao título de uma nota, use este pequeno script:", | ||||
|     "execute_script": "Executar script", | ||||
|     "help_text": "Você pode executar scripts simples nas notas correspondentes.", | ||||
|     "example_2": "Um exemplo mais complexo seria excluir todos os atributos das notas correspondentes:" | ||||
|   }, | ||||
|   "attribute_editor": { | ||||
|     "help_text_body1": "Para adicionar uma etiqueta, digite por exemplo <code>#rock</code> ou se você também quer adicionar um valor então por exemplo <code>#year = 2020</code>", | ||||
|     "help_text_body2": "Para relação, digite <code>~author = @</code>, que deve ser exibido um autocompletar onde você pode encontrar a nota desejada.", | ||||
|     "help_text_body3": "Alternativamente, você pode adicionar etiqueta e relação usando o botão <code>+</code> no lado direito.", | ||||
|     "save_attributes": "Salvar atributos <enter>", | ||||
|     "add_a_new_attribute": "Adicionar um novo atributo", | ||||
|     "add_new_label": "Adicionar nova etiqueta <kbd data-command=\"addNewLabel\"></kbd>", | ||||
|     "add_new_relation": "Adicionar nova relação <kbd data-command=\"addNewRelation\"></kbd>", | ||||
|     "add_new_label_definition": "Adicionar nova definição de etiqueta", | ||||
|     "add_new_relation_definition": "Adicionar nova definição de relação", | ||||
|     "placeholder": "Digite as etiquetas e relações aqui" | ||||
|   }, | ||||
|   "abstract_bulk_action": { | ||||
|     "remove_this_search_action": "Remover esta ação de busca" | ||||
|   }, | ||||
|   "add_label": { | ||||
|     "add_label": "Adicionar etiqueta", | ||||
|     "label_name_placeholder": "nome da etiqueta", | ||||
|     "label_name_title": "Caracteres alfanuméricos, underscore e vírgula são permitidos.", | ||||
|     "to_value": "para o valor", | ||||
|     "new_value_placeholder": "novo valor", | ||||
|     "help_text": "Em todas as notas correspondentes:", | ||||
|     "help_text_item1": "criar a etiqueta indicada se a nota ainda não tiver uma", | ||||
|     "help_text_item2": "ou altere o valor da etiqueta existente", | ||||
|     "help_text_note": "Você também pode chamar este método sem valor, neste caso a etiqueta será atribuída à nota sem valor." | ||||
|   }, | ||||
|   "update_label_value": { | ||||
|     "update_label_value": "Atualizar valor da etiqueta", | ||||
|     "label_name_placeholder": "nome da etiqueta", | ||||
|     "label_name_title": "Caracteres alfanuméricos, underscore e vírgula são permitidos.", | ||||
|     "new_value_placeholder": "novo valor", | ||||
|     "to_value": "para o valor", | ||||
|     "help_text": "Em todas as notas correspondentes, altera o valor da etiqueta existente.", | ||||
|     "help_text_note": "Você também pode chamar este método sem um valor, neste caso a etiqueta será à nota sem um valor." | ||||
|   }, | ||||
|   "delete_relation": { | ||||
|     "allowed_characters": "Caracteres alfanuméricos, underscore e vírgula são permitidos.", | ||||
|     "delete_relation": "Excluir relação", | ||||
|     "relation_name": "nome da relação" | ||||
|   }, | ||||
|   "rename_relation": { | ||||
|     "allowed_characters": "Caracteres alfanuméricos, underscore e vírgula são permitidos.", | ||||
|     "rename_relation": "Renomar relação", | ||||
|     "rename_relation_from": "Renomear relação de", | ||||
|     "old_name": "nome antigo", | ||||
|     "to": "Para", | ||||
|     "new_name": "novo nome" | ||||
|   }, | ||||
|   "update_relation_target": { | ||||
|     "allowed_characters": "Caracteres alfanuméricos, underscore e vírgula são permitidos.", | ||||
|     "to": "para", | ||||
|     "target_note": "nota destino", | ||||
|     "on_all_matched_notes": "Em todas as notas correspondentes", | ||||
|     "change_target_note": "alterar nota destino da relação existente", | ||||
|     "update_relation_target": "Atualizar destino da relação", | ||||
|     "update_relation": "Atualizar relação", | ||||
|     "relation_name": "nome da relação" | ||||
|   }, | ||||
|   "content_renderer": { | ||||
|     "open_externally": "Abrir externamente" | ||||
|   }, | ||||
|   "modal": { | ||||
|     "close": "Fechar" | ||||
|   }, | ||||
|   "api_log": { | ||||
|     "close": "Fechar" | ||||
|   }, | ||||
|   "attachment_detail_2": { | ||||
|     "will_be_deleted_in": "Este anexo será excluído automaticamente em {{time}}", | ||||
|     "will_be_deleted_soon": "Este anexo será excluído automaticamente em breve", | ||||
|     "deletion_reason": ", porque o anexo não está associado ao conteúdo da nota. Para evitar a exclusão, adicione o anexo novamente ao conteúdo ou converta o anexo em uma nota.", | ||||
|     "role_and_size": "Regra: {{role}}, Tamanho: {{size}}", | ||||
|     "link_copied": "Link do anexo copiado para a área de transferência.", | ||||
|     "unrecognized_role": "Regra desconhecida de anexo '{{role}}'." | ||||
|   }, | ||||
|   "bookmark_switch": { | ||||
|     "bookmark": "Favorito", | ||||
|     "bookmark_this_note": "Favoritar esta nota no painel da esquerda", | ||||
|     "remove_bookmark": "Remover favorito" | ||||
|   }, | ||||
|   "editability_select": { | ||||
|     "auto": "Auto", | ||||
|     "read_only": "Somente leitura", | ||||
|     "always_editable": "Sempre Editável", | ||||
|     "note_is_editable": "A nota é editável se não for muito longa.", | ||||
|     "note_is_read_only": "A nota é somente leitura, mas pode ser editada com um clique no botão.", | ||||
|     "note_is_always_editable": "A nota é sempre editável, independentemente do seu tamanho." | ||||
|   }, | ||||
|   "note-map": { | ||||
|     "button-link-map": "Mapa de Links", | ||||
|     "button-tree-map": "Mapa em Árvore" | ||||
|   }, | ||||
|   "tree-context-menu": { | ||||
|     "open-in-a-new-tab": "Abrir em uma nova aba <kbd>Ctrl+Click</kbd>", | ||||
|     "open-in-a-new-split": "Abrir em um novo painel dividido", | ||||
|     "insert-note-after": "Inserir nota após", | ||||
|     "insert-child-note": "Inserir nota filha", | ||||
|     "delete": "Excluir", | ||||
|     "search-in-subtree": "Buscar na subárvore" | ||||
|   }, | ||||
|   "command_palette": { | ||||
|     "search_subtree_title": "Buscar na Subárvore", | ||||
|     "search_subtree_description": "Buscar dentro da subárvore atual", | ||||
|     "search_history_title": "Exibir Histórico de Busca", | ||||
|     "search_history_description": "Visualizar buscas anteriores", | ||||
|     "configure_launch_bar_title": "Configurar Barra de Execução" | ||||
|   }, | ||||
|   "delete_note": { | ||||
|     "delete_note": "Excluir nota", | ||||
|     "delete_matched_notes": "Excluir notas correspondentes", | ||||
|     "delete_matched_notes_description": "Isso irá excluir as notas correspondentes.", | ||||
|     "undelete_notes_instruction": "Depois da exclusão, é possível desfazer através da janela de Alterações Recentes.", | ||||
|     "erase_notes_instruction": "Para apagar notas permanentemente, você pode fazer isso depois da exclusão indo em Opções -> Outros e clicar no botão \"Apagar notas excluídas agora\"." | ||||
|   }, | ||||
|   "delete_revisions": { | ||||
|     "delete_note_revisions": "Excluir revisões da nota", | ||||
|     "all_past_note_revisions": "Todas as revisões anteriores das notas correspondentes serão excluídas. A nota em si será perservada. Ou seja, o histórico da nota será removido." | ||||
|   }, | ||||
|   "move_note": { | ||||
|     "move_note": "Mover nota", | ||||
|     "to": "para", | ||||
|     "target_parent_note": "nota pai destino", | ||||
|     "on_all_matched_notes": "Em todas as notas correspondentes" | ||||
|   }, | ||||
|   "rename_note": { | ||||
|     "rename_note": "Renomear nota", | ||||
|     "rename_note_title_to": "Renomear título da nota para", | ||||
|     "new_note_title": "novo título da nota", | ||||
|     "click_help_icon": "Clique no ícone de ajuda a direita para ver todas as opções", | ||||
|     "example_note": "<code>Nota</code> - todas as notas correspondentes serão renomeadas para 'Nota'", | ||||
|     "example_new_title": "<code>NOVO: ${note.title}</code> - o título das notas correspondentes receberá o prefixo 'NOVO: '", | ||||
|     "example_date_prefix": "<code>${note.dateCreatedObj.format('MM-DD:')}: ${note.title}</code> - notas correspondentes receberão um prefixo com o mês-dia da data de criação da nota", | ||||
|     "api_docs": "Veja da documentação da API para <a href='https://zadam.github.io/trilium/backend_api/Note.html'>nota</a> e suas <a href='https://day.js.org/docs/en/display/format'>propriedades dateCreatedObj / utcDateCreatedObj</a> para detalhes." | ||||
|   }, | ||||
|   "calendar": { | ||||
|     "mon": "Seg", | ||||
|     "tue": "Ter", | ||||
|     "wed": "Qua", | ||||
|     "thu": "Qui", | ||||
|     "fri": "Sex", | ||||
|     "sat": "Sáb", | ||||
|     "sun": "Dom", | ||||
|     "cannot_find_day_note": "Nota do dia não encontrada", | ||||
|     "cannot_find_week_note": "Nota semanal não encontrada", | ||||
|     "january": "Janeiro", | ||||
|     "febuary": "Fevereiro", | ||||
|     "march": "Março", | ||||
|     "april": "Abril", | ||||
|     "may": "Maio", | ||||
|     "june": "Junho", | ||||
|     "july": "Julho", | ||||
|     "august": "Agosto", | ||||
|     "september": "Setembro", | ||||
|     "october": "Outubro", | ||||
|     "november": "Novembro", | ||||
|     "december": "Dezembro" | ||||
|   }, | ||||
|   "close_pane_button": { | ||||
|     "close_this_pane": "Fechar este painel" | ||||
|   }, | ||||
|   "create_pane_button": { | ||||
|     "create_new_split": "Criar nova divisão" | ||||
|   }, | ||||
|   "edit_button": { | ||||
|     "edit_this_note": "Editar esta nota" | ||||
|   }, | ||||
|   "show_toc_widget_button": { | ||||
|     "show_toc": "Mostrar Tabela de Conteúdo" | ||||
|   }, | ||||
|   "show_highlights_list_widget_button": { | ||||
|     "show_highlights_list": "Mostrar Lista de Destaques" | ||||
|   }, | ||||
|   "global_menu": { | ||||
|     "menu": "Menu", | ||||
|     "options": "Opções", | ||||
|     "open_new_window": "Abrir Nova Janela", | ||||
|     "switch_to_mobile_version": "Alternar para Versão Mobile", | ||||
|     "switch_to_desktop_version": "Alternar para Versão Desktop", | ||||
|     "zoom": "Zoom", | ||||
|     "toggle_fullscreen": "Alternar Tela Cheia", | ||||
|     "zoom_out": "Reduzir", | ||||
|     "reset_zoom_level": "Redefinir Zoom", | ||||
|     "zoom_in": "Aumentar", | ||||
|     "configure_launchbar": "Configurar Barra de Lançamento", | ||||
|     "show_shared_notes_subtree": "Exibir Subárvore de Notas Compartilhadas", | ||||
|     "advanced": "Avançado", | ||||
|     "open_dev_tools": "Abrir Ferramentas de Desenvolvedor", | ||||
|     "open_sql_console": "Abrir Console SQL", | ||||
|     "open_sql_console_history": "Abrir Histórico de Console SQL", | ||||
|     "open_search_history": "Abrir Histórico de Busca", | ||||
|     "show_backend_log": "Abrir Log do Servidor", | ||||
|     "reload_frontend": "Recarregar Frontend", | ||||
|     "show_hidden_subtree": "Exibir Subárvore Oculta", | ||||
|     "show_help": "Exibir Ajuda", | ||||
|     "about": "Sobre o Trilium Notes", | ||||
|     "logout": "Sair", | ||||
|     "show-cheatsheet": "Exibir Cheatsheet", | ||||
|     "toggle-zen-mode": "Modo Zen", | ||||
|     "reload_hint": "Recarregar pode ajudar com alguns problemas visuais sem reiniciar toda a aplicação." | ||||
|   }, | ||||
|   "zen_mode": { | ||||
|     "button_exit": "Sair do Modo Zen" | ||||
|   }, | ||||
|   "sync_status": { | ||||
|     "in_progress": "Sincronização com o servidor em andamento.", | ||||
|     "unknown": "<p>O estado da sincronização será conhecido assim que a próxima tentativa começar.</p><p>Clique para iniciar a sincronização agora.</p>", | ||||
|     "connected_with_changes": "<p>Conectado ao servidor de sincronização.<br>Existem algumas alterações esperando para serem sincronizadas.</p><p>Clique para sincronizar.</p>", | ||||
|     "connected_no_changes": "<p>Conectado ao servidor de sincronização.<br>Todas as alterações já foram sincronizadas.</p><p>Clique para sincronizar.</p>", | ||||
|     "disconnected_with_changes": "<p>A conexão ao servidor de sincronização falhou.<br>Existem algumas alterações esperando para serem sincronizadas.</p><p>Clique para sincronizar.</p>", | ||||
|     "disconnected_no_changes": "<p>A conexão ao servidor de sincronização falhou.<br>Todas as alterações já foram sincronizadas.</p><p>Clique para sincronizar.</p>" | ||||
|   }, | ||||
|   "left_pane_toggle": { | ||||
|     "show_panel": "Exibir painel", | ||||
|     "hide_panel": "Esconder painel" | ||||
|   }, | ||||
|   "move_pane_button": { | ||||
|     "move_left": "Mover para a esquerda", | ||||
|     "move_right": "Mover para a direita" | ||||
|   }, | ||||
|   "note_actions": { | ||||
|     "convert_into_attachment": "Converter para anexo", | ||||
|     "re_render_note": "Renderizar nota novamente", | ||||
|     "search_in_note": "Buscar na nota", | ||||
|     "note_source": "Código Fonte da nota", | ||||
|     "note_attachments": "Anexos da nota", | ||||
|     "open_note_externally": "Abrir nota externamente", | ||||
|     "open_note_custom": "Abrir nota de forma customizada", | ||||
|     "import_files": "Importar arquivos", | ||||
|     "export_note": "Exportar nota", | ||||
|     "delete_note": "Excluir nota", | ||||
|     "print_note": "Imprimir nota", | ||||
|     "save_revision": "Salvar revisão", | ||||
|     "convert_into_attachment_failed": "A conversão da nota '{{title}}' falhou.", | ||||
|     "convert_into_attachment_successful": "A nota '{{title}}' foi convertida para anexo.", | ||||
|     "print_pdf": "Exportar como PDF…", | ||||
|     "open_note_externally_title": "O arquivo será aberto em uma aplicação externa e monitorado por alterações. Você então poderá enviar a versão modificada de volta para o Trilium.", | ||||
|     "convert_into_attachment_prompt": "Você tem certeza que quer converter a nota '{{title}}' em um anexo da nota pai?" | ||||
|   }, | ||||
|   "protected_session_status": { | ||||
|     "inactive": "Clique para entrar na sessão protegida", | ||||
|     "active": "Sessão protegida está ativada. Clique para deixar a sessão protegida." | ||||
|   }, | ||||
|   "revisions_button": { | ||||
|     "note_revisions": "Revisões da Nota" | ||||
|   }, | ||||
|   "update_available": { | ||||
|     "update_available": "Atualização disponível" | ||||
|   }, | ||||
|   "code_buttons": { | ||||
|     "execute_button_title": "Executar script", | ||||
|     "trilium_api_docs_button_title": "Abrir documentação da Trilium API", | ||||
|     "save_to_note_button_title": "Salvar para uma nota", | ||||
|     "opening_api_docs_message": "Abrindo documentação da API…", | ||||
|     "sql_console_saved_message": "Nota do Console SQL foi salva no caminho {{note_path}}" | ||||
|   }, | ||||
|   "hide_floating_buttons_button": { | ||||
|     "button_title": "Esconder botões" | ||||
|   }, | ||||
|   "show_floating_buttons_button": { | ||||
|     "button_title": "Exibir botões" | ||||
|   }, | ||||
|   "svg_export_button": { | ||||
|     "button_title": "Exportar diagrama como SVG" | ||||
|   }, | ||||
|   "relation_map_buttons": { | ||||
|     "zoom_in_title": "Aumentar", | ||||
|     "zoom_out_title": "Reduzir", | ||||
|     "create_child_note_title": "Criar nova nota filha e adicione neste mapa de relação" | ||||
|   }, | ||||
|   "zpetne_odkazy": { | ||||
|     "backlink": "{{count}} Links Reversos", | ||||
|     "backlinks": "{{count}} Links Reversos", | ||||
|     "relation": "relação" | ||||
|   }, | ||||
|   "mobile_detail_menu": { | ||||
|     "insert_child_note": "Inserir nota filha", | ||||
|     "delete_this_note": "Excluir essa nota", | ||||
|     "error_unrecognized_command": "Comando não reconhecido {{command}}" | ||||
|   }, | ||||
|   "note_icon": { | ||||
|     "change_note_icon": "Alterar ícone da nota", | ||||
|     "category": "Categoria:", | ||||
|     "search": "Busca:", | ||||
|     "reset-default": "Redefinir para o ícone padrão" | ||||
|   }, | ||||
|   "basic_properties": { | ||||
|     "note_type": "Tipo da nota", | ||||
|     "editable": "Editável", | ||||
|     "basic_properties": "Propriedades Básicas", | ||||
|     "language": "Idioma" | ||||
|   }, | ||||
|   "book_properties": { | ||||
|     "view_type": "Tipo de visualização", | ||||
|     "grid": "Grade", | ||||
|     "list": "Lista", | ||||
|     "collapse_all_notes": "Recolher todas as notas", | ||||
|     "expand_all_children": "Expandir todos os filhos", | ||||
|     "collapse": "Recolher", | ||||
|     "expand": "Expandir", | ||||
|     "book_properties": "Propriedades da Coleção", | ||||
|     "invalid_view_type": "Tipo de visualização inválido '{{type}}'", | ||||
|     "calendar": "Calendário", | ||||
|     "table": "Tabela", | ||||
|     "geo-map": "Geo Map", | ||||
|     "board": "Quadro" | ||||
|   }, | ||||
|   "edited_notes": { | ||||
|     "no_edited_notes_found": "Ainda não há nenhuma nota editada neste dia…", | ||||
|     "title": "Notas Editadas", | ||||
|     "deleted": "(excluído)" | ||||
|   }, | ||||
|   "file_properties": { | ||||
|     "note_id": "ID da Nota", | ||||
|     "original_file_name": "Nome original do arquivo", | ||||
|     "file_type": "Tipo do arquivo", | ||||
|     "file_size": "Tamanho do arquivo", | ||||
|     "download": "Baixar", | ||||
|     "open": "Abrir", | ||||
|     "upload_new_revision": "Enviar nova revisão", | ||||
|     "upload_success": "Uma nova revisão de arquivo foi enviada.", | ||||
|     "upload_failed": "O envio de uma nova revisão de arquivo falhou.", | ||||
|     "title": "Arquivo" | ||||
|   }, | ||||
|   "image_properties": { | ||||
|     "original_file_name": "Nome original do arquivo", | ||||
|     "file_type": "Tipo do arquivo", | ||||
|     "file_size": "Tamanho do arquivo", | ||||
|     "download": "Baixar", | ||||
|     "open": "Abrir", | ||||
|     "copy_reference_to_clipboard": "Copiar referência para a área de transferência", | ||||
|     "upload_new_revision": "Enviar nova revisão", | ||||
|     "upload_success": "Uma nova revisão de imagem foi enviado.", | ||||
|     "upload_failed": "O envio de uma nova revisão de imagem falhou: {{message}}", | ||||
|     "title": "Imagem" | ||||
|   }, | ||||
|   "inherited_attribute_list": { | ||||
|     "title": "Atributos Herdados", | ||||
|     "no_inherited_attributes": "Nenhum atributo herdado." | ||||
|   }, | ||||
|   "note_info_widget": { | ||||
|     "note_id": "ID da Nota", | ||||
|     "created": "Criado", | ||||
|     "modified": "Editado", | ||||
|     "type": "Tipo", | ||||
|     "note_size": "Tamanho da nota", | ||||
|     "calculate": "calcular", | ||||
|     "title": "Informações da nota", | ||||
|     "subtree_size": "(tamanho da subárvore: {{size}} em {{count}} notas)" | ||||
|   }, | ||||
|   "note_map": { | ||||
|     "open_full": "Expandir completamente", | ||||
|     "collapse": "Recolher para tamanho normal", | ||||
|     "title": "Mapa de Notas", | ||||
|     "fix-nodes": "Fixar nós", | ||||
|     "link-distance": "Distância do Link" | ||||
|   }, | ||||
|   "note_paths": { | ||||
|     "title": "Caminho das Notas", | ||||
|     "clone_button": "Clonar nota para novo local…", | ||||
|     "intro_placed": "Esta nova está localizada nos caminhos:", | ||||
|     "intro_not_placed": "Esta nota ainda não está em nenhuma árvore de notas.", | ||||
|     "archived": "Arquivado", | ||||
|     "search": "Pesquisar" | ||||
|   }, | ||||
|   "note_properties": { | ||||
|     "this_note_was_originally_taken_from": "Esta nota foi originalmente obtida de:", | ||||
|     "info": "Informações" | ||||
|   }, | ||||
|   "promoted_attributes": { | ||||
|     "promoted_attributes": "Atributos Promovidos", | ||||
|     "unset-field-placeholder": "não atribuído", | ||||
|     "open_external_link": "Abrir link externo", | ||||
|     "unknown_label_type": "Tipo de etiqueta desconhecido '{{type}}'", | ||||
|     "unknown_attribute_type": "Tipo de atributo desconhecido '{{type}}'", | ||||
|     "add_new_attribute": "Adicionar novo atributo", | ||||
|     "remove_this_attribute": "Remover este atributo", | ||||
|     "remove_color": "Remover a etiqueta de cor" | ||||
|   }, | ||||
|   "script_executor": { | ||||
|     "query": "Consulta", | ||||
|     "script": "Script", | ||||
|     "execute_query": "Executar Consulta", | ||||
|     "execute_script": "Executar Script" | ||||
|   }, | ||||
|   "search_definition": { | ||||
|     "add_search_option": "Adicionar opção de pesquisa:", | ||||
|     "search_string": "pesquisa de texto", | ||||
|     "search_script": "pesquisa de script", | ||||
|     "ancestor": "ancestral", | ||||
|     "fast_search": "pesquisa rápida", | ||||
|     "include_archived": "incluir arquivados", | ||||
|     "order_by": "ordenar por", | ||||
|     "limit": "limite", | ||||
|     "limit_description": "Limitar número de resultados", | ||||
|     "debug": "depurar", | ||||
|     "action": "ação", | ||||
|     "search_button": "Pesquisar <kbd>enter</kbd>", | ||||
|     "search_execute": "Pesquisar & Executar ações", | ||||
|     "save_to_note": "Salvar para nota", | ||||
|     "search_parameters": "Parâmetros de Pesquisa", | ||||
|     "unknown_search_option": "Opção de pesquisa desconhecida {{searchOptionName}}", | ||||
|     "actions_executed": "As ações foram executadas.", | ||||
|     "search_note_saved": "Nota de pesquisa foi salva em {{- notePathTitle}}" | ||||
|   }, | ||||
|   "similar_notes": { | ||||
|     "title": "Notas Similares", | ||||
|     "no_similar_notes_found": "Nenhum nota similar encontrada." | ||||
|   }, | ||||
|   "abstract_search_option": { | ||||
|     "remove_this_search_option": "Remover esta opção de pesquisa", | ||||
|     "failed_rendering": "A renderização da opção de busca falhou: {{dto}} com o erro: {{error}} {{stack}}" | ||||
|   }, | ||||
|   "debug": { | ||||
|     "debug": "Depurar" | ||||
|   }, | ||||
|   "fast_search": { | ||||
|     "fast_search": "Pesquisa rápida" | ||||
|   }, | ||||
|   "include_archived_notes": { | ||||
|     "include_archived_notes": "Incluir notas arquivadas" | ||||
|   }, | ||||
|   "limit": { | ||||
|     "limit": "Limite", | ||||
|     "take_first_x_results": "Pegar apenas os X primeiros resultados." | ||||
|   }, | ||||
|   "order_by": { | ||||
|     "order_by": "Ordenar por", | ||||
|     "relevancy": "Relevância (padrão)", | ||||
|     "title": "Título", | ||||
|     "date_created": "Data de criação", | ||||
|     "date_modified": "Data da última modificação", | ||||
|     "content_size": "Tamaho do conteúdo da nota", | ||||
|     "content_and_attachments_size": "Tamanho do conteúdo da nota incluindo anexos", | ||||
|     "content_and_attachments_and_revisions_size": "Tamanho do conteúdo da nota incluindo anexos e revisões", | ||||
|     "revision_count": "Número de revisões", | ||||
|     "children_count": "Número de notas filhas", | ||||
|     "parent_count": "Número de clones", | ||||
|     "owned_label_count": "Número de etiquetas", | ||||
|     "owned_relation_count": "Número de relações", | ||||
|     "target_relation_count": "Número de relações para esta nota", | ||||
|     "random": "Ordem aleatória", | ||||
|     "asc": "Crescente (padrão)", | ||||
|     "desc": "Decrescente" | ||||
|   }, | ||||
|   "search_script": { | ||||
|     "title": "Buscar script:", | ||||
|     "placeholder": "buscar notas pelo nome", | ||||
|     "example_title": "Veja este exemplo:", | ||||
|     "example_code": "// 1. pré-filtro usando pesquisa padrão\nconst candidateNotes = api.searchForNotes(\"#journal\"); \n\n// 2. aplicando critérios de pesquisa customizados\nconst matchedNotes = candidateNotes\n    .filter(note => note.title.match(/[0-9]{1,2}\\. ?[0-9]{1,2}\\. ?[0-9]{4}/));\n\nreturn matchedNotes;" | ||||
|   }, | ||||
|   "search_string": { | ||||
|     "title_column": "Buscar texto:", | ||||
|     "search_syntax": "Sintaxe de pesquisa", | ||||
|     "also_see": "veja também", | ||||
|     "full_text_search": "Digite qualquer texto para busca por texto completo", | ||||
|     "label_abc": "retorna notas com a etiqueta abc", | ||||
|     "label_year": "corresponde notas com a etiqueta de ano 2019", | ||||
|     "label_rock_pop": "corresponde notas que tenham tanto a etiqueta rock quando pop", | ||||
|     "label_rock_or_pop": "apenas uma das etiquetas deve estar presente", | ||||
|     "label_year_comparison": "comparação numérica (também >, >=, <).", | ||||
|     "label_date_created": "notas criadas no último mês", | ||||
|     "error": "Erro na busca: {{error}}", | ||||
|     "search_prefix": "Busca:" | ||||
|   }, | ||||
|   "attachment_list": { | ||||
|     "open_help_page": "Abrir página de ajuda nos anexos", | ||||
|     "upload_attachments": "Enviar anexos", | ||||
|     "no_attachments": "Esta nota não possuí anexos." | ||||
|   }, | ||||
|   "editable_code": { | ||||
|     "placeholder": "Digite o conteúdo da sua nota de código aqui…" | ||||
|   }, | ||||
|   "editable_text": { | ||||
|     "placeholder": "Digite o conteúdo da sua nota aqui…" | ||||
|   }, | ||||
|   "empty": { | ||||
|     "search_placeholder": "buscar uma nota pelo nome", | ||||
|     "enter_workspace": "Entrar no workspace {{title}}" | ||||
|   }, | ||||
|   "file": { | ||||
|     "file_preview_not_available": "Prévia não disponível para este formato de arquivo." | ||||
|   }, | ||||
|   "protected_session": { | ||||
|     "enter_password_instruction": "É necessário digitar sua senha para mostar notas protegidas:", | ||||
|     "started": "A sessão protegida foi iniciada.", | ||||
|     "wrong_password": "Senha incorreta.", | ||||
|     "protecting-finished-successfully": "A proteção foi finalizada com sucesso.", | ||||
|     "unprotecting-finished-successfully": "A remoção da proteção foi finalizada com sucesso.", | ||||
|     "protecting-in-progress": "Proteções em andamento: {{count}}", | ||||
|     "unprotecting-in-progress-count": "Remoções de proteção em andamento: {{count}}", | ||||
|     "protecting-title": "Estado da proteção", | ||||
|     "unprotecting-title": "Estado da remoção de proteção" | ||||
|   }, | ||||
|   "relation_map": { | ||||
|     "open_in_new_tab": "Abrir em nova aba", | ||||
|     "remove_note": "Remover nota", | ||||
|     "edit_title": "Editar título", | ||||
|     "rename_note": "Renomear nota", | ||||
|     "enter_new_title": "Digite o novo título da nota:", | ||||
|     "remove_relation": "Remover relação", | ||||
|     "confirm_remove_relation": "Tem certeza que deseja remover esta relação?", | ||||
|     "connection_exists": "A conexão '{{name}}' já existe entre estas notas.", | ||||
|     "note_not_found": "Nota {{noteId}} não encontrada!", | ||||
|     "note_already_in_diagram": "A nota \"{{title}}\" já está no diagrama.", | ||||
|     "enter_title_of_new_note": "Digite o título da nova nota", | ||||
|     "default_new_note_title": "nova nota", | ||||
|     "click_on_canvas_to_place_new_note": "Clique no quadro para incluir uma nova nota" | ||||
|   }, | ||||
|   "web_view": { | ||||
|     "web_view": "Web View" | ||||
|   }, | ||||
|   "backend_log": { | ||||
|     "refresh": "Recarregar" | ||||
|   }, | ||||
|   "consistency_checks": { | ||||
|     "title": "Chegagem de Consistência", | ||||
|     "find_and_fix_button": "Encontrar e corrigir problemas de consistência", | ||||
|     "finding_and_fixing_message": "Buscando e corrigindo problemas de consistência…", | ||||
|     "issues_fixed_message": "Qualquer problema de consistência encontrado foi corrigido." | ||||
|   }, | ||||
|   "database_integrity_check": { | ||||
|     "check_button": "Verificar integridade do banco de dados", | ||||
|     "checking_integrity": "Verificando integridade do banco de dados…", | ||||
|     "integrity_check_succeeded": "Verificação de integridade bem sucedida - nenhum problema encontrado.", | ||||
|     "integrity_check_failed": "Verificação de integridade falhou: {{results}}" | ||||
|   }, | ||||
|   "sync": { | ||||
|     "title": "Sincronizar", | ||||
|     "force_full_sync_button": "Forçar sincronização completa", | ||||
|     "full_sync_triggered": "Sincronização completa iniciada", | ||||
|     "finished-successfully": "Sincronização finalizada com sucesso.", | ||||
|     "failed": "Sincronização falhou: {{message}}" | ||||
|   }, | ||||
|   "vacuum_database": { | ||||
|     "description": "Isso irá reconstruir o banco de dados, o que normalmente irá resultar em uma redução do arquivo do banco de dados. Nenhum dado será alterado." | ||||
|   }, | ||||
|   "fonts": { | ||||
|     "theme_defined": "Tema definido", | ||||
|     "fonts": "Fontes", | ||||
|     "main_font": "Fonte Principal", | ||||
|     "font_family": "Família da fonte", | ||||
|     "size": "Tamanho", | ||||
|     "note_tree_font": "Fonte da Árvore de Notas", | ||||
|     "note_detail_font": "Fonte Padrão da Nota", | ||||
|     "monospace_font": "Fonte Monospace (código)", | ||||
|     "not_all_fonts_available": "Nem todas as fontes listadas podem estar disponíveis em seu sistema.", | ||||
|     "apply_font_changes": "Para aplicar as alterações de fonte, clique em", | ||||
|     "reload_frontend": "recarregar frontend", | ||||
|     "generic-fonts": "Fontes genéricas", | ||||
|     "sans-serif-system-fonts": "Fontes sem serifa de sistema", | ||||
|     "serif-system-fonts": "Fontes serifadas de sistema", | ||||
|     "monospace-system-fonts": "Fontes monospace de sistema", | ||||
|     "handwriting-system-fonts": "Fontes de escrita à mão de sistema", | ||||
|     "serif": "Serifa", | ||||
|     "sans-serif": "Sem Serifa", | ||||
|     "monospace": "Monospace", | ||||
|     "system-default": "Padrão do Sistema" | ||||
|   }, | ||||
|   "max_content_width": { | ||||
|     "title": "Largura do Conteúdo", | ||||
|     "max_width_label": "Largura máxima do conteúdo", | ||||
|     "max_width_unit": "pixels", | ||||
|     "apply_changes_description": "Para aplicar as alterações de largura do conteúdo, clique em", | ||||
|     "reload_button": "recarregar frontend", | ||||
|     "reload_description": "alterações de opções de aparência" | ||||
|   }, | ||||
|   "native_title_bar": { | ||||
|     "title": "Barra de Título Nativa (requer recarregar o app)", | ||||
|     "enabled": "ativada", | ||||
|     "disabled": "desativada" | ||||
|   }, | ||||
|   "theme": { | ||||
|     "title": "Tema da Aplicação", | ||||
|     "theme_label": "Tema", | ||||
|     "override_theme_fonts_label": "Sobrepor fontes do tema", | ||||
|     "auto_theme": "Legado (Seguir esquema de cor do sistema)", | ||||
|     "light_theme": "Legado (Claro)", | ||||
|     "dark_theme": "Legado (Escuro)", | ||||
|     "triliumnext": "Trilium (Seguir esquema de cor do sistema)", | ||||
|     "triliumnext-light": "Trilium (Claro)", | ||||
|     "triliumnext-dark": "Trilium (Escuro)", | ||||
|     "layout": "Layout", | ||||
|     "layout-vertical-title": "Vertical", | ||||
|     "layout-horizontal-title": "Horizontal", | ||||
|     "layout-vertical-description": "barra de lançamento está a esquerda (padrão)", | ||||
|     "layout-horizontal-description": "barra de lançamento está abaixo da barra de abas, a barra de abas agora tem a largura total." | ||||
|   }, | ||||
|   "note_launcher": { | ||||
|     "this_launcher_doesnt_define_target_note": "Este lançador não define uma nota destino." | ||||
|   }, | ||||
|   "copy_image_reference_button": { | ||||
|     "button_title": "Copiar referência da imagem para a área de transferência, pode ser colado em uma nota de texto." | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -98,7 +98,8 @@ | ||||
|     "snapshot_number_limit_unit": "снимков", | ||||
|     "note_revisions_snapshot_limit_title": "Максимальное количество снимков заметок", | ||||
|     "snapshot_number_limit_label": "Максимальное количество снимков:", | ||||
|     "erase_excess_revision_snapshots_prompt": "Удалить лишние снимки." | ||||
|     "erase_excess_revision_snapshots_prompt": "Удалить лишние снимки.", | ||||
|     "erase_excess_revision_snapshots": "Удалить лишние снимки версий" | ||||
|   }, | ||||
|   "password": { | ||||
|     "alert_message": "Пожалуйста, запомните новый пароль. Пароль используется для входа в веб-интерфейс и шифрования защищённых заметок. Если вы забудете пароль, все ваши защищённые заметки будут потеряны навсегда.", | ||||
| @@ -115,10 +116,14 @@ | ||||
|     "protected_session_timeout": "Тайм-аут защищенного сеанса", | ||||
|     "protected_session_timeout_label": "Тайм-аут защищенного сеанса:", | ||||
|     "protected_session_timeout_description": "Тайм-аут защищенного сеанса - это период времени, по истечении которого защищенный сеанс удаляется из памяти браузера. Он отсчитывается с момента последнего взаимодействия с защищенными заметками. См", | ||||
|     "for_more_info": "для получения более подробной информации." | ||||
|     "for_more_info": "для получения более подробной информации.", | ||||
|     "reset_confirmation": "Сбросив пароль, вы навсегда потеряете доступ ко всем своим защищённым заметкам. Вы действительно хотите сбросить пароль?", | ||||
|     "password_changed_success": "Пароль изменён. Trilium будет перезагружен после нажатия кнопки «ОК».", | ||||
|     "password_mismatch": "Новые пароли не совпадают.", | ||||
|     "reset_success_message": "Пароль был сброшен. Пожалуйста, установите новый пароль" | ||||
|   }, | ||||
|   "content_language": { | ||||
|     "description": "Выберите один или несколько языков, которые должны отображаться в разделе «Основные свойства» текстовой заметки, доступной только для чтения или редактируемой. Это позволит реализовать такие функции, как проверка орфографии и поддержка письма справа налево.", | ||||
|     "description": "Выберите один или несколько языков, которые должны отображаться в разделе «Общее» текстовой заметки, доступной только для чтения или редактируемой. Это позволит реализовать такие функции, как проверка орфографии и поддержка письма справа налево.", | ||||
|     "title": "Языки контента" | ||||
|   }, | ||||
|   "theme": { | ||||
| @@ -133,7 +138,9 @@ | ||||
|     "layout-horizontal-title": "Горизонтальный", | ||||
|     "auto_theme": "Legacy (следует системной цветовой схеме)", | ||||
|     "light_theme": "Legacy (светлая)", | ||||
|     "dark_theme": "Legacy (темная)" | ||||
|     "dark_theme": "Legacy (темная)", | ||||
|     "layout-horizontal-description": "панель запуска находится под панелью вкладок, панель вкладок теперь занимает всю ширину.", | ||||
|     "layout-vertical-description": "панель запуска находится слева (по умолчанию)" | ||||
|   }, | ||||
|   "tasks": { | ||||
|     "due": { | ||||
| @@ -154,7 +161,8 @@ | ||||
|     "reopen_last_tab": "Повторно открыть последнюю закрытую вкладку", | ||||
|     "close_all_tabs": "Закрыть все вкладки", | ||||
|     "close_other_tabs": "Закрыть остальные вкладки", | ||||
|     "add_new_tab": "Добавить новую вкладку" | ||||
|     "add_new_tab": "Добавить новую вкладку", | ||||
|     "close_right_tabs": "Закрыть вкладки справа" | ||||
|   }, | ||||
|   "table_view": { | ||||
|     "new-row": "Новая строка", | ||||
| @@ -182,7 +190,10 @@ | ||||
|     "label_name_title": "Разрешены буквенно-цифровые символы, подчеркивание и двоеточие.", | ||||
|     "new_value_placeholder": "новое значение", | ||||
|     "to_value": "на значение", | ||||
|     "help_text": "На всех совпадающих заметках:" | ||||
|     "help_text": "На всех совпадающих заметках:", | ||||
|     "help_text_note": "Вы также можете вызвать этот метод без значения, в таком случае метка будет присвоена заметке без значения.", | ||||
|     "help_text_item1": "создать заданную метку, если у заметки ее еще нет", | ||||
|     "help_text_item2": "или изменить значение существующей метки" | ||||
|   }, | ||||
|   "delete_label": { | ||||
|     "delete_label": "Удалить метку", | ||||
| @@ -273,16 +284,17 @@ | ||||
|     "selectNote": "выбрать заметку", | ||||
|     "copyNotes": "скопировать активную заметку (или выделение) в буфер обмер (используется для <a class=\"external\" href=\"https://triliumnext.github.io/Docs/Wiki/cloning-notes.html#cloning-notes\">клонирования</a>)", | ||||
|     "createEditLink": "создать/редактировать внешнюю ссылку", | ||||
|     "headings": "<code>##</code>, <code>###</code>, <code>####</code>  и т. д., за которыми следует пробел для заголовков.", | ||||
|     "headings": "<code>##</code>, <code>###</code>, <code>####</code>  и т. д., за которыми следует пробел для заголовков", | ||||
|     "bulletList": "<code>*</code> или <code>-</code> с последующим пробелом для маркированного списка", | ||||
|     "numberedList": "<code>1.</code> или <code>1)</code> с последующим пробелом для нумерованного списка", | ||||
|     "blockQuote": "начните строку с <code>></code>, а затем пробела для блока цитаты", | ||||
|     "quickSearch": "сфокусироваться на полее ввода быстрого поиска", | ||||
|     "editNoteTitle": "В области дерева переключится с области дерева на заголовок заметки. Сочетание клавиш Enter из области заголовка заметки переключит фокус на текстовый редактор. <kbd>Ctrl+.</kbd> переключит обратно с редактора на область дерева.", | ||||
|     "editNoteTitle": "в области дерева переключится с области дерева на заголовок заметки. Сочетание клавиш Enter из области заголовка заметки переключит фокус на текстовый редактор. <kbd>Ctrl+.</kbd> переключит обратно с редактора на область дерева.", | ||||
|     "title": "Справка" | ||||
|   }, | ||||
|   "modal": { | ||||
|     "close": "Закрыть" | ||||
|     "close": "Закрыть", | ||||
|     "help_title": "Показать больше информации об этом экране" | ||||
|   }, | ||||
|   "import": { | ||||
|     "importIntoNote": "Импортировать в заметку", | ||||
| @@ -327,7 +339,7 @@ | ||||
|   "password_not_set": { | ||||
|     "title": "Пароль не установлен", | ||||
|     "body1": "Защищенные заметки шифруются с помощью пароля пользователя, но пароль еще не установлен.", | ||||
|     "body2": "Чтобы иметь возможность защищать заметки, нажмите <a class=\"open-password-options-button\" href=\"javascript:\">здесь</a>, чтобы установить пароль.", | ||||
|     "body2": "Чтобы защитить заметки, нажмите кнопку ниже, чтобы открыть диалоговое окно «Параметры» и установить пароль.", | ||||
|     "go_to_password_options": "Перейти к параметрам пароля" | ||||
|   }, | ||||
|   "protected_session_password": { | ||||
| @@ -339,7 +351,7 @@ | ||||
|   }, | ||||
|   "recent_changes": { | ||||
|     "title": "Последние изменения", | ||||
|     "erase_notes_button": "Удалить заметки, помеченные на удаление сейчас", | ||||
|     "erase_notes_button": "Стереть удаленные заметки сейчас", | ||||
|     "undelete_link": "восстановить", | ||||
|     "no_changes_message": "Еще нет изменений...", | ||||
|     "deleted_notes_message": "Удаленные заметки были стерты окончательно.", | ||||
| @@ -389,7 +401,7 @@ | ||||
|   "upload_attachments": { | ||||
|     "upload_attachments_to_note": "Загрузить вложения к заметке", | ||||
|     "choose_files": "Выберите файлы", | ||||
|     "files_will_be_uploaded": "Файлы будут загружены как приложения в", | ||||
|     "files_will_be_uploaded": "Файлы будут загружены как приложения в {{noteTitle}}", | ||||
|     "options": "Параметры", | ||||
|     "shrink_images": "Сжать изображения", | ||||
|     "tooltip": "Если этот параметр включен, Trilium попытается уменьшить размер загружаемых изображений путём масштабирования и оптимизации, что может повлиять на воспринимаемое качество изображения. Если этот параметр не включен, изображения будут загружаться без изменений.", | ||||
| @@ -425,7 +437,7 @@ | ||||
|     "target_note_title": "Отношение — это именованная связь между исходной и целевой заметками.", | ||||
|     "promoted_title": "Выделенный атрибут отображается в заметке явно.", | ||||
|     "promoted": "Выделенный", | ||||
|     "promoted_alias_title": "Название, которое будет отображаться в интерфейсе выделенных атрибутов.", | ||||
|     "promoted_alias_title": "Название, которое будет отображаться в интерфейсе продвигаемых атрибутов.", | ||||
|     "multiplicity_title": "Множественность определяет, сколько атрибутов с одним и тем же именем можно создать — максимум 1 или более 1.", | ||||
|     "label_type_title": "Тип метки поможет Trilium выбрать подходящий интерфейс для ввода значения метки.", | ||||
|     "precision_title": "Какое количество цифр после плавающей запятой должно быть доступно в интерфейсе настройки значения.", | ||||
| @@ -446,10 +458,10 @@ | ||||
|     "run_at_hour": "В какой час это должно выполняться? Следует использовать вместе с <code>#run=hourly</code>. Можно задать несколько раз для большего количества запусков в течение дня.", | ||||
|     "disable_inclusion": "скрипты с этой меткой не будут включены в выполнение родительского скрипта.", | ||||
|     "sorted": "сохраняет алфавитную сортировку дочерних заметок", | ||||
|     "sort_direction": "ASC (по умолчанию) или DESC", | ||||
|     "sort_direction": "ASC (по возрастани, по умолчанию) или DESC (по убыванию)", | ||||
|     "sort_folders_first": "Папки (заметки, включая дочерние) должны быть отсортированы вверх", | ||||
|     "top": "закрепить заданную заметку наверху в ее родителе (применяется только к отсортированным родительским заметкам)", | ||||
|     "hide_promoted_attributes": "Скрыть выделенные атрибуты в этой заметке", | ||||
|     "hide_promoted_attributes": "Скрыть продвигаемых атрибуты в этой заметке", | ||||
|     "read_only": "редактор находится в режиме только для чтения. Работает только с текстом и заметками типа \"код\".", | ||||
|     "auto_read_only_disabled": "текстовые/заметки с кодом могут автоматически переводиться в режим чтения, если они слишком большие. Вы можете отключить это поведение для каждой заметки, добавив к ней соответствующую метку", | ||||
|     "app_css": "отмечает заметки CSS, которые загружаются в приложение Trilium и, таким образом, могут использоваться для изменения внешнего вида Trilium.", | ||||
| @@ -461,10 +473,64 @@ | ||||
|     "workspace_template": "Эта заметка появится в списке доступных шаблонов при создании новой заметки, но только если она будет перемещена в рабочую область, содержащую этот шаблон", | ||||
|     "workspace_search_home": "новые заметки поиска будут созданы как дочерние записи этой заметки при перемещении их к какому-либо предку этой заметки рабочей области", | ||||
|     "workspace_calendar_root": "Определяет корень календаря для каждого рабочего пространства", | ||||
|     "hide_highlight_widget": "Скрыть виджет «Список выделенного»", | ||||
|     "hide_highlight_widget": "Скрыть виджет «Выделенное»", | ||||
|     "is_owned_by_note": "принадлежит записке", | ||||
|     "and_more": "... и ещё {{count}}.", | ||||
|     "app_theme": "отмечает заметки CSS, которые являются полноценными темами Trilium и, таким образом, доступны в опциях Trilium." | ||||
|     "app_theme": "отмечает заметки CSS, которые являются полноценными темами Trilium и, таким образом, доступны в опциях Trilium.", | ||||
|     "title_template": "Заголовок по умолчанию для заметок, создаваемых как дочерние элементы данной заметки. Значение вычисляется как строка JavaScript\n                        и, таким образом, может быть дополнено динамическим контентом с помощью внедренных переменных <code>now</code> и <code>parentNote</code>. Примеры:\n                        \n                        <ul>\n                            <li><code>Литературные произведения ${parentNote.getLabelValue('authorName')}</code></li>\n                            <li><code>Лог для ${now.format('YYYY-MM-DD HH:mm:ss')}</code></li>\n                        </ul>\n                        \n                        Подробности см. в <a href=\"https://triliumnext.github.io/Docs/Wiki/default-note-title.html\">вики</a>, документации API для <a href=\"https://zadam.github.io/trilium/backend_api/Note.html\">parentNote</a> и <a href=\"https://day.js.org/docs/en/display/format\">now</a>.", | ||||
|     "icon_class": "значение этой метки добавляется в виде CSS-класса к значку в дереве, что помогает визуально различать заметки в дереве. Примером может служить bx bx-home — значки берутся из boxicons. Может использоваться в шаблонах заметок.", | ||||
|     "share_favicon": "Заметка о фавиконе должна быть размещена на странице общего доступа. Обычно её назначают корневой папке общего доступа и делают наследуемой. Заметка о фавиконе также должна находиться в поддереве общего доступа. Рассмотрите возможность использования атрибута 'share_hidden_from_tree'.", | ||||
|     "inbox": "расположение папки «Входящие» по умолчанию для новых заметок — при создании заметки с помощью кнопки «Новая заметка» на боковой панели заметки будут созданы как дочерние заметки в заметке, помеченной меткой <code>#inbox</code>.", | ||||
|     "share_css": "CSS-заметка, которая будет добавлена на страницу общего доступа. CSS-заметка также должна находиться в общем поддереве. Также рассмотрите возможность использования 'share_hidden_from_tree' и 'share_omit_default_css'.", | ||||
|     "run_on_branch_deletion": "выполняется при удалении ветви. Ветка — это связь между родительской и дочерней заметками и удаляется, например, при перемещении заметки (старая ветвь/ссылка удаляется).", | ||||
|     "share_template": "Встроенная заметка JavaScript, которая будет использоваться в качестве шаблона для отображения общей заметки. Возвращается к шаблону по умолчанию. Рекомендуется использовать 'share_hidden_from_tree'.", | ||||
|     "print_page_size": "При экспорте в PDF изменяет размер страницы. Поддерживаемые значения: <code>A0</code>, <code>A1</code>, <code>A2</code>, <code>A3</code>, <code>A4</code>, <code>A5</code>, <code>A6</code>, <code>Legal</code>, <code>Letter</code>, <code>Tabloid</code>, <code>Ledger</code>.", | ||||
|     "keyboard_shortcut": "Определяет сочетание клавиш для немедленного перехода к этой заметке. Пример: Ctrl+Alt+E. Для вступления изменений в силу требуется перезагрузка интерфейса.", | ||||
|     "new_notes_on_top": "Новые заметки будут создаваться вверху родительской заметки, а не внизу.", | ||||
|     "print_landscape": "При экспорте в PDF изменяет ориентацию страницы с книжной на альбомную.", | ||||
|     "hide_relations": "имена отношений, которые следует скрыть, разделённые запятыми. Все остальные будут отображены.", | ||||
|     "run_on_note_change": "выполняется при изменении заметки (включая создание заметки). Не включает изменения содержимого", | ||||
|     "display_relations": "названия отношений, разделённые запятыми, которые следует отобразить. Все остальные будут скрыты.", | ||||
|     "template": "Эта заметка появится в списке доступных шаблонов при создании новой заметки", | ||||
|     "execute_button": "Название кнопки, которая выполнит текущую заметку типа \"Код\"", | ||||
|     "page_size": "количество элементов на странице в списке заметок", | ||||
|     "custom_request_handler": "см. <a href=\"javascript:\" data-help-page=\"custom-request-handler.html\">Пользовательский обработчик запросов</a>", | ||||
|     "custom_resource_provider": "см. <a href=\"javascript:\" data-help-page=\"custom-request-handler.html\">Пользовательский обработчик запросов</a>", | ||||
|     "widget": "отмечает эту заметку как пользовательский виджет, который будет добавлен в дерево компонентов Trilium", | ||||
|     "search_home": "новые заметки поиска будут созданы как дочерние записи этой заметки", | ||||
|     "workspace_inbox": "расположение в папке «Входящие» по умолчанию для новых заметок при перемещении их в некую родственную папку этой заметки в рабочей области", | ||||
|     "sql_console_home": "расположение заметок консоли SQL по умолчанию", | ||||
|     "css_class": "значение этой метки затем добавляется как CSS-класс к узлу, представляющему данную заметку в дереве. Это может быть полезно для изменения внешнего вида заметки. Может использоваться в шаблонах заметок.", | ||||
|     "bookmark_folder": "заметка с этой меткой появится в закладках как папка (с предоставлением доступа к ее дочерним элементам)", | ||||
|     "share_hidden_from_tree": "эта заметка скрыта в левом навигационном дереве, но по-прежнему доступна по ее URL-адресу", | ||||
|     "share_external_link": "заметка будет действовать как ссылка на внешний веб-сайт в дереве общего доступа", | ||||
|     "share_alias": "определить псевдоним, с помощью которого заметка будет доступна по адресу https://ссылка_на_ваш_trilium/share/[ваш_псевдоним]", | ||||
|     "share_omit_default_css": "CSS-код страницы общего доступа по умолчанию будет пропущен. Используйте его при внесении существенных изменений в стили.", | ||||
|     "share_root": "помечает заметку, которая будет выступать корневой страницей /share общедоступного сайта.", | ||||
|     "share_description": "определение текста, который будет добавлен в HTML-тег meta для описания", | ||||
|     "share_raw": "заметка будет передана в исходном виде, без HTML-обертки", | ||||
|     "share_disallow_robot_indexing": "запретит индексацию этой заметки роботами через заголовок <code>X-Robots-Tag: noindex</code>", | ||||
|     "share_credentials": "для доступа к этой общедоступной заметке требуются учётные данные. Значение должно быть в формате 'имя пользователя:пароль'. Не забудьте сделать этот атрибут наследуемым для применения к дочерним заметкам/изображениям.", | ||||
|     "share_index": "заметка с этой меткой будет содержать список всех корневых узлов общедоступных заметок", | ||||
|     "toc": "<code>#toc</code> или <code>#toc=show</code> принудительно отобразят оглавление, <code>#toc=hide</code> — скроют его. Если метка отсутствует, применяется глобальная настройка", | ||||
|     "color": "определяет цвет заметки в дереве заметок, ссылках и т. д. Используйте любое допустимое значение цвета CSS, например «red» или #a13d5f", | ||||
|     "keep_current_hoisting": "Открытие этой ссылки не изменит закрепление, даже если заметка не отображается в текущем закрепленном поддереве.", | ||||
|     "execute_description": "Более подробное описание текущей заметки типа \"Код\", отображаемое вместе с кнопкой \"Выполнить\"", | ||||
|     "run_on_note_creation": "выполняется при создании заметки на сервере. Используйте это отношение, если хотите запустить скрипт для всех заметок, созданных в определённом поддереве. В этом случае создайте его в корневой заметке поддерева и сделайте его наследуемым. Новая заметка, созданная в поддереве (любой глубины), запустит скрипт.", | ||||
|     "run_on_child_note_creation": "выполняется, когда создается новая заметка под заметкой, в которой определено это отношение", | ||||
|     "run_on_note_title_change": "выполняется при изменении заголовка заметки (включая создание заметки)", | ||||
|     "run_on_note_content_change": "выполняется при изменении содержимого заметки (включая создание заметки).", | ||||
|     "run_on_note_deletion": "выполняется при удалении заметки", | ||||
|     "run_on_branch_creation": "выполняется при создании ветви. Ветвь — это связующее звено между родительской и дочерней заметками и создаётся, например, при клонировании или перемещении заметки.", | ||||
|     "run_on_branch_change": "выполняется при обновлении ветки.", | ||||
|     "run_on_attribute_creation": "выполняется, когда создается новый атрибут для заметка, определяющей это отношение", | ||||
|     "run_on_attribute_change": " выполняется при изменении атрибута заметки, определяющей это отношение. Также срабатывает при удалении атрибута", | ||||
|     "relation_template": "атрибуты заметки будут унаследованы даже без родительско-дочерних отношений. Содержимое заметки и её поддерево будут добавлены к экземпляру заметки, если оно пустое. Подробности см. в документации.", | ||||
|     "inherit": "атрибуты заметки будут унаследованы даже без родительско-дочерних отношений. См. описание шаблонных отношений для получения аналогичной информации. См. раздел «Наследование атрибутов» в документации.", | ||||
|     "render_note": "заметки типа «Рендер HTML» будут отображаться с использованием кодовой заметки (HTML или скрипта), и необходимо указать с помощью этой связи, какую заметку следует отобразить", | ||||
|     "widget_relation": "заметка, на которую ссылается отношение будет выполнена и отображена как виджет на боковой панели", | ||||
|     "share_js": "JavaScript-заметка, которая будет добавлена на страницу общего доступа. JavaScript-заметка также должна находиться в общем поддереве. Рекомендуется использовать 'share_hidden_from_tree'.", | ||||
|     "other_notes_with_name": "Другие заметки с {{attributeType}} названием \"{{attributeName}}\"" | ||||
|   }, | ||||
|   "command_palette": { | ||||
|     "configure_launch_bar_description": "Откройте конфигурацию панели запуска, чтобы добавить или удалить элементы.", | ||||
| @@ -542,7 +608,8 @@ | ||||
|     "not_set": "Не установлен" | ||||
|   }, | ||||
|   "time_selector": { | ||||
|     "invalid_input": "Введенное значение времени не является допустимым числом." | ||||
|     "invalid_input": "Введенное значение времени не является допустимым числом.", | ||||
|     "minimum_input": "Введенное значение времени должно быть не менее {{minimumSeconds}} секунд." | ||||
|   }, | ||||
|   "share": { | ||||
|     "share_root_not_found": "Заметка с меткой #shareRoot не найдена", | ||||
| @@ -550,7 +617,10 @@ | ||||
|     "redirect_bare_domain_description": "Перенаправлять анонимных пользователей на страницу общедоступных заметок вместо отображения страницы входа", | ||||
|     "show_login_link": "Показать ссылку для аутентификации в интерфейсе общедоступных заметок", | ||||
|     "show_login_link_description": "Добавить ссылку для аутентификации в нижний колонтитул интерфейса общедоступных заметок", | ||||
|     "title": "Настройки общего доступа" | ||||
|     "title": "Настройки общего доступа", | ||||
|     "check_share_root": "Проверка состояния корневой заметки для общедоступного сайта", | ||||
|     "share_root_not_shared": "Заметка '{{noteTitle}}' имеет метку #shareRoot, но не является общедоступной", | ||||
|     "share_root_found": "Заметка корня общедоступного сайта '{{noteTitle}}' готова" | ||||
|   }, | ||||
|   "duration": { | ||||
|     "days": "Дни", | ||||
| @@ -564,7 +634,9 @@ | ||||
|     "open-location": "Открыть местоположение" | ||||
|   }, | ||||
|   "geo-map": { | ||||
|     "unable-to-load-map": "Не удалось загрузить карту." | ||||
|     "unable-to-load-map": "Не удалось загрузить карту.", | ||||
|     "create-child-note-instruction": "Щелкните по карте, чтобы создать новую заметку в этом месте, или нажмите Escape, чтобы закрыть ее.", | ||||
|     "create-child-note-title": "Создать новую дочернюю заметку и добавить ее на карту" | ||||
|   }, | ||||
|   "note_tooltip": { | ||||
|     "quick-edit": "Быстрое редактирование", | ||||
| @@ -574,7 +646,9 @@ | ||||
|     "full-text-search": "Полнотекстовый поиск", | ||||
|     "show-recent-notes": "Показать последние заметки", | ||||
|     "search-for": "Поиск \"{{term}}\"", | ||||
|     "clear-text-field": "Очистить текстовое поле" | ||||
|     "clear-text-field": "Очистить текстовое поле", | ||||
|     "insert-external-link": "Вставить внешнюю ссылку \"{{term}}\"", | ||||
|     "create-note": "Создать и связать дочернюю заметку \"{{term}}\"" | ||||
|   }, | ||||
|   "electron_integration": { | ||||
|     "zoom-factor": "Коэффициент масштабирования", | ||||
| @@ -592,7 +666,8 @@ | ||||
|     "open_note_in_new_split": "Открыть заметку в новой панели" | ||||
|   }, | ||||
|   "image_context_menu": { | ||||
|     "copy_image_to_clipboard": "Копировать изображение в буфер обмена" | ||||
|     "copy_image_to_clipboard": "Копировать изображение в буфер обмена", | ||||
|     "copy_reference_to_clipboard": "Скопировать ссылку в буфер обмена" | ||||
|   }, | ||||
|   "electron_context_menu": { | ||||
|     "paste-as-plain-text": "Вставить как обычный текст", | ||||
| @@ -600,7 +675,8 @@ | ||||
|     "copy-link": "Скопировать ссылку", | ||||
|     "copy": "Скопировать", | ||||
|     "cut": "Вырезать", | ||||
|     "search_online": "Поиск \"{{term}}\" в {{searchEngine}}" | ||||
|     "search_online": "Поиск \"{{term}}\" в {{searchEngine}}", | ||||
|     "add-term-to-dictionary": "Добавить \"{{term}}\" в словарь" | ||||
|   }, | ||||
|   "editing": { | ||||
|     "editor_type": { | ||||
| @@ -631,7 +707,8 @@ | ||||
|   }, | ||||
|   "highlighting": { | ||||
|     "color-scheme": "Цветовая схема", | ||||
|     "title": "Блоки кода" | ||||
|     "title": "Блоки кода", | ||||
|     "description": "Управляет подсветкой синтаксиса для блоков кода внутри текстовых заметок. Заметки с типом \"Код\" не будут затронуты." | ||||
|   }, | ||||
|   "editable-text": { | ||||
|     "auto-detect-language": "Определен автоматически" | ||||
| @@ -642,7 +719,11 @@ | ||||
|     "add-custom-widget": "Добавить пользовательский виджет", | ||||
|     "move-to-visible-launchers": "Переместить к видимым лаунчерам", | ||||
|     "move-to-available-launchers": "Переместить к доступным лаунчерам", | ||||
|     "delete": "Удалить <kbd data-command=\"deleteNotes\"></kbd>" | ||||
|     "delete": "Удалить <kbd data-command=\"deleteNotes\"></kbd>", | ||||
|     "add-note-launcher": "Добавить лаунчер заметки", | ||||
|     "add-script-launcher": "Добавить лаунчер скрипта", | ||||
|     "duplicate-launcher": "Создать копию лаунчера <kbd data-command=\"duplicateSubtree\">", | ||||
|     "reset_launcher_confirm": "Вы действительно хотите сбросить \"{{title}}\"? Все данные/настройки в этой заметке (и её дочерних заметках) будут потеряны, а панель запуска будет возвращена в исходное местоположение." | ||||
|   }, | ||||
|   "toc": { | ||||
|     "table_of_contents": "Оглавление", | ||||
| @@ -660,13 +741,15 @@ | ||||
|     "create-child-note": "Создать дочернюю заметку", | ||||
|     "save-changes": "Сохранить и применить изменения", | ||||
|     "saved-search-note-refreshed": "Сохраненная поисковая заметка обновлена.", | ||||
|     "refresh-saved-search-results": "Обновить сохраненные результаты поиска" | ||||
|     "refresh-saved-search-results": "Обновить сохраненные результаты поиска", | ||||
|     "automatically-collapse-notes-title": "Заметки будут свернуты после определенного периода бездействия, чтобы навести порядок в дереве." | ||||
|   }, | ||||
|   "quick-search": { | ||||
|     "no-results": "Результаты не найдены", | ||||
|     "placeholder": "Быстрый поиск", | ||||
|     "searching": "Поиск...", | ||||
|     "show-in-full-search": "Расширенный поиск" | ||||
|     "show-in-full-search": "Расширенный поиск", | ||||
|     "more-results": "... и еще {{number}} результатов." | ||||
|   }, | ||||
|   "find": { | ||||
|     "replace_all": "Заменить все", | ||||
| @@ -678,7 +761,8 @@ | ||||
|   }, | ||||
|   "template_switch": { | ||||
|     "template": "Шаблон", | ||||
|     "toggle-off-hint": "Удалить заметку как шаблон" | ||||
|     "toggle-off-hint": "Удалить заметку как шаблон", | ||||
|     "toggle-on-hint": "Сделать заметку шаблоном" | ||||
|   }, | ||||
|   "note_types": { | ||||
|     "collections": "Коллекции", | ||||
| @@ -702,7 +786,8 @@ | ||||
|     "mind-map": "Mind Map", | ||||
|     "geo-map": "Географическая карта", | ||||
|     "ai-chat": "ИИ Чат", | ||||
|     "task-list": "Список задач" | ||||
|     "task-list": "Список задач", | ||||
|     "confirm-change": "Не рекомендуется менять тип заметки, если её содержимое не пустое. Вы всё равно хотите продолжить?" | ||||
|   }, | ||||
|   "tree-context-menu": { | ||||
|     "open-in-popup": "Быстрое редактирование", | ||||
| @@ -732,7 +817,10 @@ | ||||
|     "edit-branch-prefix": "Изменить префикс ветки", | ||||
|     "convert-to-attachment": "Преобразовать в приложение", | ||||
|     "apply-bulk-actions": "Применить массовые действия", | ||||
|     "recent-changes-in-subtree": "Последние изменения в поддереве" | ||||
|     "recent-changes-in-subtree": "Последние изменения в поддереве", | ||||
|     "copy-note-path-to-clipboard": "Копировать путь к заметке в буфер обмена", | ||||
|     "convert-to-attachment-confirm": "Вы уверены, что хотите преобразовать выбранные заметки во вложения их родительских заметок?", | ||||
|     "converted-to-attachments": "{{count}} заметок были преобразованы во вложения." | ||||
|   }, | ||||
|   "info": { | ||||
|     "closeButton": "Закрыть", | ||||
| @@ -770,14 +858,17 @@ | ||||
|     "to": "в", | ||||
|     "add_relation": "Добавить отношение", | ||||
|     "relation_name": "название отношения", | ||||
|     "target_note": "целевая заметка" | ||||
|     "target_note": "целевая заметка", | ||||
|     "allowed_characters": "Разрешены буквенно-цифровые символы, подчеркивание и двоеточие.", | ||||
|     "create_relation_on_all_matched_notes": "Для всех соответствующих заметок создать заданную связь." | ||||
|   }, | ||||
|   "rename_relation": { | ||||
|     "to": "В", | ||||
|     "rename_relation": "Переименовать отношение", | ||||
|     "old_name": "старое наименование", | ||||
|     "new_name": "новое наименование", | ||||
|     "rename_relation_from": "Переименовать отношение из" | ||||
|     "rename_relation_from": "Переименовать отношение из", | ||||
|     "allowed_characters": "Разрешены буквенно-цифровые символы, подчеркивание и двоеточие." | ||||
|   }, | ||||
|   "update_relation_target": { | ||||
|     "to": "в", | ||||
| @@ -785,7 +876,9 @@ | ||||
|     "relation_name": "название отношения", | ||||
|     "target_note": "целевая заметка", | ||||
|     "update_relation_target": "Обновить целевой элемент отношения", | ||||
|     "on_all_matched_notes": "На всех совпадающих заметках" | ||||
|     "on_all_matched_notes": "На всех совпадающих заметках", | ||||
|     "allowed_characters": "Разрешены буквенно-цифровые символы, подчеркивание и двоеточие.", | ||||
|     "change_target_note": "изменить целевую заметку существующего отношения" | ||||
|   }, | ||||
|   "attachments_actions": { | ||||
|     "download": "Скачать", | ||||
| @@ -799,7 +892,14 @@ | ||||
|     "open_custom_title": "Файл будет открыт во внешнем приложении и отслеживаться на наличие изменений. После этого вы сможете загрузить изменённую версию обратно в Trilium.", | ||||
|     "open_externally_title": "Файл будет открыт во внешнем приложении и отслеживаться на наличие изменений. После этого вы сможете загрузить изменённую версию обратно в Trilium.", | ||||
|     "copy_link_to_clipboard": "Копировать ссылку в буфер обмена", | ||||
|     "convert_attachment_into_note": "Преобразовать вложение в заметку" | ||||
|     "convert_attachment_into_note": "Преобразовать вложение в заметку", | ||||
|     "delete_success": "Вложение \"{{title}}\" удалено.", | ||||
|     "enter_new_name": "Введите новое название вложения", | ||||
|     "upload_success": "Загружена новая версия вложения.", | ||||
|     "upload_failed": "Не удалось загрузить новую версию вложения.", | ||||
|     "delete_confirm": "Вы уверены, что хотите удалить вложение '{{title}}'?", | ||||
|     "convert_confirm": "Вы уверены, что хотите преобразовать вложение '{{title}}' в отдельную заметку?", | ||||
|     "convert_success": "Вложение '{{title}}' преобразовано в заметку." | ||||
|   }, | ||||
|   "calendar": { | ||||
|     "mon": "Пн", | ||||
| @@ -867,7 +967,7 @@ | ||||
|     "editable": "Изменяемое", | ||||
|     "language": "Язык", | ||||
|     "note_type": "Тип", | ||||
|     "basic_properties": "Общие параметры" | ||||
|     "basic_properties": "Общее" | ||||
|   }, | ||||
|   "book_properties": { | ||||
|     "grid": "Сетка", | ||||
| @@ -886,8 +986,8 @@ | ||||
|   }, | ||||
|   "edited_notes": { | ||||
|     "deleted": "(удалено)", | ||||
|     "title": "Отредактированные заметки", | ||||
|     "no_edited_notes_found": "Пока нет отредактированных заметок за этот день..." | ||||
|     "title": "Измененные заметки", | ||||
|     "no_edited_notes_found": "Пока нет измененных заметок за этот день..." | ||||
|   }, | ||||
|   "file_properties": { | ||||
|     "download": "Скачать", | ||||
| @@ -898,7 +998,8 @@ | ||||
|     "file_size": "Размер файла", | ||||
|     "file_type": "Тип файла", | ||||
|     "original_file_name": "Исходное имя файла", | ||||
|     "note_id": "ID заметки" | ||||
|     "note_id": "ID заметки", | ||||
|     "upload_failed": "Загрузка новой версии файла не удалась." | ||||
|   }, | ||||
|   "image_properties": { | ||||
|     "download": "Скачать", | ||||
| @@ -919,7 +1020,9 @@ | ||||
|     "note_id": "ID заметки", | ||||
|     "note_size": "Размер заметки", | ||||
|     "title": "Информация", | ||||
|     "calculate": "подсчитать" | ||||
|     "calculate": "подсчитать", | ||||
|     "note_size_info": "Размер заметки позволяет приблизительно оценить требования к объёму хранилища для данной заметки. Он учитывает её содержание и содержание её сохраненных версий.", | ||||
|     "subtree_size": "(размер поддерева: {{size}} в {{count}} заметках)" | ||||
|   }, | ||||
|   "note_paths": { | ||||
|     "search": "Поиск", | ||||
| @@ -927,11 +1030,12 @@ | ||||
|     "clone_button": "Клонировать заметку в новое место...", | ||||
|     "intro_placed": "Эта заметка размещена по следующим путям:", | ||||
|     "intro_not_placed": "Эта заметка еще не помещена в дерево заметок.", | ||||
|     "outside_hoisted": "Этот путь находится за пределами выделенный заметки, и вам придется снять выделение.", | ||||
|     "outside_hoisted": "Этот путь находится за пределами закрепленной заметки, и вам придется снять закрепление.", | ||||
|     "archived": "Архивировано" | ||||
|   }, | ||||
|   "note_properties": { | ||||
|     "info": "Информация" | ||||
|     "info": "Информация", | ||||
|     "this_note_was_originally_taken_from": "Эта заметка была первоначально взята из:" | ||||
|   }, | ||||
|   "promoted_attributes": { | ||||
|     "url_placeholder": "http://website...", | ||||
| @@ -989,7 +1093,8 @@ | ||||
|     "access_info": "Чтобы получить доступ к отладочной информации, выполните запрос и нажмите «Показать лог бэкенда» в левом верхнем углу." | ||||
|   }, | ||||
|   "limit": { | ||||
|     "limit": "Ограничение" | ||||
|     "limit": "Ограничение", | ||||
|     "take_first_x_results": "Взять только первые X указанных результатов." | ||||
|   }, | ||||
|   "order_by": { | ||||
|     "title": "Названию", | ||||
| @@ -1005,7 +1110,10 @@ | ||||
|     "owned_label_count": "Количество меток", | ||||
|     "owned_relation_count": "Количество отношений", | ||||
|     "date_modified": "Дата последнего изменения", | ||||
|     "children_count": "Количество дочерних заметок" | ||||
|     "children_count": "Количество дочерних заметок", | ||||
|     "content_and_attachments_size": "Размер содержимого заметки, включая вложения", | ||||
|     "content_and_attachments_and_revisions_size": "Размер содержимого заметки, включая вложения и версии.", | ||||
|     "target_relation_count": "Количество отношений, направленных на заметку" | ||||
|   }, | ||||
|   "search_string": { | ||||
|     "search_prefix": "Поиск:", | ||||
| @@ -1020,7 +1128,8 @@ | ||||
|     "label_rock_or_pop": "должна присутствовать только одна из vtnjr", | ||||
|     "label_year_comparison": "числовое сравнение (также >, >=, <).", | ||||
|     "label_date_created": "заметки, созданные за последний месяц", | ||||
|     "error": "Ошибка поиска: {{error}}" | ||||
|     "error": "Ошибка поиска: {{error}}", | ||||
|     "placeholder": "полнотекстовые ключевые слова, #tag = value..." | ||||
|   }, | ||||
|   "backend_log": { | ||||
|     "refresh": "Обновить" | ||||
| @@ -1032,7 +1141,8 @@ | ||||
|     "full_sync_triggered": "Полная синхронизация запущена", | ||||
|     "finished-successfully": "Синхронизация успешно завершена.", | ||||
|     "failed": "Синхронизация не удалась: {{message}}", | ||||
|     "sync_rows_filled_successfully": "Строки синхронизации успешно заполнены" | ||||
|     "sync_rows_filled_successfully": "Строки синхронизации успешно заполнены", | ||||
|     "filling_entity_changes": "Заполнение строк изменений сущностей..." | ||||
|   }, | ||||
|   "fonts": { | ||||
|     "fonts": "Шрифты", | ||||
| @@ -1052,7 +1162,10 @@ | ||||
|     "sans-serif-system-fonts": "Системные шрифты без засечек", | ||||
|     "serif-system-fonts": "Системные шрифты с засечками", | ||||
|     "monospace-system-fonts": "Моноширинные системные шрифты", | ||||
|     "handwriting-system-fonts": "Шрифты системы рукописного ввода" | ||||
|     "handwriting-system-fonts": "Шрифты системы рукописного ввода", | ||||
|     "note_tree_and_detail_font_sizing": "Обратите внимание, что размер шрифта дерева и детальной страницы зависит от настройки размера основного шрифта.", | ||||
|     "apply_font_changes": "Чтобы применить изменения шрифта, нажмите", | ||||
|     "not_all_fonts_available": "Не все перечисленные шрифты могут быть доступны в вашей системе." | ||||
|   }, | ||||
|   "max_content_width": { | ||||
|     "max_width_unit": "пикселей", | ||||
| @@ -1129,13 +1242,14 @@ | ||||
|     "index_status": "Статус индексирования", | ||||
|     "indexed_notes": "Проиндексированные заметки", | ||||
|     "indexing_stopped": "Индексирование остановлено", | ||||
|     "last_indexed": "Последние проиндексированные", | ||||
|     "last_indexed": "Индексировано в последний раз", | ||||
|     "note_chat": "Чат по заметке", | ||||
|     "start_indexing": "Начать индексирование", | ||||
|     "chat": { | ||||
|       "root_note_title": "Чаты с AI", | ||||
|       "new_chat_title": "Новый чат", | ||||
|       "create_new_ai_chat": "Создать новый чат с ИИ" | ||||
|       "create_new_ai_chat": "Создать новый чат с ИИ", | ||||
|       "root_note_content": "В этой заметке содержатся сохраненные вами разговоры в чате ИИ." | ||||
|     }, | ||||
|     "selected_provider": "Выбранный провайдер", | ||||
|     "select_model": "Выбрать модель...", | ||||
| @@ -1151,7 +1265,10 @@ | ||||
|     "temperature_description": "Контролирует случайность ответов (0 = детерминированный, 2 = максимальная случайность)", | ||||
|     "system_prompt_description": "Системный промпт по умолчанию, используемый для всех взаимодействий с ИИ", | ||||
|     "empty_key_warning": { | ||||
|       "openai": "Ключ API OpenAI пуст. Введите действительный ключ API." | ||||
|       "openai": "Ключ API OpenAI пуст. Введите действительный ключ API.", | ||||
|       "ollama": "API-ключ Ollama пуст. Введите действительный API-ключ.", | ||||
|       "voyage": "Ключ API Voyage пуст. Введите действительный ключ API.", | ||||
|       "anthropic": "Ключ API Anthropic пуст. Введите действительный ключ API." | ||||
|     }, | ||||
|     "openai_api_key_description": "Ваш ключ API OpenAI для доступа к их службам ИИ", | ||||
|     "provider_precedence_description": "Список провайдеров, разделенных запятыми, в порядке приоритета (например, \"openai,anthropic,ollama\")", | ||||
| @@ -1181,7 +1298,37 @@ | ||||
|     "failed_to_retry_note": "Не удалось повторить попытку", | ||||
|     "failed_to_retry_all": "Не удалось повторить попытку", | ||||
|     "error_generating_response": "Ошибка генерации ответа ИИ", | ||||
|     "create_new_ai_chat": "Создать новый чат с ИИ" | ||||
|     "create_new_ai_chat": "Создать новый чат с ИИ", | ||||
|     "ai_enabled": "Возможности ИИ активны", | ||||
|     "ai_disabled": "Возможности ИИ неактивны", | ||||
|     "restore_provider": "Восстановить значение провайдера", | ||||
|     "error_fetching": "Ошибка получения списка моделей: {{error}}", | ||||
|     "index_rebuild_status_error": "Ошибка проверки статуса перестроения индекса", | ||||
|     "enhanced_context_description": "Предоставляет ИИ больше контекста из заметки и связанных с ней заметок для более точных ответов", | ||||
|     "n_notes_queued_0": "{{ count }} заметка в очереди на индексирование", | ||||
|     "n_notes_queued_1": "{{ count }} заметки в очереди на индексирование", | ||||
|     "n_notes_queued_2": "{{ count }} заметок в очереди на индексирование", | ||||
|     "no_models_found_ollama": "Модели Ollama не найдены. Проверьте, запущена ли Ollama.", | ||||
|     "no_models_found_online": "Модели не найдены. Проверьте ваш ключ API и настройки.", | ||||
|     "experimental_warning": "Функция LLM в настоящее время является экспериментальной — вы предупреждены.", | ||||
|     "ollama_no_url": "Ollama не настроена. Введите корректный URL-адрес.", | ||||
|     "notes_indexed_0": "{{ count }} заметка проиндексирована", | ||||
|     "notes_indexed_1": "{{ count }} заметки проиндексировано", | ||||
|     "notes_indexed_2": "{{ count }} заметок проиндексировано", | ||||
|     "show_thinking_description": "Показать цепочку мыслительного процесса ИИ", | ||||
|     "api_key_tooltip": "API-ключ для доступа к сервису", | ||||
|     "all_notes_queued_for_retry": "Все неудачные заметки поставлены в очередь на повторную попытку", | ||||
|     "reprocess_index_started": "Оптимизация поискового индекса запущена в фоновом режиме", | ||||
|     "similarity_threshold_description": "Минимальный показатель сходства (similarity score, 0–1) для заметок, которые следует включить в контекст запросов LLM", | ||||
|     "max_notes_per_llm_query_description": "Максимальное количество похожих заметок для включения в контекст ИИ", | ||||
|     "retry_failed": "Не удалось поставить заметку в очередь для повторной попытки", | ||||
|     "rebuild_index_error": "Ошибка при запуске перестроения индекса. Подробности смотрите в логах.", | ||||
|     "enable_ollama_description": "Включить Ollama для использования локальной модели ИИ", | ||||
|     "anthropic_model_description": "Модели Anthropic Claude для автодополнения чата", | ||||
|     "anthropic_url_description": "Базовый URL для Anthropic API (по умолчанию: https://api.anthropic.com)", | ||||
|     "anthropic_api_key_description": "Ваш ключ Anthropic API для доступа к моделям Claude", | ||||
|     "enable_ai_desc": "Включить функции ИИ, такие как резюмирование заметок, генерация контента и другие возможности LLM", | ||||
|     "enable_ai_description": "Включить функции ИИ, такие как резюмирование заметок, генерация контента и другие возможности LLM" | ||||
|   }, | ||||
|   "code-editor-options": { | ||||
|     "title": "Редактор" | ||||
| @@ -1245,7 +1392,7 @@ | ||||
|   }, | ||||
|   "backup": { | ||||
|     "path": "Путь", | ||||
|     "backup_now": "Резервное копирование сейчас", | ||||
|     "backup_now": "Принудительное резервное копирование", | ||||
|     "existing_backups": "Существующие резервные копии", | ||||
|     "automatic_backup": "Автоматическое резервное копирование", | ||||
|     "automatic_backup_description": "Trilium может автоматически создавать резервную копию базы данных:", | ||||
| @@ -1254,7 +1401,9 @@ | ||||
|     "enable_monthly_backup": "Включить ежемесячное резервное копирование", | ||||
|     "backup_database_now": "Создать резервную копию", | ||||
|     "date-and-time": "Дата и время", | ||||
|     "no_backup_yet": "нет резервных копий" | ||||
|     "no_backup_yet": "нет резервных копий", | ||||
|     "database_backed_up_to": "Резервная копия базы данных создана в {{backupFilePath}}", | ||||
|     "backup_recommendation": "Рекомендуется держать резервное копирование включенным, но это может замедлить запуск приложений при использовании больших баз данных и/или медленных устройств хранения." | ||||
|   }, | ||||
|   "etapi": { | ||||
|     "title": "ETAPI", | ||||
| @@ -1272,7 +1421,14 @@ | ||||
|     "swagger_ui": "Пользовательский интерфейс ETAPI Swagger", | ||||
|     "new_token_title": "Новый токен ETAPI", | ||||
|     "token_created_title": "Создан токен ETAPI", | ||||
|     "rename_token": "Переименовать этот токен" | ||||
|     "rename_token": "Переименовать этот токен", | ||||
|     "new_token_message": "Введите название нового токена", | ||||
|     "error_empty_name": "Имя токена не может быть пустым", | ||||
|     "delete_token": "Удалить/деактивировать этот токен", | ||||
|     "rename_token_message": "Пожалуйста, введите имя нового токена", | ||||
|     "token_created_message": "Скопируйте созданный токен в буфер обмена. Trilium сохранит хеш токена, и вы его больше не сможете увидеть.", | ||||
|     "delete_token_confirmation": "Вы уверены, что хотите удалить токен ETAPI \"{{name}}\"?", | ||||
|     "no_tokens_yet": "Токенов пока нет. Нажмите кнопку выше, чтобы создать токен." | ||||
|   }, | ||||
|   "multi_factor_authentication": { | ||||
|     "oauth_title": "OAuth/OpenID", | ||||
| @@ -1296,7 +1452,16 @@ | ||||
|     "totp_title": "Одноразовый пароль с ограничением по времени (TOTP)", | ||||
|     "recovery_keys_title": "Ключи восстановления единого входа", | ||||
|     "recovery_keys_error": "Ошибка генерации кодов восстановления", | ||||
|     "recovery_keys_no_key_set": "Коды восстановления не установлены" | ||||
|     "recovery_keys_no_key_set": "Коды восстановления не установлены", | ||||
|     "recovery_keys_unused": "Код восстановления {{index}} не используется", | ||||
|     "description": "Многофакторная аутентификация (MFA) добавляет дополнительный уровень безопасности вашей учётной записи. Вместо простого ввода пароля для входа MFA требует предоставить один или несколько дополнительных документов для подтверждения вашей личности. Таким образом, даже если кто-то узнает ваш пароль, он всё равно не сможет получить доступ к вашей учётной записи без второго элемента данных. Это похоже на установку дополнительного замка на вашу дверь, значительно усложняя взлом.<br><br>Следуйте инструкциям ниже, чтобы включить MFA. Если вы настроите её неправильно, для входа будет использоваться только пароль.", | ||||
|     "totp_description": "TOTP (одноразовый пароль с ограничением по времени) — это функция безопасности, которая генерирует уникальный временный код, который меняется каждые 30 секунд. Вы используете этот код вместе с паролем для входа в свою учётную запись, что значительно затрудняет доступ к ней посторонним лицам.", | ||||
|     "recovery_keys_description_warning": "Ключи восстановления больше не будут отображаться после выхода со страницы, сохраните их в надежном и безопасном месте.<br>После использования ключа восстановления его нельзя будет использовать повторно.", | ||||
|     "totp_secret_regenerate_confirm": "Вы уверены, что хотите восстановить секрет TOTP? Это аннулирует предыдущий секрет TOTP и все существующие коды восстановления.", | ||||
|     "totp_secret_description_warning": "После создания нового секрета TOTP вам потребуется снова войти в систему, используя новый секрет TOTP.", | ||||
|     "recovery_keys_description": "Ключи восстановления единого входа используются для входа в систему, даже если вы не можете получить доступ к своим кодам аутентификатора.", | ||||
|     "totp_secret_warning": "Сохраните сгенерированный секретный ключ в безопасном месте. Он больше не будет показан.", | ||||
|     "no_totp_secret_warning": "Чтобы включить TOTP, вам сначала нужно сгенерировать секрет TOTP." | ||||
|   }, | ||||
|   "shortcuts": { | ||||
|     "shortcuts": "Сочетания клавиш", | ||||
| @@ -1305,7 +1470,11 @@ | ||||
|     "action_name": "Название действия", | ||||
|     "default_shortcuts": "Сочетания клавиш по умолчанию", | ||||
|     "multiple_shortcuts": "Несколько сочетаний клавиш для одного и того же действия можно разделить запятой.", | ||||
|     "electron_documentation": "Информацию о доступных модификаторах и кодах клавиш см. в <a href=\"https://www.electronjs.org/docs/latest/api/accelerator\">документации Electron</a>." | ||||
|     "electron_documentation": "Информацию о доступных модификаторах и кодах клавиш см. в <a href=\"https://www.electronjs.org/docs/latest/api/accelerator\">документации Electron</a>.", | ||||
|     "type_text_to_filter": "Введите текст для фильтрации сочетаний клавиш...", | ||||
|     "reload_app": "Перезагрузить приложение, чтобы применить изменения", | ||||
|     "confirm_reset": "Вы действительно хотите сбросить все сочетания клавиш до значений по умолчанию?", | ||||
|     "set_all_to_default": "Установить все сочетания клавиш по умолчанию" | ||||
|   }, | ||||
|   "sync_2": { | ||||
|     "timeout_unit": "миллисекунд", | ||||
| @@ -1320,28 +1489,35 @@ | ||||
|     "timeout": "Тайм-аут синхронизации", | ||||
|     "test_description": "Это проверит подключение и подтверждение связи с сервером синхронизации. Если сервер синхронизации не инициализирован, он будет настроен на синхронизацию с локальным документом.", | ||||
|     "test_title": "Тест синхронизации", | ||||
|     "test_button": "Проверка синхронизации" | ||||
|     "test_button": "Проверка синхронизации", | ||||
|     "handshake_failed": "Синхронизация с сервером не удалась, ошибка: {{message}}" | ||||
|   }, | ||||
|   "api_log": { | ||||
|     "close": "Закрыть" | ||||
|   }, | ||||
|   "bookmark_switch": { | ||||
|     "bookmark": "В закладки", | ||||
|     "remove_bookmark": "Удалить закладку" | ||||
|     "remove_bookmark": "Удалить закладку", | ||||
|     "bookmark_this_note": "Добавить эту заметку в закладки на левой боковой панели" | ||||
|   }, | ||||
|   "editability_select": { | ||||
|     "auto": "Авто", | ||||
|     "read_only": "Только для чтения", | ||||
|     "always_editable": "Всегда доступно для редактирования" | ||||
|     "always_editable": "Всегда доступно для редактирования", | ||||
|     "note_is_always_editable": "Заметку всегда можно редактировать, независимо от ее длины.", | ||||
|     "note_is_read_only": "Заметка доступна только для чтения, но ее можно редактировать одним нажатием кнопки.", | ||||
|     "note_is_editable": "Заметку можно редактировать, если она не слишком длинная." | ||||
|   }, | ||||
|   "shared_switch": { | ||||
|     "shared": "Общий доступ", | ||||
|     "toggle-on-title": "Сделать заметку общедоступной", | ||||
|     "toggle-off-title": "Отменить общий доступ к заметке" | ||||
|     "toggle-off-title": "Отменить общий доступ к заметке", | ||||
|     "shared-branch": "Эта заметка существует только как общая, и если отменить общий доступ, она будет удалена. Хотите продолжить и удалить эту заметку?", | ||||
|     "inherited": "Заметка не может быть убрана из общего доступа в данном случае, поскольку общий доступ передан по наследству от предка." | ||||
|   }, | ||||
|   "highlights_list_2": { | ||||
|     "options": "Параметры", | ||||
|     "title": "Список выделений" | ||||
|     "title": "Список выделенного" | ||||
|   }, | ||||
|   "include_note": { | ||||
|     "dialog_title": "Вставить заметку", | ||||
| @@ -1354,26 +1530,42 @@ | ||||
|     "box_size_prompt": "Размер рамки вставленной заметки:" | ||||
|   }, | ||||
|   "execute_script": { | ||||
|     "execute_script": "Выполнить скрипт" | ||||
|     "execute_script": "Выполнить скрипт", | ||||
|     "help_text": "Вы можете выполнять простые скрипты на соответствующих заметках.", | ||||
|     "example_1": "Например, чтобы добавить строку к заголовку заметки, используйте этот небольшой скрипт:", | ||||
|     "example_2": "Более сложным примером будет удаление всех соответствующих атрибутов заметки:" | ||||
|   }, | ||||
|   "update_label_value": { | ||||
|     "label_name_placeholder": "название метки", | ||||
|     "new_value_placeholder": "новое значение", | ||||
|     "update_label_value": "Обновить значение метки", | ||||
|     "to_value": "на значение" | ||||
|     "to_value": "на значение", | ||||
|     "help_text_note": "Вы также можете вызвать этот метод без значения, в таком случае метка будет присвоена заметке без значения.", | ||||
|     "label_name_title": "Разрешены буквенно-цифровые символы, подчеркивание и двоеточие.", | ||||
|     "help_text": "На всех соответствующих заметках изменить значение существующей метки." | ||||
|   }, | ||||
|   "delete_note": { | ||||
|     "delete_note": "Удалить заметку", | ||||
|     "delete_matched_notes": "Удалить совпадающие заметки" | ||||
|     "delete_matched_notes": "Удалить совпадающие заметки", | ||||
|     "delete_matched_notes_description": "Это приведет к удалению соответствующих заметок.", | ||||
|     "erase_notes_instruction": "Чтобы навсегда стереть заметки, после удаления перейдите в раздел «Параметры» -> «Другие» и нажмите кнопку «Стереть удаленные заметки сейчас».", | ||||
|     "undelete_notes_instruction": "После удаления их можно восстановить из диалогового окна «Последние изменения»." | ||||
|   }, | ||||
|   "rename_note": { | ||||
|     "rename_note": "Переименовать заметку", | ||||
|     "new_note_title": "название новой заметки", | ||||
|     "rename_note_title_to": "Переименовать заголовок заметки на" | ||||
|     "rename_note_title_to": "Переименовать заголовок заметки на", | ||||
|     "click_help_icon": "Нажмите значок справки справа, чтобы увидеть все параметры", | ||||
|     "evaluated_as_js_string": "Указанное значение обрабатывается как строка JavaScript и, таким образом, может быть дополнено динамическим содержимым через внедренную переменную <code>note</code> (при этом заметка переименовывается). Примеры:", | ||||
|     "example_note": "<code>Note</code> — все соответствующие примечания переименовываются в \"Note\"", | ||||
|     "example_new_title": "<code>NEW: ${note.title}</code> — заголовки соответствующих заметок начинаются с префикса \"NEW:\"", | ||||
|     "example_date_prefix": "<code>${note.dateCreatedObj.format('MM-DD:')}: ${note.title}</code> — соответствующие заметки имеют префикс в виде месяца и даты создания заметки", | ||||
|     "api_docs": "Подробную информацию см. в документации API для <a href='https://zadam.github.io/trilium/backend_api/Note.html'>note</a> и его <a href='https://day.js.org/docs/en/display/format'>свойства dateCreatedObj / utcDateCreatedObj</a>." | ||||
|   }, | ||||
|   "delete_relation": { | ||||
|     "delete_relation": "Удалить отношение", | ||||
|     "relation_name": "название отношения" | ||||
|     "relation_name": "название отношения", | ||||
|     "allowed_characters": "Разрешены буквенно-цифровые символы, подчеркивание и двоеточие." | ||||
|   }, | ||||
|   "left_pane_toggle": { | ||||
|     "show_panel": "Показать панель", | ||||
| @@ -1394,7 +1586,7 @@ | ||||
|     "save_revision": "Сохранить версию", | ||||
|     "convert_into_attachment": "Конвертировать во вложение", | ||||
|     "search_in_note": "Поиск в заметке", | ||||
|     "print_pdf": "Экспорт в PDF", | ||||
|     "print_pdf": "Экспорт в PDF...", | ||||
|     "convert_into_attachment_prompt": "Вы уверены, что хотите преобразовать заметку '{{title}}' во вложение родительской заметки?", | ||||
|     "convert_into_attachment_successful": "Примечание '{{title}}' преобразовано во вложение.", | ||||
|     "convert_into_attachment_failed": "Не удалось преобразовать заметку '{{title}}'.", | ||||
| @@ -1457,7 +1649,9 @@ | ||||
|   }, | ||||
|   "attachment_list": { | ||||
|     "upload_attachments": "Загрузка вложений", | ||||
|     "owning_note": "Заметка-владелец: " | ||||
|     "owning_note": "Заметка-владелец: ", | ||||
|     "open_help_page": "Открыть справку по вложениям", | ||||
|     "no_attachments": "Заметка не содержит вложений." | ||||
|   }, | ||||
|   "protected_session": { | ||||
|     "wrong_password": "Неверный пароль.", | ||||
| @@ -1467,7 +1661,9 @@ | ||||
|     "unprotecting-finished-successfully": "Снятие защиты успешно завершено.", | ||||
|     "start_session_button": "Начать защищенный сеанс <kbd>enter</kbd>", | ||||
|     "protecting-in-progress": "Защита в процессе: {{count}}", | ||||
|     "unprotecting-in-progress-count": "Снятие защиты в процессе: {{count}}" | ||||
|     "unprotecting-in-progress-count": "Снятие защиты в процессе: {{count}}", | ||||
|     "started": "Защищенный сеанс запущен.", | ||||
|     "enter_password_instruction": "Для отображения защищенной заметки требуется ввести пароль:" | ||||
|   }, | ||||
|   "relation_map": { | ||||
|     "remove_note": "Удалить заметку", | ||||
| @@ -1479,7 +1675,13 @@ | ||||
|     "confirm_remove_relation": "Вы уверены, что хотите удалить отношение?", | ||||
|     "enter_new_title": "Введите новое название заметки:", | ||||
|     "note_not_found": "Заметка {{noteId}} не найдена!", | ||||
|     "cannot_match_transform": "Невозможно сопоставить преобразование: {{transform}}" | ||||
|     "cannot_match_transform": "Невозможно сопоставить преобразование: {{transform}}", | ||||
|     "enter_title_of_new_note": "Введите название новой заметки", | ||||
|     "click_on_canvas_to_place_new_note": "Щелкните по холсту, чтобы разместить новую заметку", | ||||
|     "note_already_in_diagram": "Заметка \"{{title}}\" уже есть на диаграмме.", | ||||
|     "connection_exists": "Связь '{{name}}' между этими заметками уже существует.", | ||||
|     "specify_new_relation_name": "Укажите новое имя отношения (допустимые символы: буквы, цифры, двоеточие и подчеркивание):", | ||||
|     "start_dragging_relations": "Начните перетягивать отношения отсюда на другую заметку." | ||||
|   }, | ||||
|   "vacuum_database": { | ||||
|     "title": "Сжатие базы данных", | ||||
| @@ -1501,22 +1703,22 @@ | ||||
|     "enable_tray": "Включить отображение иконки в системном трее (чтобы изменения вступили в силу, необходимо перезапустить Trilium)" | ||||
|   }, | ||||
|   "highlights_list": { | ||||
|     "title": "Список выделений", | ||||
|     "title": "Список выделенного", | ||||
|     "bold": "Жирный текст", | ||||
|     "italic": "Наклонный текст", | ||||
|     "underline": "Подчеркнутый текст", | ||||
|     "color": "Цветной текст", | ||||
|     "description": "Вы можете настроить список выделений, отображаемый на правой панели:", | ||||
|     "description": "Вы можете настроить список выделенного, отображаемый на правой панели:", | ||||
|     "bg_color": "Текст с заливкой фона", | ||||
|     "visibility_title": "Видимость списка выделений", | ||||
|     "visibility_description": "Вы можете скрыть виджет списка выделений, добавив атрибут #hideHighlightWidget к заметке.", | ||||
|     "shortcut_info": "Вы можете настроить сочетание клавиш для быстрого переключения правой панели (включая список выделений) в меню Параметры -> Сочетания клавиш (название \"toggleRightPane\")." | ||||
|     "visibility_description": "Вы можете скрыть виджет списка выделенного, добавив атрибут #hideHighlightWidget к заметке.", | ||||
|     "shortcut_info": "Вы можете настроить сочетание клавиш для быстрого переключения правой панели (включая список выделенного) в меню Параметры -> Сочетания клавиш (название \"toggleRightPane\")." | ||||
|   }, | ||||
|   "custom_date_time_format": { | ||||
|     "format_string": "Строка форматирования:", | ||||
|     "formatted_time": "Пример форматирования:", | ||||
|     "title": "Пользовательский формат даты и времени", | ||||
|     "description": "Настройте формат даты и времени, вставляемых с помощью <kbd>Alt+T</kbd> или панели инструментов. Доступные токены формата см. в <a href=\"https://day.js.org/docs/en/display/format\" target=\"_blank\" rel=\"noopener noreferrer\">документации Day.js</a>." | ||||
|     "description": "Настройте формат даты и времени, вставляемых с помощью <shortcut /> или панели инструментов. Доступные токены форматирования см. в <doc>документации Day.js</doc>." | ||||
|   }, | ||||
|   "spellcheck": { | ||||
|     "title": "Проверка орфографии", | ||||
| @@ -1525,13 +1727,20 @@ | ||||
|     "multiple_languages_info": "Несколько языков можно разделять запятой, например, \"en-US, de-DE, cs\". ", | ||||
|     "available_language_codes_label": "Доступные коды языков:", | ||||
|     "restart-required": "Изменения параметров проверки орфографии вступят в силу после перезапуска приложения.", | ||||
|     "language_code_placeholder": "например \"en-US\", \"de-AT\"" | ||||
|     "language_code_placeholder": "например \"en-US\", \"de-AT\"", | ||||
|     "description": "Эти параметры применимы только для десктопных сборок, браузеры будут использовать собственную встроенную проверку орфографии." | ||||
|   }, | ||||
|   "attribute_editor": { | ||||
|     "save_attributes": "Сохранить атрибуты <enter>", | ||||
|     "add_a_new_attribute": "Добавить новый атрибут", | ||||
|     "add_new_label_definition": "Добавить новое определение метки", | ||||
|     "add_new_relation_definition": "Добавить новое определение отношения" | ||||
|     "add_new_relation_definition": "Добавить новое определение отношения", | ||||
|     "add_new_label": "Добавить новую метку <kbd data-command=\"addNewLabel\"></kbd>", | ||||
|     "add_new_relation": "Добавить новое отношение <kbd data-command=\"addNewRelation\"></kbd>", | ||||
|     "help_text_body1": "Чтобы добавить метку, просто введите, например, <code>#rock</code> или, если вы хотите добавить также значение, то, например, <code>#year = 2020</code>", | ||||
|     "help_text_body2": "Для отношения введите <code>~author = @</code>, после чего должно появиться окно автозаполнения, где вы сможете найти нужную заметку.", | ||||
|     "help_text_body3": "В качестве альтернативы вы можете добавить метку и отношение, используя кнопку <code>+</code> с правой стороны.", | ||||
|     "placeholder": "Введите здесь метки и отношения" | ||||
|   }, | ||||
|   "delete_revisions": { | ||||
|     "delete_note_revisions": "Удалить версии заметки", | ||||
| @@ -1547,7 +1756,7 @@ | ||||
|     "edit_this_note": "Редактировать заметку" | ||||
|   }, | ||||
|   "show_highlights_list_widget_button": { | ||||
|     "show_highlights_list": "Показать список выделений" | ||||
|     "show_highlights_list": "Показать список выделенного" | ||||
|   }, | ||||
|   "zen_mode": { | ||||
|     "button_exit": "Покинуть режим \"дзен\"" | ||||
| @@ -1587,7 +1796,9 @@ | ||||
|   }, | ||||
|   "protect_note": { | ||||
|     "toggle-on": "Защитить заметку", | ||||
|     "toggle-off": "Снять защиту с заметки" | ||||
|     "toggle-off": "Снять защиту с заметки", | ||||
|     "toggle-off-hint": "Заметка защищена. Щелкните, чтобы снять защиту", | ||||
|     "toggle-on-hint": "Заметка не защищена. Щелкните, чтобы установить защиту" | ||||
|   }, | ||||
|   "note-map": { | ||||
|     "button-link-map": "Карта связей", | ||||
| @@ -1604,15 +1815,21 @@ | ||||
|   "consistency_checks": { | ||||
|     "find_and_fix_button": "Найти и устранить проблемы целостности", | ||||
|     "finding_and_fixing_message": "Поиск и устранение проблем целостности...", | ||||
|     "title": "Проверки целостности" | ||||
|     "title": "Проверки целостности", | ||||
|     "issues_fixed_message": "Все обнаруженные проблемы с согласованностью теперь устранены." | ||||
|   }, | ||||
|   "call_to_action": { | ||||
|     "next_theme_message": "В настоящее время вы используете старую тему оформления. Хотите попробовать новую тему?", | ||||
|     "dismiss": "Отклонить", | ||||
|     "background_effects_button": "Включить эффекты фона" | ||||
|     "background_effects_button": "Включить эффекты фона", | ||||
|     "next_theme_button": "Попробовать новую тему", | ||||
|     "background_effects_message": "На устройствах Windows фоновые эффекты теперь полностью стабильны. Они добавляют цвет в пользовательский интерфейс, размывая фон за ним. Этот приём также используется в других приложениях, например, в проводнике Windows.", | ||||
|     "background_effects_title": "Фоновые эффекты теперь стабильны", | ||||
|     "next_theme_title": "Попробуйте новую тему Trilium" | ||||
|   }, | ||||
|   "zoom_factor": { | ||||
|     "description": "Масштабированием также можно управлять с помощью сочетаний клавиш CTRL+- и CTRL+=." | ||||
|     "description": "Масштабированием также можно управлять с помощью сочетаний клавиш CTRL+- и CTRL+=.", | ||||
|     "title": "Коэффициент масштабирования (только для настольной версии)" | ||||
|   }, | ||||
|   "show_toc_widget_button": { | ||||
|     "show_toc": "Показать оглавление" | ||||
| @@ -1621,7 +1838,8 @@ | ||||
|     "title": "Доступные типы в выпадающем списке" | ||||
|   }, | ||||
|   "search_result": { | ||||
|     "no_notes_found": "По заданным параметрам поиска заметки не найдены." | ||||
|     "no_notes_found": "По заданным параметрам поиска заметки не найдены.", | ||||
|     "search_not_executed": "Поиск ещё не выполнен. Нажмите кнопку «Поиск» выше, чтобы увидеть результаты." | ||||
|   }, | ||||
|   "empty": { | ||||
|     "search_placeholder": "поиск заметки по ее названию", | ||||
| @@ -1631,7 +1849,11 @@ | ||||
|   "search_script": { | ||||
|     "placeholder": "поиск заметки по ее названию", | ||||
|     "title": "Скрипт поиска:", | ||||
|     "example_title": "См. этот пример:" | ||||
|     "example_title": "См. этот пример:", | ||||
|     "description1": "Скрипт поиска позволяет определить результаты поиска, запустив его. Это обеспечивает максимальную гибкость, когда стандартного поиска недостаточно.", | ||||
|     "description2": "Скрипт поиска должен иметь тип «Код» и подтип «JavaScript backend». Скрипт должен возвращать массив идентификаторов заметок или заметок.", | ||||
|     "note": "Обратите внимание, что скрипт поиска и строка поиска не могут быть объединены друг с другом.", | ||||
|     "example_code": "// 1. Предварительная фильтрация с использованием стандартного поиска\nconst candidateNotes = api.searchForNotes(\"#journal\"); \n\n// 2. Применение пользовательских критериев поиска\nconst matchedNotes = candidateNotes\n    .filter(note => note.title.match(/[0-9]{1,2}\\. ?[0-9]{1,2}\\. ?[0-9]{4}/));\n\nreturn matchedNotes;" | ||||
|   }, | ||||
|   "note_erasure_timeout": { | ||||
|     "note_erasure_timeout_title": "Срок окончательного удаления заметок", | ||||
| @@ -1645,21 +1867,25 @@ | ||||
|     "erase_unused_attachments_now": "Удалить неиспользуемые вложения прямо сейчас", | ||||
|     "attachment_auto_deletion_description": "Вложения автоматически удаляются, если в заметке на них больше не ссылаются по истечении определенного времени.", | ||||
|     "attachment_erasure_timeout": "Тайм-аут удаления вложения", | ||||
|     "erase_attachments_after": "Удалять неиспользуемые вложения через:" | ||||
|     "erase_attachments_after": "Удалять неиспользуемые вложения через:", | ||||
|     "unused_attachments_erased": "Неиспользуемые вложения были удалены.", | ||||
|     "manual_erasing_description": "Вы также можете запустить стирание вручную (без учета тайм-аута, определенного выше):" | ||||
|   }, | ||||
|   "revisions_snapshot_interval": { | ||||
|     "note_revisions_snapshot_interval_title": "Интервал создания снимка версии заметки", | ||||
|     "note_revisions_snapshot_description": "Интервал создания снимка версии заметки - это время, по истечении которого для неё будет создана новая версия. Подробнее см. в <a href=\"https://triliumnext.github.io/Docs/Wiki/note-revisions.html\" class=\"external\">wiki</a>.", | ||||
|     "note_revisions_snapshot_description": "Интервал между снимками редакции заметки — это время, по истечении которого для заметки будет создана новая редакция. Подробнее см. <doc>wiki</doc>.", | ||||
|     "snapshot_time_interval_label": "Интервал создания снимка версии заметки:" | ||||
|   }, | ||||
|   "title_bar_buttons": { | ||||
|     "window-on-top": "Закрепить окно" | ||||
|   }, | ||||
|   "abstract_search_option": { | ||||
|     "remove_this_search_option": "Удалить эту опцию поиска" | ||||
|     "remove_this_search_option": "Удалить эту опцию поиска", | ||||
|     "failed_rendering": "Не удалось выполнить рендеринг опции поиска: {{dto}} с ошибкой: {{error}} {{stack}}" | ||||
|   }, | ||||
|   "image": { | ||||
|     "copied-to-clipboard": "Ссылка на изображение скопирована в буфер обмена. Её можно вставить в любую текстовую заметку." | ||||
|     "copied-to-clipboard": "Ссылка на изображение скопирована в буфер обмена. Её можно вставить в любую текстовую заметку.", | ||||
|     "cannot-copy": "Не удалось скопировать ссылку на изображение в буфер обмена." | ||||
|   }, | ||||
|   "abstract_bulk_action": { | ||||
|     "remove_this_search_action": "Удалить это действие поиска" | ||||
| @@ -1668,26 +1894,38 @@ | ||||
|     "cannot-move-notes-here": "Невозможно переместить заметки сюда.", | ||||
|     "delete-status": "Статус удаления", | ||||
|     "delete-finished-successfully": "Удаление успешно завершено.", | ||||
|     "undeleting-notes-finished-successfully": "Восстановление заметок успешно завершено." | ||||
|     "undeleting-notes-finished-successfully": "Восстановление заметок успешно завершено.", | ||||
|     "delete-notes-in-progress": "Удаление заметок в процессе: {{count}}", | ||||
|     "undeleting-notes-in-progress": "Идет восстановление заметок: {{count}}" | ||||
|   }, | ||||
|   "attachment_detail": { | ||||
|     "owning_note": "Заметка-владелец: ", | ||||
|     "list_of_all_attachments": "Список всех вложений" | ||||
|     "list_of_all_attachments": "Список всех вложений", | ||||
|     "open_help_page": "Открыть справку по вложениям", | ||||
|     "attachment_deleted": "Это вложение было удалено.", | ||||
|     "you_can_also_open": ", вы также можете открыть " | ||||
|   }, | ||||
|   "web_view": { | ||||
|     "web_view": "Веб-страница" | ||||
|     "web_view": "Веб-страница", | ||||
|     "create_label": "Для начала создайте метку с URL-адресом, который вы хотите встроить, например, #webViewSrc=\"https://www.google.com\"", | ||||
|     "embed_websites": "Заметки типа \"Веб-страница\" позволяет встраивать веб-сайты в Trilium." | ||||
|   }, | ||||
|   "ribbon": { | ||||
|     "widgets": "Виджеты ленты" | ||||
|     "widgets": "Виджеты ленты", | ||||
|     "promoted_attributes_message": "Вкладка \"Продвигаемые атрибуты\" будет автоматически открыта, если таковые атрибуты установлены у заметки", | ||||
|     "edited_notes_message": "Вкладка ленты «Измененные заметки» будет автоматически открываться в заметках дня" | ||||
|   }, | ||||
|   "options_widget": { | ||||
|     "options_status": "Статус опций" | ||||
|     "options_status": "Статус опций", | ||||
|     "options_change_saved": "Изменения параметров сохранены." | ||||
|   }, | ||||
|   "spacer": { | ||||
|     "configure_launchbar": "Конфигурация лаунчбара" | ||||
|   }, | ||||
|   "entrypoints": { | ||||
|     "note-executed": "Заметка выполнена." | ||||
|     "note-executed": "Заметка выполнена.", | ||||
|     "note-revision-created": "Снимок версии заметки создан успешно.", | ||||
|     "sql-error": "Произошла ошибка при выполнении SQL-запроса: {{message}}" | ||||
|   }, | ||||
|   "include_archived_notes": { | ||||
|     "include_archived_notes": "Включить архивные заметки" | ||||
| @@ -1695,19 +1933,96 @@ | ||||
|   "open-help-page": "Открыть страницу справки", | ||||
|   "watched_file_update_status": { | ||||
|     "upload_modified_file": "Загрузить измененный файл", | ||||
|     "ignore_this_change": "Игнорировать это изменение" | ||||
|     "ignore_this_change": "Игнорировать это изменение", | ||||
|     "file_last_modified": "Файл <code class=\"file-path\"></code> был последний раз изменен <span class=\"file-last-modified\"></span>." | ||||
|   }, | ||||
|   "clipboard": { | ||||
|     "copy_success": "Скопировано в буфер обмена." | ||||
|     "copy_success": "Скопировано в буфер обмена.", | ||||
|     "copy_failed": "Невозможно скопировать в буфер обмена из-за проблем с правами доступа.", | ||||
|     "copied": "Заметки скопированы в буфер обмена.", | ||||
|     "cut": "Заметки вырезаны в буфер обмена." | ||||
|   }, | ||||
|   "ws": { | ||||
|     "sync-check-failed": "Проверка синхронизации не удалась!" | ||||
|     "sync-check-failed": "Проверка синхронизации не удалась!", | ||||
|     "encountered-error": "Обнаружена ошибка \"{{message}}\", проверьте консоль.", | ||||
|     "consistency-checks-failed": "Проверка целостности не пройдена! Подробности смотрите в логах." | ||||
|   }, | ||||
|   "attachment_detail_2": { | ||||
|     "role_and_size": "Роль: {{role}}, Размер: {{size}}", | ||||
|     "unrecognized_role": "Нераспознанная роль вложения '{{role}}'." | ||||
|     "unrecognized_role": "Нераспознанная роль вложения '{{role}}'.", | ||||
|     "link_copied": "Ссылка на вложение скопирована в буфер обмена.", | ||||
|     "will_be_deleted_soon": "Это вложение скоро будет автоматически удалено", | ||||
|     "will_be_deleted_in": "Это вложение будет автоматически удалено через {{time}}", | ||||
|     "deletion_reason": ", поскольку вложение не связано с содержимым заметки. Чтобы предотвратить удаление, добавьте ссылку на вложение обратно в содержимое или преобразуйте вложение в заметку." | ||||
|   }, | ||||
|   "note_title": { | ||||
|     "placeholder": "введите здесь название заметки..." | ||||
|   }, | ||||
|   "units": { | ||||
|     "percentage": "%" | ||||
|   }, | ||||
|   "settings": { | ||||
|     "related_settings": "Связанные настройки" | ||||
|   }, | ||||
|   "content_widget": { | ||||
|     "unknown_widget": "Неизвестный виджет \"{{id}}\"." | ||||
|   }, | ||||
|   "wrap_lines": { | ||||
|     "wrap_lines_in_code_notes": "Переносить строки в заметках типа \"Код\"", | ||||
|     "enable_line_wrap": "Включить перенос строк (для вступления изменений в силу может потребоваться перезагрузка интерфейса)" | ||||
|   }, | ||||
|   "shared_info": { | ||||
|     "help_link": "Для получения справки посетите <a href=\"https://triliumnext.github.io/Docs/Wiki/sharing.html\">вики</a>.", | ||||
|     "shared_locally": "Заметка общедоступна локально в", | ||||
|     "shared_publicly": "Заметка общедоступна публично в" | ||||
|   }, | ||||
|   "note_create": { | ||||
|     "duplicated": "Создан дубль заметки \"{{title}}\"." | ||||
|   }, | ||||
|   "help-button": { | ||||
|     "title": "Открыть соответствующую страницу справки" | ||||
|   }, | ||||
|   "render": { | ||||
|     "note_detail_render_help_2": "Тип заметки «Рендер HTML» используется для <a class=\"external\" href=\"https://triliumnext.github.io/Docs/Wiki/scripts.html\">скриптинга</a>. Если коротко, у вас есть заметка с HTML-кодом (возможно, с добавлением JavaScript), и эта заметка её отобразит. Для этого необходимо определить <a class=\"external\" href=\"https://triliumnext.github.io/Docs/Wiki/attributes.html\">отношение</a> с именем «renderNote», указывающее на HTML-заметку для отрисовки.", | ||||
|     "note_detail_render_help_1": "Эта справочная заметка отображается, поскольку эта справка типа Render HTML не имеет необходимой связи для правильной работы." | ||||
|   }, | ||||
|   "file": { | ||||
|     "too_big": "В целях повышения производительности в режиме предварительного просмотра отображаются только первые {{maxNumChars}} символов файла. Загрузите файл и откройте его во внешнем браузере, чтобы увидеть всё содержимое.", | ||||
|     "file_preview_not_available": "Предварительный просмотр файла недоступен для этого файла." | ||||
|   }, | ||||
|   "app_context": { | ||||
|     "please_wait_for_save": "Подождите несколько секунд, пока сохранение завершится, затем попробуйте еще раз." | ||||
|   }, | ||||
|   "settings_appearance": { | ||||
|     "related_code_blocks": "Цветовая схема для блоков кода в текстовых заметках", | ||||
|     "related_code_notes": "Цветовая схема для заметок типа \"Код\"" | ||||
|   }, | ||||
|   "sql_result": { | ||||
|     "no_rows": "По этому запросу не возвращено ни одной строки" | ||||
|   }, | ||||
|   "editable_code": { | ||||
|     "placeholder": "Введите содержимое для заметки с кодом..." | ||||
|   }, | ||||
|   "editable_text": { | ||||
|     "placeholder": "Введите содержимое для заметки..." | ||||
|   }, | ||||
|   "hoisted_note": { | ||||
|     "confirm_unhoisting": "Запрошенная заметка «{{requestedNote}}» находится за пределами поддерева закрепленной заметки \"{{hoistedNote}}\", и для доступа к ней необходимо снять закрепление. Открепить заметку?" | ||||
|   }, | ||||
|   "frontend_script_api": { | ||||
|     "sync_warning": "Вы передаете синхронную функцию в `api.runAsyncOnBackendWithManualTransactionHandling()`, \\nхотя вместо этого вам, скорее всего, следует использовать `api.runOnBackend()`.", | ||||
|     "async_warning": "Вы передаете асинхронную функцию в `api.runOnBackend()`, которая, скорее всего, не будет работать так, как вы предполагали.\\nЛибо сделайте функцию синхронной (удалив ключевое слово `async`), либо используйте `api.runAsyncOnBackendWithManualTransactionHandling()`." | ||||
|   }, | ||||
|   "note_detail": { | ||||
|     "could_not_find_typewidget": "Не удалось найти typeWidget для типа '{{type}}'" | ||||
|   }, | ||||
|   "book": { | ||||
|     "no_children_help": "В этой коллекции нет дочерних заметок, поэтому отображать нечего. Подробности см. на <a href=\"https://triliumnext.github.io/Docs/Wiki/book-note.html\">wiki</a>." | ||||
|   }, | ||||
|   "ui-performance": { | ||||
|     "title": "Производительность", | ||||
|     "enable-motion": "Включить визуальные эффекты и анимации", | ||||
|     "enable-shadows": "Включить тени", | ||||
|     "enable-backdrop-effects": "Включить эффекты размытия фона меню, всплывающих окон и панелей" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1643,13 +1643,13 @@ | ||||
|     "failed_notes": "失敗筆記", | ||||
|     "last_processed": "最後處理時間", | ||||
|     "refresh_stats": "更新統計資料", | ||||
|     "enable_ai_features": "啟用 AI / LLM 功能", | ||||
|     "enable_ai_features": "啟用 AI/LLM 功能", | ||||
|     "enable_ai_description": "啟用筆記摘要、內容生成等 AI 功能及其他 LLM 能力", | ||||
|     "openai_tab": "OpenAI", | ||||
|     "anthropic_tab": "Anthropic", | ||||
|     "voyage_tab": "Voyage AI", | ||||
|     "ollama_tab": "Ollama", | ||||
|     "enable_ai": "啟用 AI / LLM 功能", | ||||
|     "enable_ai": "啟用 AI/LLM 功能", | ||||
|     "enable_ai_desc": "啟用筆記摘要、內容生成等 AI 功能及其他 LLM 能力", | ||||
|     "provider_configuration": "AI 提供者設定", | ||||
|     "provider_precedence": "提供者優先級", | ||||
| @@ -1771,7 +1771,12 @@ | ||||
|     "selected_provider": "已選提供者", | ||||
|     "selected_provider_description": "選擇用於聊天和補全功能的 AI 提供者", | ||||
|     "select_model": "選擇模型…", | ||||
|     "select_provider": "選擇提供者…" | ||||
|     "select_provider": "選擇提供者…", | ||||
|     "ai_enabled": "已啟用 AI 功能", | ||||
|     "ai_disabled": "已禁用 AI 功能", | ||||
|     "no_models_found_online": "找不到模型。請檢查您的 API 金鑰及設定。", | ||||
|     "no_models_found_ollama": "找不到 Ollama 模型。請確認 Ollama 是否正在執行。", | ||||
|     "error_fetching": "獲取模型失敗:{{error}}" | ||||
|   }, | ||||
|   "code-editor-options": { | ||||
|     "title": "編輯器" | ||||
| @@ -1999,5 +2004,21 @@ | ||||
|     "next_theme_message": "您正在使用舊版主題,要試用新主題嗎?", | ||||
|     "next_theme_button": "試用新主題", | ||||
|     "dismiss": "關閉" | ||||
|   }, | ||||
|   "settings": { | ||||
|     "related_settings": "相關設定" | ||||
|   }, | ||||
|   "settings_appearance": { | ||||
|     "related_code_blocks": "文字筆記中程式碼區塊的配色方案", | ||||
|     "related_code_notes": "程式碼筆記的配色方案" | ||||
|   }, | ||||
|   "units": { | ||||
|     "percentage": "%" | ||||
|   }, | ||||
|   "ui-performance": { | ||||
|     "title": "效能", | ||||
|     "enable-motion": "啟用轉場與動畫", | ||||
|     "enable-shadows": "啟用陰影", | ||||
|     "enable-backdrop-effects": "啟用選單、彈出視窗和面板的背景特效" | ||||
|   } | ||||
| } | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,6 +1,8 @@ | ||||
| import utils from "../../services/utils.js"; | ||||
| import type BasicWidget from "../basic_widget.js"; | ||||
| import { EventData } from "../../components/app_context.js"; | ||||
| import FlexContainer from "./flex_container.js"; | ||||
| import options from "../../services/options.js"; | ||||
| import type BasicWidget from "../basic_widget.js"; | ||||
| import utils from "../../services/utils.js"; | ||||
|  | ||||
| /** | ||||
|  * The root container is the top-most widget/container, from which the entire layout derives. | ||||
| @@ -27,15 +29,45 @@ export default class RootContainer extends FlexContainer<BasicWidget> { | ||||
|             window.visualViewport?.addEventListener("resize", () => this.#onMobileResize()); | ||||
|         } | ||||
|  | ||||
|         this.#setMotion(options.is("motionEnabled")); | ||||
|         this.#setShadows(options.is("shadowsEnabled")); | ||||
|         this.#setBackdropEffects(options.is("backdropEffectsEnabled")); | ||||
|  | ||||
|         return super.render(); | ||||
|     } | ||||
|  | ||||
|     entitiesReloadedEvent({ loadResults }: EventData<"entitiesReloaded">) { | ||||
|         if (loadResults.isOptionReloaded("motionEnabled")) { | ||||
|             this.#setMotion(options.is("motionEnabled")); | ||||
|         } | ||||
|  | ||||
|         if (loadResults.isOptionReloaded("shadowsEnabled")) { | ||||
|             this.#setShadows(options.is("shadowsEnabled")); | ||||
|         } | ||||
|  | ||||
|         if (loadResults.isOptionReloaded("backdropEffectsEnabled")) { | ||||
|             this.#setBackdropEffects(options.is("backdropEffectsEnabled")); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     #onMobileResize() { | ||||
|         const currentViewportHeight = getViewportHeight(); | ||||
|         const isKeyboardOpened = (currentViewportHeight < this.originalViewportHeight); | ||||
|         this.$widget.toggleClass("virtual-keyboard-opened", isKeyboardOpened); | ||||
|     } | ||||
|  | ||||
|     #setMotion(enabled: boolean) { | ||||
|         document.body.classList.toggle("motion-disabled", !enabled); | ||||
|         jQuery.fx.off = !enabled; | ||||
|     } | ||||
|  | ||||
|     #setShadows(enabled: boolean) { | ||||
|         document.body.classList.toggle("shadows-disabled", !enabled); | ||||
|     } | ||||
|  | ||||
|     #setBackdropEffects(enabled: boolean) { | ||||
|         document.body.classList.toggle("backdrop-effects-disabled", !enabled); | ||||
|     } | ||||
| } | ||||
|  | ||||
| function getViewportHeight() { | ||||
|   | ||||
| @@ -13,6 +13,7 @@ import noteAutocompleteService, { type Suggestion } from "../../../services/note | ||||
| import mimeTypesService from "../../../services/mime_types.js"; | ||||
| import { normalizeMimeTypeForCKEditor } from "@triliumnext/commons"; | ||||
| import { buildToolbarConfig } from "./toolbar.js"; | ||||
| import ckeditorPluginConfigService from "../../../services/ckeditor_plugin_config.js"; | ||||
|  | ||||
| export const OPEN_SOURCE_LICENSE_KEY = "GPL"; | ||||
|  | ||||
| @@ -164,7 +165,7 @@ export async function buildConfig(opts: BuildEditorOptions): Promise<EditorConfi | ||||
|         }, | ||||
|         // This value must be kept in sync with the language defined in webpack.config.js. | ||||
|         language: "en", | ||||
|         removePlugins: getDisabledPlugins() | ||||
|         removePlugins: await getDisabledPlugins() | ||||
|     }; | ||||
|  | ||||
|     // Set up content language. | ||||
| @@ -203,9 +204,11 @@ export async function buildConfig(opts: BuildEditorOptions): Promise<EditorConfi | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     const toolbarConfig = await buildToolbarConfig(opts.isClassicEditor); | ||||
|      | ||||
|     return { | ||||
|         ...config, | ||||
|         ...buildToolbarConfig(opts.isClassicEditor) | ||||
|         ...toolbarConfig | ||||
|     }; | ||||
| } | ||||
|  | ||||
| @@ -237,9 +240,18 @@ function getLicenseKey() { | ||||
|     return premiumLicenseKey; | ||||
| } | ||||
|  | ||||
| function getDisabledPlugins() { | ||||
| async function getDisabledPlugins() { | ||||
|     let disabledPlugins: string[] = []; | ||||
|  | ||||
|     // Check user's plugin configuration | ||||
|     try { | ||||
|         const userDisabledPlugins = await ckeditorPluginConfigService.getDisabledPlugins(); | ||||
|         disabledPlugins.push(...userDisabledPlugins); | ||||
|     } catch (error) { | ||||
|         console.warn("Failed to load user plugin configuration, using defaults:", error); | ||||
|     } | ||||
|  | ||||
|     // Legacy emoji setting override | ||||
|     if (options.get("textNoteEmojiCompletionEnabled") !== "true") { | ||||
|         disabledPlugins.push("EmojiMention"); | ||||
|     } | ||||
|   | ||||
| @@ -1,33 +1,73 @@ | ||||
| import utils from "../../../services/utils.js"; | ||||
| import options from "../../../services/options.js"; | ||||
| import ckeditorPluginConfigService from "../../../services/ckeditor_plugin_config.js"; | ||||
|  | ||||
| const TEXT_FORMATTING_GROUP = { | ||||
|     label: "Text formatting", | ||||
|     icon: "text" | ||||
| }; | ||||
|  | ||||
| export function buildToolbarConfig(isClassicToolbar: boolean) { | ||||
| export async function buildToolbarConfig(isClassicToolbar: boolean) { | ||||
|     const hiddenItems = await getHiddenToolbarItems(); | ||||
|      | ||||
|     if (utils.isMobile()) { | ||||
|         return buildMobileToolbar(); | ||||
|         return buildMobileToolbar(hiddenItems); | ||||
|     } else if (isClassicToolbar) { | ||||
|         const multilineToolbar = utils.isDesktop() && options.get("textNoteEditorMultilineToolbar") === "true"; | ||||
|         return buildClassicToolbar(multilineToolbar); | ||||
|         return buildClassicToolbar(multilineToolbar, hiddenItems); | ||||
|     } else { | ||||
|         return buildFloatingToolbar(); | ||||
|         return buildFloatingToolbar(hiddenItems); | ||||
|     } | ||||
| } | ||||
|  | ||||
| export function buildMobileToolbar() { | ||||
|     const classicConfig = buildClassicToolbar(false); | ||||
| async function getHiddenToolbarItems(): Promise<Set<string>> { | ||||
|     try { | ||||
|         const hiddenItems = await ckeditorPluginConfigService.getHiddenToolbarItems(); | ||||
|         return new Set(hiddenItems); | ||||
|     } catch (error) { | ||||
|         console.warn("Failed to get hidden toolbar items, using empty set:", error); | ||||
|         return new Set(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Filter toolbar items based on disabled plugins | ||||
|  */ | ||||
| function filterToolbarItems(items: (string | object)[], hiddenItems: Set<string>): (string | object)[] { | ||||
|     return items.filter(item => { | ||||
|         if (typeof item === 'string') { | ||||
|             // Don't hide separators | ||||
|             if (item === '|') return true; | ||||
|             // Check if this item should be hidden | ||||
|             return !hiddenItems.has(item); | ||||
|         } else if (typeof item === 'object' && item !== null && 'items' in item) { | ||||
|             // Filter nested items recursively | ||||
|             const nestedItem = item as { items: (string | object)[] }; | ||||
|             const filteredNested = filterToolbarItems(nestedItem.items, hiddenItems); | ||||
|             // Only keep the group if it has at least one non-separator item | ||||
|             const hasNonSeparatorItems = filteredNested.some(subItem =>  | ||||
|                 typeof subItem === 'string' ? subItem !== '|' : true | ||||
|             ); | ||||
|             if (hasNonSeparatorItems) { | ||||
|                 return { ...item, items: filteredNested }; | ||||
|             } | ||||
|             return null; | ||||
|         } | ||||
|         return true; | ||||
|     }).filter(item => item !== null) as (string | object)[]; | ||||
| } | ||||
|  | ||||
| export function buildMobileToolbar(hiddenItems: Set<string>) { | ||||
|     const classicConfig = buildClassicToolbar(false, hiddenItems); | ||||
|     const items: string[] = []; | ||||
|  | ||||
|     for (const item of classicConfig.toolbar.items) { | ||||
|         if (typeof item === "object" && "items" in item) { | ||||
|             for (const subitem of item.items) { | ||||
|             for (const subitem of (item as any).items) { | ||||
|                 items.push(subitem); | ||||
|             } | ||||
|         } else { | ||||
|             items.push(item); | ||||
|             items.push(item as string); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -40,110 +80,115 @@ export function buildMobileToolbar() { | ||||
|     }; | ||||
| } | ||||
|  | ||||
| export function buildClassicToolbar(multilineToolbar: boolean) { | ||||
| export function buildClassicToolbar(multilineToolbar: boolean, hiddenItems: Set<string>) { | ||||
|     // For nested toolbars, refer to https://ckeditor.com/docs/ckeditor5/latest/getting-started/setup/toolbar.html#grouping-toolbar-items-in-dropdowns-nested-toolbars. | ||||
|     const items = [ | ||||
|         "heading", | ||||
|         "fontSize", | ||||
|         "|", | ||||
|         "bold", | ||||
|         "italic", | ||||
|         { | ||||
|             ...TEXT_FORMATTING_GROUP, | ||||
|             items: ["underline", "strikethrough", "|", "superscript", "subscript", "|", "kbd"] | ||||
|         }, | ||||
|         "|", | ||||
|         "fontColor", | ||||
|         "fontBackgroundColor", | ||||
|         "removeFormat", | ||||
|         "|", | ||||
|         "bulletedList", | ||||
|         "numberedList", | ||||
|         "todoList", | ||||
|         "|", | ||||
|         "blockQuote", | ||||
|         "admonition", | ||||
|         "insertTable", | ||||
|         "|", | ||||
|         "code", | ||||
|         "codeBlock", | ||||
|         "|", | ||||
|         "footnote", | ||||
|         { | ||||
|             label: "Insert", | ||||
|             icon: "plus", | ||||
|             items: ["imageUpload", "|", "link", "bookmark", "internallink", "includeNote", "|", "specialCharacters", "emoji", "math", "mermaid", "horizontalLine", "pageBreak", "dateTime"] | ||||
|         }, | ||||
|         "|", | ||||
|         "alignment", | ||||
|         "outdent", | ||||
|         "indent", | ||||
|         "|", | ||||
|         "insertTemplate", | ||||
|         "markdownImport", | ||||
|         "cuttonote", | ||||
|         "findAndReplace" | ||||
|     ]; | ||||
|  | ||||
|     return { | ||||
|         toolbar: { | ||||
|             items: [ | ||||
|                 "heading", | ||||
|                 "fontSize", | ||||
|                 "|", | ||||
|                 "bold", | ||||
|                 "italic", | ||||
|                 { | ||||
|                     ...TEXT_FORMATTING_GROUP, | ||||
|                     items: ["underline", "strikethrough", "|", "superscript", "subscript", "|", "kbd"] | ||||
|                 }, | ||||
|                 "|", | ||||
|                 "fontColor", | ||||
|                 "fontBackgroundColor", | ||||
|                 "removeFormat", | ||||
|                 "|", | ||||
|                 "bulletedList", | ||||
|                 "numberedList", | ||||
|                 "todoList", | ||||
|                 "|", | ||||
|                 "blockQuote", | ||||
|                 "admonition", | ||||
|                 "insertTable", | ||||
|                 "|", | ||||
|                 "code", | ||||
|                 "codeBlock", | ||||
|                 "|", | ||||
|                 "footnote", | ||||
|                 { | ||||
|                     label: "Insert", | ||||
|                     icon: "plus", | ||||
|                     items: ["imageUpload", "|", "link", "bookmark", "internallink", "includeNote", "|", "specialCharacters", "emoji", "math", "mermaid", "horizontalLine", "pageBreak", "dateTime"] | ||||
|                 }, | ||||
|                 "|", | ||||
|                 "alignment", | ||||
|                 "outdent", | ||||
|                 "indent", | ||||
|                 "|", | ||||
|                 "insertTemplate", | ||||
|                 "markdownImport", | ||||
|                 "cuttonote", | ||||
|                 "findAndReplace" | ||||
|             ], | ||||
|             items: filterToolbarItems(items, hiddenItems), | ||||
|             shouldNotGroupWhenFull: multilineToolbar | ||||
|         } | ||||
|     }; | ||||
| } | ||||
|  | ||||
| export function buildFloatingToolbar() { | ||||
| export function buildFloatingToolbar(hiddenItems: Set<string>) { | ||||
|     const toolbarItems = [ | ||||
|         "fontSize", | ||||
|         "bold", | ||||
|         "italic", | ||||
|         "underline", | ||||
|         { | ||||
|             ...TEXT_FORMATTING_GROUP, | ||||
|             items: [ "strikethrough", "|", "superscript", "subscript", "|", "kbd" ] | ||||
|         }, | ||||
|         "|", | ||||
|         "fontColor", | ||||
|         "fontBackgroundColor", | ||||
|         "|", | ||||
|         "code", | ||||
|         "link", | ||||
|         "bookmark", | ||||
|         "removeFormat", | ||||
|         "internallink", | ||||
|         "cuttonote" | ||||
|     ]; | ||||
|  | ||||
|     const blockToolbarItems = [ | ||||
|         "heading", | ||||
|         "|", | ||||
|         "bulletedList", | ||||
|         "numberedList", | ||||
|         "todoList", | ||||
|         "|", | ||||
|         "blockQuote", | ||||
|         "admonition", | ||||
|         "codeBlock", | ||||
|         "insertTable", | ||||
|         "footnote", | ||||
|         { | ||||
|             label: "Insert", | ||||
|             icon: "plus", | ||||
|             items: ["link", "bookmark", "internallink", "includeNote", "|", "math", "mermaid", "horizontalLine", "pageBreak", "dateTime"] | ||||
|         }, | ||||
|         "|", | ||||
|         "alignment", | ||||
|         "outdent", | ||||
|         "indent", | ||||
|         "|", | ||||
|         "insertTemplate", | ||||
|         "imageUpload", | ||||
|         "markdownImport", | ||||
|         "specialCharacters", | ||||
|         "emoji", | ||||
|         "findAndReplace" | ||||
|     ]; | ||||
|  | ||||
|     return { | ||||
|         toolbar: { | ||||
|             items: [ | ||||
|                 "fontSize", | ||||
|                 "bold", | ||||
|                 "italic", | ||||
|                 "underline", | ||||
|                 { | ||||
|                     ...TEXT_FORMATTING_GROUP, | ||||
|                     items: [ "strikethrough", "|", "superscript", "subscript", "|", "kbd" ] | ||||
|                 }, | ||||
|                 "|", | ||||
|                 "fontColor", | ||||
|                 "fontBackgroundColor", | ||||
|                 "|", | ||||
|                 "code", | ||||
|                 "link", | ||||
|                 "bookmark", | ||||
|                 "removeFormat", | ||||
|                 "internallink", | ||||
|                 "cuttonote" | ||||
|             ] | ||||
|             items: filterToolbarItems(toolbarItems, hiddenItems) | ||||
|         }, | ||||
|  | ||||
|         blockToolbar: [ | ||||
|             "heading", | ||||
|             "|", | ||||
|             "bulletedList", | ||||
|             "numberedList", | ||||
|             "todoList", | ||||
|             "|", | ||||
|             "blockQuote", | ||||
|             "admonition", | ||||
|             "codeBlock", | ||||
|             "insertTable", | ||||
|             "footnote", | ||||
|             { | ||||
|                 label: "Insert", | ||||
|                 icon: "plus", | ||||
|                 items: ["link", "bookmark", "internallink", "includeNote", "|", "math", "mermaid", "horizontalLine", "pageBreak", "dateTime"] | ||||
|             }, | ||||
|             "|", | ||||
|             "alignment", | ||||
|             "outdent", | ||||
|             "indent", | ||||
|             "|", | ||||
|             "insertTemplate", | ||||
|             "imageUpload", | ||||
|             "markdownImport", | ||||
|             "specialCharacters", | ||||
|             "emoji", | ||||
|             "findAndReplace" | ||||
|         ] | ||||
|         blockToolbar: filterToolbarItems(blockToolbarItems, hiddenItems) | ||||
|     }; | ||||
| } | ||||
|   | ||||
| @@ -17,6 +17,7 @@ import PasswordSettings from "./options/password.jsx"; | ||||
| import ShortcutSettings from "./options/shortcuts.js"; | ||||
| import TextNoteSettings from "./options/text_notes.jsx"; | ||||
| import CodeNoteSettings from "./options/code_notes.jsx"; | ||||
| import CKEditorPluginSettings from "./options/ckeditor_plugins.jsx"; | ||||
| import OtherSettings from "./options/other.jsx"; | ||||
| import BackendLogWidget from "./content/backend_log.js"; | ||||
| import MultiFactorAuthenticationSettings from "./options/multi_factor_authentication.js"; | ||||
| @@ -45,13 +46,14 @@ const TPL = /*html*/`<div class="note-detail-content-widget note-detail-printabl | ||||
|     <div class="note-detail-content-widget-content"></div> | ||||
| </div>`; | ||||
|  | ||||
| export type OptionPages = "_optionsAppearance" | "_optionsShortcuts" | "_optionsTextNotes" | "_optionsCodeNotes" | "_optionsImages" | "_optionsSpellcheck" | "_optionsPassword" | "_optionsMFA" | "_optionsEtapi" | "_optionsBackup" | "_optionsSync" | "_optionsAi" | "_optionsOther" | "_optionsLocalization" | "_optionsAdvanced"; | ||||
| export type OptionPages = "_optionsAppearance" | "_optionsShortcuts" | "_optionsTextNotes" | "_optionsCodeNotes" | "_optionsCKEditorPlugins" | "_optionsImages" | "_optionsSpellcheck" | "_optionsPassword" | "_optionsMFA" | "_optionsEtapi" | "_optionsBackup" | "_optionsSync" | "_optionsAi" | "_optionsOther" | "_optionsLocalization" | "_optionsAdvanced"; | ||||
|  | ||||
| const CONTENT_WIDGETS: Record<OptionPages | "_backendLog", ((typeof NoteContextAwareWidget)[] | JSX.Element)> = { | ||||
|     _optionsAppearance: <AppearanceSettings />, | ||||
|     _optionsShortcuts: <ShortcutSettings />, | ||||
|     _optionsTextNotes: <TextNoteSettings />, | ||||
|     _optionsCodeNotes: <CodeNoteSettings />, | ||||
|     _optionsCKEditorPlugins: <CKEditorPluginSettings />, | ||||
|     _optionsImages: <ImageSettings />, | ||||
|     _optionsSpellcheck: <SpellcheckSettings />, | ||||
|     _optionsPassword: <PasswordSettings />, | ||||
|   | ||||
| @@ -88,6 +88,7 @@ export default function AppearanceSettings() { | ||||
|             <ApplicationTheme /> | ||||
|             {overrideThemeFonts === "true" && <Fonts />} | ||||
|             {isElectron() && <ElectronIntegration /> } | ||||
|             <Performance /> | ||||
|             <MaxContentWidth /> | ||||
|             <RelatedSettings items={[ | ||||
|                 { | ||||
| @@ -245,6 +246,30 @@ function ElectronIntegration() { | ||||
|     ) | ||||
| } | ||||
|  | ||||
| function Performance() { | ||||
|     const [ motionEnabled, setMotionEnabled ] = useTriliumOptionBool("motionEnabled"); | ||||
|     const [ shadowsEnabled, setShadowsEnabled ] = useTriliumOptionBool("shadowsEnabled"); | ||||
|     const [ backdropEffectsEnabled, setBackdropEffectsEnabled ] = useTriliumOptionBool("backdropEffectsEnabled"); | ||||
|  | ||||
|     return <OptionsSection title={t("ui-performance.title")}> | ||||
|         <FormCheckbox | ||||
|             label={t("ui-performance.enable-motion")} | ||||
|             currentValue={motionEnabled} onChange={setMotionEnabled} | ||||
|         /> | ||||
|  | ||||
|         <FormCheckbox | ||||
|             label={t("ui-performance.enable-shadows")} | ||||
|             currentValue={shadowsEnabled} onChange={setShadowsEnabled} | ||||
|         /> | ||||
|  | ||||
|         <FormCheckbox | ||||
|             label={t("ui-performance.enable-backdrop-effects")} | ||||
|             currentValue={backdropEffectsEnabled} onChange={setBackdropEffectsEnabled} | ||||
|         /> | ||||
|     </OptionsSection> | ||||
| } | ||||
|  | ||||
|  | ||||
| function MaxContentWidth() { | ||||
|     const [ maxContentWidth, setMaxContentWidth ] = useTriliumOption("maxContentWidth"); | ||||
|  | ||||
|   | ||||
| @@ -0,0 +1,397 @@ | ||||
| import { useEffect, useState, useCallback, useMemo } from "preact/hooks"; | ||||
| import { t } from "../../../services/i18n"; | ||||
| import server from "../../../services/server"; | ||||
| import FormCheckbox from "../../react/FormCheckbox"; | ||||
| import FormGroup from "../../react/FormGroup"; | ||||
| import FormText from "../../react/FormText"; | ||||
| import OptionsSection from "./components/OptionsSection"; | ||||
| import Button from "../../react/Button"; | ||||
| import toast from "../../../services/toast"; | ||||
| import type {  | ||||
|     PluginMetadata,  | ||||
|     PluginConfiguration,  | ||||
|     PluginRegistry, | ||||
|     PluginValidationResult, | ||||
|     UpdatePluginConfigRequest, | ||||
|     UpdatePluginConfigResponse, | ||||
|     QueryPluginsResult, | ||||
|     PluginCategory | ||||
| } from "@triliumnext/commons"; | ||||
|  | ||||
| interface PluginStats { | ||||
|     enabled: number; | ||||
|     total: number; | ||||
|     core: number; | ||||
|     premium: number; | ||||
|     configurable: number; | ||||
|     categories: Record<string, number>; | ||||
|     hasPremiumLicense: boolean; | ||||
| } | ||||
|  | ||||
| const CATEGORY_DISPLAY_NAMES: Record<PluginCategory, string> = { | ||||
|     formatting: t("ckeditor_plugins.category_formatting"), | ||||
|     structure: t("ckeditor_plugins.category_structure"), | ||||
|     media: t("ckeditor_plugins.category_media"), | ||||
|     tables: t("ckeditor_plugins.category_tables"), | ||||
|     advanced: t("ckeditor_plugins.category_advanced"), | ||||
|     trilium: t("ckeditor_plugins.category_trilium"), | ||||
|     external: t("ckeditor_plugins.category_external") | ||||
| }; | ||||
|  | ||||
| export default function CKEditorPluginSettings() { | ||||
|     const [pluginRegistry, setPluginRegistry] = useState<PluginRegistry | null>(null); | ||||
|     const [userConfig, setUserConfig] = useState<PluginConfiguration[]>([]); | ||||
|     const [stats, setStats] = useState<PluginStats | null>(null); | ||||
|     const [loading, setLoading] = useState(true); | ||||
|     const [saving, setSaving] = useState(false); | ||||
|     const [validationResult, setValidationResult] = useState<PluginValidationResult | null>(null); | ||||
|     const [showValidation, setShowValidation] = useState(false); | ||||
|  | ||||
|     // Load initial data | ||||
|     useEffect(() => { | ||||
|         loadData(); | ||||
|     }, []); | ||||
|  | ||||
|     const loadData = useCallback(async () => { | ||||
|         setLoading(true); | ||||
|         try { | ||||
|             const [registry, config, statsData] = await Promise.all([ | ||||
|                 server.get<PluginRegistry>('ckeditor-plugins/registry'), | ||||
|                 server.get<PluginConfiguration[]>('ckeditor-plugins/config'), | ||||
|                 server.get<PluginStats>('ckeditor-plugins/stats') | ||||
|             ]); | ||||
|              | ||||
|             setPluginRegistry(registry); | ||||
|             setUserConfig(config); | ||||
|             setStats(statsData); | ||||
|         } catch (error) { | ||||
|             toast.showError(`${t("ckeditor_plugins.load_error")}: ${error}`); | ||||
|         } finally { | ||||
|             setLoading(false); | ||||
|         } | ||||
|     }, []); | ||||
|  | ||||
|     // Organize plugins by category | ||||
|     const pluginsByCategory = useMemo(() => { | ||||
|         if (!pluginRegistry) return {}; | ||||
|          | ||||
|         const categories: Record<PluginCategory, PluginMetadata[]> = { | ||||
|             formatting: [], | ||||
|             structure: [], | ||||
|             media: [], | ||||
|             tables: [], | ||||
|             advanced: [], | ||||
|             trilium: [], | ||||
|             external: [] | ||||
|         }; | ||||
|  | ||||
|         Object.values(pluginRegistry.plugins).forEach(plugin => { | ||||
|             if (!plugin.isCore) { // Don't show core plugins in settings | ||||
|                 categories[plugin.category].push(plugin); | ||||
|             } | ||||
|         }); | ||||
|  | ||||
|         // Sort plugins within each category by name | ||||
|         Object.keys(categories).forEach(category => { | ||||
|             categories[category as PluginCategory].sort((a, b) => a.name.localeCompare(b.name)); | ||||
|         }); | ||||
|  | ||||
|         return categories; | ||||
|     }, [pluginRegistry]); | ||||
|  | ||||
|     // Get enabled status for a plugin | ||||
|     const isPluginEnabled = useCallback((pluginId: string): boolean => { | ||||
|         return userConfig.find(config => config.id === pluginId)?.enabled ?? false; | ||||
|     }, [userConfig]); | ||||
|  | ||||
|     // Toggle plugin enabled state | ||||
|     const togglePlugin = useCallback((pluginId: string) => { | ||||
|         setUserConfig(prev => prev.map(config =>  | ||||
|             config.id === pluginId  | ||||
|                 ? { ...config, enabled: !config.enabled } | ||||
|                 : config | ||||
|         )); | ||||
|     }, []); | ||||
|  | ||||
|     // Validate current configuration | ||||
|     const validateConfig = useCallback(async () => { | ||||
|         if (!userConfig.length) return; | ||||
|          | ||||
|         try { | ||||
|             const result = await server.post<PluginValidationResult>('ckeditor-plugins/validate', { | ||||
|                 plugins: userConfig | ||||
|             }); | ||||
|             setValidationResult(result); | ||||
|             setShowValidation(true); | ||||
|             return result; | ||||
|         } catch (error) { | ||||
|             toast.showError(`${t("ckeditor_plugins.validation_error")}: ${error}`); | ||||
|             return null; | ||||
|         } | ||||
|     }, [userConfig]); | ||||
|  | ||||
|     // Save configuration | ||||
|     const saveConfiguration = useCallback(async () => { | ||||
|         setSaving(true); | ||||
|         setShowValidation(false); | ||||
|          | ||||
|         try { | ||||
|             const request: UpdatePluginConfigRequest = { | ||||
|                 plugins: userConfig, | ||||
|                 validate: true | ||||
|             }; | ||||
|  | ||||
|             const response = await server.put<UpdatePluginConfigResponse>('ckeditor-plugins/config', request); | ||||
|              | ||||
|             if (response.success) { | ||||
|                 toast.showMessage(t("ckeditor_plugins.save_success")); | ||||
|                 await loadData(); // Reload stats | ||||
|                  | ||||
|                 // Notify user that editor reload might be needed | ||||
|                 toast.showMessage(t("ckeditor_plugins.reload_editor_notice"), { | ||||
|                     timeout: 5000 | ||||
|                 }); | ||||
|             } else { | ||||
|                 setValidationResult(response.validation); | ||||
|                 setShowValidation(true); | ||||
|                 toast.showError(`${t("ckeditor_plugins.save_error")}: ${response.errors?.join(", ")}`); | ||||
|             } | ||||
|         } catch (error) { | ||||
|             toast.showError(`${t("ckeditor_plugins.save_error")}: ${error}`); | ||||
|         } finally { | ||||
|             setSaving(false); | ||||
|         } | ||||
|     }, [userConfig, loadData]); | ||||
|  | ||||
|     // Reset to defaults | ||||
|     const resetToDefaults = useCallback(async () => { | ||||
|         if (!confirm(t("ckeditor_plugins.reset_confirm"))) return; | ||||
|          | ||||
|         setSaving(true); | ||||
|         try { | ||||
|             const response = await server.post<UpdatePluginConfigResponse>('ckeditor-plugins/reset'); | ||||
|             if (response.success) { | ||||
|                 setUserConfig(response.plugins); | ||||
|                 toast.showMessage(t("ckeditor_plugins.reset_success")); | ||||
|                 await loadData(); | ||||
|             } else { | ||||
|                 toast.showError(`${t("ckeditor_plugins.reset_error")}: ${response.errors?.join(", ")}`); | ||||
|             } | ||||
|         } catch (error) { | ||||
|             toast.showError(`${t("ckeditor_plugins.reset_error")}: ${error}`); | ||||
|         } finally { | ||||
|             setSaving(false); | ||||
|         } | ||||
|     }, [loadData]); | ||||
|  | ||||
|     if (loading) { | ||||
|         return ( | ||||
|             <OptionsSection title={t("ckeditor_plugins.title")}> | ||||
|                 <FormText>{t("ckeditor_plugins.loading")}</FormText> | ||||
|             </OptionsSection> | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     if (!pluginRegistry || !stats) { | ||||
|         return ( | ||||
|             <OptionsSection title={t("ckeditor_plugins.title")}> | ||||
|                 <FormText>{t("ckeditor_plugins.load_failed")}</FormText> | ||||
|                 <Button text={t("ckeditor_plugins.retry")} onClick={loadData} /> | ||||
|             </OptionsSection> | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     return ( | ||||
|         <div> | ||||
|             <OptionsSection title={t("ckeditor_plugins.title")}> | ||||
|                 <FormText>{t("ckeditor_plugins.description")}</FormText> | ||||
|                  | ||||
|                 {/* Stats overview */} | ||||
|                 <div className="plugin-stats" style={{  | ||||
|                     backgroundColor: 'var(--accented-background-color)',  | ||||
|                     padding: '12px',  | ||||
|                     borderRadius: '4px', | ||||
|                     marginBottom: '20px' | ||||
|                 }}> | ||||
|                     <div className="row"> | ||||
|                         <div className="col-md-3"> | ||||
|                             <strong>{t("ckeditor_plugins.stats_enabled")}</strong><br /> | ||||
|                             <span style={{ fontSize: '1.2em', color: 'var(--main-text-color)' }}> | ||||
|                                 {stats.enabled}/{stats.configurable} | ||||
|                             </span> | ||||
|                         </div> | ||||
|                         <div className="col-md-3"> | ||||
|                             <strong>{t("ckeditor_plugins.stats_total")}</strong><br /> | ||||
|                             <span style={{ fontSize: '1.2em', color: 'var(--main-text-color)' }}> | ||||
|                                 {stats.total} | ||||
|                             </span> | ||||
|                         </div> | ||||
|                         <div className="col-md-3"> | ||||
|                             <strong>{t("ckeditor_plugins.stats_core")}</strong><br /> | ||||
|                             <span style={{ fontSize: '1.2em', color: 'var(--main-text-color)' }}> | ||||
|                                 {stats.core} | ||||
|                             </span> | ||||
|                         </div> | ||||
|                         <div className="col-md-3"> | ||||
|                             <strong>{t("ckeditor_plugins.stats_premium")}</strong><br /> | ||||
|                             <span style={{ fontSize: '1.2em', color: stats.hasPremiumLicense ? 'var(--success-color)' : 'var(--muted-text-color)' }}> | ||||
|                                 {stats.premium} {!stats.hasPremiumLicense && `(${t("ckeditor_plugins.no_license")})`} | ||||
|                             </span> | ||||
|                         </div> | ||||
|                     </div> | ||||
|                 </div> | ||||
|  | ||||
|                 {/* Validation results */} | ||||
|                 {showValidation && validationResult && ( | ||||
|                     <div className="validation-results" style={{ marginBottom: '20px' }}> | ||||
|                         {!validationResult.valid && ( | ||||
|                             <div className="alert alert-danger"> | ||||
|                                 <strong>{t("ckeditor_plugins.validation_errors")}</strong> | ||||
|                                 <ul> | ||||
|                                     {validationResult.errors.map((error, index) => ( | ||||
|                                         <li key={index}>{error.message}</li> | ||||
|                                     ))} | ||||
|                                 </ul> | ||||
|                             </div> | ||||
|                         )} | ||||
|                         {validationResult.warnings.length > 0 && ( | ||||
|                             <div className="alert alert-warning"> | ||||
|                                 <strong>{t("ckeditor_plugins.validation_warnings")}</strong> | ||||
|                                 <ul> | ||||
|                                     {validationResult.warnings.map((warning, index) => ( | ||||
|                                         <li key={index}>{warning.message}</li> | ||||
|                                     ))} | ||||
|                                 </ul> | ||||
|                             </div> | ||||
|                         )} | ||||
|                     </div> | ||||
|                 )} | ||||
|  | ||||
|                 {/* Action buttons */} | ||||
|                 <div className="plugin-actions" style={{ marginBottom: '20px' }}> | ||||
|                     <Button  | ||||
|                         text={t("ckeditor_plugins.validate")}  | ||||
|                         onClick={validateConfig} | ||||
|                         disabled={saving} | ||||
|                         className="btn-secondary" | ||||
|                         size="small" | ||||
|                     /> | ||||
|                     <Button  | ||||
|                         text={t("ckeditor_plugins.save")}  | ||||
|                         onClick={saveConfiguration} | ||||
|                         disabled={saving} | ||||
|                         className="btn-primary" | ||||
|                         size="small" | ||||
|                         style={{ marginLeft: '10px' }} | ||||
|                     /> | ||||
|                     <Button  | ||||
|                         text={t("ckeditor_plugins.reset_defaults")}  | ||||
|                         onClick={resetToDefaults} | ||||
|                         disabled={saving} | ||||
|                         className="btn-secondary" | ||||
|                         size="small" | ||||
|                         style={{ marginLeft: '10px' }} | ||||
|                     /> | ||||
|                 </div> | ||||
|             </OptionsSection> | ||||
|  | ||||
|             {/* Plugin categories */} | ||||
|             {Object.entries(pluginsByCategory).map(([categoryKey, plugins]) => { | ||||
|                 if (plugins.length === 0) return null; | ||||
|                  | ||||
|                 const category = categoryKey as PluginCategory; | ||||
|                 return ( | ||||
|                     <OptionsSection key={category} title={CATEGORY_DISPLAY_NAMES[category]} level={2}> | ||||
|                         <div className="plugin-category"> | ||||
|                             {plugins.map(plugin => ( | ||||
|                                 <PluginConfigItem  | ||||
|                                     key={plugin.id} | ||||
|                                     plugin={plugin} | ||||
|                                     enabled={isPluginEnabled(plugin.id)} | ||||
|                                     onToggle={() => togglePlugin(plugin.id)} | ||||
|                                     hasPremiumLicense={stats.hasPremiumLicense} | ||||
|                                 /> | ||||
|                             ))} | ||||
|                         </div> | ||||
|                     </OptionsSection> | ||||
|                 ); | ||||
|             })} | ||||
|         </div> | ||||
|     ); | ||||
| } | ||||
|  | ||||
| interface PluginConfigItemProps { | ||||
|     plugin: PluginMetadata; | ||||
|     enabled: boolean; | ||||
|     onToggle: () => void; | ||||
|     hasPremiumLicense: boolean; | ||||
| } | ||||
|  | ||||
| function PluginConfigItem({ plugin, enabled, onToggle, hasPremiumLicense }: PluginConfigItemProps) { | ||||
|     const canEnable = !plugin.requiresPremium || hasPremiumLicense; | ||||
|      | ||||
|     return ( | ||||
|         <FormGroup name={`plugin-${plugin.id}`} style={{ marginBottom: '15px' }}> | ||||
|             <div className="plugin-item" style={{ | ||||
|                 display: 'flex', | ||||
|                 alignItems: 'flex-start', | ||||
|                 opacity: canEnable ? 1 : 0.6 | ||||
|             }}> | ||||
|                 <FormCheckbox | ||||
|                     label="" | ||||
|                     currentValue={enabled && canEnable} | ||||
|                     onChange={canEnable ? onToggle : undefined} | ||||
|                     disabled={!canEnable} | ||||
|                     containerStyle={{ marginRight: '10px', marginTop: '2px' }} | ||||
|                 /> | ||||
|                 <div style={{ flex: 1 }}> | ||||
|                     <div style={{  | ||||
|                         fontWeight: 'bold', | ||||
|                         marginBottom: '4px', | ||||
|                         display: 'flex', | ||||
|                         alignItems: 'center', | ||||
|                         gap: '8px' | ||||
|                     }}> | ||||
|                         <span>{plugin.name}</span> | ||||
|                         {plugin.requiresPremium && ( | ||||
|                             <span className="badge badge-warning" style={{ fontSize: '0.75em' }}> | ||||
|                                 {t("ckeditor_plugins.premium")} | ||||
|                             </span> | ||||
|                         )} | ||||
|                         {plugin.dependencies.length > 0 && ( | ||||
|                             <span className="badge badge-info" style={{ fontSize: '0.75em' }}> | ||||
|                                 {t("ckeditor_plugins.has_dependencies")} | ||||
|                             </span> | ||||
|                         )} | ||||
|                     </div> | ||||
|                     <div style={{  | ||||
|                         fontSize: '0.9em',  | ||||
|                         color: 'var(--muted-text-color)', | ||||
|                         marginBottom: '4px' | ||||
|                     }}> | ||||
|                         {plugin.description} | ||||
|                     </div> | ||||
|                     {plugin.dependencies.length > 0 && ( | ||||
|                         <div style={{ fontSize: '0.8em', color: 'var(--muted-text-color)' }}> | ||||
|                             {t("ckeditor_plugins.depends_on")}: {plugin.dependencies.join(', ')} | ||||
|                         </div> | ||||
|                     )} | ||||
|                     {plugin.toolbarItems && plugin.toolbarItems.length > 0 && ( | ||||
|                         <div style={{ fontSize: '0.8em', color: 'var(--muted-text-color)' }}> | ||||
|                             {t("ckeditor_plugins.toolbar_items")}: {plugin.toolbarItems.join(', ')} | ||||
|                         </div> | ||||
|                     )} | ||||
|                     {!canEnable && ( | ||||
|                         <div style={{  | ||||
|                             fontSize: '0.8em',  | ||||
|                             color: 'var(--error-color)', | ||||
|                             fontStyle: 'italic' | ||||
|                         }}> | ||||
|                             {t("ckeditor_plugins.premium_required")} | ||||
|                         </div> | ||||
|                     )} | ||||
|                 </div> | ||||
|             </div> | ||||
|         </FormGroup> | ||||
|     ); | ||||
| } | ||||
| @@ -676,6 +676,10 @@ export async function getFullCalendarLocale(locale: string) { | ||||
|             return (await import("@fullcalendar/core/locales/ru")).default; | ||||
|         case "ja": | ||||
|             return (await import("@fullcalendar/core/locales/ja")).default; | ||||
|         case "pt_br": | ||||
|             return (await import("@fullcalendar/core/locales/pt-br")).default; | ||||
|         case "uk": | ||||
|             return (await import("@fullcalendar/core/locales/uk")).default; | ||||
|         case "en": | ||||
|         default: | ||||
|             return undefined; | ||||
|   | ||||
| @@ -74,7 +74,7 @@ | ||||
|     "html2plaintext": "2.1.4", | ||||
|     "http-proxy-agent": "7.0.2", | ||||
|     "https-proxy-agent": "7.0.6", | ||||
|     "i18next": "25.3.6", | ||||
|     "i18next": "25.4.1", | ||||
|     "i18next-fs-backend": "2.6.0", | ||||
|     "image-type": "6.0.0", | ||||
|     "ini": "5.0.0", | ||||
|   | ||||
| @@ -1,305 +1,397 @@ | ||||
| { | ||||
|     "keyboard_actions": { | ||||
|         "open-jump-to-note-dialog": "Öffne das Dialogfeld \"Zu Notiz springen\"", | ||||
|         "search-in-subtree": "Nach Notizen im Unterbaum der aktuellen Notiz suchen", | ||||
|         "expand-subtree": "Unterbaum der aktuellen Notiz ausklappen", | ||||
|         "collapse-tree": "Gesamten Notizbaum einklappen", | ||||
|         "collapse-subtree": "Unterbaum der aktuellen Notiz einklappen", | ||||
|         "sort-child-notes": "Untergeordnete Notizen sortieren", | ||||
|         "creating-and-moving-notes": "Notizen erstellen und verschieben", | ||||
|         "create-note-into-inbox": "Erstelle eine Notiz im Posteingang (falls definiert) oder in der Tagesnotiz", | ||||
|         "delete-note": "Notiz löschen", | ||||
|         "move-note-up": "Notiz nach oben verschieben", | ||||
|         "move-note-down": "Notiz nach unten verschieben", | ||||
|         "move-note-up-in-hierarchy": "Notiz in der Hierarchie nach oben verschieben", | ||||
|         "move-note-down-in-hierarchy": "Notiz in der Hierarchie nach unten verschieben", | ||||
|         "edit-note-title": "Vom Notiz-Baum zur Notiz-Detailansicht springen und den Titel bearbeiten", | ||||
|         "edit-branch-prefix": "Dialog zum Bearbeiten des Zweigpräfixes anzeigen", | ||||
|         "note-clipboard": "Notiz-Zwischenablage", | ||||
|         "copy-notes-to-clipboard": "Ausgewählte Notizen in die Zwischenablage kopieren", | ||||
|         "paste-notes-from-clipboard": "Notizen aus der Zwischenablage in die aktive Notiz einfügen", | ||||
|         "cut-notes-to-clipboard": "Ausgewählte Notizen in die Zwischenablage ausschneiden", | ||||
|         "select-all-notes-in-parent": "Alle Notizen der aktuellen Notizenebene auswählen", | ||||
|         "add-note-above-to-the-selection": "Notiz oberhalb der Auswahl hinzufügen", | ||||
|         "add-note-below-to-selection": "Notiz unterhalb der Auswahl hinzufügen", | ||||
|         "duplicate-subtree": "Unterbaum duplizieren", | ||||
|         "tabs-and-windows": "Tabs & Fenster", | ||||
|         "open-new-tab": "Neuen Tab öffnen", | ||||
|         "close-active-tab": "Aktiven Tab schließen", | ||||
|         "reopen-last-tab": "Zuletzt geschlossenen Tab wieder öffnen", | ||||
|         "activate-next-tab": "Rechten Tab aktivieren", | ||||
|         "activate-previous-tab": "Linken Tab aktivieren", | ||||
|         "open-new-window": "Neues leeres Fenster öffnen", | ||||
|         "toggle-tray": "Anwendung im Systemtray anzeigen/verstecken", | ||||
|         "first-tab": "Ersten Tab in der Liste aktivieren", | ||||
|         "second-tab": "Zweiten Tab in der Liste aktivieren", | ||||
|         "third-tab": "Dritten Tab in der Liste aktivieren", | ||||
|         "fourth-tab": "Vierten Tab in der Liste aktivieren", | ||||
|         "fifth-tab": "Fünften Tab in der Liste aktivieren", | ||||
|         "sixth-tab": "Sechsten Tab in der Liste aktivieren", | ||||
|         "seventh-tab": "Siebten Tab in der Liste aktivieren", | ||||
|         "eight-tab": "Achten Tab in der Liste aktivieren", | ||||
|         "ninth-tab": "Neunten Tab in der Liste aktivieren", | ||||
|         "last-tab": "Letzten Tab in der Liste aktivieren", | ||||
|         "dialogs": "Dialoge", | ||||
|         "show-note-source": "Notizquellen-Dialog anzeigen", | ||||
|         "show-options": "Optionen-Dialog anzeigen", | ||||
|         "show-revisions": "Notizrevisionen-Dialog anzeigen", | ||||
|         "show-recent-changes": "Letzte Änderungen-Dialog anzeigen", | ||||
|         "show-sql-console": "SQL-Konsole-Dialog anzeigen", | ||||
|         "show-backend-log": "Backend-Logs-Dialog anzeigen", | ||||
|         "text-note-operations": "Textnotiz-Operationen", | ||||
|         "add-link-to-text": "Dialogfeld zum Hinzufügen eines Links zum Text öffnen", | ||||
|         "follow-link-under-cursor": "Folge dem Link, unter dem Mauszeiger", | ||||
|         "insert-date-and-time-to-text": "Aktuelles Datum & Uhrzeit in den Text einfügen", | ||||
|         "paste-markdown-into-text": "Markdown aus der Zwischenablage in die Textnotiz einfügen", | ||||
|         "cut-into-note": "Auswahl aus der aktuellen Notiz ausschneiden und eine Unternotiz mit dem ausgewählten Text erstellen", | ||||
|         "add-include-note-to-text": "Notiz-Einfügen-Dialog öffnen", | ||||
|         "edit-readonly-note": "Schreibgeschützte Notiz bearbeiten", | ||||
|         "attributes-labels-and-relations": "Attribute (Labels & Verknüpfungen)", | ||||
|         "add-new-label": "Neues Label erstellen", | ||||
|         "create-new-relation": "Neue Verknüpfungen", | ||||
|         "ribbon-tabs": "Ribbon-Tabs", | ||||
|         "toggle-basic-properties": "Grundattribute umschalten", | ||||
|         "toggle-file-properties": "Dateiattribute umschalten", | ||||
|         "toggle-image-properties": "Bildattribute umschalten", | ||||
|         "toggle-owned-attributes": "Eigene Attribute umschalten", | ||||
|         "toggle-inherited-attributes": "Vererbte Attribute umschalten", | ||||
|         "toggle-promoted-attributes": "Beworbene Attribute umschalten", | ||||
|         "toggle-link-map": "Link-Karte umschalten", | ||||
|         "toggle-note-info": "Notizinformationen umschalten", | ||||
|         "toggle-note-paths": "Notizpfade umschalten", | ||||
|         "toggle-similar-notes": "Ähnliche Notizen umschalten", | ||||
|         "other": "Sonstige", | ||||
|         "toggle-right-pane": "Anzeige der rechten Leiste umschalten, das Inhaltsverzeichnis und Markierungen enthält", | ||||
|         "print-active-note": "Aktive Notiz drucken", | ||||
|         "open-note-externally": "Notiz als Datei mit Standardanwendung öffnen", | ||||
|         "render-active-note": "Aktive Notiz rendern (erneut rendern)", | ||||
|         "run-active-note": "Aktive JavaScript(Frontend/Backend)-Codenotiz ausführen", | ||||
|         "toggle-note-hoisting": "Notiz-Fokus der aktiven Notiz umschalten", | ||||
|         "unhoist": "Notiz-Fokus aufheben", | ||||
|         "reload-frontend-app": "Frontend-App neuladen", | ||||
|         "open-dev-tools": "Entwicklertools öffnen", | ||||
|         "toggle-left-note-tree-panel": "Linke Notizbaum-Leiste umschalten", | ||||
|         "toggle-full-screen": "Vollbildmodus umschalten", | ||||
|         "zoom-out": "Herauszoomen", | ||||
|         "zoom-in": "Hineinzoomen", | ||||
|         "note-navigation": "Notiznavigation", | ||||
|         "reset-zoom-level": "Zoomlevel zurücksetzen", | ||||
|         "copy-without-formatting": "Ausgewählten Text ohne Formatierung kopieren", | ||||
|         "force-save-revision": "Erstellen / Speichern einer neuen Notizrevision der aktiven Notiz erzwingen", | ||||
|         "show-help": "Eingebaute Hilfe / Cheat-Sheet anzeigen", | ||||
|         "toggle-book-properties": "Buch-Eigenschaften umschalten", | ||||
|         "clone-notes-to": "Ausgewählte Notizen duplizieren", | ||||
|         "open-command-palette": "Kommandopalette öffnen", | ||||
|         "export-as-pdf": "Aktuelle Notiz als PDF exportieren", | ||||
|         "back-in-note-history": "Navigiere zur vorherigen Notiz im Verlauf", | ||||
|         "forward-in-note-history": "Navigiere zur nächsten Notiz im Verlauf", | ||||
|         "scroll-to-active-note": "Zum aktiven Notizbaumeintrag springen", | ||||
|         "quick-search": "Schnellsuche öffnen", | ||||
|         "create-note-after": "Erstelle eine neue Notiz nach der aktuellen Notiz", | ||||
|         "create-note-into": "Unternotiz zur aktiven Notiz anlegen", | ||||
|         "move-notes-to": "Ausgewählte Notizen verschieben", | ||||
|         "show-cheatsheet": "Übersicht der Tastenkombinationen anzeigen", | ||||
|         "find-in-text": "Suchleiste umschalten", | ||||
|         "toggle-classic-editor-toolbar": "Schalte um zum Formatierungs-Tab für den Editor mit fester Werkzeugleiste", | ||||
|         "toggle-zen-mode": "Zen-Modus ein-/ausschalten (reduzierte Benutzeroberfläche für ablenkungsfreies Arbeiten)" | ||||
|     }, | ||||
|     "login": { | ||||
|         "title": "Anmeldung", | ||||
|         "heading": "Trilium Anmeldung", | ||||
|         "incorrect-password": "Das Passwort ist falsch. Bitte versuche es erneut.", | ||||
|         "password": "Passwort", | ||||
|         "remember-me": "Angemeldet bleiben", | ||||
|         "button": "Anmelden" | ||||
|     }, | ||||
|     "set_password": { | ||||
|         "title": "Passwort festlegen", | ||||
|         "heading": "Passwort festlegen", | ||||
|         "description": "Bevor du Trilium im Web verwenden kannst, musst du zuerst ein Passwort festlegen. Du wirst dieses Passwort dann zur Anmeldung verwenden.", | ||||
|         "password": "Passwort", | ||||
|         "password-confirmation": "Passwortbestätigung", | ||||
|         "button": "Passwort festlegen" | ||||
|     }, | ||||
|     "javascript-required": "Trilium erfordert, dass JavaScript aktiviert ist.", | ||||
|     "setup": { | ||||
|         "heading": "Trilium Notes Setup", | ||||
|         "new-document": "Ich bin ein neuer Benutzer und möchte ein neues Trilium-Dokument für meine Notizen erstellen", | ||||
|         "sync-from-desktop": "Ich habe bereits eine Desktop-Instanz und möchte die Synchronisierung damit einrichten", | ||||
|         "sync-from-server": "Ich habe bereits eine Server-Instanz und möchte die Synchronisierung damit einrichten", | ||||
|         "next": "Weiter", | ||||
|         "init-in-progress": "Dokumenteninitialisierung läuft", | ||||
|         "redirecting": "Du wirst in Kürze zur Anwendung weitergeleitet.", | ||||
|         "title": "Setup" | ||||
|     }, | ||||
|     "setup_sync-from-desktop": { | ||||
|         "heading": "Synchronisation vom Desktop", | ||||
|         "description": "Dieses Setup muss von der Desktop-Instanz aus initiiert werden:", | ||||
|         "step1": "Öffne deine Trilium Notes Desktop-Instanz.", | ||||
|         "step2": "Klicke im Trilium-Menü auf Optionen.", | ||||
|         "step3": "Klicke auf die Kategorie Synchronisation.", | ||||
|         "step4": "Ändere die Server-Instanzadresse auf: {{- host}} und klicke auf Speichern.", | ||||
|         "step5": "Klicke auf den Button \"Test-Synchronisation\", um zu überprüfen, ob die Verbindung erfolgreich ist.", | ||||
|         "step6": "Sobald du diese Schritte abgeschlossen hast, klicke auf {{- link}}.", | ||||
|         "step6-here": "hier" | ||||
|     }, | ||||
|     "setup_sync-from-server": { | ||||
|         "heading": "Synchronisation vom Server", | ||||
|         "instructions": "Bitte gib unten die Trilium-Server-Adresse und die Zugangsdaten ein. Dies wird das gesamte Trilium-Dokument vom Server herunterladen und die Synchronisation einrichten. Je nach Dokumentgröße und Verbindungsgeschwindigkeit kann dies eine Weile dauern.", | ||||
|         "server-host": "Trilium Server-Adresse", | ||||
|         "server-host-placeholder": "https://<hostname>:<port>", | ||||
|         "proxy-server": "Proxy-Server (optional)", | ||||
|         "proxy-server-placeholder": "https://<hostname>:<port>", | ||||
|         "note": "Hinweis:", | ||||
|         "proxy-instruction": "Wenn du die Proxy-Einstellung leer lässt, wird der System-Proxy verwendet (gilt nur für die Desktop-Anwendung)", | ||||
|         "password": "Passwort", | ||||
|         "password-placeholder": "Passwort", | ||||
|         "back": "Zurück", | ||||
|         "finish-setup": "Setup abschließen" | ||||
|     }, | ||||
|     "setup_sync-in-progress": { | ||||
|         "heading": "Synchronisation läuft", | ||||
|         "successful": "Die Synchronisation wurde erfolgreich eingerichtet. Es wird eine Weile dauern, bis die erste Synchronisation abgeschlossen ist. Sobald dies erledigt ist, wirst du zur Anmeldeseite weitergeleitet.", | ||||
|         "outstanding-items": "Ausstehende Synchronisationselemente:", | ||||
|         "outstanding-items-default": "N/A" | ||||
|     }, | ||||
|     "share_404": { | ||||
|         "title": "Nicht gefunden", | ||||
|         "heading": "Nicht gefunden" | ||||
|     }, | ||||
|     "share_page": { | ||||
|         "parent": "Übergeordnete Notiz:", | ||||
|         "clipped-from": "Diese Notiz wurde ursprünglich von {{- url}} ausgeschnitten", | ||||
|         "child-notes": "Untergeordnete Notizen:", | ||||
|         "no-content": "Diese Notiz hat keinen Inhalt." | ||||
|     }, | ||||
|     "weekdays": { | ||||
|         "monday": "Montag", | ||||
|         "tuesday": "Dienstag", | ||||
|         "wednesday": "Mittwoch", | ||||
|         "thursday": "Donnerstag", | ||||
|         "friday": "Freitag", | ||||
|         "saturday": "Samstag", | ||||
|         "sunday": "Sonntag" | ||||
|     }, | ||||
|     "months": { | ||||
|         "january": "Januar", | ||||
|         "february": "Februar", | ||||
|         "march": "März", | ||||
|         "april": "April", | ||||
|         "may": "Mai", | ||||
|         "june": "Juni", | ||||
|         "july": "Juli", | ||||
|         "august": "August", | ||||
|         "september": "September", | ||||
|         "october": "Oktober", | ||||
|         "november": "November", | ||||
|         "december": "Dezember" | ||||
|     }, | ||||
|     "special_notes": { | ||||
|         "search_prefix": "Suche:" | ||||
|     }, | ||||
|     "test_sync": { | ||||
|         "not-configured": "Der Synchronisations-Server-Host ist nicht konfiguriert. Bitte konfiguriere zuerst die Synchronisation.", | ||||
|         "successful": "Die Server-Verbindung wurde erfolgreich hergestellt, die Synchronisation wurde gestartet." | ||||
|     }, | ||||
|     "hidden-subtree": { | ||||
|         "root-title": "Versteckte Notizen", | ||||
|         "search-history-title": "Suchverlauf", | ||||
|         "note-map-title": "Notiz Karte", | ||||
|         "sql-console-history-title": "SQL Konsolen Verlauf", | ||||
|         "shared-notes-title": "Geteilte Notizen", | ||||
|         "bulk-action-title": "Massenverarbeitung", | ||||
|         "backend-log-title": "Backend Log", | ||||
|         "user-hidden-title": "Versteckt vom Nutzer", | ||||
|         "launch-bar-templates-title": "Startleiste Vorlagen", | ||||
|         "base-abstract-launcher-title": "Basis Abstrakte Startleiste", | ||||
|         "command-launcher-title": "Befehlslauncher", | ||||
|         "note-launcher-title": "Notiz Launcher", | ||||
|         "script-launcher-title": "Script Launcher", | ||||
|         "built-in-widget-title": "Eingebautes Widget", | ||||
|         "spacer-title": "Freifeld", | ||||
|         "custom-widget-title": "Custom Widget", | ||||
|         "launch-bar-title": "Launchbar", | ||||
|         "available-launchers-title": "Verfügbare Launchers", | ||||
|         "go-to-previous-note-title": "Zur vorherigen Notiz gehen", | ||||
|         "go-to-next-note-title": "Zur nächsten Notiz gehen", | ||||
|         "new-note-title": "Neue Notiz", | ||||
|         "search-notes-title": "Notizen durchsuchen", | ||||
|         "calendar-title": "Kalender", | ||||
|         "recent-changes-title": "neue Änderungen", | ||||
|         "bookmarks-title": "Lesezeichen", | ||||
|         "open-today-journal-note-title": "Heutigen Journaleintrag öffnen", | ||||
|         "quick-search-title": "Schnellsuche", | ||||
|         "protected-session-title": "Geschützte Sitzung", | ||||
|         "sync-status-title": "Sync Status", | ||||
|         "settings-title": "Einstellungen", | ||||
|         "options-title": "Optionen", | ||||
|         "appearance-title": "Erscheinungsbild", | ||||
|         "shortcuts-title": "Tastaturkürzel", | ||||
|         "text-notes": "Text Notizen", | ||||
|         "code-notes-title": "Code Notizen", | ||||
|         "images-title": "Bilder", | ||||
|         "spellcheck-title": "Rechtschreibprüfung", | ||||
|         "password-title": "Passwort", | ||||
|         "etapi-title": "ETAPI", | ||||
|         "backup-title": "Sicherung", | ||||
|         "sync-title": "Sync", | ||||
|         "other": "Weitere", | ||||
|         "advanced-title": "Erweitert", | ||||
|         "visible-launchers-title": "Sichtbare Launcher", | ||||
|         "user-guide": "Nutzerhandbuch" | ||||
|     }, | ||||
|     "notes": { | ||||
|         "new-note": "Neue Notiz", | ||||
|         "duplicate-note-suffix": "(dup)", | ||||
|         "duplicate-note-title": "{{- noteTitle }} {{ duplicateNoteSuffix }}" | ||||
|     }, | ||||
|     "backend_log": { | ||||
|         "log-does-not-exist": "Die Backend-Log-Datei '{{ fileName }}' existiert (noch) nicht.", | ||||
|         "reading-log-failed": "Das Lesen der Backend-Log-Datei '{{ fileName }}' ist fehlgeschlagen." | ||||
|     }, | ||||
|     "content_renderer": { | ||||
|         "note-cannot-be-displayed": "Dieser Notiztyp kann nicht angezeigt werden." | ||||
|     }, | ||||
|     "pdf": { | ||||
|         "export_filter": "PDF Dokument (*.pdf)", | ||||
|         "unable-to-export-message": "Die aktuelle Notiz konnte nicht als PDF exportiert werden.", | ||||
|         "unable-to-export-title": "Export als PDF fehlgeschlagen", | ||||
|         "unable-to-save-message": "Die ausgewählte Datei konnte nicht beschrieben werden. Erneut versuchen oder ein anderes Ziel auswählen." | ||||
|     }, | ||||
|     "tray": { | ||||
|         "tooltip": "Trilium Notes", | ||||
|         "close": "Trilium schließen", | ||||
|         "recents": "Kürzliche Notizen", | ||||
|         "bookmarks": "Lesezeichen", | ||||
|         "today": "Heutigen Journal Eintrag öffnen", | ||||
|         "new-note": "Neue Notiz", | ||||
|         "show-windows": "Fenster anzeigen" | ||||
|     }, | ||||
|     "hidden_subtree_templates": { | ||||
|         "table": "Tabelle", | ||||
|         "board_status_done": "Erledigt" | ||||
|     }, | ||||
|     "keyboard_action_names": { | ||||
|         "copy-notes-to-clipboard": "Notizen in Zwischenablage kopieren", | ||||
|         "paste-notes-from-clipboard": "Notizen aus Zwischenablage einfügen", | ||||
|         "back-in-note-history": "Zurück im Notizverlauf", | ||||
|         "forward-in-note-history": "Vorwärts im Notizverlauf", | ||||
|         "jump-to-note": "Wechseln zu...", | ||||
|         "command-palette": "Befehlsübersicht", | ||||
|         "scroll-to-active-note": "Zur aktiven Notiz scrollen", | ||||
|         "quick-search": "Schnellsuche", | ||||
|         "search-in-subtree": "In Unterzweig suchen", | ||||
|         "expand-subtree": "Unterzweig aufklappen", | ||||
|         "collapse-tree": "Baumstruktur einklappen", | ||||
|         "collapse-subtree": "Unterzweig einklappen", | ||||
|         "sort-child-notes": "Unternotizen sortieren", | ||||
|         "create-note-after": "Erstelle eine neue Notiz dahinter", | ||||
|         "create-note-into": "Erstelle eine neue Notiz davor", | ||||
|         "create-note-into-inbox": "Neue Notiz in Inbox erstellen", | ||||
|         "delete-notes": "Notizen löschen", | ||||
|         "move-note-up": "Notiz nach oben verschieben", | ||||
|         "move-note-down": "Notiz nach unten verschieben" | ||||
|     } | ||||
|   "keyboard_actions": { | ||||
|     "open-jump-to-note-dialog": "Öffne das Dialogfeld \"Zu Notiz springen\"", | ||||
|     "search-in-subtree": "Nach Notizen im Unterbaum der aktuellen Notiz suchen", | ||||
|     "expand-subtree": "Unterbaum der aktuellen Notiz ausklappen", | ||||
|     "collapse-tree": "Gesamten Notizbaum einklappen", | ||||
|     "collapse-subtree": "Unterbaum der aktuellen Notiz einklappen", | ||||
|     "sort-child-notes": "Untergeordnete Notizen sortieren", | ||||
|     "creating-and-moving-notes": "Notizen erstellen und verschieben", | ||||
|     "create-note-into-inbox": "Erstelle eine Notiz im Posteingang (falls definiert) oder in der Tagesnotiz", | ||||
|     "delete-note": "Notiz löschen", | ||||
|     "move-note-up": "Notiz nach oben verschieben", | ||||
|     "move-note-down": "Notiz nach unten verschieben", | ||||
|     "move-note-up-in-hierarchy": "Notiz in der Hierarchie nach oben verschieben", | ||||
|     "move-note-down-in-hierarchy": "Notiz in der Hierarchie nach unten verschieben", | ||||
|     "edit-note-title": "Vom Notiz-Baum zur Notiz-Detailansicht springen und den Titel bearbeiten", | ||||
|     "edit-branch-prefix": "Dialog zum Bearbeiten des Zweigpräfixes anzeigen", | ||||
|     "note-clipboard": "Notiz-Zwischenablage", | ||||
|     "copy-notes-to-clipboard": "Ausgewählte Notizen in die Zwischenablage kopieren", | ||||
|     "paste-notes-from-clipboard": "Notizen aus der Zwischenablage in die aktive Notiz einfügen", | ||||
|     "cut-notes-to-clipboard": "Ausgewählte Notizen in die Zwischenablage ausschneiden", | ||||
|     "select-all-notes-in-parent": "Alle Notizen der aktuellen Notizenebene auswählen", | ||||
|     "add-note-above-to-the-selection": "Notiz oberhalb der Auswahl hinzufügen", | ||||
|     "add-note-below-to-selection": "Notiz unterhalb der Auswahl hinzufügen", | ||||
|     "duplicate-subtree": "Unterbaum duplizieren", | ||||
|     "tabs-and-windows": "Tabs & Fenster", | ||||
|     "open-new-tab": "Neuen Tab öffnen", | ||||
|     "close-active-tab": "Aktiven Tab schließen", | ||||
|     "reopen-last-tab": "Zuletzt geschlossenen Tab wieder öffnen", | ||||
|     "activate-next-tab": "Rechten Tab aktivieren", | ||||
|     "activate-previous-tab": "Linken Tab aktivieren", | ||||
|     "open-new-window": "Neues leeres Fenster öffnen", | ||||
|     "toggle-tray": "Anwendung im Systemtray anzeigen/verstecken", | ||||
|     "first-tab": "Ersten Tab in der Liste aktivieren", | ||||
|     "second-tab": "Zweiten Tab in der Liste aktivieren", | ||||
|     "third-tab": "Dritten Tab in der Liste aktivieren", | ||||
|     "fourth-tab": "Vierten Tab in der Liste aktivieren", | ||||
|     "fifth-tab": "Fünften Tab in der Liste aktivieren", | ||||
|     "sixth-tab": "Sechsten Tab in der Liste aktivieren", | ||||
|     "seventh-tab": "Siebten Tab in der Liste aktivieren", | ||||
|     "eight-tab": "Achten Tab in der Liste aktivieren", | ||||
|     "ninth-tab": "Neunten Tab in der Liste aktivieren", | ||||
|     "last-tab": "Letzten Tab in der Liste aktivieren", | ||||
|     "dialogs": "Dialoge", | ||||
|     "show-note-source": "Notizquellen-Dialog anzeigen", | ||||
|     "show-options": "Optionen-Dialog anzeigen", | ||||
|     "show-revisions": "Notizrevisionen-Dialog anzeigen", | ||||
|     "show-recent-changes": "Letzte Änderungen-Dialog anzeigen", | ||||
|     "show-sql-console": "SQL-Konsole-Dialog anzeigen", | ||||
|     "show-backend-log": "Backend-Logs-Dialog anzeigen", | ||||
|     "text-note-operations": "Textnotiz-Operationen", | ||||
|     "add-link-to-text": "Dialogfeld zum Hinzufügen eines Links zum Text öffnen", | ||||
|     "follow-link-under-cursor": "Folge dem Link, unter dem Mauszeiger", | ||||
|     "insert-date-and-time-to-text": "Aktuelles Datum & Uhrzeit in den Text einfügen", | ||||
|     "paste-markdown-into-text": "Markdown aus der Zwischenablage in die Textnotiz einfügen", | ||||
|     "cut-into-note": "Auswahl aus der aktuellen Notiz ausschneiden und eine Unternotiz mit dem ausgewählten Text erstellen", | ||||
|     "add-include-note-to-text": "Notiz-Einfügen-Dialog öffnen", | ||||
|     "edit-readonly-note": "Schreibgeschützte Notiz bearbeiten", | ||||
|     "attributes-labels-and-relations": "Attribute (Labels & Verknüpfungen)", | ||||
|     "add-new-label": "Neues Label erstellen", | ||||
|     "create-new-relation": "Neue Verknüpfungen", | ||||
|     "ribbon-tabs": "Ribbon-Tabs", | ||||
|     "toggle-basic-properties": "Grundattribute umschalten", | ||||
|     "toggle-file-properties": "Dateiattribute umschalten", | ||||
|     "toggle-image-properties": "Bildattribute umschalten", | ||||
|     "toggle-owned-attributes": "Eigene Attribute umschalten", | ||||
|     "toggle-inherited-attributes": "Vererbte Attribute umschalten", | ||||
|     "toggle-promoted-attributes": "Beworbene Attribute umschalten", | ||||
|     "toggle-link-map": "Link-Karte umschalten", | ||||
|     "toggle-note-info": "Notizinformationen umschalten", | ||||
|     "toggle-note-paths": "Notizpfade umschalten", | ||||
|     "toggle-similar-notes": "Ähnliche Notizen umschalten", | ||||
|     "other": "Sonstige", | ||||
|     "toggle-right-pane": "Anzeige der rechten Leiste umschalten, das Inhaltsverzeichnis und Markierungen enthält", | ||||
|     "print-active-note": "Aktive Notiz drucken", | ||||
|     "open-note-externally": "Notiz als Datei mit Standardanwendung öffnen", | ||||
|     "render-active-note": "Aktive Notiz rendern (erneut rendern)", | ||||
|     "run-active-note": "Aktive JavaScript(Frontend/Backend)-Codenotiz ausführen", | ||||
|     "toggle-note-hoisting": "Notiz-Fokus der aktiven Notiz umschalten", | ||||
|     "unhoist": "Notiz-Fokus aufheben", | ||||
|     "reload-frontend-app": "Frontend-App neuladen", | ||||
|     "open-dev-tools": "Entwicklertools öffnen", | ||||
|     "toggle-left-note-tree-panel": "Linke Notizbaum-Leiste umschalten", | ||||
|     "toggle-full-screen": "Vollbildmodus umschalten", | ||||
|     "zoom-out": "Herauszoomen", | ||||
|     "zoom-in": "Hineinzoomen", | ||||
|     "note-navigation": "Notiznavigation", | ||||
|     "reset-zoom-level": "Zoomlevel zurücksetzen", | ||||
|     "copy-without-formatting": "Ausgewählten Text ohne Formatierung kopieren", | ||||
|     "force-save-revision": "Erstellen / Speichern einer neuen Notizrevision der aktiven Notiz erzwingen", | ||||
|     "show-help": "Eingebaute Hilfe / Cheat-Sheet anzeigen", | ||||
|     "toggle-book-properties": "Buch-Eigenschaften umschalten", | ||||
|     "clone-notes-to": "Ausgewählte Notizen duplizieren", | ||||
|     "open-command-palette": "Kommandopalette öffnen", | ||||
|     "export-as-pdf": "Aktuelle Notiz als PDF exportieren", | ||||
|     "back-in-note-history": "Navigiere zur vorherigen Notiz im Verlauf", | ||||
|     "forward-in-note-history": "Navigiere zur nächsten Notiz im Verlauf", | ||||
|     "scroll-to-active-note": "Zum aktiven Notizbaumeintrag springen", | ||||
|     "quick-search": "Schnellsuche öffnen", | ||||
|     "create-note-after": "Erstelle eine neue Notiz nach der aktuellen Notiz", | ||||
|     "create-note-into": "Unternotiz zur aktiven Notiz anlegen", | ||||
|     "move-notes-to": "Ausgewählte Notizen verschieben", | ||||
|     "show-cheatsheet": "Übersicht der Tastenkombinationen anzeigen", | ||||
|     "find-in-text": "Suchleiste umschalten", | ||||
|     "toggle-classic-editor-toolbar": "Schalte um zum Formatierungs-Tab für den Editor mit fester Werkzeugleiste", | ||||
|     "toggle-zen-mode": "Zen-Modus ein-/ausschalten (reduzierte Benutzeroberfläche für ablenkungsfreies Arbeiten)" | ||||
|   }, | ||||
|   "login": { | ||||
|     "title": "Anmeldung", | ||||
|     "heading": "Trilium Anmeldung", | ||||
|     "incorrect-password": "Das Passwort ist falsch. Bitte versuche es erneut.", | ||||
|     "password": "Passwort", | ||||
|     "remember-me": "Angemeldet bleiben", | ||||
|     "button": "Anmelden", | ||||
|     "incorrect-totp": "Einmaltoken ist falsch. Bitte erneut versuchen.", | ||||
|     "sign_in_with_sso": "Mit {{ ssoIssuerName }} anmelden" | ||||
|   }, | ||||
|   "set_password": { | ||||
|     "title": "Passwort festlegen", | ||||
|     "heading": "Passwort festlegen", | ||||
|     "description": "Bevor du Trilium im Web verwenden kannst, musst du zuerst ein Passwort festlegen. Du wirst dieses Passwort dann zur Anmeldung verwenden.", | ||||
|     "password": "Passwort", | ||||
|     "password-confirmation": "Passwortbestätigung", | ||||
|     "button": "Passwort festlegen" | ||||
|   }, | ||||
|   "javascript-required": "Trilium erfordert, dass JavaScript aktiviert ist.", | ||||
|   "setup": { | ||||
|     "heading": "Trilium Notes Setup", | ||||
|     "new-document": "Ich bin ein neuer Benutzer und möchte ein neues Trilium-Dokument für meine Notizen erstellen", | ||||
|     "sync-from-desktop": "Ich habe bereits eine Desktop-Instanz und möchte die Synchronisierung damit einrichten", | ||||
|     "sync-from-server": "Ich habe bereits eine Server-Instanz und möchte die Synchronisierung damit einrichten", | ||||
|     "next": "Weiter", | ||||
|     "init-in-progress": "Dokumenteninitialisierung läuft", | ||||
|     "redirecting": "Du wirst in Kürze zur Anwendung weitergeleitet.", | ||||
|     "title": "Setup" | ||||
|   }, | ||||
|   "setup_sync-from-desktop": { | ||||
|     "heading": "Synchronisation vom Desktop", | ||||
|     "description": "Dieses Setup muss von der Desktop-Instanz aus initiiert werden:", | ||||
|     "step1": "Öffne deine Trilium Notes Desktop-Instanz.", | ||||
|     "step2": "Klicke im Trilium-Menü auf Optionen.", | ||||
|     "step3": "Klicke auf die Kategorie Synchronisation.", | ||||
|     "step4": "Ändere die Server-Instanzadresse auf: {{- host}} und klicke auf Speichern.", | ||||
|     "step5": "Klicke auf den Button \"Test-Synchronisation\", um zu überprüfen, ob die Verbindung erfolgreich ist.", | ||||
|     "step6": "Sobald du diese Schritte abgeschlossen hast, klicke auf {{- link}}.", | ||||
|     "step6-here": "hier" | ||||
|   }, | ||||
|   "setup_sync-from-server": { | ||||
|     "heading": "Synchronisation vom Server", | ||||
|     "instructions": "Bitte gib unten die Trilium-Server-Adresse und die Zugangsdaten ein. Dies wird das gesamte Trilium-Dokument vom Server herunterladen und die Synchronisation einrichten. Je nach Dokumentgröße und Verbindungsgeschwindigkeit kann dies eine Weile dauern.", | ||||
|     "server-host": "Trilium Server-Adresse", | ||||
|     "server-host-placeholder": "https://<hostname>:<port>", | ||||
|     "proxy-server": "Proxy-Server (optional)", | ||||
|     "proxy-server-placeholder": "https://<hostname>:<port>", | ||||
|     "note": "Hinweis:", | ||||
|     "proxy-instruction": "Wenn du die Proxy-Einstellung leer lässt, wird der System-Proxy verwendet (gilt nur für die Desktop-Anwendung)", | ||||
|     "password": "Passwort", | ||||
|     "password-placeholder": "Passwort", | ||||
|     "back": "Zurück", | ||||
|     "finish-setup": "Setup abschließen" | ||||
|   }, | ||||
|   "setup_sync-in-progress": { | ||||
|     "heading": "Synchronisation läuft", | ||||
|     "successful": "Die Synchronisation wurde erfolgreich eingerichtet. Es wird eine Weile dauern, bis die erste Synchronisation abgeschlossen ist. Sobald dies erledigt ist, wirst du zur Anmeldeseite weitergeleitet.", | ||||
|     "outstanding-items": "Ausstehende Synchronisationselemente:", | ||||
|     "outstanding-items-default": "N/A" | ||||
|   }, | ||||
|   "share_404": { | ||||
|     "title": "Nicht gefunden", | ||||
|     "heading": "Nicht gefunden" | ||||
|   }, | ||||
|   "share_page": { | ||||
|     "parent": "Übergeordnete Notiz:", | ||||
|     "clipped-from": "Diese Notiz wurde ursprünglich von {{- url}} ausgeschnitten", | ||||
|     "child-notes": "Untergeordnete Notizen:", | ||||
|     "no-content": "Diese Notiz hat keinen Inhalt." | ||||
|   }, | ||||
|   "weekdays": { | ||||
|     "monday": "Montag", | ||||
|     "tuesday": "Dienstag", | ||||
|     "wednesday": "Mittwoch", | ||||
|     "thursday": "Donnerstag", | ||||
|     "friday": "Freitag", | ||||
|     "saturday": "Samstag", | ||||
|     "sunday": "Sonntag" | ||||
|   }, | ||||
|   "months": { | ||||
|     "january": "Januar", | ||||
|     "february": "Februar", | ||||
|     "march": "März", | ||||
|     "april": "April", | ||||
|     "may": "Mai", | ||||
|     "june": "Juni", | ||||
|     "july": "Juli", | ||||
|     "august": "August", | ||||
|     "september": "September", | ||||
|     "october": "Oktober", | ||||
|     "november": "November", | ||||
|     "december": "Dezember" | ||||
|   }, | ||||
|   "special_notes": { | ||||
|     "search_prefix": "Suche:" | ||||
|   }, | ||||
|   "test_sync": { | ||||
|     "not-configured": "Der Synchronisations-Server-Host ist nicht konfiguriert. Bitte konfiguriere zuerst die Synchronisation.", | ||||
|     "successful": "Die Server-Verbindung wurde erfolgreich hergestellt, die Synchronisation wurde gestartet." | ||||
|   }, | ||||
|   "hidden-subtree": { | ||||
|     "root-title": "Versteckte Notizen", | ||||
|     "search-history-title": "Suchverlauf", | ||||
|     "note-map-title": "Notiz Karte", | ||||
|     "sql-console-history-title": "SQL Konsolen Verlauf", | ||||
|     "shared-notes-title": "Geteilte Notizen", | ||||
|     "bulk-action-title": "Massenverarbeitung", | ||||
|     "backend-log-title": "Backend Log", | ||||
|     "user-hidden-title": "Versteckt vom Nutzer", | ||||
|     "launch-bar-templates-title": "Startleiste Vorlagen", | ||||
|     "base-abstract-launcher-title": "Basis Abstrakte Startleiste", | ||||
|     "command-launcher-title": "Befehlslauncher", | ||||
|     "note-launcher-title": "Notiz Launcher", | ||||
|     "script-launcher-title": "Script Launcher", | ||||
|     "built-in-widget-title": "Eingebautes Widget", | ||||
|     "spacer-title": "Freifeld", | ||||
|     "custom-widget-title": "Custom Widget", | ||||
|     "launch-bar-title": "Launchbar", | ||||
|     "available-launchers-title": "Verfügbare Launchers", | ||||
|     "go-to-previous-note-title": "Zur vorherigen Notiz gehen", | ||||
|     "go-to-next-note-title": "Zur nächsten Notiz gehen", | ||||
|     "new-note-title": "Neue Notiz", | ||||
|     "search-notes-title": "Notizen durchsuchen", | ||||
|     "calendar-title": "Kalender", | ||||
|     "recent-changes-title": "neue Änderungen", | ||||
|     "bookmarks-title": "Lesezeichen", | ||||
|     "open-today-journal-note-title": "Heutigen Journaleintrag öffnen", | ||||
|     "quick-search-title": "Schnellsuche", | ||||
|     "protected-session-title": "Geschützte Sitzung", | ||||
|     "sync-status-title": "Sync Status", | ||||
|     "settings-title": "Einstellungen", | ||||
|     "options-title": "Optionen", | ||||
|     "appearance-title": "Erscheinungsbild", | ||||
|     "shortcuts-title": "Tastaturkürzel", | ||||
|     "text-notes": "Text Notizen", | ||||
|     "code-notes-title": "Code Notizen", | ||||
|     "images-title": "Bilder", | ||||
|     "spellcheck-title": "Rechtschreibprüfung", | ||||
|     "password-title": "Passwort", | ||||
|     "etapi-title": "ETAPI", | ||||
|     "backup-title": "Sicherung", | ||||
|     "sync-title": "Sync", | ||||
|     "other": "Weitere", | ||||
|     "advanced-title": "Erweitert", | ||||
|     "visible-launchers-title": "Sichtbare Launcher", | ||||
|     "user-guide": "Nutzerhandbuch", | ||||
|     "jump-to-note-title": "Springe zu...", | ||||
|     "llm-chat-title": "Chat mit Notizen", | ||||
|     "multi-factor-authentication-title": "MFA", | ||||
|     "ai-llm-title": "AI/LLM", | ||||
|     "localization": "Sprache & Region", | ||||
|     "inbox-title": "Posteingang" | ||||
|   }, | ||||
|   "notes": { | ||||
|     "new-note": "Neue Notiz", | ||||
|     "duplicate-note-suffix": "(dup)", | ||||
|     "duplicate-note-title": "{{- noteTitle }} {{ duplicateNoteSuffix }}" | ||||
|   }, | ||||
|   "backend_log": { | ||||
|     "log-does-not-exist": "Die Backend-Log-Datei '{{ fileName }}' existiert (noch) nicht.", | ||||
|     "reading-log-failed": "Das Lesen der Backend-Log-Datei '{{ fileName }}' ist fehlgeschlagen." | ||||
|   }, | ||||
|   "content_renderer": { | ||||
|     "note-cannot-be-displayed": "Dieser Notiztyp kann nicht angezeigt werden." | ||||
|   }, | ||||
|   "pdf": { | ||||
|     "export_filter": "PDF Dokument (*.pdf)", | ||||
|     "unable-to-export-message": "Die aktuelle Notiz konnte nicht als PDF exportiert werden.", | ||||
|     "unable-to-export-title": "Export als PDF fehlgeschlagen", | ||||
|     "unable-to-save-message": "Die ausgewählte Datei konnte nicht beschrieben werden. Erneut versuchen oder ein anderes Ziel auswählen." | ||||
|   }, | ||||
|   "tray": { | ||||
|     "tooltip": "Trilium Notes", | ||||
|     "close": "Trilium schließen", | ||||
|     "recents": "Kürzliche Notizen", | ||||
|     "bookmarks": "Lesezeichen", | ||||
|     "today": "Heutigen Journal Eintrag öffnen", | ||||
|     "new-note": "Neue Notiz", | ||||
|     "show-windows": "Fenster anzeigen", | ||||
|     "open_new_window": "Öffne neues Fenster" | ||||
|   }, | ||||
|   "hidden_subtree_templates": { | ||||
|     "table": "Tabelle", | ||||
|     "board_status_done": "Erledigt" | ||||
|   }, | ||||
|   "keyboard_action_names": { | ||||
|     "copy-notes-to-clipboard": "Notizen in Zwischenablage kopieren", | ||||
|     "paste-notes-from-clipboard": "Notizen aus Zwischenablage einfügen", | ||||
|     "back-in-note-history": "Zurück im Notizverlauf", | ||||
|     "forward-in-note-history": "Vorwärts im Notizverlauf", | ||||
|     "jump-to-note": "Wechseln zu...", | ||||
|     "command-palette": "Befehlsübersicht", | ||||
|     "scroll-to-active-note": "Zur aktiven Notiz scrollen", | ||||
|     "quick-search": "Schnellsuche", | ||||
|     "search-in-subtree": "In Unterzweig suchen", | ||||
|     "expand-subtree": "Unterzweig aufklappen", | ||||
|     "collapse-tree": "Baumstruktur einklappen", | ||||
|     "collapse-subtree": "Unterzweig einklappen", | ||||
|     "sort-child-notes": "Unternotizen sortieren", | ||||
|     "create-note-after": "Erstelle eine neue Notiz dahinter", | ||||
|     "create-note-into": "Erstelle eine neue Notiz davor", | ||||
|     "create-note-into-inbox": "Neue Notiz in Inbox erstellen", | ||||
|     "delete-notes": "Notizen löschen", | ||||
|     "move-note-up": "Notiz nach oben verschieben", | ||||
|     "move-note-down": "Notiz nach unten verschieben", | ||||
|     "edit-note-title": "Bearbeite Notiz Titel", | ||||
|     "clone-notes-to": "Vervielfältige Notiz nach", | ||||
|     "move-notes-to": "Verschiebe Notiz nach", | ||||
|     "cut-notes-to-clipboard": "Notizen in Zwischenablage ausschneiden", | ||||
|     "add-note-above-to-selection": "Notiz oberhalb der Selektion hinzufügen", | ||||
|     "add-note-below-to-selection": "Notiz unterhalb der Selektion hinzufügen", | ||||
|     "open-new-tab": "Öffne im neuen Tab", | ||||
|     "close-active-tab": "Schließe aktiven Tab", | ||||
|     "reopen-last-tab": "Öffne zuletzt geschlossenen Tab", | ||||
|     "activate-next-tab": "Aktiviere nächsten Tab", | ||||
|     "activate-previous-tab": "Aktiviere vorherigen Tab", | ||||
|     "open-new-window": "Öffne im neuen Fenster", | ||||
|     "toggle-system-tray-icon": "Systemablage-Symbol umschalten", | ||||
|     "toggle-zen-mode": "Zen-Modus umschalten", | ||||
|     "switch-to-first-tab": "Wechsle zum ersten Tab", | ||||
|     "switch-to-second-tab": "Wechsle zum zweiten Tab", | ||||
|     "switch-to-third-tab": "Wechsle zum dritten Tab", | ||||
|     "switch-to-fourth-tab": "Wechsle zum vierten Tab", | ||||
|     "switch-to-fifth-tab": "Wechsle zum fünften Tab", | ||||
|     "switch-to-sixth-tab": "Wechsle zum sechsten Tab", | ||||
|     "switch-to-seventh-tab": "Wechsle zum siebten Tab", | ||||
|     "switch-to-eighth-tab": "Wechsle zum achten Tab", | ||||
|     "switch-to-ninth-tab": "Wechsle zum neunten Tab", | ||||
|     "switch-to-last-tab": "Wechsle zum letzten Tab", | ||||
|     "show-note-source": "Zeige Notiz Quelle", | ||||
|     "show-options": "Zeige Optionen", | ||||
|     "show-revisions": "Zeige Revisionen", | ||||
|     "show-recent-changes": "Zeige letzte Änderungen", | ||||
|     "show-sql-console": "Zeige SQL Konsole", | ||||
|     "show-backend-log": "Zeige Backend-Protokoll", | ||||
|     "show-help": "Zeige Hilfe", | ||||
|     "show-cheatsheet": "Zeige Cheatsheet", | ||||
|     "add-link-to-text": "Link zum Text hinzufügen", | ||||
|     "cut-into-note": "In neue Notiz verschieben", | ||||
|     "add-new-label": "Neues Label hinzufügen", | ||||
|     "add-new-relation": "Neue Beziehung hinzufügen", | ||||
|     "print-active-note": "Drucke aktive Notiz", | ||||
|     "export-active-note-as-pdf": "Exportiere aktive Notiz als PDF", | ||||
|     "move-note-up-in-hierarchy": "Notiz in der Hierarchie nach oben verschieben", | ||||
|     "move-note-down-in-hierarchy": "Notiz in der Hierarchie nach unten verschieben", | ||||
|     "edit-branch-prefix": "Zweigpräfix bearbeiten", | ||||
|     "select-all-notes-in-parent": "Alle Notizen in übergeordnetem Element auswählen", | ||||
|     "duplicate-subtree": "Unterbaum duplizieren", | ||||
|     "follow-link-under-cursor": "Folge Link unterhalb des Mauszeigers", | ||||
|     "insert-date-and-time-to-text": "Datum und Uhrzeit in Text einfügen", | ||||
|     "paste-markdown-into-text": "Markdown in Text einfügen", | ||||
|     "add-include-note-to-text": "Notiz in Text einfügen", | ||||
|     "edit-read-only-note": "Schreibgeschützte Notiz bearbeiten", | ||||
|     "toggle-ribbon-tab-classic-editor": "Registerkarte Klassischer Editor umschalten", | ||||
|     "toggle-ribbon-tab-basic-properties": "Registerkarte Grundlegende Eigenschaften umschalten", | ||||
|     "toggle-ribbon-tab-book-properties": "Registerkarte Buch-Eigenschaften umschalten", | ||||
|     "toggle-ribbon-tab-file-properties": "Registerkarte Datei-Eigenschaften umschalten", | ||||
|     "toggle-ribbon-tab-image-properties": "Registerkarte Bilder-Eigenschaften umschalten", | ||||
|     "toggle-ribbon-tab-owned-attributes": "Registerkarte Besitzerattribute umschalten", | ||||
|     "toggle-ribbon-tab-inherited-attributes": "Registerkarte geerbte Attribute umschalten", | ||||
|     "toggle-ribbon-tab-promoted-attributes": "Registerkarte verliehene Attribute umschalten", | ||||
|     "toggle-ribbon-tab-note-map": "Registerkarte Notizkarte umschalten", | ||||
|     "toggle-ribbon-tab-note-info": "Registerkarte Notiz-Info umschalten", | ||||
|     "toggle-ribbon-tab-note-paths": "Registerkarte Notiz-Pfad umschalten", | ||||
|     "toggle-ribbon-tab-similar-notes": "Registerkarte ähnliche Notizen umschalten", | ||||
|     "toggle-right-pane": "Rechten Bereich ein-/ausblenden", | ||||
|     "open-note-externally": "Notiz extern öffnen", | ||||
|     "render-active-note": "Aktive Notiz rendern", | ||||
|     "run-active-note": "Aktive Notiz ausführen", | ||||
|     "toggle-note-hoisting": "Notiz hochziehen umschalten", | ||||
|     "unhoist-note": "Notiz hochziehen rückgängig machen", | ||||
|     "reload-frontend-app": "Oberfläche neu laden", | ||||
|     "open-developer-tools": "Öffne Entwickler-Tools", | ||||
|     "find-in-text": "Im Text suchen", | ||||
|     "toggle-left-pane": "Linken Bereich ein-/ausblenden", | ||||
|     "toggle-full-screen": "Vollbild-Modus de-/aktivieren", | ||||
|     "zoom-out": "Vergrößern", | ||||
|     "zoom-in": "Verkleinern", | ||||
|     "reset-zoom-level": "Zoom zurücksetzen", | ||||
|     "copy-without-formatting": "Kopieren ohne Formatierung", | ||||
|     "force-save-revision": "Speichern der Notizrevision erzwingen" | ||||
|   }, | ||||
|   "weekdayNumber": "Woche {weekNumber}", | ||||
|   "quarterNumber": "Quartal {quarterNumber}", | ||||
|   "migration": { | ||||
|     "old_version": "Eine direkte Migration von Ihrer aktuellen Version wird nicht unterstützt. Bitte führen Sie zunächst ein Upgrade auf die neueste Version v0.60.4 durch und erst dann auf diese Version.", | ||||
|     "error_message": "Fehler bei der Migration zu Version {{version}}: {{stack}}", | ||||
|     "wrong_db_version": "Die Version der Datenbank ({{version}}) ist neuer als die von der Anwendung erwartete Version ({{targetVersion}}), was bedeutet, dass diese mit einer neueren und inkompatiblen Version von Trilium erstellt wurde. Führen Sie ein Upgrade auf die neueste Version von Trilium durch, um dieses Problem zu beheben." | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -350,6 +350,13 @@ | ||||
|     "add-include-note-to-text": "Ajouter une note inclusion au texte", | ||||
|     "edit-read-only-note": "Modifier une note en lecture seule", | ||||
|     "add-new-label": "Ajouter une nouvelle étiquette", | ||||
|     "add-new-relation": "Ajouter une nouvelle relation" | ||||
|     "add-new-relation": "Ajouter une nouvelle relation", | ||||
|     "toggle-ribbon-tab-classic-editor": "Basculer l'onglet Mise en forme de l'éditeur avec la barre d'outils fixe", | ||||
|     "toggle-ribbon-tab-basic-properties": "Afficher/masquer les Propriétés de base de la note", | ||||
|     "toggle-ribbon-tab-book-properties": "Afficher/masquer les Propriétés du Livre", | ||||
|     "toggle-ribbon-tab-file-properties": "Afficher/masquer les Propriétés du fichier", | ||||
|     "toggle-ribbon-tab-image-properties": "Afficher/masquer les Propriétés de l'image", | ||||
|     "toggle-ribbon-tab-owned-attributes": "Afficher/masquer les Attributs propres", | ||||
|     "toggle-ribbon-tab-inherited-attributes": "Afficher/masquer les Attributs hérités" | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										43
									
								
								apps/server/src/assets/translations/ko/server.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								apps/server/src/assets/translations/ko/server.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| { | ||||
|   "keyboard_actions": { | ||||
|     "back-in-note-history": "기록 내 이전 노트로 이동하기", | ||||
|     "open-command-palette": "명령어 팔레트 열기", | ||||
|     "delete-note": "노트 삭제", | ||||
|     "quick-search": "빠른 검색바 활성화", | ||||
|     "move-note-up": "노트를 위로 이동", | ||||
|     "move-note-down": "노트를 아래로 이동", | ||||
|     "move-note-up-in-hierarchy": "노트를 상위 계층으로 이동", | ||||
|     "move-note-down-in-hierarchy": "노트를 하위 계층으로 이동", | ||||
|     "sort-child-notes": "자식 노트 정렬", | ||||
|     "forward-in-note-history": "기록 내 다음 노트로 이동하기", | ||||
|     "open-jump-to-note-dialog": "\"노트로 이동하기\" 대화창 열기", | ||||
|     "scroll-to-active-note": "노트 트리를 활성화된 노트로 스크롤하기", | ||||
|     "search-in-subtree": "활성화된 노트 하위에서 노트 찾기", | ||||
|     "expand-subtree": "현재 노트의 서브트리 펼치기", | ||||
|     "collapse-tree": "전체 노트 트리 접기", | ||||
|     "collapse-subtree": "현재 노트의 서브트리 접기", | ||||
|     "creating-and-moving-notes": "노트 만들기 및 이동하기", | ||||
|     "create-note-after": "노트 활성화 후 노트 만들기", | ||||
|     "create-note-into": "활성화된 노트 하위에 노트 만들기", | ||||
|     "create-note-into-inbox": "받은 편지함(정의된 경우) 또는 당일 노트에 노트 만들기", | ||||
|     "edit-note-title": "트리에서 노트 세부 사항으로 이동하고 제목 편집하기", | ||||
|     "edit-branch-prefix": "\"브랜치 접두사 편집\" 대화창 표시하기", | ||||
|     "clone-notes-to": "선택된 노트들을 복사하기", | ||||
|     "move-notes-to": "선택된 노트들을 이동하기", | ||||
|     "note-clipboard": "노트 클립보드", | ||||
|     "copy-notes-to-clipboard": "선택된 노트들을 클립보드에 복사하기", | ||||
|     "paste-notes-from-clipboard": "클립보드 내 노트를 활성화된 노트에 붙여넣기", | ||||
|     "cut-notes-to-clipboard": "선택된 노트들을 클립보드에 잘라내기", | ||||
|     "select-all-notes-in-parent": "현재 노트 레벨에서 모든 노트 선택하기", | ||||
|     "add-note-above-to-the-selection": "선택한 영역 위에 노트 추가하기", | ||||
|     "add-note-below-to-selection": "선택한 영역 아래에 노트 추가하기", | ||||
|     "duplicate-subtree": "서브트리 복사하기", | ||||
|     "open-new-tab": "새 탭 열기", | ||||
|     "close-active-tab": "활성화된 탭 닫기", | ||||
|     "reopen-last-tab": "마지막으로 닫은 탭 열기", | ||||
|     "activate-next-tab": "우측 탭 활성화", | ||||
|     "activate-previous-tab": "좌측 탭 활성화", | ||||
|     "open-new-window": "새 비어있는 창 열기", | ||||
|     "toggle-tray": "시스템 트레이에서 애플리케이션 보여주기/숨기기" | ||||
|   } | ||||
| } | ||||
| @@ -71,10 +71,10 @@ | ||||
|     "add-new-label": "Создать новую заметку", | ||||
|     "create-new-relation": "Создать новое отношение", | ||||
|     "ribbon-tabs": "Вкладки ленты", | ||||
|     "toggle-basic-properties": "Перейти к основным свойствам", | ||||
|     "toggle-basic-properties": "Перейти к общим параметрам", | ||||
|     "toggle-file-properties": "Перейти к свойствам файла", | ||||
|     "toggle-image-properties": "Перейти к свойствам изображения", | ||||
|     "toggle-owned-attributes": "Перейти к собственным атрибутами", | ||||
|     "toggle-owned-attributes": "Перейти к атрибутам", | ||||
|     "toggle-inherited-attributes": "Перейти к унаследованным атрибутам", | ||||
|     "toggle-promoted-attributes": "Перейти к продвигаемым атрибутам", | ||||
|     "toggle-link-map": "Перейти к картуессылок", | ||||
| @@ -100,7 +100,10 @@ | ||||
|     "toggle-book-properties": "Перейти к свойствам коллекции", | ||||
|     "toggle-classic-editor-toolbar": "Перейти на вкладку «Форматирование» для редактора с фиксированной панелью инструментов", | ||||
|     "export-as-pdf": "Экспортировать текущую заметку в формате PDF", | ||||
|     "toggle-zen-mode": "Включает/отключает режим дзен (минимальный пользовательский интерфейс для фокусирования)" | ||||
|     "toggle-zen-mode": "Включает/отключает режим дзен (минимальный пользовательский интерфейс для фокусирования)", | ||||
|     "toggle-note-hoisting": "Переключить закрепление активной заметки", | ||||
|     "unhoist": "Убрать закрепление везде", | ||||
|     "force-save-revision": "Принудительное создать/сохранить снимок версии активной заметки" | ||||
|   }, | ||||
|   "hidden-subtree": { | ||||
|     "localization": "Язык и регион", | ||||
| @@ -147,12 +150,13 @@ | ||||
|     "visible-launchers-title": "Видимые лаунчеры", | ||||
|     "user-guide": "Руководство пользователя", | ||||
|     "sql-console-history-title": "История консоли SQL", | ||||
|     "user-hidden-title": "Пользователь скрыт", | ||||
|     "user-hidden-title": "Скрытый пользователь", | ||||
|     "launch-bar-templates-title": "Шаблоны панели запуска", | ||||
|     "base-abstract-launcher-title": "Базовый абстрактный лаунчер", | ||||
|     "go-to-previous-note-title": "К предыдущей заметке", | ||||
|     "go-to-next-note-title": "К следующей заметке", | ||||
|     "open-today-journal-note-title": "Открыть сегодняшнюю заметку в журнале" | ||||
|     "open-today-journal-note-title": "Открыть сегодняшнюю заметку в журнале", | ||||
|     "llm-chat-title": "ИИ чат с заметками" | ||||
|   }, | ||||
|   "tray": { | ||||
|     "bookmarks": "Закладки", | ||||
| @@ -160,7 +164,9 @@ | ||||
|     "close": "Выйти из Trilium", | ||||
|     "recents": "Последние заметки", | ||||
|     "new-note": "Новая заметка", | ||||
|     "show-windows": "Показать окна" | ||||
|     "show-windows": "Показать окна", | ||||
|     "open_new_window": "Открыть новое окно", | ||||
|     "today": "Открыть заметку дня" | ||||
|   }, | ||||
|   "keyboard_action_names": { | ||||
|     "scroll-to-active-note": "Прокрутить к активной заметке", | ||||
| @@ -238,7 +244,26 @@ | ||||
|     "toggle-full-screen": "Переключить на полный экран", | ||||
|     "reset-zoom-level": "Сбросить уровень масштабирования", | ||||
|     "copy-without-formatting": "Копировать без форматирования", | ||||
|     "force-save-revision": "Принудительное сохранение версии" | ||||
|     "force-save-revision": "Принудительное сохранение версии", | ||||
|     "unhoist-note": "Открепить заметку", | ||||
|     "toggle-right-pane": "Переключить правую панель", | ||||
|     "print-active-note": "Печать активной заметки", | ||||
|     "render-active-note": "Рендеринг активной заметки", | ||||
|     "run-active-note": "Запуск активной заметки", | ||||
|     "add-include-note-to-text": "Добавить включение другой заметки в текст", | ||||
|     "toggle-ribbon-tab-basic-properties": "Переключить на вкладку \"Общее\"", | ||||
|     "toggle-ribbon-tab-book-properties": "Переключить на вкладку \"Свойства книги\"", | ||||
|     "toggle-ribbon-tab-file-properties": "Переключить на вкладку \"Свойства файла\"", | ||||
|     "toggle-ribbon-tab-image-properties": "Переключить на вкладку \"Свойства изображения\"", | ||||
|     "toggle-ribbon-tab-owned-attributes": "Переключить на вкладку \"Атрибуты\"", | ||||
|     "toggle-ribbon-tab-inherited-attributes": "Переключить на вкладку \"Унаследованные атрибуты\"", | ||||
|     "toggle-ribbon-tab-promoted-attributes": "Переключить на вкладку \"Продвигаемые атрибуты\"", | ||||
|     "toggle-ribbon-tab-note-map": "Переключить на вкладку \"Карта заметок\"", | ||||
|     "toggle-ribbon-tab-note-info": "Переключить на вкладку \"Информация о заметке\"", | ||||
|     "toggle-ribbon-tab-note-paths": "Переключить на вкладку \"Пути к заметке\"", | ||||
|     "toggle-ribbon-tab-similar-notes": "Переключить на вкладку \"Похожие заметки\"", | ||||
|     "export-active-note-as-pdf": "Экспортировать активную заметку в формате PDF", | ||||
|     "toggle-note-hoisting": "Переключить закрепление заметки" | ||||
|   }, | ||||
|   "months": { | ||||
|     "august": "Август", | ||||
| @@ -318,7 +343,8 @@ | ||||
|   }, | ||||
|   "notes": { | ||||
|     "duplicate-note-suffix": "(дубликат)", | ||||
|     "new-note": "Новая заметка" | ||||
|     "new-note": "Новая заметка", | ||||
|     "duplicate-note-title": "{{- noteTitle }} {{ duplicateNoteSuffix }}" | ||||
|   }, | ||||
|   "modals": { | ||||
|     "error_title": "Ошибка" | ||||
| @@ -328,7 +354,9 @@ | ||||
|     "subpages": "Подстраницы:", | ||||
|     "expand": "Развернуть", | ||||
|     "site-theme": "Тема оформления сайта", | ||||
|     "image_alt": "Изображение статьи" | ||||
|     "image_alt": "Изображение статьи", | ||||
|     "on-this-page": "На текущей странице", | ||||
|     "last-updated": "Последнее обновление: {{- date}}" | ||||
|   }, | ||||
|   "hidden_subtree_templates": { | ||||
|     "description": "Описание", | ||||
| @@ -359,12 +387,14 @@ | ||||
|   }, | ||||
|   "share_page": { | ||||
|     "child-notes": "Дочерние заметки:", | ||||
|     "no-content": "Эта заметка пуста." | ||||
|     "no-content": "Эта заметка пуста.", | ||||
|     "parent": "родитель:", | ||||
|     "clipped-from": "Эта заметка изначально была вырезана из {{- url}}" | ||||
|   }, | ||||
|   "javascript-required": "Для работы Trilium требуется JavaScript.", | ||||
|   "setup_sync-from-desktop": { | ||||
|     "heading": "Синхронизация с приложения ПК", | ||||
|     "description": "Эту настройку необходимо инициировать из приложения для ПК.", | ||||
|     "description": "Эту настройку необходимо инициировать из приложения для ПК:", | ||||
|     "step1": "Откройте приложение Trilium Notes на ПК.", | ||||
|     "step2": "В меню Trilium выберите «Параметры».", | ||||
|     "step3": "Нажмите на категорию «Синхронизация».", | ||||
| @@ -376,5 +406,23 @@ | ||||
|   "test_sync": { | ||||
|     "not-configured": "Адрес сервера синхронизации не установлен. Сначала настройте синхронизацию.", | ||||
|     "successful": "Установление связи с сервером синхронизации прошло успешно, синхронизация начата." | ||||
|   }, | ||||
|   "pdf": { | ||||
|     "export_filter": "Документ PDF (*.pdf)", | ||||
|     "unable-to-save-message": "Не удалось записать выбранный файл. Попробуйте ещё раз или выберите другой путь.", | ||||
|     "unable-to-export-message": "Текущую заметку невозможно экспортировать в формате PDF.", | ||||
|     "unable-to-export-title": "Невозможно экспортировать в PDF" | ||||
|   }, | ||||
|   "migration": { | ||||
|     "wrong_db_version": "Версия базы данных ({{version}}) новее, чем та, которую ожидает приложение ({{targetVersion}}). Это означает, что она была создана более новой и несовместимой версией Trilium. Для решения этой проблемы обновите Trilium до последней версии.", | ||||
|     "old_version": "Прямая миграция с текущей версии не поддерживается. Сначала обновитесь до последней версии 0.60.4, а затем — до этой.", | ||||
|     "error_message": "Ошибка при миграции на версию {{version}}: {{stack}}" | ||||
|   }, | ||||
|   "backend_log": { | ||||
|     "reading-log-failed": "Не удалось прочитать файл лога бэкенда '{{ fileName }}'.", | ||||
|     "log-does-not-exist": "Файл лога бэкенда '{{ fileName }}' (пока) не существует." | ||||
|   }, | ||||
|   "content_renderer": { | ||||
|     "note-cannot-be-displayed": "Этот тип заметки не может быть отображен." | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -349,7 +349,7 @@ | ||||
|     "etapi-title": "ETAPI", | ||||
|     "backup-title": "備份", | ||||
|     "sync-title": "同步", | ||||
|     "ai-llm-title": "AI / LLM", | ||||
|     "ai-llm-title": "AI/LLM", | ||||
|     "other": "其他", | ||||
|     "advanced-title": "進階", | ||||
|     "visible-launchers-title": "可見啟動器", | ||||
|   | ||||
| @@ -2,40 +2,40 @@ | ||||
|   "keyboard_actions": { | ||||
|     "back-in-note-history": "Перейти до попередньої нотатки в історії", | ||||
|     "forward-in-note-history": "Перейти до наступної нотатки в історії", | ||||
|     "open-command-palette": "Відкрити панель команд", | ||||
|     "scroll-to-active-note": "Прокрутити дерево нотаток до активної нотатки", | ||||
|     "quick-search": "Показати панель швидкого пошуку", | ||||
|     "open-command-palette": "Відкрити палітру команд", | ||||
|     "scroll-to-active-note": "Прокрутити дерево до активної нотатки", | ||||
|     "quick-search": "Активувати панель швидкого пошуку", | ||||
|     "search-in-subtree": "Пошук нотаток в піддереві активної нотатки", | ||||
|     "expand-subtree": "Розкрити піддерево поточної нотатки", | ||||
|     "expand-subtree": "Розгорнути піддерево поточної нотатки", | ||||
|     "collapse-tree": "Згорнути все дерево нотаток", | ||||
|     "open-jump-to-note-dialog": "Відкрити вікно \"Перейти до нотатки\"", | ||||
|     "open-jump-to-note-dialog": "Відкрити діалог \"Перейти до нотатки\"", | ||||
|     "collapse-subtree": "Згорнути піддерево поточної нотатки", | ||||
|     "sort-child-notes": "Сортувати дочірні нотатки", | ||||
|     "creating-and-moving-notes": "Створення та переміщення нотаток", | ||||
|     "create-note-after": "Створити нотатку після активної нотатки", | ||||
|     "create-note-into": "Створити нотатку як дочірній елемент активної нотатки", | ||||
|     "create-note-into-inbox": "Створити нотатку у «Вхідні» (якщо визначено) або в щоденнику", | ||||
|     "create-note-into": "Створити нотатку як дочірню до активної нотатки", | ||||
|     "create-note-into-inbox": "Створити нотатку у вхідні (якщо визначено) або денну нотатку", | ||||
|     "delete-note": "Видалити нотатку", | ||||
|     "move-note-up": "Перемістити нотатку вгору", | ||||
|     "move-note-down": "Перемістити нотатку вниз", | ||||
|     "move-note-up-in-hierarchy": "Перемістити нотатку вище в ієрархії", | ||||
|     "move-note-down-in-hierarchy": "Перемістити нотатку вниз в ієрархії", | ||||
|     "edit-note-title": "Перейти від дерева до деталей нотатки та редагувати заголовок", | ||||
|     "edit-branch-prefix": "Показати вікно \"Редагувати префікс гілки\"", | ||||
|     "edit-branch-prefix": "Показати діалог \"Редагувати префікс гілки\"", | ||||
|     "clone-notes-to": "Клонувати вибрані нотатки", | ||||
|     "move-notes-to": "Перемістити вибрані нотатки", | ||||
|     "note-clipboard": "Буфер обміну нотаток", | ||||
|     "note-clipboard": "Буфер обміну нотатки", | ||||
|     "copy-notes-to-clipboard": "Копіювати вибрані нотатки в буфер обміну", | ||||
|     "paste-notes-from-clipboard": "Вставити нотатки з буфера обміну в активну нотатку", | ||||
|     "paste-notes-from-clipboard": "Вставити нотатки з буферу обміну в активну нотатку", | ||||
|     "cut-notes-to-clipboard": "Вирізати вибрані нотатки в буфер обміну", | ||||
|     "select-all-notes-in-parent": "Вибрати всі нотатки з поточного рівня нотаток", | ||||
|     "add-note-above-to-the-selection": "Додати нотатку вище до виділення", | ||||
|     "add-note-above-to-the-selection": "Додати нотатку вище до вибраного", | ||||
|     "add-note-below-to-selection": "Додати нотатку нижче до вибраного", | ||||
|     "duplicate-subtree": "Дублікат гілок дерева", | ||||
|     "duplicate-subtree": "Дублювання піддерева", | ||||
|     "tabs-and-windows": "Вкладки & Вікна", | ||||
|     "open-new-tab": "Відкрити нову вкладку", | ||||
|     "close-active-tab": "Закрити активну вкладку", | ||||
|     "reopen-last-tab": "Знову відкрити останню закриту вкладку", | ||||
|     "reopen-last-tab": "Відкрити останню закриту вкладку", | ||||
|     "activate-next-tab": "Активувати вкладку праворуч", | ||||
|     "activate-previous-tab": "Активувати вкладку ліворуч", | ||||
|     "open-new-window": "Відкрити нове порожнє вікно", | ||||
| @@ -51,48 +51,48 @@ | ||||
|     "ninth-tab": "Активувати дев'яту вкладку у списку", | ||||
|     "last-tab": "Активувати останню вкладку у списку", | ||||
|     "dialogs": "Діалоги", | ||||
|     "show-note-source": "Показати вікно «Джерело нотатки»", | ||||
|     "show-options": "Відкрити \"Параметри\"", | ||||
|     "show-revisions": "Показати вікно \"Зміни нотаток\"", | ||||
|     "show-recent-changes": "Показати вікно \"Останні зміни\"", | ||||
|     "show-sql-console": "Відкрити \"Консоль SQL\"", | ||||
|     "show-backend-log": "Відкрити \"Backend Log\"", | ||||
|     "show-note-source": "Показати діалог \"Джерело нотатки\"", | ||||
|     "show-options": "Відкрити сторінку \"Параметри\"", | ||||
|     "show-revisions": "Показати діалог \"Версії нотаток\"", | ||||
|     "show-recent-changes": "Показати діалог \"Останні зміни\"", | ||||
|     "show-sql-console": "Відкрити сторінку \"Консоль SQL\"", | ||||
|     "show-backend-log": "Відкрити сторінку \"Backend Log\"", | ||||
|     "show-help": "Відкрити вбудований Посібник користувача", | ||||
|     "show-cheatsheet": "Показати вікно зі стандартними діями клавіатури", | ||||
|     "text-note-operations": "Дії з текстовими нотатками", | ||||
|     "add-link-to-text": "Відкрити діалогове вікно для додавання посилання до тексту", | ||||
|     "follow-link-under-cursor": "Перейдіть за посиланням, на якому знаходиться курсор", | ||||
|     "insert-date-and-time-to-text": "Вставити поточну дату та час у текст", | ||||
|     "add-link-to-text": "Відкрити діалог для додавання посилання до тексту", | ||||
|     "follow-link-under-cursor": "Перехід за посиланням, на якому знаходиться курсор", | ||||
|     "insert-date-and-time-to-text": "Вставити поточну дату & час у текст", | ||||
|     "paste-markdown-into-text": "Вставити Markdown з буфера обміну в текстову нотатку", | ||||
|     "cut-into-note": "Вирізати виділений фрагмент із поточної нотатки та створити піднотатку з виділеним текстом", | ||||
|     "add-include-note-to-text": "Відкрити вікно для додавання примітки", | ||||
|     "add-include-note-to-text": "Відкрити діалог для додавання нотатки", | ||||
|     "edit-readonly-note": "Редагувати нотатку, доступну тільки для читання", | ||||
|     "attributes-labels-and-relations": "Атрибути (мітки та зв'язки)", | ||||
|     "attributes-labels-and-relations": "Атрибути (мітки & зв'язки)", | ||||
|     "add-new-label": "Створити нову мітку", | ||||
|     "create-new-relation": "Створити новий зв'язок", | ||||
|     "ribbon-tabs": "Вкладки стрічки", | ||||
|     "toggle-basic-properties": "Увімкнути Основні властивості", | ||||
|     "toggle-file-properties": "Увімкнути Властивості файлу", | ||||
|     "toggle-image-properties": "Увімкнути Властивості зображення", | ||||
|     "toggle-owned-attributes": "Увімкнути Власні атрибути", | ||||
|     "toggle-inherited-attributes": "Увімкнути Успадковані атрибути", | ||||
|     "toggle-promoted-attributes": "Увімкнути Рекламні атрибути", | ||||
|     "toggle-link-map": "Увімкнути Карта посилань", | ||||
|     "toggle-note-info": "Увімкнути Інформація про нотатку", | ||||
|     "toggle-note-paths": "Увімкнути Шляхі нотатки", | ||||
|     "toggle-similar-notes": "Увімкнути Схожі нотатки", | ||||
|     "toggle-basic-properties": "Увімкнути Основні Властивості", | ||||
|     "toggle-file-properties": "Увімкнути Властивості Файлу", | ||||
|     "toggle-image-properties": "Увімкнути Властивості Зображення", | ||||
|     "toggle-owned-attributes": "Увімкнути Власні Атрибути", | ||||
|     "toggle-inherited-attributes": "Увімкнути Успадковані Атрибути", | ||||
|     "toggle-promoted-attributes": "Увімкнути Просунуті Атрибути", | ||||
|     "toggle-link-map": "Увімкнути Карта Посилань", | ||||
|     "toggle-note-info": "Увімкнути Інформація про Нотатку", | ||||
|     "toggle-note-paths": "Увімкнути Шляхи Нотатки", | ||||
|     "toggle-similar-notes": "Увімкнути Схожі Нотатки", | ||||
|     "other": "Інше", | ||||
|     "toggle-right-pane": "Увімкнути відображення правої панелі, яка містить Зміст та Основні моменти", | ||||
|     "print-active-note": "Друк активної нотатки", | ||||
|     "open-note-externally": "Відкрити нотатку як файл у програмі за замовчуванням", | ||||
|     "render-active-note": "Відтворити (повторно відтворити) активну нотатку", | ||||
|     "run-active-note": "Виконати активний код JavaScript (фронтенд/бекенд) нотатки", | ||||
|     "toggle-note-hoisting": "Увімкнути Хостинг активної нотатки", | ||||
|     "unhoist": "Зніміть з будь-якого місця", | ||||
|     "render-active-note": "Рендеринг (перерендерінг) активної нотатки", | ||||
|     "run-active-note": "Виконати активний код JavaScript (frontend/backend) нотатки з кодом", | ||||
|     "toggle-note-hoisting": "Увімкнути хостинг активної нотатки", | ||||
|     "unhoist": "Відкріпити з будь-якого місця", | ||||
|     "reload-frontend-app": "Перезавантажити інтерфейс", | ||||
|     "open-dev-tools": "Відкрити інструменти розробника", | ||||
|     "find-in-text": "Увімкнути панель пошуку", | ||||
|     "toggle-left-note-tree-panel": "Увімкнути ліву панель (дерево нотаток)", | ||||
|     "toggle-left-note-tree-panel": "Увімкнути ліву панель (дерево нотатки)", | ||||
|     "toggle-full-screen": "Увімкнути повноекранний режим", | ||||
|     "zoom-out": "Зменшити масштаб", | ||||
|     "zoom-in": "Збільшити масштаб", | ||||
| @@ -100,66 +100,107 @@ | ||||
|     "reset-zoom-level": "Скинути рівень масштабування", | ||||
|     "copy-without-formatting": "Копіювати виділений текст без форматування", | ||||
|     "force-save-revision": "Примусове створення/збереження нової версії активної нотатки", | ||||
|     "toggle-book-properties": "Увімкнути Властивості колекції", | ||||
|     "toggle-book-properties": "Увімкнути Властивості Колекції", | ||||
|     "toggle-classic-editor-toolbar": "Увімкнути вкладку Форматування для редактора з фіксованою панеллю інструментів", | ||||
|     "export-as-pdf": "Експортувати поточну нотатку у PDF", | ||||
|     "toggle-zen-mode": "Вмикає/вимикає дзен-режим (мінімальний інтерфейс для більш цілеспрямованого редагування)" | ||||
|     "export-as-pdf": "Експортувати поточну нотатку в PDF", | ||||
|     "toggle-zen-mode": "Вмикає/вимикає Дзен-режим (мінімальний інтерфейс сфокусованого редагування)" | ||||
|   }, | ||||
|   "keyboard_action_names": { | ||||
|     "back-in-note-history": "Назад до Історії нотаток", | ||||
|     "back-in-note-history": "Назад в Історії нотаток", | ||||
|     "forward-in-note-history": "Вперед в Історії нотаток", | ||||
|     "jump-to-note": "Перейти до...", | ||||
|     "command-palette": "Палітра команд", | ||||
|     "command-palette": "Палітра Команд", | ||||
|     "scroll-to-active-note": "Прокрутити до Активної нотатки", | ||||
|     "quick-search": "Швидкий пошук", | ||||
|     "search-in-subtree": "Пошук у піддереві", | ||||
|     "expand-subtree": "Розгорнути піддерево", | ||||
|     "collapse-tree": "Згорнути дерево", | ||||
|     "collapse-subtree": "Згорнути піддерево", | ||||
|     "sort-child-notes": "Сортувати дочірні нотатки", | ||||
|     "create-note-after": "Створити нотатку після", | ||||
|     "create-note-into": "Створити нотатку в", | ||||
|     "create-note-into-inbox": "Створити нотатку у \"Вхідні\"", | ||||
|     "delete-notes": "Видалити нотатки", | ||||
|     "move-note-up": "Перемістити нотатку вгору", | ||||
|     "move-note-down": "Перемістити нотатку вниз", | ||||
|     "move-note-up-in-hierarchy": "Перемістити нотатку вгору в ієрархії", | ||||
|     "move-note-down-in-hierarchy": "Перемістити нотатку вниз в ієрархії", | ||||
|     "edit-note-title": "Редагувати назву нотатки", | ||||
|     "edit-branch-prefix": "Редагувати префікс гілки", | ||||
|     "quick-search": "Швидкий Пошук", | ||||
|     "search-in-subtree": "Пошук у Піддереві", | ||||
|     "expand-subtree": "Розгорнути Піддерево", | ||||
|     "collapse-tree": "Згорнути Дерево", | ||||
|     "collapse-subtree": "Згорнути Піддерево", | ||||
|     "sort-child-notes": "Сортувати Дочірні нотатки", | ||||
|     "create-note-after": "Створити Нотатку після", | ||||
|     "create-note-into": "Створити Нотатку в", | ||||
|     "create-note-into-inbox": "Створити Нотатку у Вхідні", | ||||
|     "delete-notes": "Видалити Нотатки", | ||||
|     "move-note-up": "Перемістити Нотатку вгору", | ||||
|     "move-note-down": "Перемістити Нотатку вниз", | ||||
|     "move-note-up-in-hierarchy": "Перемістити Нотатку вгору в Ієрархії", | ||||
|     "move-note-down-in-hierarchy": "Перемістити Нотатку вниз в Ієрархії", | ||||
|     "edit-note-title": "Редагувати Заголовок нотатки", | ||||
|     "edit-branch-prefix": "Редагувати префікс Гілки", | ||||
|     "clone-notes-to": "Клонувати нотатки до", | ||||
|     "move-notes-to": "Перемістити нотатки до", | ||||
|     "copy-notes-to-clipboard": "Копіювати нотатки в буфер обміну", | ||||
|     "paste-notes-from-clipboard": "Вставити нотатки з буфера обміну", | ||||
|     "cut-notes-to-clipboard": "Вирізати нотатки в буфер обміну", | ||||
|     "copy-notes-to-clipboard": "Копіювати нотатки в Буфер обміну", | ||||
|     "paste-notes-from-clipboard": "Вставити нотатки з Буфера обміну", | ||||
|     "cut-notes-to-clipboard": "Вирізати нотатки в Буфер обміну", | ||||
|     "select-all-notes-in-parent": "Вибрати всі нотатки в Батьківські", | ||||
|     "add-note-above-to-selection": "Додати примітку вище до виділення", | ||||
|     "add-note-below-to-selection": "Додати нотатку нижче до виділення", | ||||
|     "duplicate-subtree": "Дублікат піддерева", | ||||
|     "open-new-tab": "Відкрити нову вкладку", | ||||
|     "add-note-above-to-selection": "Додати Нотатку вище до вибраного", | ||||
|     "add-note-below-to-selection": "Додати Нотатку нижче до вибраного", | ||||
|     "duplicate-subtree": "Дублікат Піддерева", | ||||
|     "open-new-tab": "Відкрити Нову вкладку", | ||||
|     "close-active-tab": "Закрити активну вкладку", | ||||
|     "reopen-last-tab": "Відкрити останню вкладку", | ||||
|     "activate-next-tab": "Активувати наступну вкладку", | ||||
|     "activate-previous-tab": "Активувати попередню вкладку", | ||||
|     "open-new-window": "Відкрити нове вікно", | ||||
|     "open-new-window": "Відкрити Нове вікно", | ||||
|     "toggle-system-tray-icon": "Увімкнути Значок системного трея", | ||||
|     "toggle-zen-mode": "Увімкнути режим дзен", | ||||
|     "toggle-zen-mode": "Увімкнути Дзен-режим", | ||||
|     "switch-to-first-tab": "Перейти до першої вкладки", | ||||
|     "switch-to-second-tab": "Перейти до другої вкладки", | ||||
|     "switch-to-third-tab": "Перейти до третьої вкладки", | ||||
|     "switch-to-fourth-tab": "Перейти до четвертої вкладки", | ||||
|     "switch-to-fifth-tab": "Перейти на п'яту вкладку", | ||||
|     "switch-to-fifth-tab": "Перейти до п'ятої вкладки", | ||||
|     "switch-to-sixth-tab": "Перейти до шостої вкладки", | ||||
|     "switch-to-seventh-tab": "Перейти до сьомої вкладки", | ||||
|     "switch-to-eighth-tab": "Перейти до восьмої вкладки", | ||||
|     "find-in-text": "Знайти в тексті", | ||||
|     "toggle-left-pane": "Перемкнути ліву панель", | ||||
|     "toggle-full-screen": "Перемкнути повноекранний режим", | ||||
|     "toggle-left-pane": "Увімкнути Ліву панель", | ||||
|     "toggle-full-screen": "Увімкнути Повноекранний режим", | ||||
|     "zoom-out": "Зменшити масштаб", | ||||
|     "zoom-in": "Збільшити масштаб", | ||||
|     "reset-zoom-level": "Скинути рівень масштабування", | ||||
|     "reset-zoom-level": "Скинути Масштабування", | ||||
|     "copy-without-formatting": "Копіювати без форматування", | ||||
|     "force-save-revision": "Примусове збереження версії" | ||||
|     "force-save-revision": "Примусове збереження версії", | ||||
|     "switch-to-ninth-tab": "Перейти до дев'ятої вкладки", | ||||
|     "switch-to-last-tab": "Перейти до останньої вкладки", | ||||
|     "show-note-source": "Показати Джерело нотатки", | ||||
|     "show-options": "Показати Параметри", | ||||
|     "show-revisions": "Показати Версії", | ||||
|     "show-recent-changes": "Показати Останні зміни", | ||||
|     "show-sql-console": "Показати Консоль SQL", | ||||
|     "show-backend-log": "Показати Backend Log", | ||||
|     "show-help": "Показати Допомогу", | ||||
|     "show-cheatsheet": "Показати Шпаргалку", | ||||
|     "add-link-to-text": "Додати Посилання до тексту", | ||||
|     "follow-link-under-cursor": "Перейти за Посиланням під курсором", | ||||
|     "insert-date-and-time-to-text": "Вставити Дату та Час у текст", | ||||
|     "paste-markdown-into-text": "Вставити Markdown у текст", | ||||
|     "cut-into-note": "Вирізати у нотатку", | ||||
|     "add-include-note-to-text": "Додати включену нотатку до тексту", | ||||
|     "edit-read-only-note": "Редагувати нотатку лише для читання", | ||||
|     "add-new-label": "Додати Нову мітку", | ||||
|     "add-new-relation": "Додати Новий зв'язок", | ||||
|     "toggle-ribbon-tab-classic-editor": "Включити Вкладку стрічки Класичний Редактор", | ||||
|     "toggle-ribbon-tab-basic-properties": "Включити Вкладку стрічки Основні властивості", | ||||
|     "toggle-ribbon-tab-book-properties": "Включити вкладку стрічки Властивості книги", | ||||
|     "toggle-ribbon-tab-file-properties": "Включити вкладку стрічки Властивості Файлу", | ||||
|     "toggle-ribbon-tab-image-properties": "Включити вкладку стрічки Властивості Зображення", | ||||
|     "toggle-ribbon-tab-owned-attributes": "Включити вкладку стрічки Власні Атрибути", | ||||
|     "toggle-ribbon-tab-inherited-attributes": "Включити вкладку стрічки Успадковані Атрибути", | ||||
|     "toggle-ribbon-tab-promoted-attributes": "Включити вкладку стрічки Просунуті Атрибути", | ||||
|     "toggle-ribbon-tab-note-map": "Включити вкладку стрічки Карта Нотатки", | ||||
|     "toggle-ribbon-tab-note-info": "Включити вкладку стрічки Інформація про Нотатку", | ||||
|     "toggle-ribbon-tab-note-paths": "Включити вкладку стрічки Шляхи Нотатки", | ||||
|     "toggle-ribbon-tab-similar-notes": "Включити вкладку стрічки Схожі Нотатки", | ||||
|     "toggle-right-pane": "Включити Праву панель", | ||||
|     "print-active-note": "Друк Активної Нотатки", | ||||
|     "export-active-note-as-pdf": "Експорт Активної нотатки у PDF", | ||||
|     "open-note-externally": "Відкрити Нотатку зовнішньою програмою", | ||||
|     "render-active-note": "Рендеринг Активної Нотатки", | ||||
|     "run-active-note": "Запустити Активну Нотатку", | ||||
|     "toggle-note-hoisting": "Включити хостинг нотатки", | ||||
|     "reload-frontend-app": "Перезавантажити Інтерфейс програми", | ||||
|     "open-developer-tools": "Відкрити Інструменти розробника", | ||||
|     "unhoist-note": "Відкріпити Нотатку" | ||||
|   }, | ||||
|   "login": { | ||||
|     "title": "Увійти", | ||||
| @@ -172,6 +213,216 @@ | ||||
|     "sign_in_with_sso": "Увійти за допомогою {{ ssoIssuerName }}" | ||||
|   }, | ||||
|   "set_password": { | ||||
|     "title": "Встановити пароль" | ||||
|     "title": "Встановити Пароль", | ||||
|     "heading": "Встановити пароль", | ||||
|     "description": "Перш ніж почати користуватися Trilium з веб-сайту, вам потрібно спочатку встановити пароль. Потім ви будете використовувати цей пароль для входу в систему.", | ||||
|     "password": "Пароль", | ||||
|     "password-confirmation": "Підтвердження пароля", | ||||
|     "button": "Встановити пароль" | ||||
|   }, | ||||
|   "javascript-required": "Для роботи Trilium потрібен JavaScript.", | ||||
|   "setup": { | ||||
|     "heading": "Налаштування Trilium Notes", | ||||
|     "new-document": "Я новий користувач і хочу створити новий документ Trilium для своїх нотаток", | ||||
|     "sync-from-desktop": "У мене вже є екземпляр для ПК і я хочу налаштувати синхронізацію з ним", | ||||
|     "sync-from-server": "У мене вже є екземпляр сервера, і я хочу налаштувати синхронізацію з ним", | ||||
|     "next": "Наступна", | ||||
|     "init-in-progress": "Триває ініціалізація документа", | ||||
|     "redirecting": "Невдовзі вас буде перенаправлено до програми.", | ||||
|     "title": "Налаштування" | ||||
|   }, | ||||
|   "setup_sync-from-desktop": { | ||||
|     "heading": "Синхронізація з ПК", | ||||
|     "description": "Це налаштування потрібно ініціювати з екземпляра ПК:", | ||||
|     "step1": "Відкрийте екземпляр Trilium Notes на ПК.", | ||||
|     "step2": "У меню Trilium натисніть Параметри.", | ||||
|     "step3": "Натисніть на Категорія Синхронізації.", | ||||
|     "step4": "Змініть адресу екземпляра сервера на: {{- host}} та натисніть кнопку Зберегти.", | ||||
|     "step5": "Натисніть \"Тест синхронізації\", щоб перевірити успішність підключення.", | ||||
|     "step6": "Після виконання цих кроків натисніть {{- link}}.", | ||||
|     "step6-here": "тут" | ||||
|   }, | ||||
|   "setup_sync-from-server": { | ||||
|     "heading": "Синхронізація з сервера", | ||||
|     "instructions": "Будь ласка, введіть адресу сервера Trilium та облікові дані нижче. Це завантажить весь документ Trilium із сервера та налаштує синхронізацію з ним. Залежно від розміру документа та швидкості вашого з’єднання, це може зайняти деякий час.", | ||||
|     "server-host": "Адреса сервера Trilium", | ||||
|     "server-host-placeholder": "https://<hostname>:<port>", | ||||
|     "proxy-server": "Проксі-сервер (необов'язково)", | ||||
|     "proxy-server-placeholder": "https://<hostname>:<port>", | ||||
|     "note": "Нотатка:", | ||||
|     "proxy-instruction": "Якщо залишити налаштування проксі-сервера порожнім, використовуватиметься системний проксі-сервер (стосується лише ПК програми)", | ||||
|     "password": "Пароль", | ||||
|     "password-placeholder": "Пароль", | ||||
|     "back": "Назад", | ||||
|     "finish-setup": "Завершити налаштування" | ||||
|   }, | ||||
|   "setup_sync-in-progress": { | ||||
|     "heading": "Триває синхронізація", | ||||
|     "successful": "Синхронізацію налаштовано правильно. Початкова синхронізація завершиться через деякий час. Після її завершення вас буде перенаправлено на сторінку входу.", | ||||
|     "outstanding-items": "Незавершені елементи синхронізації:", | ||||
|     "outstanding-items-default": "Недоступно" | ||||
|   }, | ||||
|   "share_404": { | ||||
|     "title": "Не знайдено", | ||||
|     "heading": "Не знайдено" | ||||
|   }, | ||||
|   "share_page": { | ||||
|     "parent": "батьківська:", | ||||
|     "clipped-from": "Цю нотатку було спочатку вирізано з {{- url}}", | ||||
|     "child-notes": "Дочірні нотатки:", | ||||
|     "no-content": "Ця нотатка не має вмісту." | ||||
|   }, | ||||
|   "weekdays": { | ||||
|     "monday": "Понеділок", | ||||
|     "tuesday": "Вівторок", | ||||
|     "wednesday": "Середа", | ||||
|     "thursday": "Четвер", | ||||
|     "friday": "П'ятниця", | ||||
|     "saturday": "Субота", | ||||
|     "sunday": "Неділя" | ||||
|   }, | ||||
|   "weekdayNumber": "Тиждень {weekNumber}", | ||||
|   "months": { | ||||
|     "january": "Січень", | ||||
|     "february": "Лютий", | ||||
|     "march": "Березень", | ||||
|     "april": "Квітень", | ||||
|     "may": "Травень", | ||||
|     "june": "Червень", | ||||
|     "july": "Липень", | ||||
|     "august": "Серпень", | ||||
|     "september": "Вересень", | ||||
|     "october": "Жовтень", | ||||
|     "november": "Листопад", | ||||
|     "december": "Грудень" | ||||
|   }, | ||||
|   "quarterNumber": "Квартал {quarterNumber}", | ||||
|   "special_notes": { | ||||
|     "search_prefix": "Пошук:" | ||||
|   }, | ||||
|   "test_sync": { | ||||
|     "not-configured": "Хост сервера синхронізації не налаштовано. Спочатку налаштуйте синхронізацію.", | ||||
|     "successful": "Установлення зв'язку з сервером синхронізації пройшло успішно, синхронізація розпочалася." | ||||
|   }, | ||||
|   "hidden-subtree": { | ||||
|     "root-title": "Приховані Нотатки", | ||||
|     "search-history-title": "Історія пошуку", | ||||
|     "note-map-title": "Карта Нотатки", | ||||
|     "sql-console-history-title": "Історія консолі SQL", | ||||
|     "shared-notes-title": "Спільні Нотатки", | ||||
|     "bulk-action-title": "Масова дія", | ||||
|     "backend-log-title": "Backend Log", | ||||
|     "user-hidden-title": "Прихований користувач", | ||||
|     "launch-bar-templates-title": "Запуск Шаблони панелей", | ||||
|     "base-abstract-launcher-title": "Базовий Лаунчер Abstract", | ||||
|     "command-launcher-title": "Лаунчер Command", | ||||
|     "note-launcher-title": "Лаунчер Note", | ||||
|     "script-launcher-title": "Лаунчер Script", | ||||
|     "built-in-widget-title": "Вбудований віджет", | ||||
|     "custom-widget-title": "Користувацький віджет", | ||||
|     "launch-bar-title": "Панель запуску", | ||||
|     "available-launchers-title": "Доступні Лаунчери", | ||||
|     "go-to-previous-note-title": "Перейти до попередньої нотатки", | ||||
|     "go-to-next-note-title": "Перейти до наступної нотатки", | ||||
|     "new-note-title": "Нова Нотатка", | ||||
|     "search-notes-title": "Пошук нотаток", | ||||
|     "jump-to-note-title": "Перейти до...", | ||||
|     "calendar-title": "Календар", | ||||
|     "recent-changes-title": "Останні Зміни", | ||||
|     "bookmarks-title": "Закладки", | ||||
|     "open-today-journal-note-title": "Відкрити Щоденник за Сьогодні", | ||||
|     "quick-search-title": "Швидкий Пошук", | ||||
|     "protected-session-title": "Захищений Сеанс", | ||||
|     "sync-status-title": "Статус синхронізації", | ||||
|     "settings-title": "Налаштування", | ||||
|     "llm-chat-title": "Чат з Нотатками", | ||||
|     "options-title": "Параметри", | ||||
|     "appearance-title": "Зовнішній вигляд", | ||||
|     "shortcuts-title": "Комбінації клавіші", | ||||
|     "text-notes": "Текстові Нотатки", | ||||
|     "code-notes-title": "Нотатка з кодом", | ||||
|     "images-title": "Зображення", | ||||
|     "spellcheck-title": "Перевірка Орфографії", | ||||
|     "password-title": "Пароль", | ||||
|     "multi-factor-authentication-title": "MFA", | ||||
|     "etapi-title": "ETAPI", | ||||
|     "backup-title": "Резервне копіювання", | ||||
|     "sync-title": "Синхронізація", | ||||
|     "ai-llm-title": "AI/LLM", | ||||
|     "other": "Інше", | ||||
|     "advanced-title": "Розширені", | ||||
|     "visible-launchers-title": "Видимі Лаунчери", | ||||
|     "user-guide": "Посібник користувача", | ||||
|     "localization": "Мова & Регіон", | ||||
|     "inbox-title": "Вхідні", | ||||
|     "spacer-title": "Роздільник" | ||||
|   }, | ||||
|   "notes": { | ||||
|     "new-note": "Нова нотатка", | ||||
|     "duplicate-note-suffix": "(дублікат)", | ||||
|     "duplicate-note-title": "{{- noteTitle }} {{ duplicateNoteSuffix }}" | ||||
|   }, | ||||
|   "backend_log": { | ||||
|     "log-does-not-exist": "Файл backend log '{{ fileName }}' не існує.", | ||||
|     "reading-log-failed": "Не вдалося прочитати backend log file {{fileName}}." | ||||
|   }, | ||||
|   "content_renderer": { | ||||
|     "note-cannot-be-displayed": "Цей тип нотатки не може бути відображений." | ||||
|   }, | ||||
|   "pdf": { | ||||
|     "export_filter": "PDF Document (*.pdf)", | ||||
|     "unable-to-export-message": "Поточну нотатку не вдалося експортувати у PDF.", | ||||
|     "unable-to-export-title": "Не вдається експортувати у PDF", | ||||
|     "unable-to-save-message": "Не вдалося записати вибраний файл. Спробуйте ще раз або виберіть інше місце призначення." | ||||
|   }, | ||||
|   "tray": { | ||||
|     "tooltip": "Trilium Notes", | ||||
|     "close": "Вихід з Trilium", | ||||
|     "recents": "Останні нотатки", | ||||
|     "bookmarks": "Закладки", | ||||
|     "today": "Відкрити щоденник за сьогодні", | ||||
|     "new-note": "Нова нотатка", | ||||
|     "show-windows": "Показати вікна", | ||||
|     "open_new_window": "Відкрити нове вікно" | ||||
|   }, | ||||
|   "migration": { | ||||
|     "old_version": "Пряма міграція з вашої поточної версії не підтримується. Будь ласка, спочатку оновіть систему до останньої версії v0.60.4, а потім лише потім до цієї.", | ||||
|     "error_message": "Помилка під час міграції до версії {{version}}: {{stack}}", | ||||
|     "wrong_db_version": "Версія бази даних ({{version}}) новіша за ту, яку очікує програма ({{targetVersion}}), це означає, що її було створено новішою та несумісною версією Trilium. Оновіть Trilium до останньої версії, щоб вирішити цю проблему." | ||||
|   }, | ||||
|   "modals": { | ||||
|     "error_title": "Помилка" | ||||
|   }, | ||||
|   "share_theme": { | ||||
|     "site-theme": "Тема сайту", | ||||
|     "search_placeholder": "Пошук...", | ||||
|     "image_alt": "Зображення до статті", | ||||
|     "last-updated": "Останнє оновлення: {{- date}}", | ||||
|     "subpages": "Підсторінки:", | ||||
|     "on-this-page": "На цій сторінці", | ||||
|     "expand": "Розгорнути" | ||||
|   }, | ||||
|   "hidden_subtree_templates": { | ||||
|     "text-snippet": "Фрагмент тексту", | ||||
|     "description": "Опис", | ||||
|     "list-view": "Список", | ||||
|     "grid-view": "Сітка", | ||||
|     "calendar": "Календар", | ||||
|     "table": "Таблиця", | ||||
|     "geo-map": "Географічна карта", | ||||
|     "start-date": "Дата початку", | ||||
|     "end-date": "Дата завершення", | ||||
|     "start-time": "Час початку", | ||||
|     "end-time": "Час завершення", | ||||
|     "geolocation": "Геолокація", | ||||
|     "built-in-templates": "Вбудовані шаблони", | ||||
|     "board": "Дошка", | ||||
|     "status": "Статус", | ||||
|     "board_note_first": "Перша нотатка", | ||||
|     "board_note_second": "Друга нотатка", | ||||
|     "board_note_third": "Третя нотатка", | ||||
|     "board_status_todo": "Зробити", | ||||
|     "board_status_progress": "У процесі", | ||||
|     "board_status_done": "Готово" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
|     <link rel="manifest" crossorigin="use-credentials" href="manifest.webmanifest"> | ||||
|     <title>Trilium Notes</title> | ||||
| </head> | ||||
| <body class="desktop heading-style-<%= headingStyle %> layout-<%= layoutOrientation %> platform-<%= platform %> <%= isElectron ? 'electron' : '' %> <%= hasNativeTitleBar ? 'native-titlebar' : '' %> <%= hasBackgroundEffects ? 'background-effects' : '' %>"> | ||||
| <body id="trilium-app" class="desktop heading-style-<%= headingStyle %> layout-<%= layoutOrientation %> platform-<%= platform %> <%= isElectron ? 'electron' : '' %> <%= hasNativeTitleBar ? 'native-titlebar' : '' %> <%= hasBackgroundEffects ? 'background-effects' : '' %>"> | ||||
| <noscript><%= t("javascript-required") %></noscript> | ||||
|  | ||||
| <script> | ||||
|   | ||||
							
								
								
									
										388
									
								
								apps/server/src/routes/api/ckeditor_plugins.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										388
									
								
								apps/server/src/routes/api/ckeditor_plugins.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,388 @@ | ||||
| /** | ||||
|  * @module CKEditor Plugins API | ||||
|  *  | ||||
|  * This module provides REST endpoints for managing CKEditor plugin configuration | ||||
|  * in Trilium Notes. It handles plugin enablement, validation, and user preferences. | ||||
|  */ | ||||
|  | ||||
| import options from "../../services/options.js"; | ||||
| import type { Request, Response } from "express"; | ||||
| import type {  | ||||
|     PluginConfiguration,  | ||||
|     PluginRegistry,  | ||||
|     PluginValidationResult,  | ||||
|     UpdatePluginConfigRequest, | ||||
|     UpdatePluginConfigResponse, | ||||
|     QueryPluginsOptions, | ||||
|     QueryPluginsResult, | ||||
|     PluginMetadata | ||||
| } from "@triliumnext/commons"; | ||||
| import { PLUGIN_REGISTRY, getPluginMetadata, getPluginsByCategory, getConfigurablePlugins } from "@triliumnext/ckeditor5"; | ||||
| import log from "../../services/log.js"; | ||||
|  | ||||
| /** | ||||
|  * Get the complete plugin registry with metadata | ||||
|  */ | ||||
| export function getPluginRegistry(): PluginRegistry { | ||||
|     return PLUGIN_REGISTRY; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get current user's plugin configuration | ||||
|  */ | ||||
| export function getUserPluginConfig(): PluginConfiguration[] { | ||||
|     const enabledPluginsJson = options.getOptionOrNull("ckeditorEnabledPlugins"); | ||||
|      | ||||
|     if (!enabledPluginsJson) { | ||||
|         // Return default configuration if none exists | ||||
|         return getDefaultPluginConfiguration(); | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|         const enabledPlugins = JSON.parse(enabledPluginsJson) as string[]; | ||||
|          | ||||
|         // Convert to PluginConfiguration array | ||||
|         return Object.keys(PLUGIN_REGISTRY.plugins).map(pluginId => ({ | ||||
|             id: pluginId, | ||||
|             enabled: enabledPlugins.includes(pluginId) | ||||
|         })); | ||||
|     } catch (error) { | ||||
|         log.error(`Failed to parse CKEditor plugin configuration: ${error}`); | ||||
|         return getDefaultPluginConfiguration(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get default plugin configuration (all non-premium plugins enabled) | ||||
|  */ | ||||
| function getDefaultPluginConfiguration(): PluginConfiguration[] { | ||||
|     return Object.values(PLUGIN_REGISTRY.plugins).map(plugin => ({ | ||||
|         id: plugin.id, | ||||
|         enabled: plugin.defaultEnabled && !plugin.requiresPremium | ||||
|     })); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Update user's plugin configuration | ||||
|  */ | ||||
| export function updateUserPluginConfig(request: UpdatePluginConfigRequest): UpdatePluginConfigResponse { | ||||
|     const { plugins, validate = true } = request; | ||||
|      | ||||
|     try { | ||||
|         // Validate if requested | ||||
|         let validation: PluginValidationResult | undefined; | ||||
|         if (validate) { | ||||
|             validation = validatePluginConfiguration(plugins); | ||||
|             if (!validation.valid) { | ||||
|                 return { | ||||
|                     success: false, | ||||
|                     validation, | ||||
|                     plugins: [], | ||||
|                     errors: validation.errors.map(err => err.message) | ||||
|                 }; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Save configuration | ||||
|         const enabledPluginIds = plugins | ||||
|             .filter(plugin => plugin.enabled) | ||||
|             .map(plugin => plugin.id); | ||||
|  | ||||
|         options.setOption("ckeditorEnabledPlugins", JSON.stringify(enabledPluginIds)); | ||||
|  | ||||
|         log.info(`Updated CKEditor plugin configuration: ${enabledPluginIds.length} plugins enabled`); | ||||
|  | ||||
|         return { | ||||
|             success: true, | ||||
|             validation, | ||||
|             plugins: plugins, | ||||
|             errors: [] | ||||
|         }; | ||||
|     } catch (error) { | ||||
|         log.error(`Failed to update CKEditor plugin configuration: ${error}`); | ||||
|         return { | ||||
|             success: false, | ||||
|             plugins: [], | ||||
|             errors: [`Failed to update configuration: ${error}`] | ||||
|         }; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Query plugins with filtering options | ||||
|  */ | ||||
| export function queryPlugins(options: QueryPluginsOptions = {}): QueryPluginsResult { | ||||
|     const { category, enabled, coreOnly, includeConfig } = options; | ||||
|      | ||||
|     let plugins = Object.values(PLUGIN_REGISTRY.plugins); | ||||
|  | ||||
|     // Apply filters | ||||
|     if (category) { | ||||
|         plugins = plugins.filter(plugin => plugin.category === category); | ||||
|     } | ||||
|  | ||||
|     if (coreOnly === true) { | ||||
|         plugins = plugins.filter(plugin => plugin.isCore); | ||||
|     } else if (coreOnly === false) { | ||||
|         plugins = plugins.filter(plugin => !plugin.isCore); | ||||
|     } | ||||
|  | ||||
|     // Get user configuration if requested or filtering by enabled status | ||||
|     let userConfig: PluginConfiguration[] = []; | ||||
|     if (includeConfig || enabled !== undefined) { | ||||
|         userConfig = getUserPluginConfig(); | ||||
|     } | ||||
|  | ||||
|     // Filter by enabled status | ||||
|     if (enabled !== undefined) { | ||||
|         const enabledPluginIds = new Set( | ||||
|             userConfig.filter(config => config.enabled).map(config => config.id) | ||||
|         ); | ||||
|         plugins = plugins.filter(plugin =>  | ||||
|             enabled ? enabledPluginIds.has(plugin.id) : !enabledPluginIds.has(plugin.id) | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     // Add user configuration if requested | ||||
|     const result = plugins.map(plugin => { | ||||
|         if (includeConfig) { | ||||
|             const config = userConfig.find(config => config.id === plugin.id); | ||||
|             return { | ||||
|                 ...plugin, | ||||
|                 enabled: config?.enabled ?? false, | ||||
|                 config: config?.config | ||||
|             }; | ||||
|         } | ||||
|         return plugin; | ||||
|     }); | ||||
|  | ||||
|     // Get available categories | ||||
|     const categories = [...new Set(Object.values(PLUGIN_REGISTRY.plugins).map(plugin => plugin.category))]; | ||||
|  | ||||
|     return { | ||||
|         plugins: result, | ||||
|         totalCount: Object.keys(PLUGIN_REGISTRY.plugins).length, | ||||
|         categories: categories as any[] | ||||
|     }; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Validate plugin configuration for dependencies and conflicts | ||||
|  */ | ||||
| export function validatePluginConfiguration(plugins: PluginConfiguration[]): PluginValidationResult { | ||||
|     const errors: any[] = []; | ||||
|     const warnings: any[] = []; | ||||
|     const enabledPlugins = new Set(plugins.filter(p => p.enabled).map(p => p.id)); | ||||
|     const resolvedPlugins = new Set<string>(); | ||||
|  | ||||
|     // Check each enabled plugin | ||||
|     for (const plugin of plugins.filter(p => p.enabled)) { | ||||
|         const metadata = getPluginMetadata(plugin.id); | ||||
|          | ||||
|         if (!metadata) { | ||||
|             errors.push({ | ||||
|                 type: "missing_dependency", | ||||
|                 pluginId: plugin.id, | ||||
|                 message: `Plugin '${plugin.id}' not found in registry` | ||||
|             }); | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         // Check premium requirements | ||||
|         if (metadata.requiresPremium && !hasPremiumLicense()) { | ||||
|             errors.push({ | ||||
|                 type: "premium_required", | ||||
|                 pluginId: plugin.id, | ||||
|                 message: `Plugin '${metadata.name}' requires a premium CKEditor license` | ||||
|             }); | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         // Check dependencies | ||||
|         for (const depId of metadata.dependencies) { | ||||
|             if (!enabledPlugins.has(depId)) { | ||||
|                 const depMetadata = getPluginMetadata(depId); | ||||
|                 errors.push({ | ||||
|                     type: "missing_dependency", | ||||
|                     pluginId: plugin.id, | ||||
|                     message: `Plugin '${metadata.name}' requires '${depMetadata?.name || depId}' to be enabled`, | ||||
|                     details: { dependency: depId } | ||||
|                 }); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Check conflicts | ||||
|         for (const conflictId of metadata.conflicts) { | ||||
|             if (enabledPlugins.has(conflictId)) { | ||||
|                 const conflictMetadata = getPluginMetadata(conflictId); | ||||
|                 errors.push({ | ||||
|                     type: "plugin_conflict", | ||||
|                     pluginId: plugin.id, | ||||
|                     message: `Plugin '${metadata.name}' conflicts with '${conflictMetadata?.name || conflictId}'`, | ||||
|                     details: { conflict: conflictId } | ||||
|                 }); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         resolvedPlugins.add(plugin.id); | ||||
|     } | ||||
|  | ||||
|     // Add core plugins to resolved list (they're always enabled) | ||||
|     for (const plugin of Object.values(PLUGIN_REGISTRY.plugins)) { | ||||
|         if (plugin.isCore) { | ||||
|             resolvedPlugins.add(plugin.id); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Check for circular dependencies (simplified check) | ||||
|     const visited = new Set<string>(); | ||||
|     const visiting = new Set<string>(); | ||||
|  | ||||
|     function hasCircularDependency(pluginId: string): boolean { | ||||
|         if (visiting.has(pluginId)) { | ||||
|             return true; | ||||
|         } | ||||
|         if (visited.has(pluginId)) { | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         visiting.add(pluginId); | ||||
|         const metadata = getPluginMetadata(pluginId); | ||||
|          | ||||
|         if (metadata) { | ||||
|             for (const depId of metadata.dependencies) { | ||||
|                 if (enabledPlugins.has(depId) && hasCircularDependency(depId)) { | ||||
|                     return true; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         visiting.delete(pluginId); | ||||
|         visited.add(pluginId); | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     for (const pluginId of enabledPlugins) { | ||||
|         if (hasCircularDependency(pluginId)) { | ||||
|             errors.push({ | ||||
|                 type: "circular_dependency", | ||||
|                 pluginId: pluginId, | ||||
|                 message: `Circular dependency detected for plugin '${getPluginMetadata(pluginId)?.name || pluginId}'` | ||||
|             }); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return { | ||||
|         valid: errors.length === 0, | ||||
|         errors, | ||||
|         warnings, | ||||
|         resolvedPlugins: Array.from(resolvedPlugins) | ||||
|     }; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Check if user has premium CKEditor license | ||||
|  */ | ||||
| function hasPremiumLicense(): boolean { | ||||
|     // This would check the actual license key | ||||
|     // For now, assume no premium license | ||||
|     return process.env.VITE_CKEDITOR_KEY !== undefined && process.env.VITE_CKEDITOR_KEY !== ""; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Reset plugin configuration to defaults | ||||
|  */ | ||||
| export function resetPluginConfigToDefaults(): UpdatePluginConfigResponse { | ||||
|     const defaultConfig = getDefaultPluginConfiguration(); | ||||
|      | ||||
|     return updateUserPluginConfig({ | ||||
|         plugins: defaultConfig, | ||||
|         validate: false | ||||
|     }); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get plugin statistics | ||||
|  */ | ||||
| export function getPluginStats() { | ||||
|     const userConfig = getUserPluginConfig(); | ||||
|     const enabledCount = userConfig.filter(p => p.enabled).length; | ||||
|     const totalCount = Object.keys(PLUGIN_REGISTRY.plugins).length; | ||||
|     const coreCount = Object.values(PLUGIN_REGISTRY.plugins).filter(p => p.isCore).length; | ||||
|     const premiumCount = Object.values(PLUGIN_REGISTRY.plugins).filter(p => p.requiresPremium).length; | ||||
|  | ||||
|     const categoryCounts = Object.values(PLUGIN_REGISTRY.plugins).reduce((acc, plugin) => { | ||||
|         acc[plugin.category] = (acc[plugin.category] || 0) + 1; | ||||
|         return acc; | ||||
|     }, {} as Record<string, number>); | ||||
|  | ||||
|     return { | ||||
|         enabled: enabledCount, | ||||
|         total: totalCount, | ||||
|         core: coreCount, | ||||
|         premium: premiumCount, | ||||
|         configurable: totalCount - coreCount, | ||||
|         categories: categoryCounts, | ||||
|         hasPremiumLicense: hasPremiumLicense() | ||||
|     }; | ||||
| } | ||||
|  | ||||
| // Express route handlers | ||||
| function getPluginRegistryHandler(req: Request, res: Response) { | ||||
|     res.json(getPluginRegistry()); | ||||
| } | ||||
|  | ||||
| function getUserPluginConfigHandler(req: Request, res: Response) { | ||||
|     res.json(getUserPluginConfig()); | ||||
| } | ||||
|  | ||||
| function updateUserPluginConfigHandler(req: Request, res: Response) { | ||||
|     const updateRequest: UpdatePluginConfigRequest = req.body; | ||||
|     const result = updateUserPluginConfig(updateRequest); | ||||
|      | ||||
|     if (!result.success) { | ||||
|         res.status(400).json(result); | ||||
|     } else { | ||||
|         res.json(result); | ||||
|     } | ||||
| } | ||||
|  | ||||
| function queryPluginsHandler(req: Request, res: Response) { | ||||
|     const queryOptions: QueryPluginsOptions = { | ||||
|         category: req.query.category as string, | ||||
|         enabled: req.query.enabled === 'true' ? true : req.query.enabled === 'false' ? false : undefined, | ||||
|         coreOnly: req.query.coreOnly === 'true' ? true : req.query.coreOnly === 'false' ? false : undefined, | ||||
|         includeConfig: req.query.includeConfig === 'true' | ||||
|     }; | ||||
|      | ||||
|     res.json(queryPlugins(queryOptions)); | ||||
| } | ||||
|  | ||||
| function validatePluginConfigurationHandler(req: Request, res: Response) { | ||||
|     const plugins: PluginConfiguration[] = req.body.plugins || []; | ||||
|     res.json(validatePluginConfiguration(plugins)); | ||||
| } | ||||
|  | ||||
| function resetPluginConfigToDefaultsHandler(req: Request, res: Response) { | ||||
|     const result = resetPluginConfigToDefaults(); | ||||
|      | ||||
|     if (!result.success) { | ||||
|         res.status(400).json(result); | ||||
|     } else { | ||||
|         res.json(result); | ||||
|     } | ||||
| } | ||||
|  | ||||
| function getPluginStatsHandler(req: Request, res: Response) { | ||||
|     res.json(getPluginStats()); | ||||
| } | ||||
|  | ||||
| export default { | ||||
|     getPluginRegistry: getPluginRegistryHandler, | ||||
|     getUserPluginConfig: getUserPluginConfigHandler, | ||||
|     updateUserPluginConfig: updateUserPluginConfigHandler, | ||||
|     queryPlugins: queryPluginsHandler, | ||||
|     validatePluginConfiguration: validatePluginConfigurationHandler, | ||||
|     resetPluginConfigToDefaults: resetPluginConfigToDefaultsHandler, | ||||
|     getPluginStats: getPluginStatsHandler | ||||
| }; | ||||
| @@ -63,6 +63,9 @@ const ALLOWED_OPTIONS = new Set<OptionNames>([ | ||||
|     "dailyBackupEnabled", | ||||
|     "weeklyBackupEnabled", | ||||
|     "monthlyBackupEnabled", | ||||
|     "motionEnabled", | ||||
|     "shadowsEnabled", | ||||
|     "backdropEffectsEnabled", | ||||
|     "maxContentWidth", | ||||
|     "compressImages", | ||||
|     "downloadImagesAutomatically", | ||||
| @@ -94,6 +97,7 @@ const ALLOWED_OPTIONS = new Set<OptionNames>([ | ||||
|     "showLoginInShareTheme", | ||||
|     "splitEditorOrientation", | ||||
|     "seenCallToActions", | ||||
|     "ckeditorEnabledPlugins", | ||||
|  | ||||
|     // AI/LLM integration options | ||||
|     "aiEnabled", | ||||
|   | ||||
| @@ -59,6 +59,7 @@ import openaiRoute from "./api/openai.js"; | ||||
| import anthropicRoute from "./api/anthropic.js"; | ||||
| import llmRoute from "./api/llm.js"; | ||||
| import systemInfoRoute from "./api/system_info.js"; | ||||
| import ckeditorPluginsRoute from "./api/ckeditor_plugins.js"; | ||||
|  | ||||
| import etapiAuthRoutes from "../etapi/auth.js"; | ||||
| import etapiAppInfoRoutes from "../etapi/app_info.js"; | ||||
| @@ -220,6 +221,15 @@ function register(app: express.Application) { | ||||
|     apiRoute(PUT, "/api/options", optionsApiRoute.updateOptions); | ||||
|     apiRoute(GET, "/api/options/user-themes", optionsApiRoute.getUserThemes); | ||||
|     apiRoute(GET, "/api/options/locales", optionsApiRoute.getSupportedLocales); | ||||
|      | ||||
|     // CKEditor plugins management | ||||
|     apiRoute(GET, "/api/ckeditor-plugins/registry", ckeditorPluginsRoute.getPluginRegistry); | ||||
|     apiRoute(GET, "/api/ckeditor-plugins/config", ckeditorPluginsRoute.getUserPluginConfig); | ||||
|     apiRoute(PUT, "/api/ckeditor-plugins/config", ckeditorPluginsRoute.updateUserPluginConfig); | ||||
|     apiRoute(GET, "/api/ckeditor-plugins/query", ckeditorPluginsRoute.queryPlugins); | ||||
|     apiRoute(PST, "/api/ckeditor-plugins/validate", ckeditorPluginsRoute.validatePluginConfiguration); | ||||
|     apiRoute(PST, "/api/ckeditor-plugins/reset", ckeditorPluginsRoute.resetPluginConfigToDefaults); | ||||
|     apiRoute(GET, "/api/ckeditor-plugins/stats", ckeditorPluginsRoute.getPluginStats); | ||||
|  | ||||
|     apiRoute(PST, "/api/password/change", passwordApiRoute.changePassword); | ||||
|     apiRoute(PST, "/api/password/reset", passwordApiRoute.resetPassword); | ||||
|   | ||||
| @@ -16,11 +16,13 @@ const DAYJS_LOADER: Record<LOCALE_IDS, () => Promise<typeof import("dayjs/locale | ||||
|     "fa": () => import("dayjs/locale/fa.js"), | ||||
|     "fr": () => import("dayjs/locale/fr.js"), | ||||
|     "he": () => import("dayjs/locale/he.js"), | ||||
|     "ja": () => import("dayjs/locale/ja.js"), | ||||
|     "ku": () => import("dayjs/locale/ku.js"), | ||||
|     "pt_br": () => import("dayjs/locale/pt-br.js"), | ||||
|     "ro": () => import("dayjs/locale/ro.js"), | ||||
|     "ru": () => import("dayjs/locale/ru.js"), | ||||
|     "tw": () => import("dayjs/locale/zh-tw.js"), | ||||
|     "ja": () => import("dayjs/locale/ja.js") | ||||
|     "uk": () => import("dayjs/locale/uk.js"), | ||||
| } | ||||
|  | ||||
| export async function initializeTranslations() { | ||||
| @@ -39,8 +41,10 @@ export async function initializeTranslations() { | ||||
|     }); | ||||
|  | ||||
|     // Initialize dayjs locale. | ||||
|     const dayjsLocale = await DAYJS_LOADER[locale](); | ||||
|     dayjs.locale(dayjsLocale); | ||||
|     const dayjsLocale = DAYJS_LOADER[locale]; | ||||
|     if (dayjsLocale) { | ||||
|         dayjs.locale(await dayjsLocale()); | ||||
|     } | ||||
| } | ||||
|  | ||||
| export function ordinal(date: Dayjs) { | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import log from "./log.js"; | ||||
| import dateUtils from "./date_utils.js"; | ||||
| import keyboardActions from "./keyboard_actions.js"; | ||||
| import { SANITIZER_DEFAULT_ALLOWED_TAGS, type KeyboardShortcutWithRequiredActionName, type OptionMap, type OptionNames } from "@triliumnext/commons"; | ||||
| import { getDefaultPluginConfiguration } from "@triliumnext/ckeditor5"; | ||||
|  | ||||
| function initDocumentOptions() { | ||||
|     optionService.createOption("documentId", randomSecureToken(16), false); | ||||
| @@ -152,6 +153,10 @@ const defaultOptions: DefaultOption[] = [ | ||||
|         }, | ||||
|         isSynced: false | ||||
|     }, | ||||
|     { name: "motionEnabled", value: "true", isSynced: false }, | ||||
|     { name: "shadowsEnabled", value: "true", isSynced: false }, | ||||
|     { name: "backdropEffectsEnabled", value: "true", isSynced: false }, | ||||
|  | ||||
|  | ||||
|     // Internationalization | ||||
|     { name: "locale", value: "en", isSynced: true }, | ||||
| @@ -180,6 +185,9 @@ const defaultOptions: DefaultOption[] = [ | ||||
|     { name: "textNoteEditorMultilineToolbar", value: "false", isSynced: true }, | ||||
|     { name: "textNoteEmojiCompletionEnabled", value: "true", isSynced: true }, | ||||
|     { name: "textNoteCompletionEnabled", value: "true", isSynced: true }, | ||||
|      | ||||
|     // CKEditor plugin configuration | ||||
|     { name: "ckeditorEnabledPlugins", value: getDefaultPluginConfiguration, isSynced: true }, | ||||
|  | ||||
|     // HTML import configuration | ||||
|     { name: "layoutOrientation", value: "vertical", isSynced: false }, | ||||
|   | ||||
| @@ -56,7 +56,7 @@ | ||||
|     "jsonc-eslint-parser": "^2.1.0", | ||||
|     "nx": "21.3.11", | ||||
|     "react-refresh": "^0.17.0", | ||||
|     "rollup-plugin-webpack-stats": "2.1.3", | ||||
|     "rollup-plugin-webpack-stats": "2.1.4", | ||||
|     "tslib": "^2.3.0", | ||||
|     "tsx": "4.20.4", | ||||
|     "typescript": "~5.9.0", | ||||
| @@ -89,7 +89,7 @@ | ||||
|       "@nx/js": "patches/@nx__js.patch" | ||||
|     }, | ||||
|     "overrides": { | ||||
|       "mermaid": "11.10.0", | ||||
|       "mermaid": "11.10.1", | ||||
|       "preact": "10.27.1", | ||||
|       "roughjs": "4.6.6", | ||||
|       "@types/express-serve-static-core": "5.0.7", | ||||
|   | ||||
| @@ -4,7 +4,7 @@ import { COMMON_PLUGINS, CORE_PLUGINS, POPUP_EDITOR_PLUGINS } from "./plugins.js | ||||
| import { BalloonEditor, DecoupledEditor, FindAndReplaceEditing, FindCommand } from "ckeditor5"; | ||||
| import "./translation_overrides.js"; | ||||
| export { EditorWatchdog } from "ckeditor5"; | ||||
| export { PREMIUM_PLUGINS } from "./plugins.js"; | ||||
| export { PREMIUM_PLUGINS, PLUGIN_REGISTRY, getPluginMetadata, getPluginsByCategory, getConfigurablePlugins, getDefaultPluginConfiguration } from "./plugins.js"; | ||||
| export type { EditorConfig, MentionFeed, MentionFeedObjectItem, ModelNode, ModelPosition, ModelElement, WatchdogConfig } from "ckeditor5"; | ||||
| export type { TemplateDefinition } from "ckeditor5-premium-features"; | ||||
| export { default as buildExtraCommands } from "./extra_slash_commands.js"; | ||||
|   | ||||
| @@ -30,6 +30,8 @@ import CodeBlockLanguageDropdown from "./plugins/code_block_language_dropdown.js | ||||
| import MoveBlockUpDownPlugin from "./plugins/move_block_updown.js"; | ||||
| import ScrollOnUndoRedoPlugin from "./plugins/scroll_on_undo_redo.js" | ||||
|  | ||||
| import type { PluginMetadata, PluginRegistry } from "@triliumnext/commons"; | ||||
|  | ||||
| /** | ||||
|  * Plugins that are specific to Trilium and not part of the CKEditor 5 core, included in both text editors but not in the attribute editor. | ||||
|  */ | ||||
| @@ -159,3 +161,613 @@ export const POPUP_EDITOR_PLUGINS: typeof Plugin[] = [ | ||||
|     ...COMMON_PLUGINS, | ||||
|     BlockToolbar, | ||||
| ]; | ||||
|  | ||||
| /** | ||||
|  * Plugin metadata registry for CKEditor plugins in Trilium. | ||||
|  * This defines the configurable plugins with their metadata, dependencies, and categorization. | ||||
|  */ | ||||
| export const PLUGIN_REGISTRY: PluginRegistry = { | ||||
|     plugins: { | ||||
|         // Core plugins (cannot be disabled) | ||||
|         "clipboard": { | ||||
|             id: "clipboard", | ||||
|             name: "Clipboard", | ||||
|             description: "Basic clipboard operations (copy, paste, cut)", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: true, | ||||
|             commands: ["copy", "paste", "cut"] | ||||
|         }, | ||||
|         "enter": { | ||||
|             id: "enter", | ||||
|             name: "Enter Key", | ||||
|             description: "Enter key handling for line breaks and paragraphs", | ||||
|             category: "structure", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: true, | ||||
|         }, | ||||
|         "typing": { | ||||
|             id: "typing", | ||||
|             name: "Typing", | ||||
|             description: "Basic text input and keyboard handling", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: true, | ||||
|         }, | ||||
|         "undo": { | ||||
|             id: "undo", | ||||
|             name: "Undo/Redo", | ||||
|             description: "Undo and redo functionality", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: true, | ||||
|             commands: ["undo", "redo"], | ||||
|             toolbarItems: ["undo", "redo"] | ||||
|         }, | ||||
|         "paragraph": { | ||||
|             id: "paragraph", | ||||
|             name: "Paragraph", | ||||
|             description: "Basic paragraph formatting", | ||||
|             category: "structure", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: true, | ||||
|             commands: ["paragraph"] | ||||
|         }, | ||||
|  | ||||
|         // Formatting plugins | ||||
|         "bold": { | ||||
|             id: "bold", | ||||
|             name: "Bold", | ||||
|             description: "Bold text formatting", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["bold"], | ||||
|             toolbarItems: ["bold"] | ||||
|         }, | ||||
|         "italic": { | ||||
|             id: "italic", | ||||
|             name: "Italic", | ||||
|             description: "Italic text formatting", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["italic"], | ||||
|             toolbarItems: ["italic"] | ||||
|         }, | ||||
|         "underline": { | ||||
|             id: "underline", | ||||
|             name: "Underline", | ||||
|             description: "Underline text formatting", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["underline"], | ||||
|             toolbarItems: ["underline"] | ||||
|         }, | ||||
|         "strikethrough": { | ||||
|             id: "strikethrough", | ||||
|             name: "Strikethrough", | ||||
|             description: "Strikethrough text formatting", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["strikethrough"], | ||||
|             toolbarItems: ["strikethrough"] | ||||
|         }, | ||||
|         "code": { | ||||
|             id: "code", | ||||
|             name: "Inline Code", | ||||
|             description: "Inline code formatting", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["code"], | ||||
|             toolbarItems: ["code"] | ||||
|         }, | ||||
|         "subscript": { | ||||
|             id: "subscript", | ||||
|             name: "Subscript", | ||||
|             description: "Subscript text formatting", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["subscript"], | ||||
|             toolbarItems: ["subscript"] | ||||
|         }, | ||||
|         "superscript": { | ||||
|             id: "superscript", | ||||
|             name: "Superscript", | ||||
|             description: "Superscript text formatting", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["superscript"], | ||||
|             toolbarItems: ["superscript"] | ||||
|         }, | ||||
|  | ||||
|         // Structure plugins | ||||
|         "heading": { | ||||
|             id: "heading", | ||||
|             name: "Headings", | ||||
|             description: "Heading levels (H2-H6)", | ||||
|             category: "structure", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["heading"], | ||||
|             toolbarItems: ["heading"] | ||||
|         }, | ||||
|         "blockquote": { | ||||
|             id: "blockquote", | ||||
|             name: "Block Quote", | ||||
|             description: "Block quote formatting", | ||||
|             category: "structure", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["blockQuote"], | ||||
|             toolbarItems: ["blockQuote"] | ||||
|         }, | ||||
|         "list": { | ||||
|             id: "list", | ||||
|             name: "Lists", | ||||
|             description: "Bulleted and numbered lists", | ||||
|             category: "structure", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["bulletedList", "numberedList"], | ||||
|             toolbarItems: ["bulletedList", "numberedList"] | ||||
|         }, | ||||
|         "todolist": { | ||||
|             id: "todolist", | ||||
|             name: "Todo List", | ||||
|             description: "Checkable todo list items", | ||||
|             category: "structure", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: ["list"], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["todoList"], | ||||
|             toolbarItems: ["todoList"] | ||||
|         }, | ||||
|         "alignment": { | ||||
|             id: "alignment", | ||||
|             name: "Text Alignment", | ||||
|             description: "Text alignment (left, center, right, justify)", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["alignment"], | ||||
|             toolbarItems: ["alignment"] | ||||
|         }, | ||||
|         "indent": { | ||||
|             id: "indent", | ||||
|             name: "Indentation", | ||||
|             description: "Text and block indentation", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["indent", "outdent"], | ||||
|             toolbarItems: ["indent", "outdent"] | ||||
|         }, | ||||
|  | ||||
|         // Media plugins | ||||
|         "image": { | ||||
|             id: "image", | ||||
|             name: "Images", | ||||
|             description: "Image insertion, resizing and styling", | ||||
|             category: "media", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["insertImage"], | ||||
|             toolbarItems: ["insertImage"] | ||||
|         }, | ||||
|         "link": { | ||||
|             id: "link", | ||||
|             name: "Links", | ||||
|             description: "Hyperlinks and internal note links", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["link", "unlink"], | ||||
|             toolbarItems: ["link", "unlink"] | ||||
|         }, | ||||
|  | ||||
|         // Table plugins | ||||
|         "table": { | ||||
|             id: "table", | ||||
|             name: "Tables", | ||||
|             description: "Table creation and editing", | ||||
|             category: "tables", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["insertTable"], | ||||
|             toolbarItems: ["insertTable"] | ||||
|         }, | ||||
|  | ||||
|         // Advanced plugins | ||||
|         "codeblock": { | ||||
|             id: "codeblock", | ||||
|             name: "Code Blocks", | ||||
|             description: "Syntax-highlighted code blocks", | ||||
|             category: "advanced", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["codeBlock"], | ||||
|             toolbarItems: ["codeBlock"] | ||||
|         }, | ||||
|         "math": { | ||||
|             id: "math", | ||||
|             name: "Math Formulas", | ||||
|             description: "Mathematical formulas using KaTeX", | ||||
|             category: "advanced", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["math"], | ||||
|             toolbarItems: ["math"] | ||||
|         }, | ||||
|         "mermaid": { | ||||
|             id: "mermaid", | ||||
|             name: "Mermaid Diagrams", | ||||
|             description: "Diagram creation using Mermaid syntax", | ||||
|             category: "advanced", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["mermaid"], | ||||
|             toolbarItems: ["mermaid"] | ||||
|         }, | ||||
|         "admonition": { | ||||
|             id: "admonition", | ||||
|             name: "Admonitions", | ||||
|             description: "Callout boxes and admonition blocks", | ||||
|             category: "advanced", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["admonition"], | ||||
|             toolbarItems: ["admonition"] | ||||
|         }, | ||||
|         "footnotes": { | ||||
|             id: "footnotes", | ||||
|             name: "Footnotes", | ||||
|             description: "Footnote references and definitions", | ||||
|             category: "advanced", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["footnote"], | ||||
|             toolbarItems: ["footnote"] | ||||
|         }, | ||||
|         "keyboard": { | ||||
|             id: "keyboard", | ||||
|             name: "Keyboard Shortcuts", | ||||
|             description: "Visual keyboard shortcut markers", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["kbd"], | ||||
|             toolbarItems: ["kbd"] | ||||
|         }, | ||||
|         "horizontalline": { | ||||
|             id: "horizontalline", | ||||
|             name: "Horizontal Line", | ||||
|             description: "Horizontal rule/divider line", | ||||
|             category: "structure", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["horizontalLine"], | ||||
|             toolbarItems: ["horizontalLine"] | ||||
|         }, | ||||
|         "pagebreak": { | ||||
|             id: "pagebreak", | ||||
|             name: "Page Break", | ||||
|             description: "Page break for printing", | ||||
|             category: "structure", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["pageBreak"], | ||||
|             toolbarItems: ["pageBreak"] | ||||
|         }, | ||||
|         "removeformat": { | ||||
|             id: "removeformat", | ||||
|             name: "Remove Formatting", | ||||
|             description: "Remove text formatting", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["removeFormat"], | ||||
|             toolbarItems: ["removeFormat"] | ||||
|         }, | ||||
|         "findandreplace": { | ||||
|             id: "findandreplace", | ||||
|             name: "Find and Replace", | ||||
|             description: "Text search and replace functionality", | ||||
|             category: "advanced", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["findAndReplace"], | ||||
|             toolbarItems: ["findAndReplace"] | ||||
|         }, | ||||
|         "font": { | ||||
|             id: "font", | ||||
|             name: "Font Styling", | ||||
|             description: "Font family, size, color, and background color", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["fontFamily", "fontSize", "fontColor", "fontBackgroundColor"], | ||||
|             toolbarItems: ["fontFamily", "fontSize", "fontColor", "fontBackgroundColor"] | ||||
|         }, | ||||
|         "specialcharacters": { | ||||
|             id: "specialcharacters", | ||||
|             name: "Special Characters", | ||||
|             description: "Insert special characters and symbols", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["specialCharacters"], | ||||
|             toolbarItems: ["specialCharacters"] | ||||
|         }, | ||||
|         "emoji": { | ||||
|             id: "emoji", | ||||
|             name: "Emoji Support", | ||||
|             description: "Emoji insertion and autocomplete", | ||||
|             category: "formatting", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["emoji"], | ||||
|             toolbarItems: ["emoji"] | ||||
|         }, | ||||
|  | ||||
|         // Premium plugins | ||||
|         "slashcommand": { | ||||
|             id: "slashcommand", | ||||
|             name: "Slash Commands", | ||||
|             description: "Quick command insertion with / key", | ||||
|             category: "advanced", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: true, | ||||
|             isCore: false, | ||||
|             commands: ["slashCommand"] | ||||
|         }, | ||||
|         "template": { | ||||
|             id: "template", | ||||
|             name: "Templates", | ||||
|             description: "Text templates and snippets", | ||||
|             category: "advanced", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: true, | ||||
|             isCore: false, | ||||
|             commands: ["template"], | ||||
|             toolbarItems: ["template"] | ||||
|         }, | ||||
|  | ||||
|         // Trilium-specific plugins | ||||
|         "uploadimage": { | ||||
|             id: "uploadimage", | ||||
|             name: "Image Upload", | ||||
|             description: "Trilium-specific image upload handling", | ||||
|             category: "trilium", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: ["image"], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|         }, | ||||
|         "cuttonote": { | ||||
|             id: "cuttonote", | ||||
|             name: "Cut to Note", | ||||
|             description: "Cut selected text to create a new note", | ||||
|             category: "trilium", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["cutToNote"] | ||||
|         }, | ||||
|         "internallink": { | ||||
|             id: "internallink", | ||||
|             name: "Internal Links", | ||||
|             description: "Trilium-specific internal note linking", | ||||
|             category: "trilium", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: ["link"], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|         }, | ||||
|         "insertdatetime": { | ||||
|             id: "insertdatetime", | ||||
|             name: "Insert Date/Time", | ||||
|             description: "Insert current date and time", | ||||
|             category: "trilium", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["insertDateTime"] | ||||
|         }, | ||||
|         "includenote": { | ||||
|             id: "includenote", | ||||
|             name: "Include Note", | ||||
|             description: "Include content from other notes", | ||||
|             category: "trilium", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["includeNote"] | ||||
|         }, | ||||
|         "uploadfile": { | ||||
|             id: "uploadfile", | ||||
|             name: "File Upload", | ||||
|             description: "Upload and attach files", | ||||
|             category: "trilium", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["uploadFile"] | ||||
|         }, | ||||
|         "markdownimport": { | ||||
|             id: "markdownimport", | ||||
|             name: "Markdown Import", | ||||
|             description: "Import markdown content", | ||||
|             category: "trilium", | ||||
|             defaultEnabled: true, | ||||
|             dependencies: [], | ||||
|             conflicts: [], | ||||
|             requiresPremium: false, | ||||
|             isCore: false, | ||||
|             commands: ["markdownImport"] | ||||
|         } | ||||
|     }, | ||||
|     version: "1.0.0", | ||||
|     lastModified: new Date().toISOString() | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Get plugin metadata by ID | ||||
|  */ | ||||
| export function getPluginMetadata(pluginId: string): PluginMetadata | undefined { | ||||
|     return PLUGIN_REGISTRY.plugins[pluginId]; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get all plugins in a category | ||||
|  */ | ||||
| export function getPluginsByCategory(category: string): PluginMetadata[] { | ||||
|     return Object.values(PLUGIN_REGISTRY.plugins).filter(plugin => plugin.category === category); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get core plugins (cannot be disabled) | ||||
|  */ | ||||
| export function getCorePlugins(): PluginMetadata[] { | ||||
|     return Object.values(PLUGIN_REGISTRY.plugins).filter(plugin => plugin.isCore); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get configurable plugins (can be enabled/disabled) | ||||
|  */ | ||||
| export function getConfigurablePlugins(): PluginMetadata[] { | ||||
|     return Object.values(PLUGIN_REGISTRY.plugins).filter(plugin => !plugin.isCore); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get default enabled plugins configuration as JSON string | ||||
|  */ | ||||
| export function getDefaultPluginConfiguration(): string { | ||||
|     const defaultConfig: Record<string, boolean> = {}; | ||||
|     Object.values(PLUGIN_REGISTRY.plugins).forEach(plugin => { | ||||
|         if (!plugin.isCore) { | ||||
|             defaultConfig[plugin.id] = plugin.defaultEnabled; | ||||
|         } | ||||
|     }); | ||||
|     return JSON.stringify(defaultConfig); | ||||
| } | ||||
|   | ||||
| @@ -4,20 +4,23 @@ | ||||
|   "include": [], | ||||
|   "references": [ | ||||
|     { | ||||
|       "path": "../ckeditor5-footnotes" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-math" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-admonition" | ||||
|       "path": "../commons" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-mermaid" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-math" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-keyboard-marker" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-footnotes" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-admonition" | ||||
|     }, | ||||
|     { | ||||
|       "path": "./tsconfig.lib.json" | ||||
|     } | ||||
|   | ||||
| @@ -20,19 +20,22 @@ | ||||
|   ], | ||||
|   "references": [ | ||||
|     { | ||||
|       "path": "../ckeditor5-footnotes" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-math" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-admonition" | ||||
|       "path": "../commons/tsconfig.lib.json" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-mermaid" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-math" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-keyboard-marker" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-footnotes" | ||||
|     }, | ||||
|     { | ||||
|       "path": "../ckeditor5-admonition" | ||||
|     } | ||||
|   ] | ||||
| } | ||||
|   | ||||
| @@ -63,6 +63,6 @@ | ||||
|     "codemirror-lang-elixir": "4.0.0", | ||||
|     "codemirror-lang-hcl": "0.1.0", | ||||
|     "codemirror-lang-mermaid": "0.5.0", | ||||
|     "eslint-linter-browserify": "9.33.0" | ||||
|     "eslint-linter-browserify": "9.34.0" | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| export * from "./lib/i18n.js"; | ||||
| export * from "./lib/options_interface.js"; | ||||
| export * from "./lib/ckeditor_plugin_interface.js"; | ||||
| export * from "./lib/keyboard_actions_interface.js"; | ||||
| export * from "./lib/hidden_subtree.js"; | ||||
| export * from "./lib/rows.js"; | ||||
|   | ||||
							
								
								
									
										163
									
								
								packages/commons/src/lib/ckeditor_plugin_interface.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										163
									
								
								packages/commons/src/lib/ckeditor_plugin_interface.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,163 @@ | ||||
| /** | ||||
|  * @module CKEditor Plugin Interface | ||||
|  *  | ||||
|  * This module defines the TypeScript interfaces and types for managing | ||||
|  * CKEditor plugins in Trilium Notes. It provides type-safe configuration | ||||
|  * for plugin enablement, metadata, and dependency management. | ||||
|  */ | ||||
|  | ||||
| /** | ||||
|  * Defines the categories of CKEditor plugins available in Trilium. | ||||
|  */ | ||||
| export type PluginCategory =  | ||||
|     | "formatting"    // Text formatting (bold, italic, etc.) | ||||
|     | "structure"     // Document structure (headings, lists, etc.) | ||||
|     | "media"        // Images, files, embeds | ||||
|     | "tables"       // Table-related functionality | ||||
|     | "advanced"     // Advanced features (math, mermaid, etc.) | ||||
|     | "trilium"      // Trilium-specific plugins | ||||
|     | "external";    // Third-party plugins | ||||
|  | ||||
| /** | ||||
|  * Represents the metadata for a CKEditor plugin. | ||||
|  */ | ||||
| export interface PluginMetadata { | ||||
|     /** Unique identifier for the plugin */ | ||||
|     id: string; | ||||
|     /** Human-readable display name */ | ||||
|     name: string; | ||||
|     /** Brief description of the plugin's functionality */ | ||||
|     description: string; | ||||
|     /** Category this plugin belongs to */ | ||||
|     category: PluginCategory; | ||||
|     /** Whether this plugin is enabled by default for new users */ | ||||
|     defaultEnabled: boolean; | ||||
|     /** Array of plugin IDs that this plugin depends on */ | ||||
|     dependencies: string[]; | ||||
|     /** Array of plugin IDs that conflict with this plugin */ | ||||
|     conflicts: string[]; | ||||
|     /** Whether this plugin requires a premium CKEditor license */ | ||||
|     requiresPremium: boolean; | ||||
|     /** Whether this plugin is part of the core editor functionality (cannot be disabled) */ | ||||
|     isCore: boolean; | ||||
|     /** Toolbar items/commands provided by this plugin */ | ||||
|     toolbarItems?: string[]; | ||||
|     /** Commands provided by this plugin */ | ||||
|     commands?: string[]; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Configuration for a user's CKEditor plugin preferences. | ||||
|  */ | ||||
| export interface PluginConfiguration { | ||||
|     /** Plugin ID */ | ||||
|     id: string; | ||||
|     /** Whether the plugin is enabled for this user */ | ||||
|     enabled: boolean; | ||||
|     /** User-specific configuration for the plugin (if any) */ | ||||
|     config?: Record<string, unknown>; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * The complete registry of available CKEditor plugins. | ||||
|  */ | ||||
| export interface PluginRegistry { | ||||
|     /** Map of plugin ID to plugin metadata */ | ||||
|     plugins: Record<string, PluginMetadata>; | ||||
|     /** Version of the plugin registry (for cache invalidation) */ | ||||
|     version: string; | ||||
|     /** Last modified timestamp */ | ||||
|     lastModified: string; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Result of plugin dependency validation. | ||||
|  */ | ||||
| export interface PluginValidationResult { | ||||
|     /** Whether the configuration is valid */ | ||||
|     valid: boolean; | ||||
|     /** Array of validation errors */ | ||||
|     errors: PluginValidationError[]; | ||||
|     /** Array of warnings (non-blocking issues) */ | ||||
|     warnings: PluginValidationWarning[]; | ||||
|     /** Resolved list of plugins that should be enabled */ | ||||
|     resolvedPlugins: string[]; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Validation error for plugin configuration. | ||||
|  */ | ||||
| export interface PluginValidationError { | ||||
|     /** Type of error */ | ||||
|     type: "missing_dependency" | "circular_dependency" | "plugin_conflict" | "premium_required"; | ||||
|     /** Plugin ID that caused the error */ | ||||
|     pluginId: string; | ||||
|     /** Human-readable error message */ | ||||
|     message: string; | ||||
|     /** Additional context about the error */ | ||||
|     details?: Record<string, unknown>; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Validation warning for plugin configuration. | ||||
|  */ | ||||
| export interface PluginValidationWarning { | ||||
|     /** Type of warning */ | ||||
|     type: "dependency_disabled" | "unused_dependency" | "performance_impact"; | ||||
|     /** Plugin ID that caused the warning */ | ||||
|     pluginId: string; | ||||
|     /** Human-readable warning message */ | ||||
|     message: string; | ||||
|     /** Additional context about the warning */ | ||||
|     details?: Record<string, unknown>; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Request to update plugin configuration. | ||||
|  */ | ||||
| export interface UpdatePluginConfigRequest { | ||||
|     /** Array of plugin configurations to update */ | ||||
|     plugins: PluginConfiguration[]; | ||||
|     /** Whether to validate dependencies before saving */ | ||||
|     validate?: boolean; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Response from updating plugin configuration. | ||||
|  */ | ||||
| export interface UpdatePluginConfigResponse { | ||||
|     /** Whether the update was successful */ | ||||
|     success: boolean; | ||||
|     /** Validation result (if validation was requested) */ | ||||
|     validation?: PluginValidationResult; | ||||
|     /** Updated plugin configurations */ | ||||
|     plugins: PluginConfiguration[]; | ||||
|     /** Any errors that occurred during the update */ | ||||
|     errors?: string[]; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Options for querying the plugin registry. | ||||
|  */ | ||||
| export interface QueryPluginsOptions { | ||||
|     /** Filter by category */ | ||||
|     category?: PluginCategory; | ||||
|     /** Filter by enabled status */ | ||||
|     enabled?: boolean; | ||||
|     /** Filter by core status */ | ||||
|     coreOnly?: boolean; | ||||
|     /** Include user configuration in results */ | ||||
|     includeConfig?: boolean; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Result of querying plugins. | ||||
|  */ | ||||
| export interface QueryPluginsResult { | ||||
|     /** Array of plugin metadata */ | ||||
|     plugins: (PluginMetadata & { enabled?: boolean; config?: Record<string, unknown> })[]; | ||||
|     /** Total count of plugins (before filtering) */ | ||||
|     totalCount: number; | ||||
|     /** Categories available in the registry */ | ||||
|     categories: PluginCategory[]; | ||||
| } | ||||
| @@ -10,51 +10,17 @@ export interface Locale { | ||||
| } | ||||
|  | ||||
| const UNSORTED_LOCALES: Locale[] = [ | ||||
|     { | ||||
|         id: "en", | ||||
|         name: "English", | ||||
|         electronLocale: "en" | ||||
|     }, | ||||
|     { | ||||
|         id: "de", | ||||
|         name: "Deutsch", | ||||
|         electronLocale: "de" | ||||
|     }, | ||||
|     { | ||||
|         id: "es", | ||||
|         name: "Español", | ||||
|         electronLocale: "es" | ||||
|     }, | ||||
|     { | ||||
|         id: "fr", | ||||
|         name: "Français", | ||||
|         electronLocale: "fr" | ||||
|     }, | ||||
|     { | ||||
|         id: "cn", | ||||
|         name: "简体中文", | ||||
|         electronLocale: "zh_CN" | ||||
|     }, | ||||
|     { | ||||
|         id: "tw", | ||||
|         name: "繁體中文", | ||||
|         electronLocale: "zh_TW" | ||||
|     }, | ||||
|     { | ||||
|         id: "ro", | ||||
|         name: "Română", | ||||
|         electronLocale: "ro" | ||||
|     }, | ||||
|     { | ||||
|         id: "ru", | ||||
|         name: "Русский", | ||||
|         electronLocale: "ru" | ||||
|     }, | ||||
|     { | ||||
|         id: "ja", | ||||
|         name: "日本語", | ||||
|         electronLocale: "ja" | ||||
|     }, | ||||
|     { id: "cn", name: "简体中文", electronLocale: "zh_CN" }, | ||||
|     { id: "de", name: "Deutsch", electronLocale: "de" }, | ||||
|     { id: "en", name: "English", electronLocale: "en" }, | ||||
|     { id: "es", name: "Español", electronLocale: "es" }, | ||||
|     { id: "fr", name: "Français", electronLocale: "fr" }, | ||||
|     { id: "ja", name: "日本語", electronLocale: "ja" }, | ||||
|     { id: "pt_br", name: "Português (Brasil)", electronLocale: "pt-BR" }, | ||||
|     { id: "ro", name: "Română", electronLocale: "ro" }, | ||||
|     { id: "ru", name: "Русский", electronLocale: "ru" }, | ||||
|     { id: "tw", name: "繁體中文", electronLocale: "zh_TW" }, | ||||
|     { id: "uk", name: "Українська", electronLocale: "uk" }, | ||||
|  | ||||
|     /* | ||||
|      * Right to left languages | ||||
|   | ||||
| @@ -93,6 +93,9 @@ export interface OptionDefinitions extends KeyboardShortcutsOptions<KeyboardActi | ||||
|  | ||||
|     // Appearance | ||||
|     splitEditorOrientation: "horziontal" | "vertical"; | ||||
|     motionEnabled: boolean; | ||||
|     shadowsEnabled: boolean; | ||||
|     backdropEffectsEnabled: boolean; | ||||
|     codeNoteTheme: string; | ||||
|  | ||||
|     initialized: boolean; | ||||
| @@ -147,6 +150,9 @@ export interface OptionDefinitions extends KeyboardShortcutsOptions<KeyboardActi | ||||
|     codeOpenAiModel: string; | ||||
|     aiSelectedProvider: string; | ||||
|     seenCallToActions: string; | ||||
|      | ||||
|     // CKEditor plugin options | ||||
|     ckeditorEnabledPlugins: string; | ||||
| } | ||||
|  | ||||
| export type OptionNames = keyof OptionDefinitions; | ||||
|   | ||||
							
								
								
									
										681
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										681
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user