Files
esengine/packages/editor/plugins/sprite-editor/src/index.ts
YHH 155411e743 refactor: reorganize package structure and decouple framework packages (#338)
* refactor: reorganize package structure and decouple framework packages

## Package Structure Reorganization
- Reorganized 55 packages into categorized subdirectories:
  - packages/framework/ - Generic framework (Laya/Cocos compatible)
  - packages/engine/ - ESEngine core modules
  - packages/rendering/ - Rendering modules (WASM dependent)
  - packages/physics/ - Physics modules
  - packages/streaming/ - World streaming
  - packages/network-ext/ - Network extensions
  - packages/editor/ - Editor framework and plugins
  - packages/rust/ - Rust WASM engine
  - packages/tools/ - Build tools and SDK

## Framework Package Decoupling
- Decoupled behavior-tree and blueprint packages from ESEngine dependencies
- Created abstracted interfaces (IBTAssetManager, IBehaviorTreeAssetContent)
- ESEngine-specific code moved to esengine/ subpath exports
- Framework packages now usable with Cocos/Laya without ESEngine

## CI Configuration
- Updated CI to only type-check and lint framework packages
- Added type-check:framework and lint:framework scripts

## Breaking Changes
- Package import paths changed due to directory reorganization
- ESEngine integrations now use subpath imports (e.g., '@esengine/behavior-tree/esengine')

* fix: update es-engine file path after directory reorganization

* docs: update README to focus on framework over engine

* ci: only build framework packages, remove Rust/WASM dependencies

* fix: remove esengine subpath from behavior-tree and blueprint builds

ESEngine integration code will only be available in full engine builds.
Framework packages are now purely engine-agnostic.

* fix: move network-protocols to framework, build both in CI

* fix: update workflow paths from packages/core to packages/framework/core

* fix: exclude esengine folder from type-check in behavior-tree and blueprint

* fix: update network tsconfig references to new paths

* fix: add test:ci:framework to only test framework packages in CI

* fix: only build core and math npm packages in CI

* fix: exclude test files from CodeQL and fix string escaping security issue
2025-12-26 14:50:35 +08:00

195 lines
5.9 KiB
TypeScript

/**
* @esengine/sprite-editor
*
* Editor support for @esengine/sprite - inspectors, field editors, and entity templates
* 精灵编辑器支持 - 检视器、字段编辑器和实体模板
*/
import type { Entity, ServiceContainer } from '@esengine/ecs-framework';
import { Core } from '@esengine/ecs-framework';
import type {
IEditorModuleLoader,
EntityCreationTemplate,
IEditorPlugin,
ModuleManifest
} from '@esengine/editor-core';
import {
EntityStoreService,
MessageHub,
EditorComponentRegistry,
ComponentInspectorRegistry
} from '@esengine/editor-core';
import { TransformComponent } from '@esengine/engine-core';
// Runtime imports from @esengine/sprite
import {
SpriteComponent,
SpriteAnimatorComponent,
SpriteRuntimeModule
} from '@esengine/sprite';
// Inspector
import { SpriteComponentInspector } from './SpriteComponentInspector';
// Export inspector
export { SpriteComponentInspector } from './SpriteComponentInspector';
/**
* 精灵编辑器模块
* Sprite Editor Module
*/
export class SpriteEditorModule implements IEditorModuleLoader {
async install(services: ServiceContainer): Promise<void> {
// 注册组件检查器 | Register component inspectors
const componentInspectorRegistry = services.tryResolve(ComponentInspectorRegistry);
if (componentInspectorRegistry) {
componentInspectorRegistry.register(new SpriteComponentInspector());
}
// 注册 Sprite 组件到编辑器组件注册表 | Register Sprite components to editor component registry
const componentRegistry = services.resolve(EditorComponentRegistry);
if (componentRegistry) {
const spriteComponents = [
{
name: 'Sprite',
type: SpriteComponent,
category: 'components.category.rendering',
description: '2D sprite rendering component',
icon: 'Image'
},
{
name: 'SpriteAnimator',
type: SpriteAnimatorComponent,
category: 'components.category.rendering',
description: 'Sprite frame animation component',
icon: 'Film'
},
];
for (const comp of spriteComponents) {
componentRegistry.register({
name: comp.name,
type: comp.type,
category: comp.category,
description: comp.description,
icon: comp.icon
});
}
}
}
async uninstall(): Promise<void> {
// Nothing to cleanup
}
getEntityCreationTemplates(): EntityCreationTemplate[] {
return [
// Sprite Entity
{
id: 'create-sprite',
label: 'Sprite',
icon: 'Image',
category: 'rendering',
order: 100,
create: (): number => {
return this.createSpriteEntity('Sprite');
}
},
// Animated Sprite Entity
{
id: 'create-animated-sprite',
label: 'Animated Sprite',
icon: 'Film',
category: 'rendering',
order: 101,
create: (): number => {
return this.createSpriteEntity('AnimatedSprite', (entity) => {
const animator = new SpriteAnimatorComponent();
animator.autoPlay = true;
entity.addComponent(animator);
});
}
},
];
}
/**
* 创建 Sprite 实体的辅助方法
* Helper method to create Sprite entity
*/
private createSpriteEntity(baseName: string, configure?: (entity: Entity) => void): number {
const scene = Core.scene;
if (!scene) {
throw new Error('Scene not available');
}
const entityStore = Core.services.resolve(EntityStoreService);
const messageHub = Core.services.resolve(MessageHub);
if (!entityStore || !messageHub) {
throw new Error('EntityStoreService or MessageHub not available');
}
const existingCount = entityStore.getAllEntities()
.filter((e: Entity) => e.name.startsWith(baseName)).length;
const entityName = existingCount > 0 ? `${baseName} ${existingCount + 1}` : baseName;
const entity = scene.createEntity(entityName);
// Add Transform component
const transform = new TransformComponent();
entity.addComponent(transform);
// Add Sprite component
const sprite = new SpriteComponent();
entity.addComponent(sprite);
if (configure) {
configure(entity);
}
entityStore.addEntity(entity);
messageHub.publish('entity:added', { entity });
messageHub.publish('scene:modified', {});
entityStore.selectEntity(entity);
return entity.id;
}
}
export const spriteEditorModule = new SpriteEditorModule();
/**
* Sprite 插件清单
* Sprite Plugin Manifest
*/
const manifest: ModuleManifest = {
id: '@esengine/sprite',
name: '@esengine/sprite',
displayName: 'Sprite',
version: '1.0.0',
description: 'Sprite and animation components for 2D rendering',
category: 'Rendering',
isCore: false,
defaultEnabled: true,
isEngineModule: true,
dependencies: ['engine-core'],
exports: {
components: ['SpriteComponent', 'SpriteAnimatorComponent'],
systems: ['SpriteRenderSystem']
}
};
/**
* 完整的 Sprite 插件(运行时 + 编辑器)
* Complete Sprite Plugin (runtime + editor)
*/
export const SpritePlugin: IEditorPlugin = {
manifest,
runtimeModule: new SpriteRuntimeModule(),
editorModule: spriteEditorModule
};
export default spriteEditorModule;