UNPKG

@dcl-sdk/utils

Version:

A collection of helpers to make it easier to build a Decentraland scene using the SDK 7.

126 lines 22.1 kB
import { engine, EntityState, Schemas, Transform } from '@dcl/sdk/ecs'; import { Scalar, Vector3, Quaternion } from '@dcl/sdk/math'; import { createCatmullRomSpline } from './math'; import { priority } from './priority'; function createPaths(targetEngine) { const FollowPath = targetEngine.defineComponent('dcl.utils.FollowPath', { points: Schemas.Array(Schemas.Vector3), faceDirection: Schemas.Boolean, speed: Schemas.Number, normalizedTime: Schemas.Number, currentIndex: Schemas.Number, segmentTimes: Schemas.Array(Schemas.Number), curveSegmentCount: Schemas.Number }); const finishCbs = new Map(); const pointReachedCbs = new Map(); function unregisterEntity(entity) { finishCbs.delete(entity); pointReachedCbs.delete(entity); FollowPath.deleteFrom(entity); } function system(dt) { const deadPaths = []; const pointReachedPaths = []; for (const entity of finishCbs.keys()) { if (targetEngine.getEntityState(entity) == EntityState.Removed || !FollowPath.has(entity)) { unregisterEntity(entity); continue; } const transform = Transform.getMutable(entity); const path = FollowPath.getMutable(entity); path.normalizedTime = Scalar.clamp(path.normalizedTime + dt * path.speed, 0, 1); if (path.normalizedTime >= 1) deadPaths.push(entity); while (path.normalizedTime >= path.segmentTimes[path.currentIndex] && path.currentIndex < path.points.length - 1) { if (path.faceDirection) { const direction = Vector3.subtract(path.points[path.currentIndex + 1], path.points[path.currentIndex]); transform.rotation = Quaternion.lookRotation(direction); } if (path.currentIndex > 0 && path.currentIndex % path.curveSegmentCount == 0) { const pointIndex = path.currentIndex / path.curveSegmentCount; const pointCoords = path.points[path.currentIndex]; const nextPointCoords = path.points[path.currentIndex + path.curveSegmentCount]; pointReachedPaths.push({ entity: entity, index: pointIndex, coords: pointCoords, nextCoords: nextPointCoords }); } path.currentIndex += 1; } const timeDiff = path.segmentTimes[path.currentIndex] - path.segmentTimes[path.currentIndex - 1]; const coef = (path.segmentTimes[path.currentIndex] - path.normalizedTime) / timeDiff; transform.position = Vector3.lerp(path.points[path.currentIndex], path.points[path.currentIndex - 1], coef); } for (const pointReached of pointReachedPaths) { const callback = pointReachedCbs.get(pointReached.entity); if (callback) { callback(pointReached.index, pointReached.coords, pointReached.nextCoords); } } for (const entity of deadPaths) { const callback = finishCbs.get(entity); unregisterEntity(entity); if (callback) callback(); } } targetEngine.addSystem(system, priority.PathSystemPriority); function startPath(entity, points, duration, faceDirection, curveSegmentCount, onFinishCallback, onPointReachedCallback) { if (points.length < 2) throw new Error('At least 2 points are required to form a path.'); if (duration == 0) throw new Error('Path duration must not be zero'); if (curveSegmentCount) { const loop = Vector3.equals(points[0], points[points.length - 1]); if (loop) { points.pop(); points.unshift(points.pop()); } points = createCatmullRomSpline(points, curveSegmentCount, loop); } else { curveSegmentCount = 1; } finishCbs.set(entity, onFinishCallback); pointReachedCbs.set(entity, onPointReachedCallback); let totalLength = 0; const segmentLengths = []; for (let i = 0; i < points.length - 1; i++) { let sqDist = Vector3.distance(points[i], points[i + 1]); totalLength += sqDist; segmentLengths.push(sqDist); } const segmentTimes = [0]; for (let i = 0; i < segmentLengths.length; i++) { segmentTimes.push(segmentLengths[i] / totalLength + segmentTimes[i]); } FollowPath.createOrReplace(entity, { points: points, segmentTimes: segmentTimes, curveSegmentCount: curveSegmentCount, speed: 1 / duration, normalizedTime: 0, currentIndex: 0, faceDirection: faceDirection }); } return { startStraightPath(entity, points, duration, faceDirection, onFinishCallback, onPointReachedCallback) { return startPath(entity, points, duration, faceDirection, 0, onFinishCallback, onPointReachedCallback); }, startSmoothPath(entity, points, duration, segmentCount, faceDirection, onFinishCallback, onPointReachedCallback) { if (segmentCount < 2 || !Number.isInteger(segmentCount)) throw new Error(`segmentCount must be an integer that is greater than 2, got: ${segmentCount}`); return startPath(entity, points, duration, faceDirection, segmentCount, onFinishCallback, onPointReachedCallback); }, stopPath(entity) { unregisterEntity(entity); }, getOnFinishCallback(entity) { if (!finishCbs.has(entity)) throw new Error(`Entity ${entity} is not registered in triggers system`); return finishCbs.get(entity); } }; } export const paths = createPaths(engine); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"path.js","sourceRoot":"","sources":["../src/path.ts"],"names":[],"mappings":"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","sourcesContent":["import { engine, Entity, EntityState, IEngine, Schemas, Transform } from '@dcl/sdk/ecs'\nimport { Scalar, Vector3, Quaternion } from '@dcl/sdk/math'\nimport { createCatmullRomSpline } from './math'\nimport { priority } from './priority'\n\nexport type Paths = ReturnType<typeof createPaths>\n\nexport type OnFinishCallback = () => void\nexport type OnPointReachedCallback = (pointIndex: number, point: Vector3, nextPoint: Vector3) => void\n\nfunction createPaths(targetEngine: IEngine) {\n  const FollowPath = targetEngine.defineComponent('dcl.utils.FollowPath', {\n    points: Schemas.Array(Schemas.Vector3),\n    faceDirection: Schemas.Boolean,\n    speed: Schemas.Number,\n    normalizedTime: Schemas.Number,\n    currentIndex: Schemas.Number,\n    segmentTimes: Schemas.Array(Schemas.Number),\n    curveSegmentCount: Schemas.Number\n  })\n\n  type FinishCallbackMap = Map<Entity, OnFinishCallback | undefined>\n  type OnPointReachedCallbackMap = Map<Entity, OnPointReachedCallback | undefined>\n\n  const finishCbs: FinishCallbackMap = new Map()\n  const pointReachedCbs: OnPointReachedCallbackMap = new Map()\n\n  function unregisterEntity(entity: Entity) {\n    finishCbs.delete(entity)\n    pointReachedCbs.delete(entity)\n    FollowPath.deleteFrom(entity)\n  }\n\n  function system(dt: number) {\n    const deadPaths = []\n    const pointReachedPaths = []\n\n    for (const entity of finishCbs.keys()) {\n      if (targetEngine.getEntityState(entity) == EntityState.Removed || !FollowPath.has(entity)) {\n        unregisterEntity(entity)\n        continue\n      }\n\n      const transform = Transform.getMutable(entity)\n      const path = FollowPath.getMutable(entity)\n      path.normalizedTime = Scalar.clamp(path.normalizedTime + dt * path.speed, 0, 1)\n      if (path.normalizedTime >= 1)\n        deadPaths.push(entity)\n\n      while (\n        path.normalizedTime >= path.segmentTimes[path.currentIndex] &&\n        path.currentIndex < path.points.length - 1\n      ) {\n        if (path.faceDirection) {\n          const direction = Vector3.subtract(path.points[path.currentIndex + 1], path.points[path.currentIndex])\n          transform.rotation = Quaternion.lookRotation(direction)\n        }\n        if (path.currentIndex > 0 && path.currentIndex % path.curveSegmentCount == 0) {\n          const pointIndex = path.currentIndex / path.curveSegmentCount\n          const pointCoords = path.points[path.currentIndex]\n          const nextPointCoords = path.points[path.currentIndex + path.curveSegmentCount]\n          pointReachedPaths.push({entity: entity, index: pointIndex, coords: pointCoords, nextCoords: nextPointCoords})\n        }\n        path.currentIndex += 1\n      }\n\n      const timeDiff = path.segmentTimes[path.currentIndex] - path.segmentTimes[path.currentIndex - 1]\n      const coef = (path.segmentTimes[path.currentIndex] - path.normalizedTime) / timeDiff\n      transform.position = Vector3.lerp(path.points[path.currentIndex], path.points[path.currentIndex - 1], coef)\n    }\n\n    for (const pointReached of pointReachedPaths) {\n      const callback = pointReachedCbs.get(pointReached.entity)\n      if (callback) {\n        callback(pointReached.index, pointReached.coords, pointReached.nextCoords)\n      }\n    }\n\n    for (const entity of deadPaths) {\n      const callback = finishCbs.get(entity)\n      unregisterEntity(entity)\n      if (callback)\n        callback()\n    }\n  }\n\n  targetEngine.addSystem(system, priority.PathSystemPriority)\n\n  function startPath(\n    entity: Entity,\n    points: Vector3[],\n    duration: number,\n    faceDirection?: boolean,\n    curveSegmentCount?: number,\n    onFinishCallback?: OnFinishCallback,\n    onPointReachedCallback?: OnPointReachedCallback\n  ) {\n    if (points.length < 2)\n      throw new Error('At least 2 points are required to form a path.')\n\n    if (duration == 0)\n      throw new Error('Path duration must not be zero')\n\n    if (curveSegmentCount) {\n      const loop = Vector3.equals(points[0], points[points.length - 1])\n      if (loop) {\n        points.pop()\n        points.unshift(points.pop()!)\n      }\n      points = createCatmullRomSpline(points, curveSegmentCount, loop)\n    } else {\n      curveSegmentCount = 1\n    }\n\n    finishCbs.set(entity, onFinishCallback)\n    pointReachedCbs.set(entity, onPointReachedCallback)\n\n    let totalLength = 0\n    const segmentLengths = []\n    for (let i = 0; i < points.length - 1; i++) {\n      let sqDist = Vector3.distance(points[i], points[i + 1])\n      totalLength += sqDist\n      segmentLengths.push(sqDist)\n    }\n\n    const segmentTimes = [0]\n    for (let i = 0; i < segmentLengths.length; i++) {\n      segmentTimes.push(segmentLengths[i] / totalLength + segmentTimes[i])\n    }\n\n    FollowPath.createOrReplace(entity, {\n      points: points,\n      segmentTimes: segmentTimes,\n      curveSegmentCount: curveSegmentCount,\n      speed: 1 / duration,\n      normalizedTime: 0,\n      currentIndex: 0,\n      faceDirection: faceDirection\n    })\n  }\n\n  return {\n    startStraightPath(\n      entity: Entity,\n      points: Vector3[],\n      duration: number,\n      faceDirection?: boolean,\n      onFinishCallback?: OnFinishCallback,\n      onPointReachedCallback?: OnPointReachedCallback\n    ) {\n      return startPath(entity, points, duration, faceDirection, 0, onFinishCallback, onPointReachedCallback)\n    },\n    startSmoothPath(\n      entity: Entity,\n      points: Vector3[],\n      duration: number,\n      segmentCount: number,\n      faceDirection?: boolean,\n      onFinishCallback?: OnFinishCallback,\n      onPointReachedCallback?: OnPointReachedCallback\n    ) {\n      if (segmentCount < 2 || !Number.isInteger(segmentCount))\n        throw new Error(`segmentCount must be an integer that is greater than 2, got: ${segmentCount}`)\n      return startPath(entity, points, duration, faceDirection, segmentCount, onFinishCallback, onPointReachedCallback)\n    },\n    stopPath(entity: Entity) {\n      unregisterEntity(entity)\n    },\n    getOnFinishCallback(entity: Entity) {\n      if (!finishCbs.has(entity))\n        throw new Error(`Entity ${entity} is not registered in triggers system`)\n      return finishCbs.get(entity)\n    }\n  }\n}\n\nexport const paths = createPaths(engine)\n"]}