Compare commits

..

No commits in common. "6f5f7390593f1454101118f1b8d408c4cc385664" and "36fd04f975fbdd10ba44ffe2f935ec98f1bd3dce" have entirely different histories.

6 changed files with 23 additions and 174 deletions

View File

@ -12,8 +12,6 @@ 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) 反馈,本站已开放注册。

View File

@ -1,4 +1,4 @@
import React, { useEffect, useMemo } from 'react' import React, { useEffect } from 'react'
import { useState } from 'react' import { useState } from 'react'
import { import {
Button, Button,
@ -35,17 +35,12 @@ 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, OnFileDrop } from '../wailsjs/runtime/runtime' import { EventsOn } from '../wailsjs/runtime/runtime'
const columnsDef: TableColumnDefinition<Item>[] = [ const columnsDef: TableColumnDefinition<Item>[] = [
createTableColumn<Item>({ createTableColumn<Item>({
@ -56,10 +51,6 @@ const columnsDef: TableColumnDefinition<Item>[] = [
columnId: 'file', columnId: 'file',
renderHeaderCell: () => <></>, renderHeaderCell: () => <></>,
}), }),
createTableColumn<Item>({
columnId: 'operation',
renderHeaderCell: () => <></>,
}),
] ]
const useStyles = makeStyles({ const useStyles = makeStyles({
@ -78,9 +69,10 @@ export const App = () => {
const styles = useStyles() const styles = useStyles()
const [items, setItems] = useState<Item[]>([]) const [items, setItems] = useState<Item[]>([])
const isProcessing = useMemo(() => { const [isProcessing, setIsProcessing] = useState(false)
return items.some(item => item.status === 'processing')
}, [items]) const [totalFilesNeedToProcess, setTotalFilesNeedToProcess] = useState(0)
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('')
@ -95,13 +87,9 @@ export const App = () => {
minWidth: 100, minWidth: 100,
}, },
file: { file: {
idealWidth: 1000, idealWidth: 150,
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
@ -143,13 +131,11 @@ 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)
}) })
} }
@ -176,10 +162,7 @@ export const App = () => {
showDialog('当前文件列表已全部处理完毕,请重新添加新的文件。') showDialog('当前文件列表已全部处理完毕,请重新添加新的文件。')
return return
} }
if(saveTo === 'custom' && savePath === '') { setIsProcessing(true)
showDialog('保存路径为空,请先设置保存路径。')
return
}
const ncmFiles: main.NcmFile[] = [] const ncmFiles: main.NcmFile[] = []
items.forEach(item => { items.forEach(item => {
ncmFiles.push({ ncmFiles.push({
@ -192,37 +175,24 @@ 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(() => {
Save({ if (finishedCount === totalFilesNeedToProcess) {
save_to: saveTo, setFinishedCount(0)
path: savePath, setTotalFilesNeedToProcess(0)
}).then(_ => {}) setIsProcessing(false)
}, [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' }])
} }
}, false) }, [finishedCount])
return ( return (
<div className="p-3"> <div className="p-3">
@ -247,22 +217,10 @@ export const App = () => {
</DialogSurface> </DialogSurface>
</Dialog> </Dialog>
<div className="flex space-between gap-3"> <div className="flex space-between gap-3">
<Button onClick={selectFiles} icon={<DocumentAddRegular />}> <Button onClick={selectFiles}></Button>
<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()
}} }}
@ -331,18 +289,6 @@ 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
View File

@ -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.4 git.taurusxin.com/taurusxin/ncmdump-go v1.7.3
github.com/wailsapp/wails/v2 v2.9.2 github.com/wailsapp/wails/v2 v2.9.2
) )

4
go.sum
View File

@ -1,5 +1,5 @@
git.taurusxin.com/taurusxin/ncmdump-go v1.7.4 h1:Uk7tP58yNMOMqH1e9+BOFwwQ/lqScK3F10mAIFB/jtM= git.taurusxin.com/taurusxin/ncmdump-go v1.7.3 h1:UbaWVqWpCHJxle2ud631TzNavhSi7cuwu6+MudG3/tQ=
git.taurusxin.com/taurusxin/ncmdump-go v1.7.4/go.mod h1:6kRSwUFM9BZwvDrg6MEPBn+29+Q2131QjK/URWO7seg= git.taurusxin.com/taurusxin/ncmdump-go v1.7.3/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
View File

@ -6,8 +6,6 @@ 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
@ -17,13 +15,8 @@ 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,
@ -34,11 +27,6 @@ func main() {
OnStartup: app.startup, OnStartup: app.startup,
Bind: []interface{}{ Bind: []interface{}{
app, app,
config_manager,
},
DragAndDrop: &options.DragAndDrop{
EnableFileDrop: true,
DisableWebViewDrop: true,
}, },
}) })

View File

@ -1,83 +0,0 @@
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
}