@joydle/ui 0.1.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.
Files changed (161) hide show
  1. package/dist/controls/ActionButton.d.ts +18 -0
  2. package/dist/controls/ActionButton.d.ts.map +1 -0
  3. package/dist/controls/ActionButton.js +108 -0
  4. package/dist/controls/ActionButton.js.map +1 -0
  5. package/dist/controls/ActionButtonGroup.d.ts +15 -0
  6. package/dist/controls/ActionButtonGroup.d.ts.map +1 -0
  7. package/dist/controls/ActionButtonGroup.js +55 -0
  8. package/dist/controls/ActionButtonGroup.js.map +1 -0
  9. package/dist/controls/Joystick.d.ts +9 -0
  10. package/dist/controls/Joystick.d.ts.map +1 -0
  11. package/dist/controls/Joystick.js +168 -0
  12. package/dist/controls/Joystick.js.map +1 -0
  13. package/dist/controls/SwipeZone.d.ts +10 -0
  14. package/dist/controls/SwipeZone.d.ts.map +1 -0
  15. package/dist/controls/SwipeZone.js +84 -0
  16. package/dist/controls/SwipeZone.js.map +1 -0
  17. package/dist/controls/TouchDPad.d.ts +9 -0
  18. package/dist/controls/TouchDPad.d.ts.map +1 -0
  19. package/dist/controls/TouchDPad.js +90 -0
  20. package/dist/controls/TouchDPad.js.map +1 -0
  21. package/dist/controls/index.d.ts +6 -0
  22. package/dist/controls/index.d.ts.map +1 -0
  23. package/dist/controls/index.js +9 -0
  24. package/dist/controls/index.js.map +1 -0
  25. package/dist/core/GameShell.d.ts +43 -0
  26. package/dist/core/GameShell.d.ts.map +1 -0
  27. package/dist/core/GameShell.js +91 -0
  28. package/dist/core/GameShell.js.map +1 -0
  29. package/dist/core/GameState.d.ts +35 -0
  30. package/dist/core/GameState.d.ts.map +1 -0
  31. package/dist/core/GameState.js +74 -0
  32. package/dist/core/GameState.js.map +1 -0
  33. package/dist/core/InputManager.d.ts +39 -0
  34. package/dist/core/InputManager.d.ts.map +1 -0
  35. package/dist/core/InputManager.js +149 -0
  36. package/dist/core/InputManager.js.map +1 -0
  37. package/dist/core/ScaleManager.d.ts +38 -0
  38. package/dist/core/ScaleManager.d.ts.map +1 -0
  39. package/dist/core/ScaleManager.js +106 -0
  40. package/dist/core/ScaleManager.js.map +1 -0
  41. package/dist/core/index.d.ts +9 -0
  42. package/dist/core/index.d.ts.map +1 -0
  43. package/dist/core/index.js +14 -0
  44. package/dist/core/index.js.map +1 -0
  45. package/dist/core/types.d.ts +113 -0
  46. package/dist/core/types.d.ts.map +1 -0
  47. package/dist/core/types.js +5 -0
  48. package/dist/core/types.js.map +1 -0
  49. package/dist/core/useGame.d.ts +22 -0
  50. package/dist/core/useGame.d.ts.map +1 -0
  51. package/dist/core/useGame.js +40 -0
  52. package/dist/core/useGame.js.map +1 -0
  53. package/dist/core/useGameLoop.d.ts +16 -0
  54. package/dist/core/useGameLoop.d.ts.map +1 -0
  55. package/dist/core/useGameLoop.js +35 -0
  56. package/dist/core/useGameLoop.js.map +1 -0
  57. package/dist/core/useGameSetup.d.ts +16 -0
  58. package/dist/core/useGameSetup.d.ts.map +1 -0
  59. package/dist/core/useGameSetup.js +27 -0
  60. package/dist/core/useGameSetup.js.map +1 -0
  61. package/dist/hud/BottomHint.d.ts +17 -0
  62. package/dist/hud/BottomHint.d.ts.map +1 -0
  63. package/dist/hud/BottomHint.js +68 -0
  64. package/dist/hud/BottomHint.js.map +1 -0
  65. package/dist/hud/ComboLabel.d.ts +18 -0
  66. package/dist/hud/ComboLabel.d.ts.map +1 -0
  67. package/dist/hud/ComboLabel.js +125 -0
  68. package/dist/hud/ComboLabel.js.map +1 -0
  69. package/dist/hud/HUD.d.ts +18 -0
  70. package/dist/hud/HUD.d.ts.map +1 -0
  71. package/dist/hud/HUD.js +72 -0
  72. package/dist/hud/HUD.js.map +1 -0
  73. package/dist/hud/Lives.d.ts +21 -0
  74. package/dist/hud/Lives.d.ts.map +1 -0
  75. package/dist/hud/Lives.js +92 -0
  76. package/dist/hud/Lives.js.map +1 -0
  77. package/dist/hud/Meter.d.ts +24 -0
  78. package/dist/hud/Meter.d.ts.map +1 -0
  79. package/dist/hud/Meter.js +133 -0
  80. package/dist/hud/Meter.js.map +1 -0
  81. package/dist/hud/MiniMap.d.ts +41 -0
  82. package/dist/hud/MiniMap.d.ts.map +1 -0
  83. package/dist/hud/MiniMap.js +103 -0
  84. package/dist/hud/MiniMap.js.map +1 -0
  85. package/dist/hud/Score.d.ts +15 -0
  86. package/dist/hud/Score.d.ts.map +1 -0
  87. package/dist/hud/Score.js +74 -0
  88. package/dist/hud/Score.js.map +1 -0
  89. package/dist/hud/Timer.d.ts +21 -0
  90. package/dist/hud/Timer.d.ts.map +1 -0
  91. package/dist/hud/Timer.js +111 -0
  92. package/dist/hud/Timer.js.map +1 -0
  93. package/dist/hud/WaveLabel.d.ts +15 -0
  94. package/dist/hud/WaveLabel.d.ts.map +1 -0
  95. package/dist/hud/WaveLabel.js +70 -0
  96. package/dist/hud/WaveLabel.js.map +1 -0
  97. package/dist/hud/index.d.ts +10 -0
  98. package/dist/hud/index.d.ts.map +1 -0
  99. package/dist/hud/index.js +13 -0
  100. package/dist/hud/index.js.map +1 -0
  101. package/dist/index.d.ts +8 -0
  102. package/dist/index.d.ts.map +1 -0
  103. package/dist/index.js +13 -0
  104. package/dist/index.js.map +1 -0
  105. package/dist/screens/GameOverScreen.d.ts +36 -0
  106. package/dist/screens/GameOverScreen.d.ts.map +1 -0
  107. package/dist/screens/GameOverScreen.js +255 -0
  108. package/dist/screens/GameOverScreen.js.map +1 -0
  109. package/dist/screens/LoadingScreen.d.ts +21 -0
  110. package/dist/screens/LoadingScreen.d.ts.map +1 -0
  111. package/dist/screens/LoadingScreen.js +129 -0
  112. package/dist/screens/LoadingScreen.js.map +1 -0
  113. package/dist/screens/LobbyScreen.d.ts +39 -0
  114. package/dist/screens/LobbyScreen.d.ts.map +1 -0
  115. package/dist/screens/LobbyScreen.js +266 -0
  116. package/dist/screens/LobbyScreen.js.map +1 -0
  117. package/dist/screens/PauseOverlay.d.ts +29 -0
  118. package/dist/screens/PauseOverlay.d.ts.map +1 -0
  119. package/dist/screens/PauseOverlay.js +158 -0
  120. package/dist/screens/PauseOverlay.js.map +1 -0
  121. package/dist/screens/ScreenManager.d.ts +26 -0
  122. package/dist/screens/ScreenManager.d.ts.map +1 -0
  123. package/dist/screens/ScreenManager.js +61 -0
  124. package/dist/screens/ScreenManager.js.map +1 -0
  125. package/dist/screens/TitleScreen.d.ts +30 -0
  126. package/dist/screens/TitleScreen.d.ts.map +1 -0
  127. package/dist/screens/TitleScreen.js +263 -0
  128. package/dist/screens/TitleScreen.js.map +1 -0
  129. package/dist/screens/index.d.ts +7 -0
  130. package/dist/screens/index.d.ts.map +1 -0
  131. package/dist/screens/index.js +10 -0
  132. package/dist/screens/index.js.map +1 -0
  133. package/dist/theme/ThemeProvider.d.ts +25 -0
  134. package/dist/theme/ThemeProvider.d.ts.map +1 -0
  135. package/dist/theme/ThemeProvider.js +51 -0
  136. package/dist/theme/ThemeProvider.js.map +1 -0
  137. package/dist/theme/index.d.ts +4 -0
  138. package/dist/theme/index.d.ts.map +1 -0
  139. package/dist/theme/index.js +7 -0
  140. package/dist/theme/index.js.map +1 -0
  141. package/dist/theme/palettes.d.ts +10 -0
  142. package/dist/theme/palettes.d.ts.map +1 -0
  143. package/dist/theme/palettes.js +173 -0
  144. package/dist/theme/palettes.js.map +1 -0
  145. package/dist/theme/theme.d.ts +24 -0
  146. package/dist/theme/theme.d.ts.map +1 -0
  147. package/dist/theme/theme.js +29 -0
  148. package/dist/theme/theme.js.map +1 -0
  149. package/dist/transitions/ScreenTransition.d.ts +19 -0
  150. package/dist/transitions/ScreenTransition.d.ts.map +1 -0
  151. package/dist/transitions/ScreenTransition.js +105 -0
  152. package/dist/transitions/ScreenTransition.js.map +1 -0
  153. package/dist/transitions/index.d.ts +3 -0
  154. package/dist/transitions/index.d.ts.map +1 -0
  155. package/dist/transitions/index.js +6 -0
  156. package/dist/transitions/index.js.map +1 -0
  157. package/dist/transitions/presets.d.ts +21 -0
  158. package/dist/transitions/presets.d.ts.map +1 -0
  159. package/dist/transitions/presets.js +37 -0
  160. package/dist/transitions/presets.js.map +1 -0
  161. package/package.json +62 -0
@@ -0,0 +1,129 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "preact/jsx-runtime";
2
+ // ---------------------------------------------------------------------------
3
+ // @joydle/ui — Loading screen with animated progress bar
4
+ // ---------------------------------------------------------------------------
5
+ import { useRef, useEffect } from 'preact/hooks';
6
+ // ---- GSAP helper ----------------------------------------------------------
7
+ function getGsap() {
8
+ try {
9
+ return globalThis.gsap ?? require('gsap').default;
10
+ }
11
+ catch {
12
+ return null;
13
+ }
14
+ }
15
+ // ---- Style injection ------------------------------------------------------
16
+ const LOADING_CSS = `
17
+ .joydle-loading {
18
+ position: absolute;
19
+ inset: 0;
20
+ z-index: 20;
21
+ display: flex;
22
+ flex-direction: column;
23
+ align-items: center;
24
+ justify-content: center;
25
+ gap: 16px;
26
+ font-family: var(--joydle-font, sans-serif);
27
+ color: var(--joydle-text, #fff);
28
+ background: var(--joydle-bg, #06060f);
29
+ }
30
+
31
+ .joydle-loading__label {
32
+ font-size: 0.9rem;
33
+ font-weight: 600;
34
+ letter-spacing: 0.08em;
35
+ color: var(--joydle-text-dim, rgba(255,255,255,0.5));
36
+ text-transform: uppercase;
37
+ margin: 0;
38
+ }
39
+
40
+ .joydle-loading__bar-track {
41
+ width: min(320px, 70vw);
42
+ height: 8px;
43
+ background: rgba(255, 255, 255, 0.1);
44
+ border-radius: 4px;
45
+ overflow: hidden;
46
+ position: relative;
47
+ }
48
+
49
+ .joydle-loading__bar-fill {
50
+ height: 100%;
51
+ background: var(--joydle-primary, #00f5ff);
52
+ border-radius: 4px;
53
+ transition: width 0.3s ease-out;
54
+ position: relative;
55
+ box-shadow: 0 0 var(--joydle-glow-blur, 0) var(--joydle-glow-spread, 0) var(--joydle-primary, #00f5ff);
56
+ }
57
+
58
+ .joydle-loading__bar-fill::after {
59
+ content: '';
60
+ position: absolute;
61
+ top: 0;
62
+ right: 0;
63
+ width: 40px;
64
+ height: 100%;
65
+ background: linear-gradient(
66
+ 90deg,
67
+ transparent,
68
+ rgba(255, 255, 255, 0.3)
69
+ );
70
+ animation: joydle-loading-shimmer 1.5s ease-in-out infinite;
71
+ }
72
+
73
+ @keyframes joydle-loading-shimmer {
74
+ 0% { opacity: 0; }
75
+ 50% { opacity: 1; }
76
+ 100% { opacity: 0; }
77
+ }
78
+
79
+ .joydle-loading__percent {
80
+ font-size: 0.75rem;
81
+ font-weight: 600;
82
+ color: var(--joydle-text-dim, rgba(255,255,255,0.5));
83
+ letter-spacing: 0.04em;
84
+ margin: 0;
85
+ }
86
+ `;
87
+ let styleInjected = false;
88
+ function injectStyle() {
89
+ if (styleInjected)
90
+ return;
91
+ styleInjected = true;
92
+ const el = document.createElement('style');
93
+ el.textContent = LOADING_CSS;
94
+ document.head.appendChild(el);
95
+ }
96
+ // ---- Component ------------------------------------------------------------
97
+ /**
98
+ * Centred loading screen with an animated progress bar.
99
+ *
100
+ * The bar fill uses `--joydle-primary` and animates smoothly via CSS
101
+ * transition. A subtle shimmer effect runs along the bar edge.
102
+ *
103
+ * ```tsx
104
+ * <LoadingScreen progress={0.65} label="Loading assets..." />
105
+ * ```
106
+ */
107
+ export function LoadingScreen({ progress, label = 'Loading...', screen: _screen, }) {
108
+ injectStyle();
109
+ const containerRef = useRef(null);
110
+ // Clamp progress to [0, 1].
111
+ const clamped = Math.max(0, Math.min(1, progress));
112
+ const pct = Math.round(clamped * 100);
113
+ // GSAP entrance
114
+ useEffect(() => {
115
+ const el = containerRef.current;
116
+ if (!el)
117
+ return;
118
+ const gsap = getGsap();
119
+ if (!gsap)
120
+ return;
121
+ const tl = gsap.timeline();
122
+ tl.fromTo(el, { opacity: 0 }, { opacity: 1, duration: 0.35, ease: 'power2.out' });
123
+ return () => {
124
+ tl.kill();
125
+ };
126
+ }, []);
127
+ return (_jsxs("div", { class: "joydle-loading", ref: containerRef, children: [_jsx("p", { class: "joydle-loading__label", children: label }), _jsx("div", { class: "joydle-loading__bar-track", children: _jsx("div", { class: "joydle-loading__bar-fill", style: { width: `${pct}%` } }) }), _jsxs("p", { class: "joydle-loading__percent", children: [pct, "%"] })] }));
128
+ }
129
+ //# sourceMappingURL=LoadingScreen.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LoadingScreen.js","sourceRoot":"","sources":["../../src/screens/LoadingScreen.tsx"],"names":[],"mappings":";AAAA,8EAA8E;AAC9E,0DAA0D;AAC1D,8EAA8E;AAE9E,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AAGjD,8EAA8E;AAE9E,SAAS,OAAO;IACd,IAAI,CAAC;QACH,OAAQ,UAAkB,CAAC,IAAI,IAAI,OAAO,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC;IAC7D,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED,8EAA8E;AAE9E,MAAM,WAAW,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAsEnB,CAAC;AAEF,IAAI,aAAa,GAAG,KAAK,CAAC;AAC1B,SAAS,WAAW;IAClB,IAAI,aAAa;QAAE,OAAO;IAC1B,aAAa,GAAG,IAAI,CAAC;IACrB,MAAM,EAAE,GAAG,QAAQ,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;IAC3C,EAAE,CAAC,WAAW,GAAG,WAAW,CAAC;IAC7B,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;AAChC,CAAC;AAaD,8EAA8E;AAE9E;;;;;;;;;GASG;AACH,MAAM,UAAU,aAAa,CAAC,EAC5B,QAAQ,EACR,KAAK,GAAG,YAAY,EACpB,MAAM,EAAE,OAAO,GACI;IACnB,WAAW,EAAE,CAAC;IAEd,MAAM,YAAY,GAAG,MAAM,CAAiB,IAAI,CAAC,CAAC;IAElD,4BAA4B;IAC5B,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,CAAC;IACnD,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,GAAG,CAAC,CAAC;IAEtC,gBAAgB;IAChB,SAAS,CAAC,GAAG,EAAE;QACb,MAAM,EAAE,GAAG,YAAY,CAAC,OAAO,CAAC;QAChC,IAAI,CAAC,EAAE;YAAE,OAAO;QAEhB,MAAM,IAAI,GAAG,OAAO,EAAE,CAAC;QACvB,IAAI,CAAC,IAAI;YAAE,OAAO;QAElB,MAAM,EAAE,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;QAC3B,EAAE,CAAC,MAAM,CAAC,EAAE,EAAE,EAAE,OAAO,EAAE,CAAC,EAAE,EAAE,EAAE,OAAO,EAAE,CAAC,EAAE,QAAQ,EAAE,IAAI,EAAE,IAAI,EAAE,YAAY,EAAE,CAAC,CAAC;QAElF,OAAO,GAAG,EAAE;YACV,EAAE,CAAC,IAAI,EAAE,CAAC;QACZ,CAAC,CAAC;IACJ,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,OAAO,CACL,eAAK,KAAK,EAAC,gBAAgB,EAAC,GAAG,EAAE,YAAY,aAC3C,YAAG,KAAK,EAAC,uBAAuB,YAAE,KAAK,GAAK,EAE5C,cAAK,KAAK,EAAC,2BAA2B,YACpC,cACE,KAAK,EAAC,0BAA0B,EAChC,KAAK,EAAE,EAAE,KAAK,EAAE,GAAG,GAAG,GAAG,EAAE,GAC3B,GACE,EAEN,aAAG,KAAK,EAAC,yBAAyB,aAAE,GAAG,SAAM,IACzC,CACP,CAAC;AACJ,CAAC"}
@@ -0,0 +1,39 @@
1
+ import type { JSX } from 'preact';
2
+ export interface LobbyPlayer {
3
+ id: string;
4
+ name: string;
5
+ ready: boolean;
6
+ }
7
+ export interface LobbyScreenProps {
8
+ /** List of players in the lobby. */
9
+ players: LobbyPlayer[];
10
+ /** Match code to display for others to join. */
11
+ matchCode?: string;
12
+ /** Callback when the Ready button is clicked. */
13
+ onReady: () => void;
14
+ /** Callback when the Start button is clicked (host only). */
15
+ onStart?: () => void;
16
+ /** Maximum number of players (shown in counter). */
17
+ maxPlayers?: number;
18
+ /** Screen identifier for ScreenManager matching. */
19
+ screen?: string;
20
+ }
21
+ /**
22
+ * Multiplayer lobby screen with a player list, ready indicators, and
23
+ * optional match code display.
24
+ *
25
+ * ```tsx
26
+ * <LobbyScreen
27
+ * players={[
28
+ * { id: '1', name: 'Player 1', ready: true },
29
+ * { id: '2', name: 'Player 2', ready: false },
30
+ * ]}
31
+ * matchCode="ABC123"
32
+ * onReady={() => toggleReady()}
33
+ * onStart={() => startGame()}
34
+ * maxPlayers={4}
35
+ * />
36
+ * ```
37
+ */
38
+ export declare function LobbyScreen({ players, matchCode, onReady, onStart, maxPlayers, screen: _screen, }: LobbyScreenProps): JSX.Element;
39
+ //# sourceMappingURL=LobbyScreen.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LobbyScreen.d.ts","sourceRoot":"","sources":["../../src/screens/LobbyScreen.tsx"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,QAAQ,CAAC;AA6NlC,MAAM,WAAW,WAAW;IAC1B,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,OAAO,CAAC;CAChB;AAED,MAAM,WAAW,gBAAgB;IAC/B,oCAAoC;IACpC,OAAO,EAAE,WAAW,EAAE,CAAC;IACvB,gDAAgD;IAChD,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,iDAAiD;IACjD,OAAO,EAAE,MAAM,IAAI,CAAC;IACpB,6DAA6D;IAC7D,OAAO,CAAC,EAAE,MAAM,IAAI,CAAC;IACrB,oDAAoD;IACpD,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,oDAAoD;IACpD,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB;AAID;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,WAAW,CAAC,EAC1B,OAAO,EACP,SAAS,EACT,OAAO,EACP,OAAO,EACP,UAAU,EACV,MAAM,EAAE,OAAO,GAChB,EAAE,gBAAgB,GAAG,GAAG,CAAC,OAAO,CA0FhC"}
@@ -0,0 +1,266 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "preact/jsx-runtime";
2
+ // ---------------------------------------------------------------------------
3
+ // @joydle/ui — Multiplayer lobby screen
4
+ // ---------------------------------------------------------------------------
5
+ import { useRef, useEffect } from 'preact/hooks';
6
+ // ---- GSAP helper ----------------------------------------------------------
7
+ function getGsap() {
8
+ try {
9
+ return globalThis.gsap ?? require('gsap').default;
10
+ }
11
+ catch {
12
+ return null;
13
+ }
14
+ }
15
+ // ---- Style injection ------------------------------------------------------
16
+ const LOBBY_CSS = `
17
+ .joydle-lobby {
18
+ position: absolute;
19
+ inset: 0;
20
+ z-index: 20;
21
+ display: flex;
22
+ flex-direction: column;
23
+ align-items: center;
24
+ justify-content: center;
25
+ gap: 20px;
26
+ font-family: var(--joydle-font, sans-serif);
27
+ color: var(--joydle-text, #fff);
28
+ background: var(--joydle-bg, #06060f);
29
+ overflow: hidden;
30
+ }
31
+
32
+ .joydle-lobby__backdrop {
33
+ position: absolute;
34
+ inset: 0;
35
+ background: rgba(0, 0, 0, 0.45);
36
+ z-index: 0;
37
+ }
38
+
39
+ .joydle-lobby__content {
40
+ position: relative;
41
+ z-index: 1;
42
+ display: flex;
43
+ flex-direction: column;
44
+ align-items: center;
45
+ gap: 16px;
46
+ padding: 24px;
47
+ width: min(400px, 85vw);
48
+ }
49
+
50
+ .joydle-lobby__title {
51
+ font-size: clamp(1.5rem, 4vw, 2.2rem);
52
+ font-weight: 900;
53
+ letter-spacing: 0.08em;
54
+ color: var(--joydle-primary, #00f5ff);
55
+ text-transform: uppercase;
56
+ margin: 0;
57
+ text-shadow: 0 0 var(--joydle-glow-blur, 0) var(--joydle-primary, #00f5ff);
58
+ }
59
+
60
+ .joydle-lobby__code-container {
61
+ display: flex;
62
+ flex-direction: column;
63
+ align-items: center;
64
+ gap: 4px;
65
+ }
66
+
67
+ .joydle-lobby__code-label {
68
+ font-size: 0.7rem;
69
+ color: var(--joydle-text-dim, rgba(255,255,255,0.5));
70
+ letter-spacing: 0.1em;
71
+ text-transform: uppercase;
72
+ margin: 0;
73
+ }
74
+
75
+ .joydle-lobby__code {
76
+ font-size: 1.6rem;
77
+ font-weight: 900;
78
+ letter-spacing: 0.2em;
79
+ color: var(--joydle-gold, #ffd700);
80
+ font-family: var(--joydle-font, monospace);
81
+ margin: 0;
82
+ padding: 6px 16px;
83
+ background: rgba(255, 255, 255, 0.06);
84
+ border-radius: var(--joydle-border-radius, 4px);
85
+ border: 1px solid rgba(255, 255, 255, 0.1);
86
+ }
87
+
88
+ .joydle-lobby__player-count {
89
+ font-size: 0.8rem;
90
+ color: var(--joydle-text-dim, rgba(255,255,255,0.5));
91
+ letter-spacing: 0.06em;
92
+ margin: 0;
93
+ }
94
+
95
+ .joydle-lobby__player-list {
96
+ list-style: none;
97
+ margin: 0;
98
+ padding: 0;
99
+ width: 100%;
100
+ display: flex;
101
+ flex-direction: column;
102
+ gap: 6px;
103
+ }
104
+
105
+ .joydle-lobby__player {
106
+ display: flex;
107
+ align-items: center;
108
+ gap: 10px;
109
+ padding: 10px 14px;
110
+ background: rgba(255, 255, 255, 0.05);
111
+ border-radius: var(--joydle-border-radius, 4px);
112
+ border: 1px solid rgba(255, 255, 255, 0.08);
113
+ transition: background 0.15s ease;
114
+ }
115
+
116
+ .joydle-lobby__player--ready {
117
+ border-color: var(--joydle-success, #00ff88);
118
+ background: rgba(0, 255, 136, 0.06);
119
+ }
120
+
121
+ .joydle-lobby__player-dot {
122
+ width: 10px;
123
+ height: 10px;
124
+ border-radius: 50%;
125
+ flex-shrink: 0;
126
+ background: rgba(255, 255, 255, 0.2);
127
+ transition: background 0.2s ease;
128
+ }
129
+
130
+ .joydle-lobby__player--ready .joydle-lobby__player-dot {
131
+ background: var(--joydle-success, #00ff88);
132
+ box-shadow: 0 0 6px 2px var(--joydle-success, #00ff88);
133
+ }
134
+
135
+ .joydle-lobby__player-name {
136
+ font-size: 0.9rem;
137
+ font-weight: 600;
138
+ color: var(--joydle-text, #fff);
139
+ flex: 1;
140
+ }
141
+
142
+ .joydle-lobby__player-status {
143
+ font-size: 0.7rem;
144
+ letter-spacing: 0.08em;
145
+ text-transform: uppercase;
146
+ color: var(--joydle-text-dim, rgba(255,255,255,0.5));
147
+ }
148
+
149
+ .joydle-lobby__player--ready .joydle-lobby__player-status {
150
+ color: var(--joydle-success, #00ff88);
151
+ }
152
+
153
+ .joydle-lobby__buttons {
154
+ display: flex;
155
+ flex-direction: column;
156
+ align-items: center;
157
+ gap: 10px;
158
+ margin-top: 8px;
159
+ width: 100%;
160
+ }
161
+
162
+ .joydle-lobby__btn {
163
+ padding: 12px 40px;
164
+ font-size: 1rem;
165
+ font-weight: 700;
166
+ font-family: var(--joydle-font, sans-serif);
167
+ letter-spacing: 0.1em;
168
+ text-transform: uppercase;
169
+ border: none;
170
+ border-radius: var(--joydle-border-radius, 4px);
171
+ cursor: pointer;
172
+ transition: transform 0.12s ease, box-shadow 0.12s ease;
173
+ width: 100%;
174
+ max-width: 240px;
175
+ }
176
+
177
+ .joydle-lobby__btn:hover {
178
+ transform: scale(1.03);
179
+ }
180
+
181
+ .joydle-lobby__btn:active {
182
+ transform: scale(0.97);
183
+ }
184
+
185
+ .joydle-lobby__btn--ready {
186
+ color: var(--joydle-bg, #06060f);
187
+ background: var(--joydle-success, #00ff88);
188
+ box-shadow: 0 0 var(--joydle-glow-blur, 0) var(--joydle-glow-spread, 0) var(--joydle-success, #00ff88);
189
+ }
190
+
191
+ .joydle-lobby__btn--ready:hover {
192
+ box-shadow: 0 0 12px 3px var(--joydle-success, #00ff88);
193
+ }
194
+
195
+ .joydle-lobby__btn--start {
196
+ color: var(--joydle-bg, #06060f);
197
+ background: var(--joydle-accent, #ff2d7b);
198
+ box-shadow: 0 0 var(--joydle-glow-blur, 0) var(--joydle-glow-spread, 0) var(--joydle-accent, #ff2d7b);
199
+ }
200
+
201
+ .joydle-lobby__btn--start:hover {
202
+ box-shadow: 0 0 12px 3px var(--joydle-accent, #ff2d7b);
203
+ }
204
+
205
+ .joydle-lobby__btn:disabled {
206
+ opacity: 0.4;
207
+ cursor: not-allowed;
208
+ transform: none;
209
+ }
210
+ `;
211
+ let styleInjected = false;
212
+ function injectStyle() {
213
+ if (styleInjected)
214
+ return;
215
+ styleInjected = true;
216
+ const el = document.createElement('style');
217
+ el.textContent = LOBBY_CSS;
218
+ document.head.appendChild(el);
219
+ }
220
+ // ---- Component ------------------------------------------------------------
221
+ /**
222
+ * Multiplayer lobby screen with a player list, ready indicators, and
223
+ * optional match code display.
224
+ *
225
+ * ```tsx
226
+ * <LobbyScreen
227
+ * players={[
228
+ * { id: '1', name: 'Player 1', ready: true },
229
+ * { id: '2', name: 'Player 2', ready: false },
230
+ * ]}
231
+ * matchCode="ABC123"
232
+ * onReady={() => toggleReady()}
233
+ * onStart={() => startGame()}
234
+ * maxPlayers={4}
235
+ * />
236
+ * ```
237
+ */
238
+ export function LobbyScreen({ players, matchCode, onReady, onStart, maxPlayers, screen: _screen, }) {
239
+ injectStyle();
240
+ const containerRef = useRef(null);
241
+ // GSAP entrance
242
+ useEffect(() => {
243
+ const el = containerRef.current;
244
+ if (!el)
245
+ return;
246
+ const gsap = getGsap();
247
+ if (!gsap)
248
+ return;
249
+ const tl = gsap.timeline();
250
+ tl.fromTo(el, { opacity: 0, y: 20 }, { opacity: 1, y: 0, duration: 0.4, ease: 'power2.out' });
251
+ // Stagger the player rows
252
+ const rows = el.querySelectorAll('.joydle-lobby__player');
253
+ if (rows.length > 0) {
254
+ tl.fromTo(rows, { opacity: 0, x: -20 }, { opacity: 1, x: 0, duration: 0.3, ease: 'power2.out', stagger: 0.06 }, 0.15);
255
+ }
256
+ return () => {
257
+ tl.kill();
258
+ };
259
+ }, []);
260
+ const allReady = players.length > 0 && players.every((p) => p.ready);
261
+ const countLabel = maxPlayers
262
+ ? `${players.length} / ${maxPlayers} players`
263
+ : `${players.length} player${players.length !== 1 ? 's' : ''}`;
264
+ return (_jsxs("div", { class: "joydle-lobby", ref: containerRef, children: [_jsx("div", { class: "joydle-lobby__backdrop" }), _jsxs("div", { class: "joydle-lobby__content", children: [_jsx("h1", { class: "joydle-lobby__title", children: "LOBBY" }), matchCode && (_jsxs("div", { class: "joydle-lobby__code-container", children: [_jsx("p", { class: "joydle-lobby__code-label", children: "Match Code" }), _jsx("p", { class: "joydle-lobby__code", children: matchCode })] })), _jsx("p", { class: "joydle-lobby__player-count", children: countLabel }), _jsx("ul", { class: "joydle-lobby__player-list", children: players.map((player) => (_jsxs("li", { class: `joydle-lobby__player${player.ready ? ' joydle-lobby__player--ready' : ''}`, children: [_jsx("span", { class: "joydle-lobby__player-dot" }), _jsx("span", { class: "joydle-lobby__player-name", children: player.name }), _jsx("span", { class: "joydle-lobby__player-status", children: player.ready ? 'READY' : 'WAITING' })] }, player.id))) }), _jsxs("div", { class: "joydle-lobby__buttons", children: [_jsx("button", { class: "joydle-lobby__btn joydle-lobby__btn--ready", onClick: onReady, type: "button", children: "READY" }), onStart && (_jsx("button", { class: "joydle-lobby__btn joydle-lobby__btn--start", onClick: onStart, disabled: !allReady, type: "button", children: "START GAME" }))] })] })] }));
265
+ }
266
+ //# sourceMappingURL=LobbyScreen.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LobbyScreen.js","sourceRoot":"","sources":["../../src/screens/LobbyScreen.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,29 @@
1
+ import type { JSX } from 'preact';
2
+ export interface PauseOverlayProps {
3
+ /** Callback when the resume button is clicked. */
4
+ onResume: () => void;
5
+ /** Whether to show the restart button. */
6
+ showRestart?: boolean;
7
+ /** Callback when the restart button is clicked. */
8
+ onRestart?: () => void;
9
+ /** Screen identifier for ScreenManager matching. */
10
+ screen?: string;
11
+ }
12
+ /**
13
+ * Semi-transparent pause overlay with blur backdrop.
14
+ *
15
+ * Auto-dismisses on the Escape key. Renders a Resume button and an
16
+ * optional Restart button.
17
+ *
18
+ * ```tsx
19
+ * {state.screen === 'paused' && (
20
+ * <PauseOverlay
21
+ * onResume={() => setState({ screen: 'playing' })}
22
+ * showRestart
23
+ * onRestart={() => setState({ screen: 'title' })}
24
+ * />
25
+ * )}
26
+ * ```
27
+ */
28
+ export declare function PauseOverlay({ onResume, showRestart, onRestart, screen: _screen, }: PauseOverlayProps): JSX.Element;
29
+ //# sourceMappingURL=PauseOverlay.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PauseOverlay.d.ts","sourceRoot":"","sources":["../../src/screens/PauseOverlay.tsx"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,GAAG,EAAE,MAAM,QAAQ,CAAC;AA8GlC,MAAM,WAAW,iBAAiB;IAChC,kDAAkD;IAClD,QAAQ,EAAE,MAAM,IAAI,CAAC;IACrB,0CAA0C;IAC1C,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,mDAAmD;IACnD,SAAS,CAAC,EAAE,MAAM,IAAI,CAAC;IACvB,oDAAoD;IACpD,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB;AAID;;;;;;;;;;;;;;;GAeG;AACH,wBAAgB,YAAY,CAAC,EAC3B,QAAQ,EACR,WAAmB,EACnB,SAAS,EACT,MAAM,EAAE,OAAO,GAChB,EAAE,iBAAiB,GAAG,GAAG,CAAC,OAAO,CA+DjC"}
@@ -0,0 +1,158 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "preact/jsx-runtime";
2
+ // ---------------------------------------------------------------------------
3
+ // @joydle/ui — Pause overlay with blur backdrop
4
+ // ---------------------------------------------------------------------------
5
+ import { useRef, useEffect } from 'preact/hooks';
6
+ // ---- GSAP helper ----------------------------------------------------------
7
+ function getGsap() {
8
+ try {
9
+ return globalThis.gsap ?? require('gsap').default;
10
+ }
11
+ catch {
12
+ return null;
13
+ }
14
+ }
15
+ // ---- Style injection ------------------------------------------------------
16
+ const PAUSE_CSS = `
17
+ .joydle-pause {
18
+ position: absolute;
19
+ inset: 0;
20
+ z-index: 20;
21
+ display: flex;
22
+ flex-direction: column;
23
+ align-items: center;
24
+ justify-content: center;
25
+ gap: 16px;
26
+ font-family: var(--joydle-font, sans-serif);
27
+ color: var(--joydle-text, #fff);
28
+ background: rgba(0, 0, 0, 0.7);
29
+ backdrop-filter: blur(4px);
30
+ -webkit-backdrop-filter: blur(4px);
31
+ }
32
+
33
+ .joydle-pause__title {
34
+ font-size: clamp(2rem, 5vw, 3.5rem);
35
+ font-weight: 900;
36
+ letter-spacing: 0.12em;
37
+ color: var(--joydle-text, #fff);
38
+ text-transform: uppercase;
39
+ margin: 0;
40
+ text-shadow: 0 0 var(--joydle-glow-blur, 0) var(--joydle-primary, #00f5ff);
41
+ }
42
+
43
+ .joydle-pause__buttons {
44
+ display: flex;
45
+ flex-direction: column;
46
+ align-items: center;
47
+ gap: 12px;
48
+ margin-top: 8px;
49
+ }
50
+
51
+ .joydle-pause__btn {
52
+ padding: 12px 40px;
53
+ font-size: 1rem;
54
+ font-weight: 700;
55
+ font-family: var(--joydle-font, sans-serif);
56
+ letter-spacing: 0.1em;
57
+ text-transform: uppercase;
58
+ border: none;
59
+ border-radius: var(--joydle-border-radius, 4px);
60
+ cursor: pointer;
61
+ transition: transform 0.12s ease, box-shadow 0.12s ease;
62
+ min-width: 180px;
63
+ }
64
+
65
+ .joydle-pause__btn:hover {
66
+ transform: scale(1.05);
67
+ }
68
+
69
+ .joydle-pause__btn:active {
70
+ transform: scale(0.97);
71
+ }
72
+
73
+ .joydle-pause__btn--resume {
74
+ color: var(--joydle-bg, #06060f);
75
+ background: var(--joydle-primary, #00f5ff);
76
+ box-shadow: 0 0 var(--joydle-glow-blur, 0) var(--joydle-glow-spread, 0) var(--joydle-primary, #00f5ff);
77
+ }
78
+
79
+ .joydle-pause__btn--resume:hover {
80
+ box-shadow: 0 0 16px 4px var(--joydle-primary, #00f5ff);
81
+ }
82
+
83
+ .joydle-pause__btn--restart {
84
+ color: var(--joydle-text, #fff);
85
+ background: transparent;
86
+ border: 2px solid var(--joydle-text-dim, rgba(255,255,255,0.5));
87
+ }
88
+
89
+ .joydle-pause__btn--restart:hover {
90
+ border-color: var(--joydle-text, #fff);
91
+ }
92
+
93
+ .joydle-pause__hint {
94
+ margin-top: 8px;
95
+ font-size: 0.7rem;
96
+ color: var(--joydle-text-dim, rgba(255,255,255,0.5));
97
+ letter-spacing: 0.06em;
98
+ }
99
+ `;
100
+ let styleInjected = false;
101
+ function injectStyle() {
102
+ if (styleInjected)
103
+ return;
104
+ styleInjected = true;
105
+ const el = document.createElement('style');
106
+ el.textContent = PAUSE_CSS;
107
+ document.head.appendChild(el);
108
+ }
109
+ // ---- Component ------------------------------------------------------------
110
+ /**
111
+ * Semi-transparent pause overlay with blur backdrop.
112
+ *
113
+ * Auto-dismisses on the Escape key. Renders a Resume button and an
114
+ * optional Restart button.
115
+ *
116
+ * ```tsx
117
+ * {state.screen === 'paused' && (
118
+ * <PauseOverlay
119
+ * onResume={() => setState({ screen: 'playing' })}
120
+ * showRestart
121
+ * onRestart={() => setState({ screen: 'title' })}
122
+ * />
123
+ * )}
124
+ * ```
125
+ */
126
+ export function PauseOverlay({ onResume, showRestart = false, onRestart, screen: _screen, }) {
127
+ injectStyle();
128
+ const overlayRef = useRef(null);
129
+ // GSAP entrance
130
+ useEffect(() => {
131
+ const el = overlayRef.current;
132
+ if (!el)
133
+ return;
134
+ const gsap = getGsap();
135
+ if (!gsap)
136
+ return;
137
+ const tl = gsap.timeline();
138
+ tl.fromTo(el, { opacity: 0 }, { opacity: 1, duration: 0.25, ease: 'power2.out' });
139
+ return () => {
140
+ tl.kill();
141
+ };
142
+ }, []);
143
+ // ESC key handler
144
+ useEffect(() => {
145
+ const handleKey = (e) => {
146
+ if (e.key === 'Escape') {
147
+ e.preventDefault();
148
+ onResume();
149
+ }
150
+ };
151
+ document.addEventListener('keydown', handleKey);
152
+ return () => {
153
+ document.removeEventListener('keydown', handleKey);
154
+ };
155
+ }, [onResume]);
156
+ return (_jsxs("div", { class: "joydle-pause", ref: overlayRef, children: [_jsx("h1", { class: "joydle-pause__title", children: "PAUSED" }), _jsxs("div", { class: "joydle-pause__buttons", children: [_jsx("button", { class: "joydle-pause__btn joydle-pause__btn--resume", onClick: onResume, type: "button", children: "RESUME" }), showRestart && onRestart && (_jsx("button", { class: "joydle-pause__btn joydle-pause__btn--restart", onClick: onRestart, type: "button", children: "RESTART" }))] }), _jsx("p", { class: "joydle-pause__hint", children: "Press ESC to resume" })] }));
157
+ }
158
+ //# sourceMappingURL=PauseOverlay.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PauseOverlay.js","sourceRoot":"","sources":["../../src/screens/PauseOverlay.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,26 @@
1
+ import type { ComponentChildren } from 'preact';
2
+ export interface ScreenManagerProps {
3
+ /** The currently active screen identifier. */
4
+ screen: string;
5
+ /** Transition preset name (default: `"fade"`). */
6
+ transition?: 'fade' | 'slide' | 'zoom' | 'none';
7
+ /** Transition duration override in seconds. */
8
+ duration?: number;
9
+ /** Screen components — each should have a `screen` prop. */
10
+ children?: ComponentChildren;
11
+ }
12
+ /**
13
+ * Renders only the child whose `screen` prop matches the current `screen`
14
+ * value. Wraps the visible screen in a `<ScreenTransition>` for animated
15
+ * enter/exit effects.
16
+ *
17
+ * ```tsx
18
+ * <ScreenManager screen={state.screen} transition="fade">
19
+ * <TitleScreen screen="title" title="My Game" onPlay={start} />
20
+ * <GameOverScreen screen="gameover" score={42} onRetry={restart} />
21
+ * <PauseOverlay screen="paused" onResume={resume} />
22
+ * </ScreenManager>
23
+ * ```
24
+ */
25
+ export declare function ScreenManager({ screen, transition, duration, children, }: ScreenManagerProps): import("preact").JSX.Element | null;
26
+ //# sourceMappingURL=ScreenManager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ScreenManager.d.ts","sourceRoot":"","sources":["../../src/screens/ScreenManager.tsx"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,iBAAiB,EAAS,MAAM,QAAQ,CAAC;AA4BvD,MAAM,WAAW,kBAAkB;IACjC,8CAA8C;IAC9C,MAAM,EAAE,MAAM,CAAC;IACf,kDAAkD;IAClD,UAAU,CAAC,EAAE,MAAM,GAAG,OAAO,GAAG,MAAM,GAAG,MAAM,CAAC;IAChD,+CAA+C;IAC/C,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,4DAA4D;IAC5D,QAAQ,CAAC,EAAE,iBAAiB,CAAC;CAC9B;AAID;;;;;;;;;;;;GAYG;AACH,wBAAgB,aAAa,CAAC,EAC5B,MAAM,EACN,UAAmB,EACnB,QAAQ,EACR,QAAQ,GACT,EAAE,kBAAkB,uCAgCpB"}