@tsparticles/effect-trail 4.0.0-alpha.8 → 4.0.0-beta.0

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/543.min.js ADDED
@@ -0,0 +1 @@
1
+ "use strict";(this.webpackChunk_tsparticles_effect_trail=this.webpackChunk_tsparticles_effect_trail||[]).push([[543],{543(t,a,i){i.d(a,{TrailDrawer:()=>n});var e=i(303);let l={a:1,b:0,c:0,d:1};class n{drawAfter(t){let{context:a,drawPosition:i,drawRadius:n,drawScale:r,particle:h,transformData:o}=t,s=n*e.double,d=h.container.retina.pixelRatio,f=h.trail;if(!f||!h.trailLength)return;let c=h.trailLength*r+n;if(f.push({color:a.fillStyle||a.strokeStyle,position:{x:i.x,y:i.y},transformData:o}),f.length<3)return;let g=Math.floor(c);f.length>g&&f.splice(0,f.length-g);let x=Math.min(f.length,c),p={width:h.container.canvas.size.width*r+s,height:h.container.canvas.size.height*r+s};a.save(),a.lineCap="butt",a.lineJoin="round";for(let t=x-2;t>0;t--){let i=f[t+1],n=f[t],o=f[t-1];if(!i||!n||!o)continue;let g=i.position,u=n.position,y=o.position,b=h.trailTransform?n.transformData??l:l,{distance:m}=(0,e.getDistances)(g,u),{distance:M}=(0,e.getDistances)(u,y);if(m>c*e.double||M>c*e.double||Math.abs(g.x-u.x)>p.width*e.half||Math.abs(g.y-u.y)>p.height*e.half||Math.abs(u.x-y.x)>p.width*e.half||Math.abs(u.y-y.y)>p.height*e.half)continue;a.setTransform(b.a,b.b,b.c,b.d,u.x,u.y);let W=(g.x+u.x)*e.half-u.x,v=(g.y+u.y)*e.half-u.y,w=(u.x+y.x)*e.half-u.x,k=(u.y+y.y)*e.half-u.y;a.beginPath(),a.moveTo(W,v),a.quadraticCurveTo(e.originPoint.x,e.originPoint.y,w,k);let R=Math.max(t/x*s,d,(h.trailMinWidth??-1)*r),T=a.globalAlpha;a.globalAlpha=h.trailFade?t/x:e.defaultAlpha,a.lineWidth=h.trailMaxWidth?Math.min(R,h.trailMaxWidth*r):R,a.strokeStyle=n.color,a.stroke(),a.globalAlpha=T}a.restore()}particleInit(t,a){a.trail=[];let i=a.effectData;a.trailFade=i?.fade??!0,a.trailLength=(0,e.getRangeValue)(i?.length??10)*t.retina.pixelRatio,a.trailMaxWidth=i?.maxWidth?(0,e.getRangeValue)(i.maxWidth)*t.retina.pixelRatio:void 0,a.trailMinWidth=i?.minWidth?(0,e.getRangeValue)(i.minWidth)*t.retina.pixelRatio:void 0,a.trailTransform=i?.transform??!1}}}}]);
@@ -1,5 +1,5 @@
1
- import { defaultAlpha, double, getRangeValue, half, none, originPoint, } from "@tsparticles/engine";
2
- const minTrailLength = 2, trailLengthOffset = 1, minWidth = -1, defaultLength = 10;
1
+ import { defaultAlpha, double, getDistances, getRangeValue, half, originPoint, } from "@tsparticles/engine";
2
+ const minTrailLength = 3, trailLengthOffset = 1, minWidth = -1, firstIndex = 0, defaultLength = 10, loopTrailLengthOffset = 2, loopTrailLengthMinIndex = 0;
3
3
  const defaultTransform = {
4
4
  a: 1,
5
5
  b: 0,
@@ -8,11 +8,11 @@ const defaultTransform = {
8
8
  };
9
9
  export class TrailDrawer {
10
10
  drawAfter(data) {
11
- const { context, radius, particle, transformData } = data, diameter = radius * double, pxRatio = particle.container.retina.pixelRatio, currentPos = particle.getPosition(), trail = particle.trail;
11
+ const { context, drawPosition, drawRadius, drawScale, particle, transformData } = data, diameter = drawRadius * double, pxRatio = particle.container.retina.pixelRatio, trail = particle.trail;
12
12
  if (!trail || !particle.trailLength) {
13
13
  return;
14
14
  }
15
- const pathLength = particle.trailLength + radius;
15
+ const currentPos = drawPosition, pathLength = particle.trailLength * drawScale + drawRadius;
16
16
  trail.push({
17
17
  color: context.fillStyle || context.strokeStyle,
18
18
  position: {
@@ -24,46 +24,45 @@ export class TrailDrawer {
24
24
  if (trail.length < minTrailLength) {
25
25
  return;
26
26
  }
27
- while (trail.length > pathLength) {
28
- trail.shift();
27
+ const pathLengthFloor = Math.floor(pathLength);
28
+ if (trail.length > pathLengthFloor) {
29
+ trail.splice(firstIndex, trail.length - pathLengthFloor);
29
30
  }
30
31
  const trailLength = Math.min(trail.length, pathLength), canvasSize = {
31
- width: particle.container.canvas.size.width + diameter,
32
- height: particle.container.canvas.size.height + diameter,
32
+ width: particle.container.canvas.size.width * drawScale + diameter,
33
+ height: particle.container.canvas.size.height * drawScale + diameter,
33
34
  };
34
- const trailPos = trail[trailLength - trailLengthOffset];
35
- if (!trailPos) {
36
- return;
37
- }
38
- let lastPos = trailPos.position;
39
- for (let i = trailLength; i > none; i--) {
40
- const step = trail[i - trailLengthOffset];
41
- if (!step) {
35
+ context.save();
36
+ context.lineCap = "butt";
37
+ context.lineJoin = "round";
38
+ for (let i = trailLength - loopTrailLengthOffset; i > loopTrailLengthMinIndex; i--) {
39
+ const previousStep = trail[i + trailLengthOffset], step = trail[i], nextStep = trail[i - trailLengthOffset];
40
+ if (!previousStep || !step || !nextStep) {
42
41
  continue;
43
42
  }
44
- const position = step.position, stepTransformData = particle.trailTransform ? (step.transformData ?? defaultTransform) : defaultTransform;
45
- context.setTransform(stepTransformData.a, stepTransformData.b, stepTransformData.c, stepTransformData.d, position.x, position.y);
46
- context.beginPath();
47
- context.moveTo(lastPos.x - position.x, lastPos.y - position.y);
48
- const warp = {
49
- x: (lastPos.x + canvasSize.width) % canvasSize.width,
50
- y: (lastPos.y + canvasSize.height) % canvasSize.height,
51
- };
52
- if (Math.abs(lastPos.x - position.x) > canvasSize.width * half ||
53
- Math.abs(lastPos.y - position.y) > canvasSize.height * half) {
54
- lastPos = position;
43
+ const previousPosition = previousStep.position, position = step.position, nextPosition = nextStep.position, stepTransformData = particle.trailTransform ? (step.transformData ?? defaultTransform) : defaultTransform, { distance: previousDistance } = getDistances(previousPosition, position), { distance: nextDistance } = getDistances(position, nextPosition);
44
+ if (previousDistance > pathLength * double || nextDistance > pathLength * double) {
55
45
  continue;
56
46
  }
57
- context.lineTo(Math.abs(lastPos.x - position.x) > canvasSize.width * half ? warp.x : originPoint.x, Math.abs(lastPos.y - position.y) > canvasSize.height * half ? warp.y : originPoint.y);
58
- const width = Math.max((i / trailLength) * diameter, pxRatio, particle.trailMinWidth ?? minWidth), oldAlpha = context.globalAlpha;
47
+ if (Math.abs(previousPosition.x - position.x) > canvasSize.width * half ||
48
+ Math.abs(previousPosition.y - position.y) > canvasSize.height * half ||
49
+ Math.abs(position.x - nextPosition.x) > canvasSize.width * half ||
50
+ Math.abs(position.y - nextPosition.y) > canvasSize.height * half) {
51
+ continue;
52
+ }
53
+ context.setTransform(stepTransformData.a, stepTransformData.b, stepTransformData.c, stepTransformData.d, position.x, position.y);
54
+ const startX = (previousPosition.x + position.x) * half - position.x, startY = (previousPosition.y + position.y) * half - position.y, endX = (position.x + nextPosition.x) * half - position.x, endY = (position.y + nextPosition.y) * half - position.y;
55
+ context.beginPath();
56
+ context.moveTo(startX, startY);
57
+ context.quadraticCurveTo(originPoint.x, originPoint.y, endX, endY);
58
+ const width = Math.max((i / trailLength) * diameter, pxRatio, (particle.trailMinWidth ?? minWidth) * drawScale), oldAlpha = context.globalAlpha;
59
59
  context.globalAlpha = particle.trailFade ? i / trailLength : defaultAlpha;
60
- context.lineWidth = particle.trailMaxWidth ? Math.min(width, particle.trailMaxWidth) : width;
60
+ context.lineWidth = particle.trailMaxWidth ? Math.min(width, particle.trailMaxWidth * drawScale) : width;
61
61
  context.strokeStyle = step.color;
62
62
  context.stroke();
63
63
  context.globalAlpha = oldAlpha;
64
- lastPos = position;
65
64
  }
66
- context.setTransform(transformData.a, transformData.b, transformData.c, transformData.d, currentPos.x, currentPos.y);
65
+ context.restore();
67
66
  }
68
67
  particleInit(container, particle) {
69
68
  particle.trail = [];
package/browser/index.js CHANGED
@@ -1,7 +1,9 @@
1
1
  export async function loadTrailEffect(engine) {
2
- engine.checkVersion("4.0.0-alpha.8");
3
- await engine.register(async (e) => {
4
- const { TrailDrawer } = await import("./TrailDrawer.js");
5
- e.addEffect("trail", new TrailDrawer());
2
+ engine.checkVersion("4.0.0-beta.0");
3
+ await engine.register(e => {
4
+ e.addEffect("trail", async () => {
5
+ const { TrailDrawer } = await import("./TrailDrawer.js");
6
+ return new TrailDrawer();
7
+ });
6
8
  });
7
9
  }
@@ -1,5 +1,5 @@
1
- import { defaultAlpha, double, getRangeValue, half, none, originPoint, } from "@tsparticles/engine";
2
- const minTrailLength = 2, trailLengthOffset = 1, minWidth = -1, defaultLength = 10;
1
+ import { defaultAlpha, double, getDistances, getRangeValue, half, originPoint, } from "@tsparticles/engine";
2
+ const minTrailLength = 3, trailLengthOffset = 1, minWidth = -1, firstIndex = 0, defaultLength = 10, loopTrailLengthOffset = 2, loopTrailLengthMinIndex = 0;
3
3
  const defaultTransform = {
4
4
  a: 1,
5
5
  b: 0,
@@ -8,11 +8,11 @@ const defaultTransform = {
8
8
  };
9
9
  export class TrailDrawer {
10
10
  drawAfter(data) {
11
- const { context, radius, particle, transformData } = data, diameter = radius * double, pxRatio = particle.container.retina.pixelRatio, currentPos = particle.getPosition(), trail = particle.trail;
11
+ const { context, drawPosition, drawRadius, drawScale, particle, transformData } = data, diameter = drawRadius * double, pxRatio = particle.container.retina.pixelRatio, trail = particle.trail;
12
12
  if (!trail || !particle.trailLength) {
13
13
  return;
14
14
  }
15
- const pathLength = particle.trailLength + radius;
15
+ const currentPos = drawPosition, pathLength = particle.trailLength * drawScale + drawRadius;
16
16
  trail.push({
17
17
  color: context.fillStyle || context.strokeStyle,
18
18
  position: {
@@ -24,46 +24,45 @@ export class TrailDrawer {
24
24
  if (trail.length < minTrailLength) {
25
25
  return;
26
26
  }
27
- while (trail.length > pathLength) {
28
- trail.shift();
27
+ const pathLengthFloor = Math.floor(pathLength);
28
+ if (trail.length > pathLengthFloor) {
29
+ trail.splice(firstIndex, trail.length - pathLengthFloor);
29
30
  }
30
31
  const trailLength = Math.min(trail.length, pathLength), canvasSize = {
31
- width: particle.container.canvas.size.width + diameter,
32
- height: particle.container.canvas.size.height + diameter,
32
+ width: particle.container.canvas.size.width * drawScale + diameter,
33
+ height: particle.container.canvas.size.height * drawScale + diameter,
33
34
  };
34
- const trailPos = trail[trailLength - trailLengthOffset];
35
- if (!trailPos) {
36
- return;
37
- }
38
- let lastPos = trailPos.position;
39
- for (let i = trailLength; i > none; i--) {
40
- const step = trail[i - trailLengthOffset];
41
- if (!step) {
35
+ context.save();
36
+ context.lineCap = "butt";
37
+ context.lineJoin = "round";
38
+ for (let i = trailLength - loopTrailLengthOffset; i > loopTrailLengthMinIndex; i--) {
39
+ const previousStep = trail[i + trailLengthOffset], step = trail[i], nextStep = trail[i - trailLengthOffset];
40
+ if (!previousStep || !step || !nextStep) {
42
41
  continue;
43
42
  }
44
- const position = step.position, stepTransformData = particle.trailTransform ? (step.transformData ?? defaultTransform) : defaultTransform;
45
- context.setTransform(stepTransformData.a, stepTransformData.b, stepTransformData.c, stepTransformData.d, position.x, position.y);
46
- context.beginPath();
47
- context.moveTo(lastPos.x - position.x, lastPos.y - position.y);
48
- const warp = {
49
- x: (lastPos.x + canvasSize.width) % canvasSize.width,
50
- y: (lastPos.y + canvasSize.height) % canvasSize.height,
51
- };
52
- if (Math.abs(lastPos.x - position.x) > canvasSize.width * half ||
53
- Math.abs(lastPos.y - position.y) > canvasSize.height * half) {
54
- lastPos = position;
43
+ const previousPosition = previousStep.position, position = step.position, nextPosition = nextStep.position, stepTransformData = particle.trailTransform ? (step.transformData ?? defaultTransform) : defaultTransform, { distance: previousDistance } = getDistances(previousPosition, position), { distance: nextDistance } = getDistances(position, nextPosition);
44
+ if (previousDistance > pathLength * double || nextDistance > pathLength * double) {
55
45
  continue;
56
46
  }
57
- context.lineTo(Math.abs(lastPos.x - position.x) > canvasSize.width * half ? warp.x : originPoint.x, Math.abs(lastPos.y - position.y) > canvasSize.height * half ? warp.y : originPoint.y);
58
- const width = Math.max((i / trailLength) * diameter, pxRatio, particle.trailMinWidth ?? minWidth), oldAlpha = context.globalAlpha;
47
+ if (Math.abs(previousPosition.x - position.x) > canvasSize.width * half ||
48
+ Math.abs(previousPosition.y - position.y) > canvasSize.height * half ||
49
+ Math.abs(position.x - nextPosition.x) > canvasSize.width * half ||
50
+ Math.abs(position.y - nextPosition.y) > canvasSize.height * half) {
51
+ continue;
52
+ }
53
+ context.setTransform(stepTransformData.a, stepTransformData.b, stepTransformData.c, stepTransformData.d, position.x, position.y);
54
+ const startX = (previousPosition.x + position.x) * half - position.x, startY = (previousPosition.y + position.y) * half - position.y, endX = (position.x + nextPosition.x) * half - position.x, endY = (position.y + nextPosition.y) * half - position.y;
55
+ context.beginPath();
56
+ context.moveTo(startX, startY);
57
+ context.quadraticCurveTo(originPoint.x, originPoint.y, endX, endY);
58
+ const width = Math.max((i / trailLength) * diameter, pxRatio, (particle.trailMinWidth ?? minWidth) * drawScale), oldAlpha = context.globalAlpha;
59
59
  context.globalAlpha = particle.trailFade ? i / trailLength : defaultAlpha;
60
- context.lineWidth = particle.trailMaxWidth ? Math.min(width, particle.trailMaxWidth) : width;
60
+ context.lineWidth = particle.trailMaxWidth ? Math.min(width, particle.trailMaxWidth * drawScale) : width;
61
61
  context.strokeStyle = step.color;
62
62
  context.stroke();
63
63
  context.globalAlpha = oldAlpha;
64
- lastPos = position;
65
64
  }
66
- context.setTransform(transformData.a, transformData.b, transformData.c, transformData.d, currentPos.x, currentPos.y);
65
+ context.restore();
67
66
  }
68
67
  particleInit(container, particle) {
69
68
  particle.trail = [];
package/cjs/index.js CHANGED
@@ -1,7 +1,9 @@
1
1
  export async function loadTrailEffect(engine) {
2
- engine.checkVersion("4.0.0-alpha.8");
3
- await engine.register(async (e) => {
4
- const { TrailDrawer } = await import("./TrailDrawer.js");
5
- e.addEffect("trail", new TrailDrawer());
2
+ engine.checkVersion("4.0.0-beta.0");
3
+ await engine.register(e => {
4
+ e.addEffect("trail", async () => {
5
+ const { TrailDrawer } = await import("./TrailDrawer.js");
6
+ return new TrailDrawer();
7
+ });
6
8
  });
7
9
  }
@@ -4,7 +4,7 @@
4
4
  * Demo / Generator : https://particles.js.org/
5
5
  * GitHub : https://www.github.com/matteobruni/tsparticles
6
6
  * How to use? : Check the GitHub README
7
- * v4.0.0-alpha.8
7
+ * v4.0.0-beta.0
8
8
  */
9
9
  "use strict";
10
10
  /*
@@ -23,7 +23,7 @@
23
23
  \*************************************/
24
24
  (__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) {
25
25
 
26
- eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ TrailDrawer: () => (/* binding */ TrailDrawer)\n/* harmony export */ });\n/* harmony import */ var _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @tsparticles/engine */ \"@tsparticles/engine\");\n\nconst minTrailLength = 2,\n trailLengthOffset = 1,\n minWidth = -1,\n defaultLength = 10;\nconst defaultTransform = {\n a: 1,\n b: 0,\n c: 0,\n d: 1\n};\nclass TrailDrawer {\n drawAfter(data) {\n const {\n context,\n radius,\n particle,\n transformData\n } = data,\n diameter = radius * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.double,\n pxRatio = particle.container.retina.pixelRatio,\n currentPos = particle.getPosition(),\n trail = particle.trail;\n if (!trail || !particle.trailLength) {\n return;\n }\n const pathLength = particle.trailLength + radius;\n trail.push({\n color: context.fillStyle || context.strokeStyle,\n position: {\n x: currentPos.x,\n y: currentPos.y\n },\n transformData\n });\n if (trail.length < minTrailLength) {\n return;\n }\n while (trail.length > pathLength) {\n trail.shift();\n }\n const trailLength = Math.min(trail.length, pathLength),\n canvasSize = {\n width: particle.container.canvas.size.width + diameter,\n height: particle.container.canvas.size.height + diameter\n };\n const trailPos = trail[trailLength - trailLengthOffset];\n if (!trailPos) {\n return;\n }\n let lastPos = trailPos.position;\n for (let i = trailLength; i > _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.none; i--) {\n const step = trail[i - trailLengthOffset];\n if (!step) {\n continue;\n }\n const position = step.position,\n stepTransformData = particle.trailTransform ? step.transformData ?? defaultTransform : defaultTransform;\n context.setTransform(stepTransformData.a, stepTransformData.b, stepTransformData.c, stepTransformData.d, position.x, position.y);\n context.beginPath();\n context.moveTo(lastPos.x - position.x, lastPos.y - position.y);\n const warp = {\n x: (lastPos.x + canvasSize.width) % canvasSize.width,\n y: (lastPos.y + canvasSize.height) % canvasSize.height\n };\n if (Math.abs(lastPos.x - position.x) > canvasSize.width * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half || Math.abs(lastPos.y - position.y) > canvasSize.height * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half) {\n lastPos = position;\n continue;\n }\n context.lineTo(Math.abs(lastPos.x - position.x) > canvasSize.width * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half ? warp.x : _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.originPoint.x, Math.abs(lastPos.y - position.y) > canvasSize.height * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half ? warp.y : _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.originPoint.y);\n const width = Math.max(i / trailLength * diameter, pxRatio, particle.trailMinWidth ?? minWidth),\n oldAlpha = context.globalAlpha;\n context.globalAlpha = particle.trailFade ? i / trailLength : _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.defaultAlpha;\n context.lineWidth = particle.trailMaxWidth ? Math.min(width, particle.trailMaxWidth) : width;\n context.strokeStyle = step.color;\n context.stroke();\n context.globalAlpha = oldAlpha;\n lastPos = position;\n }\n context.setTransform(transformData.a, transformData.b, transformData.c, transformData.d, currentPos.x, currentPos.y);\n }\n particleInit(container, particle) {\n particle.trail = [];\n const effectData = particle.effectData;\n particle.trailFade = effectData?.fade ?? true;\n particle.trailLength = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeValue)(effectData?.length ?? defaultLength) * container.retina.pixelRatio;\n particle.trailMaxWidth = effectData?.maxWidth ? (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeValue)(effectData.maxWidth) * container.retina.pixelRatio : undefined;\n particle.trailMinWidth = effectData?.minWidth ? (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeValue)(effectData.minWidth) * container.retina.pixelRatio : undefined;\n particle.trailTransform = effectData?.transform ?? false;\n }\n}\n\n//# sourceURL=webpack://@tsparticles/effect-trail/./dist/browser/TrailDrawer.js?\n}");
26
+ eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ TrailDrawer: () => (/* binding */ TrailDrawer)\n/* harmony export */ });\n/* harmony import */ var _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @tsparticles/engine */ \"@tsparticles/engine\");\n\nconst minTrailLength = 3, trailLengthOffset = 1, minWidth = -1, firstIndex = 0, defaultLength = 10, loopTrailLengthOffset = 2, loopTrailLengthMinIndex = 0;\nconst defaultTransform = {\n a: 1,\n b: 0,\n c: 0,\n d: 1\n};\nclass TrailDrawer {\n drawAfter(data) {\n const { context, drawPosition, drawRadius, drawScale, particle, transformData } = data, diameter = drawRadius * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.double, pxRatio = particle.container.retina.pixelRatio, trail = particle.trail;\n if (!trail || !particle.trailLength) {\n return;\n }\n const currentPos = drawPosition, pathLength = particle.trailLength * drawScale + drawRadius;\n trail.push({\n color: context.fillStyle || context.strokeStyle,\n position: {\n x: currentPos.x,\n y: currentPos.y\n },\n transformData\n });\n if (trail.length < minTrailLength) {\n return;\n }\n const pathLengthFloor = Math.floor(pathLength);\n if (trail.length > pathLengthFloor) {\n trail.splice(firstIndex, trail.length - pathLengthFloor);\n }\n const trailLength = Math.min(trail.length, pathLength), canvasSize = {\n width: particle.container.canvas.size.width * drawScale + diameter,\n height: particle.container.canvas.size.height * drawScale + diameter\n };\n context.save();\n context.lineCap = \"butt\";\n context.lineJoin = \"round\";\n for(let i = trailLength - loopTrailLengthOffset; i > loopTrailLengthMinIndex; i--){\n const previousStep = trail[i + trailLengthOffset], step = trail[i], nextStep = trail[i - trailLengthOffset];\n if (!previousStep || !step || !nextStep) {\n continue;\n }\n const previousPosition = previousStep.position, position = step.position, nextPosition = nextStep.position, stepTransformData = particle.trailTransform ? step.transformData ?? defaultTransform : defaultTransform, { distance: previousDistance } = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getDistances)(previousPosition, position), { distance: nextDistance } = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getDistances)(position, nextPosition);\n if (previousDistance > pathLength * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.double || nextDistance > pathLength * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.double) {\n continue;\n }\n if (Math.abs(previousPosition.x - position.x) > canvasSize.width * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half || Math.abs(previousPosition.y - position.y) > canvasSize.height * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half || Math.abs(position.x - nextPosition.x) > canvasSize.width * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half || Math.abs(position.y - nextPosition.y) > canvasSize.height * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half) {\n continue;\n }\n context.setTransform(stepTransformData.a, stepTransformData.b, stepTransformData.c, stepTransformData.d, position.x, position.y);\n const startX = (previousPosition.x + position.x) * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half - position.x, startY = (previousPosition.y + position.y) * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half - position.y, endX = (position.x + nextPosition.x) * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half - position.x, endY = (position.y + nextPosition.y) * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half - position.y;\n context.beginPath();\n context.moveTo(startX, startY);\n context.quadraticCurveTo(_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.originPoint.x, _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.originPoint.y, endX, endY);\n const width = Math.max(i / trailLength * diameter, pxRatio, (particle.trailMinWidth ?? minWidth) * drawScale), oldAlpha = context.globalAlpha;\n context.globalAlpha = particle.trailFade ? i / trailLength : _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.defaultAlpha;\n context.lineWidth = particle.trailMaxWidth ? Math.min(width, particle.trailMaxWidth * drawScale) : width;\n context.strokeStyle = step.color;\n context.stroke();\n context.globalAlpha = oldAlpha;\n }\n context.restore();\n }\n particleInit(container, particle) {\n particle.trail = [];\n const effectData = particle.effectData;\n particle.trailFade = effectData?.fade ?? true;\n particle.trailLength = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeValue)(effectData?.length ?? defaultLength) * container.retina.pixelRatio;\n particle.trailMaxWidth = effectData?.maxWidth ? (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeValue)(effectData.maxWidth) * container.retina.pixelRatio : undefined;\n particle.trailMinWidth = effectData?.minWidth ? (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeValue)(effectData.minWidth) * container.retina.pixelRatio : undefined;\n particle.trailTransform = effectData?.transform ?? false;\n }\n}\n\n\n//# sourceURL=webpack://@tsparticles/effect-trail/./dist/browser/TrailDrawer.js?\n}");
27
27
 
28
28
  /***/ }
29
29
 
@@ -1,5 +1,5 @@
1
- import { defaultAlpha, double, getRangeValue, half, none, originPoint, } from "@tsparticles/engine";
2
- const minTrailLength = 2, trailLengthOffset = 1, minWidth = -1, defaultLength = 10;
1
+ import { defaultAlpha, double, getDistances, getRangeValue, half, originPoint, } from "@tsparticles/engine";
2
+ const minTrailLength = 3, trailLengthOffset = 1, minWidth = -1, firstIndex = 0, defaultLength = 10, loopTrailLengthOffset = 2, loopTrailLengthMinIndex = 0;
3
3
  const defaultTransform = {
4
4
  a: 1,
5
5
  b: 0,
@@ -8,11 +8,11 @@ const defaultTransform = {
8
8
  };
9
9
  export class TrailDrawer {
10
10
  drawAfter(data) {
11
- const { context, radius, particle, transformData } = data, diameter = radius * double, pxRatio = particle.container.retina.pixelRatio, currentPos = particle.getPosition(), trail = particle.trail;
11
+ const { context, drawPosition, drawRadius, drawScale, particle, transformData } = data, diameter = drawRadius * double, pxRatio = particle.container.retina.pixelRatio, trail = particle.trail;
12
12
  if (!trail || !particle.trailLength) {
13
13
  return;
14
14
  }
15
- const pathLength = particle.trailLength + radius;
15
+ const currentPos = drawPosition, pathLength = particle.trailLength * drawScale + drawRadius;
16
16
  trail.push({
17
17
  color: context.fillStyle || context.strokeStyle,
18
18
  position: {
@@ -24,46 +24,45 @@ export class TrailDrawer {
24
24
  if (trail.length < minTrailLength) {
25
25
  return;
26
26
  }
27
- while (trail.length > pathLength) {
28
- trail.shift();
27
+ const pathLengthFloor = Math.floor(pathLength);
28
+ if (trail.length > pathLengthFloor) {
29
+ trail.splice(firstIndex, trail.length - pathLengthFloor);
29
30
  }
30
31
  const trailLength = Math.min(trail.length, pathLength), canvasSize = {
31
- width: particle.container.canvas.size.width + diameter,
32
- height: particle.container.canvas.size.height + diameter,
32
+ width: particle.container.canvas.size.width * drawScale + diameter,
33
+ height: particle.container.canvas.size.height * drawScale + diameter,
33
34
  };
34
- const trailPos = trail[trailLength - trailLengthOffset];
35
- if (!trailPos) {
36
- return;
37
- }
38
- let lastPos = trailPos.position;
39
- for (let i = trailLength; i > none; i--) {
40
- const step = trail[i - trailLengthOffset];
41
- if (!step) {
35
+ context.save();
36
+ context.lineCap = "butt";
37
+ context.lineJoin = "round";
38
+ for (let i = trailLength - loopTrailLengthOffset; i > loopTrailLengthMinIndex; i--) {
39
+ const previousStep = trail[i + trailLengthOffset], step = trail[i], nextStep = trail[i - trailLengthOffset];
40
+ if (!previousStep || !step || !nextStep) {
42
41
  continue;
43
42
  }
44
- const position = step.position, stepTransformData = particle.trailTransform ? (step.transformData ?? defaultTransform) : defaultTransform;
45
- context.setTransform(stepTransformData.a, stepTransformData.b, stepTransformData.c, stepTransformData.d, position.x, position.y);
46
- context.beginPath();
47
- context.moveTo(lastPos.x - position.x, lastPos.y - position.y);
48
- const warp = {
49
- x: (lastPos.x + canvasSize.width) % canvasSize.width,
50
- y: (lastPos.y + canvasSize.height) % canvasSize.height,
51
- };
52
- if (Math.abs(lastPos.x - position.x) > canvasSize.width * half ||
53
- Math.abs(lastPos.y - position.y) > canvasSize.height * half) {
54
- lastPos = position;
43
+ const previousPosition = previousStep.position, position = step.position, nextPosition = nextStep.position, stepTransformData = particle.trailTransform ? (step.transformData ?? defaultTransform) : defaultTransform, { distance: previousDistance } = getDistances(previousPosition, position), { distance: nextDistance } = getDistances(position, nextPosition);
44
+ if (previousDistance > pathLength * double || nextDistance > pathLength * double) {
55
45
  continue;
56
46
  }
57
- context.lineTo(Math.abs(lastPos.x - position.x) > canvasSize.width * half ? warp.x : originPoint.x, Math.abs(lastPos.y - position.y) > canvasSize.height * half ? warp.y : originPoint.y);
58
- const width = Math.max((i / trailLength) * diameter, pxRatio, particle.trailMinWidth ?? minWidth), oldAlpha = context.globalAlpha;
47
+ if (Math.abs(previousPosition.x - position.x) > canvasSize.width * half ||
48
+ Math.abs(previousPosition.y - position.y) > canvasSize.height * half ||
49
+ Math.abs(position.x - nextPosition.x) > canvasSize.width * half ||
50
+ Math.abs(position.y - nextPosition.y) > canvasSize.height * half) {
51
+ continue;
52
+ }
53
+ context.setTransform(stepTransformData.a, stepTransformData.b, stepTransformData.c, stepTransformData.d, position.x, position.y);
54
+ const startX = (previousPosition.x + position.x) * half - position.x, startY = (previousPosition.y + position.y) * half - position.y, endX = (position.x + nextPosition.x) * half - position.x, endY = (position.y + nextPosition.y) * half - position.y;
55
+ context.beginPath();
56
+ context.moveTo(startX, startY);
57
+ context.quadraticCurveTo(originPoint.x, originPoint.y, endX, endY);
58
+ const width = Math.max((i / trailLength) * diameter, pxRatio, (particle.trailMinWidth ?? minWidth) * drawScale), oldAlpha = context.globalAlpha;
59
59
  context.globalAlpha = particle.trailFade ? i / trailLength : defaultAlpha;
60
- context.lineWidth = particle.trailMaxWidth ? Math.min(width, particle.trailMaxWidth) : width;
60
+ context.lineWidth = particle.trailMaxWidth ? Math.min(width, particle.trailMaxWidth * drawScale) : width;
61
61
  context.strokeStyle = step.color;
62
62
  context.stroke();
63
63
  context.globalAlpha = oldAlpha;
64
- lastPos = position;
65
64
  }
66
- context.setTransform(transformData.a, transformData.b, transformData.c, transformData.d, currentPos.x, currentPos.y);
65
+ context.restore();
67
66
  }
68
67
  particleInit(container, particle) {
69
68
  particle.trail = [];
package/esm/index.js CHANGED
@@ -1,7 +1,9 @@
1
1
  export async function loadTrailEffect(engine) {
2
- engine.checkVersion("4.0.0-alpha.8");
3
- await engine.register(async (e) => {
4
- const { TrailDrawer } = await import("./TrailDrawer.js");
5
- e.addEffect("trail", new TrailDrawer());
2
+ engine.checkVersion("4.0.0-beta.0");
3
+ await engine.register(e => {
4
+ e.addEffect("trail", async () => {
5
+ const { TrailDrawer } = await import("./TrailDrawer.js");
6
+ return new TrailDrawer();
7
+ });
6
8
  });
7
9
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@tsparticles/effect-trail",
3
- "version": "4.0.0-alpha.8",
3
+ "version": "4.0.0-beta.0",
4
4
  "description": "tsParticles trail effect",
5
5
  "homepage": "https://particles.js.org",
6
6
  "repository": {
@@ -100,7 +100,7 @@
100
100
  "./package.json": "./package.json"
101
101
  },
102
102
  "dependencies": {
103
- "@tsparticles/engine": "4.0.0-alpha.8"
103
+ "@tsparticles/engine": "4.0.0-beta.0"
104
104
  },
105
105
  "publishConfig": {
106
106
  "access": "public"
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/effect-trail [23 Jan 2026 at 23:40]</title>
6
+ <title>@tsparticles/effect-trail [19 Mar 2026 at 14:03]</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>
@@ -4,7 +4,7 @@
4
4
  * Demo / Generator : https://particles.js.org/
5
5
  * GitHub : https://www.github.com/matteobruni/tsparticles
6
6
  * How to use? : Check the GitHub README
7
- * v4.0.0-alpha.8
7
+ * v4.0.0-beta.0
8
8
  */
9
9
  /*
10
10
  * ATTENTION: The "eval" devtool has been used (maybe by default in mode: "development").
@@ -28,16 +28,6 @@ return /******/ (() => { // webpackBootstrap
28
28
  /******/ "use strict";
29
29
  /******/ var __webpack_modules__ = ({
30
30
 
31
- /***/ "./dist/browser/index.js"
32
- /*!*******************************!*\
33
- !*** ./dist/browser/index.js ***!
34
- \*******************************/
35
- (__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) {
36
-
37
- eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ loadTrailEffect: () => (/* binding */ loadTrailEffect)\n/* harmony export */ });\nasync function loadTrailEffect(engine) {\n engine.checkVersion(\"4.0.0-alpha.8\");\n await engine.register(async e => {\n const {\n TrailDrawer\n } = await __webpack_require__.e(/*! import() */ \"dist_browser_TrailDrawer_js\").then(__webpack_require__.bind(__webpack_require__, /*! ./TrailDrawer.js */ \"./dist/browser/TrailDrawer.js\"));\n e.addEffect(\"trail\", new TrailDrawer());\n });\n}\n\n//# sourceURL=webpack://@tsparticles/effect-trail/./dist/browser/index.js?\n}");
38
-
39
- /***/ },
40
-
41
31
  /***/ "@tsparticles/engine"
42
32
  /*!*********************************************************************************************************************************!*\
43
33
  !*** external {"commonjs":"@tsparticles/engine","commonjs2":"@tsparticles/engine","amd":"@tsparticles/engine","root":"window"} ***!
@@ -46,6 +36,16 @@ eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpa
46
36
 
47
37
  module.exports = __WEBPACK_EXTERNAL_MODULE__tsparticles_engine__;
48
38
 
39
+ /***/ },
40
+
41
+ /***/ "./dist/browser/index.js"
42
+ /*!*******************************!*\
43
+ !*** ./dist/browser/index.js ***!
44
+ \*******************************/
45
+ (__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) {
46
+
47
+ eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ loadTrailEffect: () => (/* binding */ loadTrailEffect)\n/* harmony export */ });\nasync function loadTrailEffect(engine) {\n engine.checkVersion(\"4.0.0-beta.0\");\n await engine.register((e)=>{\n e.addEffect(\"trail\", async ()=>{\n const { TrailDrawer } = await __webpack_require__.e(/*! import() */ \"dist_browser_TrailDrawer_js\").then(__webpack_require__.bind(__webpack_require__, /*! ./TrailDrawer.js */ \"./dist/browser/TrailDrawer.js\"));\n return new TrailDrawer();\n });\n });\n}\n\n\n//# sourceURL=webpack://@tsparticles/effect-trail/./dist/browser/index.js?\n}");
48
+
49
49
  /***/ }
50
50
 
51
51
  /******/ });
@@ -60,12 +60,6 @@ module.exports = __WEBPACK_EXTERNAL_MODULE__tsparticles_engine__;
60
60
  /******/ if (cachedModule !== undefined) {
61
61
  /******/ return cachedModule.exports;
62
62
  /******/ }
63
- /******/ // Check if module exists (development only)
64
- /******/ if (__webpack_modules__[moduleId] === undefined) {
65
- /******/ var e = new Error("Cannot find module '" + moduleId + "'");
66
- /******/ e.code = 'MODULE_NOT_FOUND';
67
- /******/ throw e;
68
- /******/ }
69
63
  /******/ // Create a new module (and put it into the cache)
70
64
  /******/ var module = __webpack_module_cache__[moduleId] = {
71
65
  /******/ // no module.id needed
@@ -74,6 +68,12 @@ module.exports = __WEBPACK_EXTERNAL_MODULE__tsparticles_engine__;
74
68
  /******/ };
75
69
  /******/
76
70
  /******/ // Execute the module function
71
+ /******/ if (!(moduleId in __webpack_modules__)) {
72
+ /******/ delete __webpack_module_cache__[moduleId];
73
+ /******/ var e = new Error("Cannot find module '" + moduleId + "'");
74
+ /******/ e.code = 'MODULE_NOT_FOUND';
75
+ /******/ throw e;
76
+ /******/ }
77
77
  /******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__);
78
78
  /******/
79
79
  /******/ // Return the exports of the module
@@ -118,6 +118,18 @@ module.exports = __WEBPACK_EXTERNAL_MODULE__tsparticles_engine__;
118
118
  /******/ };
119
119
  /******/ })();
120
120
  /******/
121
+ /******/ /* webpack/runtime/global */
122
+ /******/ (() => {
123
+ /******/ __webpack_require__.g = (function() {
124
+ /******/ if (typeof globalThis === 'object') return globalThis;
125
+ /******/ try {
126
+ /******/ return this || new Function('return this')();
127
+ /******/ } catch (e) {
128
+ /******/ if (typeof window === 'object') return window;
129
+ /******/ }
130
+ /******/ })();
131
+ /******/ })();
132
+ /******/
121
133
  /******/ /* webpack/runtime/hasOwnProperty shorthand */
122
134
  /******/ (() => {
123
135
  /******/ __webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))
@@ -182,8 +194,8 @@ module.exports = __WEBPACK_EXTERNAL_MODULE__tsparticles_engine__;
182
194
  /******/ /* webpack/runtime/publicPath */
183
195
  /******/ (() => {
184
196
  /******/ var scriptUrl;
185
- /******/ if (globalThis.importScripts) scriptUrl = globalThis.location + "";
186
- /******/ var document = globalThis.document;
197
+ /******/ if (__webpack_require__.g.importScripts) scriptUrl = __webpack_require__.g.location + "";
198
+ /******/ var document = __webpack_require__.g.document;
187
199
  /******/ if (!scriptUrl && document) {
188
200
  /******/ if (document.currentScript && document.currentScript.tagName.toUpperCase() === 'SCRIPT')
189
201
  /******/ scriptUrl = document.currentScript.src;
@@ -1,2 +1,2 @@
1
- /*! For license information please see tsparticles.effect.trail.min.js.LICENSE.txt */
2
- !function(e,t){if("object"==typeof exports&&"object"==typeof module)module.exports=t(require("@tsparticles/engine"));else if("function"==typeof define&&define.amd)define(["@tsparticles/engine"],t);else{var r="object"==typeof exports?t(require("@tsparticles/engine")):t(e.window);for(var o in r)("object"==typeof exports?exports:e)[o]=r[o]}}(this,(e=>(()=>{var t,r,o={303(t){t.exports=e}},a={};function i(e){var t=a[e];if(void 0!==t)return t.exports;var r=a[e]={exports:{}};return o[e](r,r.exports,i),r.exports}i.m=o,i.d=(e,t)=>{for(var r in t)i.o(t,r)&&!i.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},i.f={},i.e=e=>Promise.all(Object.keys(i.f).reduce(((t,r)=>(i.f[r](e,t),t)),[])),i.u=e=>e+".min.js",i.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),t={},r="@tsparticles/effect-trail:",i.l=(e,o,a,n)=>{if(t[e])t[e].push(o);else{var s,l;if(void 0!==a)for(var c=document.getElementsByTagName("script"),p=0;p<c.length;p++){var f=c[p];if(f.getAttribute("src")==e||f.getAttribute("data-webpack")==r+a){s=f;break}}s||(l=!0,(s=document.createElement("script")).charset="utf-8",i.nc&&s.setAttribute("nonce",i.nc),s.setAttribute("data-webpack",r+a),s.src=e),t[e]=[o];var u=(r,o)=>{s.onerror=s.onload=null,clearTimeout(d);var a=t[e];if(delete t[e],s.parentNode&&s.parentNode.removeChild(s),a&&a.forEach((e=>e(o))),r)return r(o)},d=setTimeout(u.bind(null,void 0,{type:"timeout",target:s}),12e4);s.onerror=u.bind(null,s.onerror),s.onload=u.bind(null,s.onload),l&&document.head.appendChild(s)}},i.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},(()=>{var e;globalThis.importScripts&&(e=globalThis.location+"");var t=globalThis.document;if(!e&&t&&(t.currentScript&&"SCRIPT"===t.currentScript.tagName.toUpperCase()&&(e=t.currentScript.src),!e)){var r=t.getElementsByTagName("script");if(r.length)for(var o=r.length-1;o>-1&&(!e||!/^http(s?):/.test(e));)e=r[o--].src}if(!e)throw new Error("Automatic publicPath is not supported in this browser");e=e.replace(/^blob:/,"").replace(/#.*$/,"").replace(/\?.*$/,"").replace(/\/[^\/]+$/,"/"),i.p=e})(),(()=>{var e={826:0};i.f.j=(t,r)=>{var o=i.o(e,t)?e[t]:void 0;if(0!==o)if(o)r.push(o[2]);else{var a=new Promise(((r,a)=>o=e[t]=[r,a]));r.push(o[2]=a);var n=i.p+i.u(t),s=new Error;i.l(n,(r=>{if(i.o(e,t)&&(0!==(o=e[t])&&(e[t]=void 0),o)){var a=r&&("load"===r.type?"missing":r.type),n=r&&r.target&&r.target.src;s.message="Loading chunk "+t+" failed.\n("+a+": "+n+")",s.name="ChunkLoadError",s.type=a,s.request=n,o[1](s)}}),"chunk-"+t,t)}};var t=(t,r)=>{var o,a,[n,s,l]=r,c=0;if(n.some((t=>0!==e[t]))){for(o in s)i.o(s,o)&&(i.m[o]=s[o]);if(l)l(i)}for(t&&t(r);c<n.length;c++)a=n[c],i.o(e,a)&&e[a]&&e[a][0](),e[a]=0},r=this.webpackChunk_tsparticles_effect_trail=this.webpackChunk_tsparticles_effect_trail||[];r.forEach(t.bind(null,0)),r.push=t.bind(null,r.push.bind(r))})();var n={};async function s(e){e.checkVersion("4.0.0-alpha.8"),await e.register((async e=>{const{TrailDrawer:t}=await i.e(593).then(i.bind(i,593));e.addEffect("trail",new t)}))}return i.r(n),i.d(n,{loadTrailEffect:()=>s}),n})()));
1
+ !function(e,t){if("object"==typeof exports&&"object"==typeof module)module.exports=t(require("@tsparticles/engine"));else if("function"==typeof define&&define.amd)define(["@tsparticles/engine"],t);else{var r="object"==typeof exports?t(require("@tsparticles/engine")):t(e.window);for(var o in r)("object"==typeof exports?exports:e)[o]=r[o]}}(this,e=>(()=>{"use strict";var t,r,o,i={303(t){t.exports=e}},n={};function a(e){var t=n[e];if(void 0!==t)return t.exports;var r=n[e]={exports:{}};return i[e](r,r.exports,a),r.exports}a.m=i,a.d=(e,t)=>{for(var r in t)a.o(t,r)&&!a.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},a.f={},a.e=e=>Promise.all(Object.keys(a.f).reduce((t,r)=>(a.f[r](e,t),t),[])),a.u=e=>""+e+".min.js",a.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||Function("return this")()}catch(e){if("object"==typeof window)return window}}(),a.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),c={},a.l=(e,t,r,o)=>{if(c[e])return void c[e].push(t);if(void 0!==r)for(var i,n,s=document.getElementsByTagName("script"),l=0;l<s.length;l++){var p=s[l];if(p.getAttribute("src")==e||p.getAttribute("data-webpack")=="@tsparticles/effect-trail:"+r){i=p;break}}i||(n=!0,(i=document.createElement("script")).charset="utf-8",a.nc&&i.setAttribute("nonce",a.nc),i.setAttribute("data-webpack","@tsparticles/effect-trail:"+r),i.src=e),c[e]=[t];var u=(t,r)=>{i.onerror=i.onload=null,clearTimeout(f);var o=c[e];if(delete c[e],i.parentNode&&i.parentNode.removeChild(i),o&&o.forEach(e=>e(r)),t)return t(r)},f=setTimeout(u.bind(null,void 0,{type:"timeout",target:i}),12e4);i.onerror=u.bind(null,i.onerror),i.onload=u.bind(null,i.onload),n&&document.head.appendChild(i)},a.r=e=>{"u">typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},a.g.importScripts&&(s=a.g.location+"");var c,s,l=a.g.document;if(!s&&l&&(l.currentScript&&"SCRIPT"===l.currentScript.tagName.toUpperCase()&&(s=l.currentScript.src),!s)){var p=l.getElementsByTagName("script");if(p.length)for(var u=p.length-1;u>-1&&(!s||!/^http(s?):/.test(s));)s=p[u--].src}if(!s)throw Error("Automatic publicPath is not supported in this browser");a.p=s=s.replace(/^blob:/,"").replace(/#.*$/,"").replace(/\?.*$/,"").replace(/\/[^\/]+$/,"/"),t={826:0},a.f.j=(e,r)=>{var o=a.o(t,e)?t[e]:void 0;if(0!==o)if(o)r.push(o[2]);else{var i=new Promise((r,i)=>o=t[e]=[r,i]);r.push(o[2]=i);var n=a.p+a.u(e),c=Error();a.l(n,r=>{if(a.o(t,e)&&(0!==(o=t[e])&&(t[e]=void 0),o)){var i=r&&("load"===r.type?"missing":r.type),n=r&&r.target&&r.target.src;c.message="Loading chunk "+e+` failed.
2
+ (`+i+": "+n+")",c.name="ChunkLoadError",c.type=i,c.request=n,o[1](c)}},"chunk-"+e,e)}},r=(e,r)=>{var o,i,[n,c,s]=r,l=0;if(n.some(e=>0!==t[e])){for(o in c)a.o(c,o)&&(a.m[o]=c[o]);s&&s(a)}for(e&&e(r);l<n.length;l++)i=n[l],a.o(t,i)&&t[i]&&t[i][0](),t[i]=0},(o=this.webpackChunk_tsparticles_effect_trail=this.webpackChunk_tsparticles_effect_trail||[]).forEach(r.bind(null,0)),o.push=r.bind(null,o.push.bind(o));var f={};async function d(e){e.checkVersion("4.0.0-beta.0"),await e.register(e=>{e.addEffect("trail",async()=>{let{TrailDrawer:e}=await a.e(543).then(a.bind(a,543));return new e})})}return a.r(f),a.d(f,{loadTrailEffect:()=>d}),f})());
@@ -11,7 +11,7 @@
11
11
  Object.defineProperty(exports, "__esModule", { value: true });
12
12
  exports.TrailDrawer = void 0;
13
13
  const engine_1 = require("@tsparticles/engine");
14
- const minTrailLength = 2, trailLengthOffset = 1, minWidth = -1, defaultLength = 10;
14
+ const minTrailLength = 3, trailLengthOffset = 1, minWidth = -1, firstIndex = 0, defaultLength = 10, loopTrailLengthOffset = 2, loopTrailLengthMinIndex = 0;
15
15
  const defaultTransform = {
16
16
  a: 1,
17
17
  b: 0,
@@ -20,11 +20,11 @@
20
20
  };
21
21
  class TrailDrawer {
22
22
  drawAfter(data) {
23
- const { context, radius, particle, transformData } = data, diameter = radius * engine_1.double, pxRatio = particle.container.retina.pixelRatio, currentPos = particle.getPosition(), trail = particle.trail;
23
+ const { context, drawPosition, drawRadius, drawScale, particle, transformData } = data, diameter = drawRadius * engine_1.double, pxRatio = particle.container.retina.pixelRatio, trail = particle.trail;
24
24
  if (!trail || !particle.trailLength) {
25
25
  return;
26
26
  }
27
- const pathLength = particle.trailLength + radius;
27
+ const currentPos = drawPosition, pathLength = particle.trailLength * drawScale + drawRadius;
28
28
  trail.push({
29
29
  color: context.fillStyle || context.strokeStyle,
30
30
  position: {
@@ -36,46 +36,45 @@
36
36
  if (trail.length < minTrailLength) {
37
37
  return;
38
38
  }
39
- while (trail.length > pathLength) {
40
- trail.shift();
39
+ const pathLengthFloor = Math.floor(pathLength);
40
+ if (trail.length > pathLengthFloor) {
41
+ trail.splice(firstIndex, trail.length - pathLengthFloor);
41
42
  }
42
43
  const trailLength = Math.min(trail.length, pathLength), canvasSize = {
43
- width: particle.container.canvas.size.width + diameter,
44
- height: particle.container.canvas.size.height + diameter,
44
+ width: particle.container.canvas.size.width * drawScale + diameter,
45
+ height: particle.container.canvas.size.height * drawScale + diameter,
45
46
  };
46
- const trailPos = trail[trailLength - trailLengthOffset];
47
- if (!trailPos) {
48
- return;
49
- }
50
- let lastPos = trailPos.position;
51
- for (let i = trailLength; i > engine_1.none; i--) {
52
- const step = trail[i - trailLengthOffset];
53
- if (!step) {
47
+ context.save();
48
+ context.lineCap = "butt";
49
+ context.lineJoin = "round";
50
+ for (let i = trailLength - loopTrailLengthOffset; i > loopTrailLengthMinIndex; i--) {
51
+ const previousStep = trail[i + trailLengthOffset], step = trail[i], nextStep = trail[i - trailLengthOffset];
52
+ if (!previousStep || !step || !nextStep) {
54
53
  continue;
55
54
  }
56
- const position = step.position, stepTransformData = particle.trailTransform ? (step.transformData ?? defaultTransform) : defaultTransform;
57
- context.setTransform(stepTransformData.a, stepTransformData.b, stepTransformData.c, stepTransformData.d, position.x, position.y);
58
- context.beginPath();
59
- context.moveTo(lastPos.x - position.x, lastPos.y - position.y);
60
- const warp = {
61
- x: (lastPos.x + canvasSize.width) % canvasSize.width,
62
- y: (lastPos.y + canvasSize.height) % canvasSize.height,
63
- };
64
- if (Math.abs(lastPos.x - position.x) > canvasSize.width * engine_1.half ||
65
- Math.abs(lastPos.y - position.y) > canvasSize.height * engine_1.half) {
66
- lastPos = position;
55
+ const previousPosition = previousStep.position, position = step.position, nextPosition = nextStep.position, stepTransformData = particle.trailTransform ? (step.transformData ?? defaultTransform) : defaultTransform, { distance: previousDistance } = (0, engine_1.getDistances)(previousPosition, position), { distance: nextDistance } = (0, engine_1.getDistances)(position, nextPosition);
56
+ if (previousDistance > pathLength * engine_1.double || nextDistance > pathLength * engine_1.double) {
67
57
  continue;
68
58
  }
69
- context.lineTo(Math.abs(lastPos.x - position.x) > canvasSize.width * engine_1.half ? warp.x : engine_1.originPoint.x, Math.abs(lastPos.y - position.y) > canvasSize.height * engine_1.half ? warp.y : engine_1.originPoint.y);
70
- const width = Math.max((i / trailLength) * diameter, pxRatio, particle.trailMinWidth ?? minWidth), oldAlpha = context.globalAlpha;
59
+ if (Math.abs(previousPosition.x - position.x) > canvasSize.width * engine_1.half ||
60
+ Math.abs(previousPosition.y - position.y) > canvasSize.height * engine_1.half ||
61
+ Math.abs(position.x - nextPosition.x) > canvasSize.width * engine_1.half ||
62
+ Math.abs(position.y - nextPosition.y) > canvasSize.height * engine_1.half) {
63
+ continue;
64
+ }
65
+ context.setTransform(stepTransformData.a, stepTransformData.b, stepTransformData.c, stepTransformData.d, position.x, position.y);
66
+ const startX = (previousPosition.x + position.x) * engine_1.half - position.x, startY = (previousPosition.y + position.y) * engine_1.half - position.y, endX = (position.x + nextPosition.x) * engine_1.half - position.x, endY = (position.y + nextPosition.y) * engine_1.half - position.y;
67
+ context.beginPath();
68
+ context.moveTo(startX, startY);
69
+ context.quadraticCurveTo(engine_1.originPoint.x, engine_1.originPoint.y, endX, endY);
70
+ const width = Math.max((i / trailLength) * diameter, pxRatio, (particle.trailMinWidth ?? minWidth) * drawScale), oldAlpha = context.globalAlpha;
71
71
  context.globalAlpha = particle.trailFade ? i / trailLength : engine_1.defaultAlpha;
72
- context.lineWidth = particle.trailMaxWidth ? Math.min(width, particle.trailMaxWidth) : width;
72
+ context.lineWidth = particle.trailMaxWidth ? Math.min(width, particle.trailMaxWidth * drawScale) : width;
73
73
  context.strokeStyle = step.color;
74
74
  context.stroke();
75
75
  context.globalAlpha = oldAlpha;
76
- lastPos = position;
77
76
  }
78
- context.setTransform(transformData.a, transformData.b, transformData.c, transformData.d, currentPos.x, currentPos.y);
77
+ context.restore();
79
78
  }
80
79
  particleInit(container, particle) {
81
80
  particle.trail = [];
package/umd/index.js CHANGED
@@ -45,10 +45,12 @@ var __importStar = (this && this.__importStar) || (function () {
45
45
  Object.defineProperty(exports, "__esModule", { value: true });
46
46
  exports.loadTrailEffect = loadTrailEffect;
47
47
  async function loadTrailEffect(engine) {
48
- engine.checkVersion("4.0.0-alpha.8");
49
- await engine.register(async (e) => {
50
- const { TrailDrawer } = await (__syncRequire ? Promise.resolve().then(() => __importStar(require("./TrailDrawer.js"))) : new Promise((resolve_1, reject_1) => { require(["./TrailDrawer.js"], resolve_1, reject_1); }).then(__importStar));
51
- e.addEffect("trail", new TrailDrawer());
48
+ engine.checkVersion("4.0.0-beta.0");
49
+ await engine.register(e => {
50
+ e.addEffect("trail", async () => {
51
+ const { TrailDrawer } = await (__syncRequire ? Promise.resolve().then(() => __importStar(require("./TrailDrawer.js"))) : new Promise((resolve_1, reject_1) => { require(["./TrailDrawer.js"], resolve_1, reject_1); }).then(__importStar));
52
+ return new TrailDrawer();
53
+ });
52
54
  });
53
55
  }
54
56
  });
package/593.min.js DELETED
@@ -1,2 +0,0 @@
1
- /*! For license information please see 593.min.js.LICENSE.txt */
2
- (this.webpackChunk_tsparticles_effect_trail=this.webpackChunk_tsparticles_effect_trail||[]).push([[593],{593(t,a,i){i.d(a,{TrailDrawer:()=>r});var e=i(303);const n={a:1,b:0,c:0,d:1};class r{drawAfter(t){const{context:a,radius:i,particle:r,transformData:l}=t,o=i*e.double,s=r.container.retina.pixelRatio,h=r.getPosition(),c=r.trail;if(!c||!r.trailLength)return;const f=r.trailLength+i;if(c.push({color:a.fillStyle||a.strokeStyle,position:{x:h.x,y:h.y},transformData:l}),c.length<2)return;for(;c.length>f;)c.shift();const x=Math.min(c.length,f),d=r.container.canvas.size.width+o,g=r.container.canvas.size.height+o,p=c[x-1];if(!p)return;let y=p.position;for(let t=x;t>e.none;t--){const i=c[t-1];if(!i)continue;const l=i.position,h=r.trailTransform?i.transformData??n:n;a.setTransform(h.a,h.b,h.c,h.d,l.x,l.y),a.beginPath(),a.moveTo(y.x-l.x,y.y-l.y);const f={x:(y.x+d)%d,y:(y.y+g)%g};if(Math.abs(y.x-l.x)>d*e.half||Math.abs(y.y-l.y)>g*e.half){y=l;continue}a.lineTo(Math.abs(y.x-l.x)>d*e.half?f.x:e.originPoint.x,Math.abs(y.y-l.y)>g*e.half?f.y:e.originPoint.y);const p=Math.max(t/x*o,s,r.trailMinWidth??-1),m=a.globalAlpha;a.globalAlpha=r.trailFade?t/x:e.defaultAlpha,a.lineWidth=r.trailMaxWidth?Math.min(p,r.trailMaxWidth):p,a.strokeStyle=i.color,a.stroke(),a.globalAlpha=m,y=l}a.setTransform(l.a,l.b,l.c,l.d,h.x,h.y)}particleInit(t,a){a.trail=[];const i=a.effectData;a.trailFade=i?.fade??!0,a.trailLength=(0,e.getRangeValue)(i?.length??10)*t.retina.pixelRatio,a.trailMaxWidth=i?.maxWidth?(0,e.getRangeValue)(i.maxWidth)*t.retina.pixelRatio:void 0,a.trailMinWidth=i?.minWidth?(0,e.getRangeValue)(i.minWidth)*t.retina.pixelRatio:void 0,a.trailTransform=i?.transform??!1}}}}]);
@@ -1 +0,0 @@
1
- /*! tsParticles Trail Shape v4.0.0-alpha.8 by Matteo Bruni */
@@ -1 +0,0 @@
1
- /*! tsParticles Trail Shape v4.0.0-alpha.8 by Matteo Bruni */