@dcl/ecs
Version:
Decentraland ECS
141 lines (140 loc) • 6.41 kB
JavaScript
;
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
return result;
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.createTriggerAreaEventsSystem = void 0;
const components = __importStar(require("../components"));
const entity_1 = require("../engine/entity");
/**
* @internal
*/
function createTriggerAreaEventsSystem(engine) {
const triggerAreaResultComponent = components.TriggerAreaResult(engine);
const entitiesMap = new Map();
function hasCallbacksMap(entity) {
return entitiesMap.has(entity) && entitiesMap.get(entity) !== undefined;
}
function addEntityCallback(entity, triggerType, callback) {
if (hasCallbacksMap(entity)) {
entitiesMap.get(entity).triggerCallbackMap.set(triggerType, callback);
}
else {
entitiesMap.set(entity, {
triggerCallbackMap: new Map([[triggerType, callback]]),
lastConsumedTimestamp: -1
});
}
}
function removeEntityCallback(entity, triggerType) {
if (!entitiesMap.has(entity) || !entitiesMap.get(entity).triggerCallbackMap.has(triggerType))
return;
const triggerCallbackMap = entitiesMap.get(entity).triggerCallbackMap;
triggerCallbackMap.delete(triggerType);
// Remove entity if no more trigger callbacks are registered
if (triggerCallbackMap.size === 0)
entitiesMap.delete(entity);
}
function onTriggerEnter(entity, cb) {
addEntityCallback(entity, 0 /* TriggerAreaEventType.TAET_ENTER */, cb);
}
function removeOnTriggerEnter(entity) {
removeEntityCallback(entity, 0 /* TriggerAreaEventType.TAET_ENTER */);
}
function onTriggerStay(entity, cb) {
addEntityCallback(entity, 1 /* TriggerAreaEventType.TAET_STAY */, cb);
}
function removeOnTriggerStay(entity) {
removeEntityCallback(entity, 1 /* TriggerAreaEventType.TAET_STAY */);
}
function onTriggerExit(entity, cb) {
addEntityCallback(entity, 2 /* TriggerAreaEventType.TAET_EXIT */, cb);
}
function removeOnTriggerExit(entity) {
removeEntityCallback(entity, 2 /* TriggerAreaEventType.TAET_EXIT */);
}
engine.addSystem(function TriggerAreaResultSystem() {
const garbageEntries = [];
for (const [entity, data] of entitiesMap) {
if (engine.getEntityState(entity) === entity_1.EntityState.Removed) {
garbageEntries.push(entity);
continue;
}
const result = triggerAreaResultComponent.get(entity);
// The Explorer may be taking time before the result component is put
if (result.size === 0)
continue;
const values = Array.from(result.values());
// determine starting index for new values (more than one could be added between System updates)
// search backwards to find the anchor at lastConsumedTimestamp
let startIndex = 0;
if (data.lastConsumedTimestamp >= 0) {
const newestTimestamp = values[values.length - 1].timestamp;
// if nothing new, skip processing
if (newestTimestamp <= data.lastConsumedTimestamp) {
continue;
}
// Find index of value with the lastConsumedTimestamp
let i = values.length - 2;
while (i >= 0 && values[i].timestamp > data.lastConsumedTimestamp)
i--;
// Mark the following value index as the starting point to trigger all the new value callbacks
startIndex = i + 1;
}
if (startIndex >= values.length)
continue;
// Trigger callbacks for all the new values
for (let i = startIndex; i < values.length; i++) {
switch (values[i].eventType) {
case 0 /* TriggerAreaEventType.TAET_ENTER */:
if (!data.triggerCallbackMap.has(0 /* TriggerAreaEventType.TAET_ENTER */))
continue;
data.triggerCallbackMap.get(0 /* TriggerAreaEventType.TAET_ENTER */)(values[i]);
break;
case 1 /* TriggerAreaEventType.TAET_STAY */:
if (!data.triggerCallbackMap.has(1 /* TriggerAreaEventType.TAET_STAY */))
continue;
data.triggerCallbackMap.get(1 /* TriggerAreaEventType.TAET_STAY */)(values[i]);
break;
case 2 /* TriggerAreaEventType.TAET_EXIT */:
if (!data.triggerCallbackMap.has(2 /* TriggerAreaEventType.TAET_EXIT */))
continue;
data.triggerCallbackMap.get(2 /* TriggerAreaEventType.TAET_EXIT */)(values[i]);
break;
}
}
data.lastConsumedTimestamp = values[values.length - 1].timestamp;
}
// Clean up garbage entries
garbageEntries.forEach((garbageEntity) => entitiesMap.delete(garbageEntity));
});
return {
onTriggerEnter,
removeOnTriggerEnter,
onTriggerStay,
removeOnTriggerStay,
onTriggerExit,
removeOnTriggerExit
};
}
exports.createTriggerAreaEventsSystem = createTriggerAreaEventsSystem;