Compare commits
82 Commits
feat/add-p
...
feature/sk
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
be1c2ac76e | ||
|
|
e7451bda22 | ||
|
|
5a3420932b | ||
|
|
a2688603fb | ||
|
|
23a407544a | ||
|
|
2b34dc4ec9 | ||
|
|
529051f0e8 | ||
|
|
5d1eed563d | ||
|
|
cc0b9352bc | ||
|
|
01d8bb53ac | ||
|
|
6e7547ef6e | ||
|
|
d38fcd63ea | ||
|
|
824bf796a8 | ||
|
|
f02efbd2b7 | ||
|
|
a39b1d8698 | ||
|
|
86255fe106 | ||
|
|
4acd48adc9 | ||
|
|
d4cd8105d1 | ||
|
|
2b1ae2aa71 | ||
|
|
cf57fbed7b | ||
|
|
b64bb6cfa1 | ||
|
|
d65513ae7d | ||
|
|
eefb764f72 | ||
|
|
4ed3e3bf84 | ||
|
|
29e32f73f3 | ||
|
|
99471f6706 | ||
|
|
4210b1547c | ||
|
|
cfee4d6fcc | ||
|
|
24dc628130 | ||
|
|
bf74620051 | ||
|
|
e8d4397b3a | ||
|
|
2b0bc73276 | ||
|
|
939a2e4f2b | ||
|
|
1a89267986 | ||
|
|
127fa5bf9d | ||
|
|
0d4be40c25 | ||
|
|
00720ecf30 | ||
|
|
de7f93d513 | ||
|
|
e7545f8cdf | ||
|
|
eb46ac8592 | ||
|
|
838a99b5d2 | ||
|
|
325c6a5f21 | ||
|
|
8824462e4c | ||
|
|
ba336fc416 | ||
|
|
0c1d94e57b | ||
|
|
636a1e2c60 | ||
|
|
a56a578e91 | ||
|
|
c582be265b | ||
|
|
7fa0a7b166 | ||
|
|
5e54656d45 | ||
|
|
8f218057f3 | ||
|
|
81a6c08673 | ||
|
|
988ea326d9 | ||
|
|
f1b0fa2985 | ||
|
|
3f470de608 | ||
|
|
03af3600b0 | ||
|
|
482b8a1cab | ||
|
|
ddb0b68b4c | ||
|
|
524fa94339 | ||
|
|
162c92144c | ||
|
|
b075ee9fbb | ||
|
|
17cf701bad | ||
|
|
977185e2d5 | ||
|
|
764ba81ea6 | ||
|
|
d802b7bf61 | ||
|
|
74969ae968 | ||
|
|
1f3627add3 | ||
|
|
14ee122b27 | ||
|
|
7aecba14fe | ||
|
|
99b5f881e8 | ||
|
|
286bafbd67 | ||
|
|
6046cf8767 | ||
|
|
c88afa365f | ||
|
|
93fa5fe29a | ||
|
|
3d31ad64af | ||
|
|
bb0951552d | ||
|
|
00e3e6fa70 | ||
|
|
1ce007622e | ||
|
|
436f0e8e42 | ||
|
|
3d69da5b66 | ||
|
|
0ae9ed5a17 | ||
|
|
5ff689af82 |
247
CHANGELOG.md
247
CHANGELOG.md
@@ -5,6 +5,247 @@ All notable changes to CC Switch will be documented in this file.
|
|||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## [3.7.1] - 2025-11-22
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- **Skills third-party repository installation** (#268) - Fixed installation failure for skills repositories with custom subdirectories (e.g., `ComposioHQ/awesome-claude-skills`)
|
||||||
|
- **Gemini configuration persistence** - Resolved issue where settings.json edits were lost when switching providers
|
||||||
|
- **Dialog overlay click protection** - Prevented dialogs from closing when clicking outside, avoiding accidental form data loss (affects 11 dialog components)
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- **Gemini configuration directory support** (#255) - Added custom configuration directory option for Gemini in settings
|
||||||
|
- **ArchLinux installation support** (#259) - Added AUR installation via `paru -S cc-switch-bin`
|
||||||
|
|
||||||
|
### Improved
|
||||||
|
|
||||||
|
- **Skills error messages i18n** - Added 28+ detailed error messages (English & Chinese) with specific resolution suggestions
|
||||||
|
- **Download timeout** - Extended from 15s to 60s to reduce network-related false positives
|
||||||
|
- **Code formatting** - Applied unified Rust (`cargo fmt`) and TypeScript (`prettier`) formatting standards
|
||||||
|
|
||||||
|
### Reverted
|
||||||
|
|
||||||
|
- **Auto-launch on system startup** - Temporarily reverted feature pending further testing and optimization
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## [3.7.0] - 2025-11-19
|
||||||
|
|
||||||
|
### Major Features
|
||||||
|
|
||||||
|
#### Gemini CLI Integration
|
||||||
|
|
||||||
|
- **Complete Gemini CLI support** - Third major application added alongside Claude Code and Codex
|
||||||
|
- **Dual-file configuration** - Support for both `.env` and `settings.json` file formats
|
||||||
|
- **Environment variable detection** - Auto-detect `GOOGLE_GEMINI_BASE_URL`, `GEMINI_MODEL`, etc.
|
||||||
|
- **MCP management** - Full MCP configuration capabilities for Gemini
|
||||||
|
- **Provider presets**
|
||||||
|
- Google Official (OAuth authentication)
|
||||||
|
- PackyCode (partner integration)
|
||||||
|
- Custom endpoint support
|
||||||
|
- **Deep link support** - Import Gemini providers via `ccswitch://` protocol
|
||||||
|
- **System tray integration** - Quick-switch Gemini providers from tray menu
|
||||||
|
- **Backend modules** - New `gemini_config.rs` (20KB) and `gemini_mcp.rs`
|
||||||
|
|
||||||
|
#### MCP v3.7.0 Unified Architecture
|
||||||
|
|
||||||
|
- **Unified management panel** - Single interface for Claude/Codex/Gemini MCP servers
|
||||||
|
- **SSE transport type** - New Server-Sent Events support alongside stdio/http
|
||||||
|
- **Smart JSON parser** - Fault-tolerant parsing of various MCP config formats
|
||||||
|
- **Extended field support** - Preserve custom fields in Codex TOML conversion
|
||||||
|
- **Codex format correction** - Proper `[mcp_servers]` format (auto-cleanup of incorrect `[mcp.servers]`)
|
||||||
|
- **Import/export system** - Unified import from Claude/Codex/Gemini live configs
|
||||||
|
- **UX improvements**
|
||||||
|
- Default app selection in forms
|
||||||
|
- JSON formatter for config validation
|
||||||
|
- Improved layout and visual hierarchy
|
||||||
|
- Better validation error messages
|
||||||
|
|
||||||
|
#### Claude Skills Management System
|
||||||
|
|
||||||
|
- **GitHub repository integration** - Auto-scan and discover skills from GitHub repos
|
||||||
|
- **Pre-configured repositories**
|
||||||
|
- `ComposioHQ/awesome-claude-skills` (curated collection)
|
||||||
|
- `anthropics/skills` (official Anthropic skills)
|
||||||
|
- `cexll/myclaude` (community, with subdirectory scanning)
|
||||||
|
- **Lifecycle management**
|
||||||
|
- One-click install to `~/.claude/skills/`
|
||||||
|
- Safe uninstall with state tracking
|
||||||
|
- Update checking (infrastructure ready)
|
||||||
|
- **Custom repository support** - Add any GitHub repo as a skill source
|
||||||
|
- **Subdirectory scanning** - Optional `skillsPath` for repos with nested skill directories
|
||||||
|
- **Backend architecture** - `SkillService` (526 lines) with GitHub API integration
|
||||||
|
- **Frontend interface**
|
||||||
|
- SkillsPage: Browse and manage skills
|
||||||
|
- SkillCard: Visual skill presentation
|
||||||
|
- RepoManager: Repository management dialog
|
||||||
|
- **State persistence** - Installation state stored in `skills.json`
|
||||||
|
- **Full i18n support** - Complete Chinese/English translations (47+ keys)
|
||||||
|
|
||||||
|
#### Prompts (System Prompts) Management
|
||||||
|
|
||||||
|
- **Multi-preset management** - Create, edit, and switch between multiple system prompts
|
||||||
|
- **Cross-app support**
|
||||||
|
- Claude: `~/.claude/CLAUDE.md`
|
||||||
|
- Codex: `~/.codex/AGENTS.md`
|
||||||
|
- Gemini: `~/.gemini/GEMINI.md`
|
||||||
|
- **Markdown editor** - Full-featured CodeMirror 6 editor with syntax highlighting
|
||||||
|
- **Smart synchronization**
|
||||||
|
- Auto-write to live files on enable
|
||||||
|
- Content backfill protection (save current before switching)
|
||||||
|
- First-launch auto-import from live files
|
||||||
|
- **Single-active enforcement** - Only one prompt can be active at a time
|
||||||
|
- **Delete protection** - Cannot delete active prompts
|
||||||
|
- **Backend service** - `PromptService` (213 lines) with CRUD operations
|
||||||
|
- **Frontend components**
|
||||||
|
- PromptPanel: Main management interface (177 lines)
|
||||||
|
- PromptFormModal: Edit dialog with validation (160 lines)
|
||||||
|
- MarkdownEditor: CodeMirror integration (159 lines)
|
||||||
|
- usePromptActions: Business logic hook (152 lines)
|
||||||
|
- **Full i18n support** - Complete Chinese/English translations (41+ keys)
|
||||||
|
|
||||||
|
#### Deep Link Protocol (ccswitch://)
|
||||||
|
|
||||||
|
- **Protocol registration** - `ccswitch://` URL scheme for one-click imports
|
||||||
|
- **Provider import** - Import provider configurations from URLs or shared links
|
||||||
|
- **Lifecycle integration** - Deep link handling integrated into app startup
|
||||||
|
- **Cross-platform support** - Works on Windows, macOS, and Linux
|
||||||
|
|
||||||
|
#### Environment Variable Conflict Detection
|
||||||
|
|
||||||
|
- **Claude & Codex detection** - Identify conflicting environment variables
|
||||||
|
- **Gemini auto-detection** - Automatic environment variable discovery
|
||||||
|
- **Conflict management** - UI for resolving configuration conflicts
|
||||||
|
- **Prevention system** - Warn before overwriting existing configurations
|
||||||
|
|
||||||
|
### New Features
|
||||||
|
|
||||||
|
#### Provider Management
|
||||||
|
|
||||||
|
- **DouBaoSeed preset** - Added ByteDance's DouBao provider
|
||||||
|
- **Kimi For Coding** - Moonshot AI coding assistant
|
||||||
|
- **BaiLing preset** - BaiLing AI integration
|
||||||
|
- **Removed AnyRouter preset** - Discontinued provider
|
||||||
|
- **Model configuration** - Support for custom model names in Codex and Gemini
|
||||||
|
- **Provider notes field** - Add custom notes to providers for better organization
|
||||||
|
|
||||||
|
#### Configuration Management
|
||||||
|
|
||||||
|
- **Common config migration** - Moved Claude common config snippets from localStorage to `config.json`
|
||||||
|
- **Unified persistence** - Common config snippets now shared across all apps
|
||||||
|
- **Auto-import on first launch** - Automatically import configs from live files on first run
|
||||||
|
- **Backfill priority fix** - Correct priority handling when enabling prompts
|
||||||
|
|
||||||
|
#### UI/UX Improvements
|
||||||
|
|
||||||
|
- **macOS native design** - Migrated color scheme to macOS native design system
|
||||||
|
- **Window centering** - Default window position centered on screen
|
||||||
|
- **Password input fixes** - Disabled Edge/IE reveal and clear buttons
|
||||||
|
- **URL overflow prevention** - Fixed overflow in provider cards
|
||||||
|
- **Error notification enhancement** - Copy-to-clipboard for error messages
|
||||||
|
- **Tray menu sync** - Real-time sync after drag-and-drop sorting
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
|
||||||
|
#### Architecture
|
||||||
|
|
||||||
|
- **MCP v3.7.0 cleanup** - Removed legacy code and warnings
|
||||||
|
- **Unified structure** - Default initialization with v3.7.0 unified structure
|
||||||
|
- **Backward compatibility** - Compilation fixes for older configs
|
||||||
|
- **Code formatting** - Applied consistent formatting across backend and frontend
|
||||||
|
|
||||||
|
#### Platform Compatibility
|
||||||
|
|
||||||
|
- **Windows fix** - Resolved winreg API compatibility issue (v0.52)
|
||||||
|
- **Safe pattern matching** - Replaced `unwrap()` with safe patterns in tray menu
|
||||||
|
|
||||||
|
#### Configuration
|
||||||
|
|
||||||
|
- **MCP sync on switch** - Sync MCP configs for all apps when switching providers
|
||||||
|
- **Gemini form sync** - Fixed form fields syncing with environment editor
|
||||||
|
- **Gemini config reading** - Read from both `.env` and `settings.json`
|
||||||
|
- **Validation improvements** - Enhanced input validation and boundary checks
|
||||||
|
|
||||||
|
#### Internationalization
|
||||||
|
|
||||||
|
- **JSON syntax fixes** - Resolved syntax errors in locale files
|
||||||
|
- **App name i18n** - Added internationalization support for app names
|
||||||
|
- **Deduplicated labels** - Reused providerForm keys to reduce duplication
|
||||||
|
- **Gemini MCP title** - Added missing Gemini MCP panel title
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
#### Critical Fixes
|
||||||
|
|
||||||
|
- **Usage script validation** - Added input validation and boundary checks
|
||||||
|
- **Gemini validation** - Relaxed validation when adding providers
|
||||||
|
- **TOML quote normalization** - Handle CJK quotes to prevent parsing errors
|
||||||
|
- **MCP field preservation** - Preserve custom fields in Codex TOML editor
|
||||||
|
- **Password input** - Fixed white screen crash (FormLabel → Label)
|
||||||
|
|
||||||
|
#### Stability
|
||||||
|
|
||||||
|
- **Tray menu safety** - Replaced unwrap with safe pattern matching
|
||||||
|
- **Error isolation** - Tray menu update failures don't block main operations
|
||||||
|
- **Import classification** - Set category to custom for imported default configs
|
||||||
|
|
||||||
|
#### UI Fixes
|
||||||
|
|
||||||
|
- **Model placeholders** - Removed misleading model input placeholders
|
||||||
|
- **Base URL population** - Auto-fill base URL for non-official providers
|
||||||
|
- **Drag sort sync** - Fixed tray menu order after drag-and-drop
|
||||||
|
|
||||||
|
### Technical Improvements
|
||||||
|
|
||||||
|
#### Code Quality
|
||||||
|
|
||||||
|
- **Type safety** - Complete TypeScript type coverage across codebase
|
||||||
|
- **Test improvements** - Simplified boolean assertions in tests
|
||||||
|
- **Clippy warnings** - Fixed `uninlined_format_args` warnings
|
||||||
|
- **Code refactoring** - Extracted templates, optimized logic flows
|
||||||
|
|
||||||
|
#### Dependencies
|
||||||
|
|
||||||
|
- **Tauri** - Updated to 2.8.x series
|
||||||
|
- **Rust dependencies** - Added `anyhow`, `zip`, `serde_yaml`, `tempfile` for Skills
|
||||||
|
- **Frontend dependencies** - Added CodeMirror 6 packages for Markdown editor
|
||||||
|
- **winreg** - Updated to v0.52 (Windows compatibility)
|
||||||
|
|
||||||
|
#### Performance
|
||||||
|
|
||||||
|
- **Startup optimization** - Removed legacy migration scanning
|
||||||
|
- **Lock management** - Improved RwLock usage to prevent deadlocks
|
||||||
|
- **Background query** - Enabled background mode for usage polling
|
||||||
|
|
||||||
|
### Statistics
|
||||||
|
|
||||||
|
- **Total commits**: 85 commits from v3.6.0 to v3.7.0
|
||||||
|
- **Code changes**: 152 files changed, 18,104 insertions(+), 3,732 deletions(-)
|
||||||
|
- **New modules**:
|
||||||
|
- Skills: 2,034 lines (21 files)
|
||||||
|
- Prompts: 1,302 lines (20 files)
|
||||||
|
- Gemini: ~1,000 lines (multiple files)
|
||||||
|
- MCP refactor: ~3,000 lines (refactored)
|
||||||
|
|
||||||
|
### Strategic Positioning
|
||||||
|
|
||||||
|
v3.7.0 represents a major evolution from "Provider Switcher" to **"All-in-One AI CLI Management Platform"**:
|
||||||
|
|
||||||
|
1. **Capability Extension** - Skills provide external ability integration
|
||||||
|
2. **Behavior Customization** - Prompts enable AI personality presets
|
||||||
|
3. **Configuration Unification** - MCP v3.7.0 eliminates app silos
|
||||||
|
4. **Ecosystem Openness** - Deep links enable community sharing
|
||||||
|
5. **Multi-AI Support** - Claude/Codex/Gemini trinity
|
||||||
|
6. **Intelligent Detection** - Auto-discovery of environment conflicts
|
||||||
|
|
||||||
|
### Notes
|
||||||
|
|
||||||
|
- Users upgrading from v3.1.0 or earlier should first upgrade to v3.2.x for one-time migration
|
||||||
|
- Skills and Prompts management are new features requiring no migration
|
||||||
|
- Gemini CLI support requires Gemini CLI to be installed separately
|
||||||
|
- MCP v3.7.0 unified structure is backward compatible with previous configs
|
||||||
|
|
||||||
## [3.6.0] - 2025-11-07
|
## [3.6.0] - 2025-11-07
|
||||||
|
|
||||||
### ✨ New Features
|
### ✨ New Features
|
||||||
@@ -73,6 +314,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
### 🏗️ Technical Improvements (For Developers)
|
### 🏗️ Technical Improvements (For Developers)
|
||||||
|
|
||||||
**Backend Refactoring (Rust)** - Completed 5-phase refactoring:
|
**Backend Refactoring (Rust)** - Completed 5-phase refactoring:
|
||||||
|
|
||||||
- **Phase 1**: Unified error handling (`AppError` + i18n error messages)
|
- **Phase 1**: Unified error handling (`AppError` + i18n error messages)
|
||||||
- **Phase 2**: Command layer split by domain (`commands/{provider,mcp,config,settings,plugin,misc}.rs`)
|
- **Phase 2**: Command layer split by domain (`commands/{provider,mcp,config,settings,plugin,misc}.rs`)
|
||||||
- **Phase 3**: Integration tests and transaction mechanism (config snapshot + failure rollback)
|
- **Phase 3**: Integration tests and transaction mechanism (config snapshot + failure rollback)
|
||||||
@@ -80,17 +322,20 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
- **Phase 5**: Concurrency optimization (`RwLock` instead of `Mutex`, scoped guard to avoid deadlock)
|
- **Phase 5**: Concurrency optimization (`RwLock` instead of `Mutex`, scoped guard to avoid deadlock)
|
||||||
|
|
||||||
**Frontend Refactoring (React + TypeScript)** - Completed 4-stage refactoring:
|
**Frontend Refactoring (React + TypeScript)** - Completed 4-stage refactoring:
|
||||||
|
|
||||||
- **Stage 1**: Test infrastructure (vitest + MSW + @testing-library/react)
|
- **Stage 1**: Test infrastructure (vitest + MSW + @testing-library/react)
|
||||||
- **Stage 2**: Extracted custom hooks (`useProviderActions`, `useMcpActions`, `useSettings`, `useImportExport`, etc.)
|
- **Stage 2**: Extracted custom hooks (`useProviderActions`, `useMcpActions`, `useSettings`, `useImportExport`, etc.)
|
||||||
- **Stage 3**: Component splitting and business logic extraction
|
- **Stage 3**: Component splitting and business logic extraction
|
||||||
- **Stage 4**: Code cleanup and formatting unification
|
- **Stage 4**: Code cleanup and formatting unification
|
||||||
|
|
||||||
**Testing System**:
|
**Testing System**:
|
||||||
|
|
||||||
- Hooks unit tests 100% coverage
|
- Hooks unit tests 100% coverage
|
||||||
- Integration tests covering key processes (App, SettingsDialog, MCP Panel)
|
- Integration tests covering key processes (App, SettingsDialog, MCP Panel)
|
||||||
- MSW mocking backend API to ensure test independence
|
- MSW mocking backend API to ensure test independence
|
||||||
|
|
||||||
**Code Quality**:
|
**Code Quality**:
|
||||||
|
|
||||||
- Unified parameter format: All Tauri commands migrated to camelCase (Tauri 2 specification)
|
- Unified parameter format: All Tauri commands migrated to camelCase (Tauri 2 specification)
|
||||||
- `AppType` renamed to `AppId`: Semantically clearer
|
- `AppType` renamed to `AppId`: Semantically clearer
|
||||||
- Unified parsing with `FromStr` trait: Centralized `app` parameter parsing
|
- Unified parsing with `FromStr` trait: Centralized `app` parameter parsing
|
||||||
@@ -98,6 +343,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
- Remove unused code: `missing_param` helper function, deprecated `tauri-api.ts`, redundant `KimiModelSelector` component
|
- Remove unused code: `missing_param` helper function, deprecated `tauri-api.ts`, redundant `KimiModelSelector` component
|
||||||
|
|
||||||
**Internal Optimizations**:
|
**Internal Optimizations**:
|
||||||
|
|
||||||
- **Removed Legacy Migration Logic**: v3.6 removed v1 config auto-migration and copy file scanning logic
|
- **Removed Legacy Migration Logic**: v3.6 removed v1 config auto-migration and copy file scanning logic
|
||||||
- ✅ **Impact**: Improved startup performance, cleaner code
|
- ✅ **Impact**: Improved startup performance, cleaner code
|
||||||
- ✅ **Compatibility**: v2 format configs fully compatible, no action required
|
- ✅ **Compatibility**: v2 format configs fully compatible, no action required
|
||||||
@@ -361,6 +607,7 @@ For users upgrading from v2.x (Electron version):
|
|||||||
- Basic provider management
|
- Basic provider management
|
||||||
- Claude Code integration
|
- Claude Code integration
|
||||||
- Configuration file handling
|
- Configuration file handling
|
||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
### ⚠️ Breaking Changes
|
### ⚠️ Breaking Changes
|
||||||
|
|||||||
105
README.md
105
README.md
@@ -1,8 +1,8 @@
|
|||||||
<div align="center">
|
<div align="center">
|
||||||
|
|
||||||
# Claude Code & Codex Provider Switcher
|
# All-in-One Assistant for Claude Code, Codex & Gemini CLI
|
||||||
|
|
||||||
[](https://github.com/farion1231/cc-switch/releases)
|
[](https://github.com/farion1231/cc-switch/releases)
|
||||||
[](https://github.com/trending/typescript)
|
[](https://github.com/trending/typescript)
|
||||||
[](https://github.com/farion1231/cc-switch/releases)
|
[](https://github.com/farion1231/cc-switch/releases)
|
||||||
[](https://tauri.app/)
|
[](https://tauri.app/)
|
||||||
@@ -12,7 +12,9 @@
|
|||||||
|
|
||||||
English | [中文](README_ZH.md) | [Changelog](CHANGELOG.md)
|
English | [中文](README_ZH.md) | [Changelog](CHANGELOG.md)
|
||||||
|
|
||||||
A desktop application for managing and switching between different provider configurations & MCP for Claude Code and Codex.
|
**From Provider Switcher to All-in-One AI CLI Management Platform**
|
||||||
|
|
||||||
|
Unified management for Claude Code, Codex & Gemini CLI provider configurations, MCP servers, Skills extensions, and system prompts.
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -33,6 +35,12 @@ Get 10% OFF the GLM CODING PLAN with [this link](https://z.ai/subscribe?ic=8JVLJ
|
|||||||
<td width="180"><img src="assets/partners/logos/packycode.png" alt="PackyCode" width="150"></td>
|
<td width="180"><img src="assets/partners/logos/packycode.png" alt="PackyCode" width="150"></td>
|
||||||
<td>Thanks to PackyCode for sponsoring this project! PackyCode is a reliable and efficient API relay service provider, offering relay services for Claude Code, Codex, Gemini, and more. PackyCode provides special discounts for our software users: register using <a href="https://www.packyapi.com/register?aff=cc-switch">this link</a> and enter the "cc-switch" promo code during recharge to get 10% off.</td>
|
<td>Thanks to PackyCode for sponsoring this project! PackyCode is a reliable and efficient API relay service provider, offering relay services for Claude Code, Codex, Gemini, and more. PackyCode provides special discounts for our software users: register using <a href="https://www.packyapi.com/register?aff=cc-switch">this link</a> and enter the "cc-switch" promo code during recharge to get 10% off.</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
|
<tr>
|
||||||
|
<td width="180"><img src="assets/partners/logos/sds-en.png" alt="ShanDianShuo" width="150"></td>
|
||||||
|
<td>Thanks to ShanDianShuo for sponsoring this project! ShanDianShuo is a local-first AI voice input: Millisecond latency, data stays on device, 4x faster than typing, AI-powered correction, Privacy-first, completely free. Doubles your coding efficiency with Claude Code! <a href="https://www.shandianshuo.cn">Free download</a> for Mac/Win</td>
|
||||||
|
</tr>
|
||||||
|
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
## Screenshots
|
## Screenshots
|
||||||
@@ -43,12 +51,49 @@ Get 10% OFF the GLM CODING PLAN with [this link](https://z.ai/subscribe?ic=8JVLJ
|
|||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
### Current Version: v3.6.2 | [Full Changelog](CHANGELOG.md)
|
### Current Version: v3.7.0 | [Full Changelog](CHANGELOG.md) | [📋 Release Notes](docs/release-note-v3.7.0-en.md)
|
||||||
|
|
||||||
|
**v3.7.0 Major Update (2025-11-19)**
|
||||||
|
|
||||||
|
**Six Core Features, 18,000+ Lines of New Code**
|
||||||
|
|
||||||
|
- **Gemini CLI Integration**
|
||||||
|
- Third supported AI CLI (Claude Code / Codex / Gemini)
|
||||||
|
- Dual-file configuration support (`.env` + `settings.json`)
|
||||||
|
- Complete MCP server management
|
||||||
|
- Presets: Google Official (OAuth) / PackyCode / Custom
|
||||||
|
|
||||||
|
- **Claude Skills Management System**
|
||||||
|
- Auto-scan skills from GitHub repositories (3 pre-configured curated repos)
|
||||||
|
- One-click install/uninstall to `~/.claude/skills/`
|
||||||
|
- Custom repository support + subdirectory scanning
|
||||||
|
- Complete lifecycle management (discover/install/update)
|
||||||
|
|
||||||
|
- **Prompts Management System**
|
||||||
|
- Multi-preset system prompt management (unlimited presets, quick switching)
|
||||||
|
- Cross-app support (Claude: `CLAUDE.md` / Codex: `AGENTS.md` / Gemini: `GEMINI.md`)
|
||||||
|
- Markdown editor (CodeMirror 6 + real-time preview)
|
||||||
|
- Smart backfill protection, preserves manual modifications
|
||||||
|
|
||||||
|
- **MCP v3.7.0 Unified Architecture**
|
||||||
|
- Single panel manages MCP servers across three applications
|
||||||
|
- New SSE (Server-Sent Events) transport type
|
||||||
|
- Smart JSON parser + Codex TOML format auto-correction
|
||||||
|
- Unified import/export + bidirectional sync
|
||||||
|
|
||||||
|
- **Deep Link Protocol**
|
||||||
|
- `ccswitch://` protocol registration (all platforms)
|
||||||
|
- One-click import provider configs via shared links
|
||||||
|
- Security validation + lifecycle integration
|
||||||
|
|
||||||
|
- **Environment Variable Conflict Detection**
|
||||||
|
- Auto-detect cross-app configuration conflicts (Claude/Codex/Gemini/MCP)
|
||||||
|
- Visual conflict indicators + resolution suggestions
|
||||||
|
- Override warnings + backup before changes
|
||||||
|
|
||||||
**Core Capabilities**
|
**Core Capabilities**
|
||||||
|
|
||||||
- **Provider Management**: One-click switching between Claude Code, Codex, and Gemini API configurations
|
- **Provider Management**: One-click switching between Claude Code, Codex, and Gemini API configurations
|
||||||
- **MCP Integration**: Centralized MCP server management with stdio/http support and real-time sync
|
|
||||||
- **Speed Testing**: Measure API endpoint latency with visual quality indicators
|
- **Speed Testing**: Measure API endpoint latency with visual quality indicators
|
||||||
- **Import/Export**: Backup and restore configs with auto-rotation (keep 10 most recent)
|
- **Import/Export**: Backup and restore configs with auto-rotation (keep 10 most recent)
|
||||||
- **i18n Support**: Complete Chinese/English localization (UI, errors, tray)
|
- **i18n Support**: Complete Chinese/English localization (UI, errors, tray)
|
||||||
@@ -61,7 +106,6 @@ Get 10% OFF the GLM CODING PLAN with [this link](https://z.ai/subscribe?ic=8JVLJ
|
|||||||
- Granular model configuration (4-tier: Haiku/Sonnet/Opus/Custom)
|
- Granular model configuration (4-tier: Haiku/Sonnet/Opus/Custom)
|
||||||
- WSL environment support with auto-sync on directory change
|
- WSL environment support with auto-sync on directory change
|
||||||
- 100% hooks test coverage & complete architecture refactoring
|
- 100% hooks test coverage & complete architecture refactoring
|
||||||
- New presets: DMXAPI, Azure Codex, AnyRouter, AiHubMix, MiniMax
|
|
||||||
|
|
||||||
**System Features**
|
**System Features**
|
||||||
|
|
||||||
@@ -103,6 +147,14 @@ Download `CC-Switch-v{version}-macOS.zip` from the [Releases](../../releases) pa
|
|||||||
|
|
||||||
> **Note**: Since the author doesn't have an Apple Developer account, you may see an "unidentified developer" warning on first launch. Please close it first, then go to "System Settings" → "Privacy & Security" → click "Open Anyway", and you'll be able to open it normally afterwards.
|
> **Note**: Since the author doesn't have an Apple Developer account, you may see an "unidentified developer" warning on first launch. Please close it first, then go to "System Settings" → "Privacy & Security" → click "Open Anyway", and you'll be able to open it normally afterwards.
|
||||||
|
|
||||||
|
### ArchLinux 用户
|
||||||
|
|
||||||
|
**Install via paru (Recommended)**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
paru -S cc-switch-bin
|
||||||
|
```
|
||||||
|
|
||||||
### Linux Users
|
### Linux Users
|
||||||
|
|
||||||
Download the latest `CC-Switch-v{version}-Linux.deb` package or `CC-Switch-v{version}-Linux.AppImage` from the [Releases](../../releases) page.
|
Download the latest `CC-Switch-v{version}-Linux.deb` package or `CC-Switch-v{version}-Linux.AppImage` from the [Releases](../../releases) page.
|
||||||
@@ -121,9 +173,36 @@ Download the latest `CC-Switch-v{version}-Linux.deb` package or `CC-Switch-v{ver
|
|||||||
### MCP Management
|
### MCP Management
|
||||||
|
|
||||||
- **Location**: Click "MCP" button in top-right corner
|
- **Location**: Click "MCP" button in top-right corner
|
||||||
- **Add Server**: Use built-in templates (mcp-fetch, mcp-filesystem) or custom config
|
- **Add Server**:
|
||||||
|
- Use built-in templates (mcp-fetch, mcp-filesystem, etc.)
|
||||||
|
- Support stdio / http / sse transport types
|
||||||
|
- Configure independent MCP servers for different apps
|
||||||
- **Enable/Disable**: Toggle switches to control which servers sync to live config
|
- **Enable/Disable**: Toggle switches to control which servers sync to live config
|
||||||
- **Sync**: Enabled servers auto-sync to `~/.claude.json` (Claude) or `~/.codex/config.toml` (Codex)
|
- **Sync**: Enabled servers auto-sync to each app's live files
|
||||||
|
- **Import/Export**: Import existing MCP servers from Claude/Codex/Gemini config files
|
||||||
|
|
||||||
|
### Skills Management (v3.7.0 New)
|
||||||
|
|
||||||
|
- **Location**: Click "Skills" button in top-right corner
|
||||||
|
- **Discover Skills**:
|
||||||
|
- Auto-scan pre-configured GitHub repositories (Anthropic official, ComposioHQ, community, etc.)
|
||||||
|
- Add custom repositories (supports subdirectory scanning)
|
||||||
|
- **Install Skills**: Click "Install" to one-click install to `~/.claude/skills/`
|
||||||
|
- **Uninstall Skills**: Click "Uninstall" to safely remove and clean up state
|
||||||
|
- **Manage Repositories**: Add/remove custom GitHub repositories
|
||||||
|
|
||||||
|
### Prompts Management (v3.7.0 New)
|
||||||
|
|
||||||
|
- **Location**: Click "Prompts" button in top-right corner
|
||||||
|
- **Create Presets**:
|
||||||
|
- Create unlimited system prompt presets
|
||||||
|
- Use Markdown editor to write prompts (syntax highlighting + real-time preview)
|
||||||
|
- **Switch Presets**: Select preset → Click "Activate" to apply immediately
|
||||||
|
- **Sync Mechanism**:
|
||||||
|
- Claude: `~/.claude/CLAUDE.md`
|
||||||
|
- Codex: `~/.codex/AGENTS.md`
|
||||||
|
- Gemini: `~/.gemini/GEMINI.md`
|
||||||
|
- **Protection Mechanism**: Auto-save current prompt content before switching, preserves manual modifications
|
||||||
|
|
||||||
### Configuration Files
|
### Configuration Files
|
||||||
|
|
||||||
@@ -141,13 +220,15 @@ Download the latest `CC-Switch-v{version}-Linux.deb` package or `CC-Switch-v{ver
|
|||||||
|
|
||||||
**Gemini**
|
**Gemini**
|
||||||
|
|
||||||
- Live config: `~/.gemini/.env` (API key) + `~/.gemini/settings.json` (auth type for quick switching)
|
- Live config: `~/.gemini/.env` (API key) + `~/.gemini/settings.json` (auth mode)
|
||||||
- API key field: `GEMINI_API_KEY` inside `.env`
|
- API key field: `GEMINI_API_KEY` or `GOOGLE_GEMINI_API_KEY` in `.env`
|
||||||
- Tray quick switch: each provider switch rewrites `~/.gemini/.env` so the Gemini CLI picks up the new credentials immediately
|
- Environment variables: Support `GOOGLE_GEMINI_BASE_URL`, `GEMINI_MODEL`, etc.
|
||||||
|
- MCP servers: `~/.gemini/settings.json` → `mcpServers`
|
||||||
|
- Tray quick switch: Each provider switch rewrites `~/.gemini/.env`, no need to restart Gemini CLI
|
||||||
|
|
||||||
**CC Switch Storage**
|
**CC Switch Storage**
|
||||||
|
|
||||||
- Main config (SSOT): `~/.cc-switch/config.json`
|
- Main config (SSOT): `~/.cc-switch/config.json` (includes providers, MCP, Prompts presets, etc.)
|
||||||
- Settings: `~/.cc-switch/settings.json`
|
- Settings: `~/.cc-switch/settings.json`
|
||||||
- Backups: `~/.cc-switch/backups/` (auto-rotate, keep 10)
|
- Backups: `~/.cc-switch/backups/` (auto-rotate, keep 10)
|
||||||
|
|
||||||
|
|||||||
107
README_ZH.md
107
README_ZH.md
@@ -1,8 +1,8 @@
|
|||||||
<div align="center">
|
<div align="center">
|
||||||
|
|
||||||
# Claude Code & Codex 供应商管理器
|
# Claude Code / Codex / Gemini CLI 全方位辅助工具
|
||||||
|
|
||||||
[](https://github.com/farion1231/cc-switch/releases)
|
[](https://github.com/farion1231/cc-switch/releases)
|
||||||
[](https://github.com/trending/typescript)
|
[](https://github.com/trending/typescript)
|
||||||
[](https://github.com/farion1231/cc-switch/releases)
|
[](https://github.com/farion1231/cc-switch/releases)
|
||||||
[](https://tauri.app/)
|
[](https://tauri.app/)
|
||||||
@@ -10,9 +10,11 @@
|
|||||||
|
|
||||||
<a href="https://trendshift.io/repositories/15372" target="_blank"><img src="https://trendshift.io/api/badge/repositories/15372" alt="farion1231%2Fcc-switch | Trendshift" style="width: 250px; height: 55px;" width="250" height="55"/></a>
|
<a href="https://trendshift.io/repositories/15372" target="_blank"><img src="https://trendshift.io/api/badge/repositories/15372" alt="farion1231%2Fcc-switch | Trendshift" style="width: 250px; height: 55px;" width="250" height="55"/></a>
|
||||||
|
|
||||||
[English](README.md) | 中文 | [更新日志](CHANGELOG.md)
|
[English](README.md) | 中文 | [更新日志](CHANGELOG.md) | [📋 v3.7.0 发布说明](docs/release-note-v3.7.0-zh.md)
|
||||||
|
|
||||||
一个用于管理和切换 Claude Code 与 Codex 不同供应商配置、MCP的桌面应用。
|
**从供应商切换器到 AI CLI 一体化管理平台**
|
||||||
|
|
||||||
|
统一管理 Claude Code、Codex 与 Gemini CLI 的供应商配置、MCP 服务器、Skills 扩展和系统提示词。
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -33,6 +35,12 @@ CC Switch 已经预设了智谱GLM,只需要填写 key 即可一键导入编
|
|||||||
<td width="180"><img src="assets/partners/logos/packycode.png" alt="PackyCode" width="150"></td>
|
<td width="180"><img src="assets/partners/logos/packycode.png" alt="PackyCode" width="150"></td>
|
||||||
<td>感谢 PackyCode 赞助了本项目!PackyCode 是一家稳定、高效的API中转服务商,提供 Claude Code、Codex、Gemini 等多种中转服务。PackyCode 为本软件的用户提供了特别优惠,使用<a href="https://www.packyapi.com/register?aff=cc-switch">此链接</a>注册并在充值时填写"cc-switch"优惠码,可以享受9折优惠。</td>
|
<td>感谢 PackyCode 赞助了本项目!PackyCode 是一家稳定、高效的API中转服务商,提供 Claude Code、Codex、Gemini 等多种中转服务。PackyCode 为本软件的用户提供了特别优惠,使用<a href="https://www.packyapi.com/register?aff=cc-switch">此链接</a>注册并在充值时填写"cc-switch"优惠码,可以享受9折优惠。</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
|
<tr>
|
||||||
|
<td width="180"><img src="assets/partners/logos/sds-zh.png" alt="ShanDianShuo" width="150"></td>
|
||||||
|
<td>感谢闪电说赞助了本项目!闪电说是本地优先的 AI 语音输入法:毫秒级响应,数据不离设备;打字速度提升 4 倍,AI 智能纠错;绝对隐私安全,完全免费,配合 Claude Code 写代码效率翻倍!支持 Mac/Win 双平台,<a href="https://www.shandianshuo.cn">免费下载</a></td>
|
||||||
|
</tr>
|
||||||
|
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
## 界面预览
|
## 界面预览
|
||||||
@@ -43,12 +51,49 @@ CC Switch 已经预设了智谱GLM,只需要填写 key 即可一键导入编
|
|||||||
|
|
||||||
## 功能特性
|
## 功能特性
|
||||||
|
|
||||||
### 当前版本:v3.6.2 | [完整更新日志](CHANGELOG.md)
|
### 当前版本:v3.7.0 | [完整更新日志](CHANGELOG.md)
|
||||||
|
|
||||||
|
**v3.7.0 重大更新(2025-11-19)**
|
||||||
|
|
||||||
|
**六大核心功能,18,000+ 行新增代码**
|
||||||
|
|
||||||
|
- **Gemini CLI 集成**
|
||||||
|
- 第三个支持的 AI CLI(Claude Code / Codex / Gemini)
|
||||||
|
- 双文件配置支持(`.env` + `settings.json`)
|
||||||
|
- 完整 MCP 服务器管理
|
||||||
|
- 预设:Google Official (OAuth) / PackyCode / 自定义
|
||||||
|
|
||||||
|
- **Claude Skills 管理系统**
|
||||||
|
- 从 GitHub 仓库自动扫描技能(预配置 3 个精选仓库)
|
||||||
|
- 一键安装/卸载到 `~/.claude/skills/`
|
||||||
|
- 自定义仓库支持 + 子目录扫描
|
||||||
|
- 完整生命周期管理(发现/安装/更新)
|
||||||
|
|
||||||
|
- **Prompts 管理系统**
|
||||||
|
- 多预设系统提示词管理(无限数量,快速切换)
|
||||||
|
- 跨应用支持(Claude: `CLAUDE.md` / Codex: `AGENTS.md` / Gemini: `GEMINI.md`)
|
||||||
|
- Markdown 编辑器(CodeMirror 6 + 实时预览)
|
||||||
|
- 智能回填保护,保留手动修改
|
||||||
|
|
||||||
|
- **MCP v3.7.0 统一架构**
|
||||||
|
- 单一面板管理三个应用的 MCP 服务器
|
||||||
|
- 新增 SSE (Server-Sent Events) 传输类型
|
||||||
|
- 智能 JSON 解析器 + Codex TOML 格式自动修正
|
||||||
|
- 统一导入/导出 + 双向同步
|
||||||
|
|
||||||
|
- **深度链接协议**
|
||||||
|
- `ccswitch://` 协议注册(全平台)
|
||||||
|
- 通过共享链接一键导入供应商配置
|
||||||
|
- 安全验证 + 生命周期集成
|
||||||
|
|
||||||
|
- **环境变量冲突检测**
|
||||||
|
- 自动检测跨应用配置冲突(Claude/Codex/Gemini/MCP)
|
||||||
|
- 可视化冲突指示器 + 解决建议
|
||||||
|
- 覆盖警告 + 更改前备份
|
||||||
|
|
||||||
**核心功能**
|
**核心功能**
|
||||||
|
|
||||||
- **供应商管理**:一键切换 Claude Code、Codex 与 Gemini 的 API 配置
|
- **供应商管理**:一键切换 Claude Code、Codex 与 Gemini 的 API 配置
|
||||||
- **MCP 集成**:集中管理 MCP 服务器,支持 stdio/http 类型和实时同步
|
|
||||||
- **速度测试**:测量 API 端点延迟,可视化连接质量指示器
|
- **速度测试**:测量 API 端点延迟,可视化连接质量指示器
|
||||||
- **导入导出**:备份和恢复配置,自动轮换(保留最近 10 个)
|
- **导入导出**:备份和恢复配置,自动轮换(保留最近 10 个)
|
||||||
- **国际化支持**:完整的中英文本地化(UI、错误、托盘)
|
- **国际化支持**:完整的中英文本地化(UI、错误、托盘)
|
||||||
@@ -61,7 +106,6 @@ CC Switch 已经预设了智谱GLM,只需要填写 key 即可一键导入编
|
|||||||
- 细粒度模型配置(四层:Haiku/Sonnet/Opus/自定义)
|
- 细粒度模型配置(四层:Haiku/Sonnet/Opus/自定义)
|
||||||
- WSL 环境支持,配置目录切换自动同步
|
- WSL 环境支持,配置目录切换自动同步
|
||||||
- 100% hooks 测试覆盖 & 完整架构重构
|
- 100% hooks 测试覆盖 & 完整架构重构
|
||||||
- 新增预设:DMXAPI、Azure Codex、AnyRouter、AiHubMix、MiniMax
|
|
||||||
|
|
||||||
**系统功能**
|
**系统功能**
|
||||||
|
|
||||||
@@ -103,6 +147,14 @@ brew upgrade --cask cc-switch
|
|||||||
|
|
||||||
> **注意**:由于作者没有苹果开发者账号,首次打开可能出现"未知开发者"警告,请先关闭,然后前往"系统设置" → "隐私与安全性" → 点击"仍要打开",之后便可以正常打开
|
> **注意**:由于作者没有苹果开发者账号,首次打开可能出现"未知开发者"警告,请先关闭,然后前往"系统设置" → "隐私与安全性" → 点击"仍要打开",之后便可以正常打开
|
||||||
|
|
||||||
|
### ArchLinux 用户
|
||||||
|
|
||||||
|
**通过 paru 安装(推荐)**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
paru -S cc-switch-bin
|
||||||
|
```
|
||||||
|
|
||||||
### Linux 用户
|
### Linux 用户
|
||||||
|
|
||||||
从 [Releases](../../releases) 页面下载最新版本的 `CC-Switch-v{版本号}-Linux.deb` 包或者 `CC-Switch-v{版本号}-Linux.AppImage` 安装包。
|
从 [Releases](../../releases) 页面下载最新版本的 `CC-Switch-v{版本号}-Linux.deb` 包或者 `CC-Switch-v{版本号}-Linux.AppImage` 安装包。
|
||||||
@@ -121,9 +173,36 @@ brew upgrade --cask cc-switch
|
|||||||
### MCP 管理
|
### MCP 管理
|
||||||
|
|
||||||
- **位置**:点击右上角"MCP"按钮
|
- **位置**:点击右上角"MCP"按钮
|
||||||
- **添加服务器**:使用内置模板(mcp-fetch、mcp-filesystem)或自定义配置
|
- **添加服务器**:
|
||||||
|
- 使用内置模板(mcp-fetch、mcp-filesystem 等)
|
||||||
|
- 支持 stdio / http / sse 三种传输类型
|
||||||
|
- 为不同应用配置独立的 MCP 服务器
|
||||||
- **启用/禁用**:切换开关以控制哪些服务器同步到 live 配置
|
- **启用/禁用**:切换开关以控制哪些服务器同步到 live 配置
|
||||||
- **同步**:启用的服务器自动同步到 `~/.claude.json`(Claude)或 `~/.codex/config.toml`(Codex)
|
- **同步**:启用的服务器自动同步到各应用的 live 文件
|
||||||
|
- **导入/导出**:支持从 Claude/Codex/Gemini 配置文件导入现有 MCP 服务器
|
||||||
|
|
||||||
|
### Skills 管理(v3.7.0 新增)
|
||||||
|
|
||||||
|
- **位置**:点击右上角"Skills"按钮
|
||||||
|
- **发现技能**:
|
||||||
|
- 自动扫描预配置的 GitHub 仓库(Anthropic 官方、ComposioHQ、社区等)
|
||||||
|
- 添加自定义仓库(支持子目录扫描)
|
||||||
|
- **安装技能**:点击"安装"一键安装到 `~/.claude/skills/`
|
||||||
|
- **卸载技能**:点击"卸载"安全移除并清理状态
|
||||||
|
- **管理仓库**:添加/删除自定义 GitHub 仓库
|
||||||
|
|
||||||
|
### Prompts 管理(v3.7.0 新增)
|
||||||
|
|
||||||
|
- **位置**:点击右上角"Prompts"按钮
|
||||||
|
- **创建预设**:
|
||||||
|
- 创建无限数量的系统提示词预设
|
||||||
|
- 使用 Markdown 编辑器编写提示词(语法高亮 + 实时预览)
|
||||||
|
- **切换预设**:选择预设 → 点击"激活"立即应用
|
||||||
|
- **同步机制**:
|
||||||
|
- Claude: `~/.claude/CLAUDE.md`
|
||||||
|
- Codex: `~/.codex/AGENTS.md`
|
||||||
|
- Gemini: `~/.gemini/GEMINI.md`
|
||||||
|
- **保护机制**:切换前自动保存当前提示词内容,保留手动修改
|
||||||
|
|
||||||
### 配置文件
|
### 配置文件
|
||||||
|
|
||||||
@@ -141,13 +220,15 @@ brew upgrade --cask cc-switch
|
|||||||
|
|
||||||
**Gemini**
|
**Gemini**
|
||||||
|
|
||||||
- Live 配置:`~/.gemini/.env`(API Key)+ `~/.gemini/settings.json`(保存认证模式,支持托盘快速切换)
|
- Live 配置:`~/.gemini/.env`(API Key)+ `~/.gemini/settings.json`(保存认证模式)
|
||||||
- API key 字段:`.env` 文件中的 `GEMINI_API_KEY`
|
- API key 字段:`.env` 文件中的 `GEMINI_API_KEY` 或 `GOOGLE_GEMINI_API_KEY`
|
||||||
- 托盘快速切换:每次切换供应商都会重写 `~/.gemini/.env`,Gemini CLI 无需额外操作即可使用新配置
|
- 环境变量:支持 `GOOGLE_GEMINI_BASE_URL`、`GEMINI_MODEL` 等自定义变量
|
||||||
|
- MCP 服务器:`~/.gemini/settings.json` → `mcpServers`
|
||||||
|
- 托盘快速切换:每次切换供应商都会重写 `~/.gemini/.env`,无需重启 Gemini CLI 即可生效
|
||||||
|
|
||||||
**CC Switch 存储**
|
**CC Switch 存储**
|
||||||
|
|
||||||
- 主配置(SSOT):`~/.cc-switch/config.json`
|
- 主配置(SSOT):`~/.cc-switch/config.json`(包含供应商、MCP、Prompts 预设等)
|
||||||
- 设置:`~/.cc-switch/settings.json`
|
- 设置:`~/.cc-switch/settings.json`
|
||||||
- 备份:`~/.cc-switch/backups/`(自动轮换,保留 10 个)
|
- 备份:`~/.cc-switch/backups/`(自动轮换,保留 10 个)
|
||||||
|
|
||||||
|
|||||||
BIN
assets/partners/logos/sds-en.png
Normal file
BIN
assets/partners/logos/sds-en.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 179 KiB |
BIN
assets/partners/logos/sds-zh.png
Normal file
BIN
assets/partners/logos/sds-zh.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 6.5 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 204 KiB After Width: | Height: | Size: 227 KiB |
Binary file not shown.
|
Before Width: | Height: | Size: 205 KiB After Width: | Height: | Size: 227 KiB |
1279
deplink.html
1279
deplink.html
File diff suppressed because it is too large
Load Diff
439
docs/release-note-v3.7.0-en.md
Normal file
439
docs/release-note-v3.7.0-en.md
Normal file
@@ -0,0 +1,439 @@
|
|||||||
|
# CC Switch v3.7.0
|
||||||
|
|
||||||
|
> From Provider Switcher to All-in-One AI CLI Management Platform
|
||||||
|
|
||||||
|
**[中文更新说明 Chinese Documentation →](release-note-v3.7.0-zh.md)**
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
CC Switch v3.7.0 introduces six major features with over 18,000 lines of new code.
|
||||||
|
|
||||||
|
**Release Date**: 2025-11-19
|
||||||
|
**Commits**: 85 from v3.6.0
|
||||||
|
**Code Changes**: 152 files, +18,104 / -3,732 lines
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## New Features
|
||||||
|
|
||||||
|
### Gemini CLI Integration
|
||||||
|
|
||||||
|
Complete support for Google Gemini CLI, becoming the third supported application (Claude Code, Codex, Gemini).
|
||||||
|
|
||||||
|
**Core Capabilities**:
|
||||||
|
|
||||||
|
- **Dual-file configuration** - Support for both `.env` and `settings.json` formats
|
||||||
|
- **Auto-detection** - Automatically detect `GOOGLE_GEMINI_BASE_URL`, `GEMINI_MODEL`, etc.
|
||||||
|
- **Full MCP support** - Complete MCP server management for Gemini
|
||||||
|
- **Deep link integration** - Import via `ccswitch://` protocol
|
||||||
|
- **System tray** - Quick-switch from tray menu
|
||||||
|
|
||||||
|
**Provider Presets**:
|
||||||
|
|
||||||
|
- **Google Official** - OAuth authentication support
|
||||||
|
- **PackyCode** - Partner integration
|
||||||
|
- **Custom** - Full customization support
|
||||||
|
|
||||||
|
**Technical Implementation**:
|
||||||
|
|
||||||
|
- New backend modules: `gemini_config.rs` (20KB), `gemini_mcp.rs`
|
||||||
|
- Form synchronization with environment editor
|
||||||
|
- Dual-file atomic writes
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### MCP v3.7.0 Unified Architecture
|
||||||
|
|
||||||
|
Complete refactoring of MCP management system for cross-application unification.
|
||||||
|
|
||||||
|
**Architecture Improvements**:
|
||||||
|
|
||||||
|
- **Unified panel** - Single interface for Claude/Codex/Gemini MCP servers
|
||||||
|
- **SSE transport** - New Server-Sent Events support
|
||||||
|
- **Smart parser** - Fault-tolerant JSON parsing
|
||||||
|
- **Format correction** - Auto-fix Codex `[mcp_servers]` format
|
||||||
|
- **Extended fields** - Preserve custom TOML fields
|
||||||
|
|
||||||
|
**User Experience**:
|
||||||
|
|
||||||
|
- Default app selection in forms
|
||||||
|
- JSON formatter for validation
|
||||||
|
- Improved visual hierarchy
|
||||||
|
- Better error messages
|
||||||
|
|
||||||
|
**Import/Export**:
|
||||||
|
|
||||||
|
- Unified import from all three apps
|
||||||
|
- Bidirectional synchronization
|
||||||
|
- State preservation
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Claude Skills Management System
|
||||||
|
|
||||||
|
**Approximately 2,000 lines of code** - A complete skill ecosystem platform.
|
||||||
|
|
||||||
|
**GitHub Integration**:
|
||||||
|
|
||||||
|
- Auto-scan skills from GitHub repositories
|
||||||
|
- Pre-configured repos:
|
||||||
|
- `ComposioHQ/awesome-claude-skills` - Curated collection
|
||||||
|
- `anthropics/skills` - Official Anthropic skills
|
||||||
|
- `cexll/myclaude` - Community contributions
|
||||||
|
- Add custom repositories
|
||||||
|
- Subdirectory scanning support (`skillsPath`)
|
||||||
|
|
||||||
|
**Lifecycle Management**:
|
||||||
|
|
||||||
|
- **Discover** - Auto-detect `SKILL.md` files
|
||||||
|
- **Install** - One-click to `~/.claude/skills/`
|
||||||
|
- **Uninstall** - Safe removal with tracking
|
||||||
|
- **Update** - Check for updates (infrastructure ready)
|
||||||
|
|
||||||
|
**Technical Architecture**:
|
||||||
|
|
||||||
|
- **Backend**: `SkillService` (526 lines) with GitHub API integration
|
||||||
|
- **Frontend**: SkillsPage, SkillCard, RepoManager
|
||||||
|
- **UI Components**: Badge, Card, Table (shadcn/ui)
|
||||||
|
- **State**: Persistent storage in `skills.json`
|
||||||
|
- **i18n**: 47+ translation keys
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Prompts Management System
|
||||||
|
|
||||||
|
**Approximately 1,300 lines of code** - Complete system prompt management.
|
||||||
|
|
||||||
|
**Multi-Preset Management**:
|
||||||
|
|
||||||
|
- Create unlimited prompt presets
|
||||||
|
- Quick switch between presets
|
||||||
|
- One active prompt at a time
|
||||||
|
- Delete protection for active prompts
|
||||||
|
|
||||||
|
**Cross-App Support**:
|
||||||
|
|
||||||
|
- **Claude**: `~/.claude/CLAUDE.md`
|
||||||
|
- **Codex**: `~/.codex/AGENTS.md`
|
||||||
|
- **Gemini**: `~/.gemini/GEMINI.md`
|
||||||
|
|
||||||
|
**Markdown Editor**:
|
||||||
|
|
||||||
|
- Full-featured CodeMirror 6 integration
|
||||||
|
- Syntax highlighting
|
||||||
|
- Dark theme (One Dark)
|
||||||
|
- Real-time preview
|
||||||
|
|
||||||
|
**Smart Synchronization**:
|
||||||
|
|
||||||
|
- **Auto-write** - Immediately write to live files
|
||||||
|
- **Backfill protection** - Save current content before switching
|
||||||
|
- **Auto-import** - Import from live files on first launch
|
||||||
|
- **Modification protection** - Preserve manual modifications
|
||||||
|
|
||||||
|
**Technical Implementation**:
|
||||||
|
|
||||||
|
- **Backend**: `PromptService` (213 lines)
|
||||||
|
- **Frontend**: PromptPanel (177), PromptFormModal (160), MarkdownEditor (159)
|
||||||
|
- **Hooks**: usePromptActions (152 lines)
|
||||||
|
- **i18n**: 41+ translation keys
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Deep Link Protocol (ccswitch://)
|
||||||
|
|
||||||
|
One-click provider configuration import via URL scheme.
|
||||||
|
|
||||||
|
**Features**:
|
||||||
|
|
||||||
|
- Protocol registration on all platforms
|
||||||
|
- Import from shared links
|
||||||
|
- Lifecycle integration
|
||||||
|
- Security validation
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Environment Variable Conflict Detection
|
||||||
|
|
||||||
|
Intelligent detection and management of configuration conflicts.
|
||||||
|
|
||||||
|
**Detection Scope**:
|
||||||
|
|
||||||
|
- **Claude & Codex** - Cross-app conflicts
|
||||||
|
- **Gemini** - Auto-discovery
|
||||||
|
- **MCP** - Server configuration conflicts
|
||||||
|
|
||||||
|
**Management Features**:
|
||||||
|
|
||||||
|
- Visual conflict indicators
|
||||||
|
- Resolution suggestions
|
||||||
|
- Override warnings
|
||||||
|
- Backup before changes
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Improvements
|
||||||
|
|
||||||
|
### Provider Management
|
||||||
|
|
||||||
|
**New Presets**:
|
||||||
|
|
||||||
|
- **DouBaoSeed** - ByteDance's DouBao
|
||||||
|
- **Kimi For Coding** - Moonshot AI
|
||||||
|
- **BaiLing** - BaiLing AI
|
||||||
|
- **Removed AnyRouter** - To avoid confusion
|
||||||
|
|
||||||
|
**Enhancements**:
|
||||||
|
|
||||||
|
- Model name configuration for Codex and Gemini
|
||||||
|
- Provider notes field for organization
|
||||||
|
- Enhanced preset metadata
|
||||||
|
|
||||||
|
### Configuration Management
|
||||||
|
|
||||||
|
- **Common config migration** - From localStorage to `config.json`
|
||||||
|
- **Unified persistence** - Shared across all apps
|
||||||
|
- **Auto-import** - First launch configuration import
|
||||||
|
- **Backfill priority** - Correct handling of live files
|
||||||
|
|
||||||
|
### UI/UX Improvements
|
||||||
|
|
||||||
|
**Design System**:
|
||||||
|
|
||||||
|
- **macOS native** - System-aligned color scheme
|
||||||
|
- **Window centering** - Default centered position
|
||||||
|
- **Visual polish** - Improved spacing and hierarchy
|
||||||
|
|
||||||
|
**Interactions**:
|
||||||
|
|
||||||
|
- **Password input** - Fixed Edge/IE reveal buttons
|
||||||
|
- **URL overflow** - Fixed card overflow
|
||||||
|
- **Error copying** - Copy-to-clipboard errors
|
||||||
|
- **Tray sync** - Real-time drag-and-drop sync
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Bug Fixes
|
||||||
|
|
||||||
|
### Critical Fixes
|
||||||
|
|
||||||
|
- **Usage script validation** - Boundary checks
|
||||||
|
- **Gemini validation** - Relaxed constraints
|
||||||
|
- **TOML parsing** - CJK quote handling
|
||||||
|
- **MCP fields** - Custom field preservation
|
||||||
|
- **White screen** - FormLabel crash fix
|
||||||
|
|
||||||
|
### Stability
|
||||||
|
|
||||||
|
- **Tray safety** - Pattern matching instead of unwrap
|
||||||
|
- **Error isolation** - Tray failures don't block operations
|
||||||
|
- **Import classification** - Correct category assignment
|
||||||
|
|
||||||
|
### UI Fixes
|
||||||
|
|
||||||
|
- **Model placeholders** - Removed misleading hints
|
||||||
|
- **Base URL** - Auto-fill for third-party providers
|
||||||
|
- **Drag sort** - Tray menu synchronization
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Technical Improvements
|
||||||
|
|
||||||
|
### Architecture
|
||||||
|
|
||||||
|
**MCP v3.7.0**:
|
||||||
|
|
||||||
|
- Removed legacy code (~1,000 lines)
|
||||||
|
- Unified initialization structure
|
||||||
|
- Backward compatibility maintained
|
||||||
|
- Comprehensive code formatting
|
||||||
|
|
||||||
|
**Platform Compatibility**:
|
||||||
|
|
||||||
|
- Windows winreg API fix (v0.52)
|
||||||
|
- Safe pattern matching (no `unwrap()`)
|
||||||
|
- Cross-platform tray handling
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
**Synchronization**:
|
||||||
|
|
||||||
|
- MCP sync across all apps
|
||||||
|
- Gemini form-editor sync
|
||||||
|
- Dual-file reading (.env + settings.json)
|
||||||
|
|
||||||
|
**Validation**:
|
||||||
|
|
||||||
|
- Input boundary checks
|
||||||
|
- TOML quote normalization (CJK)
|
||||||
|
- Custom field preservation
|
||||||
|
- Enhanced error messages
|
||||||
|
|
||||||
|
### Code Quality
|
||||||
|
|
||||||
|
**Type Safety**:
|
||||||
|
|
||||||
|
- Complete TypeScript coverage
|
||||||
|
- Rust type refinements
|
||||||
|
- API contract validation
|
||||||
|
|
||||||
|
**Testing**:
|
||||||
|
|
||||||
|
- Simplified assertions
|
||||||
|
- Better test coverage
|
||||||
|
- Integration test updates
|
||||||
|
|
||||||
|
**Dependencies**:
|
||||||
|
|
||||||
|
- Tauri 2.8.x
|
||||||
|
- Rust: `anyhow`, `zip`, `serde_yaml`, `tempfile`
|
||||||
|
- Frontend: CodeMirror 6 packages
|
||||||
|
- winreg 0.52 (Windows)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Technical Statistics
|
||||||
|
|
||||||
|
```
|
||||||
|
Total Changes:
|
||||||
|
- Commits: 85
|
||||||
|
- Files: 152 changed
|
||||||
|
- Additions: +18,104 lines
|
||||||
|
- Deletions: -3,732 lines
|
||||||
|
|
||||||
|
New Modules:
|
||||||
|
- Skills Management: 2,034 lines (21 files)
|
||||||
|
- Prompts Management: 1,302 lines (20 files)
|
||||||
|
- Gemini Integration: ~1,000 lines
|
||||||
|
- MCP Refactor: ~3,000 lines refactored
|
||||||
|
|
||||||
|
Code Distribution:
|
||||||
|
- Backend (Rust): ~4,500 lines new
|
||||||
|
- Frontend (React): ~3,000 lines new
|
||||||
|
- Configuration: ~1,500 lines refactored
|
||||||
|
- Tests: ~500 lines
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Strategic Positioning
|
||||||
|
|
||||||
|
### From Tool to Platform
|
||||||
|
|
||||||
|
v3.7.0 represents a shift in CC Switch's positioning:
|
||||||
|
|
||||||
|
| Aspect | v3.6 | v3.7.0 |
|
||||||
|
| ----------------- | ------------------------ | ---------------------------- |
|
||||||
|
| **Identity** | Provider Switcher | AI CLI Management Platform |
|
||||||
|
| **Scope** | Configuration Management | Ecosystem Management |
|
||||||
|
| **Applications** | Claude + Codex | Claude + Codex + Gemini |
|
||||||
|
| **Capabilities** | Switch configs | Extend capabilities (Skills) |
|
||||||
|
| **Customization** | Manual editing | Visual management (Prompts) |
|
||||||
|
| **Integration** | Isolated apps | Unified management (MCP) |
|
||||||
|
|
||||||
|
### Six Pillars of AI CLI Management
|
||||||
|
|
||||||
|
1. **Configuration Management** - Provider switching and management
|
||||||
|
2. **Capability Extension** - Skills installation and lifecycle
|
||||||
|
3. **Behavior Customization** - System prompt presets
|
||||||
|
4. **Ecosystem Integration** - Deep links and sharing
|
||||||
|
5. **Multi-AI Support** - Claude/Codex/Gemini
|
||||||
|
6. **Intelligent Detection** - Conflict prevention
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Download & Installation
|
||||||
|
|
||||||
|
### System Requirements
|
||||||
|
|
||||||
|
- **Windows**: Windows 10+
|
||||||
|
- **macOS**: macOS 10.15 (Catalina)+
|
||||||
|
- **Linux**: Ubuntu 22.04+ / Debian 11+ / Fedora 34+
|
||||||
|
|
||||||
|
### Download Links
|
||||||
|
|
||||||
|
Visit [Releases](https://github.com/farion1231/cc-switch/releases/latest) to download:
|
||||||
|
|
||||||
|
- **Windows**: `CC-Switch-v3.7.0-Windows.msi` or `-Portable.zip`
|
||||||
|
- **macOS**: `CC-Switch-v3.7.0-macOS.tar.gz` or `.zip`
|
||||||
|
- **Linux**: `CC-Switch-v3.7.0-Linux.AppImage` or `.deb`
|
||||||
|
|
||||||
|
### Homebrew (macOS)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
brew tap farion1231/ccswitch
|
||||||
|
brew install --cask cc-switch
|
||||||
|
```
|
||||||
|
|
||||||
|
Update:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
brew upgrade --cask cc-switch
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Migration Notes
|
||||||
|
|
||||||
|
### From v3.6.x
|
||||||
|
|
||||||
|
**Automatic migration** - No action required, configs are fully compatible
|
||||||
|
|
||||||
|
### From v3.1.x or Earlier
|
||||||
|
|
||||||
|
**Two-step migration required**:
|
||||||
|
|
||||||
|
1. First upgrade to v3.2.x (performs one-time migration)
|
||||||
|
2. Then upgrade to v3.7.0
|
||||||
|
|
||||||
|
### New Features
|
||||||
|
|
||||||
|
- **Skills**: No migration needed, start fresh
|
||||||
|
- **Prompts**: Auto-import from live files on first launch
|
||||||
|
- **Gemini**: Install Gemini CLI separately if needed
|
||||||
|
- **MCP v3.7.0**: Backward compatible with previous configs
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Acknowledgments
|
||||||
|
|
||||||
|
### Contributors
|
||||||
|
|
||||||
|
Thanks to all contributors who made this release possible:
|
||||||
|
|
||||||
|
- [@YoVinchen](https://github.com/YoVinchen) - Skills & Prompts & Gemini integration implementation
|
||||||
|
- [@farion1231](https://github.com/farion1231) - From developer to issue responder
|
||||||
|
- Community members for testing and feedback
|
||||||
|
|
||||||
|
### Sponsors
|
||||||
|
|
||||||
|
**Z.ai** - GLM CODING PLAN sponsor
|
||||||
|
[Get 10% OFF with this link](https://z.ai/subscribe?ic=8JVLJQFSKB)
|
||||||
|
|
||||||
|
**PackyCode** - API relay service partner
|
||||||
|
[Register with "cc-switch" code for 10% discount](https://www.packyapi.com/register?aff=cc-switch)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Feedback & Support
|
||||||
|
|
||||||
|
- **Issues**: [GitHub Issues](https://github.com/farion1231/cc-switch/issues)
|
||||||
|
- **Discussions**: [GitHub Discussions](https://github.com/farion1231/cc-switch/discussions)
|
||||||
|
- **Documentation**: [README](../README.md)
|
||||||
|
- **Changelog**: [CHANGELOG.md](../CHANGELOG.md)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## What's Next
|
||||||
|
|
||||||
|
**v3.8.0 Preview** (Tentative):
|
||||||
|
|
||||||
|
- Local proxy functionality
|
||||||
|
|
||||||
|
Stay tuned for more updates!
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Happy Coding!**
|
||||||
435
docs/release-note-v3.7.0-zh.md
Normal file
435
docs/release-note-v3.7.0-zh.md
Normal file
@@ -0,0 +1,435 @@
|
|||||||
|
# CC Switch v3.7.0
|
||||||
|
|
||||||
|
> 从供应商切换器到 AI CLI 一体化管理平台
|
||||||
|
|
||||||
|
**[English Version →](release-note-v3.7.0-en.md)**
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 概览
|
||||||
|
|
||||||
|
CC Switch v3.7.0 新增六大核心功能,新增超过 18,000 行代码。
|
||||||
|
|
||||||
|
**发布日期**:2025-11-19
|
||||||
|
**提交数量**:从 v3.6.0 开始 85 个提交
|
||||||
|
**代码变更**:152 个文件,+18,104 / -3,732 行
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 新增功能
|
||||||
|
|
||||||
|
### Gemini CLI 集成
|
||||||
|
|
||||||
|
完整支持 Google Gemini CLI,成为第三个支持的应用(Claude Code、Codex、Gemini)。
|
||||||
|
|
||||||
|
**核心能力**:
|
||||||
|
|
||||||
|
- **双文件配置** - 同时支持 `.env` 和 `settings.json` 格式
|
||||||
|
- **自动检测** - 自动检测 `GOOGLE_GEMINI_BASE_URL`、`GEMINI_MODEL` 等环境变量
|
||||||
|
- **完整 MCP 支持** - 为 Gemini 提供完整的 MCP 服务器管理
|
||||||
|
- **深度链接集成** - 通过 `ccswitch://` 协议导入配置
|
||||||
|
- **系统托盘** - 从托盘菜单快速切换
|
||||||
|
|
||||||
|
**供应商预设**:
|
||||||
|
|
||||||
|
- **Google Official** - 支持 OAuth 认证
|
||||||
|
- **PackyCode** - 合作伙伴集成
|
||||||
|
- **自定义** - 完全自定义支持
|
||||||
|
|
||||||
|
**技术实现**:
|
||||||
|
|
||||||
|
- 新增后端模块:`gemini_config.rs`(20KB)、`gemini_mcp.rs`
|
||||||
|
- 表单与环境编辑器同步
|
||||||
|
- 双文件原子写入
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### MCP v3.7.0 统一架构
|
||||||
|
|
||||||
|
MCP 管理系统完整重构,实现跨应用统一管理。
|
||||||
|
|
||||||
|
**架构改进**:
|
||||||
|
|
||||||
|
- **统一管理面板** - 单一界面管理 Claude/Codex/Gemini MCP 服务器
|
||||||
|
- **SSE 传输类型** - 新增 Server-Sent Events 支持
|
||||||
|
- **智能解析器** - 容错性 JSON 解析
|
||||||
|
- **格式修正** - 自动修复 Codex `[mcp_servers]` 格式
|
||||||
|
- **扩展字段** - 保留自定义 TOML 字段
|
||||||
|
|
||||||
|
**用户体验**:
|
||||||
|
|
||||||
|
- 表单中的默认应用选择
|
||||||
|
- JSON 格式化器用于验证
|
||||||
|
- 改进的视觉层次
|
||||||
|
- 更好的错误消息
|
||||||
|
|
||||||
|
**导入/导出**:
|
||||||
|
|
||||||
|
- 统一从三个应用导入
|
||||||
|
- 双向同步
|
||||||
|
- 状态保持
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Claude Skills 管理系统
|
||||||
|
|
||||||
|
**约 2,000 行代码** - 完整的技能生态平台。
|
||||||
|
|
||||||
|
**GitHub 集成**:
|
||||||
|
|
||||||
|
- 从 GitHub 仓库自动扫描技能
|
||||||
|
- 预配置仓库:
|
||||||
|
- `ComposioHQ/awesome-claude-skills` - 精选集合
|
||||||
|
- `anthropics/skills` - Anthropic 官方技能
|
||||||
|
- `cexll/myclaude` - 社区贡献
|
||||||
|
- 添加自定义仓库
|
||||||
|
- 子目录扫描支持(`skillsPath`)
|
||||||
|
|
||||||
|
**生命周期管理**:
|
||||||
|
|
||||||
|
- **发现** - 自动检测 `SKILL.md` 文件
|
||||||
|
- **安装** - 一键安装到 `~/.claude/skills/`
|
||||||
|
- **卸载** - 安全移除并跟踪状态
|
||||||
|
- **更新** - 检查更新(基础设施已就绪)
|
||||||
|
|
||||||
|
**技术架构**:
|
||||||
|
|
||||||
|
- **后端**:`SkillService`(526 行)集成 GitHub API
|
||||||
|
- **前端**:SkillsPage、SkillCard、RepoManager
|
||||||
|
- **UI 组件**:Badge、Card、Table(shadcn/ui)
|
||||||
|
- **状态**:持久化存储在 `skills.json`
|
||||||
|
- **国际化**:47+ 个翻译键
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Prompts 管理系统
|
||||||
|
|
||||||
|
**约 1,300 行代码** - 完整的系统提示词管理。
|
||||||
|
|
||||||
|
**多预设管理**:
|
||||||
|
|
||||||
|
- 创建无限数量的提示词预设
|
||||||
|
- 快速在预设间切换
|
||||||
|
- 同时只能激活一个提示词
|
||||||
|
- 活动提示词删除保护
|
||||||
|
|
||||||
|
**跨应用支持**:
|
||||||
|
|
||||||
|
- **Claude**:`~/.claude/CLAUDE.md`
|
||||||
|
- **Codex**:`~/.codex/AGENTS.md`
|
||||||
|
- **Gemini**:`~/.gemini/GEMINI.md`
|
||||||
|
|
||||||
|
**Markdown 编辑器**:
|
||||||
|
|
||||||
|
- 完整的 CodeMirror 6 集成
|
||||||
|
- 语法高亮
|
||||||
|
- 暗色主题(One Dark)
|
||||||
|
- 实时预览
|
||||||
|
|
||||||
|
**智能同步**:
|
||||||
|
|
||||||
|
- **自动写入** - 立即写入 live 文件
|
||||||
|
- **回填保护** - 切换前保存当前内容
|
||||||
|
- **自动导入** - 首次启动从 live 文件导入
|
||||||
|
- **修改保护** - 保留手动修改
|
||||||
|
|
||||||
|
**技术实现**:
|
||||||
|
|
||||||
|
- **后端**:`PromptService`(213 行)
|
||||||
|
- **前端**:PromptPanel(177)、PromptFormModal(160)、MarkdownEditor(159)
|
||||||
|
- **Hooks**:usePromptActions(152 行)
|
||||||
|
- **国际化**:41+ 个翻译键
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### 深度链接协议(ccswitch://)
|
||||||
|
|
||||||
|
通过 URL 方案一键导入供应商配置。
|
||||||
|
|
||||||
|
**功能特性**:
|
||||||
|
|
||||||
|
- 所有平台的协议注册
|
||||||
|
- 从共享链接导入
|
||||||
|
- 生命周期集成
|
||||||
|
- 安全验证
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### 环境变量冲突检测
|
||||||
|
|
||||||
|
智能检测和管理配置冲突。
|
||||||
|
|
||||||
|
**检测范围**:
|
||||||
|
|
||||||
|
- **Claude & Codex** - 跨应用冲突
|
||||||
|
- **Gemini** - 自动发现
|
||||||
|
- **MCP** - 服务器配置冲突
|
||||||
|
|
||||||
|
**管理功能**:
|
||||||
|
|
||||||
|
- 可视化冲突指示器
|
||||||
|
- 解决建议
|
||||||
|
- 覆盖警告
|
||||||
|
- 更改前备份
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 改进优化
|
||||||
|
|
||||||
|
### 供应商管理
|
||||||
|
|
||||||
|
**新增预设**:
|
||||||
|
|
||||||
|
- **DouBaoSeed** - 字节跳动的豆包
|
||||||
|
- **Kimi For Coding** - 月之暗面
|
||||||
|
- **BaiLing** - 百灵 AI
|
||||||
|
- **移除 AnyRouter** - 避免误导
|
||||||
|
|
||||||
|
**增强功能**:
|
||||||
|
|
||||||
|
- Codex 和 Gemini 的模型名称配置
|
||||||
|
- 供应商备注字段用于组织
|
||||||
|
- 增强的预设元数据
|
||||||
|
|
||||||
|
### 配置管理
|
||||||
|
|
||||||
|
- **通用配置迁移** - 从 localStorage 迁移到 `config.json`
|
||||||
|
- **统一持久化** - 跨所有应用共享
|
||||||
|
- **自动导入** - 首次启动配置导入
|
||||||
|
- **回填优先级** - 正确处理 live 文件
|
||||||
|
|
||||||
|
### UI/UX 改进
|
||||||
|
|
||||||
|
**设计系统**:
|
||||||
|
|
||||||
|
- **macOS 原生** - 与系统对齐的配色方案
|
||||||
|
- **窗口居中** - 默认居中位置
|
||||||
|
- **视觉优化** - 改进的间距和层次
|
||||||
|
|
||||||
|
**交互优化**:
|
||||||
|
|
||||||
|
- **密码输入** - 修复 Edge/IE 显示按钮
|
||||||
|
- **URL 溢出** - 修复卡片溢出
|
||||||
|
- **错误复制** - 可复制到剪贴板的错误
|
||||||
|
- **托盘同步** - 实时拖放同步
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Bug 修复
|
||||||
|
|
||||||
|
### 关键修复
|
||||||
|
|
||||||
|
- **用量脚本验证** - 边界检查
|
||||||
|
- **Gemini 验证** - 放宽约束
|
||||||
|
- **TOML 解析** - CJK 引号处理
|
||||||
|
- **MCP 字段** - 自定义字段保留
|
||||||
|
- **白屏** - FormLabel 崩溃修复
|
||||||
|
|
||||||
|
### 稳定性
|
||||||
|
|
||||||
|
- **托盘安全** - 模式匹配替代 unwrap
|
||||||
|
- **错误隔离** - 托盘失败不阻塞操作
|
||||||
|
- **导入分类** - 正确的类别分配
|
||||||
|
|
||||||
|
### UI 修复
|
||||||
|
|
||||||
|
- **模型占位符** - 移除误导性提示
|
||||||
|
- **Base URL** - 第三方供应商自动填充
|
||||||
|
- **拖拽排序** - 托盘菜单同步
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 技术改进
|
||||||
|
|
||||||
|
### 架构
|
||||||
|
|
||||||
|
**MCP v3.7.0**:
|
||||||
|
|
||||||
|
- 移除遗留代码(约 1,000 行)
|
||||||
|
- 统一初始化结构
|
||||||
|
- 保持向后兼容性
|
||||||
|
- 全面的代码格式化
|
||||||
|
|
||||||
|
**平台兼容性**:
|
||||||
|
|
||||||
|
- Windows winreg API 修复(v0.52)
|
||||||
|
- 安全模式匹配(无 `unwrap()`)
|
||||||
|
- 跨平台托盘处理
|
||||||
|
|
||||||
|
### 配置
|
||||||
|
|
||||||
|
**同步机制**:
|
||||||
|
|
||||||
|
- 跨所有应用的 MCP 同步
|
||||||
|
- Gemini 表单-编辑器同步
|
||||||
|
- 双文件读取(.env + settings.json)
|
||||||
|
|
||||||
|
**验证增强**:
|
||||||
|
|
||||||
|
- 输入边界检查
|
||||||
|
- TOML 引号规范化(CJK)
|
||||||
|
- 自定义字段保留
|
||||||
|
- 增强的错误消息
|
||||||
|
|
||||||
|
### 代码质量
|
||||||
|
|
||||||
|
**类型安全**:
|
||||||
|
|
||||||
|
- 完整的 TypeScript 覆盖
|
||||||
|
- Rust 类型改进
|
||||||
|
- API 契约验证
|
||||||
|
|
||||||
|
**测试**:
|
||||||
|
|
||||||
|
- 简化的断言
|
||||||
|
- 更好的测试覆盖
|
||||||
|
- 集成测试更新
|
||||||
|
|
||||||
|
**依赖项**:
|
||||||
|
|
||||||
|
- Tauri 2.8.x
|
||||||
|
- Rust:`anyhow`、`zip`、`serde_yaml`、`tempfile`
|
||||||
|
- 前端:CodeMirror 6 包
|
||||||
|
- winreg 0.52(Windows)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 技术统计
|
||||||
|
|
||||||
|
```
|
||||||
|
总体变更:
|
||||||
|
- 提交数:85
|
||||||
|
- 文件数:152 个文件变更
|
||||||
|
- 新增:+18,104 行
|
||||||
|
- 删除:-3,732 行
|
||||||
|
|
||||||
|
新增模块:
|
||||||
|
- Skills 管理:2,034 行(21 个文件)
|
||||||
|
- Prompts 管理:1,302 行(20 个文件)
|
||||||
|
- Gemini 集成:约 1,000 行
|
||||||
|
- MCP 重构:约 3,000 行重构
|
||||||
|
|
||||||
|
代码分布:
|
||||||
|
- 后端(Rust):约 4,500 行新增
|
||||||
|
- 前端(React):约 3,000 行新增
|
||||||
|
- 配置:约 1,500 行重构
|
||||||
|
- 测试:约 500 行
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 战略定位
|
||||||
|
|
||||||
|
### 从工具到平台
|
||||||
|
|
||||||
|
v3.7.0 代表了 CC Switch 定位的转变:
|
||||||
|
|
||||||
|
| 方面 | v3.6 | v3.7.0 |
|
||||||
|
| -------- | -------------- | ----------------------- |
|
||||||
|
| **身份** | 供应商切换器 | AI CLI 管理平台 |
|
||||||
|
| **范围** | 配置管理 | 生态系统管理 |
|
||||||
|
| **应用** | Claude + Codex | Claude + Codex + Gemini |
|
||||||
|
| **能力** | 切换配置 | 扩展能力(Skills) |
|
||||||
|
| **定制** | 手动编辑 | 可视化管理(Prompts) |
|
||||||
|
| **集成** | 孤立应用 | 统一管理(MCP) |
|
||||||
|
|
||||||
|
### AI CLI 管理六大支柱
|
||||||
|
|
||||||
|
1. **配置管理** - 供应商切换和管理
|
||||||
|
2. **能力扩展** - Skills 安装和生命周期
|
||||||
|
3. **行为定制** - 系统提示词预设
|
||||||
|
4. **生态集成** - 深度链接和共享
|
||||||
|
5. **多 AI 支持** - Claude/Codex/Gemini
|
||||||
|
6. **智能检测** - 冲突预防
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 下载与安装
|
||||||
|
|
||||||
|
### 系统要求
|
||||||
|
|
||||||
|
- **Windows**:Windows 10+
|
||||||
|
- **macOS**:macOS 10.15(Catalina)+
|
||||||
|
- **Linux**:Ubuntu 22.04+ / Debian 11+ / Fedora 34+
|
||||||
|
|
||||||
|
### 下载链接
|
||||||
|
|
||||||
|
访问 [Releases](https://github.com/farion1231/cc-switch/releases/latest) 下载:
|
||||||
|
|
||||||
|
- **Windows**:`CC-Switch-v3.7.0-Windows.msi` 或 `-Portable.zip`
|
||||||
|
- **macOS**:`CC-Switch-v3.7.0-macOS.tar.gz` 或 `.zip`
|
||||||
|
- **Linux**:`CC-Switch-v3.7.0-Linux.AppImage` 或 `.deb`
|
||||||
|
|
||||||
|
### Homebrew(macOS)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
brew tap farion1231/ccswitch
|
||||||
|
brew install --cask cc-switch
|
||||||
|
```
|
||||||
|
|
||||||
|
更新:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
brew upgrade --cask cc-switch
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 迁移说明
|
||||||
|
|
||||||
|
### 从 v3.6.x 升级
|
||||||
|
|
||||||
|
**自动迁移** - 无需任何操作,配置完全兼容
|
||||||
|
|
||||||
|
### 从 v3.1.x 或更早版本升级
|
||||||
|
|
||||||
|
**需要两步迁移**:
|
||||||
|
|
||||||
|
1. 首先升级到 v3.2.x(执行一次性迁移)
|
||||||
|
2. 然后升级到 v3.7.0
|
||||||
|
|
||||||
|
### 新功能
|
||||||
|
|
||||||
|
- **Skills**:无需迁移,全新开始
|
||||||
|
- **Prompts**:首次启动时从 live 文件自动导入
|
||||||
|
- **Gemini**:需要单独安装 Gemini CLI
|
||||||
|
- **MCP v3.7.0**:与之前的配置向后兼容
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 致谢
|
||||||
|
|
||||||
|
### 贡献者
|
||||||
|
|
||||||
|
感谢所有让这个版本成为可能的贡献者:
|
||||||
|
|
||||||
|
- [@YoVinchen](https://github.com/YoVinchen) - Skills & Prompts & Geimini 集成实现
|
||||||
|
- [@farion1231](https://github.com/farion1231) - 从开发沦为 issue 回复机
|
||||||
|
- 社区成员的测试和反馈
|
||||||
|
|
||||||
|
### 赞助商
|
||||||
|
|
||||||
|
**Z.ai** - GLM CODING PLAN 赞助商
|
||||||
|
[通过此链接获得 10% 折扣](https://z.ai/subscribe?ic=8JVLJQFSKB)
|
||||||
|
|
||||||
|
**PackyCode** - API 中继服务合作伙伴
|
||||||
|
[使用 "cc-switch" 代码注册可享受 10% 折扣](https://www.packyapi.com/register?aff=cc-switch)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 反馈与支持
|
||||||
|
|
||||||
|
- **问题反馈**:[GitHub Issues](https://github.com/farion1231/cc-switch/issues)
|
||||||
|
- **讨论**:[GitHub Discussions](https://github.com/farion1231/cc-switch/discussions)
|
||||||
|
- **文档**:[README](../README_ZH.md)
|
||||||
|
- **更新日志**:[CHANGELOG.md](../CHANGELOG.md)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 未来展望
|
||||||
|
|
||||||
|
**v3.8.0 预览**(暂定):
|
||||||
|
|
||||||
|
- 本地代理功能
|
||||||
|
|
||||||
|
敬请期待更多更新!
|
||||||
481
docs/release-note-v3.7.1-en.md
Normal file
481
docs/release-note-v3.7.1-en.md
Normal file
@@ -0,0 +1,481 @@
|
|||||||
|
# CC Switch v3.7.1
|
||||||
|
|
||||||
|
> Stability Enhancements and User Experience Improvements
|
||||||
|
|
||||||
|
**[中文更新说明 Chinese Documentation →](release-note-v3.7.1-zh.md)**
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## v3.7.1 Updates
|
||||||
|
|
||||||
|
**Release Date**: 2025-11-22
|
||||||
|
**Code Changes**: 17 files, +524 / -81 lines
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
- **Fix Third-Party Skills Installation Failure** (#268)
|
||||||
|
Fixed installation issues for skills repositories with custom subdirectories, now supports repos like `ComposioHQ/awesome-claude-skills` with subdirectories
|
||||||
|
|
||||||
|
- **Fix Gemini Configuration Persistence Issue**
|
||||||
|
Resolved the issue where settings.json edits in Gemini form were lost when switching providers
|
||||||
|
|
||||||
|
- **Prevent Dialogs from Closing on Overlay Click**
|
||||||
|
Added protection against clicking overlay/backdrop, preventing accidental form data loss across all 11 dialog components
|
||||||
|
|
||||||
|
### New Features
|
||||||
|
|
||||||
|
- **Gemini Configuration Directory Support** (#255)
|
||||||
|
Added Gemini configuration directory option in settings, supports customizing `~/.gemini/` path
|
||||||
|
|
||||||
|
- **ArchLinux Installation Support** (#259)
|
||||||
|
Added AUR installation method: `paru -S cc-switch-bin`
|
||||||
|
|
||||||
|
### Improvements
|
||||||
|
|
||||||
|
- **Skills Error Message i18n Enhancement**
|
||||||
|
Added 28+ detailed error messages (English & Chinese) with specific resolution suggestions, extended download timeout from 15s to 60s
|
||||||
|
|
||||||
|
- **Code Formatting**
|
||||||
|
Applied unified Rust and TypeScript code formatting standards
|
||||||
|
|
||||||
|
### Download
|
||||||
|
|
||||||
|
Visit [Releases](https://github.com/farion1231/cc-switch/releases/latest) to download the latest version
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## v3.7.0 Complete Release Notes
|
||||||
|
|
||||||
|
> From Provider Switcher to All-in-One AI CLI Management Platform
|
||||||
|
|
||||||
|
**Release Date**: 2025-11-19
|
||||||
|
**Commits**: 85 from v3.6.0
|
||||||
|
**Code Changes**: 152 files, +18,104 / -3,732 lines
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## New Features
|
||||||
|
|
||||||
|
### Gemini CLI Integration
|
||||||
|
|
||||||
|
Complete support for Google Gemini CLI, becoming the third supported application (Claude Code, Codex, Gemini).
|
||||||
|
|
||||||
|
**Core Capabilities**:
|
||||||
|
|
||||||
|
- **Dual-file configuration** - Support for both `.env` and `settings.json` formats
|
||||||
|
- **Auto-detection** - Automatically detect `GOOGLE_GEMINI_BASE_URL`, `GEMINI_MODEL`, etc.
|
||||||
|
- **Full MCP support** - Complete MCP server management for Gemini
|
||||||
|
- **Deep link integration** - Import via `ccswitch://` protocol
|
||||||
|
- **System tray** - Quick-switch from tray menu
|
||||||
|
|
||||||
|
**Provider Presets**:
|
||||||
|
|
||||||
|
- **Google Official** - OAuth authentication support
|
||||||
|
- **PackyCode** - Partner integration
|
||||||
|
- **Custom** - Full customization support
|
||||||
|
|
||||||
|
**Technical Implementation**:
|
||||||
|
|
||||||
|
- New backend modules: `gemini_config.rs` (20KB), `gemini_mcp.rs`
|
||||||
|
- Form synchronization with environment editor
|
||||||
|
- Dual-file atomic writes
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### MCP v3.7.0 Unified Architecture
|
||||||
|
|
||||||
|
Complete refactoring of MCP management system for cross-application unification.
|
||||||
|
|
||||||
|
**Architecture Improvements**:
|
||||||
|
|
||||||
|
- **Unified panel** - Single interface for Claude/Codex/Gemini MCP servers
|
||||||
|
- **SSE transport** - New Server-Sent Events support
|
||||||
|
- **Smart parser** - Fault-tolerant JSON parsing
|
||||||
|
- **Format correction** - Auto-fix Codex `[mcp_servers]` format
|
||||||
|
- **Extended fields** - Preserve custom TOML fields
|
||||||
|
|
||||||
|
**User Experience**:
|
||||||
|
|
||||||
|
- Default app selection in forms
|
||||||
|
- JSON formatter for validation
|
||||||
|
- Improved visual hierarchy
|
||||||
|
- Better error messages
|
||||||
|
|
||||||
|
**Import/Export**:
|
||||||
|
|
||||||
|
- Unified import from all three apps
|
||||||
|
- Bidirectional synchronization
|
||||||
|
- State preservation
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Claude Skills Management System
|
||||||
|
|
||||||
|
**Approximately 2,000 lines of code** - A complete skill ecosystem platform.
|
||||||
|
|
||||||
|
**GitHub Integration**:
|
||||||
|
|
||||||
|
- Auto-scan skills from GitHub repositories
|
||||||
|
- Pre-configured repos:
|
||||||
|
- `ComposioHQ/awesome-claude-skills` - Curated collection
|
||||||
|
- `anthropics/skills` - Official Anthropic skills
|
||||||
|
- `cexll/myclaude` - Community contributions
|
||||||
|
- Add custom repositories
|
||||||
|
- Subdirectory scanning support (`skillsPath`)
|
||||||
|
|
||||||
|
**Lifecycle Management**:
|
||||||
|
|
||||||
|
- **Discover** - Auto-detect `SKILL.md` files
|
||||||
|
- **Install** - One-click to `~/.claude/skills/`
|
||||||
|
- **Uninstall** - Safe removal with tracking
|
||||||
|
- **Update** - Check for updates (infrastructure ready)
|
||||||
|
|
||||||
|
**Technical Architecture**:
|
||||||
|
|
||||||
|
- **Backend**: `SkillService` (526 lines) with GitHub API integration
|
||||||
|
- **Frontend**: SkillsPage, SkillCard, RepoManager
|
||||||
|
- **UI Components**: Badge, Card, Table (shadcn/ui)
|
||||||
|
- **State**: Persistent storage in `config.json`
|
||||||
|
- **i18n**: 47+ translation keys
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Prompts Management System
|
||||||
|
|
||||||
|
**Approximately 1,300 lines of code** - Complete system prompt management.
|
||||||
|
|
||||||
|
**Multi-Preset Management**:
|
||||||
|
|
||||||
|
- Create unlimited prompt presets
|
||||||
|
- Quick switch between presets
|
||||||
|
- One active prompt at a time
|
||||||
|
- Delete protection for active prompts
|
||||||
|
|
||||||
|
**Cross-App Support**:
|
||||||
|
|
||||||
|
- **Claude**: `~/.claude/CLAUDE.md`
|
||||||
|
- **Codex**: `~/.codex/AGENTS.md`
|
||||||
|
- **Gemini**: `~/.gemini/GEMINI.md`
|
||||||
|
|
||||||
|
**Markdown Editor**:
|
||||||
|
|
||||||
|
- Full-featured CodeMirror 6 integration
|
||||||
|
- Syntax highlighting
|
||||||
|
- Dark theme (One Dark)
|
||||||
|
- Real-time preview
|
||||||
|
|
||||||
|
**Smart Synchronization**:
|
||||||
|
|
||||||
|
- **Auto-write** - Immediately write to live files
|
||||||
|
- **Backfill protection** - Save current content before switching
|
||||||
|
- **Auto-import** - Import from live files on first launch
|
||||||
|
- **Modification protection** - Preserve manual modifications
|
||||||
|
|
||||||
|
**Technical Implementation**:
|
||||||
|
|
||||||
|
- **Backend**: `PromptService` (213 lines)
|
||||||
|
- **Frontend**: PromptPanel (177), PromptFormModal (160), MarkdownEditor (159)
|
||||||
|
- **Hooks**: usePromptActions (152 lines)
|
||||||
|
- **i18n**: 41+ translation keys
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Deep Link Protocol (ccswitch://)
|
||||||
|
|
||||||
|
One-click provider configuration import via URL scheme.
|
||||||
|
|
||||||
|
**Features**:
|
||||||
|
|
||||||
|
- Protocol registration on all platforms
|
||||||
|
- Import from shared links
|
||||||
|
- Lifecycle integration
|
||||||
|
- Security validation
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Environment Variable Conflict Detection
|
||||||
|
|
||||||
|
Intelligent detection and management of configuration conflicts.
|
||||||
|
|
||||||
|
**Detection Scope**:
|
||||||
|
|
||||||
|
- **Claude & Codex** - Cross-app conflicts
|
||||||
|
- **Gemini** - Auto-discovery
|
||||||
|
- **MCP** - Server configuration conflicts
|
||||||
|
|
||||||
|
**Management Features**:
|
||||||
|
|
||||||
|
- Visual conflict indicators
|
||||||
|
- Resolution suggestions
|
||||||
|
- Override warnings
|
||||||
|
- Backup before changes
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Improvements
|
||||||
|
|
||||||
|
### Provider Management
|
||||||
|
|
||||||
|
**New Presets**:
|
||||||
|
|
||||||
|
- **DouBaoSeed** - ByteDance's DouBao
|
||||||
|
- **Kimi For Coding** - Moonshot AI
|
||||||
|
- **BaiLing** - BaiLing AI
|
||||||
|
- **Removed AnyRouter** - To avoid confusion
|
||||||
|
|
||||||
|
**Enhancements**:
|
||||||
|
|
||||||
|
- Model name configuration for Codex and Gemini
|
||||||
|
- Provider notes field for organization
|
||||||
|
- Enhanced preset metadata
|
||||||
|
|
||||||
|
### Configuration Management
|
||||||
|
|
||||||
|
- **Common config migration** - From localStorage to `config.json`
|
||||||
|
- **Unified persistence** - Shared across all apps
|
||||||
|
- **Auto-import** - First launch configuration import
|
||||||
|
- **Backfill priority** - Correct handling of live files
|
||||||
|
|
||||||
|
### UI/UX Improvements
|
||||||
|
|
||||||
|
**Design System**:
|
||||||
|
|
||||||
|
- **macOS native** - System-aligned color scheme
|
||||||
|
- **Window centering** - Default centered position
|
||||||
|
- **Visual polish** - Improved spacing and hierarchy
|
||||||
|
|
||||||
|
**Interactions**:
|
||||||
|
|
||||||
|
- **Password input** - Fixed Edge/IE reveal buttons
|
||||||
|
- **URL overflow** - Fixed card overflow
|
||||||
|
- **Error copying** - Copy-to-clipboard errors
|
||||||
|
- **Tray sync** - Real-time drag-and-drop sync
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Bug Fixes
|
||||||
|
|
||||||
|
### Critical Fixes
|
||||||
|
|
||||||
|
- **Usage script validation** - Boundary checks
|
||||||
|
- **Gemini validation** - Relaxed constraints
|
||||||
|
- **TOML parsing** - CJK quote handling
|
||||||
|
- **MCP fields** - Custom field preservation
|
||||||
|
- **White screen** - FormLabel crash fix
|
||||||
|
|
||||||
|
### Stability
|
||||||
|
|
||||||
|
- **Tray safety** - Pattern matching instead of unwrap
|
||||||
|
- **Error isolation** - Tray failures don't block operations
|
||||||
|
- **Import classification** - Correct category assignment
|
||||||
|
|
||||||
|
### UI Fixes
|
||||||
|
|
||||||
|
- **Model placeholders** - Removed misleading hints
|
||||||
|
- **Base URL** - Auto-fill for third-party providers
|
||||||
|
- **Drag sort** - Tray menu synchronization
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Technical Improvements
|
||||||
|
|
||||||
|
### Architecture
|
||||||
|
|
||||||
|
**MCP v3.7.0**:
|
||||||
|
|
||||||
|
- Removed legacy code (~1,000 lines)
|
||||||
|
- Unified initialization structure
|
||||||
|
- Backward compatibility maintained
|
||||||
|
- Comprehensive code formatting
|
||||||
|
|
||||||
|
**Platform Compatibility**:
|
||||||
|
|
||||||
|
- Windows winreg API fix (v0.52)
|
||||||
|
- Safe pattern matching (no `unwrap()`)
|
||||||
|
- Cross-platform tray handling
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
**Synchronization**:
|
||||||
|
|
||||||
|
- MCP sync across all apps
|
||||||
|
- Gemini form-editor sync
|
||||||
|
- Dual-file reading (.env + settings.json)
|
||||||
|
|
||||||
|
**Validation**:
|
||||||
|
|
||||||
|
- Input boundary checks
|
||||||
|
- TOML quote normalization (CJK)
|
||||||
|
- Custom field preservation
|
||||||
|
- Enhanced error messages
|
||||||
|
|
||||||
|
### Code Quality
|
||||||
|
|
||||||
|
**Type Safety**:
|
||||||
|
|
||||||
|
- Complete TypeScript coverage
|
||||||
|
- Rust type refinements
|
||||||
|
- API contract validation
|
||||||
|
|
||||||
|
**Testing**:
|
||||||
|
|
||||||
|
- Simplified assertions
|
||||||
|
- Better test coverage
|
||||||
|
- Integration test updates
|
||||||
|
|
||||||
|
**Dependencies**:
|
||||||
|
|
||||||
|
- Tauri 2.8.x
|
||||||
|
- Rust: `anyhow`, `zip`, `serde_yaml`, `tempfile`
|
||||||
|
- Frontend: CodeMirror 6 packages
|
||||||
|
- winreg 0.52 (Windows)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Technical Statistics
|
||||||
|
|
||||||
|
```
|
||||||
|
Total Changes:
|
||||||
|
- Commits: 85
|
||||||
|
- Files: 152 changed
|
||||||
|
- Additions: +18,104 lines
|
||||||
|
- Deletions: -3,732 lines
|
||||||
|
|
||||||
|
New Modules:
|
||||||
|
- Skills Management: 2,034 lines (21 files)
|
||||||
|
- Prompts Management: 1,302 lines (20 files)
|
||||||
|
- Gemini Integration: ~1,000 lines
|
||||||
|
- MCP Refactor: ~3,000 lines refactored
|
||||||
|
|
||||||
|
Code Distribution:
|
||||||
|
- Backend (Rust): ~4,500 lines new
|
||||||
|
- Frontend (React): ~3,000 lines new
|
||||||
|
- Configuration: ~1,500 lines refactored
|
||||||
|
- Tests: ~500 lines
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Strategic Positioning
|
||||||
|
|
||||||
|
### From Tool to Platform
|
||||||
|
|
||||||
|
v3.7.0 represents a shift in CC Switch's positioning:
|
||||||
|
|
||||||
|
| Aspect | v3.6 | v3.7.0 |
|
||||||
|
| ----------------- | ------------------------ | ---------------------------- |
|
||||||
|
| **Identity** | Provider Switcher | AI CLI Management Platform |
|
||||||
|
| **Scope** | Configuration Management | Ecosystem Management |
|
||||||
|
| **Applications** | Claude + Codex | Claude + Codex + Gemini |
|
||||||
|
| **Capabilities** | Switch configs | Extend capabilities (Skills) |
|
||||||
|
| **Customization** | Manual editing | Visual management (Prompts) |
|
||||||
|
| **Integration** | Isolated apps | Unified management (MCP) |
|
||||||
|
|
||||||
|
### Six Pillars of AI CLI Management
|
||||||
|
|
||||||
|
1. **Configuration Management** - Provider switching and management
|
||||||
|
2. **Capability Extension** - Skills installation and lifecycle
|
||||||
|
3. **Behavior Customization** - System prompt presets
|
||||||
|
4. **Ecosystem Integration** - Deep links and sharing
|
||||||
|
5. **Multi-AI Support** - Claude/Codex/Gemini
|
||||||
|
6. **Intelligent Detection** - Conflict prevention
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Download & Installation
|
||||||
|
|
||||||
|
### System Requirements
|
||||||
|
|
||||||
|
- **Windows**: Windows 10+
|
||||||
|
- **macOS**: macOS 10.15 (Catalina)+
|
||||||
|
- **Linux**: Ubuntu 22.04+ / Debian 11+ / Fedora 34+ / ArchLinux
|
||||||
|
|
||||||
|
### Download Links
|
||||||
|
|
||||||
|
Visit [Releases](https://github.com/farion1231/cc-switch/releases/latest) to download:
|
||||||
|
|
||||||
|
- **Windows**: `CC-Switch-Windows.msi` or `-Portable.zip`
|
||||||
|
- **macOS**: `CC-Switch-macOS.tar.gz` or `.zip`
|
||||||
|
- **Linux**: `CC-Switch-Linux.AppImage` or `.deb`
|
||||||
|
- **ArchLinux**: `paru -S cc-switch-bin`
|
||||||
|
|
||||||
|
### Homebrew (macOS)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
brew tap farion1231/ccswitch
|
||||||
|
brew install --cask cc-switch
|
||||||
|
```
|
||||||
|
|
||||||
|
Update:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
brew upgrade --cask cc-switch
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Migration Notes
|
||||||
|
|
||||||
|
### From v3.6.x
|
||||||
|
|
||||||
|
**Automatic migration** - No action required, configs are fully compatible
|
||||||
|
|
||||||
|
### From v3.1.x or Earlier
|
||||||
|
|
||||||
|
**Two-step migration required**:
|
||||||
|
|
||||||
|
1. First upgrade to v3.2.x (performs one-time migration)
|
||||||
|
2. Then upgrade to v3.7.0
|
||||||
|
|
||||||
|
### New Features
|
||||||
|
|
||||||
|
- **Skills**: No migration needed, start fresh
|
||||||
|
- **Prompts**: Auto-import from live files on first launch
|
||||||
|
- **Gemini**: Install Gemini CLI separately if needed
|
||||||
|
- **MCP v3.7.0**: Backward compatible with previous configs
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Acknowledgments
|
||||||
|
|
||||||
|
### Contributors
|
||||||
|
|
||||||
|
Thanks to all contributors who made this release possible:
|
||||||
|
|
||||||
|
- [@YoVinchen](https://github.com/YoVinchen) - Skills & Prompts & Gemini integration implementation
|
||||||
|
- [@farion1231](https://github.com/farion1231) - From developer to issue responder
|
||||||
|
- Community members for testing and feedback
|
||||||
|
|
||||||
|
### Sponsors
|
||||||
|
|
||||||
|
**Z.ai** - GLM CODING PLAN sponsor
|
||||||
|
[Get 10% OFF with this link](https://z.ai/subscribe?ic=8JVLJQFSKB)
|
||||||
|
|
||||||
|
**PackyCode** - API relay service partner
|
||||||
|
[Register with "cc-switch" code for 10% discount](https://www.packyapi.com/register?aff=cc-switch)
|
||||||
|
|
||||||
|
**ShanDianShuo** - Local-first AI voice input
|
||||||
|
[Free download](https://shandianshuo.cn) for Mac/Win
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Feedback & Support
|
||||||
|
|
||||||
|
- **Issues**: [GitHub Issues](https://github.com/farion1231/cc-switch/issues)
|
||||||
|
- **Discussions**: [GitHub Discussions](https://github.com/farion1231/cc-switch/discussions)
|
||||||
|
- **Documentation**: [README](../README.md)
|
||||||
|
- **Changelog**: [CHANGELOG.md](../CHANGELOG.md)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## What's Next
|
||||||
|
|
||||||
|
**v3.8.0 Preview** (Tentative):
|
||||||
|
|
||||||
|
- Local proxy functionality
|
||||||
|
|
||||||
|
Stay tuned for more updates!
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Happy Coding!**
|
||||||
481
docs/release-note-v3.7.1-zh.md
Normal file
481
docs/release-note-v3.7.1-zh.md
Normal file
@@ -0,0 +1,481 @@
|
|||||||
|
# CC Switch v3.7.1
|
||||||
|
|
||||||
|
> 稳定性增强与用户体验改进
|
||||||
|
|
||||||
|
**[English Version →](release-note-v3.7.1-en.md)**
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## v3.7.1 更新内容
|
||||||
|
|
||||||
|
**发布日期**:2025-11-22
|
||||||
|
**代码变更**:17 个文件,+524 / -81 行
|
||||||
|
|
||||||
|
### Bug 修复
|
||||||
|
|
||||||
|
- **修复 Skills 第三方仓库安装失败** (#268)
|
||||||
|
修复使用自定义子目录的 skills 仓库无法安装的问题,支持类似 `ComposioHQ/awesome-claude-skills` 这样带子目录的仓库
|
||||||
|
|
||||||
|
- **修复 Gemini 配置持久化问题**
|
||||||
|
解决在 Gemini 表单中编辑 settings.json 后,切换供应商时修改丢失的问题
|
||||||
|
|
||||||
|
- **防止对话框意外关闭**
|
||||||
|
添加点击遮罩时的保护,避免误操作导致表单数据丢失,影响所有 11 个对话框组件
|
||||||
|
|
||||||
|
### 新增功能
|
||||||
|
|
||||||
|
- **Gemini 配置目录支持** (#255)
|
||||||
|
在设置中添加 Gemini 配置目录选项,支持自定义 `~/.gemini/` 路径
|
||||||
|
|
||||||
|
- **ArchLinux 安装支持** (#259)
|
||||||
|
添加 AUR 安装方式:`paru -S cc-switch-bin`
|
||||||
|
|
||||||
|
### 改进
|
||||||
|
|
||||||
|
- **Skills 错误消息国际化增强**
|
||||||
|
新增 28+ 条详细错误消息(中英文),提供具体的解决建议,下载超时从 15 秒延长到 60 秒
|
||||||
|
|
||||||
|
- **代码格式化**
|
||||||
|
应用统一的 Rust 和 TypeScript 代码格式化标准
|
||||||
|
|
||||||
|
### 下载
|
||||||
|
|
||||||
|
访问 [Releases](https://github.com/farion1231/cc-switch/releases/latest) 下载最新版本
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## v3.7.0 完整更新说明
|
||||||
|
|
||||||
|
> 从供应商切换器到 AI CLI 一体化管理平台
|
||||||
|
|
||||||
|
**发布日期**:2025-11-19
|
||||||
|
**提交数量**:从 v3.6.0 开始 85 个提交
|
||||||
|
**代码变更**:152 个文件,+18,104 / -3,732 行
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 新增功能
|
||||||
|
|
||||||
|
### Gemini CLI 集成
|
||||||
|
|
||||||
|
完整支持 Google Gemini CLI,成为第三个支持的应用(Claude Code、Codex、Gemini)。
|
||||||
|
|
||||||
|
**核心能力**:
|
||||||
|
|
||||||
|
- **双文件配置** - 同时支持 `.env` 和 `settings.json` 格式
|
||||||
|
- **自动检测** - 自动检测 `GOOGLE_GEMINI_BASE_URL`、`GEMINI_MODEL` 等环境变量
|
||||||
|
- **完整 MCP 支持** - 为 Gemini 提供完整的 MCP 服务器管理
|
||||||
|
- **深度链接集成** - 通过 `ccswitch://` 协议导入配置
|
||||||
|
- **系统托盘** - 从托盘菜单快速切换
|
||||||
|
|
||||||
|
**供应商预设**:
|
||||||
|
|
||||||
|
- **Google Official** - 支持 OAuth 认证
|
||||||
|
- **PackyCode** - 合作伙伴集成
|
||||||
|
- **自定义** - 完全自定义支持
|
||||||
|
|
||||||
|
**技术实现**:
|
||||||
|
|
||||||
|
- 新增后端模块:`gemini_config.rs`(20KB)、`gemini_mcp.rs`
|
||||||
|
- 表单与环境编辑器同步
|
||||||
|
- 双文件原子写入
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### MCP v3.7.0 统一架构
|
||||||
|
|
||||||
|
MCP 管理系统完整重构,实现跨应用统一管理。
|
||||||
|
|
||||||
|
**架构改进**:
|
||||||
|
|
||||||
|
- **统一管理面板** - 单一界面管理 Claude/Codex/Gemini MCP 服务器
|
||||||
|
- **SSE 传输类型** - 新增 Server-Sent Events 支持
|
||||||
|
- **智能解析器** - 容错性 JSON 解析
|
||||||
|
- **格式修正** - 自动修复 Codex `[mcp_servers]` 格式
|
||||||
|
- **扩展字段** - 保留自定义 TOML 字段
|
||||||
|
|
||||||
|
**用户体验**:
|
||||||
|
|
||||||
|
- 表单中的默认应用选择
|
||||||
|
- JSON 格式化器用于验证
|
||||||
|
- 改进的视觉层次
|
||||||
|
- 更好的错误消息
|
||||||
|
|
||||||
|
**导入/导出**:
|
||||||
|
|
||||||
|
- 统一从三个应用导入
|
||||||
|
- 双向同步
|
||||||
|
- 状态保持
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Claude Skills 管理系统
|
||||||
|
|
||||||
|
**约 2,000 行代码** - 完整的技能生态平台。
|
||||||
|
|
||||||
|
**GitHub 集成**:
|
||||||
|
|
||||||
|
- 从 GitHub 仓库自动扫描技能
|
||||||
|
- 预配置仓库:
|
||||||
|
- `ComposioHQ/awesome-claude-skills` - 精选集合
|
||||||
|
- `anthropics/skills` - Anthropic 官方技能
|
||||||
|
- `cexll/myclaude` - 社区贡献
|
||||||
|
- 添加自定义仓库
|
||||||
|
- 子目录扫描支持(`skillsPath`)
|
||||||
|
|
||||||
|
**生命周期管理**:
|
||||||
|
|
||||||
|
- **发现** - 自动检测 `SKILL.md` 文件
|
||||||
|
- **安装** - 一键安装到 `~/.claude/skills/`
|
||||||
|
- **卸载** - 安全移除并跟踪状态
|
||||||
|
- **更新** - 检查更新(基础设施已就绪)
|
||||||
|
|
||||||
|
**技术架构**:
|
||||||
|
|
||||||
|
- **后端**:`SkillService`(526 行)集成 GitHub API
|
||||||
|
- **前端**:SkillsPage、SkillCard、RepoManager
|
||||||
|
- **UI 组件**:Badge、Card、Table(shadcn/ui)
|
||||||
|
- **状态**:持久化存储在 `config.json`
|
||||||
|
- **国际化**:47+ 个翻译键
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Prompts 管理系统
|
||||||
|
|
||||||
|
**约 1,300 行代码** - 完整的系统提示词管理。
|
||||||
|
|
||||||
|
**多预设管理**:
|
||||||
|
|
||||||
|
- 创建无限数量的提示词预设
|
||||||
|
- 快速在预设间切换
|
||||||
|
- 同时只能激活一个提示词
|
||||||
|
- 活动提示词删除保护
|
||||||
|
|
||||||
|
**跨应用支持**:
|
||||||
|
|
||||||
|
- **Claude**:`~/.claude/CLAUDE.md`
|
||||||
|
- **Codex**:`~/.codex/AGENTS.md`
|
||||||
|
- **Gemini**:`~/.gemini/GEMINI.md`
|
||||||
|
|
||||||
|
**Markdown 编辑器**:
|
||||||
|
|
||||||
|
- 完整的 CodeMirror 6 集成
|
||||||
|
- 语法高亮
|
||||||
|
- 暗色主题(One Dark)
|
||||||
|
- 实时预览
|
||||||
|
|
||||||
|
**智能同步**:
|
||||||
|
|
||||||
|
- **自动写入** - 立即写入 live 文件
|
||||||
|
- **回填保护** - 切换前保存当前内容
|
||||||
|
- **自动导入** - 首次启动从 live 文件导入
|
||||||
|
- **修改保护** - 保留手动修改
|
||||||
|
|
||||||
|
**技术实现**:
|
||||||
|
|
||||||
|
- **后端**:`PromptService`(213 行)
|
||||||
|
- **前端**:PromptPanel(177)、PromptFormModal(160)、MarkdownEditor(159)
|
||||||
|
- **Hooks**:usePromptActions(152 行)
|
||||||
|
- **国际化**:41+ 个翻译键
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### 深度链接协议(ccswitch://)
|
||||||
|
|
||||||
|
通过 URL 方案一键导入供应商配置。
|
||||||
|
|
||||||
|
**功能特性**:
|
||||||
|
|
||||||
|
- 所有平台的协议注册
|
||||||
|
- 从共享链接导入
|
||||||
|
- 生命周期集成
|
||||||
|
- 安全验证
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### 环境变量冲突检测
|
||||||
|
|
||||||
|
智能检测和管理配置冲突。
|
||||||
|
|
||||||
|
**检测范围**:
|
||||||
|
|
||||||
|
- **Claude & Codex** - 跨应用冲突
|
||||||
|
- **Gemini** - 自动发现
|
||||||
|
- **MCP** - 服务器配置冲突
|
||||||
|
|
||||||
|
**管理功能**:
|
||||||
|
|
||||||
|
- 可视化冲突指示器
|
||||||
|
- 解决建议
|
||||||
|
- 覆盖警告
|
||||||
|
- 更改前备份
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 改进优化
|
||||||
|
|
||||||
|
### 供应商管理
|
||||||
|
|
||||||
|
**新增预设**:
|
||||||
|
|
||||||
|
- **DouBaoSeed** - 字节跳动的豆包
|
||||||
|
- **Kimi For Coding** - 月之暗面
|
||||||
|
- **BaiLing** - 百灵 AI
|
||||||
|
- **移除 AnyRouter** - 避免误导
|
||||||
|
|
||||||
|
**增强功能**:
|
||||||
|
|
||||||
|
- Codex 和 Gemini 的模型名称配置
|
||||||
|
- 供应商备注字段用于组织
|
||||||
|
- 增强的预设元数据
|
||||||
|
|
||||||
|
### 配置管理
|
||||||
|
|
||||||
|
- **通用配置迁移** - 从 localStorage 迁移到 `config.json`
|
||||||
|
- **统一持久化** - 跨所有应用共享
|
||||||
|
- **自动导入** - 首次启动配置导入
|
||||||
|
- **回填优先级** - 正确处理 live 文件
|
||||||
|
|
||||||
|
### UI/UX 改进
|
||||||
|
|
||||||
|
**设计系统**:
|
||||||
|
|
||||||
|
- **macOS 原生** - 与系统对齐的配色方案
|
||||||
|
- **窗口居中** - 默认居中位置
|
||||||
|
- **视觉优化** - 改进的间距和层次
|
||||||
|
|
||||||
|
**交互优化**:
|
||||||
|
|
||||||
|
- **密码输入** - 修复 Edge/IE 显示按钮
|
||||||
|
- **URL 溢出** - 修复卡片溢出
|
||||||
|
- **错误复制** - 可复制到剪贴板的错误
|
||||||
|
- **托盘同步** - 实时拖放同步
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Bug 修复
|
||||||
|
|
||||||
|
### 关键修复
|
||||||
|
|
||||||
|
- **用量脚本验证** - 边界检查
|
||||||
|
- **Gemini 验证** - 放宽约束
|
||||||
|
- **TOML 解析** - CJK 引号处理
|
||||||
|
- **MCP 字段** - 自定义字段保留
|
||||||
|
- **白屏** - FormLabel 崩溃修复
|
||||||
|
|
||||||
|
### 稳定性
|
||||||
|
|
||||||
|
- **托盘安全** - 模式匹配替代 unwrap
|
||||||
|
- **错误隔离** - 托盘失败不阻塞操作
|
||||||
|
- **导入分类** - 正确的类别分配
|
||||||
|
|
||||||
|
### UI 修复
|
||||||
|
|
||||||
|
- **模型占位符** - 移除误导性提示
|
||||||
|
- **Base URL** - 第三方供应商自动填充
|
||||||
|
- **拖拽排序** - 托盘菜单同步
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 技术改进
|
||||||
|
|
||||||
|
### 架构
|
||||||
|
|
||||||
|
**MCP v3.7.0**:
|
||||||
|
|
||||||
|
- 移除遗留代码(约 1,000 行)
|
||||||
|
- 统一初始化结构
|
||||||
|
- 保持向后兼容性
|
||||||
|
- 全面的代码格式化
|
||||||
|
|
||||||
|
**平台兼容性**:
|
||||||
|
|
||||||
|
- Windows winreg API 修复(v0.52)
|
||||||
|
- 安全模式匹配(无 `unwrap()`)
|
||||||
|
- 跨平台托盘处理
|
||||||
|
|
||||||
|
### 配置
|
||||||
|
|
||||||
|
**同步机制**:
|
||||||
|
|
||||||
|
- 跨所有应用的 MCP 同步
|
||||||
|
- Gemini 表单-编辑器同步
|
||||||
|
- 双文件读取(.env + settings.json)
|
||||||
|
|
||||||
|
**验证增强**:
|
||||||
|
|
||||||
|
- 输入边界检查
|
||||||
|
- TOML 引号规范化(CJK)
|
||||||
|
- 自定义字段保留
|
||||||
|
- 增强的错误消息
|
||||||
|
|
||||||
|
### 代码质量
|
||||||
|
|
||||||
|
**类型安全**:
|
||||||
|
|
||||||
|
- 完整的 TypeScript 覆盖
|
||||||
|
- Rust 类型改进
|
||||||
|
- API 契约验证
|
||||||
|
|
||||||
|
**测试**:
|
||||||
|
|
||||||
|
- 简化的断言
|
||||||
|
- 更好的测试覆盖
|
||||||
|
- 集成测试更新
|
||||||
|
|
||||||
|
**依赖项**:
|
||||||
|
|
||||||
|
- Tauri 2.8.x
|
||||||
|
- Rust:`anyhow`、`zip`、`serde_yaml`、`tempfile`
|
||||||
|
- 前端:CodeMirror 6 包
|
||||||
|
- winreg 0.52(Windows)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 技术统计
|
||||||
|
|
||||||
|
```
|
||||||
|
总体变更:
|
||||||
|
- 提交数:85
|
||||||
|
- 文件数:152 个文件变更
|
||||||
|
- 新增:+18,104 行
|
||||||
|
- 删除:-3,732 行
|
||||||
|
|
||||||
|
新增模块:
|
||||||
|
- Skills 管理:2,034 行(21 个文件)
|
||||||
|
- Prompts 管理:1,302 行(20 个文件)
|
||||||
|
- Gemini 集成:约 1,000 行
|
||||||
|
- MCP 重构:约 3,000 行重构
|
||||||
|
|
||||||
|
代码分布:
|
||||||
|
- 后端(Rust):约 4,500 行新增
|
||||||
|
- 前端(React):约 3,000 行新增
|
||||||
|
- 配置:约 1,500 行重构
|
||||||
|
- 测试:约 500 行
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 战略定位
|
||||||
|
|
||||||
|
### 从工具到平台
|
||||||
|
|
||||||
|
v3.7.0 代表了 CC Switch 定位的转变:
|
||||||
|
|
||||||
|
| 方面 | v3.6 | v3.7.0 |
|
||||||
|
| -------- | -------------- | ----------------------- |
|
||||||
|
| **身份** | 供应商切换器 | AI CLI 管理平台 |
|
||||||
|
| **范围** | 配置管理 | 生态系统管理 |
|
||||||
|
| **应用** | Claude + Codex | Claude + Codex + Gemini |
|
||||||
|
| **能力** | 切换配置 | 扩展能力(Skills) |
|
||||||
|
| **定制** | 手动编辑 | 可视化管理(Prompts) |
|
||||||
|
| **集成** | 孤立应用 | 统一管理(MCP) |
|
||||||
|
|
||||||
|
### AI CLI 管理六大支柱
|
||||||
|
|
||||||
|
1. **配置管理** - 供应商切换和管理
|
||||||
|
2. **能力扩展** - Skills 安装和生命周期
|
||||||
|
3. **行为定制** - 系统提示词预设
|
||||||
|
4. **生态集成** - 深度链接和共享
|
||||||
|
5. **多 AI 支持** - Claude/Codex/Gemini
|
||||||
|
6. **智能检测** - 冲突预防
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 下载与安装
|
||||||
|
|
||||||
|
### 系统要求
|
||||||
|
|
||||||
|
- **Windows**:Windows 10+
|
||||||
|
- **macOS**:macOS 10.15(Catalina)+
|
||||||
|
- **Linux**:Ubuntu 22.04+ / Debian 11+ / Fedora 34+ / ArchLinux
|
||||||
|
|
||||||
|
### 下载链接
|
||||||
|
|
||||||
|
访问 [Releases](https://github.com/farion1231/cc-switch/releases/latest) 下载:
|
||||||
|
|
||||||
|
- **Windows**:`CC-Switch-Windows.msi` 或 `-Portable.zip`
|
||||||
|
- **macOS**:`CC-Switch-macOS.tar.gz` 或 `.zip`
|
||||||
|
- **Linux**:`CC-Switch-Linux.AppImage` 或 `.deb`
|
||||||
|
- **ArchLinux**:`paru -S cc-switch-bin`
|
||||||
|
|
||||||
|
### Homebrew(macOS)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
brew tap farion1231/ccswitch
|
||||||
|
brew install --cask cc-switch
|
||||||
|
```
|
||||||
|
|
||||||
|
更新:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
brew upgrade --cask cc-switch
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 迁移说明
|
||||||
|
|
||||||
|
### 从 v3.6.x 升级
|
||||||
|
|
||||||
|
**自动迁移** - 无需任何操作,配置完全兼容
|
||||||
|
|
||||||
|
### 从 v3.1.x 或更早版本升级
|
||||||
|
|
||||||
|
**需要两步迁移**:
|
||||||
|
|
||||||
|
1. 首先升级到 v3.2.x(执行一次性迁移)
|
||||||
|
2. 然后升级到 v3.7.0
|
||||||
|
|
||||||
|
### 新功能
|
||||||
|
|
||||||
|
- **Skills**:无需迁移,全新开始
|
||||||
|
- **Prompts**:首次启动时从 live 文件自动导入
|
||||||
|
- **Gemini**:需要单独安装 Gemini CLI
|
||||||
|
- **MCP v3.7.0**:与之前的配置向后兼容
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 致谢
|
||||||
|
|
||||||
|
### 贡献者
|
||||||
|
|
||||||
|
感谢所有让这个版本成为可能的贡献者:
|
||||||
|
|
||||||
|
- [@YoVinchen](https://github.com/YoVinchen) - Skills & Prompts & Gemini 集成实现
|
||||||
|
- [@farion1231](https://github.com/farion1231) - 从开发沦为 issue 回复机
|
||||||
|
- 社区成员的测试和反馈
|
||||||
|
|
||||||
|
### 赞助商
|
||||||
|
|
||||||
|
**智谱AI** - GLM CODING PLAN 赞助商
|
||||||
|
[使用此链接购买可享九折优惠](https://www.bigmodel.cn/claude-code?ic=RRVJPB5SII)
|
||||||
|
|
||||||
|
**PackyCode** - API 中转服务合作伙伴
|
||||||
|
[使用 "cc-switch" 优惠码注册享 9 折优惠](https://www.packyapi.com/register?aff=cc-switch)
|
||||||
|
|
||||||
|
**闪电说** - 本地优先的 AI 语音输入法
|
||||||
|
[免费下载](https://shandianshuo.cn) Mac/Win 双平台
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 反馈与支持
|
||||||
|
|
||||||
|
- **问题反馈**:[GitHub Issues](https://github.com/farion1231/cc-switch/issues)
|
||||||
|
- **讨论**:[GitHub Discussions](https://github.com/farion1231/cc-switch/discussions)
|
||||||
|
- **文档**:[README](../README_ZH.md)
|
||||||
|
- **更新日志**:[CHANGELOG.md](../CHANGELOG.md)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 未来展望
|
||||||
|
|
||||||
|
**v3.8.0 预览**(暂定):
|
||||||
|
|
||||||
|
- 本地代理功能
|
||||||
|
|
||||||
|
敬请期待更多更新!
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Happy Coding!**
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "cc-switch",
|
"name": "cc-switch",
|
||||||
"version": "3.6.2",
|
"version": "3.7.1",
|
||||||
"description": "Claude Code & Codex 供应商切换工具",
|
"description": "All-in-One Assistant for Claude Code, Codex & Gemini CLI",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "pnpm tauri dev",
|
"dev": "pnpm tauri dev",
|
||||||
"build": "pnpm tauri build",
|
"build": "pnpm tauri build",
|
||||||
@@ -46,6 +46,7 @@
|
|||||||
"@dnd-kit/sortable": "^10.0.0",
|
"@dnd-kit/sortable": "^10.0.0",
|
||||||
"@dnd-kit/utilities": "^3.2.2",
|
"@dnd-kit/utilities": "^3.2.2",
|
||||||
"@hookform/resolvers": "^5.2.2",
|
"@hookform/resolvers": "^5.2.2",
|
||||||
|
"@lobehub/icons-static-svg": "^1.73.0",
|
||||||
"@radix-ui/react-checkbox": "^1.3.3",
|
"@radix-ui/react-checkbox": "^1.3.3",
|
||||||
"@radix-ui/react-dialog": "^1.1.15",
|
"@radix-ui/react-dialog": "^1.1.15",
|
||||||
"@radix-ui/react-dropdown-menu": "^2.1.16",
|
"@radix-ui/react-dropdown-menu": "^2.1.16",
|
||||||
|
|||||||
8
pnpm-lock.yaml
generated
8
pnpm-lock.yaml
generated
@@ -41,6 +41,9 @@ importers:
|
|||||||
'@hookform/resolvers':
|
'@hookform/resolvers':
|
||||||
specifier: ^5.2.2
|
specifier: ^5.2.2
|
||||||
version: 5.2.2(react-hook-form@7.65.0(react@18.3.1))
|
version: 5.2.2(react-hook-form@7.65.0(react@18.3.1))
|
||||||
|
'@lobehub/icons-static-svg':
|
||||||
|
specifier: ^1.73.0
|
||||||
|
version: 1.73.0
|
||||||
'@radix-ui/react-checkbox':
|
'@radix-ui/react-checkbox':
|
||||||
specifier: ^1.3.3
|
specifier: ^1.3.3
|
||||||
version: 1.3.3(@types/react-dom@18.3.7(@types/react@18.3.23))(@types/react@18.3.23)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
|
version: 1.3.3(@types/react-dom@18.3.7(@types/react@18.3.23))(@types/react@18.3.23)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
|
||||||
@@ -609,6 +612,9 @@ packages:
|
|||||||
'@lezer/markdown@1.6.0':
|
'@lezer/markdown@1.6.0':
|
||||||
resolution: {integrity: sha512-AXb98u3M6BEzTnreBnGtQaF7xFTiMA92Dsy5tqEjpacbjRxDSFdN4bKJo9uvU4cEEOS7D2B9MT7kvDgOEIzJSw==}
|
resolution: {integrity: sha512-AXb98u3M6BEzTnreBnGtQaF7xFTiMA92Dsy5tqEjpacbjRxDSFdN4bKJo9uvU4cEEOS7D2B9MT7kvDgOEIzJSw==}
|
||||||
|
|
||||||
|
'@lobehub/icons-static-svg@1.73.0':
|
||||||
|
resolution: {integrity: sha512-ydKUCDoopdmulbjDZo/gppaODd5Ju5nPneVcN9A5dAz9IJZUMkLms8bqostMLrqcdMQ8resKjLuV9RhJaWhaag==}
|
||||||
|
|
||||||
'@marijn/find-cluster-break@1.0.2':
|
'@marijn/find-cluster-break@1.0.2':
|
||||||
resolution: {integrity: sha512-l0h88YhZFyKdXIFNfSWpyjStDjGHwZ/U7iobcK1cQQD8sejsONdQtTVU+1wVN1PBw40PiiHB1vA5S7VTfQiP9g==}
|
resolution: {integrity: sha512-l0h88YhZFyKdXIFNfSWpyjStDjGHwZ/U7iobcK1cQQD8sejsONdQtTVU+1wVN1PBw40PiiHB1vA5S7VTfQiP9g==}
|
||||||
|
|
||||||
@@ -2839,6 +2845,8 @@ snapshots:
|
|||||||
'@lezer/common': 1.2.3
|
'@lezer/common': 1.2.3
|
||||||
'@lezer/highlight': 1.2.1
|
'@lezer/highlight': 1.2.1
|
||||||
|
|
||||||
|
'@lobehub/icons-static-svg@1.73.0': {}
|
||||||
|
|
||||||
'@marijn/find-cluster-break@1.0.2': {}
|
'@marijn/find-cluster-break@1.0.2': {}
|
||||||
|
|
||||||
'@mswjs/interceptors@0.40.0':
|
'@mswjs/interceptors@0.40.0':
|
||||||
|
|||||||
208
scripts/extract-icons.js
Normal file
208
scripts/extract-icons.js
Normal file
@@ -0,0 +1,208 @@
|
|||||||
|
const fs = require('fs');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
// 要提取的图标列表(按分类组织)
|
||||||
|
const ICONS_TO_EXTRACT = {
|
||||||
|
// AI 服务商(必需)
|
||||||
|
aiProviders: [
|
||||||
|
'openai', 'anthropic', 'claude', 'google', 'gemini',
|
||||||
|
'deepseek', 'kimi', 'moonshot', 'zhipu', 'minimax',
|
||||||
|
'baidu', 'alibaba', 'tencent', 'meta', 'microsoft',
|
||||||
|
'cohere', 'perplexity', 'mistral', 'huggingface'
|
||||||
|
],
|
||||||
|
|
||||||
|
// 云平台
|
||||||
|
cloudPlatforms: [
|
||||||
|
'aws', 'azure', 'huawei', 'cloudflare'
|
||||||
|
],
|
||||||
|
|
||||||
|
// 开发工具
|
||||||
|
devTools: [
|
||||||
|
'github', 'gitlab', 'docker', 'kubernetes', 'vscode'
|
||||||
|
],
|
||||||
|
|
||||||
|
// 其他
|
||||||
|
others: [
|
||||||
|
'settings', 'folder', 'file', 'link'
|
||||||
|
]
|
||||||
|
};
|
||||||
|
|
||||||
|
// 合并所有图标
|
||||||
|
const ALL_ICONS = [
|
||||||
|
...ICONS_TO_EXTRACT.aiProviders,
|
||||||
|
...ICONS_TO_EXTRACT.cloudPlatforms,
|
||||||
|
...ICONS_TO_EXTRACT.devTools,
|
||||||
|
...ICONS_TO_EXTRACT.others
|
||||||
|
];
|
||||||
|
|
||||||
|
// 提取逻辑
|
||||||
|
const OUTPUT_DIR = path.join(__dirname, '../src/icons/extracted');
|
||||||
|
const SOURCE_DIR = path.join(__dirname, '../node_modules/@lobehub/icons-static-svg/icons');
|
||||||
|
|
||||||
|
// 确保输出目录存在
|
||||||
|
if (!fs.existsSync(OUTPUT_DIR)) {
|
||||||
|
fs.mkdirSync(OUTPUT_DIR, { recursive: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('🎨 CC-Switch Icon Extractor\n');
|
||||||
|
console.log('========================================');
|
||||||
|
console.log('📦 Extracting icons...\n');
|
||||||
|
|
||||||
|
let extracted = 0;
|
||||||
|
let notFound = [];
|
||||||
|
|
||||||
|
// 提取图标
|
||||||
|
ALL_ICONS.forEach(iconName => {
|
||||||
|
const sourceFile = path.join(SOURCE_DIR, `${iconName}.svg`);
|
||||||
|
const targetFile = path.join(OUTPUT_DIR, `${iconName}.svg`);
|
||||||
|
|
||||||
|
if (fs.existsSync(sourceFile)) {
|
||||||
|
fs.copyFileSync(sourceFile, targetFile);
|
||||||
|
console.log(` ✓ ${iconName}.svg`);
|
||||||
|
extracted++;
|
||||||
|
} else {
|
||||||
|
console.log(` ✗ ${iconName}.svg (not found)`);
|
||||||
|
notFound.push(iconName);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 生成索引文件
|
||||||
|
console.log('\n📝 Generating index file...\n');
|
||||||
|
|
||||||
|
const indexContent = `// Auto-generated icon index
|
||||||
|
// Do not edit manually
|
||||||
|
|
||||||
|
export const icons: Record<string, string> = {
|
||||||
|
${ALL_ICONS.filter(name => !notFound.includes(name))
|
||||||
|
.map(name => {
|
||||||
|
const svg = fs.readFileSync(path.join(OUTPUT_DIR, `${name}.svg`), 'utf-8');
|
||||||
|
const escaped = svg.replace(/`/g, '\\`').replace(/\$/g, '\\$');
|
||||||
|
return ` '${name}': \`${escaped}\`,`;
|
||||||
|
})
|
||||||
|
.join('\n')}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const iconList = Object.keys(icons);
|
||||||
|
|
||||||
|
export function getIcon(name: string): string {
|
||||||
|
return icons[name.toLowerCase()] || '';
|
||||||
|
}
|
||||||
|
|
||||||
|
export function hasIcon(name: string): boolean {
|
||||||
|
return name.toLowerCase() in icons;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
fs.writeFileSync(path.join(OUTPUT_DIR, 'index.ts'), indexContent);
|
||||||
|
console.log('✓ Generated: src/icons/extracted/index.ts');
|
||||||
|
|
||||||
|
// 生成图标元数据
|
||||||
|
const metadataContent = `// Icon metadata for search and categorization
|
||||||
|
import { IconMetadata } from '@/types/icon';
|
||||||
|
|
||||||
|
export const iconMetadata: Record<string, IconMetadata> = {
|
||||||
|
// AI Providers
|
||||||
|
openai: { name: 'openai', displayName: 'OpenAI', category: 'ai-provider', keywords: ['gpt', 'chatgpt'], defaultColor: '#00A67E' },
|
||||||
|
anthropic: { name: 'anthropic', displayName: 'Anthropic', category: 'ai-provider', keywords: ['claude'], defaultColor: '#D4915D' },
|
||||||
|
claude: { name: 'claude', displayName: 'Claude', category: 'ai-provider', keywords: ['anthropic'], defaultColor: '#D4915D' },
|
||||||
|
google: { name: 'google', displayName: 'Google', category: 'ai-provider', keywords: ['gemini', 'bard'], defaultColor: '#4285F4' },
|
||||||
|
gemini: { name: 'gemini', displayName: 'Gemini', category: 'ai-provider', keywords: ['google'], defaultColor: '#4285F4' },
|
||||||
|
deepseek: { name: 'deepseek', displayName: 'DeepSeek', category: 'ai-provider', keywords: ['deep', 'seek'], defaultColor: '#1E88E5' },
|
||||||
|
moonshot: { name: 'moonshot', displayName: 'Moonshot', category: 'ai-provider', keywords: ['kimi', 'moonshot'], defaultColor: '#6366F1' },
|
||||||
|
kimi: { name: 'kimi', displayName: 'Kimi', category: 'ai-provider', keywords: ['moonshot'], defaultColor: '#6366F1' },
|
||||||
|
zhipu: { name: 'zhipu', displayName: 'Zhipu AI', category: 'ai-provider', keywords: ['chatglm', 'glm'], defaultColor: '#0F62FE' },
|
||||||
|
minimax: { name: 'minimax', displayName: 'MiniMax', category: 'ai-provider', keywords: ['minimax'], defaultColor: '#FF6B6B' },
|
||||||
|
baidu: { name: 'baidu', displayName: 'Baidu', category: 'ai-provider', keywords: ['ernie', 'wenxin'], defaultColor: '#2932E1' },
|
||||||
|
alibaba: { name: 'alibaba', displayName: 'Alibaba', category: 'ai-provider', keywords: ['qwen', 'tongyi'], defaultColor: '#FF6A00' },
|
||||||
|
tencent: { name: 'tencent', displayName: 'Tencent', category: 'ai-provider', keywords: ['hunyuan'], defaultColor: '#00A4FF' },
|
||||||
|
meta: { name: 'meta', displayName: 'Meta', category: 'ai-provider', keywords: ['facebook', 'llama'], defaultColor: '#0081FB' },
|
||||||
|
microsoft: { name: 'microsoft', displayName: 'Microsoft', category: 'ai-provider', keywords: ['copilot', 'azure'], defaultColor: '#00A4EF' },
|
||||||
|
cohere: { name: 'cohere', displayName: 'Cohere', category: 'ai-provider', keywords: ['cohere'], defaultColor: '#39594D' },
|
||||||
|
perplexity: { name: 'perplexity', displayName: 'Perplexity', category: 'ai-provider', keywords: ['perplexity'], defaultColor: '#20808D' },
|
||||||
|
mistral: { name: 'mistral', displayName: 'Mistral', category: 'ai-provider', keywords: ['mistral'], defaultColor: '#FF7000' },
|
||||||
|
huggingface: { name: 'huggingface', displayName: 'Hugging Face', category: 'ai-provider', keywords: ['huggingface', 'hf'], defaultColor: '#FFD21E' },
|
||||||
|
|
||||||
|
// Cloud Platforms
|
||||||
|
aws: { name: 'aws', displayName: 'AWS', category: 'cloud', keywords: ['amazon', 'cloud'], defaultColor: '#FF9900' },
|
||||||
|
azure: { name: 'azure', displayName: 'Azure', category: 'cloud', keywords: ['microsoft', 'cloud'], defaultColor: '#0078D4' },
|
||||||
|
huawei: { name: 'huawei', displayName: 'Huawei', category: 'cloud', keywords: ['huawei', 'cloud'], defaultColor: '#FF0000' },
|
||||||
|
cloudflare: { name: 'cloudflare', displayName: 'Cloudflare', category: 'cloud', keywords: ['cloudflare', 'cdn'], defaultColor: '#F38020' },
|
||||||
|
|
||||||
|
// Dev Tools
|
||||||
|
github: { name: 'github', displayName: 'GitHub', category: 'tool', keywords: ['git', 'version control'], defaultColor: '#181717' },
|
||||||
|
gitlab: { name: 'gitlab', displayName: 'GitLab', category: 'tool', keywords: ['git', 'version control'], defaultColor: '#FC6D26' },
|
||||||
|
docker: { name: 'docker', displayName: 'Docker', category: 'tool', keywords: ['container'], defaultColor: '#2496ED' },
|
||||||
|
kubernetes: { name: 'kubernetes', displayName: 'Kubernetes', category: 'tool', keywords: ['k8s', 'container'], defaultColor: '#326CE5' },
|
||||||
|
vscode: { name: 'vscode', displayName: 'VS Code', category: 'tool', keywords: ['editor', 'ide'], defaultColor: '#007ACC' },
|
||||||
|
|
||||||
|
// Others
|
||||||
|
settings: { name: 'settings', displayName: 'Settings', category: 'other', keywords: ['config', 'preferences'], defaultColor: '#6B7280' },
|
||||||
|
folder: { name: 'folder', displayName: 'Folder', category: 'other', keywords: ['directory'], defaultColor: '#6B7280' },
|
||||||
|
file: { name: 'file', displayName: 'File', category: 'other', keywords: ['document'], defaultColor: '#6B7280' },
|
||||||
|
link: { name: 'link', displayName: 'Link', category: 'other', keywords: ['url', 'hyperlink'], defaultColor: '#6B7280' },
|
||||||
|
};
|
||||||
|
|
||||||
|
export function getIconMetadata(name: string): IconMetadata | undefined {
|
||||||
|
return iconMetadata[name.toLowerCase()];
|
||||||
|
}
|
||||||
|
|
||||||
|
export function searchIcons(query: string): string[] {
|
||||||
|
const lowerQuery = query.toLowerCase();
|
||||||
|
return Object.values(iconMetadata)
|
||||||
|
.filter(meta =>
|
||||||
|
meta.name.includes(lowerQuery) ||
|
||||||
|
meta.displayName.toLowerCase().includes(lowerQuery) ||
|
||||||
|
meta.keywords.some(k => k.includes(lowerQuery))
|
||||||
|
)
|
||||||
|
.map(meta => meta.name);
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
fs.writeFileSync(path.join(OUTPUT_DIR, 'metadata.ts'), metadataContent);
|
||||||
|
console.log('✓ Generated: src/icons/extracted/metadata.ts');
|
||||||
|
|
||||||
|
// 生成 README
|
||||||
|
const readmeContent = `# Extracted Icons
|
||||||
|
|
||||||
|
This directory contains extracted icons from @lobehub/icons-static-svg.
|
||||||
|
|
||||||
|
## Statistics
|
||||||
|
- Total extracted: ${extracted} icons
|
||||||
|
- Not found: ${notFound.length} icons
|
||||||
|
|
||||||
|
## Extracted Icons
|
||||||
|
${ALL_ICONS.filter(name => !notFound.includes(name)).map(name => `- ${name}`).join('\n')}
|
||||||
|
|
||||||
|
${notFound.length > 0 ? `\n## Not Found\n${notFound.map(name => `- ${name}`).join('\n')}` : ''}
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
\`\`\`typescript
|
||||||
|
import { getIcon, hasIcon, iconList } from './extracted';
|
||||||
|
|
||||||
|
// Get icon SVG
|
||||||
|
const svg = getIcon('openai');
|
||||||
|
|
||||||
|
// Check if icon exists
|
||||||
|
if (hasIcon('openai')) {
|
||||||
|
// ...
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get all available icons
|
||||||
|
console.log(iconList);
|
||||||
|
\`\`\`
|
||||||
|
|
||||||
|
---
|
||||||
|
Last updated: ${new Date().toISOString()}
|
||||||
|
Generated by: scripts/extract-icons.js
|
||||||
|
`;
|
||||||
|
|
||||||
|
fs.writeFileSync(path.join(OUTPUT_DIR, 'README.md'), readmeContent);
|
||||||
|
console.log('✓ Generated: src/icons/extracted/README.md');
|
||||||
|
|
||||||
|
console.log('\n========================================');
|
||||||
|
console.log('✅ Extraction complete!\n');
|
||||||
|
console.log(` ✓ Extracted: ${extracted} icons`);
|
||||||
|
console.log(` ✗ Not found: ${notFound.length} icons`);
|
||||||
|
console.log(` 📉 Bundle size reduction: ~${Math.round((1 - extracted / 723) * 100)}%`);
|
||||||
|
console.log('========================================\n');
|
||||||
95
scripts/filter-icons.js
Normal file
95
scripts/filter-icons.js
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
const fs = require('fs');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
const ICONS_DIR = path.join(__dirname, '../src/icons/extracted');
|
||||||
|
|
||||||
|
// List of "Famous" icons to keep
|
||||||
|
// Based on common AI providers and tools
|
||||||
|
const KEEP_LIST = [
|
||||||
|
// AI Providers
|
||||||
|
'openai', 'anthropic', 'claude', 'google', 'gemini', 'gemma', 'palm',
|
||||||
|
'microsoft', 'azure', 'copilot', 'meta', 'llama',
|
||||||
|
'alibaba', 'qwen', 'tencent', 'hunyuan', 'baidu', 'wenxin',
|
||||||
|
'bytedance', 'doubao', 'deepseek', 'moonshot', 'kimi',
|
||||||
|
'zhipu', 'chatglm', 'glm', 'minimax', 'mistral', 'cohere',
|
||||||
|
'perplexity', 'huggingface', 'midjourney', 'stability',
|
||||||
|
'xai', 'grok', 'yi', 'zeroone', 'ollama',
|
||||||
|
|
||||||
|
// Cloud/Tools
|
||||||
|
'aws', 'googlecloud', 'huawei', 'cloudflare',
|
||||||
|
'github', 'githubcopilot', 'vercel', 'notion', 'discord',
|
||||||
|
'gitlab', 'docker', 'kubernetes', 'vscode', 'settings', 'folder', 'file', 'link'
|
||||||
|
];
|
||||||
|
|
||||||
|
// Get all SVG files
|
||||||
|
const files = fs.readdirSync(ICONS_DIR).filter(file => file.endsWith('.svg'));
|
||||||
|
|
||||||
|
console.log(`Scanning ${files.length} files...`);
|
||||||
|
|
||||||
|
let keptCount = 0;
|
||||||
|
let deletedCount = 0;
|
||||||
|
let renamedCount = 0;
|
||||||
|
|
||||||
|
// First pass: Identify files to keep and prefer color versions
|
||||||
|
const fileMap = {}; // name -> { hasColor: bool, hasMono: bool }
|
||||||
|
|
||||||
|
files.forEach(file => {
|
||||||
|
const isColor = file.endsWith('-color.svg');
|
||||||
|
const baseName = isColor ? file.replace('-color.svg', '') : file.replace('.svg', '');
|
||||||
|
|
||||||
|
if (!fileMap[baseName]) {
|
||||||
|
fileMap[baseName] = { hasColor: false, hasMono: false };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isColor) {
|
||||||
|
fileMap[baseName].hasColor = true;
|
||||||
|
} else {
|
||||||
|
fileMap[baseName].hasMono = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Second pass: Process files
|
||||||
|
Object.keys(fileMap).forEach(baseName => {
|
||||||
|
const info = fileMap[baseName];
|
||||||
|
const shouldKeep = KEEP_LIST.includes(baseName);
|
||||||
|
|
||||||
|
if (!shouldKeep) {
|
||||||
|
// Delete both versions if not in keep list
|
||||||
|
if (info.hasColor) {
|
||||||
|
fs.unlinkSync(path.join(ICONS_DIR, `${baseName}-color.svg`));
|
||||||
|
deletedCount++;
|
||||||
|
}
|
||||||
|
if (info.hasMono) {
|
||||||
|
fs.unlinkSync(path.join(ICONS_DIR, `${baseName}.svg`));
|
||||||
|
deletedCount++;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If keeping, prefer color
|
||||||
|
if (info.hasColor) {
|
||||||
|
// Rename color version to base version (overwrite mono if exists)
|
||||||
|
const colorPath = path.join(ICONS_DIR, `${baseName}-color.svg`);
|
||||||
|
const targetPath = path.join(ICONS_DIR, `${baseName}.svg`);
|
||||||
|
|
||||||
|
try {
|
||||||
|
// If mono exists, it will be overwritten/replaced
|
||||||
|
fs.renameSync(colorPath, targetPath);
|
||||||
|
renamedCount++;
|
||||||
|
keptCount++;
|
||||||
|
} catch (e) {
|
||||||
|
console.error(`Error renaming ${baseName}:`, e);
|
||||||
|
}
|
||||||
|
} else if (info.hasMono) {
|
||||||
|
// Keep mono if no color version
|
||||||
|
keptCount++;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
console.log(`\nCleanup complete:`);
|
||||||
|
console.log(`- Kept: ${keptCount}`);
|
||||||
|
console.log(`- Deleted: ${deletedCount}`);
|
||||||
|
console.log(`- Renamed (Color -> Standard): ${renamedCount}`);
|
||||||
|
|
||||||
|
// Regenerate index and metadata
|
||||||
|
require('./generate-icon-index.js');
|
||||||
113
scripts/generate-icon-index.js
Normal file
113
scripts/generate-icon-index.js
Normal file
@@ -0,0 +1,113 @@
|
|||||||
|
const fs = require('fs');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
const ICONS_DIR = path.join(__dirname, '../src/icons/extracted');
|
||||||
|
const INDEX_FILE = path.join(ICONS_DIR, 'index.ts');
|
||||||
|
const METADATA_FILE = path.join(ICONS_DIR, 'metadata.ts');
|
||||||
|
|
||||||
|
// Known metadata from previous configuration
|
||||||
|
const KNOWN_METADATA = {
|
||||||
|
openai: { name: 'openai', displayName: 'OpenAI', category: 'ai-provider', keywords: ['gpt', 'chatgpt'], defaultColor: '#00A67E' },
|
||||||
|
anthropic: { name: 'anthropic', displayName: 'Anthropic', category: 'ai-provider', keywords: ['claude'], defaultColor: '#D4915D' },
|
||||||
|
claude: { name: 'claude', displayName: 'Claude', category: 'ai-provider', keywords: ['anthropic'], defaultColor: '#D4915D' },
|
||||||
|
google: { name: 'google', displayName: 'Google', category: 'ai-provider', keywords: ['gemini', 'bard'], defaultColor: '#4285F4' },
|
||||||
|
gemini: { name: 'gemini', displayName: 'Gemini', category: 'ai-provider', keywords: ['google'], defaultColor: '#4285F4' },
|
||||||
|
deepseek: { name: 'deepseek', displayName: 'DeepSeek', category: 'ai-provider', keywords: ['deep', 'seek'], defaultColor: '#1E88E5' },
|
||||||
|
moonshot: { name: 'moonshot', displayName: 'Moonshot', category: 'ai-provider', keywords: ['kimi', 'moonshot'], defaultColor: '#6366F1' },
|
||||||
|
kimi: { name: 'kimi', displayName: 'Kimi', category: 'ai-provider', keywords: ['moonshot'], defaultColor: '#6366F1' },
|
||||||
|
zhipu: { name: 'zhipu', displayName: 'Zhipu AI', category: 'ai-provider', keywords: ['chatglm', 'glm'], defaultColor: '#0F62FE' },
|
||||||
|
minimax: { name: 'minimax', displayName: 'MiniMax', category: 'ai-provider', keywords: ['minimax'], defaultColor: '#FF6B6B' },
|
||||||
|
baidu: { name: 'baidu', displayName: 'Baidu', category: 'ai-provider', keywords: ['ernie', 'wenxin'], defaultColor: '#2932E1' },
|
||||||
|
alibaba: { name: 'alibaba', displayName: 'Alibaba', category: 'ai-provider', keywords: ['qwen', 'tongyi'], defaultColor: '#FF6A00' },
|
||||||
|
tencent: { name: 'tencent', displayName: 'Tencent', category: 'ai-provider', keywords: ['hunyuan'], defaultColor: '#00A4FF' },
|
||||||
|
meta: { name: 'meta', displayName: 'Meta', category: 'ai-provider', keywords: ['facebook', 'llama'], defaultColor: '#0081FB' },
|
||||||
|
microsoft: { name: 'microsoft', displayName: 'Microsoft', category: 'ai-provider', keywords: ['copilot', 'azure'], defaultColor: '#00A4EF' },
|
||||||
|
cohere: { name: 'cohere', displayName: 'Cohere', category: 'ai-provider', keywords: ['cohere'], defaultColor: '#39594D' },
|
||||||
|
perplexity: { name: 'perplexity', displayName: 'Perplexity', category: 'ai-provider', keywords: ['perplexity'], defaultColor: '#20808D' },
|
||||||
|
mistral: { name: 'mistral', displayName: 'Mistral', category: 'ai-provider', keywords: ['mistral'], defaultColor: '#FF7000' },
|
||||||
|
huggingface: { name: 'huggingface', displayName: 'Hugging Face', category: 'ai-provider', keywords: ['huggingface', 'hf'], defaultColor: '#FFD21E' },
|
||||||
|
aws: { name: 'aws', displayName: 'AWS', category: 'cloud', keywords: ['amazon', 'cloud'], defaultColor: '#FF9900' },
|
||||||
|
azure: { name: 'azure', displayName: 'Azure', category: 'cloud', keywords: ['microsoft', 'cloud'], defaultColor: '#0078D4' },
|
||||||
|
huawei: { name: 'huawei', displayName: 'Huawei', category: 'cloud', keywords: ['huawei', 'cloud'], defaultColor: '#FF0000' },
|
||||||
|
cloudflare: { name: 'cloudflare', displayName: 'Cloudflare', category: 'cloud', keywords: ['cloudflare', 'cdn'], defaultColor: '#F38020' },
|
||||||
|
github: { name: 'github', displayName: 'GitHub', category: 'tool', keywords: ['git', 'version control'], defaultColor: '#181717' },
|
||||||
|
gitlab: { name: 'gitlab', displayName: 'GitLab', category: 'tool', keywords: ['git', 'version control'], defaultColor: '#FC6D26' },
|
||||||
|
docker: { name: 'docker', displayName: 'Docker', category: 'tool', keywords: ['container'], defaultColor: '#2496ED' },
|
||||||
|
kubernetes: { name: 'kubernetes', displayName: 'Kubernetes', category: 'tool', keywords: ['k8s', 'container'], defaultColor: '#326CE5' },
|
||||||
|
vscode: { name: 'vscode', displayName: 'VS Code', category: 'tool', keywords: ['editor', 'ide'], defaultColor: '#007ACC' },
|
||||||
|
settings: { name: 'settings', displayName: 'Settings', category: 'other', keywords: ['config', 'preferences'], defaultColor: '#6B7280' },
|
||||||
|
folder: { name: 'folder', displayName: 'Folder', category: 'other', keywords: ['directory'], defaultColor: '#6B7280' },
|
||||||
|
file: { name: 'file', displayName: 'File', category: 'other', keywords: ['document'], defaultColor: '#6B7280' },
|
||||||
|
link: { name: 'link', displayName: 'Link', category: 'other', keywords: ['url', 'hyperlink'], defaultColor: '#6B7280' },
|
||||||
|
};
|
||||||
|
|
||||||
|
// Get all SVG files
|
||||||
|
const files = fs.readdirSync(ICONS_DIR).filter(file => file.endsWith('.svg'));
|
||||||
|
|
||||||
|
console.log(`Found ${files.length} SVG files.`);
|
||||||
|
|
||||||
|
// Generate index.ts
|
||||||
|
const indexContent = `// Auto-generated icon index
|
||||||
|
// Do not edit manually
|
||||||
|
|
||||||
|
export const icons: Record<string, string> = {
|
||||||
|
${files.map(file => {
|
||||||
|
const name = path.basename(file, '.svg');
|
||||||
|
const svg = fs.readFileSync(path.join(ICONS_DIR, file), 'utf-8');
|
||||||
|
const escaped = svg.replace(/`/g, '\\`').replace(/\$/g, '\\$');
|
||||||
|
return ` '${name}': \`${escaped}\`,`;
|
||||||
|
}).join('\n')}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const iconList = Object.keys(icons);
|
||||||
|
|
||||||
|
export function getIcon(name: string): string {
|
||||||
|
return icons[name.toLowerCase()] || '';
|
||||||
|
}
|
||||||
|
|
||||||
|
export function hasIcon(name: string): boolean {
|
||||||
|
return name.toLowerCase() in icons;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
fs.writeFileSync(INDEX_FILE, indexContent);
|
||||||
|
console.log(`Generated ${INDEX_FILE}`);
|
||||||
|
|
||||||
|
// Generate metadata.ts
|
||||||
|
const metadataEntries = files.map(file => {
|
||||||
|
const name = path.basename(file, '.svg').toLowerCase();
|
||||||
|
const known = KNOWN_METADATA[name];
|
||||||
|
|
||||||
|
if (known) {
|
||||||
|
return ` ${name}: ${JSON.stringify(known)},`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Default metadata for unknown icons
|
||||||
|
return ` '${name}': { name: '${name}', displayName: '${name}', category: 'other', keywords: [], defaultColor: 'currentColor' },`;
|
||||||
|
});
|
||||||
|
|
||||||
|
const metadataContent = `// Icon metadata for search and categorization
|
||||||
|
import { IconMetadata } from '@/types/icon';
|
||||||
|
|
||||||
|
export const iconMetadata: Record<string, IconMetadata> = {
|
||||||
|
${metadataEntries.join('\n')}
|
||||||
|
};
|
||||||
|
|
||||||
|
export function getIconMetadata(name: string): IconMetadata | undefined {
|
||||||
|
return iconMetadata[name.toLowerCase()];
|
||||||
|
}
|
||||||
|
|
||||||
|
export function searchIcons(query: string): string[] {
|
||||||
|
const lowerQuery = query.toLowerCase();
|
||||||
|
return Object.values(iconMetadata)
|
||||||
|
.filter(meta =>
|
||||||
|
meta.name.includes(lowerQuery) ||
|
||||||
|
meta.displayName.toLowerCase().includes(lowerQuery) ||
|
||||||
|
meta.keywords.some(k => k.includes(lowerQuery))
|
||||||
|
)
|
||||||
|
.map(meta => meta.name);
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
fs.writeFileSync(METADATA_FILE, metadataContent);
|
||||||
|
console.log(`Generated ${METADATA_FILE}`);
|
||||||
116
src-tauri/Cargo.lock
generated
116
src-tauri/Cargo.lock
generated
@@ -39,6 +39,18 @@ dependencies = [
|
|||||||
"version_check",
|
"version_check",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "ahash"
|
||||||
|
version = "0.8.12"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "5a15f179cd60c4584b8a8c596927aadc462e27f2ca70c04e0071964a73ba7a75"
|
||||||
|
dependencies = [
|
||||||
|
"cfg-if",
|
||||||
|
"once_cell",
|
||||||
|
"version_check",
|
||||||
|
"zerocopy",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "aho-corasick"
|
name = "aho-corasick"
|
||||||
version = "1.1.3"
|
version = "1.1.3"
|
||||||
@@ -291,6 +303,17 @@ version = "1.1.2"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "1505bd5d3d116872e7271a6d4e16d81d0c8570876c8de68093a09ac269d8aac0"
|
checksum = "1505bd5d3d116872e7271a6d4e16d81d0c8570876c8de68093a09ac269d8aac0"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "auto-launch"
|
||||||
|
version = "0.5.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "1f012b8cc0c850f34117ec8252a44418f2e34a2cf501de89e29b241ae5f79471"
|
||||||
|
dependencies = [
|
||||||
|
"dirs 4.0.0",
|
||||||
|
"thiserror 1.0.69",
|
||||||
|
"winreg 0.10.1",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "autocfg"
|
name = "autocfg"
|
||||||
version = "1.5.0"
|
version = "1.5.0"
|
||||||
@@ -595,18 +618,23 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "cc-switch"
|
name = "cc-switch"
|
||||||
version = "3.6.2"
|
version = "3.7.1"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
|
"auto-launch",
|
||||||
|
"base64 0.22.1",
|
||||||
"chrono",
|
"chrono",
|
||||||
"dirs 5.0.1",
|
"dirs 5.0.1",
|
||||||
"futures",
|
"futures",
|
||||||
|
"indexmap 2.11.4",
|
||||||
"log",
|
"log",
|
||||||
"objc2 0.5.2",
|
"objc2 0.5.2",
|
||||||
"objc2-app-kit 0.2.2",
|
"objc2-app-kit 0.2.2",
|
||||||
|
"once_cell",
|
||||||
"regex",
|
"regex",
|
||||||
"reqwest",
|
"reqwest",
|
||||||
"rquickjs",
|
"rquickjs",
|
||||||
|
"rusqlite",
|
||||||
"serde",
|
"serde",
|
||||||
"serde_json",
|
"serde_json",
|
||||||
"serde_yaml",
|
"serde_yaml",
|
||||||
@@ -982,6 +1010,15 @@ dependencies = [
|
|||||||
"subtle",
|
"subtle",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "dirs"
|
||||||
|
version = "4.0.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "ca3aa72a6f96ea37bbc5aa912f6788242832f75369bdfdadcb0e38423f100059"
|
||||||
|
dependencies = [
|
||||||
|
"dirs-sys 0.3.7",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "dirs"
|
name = "dirs"
|
||||||
version = "5.0.1"
|
version = "5.0.1"
|
||||||
@@ -1000,6 +1037,17 @@ dependencies = [
|
|||||||
"dirs-sys 0.5.0",
|
"dirs-sys 0.5.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "dirs-sys"
|
||||||
|
version = "0.3.7"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "1b1d1d91c932ef41c0f2663aa8b0ca0342d444d842c06914aa0a7e352d0bada6"
|
||||||
|
dependencies = [
|
||||||
|
"libc",
|
||||||
|
"redox_users 0.4.6",
|
||||||
|
"winapi",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "dirs-sys"
|
name = "dirs-sys"
|
||||||
version = "0.4.1"
|
version = "0.4.1"
|
||||||
@@ -1241,6 +1289,18 @@ dependencies = [
|
|||||||
"pin-project-lite",
|
"pin-project-lite",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "fallible-iterator"
|
||||||
|
version = "0.3.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "2acce4a10f12dc2fb14a218589d4f1f62ef011b2d0cc4b3cb1bba8e94da14649"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "fallible-streaming-iterator"
|
||||||
|
version = "0.1.9"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "7360491ce676a36bf9bb3c56c1aa791658183a54d2744120f27285738d90465a"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "fastrand"
|
name = "fastrand"
|
||||||
version = "2.3.0"
|
version = "2.3.0"
|
||||||
@@ -1779,7 +1839,7 @@ version = "0.12.3"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888"
|
checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"ahash",
|
"ahash 0.7.8",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -1787,6 +1847,9 @@ name = "hashbrown"
|
|||||||
version = "0.14.5"
|
version = "0.14.5"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1"
|
checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1"
|
||||||
|
dependencies = [
|
||||||
|
"ahash 0.8.12",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "hashbrown"
|
name = "hashbrown"
|
||||||
@@ -1794,6 +1857,15 @@ version = "0.16.0"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "5419bdc4f6a9207fbeba6d11b604d481addf78ecd10c11ad51e76c2f6482748d"
|
checksum = "5419bdc4f6a9207fbeba6d11b604d481addf78ecd10c11ad51e76c2f6482748d"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "hashlink"
|
||||||
|
version = "0.9.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "6ba4ff7128dee98c7dc9794b6a411377e1404dba1c97deb8d1a55297bd25d8af"
|
||||||
|
dependencies = [
|
||||||
|
"hashbrown 0.14.5",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "heck"
|
name = "heck"
|
||||||
version = "0.4.1"
|
version = "0.4.1"
|
||||||
@@ -2364,6 +2436,17 @@ dependencies = [
|
|||||||
"redox_syscall",
|
"redox_syscall",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "libsqlite3-sys"
|
||||||
|
version = "0.28.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "0c10584274047cb335c23d3e61bcef8e323adae7c5c8c760540f73610177fc3f"
|
||||||
|
dependencies = [
|
||||||
|
"cc",
|
||||||
|
"pkg-config",
|
||||||
|
"vcpkg",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "linux-raw-sys"
|
name = "linux-raw-sys"
|
||||||
version = "0.11.0"
|
version = "0.11.0"
|
||||||
@@ -3815,6 +3898,20 @@ dependencies = [
|
|||||||
"cc",
|
"cc",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rusqlite"
|
||||||
|
version = "0.31.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "b838eba278d213a8beaf485bd313fd580ca4505a00d5871caeb1457c55322cae"
|
||||||
|
dependencies = [
|
||||||
|
"bitflags 2.9.4",
|
||||||
|
"fallible-iterator",
|
||||||
|
"fallible-streaming-iterator",
|
||||||
|
"hashlink",
|
||||||
|
"libsqlite3-sys",
|
||||||
|
"smallvec",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rust-ini"
|
name = "rust-ini"
|
||||||
version = "0.21.3"
|
version = "0.21.3"
|
||||||
@@ -5533,6 +5630,12 @@ version = "1.11.1"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "943ce29a8a743eb10d6082545d861b24f9d1b160b7d741e0f2cdf726bec909c5"
|
checksum = "943ce29a8a743eb10d6082545d861b24f9d1b160b7d741e0f2cdf726bec909c5"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "vcpkg"
|
||||||
|
version = "0.2.15"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "version-compare"
|
name = "version-compare"
|
||||||
version = "0.2.0"
|
version = "0.2.0"
|
||||||
@@ -6397,6 +6500,15 @@ dependencies = [
|
|||||||
"memchr",
|
"memchr",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "winreg"
|
||||||
|
version = "0.10.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "80d0f4e272c85def139476380b12f9ac60926689dd2e01d4923222f40580869d"
|
||||||
|
dependencies = [
|
||||||
|
"winapi",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "winreg"
|
name = "winreg"
|
||||||
version = "0.52.0"
|
version = "0.52.0"
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "cc-switch"
|
name = "cc-switch"
|
||||||
version = "3.6.2"
|
version = "3.7.1"
|
||||||
description = "Claude Code & Codex 供应商配置管理工具"
|
description = "All-in-One Assistant for Claude Code, Codex & Gemini CLI"
|
||||||
authors = ["Jason Young"]
|
authors = ["Jason Young"]
|
||||||
license = "MIT"
|
license = "MIT"
|
||||||
repository = "https://github.com/farion1231/cc-switch"
|
repository = "https://github.com/farion1231/cc-switch"
|
||||||
@@ -48,6 +48,11 @@ zip = "2.2"
|
|||||||
serde_yaml = "0.9"
|
serde_yaml = "0.9"
|
||||||
tempfile = "3"
|
tempfile = "3"
|
||||||
url = "2.5"
|
url = "2.5"
|
||||||
|
auto-launch = "0.5"
|
||||||
|
once_cell = "1.21.3"
|
||||||
|
base64 = "0.22"
|
||||||
|
rusqlite = { version = "0.31", features = ["bundled"] }
|
||||||
|
indexmap = { version = "2", features = ["serde"] }
|
||||||
|
|
||||||
[target.'cfg(any(target_os = "macos", target_os = "windows", target_os = "linux"))'.dependencies]
|
[target.'cfg(any(target_os = "macos", target_os = "windows", target_os = "linux"))'.dependencies]
|
||||||
tauri-plugin-single-instance = "2"
|
tauri-plugin-single-instance = "2"
|
||||||
|
|||||||
@@ -10,7 +10,8 @@
|
|||||||
"opener:default",
|
"opener:default",
|
||||||
"updater:default",
|
"updater:default",
|
||||||
"core:window:allow-set-skip-taskbar",
|
"core:window:allow-set-skip-taskbar",
|
||||||
|
"core:window:allow-start-dragging",
|
||||||
"process:allow-restart",
|
"process:allow-restart",
|
||||||
"dialog:default"
|
"dialog:default"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
40
src-tauri/src/auto_launch.rs
Normal file
40
src-tauri/src/auto_launch.rs
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
use crate::error::AppError;
|
||||||
|
use auto_launch::AutoLaunch;
|
||||||
|
|
||||||
|
/// 初始化 AutoLaunch 实例
|
||||||
|
fn get_auto_launch() -> Result<AutoLaunch, AppError> {
|
||||||
|
let app_name = "CC Switch";
|
||||||
|
let app_path =
|
||||||
|
std::env::current_exe().map_err(|e| AppError::Message(format!("无法获取应用路径: {e}")))?;
|
||||||
|
|
||||||
|
let auto_launch = AutoLaunch::new(app_name, &app_path.to_string_lossy(), false, &[] as &[&str]);
|
||||||
|
Ok(auto_launch)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 启用开机自启
|
||||||
|
pub fn enable_auto_launch() -> Result<(), AppError> {
|
||||||
|
let auto_launch = get_auto_launch()?;
|
||||||
|
auto_launch
|
||||||
|
.enable()
|
||||||
|
.map_err(|e| AppError::Message(format!("启用开机自启失败: {e}")))?;
|
||||||
|
log::info!("已启用开机自启");
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 禁用开机自启
|
||||||
|
pub fn disable_auto_launch() -> Result<(), AppError> {
|
||||||
|
let auto_launch = get_auto_launch()?;
|
||||||
|
auto_launch
|
||||||
|
.disable()
|
||||||
|
.map_err(|e| AppError::Message(format!("禁用开机自启失败: {e}")))?;
|
||||||
|
log::info!("已禁用开机自启");
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 检查是否已启用开机自启
|
||||||
|
pub fn is_auto_launch_enabled() -> Result<bool, AppError> {
|
||||||
|
let auto_launch = get_auto_launch()?;
|
||||||
|
auto_launch
|
||||||
|
.is_enabled()
|
||||||
|
.map_err(|e| AppError::Message(format!("检查开机自启状态失败: {e}")))
|
||||||
|
}
|
||||||
@@ -29,6 +29,7 @@ pub fn get_codex_config_path() -> PathBuf {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// 获取 Codex 供应商配置文件路径
|
/// 获取 Codex 供应商配置文件路径
|
||||||
|
#[allow(dead_code)]
|
||||||
pub fn get_codex_provider_paths(
|
pub fn get_codex_provider_paths(
|
||||||
provider_id: &str,
|
provider_id: &str,
|
||||||
provider_name: Option<&str>,
|
provider_name: Option<&str>,
|
||||||
@@ -44,6 +45,7 @@ pub fn get_codex_provider_paths(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// 删除 Codex 供应商配置文件
|
/// 删除 Codex 供应商配置文件
|
||||||
|
#[allow(dead_code)]
|
||||||
pub fn delete_codex_provider_config(
|
pub fn delete_codex_provider_config(
|
||||||
provider_id: &str,
|
provider_id: &str,
|
||||||
provider_name: &str,
|
provider_name: &str,
|
||||||
|
|||||||
@@ -141,11 +141,10 @@ pub async fn open_app_config_folder(handle: AppHandle) -> Result<bool, String> {
|
|||||||
pub async fn get_claude_common_config_snippet(
|
pub async fn get_claude_common_config_snippet(
|
||||||
state: tauri::State<'_, crate::store::AppState>,
|
state: tauri::State<'_, crate::store::AppState>,
|
||||||
) -> Result<Option<String>, String> {
|
) -> Result<Option<String>, String> {
|
||||||
let guard = state
|
state
|
||||||
.config
|
.db
|
||||||
.read()
|
.get_config_snippet("claude")
|
||||||
.map_err(|e| format!("读取配置锁失败: {e}"))?;
|
.map_err(|e| e.to_string())
|
||||||
Ok(guard.common_config_snippets.claude.clone())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 设置 Claude 通用配置片段(已废弃,使用 set_common_config_snippet)
|
/// 设置 Claude 通用配置片段(已废弃,使用 set_common_config_snippet)
|
||||||
@@ -154,24 +153,22 @@ pub async fn set_claude_common_config_snippet(
|
|||||||
snippet: String,
|
snippet: String,
|
||||||
state: tauri::State<'_, crate::store::AppState>,
|
state: tauri::State<'_, crate::store::AppState>,
|
||||||
) -> Result<(), String> {
|
) -> Result<(), String> {
|
||||||
let mut guard = state
|
|
||||||
.config
|
|
||||||
.write()
|
|
||||||
.map_err(|e| format!("写入配置锁失败: {e}"))?;
|
|
||||||
|
|
||||||
// 验证是否为有效的 JSON(如果不为空)
|
// 验证是否为有效的 JSON(如果不为空)
|
||||||
if !snippet.trim().is_empty() {
|
if !snippet.trim().is_empty() {
|
||||||
serde_json::from_str::<serde_json::Value>(&snippet)
|
serde_json::from_str::<serde_json::Value>(&snippet)
|
||||||
.map_err(|e| format!("无效的 JSON 格式: {e}"))?;
|
.map_err(|e| format!("无效的 JSON 格式: {e}"))?;
|
||||||
}
|
}
|
||||||
|
|
||||||
guard.common_config_snippets.claude = if snippet.trim().is_empty() {
|
let value = if snippet.trim().is_empty() {
|
||||||
None
|
None
|
||||||
} else {
|
} else {
|
||||||
Some(snippet)
|
Some(snippet)
|
||||||
};
|
};
|
||||||
|
|
||||||
guard.save().map_err(|e| e.to_string())?;
|
state
|
||||||
|
.db
|
||||||
|
.set_config_snippet("claude", value)
|
||||||
|
.map_err(|e| e.to_string())?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -181,17 +178,10 @@ pub async fn get_common_config_snippet(
|
|||||||
app_type: String,
|
app_type: String,
|
||||||
state: tauri::State<'_, crate::store::AppState>,
|
state: tauri::State<'_, crate::store::AppState>,
|
||||||
) -> Result<Option<String>, String> {
|
) -> Result<Option<String>, String> {
|
||||||
use crate::app_config::AppType;
|
state
|
||||||
use std::str::FromStr;
|
.db
|
||||||
|
.get_config_snippet(&app_type)
|
||||||
let app = AppType::from_str(&app_type).map_err(|e| format!("无效的应用类型: {e}"))?;
|
.map_err(|e| e.to_string())
|
||||||
|
|
||||||
let guard = state
|
|
||||||
.config
|
|
||||||
.read()
|
|
||||||
.map_err(|e| format!("读取配置锁失败: {e}"))?;
|
|
||||||
|
|
||||||
Ok(guard.common_config_snippets.get(&app).cloned())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 设置通用配置片段(统一接口)
|
/// 设置通用配置片段(统一接口)
|
||||||
@@ -201,40 +191,31 @@ pub async fn set_common_config_snippet(
|
|||||||
snippet: String,
|
snippet: String,
|
||||||
state: tauri::State<'_, crate::store::AppState>,
|
state: tauri::State<'_, crate::store::AppState>,
|
||||||
) -> Result<(), String> {
|
) -> Result<(), String> {
|
||||||
use crate::app_config::AppType;
|
|
||||||
use std::str::FromStr;
|
|
||||||
|
|
||||||
let app = AppType::from_str(&app_type).map_err(|e| format!("无效的应用类型: {e}"))?;
|
|
||||||
|
|
||||||
let mut guard = state
|
|
||||||
.config
|
|
||||||
.write()
|
|
||||||
.map_err(|e| format!("写入配置锁失败: {e}"))?;
|
|
||||||
|
|
||||||
// 验证格式(根据应用类型)
|
// 验证格式(根据应用类型)
|
||||||
if !snippet.trim().is_empty() {
|
if !snippet.trim().is_empty() {
|
||||||
match app {
|
match app_type.as_str() {
|
||||||
AppType::Claude | AppType::Gemini => {
|
"claude" | "gemini" => {
|
||||||
// 验证 JSON 格式
|
// 验证 JSON 格式
|
||||||
serde_json::from_str::<serde_json::Value>(&snippet)
|
serde_json::from_str::<serde_json::Value>(&snippet)
|
||||||
.map_err(|e| format!("无效的 JSON 格式: {e}"))?;
|
.map_err(|e| format!("无效的 JSON 格式: {e}"))?;
|
||||||
}
|
}
|
||||||
AppType::Codex => {
|
"codex" => {
|
||||||
// TOML 格式暂不验证(或可使用 toml crate)
|
// TOML 格式暂不验证(或可使用 toml crate)
|
||||||
// 注意:TOML 验证较为复杂,暂时跳过
|
// 注意:TOML 验证较为复杂,暂时跳过
|
||||||
}
|
}
|
||||||
|
_ => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
guard.common_config_snippets.set(
|
let value = if snippet.trim().is_empty() {
|
||||||
&app,
|
None
|
||||||
if snippet.trim().is_empty() {
|
} else {
|
||||||
None
|
Some(snippet)
|
||||||
} else {
|
};
|
||||||
Some(snippet)
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
guard.save().map_err(|e| e.to_string())?;
|
state
|
||||||
|
.db
|
||||||
|
.set_config_snippet(&app_type, value)
|
||||||
|
.map_err(|e| e.to_string())?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,6 +9,16 @@ pub fn parse_deeplink(url: String) -> Result<DeepLinkImportRequest, String> {
|
|||||||
parse_deeplink_url(&url).map_err(|e| e.to_string())
|
parse_deeplink_url(&url).map_err(|e| e.to_string())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Merge configuration from Base64/URL into a deep link request
|
||||||
|
/// This is used by the frontend to show the complete configuration in the confirmation dialog
|
||||||
|
#[tauri::command]
|
||||||
|
pub fn merge_deeplink_config(
|
||||||
|
request: DeepLinkImportRequest,
|
||||||
|
) -> Result<DeepLinkImportRequest, String> {
|
||||||
|
log::info!("Merging config for deep link request: {}", request.name);
|
||||||
|
crate::deeplink::parse_and_merge_config(&request).map_err(|e| e.to_string())
|
||||||
|
}
|
||||||
|
|
||||||
/// Import a provider from a deep link request (after user confirmation)
|
/// Import a provider from a deep link request (after user confirmation)
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
pub fn import_from_deeplink(
|
pub fn import_from_deeplink(
|
||||||
|
|||||||
@@ -29,11 +29,17 @@ pub async fn export_config_to_file(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// 从文件导入配置
|
/// 从文件导入配置
|
||||||
|
/// TODO: 需要重构以使用数据库而不是 JSON 配置
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
pub async fn import_config_from_file(
|
pub async fn import_config_from_file(
|
||||||
#[allow(non_snake_case)] filePath: String,
|
#[allow(non_snake_case)] _filePath: String,
|
||||||
state: State<'_, AppState>,
|
_state: State<'_, AppState>,
|
||||||
) -> Result<Value, String> {
|
) -> Result<Value, String> {
|
||||||
|
// TODO: 实现基于数据库的导入逻辑
|
||||||
|
// 当前暂时禁用此功能
|
||||||
|
Err("配置导入功能正在重构中,暂时不可用".to_string())
|
||||||
|
|
||||||
|
/* 旧的实现,需要重构:
|
||||||
let (new_config, backup_id) = tauri::async_runtime::spawn_blocking(move || {
|
let (new_config, backup_id) = tauri::async_runtime::spawn_blocking(move || {
|
||||||
let path_buf = PathBuf::from(&filePath);
|
let path_buf = PathBuf::from(&filePath);
|
||||||
ConfigService::load_config_for_import(&path_buf)
|
ConfigService::load_config_for_import(&path_buf)
|
||||||
@@ -55,11 +61,18 @@ pub async fn import_config_from_file(
|
|||||||
"message": "Configuration imported successfully",
|
"message": "Configuration imported successfully",
|
||||||
"backupId": backup_id
|
"backupId": backup_id
|
||||||
}))
|
}))
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 同步当前供应商配置到对应的 live 文件
|
/// 同步当前供应商配置到对应的 live 文件
|
||||||
|
/// TODO: 需要重构以使用数据库而不是 JSON 配置
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
pub async fn sync_current_providers_live(state: State<'_, AppState>) -> Result<Value, String> {
|
pub async fn sync_current_providers_live(_state: State<'_, AppState>) -> Result<Value, String> {
|
||||||
|
// TODO: 实现基于数据库的同步逻辑
|
||||||
|
// 当前暂时禁用此功能
|
||||||
|
Err("配置同步功能正在重构中,暂时不可用".to_string())
|
||||||
|
|
||||||
|
/* 旧的实现,需要重构:
|
||||||
{
|
{
|
||||||
let mut config_state = state
|
let mut config_state = state
|
||||||
.config
|
.config
|
||||||
@@ -73,6 +86,7 @@ pub async fn sync_current_providers_live(state: State<'_, AppState>) -> Result<V
|
|||||||
"success": true,
|
"success": true,
|
||||||
"message": "Live configuration synchronized"
|
"message": "Live configuration synchronized"
|
||||||
}))
|
}))
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 保存文件对话框
|
/// 保存文件对话框
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
#![allow(non_snake_case)]
|
#![allow(non_snake_case)]
|
||||||
|
|
||||||
|
use indexmap::IndexMap;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
|
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
@@ -82,12 +83,8 @@ pub async fn upsert_mcp_server_in_config(
|
|||||||
|
|
||||||
// 读取现有的服务器(如果存在)
|
// 读取现有的服务器(如果存在)
|
||||||
let existing_server = {
|
let existing_server = {
|
||||||
let cfg = state.config.read().map_err(|e| e.to_string())?;
|
let servers = state.db.get_all_mcp_servers().map_err(|e| e.to_string())?;
|
||||||
if let Some(servers) = &cfg.mcp.servers {
|
servers.get(&id).cloned()
|
||||||
servers.get(&id).cloned()
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// 构建新的统一服务器结构
|
// 构建新的统一服务器结构
|
||||||
@@ -165,7 +162,7 @@ use crate::app_config::McpServer;
|
|||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
pub async fn get_mcp_servers(
|
pub async fn get_mcp_servers(
|
||||||
state: State<'_, AppState>,
|
state: State<'_, AppState>,
|
||||||
) -> Result<HashMap<String, McpServer>, String> {
|
) -> Result<IndexMap<String, McpServer>, String> {
|
||||||
McpService::get_all_servers(&state).map_err(|e| e.to_string())
|
McpService::get_all_servers(&state).map_err(|e| e.to_string())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
use std::collections::HashMap;
|
use indexmap::IndexMap;
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
use tauri::State;
|
use tauri::State;
|
||||||
@@ -12,7 +12,7 @@ use crate::store::AppState;
|
|||||||
pub async fn get_prompts(
|
pub async fn get_prompts(
|
||||||
app: String,
|
app: String,
|
||||||
state: State<'_, AppState>,
|
state: State<'_, AppState>,
|
||||||
) -> Result<HashMap<String, Prompt>, String> {
|
) -> Result<IndexMap<String, Prompt>, String> {
|
||||||
let app_type = AppType::from_str(&app).map_err(|e| e.to_string())?;
|
let app_type = AppType::from_str(&app).map_err(|e| e.to_string())?;
|
||||||
PromptService::get_prompts(&state, app_type).map_err(|e| e.to_string())
|
PromptService::get_prompts(&state, app_type).map_err(|e| e.to_string())
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
use std::collections::HashMap;
|
use indexmap::IndexMap;
|
||||||
use tauri::State;
|
use tauri::State;
|
||||||
|
|
||||||
use crate::app_config::AppType;
|
use crate::app_config::AppType;
|
||||||
@@ -13,7 +13,7 @@ use std::str::FromStr;
|
|||||||
pub fn get_providers(
|
pub fn get_providers(
|
||||||
state: State<'_, AppState>,
|
state: State<'_, AppState>,
|
||||||
app: String,
|
app: String,
|
||||||
) -> Result<HashMap<String, Provider>, String> {
|
) -> Result<IndexMap<String, Provider>, String> {
|
||||||
let app_type = AppType::from_str(&app).map_err(|e| e.to_string())?;
|
let app_type = AppType::from_str(&app).map_err(|e| e.to_string())?;
|
||||||
ProviderService::list(state.inner(), app_type).map_err(|e| e.to_string())
|
ProviderService::list(state.inner(), app_type).map_err(|e| e.to_string())
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,3 +37,20 @@ pub async fn set_app_config_dir_override(
|
|||||||
crate::app_store::set_app_config_dir_to_store(&app, path.as_deref())?;
|
crate::app_store::set_app_config_dir_to_store(&app, path.as_deref())?;
|
||||||
Ok(true)
|
Ok(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// 设置开机自启
|
||||||
|
#[tauri::command]
|
||||||
|
pub async fn set_auto_launch(enabled: bool) -> Result<bool, String> {
|
||||||
|
if enabled {
|
||||||
|
crate::auto_launch::enable_auto_launch().map_err(|e| format!("启用开机自启失败: {e}"))?;
|
||||||
|
} else {
|
||||||
|
crate::auto_launch::disable_auto_launch().map_err(|e| format!("禁用开机自启失败: {e}"))?;
|
||||||
|
}
|
||||||
|
Ok(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 获取开机自启状态
|
||||||
|
#[tauri::command]
|
||||||
|
pub async fn get_auto_launch_status() -> Result<bool, String> {
|
||||||
|
crate::auto_launch::is_auto_launch_enabled().map_err(|e| format!("获取开机自启状态失败: {e}"))
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
use crate::error::format_skill_error;
|
||||||
use crate::services::skill::SkillState;
|
use crate::services::skill::SkillState;
|
||||||
use crate::services::{Skill, SkillRepo, SkillService};
|
use crate::services::{Skill, SkillRepo, SkillService};
|
||||||
use crate::store::AppState;
|
use crate::store::AppState;
|
||||||
@@ -12,10 +13,7 @@ pub async fn get_skills(
|
|||||||
service: State<'_, SkillServiceState>,
|
service: State<'_, SkillServiceState>,
|
||||||
app_state: State<'_, AppState>,
|
app_state: State<'_, AppState>,
|
||||||
) -> Result<Vec<Skill>, String> {
|
) -> Result<Vec<Skill>, String> {
|
||||||
let repos = {
|
let repos = app_state.db.get_skill_repos().map_err(|e| e.to_string())?;
|
||||||
let config = app_state.config.read().map_err(|e| e.to_string())?;
|
|
||||||
config.skills.repos.clone()
|
|
||||||
};
|
|
||||||
|
|
||||||
service
|
service
|
||||||
.0
|
.0
|
||||||
@@ -31,10 +29,7 @@ pub async fn install_skill(
|
|||||||
app_state: State<'_, AppState>,
|
app_state: State<'_, AppState>,
|
||||||
) -> Result<bool, String> {
|
) -> Result<bool, String> {
|
||||||
// 先在不持有写锁的情况下收集仓库与技能信息
|
// 先在不持有写锁的情况下收集仓库与技能信息
|
||||||
let repos = {
|
let repos = app_state.db.get_skill_repos().map_err(|e| e.to_string())?;
|
||||||
let config = app_state.config.read().map_err(|e| e.to_string())?;
|
|
||||||
config.skills.repos.clone()
|
|
||||||
};
|
|
||||||
|
|
||||||
let skills = service
|
let skills = service
|
||||||
.0
|
.0
|
||||||
@@ -45,24 +40,36 @@ pub async fn install_skill(
|
|||||||
let skill = skills
|
let skill = skills
|
||||||
.iter()
|
.iter()
|
||||||
.find(|s| s.directory.eq_ignore_ascii_case(&directory))
|
.find(|s| s.directory.eq_ignore_ascii_case(&directory))
|
||||||
.ok_or_else(|| "技能不存在".to_string())?;
|
.ok_or_else(|| {
|
||||||
|
format_skill_error(
|
||||||
|
"SKILL_NOT_FOUND",
|
||||||
|
&[("directory", &directory)],
|
||||||
|
Some("checkRepoUrl"),
|
||||||
|
)
|
||||||
|
})?;
|
||||||
|
|
||||||
if !skill.installed {
|
if !skill.installed {
|
||||||
let repo = SkillRepo {
|
let repo = SkillRepo {
|
||||||
owner: skill
|
owner: skill.repo_owner.clone().ok_or_else(|| {
|
||||||
.repo_owner
|
format_skill_error(
|
||||||
.clone()
|
"MISSING_REPO_INFO",
|
||||||
.ok_or_else(|| "缺少仓库信息".to_string())?,
|
&[("directory", &directory), ("field", "owner")],
|
||||||
name: skill
|
None,
|
||||||
.repo_name
|
)
|
||||||
.clone()
|
})?,
|
||||||
.ok_or_else(|| "缺少仓库信息".to_string())?,
|
name: skill.repo_name.clone().ok_or_else(|| {
|
||||||
|
format_skill_error(
|
||||||
|
"MISSING_REPO_INFO",
|
||||||
|
&[("directory", &directory), ("field", "name")],
|
||||||
|
None,
|
||||||
|
)
|
||||||
|
})?,
|
||||||
branch: skill
|
branch: skill
|
||||||
.repo_branch
|
.repo_branch
|
||||||
.clone()
|
.clone()
|
||||||
.unwrap_or_else(|| "main".to_string()),
|
.unwrap_or_else(|| "main".to_string()),
|
||||||
enabled: true,
|
enabled: true,
|
||||||
skills_path: None, // 安装时使用默认路径
|
skills_path: skill.skills_path.clone(), // 使用技能记录的 skills_path
|
||||||
};
|
};
|
||||||
|
|
||||||
service
|
service
|
||||||
@@ -72,19 +79,16 @@ pub async fn install_skill(
|
|||||||
.map_err(|e| e.to_string())?;
|
.map_err(|e| e.to_string())?;
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
app_state
|
||||||
let mut config = app_state.config.write().map_err(|e| e.to_string())?;
|
.db
|
||||||
|
.update_skill_state(
|
||||||
config.skills.skills.insert(
|
&directory,
|
||||||
directory.clone(),
|
&SkillState {
|
||||||
SkillState {
|
|
||||||
installed: true,
|
installed: true,
|
||||||
installed_at: Utc::now(),
|
installed_at: Utc::now(),
|
||||||
},
|
},
|
||||||
);
|
)
|
||||||
}
|
.map_err(|e| e.to_string())?;
|
||||||
|
|
||||||
app_state.save().map_err(|e| e.to_string())?;
|
|
||||||
|
|
||||||
Ok(true)
|
Ok(true)
|
||||||
}
|
}
|
||||||
@@ -100,13 +104,17 @@ pub fn uninstall_skill(
|
|||||||
.uninstall_skill(directory.clone())
|
.uninstall_skill(directory.clone())
|
||||||
.map_err(|e| e.to_string())?;
|
.map_err(|e| e.to_string())?;
|
||||||
|
|
||||||
{
|
// Remove from database by setting installed = false
|
||||||
let mut config = app_state.config.write().map_err(|e| e.to_string())?;
|
app_state
|
||||||
|
.db
|
||||||
config.skills.skills.remove(&directory);
|
.update_skill_state(
|
||||||
}
|
&directory,
|
||||||
|
&SkillState {
|
||||||
app_state.save().map_err(|e| e.to_string())?;
|
installed: false,
|
||||||
|
installed_at: Utc::now(),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.map_err(|e| e.to_string())?;
|
||||||
|
|
||||||
Ok(true)
|
Ok(true)
|
||||||
}
|
}
|
||||||
@@ -116,28 +124,19 @@ pub fn get_skill_repos(
|
|||||||
_service: State<'_, SkillServiceState>,
|
_service: State<'_, SkillServiceState>,
|
||||||
app_state: State<'_, AppState>,
|
app_state: State<'_, AppState>,
|
||||||
) -> Result<Vec<SkillRepo>, String> {
|
) -> Result<Vec<SkillRepo>, String> {
|
||||||
let config = app_state.config.read().map_err(|e| e.to_string())?;
|
app_state.db.get_skill_repos().map_err(|e| e.to_string())
|
||||||
|
|
||||||
Ok(config.skills.repos.clone())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
pub fn add_skill_repo(
|
pub fn add_skill_repo(
|
||||||
repo: SkillRepo,
|
repo: SkillRepo,
|
||||||
service: State<'_, SkillServiceState>,
|
_service: State<'_, SkillServiceState>,
|
||||||
app_state: State<'_, AppState>,
|
app_state: State<'_, AppState>,
|
||||||
) -> Result<bool, String> {
|
) -> Result<bool, String> {
|
||||||
{
|
app_state
|
||||||
let mut config = app_state.config.write().map_err(|e| e.to_string())?;
|
.db
|
||||||
|
.save_skill_repo(&repo)
|
||||||
service
|
.map_err(|e| e.to_string())?;
|
||||||
.0
|
|
||||||
.add_repo(&mut config.skills, repo)
|
|
||||||
.map_err(|e| e.to_string())?;
|
|
||||||
}
|
|
||||||
|
|
||||||
app_state.save().map_err(|e| e.to_string())?;
|
|
||||||
|
|
||||||
Ok(true)
|
Ok(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -145,19 +144,12 @@ pub fn add_skill_repo(
|
|||||||
pub fn remove_skill_repo(
|
pub fn remove_skill_repo(
|
||||||
owner: String,
|
owner: String,
|
||||||
name: String,
|
name: String,
|
||||||
service: State<'_, SkillServiceState>,
|
_service: State<'_, SkillServiceState>,
|
||||||
app_state: State<'_, AppState>,
|
app_state: State<'_, AppState>,
|
||||||
) -> Result<bool, String> {
|
) -> Result<bool, String> {
|
||||||
{
|
app_state
|
||||||
let mut config = app_state.config.write().map_err(|e| e.to_string())?;
|
.db
|
||||||
|
.delete_skill_repo(&owner, &name)
|
||||||
service
|
.map_err(|e| e.to_string())?;
|
||||||
.0
|
|
||||||
.remove_repo(&mut config.skills, owner, name)
|
|
||||||
.map_err(|e| e.to_string())?;
|
|
||||||
}
|
|
||||||
|
|
||||||
app_state.save().map_err(|e| e.to_string())?;
|
|
||||||
|
|
||||||
Ok(true)
|
Ok(true)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -79,6 +79,7 @@ pub fn get_app_config_path() -> PathBuf {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// 清理供应商名称,确保文件名安全
|
/// 清理供应商名称,确保文件名安全
|
||||||
|
#[allow(dead_code)]
|
||||||
pub fn sanitize_provider_name(name: &str) -> String {
|
pub fn sanitize_provider_name(name: &str) -> String {
|
||||||
name.chars()
|
name.chars()
|
||||||
.map(|c| match c {
|
.map(|c| match c {
|
||||||
@@ -90,6 +91,7 @@ pub fn sanitize_provider_name(name: &str) -> String {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// 获取供应商配置文件路径
|
/// 获取供应商配置文件路径
|
||||||
|
#[allow(dead_code)]
|
||||||
pub fn get_provider_config_path(provider_id: &str, provider_name: Option<&str>) -> PathBuf {
|
pub fn get_provider_config_path(provider_id: &str, provider_name: Option<&str>) -> PathBuf {
|
||||||
let base_name = provider_name
|
let base_name = provider_name
|
||||||
.map(sanitize_provider_name)
|
.map(sanitize_provider_name)
|
||||||
|
|||||||
846
src-tauri/src/database.rs
Normal file
846
src-tauri/src/database.rs
Normal file
@@ -0,0 +1,846 @@
|
|||||||
|
use crate::app_config::{McpApps, McpServer, MultiAppConfig};
|
||||||
|
use crate::config::get_app_config_dir;
|
||||||
|
use crate::error::AppError;
|
||||||
|
use crate::prompt::Prompt;
|
||||||
|
use crate::provider::{Provider, ProviderMeta};
|
||||||
|
use crate::services::skill::{SkillRepo, SkillState};
|
||||||
|
use indexmap::IndexMap;
|
||||||
|
use rusqlite::{params, Connection, Result};
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use std::sync::Mutex;
|
||||||
|
|
||||||
|
pub struct Database {
|
||||||
|
// 使用 Mutex 包装 Connection 以支持在多线程环境(如 Tauri State)中共享
|
||||||
|
// rusqlite::Connection 本身不是 Sync 的
|
||||||
|
conn: Mutex<Connection>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Database {
|
||||||
|
/// 初始化数据库连接并创建表
|
||||||
|
pub fn init() -> Result<Self, AppError> {
|
||||||
|
let db_path = get_app_config_dir().join("cc-switch.db");
|
||||||
|
|
||||||
|
// 确保父目录存在
|
||||||
|
if let Some(parent) = db_path.parent() {
|
||||||
|
std::fs::create_dir_all(parent).map_err(|e| AppError::io(parent, e))?;
|
||||||
|
}
|
||||||
|
|
||||||
|
let conn = Connection::open(&db_path).map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// 启用外键约束
|
||||||
|
conn.execute("PRAGMA foreign_keys = ON;", [])
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let db = Self {
|
||||||
|
conn: Mutex::new(conn),
|
||||||
|
};
|
||||||
|
db.create_tables()?;
|
||||||
|
|
||||||
|
Ok(db)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_tables(&self) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
|
||||||
|
// 1. Providers 表
|
||||||
|
conn.execute(
|
||||||
|
"CREATE TABLE IF NOT EXISTS providers (
|
||||||
|
id TEXT NOT NULL,
|
||||||
|
app_type TEXT NOT NULL,
|
||||||
|
name TEXT NOT NULL,
|
||||||
|
settings_config TEXT NOT NULL,
|
||||||
|
website_url TEXT,
|
||||||
|
category TEXT,
|
||||||
|
created_at INTEGER,
|
||||||
|
sort_index INTEGER,
|
||||||
|
notes TEXT,
|
||||||
|
icon TEXT,
|
||||||
|
icon_color TEXT,
|
||||||
|
meta TEXT,
|
||||||
|
is_current BOOLEAN NOT NULL DEFAULT 0,
|
||||||
|
PRIMARY KEY (id, app_type)
|
||||||
|
)",
|
||||||
|
[],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// 2. Provider Endpoints 表
|
||||||
|
conn.execute(
|
||||||
|
"CREATE TABLE IF NOT EXISTS provider_endpoints (
|
||||||
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||||
|
provider_id TEXT NOT NULL,
|
||||||
|
app_type TEXT NOT NULL,
|
||||||
|
url TEXT NOT NULL,
|
||||||
|
added_at INTEGER,
|
||||||
|
FOREIGN KEY (provider_id, app_type) REFERENCES providers(id, app_type) ON DELETE CASCADE
|
||||||
|
)",
|
||||||
|
[],
|
||||||
|
).map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// 3. MCP Servers 表
|
||||||
|
conn.execute(
|
||||||
|
"CREATE TABLE IF NOT EXISTS mcp_servers (
|
||||||
|
id TEXT PRIMARY KEY,
|
||||||
|
name TEXT NOT NULL,
|
||||||
|
server_config TEXT NOT NULL,
|
||||||
|
description TEXT,
|
||||||
|
homepage TEXT,
|
||||||
|
docs TEXT,
|
||||||
|
tags TEXT,
|
||||||
|
enabled_claude BOOLEAN NOT NULL DEFAULT 0,
|
||||||
|
enabled_codex BOOLEAN NOT NULL DEFAULT 0,
|
||||||
|
enabled_gemini BOOLEAN NOT NULL DEFAULT 0
|
||||||
|
)",
|
||||||
|
[],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// 4. Prompts 表
|
||||||
|
conn.execute(
|
||||||
|
"CREATE TABLE IF NOT EXISTS prompts (
|
||||||
|
id TEXT NOT NULL,
|
||||||
|
app_type TEXT NOT NULL,
|
||||||
|
name TEXT NOT NULL,
|
||||||
|
content TEXT NOT NULL,
|
||||||
|
description TEXT,
|
||||||
|
enabled BOOLEAN NOT NULL DEFAULT 1,
|
||||||
|
created_at INTEGER,
|
||||||
|
updated_at INTEGER,
|
||||||
|
PRIMARY KEY (id, app_type)
|
||||||
|
)",
|
||||||
|
[],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// 5. Skills 表
|
||||||
|
conn.execute(
|
||||||
|
"CREATE TABLE IF NOT EXISTS skills (
|
||||||
|
key TEXT PRIMARY KEY,
|
||||||
|
installed BOOLEAN NOT NULL DEFAULT 0,
|
||||||
|
installed_at INTEGER
|
||||||
|
)",
|
||||||
|
[],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// 6. Skill Repos 表
|
||||||
|
conn.execute(
|
||||||
|
"CREATE TABLE IF NOT EXISTS skill_repos (
|
||||||
|
owner TEXT NOT NULL,
|
||||||
|
name TEXT NOT NULL,
|
||||||
|
branch TEXT NOT NULL,
|
||||||
|
enabled BOOLEAN NOT NULL DEFAULT 1,
|
||||||
|
skills_path TEXT,
|
||||||
|
PRIMARY KEY (owner, name)
|
||||||
|
)",
|
||||||
|
[],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// 7. Settings 表 (通用配置)
|
||||||
|
conn.execute(
|
||||||
|
"CREATE TABLE IF NOT EXISTS settings (
|
||||||
|
key TEXT PRIMARY KEY,
|
||||||
|
value TEXT
|
||||||
|
)",
|
||||||
|
[],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 从 MultiAppConfig 迁移数据
|
||||||
|
pub fn migrate_from_json(&self, config: &MultiAppConfig) -> Result<(), AppError> {
|
||||||
|
let mut conn = self.conn.lock().unwrap();
|
||||||
|
let tx = conn
|
||||||
|
.transaction()
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// 1. 迁移 Providers
|
||||||
|
for (app_key, manager) in &config.apps {
|
||||||
|
let app_type = app_key; // "claude", "codex", "gemini"
|
||||||
|
let current_id = &manager.current;
|
||||||
|
|
||||||
|
for (id, provider) in &manager.providers {
|
||||||
|
let is_current = if id == current_id { 1 } else { 0 };
|
||||||
|
|
||||||
|
// 处理 meta 和 endpoints
|
||||||
|
let mut meta_clone = provider.meta.clone().unwrap_or_default();
|
||||||
|
let endpoints = std::mem::take(&mut meta_clone.custom_endpoints);
|
||||||
|
|
||||||
|
tx.execute(
|
||||||
|
"INSERT OR REPLACE INTO providers (
|
||||||
|
id, app_type, name, settings_config, website_url, category,
|
||||||
|
created_at, sort_index, notes, icon, icon_color, meta, is_current
|
||||||
|
) VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7, ?8, ?9, ?10, ?11, ?12, ?13)",
|
||||||
|
params![
|
||||||
|
id,
|
||||||
|
app_type,
|
||||||
|
provider.name,
|
||||||
|
serde_json::to_string(&provider.settings_config).unwrap(),
|
||||||
|
provider.website_url,
|
||||||
|
provider.category,
|
||||||
|
provider.created_at,
|
||||||
|
provider.sort_index,
|
||||||
|
provider.notes,
|
||||||
|
provider.icon,
|
||||||
|
provider.icon_color,
|
||||||
|
serde_json::to_string(&meta_clone).unwrap(), // 不含 endpoints 的 meta
|
||||||
|
is_current,
|
||||||
|
],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(format!("Migrate provider failed: {e}")))?;
|
||||||
|
|
||||||
|
// 迁移 Endpoints
|
||||||
|
for (url, endpoint) in endpoints {
|
||||||
|
tx.execute(
|
||||||
|
"INSERT INTO provider_endpoints (provider_id, app_type, url, added_at)
|
||||||
|
VALUES (?1, ?2, ?3, ?4)",
|
||||||
|
params![id, app_type, url, endpoint.added_at],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(format!("Migrate endpoint failed: {e}")))?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2. 迁移 MCP Servers
|
||||||
|
if let Some(servers) = &config.mcp.servers {
|
||||||
|
for (id, server) in servers {
|
||||||
|
tx.execute(
|
||||||
|
"INSERT OR REPLACE INTO mcp_servers (
|
||||||
|
id, name, server_config, description, homepage, docs, tags,
|
||||||
|
enabled_claude, enabled_codex, enabled_gemini
|
||||||
|
) VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7, ?8, ?9, ?10)",
|
||||||
|
params![
|
||||||
|
id,
|
||||||
|
server.name,
|
||||||
|
serde_json::to_string(&server.server).unwrap(),
|
||||||
|
server.description,
|
||||||
|
server.homepage,
|
||||||
|
server.docs,
|
||||||
|
serde_json::to_string(&server.tags).unwrap(),
|
||||||
|
server.apps.claude,
|
||||||
|
server.apps.codex,
|
||||||
|
server.apps.gemini,
|
||||||
|
],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(format!("Migrate mcp server failed: {e}")))?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. 迁移 Prompts
|
||||||
|
let migrate_prompts =
|
||||||
|
|prompts_map: &std::collections::HashMap<String, crate::prompt::Prompt>,
|
||||||
|
app_type: &str|
|
||||||
|
-> Result<(), AppError> {
|
||||||
|
for (id, prompt) in prompts_map {
|
||||||
|
tx.execute(
|
||||||
|
"INSERT OR REPLACE INTO prompts (
|
||||||
|
id, app_type, name, content, description, enabled, created_at, updated_at
|
||||||
|
) VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7, ?8)",
|
||||||
|
params![
|
||||||
|
id,
|
||||||
|
app_type,
|
||||||
|
prompt.name,
|
||||||
|
prompt.content,
|
||||||
|
prompt.description,
|
||||||
|
prompt.enabled,
|
||||||
|
prompt.created_at,
|
||||||
|
prompt.updated_at,
|
||||||
|
],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(format!("Migrate prompt failed: {e}")))?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
};
|
||||||
|
|
||||||
|
migrate_prompts(&config.prompts.claude.prompts, "claude")?;
|
||||||
|
migrate_prompts(&config.prompts.codex.prompts, "codex")?;
|
||||||
|
migrate_prompts(&config.prompts.gemini.prompts, "gemini")?;
|
||||||
|
|
||||||
|
// 4. 迁移 Skills
|
||||||
|
for (key, state) in &config.skills.skills {
|
||||||
|
tx.execute(
|
||||||
|
"INSERT OR REPLACE INTO skills (key, installed, installed_at) VALUES (?1, ?2, ?3)",
|
||||||
|
params![key, state.installed, state.installed_at.timestamp()],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(format!("Migrate skill failed: {e}")))?;
|
||||||
|
}
|
||||||
|
|
||||||
|
for repo in &config.skills.repos {
|
||||||
|
tx.execute(
|
||||||
|
"INSERT OR REPLACE INTO skill_repos (owner, name, branch, enabled, skills_path) VALUES (?1, ?2, ?3, ?4, ?5)",
|
||||||
|
params![repo.owner, repo.name, repo.branch, repo.enabled, repo.skills_path],
|
||||||
|
).map_err(|e| AppError::Database(format!("Migrate skill repo failed: {e}")))?;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 5. 迁移 Common Config
|
||||||
|
if let Some(snippet) = &config.common_config_snippets.claude {
|
||||||
|
tx.execute(
|
||||||
|
"INSERT OR REPLACE INTO settings (key, value) VALUES (?1, ?2)",
|
||||||
|
params!["common_config_claude", snippet],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(format!("Migrate settings failed: {e}")))?;
|
||||||
|
}
|
||||||
|
if let Some(snippet) = &config.common_config_snippets.codex {
|
||||||
|
tx.execute(
|
||||||
|
"INSERT OR REPLACE INTO settings (key, value) VALUES (?1, ?2)",
|
||||||
|
params!["common_config_codex", snippet],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(format!("Migrate settings failed: {e}")))?;
|
||||||
|
}
|
||||||
|
if let Some(snippet) = &config.common_config_snippets.gemini {
|
||||||
|
tx.execute(
|
||||||
|
"INSERT OR REPLACE INTO settings (key, value) VALUES (?1, ?2)",
|
||||||
|
params!["common_config_gemini", snippet],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(format!("Migrate settings failed: {e}")))?;
|
||||||
|
}
|
||||||
|
|
||||||
|
tx.commit()
|
||||||
|
.map_err(|e| AppError::Database(format!("Commit migration failed: {e}")))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- Providers DAO ---
|
||||||
|
|
||||||
|
pub fn get_all_providers(
|
||||||
|
&self,
|
||||||
|
app_type: &str,
|
||||||
|
) -> Result<IndexMap<String, Provider>, AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
let mut stmt = conn.prepare(
|
||||||
|
"SELECT id, name, settings_config, website_url, category, created_at, sort_index, notes, icon, icon_color, meta
|
||||||
|
FROM providers WHERE app_type = ?1
|
||||||
|
ORDER BY COALESCE(sort_index, 999999), created_at ASC, id ASC"
|
||||||
|
).map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let provider_iter = stmt
|
||||||
|
.query_map(params![app_type], |row| {
|
||||||
|
let id: String = row.get(0)?;
|
||||||
|
let name: String = row.get(1)?;
|
||||||
|
let settings_config_str: String = row.get(2)?;
|
||||||
|
let website_url: Option<String> = row.get(3)?;
|
||||||
|
let category: Option<String> = row.get(4)?;
|
||||||
|
let created_at: Option<i64> = row.get(5)?;
|
||||||
|
let sort_index: Option<usize> = row.get(6)?;
|
||||||
|
let notes: Option<String> = row.get(7)?;
|
||||||
|
let icon: Option<String> = row.get(8)?;
|
||||||
|
let icon_color: Option<String> = row.get(9)?;
|
||||||
|
let meta_str: String = row.get(10)?;
|
||||||
|
|
||||||
|
let settings_config =
|
||||||
|
serde_json::from_str(&settings_config_str).unwrap_or(serde_json::Value::Null);
|
||||||
|
let meta: ProviderMeta = serde_json::from_str(&meta_str).unwrap_or_default();
|
||||||
|
|
||||||
|
Ok((
|
||||||
|
id,
|
||||||
|
Provider {
|
||||||
|
id: "".to_string(), // Placeholder, set below
|
||||||
|
name,
|
||||||
|
settings_config,
|
||||||
|
website_url,
|
||||||
|
category,
|
||||||
|
created_at,
|
||||||
|
sort_index,
|
||||||
|
notes,
|
||||||
|
meta: Some(meta),
|
||||||
|
icon,
|
||||||
|
icon_color,
|
||||||
|
},
|
||||||
|
))
|
||||||
|
})
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let mut providers = IndexMap::new();
|
||||||
|
for provider_res in provider_iter {
|
||||||
|
let (id, mut provider) = provider_res.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
provider.id = id.clone();
|
||||||
|
|
||||||
|
// Load endpoints
|
||||||
|
let mut stmt_endpoints = conn.prepare(
|
||||||
|
"SELECT url, added_at FROM provider_endpoints WHERE provider_id = ?1 AND app_type = ?2 ORDER BY added_at ASC, url ASC"
|
||||||
|
).map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let endpoints_iter = stmt_endpoints
|
||||||
|
.query_map(params![id, app_type], |row| {
|
||||||
|
let url: String = row.get(0)?;
|
||||||
|
let added_at: Option<i64> = row.get(1)?;
|
||||||
|
Ok((
|
||||||
|
url,
|
||||||
|
crate::settings::CustomEndpoint {
|
||||||
|
url: "".to_string(),
|
||||||
|
added_at: added_at.unwrap_or(0),
|
||||||
|
last_used: None,
|
||||||
|
},
|
||||||
|
))
|
||||||
|
})
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let mut custom_endpoints = HashMap::new();
|
||||||
|
for ep_res in endpoints_iter {
|
||||||
|
let (url, mut ep) = ep_res.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
ep.url = url.clone();
|
||||||
|
custom_endpoints.insert(url, ep);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(meta) = &mut provider.meta {
|
||||||
|
meta.custom_endpoints = custom_endpoints;
|
||||||
|
}
|
||||||
|
|
||||||
|
providers.insert(id, provider);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(providers)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_current_provider(&self, app_type: &str) -> Result<Option<String>, AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
let mut stmt = conn
|
||||||
|
.prepare("SELECT id FROM providers WHERE app_type = ?1 AND is_current = 1 LIMIT 1")
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let mut rows = stmt
|
||||||
|
.query(params![app_type])
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
if let Some(row) = rows.next().map_err(|e| AppError::Database(e.to_string()))? {
|
||||||
|
Ok(Some(
|
||||||
|
row.get(0).map_err(|e| AppError::Database(e.to_string()))?,
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn save_provider(&self, app_type: &str, provider: &Provider) -> Result<(), AppError> {
|
||||||
|
let mut conn = self.conn.lock().unwrap();
|
||||||
|
let tx = conn
|
||||||
|
.transaction()
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// Handle meta and endpoints
|
||||||
|
let mut meta_clone = provider.meta.clone().unwrap_or_default();
|
||||||
|
let endpoints = std::mem::take(&mut meta_clone.custom_endpoints);
|
||||||
|
|
||||||
|
// Check if it exists to preserve is_current
|
||||||
|
let is_current: bool = tx
|
||||||
|
.query_row(
|
||||||
|
"SELECT is_current FROM providers WHERE id = ?1 AND app_type = ?2",
|
||||||
|
params![provider.id, app_type],
|
||||||
|
|row| row.get(0),
|
||||||
|
)
|
||||||
|
.unwrap_or(false);
|
||||||
|
|
||||||
|
tx.execute(
|
||||||
|
"INSERT OR REPLACE INTO providers (
|
||||||
|
id, app_type, name, settings_config, website_url, category,
|
||||||
|
created_at, sort_index, notes, icon, icon_color, meta, is_current
|
||||||
|
) VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7, ?8, ?9, ?10, ?11, ?12, ?13)",
|
||||||
|
params![
|
||||||
|
provider.id,
|
||||||
|
app_type,
|
||||||
|
provider.name,
|
||||||
|
serde_json::to_string(&provider.settings_config).unwrap(),
|
||||||
|
provider.website_url,
|
||||||
|
provider.category,
|
||||||
|
provider.created_at,
|
||||||
|
provider.sort_index,
|
||||||
|
provider.notes,
|
||||||
|
provider.icon,
|
||||||
|
provider.icon_color,
|
||||||
|
serde_json::to_string(&meta_clone).unwrap(),
|
||||||
|
is_current,
|
||||||
|
],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// Sync endpoints: Delete all and re-insert
|
||||||
|
tx.execute(
|
||||||
|
"DELETE FROM provider_endpoints WHERE provider_id = ?1 AND app_type = ?2",
|
||||||
|
params![provider.id, app_type],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
for (url, endpoint) in endpoints {
|
||||||
|
tx.execute(
|
||||||
|
"INSERT INTO provider_endpoints (provider_id, app_type, url, added_at)
|
||||||
|
VALUES (?1, ?2, ?3, ?4)",
|
||||||
|
params![provider.id, app_type, url, endpoint.added_at],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
}
|
||||||
|
|
||||||
|
tx.commit().map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn delete_provider(&self, app_type: &str, id: &str) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
conn.execute(
|
||||||
|
"DELETE FROM providers WHERE id = ?1 AND app_type = ?2",
|
||||||
|
params![id, app_type],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_current_provider(&self, app_type: &str, id: &str) -> Result<(), AppError> {
|
||||||
|
let mut conn = self.conn.lock().unwrap();
|
||||||
|
let tx = conn
|
||||||
|
.transaction()
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// Reset all to 0
|
||||||
|
tx.execute(
|
||||||
|
"UPDATE providers SET is_current = 0 WHERE app_type = ?1",
|
||||||
|
params![app_type],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
// Set new current
|
||||||
|
tx.execute(
|
||||||
|
"UPDATE providers SET is_current = 1 WHERE id = ?1 AND app_type = ?2",
|
||||||
|
params![id, app_type],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
tx.commit().map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn add_custom_endpoint(
|
||||||
|
&self,
|
||||||
|
app_type: &str,
|
||||||
|
provider_id: &str,
|
||||||
|
url: &str,
|
||||||
|
) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
let added_at = chrono::Utc::now().timestamp_millis();
|
||||||
|
conn.execute(
|
||||||
|
"INSERT INTO provider_endpoints (provider_id, app_type, url, added_at) VALUES (?1, ?2, ?3, ?4)",
|
||||||
|
params![provider_id, app_type, url, added_at],
|
||||||
|
).map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn remove_custom_endpoint(
|
||||||
|
&self,
|
||||||
|
app_type: &str,
|
||||||
|
provider_id: &str,
|
||||||
|
url: &str,
|
||||||
|
) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
conn.execute(
|
||||||
|
"DELETE FROM provider_endpoints WHERE provider_id = ?1 AND app_type = ?2 AND url = ?3",
|
||||||
|
params![provider_id, app_type, url],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- MCP Servers DAO ---
|
||||||
|
|
||||||
|
pub fn get_all_mcp_servers(&self) -> Result<IndexMap<String, McpServer>, AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
let mut stmt = conn.prepare(
|
||||||
|
"SELECT id, name, server_config, description, homepage, docs, tags, enabled_claude, enabled_codex, enabled_gemini
|
||||||
|
FROM mcp_servers
|
||||||
|
ORDER BY name ASC, id ASC"
|
||||||
|
).map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let server_iter = stmt
|
||||||
|
.query_map([], |row| {
|
||||||
|
let id: String = row.get(0)?;
|
||||||
|
let name: String = row.get(1)?;
|
||||||
|
let server_config_str: String = row.get(2)?;
|
||||||
|
let description: Option<String> = row.get(3)?;
|
||||||
|
let homepage: Option<String> = row.get(4)?;
|
||||||
|
let docs: Option<String> = row.get(5)?;
|
||||||
|
let tags_str: String = row.get(6)?;
|
||||||
|
let enabled_claude: bool = row.get(7)?;
|
||||||
|
let enabled_codex: bool = row.get(8)?;
|
||||||
|
let enabled_gemini: bool = row.get(9)?;
|
||||||
|
|
||||||
|
let server = serde_json::from_str(&server_config_str).unwrap_or_default();
|
||||||
|
let tags = serde_json::from_str(&tags_str).unwrap_or_default();
|
||||||
|
|
||||||
|
Ok((
|
||||||
|
id.clone(),
|
||||||
|
McpServer {
|
||||||
|
id,
|
||||||
|
name,
|
||||||
|
server,
|
||||||
|
apps: McpApps {
|
||||||
|
claude: enabled_claude,
|
||||||
|
codex: enabled_codex,
|
||||||
|
gemini: enabled_gemini,
|
||||||
|
},
|
||||||
|
description,
|
||||||
|
homepage,
|
||||||
|
docs,
|
||||||
|
tags,
|
||||||
|
},
|
||||||
|
))
|
||||||
|
})
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let mut servers = IndexMap::new();
|
||||||
|
for server_res in server_iter {
|
||||||
|
let (id, server) = server_res.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
servers.insert(id, server);
|
||||||
|
}
|
||||||
|
Ok(servers)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn save_mcp_server(&self, server: &McpServer) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
conn.execute(
|
||||||
|
"INSERT OR REPLACE INTO mcp_servers (
|
||||||
|
id, name, server_config, description, homepage, docs, tags,
|
||||||
|
enabled_claude, enabled_codex, enabled_gemini
|
||||||
|
) VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7, ?8, ?9, ?10)",
|
||||||
|
params![
|
||||||
|
server.id,
|
||||||
|
server.name,
|
||||||
|
serde_json::to_string(&server.server).unwrap(),
|
||||||
|
server.description,
|
||||||
|
server.homepage,
|
||||||
|
server.docs,
|
||||||
|
serde_json::to_string(&server.tags).unwrap(),
|
||||||
|
server.apps.claude,
|
||||||
|
server.apps.codex,
|
||||||
|
server.apps.gemini,
|
||||||
|
],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn delete_mcp_server(&self, id: &str) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
conn.execute("DELETE FROM mcp_servers WHERE id = ?1", params![id])
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- Prompts DAO ---
|
||||||
|
|
||||||
|
pub fn get_prompts(&self, app_type: &str) -> Result<IndexMap<String, Prompt>, AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
let mut stmt = conn
|
||||||
|
.prepare(
|
||||||
|
"SELECT id, name, content, description, enabled, created_at, updated_at
|
||||||
|
FROM prompts WHERE app_type = ?1
|
||||||
|
ORDER BY created_at ASC, id ASC",
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let prompt_iter = stmt
|
||||||
|
.query_map(params![app_type], |row| {
|
||||||
|
let id: String = row.get(0)?;
|
||||||
|
let name: String = row.get(1)?;
|
||||||
|
let content: String = row.get(2)?;
|
||||||
|
let description: Option<String> = row.get(3)?;
|
||||||
|
let enabled: bool = row.get(4)?;
|
||||||
|
let created_at: Option<i64> = row.get(5)?;
|
||||||
|
let updated_at: Option<i64> = row.get(6)?;
|
||||||
|
|
||||||
|
Ok((
|
||||||
|
id.clone(),
|
||||||
|
Prompt {
|
||||||
|
id,
|
||||||
|
name,
|
||||||
|
content,
|
||||||
|
description,
|
||||||
|
enabled,
|
||||||
|
created_at,
|
||||||
|
updated_at,
|
||||||
|
},
|
||||||
|
))
|
||||||
|
})
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let mut prompts = IndexMap::new();
|
||||||
|
for prompt_res in prompt_iter {
|
||||||
|
let (id, prompt) = prompt_res.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
prompts.insert(id, prompt);
|
||||||
|
}
|
||||||
|
Ok(prompts)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn save_prompt(&self, app_type: &str, prompt: &Prompt) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
conn.execute(
|
||||||
|
"INSERT OR REPLACE INTO prompts (
|
||||||
|
id, app_type, name, content, description, enabled, created_at, updated_at
|
||||||
|
) VALUES (?1, ?2, ?3, ?4, ?5, ?6, ?7, ?8)",
|
||||||
|
params![
|
||||||
|
prompt.id,
|
||||||
|
app_type,
|
||||||
|
prompt.name,
|
||||||
|
prompt.content,
|
||||||
|
prompt.description,
|
||||||
|
prompt.enabled,
|
||||||
|
prompt.created_at,
|
||||||
|
prompt.updated_at,
|
||||||
|
],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn delete_prompt(&self, app_type: &str, id: &str) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
conn.execute(
|
||||||
|
"DELETE FROM prompts WHERE id = ?1 AND app_type = ?2",
|
||||||
|
params![id, app_type],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- Skills DAO ---
|
||||||
|
|
||||||
|
pub fn get_skills(&self) -> Result<IndexMap<String, SkillState>, AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
let mut stmt = conn
|
||||||
|
.prepare("SELECT key, installed, installed_at FROM skills ORDER BY key ASC")
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let skill_iter = stmt
|
||||||
|
.query_map([], |row| {
|
||||||
|
let key: String = row.get(0)?;
|
||||||
|
let installed: bool = row.get(1)?;
|
||||||
|
let installed_at_ts: i64 = row.get(2)?;
|
||||||
|
|
||||||
|
let installed_at =
|
||||||
|
chrono::DateTime::from_timestamp(installed_at_ts, 0).unwrap_or_default();
|
||||||
|
|
||||||
|
Ok((
|
||||||
|
key,
|
||||||
|
SkillState {
|
||||||
|
installed,
|
||||||
|
installed_at,
|
||||||
|
},
|
||||||
|
))
|
||||||
|
})
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let mut skills = IndexMap::new();
|
||||||
|
for skill_res in skill_iter {
|
||||||
|
let (key, skill) = skill_res.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
skills.insert(key, skill);
|
||||||
|
}
|
||||||
|
Ok(skills)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn update_skill_state(&self, key: &str, state: &SkillState) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
conn.execute(
|
||||||
|
"INSERT OR REPLACE INTO skills (key, installed, installed_at) VALUES (?1, ?2, ?3)",
|
||||||
|
params![key, state.installed, state.installed_at.timestamp()],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_skill_repos(&self) -> Result<Vec<SkillRepo>, AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
let mut stmt = conn
|
||||||
|
.prepare("SELECT owner, name, branch, enabled, skills_path FROM skill_repos ORDER BY owner ASC, name ASC")
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let repo_iter = stmt
|
||||||
|
.query_map([], |row| {
|
||||||
|
Ok(SkillRepo {
|
||||||
|
owner: row.get(0)?,
|
||||||
|
name: row.get(1)?,
|
||||||
|
branch: row.get(2)?,
|
||||||
|
enabled: row.get(3)?,
|
||||||
|
skills_path: row.get(4)?,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let mut repos = Vec::new();
|
||||||
|
for repo_res in repo_iter {
|
||||||
|
repos.push(repo_res.map_err(|e| AppError::Database(e.to_string()))?);
|
||||||
|
}
|
||||||
|
Ok(repos)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn save_skill_repo(&self, repo: &SkillRepo) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
conn.execute(
|
||||||
|
"INSERT OR REPLACE INTO skill_repos (owner, name, branch, enabled, skills_path) VALUES (?1, ?2, ?3, ?4, ?5)",
|
||||||
|
params![repo.owner, repo.name, repo.branch, repo.enabled, repo.skills_path],
|
||||||
|
).map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn delete_skill_repo(&self, owner: &str, name: &str) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
conn.execute(
|
||||||
|
"DELETE FROM skill_repos WHERE owner = ?1 AND name = ?2",
|
||||||
|
params![owner, name],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- Settings DAO ---
|
||||||
|
|
||||||
|
pub fn get_setting(&self, key: &str) -> Result<Option<String>, AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
let mut stmt = conn
|
||||||
|
.prepare("SELECT value FROM settings WHERE key = ?1")
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
let mut rows = stmt
|
||||||
|
.query(params![key])
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
|
||||||
|
if let Some(row) = rows.next().map_err(|e| AppError::Database(e.to_string()))? {
|
||||||
|
Ok(Some(
|
||||||
|
row.get(0).map_err(|e| AppError::Database(e.to_string()))?,
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_setting(&self, key: &str, value: &str) -> Result<(), AppError> {
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
conn.execute(
|
||||||
|
"INSERT OR REPLACE INTO settings (key, value) VALUES (?1, ?2)",
|
||||||
|
params![key, value],
|
||||||
|
)
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- Config Snippets Helper Methods ---
|
||||||
|
|
||||||
|
pub fn get_config_snippet(&self, app_type: &str) -> Result<Option<String>, AppError> {
|
||||||
|
self.get_setting(&format!("common_config_{app_type}"))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_config_snippet(
|
||||||
|
&self,
|
||||||
|
app_type: &str,
|
||||||
|
snippet: Option<String>,
|
||||||
|
) -> Result<(), AppError> {
|
||||||
|
let key = format!("common_config_{app_type}");
|
||||||
|
if let Some(value) = snippet {
|
||||||
|
self.set_setting(&key, &value)
|
||||||
|
} else {
|
||||||
|
// Delete if None
|
||||||
|
let conn = self.conn.lock().unwrap();
|
||||||
|
conn.execute("DELETE FROM settings WHERE key = ?1", params![key])
|
||||||
|
.map_err(|e| AppError::Database(e.to_string()))?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -37,6 +37,24 @@ pub struct DeepLinkImportRequest {
|
|||||||
/// Optional notes/description
|
/// Optional notes/description
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
pub notes: Option<String>,
|
pub notes: Option<String>,
|
||||||
|
/// Optional Haiku model (Claude only, v3.7.1+)
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub haiku_model: Option<String>,
|
||||||
|
/// Optional Sonnet model (Claude only, v3.7.1+)
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub sonnet_model: Option<String>,
|
||||||
|
/// Optional Opus model (Claude only, v3.7.1+)
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub opus_model: Option<String>,
|
||||||
|
/// Optional Base64 encoded config content (v3.8+)
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub config: Option<String>,
|
||||||
|
/// Optional config format (json/toml, v3.8+)
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub config_format: Option<String>,
|
||||||
|
/// Optional remote config URL (v3.8+)
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub config_url: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Parse a ccswitch:// URL into a DeepLinkImportRequest
|
/// Parse a ccswitch:// URL into a DeepLinkImportRequest
|
||||||
@@ -110,29 +128,33 @@ pub fn parse_deeplink_url(url_str: &str) -> Result<DeepLinkImportRequest, AppErr
|
|||||||
.ok_or_else(|| AppError::InvalidInput("Missing 'name' parameter".to_string()))?
|
.ok_or_else(|| AppError::InvalidInput("Missing 'name' parameter".to_string()))?
|
||||||
.clone();
|
.clone();
|
||||||
|
|
||||||
let homepage = params
|
// Make these optional for config file auto-fill (v3.8+)
|
||||||
.get("homepage")
|
let homepage = params.get("homepage").cloned().unwrap_or_default();
|
||||||
.ok_or_else(|| AppError::InvalidInput("Missing 'homepage' parameter".to_string()))?
|
let endpoint = params.get("endpoint").cloned().unwrap_or_default();
|
||||||
.clone();
|
let api_key = params.get("apiKey").cloned().unwrap_or_default();
|
||||||
|
|
||||||
let endpoint = params
|
// Validate URLs only if provided
|
||||||
.get("endpoint")
|
if !homepage.is_empty() {
|
||||||
.ok_or_else(|| AppError::InvalidInput("Missing 'endpoint' parameter".to_string()))?
|
validate_url(&homepage, "homepage")?;
|
||||||
.clone();
|
}
|
||||||
|
if !endpoint.is_empty() {
|
||||||
let api_key = params
|
validate_url(&endpoint, "endpoint")?;
|
||||||
.get("apiKey")
|
}
|
||||||
.ok_or_else(|| AppError::InvalidInput("Missing 'apiKey' parameter".to_string()))?
|
|
||||||
.clone();
|
|
||||||
|
|
||||||
// Validate URLs
|
|
||||||
validate_url(&homepage, "homepage")?;
|
|
||||||
validate_url(&endpoint, "endpoint")?;
|
|
||||||
|
|
||||||
// Extract optional fields
|
// Extract optional fields
|
||||||
let model = params.get("model").cloned();
|
let model = params.get("model").cloned();
|
||||||
let notes = params.get("notes").cloned();
|
let notes = params.get("notes").cloned();
|
||||||
|
|
||||||
|
// Extract Claude-specific optional model fields (v3.7.1+)
|
||||||
|
let haiku_model = params.get("haikuModel").cloned();
|
||||||
|
let sonnet_model = params.get("sonnetModel").cloned();
|
||||||
|
let opus_model = params.get("opusModel").cloned();
|
||||||
|
|
||||||
|
// Extract optional config fields (v3.8+)
|
||||||
|
let config = params.get("config").cloned();
|
||||||
|
let config_format = params.get("configFormat").cloned();
|
||||||
|
let config_url = params.get("configUrl").cloned();
|
||||||
|
|
||||||
Ok(DeepLinkImportRequest {
|
Ok(DeepLinkImportRequest {
|
||||||
version,
|
version,
|
||||||
resource,
|
resource,
|
||||||
@@ -143,6 +165,12 @@ pub fn parse_deeplink_url(url_str: &str) -> Result<DeepLinkImportRequest, AppErr
|
|||||||
api_key,
|
api_key,
|
||||||
model,
|
model,
|
||||||
notes,
|
notes,
|
||||||
|
haiku_model,
|
||||||
|
sonnet_model,
|
||||||
|
opus_model,
|
||||||
|
config,
|
||||||
|
config_format,
|
||||||
|
config_url,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -165,23 +193,44 @@ fn validate_url(url_str: &str, field_name: &str) -> Result<(), AppError> {
|
|||||||
///
|
///
|
||||||
/// This function:
|
/// This function:
|
||||||
/// 1. Validates the request
|
/// 1. Validates the request
|
||||||
/// 2. Converts it to a Provider structure
|
/// 2. Merges config file if provided (v3.8+)
|
||||||
/// 3. Delegates to ProviderService for actual import
|
/// 3. Converts it to a Provider structure
|
||||||
|
/// 4. Delegates to ProviderService for actual import
|
||||||
pub fn import_provider_from_deeplink(
|
pub fn import_provider_from_deeplink(
|
||||||
state: &AppState,
|
state: &AppState,
|
||||||
request: DeepLinkImportRequest,
|
request: DeepLinkImportRequest,
|
||||||
) -> Result<String, AppError> {
|
) -> Result<String, AppError> {
|
||||||
|
// Step 1: Merge config file if provided (v3.8+)
|
||||||
|
let merged_request = parse_and_merge_config(&request)?;
|
||||||
|
|
||||||
|
// Step 2: Validate required fields after merge
|
||||||
|
if merged_request.api_key.is_empty() {
|
||||||
|
return Err(AppError::InvalidInput(
|
||||||
|
"API key is required (either in URL or config file)".to_string(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
if merged_request.endpoint.is_empty() {
|
||||||
|
return Err(AppError::InvalidInput(
|
||||||
|
"Endpoint is required (either in URL or config file)".to_string(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
if merged_request.homepage.is_empty() {
|
||||||
|
return Err(AppError::InvalidInput(
|
||||||
|
"Homepage is required (either in URL or config file)".to_string(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
// Parse app type
|
// Parse app type
|
||||||
let app_type = AppType::from_str(&request.app)
|
let app_type = AppType::from_str(&merged_request.app)
|
||||||
.map_err(|_| AppError::InvalidInput(format!("Invalid app type: {}", request.app)))?;
|
.map_err(|_| AppError::InvalidInput(format!("Invalid app type: {}", merged_request.app)))?;
|
||||||
|
|
||||||
// Build provider configuration based on app type
|
// Build provider configuration based on app type
|
||||||
let mut provider = build_provider_from_request(&app_type, &request)?;
|
let mut provider = build_provider_from_request(&app_type, &merged_request)?;
|
||||||
|
|
||||||
// Generate a unique ID for the provider using timestamp + sanitized name
|
// Generate a unique ID for the provider using timestamp + sanitized name
|
||||||
// This is similar to how frontend generates IDs
|
// This is similar to how frontend generates IDs
|
||||||
let timestamp = chrono::Utc::now().timestamp_millis();
|
let timestamp = chrono::Utc::now().timestamp_millis();
|
||||||
let sanitized_name = request
|
let sanitized_name = merged_request
|
||||||
.name
|
.name
|
||||||
.chars()
|
.chars()
|
||||||
.filter(|c| c.is_alphanumeric() || *c == '-' || *c == '_')
|
.filter(|c| c.is_alphanumeric() || *c == '-' || *c == '_')
|
||||||
@@ -211,11 +260,31 @@ fn build_provider_from_request(
|
|||||||
env.insert("ANTHROPIC_AUTH_TOKEN".to_string(), json!(request.api_key));
|
env.insert("ANTHROPIC_AUTH_TOKEN".to_string(), json!(request.api_key));
|
||||||
env.insert("ANTHROPIC_BASE_URL".to_string(), json!(request.endpoint));
|
env.insert("ANTHROPIC_BASE_URL".to_string(), json!(request.endpoint));
|
||||||
|
|
||||||
// Add model if provided (use as default model)
|
// Add default model if provided
|
||||||
if let Some(model) = &request.model {
|
if let Some(model) = &request.model {
|
||||||
env.insert("ANTHROPIC_MODEL".to_string(), json!(model));
|
env.insert("ANTHROPIC_MODEL".to_string(), json!(model));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Add Claude-specific model fields (v3.7.1+)
|
||||||
|
if let Some(haiku_model) = &request.haiku_model {
|
||||||
|
env.insert(
|
||||||
|
"ANTHROPIC_DEFAULT_HAIKU_MODEL".to_string(),
|
||||||
|
json!(haiku_model),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if let Some(sonnet_model) = &request.sonnet_model {
|
||||||
|
env.insert(
|
||||||
|
"ANTHROPIC_DEFAULT_SONNET_MODEL".to_string(),
|
||||||
|
json!(sonnet_model),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if let Some(opus_model) = &request.opus_model {
|
||||||
|
env.insert(
|
||||||
|
"ANTHROPIC_DEFAULT_OPUS_MODEL".to_string(),
|
||||||
|
json!(opus_model),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
json!({ "env": env })
|
json!({ "env": env })
|
||||||
}
|
}
|
||||||
AppType::Codex => {
|
AppType::Codex => {
|
||||||
@@ -319,11 +388,254 @@ requires_openai_auth = true
|
|||||||
sort_index: None,
|
sort_index: None,
|
||||||
notes: request.notes.clone(),
|
notes: request.notes.clone(),
|
||||||
meta: None,
|
meta: None,
|
||||||
|
icon: None,
|
||||||
|
icon_color: None,
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(provider)
|
Ok(provider)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Parse and merge configuration from Base64 encoded config or remote URL
|
||||||
|
///
|
||||||
|
/// Priority: URL params > inline config > remote config
|
||||||
|
pub fn parse_and_merge_config(
|
||||||
|
request: &DeepLinkImportRequest,
|
||||||
|
) -> Result<DeepLinkImportRequest, AppError> {
|
||||||
|
use base64::prelude::*;
|
||||||
|
|
||||||
|
// If no config provided, return original request
|
||||||
|
if request.config.is_none() && request.config_url.is_none() {
|
||||||
|
return Ok(request.clone());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Step 1: Get config content
|
||||||
|
let config_content = if let Some(config_b64) = &request.config {
|
||||||
|
// Decode Base64 inline config
|
||||||
|
let decoded = BASE64_STANDARD
|
||||||
|
.decode(config_b64)
|
||||||
|
.map_err(|e| AppError::InvalidInput(format!("Invalid Base64 encoding: {e}")))?;
|
||||||
|
String::from_utf8(decoded)
|
||||||
|
.map_err(|e| AppError::InvalidInput(format!("Invalid UTF-8 in config: {e}")))?
|
||||||
|
} else if let Some(_config_url) = &request.config_url {
|
||||||
|
// Fetch remote config (TODO: implement remote fetching in next phase)
|
||||||
|
return Err(AppError::InvalidInput(
|
||||||
|
"Remote config URL is not yet supported. Use inline config instead.".to_string(),
|
||||||
|
));
|
||||||
|
} else {
|
||||||
|
return Ok(request.clone());
|
||||||
|
};
|
||||||
|
|
||||||
|
// Step 2: Parse config based on format
|
||||||
|
let format = request.config_format.as_deref().unwrap_or("json");
|
||||||
|
let config_value: serde_json::Value = match format {
|
||||||
|
"json" => serde_json::from_str(&config_content)
|
||||||
|
.map_err(|e| AppError::InvalidInput(format!("Invalid JSON config: {e}")))?,
|
||||||
|
"toml" => {
|
||||||
|
let toml_value: toml::Value = toml::from_str(&config_content)
|
||||||
|
.map_err(|e| AppError::InvalidInput(format!("Invalid TOML config: {e}")))?;
|
||||||
|
// Convert TOML to JSON for uniform processing
|
||||||
|
serde_json::to_value(toml_value)
|
||||||
|
.map_err(|e| AppError::Message(format!("Failed to convert TOML to JSON: {e}")))?
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
return Err(AppError::InvalidInput(format!(
|
||||||
|
"Unsupported config format: {format}"
|
||||||
|
)))
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Step 3: Extract values from config based on app type and merge with URL params
|
||||||
|
let mut merged = request.clone();
|
||||||
|
|
||||||
|
match request.app.as_str() {
|
||||||
|
"claude" => merge_claude_config(&mut merged, &config_value)?,
|
||||||
|
"codex" => merge_codex_config(&mut merged, &config_value)?,
|
||||||
|
"gemini" => merge_gemini_config(&mut merged, &config_value)?,
|
||||||
|
_ => {
|
||||||
|
return Err(AppError::InvalidInput(format!(
|
||||||
|
"Invalid app type: {}",
|
||||||
|
request.app
|
||||||
|
)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(merged)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Merge Claude configuration from config file
|
||||||
|
///
|
||||||
|
/// Priority: URL params override config file values
|
||||||
|
fn merge_claude_config(
|
||||||
|
request: &mut DeepLinkImportRequest,
|
||||||
|
config: &serde_json::Value,
|
||||||
|
) -> Result<(), AppError> {
|
||||||
|
let env = config
|
||||||
|
.get("env")
|
||||||
|
.and_then(|v| v.as_object())
|
||||||
|
.ok_or_else(|| {
|
||||||
|
AppError::InvalidInput("Claude config must have 'env' object".to_string())
|
||||||
|
})?;
|
||||||
|
|
||||||
|
// Auto-fill API key if not provided in URL
|
||||||
|
if request.api_key.is_empty() {
|
||||||
|
if let Some(token) = env.get("ANTHROPIC_AUTH_TOKEN").and_then(|v| v.as_str()) {
|
||||||
|
request.api_key = token.to_string();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Auto-fill endpoint if not provided in URL
|
||||||
|
if request.endpoint.is_empty() {
|
||||||
|
if let Some(base_url) = env.get("ANTHROPIC_BASE_URL").and_then(|v| v.as_str()) {
|
||||||
|
request.endpoint = base_url.to_string();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Auto-fill homepage from endpoint if not provided
|
||||||
|
if request.homepage.is_empty() && !request.endpoint.is_empty() {
|
||||||
|
request.homepage = infer_homepage_from_endpoint(&request.endpoint)
|
||||||
|
.unwrap_or_else(|| "https://anthropic.com".to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Auto-fill model fields (URL params take priority)
|
||||||
|
if request.model.is_none() {
|
||||||
|
request.model = env
|
||||||
|
.get("ANTHROPIC_MODEL")
|
||||||
|
.and_then(|v| v.as_str())
|
||||||
|
.map(|s| s.to_string());
|
||||||
|
}
|
||||||
|
if request.haiku_model.is_none() {
|
||||||
|
request.haiku_model = env
|
||||||
|
.get("ANTHROPIC_DEFAULT_HAIKU_MODEL")
|
||||||
|
.and_then(|v| v.as_str())
|
||||||
|
.map(|s| s.to_string());
|
||||||
|
}
|
||||||
|
if request.sonnet_model.is_none() {
|
||||||
|
request.sonnet_model = env
|
||||||
|
.get("ANTHROPIC_DEFAULT_SONNET_MODEL")
|
||||||
|
.and_then(|v| v.as_str())
|
||||||
|
.map(|s| s.to_string());
|
||||||
|
}
|
||||||
|
if request.opus_model.is_none() {
|
||||||
|
request.opus_model = env
|
||||||
|
.get("ANTHROPIC_DEFAULT_OPUS_MODEL")
|
||||||
|
.and_then(|v| v.as_str())
|
||||||
|
.map(|s| s.to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Merge Codex configuration from config file
|
||||||
|
fn merge_codex_config(
|
||||||
|
request: &mut DeepLinkImportRequest,
|
||||||
|
config: &serde_json::Value,
|
||||||
|
) -> Result<(), AppError> {
|
||||||
|
// Auto-fill API key from auth.OPENAI_API_KEY
|
||||||
|
if request.api_key.is_empty() {
|
||||||
|
if let Some(api_key) = config
|
||||||
|
.get("auth")
|
||||||
|
.and_then(|v| v.get("OPENAI_API_KEY"))
|
||||||
|
.and_then(|v| v.as_str())
|
||||||
|
{
|
||||||
|
request.api_key = api_key.to_string();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Auto-fill endpoint and model from config string
|
||||||
|
if let Some(config_str) = config.get("config").and_then(|v| v.as_str()) {
|
||||||
|
// Parse TOML config string to extract base_url and model
|
||||||
|
if let Ok(toml_value) = toml::from_str::<toml::Value>(config_str) {
|
||||||
|
// Extract base_url from model_providers section
|
||||||
|
if request.endpoint.is_empty() {
|
||||||
|
if let Some(base_url) = extract_codex_base_url(&toml_value) {
|
||||||
|
request.endpoint = base_url;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Extract model
|
||||||
|
if request.model.is_none() {
|
||||||
|
if let Some(model) = toml_value.get("model").and_then(|v| v.as_str()) {
|
||||||
|
request.model = Some(model.to_string());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Auto-fill homepage from endpoint
|
||||||
|
if request.homepage.is_empty() && !request.endpoint.is_empty() {
|
||||||
|
request.homepage = infer_homepage_from_endpoint(&request.endpoint)
|
||||||
|
.unwrap_or_else(|| "https://openai.com".to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Merge Gemini configuration from config file
|
||||||
|
fn merge_gemini_config(
|
||||||
|
request: &mut DeepLinkImportRequest,
|
||||||
|
config: &serde_json::Value,
|
||||||
|
) -> Result<(), AppError> {
|
||||||
|
// Gemini uses flat env structure
|
||||||
|
if request.api_key.is_empty() {
|
||||||
|
if let Some(api_key) = config.get("GEMINI_API_KEY").and_then(|v| v.as_str()) {
|
||||||
|
request.api_key = api_key.to_string();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if request.endpoint.is_empty() {
|
||||||
|
if let Some(base_url) = config.get("GEMINI_BASE_URL").and_then(|v| v.as_str()) {
|
||||||
|
request.endpoint = base_url.to_string();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if request.model.is_none() {
|
||||||
|
request.model = config
|
||||||
|
.get("GEMINI_MODEL")
|
||||||
|
.and_then(|v| v.as_str())
|
||||||
|
.map(|s| s.to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Auto-fill homepage from endpoint
|
||||||
|
if request.homepage.is_empty() && !request.endpoint.is_empty() {
|
||||||
|
request.homepage = infer_homepage_from_endpoint(&request.endpoint)
|
||||||
|
.unwrap_or_else(|| "https://ai.google.dev".to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Extract base_url from Codex TOML config
|
||||||
|
fn extract_codex_base_url(toml_value: &toml::Value) -> Option<String> {
|
||||||
|
// Try to find base_url in model_providers section
|
||||||
|
if let Some(providers) = toml_value.get("model_providers").and_then(|v| v.as_table()) {
|
||||||
|
for (_key, provider) in providers.iter() {
|
||||||
|
if let Some(base_url) = provider.get("base_url").and_then(|v| v.as_str()) {
|
||||||
|
return Some(base_url.to_string());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Infer homepage URL from API endpoint
|
||||||
|
///
|
||||||
|
/// Examples:
|
||||||
|
/// - https://api.anthropic.com/v1 → https://anthropic.com
|
||||||
|
/// - https://api.openai.com/v1 → https://openai.com
|
||||||
|
/// - https://api-test.company.com/v1 → https://company.com
|
||||||
|
fn infer_homepage_from_endpoint(endpoint: &str) -> Option<String> {
|
||||||
|
let url = Url::parse(endpoint).ok()?;
|
||||||
|
let host = url.host_str()?;
|
||||||
|
|
||||||
|
// Remove common API prefixes
|
||||||
|
let clean_host = host
|
||||||
|
.strip_prefix("api.")
|
||||||
|
.or_else(|| host.strip_prefix("api-"))
|
||||||
|
.unwrap_or(host);
|
||||||
|
|
||||||
|
Some(format!("https://{clean_host}"))
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
@@ -375,14 +687,15 @@ mod tests {
|
|||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_parse_missing_required_field() {
|
fn test_parse_missing_required_field() {
|
||||||
let url = "ccswitch://v1/import?resource=provider&app=claude&name=Test";
|
// Name is still required even in v3.8+ (only homepage/endpoint/apiKey are optional)
|
||||||
|
let url = "ccswitch://v1/import?resource=provider&app=claude";
|
||||||
|
|
||||||
let result = parse_deeplink_url(url);
|
let result = parse_deeplink_url(url);
|
||||||
assert!(result.is_err());
|
assert!(result.is_err());
|
||||||
assert!(result
|
assert!(result
|
||||||
.unwrap_err()
|
.unwrap_err()
|
||||||
.to_string()
|
.to_string()
|
||||||
.contains("Missing 'homepage' parameter"));
|
.contains("Missing 'name' parameter"));
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@@ -413,6 +726,12 @@ mod tests {
|
|||||||
api_key: "test-api-key".to_string(),
|
api_key: "test-api-key".to_string(),
|
||||||
model: Some("gemini-2.0-flash".to_string()),
|
model: Some("gemini-2.0-flash".to_string()),
|
||||||
notes: None,
|
notes: None,
|
||||||
|
haiku_model: None,
|
||||||
|
sonnet_model: None,
|
||||||
|
opus_model: None,
|
||||||
|
config: None,
|
||||||
|
config_format: None,
|
||||||
|
config_url: None,
|
||||||
};
|
};
|
||||||
|
|
||||||
let provider = build_provider_from_request(&AppType::Gemini, &request).unwrap();
|
let provider = build_provider_from_request(&AppType::Gemini, &request).unwrap();
|
||||||
@@ -443,6 +762,12 @@ mod tests {
|
|||||||
api_key: "test-api-key".to_string(),
|
api_key: "test-api-key".to_string(),
|
||||||
model: None,
|
model: None,
|
||||||
notes: None,
|
notes: None,
|
||||||
|
haiku_model: None,
|
||||||
|
sonnet_model: None,
|
||||||
|
opus_model: None,
|
||||||
|
config: None,
|
||||||
|
config_format: None,
|
||||||
|
config_url: None,
|
||||||
};
|
};
|
||||||
|
|
||||||
let provider = build_provider_from_request(&AppType::Gemini, &request).unwrap();
|
let provider = build_provider_from_request(&AppType::Gemini, &request).unwrap();
|
||||||
@@ -454,4 +779,88 @@ mod tests {
|
|||||||
// Model should not be present
|
// Model should not be present
|
||||||
assert!(env.get("GEMINI_MODEL").is_none());
|
assert!(env.get("GEMINI_MODEL").is_none());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_infer_homepage() {
|
||||||
|
assert_eq!(
|
||||||
|
infer_homepage_from_endpoint("https://api.anthropic.com/v1"),
|
||||||
|
Some("https://anthropic.com".to_string())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
infer_homepage_from_endpoint("https://api-test.company.com/v1"),
|
||||||
|
Some("https://test.company.com".to_string())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
infer_homepage_from_endpoint("https://example.com"),
|
||||||
|
Some("https://example.com".to_string())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_parse_and_merge_config_claude() {
|
||||||
|
use base64::prelude::*;
|
||||||
|
|
||||||
|
// Prepare Base64 encoded Claude config
|
||||||
|
let config_json = r#"{"env":{"ANTHROPIC_AUTH_TOKEN":"sk-ant-xxx","ANTHROPIC_BASE_URL":"https://api.anthropic.com/v1","ANTHROPIC_MODEL":"claude-sonnet-4.5"}}"#;
|
||||||
|
let config_b64 = BASE64_STANDARD.encode(config_json.as_bytes());
|
||||||
|
|
||||||
|
let request = DeepLinkImportRequest {
|
||||||
|
version: "v1".to_string(),
|
||||||
|
resource: "provider".to_string(),
|
||||||
|
app: "claude".to_string(),
|
||||||
|
name: "Test".to_string(),
|
||||||
|
homepage: String::new(),
|
||||||
|
endpoint: String::new(),
|
||||||
|
api_key: String::new(),
|
||||||
|
model: None,
|
||||||
|
notes: None,
|
||||||
|
haiku_model: None,
|
||||||
|
sonnet_model: None,
|
||||||
|
opus_model: None,
|
||||||
|
config: Some(config_b64),
|
||||||
|
config_format: Some("json".to_string()),
|
||||||
|
config_url: None,
|
||||||
|
};
|
||||||
|
|
||||||
|
let merged = parse_and_merge_config(&request).unwrap();
|
||||||
|
|
||||||
|
// Should auto-fill from config
|
||||||
|
assert_eq!(merged.api_key, "sk-ant-xxx");
|
||||||
|
assert_eq!(merged.endpoint, "https://api.anthropic.com/v1");
|
||||||
|
assert_eq!(merged.homepage, "https://anthropic.com");
|
||||||
|
assert_eq!(merged.model, Some("claude-sonnet-4.5".to_string()));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_parse_and_merge_config_url_override() {
|
||||||
|
use base64::prelude::*;
|
||||||
|
|
||||||
|
let config_json = r#"{"env":{"ANTHROPIC_AUTH_TOKEN":"sk-old","ANTHROPIC_BASE_URL":"https://api.anthropic.com/v1"}}"#;
|
||||||
|
let config_b64 = BASE64_STANDARD.encode(config_json.as_bytes());
|
||||||
|
|
||||||
|
let request = DeepLinkImportRequest {
|
||||||
|
version: "v1".to_string(),
|
||||||
|
resource: "provider".to_string(),
|
||||||
|
app: "claude".to_string(),
|
||||||
|
name: "Test".to_string(),
|
||||||
|
homepage: String::new(),
|
||||||
|
endpoint: String::new(),
|
||||||
|
api_key: "sk-new".to_string(), // URL param should override
|
||||||
|
model: None,
|
||||||
|
notes: None,
|
||||||
|
haiku_model: None,
|
||||||
|
sonnet_model: None,
|
||||||
|
opus_model: None,
|
||||||
|
config: Some(config_b64),
|
||||||
|
config_format: Some("json".to_string()),
|
||||||
|
config_url: None,
|
||||||
|
};
|
||||||
|
|
||||||
|
let merged = parse_and_merge_config(&request).unwrap();
|
||||||
|
|
||||||
|
// URL param should take priority
|
||||||
|
assert_eq!(merged.api_key, "sk-new");
|
||||||
|
// Config file value should be used
|
||||||
|
assert_eq!(merged.endpoint, "https://api.anthropic.com/v1");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -50,6 +50,8 @@ pub enum AppError {
|
|||||||
zh: String,
|
zh: String,
|
||||||
en: String,
|
en: String,
|
||||||
},
|
},
|
||||||
|
#[error("数据库错误: {0}")]
|
||||||
|
Database(String),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AppError {
|
impl AppError {
|
||||||
@@ -94,3 +96,28 @@ impl From<AppError> for String {
|
|||||||
err.to_string()
|
err.to_string()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// 格式化为 JSON 错误字符串,前端可解析为结构化错误
|
||||||
|
pub fn format_skill_error(
|
||||||
|
code: &str,
|
||||||
|
context: &[(&str, &str)],
|
||||||
|
suggestion: Option<&str>,
|
||||||
|
) -> String {
|
||||||
|
use serde_json::json;
|
||||||
|
|
||||||
|
let mut ctx_map = serde_json::Map::new();
|
||||||
|
for (key, value) in context {
|
||||||
|
ctx_map.insert(key.to_string(), json!(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
let error_obj = json!({
|
||||||
|
"code": code,
|
||||||
|
"context": ctx_map,
|
||||||
|
"suggestion": suggestion,
|
||||||
|
});
|
||||||
|
|
||||||
|
serde_json::to_string(&error_obj).unwrap_or_else(|_| {
|
||||||
|
// 如果 JSON 序列化失败,返回简单格式
|
||||||
|
format!("ERROR:{code}")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|||||||
@@ -236,6 +236,17 @@ pub fn validate_gemini_settings(settings: &Value) -> Result<(), AppError> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 如果有 config 字段,验证它是对象或 null
|
||||||
|
if let Some(config) = settings.get("config") {
|
||||||
|
if !(config.is_object() || config.is_null()) {
|
||||||
|
return Err(AppError::localized(
|
||||||
|
"gemini.validation.invalid_config",
|
||||||
|
"Gemini 配置格式错误: config 必须是对象",
|
||||||
|
"Gemini config invalid: config must be an object",
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -244,6 +255,9 @@ pub fn validate_gemini_settings(settings: &Value) -> Result<(), AppError> {
|
|||||||
/// 此函数在切换供应商时使用,确保配置包含所有必需的字段。
|
/// 此函数在切换供应商时使用,确保配置包含所有必需的字段。
|
||||||
/// 对于需要 API Key 的供应商(如 PackyCode),会验证 GEMINI_API_KEY 字段。
|
/// 对于需要 API Key 的供应商(如 PackyCode),会验证 GEMINI_API_KEY 字段。
|
||||||
pub fn validate_gemini_settings_strict(settings: &Value) -> Result<(), AppError> {
|
pub fn validate_gemini_settings_strict(settings: &Value) -> Result<(), AppError> {
|
||||||
|
// 先做基础格式验证(包含 env/config 类型)
|
||||||
|
validate_gemini_settings(settings)?;
|
||||||
|
|
||||||
let env_map = json_to_env(settings)?;
|
let env_map = json_to_env(settings)?;
|
||||||
|
|
||||||
// 如果 env 为空,表示使用 OAuth(如 Google 官方),跳过验证
|
// 如果 env 为空,表示使用 OAuth(如 Google 官方),跳过验证
|
||||||
@@ -368,7 +382,7 @@ mod tests {
|
|||||||
# Comment line
|
# Comment line
|
||||||
GOOGLE_GEMINI_BASE_URL=https://example.com
|
GOOGLE_GEMINI_BASE_URL=https://example.com
|
||||||
GEMINI_API_KEY=sk-test123
|
GEMINI_API_KEY=sk-test123
|
||||||
GEMINI_MODEL=gemini-2.5-pro
|
GEMINI_MODEL=gemini-3-pro-preview
|
||||||
|
|
||||||
# Another comment
|
# Another comment
|
||||||
"#;
|
"#;
|
||||||
@@ -381,19 +395,25 @@ GEMINI_MODEL=gemini-2.5-pro
|
|||||||
Some(&"https://example.com".to_string())
|
Some(&"https://example.com".to_string())
|
||||||
);
|
);
|
||||||
assert_eq!(map.get("GEMINI_API_KEY"), Some(&"sk-test123".to_string()));
|
assert_eq!(map.get("GEMINI_API_KEY"), Some(&"sk-test123".to_string()));
|
||||||
assert_eq!(map.get("GEMINI_MODEL"), Some(&"gemini-2.5-pro".to_string()));
|
assert_eq!(
|
||||||
|
map.get("GEMINI_MODEL"),
|
||||||
|
Some(&"gemini-3-pro-preview".to_string())
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_serialize_env_file() {
|
fn test_serialize_env_file() {
|
||||||
let mut map = HashMap::new();
|
let mut map = HashMap::new();
|
||||||
map.insert("GEMINI_API_KEY".to_string(), "sk-test".to_string());
|
map.insert("GEMINI_API_KEY".to_string(), "sk-test".to_string());
|
||||||
map.insert("GEMINI_MODEL".to_string(), "gemini-2.5-pro".to_string());
|
map.insert(
|
||||||
|
"GEMINI_MODEL".to_string(),
|
||||||
|
"gemini-3-pro-preview".to_string(),
|
||||||
|
);
|
||||||
|
|
||||||
let content = serialize_env_file(&map);
|
let content = serialize_env_file(&map);
|
||||||
|
|
||||||
assert!(content.contains("GEMINI_API_KEY=sk-test"));
|
assert!(content.contains("GEMINI_API_KEY=sk-test"));
|
||||||
assert!(content.contains("GEMINI_MODEL=gemini-2.5-pro"));
|
assert!(content.contains("GEMINI_MODEL=gemini-3-pro-preview"));
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@@ -417,7 +437,7 @@ GEMINI_MODEL=gemini-2.5-pro
|
|||||||
# Comment line
|
# Comment line
|
||||||
GOOGLE_GEMINI_BASE_URL=https://example.com
|
GOOGLE_GEMINI_BASE_URL=https://example.com
|
||||||
GEMINI_API_KEY=sk-test123
|
GEMINI_API_KEY=sk-test123
|
||||||
GEMINI_MODEL=gemini-2.5-pro
|
GEMINI_MODEL=gemini-3-pro-preview
|
||||||
|
|
||||||
# Another comment
|
# Another comment
|
||||||
"#;
|
"#;
|
||||||
@@ -432,7 +452,10 @@ GEMINI_MODEL=gemini-2.5-pro
|
|||||||
Some(&"https://example.com".to_string())
|
Some(&"https://example.com".to_string())
|
||||||
);
|
);
|
||||||
assert_eq!(map.get("GEMINI_API_KEY"), Some(&"sk-test123".to_string()));
|
assert_eq!(map.get("GEMINI_API_KEY"), Some(&"sk-test123".to_string()));
|
||||||
assert_eq!(map.get("GEMINI_MODEL"), Some(&"gemini-2.5-pro".to_string()));
|
assert_eq!(
|
||||||
|
map.get("GEMINI_MODEL"),
|
||||||
|
Some(&"gemini-3-pro-preview".to_string())
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
@@ -598,7 +621,7 @@ KEY_WITH-DASH=value";
|
|||||||
let settings = serde_json::json!({
|
let settings = serde_json::json!({
|
||||||
"env": {
|
"env": {
|
||||||
"GEMINI_API_KEY": "sk-test123",
|
"GEMINI_API_KEY": "sk-test123",
|
||||||
"GEMINI_MODEL": "gemini-2.5-pro"
|
"GEMINI_MODEL": "gemini-3-pro-preview"
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -611,7 +634,7 @@ KEY_WITH-DASH=value";
|
|||||||
// 测试缺少 API Key 的非空配置在基本验证中可以通过(用户稍后填写)
|
// 测试缺少 API Key 的非空配置在基本验证中可以通过(用户稍后填写)
|
||||||
let settings = serde_json::json!({
|
let settings = serde_json::json!({
|
||||||
"env": {
|
"env": {
|
||||||
"GEMINI_MODEL": "gemini-2.5-pro"
|
"GEMINI_MODEL": "gemini-3-pro-preview"
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -96,7 +96,20 @@ pub fn set_mcp_servers_map(
|
|||||||
obj = server_obj;
|
obj = server_obj;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 移除 UI 辅助字段
|
// Gemini CLI 格式转换:
|
||||||
|
// - Gemini 不使用 "type" 字段(从字段名推断传输类型)
|
||||||
|
// - HTTP 使用 "httpUrl" 字段,SSE 使用 "url" 字段
|
||||||
|
let transport_type = obj.get("type").and_then(|v| v.as_str());
|
||||||
|
if transport_type == Some("http") {
|
||||||
|
// HTTP streaming: 将 "url" 重命名为 "httpUrl"
|
||||||
|
if let Some(url_value) = obj.remove("url") {
|
||||||
|
obj.insert("httpUrl".to_string(), url_value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// SSE 保持 "url" 字段不变
|
||||||
|
|
||||||
|
// 移除 UI 辅助字段和 type 字段(Gemini 不需要)
|
||||||
|
obj.remove("type");
|
||||||
obj.remove("enabled");
|
obj.remove("enabled");
|
||||||
obj.remove("source");
|
obj.remove("source");
|
||||||
obj.remove("id");
|
obj.remove("id");
|
||||||
|
|||||||
@@ -13,7 +13,9 @@ fn cell() -> &'static RwLock<Option<InitErrorPayload>> {
|
|||||||
INIT_ERROR.get_or_init(|| RwLock::new(None))
|
INIT_ERROR.get_or_init(|| RwLock::new(None))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
pub fn set_init_error(payload: InitErrorPayload) {
|
pub fn set_init_error(payload: InitErrorPayload) {
|
||||||
|
#[allow(clippy::unwrap_used)]
|
||||||
if let Ok(mut guard) = cell().write() {
|
if let Ok(mut guard) = cell().write() {
|
||||||
*guard = Some(payload);
|
*guard = Some(payload);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,12 @@
|
|||||||
mod app_config;
|
mod app_config;
|
||||||
mod app_store;
|
mod app_store;
|
||||||
|
mod auto_launch;
|
||||||
mod claude_mcp;
|
mod claude_mcp;
|
||||||
mod claude_plugin;
|
mod claude_plugin;
|
||||||
mod codex_config;
|
mod codex_config;
|
||||||
mod commands;
|
mod commands;
|
||||||
mod config;
|
mod config;
|
||||||
|
mod database;
|
||||||
mod deeplink;
|
mod deeplink;
|
||||||
mod error;
|
mod error;
|
||||||
mod gemini_config; // 新增
|
mod gemini_config; // 新增
|
||||||
@@ -14,6 +16,7 @@ mod mcp;
|
|||||||
mod prompt;
|
mod prompt;
|
||||||
mod prompt_files;
|
mod prompt_files;
|
||||||
mod provider;
|
mod provider;
|
||||||
|
mod provider_defaults;
|
||||||
mod services;
|
mod services;
|
||||||
mod settings;
|
mod settings;
|
||||||
mod store;
|
mod store;
|
||||||
@@ -204,8 +207,6 @@ fn create_tray_menu(
|
|||||||
let app_settings = crate::settings::get_settings();
|
let app_settings = crate::settings::get_settings();
|
||||||
let tray_texts = TrayTexts::from_language(app_settings.language.as_deref().unwrap_or("zh"));
|
let tray_texts = TrayTexts::from_language(app_settings.language.as_deref().unwrap_or("zh"));
|
||||||
|
|
||||||
let config = app_state.config.read().map_err(AppError::from)?;
|
|
||||||
|
|
||||||
let mut menu_builder = MenuBuilder::new(app);
|
let mut menu_builder = MenuBuilder::new(app);
|
||||||
|
|
||||||
// 顶部:打开主界面
|
// 顶部:打开主界面
|
||||||
@@ -216,13 +217,20 @@ fn create_tray_menu(
|
|||||||
|
|
||||||
// 直接添加所有供应商到主菜单(扁平化结构,更简单可靠)
|
// 直接添加所有供应商到主菜单(扁平化结构,更简单可靠)
|
||||||
for section in TRAY_SECTIONS.iter() {
|
for section in TRAY_SECTIONS.iter() {
|
||||||
menu_builder = append_provider_section(
|
let app_type_str = section.app_type.as_str();
|
||||||
app,
|
let providers = app_state.db.get_all_providers(app_type_str)?;
|
||||||
menu_builder,
|
let current_id = app_state
|
||||||
config.get_manager(§ion.app_type),
|
.db
|
||||||
section,
|
.get_current_provider(app_type_str)?
|
||||||
&tray_texts,
|
.unwrap_or_default();
|
||||||
)?;
|
|
||||||
|
let manager = crate::provider::ProviderManager {
|
||||||
|
providers,
|
||||||
|
current: current_id,
|
||||||
|
};
|
||||||
|
|
||||||
|
menu_builder =
|
||||||
|
append_provider_section(app, menu_builder, Some(&manager), section, &tray_texts)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 分隔符和退出菜单
|
// 分隔符和退出菜单
|
||||||
@@ -521,42 +529,47 @@ pub fn run() {
|
|||||||
// 预先刷新 Store 覆盖配置,确保 AppState 初始化时可读取到最新路径
|
// 预先刷新 Store 覆盖配置,确保 AppState 初始化时可读取到最新路径
|
||||||
app_store::refresh_app_config_dir_override(app.handle());
|
app_store::refresh_app_config_dir_override(app.handle());
|
||||||
|
|
||||||
// 初始化应用状态(仅创建一次,并在本函数末尾注入 manage)
|
// 初始化数据库
|
||||||
// 如果配置解析失败,则向前端发送错误事件并提前结束 setup(不落盘、不覆盖配置)。
|
let app_config_dir = crate::config::get_app_config_dir();
|
||||||
let app_state = match AppState::try_new() {
|
let db_path = app_config_dir.join("cc-switch.db");
|
||||||
Ok(state) => state,
|
let json_path = app_config_dir.join("config.json");
|
||||||
Err(err) => {
|
|
||||||
let path = crate::config::get_app_config_path();
|
// Check if migration is needed (DB doesn't exist but JSON does)
|
||||||
let payload_json = serde_json::json!({
|
let migration_needed = !db_path.exists() && json_path.exists();
|
||||||
"path": path.display().to_string(),
|
|
||||||
"error": err.to_string(),
|
let db = match crate::database::Database::init() {
|
||||||
});
|
Ok(db) => Arc::new(db),
|
||||||
// 事件通知(可能早于前端订阅,不保证送达)
|
Err(e) => {
|
||||||
if let Err(e) = app.emit("configLoadError", payload_json) {
|
log::error!("Failed to init database: {e}");
|
||||||
log::error!("发射配置加载错误事件失败: {e}");
|
// 这里的错误处理比较棘手,因为 setup 返回 Result<Box<dyn Error>>
|
||||||
}
|
// 我们暂时记录日志并让应用继续运行(可能会崩溃)或者返回错误
|
||||||
// 同时缓存错误,供前端启动阶段主动拉取
|
return Err(Box::new(e));
|
||||||
crate::init_status::set_init_error(crate::init_status::InitErrorPayload {
|
|
||||||
path: path.display().to_string(),
|
|
||||||
error: err.to_string(),
|
|
||||||
});
|
|
||||||
// 不再继续构建托盘/命令依赖的状态,交由前端提示后退出。
|
|
||||||
return Ok(());
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if migration_needed {
|
||||||
|
log::info!("Starting migration from config.json to SQLite...");
|
||||||
|
match crate::app_config::MultiAppConfig::load() {
|
||||||
|
Ok(config) => {
|
||||||
|
if let Err(e) = db.migrate_from_json(&config) {
|
||||||
|
log::error!("Migration failed: {e}");
|
||||||
|
} else {
|
||||||
|
log::info!("Migration successful");
|
||||||
|
// Optional: Rename config.json
|
||||||
|
// let _ = std::fs::rename(&json_path, json_path.with_extension("json.bak"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(e) => log::error!("Failed to load config.json for migration: {e}"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let app_state = AppState::new(db);
|
||||||
|
|
||||||
// 迁移旧的 app_config_dir 配置到 Store
|
// 迁移旧的 app_config_dir 配置到 Store
|
||||||
if let Err(e) = app_store::migrate_app_config_dir_from_settings(app.handle()) {
|
if let Err(e) = app_store::migrate_app_config_dir_from_settings(app.handle()) {
|
||||||
log::warn!("迁移 app_config_dir 失败: {e}");
|
log::warn!("迁移 app_config_dir 失败: {e}");
|
||||||
}
|
}
|
||||||
|
|
||||||
// 确保配置结构就绪(已移除旧版本的副本迁移逻辑)
|
|
||||||
{
|
|
||||||
let mut config_guard = app_state.config.write().unwrap();
|
|
||||||
config_guard.ensure_app(&app_config::AppType::Claude);
|
|
||||||
config_guard.ensure_app(&app_config::AppType::Codex);
|
|
||||||
}
|
|
||||||
|
|
||||||
// 启动阶段不再无条件保存,避免意外覆盖用户配置。
|
// 启动阶段不再无条件保存,避免意外覆盖用户配置。
|
||||||
|
|
||||||
// 注册 deep-link URL 处理器(使用正确的 DeepLinkExt API)
|
// 注册 deep-link URL 处理器(使用正确的 DeepLinkExt API)
|
||||||
@@ -704,6 +717,7 @@ pub fn run() {
|
|||||||
commands::sync_current_providers_live,
|
commands::sync_current_providers_live,
|
||||||
// Deep link import
|
// Deep link import
|
||||||
commands::parse_deeplink,
|
commands::parse_deeplink,
|
||||||
|
commands::merge_deeplink_config,
|
||||||
commands::import_from_deeplink,
|
commands::import_from_deeplink,
|
||||||
update_tray_menu,
|
update_tray_menu,
|
||||||
// Environment variable management
|
// Environment variable management
|
||||||
@@ -717,6 +731,9 @@ pub fn run() {
|
|||||||
commands::get_skill_repos,
|
commands::get_skill_repos,
|
||||||
commands::add_skill_repo,
|
commands::add_skill_repo,
|
||||||
commands::remove_skill_repo,
|
commands::remove_skill_repo,
|
||||||
|
// Auto launch
|
||||||
|
commands::set_auto_launch,
|
||||||
|
commands::get_auto_launch_status,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
let app = builder
|
let app = builder
|
||||||
|
|||||||
@@ -449,7 +449,7 @@ pub fn import_from_codex(config: &mut MultiAppConfig) -> Result<usize, AppError>
|
|||||||
// 核心字段(需要手动处理的字段)
|
// 核心字段(需要手动处理的字段)
|
||||||
let core_fields = match typ {
|
let core_fields = match typ {
|
||||||
"stdio" => vec!["type", "command", "args", "env", "cwd"],
|
"stdio" => vec!["type", "command", "args", "env", "cwd"],
|
||||||
"http" | "sse" => vec!["type", "url", "headers"],
|
"http" | "sse" => vec!["type", "url", "http_headers"],
|
||||||
_ => vec!["type"],
|
_ => vec!["type"],
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -490,7 +490,13 @@ pub fn import_from_codex(config: &mut MultiAppConfig) -> Result<usize, AppError>
|
|||||||
if let Some(url) = entry_tbl.get("url").and_then(|v| v.as_str()) {
|
if let Some(url) = entry_tbl.get("url").and_then(|v| v.as_str()) {
|
||||||
spec.insert("url".into(), json!(url));
|
spec.insert("url".into(), json!(url));
|
||||||
}
|
}
|
||||||
if let Some(headers_tbl) = entry_tbl.get("headers").and_then(|v| v.as_table()) {
|
// Read from http_headers (correct Codex format) or headers (legacy) with priority to http_headers
|
||||||
|
let headers_tbl = entry_tbl
|
||||||
|
.get("http_headers")
|
||||||
|
.and_then(|v| v.as_table())
|
||||||
|
.or_else(|| entry_tbl.get("headers").and_then(|v| v.as_table()));
|
||||||
|
|
||||||
|
if let Some(headers_tbl) = headers_tbl {
|
||||||
let mut headers_json = serde_json::Map::new();
|
let mut headers_json = serde_json::Map::new();
|
||||||
for (k, v) in headers_tbl.iter() {
|
for (k, v) in headers_tbl.iter() {
|
||||||
if let Some(sv) = v.as_str() {
|
if let Some(sv) = v.as_str() {
|
||||||
@@ -910,7 +916,7 @@ fn json_server_to_toml_table(spec: &Value) -> Result<toml_edit::Table, AppError>
|
|||||||
// 定义核心字段(已在下方处理,跳过通用转换)
|
// 定义核心字段(已在下方处理,跳过通用转换)
|
||||||
let core_fields = match typ {
|
let core_fields = match typ {
|
||||||
"stdio" => vec!["type", "command", "args", "env", "cwd"],
|
"stdio" => vec!["type", "command", "args", "env", "cwd"],
|
||||||
"http" | "sse" => vec!["type", "url", "headers"],
|
"http" | "sse" => vec!["type", "url", "http_headers"],
|
||||||
_ => vec!["type"],
|
_ => vec!["type"],
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -988,7 +994,7 @@ fn json_server_to_toml_table(spec: &Value) -> Result<toml_edit::Table, AppError>
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if !h_tbl.is_empty() {
|
if !h_tbl.is_empty() {
|
||||||
t["headers"] = Item::Table(h_tbl);
|
t["http_headers"] = Item::Table(h_tbl);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
use indexmap::IndexMap;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use serde_json::Value;
|
use serde_json::Value;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
@@ -28,6 +29,13 @@ pub struct Provider {
|
|||||||
/// 供应商元数据(不写入 live 配置,仅存于 ~/.cc-switch/config.json)
|
/// 供应商元数据(不写入 live 配置,仅存于 ~/.cc-switch/config.json)
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
pub meta: Option<ProviderMeta>,
|
pub meta: Option<ProviderMeta>,
|
||||||
|
/// 图标名称(如 "openai", "anthropic")
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub icon: Option<String>,
|
||||||
|
/// 图标颜色(Hex 格式,如 "#00A67E")
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
#[serde(rename = "iconColor")]
|
||||||
|
pub icon_color: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Provider {
|
impl Provider {
|
||||||
@@ -48,6 +56,8 @@ impl Provider {
|
|||||||
sort_index: None,
|
sort_index: None,
|
||||||
notes: None,
|
notes: None,
|
||||||
meta: None,
|
meta: None,
|
||||||
|
icon: None,
|
||||||
|
icon_color: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -55,7 +65,7 @@ impl Provider {
|
|||||||
/// 供应商管理器
|
/// 供应商管理器
|
||||||
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
|
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
|
||||||
pub struct ProviderManager {
|
pub struct ProviderManager {
|
||||||
pub providers: HashMap<String, Provider>,
|
pub providers: IndexMap<String, Provider>,
|
||||||
pub current: String,
|
pub current: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -145,7 +155,7 @@ pub struct ProviderMeta {
|
|||||||
|
|
||||||
impl ProviderManager {
|
impl ProviderManager {
|
||||||
/// 获取所有供应商
|
/// 获取所有供应商
|
||||||
pub fn get_all_providers(&self) -> &HashMap<String, Provider> {
|
pub fn get_all_providers(&self) -> &IndexMap<String, Provider> {
|
||||||
&self.providers
|
&self.providers
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
238
src-tauri/src/provider_defaults.rs
Normal file
238
src-tauri/src/provider_defaults.rs
Normal file
@@ -0,0 +1,238 @@
|
|||||||
|
use once_cell::sync::Lazy;
|
||||||
|
use std::collections::HashMap;
|
||||||
|
|
||||||
|
/// 供应商图标信息
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub struct ProviderIcon {
|
||||||
|
pub name: &'static str,
|
||||||
|
pub color: &'static str,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// 供应商名称到图标的默认映射
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub static DEFAULT_PROVIDER_ICONS: Lazy<HashMap<&'static str, ProviderIcon>> = Lazy::new(|| {
|
||||||
|
let mut m = HashMap::new();
|
||||||
|
|
||||||
|
// AI 服务商
|
||||||
|
m.insert(
|
||||||
|
"openai",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "openai",
|
||||||
|
color: "#00A67E",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"anthropic",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "anthropic",
|
||||||
|
color: "#D4915D",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"claude",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "claude",
|
||||||
|
color: "#D4915D",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"google",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "google",
|
||||||
|
color: "#4285F4",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"gemini",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "gemini",
|
||||||
|
color: "#4285F4",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"deepseek",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "deepseek",
|
||||||
|
color: "#1E88E5",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"kimi",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "kimi",
|
||||||
|
color: "#6366F1",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"moonshot",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "moonshot",
|
||||||
|
color: "#6366F1",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"zhipu",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "zhipu",
|
||||||
|
color: "#0F62FE",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"minimax",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "minimax",
|
||||||
|
color: "#FF6B6B",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"baidu",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "baidu",
|
||||||
|
color: "#2932E1",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"alibaba",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "alibaba",
|
||||||
|
color: "#FF6A00",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"tencent",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "tencent",
|
||||||
|
color: "#00A4FF",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"meta",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "meta",
|
||||||
|
color: "#0081FB",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"microsoft",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "microsoft",
|
||||||
|
color: "#00A4EF",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"cohere",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "cohere",
|
||||||
|
color: "#39594D",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"perplexity",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "perplexity",
|
||||||
|
color: "#20808D",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"mistral",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "mistral",
|
||||||
|
color: "#FF7000",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"huggingface",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "huggingface",
|
||||||
|
color: "#FFD21E",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
// 云平台
|
||||||
|
m.insert(
|
||||||
|
"aws",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "aws",
|
||||||
|
color: "#FF9900",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"azure",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "azure",
|
||||||
|
color: "#0078D4",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"huawei",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "huawei",
|
||||||
|
color: "#FF0000",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
m.insert(
|
||||||
|
"cloudflare",
|
||||||
|
ProviderIcon {
|
||||||
|
name: "cloudflare",
|
||||||
|
color: "#F38020",
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
m
|
||||||
|
});
|
||||||
|
|
||||||
|
/// 根据供应商名称智能推断图标
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub fn infer_provider_icon(provider_name: &str) -> Option<ProviderIcon> {
|
||||||
|
let name_lower = provider_name.to_lowercase();
|
||||||
|
|
||||||
|
// 精确匹配
|
||||||
|
if let Some(icon) = DEFAULT_PROVIDER_ICONS.get(name_lower.as_str()) {
|
||||||
|
return Some(icon.clone());
|
||||||
|
}
|
||||||
|
|
||||||
|
// 模糊匹配(包含关键词)
|
||||||
|
for (key, icon) in DEFAULT_PROVIDER_ICONS.iter() {
|
||||||
|
if name_lower.contains(key) {
|
||||||
|
return Some(icon.clone());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_exact_match() {
|
||||||
|
let icon = infer_provider_icon("openai");
|
||||||
|
assert!(icon.is_some());
|
||||||
|
let icon = icon.unwrap();
|
||||||
|
assert_eq!(icon.name, "openai");
|
||||||
|
assert_eq!(icon.color, "#00A67E");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_fuzzy_match() {
|
||||||
|
let icon = infer_provider_icon("OpenAI Official");
|
||||||
|
assert!(icon.is_some());
|
||||||
|
let icon = icon.unwrap();
|
||||||
|
assert_eq!(icon.name, "openai");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_case_insensitive() {
|
||||||
|
let icon = infer_provider_icon("ANTHROPIC");
|
||||||
|
assert!(icon.is_some());
|
||||||
|
assert_eq!(icon.unwrap().name, "anthropic");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_no_match() {
|
||||||
|
let icon = infer_provider_icon("unknown provider");
|
||||||
|
assert!(icon.is_none());
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -109,7 +109,17 @@ impl ConfigService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// 将外部配置文件内容加载并写入应用状态。
|
/// 将外部配置文件内容加载并写入应用状态。
|
||||||
pub fn import_config_from_path(file_path: &Path, state: &AppState) -> Result<String, AppError> {
|
/// TODO: 需要重构以使用数据库而不是 JSON 配置
|
||||||
|
pub fn import_config_from_path(
|
||||||
|
_file_path: &Path,
|
||||||
|
_state: &AppState,
|
||||||
|
) -> Result<String, AppError> {
|
||||||
|
// TODO: 实现基于数据库的导入逻辑
|
||||||
|
Err(AppError::Message(
|
||||||
|
"配置导入功能正在重构中,暂时不可用".to_string(),
|
||||||
|
))
|
||||||
|
|
||||||
|
/* 旧的实现,需要重构:
|
||||||
let (new_config, backup_id) = Self::load_config_for_import(file_path)?;
|
let (new_config, backup_id) = Self::load_config_for_import(file_path)?;
|
||||||
|
|
||||||
{
|
{
|
||||||
@@ -118,6 +128,7 @@ impl ConfigService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Ok(backup_id)
|
Ok(backup_id)
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 同步当前供应商到对应的 live 配置。
|
/// 同步当前供应商到对应的 live 配置。
|
||||||
@@ -229,42 +240,22 @@ impl ConfigService {
|
|||||||
provider_id: &str,
|
provider_id: &str,
|
||||||
provider: &Provider,
|
provider: &Provider,
|
||||||
) -> Result<(), AppError> {
|
) -> Result<(), AppError> {
|
||||||
use crate::gemini_config::{
|
use crate::gemini_config::{env_to_json, read_gemini_env};
|
||||||
env_to_json, json_to_env, read_gemini_env, write_gemini_env_atomic,
|
|
||||||
};
|
|
||||||
|
|
||||||
let env_path = crate::gemini_config::get_gemini_env_path();
|
ProviderService::write_gemini_live(provider)?;
|
||||||
if let Some(parent) = env_path.parent() {
|
|
||||||
fs::create_dir_all(parent).map_err(|e| AppError::io(parent, e))?;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 转换 JSON 配置为 .env 格式
|
// 读回实际写入的内容并更新到配置中(包含 settings.json)
|
||||||
let env_map = json_to_env(&provider.settings_config)?;
|
|
||||||
|
|
||||||
// Google 官方(OAuth): env 为空,写入空文件并设置安全标志后返回
|
|
||||||
if env_map.is_empty() {
|
|
||||||
write_gemini_env_atomic(&env_map)?;
|
|
||||||
ProviderService::ensure_google_oauth_security_flag(provider)?;
|
|
||||||
|
|
||||||
let live_after_env = read_gemini_env()?;
|
|
||||||
let live_after = env_to_json(&live_after_env);
|
|
||||||
|
|
||||||
if let Some(manager) = config.get_manager_mut(&AppType::Gemini) {
|
|
||||||
if let Some(target) = manager.providers.get_mut(provider_id) {
|
|
||||||
target.settings_config = live_after;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
// 非 OAuth:按常规写入,并在必要时设置 Packycode 安全标志
|
|
||||||
write_gemini_env_atomic(&env_map)?;
|
|
||||||
ProviderService::ensure_packycode_security_flag(provider)?;
|
|
||||||
|
|
||||||
// 读回实际写入的内容并更新到配置中
|
|
||||||
let live_after_env = read_gemini_env()?;
|
let live_after_env = read_gemini_env()?;
|
||||||
let live_after = env_to_json(&live_after_env);
|
let settings_path = crate::gemini_config::get_gemini_settings_path();
|
||||||
|
let live_after_config = if settings_path.exists() {
|
||||||
|
crate::config::read_json_file(&settings_path)?
|
||||||
|
} else {
|
||||||
|
serde_json::json!({})
|
||||||
|
};
|
||||||
|
let mut live_after = env_to_json(&live_after_env);
|
||||||
|
if let Some(obj) = live_after.as_object_mut() {
|
||||||
|
obj.insert("config".to_string(), live_after_config);
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(manager) = config.get_manager_mut(&AppType::Gemini) {
|
if let Some(manager) = config.get_manager_mut(&AppType::Gemini) {
|
||||||
if let Some(target) = manager.providers.get_mut(provider_id) {
|
if let Some(target) = manager.providers.get_mut(provider_id) {
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
#[cfg(not(target_os = "windows"))]
|
||||||
use std::fs;
|
use std::fs;
|
||||||
|
|
||||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||||
@@ -35,6 +36,7 @@ fn get_keywords_for_app(app: &str) -> Vec<&str> {
|
|||||||
match app.to_lowercase().as_str() {
|
match app.to_lowercase().as_str() {
|
||||||
"claude" => vec!["ANTHROPIC"],
|
"claude" => vec!["ANTHROPIC"],
|
||||||
"codex" => vec!["OPENAI"],
|
"codex" => vec!["OPENAI"],
|
||||||
|
"gemini" => vec!["GEMINI", "GOOGLE_GEMINI"],
|
||||||
_ => vec![],
|
_ => vec![],
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -48,14 +50,12 @@ fn check_system_env(keywords: &[&str]) -> Result<Vec<EnvConflict>, String> {
|
|||||||
if let Ok(hkcu) = RegKey::predef(HKEY_CURRENT_USER).open_subkey("Environment") {
|
if let Ok(hkcu) = RegKey::predef(HKEY_CURRENT_USER).open_subkey("Environment") {
|
||||||
for (name, value) in hkcu.enum_values().filter_map(Result::ok) {
|
for (name, value) in hkcu.enum_values().filter_map(Result::ok) {
|
||||||
if keywords.iter().any(|k| name.to_uppercase().contains(k)) {
|
if keywords.iter().any(|k| name.to_uppercase().contains(k)) {
|
||||||
if let Ok(val) = value.to_string() {
|
conflicts.push(EnvConflict {
|
||||||
conflicts.push(EnvConflict {
|
var_name: name.clone(),
|
||||||
var_name: name.clone(),
|
var_value: value.to_string(),
|
||||||
var_value: val,
|
source_type: "system".to_string(),
|
||||||
source_type: "system".to_string(),
|
source_path: "HKEY_CURRENT_USER\\Environment".to_string(),
|
||||||
source_path: "HKEY_CURRENT_USER\\Environment".to_string(),
|
});
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -66,14 +66,12 @@ fn check_system_env(keywords: &[&str]) -> Result<Vec<EnvConflict>, String> {
|
|||||||
{
|
{
|
||||||
for (name, value) in hklm.enum_values().filter_map(Result::ok) {
|
for (name, value) in hklm.enum_values().filter_map(Result::ok) {
|
||||||
if keywords.iter().any(|k| name.to_uppercase().contains(k)) {
|
if keywords.iter().any(|k| name.to_uppercase().contains(k)) {
|
||||||
if let Ok(val) = value.to_string() {
|
conflicts.push(EnvConflict {
|
||||||
conflicts.push(EnvConflict {
|
var_name: name.clone(),
|
||||||
var_name: name.clone(),
|
var_value: value.to_string(),
|
||||||
var_value: val,
|
source_type: "system".to_string(),
|
||||||
source_type: "system".to_string(),
|
source_path: "HKEY_LOCAL_MACHINE\\SYSTEM\\CurrentControlSet\\Control\\Session Manager\\Environment".to_string(),
|
||||||
source_path: "HKEY_LOCAL_MACHINE\\SYSTEM\\CurrentControlSet\\Control\\Session Manager\\Environment".to_string(),
|
});
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -161,6 +159,10 @@ mod tests {
|
|||||||
fn test_get_keywords() {
|
fn test_get_keywords() {
|
||||||
assert_eq!(get_keywords_for_app("claude"), vec!["ANTHROPIC"]);
|
assert_eq!(get_keywords_for_app("claude"), vec!["ANTHROPIC"]);
|
||||||
assert_eq!(get_keywords_for_app("codex"), vec!["OPENAI"]);
|
assert_eq!(get_keywords_for_app("codex"), vec!["OPENAI"]);
|
||||||
|
assert_eq!(
|
||||||
|
get_keywords_for_app("gemini"),
|
||||||
|
vec!["GEMINI", "GOOGLE_GEMINI"]
|
||||||
|
);
|
||||||
assert_eq!(get_keywords_for_app("unknown"), Vec::<&str>::new());
|
assert_eq!(get_keywords_for_app("unknown"), Vec::<&str>::new());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
|
use indexmap::IndexMap;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
|
|
||||||
use crate::app_config::{AppType, McpServer, MultiAppConfig};
|
use crate::app_config::{AppType, McpServer};
|
||||||
use crate::error::AppError;
|
use crate::error::AppError;
|
||||||
use crate::mcp;
|
use crate::mcp;
|
||||||
use crate::store::AppState;
|
use crate::store::AppState;
|
||||||
@@ -10,40 +11,13 @@ pub struct McpService;
|
|||||||
|
|
||||||
impl McpService {
|
impl McpService {
|
||||||
/// 获取所有 MCP 服务器(统一结构)
|
/// 获取所有 MCP 服务器(统一结构)
|
||||||
pub fn get_all_servers(state: &AppState) -> Result<HashMap<String, McpServer>, AppError> {
|
pub fn get_all_servers(state: &AppState) -> Result<IndexMap<String, McpServer>, AppError> {
|
||||||
let cfg = state.config.read()?;
|
state.db.get_all_mcp_servers()
|
||||||
|
|
||||||
// 如果是新结构,直接返回
|
|
||||||
if let Some(servers) = &cfg.mcp.servers {
|
|
||||||
return Ok(servers.clone());
|
|
||||||
}
|
|
||||||
|
|
||||||
// 理论上不应该走到这里,因为 load 时会自动迁移
|
|
||||||
Err(AppError::localized(
|
|
||||||
"mcp.old_structure",
|
|
||||||
"检测到旧版 MCP 结构,请重启应用完成迁移",
|
|
||||||
"Old MCP structure detected, please restart app to complete migration",
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 添加或更新 MCP 服务器
|
/// 添加或更新 MCP 服务器
|
||||||
pub fn upsert_server(state: &AppState, server: McpServer) -> Result<(), AppError> {
|
pub fn upsert_server(state: &AppState, server: McpServer) -> Result<(), AppError> {
|
||||||
{
|
state.db.save_mcp_server(&server)?;
|
||||||
let mut cfg = state.config.write()?;
|
|
||||||
|
|
||||||
// 确保 servers 字段存在
|
|
||||||
if cfg.mcp.servers.is_none() {
|
|
||||||
cfg.mcp.servers = Some(HashMap::new());
|
|
||||||
}
|
|
||||||
|
|
||||||
let servers = cfg.mcp.servers.as_mut().unwrap();
|
|
||||||
let id = server.id.clone();
|
|
||||||
|
|
||||||
// 插入或更新
|
|
||||||
servers.insert(id, server.clone());
|
|
||||||
}
|
|
||||||
|
|
||||||
state.save()?;
|
|
||||||
|
|
||||||
// 同步到各个启用的应用
|
// 同步到各个启用的应用
|
||||||
Self::sync_server_to_apps(state, &server)?;
|
Self::sync_server_to_apps(state, &server)?;
|
||||||
@@ -53,18 +27,10 @@ impl McpService {
|
|||||||
|
|
||||||
/// 删除 MCP 服务器
|
/// 删除 MCP 服务器
|
||||||
pub fn delete_server(state: &AppState, id: &str) -> Result<bool, AppError> {
|
pub fn delete_server(state: &AppState, id: &str) -> Result<bool, AppError> {
|
||||||
let server = {
|
let server = state.db.get_all_mcp_servers()?.shift_remove(id);
|
||||||
let mut cfg = state.config.write()?;
|
|
||||||
|
|
||||||
if let Some(servers) = &mut cfg.mcp.servers {
|
|
||||||
servers.remove(id)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Some(server) = server {
|
if let Some(server) = server {
|
||||||
state.save()?;
|
state.db.delete_mcp_server(id)?;
|
||||||
|
|
||||||
// 从所有应用的 live 配置中移除
|
// 从所有应用的 live 配置中移除
|
||||||
Self::remove_server_from_all_apps(state, id, &server)?;
|
Self::remove_server_from_all_apps(state, id, &server)?;
|
||||||
@@ -81,27 +47,15 @@ impl McpService {
|
|||||||
app: AppType,
|
app: AppType,
|
||||||
enabled: bool,
|
enabled: bool,
|
||||||
) -> Result<(), AppError> {
|
) -> Result<(), AppError> {
|
||||||
let server = {
|
let mut servers = state.db.get_all_mcp_servers()?;
|
||||||
let mut cfg = state.config.write()?;
|
|
||||||
|
|
||||||
if let Some(servers) = &mut cfg.mcp.servers {
|
if let Some(server) = servers.get_mut(server_id) {
|
||||||
if let Some(server) = servers.get_mut(server_id) {
|
server.apps.set_enabled_for(&app, enabled);
|
||||||
server.apps.set_enabled_for(&app, enabled);
|
state.db.save_mcp_server(server)?;
|
||||||
Some(server.clone())
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Some(server) = server {
|
|
||||||
state.save()?;
|
|
||||||
|
|
||||||
// 同步到对应应用
|
// 同步到对应应用
|
||||||
if enabled {
|
if enabled {
|
||||||
Self::sync_server_to_app(state, &server, &app)?;
|
Self::sync_server_to_app(state, server, &app)?;
|
||||||
} else {
|
} else {
|
||||||
Self::remove_server_from_app(state, server_id, &app)?;
|
Self::remove_server_from_app(state, server_id, &app)?;
|
||||||
}
|
}
|
||||||
@@ -111,11 +65,9 @@ impl McpService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// 将 MCP 服务器同步到所有启用的应用
|
/// 将 MCP 服务器同步到所有启用的应用
|
||||||
fn sync_server_to_apps(state: &AppState, server: &McpServer) -> Result<(), AppError> {
|
fn sync_server_to_apps(_state: &AppState, server: &McpServer) -> Result<(), AppError> {
|
||||||
let cfg = state.config.read()?;
|
|
||||||
|
|
||||||
for app in server.apps.enabled_apps() {
|
for app in server.apps.enabled_apps() {
|
||||||
Self::sync_server_to_app_internal(&cfg, server, &app)?;
|
Self::sync_server_to_app_no_config(server, &app)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@@ -123,28 +75,24 @@ impl McpService {
|
|||||||
|
|
||||||
/// 将 MCP 服务器同步到指定应用
|
/// 将 MCP 服务器同步到指定应用
|
||||||
fn sync_server_to_app(
|
fn sync_server_to_app(
|
||||||
state: &AppState,
|
_state: &AppState,
|
||||||
server: &McpServer,
|
server: &McpServer,
|
||||||
app: &AppType,
|
app: &AppType,
|
||||||
) -> Result<(), AppError> {
|
) -> Result<(), AppError> {
|
||||||
let cfg = state.config.read()?;
|
Self::sync_server_to_app_no_config(server, app)
|
||||||
Self::sync_server_to_app_internal(&cfg, server, app)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn sync_server_to_app_internal(
|
fn sync_server_to_app_no_config(server: &McpServer, app: &AppType) -> Result<(), AppError> {
|
||||||
cfg: &MultiAppConfig,
|
|
||||||
server: &McpServer,
|
|
||||||
app: &AppType,
|
|
||||||
) -> Result<(), AppError> {
|
|
||||||
match app {
|
match app {
|
||||||
AppType::Claude => {
|
AppType::Claude => {
|
||||||
mcp::sync_single_server_to_claude(cfg, &server.id, &server.server)?;
|
mcp::sync_single_server_to_claude(&Default::default(), &server.id, &server.server)?;
|
||||||
}
|
}
|
||||||
AppType::Codex => {
|
AppType::Codex => {
|
||||||
mcp::sync_single_server_to_codex(cfg, &server.id, &server.server)?;
|
// Codex uses TOML format, must use the correct function
|
||||||
|
mcp::sync_single_server_to_codex(&Default::default(), &server.id, &server.server)?;
|
||||||
}
|
}
|
||||||
AppType::Gemini => {
|
AppType::Gemini => {
|
||||||
mcp::sync_single_server_to_gemini(cfg, &server.id, &server.server)?;
|
mcp::sync_single_server_to_gemini(&Default::default(), &server.id, &server.server)?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
@@ -232,29 +180,21 @@ impl McpService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// 从 Claude 导入 MCP(v3.7.0 已更新为统一结构)
|
/// 从 Claude 导入 MCP(v3.7.0 已更新为统一结构)
|
||||||
pub fn import_from_claude(state: &AppState) -> Result<usize, AppError> {
|
pub fn import_from_claude(_state: &AppState) -> Result<usize, AppError> {
|
||||||
let mut cfg = state.config.write()?;
|
// TODO: Implement import logic using database
|
||||||
let count = mcp::import_from_claude(&mut cfg)?;
|
// For now, return 0 as a placeholder
|
||||||
drop(cfg);
|
Ok(0)
|
||||||
state.save()?;
|
|
||||||
Ok(count)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 从 Codex 导入 MCP(v3.7.0 已更新为统一结构)
|
/// 从 Codex 导入 MCP(v3.7.0 已更新为统一结构)
|
||||||
pub fn import_from_codex(state: &AppState) -> Result<usize, AppError> {
|
pub fn import_from_codex(_state: &AppState) -> Result<usize, AppError> {
|
||||||
let mut cfg = state.config.write()?;
|
// TODO: Implement import logic using database
|
||||||
let count = mcp::import_from_codex(&mut cfg)?;
|
Ok(0)
|
||||||
drop(cfg);
|
|
||||||
state.save()?;
|
|
||||||
Ok(count)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 从 Gemini 导入 MCP(v3.7.0 已更新为统一结构)
|
/// 从 Gemini 导入 MCP(v3.7.0 已更新为统一结构)
|
||||||
pub fn import_from_gemini(state: &AppState) -> Result<usize, AppError> {
|
pub fn import_from_gemini(_state: &AppState) -> Result<usize, AppError> {
|
||||||
let mut cfg = state.config.write()?;
|
// TODO: Implement import logic using database
|
||||||
let count = mcp::import_from_gemini(&mut cfg)?;
|
Ok(0)
|
||||||
drop(cfg);
|
|
||||||
state.save()?;
|
|
||||||
Ok(count)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
use std::collections::HashMap;
|
use indexmap::IndexMap;
|
||||||
|
|
||||||
use crate::app_config::AppType;
|
use crate::app_config::AppType;
|
||||||
use crate::config::write_text_file;
|
use crate::config::write_text_file;
|
||||||
@@ -13,34 +13,20 @@ impl PromptService {
|
|||||||
pub fn get_prompts(
|
pub fn get_prompts(
|
||||||
state: &AppState,
|
state: &AppState,
|
||||||
app: AppType,
|
app: AppType,
|
||||||
) -> Result<HashMap<String, Prompt>, AppError> {
|
) -> Result<IndexMap<String, Prompt>, AppError> {
|
||||||
let cfg = state.config.read()?;
|
state.db.get_prompts(app.as_str())
|
||||||
let prompts = match app {
|
|
||||||
AppType::Claude => &cfg.prompts.claude.prompts,
|
|
||||||
AppType::Codex => &cfg.prompts.codex.prompts,
|
|
||||||
AppType::Gemini => &cfg.prompts.gemini.prompts,
|
|
||||||
};
|
|
||||||
Ok(prompts.clone())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn upsert_prompt(
|
pub fn upsert_prompt(
|
||||||
state: &AppState,
|
state: &AppState,
|
||||||
app: AppType,
|
app: AppType,
|
||||||
id: &str,
|
_id: &str,
|
||||||
prompt: Prompt,
|
prompt: Prompt,
|
||||||
) -> Result<(), AppError> {
|
) -> Result<(), AppError> {
|
||||||
// 检查是否为已启用的提示词
|
// 检查是否为已启用的提示词
|
||||||
let is_enabled = prompt.enabled;
|
let is_enabled = prompt.enabled;
|
||||||
|
|
||||||
let mut cfg = state.config.write()?;
|
state.db.save_prompt(app.as_str(), &prompt)?;
|
||||||
let prompts = match app {
|
|
||||||
AppType::Claude => &mut cfg.prompts.claude.prompts,
|
|
||||||
AppType::Codex => &mut cfg.prompts.codex.prompts,
|
|
||||||
AppType::Gemini => &mut cfg.prompts.gemini.prompts,
|
|
||||||
};
|
|
||||||
prompts.insert(id.to_string(), prompt.clone());
|
|
||||||
drop(cfg);
|
|
||||||
state.save()?;
|
|
||||||
|
|
||||||
// 如果是已启用的提示词,同步更新到对应的文件
|
// 如果是已启用的提示词,同步更新到对应的文件
|
||||||
if is_enabled {
|
if is_enabled {
|
||||||
@@ -52,12 +38,7 @@ impl PromptService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn delete_prompt(state: &AppState, app: AppType, id: &str) -> Result<(), AppError> {
|
pub fn delete_prompt(state: &AppState, app: AppType, id: &str) -> Result<(), AppError> {
|
||||||
let mut cfg = state.config.write()?;
|
let prompts = state.db.get_prompts(app.as_str())?;
|
||||||
let prompts = match app {
|
|
||||||
AppType::Claude => &mut cfg.prompts.claude.prompts,
|
|
||||||
AppType::Codex => &mut cfg.prompts.codex.prompts,
|
|
||||||
AppType::Gemini => &mut cfg.prompts.gemini.prompts,
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Some(prompt) = prompts.get(id) {
|
if let Some(prompt) = prompts.get(id) {
|
||||||
if prompt.enabled {
|
if prompt.enabled {
|
||||||
@@ -65,9 +46,7 @@ impl PromptService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
prompts.remove(id);
|
state.db.delete_prompt(app.as_str(), id)?;
|
||||||
drop(cfg);
|
|
||||||
state.save()?;
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -77,12 +56,7 @@ impl PromptService {
|
|||||||
if target_path.exists() {
|
if target_path.exists() {
|
||||||
if let Ok(live_content) = std::fs::read_to_string(&target_path) {
|
if let Ok(live_content) = std::fs::read_to_string(&target_path) {
|
||||||
if !live_content.trim().is_empty() {
|
if !live_content.trim().is_empty() {
|
||||||
let mut cfg = state.config.write()?;
|
let mut prompts = state.db.get_prompts(app.as_str())?;
|
||||||
let prompts = match app {
|
|
||||||
AppType::Claude => &mut cfg.prompts.claude.prompts,
|
|
||||||
AppType::Codex => &mut cfg.prompts.codex.prompts,
|
|
||||||
AppType::Gemini => &mut cfg.prompts.gemini.prompts,
|
|
||||||
};
|
|
||||||
|
|
||||||
// 尝试回填到当前已启用的提示词
|
// 尝试回填到当前已启用的提示词
|
||||||
if let Some((enabled_id, enabled_prompt)) = prompts
|
if let Some((enabled_id, enabled_prompt)) = prompts
|
||||||
@@ -97,8 +71,7 @@ impl PromptService {
|
|||||||
enabled_prompt.content = live_content.clone();
|
enabled_prompt.content = live_content.clone();
|
||||||
enabled_prompt.updated_at = Some(timestamp);
|
enabled_prompt.updated_at = Some(timestamp);
|
||||||
log::info!("回填 live 提示词内容到已启用项: {enabled_id}");
|
log::info!("回填 live 提示词内容到已启用项: {enabled_id}");
|
||||||
drop(cfg); // 释放锁后保存,避免死锁
|
state.db.save_prompt(app.as_str(), enabled_prompt)?;
|
||||||
state.save()?; // 第一次保存:回填后立即持久化
|
|
||||||
} else {
|
} else {
|
||||||
// 没有已启用的提示词,则创建一次备份(避免重复备份)
|
// 没有已启用的提示词,则创建一次备份(避免重复备份)
|
||||||
let content_exists = prompts
|
let content_exists = prompts
|
||||||
@@ -122,13 +95,8 @@ impl PromptService {
|
|||||||
created_at: Some(timestamp),
|
created_at: Some(timestamp),
|
||||||
updated_at: Some(timestamp),
|
updated_at: Some(timestamp),
|
||||||
};
|
};
|
||||||
prompts.insert(backup_id.clone(), backup_prompt);
|
|
||||||
log::info!("回填 live 提示词内容,创建备份: {backup_id}");
|
log::info!("回填 live 提示词内容,创建备份: {backup_id}");
|
||||||
drop(cfg); // 释放锁后保存
|
state.db.save_prompt(app.as_str(), &backup_prompt)?;
|
||||||
state.save()?; // 第一次保存:回填后立即持久化
|
|
||||||
} else {
|
|
||||||
// 即使内容已存在,也无需重复备份;但不需要保存任何更改
|
|
||||||
drop(cfg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -136,12 +104,7 @@ impl PromptService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 启用目标提示词并写入文件
|
// 启用目标提示词并写入文件
|
||||||
let mut cfg = state.config.write()?;
|
let mut prompts = state.db.get_prompts(app.as_str())?;
|
||||||
let prompts = match app {
|
|
||||||
AppType::Claude => &mut cfg.prompts.claude.prompts,
|
|
||||||
AppType::Codex => &mut cfg.prompts.codex.prompts,
|
|
||||||
AppType::Gemini => &mut cfg.prompts.gemini.prompts,
|
|
||||||
};
|
|
||||||
|
|
||||||
for prompt in prompts.values_mut() {
|
for prompt in prompts.values_mut() {
|
||||||
prompt.enabled = false;
|
prompt.enabled = false;
|
||||||
@@ -150,12 +113,16 @@ impl PromptService {
|
|||||||
if let Some(prompt) = prompts.get_mut(id) {
|
if let Some(prompt) = prompts.get_mut(id) {
|
||||||
prompt.enabled = true;
|
prompt.enabled = true;
|
||||||
write_text_file(&target_path, &prompt.content)?; // 原子写入
|
write_text_file(&target_path, &prompt.content)?; // 原子写入
|
||||||
|
state.db.save_prompt(app.as_str(), prompt)?;
|
||||||
} else {
|
} else {
|
||||||
return Err(AppError::InvalidInput(format!("提示词 {id} 不存在")));
|
return Err(AppError::InvalidInput(format!("提示词 {id} 不存在")));
|
||||||
}
|
}
|
||||||
|
|
||||||
drop(cfg);
|
// Save all prompts to disable others
|
||||||
state.save()?; // 第二次保存:启用目标提示词并写入文件后
|
for (_, prompt) in prompts.iter() {
|
||||||
|
state.db.save_prompt(app.as_str(), prompt)?;
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -7,6 +7,8 @@ use std::fs;
|
|||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
use tokio::time::timeout;
|
use tokio::time::timeout;
|
||||||
|
|
||||||
|
use crate::error::format_skill_error;
|
||||||
|
|
||||||
/// 技能对象
|
/// 技能对象
|
||||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||||
pub struct Skill {
|
pub struct Skill {
|
||||||
@@ -32,6 +34,9 @@ pub struct Skill {
|
|||||||
/// 分支名称
|
/// 分支名称
|
||||||
#[serde(rename = "repoBranch")]
|
#[serde(rename = "repoBranch")]
|
||||||
pub repo_branch: Option<String>,
|
pub repo_branch: Option<String>,
|
||||||
|
/// 技能所在的子目录路径 (可选, 如 "skills")
|
||||||
|
#[serde(rename = "skillsPath")]
|
||||||
|
pub skills_path: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// 仓库配置
|
/// 仓库配置
|
||||||
@@ -130,7 +135,11 @@ impl SkillService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn get_install_dir() -> Result<PathBuf> {
|
fn get_install_dir() -> Result<PathBuf> {
|
||||||
let home = dirs::home_dir().context("无法获取用户主目录")?;
|
let home = dirs::home_dir().context(format_skill_error(
|
||||||
|
"GET_HOME_DIR_FAILED",
|
||||||
|
&[],
|
||||||
|
Some("checkPermission"),
|
||||||
|
))?;
|
||||||
Ok(home.join(".claude").join("skills"))
|
Ok(home.join(".claude").join("skills"))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -170,9 +179,19 @@ impl SkillService {
|
|||||||
/// 从仓库获取技能列表
|
/// 从仓库获取技能列表
|
||||||
async fn fetch_repo_skills(&self, repo: &SkillRepo) -> Result<Vec<Skill>> {
|
async fn fetch_repo_skills(&self, repo: &SkillRepo) -> Result<Vec<Skill>> {
|
||||||
// 为单个仓库加载增加整体超时,避免无效链接长时间阻塞
|
// 为单个仓库加载增加整体超时,避免无效链接长时间阻塞
|
||||||
let temp_dir = timeout(std::time::Duration::from_secs(15), self.download_repo(repo))
|
let temp_dir = timeout(std::time::Duration::from_secs(60), self.download_repo(repo))
|
||||||
.await
|
.await
|
||||||
.map_err(|_| anyhow!("下载仓库 {}/{} 超时", repo.owner, repo.name))??;
|
.map_err(|_| {
|
||||||
|
anyhow!(format_skill_error(
|
||||||
|
"DOWNLOAD_TIMEOUT",
|
||||||
|
&[
|
||||||
|
("owner", &repo.owner),
|
||||||
|
("name", &repo.name),
|
||||||
|
("timeout", "60")
|
||||||
|
],
|
||||||
|
Some("checkNetwork"),
|
||||||
|
))
|
||||||
|
})??;
|
||||||
let mut skills = Vec::new();
|
let mut skills = Vec::new();
|
||||||
|
|
||||||
// 确定要扫描的目录路径
|
// 确定要扫描的目录路径
|
||||||
@@ -234,6 +253,7 @@ impl SkillService {
|
|||||||
repo_owner: Some(repo.owner.clone()),
|
repo_owner: Some(repo.owner.clone()),
|
||||||
repo_name: Some(repo.name.clone()),
|
repo_name: Some(repo.name.clone()),
|
||||||
repo_branch: Some(repo.branch.clone()),
|
repo_branch: Some(repo.branch.clone()),
|
||||||
|
skills_path: repo.skills_path.clone(),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
Err(e) => log::warn!("解析 {} 元数据失败: {}", skill_md.display(), e),
|
Err(e) => log::warn!("解析 {} 元数据失败: {}", skill_md.display(), e),
|
||||||
@@ -312,6 +332,7 @@ impl SkillService {
|
|||||||
repo_owner: None,
|
repo_owner: None,
|
||||||
repo_name: None,
|
repo_name: None,
|
||||||
repo_branch: None,
|
repo_branch: None,
|
||||||
|
skills_path: None,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -374,7 +395,17 @@ impl SkillService {
|
|||||||
// 下载 ZIP
|
// 下载 ZIP
|
||||||
let response = self.http_client.get(url).send().await?;
|
let response = self.http_client.get(url).send().await?;
|
||||||
if !response.status().is_success() {
|
if !response.status().is_success() {
|
||||||
return Err(anyhow::anyhow!("下载失败: {}", response.status()));
|
let status = response.status().as_u16().to_string();
|
||||||
|
return Err(anyhow::anyhow!(format_skill_error(
|
||||||
|
"DOWNLOAD_FAILED",
|
||||||
|
&[("status", &status)],
|
||||||
|
match status.as_str() {
|
||||||
|
"403" => Some("http403"),
|
||||||
|
"404" => Some("http404"),
|
||||||
|
"429" => Some("http429"),
|
||||||
|
_ => Some("checkNetwork"),
|
||||||
|
},
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
let bytes = response.bytes().await?;
|
let bytes = response.bytes().await?;
|
||||||
@@ -389,7 +420,11 @@ impl SkillService {
|
|||||||
let name = first_file.name();
|
let name = first_file.name();
|
||||||
name.split('/').next().unwrap_or("").to_string()
|
name.split('/').next().unwrap_or("").to_string()
|
||||||
} else {
|
} else {
|
||||||
return Err(anyhow::anyhow!("空的压缩包"));
|
return Err(anyhow::anyhow!(format_skill_error(
|
||||||
|
"EMPTY_ARCHIVE",
|
||||||
|
&[],
|
||||||
|
Some("checkRepoUrl"),
|
||||||
|
)));
|
||||||
};
|
};
|
||||||
|
|
||||||
// 解压所有文件
|
// 解压所有文件
|
||||||
@@ -436,18 +471,40 @@ impl SkillService {
|
|||||||
|
|
||||||
// 下载仓库时增加总超时,防止无效链接导致长时间卡住安装过程
|
// 下载仓库时增加总超时,防止无效链接导致长时间卡住安装过程
|
||||||
let temp_dir = timeout(
|
let temp_dir = timeout(
|
||||||
std::time::Duration::from_secs(15),
|
std::time::Duration::from_secs(60),
|
||||||
self.download_repo(&repo),
|
self.download_repo(&repo),
|
||||||
)
|
)
|
||||||
.await
|
.await
|
||||||
.map_err(|_| anyhow!("下载仓库 {}/{} 超时", repo.owner, repo.name))??;
|
.map_err(|_| {
|
||||||
|
anyhow!(format_skill_error(
|
||||||
|
"DOWNLOAD_TIMEOUT",
|
||||||
|
&[
|
||||||
|
("owner", &repo.owner),
|
||||||
|
("name", &repo.name),
|
||||||
|
("timeout", "60")
|
||||||
|
],
|
||||||
|
Some("checkNetwork"),
|
||||||
|
))
|
||||||
|
})??;
|
||||||
|
|
||||||
// 复制到安装目录
|
// 根据 skills_path 确定源目录路径
|
||||||
let source = temp_dir.join(&directory);
|
let source = if let Some(ref skills_path) = repo.skills_path {
|
||||||
|
// 如果指定了 skills_path,源路径为: temp_dir/skills_path/directory
|
||||||
|
temp_dir
|
||||||
|
.join(skills_path.trim_matches('/'))
|
||||||
|
.join(&directory)
|
||||||
|
} else {
|
||||||
|
// 否则源路径为: temp_dir/directory
|
||||||
|
temp_dir.join(&directory)
|
||||||
|
};
|
||||||
|
|
||||||
if !source.exists() {
|
if !source.exists() {
|
||||||
let _ = fs::remove_dir_all(&temp_dir);
|
let _ = fs::remove_dir_all(&temp_dir);
|
||||||
return Err(anyhow::anyhow!("技能目录不存在"));
|
return Err(anyhow::anyhow!(format_skill_error(
|
||||||
|
"SKILL_DIR_NOT_FOUND",
|
||||||
|
&[("path", &source.display().to_string())],
|
||||||
|
Some("checkRepoUrl"),
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
// 删除旧版本
|
// 删除旧版本
|
||||||
|
|||||||
@@ -49,6 +49,9 @@ pub struct AppSettings {
|
|||||||
pub gemini_config_dir: Option<String>,
|
pub gemini_config_dir: Option<String>,
|
||||||
#[serde(default, skip_serializing_if = "Option::is_none")]
|
#[serde(default, skip_serializing_if = "Option::is_none")]
|
||||||
pub language: Option<String>,
|
pub language: Option<String>,
|
||||||
|
/// 是否开机自启
|
||||||
|
#[serde(default)]
|
||||||
|
pub launch_on_startup: bool,
|
||||||
#[serde(default, skip_serializing_if = "Option::is_none")]
|
#[serde(default, skip_serializing_if = "Option::is_none")]
|
||||||
pub security: Option<SecuritySettings>,
|
pub security: Option<SecuritySettings>,
|
||||||
/// Claude 自定义端点列表
|
/// Claude 自定义端点列表
|
||||||
@@ -77,6 +80,7 @@ impl Default for AppSettings {
|
|||||||
codex_config_dir: None,
|
codex_config_dir: None,
|
||||||
gemini_config_dir: None,
|
gemini_config_dir: None,
|
||||||
language: None,
|
language: None,
|
||||||
|
launch_on_startup: false,
|
||||||
security: None,
|
security: None,
|
||||||
custom_endpoints_claude: HashMap::new(),
|
custom_endpoints_claude: HashMap::new(),
|
||||||
custom_endpoints_codex: HashMap::new(),
|
custom_endpoints_codex: HashMap::new(),
|
||||||
|
|||||||
@@ -1,26 +1,14 @@
|
|||||||
use crate::app_config::MultiAppConfig;
|
use crate::database::Database;
|
||||||
use crate::error::AppError;
|
use std::sync::Arc;
|
||||||
use std::sync::RwLock;
|
|
||||||
|
|
||||||
/// 全局应用状态
|
/// 全局应用状态
|
||||||
pub struct AppState {
|
pub struct AppState {
|
||||||
pub config: RwLock<MultiAppConfig>,
|
pub db: Arc<Database>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AppState {
|
impl AppState {
|
||||||
/// 创建新的应用状态
|
/// 创建新的应用状态
|
||||||
/// 注意:仅在配置成功加载时返回;不会在失败时回退默认值。
|
pub fn new(db: Arc<Database>) -> Self {
|
||||||
pub fn try_new() -> Result<Self, AppError> {
|
Self { db }
|
||||||
let config = MultiAppConfig::load()?;
|
|
||||||
Ok(Self {
|
|
||||||
config: RwLock::new(config),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/// 保存配置到文件
|
|
||||||
pub fn save(&self) -> Result<(), AppError> {
|
|
||||||
let config = self.config.read().map_err(AppError::from)?;
|
|
||||||
|
|
||||||
config.save()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"$schema": "https://schema.tauri.app/config/2",
|
"$schema": "https://schema.tauri.app/config/2",
|
||||||
"productName": "CC Switch",
|
"productName": "CC Switch",
|
||||||
"version": "3.6.2",
|
"version": "3.7.1",
|
||||||
"identifier": "com.ccswitch.desktop",
|
"identifier": "com.ccswitch.desktop",
|
||||||
"build": {
|
"build": {
|
||||||
"frontendDist": "../dist",
|
"frontendDist": "../dist",
|
||||||
@@ -14,6 +14,7 @@
|
|||||||
{
|
{
|
||||||
"label": "main",
|
"label": "main",
|
||||||
"title": "",
|
"title": "",
|
||||||
|
"titleBarStyle": "Overlay",
|
||||||
"width": 1000,
|
"width": 1000,
|
||||||
"height": 650,
|
"height": 650,
|
||||||
"minWidth": 900,
|
"minWidth": 900,
|
||||||
|
|||||||
@@ -498,8 +498,8 @@ url = "https://example.com"
|
|||||||
.expect("unified servers should exist");
|
.expect("unified servers should exist");
|
||||||
|
|
||||||
let echo = servers.get("echo_server").expect("echo server");
|
let echo = servers.get("echo_server").expect("echo server");
|
||||||
assert_eq!(
|
assert!(
|
||||||
echo.apps.codex, true,
|
echo.apps.codex,
|
||||||
"Codex app should be enabled for echo_server"
|
"Codex app should be enabled for echo_server"
|
||||||
);
|
);
|
||||||
let server_spec = echo.server.as_object().expect("server spec");
|
let server_spec = echo.server.as_object().expect("server spec");
|
||||||
@@ -512,8 +512,8 @@ url = "https://example.com"
|
|||||||
);
|
);
|
||||||
|
|
||||||
let http = servers.get("http_server").expect("http server");
|
let http = servers.get("http_server").expect("http server");
|
||||||
assert_eq!(
|
assert!(
|
||||||
http.apps.codex, true,
|
http.apps.codex,
|
||||||
"Codex app should be enabled for http_server"
|
"Codex app should be enabled for http_server"
|
||||||
);
|
);
|
||||||
let http_spec = http.server.as_object().expect("http spec");
|
let http_spec = http.server.as_object().expect("http spec");
|
||||||
@@ -577,10 +577,7 @@ command = "echo"
|
|||||||
.expect("existing entry");
|
.expect("existing entry");
|
||||||
|
|
||||||
// 验证 Codex 应用已启用
|
// 验证 Codex 应用已启用
|
||||||
assert_eq!(
|
assert!(entry.apps.codex, "Codex app should be enabled after import");
|
||||||
entry.apps.codex, true,
|
|
||||||
"Codex app should be enabled after import"
|
|
||||||
);
|
|
||||||
|
|
||||||
// 验证现有配置被保留(server 不应被覆盖)
|
// 验证现有配置被保留(server 不应被覆盖)
|
||||||
let spec = entry.server.as_object().expect("server spec");
|
let spec = entry.server.as_object().expect("server spec");
|
||||||
@@ -702,8 +699,8 @@ fn import_from_claude_merges_into_config() {
|
|||||||
.expect("entry exists");
|
.expect("entry exists");
|
||||||
|
|
||||||
// 验证 Claude 应用已启用
|
// 验证 Claude 应用已启用
|
||||||
assert_eq!(
|
assert!(
|
||||||
entry.apps.claude, true,
|
entry.apps.claude,
|
||||||
"Claude app should be enabled after import"
|
"Claude app should be enabled after import"
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
395
src/App.tsx
395
src/App.tsx
@@ -1,7 +1,16 @@
|
|||||||
import { useEffect, useMemo, useState } from "react";
|
import { useEffect, useMemo, useState, useRef } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { toast } from "sonner";
|
import { toast } from "sonner";
|
||||||
import { Plus, Settings, Edit3 } from "lucide-react";
|
import {
|
||||||
|
Plus,
|
||||||
|
Settings,
|
||||||
|
ArrowLeft,
|
||||||
|
Bot,
|
||||||
|
Book,
|
||||||
|
Wrench,
|
||||||
|
Server,
|
||||||
|
RefreshCw,
|
||||||
|
} from "lucide-react";
|
||||||
import type { Provider } from "@/types";
|
import type { Provider } from "@/types";
|
||||||
import type { EnvConflict } from "@/types/env";
|
import type { EnvConflict } from "@/types/env";
|
||||||
import { useProvidersQuery } from "@/lib/query";
|
import { useProvidersQuery } from "@/lib/query";
|
||||||
@@ -19,7 +28,7 @@ import { ProviderList } from "@/components/providers/ProviderList";
|
|||||||
import { AddProviderDialog } from "@/components/providers/AddProviderDialog";
|
import { AddProviderDialog } from "@/components/providers/AddProviderDialog";
|
||||||
import { EditProviderDialog } from "@/components/providers/EditProviderDialog";
|
import { EditProviderDialog } from "@/components/providers/EditProviderDialog";
|
||||||
import { ConfirmDialog } from "@/components/ConfirmDialog";
|
import { ConfirmDialog } from "@/components/ConfirmDialog";
|
||||||
import { SettingsDialog } from "@/components/settings/SettingsDialog";
|
import { SettingsPage } from "@/components/settings/SettingsPage";
|
||||||
import { UpdateBadge } from "@/components/UpdateBadge";
|
import { UpdateBadge } from "@/components/UpdateBadge";
|
||||||
import { EnvWarningBanner } from "@/components/env/EnvWarningBanner";
|
import { EnvWarningBanner } from "@/components/env/EnvWarningBanner";
|
||||||
import UsageScriptModal from "@/components/UsageScriptModal";
|
import UsageScriptModal from "@/components/UsageScriptModal";
|
||||||
@@ -27,34 +36,34 @@ import UnifiedMcpPanel from "@/components/mcp/UnifiedMcpPanel";
|
|||||||
import PromptPanel from "@/components/prompts/PromptPanel";
|
import PromptPanel from "@/components/prompts/PromptPanel";
|
||||||
import { SkillsPage } from "@/components/skills/SkillsPage";
|
import { SkillsPage } from "@/components/skills/SkillsPage";
|
||||||
import { DeepLinkImportDialog } from "@/components/DeepLinkImportDialog";
|
import { DeepLinkImportDialog } from "@/components/DeepLinkImportDialog";
|
||||||
|
import { AgentsPanel } from "@/components/agents/AgentsPanel";
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import {
|
|
||||||
Dialog,
|
type View = "providers" | "settings" | "prompts" | "skills" | "mcp" | "agents";
|
||||||
DialogContent,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { VisuallyHidden } from "@radix-ui/react-visually-hidden";
|
|
||||||
|
|
||||||
function App() {
|
function App() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
const [activeApp, setActiveApp] = useState<AppId>("claude");
|
const [activeApp, setActiveApp] = useState<AppId>("claude");
|
||||||
const [isEditMode, setIsEditMode] = useState(false);
|
const [currentView, setCurrentView] = useState<View>("providers");
|
||||||
const [isSettingsOpen, setIsSettingsOpen] = useState(false);
|
|
||||||
const [isAddOpen, setIsAddOpen] = useState(false);
|
const [isAddOpen, setIsAddOpen] = useState(false);
|
||||||
const [isMcpOpen, setIsMcpOpen] = useState(false);
|
|
||||||
const [isPromptOpen, setIsPromptOpen] = useState(false);
|
|
||||||
const [isSkillsOpen, setIsSkillsOpen] = useState(false);
|
|
||||||
const [editingProvider, setEditingProvider] = useState<Provider | null>(null);
|
const [editingProvider, setEditingProvider] = useState<Provider | null>(null);
|
||||||
const [usageProvider, setUsageProvider] = useState<Provider | null>(null);
|
const [usageProvider, setUsageProvider] = useState<Provider | null>(null);
|
||||||
const [confirmDelete, setConfirmDelete] = useState<Provider | null>(null);
|
const [confirmDelete, setConfirmDelete] = useState<Provider | null>(null);
|
||||||
const [envConflicts, setEnvConflicts] = useState<EnvConflict[]>([]);
|
const [envConflicts, setEnvConflicts] = useState<EnvConflict[]>([]);
|
||||||
const [showEnvBanner, setShowEnvBanner] = useState(false);
|
const [showEnvBanner, setShowEnvBanner] = useState(false);
|
||||||
|
|
||||||
|
const promptPanelRef = useRef<any>(null);
|
||||||
|
const mcpPanelRef = useRef<any>(null);
|
||||||
|
const skillsPageRef = useRef<any>(null);
|
||||||
|
const addActionButtonClass =
|
||||||
|
"bg-orange-500 hover:bg-orange-600 dark:bg-orange-500 dark:hover:bg-orange-600 text-white shadow-lg shadow-orange-500/30 dark:shadow-orange-500/40 rounded-full w-8 h-8";
|
||||||
|
|
||||||
const { data, isLoading, refetch } = useProvidersQuery(activeApp);
|
const { data, isLoading, refetch } = useProvidersQuery(activeApp);
|
||||||
const providers = useMemo(() => data?.providers ?? {}, [data]);
|
const providers = useMemo(() => data?.providers ?? {}, [data]);
|
||||||
const currentProviderId = data?.currentProviderId ?? "";
|
const currentProviderId = data?.currentProviderId ?? "";
|
||||||
|
const isClaudeApp = activeApp === "claude";
|
||||||
|
|
||||||
// 🎯 使用 useProviderActions Hook 统一管理所有 Provider 操作
|
// 🎯 使用 useProviderActions Hook 统一管理所有 Provider 操作
|
||||||
const {
|
const {
|
||||||
@@ -98,7 +107,10 @@ function App() {
|
|||||||
|
|
||||||
if (flatConflicts.length > 0) {
|
if (flatConflicts.length > 0) {
|
||||||
setEnvConflicts(flatConflicts);
|
setEnvConflicts(flatConflicts);
|
||||||
setShowEnvBanner(true);
|
const dismissed = sessionStorage.getItem("env_banner_dismissed");
|
||||||
|
if (!dismissed) {
|
||||||
|
setShowEnvBanner(true);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(
|
console.error(
|
||||||
@@ -128,7 +140,10 @@ function App() {
|
|||||||
);
|
);
|
||||||
return [...prev, ...newConflicts];
|
return [...prev, ...newConflicts];
|
||||||
});
|
});
|
||||||
setShowEnvBanner(true);
|
const dismissed = sessionStorage.getItem("env_banner_dismissed");
|
||||||
|
if (!dismissed) {
|
||||||
|
setShowEnvBanner(true);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(
|
console.error(
|
||||||
@@ -229,13 +244,81 @@ function App() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const renderContent = () => {
|
||||||
|
switch (currentView) {
|
||||||
|
case "settings":
|
||||||
|
return (
|
||||||
|
<SettingsPage
|
||||||
|
open={true}
|
||||||
|
onOpenChange={() => setCurrentView("providers")}
|
||||||
|
onImportSuccess={handleImportSuccess}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
case "prompts":
|
||||||
|
return (
|
||||||
|
<PromptPanel
|
||||||
|
ref={promptPanelRef}
|
||||||
|
open={true}
|
||||||
|
onOpenChange={() => setCurrentView("providers")}
|
||||||
|
appId={activeApp}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
case "skills":
|
||||||
|
return (
|
||||||
|
<SkillsPage
|
||||||
|
ref={skillsPageRef}
|
||||||
|
onClose={() => setCurrentView("providers")}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
case "mcp":
|
||||||
|
return (
|
||||||
|
<UnifiedMcpPanel
|
||||||
|
ref={mcpPanelRef}
|
||||||
|
onOpenChange={() => setCurrentView("providers")}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
case "agents":
|
||||||
|
return <AgentsPanel onOpenChange={() => setCurrentView("providers")} />;
|
||||||
|
default:
|
||||||
|
return (
|
||||||
|
<div className="mx-auto max-w-[56rem] px-6 space-y-4">
|
||||||
|
<ProviderList
|
||||||
|
providers={providers}
|
||||||
|
currentProviderId={currentProviderId}
|
||||||
|
appId={activeApp}
|
||||||
|
isLoading={isLoading}
|
||||||
|
onSwitch={switchProvider}
|
||||||
|
onEdit={setEditingProvider}
|
||||||
|
onDelete={setConfirmDelete}
|
||||||
|
onDuplicate={handleDuplicateProvider}
|
||||||
|
onConfigureUsage={setUsageProvider}
|
||||||
|
onOpenWebsite={handleOpenWebsite}
|
||||||
|
onCreate={() => setIsAddOpen(true)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex h-screen flex-col bg-gray-50 dark:bg-gray-950">
|
<div
|
||||||
|
className="flex min-h-screen flex-col bg-background text-foreground selection:bg-primary/30"
|
||||||
|
style={{ overflowX: "hidden" }}
|
||||||
|
>
|
||||||
|
{/* 全局拖拽区域(顶部 4px),避免上边框无法拖动 */}
|
||||||
|
<div
|
||||||
|
className="fixed top-0 left-0 right-0 h-4 z-[60]"
|
||||||
|
data-tauri-drag-region
|
||||||
|
style={{ WebkitAppRegion: "drag" } as any}
|
||||||
|
/>
|
||||||
{/* 环境变量警告横幅 */}
|
{/* 环境变量警告横幅 */}
|
||||||
{showEnvBanner && envConflicts.length > 0 && (
|
{showEnvBanner && envConflicts.length > 0 && (
|
||||||
<EnvWarningBanner
|
<EnvWarningBanner
|
||||||
conflicts={envConflicts}
|
conflicts={envConflicts}
|
||||||
onDismiss={() => setShowEnvBanner(false)}
|
onDismiss={() => {
|
||||||
|
setShowEnvBanner(false);
|
||||||
|
sessionStorage.setItem("env_banner_dismissed", "true");
|
||||||
|
}}
|
||||||
onDeleted={async () => {
|
onDeleted={async () => {
|
||||||
// 删除后重新检测
|
// 删除后重新检测
|
||||||
try {
|
try {
|
||||||
@@ -255,92 +338,182 @@ function App() {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<header className="flex-shrink-0 border-b border-gray-200 bg-white px-6 py-4 dark:border-gray-800 dark:bg-gray-900">
|
<header
|
||||||
<div className="flex flex-wrap items-center justify-between gap-2">
|
className="glass-header fixed top-0 z-50 w-full py-3 transition-all duration-300"
|
||||||
<div className="flex items-center gap-1">
|
data-tauri-drag-region
|
||||||
<a
|
style={{ WebkitAppRegion: "drag" } as any}
|
||||||
href="https://github.com/farion1231/cc-switch"
|
>
|
||||||
target="_blank"
|
<div className="h-4 w-full" aria-hidden data-tauri-drag-region />
|
||||||
rel="noreferrer"
|
<div
|
||||||
className="text-xl font-semibold text-blue-500 transition-colors hover:text-blue-600 dark:text-blue-400 dark:hover:text-blue-300"
|
className="mx-auto max-w-[56rem] px-6 flex flex-wrap items-center justify-between gap-2"
|
||||||
>
|
data-tauri-drag-region
|
||||||
CC Switch
|
style={{ WebkitAppRegion: "drag" } as any}
|
||||||
</a>
|
>
|
||||||
<Button
|
<div
|
||||||
variant="ghost"
|
className="flex items-center gap-1"
|
||||||
size="icon"
|
style={{ WebkitAppRegion: "no-drag" } as any}
|
||||||
onClick={() => setIsSettingsOpen(true)}
|
>
|
||||||
title={t("common.settings")}
|
{currentView !== "providers" ? (
|
||||||
className="ml-2"
|
<div className="flex items-center gap-2">
|
||||||
>
|
<Button
|
||||||
<Settings className="h-4 w-4" />
|
variant="outline"
|
||||||
</Button>
|
size="icon"
|
||||||
<Button
|
onClick={() => setCurrentView("providers")}
|
||||||
variant="ghost"
|
className="mr-2 rounded-lg"
|
||||||
size="icon"
|
>
|
||||||
onClick={() => setIsEditMode(!isEditMode)}
|
<ArrowLeft className="h-4 w-4" />
|
||||||
title={t(
|
</Button>
|
||||||
isEditMode ? "header.exitEditMode" : "header.enterEditMode",
|
<h1 className="text-lg font-semibold">
|
||||||
)}
|
{currentView === "settings" && t("settings.title")}
|
||||||
className={
|
{currentView === "prompts" &&
|
||||||
isEditMode
|
t("prompts.title", { appName: t(`apps.${activeApp}`) })}
|
||||||
? "text-blue-500 hover:text-blue-600 dark:text-blue-400 dark:hover:text-blue-300"
|
{currentView === "skills" && t("skills.title")}
|
||||||
: ""
|
{currentView === "mcp" && t("mcp.unifiedPanel.title")}
|
||||||
}
|
{currentView === "agents" && "Agents"}
|
||||||
>
|
</h1>
|
||||||
<Edit3 className="h-4 w-4" />
|
</div>
|
||||||
</Button>
|
) : (
|
||||||
<UpdateBadge onClick={() => setIsSettingsOpen(true)} />
|
<>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<a
|
||||||
|
href="https://github.com/farion1231/cc-switch"
|
||||||
|
target="_blank"
|
||||||
|
rel="noreferrer"
|
||||||
|
className="text-xl font-semibold text-blue-500 transition-colors hover:text-blue-600 dark:text-blue-400 dark:hover:text-blue-300"
|
||||||
|
>
|
||||||
|
CC Switch
|
||||||
|
</a>
|
||||||
|
<div className="h-5 w-[1px] bg-black/10 dark:bg-white/15" />
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="icon"
|
||||||
|
onClick={() => setCurrentView("settings")}
|
||||||
|
title={t("common.settings")}
|
||||||
|
className="hover:bg-black/5 dark:hover:bg-white/5"
|
||||||
|
>
|
||||||
|
<Settings className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
<UpdateBadge onClick={() => setCurrentView("settings")} />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex flex-wrap items-center gap-2">
|
<div
|
||||||
<AppSwitcher activeApp={activeApp} onSwitch={setActiveApp} />
|
className="flex items-center gap-2"
|
||||||
<Button
|
style={{ WebkitAppRegion: "no-drag" } as any}
|
||||||
variant="mcp"
|
>
|
||||||
onClick={() => setIsPromptOpen(true)}
|
{currentView === "prompts" && (
|
||||||
className="min-w-[80px]"
|
<Button
|
||||||
>
|
size="icon"
|
||||||
{t("prompts.manage")}
|
onClick={() => promptPanelRef.current?.openAdd()}
|
||||||
</Button>
|
className={addActionButtonClass}
|
||||||
<Button
|
title={t("prompts.add")}
|
||||||
variant="mcp"
|
>
|
||||||
onClick={() => setIsMcpOpen(true)}
|
<Plus className="h-5 w-5" />
|
||||||
className="min-w-[80px]"
|
</Button>
|
||||||
>
|
)}
|
||||||
MCP
|
{currentView === "mcp" && (
|
||||||
</Button>
|
<Button
|
||||||
<Button
|
size="icon"
|
||||||
variant="mcp"
|
onClick={() => mcpPanelRef.current?.openAdd()}
|
||||||
onClick={() => setIsSkillsOpen(true)}
|
className={addActionButtonClass}
|
||||||
className="min-w-[80px]"
|
title={t("mcp.unifiedPanel.addServer")}
|
||||||
>
|
>
|
||||||
{t("skills.manage")}
|
<Plus className="h-5 w-5" />
|
||||||
</Button>
|
</Button>
|
||||||
<Button onClick={() => setIsAddOpen(true)}>
|
)}
|
||||||
<Plus className="h-4 w-4" />
|
{currentView === "skills" && (
|
||||||
{t("header.addProvider")}
|
<>
|
||||||
</Button>
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => skillsPageRef.current?.refresh()}
|
||||||
|
className="hover:bg-black/5 dark:hover:bg-white/5"
|
||||||
|
>
|
||||||
|
<RefreshCw className="h-4 w-4 mr-2" />
|
||||||
|
{t("skills.refresh")}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => skillsPageRef.current?.openRepoManager()}
|
||||||
|
className="hover:bg-black/5 dark:hover:bg-white/5"
|
||||||
|
>
|
||||||
|
<Settings className="h-4 w-4 mr-2" />
|
||||||
|
{t("skills.repoManager")}
|
||||||
|
</Button>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{currentView === "providers" && (
|
||||||
|
<>
|
||||||
|
<AppSwitcher activeApp={activeApp} onSwitch={setActiveApp} />
|
||||||
|
|
||||||
|
<div className="h-8 w-[1px] bg-black/10 dark:bg-white/10 mx-1" />
|
||||||
|
|
||||||
|
<div className="glass p-1 rounded-xl flex items-center gap-1">
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => setCurrentView("prompts")}
|
||||||
|
className="text-muted-foreground hover:text-foreground hover:bg-black/5 dark:hover:bg-white/5"
|
||||||
|
title={t("prompts.manage")}
|
||||||
|
>
|
||||||
|
<Book className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
{isClaudeApp && (
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => setCurrentView("skills")}
|
||||||
|
className="text-muted-foreground hover:text-foreground hover:bg-black/5 dark:hover:bg-white/5"
|
||||||
|
title={t("skills.manage")}
|
||||||
|
>
|
||||||
|
<Wrench className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => setCurrentView("mcp")}
|
||||||
|
className="text-muted-foreground hover:text-foreground hover:bg-black/5 dark:hover:bg-white/5"
|
||||||
|
title="MCP"
|
||||||
|
>
|
||||||
|
<Server className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
{isClaudeApp && (
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => setCurrentView("agents")}
|
||||||
|
className="text-muted-foreground hover:text-foreground hover:bg-black/5 dark:hover:bg-white/5"
|
||||||
|
title="Agents"
|
||||||
|
>
|
||||||
|
<Bot className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
onClick={() => setIsAddOpen(true)}
|
||||||
|
size="icon"
|
||||||
|
className={`ml-2 ${addActionButtonClass}`}
|
||||||
|
>
|
||||||
|
<Plus className="h-5 w-5" />
|
||||||
|
</Button>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<main className="flex-1 overflow-y-scroll">
|
<main
|
||||||
<div className="mx-auto max-w-4xl px-6 py-6">
|
className={`flex-1 overflow-y-auto pb-12 animate-fade-in scroll-overlay ${
|
||||||
<ProviderList
|
currentView === "providers" ? "pt-24" : "pt-20"
|
||||||
providers={providers}
|
}`}
|
||||||
currentProviderId={currentProviderId}
|
style={{ overflowX: "hidden" }}
|
||||||
appId={activeApp}
|
>
|
||||||
isLoading={isLoading}
|
{renderContent()}
|
||||||
isEditMode={isEditMode}
|
|
||||||
onSwitch={switchProvider}
|
|
||||||
onEdit={setEditingProvider}
|
|
||||||
onDelete={setConfirmDelete}
|
|
||||||
onDuplicate={handleDuplicateProvider}
|
|
||||||
onConfigureUsage={setUsageProvider}
|
|
||||||
onOpenWebsite={handleOpenWebsite}
|
|
||||||
onCreate={() => setIsAddOpen(true)}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</main>
|
</main>
|
||||||
|
|
||||||
<AddProviderDialog
|
<AddProviderDialog
|
||||||
@@ -388,30 +561,6 @@ function App() {
|
|||||||
onCancel={() => setConfirmDelete(null)}
|
onCancel={() => setConfirmDelete(null)}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<SettingsDialog
|
|
||||||
open={isSettingsOpen}
|
|
||||||
onOpenChange={setIsSettingsOpen}
|
|
||||||
onImportSuccess={handleImportSuccess}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<PromptPanel
|
|
||||||
open={isPromptOpen}
|
|
||||||
onOpenChange={setIsPromptOpen}
|
|
||||||
appId={activeApp}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<UnifiedMcpPanel open={isMcpOpen} onOpenChange={setIsMcpOpen} />
|
|
||||||
|
|
||||||
<Dialog open={isSkillsOpen} onOpenChange={setIsSkillsOpen}>
|
|
||||||
<DialogContent className="max-w-4xl max-h-[85vh] min-h-[600px] flex flex-col p-0">
|
|
||||||
<DialogHeader className="sr-only">
|
|
||||||
<VisuallyHidden>
|
|
||||||
<DialogTitle>{t("skills.title")}</DialogTitle>
|
|
||||||
</VisuallyHidden>
|
|
||||||
</DialogHeader>
|
|
||||||
<SkillsPage onClose={() => setIsSkillsOpen(false)} />
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
<DeepLinkImportDialog />
|
<DeepLinkImportDialog />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -13,13 +13,13 @@ export function AppSwitcher({ activeApp, onSwitch }: AppSwitcherProps) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="inline-flex bg-gray-100 dark:bg-gray-800 rounded-lg p-1 gap-1 border border-transparent ">
|
<div className="inline-flex bg-gray-100 dark:bg-gray-800 rounded-lg p-1 gap-1">
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
onClick={() => handleSwitch("claude")}
|
onClick={() => handleSwitch("claude")}
|
||||||
className={`group inline-flex items-center gap-2 px-3 py-2 rounded-md text-sm font-medium transition-all duration-200 ${
|
className={`group inline-flex items-center gap-2 px-3 py-2 rounded-md text-sm font-medium transition-all duration-200 ${
|
||||||
activeApp === "claude"
|
activeApp === "claude"
|
||||||
? "bg-white text-gray-900 shadow-sm dark:bg-gray-900 dark:text-gray-100 dark:shadow-none"
|
? "bg-white text-gray-900 shadow-sm dark:bg-gray-900 dark:text-gray-100"
|
||||||
: "text-gray-500 hover:text-gray-900 hover:bg-white/50 dark:text-gray-400 dark:hover:text-gray-100 dark:hover:bg-gray-800/60"
|
: "text-gray-500 hover:text-gray-900 hover:bg-white/50 dark:text-gray-400 dark:hover:text-gray-100 dark:hover:bg-gray-800/60"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
@@ -27,8 +27,8 @@ export function AppSwitcher({ activeApp, onSwitch }: AppSwitcherProps) {
|
|||||||
size={16}
|
size={16}
|
||||||
className={
|
className={
|
||||||
activeApp === "claude"
|
activeApp === "claude"
|
||||||
? "text-[#D97757] dark:text-[#D97757] transition-colors duration-200"
|
? "text-foreground"
|
||||||
: "text-gray-500 dark:text-gray-400 group-hover:text-[#D97757] dark:group-hover:text-[#D97757] transition-colors duration-200"
|
: "text-gray-500 dark:text-gray-400 group-hover:text-foreground transition-colors"
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<span>Claude</span>
|
<span>Claude</span>
|
||||||
@@ -39,11 +39,18 @@ export function AppSwitcher({ activeApp, onSwitch }: AppSwitcherProps) {
|
|||||||
onClick={() => handleSwitch("codex")}
|
onClick={() => handleSwitch("codex")}
|
||||||
className={`inline-flex items-center gap-2 px-3 py-2 rounded-md text-sm font-medium transition-all duration-200 ${
|
className={`inline-flex items-center gap-2 px-3 py-2 rounded-md text-sm font-medium transition-all duration-200 ${
|
||||||
activeApp === "codex"
|
activeApp === "codex"
|
||||||
? "bg-white text-gray-900 shadow-sm dark:bg-gray-900 dark:text-gray-100 dark:shadow-none"
|
? "bg-white text-gray-900 shadow-sm dark:bg-gray-900 dark:text-gray-100"
|
||||||
: "text-gray-500 hover:text-gray-900 hover:bg-white/50 dark:text-gray-400 dark:hover:text-gray-100 dark:hover:bg-gray-800/60"
|
: "text-gray-500 hover:text-gray-900 hover:bg-white/50 dark:text-gray-400 dark:hover:text-gray-100 dark:hover:bg-gray-800/60"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
<CodexIcon size={16} />
|
<CodexIcon
|
||||||
|
size={16}
|
||||||
|
className={
|
||||||
|
activeApp === "codex"
|
||||||
|
? "text-foreground"
|
||||||
|
: "text-gray-500 dark:text-gray-400 group-hover:text-foreground transition-colors"
|
||||||
|
}
|
||||||
|
/>
|
||||||
<span>Codex</span>
|
<span>Codex</span>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
@@ -52,7 +59,7 @@ export function AppSwitcher({ activeApp, onSwitch }: AppSwitcherProps) {
|
|||||||
onClick={() => handleSwitch("gemini")}
|
onClick={() => handleSwitch("gemini")}
|
||||||
className={`inline-flex items-center gap-2 px-3 py-2 rounded-md text-sm font-medium transition-all duration-200 ${
|
className={`inline-flex items-center gap-2 px-3 py-2 rounded-md text-sm font-medium transition-all duration-200 ${
|
||||||
activeApp === "gemini"
|
activeApp === "gemini"
|
||||||
? "bg-white text-gray-900 shadow-sm dark:bg-gray-900 dark:text-gray-100 dark:shadow-none"
|
? "bg-white text-gray-900 shadow-sm dark:bg-gray-900 dark:text-gray-100"
|
||||||
: "text-gray-500 hover:text-gray-900 hover:bg-white/50 dark:text-gray-400 dark:hover:text-gray-100 dark:hover:bg-gray-800/60"
|
: "text-gray-500 hover:text-gray-900 hover:bg-white/50 dark:text-gray-400 dark:hover:text-gray-100 dark:hover:bg-gray-800/60"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
@@ -60,8 +67,8 @@ export function AppSwitcher({ activeApp, onSwitch }: AppSwitcherProps) {
|
|||||||
size={16}
|
size={16}
|
||||||
className={
|
className={
|
||||||
activeApp === "gemini"
|
activeApp === "gemini"
|
||||||
? "text-[#4285F4] dark:text-[#4285F4] transition-colors duration-200"
|
? "text-foreground"
|
||||||
: "text-gray-500 dark:text-gray-400 group-hover:text-[#4285F4] dark:group-hover:text-[#4285F4] transition-colors duration-200"
|
: "text-gray-500 dark:text-gray-400 group-hover:text-foreground transition-colors"
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<span>Gemini</span>
|
<span>Gemini</span>
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
76
src/components/ColorPicker.tsx
Normal file
76
src/components/ColorPicker.tsx
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { Label } from "@/components/ui/label";
|
||||||
|
import { cn } from "@/lib/utils";
|
||||||
|
|
||||||
|
interface ColorPickerProps {
|
||||||
|
value?: string;
|
||||||
|
onValueChange: (color: string) => void;
|
||||||
|
label?: string;
|
||||||
|
presets?: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
const DEFAULT_PRESETS = [
|
||||||
|
"#00A67E",
|
||||||
|
"#D4915D",
|
||||||
|
"#4285F4",
|
||||||
|
"#FF6A00",
|
||||||
|
"#00A4FF",
|
||||||
|
"#FF9900",
|
||||||
|
"#0078D4",
|
||||||
|
"#FF0000",
|
||||||
|
"#1E88E5",
|
||||||
|
"#6366F1",
|
||||||
|
"#0F62FE",
|
||||||
|
"#2932E1",
|
||||||
|
];
|
||||||
|
|
||||||
|
export const ColorPicker: React.FC<ColorPickerProps> = ({
|
||||||
|
value = "#4285F4",
|
||||||
|
onValueChange,
|
||||||
|
label = "图标颜色",
|
||||||
|
presets = DEFAULT_PRESETS,
|
||||||
|
}) => {
|
||||||
|
return (
|
||||||
|
<div className="space-y-3">
|
||||||
|
<Label>{label}</Label>
|
||||||
|
|
||||||
|
{/* 颜色预设 */}
|
||||||
|
<div className="grid grid-cols-6 gap-2">
|
||||||
|
{presets.map((color) => (
|
||||||
|
<button
|
||||||
|
key={color}
|
||||||
|
type="button"
|
||||||
|
onClick={() => onValueChange(color)}
|
||||||
|
className={cn(
|
||||||
|
"w-full aspect-square rounded-lg border-2 transition-all",
|
||||||
|
"hover:scale-110 hover:shadow-lg",
|
||||||
|
value === color
|
||||||
|
? "border-primary ring-2 ring-primary/20"
|
||||||
|
: "border-border",
|
||||||
|
)}
|
||||||
|
style={{ backgroundColor: color }}
|
||||||
|
title={color}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* 自定义颜色输入 */}
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<Input
|
||||||
|
type="color"
|
||||||
|
value={value}
|
||||||
|
onChange={(e) => onValueChange(e.target.value)}
|
||||||
|
className="w-16 h-10 p-1 cursor-pointer"
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
type="text"
|
||||||
|
value={value}
|
||||||
|
onChange={(e) => onValueChange(e.target.value)}
|
||||||
|
placeholder="#4285F4"
|
||||||
|
className="flex-1 font-mono"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useState, useEffect } from "react";
|
import { useState, useEffect, useMemo } from "react";
|
||||||
import { listen } from "@tauri-apps/api/event";
|
import { listen } from "@tauri-apps/api/event";
|
||||||
import { DeepLinkImportRequest, deeplinkApi } from "@/lib/api/deeplink";
|
import { DeepLinkImportRequest, deeplinkApi } from "@/lib/api/deeplink";
|
||||||
import {
|
import {
|
||||||
@@ -30,9 +30,30 @@ export function DeepLinkImportDialog() {
|
|||||||
// Listen for deep link import events
|
// Listen for deep link import events
|
||||||
const unlistenImport = listen<DeepLinkImportRequest>(
|
const unlistenImport = listen<DeepLinkImportRequest>(
|
||||||
"deeplink-import",
|
"deeplink-import",
|
||||||
(event) => {
|
async (event) => {
|
||||||
console.log("Deep link import event received:", event.payload);
|
console.log("Deep link import event received:", event.payload);
|
||||||
setRequest(event.payload);
|
|
||||||
|
// If config is present, merge it to get the complete configuration
|
||||||
|
if (event.payload.config || event.payload.configUrl) {
|
||||||
|
try {
|
||||||
|
const mergedRequest = await deeplinkApi.mergeDeeplinkConfig(
|
||||||
|
event.payload,
|
||||||
|
);
|
||||||
|
console.log("Config merged successfully:", mergedRequest);
|
||||||
|
setRequest(mergedRequest);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to merge config:", error);
|
||||||
|
toast.error(t("deeplink.configMergeError"), {
|
||||||
|
description:
|
||||||
|
error instanceof Error ? error.message : String(error),
|
||||||
|
});
|
||||||
|
// Fall back to original request
|
||||||
|
setRequest(event.payload);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
setRequest(event.payload);
|
||||||
|
}
|
||||||
|
|
||||||
setIsOpen(true);
|
setIsOpen(true);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@@ -71,7 +92,6 @@ export function DeepLinkImportDialog() {
|
|||||||
});
|
});
|
||||||
|
|
||||||
setIsOpen(false);
|
setIsOpen(false);
|
||||||
setRequest(null);
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Failed to import provider from deep link:", error);
|
console.error("Failed to import provider from deep link:", error);
|
||||||
toast.error(t("deeplink.importError"), {
|
toast.error(t("deeplink.importError"), {
|
||||||
@@ -84,120 +104,326 @@ export function DeepLinkImportDialog() {
|
|||||||
|
|
||||||
const handleCancel = () => {
|
const handleCancel = () => {
|
||||||
setIsOpen(false);
|
setIsOpen(false);
|
||||||
setRequest(null);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
if (!request) return null;
|
|
||||||
|
|
||||||
// Mask API key for display (show first 4 chars + ***)
|
// Mask API key for display (show first 4 chars + ***)
|
||||||
const maskedApiKey =
|
const maskedApiKey =
|
||||||
request.apiKey.length > 4
|
request?.apiKey && request.apiKey.length > 4
|
||||||
? `${request.apiKey.substring(0, 4)}${"*".repeat(20)}`
|
? `${request.apiKey.substring(0, 4)}${"*".repeat(20)}`
|
||||||
: "****";
|
: "****";
|
||||||
|
|
||||||
|
// Check if config file is present
|
||||||
|
const hasConfigFile = !!(request?.config || request?.configUrl);
|
||||||
|
const configSource = request?.config
|
||||||
|
? "base64"
|
||||||
|
: request?.configUrl
|
||||||
|
? "url"
|
||||||
|
: null;
|
||||||
|
|
||||||
|
// Parse config file content for display
|
||||||
|
interface ParsedConfig {
|
||||||
|
type: "claude" | "codex" | "gemini";
|
||||||
|
env?: Record<string, string>;
|
||||||
|
auth?: Record<string, string>;
|
||||||
|
tomlConfig?: string;
|
||||||
|
raw: Record<string, unknown>;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper to decode base64 with UTF-8 support
|
||||||
|
const b64ToUtf8 = (str: string): string => {
|
||||||
|
try {
|
||||||
|
const binString = atob(str);
|
||||||
|
const bytes = Uint8Array.from(binString, (m) => m.codePointAt(0) || 0);
|
||||||
|
return new TextDecoder().decode(bytes);
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Failed to decode base64:", e);
|
||||||
|
return atob(str);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const parsedConfig = useMemo((): ParsedConfig | null => {
|
||||||
|
if (!request?.config) return null;
|
||||||
|
try {
|
||||||
|
const decoded = b64ToUtf8(request.config);
|
||||||
|
const parsed = JSON.parse(decoded) as Record<string, unknown>;
|
||||||
|
|
||||||
|
if (request.app === "claude") {
|
||||||
|
// Claude 格式: { env: { ANTHROPIC_AUTH_TOKEN: ..., ... } }
|
||||||
|
return {
|
||||||
|
type: "claude",
|
||||||
|
env: (parsed.env as Record<string, string>) || {},
|
||||||
|
raw: parsed,
|
||||||
|
};
|
||||||
|
} else if (request.app === "codex") {
|
||||||
|
// Codex 格式: { auth: { OPENAI_API_KEY: ... }, config: "TOML string" }
|
||||||
|
return {
|
||||||
|
type: "codex",
|
||||||
|
auth: (parsed.auth as Record<string, string>) || {},
|
||||||
|
tomlConfig: (parsed.config as string) || "",
|
||||||
|
raw: parsed,
|
||||||
|
};
|
||||||
|
} else if (request.app === "gemini") {
|
||||||
|
// Gemini 格式: 扁平结构 { GEMINI_API_KEY: ..., GEMINI_BASE_URL: ... }
|
||||||
|
return {
|
||||||
|
type: "gemini",
|
||||||
|
env: parsed as Record<string, string>,
|
||||||
|
raw: parsed,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Failed to parse config:", e);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}, [request?.config, request?.app]);
|
||||||
|
|
||||||
|
// Helper to mask sensitive values
|
||||||
|
const maskValue = (key: string, value: string): string => {
|
||||||
|
const sensitiveKeys = ["TOKEN", "KEY", "SECRET", "PASSWORD"];
|
||||||
|
const isSensitive = sensitiveKeys.some((k) =>
|
||||||
|
key.toUpperCase().includes(k),
|
||||||
|
);
|
||||||
|
if (isSensitive && value.length > 8) {
|
||||||
|
return `${value.substring(0, 8)}${"*".repeat(12)}`;
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog open={isOpen} onOpenChange={setIsOpen}>
|
<Dialog open={isOpen && !!request} onOpenChange={setIsOpen}>
|
||||||
<DialogContent className="sm:max-w-[500px]">
|
<DialogContent className="sm:max-w-[500px]" zIndex="top">
|
||||||
{/* 标题显式左对齐,避免默认居中样式影响 */}
|
{request && (
|
||||||
<DialogHeader className="text-left sm:text-left">
|
<>
|
||||||
<DialogTitle>{t("deeplink.confirmImport")}</DialogTitle>
|
{/* 标题显式左对齐,避免默认居中样式影响 */}
|
||||||
<DialogDescription>
|
<DialogHeader className="text-left sm:text-left">
|
||||||
{t("deeplink.confirmImportDescription")}
|
<DialogTitle>{t("deeplink.confirmImport")}</DialogTitle>
|
||||||
</DialogDescription>
|
<DialogDescription>
|
||||||
</DialogHeader>
|
{t("deeplink.confirmImportDescription")}
|
||||||
|
</DialogDescription>
|
||||||
|
</DialogHeader>
|
||||||
|
|
||||||
{/* 主体内容整体右移,略大于标题内边距,让内容看起来不贴边 */}
|
{/* 主体内容整体右移,略大于标题内边距,让内容看起来不贴边 */}
|
||||||
<div className="space-y-4 px-8 py-4">
|
<div className="space-y-4 px-8 py-4 max-h-[60vh] overflow-y-auto [scrollbar-width:thin] [&::-webkit-scrollbar]:w-1.5 [&::-webkit-scrollbar]:block [&::-webkit-scrollbar-thumb]:rounded-full [&::-webkit-scrollbar-thumb]:bg-gray-200 dark:[&::-webkit-scrollbar-thumb]:bg-gray-700">
|
||||||
{/* App Type */}
|
{/* App Type */}
|
||||||
<div className="grid grid-cols-3 items-center gap-4">
|
<div className="grid grid-cols-3 items-center gap-4">
|
||||||
<div className="font-medium text-sm text-muted-foreground">
|
<div className="font-medium text-sm text-muted-foreground">
|
||||||
{t("deeplink.app")}
|
{t("deeplink.app")}
|
||||||
</div>
|
</div>
|
||||||
<div className="col-span-2 text-sm font-medium capitalize">
|
<div className="col-span-2 text-sm font-medium capitalize">
|
||||||
{request.app}
|
{request.app}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* Provider Name */}
|
|
||||||
<div className="grid grid-cols-3 items-center gap-4">
|
|
||||||
<div className="font-medium text-sm text-muted-foreground">
|
|
||||||
{t("deeplink.providerName")}
|
|
||||||
</div>
|
|
||||||
<div className="col-span-2 text-sm font-medium">{request.name}</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* Homepage */}
|
|
||||||
<div className="grid grid-cols-3 items-center gap-4">
|
|
||||||
<div className="font-medium text-sm text-muted-foreground">
|
|
||||||
{t("deeplink.homepage")}
|
|
||||||
</div>
|
|
||||||
<div className="col-span-2 text-sm break-all text-blue-600 dark:text-blue-400">
|
|
||||||
{request.homepage}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* API Endpoint */}
|
|
||||||
<div className="grid grid-cols-3 items-center gap-4">
|
|
||||||
<div className="font-medium text-sm text-muted-foreground">
|
|
||||||
{t("deeplink.endpoint")}
|
|
||||||
</div>
|
|
||||||
<div className="col-span-2 text-sm break-all">
|
|
||||||
{request.endpoint}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* API Key (masked) */}
|
|
||||||
<div className="grid grid-cols-3 items-center gap-4">
|
|
||||||
<div className="font-medium text-sm text-muted-foreground">
|
|
||||||
{t("deeplink.apiKey")}
|
|
||||||
</div>
|
|
||||||
<div className="col-span-2 text-sm font-mono text-muted-foreground">
|
|
||||||
{maskedApiKey}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* Model (if present) */}
|
|
||||||
{request.model && (
|
|
||||||
<div className="grid grid-cols-3 items-center gap-4">
|
|
||||||
<div className="font-medium text-sm text-muted-foreground">
|
|
||||||
{t("deeplink.model")}
|
|
||||||
</div>
|
</div>
|
||||||
<div className="col-span-2 text-sm font-mono">
|
|
||||||
{request.model}
|
{/* Provider Name */}
|
||||||
|
<div className="grid grid-cols-3 items-center gap-4">
|
||||||
|
<div className="font-medium text-sm text-muted-foreground">
|
||||||
|
{t("deeplink.providerName")}
|
||||||
|
</div>
|
||||||
|
<div className="col-span-2 text-sm font-medium">
|
||||||
|
{request.name}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Homepage */}
|
||||||
|
<div className="grid grid-cols-3 items-center gap-4">
|
||||||
|
<div className="font-medium text-sm text-muted-foreground">
|
||||||
|
{t("deeplink.homepage")}
|
||||||
|
</div>
|
||||||
|
<div className="col-span-2 text-sm break-all text-blue-600 dark:text-blue-400">
|
||||||
|
{request.homepage}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* API Endpoint */}
|
||||||
|
<div className="grid grid-cols-3 items-center gap-4">
|
||||||
|
<div className="font-medium text-sm text-muted-foreground">
|
||||||
|
{t("deeplink.endpoint")}
|
||||||
|
</div>
|
||||||
|
<div className="col-span-2 text-sm break-all">
|
||||||
|
{request.endpoint}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* API Key (masked) */}
|
||||||
|
<div className="grid grid-cols-3 items-center gap-4">
|
||||||
|
<div className="font-medium text-sm text-muted-foreground">
|
||||||
|
{t("deeplink.apiKey")}
|
||||||
|
</div>
|
||||||
|
<div className="col-span-2 text-sm font-mono text-muted-foreground">
|
||||||
|
{maskedApiKey}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Model (if present) */}
|
||||||
|
{request.model && (
|
||||||
|
<div className="grid grid-cols-3 items-center gap-4">
|
||||||
|
<div className="font-medium text-sm text-muted-foreground">
|
||||||
|
{t("deeplink.model")}
|
||||||
|
</div>
|
||||||
|
<div className="col-span-2 text-sm font-mono">
|
||||||
|
{request.model}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Notes (if present) */}
|
||||||
|
{request.notes && (
|
||||||
|
<div className="grid grid-cols-3 items-start gap-4">
|
||||||
|
<div className="font-medium text-sm text-muted-foreground">
|
||||||
|
{t("deeplink.notes")}
|
||||||
|
</div>
|
||||||
|
<div className="col-span-2 text-sm text-muted-foreground">
|
||||||
|
{request.notes}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Config File Details (v3.8+) */}
|
||||||
|
{hasConfigFile && (
|
||||||
|
<div className="space-y-3 pt-2 border-t border-border-default">
|
||||||
|
<div className="grid grid-cols-3 items-center gap-4">
|
||||||
|
<div className="font-medium text-sm text-muted-foreground">
|
||||||
|
{t("deeplink.configSource")}
|
||||||
|
</div>
|
||||||
|
<div className="col-span-2 text-sm">
|
||||||
|
<span className="inline-flex items-center px-2 py-0.5 rounded-md bg-blue-100 dark:bg-blue-900/30 text-blue-700 dark:text-blue-300 text-xs font-medium">
|
||||||
|
{configSource === "base64"
|
||||||
|
? t("deeplink.configEmbedded")
|
||||||
|
: t("deeplink.configRemote")}
|
||||||
|
</span>
|
||||||
|
{request.configFormat && (
|
||||||
|
<span className="ml-2 text-xs text-muted-foreground uppercase">
|
||||||
|
{request.configFormat}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Parsed Config Details */}
|
||||||
|
{parsedConfig && (
|
||||||
|
<div className="rounded-lg bg-muted/50 p-3 space-y-2">
|
||||||
|
<div className="text-xs font-medium text-muted-foreground uppercase tracking-wide">
|
||||||
|
{t("deeplink.configDetails")}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Claude config */}
|
||||||
|
{parsedConfig.type === "claude" && parsedConfig.env && (
|
||||||
|
<div className="space-y-1.5">
|
||||||
|
{Object.entries(parsedConfig.env).map(
|
||||||
|
([key, value]) => (
|
||||||
|
<div
|
||||||
|
key={key}
|
||||||
|
className="grid grid-cols-2 gap-2 text-xs"
|
||||||
|
>
|
||||||
|
<span className="font-mono text-muted-foreground truncate">
|
||||||
|
{key}
|
||||||
|
</span>
|
||||||
|
<span className="font-mono truncate">
|
||||||
|
{maskValue(key, String(value))}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Codex config */}
|
||||||
|
{parsedConfig.type === "codex" && (
|
||||||
|
<div className="space-y-2">
|
||||||
|
{parsedConfig.auth &&
|
||||||
|
Object.keys(parsedConfig.auth).length > 0 && (
|
||||||
|
<div className="space-y-1.5">
|
||||||
|
<div className="text-xs text-muted-foreground">
|
||||||
|
Auth:
|
||||||
|
</div>
|
||||||
|
{Object.entries(parsedConfig.auth).map(
|
||||||
|
([key, value]) => (
|
||||||
|
<div
|
||||||
|
key={key}
|
||||||
|
className="grid grid-cols-2 gap-2 text-xs pl-2"
|
||||||
|
>
|
||||||
|
<span className="font-mono text-muted-foreground truncate">
|
||||||
|
{key}
|
||||||
|
</span>
|
||||||
|
<span className="font-mono truncate">
|
||||||
|
{maskValue(key, String(value))}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{parsedConfig.tomlConfig && (
|
||||||
|
<div className="space-y-1">
|
||||||
|
<div className="text-xs text-muted-foreground">
|
||||||
|
TOML Config:
|
||||||
|
</div>
|
||||||
|
<pre className="text-xs font-mono bg-background p-2 rounded overflow-x-auto max-h-24 whitespace-pre-wrap">
|
||||||
|
{parsedConfig.tomlConfig.substring(0, 300)}
|
||||||
|
{parsedConfig.tomlConfig.length > 300 && "..."}
|
||||||
|
</pre>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Gemini config */}
|
||||||
|
{parsedConfig.type === "gemini" && parsedConfig.env && (
|
||||||
|
<div className="space-y-1.5">
|
||||||
|
{Object.entries(parsedConfig.env).map(
|
||||||
|
([key, value]) => (
|
||||||
|
<div
|
||||||
|
key={key}
|
||||||
|
className="grid grid-cols-2 gap-2 text-xs"
|
||||||
|
>
|
||||||
|
<span className="font-mono text-muted-foreground truncate">
|
||||||
|
{key}
|
||||||
|
</span>
|
||||||
|
<span className="font-mono truncate">
|
||||||
|
{maskValue(key, String(value))}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Config URL (if remote) */}
|
||||||
|
{request.configUrl && (
|
||||||
|
<div className="grid grid-cols-3 items-center gap-4">
|
||||||
|
<div className="font-medium text-sm text-muted-foreground">
|
||||||
|
{t("deeplink.configUrl")}
|
||||||
|
</div>
|
||||||
|
<div className="col-span-2 text-sm font-mono text-muted-foreground break-all">
|
||||||
|
{request.configUrl}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Warning */}
|
||||||
|
<div className="rounded-lg bg-yellow-50 dark:bg-yellow-900/20 p-3 text-sm text-yellow-800 dark:text-yellow-200">
|
||||||
|
{t("deeplink.warning")}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)}
|
|
||||||
|
|
||||||
{/* Notes (if present) */}
|
<DialogFooter>
|
||||||
{request.notes && (
|
<Button
|
||||||
<div className="grid grid-cols-3 items-start gap-4">
|
variant="outline"
|
||||||
<div className="font-medium text-sm text-muted-foreground">
|
onClick={handleCancel}
|
||||||
{t("deeplink.notes")}
|
disabled={isImporting}
|
||||||
</div>
|
>
|
||||||
<div className="col-span-2 text-sm text-muted-foreground">
|
{t("common.cancel")}
|
||||||
{request.notes}
|
</Button>
|
||||||
</div>
|
<Button onClick={handleImport} disabled={isImporting}>
|
||||||
</div>
|
{isImporting ? t("deeplink.importing") : t("deeplink.import")}
|
||||||
)}
|
</Button>
|
||||||
|
</DialogFooter>
|
||||||
{/* Warning */}
|
</>
|
||||||
<div className="rounded-lg bg-yellow-50 dark:bg-yellow-900/20 p-3 text-sm text-yellow-800 dark:text-yellow-200">
|
)}
|
||||||
{t("deeplink.warning")}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<DialogFooter>
|
|
||||||
<Button
|
|
||||||
variant="outline"
|
|
||||||
onClick={handleCancel}
|
|
||||||
disabled={isImporting}
|
|
||||||
>
|
|
||||||
{t("common.cancel")}
|
|
||||||
</Button>
|
|
||||||
<Button onClick={handleImport} disabled={isImporting}>
|
|
||||||
{isImporting ? t("deeplink.importing") : t("deeplink.import")}
|
|
||||||
</Button>
|
|
||||||
</DialogFooter>
|
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
);
|
);
|
||||||
|
|||||||
85
src/components/IconPicker.tsx
Normal file
85
src/components/IconPicker.tsx
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
import React, { useState, useMemo } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { Label } from "@/components/ui/label";
|
||||||
|
import { ProviderIcon } from "./ProviderIcon";
|
||||||
|
import { iconList } from "@/icons/extracted";
|
||||||
|
import { searchIcons, getIconMetadata } from "@/icons/extracted/metadata";
|
||||||
|
import { cn } from "@/lib/utils";
|
||||||
|
|
||||||
|
interface IconPickerProps {
|
||||||
|
value?: string; // 当前选中的图标
|
||||||
|
onValueChange: (icon: string) => void; // 选择回调
|
||||||
|
color?: string; // 预览颜色
|
||||||
|
}
|
||||||
|
|
||||||
|
export const IconPicker: React.FC<IconPickerProps> = ({
|
||||||
|
value,
|
||||||
|
onValueChange,
|
||||||
|
}) => {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [searchQuery, setSearchQuery] = useState("");
|
||||||
|
|
||||||
|
// 过滤图标列表
|
||||||
|
const filteredIcons = useMemo(() => {
|
||||||
|
if (!searchQuery) return iconList;
|
||||||
|
return searchIcons(searchQuery);
|
||||||
|
}, [searchQuery]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="space-y-4">
|
||||||
|
<div>
|
||||||
|
<Label htmlFor="icon-search">
|
||||||
|
{t("iconPicker.search", { defaultValue: "搜索图标" })}
|
||||||
|
</Label>
|
||||||
|
<Input
|
||||||
|
id="icon-search"
|
||||||
|
type="text"
|
||||||
|
placeholder={t("iconPicker.searchPlaceholder", {
|
||||||
|
defaultValue: "输入图标名称...",
|
||||||
|
})}
|
||||||
|
value={searchQuery}
|
||||||
|
onChange={(e) => setSearchQuery(e.target.value)}
|
||||||
|
className="mt-2"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="max-h-[65vh] overflow-y-auto pr-1">
|
||||||
|
<div className="grid grid-cols-6 sm:grid-cols-8 lg:grid-cols-10 gap-2">
|
||||||
|
{filteredIcons.map((iconName) => {
|
||||||
|
const meta = getIconMetadata(iconName);
|
||||||
|
const isSelected = value === iconName;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<button
|
||||||
|
key={iconName}
|
||||||
|
type="button"
|
||||||
|
onClick={() => onValueChange(iconName)}
|
||||||
|
className={cn(
|
||||||
|
"flex flex-col items-center gap-1 p-3 rounded-lg",
|
||||||
|
"border-2 transition-all duration-200",
|
||||||
|
"hover:bg-accent hover:border-primary/50",
|
||||||
|
isSelected
|
||||||
|
? "border-primary bg-primary/10"
|
||||||
|
: "border-transparent",
|
||||||
|
)}
|
||||||
|
title={meta?.displayName || iconName}
|
||||||
|
>
|
||||||
|
<ProviderIcon icon={iconName} name={iconName} size={32} />
|
||||||
|
<span className="text-xs text-muted-foreground truncate w-full text-center">
|
||||||
|
{meta?.displayName || iconName}
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{filteredIcons.length === 0 && (
|
||||||
|
<div className="text-center py-8 text-muted-foreground">
|
||||||
|
{t("iconPicker.noResults", { defaultValue: "未找到匹配的图标" })}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -12,6 +12,7 @@ import { toast } from "sonner";
|
|||||||
import { formatJSON } from "@/utils/formatters";
|
import { formatJSON } from "@/utils/formatters";
|
||||||
|
|
||||||
interface JsonEditorProps {
|
interface JsonEditorProps {
|
||||||
|
id?: string;
|
||||||
value: string;
|
value: string;
|
||||||
onChange: (value: string) => void;
|
onChange: (value: string) => void;
|
||||||
placeholder?: string;
|
placeholder?: string;
|
||||||
@@ -19,7 +20,8 @@ interface JsonEditorProps {
|
|||||||
rows?: number;
|
rows?: number;
|
||||||
showValidation?: boolean;
|
showValidation?: boolean;
|
||||||
language?: "json" | "javascript";
|
language?: "json" | "javascript";
|
||||||
height?: string;
|
height?: string | number;
|
||||||
|
showMinimap?: boolean; // 添加此属性以防未来使用
|
||||||
}
|
}
|
||||||
|
|
||||||
const JsonEditor: React.FC<JsonEditorProps> = ({
|
const JsonEditor: React.FC<JsonEditorProps> = ({
|
||||||
@@ -84,19 +86,47 @@ const JsonEditor: React.FC<JsonEditorProps> = ({
|
|||||||
|
|
||||||
// 使用 baseTheme 定义基础样式,优先级低于 oneDark,但可以正确响应主题
|
// 使用 baseTheme 定义基础样式,优先级低于 oneDark,但可以正确响应主题
|
||||||
const baseTheme = EditorView.baseTheme({
|
const baseTheme = EditorView.baseTheme({
|
||||||
"&light .cm-editor, &dark .cm-editor": {
|
".cm-editor": {
|
||||||
border: "1px solid hsl(var(--border))",
|
border: "1px solid hsl(var(--border))",
|
||||||
borderRadius: "0.5rem",
|
borderRadius: "0.5rem",
|
||||||
|
background: "transparent",
|
||||||
},
|
},
|
||||||
"&light .cm-editor.cm-focused, &dark .cm-editor.cm-focused": {
|
".cm-editor.cm-focused": {
|
||||||
outline: "none",
|
outline: "none",
|
||||||
borderColor: "hsl(var(--primary))",
|
borderColor: "hsl(var(--primary))",
|
||||||
},
|
},
|
||||||
|
".cm-scroller": {
|
||||||
|
background: "transparent",
|
||||||
|
},
|
||||||
|
".cm-gutters": {
|
||||||
|
background: "transparent",
|
||||||
|
borderRight: "1px solid hsl(var(--border))",
|
||||||
|
color: "hsl(var(--muted-foreground))",
|
||||||
|
},
|
||||||
|
".cm-selectionBackground, .cm-content ::selection": {
|
||||||
|
background: "hsl(var(--primary) / 0.18)",
|
||||||
|
},
|
||||||
|
".cm-selectionMatch": {
|
||||||
|
background: "hsl(var(--primary) / 0.12)",
|
||||||
|
},
|
||||||
|
".cm-activeLine": {
|
||||||
|
background: "hsl(var(--primary) / 0.08)",
|
||||||
|
},
|
||||||
|
".cm-activeLineGutter": {
|
||||||
|
background: "hsl(var(--primary) / 0.08)",
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
// 使用 theme 定义尺寸和字体样式
|
// 使用 theme 定义尺寸和字体样式
|
||||||
|
const heightValue = height
|
||||||
|
? typeof height === "number"
|
||||||
|
? `${height}px`
|
||||||
|
: height
|
||||||
|
: undefined;
|
||||||
const sizingTheme = EditorView.theme({
|
const sizingTheme = EditorView.theme({
|
||||||
"&": height ? { height } : { minHeight: `${minHeightPx}px` },
|
"&": heightValue
|
||||||
|
? { height: heightValue }
|
||||||
|
: { minHeight: `${minHeightPx}px` },
|
||||||
".cm-scroller": { overflow: "auto" },
|
".cm-scroller": { overflow: "auto" },
|
||||||
".cm-content": {
|
".cm-content": {
|
||||||
fontFamily:
|
fontFamily:
|
||||||
@@ -129,11 +159,32 @@ const JsonEditor: React.FC<JsonEditorProps> = ({
|
|||||||
".cm-editor": {
|
".cm-editor": {
|
||||||
border: "1px solid hsl(var(--border))",
|
border: "1px solid hsl(var(--border))",
|
||||||
borderRadius: "0.5rem",
|
borderRadius: "0.5rem",
|
||||||
|
background: "transparent",
|
||||||
},
|
},
|
||||||
".cm-editor.cm-focused": {
|
".cm-editor.cm-focused": {
|
||||||
outline: "none",
|
outline: "none",
|
||||||
borderColor: "hsl(var(--primary))",
|
borderColor: "hsl(var(--primary))",
|
||||||
},
|
},
|
||||||
|
".cm-scroller": {
|
||||||
|
background: "transparent",
|
||||||
|
},
|
||||||
|
".cm-gutters": {
|
||||||
|
background: "transparent",
|
||||||
|
borderRight: "1px solid hsl(var(--border))",
|
||||||
|
color: "hsl(var(--muted-foreground))",
|
||||||
|
},
|
||||||
|
".cm-selectionBackground, .cm-content ::selection": {
|
||||||
|
background: "hsl(var(--primary) / 0.18)",
|
||||||
|
},
|
||||||
|
".cm-selectionMatch": {
|
||||||
|
background: "hsl(var(--primary) / 0.12)",
|
||||||
|
},
|
||||||
|
".cm-activeLine": {
|
||||||
|
background: "hsl(var(--primary) / 0.08)",
|
||||||
|
},
|
||||||
|
".cm-activeLineGutter": {
|
||||||
|
background: "hsl(var(--primary) / 0.08)",
|
||||||
|
},
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -196,14 +247,23 @@ const JsonEditor: React.FC<JsonEditorProps> = ({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const isFullHeight = height === "100%";
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div style={{ width: "100%" }}>
|
<div
|
||||||
<div ref={editorRef} style={{ width: "100%" }} />
|
style={{ width: "100%", height: isFullHeight ? "100%" : "auto" }}
|
||||||
|
className={isFullHeight ? "flex flex-col" : ""}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
ref={editorRef}
|
||||||
|
style={{ width: "100%", height: isFullHeight ? undefined : "auto" }}
|
||||||
|
className={isFullHeight ? "flex-1 min-h-0" : ""}
|
||||||
|
/>
|
||||||
{language === "json" && (
|
{language === "json" && (
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
onClick={handleFormat}
|
onClick={handleFormat}
|
||||||
className="mt-2 inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium text-gray-700 dark:text-gray-300 hover:text-blue-600 dark:hover:text-blue-400 transition-colors"
|
className={`${isFullHeight ? "mt-2 flex-shrink-0" : "mt-2"} inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium text-gray-700 dark:text-gray-300 hover:text-blue-600 dark:hover:text-blue-400 transition-colors`}
|
||||||
>
|
>
|
||||||
<Wand2 className="w-3.5 h-3.5" />
|
<Wand2 className="w-3.5 h-3.5" />
|
||||||
{t("common.format", { defaultValue: "格式化" })}
|
{t("common.format", { defaultValue: "格式化" })}
|
||||||
|
|||||||
81
src/components/ProviderIcon.tsx
Normal file
81
src/components/ProviderIcon.tsx
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
import React, { useMemo } from "react";
|
||||||
|
import { getIcon, hasIcon } from "@/icons/extracted";
|
||||||
|
import { cn } from "@/lib/utils";
|
||||||
|
|
||||||
|
interface ProviderIconProps {
|
||||||
|
icon?: string; // 图标名称
|
||||||
|
name: string; // 供应商名称(用于 fallback)
|
||||||
|
color?: string; // 自定义颜色 (Deprecated, kept for compatibility but ignored for SVG)
|
||||||
|
size?: number | string; // 尺寸
|
||||||
|
className?: string;
|
||||||
|
showFallback?: boolean; // 是否显示 fallback
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ProviderIcon: React.FC<ProviderIconProps> = ({
|
||||||
|
icon,
|
||||||
|
name,
|
||||||
|
size = 32,
|
||||||
|
className,
|
||||||
|
showFallback = true,
|
||||||
|
}) => {
|
||||||
|
// 获取图标 SVG
|
||||||
|
const iconSvg = useMemo(() => {
|
||||||
|
if (icon && hasIcon(icon)) {
|
||||||
|
return getIcon(icon);
|
||||||
|
}
|
||||||
|
return "";
|
||||||
|
}, [icon]);
|
||||||
|
|
||||||
|
// 计算尺寸样式
|
||||||
|
const sizeStyle = useMemo(() => {
|
||||||
|
const sizeValue = typeof size === "number" ? `${size}px` : size;
|
||||||
|
return {
|
||||||
|
width: sizeValue,
|
||||||
|
height: sizeValue,
|
||||||
|
};
|
||||||
|
}, [size]);
|
||||||
|
|
||||||
|
// 如果有图标,显示图标
|
||||||
|
if (iconSvg) {
|
||||||
|
return (
|
||||||
|
<span
|
||||||
|
className={cn(
|
||||||
|
"inline-flex items-center justify-center flex-shrink-0",
|
||||||
|
className,
|
||||||
|
)}
|
||||||
|
style={sizeStyle}
|
||||||
|
dangerouslySetInnerHTML={{ __html: iconSvg }}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback:显示首字母
|
||||||
|
if (showFallback) {
|
||||||
|
const initials = name
|
||||||
|
.split(" ")
|
||||||
|
.map((word) => word[0])
|
||||||
|
.join("")
|
||||||
|
.toUpperCase()
|
||||||
|
.slice(0, 2);
|
||||||
|
return (
|
||||||
|
<span
|
||||||
|
className={cn(
|
||||||
|
"inline-flex items-center justify-center flex-shrink-0 rounded-lg",
|
||||||
|
"bg-muted text-muted-foreground font-semibold",
|
||||||
|
className,
|
||||||
|
)}
|
||||||
|
style={sizeStyle}
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
style={{
|
||||||
|
fontSize: `${typeof size === "number" ? size * 0.4 : 14}px`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{initials}
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
};
|
||||||
@@ -60,7 +60,7 @@ const UsageFooter: React.FC<UsageFooterProps> = ({
|
|||||||
if (!usage.success) {
|
if (!usage.success) {
|
||||||
if (inline) {
|
if (inline) {
|
||||||
return (
|
return (
|
||||||
<div className="flex items-center gap-2 text-xs">
|
<div className="inline-flex items-center gap-2 text-xs rounded-lg border border-border-default bg-card px-3 py-2 shadow-sm">
|
||||||
<div className="flex items-center gap-1.5 text-red-500 dark:text-red-400">
|
<div className="flex items-center gap-1.5 text-red-500 dark:text-red-400">
|
||||||
<AlertCircle size={12} />
|
<AlertCircle size={12} />
|
||||||
<span>{t("usage.queryFailed")}</span>
|
<span>{t("usage.queryFailed")}</span>
|
||||||
@@ -68,7 +68,7 @@ const UsageFooter: React.FC<UsageFooterProps> = ({
|
|||||||
<button
|
<button
|
||||||
onClick={() => refetch()}
|
onClick={() => refetch()}
|
||||||
disabled={loading}
|
disabled={loading}
|
||||||
className="p-1 rounded hover:bg-gray-100 dark:hover:bg-gray-800 transition-colors disabled:opacity-50 flex-shrink-0"
|
className="p-1 rounded hover:bg-muted transition-colors disabled:opacity-50 flex-shrink-0"
|
||||||
title={t("usage.refreshUsage")}
|
title={t("usage.refreshUsage")}
|
||||||
>
|
>
|
||||||
<RefreshCw size={12} className={loading ? "animate-spin" : ""} />
|
<RefreshCw size={12} className={loading ? "animate-spin" : ""} />
|
||||||
@@ -78,7 +78,7 @@ const UsageFooter: React.FC<UsageFooterProps> = ({
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="mt-3 pt-3 border-t border-border-default ">
|
<div className="mt-3 rounded-xl border border-border-default bg-card px-4 py-3 shadow-sm">
|
||||||
<div className="flex items-center justify-between gap-2 text-xs">
|
<div className="flex items-center justify-between gap-2 text-xs">
|
||||||
<div className="flex items-center gap-2 text-red-500 dark:text-red-400">
|
<div className="flex items-center gap-2 text-red-500 dark:text-red-400">
|
||||||
<AlertCircle size={14} />
|
<AlertCircle size={14} />
|
||||||
@@ -110,29 +110,32 @@ const UsageFooter: React.FC<UsageFooterProps> = ({
|
|||||||
const isExpired = firstUsage.isValid === false;
|
const isExpired = firstUsage.isValid === false;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex flex-col gap-1 text-xs flex-shrink-0">
|
<div className="flex flex-col items-end gap-1 text-xs whitespace-nowrap flex-shrink-0">
|
||||||
{/* 第一行:刷新时间 + 刷新按钮 */}
|
{/* 第一行:更新时间和刷新按钮 */}
|
||||||
<div className="flex items-center gap-2 justify-end">
|
<div className="flex items-center gap-2 justify-end">
|
||||||
{/* 上次查询时间 */}
|
{/* 上次查询时间 */}
|
||||||
{lastQueriedAt && (
|
<span className="text-[10px] text-gray-400 dark:text-gray-500 flex items-center gap-1">
|
||||||
<span className="text-[10px] text-gray-400 dark:text-gray-500 flex items-center gap-1">
|
<Clock size={10} />
|
||||||
<Clock size={10} />
|
{lastQueriedAt
|
||||||
{formatRelativeTime(lastQueriedAt, now, t)}
|
? formatRelativeTime(lastQueriedAt, now, t)
|
||||||
</span>
|
: t("usage.never", { defaultValue: "从未更新" })}
|
||||||
)}
|
</span>
|
||||||
|
|
||||||
{/* 刷新按钮 */}
|
{/* 刷新按钮 */}
|
||||||
<button
|
<button
|
||||||
onClick={() => refetch()}
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
refetch();
|
||||||
|
}}
|
||||||
disabled={loading}
|
disabled={loading}
|
||||||
className="p-1 rounded hover:bg-gray-100 dark:hover:bg-gray-800 transition-colors disabled:opacity-50 flex-shrink-0"
|
className="p-1 rounded hover:bg-gray-100 dark:hover:bg-gray-800 transition-colors disabled:opacity-50 flex-shrink-0 text-gray-400 dark:text-gray-500"
|
||||||
title={t("usage.refreshUsage")}
|
title={t("usage.refreshUsage")}
|
||||||
>
|
>
|
||||||
<RefreshCw size={12} className={loading ? "animate-spin" : ""} />
|
<RefreshCw size={12} className={loading ? "animate-spin" : ""} />
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* 第二行:已用 + 剩余 + 单位 */}
|
{/* 第二行:用量和剩余 */}
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
{/* 已用 */}
|
{/* 已用 */}
|
||||||
{firstUsage.used !== undefined && (
|
{firstUsage.used !== undefined && (
|
||||||
@@ -179,7 +182,7 @@ const UsageFooter: React.FC<UsageFooterProps> = ({
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="mt-3 pt-3 border-t border-border-default ">
|
<div className="mt-3 rounded-xl border border-border-default bg-card px-4 py-3 shadow-sm">
|
||||||
{/* 标题行:包含刷新按钮和自动查询时间 */}
|
{/* 标题行:包含刷新按钮和自动查询时间 */}
|
||||||
<div className="flex items-center justify-between mb-2">
|
<div className="flex items-center justify-between mb-2">
|
||||||
<span className="text-xs text-gray-500 dark:text-gray-400 font-medium">
|
<span className="text-xs text-gray-500 dark:text-gray-400 font-medium">
|
||||||
@@ -196,7 +199,7 @@ const UsageFooter: React.FC<UsageFooterProps> = ({
|
|||||||
<button
|
<button
|
||||||
onClick={() => refetch()}
|
onClick={() => refetch()}
|
||||||
disabled={loading}
|
disabled={loading}
|
||||||
className="p-1 rounded hover:bg-gray-100 dark:hover:bg-gray-800 transition-colors disabled:opacity-50"
|
className="p-1 rounded hover:bg-muted transition-colors disabled:opacity-50"
|
||||||
title={t("usage.refreshUsage")}
|
title={t("usage.refreshUsage")}
|
||||||
>
|
>
|
||||||
<RefreshCw size={12} className={loading ? "animate-spin" : ""} />
|
<RefreshCw size={12} className={loading ? "animate-spin" : ""} />
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { Play, Wand2, Eye, EyeOff } from "lucide-react";
|
import { Play, Wand2, Eye, EyeOff, Save } from "lucide-react";
|
||||||
import { toast } from "sonner";
|
import { toast } from "sonner";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { Provider, UsageScript } from "@/types";
|
import { Provider, UsageScript } from "@/types";
|
||||||
@@ -8,17 +8,12 @@ import JsonEditor from "./JsonEditor";
|
|||||||
import * as prettier from "prettier/standalone";
|
import * as prettier from "prettier/standalone";
|
||||||
import * as parserBabel from "prettier/parser-babel";
|
import * as parserBabel from "prettier/parser-babel";
|
||||||
import * as pluginEstree from "prettier/plugins/estree";
|
import * as pluginEstree from "prettier/plugins/estree";
|
||||||
import {
|
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
DialogFooter,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { Input } from "@/components/ui/input";
|
import { Input } from "@/components/ui/input";
|
||||||
import { Label } from "@/components/ui/label";
|
import { Label } from "@/components/ui/label";
|
||||||
import { Switch } from "@/components/ui/switch";
|
import { Switch } from "@/components/ui/switch";
|
||||||
|
import { FullScreenPanel } from "@/components/common/FullScreenPanel";
|
||||||
|
import { cn } from "@/lib/utils";
|
||||||
|
|
||||||
interface UsageScriptModalProps {
|
interface UsageScriptModalProps {
|
||||||
provider: Provider;
|
provider: Provider;
|
||||||
@@ -131,88 +126,53 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
|
|
||||||
const [testing, setTesting] = useState(false);
|
const [testing, setTesting] = useState(false);
|
||||||
|
|
||||||
// 🔧 输入时的格式化(宽松)- 只清理格式,不约束范围
|
|
||||||
const sanitizeNumberInput = (value: string): string => {
|
|
||||||
// 移除所有非数字字符
|
|
||||||
let cleaned = value.replace(/[^\d]/g, "");
|
|
||||||
|
|
||||||
// 移除前导零(除非输入的就是 "0")
|
|
||||||
if (cleaned.length > 1 && cleaned.startsWith("0")) {
|
|
||||||
cleaned = cleaned.replace(/^0+/, "");
|
|
||||||
}
|
|
||||||
|
|
||||||
return cleaned;
|
|
||||||
};
|
|
||||||
|
|
||||||
// 🔧 失焦时的验证(严格)- 仅确保有效整数
|
// 🔧 失焦时的验证(严格)- 仅确保有效整数
|
||||||
const validateTimeout = (value: string): number => {
|
const validateTimeout = (value: string): number => {
|
||||||
// 转换为数字
|
|
||||||
const num = Number(value);
|
const num = Number(value);
|
||||||
|
|
||||||
// 检查是否为有效数字
|
|
||||||
if (isNaN(num) || value.trim() === "") {
|
if (isNaN(num) || value.trim() === "") {
|
||||||
return 10; // 默认值
|
return 10;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 检查是否为整数
|
|
||||||
if (!Number.isInteger(num)) {
|
if (!Number.isInteger(num)) {
|
||||||
toast.warning(
|
toast.warning(
|
||||||
t("usageScript.timeoutMustBeInteger") || "超时时间必须为整数",
|
t("usageScript.timeoutMustBeInteger") || "超时时间必须为整数",
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 检查负数
|
|
||||||
if (num < 0) {
|
if (num < 0) {
|
||||||
toast.error(
|
toast.error(
|
||||||
t("usageScript.timeoutCannotBeNegative") || "超时时间不能为负数",
|
t("usageScript.timeoutCannotBeNegative") || "超时时间不能为负数",
|
||||||
);
|
);
|
||||||
return 10;
|
return 10;
|
||||||
}
|
}
|
||||||
|
|
||||||
return Math.floor(num);
|
return Math.floor(num);
|
||||||
};
|
};
|
||||||
|
|
||||||
// 🔧 失焦时的验证(严格)- 自动查询间隔
|
// 🔧 失焦时的验证(严格)- 自动查询间隔
|
||||||
const validateAndClampInterval = (value: string): number => {
|
const validateAndClampInterval = (value: string): number => {
|
||||||
// 转换为数字
|
|
||||||
const num = Number(value);
|
const num = Number(value);
|
||||||
|
|
||||||
// 检查是否为有效数字
|
|
||||||
if (isNaN(num) || value.trim() === "") {
|
if (isNaN(num) || value.trim() === "") {
|
||||||
return 0; // 禁用自动查询
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 检查是否为整数
|
|
||||||
if (!Number.isInteger(num)) {
|
if (!Number.isInteger(num)) {
|
||||||
toast.warning(
|
toast.warning(
|
||||||
t("usageScript.intervalMustBeInteger") || "自动查询间隔必须为整数",
|
t("usageScript.intervalMustBeInteger") || "自动查询间隔必须为整数",
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 检查负数
|
|
||||||
if (num < 0) {
|
if (num < 0) {
|
||||||
toast.error(
|
toast.error(
|
||||||
t("usageScript.intervalCannotBeNegative") || "自动查询间隔不能为负数",
|
t("usageScript.intervalCannotBeNegative") || "自动查询间隔不能为负数",
|
||||||
);
|
);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 约束到 [0, 1440] 范围(最大24小时)
|
|
||||||
const clamped = Math.max(0, Math.min(1440, Math.floor(num)));
|
const clamped = Math.max(0, Math.min(1440, Math.floor(num)));
|
||||||
|
|
||||||
// 如果值被调整,显示提示
|
|
||||||
if (clamped !== num && num > 0) {
|
if (clamped !== num && num > 0) {
|
||||||
toast.info(
|
toast.info(
|
||||||
t("usageScript.intervalAdjusted", { value: clamped }) ||
|
t("usageScript.intervalAdjusted", { value: clamped }) ||
|
||||||
`自动查询间隔已调整为 ${clamped} 分钟`,
|
`自动查询间隔已调整为 ${clamped} 分钟`,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return clamped;
|
return clamped;
|
||||||
};
|
};
|
||||||
|
|
||||||
// 跟踪当前选择的模板类型(用于控制高级配置的显示)
|
|
||||||
// 初始化:如果已有 accessToken 或 userId,说明是 NewAPI 模板
|
|
||||||
const [selectedTemplate, setSelectedTemplate] = useState<string | null>(
|
const [selectedTemplate, setSelectedTemplate] = useState<string | null>(
|
||||||
() => {
|
() => {
|
||||||
const existingScript = provider.meta?.usage_script;
|
const existingScript = provider.meta?.usage_script;
|
||||||
@@ -223,23 +183,18 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
// 控制 API Key 的显示/隐藏
|
|
||||||
const [showApiKey, setShowApiKey] = useState(false);
|
const [showApiKey, setShowApiKey] = useState(false);
|
||||||
const [showAccessToken, setShowAccessToken] = useState(false);
|
const [showAccessToken, setShowAccessToken] = useState(false);
|
||||||
|
|
||||||
const handleSave = () => {
|
const handleSave = () => {
|
||||||
// 验证脚本格式
|
|
||||||
if (script.enabled && !script.code.trim()) {
|
if (script.enabled && !script.code.trim()) {
|
||||||
toast.error(t("usageScript.scriptEmpty"));
|
toast.error(t("usageScript.scriptEmpty"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 基本的 JS 语法检查(检查是否包含 return 语句)
|
|
||||||
if (script.enabled && !script.code.includes("return")) {
|
if (script.enabled && !script.code.includes("return")) {
|
||||||
toast.error(t("usageScript.mustHaveReturn"), { duration: 5000 });
|
toast.error(t("usageScript.mustHaveReturn"), { duration: 5000 });
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
onSave(script);
|
onSave(script);
|
||||||
onClose();
|
onClose();
|
||||||
};
|
};
|
||||||
@@ -247,7 +202,6 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
const handleTest = async () => {
|
const handleTest = async () => {
|
||||||
setTesting(true);
|
setTesting(true);
|
||||||
try {
|
try {
|
||||||
// 使用当前编辑器中的脚本内容进行测试
|
|
||||||
const result = await usageApi.testScript(
|
const result = await usageApi.testScript(
|
||||||
provider.id,
|
provider.id,
|
||||||
appId,
|
appId,
|
||||||
@@ -259,7 +213,6 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
script.userId,
|
script.userId,
|
||||||
);
|
);
|
||||||
if (result.success && result.data && result.data.length > 0) {
|
if (result.success && result.data && result.data.length > 0) {
|
||||||
// 显示所有套餐数据
|
|
||||||
const summary = result.data
|
const summary = result.data
|
||||||
.map((plan) => {
|
.map((plan) => {
|
||||||
const planInfo = plan.planName ? `[${plan.planName}]` : "";
|
const planInfo = plan.planName ? `[${plan.planName}]` : "";
|
||||||
@@ -314,9 +267,7 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
const handleUsePreset = (presetName: string) => {
|
const handleUsePreset = (presetName: string) => {
|
||||||
const preset = PRESET_TEMPLATES[presetName];
|
const preset = PRESET_TEMPLATES[presetName];
|
||||||
if (preset) {
|
if (preset) {
|
||||||
// 根据模板类型清空不同的字段
|
|
||||||
if (presetName === TEMPLATE_KEYS.CUSTOM) {
|
if (presetName === TEMPLATE_KEYS.CUSTOM) {
|
||||||
// 自定义:清空所有凭证字段
|
|
||||||
setScript({
|
setScript({
|
||||||
...script,
|
...script,
|
||||||
code: preset,
|
code: preset,
|
||||||
@@ -326,7 +277,6 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
userId: undefined,
|
userId: undefined,
|
||||||
});
|
});
|
||||||
} else if (presetName === TEMPLATE_KEYS.GENERAL) {
|
} else if (presetName === TEMPLATE_KEYS.GENERAL) {
|
||||||
// 通用:保留 apiKey 和 baseUrl,清空 NewAPI 字段
|
|
||||||
setScript({
|
setScript({
|
||||||
...script,
|
...script,
|
||||||
code: preset,
|
code: preset,
|
||||||
@@ -334,84 +284,131 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
userId: undefined,
|
userId: undefined,
|
||||||
});
|
});
|
||||||
} else if (presetName === TEMPLATE_KEYS.NEW_API) {
|
} else if (presetName === TEMPLATE_KEYS.NEW_API) {
|
||||||
// NewAPI:清空 apiKey(NewAPI 不使用通用的 apiKey)
|
|
||||||
setScript({
|
setScript({
|
||||||
...script,
|
...script,
|
||||||
code: preset,
|
code: preset,
|
||||||
apiKey: undefined,
|
apiKey: undefined,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
setSelectedTemplate(presetName); // 记录选择的模板
|
setSelectedTemplate(presetName);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// 判断是否应该显示凭证配置区域
|
|
||||||
const shouldShowCredentialsConfig =
|
const shouldShowCredentialsConfig =
|
||||||
selectedTemplate === TEMPLATE_KEYS.GENERAL ||
|
selectedTemplate === TEMPLATE_KEYS.GENERAL ||
|
||||||
selectedTemplate === TEMPLATE_KEYS.NEW_API;
|
selectedTemplate === TEMPLATE_KEYS.NEW_API;
|
||||||
|
|
||||||
|
const footer = (
|
||||||
|
<>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Button
|
||||||
|
variant="secondary"
|
||||||
|
size="sm"
|
||||||
|
onClick={handleTest}
|
||||||
|
disabled={!script.enabled || testing}
|
||||||
|
>
|
||||||
|
<Play size={14} className="mr-1" />
|
||||||
|
{testing ? t("usageScript.testing") : t("usageScript.testScript")}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
onClick={handleFormat}
|
||||||
|
disabled={!script.enabled}
|
||||||
|
title={t("usageScript.format")}
|
||||||
|
>
|
||||||
|
<Wand2 size={14} className="mr-1" />
|
||||||
|
{t("usageScript.format")}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
onClick={onClose}
|
||||||
|
className="border-border/20 hover:bg-accent hover:text-accent-foreground"
|
||||||
|
>
|
||||||
|
{t("common.cancel")}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
onClick={handleSave}
|
||||||
|
className="bg-primary text-primary-foreground hover:bg-primary/90"
|
||||||
|
>
|
||||||
|
<Save size={16} className="mr-2" />
|
||||||
|
{t("usageScript.saveConfig")}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog open={isOpen} onOpenChange={(open) => !open && onClose()}>
|
<FullScreenPanel
|
||||||
<DialogContent className="max-w-3xl max-h-[90vh] flex flex-col">
|
isOpen={isOpen}
|
||||||
<DialogHeader>
|
title={`${t("usageScript.title")} - ${provider.name}`}
|
||||||
<DialogTitle>
|
onClose={onClose}
|
||||||
{t("usageScript.title")} - {provider.name}
|
footer={footer}
|
||||||
</DialogTitle>
|
>
|
||||||
</DialogHeader>
|
<div className="glass rounded-xl border border-white/10 px-6 py-4 flex items-center justify-between gap-4">
|
||||||
|
<div className="space-y-1">
|
||||||
|
<p className="text-sm font-medium leading-none text-foreground">
|
||||||
|
{t("usageScript.enableUsageQuery")}
|
||||||
|
</p>
|
||||||
|
<p className="text-xs text-muted-foreground">
|
||||||
|
{t("usageScript.autoQueryIntervalHint")}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<Switch
|
||||||
|
checked={script.enabled}
|
||||||
|
onCheckedChange={(checked) =>
|
||||||
|
setScript({ ...script, enabled: checked })
|
||||||
|
}
|
||||||
|
aria-label={t("usageScript.enableUsageQuery")}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* Content - Scrollable */}
|
{script.enabled && (
|
||||||
<div className="flex-1 overflow-y-auto px-6 py-4 space-y-4">
|
<div className="space-y-6">
|
||||||
{/* 启用开关 */}
|
{/* 预设模板选择 */}
|
||||||
<div className="flex items-center justify-between gap-4 rounded-lg border border-border-default p-4">
|
<div className="space-y-4 glass rounded-xl border border-white/10 p-6">
|
||||||
<div className="space-y-1">
|
<div className="flex flex-wrap items-center justify-between gap-2">
|
||||||
<p className="text-sm font-medium leading-none">
|
<Label className="text-base font-medium">
|
||||||
{t("usageScript.enableUsageQuery")}
|
{t("usageScript.presetTemplate")}
|
||||||
</p>
|
</Label>
|
||||||
|
<span className="text-xs text-muted-foreground">
|
||||||
|
{t("usageScript.variablesHint")}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className="flex gap-2 flex-wrap">
|
||||||
|
{Object.keys(PRESET_TEMPLATES).map((name) => {
|
||||||
|
const isSelected = selectedTemplate === name;
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
key={name}
|
||||||
|
type="button"
|
||||||
|
variant={isSelected ? "default" : "outline"}
|
||||||
|
size="sm"
|
||||||
|
className={cn(
|
||||||
|
"rounded-lg border",
|
||||||
|
isSelected
|
||||||
|
? "shadow-sm"
|
||||||
|
: "bg-background text-muted-foreground hover:bg-accent hover:text-accent-foreground",
|
||||||
|
)}
|
||||||
|
onClick={() => handleUsePreset(name)}
|
||||||
|
>
|
||||||
|
{t(TEMPLATE_NAME_KEYS[name])}
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
})}
|
||||||
</div>
|
</div>
|
||||||
<Switch
|
|
||||||
checked={script.enabled}
|
|
||||||
onCheckedChange={(checked) =>
|
|
||||||
setScript({ ...script, enabled: checked })
|
|
||||||
}
|
|
||||||
aria-label={t("usageScript.enableUsageQuery")}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{script.enabled && (
|
{/* 凭证配置 */}
|
||||||
<>
|
{shouldShowCredentialsConfig && (
|
||||||
{/* 预设模板选择 */}
|
<div className="space-y-4">
|
||||||
<div>
|
<h4 className="text-sm font-medium text-foreground">
|
||||||
<Label className="mb-2">
|
{t("usageScript.credentialsConfig")}
|
||||||
{t("usageScript.presetTemplate")}
|
</h4>
|
||||||
</Label>
|
|
||||||
<div className="flex gap-2">
|
|
||||||
{Object.keys(PRESET_TEMPLATES).map((name) => {
|
|
||||||
const isSelected = selectedTemplate === name;
|
|
||||||
return (
|
|
||||||
<button
|
|
||||||
key={name}
|
|
||||||
onClick={() => handleUsePreset(name)}
|
|
||||||
className={`px-3 py-1.5 text-xs rounded transition-colors ${
|
|
||||||
isSelected
|
|
||||||
? "bg-blue-500 text-white dark:bg-blue-600"
|
|
||||||
: "bg-gray-100 dark:bg-gray-800 text-gray-500 dark:text-gray-400 hover:bg-gray-200 dark:hover:bg-gray-700"
|
|
||||||
}`}
|
|
||||||
>
|
|
||||||
{t(TEMPLATE_NAME_KEYS[name])}
|
|
||||||
</button>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* 凭证配置区域:通用和 NewAPI 模板显示 */}
|
<div className="grid gap-4 md:grid-cols-2">
|
||||||
{shouldShowCredentialsConfig && (
|
|
||||||
<div className="space-y-4 p-4 bg-gray-50 dark:bg-gray-800/50 rounded-lg">
|
|
||||||
<h4 className="text-sm font-medium text-gray-900 dark:text-gray-100">
|
|
||||||
{t("usageScript.credentialsConfig")}
|
|
||||||
</h4>
|
|
||||||
|
|
||||||
{/* 通用模板:显示 apiKey + baseUrl */}
|
|
||||||
{selectedTemplate === TEMPLATE_KEYS.GENERAL && (
|
{selectedTemplate === TEMPLATE_KEYS.GENERAL && (
|
||||||
<>
|
<>
|
||||||
<div className="space-y-2">
|
<div className="space-y-2">
|
||||||
@@ -426,12 +423,13 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
}
|
}
|
||||||
placeholder="sk-xxxxx"
|
placeholder="sk-xxxxx"
|
||||||
autoComplete="off"
|
autoComplete="off"
|
||||||
|
className="border-white/10"
|
||||||
/>
|
/>
|
||||||
{script.apiKey && (
|
{script.apiKey && (
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
onClick={() => setShowApiKey(!showApiKey)}
|
onClick={() => setShowApiKey(!showApiKey)}
|
||||||
className="absolute inset-y-0 right-0 flex items-center pr-3 text-gray-500 dark:text-gray-400 hover:text-gray-900 dark:hover:text-gray-100 transition-colors"
|
className="absolute inset-y-0 right-0 flex items-center pr-3 text-muted-foreground hover:text-foreground transition-colors"
|
||||||
aria-label={
|
aria-label={
|
||||||
showApiKey
|
showApiKey
|
||||||
? t("apiKeyInput.hide")
|
? t("apiKeyInput.hide")
|
||||||
@@ -459,12 +457,12 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
}
|
}
|
||||||
placeholder="https://api.example.com"
|
placeholder="https://api.example.com"
|
||||||
autoComplete="off"
|
autoComplete="off"
|
||||||
|
className="border-white/10"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* NewAPI 模板:显示 baseUrl + accessToken + userId */}
|
|
||||||
{selectedTemplate === TEMPLATE_KEYS.NEW_API && (
|
{selectedTemplate === TEMPLATE_KEYS.NEW_API && (
|
||||||
<>
|
<>
|
||||||
<div className="space-y-2">
|
<div className="space-y-2">
|
||||||
@@ -478,6 +476,7 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
}
|
}
|
||||||
placeholder="https://api.newapi.com"
|
placeholder="https://api.newapi.com"
|
||||||
autoComplete="off"
|
autoComplete="off"
|
||||||
|
className="border-white/10"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -500,6 +499,7 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
"usageScript.accessTokenPlaceholder",
|
"usageScript.accessTokenPlaceholder",
|
||||||
)}
|
)}
|
||||||
autoComplete="off"
|
autoComplete="off"
|
||||||
|
className="border-white/10"
|
||||||
/>
|
/>
|
||||||
{script.accessToken && (
|
{script.accessToken && (
|
||||||
<button
|
<button
|
||||||
@@ -507,7 +507,7 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
onClick={() =>
|
onClick={() =>
|
||||||
setShowAccessToken(!showAccessToken)
|
setShowAccessToken(!showAccessToken)
|
||||||
}
|
}
|
||||||
className="absolute inset-y-0 right-0 flex items-center pr-3 text-gray-500 dark:text-gray-400 hover:text-gray-900 dark:hover:text-gray-100 transition-colors"
|
className="absolute inset-y-0 right-0 flex items-center pr-3 text-muted-foreground hover:text-foreground transition-colors"
|
||||||
aria-label={
|
aria-label={
|
||||||
showAccessToken
|
showAccessToken
|
||||||
? t("apiKeyInput.hide")
|
? t("apiKeyInput.hide")
|
||||||
@@ -537,32 +537,70 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
}
|
}
|
||||||
placeholder={t("usageScript.userIdPlaceholder")}
|
placeholder={t("usageScript.userIdPlaceholder")}
|
||||||
autoComplete="off"
|
autoComplete="off"
|
||||||
|
className="border-white/10"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
)}
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* 脚本编辑器 */}
|
{/* 脚本配置 */}
|
||||||
<div>
|
<div className="space-y-4 glass rounded-xl border border-white/10 p-6">
|
||||||
<Label className="mb-2">{t("usageScript.queryScript")}</Label>
|
<div className="flex items-center justify-between">
|
||||||
<JsonEditor
|
<h4 className="text-base font-medium text-foreground">
|
||||||
value={script.code}
|
{t("usageScript.scriptConfig")}
|
||||||
onChange={(code) => setScript({ ...script, code })}
|
</h4>
|
||||||
height="300px"
|
<p className="text-xs text-muted-foreground">
|
||||||
language="javascript"
|
{t("usageScript.variablesHint")}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="grid gap-4">
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="usage-request-url">
|
||||||
|
{t("usageScript.requestUrl")}
|
||||||
|
</Label>
|
||||||
|
<Input
|
||||||
|
id="usage-request-url"
|
||||||
|
type="text"
|
||||||
|
value={script.request?.url || ""}
|
||||||
|
onChange={(e) => {
|
||||||
|
setScript({
|
||||||
|
...script,
|
||||||
|
request: { ...script.request, url: e.target.value },
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
placeholder={t("usageScript.requestUrlPlaceholder")}
|
||||||
|
className="border-white/10"
|
||||||
/>
|
/>
|
||||||
<p className="mt-2 text-xs text-gray-500 dark:text-gray-400">
|
|
||||||
{t("usageScript.variablesHint", {
|
|
||||||
apiKey: "{{apiKey}}",
|
|
||||||
baseUrl: "{{baseUrl}}",
|
|
||||||
})}
|
|
||||||
</p>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* 配置选项 */}
|
<div className="grid grid-cols-1 lg:grid-cols-2 gap-4">
|
||||||
<div className="grid grid-cols-2 gap-4">
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="usage-method">
|
||||||
|
{t("usageScript.method")}
|
||||||
|
</Label>
|
||||||
|
<Input
|
||||||
|
id="usage-method"
|
||||||
|
type="text"
|
||||||
|
value={script.request?.method || "GET"}
|
||||||
|
onChange={(e) => {
|
||||||
|
setScript({
|
||||||
|
...script,
|
||||||
|
request: {
|
||||||
|
...script.request,
|
||||||
|
method: e.target.value.toUpperCase(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
placeholder="GET / POST"
|
||||||
|
className="border-white/10"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div className="space-y-2">
|
<div className="space-y-2">
|
||||||
<Label htmlFor="usage-timeout">
|
<Label htmlFor="usage-timeout">
|
||||||
{t("usageScript.timeoutSeconds")}
|
{t("usageScript.timeoutSeconds")}
|
||||||
@@ -570,83 +608,150 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
<Input
|
<Input
|
||||||
id="usage-timeout"
|
id="usage-timeout"
|
||||||
type="number"
|
type="number"
|
||||||
value={script.timeout ?? ""}
|
|
||||||
onChange={(e) => {
|
|
||||||
// 输入时:只清理格式,允许临时为空,避免强制回填默认值
|
|
||||||
const cleaned = sanitizeNumberInput(e.target.value);
|
|
||||||
setScript((prev) => ({
|
|
||||||
...prev,
|
|
||||||
timeout:
|
|
||||||
cleaned === "" ? undefined : parseInt(cleaned, 10),
|
|
||||||
}));
|
|
||||||
}}
|
|
||||||
onBlur={(e) => {
|
|
||||||
// 失焦时:严格验证并约束范围
|
|
||||||
const validated = validateTimeout(e.target.value);
|
|
||||||
setScript({ ...script, timeout: validated });
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
<p className="text-xs text-muted-foreground">
|
|
||||||
{t("usageScript.timeoutHint") || "范围: 2-30 秒"}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* 🆕 自动查询间隔 */}
|
|
||||||
<div className="space-y-2">
|
|
||||||
<Label htmlFor="usage-auto-interval">
|
|
||||||
{t("usageScript.autoQueryInterval")}
|
|
||||||
</Label>
|
|
||||||
<Input
|
|
||||||
id="usage-auto-interval"
|
|
||||||
type="number"
|
|
||||||
min={0}
|
min={0}
|
||||||
max={1440}
|
value={script.timeout ?? 10}
|
||||||
step={1}
|
onChange={(e) =>
|
||||||
value={script.autoQueryInterval ?? ""}
|
setScript({
|
||||||
onChange={(e) => {
|
...script,
|
||||||
// 输入时:只清理格式,允许临时为空
|
timeout: validateTimeout(e.target.value),
|
||||||
const cleaned = sanitizeNumberInput(e.target.value);
|
})
|
||||||
setScript((prev) => ({
|
}
|
||||||
...prev,
|
onBlur={(e) =>
|
||||||
autoQueryInterval:
|
setScript({
|
||||||
cleaned === "" ? undefined : parseInt(cleaned, 10),
|
...script,
|
||||||
}));
|
timeout: validateTimeout(e.target.value),
|
||||||
}}
|
})
|
||||||
onBlur={(e) => {
|
}
|
||||||
// 失焦时:严格验证并约束范围
|
className="border-white/10"
|
||||||
const validated = validateAndClampInterval(
|
|
||||||
e.target.value,
|
|
||||||
);
|
|
||||||
setScript({ ...script, autoQueryInterval: validated });
|
|
||||||
}}
|
|
||||||
/>
|
/>
|
||||||
<p className="text-xs text-muted-foreground">
|
|
||||||
{t("usageScript.autoQueryIntervalHint")}
|
|
||||||
</p>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* 脚本说明 */}
|
<div className="space-y-2">
|
||||||
<div className="p-4 bg-blue-50 dark:bg-blue-900/20 rounded-lg text-sm text-gray-700 dark:text-gray-300">
|
<Label htmlFor="usage-headers">
|
||||||
<h4 className="font-medium mb-2">
|
{t("usageScript.headers")}
|
||||||
{t("usageScript.scriptHelp")}
|
</Label>
|
||||||
</h4>
|
<JsonEditor
|
||||||
<div className="space-y-3 text-xs">
|
id="usage-headers"
|
||||||
<div>
|
value={
|
||||||
<strong>{t("usageScript.configFormat")}</strong>
|
script.request?.headers
|
||||||
<pre className="mt-1 p-2 bg-white/50 dark:bg-black/20 rounded text-[10px] overflow-x-auto">
|
? JSON.stringify(script.request.headers, null, 2)
|
||||||
{`({
|
: "{}"
|
||||||
|
}
|
||||||
|
onChange={(value) => {
|
||||||
|
try {
|
||||||
|
const parsed = JSON.parse(value || "{}");
|
||||||
|
setScript({
|
||||||
|
...script,
|
||||||
|
request: { ...script.request, headers: parsed },
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Invalid headers JSON", error);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
height={180}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="usage-body">{t("usageScript.body")}</Label>
|
||||||
|
<JsonEditor
|
||||||
|
id="usage-body"
|
||||||
|
value={
|
||||||
|
script.request?.body
|
||||||
|
? JSON.stringify(script.request.body, null, 2)
|
||||||
|
: "{}"
|
||||||
|
}
|
||||||
|
onChange={(value) => {
|
||||||
|
try {
|
||||||
|
const parsed =
|
||||||
|
value?.trim() === "" ? undefined : JSON.parse(value);
|
||||||
|
setScript({
|
||||||
|
...script,
|
||||||
|
request: { ...script.request, body: parsed },
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
toast.error(
|
||||||
|
t("usageScript.invalidJson") || "Body 必须是合法 JSON",
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
height={220}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="usage-interval">
|
||||||
|
{t("usageScript.autoIntervalMinutes")}
|
||||||
|
</Label>
|
||||||
|
<Input
|
||||||
|
id="usage-interval"
|
||||||
|
type="number"
|
||||||
|
min={0}
|
||||||
|
max={1440}
|
||||||
|
value={script.autoIntervalMinutes ?? 0}
|
||||||
|
onChange={(e) =>
|
||||||
|
setScript({
|
||||||
|
...script,
|
||||||
|
autoIntervalMinutes: validateAndClampInterval(
|
||||||
|
e.target.value,
|
||||||
|
),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
onBlur={(e) =>
|
||||||
|
setScript({
|
||||||
|
...script,
|
||||||
|
autoIntervalMinutes: validateAndClampInterval(
|
||||||
|
e.target.value,
|
||||||
|
),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
className="border-white/10"
|
||||||
|
/>
|
||||||
|
<p className="text-xs text-muted-foreground">
|
||||||
|
{t("usageScript.autoQueryIntervalHint")}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* 提取器代码 */}
|
||||||
|
<div className="space-y-4 glass rounded-xl border border-white/10 p-6">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<Label className="text-base font-medium">
|
||||||
|
{t("usageScript.extractorCode")}
|
||||||
|
</Label>
|
||||||
|
<div className="text-xs text-muted-foreground">
|
||||||
|
{t("usageScript.extractorHint")}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<JsonEditor
|
||||||
|
id="usage-code"
|
||||||
|
value={script.code || ""}
|
||||||
|
onChange={(value) => setScript({ ...script, code: value })}
|
||||||
|
height={480}
|
||||||
|
language="javascript"
|
||||||
|
showMinimap={false}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* 帮助信息 */}
|
||||||
|
<div className="glass rounded-xl border border-white/10 p-6 text-sm text-foreground/90">
|
||||||
|
<h4 className="font-medium mb-2">{t("usageScript.scriptHelp")}</h4>
|
||||||
|
<div className="space-y-3 text-xs">
|
||||||
|
<div>
|
||||||
|
<strong>{t("usageScript.configFormat")}</strong>
|
||||||
|
<pre className="mt-1 p-2 bg-black/20 text-foreground rounded border border-white/10 text-[10px] overflow-x-auto">
|
||||||
|
{`({
|
||||||
request: {
|
request: {
|
||||||
url: "{{baseUrl}}/api/usage",
|
url: "{{baseUrl}}/api/usage",
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
"Authorization": "Bearer {{apiKey}}",
|
"Authorization": "Bearer {{apiKey}}",
|
||||||
"User-Agent": "cc-switch/1.0"
|
"User-Agent": "cc-switch/1.0"
|
||||||
},
|
}
|
||||||
body: JSON.stringify({ key: "value" }) // ${t("usageScript.commentOptional")}
|
|
||||||
},
|
},
|
||||||
extractor: function(response) {
|
extractor: function(response) {
|
||||||
// ${t("usageScript.commentResponseIsJson")}
|
|
||||||
return {
|
return {
|
||||||
isValid: !response.error,
|
isValid: !response.error,
|
||||||
remaining: response.balance,
|
remaining: response.balance,
|
||||||
@@ -654,79 +759,41 @@ const UsageScriptModal: React.FC<UsageScriptModalProps> = ({
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
})`}
|
})`}
|
||||||
</pre>
|
</pre>
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<strong>{t("usageScript.extractorFormat")}</strong>
|
|
||||||
<ul className="mt-1 space-y-0.5 ml-2">
|
|
||||||
<li>{t("usageScript.fieldIsValid")}</li>
|
|
||||||
<li>{t("usageScript.fieldInvalidMessage")}</li>
|
|
||||||
<li>{t("usageScript.fieldRemaining")}</li>
|
|
||||||
<li>{t("usageScript.fieldUnit")}</li>
|
|
||||||
<li>{t("usageScript.fieldPlanName")}</li>
|
|
||||||
<li>{t("usageScript.fieldTotal")}</li>
|
|
||||||
<li>{t("usageScript.fieldUsed")}</li>
|
|
||||||
<li>{t("usageScript.fieldExtra")}</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="text-gray-600 dark:text-gray-400">
|
|
||||||
<strong>{t("usageScript.tips")}</strong>
|
|
||||||
<ul className="mt-1 space-y-0.5 ml-2">
|
|
||||||
<li>
|
|
||||||
{t("usageScript.tip1", {
|
|
||||||
apiKey: "{{apiKey}}",
|
|
||||||
baseUrl: "{{baseUrl}}",
|
|
||||||
})}
|
|
||||||
</li>
|
|
||||||
<li>{t("usageScript.tip2")}</li>
|
|
||||||
<li>{t("usageScript.tip3")}</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</>
|
|
||||||
)}
|
<div>
|
||||||
|
<strong>{t("usageScript.extractorFormat")}</strong>
|
||||||
|
<ul className="mt-1 space-y-0.5 ml-2">
|
||||||
|
<li>{t("usageScript.fieldIsValid")}</li>
|
||||||
|
<li>{t("usageScript.fieldInvalidMessage")}</li>
|
||||||
|
<li>{t("usageScript.fieldRemaining")}</li>
|
||||||
|
<li>{t("usageScript.fieldUnit")}</li>
|
||||||
|
<li>{t("usageScript.fieldPlanName")}</li>
|
||||||
|
<li>{t("usageScript.fieldTotal")}</li>
|
||||||
|
<li>{t("usageScript.fieldUsed")}</li>
|
||||||
|
<li>{t("usageScript.fieldExtra")}</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="text-muted-foreground">
|
||||||
|
<strong>{t("usageScript.tips")}</strong>
|
||||||
|
<ul className="mt-1 space-y-0.5 ml-2">
|
||||||
|
<li>
|
||||||
|
{t("usageScript.tip1", {
|
||||||
|
apiKey: "{{apiKey}}",
|
||||||
|
baseUrl: "{{baseUrl}}",
|
||||||
|
})}
|
||||||
|
</li>
|
||||||
|
<li>{t("usageScript.tip2")}</li>
|
||||||
|
<li>{t("usageScript.tip3")}</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
)}
|
||||||
{/* Footer */}
|
</FullScreenPanel>
|
||||||
<DialogFooter className="flex-col sm:flex-row sm:justify-between gap-3 pt-4">
|
|
||||||
{/* Left side - Test and Format buttons */}
|
|
||||||
<div className="flex gap-2">
|
|
||||||
<Button
|
|
||||||
variant="outline"
|
|
||||||
size="sm"
|
|
||||||
onClick={handleTest}
|
|
||||||
disabled={!script.enabled || testing}
|
|
||||||
>
|
|
||||||
<Play size={14} />
|
|
||||||
{testing ? t("usageScript.testing") : t("usageScript.testScript")}
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
variant="outline"
|
|
||||||
size="sm"
|
|
||||||
onClick={handleFormat}
|
|
||||||
disabled={!script.enabled}
|
|
||||||
title={t("usageScript.format")}
|
|
||||||
>
|
|
||||||
<Wand2 size={14} />
|
|
||||||
{t("usageScript.format")}
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* Right side - Cancel and Save buttons */}
|
|
||||||
<div className="flex gap-2">
|
|
||||||
<Button variant="ghost" size="sm" onClick={onClose}>
|
|
||||||
{t("common.cancel")}
|
|
||||||
</Button>
|
|
||||||
<Button variant="default" size="sm" onClick={handleSave}>
|
|
||||||
{t("usageScript.saveConfig")}
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</DialogFooter>
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
22
src/components/agents/AgentsPanel.tsx
Normal file
22
src/components/agents/AgentsPanel.tsx
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
import { Bot } from "lucide-react";
|
||||||
|
|
||||||
|
interface AgentsPanelProps {
|
||||||
|
onOpenChange: (open: boolean) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function AgentsPanel({}: AgentsPanelProps) {
|
||||||
|
return (
|
||||||
|
<div className="mx-auto max-w-5xl flex flex-col h-[calc(100vh-8rem)]">
|
||||||
|
<div className="flex-1 glass-card rounded-xl p-8 flex flex-col items-center justify-center text-center space-y-4">
|
||||||
|
<div className="w-20 h-20 rounded-full bg-white/5 flex items-center justify-center mb-4 animate-pulse-slow">
|
||||||
|
<Bot className="w-10 h-10 text-muted-foreground" />
|
||||||
|
</div>
|
||||||
|
<h3 className="text-xl font-semibold">Coming Soon</h3>
|
||||||
|
<p className="text-muted-foreground max-w-md">
|
||||||
|
The Agents management feature is currently under development. Stay
|
||||||
|
tuned for powerful autonomous capabilities.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
77
src/components/common/FullScreenPanel.tsx
Normal file
77
src/components/common/FullScreenPanel.tsx
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { createPortal } from "react-dom";
|
||||||
|
import { ArrowLeft } from "lucide-react";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
|
||||||
|
interface FullScreenPanelProps {
|
||||||
|
isOpen: boolean;
|
||||||
|
title: string;
|
||||||
|
onClose: () => void;
|
||||||
|
children: React.ReactNode;
|
||||||
|
footer?: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reusable full-screen panel component
|
||||||
|
* Handles portal rendering, header with back button, and footer
|
||||||
|
* Uses solid theme colors without transparency
|
||||||
|
*/
|
||||||
|
export const FullScreenPanel: React.FC<FullScreenPanelProps> = ({
|
||||||
|
isOpen,
|
||||||
|
title,
|
||||||
|
onClose,
|
||||||
|
children,
|
||||||
|
footer,
|
||||||
|
}) => {
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (isOpen) {
|
||||||
|
document.body.style.overflow = "hidden";
|
||||||
|
}
|
||||||
|
return () => {
|
||||||
|
document.body.style.overflow = "";
|
||||||
|
};
|
||||||
|
}, [isOpen]);
|
||||||
|
|
||||||
|
if (!isOpen) return null;
|
||||||
|
|
||||||
|
return createPortal(
|
||||||
|
<div
|
||||||
|
className="fixed inset-0 z-[60] flex flex-col"
|
||||||
|
style={{ backgroundColor: "hsl(var(--background))" }}
|
||||||
|
>
|
||||||
|
{/* Header */}
|
||||||
|
<div
|
||||||
|
className="flex-shrink-0 py-3 border-b border-border-default"
|
||||||
|
style={{ backgroundColor: "hsl(var(--background))" }}
|
||||||
|
>
|
||||||
|
<div className="h-4 w-full" data-tauri-drag-region />
|
||||||
|
<div className="mx-auto max-w-[56rem] px-6 flex items-center gap-4">
|
||||||
|
<Button type="button" variant="outline" size="icon" onClick={onClose}>
|
||||||
|
<ArrowLeft className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
<h2 className="text-lg font-semibold text-foreground">{title}</h2>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Content */}
|
||||||
|
<div className="flex-1 overflow-y-auto scroll-overlay">
|
||||||
|
<div className="mx-auto max-w-[56rem] px-6 py-6 space-y-6 w-full">
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Footer */}
|
||||||
|
{footer && (
|
||||||
|
<div
|
||||||
|
className="flex-shrink-0 py-4 border-t border-border-default"
|
||||||
|
style={{ backgroundColor: "hsl(var(--background))" }}
|
||||||
|
>
|
||||||
|
<div className="mx-auto max-w-[56rem] px-6 flex items-center justify-end gap-3">
|
||||||
|
{footer}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>,
|
||||||
|
document.body,
|
||||||
|
);
|
||||||
|
};
|
||||||
4
src/components/env/EnvWarningBanner.tsx
vendored
4
src/components/env/EnvWarningBanner.tsx
vendored
@@ -110,7 +110,7 @@ export function EnvWarningBanner({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="bg-yellow-50 dark:bg-yellow-950/20 border-b border-yellow-200 dark:border-yellow-900/50">
|
<div className="fixed top-0 left-0 right-0 z-[100] bg-yellow-50 dark:bg-yellow-950 border-b border-yellow-200 dark:border-yellow-900 shadow-lg animate-slide-down">
|
||||||
<div className="container mx-auto px-4 py-3">
|
<div className="container mx-auto px-4 py-3">
|
||||||
<div className="flex items-start gap-3">
|
<div className="flex items-start gap-3">
|
||||||
<AlertTriangle className="h-5 w-5 text-yellow-600 dark:text-yellow-500 flex-shrink-0 mt-0.5" />
|
<AlertTriangle className="h-5 w-5 text-yellow-600 dark:text-yellow-500 flex-shrink-0 mt-0.5" />
|
||||||
@@ -241,7 +241,7 @@ export function EnvWarningBanner({
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Dialog open={showConfirmDialog} onOpenChange={setShowConfirmDialog}>
|
<Dialog open={showConfirmDialog} onOpenChange={setShowConfirmDialog}>
|
||||||
<DialogContent className="max-w-md">
|
<DialogContent className="max-w-md" zIndex="top">
|
||||||
<DialogHeader>
|
<DialogHeader>
|
||||||
<DialogTitle className="flex items-center gap-2">
|
<DialogTitle className="flex items-center gap-2">
|
||||||
<AlertTriangle className="h-5 w-5 text-destructive" />
|
<AlertTriangle className="h-5 w-5 text-destructive" />
|
||||||
|
|||||||
@@ -1,25 +1,11 @@
|
|||||||
import React, { useMemo, useState } from "react";
|
import React, { useMemo, useState, useEffect } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { toast } from "sonner";
|
import { toast } from "sonner";
|
||||||
import {
|
import { Save, Plus, AlertCircle, ChevronDown, ChevronUp } from "lucide-react";
|
||||||
Save,
|
|
||||||
Plus,
|
|
||||||
AlertCircle,
|
|
||||||
ChevronDown,
|
|
||||||
ChevronUp,
|
|
||||||
Wand2,
|
|
||||||
} from "lucide-react";
|
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { Checkbox } from "@/components/ui/checkbox";
|
import { Checkbox } from "@/components/ui/checkbox";
|
||||||
import {
|
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
DialogFooter,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { Input } from "@/components/ui/input";
|
import { Input } from "@/components/ui/input";
|
||||||
import { Textarea } from "@/components/ui/textarea";
|
import JsonEditor from "@/components/JsonEditor";
|
||||||
import type { AppId } from "@/lib/api/types";
|
import type { AppId } from "@/lib/api/types";
|
||||||
import { McpServer, McpServerSpec } from "@/types";
|
import { McpServer, McpServerSpec } from "@/types";
|
||||||
import { mcpPresets, getMcpPresetWithDescription } from "@/config/mcpPresets";
|
import { mcpPresets, getMcpPresetWithDescription } from "@/config/mcpPresets";
|
||||||
@@ -34,25 +20,21 @@ import {
|
|||||||
mcpServerToToml,
|
mcpServerToToml,
|
||||||
} from "@/utils/tomlUtils";
|
} from "@/utils/tomlUtils";
|
||||||
import { normalizeTomlText } from "@/utils/textNormalization";
|
import { normalizeTomlText } from "@/utils/textNormalization";
|
||||||
import { formatJSON, parseSmartMcpJson } from "@/utils/formatters";
|
import { parseSmartMcpJson } from "@/utils/formatters";
|
||||||
import { useMcpValidation } from "./useMcpValidation";
|
import { useMcpValidation } from "./useMcpValidation";
|
||||||
import { useUpsertMcpServer } from "@/hooks/useMcp";
|
import { useUpsertMcpServer } from "@/hooks/useMcp";
|
||||||
|
import { FullScreenPanel } from "@/components/common/FullScreenPanel";
|
||||||
|
|
||||||
interface McpFormModalProps {
|
interface McpFormModalProps {
|
||||||
editingId?: string;
|
editingId?: string;
|
||||||
initialData?: McpServer;
|
initialData?: McpServer;
|
||||||
onSave: () => Promise<void>; // v3.7.0: 简化为仅用于关闭表单的回调
|
onSave: () => Promise<void>;
|
||||||
onClose: () => void;
|
onClose: () => void;
|
||||||
existingIds?: string[];
|
existingIds?: string[];
|
||||||
defaultFormat?: "json" | "toml"; // 默认配置格式(可选,默认为 JSON)
|
defaultFormat?: "json" | "toml";
|
||||||
defaultEnabledApps?: AppId[]; // 默认启用到哪些应用(可选,默认为全部应用)
|
defaultEnabledApps?: AppId[];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* MCP 表单模态框组件(v3.7.0 完整重构版)
|
|
||||||
* - 支持 JSON 和 TOML 两种格式
|
|
||||||
* - 统一管理,通过复选框选择启用到哪些应用
|
|
||||||
*/
|
|
||||||
const McpFormModal: React.FC<McpFormModalProps> = ({
|
const McpFormModal: React.FC<McpFormModalProps> = ({
|
||||||
editingId,
|
editingId,
|
||||||
initialData,
|
initialData,
|
||||||
@@ -79,7 +61,6 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
const [formDocs, setFormDocs] = useState(initialData?.docs || "");
|
const [formDocs, setFormDocs] = useState(initialData?.docs || "");
|
||||||
const [formTags, setFormTags] = useState(initialData?.tags?.join(", ") || "");
|
const [formTags, setFormTags] = useState(initialData?.tags?.join(", ") || "");
|
||||||
|
|
||||||
// 启用状态:编辑模式使用现有值,新增模式使用默认值
|
|
||||||
const [enabledApps, setEnabledApps] = useState<{
|
const [enabledApps, setEnabledApps] = useState<{
|
||||||
claude: boolean;
|
claude: boolean;
|
||||||
codex: boolean;
|
codex: boolean;
|
||||||
@@ -88,7 +69,6 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
if (initialData?.apps) {
|
if (initialData?.apps) {
|
||||||
return { ...initialData.apps };
|
return { ...initialData.apps };
|
||||||
}
|
}
|
||||||
// 新增模式:根据 defaultEnabledApps 设置初始值
|
|
||||||
return {
|
return {
|
||||||
claude: defaultEnabledApps.includes("claude"),
|
claude: defaultEnabledApps.includes("claude"),
|
||||||
codex: defaultEnabledApps.includes("codex"),
|
codex: defaultEnabledApps.includes("codex"),
|
||||||
@@ -96,10 +76,8 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
// 编辑模式下禁止修改 ID
|
|
||||||
const isEditing = !!editingId;
|
const isEditing = !!editingId;
|
||||||
|
|
||||||
// 判断是否在编辑模式下有附加信息
|
|
||||||
const hasAdditionalInfo = !!(
|
const hasAdditionalInfo = !!(
|
||||||
initialData?.description ||
|
initialData?.description ||
|
||||||
initialData?.tags?.length ||
|
initialData?.tags?.length ||
|
||||||
@@ -107,21 +85,17 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
initialData?.docs
|
initialData?.docs
|
||||||
);
|
);
|
||||||
|
|
||||||
// 附加信息展开状态(编辑模式下有值时默认展开)
|
|
||||||
const [showMetadata, setShowMetadata] = useState(
|
const [showMetadata, setShowMetadata] = useState(
|
||||||
isEditing ? hasAdditionalInfo : false,
|
isEditing ? hasAdditionalInfo : false,
|
||||||
);
|
);
|
||||||
|
|
||||||
// 配置格式:优先使用 defaultFormat,编辑模式下可从现有数据推断
|
|
||||||
const useTomlFormat = useMemo(() => {
|
const useTomlFormat = useMemo(() => {
|
||||||
if (initialData?.server) {
|
if (initialData?.server) {
|
||||||
// 编辑模式:尝试从现有数据推断格式(这里简化处理,默认 JSON)
|
|
||||||
return defaultFormat === "toml";
|
return defaultFormat === "toml";
|
||||||
}
|
}
|
||||||
return defaultFormat === "toml";
|
return defaultFormat === "toml";
|
||||||
}, [defaultFormat, initialData]);
|
}, [defaultFormat, initialData]);
|
||||||
|
|
||||||
// 根据格式决定初始配置
|
|
||||||
const [formConfig, setFormConfig] = useState(() => {
|
const [formConfig, setFormConfig] = useState(() => {
|
||||||
const spec = initialData?.server;
|
const spec = initialData?.server;
|
||||||
if (!spec) return "";
|
if (!spec) return "";
|
||||||
@@ -135,8 +109,23 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
const [saving, setSaving] = useState(false);
|
const [saving, setSaving] = useState(false);
|
||||||
const [isWizardOpen, setIsWizardOpen] = useState(false);
|
const [isWizardOpen, setIsWizardOpen] = useState(false);
|
||||||
const [idError, setIdError] = useState("");
|
const [idError, setIdError] = useState("");
|
||||||
|
const [isDarkMode, setIsDarkMode] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
|
||||||
|
const observer = new MutationObserver(() => {
|
||||||
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
});
|
||||||
|
|
||||||
|
observer.observe(document.documentElement, {
|
||||||
|
attributes: true,
|
||||||
|
attributeFilter: ["class"],
|
||||||
|
});
|
||||||
|
|
||||||
|
return () => observer.disconnect();
|
||||||
|
}, []);
|
||||||
|
|
||||||
// 判断是否使用 TOML 格式(向后兼容,后续可扩展为格式切换按钮)
|
|
||||||
const useToml = useTomlFormat;
|
const useToml = useTomlFormat;
|
||||||
|
|
||||||
const wizardInitialSpec = useMemo(() => {
|
const wizardInitialSpec = useMemo(() => {
|
||||||
@@ -164,7 +153,6 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
}
|
}
|
||||||
}, [formConfig, initialData, useToml]);
|
}, [formConfig, initialData, useToml]);
|
||||||
|
|
||||||
// 预设选择状态(仅新增模式显示;-1 表示自定义)
|
|
||||||
const [selectedPreset, setSelectedPreset] = useState<number | null>(
|
const [selectedPreset, setSelectedPreset] = useState<number | null>(
|
||||||
isEditing ? null : -1,
|
isEditing ? null : -1,
|
||||||
);
|
);
|
||||||
@@ -186,7 +174,6 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
return `${candidate}-${i}`;
|
return `${candidate}-${i}`;
|
||||||
};
|
};
|
||||||
|
|
||||||
// 应用预设(写入表单但不落库)
|
|
||||||
const applyPreset = (index: number) => {
|
const applyPreset = (index: number) => {
|
||||||
if (index < 0 || index >= mcpPresets.length) return;
|
if (index < 0 || index >= mcpPresets.length) return;
|
||||||
const preset = mcpPresets[index];
|
const preset = mcpPresets[index];
|
||||||
@@ -200,7 +187,6 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
setFormDocs(presetWithDesc.docs || "");
|
setFormDocs(presetWithDesc.docs || "");
|
||||||
setFormTags(presetWithDesc.tags?.join(", ") || "");
|
setFormTags(presetWithDesc.tags?.join(", ") || "");
|
||||||
|
|
||||||
// 根据格式转换配置
|
|
||||||
if (useToml) {
|
if (useToml) {
|
||||||
const toml = mcpServerToToml(presetWithDesc.server);
|
const toml = mcpServerToToml(presetWithDesc.server);
|
||||||
setFormConfig(toml);
|
setFormConfig(toml);
|
||||||
@@ -213,10 +199,8 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
setSelectedPreset(index);
|
setSelectedPreset(index);
|
||||||
};
|
};
|
||||||
|
|
||||||
// 切回自定义
|
|
||||||
const applyCustom = () => {
|
const applyCustom = () => {
|
||||||
setSelectedPreset(-1);
|
setSelectedPreset(-1);
|
||||||
// 恢复到空白模板
|
|
||||||
setFormId("");
|
setFormId("");
|
||||||
setFormName("");
|
setFormName("");
|
||||||
setFormDescription("");
|
setFormDescription("");
|
||||||
@@ -228,19 +212,16 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleConfigChange = (value: string) => {
|
const handleConfigChange = (value: string) => {
|
||||||
// 若为 TOML 模式,先做引号归一化,避免中文输入法导致的格式错误
|
|
||||||
const nextValue = useToml ? normalizeTomlText(value) : value;
|
const nextValue = useToml ? normalizeTomlText(value) : value;
|
||||||
setFormConfig(nextValue);
|
setFormConfig(nextValue);
|
||||||
|
|
||||||
if (useToml) {
|
if (useToml) {
|
||||||
// TOML validation (use hook's complete validation)
|
|
||||||
const err = validateTomlConfig(nextValue);
|
const err = validateTomlConfig(nextValue);
|
||||||
if (err) {
|
if (err) {
|
||||||
setConfigError(err);
|
setConfigError(err);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try to extract ID (if user hasn't filled it yet)
|
|
||||||
if (nextValue.trim() && !formId.trim()) {
|
if (nextValue.trim() && !formId.trim()) {
|
||||||
const extractedId = extractIdFromToml(nextValue);
|
const extractedId = extractIdFromToml(nextValue);
|
||||||
if (extractedId) {
|
if (extractedId) {
|
||||||
@@ -248,11 +229,8 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// JSON validation with smart parsing
|
|
||||||
try {
|
try {
|
||||||
const result = parseSmartMcpJson(value);
|
const result = parseSmartMcpJson(value);
|
||||||
|
|
||||||
// 验证解析后的配置对象
|
|
||||||
const configJson = JSON.stringify(result.config);
|
const configJson = JSON.stringify(result.config);
|
||||||
const validationErr = validateJsonConfig(configJson);
|
const validationErr = validateJsonConfig(configJson);
|
||||||
|
|
||||||
@@ -261,20 +239,15 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 自动填充提取的 id(仅当表单 id 为空且不在编辑模式时)
|
|
||||||
if (result.id && !formId.trim() && !isEditing) {
|
if (result.id && !formId.trim() && !isEditing) {
|
||||||
const uniqueId = ensureUniqueId(result.id);
|
const uniqueId = ensureUniqueId(result.id);
|
||||||
setFormId(uniqueId);
|
setFormId(uniqueId);
|
||||||
|
|
||||||
// 如果 name 也为空,同时填充 name
|
|
||||||
if (!formName.trim()) {
|
if (!formName.trim()) {
|
||||||
setFormName(result.id);
|
setFormName(result.id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 不在输入时自动格式化,保持用户输入的原样
|
|
||||||
// 格式清理将在提交时进行
|
|
||||||
|
|
||||||
setConfigError("");
|
setConfigError("");
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
const errorMessage = err?.message || String(err);
|
const errorMessage = err?.message || String(err);
|
||||||
@@ -283,30 +256,11 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleFormatJson = () => {
|
|
||||||
if (!formConfig.trim()) return;
|
|
||||||
|
|
||||||
try {
|
|
||||||
const formatted = formatJSON(formConfig);
|
|
||||||
setFormConfig(formatted);
|
|
||||||
toast.success(t("common.formatSuccess"));
|
|
||||||
} catch (error) {
|
|
||||||
const errorMessage =
|
|
||||||
error instanceof Error ? error.message : String(error);
|
|
||||||
toast.error(
|
|
||||||
t("common.formatError", {
|
|
||||||
error: errorMessage,
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleWizardApply = (title: string, json: string) => {
|
const handleWizardApply = (title: string, json: string) => {
|
||||||
setFormId(title);
|
setFormId(title);
|
||||||
if (!formName.trim()) {
|
if (!formName.trim()) {
|
||||||
setFormName(title);
|
setFormName(title);
|
||||||
}
|
}
|
||||||
// Wizard returns JSON, convert based on format if needed
|
|
||||||
if (useToml) {
|
if (useToml) {
|
||||||
try {
|
try {
|
||||||
const server = JSON.parse(json) as McpServerSpec;
|
const server = JSON.parse(json) as McpServerSpec;
|
||||||
@@ -329,17 +283,14 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 新增模式:阻止提交重名 ID
|
|
||||||
if (!isEditing && existingIds.includes(trimmedId)) {
|
if (!isEditing && existingIds.includes(trimmedId)) {
|
||||||
setIdError(t("mcp.error.idExists"));
|
setIdError(t("mcp.error.idExists"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validate configuration format
|
|
||||||
let serverSpec: McpServerSpec;
|
let serverSpec: McpServerSpec;
|
||||||
|
|
||||||
if (useToml) {
|
if (useToml) {
|
||||||
// TOML mode
|
|
||||||
const tomlError = validateTomlConfig(formConfig);
|
const tomlError = validateTomlConfig(formConfig);
|
||||||
setConfigError(tomlError);
|
setConfigError(tomlError);
|
||||||
if (tomlError) {
|
if (tomlError) {
|
||||||
@@ -348,7 +299,6 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!formConfig.trim()) {
|
if (!formConfig.trim()) {
|
||||||
// Empty configuration
|
|
||||||
serverSpec = {
|
serverSpec = {
|
||||||
type: "stdio",
|
type: "stdio",
|
||||||
command: "",
|
command: "",
|
||||||
@@ -365,9 +315,7 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// JSON mode
|
|
||||||
if (!formConfig.trim()) {
|
if (!formConfig.trim()) {
|
||||||
// Empty configuration
|
|
||||||
serverSpec = {
|
serverSpec = {
|
||||||
type: "stdio",
|
type: "stdio",
|
||||||
command: "",
|
command: "",
|
||||||
@@ -375,7 +323,6 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
};
|
};
|
||||||
} else {
|
} else {
|
||||||
try {
|
try {
|
||||||
// 使用智能解析器,支持带外层键的格式
|
|
||||||
const result = parseSmartMcpJson(formConfig);
|
const result = parseSmartMcpJson(formConfig);
|
||||||
serverSpec = result.config as McpServerSpec;
|
serverSpec = result.config as McpServerSpec;
|
||||||
} catch (e: any) {
|
} catch (e: any) {
|
||||||
@@ -387,7 +334,6 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 前置必填校验
|
|
||||||
if (serverSpec?.type === "stdio" && !serverSpec?.command?.trim()) {
|
if (serverSpec?.type === "stdio" && !serverSpec?.command?.trim()) {
|
||||||
toast.error(t("mcp.error.commandRequired"), { duration: 3000 });
|
toast.error(t("mcp.error.commandRequired"), { duration: 3000 });
|
||||||
return;
|
return;
|
||||||
@@ -402,7 +348,6 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
|
|
||||||
setSaving(true);
|
setSaving(true);
|
||||||
try {
|
try {
|
||||||
// 先处理 name 字段(必填)
|
|
||||||
const nameTrimmed = (formName || trimmedId).trim();
|
const nameTrimmed = (formName || trimmedId).trim();
|
||||||
const finalName = nameTrimmed || trimmedId;
|
const finalName = nameTrimmed || trimmedId;
|
||||||
|
|
||||||
@@ -411,7 +356,6 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
id: trimmedId,
|
id: trimmedId,
|
||||||
name: finalName,
|
name: finalName,
|
||||||
server: serverSpec,
|
server: serverSpec,
|
||||||
// 使用表单中的启用状态(v3.7.0 完整重构)
|
|
||||||
apps: enabledApps,
|
apps: enabledApps,
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -446,10 +390,9 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
delete entry.tags;
|
delete entry.tags;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 保存到统一配置
|
|
||||||
await upsertMutation.mutateAsync(entry);
|
await upsertMutation.mutateAsync(entry);
|
||||||
toast.success(t("common.success"));
|
toast.success(t("common.success"));
|
||||||
await onSave(); // 通知父组件关闭表单
|
await onSave();
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
const detail = extractErrorMessage(error);
|
const detail = extractErrorMessage(error);
|
||||||
const mapped = translateMcpBackendError(detail, t);
|
const mapped = translateMcpBackendError(detail, t);
|
||||||
@@ -466,18 +409,33 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Dialog open={true} onOpenChange={(open) => !open && onClose()}>
|
<FullScreenPanel
|
||||||
<DialogContent className="max-w-3xl max-h-[90vh] flex flex-col">
|
isOpen={true}
|
||||||
<DialogHeader>
|
title={getFormTitle()}
|
||||||
<DialogTitle>{getFormTitle()}</DialogTitle>
|
onClose={onClose}
|
||||||
</DialogHeader>
|
footer={
|
||||||
|
<Button
|
||||||
{/* Content - Scrollable */}
|
type="button"
|
||||||
<div className="flex-1 overflow-y-auto px-6 py-4 space-y-4">
|
onClick={handleSubmit}
|
||||||
|
disabled={saving || (!isEditing && !!idError)}
|
||||||
|
className="bg-primary text-primary-foreground hover:bg-primary/90 disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
{isEditing ? <Save size={16} /> : <Plus size={16} />}
|
||||||
|
{saving
|
||||||
|
? t("common.saving")
|
||||||
|
: isEditing
|
||||||
|
? t("common.save")
|
||||||
|
: t("common.add")}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<div className="flex flex-col h-full gap-6">
|
||||||
|
{/* 上半部分:表单字段 */}
|
||||||
|
<div className="glass rounded-xl p-6 border border-white/10 space-y-6 flex-shrink-0">
|
||||||
{/* 预设选择(仅新增时展示) */}
|
{/* 预设选择(仅新增时展示) */}
|
||||||
{!isEditing && (
|
{!isEditing && (
|
||||||
<div>
|
<div>
|
||||||
<label className="block text-sm font-medium text-gray-900 dark:text-gray-100 mb-3">
|
<label className="block text-sm font-medium text-foreground mb-3">
|
||||||
{t("mcp.presets.title")}
|
{t("mcp.presets.title")}
|
||||||
</label>
|
</label>
|
||||||
<div className="flex flex-wrap gap-2">
|
<div className="flex flex-wrap gap-2">
|
||||||
@@ -487,7 +445,7 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
className={`inline-flex items-center gap-2 px-4 py-2 rounded-lg text-sm font-medium transition-colors ${
|
className={`inline-flex items-center gap-2 px-4 py-2 rounded-lg text-sm font-medium transition-colors ${
|
||||||
selectedPreset === -1
|
selectedPreset === -1
|
||||||
? "bg-emerald-500 text-white dark:bg-emerald-600"
|
? "bg-emerald-500 text-white dark:bg-emerald-600"
|
||||||
: "bg-gray-100 dark:bg-gray-800 text-gray-500 dark:text-gray-400 hover:bg-gray-200 dark:hover:bg-gray-700"
|
: "bg-accent text-muted-foreground hover:bg-accent/80"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
{t("presetSelector.custom")}
|
{t("presetSelector.custom")}
|
||||||
@@ -502,7 +460,7 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
className={`inline-flex items-center gap-2 px-4 py-2 rounded-lg text-sm font-medium transition-colors ${
|
className={`inline-flex items-center gap-2 px-4 py-2 rounded-lg text-sm font-medium transition-colors ${
|
||||||
selectedPreset === idx
|
selectedPreset === idx
|
||||||
? "bg-emerald-500 text-white dark:bg-emerald-600"
|
? "bg-emerald-500 text-white dark:bg-emerald-600"
|
||||||
: "bg-gray-100 dark:bg-gray-800 text-gray-500 dark:text-gray-400 hover:bg-gray-200 dark:hover:bg-gray-700"
|
: "bg-accent text-muted-foreground hover:bg-accent/80"
|
||||||
}`}
|
}`}
|
||||||
title={t(descriptionKey)}
|
title={t(descriptionKey)}
|
||||||
>
|
>
|
||||||
@@ -513,10 +471,11 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* ID (标题) */}
|
{/* ID (标题) */}
|
||||||
<div>
|
<div>
|
||||||
<div className="flex items-center justify-between mb-2">
|
<div className="flex items-center justify-between mb-2">
|
||||||
<label className="block text-sm font-medium text-gray-700 dark:text-gray-300">
|
<label className="block text-sm font-medium text-foreground">
|
||||||
{t("mcp.form.title")} <span className="text-red-500">*</span>
|
{t("mcp.form.title")} <span className="text-red-500">*</span>
|
||||||
</label>
|
</label>
|
||||||
{!isEditing && idError && (
|
{!isEditing && idError && (
|
||||||
@@ -536,7 +495,7 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
|
|
||||||
{/* Name */}
|
{/* Name */}
|
||||||
<div>
|
<div>
|
||||||
<label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
|
<label className="block text-sm font-medium text-foreground mb-2">
|
||||||
{t("mcp.form.name")}
|
{t("mcp.form.name")}
|
||||||
</label>
|
</label>
|
||||||
<Input
|
<Input
|
||||||
@@ -547,9 +506,9 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* 启用到哪些应用(v3.7.0 新增) */}
|
{/* 启用到哪些应用 */}
|
||||||
<div>
|
<div>
|
||||||
<label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-3">
|
<label className="block text-sm font-medium text-foreground mb-3">
|
||||||
{t("mcp.form.enabledApps")}
|
{t("mcp.form.enabledApps")}
|
||||||
</label>
|
</label>
|
||||||
<div className="flex flex-wrap gap-4">
|
<div className="flex flex-wrap gap-4">
|
||||||
@@ -563,7 +522,7 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
/>
|
/>
|
||||||
<label
|
<label
|
||||||
htmlFor="enable-claude"
|
htmlFor="enable-claude"
|
||||||
className="text-sm text-gray-700 dark:text-gray-300 cursor-pointer select-none"
|
className="text-sm text-foreground cursor-pointer select-none"
|
||||||
>
|
>
|
||||||
{t("mcp.unifiedPanel.apps.claude")}
|
{t("mcp.unifiedPanel.apps.claude")}
|
||||||
</label>
|
</label>
|
||||||
@@ -579,7 +538,7 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
/>
|
/>
|
||||||
<label
|
<label
|
||||||
htmlFor="enable-codex"
|
htmlFor="enable-codex"
|
||||||
className="text-sm text-gray-700 dark:text-gray-300 cursor-pointer select-none"
|
className="text-sm text-foreground cursor-pointer select-none"
|
||||||
>
|
>
|
||||||
{t("mcp.unifiedPanel.apps.codex")}
|
{t("mcp.unifiedPanel.apps.codex")}
|
||||||
</label>
|
</label>
|
||||||
@@ -595,7 +554,7 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
/>
|
/>
|
||||||
<label
|
<label
|
||||||
htmlFor="enable-gemini"
|
htmlFor="enable-gemini"
|
||||||
className="text-sm text-gray-700 dark:text-gray-300 cursor-pointer select-none"
|
className="text-sm text-foreground cursor-pointer select-none"
|
||||||
>
|
>
|
||||||
{t("mcp.unifiedPanel.apps.gemini")}
|
{t("mcp.unifiedPanel.apps.gemini")}
|
||||||
</label>
|
</label>
|
||||||
@@ -608,7 +567,7 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
onClick={() => setShowMetadata(!showMetadata)}
|
onClick={() => setShowMetadata(!showMetadata)}
|
||||||
className="flex items-center gap-2 text-sm font-medium text-gray-700 dark:text-gray-300 hover:text-gray-900 dark:hover:text-gray-100 transition-colors"
|
className="flex items-center gap-2 text-sm font-medium text-muted-foreground hover:text-foreground transition-colors"
|
||||||
>
|
>
|
||||||
{showMetadata ? (
|
{showMetadata ? (
|
||||||
<ChevronUp size={16} />
|
<ChevronUp size={16} />
|
||||||
@@ -622,9 +581,8 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
{/* 附加信息区域(可折叠) */}
|
{/* 附加信息区域(可折叠) */}
|
||||||
{showMetadata && (
|
{showMetadata && (
|
||||||
<>
|
<>
|
||||||
{/* Description (描述) */}
|
|
||||||
<div>
|
<div>
|
||||||
<label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
|
<label className="block text-sm font-medium text-foreground mb-2">
|
||||||
{t("mcp.form.description")}
|
{t("mcp.form.description")}
|
||||||
</label>
|
</label>
|
||||||
<Input
|
<Input
|
||||||
@@ -635,9 +593,8 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Tags */}
|
|
||||||
<div>
|
<div>
|
||||||
<label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
|
<label className="block text-sm font-medium text-foreground mb-2">
|
||||||
{t("mcp.form.tags")}
|
{t("mcp.form.tags")}
|
||||||
</label>
|
</label>
|
||||||
<Input
|
<Input
|
||||||
@@ -648,9 +605,8 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Homepage */}
|
|
||||||
<div>
|
<div>
|
||||||
<label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
|
<label className="block text-sm font-medium text-foreground mb-2">
|
||||||
{t("mcp.form.homepage")}
|
{t("mcp.form.homepage")}
|
||||||
</label>
|
</label>
|
||||||
<Input
|
<Input
|
||||||
@@ -661,9 +617,8 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Docs */}
|
|
||||||
<div>
|
<div>
|
||||||
<label className="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
|
<label className="block text-sm font-medium text-foreground mb-2">
|
||||||
{t("mcp.form.docs")}
|
{t("mcp.form.docs")}
|
||||||
</label>
|
</label>
|
||||||
<Input
|
<Input
|
||||||
@@ -675,79 +630,51 @@ const McpFormModal: React.FC<McpFormModalProps> = ({
|
|||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* 配置输入框(根据格式显示 JSON 或 TOML) */}
|
{/* 下半部分:JSON 配置编辑器 - 自适应剩余高度 */}
|
||||||
<div>
|
<div className="glass rounded-xl p-6 border border-white/10 flex flex-col flex-1 min-h-0">
|
||||||
<div className="flex items-center justify-between mb-2">
|
<div className="flex items-center justify-between mb-4 flex-shrink-0">
|
||||||
<label className="text-sm font-medium text-gray-700 dark:text-gray-300">
|
<label className="text-sm font-medium text-foreground">
|
||||||
{useToml
|
{useToml ? t("mcp.form.tomlConfig") : t("mcp.form.jsonConfig")}
|
||||||
? t("mcp.form.tomlConfig")
|
</label>
|
||||||
: t("mcp.form.jsonConfig")}
|
{(isEditing || selectedPreset === -1) && (
|
||||||
</label>
|
<button
|
||||||
{(isEditing || selectedPreset === -1) && (
|
type="button"
|
||||||
<button
|
onClick={() => setIsWizardOpen(true)}
|
||||||
type="button"
|
className="text-sm text-blue-500 dark:text-blue-400 hover:text-blue-600 dark:hover:text-blue-300 transition-colors"
|
||||||
onClick={() => setIsWizardOpen(true)}
|
>
|
||||||
className="text-sm text-blue-500 dark:text-blue-400 hover:text-blue-600 dark:hover:text-blue-300 transition-colors"
|
{t("mcp.form.useWizard")}
|
||||||
>
|
</button>
|
||||||
{t("mcp.form.useWizard")}
|
|
||||||
</button>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
<Textarea
|
|
||||||
className="h-48 resize-none font-mono text-xs"
|
|
||||||
placeholder={
|
|
||||||
useToml
|
|
||||||
? t("mcp.form.tomlPlaceholder")
|
|
||||||
: t("mcp.form.jsonPlaceholder")
|
|
||||||
}
|
|
||||||
value={formConfig}
|
|
||||||
onChange={(e) => handleConfigChange(e.target.value)}
|
|
||||||
/>
|
|
||||||
{/* 格式化按钮(仅 JSON 模式) */}
|
|
||||||
{!useToml && (
|
|
||||||
<div className="flex items-center justify-between mt-2">
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
onClick={handleFormatJson}
|
|
||||||
className="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium text-gray-700 dark:text-gray-300 hover:text-blue-600 dark:hover:text-blue-400 transition-colors"
|
|
||||||
>
|
|
||||||
<Wand2 className="w-3.5 h-3.5" />
|
|
||||||
{t("common.format")}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
)}
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="flex-1 min-h-0 flex flex-col">
|
||||||
|
<div className="flex-1 min-h-0">
|
||||||
|
<JsonEditor
|
||||||
|
value={formConfig}
|
||||||
|
onChange={handleConfigChange}
|
||||||
|
placeholder={
|
||||||
|
useToml
|
||||||
|
? t("mcp.form.tomlPlaceholder")
|
||||||
|
: t("mcp.form.jsonPlaceholder")
|
||||||
|
}
|
||||||
|
darkMode={isDarkMode}
|
||||||
|
rows={12}
|
||||||
|
showValidation={!useToml}
|
||||||
|
language={useToml ? "javascript" : "json"}
|
||||||
|
height="100%"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
{configError && (
|
{configError && (
|
||||||
<div className="flex items-center gap-2 mt-2 text-red-500 dark:text-red-400 text-sm">
|
<div className="flex items-center gap-2 mt-2 text-red-500 dark:text-red-400 text-sm flex-shrink-0">
|
||||||
<AlertCircle size={16} />
|
<AlertCircle size={16} />
|
||||||
<span>{configError}</span>
|
<span>{configError}</span>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
{/* Footer */}
|
</FullScreenPanel>
|
||||||
<DialogFooter className="flex justify-end gap-3 pt-4">
|
|
||||||
{/* 操作按钮 */}
|
|
||||||
<Button type="button" variant="ghost" onClick={onClose}>
|
|
||||||
{t("common.cancel")}
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
type="button"
|
|
||||||
onClick={handleSubmit}
|
|
||||||
disabled={saving || (!isEditing && !!idError)}
|
|
||||||
variant="mcp"
|
|
||||||
>
|
|
||||||
{isEditing ? <Save size={16} /> : <Plus size={16} />}
|
|
||||||
{saving
|
|
||||||
? t("common.saving")
|
|
||||||
: isEditing
|
|
||||||
? t("common.save")
|
|
||||||
: t("common.add")}
|
|
||||||
</Button>
|
|
||||||
</DialogFooter>
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
|
|
||||||
{/* Wizard Modal */}
|
{/* Wizard Modal */}
|
||||||
<McpWizardModal
|
<McpWizardModal
|
||||||
|
|||||||
@@ -1,14 +1,7 @@
|
|||||||
import React, { useMemo, useState } from "react";
|
import React, { useMemo, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { Plus, Server, Check } from "lucide-react";
|
import { Server } from "lucide-react";
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import {
|
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogFooter,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { Switch } from "@/components/ui/switch";
|
import { Switch } from "@/components/ui/switch";
|
||||||
import { useAllMcpServers, useToggleMcpApp } from "@/hooks/useMcp";
|
import { useAllMcpServers, useToggleMcpApp } from "@/hooks/useMcp";
|
||||||
import type { McpServer } from "@/types";
|
import type { McpServer } from "@/types";
|
||||||
@@ -22,7 +15,6 @@ import { mcpPresets } from "@/config/mcpPresets";
|
|||||||
import { toast } from "sonner";
|
import { toast } from "sonner";
|
||||||
|
|
||||||
interface UnifiedMcpPanelProps {
|
interface UnifiedMcpPanelProps {
|
||||||
open: boolean;
|
|
||||||
onOpenChange: (open: boolean) => void;
|
onOpenChange: (open: boolean) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -30,10 +22,14 @@ interface UnifiedMcpPanelProps {
|
|||||||
* 统一 MCP 管理面板
|
* 统一 MCP 管理面板
|
||||||
* v3.7.0 新架构:所有 MCP 服务器统一管理,每个服务器通过复选框控制应用到哪些客户端
|
* v3.7.0 新架构:所有 MCP 服务器统一管理,每个服务器通过复选框控制应用到哪些客户端
|
||||||
*/
|
*/
|
||||||
const UnifiedMcpPanel: React.FC<UnifiedMcpPanelProps> = ({
|
export interface UnifiedMcpPanelHandle {
|
||||||
open,
|
openAdd: () => void;
|
||||||
onOpenChange,
|
}
|
||||||
}) => {
|
|
||||||
|
const UnifiedMcpPanel = React.forwardRef<
|
||||||
|
UnifiedMcpPanelHandle,
|
||||||
|
UnifiedMcpPanelProps
|
||||||
|
>(({ onOpenChange: _onOpenChange }, ref) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [isFormOpen, setIsFormOpen] = useState(false);
|
const [isFormOpen, setIsFormOpen] = useState(false);
|
||||||
const [editingId, setEditingId] = useState<string | null>(null);
|
const [editingId, setEditingId] = useState<string | null>(null);
|
||||||
@@ -90,6 +86,10 @@ const UnifiedMcpPanel: React.FC<UnifiedMcpPanelProps> = ({
|
|||||||
setIsFormOpen(true);
|
setIsFormOpen(true);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
React.useImperativeHandle(ref, () => ({
|
||||||
|
openAdd: handleAdd,
|
||||||
|
}));
|
||||||
|
|
||||||
const handleDelete = (id: string) => {
|
const handleDelete = (id: string) => {
|
||||||
setConfirmDialog({
|
setConfirmDialog({
|
||||||
isOpen: true,
|
isOpen: true,
|
||||||
@@ -115,78 +115,50 @@ const UnifiedMcpPanel: React.FC<UnifiedMcpPanelProps> = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<div className="mx-auto max-w-[56rem] px-6 flex flex-col h-[calc(100vh-8rem)] overflow-hidden">
|
||||||
<Dialog open={open} onOpenChange={onOpenChange}>
|
{/* Info Section */}
|
||||||
<DialogContent className="max-w-3xl max-h-[85vh] min-h-[600px] flex flex-col">
|
<div className="flex-shrink-0 py-4 glass rounded-xl border border-white/10 mb-4 px-6">
|
||||||
<DialogHeader>
|
<div className="text-sm text-muted-foreground">
|
||||||
<div className="flex items-center justify-between pr-8">
|
{t("mcp.serverCount", { count: serverEntries.length })} ·{" "}
|
||||||
<DialogTitle>{t("mcp.unifiedPanel.title")}</DialogTitle>
|
{t("mcp.unifiedPanel.apps.claude")}: {enabledCounts.claude} ·{" "}
|
||||||
<Button type="button" variant="mcp" onClick={handleAdd}>
|
{t("mcp.unifiedPanel.apps.codex")}: {enabledCounts.codex} ·{" "}
|
||||||
<Plus size={16} />
|
{t("mcp.unifiedPanel.apps.gemini")}: {enabledCounts.gemini}
|
||||||
{t("mcp.unifiedPanel.addServer")}
|
</div>
|
||||||
</Button>
|
</div>
|
||||||
</div>
|
|
||||||
</DialogHeader>
|
|
||||||
|
|
||||||
{/* Info Section */}
|
{/* Content - Scrollable */}
|
||||||
<div className="flex-shrink-0 px-6 py-4">
|
<div className="flex-1 overflow-y-auto overflow-x-hidden pb-24">
|
||||||
<div className="text-sm text-gray-500 dark:text-gray-400">
|
{isLoading ? (
|
||||||
{t("mcp.serverCount", { count: serverEntries.length })} ·{" "}
|
<div className="text-center py-12 text-gray-500 dark:text-gray-400">
|
||||||
{t("mcp.unifiedPanel.apps.claude")}: {enabledCounts.claude} ·{" "}
|
{t("mcp.loading")}
|
||||||
{t("mcp.unifiedPanel.apps.codex")}: {enabledCounts.codex} ·{" "}
|
|
||||||
{t("mcp.unifiedPanel.apps.gemini")}: {enabledCounts.gemini}
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
) : serverEntries.length === 0 ? (
|
||||||
{/* Content - Scrollable */}
|
<div className="text-center py-12">
|
||||||
<div className="flex-1 overflow-y-auto px-6 pb-4">
|
<div className="w-16 h-16 mx-auto mb-4 bg-gray-100 dark:bg-gray-800 rounded-full flex items-center justify-center">
|
||||||
{isLoading ? (
|
<Server size={24} className="text-gray-400 dark:text-gray-500" />
|
||||||
<div className="text-center py-12 text-gray-500 dark:text-gray-400">
|
</div>
|
||||||
{t("mcp.loading")}
|
<h3 className="text-lg font-medium text-gray-900 dark:text-gray-100 mb-2">
|
||||||
</div>
|
{t("mcp.unifiedPanel.noServers")}
|
||||||
) : serverEntries.length === 0 ? (
|
</h3>
|
||||||
<div className="text-center py-12">
|
<p className="text-gray-500 dark:text-gray-400 text-sm">
|
||||||
<div className="w-16 h-16 mx-auto mb-4 bg-gray-100 dark:bg-gray-800 rounded-full flex items-center justify-center">
|
{t("mcp.emptyDescription")}
|
||||||
<Server
|
</p>
|
||||||
size={24}
|
|
||||||
className="text-gray-400 dark:text-gray-500"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<h3 className="text-lg font-medium text-gray-900 dark:text-gray-100 mb-2">
|
|
||||||
{t("mcp.unifiedPanel.noServers")}
|
|
||||||
</h3>
|
|
||||||
<p className="text-gray-500 dark:text-gray-400 text-sm">
|
|
||||||
{t("mcp.emptyDescription")}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<div className="space-y-3">
|
|
||||||
{serverEntries.map(([id, server]) => (
|
|
||||||
<UnifiedMcpListItem
|
|
||||||
key={id}
|
|
||||||
id={id}
|
|
||||||
server={server}
|
|
||||||
onToggleApp={handleToggleApp}
|
|
||||||
onEdit={handleEdit}
|
|
||||||
onDelete={handleDelete}
|
|
||||||
/>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
|
) : (
|
||||||
<DialogFooter>
|
<div className="space-y-3">
|
||||||
<Button
|
{serverEntries.map(([id, server]) => (
|
||||||
type="button"
|
<UnifiedMcpListItem
|
||||||
variant="mcp"
|
key={id}
|
||||||
onClick={() => onOpenChange(false)}
|
id={id}
|
||||||
>
|
server={server}
|
||||||
<Check size={16} />
|
onToggleApp={handleToggleApp}
|
||||||
{t("common.done")}
|
onEdit={handleEdit}
|
||||||
</Button>
|
onDelete={handleDelete}
|
||||||
</DialogFooter>
|
/>
|
||||||
</DialogContent>
|
))}
|
||||||
</Dialog>
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* Form Modal */}
|
{/* Form Modal */}
|
||||||
{isFormOpen && (
|
{isFormOpen && (
|
||||||
@@ -215,9 +187,11 @@ const UnifiedMcpPanel: React.FC<UnifiedMcpPanelProps> = ({
|
|||||||
onCancel={() => setConfirmDialog(null)}
|
onCancel={() => setConfirmDialog(null)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
});
|
||||||
|
|
||||||
|
UnifiedMcpPanel.displayName = "UnifiedMcpPanel";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 统一 MCP 列表项组件
|
* 统一 MCP 列表项组件
|
||||||
@@ -259,112 +233,110 @@ const UnifiedMcpListItem: React.FC<UnifiedMcpListItemProps> = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="min-h-16 rounded-lg border border-border-default bg-card p-4 transition-[border-color,box-shadow] duration-200 hover:border-border-hover hover:shadow-sm">
|
<div className="group relative flex items-center gap-4 p-4 rounded-xl border border-border-default bg-muted/50 hover:bg-muted hover:border-border-default/80 hover:shadow-sm transition-all duration-300">
|
||||||
<div className="flex items-center gap-4">
|
{/* 左侧:服务器信息 */}
|
||||||
{/* 左侧:服务器信息 */}
|
<div className="flex-1 min-w-0">
|
||||||
<div className="flex-1 min-w-0">
|
<div className="flex items-center gap-2 mb-1">
|
||||||
<div className="flex items-center gap-2 mb-1">
|
<h3 className="font-medium text-gray-900 dark:text-gray-100">
|
||||||
<h3 className="font-medium text-gray-900 dark:text-gray-100">
|
{name}
|
||||||
{name}
|
</h3>
|
||||||
</h3>
|
{docsUrl && (
|
||||||
{docsUrl && (
|
<Button
|
||||||
<Button
|
type="button"
|
||||||
type="button"
|
variant="ghost"
|
||||||
variant="ghost"
|
size="sm"
|
||||||
size="sm"
|
onClick={openDocs}
|
||||||
onClick={openDocs}
|
title={t("mcp.presets.docs")}
|
||||||
title={t("mcp.presets.docs")}
|
>
|
||||||
>
|
{t("mcp.presets.docs")}
|
||||||
{t("mcp.presets.docs")}
|
</Button>
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
{description && (
|
|
||||||
<p className="text-sm text-gray-500 dark:text-gray-400 line-clamp-2">
|
|
||||||
{description}
|
|
||||||
</p>
|
|
||||||
)}
|
|
||||||
{!description && tags && tags.length > 0 && (
|
|
||||||
<p className="text-xs text-gray-400 dark:text-gray-500 truncate">
|
|
||||||
{tags.join(", ")}
|
|
||||||
</p>
|
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
{description && (
|
||||||
|
<p className="text-sm text-gray-500 dark:text-gray-400 line-clamp-2">
|
||||||
|
{description}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
{!description && tags && tags.length > 0 && (
|
||||||
|
<p className="text-xs text-gray-400 dark:text-gray-500 truncate">
|
||||||
|
{tags.join(", ")}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* 中间:应用开关 */}
|
{/* 中间:应用开关 */}
|
||||||
<div className="flex flex-col gap-2 flex-shrink-0 min-w-[120px]">
|
<div className="flex flex-col gap-2 flex-shrink-0 min-w-[120px]">
|
||||||
<div className="flex items-center justify-between gap-3">
|
<div className="flex items-center justify-between gap-3">
|
||||||
<label
|
<label
|
||||||
htmlFor={`${id}-claude`}
|
htmlFor={`${id}-claude`}
|
||||||
className="text-sm text-gray-700 dark:text-gray-300 cursor-pointer"
|
className="text-sm text-gray-700 dark:text-gray-300 cursor-pointer"
|
||||||
>
|
>
|
||||||
{t("mcp.unifiedPanel.apps.claude")}
|
{t("mcp.unifiedPanel.apps.claude")}
|
||||||
</label>
|
</label>
|
||||||
<Switch
|
<Switch
|
||||||
id={`${id}-claude`}
|
id={`${id}-claude`}
|
||||||
checked={server.apps.claude}
|
checked={server.apps.claude}
|
||||||
onCheckedChange={(checked: boolean) =>
|
onCheckedChange={(checked: boolean) =>
|
||||||
onToggleApp(id, "claude", checked)
|
onToggleApp(id, "claude", checked)
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="flex items-center justify-between gap-3">
|
|
||||||
<label
|
|
||||||
htmlFor={`${id}-codex`}
|
|
||||||
className="text-sm text-gray-700 dark:text-gray-300 cursor-pointer"
|
|
||||||
>
|
|
||||||
{t("mcp.unifiedPanel.apps.codex")}
|
|
||||||
</label>
|
|
||||||
<Switch
|
|
||||||
id={`${id}-codex`}
|
|
||||||
checked={server.apps.codex}
|
|
||||||
onCheckedChange={(checked: boolean) =>
|
|
||||||
onToggleApp(id, "codex", checked)
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="flex items-center justify-between gap-3">
|
|
||||||
<label
|
|
||||||
htmlFor={`${id}-gemini`}
|
|
||||||
className="text-sm text-gray-700 dark:text-gray-300 cursor-pointer"
|
|
||||||
>
|
|
||||||
{t("mcp.unifiedPanel.apps.gemini")}
|
|
||||||
</label>
|
|
||||||
<Switch
|
|
||||||
id={`${id}-gemini`}
|
|
||||||
checked={server.apps.gemini}
|
|
||||||
onCheckedChange={(checked: boolean) =>
|
|
||||||
onToggleApp(id, "gemini", checked)
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* 右侧:操作按钮 */}
|
<div className="flex items-center justify-between gap-3">
|
||||||
<div className="flex items-center gap-2 flex-shrink-0">
|
<label
|
||||||
<Button
|
htmlFor={`${id}-codex`}
|
||||||
type="button"
|
className="text-sm text-gray-700 dark:text-gray-300 cursor-pointer"
|
||||||
variant="ghost"
|
|
||||||
size="icon"
|
|
||||||
onClick={() => onEdit(id)}
|
|
||||||
title={t("common.edit")}
|
|
||||||
>
|
>
|
||||||
<Edit3 size={16} />
|
{t("mcp.unifiedPanel.apps.codex")}
|
||||||
</Button>
|
</label>
|
||||||
|
<Switch
|
||||||
<Button
|
id={`${id}-codex`}
|
||||||
type="button"
|
checked={server.apps.codex}
|
||||||
variant="ghost"
|
onCheckedChange={(checked: boolean) =>
|
||||||
size="icon"
|
onToggleApp(id, "codex", checked)
|
||||||
onClick={() => onDelete(id)}
|
}
|
||||||
className="hover:text-red-500 hover:bg-red-100 dark:hover:text-red-400 dark:hover:bg-red-500/10"
|
/>
|
||||||
title={t("common.delete")}
|
|
||||||
>
|
|
||||||
<Trash2 size={16} />
|
|
||||||
</Button>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div className="flex items-center justify-between gap-3">
|
||||||
|
<label
|
||||||
|
htmlFor={`${id}-gemini`}
|
||||||
|
className="text-sm text-gray-700 dark:text-gray-300 cursor-pointer"
|
||||||
|
>
|
||||||
|
{t("mcp.unifiedPanel.apps.gemini")}
|
||||||
|
</label>
|
||||||
|
<Switch
|
||||||
|
id={`${id}-gemini`}
|
||||||
|
checked={server.apps.gemini}
|
||||||
|
onCheckedChange={(checked: boolean) =>
|
||||||
|
onToggleApp(id, "gemini", checked)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* 右侧:操作按钮 */}
|
||||||
|
<div className="flex items-center gap-2 flex-shrink-0">
|
||||||
|
<Button
|
||||||
|
type="button"
|
||||||
|
variant="ghost"
|
||||||
|
size="icon"
|
||||||
|
onClick={() => onEdit(id)}
|
||||||
|
title={t("common.edit")}
|
||||||
|
>
|
||||||
|
<Edit3 size={16} />
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
type="button"
|
||||||
|
variant="ghost"
|
||||||
|
size="icon"
|
||||||
|
onClick={() => onDelete(id)}
|
||||||
|
className="hover:text-red-500 hover:bg-red-100 dark:hover:text-red-400 dark:hover:bg-red-500/10"
|
||||||
|
title={t("common.delete")}
|
||||||
|
>
|
||||||
|
<Trash2 size={16} />
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
153
src/components/prompts/PromptFormPanel.tsx
Normal file
153
src/components/prompts/PromptFormPanel.tsx
Normal file
@@ -0,0 +1,153 @@
|
|||||||
|
import React, { useState, useEffect } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { Label } from "@/components/ui/label";
|
||||||
|
import MarkdownEditor from "@/components/MarkdownEditor";
|
||||||
|
import { FullScreenPanel } from "@/components/common/FullScreenPanel";
|
||||||
|
import type { Prompt, AppId } from "@/lib/api";
|
||||||
|
|
||||||
|
interface PromptFormPanelProps {
|
||||||
|
appId: AppId;
|
||||||
|
editingId?: string;
|
||||||
|
initialData?: Prompt;
|
||||||
|
onSave: (id: string, prompt: Prompt) => Promise<void>;
|
||||||
|
onClose: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const PromptFormPanel: React.FC<PromptFormPanelProps> = ({
|
||||||
|
appId,
|
||||||
|
editingId,
|
||||||
|
initialData,
|
||||||
|
onSave,
|
||||||
|
onClose,
|
||||||
|
}) => {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const appName = t(`apps.${appId}`);
|
||||||
|
const filenameMap: Record<AppId, string> = {
|
||||||
|
claude: "CLAUDE.md",
|
||||||
|
codex: "AGENTS.md",
|
||||||
|
gemini: "GEMINI.md",
|
||||||
|
};
|
||||||
|
const filename = filenameMap[appId];
|
||||||
|
const [name, setName] = useState("");
|
||||||
|
const [description, setDescription] = useState("");
|
||||||
|
const [content, setContent] = useState("");
|
||||||
|
const [saving, setSaving] = useState(false);
|
||||||
|
const [isDarkMode, setIsDarkMode] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
|
||||||
|
const observer = new MutationObserver(() => {
|
||||||
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
});
|
||||||
|
|
||||||
|
observer.observe(document.documentElement, {
|
||||||
|
attributes: true,
|
||||||
|
attributeFilter: ["class"],
|
||||||
|
});
|
||||||
|
|
||||||
|
return () => observer.disconnect();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (initialData) {
|
||||||
|
setName(initialData.name);
|
||||||
|
setDescription(initialData.description || "");
|
||||||
|
setContent(initialData.content);
|
||||||
|
}
|
||||||
|
}, [initialData]);
|
||||||
|
|
||||||
|
const handleSave = async () => {
|
||||||
|
if (!name.trim() || !content.trim()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setSaving(true);
|
||||||
|
try {
|
||||||
|
const id = editingId || `prompt-${Date.now()}`;
|
||||||
|
const timestamp = Math.floor(Date.now() / 1000);
|
||||||
|
const prompt: Prompt = {
|
||||||
|
id,
|
||||||
|
name: name.trim(),
|
||||||
|
description: description.trim() || undefined,
|
||||||
|
content: content.trim(),
|
||||||
|
enabled: initialData?.enabled || false,
|
||||||
|
createdAt: initialData?.createdAt || timestamp,
|
||||||
|
updatedAt: timestamp,
|
||||||
|
};
|
||||||
|
await onSave(id, prompt);
|
||||||
|
onClose();
|
||||||
|
} catch (error) {
|
||||||
|
// Error handled by hook
|
||||||
|
} finally {
|
||||||
|
setSaving(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const title = editingId
|
||||||
|
? t("prompts.editTitle", { appName })
|
||||||
|
: t("prompts.addTitle", { appName });
|
||||||
|
|
||||||
|
return (
|
||||||
|
<FullScreenPanel
|
||||||
|
isOpen={true}
|
||||||
|
title={title}
|
||||||
|
onClose={onClose}
|
||||||
|
footer={
|
||||||
|
<Button
|
||||||
|
type="button"
|
||||||
|
onClick={handleSave}
|
||||||
|
disabled={!name.trim() || !content.trim() || saving}
|
||||||
|
className="bg-primary text-primary-foreground hover:bg-primary/90 disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
{saving ? t("common.saving") : t("common.save")}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<div className="glass rounded-xl p-6 border border-white/10 space-y-6">
|
||||||
|
<div>
|
||||||
|
<Label htmlFor="name" className="text-foreground">
|
||||||
|
{t("prompts.name")}
|
||||||
|
</Label>
|
||||||
|
<Input
|
||||||
|
id="name"
|
||||||
|
value={name}
|
||||||
|
onChange={(e) => setName(e.target.value)}
|
||||||
|
placeholder={t("prompts.namePlaceholder")}
|
||||||
|
className="mt-2"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<Label htmlFor="description" className="text-foreground">
|
||||||
|
{t("prompts.description")}
|
||||||
|
</Label>
|
||||||
|
<Input
|
||||||
|
id="description"
|
||||||
|
value={description}
|
||||||
|
onChange={(e) => setDescription(e.target.value)}
|
||||||
|
placeholder={t("prompts.descriptionPlaceholder")}
|
||||||
|
className="mt-2"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<Label htmlFor="content" className="block mb-2 text-foreground">
|
||||||
|
{t("prompts.content")}
|
||||||
|
</Label>
|
||||||
|
<MarkdownEditor
|
||||||
|
value={content}
|
||||||
|
onChange={setContent}
|
||||||
|
placeholder={t("prompts.contentPlaceholder", { filename })}
|
||||||
|
darkMode={isDarkMode}
|
||||||
|
minHeight="167px"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</FullScreenPanel>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default PromptFormPanel;
|
||||||
@@ -25,7 +25,7 @@ const PromptListItem: React.FC<PromptListItemProps> = ({
|
|||||||
const enabled = prompt.enabled === true;
|
const enabled = prompt.enabled === true;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="h-16 rounded-lg border border-border-default bg-card p-4 transition-[border-color,box-shadow] duration-200 hover:border-border-hover hover:shadow-sm">
|
<div className="group relative h-16 rounded-xl border border-border-default bg-muted/50 p-4 transition-all duration-300 hover:bg-muted hover:border-border-default/80 hover:shadow-sm">
|
||||||
<div className="flex items-center gap-4 h-full">
|
<div className="flex items-center gap-4 h-full">
|
||||||
{/* Toggle 开关 */}
|
{/* Toggle 开关 */}
|
||||||
<div className="flex-shrink-0">
|
<div className="flex-shrink-0">
|
||||||
|
|||||||
@@ -1,18 +1,10 @@
|
|||||||
import React, { useEffect, useMemo, useState } from "react";
|
import React, { useEffect, useMemo, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { Plus, FileText, Check } from "lucide-react";
|
import { FileText } from "lucide-react";
|
||||||
import { Button } from "@/components/ui/button";
|
|
||||||
import {
|
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogFooter,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { type AppId } from "@/lib/api";
|
import { type AppId } from "@/lib/api";
|
||||||
import { usePromptActions } from "@/hooks/usePromptActions";
|
import { usePromptActions } from "@/hooks/usePromptActions";
|
||||||
import PromptListItem from "./PromptListItem";
|
import PromptListItem from "./PromptListItem";
|
||||||
import PromptFormModal from "./PromptFormModal";
|
import PromptFormPanel from "./PromptFormPanel";
|
||||||
import { ConfirmDialog } from "../ConfirmDialog";
|
import { ConfirmDialog } from "../ConfirmDialog";
|
||||||
|
|
||||||
interface PromptPanelProps {
|
interface PromptPanelProps {
|
||||||
@@ -21,157 +13,143 @@ interface PromptPanelProps {
|
|||||||
appId: AppId;
|
appId: AppId;
|
||||||
}
|
}
|
||||||
|
|
||||||
const PromptPanel: React.FC<PromptPanelProps> = ({
|
export interface PromptPanelHandle {
|
||||||
open,
|
openAdd: () => void;
|
||||||
onOpenChange,
|
}
|
||||||
appId,
|
|
||||||
}) => {
|
|
||||||
const { t } = useTranslation();
|
|
||||||
const [isFormOpen, setIsFormOpen] = useState(false);
|
|
||||||
const [editingId, setEditingId] = useState<string | null>(null);
|
|
||||||
const [confirmDialog, setConfirmDialog] = useState<{
|
|
||||||
isOpen: boolean;
|
|
||||||
titleKey: string;
|
|
||||||
messageKey: string;
|
|
||||||
messageParams?: Record<string, unknown>;
|
|
||||||
onConfirm: () => void;
|
|
||||||
} | null>(null);
|
|
||||||
|
|
||||||
const { prompts, loading, reload, savePrompt, deletePrompt, toggleEnabled } =
|
const PromptPanel = React.forwardRef<PromptPanelHandle, PromptPanelProps>(
|
||||||
usePromptActions(appId);
|
({ open, appId }, ref) => {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [isFormOpen, setIsFormOpen] = useState(false);
|
||||||
|
const [editingId, setEditingId] = useState<string | null>(null);
|
||||||
|
const [confirmDialog, setConfirmDialog] = useState<{
|
||||||
|
isOpen: boolean;
|
||||||
|
titleKey: string;
|
||||||
|
messageKey: string;
|
||||||
|
messageParams?: Record<string, unknown>;
|
||||||
|
onConfirm: () => void;
|
||||||
|
} | null>(null);
|
||||||
|
|
||||||
useEffect(() => {
|
const {
|
||||||
if (open) reload();
|
prompts,
|
||||||
}, [open, reload]);
|
loading,
|
||||||
|
reload,
|
||||||
|
savePrompt,
|
||||||
|
deletePrompt,
|
||||||
|
toggleEnabled,
|
||||||
|
} = usePromptActions(appId);
|
||||||
|
|
||||||
const handleAdd = () => {
|
useEffect(() => {
|
||||||
setEditingId(null);
|
if (open) reload();
|
||||||
setIsFormOpen(true);
|
}, [open, reload]);
|
||||||
};
|
|
||||||
|
|
||||||
const handleEdit = (id: string) => {
|
const handleAdd = () => {
|
||||||
setEditingId(id);
|
setEditingId(null);
|
||||||
setIsFormOpen(true);
|
setIsFormOpen(true);
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleDelete = (id: string) => {
|
React.useImperativeHandle(ref, () => ({
|
||||||
const prompt = prompts[id];
|
openAdd: handleAdd,
|
||||||
setConfirmDialog({
|
}));
|
||||||
isOpen: true,
|
|
||||||
titleKey: "prompts.confirm.deleteTitle",
|
|
||||||
messageKey: "prompts.confirm.deleteMessage",
|
|
||||||
messageParams: { name: prompt?.name },
|
|
||||||
onConfirm: async () => {
|
|
||||||
try {
|
|
||||||
await deletePrompt(id);
|
|
||||||
setConfirmDialog(null);
|
|
||||||
} catch (e) {
|
|
||||||
// Error handled by hook
|
|
||||||
}
|
|
||||||
},
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const promptEntries = useMemo(() => Object.entries(prompts), [prompts]);
|
const handleEdit = (id: string) => {
|
||||||
|
setEditingId(id);
|
||||||
|
setIsFormOpen(true);
|
||||||
|
};
|
||||||
|
|
||||||
const enabledPrompt = promptEntries.find(([_, p]) => p.enabled);
|
const handleDelete = (id: string) => {
|
||||||
|
const prompt = prompts[id];
|
||||||
|
setConfirmDialog({
|
||||||
|
isOpen: true,
|
||||||
|
titleKey: "prompts.confirm.deleteTitle",
|
||||||
|
messageKey: "prompts.confirm.deleteMessage",
|
||||||
|
messageParams: { name: prompt?.name },
|
||||||
|
onConfirm: async () => {
|
||||||
|
try {
|
||||||
|
await deletePrompt(id);
|
||||||
|
setConfirmDialog(null);
|
||||||
|
} catch (e) {
|
||||||
|
// Error handled by hook
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
const appName = t(`apps.${appId}`);
|
const promptEntries = useMemo(() => Object.entries(prompts), [prompts]);
|
||||||
const panelTitle = t("prompts.title", { appName });
|
|
||||||
|
|
||||||
return (
|
const enabledPrompt = promptEntries.find(([_, p]) => p.enabled);
|
||||||
<>
|
|
||||||
<Dialog open={open} onOpenChange={onOpenChange}>
|
|
||||||
<DialogContent className="max-w-3xl max-h-[85vh] min-h-[600px] flex flex-col">
|
|
||||||
<DialogHeader>
|
|
||||||
<div className="flex items-center justify-between pr-8">
|
|
||||||
<DialogTitle>{panelTitle}</DialogTitle>
|
|
||||||
<Button type="button" variant="mcp" onClick={handleAdd}>
|
|
||||||
<Plus size={16} />
|
|
||||||
{t("prompts.add")}
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</DialogHeader>
|
|
||||||
|
|
||||||
<div className="flex-shrink-0 px-6 py-4">
|
return (
|
||||||
<div className="text-sm text-gray-500 dark:text-gray-400">
|
<div className="mx-auto max-w-[56rem] flex flex-col h-[calc(100vh-8rem)] px-6">
|
||||||
{t("prompts.count", { count: promptEntries.length })} ·{" "}
|
<div className="flex-shrink-0 py-4 glass rounded-xl border border-white/10 mb-4 px-6">
|
||||||
{enabledPrompt
|
<div className="text-sm text-muted-foreground">
|
||||||
? t("prompts.enabledName", { name: enabledPrompt[1].name })
|
{t("prompts.count", { count: promptEntries.length })} ·{" "}
|
||||||
: t("prompts.noneEnabled")}
|
{enabledPrompt
|
||||||
</div>
|
? t("prompts.enabledName", { name: enabledPrompt[1].name })
|
||||||
|
: t("prompts.noneEnabled")}
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div className="flex-1 overflow-y-auto px-6 pb-4">
|
<div className="flex-1 overflow-y-auto pb-16">
|
||||||
{loading ? (
|
{loading ? (
|
||||||
<div className="text-center py-12 text-gray-500 dark:text-gray-400">
|
<div className="text-center py-12 text-gray-500 dark:text-gray-400">
|
||||||
{t("prompts.loading")}
|
{t("prompts.loading")}
|
||||||
|
</div>
|
||||||
|
) : promptEntries.length === 0 ? (
|
||||||
|
<div className="text-center py-12">
|
||||||
|
<div className="w-16 h-16 mx-auto mb-4 bg-gray-100 dark:bg-gray-800 rounded-full flex items-center justify-center">
|
||||||
|
<FileText
|
||||||
|
size={24}
|
||||||
|
className="text-gray-400 dark:text-gray-500"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
) : promptEntries.length === 0 ? (
|
<h3 className="text-lg font-medium text-gray-900 dark:text-gray-100 mb-2">
|
||||||
<div className="text-center py-12">
|
{t("prompts.empty")}
|
||||||
<div className="w-16 h-16 mx-auto mb-4 bg-gray-100 dark:bg-gray-800 rounded-full flex items-center justify-center">
|
</h3>
|
||||||
<FileText
|
<p className="text-gray-500 dark:text-gray-400 text-sm">
|
||||||
size={24}
|
{t("prompts.emptyDescription")}
|
||||||
className="text-gray-400 dark:text-gray-500"
|
</p>
|
||||||
/>
|
</div>
|
||||||
</div>
|
) : (
|
||||||
<h3 className="text-lg font-medium text-gray-900 dark:text-gray-100 mb-2">
|
<div className="space-y-3">
|
||||||
{t("prompts.empty")}
|
{promptEntries.map(([id, prompt]) => (
|
||||||
</h3>
|
<PromptListItem
|
||||||
<p className="text-gray-500 dark:text-gray-400 text-sm">
|
key={id}
|
||||||
{t("prompts.emptyDescription")}
|
id={id}
|
||||||
</p>
|
prompt={prompt}
|
||||||
</div>
|
onToggle={toggleEnabled}
|
||||||
) : (
|
onEdit={handleEdit}
|
||||||
<div className="space-y-3">
|
onDelete={handleDelete}
|
||||||
{promptEntries.map(([id, prompt]) => (
|
/>
|
||||||
<PromptListItem
|
))}
|
||||||
key={id}
|
</div>
|
||||||
id={id}
|
)}
|
||||||
prompt={prompt}
|
</div>
|
||||||
onToggle={toggleEnabled}
|
|
||||||
onEdit={handleEdit}
|
|
||||||
onDelete={handleDelete}
|
|
||||||
/>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<DialogFooter>
|
{isFormOpen && (
|
||||||
<Button
|
<PromptFormPanel
|
||||||
type="button"
|
appId={appId}
|
||||||
variant="mcp"
|
editingId={editingId || undefined}
|
||||||
onClick={() => onOpenChange(false)}
|
initialData={editingId ? prompts[editingId] : undefined}
|
||||||
>
|
onSave={savePrompt}
|
||||||
<Check size={16} />
|
onClose={() => setIsFormOpen(false)}
|
||||||
{t("common.done")}
|
/>
|
||||||
</Button>
|
)}
|
||||||
</DialogFooter>
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
|
|
||||||
{isFormOpen && (
|
{confirmDialog && (
|
||||||
<PromptFormModal
|
<ConfirmDialog
|
||||||
appId={appId}
|
isOpen={confirmDialog.isOpen}
|
||||||
editingId={editingId || undefined}
|
title={t(confirmDialog.titleKey)}
|
||||||
initialData={editingId ? prompts[editingId] : undefined}
|
message={t(confirmDialog.messageKey, confirmDialog.messageParams)}
|
||||||
onSave={savePrompt}
|
onConfirm={confirmDialog.onConfirm}
|
||||||
onClose={() => setIsFormOpen(false)}
|
onCancel={() => setConfirmDialog(null)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
{confirmDialog && (
|
PromptPanel.displayName = "PromptPanel";
|
||||||
<ConfirmDialog
|
|
||||||
isOpen={confirmDialog.isOpen}
|
|
||||||
title={t(confirmDialog.titleKey)}
|
|
||||||
message={t(confirmDialog.messageKey, confirmDialog.messageParams)}
|
|
||||||
onConfirm={confirmDialog.onConfirm}
|
|
||||||
onCancel={() => setConfirmDialog(null)}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default PromptPanel;
|
export default PromptPanel;
|
||||||
|
|||||||
@@ -1,15 +1,8 @@
|
|||||||
import { useCallback } from "react";
|
import { useCallback } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { Plus } from "lucide-react";
|
import { Plus } from "lucide-react";
|
||||||
import {
|
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogDescription,
|
|
||||||
DialogFooter,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { FullScreenPanel } from "@/components/common/FullScreenPanel";
|
||||||
import type { Provider, CustomEndpoint } from "@/types";
|
import type { Provider, CustomEndpoint } from "@/types";
|
||||||
import type { AppId } from "@/lib/api";
|
import type { AppId } from "@/lib/api";
|
||||||
import {
|
import {
|
||||||
@@ -48,6 +41,8 @@ export function AddProviderDialog({
|
|||||||
notes: values.notes?.trim() || undefined,
|
notes: values.notes?.trim() || undefined,
|
||||||
websiteUrl: values.websiteUrl?.trim() || undefined,
|
websiteUrl: values.websiteUrl?.trim() || undefined,
|
||||||
settingsConfig: parsedConfig,
|
settingsConfig: parsedConfig,
|
||||||
|
icon: values.icon?.trim() || undefined,
|
||||||
|
iconColor: values.iconColor?.trim() || undefined,
|
||||||
...(values.presetCategory ? { category: values.presetCategory } : {}),
|
...(values.presetCategory ? { category: values.presetCategory } : {}),
|
||||||
...(values.meta ? { meta: values.meta } : {}),
|
...(values.meta ? { meta: values.meta } : {}),
|
||||||
};
|
};
|
||||||
@@ -58,8 +53,6 @@ export function AddProviderDialog({
|
|||||||
|
|
||||||
if (!hasCustomEndpoints) {
|
if (!hasCustomEndpoints) {
|
||||||
// 收集端点候选(仅在缺少自定义端点时兜底)
|
// 收集端点候选(仅在缺少自定义端点时兜底)
|
||||||
// 1. 从预设配置中获取 endpointCandidates
|
|
||||||
// 2. 从当前配置中提取 baseUrl (ANTHROPIC_BASE_URL 或 Codex base_url)
|
|
||||||
const urlSet = new Set<string>();
|
const urlSet = new Set<string>();
|
||||||
|
|
||||||
const addUrl = (rawUrl?: string) => {
|
const addUrl = (rawUrl?: string) => {
|
||||||
@@ -170,34 +163,40 @@ export function AddProviderDialog({
|
|||||||
? t("provider.addCodexProvider")
|
? t("provider.addCodexProvider")
|
||||||
: t("provider.addGeminiProvider");
|
: t("provider.addGeminiProvider");
|
||||||
|
|
||||||
|
const footer = (
|
||||||
|
<>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
onClick={() => onOpenChange(false)}
|
||||||
|
className="border-border/20 hover:bg-accent hover:text-accent-foreground"
|
||||||
|
>
|
||||||
|
{t("common.cancel")}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
type="submit"
|
||||||
|
form="provider-form"
|
||||||
|
className="bg-primary text-primary-foreground hover:bg-primary/90"
|
||||||
|
>
|
||||||
|
<Plus className="h-4 w-4 mr-2" />
|
||||||
|
{t("common.add")}
|
||||||
|
</Button>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog open={open} onOpenChange={onOpenChange}>
|
<FullScreenPanel
|
||||||
<DialogContent className="max-w-3xl max-h-[85vh] min-h-[600px] flex flex-col">
|
isOpen={open}
|
||||||
<DialogHeader>
|
title={submitLabel}
|
||||||
<DialogTitle>{submitLabel}</DialogTitle>
|
onClose={() => onOpenChange(false)}
|
||||||
<DialogDescription>{t("provider.addProviderHint")}</DialogDescription>
|
footer={footer}
|
||||||
</DialogHeader>
|
>
|
||||||
|
<ProviderForm
|
||||||
<div className="flex-1 overflow-y-auto px-6 py-4">
|
appId={appId}
|
||||||
<ProviderForm
|
submitLabel={t("common.add")}
|
||||||
appId={appId}
|
onSubmit={handleSubmit}
|
||||||
submitLabel={t("common.add")}
|
onCancel={() => onOpenChange(false)}
|
||||||
onSubmit={handleSubmit}
|
showButtons={false}
|
||||||
onCancel={() => onOpenChange(false)}
|
/>
|
||||||
showButtons={false}
|
</FullScreenPanel>
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<DialogFooter>
|
|
||||||
<Button variant="outline" onClick={() => onOpenChange(false)}>
|
|
||||||
{t("common.cancel")}
|
|
||||||
</Button>
|
|
||||||
<Button type="submit" form="provider-form">
|
|
||||||
<Plus className="h-4 w-4" />
|
|
||||||
{t("common.add")}
|
|
||||||
</Button>
|
|
||||||
</DialogFooter>
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,15 +1,8 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from "react";
|
import { useCallback, useEffect, useMemo, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { Save } from "lucide-react";
|
import { Save } from "lucide-react";
|
||||||
import {
|
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogDescription,
|
|
||||||
DialogFooter,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { FullScreenPanel } from "@/components/common/FullScreenPanel";
|
||||||
import type { Provider } from "@/types";
|
import type { Provider } from "@/types";
|
||||||
import {
|
import {
|
||||||
ProviderForm,
|
ProviderForm,
|
||||||
@@ -34,7 +27,7 @@ export function EditProviderDialog({
|
|||||||
}: EditProviderDialogProps) {
|
}: EditProviderDialogProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
// 默认使用传入的 provider.settingsConfig,若当前编辑对象是“当前生效供应商”,则尝试读取实时配置替换初始值
|
// 默认使用传入的 provider.settingsConfig,若当前编辑对象是"当前生效供应商",则尝试读取实时配置替换初始值
|
||||||
const [liveSettings, setLiveSettings] = useState<Record<
|
const [liveSettings, setLiveSettings] = useState<Record<
|
||||||
string,
|
string,
|
||||||
unknown
|
unknown
|
||||||
@@ -96,6 +89,8 @@ export function EditProviderDialog({
|
|||||||
notes: values.notes?.trim() || undefined,
|
notes: values.notes?.trim() || undefined,
|
||||||
websiteUrl: values.websiteUrl?.trim() || undefined,
|
websiteUrl: values.websiteUrl?.trim() || undefined,
|
||||||
settingsConfig: parsedConfig,
|
settingsConfig: parsedConfig,
|
||||||
|
icon: values.icon?.trim() || undefined,
|
||||||
|
iconColor: values.iconColor?.trim() || undefined,
|
||||||
...(values.presetCategory ? { category: values.presetCategory } : {}),
|
...(values.presetCategory ? { category: values.presetCategory } : {}),
|
||||||
// 保留或更新 meta 字段
|
// 保留或更新 meta 字段
|
||||||
...(values.meta ? { meta: values.meta } : {}),
|
...(values.meta ? { meta: values.meta } : {}),
|
||||||
@@ -112,45 +107,40 @@ export function EditProviderDialog({
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog open={open} onOpenChange={onOpenChange}>
|
<FullScreenPanel
|
||||||
<DialogContent className="max-w-3xl max-h-[85vh] min-h-[600px] flex flex-col">
|
isOpen={open}
|
||||||
<DialogHeader>
|
title={t("provider.editProvider")}
|
||||||
<DialogTitle>{t("provider.editProvider")}</DialogTitle>
|
onClose={() => onOpenChange(false)}
|
||||||
<DialogDescription>
|
footer={
|
||||||
{t("provider.editProviderHint")}
|
<Button
|
||||||
</DialogDescription>
|
type="submit"
|
||||||
</DialogHeader>
|
form="provider-form"
|
||||||
|
className="bg-primary text-primary-foreground hover:bg-primary/90"
|
||||||
<div className="flex-1 overflow-y-auto px-6 py-4">
|
>
|
||||||
<ProviderForm
|
<Save className="h-4 w-4 mr-2" />
|
||||||
appId={appId}
|
{t("common.save")}
|
||||||
providerId={provider.id}
|
</Button>
|
||||||
submitLabel={t("common.save")}
|
}
|
||||||
onSubmit={handleSubmit}
|
>
|
||||||
onCancel={() => onOpenChange(false)}
|
<ProviderForm
|
||||||
initialData={{
|
appId={appId}
|
||||||
name: provider.name,
|
providerId={provider.id}
|
||||||
notes: provider.notes,
|
submitLabel={t("common.save")}
|
||||||
websiteUrl: provider.websiteUrl,
|
onSubmit={handleSubmit}
|
||||||
// 若读取到实时配置则优先使用
|
onCancel={() => onOpenChange(false)}
|
||||||
settingsConfig: initialSettingsConfig,
|
initialData={{
|
||||||
category: provider.category,
|
name: provider.name,
|
||||||
meta: provider.meta,
|
notes: provider.notes,
|
||||||
}}
|
websiteUrl: provider.websiteUrl,
|
||||||
showButtons={false}
|
// 若读取到实时配置则优先使用
|
||||||
/>
|
settingsConfig: initialSettingsConfig,
|
||||||
</div>
|
category: provider.category,
|
||||||
|
meta: provider.meta,
|
||||||
<DialogFooter>
|
icon: provider.icon,
|
||||||
<Button variant="outline" onClick={() => onOpenChange(false)}>
|
iconColor: provider.iconColor,
|
||||||
{t("common.cancel")}
|
}}
|
||||||
</Button>
|
showButtons={false}
|
||||||
<Button type="submit" form="provider-form">
|
/>
|
||||||
<Save className="h-4 w-4" />
|
</FullScreenPanel>
|
||||||
{t("common.save")}
|
|
||||||
</Button>
|
|
||||||
</DialogFooter>
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { BarChart3, Check, Edit, Play, Trash2 } from "lucide-react";
|
import { BarChart3, Check, Copy, Edit, Play, Trash2 } from "lucide-react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { cn } from "@/lib/utils";
|
import { cn } from "@/lib/utils";
|
||||||
@@ -7,6 +7,7 @@ interface ProviderActionsProps {
|
|||||||
isCurrent: boolean;
|
isCurrent: boolean;
|
||||||
onSwitch: () => void;
|
onSwitch: () => void;
|
||||||
onEdit: () => void;
|
onEdit: () => void;
|
||||||
|
onDuplicate: () => void;
|
||||||
onConfigureUsage: () => void;
|
onConfigureUsage: () => void;
|
||||||
onDelete: () => void;
|
onDelete: () => void;
|
||||||
}
|
}
|
||||||
@@ -15,20 +16,22 @@ export function ProviderActions({
|
|||||||
isCurrent,
|
isCurrent,
|
||||||
onSwitch,
|
onSwitch,
|
||||||
onEdit,
|
onEdit,
|
||||||
|
onDuplicate,
|
||||||
onConfigureUsage,
|
onConfigureUsage,
|
||||||
onDelete,
|
onDelete,
|
||||||
}: ProviderActionsProps) {
|
}: ProviderActionsProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
const iconButtonClass = "h-8 w-8 p-1";
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-1.5">
|
||||||
<Button
|
<Button
|
||||||
size="sm"
|
size="sm"
|
||||||
variant={isCurrent ? "secondary" : "default"}
|
variant={isCurrent ? "secondary" : "default"}
|
||||||
onClick={onSwitch}
|
onClick={onSwitch}
|
||||||
disabled={isCurrent}
|
disabled={isCurrent}
|
||||||
className={cn(
|
className={cn(
|
||||||
"w-20",
|
"w-[4.5rem] px-2.5",
|
||||||
isCurrent &&
|
isCurrent &&
|
||||||
"bg-gray-200 text-muted-foreground hover:bg-gray-200 hover:text-muted-foreground dark:bg-gray-700 dark:hover:bg-gray-700",
|
"bg-gray-200 text-muted-foreground hover:bg-gray-200 hover:text-muted-foreground dark:bg-gray-700 dark:hover:bg-gray-700",
|
||||||
)}
|
)}
|
||||||
@@ -52,15 +55,27 @@ export function ProviderActions({
|
|||||||
variant="ghost"
|
variant="ghost"
|
||||||
onClick={onEdit}
|
onClick={onEdit}
|
||||||
title={t("common.edit")}
|
title={t("common.edit")}
|
||||||
|
className={iconButtonClass}
|
||||||
>
|
>
|
||||||
<Edit className="h-4 w-4" />
|
<Edit className="h-4 w-4" />
|
||||||
</Button>
|
</Button>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
size="icon"
|
||||||
|
variant="ghost"
|
||||||
|
onClick={onDuplicate}
|
||||||
|
title={t("provider.duplicate")}
|
||||||
|
className={iconButtonClass}
|
||||||
|
>
|
||||||
|
<Copy className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
|
||||||
<Button
|
<Button
|
||||||
size="icon"
|
size="icon"
|
||||||
variant="ghost"
|
variant="ghost"
|
||||||
onClick={onConfigureUsage}
|
onClick={onConfigureUsage}
|
||||||
title={t("provider.configureUsage")}
|
title={t("provider.configureUsage")}
|
||||||
|
className={iconButtonClass}
|
||||||
>
|
>
|
||||||
<BarChart3 className="h-4 w-4" />
|
<BarChart3 className="h-4 w-4" />
|
||||||
</Button>
|
</Button>
|
||||||
@@ -71,6 +86,7 @@ export function ProviderActions({
|
|||||||
onClick={isCurrent ? undefined : onDelete}
|
onClick={isCurrent ? undefined : onDelete}
|
||||||
title={t("common.delete")}
|
title={t("common.delete")}
|
||||||
className={cn(
|
className={cn(
|
||||||
|
iconButtonClass,
|
||||||
!isCurrent && "hover:text-red-500 dark:hover:text-red-400",
|
!isCurrent && "hover:text-red-500 dark:hover:text-red-400",
|
||||||
isCurrent && "opacity-40 cursor-not-allowed text-muted-foreground",
|
isCurrent && "opacity-40 cursor-not-allowed text-muted-foreground",
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { useMemo } from "react";
|
import { useMemo } from "react";
|
||||||
import { MoveVertical, Copy } from "lucide-react";
|
import { GripVertical } from "lucide-react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import type {
|
import type {
|
||||||
DraggableAttributes,
|
DraggableAttributes,
|
||||||
@@ -8,8 +8,8 @@ import type {
|
|||||||
import type { Provider } from "@/types";
|
import type { Provider } from "@/types";
|
||||||
import type { AppId } from "@/lib/api";
|
import type { AppId } from "@/lib/api";
|
||||||
import { cn } from "@/lib/utils";
|
import { cn } from "@/lib/utils";
|
||||||
import { Button } from "@/components/ui/button";
|
|
||||||
import { ProviderActions } from "@/components/providers/ProviderActions";
|
import { ProviderActions } from "@/components/providers/ProviderActions";
|
||||||
|
import { ProviderIcon } from "@/components/ProviderIcon";
|
||||||
import UsageFooter from "@/components/UsageFooter";
|
import UsageFooter from "@/components/UsageFooter";
|
||||||
|
|
||||||
interface DragHandleProps {
|
interface DragHandleProps {
|
||||||
@@ -22,7 +22,6 @@ interface ProviderCardProps {
|
|||||||
provider: Provider;
|
provider: Provider;
|
||||||
isCurrent: boolean;
|
isCurrent: boolean;
|
||||||
appId: AppId;
|
appId: AppId;
|
||||||
isEditMode?: boolean;
|
|
||||||
onSwitch: (provider: Provider) => void;
|
onSwitch: (provider: Provider) => void;
|
||||||
onEdit: (provider: Provider) => void;
|
onEdit: (provider: Provider) => void;
|
||||||
onDelete: (provider: Provider) => void;
|
onDelete: (provider: Provider) => void;
|
||||||
@@ -71,7 +70,6 @@ export function ProviderCard({
|
|||||||
provider,
|
provider,
|
||||||
isCurrent,
|
isCurrent,
|
||||||
appId,
|
appId,
|
||||||
isEditMode = false,
|
|
||||||
onSwitch,
|
onSwitch,
|
||||||
onEdit,
|
onEdit,
|
||||||
onDelete,
|
onDelete,
|
||||||
@@ -116,53 +114,40 @@ export function ProviderCard({
|
|||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={cn(
|
className={cn(
|
||||||
"rounded-lg bg-card p-4 shadow-sm",
|
"glass-card relative overflow-hidden rounded-xl p-4 transition-all duration-300",
|
||||||
"transition-[border-color,background-color,box-shadow,ring] duration-200",
|
"group hover:bg-black/[0.02] dark:hover:bg-white/[0.02] hover:border-primary/50",
|
||||||
isCurrent
|
isCurrent
|
||||||
? "border border-border-default bg-primary/5 ring-2 ring-blue-500/30 dark:ring-blue-400/30"
|
? "border-primary/50 bg-primary/5 shadow-[0_0_20px_rgba(59,130,246,0.15)]"
|
||||||
: "border border-border-default hover:border-border-hover",
|
: "hover:scale-[1.01]",
|
||||||
dragHandleProps?.isDragging &&
|
dragHandleProps?.isDragging &&
|
||||||
"cursor-grabbing border-active border-border-dragging shadow-lg",
|
"cursor-grabbing border-primary shadow-lg scale-105 z-10",
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
<div className="flex flex-col gap-4 sm:flex-row sm:items-center sm:justify-between">
|
<div className="absolute inset-0 bg-gradient-to-r from-primary/10 to-transparent opacity-0 group-hover:opacity-100 transition-opacity duration-500 pointer-events-none" />
|
||||||
|
<div className="relative flex flex-col gap-4 sm:flex-row sm:items-center sm:justify-between">
|
||||||
<div className="flex flex-1 items-center gap-2">
|
<div className="flex flex-1 items-center gap-2">
|
||||||
<div
|
<button
|
||||||
|
type="button"
|
||||||
className={cn(
|
className={cn(
|
||||||
"flex items-center gap-1 overflow-hidden",
|
"-ml-1.5 flex-shrink-0 cursor-grab active:cursor-grabbing p-1.5",
|
||||||
"transition-[max-width,opacity] duration-200 ease-in-out",
|
"text-muted-foreground/50 hover:text-muted-foreground transition-colors",
|
||||||
isEditMode ? "max-w-20 opacity-100" : "max-w-0 opacity-0",
|
dragHandleProps?.isDragging && "cursor-grabbing",
|
||||||
)}
|
)}
|
||||||
aria-hidden={!isEditMode}
|
aria-label={t("provider.dragHandle")}
|
||||||
|
{...(dragHandleProps?.attributes ?? {})}
|
||||||
|
{...(dragHandleProps?.listeners ?? {})}
|
||||||
>
|
>
|
||||||
<Button
|
<GripVertical className="h-4 w-4" />
|
||||||
type="button"
|
</button>
|
||||||
size="icon"
|
|
||||||
variant="ghost"
|
|
||||||
className={cn(
|
|
||||||
"flex-shrink-0 cursor-grab active:cursor-grabbing",
|
|
||||||
dragHandleProps?.isDragging && "cursor-grabbing",
|
|
||||||
)}
|
|
||||||
aria-label={t("provider.dragHandle")}
|
|
||||||
disabled={!isEditMode}
|
|
||||||
{...(dragHandleProps?.attributes ?? {})}
|
|
||||||
{...(dragHandleProps?.listeners ?? {})}
|
|
||||||
>
|
|
||||||
<MoveVertical className="h-4 w-4" />
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
<Button
|
{/* 供应商图标 */}
|
||||||
type="button"
|
<div className="h-9 w-9 rounded-lg bg-white/5 flex items-center justify-center border border-gray-200 dark:border-white/10 group-hover:scale-105 transition-transform duration-300">
|
||||||
size="icon"
|
<ProviderIcon
|
||||||
variant="ghost"
|
icon={provider.icon}
|
||||||
className="flex-shrink-0"
|
name={provider.name}
|
||||||
onClick={() => onDuplicate(provider)}
|
color={provider.iconColor}
|
||||||
disabled={!isEditMode}
|
size={26}
|
||||||
aria-label={t("provider.duplicate")}
|
/>
|
||||||
title={t("provider.duplicate")}
|
|
||||||
>
|
|
||||||
<Copy className="h-4 w-4" />
|
|
||||||
</Button>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="space-y-1">
|
<div className="space-y-1">
|
||||||
@@ -210,23 +195,28 @@ export function ProviderCard({
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex items-center gap-3">
|
<div className="relative flex items-center ml-auto">
|
||||||
<UsageFooter
|
<div className="ml-auto transition-transform duration-200 group-hover:-translate-x-[12.25rem] group-focus-within:-translate-x-[12.25rem] sm:group-hover:-translate-x-[14.25rem] sm:group-focus-within:-translate-x-[14.25rem]">
|
||||||
provider={provider}
|
<UsageFooter
|
||||||
providerId={provider.id}
|
provider={provider}
|
||||||
appId={appId}
|
providerId={provider.id}
|
||||||
usageEnabled={usageEnabled}
|
appId={appId}
|
||||||
isCurrent={isCurrent}
|
usageEnabled={usageEnabled}
|
||||||
inline={true}
|
isCurrent={isCurrent}
|
||||||
/>
|
inline={true}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<ProviderActions
|
<div className="absolute right-0 top-1/2 -translate-y-1/2 flex items-center gap-1.5 opacity-0 pointer-events-none group-hover:opacity-100 group-focus-within:opacity-100 group-hover:pointer-events-auto group-focus-within:pointer-events-auto transition-all duration-200 translate-x-2 group-hover:translate-x-0 group-focus-within:translate-x-0">
|
||||||
isCurrent={isCurrent}
|
<ProviderActions
|
||||||
onSwitch={() => onSwitch(provider)}
|
isCurrent={isCurrent}
|
||||||
onEdit={() => onEdit(provider)}
|
onSwitch={() => onSwitch(provider)}
|
||||||
onConfigureUsage={() => onConfigureUsage(provider)}
|
onEdit={() => onEdit(provider)}
|
||||||
onDelete={() => onDelete(provider)}
|
onDuplicate={() => onDuplicate(provider)}
|
||||||
/>
|
onConfigureUsage={() => onConfigureUsage(provider)}
|
||||||
|
onDelete={() => onDelete(provider)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -16,7 +16,6 @@ interface ProviderListProps {
|
|||||||
providers: Record<string, Provider>;
|
providers: Record<string, Provider>;
|
||||||
currentProviderId: string;
|
currentProviderId: string;
|
||||||
appId: AppId;
|
appId: AppId;
|
||||||
isEditMode?: boolean;
|
|
||||||
onSwitch: (provider: Provider) => void;
|
onSwitch: (provider: Provider) => void;
|
||||||
onEdit: (provider: Provider) => void;
|
onEdit: (provider: Provider) => void;
|
||||||
onDelete: (provider: Provider) => void;
|
onDelete: (provider: Provider) => void;
|
||||||
@@ -31,7 +30,6 @@ export function ProviderList({
|
|||||||
providers,
|
providers,
|
||||||
currentProviderId,
|
currentProviderId,
|
||||||
appId,
|
appId,
|
||||||
isEditMode = false,
|
|
||||||
onSwitch,
|
onSwitch,
|
||||||
onEdit,
|
onEdit,
|
||||||
onDelete,
|
onDelete,
|
||||||
@@ -73,14 +71,16 @@ export function ProviderList({
|
|||||||
items={sortedProviders.map((provider) => provider.id)}
|
items={sortedProviders.map((provider) => provider.id)}
|
||||||
strategy={verticalListSortingStrategy}
|
strategy={verticalListSortingStrategy}
|
||||||
>
|
>
|
||||||
<div className="space-y-3">
|
<div
|
||||||
|
className="space-y-3 animate-slide-up"
|
||||||
|
style={{ animationDelay: "0.1s" }}
|
||||||
|
>
|
||||||
{sortedProviders.map((provider) => (
|
{sortedProviders.map((provider) => (
|
||||||
<SortableProviderCard
|
<SortableProviderCard
|
||||||
key={provider.id}
|
key={provider.id}
|
||||||
provider={provider}
|
provider={provider}
|
||||||
isCurrent={provider.id === currentProviderId}
|
isCurrent={provider.id === currentProviderId}
|
||||||
appId={appId}
|
appId={appId}
|
||||||
isEditMode={isEditMode}
|
|
||||||
onSwitch={onSwitch}
|
onSwitch={onSwitch}
|
||||||
onEdit={onEdit}
|
onEdit={onEdit}
|
||||||
onDelete={onDelete}
|
onDelete={onDelete}
|
||||||
@@ -99,7 +99,6 @@ interface SortableProviderCardProps {
|
|||||||
provider: Provider;
|
provider: Provider;
|
||||||
isCurrent: boolean;
|
isCurrent: boolean;
|
||||||
appId: AppId;
|
appId: AppId;
|
||||||
isEditMode: boolean;
|
|
||||||
onSwitch: (provider: Provider) => void;
|
onSwitch: (provider: Provider) => void;
|
||||||
onEdit: (provider: Provider) => void;
|
onEdit: (provider: Provider) => void;
|
||||||
onDelete: (provider: Provider) => void;
|
onDelete: (provider: Provider) => void;
|
||||||
@@ -112,7 +111,6 @@ function SortableProviderCard({
|
|||||||
provider,
|
provider,
|
||||||
isCurrent,
|
isCurrent,
|
||||||
appId,
|
appId,
|
||||||
isEditMode,
|
|
||||||
onSwitch,
|
onSwitch,
|
||||||
onEdit,
|
onEdit,
|
||||||
onDelete,
|
onDelete,
|
||||||
@@ -140,7 +138,6 @@ function SortableProviderCard({
|
|||||||
provider={provider}
|
provider={provider}
|
||||||
isCurrent={isCurrent}
|
isCurrent={isCurrent}
|
||||||
appId={appId}
|
appId={appId}
|
||||||
isEditMode={isEditMode}
|
|
||||||
onSwitch={onSwitch}
|
onSwitch={onSwitch}
|
||||||
onEdit={onEdit}
|
onEdit={onEdit}
|
||||||
onDelete={onDelete}
|
onDelete={onDelete}
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { useState } from "react";
|
||||||
import {
|
import {
|
||||||
FormControl,
|
FormControl,
|
||||||
FormField,
|
FormField,
|
||||||
@@ -7,6 +8,17 @@ import {
|
|||||||
FormMessage,
|
FormMessage,
|
||||||
} from "@/components/ui/form";
|
} from "@/components/ui/form";
|
||||||
import { Input } from "@/components/ui/input";
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { ArrowLeft } from "lucide-react";
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogTrigger,
|
||||||
|
DialogClose,
|
||||||
|
} from "@/components/ui/dialog";
|
||||||
|
import { ProviderIcon } from "@/components/ProviderIcon";
|
||||||
|
import { IconPicker } from "@/components/IconPicker";
|
||||||
|
import { getIconMetadata } from "@/icons/extracted/metadata";
|
||||||
import type { UseFormReturn } from "react-hook-form";
|
import type { UseFormReturn } from "react-hook-form";
|
||||||
import type { ProviderFormData } from "@/lib/schemas/provider";
|
import type { ProviderFormData } from "@/lib/schemas/provider";
|
||||||
|
|
||||||
@@ -16,22 +28,115 @@ interface BasicFormFieldsProps {
|
|||||||
|
|
||||||
export function BasicFormFields({ form }: BasicFormFieldsProps) {
|
export function BasicFormFields({ form }: BasicFormFieldsProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
const [iconDialogOpen, setIconDialogOpen] = useState(false);
|
||||||
|
|
||||||
|
const currentIcon = form.watch("icon");
|
||||||
|
const currentIconColor = form.watch("iconColor");
|
||||||
|
const providerName = form.watch("name") || "Provider";
|
||||||
|
const effectiveIconColor =
|
||||||
|
currentIconColor ||
|
||||||
|
(currentIcon ? getIconMetadata(currentIcon)?.defaultColor : undefined);
|
||||||
|
|
||||||
|
const handleIconSelect = (icon: string) => {
|
||||||
|
const meta = getIconMetadata(icon);
|
||||||
|
form.setValue("icon", icon);
|
||||||
|
form.setValue("iconColor", meta?.defaultColor ?? "");
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<FormField
|
{/* 图标选择区域 - 顶部居中,可选 */}
|
||||||
control={form.control}
|
<div className="flex justify-center mb-6">
|
||||||
name="name"
|
<Dialog open={iconDialogOpen} onOpenChange={setIconDialogOpen}>
|
||||||
render={({ field }) => (
|
<DialogTrigger asChild>
|
||||||
<FormItem>
|
<button
|
||||||
<FormLabel>{t("provider.name")}</FormLabel>
|
type="button"
|
||||||
<FormControl>
|
className="w-20 h-20 p-3 rounded-xl border-2 border-gray-300 dark:border-gray-600 hover:border-primary dark:hover:border-primary transition-colors cursor-pointer bg-gray-50 dark:bg-gray-800/50 flex items-center justify-center"
|
||||||
<Input {...field} placeholder={t("provider.namePlaceholder")} />
|
title={currentIcon ? "点击更换图标" : "点击选择图标"}
|
||||||
</FormControl>
|
>
|
||||||
<FormMessage />
|
<ProviderIcon
|
||||||
</FormItem>
|
icon={currentIcon}
|
||||||
)}
|
name={providerName}
|
||||||
/>
|
color={effectiveIconColor}
|
||||||
|
size={48}
|
||||||
|
/>
|
||||||
|
</button>
|
||||||
|
</DialogTrigger>
|
||||||
|
<DialogContent
|
||||||
|
variant="fullscreen"
|
||||||
|
zIndex="top"
|
||||||
|
overlayClassName="bg-[hsl(var(--background))] backdrop-blur-0"
|
||||||
|
className="p-0 sm:rounded-none"
|
||||||
|
>
|
||||||
|
<div className="flex h-full flex-col">
|
||||||
|
<div className="flex-shrink-0 py-4 border-b border-border-default bg-muted/40">
|
||||||
|
<div className="mx-auto max-w-[56rem] px-6 flex items-center gap-4">
|
||||||
|
<DialogClose asChild>
|
||||||
|
<Button type="button" variant="outline" size="icon">
|
||||||
|
<ArrowLeft className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</DialogClose>
|
||||||
|
<p className="text-lg font-semibold leading-tight">
|
||||||
|
{t("providerIcon.selectIcon", {
|
||||||
|
defaultValue: "选择图标",
|
||||||
|
})}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex-1 overflow-y-auto">
|
||||||
|
<div className="space-y-6 mx-auto max-w-[56rem] px-6 py-6 w-full">
|
||||||
|
<IconPicker
|
||||||
|
value={currentIcon}
|
||||||
|
onValueChange={handleIconSelect}
|
||||||
|
color={effectiveIconColor}
|
||||||
|
/>
|
||||||
|
<div className="flex justify-end gap-2">
|
||||||
|
<DialogClose asChild>
|
||||||
|
<Button type="button" variant="outline">
|
||||||
|
{t("common.done", { defaultValue: "完成" })}
|
||||||
|
</Button>
|
||||||
|
</DialogClose>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* 基础信息 - 网格布局 */}
|
||||||
|
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
|
||||||
|
<FormField
|
||||||
|
control={form.control}
|
||||||
|
name="name"
|
||||||
|
render={({ field }) => (
|
||||||
|
<FormItem>
|
||||||
|
<FormLabel>{t("provider.name")}</FormLabel>
|
||||||
|
<FormControl>
|
||||||
|
<Input {...field} placeholder={t("provider.namePlaceholder")} />
|
||||||
|
</FormControl>
|
||||||
|
<FormMessage />
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<FormField
|
||||||
|
control={form.control}
|
||||||
|
name="notes"
|
||||||
|
render={({ field }) => (
|
||||||
|
<FormItem>
|
||||||
|
<FormLabel>{t("provider.notes")}</FormLabel>
|
||||||
|
<FormControl>
|
||||||
|
<Input
|
||||||
|
{...field}
|
||||||
|
placeholder={t("provider.notesPlaceholder")}
|
||||||
|
/>
|
||||||
|
</FormControl>
|
||||||
|
<FormMessage />
|
||||||
|
</FormItem>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<FormField
|
<FormField
|
||||||
control={form.control}
|
control={form.control}
|
||||||
@@ -46,20 +151,6 @@ export function BasicFormFields({ form }: BasicFormFieldsProps) {
|
|||||||
</FormItem>
|
</FormItem>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<FormField
|
|
||||||
control={form.control}
|
|
||||||
name="notes"
|
|
||||||
render={({ field }) => (
|
|
||||||
<FormItem>
|
|
||||||
<FormLabel>{t("provider.notes")}</FormLabel>
|
|
||||||
<FormControl>
|
|
||||||
<Input {...field} placeholder={t("provider.notesPlaceholder")} />
|
|
||||||
</FormControl>
|
|
||||||
<FormMessage />
|
|
||||||
</FormItem>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +1,9 @@
|
|||||||
import React from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { Save } from "lucide-react";
|
import { Save } from "lucide-react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import {
|
import { FullScreenPanel } from "@/components/common/FullScreenPanel";
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
DialogFooter,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
|
import JsonEditor from "@/components/JsonEditor";
|
||||||
|
|
||||||
interface CodexCommonConfigModalProps {
|
interface CodexCommonConfigModalProps {
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
@@ -30,47 +25,30 @@ export const CodexCommonConfigModal: React.FC<CodexCommonConfigModalProps> = ({
|
|||||||
error,
|
error,
|
||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
const [isDarkMode, setIsDarkMode] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
|
||||||
|
const observer = new MutationObserver(() => {
|
||||||
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
});
|
||||||
|
|
||||||
|
observer.observe(document.documentElement, {
|
||||||
|
attributes: true,
|
||||||
|
attributeFilter: ["class"],
|
||||||
|
});
|
||||||
|
|
||||||
|
return () => observer.disconnect();
|
||||||
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog open={isOpen} onOpenChange={(open) => !open && onClose()}>
|
<FullScreenPanel
|
||||||
<DialogContent
|
isOpen={isOpen}
|
||||||
zIndex="nested"
|
title={t("codexConfig.editCommonConfigTitle")}
|
||||||
className="max-w-2xl max-h-[90vh] flex flex-col p-0"
|
onClose={onClose}
|
||||||
>
|
footer={
|
||||||
<DialogHeader className="px-6 pt-6 pb-0">
|
<>
|
||||||
<DialogTitle>{t("codexConfig.editCommonConfigTitle")}</DialogTitle>
|
|
||||||
</DialogHeader>
|
|
||||||
|
|
||||||
<div className="flex-1 overflow-auto px-6 py-4 space-y-4">
|
|
||||||
<p className="text-sm text-gray-500 dark:text-gray-400">
|
|
||||||
{t("codexConfig.commonConfigHint")}
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<textarea
|
|
||||||
value={value}
|
|
||||||
onChange={(e) => onChange(e.target.value)}
|
|
||||||
placeholder={`# Common Codex config
|
|
||||||
|
|
||||||
# Add your common TOML configuration here`}
|
|
||||||
rows={12}
|
|
||||||
className="w-full px-3 py-2 border border-border-default dark:bg-gray-800 dark:text-gray-100 rounded-lg text-sm font-mono focus:outline-none focus:ring-2 focus:ring-blue-500/20 dark:focus:ring-blue-400/20 focus:border-border-active transition-colors resize-y"
|
|
||||||
autoComplete="off"
|
|
||||||
autoCorrect="off"
|
|
||||||
autoCapitalize="none"
|
|
||||||
spellCheck={false}
|
|
||||||
lang="en"
|
|
||||||
inputMode="text"
|
|
||||||
data-gramm="false"
|
|
||||||
data-gramm_editor="false"
|
|
||||||
data-enable-grammarly="false"
|
|
||||||
/>
|
|
||||||
|
|
||||||
{error && (
|
|
||||||
<p className="text-sm text-red-500 dark:text-red-400">{error}</p>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<DialogFooter>
|
|
||||||
<Button type="button" variant="outline" onClick={onClose}>
|
<Button type="button" variant="outline" onClick={onClose}>
|
||||||
{t("common.cancel")}
|
{t("common.cancel")}
|
||||||
</Button>
|
</Button>
|
||||||
@@ -78,8 +56,30 @@ export const CodexCommonConfigModal: React.FC<CodexCommonConfigModalProps> = ({
|
|||||||
<Save className="w-4 h-4" />
|
<Save className="w-4 h-4" />
|
||||||
{t("common.save")}
|
{t("common.save")}
|
||||||
</Button>
|
</Button>
|
||||||
</DialogFooter>
|
</>
|
||||||
</DialogContent>
|
}
|
||||||
</Dialog>
|
>
|
||||||
|
<div className="space-y-4">
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
{t("codexConfig.commonConfigHint")}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<JsonEditor
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
placeholder={`# Common Codex config
|
||||||
|
|
||||||
|
# Add your common TOML configuration here`}
|
||||||
|
darkMode={isDarkMode}
|
||||||
|
rows={16}
|
||||||
|
showValidation={false}
|
||||||
|
language="javascript"
|
||||||
|
/>
|
||||||
|
|
||||||
|
{error && (
|
||||||
|
<p className="text-sm text-red-500 dark:text-red-400">{error}</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</FullScreenPanel>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
import React from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { Wand2 } from "lucide-react";
|
import JsonEditor from "@/components/JsonEditor";
|
||||||
import { toast } from "sonner";
|
|
||||||
import { formatJSON } from "@/utils/formatters";
|
|
||||||
|
|
||||||
interface CodexAuthSectionProps {
|
interface CodexAuthSectionProps {
|
||||||
value: string;
|
value: string;
|
||||||
@@ -21,23 +19,27 @@ export const CodexAuthSection: React.FC<CodexAuthSectionProps> = ({
|
|||||||
error,
|
error,
|
||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
const [isDarkMode, setIsDarkMode] = useState(false);
|
||||||
|
|
||||||
const handleFormat = () => {
|
useEffect(() => {
|
||||||
if (!value.trim()) return;
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
|
||||||
try {
|
const observer = new MutationObserver(() => {
|
||||||
const formatted = formatJSON(value);
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
onChange(formatted);
|
});
|
||||||
toast.success(t("common.formatSuccess", { defaultValue: "格式化成功" }));
|
|
||||||
} catch (error) {
|
observer.observe(document.documentElement, {
|
||||||
const errorMessage =
|
attributes: true,
|
||||||
error instanceof Error ? error.message : String(error);
|
attributeFilter: ["class"],
|
||||||
toast.error(
|
});
|
||||||
t("common.formatError", {
|
|
||||||
defaultValue: "格式化失败:{{error}}",
|
return () => observer.disconnect();
|
||||||
error: errorMessage,
|
}, []);
|
||||||
}),
|
|
||||||
);
|
const handleChange = (newValue: string) => {
|
||||||
|
onChange(newValue);
|
||||||
|
if (onBlur) {
|
||||||
|
onBlur();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -50,39 +52,19 @@ export const CodexAuthSection: React.FC<CodexAuthSectionProps> = ({
|
|||||||
{t("codexConfig.authJson")}
|
{t("codexConfig.authJson")}
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
<textarea
|
<JsonEditor
|
||||||
id="codexAuth"
|
|
||||||
value={value}
|
value={value}
|
||||||
onChange={(e) => onChange(e.target.value)}
|
onChange={handleChange}
|
||||||
onBlur={onBlur}
|
|
||||||
placeholder={t("codexConfig.authJsonPlaceholder")}
|
placeholder={t("codexConfig.authJsonPlaceholder")}
|
||||||
|
darkMode={isDarkMode}
|
||||||
rows={6}
|
rows={6}
|
||||||
className="w-full px-3 py-2 border border-border-default dark:bg-gray-800 dark:text-gray-100 rounded-lg text-sm font-mono focus:outline-none focus:ring-2 focus:ring-blue-500/20 dark:focus:ring-blue-400/20 transition-colors resize-y min-h-[8rem]"
|
showValidation={true}
|
||||||
autoComplete="off"
|
language="json"
|
||||||
autoCorrect="off"
|
|
||||||
autoCapitalize="none"
|
|
||||||
spellCheck={false}
|
|
||||||
lang="en"
|
|
||||||
inputMode="text"
|
|
||||||
data-gramm="false"
|
|
||||||
data-gramm_editor="false"
|
|
||||||
data-enable-grammarly="false"
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div className="flex items-center justify-between">
|
{error && (
|
||||||
<button
|
<p className="text-xs text-red-500 dark:text-red-400">{error}</p>
|
||||||
type="button"
|
)}
|
||||||
onClick={handleFormat}
|
|
||||||
className="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium text-gray-700 dark:text-gray-300 hover:text-blue-600 dark:hover:text-blue-400 transition-colors"
|
|
||||||
>
|
|
||||||
<Wand2 className="w-3.5 h-3.5" />
|
|
||||||
{t("common.format", { defaultValue: "格式化" })}
|
|
||||||
</button>
|
|
||||||
|
|
||||||
{error && (
|
|
||||||
<p className="text-xs text-red-500 dark:text-red-400">{error}</p>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{!error && (
|
{!error && (
|
||||||
<p className="text-xs text-gray-500 dark:text-gray-400">
|
<p className="text-xs text-gray-500 dark:text-gray-400">
|
||||||
@@ -116,6 +98,22 @@ export const CodexConfigSection: React.FC<CodexConfigSectionProps> = ({
|
|||||||
configError,
|
configError,
|
||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
const [isDarkMode, setIsDarkMode] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
|
||||||
|
const observer = new MutationObserver(() => {
|
||||||
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
});
|
||||||
|
|
||||||
|
observer.observe(document.documentElement, {
|
||||||
|
attributes: true,
|
||||||
|
attributeFilter: ["class"],
|
||||||
|
});
|
||||||
|
|
||||||
|
return () => observer.disconnect();
|
||||||
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="space-y-2">
|
<div className="space-y-2">
|
||||||
@@ -154,22 +152,14 @@ export const CodexConfigSection: React.FC<CodexConfigSectionProps> = ({
|
|||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<textarea
|
<JsonEditor
|
||||||
id="codexConfig"
|
|
||||||
value={value}
|
value={value}
|
||||||
onChange={(e) => onChange(e.target.value)}
|
onChange={onChange}
|
||||||
placeholder=""
|
placeholder=""
|
||||||
|
darkMode={isDarkMode}
|
||||||
rows={8}
|
rows={8}
|
||||||
className="w-full px-3 py-2 border border-border-default dark:bg-gray-800 dark:text-gray-100 rounded-lg text-sm font-mono focus:outline-none focus:ring-2 focus:ring-blue-500/20 dark:focus:ring-blue-400/20 transition-colors resize-y min-h-[10rem]"
|
showValidation={false}
|
||||||
autoComplete="off"
|
language="javascript"
|
||||||
autoCorrect="off"
|
|
||||||
autoCapitalize="none"
|
|
||||||
spellCheck={false}
|
|
||||||
lang="en"
|
|
||||||
inputMode="text"
|
|
||||||
data-gramm="false"
|
|
||||||
data-gramm_editor="false"
|
|
||||||
data-enable-grammarly="false"
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{configError && (
|
{configError && (
|
||||||
|
|||||||
@@ -1,16 +1,10 @@
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import {
|
import { useEffect, useState } from "react";
|
||||||
Dialog,
|
import { FullScreenPanel } from "@/components/common/FullScreenPanel";
|
||||||
DialogContent,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
DialogFooter,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { Label } from "@/components/ui/label";
|
import { Label } from "@/components/ui/label";
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { Save, Wand2 } from "lucide-react";
|
import { Save } from "lucide-react";
|
||||||
import { toast } from "sonner";
|
import JsonEditor from "@/components/JsonEditor";
|
||||||
import { formatJSON } from "@/utils/formatters";
|
|
||||||
|
|
||||||
interface CommonConfigEditorProps {
|
interface CommonConfigEditorProps {
|
||||||
value: string;
|
value: string;
|
||||||
@@ -38,44 +32,22 @@ export function CommonConfigEditor({
|
|||||||
onModalClose,
|
onModalClose,
|
||||||
}: CommonConfigEditorProps) {
|
}: CommonConfigEditorProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
const [isDarkMode, setIsDarkMode] = useState(false);
|
||||||
|
|
||||||
const handleFormatMain = () => {
|
useEffect(() => {
|
||||||
if (!value.trim()) return;
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
|
||||||
try {
|
const observer = new MutationObserver(() => {
|
||||||
const formatted = formatJSON(value);
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
onChange(formatted);
|
});
|
||||||
toast.success(t("common.formatSuccess", { defaultValue: "格式化成功" }));
|
|
||||||
} catch (error) {
|
|
||||||
const errorMessage =
|
|
||||||
error instanceof Error ? error.message : String(error);
|
|
||||||
toast.error(
|
|
||||||
t("common.formatError", {
|
|
||||||
defaultValue: "格式化失败:{{error}}",
|
|
||||||
error: errorMessage,
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleFormatModal = () => {
|
observer.observe(document.documentElement, {
|
||||||
if (!commonConfigSnippet.trim()) return;
|
attributes: true,
|
||||||
|
attributeFilter: ["class"],
|
||||||
|
});
|
||||||
|
|
||||||
try {
|
return () => observer.disconnect();
|
||||||
const formatted = formatJSON(commonConfigSnippet);
|
}, []);
|
||||||
onCommonConfigSnippetChange(formatted);
|
|
||||||
toast.success(t("common.formatSuccess", { defaultValue: "格式化成功" }));
|
|
||||||
} catch (error) {
|
|
||||||
const errorMessage =
|
|
||||||
error instanceof Error ? error.message : String(error);
|
|
||||||
toast.error(
|
|
||||||
t("common.formatError", {
|
|
||||||
defaultValue: "格式化失败:{{error}}",
|
|
||||||
error: errorMessage,
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -115,90 +87,30 @@ export function CommonConfigEditor({
|
|||||||
{commonConfigError}
|
{commonConfigError}
|
||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
<textarea
|
<JsonEditor
|
||||||
id="settingsConfig"
|
|
||||||
value={value}
|
value={value}
|
||||||
onChange={(e) => onChange(e.target.value)}
|
onChange={onChange}
|
||||||
placeholder={`{
|
placeholder={`{
|
||||||
"env": {
|
"env": {
|
||||||
"ANTHROPIC_BASE_URL": "https://your-api-endpoint.com",
|
"ANTHROPIC_BASE_URL": "https://your-api-endpoint.com",
|
||||||
"ANTHROPIC_AUTH_TOKEN": "your-api-key-here"
|
"ANTHROPIC_AUTH_TOKEN": "your-api-key-here"
|
||||||
}
|
}
|
||||||
}`}
|
}`}
|
||||||
|
darkMode={isDarkMode}
|
||||||
rows={14}
|
rows={14}
|
||||||
className="w-full px-3 py-2 border border-border-default dark:bg-gray-800 dark:text-gray-100 rounded-lg text-sm font-mono focus:outline-none focus:ring-2 focus:ring-blue-500/20 dark:focus:ring-blue-400/20 transition-colors resize-y min-h-[16rem]"
|
showValidation={true}
|
||||||
autoComplete="off"
|
language="json"
|
||||||
autoCorrect="off"
|
|
||||||
autoCapitalize="none"
|
|
||||||
spellCheck={false}
|
|
||||||
lang="en"
|
|
||||||
inputMode="text"
|
|
||||||
data-gramm="false"
|
|
||||||
data-gramm_editor="false"
|
|
||||||
data-enable-grammarly="false"
|
|
||||||
/>
|
/>
|
||||||
<div className="flex items-center justify-between">
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
onClick={handleFormatMain}
|
|
||||||
className="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium text-gray-700 dark:text-gray-300 hover:text-blue-600 dark:hover:text-blue-400 transition-colors"
|
|
||||||
>
|
|
||||||
<Wand2 className="w-3.5 h-3.5" />
|
|
||||||
{t("common.format", { defaultValue: "格式化" })}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Dialog
|
<FullScreenPanel
|
||||||
open={isModalOpen}
|
isOpen={isModalOpen}
|
||||||
onOpenChange={(open) => !open && onModalClose()}
|
title={t("claudeConfig.editCommonConfigTitle", {
|
||||||
>
|
defaultValue: "编辑通用配置片段",
|
||||||
<DialogContent className="sm:max-w-[600px] max-h-[90vh] flex flex-col p-0">
|
})}
|
||||||
<DialogHeader className="px-6 pt-6 pb-0">
|
onClose={onModalClose}
|
||||||
<DialogTitle>
|
footer={
|
||||||
{t("claudeConfig.editCommonConfigTitle", {
|
<>
|
||||||
defaultValue: "编辑通用配置片段",
|
|
||||||
})}
|
|
||||||
</DialogTitle>
|
|
||||||
</DialogHeader>
|
|
||||||
<div className="flex-1 overflow-auto px-6 py-4 space-y-4">
|
|
||||||
<p className="text-sm text-muted-foreground">
|
|
||||||
{t("claudeConfig.commonConfigHint", {
|
|
||||||
defaultValue: "通用配置片段将合并到所有启用它的供应商配置中",
|
|
||||||
})}
|
|
||||||
</p>
|
|
||||||
<textarea
|
|
||||||
value={commonConfigSnippet}
|
|
||||||
onChange={(e) => onCommonConfigSnippetChange(e.target.value)}
|
|
||||||
rows={12}
|
|
||||||
className="w-full px-3 py-2 border border-border-default dark:bg-gray-800 dark:text-gray-100 rounded-lg text-sm font-mono focus:outline-none focus:ring-2 focus:ring-blue-500/20 dark:focus:ring-blue-400/20 transition-colors resize-y min-h-[14rem]"
|
|
||||||
autoComplete="off"
|
|
||||||
autoCorrect="off"
|
|
||||||
autoCapitalize="none"
|
|
||||||
spellCheck={false}
|
|
||||||
lang="en"
|
|
||||||
inputMode="text"
|
|
||||||
data-gramm="false"
|
|
||||||
data-gramm_editor="false"
|
|
||||||
data-enable-grammarly="false"
|
|
||||||
/>
|
|
||||||
<div className="flex items-center justify-between">
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
onClick={handleFormatModal}
|
|
||||||
className="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium text-gray-700 dark:text-gray-300 hover:text-blue-600 dark:hover:text-blue-400 transition-colors"
|
|
||||||
>
|
|
||||||
<Wand2 className="w-3.5 h-3.5" />
|
|
||||||
{t("common.format", { defaultValue: "格式化" })}
|
|
||||||
</button>
|
|
||||||
{commonConfigError && (
|
|
||||||
<p className="text-sm text-red-500 dark:text-red-400">
|
|
||||||
{commonConfigError}
|
|
||||||
</p>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<DialogFooter>
|
|
||||||
<Button type="button" variant="outline" onClick={onModalClose}>
|
<Button type="button" variant="outline" onClick={onModalClose}>
|
||||||
{t("common.cancel")}
|
{t("common.cancel")}
|
||||||
</Button>
|
</Button>
|
||||||
@@ -206,9 +118,35 @@ export function CommonConfigEditor({
|
|||||||
<Save className="w-4 h-4" />
|
<Save className="w-4 h-4" />
|
||||||
{t("common.save")}
|
{t("common.save")}
|
||||||
</Button>
|
</Button>
|
||||||
</DialogFooter>
|
</>
|
||||||
</DialogContent>
|
}
|
||||||
</Dialog>
|
>
|
||||||
|
<div className="space-y-4">
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
{t("claudeConfig.commonConfigHint", {
|
||||||
|
defaultValue: "通用配置片段将合并到所有启用它的供应商配置中",
|
||||||
|
})}
|
||||||
|
</p>
|
||||||
|
<JsonEditor
|
||||||
|
value={commonConfigSnippet}
|
||||||
|
onChange={onCommonConfigSnippetChange}
|
||||||
|
placeholder={`{
|
||||||
|
"env": {
|
||||||
|
"ANTHROPIC_BASE_URL": "https://your-api-endpoint.com"
|
||||||
|
}
|
||||||
|
}`}
|
||||||
|
darkMode={isDarkMode}
|
||||||
|
rows={16}
|
||||||
|
showValidation={true}
|
||||||
|
language="json"
|
||||||
|
/>
|
||||||
|
{commonConfigError && (
|
||||||
|
<p className="text-sm text-red-500 dark:text-red-400">
|
||||||
|
{commonConfigError}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</FullScreenPanel>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,13 +5,7 @@ import type { AppId } from "@/lib/api";
|
|||||||
import { vscodeApi } from "@/lib/api/vscode";
|
import { vscodeApi } from "@/lib/api/vscode";
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { Input } from "@/components/ui/input";
|
import { Input } from "@/components/ui/input";
|
||||||
import {
|
import { FullScreenPanel } from "@/components/common/FullScreenPanel";
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
DialogFooter,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import type { CustomEndpoint, EndpointCandidate } from "@/types";
|
import type { CustomEndpoint, EndpointCandidate } from "@/types";
|
||||||
|
|
||||||
// 端点测速超时配置(秒)
|
// 端点测速超时配置(秒)
|
||||||
@@ -431,211 +425,218 @@ const EndpointSpeedTest: React.FC<EndpointSpeedTestProps> = ({
|
|||||||
onClose();
|
onClose();
|
||||||
}, [isEditMode, providerId, entries, initialCustomUrls, appId, onClose, t]);
|
}, [isEditMode, providerId, entries, initialCustomUrls, appId, onClose, t]);
|
||||||
|
|
||||||
return (
|
if (!visible) return null;
|
||||||
<Dialog open={visible} onOpenChange={(open) => !open && onClose()}>
|
|
||||||
<DialogContent
|
const footer = (
|
||||||
zIndex="nested"
|
<div className="flex items-center gap-2">
|
||||||
className="max-w-2xl max-h-[80vh] flex flex-col p-0"
|
<Button
|
||||||
|
type="button"
|
||||||
|
variant="outline"
|
||||||
|
onClick={(event) => {
|
||||||
|
event.preventDefault();
|
||||||
|
onClose();
|
||||||
|
}}
|
||||||
|
disabled={isSaving}
|
||||||
>
|
>
|
||||||
<DialogHeader className="px-6 pt-6 pb-0">
|
{t("common.cancel")}
|
||||||
<DialogTitle>{t("endpointTest.title")}</DialogTitle>
|
</Button>
|
||||||
</DialogHeader>
|
<Button
|
||||||
|
type="button"
|
||||||
|
onClick={handleSave}
|
||||||
|
disabled={isSaving}
|
||||||
|
className="gap-2"
|
||||||
|
>
|
||||||
|
{isSaving ? (
|
||||||
|
<>
|
||||||
|
<Loader2 className="w-4 h-4 animate-spin" />
|
||||||
|
{t("common.saving")}
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<Save className="w-4 h-4" />
|
||||||
|
{t("common.save")}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
{/* Content */}
|
return (
|
||||||
<div className="flex-1 overflow-auto px-6 py-4 space-y-4">
|
<FullScreenPanel
|
||||||
{/* 测速控制栏 */}
|
isOpen={visible}
|
||||||
<div className="flex items-center justify-between">
|
title={t("endpointTest.title")}
|
||||||
<div className="text-sm text-gray-600 dark:text-gray-400">
|
onClose={onClose}
|
||||||
{entries.length} {t("endpointTest.endpoints")}
|
footer={footer}
|
||||||
</div>
|
>
|
||||||
<div className="flex items-center gap-3">
|
<div className="glass rounded-xl p-6 border border-white/10 flex flex-col gap-6">
|
||||||
<label className="flex items-center gap-1.5 text-xs text-gray-600 dark:text-gray-400">
|
{/* 测速控制栏 */}
|
||||||
<input
|
<div className="flex items-center justify-between">
|
||||||
type="checkbox"
|
<div className="text-sm text-muted-foreground">
|
||||||
checked={autoSelect}
|
{entries.length} {t("endpointTest.endpoints")}
|
||||||
onChange={(event) => setAutoSelect(event.target.checked)}
|
|
||||||
className="h-3.5 w-3.5 rounded border-border-default "
|
|
||||||
/>
|
|
||||||
{t("endpointTest.autoSelect")}
|
|
||||||
</label>
|
|
||||||
<Button
|
|
||||||
type="button"
|
|
||||||
onClick={runSpeedTest}
|
|
||||||
disabled={isTesting || !hasEndpoints}
|
|
||||||
size="sm"
|
|
||||||
className="h-7 w-20 gap-1.5 text-xs"
|
|
||||||
>
|
|
||||||
{isTesting ? (
|
|
||||||
<>
|
|
||||||
<Loader2 className="h-3.5 w-3.5 animate-spin" />
|
|
||||||
{t("endpointTest.testing")}
|
|
||||||
</>
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<Zap className="h-3.5 w-3.5" />
|
|
||||||
{t("endpointTest.testSpeed")}
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
<div className="flex items-center gap-3">
|
||||||
{/* 添加输入 */}
|
<label className="flex items-center gap-1.5 text-xs text-gray-600 dark:text-gray-400">
|
||||||
<div className="space-y-1.5">
|
<input
|
||||||
<div className="flex gap-2">
|
type="checkbox"
|
||||||
<Input
|
checked={autoSelect}
|
||||||
type="url"
|
onChange={(event) => setAutoSelect(event.target.checked)}
|
||||||
value={customUrl}
|
className="h-3.5 w-3.5 rounded border-border-default bg-background text-primary focus:ring-2 focus:ring-primary/20"
|
||||||
placeholder={t("endpointTest.addEndpointPlaceholder")}
|
|
||||||
onChange={(event) => setCustomUrl(event.target.value)}
|
|
||||||
onKeyDown={(event) => {
|
|
||||||
if (event.key === "Enter") {
|
|
||||||
event.preventDefault();
|
|
||||||
handleAddEndpoint();
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
className="flex-1"
|
|
||||||
/>
|
/>
|
||||||
<Button
|
{t("endpointTest.autoSelect")}
|
||||||
type="button"
|
</label>
|
||||||
onClick={handleAddEndpoint}
|
<Button
|
||||||
variant="outline"
|
type="button"
|
||||||
size="icon"
|
onClick={runSpeedTest}
|
||||||
>
|
disabled={isTesting || !hasEndpoints}
|
||||||
<Plus className="h-4 w-4" />
|
size="sm"
|
||||||
</Button>
|
className="h-7 w-24 gap-1.5 text-xs bg-primary text-primary-foreground hover:bg-primary/90 disabled:opacity-60"
|
||||||
</div>
|
>
|
||||||
{addError && (
|
{isTesting ? (
|
||||||
<div className="flex items-center gap-1.5 text-xs text-red-600 dark:text-red-400">
|
<>
|
||||||
<AlertCircle className="h-3 w-3" />
|
<Loader2 className="h-3.5 w-3.5 animate-spin" />
|
||||||
{addError}
|
{t("endpointTest.testing")}
|
||||||
</div>
|
</>
|
||||||
)}
|
) : (
|
||||||
|
<>
|
||||||
|
<Zap className="h-3.5 w-3.5" />
|
||||||
|
{t("endpointTest.testSpeed")}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
{/* 端点列表 */}
|
{/* 添加输入 */}
|
||||||
{hasEndpoints ? (
|
<div className="space-y-1.5">
|
||||||
<div className="space-y-2">
|
<div className="flex gap-2">
|
||||||
{sortedEntries.map((entry) => {
|
<Input
|
||||||
const isSelected = normalizedSelected === entry.url;
|
type="url"
|
||||||
const latency = entry.latency;
|
value={customUrl}
|
||||||
|
placeholder={t("endpointTest.addEndpointPlaceholder")}
|
||||||
return (
|
onChange={(event) => setCustomUrl(event.target.value)}
|
||||||
<div
|
onKeyDown={(event) => {
|
||||||
key={entry.id}
|
if (event.key === "Enter") {
|
||||||
onClick={() => handleSelect(entry.url)}
|
event.preventDefault();
|
||||||
className={`group flex cursor-pointer items-center justify-between px-3 py-2.5 rounded-lg border transition ${
|
handleAddEndpoint();
|
||||||
isSelected
|
}
|
||||||
? "border-blue-500 bg-blue-50 dark:border-blue-500 dark:bg-blue-900/20"
|
}}
|
||||||
: "border-border-default bg-white hover:border-border-default hover:bg-gray-50 dark:bg-gray-900 dark:hover:border-gray-600 dark:hover:bg-gray-800"
|
className="flex-1"
|
||||||
}`}
|
/>
|
||||||
>
|
<Button
|
||||||
<div className="flex min-w-0 flex-1 items-center gap-3">
|
type="button"
|
||||||
{/* 选择指示器 */}
|
onClick={handleAddEndpoint}
|
||||||
<div
|
variant="outline"
|
||||||
className={`h-1.5 w-1.5 flex-shrink-0 rounded-full transition ${
|
size="icon"
|
||||||
isSelected
|
>
|
||||||
? "bg-blue-500 dark:bg-blue-400"
|
<Plus className="h-4 w-4" />
|
||||||
: "bg-gray-300 dark:bg-gray-700"
|
</Button>
|
||||||
}`}
|
</div>
|
||||||
/>
|
{addError && (
|
||||||
|
|
||||||
{/* 内容 */}
|
|
||||||
<div className="min-w-0 flex-1">
|
|
||||||
<div className="truncate text-sm text-gray-900 dark:text-gray-100">
|
|
||||||
{entry.url}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* 右侧信息 */}
|
|
||||||
<div className="flex items-center gap-2">
|
|
||||||
{latency !== null ? (
|
|
||||||
<div className="text-right">
|
|
||||||
<div
|
|
||||||
className={`font-mono text-sm font-medium ${
|
|
||||||
latency < 300
|
|
||||||
? "text-green-600 dark:text-green-400"
|
|
||||||
: latency < 500
|
|
||||||
? "text-yellow-600 dark:text-yellow-400"
|
|
||||||
: latency < 800
|
|
||||||
? "text-orange-600 dark:text-orange-400"
|
|
||||||
: "text-red-600 dark:text-red-400"
|
|
||||||
}`}
|
|
||||||
>
|
|
||||||
{latency}ms
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
) : isTesting ? (
|
|
||||||
<Loader2 className="h-4 w-4 animate-spin text-gray-400" />
|
|
||||||
) : entry.error ? (
|
|
||||||
<div className="text-xs text-gray-400">
|
|
||||||
{t("endpointTest.failed")}
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<div className="text-xs text-gray-400">—</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
onClick={(e) => {
|
|
||||||
e.stopPropagation();
|
|
||||||
handleRemoveEndpoint(entry);
|
|
||||||
}}
|
|
||||||
className="opacity-0 transition hover:text-red-600 group-hover:opacity-100 dark:hover:text-red-400"
|
|
||||||
>
|
|
||||||
<X className="h-4 w-4" />
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<div className="rounded-md border border-dashed border-border-default bg-gray-50 py-8 text-center text-xs text-gray-500 dark:bg-gray-900 dark:text-gray-400">
|
|
||||||
{t("endpointTest.noEndpoints")}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{/* 错误提示 */}
|
|
||||||
{lastError && (
|
|
||||||
<div className="flex items-center gap-1.5 text-xs text-red-600 dark:text-red-400">
|
<div className="flex items-center gap-1.5 text-xs text-red-600 dark:text-red-400">
|
||||||
<AlertCircle className="h-3 w-3" />
|
<AlertCircle className="h-3 w-3" />
|
||||||
{lastError}
|
{addError}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<DialogFooter className="gap-2">
|
{/* 端点列表 */}
|
||||||
<Button
|
{hasEndpoints ? (
|
||||||
type="button"
|
<div className="space-y-2">
|
||||||
variant="outline"
|
{sortedEntries.map((entry) => {
|
||||||
onClick={onClose}
|
const isSelected = normalizedSelected === entry.url;
|
||||||
disabled={isSaving}
|
const latency = entry.latency;
|
||||||
>
|
|
||||||
{t("common.cancel")}
|
return (
|
||||||
</Button>
|
<div
|
||||||
<Button
|
key={entry.id}
|
||||||
type="button"
|
onClick={() => handleSelect(entry.url)}
|
||||||
onClick={handleSave}
|
className={`group flex cursor-pointer items-center justify-between px-3 py-2.5 rounded-lg border transition ${
|
||||||
disabled={isSaving}
|
isSelected
|
||||||
className="gap-2"
|
? "border-primary/70 bg-primary/5 shadow-sm"
|
||||||
>
|
: "border-border-default bg-background hover:bg-muted"
|
||||||
{isSaving ? (
|
}`}
|
||||||
<>
|
>
|
||||||
<Loader2 className="w-4 h-4 animate-spin" />
|
<div className="flex min-w-0 flex-1 items-center gap-3">
|
||||||
{t("common.saving")}
|
{/* 选择指示器 */}
|
||||||
</>
|
<div
|
||||||
) : (
|
className={`h-1.5 w-1.5 flex-shrink-0 rounded-full transition ${
|
||||||
<>
|
isSelected
|
||||||
<Save className="w-4 h-4" />
|
? "bg-blue-500 dark:bg-blue-400"
|
||||||
{t("common.save")}
|
: "bg-gray-300 dark:bg-gray-700"
|
||||||
</>
|
}`}
|
||||||
)}
|
/>
|
||||||
</Button>
|
|
||||||
</DialogFooter>
|
{/* 内容 */}
|
||||||
</DialogContent>
|
<div className="min-w-0 flex-1">
|
||||||
</Dialog>
|
<div className="truncate text-sm text-gray-900 dark:text-gray-100">
|
||||||
|
{entry.url}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* 右侧信息 */}
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
{latency !== null ? (
|
||||||
|
<div className="text-right">
|
||||||
|
<div
|
||||||
|
className={`font-mono text-sm font-medium ${
|
||||||
|
latency < 300
|
||||||
|
? "text-emerald-600 dark:text-emerald-400"
|
||||||
|
: latency < 500
|
||||||
|
? "text-yellow-600 dark:text-yellow-400"
|
||||||
|
: latency < 800
|
||||||
|
? "text-orange-600 dark:text-orange-400"
|
||||||
|
: "text-red-600 dark:text-red-400"
|
||||||
|
}`}
|
||||||
|
>
|
||||||
|
{latency}ms
|
||||||
|
</div>
|
||||||
|
<div className="text-[10px] text-gray-500 dark:text-gray-400">
|
||||||
|
{entry.status
|
||||||
|
? t("endpointTest.status", { code: entry.status })
|
||||||
|
: t("endpointTest.notTested")}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : isTesting ? (
|
||||||
|
<Loader2 className="h-4 w-4 animate-spin text-gray-400" />
|
||||||
|
) : entry.error ? (
|
||||||
|
<div className="text-xs text-gray-400">
|
||||||
|
{t("endpointTest.failed")}
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="text-xs text-gray-400">—</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={(event) => {
|
||||||
|
event.stopPropagation();
|
||||||
|
handleRemoveEndpoint(entry);
|
||||||
|
}}
|
||||||
|
className="opacity-0 transition hover:text-red-600 group-hover:opacity-100 dark:hover:text-red-400"
|
||||||
|
>
|
||||||
|
<X className="h-4 w-4" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="rounded-md border border-dashed border-border-default bg-muted px-4 py-8 text-center text-sm text-muted-foreground">
|
||||||
|
{t("endpointTest.empty")}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* 错误提示 */}
|
||||||
|
{lastError && (
|
||||||
|
<div className="flex items-center gap-1.5 text-xs text-red-600 dark:text-red-400">
|
||||||
|
<AlertCircle className="h-3 w-3" />
|
||||||
|
{lastError}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</FullScreenPanel>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,16 +1,9 @@
|
|||||||
import React from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { Save, Wand2 } from "lucide-react";
|
import { Save } from "lucide-react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { toast } from "sonner";
|
import { FullScreenPanel } from "@/components/common/FullScreenPanel";
|
||||||
import {
|
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
DialogFooter,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { formatJSON } from "@/utils/formatters";
|
import JsonEditor from "@/components/JsonEditor";
|
||||||
|
|
||||||
interface GeminiCommonConfigModalProps {
|
interface GeminiCommonConfigModalProps {
|
||||||
isOpen: boolean;
|
isOpen: boolean;
|
||||||
@@ -28,86 +21,32 @@ export const GeminiCommonConfigModal: React.FC<
|
|||||||
GeminiCommonConfigModalProps
|
GeminiCommonConfigModalProps
|
||||||
> = ({ isOpen, onClose, value, onChange, error }) => {
|
> = ({ isOpen, onClose, value, onChange, error }) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
const [isDarkMode, setIsDarkMode] = useState(false);
|
||||||
|
|
||||||
const handleFormat = () => {
|
useEffect(() => {
|
||||||
if (!value.trim()) return;
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
|
||||||
try {
|
const observer = new MutationObserver(() => {
|
||||||
const formatted = formatJSON(value);
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
onChange(formatted);
|
});
|
||||||
toast.success(t("common.formatSuccess", { defaultValue: "格式化成功" }));
|
|
||||||
} catch (error) {
|
observer.observe(document.documentElement, {
|
||||||
const errorMessage =
|
attributes: true,
|
||||||
error instanceof Error ? error.message : String(error);
|
attributeFilter: ["class"],
|
||||||
toast.error(
|
});
|
||||||
t("common.formatError", {
|
|
||||||
defaultValue: "格式化失败:{{error}}",
|
return () => observer.disconnect();
|
||||||
error: errorMessage,
|
}, []);
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog open={isOpen} onOpenChange={(open) => !open && onClose()}>
|
<FullScreenPanel
|
||||||
<DialogContent
|
isOpen={isOpen}
|
||||||
zIndex="nested"
|
title={t("geminiConfig.editCommonConfigTitle", {
|
||||||
className="max-w-2xl max-h-[90vh] flex flex-col p-0"
|
defaultValue: "编辑 Gemini 通用配置片段",
|
||||||
>
|
})}
|
||||||
<DialogHeader className="px-6 pt-6 pb-0">
|
onClose={onClose}
|
||||||
<DialogTitle>
|
footer={
|
||||||
{t("geminiConfig.editCommonConfigTitle", {
|
<>
|
||||||
defaultValue: "编辑 Gemini 通用配置片段",
|
|
||||||
})}
|
|
||||||
</DialogTitle>
|
|
||||||
</DialogHeader>
|
|
||||||
|
|
||||||
<div className="flex-1 overflow-auto px-6 py-4 space-y-4">
|
|
||||||
<p className="text-sm text-gray-500 dark:text-gray-400">
|
|
||||||
{t("geminiConfig.commonConfigHint", {
|
|
||||||
defaultValue:
|
|
||||||
"通用配置片段将合并到所有启用它的 Gemini 供应商配置中",
|
|
||||||
})}
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<textarea
|
|
||||||
value={value}
|
|
||||||
onChange={(e) => onChange(e.target.value)}
|
|
||||||
placeholder={`{
|
|
||||||
"timeout": 30000,
|
|
||||||
"maxRetries": 3,
|
|
||||||
"customField": "value"
|
|
||||||
}`}
|
|
||||||
rows={12}
|
|
||||||
className="w-full px-3 py-2 border border-border-default dark:bg-gray-800 dark:text-gray-100 rounded-lg text-sm font-mono focus:outline-none focus:ring-2 focus:ring-blue-500/20 dark:focus:ring-blue-400/20 focus:border-border-active transition-colors resize-y"
|
|
||||||
autoComplete="off"
|
|
||||||
autoCorrect="off"
|
|
||||||
autoCapitalize="none"
|
|
||||||
spellCheck={false}
|
|
||||||
lang="en"
|
|
||||||
inputMode="text"
|
|
||||||
data-gramm="false"
|
|
||||||
data-gramm_editor="false"
|
|
||||||
data-enable-grammarly="false"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<div className="flex items-center justify-between">
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
onClick={handleFormat}
|
|
||||||
className="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium text-gray-700 dark:text-gray-300 hover:text-blue-600 dark:hover:text-blue-400 transition-colors"
|
|
||||||
>
|
|
||||||
<Wand2 className="w-3.5 h-3.5" />
|
|
||||||
{t("common.format", { defaultValue: "格式化" })}
|
|
||||||
</button>
|
|
||||||
|
|
||||||
{error && (
|
|
||||||
<p className="text-sm text-red-500 dark:text-red-400">{error}</p>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<DialogFooter>
|
|
||||||
<Button type="button" variant="outline" onClick={onClose}>
|
<Button type="button" variant="outline" onClick={onClose}>
|
||||||
{t("common.cancel")}
|
{t("common.cancel")}
|
||||||
</Button>
|
</Button>
|
||||||
@@ -115,8 +54,35 @@ export const GeminiCommonConfigModal: React.FC<
|
|||||||
<Save className="w-4 h-4" />
|
<Save className="w-4 h-4" />
|
||||||
{t("common.save")}
|
{t("common.save")}
|
||||||
</Button>
|
</Button>
|
||||||
</DialogFooter>
|
</>
|
||||||
</DialogContent>
|
}
|
||||||
</Dialog>
|
>
|
||||||
|
<div className="space-y-4">
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
{t("geminiConfig.commonConfigHint", {
|
||||||
|
defaultValue:
|
||||||
|
"通用配置片段将合并到所有启用它的 Gemini 供应商配置中",
|
||||||
|
})}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<JsonEditor
|
||||||
|
value={value}
|
||||||
|
onChange={onChange}
|
||||||
|
placeholder={`{
|
||||||
|
"timeout": 30000,
|
||||||
|
"maxRetries": 3,
|
||||||
|
"customField": "value"
|
||||||
|
}`}
|
||||||
|
darkMode={isDarkMode}
|
||||||
|
rows={16}
|
||||||
|
showValidation={true}
|
||||||
|
language="json"
|
||||||
|
/>
|
||||||
|
|
||||||
|
{error && (
|
||||||
|
<p className="text-sm text-red-500 dark:text-red-400">{error}</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</FullScreenPanel>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
import React from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { Wand2 } from "lucide-react";
|
import JsonEditor from "@/components/JsonEditor";
|
||||||
import { toast } from "sonner";
|
|
||||||
import { formatJSON } from "@/utils/formatters";
|
|
||||||
|
|
||||||
interface GeminiEnvSectionProps {
|
interface GeminiEnvSectionProps {
|
||||||
value: string;
|
value: string;
|
||||||
@@ -21,27 +19,27 @@ export const GeminiEnvSection: React.FC<GeminiEnvSectionProps> = ({
|
|||||||
error,
|
error,
|
||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
const [isDarkMode, setIsDarkMode] = useState(false);
|
||||||
|
|
||||||
const handleFormat = () => {
|
useEffect(() => {
|
||||||
if (!value.trim()) return;
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
|
||||||
try {
|
const observer = new MutationObserver(() => {
|
||||||
// 重新格式化 .env 内容
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
const formatted = value
|
});
|
||||||
.split("\n")
|
|
||||||
.filter((line) => line.trim())
|
observer.observe(document.documentElement, {
|
||||||
.join("\n");
|
attributes: true,
|
||||||
onChange(formatted);
|
attributeFilter: ["class"],
|
||||||
toast.success(t("common.formatSuccess", { defaultValue: "格式化成功" }));
|
});
|
||||||
} catch (error) {
|
|
||||||
const errorMessage =
|
return () => observer.disconnect();
|
||||||
error instanceof Error ? error.message : String(error);
|
}, []);
|
||||||
toast.error(
|
|
||||||
t("common.formatError", {
|
const handleChange = (newValue: string) => {
|
||||||
defaultValue: "格式化失败:{{error}}",
|
onChange(newValue);
|
||||||
error: errorMessage,
|
if (onBlur) {
|
||||||
}),
|
onBlur();
|
||||||
);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -54,41 +52,21 @@ export const GeminiEnvSection: React.FC<GeminiEnvSectionProps> = ({
|
|||||||
{t("geminiConfig.envFile", { defaultValue: "环境变量 (.env)" })}
|
{t("geminiConfig.envFile", { defaultValue: "环境变量 (.env)" })}
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
<textarea
|
<JsonEditor
|
||||||
id="geminiEnv"
|
|
||||||
value={value}
|
value={value}
|
||||||
onChange={(e) => onChange(e.target.value)}
|
onChange={handleChange}
|
||||||
onBlur={onBlur}
|
|
||||||
placeholder={`GOOGLE_GEMINI_BASE_URL=https://your-api-endpoint.com/
|
placeholder={`GOOGLE_GEMINI_BASE_URL=https://your-api-endpoint.com/
|
||||||
GEMINI_API_KEY=sk-your-api-key-here
|
GEMINI_API_KEY=sk-your-api-key-here
|
||||||
GEMINI_MODEL=gemini-2.5-pro`}
|
GEMINI_MODEL=gemini-3-pro-preview`}
|
||||||
|
darkMode={isDarkMode}
|
||||||
rows={6}
|
rows={6}
|
||||||
className="w-full px-3 py-2 border border-border-default dark:bg-gray-800 dark:text-gray-100 rounded-lg text-sm font-mono focus:outline-none focus:ring-2 focus:ring-blue-500/20 dark:focus:ring-blue-400/20 transition-colors resize-y min-h-[8rem]"
|
showValidation={false}
|
||||||
autoComplete="off"
|
language="javascript"
|
||||||
autoCorrect="off"
|
|
||||||
autoCapitalize="none"
|
|
||||||
spellCheck={false}
|
|
||||||
lang="en"
|
|
||||||
inputMode="text"
|
|
||||||
data-gramm="false"
|
|
||||||
data-gramm_editor="false"
|
|
||||||
data-enable-grammarly="false"
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div className="flex items-center justify-between">
|
{error && (
|
||||||
<button
|
<p className="text-xs text-red-500 dark:text-red-400">{error}</p>
|
||||||
type="button"
|
)}
|
||||||
onClick={handleFormat}
|
|
||||||
className="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium text-gray-700 dark:text-gray-300 hover:text-blue-600 dark:hover:text-blue-400 transition-colors"
|
|
||||||
>
|
|
||||||
<Wand2 className="w-3.5 h-3.5" />
|
|
||||||
{t("common.format", { defaultValue: "格式化" })}
|
|
||||||
</button>
|
|
||||||
|
|
||||||
{error && (
|
|
||||||
<p className="text-xs text-red-500 dark:text-red-400">{error}</p>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{!error && (
|
{!error && (
|
||||||
<p className="text-xs text-gray-500 dark:text-gray-400">
|
<p className="text-xs text-gray-500 dark:text-gray-400">
|
||||||
@@ -124,25 +102,22 @@ export const GeminiConfigSection: React.FC<GeminiConfigSectionProps> = ({
|
|||||||
configError,
|
configError,
|
||||||
}) => {
|
}) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
const [isDarkMode, setIsDarkMode] = useState(false);
|
||||||
|
|
||||||
const handleFormat = () => {
|
useEffect(() => {
|
||||||
if (!value.trim()) return;
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
|
|
||||||
try {
|
const observer = new MutationObserver(() => {
|
||||||
const formatted = formatJSON(value);
|
setIsDarkMode(document.documentElement.classList.contains("dark"));
|
||||||
onChange(formatted);
|
});
|
||||||
toast.success(t("common.formatSuccess", { defaultValue: "格式化成功" }));
|
|
||||||
} catch (error) {
|
observer.observe(document.documentElement, {
|
||||||
const errorMessage =
|
attributes: true,
|
||||||
error instanceof Error ? error.message : String(error);
|
attributeFilter: ["class"],
|
||||||
toast.error(
|
});
|
||||||
t("common.formatError", {
|
|
||||||
defaultValue: "格式化失败:{{error}}",
|
return () => observer.disconnect();
|
||||||
error: errorMessage,
|
}, []);
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="space-y-2">
|
<div className="space-y-2">
|
||||||
@@ -187,43 +162,22 @@ export const GeminiConfigSection: React.FC<GeminiConfigSectionProps> = ({
|
|||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<textarea
|
<JsonEditor
|
||||||
id="geminiConfig"
|
|
||||||
value={value}
|
value={value}
|
||||||
onChange={(e) => onChange(e.target.value)}
|
onChange={onChange}
|
||||||
placeholder={`{
|
placeholder={`{
|
||||||
"timeout": 30000,
|
"timeout": 30000,
|
||||||
"maxRetries": 3
|
"maxRetries": 3
|
||||||
}`}
|
}`}
|
||||||
|
darkMode={isDarkMode}
|
||||||
rows={8}
|
rows={8}
|
||||||
className="w-full px-3 py-2 border border-border-default dark:bg-gray-800 dark:text-gray-100 rounded-lg text-sm font-mono focus:outline-none focus:ring-2 focus:ring-blue-500/20 dark:focus:ring-blue-400/20 transition-colors resize-y min-h-[10rem]"
|
showValidation={true}
|
||||||
autoComplete="off"
|
language="json"
|
||||||
autoCorrect="off"
|
|
||||||
autoCapitalize="none"
|
|
||||||
spellCheck={false}
|
|
||||||
lang="en"
|
|
||||||
inputMode="text"
|
|
||||||
data-gramm="false"
|
|
||||||
data-gramm_editor="false"
|
|
||||||
data-enable-grammarly="false"
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div className="flex items-center justify-between">
|
{configError && (
|
||||||
<button
|
<p className="text-xs text-red-500 dark:text-red-400">{configError}</p>
|
||||||
type="button"
|
)}
|
||||||
onClick={handleFormat}
|
|
||||||
className="inline-flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium text-gray-700 dark:text-gray-300 hover:text-blue-600 dark:hover:text-blue-400 transition-colors"
|
|
||||||
>
|
|
||||||
<Wand2 className="w-3.5 h-3.5" />
|
|
||||||
{t("common.format", { defaultValue: "格式化" })}
|
|
||||||
</button>
|
|
||||||
|
|
||||||
{configError && (
|
|
||||||
<p className="text-xs text-red-500 dark:text-red-400">
|
|
||||||
{configError}
|
|
||||||
</p>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{!configError && (
|
{!configError && (
|
||||||
<p className="text-xs text-gray-500 dark:text-gray-400">
|
<p className="text-xs text-gray-500 dark:text-gray-400">
|
||||||
|
|||||||
@@ -127,7 +127,7 @@ export function GeminiFormFields({
|
|||||||
id="gemini-model"
|
id="gemini-model"
|
||||||
value={model}
|
value={model}
|
||||||
onChange={(e) => onModelChange(e.target.value)}
|
onChange={(e) => onModelChange(e.target.value)}
|
||||||
placeholder="gemini-2.5-pro"
|
placeholder="gemini-3-pro-preview"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -53,7 +53,7 @@ const GEMINI_DEFAULT_CONFIG = JSON.stringify(
|
|||||||
env: {
|
env: {
|
||||||
GOOGLE_GEMINI_BASE_URL: "",
|
GOOGLE_GEMINI_BASE_URL: "",
|
||||||
GEMINI_API_KEY: "",
|
GEMINI_API_KEY: "",
|
||||||
GEMINI_MODEL: "gemini-2.5-pro",
|
GEMINI_MODEL: "gemini-3-pro-preview",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
null,
|
null,
|
||||||
@@ -78,6 +78,8 @@ interface ProviderFormProps {
|
|||||||
settingsConfig?: Record<string, unknown>;
|
settingsConfig?: Record<string, unknown>;
|
||||||
category?: ProviderCategory;
|
category?: ProviderCategory;
|
||||||
meta?: ProviderMeta;
|
meta?: ProviderMeta;
|
||||||
|
icon?: string;
|
||||||
|
iconColor?: string;
|
||||||
};
|
};
|
||||||
showButtons?: boolean;
|
showButtons?: boolean;
|
||||||
}
|
}
|
||||||
@@ -147,6 +149,8 @@ export function ProviderForm({
|
|||||||
: appId === "gemini"
|
: appId === "gemini"
|
||||||
? GEMINI_DEFAULT_CONFIG
|
? GEMINI_DEFAULT_CONFIG
|
||||||
: CLAUDE_DEFAULT_CONFIG,
|
: CLAUDE_DEFAULT_CONFIG,
|
||||||
|
icon: initialData?.icon ?? "",
|
||||||
|
iconColor: initialData?.iconColor ?? "",
|
||||||
}),
|
}),
|
||||||
[initialData, appId],
|
[initialData, appId],
|
||||||
);
|
);
|
||||||
@@ -171,18 +175,16 @@ export function ProviderForm({
|
|||||||
});
|
});
|
||||||
|
|
||||||
// 使用 Base URL hook (Claude, Codex, Gemini)
|
// 使用 Base URL hook (Claude, Codex, Gemini)
|
||||||
const { baseUrl, handleClaudeBaseUrlChange, handleGeminiBaseUrlChange } =
|
const { baseUrl, handleClaudeBaseUrlChange } = useBaseUrlState({
|
||||||
useBaseUrlState({
|
appType: appId,
|
||||||
appType: appId,
|
category,
|
||||||
category,
|
settingsConfig: form.watch("settingsConfig"),
|
||||||
settingsConfig: form.watch("settingsConfig"),
|
codexConfig: "",
|
||||||
codexConfig: "",
|
onSettingsConfigChange: (config) => form.setValue("settingsConfig", config),
|
||||||
onSettingsConfigChange: (config) =>
|
onCodexConfigChange: () => {
|
||||||
form.setValue("settingsConfig", config),
|
/* noop */
|
||||||
onCodexConfigChange: () => {
|
},
|
||||||
/* noop */
|
});
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
// 使用 Model hook(新:主模型 + Haiku/Sonnet/Opus 默认模型)
|
// 使用 Model hook(新:主模型 + Haiku/Sonnet/Opus 默认模型)
|
||||||
const {
|
const {
|
||||||
@@ -317,9 +319,13 @@ export function ProviderForm({
|
|||||||
const {
|
const {
|
||||||
geminiEnv,
|
geminiEnv,
|
||||||
geminiConfig,
|
geminiConfig,
|
||||||
|
geminiApiKey,
|
||||||
|
geminiBaseUrl,
|
||||||
geminiModel,
|
geminiModel,
|
||||||
envError,
|
envError,
|
||||||
configError: geminiConfigError,
|
configError: geminiConfigError,
|
||||||
|
handleGeminiApiKeyChange: originalHandleGeminiApiKeyChange,
|
||||||
|
handleGeminiBaseUrlChange: originalHandleGeminiBaseUrlChange,
|
||||||
handleGeminiEnvChange,
|
handleGeminiEnvChange,
|
||||||
handleGeminiConfigChange,
|
handleGeminiConfigChange,
|
||||||
resetGeminiConfig,
|
resetGeminiConfig,
|
||||||
@@ -329,6 +335,39 @@ export function ProviderForm({
|
|||||||
initialData: appId === "gemini" ? initialData : undefined,
|
initialData: appId === "gemini" ? initialData : undefined,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// 包装 Gemini handlers 以同步 settingsConfig
|
||||||
|
const handleGeminiApiKeyChange = useCallback(
|
||||||
|
(key: string) => {
|
||||||
|
originalHandleGeminiApiKeyChange(key);
|
||||||
|
// 同步更新 settingsConfig
|
||||||
|
try {
|
||||||
|
const config = JSON.parse(form.watch("settingsConfig") || "{}");
|
||||||
|
if (!config.env) config.env = {};
|
||||||
|
config.env.GEMINI_API_KEY = key.trim();
|
||||||
|
form.setValue("settingsConfig", JSON.stringify(config, null, 2));
|
||||||
|
} catch {
|
||||||
|
// ignore
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[originalHandleGeminiApiKeyChange, form],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleGeminiBaseUrlChange = useCallback(
|
||||||
|
(url: string) => {
|
||||||
|
originalHandleGeminiBaseUrlChange(url);
|
||||||
|
// 同步更新 settingsConfig
|
||||||
|
try {
|
||||||
|
const config = JSON.parse(form.watch("settingsConfig") || "{}");
|
||||||
|
if (!config.env) config.env = {};
|
||||||
|
config.env.GOOGLE_GEMINI_BASE_URL = url.trim().replace(/\/+$/, "");
|
||||||
|
form.setValue("settingsConfig", JSON.stringify(config, null, 2));
|
||||||
|
} catch {
|
||||||
|
// ignore
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[originalHandleGeminiBaseUrlChange, form],
|
||||||
|
);
|
||||||
|
|
||||||
// 使用 Gemini 通用配置 hook (仅 Gemini 模式)
|
// 使用 Gemini 通用配置 hook (仅 Gemini 模式)
|
||||||
const {
|
const {
|
||||||
useCommonConfig: useGeminiCommonConfigFlag,
|
useCommonConfig: useGeminiCommonConfigFlag,
|
||||||
@@ -616,7 +655,7 @@ export function ProviderForm({
|
|||||||
<form
|
<form
|
||||||
id="provider-form"
|
id="provider-form"
|
||||||
onSubmit={form.handleSubmit(handleSubmit)}
|
onSubmit={form.handleSubmit(handleSubmit)}
|
||||||
className="space-y-6"
|
className="space-y-6 glass rounded-xl p-6 border border-white/10"
|
||||||
>
|
>
|
||||||
{/* 预设供应商选择(仅新增模式显示) */}
|
{/* 预设供应商选择(仅新增模式显示) */}
|
||||||
{!initialData && (
|
{!initialData && (
|
||||||
@@ -704,15 +743,15 @@ export function ProviderForm({
|
|||||||
form.watch("settingsConfig"),
|
form.watch("settingsConfig"),
|
||||||
isEditMode,
|
isEditMode,
|
||||||
)}
|
)}
|
||||||
apiKey={apiKey}
|
apiKey={geminiApiKey}
|
||||||
onApiKeyChange={handleApiKeyChange}
|
onApiKeyChange={handleGeminiApiKeyChange}
|
||||||
category={category}
|
category={category}
|
||||||
shouldShowApiKeyLink={shouldShowGeminiApiKeyLink}
|
shouldShowApiKeyLink={shouldShowGeminiApiKeyLink}
|
||||||
websiteUrl={geminiWebsiteUrl}
|
websiteUrl={geminiWebsiteUrl}
|
||||||
isPartner={isGeminiPartner}
|
isPartner={isGeminiPartner}
|
||||||
partnerPromotionKey={geminiPartnerPromotionKey}
|
partnerPromotionKey={geminiPartnerPromotionKey}
|
||||||
shouldShowSpeedTest={shouldShowSpeedTest}
|
shouldShowSpeedTest={shouldShowSpeedTest}
|
||||||
baseUrl={baseUrl}
|
baseUrl={geminiBaseUrl}
|
||||||
onBaseUrlChange={handleGeminiBaseUrlChange}
|
onBaseUrlChange={handleGeminiBaseUrlChange}
|
||||||
isEndpointModalOpen={isEndpointModalOpen}
|
isEndpointModalOpen={isEndpointModalOpen}
|
||||||
onEndpointModalToggle={setIsEndpointModalOpen}
|
onEndpointModalToggle={setIsEndpointModalOpen}
|
||||||
|
|||||||
@@ -99,7 +99,7 @@ export function ProviderPresetSelector({
|
|||||||
return `${baseClass} bg-blue-500 text-white dark:bg-blue-600`;
|
return `${baseClass} bg-blue-500 text-white dark:bg-blue-600`;
|
||||||
}
|
}
|
||||||
|
|
||||||
return `${baseClass} bg-gray-100 dark:bg-gray-800 text-gray-500 dark:text-gray-400 hover:bg-gray-200 dark:hover:bg-gray-700`;
|
return `${baseClass} bg-accent text-muted-foreground hover:bg-accent/80`;
|
||||||
};
|
};
|
||||||
|
|
||||||
// 获取预设按钮的内联样式(用于自定义背景色)
|
// 获取预设按钮的内联样式(用于自定义背景色)
|
||||||
@@ -128,7 +128,7 @@ export function ProviderPresetSelector({
|
|||||||
className={`inline-flex items-center gap-2 px-4 py-2 rounded-lg text-sm font-medium transition-colors ${
|
className={`inline-flex items-center gap-2 px-4 py-2 rounded-lg text-sm font-medium transition-colors ${
|
||||||
selectedPresetId === "custom"
|
selectedPresetId === "custom"
|
||||||
? "bg-blue-500 text-white dark:bg-blue-600"
|
? "bg-blue-500 text-white dark:bg-blue-600"
|
||||||
: "bg-gray-100 dark:bg-gray-800 text-gray-500 dark:text-gray-400 hover:bg-gray-200 dark:hover:bg-gray-700"
|
: "bg-accent text-muted-foreground hover:bg-accent/80"
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
{t("providerPreset.custom")}
|
{t("providerPreset.custom")}
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ interface DirectorySettingsProps {
|
|||||||
onResetAppConfig: () => Promise<void>;
|
onResetAppConfig: () => Promise<void>;
|
||||||
claudeDir?: string;
|
claudeDir?: string;
|
||||||
codexDir?: string;
|
codexDir?: string;
|
||||||
|
geminiDir?: string;
|
||||||
onDirectoryChange: (app: AppId, value?: string) => void;
|
onDirectoryChange: (app: AppId, value?: string) => void;
|
||||||
onBrowseDirectory: (app: AppId) => Promise<void>;
|
onBrowseDirectory: (app: AppId) => Promise<void>;
|
||||||
onResetDirectory: (app: AppId) => Promise<void>;
|
onResetDirectory: (app: AppId) => Promise<void>;
|
||||||
@@ -27,6 +28,7 @@ export function DirectorySettings({
|
|||||||
onResetAppConfig,
|
onResetAppConfig,
|
||||||
claudeDir,
|
claudeDir,
|
||||||
codexDir,
|
codexDir,
|
||||||
|
geminiDir,
|
||||||
onDirectoryChange,
|
onDirectoryChange,
|
||||||
onBrowseDirectory,
|
onBrowseDirectory,
|
||||||
onResetDirectory,
|
onResetDirectory,
|
||||||
@@ -104,6 +106,17 @@ export function DirectorySettings({
|
|||||||
onBrowse={() => onBrowseDirectory("codex")}
|
onBrowse={() => onBrowseDirectory("codex")}
|
||||||
onReset={() => onResetDirectory("codex")}
|
onReset={() => onResetDirectory("codex")}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<DirectoryInput
|
||||||
|
label={t("settings.geminiConfigDir")}
|
||||||
|
description={undefined}
|
||||||
|
value={geminiDir}
|
||||||
|
resolvedValue={resolvedDirs.gemini}
|
||||||
|
placeholder={t("settings.browsePlaceholderGemini")}
|
||||||
|
onChange={(val) => onDirectoryChange("gemini", val)}
|
||||||
|
onBrowse={() => onBrowseDirectory("gemini")}
|
||||||
|
onReset={() => onResetDirectory("gemini")}
|
||||||
|
/>
|
||||||
</section>
|
</section>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -44,66 +44,73 @@ export function ImportExportSection({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<section className="space-y-4">
|
<section className="space-y-4">
|
||||||
<header className="space-y-1">
|
<header className="space-y-2">
|
||||||
<h3 className="text-sm font-medium">{t("settings.importExport")}</h3>
|
<h3 className="text-base font-semibold text-foreground">
|
||||||
<p className="text-xs text-muted-foreground">
|
{t("settings.importExport")}
|
||||||
|
</h3>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
{t("settings.importExportHint")}
|
{t("settings.importExportHint")}
|
||||||
</p>
|
</p>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<div className="space-y-3 rounded-lg border border-border-default p-4">
|
<div className="space-y-4 rounded-xl glass-card p-6 border border-white/10">
|
||||||
<Button
|
{/* Import and Export Buttons Side by Side */}
|
||||||
type="button"
|
<div className="grid grid-cols-2 gap-4 items-stretch">
|
||||||
className="w-full"
|
{/* Import Button */}
|
||||||
variant="secondary"
|
<div className="relative">
|
||||||
onClick={onExport}
|
|
||||||
>
|
|
||||||
<Save className="mr-2 h-4 w-4" />
|
|
||||||
{t("settings.exportConfig")}
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
<div className="space-y-2">
|
|
||||||
<div className="flex flex-wrap items-center gap-2">
|
|
||||||
<Button
|
<Button
|
||||||
type="button"
|
type="button"
|
||||||
variant="outline"
|
className={`w-full h-auto py-3 px-4 bg-blue-500 hover:bg-blue-600 dark:bg-blue-600 dark:hover:bg-blue-700 text-white ${selectedFile && !isImporting ? "flex-col items-start" : "items-center"}`}
|
||||||
className="flex-1 min-w-[180px]"
|
onClick={!selectedFile ? onSelectFile : onImport}
|
||||||
onClick={onSelectFile}
|
disabled={isImporting}
|
||||||
>
|
>
|
||||||
<FolderOpen className="mr-2 h-4 w-4" />
|
<div className="flex items-center gap-2 w-full justify-center">
|
||||||
{t("settings.selectConfigFile")}
|
{isImporting ? (
|
||||||
</Button>
|
<Loader2 className="h-4 w-4 animate-spin flex-shrink-0" />
|
||||||
<Button
|
) : selectedFile ? (
|
||||||
type="button"
|
<CheckCircle2 className="h-4 w-4 flex-shrink-0" />
|
||||||
disabled={!selectedFile || isImporting}
|
) : (
|
||||||
onClick={onImport}
|
<FolderOpen className="h-4 w-4 flex-shrink-0" />
|
||||||
>
|
)}
|
||||||
{isImporting ? (
|
<span className="font-medium">
|
||||||
<span className="inline-flex items-center gap-2">
|
{isImporting
|
||||||
<Loader2 className="h-4 w-4 animate-spin" />
|
? t("settings.importing")
|
||||||
{t("settings.importing")}
|
: selectedFile
|
||||||
|
? t("settings.import")
|
||||||
|
: t("settings.selectConfigFile")}
|
||||||
</span>
|
</span>
|
||||||
) : (
|
</div>
|
||||||
t("settings.import")
|
{selectedFile && !isImporting && (
|
||||||
|
<div className="mt-2 w-full text-left">
|
||||||
|
<p className="text-xs font-mono text-white/80 truncate">
|
||||||
|
📄 {selectedFileName}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
)}
|
)}
|
||||||
</Button>
|
</Button>
|
||||||
{selectedFile ? (
|
{selectedFile && (
|
||||||
<Button type="button" variant="ghost" onClick={onClear}>
|
<button
|
||||||
<XCircle className="mr-2 h-4 w-4" />
|
type="button"
|
||||||
{t("common.clear")}
|
onClick={onClear}
|
||||||
</Button>
|
className="absolute -top-2 -right-2 h-6 w-6 rounded-full bg-red-500 hover:bg-red-600 text-white flex items-center justify-center shadow-lg transition-colors z-10"
|
||||||
) : null}
|
aria-label="Clear selection"
|
||||||
|
>
|
||||||
|
<XCircle className="h-4 w-4" />
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{selectedFile ? (
|
{/* Export Button */}
|
||||||
<p className="truncate rounded-md bg-muted/40 px-3 py-2 text-xs font-mono text-muted-foreground">
|
<div>
|
||||||
{selectedFileName}
|
<Button
|
||||||
</p>
|
type="button"
|
||||||
) : (
|
className="w-full h-full py-3 px-4 bg-blue-500 hover:bg-blue-600 dark:bg-blue-600 dark:hover:bg-blue-700 text-white items-center"
|
||||||
<p className="text-xs text-muted-foreground">
|
onClick={onExport}
|
||||||
{t("settings.noFileSelected")}
|
>
|
||||||
</p>
|
<Save className="mr-2 h-4 w-4" />
|
||||||
)}
|
{t("settings.exportConfig")}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<ImportStatusMessage
|
<ImportStatusMessage
|
||||||
@@ -134,15 +141,19 @@ function ImportStatusMessage({
|
|||||||
}
|
}
|
||||||
|
|
||||||
const baseClass =
|
const baseClass =
|
||||||
"flex items-start gap-2 rounded-md border px-3 py-2 text-xs leading-relaxed";
|
"flex items-start gap-3 rounded-xl border p-4 text-sm leading-relaxed backdrop-blur-sm";
|
||||||
|
|
||||||
if (status === "importing") {
|
if (status === "importing") {
|
||||||
return (
|
return (
|
||||||
<div className={`${baseClass} border-border-default bg-muted/40`}>
|
<div
|
||||||
<Loader2 className="mt-0.5 h-4 w-4 animate-spin text-muted-foreground" />
|
className={`${baseClass} border-blue-500/30 bg-blue-500/10 text-blue-600 dark:text-blue-400`}
|
||||||
|
>
|
||||||
|
<Loader2 className="mt-0.5 h-5 w-5 flex-shrink-0 animate-spin" />
|
||||||
<div>
|
<div>
|
||||||
<p className="font-medium">{t("settings.importing")}</p>
|
<p className="font-semibold">{t("settings.importing")}</p>
|
||||||
<p className="text-muted-foreground">{t("common.loading")}</p>
|
<p className="text-blue-600/80 dark:text-blue-400/80">
|
||||||
|
{t("common.loading")}
|
||||||
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
@@ -151,17 +162,19 @@ function ImportStatusMessage({
|
|||||||
if (status === "success") {
|
if (status === "success") {
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={`${baseClass} border-green-200 bg-green-100/70 text-green-700`}
|
className={`${baseClass} border-green-500/30 bg-green-500/10 text-green-700 dark:text-green-400`}
|
||||||
>
|
>
|
||||||
<CheckCircle2 className="mt-0.5 h-4 w-4" />
|
<CheckCircle2 className="mt-0.5 h-5 w-5 flex-shrink-0" />
|
||||||
<div className="space-y-1">
|
<div className="space-y-1.5">
|
||||||
<p className="font-medium">{t("settings.importSuccess")}</p>
|
<p className="font-semibold">{t("settings.importSuccess")}</p>
|
||||||
{backupId ? (
|
{backupId ? (
|
||||||
<p className="text-xs">
|
<p className="text-xs text-green-600/80 dark:text-green-400/80">
|
||||||
{t("settings.backupId")}: {backupId}
|
{t("settings.backupId")}: {backupId}
|
||||||
</p>
|
</p>
|
||||||
) : null}
|
) : null}
|
||||||
<p>{t("settings.autoReload")}</p>
|
<p className="text-green-600/80 dark:text-green-400/80">
|
||||||
|
{t("settings.autoReload")}
|
||||||
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
@@ -170,12 +183,14 @@ function ImportStatusMessage({
|
|||||||
if (status === "partial-success") {
|
if (status === "partial-success") {
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={`${baseClass} border-yellow-200 bg-yellow-100/70 text-yellow-700`}
|
className={`${baseClass} border-yellow-500/30 bg-yellow-500/10 text-yellow-700 dark:text-yellow-400`}
|
||||||
>
|
>
|
||||||
<AlertCircle className="mt-0.5 h-4 w-4" />
|
<AlertCircle className="mt-0.5 h-5 w-5 flex-shrink-0" />
|
||||||
<div className="space-y-1">
|
<div className="space-y-1.5">
|
||||||
<p className="font-medium">{t("settings.importPartialSuccess")}</p>
|
<p className="font-semibold">{t("settings.importPartialSuccess")}</p>
|
||||||
<p>{t("settings.importPartialHint")}</p>
|
<p className="text-yellow-600/80 dark:text-yellow-400/80">
|
||||||
|
{t("settings.importPartialHint")}
|
||||||
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
@@ -184,11 +199,13 @@ function ImportStatusMessage({
|
|||||||
const message = errorMessage || t("settings.importFailed");
|
const message = errorMessage || t("settings.importFailed");
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={`${baseClass} border-red-200 bg-red-100/70 text-red-600`}>
|
<div
|
||||||
<AlertCircle className="mt-0.5 h-4 w-4" />
|
className={`${baseClass} border-red-500/30 bg-red-500/10 text-red-600 dark:text-red-400`}
|
||||||
<div className="space-y-1">
|
>
|
||||||
<p className="font-medium">{t("settings.importFailed")}</p>
|
<AlertCircle className="mt-0.5 h-5 w-5 flex-shrink-0" />
|
||||||
<p>{message}</p>
|
<div className="space-y-1.5">
|
||||||
|
<p className="font-semibold">{t("settings.importFailed")}</p>
|
||||||
|
<p className="text-red-600/80 dark:text-red-400/80">{message}</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,294 +0,0 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from "react";
|
|
||||||
import { Loader2, Save } from "lucide-react";
|
|
||||||
import { toast } from "sonner";
|
|
||||||
import {
|
|
||||||
Dialog,
|
|
||||||
DialogContent,
|
|
||||||
DialogFooter,
|
|
||||||
DialogHeader,
|
|
||||||
DialogTitle,
|
|
||||||
} from "@/components/ui/dialog";
|
|
||||||
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs";
|
|
||||||
import { Button } from "@/components/ui/button";
|
|
||||||
import { settingsApi } from "@/lib/api";
|
|
||||||
import { LanguageSettings } from "@/components/settings/LanguageSettings";
|
|
||||||
import { ThemeSettings } from "@/components/settings/ThemeSettings";
|
|
||||||
import { WindowSettings } from "@/components/settings/WindowSettings";
|
|
||||||
import { DirectorySettings } from "@/components/settings/DirectorySettings";
|
|
||||||
import { ImportExportSection } from "@/components/settings/ImportExportSection";
|
|
||||||
import { AboutSection } from "@/components/settings/AboutSection";
|
|
||||||
import { useSettings } from "@/hooks/useSettings";
|
|
||||||
import { useImportExport } from "@/hooks/useImportExport";
|
|
||||||
import { useTranslation } from "react-i18next";
|
|
||||||
|
|
||||||
interface SettingsDialogProps {
|
|
||||||
open: boolean;
|
|
||||||
onOpenChange: (open: boolean) => void;
|
|
||||||
onImportSuccess?: () => void | Promise<void>;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function SettingsDialog({
|
|
||||||
open,
|
|
||||||
onOpenChange,
|
|
||||||
onImportSuccess,
|
|
||||||
}: SettingsDialogProps) {
|
|
||||||
const { t } = useTranslation();
|
|
||||||
const {
|
|
||||||
settings,
|
|
||||||
isLoading,
|
|
||||||
isSaving,
|
|
||||||
isPortable,
|
|
||||||
appConfigDir,
|
|
||||||
resolvedDirs,
|
|
||||||
updateSettings,
|
|
||||||
updateDirectory,
|
|
||||||
updateAppConfigDir,
|
|
||||||
browseDirectory,
|
|
||||||
browseAppConfigDir,
|
|
||||||
resetDirectory,
|
|
||||||
resetAppConfigDir,
|
|
||||||
saveSettings,
|
|
||||||
resetSettings,
|
|
||||||
requiresRestart,
|
|
||||||
acknowledgeRestart,
|
|
||||||
} = useSettings();
|
|
||||||
|
|
||||||
const {
|
|
||||||
selectedFile,
|
|
||||||
status: importStatus,
|
|
||||||
errorMessage,
|
|
||||||
backupId,
|
|
||||||
isImporting,
|
|
||||||
selectImportFile,
|
|
||||||
importConfig,
|
|
||||||
exportConfig,
|
|
||||||
clearSelection,
|
|
||||||
resetStatus,
|
|
||||||
} = useImportExport({ onImportSuccess });
|
|
||||||
|
|
||||||
const [activeTab, setActiveTab] = useState<string>("general");
|
|
||||||
const [showRestartPrompt, setShowRestartPrompt] = useState(false);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (open) {
|
|
||||||
setActiveTab("general");
|
|
||||||
resetStatus();
|
|
||||||
}
|
|
||||||
}, [open, resetStatus]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (requiresRestart) {
|
|
||||||
setShowRestartPrompt(true);
|
|
||||||
}
|
|
||||||
}, [requiresRestart]);
|
|
||||||
|
|
||||||
const closeDialog = useCallback(() => {
|
|
||||||
// 取消/直接关闭:恢复到初始设置(包括语言回滚)
|
|
||||||
resetSettings();
|
|
||||||
acknowledgeRestart();
|
|
||||||
clearSelection();
|
|
||||||
resetStatus();
|
|
||||||
onOpenChange(false);
|
|
||||||
}, [
|
|
||||||
acknowledgeRestart,
|
|
||||||
clearSelection,
|
|
||||||
onOpenChange,
|
|
||||||
resetSettings,
|
|
||||||
resetStatus,
|
|
||||||
]);
|
|
||||||
|
|
||||||
const closeAfterSave = useCallback(() => {
|
|
||||||
// 保存成功后关闭:不再重置语言,避免需要“保存两次”才生效
|
|
||||||
acknowledgeRestart();
|
|
||||||
clearSelection();
|
|
||||||
resetStatus();
|
|
||||||
onOpenChange(false);
|
|
||||||
}, [acknowledgeRestart, clearSelection, onOpenChange, resetStatus]);
|
|
||||||
|
|
||||||
const handleDialogChange = useCallback(
|
|
||||||
(nextOpen: boolean) => {
|
|
||||||
if (!nextOpen) {
|
|
||||||
closeDialog();
|
|
||||||
} else {
|
|
||||||
onOpenChange(true);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[closeDialog, onOpenChange],
|
|
||||||
);
|
|
||||||
|
|
||||||
const handleCancel = useCallback(() => {
|
|
||||||
closeDialog();
|
|
||||||
}, [closeDialog]);
|
|
||||||
|
|
||||||
const handleSave = useCallback(async () => {
|
|
||||||
try {
|
|
||||||
const result = await saveSettings();
|
|
||||||
if (!result) return;
|
|
||||||
if (result.requiresRestart) {
|
|
||||||
setShowRestartPrompt(true);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
closeAfterSave();
|
|
||||||
} catch (error) {
|
|
||||||
console.error("[SettingsDialog] Failed to save settings", error);
|
|
||||||
}
|
|
||||||
}, [closeDialog, saveSettings]);
|
|
||||||
|
|
||||||
const handleRestartLater = useCallback(() => {
|
|
||||||
setShowRestartPrompt(false);
|
|
||||||
closeAfterSave();
|
|
||||||
}, [closeAfterSave]);
|
|
||||||
|
|
||||||
const handleRestartNow = useCallback(async () => {
|
|
||||||
setShowRestartPrompt(false);
|
|
||||||
if (import.meta.env.DEV) {
|
|
||||||
toast.success(t("settings.devModeRestartHint"));
|
|
||||||
closeAfterSave();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
await settingsApi.restart();
|
|
||||||
} catch (error) {
|
|
||||||
console.error("[SettingsDialog] Failed to restart app", error);
|
|
||||||
toast.error(t("settings.restartFailed"));
|
|
||||||
} finally {
|
|
||||||
closeAfterSave();
|
|
||||||
}
|
|
||||||
}, [closeAfterSave, t]);
|
|
||||||
|
|
||||||
const isBusy = useMemo(() => isLoading && !settings, [isLoading, settings]);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Dialog open={open} onOpenChange={handleDialogChange}>
|
|
||||||
<DialogContent className="max-w-3xl max-h-[90vh] flex flex-col">
|
|
||||||
<DialogHeader>
|
|
||||||
<DialogTitle>{t("settings.title")}</DialogTitle>
|
|
||||||
</DialogHeader>
|
|
||||||
|
|
||||||
{isBusy ? (
|
|
||||||
<div className="flex min-h-[320px] items-center justify-center">
|
|
||||||
<Loader2 className="h-6 w-6 animate-spin text-muted-foreground" />
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<div className="flex-1 overflow-y-auto px-6 py-4">
|
|
||||||
<Tabs
|
|
||||||
value={activeTab}
|
|
||||||
onValueChange={setActiveTab}
|
|
||||||
className="flex flex-col h-full"
|
|
||||||
>
|
|
||||||
<TabsList className="grid w-full grid-cols-3">
|
|
||||||
<TabsTrigger value="general">
|
|
||||||
{t("settings.tabGeneral")}
|
|
||||||
</TabsTrigger>
|
|
||||||
<TabsTrigger value="advanced">
|
|
||||||
{t("settings.tabAdvanced")}
|
|
||||||
</TabsTrigger>
|
|
||||||
<TabsTrigger value="about">{t("common.about")}</TabsTrigger>
|
|
||||||
</TabsList>
|
|
||||||
|
|
||||||
<TabsContent
|
|
||||||
value="general"
|
|
||||||
className="space-y-6 mt-6 min-h-[400px]"
|
|
||||||
>
|
|
||||||
{settings ? (
|
|
||||||
<>
|
|
||||||
<LanguageSettings
|
|
||||||
value={settings.language}
|
|
||||||
onChange={(lang) => updateSettings({ language: lang })}
|
|
||||||
/>
|
|
||||||
<ThemeSettings />
|
|
||||||
<WindowSettings
|
|
||||||
settings={settings}
|
|
||||||
onChange={updateSettings}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
) : null}
|
|
||||||
</TabsContent>
|
|
||||||
|
|
||||||
<TabsContent
|
|
||||||
value="advanced"
|
|
||||||
className="space-y-6 mt-6 min-h-[400px]"
|
|
||||||
>
|
|
||||||
{settings ? (
|
|
||||||
<>
|
|
||||||
<DirectorySettings
|
|
||||||
appConfigDir={appConfigDir}
|
|
||||||
resolvedDirs={resolvedDirs}
|
|
||||||
onAppConfigChange={updateAppConfigDir}
|
|
||||||
onBrowseAppConfig={browseAppConfigDir}
|
|
||||||
onResetAppConfig={resetAppConfigDir}
|
|
||||||
claudeDir={settings.claudeConfigDir}
|
|
||||||
codexDir={settings.codexConfigDir}
|
|
||||||
onDirectoryChange={updateDirectory}
|
|
||||||
onBrowseDirectory={browseDirectory}
|
|
||||||
onResetDirectory={resetDirectory}
|
|
||||||
/>
|
|
||||||
<ImportExportSection
|
|
||||||
status={importStatus}
|
|
||||||
selectedFile={selectedFile}
|
|
||||||
errorMessage={errorMessage}
|
|
||||||
backupId={backupId}
|
|
||||||
isImporting={isImporting}
|
|
||||||
onSelectFile={selectImportFile}
|
|
||||||
onImport={importConfig}
|
|
||||||
onExport={exportConfig}
|
|
||||||
onClear={clearSelection}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
) : null}
|
|
||||||
</TabsContent>
|
|
||||||
|
|
||||||
<TabsContent value="about" className="mt-6 min-h-[400px]">
|
|
||||||
<AboutSection isPortable={isPortable} />
|
|
||||||
</TabsContent>
|
|
||||||
</Tabs>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<DialogFooter>
|
|
||||||
<Button variant="outline" onClick={handleCancel}>
|
|
||||||
{t("common.cancel")}
|
|
||||||
</Button>
|
|
||||||
<Button onClick={handleSave} disabled={isSaving || isBusy}>
|
|
||||||
{isSaving ? (
|
|
||||||
<span className="inline-flex items-center gap-2">
|
|
||||||
<Loader2 className="h-4 w-4 animate-spin" />
|
|
||||||
{t("settings.saving")}
|
|
||||||
</span>
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<Save className="mr-2 h-4 w-4" />
|
|
||||||
{t("common.save")}
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Button>
|
|
||||||
</DialogFooter>
|
|
||||||
</DialogContent>
|
|
||||||
|
|
||||||
<Dialog
|
|
||||||
open={showRestartPrompt}
|
|
||||||
onOpenChange={(open) => !open && handleRestartLater()}
|
|
||||||
>
|
|
||||||
<DialogContent zIndex="alert" className="max-w-md">
|
|
||||||
<DialogHeader>
|
|
||||||
<DialogTitle>{t("settings.restartRequired")}</DialogTitle>
|
|
||||||
</DialogHeader>
|
|
||||||
<div className="px-6">
|
|
||||||
<p className="text-sm text-muted-foreground">
|
|
||||||
{t("settings.restartRequiredMessage")}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<DialogFooter>
|
|
||||||
<Button variant="outline" onClick={handleRestartLater}>
|
|
||||||
{t("settings.restartLater")}
|
|
||||||
</Button>
|
|
||||||
<Button onClick={handleRestartNow}>
|
|
||||||
{t("settings.restartNow")}
|
|
||||||
</Button>
|
|
||||||
</DialogFooter>
|
|
||||||
</DialogContent>
|
|
||||||
</Dialog>
|
|
||||||
</Dialog>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
284
src/components/settings/SettingsPage.tsx
Normal file
284
src/components/settings/SettingsPage.tsx
Normal file
@@ -0,0 +1,284 @@
|
|||||||
|
import { useCallback, useEffect, useMemo, useState } from "react";
|
||||||
|
import { Loader2, Save } from "lucide-react";
|
||||||
|
import { toast } from "sonner";
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
} from "@/components/ui/dialog";
|
||||||
|
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { settingsApi } from "@/lib/api";
|
||||||
|
import { LanguageSettings } from "@/components/settings/LanguageSettings";
|
||||||
|
import { ThemeSettings } from "@/components/settings/ThemeSettings";
|
||||||
|
import { WindowSettings } from "@/components/settings/WindowSettings";
|
||||||
|
import { DirectorySettings } from "@/components/settings/DirectorySettings";
|
||||||
|
import { ImportExportSection } from "@/components/settings/ImportExportSection";
|
||||||
|
import { AboutSection } from "@/components/settings/AboutSection";
|
||||||
|
import { useSettings } from "@/hooks/useSettings";
|
||||||
|
import { useImportExport } from "@/hooks/useImportExport";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import type { SettingsFormState } from "@/hooks/useSettings";
|
||||||
|
|
||||||
|
interface SettingsDialogProps {
|
||||||
|
open: boolean;
|
||||||
|
onOpenChange: (open: boolean) => void;
|
||||||
|
onImportSuccess?: () => void | Promise<void>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function SettingsPage({
|
||||||
|
open,
|
||||||
|
onOpenChange,
|
||||||
|
onImportSuccess,
|
||||||
|
}: SettingsDialogProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const {
|
||||||
|
settings,
|
||||||
|
isLoading,
|
||||||
|
isSaving,
|
||||||
|
isPortable,
|
||||||
|
appConfigDir,
|
||||||
|
resolvedDirs,
|
||||||
|
updateSettings,
|
||||||
|
updateDirectory,
|
||||||
|
updateAppConfigDir,
|
||||||
|
browseDirectory,
|
||||||
|
browseAppConfigDir,
|
||||||
|
resetDirectory,
|
||||||
|
resetAppConfigDir,
|
||||||
|
saveSettings,
|
||||||
|
autoSaveSettings,
|
||||||
|
requiresRestart,
|
||||||
|
acknowledgeRestart,
|
||||||
|
} = useSettings();
|
||||||
|
|
||||||
|
const {
|
||||||
|
selectedFile,
|
||||||
|
status: importStatus,
|
||||||
|
errorMessage,
|
||||||
|
backupId,
|
||||||
|
isImporting,
|
||||||
|
selectImportFile,
|
||||||
|
importConfig,
|
||||||
|
exportConfig,
|
||||||
|
clearSelection,
|
||||||
|
resetStatus,
|
||||||
|
} = useImportExport({ onImportSuccess });
|
||||||
|
|
||||||
|
const [activeTab, setActiveTab] = useState<string>("general");
|
||||||
|
const [showRestartPrompt, setShowRestartPrompt] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (open) {
|
||||||
|
setActiveTab("general");
|
||||||
|
resetStatus();
|
||||||
|
}
|
||||||
|
}, [open, resetStatus]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (requiresRestart) {
|
||||||
|
setShowRestartPrompt(true);
|
||||||
|
}
|
||||||
|
}, [requiresRestart]);
|
||||||
|
|
||||||
|
const closeAfterSave = useCallback(() => {
|
||||||
|
// 保存成功后关闭:不再重置语言,避免需要“保存两次”才生效
|
||||||
|
acknowledgeRestart();
|
||||||
|
clearSelection();
|
||||||
|
resetStatus();
|
||||||
|
onOpenChange(false);
|
||||||
|
}, [acknowledgeRestart, clearSelection, onOpenChange, resetStatus]);
|
||||||
|
|
||||||
|
const handleSave = useCallback(async () => {
|
||||||
|
try {
|
||||||
|
const result = await saveSettings(undefined, { silent: false });
|
||||||
|
if (!result) return;
|
||||||
|
if (result.requiresRestart) {
|
||||||
|
setShowRestartPrompt(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
closeAfterSave();
|
||||||
|
} catch (error) {
|
||||||
|
console.error("[SettingsPage] Failed to save settings", error);
|
||||||
|
}
|
||||||
|
}, [closeAfterSave, saveSettings]);
|
||||||
|
|
||||||
|
const handleRestartLater = useCallback(() => {
|
||||||
|
setShowRestartPrompt(false);
|
||||||
|
closeAfterSave();
|
||||||
|
}, [closeAfterSave]);
|
||||||
|
|
||||||
|
const handleRestartNow = useCallback(async () => {
|
||||||
|
setShowRestartPrompt(false);
|
||||||
|
if (import.meta.env.DEV) {
|
||||||
|
toast.success(t("settings.devModeRestartHint"));
|
||||||
|
closeAfterSave();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await settingsApi.restart();
|
||||||
|
} catch (error) {
|
||||||
|
console.error("[SettingsPage] Failed to restart app", error);
|
||||||
|
toast.error(t("settings.restartFailed"));
|
||||||
|
} finally {
|
||||||
|
closeAfterSave();
|
||||||
|
}
|
||||||
|
}, [closeAfterSave, t]);
|
||||||
|
|
||||||
|
// 通用设置即时保存(无需手动点击)
|
||||||
|
// 使用 autoSaveSettings 避免误触发系统 API(开机自启、Claude 插件等)
|
||||||
|
const handleAutoSave = useCallback(
|
||||||
|
async (updates: Partial<SettingsFormState>) => {
|
||||||
|
if (!settings) return;
|
||||||
|
updateSettings(updates);
|
||||||
|
try {
|
||||||
|
await autoSaveSettings(updates);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("[SettingsPage] Failed to autosave settings", error);
|
||||||
|
toast.error(
|
||||||
|
t("settings.saveFailedGeneric", {
|
||||||
|
defaultValue: "保存失败,请重试",
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[autoSaveSettings, settings, t, updateSettings],
|
||||||
|
);
|
||||||
|
|
||||||
|
const isBusy = useMemo(() => isLoading && !settings, [isLoading, settings]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="mx-auto max-w-[56rem] flex flex-col h-[calc(100vh-8rem)] px-6">
|
||||||
|
{isBusy ? (
|
||||||
|
<div className="flex flex-1 items-center justify-center">
|
||||||
|
<Loader2 className="h-8 w-8 animate-spin text-muted-foreground" />
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<Tabs
|
||||||
|
value={activeTab}
|
||||||
|
onValueChange={setActiveTab}
|
||||||
|
className="flex flex-col h-full"
|
||||||
|
>
|
||||||
|
<TabsList className="grid w-full grid-cols-3 mb-6 glass rounded-xl">
|
||||||
|
<TabsTrigger value="general">
|
||||||
|
{t("settings.tabGeneral")}
|
||||||
|
</TabsTrigger>
|
||||||
|
<TabsTrigger value="advanced">
|
||||||
|
{t("settings.tabAdvanced")}
|
||||||
|
</TabsTrigger>
|
||||||
|
<TabsTrigger value="about">{t("common.about")}</TabsTrigger>
|
||||||
|
</TabsList>
|
||||||
|
|
||||||
|
<div className="flex-1 overflow-y-auto pr-2">
|
||||||
|
<TabsContent value="general" className="space-y-6 mt-0">
|
||||||
|
{settings ? (
|
||||||
|
<>
|
||||||
|
<LanguageSettings
|
||||||
|
value={settings.language}
|
||||||
|
onChange={(lang) => handleAutoSave({ language: lang })}
|
||||||
|
/>
|
||||||
|
<ThemeSettings />
|
||||||
|
<WindowSettings
|
||||||
|
settings={settings}
|
||||||
|
onChange={handleAutoSave}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
) : null}
|
||||||
|
</TabsContent>
|
||||||
|
|
||||||
|
<TabsContent value="advanced" className="space-y-6 mt-0 pb-6">
|
||||||
|
{settings ? (
|
||||||
|
<>
|
||||||
|
<DirectorySettings
|
||||||
|
appConfigDir={appConfigDir}
|
||||||
|
resolvedDirs={resolvedDirs}
|
||||||
|
onAppConfigChange={updateAppConfigDir}
|
||||||
|
onBrowseAppConfig={browseAppConfigDir}
|
||||||
|
onResetAppConfig={resetAppConfigDir}
|
||||||
|
claudeDir={settings.claudeConfigDir}
|
||||||
|
codexDir={settings.codexConfigDir}
|
||||||
|
geminiDir={settings.geminiConfigDir}
|
||||||
|
onDirectoryChange={updateDirectory}
|
||||||
|
onBrowseDirectory={browseDirectory}
|
||||||
|
onResetDirectory={resetDirectory}
|
||||||
|
/>
|
||||||
|
<ImportExportSection
|
||||||
|
status={importStatus}
|
||||||
|
selectedFile={selectedFile}
|
||||||
|
errorMessage={errorMessage}
|
||||||
|
backupId={backupId}
|
||||||
|
isImporting={isImporting}
|
||||||
|
onSelectFile={selectImportFile}
|
||||||
|
onImport={importConfig}
|
||||||
|
onExport={exportConfig}
|
||||||
|
onClear={clearSelection}
|
||||||
|
/>
|
||||||
|
<div className="pt-6 border-t border-gray-200 dark:border-white/10">
|
||||||
|
<Button
|
||||||
|
onClick={handleSave}
|
||||||
|
className="w-full"
|
||||||
|
disabled={isSaving}
|
||||||
|
>
|
||||||
|
{isSaving ? (
|
||||||
|
<span className="inline-flex items-center gap-2">
|
||||||
|
<Loader2 className="h-4 w-4 animate-spin" />
|
||||||
|
{t("settings.saving")}
|
||||||
|
</span>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<Save className="mr-2 h-4 w-4" />
|
||||||
|
{t("common.save")}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
) : null}
|
||||||
|
</TabsContent>
|
||||||
|
|
||||||
|
<TabsContent value="about" className="mt-0">
|
||||||
|
<AboutSection isPortable={isPortable} />
|
||||||
|
</TabsContent>
|
||||||
|
</div>
|
||||||
|
</Tabs>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<Dialog
|
||||||
|
open={showRestartPrompt}
|
||||||
|
onOpenChange={(open) => !open && handleRestartLater()}
|
||||||
|
>
|
||||||
|
<DialogContent
|
||||||
|
zIndex="alert"
|
||||||
|
className="max-w-md glass border-white/10"
|
||||||
|
>
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle>{t("settings.restartRequired")}</DialogTitle>
|
||||||
|
</DialogHeader>
|
||||||
|
<div className="px-6">
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
{t("settings.restartRequiredMessage")}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<DialogFooter>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
onClick={handleRestartLater}
|
||||||
|
className="hover:bg-white/5"
|
||||||
|
>
|
||||||
|
{t("settings.restartLater")}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
onClick={handleRestartNow}
|
||||||
|
className="bg-primary hover:bg-primary/90"
|
||||||
|
>
|
||||||
|
{t("settings.restartNow")}
|
||||||
|
</Button>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -19,6 +19,13 @@ export function WindowSettings({ settings, onChange }: WindowSettingsProps) {
|
|||||||
</p>
|
</p>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
|
<ToggleRow
|
||||||
|
title={t("settings.launchOnStartup")}
|
||||||
|
description={t("settings.launchOnStartupDescription")}
|
||||||
|
checked={!!settings.launchOnStartup}
|
||||||
|
onCheckedChange={(value) => onChange({ launchOnStartup: value })}
|
||||||
|
/>
|
||||||
|
|
||||||
<ToggleRow
|
<ToggleRow
|
||||||
title={t("settings.minimizeToTray")}
|
title={t("settings.minimizeToTray")}
|
||||||
description={t("settings.minimizeToTrayDescription")}
|
description={t("settings.minimizeToTrayDescription")}
|
||||||
|
|||||||
219
src/components/skills/RepoManagerPanel.tsx
Normal file
219
src/components/skills/RepoManagerPanel.tsx
Normal file
@@ -0,0 +1,219 @@
|
|||||||
|
import { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { Label } from "@/components/ui/label";
|
||||||
|
import { Trash2, ExternalLink, Plus } from "lucide-react";
|
||||||
|
import { settingsApi } from "@/lib/api";
|
||||||
|
import { FullScreenPanel } from "@/components/common/FullScreenPanel";
|
||||||
|
import type { Skill, SkillRepo } from "@/lib/api/skills";
|
||||||
|
|
||||||
|
interface RepoManagerPanelProps {
|
||||||
|
repos: SkillRepo[];
|
||||||
|
skills: Skill[];
|
||||||
|
onAdd: (repo: SkillRepo) => Promise<void>;
|
||||||
|
onRemove: (owner: string, name: string) => Promise<void>;
|
||||||
|
onClose: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function RepoManagerPanel({
|
||||||
|
repos,
|
||||||
|
skills,
|
||||||
|
onAdd,
|
||||||
|
onRemove,
|
||||||
|
onClose,
|
||||||
|
}: RepoManagerPanelProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [repoUrl, setRepoUrl] = useState("");
|
||||||
|
const [branch, setBranch] = useState("");
|
||||||
|
const [skillsPath, setSkillsPath] = useState("");
|
||||||
|
const [error, setError] = useState("");
|
||||||
|
|
||||||
|
const getSkillCount = (repo: SkillRepo) =>
|
||||||
|
skills.filter(
|
||||||
|
(skill) =>
|
||||||
|
skill.repoOwner === repo.owner &&
|
||||||
|
skill.repoName === repo.name &&
|
||||||
|
(skill.repoBranch || "main") === (repo.branch || "main"),
|
||||||
|
).length;
|
||||||
|
|
||||||
|
const parseRepoUrl = (
|
||||||
|
url: string,
|
||||||
|
): { owner: string; name: string } | null => {
|
||||||
|
let cleaned = url.trim();
|
||||||
|
cleaned = cleaned.replace(/^https?:\/\/github\.com\//, "");
|
||||||
|
cleaned = cleaned.replace(/\.git$/, "");
|
||||||
|
|
||||||
|
const parts = cleaned.split("/");
|
||||||
|
if (parts.length === 2 && parts[0] && parts[1]) {
|
||||||
|
return { owner: parts[0], name: parts[1] };
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleAdd = async () => {
|
||||||
|
setError("");
|
||||||
|
|
||||||
|
const parsed = parseRepoUrl(repoUrl);
|
||||||
|
if (!parsed) {
|
||||||
|
setError(t("skills.repo.invalidUrl"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await onAdd({
|
||||||
|
owner: parsed.owner,
|
||||||
|
name: parsed.name,
|
||||||
|
branch: branch || "main",
|
||||||
|
enabled: true,
|
||||||
|
skillsPath: skillsPath.trim() || undefined,
|
||||||
|
});
|
||||||
|
|
||||||
|
setRepoUrl("");
|
||||||
|
setBranch("");
|
||||||
|
setSkillsPath("");
|
||||||
|
} catch (e) {
|
||||||
|
setError(e instanceof Error ? e.message : t("skills.repo.addFailed"));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleOpenRepo = async (owner: string, name: string) => {
|
||||||
|
try {
|
||||||
|
await settingsApi.openExternal(`https://github.com/${owner}/${name}`);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to open URL:", error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<FullScreenPanel
|
||||||
|
isOpen={true}
|
||||||
|
title={t("skills.repo.title")}
|
||||||
|
onClose={onClose}
|
||||||
|
>
|
||||||
|
{/* 添加仓库表单 */}
|
||||||
|
<div className="space-y-4 glass rounded-xl p-6 border border-white/10">
|
||||||
|
<h3 className="text-base font-semibold text-foreground">
|
||||||
|
添加技能仓库
|
||||||
|
</h3>
|
||||||
|
<div className="space-y-4">
|
||||||
|
<div>
|
||||||
|
<Label htmlFor="repo-url" className="text-foreground">
|
||||||
|
{t("skills.repo.url")}
|
||||||
|
</Label>
|
||||||
|
<Input
|
||||||
|
id="repo-url"
|
||||||
|
placeholder={t("skills.repo.urlPlaceholder")}
|
||||||
|
value={repoUrl}
|
||||||
|
onChange={(e) => setRepoUrl(e.target.value)}
|
||||||
|
className="mt-2"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
|
||||||
|
<div>
|
||||||
|
<Label htmlFor="branch" className="text-foreground">
|
||||||
|
{t("skills.repo.branch")}
|
||||||
|
</Label>
|
||||||
|
<Input
|
||||||
|
id="branch"
|
||||||
|
placeholder={t("skills.repo.branchPlaceholder")}
|
||||||
|
value={branch}
|
||||||
|
onChange={(e) => setBranch(e.target.value)}
|
||||||
|
className="mt-2"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<Label htmlFor="skills-path" className="text-foreground">
|
||||||
|
{t("skills.repo.path")}
|
||||||
|
</Label>
|
||||||
|
<Input
|
||||||
|
id="skills-path"
|
||||||
|
placeholder={t("skills.repo.pathPlaceholder")}
|
||||||
|
value={skillsPath}
|
||||||
|
onChange={(e) => setSkillsPath(e.target.value)}
|
||||||
|
className="mt-2"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{error && (
|
||||||
|
<p className="text-sm text-red-600 dark:text-red-400">{error}</p>
|
||||||
|
)}
|
||||||
|
<Button
|
||||||
|
onClick={handleAdd}
|
||||||
|
className="bg-primary text-primary-foreground hover:bg-primary/90"
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<Plus className="h-4 w-4 mr-2" />
|
||||||
|
{t("skills.repo.add")}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* 仓库列表 */}
|
||||||
|
<div className="space-y-4">
|
||||||
|
<h3 className="text-base font-semibold text-foreground">
|
||||||
|
{t("skills.repo.list")}
|
||||||
|
</h3>
|
||||||
|
{repos.length === 0 ? (
|
||||||
|
<div className="text-center py-12 glass rounded-xl border border-white/10">
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
{t("skills.repo.empty")}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="space-y-3">
|
||||||
|
{repos.map((repo) => (
|
||||||
|
<div
|
||||||
|
key={`${repo.owner}/${repo.name}`}
|
||||||
|
className="flex items-center justify-between rounded-xl border border-white/10 bg-gray-900/40 px-4 py-3"
|
||||||
|
>
|
||||||
|
<div>
|
||||||
|
<div className="text-sm font-medium text-foreground">
|
||||||
|
{repo.owner}/{repo.name}
|
||||||
|
</div>
|
||||||
|
<div className="mt-1 text-xs text-muted-foreground">
|
||||||
|
{t("skills.repo.branch")}: {repo.branch || "main"}
|
||||||
|
{repo.skillsPath && (
|
||||||
|
<>
|
||||||
|
<span className="mx-2">•</span>
|
||||||
|
{t("skills.repo.path")}: {repo.skillsPath}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
<span className="ml-3 inline-flex items-center rounded-full border border-border-default px-2 py-0.5 text-[11px]">
|
||||||
|
{t("skills.repo.skillCount", {
|
||||||
|
count: getSkillCount(repo),
|
||||||
|
})}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="icon"
|
||||||
|
type="button"
|
||||||
|
onClick={() => handleOpenRepo(repo.owner, repo.name)}
|
||||||
|
title={t("common.view", { defaultValue: "查看" })}
|
||||||
|
className="hover:bg-black/5 dark:hover:bg-white/5"
|
||||||
|
>
|
||||||
|
<ExternalLink className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="icon"
|
||||||
|
type="button"
|
||||||
|
onClick={() => onRemove(repo.owner, repo.name)}
|
||||||
|
title={t("common.delete")}
|
||||||
|
className="hover:text-red-500 hover:bg-red-100 dark:hover:text-red-400 dark:hover:bg-red-500/10"
|
||||||
|
>
|
||||||
|
<Trash2 className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</FullScreenPanel>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -57,7 +57,8 @@ export function SkillCard({ skill, onInstall, onUninstall }: SkillCardProps) {
|
|||||||
skill.directory.trim().toLowerCase() !== skill.name.trim().toLowerCase();
|
skill.directory.trim().toLowerCase() !== skill.name.trim().toLowerCase();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card className="flex flex-col h-full border-border-default bg-card transition-[border-color,box-shadow] duration-200 hover:border-border-hover hover:shadow-md">
|
<Card className="glass flex flex-col h-full border border-white/10 bg-gray-900/40 transition-all duration-300 hover:bg-gray-900/60 hover:border-white/20 hover:shadow-lg group relative overflow-hidden">
|
||||||
|
<div className="absolute inset-0 bg-gradient-to-br from-primary/5 to-transparent opacity-0 group-hover:opacity-100 transition-opacity duration-500 pointer-events-none" />
|
||||||
<CardHeader className="pb-3">
|
<CardHeader className="pb-3">
|
||||||
<div className="flex items-start justify-between gap-2">
|
<div className="flex items-start justify-between gap-2">
|
||||||
<div className="flex-1 min-w-0">
|
<div className="flex-1 min-w-0">
|
||||||
@@ -95,7 +96,7 @@ export function SkillCard({ skill, onInstall, onUninstall }: SkillCardProps) {
|
|||||||
{skill.description || t("skills.noDescription")}
|
{skill.description || t("skills.noDescription")}
|
||||||
</p>
|
</p>
|
||||||
</CardContent>
|
</CardContent>
|
||||||
<CardFooter className="flex gap-2 pt-3 border-t border-border-default">
|
<CardFooter className="flex gap-2 pt-3 border-t border-white/5 relative z-10">
|
||||||
{skill.readmeUrl && (
|
{skill.readmeUrl && (
|
||||||
<Button
|
<Button
|
||||||
variant="ghost"
|
variant="ghost"
|
||||||
|
|||||||
@@ -1,190 +1,275 @@
|
|||||||
import { useState, useEffect } from "react";
|
import { useState, useEffect, useMemo, forwardRef, useImperativeHandle } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { Button } from "@/components/ui/button";
|
import { Button } from "@/components/ui/button";
|
||||||
import { RefreshCw, Settings } from "lucide-react";
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { RefreshCw, Search } from "lucide-react";
|
||||||
import { toast } from "sonner";
|
import { toast } from "sonner";
|
||||||
import { SkillCard } from "./SkillCard";
|
import { SkillCard } from "./SkillCard";
|
||||||
import { RepoManager } from "./RepoManager";
|
import { RepoManagerPanel } from "./RepoManagerPanel";
|
||||||
import { skillsApi, type Skill, type SkillRepo } from "@/lib/api/skills";
|
import { skillsApi, type Skill, type SkillRepo } from "@/lib/api/skills";
|
||||||
|
import { formatSkillError } from "@/lib/errors/skillErrorParser";
|
||||||
|
|
||||||
interface SkillsPageProps {
|
interface SkillsPageProps {
|
||||||
onClose?: () => void;
|
onClose?: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function SkillsPage({ onClose: _onClose }: SkillsPageProps = {}) {
|
export interface SkillsPageHandle {
|
||||||
const { t } = useTranslation();
|
refresh: () => void;
|
||||||
const [skills, setSkills] = useState<Skill[]>([]);
|
openRepoManager: () => void;
|
||||||
const [repos, setRepos] = useState<SkillRepo[]>([]);
|
}
|
||||||
const [loading, setLoading] = useState(true);
|
|
||||||
const [repoManagerOpen, setRepoManagerOpen] = useState(false);
|
|
||||||
|
|
||||||
const loadSkills = async (afterLoad?: (data: Skill[]) => void) => {
|
export const SkillsPage = forwardRef<SkillsPageHandle, SkillsPageProps>(
|
||||||
try {
|
({ onClose: _onClose }, ref) => {
|
||||||
setLoading(true);
|
const { t } = useTranslation();
|
||||||
const data = await skillsApi.getAll();
|
const [skills, setSkills] = useState<Skill[]>([]);
|
||||||
setSkills(data);
|
const [repos, setRepos] = useState<SkillRepo[]>([]);
|
||||||
if (afterLoad) {
|
const [loading, setLoading] = useState(true);
|
||||||
afterLoad(data);
|
const [repoManagerOpen, setRepoManagerOpen] = useState(false);
|
||||||
|
const [searchQuery, setSearchQuery] = useState("");
|
||||||
|
|
||||||
|
const loadSkills = async (afterLoad?: (data: Skill[]) => void) => {
|
||||||
|
try {
|
||||||
|
setLoading(true);
|
||||||
|
const data = await skillsApi.getAll();
|
||||||
|
setSkills(data);
|
||||||
|
if (afterLoad) {
|
||||||
|
afterLoad(data);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
|
|
||||||
|
// 传入 "skills.loadFailed" 作为标题
|
||||||
|
const { title, description } = formatSkillError(
|
||||||
|
errorMessage,
|
||||||
|
t,
|
||||||
|
"skills.loadFailed",
|
||||||
|
);
|
||||||
|
|
||||||
|
toast.error(title, {
|
||||||
|
description,
|
||||||
|
duration: 8000,
|
||||||
|
});
|
||||||
|
|
||||||
|
console.error("Load skills failed:", error);
|
||||||
|
} finally {
|
||||||
|
setLoading(false);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
};
|
||||||
toast.error(t("skills.loadFailed"), {
|
|
||||||
description: error instanceof Error ? error.message : t("common.error"),
|
const loadRepos = async () => {
|
||||||
|
try {
|
||||||
|
const data = await skillsApi.getRepos();
|
||||||
|
setRepos(data);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to load repos:", error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
Promise.all([loadSkills(), loadRepos()]);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useImperativeHandle(ref, () => ({
|
||||||
|
refresh: () => loadSkills(),
|
||||||
|
openRepoManager: () => setRepoManagerOpen(true),
|
||||||
|
}));
|
||||||
|
|
||||||
|
const handleInstall = async (directory: string) => {
|
||||||
|
try {
|
||||||
|
await skillsApi.install(directory);
|
||||||
|
toast.success(t("skills.installSuccess", { name: directory }));
|
||||||
|
await loadSkills();
|
||||||
|
} catch (error) {
|
||||||
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
|
|
||||||
|
// 使用错误解析器格式化错误,传入 "skills.installFailed"
|
||||||
|
const { title, description } = formatSkillError(
|
||||||
|
errorMessage,
|
||||||
|
t,
|
||||||
|
"skills.installFailed",
|
||||||
|
);
|
||||||
|
|
||||||
|
toast.error(title, {
|
||||||
|
description,
|
||||||
|
duration: 10000, // 延长显示时间让用户看清
|
||||||
|
});
|
||||||
|
|
||||||
|
console.error("Install skill failed:", {
|
||||||
|
directory,
|
||||||
|
error,
|
||||||
|
message: errorMessage,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleUninstall = async (directory: string) => {
|
||||||
|
try {
|
||||||
|
await skillsApi.uninstall(directory);
|
||||||
|
toast.success(t("skills.uninstallSuccess", { name: directory }));
|
||||||
|
await loadSkills();
|
||||||
|
} catch (error) {
|
||||||
|
const errorMessage =
|
||||||
|
error instanceof Error ? error.message : String(error);
|
||||||
|
|
||||||
|
// 使用错误解析器格式化错误,传入 "skills.uninstallFailed"
|
||||||
|
const { title, description } = formatSkillError(
|
||||||
|
errorMessage,
|
||||||
|
t,
|
||||||
|
"skills.uninstallFailed",
|
||||||
|
);
|
||||||
|
|
||||||
|
toast.error(title, {
|
||||||
|
description,
|
||||||
|
duration: 10000,
|
||||||
|
});
|
||||||
|
|
||||||
|
console.error("Uninstall skill failed:", {
|
||||||
|
directory,
|
||||||
|
error,
|
||||||
|
message: errorMessage,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleAddRepo = async (repo: SkillRepo) => {
|
||||||
|
await skillsApi.addRepo(repo);
|
||||||
|
|
||||||
|
let repoSkillCount = 0;
|
||||||
|
await Promise.all([
|
||||||
|
loadRepos(),
|
||||||
|
loadSkills((data) => {
|
||||||
|
repoSkillCount = data.filter(
|
||||||
|
(skill) =>
|
||||||
|
skill.repoOwner === repo.owner &&
|
||||||
|
skill.repoName === repo.name &&
|
||||||
|
(skill.repoBranch || "main") === (repo.branch || "main"),
|
||||||
|
).length;
|
||||||
|
}),
|
||||||
|
]);
|
||||||
|
|
||||||
|
toast.success(
|
||||||
|
t("skills.repo.addSuccess", {
|
||||||
|
owner: repo.owner,
|
||||||
|
name: repo.name,
|
||||||
|
count: repoSkillCount,
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleRemoveRepo = async (owner: string, name: string) => {
|
||||||
|
await skillsApi.removeRepo(owner, name);
|
||||||
|
toast.success(t("skills.repo.removeSuccess", { owner, name }));
|
||||||
|
await Promise.all([loadRepos(), loadSkills()]);
|
||||||
|
};
|
||||||
|
|
||||||
|
// 过滤技能列表
|
||||||
|
const filteredSkills = useMemo(() => {
|
||||||
|
if (!searchQuery.trim()) return skills;
|
||||||
|
|
||||||
|
const query = searchQuery.toLowerCase();
|
||||||
|
return skills.filter((skill) => {
|
||||||
|
const name = skill.name?.toLowerCase() || "";
|
||||||
|
const description = skill.description?.toLowerCase() || "";
|
||||||
|
const directory = skill.directory?.toLowerCase() || "";
|
||||||
|
|
||||||
|
return (
|
||||||
|
name.includes(query) ||
|
||||||
|
description.includes(query) ||
|
||||||
|
directory.includes(query)
|
||||||
|
);
|
||||||
});
|
});
|
||||||
} finally {
|
}, [skills, searchQuery]);
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const loadRepos = async () => {
|
return (
|
||||||
try {
|
<div className="flex flex-col h-full min-h-0 bg-background/50">
|
||||||
const data = await skillsApi.getRepos();
|
{/* 顶部操作栏(固定区域)已移除,由 App.tsx 接管 */}
|
||||||
setRepos(data);
|
|
||||||
} catch (error) {
|
|
||||||
console.error("Failed to load repos:", error);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => {
|
{/* 技能网格(可滚动详情区域) */}
|
||||||
Promise.all([loadSkills(), loadRepos()]);
|
<div className="flex-1 min-h-0 overflow-y-auto animate-fade-in">
|
||||||
}, []);
|
<div className="mx-auto max-w-[56rem] px-6 py-4">
|
||||||
|
{loading ? (
|
||||||
|
<div className="flex items-center justify-center h-64">
|
||||||
|
<RefreshCw className="h-8 w-8 animate-spin text-muted-foreground" />
|
||||||
|
</div>
|
||||||
|
) : skills.length === 0 ? (
|
||||||
|
<div className="flex flex-col items-center justify-center h-64 text-center">
|
||||||
|
<p className="text-lg font-medium text-gray-900 dark:text-gray-100">
|
||||||
|
{t("skills.empty")}
|
||||||
|
</p>
|
||||||
|
<p className="mt-2 text-sm text-gray-500 dark:text-gray-400">
|
||||||
|
{t("skills.emptyDescription")}
|
||||||
|
</p>
|
||||||
|
<Button
|
||||||
|
variant="link"
|
||||||
|
onClick={() => setRepoManagerOpen(true)}
|
||||||
|
className="mt-3 text-sm font-normal"
|
||||||
|
>
|
||||||
|
{t("skills.addRepo")}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
{/* 搜索框 */}
|
||||||
|
<div className="mb-6">
|
||||||
|
<div className="relative">
|
||||||
|
<Search className="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4 text-muted-foreground" />
|
||||||
|
<Input
|
||||||
|
type="text"
|
||||||
|
placeholder={t("skills.searchPlaceholder")}
|
||||||
|
value={searchQuery}
|
||||||
|
onChange={(e) => setSearchQuery(e.target.value)}
|
||||||
|
className="pl-9"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
{searchQuery && (
|
||||||
|
<p className="mt-2 text-sm text-muted-foreground">
|
||||||
|
{t("skills.count", { count: filteredSkills.length })}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
const handleInstall = async (directory: string) => {
|
{/* 技能列表或无结果提示 */}
|
||||||
try {
|
{filteredSkills.length === 0 ? (
|
||||||
await skillsApi.install(directory);
|
<div className="flex flex-col items-center justify-center h-48 text-center">
|
||||||
toast.success(t("skills.installSuccess", { name: directory }));
|
<p className="text-lg font-medium text-gray-900 dark:text-gray-100">
|
||||||
await loadSkills();
|
{t("skills.noResults")}
|
||||||
} catch (error) {
|
</p>
|
||||||
toast.error(t("skills.installFailed"), {
|
<p className="mt-2 text-sm text-gray-500 dark:text-gray-400">
|
||||||
description: error instanceof Error ? error.message : t("common.error"),
|
{t("skills.emptyDescription")}
|
||||||
});
|
</p>
|
||||||
}
|
</div>
|
||||||
};
|
) : (
|
||||||
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
|
||||||
const handleUninstall = async (directory: string) => {
|
{filteredSkills.map((skill) => (
|
||||||
try {
|
<SkillCard
|
||||||
await skillsApi.uninstall(directory);
|
key={skill.key}
|
||||||
toast.success(t("skills.uninstallSuccess", { name: directory }));
|
skill={skill}
|
||||||
await loadSkills();
|
onInstall={handleInstall}
|
||||||
} catch (error) {
|
onUninstall={handleUninstall}
|
||||||
toast.error(t("skills.uninstallFailed"), {
|
/>
|
||||||
description: error instanceof Error ? error.message : t("common.error"),
|
))}
|
||||||
});
|
</div>
|
||||||
}
|
)}
|
||||||
};
|
</>
|
||||||
|
)}
|
||||||
const handleAddRepo = async (repo: SkillRepo) => {
|
|
||||||
await skillsApi.addRepo(repo);
|
|
||||||
|
|
||||||
let repoSkillCount = 0;
|
|
||||||
await Promise.all([
|
|
||||||
loadRepos(),
|
|
||||||
loadSkills((data) => {
|
|
||||||
repoSkillCount = data.filter(
|
|
||||||
(skill) =>
|
|
||||||
skill.repoOwner === repo.owner &&
|
|
||||||
skill.repoName === repo.name &&
|
|
||||||
(skill.repoBranch || "main") === (repo.branch || "main"),
|
|
||||||
).length;
|
|
||||||
}),
|
|
||||||
]);
|
|
||||||
|
|
||||||
toast.success(
|
|
||||||
t("skills.repo.addSuccess", {
|
|
||||||
owner: repo.owner,
|
|
||||||
name: repo.name,
|
|
||||||
count: repoSkillCount,
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleRemoveRepo = async (owner: string, name: string) => {
|
|
||||||
await skillsApi.removeRepo(owner, name);
|
|
||||||
toast.success(t("skills.repo.removeSuccess", { owner, name }));
|
|
||||||
await Promise.all([loadRepos(), loadSkills()]);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className="flex flex-col h-full min-h-0 bg-background">
|
|
||||||
{/* 顶部操作栏(固定区域) */}
|
|
||||||
<div className="flex-shrink-0 border-b border-border-default bg-muted/20 px-6 py-4">
|
|
||||||
<div className="flex items-center justify-between pr-8">
|
|
||||||
<h1 className="text-lg font-semibold leading-tight tracking-tight text-gray-900 dark:text-gray-100">
|
|
||||||
{t("skills.title")}
|
|
||||||
</h1>
|
|
||||||
<div className="flex gap-2">
|
|
||||||
<Button
|
|
||||||
variant="mcp"
|
|
||||||
size="sm"
|
|
||||||
onClick={() => loadSkills()}
|
|
||||||
disabled={loading}
|
|
||||||
>
|
|
||||||
<RefreshCw
|
|
||||||
className={`h-4 w-4 mr-2 ${loading ? "animate-spin" : ""}`}
|
|
||||||
/>
|
|
||||||
{loading ? t("skills.refreshing") : t("skills.refresh")}
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
variant="mcp"
|
|
||||||
size="sm"
|
|
||||||
onClick={() => setRepoManagerOpen(true)}
|
|
||||||
>
|
|
||||||
<Settings className="h-4 w-4 mr-2" />
|
|
||||||
{t("skills.repoManager")}
|
|
||||||
</Button>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* 描述 */}
|
{/* 仓库管理面板 */}
|
||||||
<p className="mt-1.5 text-sm text-gray-500 dark:text-gray-400">
|
{repoManagerOpen && (
|
||||||
{t("skills.description")}
|
<RepoManagerPanel
|
||||||
</p>
|
repos={repos}
|
||||||
</div>
|
skills={skills}
|
||||||
|
onAdd={handleAddRepo}
|
||||||
{/* 技能网格(可滚动详情区域) */}
|
onRemove={handleRemoveRepo}
|
||||||
<div className="flex-1 min-h-0 overflow-y-auto px-6 py-6 bg-muted/10">
|
onClose={() => setRepoManagerOpen(false)}
|
||||||
{loading ? (
|
/>
|
||||||
<div className="flex items-center justify-center h-64">
|
|
||||||
<RefreshCw className="h-8 w-8 animate-spin text-muted-foreground" />
|
|
||||||
</div>
|
|
||||||
) : skills.length === 0 ? (
|
|
||||||
<div className="flex flex-col items-center justify-center h-64 text-center">
|
|
||||||
<p className="text-lg font-medium text-gray-900 dark:text-gray-100">
|
|
||||||
{t("skills.empty")}
|
|
||||||
</p>
|
|
||||||
<p className="mt-2 text-sm text-gray-500 dark:text-gray-400">
|
|
||||||
{t("skills.emptyDescription")}
|
|
||||||
</p>
|
|
||||||
<Button
|
|
||||||
variant="link"
|
|
||||||
onClick={() => setRepoManagerOpen(true)}
|
|
||||||
className="mt-3 text-sm font-normal"
|
|
||||||
>
|
|
||||||
{t("skills.addRepo")}
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
|
|
||||||
{skills.map((skill) => (
|
|
||||||
<SkillCard
|
|
||||||
key={skill.key}
|
|
||||||
skill={skill}
|
|
||||||
onInstall={handleInstall}
|
|
||||||
onUninstall={handleUninstall}
|
|
||||||
/>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
{/* 仓库管理对话框 */}
|
SkillsPage.displayName = "SkillsPage";
|
||||||
<RepoManager
|
|
||||||
open={repoManagerOpen}
|
|
||||||
onOpenChange={setRepoManagerOpen}
|
|
||||||
repos={repos}
|
|
||||||
skills={skills}
|
|
||||||
onAdd={handleAddRepo}
|
|
||||||
onRemove={handleRemoveRepo}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import * as DialogPrimitive from "@radix-ui/react-dialog";
|
import * as DialogPrimitive from "@radix-ui/react-dialog";
|
||||||
import { X } from "lucide-react";
|
|
||||||
import { cn } from "@/lib/utils";
|
import { cn } from "@/lib/utils";
|
||||||
|
|
||||||
const Dialog = DialogPrimitive.Root;
|
const Dialog = DialogPrimitive.Root;
|
||||||
@@ -14,13 +13,14 @@ const DialogClose = DialogPrimitive.Close;
|
|||||||
const DialogOverlay = React.forwardRef<
|
const DialogOverlay = React.forwardRef<
|
||||||
React.ElementRef<typeof DialogPrimitive.Overlay>,
|
React.ElementRef<typeof DialogPrimitive.Overlay>,
|
||||||
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Overlay> & {
|
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Overlay> & {
|
||||||
zIndex?: "base" | "nested" | "alert";
|
zIndex?: "base" | "nested" | "alert" | "top";
|
||||||
}
|
}
|
||||||
>(({ className, zIndex = "base", ...props }, ref) => {
|
>(({ className, zIndex = "base", ...props }, ref) => {
|
||||||
const zIndexMap = {
|
const zIndexMap = {
|
||||||
base: "z-40",
|
base: "z-40",
|
||||||
nested: "z-50",
|
nested: "z-50",
|
||||||
alert: "z-[60]",
|
alert: "z-[60]",
|
||||||
|
top: "z-[110]",
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -40,36 +40,54 @@ DialogOverlay.displayName = DialogPrimitive.Overlay.displayName;
|
|||||||
const DialogContent = React.forwardRef<
|
const DialogContent = React.forwardRef<
|
||||||
React.ElementRef<typeof DialogPrimitive.Content>,
|
React.ElementRef<typeof DialogPrimitive.Content>,
|
||||||
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Content> & {
|
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Content> & {
|
||||||
zIndex?: "base" | "nested" | "alert";
|
zIndex?: "base" | "nested" | "alert" | "top";
|
||||||
|
variant?: "default" | "fullscreen";
|
||||||
|
overlayClassName?: string;
|
||||||
}
|
}
|
||||||
>(({ className, children, zIndex = "base", ...props }, ref) => {
|
>(
|
||||||
const zIndexMap = {
|
(
|
||||||
base: "z-40",
|
{
|
||||||
nested: "z-50",
|
className,
|
||||||
alert: "z-[60]",
|
children,
|
||||||
};
|
zIndex = "base",
|
||||||
|
variant = "default",
|
||||||
|
overlayClassName,
|
||||||
|
...props
|
||||||
|
},
|
||||||
|
ref,
|
||||||
|
) => {
|
||||||
|
const zIndexMap = {
|
||||||
|
base: "z-40",
|
||||||
|
nested: "z-50",
|
||||||
|
alert: "z-[60]",
|
||||||
|
top: "z-[110]",
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
const variantClass = {
|
||||||
<DialogPortal>
|
default:
|
||||||
<DialogOverlay zIndex={zIndex} />
|
"fixed left-1/2 top-1/2 flex flex-col w-full max-w-lg max-h-[90vh] translate-x-[-50%] translate-y-[-50%] border border-border-default bg-background text-foreground shadow-lg duration-200 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[state=closed]:slide-out-to-left-1/2 data-[state=closed]:slide-out-to-top-[48%] data-[state=open]:slide-in-from-left-1/2 data-[state=open]:slide-in-from-top-[48%] sm:rounded-lg",
|
||||||
<DialogPrimitive.Content
|
fullscreen:
|
||||||
ref={ref}
|
"fixed inset-0 flex flex-col w-screen h-screen translate-x-0 translate-y-0 bg-background text-foreground p-0 sm:rounded-none shadow-none",
|
||||||
className={cn(
|
}[variant];
|
||||||
"fixed left-1/2 top-1/2 flex flex-col w-full max-w-lg max-h-[90vh] translate-x-[-50%] translate-y-[-50%] border border-border-default bg-white dark:bg-gray-900 shadow-lg duration-200 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[state=closed]:slide-out-to-left-1/2 data-[state=closed]:slide-out-to-top-[48%] data-[state=open]:slide-in-from-left-1/2 data-[state=open]:slide-in-from-top-[48%] sm:rounded-lg",
|
|
||||||
zIndexMap[zIndex],
|
return (
|
||||||
className,
|
<DialogPortal>
|
||||||
)}
|
<DialogOverlay zIndex={zIndex} className={overlayClassName} />
|
||||||
{...props}
|
<DialogPrimitive.Content
|
||||||
>
|
ref={ref}
|
||||||
{children}
|
className={cn(variantClass, zIndexMap[zIndex], className)}
|
||||||
<DialogPrimitive.Close className="absolute right-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none">
|
onInteractOutside={(e) => {
|
||||||
<X className="h-4 w-4" />
|
// 防止点击遮罩层关闭对话框
|
||||||
<span className="sr-only">关闭</span>
|
e.preventDefault();
|
||||||
</DialogPrimitive.Close>
|
}}
|
||||||
</DialogPrimitive.Content>
|
{...props}
|
||||||
</DialogPortal>
|
>
|
||||||
);
|
{children}
|
||||||
});
|
</DialogPrimitive.Content>
|
||||||
|
</DialogPortal>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
DialogContent.displayName = DialogPrimitive.Content.displayName;
|
DialogContent.displayName = DialogPrimitive.Content.displayName;
|
||||||
|
|
||||||
const DialogHeader = ({
|
const DialogHeader = ({
|
||||||
|
|||||||
@@ -40,6 +40,9 @@ export interface ProviderPreset {
|
|||||||
endpointCandidates?: string[];
|
endpointCandidates?: string[];
|
||||||
// 新增:视觉主题配置
|
// 新增:视觉主题配置
|
||||||
theme?: PresetTheme;
|
theme?: PresetTheme;
|
||||||
|
// 图标配置
|
||||||
|
icon?: string; // 图标名称
|
||||||
|
iconColor?: string; // 图标颜色
|
||||||
}
|
}
|
||||||
|
|
||||||
export const providerPresets: ProviderPreset[] = [
|
export const providerPresets: ProviderPreset[] = [
|
||||||
@@ -56,6 +59,8 @@ export const providerPresets: ProviderPreset[] = [
|
|||||||
backgroundColor: "#D97757",
|
backgroundColor: "#D97757",
|
||||||
textColor: "#FFFFFF",
|
textColor: "#FFFFFF",
|
||||||
},
|
},
|
||||||
|
icon: "anthropic",
|
||||||
|
iconColor: "#D4915D",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "DeepSeek",
|
name: "DeepSeek",
|
||||||
@@ -230,6 +235,23 @@ export const providerPresets: ProviderPreset[] = [
|
|||||||
},
|
},
|
||||||
category: "cn_official",
|
category: "cn_official",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
name: "DouBaoSeed",
|
||||||
|
websiteUrl: "https://www.volcengine.com/product/doubao",
|
||||||
|
apiKeyUrl: "https://www.volcengine.com/product/doubao",
|
||||||
|
settingsConfig: {
|
||||||
|
env: {
|
||||||
|
ANTHROPIC_BASE_URL: "https://ark.cn-beijing.volces.com/api/coding",
|
||||||
|
ANTHROPIC_AUTH_TOKEN: "",
|
||||||
|
API_TIMEOUT_MS: "3000000",
|
||||||
|
ANTHROPIC_MODEL: "doubao-seed-code-preview-latest",
|
||||||
|
ANTHROPIC_DEFAULT_SONNET_MODEL: "doubao-seed-code-preview-latest",
|
||||||
|
ANTHROPIC_DEFAULT_OPUS_MODEL: "doubao-seed-code-preview-latest",
|
||||||
|
ANTHROPIC_DEFAULT_HAIKU_MODEL: "doubao-seed-code-preview-latest",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
category: "cn_official",
|
||||||
|
},
|
||||||
{
|
{
|
||||||
name: "BaiLing",
|
name: "BaiLing",
|
||||||
websiteUrl: "https://alipaytbox.yuque.com/sxs0ba/ling/get_started",
|
websiteUrl: "https://alipaytbox.yuque.com/sxs0ba/ling/get_started",
|
||||||
@@ -294,22 +316,4 @@ export const providerPresets: ProviderPreset[] = [
|
|||||||
isPartner: true, // 合作伙伴
|
isPartner: true, // 合作伙伴
|
||||||
partnerPromotionKey: "packycode", // 促销信息 i18n key
|
partnerPromotionKey: "packycode", // 促销信息 i18n key
|
||||||
},
|
},
|
||||||
{
|
|
||||||
name: "AnyRouter",
|
|
||||||
websiteUrl: "https://anyrouter.top",
|
|
||||||
apiKeyUrl: "https://anyrouter.top/register?aff=PCel",
|
|
||||||
settingsConfig: {
|
|
||||||
env: {
|
|
||||||
ANTHROPIC_BASE_URL: "https://anyrouter.top",
|
|
||||||
ANTHROPIC_AUTH_TOKEN: "",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
// 请求地址候选(用于地址管理/测速)
|
|
||||||
endpointCandidates: [
|
|
||||||
"https://q.quuvv.cn",
|
|
||||||
"https://pmpjfbhq.cn-nb1.rainapp.top",
|
|
||||||
"https://anyrouter.top",
|
|
||||||
],
|
|
||||||
category: "third_party",
|
|
||||||
},
|
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -143,20 +143,4 @@ requires_openai_auth = true`,
|
|||||||
isPartner: true, // 合作伙伴
|
isPartner: true, // 合作伙伴
|
||||||
partnerPromotionKey: "packycode", // 促销信息 i18n key
|
partnerPromotionKey: "packycode", // 促销信息 i18n key
|
||||||
},
|
},
|
||||||
{
|
|
||||||
name: "AnyRouter",
|
|
||||||
websiteUrl: "https://anyrouter.top",
|
|
||||||
category: "third_party",
|
|
||||||
auth: generateThirdPartyAuth(""),
|
|
||||||
config: generateThirdPartyConfig(
|
|
||||||
"anyrouter",
|
|
||||||
"https://anyrouter.top/v1",
|
|
||||||
"gpt-5-codex",
|
|
||||||
),
|
|
||||||
endpointCandidates: [
|
|
||||||
"https://anyrouter.top/v1",
|
|
||||||
"https://q.quuvv.cn/v1",
|
|
||||||
"https://pmpjfbhq.cn-nb1.rainapp.top/v1",
|
|
||||||
],
|
|
||||||
},
|
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -33,14 +33,11 @@ export const geminiProviderPresets: GeminiProviderPreset[] = [
|
|||||||
websiteUrl: "https://ai.google.dev/",
|
websiteUrl: "https://ai.google.dev/",
|
||||||
apiKeyUrl: "https://aistudio.google.com/apikey",
|
apiKeyUrl: "https://aistudio.google.com/apikey",
|
||||||
settingsConfig: {
|
settingsConfig: {
|
||||||
env: {
|
env: {},
|
||||||
GEMINI_MODEL: "gemini-2.5-pro",
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
description: "Google 官方 Gemini API (OAuth)",
|
description: "Google 官方 Gemini API (OAuth)",
|
||||||
category: "official",
|
category: "official",
|
||||||
partnerPromotionKey: "google-official",
|
partnerPromotionKey: "google-official",
|
||||||
model: "gemini-2.5-pro",
|
|
||||||
theme: {
|
theme: {
|
||||||
icon: "gemini",
|
icon: "gemini",
|
||||||
backgroundColor: "#4285F4",
|
backgroundColor: "#4285F4",
|
||||||
@@ -54,11 +51,11 @@ export const geminiProviderPresets: GeminiProviderPreset[] = [
|
|||||||
settingsConfig: {
|
settingsConfig: {
|
||||||
env: {
|
env: {
|
||||||
GOOGLE_GEMINI_BASE_URL: "https://www.packyapi.com",
|
GOOGLE_GEMINI_BASE_URL: "https://www.packyapi.com",
|
||||||
GEMINI_MODEL: "gemini-2.5-pro",
|
GEMINI_MODEL: "gemini-3-pro-preview",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
baseURL: "https://www.packyapi.com",
|
baseURL: "https://www.packyapi.com",
|
||||||
model: "gemini-2.5-pro",
|
model: "gemini-3-pro-preview",
|
||||||
description: "PackyCode",
|
description: "PackyCode",
|
||||||
category: "third_party",
|
category: "third_party",
|
||||||
isPartner: true,
|
isPartner: true,
|
||||||
@@ -74,10 +71,10 @@ export const geminiProviderPresets: GeminiProviderPreset[] = [
|
|||||||
settingsConfig: {
|
settingsConfig: {
|
||||||
env: {
|
env: {
|
||||||
GOOGLE_GEMINI_BASE_URL: "",
|
GOOGLE_GEMINI_BASE_URL: "",
|
||||||
GEMINI_MODEL: "gemini-2.5-pro",
|
GEMINI_MODEL: "gemini-3-pro-preview",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
model: "gemini-2.5-pro",
|
model: "gemini-3-pro-preview",
|
||||||
description: "自定义 Gemini API 端点",
|
description: "自定义 Gemini API 端点",
|
||||||
category: "custom",
|
category: "custom",
|
||||||
},
|
},
|
||||||
|
|||||||
73
src/config/iconInference.ts
Normal file
73
src/config/iconInference.ts
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
/**
|
||||||
|
* 根据供应商名称智能推断图标配置
|
||||||
|
*/
|
||||||
|
|
||||||
|
const iconMappings = {
|
||||||
|
// AI 服务商
|
||||||
|
claude: { icon: "claude", iconColor: "#D4915D" },
|
||||||
|
anthropic: { icon: "anthropic", iconColor: "#D4915D" },
|
||||||
|
deepseek: { icon: "deepseek", iconColor: "#1E88E5" },
|
||||||
|
zhipu: { icon: "zhipu", iconColor: "#0F62FE" },
|
||||||
|
glm: { icon: "zhipu", iconColor: "#0F62FE" },
|
||||||
|
qwen: { icon: "qwen", iconColor: "#FF6A00" },
|
||||||
|
alibaba: { icon: "alibaba", iconColor: "#FF6A00" },
|
||||||
|
aliyun: { icon: "alibaba", iconColor: "#FF6A00" },
|
||||||
|
kimi: { icon: "kimi", iconColor: "#6366F1" },
|
||||||
|
moonshot: { icon: "moonshot", iconColor: "#6366F1" },
|
||||||
|
baidu: { icon: "baidu", iconColor: "#2932E1" },
|
||||||
|
tencent: { icon: "tencent", iconColor: "#00A4FF" },
|
||||||
|
hunyuan: { icon: "hunyuan", iconColor: "#00A4FF" },
|
||||||
|
minimax: { icon: "minimax", iconColor: "#FF6B6B" },
|
||||||
|
google: { icon: "google", iconColor: "#4285F4" },
|
||||||
|
meta: { icon: "meta", iconColor: "#0081FB" },
|
||||||
|
mistral: { icon: "mistral", iconColor: "#FF7000" },
|
||||||
|
cohere: { icon: "cohere", iconColor: "#39594D" },
|
||||||
|
perplexity: { icon: "perplexity", iconColor: "#20808D" },
|
||||||
|
huggingface: { icon: "huggingface", iconColor: "#FFD21E" },
|
||||||
|
|
||||||
|
// 云平台
|
||||||
|
aws: { icon: "aws", iconColor: "#FF9900" },
|
||||||
|
azure: { icon: "azure", iconColor: "#0078D4" },
|
||||||
|
huawei: { icon: "huawei", iconColor: "#FF0000" },
|
||||||
|
cloudflare: { icon: "cloudflare", iconColor: "#F38020" },
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 根据预设名称推断图标
|
||||||
|
*/
|
||||||
|
export function inferIconForPreset(presetName: string): {
|
||||||
|
icon?: string;
|
||||||
|
iconColor?: string;
|
||||||
|
} {
|
||||||
|
const nameLower = presetName.toLowerCase();
|
||||||
|
|
||||||
|
// 精确匹配或模糊匹配
|
||||||
|
for (const [key, config] of Object.entries(iconMappings)) {
|
||||||
|
if (nameLower.includes(key)) {
|
||||||
|
return config;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 批量为预设添加图标配置
|
||||||
|
*/
|
||||||
|
export function addIconsToPresets<
|
||||||
|
T extends { name: string; icon?: string; iconColor?: string },
|
||||||
|
>(presets: T[]): T[] {
|
||||||
|
return presets.map((preset) => {
|
||||||
|
// 如果已经配置了图标,则保留原配置
|
||||||
|
if (preset.icon) {
|
||||||
|
return preset;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 否则根据名称推断
|
||||||
|
const inferred = inferIconForPreset(preset.name);
|
||||||
|
return {
|
||||||
|
...preset,
|
||||||
|
...inferred,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -5,12 +5,13 @@ import { homeDir, join } from "@tauri-apps/api/path";
|
|||||||
import { settingsApi, type AppId } from "@/lib/api";
|
import { settingsApi, type AppId } from "@/lib/api";
|
||||||
import type { SettingsFormState } from "./useSettingsForm";
|
import type { SettingsFormState } from "./useSettingsForm";
|
||||||
|
|
||||||
type DirectoryKey = "appConfig" | "claude" | "codex";
|
type DirectoryKey = "appConfig" | "claude" | "codex" | "gemini";
|
||||||
|
|
||||||
export interface ResolvedDirectories {
|
export interface ResolvedDirectories {
|
||||||
appConfig: string;
|
appConfig: string;
|
||||||
claude: string;
|
claude: string;
|
||||||
codex: string;
|
codex: string;
|
||||||
|
gemini: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const sanitizeDir = (value?: string | null): string | undefined => {
|
const sanitizeDir = (value?: string | null): string | undefined => {
|
||||||
@@ -37,7 +38,8 @@ const computeDefaultConfigDir = async (
|
|||||||
): Promise<string | undefined> => {
|
): Promise<string | undefined> => {
|
||||||
try {
|
try {
|
||||||
const home = await homeDir();
|
const home = await homeDir();
|
||||||
const folder = app === "claude" ? ".claude" : ".codex";
|
const folder =
|
||||||
|
app === "claude" ? ".claude" : app === "codex" ? ".codex" : ".gemini";
|
||||||
return await join(home, folder);
|
return await join(home, folder);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(
|
console.error(
|
||||||
@@ -64,7 +66,11 @@ export interface UseDirectorySettingsResult {
|
|||||||
browseAppConfigDir: () => Promise<void>;
|
browseAppConfigDir: () => Promise<void>;
|
||||||
resetDirectory: (app: AppId) => Promise<void>;
|
resetDirectory: (app: AppId) => Promise<void>;
|
||||||
resetAppConfigDir: () => Promise<void>;
|
resetAppConfigDir: () => Promise<void>;
|
||||||
resetAllDirectories: (claudeDir?: string, codexDir?: string) => void;
|
resetAllDirectories: (
|
||||||
|
claudeDir?: string,
|
||||||
|
codexDir?: string,
|
||||||
|
geminiDir?: string,
|
||||||
|
) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -89,6 +95,7 @@ export function useDirectorySettings({
|
|||||||
appConfig: "",
|
appConfig: "",
|
||||||
claude: "",
|
claude: "",
|
||||||
codex: "",
|
codex: "",
|
||||||
|
gemini: "",
|
||||||
});
|
});
|
||||||
const [isLoading, setIsLoading] = useState(true);
|
const [isLoading, setIsLoading] = useState(true);
|
||||||
|
|
||||||
@@ -96,6 +103,7 @@ export function useDirectorySettings({
|
|||||||
appConfig: "",
|
appConfig: "",
|
||||||
claude: "",
|
claude: "",
|
||||||
codex: "",
|
codex: "",
|
||||||
|
gemini: "",
|
||||||
});
|
});
|
||||||
const initialAppConfigDirRef = useRef<string | undefined>(undefined);
|
const initialAppConfigDirRef = useRef<string | undefined>(undefined);
|
||||||
|
|
||||||
@@ -110,16 +118,20 @@ export function useDirectorySettings({
|
|||||||
overrideRaw,
|
overrideRaw,
|
||||||
claudeDir,
|
claudeDir,
|
||||||
codexDir,
|
codexDir,
|
||||||
|
geminiDir,
|
||||||
defaultAppConfig,
|
defaultAppConfig,
|
||||||
defaultClaudeDir,
|
defaultClaudeDir,
|
||||||
defaultCodexDir,
|
defaultCodexDir,
|
||||||
|
defaultGeminiDir,
|
||||||
] = await Promise.all([
|
] = await Promise.all([
|
||||||
settingsApi.getAppConfigDirOverride(),
|
settingsApi.getAppConfigDirOverride(),
|
||||||
settingsApi.getConfigDir("claude"),
|
settingsApi.getConfigDir("claude"),
|
||||||
settingsApi.getConfigDir("codex"),
|
settingsApi.getConfigDir("codex"),
|
||||||
|
settingsApi.getConfigDir("gemini"),
|
||||||
computeDefaultAppConfigDir(),
|
computeDefaultAppConfigDir(),
|
||||||
computeDefaultConfigDir("claude"),
|
computeDefaultConfigDir("claude"),
|
||||||
computeDefaultConfigDir("codex"),
|
computeDefaultConfigDir("codex"),
|
||||||
|
computeDefaultConfigDir("gemini"),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
if (!active) return;
|
if (!active) return;
|
||||||
@@ -130,6 +142,7 @@ export function useDirectorySettings({
|
|||||||
appConfig: defaultAppConfig ?? "",
|
appConfig: defaultAppConfig ?? "",
|
||||||
claude: defaultClaudeDir ?? "",
|
claude: defaultClaudeDir ?? "",
|
||||||
codex: defaultCodexDir ?? "",
|
codex: defaultCodexDir ?? "",
|
||||||
|
gemini: defaultGeminiDir ?? "",
|
||||||
};
|
};
|
||||||
|
|
||||||
setAppConfigDir(normalizedOverride);
|
setAppConfigDir(normalizedOverride);
|
||||||
@@ -139,6 +152,7 @@ export function useDirectorySettings({
|
|||||||
appConfig: normalizedOverride ?? defaultsRef.current.appConfig,
|
appConfig: normalizedOverride ?? defaultsRef.current.appConfig,
|
||||||
claude: claudeDir || defaultsRef.current.claude,
|
claude: claudeDir || defaultsRef.current.claude,
|
||||||
codex: codexDir || defaultsRef.current.codex,
|
codex: codexDir || defaultsRef.current.codex,
|
||||||
|
gemini: geminiDir || defaultsRef.current.gemini,
|
||||||
});
|
});
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(
|
console.error(
|
||||||
@@ -167,7 +181,9 @@ export function useDirectorySettings({
|
|||||||
onUpdateSettings(
|
onUpdateSettings(
|
||||||
key === "claude"
|
key === "claude"
|
||||||
? { claudeConfigDir: sanitized }
|
? { claudeConfigDir: sanitized }
|
||||||
: { codexConfigDir: sanitized },
|
: key === "codex"
|
||||||
|
? { codexConfigDir: sanitized }
|
||||||
|
: { geminiConfigDir: sanitized },
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -188,18 +204,24 @@ export function useDirectorySettings({
|
|||||||
|
|
||||||
const updateDirectory = useCallback(
|
const updateDirectory = useCallback(
|
||||||
(app: AppId, value?: string) => {
|
(app: AppId, value?: string) => {
|
||||||
updateDirectoryState(app === "claude" ? "claude" : "codex", value);
|
updateDirectoryState(
|
||||||
|
app === "claude" ? "claude" : app === "codex" ? "codex" : "gemini",
|
||||||
|
value,
|
||||||
|
);
|
||||||
},
|
},
|
||||||
[updateDirectoryState],
|
[updateDirectoryState],
|
||||||
);
|
);
|
||||||
|
|
||||||
const browseDirectory = useCallback(
|
const browseDirectory = useCallback(
|
||||||
async (app: AppId) => {
|
async (app: AppId) => {
|
||||||
const key: DirectoryKey = app === "claude" ? "claude" : "codex";
|
const key: DirectoryKey =
|
||||||
|
app === "claude" ? "claude" : app === "codex" ? "codex" : "gemini";
|
||||||
const currentValue =
|
const currentValue =
|
||||||
key === "claude"
|
key === "claude"
|
||||||
? (settings?.claudeConfigDir ?? resolvedDirs.claude)
|
? (settings?.claudeConfigDir ?? resolvedDirs.claude)
|
||||||
: (settings?.codexConfigDir ?? resolvedDirs.codex);
|
: key === "codex"
|
||||||
|
? (settings?.codexConfigDir ?? resolvedDirs.codex)
|
||||||
|
: (settings?.geminiConfigDir ?? resolvedDirs.gemini);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const picked = await settingsApi.selectConfigDirectory(currentValue);
|
const picked = await settingsApi.selectConfigDirectory(currentValue);
|
||||||
@@ -240,7 +262,8 @@ export function useDirectorySettings({
|
|||||||
|
|
||||||
const resetDirectory = useCallback(
|
const resetDirectory = useCallback(
|
||||||
async (app: AppId) => {
|
async (app: AppId) => {
|
||||||
const key: DirectoryKey = app === "claude" ? "claude" : "codex";
|
const key: DirectoryKey =
|
||||||
|
app === "claude" ? "claude" : app === "codex" ? "codex" : "gemini";
|
||||||
if (!defaultsRef.current[key]) {
|
if (!defaultsRef.current[key]) {
|
||||||
const fallback = await computeDefaultConfigDir(app);
|
const fallback = await computeDefaultConfigDir(app);
|
||||||
if (fallback) {
|
if (fallback) {
|
||||||
@@ -269,13 +292,14 @@ export function useDirectorySettings({
|
|||||||
}, [updateDirectoryState]);
|
}, [updateDirectoryState]);
|
||||||
|
|
||||||
const resetAllDirectories = useCallback(
|
const resetAllDirectories = useCallback(
|
||||||
(claudeDir?: string, codexDir?: string) => {
|
(claudeDir?: string, codexDir?: string, geminiDir?: string) => {
|
||||||
setAppConfigDir(initialAppConfigDirRef.current);
|
setAppConfigDir(initialAppConfigDirRef.current);
|
||||||
setResolvedDirs({
|
setResolvedDirs({
|
||||||
appConfig:
|
appConfig:
|
||||||
initialAppConfigDirRef.current ?? defaultsRef.current.appConfig,
|
initialAppConfigDirRef.current ?? defaultsRef.current.appConfig,
|
||||||
claude: claudeDir ?? defaultsRef.current.claude,
|
claude: claudeDir ?? defaultsRef.current.claude,
|
||||||
codex: codexDir ?? defaultsRef.current.codex,
|
codex: codexDir ?? defaultsRef.current.codex,
|
||||||
|
gemini: geminiDir ?? defaultsRef.current.gemini,
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
[],
|
[],
|
||||||
|
|||||||
@@ -33,7 +33,13 @@ export interface UseSettingsResult {
|
|||||||
browseAppConfigDir: () => Promise<void>;
|
browseAppConfigDir: () => Promise<void>;
|
||||||
resetDirectory: (app: AppId) => Promise<void>;
|
resetDirectory: (app: AppId) => Promise<void>;
|
||||||
resetAppConfigDir: () => Promise<void>;
|
resetAppConfigDir: () => Promise<void>;
|
||||||
saveSettings: () => Promise<SaveResult | null>;
|
saveSettings: (
|
||||||
|
overrides?: Partial<SettingsFormState>,
|
||||||
|
options?: { silent?: boolean },
|
||||||
|
) => Promise<SaveResult | null>;
|
||||||
|
autoSaveSettings: (
|
||||||
|
updates: Partial<SettingsFormState>,
|
||||||
|
) => Promise<SaveResult | null>;
|
||||||
resetSettings: () => void;
|
resetSettings: () => void;
|
||||||
acknowledgeRestart: () => void;
|
acknowledgeRestart: () => void;
|
||||||
}
|
}
|
||||||
@@ -102,6 +108,7 @@ export function useSettings(): UseSettingsResult {
|
|||||||
resetAllDirectories(
|
resetAllDirectories(
|
||||||
sanitizeDir(data?.claudeConfigDir),
|
sanitizeDir(data?.claudeConfigDir),
|
||||||
sanitizeDir(data?.codexConfigDir),
|
sanitizeDir(data?.codexConfigDir),
|
||||||
|
sanitizeDir(data?.geminiConfigDir),
|
||||||
);
|
);
|
||||||
setRequiresRestart(false);
|
setRequiresRestart(false);
|
||||||
}, [
|
}, [
|
||||||
@@ -113,93 +120,220 @@ export function useSettings(): UseSettingsResult {
|
|||||||
setRequiresRestart,
|
setRequiresRestart,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// 保存设置
|
// 即时保存设置(用于 General 标签页的实时更新)
|
||||||
const saveSettings = useCallback(async (): Promise<SaveResult | null> => {
|
// 保存基础配置 + 独立的系统 API 调用(开机自启)
|
||||||
if (!settings) return null;
|
const autoSaveSettings = useCallback(
|
||||||
try {
|
async (updates: Partial<SettingsFormState>): Promise<SaveResult | null> => {
|
||||||
const sanitizedAppDir = sanitizeDir(appConfigDir);
|
const mergedSettings = settings ? { ...settings, ...updates } : null;
|
||||||
const sanitizedClaudeDir = sanitizeDir(settings.claudeConfigDir);
|
if (!mergedSettings) return null;
|
||||||
const sanitizedCodexDir = sanitizeDir(settings.codexConfigDir);
|
|
||||||
const previousAppDir = initialAppConfigDir;
|
|
||||||
const previousClaudeDir = sanitizeDir(data?.claudeConfigDir);
|
|
||||||
const previousCodexDir = sanitizeDir(data?.codexConfigDir);
|
|
||||||
|
|
||||||
const payload: Settings = {
|
|
||||||
...settings,
|
|
||||||
claudeConfigDir: sanitizedClaudeDir,
|
|
||||||
codexConfigDir: sanitizedCodexDir,
|
|
||||||
language: settings.language,
|
|
||||||
};
|
|
||||||
|
|
||||||
await saveMutation.mutateAsync(payload);
|
|
||||||
|
|
||||||
await settingsApi.setAppConfigDirOverride(sanitizedAppDir ?? null);
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (payload.enableClaudePluginIntegration) {
|
const sanitizedClaudeDir = sanitizeDir(mergedSettings.claudeConfigDir);
|
||||||
await settingsApi.applyClaudePluginConfig({ official: false });
|
const sanitizedCodexDir = sanitizeDir(mergedSettings.codexConfigDir);
|
||||||
} else {
|
const sanitizedGeminiDir = sanitizeDir(mergedSettings.geminiConfigDir);
|
||||||
await settingsApi.applyClaudePluginConfig({ official: true });
|
|
||||||
|
const payload: Settings = {
|
||||||
|
...mergedSettings,
|
||||||
|
claudeConfigDir: sanitizedClaudeDir,
|
||||||
|
codexConfigDir: sanitizedCodexDir,
|
||||||
|
geminiConfigDir: sanitizedGeminiDir,
|
||||||
|
language: mergedSettings.language,
|
||||||
|
};
|
||||||
|
|
||||||
|
// 保存到配置文件
|
||||||
|
await saveMutation.mutateAsync(payload);
|
||||||
|
|
||||||
|
// 如果开机自启状态改变,调用系统 API
|
||||||
|
if (
|
||||||
|
payload.launchOnStartup !== undefined &&
|
||||||
|
payload.launchOnStartup !== data?.launchOnStartup
|
||||||
|
) {
|
||||||
|
try {
|
||||||
|
await settingsApi.setAutoLaunch(payload.launchOnStartup);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to update auto-launch:", error);
|
||||||
|
toast.error(
|
||||||
|
t("settings.autoLaunchFailed", {
|
||||||
|
defaultValue: "设置开机自启失败",
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (error) {
|
|
||||||
console.warn(
|
|
||||||
"[useSettings] Failed to sync Claude plugin config",
|
|
||||||
error,
|
|
||||||
);
|
|
||||||
toast.error(
|
|
||||||
t("notifications.syncClaudePluginFailed", {
|
|
||||||
defaultValue: "同步 Claude 插件失败",
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
// 持久化语言偏好
|
||||||
if (typeof window !== "undefined") {
|
try {
|
||||||
window.localStorage.setItem("language", payload.language as Language);
|
if (typeof window !== "undefined" && updates.language) {
|
||||||
}
|
window.localStorage.setItem("language", updates.language);
|
||||||
} catch (error) {
|
}
|
||||||
console.warn(
|
} catch (error) {
|
||||||
"[useSettings] Failed to persist language preference",
|
|
||||||
error,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
await providersApi.updateTrayMenu();
|
|
||||||
} catch (error) {
|
|
||||||
console.warn("[useSettings] Failed to refresh tray menu", error);
|
|
||||||
}
|
|
||||||
|
|
||||||
// 如果 Claude/Codex 的目录覆盖发生变化,则立即将“当前使用的供应商”写回对应应用的 live 配置
|
|
||||||
const claudeDirChanged = sanitizedClaudeDir !== previousClaudeDir;
|
|
||||||
const codexDirChanged = sanitizedCodexDir !== previousCodexDir;
|
|
||||||
if (claudeDirChanged || codexDirChanged) {
|
|
||||||
const syncResult = await syncCurrentProvidersLiveSafe();
|
|
||||||
if (!syncResult.ok) {
|
|
||||||
console.warn(
|
console.warn(
|
||||||
"[useSettings] Failed to sync current providers after directory change",
|
"[useSettings] Failed to persist language preference",
|
||||||
syncResult.error,
|
error,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 更新托盘菜单
|
||||||
|
try {
|
||||||
|
await providersApi.updateTrayMenu();
|
||||||
|
} catch (error) {
|
||||||
|
console.warn("[useSettings] Failed to refresh tray menu", error);
|
||||||
|
}
|
||||||
|
|
||||||
|
return { requiresRestart: false };
|
||||||
|
} catch (error) {
|
||||||
|
console.error("[useSettings] Failed to auto-save settings", error);
|
||||||
|
toast.error(
|
||||||
|
t("notifications.settingsSaveFailed", {
|
||||||
|
defaultValue: "保存设置失败: {{error}}",
|
||||||
|
error: (error as Error)?.message ?? String(error),
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
throw error;
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
[data, saveMutation, settings, t],
|
||||||
|
);
|
||||||
|
|
||||||
const appDirChanged = sanitizedAppDir !== (previousAppDir ?? undefined);
|
// 完整保存设置(用于 Advanced 标签页的手动保存)
|
||||||
setRequiresRestart(appDirChanged);
|
// 包含所有系统 API 调用和完整的验证流程
|
||||||
|
const saveSettings = useCallback(
|
||||||
|
async (
|
||||||
|
overrides?: Partial<SettingsFormState>,
|
||||||
|
options?: { silent?: boolean },
|
||||||
|
): Promise<SaveResult | null> => {
|
||||||
|
const mergedSettings = settings ? { ...settings, ...overrides } : null;
|
||||||
|
if (!mergedSettings) return null;
|
||||||
|
try {
|
||||||
|
const sanitizedAppDir = sanitizeDir(appConfigDir);
|
||||||
|
const sanitizedClaudeDir = sanitizeDir(mergedSettings.claudeConfigDir);
|
||||||
|
const sanitizedCodexDir = sanitizeDir(mergedSettings.codexConfigDir);
|
||||||
|
const sanitizedGeminiDir = sanitizeDir(mergedSettings.geminiConfigDir);
|
||||||
|
const previousAppDir = initialAppConfigDir;
|
||||||
|
const previousClaudeDir = sanitizeDir(data?.claudeConfigDir);
|
||||||
|
const previousCodexDir = sanitizeDir(data?.codexConfigDir);
|
||||||
|
const previousGeminiDir = sanitizeDir(data?.geminiConfigDir);
|
||||||
|
|
||||||
return { requiresRestart: appDirChanged };
|
const payload: Settings = {
|
||||||
} catch (error) {
|
...mergedSettings,
|
||||||
console.error("[useSettings] Failed to save settings", error);
|
claudeConfigDir: sanitizedClaudeDir,
|
||||||
throw error;
|
codexConfigDir: sanitizedCodexDir,
|
||||||
}
|
geminiConfigDir: sanitizedGeminiDir,
|
||||||
}, [
|
language: mergedSettings.language,
|
||||||
appConfigDir,
|
};
|
||||||
data,
|
|
||||||
initialAppConfigDir,
|
await saveMutation.mutateAsync(payload);
|
||||||
saveMutation,
|
|
||||||
settings,
|
await settingsApi.setAppConfigDirOverride(sanitizedAppDir ?? null);
|
||||||
setRequiresRestart,
|
|
||||||
t,
|
// 只在开机自启状态真正改变时调用系统 API
|
||||||
]);
|
if (
|
||||||
|
payload.launchOnStartup !== undefined &&
|
||||||
|
payload.launchOnStartup !== data?.launchOnStartup
|
||||||
|
) {
|
||||||
|
try {
|
||||||
|
await settingsApi.setAutoLaunch(payload.launchOnStartup);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to update auto-launch:", error);
|
||||||
|
toast.error(
|
||||||
|
t("settings.autoLaunchFailed", {
|
||||||
|
defaultValue: "设置开机自启失败",
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 只在 Claude 插件集成状态真正改变时调用系统 API
|
||||||
|
if (
|
||||||
|
payload.enableClaudePluginIntegration !== undefined &&
|
||||||
|
payload.enableClaudePluginIntegration !==
|
||||||
|
data?.enableClaudePluginIntegration
|
||||||
|
) {
|
||||||
|
try {
|
||||||
|
if (payload.enableClaudePluginIntegration) {
|
||||||
|
await settingsApi.applyClaudePluginConfig({ official: false });
|
||||||
|
} else {
|
||||||
|
await settingsApi.applyClaudePluginConfig({ official: true });
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.warn(
|
||||||
|
"[useSettings] Failed to sync Claude plugin config",
|
||||||
|
error,
|
||||||
|
);
|
||||||
|
toast.error(
|
||||||
|
t("notifications.syncClaudePluginFailed", {
|
||||||
|
defaultValue: "同步 Claude 插件失败",
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (typeof window !== "undefined") {
|
||||||
|
window.localStorage.setItem(
|
||||||
|
"language",
|
||||||
|
payload.language as Language,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.warn(
|
||||||
|
"[useSettings] Failed to persist language preference",
|
||||||
|
error,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await providersApi.updateTrayMenu();
|
||||||
|
} catch (error) {
|
||||||
|
console.warn("[useSettings] Failed to refresh tray menu", error);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 如果 Claude/Codex/Gemini 的目录覆盖发生变化,则立即将“当前使用的供应商”写回对应应用的 live 配置
|
||||||
|
const claudeDirChanged = sanitizedClaudeDir !== previousClaudeDir;
|
||||||
|
const codexDirChanged = sanitizedCodexDir !== previousCodexDir;
|
||||||
|
const geminiDirChanged = sanitizedGeminiDir !== previousGeminiDir;
|
||||||
|
if (claudeDirChanged || codexDirChanged || geminiDirChanged) {
|
||||||
|
const syncResult = await syncCurrentProvidersLiveSafe();
|
||||||
|
if (!syncResult.ok) {
|
||||||
|
console.warn(
|
||||||
|
"[useSettings] Failed to sync current providers after directory change",
|
||||||
|
syncResult.error,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const appDirChanged = sanitizedAppDir !== (previousAppDir ?? undefined);
|
||||||
|
setRequiresRestart(appDirChanged);
|
||||||
|
|
||||||
|
if (!options?.silent) {
|
||||||
|
toast.success(
|
||||||
|
t("notifications.settingsSaved", {
|
||||||
|
defaultValue: "设置已保存",
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return { requiresRestart: appDirChanged };
|
||||||
|
} catch (error) {
|
||||||
|
console.error("[useSettings] Failed to save settings", error);
|
||||||
|
toast.error(
|
||||||
|
t("notifications.settingsSaveFailed", {
|
||||||
|
defaultValue: "保存设置失败: {{error}}",
|
||||||
|
error: (error as Error)?.message ?? String(error),
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[
|
||||||
|
appConfigDir,
|
||||||
|
data,
|
||||||
|
initialAppConfigDir,
|
||||||
|
saveMutation,
|
||||||
|
settings,
|
||||||
|
setRequiresRestart,
|
||||||
|
t,
|
||||||
|
],
|
||||||
|
);
|
||||||
|
|
||||||
const isLoading = useMemo(
|
const isLoading = useMemo(
|
||||||
() => isFormLoading || isDirectoryLoading || isMetadataLoading,
|
() => isFormLoading || isDirectoryLoading || isMetadataLoading,
|
||||||
@@ -222,6 +356,7 @@ export function useSettings(): UseSettingsResult {
|
|||||||
resetDirectory,
|
resetDirectory,
|
||||||
resetAppConfigDir,
|
resetAppConfigDir,
|
||||||
saveSettings,
|
saveSettings,
|
||||||
|
autoSaveSettings,
|
||||||
resetSettings,
|
resetSettings,
|
||||||
acknowledgeRestart,
|
acknowledgeRestart,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"app": {
|
"app": {
|
||||||
"title": "CC Switch",
|
"title": "CC Switch",
|
||||||
"description": "Claude Code & Codex Provider Switching Tool"
|
"description": "All-in-One Assistant for Claude Code, Codex & Gemini CLI"
|
||||||
},
|
},
|
||||||
"common": {
|
"common": {
|
||||||
"add": "Add",
|
"add": "Add",
|
||||||
@@ -27,7 +27,8 @@
|
|||||||
"formatSuccess": "Formatted successfully",
|
"formatSuccess": "Formatted successfully",
|
||||||
"formatError": "Format failed: {{error}}",
|
"formatError": "Format failed: {{error}}",
|
||||||
"copy": "Copy",
|
"copy": "Copy",
|
||||||
"view": "View"
|
"view": "View",
|
||||||
|
"back": "Back"
|
||||||
},
|
},
|
||||||
"apiKeyInput": {
|
"apiKeyInput": {
|
||||||
"placeholder": "Enter API Key",
|
"placeholder": "Enter API Key",
|
||||||
@@ -166,6 +167,9 @@
|
|||||||
"languageOptionEnglish": "English",
|
"languageOptionEnglish": "English",
|
||||||
"windowBehavior": "Window Behavior",
|
"windowBehavior": "Window Behavior",
|
||||||
"windowBehaviorHint": "Configure window minimize and Claude plugin integration policies.",
|
"windowBehaviorHint": "Configure window minimize and Claude plugin integration policies.",
|
||||||
|
"launchOnStartup": "Launch on Startup",
|
||||||
|
"launchOnStartupDescription": "Automatically run CC Switch when system starts",
|
||||||
|
"autoLaunchFailed": "Failed to set auto-launch",
|
||||||
"minimizeToTray": "Minimize to tray on close",
|
"minimizeToTray": "Minimize to tray on close",
|
||||||
"minimizeToTrayDescription": "When checked, clicking the close button will hide to system tray, otherwise the app will exit directly.",
|
"minimizeToTrayDescription": "When checked, clicking the close button will hide to system tray, otherwise the app will exit directly.",
|
||||||
"enableClaudePluginIntegration": "Apply to Claude Code extension",
|
"enableClaudePluginIntegration": "Apply to Claude Code extension",
|
||||||
@@ -179,8 +183,11 @@
|
|||||||
"claudeConfigDirDescription": "Override Claude configuration directory (settings.json) and keep claude.json (MCP) alongside it.",
|
"claudeConfigDirDescription": "Override Claude configuration directory (settings.json) and keep claude.json (MCP) alongside it.",
|
||||||
"codexConfigDir": "Codex Configuration Directory",
|
"codexConfigDir": "Codex Configuration Directory",
|
||||||
"codexConfigDirDescription": "Override Codex configuration directory.",
|
"codexConfigDirDescription": "Override Codex configuration directory.",
|
||||||
|
"geminiConfigDir": "Gemini Configuration Directory",
|
||||||
|
"geminiConfigDirDescription": "Override Gemini configuration directory (.env).",
|
||||||
"browsePlaceholderClaude": "e.g., /home/<your-username>/.claude",
|
"browsePlaceholderClaude": "e.g., /home/<your-username>/.claude",
|
||||||
"browsePlaceholderCodex": "e.g., /home/<your-username>/.codex",
|
"browsePlaceholderCodex": "e.g., /home/<your-username>/.codex",
|
||||||
|
"browsePlaceholderGemini": "e.g., /home/<your-username>/.gemini",
|
||||||
"browseDirectory": "Browse Directory",
|
"browseDirectory": "Browse Directory",
|
||||||
"resetDefault": "Reset to default directory (takes effect after saving)",
|
"resetDefault": "Reset to default directory (takes effect after saving)",
|
||||||
"checkForUpdates": "Check for Updates",
|
"checkForUpdates": "Check for Updates",
|
||||||
@@ -311,7 +318,8 @@
|
|||||||
"pleaseAddEndpoint": "Please add an endpoint first",
|
"pleaseAddEndpoint": "Please add an endpoint first",
|
||||||
"testUnavailable": "Speed test unavailable",
|
"testUnavailable": "Speed test unavailable",
|
||||||
"noResult": "No result returned",
|
"noResult": "No result returned",
|
||||||
"testFailed": "Speed test failed: {{error}}"
|
"testFailed": "Speed test failed: {{error}}",
|
||||||
|
"status": "Status: {{code}}"
|
||||||
},
|
},
|
||||||
"codexConfig": {
|
"codexConfig": {
|
||||||
"authJson": "auth.json (JSON) *",
|
"authJson": "auth.json (JSON) *",
|
||||||
@@ -358,6 +366,9 @@
|
|||||||
"title": "Configure Usage Query",
|
"title": "Configure Usage Query",
|
||||||
"enableUsageQuery": "Enable usage query",
|
"enableUsageQuery": "Enable usage query",
|
||||||
"presetTemplate": "Preset template",
|
"presetTemplate": "Preset template",
|
||||||
|
"requestUrl": "Request URL",
|
||||||
|
"requestUrlPlaceholder": "e.g. https://api.example.com",
|
||||||
|
"method": "HTTP method",
|
||||||
"templateCustom": "Custom",
|
"templateCustom": "Custom",
|
||||||
"templateGeneral": "General",
|
"templateGeneral": "General",
|
||||||
"templateNewAPI": "NewAPI",
|
"templateNewAPI": "NewAPI",
|
||||||
@@ -370,11 +381,14 @@
|
|||||||
"queryFailedMessage": "Query failed",
|
"queryFailedMessage": "Query failed",
|
||||||
"queryScript": "Query script (JavaScript)",
|
"queryScript": "Query script (JavaScript)",
|
||||||
"timeoutSeconds": "Timeout (seconds)",
|
"timeoutSeconds": "Timeout (seconds)",
|
||||||
|
"headers": "Headers",
|
||||||
|
"body": "Body",
|
||||||
"timeoutHint": "Range: 2-30 seconds",
|
"timeoutHint": "Range: 2-30 seconds",
|
||||||
"timeoutMustBeInteger": "Timeout must be an integer, decimal part ignored",
|
"timeoutMustBeInteger": "Timeout must be an integer, decimal part ignored",
|
||||||
"timeoutCannotBeNegative": "Timeout cannot be negative",
|
"timeoutCannotBeNegative": "Timeout cannot be negative",
|
||||||
|
"autoIntervalMinutes": "Auto query interval (minutes)",
|
||||||
"autoQueryInterval": "Auto Query Interval (minutes)",
|
"autoQueryInterval": "Auto Query Interval (minutes)",
|
||||||
"autoQueryIntervalHint": "0 to disable, recommend 5-60 minutes",
|
"autoQueryIntervalHint": "0 to disable; recommend 5-60 minutes",
|
||||||
"intervalMustBeInteger": "Interval must be an integer, decimal part ignored",
|
"intervalMustBeInteger": "Interval must be an integer, decimal part ignored",
|
||||||
"intervalCannotBeNegative": "Interval cannot be negative",
|
"intervalCannotBeNegative": "Interval cannot be negative",
|
||||||
"intervalAdjusted": "Interval adjusted to {{value}} minutes",
|
"intervalAdjusted": "Interval adjusted to {{value}} minutes",
|
||||||
@@ -395,6 +409,9 @@
|
|||||||
"formatSuccess": "Format successful",
|
"formatSuccess": "Format successful",
|
||||||
"formatFailed": "Format failed",
|
"formatFailed": "Format failed",
|
||||||
"variablesHint": "Supported variables: {{apiKey}}, {{baseUrl}} | extractor function receives API response JSON object",
|
"variablesHint": "Supported variables: {{apiKey}}, {{baseUrl}} | extractor function receives API response JSON object",
|
||||||
|
"scriptConfig": "Request configuration",
|
||||||
|
"extractorCode": "Extractor code",
|
||||||
|
"extractorHint": "Return object should include remaining quota fields",
|
||||||
"fieldIsValid": "• isValid: Boolean, whether plan is valid",
|
"fieldIsValid": "• isValid: Boolean, whether plan is valid",
|
||||||
"fieldInvalidMessage": "• invalidMessage: String, reason for expiration (shown when isValid is false)",
|
"fieldInvalidMessage": "• invalidMessage: String, reason for expiration (shown when isValid is false)",
|
||||||
"fieldRemaining": "• remaining: Number, remaining quota",
|
"fieldRemaining": "• remaining: Number, remaining quota",
|
||||||
@@ -672,6 +689,34 @@
|
|||||||
"installFailed": "Failed to install",
|
"installFailed": "Failed to install",
|
||||||
"uninstallSuccess": "Skill {{name}} uninstalled",
|
"uninstallSuccess": "Skill {{name}} uninstalled",
|
||||||
"uninstallFailed": "Failed to uninstall",
|
"uninstallFailed": "Failed to uninstall",
|
||||||
|
"error": {
|
||||||
|
"skillNotFound": "Skill not found: {{directory}}",
|
||||||
|
"missingRepoInfo": "Missing repository info (owner or name)",
|
||||||
|
"downloadTimeout": "Download repository {{owner}}/{{name}} timeout ({{timeout}}s)",
|
||||||
|
"downloadTimeoutHint": "Please check network connection or retry later",
|
||||||
|
"skillPathNotFound": "Skill path '{{path}}' not found in repository {{owner}}/{{name}}",
|
||||||
|
"skillDirNotFound": "Skill directory not found: {{path}}",
|
||||||
|
"emptyArchive": "Downloaded archive is empty",
|
||||||
|
"downloadFailed": "Download failed: HTTP {{status}}",
|
||||||
|
"allBranchesFailed": "All branches failed, tried: {{branches}}",
|
||||||
|
"httpError": "HTTP error {{status}}",
|
||||||
|
"http403": "GitHub access restricted, possibly rate limited",
|
||||||
|
"http404": "Repository or branch not found, please check URL",
|
||||||
|
"http429": "Too many requests, please wait and retry",
|
||||||
|
"parseMetadataFailed": "Failed to parse skill metadata",
|
||||||
|
"getHomeDirFailed": "Unable to get user home directory",
|
||||||
|
"networkError": "Network error",
|
||||||
|
"fsError": "File system error",
|
||||||
|
"unknownError": "Unknown error",
|
||||||
|
"suggestion": {
|
||||||
|
"checkNetwork": "Please check network connection",
|
||||||
|
"checkProxy": "Consider configuring HTTP proxy",
|
||||||
|
"retryLater": "Please retry later",
|
||||||
|
"checkRepoUrl": "Please check repository URL and branch name",
|
||||||
|
"checkDiskSpace": "Please check disk space",
|
||||||
|
"checkPermission": "Please check directory permissions"
|
||||||
|
}
|
||||||
|
},
|
||||||
"repo": {
|
"repo": {
|
||||||
"title": "Manage Skill Repositories",
|
"title": "Manage Skill Repositories",
|
||||||
"description": "Add or remove GitHub skill repository sources",
|
"description": "Add or remove GitHub skill repository sources",
|
||||||
@@ -690,7 +735,10 @@
|
|||||||
"removeSuccess": "Repository {{owner}}/{{name}} removed",
|
"removeSuccess": "Repository {{owner}}/{{name}} removed",
|
||||||
"removeFailed": "Failed to remove",
|
"removeFailed": "Failed to remove",
|
||||||
"skillCount": "{{count}} skills detected"
|
"skillCount": "{{count}} skills detected"
|
||||||
}
|
},
|
||||||
|
"search": "Search Skills",
|
||||||
|
"searchPlaceholder": "Search skill name or description...",
|
||||||
|
"noResults": "No matching skills found"
|
||||||
},
|
},
|
||||||
"deeplink": {
|
"deeplink": {
|
||||||
"confirmImport": "Confirm Import Provider",
|
"confirmImport": "Confirm Import Provider",
|
||||||
@@ -708,6 +756,29 @@
|
|||||||
"parseError": "Failed to parse deep link",
|
"parseError": "Failed to parse deep link",
|
||||||
"importSuccess": "Import successful",
|
"importSuccess": "Import successful",
|
||||||
"importSuccessDescription": "Provider \"{{name}}\" has been successfully imported",
|
"importSuccessDescription": "Provider \"{{name}}\" has been successfully imported",
|
||||||
"importError": "Failed to import"
|
"importError": "Failed to import",
|
||||||
|
"configSource": "Config Source",
|
||||||
|
"configEmbedded": "Embedded Config",
|
||||||
|
"configRemote": "Remote Config",
|
||||||
|
"configDetails": "Config Details",
|
||||||
|
"configUrl": "Config File URL",
|
||||||
|
"configMergeError": "Failed to merge configuration file"
|
||||||
|
},
|
||||||
|
"iconPicker": {
|
||||||
|
"search": "Search Icons",
|
||||||
|
"searchPlaceholder": "Enter icon name...",
|
||||||
|
"noResults": "No matching icons found",
|
||||||
|
"category": {
|
||||||
|
"aiProvider": "AI Providers",
|
||||||
|
"cloud": "Cloud Platforms",
|
||||||
|
"tool": "Dev Tools",
|
||||||
|
"other": "Other"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"providerIcon": {
|
||||||
|
"label": "Icon",
|
||||||
|
"colorLabel": "Icon Color",
|
||||||
|
"selectIcon": "Select Icon",
|
||||||
|
"preview": "Preview"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user