Compare commits

...

4 Commits

Author SHA1 Message Date
Memory
a7de9b2588
refactor: Redesign and migrate WebUI to global settings 2025-10-03 22:49:41 +08:00
ezequielnick
f34cc976b4 style: update macos dock icon 2025-10-03 22:43:05 +08:00
ezequielnick
814112f541 style: update logo & changelog 2025-10-03 21:01:41 +08:00
Jiawen Geng
717239f56b
feat: add placeholder for subscription URL in profile management across multiple languages (#1244) 2025-10-03 18:48:32 +08:00
27 changed files with 422 additions and 129 deletions

Binary file not shown.

Before

Width:  |  Height:  |  Size: 20 KiB

After

Width:  |  Height:  |  Size: 19 KiB

Binary file not shown.

Binary file not shown.

Before

Width:  |  Height:  |  Size: 211 KiB

After

Width:  |  Height:  |  Size: 76 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 128 KiB

After

Width:  |  Height:  |  Size: 19 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 229 KiB

After

Width:  |  Height:  |  Size: 204 KiB

View File

@ -1,8 +1,31 @@
## 1.8.8
### 新功能 (Feat)
- 增加内核版本选择
- 记住日志页面的筛选关键字
- Webdav增加Cron定时备份
- 连接卡片纯数字显示样式
- 支持修改窗口触发行为
### 修复 (Fix)
- MacOS 首次启动时的 ENOENT: no such file or directory(config.yaml)
- 自动更新获取老的文件名称
- 修复 mihomo.yaml 文件缺失的问题
- Smart 配置文件验证出错的问题
- 开发环境的 electron 问题
### 优化 (Optimize)
- 加快以管理员模式重启速度
- 优化仅用户滚动滚轮时触发自动滚动
- 改进俄语翻译
- 使用重载替换不必要的重启
# 其他 (chore)
- 更新依赖
### 样式调整 (Sytle)
- 改进 logo 设计
- 卡片尺寸
## 1.8.7

Binary file not shown.

Before

Width:  |  Height:  |  Size: 44 KiB

After

Width:  |  Height:  |  Size: 38 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 42 KiB

After

Width:  |  Height:  |  Size: 37 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 200 KiB

After

Width:  |  Height:  |  Size: 76 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 41 KiB

After

Width:  |  Height:  |  Size: 19 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.9 KiB

After

Width:  |  Height:  |  Size: 1.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 196 KiB

After

Width:  |  Height:  |  Size: 76 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 40 KiB

After

Width:  |  Height:  |  Size: 20 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 194 KiB

After

Width:  |  Height:  |  Size: 75 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 40 KiB

After

Width:  |  Height:  |  Size: 20 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 146 KiB

After

Width:  |  Height:  |  Size: 75 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 40 KiB

After

Width:  |  Height:  |  Size: 20 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.8 KiB

After

Width:  |  Height:  |  Size: 3.2 KiB

View File

@ -4,12 +4,12 @@ import { GenIcon } from 'react-icons'
function MihomoIcon(props: any): React.JSX.Element {
return GenIcon({
tag: 'svg',
attr: { viewBox: '0 0 120 150' },
attr: { viewBox: '0 0 58 61.53' },
child: [
{
tag: 'path',
attr: {
d: 'M123,105.41c0,15.99-13.44,20.46-20.54,20.6h-29.1c5.33-1.74,8.97-5.24,9.18-5.44.05-.05.28-.27.55-.56h19.31c1.44-.04,14.6-.78,14.6-14.6,0-9.22-3.85-21.88-34.64-29.67l-1.47-6.53c28.67,6.6,42.11,18.22,42.11,36.2Z',
d: 'M57.99,14.06c-.26-4.98-4.34-9.05-9.31-9.31-5.69-.3-10.39,4.22-10.39,9.84h0c0,.39-.24.73-.6.86l-1.08.38c-.49.17-.77.68-.65,1.18l.2.9c.13.6.76.95,1.34.74l.7-.25c.42-.15.88.04,1.07.45,1.59,3.31,4.96,5.59,8.88,5.59,5.62,0,10.14-4.7,9.84-10.39Z',
fill: 'currentColor'
},
child: []
@ -17,27 +17,7 @@ function MihomoIcon(props: any): React.JSX.Element {
{
tag: 'path',
attr: {
d: 'M81.1,37.6c-2.42.76-4.68,1.55-6.78,2.37l-1.33-5.92c2.05-.77,4.21-1.51,6.47-2.21.27,2.03.83,3.96,1.64,5.76Z',
fill: 'currentColor'
},
child: []
},
{
tag: 'circle',
attr: {
cx: '53.83',
cy: '33.01',
r: '6',
fill: 'currentColor'
},
child: []
},
{
tag: 'circle',
attr: {
cx: '33.83',
cy: '33.01',
r: '6',
d: 'M40.49,33.96l-1.01-.08.84,3.69h.03s.14,0,.14,0c6.11,0,12.43,4.21,12.43,11.26,0,5.02-3.54,9.1-7.89,9.1h-3.81c-.33.57-.71,1.13-1.13,1.66-.58.72-1.22,1.37-1.91,1.95h6.85c6.35,0,11.49-5.69,11.49-12.71h0c0-9-7.9-14.87-16.04-14.87Z',
fill: 'currentColor'
},
child: []
@ -45,23 +25,7 @@ function MihomoIcon(props: any): React.JSX.Element {
{
tag: 'path',
attr: {
d: 'M67.53,126.01h-1.49c-1.83,0-3.4-1.54-3.19-3.36.17-1.52,1.4-2.64,2.98-2.64h1.7c4.3,0,8.3-1.9,11-5.3,2.7-3.3,3.6-7.6,2.7-11.8L60.13,8.91l-.4.4c-4.3,4.4-9.9,6.7-15.9,6.7s-11.6-2.3-15.8-6.6l-.5-.4L6.33,103.01c-.9,4.1,0,8.4,2.7,11.8s6.7,5.3,11,5.3h.59c1.83,0,3.4,1.54,3.19,3.36-.17,1.52-1.4,2.64-2.98,2.64h-.8c-6.1,0-11.8-2.7-15.7-7.5-3.8-4.9-5.2-11-3.8-17L22.93,2.31c.2-1.1,1-1.9,2.1-2.2,1-.3,2.2,0,3,.8l4.3,4.3c3,3.1,7.1,4.8,11.5,4.8s8.5-1.7,11.6-4.8l4.24-4.24c.37-.37.81-.68,1.31-.83,1.95-.58,3.56.68,3.86,2.16l22.3,99.3c1.3,5.9-.1,12.1-3.9,16.8-.4.5-1.4,1.5-1.4,1.5,0,0-6.26,6.1-14.3,6.1h0Z',
fill: 'currentColor'
},
child: []
},
{
tag: 'path',
attr: {
d: 'M22.23,70.11c1.6-.3,3.2.7,3.5,2.4l9,45.1c.3,1.5,1.5,2.5,3,2.5,1.7,0,3-1.3,3-3v-34.1c0-1.7,1.3-3,3-3s3,1.3,3,3v34c0,1.6,1.4,3,3.1,3h1c1.5,0,2.7-1,3-2.5l9.1-45.1c.3-1.6,1.9-2.7,3.5-2.3,1.6.3,2.7,1.9,2.3,3.5l-9.1,45c-.7,4.3-4.5,7.4-8.9,7.4h-1c-2.3,0-4.5-.9-6.1-2.3-1.6,1.4-3.7,2.3-6,2.3-4.3,0-7.9-3-8.8-7.3l-9-45.1c-.3-1.6.8-3.2,2.4-3.5Z',
fill: 'currentColor'
},
child: []
},
{
tag: 'path',
attr: {
d: 'M100.75,7.39c-11.85,0-21.5,9.64-21.5,21.5,0,1,.07,1.99.21,2.95.27,2.03.83,3.96,1.64,5.76,3.35,7.53,10.9,12.79,19.65,12.79,11.86,0,21.5-9.65,21.5-21.5s-9.64-21.5-21.5-21.5ZM100.75,45.39c-9.1,0-16.5-7.4-16.5-16.5s7.4-16.5,16.5-16.5,16.5,7.4,16.5,16.5-7.4,16.5-16.5,16.5Z',
d: 'M14.79,13.24c1.93,0,3.49,1.56,3.49,3.49s-1.56,3.49-3.49,3.49-3.49-1.56-3.49-3.49,1.56-3.49,3.49-3.49ZM25.6,13.24c1.93,0,3.49,1.56,3.49,3.49s-1.56,3.49-3.49,3.49-3.49-1.56-3.49-3.49,1.56-3.49,3.49-3.49ZM40.12,48.91L29.04,0h-.02c0,.07.02.13.02.2,0,3.3-3.96,5.98-8.84,5.98S11.36,3.51,11.36.2c0-.07.01-.13.02-.2h-.04L.26,48.91c-1.47,6.47,3.53,12.62,10.26,12.62h19.34c6.73,0,11.73-6.15,10.26-12.62ZM16.61,58.42c-.78,0-1.4-.63-1.4-1.4,0-14.27-6.25-14.65-6.52-14.66-.78,0-1.39-.63-1.39-1.4s.64-1.4,1.42-1.4c.38,0,9.3.2,9.3,17.47,0,.78-.63,1.4-1.4,1.4ZM31.7,42.36c-.31.01-6.52.45-6.52,14.66,0,.78-.63,1.4-1.4,1.4s-1.4-.63-1.4-1.4c0-17.27,8.92-17.47,9.3-17.47.78,0,1.4.63,1.4,1.4s-.62,1.39-1.38,1.4Z',
fill: 'currentColor'
},
child: []

View File

@ -0,0 +1,356 @@
import React, { useState, useEffect, useRef } from 'react'
import SettingCard from '../base/base-setting-card'
import SettingItem from '../base/base-setting-item'
import { Button, Modal, ModalContent, ModalHeader, ModalBody, ModalFooter, Input } from '@heroui/react'
import { useControledMihomoConfig } from '@renderer/hooks/use-controled-mihomo-config'
import { useTranslation } from 'react-i18next'
import { MdEdit, MdDelete, MdOpenInNew } from 'react-icons/md'
interface WebUIPanel {
id: string
name: string
url: string
isDefault?: boolean
}
// 用于高亮显示URL中的变量
const HighlightedUrl: React.FC<{ url: string }> = ({ url }) => {
const parts = url.split(/(%host|%port|%secret)/g)
return (
<p className="text-sm text-default-500 break-all">
{parts.map((part, index) => {
if (part === '%host' || part === '%port' || part === '%secret') {
return (
<span key={index} className="bg-warning-200 text-warning-800 px-1 rounded">
{part}
</span>
)
}
return part
})}
</p>
)
}
// 可点击的变量标签组件
const ClickableVariableTag: React.FC<{
variable: string;
onClick: (variable: string) => void
}> = ({ variable, onClick }) => {
return (
<span
className="bg-warning-200 text-warning-800 px-1 rounded ml-1 cursor-pointer hover:bg-warning-300"
onClick={() => onClick(variable)}
>
{variable}
</span>
)
}
const WebUIConfig: React.FC = () => {
const { t } = useTranslation()
const { controledMihomoConfig } = useControledMihomoConfig()
const externalController = controledMihomoConfig?.['external-controller'] || ''
const secret = controledMihomoConfig?.secret || ''
// 解析主机和端口
const parseController = () => {
if (externalController) {
const [host, port] = externalController.split(':')
return { host: host.replace('0.0.0.0', '127.0.0.1'), port }
}
return { host: '127.0.0.1', port: '9090' }
}
const { host, port } = parseController()
// 默认WebUI面板选项
const defaultWebUIPanels: WebUIPanel[] = [
{
id: 'metacubexd',
name: 'MetaCubeXD',
url: 'https://metacubex.github.io/metacubexd/#/setup?http=true&hostname=%host&port=%port&secret=%secret',
isDefault: true
},
{
id: 'yacd',
name: 'YACD',
url: 'https://yacd.metacubex.one/?hostname=%host&port=%port&secret=%secret',
isDefault: true
},
{
id: 'zashboard',
name: 'Zashboard',
url: 'https://board.zash.run.place/#/setup?http=true&hostname=%host&port=%port&secret=%secret',
isDefault: true
}
]
const [isModalOpen, setIsModalOpen] = useState(false)
const [allPanels, setAllPanels] = useState<WebUIPanel[]>([])
const [editingPanel, setEditingPanel] = useState<WebUIPanel | null>(null)
const [newPanelName, setNewPanelName] = useState('')
const [newPanelUrl, setNewPanelUrl] = useState('')
const urlInputRef = useRef<HTMLInputElement>(null)
// 初始化面板列表
useEffect(() => {
const savedPanels = localStorage.getItem('webui-panels')
if (savedPanels) {
setAllPanels(JSON.parse(savedPanels))
} else {
setAllPanels(defaultWebUIPanels)
}
}, [])
// 保存面板列表到localStorage
useEffect(() => {
if (allPanels.length > 0) {
localStorage.setItem('webui-panels', JSON.stringify(allPanels))
}
}, [allPanels])
// 在URL输入框光标处插入或替换变量
const insertVariableAtCursor = (variable: string) => {
if (!urlInputRef.current) return
const input = urlInputRef.current
const start = input.selectionStart || 0
const end = input.selectionEnd || 0
const currentValue = newPanelUrl || ''
// 如果有选中文本,则替换选中的文本
const newValue = currentValue.substring(0, start) + variable + currentValue.substring(end)
setNewPanelUrl(newValue)
// 设置光标位置到插入变量之后
setTimeout(() => {
if (urlInputRef.current) {
const newCursorPos = start + variable.length
urlInputRef.current.setSelectionRange(newCursorPos, newCursorPos)
urlInputRef.current.focus()
}
}, 0)
}
// 打开WebUI面板
const openWebUI = (panel: WebUIPanel) => {
const url = panel.url
.replace('%host', host)
.replace('%port', port)
.replace('%secret', secret)
window.open(url, '_blank')
}
// 添加新面板
const addNewPanel = () => {
if (newPanelName && newPanelUrl) {
const newPanel: WebUIPanel = {
id: Date.now().toString(),
name: newPanelName,
url: newPanelUrl
}
setAllPanels([...allPanels, newPanel])
setNewPanelName('')
setNewPanelUrl('')
setEditingPanel(null)
}
}
// 更新面板
const updatePanel = () => {
if (editingPanel && newPanelName && newPanelUrl) {
const updatedPanels = allPanels.map(panel =>
panel.id === editingPanel.id
? { ...panel, name: newPanelName, url: newPanelUrl }
: panel
)
setAllPanels(updatedPanels)
setEditingPanel(null)
setNewPanelName('')
setNewPanelUrl('')
}
}
// 删除面板
const deletePanel = (id: string) => {
setAllPanels(allPanels.filter(panel => panel.id !== id))
}
// 开始编辑面板
const startEditing = (panel: WebUIPanel) => {
setEditingPanel(panel)
setNewPanelName(panel.name)
setNewPanelUrl(panel.url)
}
// 取消编辑
const cancelEditing = () => {
setEditingPanel(null)
setNewPanelName('')
setNewPanelUrl('')
}
// 恢复默认面板
const restoreDefaultPanels = () => {
setAllPanels(defaultWebUIPanels)
}
return (
<SettingCard>
<SettingItem title={t('settings.webui.title')} divider>
<div className="flex gap-2">
<Button
size="sm"
color="primary"
onPress={() => setIsModalOpen(true)}
>
{t('settings.webui.manage')}
</Button>
</div>
</SettingItem>
<SettingItem title={t('settings.webui.currentConfig')}>
<div className="text-sm text-default-500">
<p>{t('settings.webui.host')}: {host}</p>
<p>{t('settings.webui.port')}: {port}</p>
</div>
</SettingItem>
{/* 面板管理模态框 */}
<Modal
isOpen={isModalOpen}
onOpenChange={setIsModalOpen}
size="5xl"
scrollBehavior="inside"
backdrop="blur"
classNames={{ backdrop: 'top-[48px]' }}
hideCloseButton
>
<ModalContent className="h-full w-[calc(100%-100px)]">
<ModalHeader className="flex pb-0 app-drag">
{t('settings.webui.manage')}
</ModalHeader>
<ModalBody className="flex flex-col h-full">
<div className="flex flex-col h-full">
{/* 添加/编辑面板表单 */}
<div className="flex flex-col gap-2 p-3 bg-default-100 rounded-lg flex-shrink-0">
<Input
label={t('settings.webui.panelName')}
placeholder={t('settings.webui.panelNamePlaceholder')}
value={newPanelName}
onValueChange={setNewPanelName}
/>
<Input
ref={urlInputRef}
label={t('settings.webui.panelUrl')}
placeholder={t('settings.webui.panelUrlPlaceholder')}
value={newPanelUrl}
onValueChange={setNewPanelUrl}
/>
<div className="text-xs text-default-500">
{t('settings.webui.variableHint')}:
<ClickableVariableTag variable="%host" onClick={insertVariableAtCursor} />
<ClickableVariableTag variable="%port" onClick={insertVariableAtCursor} />
<ClickableVariableTag variable="%secret" onClick={insertVariableAtCursor} />
</div>
<div className="flex gap-2">
{editingPanel ? (
<>
<Button
size="sm"
color="primary"
onPress={updatePanel}
isDisabled={!newPanelName || !newPanelUrl}
>
{t('common.save')}
</Button>
<Button
size="sm"
color="default"
variant="bordered"
onPress={cancelEditing}
>
{t('common.cancel')}
</Button>
</>
) : (
<Button
size="sm"
color="primary"
onPress={addNewPanel}
isDisabled={!newPanelName || !newPanelUrl}
>
{t('settings.webui.addPanel')}
</Button>
)}
<Button
size="sm"
color="warning"
variant="bordered"
onPress={restoreDefaultPanels}
>
{t('settings.webui.restoreDefaults')}
</Button>
</div>
</div>
{/* 面板列表 */}
<div className="flex flex-col gap-2 mt-2 overflow-y-auto flex-grow">
<h3 className="text-lg font-semibold">{t('settings.webui.panels')}</h3>
{allPanels.map(panel => (
<div key={panel.id} className="flex items-start justify-between p-3 bg-default-50 rounded-lg flex-shrink-0">
<div className="flex-1 mr-2">
<p className="font-medium">{panel.name}</p>
<HighlightedUrl url={panel.url} />
</div>
<div className="flex gap-2">
<Button
isIconOnly
size="sm"
color="primary"
onPress={() => openWebUI(panel)}
>
<MdOpenInNew />
</Button>
<Button
isIconOnly
size="sm"
color="warning"
onPress={() => startEditing(panel)}
>
<MdEdit />
</Button>
<Button
isIconOnly
size="sm"
color="danger"
onPress={() => deletePanel(panel.id)}
>
<MdDelete />
</Button>
</div>
</div>
))}
</div>
</div>
</ModalBody>
<ModalFooter className="pt-0">
<Button
color="primary"
onPress={() => setIsModalOpen(false)}
>
{t('common.close')}
</Button>
</ModalFooter>
</ModalContent>
</Modal>
</SettingCard>
)
}
export default WebUIConfig

View File

@ -296,4 +296,4 @@ const loadImage = (url: string): Promise<string> => {
})
}
const trayIconBase64 = ``
const trayIconBase64 = ``

View File

@ -91,6 +91,21 @@
"settings.links.github": "GitHub Repository",
"settings.links.telegram": "Telegram Group",
"settings.title": "Application Settings",
"settings.webui.title": "WebUI Management Panel",
"settings.webui.manage": "Manage Panels",
"settings.webui.currentConfig": "Current Configuration",
"settings.webui.host": "Host",
"settings.webui.port": "Port",
"settings.webui.secret": "Secret",
"settings.webui.noSecret": "No Secret",
"settings.webui.panelName": "Panel Name",
"settings.webui.panelUrl": "Panel URL",
"settings.webui.panelNamePlaceholder": "Enter panel name",
"settings.webui.panelUrlPlaceholder": "Enter panel URL",
"settings.webui.variableHint": "Available Variables",
"settings.webui.addPanel": "Add Panel",
"settings.webui.panels": "Panel List",
"settings.webui.restoreDefaults": "Restore Defaults",
"mihomo.userAgent": "Subscription User Agent",
"mihomo.userAgentPlaceholder": "Default: mihomo.party/v{{version}} (clash.meta)",
"mihomo.delayTest.url": "Delay Test URL",
@ -388,6 +403,7 @@
"dns.customHosts.valuePlaceholder": "Domain or IP",
"dns.saveOnly": "Save Only",
"profiles.title": "Profile Management",
"profiles.input.placeholder": "Enter your subscription URL",
"profiles.updateAll": "Update All Profiles",
"profiles.useProxy": "Proxy",
"profiles.import": "Import",
@ -404,12 +420,6 @@
"profiles.remote": "Remote",
"profiles.local": "Local",
"profiles.trafficUsage": "Traffic Usage Progress",
"profiles.openWebUI.title": "WebUI Management Panel",
"profiles.openWebUI.description": "Select a WebUI panel to open",
"profiles.openWebUI.local": "Local WebUI",
"profiles.updateWebUI.button": "Update Panel",
"profiles.updateWebUI.success": "WebUI panel updated successfully",
"profiles.updateWebUI.failed": "WebUI panel update failed: {{error}}",
"profiles.editInfo.title": "Edit Information",
"profiles.editInfo.name": "Name",
"profiles.editInfo.url": "Subscription URL",

View File

@ -377,6 +377,7 @@
"dns.customHosts.valuePlaceholder": "دامنه یا IP",
"dns.saveOnly": "فقط ذخیره",
"profiles.title": "مدیریت پروفایل",
"profiles.input.placeholder": "آدرس اشتراک خود را وارد کنید",
"profiles.updateAll": "به‌روزرسانی همه پروفایل‌ها",
"profiles.useProxy": "پراکسی",
"profiles.import": "وارد کردن",

View File

@ -377,6 +377,7 @@
"dns.customHosts.valuePlaceholder": "Домен или IP",
"dns.saveOnly": "Только сохранить",
"profiles.title": "Управление профилями",
"profiles.input.placeholder": "Введите URL вашей подписки",
"profiles.updateAll": "Обновить все профили",
"profiles.useProxy": "Прокси",
"profiles.import": "Импорт",

View File

@ -91,6 +91,21 @@
"settings.links.github": "GitHub 仓库",
"settings.links.telegram": "Telegram 群组",
"settings.title": "应用设置",
"settings.webui.title": "WebUI 管理面板",
"settings.webui.manage": "管理面板",
"settings.webui.currentConfig": "当前配置信息",
"settings.webui.host": "主机",
"settings.webui.port": "端口",
"settings.webui.secret": "密钥",
"settings.webui.noSecret": "无密钥",
"settings.webui.panelName": "面板名称",
"settings.webui.panelUrl": "面板URL",
"settings.webui.panelNamePlaceholder": "输入面板名称",
"settings.webui.panelUrlPlaceholder": "输入面板URL",
"settings.webui.variableHint": "可用变量",
"settings.webui.addPanel": "添加面板",
"settings.webui.panels": "面板列表",
"settings.webui.restoreDefaults": "恢复默认",
"mihomo.title": "内核设置",
"mihomo.restart": "重启内核",
"mihomo.memory": "内存使用",
@ -388,6 +403,7 @@
"dns.fallbackFilter.domainPlaceholder": "例:+.google.com",
"dns.saveOnly": "仅保存",
"profiles.title": "订阅管理",
"profiles.input.placeholder": "请输入您的订阅网址",
"profiles.updateAll": "更新全部订阅",
"profiles.useProxy": "代理",
"profiles.import": "导入",
@ -409,12 +425,6 @@
"profiles.traffic.expired": "已过期",
"profiles.traffic.remainingDays": "剩余 {{days}} 天",
"profiles.traffic.lastUpdate": "最后更新:{{time}}",
"profiles.openWebUI.title": "WebUI 管理面板",
"profiles.openWebUI.description": "选择一个 WebUI 面板打开",
"profiles.openWebUI.local": "本地 WebUI",
"profiles.updateWebUI.button": "更新面板",
"profiles.updateWebUI.success": "WebUI 面板更新成功",
"profiles.updateWebUI.failed": "WebUI 面板更新失败: {{error}}",
"profiles.editInfo.title": "编辑信息",
"profiles.editInfo.name": "名称",
"profiles.editInfo.url": "订阅地址",

View File

@ -7,16 +7,12 @@ import {
DropdownItem,
DropdownMenu,
DropdownTrigger,
Input,
Card,
CardBody,
CardHeader
Input
} from '@heroui/react'
import BasePage from '@renderer/components/base/base-page'
import ProfileItem from '@renderer/components/profiles/profile-item'
import { useProfileConfig } from '@renderer/hooks/use-profile-config'
import { useAppConfig } from '@renderer/hooks/use-app-config'
import { useControledMihomoConfig } from '@renderer/hooks/use-controled-mihomo-config'
import { getFilePath, readTextFile, subStoreCollections, subStoreSubs } from '@renderer/utils/ipc'
import type { KeyboardEvent } from 'react'
import { ReactNode, useCallback, useEffect, useMemo, useRef, useState } from 'react'
@ -36,7 +32,6 @@ import SubStoreIcon from '@renderer/components/base/substore-icon'
import useSWR from 'swr'
import { useNavigate } from 'react-router-dom'
import { useTranslation } from 'react-i18next'
import { mihomoUpgradeUI } from '@renderer/utils/ipc'
const Profiles: React.FC = () => {
const { t } = useTranslation()
@ -50,9 +45,6 @@ const Profiles: React.FC = () => {
mutateProfileConfig
} = useProfileConfig()
const { appConfig } = useAppConfig()
const { controledMihomoConfig } = useControledMihomoConfig()
const externalController = controledMihomoConfig?.['external-controller'] || ''
const externalUI = (controledMihomoConfig as any)?.['external-ui']
const { useSubStore = true, useCustomSubStore = false, customSubStoreUrl = '' } = appConfig || {}
const { current, items = [] } = profileConfig || {}
const navigate = useNavigate()
@ -203,26 +195,6 @@ const Profiles: React.FC = () => {
setSortedItems(items)
}, [items])
// 获取本地WebUI的URL
const getLocalWebUIUrl = (): string => {
if (externalController) {
// 将地址转换为WebUI URL
// 例如: 127.0.0.1:9090 -> http://127.0.0.1:9090/ui
const controller = externalController.replace('0.0.0.0', '127.0.0.1')
// 如果配置了external-ui使用/ui路径否则可能需要使用不同的路径
const uiPath = externalUI ? '/ui' : '/ui' // 默认使用/ui路径
return `http://${controller}${uiPath}`
}
// 默认URL
return 'http://127.0.0.1:9090/ui'
}
// 检查本地WebUI是否可用
const isLocalWebUIAvailable = (): boolean => {
// 如果有配置的external-controller则认为本地WebUI可用
return !!externalController
}
return (
<BasePage
ref={pageRef}
@ -256,6 +228,7 @@ const Profiles: React.FC = () => {
<div className="flex p-2">
<Input
size="sm"
placeholder={t('profiles.input.placeholder')}
value={url}
onValueChange={setUrl}
onKeyUp={handleInputKeyUp}
@ -406,53 +379,6 @@ const Profiles: React.FC = () => {
</div>
<Divider />
</div>
{/* WebUI Card with Multiple Options */}
<div className="m-2">
<Card>
<CardHeader className="flex gap-3">
<div className="flex flex-col">
<p className="text-md">{t('profiles.openWebUI.title')}</p>
<p className="text-small text-default-500">{t('profiles.openWebUI.description')}</p>
</div>
</CardHeader>
<CardBody>
<div className="flex gap-2 flex-wrap">
<Button
onPress={() => window.open('https://metacubexd.pages.dev/', '_blank')}
>
MetaCubeXD
</Button>
<Button
onPress={() => window.open('https://zashboard.pages.dev/', '_blank')}
>
Zashboard
</Button>
{isLocalWebUIAvailable() && (
<>
<Button
onPress={() => window.open(getLocalWebUIUrl(), '_blank')}
>
{t('profiles.openWebUI.local')}
</Button>
<Button
color="success"
onPress={async () => {
try {
await mihomoUpgradeUI()
new Notification(t('profiles.updateWebUI.success'))
} catch (e) {
new Notification(t('profiles.updateWebUI.failed', { error: String(e) }))
}
}}
>
{t('profiles.updateWebUI.button')}
</Button>
</>
)}
</div>
</CardBody>
</Card>
</div>
<DndContext sensors={sensors} collisionDetection={closestCenter} onDragEnd={onDragEnd}>
<div
className={`${fileOver ? 'blur-sm' : ''} grid sm:grid-cols-2 md:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-2 m-2`}

View File

@ -10,6 +10,7 @@ import ShortcutConfig from '@renderer/components/settings/shortcut-config'
import { FaTelegramPlane } from 'react-icons/fa'
import SiderConfig from '@renderer/components/settings/sider-config'
import SubStoreConfig from '@renderer/components/settings/substore-config'
import WebUIConfig from '@renderer/components/settings/webui-config'
import { useTranslation } from 'react-i18next'
const Settings: React.FC = () => {
@ -60,6 +61,7 @@ const Settings: React.FC = () => {
}
>
<GeneralConfig />
<WebUIConfig />
<SubStoreConfig />
<SiderConfig />
<WebdavConfig />