13 Commits

Author SHA1 Message Date
e3402dbfa3 update: app image 2025-02-01 19:24:31 +08:00
73687b889e feat: macOS about menu 2025-02-01 19:23:17 +08:00
4333d0ea2a fix: eof when save before load 2025-02-01 19:07:09 +08:00
e4f2d1aaab remove: debug print 2025-02-01 19:07:09 +08:00
bc67dab122 ignore: frontend files 2025-02-01 19:07:08 +08:00
ce5ca88694 ignore: frontend files 2025-02-01 19:07:08 +08:00
70ec35653d feat: change version 2025-02-01 19:07:08 +08:00
6f5f739059 feat: save preference to config file 2025-02-01 17:42:29 +08:00
b0bfb3e606 feat: prevent click when processing 2025-02-01 13:43:44 +08:00
4df7df678b feat: icons and remove button 2025-02-01 13:39:13 +08:00
33d9b91c52 update: ncmdump-go to v1.7.4 2025-02-01 13:21:07 +08:00
f7a7a7b401 feat: webview2 instruction 2025-02-01 12:11:46 +08:00
7402e35d66 feat: file drop 2025-02-01 12:11:13 +08:00
16 changed files with 188 additions and 579 deletions

4
.gitignore vendored
View File

@@ -1,3 +1,7 @@
build/bin build/bin
node_modules node_modules
frontend/dist frontend/dist
frontend/wailsjs/
frontend/package.json.md5

View File

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

View File

@@ -1 +0,0 @@
d96d5e55cba5f9d68f0aa1d714fb5f1a

View File

@@ -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({
@@ -65,14 +74,15 @@ const useStyles = makeStyles({
}, },
}) })
let loaded = false
export const App = () => { 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 +97,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 +145,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 +178,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 +194,39 @@ 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 => {
setSaveTo(res.save_to as SaveTo)
setSavePath(res.path)
loaded = true
})
}, []) }, [])
useEffect(() => { useEffect(() => {
if (finishedCount === totalFilesNeedToProcess) { if (loaded) {
setFinishedCount(0) Save({
setTotalFilesNeedToProcess(0) save_to: saveTo,
setIsProcessing(false) path: savePath,
}).then(_ => {})
} }
}, [finishedCount]) }, [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)
return ( return (
<div className="p-3"> <div className="p-3">
@@ -217,10 +251,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 +335,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>

View File

@@ -1,9 +0,0 @@
// Cynhyrchwyd y ffeil hon yn awtomatig. PEIDIWCH Â MODIWL
// This file is automatically generated. DO NOT EDIT
import {main} from '../models';
export function ProcessFiles(arg1:Array<main.NcmFile>,arg2:string):Promise<void>;
export function SelectFiles():Promise<Array<string>>;
export function SelectFolder():Promise<string>;

View File

@@ -1,15 +0,0 @@
// @ts-check
// Cynhyrchwyd y ffeil hon yn awtomatig. PEIDIWCH Â MODIWL
// This file is automatically generated. DO NOT EDIT
export function ProcessFiles(arg1, arg2) {
return window['go']['main']['App']['ProcessFiles'](arg1, arg2);
}
export function SelectFiles() {
return window['go']['main']['App']['SelectFiles']();
}
export function SelectFolder() {
return window['go']['main']['App']['SelectFolder']();
}

View File

@@ -1,19 +0,0 @@
export namespace main {
export class NcmFile {
Name: string;
Status: string;
static createFrom(source: any = {}) {
return new NcmFile(source);
}
constructor(source: any = {}) {
if ('string' === typeof source) source = JSON.parse(source);
this.Name = source["Name"];
this.Status = source["Status"];
}
}
}

View File

@@ -1,24 +0,0 @@
{
"name": "@wailsapp/runtime",
"version": "2.0.0",
"description": "Wails Javascript runtime library",
"main": "runtime.js",
"types": "runtime.d.ts",
"scripts": {
},
"repository": {
"type": "git",
"url": "git+https://github.com/wailsapp/wails.git"
},
"keywords": [
"Wails",
"Javascript",
"Go"
],
"author": "Lea Anthony <lea.anthony@gmail.com>",
"license": "MIT",
"bugs": {
"url": "https://github.com/wailsapp/wails/issues"
},
"homepage": "https://github.com/wailsapp/wails#readme"
}

View File

@@ -1,249 +0,0 @@
/*
_ __ _ __
| | / /___ _(_) /____
| | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/
The electron alternative for Go
(c) Lea Anthony 2019-present
*/
export interface Position {
x: number;
y: number;
}
export interface Size {
w: number;
h: number;
}
export interface Screen {
isCurrent: boolean;
isPrimary: boolean;
width : number
height : number
}
// Environment information such as platform, buildtype, ...
export interface EnvironmentInfo {
buildType: string;
platform: string;
arch: string;
}
// [EventsEmit](https://wails.io/docs/reference/runtime/events#eventsemit)
// emits the given event. Optional data may be passed with the event.
// This will trigger any event listeners.
export function EventsEmit(eventName: string, ...data: any): void;
// [EventsOn](https://wails.io/docs/reference/runtime/events#eventson) sets up a listener for the given event name.
export function EventsOn(eventName: string, callback: (...data: any) => void): () => void;
// [EventsOnMultiple](https://wails.io/docs/reference/runtime/events#eventsonmultiple)
// sets up a listener for the given event name, but will only trigger a given number times.
export function EventsOnMultiple(eventName: string, callback: (...data: any) => void, maxCallbacks: number): () => void;
// [EventsOnce](https://wails.io/docs/reference/runtime/events#eventsonce)
// sets up a listener for the given event name, but will only trigger once.
export function EventsOnce(eventName: string, callback: (...data: any) => void): () => void;
// [EventsOff](https://wails.io/docs/reference/runtime/events#eventsoff)
// unregisters the listener for the given event name.
export function EventsOff(eventName: string, ...additionalEventNames: string[]): void;
// [EventsOffAll](https://wails.io/docs/reference/runtime/events#eventsoffall)
// unregisters all listeners.
export function EventsOffAll(): void;
// [LogPrint](https://wails.io/docs/reference/runtime/log#logprint)
// logs the given message as a raw message
export function LogPrint(message: string): void;
// [LogTrace](https://wails.io/docs/reference/runtime/log#logtrace)
// logs the given message at the `trace` log level.
export function LogTrace(message: string): void;
// [LogDebug](https://wails.io/docs/reference/runtime/log#logdebug)
// logs the given message at the `debug` log level.
export function LogDebug(message: string): void;
// [LogError](https://wails.io/docs/reference/runtime/log#logerror)
// logs the given message at the `error` log level.
export function LogError(message: string): void;
// [LogFatal](https://wails.io/docs/reference/runtime/log#logfatal)
// logs the given message at the `fatal` log level.
// The application will quit after calling this method.
export function LogFatal(message: string): void;
// [LogInfo](https://wails.io/docs/reference/runtime/log#loginfo)
// logs the given message at the `info` log level.
export function LogInfo(message: string): void;
// [LogWarning](https://wails.io/docs/reference/runtime/log#logwarning)
// logs the given message at the `warning` log level.
export function LogWarning(message: string): void;
// [WindowReload](https://wails.io/docs/reference/runtime/window#windowreload)
// Forces a reload by the main application as well as connected browsers.
export function WindowReload(): void;
// [WindowReloadApp](https://wails.io/docs/reference/runtime/window#windowreloadapp)
// Reloads the application frontend.
export function WindowReloadApp(): void;
// [WindowSetAlwaysOnTop](https://wails.io/docs/reference/runtime/window#windowsetalwaysontop)
// Sets the window AlwaysOnTop or not on top.
export function WindowSetAlwaysOnTop(b: boolean): void;
// [WindowSetSystemDefaultTheme](https://wails.io/docs/next/reference/runtime/window#windowsetsystemdefaulttheme)
// *Windows only*
// Sets window theme to system default (dark/light).
export function WindowSetSystemDefaultTheme(): void;
// [WindowSetLightTheme](https://wails.io/docs/next/reference/runtime/window#windowsetlighttheme)
// *Windows only*
// Sets window to light theme.
export function WindowSetLightTheme(): void;
// [WindowSetDarkTheme](https://wails.io/docs/next/reference/runtime/window#windowsetdarktheme)
// *Windows only*
// Sets window to dark theme.
export function WindowSetDarkTheme(): void;
// [WindowCenter](https://wails.io/docs/reference/runtime/window#windowcenter)
// Centers the window on the monitor the window is currently on.
export function WindowCenter(): void;
// [WindowSetTitle](https://wails.io/docs/reference/runtime/window#windowsettitle)
// Sets the text in the window title bar.
export function WindowSetTitle(title: string): void;
// [WindowFullscreen](https://wails.io/docs/reference/runtime/window#windowfullscreen)
// Makes the window full screen.
export function WindowFullscreen(): void;
// [WindowUnfullscreen](https://wails.io/docs/reference/runtime/window#windowunfullscreen)
// Restores the previous window dimensions and position prior to full screen.
export function WindowUnfullscreen(): void;
// [WindowIsFullscreen](https://wails.io/docs/reference/runtime/window#windowisfullscreen)
// Returns the state of the window, i.e. whether the window is in full screen mode or not.
export function WindowIsFullscreen(): Promise<boolean>;
// [WindowSetSize](https://wails.io/docs/reference/runtime/window#windowsetsize)
// Sets the width and height of the window.
export function WindowSetSize(width: number, height: number): Promise<Size>;
// [WindowGetSize](https://wails.io/docs/reference/runtime/window#windowgetsize)
// Gets the width and height of the window.
export function WindowGetSize(): Promise<Size>;
// [WindowSetMaxSize](https://wails.io/docs/reference/runtime/window#windowsetmaxsize)
// Sets the maximum window size. Will resize the window if the window is currently larger than the given dimensions.
// Setting a size of 0,0 will disable this constraint.
export function WindowSetMaxSize(width: number, height: number): void;
// [WindowSetMinSize](https://wails.io/docs/reference/runtime/window#windowsetminsize)
// Sets the minimum window size. Will resize the window if the window is currently smaller than the given dimensions.
// Setting a size of 0,0 will disable this constraint.
export function WindowSetMinSize(width: number, height: number): void;
// [WindowSetPosition](https://wails.io/docs/reference/runtime/window#windowsetposition)
// Sets the window position relative to the monitor the window is currently on.
export function WindowSetPosition(x: number, y: number): void;
// [WindowGetPosition](https://wails.io/docs/reference/runtime/window#windowgetposition)
// Gets the window position relative to the monitor the window is currently on.
export function WindowGetPosition(): Promise<Position>;
// [WindowHide](https://wails.io/docs/reference/runtime/window#windowhide)
// Hides the window.
export function WindowHide(): void;
// [WindowShow](https://wails.io/docs/reference/runtime/window#windowshow)
// Shows the window, if it is currently hidden.
export function WindowShow(): void;
// [WindowMaximise](https://wails.io/docs/reference/runtime/window#windowmaximise)
// Maximises the window to fill the screen.
export function WindowMaximise(): void;
// [WindowToggleMaximise](https://wails.io/docs/reference/runtime/window#windowtogglemaximise)
// Toggles between Maximised and UnMaximised.
export function WindowToggleMaximise(): void;
// [WindowUnmaximise](https://wails.io/docs/reference/runtime/window#windowunmaximise)
// Restores the window to the dimensions and position prior to maximising.
export function WindowUnmaximise(): void;
// [WindowIsMaximised](https://wails.io/docs/reference/runtime/window#windowismaximised)
// Returns the state of the window, i.e. whether the window is maximised or not.
export function WindowIsMaximised(): Promise<boolean>;
// [WindowMinimise](https://wails.io/docs/reference/runtime/window#windowminimise)
// Minimises the window.
export function WindowMinimise(): void;
// [WindowUnminimise](https://wails.io/docs/reference/runtime/window#windowunminimise)
// Restores the window to the dimensions and position prior to minimising.
export function WindowUnminimise(): void;
// [WindowIsMinimised](https://wails.io/docs/reference/runtime/window#windowisminimised)
// Returns the state of the window, i.e. whether the window is minimised or not.
export function WindowIsMinimised(): Promise<boolean>;
// [WindowIsNormal](https://wails.io/docs/reference/runtime/window#windowisnormal)
// Returns the state of the window, i.e. whether the window is normal or not.
export function WindowIsNormal(): Promise<boolean>;
// [WindowSetBackgroundColour](https://wails.io/docs/reference/runtime/window#windowsetbackgroundcolour)
// Sets the background colour of the window to the given RGBA colour definition. This colour will show through for all transparent pixels.
export function WindowSetBackgroundColour(R: number, G: number, B: number, A: number): void;
// [ScreenGetAll](https://wails.io/docs/reference/runtime/window#screengetall)
// Gets the all screens. Call this anew each time you want to refresh data from the underlying windowing system.
export function ScreenGetAll(): Promise<Screen[]>;
// [BrowserOpenURL](https://wails.io/docs/reference/runtime/browser#browseropenurl)
// Opens the given URL in the system browser.
export function BrowserOpenURL(url: string): void;
// [Environment](https://wails.io/docs/reference/runtime/intro#environment)
// Returns information about the environment
export function Environment(): Promise<EnvironmentInfo>;
// [Quit](https://wails.io/docs/reference/runtime/intro#quit)
// Quits the application.
export function Quit(): void;
// [Hide](https://wails.io/docs/reference/runtime/intro#hide)
// Hides the application.
export function Hide(): void;
// [Show](https://wails.io/docs/reference/runtime/intro#show)
// Shows the application.
export function Show(): void;
// [ClipboardGetText](https://wails.io/docs/reference/runtime/clipboard#clipboardgettext)
// Returns the current text stored on clipboard
export function ClipboardGetText(): Promise<string>;
// [ClipboardSetText](https://wails.io/docs/reference/runtime/clipboard#clipboardsettext)
// Sets a text on the clipboard
export function ClipboardSetText(text: string): Promise<boolean>;
// [OnFileDrop](https://wails.io/docs/reference/runtime/draganddrop#onfiledrop)
// OnFileDrop listens to drag and drop events and calls the callback with the coordinates of the drop and an array of path strings.
export function OnFileDrop(callback: (x: number, y: number ,paths: string[]) => void, useDropTarget: boolean) :void
// [OnFileDropOff](https://wails.io/docs/reference/runtime/draganddrop#dragandddropoff)
// OnFileDropOff removes the drag and drop listeners and handlers.
export function OnFileDropOff() :void
// Check if the file path resolver is available
export function CanResolveFilePaths(): boolean;
// Resolves file paths for an array of files
export function ResolveFilePaths(files: File[]): void

View File

@@ -1,238 +0,0 @@
/*
_ __ _ __
| | / /___ _(_) /____
| | /| / / __ `/ / / ___/
| |/ |/ / /_/ / / (__ )
|__/|__/\__,_/_/_/____/
The electron alternative for Go
(c) Lea Anthony 2019-present
*/
export function LogPrint(message) {
window.runtime.LogPrint(message);
}
export function LogTrace(message) {
window.runtime.LogTrace(message);
}
export function LogDebug(message) {
window.runtime.LogDebug(message);
}
export function LogInfo(message) {
window.runtime.LogInfo(message);
}
export function LogWarning(message) {
window.runtime.LogWarning(message);
}
export function LogError(message) {
window.runtime.LogError(message);
}
export function LogFatal(message) {
window.runtime.LogFatal(message);
}
export function EventsOnMultiple(eventName, callback, maxCallbacks) {
return window.runtime.EventsOnMultiple(eventName, callback, maxCallbacks);
}
export function EventsOn(eventName, callback) {
return EventsOnMultiple(eventName, callback, -1);
}
export function EventsOff(eventName, ...additionalEventNames) {
return window.runtime.EventsOff(eventName, ...additionalEventNames);
}
export function EventsOnce(eventName, callback) {
return EventsOnMultiple(eventName, callback, 1);
}
export function EventsEmit(eventName) {
let args = [eventName].slice.call(arguments);
return window.runtime.EventsEmit.apply(null, args);
}
export function WindowReload() {
window.runtime.WindowReload();
}
export function WindowReloadApp() {
window.runtime.WindowReloadApp();
}
export function WindowSetAlwaysOnTop(b) {
window.runtime.WindowSetAlwaysOnTop(b);
}
export function WindowSetSystemDefaultTheme() {
window.runtime.WindowSetSystemDefaultTheme();
}
export function WindowSetLightTheme() {
window.runtime.WindowSetLightTheme();
}
export function WindowSetDarkTheme() {
window.runtime.WindowSetDarkTheme();
}
export function WindowCenter() {
window.runtime.WindowCenter();
}
export function WindowSetTitle(title) {
window.runtime.WindowSetTitle(title);
}
export function WindowFullscreen() {
window.runtime.WindowFullscreen();
}
export function WindowUnfullscreen() {
window.runtime.WindowUnfullscreen();
}
export function WindowIsFullscreen() {
return window.runtime.WindowIsFullscreen();
}
export function WindowGetSize() {
return window.runtime.WindowGetSize();
}
export function WindowSetSize(width, height) {
window.runtime.WindowSetSize(width, height);
}
export function WindowSetMaxSize(width, height) {
window.runtime.WindowSetMaxSize(width, height);
}
export function WindowSetMinSize(width, height) {
window.runtime.WindowSetMinSize(width, height);
}
export function WindowSetPosition(x, y) {
window.runtime.WindowSetPosition(x, y);
}
export function WindowGetPosition() {
return window.runtime.WindowGetPosition();
}
export function WindowHide() {
window.runtime.WindowHide();
}
export function WindowShow() {
window.runtime.WindowShow();
}
export function WindowMaximise() {
window.runtime.WindowMaximise();
}
export function WindowToggleMaximise() {
window.runtime.WindowToggleMaximise();
}
export function WindowUnmaximise() {
window.runtime.WindowUnmaximise();
}
export function WindowIsMaximised() {
return window.runtime.WindowIsMaximised();
}
export function WindowMinimise() {
window.runtime.WindowMinimise();
}
export function WindowUnminimise() {
window.runtime.WindowUnminimise();
}
export function WindowSetBackgroundColour(R, G, B, A) {
window.runtime.WindowSetBackgroundColour(R, G, B, A);
}
export function ScreenGetAll() {
return window.runtime.ScreenGetAll();
}
export function WindowIsMinimised() {
return window.runtime.WindowIsMinimised();
}
export function WindowIsNormal() {
return window.runtime.WindowIsNormal();
}
export function BrowserOpenURL(url) {
window.runtime.BrowserOpenURL(url);
}
export function Environment() {
return window.runtime.Environment();
}
export function Quit() {
window.runtime.Quit();
}
export function Hide() {
window.runtime.Hide();
}
export function Show() {
window.runtime.Show();
}
export function ClipboardGetText() {
return window.runtime.ClipboardGetText();
}
export function ClipboardSetText(text) {
return window.runtime.ClipboardSetText(text);
}
/**
* Callback for OnFileDrop returns a slice of file path strings when a drop is finished.
*
* @export
* @callback OnFileDropCallback
* @param {number} x - x coordinate of the drop
* @param {number} y - y coordinate of the drop
* @param {string[]} paths - A list of file paths.
*/
/**
* OnFileDrop listens to drag and drop events and calls the callback with the coordinates of the drop and an array of path strings.
*
* @export
* @param {OnFileDropCallback} callback - Callback for OnFileDrop returns a slice of file path strings when a drop is finished.
* @param {boolean} [useDropTarget=true] - Only call the callback when the drop finished on an element that has the drop target style. (--wails-drop-target)
*/
export function OnFileDrop(callback, useDropTarget) {
return window.runtime.OnFileDrop(callback, useDropTarget);
}
/**
* OnFileDropOff removes the drag and drop listeners and handlers.
*/
export function OnFileDropOff() {
return window.runtime.OnFileDropOff();
}
export function CanResolveFilePaths() {
return window.runtime.CanResolveFilePaths();
}
export function ResolveFilePaths(files) {
return window.runtime.ResolveFilePaths(files);
}

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.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
View File

@@ -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=

Binary file not shown.

Before

Width:  |  Height:  |  Size: 407 KiB

After

Width:  |  Height:  |  Size: 425 KiB

21
main.go
View File

@@ -6,6 +6,9 @@ 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"
"github.com/wailsapp/wails/v2/pkg/options/mac"
"git.taurusxin.com/taurusxin/ncmdump-gui/utils"
) )
//go:embed all:frontend/dist //go:embed all:frontend/dist
@@ -15,8 +18,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 +35,17 @@ func main() {
OnStartup: app.startup, OnStartup: app.startup,
Bind: []interface{}{ Bind: []interface{}{
app, app,
config_manager,
},
DragAndDrop: &options.DragAndDrop{
EnableFileDrop: true,
DisableWebViewDrop: true,
},
Mac: &mac.Options{
About: &mac.AboutInfo{
Title: "ncmdump-gui",
Message: "Copyright © 2025 TaurusXin",
},
}, },
}) })

81
utils/config_manager.go Normal file
View File

@@ -0,0 +1,81 @@
package utils
import (
"encoding/json"
"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)
if err != nil {
return nil
}
return preference
}

View File

@@ -13,7 +13,7 @@
"info": { "info": {
"companyName": "TaurusXin", "companyName": "TaurusXin",
"productName": "ncmdump-gui", "productName": "ncmdump-gui",
"productVersion": "1.0.1", "productVersion": "1.1.0",
"copyright": "Copyright © 2025 TaurusXin", "copyright": "Copyright © 2025 TaurusXin",
"comments": "Convert ncm to mp3/flac" "comments": "Convert ncm to mp3/flac"
} }