@tsparticles/interaction-external-bubble 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/537.min.js +1 -0
- package/README.md +5 -0
- package/browser/Bubbler.js +206 -198
- package/browser/Options/Classes/Bubble.js +1 -0
- package/browser/Options/Classes/BubbleBase.js +6 -0
- package/browser/Options/Classes/BubbleDiv.js +1 -0
- package/browser/index.js +3 -3
- package/cjs/Bubbler.js +206 -198
- package/cjs/Options/Classes/Bubble.js +1 -0
- package/cjs/Options/Classes/BubbleBase.js +6 -0
- package/cjs/Options/Classes/BubbleDiv.js +1 -0
- package/cjs/index.js +3 -3
- package/dist_browser_Bubbler_js.js +4 -4
- package/esm/Bubbler.js +206 -198
- package/esm/Options/Classes/Bubble.js +1 -0
- package/esm/Options/Classes/BubbleBase.js +6 -0
- package/esm/Options/Classes/BubbleDiv.js +1 -0
- package/esm/index.js +3 -3
- package/package.json +3 -3
- package/report.html +1 -1
- package/tsparticles.interaction.external.bubble.js +45 -33
- package/tsparticles.interaction.external.bubble.min.js +2 -2
- package/types/Bubbler.d.ts +2 -0
- package/umd/Bubbler.js +206 -198
- package/umd/Options/Classes/Bubble.js +1 -0
- package/umd/Options/Classes/BubbleBase.js +6 -0
- package/umd/Options/Classes/BubbleDiv.js +1 -0
- package/umd/index.js +3 -3
- package/132.min.js +0 -2
- package/132.min.js.LICENSE.txt +0 -1
- package/tsparticles.interaction.external.bubble.min.js.LICENSE.txt +0 -1
package/cjs/Bubbler.js
CHANGED
|
@@ -5,207 +5,13 @@ import { ProcessBubbleType } from "./Enums.js";
|
|
|
5
5
|
import { calculateBubbleValue } from "./Utils.js";
|
|
6
6
|
const bubbleMode = "bubble", minDistance = 0, defaultClickTime = 0, defaultOpacity = 1, ratioOffset = 1, defaultBubbleValue = 0, minRatio = 0, defaultRatio = 1;
|
|
7
7
|
export class Bubbler extends ExternalInteractorBase {
|
|
8
|
+
handleClickMode;
|
|
9
|
+
_engine;
|
|
10
|
+
_maxDistance;
|
|
8
11
|
constructor(engine, container) {
|
|
9
12
|
super(container);
|
|
10
|
-
this._clickBubble = interactivityData => {
|
|
11
|
-
const container = this.container, options = container.actualOptions, mouseClickPos = interactivityData.mouse.clickPosition, bubbleOptions = options.interactivity?.modes.bubble;
|
|
12
|
-
if (!bubbleOptions || !mouseClickPos) {
|
|
13
|
-
return;
|
|
14
|
-
}
|
|
15
|
-
container.bubble ??= {};
|
|
16
|
-
const distance = container.retina.bubbleModeDistance;
|
|
17
|
-
if (!distance || distance < minDistance) {
|
|
18
|
-
return;
|
|
19
|
-
}
|
|
20
|
-
const query = container.particles.quadTree.queryCircle(mouseClickPos, distance, p => this.isEnabled(interactivityData, p)), { bubble } = container;
|
|
21
|
-
for (const particle of query) {
|
|
22
|
-
if (!bubble.clicking) {
|
|
23
|
-
continue;
|
|
24
|
-
}
|
|
25
|
-
particle.bubble.inRange = !bubble.durationEnd;
|
|
26
|
-
const pos = particle.getPosition(), distMouse = getDistance(pos, mouseClickPos), timeSpent = (new Date().getTime() - (interactivityData.mouse.clickTime ?? defaultClickTime)) / millisecondsToSeconds;
|
|
27
|
-
if (timeSpent > bubbleOptions.duration) {
|
|
28
|
-
bubble.durationEnd = true;
|
|
29
|
-
}
|
|
30
|
-
if (timeSpent > bubbleOptions.duration * double) {
|
|
31
|
-
bubble.clicking = false;
|
|
32
|
-
bubble.durationEnd = false;
|
|
33
|
-
}
|
|
34
|
-
const sizeData = {
|
|
35
|
-
bubbleObj: {
|
|
36
|
-
optValue: container.retina.bubbleModeSize,
|
|
37
|
-
value: particle.bubble.radius,
|
|
38
|
-
},
|
|
39
|
-
particlesObj: {
|
|
40
|
-
optValue: getRangeMax(particle.options.size.value) * container.retina.pixelRatio,
|
|
41
|
-
value: particle.size.value,
|
|
42
|
-
},
|
|
43
|
-
type: ProcessBubbleType.size,
|
|
44
|
-
};
|
|
45
|
-
this._process(particle, distMouse, timeSpent, sizeData);
|
|
46
|
-
const opacityData = {
|
|
47
|
-
bubbleObj: {
|
|
48
|
-
optValue: bubbleOptions.opacity,
|
|
49
|
-
value: particle.bubble.opacity,
|
|
50
|
-
},
|
|
51
|
-
particlesObj: {
|
|
52
|
-
optValue: getRangeMax(particle.options.opacity.value),
|
|
53
|
-
value: particle.opacity?.value ?? defaultOpacity,
|
|
54
|
-
},
|
|
55
|
-
type: ProcessBubbleType.opacity,
|
|
56
|
-
};
|
|
57
|
-
this._process(particle, distMouse, timeSpent, opacityData);
|
|
58
|
-
if (!bubble.durationEnd && distMouse <= distance) {
|
|
59
|
-
this._hoverBubbleColor(particle, distMouse);
|
|
60
|
-
}
|
|
61
|
-
else {
|
|
62
|
-
delete particle.bubble.color;
|
|
63
|
-
}
|
|
64
|
-
}
|
|
65
|
-
};
|
|
66
|
-
this._hoverBubble = interactivityData => {
|
|
67
|
-
const container = this.container, mousePos = interactivityData.mouse.position, distance = container.retina.bubbleModeDistance;
|
|
68
|
-
if (!distance || distance < minDistance || !mousePos) {
|
|
69
|
-
return;
|
|
70
|
-
}
|
|
71
|
-
const query = container.particles.quadTree.queryCircle(mousePos, distance, p => this.isEnabled(interactivityData, p));
|
|
72
|
-
for (const particle of query) {
|
|
73
|
-
particle.bubble.inRange = true;
|
|
74
|
-
const pos = particle.getPosition(), pointDistance = getDistance(pos, mousePos), ratio = ratioOffset - pointDistance / distance;
|
|
75
|
-
if (pointDistance <= distance) {
|
|
76
|
-
if (ratio >= minRatio && interactivityData.status === mouseMoveEvent) {
|
|
77
|
-
this._hoverBubbleSize(particle, ratio);
|
|
78
|
-
this._hoverBubbleOpacity(particle, ratio);
|
|
79
|
-
this._hoverBubbleColor(particle, ratio);
|
|
80
|
-
}
|
|
81
|
-
}
|
|
82
|
-
else {
|
|
83
|
-
this.reset(interactivityData, particle);
|
|
84
|
-
}
|
|
85
|
-
if (interactivityData.status === mouseLeaveEvent) {
|
|
86
|
-
this.reset(interactivityData, particle);
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
};
|
|
90
|
-
this._hoverBubbleColor = (particle, ratio, divBubble) => {
|
|
91
|
-
const options = this.container.actualOptions, bubbleOptions = divBubble ?? options.interactivity?.modes.bubble;
|
|
92
|
-
if (!bubbleOptions) {
|
|
93
|
-
return;
|
|
94
|
-
}
|
|
95
|
-
if (!particle.bubble.finalColor) {
|
|
96
|
-
const modeColor = bubbleOptions.color;
|
|
97
|
-
if (!modeColor) {
|
|
98
|
-
return;
|
|
99
|
-
}
|
|
100
|
-
const bubbleColor = itemFromSingleOrMultiple(modeColor);
|
|
101
|
-
particle.bubble.finalColor = rangeColorToHsl(this._engine, bubbleColor);
|
|
102
|
-
}
|
|
103
|
-
if (!particle.bubble.finalColor) {
|
|
104
|
-
return;
|
|
105
|
-
}
|
|
106
|
-
if (bubbleOptions.mix) {
|
|
107
|
-
particle.bubble.color = undefined;
|
|
108
|
-
const pColor = particle.getFillColor();
|
|
109
|
-
particle.bubble.color = pColor
|
|
110
|
-
? rgbToHsl(colorMix(pColor, particle.bubble.finalColor, ratioOffset - ratio, ratio))
|
|
111
|
-
: particle.bubble.finalColor;
|
|
112
|
-
}
|
|
113
|
-
else {
|
|
114
|
-
particle.bubble.color = particle.bubble.finalColor;
|
|
115
|
-
}
|
|
116
|
-
};
|
|
117
|
-
this._hoverBubbleOpacity = (particle, ratio, divBubble) => {
|
|
118
|
-
const container = this.container, options = container.actualOptions, modeOpacity = divBubble?.opacity ?? options.interactivity?.modes.bubble?.opacity;
|
|
119
|
-
if (!modeOpacity) {
|
|
120
|
-
return;
|
|
121
|
-
}
|
|
122
|
-
const optOpacity = particle.options.opacity.value, pOpacity = particle.opacity?.value ?? defaultOpacity, opacity = calculateBubbleValue(pOpacity, modeOpacity, getRangeMax(optOpacity), ratio);
|
|
123
|
-
if (opacity !== undefined) {
|
|
124
|
-
particle.bubble.opacity = opacity;
|
|
125
|
-
}
|
|
126
|
-
};
|
|
127
|
-
this._hoverBubbleSize = (particle, ratio, divBubble) => {
|
|
128
|
-
const container = this.container, modeSize = divBubble?.size ? divBubble.size * container.retina.pixelRatio : container.retina.bubbleModeSize;
|
|
129
|
-
if (modeSize === undefined) {
|
|
130
|
-
return;
|
|
131
|
-
}
|
|
132
|
-
const optSize = getRangeMax(particle.options.size.value) * container.retina.pixelRatio, pSize = particle.size.value, size = calculateBubbleValue(pSize, modeSize, optSize, ratio);
|
|
133
|
-
if (size !== undefined) {
|
|
134
|
-
particle.bubble.radius = size;
|
|
135
|
-
}
|
|
136
|
-
};
|
|
137
|
-
this._process = (particle, distMouse, timeSpent, data) => {
|
|
138
|
-
const container = this.container, bubbleParam = data.bubbleObj.optValue, options = container.actualOptions, bubbleOptions = options.interactivity?.modes.bubble;
|
|
139
|
-
if (!bubbleOptions || bubbleParam === undefined) {
|
|
140
|
-
return;
|
|
141
|
-
}
|
|
142
|
-
const bubbleDuration = bubbleOptions.duration, bubbleDistance = container.retina.bubbleModeDistance, particlesParam = data.particlesObj.optValue, pObjBubble = data.bubbleObj.value, pObj = data.particlesObj.value ?? defaultBubbleValue, type = data.type;
|
|
143
|
-
if (!bubbleDistance || bubbleDistance < minDistance || bubbleParam === particlesParam) {
|
|
144
|
-
return;
|
|
145
|
-
}
|
|
146
|
-
container.bubble ??= {};
|
|
147
|
-
if (container.bubble.durationEnd) {
|
|
148
|
-
if (pObjBubble) {
|
|
149
|
-
if (type === ProcessBubbleType.size) {
|
|
150
|
-
delete particle.bubble.radius;
|
|
151
|
-
}
|
|
152
|
-
if (type === ProcessBubbleType.opacity) {
|
|
153
|
-
delete particle.bubble.opacity;
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
}
|
|
157
|
-
else {
|
|
158
|
-
if (distMouse <= bubbleDistance) {
|
|
159
|
-
const obj = pObjBubble ?? pObj;
|
|
160
|
-
if (obj !== bubbleParam) {
|
|
161
|
-
const value = pObj - (timeSpent * (pObj - bubbleParam)) / bubbleDuration;
|
|
162
|
-
if (type === ProcessBubbleType.size) {
|
|
163
|
-
particle.bubble.radius = value;
|
|
164
|
-
}
|
|
165
|
-
if (type === ProcessBubbleType.opacity) {
|
|
166
|
-
particle.bubble.opacity = value;
|
|
167
|
-
}
|
|
168
|
-
}
|
|
169
|
-
}
|
|
170
|
-
else {
|
|
171
|
-
if (type === ProcessBubbleType.size) {
|
|
172
|
-
delete particle.bubble.radius;
|
|
173
|
-
}
|
|
174
|
-
if (type === ProcessBubbleType.opacity) {
|
|
175
|
-
delete particle.bubble.opacity;
|
|
176
|
-
}
|
|
177
|
-
}
|
|
178
|
-
}
|
|
179
|
-
};
|
|
180
|
-
this._singleSelectorHover = (interactivityData, delta, selector, div) => {
|
|
181
|
-
const container = this.container, selectors = safeDocument().querySelectorAll(selector), bubble = container.actualOptions.interactivity?.modes.bubble;
|
|
182
|
-
if (!bubble || !selectors.length) {
|
|
183
|
-
return;
|
|
184
|
-
}
|
|
185
|
-
selectors.forEach(item => {
|
|
186
|
-
const elem = item, pxRatio = container.retina.pixelRatio, pos = {
|
|
187
|
-
x: (elem.offsetLeft + elem.offsetWidth * half) * pxRatio,
|
|
188
|
-
y: (elem.offsetTop + elem.offsetHeight * half) * pxRatio,
|
|
189
|
-
}, repulseRadius = elem.offsetWidth * half * pxRatio, area = div.type === DivType.circle
|
|
190
|
-
? new Circle(pos.x, pos.y, repulseRadius)
|
|
191
|
-
: new Rectangle(elem.offsetLeft * pxRatio, elem.offsetTop * pxRatio, elem.offsetWidth * pxRatio, elem.offsetHeight * pxRatio), query = container.particles.quadTree.query(area, p => this.isEnabled(interactivityData, p));
|
|
192
|
-
for (const particle of query) {
|
|
193
|
-
if (!area.contains(particle.getPosition())) {
|
|
194
|
-
continue;
|
|
195
|
-
}
|
|
196
|
-
particle.bubble.inRange = true;
|
|
197
|
-
const divs = bubble.divs, divBubble = divMode(divs, elem);
|
|
198
|
-
if (!particle.bubble.div || particle.bubble.div !== elem) {
|
|
199
|
-
this.clear(particle, delta, true);
|
|
200
|
-
particle.bubble.div = elem;
|
|
201
|
-
}
|
|
202
|
-
this._hoverBubbleSize(particle, defaultRatio, divBubble);
|
|
203
|
-
this._hoverBubbleOpacity(particle, defaultRatio, divBubble);
|
|
204
|
-
this._hoverBubbleColor(particle, defaultRatio, divBubble);
|
|
205
|
-
}
|
|
206
|
-
});
|
|
207
|
-
};
|
|
208
13
|
this._engine = engine;
|
|
14
|
+
this._maxDistance = 0;
|
|
209
15
|
container.bubble ??= {};
|
|
210
16
|
this.handleClickMode = (mode) => {
|
|
211
17
|
if (mode !== bubbleMode) {
|
|
@@ -215,6 +21,9 @@ export class Bubbler extends ExternalInteractorBase {
|
|
|
215
21
|
container.bubble.clicking = true;
|
|
216
22
|
};
|
|
217
23
|
}
|
|
24
|
+
get maxDistance() {
|
|
25
|
+
return this._maxDistance;
|
|
26
|
+
}
|
|
218
27
|
clear(particle, _delta, force) {
|
|
219
28
|
if (particle.bubble.inRange && !force) {
|
|
220
29
|
return;
|
|
@@ -229,6 +38,7 @@ export class Bubbler extends ExternalInteractorBase {
|
|
|
229
38
|
if (!bubble) {
|
|
230
39
|
return;
|
|
231
40
|
}
|
|
41
|
+
this._maxDistance = bubble.distance;
|
|
232
42
|
container.retina.bubbleModeDistance = bubble.distance * container.retina.pixelRatio;
|
|
233
43
|
if (bubble.size !== undefined) {
|
|
234
44
|
container.retina.bubbleModeSize = bubble.size * container.retina.pixelRatio;
|
|
@@ -272,4 +82,202 @@ export class Bubbler extends ExternalInteractorBase {
|
|
|
272
82
|
reset(_interactivityData, particle) {
|
|
273
83
|
particle.bubble.inRange = false;
|
|
274
84
|
}
|
|
85
|
+
_clickBubble = interactivityData => {
|
|
86
|
+
const container = this.container, options = container.actualOptions, mouseClickPos = interactivityData.mouse.clickPosition, bubbleOptions = options.interactivity?.modes.bubble;
|
|
87
|
+
if (!bubbleOptions || !mouseClickPos) {
|
|
88
|
+
return;
|
|
89
|
+
}
|
|
90
|
+
container.bubble ??= {};
|
|
91
|
+
const distance = container.retina.bubbleModeDistance;
|
|
92
|
+
if (!distance || distance < minDistance) {
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
const query = container.particles.grid.queryCircle(mouseClickPos, distance, p => this.isEnabled(interactivityData, p)), { bubble } = container;
|
|
96
|
+
for (const particle of query) {
|
|
97
|
+
if (!bubble.clicking) {
|
|
98
|
+
continue;
|
|
99
|
+
}
|
|
100
|
+
particle.bubble.inRange = !bubble.durationEnd;
|
|
101
|
+
const pos = particle.getPosition(), distMouse = getDistance(pos, mouseClickPos), timeSpent = (performance.now() - (interactivityData.mouse.clickTime ?? defaultClickTime)) / millisecondsToSeconds;
|
|
102
|
+
if (timeSpent > bubbleOptions.duration) {
|
|
103
|
+
bubble.durationEnd = true;
|
|
104
|
+
}
|
|
105
|
+
if (timeSpent > bubbleOptions.duration * double) {
|
|
106
|
+
bubble.clicking = false;
|
|
107
|
+
bubble.durationEnd = false;
|
|
108
|
+
}
|
|
109
|
+
const sizeData = {
|
|
110
|
+
bubbleObj: {
|
|
111
|
+
optValue: container.retina.bubbleModeSize,
|
|
112
|
+
value: particle.bubble.radius,
|
|
113
|
+
},
|
|
114
|
+
particlesObj: {
|
|
115
|
+
optValue: getRangeMax(particle.options.size.value) * container.retina.pixelRatio,
|
|
116
|
+
value: particle.size.value,
|
|
117
|
+
},
|
|
118
|
+
type: ProcessBubbleType.size,
|
|
119
|
+
};
|
|
120
|
+
this._process(particle, distMouse, timeSpent, sizeData);
|
|
121
|
+
const opacityData = {
|
|
122
|
+
bubbleObj: {
|
|
123
|
+
optValue: bubbleOptions.opacity,
|
|
124
|
+
value: particle.bubble.opacity,
|
|
125
|
+
},
|
|
126
|
+
particlesObj: {
|
|
127
|
+
optValue: getRangeMax(particle.options.opacity.value),
|
|
128
|
+
value: particle.opacity?.value ?? defaultOpacity,
|
|
129
|
+
},
|
|
130
|
+
type: ProcessBubbleType.opacity,
|
|
131
|
+
};
|
|
132
|
+
this._process(particle, distMouse, timeSpent, opacityData);
|
|
133
|
+
if (!bubble.durationEnd && distMouse <= distance) {
|
|
134
|
+
this._hoverBubbleColor(particle, distMouse);
|
|
135
|
+
}
|
|
136
|
+
else {
|
|
137
|
+
delete particle.bubble.color;
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
};
|
|
141
|
+
_hoverBubble = interactivityData => {
|
|
142
|
+
const container = this.container, mousePos = interactivityData.mouse.position, distance = container.retina.bubbleModeDistance;
|
|
143
|
+
if (!distance || distance < minDistance || !mousePos) {
|
|
144
|
+
return;
|
|
145
|
+
}
|
|
146
|
+
const query = container.particles.grid.queryCircle(mousePos, distance, p => this.isEnabled(interactivityData, p));
|
|
147
|
+
for (const particle of query) {
|
|
148
|
+
particle.bubble.inRange = true;
|
|
149
|
+
const pos = particle.getPosition(), pointDistance = getDistance(pos, mousePos), ratio = ratioOffset - pointDistance / distance;
|
|
150
|
+
if (pointDistance <= distance) {
|
|
151
|
+
if (ratio >= minRatio && interactivityData.status === mouseMoveEvent) {
|
|
152
|
+
this._hoverBubbleSize(particle, ratio);
|
|
153
|
+
this._hoverBubbleOpacity(particle, ratio);
|
|
154
|
+
this._hoverBubbleColor(particle, ratio);
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
else {
|
|
158
|
+
this.reset(interactivityData, particle);
|
|
159
|
+
}
|
|
160
|
+
if (interactivityData.status === mouseLeaveEvent) {
|
|
161
|
+
this.reset(interactivityData, particle);
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
};
|
|
165
|
+
_hoverBubbleColor = (particle, ratio, divBubble) => {
|
|
166
|
+
const options = this.container.actualOptions, bubbleOptions = divBubble ?? options.interactivity?.modes.bubble;
|
|
167
|
+
if (!bubbleOptions) {
|
|
168
|
+
return;
|
|
169
|
+
}
|
|
170
|
+
if (!particle.bubble.finalColor) {
|
|
171
|
+
const modeColor = bubbleOptions.color;
|
|
172
|
+
if (!modeColor) {
|
|
173
|
+
return;
|
|
174
|
+
}
|
|
175
|
+
const bubbleColor = itemFromSingleOrMultiple(modeColor);
|
|
176
|
+
particle.bubble.finalColor = rangeColorToHsl(this._engine, bubbleColor);
|
|
177
|
+
}
|
|
178
|
+
if (!particle.bubble.finalColor) {
|
|
179
|
+
return;
|
|
180
|
+
}
|
|
181
|
+
if (bubbleOptions.mix) {
|
|
182
|
+
particle.bubble.color = undefined;
|
|
183
|
+
const pColor = particle.getFillColor();
|
|
184
|
+
particle.bubble.color = pColor
|
|
185
|
+
? rgbToHsl(colorMix(pColor, particle.bubble.finalColor, ratioOffset - ratio, ratio))
|
|
186
|
+
: particle.bubble.finalColor;
|
|
187
|
+
}
|
|
188
|
+
else {
|
|
189
|
+
particle.bubble.color = particle.bubble.finalColor;
|
|
190
|
+
}
|
|
191
|
+
};
|
|
192
|
+
_hoverBubbleOpacity = (particle, ratio, divBubble) => {
|
|
193
|
+
const container = this.container, options = container.actualOptions, modeOpacity = divBubble?.opacity ?? options.interactivity?.modes.bubble?.opacity;
|
|
194
|
+
if (!modeOpacity) {
|
|
195
|
+
return;
|
|
196
|
+
}
|
|
197
|
+
const optOpacity = particle.options.opacity.value, pOpacity = particle.opacity?.value ?? defaultOpacity, opacity = calculateBubbleValue(pOpacity, modeOpacity, getRangeMax(optOpacity), ratio);
|
|
198
|
+
if (opacity !== undefined) {
|
|
199
|
+
particle.bubble.opacity = opacity;
|
|
200
|
+
}
|
|
201
|
+
};
|
|
202
|
+
_hoverBubbleSize = (particle, ratio, divBubble) => {
|
|
203
|
+
const container = this.container, modeSize = divBubble?.size ? divBubble.size * container.retina.pixelRatio : container.retina.bubbleModeSize;
|
|
204
|
+
if (modeSize === undefined) {
|
|
205
|
+
return;
|
|
206
|
+
}
|
|
207
|
+
const optSize = getRangeMax(particle.options.size.value) * container.retina.pixelRatio, pSize = particle.size.value, size = calculateBubbleValue(pSize, modeSize, optSize, ratio);
|
|
208
|
+
if (size !== undefined) {
|
|
209
|
+
particle.bubble.radius = size;
|
|
210
|
+
}
|
|
211
|
+
};
|
|
212
|
+
_process = (particle, distMouse, timeSpent, data) => {
|
|
213
|
+
const container = this.container, bubbleParam = data.bubbleObj.optValue, options = container.actualOptions, bubbleOptions = options.interactivity?.modes.bubble;
|
|
214
|
+
if (!bubbleOptions || bubbleParam === undefined) {
|
|
215
|
+
return;
|
|
216
|
+
}
|
|
217
|
+
const bubbleDuration = bubbleOptions.duration, bubbleDistance = container.retina.bubbleModeDistance, particlesParam = data.particlesObj.optValue, pObjBubble = data.bubbleObj.value, pObj = data.particlesObj.value ?? defaultBubbleValue, type = data.type;
|
|
218
|
+
if (!bubbleDistance || bubbleDistance < minDistance || bubbleParam === particlesParam) {
|
|
219
|
+
return;
|
|
220
|
+
}
|
|
221
|
+
container.bubble ??= {};
|
|
222
|
+
if (container.bubble.durationEnd) {
|
|
223
|
+
if (pObjBubble) {
|
|
224
|
+
if (type === ProcessBubbleType.size) {
|
|
225
|
+
delete particle.bubble.radius;
|
|
226
|
+
}
|
|
227
|
+
if (type === ProcessBubbleType.opacity) {
|
|
228
|
+
delete particle.bubble.opacity;
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
else {
|
|
233
|
+
if (distMouse <= bubbleDistance) {
|
|
234
|
+
const obj = pObjBubble ?? pObj;
|
|
235
|
+
if (obj !== bubbleParam) {
|
|
236
|
+
const value = pObj - (timeSpent * (pObj - bubbleParam)) / bubbleDuration;
|
|
237
|
+
if (type === ProcessBubbleType.size) {
|
|
238
|
+
particle.bubble.radius = value;
|
|
239
|
+
}
|
|
240
|
+
if (type === ProcessBubbleType.opacity) {
|
|
241
|
+
particle.bubble.opacity = value;
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
else {
|
|
246
|
+
if (type === ProcessBubbleType.size) {
|
|
247
|
+
delete particle.bubble.radius;
|
|
248
|
+
}
|
|
249
|
+
if (type === ProcessBubbleType.opacity) {
|
|
250
|
+
delete particle.bubble.opacity;
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
};
|
|
255
|
+
_singleSelectorHover = (interactivityData, delta, selector, div) => {
|
|
256
|
+
const container = this.container, selectors = safeDocument().querySelectorAll(selector), bubble = container.actualOptions.interactivity?.modes.bubble;
|
|
257
|
+
if (!bubble || !selectors.length) {
|
|
258
|
+
return;
|
|
259
|
+
}
|
|
260
|
+
selectors.forEach(item => {
|
|
261
|
+
const elem = item, pxRatio = container.retina.pixelRatio, pos = {
|
|
262
|
+
x: (elem.offsetLeft + elem.offsetWidth * half) * pxRatio,
|
|
263
|
+
y: (elem.offsetTop + elem.offsetHeight * half) * pxRatio,
|
|
264
|
+
}, repulseRadius = elem.offsetWidth * half * pxRatio, area = div.type === DivType.circle
|
|
265
|
+
? new Circle(pos.x, pos.y, repulseRadius)
|
|
266
|
+
: new Rectangle(elem.offsetLeft * pxRatio, elem.offsetTop * pxRatio, elem.offsetWidth * pxRatio, elem.offsetHeight * pxRatio), query = container.particles.grid.query(area, p => this.isEnabled(interactivityData, p));
|
|
267
|
+
for (const particle of query) {
|
|
268
|
+
if (!area.contains(particle.getPosition())) {
|
|
269
|
+
continue;
|
|
270
|
+
}
|
|
271
|
+
particle.bubble.inRange = true;
|
|
272
|
+
const divs = bubble.divs, divBubble = divMode(divs, elem);
|
|
273
|
+
if (!particle.bubble.div || particle.bubble.div !== elem) {
|
|
274
|
+
this.clear(particle, delta, true);
|
|
275
|
+
particle.bubble.div = elem;
|
|
276
|
+
}
|
|
277
|
+
this._hoverBubbleSize(particle, defaultRatio, divBubble);
|
|
278
|
+
this._hoverBubbleOpacity(particle, defaultRatio, divBubble);
|
|
279
|
+
this._hoverBubbleColor(particle, defaultRatio, divBubble);
|
|
280
|
+
}
|
|
281
|
+
});
|
|
282
|
+
};
|
|
275
283
|
}
|
|
@@ -2,6 +2,7 @@ import { executeOnSingleOrMultiple, isNull, } from "@tsparticles/engine";
|
|
|
2
2
|
import { BubbleBase } from "./BubbleBase.js";
|
|
3
3
|
import { BubbleDiv } from "./BubbleDiv.js";
|
|
4
4
|
export class Bubble extends BubbleBase {
|
|
5
|
+
divs;
|
|
5
6
|
load(data) {
|
|
6
7
|
super.load(data);
|
|
7
8
|
if (isNull(data)) {
|
package/cjs/index.js
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
export async function loadExternalBubbleInteraction(engine) {
|
|
2
|
-
engine.checkVersion("4.0.0-
|
|
2
|
+
engine.checkVersion("4.0.0-beta.0");
|
|
3
3
|
await engine.register(async (e) => {
|
|
4
|
-
const {
|
|
5
|
-
|
|
4
|
+
const { ensureInteractivityPluginLoaded } = await import("@tsparticles/plugin-interactivity");
|
|
5
|
+
ensureInteractivityPluginLoaded(e);
|
|
6
6
|
e.addInteractor?.("externalBubble", async (container) => {
|
|
7
7
|
const { Bubbler } = await import("./Bubbler.js");
|
|
8
8
|
return new Bubbler(e, container);
|
|
@@ -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-
|
|
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 */ Bubbler: () => (/* binding */ Bubbler)\n/* harmony export */ });\n/* harmony import */ var _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @tsparticles/engine */ \"@tsparticles/engine\");\n/* harmony import */ var _tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! @tsparticles/plugin-interactivity */ \"@tsparticles/plugin-interactivity\");\n/* harmony import */ var _Options_Classes_Bubble_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./Options/Classes/Bubble.js */ \"./dist/browser/Options/Classes/Bubble.js\");\n/* harmony import */ var _Enums_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./Enums.js */ \"./dist/browser/Enums.js\");\n/* harmony import */ var _Utils_js__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./Utils.js */ \"./dist/browser/Utils.js\");\n\n\n\n\n\nconst bubbleMode = \"bubble\",\n minDistance = 0,\n defaultClickTime = 0,\n defaultOpacity = 1,\n ratioOffset = 1,\n defaultBubbleValue = 0,\n minRatio = 0,\n defaultRatio = 1;\nclass Bubbler extends _tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.ExternalInteractorBase {\n constructor(engine, container) {\n super(container);\n this._clickBubble = interactivityData => {\n const container = this.container,\n options = container.actualOptions,\n mouseClickPos = interactivityData.mouse.clickPosition,\n bubbleOptions = options.interactivity?.modes.bubble;\n if (!bubbleOptions || !mouseClickPos) {\n return;\n }\n container.bubble ??= {};\n const distance = container.retina.bubbleModeDistance;\n if (!distance || distance < minDistance) {\n return;\n }\n const query = container.particles.quadTree.queryCircle(mouseClickPos, distance, p => this.isEnabled(interactivityData, p)),\n {\n bubble\n } = container;\n for (const particle of query) {\n if (!bubble.clicking) {\n continue;\n }\n particle.bubble.inRange = !bubble.durationEnd;\n const pos = particle.getPosition(),\n distMouse = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getDistance)(pos, mouseClickPos),\n timeSpent = (new Date().getTime() - (interactivityData.mouse.clickTime ?? defaultClickTime)) / _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.millisecondsToSeconds;\n if (timeSpent > bubbleOptions.duration) {\n bubble.durationEnd = true;\n }\n if (timeSpent > bubbleOptions.duration * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.double) {\n bubble.clicking = false;\n bubble.durationEnd = false;\n }\n const sizeData = {\n bubbleObj: {\n optValue: container.retina.bubbleModeSize,\n value: particle.bubble.radius\n },\n particlesObj: {\n optValue: (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeMax)(particle.options.size.value) * container.retina.pixelRatio,\n value: particle.size.value\n },\n type: _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.size\n };\n this._process(particle, distMouse, timeSpent, sizeData);\n const opacityData = {\n bubbleObj: {\n optValue: bubbleOptions.opacity,\n value: particle.bubble.opacity\n },\n particlesObj: {\n optValue: (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeMax)(particle.options.opacity.value),\n value: particle.opacity?.value ?? defaultOpacity\n },\n type: _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.opacity\n };\n this._process(particle, distMouse, timeSpent, opacityData);\n if (!bubble.durationEnd && distMouse <= distance) {\n this._hoverBubbleColor(particle, distMouse);\n } else {\n delete particle.bubble.color;\n }\n }\n };\n this._hoverBubble = interactivityData => {\n const container = this.container,\n mousePos = interactivityData.mouse.position,\n distance = container.retina.bubbleModeDistance;\n if (!distance || distance < minDistance || !mousePos) {\n return;\n }\n const query = container.particles.quadTree.queryCircle(mousePos, distance, p => this.isEnabled(interactivityData, p));\n for (const particle of query) {\n particle.bubble.inRange = true;\n const pos = particle.getPosition(),\n pointDistance = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getDistance)(pos, mousePos),\n ratio = ratioOffset - pointDistance / distance;\n if (pointDistance <= distance) {\n if (ratio >= minRatio && interactivityData.status === _tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.mouseMoveEvent) {\n this._hoverBubbleSize(particle, ratio);\n this._hoverBubbleOpacity(particle, ratio);\n this._hoverBubbleColor(particle, ratio);\n }\n } else {\n this.reset(interactivityData, particle);\n }\n if (interactivityData.status === _tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.mouseLeaveEvent) {\n this.reset(interactivityData, particle);\n }\n }\n };\n this._hoverBubbleColor = (particle, ratio, divBubble) => {\n const options = this.container.actualOptions,\n bubbleOptions = divBubble ?? options.interactivity?.modes.bubble;\n if (!bubbleOptions) {\n return;\n }\n if (!particle.bubble.finalColor) {\n const modeColor = bubbleOptions.color;\n if (!modeColor) {\n return;\n }\n const bubbleColor = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.itemFromSingleOrMultiple)(modeColor);\n particle.bubble.finalColor = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.rangeColorToHsl)(this._engine, bubbleColor);\n }\n if (!particle.bubble.finalColor) {\n return;\n }\n if (bubbleOptions.mix) {\n particle.bubble.color = undefined;\n const pColor = particle.getFillColor();\n particle.bubble.color = pColor ? (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.rgbToHsl)((0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.colorMix)(pColor, particle.bubble.finalColor, ratioOffset - ratio, ratio)) : particle.bubble.finalColor;\n } else {\n particle.bubble.color = particle.bubble.finalColor;\n }\n };\n this._hoverBubbleOpacity = (particle, ratio, divBubble) => {\n const container = this.container,\n options = container.actualOptions,\n modeOpacity = divBubble?.opacity ?? options.interactivity?.modes.bubble?.opacity;\n if (!modeOpacity) {\n return;\n }\n const optOpacity = particle.options.opacity.value,\n pOpacity = particle.opacity?.value ?? defaultOpacity,\n opacity = (0,_Utils_js__WEBPACK_IMPORTED_MODULE_4__.calculateBubbleValue)(pOpacity, modeOpacity, (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeMax)(optOpacity), ratio);\n if (opacity !== undefined) {\n particle.bubble.opacity = opacity;\n }\n };\n this._hoverBubbleSize = (particle, ratio, divBubble) => {\n const container = this.container,\n modeSize = divBubble?.size ? divBubble.size * container.retina.pixelRatio : container.retina.bubbleModeSize;\n if (modeSize === undefined) {\n return;\n }\n const optSize = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeMax)(particle.options.size.value) * container.retina.pixelRatio,\n pSize = particle.size.value,\n size = (0,_Utils_js__WEBPACK_IMPORTED_MODULE_4__.calculateBubbleValue)(pSize, modeSize, optSize, ratio);\n if (size !== undefined) {\n particle.bubble.radius = size;\n }\n };\n this._process = (particle, distMouse, timeSpent, data) => {\n const container = this.container,\n bubbleParam = data.bubbleObj.optValue,\n options = container.actualOptions,\n bubbleOptions = options.interactivity?.modes.bubble;\n if (!bubbleOptions || bubbleParam === undefined) {\n return;\n }\n const bubbleDuration = bubbleOptions.duration,\n bubbleDistance = container.retina.bubbleModeDistance,\n particlesParam = data.particlesObj.optValue,\n pObjBubble = data.bubbleObj.value,\n pObj = data.particlesObj.value ?? defaultBubbleValue,\n type = data.type;\n if (!bubbleDistance || bubbleDistance < minDistance || bubbleParam === particlesParam) {\n return;\n }\n container.bubble ??= {};\n if (container.bubble.durationEnd) {\n if (pObjBubble) {\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.size) {\n delete particle.bubble.radius;\n }\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.opacity) {\n delete particle.bubble.opacity;\n }\n }\n } else {\n if (distMouse <= bubbleDistance) {\n const obj = pObjBubble ?? pObj;\n if (obj !== bubbleParam) {\n const value = pObj - timeSpent * (pObj - bubbleParam) / bubbleDuration;\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.size) {\n particle.bubble.radius = value;\n }\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.opacity) {\n particle.bubble.opacity = value;\n }\n }\n } else {\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.size) {\n delete particle.bubble.radius;\n }\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.opacity) {\n delete particle.bubble.opacity;\n }\n }\n }\n };\n this._singleSelectorHover = (interactivityData, delta, selector, div) => {\n const container = this.container,\n selectors = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.safeDocument)().querySelectorAll(selector),\n bubble = container.actualOptions.interactivity?.modes.bubble;\n if (!bubble || !selectors.length) {\n return;\n }\n selectors.forEach(item => {\n const elem = item,\n pxRatio = container.retina.pixelRatio,\n pos = {\n x: (elem.offsetLeft + elem.offsetWidth * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half) * pxRatio,\n y: (elem.offsetTop + elem.offsetHeight * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half) * pxRatio\n },\n repulseRadius = elem.offsetWidth * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half * pxRatio,\n area = div.type === _tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.DivType.circle ? new _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.Circle(pos.x, pos.y, repulseRadius) : new _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.Rectangle(elem.offsetLeft * pxRatio, elem.offsetTop * pxRatio, elem.offsetWidth * pxRatio, elem.offsetHeight * pxRatio),\n query = container.particles.quadTree.query(area, p => this.isEnabled(interactivityData, p));\n for (const particle of query) {\n if (!area.contains(particle.getPosition())) {\n continue;\n }\n particle.bubble.inRange = true;\n const divs = bubble.divs,\n divBubble = (0,_tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.divMode)(divs, elem);\n if (!particle.bubble.div || particle.bubble.div !== elem) {\n this.clear(particle, delta, true);\n particle.bubble.div = elem;\n }\n this._hoverBubbleSize(particle, defaultRatio, divBubble);\n this._hoverBubbleOpacity(particle, defaultRatio, divBubble);\n this._hoverBubbleColor(particle, defaultRatio, divBubble);\n }\n });\n };\n this._engine = engine;\n container.bubble ??= {};\n this.handleClickMode = mode => {\n if (mode !== bubbleMode) {\n return;\n }\n container.bubble ??= {};\n container.bubble.clicking = true;\n };\n }\n clear(particle, _delta, force) {\n if (particle.bubble.inRange && !force) {\n return;\n }\n delete particle.bubble.div;\n delete particle.bubble.opacity;\n delete particle.bubble.radius;\n delete particle.bubble.color;\n }\n init() {\n const container = this.container,\n bubble = container.actualOptions.interactivity?.modes.bubble;\n if (!bubble) {\n return;\n }\n container.retina.bubbleModeDistance = bubble.distance * container.retina.pixelRatio;\n if (bubble.size !== undefined) {\n container.retina.bubbleModeSize = bubble.size * container.retina.pixelRatio;\n }\n }\n interact(interactivityData, delta) {\n const options = this.container.actualOptions,\n events = options.interactivity?.events;\n if (!events) {\n return;\n }\n const onHover = events.onHover,\n onClick = events.onClick,\n hoverEnabled = onHover.enable,\n hoverMode = onHover.mode,\n clickEnabled = onClick.enable,\n clickMode = onClick.mode,\n divs = events.onDiv;\n if (hoverEnabled && (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.isInArray)(bubbleMode, hoverMode)) {\n this._hoverBubble(interactivityData);\n } else if (clickEnabled && (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.isInArray)(bubbleMode, clickMode)) {\n this._clickBubble(interactivityData);\n } else {\n (0,_tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.divModeExecute)(bubbleMode, divs, (selector, div) => {\n this._singleSelectorHover(interactivityData, delta, selector, div);\n });\n }\n }\n isEnabled(interactivityData, particle) {\n const container = this.container,\n options = container.actualOptions,\n mouse = interactivityData.mouse,\n events = (particle?.interactivity ?? options.interactivity)?.events;\n if (!events) {\n return false;\n }\n const {\n onClick,\n onDiv,\n onHover\n } = events,\n divBubble = (0,_tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.isDivModeEnabled)(bubbleMode, onDiv);\n if (!(divBubble || onHover.enable && !!mouse.position || onClick.enable && mouse.clickPosition)) {\n return false;\n }\n return (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.isInArray)(bubbleMode, onHover.mode) || (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.isInArray)(bubbleMode, onClick.mode) || divBubble;\n }\n loadModeOptions(options, ...sources) {\n options.bubble ??= new _Options_Classes_Bubble_js__WEBPACK_IMPORTED_MODULE_2__.Bubble();\n for (const source of sources) {\n options.bubble.load(source?.bubble);\n }\n }\n reset(_interactivityData, particle) {\n particle.bubble.inRange = false;\n }\n}\n\n//# sourceURL=webpack://@tsparticles/interaction-external-bubble/./dist/browser/Bubbler.js?\n}");
|
|
26
|
+
eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ Bubbler: () => (/* binding */ Bubbler)\n/* harmony export */ });\n/* harmony import */ var _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @tsparticles/engine */ \"@tsparticles/engine\");\n/* harmony import */ var _tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! @tsparticles/plugin-interactivity */ \"@tsparticles/plugin-interactivity\");\n/* harmony import */ var _Options_Classes_Bubble_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./Options/Classes/Bubble.js */ \"./dist/browser/Options/Classes/Bubble.js\");\n/* harmony import */ var _Enums_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./Enums.js */ \"./dist/browser/Enums.js\");\n/* harmony import */ var _Utils_js__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./Utils.js */ \"./dist/browser/Utils.js\");\n\n\n\n\n\nconst bubbleMode = \"bubble\", minDistance = 0, defaultClickTime = 0, defaultOpacity = 1, ratioOffset = 1, defaultBubbleValue = 0, minRatio = 0, defaultRatio = 1;\nclass Bubbler extends _tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.ExternalInteractorBase {\n handleClickMode;\n _engine;\n _maxDistance;\n constructor(engine, container){\n super(container);\n this._engine = engine;\n this._maxDistance = 0;\n container.bubble ??= {};\n this.handleClickMode = (mode)=>{\n if (mode !== bubbleMode) {\n return;\n }\n container.bubble ??= {};\n container.bubble.clicking = true;\n };\n }\n get maxDistance() {\n return this._maxDistance;\n }\n clear(particle, _delta, force) {\n if (particle.bubble.inRange && !force) {\n return;\n }\n delete particle.bubble.div;\n delete particle.bubble.opacity;\n delete particle.bubble.radius;\n delete particle.bubble.color;\n }\n init() {\n const container = this.container, bubble = container.actualOptions.interactivity?.modes.bubble;\n if (!bubble) {\n return;\n }\n this._maxDistance = bubble.distance;\n container.retina.bubbleModeDistance = bubble.distance * container.retina.pixelRatio;\n if (bubble.size !== undefined) {\n container.retina.bubbleModeSize = bubble.size * container.retina.pixelRatio;\n }\n }\n interact(interactivityData, delta) {\n const options = this.container.actualOptions, events = options.interactivity?.events;\n if (!events) {\n return;\n }\n const onHover = events.onHover, onClick = events.onClick, hoverEnabled = onHover.enable, hoverMode = onHover.mode, clickEnabled = onClick.enable, clickMode = onClick.mode, divs = events.onDiv;\n if (hoverEnabled && (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.isInArray)(bubbleMode, hoverMode)) {\n this._hoverBubble(interactivityData);\n } else if (clickEnabled && (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.isInArray)(bubbleMode, clickMode)) {\n this._clickBubble(interactivityData);\n } else {\n (0,_tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.divModeExecute)(bubbleMode, divs, (selector, div)=>{\n this._singleSelectorHover(interactivityData, delta, selector, div);\n });\n }\n }\n isEnabled(interactivityData, particle) {\n const container = this.container, options = container.actualOptions, mouse = interactivityData.mouse, events = (particle?.interactivity ?? options.interactivity)?.events;\n if (!events) {\n return false;\n }\n const { onClick, onDiv, onHover } = events, divBubble = (0,_tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.isDivModeEnabled)(bubbleMode, onDiv);\n if (!(divBubble || onHover.enable && !!mouse.position || onClick.enable && mouse.clickPosition)) {\n return false;\n }\n return (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.isInArray)(bubbleMode, onHover.mode) || (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.isInArray)(bubbleMode, onClick.mode) || divBubble;\n }\n loadModeOptions(options, ...sources) {\n options.bubble ??= new _Options_Classes_Bubble_js__WEBPACK_IMPORTED_MODULE_2__.Bubble();\n for (const source of sources){\n options.bubble.load(source?.bubble);\n }\n }\n reset(_interactivityData, particle) {\n particle.bubble.inRange = false;\n }\n _clickBubble = (interactivityData)=>{\n const container = this.container, options = container.actualOptions, mouseClickPos = interactivityData.mouse.clickPosition, bubbleOptions = options.interactivity?.modes.bubble;\n if (!bubbleOptions || !mouseClickPos) {\n return;\n }\n container.bubble ??= {};\n const distance = container.retina.bubbleModeDistance;\n if (!distance || distance < minDistance) {\n return;\n }\n const query = container.particles.grid.queryCircle(mouseClickPos, distance, (p)=>this.isEnabled(interactivityData, p)), { bubble } = container;\n for (const particle of query){\n if (!bubble.clicking) {\n continue;\n }\n particle.bubble.inRange = !bubble.durationEnd;\n const pos = particle.getPosition(), distMouse = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getDistance)(pos, mouseClickPos), timeSpent = (performance.now() - (interactivityData.mouse.clickTime ?? defaultClickTime)) / _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.millisecondsToSeconds;\n if (timeSpent > bubbleOptions.duration) {\n bubble.durationEnd = true;\n }\n if (timeSpent > bubbleOptions.duration * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.double) {\n bubble.clicking = false;\n bubble.durationEnd = false;\n }\n const sizeData = {\n bubbleObj: {\n optValue: container.retina.bubbleModeSize,\n value: particle.bubble.radius\n },\n particlesObj: {\n optValue: (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeMax)(particle.options.size.value) * container.retina.pixelRatio,\n value: particle.size.value\n },\n type: _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.size\n };\n this._process(particle, distMouse, timeSpent, sizeData);\n const opacityData = {\n bubbleObj: {\n optValue: bubbleOptions.opacity,\n value: particle.bubble.opacity\n },\n particlesObj: {\n optValue: (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeMax)(particle.options.opacity.value),\n value: particle.opacity?.value ?? defaultOpacity\n },\n type: _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.opacity\n };\n this._process(particle, distMouse, timeSpent, opacityData);\n if (!bubble.durationEnd && distMouse <= distance) {\n this._hoverBubbleColor(particle, distMouse);\n } else {\n delete particle.bubble.color;\n }\n }\n };\n _hoverBubble = (interactivityData)=>{\n const container = this.container, mousePos = interactivityData.mouse.position, distance = container.retina.bubbleModeDistance;\n if (!distance || distance < minDistance || !mousePos) {\n return;\n }\n const query = container.particles.grid.queryCircle(mousePos, distance, (p)=>this.isEnabled(interactivityData, p));\n for (const particle of query){\n particle.bubble.inRange = true;\n const pos = particle.getPosition(), pointDistance = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getDistance)(pos, mousePos), ratio = ratioOffset - pointDistance / distance;\n if (pointDistance <= distance) {\n if (ratio >= minRatio && interactivityData.status === _tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.mouseMoveEvent) {\n this._hoverBubbleSize(particle, ratio);\n this._hoverBubbleOpacity(particle, ratio);\n this._hoverBubbleColor(particle, ratio);\n }\n } else {\n this.reset(interactivityData, particle);\n }\n if (interactivityData.status === _tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.mouseLeaveEvent) {\n this.reset(interactivityData, particle);\n }\n }\n };\n _hoverBubbleColor = (particle, ratio, divBubble)=>{\n const options = this.container.actualOptions, bubbleOptions = divBubble ?? options.interactivity?.modes.bubble;\n if (!bubbleOptions) {\n return;\n }\n if (!particle.bubble.finalColor) {\n const modeColor = bubbleOptions.color;\n if (!modeColor) {\n return;\n }\n const bubbleColor = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.itemFromSingleOrMultiple)(modeColor);\n particle.bubble.finalColor = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.rangeColorToHsl)(this._engine, bubbleColor);\n }\n if (!particle.bubble.finalColor) {\n return;\n }\n if (bubbleOptions.mix) {\n particle.bubble.color = undefined;\n const pColor = particle.getFillColor();\n particle.bubble.color = pColor ? (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.rgbToHsl)((0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.colorMix)(pColor, particle.bubble.finalColor, ratioOffset - ratio, ratio)) : particle.bubble.finalColor;\n } else {\n particle.bubble.color = particle.bubble.finalColor;\n }\n };\n _hoverBubbleOpacity = (particle, ratio, divBubble)=>{\n const container = this.container, options = container.actualOptions, modeOpacity = divBubble?.opacity ?? options.interactivity?.modes.bubble?.opacity;\n if (!modeOpacity) {\n return;\n }\n const optOpacity = particle.options.opacity.value, pOpacity = particle.opacity?.value ?? defaultOpacity, opacity = (0,_Utils_js__WEBPACK_IMPORTED_MODULE_4__.calculateBubbleValue)(pOpacity, modeOpacity, (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeMax)(optOpacity), ratio);\n if (opacity !== undefined) {\n particle.bubble.opacity = opacity;\n }\n };\n _hoverBubbleSize = (particle, ratio, divBubble)=>{\n const container = this.container, modeSize = divBubble?.size ? divBubble.size * container.retina.pixelRatio : container.retina.bubbleModeSize;\n if (modeSize === undefined) {\n return;\n }\n const optSize = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.getRangeMax)(particle.options.size.value) * container.retina.pixelRatio, pSize = particle.size.value, size = (0,_Utils_js__WEBPACK_IMPORTED_MODULE_4__.calculateBubbleValue)(pSize, modeSize, optSize, ratio);\n if (size !== undefined) {\n particle.bubble.radius = size;\n }\n };\n _process = (particle, distMouse, timeSpent, data)=>{\n const container = this.container, bubbleParam = data.bubbleObj.optValue, options = container.actualOptions, bubbleOptions = options.interactivity?.modes.bubble;\n if (!bubbleOptions || bubbleParam === undefined) {\n return;\n }\n const bubbleDuration = bubbleOptions.duration, bubbleDistance = container.retina.bubbleModeDistance, particlesParam = data.particlesObj.optValue, pObjBubble = data.bubbleObj.value, pObj = data.particlesObj.value ?? defaultBubbleValue, type = data.type;\n if (!bubbleDistance || bubbleDistance < minDistance || bubbleParam === particlesParam) {\n return;\n }\n container.bubble ??= {};\n if (container.bubble.durationEnd) {\n if (pObjBubble) {\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.size) {\n delete particle.bubble.radius;\n }\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.opacity) {\n delete particle.bubble.opacity;\n }\n }\n } else {\n if (distMouse <= bubbleDistance) {\n const obj = pObjBubble ?? pObj;\n if (obj !== bubbleParam) {\n const value = pObj - timeSpent * (pObj - bubbleParam) / bubbleDuration;\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.size) {\n particle.bubble.radius = value;\n }\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.opacity) {\n particle.bubble.opacity = value;\n }\n }\n } else {\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.size) {\n delete particle.bubble.radius;\n }\n if (type === _Enums_js__WEBPACK_IMPORTED_MODULE_3__.ProcessBubbleType.opacity) {\n delete particle.bubble.opacity;\n }\n }\n }\n };\n _singleSelectorHover = (interactivityData, delta, selector, div)=>{\n const container = this.container, selectors = (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.safeDocument)().querySelectorAll(selector), bubble = container.actualOptions.interactivity?.modes.bubble;\n if (!bubble || !selectors.length) {\n return;\n }\n selectors.forEach((item)=>{\n const elem = item, pxRatio = container.retina.pixelRatio, pos = {\n x: (elem.offsetLeft + elem.offsetWidth * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half) * pxRatio,\n y: (elem.offsetTop + elem.offsetHeight * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half) * pxRatio\n }, repulseRadius = elem.offsetWidth * _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.half * pxRatio, area = div.type === _tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.DivType.circle ? new _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.Circle(pos.x, pos.y, repulseRadius) : new _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.Rectangle(elem.offsetLeft * pxRatio, elem.offsetTop * pxRatio, elem.offsetWidth * pxRatio, elem.offsetHeight * pxRatio), query = container.particles.grid.query(area, (p)=>this.isEnabled(interactivityData, p));\n for (const particle of query){\n if (!area.contains(particle.getPosition())) {\n continue;\n }\n particle.bubble.inRange = true;\n const divs = bubble.divs, divBubble = (0,_tsparticles_plugin_interactivity__WEBPACK_IMPORTED_MODULE_1__.divMode)(divs, elem);\n if (!particle.bubble.div || particle.bubble.div !== elem) {\n this.clear(particle, delta, true);\n particle.bubble.div = elem;\n }\n this._hoverBubbleSize(particle, defaultRatio, divBubble);\n this._hoverBubbleOpacity(particle, defaultRatio, divBubble);\n this._hoverBubbleColor(particle, defaultRatio, divBubble);\n }\n });\n };\n}\n\n\n//# sourceURL=webpack://@tsparticles/interaction-external-bubble/./dist/browser/Bubbler.js?\n}");
|
|
27
27
|
|
|
28
28
|
/***/ },
|
|
29
29
|
|
|
@@ -33,7 +33,7 @@ eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpa
|
|
|
33
33
|
\*******************************/
|
|
34
34
|
(__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) {
|
|
35
35
|
|
|
36
|
-
eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ ProcessBubbleType: () => (/* binding */ ProcessBubbleType)\n/* harmony export */ });\nvar ProcessBubbleType;\n(function
|
|
36
|
+
eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ ProcessBubbleType: () => (/* binding */ ProcessBubbleType)\n/* harmony export */ });\nvar ProcessBubbleType;\n(function(ProcessBubbleType) {\n ProcessBubbleType[\"color\"] = \"color\";\n ProcessBubbleType[\"opacity\"] = \"opacity\";\n ProcessBubbleType[\"size\"] = \"size\";\n})(ProcessBubbleType || (ProcessBubbleType = {}));\n\n\n//# sourceURL=webpack://@tsparticles/interaction-external-bubble/./dist/browser/Enums.js?\n}");
|
|
37
37
|
|
|
38
38
|
/***/ },
|
|
39
39
|
|
|
@@ -43,7 +43,7 @@ eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpa
|
|
|
43
43
|
\*******************************/
|
|
44
44
|
(__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) {
|
|
45
45
|
|
|
46
|
-
eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ calculateBubbleValue: () => (/* binding */ calculateBubbleValue)\n/* harmony export */ });\n/* harmony import */ var _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @tsparticles/engine */ \"@tsparticles/engine\");\n\nfunction calculateBubbleValue(particleValue, modeValue, optionsValue, ratio) {\n
|
|
46
|
+
eval("{__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ calculateBubbleValue: () => (/* binding */ calculateBubbleValue)\n/* harmony export */ });\n/* harmony import */ var _tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @tsparticles/engine */ \"@tsparticles/engine\");\n\nfunction calculateBubbleValue(particleValue, modeValue, optionsValue, ratio) {\n if (modeValue >= optionsValue) {\n const value = particleValue + (modeValue - optionsValue) * ratio;\n return (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.clamp)(value, particleValue, modeValue);\n } else if (modeValue < optionsValue) {\n const value = particleValue - (optionsValue - modeValue) * ratio;\n return (0,_tsparticles_engine__WEBPACK_IMPORTED_MODULE_0__.clamp)(value, modeValue, particleValue);\n }\n return undefined;\n}\n\n\n//# sourceURL=webpack://@tsparticles/interaction-external-bubble/./dist/browser/Utils.js?\n}");
|
|
47
47
|
|
|
48
48
|
/***/ }
|
|
49
49
|
|