@tsparticles/engine 4.0.0-alpha.24 → 4.0.0-alpha.25

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.
Files changed (47) hide show
  1. package/152.min.js +1 -0
  2. package/browser/Core/Engine.js +1 -1
  3. package/browser/Core/Particles.js +8 -14
  4. package/browser/Core/Utils/Constants.js +1 -1
  5. package/browser/Core/Utils/SpatialHashGrid.js +82 -0
  6. package/browser/Core/Utils/Vectors.js +3 -10
  7. package/browser/exports.js +0 -1
  8. package/cjs/Core/Engine.js +1 -1
  9. package/cjs/Core/Particles.js +8 -14
  10. package/cjs/Core/Utils/Constants.js +1 -1
  11. package/cjs/Core/Utils/SpatialHashGrid.js +82 -0
  12. package/cjs/Core/Utils/Vectors.js +3 -10
  13. package/cjs/exports.js +0 -1
  14. package/dist_browser_Core_Container_js.js +7 -7
  15. package/esm/Core/Engine.js +1 -1
  16. package/esm/Core/Particles.js +8 -14
  17. package/esm/Core/Utils/Constants.js +1 -1
  18. package/esm/Core/Utils/SpatialHashGrid.js +82 -0
  19. package/esm/Core/Utils/Vectors.js +3 -10
  20. package/esm/exports.js +0 -1
  21. package/package.json +1 -1
  22. package/report.html +1 -1
  23. package/tsparticles.engine.js +6 -16
  24. package/tsparticles.engine.min.js +2 -2
  25. package/types/Core/Particles.d.ts +2 -2
  26. package/types/Core/Utils/Constants.d.ts +1 -1
  27. package/types/Core/Utils/SpatialHashGrid.d.ts +18 -0
  28. package/types/Core/Utils/Vectors.d.ts +2 -4
  29. package/types/export-types.d.ts +1 -1
  30. package/types/exports.d.ts +0 -1
  31. package/umd/Core/Engine.js +1 -1
  32. package/umd/Core/Particles.js +8 -14
  33. package/umd/Core/Utils/Constants.js +2 -2
  34. package/umd/Core/Utils/SpatialHashGrid.js +96 -0
  35. package/umd/Core/Utils/Vectors.js +3 -10
  36. package/umd/exports.js +1 -2
  37. package/515.min.js +0 -1
  38. package/browser/Core/Utils/Point.js +0 -8
  39. package/browser/Core/Utils/QuadTree.js +0 -85
  40. package/cjs/Core/Utils/Point.js +0 -8
  41. package/cjs/Core/Utils/QuadTree.js +0 -85
  42. package/esm/Core/Utils/Point.js +0 -8
  43. package/esm/Core/Utils/QuadTree.js +0 -85
  44. package/types/Core/Utils/Point.d.ts +0 -7
  45. package/types/Core/Utils/QuadTree.d.ts +0 -17
  46. package/umd/Core/Utils/Point.js +0 -22
  47. package/umd/Core/Utils/QuadTree.js +0 -99
@@ -1,5 +1,5 @@
1
1
  /*!
2
- * tsParticles Engine v4.0.0-alpha.24
2
+ * tsParticles Engine v4.0.0-alpha.25
3
3
  * Author: Matteo Bruni
4
4
  * MIT license: https://opensource.org/licenses/MIT
5
5
  * Website: https://particles.js.org/
@@ -55,7 +55,7 @@ eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpa
55
55
  \****************************************/
56
56
  (__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) {
57
57
 
58
- eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ Particles: () => (/* binding */ Particles)\n/* harmony export */ });\n/* harmony import */ var _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Utils/Constants.js */ \"./dist/browser/Core/Utils/Constants.js\");\n/* harmony import */ var _Enums_Types_EventType_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Enums/Types/EventType.js */ \"./dist/browser/Enums/Types/EventType.js\");\n/* harmony import */ var _Enums_Modes_LimitMode_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../Enums/Modes/LimitMode.js */ \"./dist/browser/Enums/Modes/LimitMode.js\");\n/* harmony import */ var _Particle_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./Particle.js */ \"./dist/browser/Core/Particle.js\");\n/* harmony import */ var _Utils_Point_js__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./Utils/Point.js */ \"./dist/browser/Core/Utils/Point.js\");\n/* harmony import */ var _Utils_QuadTree_js__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./Utils/QuadTree.js */ \"./dist/browser/Core/Utils/QuadTree.js\");\n/* harmony import */ var _Utils_Ranges_js__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./Utils/Ranges.js */ \"./dist/browser/Core/Utils/Ranges.js\");\n/* harmony import */ var _Utils_LogUtils_js__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../Utils/LogUtils.js */ \"./dist/browser/Utils/LogUtils.js\");\n/* harmony import */ var _Utils_OptionsUtils_js__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ../Utils/OptionsUtils.js */ \"./dist/browser/Utils/OptionsUtils.js\");\n\n\n\n\n\n\n\n\n\nconst qTreeRectangle = (canvasSize)=>{\n const { height, width } = canvasSize;\n return new _Utils_Ranges_js__WEBPACK_IMPORTED_MODULE_6__.Rectangle(_Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.posOffset * width, _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.posOffset * height, _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.sizeFactor * width, _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.sizeFactor * height);\n};\nclass Particles {\n availablePathGenerators;\n checkParticlePositionPlugins;\n effectDrawers;\n movers;\n pathGenerators;\n quadTree;\n shapeDrawers;\n updaters;\n _array;\n _container;\n _engine;\n _groupLimits;\n _limit;\n _maxZIndex;\n _minZIndex;\n _needsSort;\n _nextId;\n _particleResetPlugins;\n _particleUpdatePlugins;\n _pool;\n _postParticleUpdatePlugins;\n _postUpdatePlugins;\n _resizeFactor;\n _updatePlugins;\n _zArray;\n constructor(engine, container){\n this._engine = engine;\n this._container = container;\n this._nextId = 0;\n this._array = [];\n this._zArray = [];\n this._pool = [];\n this._limit = 0;\n this._groupLimits = new Map();\n this._needsSort = false;\n this._minZIndex = 0;\n this._maxZIndex = 0;\n const canvasSize = container.canvas.size;\n this.quadTree = new _Utils_QuadTree_js__WEBPACK_IMPORTED_MODULE_5__.QuadTree(qTreeRectangle(canvasSize), _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.qTreeCapacity);\n this.effectDrawers = new Map();\n this.movers = [];\n this.availablePathGenerators = new Map();\n this.pathGenerators = new Map();\n this.shapeDrawers = new Map();\n this.updaters = [];\n this.checkParticlePositionPlugins = [];\n this._particleResetPlugins = [];\n this._particleUpdatePlugins = [];\n this._postUpdatePlugins = [];\n this._postParticleUpdatePlugins = [];\n this._updatePlugins = [];\n }\n get count() {\n return this._array.length;\n }\n addParticle(position, overrideOptions, group, initializer) {\n const limitMode = this._container.actualOptions.particles.number.limit.mode, limit = group === undefined ? this._limit : this._groupLimits.get(group) ?? this._limit, currentCount = this.count;\n if (limit > _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minLimit) {\n switch(limitMode){\n case _Enums_Modes_LimitMode_js__WEBPACK_IMPORTED_MODULE_2__.LimitMode.delete:\n {\n const countToRemove = currentCount + _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.countOffset - limit;\n if (countToRemove > _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minCount) {\n this.removeQuantity(countToRemove);\n }\n break;\n }\n case _Enums_Modes_LimitMode_js__WEBPACK_IMPORTED_MODULE_2__.LimitMode.wait:\n if (currentCount >= limit) {\n return;\n }\n break;\n default:\n break;\n }\n }\n try {\n const particle = this._pool.pop() ?? new _Particle_js__WEBPACK_IMPORTED_MODULE_3__.Particle(this._engine, this._container);\n particle.init(this._nextId, position, overrideOptions, group);\n let canAdd = true;\n if (initializer) {\n canAdd = initializer(particle);\n }\n if (!canAdd) {\n this._pool.push(particle);\n return;\n }\n this._array.push(particle);\n this._zArray.push(particle);\n this._nextId++;\n this._engine.dispatchEvent(_Enums_Types_EventType_js__WEBPACK_IMPORTED_MODULE_1__.EventType.particleAdded, {\n container: this._container,\n data: {\n particle\n }\n });\n return particle;\n } catch (e) {\n (0,_Utils_LogUtils_js__WEBPACK_IMPORTED_MODULE_7__.getLogger)().warning(`error adding particle: ${e}`);\n }\n return undefined;\n }\n clear() {\n this._array = [];\n this._zArray = [];\n }\n destroy() {\n const container = this._container;\n for (const [, effectDrawer] of this.effectDrawers){\n effectDrawer.destroy?.(container);\n }\n for (const [, shapeDrawer] of this.shapeDrawers){\n shapeDrawer.destroy?.(container);\n }\n this._array = [];\n this._pool.length = 0;\n this._zArray = [];\n this.effectDrawers = new Map();\n this.movers = [];\n this.availablePathGenerators = new Map();\n this.pathGenerators = new Map();\n this.shapeDrawers = new Map();\n this.updaters = [];\n this.checkParticlePositionPlugins = [];\n this._particleResetPlugins = [];\n this._particleUpdatePlugins = [];\n this._postUpdatePlugins = [];\n this._postParticleUpdatePlugins = [];\n this._updatePlugins = [];\n }\n drawParticles(delta) {\n for (const particle of this._zArray){\n particle.draw(delta);\n }\n }\n filter(condition) {\n return this._array.filter(condition);\n }\n find(condition) {\n return this._array.find(condition);\n }\n get(index) {\n return this._array[index];\n }\n async init() {\n const container = this._container, options = container.actualOptions;\n this._minZIndex = 0;\n this._maxZIndex = 0;\n this._needsSort = false;\n this.checkParticlePositionPlugins = [];\n this._updatePlugins = [];\n this._particleUpdatePlugins = [];\n this._postUpdatePlugins = [];\n this._particleResetPlugins = [];\n this._postParticleUpdatePlugins = [];\n for (const plugin of container.plugins){\n if (plugin.redrawInit) {\n await plugin.redrawInit();\n }\n if (plugin.checkParticlePosition) {\n this.checkParticlePositionPlugins.push(plugin);\n }\n if (plugin.update) {\n this._updatePlugins.push(plugin);\n }\n if (plugin.particleUpdate) {\n this._particleUpdatePlugins.push(plugin);\n }\n if (plugin.postUpdate) {\n this._postUpdatePlugins.push(plugin);\n }\n if (plugin.particleReset) {\n this._particleResetPlugins.push(plugin);\n }\n if (plugin.postParticleUpdate) {\n this._postParticleUpdatePlugins.push(plugin);\n }\n }\n await this.initPlugins();\n for (const drawer of this.effectDrawers.values()){\n await drawer.init?.(container);\n }\n for (const drawer of this.shapeDrawers.values()){\n await drawer.init?.(container);\n }\n let handled = false;\n for (const plugin of container.plugins){\n handled = plugin.particlesInitialization?.() ?? handled;\n if (handled) {\n break;\n }\n }\n if (!handled) {\n const particlesOptions = options.particles, groups = particlesOptions.groups;\n for(const group in groups){\n const groupOptions = groups[group];\n if (!groupOptions) {\n continue;\n }\n for(let i = this.count, j = 0; j < groupOptions.number.value && i < particlesOptions.number.value; i++, j++){\n this.addParticle(undefined, groupOptions, group);\n }\n }\n for(let i = this.count; i < particlesOptions.number.value; i++){\n this.addParticle();\n }\n }\n }\n async initPlugins() {\n const container = this._container;\n this.effectDrawers = await this._engine.getEffectDrawers(container, true);\n this.movers = await this._engine.getMovers(container, true);\n this.availablePathGenerators = await this._engine.getPathGenerators(container, true);\n this.pathGenerators = new Map();\n this.shapeDrawers = await this._engine.getShapeDrawers(container, true);\n this.updaters = await this._engine.getUpdaters(container, true);\n for (const pathGenerator of this.pathGenerators.values()){\n pathGenerator.init();\n }\n }\n push(nb, position, overrideOptions, group) {\n for(let i = 0; i < nb; i++){\n this.addParticle(position, overrideOptions, group);\n }\n }\n async redraw() {\n this.clear();\n await this.init();\n this._container.canvas.drawParticles({\n value: 0,\n factor: 0\n });\n }\n remove(particle, group, override) {\n this.removeAt(this._array.indexOf(particle), undefined, group, override);\n }\n removeAt(index, quantity = _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.defaultRemoveQuantity, group, override) {\n if (index < _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minIndex || index > this.count) {\n return;\n }\n let deleted = 0;\n for(let i = index; deleted < quantity && i < this.count; i++){\n if (this._removeParticle(i, group, override)) {\n i--;\n deleted++;\n }\n }\n }\n removeQuantity(quantity, group) {\n this.removeAt(_Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minIndex, quantity, group);\n }\n setDensity() {\n const options = this._container.actualOptions, groups = options.particles.groups;\n let pluginsCount = 0;\n for (const plugin of this._container.plugins){\n if (plugin.particlesDensityCount) {\n pluginsCount += plugin.particlesDensityCount();\n }\n }\n for(const group in groups){\n const groupData = groups[group];\n if (!groupData) {\n continue;\n }\n const groupDataOptions = (0,_Utils_OptionsUtils_js__WEBPACK_IMPORTED_MODULE_8__.loadParticlesOptions)(this._engine, this._container, groupData);\n this._applyDensity(groupDataOptions, pluginsCount, group);\n }\n this._applyDensity(options.particles, pluginsCount);\n }\n setLastZIndex(zIndex) {\n this._needsSort ||= zIndex >= this._maxZIndex || zIndex > this._minZIndex && zIndex < this._maxZIndex;\n }\n setResizeFactor(factor) {\n this._resizeFactor = factor;\n }\n update(delta) {\n const container = this._container, particlesToDelete = new Set();\n this.quadTree = new _Utils_QuadTree_js__WEBPACK_IMPORTED_MODULE_5__.QuadTree(qTreeRectangle(container.canvas.size), _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.qTreeCapacity);\n for (const pathGenerator of this.pathGenerators.values()){\n pathGenerator.update();\n }\n for (const plugin of this._updatePlugins){\n plugin.update?.(delta);\n }\n const resizeFactor = this._resizeFactor;\n for (const particle of this._array){\n if (resizeFactor && !particle.ignoresResizeRatio) {\n particle.position.x *= resizeFactor.width;\n particle.position.y *= resizeFactor.height;\n particle.initialPosition.x *= resizeFactor.width;\n particle.initialPosition.y *= resizeFactor.height;\n }\n particle.ignoresResizeRatio = false;\n for (const plugin of this._particleResetPlugins){\n plugin.particleReset?.(particle);\n }\n for (const plugin of this._particleUpdatePlugins){\n if (particle.destroyed) {\n break;\n }\n plugin.particleUpdate?.(particle, delta);\n }\n for (const mover of this.movers){\n if (mover.isEnabled(particle)) {\n mover.move(particle, delta);\n }\n }\n if (particle.destroyed) {\n particlesToDelete.add(particle);\n continue;\n }\n this.quadTree.insert(new _Utils_Point_js__WEBPACK_IMPORTED_MODULE_4__.Point(particle.getPosition(), particle));\n }\n if (particlesToDelete.size) {\n const checkDelete = (p)=>!particlesToDelete.has(p);\n this._array = this.filter(checkDelete);\n this._zArray = this._zArray.filter(checkDelete);\n for (const particle of particlesToDelete){\n this._engine.dispatchEvent(_Enums_Types_EventType_js__WEBPACK_IMPORTED_MODULE_1__.EventType.particleRemoved, {\n container: this._container,\n data: {\n particle\n }\n });\n }\n this._addToPool(...particlesToDelete);\n }\n for (const plugin of this._postUpdatePlugins){\n plugin.postUpdate?.(delta);\n }\n for (const particle of this._array){\n for (const updater of this.updaters){\n updater.update(particle, delta);\n }\n if (!particle.destroyed && !particle.spawning) {\n for (const plugin of this._postParticleUpdatePlugins){\n plugin.postParticleUpdate?.(particle, delta);\n }\n }\n }\n delete this._resizeFactor;\n if (this._needsSort) {\n const zArray = this._zArray;\n zArray.sort((a, b)=>b.position.z - a.position.z || a.id - b.id);\n const firstItem = zArray[_Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minIndex], lastItem = zArray[zArray.length - _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.lengthOffset];\n if (!firstItem || !lastItem) {\n return;\n }\n this._maxZIndex = firstItem.position.z;\n this._minZIndex = lastItem.position.z;\n this._needsSort = false;\n }\n }\n _addToPool = (...particles)=>{\n this._pool.push(...particles);\n };\n _applyDensity = (options, pluginsCount, group, groupOptions)=>{\n const numberOptions = options.number;\n if (!numberOptions.density.enable) {\n if (group === undefined) {\n this._limit = numberOptions.limit.value;\n } else if (groupOptions?.number.limit.value ?? numberOptions.limit.value) {\n this._groupLimits.set(group, groupOptions?.number.limit.value ?? numberOptions.limit.value);\n }\n return;\n }\n const densityFactor = this._initDensityFactor(numberOptions.density), optParticlesNumber = numberOptions.value, optParticlesLimit = numberOptions.limit.value > _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minLimit ? numberOptions.limit.value : optParticlesNumber, particlesNumber = Math.min(optParticlesNumber, optParticlesLimit) * densityFactor + pluginsCount, particlesCount = Math.min(this.count, this.filter((t)=>t.group === group).length);\n if (group === undefined) {\n this._limit = numberOptions.limit.value * densityFactor;\n } else {\n this._groupLimits.set(group, numberOptions.limit.value * densityFactor);\n }\n if (particlesCount < particlesNumber) {\n this.push(Math.abs(particlesNumber - particlesCount), undefined, options, group);\n } else if (particlesCount > particlesNumber) {\n this.removeQuantity(particlesCount - particlesNumber, group);\n }\n };\n _initDensityFactor = (densityOptions)=>{\n const container = this._container;\n if (!container.canvas.element || !densityOptions.enable) {\n return _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.defaultDensityFactor;\n }\n const canvas = container.canvas.element, pxRatio = container.retina.pixelRatio;\n return canvas.width * canvas.height / (densityOptions.height * densityOptions.width * pxRatio ** _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.squareExp);\n };\n _removeParticle = (index, group, override)=>{\n const particle = this._array[index];\n if (!particle) {\n return false;\n }\n if (particle.group !== group) {\n return false;\n }\n const zIdx = this._zArray.indexOf(particle);\n this._array.splice(index, _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.deleteCount);\n this._zArray.splice(zIdx, _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.deleteCount);\n particle.destroy(override);\n this._engine.dispatchEvent(_Enums_Types_EventType_js__WEBPACK_IMPORTED_MODULE_1__.EventType.particleRemoved, {\n container: this._container,\n data: {\n particle\n }\n });\n this._addToPool(particle);\n return true;\n };\n}\n\n\n//# sourceURL=webpack://@tsparticles/engine/./dist/browser/Core/Particles.js?\n}");
58
+ eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ Particles: () => (/* binding */ Particles)\n/* harmony export */ });\n/* harmony import */ var _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Utils/Constants.js */ \"./dist/browser/Core/Utils/Constants.js\");\n/* harmony import */ var _Enums_Types_EventType_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Enums/Types/EventType.js */ \"./dist/browser/Enums/Types/EventType.js\");\n/* harmony import */ var _Enums_Modes_LimitMode_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../Enums/Modes/LimitMode.js */ \"./dist/browser/Enums/Modes/LimitMode.js\");\n/* harmony import */ var _Particle_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./Particle.js */ \"./dist/browser/Core/Particle.js\");\n/* harmony import */ var _Utils_SpatialHashGrid_js__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./Utils/SpatialHashGrid.js */ \"./dist/browser/Core/Utils/SpatialHashGrid.js\");\n/* harmony import */ var _Utils_LogUtils_js__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../Utils/LogUtils.js */ \"./dist/browser/Utils/LogUtils.js\");\n/* harmony import */ var _Utils_OptionsUtils_js__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ../Utils/OptionsUtils.js */ \"./dist/browser/Utils/OptionsUtils.js\");\n\n\n\n\n\n\n\nclass Particles {\n availablePathGenerators;\n checkParticlePositionPlugins;\n effectDrawers;\n grid;\n movers;\n pathGenerators;\n shapeDrawers;\n updaters;\n _array;\n _container;\n _engine;\n _groupLimits;\n _limit;\n _maxZIndex;\n _minZIndex;\n _needsSort;\n _nextId;\n _particleResetPlugins;\n _particleUpdatePlugins;\n _pool;\n _postParticleUpdatePlugins;\n _postUpdatePlugins;\n _resizeFactor;\n _updatePlugins;\n _zArray;\n constructor(engine, container){\n this._engine = engine;\n this._container = container;\n this._nextId = 0;\n this._array = [];\n this._zArray = [];\n this._pool = [];\n this._limit = 0;\n this._groupLimits = new Map();\n this._needsSort = false;\n this._minZIndex = 0;\n this._maxZIndex = 0;\n this.grid = new _Utils_SpatialHashGrid_js__WEBPACK_IMPORTED_MODULE_4__.SpatialHashGrid(_Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.spatialHashGridCellSize);\n this.effectDrawers = new Map();\n this.movers = [];\n this.availablePathGenerators = new Map();\n this.pathGenerators = new Map();\n this.shapeDrawers = new Map();\n this.updaters = [];\n this.checkParticlePositionPlugins = [];\n this._particleResetPlugins = [];\n this._particleUpdatePlugins = [];\n this._postUpdatePlugins = [];\n this._postParticleUpdatePlugins = [];\n this._updatePlugins = [];\n }\n get count() {\n return this._array.length;\n }\n addParticle(position, overrideOptions, group, initializer) {\n const limitMode = this._container.actualOptions.particles.number.limit.mode, limit = group === undefined ? this._limit : this._groupLimits.get(group) ?? this._limit, currentCount = this.count;\n if (limit > _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minLimit) {\n switch(limitMode){\n case _Enums_Modes_LimitMode_js__WEBPACK_IMPORTED_MODULE_2__.LimitMode.delete:\n {\n const countToRemove = currentCount + _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.countOffset - limit;\n if (countToRemove > _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minCount) {\n this.removeQuantity(countToRemove);\n }\n break;\n }\n case _Enums_Modes_LimitMode_js__WEBPACK_IMPORTED_MODULE_2__.LimitMode.wait:\n if (currentCount >= limit) {\n return;\n }\n break;\n default:\n break;\n }\n }\n try {\n const particle = this._pool.pop() ?? new _Particle_js__WEBPACK_IMPORTED_MODULE_3__.Particle(this._engine, this._container);\n particle.init(this._nextId, position, overrideOptions, group);\n let canAdd = true;\n if (initializer) {\n canAdd = initializer(particle);\n }\n if (!canAdd) {\n this._pool.push(particle);\n return;\n }\n this._array.push(particle);\n this._zArray.push(particle);\n this._nextId++;\n this._engine.dispatchEvent(_Enums_Types_EventType_js__WEBPACK_IMPORTED_MODULE_1__.EventType.particleAdded, {\n container: this._container,\n data: {\n particle\n }\n });\n return particle;\n } catch (e) {\n (0,_Utils_LogUtils_js__WEBPACK_IMPORTED_MODULE_5__.getLogger)().warning(`error adding particle: ${e}`);\n }\n return undefined;\n }\n clear() {\n this._array = [];\n this._zArray = [];\n }\n destroy() {\n const container = this._container;\n for (const [, effectDrawer] of this.effectDrawers){\n effectDrawer.destroy?.(container);\n }\n for (const [, shapeDrawer] of this.shapeDrawers){\n shapeDrawer.destroy?.(container);\n }\n this._array = [];\n this._pool.length = 0;\n this._zArray = [];\n this.effectDrawers = new Map();\n this.movers = [];\n this.availablePathGenerators = new Map();\n this.pathGenerators = new Map();\n this.shapeDrawers = new Map();\n this.updaters = [];\n this.checkParticlePositionPlugins = [];\n this._particleResetPlugins = [];\n this._particleUpdatePlugins = [];\n this._postUpdatePlugins = [];\n this._postParticleUpdatePlugins = [];\n this._updatePlugins = [];\n }\n drawParticles(delta) {\n for (const particle of this._zArray){\n particle.draw(delta);\n }\n }\n filter(condition) {\n return this._array.filter(condition);\n }\n find(condition) {\n return this._array.find(condition);\n }\n get(index) {\n return this._array[index];\n }\n async init() {\n const container = this._container, options = container.actualOptions;\n this._minZIndex = 0;\n this._maxZIndex = 0;\n this._needsSort = false;\n this.checkParticlePositionPlugins = [];\n this._updatePlugins = [];\n this._particleUpdatePlugins = [];\n this._postUpdatePlugins = [];\n this._particleResetPlugins = [];\n this._postParticleUpdatePlugins = [];\n this.grid = new _Utils_SpatialHashGrid_js__WEBPACK_IMPORTED_MODULE_4__.SpatialHashGrid(_Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.spatialHashGridCellSize * container.retina.pixelRatio);\n for (const plugin of container.plugins){\n if (plugin.redrawInit) {\n await plugin.redrawInit();\n }\n if (plugin.checkParticlePosition) {\n this.checkParticlePositionPlugins.push(plugin);\n }\n if (plugin.update) {\n this._updatePlugins.push(plugin);\n }\n if (plugin.particleUpdate) {\n this._particleUpdatePlugins.push(plugin);\n }\n if (plugin.postUpdate) {\n this._postUpdatePlugins.push(plugin);\n }\n if (plugin.particleReset) {\n this._particleResetPlugins.push(plugin);\n }\n if (plugin.postParticleUpdate) {\n this._postParticleUpdatePlugins.push(plugin);\n }\n }\n await this.initPlugins();\n for (const drawer of this.effectDrawers.values()){\n await drawer.init?.(container);\n }\n for (const drawer of this.shapeDrawers.values()){\n await drawer.init?.(container);\n }\n let handled = false;\n for (const plugin of container.plugins){\n handled = plugin.particlesInitialization?.() ?? handled;\n if (handled) {\n break;\n }\n }\n if (!handled) {\n const particlesOptions = options.particles, groups = particlesOptions.groups;\n for(const group in groups){\n const groupOptions = groups[group];\n if (!groupOptions) {\n continue;\n }\n for(let i = this.count, j = 0; j < groupOptions.number.value && i < particlesOptions.number.value; i++, j++){\n this.addParticle(undefined, groupOptions, group);\n }\n }\n for(let i = this.count; i < particlesOptions.number.value; i++){\n this.addParticle();\n }\n }\n }\n async initPlugins() {\n const container = this._container;\n this.effectDrawers = await this._engine.getEffectDrawers(container, true);\n this.movers = await this._engine.getMovers(container, true);\n this.availablePathGenerators = await this._engine.getPathGenerators(container, true);\n this.pathGenerators = new Map();\n this.shapeDrawers = await this._engine.getShapeDrawers(container, true);\n this.updaters = await this._engine.getUpdaters(container, true);\n for (const pathGenerator of this.pathGenerators.values()){\n pathGenerator.init();\n }\n }\n push(nb, position, overrideOptions, group) {\n for(let i = 0; i < nb; i++){\n this.addParticle(position, overrideOptions, group);\n }\n }\n async redraw() {\n this.clear();\n await this.init();\n this._container.canvas.drawParticles({\n value: 0,\n factor: 0\n });\n }\n remove(particle, group, override) {\n this.removeAt(this._array.indexOf(particle), undefined, group, override);\n }\n removeAt(index, quantity = _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.defaultRemoveQuantity, group, override) {\n if (index < _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minIndex || index > this.count) {\n return;\n }\n let deleted = 0;\n for(let i = index; deleted < quantity && i < this.count; i++){\n if (this._removeParticle(i, group, override)) {\n i--;\n deleted++;\n }\n }\n }\n removeQuantity(quantity, group) {\n this.removeAt(_Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minIndex, quantity, group);\n }\n setDensity() {\n const options = this._container.actualOptions, groups = options.particles.groups;\n let pluginsCount = 0;\n for (const plugin of this._container.plugins){\n if (plugin.particlesDensityCount) {\n pluginsCount += plugin.particlesDensityCount();\n }\n }\n for(const group in groups){\n const groupData = groups[group];\n if (!groupData) {\n continue;\n }\n const groupDataOptions = (0,_Utils_OptionsUtils_js__WEBPACK_IMPORTED_MODULE_6__.loadParticlesOptions)(this._engine, this._container, groupData);\n this._applyDensity(groupDataOptions, pluginsCount, group);\n }\n this._applyDensity(options.particles, pluginsCount);\n }\n setLastZIndex(zIndex) {\n this._needsSort ||= zIndex >= this._maxZIndex || zIndex > this._minZIndex && zIndex < this._maxZIndex;\n }\n setResizeFactor(factor) {\n this._resizeFactor = factor;\n }\n update(delta) {\n const particlesToDelete = new Set();\n this.grid.clear();\n for (const pathGenerator of this.pathGenerators.values()){\n pathGenerator.update();\n }\n for (const plugin of this._updatePlugins){\n plugin.update?.(delta);\n }\n const resizeFactor = this._resizeFactor;\n for (const particle of this._array){\n if (resizeFactor && !particle.ignoresResizeRatio) {\n particle.position.x *= resizeFactor.width;\n particle.position.y *= resizeFactor.height;\n particle.initialPosition.x *= resizeFactor.width;\n particle.initialPosition.y *= resizeFactor.height;\n }\n particle.ignoresResizeRatio = false;\n for (const plugin of this._particleResetPlugins){\n plugin.particleReset?.(particle);\n }\n for (const plugin of this._particleUpdatePlugins){\n if (particle.destroyed) {\n break;\n }\n plugin.particleUpdate?.(particle, delta);\n }\n for (const mover of this.movers){\n if (mover.isEnabled(particle)) {\n mover.move(particle, delta);\n }\n }\n if (particle.destroyed) {\n particlesToDelete.add(particle);\n continue;\n }\n this.grid.insert(particle);\n }\n if (particlesToDelete.size) {\n const checkDelete = (p)=>!particlesToDelete.has(p);\n this._array = this.filter(checkDelete);\n this._zArray = this._zArray.filter(checkDelete);\n for (const particle of particlesToDelete){\n this._engine.dispatchEvent(_Enums_Types_EventType_js__WEBPACK_IMPORTED_MODULE_1__.EventType.particleRemoved, {\n container: this._container,\n data: {\n particle\n }\n });\n }\n this._addToPool(...particlesToDelete);\n }\n for (const plugin of this._postUpdatePlugins){\n plugin.postUpdate?.(delta);\n }\n for (const particle of this._array){\n for (const updater of this.updaters){\n updater.update(particle, delta);\n }\n if (!particle.destroyed && !particle.spawning) {\n for (const plugin of this._postParticleUpdatePlugins){\n plugin.postParticleUpdate?.(particle, delta);\n }\n }\n }\n delete this._resizeFactor;\n if (this._needsSort) {\n const zArray = this._zArray;\n zArray.sort((a, b)=>b.position.z - a.position.z || a.id - b.id);\n const firstItem = zArray[_Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minIndex], lastItem = zArray[zArray.length - _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.lengthOffset];\n if (!firstItem || !lastItem) {\n return;\n }\n this._maxZIndex = firstItem.position.z;\n this._minZIndex = lastItem.position.z;\n this._needsSort = false;\n }\n }\n _addToPool = (...particles)=>{\n this._pool.push(...particles);\n };\n _applyDensity = (options, pluginsCount, group, groupOptions)=>{\n const numberOptions = options.number;\n if (!numberOptions.density.enable) {\n if (group === undefined) {\n this._limit = numberOptions.limit.value;\n } else if (groupOptions?.number.limit.value ?? numberOptions.limit.value) {\n this._groupLimits.set(group, groupOptions?.number.limit.value ?? numberOptions.limit.value);\n }\n return;\n }\n const densityFactor = this._initDensityFactor(numberOptions.density), optParticlesNumber = numberOptions.value, optParticlesLimit = numberOptions.limit.value > _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.minLimit ? numberOptions.limit.value : optParticlesNumber, particlesNumber = Math.min(optParticlesNumber, optParticlesLimit) * densityFactor + pluginsCount, particlesCount = Math.min(this.count, this.filter((t)=>t.group === group).length);\n if (group === undefined) {\n this._limit = numberOptions.limit.value * densityFactor;\n } else {\n this._groupLimits.set(group, numberOptions.limit.value * densityFactor);\n }\n if (particlesCount < particlesNumber) {\n this.push(Math.abs(particlesNumber - particlesCount), undefined, options, group);\n } else if (particlesCount > particlesNumber) {\n this.removeQuantity(particlesCount - particlesNumber, group);\n }\n };\n _initDensityFactor = (densityOptions)=>{\n const container = this._container;\n if (!container.canvas.element || !densityOptions.enable) {\n return _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.defaultDensityFactor;\n }\n const canvas = container.canvas.element, pxRatio = container.retina.pixelRatio;\n return canvas.width * canvas.height / (densityOptions.height * densityOptions.width * pxRatio ** _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.squareExp);\n };\n _removeParticle = (index, group, override)=>{\n const particle = this._array[index];\n if (!particle) {\n return false;\n }\n if (particle.group !== group) {\n return false;\n }\n const zIdx = this._zArray.indexOf(particle);\n this._array.splice(index, _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.deleteCount);\n this._zArray.splice(zIdx, _Utils_Constants_js__WEBPACK_IMPORTED_MODULE_0__.deleteCount);\n particle.destroy(override);\n this._engine.dispatchEvent(_Enums_Types_EventType_js__WEBPACK_IMPORTED_MODULE_1__.EventType.particleRemoved, {\n container: this._container,\n data: {\n particle\n }\n });\n this._addToPool(particle);\n return true;\n };\n}\n\n\n//# sourceURL=webpack://@tsparticles/engine/./dist/browser/Core/Particles.js?\n}");
59
59
 
60
60
  /***/ },
61
61
 
@@ -79,13 +79,13 @@ eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpa
79
79
 
80
80
  /***/ },
81
81
 
82
- /***/ "./dist/browser/Core/Utils/QuadTree.js"
83
- /*!*********************************************!*\
84
- !*** ./dist/browser/Core/Utils/QuadTree.js ***!
85
- \*********************************************/
82
+ /***/ "./dist/browser/Core/Utils/SpatialHashGrid.js"
83
+ /*!****************************************************!*\
84
+ !*** ./dist/browser/Core/Utils/SpatialHashGrid.js ***!
85
+ \****************************************************/
86
86
  (__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) {
87
87
 
88
- eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ QuadTree: () => (/* binding */ QuadTree)\n/* harmony export */ });\n/* harmony import */ var _Ranges_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Ranges.js */ \"./dist/browser/Core/Utils/Ranges.js\");\n/* harmony import */ var _Constants_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./Constants.js */ \"./dist/browser/Core/Utils/Constants.js\");\n/* harmony import */ var _Utils_MathUtils_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../../Utils/MathUtils.js */ \"./dist/browser/Utils/MathUtils.js\");\n\n\n\nclass QuadTree {\n rectangle;\n capacity;\n _points = [];\n _subs = null;\n constructor(rectangle, capacity){\n this.rectangle = rectangle;\n this.capacity = capacity;\n }\n insert(point) {\n if (!this.rectangle.contains(point.position)) {\n return false;\n }\n const subs = this._subs;\n if (!subs) {\n if (this._points.length < this.capacity) {\n this._points.push(point);\n return true;\n }\n this._subdivide();\n const newSubs = this._subs;\n for (const p of this._points){\n for(let s = 0; s < _Constants_js__WEBPACK_IMPORTED_MODULE_1__.subdivideCount; s++){\n if (newSubs?.[s]?.insert(p)) {\n break;\n }\n }\n }\n this._points.length = 0;\n for(let s = 0; s < _Constants_js__WEBPACK_IMPORTED_MODULE_1__.subdivideCount; s++){\n if (newSubs?.[s]?.insert(point)) {\n return true;\n }\n }\n return false;\n }\n for(let s = 0; s < _Constants_js__WEBPACK_IMPORTED_MODULE_1__.subdivideCount; s++){\n if (subs[s]?.insert(point)) {\n return true;\n }\n }\n return false;\n }\n query(range, check, out = []) {\n if (!range.intersects(this.rectangle)) {\n return out;\n }\n const points = this._points;\n for (const p of points){\n const particle = p.particle;\n if (!range.contains(p.position) && (0,_Utils_MathUtils_js__WEBPACK_IMPORTED_MODULE_2__.getDistance)(range.position, p.position) > particle.getRadius()) {\n continue;\n }\n if (check && !check(particle)) {\n continue;\n }\n out.push(particle);\n }\n const subs = this._subs;\n if (subs) {\n for (const s of subs){\n s.query(range, check, out);\n }\n }\n return out;\n }\n queryCircle(position, radius, check) {\n return this.query(new _Ranges_js__WEBPACK_IMPORTED_MODULE_0__.Circle(position.x, position.y, radius), check);\n }\n queryRectangle(position, size, check) {\n return this.query(new _Ranges_js__WEBPACK_IMPORTED_MODULE_0__.Rectangle(position.x, position.y, size.width, size.height), check);\n }\n _subdivide() {\n const rect = this.rectangle, { x, y } = rect.position, { width, height } = rect.size, halfWidth = width * _Constants_js__WEBPACK_IMPORTED_MODULE_1__.half, halfHeight = height * _Constants_js__WEBPACK_IMPORTED_MODULE_1__.half, capacity = this.capacity;\n this._subs = [\n new QuadTree(new _Ranges_js__WEBPACK_IMPORTED_MODULE_0__.Rectangle(x, y, halfWidth, halfHeight), capacity),\n new QuadTree(new _Ranges_js__WEBPACK_IMPORTED_MODULE_0__.Rectangle(x + halfWidth, y, halfWidth, halfHeight), capacity),\n new QuadTree(new _Ranges_js__WEBPACK_IMPORTED_MODULE_0__.Rectangle(x, y + halfHeight, halfWidth, halfHeight), capacity),\n new QuadTree(new _Ranges_js__WEBPACK_IMPORTED_MODULE_0__.Rectangle(x + halfWidth, y + halfHeight, halfWidth, halfHeight), capacity)\n ];\n }\n}\n\n\n//# sourceURL=webpack://@tsparticles/engine/./dist/browser/Core/Utils/QuadTree.js?\n}");
88
+ eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ SpatialHashGrid: () => (/* binding */ SpatialHashGrid)\n/* harmony export */ });\n/* harmony import */ var _Ranges_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Ranges.js */ \"./dist/browser/Core/Utils/Ranges.js\");\n\nclass SpatialHashGrid {\n _cellSize;\n _cells = new Map();\n _pendingCellSize;\n constructor(cellSize){\n this._cellSize = cellSize;\n }\n clear() {\n this._cells.clear();\n const pendingCellSize = this._pendingCellSize;\n if (pendingCellSize) {\n this._cellSize = pendingCellSize;\n }\n this._pendingCellSize = undefined;\n }\n insert(particle) {\n const { x, y } = particle.getPosition(), key = this._cellKeyFromCoords(x, y);\n if (!this._cells.has(key)) {\n this._cells.set(key, []);\n }\n this._cells.get(key)?.push(particle);\n }\n query(range, check, out = []) {\n const bounds = this._getRangeBounds(range);\n if (!bounds) {\n return out;\n }\n const minCellX = Math.floor(bounds.minX / this._cellSize), maxCellX = Math.floor(bounds.maxX / this._cellSize), minCellY = Math.floor(bounds.minY / this._cellSize), maxCellY = Math.floor(bounds.maxY / this._cellSize);\n for(let cx = minCellX; cx <= maxCellX; cx++){\n for(let cy = minCellY; cy <= maxCellY; cy++){\n const key = `${cx}_${cy}`, cellParticles = this._cells.get(key);\n if (!cellParticles) {\n continue;\n }\n for (const p of cellParticles){\n if (check && !check(p)) {\n continue;\n }\n if (range.contains(p.getPosition())) {\n out.push(p);\n }\n }\n }\n }\n return out;\n }\n queryCircle(position, radius, check, out = []) {\n return this.query(new _Ranges_js__WEBPACK_IMPORTED_MODULE_0__.Circle(position.x, position.y, radius), check, out);\n }\n queryRectangle(position, size, check, out = []) {\n return this.query(new _Ranges_js__WEBPACK_IMPORTED_MODULE_0__.Rectangle(position.x, position.y, size.width, size.height), check, out);\n }\n setCellSize(cellSize) {\n this._pendingCellSize = cellSize;\n }\n _cellKeyFromCoords(x, y) {\n const cellX = Math.floor(x / this._cellSize), cellY = Math.floor(y / this._cellSize);\n return `${cellX}_${cellY}`;\n }\n _getRangeBounds(range) {\n if (range instanceof _Ranges_js__WEBPACK_IMPORTED_MODULE_0__.Circle) {\n const r = range.radius, { x, y } = range.position;\n return {\n minX: x - r,\n maxX: x + r,\n minY: y - r,\n maxY: y + r\n };\n }\n if (range instanceof _Ranges_js__WEBPACK_IMPORTED_MODULE_0__.Rectangle) {\n const { x, y } = range.position, { width, height } = range.size;\n return {\n minX: x,\n maxX: x + width,\n minY: y,\n maxY: y + height\n };\n }\n return null;\n }\n}\n\n\n//# sourceURL=webpack://@tsparticles/engine/./dist/browser/Core/Utils/SpatialHashGrid.js?\n}");
89
89
 
90
90
  /***/ }
91
91
 
@@ -87,7 +87,7 @@ export class Engine {
87
87
  return this._domArray;
88
88
  }
89
89
  get version() {
90
- return "4.0.0-alpha.24";
90
+ return "4.0.0-alpha.25";
91
91
  }
92
92
  addColorManager(name, manager) {
93
93
  this.colorManagers.set(name, manager);
@@ -1,23 +1,17 @@
1
- import { countOffset, defaultDensityFactor, defaultRemoveQuantity, deleteCount, lengthOffset, minCount, minIndex, minLimit, posOffset, qTreeCapacity, sizeFactor, squareExp, } from "./Utils/Constants.js";
1
+ import { countOffset, defaultDensityFactor, defaultRemoveQuantity, deleteCount, lengthOffset, minCount, minIndex, minLimit, spatialHashGridCellSize, squareExp, } from "./Utils/Constants.js";
2
2
  import { EventType } from "../Enums/Types/EventType.js";
3
3
  import { LimitMode } from "../Enums/Modes/LimitMode.js";
4
4
  import { Particle } from "./Particle.js";
5
- import { Point } from "./Utils/Point.js";
6
- import { QuadTree } from "./Utils/QuadTree.js";
7
- import { Rectangle } from "./Utils/Ranges.js";
5
+ import { SpatialHashGrid } from "./Utils/SpatialHashGrid.js";
8
6
  import { getLogger } from "../Utils/LogUtils.js";
9
7
  import { loadParticlesOptions } from "../Utils/OptionsUtils.js";
10
- const qTreeRectangle = (canvasSize) => {
11
- const { height, width } = canvasSize;
12
- return new Rectangle(posOffset * width, posOffset * height, sizeFactor * width, sizeFactor * height);
13
- };
14
8
  export class Particles {
15
9
  availablePathGenerators;
16
10
  checkParticlePositionPlugins;
17
11
  effectDrawers;
12
+ grid;
18
13
  movers;
19
14
  pathGenerators;
20
- quadTree;
21
15
  shapeDrawers;
22
16
  updaters;
23
17
  _array;
@@ -49,8 +43,7 @@ export class Particles {
49
43
  this._needsSort = false;
50
44
  this._minZIndex = 0;
51
45
  this._maxZIndex = 0;
52
- const canvasSize = container.canvas.size;
53
- this.quadTree = new QuadTree(qTreeRectangle(canvasSize), qTreeCapacity);
46
+ this.grid = new SpatialHashGrid(spatialHashGridCellSize);
54
47
  this.effectDrawers = new Map();
55
48
  this.movers = [];
56
49
  this.availablePathGenerators = new Map();
@@ -167,6 +160,7 @@ export class Particles {
167
160
  this._postUpdatePlugins = [];
168
161
  this._particleResetPlugins = [];
169
162
  this._postParticleUpdatePlugins = [];
163
+ this.grid = new SpatialHashGrid(spatialHashGridCellSize * container.retina.pixelRatio);
170
164
  for (const plugin of container.plugins) {
171
165
  if (plugin.redrawInit) {
172
166
  await plugin.redrawInit();
@@ -285,8 +279,8 @@ export class Particles {
285
279
  this._resizeFactor = factor;
286
280
  }
287
281
  update(delta) {
288
- const container = this._container, particlesToDelete = new Set();
289
- this.quadTree = new QuadTree(qTreeRectangle(container.canvas.size), qTreeCapacity);
282
+ const particlesToDelete = new Set();
283
+ this.grid.clear();
290
284
  for (const pathGenerator of this.pathGenerators.values()) {
291
285
  pathGenerator.update();
292
286
  }
@@ -320,7 +314,7 @@ export class Particles {
320
314
  particlesToDelete.add(particle);
321
315
  continue;
322
316
  }
323
- this.quadTree.insert(new Point(particle.getPosition(), particle));
317
+ this.grid.insert(particle);
324
318
  }
325
319
  if (particlesToDelete.size) {
326
320
  const checkDelete = (p) => !particlesToDelete.has(p);
@@ -7,4 +7,4 @@ export const generatedAttribute = "generated", resizeEvent = "resize", visibilit
7
7
  b: 0,
8
8
  c: 0,
9
9
  d: 1,
10
- }, randomColorValue = "random", midColorValue = "mid", double = 2, doublePI = Math.PI * double, defaultFps = 60, defaultAlpha = 1, generatedTrue = "true", generatedFalse = "false", canvasTag = "canvas", defaultRetryCount = 0, squareExp = 2, qTreeCapacity = 4, defaultRemoveQuantity = 1, defaultRatio = 1, defaultReduceFactor = 1, subdivideCount = 4, inverseFactorNumerator = 1, rgbMax = 255, hMax = 360, sMax = 100, lMax = 100, hMin = 0, sMin = 0, hPhase = 60, empty = 0, quarter = 0.25, threeQuarter = half + quarter, minVelocity = 0, defaultTransformValue = 1, minimumSize = 0, minimumLength = 0, zIndexFactorOffset = 1, defaultOpacity = 1, clickRadius = 1, touchEndLengthOffset = 1, minCoordinate = 0, removeDeleteCount = 1, removeMinIndex = 0, defaultFpsLimit = 120, minFpsLimit = 0, canvasFirstIndex = 0, loadRandomFactor = 10000, loadMinIndex = 0, one = 1, none = 0, decayOffset = 1, tryCountIncrement = 1, minRetries = 0, minZ = 0, defaultRadius = 0, posOffset = -quarter, sizeFactor = 1.5, minLimit = 0, countOffset = 1, minCount = 0, minIndex = 0, lengthOffset = 1, defaultDensityFactor = 1, deleteCount = 1, touchDelay = 500, manualDefaultPosition = 50, defaultAngle = 0, identity = 1, minStrokeWidth = 0, lFactor = 1, lMin = 0, triple = 3, sextuple = 6, sNormalizedOffset = 1, phaseNumerator = 1, defaultRgbMin = 0, defaultVelocity = 0, defaultLoops = 0, defaultTime = 0, defaultZoom = 1;
10
+ }, randomColorValue = "random", midColorValue = "mid", double = 2, doublePI = Math.PI * double, defaultFps = 60, defaultAlpha = 1, generatedTrue = "true", generatedFalse = "false", canvasTag = "canvas", defaultRetryCount = 0, squareExp = 2, qTreeCapacity = 4, spatialHashGridCellSize = 100, defaultRemoveQuantity = 1, defaultRatio = 1, defaultReduceFactor = 1, subdivideCount = 4, inverseFactorNumerator = 1, rgbMax = 255, hMax = 360, sMax = 100, lMax = 100, hMin = 0, sMin = 0, hPhase = 60, empty = 0, quarter = 0.25, threeQuarter = half + quarter, minVelocity = 0, defaultTransformValue = 1, minimumSize = 0, minimumLength = 0, zIndexFactorOffset = 1, defaultOpacity = 1, clickRadius = 1, touchEndLengthOffset = 1, minCoordinate = 0, removeDeleteCount = 1, removeMinIndex = 0, defaultFpsLimit = 120, minFpsLimit = 0, canvasFirstIndex = 0, loadRandomFactor = 10000, loadMinIndex = 0, one = 1, none = 0, decayOffset = 1, tryCountIncrement = 1, minRetries = 0, minZ = 0, defaultRadius = 0, posOffset = -quarter, sizeFactor = 1.5, minLimit = 0, countOffset = 1, minCount = 0, minIndex = 0, lengthOffset = 1, defaultDensityFactor = 1, deleteCount = 1, touchDelay = 500, manualDefaultPosition = 50, defaultAngle = 0, identity = 1, minStrokeWidth = 0, lFactor = 1, lMin = 0, triple = 3, sextuple = 6, sNormalizedOffset = 1, phaseNumerator = 1, defaultRgbMin = 0, defaultVelocity = 0, defaultLoops = 0, defaultTime = 0, defaultZoom = 1;
@@ -0,0 +1,82 @@
1
+ import { Circle, Rectangle } from "./Ranges.js";
2
+ export class SpatialHashGrid {
3
+ _cellSize;
4
+ _cells = new Map();
5
+ _pendingCellSize;
6
+ constructor(cellSize) {
7
+ this._cellSize = cellSize;
8
+ }
9
+ clear() {
10
+ this._cells.clear();
11
+ const pendingCellSize = this._pendingCellSize;
12
+ if (pendingCellSize) {
13
+ this._cellSize = pendingCellSize;
14
+ }
15
+ this._pendingCellSize = undefined;
16
+ }
17
+ insert(particle) {
18
+ const { x, y } = particle.getPosition(), key = this._cellKeyFromCoords(x, y);
19
+ if (!this._cells.has(key)) {
20
+ this._cells.set(key, []);
21
+ }
22
+ this._cells.get(key)?.push(particle);
23
+ }
24
+ query(range, check, out = []) {
25
+ const bounds = this._getRangeBounds(range);
26
+ if (!bounds) {
27
+ return out;
28
+ }
29
+ const minCellX = Math.floor(bounds.minX / this._cellSize), maxCellX = Math.floor(bounds.maxX / this._cellSize), minCellY = Math.floor(bounds.minY / this._cellSize), maxCellY = Math.floor(bounds.maxY / this._cellSize);
30
+ for (let cx = minCellX; cx <= maxCellX; cx++) {
31
+ for (let cy = minCellY; cy <= maxCellY; cy++) {
32
+ const key = `${cx}_${cy}`, cellParticles = this._cells.get(key);
33
+ if (!cellParticles) {
34
+ continue;
35
+ }
36
+ for (const p of cellParticles) {
37
+ if (check && !check(p)) {
38
+ continue;
39
+ }
40
+ if (range.contains(p.getPosition())) {
41
+ out.push(p);
42
+ }
43
+ }
44
+ }
45
+ }
46
+ return out;
47
+ }
48
+ queryCircle(position, radius, check, out = []) {
49
+ return this.query(new Circle(position.x, position.y, radius), check, out);
50
+ }
51
+ queryRectangle(position, size, check, out = []) {
52
+ return this.query(new Rectangle(position.x, position.y, size.width, size.height), check, out);
53
+ }
54
+ setCellSize(cellSize) {
55
+ this._pendingCellSize = cellSize;
56
+ }
57
+ _cellKeyFromCoords(x, y) {
58
+ const cellX = Math.floor(x / this._cellSize), cellY = Math.floor(y / this._cellSize);
59
+ return `${cellX}_${cellY}`;
60
+ }
61
+ _getRangeBounds(range) {
62
+ if (range instanceof Circle) {
63
+ const r = range.radius, { x, y } = range.position;
64
+ return {
65
+ minX: x - r,
66
+ maxX: x + r,
67
+ minY: y - r,
68
+ maxY: y + r,
69
+ };
70
+ }
71
+ if (range instanceof Rectangle) {
72
+ const { x, y } = range.position, { width, height } = range.size;
73
+ return {
74
+ minX: x,
75
+ maxX: x + width,
76
+ minY: y,
77
+ maxY: y + height,
78
+ };
79
+ }
80
+ return null;
81
+ }
82
+ }
@@ -43,12 +43,6 @@ export class Vector3d {
43
43
  copy() {
44
44
  return Vector3d.clone(this);
45
45
  }
46
- distanceTo(v) {
47
- return this.sub(v).length;
48
- }
49
- distanceToSq(v) {
50
- return this.sub(v).getLengthSq();
51
- }
52
46
  div(n) {
53
47
  return Vector3d.create(this.x / n, this.y / n, this.z / n);
54
48
  }
@@ -80,8 +74,7 @@ export class Vector3d {
80
74
  setTo(c) {
81
75
  this.x = c.x;
82
76
  this.y = c.y;
83
- const v3d = c;
84
- this.z = v3d.z ? v3d.z : originPoint.z;
77
+ this.z = "z" in c ? c.z : originPoint.z;
85
78
  }
86
79
  sub(v) {
87
80
  return Vector3d.create(this.x - v.x, this.y - v.y, this.z - v.z);
@@ -91,10 +84,10 @@ export class Vector3d {
91
84
  this.y -= v.y;
92
85
  this.z -= v.z;
93
86
  }
94
- _updateFromAngle = (angle, length) => {
87
+ _updateFromAngle(angle, length) {
95
88
  this.x = Math.cos(angle) * length;
96
89
  this.y = Math.sin(angle) * length;
97
- };
90
+ }
98
91
  }
99
92
  export class Vector extends Vector3d {
100
93
  constructor(x = originPoint.x, y = originPoint.y) {
package/esm/exports.js CHANGED
@@ -1,5 +1,4 @@
1
1
  export * from "./Core/Utils/Constants.js";
2
- export * from "./Core/Utils/Point.js";
3
2
  export * from "./Core/Utils/Ranges.js";
4
3
  export * from "./Core/Utils/Vectors.js";
5
4
  export * from "./Enums/Directions/MoveDirection.js";
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@tsparticles/engine",
3
- "version": "4.0.0-alpha.24",
3
+ "version": "4.0.0-alpha.25",
4
4
  "description": "Easily create highly customizable particle, confetti and fireworks animations and use them as animated backgrounds for your website. Ready to use components available also for React, Vue.js (2.x and 3.x), Angular, Svelte, jQuery, Preact, Riot.js, Inferno.",
5
5
  "homepage": "https://particles.js.org",
6
6
  "scripts": {
package/report.html CHANGED
@@ -3,7 +3,7 @@
3
3
  <head>
4
4
  <meta charset="UTF-8"/>
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1"/>
6
- <title>@tsparticles/engine [12 Feb 2026 at 18:48]</title>
6
+ <title>@tsparticles/engine [21 Feb 2026 at 12:47]</title>
7
7
  <link rel="shortcut icon" href="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAEAAAABACAMAAACdt4HsAAABrVBMVEUAAAD///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////+O1foceMD///+J0/qK1Pr7/v8Xdr/9///W8P4UdL7L7P0Scr2r4Pyj3vwad8D5/f/2/f+55f3E6f34+/2H0/ojfMKpzOd0rNgQcb3F3O/j9f7c8v6g3Pz0/P/w+v/q+P7n9v6T1/uQ1vuE0vqLut/y+v+Z2fvt+f+15Pzv9fuc2/vR7v2V2Pvd6/bg9P7I6/285/2y4/yp3/zp8vk8i8kqgMT7/P31+fyv4vxGkcz6/P6/6P3j7vfS5PNnpNUxhcbO7f7F6v3O4vHK3/DA2u631Ouy0eqXweKJud5wqthfoNMMbLvY8f73+v2dxeR8sNtTmdDx9/zX6PSjyeaCtd1YnNGX2PuQveCGt95Nls42h8dLlM3F4vBtAAAAM3RSTlMAAyOx0/sKBvik8opWGBMOAe3l1snDm2E9LSb06eHcu5JpHbarfHZCN9CBb08zzkdNS0kYaptYAAAFV0lEQVRYw92X51/aYBDHHS2O2qqttVbrqNq9m+TJIAYIShBkWwqIiCgoWvfeq7Z2/s29hyQNyUcR7LveGwVyXy6XH8/9rqxglLfUPLxVduUor3h0rfp2TYvpivk37929TkG037hffoX0+peVtZQc1589rigVUdXS/ABSAyEmGIO/1XfvldSK8vs3OqB6u3m0nxmIrvgB0dj7rr7Y9IbuF68hnfFaiHA/sxqm0wciIG43P60qKv9WXWc1RXGh/mFESFABTSBi0sNAKzqet17eCtOb3kZIDwxEEU0oAIJGYxNBDhBND29e0rtXXbcpuPmED9IhEAAQ/AXEaF8EPmnrrKsv0LvWR3fg5sWDNAFZOgAgaKvZDogHNU9MFwnnYROkc56RD5CjAbQX9Ow4g7upCsvYu55aSI/Nj0H1akgKQEUM94dwK65hYRmFU9MIcH/fqJYOZYcnuJSU/waKDgTOEVaVKhwrTRP5XzgSpAITYzom7UvkhFX5VutmxeNnWDjjswTKTyfgluNDGbUpWissXhF3s7mlSml+czWkg3D0l1nNjGNjz3myOQOa1KM/jOS6ebdbAVTCi4gljHSFrviza7tOgRWcS0MOUX9zdNgag5w7rRqA44Lzw0hr1WqES36dFliSJFlh2rXIae3FFcDDgKdxrUIDePr8jGcSClV1u7A9xeN0ModY/pHMxmR1EzRh8TJiwqsHmKW0l4FCEZI+jHio+JdPPE9qwQtTRxku2D8sIeRL2LnxWSllANCQGOIiqVHAz2ye2JR0DcH+HoxDkaADLjgxjKQ+AwCX/g0+DNgdG0ukYCONAe+dbc2IAc6fwt1ARoDSezNHxV2Cmzwv3O6lDMV55edBGwGK9n1+x2F8EDfAGCxug8MhpsMEcTEAWf3rx2vZhe/LAmtIn/6apE6PN0ULKgywD9mmdxbmFl3OvD5AS5fW5zLbv/YHmcsBTjf/afDz3MaZTVCfAP9z6/Bw6ycv8EUBWJIn9zYcoAWWlW9+OzO3vkTy8H+RANLmdrpOuYWdZYEXpo+TlCJrW5EARb7fF+bWdqf3hhyZI1nWJQHgznErZhbjoEsWqi8dQNoE294aldzFurwSABL2XXMf9+H1VQGke9exw5P/AnA5Pv5ngMul7LOvO922iwACu8WkCwLCafvM4CeWPxfA8lNHcWZSoi8EwMAIciKX2Z4SWCMAa3snCZ/G4EA8D6CMLNFsGQhkkz/gQNEBbPCbWsxGUpYVu3z8IyNAknwJkfPMEhLyrdi5RTyUVACkw4GSFRNWJNEW+fgPGwHD8/JxnRuLabN4CGNRkAE23na2+VmEAUmrYymSGjMAYqH84YUIyzgzs3XC7gNgH36Vcc4zKY9o9fgPBXUAiHHwVboBHGLiX6Zcjp1f2wu4tvzZKo0ecPnDtQYDQvJXaBeNzce45Fp28ZQLrEZVuFqgBwOalArKXnW1UzlnSusQKJqKYNuz4tOnI6sZG4zanpemv+7ySU2jbA9h6uhcgpfy6G2PahirDZ6zvq6zDduMVFTKvzw8wgyEdelwY9in3XkEPs3osJuwRQ4qTkfzifndg9Gfc4pdsu82+tTnHZTBa2EAMrqr2t43pguc8tNm7JQVQ2S0ukj2d22dhXYP0/veWtwKrCkNoNimAN5+Xr/oLrxswKbVJjteWrX7eR63o4j9q0GxnaBdWgGA5VStpanIjQmEhV0/nVt5VOFUvix6awJhPcAaTEShgrG+iGyvb5a0Ndb1YGHFPEwoqAinoaykaID1o1pdPNu7XsnCKQ3R+hwWIIhGvORcJUBYXe3Xa3vq/mF/N9V13ugufMkfXn+KHsRD0B8AAAAASUVORK5CYII=" type="image/x-icon" />
8
8
 
9
9
  <script>