* feat(i18n): 统一国际化系统架构,支持插件独立翻译 ## 主要改动 ### 核心架构 - 增强 LocaleService,支持插件命名空间翻译扩展 - 新增 editor-runtime/i18n 模块,提供 createPluginLocale/createPluginTranslator - 新增 editor-core/tokens.ts,定义 LocaleServiceToken 等服务令牌 - 改进 PluginAPI 类型安全,使用 ServiceToken<T> 替代 any ### 编辑器本地化 - 扩展 en.ts/zh.ts 翻译文件,覆盖所有 UI 组件 - 新增 es.ts 西班牙语支持 - 重构 40+ 组件使用 useLocale() hook ### 插件本地化系统 - behavior-tree-editor: 新增 locales/ 和 useBTLocale hook - material-editor: 新增 locales/ 和 useMaterialLocale hook - particle-editor: 新增 locales/ 和 useParticleLocale hook - tilemap-editor: 新增 locales/ 和 useTilemapLocale hook - ui-editor: 新增 locales/ 和 useUILocale hook ### 类型安全改进 - 修复 Debug 工具使用公共接口替代 as any - 修复 ChunkStreamingSystem 添加 forEachChunk 公共方法 - 修复 blueprint-editor 移除不必要的向后兼容代码 * fix(behavior-tree-editor): 使用 ServiceToken 模式修复服务解析 - 创建 BehaviorTreeServiceToken 遵循"谁定义接口,谁导出Token"原则 - 使用 ServiceToken.id (symbol) 注册服务到 ServiceContainer - 更新 PluginSDKRegistry.resolveService 支持 ServiceToken 检测 - BehaviorTreeEditorPanel 现在使用类型安全的 PluginAPI.resolve * fix(behavior-tree-editor): 使用 ServiceContainer.resolve 获取类注册的服务 * fix: 修复多个包的依赖和类型问题 - core: EntityDataCollector.getEntityDetails 使用 HierarchySystem 获取父实体 - ui-editor: 添加 @esengine/editor-runtime 依赖 - tilemap-editor: 添加 @esengine/editor-runtime 依赖 - particle-editor: 添加 @esengine/editor-runtime 依赖
101 lines
3.2 KiB
TypeScript
101 lines
3.2 KiB
TypeScript
import { singleton } from 'tsyringe';
|
|
import { open, save } from '@tauri-apps/plugin-dialog';
|
|
import type { IDialog, OpenDialogOptions, SaveDialogOptions } from '@esengine/editor-core';
|
|
|
|
export interface ConfirmDialogData {
|
|
title: string;
|
|
message: string;
|
|
confirmText: string;
|
|
cancelText: string;
|
|
onConfirm: () => void;
|
|
onCancel?: () => void;
|
|
}
|
|
|
|
export interface IDialogExtended extends IDialog {
|
|
setConfirmCallback(callback: (data: ConfirmDialogData) => void): void;
|
|
setLocale(locale: string): void;
|
|
}
|
|
|
|
const dialogTranslations = {
|
|
en: { confirm: 'Confirm', cancel: 'Cancel' },
|
|
zh: { confirm: '确定', cancel: '取消' },
|
|
es: { confirm: 'Confirmar', cancel: 'Cancelar' }
|
|
} as const;
|
|
|
|
type LocaleKey = keyof typeof dialogTranslations;
|
|
|
|
@singleton()
|
|
export class TauriDialogService implements IDialogExtended {
|
|
private showConfirmCallback?: (data: ConfirmDialogData) => void;
|
|
private locale: string = 'zh';
|
|
|
|
dispose(): void {
|
|
this.showConfirmCallback = undefined;
|
|
}
|
|
|
|
setConfirmCallback(callback: (data: ConfirmDialogData) => void): void {
|
|
this.showConfirmCallback = callback;
|
|
}
|
|
|
|
setLocale(locale: string): void {
|
|
this.locale = locale;
|
|
}
|
|
|
|
async openDialog(options: OpenDialogOptions): Promise<string | string[] | null> {
|
|
const result = await open({
|
|
multiple: options.multiple || false,
|
|
directory: options.directory || false,
|
|
filters: options.filters,
|
|
defaultPath: options.defaultPath,
|
|
title: options.title
|
|
});
|
|
|
|
return result;
|
|
}
|
|
|
|
async saveDialog(options: SaveDialogOptions): Promise<string | null> {
|
|
const result = await save({
|
|
filters: options.filters,
|
|
defaultPath: options.defaultPath,
|
|
title: options.title
|
|
});
|
|
|
|
return result;
|
|
}
|
|
|
|
async showMessage(title: string, messageText: string, type: 'info' | 'warning' | 'error' = 'info'): Promise<void> {
|
|
console.warn('[TauriDialogService] showMessage not implemented with custom UI, use notification instead');
|
|
}
|
|
|
|
async showConfirm(title: string, messageText: string): Promise<boolean> {
|
|
return new Promise((resolve) => {
|
|
if (this.showConfirmCallback) {
|
|
let resolved = false;
|
|
const handleResolve = (result: boolean) => {
|
|
if (!resolved) {
|
|
resolved = true;
|
|
resolve(result);
|
|
}
|
|
};
|
|
|
|
const localeKey = (this.locale in dialogTranslations ? this.locale : 'en') as LocaleKey;
|
|
const texts = dialogTranslations[localeKey];
|
|
const confirmText = texts.confirm;
|
|
const cancelText = texts.cancel;
|
|
|
|
this.showConfirmCallback({
|
|
title,
|
|
message: messageText,
|
|
confirmText,
|
|
cancelText,
|
|
onConfirm: () => handleResolve(true),
|
|
onCancel: () => handleResolve(false)
|
|
});
|
|
} else {
|
|
console.warn('[TauriDialogService] showConfirmCallback not set');
|
|
resolve(false);
|
|
}
|
|
});
|
|
}
|
|
}
|