import { defineStore } from 'pinia'; import apiClient from '../utils/apiClient'; // 使用统一的 apiClient import { ref, computed } from 'vue'; // 移除 watch import i18n, { setLocale, defaultLng, availableLocales } from '../i18n'; // Import i18n instance, setLocale, defaultLng, and availableLocales import type { PaneName } from './layout.store'; import { useAuthStore } from './auth.store'; import type { ConnectionInfo } from './connections.store'; export type SortField = keyof Pick; export type SortOrder = 'asc' | 'desc'; // Assuming manual definition for now if no shared types exist: type CaptchaProvider = 'hcaptcha' | 'recaptcha' | 'none'; interface CaptchaSettings { enabled: boolean; provider: CaptchaProvider; hcaptchaSiteKey?: string; hcaptchaSecretKey?: string; // Store locally but don't expose via getters easily recaptchaSiteKey?: string; recaptchaSecretKey?: string; // Store locally but don't expose via getters easily } interface UpdateCaptchaSettingsDto { enabled?: boolean; provider?: CaptchaProvider; hcaptchaSiteKey?: string; hcaptchaSecretKey?: string; recaptchaSiteKey?: string; recaptchaSecretKey?: string; } // 移除 ITheme 和默认主题定义,这些移到 appearance.store.ts // 定义通用设置状态类型 interface SettingsState { language?: string; // 改为 string 以支持动态语言 ipWhitelist?: string; maxLoginAttempts?: string; loginBanDuration?: string; showPopupFileEditor?: string; // 'true' or 'false' shareFileEditorTabs?: string; // 'true' or 'false' ipWhitelistEnabled?: string; // 添加 IP 白名单启用状态 'true' or 'false' autoCopyOnSelect?: string; // 'true' or 'false' - 终端选中自动复制 dockerStatusIntervalSeconds?: string; // NEW: Docker 状态刷新间隔 (秒) dockerDefaultExpand?: string; // NEW: Docker 默认展开详情 'true' or 'false' statusMonitorIntervalSeconds?: string; // NEW: 状态监控轮询间隔 (秒) workspaceSidebarPersistent?: string; // NEW: 工作区侧边栏是否固定 'true' or 'false' sidebarPaneWidths?: string; // NEW: 存储各侧边栏组件宽度的 JSON 字符串 fileManagerRowSizeMultiplier?: string; // NEW: 文件管理器行大小乘数 (e.g., '1.0') fileManagerColWidths?: string; // NEW: 文件管理器列宽 JSON 字符串 (e.g., '{"name": 300, "size": 100}') commandInputSyncTarget?: 'quickCommands' | 'commandHistory' | 'none'; // NEW: 命令输入同步目标 timezone?: string; // NEW: 时区设置 (e.g., 'Asia/Shanghai', 'UTC') rdpModalWidth?: string; // NEW: RDP 模态框宽度 rdpModalHeight?: string; // NEW: RDP 模态框高度 ipBlacklistEnabled?: string; dashboardSortBy?: SortField; dashboardSortOrder?: SortOrder; showConnectionTags?: string; // 'true' or 'false' showQuickCommandTags?: string; // 'true' or 'false' layoutLocked?: string; // 'true' or 'false' - NEW: 布局锁定状态 terminalScrollbackLimit?: string; // NEW: 终端回滚行数上限 (e.g., '5000', '0' for unlimited) [key: string]: string | undefined; } export const useSettingsStore = defineStore('settings', () => { const authStore = useAuthStore(); // <--- 实例化 authStore // --- State --- const settings = ref>({}); // 通用设置状态 const parsedSidebarPaneWidths = ref>({}); // NEW: 解析后的侧边栏宽度对象 const parsedFileManagerColWidths = ref>({}); // NEW: 解析后的文件管理器列宽对象 const captchaSettings = ref(null); // NEW: CAPTCHA 设置状态 const isLoading = ref(false); const error = ref(null); // 移除外观相关状态: isStyleCustomizerVisible, currentUiTheme, currentXtermTheme // --- Actions --- /** * Fetches general settings from the backend and updates the store state. * Also sets the i18n locale based on the fetched language setting. */ async function loadInitialSettings() { isLoading.value = true; error.value = null; let determinedLang: string | undefined; // 使用 string 类型 try { console.log('[SettingsStore] 加载通用设置...'); // Fetch all settings, including the new ones const [ generalSettingsResponse, showConnectionTagsResponse, showQuickCommandTagsResponse ] = await Promise.all([ apiClient.get>('/settings'), apiClient.get<{ enabled: boolean }>('/settings/show-connection-tags'), apiClient.get<{ enabled: boolean }>('/settings/show-quick-command-tags') ]); settings.value = generalSettingsResponse.data; // Store fetched general settings // Store the specific boolean settings settings.value.showConnectionTags = String(showConnectionTagsResponse.data.enabled); settings.value.showQuickCommandTags = String(showQuickCommandTagsResponse.data.enabled); // --- 更详细的日志 --- console.log('[SettingsStore] Fetched settings from backend:', JSON.stringify(settings.value)); // --- 设置默认值 (如果后端未返回) --- if (settings.value.showPopupFileEditor === undefined) { settings.value.showPopupFileEditor = 'true'; } if (settings.value.shareFileEditorTabs === undefined) { settings.value.shareFileEditorTabs = 'true'; } if (settings.value.ipWhitelistEnabled === undefined) { settings.value.ipWhitelistEnabled = 'false'; // 默认禁用 IP 白名单 } if (settings.value.maxLoginAttempts === undefined) { settings.value.maxLoginAttempts = '5'; // 默认 5 次 } if (settings.value.loginBanDuration === undefined) { settings.value.loginBanDuration = '300'; // 默认 300 秒 } // NEW: IP Blacklist enabled default if (settings.value.ipBlacklistEnabled === undefined) { settings.value.ipBlacklistEnabled = 'true'; // 默认启用 IP 黑名单 } if (settings.value.autoCopyOnSelect === undefined) { settings.value.autoCopyOnSelect = 'false'; // 默认禁用选中即复制 } // NEW: Docker setting defaults if (settings.value.dockerStatusIntervalSeconds === undefined) { settings.value.dockerStatusIntervalSeconds = '2'; // 默认 2 秒 } if (settings.value.dockerDefaultExpand === undefined) { settings.value.dockerDefaultExpand = 'false'; // 默认不展开 } // NEW: Status Monitor interval default if (settings.value.statusMonitorIntervalSeconds === undefined) { settings.value.statusMonitorIntervalSeconds = '3'; // 默认 3 秒 } // NEW: Workspace sidebar persistent default if (settings.value.workspaceSidebarPersistent === undefined) { settings.value.workspaceSidebarPersistent = 'false'; // 默认不固定 } // NEW: Load and parse sidebar pane widths const defaultPaneWidth = '350px'; // +++ Ensure PaneName type is available or define it here +++ const knownPanes: PaneName[] = ['connections', 'fileManager', 'editor', 'statusMonitor', 'commandHistory', 'quickCommands', 'dockerManager']; // Add all possible sidebar panes let loadedWidths: Record = {}; try { if (settings.value.sidebarPaneWidths) { loadedWidths = JSON.parse(settings.value.sidebarPaneWidths); if (typeof loadedWidths !== 'object' || loadedWidths === null) { console.warn('[SettingsStore] Invalid sidebarPaneWidths format loaded, resetting.'); loadedWidths = {}; } } } catch (e) { console.error('[SettingsStore] Failed to parse sidebarPaneWidths, resetting.', e); loadedWidths = {}; } // Ensure defaults for all known panes const finalWidths: Record = {}; knownPanes.forEach(pane => { finalWidths[pane] = loadedWidths[pane] || defaultPaneWidth; }); parsedSidebarPaneWidths.value = finalWidths; // Optionally save back if defaults were added (might cause extra write on first load) // if (Object.keys(loadedWidths).length !== Object.keys(finalWidths).length) { // await updateSetting('sidebarPaneWidths', JSON.stringify(finalWidths)); // } // NEW: Load and parse file manager layout settings const defaultFileManagerRowMultiplier = '1.0'; const defaultFileManagerColWidths = { type: 50, name: 300, size: 100, permissions: 120, modified: 180 }; // Row Size Multiplier console.log(`[SettingsStore] Raw fileManagerRowSizeMultiplier from backend: '${settings.value.fileManagerRowSizeMultiplier}'`); if (settings.value.fileManagerRowSizeMultiplier === undefined) { settings.value.fileManagerRowSizeMultiplier = defaultFileManagerRowMultiplier; // Assign first console.log(`[SettingsStore] fileManagerRowSizeMultiplier not found, set to default: ${settings.value.fileManagerRowSizeMultiplier}`); // Log the assigned value } // Ensure it's a valid number string before parsing later const parsedMultiplier = parseFloat(settings.value.fileManagerRowSizeMultiplier); if (isNaN(parsedMultiplier) || parsedMultiplier <= 0) { console.warn(`[SettingsStore] Invalid fileManagerRowSizeMultiplier loaded ('${settings.value.fileManagerRowSizeMultiplier}'), resetting to default.`); settings.value.fileManagerRowSizeMultiplier = defaultFileManagerRowMultiplier; } console.log(`[SettingsStore] Final fileManagerRowSizeMultiplier value in store: '${settings.value.fileManagerRowSizeMultiplier}'`); // Column Widths let loadedFmWidths: Record = {}; console.log(`[SettingsStore] Raw fileManagerColWidths from backend: '${settings.value.fileManagerColWidths}'`); try { if (settings.value.fileManagerColWidths) { loadedFmWidths = JSON.parse(settings.value.fileManagerColWidths); console.log(`[SettingsStore] Successfully parsed fileManagerColWidths JSON: ${JSON.stringify(loadedFmWidths)}`); if (typeof loadedFmWidths !== 'object' || loadedFmWidths === null) { console.warn('[SettingsStore] Invalid fileManagerColWidths format loaded, resetting.'); loadedFmWidths = {}; } // Validate that values are numbers for (const key in loadedFmWidths) { if (typeof loadedFmWidths[key] !== 'number') { console.warn(`[SettingsStore] Invalid non-numeric value found in fileManagerColWidths for key '${key}', resetting.`); loadedFmWidths = {}; break; } } } } catch (e) { console.error('[SettingsStore] Failed to parse fileManagerColWidths, resetting.', e); loadedFmWidths = {}; } // Ensure defaults for all known columns, merging with loaded valid ones const finalFmWidths: Record = { ...defaultFileManagerColWidths }; console.log(`[SettingsStore] Default FM Col Widths: ${JSON.stringify(defaultFileManagerColWidths)}`); Object.keys(defaultFileManagerColWidths).forEach(key => { if (loadedFmWidths[key] !== undefined && loadedFmWidths[key] > 0) { // Use loaded if valid finalFmWidths[key] = loadedFmWidths[key]; } }); parsedFileManagerColWidths.value = finalFmWidths; console.log(`[SettingsStore] Final parsedFileManagerColWidths value in store: ${JSON.stringify(parsedFileManagerColWidths.value)}`); // Save back if defaults were added or structure changed (optional, might cause extra write) // const currentSavedFmWidthsString = settings.value.fileManagerColWidths; // const finalFmWidthsString = JSON.stringify(finalFmWidths); // if (currentSavedFmWidthsString !== finalFmWidthsString) { // await updateSetting('fileManagerColWidths', finalFmWidthsString); // } // NEW: Command Input Sync Target default if (settings.value.commandInputSyncTarget === undefined) { settings.value.commandInputSyncTarget = 'none'; // 默认不同步 } // NEW: Timezone default if (settings.value.timezone === undefined) { settings.value.timezone = 'UTC'; // 默认 UTC } // NEW: RDP Modal Size defaults if (settings.value.rdpModalWidth === undefined) { settings.value.rdpModalWidth = '1064'; // 默认宽度 (1024 + 40 padding) } if (settings.value.rdpModalHeight === undefined) { settings.value.rdpModalHeight = '858'; } if (settings.value.dashboardSortBy === undefined) { settings.value.dashboardSortBy = 'last_connected_at'; } if (settings.value.dashboardSortOrder === undefined) { settings.value.dashboardSortOrder = 'desc'; } // NEW: Tag visibility defaults if (settings.value.showConnectionTags === undefined) { settings.value.showConnectionTags = 'true'; // 默认显示 } if (settings.value.showQuickCommandTags === undefined) { settings.value.showQuickCommandTags = 'true'; // 默认显示 } // +++ Add missing closing brace +++ // NEW: Layout locked default - Only set if not provided by backend if (settings.value.layoutLocked === undefined) { settings.value.layoutLocked = 'false'; // 默认不锁定 console.log('[SettingsStore] layoutLocked not found in fetched settings, set to default: false'); } else { console.log(`[SettingsStore] layoutLocked found in fetched settings: ${settings.value.layoutLocked}`); } // NEW: Terminal scrollback limit default if (settings.value.terminalScrollbackLimit === undefined) { settings.value.terminalScrollbackLimit = '5000'; // 默认 5000 行 console.log(`[SettingsStore] terminalScrollbackLimit not found, set to default: ${settings.value.terminalScrollbackLimit}`); } // --- 语言设置 --- const langFromSettings = settings.value.language; console.log(`[SettingsStore] Language from fetched settings: ${langFromSettings}`); // <-- 添加日志 // 检查从设置加载的语言 (完整区域代码) 是否在可用语言列表中 if (langFromSettings && availableLocales.includes(langFromSettings)) { determinedLang = langFromSettings; } else { // 如果设置中的语言无效或缺失,尝试浏览器提供的完整区域代码 const navigatorLocale = navigator.language; if (navigatorLocale && availableLocales.includes(navigatorLocale)) { determinedLang = navigatorLocale; } else { // (可选) 尝试浏览器语言的主语言部分 const navigatorLangPart = navigatorLocale?.split('-')[0]; if (navigatorLangPart && availableLocales.includes(navigatorLangPart)) { determinedLang = navigatorLangPart; } else { // 最后回退到 i18n 配置的默认语言 determinedLang = defaultLng; } } console.warn(`[SettingsStore] Invalid or missing language setting ('${langFromSettings}') received from backend. Falling back to '${determinedLang}'.`); // Optionally save the fallback language back // await updateSetting('language', determinedLang); } if (determinedLang) { console.log(`[SettingsStore] Determined language: ${determinedLang}. Calling setLocale...`); // <-- 添加日志 setLocale(determinedLang); } else { // This case should theoretically not happen with the fallback logic above console.error('[SettingsStore] Could not determine a valid language. This should not happen.'); console.log(`[SettingsStore] Falling back to default: ${defaultLng}. Calling setLocale...`); // <-- 添加日志 setLocale(defaultLng); } } catch (err: any) { console.error('Error loading general settings:', err); // <-- 修改日志 error.value = err.response?.data?.message || err.message || 'Failed to load settings'; // 出错时(例如未登录),根据浏览器语言设置回退语言 const navigatorLang = navigator.language?.split('-')[0]; // 错误时也尝试浏览器完整区域代码,然后主语言部分,最后默认 const navigatorLocale = navigator.language; const navigatorLangPart = navigatorLocale?.split('-')[0]; let fallbackLang = defaultLng; // Start with default if (navigatorLocale && availableLocales.includes(navigatorLocale)) { fallbackLang = navigatorLocale; } else if (navigatorLangPart && availableLocales.includes(navigatorLangPart)) { fallbackLang = navigatorLangPart; } console.log(`[SettingsStore] Error loading settings. Falling back to language: ${fallbackLang}. Calling setLocale...`); // <-- 添加日志 setLocale(fallbackLang); } finally { isLoading.value = false; } } // 移除外观相关函数: loadAndApplyThemesFromSettings, applyUiTheme, saveCustomThemes, resetCustomThemes, toggleStyleCustomizer /** * Updates a single general setting value both locally and on the backend. * Uses specific endpoints for boolean settings where available. * @param key The setting key to update. * @param value The new value for the setting (string for general, boolean for specific). */ async function updateSetting(key: keyof SettingsState, value: string | boolean) { // 移除外观相关的键检查 const allowedKeys: Array = [ 'language', 'ipWhitelist', 'maxLoginAttempts', 'loginBanDuration', 'showPopupFileEditor', 'shareFileEditorTabs', 'ipWhitelistEnabled', 'autoCopyOnSelect', 'dockerStatusIntervalSeconds', 'dockerDefaultExpand', 'statusMonitorIntervalSeconds', // +++ 添加状态监控间隔键 +++ 'workspaceSidebarPersistent', // +++ 添加侧边栏固定键 +++ 'sidebarPaneWidths', // +++ 添加侧边栏宽度对象键 +++ 'fileManagerRowSizeMultiplier', // +++ 添加文件管理器行大小键 +++ 'fileManagerColWidths', // +++ 添加文件管理器列宽键 +++ 'commandInputSyncTarget', // +++ 添加命令输入同步目标键 +++ 'timezone', // NEW: 添加时区键 'rdpModalWidth', // NEW: 添加 RDP 模态框宽度键 'rdpModalHeight', // NEW: 添加 RDP 模态框高度键 'ipBlacklistEnabled', 'dashboardSortBy', 'dashboardSortOrder', 'showConnectionTags', // NEW 'showQuickCommandTags', // NEW 'layoutLocked', // NEW 'terminalScrollbackLimit' // NEW ]; if (!allowedKeys.includes(key)) { console.error(`[SettingsStore] 尝试更新不允许的设置键: ${key}`); throw new Error(`不允许更新设置项 '${key}'`); } // Use specific endpoints for boolean settings const booleanEndpoints: Partial> = { showConnectionTags: '/settings/show-connection-tags', showQuickCommandTags: '/settings/show-quick-command-tags', autoCopyOnSelect: '/settings/auto-copy-on-select', // Add other boolean settings with specific endpoints here if needed }; try { let apiPromise: Promise; const endpoint = booleanEndpoints[key]; if (endpoint && typeof value === 'boolean') { console.log(`[SettingsStore] Attempting to update boolean setting via specific endpoint - Key: ${key}, Value: ${value}, Endpoint: ${endpoint}`); apiPromise = apiClient.put(endpoint, { enabled: value }); } else if (typeof value === 'string') { console.log(`[SettingsStore] Attempting to update general setting - Key: ${key}, Value: ${value}`); const payload = { [key]: value }; console.log('[SettingsStore] Sending PUT request to /settings with payload:', payload); apiPromise = apiClient.put('/settings', payload); } else { throw new Error(`Invalid value type for setting '${key}': expected boolean for specific endpoint or string for general.`); } await apiPromise; console.log(`[SettingsStore] Successfully updated setting via API - Key: ${key}`); // Update store state *after* successful API call settings.value = { ...settings.value, [key]: String(value) }; // Store as string internally // If updating language, check if it's valid and update i18n if (key === 'language' && typeof value === 'string' && availableLocales.includes(value)) { console.log(`[SettingsStore] updateSetting: Language updated to ${value}. Calling setLocale...`); setLocale(value); } else if (key === 'language') { console.warn(`[SettingsStore] updateSetting: Attempted to set invalid language '${value}'. Ignoring i18n update.`); } } catch (err: any) { // +++ Enhanced error logging +++ console.error(`[SettingsStore] Failed to update setting '${key}' via API. Error:`, err); if (err.response) { console.error('[SettingsStore] API Error Response Data:', err.response.data); console.error('[SettingsStore] API Error Response Status:', err.response.status); console.error('[SettingsStore] API Error Response Headers:', err.response.headers); } else if (err.request) { console.error('[SettingsStore] API Error Request:', err.request); } else { console.error('[SettingsStore] API Error Message:', err.message); } // Rethrow the error but maybe provide a more specific message if possible throw new Error(err.response?.data?.message || `更新设置项 '${key}' 失败: ${err.message}`); } } /** * Updates multiple general settings values both locally and on the backend. * @param updates An object containing key-value pairs of settings to update. */ async function updateMultipleSettings(updates: Partial) { // 移除外观相关的键检查 const allowedKeys: Array = [ 'language', 'ipWhitelist', 'maxLoginAttempts', 'loginBanDuration', 'showPopupFileEditor', 'shareFileEditorTabs', 'ipWhitelistEnabled', 'autoCopyOnSelect', 'dockerStatusIntervalSeconds', 'dockerDefaultExpand', 'statusMonitorIntervalSeconds', // +++ 添加状态监控间隔键 +++ 'workspaceSidebarPersistent', // +++ 添加侧边栏固定键 +++ 'sidebarPaneWidths', // +++ 添加侧边栏宽度对象键 +++ 'fileManagerRowSizeMultiplier', // +++ 添加文件管理器行大小键 +++ 'fileManagerColWidths', // +++ 添加文件管理器列宽键 +++ 'commandInputSyncTarget', // +++ 添加命令输入同步目标键 +++ 'timezone', // NEW: 添加时区键 'rdpModalWidth', // NEW: 添加 RDP 模态框宽度键 'rdpModalHeight', // NEW: 添加 RDP 模态框高度键 'ipBlacklistEnabled', 'dashboardSortBy', 'dashboardSortOrder', 'showConnectionTags', // NEW 'showQuickCommandTags', // NEW 'layoutLocked', // NEW 'terminalScrollbackLimit' // NEW ]; const filteredUpdates: Partial = {}; let languageUpdate: string | undefined = undefined; for (const key in updates) { if (allowedKeys.includes(key as keyof SettingsState)) { filteredUpdates[key as keyof SettingsState] = updates[key]; if (key === 'language') { // Check if the language update is valid before storing it for setLocale const langValue = updates[key]; if (langValue && availableLocales.includes(langValue)) { languageUpdate = langValue; // Store the valid language code } else { console.warn(`[SettingsStore] updateMultipleSettings: Received invalid language update '${langValue}'. Ignoring.`); } } } else { console.warn(`[SettingsStore] 尝试批量更新不允许的设置键: ${key}`); } } if (Object.keys(filteredUpdates).length === 0) { console.log('[SettingsStore] 没有有效的通用设置需要更新。'); return; // 没有有效设置需要更新 } try { // 注意:后端 controller 现在会过滤,但前端也做一层检查更好 await apiClient.put('/settings', filteredUpdates); // 使用 apiClient // Update store state *after* successful API call settings.value = { ...settings.value, ...filteredUpdates }; // If language is updated, apply it if (languageUpdate) { console.log(`[SettingsStore] updateMultipleSettings: Language updated to ${languageUpdate}. Calling setLocale...`); // <-- 添加日志 setLocale(languageUpdate); } } catch (err: any) { console.error('批量更新设置失败:', err); throw new Error(err.response?.data?.message || err.message || '批量更新设置失败'); } } /** * Updates the width for a specific sidebar pane. * @param paneName The name of the pane (component). * @param width The new width string (e.g., '400px'). */ async function updateSidebarPaneWidth(paneName: PaneName, width: string) { if (!paneName) return; const newWidths = { ...parsedSidebarPaneWidths.value, [paneName]: width }; parsedSidebarPaneWidths.value = newWidths; // Update local reactive state first try { // Use updateMultipleSettings for consistency, even for one setting await updateMultipleSettings({ sidebarPaneWidths: JSON.stringify(newWidths) }); } catch (error) { console.error(`[SettingsStore] Failed to save sidebarPaneWidths after updating ${paneName}:`, error); // Optionally revert local state or show error to user } } /** * Updates the File Manager layout settings (row size multiplier and column widths). * @param multiplier The new row size multiplier (number). * @param widths The new column widths object (Record). */ async function updateFileManagerLayoutSettings(multiplier: number, widths: Record) { const multiplierString = multiplier.toFixed(2); // Store with 2 decimal places const widthsString = JSON.stringify(widths); // Update local parsed state immediately for responsiveness parsedFileManagerColWidths.value = widths; // The multiplier is handled directly by the component, but update the setting value settings.value.fileManagerRowSizeMultiplier = multiplierString; settings.value.fileManagerColWidths = widthsString; try { console.log(`[SettingsStore] Saving FM layout: multiplier=${multiplierString}, widths=${widthsString}`); await updateMultipleSettings({ fileManagerRowSizeMultiplier: multiplierString, fileManagerColWidths: widthsString, }); } catch (error) { console.error('[SettingsStore] Failed to save file manager layout settings:', error); // Optionally revert local state or show error to user } } // --- CAPTCHA Settings Actions --- /** * Fetches CAPTCHA settings from the backend. * Should be called when the settings component mounts. */ async function loadCaptchaSettings() { // Avoid reloading if already loaded, unless forced // if (captchaSettings.value !== null && !force) return; isLoading.value = true; error.value = null; try { console.log('[SettingsStore] 加载 CAPTCHA 设置...'); // Use the correct endpoint defined in the backend routes const response = await apiClient.get('/settings/captcha'); captchaSettings.value = response.data; console.log('[SettingsStore] CAPTCHA 设置加载完成:', { ...response.data, hcaptchaSecretKey: '***', recaptchaSecretKey: '***' }); // Mask secrets } catch (err: any) { console.error('加载 CAPTCHA 设置失败:', err); error.value = err.response?.data?.message || err.message || '加载 CAPTCHA 设置失败'; captchaSettings.value = null; // Reset on error } finally { isLoading.value = false; } } /** * Updates CAPTCHA settings on the backend. * @param updates - An object containing the CAPTCHA settings fields to update. */ async function updateCaptchaSettings(updates: UpdateCaptchaSettingsDto) { isLoading.value = true; error.value = null; try { console.log('[SettingsStore] 更新 CAPTCHA 设置:', { ...updates, hcaptchaSecretKey: '***', recaptchaSecretKey: '***' }); // Mask secrets // Use the correct endpoint defined in the backend routes await apiClient.put('/settings/captcha', updates); // Update local state after successful API call // Merge updates into the existing state or reload if (captchaSettings.value) { captchaSettings.value = { ...captchaSettings.value, ...updates }; } else { // If settings were null, reload them after update await loadCaptchaSettings(); } console.log('[SettingsStore] CAPTCHA 设置更新成功。'); // --- 新增:强制 authStore 重新获取配置 --- console.log('[SettingsStore] Triggering authStore to refetch CAPTCHA config...'); authStore.publicCaptchaConfig = null; // 重置 authStore 的状态以允许重新获取 await authStore.fetchCaptchaConfig(); // 让 authStore 立即获取最新的配置 // ----------------------------------------- } catch (err: any) { console.error('更新 CAPTCHA 设置失败:', err); error.value = err.response?.data?.message || err.message || '更新 CAPTCHA 设置失败'; throw error; // Re-throw to allow component to handle UI feedback } finally { isLoading.value = false; } } async function saveDashboardSortPreference(sortBy: SortField, sortOrder: SortOrder) { try { await updateMultipleSettings({ dashboardSortBy: sortBy, dashboardSortOrder: sortOrder, }); } catch (error) { console.error('[SettingsStore] Failed to save dashboard sort preference:', error); // Optionally show error to user } } // 移除外观相关 actions: saveCustomThemes, resetCustomThemes, toggleStyleCustomizer // --- Getters --- // Use defaultLng (which is now 'en-US' or the first available) from i18n.ts as the final fallback const language = computed(() => settings.value.language || defaultLng); // Getter for the popup editor setting, returning boolean const showPopupFileEditorBoolean = computed(() => { return settings.value.showPopupFileEditor !== 'false'; }); // Getter for sharing setting, returning boolean const shareFileEditorTabsBoolean = computed(() => { return settings.value.shareFileEditorTabs !== 'false'; }); // Getter for IP Whitelist enabled status const ipWhitelistEnabled = computed(() => settings.value.ipWhitelistEnabled === 'true'); // <-- NEW: Getter for IP Blacklist enabled status --> const ipBlacklistEnabledBoolean = computed(() => { // Default to true if the setting is missing or not 'false' return settings.value.ipBlacklistEnabled !== 'false'; }); // Getter for auto copy on select setting, returning boolean const autoCopyOnSelectBoolean = computed(() => { return settings.value.autoCopyOnSelect === 'true'; }); // NEW: Getter for workspace sidebar persistent setting, returning boolean const workspaceSidebarPersistentBoolean = computed(() => { return settings.value.workspaceSidebarPersistent === 'true'; }); // NEW: Getter to get width for a specific sidebar pane const getSidebarPaneWidth = computed(() => (paneName: PaneName | null): string => { const defaultWidth = '350px'; if (!paneName) return defaultWidth; // Ensure parsedSidebarPaneWidths.value is accessed correctly const widths = parsedSidebarPaneWidths.value || {}; return widths[paneName] || defaultWidth; }); // NEW: Getter for Docker default expand setting, returning boolean const dockerDefaultExpandBoolean = computed(() => { return settings.value.dockerDefaultExpand === 'true'; }); // NEW: Getter for Status Monitor interval, returning number const statusMonitorIntervalSecondsNumber = computed(() => { const val = parseInt(settings.value.statusMonitorIntervalSeconds || '3', 10); return isNaN(val) || val <= 0 ? 3 : val; // Fallback to 3 if invalid }); // NEW: Getter for File Manager row size multiplier, returning number const fileManagerRowSizeMultiplierNumber = computed(() => { const val = parseFloat(settings.value.fileManagerRowSizeMultiplier || '1.0'); return isNaN(val) || val <= 0 ? 1.0 : val; // Fallback to 1.0 if invalid }); // NEW: Getter for File Manager column widths, returning object const fileManagerColWidthsObject = computed(() => { // Return the reactive ref directly, which is updated during load and save return parsedFileManagerColWidths.value; }); // NEW: Getter for command input sync target const commandInputSyncTarget = computed(() => { const target = settings.value.commandInputSyncTarget; if (target === 'quickCommands' || target === 'commandHistory') { return target; } return 'none'; // Default to 'none' if invalid or not set }); // NEW: Getter for timezone setting const timezone = computed(() => settings.value.timezone || 'UTC'); const dashboardSortBy = computed((): SortField => { const savedSortBy = settings.value.dashboardSortBy; const validFields: SortField[] = ['created_at', 'last_connected_at', 'updated_at', 'name', 'type']; return savedSortBy && validFields.includes(savedSortBy) ? savedSortBy : 'last_connected_at'; }); const dashboardSortOrder = computed((): SortOrder => { const savedSortOrder = settings.value.dashboardSortOrder; return savedSortOrder === 'asc' || savedSortOrder === 'desc' ? savedSortOrder : 'desc'; }); const isCaptchaEnabled = computed(() => captchaSettings.value?.enabled ?? false); const captchaProvider = computed(() => captchaSettings.value?.provider ?? 'none'); const hcaptchaSiteKey = computed(() => captchaSettings.value?.hcaptchaSiteKey ?? ''); const recaptchaSiteKey = computed(() => captchaSettings.value?.recaptchaSiteKey ?? ''); // DO NOT expose secret keys via getters // NEW: Getters for tag visibility const showConnectionTagsBoolean = computed(() => { return settings.value.showConnectionTags !== 'false'; // Default to true }); const showQuickCommandTagsBoolean = computed(() => { return settings.value.showQuickCommandTags !== 'false'; // Default to true }); // NEW: Getter for layout locked status const layoutLockedBoolean = computed(() => { return settings.value.layoutLocked === 'true'; }); // NEW: Getter for terminal scrollback limit, returning number (0 means Infinity for xterm) const terminalScrollbackLimitNumber = computed(() => { const valStr = settings.value.terminalScrollbackLimit; if (valStr === null || valStr === undefined || valStr.trim() === '') { return 5000; // Default value if not set or empty } const val = parseInt(valStr, 10); if (isNaN(val) || val < 0) { return 5000; // Default value if invalid number or negative } return val; // Return 0 if it's 0, or the positive number }); return { settings, // 只包含通用设置 isLoading, error, language, showPopupFileEditorBoolean, shareFileEditorTabsBoolean, ipWhitelistEnabled, // 暴露 IP 白名单启用状态 ipBlacklistEnabledBoolean, // <-- NEW: 暴露 IP 黑名单启用状态 getter autoCopyOnSelectBoolean, dockerDefaultExpandBoolean, // +++ 暴露 Docker 默认展开 getter +++ statusMonitorIntervalSecondsNumber, // +++ 暴露状态监控间隔 getter +++ workspaceSidebarPersistentBoolean, // +++ 暴露侧边栏固定 getter +++ getSidebarPaneWidth, // +++ 暴露获取特定面板宽度的 getter +++ fileManagerRowSizeMultiplierNumber, // +++ 暴露文件管理器行大小 getter +++ fileManagerColWidthsObject, // +++ 暴露文件管理器列宽 getter +++ // CAPTCHA related exports captchaSettings, // Expose the full (but reactive) object for the settings page v-model isCaptchaEnabled, captchaProvider, hcaptchaSiteKey, recaptchaSiteKey, loadCaptchaSettings, updateCaptchaSettings, // 移除外观相关的 getters 和 actions loadInitialSettings, updateSetting, updateMultipleSettings, updateSidebarPaneWidth, // +++ 暴露更新特定面板宽度的 action +++ updateFileManagerLayoutSettings, // +++ 暴露更新文件管理器布局的 action +++ commandInputSyncTarget, // +++ 暴露命令输入同步目标 getter +++ timezone, dashboardSortBy, dashboardSortOrder, saveDashboardSortPreference, // NEW: Expose tag visibility getters showConnectionTagsBoolean, showQuickCommandTagsBoolean, // NEW: Expose layout locked getter layoutLockedBoolean, terminalScrollbackLimitNumber, // NEW: Expose terminal scrollback limit getter }; });