Skip to content

Commit 736dee0

Browse files
committed
feat: add edit rule in editor
1 parent 865c830 commit 736dee0

File tree

6 files changed

+91
-68
lines changed

6 files changed

+91
-68
lines changed

notes.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,6 @@
44

55
- [x] 快速添加规则, mc clash add-rule GUI version
66
- [x] 记住窗口位置
7-
- [ ] 导入导出(store 加密有必要, 防止扫描)
8-
- [ ] 备份不处理详情
7+
- [x] 导入导出(store 加密有必要, 防止扫描)
8+
- [ ] 备份不处理详情. (no need)
99
- [ ] 在 vscode/Atom 中编辑规则

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@
4646
"electron-updater": "^4.0.6",
4747
"electron-util": "^0.13.0",
4848
"env-paths": "^2.2.0",
49+
"execa": "^4.0.3",
4950
"fs-extra": "^9.0.1",
5051
"js-yaml": "^3.14.0",
5152
"launch-editor": "^2.2.1",

readme.md

+2-41
Original file line numberDiff line numberDiff line change
@@ -1,48 +1,9 @@
11
# clash-config-manager
22

3-
> a clash config manager
4-
5-
## Todo
6-
7-
- [ ] 配置, 在 vscode 中打开
8-
- [ ] 添加配置
9-
- [ ] mc socks clash
3+
> a clashX config manager for macOS
104
115
## Install
126

137
_macOS 10.10+, Linux, and Windows 7+ are supported (64-bit only)._
148

15-
**macOS**
16-
17-
[**Download**](https://github.com/user/repo/releases/latest) the `.dmg` file.
18-
19-
**Linux**
20-
21-
[**Download**](https://github.com/user/repo/releases/latest) the `.AppImage` or `.deb` file.
22-
23-
_The AppImage needs to be [made executable](http://discourse.appimage.org/t/how-to-make-an-appimage-executable/80) after download._
24-
25-
**Windows**
26-
27-
[**Download**](https://github.com/user/repo/releases/latest) the `.exe` file.
28-
29-
---
30-
31-
## Dev
32-
33-
Built with [Electron](https://electronjs.org).
34-
35-
### Run
36-
37-
```
38-
$ npm install
39-
$ npm start
40-
```
41-
42-
### Publish
43-
44-
```
45-
$ npm run release
46-
```
47-
48-
After Travis finishes building your app, open the release draft it created and click "Publish".
9+
[**Download**](https://github.com/magicdawn/clash-config-manager/releases/latest) the `.dmg` file.

src/page/library-rule-list/ConfigEditor.js

+15-14
Original file line numberDiff line numberDiff line change
@@ -8,13 +8,12 @@ import React, {
88
forwardRef,
99
} from 'react'
1010
import {usePersistFn, useUpdateEffect} from 'ahooks'
11-
11+
import {Spin} from 'antd'
1212
import style from './ConfigEditor.module.less'
1313

1414
export default forwardRef(ConfigEditor)
15-
1615
function ConfigEditor(props, ref) {
17-
const {value, onChange, readonly} = props
16+
const {value, onChange, readonly, spinProps} = props
1817

1918
const editorWillMount = usePersistFn((editor, monaco) => {})
2019

@@ -59,16 +58,18 @@ function ConfigEditor(props, ref) {
5958
}, [readonly, ref])
6059

6160
return (
62-
<div style={{width: '100%', height: '50vh'}} className={style.editor}>
63-
<MonacoEditor
64-
language='yaml'
65-
theme='quite-light'
66-
value={value}
67-
options={options}
68-
onChange={editorOnChange}
69-
editorWillMount={editorWillMount}
70-
editorDidMount={editorDidMount}
71-
/>
72-
</div>
61+
<Spin {...spinProps}>
62+
<div style={{width: '100%', height: '50vh'}} className={style.editor}>
63+
<MonacoEditor
64+
language='yaml'
65+
theme='quite-light'
66+
value={value}
67+
options={options}
68+
onChange={editorOnChange}
69+
editorWillMount={editorWillMount}
70+
editorDidMount={editorDidMount}
71+
/>
72+
</div>
73+
</Spin>
7374
)
7475
}

src/page/library-rule-list/index.js

+71-4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,8 @@
1+
import {remote} from 'electron'
2+
import path from 'path'
3+
import fse from 'fs-extra'
4+
import execa from 'execa'
5+
import debugFactory from 'debug'
16
import React, {useState, useCallback, useRef} from 'react'
27
import {Button, Modal, Input, message, Tooltip} from 'antd'
38
import {useMount, usePersistFn, useUpdateEffect} from 'ahooks'
@@ -10,8 +15,10 @@ import Yaml from 'js-yaml'
1015
import styles from './index.module.less'
1116
import {List, Space, Form, Select} from 'antd'
1217
const {Option} = Select
18+
const debug = debugFactory('app:libraryRuleList')
1319

1420
const namespace = 'libraryRuleList'
21+
const TEMP_EDITING_FILE = path.join(remote.app.getPath('userData'), 'temp', '临时文件-关闭生效.yml')
1522

1623
export default function LibraryRuleList() {
1724
const {effects, state} = usePlug({
@@ -214,6 +221,7 @@ function ModalAdd({visible, setVisible, editItem, editItemIndex, editMode}) {
214221
}
215222

216223
const handleCancel = useCallback(() => {
224+
if (editInEditorMaskVisible) return
217225
setVisible(false)
218226
clean()
219227
}, [])
@@ -296,6 +304,39 @@ function ModalAdd({visible, setVisible, editItem, editItemIndex, editMode}) {
296304
[form]
297305
)
298306

307+
const [editInEditorMaskVisible, setEditInEditorMaskVisible] = useState(false)
308+
const editInEditor = usePersistFn(async (editor = 'code') => {
309+
const content = form.getFieldValue('content')
310+
await fse.outputFile(TEMP_EDITING_FILE, content, 'utf8')
311+
312+
// wait edit
313+
setEditInEditorMaskVisible(true)
314+
let execResults
315+
const cmd = `${editor} --wait '${TEMP_EDITING_FILE}'`
316+
try {
317+
execResults = await execa.command(cmd, {shell: true})
318+
} catch (e) {
319+
message.error('执行命令出错: ' + e.message)
320+
return
321+
} finally {
322+
setEditInEditorMaskVisible(false)
323+
}
324+
325+
debug('exec: %o', {cmd, execResults})
326+
const {exitCode} = execResults || {}
327+
if (exitCode !== 0) {
328+
message.error(`执行命令出错: exitCode = ${exitCode}`)
329+
return
330+
}
331+
332+
// read & set
333+
const newContent = await fse.readFile(TEMP_EDITING_FILE, 'utf8')
334+
if (newContent !== content) {
335+
form.setFieldsValue({content: newContent})
336+
message.success('文件内容已更新')
337+
}
338+
})
339+
299340
return (
300341
<Modal
301342
className={styles.modal}
@@ -317,11 +358,23 @@ function ModalAdd({visible, setVisible, editItem, editItemIndex, editMode}) {
317358
/>
318359
)}
319360

320-
<Button onClick={handleAddRuleChrome}>从 Chrome 添加规则</Button>
361+
<Space direction='horizontal'>
362+
<Button disabled={editInEditorMaskVisible} onClick={handleAddRuleChrome}>
363+
从 Chrome 添加规则
364+
</Button>
365+
<Button disabled={editInEditorMaskVisible} onClick={() => editInEditor('code')}>
366+
使用 vscode 编辑
367+
</Button>
368+
<Button disabled={editInEditorMaskVisible} onClick={() => editInEditor('atom')}>
369+
使用 Atom 编辑
370+
</Button>
371+
</Space>
321372

322373
<div className='btn-wrapper'>
323-
<Button onClick={handleCancel}>取消</Button>
324-
<Button type='primary' onClick={handleOk}>
374+
<Button disabled={editInEditorMaskVisible} onClick={handleCancel}>
375+
取消
376+
</Button>
377+
<Button disabled={editInEditorMaskVisible} type='primary' onClick={handleOk}>
325378
确定
326379
</Button>
327380
</div>
@@ -362,7 +415,21 @@ function ModalAdd({visible, setVisible, editItem, editItemIndex, editMode}) {
362415
name='content'
363416
rules={[{required: true, message: '内容不能为空'}]}
364417
>
365-
<ConfigEditor ref={configEditorRef} readonly={readonly} />
418+
<ConfigEditor
419+
ref={configEditorRef}
420+
readonly={readonly}
421+
spinProps={{
422+
size: 'large',
423+
spinning: editInEditorMaskVisible,
424+
tip: (
425+
<>
426+
文件已经在编辑器中打开
427+
<br />
428+
在编辑器中关闭文件生效
429+
</>
430+
),
431+
}}
432+
/>
366433
</Form.Item>
367434
) : (
368435
<Form.Item label='URL' name='url' rules={[{required: true, message: 'url不能为空'}]}>

src/util/sync/webdav/customMerge.js

-7
Original file line numberDiff line numberDiff line change
@@ -24,13 +24,6 @@ import produce from 'immer'
2424
export default function mergeStore(a, b) {
2525
let ret = {...a}
2626

27-
const objectPath = [
28-
//
29-
'subscribe_detail',
30-
'current_config_v2.name',
31-
'preference',
32-
]
33-
3427
ret = produce(ret, (draft) => {
3528
// 'subscribe_detail',
3629
Object.assign(draft.subscribe_detail, b.subscribe_detail)

0 commit comments

Comments
 (0)