mani-game-engine 1.0.0-pre.4 → 1.0.0-pre.41
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -21
- package/README.md +69 -69
- package/lib/clock.d.ts +55 -41
- package/lib/clock.js +117 -77
- package/lib/clock.js.map +1 -1
- package/lib/ecsInjector.d.ts +46 -21
- package/lib/ecsInjector.js +218 -124
- package/lib/ecsInjector.js.map +1 -1
- package/lib/entity.d.ts +32 -22
- package/lib/entity.js +159 -83
- package/lib/entity.js.map +1 -1
- package/lib/gameEngine.d.ts +89 -66
- package/lib/gameEngine.js +361 -272
- package/lib/gameEngine.js.map +1 -1
- package/lib/index.d.ts +12 -6
- package/lib/index.js +35 -5
- package/lib/index.js.map +1 -1
- package/lib/injector.d.ts +121 -0
- package/lib/injector.js +290 -0
- package/lib/injector.js.map +1 -0
- package/lib/scope/scopeContext.d.ts +69 -0
- package/lib/scope/scopeContext.js +282 -0
- package/lib/scope/scopeContext.js.map +1 -0
- package/lib/systemContext.d.ts +15 -0
- package/lib/systemContext.js +39 -0
- package/lib/systemContext.js.map +1 -0
- package/lib/types.d.ts +80 -54
- package/lib/types.js +19 -0
- package/lib/types.js.map +1 -1
- package/lib/utils/map2k.d.ts +6 -6
- package/lib/utils/map2k.js +43 -39
- package/lib/utils/map2k.js.map +1 -1
- package/package.json +39 -45
- package/src/clock.ts +163 -102
- package/src/ecsInjector.ts +274 -164
- package/src/entity.ts +193 -113
- package/src/gameEngine.ts +463 -350
- package/src/index.ts +32 -9
- package/src/injector.ts +364 -0
- package/src/scope/scopeContext.ts +359 -0
- package/src/systemContext.ts +40 -0
- package/src/types.ts +95 -70
- package/src/utils/map2k.ts +52 -52
package/src/ecsInjector.ts
CHANGED
|
@@ -1,164 +1,274 @@
|
|
|
1
|
-
// TODO: what if systems for entity without components are created?
|
|
2
|
-
import {
|
|
3
|
-
import {
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
type
|
|
9
|
-
|
|
10
|
-
type
|
|
11
|
-
type
|
|
12
|
-
|
|
13
|
-
type
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
export const
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
};
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
1
|
+
// TODO: what if systems for entity without components are created?
|
|
2
|
+
import {Signal} from 'mani-signal';
|
|
3
|
+
import {Class, createDependencyAnnotation, Dependency, ID, Injector, putIfAbsent, ResolverContext, ResolverFunction} from './injector';
|
|
4
|
+
import {entitySignalHandlers, SystemContext} from './systemContext';
|
|
5
|
+
import {Entity} from './entity';
|
|
6
|
+
|
|
7
|
+
type ComponentClass = Class;
|
|
8
|
+
type EntityClass = Class;
|
|
9
|
+
|
|
10
|
+
type SystemResolvers<T extends Class> = [T, ResolverFunction[]]
|
|
11
|
+
type ComponentDependency = Dependency & { kind: 'component'; index: number; type: Class; };
|
|
12
|
+
type DynamicComponentDependency = Dependency & { kind: 'dynamic'; index: number; type: Class; };
|
|
13
|
+
type EntityDependency = Dependency & { kind: 'entity'; index: number; };
|
|
14
|
+
type ContextDependency = Dependency & { kind: 'context'; index: number; };
|
|
15
|
+
type SignalDependency = Dependency & { kind: 'signal'; index: number; id: ID; };
|
|
16
|
+
type EntitySignalDependency = Dependency & { kind: 'entitySignal'; index: number; id: ID; };
|
|
17
|
+
|
|
18
|
+
type EntityResolverContext = ResolverContext & { entityClass: Object; };
|
|
19
|
+
|
|
20
|
+
export const GetComponent = createDependencyAnnotation((type, index): ComponentDependency => ({kind: 'component', type, index}));
|
|
21
|
+
export const GetDynamicComponent = createDependencyAnnotation((type, index): DynamicComponentDependency => ({
|
|
22
|
+
kind: 'dynamic',
|
|
23
|
+
type,
|
|
24
|
+
index,
|
|
25
|
+
}));
|
|
26
|
+
export const GetEntity = createDependencyAnnotation((_type, index): EntityDependency => ({kind: 'entity', index}));
|
|
27
|
+
export const GetContext = createDependencyAnnotation((_type, index): ContextDependency => ({kind: 'context', index}));
|
|
28
|
+
export const GetSignal = (id: ID) => createDependencyAnnotation((_type, index): SignalDependency => ({kind: 'signal', index, id}));
|
|
29
|
+
export const GetEntitySignal = (id: ID) => createDependencyAnnotation((_type, index): EntitySignalDependency => ({
|
|
30
|
+
kind: 'entitySignal',
|
|
31
|
+
index,
|
|
32
|
+
id,
|
|
33
|
+
}));
|
|
34
|
+
export const EntityComponent = (target: object, propertyKey: string): any => {
|
|
35
|
+
const entityClass = target.constructor;
|
|
36
|
+
const componentClass = Reflect.getMetadata('design:type', target, propertyKey);
|
|
37
|
+
if (componentClass === Object) {
|
|
38
|
+
throw new Error(`Object component type not allowed. Forgot to specify type of ${entityClass.name}.${propertyKey}?`);
|
|
39
|
+
}
|
|
40
|
+
const componentSet = putIfAbsent(EcsInjector.entityComponents, entityClass, () => new Map<ComponentClass, string>());
|
|
41
|
+
componentSet.set(componentClass, propertyKey);
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
export const signalHandlers = new Map<Object, [string, ID][]>();
|
|
45
|
+
export const staticSignalHandlers = new Map<Object, [string, ID][]>();
|
|
46
|
+
export const OnSignal = (id: ID) => (target: any, propertyKey: string, descriptor: PropertyDescriptor) => {
|
|
47
|
+
if (target instanceof Function) {
|
|
48
|
+
let mappingList = putIfAbsent(staticSignalHandlers, target, (): [string, ID][] => []);
|
|
49
|
+
mappingList.push([propertyKey, id]);
|
|
50
|
+
} else {
|
|
51
|
+
let mappingList = putIfAbsent(signalHandlers, target.constructor, (): [string, ID][] => []);
|
|
52
|
+
mappingList.push([propertyKey, id]);
|
|
53
|
+
}
|
|
54
|
+
};
|
|
55
|
+
|
|
56
|
+
const extractDependenciesWithType = <T extends Dependency>(type: string) => {
|
|
57
|
+
const result = new Map<Class, T[]>();
|
|
58
|
+
Injector.dependencyMap.forEach((dependency, system) => {
|
|
59
|
+
putIfAbsent(result, system, () => [...dependency.filter((dependency) => dependency.kind === type)]);
|
|
60
|
+
});
|
|
61
|
+
return result;
|
|
62
|
+
};
|
|
63
|
+
|
|
64
|
+
const getEntityClassesForComponentDependencies = (componentTypes: Class[]): Class[] => {
|
|
65
|
+
// TODO: refactor class, use map, filter etc...
|
|
66
|
+
const result = [];
|
|
67
|
+
for (const [entityClass, componentMap] of EcsInjector.entityComponents) {
|
|
68
|
+
let allDependenciesMet = true;
|
|
69
|
+
for (const componentType of componentTypes) {
|
|
70
|
+
if (!componentMap.has(componentType)) {
|
|
71
|
+
allDependenciesMet = false;
|
|
72
|
+
break;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
if (allDependenciesMet) {
|
|
76
|
+
result.push(entityClass);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
return result;
|
|
80
|
+
};
|
|
81
|
+
|
|
82
|
+
const componentResolver = (context: ResolverContext, dependency: Dependency): ResolverFunction => {
|
|
83
|
+
const kind = context.kind;
|
|
84
|
+
const type = context.type;
|
|
85
|
+
if (kind !== 'system') {
|
|
86
|
+
throw new Error(`Could not resolve Component ${type.name}. @GetComponent only allowed in system scope.`);
|
|
87
|
+
}
|
|
88
|
+
const entityClass = (context as EntityResolverContext).entityClass;
|
|
89
|
+
const key = EcsInjector.entityComponents.get(<any>entityClass as Class)!.get((dependency as ComponentDependency).type);
|
|
90
|
+
return (context: SystemContext) => (context.entity as any)[key!];
|
|
91
|
+
};
|
|
92
|
+
|
|
93
|
+
const dynamicComponentResolver = (context: ResolverContext, dependency: Dependency): ResolverFunction => {
|
|
94
|
+
const kind = context.kind;
|
|
95
|
+
const type = context.type;
|
|
96
|
+
if (kind !== 'system') {
|
|
97
|
+
throw new Error(`Could not resolve Component ${type.name}. @GetComponent only allowed in system scope.`);
|
|
98
|
+
}
|
|
99
|
+
// const entityClass = (context as EntityResolverContext).entityClass;
|
|
100
|
+
// const key = entityComponents.get(<any>entityClass as Class)!.get((dependency as ComponentDependency).type);
|
|
101
|
+
return (context: SystemContext) => context.entity.getDynamicComponent((dependency as DynamicComponentDependency).type);
|
|
102
|
+
};
|
|
103
|
+
|
|
104
|
+
const entityResolver = ({type, kind}: ResolverContext, _dependency: Dependency): ResolverFunction => {
|
|
105
|
+
if (kind !== 'system') {
|
|
106
|
+
throw new Error(`Could not resolve Entity in ${type.name}. @GetEntity only allowed in system scope.`);
|
|
107
|
+
}
|
|
108
|
+
return (context: SystemContext) => context.entity;
|
|
109
|
+
};
|
|
110
|
+
|
|
111
|
+
const entitySignalResolver = ({type, kind}: ResolverContext, dependency: Dependency): ResolverFunction => {
|
|
112
|
+
if (kind !== 'system') {
|
|
113
|
+
throw new Error(`Could not resolve Entity in ${type.name}. @GetEntity only allowed in system scope.`);
|
|
114
|
+
}
|
|
115
|
+
return (context: SystemContext) => (<Entity>context.entity).getSignal((<EntitySignalDependency>dependency).id);
|
|
116
|
+
};
|
|
117
|
+
|
|
118
|
+
const contextResolver = ({type, kind}: ResolverContext, _dependency: Dependency): ResolverFunction => {
|
|
119
|
+
if (kind !== 'system') {
|
|
120
|
+
throw new Error(`Could not resolve Context in ${type.name}. @GetContext only allowed in system scope.`);
|
|
121
|
+
}
|
|
122
|
+
return (context: SystemContext) => context;
|
|
123
|
+
};
|
|
124
|
+
|
|
125
|
+
// const isSignalResolver = (dependency: Dependency): dependency is SignalDependency => dependency.kind === 'signal';
|
|
126
|
+
|
|
127
|
+
export class EcsInjector<SystemClass extends Class = Class> extends Injector {
|
|
128
|
+
static warnNoMatchingEntities = true;
|
|
129
|
+
static entityComponents = new Map<EntityClass, Map<ComponentClass, string>>();
|
|
130
|
+
protected componentDependencyMap = extractDependenciesWithType<ComponentDependency>('component');
|
|
131
|
+
protected dynamicComponentDependencyMap = extractDependenciesWithType<DynamicComponentDependency>('dynamic');
|
|
132
|
+
protected entitySystemMap: Map<Class, SystemClass[]>;
|
|
133
|
+
protected dynamicComponentSystemMap: Map<Class, SystemClass[]>;
|
|
134
|
+
protected entitySystemResolverTuples: Map<Class, Map<SystemClass, ResolverFunction[]>>;
|
|
135
|
+
private signalMap: Map<ID, Signal>;
|
|
136
|
+
|
|
137
|
+
constructor(parent?: EcsInjector<SystemClass>) {
|
|
138
|
+
super(parent);
|
|
139
|
+
this.signalMap = parent ? parent.signalMap : new Map<ID, Signal>();
|
|
140
|
+
|
|
141
|
+
// TODO: do we need to have this maps in every child injector?
|
|
142
|
+
this.entitySystemMap = new Map<EntityClass, SystemClass[]>();
|
|
143
|
+
this.dynamicComponentSystemMap = new Map<EntityClass, SystemClass[]>();
|
|
144
|
+
this.entitySystemResolverTuples = new Map<Class, Map<SystemClass, ResolverFunction[]>>();
|
|
145
|
+
|
|
146
|
+
this.map(EcsInjector).toValue(this);
|
|
147
|
+
|
|
148
|
+
if (!parent) {
|
|
149
|
+
// only add extension resolvers to the main/parent injector
|
|
150
|
+
this.addExtensionResolver('entity', entityResolver);
|
|
151
|
+
this.addExtensionResolver('component', componentResolver);
|
|
152
|
+
this.addExtensionResolver('dynamic', dynamicComponentResolver);
|
|
153
|
+
this.addExtensionResolver('context', contextResolver);
|
|
154
|
+
|
|
155
|
+
const signalResolver = ({kind}: ResolverContext, dependency: Dependency): ResolverFunction => {
|
|
156
|
+
const signal = putIfAbsent(this.signalMap, (<SignalDependency>dependency).id, () => new Signal());
|
|
157
|
+
return (entity: any) => signal;
|
|
158
|
+
};
|
|
159
|
+
this.addExtensionResolver('signal', signalResolver);
|
|
160
|
+
this.addExtensionResolver('entitySignal', entitySignalResolver);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
//TODO: this needs some overthinking...
|
|
165
|
+
newSignalScope() {
|
|
166
|
+
this.signalMap = new Map<ID, Signal>();
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
registerSystem<T extends SystemClass>(systemClass: T) {
|
|
170
|
+
// TODO: check if system is already mapped
|
|
171
|
+
const componentDependencies = this.componentDependencyMap.get(systemClass)?.map(it => it.type) || [];
|
|
172
|
+
const dynamicComponentDependencies = this.dynamicComponentDependencyMap.get(systemClass) || [];
|
|
173
|
+
|
|
174
|
+
for (const dynamicComponentDependency of dynamicComponentDependencies) {
|
|
175
|
+
const systemClassesForDynamicComponent = putIfAbsent(this.dynamicComponentSystemMap, dynamicComponentDependency.type, () => [] as SystemClass[]);
|
|
176
|
+
systemClassesForDynamicComponent.push(systemClass);
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
if (componentDependencies.length === 0) {
|
|
180
|
+
throw new Error(`${systemClass.name} needs at least one component dependency.`);
|
|
181
|
+
}
|
|
182
|
+
const entityClasses = getEntityClassesForComponentDependencies(componentDependencies);
|
|
183
|
+
if (entityClasses.length === 0 && EcsInjector.warnNoMatchingEntities) {
|
|
184
|
+
console.warn(`System '${systemClass.name}' has no matching entities.`);
|
|
185
|
+
return;
|
|
186
|
+
}
|
|
187
|
+
for (let entityClass of entityClasses) {
|
|
188
|
+
const systemClasses = putIfAbsent(this.entitySystemMap, entityClass, (): SystemClass[] => []);
|
|
189
|
+
systemClasses.push(systemClass);
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
createSystemsForDynamicComponents(dynamicComponentClass: Class, entity: Entity) {
|
|
194
|
+
const potentialSystems = this.getSystemResolverTuples(entity);
|
|
195
|
+
const systems = this.dynamicComponentSystemMap.get(dynamicComponentClass)!;
|
|
196
|
+
|
|
197
|
+
const finalSystemResolverEntries = new Map<SystemClass, ResolverFunction[]>();
|
|
198
|
+
for (const system of systems) {
|
|
199
|
+
const resolverTuples = potentialSystems.get(system);
|
|
200
|
+
if (resolverTuples) {
|
|
201
|
+
finalSystemResolverEntries.set(system, resolverTuples);
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
return this.createSystems(finalSystemResolverEntries, entity);
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
createSystemsForEntity(entity: Entity): Set<SystemContext> {
|
|
209
|
+
const systemResolverTuples = this.getSystemResolverTuples(entity);
|
|
210
|
+
return this.createSystems(systemResolverTuples, entity);
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
private createSystems(systemResolverMap: Map<SystemClass, ResolverFunction[]>, entity: Entity) {
|
|
214
|
+
const systemInstances = new Set<SystemContext>();
|
|
215
|
+
for (const [system, resolver] of systemResolverMap) {
|
|
216
|
+
const args = new Array(resolver.length);
|
|
217
|
+
|
|
218
|
+
const dynamicDependencies = this.dynamicComponentDependencyMap.get(system) || [];
|
|
219
|
+
const hasUnfulfilledDynamicDependencies = dynamicDependencies.some((dependency, index) => {
|
|
220
|
+
return !entity.hasDynamicComponentClass(dependency.type);
|
|
221
|
+
});
|
|
222
|
+
if (hasUnfulfilledDynamicDependencies) {
|
|
223
|
+
continue;
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
const systemContext = new SystemContext<InstanceType<SystemClass>>(entity);
|
|
227
|
+
|
|
228
|
+
for (let i = 0; i < args.length; i++) {
|
|
229
|
+
args[i] = resolver[i](systemContext);
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
const newSystem = new system(...args); // TODO: new class(...args) is very slow in firefox :/
|
|
233
|
+
|
|
234
|
+
// map entity signals
|
|
235
|
+
const entitySignals = entitySignalHandlers.get(system);
|
|
236
|
+
if (entitySignals?.length) {
|
|
237
|
+
for (const [propertyKey, id] of entitySignals) {
|
|
238
|
+
entity.getSignal(id).add(newSystem[propertyKey], newSystem);
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
// friend class :)
|
|
243
|
+
(systemContext as any).system = newSystem;
|
|
244
|
+
|
|
245
|
+
systemInstances.add(systemContext);
|
|
246
|
+
}
|
|
247
|
+
return systemInstances;
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
private getSystemResolverTuples(entity: Entity) {
|
|
251
|
+
return putIfAbsent(this.entitySystemResolverTuples, entity.constructor, (): Map<SystemClass, ResolverFunction[]> => {
|
|
252
|
+
const systems = this.entitySystemMap.get(entity.constructor as Class);
|
|
253
|
+
if (!systems) return [] as any;
|
|
254
|
+
|
|
255
|
+
// const result: SystemResolvers<SystemClass>[] = [];
|
|
256
|
+
const result = new Map<SystemClass, ResolverFunction[]>();
|
|
257
|
+
for (const systemClass of systems) {
|
|
258
|
+
result.set(systemClass, this.createResolverArray({type: systemClass, kind: 'system', entityClass: entity.constructor}));
|
|
259
|
+
}
|
|
260
|
+
return result;
|
|
261
|
+
});
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
getSignal<T>(id: ID): Signal<T> {
|
|
265
|
+
return putIfAbsent(this.signalMap, id, () => new Signal<any>());
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
dispose() {
|
|
269
|
+
// TODO: implement dispose in Injector
|
|
270
|
+
for (const [key, signal] of this.signalMap) {
|
|
271
|
+
signal.detachAll();
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
}
|