fix: grant set_skip_taskbar permission through default capability chore: align settings casing and defaults between Rust and frontend Co-authored-by: Jason <farion1231@gmail.com>
319 lines
8.3 KiB
TypeScript
319 lines
8.3 KiB
TypeScript
import { invoke } from "@tauri-apps/api/core";
|
||
import { listen, UnlistenFn } from "@tauri-apps/api/event";
|
||
import { Provider, Settings } from "../types";
|
||
|
||
// 应用类型
|
||
export type AppType = "claude" | "codex";
|
||
|
||
// 定义配置状态类型
|
||
interface ConfigStatus {
|
||
exists: boolean;
|
||
path: string;
|
||
error?: string;
|
||
}
|
||
|
||
// 定义导入结果类型
|
||
interface ImportResult {
|
||
success: boolean;
|
||
message?: string;
|
||
}
|
||
|
||
// Tauri API 封装,提供统一的全局 API 接口
|
||
export const tauriAPI = {
|
||
// 获取所有供应商
|
||
getProviders: async (app?: AppType): Promise<Record<string, Provider>> => {
|
||
try {
|
||
return await invoke("get_providers", { app_type: app, app });
|
||
} catch (error) {
|
||
console.error("获取供应商列表失败:", error);
|
||
return {};
|
||
}
|
||
},
|
||
|
||
// 获取当前供应商ID
|
||
getCurrentProvider: async (app?: AppType): Promise<string> => {
|
||
try {
|
||
return await invoke("get_current_provider", { app_type: app, app });
|
||
} catch (error) {
|
||
console.error("获取当前供应商失败:", error);
|
||
return "";
|
||
}
|
||
},
|
||
|
||
// 添加供应商
|
||
addProvider: async (provider: Provider, app?: AppType): Promise<boolean> => {
|
||
try {
|
||
return await invoke("add_provider", { provider, app_type: app, app });
|
||
} catch (error) {
|
||
console.error("添加供应商失败:", error);
|
||
throw error;
|
||
}
|
||
},
|
||
|
||
// 更新供应商
|
||
updateProvider: async (
|
||
provider: Provider,
|
||
app?: AppType,
|
||
): Promise<boolean> => {
|
||
try {
|
||
return await invoke("update_provider", { provider, app_type: app, app });
|
||
} catch (error) {
|
||
console.error("更新供应商失败:", error);
|
||
throw error;
|
||
}
|
||
},
|
||
|
||
// 删除供应商
|
||
deleteProvider: async (id: string, app?: AppType): Promise<boolean> => {
|
||
try {
|
||
return await invoke("delete_provider", { id, app_type: app, app });
|
||
} catch (error) {
|
||
console.error("删除供应商失败:", error);
|
||
throw error;
|
||
}
|
||
},
|
||
|
||
// 切换供应商
|
||
switchProvider: async (
|
||
providerId: string,
|
||
app?: AppType,
|
||
): Promise<boolean> => {
|
||
try {
|
||
return await invoke("switch_provider", {
|
||
id: providerId,
|
||
app_type: app,
|
||
app,
|
||
});
|
||
} catch (error) {
|
||
console.error("切换供应商失败:", error);
|
||
return false;
|
||
}
|
||
},
|
||
|
||
// 导入当前配置为默认供应商
|
||
importCurrentConfigAsDefault: async (
|
||
app?: AppType,
|
||
): Promise<ImportResult> => {
|
||
try {
|
||
const success = await invoke<boolean>("import_default_config", {
|
||
app_type: app,
|
||
app,
|
||
});
|
||
return {
|
||
success,
|
||
message: success ? "成功导入默认配置" : "导入失败",
|
||
};
|
||
} catch (error) {
|
||
console.error("导入默认配置失败:", error);
|
||
return {
|
||
success: false,
|
||
message: String(error),
|
||
};
|
||
}
|
||
},
|
||
|
||
// 获取 Claude Code 配置文件路径
|
||
getClaudeCodeConfigPath: async (): Promise<string> => {
|
||
try {
|
||
return await invoke("get_claude_code_config_path");
|
||
} catch (error) {
|
||
console.error("获取配置路径失败:", error);
|
||
return "";
|
||
}
|
||
},
|
||
|
||
// 获取当前生效的配置目录
|
||
getConfigDir: async (app?: AppType): Promise<string> => {
|
||
try {
|
||
return await invoke("get_config_dir", { app_type: app, app });
|
||
} catch (error) {
|
||
console.error("获取配置目录失败:", error);
|
||
return "";
|
||
}
|
||
},
|
||
|
||
// 获取 Claude Code 配置状态
|
||
getClaudeConfigStatus: async (): Promise<ConfigStatus> => {
|
||
try {
|
||
return await invoke("get_claude_config_status");
|
||
} catch (error) {
|
||
console.error("获取配置状态失败:", error);
|
||
return {
|
||
exists: false,
|
||
path: "",
|
||
error: String(error),
|
||
};
|
||
}
|
||
},
|
||
|
||
// 获取应用配置状态(通用)
|
||
getConfigStatus: async (app?: AppType): Promise<ConfigStatus> => {
|
||
try {
|
||
return await invoke("get_config_status", { app_type: app, app });
|
||
} catch (error) {
|
||
console.error("获取配置状态失败:", error);
|
||
return {
|
||
exists: false,
|
||
path: "",
|
||
error: String(error),
|
||
};
|
||
}
|
||
},
|
||
|
||
// 打开配置文件夹
|
||
openConfigFolder: async (app?: AppType): Promise<void> => {
|
||
try {
|
||
await invoke("open_config_folder", { app_type: app, app });
|
||
} catch (error) {
|
||
console.error("打开配置文件夹失败:", error);
|
||
}
|
||
},
|
||
|
||
// 打开外部链接
|
||
openExternal: async (url: string): Promise<void> => {
|
||
try {
|
||
await invoke("open_external", { url });
|
||
} catch (error) {
|
||
console.error("打开外部链接失败:", error);
|
||
}
|
||
},
|
||
|
||
// 更新托盘菜单
|
||
updateTrayMenu: async (): Promise<boolean> => {
|
||
try {
|
||
return await invoke("update_tray_menu");
|
||
} catch (error) {
|
||
console.error("更新托盘菜单失败:", error);
|
||
return false;
|
||
}
|
||
},
|
||
|
||
// 监听供应商切换事件
|
||
onProviderSwitched: async (
|
||
callback: (data: { appType: string; providerId: string }) => void,
|
||
): Promise<UnlistenFn> => {
|
||
return await listen("provider-switched", (event) => {
|
||
callback(event.payload as { appType: string; providerId: string });
|
||
});
|
||
},
|
||
|
||
// (保留空位,取消迁移提示)
|
||
|
||
// 选择配置目录
|
||
selectConfigDirectory: async (
|
||
defaultPath?: string,
|
||
): Promise<string | null> => {
|
||
try {
|
||
const sanitized =
|
||
defaultPath && defaultPath.trim() !== ""
|
||
? defaultPath
|
||
: undefined;
|
||
return await invoke<string | null>("pick_directory", {
|
||
defaultPath: sanitized,
|
||
});
|
||
} catch (error) {
|
||
console.error("选择配置目录失败:", error);
|
||
return null;
|
||
}
|
||
},
|
||
|
||
// 获取设置
|
||
getSettings: async (): Promise<Settings> => {
|
||
try {
|
||
return await invoke("get_settings");
|
||
} catch (error) {
|
||
console.error("获取设置失败:", error);
|
||
return { showInTray: true, minimizeToTrayOnClose: true };
|
||
}
|
||
},
|
||
|
||
// 保存设置
|
||
saveSettings: async (settings: Settings): Promise<boolean> => {
|
||
try {
|
||
return await invoke("save_settings", { settings });
|
||
} catch (error) {
|
||
console.error("保存设置失败:", error);
|
||
return false;
|
||
}
|
||
},
|
||
|
||
// 检查更新
|
||
checkForUpdates: async (): Promise<void> => {
|
||
try {
|
||
await invoke("check_for_updates");
|
||
} catch (error) {
|
||
console.error("检查更新失败:", error);
|
||
}
|
||
},
|
||
|
||
// 判断是否为便携模式
|
||
isPortable: async (): Promise<boolean> => {
|
||
try {
|
||
return await invoke<boolean>("is_portable_mode");
|
||
} catch (error) {
|
||
console.error("检测便携模式失败:", error);
|
||
return false;
|
||
}
|
||
},
|
||
|
||
// 获取应用配置文件路径
|
||
getAppConfigPath: async (): Promise<string> => {
|
||
try {
|
||
return await invoke("get_app_config_path");
|
||
} catch (error) {
|
||
console.error("获取应用配置路径失败:", error);
|
||
return "";
|
||
}
|
||
},
|
||
|
||
// 打开应用配置文件夹
|
||
openAppConfigFolder: async (): Promise<void> => {
|
||
try {
|
||
await invoke("open_app_config_folder");
|
||
} catch (error) {
|
||
console.error("打开应用配置文件夹失败:", error);
|
||
}
|
||
},
|
||
|
||
// VS Code: 获取 settings.json 状态
|
||
getVSCodeSettingsStatus: async (): Promise<{
|
||
exists: boolean;
|
||
path: string;
|
||
error?: string;
|
||
}> => {
|
||
try {
|
||
return await invoke("get_vscode_settings_status");
|
||
} catch (error) {
|
||
console.error("获取 VS Code 设置状态失败:", error);
|
||
return { exists: false, path: "", error: String(error) };
|
||
}
|
||
},
|
||
|
||
// VS Code: 读取 settings.json 文本
|
||
readVSCodeSettings: async (): Promise<string> => {
|
||
try {
|
||
return await invoke("read_vscode_settings");
|
||
} catch (error) {
|
||
throw new Error(`读取 VS Code 设置失败: ${String(error)}`);
|
||
}
|
||
},
|
||
|
||
// VS Code: 写回 settings.json 文本(不自动创建)
|
||
writeVSCodeSettings: async (content: string): Promise<boolean> => {
|
||
try {
|
||
return await invoke("write_vscode_settings", { content });
|
||
} catch (error) {
|
||
throw new Error(`写入 VS Code 设置失败: ${String(error)}`);
|
||
}
|
||
},
|
||
};
|
||
|
||
// 创建全局 API 对象,兼容现有代码
|
||
if (typeof window !== "undefined") {
|
||
// 绑定到 window.api,避免 Electron 命名造成误解
|
||
// API 内部已做 try/catch,非 Tauri 环境下也会安全返回默认值
|
||
(window as any).api = tauriAPI;
|
||
}
|
||
|
||
export default tauriAPI;
|