Compare commits

...

2 Commits

Author SHA1 Message Date
Memory
e4110f65c2
fix: support multiple hosts values and IPv6 addresses in DNS config 2026-04-01 21:58:33 +08:00
YangChengxxyy
93d552870e
feat: 新增全局快捷键 复制命令行命令
Co-authored-by: YangChengxxyy <yangc27@trip.com>
2026-04-01 19:11:45 +08:00
9 changed files with 52 additions and 13 deletions

View File

@ -11,7 +11,7 @@ import { patchMihomoConfig } from '../core/mihomoApi'
import { quitWithoutCore, restartCore } from '../core/manager'
import i18next from '../../shared/i18n'
import { floatingWindow, triggerFloatingWindow } from './floatingWindow'
import { updateTrayIcon } from './tray'
import { copyEnv, updateTrayIcon } from './tray'
export async function registerShortcut(
oldShortcut: string,
@ -133,6 +133,15 @@ export async function registerShortcut(
app.quit()
})
}
case 'copyEnvShortcut': {
return globalShortcut.register(newShortcut, async () => {
const shellType = process.platform === 'win32' ? 'powershell' : 'bash'
await copyEnv(shellType)
new Notification({
title: i18next.t('common.notification.copyEnvSuccess')
}).show()
})
}
}
throw new Error('Unknown action')
}
@ -147,7 +156,8 @@ export async function initShortcut(): Promise<void> {
globalModeShortcut,
directModeShortcut,
quitWithoutCoreShortcut,
restartAppShortcut
restartAppShortcut,
copyEnvShortcut
} = await getAppConfig()
if (showWindowShortcut) {
try {
@ -212,4 +222,11 @@ export async function initShortcut(): Promise<void> {
// ignore
}
}
if (copyEnvShortcut) {
try {
await registerShortcut('', copyEnvShortcut, 'copyEnvShortcut')
} catch {
// ignore
}
}
}

View File

@ -53,7 +53,8 @@ const ShortcutConfig: React.FC = () => {
globalModeShortcut = '',
directModeShortcut = '',
quitWithoutCoreShortcut = '',
restartAppShortcut = ''
restartAppShortcut = '',
copyEnvShortcut = ''
} = appConfig || {}
return (
@ -130,7 +131,7 @@ const ShortcutConfig: React.FC = () => {
/>
</div>
</SettingItem>
<SettingItem title={t('shortcuts.restartApp')}>
<SettingItem title={t('shortcuts.restartApp')} divider>
<div className="flex justify-end w-[60%]">
<ShortcutInput
value={restartAppShortcut}
@ -139,6 +140,15 @@ const ShortcutConfig: React.FC = () => {
/>
</div>
</SettingItem>
<SettingItem title={t('shortcuts.copyEnv')}>
<div className="flex justify-end w-[60%]">
<ShortcutInput
value={copyEnvShortcut}
patchAppConfig={patchAppConfig}
action="copyEnvShortcut"
/>
</div>
</SettingItem>
</SettingCard>
)
}

View File

@ -24,6 +24,7 @@
"common.notification.restartRequired": "Restart required for changes to take effect",
"common.notification.systemProxyEnabled": "System proxy enabled",
"common.notification.systemProxyDisabled": "System proxy disabled",
"common.notification.copyEnvSuccess": "Shell env command copied",
"common.notification.tunEnabled": "TUN enabled",
"common.notification.tunDisabled": "TUN disabled",
"common.notification.ruleMode": "Rule Mode",
@ -294,6 +295,7 @@
"shortcuts.toggleDirectMode": "Toggle Direct Mode",
"shortcuts.toggleLightMode": "Toggle Light Mode",
"shortcuts.restartApp": "Restart App",
"shortcuts.copyEnv": "Copy Shell Env Command",
"shortcuts.input.placeholder": "Click to input shortcut",
"sider.title": "Sidebar Settings",
"sider.cards.systemProxy": "Sys Proxy",
@ -471,7 +473,7 @@
"dns.customHosts.title": "Custom Hosts",
"dns.customHosts.list": "Hosts List",
"dns.customHosts.domainPlaceholder": "Domain",
"dns.customHosts.valuePlaceholder": "Domain or IP",
"dns.customHosts.valuePlaceholder": "Domain or IP, separate multiple with commas",
"dns.saveOnly": "Save Only",
"profiles.title": "Profile Management",
"profiles.input.placeholder": "Enter your subscription URL",

View File

@ -24,6 +24,7 @@
"common.notification.restartRequired": "برای اعمال تغییرات نیاز به راه‌اندازی مجدد است",
"common.notification.systemProxyEnabled": "پراکسی سیستم فعال شد",
"common.notification.systemProxyDisabled": "پراکسی سیستم غیرفعال شد",
"common.notification.copyEnvSuccess": "دستور محیطی کپی شد",
"common.notification.tunEnabled": "TUN فعال شد",
"common.notification.tunDisabled": "TUN غیرفعال شد",
"common.notification.ruleMode": "حالت قانون",
@ -269,6 +270,7 @@
"shortcuts.toggleDirectMode": "تغییر وضعیت حالت مستقیم",
"shortcuts.toggleLightMode": "تغییر وضعیت حالت روشن",
"shortcuts.restartApp": "راه‌اندازی مجدد برنامه",
"shortcuts.copyEnv": "کپی دستور محیطی",
"shortcuts.input.placeholder": "برای ورود میانبر کلیک کنید",
"sider.title": "تنظیمات نوار کناری",
"sider.cards.systemProxy": "پراکسی سیستم",
@ -443,7 +445,7 @@
"dns.customHosts.title": "Hosts سفارشی",
"dns.customHosts.list": "لیست Hosts",
"dns.customHosts.domainPlaceholder": "دامنه",
"dns.customHosts.valuePlaceholder": "دامنه یا IP",
"dns.customHosts.valuePlaceholder": "دامنه یا IP، چندین مورد با کاما جدا کنید",
"dns.saveOnly": "فقط ذخیره",
"profiles.title": "مدیریت پروفایل",
"profiles.input.placeholder": "آدرس اشتراک خود را وارد کنید",

View File

@ -24,6 +24,7 @@
"common.notification.restartRequired": "Требуется перезапуск для применения изменений",
"common.notification.systemProxyEnabled": "Системный прокси включен",
"common.notification.systemProxyDisabled": "Системный прокси отключен",
"common.notification.copyEnvSuccess": "Команда env скопирована",
"common.notification.tunEnabled": "TUN включен",
"common.notification.tunDisabled": "TUN отключен",
"common.notification.ruleMode": "Правило",
@ -271,6 +272,7 @@
"shortcuts.toggleDirectMode": "Переключить прямое подключение",
"shortcuts.toggleLightMode": "Переключить облегченный режим",
"shortcuts.restartApp": "Перезапустить приложение",
"shortcuts.copyEnv": "Копировать команду env",
"shortcuts.input.placeholder": "Нажмите для ввода комбинации",
"sider.title": "Настройки боковой панели",
"sider.cards.systemProxy": "Системный прокси",
@ -445,7 +447,7 @@
"dns.customHosts.title": "Пользовательский Hosts",
"dns.customHosts.list": "Список Hosts",
"dns.customHosts.domainPlaceholder": "Домен",
"dns.customHosts.valuePlaceholder": "Домен или IP",
"dns.customHosts.valuePlaceholder": "Домен или IP, несколько через запятую",
"dns.saveOnly": "Только сохранить",
"profiles.title": "Управление профилями",
"profiles.input.placeholder": "Введите URL вашей подписки",

View File

@ -24,6 +24,7 @@
"common.notification.restartRequired": "需要重启应用以使更改生效",
"common.notification.systemProxyEnabled": "系统代理已启用",
"common.notification.systemProxyDisabled": "系统代理已关闭",
"common.notification.copyEnvSuccess": "命令行命令已复制",
"common.notification.tunEnabled": "TUN 已启用",
"common.notification.tunDisabled": "TUN 已关闭",
"common.notification.ruleMode": "规则模式",
@ -294,6 +295,7 @@
"shortcuts.toggleDirectMode": "切换直连模式",
"shortcuts.toggleLightMode": "切换轻量模式",
"shortcuts.restartApp": "重启应用",
"shortcuts.copyEnv": "复制命令行命令",
"shortcuts.input.placeholder": "点击输入快捷键",
"sider.title": "侧边栏设置",
"sider.cards.systemProxy": "系统代理",
@ -462,7 +464,7 @@
"dns.customHosts.title": "自定义 Hosts",
"dns.customHosts.list": "Hosts 列表",
"dns.customHosts.domainPlaceholder": "域名",
"dns.customHosts.valuePlaceholder": "域名或 IP",
"dns.customHosts.valuePlaceholder": "域名或 IP,多个用逗号分隔",
"dns.fallback": "回退服务器",
"dns.fallbackPlaceholder": "例https://dns.alidns.com/dns-query",
"dns.fallbackFilter.title": "回退过滤设置",

View File

@ -24,6 +24,7 @@
"common.notification.restartRequired": "需要重新啟動應用以使更改生效",
"common.notification.systemProxyEnabled": "系統代理已啟用",
"common.notification.systemProxyDisabled": "系統代理已關閉",
"common.notification.copyEnvSuccess": "命令行命令已複製",
"common.notification.tunEnabled": "TUN 已啟用",
"common.notification.tunDisabled": "TUN 已關閉",
"common.notification.ruleMode": "規則模式",
@ -294,6 +295,7 @@
"shortcuts.toggleDirectMode": "切換直連模式",
"shortcuts.toggleLightMode": "切換輕量模式",
"shortcuts.restartApp": "重啟應用",
"shortcuts.copyEnv": "複製命令行命令",
"shortcuts.input.placeholder": "點擊輸入快捷鍵",
"sider.title": "側邊欄設置",
"sider.cards.systemProxy": "系統代理",
@ -462,7 +464,7 @@
"dns.customHosts.title": "自定義 Hosts",
"dns.customHosts.list": "Hosts 列表",
"dns.customHosts.domainPlaceholder": "域名",
"dns.customHosts.valuePlaceholder": "域名或 IP",
"dns.customHosts.valuePlaceholder": "域名或 IP,多個用逗號分隔",
"dns.fallback": "回退伺服器",
"dns.fallbackPlaceholder": "例https://dns.alidns.com/dns-query",
"dns.fallbackFilter.title": "回退過濾設置",

View File

@ -123,10 +123,11 @@ const DNS: React.FC = () => {
const handleSubkeyChange = (type: string, domain: string, value: string, index: number): void => {
const list = [...values[type]]
const processedValue = value.includes(',')
? value.split(',').map((s: string) => s.trim())
: value.trim()
if (domain || processedValue) list[index] = { domain: domain.trim(), value: processedValue }
const parts = value.split(',').map((s: string) => s.trim()).filter(Boolean)
const processedValue = type === 'hosts'
? parts
: (parts.length > 1 ? parts : value.trim())
if (domain || parts.length > 0) list[index] = { domain: domain.trim(), value: processedValue }
else list.splice(index, 1)
setValues({ ...values, [type]: list })
}

View File

@ -338,6 +338,7 @@ interface IAppConfig {
directModeShortcut?: string
restartAppShortcut?: string
quitWithoutCoreShortcut?: string
copyEnvShortcut?: string
language?: 'zh-CN' | 'zh-TW' | 'en-US' | 'ru-RU' | 'fa-IR'
triggerMainWindowBehavior?: 'show' | 'toggle'
showMixedPort?: number