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.
Files changed (164) hide show
  1. package/dist/DebugRenderer-DkjTAc48.js +1384 -0
  2. package/dist/DebugRenderer-DkjTAc48.js.map +1 -0
  3. package/dist/components/Button.d.ts +185 -0
  4. package/dist/components/Button.d.ts.map +1 -0
  5. package/dist/components/Canvas.d.ts +17 -0
  6. package/dist/components/Canvas.d.ts.map +1 -0
  7. package/dist/components/DOMElement.d.ts +54 -0
  8. package/dist/components/DOMElement.d.ts.map +1 -0
  9. package/dist/components/DOMSprite.d.ts +127 -0
  10. package/dist/components/DOMSprite.d.ts.map +1 -0
  11. package/dist/components/FocusContainer.d.ts +129 -0
  12. package/dist/components/FocusContainer.d.ts.map +1 -0
  13. package/dist/components/Graphic.d.ts +64 -0
  14. package/dist/components/Graphic.d.ts.map +1 -0
  15. package/dist/components/Joystick.d.ts +36 -0
  16. package/dist/components/Joystick.d.ts.map +1 -0
  17. package/dist/components/NineSliceSprite.d.ts +16 -0
  18. package/dist/components/NineSliceSprite.d.ts.map +1 -0
  19. package/dist/components/ParticleEmitter.d.ts +4 -0
  20. package/dist/components/ParticleEmitter.d.ts.map +1 -0
  21. package/dist/components/Scene.d.ts +2 -0
  22. package/dist/components/Scene.d.ts.map +1 -0
  23. package/dist/components/Text.d.ts +25 -0
  24. package/dist/components/Text.d.ts.map +1 -0
  25. package/dist/components/TilingSprite.d.ts +17 -0
  26. package/dist/components/TilingSprite.d.ts.map +1 -0
  27. package/dist/components/Video.d.ts +14 -0
  28. package/dist/components/Video.d.ts.map +1 -0
  29. package/dist/components/index.d.ts +20 -0
  30. package/dist/components/index.d.ts.map +1 -0
  31. package/dist/components/types/DisplayObject.d.ts +118 -0
  32. package/dist/components/types/DisplayObject.d.ts.map +1 -0
  33. package/dist/components/types/MouseEvent.d.ts +4 -0
  34. package/dist/components/types/MouseEvent.d.ts.map +1 -0
  35. package/dist/components/types/Spritesheet.d.ts +248 -0
  36. package/dist/components/types/Spritesheet.d.ts.map +1 -0
  37. package/dist/components/types/index.d.ts +4 -0
  38. package/dist/components/types/index.d.ts.map +1 -0
  39. package/dist/directives/Controls.d.ts +112 -0
  40. package/dist/directives/Controls.d.ts.map +1 -0
  41. package/dist/directives/ControlsBase.d.ts +199 -0
  42. package/dist/directives/ControlsBase.d.ts.map +1 -0
  43. package/dist/directives/Drag.d.ts +69 -0
  44. package/dist/directives/Drag.d.ts.map +1 -0
  45. package/dist/directives/Flash.d.ts +116 -0
  46. package/dist/directives/Flash.d.ts.map +1 -0
  47. package/dist/directives/FocusNavigation.d.ts +52 -0
  48. package/dist/directives/FocusNavigation.d.ts.map +1 -0
  49. package/dist/directives/FogVisibility.d.ts +47 -0
  50. package/dist/directives/FogVisibility.d.ts.map +1 -0
  51. package/dist/directives/GamepadControls.d.ts +224 -0
  52. package/dist/directives/GamepadControls.d.ts.map +1 -0
  53. package/dist/directives/JoystickControls.d.ts +171 -0
  54. package/dist/directives/JoystickControls.d.ts.map +1 -0
  55. package/dist/directives/KeyboardControls.d.ts +219 -0
  56. package/dist/directives/KeyboardControls.d.ts.map +1 -0
  57. package/dist/directives/Scheduler.d.ts +35 -0
  58. package/dist/directives/Scheduler.d.ts.map +1 -0
  59. package/dist/directives/Shake.d.ts +98 -0
  60. package/dist/directives/Shake.d.ts.map +1 -0
  61. package/dist/directives/Sound.d.ts +25 -0
  62. package/dist/directives/Sound.d.ts.map +1 -0
  63. package/dist/directives/Transition.d.ts +10 -0
  64. package/dist/directives/Transition.d.ts.map +1 -0
  65. package/dist/directives/ViewportCull.d.ts +11 -0
  66. package/dist/directives/ViewportCull.d.ts.map +1 -0
  67. package/dist/directives/ViewportFollow.d.ts +18 -0
  68. package/dist/directives/ViewportFollow.d.ts.map +1 -0
  69. package/dist/directives/index.d.ts +14 -0
  70. package/dist/directives/index.d.ts.map +1 -0
  71. package/dist/dist-BOOc43Qm.js +778 -0
  72. package/dist/dist-BOOc43Qm.js.map +1 -0
  73. package/dist/engine/FocusManager.d.ts +174 -0
  74. package/dist/engine/FocusManager.d.ts.map +1 -0
  75. package/dist/engine/animation.d.ts +72 -0
  76. package/dist/engine/animation.d.ts.map +1 -0
  77. package/dist/engine/bootstrap.d.ts +48 -0
  78. package/dist/engine/bootstrap.d.ts.map +1 -0
  79. package/dist/engine/directive.d.ts +13 -0
  80. package/dist/engine/directive.d.ts.map +1 -0
  81. package/dist/engine/reactive.d.ts +134 -0
  82. package/dist/engine/reactive.d.ts.map +1 -0
  83. package/dist/engine/signal.d.ts +71 -0
  84. package/dist/engine/signal.d.ts.map +1 -0
  85. package/dist/engine/trigger.d.ts +54 -0
  86. package/dist/engine/trigger.d.ts.map +1 -0
  87. package/dist/engine/utils.d.ts +89 -0
  88. package/dist/engine/utils.d.ts.map +1 -0
  89. package/dist/hooks/addContext.d.ts +2 -0
  90. package/dist/hooks/addContext.d.ts.map +1 -0
  91. package/dist/hooks/useFocus.d.ts +60 -0
  92. package/dist/hooks/useFocus.d.ts.map +1 -0
  93. package/dist/hooks/useProps.d.ts +42 -0
  94. package/dist/hooks/useProps.d.ts.map +1 -0
  95. package/dist/hooks/useRef.d.ts +4 -0
  96. package/dist/hooks/useRef.d.ts.map +1 -0
  97. package/dist/index.d.ts +19 -1107
  98. package/dist/index.d.ts.map +1 -0
  99. package/dist/index.global.js +5 -0
  100. package/dist/index.global.js.map +1 -0
  101. package/dist/index.js +9768 -3135
  102. package/dist/index.js.map +1 -1
  103. package/dist/utils/Ease.d.ts +17 -0
  104. package/dist/utils/Ease.d.ts.map +1 -0
  105. package/dist/utils/GlobalAssetLoader.d.ts +141 -0
  106. package/dist/utils/GlobalAssetLoader.d.ts.map +1 -0
  107. package/dist/utils/RadialGradient.d.ts +57 -0
  108. package/dist/utils/RadialGradient.d.ts.map +1 -0
  109. package/dist/utils/functions.d.ts +2 -0
  110. package/dist/utils/functions.d.ts.map +1 -0
  111. package/dist/utils/tabindex.d.ts +16 -0
  112. package/dist/utils/tabindex.d.ts.map +1 -0
  113. package/package.json +15 -9
  114. package/src/components/Button.ts +399 -0
  115. package/src/components/Canvas.ts +62 -46
  116. package/src/components/Container.ts +21 -2
  117. package/src/components/DOMContainer.ts +379 -0
  118. package/src/components/DOMElement.ts +556 -0
  119. package/src/components/DOMSprite.ts +1040 -0
  120. package/src/components/DisplayObject.ts +419 -201
  121. package/src/components/FocusContainer.ts +368 -0
  122. package/src/components/Graphic.ts +227 -66
  123. package/src/components/Joystick.ts +363 -0
  124. package/src/components/Mesh.ts +222 -0
  125. package/src/components/NineSliceSprite.ts +4 -1
  126. package/src/components/ParticleEmitter.ts +12 -8
  127. package/src/components/Sprite.ts +418 -52
  128. package/src/components/Text.ts +125 -18
  129. package/src/components/Viewport.ts +122 -63
  130. package/src/components/index.ts +9 -2
  131. package/src/components/types/DisplayObject.ts +53 -5
  132. package/src/components/types/Spritesheet.ts +0 -118
  133. package/src/directives/Controls.ts +254 -0
  134. package/src/directives/ControlsBase.ts +267 -0
  135. package/src/directives/Drag.ts +357 -52
  136. package/src/directives/Flash.ts +419 -0
  137. package/src/directives/FocusNavigation.ts +113 -0
  138. package/src/directives/FogVisibility.ts +273 -0
  139. package/src/directives/GamepadControls.ts +537 -0
  140. package/src/directives/JoystickControls.ts +396 -0
  141. package/src/directives/KeyboardControls.ts +85 -430
  142. package/src/directives/Scheduler.ts +12 -4
  143. package/src/directives/Shake.ts +298 -0
  144. package/src/directives/Sound.ts +94 -31
  145. package/src/directives/ViewportFollow.ts +40 -9
  146. package/src/directives/index.ts +13 -6
  147. package/src/engine/FocusManager.ts +510 -0
  148. package/src/engine/animation.ts +175 -21
  149. package/src/engine/bootstrap.ts +93 -3
  150. package/src/engine/directive.ts +4 -4
  151. package/src/engine/reactive.ts +979 -176
  152. package/src/engine/signal.ts +113 -25
  153. package/src/engine/trigger.ts +34 -7
  154. package/src/engine/utils.ts +19 -3
  155. package/src/hooks/useFocus.ts +91 -0
  156. package/src/hooks/useProps.ts +1 -1
  157. package/src/index.ts +8 -2
  158. package/src/types/pixi-cull.d.ts +7 -0
  159. package/src/utils/GlobalAssetLoader.ts +257 -0
  160. package/src/utils/functions.ts +7 -0
  161. package/src/utils/tabindex.ts +70 -0
  162. package/testing/index.ts +35 -4
  163. package/tsconfig.json +18 -0
  164. package/vite.config.ts +39 -0
@@ -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
- onMount(args) {
36
- super.onMount(args);
37
- const { props } = args;
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
- onDestroy(): void {
134
- super.onDestroy();
135
- this.subscriptionTick.unsubscribe();
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 class CanvasViewport extends DisplayObject(PixiViewport) {
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
- super(defaultOptions)
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
- const camelCaseEvent = event.replace(/-([a-z])/g, (g) => g[1].toUpperCase());
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
- onMount(element) {
59
- super.onMount(element)
60
- const { tick, renderer, canvasSize } = element.props.context
61
- let isDragging = false
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
- this.screenWidth = canvasSize().width
65
- this.screenHeight = canvasSize().height
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.timestamp)
125
+ this.viewport.update(value.deltaTime)
76
126
  })
77
127
 
78
- element.props.context.viewport = this
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
- onDestroy(): void {
134
- super.onDestroy()
135
- this.tickSubscription.unsubscribe()
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
- export interface CanvasViewport extends PixiViewport { }
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
- registerComponent('Viewport', CanvasViewport)
204
+ // Proxy methods for viewport plugins
205
+ follow(...args: any[]) {
206
+ return (this.viewport.follow as any)(...args)
207
+ }
142
208
 
143
- export interface ViewportProps {
144
- screenWidth?: number;
145
- screenHeight?: number;
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
+ }
@@ -1,6 +1,7 @@
1
1
  export { Canvas } from './Canvas'
2
2
  export { Container } from './Container'
3
- export { Graphics, Rect, Circle, Ellipse, Triangle, Svg as svg } from './Graphic'
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?: SignalOrPrimitive<{ x: number, y: number } | number>;
26
- anchor?: SignalOrPrimitive<{ x: number, y: number }>;
27
- skew?: SignalOrPrimitive<{ x: number, y: number }>;
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?: SignalOrPrimitive<{ x: number, y: number }>;
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
+ }