canvasengine 2.0.0-beta.6 → 2.0.0-beta.60
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/dist/DebugRenderer-DkjTAc48.js +1384 -0
- package/dist/DebugRenderer-DkjTAc48.js.map +1 -0
- package/dist/components/Button.d.ts +185 -0
- package/dist/components/Button.d.ts.map +1 -0
- package/dist/components/Canvas.d.ts +17 -0
- package/dist/components/Canvas.d.ts.map +1 -0
- package/dist/components/DOMElement.d.ts +54 -0
- package/dist/components/DOMElement.d.ts.map +1 -0
- package/dist/components/DOMSprite.d.ts +127 -0
- package/dist/components/DOMSprite.d.ts.map +1 -0
- package/dist/components/FocusContainer.d.ts +129 -0
- package/dist/components/FocusContainer.d.ts.map +1 -0
- package/dist/components/Graphic.d.ts +64 -0
- package/dist/components/Graphic.d.ts.map +1 -0
- package/dist/components/Joystick.d.ts +36 -0
- package/dist/components/Joystick.d.ts.map +1 -0
- package/dist/components/NineSliceSprite.d.ts +16 -0
- package/dist/components/NineSliceSprite.d.ts.map +1 -0
- package/dist/components/ParticleEmitter.d.ts +4 -0
- package/dist/components/ParticleEmitter.d.ts.map +1 -0
- package/dist/components/Scene.d.ts +2 -0
- package/dist/components/Scene.d.ts.map +1 -0
- package/dist/components/Text.d.ts +25 -0
- package/dist/components/Text.d.ts.map +1 -0
- package/dist/components/TilingSprite.d.ts +17 -0
- package/dist/components/TilingSprite.d.ts.map +1 -0
- package/dist/components/Video.d.ts +14 -0
- package/dist/components/Video.d.ts.map +1 -0
- package/dist/components/index.d.ts +20 -0
- package/dist/components/index.d.ts.map +1 -0
- package/dist/components/types/DisplayObject.d.ts +118 -0
- package/dist/components/types/DisplayObject.d.ts.map +1 -0
- package/dist/components/types/MouseEvent.d.ts +4 -0
- package/dist/components/types/MouseEvent.d.ts.map +1 -0
- package/dist/components/types/Spritesheet.d.ts +248 -0
- package/dist/components/types/Spritesheet.d.ts.map +1 -0
- package/dist/components/types/index.d.ts +4 -0
- package/dist/components/types/index.d.ts.map +1 -0
- package/dist/directives/Controls.d.ts +112 -0
- package/dist/directives/Controls.d.ts.map +1 -0
- package/dist/directives/ControlsBase.d.ts +199 -0
- package/dist/directives/ControlsBase.d.ts.map +1 -0
- package/dist/directives/Drag.d.ts +69 -0
- package/dist/directives/Drag.d.ts.map +1 -0
- package/dist/directives/Flash.d.ts +116 -0
- package/dist/directives/Flash.d.ts.map +1 -0
- package/dist/directives/FocusNavigation.d.ts +52 -0
- package/dist/directives/FocusNavigation.d.ts.map +1 -0
- package/dist/directives/FogVisibility.d.ts +47 -0
- package/dist/directives/FogVisibility.d.ts.map +1 -0
- package/dist/directives/GamepadControls.d.ts +224 -0
- package/dist/directives/GamepadControls.d.ts.map +1 -0
- package/dist/directives/JoystickControls.d.ts +171 -0
- package/dist/directives/JoystickControls.d.ts.map +1 -0
- package/dist/directives/KeyboardControls.d.ts +219 -0
- package/dist/directives/KeyboardControls.d.ts.map +1 -0
- package/dist/directives/Scheduler.d.ts +35 -0
- package/dist/directives/Scheduler.d.ts.map +1 -0
- package/dist/directives/Shake.d.ts +98 -0
- package/dist/directives/Shake.d.ts.map +1 -0
- package/dist/directives/Sound.d.ts +25 -0
- package/dist/directives/Sound.d.ts.map +1 -0
- package/dist/directives/Transition.d.ts +10 -0
- package/dist/directives/Transition.d.ts.map +1 -0
- package/dist/directives/ViewportCull.d.ts +11 -0
- package/dist/directives/ViewportCull.d.ts.map +1 -0
- package/dist/directives/ViewportFollow.d.ts +18 -0
- package/dist/directives/ViewportFollow.d.ts.map +1 -0
- package/dist/directives/index.d.ts +14 -0
- package/dist/directives/index.d.ts.map +1 -0
- package/dist/dist-BOOc43Qm.js +778 -0
- package/dist/dist-BOOc43Qm.js.map +1 -0
- package/dist/engine/FocusManager.d.ts +174 -0
- package/dist/engine/FocusManager.d.ts.map +1 -0
- package/dist/engine/animation.d.ts +72 -0
- package/dist/engine/animation.d.ts.map +1 -0
- package/dist/engine/bootstrap.d.ts +48 -0
- package/dist/engine/bootstrap.d.ts.map +1 -0
- package/dist/engine/directive.d.ts +13 -0
- package/dist/engine/directive.d.ts.map +1 -0
- package/dist/engine/reactive.d.ts +134 -0
- package/dist/engine/reactive.d.ts.map +1 -0
- package/dist/engine/signal.d.ts +71 -0
- package/dist/engine/signal.d.ts.map +1 -0
- package/dist/engine/trigger.d.ts +54 -0
- package/dist/engine/trigger.d.ts.map +1 -0
- package/dist/engine/utils.d.ts +89 -0
- package/dist/engine/utils.d.ts.map +1 -0
- package/dist/hooks/addContext.d.ts +2 -0
- package/dist/hooks/addContext.d.ts.map +1 -0
- package/dist/hooks/useFocus.d.ts +60 -0
- package/dist/hooks/useFocus.d.ts.map +1 -0
- package/dist/hooks/useProps.d.ts +42 -0
- package/dist/hooks/useProps.d.ts.map +1 -0
- package/dist/hooks/useRef.d.ts +4 -0
- package/dist/hooks/useRef.d.ts.map +1 -0
- package/dist/index.d.ts +19 -1107
- package/dist/index.d.ts.map +1 -0
- package/dist/index.global.js +5 -0
- package/dist/index.global.js.map +1 -0
- package/dist/index.js +9768 -3135
- package/dist/index.js.map +1 -1
- package/dist/utils/Ease.d.ts +17 -0
- package/dist/utils/Ease.d.ts.map +1 -0
- package/dist/utils/GlobalAssetLoader.d.ts +141 -0
- package/dist/utils/GlobalAssetLoader.d.ts.map +1 -0
- package/dist/utils/RadialGradient.d.ts +57 -0
- package/dist/utils/RadialGradient.d.ts.map +1 -0
- package/dist/utils/functions.d.ts +2 -0
- package/dist/utils/functions.d.ts.map +1 -0
- package/dist/utils/tabindex.d.ts +16 -0
- package/dist/utils/tabindex.d.ts.map +1 -0
- package/package.json +15 -9
- package/src/components/Button.ts +399 -0
- package/src/components/Canvas.ts +62 -46
- package/src/components/Container.ts +21 -2
- package/src/components/DOMContainer.ts +379 -0
- package/src/components/DOMElement.ts +556 -0
- package/src/components/DOMSprite.ts +1040 -0
- package/src/components/DisplayObject.ts +419 -201
- package/src/components/FocusContainer.ts +368 -0
- package/src/components/Graphic.ts +227 -66
- package/src/components/Joystick.ts +363 -0
- package/src/components/Mesh.ts +222 -0
- package/src/components/NineSliceSprite.ts +4 -1
- package/src/components/ParticleEmitter.ts +12 -8
- package/src/components/Sprite.ts +418 -52
- package/src/components/Text.ts +125 -18
- package/src/components/Viewport.ts +122 -63
- package/src/components/index.ts +9 -2
- package/src/components/types/DisplayObject.ts +53 -5
- package/src/components/types/Spritesheet.ts +0 -118
- package/src/directives/Controls.ts +254 -0
- package/src/directives/ControlsBase.ts +267 -0
- package/src/directives/Drag.ts +357 -52
- package/src/directives/Flash.ts +419 -0
- package/src/directives/FocusNavigation.ts +113 -0
- package/src/directives/FogVisibility.ts +273 -0
- package/src/directives/GamepadControls.ts +537 -0
- package/src/directives/JoystickControls.ts +396 -0
- package/src/directives/KeyboardControls.ts +85 -430
- package/src/directives/Scheduler.ts +12 -4
- package/src/directives/Shake.ts +298 -0
- package/src/directives/Sound.ts +94 -31
- package/src/directives/ViewportFollow.ts +40 -9
- package/src/directives/index.ts +13 -6
- package/src/engine/FocusManager.ts +510 -0
- package/src/engine/animation.ts +175 -21
- package/src/engine/bootstrap.ts +93 -3
- package/src/engine/directive.ts +4 -4
- package/src/engine/reactive.ts +979 -176
- package/src/engine/signal.ts +113 -25
- package/src/engine/trigger.ts +34 -7
- package/src/engine/utils.ts +19 -3
- package/src/hooks/useFocus.ts +91 -0
- package/src/hooks/useProps.ts +1 -1
- package/src/index.ts +8 -2
- package/src/types/pixi-cull.d.ts +7 -0
- package/src/utils/GlobalAssetLoader.ts +257 -0
- package/src/utils/functions.ts +7 -0
- package/src/utils/tabindex.ts +70 -0
- package/testing/index.ts +35 -4
- package/tsconfig.json +18 -0
- package/vite.config.ts +39 -0
package/src/components/Text.ts
CHANGED
|
@@ -1,15 +1,16 @@
|
|
|
1
1
|
import { Text as PixiText, TextStyle } from "pixi.js";
|
|
2
|
-
import { createComponent, registerComponent } from "../engine/reactive";
|
|
3
|
-
import { DisplayObject } from "./DisplayObject";
|
|
2
|
+
import { createComponent, registerComponent, Element, Props } from "../engine/reactive";
|
|
3
|
+
import { DisplayObject, ComponentInstance } from "./DisplayObject";
|
|
4
4
|
import { DisplayObjectProps } from "./types/DisplayObject";
|
|
5
5
|
import { Signal } from "@signe/reactive";
|
|
6
|
-
import { on } from "../engine/trigger";
|
|
6
|
+
import { on, isTrigger } from "../engine/trigger";
|
|
7
|
+
import { Howl } from "howler";
|
|
7
8
|
|
|
8
9
|
enum TextEffect {
|
|
9
10
|
Typewriter = "typewriter",
|
|
10
11
|
}
|
|
11
12
|
|
|
12
|
-
interface TextProps extends DisplayObjectProps {
|
|
13
|
+
export interface TextProps extends DisplayObjectProps {
|
|
13
14
|
text?: string;
|
|
14
15
|
style?: Partial<TextStyle>;
|
|
15
16
|
color?: string;
|
|
@@ -20,7 +21,13 @@ interface TextProps extends DisplayObjectProps {
|
|
|
20
21
|
start?: () => void;
|
|
21
22
|
onComplete?: () => void;
|
|
22
23
|
skip?: () => void;
|
|
24
|
+
sound?: {
|
|
25
|
+
src: string;
|
|
26
|
+
volume?: number;
|
|
27
|
+
rate?: number;
|
|
28
|
+
};
|
|
23
29
|
};
|
|
30
|
+
context?: any; // Ensure context is available, ideally typed from a base prop or injected
|
|
24
31
|
}
|
|
25
32
|
|
|
26
33
|
class CanvasText extends DisplayObject(PixiText) {
|
|
@@ -31,10 +38,19 @@ class CanvasText extends DisplayObject(PixiText) {
|
|
|
31
38
|
private _wordWrapWidth: number = 0;
|
|
32
39
|
private typewriterOptions: any = {};
|
|
33
40
|
private skipSignal?: () => void;
|
|
41
|
+
private typewriterSound?: Howl;
|
|
42
|
+
private lastSoundTime: number = 0;
|
|
43
|
+
private soundDuration: number = 0; // Duration of the sound in milliseconds
|
|
34
44
|
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
45
|
+
/**
|
|
46
|
+
* Called when the component is mounted to the scene graph.
|
|
47
|
+
* Initializes the typewriter effect if configured.
|
|
48
|
+
* @param {Element<CanvasText>} element - The element being mounted with parent and props.
|
|
49
|
+
* @param {number} [index] - The index of the component among its siblings.
|
|
50
|
+
*/
|
|
51
|
+
async onMount(element: Element<any>, index?: number): Promise<void> {
|
|
52
|
+
const { props } = element;
|
|
53
|
+
await super.onMount(element, index);
|
|
38
54
|
const tick: Signal = props.context.tick;
|
|
39
55
|
|
|
40
56
|
if (props.text && props.typewriter) {
|
|
@@ -44,12 +60,18 @@ class CanvasText extends DisplayObject(PixiText) {
|
|
|
44
60
|
// Set typewriter options
|
|
45
61
|
if (props.typewriter) {
|
|
46
62
|
this.typewriterOptions = props.typewriter;
|
|
47
|
-
if (this.typewriterOptions.skip) {
|
|
63
|
+
if (this.typewriterOptions.skip && isTrigger(this.typewriterOptions.skip)) {
|
|
48
64
|
on(this.typewriterOptions.skip, () => {
|
|
49
65
|
this.skipTypewriter();
|
|
50
66
|
});
|
|
51
67
|
}
|
|
68
|
+
// Initialize typewriter sound if configured
|
|
69
|
+
if (this.typewriterOptions.sound) {
|
|
70
|
+
this.initializeTypewriterSound();
|
|
71
|
+
}
|
|
52
72
|
}
|
|
73
|
+
// Update layout after initializing typewriter
|
|
74
|
+
this.updateLayout();
|
|
53
75
|
}
|
|
54
76
|
this.subscriptionTick = tick.observable.subscribe(() => {
|
|
55
77
|
if (props.typewriter) {
|
|
@@ -63,15 +85,21 @@ class CanvasText extends DisplayObject(PixiText) {
|
|
|
63
85
|
if (props.typewriter) {
|
|
64
86
|
if (props.typewriter) {
|
|
65
87
|
this.typewriterOptions = props.typewriter;
|
|
88
|
+
// Reinitialize sound if sound configuration changed
|
|
89
|
+
if (props.typewriter.sound) {
|
|
90
|
+
this.initializeTypewriterSound();
|
|
91
|
+
}
|
|
66
92
|
}
|
|
67
93
|
}
|
|
68
|
-
if (props.text) {
|
|
69
|
-
this.text = props.text;
|
|
94
|
+
if (props.text !== undefined) {
|
|
95
|
+
this.text = ''+props.text;
|
|
70
96
|
}
|
|
71
97
|
if (props.text !== undefined && props.text !== this.fullText && this.fullProps.typewriter) {
|
|
72
98
|
this.text = "";
|
|
73
99
|
this.currentIndex = 0;
|
|
74
100
|
this.fullText = props.text;
|
|
101
|
+
// Update layout after resetting typewriter
|
|
102
|
+
this.updateLayout();
|
|
75
103
|
}
|
|
76
104
|
if (props.style) {
|
|
77
105
|
for (const key in props.style) {
|
|
@@ -90,6 +118,59 @@ class CanvasText extends DisplayObject(PixiText) {
|
|
|
90
118
|
if (props.fontFamily) {
|
|
91
119
|
this.style.fontFamily = props.fontFamily;
|
|
92
120
|
}
|
|
121
|
+
|
|
122
|
+
// Use the centralized layout update method
|
|
123
|
+
this.updateLayout();
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
get onCompleteCallback() {
|
|
127
|
+
return this.typewriterOptions.onComplete;
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
/**
|
|
131
|
+
* Initializes the typewriter sound effect using Howler.
|
|
132
|
+
* Creates a Howl instance with the configured sound settings.
|
|
133
|
+
* Calculates the sound duration to prevent overlapping sounds.
|
|
134
|
+
*/
|
|
135
|
+
private initializeTypewriterSound() {
|
|
136
|
+
if (!this.typewriterOptions.sound?.src) return;
|
|
137
|
+
|
|
138
|
+
this.typewriterSound = new Howl({
|
|
139
|
+
src: [this.typewriterOptions.sound.src],
|
|
140
|
+
volume: this.typewriterOptions.sound.volume ?? 0.5,
|
|
141
|
+
rate: this.typewriterOptions.sound.rate ?? 1.0,
|
|
142
|
+
preload: true,
|
|
143
|
+
onload: () => {
|
|
144
|
+
// Calculate sound duration in milliseconds
|
|
145
|
+
if (this.typewriterSound) {
|
|
146
|
+
const duration = this.typewriterSound.duration();
|
|
147
|
+
const rate = this.typewriterOptions.sound?.rate ?? 1.0;
|
|
148
|
+
this.soundDuration = (duration / rate) * 1000;
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
/**
|
|
155
|
+
* Plays the typewriter sound with duration-based cooldown to prevent overlapping sounds.
|
|
156
|
+
* @param {number} currentTime - The current timestamp to check against sound duration.
|
|
157
|
+
*/
|
|
158
|
+
private playTypewriterSound(currentTime: number) {
|
|
159
|
+
if (!this.typewriterSound || !this.typewriterOptions.sound) return;
|
|
160
|
+
|
|
161
|
+
// Check if enough time has passed since the last sound play
|
|
162
|
+
// Use the actual sound duration to prevent overlap
|
|
163
|
+
if (this.soundDuration > 0 && currentTime - this.lastSoundTime < this.soundDuration) return;
|
|
164
|
+
|
|
165
|
+
this.typewriterSound.play();
|
|
166
|
+
this.lastSoundTime = currentTime;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
/**
|
|
170
|
+
* Updates the layout properties of the text component.
|
|
171
|
+
* This method ensures consistent width, height and word wrap behavior.
|
|
172
|
+
*/
|
|
173
|
+
private updateLayout() {
|
|
93
174
|
if (this._wordWrapWidth) {
|
|
94
175
|
this.setWidth(this._wordWrapWidth);
|
|
95
176
|
} else {
|
|
@@ -98,10 +179,6 @@ class CanvasText extends DisplayObject(PixiText) {
|
|
|
98
179
|
this.setHeight(this.height);
|
|
99
180
|
}
|
|
100
181
|
|
|
101
|
-
get onCompleteCallback() {
|
|
102
|
-
return this.typewriterOptions.onComplete;
|
|
103
|
-
}
|
|
104
|
-
|
|
105
182
|
private typewriterEffect() {
|
|
106
183
|
if (this.currentIndex < this.fullText.length) {
|
|
107
184
|
const nextIndex = Math.min(
|
|
@@ -111,6 +188,14 @@ class CanvasText extends DisplayObject(PixiText) {
|
|
|
111
188
|
this.text = this.fullText.slice(0, nextIndex);
|
|
112
189
|
this.currentIndex = nextIndex;
|
|
113
190
|
|
|
191
|
+
// Play typewriter sound if configured
|
|
192
|
+
if (this.typewriterOptions.sound) {
|
|
193
|
+
this.playTypewriterSound(Date.now());
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
// Update layout after text change to maintain proper word wrap and dimensions
|
|
197
|
+
this.updateLayout();
|
|
198
|
+
|
|
114
199
|
// Check if typewriter effect is complete
|
|
115
200
|
if (
|
|
116
201
|
this.currentIndex === this.fullText.length &&
|
|
@@ -128,15 +213,37 @@ class CanvasText extends DisplayObject(PixiText) {
|
|
|
128
213
|
}
|
|
129
214
|
this.text = this.fullText;
|
|
130
215
|
this.currentIndex = this.fullText.length;
|
|
216
|
+
|
|
217
|
+
// Update layout after setting full text to maintain proper word wrap and dimensions
|
|
218
|
+
this.updateLayout();
|
|
131
219
|
}
|
|
132
220
|
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
221
|
+
/**
|
|
222
|
+
* Called when the component is about to be destroyed.
|
|
223
|
+
* Unsubscribes from the tick observable and cleans up sound resources.
|
|
224
|
+
* @param {Element<any>} parent - The parent element.
|
|
225
|
+
* @param {() => void} [afterDestroy] - An optional callback function to be executed after the component's own destruction logic.
|
|
226
|
+
*/
|
|
227
|
+
async onDestroy(parent: Element<any>, afterDestroy?: () => void): Promise<void> {
|
|
228
|
+
const _afterDestroy = async () => {
|
|
229
|
+
if (this.subscriptionTick) {
|
|
230
|
+
this.subscriptionTick.unsubscribe();
|
|
231
|
+
}
|
|
232
|
+
// Clean up typewriter sound
|
|
233
|
+
if (this.typewriterSound) {
|
|
234
|
+
this.typewriterSound.stop();
|
|
235
|
+
this.typewriterSound.unload();
|
|
236
|
+
this.typewriterSound = undefined;
|
|
237
|
+
}
|
|
238
|
+
if (afterDestroy) {
|
|
239
|
+
afterDestroy();
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
await super.onDestroy(parent, _afterDestroy);
|
|
136
243
|
}
|
|
137
244
|
}
|
|
138
245
|
|
|
139
|
-
interface CanvasText extends PixiText {}
|
|
246
|
+
// interface CanvasText extends PixiText {} // Removed as it's redundant and causes type conflicts
|
|
140
247
|
|
|
141
248
|
registerComponent("Text", CanvasText);
|
|
142
249
|
|
|
@@ -1,8 +1,9 @@
|
|
|
1
1
|
import { Viewport as PixiViewport } from 'pixi-viewport';
|
|
2
2
|
import { Subscription } from 'rxjs';
|
|
3
|
-
import { createComponent, registerComponent } from '../engine/reactive';
|
|
4
|
-
import { DisplayObject } from './DisplayObject';
|
|
5
|
-
import { effect } from '@signe/reactive';
|
|
3
|
+
import { createComponent, registerComponent, Element, Props } from '../engine/reactive';
|
|
4
|
+
import { DisplayObject, ComponentInstance } from './DisplayObject';
|
|
5
|
+
import { effect, Signal } from '@signe/reactive';
|
|
6
|
+
import { Graphics, Container, ContainerChild, IRenderLayer } from 'pixi.js';
|
|
6
7
|
|
|
7
8
|
const EVENTS = [
|
|
8
9
|
'bounce-x-end',
|
|
@@ -28,55 +29,103 @@ const EVENTS = [
|
|
|
28
29
|
'zoomed-end'
|
|
29
30
|
]
|
|
30
31
|
|
|
31
|
-
export
|
|
32
|
+
export interface ViewportProps extends Props {
|
|
33
|
+
screenWidth?: number;
|
|
34
|
+
screenHeight?: number;
|
|
35
|
+
worldWidth?: number;
|
|
36
|
+
worldHeight?: number;
|
|
37
|
+
sortableChildren?: boolean;
|
|
38
|
+
clamp?: boolean | {
|
|
39
|
+
left?: number;
|
|
40
|
+
right?: number;
|
|
41
|
+
top?: number;
|
|
42
|
+
bottom?: number;
|
|
43
|
+
};
|
|
44
|
+
context?: any;
|
|
45
|
+
[key: string]: any;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
export class CanvasViewport extends DisplayObject(Container) {
|
|
32
49
|
private tickSubscription: Subscription
|
|
33
50
|
overrideProps = ['wheel']
|
|
51
|
+
#mask: Graphics
|
|
52
|
+
public viewport: PixiViewport
|
|
34
53
|
|
|
35
54
|
constructor() {
|
|
55
|
+
super()
|
|
36
56
|
const defaultOptions = {
|
|
37
57
|
noTicker: true,
|
|
38
58
|
events: {
|
|
39
59
|
domElement: {
|
|
40
|
-
addEventListener: () => {}
|
|
60
|
+
addEventListener: () => { }
|
|
41
61
|
}
|
|
42
62
|
},
|
|
43
63
|
}
|
|
44
64
|
// @ts-ignore
|
|
45
|
-
|
|
65
|
+
this.viewport = new PixiViewport(defaultOptions)
|
|
66
|
+
super.addChild(this.viewport)
|
|
67
|
+
|
|
68
|
+
this.#mask = new Graphics()
|
|
69
|
+
super.addChild(this.#mask)
|
|
70
|
+
this.mask = this.#mask
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
addChild<U extends any[]>(...children: U): U[0] {
|
|
74
|
+
return this.viewport.addChild(...children)
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
addChildAt<T extends ContainerChild | IRenderLayer>(child: T, index: number): T {
|
|
78
|
+
return this.viewport.addChildAt(child, index) as T
|
|
46
79
|
}
|
|
47
80
|
|
|
48
81
|
onInit(props) {
|
|
49
82
|
super.onInit(props)
|
|
50
83
|
for (let event of EVENTS) {
|
|
51
|
-
|
|
52
|
-
if (props[camelCaseEvent]) {
|
|
53
|
-
this.on(event, props[camelCaseEvent])
|
|
54
|
-
}
|
|
84
|
+
if (props[event]) this.viewport.on(event, props[event])
|
|
55
85
|
}
|
|
56
86
|
}
|
|
57
87
|
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
88
|
+
/**
|
|
89
|
+
* Called when the component is mounted to the scene graph.
|
|
90
|
+
* Initializes viewport settings and subscriptions.
|
|
91
|
+
* @param {Element<CanvasViewport>} element - The element being mounted. Its `props` property (of type ViewportProps) contains component properties and context.
|
|
92
|
+
* @param {number} [index] - The index of the component among its siblings.
|
|
93
|
+
*/
|
|
94
|
+
async onMount(element: Element<any>, index?: number): Promise<void> {
|
|
95
|
+
element.props.context.viewport = this.viewport
|
|
96
|
+
await super.onMount(element, index);
|
|
97
|
+
const { props } = element;
|
|
98
|
+
const { tick, app, canvasSize } = props.context;
|
|
99
|
+
|
|
63
100
|
effect(() => {
|
|
64
|
-
|
|
65
|
-
|
|
101
|
+
if (props.screenWidth === undefined) {
|
|
102
|
+
this.viewport.screenWidth = canvasSize().width
|
|
103
|
+
}
|
|
104
|
+
if (props.screenHeight === undefined) {
|
|
105
|
+
this.viewport.screenHeight = canvasSize().height
|
|
106
|
+
}
|
|
107
|
+
this.updateMask()
|
|
108
|
+
})
|
|
109
|
+
|
|
110
|
+
effect(() => {
|
|
111
|
+
const _app = app()
|
|
112
|
+
if (!_app) return
|
|
113
|
+
|
|
114
|
+
const renderer = _app.renderer
|
|
115
|
+
|
|
116
|
+
renderer.events.domElement.addEventListener(
|
|
117
|
+
'wheel',
|
|
118
|
+
this.viewport.input.wheelFunction
|
|
119
|
+
);
|
|
120
|
+
|
|
121
|
+
this.viewport.options.events = renderer.events
|
|
66
122
|
})
|
|
67
123
|
|
|
68
|
-
renderer.events.domElement.addEventListener(
|
|
69
|
-
'wheel',
|
|
70
|
-
this.input.wheelFunction
|
|
71
|
-
);
|
|
72
|
-
this.options.events = renderer.events
|
|
73
|
-
|
|
74
124
|
this.tickSubscription = tick.observable.subscribe(({ value }) => {
|
|
75
|
-
this.update(value.
|
|
125
|
+
this.viewport.update(value.deltaTime)
|
|
76
126
|
})
|
|
77
127
|
|
|
78
|
-
|
|
79
|
-
this.updateViewportSettings(element.props)
|
|
128
|
+
this.updateViewportSettings(props)
|
|
80
129
|
}
|
|
81
130
|
|
|
82
131
|
onUpdate(props) {
|
|
@@ -86,74 +135,84 @@ export class CanvasViewport extends DisplayObject(PixiViewport) {
|
|
|
86
135
|
|
|
87
136
|
private updateViewportSettings(props) {
|
|
88
137
|
if (props.screenWidth !== undefined) {
|
|
89
|
-
this.screenWidth = props.screenWidth
|
|
138
|
+
this.viewport.screenWidth = props.screenWidth
|
|
90
139
|
}
|
|
91
140
|
if (props.screenHeight !== undefined) {
|
|
92
|
-
this.screenHeight = props.screenHeight
|
|
141
|
+
this.viewport.screenHeight = props.screenHeight
|
|
93
142
|
}
|
|
143
|
+
this.updateMask()
|
|
94
144
|
if (props.worldWidth !== undefined) {
|
|
95
|
-
this.worldWidth = props.worldWidth
|
|
145
|
+
this.viewport.worldWidth = props.worldWidth
|
|
96
146
|
}
|
|
97
147
|
if (props.worldHeight !== undefined) {
|
|
98
|
-
this.worldHeight = props.worldHeight
|
|
148
|
+
this.viewport.worldHeight = props.worldHeight
|
|
149
|
+
}
|
|
150
|
+
if (props.sortableChildren !== undefined) {
|
|
151
|
+
this.viewport.sortableChildren = props.sortableChildren
|
|
152
|
+
}
|
|
153
|
+
if (props.drag) {
|
|
154
|
+
this.viewport.drag(props.drag)
|
|
99
155
|
}
|
|
100
|
-
// if (props.drag) {
|
|
101
|
-
// if (props.drag === true) {
|
|
102
|
-
|
|
103
|
-
// } else {
|
|
104
|
-
// this.drag(props.drag)
|
|
105
|
-
// }
|
|
106
|
-
// }
|
|
107
156
|
if (props.clamp) {
|
|
108
|
-
this.clamp(props.clamp)
|
|
157
|
+
this.viewport.clamp(props.clamp.value ?? props.clamp)
|
|
109
158
|
}
|
|
110
159
|
if (props.wheel) {
|
|
111
160
|
if (props.wheel === true) {
|
|
112
|
-
this.wheel()
|
|
161
|
+
this.viewport.wheel()
|
|
113
162
|
} else {
|
|
114
|
-
this.wheel(props.wheel)
|
|
163
|
+
this.viewport.wheel(props.wheel)
|
|
115
164
|
}
|
|
116
165
|
}
|
|
117
166
|
if (props.decelerate) {
|
|
118
167
|
if (props.decelerate === true) {
|
|
119
|
-
this.decelerate()
|
|
168
|
+
this.viewport.decelerate()
|
|
120
169
|
} else {
|
|
121
|
-
this.decelerate(props.decelerate)
|
|
170
|
+
this.viewport.decelerate(props.decelerate)
|
|
122
171
|
}
|
|
123
172
|
}
|
|
124
173
|
if (props.pinch) {
|
|
125
174
|
if (props.pinch === true) {
|
|
126
|
-
this.pinch()
|
|
175
|
+
this.viewport.pinch()
|
|
127
176
|
} else {
|
|
128
|
-
this.pinch(props.pinch)
|
|
177
|
+
this.viewport.pinch(props.pinch)
|
|
129
178
|
}
|
|
130
179
|
}
|
|
131
180
|
}
|
|
132
181
|
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
this.
|
|
182
|
+
private updateMask() {
|
|
183
|
+
if (!this.#mask) return
|
|
184
|
+
this.#mask.clear()
|
|
185
|
+
this.#mask.beginFill(0xffffff)
|
|
186
|
+
this.#mask.drawRect(0, 0, this.viewport.screenWidth, this.viewport.screenHeight)
|
|
187
|
+
this.#mask.endFill()
|
|
136
188
|
}
|
|
137
|
-
}
|
|
138
189
|
|
|
139
|
-
|
|
190
|
+
/**
|
|
191
|
+
* Called when the component is about to be destroyed.
|
|
192
|
+
* Unsubscribes from the tick observable.
|
|
193
|
+
* @param {Element<any>} parent - The parent element.
|
|
194
|
+
* @param {() => void} [afterDestroy] - An optional callback function to be executed after the component's own destruction logic.
|
|
195
|
+
*/
|
|
196
|
+
async onDestroy(parent: Element<any>, afterDestroy?: () => void): Promise<void> {
|
|
197
|
+
const _afterDestroy = async () => {
|
|
198
|
+
this.tickSubscription.unsubscribe()
|
|
199
|
+
afterDestroy()
|
|
200
|
+
}
|
|
201
|
+
await super.onDestroy(parent, _afterDestroy);
|
|
202
|
+
}
|
|
140
203
|
|
|
141
|
-
|
|
204
|
+
// Proxy methods for viewport plugins
|
|
205
|
+
follow(...args: any[]) {
|
|
206
|
+
return (this.viewport.follow as any)(...args)
|
|
207
|
+
}
|
|
142
208
|
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
worldWidth?: number;
|
|
147
|
-
worldHeight?: number;
|
|
148
|
-
clamp?: boolean | {
|
|
149
|
-
left?: number;
|
|
150
|
-
right?: number;
|
|
151
|
-
top?: number;
|
|
152
|
-
bottom?: number;
|
|
153
|
-
};
|
|
154
|
-
[key: string]: any;
|
|
209
|
+
get plugins() {
|
|
210
|
+
return this.viewport.plugins
|
|
211
|
+
}
|
|
155
212
|
}
|
|
156
213
|
|
|
214
|
+
registerComponent('Viewport', CanvasViewport)
|
|
215
|
+
|
|
157
216
|
export function Viewport(props: ViewportProps) {
|
|
158
217
|
return createComponent('Viewport', props);
|
|
159
|
-
}
|
|
218
|
+
}
|
package/src/components/index.ts
CHANGED
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
export { Canvas } from './Canvas'
|
|
2
2
|
export { Container } from './Container'
|
|
3
|
-
export { Graphics, Rect, Circle, Ellipse, Triangle, Svg
|
|
3
|
+
export { Graphics, Rect, Circle, Ellipse, Triangle, Svg } from './Graphic'
|
|
4
|
+
export { Mesh } from './Mesh'
|
|
4
5
|
export { Scene } from './Scene'
|
|
5
6
|
export { ParticlesEmitter } from './ParticleEmitter'
|
|
6
7
|
export { Sprite } from './Sprite'
|
|
@@ -9,4 +10,10 @@ export { Text } from './Text'
|
|
|
9
10
|
export { TilingSprite } from './TilingSprite'
|
|
10
11
|
export { Viewport } from './Viewport'
|
|
11
12
|
export { NineSliceSprite } from './NineSliceSprite'
|
|
12
|
-
export { type ComponentInstance } from './DisplayObject'
|
|
13
|
+
export { type ComponentInstance } from './DisplayObject'
|
|
14
|
+
export { DOMContainer } from './DOMContainer'
|
|
15
|
+
export { DOMElement } from './DOMElement'
|
|
16
|
+
export { DOMSprite } from './DOMSprite'
|
|
17
|
+
export { Button, ButtonState, type ButtonProps, type ButtonStyle } from './Button'
|
|
18
|
+
export { Joystick, type JoystickSettings } from './Joystick'
|
|
19
|
+
export { FocusContainer, Navigation, type FocusContainerProps } from './FocusContainer'
|
|
@@ -1,11 +1,22 @@
|
|
|
1
1
|
import * as PIXI from "pixi.js";
|
|
2
2
|
import { SignalOrPrimitive } from ".";
|
|
3
|
+
import { DragProps } from "../../directives/Drag";
|
|
4
|
+
import { ViewportFollowProps } from "../../directives/ViewportFollow";
|
|
5
|
+
import { ShakeProps } from "../../directives/Shake";
|
|
6
|
+
import { FlashProps } from "../../directives/Flash";
|
|
7
|
+
import { FogVisibilityProps } from "../../directives/FogVisibility";
|
|
3
8
|
|
|
4
9
|
export type FlexDirection = 'row' | 'column' | 'row-reverse' | 'column-reverse';
|
|
5
10
|
export type JustifyContent = 'flex-start' | 'flex-end' | 'center' | 'space-between' | 'space-around';
|
|
6
11
|
export type AlignContent = 'flex-start' | 'flex-end' | 'center' | 'space-between' | 'space-around';
|
|
7
12
|
export type Size = number | `${number}%`
|
|
8
13
|
export type EdgeSize = SignalOrPrimitive<Size | [Size, Size] | [Size, Size, Size, Size]>
|
|
14
|
+
export type ObjectFit = 'contain' | 'cover' | 'fill' | 'none' | 'scale-down';
|
|
15
|
+
export type ObjectPosition = string;
|
|
16
|
+
export type TransformOrigin = string;
|
|
17
|
+
export type PositionType = 'relative' | 'absolute' | 'static';
|
|
18
|
+
|
|
19
|
+
export type ObservablePointSignal = [number, number] | SignalOrPrimitive<[number, number]> | { x: number, y: number } | SignalOrPrimitive<{ x: number, y: number }>;
|
|
9
20
|
|
|
10
21
|
export interface DisplayObjectProps {
|
|
11
22
|
attach?: any;
|
|
@@ -14,6 +25,24 @@ export interface DisplayObjectProps {
|
|
|
14
25
|
y?: SignalOrPrimitive<number>;
|
|
15
26
|
width?: SignalOrPrimitive<Size>;
|
|
16
27
|
height?: SignalOrPrimitive<Size>;
|
|
28
|
+
minWidth?: SignalOrPrimitive<Size>;
|
|
29
|
+
minHeight?: SignalOrPrimitive<Size>;
|
|
30
|
+
maxWidth?: SignalOrPrimitive<Size>;
|
|
31
|
+
maxHeight?: SignalOrPrimitive<Size>;
|
|
32
|
+
aspectRatio?: SignalOrPrimitive<number>;
|
|
33
|
+
flexGrow?: SignalOrPrimitive<number>;
|
|
34
|
+
flexShrink?: SignalOrPrimitive<number>;
|
|
35
|
+
flexBasis?: SignalOrPrimitive<Size>;
|
|
36
|
+
rowGap?: SignalOrPrimitive<number>;
|
|
37
|
+
columnGap?: SignalOrPrimitive<number>;
|
|
38
|
+
positionType?: PositionType;
|
|
39
|
+
top?: SignalOrPrimitive<Size>;
|
|
40
|
+
right?: SignalOrPrimitive<Size>;
|
|
41
|
+
bottom?: SignalOrPrimitive<Size>;
|
|
42
|
+
left?: SignalOrPrimitive<Size>;
|
|
43
|
+
objectFit?: ObjectFit;
|
|
44
|
+
objectPosition?: ObjectPosition;
|
|
45
|
+
transformOrigin?: TransformOrigin;
|
|
17
46
|
children?: any[];
|
|
18
47
|
flexDirection?: FlexDirection;
|
|
19
48
|
justifyContent?: JustifyContent;
|
|
@@ -22,9 +51,9 @@ export interface DisplayObjectProps {
|
|
|
22
51
|
padding?: EdgeSize;
|
|
23
52
|
border?: EdgeSize;
|
|
24
53
|
absolute?: SignalOrPrimitive<boolean>;
|
|
25
|
-
scale?:
|
|
26
|
-
anchor?:
|
|
27
|
-
skew?:
|
|
54
|
+
scale?: ObservablePointSignal | number;
|
|
55
|
+
anchor?: ObservablePointSignal;
|
|
56
|
+
skew?: ObservablePointSignal;
|
|
28
57
|
tint?: SignalOrPrimitive<number>;
|
|
29
58
|
rotation?: SignalOrPrimitive<number>;
|
|
30
59
|
angle?: SignalOrPrimitive<number>;
|
|
@@ -32,11 +61,29 @@ export interface DisplayObjectProps {
|
|
|
32
61
|
roundPixels?: SignalOrPrimitive<boolean>;
|
|
33
62
|
cursor?: SignalOrPrimitive<string>;
|
|
34
63
|
visible?: SignalOrPrimitive<boolean>;
|
|
35
|
-
pivot?:
|
|
64
|
+
pivot?: ObservablePointSignal;
|
|
36
65
|
filters?: any[];
|
|
37
66
|
blendMode?: SignalOrPrimitive<PIXI.BLEND_MODES>;
|
|
38
67
|
blur?: SignalOrPrimitive<number>;
|
|
68
|
+
/**
|
|
69
|
+
* Optional metadata used by presets (for example `SpriteShadows`)
|
|
70
|
+
* to mark this display object as a shadow caster.
|
|
71
|
+
*/
|
|
72
|
+
shadowCaster?: any;
|
|
73
|
+
/**
|
|
74
|
+
* Optional metadata used by presets (for example `Footprints`)
|
|
75
|
+
* to mark this display object as a footprint caster.
|
|
76
|
+
*/
|
|
77
|
+
footprintCaster?: any;
|
|
78
|
+
|
|
79
|
+
// Directives
|
|
80
|
+
drag?: DragProps;
|
|
81
|
+
viewportFollow?: ViewportFollowProps;
|
|
82
|
+
shake?: ShakeProps;
|
|
83
|
+
flash?: FlashProps;
|
|
84
|
+
fogVisibility?: FogVisibilityProps;
|
|
39
85
|
|
|
86
|
+
// Events
|
|
40
87
|
click?: PIXI.FederatedEventHandler;
|
|
41
88
|
mousedown?: PIXI.FederatedEventHandler;
|
|
42
89
|
mouseenter?: PIXI.FederatedEventHandler;
|
|
@@ -67,4 +114,5 @@ export interface DisplayObjectProps {
|
|
|
67
114
|
touchmove?: PIXI.FederatedEventHandler;
|
|
68
115
|
touchstart?: PIXI.FederatedEventHandler;
|
|
69
116
|
wheel?: PIXI.FederatedEventHandler<PIXI.FederatedWheelEvent>;
|
|
70
|
-
|
|
117
|
+
tabindex?: SignalOrPrimitive<number>;
|
|
118
|
+
}
|