Files
nexus-terminal/packages/frontend/src/stores/settings.store.ts
T
2025-05-05 17:06:10 +08:00

784 lines
37 KiB
TypeScript

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<ConnectionInfo, 'created_at' | 'last_connected_at' | 'updated_at' | 'name' | 'type'>;
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<Partial<SettingsState>>({}); // 通用设置状态
const parsedSidebarPaneWidths = ref<Record<string, string>>({}); // NEW: 解析后的侧边栏宽度对象
const parsedFileManagerColWidths = ref<Record<string, number>>({}); // NEW: 解析后的文件管理器列宽对象
const captchaSettings = ref<CaptchaSettings | null>(null); // NEW: CAPTCHA 设置状态
const isLoading = ref(false);
const error = ref<string | null>(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<Record<string, string>>('/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<string, string> = {};
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<string, string> = {};
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<string, number> = {};
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<string, number> = { ...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<keyof SettingsState> = [
'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<Record<keyof SettingsState, string>> = {
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<any>;
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<SettingsState>) {
// 移除外观相关的键检查
const allowedKeys: Array<keyof SettingsState> = [
'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<SettingsState> = {};
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<string, number>).
*/
async function updateFileManagerLayoutSettings(multiplier: number, widths: Record<string, number>) {
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<CaptchaSettings>('/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
};
});