Compare commits
	
		
			6 Commits
		
	
	
		
			36fd04f975
			...
			6f5f739059
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 6f5f739059 | |||
| b0bfb3e606 | |||
| 4df7df678b | |||
| 33d9b91c52 | |||
| f7a7a7b401 | |||
| 7402e35d66 | 
| @ -12,6 +12,8 @@ ncmdump-gui 基于 [ncmdump-go](https://git.taurusxin.com/taurusxin/ncmdump-go) | |||||||
| 
 | 
 | ||||||
| 前往[Releases](https://git.taurusxin.com/taurusxin/ncmdump-gui/releases)下载最新版本。 | 前往[Releases](https://git.taurusxin.com/taurusxin/ncmdump-gui/releases)下载最新版本。 | ||||||
| 
 | 
 | ||||||
|  | 请注意:绝大多数较新版本的 Windows 都自带了 Webview 2 运行时,如果遇到了软件无法启动的问题,请前往微软官网下载:<https://developer.microsoft.com/zh-cn/microsoft-edge/webview2/> | ||||||
|  | 
 | ||||||
| ## Bug 及建议 | ## Bug 及建议 | ||||||
| 
 | 
 | ||||||
| 如有任何问题或建议,请直接通过 [Issue](https://git.taurusxin.com/taurusxin/ncmdump-gui/issues) 反馈,本站已开放注册。 | 如有任何问题或建议,请直接通过 [Issue](https://git.taurusxin.com/taurusxin/ncmdump-gui/issues) 反馈,本站已开放注册。 | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| import React, { useEffect } from 'react' | import React, { useEffect, useMemo } from 'react' | ||||||
| import { useState } from 'react' | import { useState } from 'react' | ||||||
| import { | import { | ||||||
|   Button, |   Button, | ||||||
| @ -35,12 +35,17 @@ import { | |||||||
|   CheckmarkRegular, |   CheckmarkRegular, | ||||||
|   DismissRegular, |   DismissRegular, | ||||||
|   DocumentArrowRightRegular, |   DocumentArrowRightRegular, | ||||||
|  |   DocumentAddRegular, | ||||||
|  |   DeleteRegular, | ||||||
|  |   DeleteDismissRegular, | ||||||
|  |   WindowPlayRegular, | ||||||
| } from '@fluentui/react-icons' | } from '@fluentui/react-icons' | ||||||
| 
 | 
 | ||||||
| import { Status, Item, SaveTo } from './types' | import { Status, Item, SaveTo } from './types' | ||||||
| import { SelectFiles, SelectFolder, ProcessFiles } from '../wailsjs/go/main/App' | import { SelectFiles, SelectFolder, ProcessFiles } from '../wailsjs/go/main/App' | ||||||
|  | import { Load, Save } from '../wailsjs/go/utils/ConfigManager' | ||||||
| import { main } from '../wailsjs/go/models' | import { main } from '../wailsjs/go/models' | ||||||
| import { EventsOn } from '../wailsjs/runtime/runtime' | import { EventsOn, OnFileDrop } from '../wailsjs/runtime/runtime' | ||||||
| 
 | 
 | ||||||
| const columnsDef: TableColumnDefinition<Item>[] = [ | const columnsDef: TableColumnDefinition<Item>[] = [ | ||||||
|   createTableColumn<Item>({ |   createTableColumn<Item>({ | ||||||
| @ -51,6 +56,10 @@ const columnsDef: TableColumnDefinition<Item>[] = [ | |||||||
|     columnId: 'file', |     columnId: 'file', | ||||||
|     renderHeaderCell: () => <>文件</>, |     renderHeaderCell: () => <>文件</>, | ||||||
|   }), |   }), | ||||||
|  |   createTableColumn<Item>({ | ||||||
|  |     columnId: 'operation', | ||||||
|  |     renderHeaderCell: () => <>操作</>, | ||||||
|  |   }), | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
| const useStyles = makeStyles({ | const useStyles = makeStyles({ | ||||||
| @ -69,10 +78,9 @@ export const App = () => { | |||||||
|   const styles = useStyles() |   const styles = useStyles() | ||||||
| 
 | 
 | ||||||
|   const [items, setItems] = useState<Item[]>([]) |   const [items, setItems] = useState<Item[]>([]) | ||||||
|   const [isProcessing, setIsProcessing] = useState(false) |   const isProcessing = useMemo(() => { | ||||||
| 
 |     return items.some(item => item.status === 'processing') | ||||||
|   const [totalFilesNeedToProcess, setTotalFilesNeedToProcess] = useState(0) |   }, [items]) | ||||||
|   const [finishedCount, setFinishedCount] = useState(0) |  | ||||||
| 
 | 
 | ||||||
|   const [saveTo, setSaveTo] = useState<SaveTo>('original') |   const [saveTo, setSaveTo] = useState<SaveTo>('original') | ||||||
|   const [savePath, setSavePath] = useState('') |   const [savePath, setSavePath] = useState('') | ||||||
| @ -87,9 +95,13 @@ export const App = () => { | |||||||
|       minWidth: 100, |       minWidth: 100, | ||||||
|     }, |     }, | ||||||
|     file: { |     file: { | ||||||
|       idealWidth: 150, |       idealWidth: 1000, | ||||||
|       minWidth: 150, |       minWidth: 150, | ||||||
|     }, |     }, | ||||||
|  |     operation: { | ||||||
|  |       idealWidth: 80, | ||||||
|  |       minWidth: 80, | ||||||
|  |     }, | ||||||
|   }) |   }) | ||||||
| 
 | 
 | ||||||
|   // eslint-disable-next-line @typescript-eslint/naming-convention
 |   // eslint-disable-next-line @typescript-eslint/naming-convention
 | ||||||
| @ -131,11 +143,13 @@ export const App = () => { | |||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const selectFiles = () => { |   const selectFiles = () => { | ||||||
|  |     if (isProcessing) { | ||||||
|  |       return | ||||||
|  |     } | ||||||
|     SelectFiles().then(files => { |     SelectFiles().then(files => { | ||||||
|       for (const file of files) { |       for (const file of files) { | ||||||
|         setItems(prev => [...prev, { file, status: 'pending' }]) |         setItems(prev => [...prev, { file, status: 'pending' }]) | ||||||
|       } |       } | ||||||
|       setTotalFilesNeedToProcess(files.length) |  | ||||||
|     }) |     }) | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| @ -162,7 +176,10 @@ export const App = () => { | |||||||
|       showDialog('当前文件列表已全部处理完毕,请重新添加新的文件。') |       showDialog('当前文件列表已全部处理完毕,请重新添加新的文件。') | ||||||
|       return |       return | ||||||
|     } |     } | ||||||
|     setIsProcessing(true) |     if(saveTo === 'custom' && savePath === '') { | ||||||
|  |       showDialog('保存路径为空,请先设置保存路径。') | ||||||
|  |       return | ||||||
|  |     } | ||||||
|     const ncmFiles: main.NcmFile[] = [] |     const ncmFiles: main.NcmFile[] = [] | ||||||
|     items.forEach(item => { |     items.forEach(item => { | ||||||
|       ncmFiles.push({ |       ncmFiles.push({ | ||||||
| @ -175,24 +192,37 @@ export const App = () => { | |||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     EventsOn('file-status-changed', (index: number, status: Status) => { |     EventsOn('file-status-changed', (index: number, status: Status) => { | ||||||
|       if (status == 'done' || status === 'error') { |  | ||||||
|         setFinishedCount(prev => prev + 1) |  | ||||||
|       } |  | ||||||
|       setItems(prev => { |       setItems(prev => { | ||||||
|         const newItems = [...prev] |         const newItems = [...prev] | ||||||
|         newItems[index].status = status |         newItems[index].status = status | ||||||
|         return newItems |         return newItems | ||||||
|       }) |       }) | ||||||
|     }) |     }) | ||||||
|  |     Load().then(res => { | ||||||
|  |       console.log(res) | ||||||
|  |       setSaveTo(res.save_to as SaveTo) | ||||||
|  |       setSavePath(res.path) | ||||||
|  |     }) | ||||||
|   }, []) |   }, []) | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (finishedCount === totalFilesNeedToProcess) { |     Save({ | ||||||
|       setFinishedCount(0) |       save_to: saveTo, | ||||||
|       setTotalFilesNeedToProcess(0) |       path: savePath, | ||||||
|       setIsProcessing(false) |     }).then(_ => {}) | ||||||
|  |   }, [saveTo, savePath]) | ||||||
|  | 
 | ||||||
|  |   OnFileDrop((_x, _y, paths) => { | ||||||
|  |     let length = paths.length | ||||||
|  |     for (const path of paths) { | ||||||
|  |       // only end with ncm
 | ||||||
|  |       if (!path.endsWith('.ncm')) { | ||||||
|  |         length-- | ||||||
|  |         continue | ||||||
|  |       } | ||||||
|  |       setItems(prev => [...prev, { file: path, status: 'pending' }]) | ||||||
|     } |     } | ||||||
|   }, [finishedCount]) |   }, false) | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <div className="p-3"> |     <div className="p-3"> | ||||||
| @ -217,10 +247,22 @@ export const App = () => { | |||||||
|         </DialogSurface> |         </DialogSurface> | ||||||
|       </Dialog> |       </Dialog> | ||||||
|       <div className="flex space-between gap-3"> |       <div className="flex space-between gap-3"> | ||||||
|         <Button onClick={selectFiles}>添加文件</Button> |         <Button onClick={selectFiles} icon={<DocumentAddRegular />}> | ||||||
|         <Button onClick={() => setItems([])}>清除列表</Button> |           添加文件 | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           onClick={() => { | ||||||
|  |             if (!isProcessing) { | ||||||
|  |               setItems([]) | ||||||
|  |             } | ||||||
|  |           }} | ||||||
|  |           icon={<DeleteDismissRegular />} | ||||||
|  |         > | ||||||
|  |           清除列表 | ||||||
|  |         </Button> | ||||||
|         <Button |         <Button | ||||||
|           appearance="primary" |           appearance="primary" | ||||||
|  |           icon={<WindowPlayRegular />} | ||||||
|           onClick={() => { |           onClick={() => { | ||||||
|             startProcess() |             startProcess() | ||||||
|           }} |           }} | ||||||
| @ -289,6 +331,18 @@ export const App = () => { | |||||||
|                 </TableCellLayout> |                 </TableCellLayout> | ||||||
|               </TableCell> |               </TableCell> | ||||||
|               <TableCell>{file.file}</TableCell> |               <TableCell>{file.file}</TableCell> | ||||||
|  |               <TableCell> | ||||||
|  |                 <Button | ||||||
|  |                   size="small" | ||||||
|  |                   icon={<DeleteRegular />} | ||||||
|  |                   appearance="transparent" | ||||||
|  |                   onClick={() => { | ||||||
|  |                     setItems(prev => prev.filter((_, i) => i !== index)) | ||||||
|  |                   }} | ||||||
|  |                 > | ||||||
|  |                   移除 | ||||||
|  |                 </Button> | ||||||
|  |               </TableCell> | ||||||
|             </TableRow> |             </TableRow> | ||||||
|           ))} |           ))} | ||||||
|         </TableBody> |         </TableBody> | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								go.mod
									
									
									
									
									
								
							| @ -5,7 +5,7 @@ go 1.23.0 | |||||||
| toolchain go1.23.3 | toolchain go1.23.3 | ||||||
| 
 | 
 | ||||||
| require ( | require ( | ||||||
| 	git.taurusxin.com/taurusxin/ncmdump-go v1.7.3 | 	git.taurusxin.com/taurusxin/ncmdump-go v1.7.4 | ||||||
| 	github.com/wailsapp/wails/v2 v2.9.2 | 	github.com/wailsapp/wails/v2 v2.9.2 | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										4
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								go.sum
									
									
									
									
									
								
							| @ -1,5 +1,5 @@ | |||||||
| git.taurusxin.com/taurusxin/ncmdump-go v1.7.3 h1:UbaWVqWpCHJxle2ud631TzNavhSi7cuwu6+MudG3/tQ= | git.taurusxin.com/taurusxin/ncmdump-go v1.7.4 h1:Uk7tP58yNMOMqH1e9+BOFwwQ/lqScK3F10mAIFB/jtM= | ||||||
| git.taurusxin.com/taurusxin/ncmdump-go v1.7.3/go.mod h1:6kRSwUFM9BZwvDrg6MEPBn+29+Q2131QjK/URWO7seg= | git.taurusxin.com/taurusxin/ncmdump-go v1.7.4/go.mod h1:6kRSwUFM9BZwvDrg6MEPBn+29+Q2131QjK/URWO7seg= | ||||||
| github.com/TwiN/go-color v1.4.1 h1:mqG0P/KBgHKVqmtL5ye7K0/Gr4l6hTksPgTgMk3mUzc= | github.com/TwiN/go-color v1.4.1 h1:mqG0P/KBgHKVqmtL5ye7K0/Gr4l6hTksPgTgMk3mUzc= | ||||||
| github.com/TwiN/go-color v1.4.1/go.mod h1:WcPf/jtiW95WBIsEeY1Lc/b8aaWoiqQpu5cf8WFxu+s= | github.com/TwiN/go-color v1.4.1/go.mod h1:WcPf/jtiW95WBIsEeY1Lc/b8aaWoiqQpu5cf8WFxu+s= | ||||||
| github.com/bep/debounce v1.2.1 h1:v67fRdBA9UQu2NhLFXrSg0Brw7CexQekrBwDMM8bzeY= | github.com/bep/debounce v1.2.1 h1:v67fRdBA9UQu2NhLFXrSg0Brw7CexQekrBwDMM8bzeY= | ||||||
|  | |||||||
							
								
								
									
										14
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								main.go
									
									
									
									
									
								
							| @ -6,6 +6,8 @@ import ( | |||||||
| 	"github.com/wailsapp/wails/v2" | 	"github.com/wailsapp/wails/v2" | ||||||
| 	"github.com/wailsapp/wails/v2/pkg/options" | 	"github.com/wailsapp/wails/v2/pkg/options" | ||||||
| 	"github.com/wailsapp/wails/v2/pkg/options/assetserver" | 	"github.com/wailsapp/wails/v2/pkg/options/assetserver" | ||||||
|  | 
 | ||||||
|  | 	"git.taurusxin.com/taurusxin/ncmdump-gui/utils" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| //go:embed all:frontend/dist | //go:embed all:frontend/dist | ||||||
| @ -15,8 +17,13 @@ func main() { | |||||||
| 	// Create an instance of the app structure | 	// Create an instance of the app structure | ||||||
| 	app := NewApp() | 	app := NewApp() | ||||||
| 
 | 
 | ||||||
|  | 	config_manager, err := utils.NewConfigManager("config.json") | ||||||
|  | 	if err != nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	// Create application with options | 	// Create application with options | ||||||
| 	err := wails.Run(&options.App{ | 	err = wails.Run(&options.App{ | ||||||
| 		Title:  "ncmdump-gui", | 		Title:  "ncmdump-gui", | ||||||
| 		Width:  750, | 		Width:  750, | ||||||
| 		Height: 500, | 		Height: 500, | ||||||
| @ -27,6 +34,11 @@ func main() { | |||||||
| 		OnStartup:        app.startup, | 		OnStartup:        app.startup, | ||||||
| 		Bind: []interface{}{ | 		Bind: []interface{}{ | ||||||
| 			app, | 			app, | ||||||
|  | 			config_manager, | ||||||
|  | 		}, | ||||||
|  | 		DragAndDrop: &options.DragAndDrop{ | ||||||
|  | 			EnableFileDrop: true, | ||||||
|  | 			DisableWebViewDrop: true, | ||||||
| 		}, | 		}, | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										83
									
								
								utils/config_manager.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								utils/config_manager.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,83 @@ | |||||||
|  | package utils | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | 	"path/filepath" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type SaveToType = string | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	Original SaveToType = "original" | ||||||
|  | 	Custom   SaveToType = "custom" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type Preference struct { | ||||||
|  | 	SaveTo SaveToType  `json:"save_to"` | ||||||
|  | 	Path   string       `json:"path"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type ConfigManager struct { | ||||||
|  | 	FilePath string | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // NewConfigManager 创建一个新的配置管理器,自动适配不同操作系统的用户目录 | ||||||
|  | func NewConfigManager(filename string) (*ConfigManager, error) { | ||||||
|  | 	configDir, err := os.UserConfigDir() // 获取用户配置目录 | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	dirPath := filepath.Join(configDir, "ncmdump-gui") | ||||||
|  | 	err = os.MkdirAll(dirPath, os.ModePerm) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	filePath := filepath.Join(dirPath, filename) | ||||||
|  | 
 | ||||||
|  | 	configManager := &ConfigManager{FilePath: filePath} | ||||||
|  | 	// if not exist, create it with default value | ||||||
|  | 	if _, err := os.Stat(filePath); os.IsNotExist(err) { | ||||||
|  | 		defaultConfig := &Preference{ | ||||||
|  | 			SaveTo: Original, | ||||||
|  | 			Path:   "", | ||||||
|  | 		} | ||||||
|  | 		configManager.Save(defaultConfig) | ||||||
|  | 	} | ||||||
|  | 	return &ConfigManager{FilePath: filePath}, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Save 将配置保存到文件 | ||||||
|  | func (cm *ConfigManager) Save(preference *Preference) bool { | ||||||
|  | 	file, err := os.Create(cm.FilePath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	defer file.Close() | ||||||
|  | 
 | ||||||
|  | 	encoder := json.NewEncoder(file) | ||||||
|  | 	encoder.SetIndent("", "  ") // 格式化输出 | ||||||
|  | 	err = encoder.Encode(preference) | ||||||
|  | 	return err == nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Load 从文件读取配置 | ||||||
|  | func (cm *ConfigManager) Load() *Preference { | ||||||
|  | 	file, err := os.Open(cm.FilePath) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	defer file.Close() | ||||||
|  | 
 | ||||||
|  | 	var preference *Preference = nil | ||||||
|  | 	decoder := json.NewDecoder(file) | ||||||
|  | 	err = decoder.Decode(&preference) | ||||||
|  | 	fmt.Println(preference) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	return preference | ||||||
|  | } | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user