@rpgjs/client 5.0.0-beta.1 → 5.0.0-beta.10
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/CHANGELOG.md +37 -0
- package/LICENSE +19 -0
- package/dist/Game/AnimationManager.d.ts +1 -1
- package/dist/Game/AnimationManager.js +18 -9
- package/dist/Game/AnimationManager.js.map +1 -1
- package/dist/Game/AnimationManager.spec.d.ts +1 -0
- package/dist/Game/Event.js.map +1 -1
- package/dist/Game/Map.d.ts +9 -1
- package/dist/Game/Map.js +63 -5
- package/dist/Game/Map.js.map +1 -1
- package/dist/Game/Object.d.ts +47 -15
- package/dist/Game/Object.js +82 -38
- package/dist/Game/Object.js.map +1 -1
- package/dist/Game/Player.js.map +1 -1
- package/dist/Gui/Gui.d.ts +17 -4
- package/dist/Gui/Gui.js +78 -48
- package/dist/Gui/Gui.js.map +1 -1
- package/dist/Gui/Gui.spec.d.ts +1 -0
- package/dist/Gui/NotificationManager.js.map +1 -1
- package/dist/Resource.js +1 -1
- package/dist/Resource.js.map +1 -1
- package/dist/RpgClient.d.ts +57 -2
- package/dist/RpgClientEngine.d.ts +61 -6
- package/dist/RpgClientEngine.js +122 -14
- package/dist/RpgClientEngine.js.map +1 -1
- package/dist/Sound.js.map +1 -1
- package/dist/_virtual/{_@oxc-project_runtime@0.122.0 → _@oxc-project_runtime@0.130.0}/helpers/decorate.js +1 -1
- package/dist/_virtual/{_@oxc-project_runtime@0.122.0 → _@oxc-project_runtime@0.130.0}/helpers/decorateMetadata.js +1 -1
- package/dist/components/animations/animation.ce.js +4 -5
- package/dist/components/animations/animation.ce.js.map +1 -1
- package/dist/components/animations/hit.ce.js +19 -25
- package/dist/components/animations/hit.ce.js.map +1 -1
- package/dist/components/animations/index.js +4 -4
- package/dist/components/animations/index.js.map +1 -1
- package/dist/components/character.ce.js +406 -226
- package/dist/components/character.ce.js.map +1 -1
- package/dist/components/dynamics/bar.ce.js +96 -0
- package/dist/components/dynamics/bar.ce.js.map +1 -0
- package/dist/components/dynamics/image.ce.js +23 -0
- package/dist/components/dynamics/image.ce.js.map +1 -0
- package/dist/components/dynamics/parse-value.d.ts +3 -0
- package/dist/components/dynamics/parse-value.js +54 -35
- package/dist/components/dynamics/parse-value.js.map +1 -1
- package/dist/components/dynamics/parse-value.spec.d.ts +1 -0
- package/dist/components/dynamics/shape-utils.d.ts +16 -0
- package/dist/components/dynamics/shape-utils.js +73 -0
- package/dist/components/dynamics/shape-utils.js.map +1 -0
- package/dist/components/dynamics/shape-utils.spec.d.ts +1 -0
- package/dist/components/dynamics/shape.ce.js +83 -0
- package/dist/components/dynamics/shape.ce.js.map +1 -0
- package/dist/components/dynamics/text.ce.js +33 -56
- package/dist/components/dynamics/text.ce.js.map +1 -1
- package/dist/components/gui/box.ce.js +6 -8
- package/dist/components/gui/box.ce.js.map +1 -1
- package/dist/components/gui/dialogbox/index.ce.js +53 -60
- package/dist/components/gui/dialogbox/index.ce.js.map +1 -1
- package/dist/components/gui/gameover.ce.js +39 -63
- package/dist/components/gui/gameover.ce.js.map +1 -1
- package/dist/components/gui/hud/hud.ce.js +21 -30
- package/dist/components/gui/hud/hud.ce.js.map +1 -1
- package/dist/components/gui/menu/equip-menu.ce.js +110 -164
- package/dist/components/gui/menu/equip-menu.ce.js.map +1 -1
- package/dist/components/gui/menu/exit-menu.ce.js +6 -5
- package/dist/components/gui/menu/exit-menu.ce.js.map +1 -1
- package/dist/components/gui/menu/items-menu.ce.js +49 -67
- package/dist/components/gui/menu/items-menu.ce.js.map +1 -1
- package/dist/components/gui/menu/main-menu.ce.js +72 -90
- package/dist/components/gui/menu/main-menu.ce.js.map +1 -1
- package/dist/components/gui/menu/options-menu.ce.js +5 -4
- package/dist/components/gui/menu/options-menu.ce.js.map +1 -1
- package/dist/components/gui/menu/skills-menu.ce.js +12 -17
- package/dist/components/gui/menu/skills-menu.ce.js.map +1 -1
- package/dist/components/gui/mobile/index.js +2 -2
- package/dist/components/gui/mobile/index.js.map +1 -1
- package/dist/components/gui/mobile/mobile.ce.js +5 -4
- package/dist/components/gui/mobile/mobile.ce.js.map +1 -1
- package/dist/components/gui/notification/notification.ce.js +22 -24
- package/dist/components/gui/notification/notification.ce.js.map +1 -1
- package/dist/components/gui/save-load.ce.js +70 -248
- package/dist/components/gui/save-load.ce.js.map +1 -1
- package/dist/components/gui/shop/shop.ce.js +87 -125
- package/dist/components/gui/shop/shop.ce.js.map +1 -1
- package/dist/components/gui/title-screen.ce.js +42 -68
- package/dist/components/gui/title-screen.ce.js.map +1 -1
- package/dist/components/player-components-utils.d.ts +67 -0
- package/dist/components/player-components-utils.js +162 -0
- package/dist/components/player-components-utils.js.map +1 -0
- package/dist/components/player-components-utils.spec.d.ts +1 -0
- package/dist/components/player-components.ce.js +188 -0
- package/dist/components/player-components.ce.js.map +1 -0
- package/dist/components/prebuilt/hp-bar.ce.js +41 -44
- package/dist/components/prebuilt/hp-bar.ce.js.map +1 -1
- package/dist/components/prebuilt/light-halo.ce.js +35 -59
- package/dist/components/prebuilt/light-halo.ce.js.map +1 -1
- package/dist/components/scenes/canvas.ce.js +157 -21
- package/dist/components/scenes/canvas.ce.js.map +1 -1
- package/dist/components/scenes/draw-map.ce.js +19 -29
- package/dist/components/scenes/draw-map.ce.js.map +1 -1
- package/dist/components/scenes/event-layer.ce.js +9 -8
- package/dist/components/scenes/event-layer.ce.js.map +1 -1
- package/dist/core/inject.js +1 -1
- package/dist/core/inject.js.map +1 -1
- package/dist/core/setup.js +1 -1
- package/dist/core/setup.js.map +1 -1
- package/dist/decorators/spritesheet.d.ts +1 -0
- package/dist/decorators/spritesheet.js +11 -0
- package/dist/decorators/spritesheet.js.map +1 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +21 -20
- package/dist/module.js +4 -1
- package/dist/module.js.map +1 -1
- package/dist/node_modules/.pnpm/{@signe_di@2.9.0 → @signe_di@3.0.1}/node_modules/@signe/di/dist/index.js +7 -117
- package/dist/node_modules/.pnpm/@signe_di@3.0.1/node_modules/@signe/di/dist/index.js.map +1 -0
- package/dist/node_modules/.pnpm/@signe_reactive@3.0.1/node_modules/@signe/reactive/dist/index.js +239 -0
- package/dist/node_modules/.pnpm/@signe_reactive@3.0.1/node_modules/@signe/reactive/dist/index.js.map +1 -0
- package/dist/node_modules/.pnpm/@signe_room@3.0.1/node_modules/@signe/room/dist/chunk-EUXUH3YW.js +13 -0
- package/dist/node_modules/.pnpm/@signe_room@3.0.1/node_modules/@signe/room/dist/chunk-EUXUH3YW.js.map +1 -0
- package/dist/node_modules/.pnpm/@signe_room@3.0.1/node_modules/@signe/room/dist/index.js +696 -0
- package/dist/node_modules/.pnpm/@signe_room@3.0.1/node_modules/@signe/room/dist/index.js.map +1 -0
- package/dist/node_modules/.pnpm/@signe_sync@3.0.1/node_modules/@signe/sync/dist/client/index.js +44 -0
- package/dist/node_modules/.pnpm/@signe_sync@3.0.1/node_modules/@signe/sync/dist/client/index.js.map +1 -0
- package/dist/node_modules/.pnpm/{@signe_sync@2.9.0 → @signe_sync@3.0.1}/node_modules/@signe/sync/dist/index.js +57 -141
- package/dist/node_modules/.pnpm/@signe_sync@3.0.1/node_modules/@signe/sync/dist/index.js.map +1 -0
- package/dist/node_modules/.pnpm/partysocket@1.1.3/node_modules/partysocket/dist/chunk-HAC622V3.js.map +1 -1
- package/dist/node_modules/.pnpm/partysocket@1.1.3/node_modules/partysocket/dist/chunk-S74YV6PU.js.map +1 -1
- package/dist/node_modules/.pnpm/zod@3.24.2/node_modules/zod/lib/index.js +27 -27
- package/dist/node_modules/.pnpm/zod@3.24.2/node_modules/zod/lib/index.js.map +1 -1
- package/dist/presets/animation.js.map +1 -1
- package/dist/presets/faceset.js.map +1 -1
- package/dist/presets/icon.js.map +1 -1
- package/dist/presets/index.js.map +1 -1
- package/dist/presets/lpc.js.map +1 -1
- package/dist/presets/rmspritesheet.js.map +1 -1
- package/dist/services/AbstractSocket.js.map +1 -1
- package/dist/services/keyboardControls.js.map +1 -1
- package/dist/services/loadMap.d.ts +6 -0
- package/dist/services/loadMap.js +1 -1
- package/dist/services/loadMap.js.map +1 -1
- package/dist/services/mmorpg.js +9 -4
- package/dist/services/mmorpg.js.map +1 -1
- package/dist/services/save.js.map +1 -1
- package/dist/services/save.spec.d.ts +1 -0
- package/dist/services/standalone.js +1 -1
- package/dist/services/standalone.js.map +1 -1
- package/dist/utils/getEntityProp.js +4 -3
- package/dist/utils/getEntityProp.js.map +1 -1
- package/dist/utils/getEntityProp.spec.d.ts +1 -0
- package/dist/utils/readPropValue.d.ts +2 -0
- package/dist/utils/readPropValue.js +13 -0
- package/dist/utils/readPropValue.js.map +1 -0
- package/package.json +13 -14
- package/src/Game/AnimationManager.spec.ts +30 -0
- package/src/Game/AnimationManager.ts +22 -10
- package/src/Game/Map.ts +91 -2
- package/src/Game/Object.ts +148 -69
- package/src/Gui/Gui.spec.ts +273 -0
- package/src/Gui/Gui.ts +105 -50
- package/src/Resource.ts +1 -2
- package/src/RpgClient.ts +63 -2
- package/src/RpgClientEngine.ts +173 -25
- package/src/components/character.ce +422 -15
- package/src/components/dynamics/bar.ce +87 -0
- package/src/components/dynamics/image.ce +20 -0
- package/src/components/dynamics/parse-value.spec.ts +83 -0
- package/src/components/dynamics/parse-value.ts +111 -37
- package/src/components/dynamics/shape-utils.spec.ts +46 -0
- package/src/components/dynamics/shape-utils.ts +61 -0
- package/src/components/dynamics/shape.ce +89 -0
- package/src/components/dynamics/text.ce +34 -149
- package/src/components/gui/dialogbox/index.ce +15 -6
- package/src/components/player-components-utils.spec.ts +109 -0
- package/src/components/player-components-utils.ts +205 -0
- package/src/components/player-components.ce +221 -0
- package/src/components/scenes/canvas.ce +165 -6
- package/src/components/scenes/draw-map.ce +2 -15
- package/src/components/scenes/event-layer.ce +1 -2
- package/src/core/setup.ts +2 -2
- package/src/decorators/spritesheet.ts +8 -0
- package/src/index.ts +1 -0
- package/src/module.ts +5 -1
- package/src/services/loadMap.ts +2 -0
- package/src/services/mmorpg.ts +8 -2
- package/src/services/save.spec.ts +127 -0
- package/src/utils/getEntityProp.spec.ts +96 -0
- package/src/utils/getEntityProp.ts +4 -3
- package/src/utils/readPropValue.ts +16 -0
- package/dist/node_modules/.pnpm/@signe_di@2.9.0/node_modules/@signe/di/dist/index.js.map +0 -1
- package/dist/node_modules/.pnpm/@signe_reactive@2.8.3/node_modules/@signe/reactive/dist/index.js +0 -457
- package/dist/node_modules/.pnpm/@signe_reactive@2.8.3/node_modules/@signe/reactive/dist/index.js.map +0 -1
- package/dist/node_modules/.pnpm/@signe_reactive@2.9.0/node_modules/@signe/reactive/dist/index.js +0 -463
- package/dist/node_modules/.pnpm/@signe_reactive@2.9.0/node_modules/@signe/reactive/dist/index.js.map +0 -1
- package/dist/node_modules/.pnpm/@signe_room@2.9.0/node_modules/@signe/room/dist/index.js +0 -2191
- package/dist/node_modules/.pnpm/@signe_room@2.9.0/node_modules/@signe/room/dist/index.js.map +0 -1
- package/dist/node_modules/.pnpm/@signe_sync@2.9.0/node_modules/@signe/sync/dist/chunk-7QVYU63E.js +0 -10
- package/dist/node_modules/.pnpm/@signe_sync@2.9.0/node_modules/@signe/sync/dist/chunk-7QVYU63E.js.map +0 -1
- package/dist/node_modules/.pnpm/@signe_sync@2.9.0/node_modules/@signe/sync/dist/client/index.js +0 -91
- package/dist/node_modules/.pnpm/@signe_sync@2.9.0/node_modules/@signe/sync/dist/client/index.js.map +0 -1
- package/dist/node_modules/.pnpm/@signe_sync@2.9.0/node_modules/@signe/sync/dist/index.js.map +0 -1
- package/dist/node_modules/.pnpm/dset@3.1.4/node_modules/dset/dist/index.js +0 -14
- package/dist/node_modules/.pnpm/dset@3.1.4/node_modules/dset/dist/index.js.map +0 -1
|
@@ -1,8 +1,26 @@
|
|
|
1
1
|
<Canvas width={engine.width} height={engine.height}>
|
|
2
2
|
<Viewport worldWidth worldHeight clamp sortableChildren={true}>
|
|
3
|
-
@if (
|
|
4
|
-
<
|
|
3
|
+
@if (lightingAmbientEnabled()) {
|
|
4
|
+
<NightAmbiant
|
|
5
|
+
spots={nightLights}
|
|
6
|
+
darkness={lightingDarknessProps}
|
|
7
|
+
haze={lightingHaze}
|
|
8
|
+
/>
|
|
5
9
|
}
|
|
10
|
+
@if (shadowEnabled()) {
|
|
11
|
+
<SpriteShadows
|
|
12
|
+
lights={shadowLights}
|
|
13
|
+
ambientLight={shadowAmbientLight}
|
|
14
|
+
minInfluence={shadowMinInfluence}
|
|
15
|
+
falloffPower={shadowFalloffPower}
|
|
16
|
+
mode={shadowMode}
|
|
17
|
+
updateHz={shadowUpdateHz}
|
|
18
|
+
scanHz={shadowScanHz}
|
|
19
|
+
cullToViewport={shadowCullToViewport}
|
|
20
|
+
shadowColor={shadowColor}
|
|
21
|
+
/>
|
|
22
|
+
}
|
|
23
|
+
<SceneMap />
|
|
6
24
|
</Viewport>
|
|
7
25
|
@for (gui of guiList) {
|
|
8
26
|
<Container display="flex">
|
|
@@ -18,21 +36,31 @@
|
|
|
18
36
|
</Canvas>
|
|
19
37
|
|
|
20
38
|
<script>
|
|
21
|
-
import {
|
|
39
|
+
import { computed, effect } from "canvasengine";
|
|
22
40
|
import { inject } from "../../core/inject";
|
|
23
41
|
import { RpgClientEngine } from "../../RpgClientEngine";
|
|
24
42
|
import SceneMap from './draw-map.ce'
|
|
25
43
|
import { RpgGui } from "../../Gui/Gui";
|
|
26
44
|
import { delay } from "@rpgjs/common";
|
|
45
|
+
import { NightAmbiant, SpriteShadows } from '@canvasengine/presets'
|
|
27
46
|
|
|
28
47
|
const engine = inject(RpgClientEngine);
|
|
29
48
|
const guiService = inject(RpgGui);
|
|
30
49
|
const sceneData = engine.sceneMap.data
|
|
50
|
+
const lighting = engine.sceneMap.lighting
|
|
31
51
|
const guiList = computed(() => {
|
|
32
52
|
return Object.values(guiService.gui()).filter((gui) => !gui.attachToSprite)
|
|
33
53
|
})
|
|
34
|
-
const worldWidth = computed(() =>
|
|
35
|
-
|
|
54
|
+
const worldWidth = computed(() => {
|
|
55
|
+
const data = sceneData?.()
|
|
56
|
+
const scale = Number(data?.params?.scale ?? 1) || 1
|
|
57
|
+
return Number(data?.width ?? data?.params?.width ?? 2048) * scale
|
|
58
|
+
})
|
|
59
|
+
const worldHeight = computed(() => {
|
|
60
|
+
const data = sceneData?.()
|
|
61
|
+
const scale = Number(data?.params?.scale ?? 1) || 1
|
|
62
|
+
return Number(data?.height ?? data?.params?.height ?? 2048) * scale
|
|
63
|
+
})
|
|
36
64
|
|
|
37
65
|
effect(() => {
|
|
38
66
|
if (sceneData() && !sceneData().component) {
|
|
@@ -49,8 +77,139 @@
|
|
|
49
77
|
const onGuiInteraction = (gui, name, data) => {
|
|
50
78
|
guiService.guiInteraction(gui.name, name, data)
|
|
51
79
|
}
|
|
52
|
-
|
|
80
|
+
|
|
53
81
|
const clamp = {
|
|
54
82
|
direction: "all"
|
|
55
83
|
}
|
|
84
|
+
const NIGHT_SPOT_RADIUS_SCALE = 4.25
|
|
85
|
+
const NIGHT_SPOT_MIN_RADIUS = 170
|
|
86
|
+
const NIGHT_SPOT_MIN_INTENSITY = 1
|
|
87
|
+
const SHADOW_SPOT_RADIUS_SCALE = 12
|
|
88
|
+
const SHADOW_SPOT_MIN_RADIUS = 480
|
|
89
|
+
const SHADOW_SPOT_MIN_INTENSITY = 1.35
|
|
90
|
+
|
|
91
|
+
const nightSpotRadius = (radius) => Math.max(radius * NIGHT_SPOT_RADIUS_SCALE, NIGHT_SPOT_MIN_RADIUS)
|
|
92
|
+
const shadowSpotRadius = (radius) => Math.max(radius * SHADOW_SPOT_RADIUS_SCALE, SHADOW_SPOT_MIN_RADIUS)
|
|
93
|
+
const nightSpotIntensity = (intensity, fallback) => Math.max(intensity ?? fallback, NIGHT_SPOT_MIN_INTENSITY)
|
|
94
|
+
const shadowSpotIntensity = (intensity) => Math.max(intensity ?? 1.3, SHADOW_SPOT_MIN_INTENSITY)
|
|
95
|
+
|
|
96
|
+
const lightingAmbient = computed(() => {
|
|
97
|
+
const state = lighting?.()
|
|
98
|
+
return state?.ambient ?? {}
|
|
99
|
+
})
|
|
100
|
+
|
|
101
|
+
const nightLights = computed(() => {
|
|
102
|
+
const state = lighting?.()
|
|
103
|
+
return (state?.spots ?? []).map((spot, index) => {
|
|
104
|
+
const radius = spot.radius ?? 180
|
|
105
|
+
return {
|
|
106
|
+
x: spot.x,
|
|
107
|
+
y: spot.y,
|
|
108
|
+
radius: nightSpotRadius(radius),
|
|
109
|
+
intensity: nightSpotIntensity(spot.intensity, index === 0 ? 1 : 0.92),
|
|
110
|
+
flicker: spot.flicker,
|
|
111
|
+
flickerSpeed: spot.flickerSpeed ?? 14,
|
|
112
|
+
pulse: spot.pulse,
|
|
113
|
+
pulseSpeed: spot.pulseSpeed,
|
|
114
|
+
phase: spot.phase,
|
|
115
|
+
}
|
|
116
|
+
})
|
|
117
|
+
})
|
|
118
|
+
const hasLightSpots = computed(() => {
|
|
119
|
+
const state = lighting?.()
|
|
120
|
+
return (state?.spots?.length ?? 0) > 0
|
|
121
|
+
})
|
|
122
|
+
|
|
123
|
+
const lightingDarkness = computed(() => {
|
|
124
|
+
const darkness = lightingAmbient().darkness
|
|
125
|
+
return typeof darkness === "number" ? darkness : 0
|
|
126
|
+
})
|
|
127
|
+
|
|
128
|
+
const lightingAmbientEnabled = computed(() => {
|
|
129
|
+
const state = lighting?.()
|
|
130
|
+
return Boolean(state && lightingDarkness() > 0)
|
|
131
|
+
})
|
|
132
|
+
|
|
133
|
+
const lightingDarknessProps = computed(() => ({
|
|
134
|
+
opacity: lightingDarkness(),
|
|
135
|
+
color: lightingAmbient().darkColor ?? "#080a12",
|
|
136
|
+
}))
|
|
137
|
+
const lightingHaze = computed(() => ({
|
|
138
|
+
color: lightingAmbient().fogColor ?? "#12182a",
|
|
139
|
+
radius: lightingAmbient().fogRadius ?? 0.44,
|
|
140
|
+
softness: lightingAmbient().fogSoftness ?? 0.3,
|
|
141
|
+
opacity: lightingAmbient().fogOpacity ?? 0.35,
|
|
142
|
+
}))
|
|
143
|
+
|
|
144
|
+
const defaultSunLight = () => {
|
|
145
|
+
const data = sceneData?.()
|
|
146
|
+
const width = Number(data?.width ?? data?.params?.width ?? 2048)
|
|
147
|
+
const height = Number(data?.height ?? data?.params?.height ?? 2048)
|
|
148
|
+
const scale = Number(data?.params?.scale ?? 1) || 1
|
|
149
|
+
const mapWidth = width * scale
|
|
150
|
+
const mapHeight = height * scale
|
|
151
|
+
|
|
152
|
+
return {
|
|
153
|
+
x: -mapWidth * 0.35,
|
|
154
|
+
y: -mapHeight * 0.45,
|
|
155
|
+
z: 520,
|
|
156
|
+
radius: Math.max(mapWidth, mapHeight) * 2.5,
|
|
157
|
+
intensity: 0.85,
|
|
158
|
+
shadowWeight: lightingDarkness() > 0 ? 2.2 : 1,
|
|
159
|
+
enabled: true,
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
const shadowState = computed(() => {
|
|
164
|
+
const state = lighting?.()
|
|
165
|
+
return state?.shadows ?? null
|
|
166
|
+
})
|
|
167
|
+
|
|
168
|
+
const shadowLights = computed(() => {
|
|
169
|
+
const state = lighting?.()
|
|
170
|
+
const defaultSun = defaultSunLight()
|
|
171
|
+
const sun = {
|
|
172
|
+
...defaultSun,
|
|
173
|
+
...(state?.sun ?? {}),
|
|
174
|
+
shadowWeight: state?.sun?.shadowWeight ?? defaultSun.shadowWeight,
|
|
175
|
+
}
|
|
176
|
+
const spotLights = (state?.spots ?? []).map((spot) => {
|
|
177
|
+
const radius = spot.radius ?? 180
|
|
178
|
+
return {
|
|
179
|
+
x: spot.x,
|
|
180
|
+
y: spot.y,
|
|
181
|
+
z: 170,
|
|
182
|
+
radius: shadowSpotRadius(radius),
|
|
183
|
+
intensity: shadowSpotIntensity(spot.intensity),
|
|
184
|
+
shadowWeight: 2.4,
|
|
185
|
+
enabled: true,
|
|
186
|
+
}
|
|
187
|
+
})
|
|
188
|
+
|
|
189
|
+
return [
|
|
190
|
+
...((sun.enabled === false || sun.intensity <= 0) ? [] : [sun]),
|
|
191
|
+
...spotLights,
|
|
192
|
+
]
|
|
193
|
+
})
|
|
194
|
+
|
|
195
|
+
const shadowAmbientLight = computed(() => {
|
|
196
|
+
const shadows = shadowState()
|
|
197
|
+
if (shadows?.ambientLight === null || shadows?.ambientLight?.enabled === false) {
|
|
198
|
+
return null
|
|
199
|
+
}
|
|
200
|
+
return shadows?.ambientLight ?? { x: -0.18, y: -1, z: 420, intensity: 0.32, shadowWeight: 1 }
|
|
201
|
+
})
|
|
202
|
+
|
|
203
|
+
const shadowEnabled = computed(() => {
|
|
204
|
+
const shadows = shadowState()
|
|
205
|
+
return Boolean((shadows?.enabled || hasLightSpots()) && (shadowLights().length > 0 || shadowAmbientLight()))
|
|
206
|
+
})
|
|
207
|
+
|
|
208
|
+
const shadowMode = computed(() => shadowState()?.mode ?? "strongest")
|
|
209
|
+
const shadowUpdateHz = computed(() => shadowState()?.updateHz ?? 60)
|
|
210
|
+
const shadowScanHz = computed(() => shadowState()?.scanHz ?? 8)
|
|
211
|
+
const shadowCullToViewport = computed(() => shadowState()?.cullToViewport ?? true)
|
|
212
|
+
const shadowMinInfluence = computed(() => shadowState()?.minInfluence ?? 0.24)
|
|
213
|
+
const shadowFalloffPower = computed(() => shadowState()?.falloffPower ?? 0.85)
|
|
214
|
+
const shadowColor = computed(() => shadowState()?.shadowColor ?? "#05070d")
|
|
56
215
|
</script>
|
|
@@ -13,14 +13,13 @@
|
|
|
13
13
|
</Container>
|
|
14
14
|
}
|
|
15
15
|
|
|
16
|
-
@if (weatherProps) {
|
|
16
|
+
@if (weatherProps()) {
|
|
17
17
|
<Weather ...weatherProps() />
|
|
18
18
|
}
|
|
19
19
|
</Container>
|
|
20
20
|
|
|
21
21
|
<script>
|
|
22
|
-
import {
|
|
23
|
-
import { effect, signal, computed, mount, on, tick } from 'canvasengine'
|
|
22
|
+
import { computed } from 'canvasengine'
|
|
24
23
|
import { inject } from "../../core/inject";
|
|
25
24
|
import { RpgClientEngine } from "../../RpgClientEngine";
|
|
26
25
|
import { Weather } from '@canvasengine/presets'
|
|
@@ -29,23 +28,11 @@
|
|
|
29
28
|
const componentAnimations = engine.componentAnimations
|
|
30
29
|
const map = engine.sceneMap?.data
|
|
31
30
|
const sceneComponent = computed(() => map()?.component)
|
|
32
|
-
const sceneParams = map()?.params
|
|
33
31
|
const mapParams = map()?.params
|
|
34
|
-
const animations = engine.sceneMap.animations
|
|
35
32
|
const weather = engine.sceneMap.weather
|
|
36
33
|
const backgroundMusic = { src: mapParams?.backgroundMusic, autoplay: true, loop: true }
|
|
37
34
|
const backgroundAmbientSound = { src: mapParams?.backgroundAmbientSound, autoplay: true, loop: true }
|
|
38
35
|
|
|
39
|
-
const data = signal(map()?.data)
|
|
40
|
-
|
|
41
|
-
const scale = mapParams?.scale || 1
|
|
42
|
-
const worldWidth = (mapParams?.width || 2048) * scale
|
|
43
|
-
const worldHeight = (mapParams?.height || 2048) * scale
|
|
44
|
-
|
|
45
|
-
const clamp = {
|
|
46
|
-
direction: "all"
|
|
47
|
-
}
|
|
48
|
-
|
|
49
36
|
const shakeConfig = {
|
|
50
37
|
trigger: engine.mapShakeTrigger,
|
|
51
38
|
intensity: 10,
|
|
@@ -13,7 +13,6 @@
|
|
|
13
13
|
</Container>
|
|
14
14
|
|
|
15
15
|
<script>
|
|
16
|
-
import { effect, signal } from 'canvasengine'
|
|
17
16
|
import { inject } from "../../core/inject";
|
|
18
17
|
import { RpgClientEngine } from "../../RpgClientEngine";
|
|
19
18
|
import Character from "../character.ce";
|
|
@@ -24,4 +23,4 @@
|
|
|
24
23
|
|
|
25
24
|
const players = engine.sceneMap.players
|
|
26
25
|
const events = engine.sceneMap.events
|
|
27
|
-
</script>
|
|
26
|
+
</script>
|
package/src/core/setup.ts
CHANGED
|
@@ -14,7 +14,7 @@ export async function startGame(options: SetupOptions) {
|
|
|
14
14
|
|
|
15
15
|
await injector(context, options.providers);
|
|
16
16
|
|
|
17
|
-
const engine = inject(context, RpgClientEngine);
|
|
17
|
+
const engine = inject<RpgClientEngine>(context, RpgClientEngine);
|
|
18
18
|
await engine.start();
|
|
19
19
|
return context;
|
|
20
|
-
}
|
|
20
|
+
}
|
package/src/index.ts
CHANGED
|
@@ -16,6 +16,7 @@ export * from "./components";
|
|
|
16
16
|
export * from "./components/gui";
|
|
17
17
|
export * from "./Sound";
|
|
18
18
|
export * from "./Resource";
|
|
19
|
+
export * from "./decorators/spritesheet";
|
|
19
20
|
export * from "./utils/getEntityProp";
|
|
20
21
|
export { Context } from "@signe/di";
|
|
21
22
|
export { KeyboardControls, Input } from "canvasengine";
|
package/src/module.ts
CHANGED
|
@@ -195,6 +195,11 @@ export function provideClientModules(modules: RpgClientModule[]): FactoryProvide
|
|
|
195
195
|
engine.addSpriteComponentInFront(component);
|
|
196
196
|
});
|
|
197
197
|
}
|
|
198
|
+
if (sprite.components) {
|
|
199
|
+
Object.entries(sprite.components).forEach(([id, component]) => {
|
|
200
|
+
engine.registerSpriteComponent(id, component);
|
|
201
|
+
});
|
|
202
|
+
}
|
|
198
203
|
},
|
|
199
204
|
};
|
|
200
205
|
}
|
|
@@ -226,4 +231,3 @@ export function provideClientGlobalConfig(config: any = {}) {
|
|
|
226
231
|
}
|
|
227
232
|
return provideGlobalConfig(config)
|
|
228
233
|
}
|
|
229
|
-
|
package/src/services/loadMap.ts
CHANGED
|
@@ -20,6 +20,8 @@ type MapData = {
|
|
|
20
20
|
height?: number;
|
|
21
21
|
/** Optional map events data (NPCs, interactive objects, etc.) */
|
|
22
22
|
events?: any;
|
|
23
|
+
/** Optional named positions, for example Tiled point objects used by changeMap("map", "name") */
|
|
24
|
+
positions?: Record<string, { x: number; y: number; z?: number }>;
|
|
23
25
|
/** Optional map identifier, defaults to the mapId parameter if not provided */
|
|
24
26
|
id?: string;
|
|
25
27
|
}
|
package/src/services/mmorpg.ts
CHANGED
|
@@ -61,7 +61,10 @@ class BridgeWebsocket extends AbstractWebsocket {
|
|
|
61
61
|
this.socket = await connectionRoom({
|
|
62
62
|
host,
|
|
63
63
|
room: this.targetRoom,
|
|
64
|
-
id: this.privateId
|
|
64
|
+
id: this.privateId,
|
|
65
|
+
query: {
|
|
66
|
+
id: this.privateId,
|
|
67
|
+
},
|
|
65
68
|
}, instance)
|
|
66
69
|
|
|
67
70
|
listeners?.(this.socket)
|
|
@@ -93,7 +96,10 @@ class BridgeWebsocket extends AbstractWebsocket {
|
|
|
93
96
|
room,
|
|
94
97
|
id: this.privateId,
|
|
95
98
|
host: host || this.options.host || window.location.host,
|
|
96
|
-
query
|
|
99
|
+
query: {
|
|
100
|
+
...query,
|
|
101
|
+
id: this.privateId,
|
|
102
|
+
},
|
|
97
103
|
})
|
|
98
104
|
}
|
|
99
105
|
|
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
import { afterEach, beforeEach, describe, expect, test, vi } from "vitest";
|
|
2
|
+
import { SaveClientService, provideSaveClient } from "./save";
|
|
3
|
+
|
|
4
|
+
const createSocket = () => {
|
|
5
|
+
const listeners = new Map<string, (data: any) => void>();
|
|
6
|
+
const emitted: Array<{ event: string; data: any }> = [];
|
|
7
|
+
|
|
8
|
+
return {
|
|
9
|
+
listeners,
|
|
10
|
+
emitted,
|
|
11
|
+
on: vi.fn((event: string, callback: (data: any) => void) => {
|
|
12
|
+
listeners.set(event, callback);
|
|
13
|
+
}),
|
|
14
|
+
off: vi.fn((event: string) => {
|
|
15
|
+
listeners.delete(event);
|
|
16
|
+
}),
|
|
17
|
+
emit: vi.fn((event: string, data: any) => {
|
|
18
|
+
emitted.push({ event, data });
|
|
19
|
+
}),
|
|
20
|
+
};
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
const createService = (socket = createSocket()) => {
|
|
24
|
+
const service = Object.create(SaveClientService.prototype) as SaveClientService;
|
|
25
|
+
(service as any).webSocket = socket;
|
|
26
|
+
(service as any).pending = new Map();
|
|
27
|
+
(service as any).requestCounter = 0;
|
|
28
|
+
return { service, socket };
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
describe("SaveClientService", () => {
|
|
32
|
+
beforeEach(() => {
|
|
33
|
+
vi.useFakeTimers();
|
|
34
|
+
vi.setSystemTime(new Date("2026-01-02T03:04:05.000Z"));
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
afterEach(() => {
|
|
38
|
+
vi.useRealTimers();
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
test("registers save result listeners idempotently", () => {
|
|
42
|
+
const { service, socket } = createService();
|
|
43
|
+
|
|
44
|
+
service.initialize();
|
|
45
|
+
|
|
46
|
+
expect(socket.off).toHaveBeenCalledTimes(4);
|
|
47
|
+
expect(socket.on).toHaveBeenCalledTimes(4);
|
|
48
|
+
expect([...socket.listeners.keys()]).toEqual([
|
|
49
|
+
"save.list.result",
|
|
50
|
+
"save.save.result",
|
|
51
|
+
"save.load.result",
|
|
52
|
+
"save.error",
|
|
53
|
+
]);
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
test("lists, saves and loads slots through request/response events", async () => {
|
|
57
|
+
const { service, socket } = createService();
|
|
58
|
+
service.initialize();
|
|
59
|
+
|
|
60
|
+
const listPromise = service.listSlots();
|
|
61
|
+
expect(socket.emitted[0]).toEqual({
|
|
62
|
+
event: "save.list",
|
|
63
|
+
data: { requestId: "1767323045000-1" },
|
|
64
|
+
});
|
|
65
|
+
socket.listeners.get("save.list.result")?.({
|
|
66
|
+
requestId: "1767323045000-1",
|
|
67
|
+
slots: [{ id: "slot-a" }],
|
|
68
|
+
});
|
|
69
|
+
await expect(listPromise).resolves.toEqual([{ id: "slot-a" }]);
|
|
70
|
+
|
|
71
|
+
const savePromise = service.saveSlot(2, { map: "forest" } as any);
|
|
72
|
+
expect(socket.emitted[1]).toEqual({
|
|
73
|
+
event: "save.save",
|
|
74
|
+
data: {
|
|
75
|
+
requestId: "1767323045000-2",
|
|
76
|
+
index: 2,
|
|
77
|
+
meta: { map: "forest" },
|
|
78
|
+
},
|
|
79
|
+
});
|
|
80
|
+
socket.listeners.get("save.save.result")?.({
|
|
81
|
+
requestId: "1767323045000-2",
|
|
82
|
+
index: 2,
|
|
83
|
+
slots: [null, null, { id: "slot-c" }],
|
|
84
|
+
});
|
|
85
|
+
await expect(savePromise).resolves.toEqual([null, null, { id: "slot-c" }]);
|
|
86
|
+
|
|
87
|
+
const loadPromise = service.loadSlot(2);
|
|
88
|
+
expect(socket.emitted[2]).toEqual({
|
|
89
|
+
event: "save.load",
|
|
90
|
+
data: { requestId: "1767323045000-3", index: 2 },
|
|
91
|
+
});
|
|
92
|
+
socket.listeners.get("save.load.result")?.({
|
|
93
|
+
requestId: "1767323045000-3",
|
|
94
|
+
index: 2,
|
|
95
|
+
ok: true,
|
|
96
|
+
});
|
|
97
|
+
await expect(loadPromise).resolves.toBe(true);
|
|
98
|
+
});
|
|
99
|
+
|
|
100
|
+
test("rejects matching pending requests on save errors and ignores stale responses", async () => {
|
|
101
|
+
const { service, socket } = createService();
|
|
102
|
+
service.initialize();
|
|
103
|
+
|
|
104
|
+
const listPromise = service.listSlots();
|
|
105
|
+
|
|
106
|
+
socket.listeners.get("save.list.result")?.({
|
|
107
|
+
requestId: "unknown",
|
|
108
|
+
slots: [{ id: "stale" }],
|
|
109
|
+
});
|
|
110
|
+
expect((service as any).pending.size).toBe(1);
|
|
111
|
+
|
|
112
|
+
socket.listeners.get("save.error")?.({
|
|
113
|
+
requestId: "1767323045000-1",
|
|
114
|
+
message: "Cannot save",
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
await expect(listPromise).rejects.toThrow("Cannot save");
|
|
118
|
+
expect((service as any).pending.size).toBe(0);
|
|
119
|
+
});
|
|
120
|
+
|
|
121
|
+
test("provides the client save service class", () => {
|
|
122
|
+
expect(provideSaveClient()).toEqual({
|
|
123
|
+
provide: SaveClientService,
|
|
124
|
+
useClass: SaveClientService,
|
|
125
|
+
});
|
|
126
|
+
});
|
|
127
|
+
});
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
import { describe, expect, test } from "vitest";
|
|
2
|
+
import { signal } from "canvasengine";
|
|
3
|
+
import { getEntityProp } from "./getEntityProp";
|
|
4
|
+
|
|
5
|
+
const createEntity = () => {
|
|
6
|
+
const name = signal("Alex");
|
|
7
|
+
|
|
8
|
+
return {
|
|
9
|
+
_level: signal(5),
|
|
10
|
+
_exp: signal(120),
|
|
11
|
+
_gold: signal(30),
|
|
12
|
+
hpSignal: signal(80),
|
|
13
|
+
spSignal: signal(12),
|
|
14
|
+
type: signal("player"),
|
|
15
|
+
x: signal(10),
|
|
16
|
+
y: signal(20),
|
|
17
|
+
z: signal(0),
|
|
18
|
+
tint: signal(0xffffff),
|
|
19
|
+
direction: signal(2),
|
|
20
|
+
hitbox: signal({ w: 32, h: 48 }),
|
|
21
|
+
animationName: signal("idle"),
|
|
22
|
+
graphics: signal(["hero"]),
|
|
23
|
+
items: signal([{ id: "potion" }]),
|
|
24
|
+
equipments: signal([]),
|
|
25
|
+
states: signal([]),
|
|
26
|
+
skills: signal([]),
|
|
27
|
+
_effects: signal([]),
|
|
28
|
+
componentsTop: signal(["nameplate"]),
|
|
29
|
+
componentsBottom: signal([]),
|
|
30
|
+
componentsCenter: signal([]),
|
|
31
|
+
componentsLeft: signal([]),
|
|
32
|
+
componentsRight: signal([]),
|
|
33
|
+
_param: signal({ maxHp: 100, custom: 7 }),
|
|
34
|
+
_canMove: signal(true),
|
|
35
|
+
get name() {
|
|
36
|
+
return name();
|
|
37
|
+
},
|
|
38
|
+
set name(value: string) {
|
|
39
|
+
name.set(value);
|
|
40
|
+
},
|
|
41
|
+
get speed() {
|
|
42
|
+
return 4;
|
|
43
|
+
},
|
|
44
|
+
get canMove() {
|
|
45
|
+
return true;
|
|
46
|
+
},
|
|
47
|
+
} as any;
|
|
48
|
+
};
|
|
49
|
+
|
|
50
|
+
describe("getEntityProp", () => {
|
|
51
|
+
test("returns undefined until a reactive entity becomes available", () => {
|
|
52
|
+
const entity = signal<any>(undefined);
|
|
53
|
+
const hp = getEntityProp(entity, "hp");
|
|
54
|
+
const maxHp = getEntityProp(entity, "params.maxHp");
|
|
55
|
+
|
|
56
|
+
expect(hp()).toBeUndefined();
|
|
57
|
+
expect(maxHp()).toBeUndefined();
|
|
58
|
+
|
|
59
|
+
const current = createEntity();
|
|
60
|
+
entity.set(current);
|
|
61
|
+
|
|
62
|
+
expect(hp()).toBe(80);
|
|
63
|
+
expect(maxHp()).toBe(100);
|
|
64
|
+
|
|
65
|
+
const next = createEntity();
|
|
66
|
+
next.hpSignal.set(42);
|
|
67
|
+
next._param.set({ maxHp: 90 });
|
|
68
|
+
entity.set(next);
|
|
69
|
+
|
|
70
|
+
expect(hp()).toBe(42);
|
|
71
|
+
expect(maxHp()).toBe(90);
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
test("reads built-in, param and unknown keys without throwing", () => {
|
|
75
|
+
const entity = createEntity();
|
|
76
|
+
|
|
77
|
+
expect(getEntityProp(entity, "name")()).toBe("Alex");
|
|
78
|
+
expect(getEntityProp(entity, "componentsTop")()).toEqual(["nameplate"]);
|
|
79
|
+
expect(getEntityProp(entity, "params.custom")()).toBe(7);
|
|
80
|
+
expect(getEntityProp(entity, "params.missing")()).toBeUndefined();
|
|
81
|
+
expect(getEntityProp(entity, "unknown" as any)()).toBeUndefined();
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
test("reads canMove from the synced value when available", () => {
|
|
85
|
+
const entity = createEntity();
|
|
86
|
+
const canMove = getEntityProp(entity, "canMove");
|
|
87
|
+
|
|
88
|
+
expect(canMove()).toBe(true);
|
|
89
|
+
|
|
90
|
+
entity._canMove.set(false);
|
|
91
|
+
expect(canMove()).toBe(false);
|
|
92
|
+
|
|
93
|
+
entity._canMove.set({ value: true } as any);
|
|
94
|
+
expect(canMove()).toBe(true);
|
|
95
|
+
});
|
|
96
|
+
});
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { computed, Signal } from "canvasengine";
|
|
2
2
|
import { RpgClientObject } from "../Game/Object";
|
|
3
|
+
import { readPropValue } from "./readPropValue";
|
|
3
4
|
|
|
4
5
|
const BUILTIN_PARAM_KEYS = [
|
|
5
6
|
"maxHp",
|
|
@@ -22,17 +23,17 @@ const entityPropMap = {
|
|
|
22
23
|
gold: (entity: RpgClientObject) => entity._gold(),
|
|
23
24
|
hp: (entity: RpgClientObject) => entity.hpSignal(),
|
|
24
25
|
sp: (entity: RpgClientObject) => entity.spSignal(),
|
|
25
|
-
name: (entity: RpgClientObject) => entity.name
|
|
26
|
+
name: (entity: RpgClientObject) => entity.name,
|
|
26
27
|
type: (entity: RpgClientObject) => entity.type(),
|
|
27
28
|
x: (entity: RpgClientObject) => entity.x(),
|
|
28
29
|
y: (entity: RpgClientObject) => entity.y(),
|
|
29
30
|
z: (entity: RpgClientObject) => entity.z(),
|
|
30
31
|
tint: (entity: RpgClientObject) => entity.tint(),
|
|
31
32
|
direction: (entity: RpgClientObject) => entity.direction(),
|
|
32
|
-
speed: (entity: RpgClientObject) => entity.speed
|
|
33
|
+
speed: (entity: RpgClientObject) => entity.speed,
|
|
33
34
|
hitbox: (entity: RpgClientObject) => entity.hitbox(),
|
|
34
35
|
animation: (entity: RpgClientObject) => entity.animationName(),
|
|
35
|
-
canMove: (entity: RpgClientObject) => entity.canMove
|
|
36
|
+
canMove: (entity: RpgClientObject) => readPropValue<boolean>((entity as any)._canMove ?? entity.canMove),
|
|
36
37
|
graphics: (entity: RpgClientObject) => entity.graphics(),
|
|
37
38
|
items: (entity: RpgClientObject) => entity.items(),
|
|
38
39
|
equipments: (entity: RpgClientObject) => entity.equipments(),
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
export const readPropValue = <T = unknown>(value: unknown): T => {
|
|
2
|
+
if (typeof value === "function") {
|
|
3
|
+
return readPropValue((value as () => unknown)());
|
|
4
|
+
}
|
|
5
|
+
|
|
6
|
+
if (value && typeof value === "object" && "value" in value) {
|
|
7
|
+
return readPropValue((value as { value: unknown }).value);
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
return value as T;
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
export const getCanMoveValue = (entity: any): boolean => {
|
|
14
|
+
const value = readPropValue(entity?._canMove ?? entity?.canMove);
|
|
15
|
+
return value !== false;
|
|
16
|
+
};
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","names":[],"sources":["../../../../../../../../../../node_modules/.pnpm/@signe+di@2.9.0/node_modules/@signe/di/dist/index.js"],"sourcesContent":["var __defProp = Object.defineProperty;\nvar __name = (target, value) => __defProp(target, \"name\", { value, configurable: true });\n\n// src/inject.ts\nvar DEFAULT_INSTANCE_KEY = \"__default__\";\nfunction toTokenName(token) {\n return typeof token === \"function\" ? token.name : token;\n}\n__name(toTokenName, \"toTokenName\");\nfunction toInstanceKey(name) {\n return name ?? DEFAULT_INSTANCE_KEY;\n}\n__name(toInstanceKey, \"toInstanceKey\");\nfunction getRecord(context, token) {\n return context.get(\"inject:\" + toTokenName(token));\n}\n__name(getRecord, \"getRecord\");\nfunction ensureRecord(context, token) {\n const key = \"inject:\" + toTokenName(token);\n let record = context.get(key);\n if (!record) {\n record = {\n multi: false,\n values: /* @__PURE__ */ new Map(),\n injected: /* @__PURE__ */ new Set()\n };\n }\n context.set(key, record);\n return record;\n}\n__name(ensureRecord, \"ensureRecord\");\nfunction provide(context, token, value, options = {}) {\n const record = ensureRecord(context, token);\n const instanceKey = toInstanceKey(options.name);\n if (options.multi) {\n record.multi = true;\n }\n if (!record.multi && instanceKey !== DEFAULT_INSTANCE_KEY) {\n record.multi = true;\n }\n record.values.set(instanceKey, value);\n return value;\n}\n__name(provide, \"provide\");\nfunction isInjected(context, token, options = {}) {\n const record = getRecord(context, token);\n if (!record) {\n return false;\n }\n if (options.name) {\n return record.injected.has(toInstanceKey(options.name));\n }\n if (record.multi) {\n return record.injected.size > 0;\n }\n return record.injected.has(DEFAULT_INSTANCE_KEY);\n}\n__name(isInjected, \"isInjected\");\nfunction isProvided(context, token, options = {}) {\n const record = getRecord(context, token);\n if (!record) {\n return false;\n }\n if (options.name) {\n return record.values.has(toInstanceKey(options.name));\n }\n if (record.multi) {\n return record.values.size > 0;\n }\n return record.values.has(DEFAULT_INSTANCE_KEY);\n}\n__name(isProvided, \"isProvided\");\nfunction hasInstance(context, token, options = {}) {\n return isProvided(context, token, options);\n}\n__name(hasInstance, \"hasInstance\");\nfunction handleMissingInjection(token, options) {\n const name = toTokenName(token);\n if (options.name) {\n throw new Error(`Injection provider ${name} with name ${options.name} not found`);\n }\n throw new Error(`Injection provider ${name} not found`);\n}\n__name(handleMissingInjection, \"handleMissingInjection\");\nfunction markInjected(record, key) {\n record.injected.add(key);\n}\n__name(markInjected, \"markInjected\");\nfunction markAllInjected(record) {\n for (const key of record.values.keys()) {\n record.injected.add(key);\n }\n}\n__name(markAllInjected, \"markAllInjected\");\nfunction inject(context, token, options = {}) {\n const record = getRecord(context, token);\n if (!record) {\n if (options.optional) {\n return options.multi ? [] : void 0;\n }\n return handleMissingInjection(token, options);\n }\n if (options.name) {\n const instanceKey = toInstanceKey(options.name);\n if (!record.values.has(instanceKey)) {\n if (options.optional) {\n return void 0;\n }\n return handleMissingInjection(token, options);\n }\n const value2 = record.values.get(instanceKey);\n markInjected(record, instanceKey);\n return value2;\n }\n if (options.multi || record.multi) {\n if (record.values.size === 0) {\n if (options.optional) {\n return [];\n }\n return handleMissingInjection(token, options);\n }\n markAllInjected(record);\n return Array.from(record.values.values());\n }\n const value = record.values.get(DEFAULT_INSTANCE_KEY);\n if (value === void 0) {\n if (options.optional) {\n return void 0;\n }\n return handleMissingInjection(token, options);\n }\n markInjected(record, DEFAULT_INSTANCE_KEY);\n return value;\n}\n__name(inject, \"inject\");\nfunction override(providers, newProvider, options) {\n let { upsert = false, key } = options ?? {};\n if (!key) {\n key = typeof newProvider === \"function\" ? newProvider.name : newProvider.provide;\n }\n const flatProviders = providers.flat();\n const exists = flatProviders.some((provider) => {\n if (typeof provider === \"function\") {\n return provider.name === key;\n } else if (typeof provider === \"object\") {\n return provider.provide === key;\n }\n return false;\n });\n const mappedProviders = flatProviders.map((provider) => {\n if (typeof provider === \"function\" && provider.name === key) {\n return newProvider;\n } else if (typeof provider === \"object\" && provider.provide === key) {\n return newProvider;\n }\n return provider;\n });\n if (upsert && !exists) {\n mappedProviders.push(newProvider);\n }\n return mappedProviders;\n}\n__name(override, \"override\");\nfunction findProviders(providers, name) {\n const results = [];\n for (const provider of providers) {\n if (Array.isArray(provider)) {\n results.push(...findProviders(provider, name));\n } else if (findProvider(provider, name)) {\n results.push(provider);\n }\n }\n return results;\n}\n__name(findProviders, \"findProviders\");\nfunction findProvider(providers, name) {\n if (!Array.isArray(providers)) {\n if (typeof providers === \"object\" && \"provide\" in providers) {\n const provider = providers;\n const providerName = typeof provider.provide === \"function\" ? provider.provide.name : provider.provide;\n if (name instanceof RegExp) {\n if (name.test(providerName)) return providers;\n } else {\n if (providerName === name) return providers;\n }\n }\n return null;\n }\n for (const provider of providers) {\n if (Array.isArray(provider)) {\n const found = findProvider(provider, name);\n if (found) return found;\n continue;\n }\n if (typeof provider === \"object\" && \"provide\" in provider) {\n const providerName = typeof provider.provide === \"function\" ? provider.provide.name : provider.provide;\n if (name instanceof RegExp) {\n if (name.test(providerName)) return provider;\n } else {\n if (providerName === name) return provider;\n }\n }\n }\n return null;\n}\n__name(findProvider, \"findProvider\");\n\n// src/merge-config.ts\nfunction processProvider(mergedConfig, baseConfig, provider) {\n if (Array.isArray(provider)) {\n for (const nestedProvider of provider) {\n processProvider(mergedConfig, baseConfig, nestedProvider);\n }\n return;\n }\n const existingProvider = findProvider(baseConfig.providers, provider.provide);\n if (existingProvider) {\n mergedConfig.providers = override(mergedConfig.providers, provider);\n } else {\n mergedConfig.providers.push(provider);\n }\n}\n__name(processProvider, \"processProvider\");\nfunction mergeConfig(baseConfig, config) {\n const mergedConfig = {\n ...baseConfig,\n ...config,\n providers: [\n ...baseConfig.providers\n ]\n // Start with a copy of base providers\n };\n for (const provider of config.providers) {\n processProvider(mergedConfig, baseConfig, provider);\n }\n return mergedConfig;\n}\n__name(mergeConfig, \"mergeConfig\");\n\n// src/provider.ts\nfunction extractProvideOptions(source) {\n if (!source) {\n return void 0;\n }\n const { multi, name } = source;\n if (multi === void 0 && name === void 0) {\n return void 0;\n }\n return {\n multi,\n name\n };\n}\n__name(extractProvideOptions, \"extractProvideOptions\");\nfunction getDeps(provider) {\n if (typeof provider === \"function\") {\n return provider.deps ?? [];\n }\n return provider.deps ?? [];\n}\n__name(getDeps, \"getDeps\");\nfunction sortProviders(providers) {\n const tokenName = /* @__PURE__ */ __name((t) => typeof t === \"function\" ? t.name : t, \"tokenName\");\n const map = /* @__PURE__ */ new Map();\n for (const p of providers) {\n const token = tokenName(typeof p === \"function\" ? p : p.provide);\n const list = map.get(token);\n if (list) {\n list.push(p);\n } else {\n map.set(token, [\n p\n ]);\n }\n }\n const result = [];\n const visited = /* @__PURE__ */ new Set();\n const stack = /* @__PURE__ */ new Set();\n const visit = /* @__PURE__ */ __name((token) => {\n const name = tokenName(token);\n if (visited.has(name)) return;\n if (stack.has(name)) {\n throw new Error(`Circular dependency detected for provider ${name}`);\n }\n stack.add(name);\n const providersForToken = map.get(name);\n if (providersForToken) {\n for (const provider of providersForToken) {\n for (const dep of getDeps(provider)) {\n visit(dep);\n }\n result.push(provider);\n }\n visited.add(name);\n }\n stack.delete(name);\n }, \"visit\");\n for (const p of providers) {\n const token = typeof p === \"function\" ? p : p.provide;\n visit(token);\n }\n return result;\n}\n__name(sortProviders, \"sortProviders\");\nasync function injector(context, providers) {\n providers = providers.flat();\n providers = sortProviders(providers);\n for (const provider of providers) {\n let token;\n let instance;\n let options;\n if (typeof provider === \"function\") {\n token = provider;\n instance = new provider(context);\n const diOptions = extractProvideOptions(provider.diOptions ?? provider.di);\n options = diOptions;\n } else {\n token = provider.provide;\n options = extractProvideOptions(provider);\n const provideUserClass = provider.useClass;\n const isClass = typeof provideUserClass === \"function\";\n if (isClass) {\n instance = new provideUserClass(context);\n } else if (\"useValue\" in provider) {\n instance = provider.useValue;\n } else if (\"useFactory\" in provider) {\n instance = provider.useFactory?.(context);\n if (instance instanceof Promise) {\n instance = await instance;\n }\n } else if (\"useExisting\" in provider) {\n instance = inject(context, provider.useExisting);\n }\n }\n provide(context, token, instance, options);\n }\n}\n__name(injector, \"injector\");\n\n// src/context.ts\nvar Context = class {\n static {\n __name(this, \"Context\");\n }\n /** Internal storage for injected values */\n values = {};\n /**\n * Sets a value in the context\n * @param key - Unique identifier for the value\n * @param value - Value to store\n */\n set(key, value) {\n this.values[key] = value;\n }\n /**\n * Retrieves a value from the context\n * @param key - Unique identifier for the value\n * @returns The stored value or undefined if not found\n */\n get(key) {\n return this.values[key];\n }\n};\nexport {\n Context,\n findProvider,\n findProviders,\n hasInstance,\n inject,\n injector,\n isInjected,\n isProvided,\n mergeConfig,\n override,\n provide\n};\n//# sourceMappingURL=index.js.map"],"x_google_ignoreList":[0],"mappings":";AAAA,IAAI,YAAY,OAAO;AACvB,IAAI,UAAU,QAAQ,UAAU,UAAU,QAAQ,QAAQ;CAAE;CAAO,cAAc;CAAM,CAAC;AAGxF,IAAI,uBAAuB;AAC3B,SAAS,YAAY,OAAO;AAC1B,QAAO,OAAO,UAAU,aAAa,MAAM,OAAO;;AAEpD,OAAO,aAAa,cAAc;AAClC,SAAS,cAAc,MAAM;AAC3B,QAAO,QAAQ;;AAEjB,OAAO,eAAe,gBAAgB;AACtC,SAAS,UAAU,SAAS,OAAO;AACjC,QAAO,QAAQ,IAAI,YAAY,YAAY,MAAM,CAAC;;AAEpD,OAAO,WAAW,YAAY;AAC9B,SAAS,aAAa,SAAS,OAAO;CACpC,MAAM,MAAM,YAAY,YAAY,MAAM;CAC1C,IAAI,SAAS,QAAQ,IAAI,IAAI;AAC7B,KAAI,CAAC,OACH,UAAS;EACP,OAAO;EACP,wBAAwB,IAAI,KAAK;EACjC,0BAA0B,IAAI,KAAK;EACpC;AAEH,SAAQ,IAAI,KAAK,OAAO;AACxB,QAAO;;AAET,OAAO,cAAc,eAAe;AACpC,SAAS,QAAQ,SAAS,OAAO,OAAO,UAAU,EAAE,EAAE;CACpD,MAAM,SAAS,aAAa,SAAS,MAAM;CAC3C,MAAM,cAAc,cAAc,QAAQ,KAAK;AAC/C,KAAI,QAAQ,MACV,QAAO,QAAQ;AAEjB,KAAI,CAAC,OAAO,SAAS,gBAAgB,qBACnC,QAAO,QAAQ;AAEjB,QAAO,OAAO,IAAI,aAAa,MAAM;AACrC,QAAO;;AAET,OAAO,SAAS,UAAU;AAC1B,SAAS,WAAW,SAAS,OAAO,UAAU,EAAE,EAAE;CAChD,MAAM,SAAS,UAAU,SAAS,MAAM;AACxC,KAAI,CAAC,OACH,QAAO;AAET,KAAI,QAAQ,KACV,QAAO,OAAO,SAAS,IAAI,cAAc,QAAQ,KAAK,CAAC;AAEzD,KAAI,OAAO,MACT,QAAO,OAAO,SAAS,OAAO;AAEhC,QAAO,OAAO,SAAS,IAAI,qBAAqB;;AAElD,OAAO,YAAY,aAAa;AAChC,SAAS,WAAW,SAAS,OAAO,UAAU,EAAE,EAAE;CAChD,MAAM,SAAS,UAAU,SAAS,MAAM;AACxC,KAAI,CAAC,OACH,QAAO;AAET,KAAI,QAAQ,KACV,QAAO,OAAO,OAAO,IAAI,cAAc,QAAQ,KAAK,CAAC;AAEvD,KAAI,OAAO,MACT,QAAO,OAAO,OAAO,OAAO;AAE9B,QAAO,OAAO,OAAO,IAAI,qBAAqB;;AAEhD,OAAO,YAAY,aAAa;AAChC,SAAS,YAAY,SAAS,OAAO,UAAU,EAAE,EAAE;AACjD,QAAO,WAAW,SAAS,OAAO,QAAQ;;AAE5C,OAAO,aAAa,cAAc;AAClC,SAAS,uBAAuB,OAAO,SAAS;CAC9C,MAAM,OAAO,YAAY,MAAM;AAC/B,KAAI,QAAQ,KACV,OAAM,IAAI,MAAM,sBAAsB,KAAK,aAAa,QAAQ,KAAK,YAAY;AAEnF,OAAM,IAAI,MAAM,sBAAsB,KAAK,YAAY;;AAEzD,OAAO,wBAAwB,yBAAyB;AACxD,SAAS,aAAa,QAAQ,KAAK;AACjC,QAAO,SAAS,IAAI,IAAI;;AAE1B,OAAO,cAAc,eAAe;AACpC,SAAS,gBAAgB,QAAQ;AAC/B,MAAK,MAAM,OAAO,OAAO,OAAO,MAAM,CACpC,QAAO,SAAS,IAAI,IAAI;;AAG5B,OAAO,iBAAiB,kBAAkB;AAC1C,SAAS,OAAO,SAAS,OAAO,UAAU,EAAE,EAAE;CAC5C,MAAM,SAAS,UAAU,SAAS,MAAM;AACxC,KAAI,CAAC,QAAQ;AACX,MAAI,QAAQ,SACV,QAAO,QAAQ,QAAQ,EAAE,GAAG,KAAK;AAEnC,SAAO,uBAAuB,OAAO,QAAQ;;AAE/C,KAAI,QAAQ,MAAM;EAChB,MAAM,cAAc,cAAc,QAAQ,KAAK;AAC/C,MAAI,CAAC,OAAO,OAAO,IAAI,YAAY,EAAE;AACnC,OAAI,QAAQ,SACV;AAEF,UAAO,uBAAuB,OAAO,QAAQ;;EAE/C,MAAM,SAAS,OAAO,OAAO,IAAI,YAAY;AAC7C,eAAa,QAAQ,YAAY;AACjC,SAAO;;AAET,KAAI,QAAQ,SAAS,OAAO,OAAO;AACjC,MAAI,OAAO,OAAO,SAAS,GAAG;AAC5B,OAAI,QAAQ,SACV,QAAO,EAAE;AAEX,UAAO,uBAAuB,OAAO,QAAQ;;AAE/C,kBAAgB,OAAO;AACvB,SAAO,MAAM,KAAK,OAAO,OAAO,QAAQ,CAAC;;CAE3C,MAAM,QAAQ,OAAO,OAAO,IAAI,qBAAqB;AACrD,KAAI,UAAU,KAAK,GAAG;AACpB,MAAI,QAAQ,SACV;AAEF,SAAO,uBAAuB,OAAO,QAAQ;;AAE/C,cAAa,QAAQ,qBAAqB;AAC1C,QAAO;;AAET,OAAO,QAAQ,SAAS;AACxB,SAAS,SAAS,WAAW,aAAa,SAAS;CACjD,IAAI,EAAE,SAAS,OAAO,QAAQ,WAAW,EAAE;AAC3C,KAAI,CAAC,IACH,OAAM,OAAO,gBAAgB,aAAa,YAAY,OAAO,YAAY;CAE3E,MAAM,gBAAgB,UAAU,MAAM;CACtC,MAAM,SAAS,cAAc,MAAM,aAAa;AAC9C,MAAI,OAAO,aAAa,WACtB,QAAO,SAAS,SAAS;WAChB,OAAO,aAAa,SAC7B,QAAO,SAAS,YAAY;AAE9B,SAAO;GACP;CACF,MAAM,kBAAkB,cAAc,KAAK,aAAa;AACtD,MAAI,OAAO,aAAa,cAAc,SAAS,SAAS,IACtD,QAAO;WACE,OAAO,aAAa,YAAY,SAAS,YAAY,IAC9D,QAAO;AAET,SAAO;GACP;AACF,KAAI,UAAU,CAAC,OACb,iBAAgB,KAAK,YAAY;AAEnC,QAAO;;AAET,OAAO,UAAU,WAAW;AAC5B,SAAS,cAAc,WAAW,MAAM;CACtC,MAAM,UAAU,EAAE;AAClB,MAAK,MAAM,YAAY,UACrB,KAAI,MAAM,QAAQ,SAAS,CACzB,SAAQ,KAAK,GAAG,cAAc,UAAU,KAAK,CAAC;UACrC,aAAa,UAAU,KAAK,CACrC,SAAQ,KAAK,SAAS;AAG1B,QAAO;;AAET,OAAO,eAAe,gBAAgB;AACtC,SAAS,aAAa,WAAW,MAAM;AACrC,KAAI,CAAC,MAAM,QAAQ,UAAU,EAAE;AAC7B,MAAI,OAAO,cAAc,YAAY,aAAa,WAAW;GAC3D,MAAM,WAAW;GACjB,MAAM,eAAe,OAAO,SAAS,YAAY,aAAa,SAAS,QAAQ,OAAO,SAAS;AAC/F,OAAI,gBAAgB;QACd,KAAK,KAAK,aAAa,CAAE,QAAO;cAEhC,iBAAiB,KAAM,QAAO;;AAGtC,SAAO;;AAET,MAAK,MAAM,YAAY,WAAW;AAChC,MAAI,MAAM,QAAQ,SAAS,EAAE;GAC3B,MAAM,QAAQ,aAAa,UAAU,KAAK;AAC1C,OAAI,MAAO,QAAO;AAClB;;AAEF,MAAI,OAAO,aAAa,YAAY,aAAa,UAAU;GACzD,MAAM,eAAe,OAAO,SAAS,YAAY,aAAa,SAAS,QAAQ,OAAO,SAAS;AAC/F,OAAI,gBAAgB;QACd,KAAK,KAAK,aAAa,CAAE,QAAO;cAEhC,iBAAiB,KAAM,QAAO;;;AAIxC,QAAO;;AAET,OAAO,cAAc,eAAe;AAGpC,SAAS,gBAAgB,cAAc,YAAY,UAAU;AAC3D,KAAI,MAAM,QAAQ,SAAS,EAAE;AAC3B,OAAK,MAAM,kBAAkB,SAC3B,iBAAgB,cAAc,YAAY,eAAe;AAE3D;;AAGF,KADyB,aAAa,WAAW,WAAW,SAAS,QAAQ,CAE3E,cAAa,YAAY,SAAS,aAAa,WAAW,SAAS;KAEnE,cAAa,UAAU,KAAK,SAAS;;AAGzC,OAAO,iBAAiB,kBAAkB;AAC1C,SAAS,YAAY,YAAY,QAAQ;CACvC,MAAM,eAAe;EACnB,GAAG;EACH,GAAG;EACH,WAAW,CACT,GAAG,WAAW,UACf;EAEF;AACD,MAAK,MAAM,YAAY,OAAO,UAC5B,iBAAgB,cAAc,YAAY,SAAS;AAErD,QAAO;;AAET,OAAO,aAAa,cAAc;AAGlC,SAAS,sBAAsB,QAAQ;AACrC,KAAI,CAAC,OACH;CAEF,MAAM,EAAE,OAAO,SAAS;AACxB,KAAI,UAAU,KAAK,KAAK,SAAS,KAAK,EACpC;AAEF,QAAO;EACL;EACA;EACD;;AAEH,OAAO,uBAAuB,wBAAwB;AACtD,SAAS,QAAQ,UAAU;AACzB,KAAI,OAAO,aAAa,WACtB,QAAO,SAAS,QAAQ,EAAE;AAE5B,QAAO,SAAS,QAAQ,EAAE;;AAE5B,OAAO,SAAS,UAAU;AAC1B,SAAS,cAAc,WAAW;CAChC,MAAM,YAA4B,wBAAQ,MAAM,OAAO,MAAM,aAAa,EAAE,OAAO,GAAG,YAAY;CAClG,MAAM,sBAAsB,IAAI,KAAK;AACrC,MAAK,MAAM,KAAK,WAAW;EACzB,MAAM,QAAQ,UAAU,OAAO,MAAM,aAAa,IAAI,EAAE,QAAQ;EAChE,MAAM,OAAO,IAAI,IAAI,MAAM;AAC3B,MAAI,KACF,MAAK,KAAK,EAAE;MAEZ,KAAI,IAAI,OAAO,CACb,EACD,CAAC;;CAGN,MAAM,SAAS,EAAE;CACjB,MAAM,0BAA0B,IAAI,KAAK;CACzC,MAAM,wBAAwB,IAAI,KAAK;CACvC,MAAM,QAAwB,wBAAQ,UAAU;EAC9C,MAAM,OAAO,UAAU,MAAM;AAC7B,MAAI,QAAQ,IAAI,KAAK,CAAE;AACvB,MAAI,MAAM,IAAI,KAAK,CACjB,OAAM,IAAI,MAAM,6CAA6C,OAAO;AAEtE,QAAM,IAAI,KAAK;EACf,MAAM,oBAAoB,IAAI,IAAI,KAAK;AACvC,MAAI,mBAAmB;AACrB,QAAK,MAAM,YAAY,mBAAmB;AACxC,SAAK,MAAM,OAAO,QAAQ,SAAS,CACjC,OAAM,IAAI;AAEZ,WAAO,KAAK,SAAS;;AAEvB,WAAQ,IAAI,KAAK;;AAEnB,QAAM,OAAO,KAAK;IACjB,QAAQ;AACX,MAAK,MAAM,KAAK,UAEd,OADc,OAAO,MAAM,aAAa,IAAI,EAAE,QAClC;AAEd,QAAO;;AAET,OAAO,eAAe,gBAAgB;AACtC,eAAe,SAAS,SAAS,WAAW;AAC1C,aAAY,UAAU,MAAM;AAC5B,aAAY,cAAc,UAAU;AACpC,MAAK,MAAM,YAAY,WAAW;EAChC,IAAI;EACJ,IAAI;EACJ,IAAI;AACJ,MAAI,OAAO,aAAa,YAAY;AAClC,WAAQ;AACR,cAAW,IAAI,SAAS,QAAQ;AAEhC,aADkB,sBAAsB,SAAS,aAAa,SAAS,GAAG;SAErE;AACL,WAAQ,SAAS;AACjB,aAAU,sBAAsB,SAAS;GACzC,MAAM,mBAAmB,SAAS;AAElC,OADgB,OAAO,qBAAqB,WAE1C,YAAW,IAAI,iBAAiB,QAAQ;YAC/B,cAAc,SACvB,YAAW,SAAS;YACX,gBAAgB,UAAU;AACnC,eAAW,SAAS,aAAa,QAAQ;AACzC,QAAI,oBAAoB,QACtB,YAAW,MAAM;cAEV,iBAAiB,SAC1B,YAAW,OAAO,SAAS,SAAS,YAAY;;AAGpD,UAAQ,SAAS,OAAO,UAAU,QAAQ;;;AAG9C,OAAO,UAAU,WAAW;AAG5B,IAAI,UAAU,MAAM;CAClB;AACE,SAAO,MAAM,UAAU;;;CAGzB,SAAS,EAAE;;;;;;CAMX,IAAI,KAAK,OAAO;AACd,OAAK,OAAO,OAAO;;;;;;;CAOrB,IAAI,KAAK;AACP,SAAO,KAAK,OAAO"}
|