@nice2dev/game-engine 1.0.5 → 1.0.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.
Files changed (721) hide show
  1. package/dist/cjs/accessibility/Accessibility.js +55 -20
  2. package/dist/cjs/accessibility/Accessibility.js.map +1 -1
  3. package/dist/cjs/ai/AI.js +74 -40
  4. package/dist/cjs/ai/AI.js.map +1 -1
  5. package/dist/cjs/ai/BehaviorTree.js +39 -47
  6. package/dist/cjs/ai/BehaviorTree.js.map +1 -1
  7. package/dist/cjs/ai/StateMachine.js +64 -100
  8. package/dist/cjs/ai/StateMachine.js.map +1 -1
  9. package/dist/cjs/animation/Animation.js +58 -34
  10. package/dist/cjs/animation/Animation.js.map +1 -1
  11. package/dist/cjs/audio/AudioAdvanced.js +71 -37
  12. package/dist/cjs/audio/AudioAdvanced.js.map +1 -1
  13. package/dist/cjs/audio/AudioBridge.js +61 -49
  14. package/dist/cjs/audio/AudioBridge.js.map +1 -1
  15. package/dist/cjs/audio/AudioManager.js +45 -29
  16. package/dist/cjs/audio/AudioManager.js.map +1 -1
  17. package/dist/cjs/build-cloud/BuildOrchestration.js +1142 -0
  18. package/dist/cjs/build-cloud/BuildOrchestration.js.map +1 -0
  19. package/dist/cjs/build-cloud/GodotBuildCloud.js +722 -0
  20. package/dist/cjs/build-cloud/GodotBuildCloud.js.map +1 -0
  21. package/dist/cjs/build-cloud/UnityBuildCloud.js +652 -0
  22. package/dist/cjs/build-cloud/UnityBuildCloud.js.map +1 -0
  23. package/dist/cjs/build-cloud/UnrealBuildCloud.js +674 -0
  24. package/dist/cjs/build-cloud/UnrealBuildCloud.js.map +1 -0
  25. package/dist/cjs/core/EventBus.js +2 -1
  26. package/dist/cjs/core/EventBus.js.map +1 -1
  27. package/dist/cjs/core/GameConfig.js +1 -1
  28. package/dist/cjs/core/GameConfig.js.map +1 -1
  29. package/dist/cjs/core/GameLoop.js +6 -3
  30. package/dist/cjs/core/GameLoop.js.map +1 -1
  31. package/dist/cjs/core/ServiceLocator.js +1 -2
  32. package/dist/cjs/core/ServiceLocator.js.map +1 -1
  33. package/dist/cjs/devtools/DevTools.js +73 -43
  34. package/dist/cjs/devtools/DevTools.js.map +1 -1
  35. package/dist/cjs/devtools/DeveloperExperience.js +84 -42
  36. package/dist/cjs/devtools/DeveloperExperience.js.map +1 -1
  37. package/dist/cjs/devtools/GameplayAnalytics.js +71 -43
  38. package/dist/cjs/devtools/GameplayAnalytics.js.map +1 -1
  39. package/dist/cjs/dialogue/DialogueSystem.js +153 -129
  40. package/dist/cjs/dialogue/DialogueSystem.js.map +1 -1
  41. package/dist/cjs/docs/DocGenerator.js +70 -28
  42. package/dist/cjs/docs/DocGenerator.js.map +1 -1
  43. package/dist/cjs/ecs/World.js +159 -33
  44. package/dist/cjs/ecs/World.js.map +1 -1
  45. package/dist/cjs/editor/AchievementEditor.js +650 -0
  46. package/dist/cjs/editor/AchievementEditor.js.map +1 -0
  47. package/dist/cjs/editor/AdvancedEditor.js +95 -23
  48. package/dist/cjs/editor/AdvancedEditor.js.map +1 -1
  49. package/dist/cjs/editor/AnimationRetargeting.js +742 -0
  50. package/dist/cjs/editor/AnimationRetargeting.js.map +1 -0
  51. package/dist/cjs/editor/AssetBrowser.js +1191 -0
  52. package/dist/cjs/editor/AssetBrowser.js.map +1 -0
  53. package/dist/cjs/editor/AssetManager.js +22 -14
  54. package/dist/cjs/editor/AssetManager.js.map +1 -1
  55. package/dist/cjs/editor/ConsolePro.js +1092 -0
  56. package/dist/cjs/editor/ConsolePro.js.map +1 -0
  57. package/dist/cjs/editor/CraftingEditor.js +749 -0
  58. package/dist/cjs/editor/CraftingEditor.js.map +1 -0
  59. package/dist/cjs/editor/DebugTools.js +46 -23
  60. package/dist/cjs/editor/DebugTools.js.map +1 -1
  61. package/dist/cjs/editor/DialogueSystem.js +924 -0
  62. package/dist/cjs/editor/DialogueSystem.js.map +1 -0
  63. package/dist/cjs/editor/LeaderboardSystem.js +496 -0
  64. package/dist/cjs/editor/LeaderboardSystem.js.map +1 -0
  65. package/dist/cjs/editor/NiceGameEditor.js +532 -142
  66. package/dist/cjs/editor/NiceGameEditor.js.map +1 -1
  67. package/dist/cjs/editor/PrefabSystem.js +864 -0
  68. package/dist/cjs/editor/PrefabSystem.js.map +1 -0
  69. package/dist/cjs/editor/ProfilerPanel.js +716 -0
  70. package/dist/cjs/editor/ProfilerPanel.js.map +1 -0
  71. package/dist/cjs/editor/QuestEditor.js +850 -0
  72. package/dist/cjs/editor/QuestEditor.js.map +1 -0
  73. package/dist/cjs/editor/ReplaySystem.js +655 -0
  74. package/dist/cjs/editor/ReplaySystem.js.map +1 -0
  75. package/dist/cjs/editor/SceneEditor.js +33 -13
  76. package/dist/cjs/editor/SceneEditor.js.map +1 -1
  77. package/dist/cjs/editor/ScreenshotMode.js +710 -0
  78. package/dist/cjs/editor/ScreenshotMode.js.map +1 -0
  79. package/dist/cjs/editor/ShaderGraph.js +7 -6
  80. package/dist/cjs/editor/ShaderGraph.js.map +1 -1
  81. package/dist/cjs/editor/TimelineEditor.js +36 -30
  82. package/dist/cjs/editor/TimelineEditor.js.map +1 -1
  83. package/dist/cjs/editor/UndoRedoPro.js +920 -0
  84. package/dist/cjs/editor/UndoRedoPro.js.map +1 -0
  85. package/dist/cjs/editor/VRARMode.js +658 -0
  86. package/dist/cjs/editor/VRARMode.js.map +1 -0
  87. package/dist/cjs/engine/NiceGameEngine.js +12 -10
  88. package/dist/cjs/engine/NiceGameEngine.js.map +1 -1
  89. package/dist/cjs/engine/SaveSystemV2.js +73 -37
  90. package/dist/cjs/engine/SaveSystemV2.js.map +1 -1
  91. package/dist/cjs/enterprise/Enterprise.js +92 -44
  92. package/dist/cjs/enterprise/Enterprise.js.map +1 -1
  93. package/dist/cjs/export/BevyExporter.js +998 -0
  94. package/dist/cjs/export/BevyExporter.js.map +1 -0
  95. package/dist/cjs/export/CocosExporter.js +706 -0
  96. package/dist/cjs/export/CocosExporter.js.map +1 -0
  97. package/dist/cjs/export/Construct3Exporter.js +832 -0
  98. package/dist/cjs/export/Construct3Exporter.js.map +1 -0
  99. package/dist/cjs/export/DefoldExporter.js +1106 -0
  100. package/dist/cjs/export/DefoldExporter.js.map +1 -0
  101. package/dist/cjs/export/GDevelopExporter.js +748 -0
  102. package/dist/cjs/export/GDevelopExporter.js.map +1 -0
  103. package/dist/cjs/export/GameMakerExporter.js +846 -0
  104. package/dist/cjs/export/GameMakerExporter.js.map +1 -0
  105. package/dist/cjs/export/GodotExporter.js +45 -50
  106. package/dist/cjs/export/GodotExporter.js.map +1 -1
  107. package/dist/cjs/export/MinecraftBedrockExporter.js +606 -0
  108. package/dist/cjs/export/MinecraftBedrockExporter.js.map +1 -0
  109. package/dist/cjs/export/MonoGameExporter.js +1334 -0
  110. package/dist/cjs/export/MonoGameExporter.js.map +1 -0
  111. package/dist/cjs/export/Pico8Exporter.js +846 -0
  112. package/dist/cjs/export/Pico8Exporter.js.map +1 -0
  113. package/dist/cjs/export/PlatformExporter.js +4 -5
  114. package/dist/cjs/export/PlatformExporter.js.map +1 -1
  115. package/dist/cjs/export/RPGMakerExporter.js +906 -0
  116. package/dist/cjs/export/RPGMakerExporter.js.map +1 -0
  117. package/dist/cjs/export/RobloxExporter.js +943 -0
  118. package/dist/cjs/export/RobloxExporter.js.map +1 -0
  119. package/dist/cjs/export/Solar2DExporter.js +1283 -0
  120. package/dist/cjs/export/Solar2DExporter.js.map +1 -0
  121. package/dist/cjs/export/ThreeJSExporter.js +52 -55
  122. package/dist/cjs/export/ThreeJSExporter.js.map +1 -1
  123. package/dist/cjs/export/Tic80Exporter.js +1315 -0
  124. package/dist/cjs/export/Tic80Exporter.js.map +1 -0
  125. package/dist/cjs/export/UnityExporter.js +41 -56
  126. package/dist/cjs/export/UnityExporter.js.map +1 -1
  127. package/dist/cjs/export/UnrealExporter.js +962 -0
  128. package/dist/cjs/export/UnrealExporter.js.map +1 -0
  129. package/dist/cjs/export/WebExporter.js +42 -50
  130. package/dist/cjs/export/WebExporter.js.map +1 -1
  131. package/dist/cjs/export/index.js +50 -2
  132. package/dist/cjs/export/index.js.map +1 -1
  133. package/dist/cjs/godot-integration/GodotAssetLibrary.js +716 -0
  134. package/dist/cjs/godot-integration/GodotAssetLibrary.js.map +1 -0
  135. package/dist/cjs/godot-integration/GodotImport.js +1069 -0
  136. package/dist/cjs/godot-integration/GodotImport.js.map +1 -0
  137. package/dist/cjs/godot-integration/GodotLiveLink.js +962 -0
  138. package/dist/cjs/godot-integration/GodotLiveLink.js.map +1 -0
  139. package/dist/cjs/godot-integration/components/GodotProjectBrowser.js +668 -0
  140. package/dist/cjs/godot-integration/components/GodotProjectBrowser.js.map +1 -0
  141. package/dist/cjs/i18n/I18n.js +56 -30
  142. package/dist/cjs/i18n/I18n.js.map +1 -1
  143. package/dist/cjs/i18n/useTranslation.js +4 -4
  144. package/dist/cjs/i18n/useTranslation.js.map +1 -1
  145. package/dist/cjs/import/AsepriteImporter.js +15 -19
  146. package/dist/cjs/import/AsepriteImporter.js.map +1 -1
  147. package/dist/cjs/import/DragonBonesImporter.js +36 -40
  148. package/dist/cjs/import/DragonBonesImporter.js.map +1 -1
  149. package/dist/cjs/import/GameMakerImporter.js +11 -17
  150. package/dist/cjs/import/GameMakerImporter.js.map +1 -1
  151. package/dist/cjs/import/GodotSceneImporter.js +43 -29
  152. package/dist/cjs/import/GodotSceneImporter.js.map +1 -1
  153. package/dist/cjs/import/LDtkImporter.js +16 -17
  154. package/dist/cjs/import/LDtkImporter.js.map +1 -1
  155. package/dist/cjs/import/Live2DImporter.js +16 -19
  156. package/dist/cjs/import/Live2DImporter.js.map +1 -1
  157. package/dist/cjs/import/NdgFormat.js +28 -35
  158. package/dist/cjs/import/NdgFormat.js.map +1 -1
  159. package/dist/cjs/import/OgmoImporter.js +16 -18
  160. package/dist/cjs/import/OgmoImporter.js.map +1 -1
  161. package/dist/cjs/import/RPGMakerImporter.js +5 -6
  162. package/dist/cjs/import/RPGMakerImporter.js.map +1 -1
  163. package/dist/cjs/import/SceneImporter.js +23 -13
  164. package/dist/cjs/import/SceneImporter.js.map +1 -1
  165. package/dist/cjs/import/SpineImporter.js +20 -20
  166. package/dist/cjs/import/SpineImporter.js.map +1 -1
  167. package/dist/cjs/import/SpriterImporter.js +38 -46
  168. package/dist/cjs/import/SpriterImporter.js.map +1 -1
  169. package/dist/cjs/import/TiledMapImporter.js +20 -24
  170. package/dist/cjs/import/TiledMapImporter.js.map +1 -1
  171. package/dist/cjs/import/UnitySceneImporter.js +15 -15
  172. package/dist/cjs/import/UnitySceneImporter.js.map +1 -1
  173. package/dist/cjs/index.js +516 -4
  174. package/dist/cjs/index.js.map +1 -1
  175. package/dist/cjs/input/GamepadNavigation.js +82 -47
  176. package/dist/cjs/input/GamepadNavigation.js.map +1 -1
  177. package/dist/cjs/input/InputManager.js +107 -52
  178. package/dist/cjs/input/InputManager.js.map +1 -1
  179. package/dist/cjs/input/useGamepads.js +35 -29
  180. package/dist/cjs/input/useGamepads.js.map +1 -1
  181. package/dist/cjs/integration/IconSprite.js +39 -21
  182. package/dist/cjs/integration/IconSprite.js.map +1 -1
  183. package/dist/cjs/inventory/InventorySystem.js +218 -127
  184. package/dist/cjs/inventory/InventorySystem.js.map +1 -1
  185. package/dist/cjs/kids/KidMode.js +491 -78
  186. package/dist/cjs/kids/KidMode.js.map +1 -1
  187. package/dist/cjs/kids/KidTools.js +154 -48
  188. package/dist/cjs/kids/KidTools.js.map +1 -1
  189. package/dist/cjs/monetization/Monetization.js +47 -35
  190. package/dist/cjs/monetization/Monetization.js.map +1 -1
  191. package/dist/cjs/multiplayer/LocalMultiplayer.js +48 -26
  192. package/dist/cjs/multiplayer/LocalMultiplayer.js.map +1 -1
  193. package/dist/cjs/multiplayer/MiniGameTypes.js +23 -10
  194. package/dist/cjs/multiplayer/MiniGameTypes.js.map +1 -1
  195. package/dist/cjs/native-engine/AssetPipeline.js +540 -0
  196. package/dist/cjs/native-engine/AssetPipeline.js.map +1 -0
  197. package/dist/cjs/native-engine/AudioEngine.js +825 -0
  198. package/dist/cjs/native-engine/AudioEngine.js.map +1 -0
  199. package/dist/cjs/native-engine/InputSystem.js +1175 -0
  200. package/dist/cjs/native-engine/InputSystem.js.map +1 -0
  201. package/dist/cjs/native-engine/NetworkingSystem.js +825 -0
  202. package/dist/cjs/native-engine/NetworkingSystem.js.map +1 -0
  203. package/dist/cjs/native-engine/PhysicsEngine.js +622 -0
  204. package/dist/cjs/native-engine/PhysicsEngine.js.map +1 -0
  205. package/dist/cjs/native-engine/RenderingEngine.js +371 -0
  206. package/dist/cjs/native-engine/RenderingEngine.js.map +1 -0
  207. package/dist/cjs/native-engine/SceneGraph.js +862 -0
  208. package/dist/cjs/native-engine/SceneGraph.js.map +1 -0
  209. package/dist/cjs/network/MultiplayerTransport.js +63 -33
  210. package/dist/cjs/network/MultiplayerTransport.js.map +1 -1
  211. package/dist/cjs/network/Networking.js +87 -62
  212. package/dist/cjs/network/Networking.js.map +1 -1
  213. package/dist/cjs/pathfinding/Pathfinding.js +71 -26
  214. package/dist/cjs/pathfinding/Pathfinding.js.map +1 -1
  215. package/dist/cjs/performance/AssetStreaming.js +656 -0
  216. package/dist/cjs/performance/AssetStreaming.js.map +1 -0
  217. package/dist/cjs/performance/BenchmarkSuite.js +37 -24
  218. package/dist/cjs/performance/BenchmarkSuite.js.map +1 -1
  219. package/dist/cjs/performance/DeltaCompression.js +566 -0
  220. package/dist/cjs/performance/DeltaCompression.js.map +1 -0
  221. package/dist/cjs/performance/DeterministicPhysics.js +603 -0
  222. package/dist/cjs/performance/DeterministicPhysics.js.map +1 -0
  223. package/dist/cjs/performance/ECSOptimization.js +726 -0
  224. package/dist/cjs/performance/ECSOptimization.js.map +1 -0
  225. package/dist/cjs/performance/HotReload.js +525 -0
  226. package/dist/cjs/performance/HotReload.js.map +1 -0
  227. package/dist/cjs/performance/InputRecording.js +590 -0
  228. package/dist/cjs/performance/InputRecording.js.map +1 -0
  229. package/dist/cjs/performance/InterestManagement.js +532 -0
  230. package/dist/cjs/performance/InterestManagement.js.map +1 -0
  231. package/dist/cjs/performance/LocalizationRuntime.js +472 -0
  232. package/dist/cjs/performance/LocalizationRuntime.js.map +1 -0
  233. package/dist/cjs/performance/MemoryManagement.js +487 -0
  234. package/dist/cjs/performance/MemoryManagement.js.map +1 -0
  235. package/dist/cjs/performance/ModSupport.js +531 -0
  236. package/dist/cjs/performance/ModSupport.js.map +1 -0
  237. package/dist/cjs/performance/MultiThreadedPhysics.js +500 -0
  238. package/dist/cjs/performance/MultiThreadedPhysics.js.map +1 -0
  239. package/dist/cjs/performance/NetworkPrediction.js +521 -0
  240. package/dist/cjs/performance/NetworkPrediction.js.map +1 -0
  241. package/dist/cjs/performance/ObjectPooling.js +653 -0
  242. package/dist/cjs/performance/ObjectPooling.js.map +1 -0
  243. package/dist/cjs/performance/Performance.js +64 -25
  244. package/dist/cjs/performance/Performance.js.map +1 -1
  245. package/dist/cjs/performance/PerformanceAdvanced.js +83 -49
  246. package/dist/cjs/performance/PerformanceAdvanced.js.map +1 -1
  247. package/dist/cjs/performance/SaveSystem.js +574 -0
  248. package/dist/cjs/performance/SaveSystem.js.map +1 -0
  249. package/dist/cjs/performance/WebGPURenderer.js +702 -0
  250. package/dist/cjs/performance/WebGPURenderer.js.map +1 -0
  251. package/dist/cjs/physics/PhysicsAdvanced.js +32 -20
  252. package/dist/cjs/physics/PhysicsAdvanced.js.map +1 -1
  253. package/dist/cjs/physics/PhysicsEngine2D.js +63 -36
  254. package/dist/cjs/physics/PhysicsEngine2D.js.map +1 -1
  255. package/dist/cjs/plugins/PluginSDK.js +46 -34
  256. package/dist/cjs/plugins/PluginSDK.js.map +1 -1
  257. package/dist/cjs/plugins/PluginTestKit.js +28 -29
  258. package/dist/cjs/plugins/PluginTestKit.js.map +1 -1
  259. package/dist/cjs/procedural/Procedural.js +24 -17
  260. package/dist/cjs/procedural/Procedural.js.map +1 -1
  261. package/dist/cjs/procedural/WorldBuilding.js +187 -55
  262. package/dist/cjs/procedural/WorldBuilding.js.map +1 -1
  263. package/dist/cjs/quest/QuestSystem.js +145 -111
  264. package/dist/cjs/quest/QuestSystem.js.map +1 -1
  265. package/dist/cjs/release/Release.js +142 -31
  266. package/dist/cjs/release/Release.js.map +1 -1
  267. package/dist/cjs/rendering/Camera2D.js +12 -7
  268. package/dist/cjs/rendering/Camera2D.js.map +1 -1
  269. package/dist/cjs/rendering/Renderer2D.js +62 -26
  270. package/dist/cjs/rendering/Renderer2D.js.map +1 -1
  271. package/dist/cjs/rendering/WebGPURenderPipeline.js +60 -31
  272. package/dist/cjs/rendering/WebGPURenderPipeline.js.map +1 -1
  273. package/dist/cjs/runtime3d/Runtime3D.js +80 -55
  274. package/dist/cjs/runtime3d/Runtime3D.js.map +1 -1
  275. package/dist/cjs/runtime3d/SceneEditor3D.js +20 -11
  276. package/dist/cjs/runtime3d/SceneEditor3D.js.map +1 -1
  277. package/dist/cjs/scene/SceneManager.js +26 -12
  278. package/dist/cjs/scene/SceneManager.js.map +1 -1
  279. package/dist/cjs/scripting/GraphToAST.js +36 -24
  280. package/dist/cjs/scripting/GraphToAST.js.map +1 -1
  281. package/dist/cjs/scripting/LanguageExporter.js +2 -3
  282. package/dist/cjs/scripting/LanguageExporter.js.map +1 -1
  283. package/dist/cjs/scripting/NodeGraph.js +49 -32
  284. package/dist/cjs/scripting/NodeGraph.js.map +1 -1
  285. package/dist/cjs/scripting/ScriptAST.js +19 -6
  286. package/dist/cjs/scripting/ScriptAST.js.map +1 -1
  287. package/dist/cjs/scripting/VisualScripting2.js +13 -12
  288. package/dist/cjs/scripting/VisualScripting2.js.map +1 -1
  289. package/dist/cjs/scripting/exporters/CSharpExporter.js +16 -16
  290. package/dist/cjs/scripting/exporters/CSharpExporter.js.map +1 -1
  291. package/dist/cjs/scripting/exporters/GDScriptExporter.js +3 -6
  292. package/dist/cjs/scripting/exporters/GDScriptExporter.js.map +1 -1
  293. package/dist/cjs/scripting/exporters/LuaExporter.js +5 -9
  294. package/dist/cjs/scripting/exporters/LuaExporter.js.map +1 -1
  295. package/dist/cjs/scripting/exporters/PythonExporter.js +7 -13
  296. package/dist/cjs/scripting/exporters/PythonExporter.js.map +1 -1
  297. package/dist/cjs/scripting/exporters/RustExporter.js +7 -14
  298. package/dist/cjs/scripting/exporters/RustExporter.js.map +1 -1
  299. package/dist/cjs/scripting/exporters/TypeScriptExporter.js +28 -29
  300. package/dist/cjs/scripting/exporters/TypeScriptExporter.js.map +1 -1
  301. package/dist/cjs/social/Social.js +27 -20
  302. package/dist/cjs/social/Social.js.map +1 -1
  303. package/dist/cjs/systems/LightingSystem.js +980 -0
  304. package/dist/cjs/systems/LightingSystem.js.map +1 -0
  305. package/dist/cjs/systems/ParticleSystem2.js +34 -17
  306. package/dist/cjs/systems/ParticleSystem2.js.map +1 -1
  307. package/dist/cjs/systems/PhysicsDebugSystem.js +1228 -0
  308. package/dist/cjs/systems/PhysicsDebugSystem.js.map +1 -0
  309. package/dist/cjs/systems/TerrainSystem.js +1137 -0
  310. package/dist/cjs/systems/TerrainSystem.js.map +1 -0
  311. package/dist/cjs/templates/ActionTemplates.js +144 -24
  312. package/dist/cjs/templates/ActionTemplates.js.map +1 -1
  313. package/dist/cjs/templates/FightingTemplate.js +536 -0
  314. package/dist/cjs/templates/FightingTemplate.js.map +1 -0
  315. package/dist/cjs/templates/MetroidvaniaTemplate.js +523 -0
  316. package/dist/cjs/templates/MetroidvaniaTemplate.js.map +1 -0
  317. package/dist/cjs/templates/PartyTemplates.js +60 -28
  318. package/dist/cjs/templates/PartyTemplates.js.map +1 -1
  319. package/dist/cjs/templates/PuzzleTemplates.js +63 -22
  320. package/dist/cjs/templates/PuzzleTemplates.js.map +1 -1
  321. package/dist/cjs/templates/RPGTemplates.js +29 -15
  322. package/dist/cjs/templates/RPGTemplates.js.map +1 -1
  323. package/dist/cjs/templates/SportsTemplates.js +41 -17
  324. package/dist/cjs/templates/SportsTemplates.js.map +1 -1
  325. package/dist/cjs/templates/StrategyTemplates.js +45 -18
  326. package/dist/cjs/templates/StrategyTemplates.js.map +1 -1
  327. package/dist/cjs/templates/SurvivalTemplate.js +479 -0
  328. package/dist/cjs/templates/SurvivalTemplate.js.map +1 -0
  329. package/dist/cjs/templates/WaveDefense.js +139 -52
  330. package/dist/cjs/templates/WaveDefense.js.map +1 -1
  331. package/dist/cjs/tilemap/Tilemap.js +48 -32
  332. package/dist/cjs/tilemap/Tilemap.js.map +1 -1
  333. package/dist/cjs/tutorials/TutorialBranching.js +111 -42
  334. package/dist/cjs/tutorials/TutorialBranching.js.map +1 -1
  335. package/dist/cjs/tutorials/TutorialEngine.js +152 -80
  336. package/dist/cjs/tutorials/TutorialEngine.js.map +1 -1
  337. package/dist/cjs/tutorials/TutorialOverlay.js +117 -40
  338. package/dist/cjs/tutorials/TutorialOverlay.js.map +1 -1
  339. package/dist/cjs/ui/GameUI.js +69 -42
  340. package/dist/cjs/ui/GameUI.js.map +1 -1
  341. package/dist/cjs/unity-integration/UnityAssetStore.js +754 -0
  342. package/dist/cjs/unity-integration/UnityAssetStore.js.map +1 -0
  343. package/dist/cjs/unity-integration/UnityImport.js +1252 -0
  344. package/dist/cjs/unity-integration/UnityImport.js.map +1 -0
  345. package/dist/cjs/unity-integration/UnityLiveLink.js +1022 -0
  346. package/dist/cjs/unity-integration/UnityLiveLink.js.map +1 -0
  347. package/dist/cjs/unity-integration/components/UnityAssetStoreBrowser.js +796 -0
  348. package/dist/cjs/unity-integration/components/UnityAssetStoreBrowser.js.map +1 -0
  349. package/dist/cjs/unity-integration/components/UnityProjectBrowser.js +833 -0
  350. package/dist/cjs/unity-integration/components/UnityProjectBrowser.js.map +1 -0
  351. package/dist/cjs/unreal-integration/UnrealImport.js +442 -0
  352. package/dist/cjs/unreal-integration/UnrealImport.js.map +1 -0
  353. package/dist/cjs/unreal-integration/UnrealLiveLink.js +1186 -0
  354. package/dist/cjs/unreal-integration/UnrealLiveLink.js.map +1 -0
  355. package/dist/cjs/unreal-integration/UnrealMarketplace.js +497 -0
  356. package/dist/cjs/unreal-integration/UnrealMarketplace.js.map +1 -0
  357. package/dist/cjs/unreal-integration/components/UnrealProjectBrowser.js +835 -0
  358. package/dist/cjs/unreal-integration/components/UnrealProjectBrowser.js.map +1 -0
  359. package/dist/cjs/xr/ARVR.js +32 -9
  360. package/dist/cjs/xr/ARVR.js.map +1 -1
  361. package/dist/esm/accessibility/Accessibility.js +55 -20
  362. package/dist/esm/accessibility/Accessibility.js.map +1 -1
  363. package/dist/esm/ai/AI.js +74 -40
  364. package/dist/esm/ai/AI.js.map +1 -1
  365. package/dist/esm/ai/BehaviorTree.js +39 -47
  366. package/dist/esm/ai/BehaviorTree.js.map +1 -1
  367. package/dist/esm/ai/StateMachine.js +65 -99
  368. package/dist/esm/ai/StateMachine.js.map +1 -1
  369. package/dist/esm/animation/Animation.js +58 -34
  370. package/dist/esm/animation/Animation.js.map +1 -1
  371. package/dist/esm/audio/AudioAdvanced.js +71 -37
  372. package/dist/esm/audio/AudioAdvanced.js.map +1 -1
  373. package/dist/esm/audio/AudioBridge.js +61 -49
  374. package/dist/esm/audio/AudioBridge.js.map +1 -1
  375. package/dist/esm/audio/AudioManager.js +45 -29
  376. package/dist/esm/audio/AudioManager.js.map +1 -1
  377. package/dist/esm/build-cloud/BuildOrchestration.js +1129 -0
  378. package/dist/esm/build-cloud/BuildOrchestration.js.map +1 -0
  379. package/dist/esm/build-cloud/GodotBuildCloud.js +715 -0
  380. package/dist/esm/build-cloud/GodotBuildCloud.js.map +1 -0
  381. package/dist/esm/build-cloud/UnityBuildCloud.js +643 -0
  382. package/dist/esm/build-cloud/UnityBuildCloud.js.map +1 -0
  383. package/dist/esm/build-cloud/UnrealBuildCloud.js +668 -0
  384. package/dist/esm/build-cloud/UnrealBuildCloud.js.map +1 -0
  385. package/dist/esm/core/EventBus.js +2 -1
  386. package/dist/esm/core/EventBus.js.map +1 -1
  387. package/dist/esm/core/GameConfig.js +1 -1
  388. package/dist/esm/core/GameConfig.js.map +1 -1
  389. package/dist/esm/core/GameLoop.js +6 -3
  390. package/dist/esm/core/GameLoop.js.map +1 -1
  391. package/dist/esm/core/ServiceLocator.js +1 -2
  392. package/dist/esm/core/ServiceLocator.js.map +1 -1
  393. package/dist/esm/devtools/DevTools.js +73 -43
  394. package/dist/esm/devtools/DevTools.js.map +1 -1
  395. package/dist/esm/devtools/DeveloperExperience.js +84 -42
  396. package/dist/esm/devtools/DeveloperExperience.js.map +1 -1
  397. package/dist/esm/devtools/GameplayAnalytics.js +71 -43
  398. package/dist/esm/devtools/GameplayAnalytics.js.map +1 -1
  399. package/dist/esm/dialogue/DialogueSystem.js +153 -129
  400. package/dist/esm/dialogue/DialogueSystem.js.map +1 -1
  401. package/dist/esm/docs/DocGenerator.js +70 -28
  402. package/dist/esm/docs/DocGenerator.js.map +1 -1
  403. package/dist/esm/ecs/World.js +159 -33
  404. package/dist/esm/ecs/World.js.map +1 -1
  405. package/dist/esm/editor/AchievementEditor.js +645 -0
  406. package/dist/esm/editor/AchievementEditor.js.map +1 -0
  407. package/dist/esm/editor/AdvancedEditor.js +95 -23
  408. package/dist/esm/editor/AdvancedEditor.js.map +1 -1
  409. package/dist/esm/editor/AnimationRetargeting.js +738 -0
  410. package/dist/esm/editor/AnimationRetargeting.js.map +1 -0
  411. package/dist/esm/editor/AssetBrowser.js +1184 -0
  412. package/dist/esm/editor/AssetBrowser.js.map +1 -0
  413. package/dist/esm/editor/AssetManager.js +22 -14
  414. package/dist/esm/editor/AssetManager.js.map +1 -1
  415. package/dist/esm/editor/ConsolePro.js +1085 -0
  416. package/dist/esm/editor/ConsolePro.js.map +1 -0
  417. package/dist/esm/editor/CraftingEditor.js +744 -0
  418. package/dist/esm/editor/CraftingEditor.js.map +1 -0
  419. package/dist/esm/editor/DebugTools.js +46 -23
  420. package/dist/esm/editor/DebugTools.js.map +1 -1
  421. package/dist/esm/editor/DialogueSystem.js +918 -0
  422. package/dist/esm/editor/DialogueSystem.js.map +1 -0
  423. package/dist/esm/editor/LeaderboardSystem.js +491 -0
  424. package/dist/esm/editor/LeaderboardSystem.js.map +1 -0
  425. package/dist/esm/editor/NiceGameEditor.js +532 -142
  426. package/dist/esm/editor/NiceGameEditor.js.map +1 -1
  427. package/dist/esm/editor/PrefabSystem.js +858 -0
  428. package/dist/esm/editor/PrefabSystem.js.map +1 -0
  429. package/dist/esm/editor/ProfilerPanel.js +710 -0
  430. package/dist/esm/editor/ProfilerPanel.js.map +1 -0
  431. package/dist/esm/editor/QuestEditor.js +846 -0
  432. package/dist/esm/editor/QuestEditor.js.map +1 -0
  433. package/dist/esm/editor/ReplaySystem.js +648 -0
  434. package/dist/esm/editor/ReplaySystem.js.map +1 -0
  435. package/dist/esm/editor/SceneEditor.js +33 -13
  436. package/dist/esm/editor/SceneEditor.js.map +1 -1
  437. package/dist/esm/editor/ScreenshotMode.js +702 -0
  438. package/dist/esm/editor/ScreenshotMode.js.map +1 -0
  439. package/dist/esm/editor/ShaderGraph.js +7 -6
  440. package/dist/esm/editor/ShaderGraph.js.map +1 -1
  441. package/dist/esm/editor/TimelineEditor.js +37 -30
  442. package/dist/esm/editor/TimelineEditor.js.map +1 -1
  443. package/dist/esm/editor/UndoRedoPro.js +914 -0
  444. package/dist/esm/editor/UndoRedoPro.js.map +1 -0
  445. package/dist/esm/editor/VRARMode.js +653 -0
  446. package/dist/esm/editor/VRARMode.js.map +1 -0
  447. package/dist/esm/engine/NiceGameEngine.js +12 -10
  448. package/dist/esm/engine/NiceGameEngine.js.map +1 -1
  449. package/dist/esm/engine/SaveSystemV2.js +73 -37
  450. package/dist/esm/engine/SaveSystemV2.js.map +1 -1
  451. package/dist/esm/enterprise/Enterprise.js +92 -44
  452. package/dist/esm/enterprise/Enterprise.js.map +1 -1
  453. package/dist/esm/export/BevyExporter.js +995 -0
  454. package/dist/esm/export/BevyExporter.js.map +1 -0
  455. package/dist/esm/export/CocosExporter.js +703 -0
  456. package/dist/esm/export/CocosExporter.js.map +1 -0
  457. package/dist/esm/export/Construct3Exporter.js +829 -0
  458. package/dist/esm/export/Construct3Exporter.js.map +1 -0
  459. package/dist/esm/export/DefoldExporter.js +1103 -0
  460. package/dist/esm/export/DefoldExporter.js.map +1 -0
  461. package/dist/esm/export/GDevelopExporter.js +745 -0
  462. package/dist/esm/export/GDevelopExporter.js.map +1 -0
  463. package/dist/esm/export/GameMakerExporter.js +843 -0
  464. package/dist/esm/export/GameMakerExporter.js.map +1 -0
  465. package/dist/esm/export/GodotExporter.js +45 -50
  466. package/dist/esm/export/GodotExporter.js.map +1 -1
  467. package/dist/esm/export/MinecraftBedrockExporter.js +603 -0
  468. package/dist/esm/export/MinecraftBedrockExporter.js.map +1 -0
  469. package/dist/esm/export/MonoGameExporter.js +1331 -0
  470. package/dist/esm/export/MonoGameExporter.js.map +1 -0
  471. package/dist/esm/export/Pico8Exporter.js +843 -0
  472. package/dist/esm/export/Pico8Exporter.js.map +1 -0
  473. package/dist/esm/export/PlatformExporter.js +4 -5
  474. package/dist/esm/export/PlatformExporter.js.map +1 -1
  475. package/dist/esm/export/RPGMakerExporter.js +903 -0
  476. package/dist/esm/export/RPGMakerExporter.js.map +1 -0
  477. package/dist/esm/export/RobloxExporter.js +940 -0
  478. package/dist/esm/export/RobloxExporter.js.map +1 -0
  479. package/dist/esm/export/Solar2DExporter.js +1280 -0
  480. package/dist/esm/export/Solar2DExporter.js.map +1 -0
  481. package/dist/esm/export/ThreeJSExporter.js +52 -55
  482. package/dist/esm/export/ThreeJSExporter.js.map +1 -1
  483. package/dist/esm/export/Tic80Exporter.js +1312 -0
  484. package/dist/esm/export/Tic80Exporter.js.map +1 -0
  485. package/dist/esm/export/UnityExporter.js +41 -56
  486. package/dist/esm/export/UnityExporter.js.map +1 -1
  487. package/dist/esm/export/UnrealExporter.js +959 -0
  488. package/dist/esm/export/UnrealExporter.js.map +1 -0
  489. package/dist/esm/export/WebExporter.js +42 -50
  490. package/dist/esm/export/WebExporter.js.map +1 -1
  491. package/dist/esm/export/index.js +38 -2
  492. package/dist/esm/export/index.js.map +1 -1
  493. package/dist/esm/godot-integration/GodotAssetLibrary.js +711 -0
  494. package/dist/esm/godot-integration/GodotAssetLibrary.js.map +1 -0
  495. package/dist/esm/godot-integration/GodotImport.js +1062 -0
  496. package/dist/esm/godot-integration/GodotImport.js.map +1 -0
  497. package/dist/esm/godot-integration/GodotLiveLink.js +958 -0
  498. package/dist/esm/godot-integration/GodotLiveLink.js.map +1 -0
  499. package/dist/esm/godot-integration/components/GodotProjectBrowser.js +666 -0
  500. package/dist/esm/godot-integration/components/GodotProjectBrowser.js.map +1 -0
  501. package/dist/esm/i18n/I18n.js +56 -30
  502. package/dist/esm/i18n/I18n.js.map +1 -1
  503. package/dist/esm/i18n/useTranslation.js +4 -4
  504. package/dist/esm/i18n/useTranslation.js.map +1 -1
  505. package/dist/esm/import/AsepriteImporter.js +15 -19
  506. package/dist/esm/import/AsepriteImporter.js.map +1 -1
  507. package/dist/esm/import/DragonBonesImporter.js +36 -40
  508. package/dist/esm/import/DragonBonesImporter.js.map +1 -1
  509. package/dist/esm/import/GameMakerImporter.js +11 -17
  510. package/dist/esm/import/GameMakerImporter.js.map +1 -1
  511. package/dist/esm/import/GodotSceneImporter.js +43 -29
  512. package/dist/esm/import/GodotSceneImporter.js.map +1 -1
  513. package/dist/esm/import/LDtkImporter.js +16 -17
  514. package/dist/esm/import/LDtkImporter.js.map +1 -1
  515. package/dist/esm/import/Live2DImporter.js +16 -19
  516. package/dist/esm/import/Live2DImporter.js.map +1 -1
  517. package/dist/esm/import/NdgFormat.js +28 -35
  518. package/dist/esm/import/NdgFormat.js.map +1 -1
  519. package/dist/esm/import/OgmoImporter.js +16 -18
  520. package/dist/esm/import/OgmoImporter.js.map +1 -1
  521. package/dist/esm/import/RPGMakerImporter.js +5 -6
  522. package/dist/esm/import/RPGMakerImporter.js.map +1 -1
  523. package/dist/esm/import/SceneImporter.js +23 -13
  524. package/dist/esm/import/SceneImporter.js.map +1 -1
  525. package/dist/esm/import/SpineImporter.js +20 -20
  526. package/dist/esm/import/SpineImporter.js.map +1 -1
  527. package/dist/esm/import/SpriterImporter.js +38 -46
  528. package/dist/esm/import/SpriterImporter.js.map +1 -1
  529. package/dist/esm/import/TiledMapImporter.js +20 -24
  530. package/dist/esm/import/TiledMapImporter.js.map +1 -1
  531. package/dist/esm/import/UnitySceneImporter.js +15 -15
  532. package/dist/esm/import/UnitySceneImporter.js.map +1 -1
  533. package/dist/esm/index.js +76 -3
  534. package/dist/esm/index.js.map +1 -1
  535. package/dist/esm/input/GamepadNavigation.js +82 -47
  536. package/dist/esm/input/GamepadNavigation.js.map +1 -1
  537. package/dist/esm/input/InputManager.js +107 -52
  538. package/dist/esm/input/InputManager.js.map +1 -1
  539. package/dist/esm/input/useGamepads.js +35 -29
  540. package/dist/esm/input/useGamepads.js.map +1 -1
  541. package/dist/esm/integration/IconSprite.js +39 -21
  542. package/dist/esm/integration/IconSprite.js.map +1 -1
  543. package/dist/esm/inventory/InventorySystem.js +218 -127
  544. package/dist/esm/inventory/InventorySystem.js.map +1 -1
  545. package/dist/esm/kids/KidMode.js +491 -78
  546. package/dist/esm/kids/KidMode.js.map +1 -1
  547. package/dist/esm/kids/KidTools.js +154 -48
  548. package/dist/esm/kids/KidTools.js.map +1 -1
  549. package/dist/esm/monetization/Monetization.js +47 -35
  550. package/dist/esm/monetization/Monetization.js.map +1 -1
  551. package/dist/esm/multiplayer/LocalMultiplayer.js +48 -26
  552. package/dist/esm/multiplayer/LocalMultiplayer.js.map +1 -1
  553. package/dist/esm/multiplayer/MiniGameTypes.js +23 -10
  554. package/dist/esm/multiplayer/MiniGameTypes.js.map +1 -1
  555. package/dist/esm/native-engine/AssetPipeline.js +537 -0
  556. package/dist/esm/native-engine/AssetPipeline.js.map +1 -0
  557. package/dist/esm/native-engine/AudioEngine.js +822 -0
  558. package/dist/esm/native-engine/AudioEngine.js.map +1 -0
  559. package/dist/esm/native-engine/InputSystem.js +1170 -0
  560. package/dist/esm/native-engine/InputSystem.js.map +1 -0
  561. package/dist/esm/native-engine/NetworkingSystem.js +817 -0
  562. package/dist/esm/native-engine/NetworkingSystem.js.map +1 -0
  563. package/dist/esm/native-engine/PhysicsEngine.js +619 -0
  564. package/dist/esm/native-engine/PhysicsEngine.js.map +1 -0
  565. package/dist/esm/native-engine/RenderingEngine.js +368 -0
  566. package/dist/esm/native-engine/RenderingEngine.js.map +1 -0
  567. package/dist/esm/native-engine/SceneGraph.js +857 -0
  568. package/dist/esm/native-engine/SceneGraph.js.map +1 -0
  569. package/dist/esm/network/MultiplayerTransport.js +63 -33
  570. package/dist/esm/network/MultiplayerTransport.js.map +1 -1
  571. package/dist/esm/network/Networking.js +87 -62
  572. package/dist/esm/network/Networking.js.map +1 -1
  573. package/dist/esm/pathfinding/Pathfinding.js +71 -26
  574. package/dist/esm/pathfinding/Pathfinding.js.map +1 -1
  575. package/dist/esm/performance/AssetStreaming.js +649 -0
  576. package/dist/esm/performance/AssetStreaming.js.map +1 -0
  577. package/dist/esm/performance/BenchmarkSuite.js +37 -24
  578. package/dist/esm/performance/BenchmarkSuite.js.map +1 -1
  579. package/dist/esm/performance/DeltaCompression.js +554 -0
  580. package/dist/esm/performance/DeltaCompression.js.map +1 -0
  581. package/dist/esm/performance/DeterministicPhysics.js +573 -0
  582. package/dist/esm/performance/DeterministicPhysics.js.map +1 -0
  583. package/dist/esm/performance/ECSOptimization.js +716 -0
  584. package/dist/esm/performance/ECSOptimization.js.map +1 -0
  585. package/dist/esm/performance/HotReload.js +517 -0
  586. package/dist/esm/performance/HotReload.js.map +1 -0
  587. package/dist/esm/performance/InputRecording.js +580 -0
  588. package/dist/esm/performance/InputRecording.js.map +1 -0
  589. package/dist/esm/performance/InterestManagement.js +524 -0
  590. package/dist/esm/performance/InterestManagement.js.map +1 -0
  591. package/dist/esm/performance/LocalizationRuntime.js +465 -0
  592. package/dist/esm/performance/LocalizationRuntime.js.map +1 -0
  593. package/dist/esm/performance/MemoryManagement.js +481 -0
  594. package/dist/esm/performance/MemoryManagement.js.map +1 -0
  595. package/dist/esm/performance/ModSupport.js +520 -0
  596. package/dist/esm/performance/ModSupport.js.map +1 -0
  597. package/dist/esm/performance/MultiThreadedPhysics.js +491 -0
  598. package/dist/esm/performance/MultiThreadedPhysics.js.map +1 -0
  599. package/dist/esm/performance/NetworkPrediction.js +508 -0
  600. package/dist/esm/performance/NetworkPrediction.js.map +1 -0
  601. package/dist/esm/performance/ObjectPooling.js +639 -0
  602. package/dist/esm/performance/ObjectPooling.js.map +1 -0
  603. package/dist/esm/performance/Performance.js +64 -25
  604. package/dist/esm/performance/Performance.js.map +1 -1
  605. package/dist/esm/performance/PerformanceAdvanced.js +83 -49
  606. package/dist/esm/performance/PerformanceAdvanced.js.map +1 -1
  607. package/dist/esm/performance/SaveSystem.js +567 -0
  608. package/dist/esm/performance/SaveSystem.js.map +1 -0
  609. package/dist/esm/performance/WebGPURenderer.js +697 -0
  610. package/dist/esm/performance/WebGPURenderer.js.map +1 -0
  611. package/dist/esm/physics/PhysicsAdvanced.js +32 -20
  612. package/dist/esm/physics/PhysicsAdvanced.js.map +1 -1
  613. package/dist/esm/physics/PhysicsEngine2D.js +63 -36
  614. package/dist/esm/physics/PhysicsEngine2D.js.map +1 -1
  615. package/dist/esm/plugins/PluginSDK.js +46 -34
  616. package/dist/esm/plugins/PluginSDK.js.map +1 -1
  617. package/dist/esm/plugins/PluginTestKit.js +28 -29
  618. package/dist/esm/plugins/PluginTestKit.js.map +1 -1
  619. package/dist/esm/procedural/Procedural.js +24 -17
  620. package/dist/esm/procedural/Procedural.js.map +1 -1
  621. package/dist/esm/procedural/WorldBuilding.js +188 -55
  622. package/dist/esm/procedural/WorldBuilding.js.map +1 -1
  623. package/dist/esm/quest/QuestSystem.js +145 -111
  624. package/dist/esm/quest/QuestSystem.js.map +1 -1
  625. package/dist/esm/release/Release.js +142 -31
  626. package/dist/esm/release/Release.js.map +1 -1
  627. package/dist/esm/rendering/Camera2D.js +12 -7
  628. package/dist/esm/rendering/Camera2D.js.map +1 -1
  629. package/dist/esm/rendering/Renderer2D.js +62 -26
  630. package/dist/esm/rendering/Renderer2D.js.map +1 -1
  631. package/dist/esm/rendering/WebGPURenderPipeline.js +60 -31
  632. package/dist/esm/rendering/WebGPURenderPipeline.js.map +1 -1
  633. package/dist/esm/runtime3d/Runtime3D.js +80 -55
  634. package/dist/esm/runtime3d/Runtime3D.js.map +1 -1
  635. package/dist/esm/runtime3d/SceneEditor3D.js +20 -11
  636. package/dist/esm/runtime3d/SceneEditor3D.js.map +1 -1
  637. package/dist/esm/scene/SceneManager.js +26 -12
  638. package/dist/esm/scene/SceneManager.js.map +1 -1
  639. package/dist/esm/scripting/GraphToAST.js +36 -24
  640. package/dist/esm/scripting/GraphToAST.js.map +1 -1
  641. package/dist/esm/scripting/LanguageExporter.js +2 -3
  642. package/dist/esm/scripting/LanguageExporter.js.map +1 -1
  643. package/dist/esm/scripting/NodeGraph.js +49 -32
  644. package/dist/esm/scripting/NodeGraph.js.map +1 -1
  645. package/dist/esm/scripting/ScriptAST.js +19 -6
  646. package/dist/esm/scripting/ScriptAST.js.map +1 -1
  647. package/dist/esm/scripting/VisualScripting2.js +13 -12
  648. package/dist/esm/scripting/VisualScripting2.js.map +1 -1
  649. package/dist/esm/scripting/exporters/CSharpExporter.js +16 -16
  650. package/dist/esm/scripting/exporters/CSharpExporter.js.map +1 -1
  651. package/dist/esm/scripting/exporters/GDScriptExporter.js +3 -6
  652. package/dist/esm/scripting/exporters/GDScriptExporter.js.map +1 -1
  653. package/dist/esm/scripting/exporters/LuaExporter.js +5 -9
  654. package/dist/esm/scripting/exporters/LuaExporter.js.map +1 -1
  655. package/dist/esm/scripting/exporters/PythonExporter.js +7 -13
  656. package/dist/esm/scripting/exporters/PythonExporter.js.map +1 -1
  657. package/dist/esm/scripting/exporters/RustExporter.js +7 -14
  658. package/dist/esm/scripting/exporters/RustExporter.js.map +1 -1
  659. package/dist/esm/scripting/exporters/TypeScriptExporter.js +28 -29
  660. package/dist/esm/scripting/exporters/TypeScriptExporter.js.map +1 -1
  661. package/dist/esm/social/Social.js +27 -20
  662. package/dist/esm/social/Social.js.map +1 -1
  663. package/dist/esm/systems/LightingSystem.js +968 -0
  664. package/dist/esm/systems/LightingSystem.js.map +1 -0
  665. package/dist/esm/systems/ParticleSystem2.js +34 -17
  666. package/dist/esm/systems/ParticleSystem2.js.map +1 -1
  667. package/dist/esm/systems/PhysicsDebugSystem.js +1219 -0
  668. package/dist/esm/systems/PhysicsDebugSystem.js.map +1 -0
  669. package/dist/esm/systems/TerrainSystem.js +1125 -0
  670. package/dist/esm/systems/TerrainSystem.js.map +1 -0
  671. package/dist/esm/templates/ActionTemplates.js +144 -24
  672. package/dist/esm/templates/ActionTemplates.js.map +1 -1
  673. package/dist/esm/templates/FightingTemplate.js +513 -0
  674. package/dist/esm/templates/FightingTemplate.js.map +1 -0
  675. package/dist/esm/templates/MetroidvaniaTemplate.js +484 -0
  676. package/dist/esm/templates/MetroidvaniaTemplate.js.map +1 -0
  677. package/dist/esm/templates/PartyTemplates.js +60 -28
  678. package/dist/esm/templates/PartyTemplates.js.map +1 -1
  679. package/dist/esm/templates/PuzzleTemplates.js +63 -22
  680. package/dist/esm/templates/PuzzleTemplates.js.map +1 -1
  681. package/dist/esm/templates/RPGTemplates.js +29 -15
  682. package/dist/esm/templates/RPGTemplates.js.map +1 -1
  683. package/dist/esm/templates/SportsTemplates.js +41 -17
  684. package/dist/esm/templates/SportsTemplates.js.map +1 -1
  685. package/dist/esm/templates/StrategyTemplates.js +45 -18
  686. package/dist/esm/templates/StrategyTemplates.js.map +1 -1
  687. package/dist/esm/templates/SurvivalTemplate.js +447 -0
  688. package/dist/esm/templates/SurvivalTemplate.js.map +1 -0
  689. package/dist/esm/templates/WaveDefense.js +139 -52
  690. package/dist/esm/templates/WaveDefense.js.map +1 -1
  691. package/dist/esm/tilemap/Tilemap.js +48 -32
  692. package/dist/esm/tilemap/Tilemap.js.map +1 -1
  693. package/dist/esm/tutorials/TutorialBranching.js +111 -42
  694. package/dist/esm/tutorials/TutorialBranching.js.map +1 -1
  695. package/dist/esm/tutorials/TutorialEngine.js +152 -80
  696. package/dist/esm/tutorials/TutorialEngine.js.map +1 -1
  697. package/dist/esm/tutorials/TutorialOverlay.js +117 -40
  698. package/dist/esm/tutorials/TutorialOverlay.js.map +1 -1
  699. package/dist/esm/ui/GameUI.js +69 -42
  700. package/dist/esm/ui/GameUI.js.map +1 -1
  701. package/dist/esm/unity-integration/UnityAssetStore.js +749 -0
  702. package/dist/esm/unity-integration/UnityAssetStore.js.map +1 -0
  703. package/dist/esm/unity-integration/UnityImport.js +1246 -0
  704. package/dist/esm/unity-integration/UnityImport.js.map +1 -0
  705. package/dist/esm/unity-integration/UnityLiveLink.js +1017 -0
  706. package/dist/esm/unity-integration/UnityLiveLink.js.map +1 -0
  707. package/dist/esm/unity-integration/components/UnityAssetStoreBrowser.js +794 -0
  708. package/dist/esm/unity-integration/components/UnityAssetStoreBrowser.js.map +1 -0
  709. package/dist/esm/unity-integration/components/UnityProjectBrowser.js +831 -0
  710. package/dist/esm/unity-integration/components/UnityProjectBrowser.js.map +1 -0
  711. package/dist/esm/unreal-integration/UnrealImport.js +437 -0
  712. package/dist/esm/unreal-integration/UnrealImport.js.map +1 -0
  713. package/dist/esm/unreal-integration/UnrealLiveLink.js +1182 -0
  714. package/dist/esm/unreal-integration/UnrealLiveLink.js.map +1 -0
  715. package/dist/esm/unreal-integration/UnrealMarketplace.js +490 -0
  716. package/dist/esm/unreal-integration/UnrealMarketplace.js.map +1 -0
  717. package/dist/esm/unreal-integration/components/UnrealProjectBrowser.js +833 -0
  718. package/dist/esm/unreal-integration/components/UnrealProjectBrowser.js.map +1 -0
  719. package/dist/esm/xr/ARVR.js +32 -9
  720. package/dist/esm/xr/ARVR.js.map +1 -1
  721. package/package.json +6 -5
@@ -17,14 +17,16 @@ class UndoStack {
17
17
  // Remove everything after current pointer
18
18
  this.stack.splice(this.pointer + 1);
19
19
  this.stack.push(entry);
20
- if (this.stack.length > this.maxSize)
20
+ if (this.stack.length > this.maxSize) {
21
21
  this.stack.shift();
22
+ }
22
23
  this.pointer = this.stack.length - 1;
23
24
  this.events.emit('change', null);
24
25
  }
25
26
  undo() {
26
- if (this.pointer < 0)
27
+ if (this.pointer < 0) {
27
28
  return null;
29
+ }
28
30
  const entry = this.stack[this.pointer];
29
31
  entry.undo();
30
32
  this.pointer--;
@@ -32,8 +34,9 @@ class UndoStack {
32
34
  return entry.description;
33
35
  }
34
36
  redo() {
35
- if (this.pointer >= this.stack.length - 1)
37
+ if (this.pointer >= this.stack.length - 1) {
36
38
  return null;
39
+ }
37
40
  this.pointer++;
38
41
  const entry = this.stack[this.pointer];
39
42
  entry.redo();
@@ -83,7 +86,10 @@ function editorReducer(state, action) {
83
86
  case 'SELECT':
84
87
  return { ...state, selectedEntities: action.entities };
85
88
  case 'ADD_SELECTION':
86
- return { ...state, selectedEntities: [...new Set([...state.selectedEntities, ...action.entities])] };
89
+ return {
90
+ ...state,
91
+ selectedEntities: [...new Set([...state.selectedEntities, ...action.entities])],
92
+ };
87
93
  case 'DESELECT_ALL':
88
94
  return { ...state, selectedEntities: [] };
89
95
  case 'SET_CLIPBOARD':
@@ -101,7 +107,7 @@ function editorReducer(state, action) {
101
107
  case 'TOGGLE_PANEL':
102
108
  return {
103
109
  ...state,
104
- panels: state.panels.map(p => p.id === action.panelId ? { ...p, visible: !p.visible } : p),
110
+ panels: state.panels.map((p) => p.id === action.panelId ? { ...p, visible: !p.visible } : p),
105
111
  };
106
112
  case 'SET_SCENE':
107
113
  return { ...state, currentSceneId: action.sceneId };
@@ -116,8 +122,9 @@ function editorReducer(state, action) {
116
122
  const EditorContext = React.createContext(null);
117
123
  function useEditor() {
118
124
  const ctx = React.useContext(EditorContext);
119
- if (!ctx)
125
+ if (!ctx) {
120
126
  throw new Error('useEditor must be used within NiceGameEditor');
127
+ }
121
128
  return ctx;
122
129
  }
123
130
  function useKeyboardShortcuts(shortcuts) {
@@ -146,7 +153,6 @@ function deserializeProject(json) {
146
153
  return JSON.parse(json);
147
154
  }
148
155
  const NiceGameEditor = ({ project, onSave, className, style, }) => {
149
- var _a, _b, _c, _d;
150
156
  const [state, dispatch] = React.useReducer(editorReducer, DEFAULT_STATE);
151
157
  const worldRef = React.useRef(new World.World());
152
158
  const undoRef = React.useRef(new UndoStack());
@@ -159,9 +165,12 @@ const NiceGameEditor = ({ project, onSave, className, style, }) => {
159
165
  const [worldVersion, setWorldVersion] = React.useState(0);
160
166
  const [consoleLogs, setConsoleLogs] = React.useState([]);
161
167
  const logIdRef = React.useRef(0);
162
- const refreshWorld = React.useCallback(() => setWorldVersion(v => v + 1), []);
168
+ const refreshWorld = React.useCallback(() => setWorldVersion((v) => v + 1), []);
163
169
  const addLog = React.useCallback((level, message) => {
164
- setConsoleLogs(prev => [...prev.slice(-199), { id: logIdRef.current++, level, message, timestamp: Date.now() }]);
170
+ setConsoleLogs((prev) => [
171
+ ...prev.slice(-199),
172
+ { id: logIdRef.current++, level, message, timestamp: Date.now() },
173
+ ]);
165
174
  }, []);
166
175
  const createNewEntity = React.useCallback((name, components = ['Transform']) => {
167
176
  const entity = world.createEntity(name);
@@ -180,31 +189,68 @@ const NiceGameEditor = ({ project, onSave, className, style, }) => {
180
189
  { key: 'z', ctrl: true, shift: true, action: () => undoStack.redo(), description: 'Redo' },
181
190
  { key: 'y', ctrl: true, action: () => undoStack.redo(), description: 'Redo' },
182
191
  { key: 's', ctrl: true, action: () => handleSave(), description: 'Save' },
183
- { key: 'v', action: () => dispatch({ type: 'SET_TOOL', tool: 'select' }), description: 'Select tool' },
184
- { key: 'w', action: () => dispatch({ type: 'SET_TOOL', tool: 'move' }), description: 'Move tool' },
185
- { key: 'e', action: () => dispatch({ type: 'SET_TOOL', tool: 'rotate' }), description: 'Rotate tool' },
186
- { key: 'r', action: () => dispatch({ type: 'SET_TOOL', tool: 'scale' }), description: 'Scale tool' },
187
- { key: 'g', ctrl: true, action: () => dispatch({ type: 'TOGGLE_GRID' }), description: 'Toggle grid' },
192
+ {
193
+ key: 'v',
194
+ action: () => dispatch({ type: 'SET_TOOL', tool: 'select' }),
195
+ description: 'Select tool',
196
+ },
197
+ {
198
+ key: 'w',
199
+ action: () => dispatch({ type: 'SET_TOOL', tool: 'move' }),
200
+ description: 'Move tool',
201
+ },
202
+ {
203
+ key: 'e',
204
+ action: () => dispatch({ type: 'SET_TOOL', tool: 'rotate' }),
205
+ description: 'Rotate tool',
206
+ },
207
+ {
208
+ key: 'r',
209
+ action: () => dispatch({ type: 'SET_TOOL', tool: 'scale' }),
210
+ description: 'Scale tool',
211
+ },
212
+ {
213
+ key: 'g',
214
+ ctrl: true,
215
+ action: () => dispatch({ type: 'TOGGLE_GRID' }),
216
+ description: 'Toggle grid',
217
+ },
188
218
  { key: 'Delete', action: () => deleteSelected(), description: 'Delete selected' },
189
- { key: 'Escape', action: () => dispatch({ type: 'DESELECT_ALL' }), description: 'Deselect all' },
219
+ {
220
+ key: 'Escape',
221
+ action: () => dispatch({ type: 'DESELECT_ALL' }),
222
+ description: 'Deselect all',
223
+ },
190
224
  ], []);
191
225
  useKeyboardShortcuts(shortcuts);
192
226
  const handleSave = React.useCallback(() => {
193
- var _a;
194
- if (!onSave)
227
+ if (!onSave) {
195
228
  return;
196
- const sceneData = sceneManager.serializeScene(world, (_a = state.currentSceneId) !== null && _a !== void 0 ? _a : 'main', 'Main Scene');
197
- const proj = project !== null && project !== void 0 ? project : {
229
+ }
230
+ const sceneData = sceneManager.serializeScene(world, state.currentSceneId ?? 'main', 'Main Scene');
231
+ const proj = project ?? {
198
232
  formatVersion: 1,
199
233
  name: 'Untitled',
200
234
  version: '0.1.0',
201
235
  author: '',
202
236
  description: '',
203
237
  settings: {
204
- width: 1280, height: 720, targetFps: 60, maxPlayers: 4,
205
- physics: { gravity: { x: 0, y: 980 }, fixedTimestep: 1 / 60, velocityIterations: 8, positionIterations: 3 },
238
+ width: 1280,
239
+ height: 720,
240
+ targetFps: 60,
241
+ maxPlayers: 4,
242
+ physics: {
243
+ gravity: { x: 0, y: 980 },
244
+ fixedTimestep: 1 / 60,
245
+ velocityIterations: 8,
246
+ positionIterations: 3,
247
+ },
206
248
  audio: { masterVolume: 1, musicVolume: 0.7, sfxVolume: 1, voiceVolume: 1 },
207
- rendering: { pixelPerfect: false, antiAlias: true, backgroundColor: { r: 0.1, g: 0.1, b: 0.15, a: 1 } },
249
+ rendering: {
250
+ pixelPerfect: false,
251
+ antiAlias: true,
252
+ backgroundColor: { r: 0.1, g: 0.1, b: 0.15, a: 1 },
253
+ },
208
254
  },
209
255
  scenes: [sceneData],
210
256
  startSceneId: sceneData.id,
@@ -215,10 +261,11 @@ const NiceGameEditor = ({ project, onSave, className, style, }) => {
215
261
  }, [world, sceneManager, state.currentSceneId, project, onSave]);
216
262
  const deleteSelected = React.useCallback(() => {
217
263
  const toDelete = [...state.selectedEntities];
218
- if (toDelete.length === 0)
264
+ if (toDelete.length === 0) {
219
265
  return;
266
+ }
220
267
  // Store for undo
221
- const snapshots = toDelete.map(id => ({
268
+ const snapshots = toDelete.map((id) => ({
222
269
  entity: world.getEntity(id),
223
270
  id,
224
271
  }));
@@ -236,8 +283,9 @@ const NiceGameEditor = ({ project, onSave, className, style, }) => {
236
283
  events.emit('editor:undo:delete', snapshots);
237
284
  },
238
285
  redo: () => {
239
- for (const id of toDelete)
286
+ for (const id of toDelete) {
240
287
  world.destroyEntity(id);
288
+ }
241
289
  },
242
290
  });
243
291
  }, [state.selectedEntities, world, undoStack]);
@@ -253,8 +301,19 @@ const NiceGameEditor = ({ project, onSave, className, style, }) => {
253
301
  createNewEntity,
254
302
  consoleLogs,
255
303
  addLog,
256
- }), [state, world, undoStack, sceneManager, events, worldVersion, refreshWorld, createNewEntity, consoleLogs, addLog]);
257
- return (jsxRuntime.jsx(EditorContext.Provider, { value: ctxValue, children: jsxRuntime.jsxs("div", { className: `nice-game-editor ${className !== null && className !== void 0 ? className : ''}`, style: {
304
+ }), [
305
+ state,
306
+ world,
307
+ undoStack,
308
+ sceneManager,
309
+ events,
310
+ worldVersion,
311
+ refreshWorld,
312
+ createNewEntity,
313
+ consoleLogs,
314
+ addLog,
315
+ ]);
316
+ return (jsxRuntime.jsx(EditorContext.Provider, { value: ctxValue, children: jsxRuntime.jsxs("div", { className: `nice-game-editor ${className ?? ''}`, style: {
258
317
  display: 'flex',
259
318
  flexDirection: 'column',
260
319
  width: '100%',
@@ -265,39 +324,52 @@ const NiceGameEditor = ({ project, onSave, className, style, }) => {
265
324
  fontFamily: 'monospace',
266
325
  fontSize: 13,
267
326
  ...style,
268
- }, children: [jsxRuntime.jsx(EditorToolbar, {}), jsxRuntime.jsxs("div", { style: { display: 'flex', flex: 1, overflow: 'hidden' }, children: [((_a = state.panels.find(p => p.id === 'hierarchy')) === null || _a === void 0 ? void 0 : _a.visible) && (jsxRuntime.jsx(EditorPanelContainer, { title: "Hierarchy", width: 250, position: "left", children: jsxRuntime.jsx(HierarchyPanel, {}) })), jsxRuntime.jsx("div", { style: { flex: 1, position: 'relative', overflow: 'hidden' }, children: jsxRuntime.jsx(EditorViewport, {}) }), ((_b = state.panels.find(p => p.id === 'inspector')) === null || _b === void 0 ? void 0 : _b.visible) && (jsxRuntime.jsx(EditorPanelContainer, { title: "Inspector", width: 300, position: "right", children: jsxRuntime.jsx(InspectorPanel, {}) }))] }), jsxRuntime.jsxs("div", { style: { display: 'flex', borderTop: '1px solid #45475a' }, children: [((_c = state.panels.find(p => p.id === 'assets')) === null || _c === void 0 ? void 0 : _c.visible) && (jsxRuntime.jsx(EditorPanelContainer, { title: "Assets", height: 180, position: "bottom", children: jsxRuntime.jsx(AssetPanel, {}) })), ((_d = state.panels.find(p => p.id === 'console')) === null || _d === void 0 ? void 0 : _d.visible) && (jsxRuntime.jsx(EditorPanelContainer, { title: "Console", height: 180, position: "bottom", children: jsxRuntime.jsx(ConsolePanel, {}) }))] })] }) }));
327
+ }, children: [jsxRuntime.jsx(EditorToolbar, {}), jsxRuntime.jsxs("div", { style: { display: 'flex', flex: 1, overflow: 'hidden' }, children: [state.panels.find((p) => p.id === 'hierarchy')?.visible && (jsxRuntime.jsx(EditorPanelContainer, { title: "Hierarchy", width: 250, position: "left", children: jsxRuntime.jsx(HierarchyPanel, {}) })), jsxRuntime.jsx("div", { style: { flex: 1, position: 'relative', overflow: 'hidden' }, children: jsxRuntime.jsx(EditorViewport, {}) }), state.panels.find((p) => p.id === 'inspector')?.visible && (jsxRuntime.jsx(EditorPanelContainer, { title: "Inspector", width: 300, position: "right", children: jsxRuntime.jsx(InspectorPanel, {}) }))] }), jsxRuntime.jsxs("div", { style: { display: 'flex', borderTop: '1px solid #45475a' }, children: [state.panels.find((p) => p.id === 'assets')?.visible && (jsxRuntime.jsx(EditorPanelContainer, { title: "Assets", height: 180, position: "bottom", children: jsxRuntime.jsx(AssetPanel, {}) })), state.panels.find((p) => p.id === 'console')?.visible && (jsxRuntime.jsx(EditorPanelContainer, { title: "Console", height: 180, position: "bottom", children: jsxRuntime.jsx(ConsolePanel, {}) }))] })] }) }));
269
328
  };
270
329
  /* ── Helpers ───────────────────────────────────────────────────── */
271
330
  function getEntityBounds(entityId, world) {
272
331
  const shape = world.getComponent(entityId, 'Shape');
273
332
  if (shape) {
274
- if (shape.shape === 'circle')
333
+ if (shape.shape === 'circle') {
275
334
  return { x: -shape.radius, y: -shape.radius, w: shape.radius * 2, h: shape.radius * 2 };
335
+ }
276
336
  return { x: -shape.width / 2, y: -shape.height / 2, w: shape.width, h: shape.height };
277
337
  }
278
338
  const sprite = world.getComponent(entityId, 'Sprite');
279
339
  if (sprite) {
280
- return { x: -sprite.width * sprite.anchor.x, y: -sprite.height * sprite.anchor.y, w: sprite.width, h: sprite.height };
340
+ return {
341
+ x: -sprite.width * sprite.anchor.x,
342
+ y: -sprite.height * sprite.anchor.y,
343
+ w: sprite.width,
344
+ h: sprite.height,
345
+ };
281
346
  }
282
347
  const text = world.getComponent(entityId, 'Text');
283
- if (text)
348
+ if (text) {
284
349
  return { x: -30, y: -text.fontSize / 2, w: 60, h: text.fontSize };
350
+ }
285
351
  return { x: -8, y: -8, w: 16, h: 16 };
286
352
  }
287
353
  function getEntityWorldBounds(entityId, world) {
288
354
  const transform = world.getComponent(entityId, 'Transform');
289
- if (!transform)
355
+ if (!transform) {
290
356
  return null;
357
+ }
291
358
  const b = getEntityBounds(entityId, world);
292
359
  return { x: transform.position.x + b.x, y: transform.position.y + b.y, w: b.w, h: b.h };
293
360
  }
294
361
  function getCursorForTool(tool) {
295
362
  switch (tool) {
296
- case 'move': return 'move';
297
- case 'rotate': return 'crosshair';
298
- case 'scale': return 'nwse-resize';
299
- case 'pan': return 'grab';
300
- default: return 'default';
363
+ case 'move':
364
+ return 'move';
365
+ case 'rotate':
366
+ return 'crosshair';
367
+ case 'scale':
368
+ return 'nwse-resize';
369
+ case 'pan':
370
+ return 'grab';
371
+ default:
372
+ return 'default';
301
373
  }
302
374
  }
303
375
  function colorToCSS(c) {
@@ -305,14 +377,26 @@ function colorToCSS(c) {
305
377
  }
306
378
  function cssToColor(hex) {
307
379
  const m = hex.match(/^#?([0-9a-f]{2})([0-9a-f]{2})([0-9a-f]{2})$/i);
308
- if (!m)
380
+ if (!m) {
309
381
  return { r: 1, g: 1, b: 1, a: 1 };
310
- return { r: parseInt(m[1], 16) / 255, g: parseInt(m[2], 16) / 255, b: parseInt(m[3], 16) / 255, a: 1 };
382
+ }
383
+ return {
384
+ r: parseInt(m[1], 16) / 255,
385
+ g: parseInt(m[2], 16) / 255,
386
+ b: parseInt(m[3], 16) / 255,
387
+ a: 1,
388
+ };
311
389
  }
312
390
  function colorToHex(c) {
313
- const r = Math.round(c.r * 255).toString(16).padStart(2, '0');
314
- const g = Math.round(c.g * 255).toString(16).padStart(2, '0');
315
- const b = Math.round(c.b * 255).toString(16).padStart(2, '0');
391
+ const r = Math.round(c.r * 255)
392
+ .toString(16)
393
+ .padStart(2, '0');
394
+ const g = Math.round(c.g * 255)
395
+ .toString(16)
396
+ .padStart(2, '0');
397
+ const b = Math.round(c.b * 255)
398
+ .toString(16)
399
+ .padStart(2, '0');
316
400
  return `#${r}${g}${b}`;
317
401
  }
318
402
  const ENTITY_PRESETS = [
@@ -320,7 +404,11 @@ const ENTITY_PRESETS = [
320
404
  { label: 'Sprite', icon: '🖼', components: ['Transform', 'Sprite'] },
321
405
  { label: 'Shape', icon: '■', components: ['Transform', 'Shape'] },
322
406
  { label: 'Text', icon: 'T', components: ['Transform', 'Text'] },
323
- { label: 'Physics Body', icon: '⚡', components: ['Transform', 'Shape', 'Collider2D', 'RigidBody2D'] },
407
+ {
408
+ label: 'Physics Body',
409
+ icon: '⚡',
410
+ components: ['Transform', 'Shape', 'Collider2D', 'RigidBody2D'],
411
+ },
324
412
  { label: 'Camera', icon: '📷', components: ['Transform', 'CameraTarget'] },
325
413
  ];
326
414
  /* ── Sub-components ───────────────────────────────────────────── */
@@ -328,25 +416,64 @@ const EditorToolbar = () => {
328
416
  const { state, dispatch, createNewEntity, undoStack } = useEditor();
329
417
  const [showAddMenu, setShowAddMenu] = React.useState(false);
330
418
  const tools = ['select', 'move', 'rotate', 'scale', 'pan'];
331
- const toolIcons = { select: '⊕', move: '✥', rotate: '↻', scale: '⇲', pan: '✋' };
419
+ const toolIcons = {
420
+ select: '⊕',
421
+ move: '✥',
422
+ rotate: '↻',
423
+ scale: '⇲',
424
+ pan: '✋',
425
+ };
332
426
  const btnStyle = (active) => ({
333
427
  background: active ? '#585b70' : 'transparent',
334
- border: 'none', color: '#cdd6f4', padding: '4px 8px',
335
- borderRadius: 4, cursor: 'pointer', fontSize: 14,
428
+ border: 'none',
429
+ color: '#cdd6f4',
430
+ padding: '4px 8px',
431
+ borderRadius: 4,
432
+ cursor: 'pointer',
433
+ fontSize: 14,
336
434
  });
337
435
  return (jsxRuntime.jsxs("div", { style: {
338
- display: 'flex', alignItems: 'center', gap: 4,
339
- padding: '4px 8px', borderBottom: '1px solid #45475a',
436
+ display: 'flex',
437
+ alignItems: 'center',
438
+ gap: 4,
439
+ padding: '4px 8px',
440
+ borderBottom: '1px solid #45475a',
340
441
  backgroundColor: '#181825',
341
- }, children: [tools.map(t => (jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'SET_TOOL', tool: t }), style: btnStyle(state.tool === t), title: `${t} (${t[0].toUpperCase()})`, children: toolIcons[t] }, t))), jsxRuntime.jsx("div", { style: { width: 1, height: 20, background: '#45475a', margin: '0 4px' } }), jsxRuntime.jsxs("div", { style: { position: 'relative' }, children: [jsxRuntime.jsx("button", { onClick: () => setShowAddMenu(!showAddMenu), style: { ...btnStyle(false), fontSize: 12, padding: '4px 10px', background: '#2f9e44', color: '#fff' }, children: "+ Entity" }), showAddMenu && (jsxRuntime.jsx("div", { style: {
342
- position: 'absolute', top: '100%', left: 0, zIndex: 100,
343
- background: '#1e1e2e', border: '1px solid #45475a', borderRadius: 4,
344
- padding: 4, minWidth: 150, marginTop: 2,
345
- }, children: ENTITY_PRESETS.map(p => (jsxRuntime.jsxs("button", { onClick: () => { createNewEntity(p.label, p.components); setShowAddMenu(false); }, style: {
346
- display: 'block', width: '100%', textAlign: 'left',
347
- background: 'transparent', border: 'none', color: '#cdd6f4',
348
- padding: '4px 8px', cursor: 'pointer', borderRadius: 2, fontSize: 12,
349
- }, children: [p.icon, " ", p.label] }, p.label))) }))] }), jsxRuntime.jsx("div", { style: { width: 1, height: 20, background: '#45475a', margin: '0 4px' } }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'SET_MODE', mode: state.mode === 'play' ? 'edit' : 'play' }), style: btnStyle(state.mode === 'play'), title: "Play/Stop", children: state.mode === 'play' ? '⏹' : '▶' }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'SET_MODE', mode: state.mode === 'pause' ? 'play' : 'pause' }), style: btnStyle(state.mode === 'pause'), title: "Pause", disabled: state.mode === 'edit', children: "\u23F8" }), jsxRuntime.jsx("div", { style: { width: 1, height: 20, background: '#45475a', margin: '0 4px' } }), jsxRuntime.jsx("button", { onClick: () => undoStack.undo(), style: btnStyle(false), title: "Undo (Ctrl+Z)", disabled: !undoStack.canUndo, children: "\u21A9" }), jsxRuntime.jsx("button", { onClick: () => undoStack.redo(), style: btnStyle(false), title: "Redo (Ctrl+Y)", disabled: !undoStack.canRedo, children: "\u21AA" }), jsxRuntime.jsx("div", { style: { width: 1, height: 20, background: '#45475a', margin: '0 4px' } }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'SET_ZOOM', zoom: state.zoom * 0.8 }), style: btnStyle(false), title: "Zoom Out", children: "\u2212" }), jsxRuntime.jsxs("span", { style: { fontSize: 11, minWidth: 40, textAlign: 'center' }, children: [(state.zoom * 100) | 0, "%"] }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'SET_ZOOM', zoom: state.zoom * 1.25 }), style: btnStyle(false), title: "Zoom In", children: "+" }), jsxRuntime.jsx("button", { onClick: () => { dispatch({ type: 'SET_ZOOM', zoom: 1 }); dispatch({ type: 'SET_PAN', offset: { x: 0, y: 0 } }); }, style: btnStyle(false), title: "Reset View", children: "\u2302" }), jsxRuntime.jsx("div", { style: { width: 1, height: 20, background: '#45475a', margin: '0 4px' } }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'TOGGLE_GRID' }), style: btnStyle(state.showGrid), title: "Toggle Grid (Ctrl+G)", children: "\u25A6" }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'TOGGLE_SNAP' }), style: btnStyle(state.snapToGrid), title: "Toggle Snap", children: "\u229E" }), jsxRuntime.jsx("div", { style: { flex: 1 } }), jsxRuntime.jsxs("span", { style: { fontSize: 11, opacity: 0.6 }, children: [state.mode.toUpperCase(), " | ", state.tool, " | ", state.dirty ? '● Unsaved' : '✓ Saved'] })] }));
442
+ }, children: [tools.map((t) => (jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'SET_TOOL', tool: t }), style: btnStyle(state.tool === t), title: `${t} (${t[0].toUpperCase()})`, children: toolIcons[t] }, t))), jsxRuntime.jsx("div", { style: { width: 1, height: 20, background: '#45475a', margin: '0 4px' } }), jsxRuntime.jsxs("div", { style: { position: 'relative' }, children: [jsxRuntime.jsx("button", { onClick: () => setShowAddMenu(!showAddMenu), style: {
443
+ ...btnStyle(false),
444
+ fontSize: 12,
445
+ padding: '4px 10px',
446
+ background: '#2f9e44',
447
+ color: '#fff',
448
+ }, children: "+ Entity" }), showAddMenu && (jsxRuntime.jsx("div", { style: {
449
+ position: 'absolute',
450
+ top: '100%',
451
+ left: 0,
452
+ zIndex: 100,
453
+ background: '#1e1e2e',
454
+ border: '1px solid #45475a',
455
+ borderRadius: 4,
456
+ padding: 4,
457
+ minWidth: 150,
458
+ marginTop: 2,
459
+ }, children: ENTITY_PRESETS.map((p) => (jsxRuntime.jsxs("button", { onClick: () => {
460
+ createNewEntity(p.label, p.components);
461
+ setShowAddMenu(false);
462
+ }, style: {
463
+ display: 'block',
464
+ width: '100%',
465
+ textAlign: 'left',
466
+ background: 'transparent',
467
+ border: 'none',
468
+ color: '#cdd6f4',
469
+ padding: '4px 8px',
470
+ cursor: 'pointer',
471
+ borderRadius: 2,
472
+ fontSize: 12,
473
+ }, children: [p.icon, " ", p.label] }, p.label))) }))] }), jsxRuntime.jsx("div", { style: { width: 1, height: 20, background: '#45475a', margin: '0 4px' } }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'SET_MODE', mode: state.mode === 'play' ? 'edit' : 'play' }), style: btnStyle(state.mode === 'play'), title: "Play/Stop", children: state.mode === 'play' ? '⏹' : '▶' }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'SET_MODE', mode: state.mode === 'pause' ? 'play' : 'pause' }), style: btnStyle(state.mode === 'pause'), title: "Pause", disabled: state.mode === 'edit', children: "\u23F8" }), jsxRuntime.jsx("div", { style: { width: 1, height: 20, background: '#45475a', margin: '0 4px' } }), jsxRuntime.jsx("button", { onClick: () => undoStack.undo(), style: btnStyle(false), title: "Undo (Ctrl+Z)", disabled: !undoStack.canUndo, children: "\u21A9" }), jsxRuntime.jsx("button", { onClick: () => undoStack.redo(), style: btnStyle(false), title: "Redo (Ctrl+Y)", disabled: !undoStack.canRedo, children: "\u21AA" }), jsxRuntime.jsx("div", { style: { width: 1, height: 20, background: '#45475a', margin: '0 4px' } }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'SET_ZOOM', zoom: state.zoom * 0.8 }), style: btnStyle(false), title: "Zoom Out", children: "\u2212" }), jsxRuntime.jsxs("span", { style: { fontSize: 11, minWidth: 40, textAlign: 'center' }, children: [(state.zoom * 100) | 0, "%"] }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'SET_ZOOM', zoom: state.zoom * 1.25 }), style: btnStyle(false), title: "Zoom In", children: "+" }), jsxRuntime.jsx("button", { onClick: () => {
474
+ dispatch({ type: 'SET_ZOOM', zoom: 1 });
475
+ dispatch({ type: 'SET_PAN', offset: { x: 0, y: 0 } });
476
+ }, style: btnStyle(false), title: "Reset View", children: "\u2302" }), jsxRuntime.jsx("div", { style: { width: 1, height: 20, background: '#45475a', margin: '0 4px' } }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'TOGGLE_GRID' }), style: btnStyle(state.showGrid), title: "Toggle Grid (Ctrl+G)", children: "\u25A6" }), jsxRuntime.jsx("button", { onClick: () => dispatch({ type: 'TOGGLE_SNAP' }), style: btnStyle(state.snapToGrid), title: "Toggle Snap", children: "\u229E" }), jsxRuntime.jsx("div", { style: { flex: 1 } }), jsxRuntime.jsxs("span", { style: { fontSize: 11, opacity: 0.6 }, children: [state.mode.toUpperCase(), " | ", state.tool, " | ", state.dirty ? '● Unsaved' : '✓ Saved'] })] }));
350
477
  };
351
478
  const EditorPanelContainer = ({ title, width, height, position, children }) => {
352
479
  const isHorizontal = position === 'bottom';
@@ -356,42 +483,73 @@ const EditorPanelContainer = ({ title, width, height, position, children }) => {
356
483
  flex: isHorizontal ? undefined : undefined,
357
484
  borderLeft: position === 'right' ? '1px solid #45475a' : undefined,
358
485
  borderRight: position === 'left' ? '1px solid #45475a' : undefined,
359
- display: 'flex', flexDirection: 'column', overflow: 'hidden',
486
+ display: 'flex',
487
+ flexDirection: 'column',
488
+ overflow: 'hidden',
360
489
  backgroundColor: '#1e1e2e',
361
490
  }, children: [jsxRuntime.jsx("div", { style: {
362
- padding: '4px 8px', fontSize: 11, fontWeight: 'bold',
363
- backgroundColor: '#181825', borderBottom: '1px solid #313244',
364
- textTransform: 'uppercase', letterSpacing: 1,
491
+ padding: '4px 8px',
492
+ fontSize: 11,
493
+ fontWeight: 'bold',
494
+ backgroundColor: '#181825',
495
+ borderBottom: '1px solid #313244',
496
+ textTransform: 'uppercase',
497
+ letterSpacing: 1,
365
498
  }, children: title }), jsxRuntime.jsx("div", { style: { flex: 1, overflow: 'auto', padding: 4 }, children: children })] }));
366
499
  };
367
500
  const HierarchyPanel = () => {
368
501
  const { world, state, dispatch, createNewEntity, refreshWorld, addLog } = useEditor();
369
- const entities = world.allEntities().filter(e => e.parent == null);
502
+ const entities = world.allEntities().filter((e) => e.parent == null);
370
503
  const getIcon = (id) => {
371
- if (world.hasComponent(id, 'CameraTarget'))
504
+ if (world.hasComponent(id, 'CameraTarget')) {
372
505
  return '📷';
373
- if (world.hasComponent(id, 'Sprite'))
506
+ }
507
+ if (world.hasComponent(id, 'Sprite')) {
374
508
  return '🖼';
375
- if (world.hasComponent(id, 'Text'))
509
+ }
510
+ if (world.hasComponent(id, 'Text')) {
376
511
  return 'T';
377
- if (world.hasComponent(id, 'Shape'))
512
+ }
513
+ if (world.hasComponent(id, 'Shape')) {
378
514
  return '■';
379
- if (world.hasComponent(id, 'ParticleEmitter'))
515
+ }
516
+ if (world.hasComponent(id, 'ParticleEmitter')) {
380
517
  return '✨';
381
- if (world.hasComponent(id, 'AudioSource'))
518
+ }
519
+ if (world.hasComponent(id, 'AudioSource')) {
382
520
  return '🔊';
521
+ }
383
522
  return '○';
384
523
  };
385
524
  const renderEntity = (e, depth) => {
386
525
  const isSelected = state.selectedEntities.includes(e.id);
387
- const children = e.children.map(cid => world.getEntity(cid)).filter(Boolean);
526
+ const children = e.children.map((cid) => world.getEntity(cid)).filter(Boolean);
388
527
  return (jsxRuntime.jsxs(React.Fragment, { children: [jsxRuntime.jsxs("div", { onClick: () => dispatch({ type: 'SELECT', entities: [e.id] }), style: {
389
- padding: '2px 6px', paddingLeft: 6 + depth * 16, cursor: 'pointer',
528
+ padding: '2px 6px',
529
+ paddingLeft: 6 + depth * 16,
530
+ cursor: 'pointer',
390
531
  backgroundColor: isSelected ? '#45475a' : 'transparent',
391
- borderRadius: 2, display: 'flex', alignItems: 'center', gap: 4, fontSize: 12,
392
- }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, width: 14, textAlign: 'center' }, children: getIcon(e.id) }), jsxRuntime.jsx("span", { style: { flex: 1 }, children: e.name }), jsxRuntime.jsxs("span", { style: { opacity: 0.3, fontSize: 10 }, children: ["#", e.id] })] }), children.map(c => renderEntity(c, depth + 1))] }, e.id));
532
+ borderRadius: 2,
533
+ display: 'flex',
534
+ alignItems: 'center',
535
+ gap: 4,
536
+ fontSize: 12,
537
+ }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, width: 14, textAlign: 'center' }, children: getIcon(e.id) }), jsxRuntime.jsx("span", { style: { flex: 1 }, children: e.name }), jsxRuntime.jsxs("span", { style: { opacity: 0.3, fontSize: 10 }, children: ["#", e.id] })] }), children.map((c) => renderEntity(c, depth + 1))] }, e.id));
393
538
  };
394
- return (jsxRuntime.jsxs("div", { children: [jsxRuntime.jsxs("div", { style: { padding: '2px 6px', fontSize: 10, display: 'flex', gap: 4, borderBottom: '1px solid #313244', marginBottom: 4 }, children: [jsxRuntime.jsx("button", { onClick: () => createNewEntity('Entity'), style: { background: 'transparent', border: 'none', color: '#a6adc8', cursor: 'pointer', fontSize: 10 }, title: "Add Entity", children: "+" }), jsxRuntime.jsx("button", { onClick: () => {
539
+ return (jsxRuntime.jsxs("div", { children: [jsxRuntime.jsxs("div", { style: {
540
+ padding: '2px 6px',
541
+ fontSize: 10,
542
+ display: 'flex',
543
+ gap: 4,
544
+ borderBottom: '1px solid #313244',
545
+ marginBottom: 4,
546
+ }, children: [jsxRuntime.jsx("button", { onClick: () => createNewEntity('Entity'), style: {
547
+ background: 'transparent',
548
+ border: 'none',
549
+ color: '#a6adc8',
550
+ cursor: 'pointer',
551
+ fontSize: 10,
552
+ }, title: "Add Entity", children: "+" }), jsxRuntime.jsx("button", { onClick: () => {
395
553
  if (state.selectedEntities.length > 0) {
396
554
  const id = state.selectedEntities[0];
397
555
  const child = world.createEntity('Child');
@@ -402,58 +560,160 @@ const HierarchyPanel = () => {
402
560
  refreshWorld();
403
561
  addLog('info', `Created child entity (#${child.id})`);
404
562
  }
405
- }, style: { background: 'transparent', border: 'none', color: '#a6adc8', cursor: 'pointer', fontSize: 10 }, title: "Add Child", disabled: state.selectedEntities.length === 0, children: "\u21B3+" }), jsxRuntime.jsx("span", { style: { flex: 1 } }), jsxRuntime.jsxs("span", { style: { color: '#6c7086' }, children: [world.entityCount, " entities"] })] }), entities.map(e => renderEntity(e, 0))] }));
563
+ }, style: {
564
+ background: 'transparent',
565
+ border: 'none',
566
+ color: '#a6adc8',
567
+ cursor: 'pointer',
568
+ fontSize: 10,
569
+ }, title: "Add Child", disabled: state.selectedEntities.length === 0, children: "\u21B3+" }), jsxRuntime.jsx("span", { style: { flex: 1 } }), jsxRuntime.jsxs("span", { style: { color: '#6c7086' }, children: [world.entityCount, " entities"] })] }), entities.map((e) => renderEntity(e, 0))] }));
406
570
  };
407
571
  /* ── Inspector sub-editors ─────────────────────────────────────── */
408
572
  const fieldStyle = {
409
- width: '100%', background: '#313244', color: '#cdd6f4', border: '1px solid #45475a',
410
- borderRadius: 3, padding: '2px 4px', fontSize: 11,
573
+ width: '100%',
574
+ background: '#313244',
575
+ color: '#cdd6f4',
576
+ border: '1px solid #45475a',
577
+ borderRadius: 3,
578
+ padding: '2px 4px',
579
+ fontSize: 11,
411
580
  };
412
- const NumField = ({ label, value, onChange, step = 1 }) => (jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: label }), jsxRuntime.jsx("input", { type: "number", value: Math.round(value * 1000) / 1000, step: step, onChange: e => onChange(Number(e.target.value)), style: { ...fieldStyle, flex: 1 } })] }));
413
- const ColField = ({ label, color, onChange }) => (jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: label }), jsxRuntime.jsx("input", { type: "color", value: colorToHex(color), onChange: e => onChange(cssToColor(e.target.value)), style: { width: 28, height: 20, border: 'none', padding: 0, cursor: 'pointer' } }), jsxRuntime.jsx("span", { style: { fontSize: 9, color: '#6c7086' }, children: colorToHex(color) })] }));
581
+ const NumField = ({ label, value, onChange, step = 1 }) => (jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: label }), jsxRuntime.jsx("input", { type: "number", value: Math.round(value * 1000) / 1000, step: step, onChange: (e) => onChange(Number(e.target.value)), style: { ...fieldStyle, flex: 1 } })] }));
582
+ const ColField = ({ label, color, onChange }) => (jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: label }), jsxRuntime.jsx("input", { type: "color", value: colorToHex(color), onChange: (e) => onChange(cssToColor(e.target.value)), style: { width: 28, height: 20, border: 'none', padding: 0, cursor: 'pointer' } }), jsxRuntime.jsx("span", { style: { fontSize: 9, color: '#6c7086' }, children: colorToHex(color) })] }));
414
583
  const TransformEditor = ({ entityId }) => {
415
584
  const { world, refreshWorld, dispatch } = useEditor();
416
585
  const t = world.getComponent(entityId, 'Transform');
417
- if (!t)
586
+ if (!t) {
418
587
  return null;
419
- const update = (fn) => { fn(); dispatch({ type: 'MARK_DIRTY' }); refreshWorld(); };
420
- return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(NumField, { label: "Pos X", value: t.position.x, onChange: v => update(() => { t.position.x = v; }) }), jsxRuntime.jsx(NumField, { label: "Pos Y", value: t.position.y, onChange: v => update(() => { t.position.y = v; }) }), jsxRuntime.jsx(NumField, { label: "Rotation", value: t.rotation * (180 / Math.PI), step: 5, onChange: v => update(() => { t.rotation = v * (Math.PI / 180); }) }), jsxRuntime.jsx(NumField, { label: "Scale X", value: t.scale.x, step: 0.1, onChange: v => update(() => { t.scale.x = v; }) }), jsxRuntime.jsx(NumField, { label: "Scale Y", value: t.scale.y, step: 0.1, onChange: v => update(() => { t.scale.y = v; }) })] }));
588
+ }
589
+ const update = (fn) => {
590
+ fn();
591
+ dispatch({ type: 'MARK_DIRTY' });
592
+ refreshWorld();
593
+ };
594
+ return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(NumField, { label: "Pos X", value: t.position.x, onChange: (v) => update(() => {
595
+ t.position.x = v;
596
+ }) }), jsxRuntime.jsx(NumField, { label: "Pos Y", value: t.position.y, onChange: (v) => update(() => {
597
+ t.position.y = v;
598
+ }) }), jsxRuntime.jsx(NumField, { label: "Rotation", value: t.rotation * (180 / Math.PI), step: 5, onChange: (v) => update(() => {
599
+ t.rotation = v * (Math.PI / 180);
600
+ }) }), jsxRuntime.jsx(NumField, { label: "Scale X", value: t.scale.x, step: 0.1, onChange: (v) => update(() => {
601
+ t.scale.x = v;
602
+ }) }), jsxRuntime.jsx(NumField, { label: "Scale Y", value: t.scale.y, step: 0.1, onChange: (v) => update(() => {
603
+ t.scale.y = v;
604
+ }) })] }));
421
605
  };
422
606
  const ShapeEditor = ({ entityId }) => {
423
607
  const { world, refreshWorld, dispatch } = useEditor();
424
608
  const s = world.getComponent(entityId, 'Shape');
425
- if (!s)
609
+ if (!s) {
426
610
  return null;
427
- const update = (fn) => { fn(); dispatch({ type: 'MARK_DIRTY' }); refreshWorld(); };
428
- return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: "Shape" }), jsxRuntime.jsxs("select", { value: s.shape, onChange: e => update(() => { s.shape = e.target.value; }), style: { ...fieldStyle, flex: 1 }, children: [jsxRuntime.jsx("option", { value: "rect", children: "Rectangle" }), jsxRuntime.jsx("option", { value: "circle", children: "Circle" }), jsxRuntime.jsx("option", { value: "polygon", children: "Polygon" }), jsxRuntime.jsx("option", { value: "line", children: "Line" })] })] }), s.shape !== 'circle' && jsxRuntime.jsx(NumField, { label: "Width", value: s.width, onChange: v => update(() => { s.width = v; }) }), s.shape !== 'circle' && jsxRuntime.jsx(NumField, { label: "Height", value: s.height, onChange: v => update(() => { s.height = v; }) }), s.shape === 'circle' && jsxRuntime.jsx(NumField, { label: "Radius", value: s.radius, onChange: v => update(() => { s.radius = v; }) }), jsxRuntime.jsx(ColField, { label: "Fill", color: s.fillColor, onChange: c => update(() => { Object.assign(s.fillColor, c); }) }), jsxRuntime.jsx(ColField, { label: "Stroke", color: s.strokeColor, onChange: c => update(() => { Object.assign(s.strokeColor, c); }) }), jsxRuntime.jsx(NumField, { label: "Stroke W", value: s.strokeWidth, onChange: v => update(() => { s.strokeWidth = v; }) })] }));
611
+ }
612
+ const update = (fn) => {
613
+ fn();
614
+ dispatch({ type: 'MARK_DIRTY' });
615
+ refreshWorld();
616
+ };
617
+ return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: "Shape" }), jsxRuntime.jsxs("select", { value: s.shape, onChange: (e) => update(() => {
618
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any -- mutates discriminated-union variant; consumer narrows by `s.shape`
619
+ s.shape = e.target.value;
620
+ }), style: { ...fieldStyle, flex: 1 }, children: [jsxRuntime.jsx("option", { value: "rect", children: "Rectangle" }), jsxRuntime.jsx("option", { value: "circle", children: "Circle" }), jsxRuntime.jsx("option", { value: "polygon", children: "Polygon" }), jsxRuntime.jsx("option", { value: "line", children: "Line" })] })] }), s.shape !== 'circle' && (jsxRuntime.jsx(NumField, { label: "Width", value: s.width, onChange: (v) => update(() => {
621
+ s.width = v;
622
+ }) })), s.shape !== 'circle' && (jsxRuntime.jsx(NumField, { label: "Height", value: s.height, onChange: (v) => update(() => {
623
+ s.height = v;
624
+ }) })), s.shape === 'circle' && (jsxRuntime.jsx(NumField, { label: "Radius", value: s.radius, onChange: (v) => update(() => {
625
+ s.radius = v;
626
+ }) })), jsxRuntime.jsx(ColField, { label: "Fill", color: s.fillColor, onChange: (c) => update(() => {
627
+ Object.assign(s.fillColor, c);
628
+ }) }), jsxRuntime.jsx(ColField, { label: "Stroke", color: s.strokeColor, onChange: (c) => update(() => {
629
+ Object.assign(s.strokeColor, c);
630
+ }) }), jsxRuntime.jsx(NumField, { label: "Stroke W", value: s.strokeWidth, onChange: (v) => update(() => {
631
+ s.strokeWidth = v;
632
+ }) })] }));
429
633
  };
430
634
  const SpriteEditor = ({ entityId }) => {
431
635
  const { world, refreshWorld, dispatch } = useEditor();
432
636
  const s = world.getComponent(entityId, 'Sprite');
433
- if (!s)
637
+ if (!s) {
434
638
  return null;
435
- const update = (fn) => { fn(); dispatch({ type: 'MARK_DIRTY' }); refreshWorld(); };
436
- return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(NumField, { label: "Width", value: s.width, onChange: v => update(() => { s.width = v; }) }), jsxRuntime.jsx(NumField, { label: "Height", value: s.height, onChange: v => update(() => { s.height = v; }) }), jsxRuntime.jsx(ColField, { label: "Tint", color: s.tint, onChange: c => update(() => { Object.assign(s.tint, c); }) }), jsxRuntime.jsx(NumField, { label: "Opacity", value: s.opacity, step: 0.1, onChange: v => update(() => { s.opacity = Math.max(0, Math.min(1, v)); }) }), jsxRuntime.jsx(NumField, { label: "Layer", value: s.layer, step: 1, onChange: v => update(() => { s.layer = v; }) }), jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: "Visible" }), jsxRuntime.jsx("input", { type: "checkbox", checked: s.visible, onChange: e => update(() => { s.visible = e.target.checked; }) })] })] }));
639
+ }
640
+ const update = (fn) => {
641
+ fn();
642
+ dispatch({ type: 'MARK_DIRTY' });
643
+ refreshWorld();
644
+ };
645
+ return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsx(NumField, { label: "Width", value: s.width, onChange: (v) => update(() => {
646
+ s.width = v;
647
+ }) }), jsxRuntime.jsx(NumField, { label: "Height", value: s.height, onChange: (v) => update(() => {
648
+ s.height = v;
649
+ }) }), jsxRuntime.jsx(ColField, { label: "Tint", color: s.tint, onChange: (c) => update(() => {
650
+ Object.assign(s.tint, c);
651
+ }) }), jsxRuntime.jsx(NumField, { label: "Opacity", value: s.opacity, step: 0.1, onChange: (v) => update(() => {
652
+ s.opacity = Math.max(0, Math.min(1, v));
653
+ }) }), jsxRuntime.jsx(NumField, { label: "Layer", value: s.layer, step: 1, onChange: (v) => update(() => {
654
+ s.layer = v;
655
+ }) }), jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: "Visible" }), jsxRuntime.jsx("input", { type: "checkbox", checked: s.visible, onChange: (e) => update(() => {
656
+ s.visible = e.target.checked;
657
+ }) })] })] }));
437
658
  };
438
659
  const TextEditor = ({ entityId }) => {
439
660
  const { world, refreshWorld, dispatch } = useEditor();
440
661
  const t = world.getComponent(entityId, 'Text');
441
- if (!t)
662
+ if (!t) {
442
663
  return null;
443
- const update = (fn) => { fn(); dispatch({ type: 'MARK_DIRTY' }); refreshWorld(); };
444
- return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsxs("div", { style: { marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, color: '#a6adc8' }, children: "Text" }), jsxRuntime.jsx("input", { type: "text", value: t.text, onChange: e => update(() => { t.text = e.target.value; }), style: { ...fieldStyle, width: '100%' } })] }), jsxRuntime.jsx(NumField, { label: "Font Size", value: t.fontSize, onChange: v => update(() => { t.fontSize = v; }) }), jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: "Align" }), jsxRuntime.jsxs("select", { value: t.align, onChange: e => update(() => { t.align = e.target.value; }), style: { ...fieldStyle, flex: 1 }, children: [jsxRuntime.jsx("option", { value: "left", children: "Left" }), jsxRuntime.jsx("option", { value: "center", children: "Center" }), jsxRuntime.jsx("option", { value: "right", children: "Right" })] })] }), jsxRuntime.jsx(ColField, { label: "Color", color: t.color, onChange: c => update(() => { Object.assign(t.color, c); }) })] }));
664
+ }
665
+ const update = (fn) => {
666
+ fn();
667
+ dispatch({ type: 'MARK_DIRTY' });
668
+ refreshWorld();
669
+ };
670
+ return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsxs("div", { style: { marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, color: '#a6adc8' }, children: "Text" }), jsxRuntime.jsx("input", { type: "text", value: t.text, onChange: (e) => update(() => {
671
+ t.text = e.target.value;
672
+ }), style: { ...fieldStyle, width: '100%' } })] }), jsxRuntime.jsx(NumField, { label: "Font Size", value: t.fontSize, onChange: (v) => update(() => {
673
+ t.fontSize = v;
674
+ }) }), jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: "Align" }), jsxRuntime.jsxs("select", { value: t.align, onChange: (e) => update(() => {
675
+ t.align = e.target.value;
676
+ }), style: { ...fieldStyle, flex: 1 }, children: [jsxRuntime.jsx("option", { value: "left", children: "Left" }), jsxRuntime.jsx("option", { value: "center", children: "Center" }), jsxRuntime.jsx("option", { value: "right", children: "Right" })] })] }), jsxRuntime.jsx(ColField, { label: "Color", color: t.color, onChange: (c) => update(() => {
677
+ Object.assign(t.color, c);
678
+ }) })] }));
445
679
  };
446
680
  const RigidBodyEditor = ({ entityId }) => {
447
681
  const { world, refreshWorld, dispatch } = useEditor();
448
682
  const rb = world.getComponent(entityId, 'RigidBody2D');
449
- if (!rb)
683
+ if (!rb) {
450
684
  return null;
451
- const update = (fn) => { fn(); dispatch({ type: 'MARK_DIRTY' }); refreshWorld(); };
452
- return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: "Body" }), jsxRuntime.jsxs("select", { value: rb.bodyType, onChange: e => update(() => { rb.bodyType = e.target.value; }), style: { ...fieldStyle, flex: 1 }, children: [jsxRuntime.jsx("option", { value: "dynamic", children: "Dynamic" }), jsxRuntime.jsx("option", { value: "static", children: "Static" }), jsxRuntime.jsx("option", { value: "kinematic", children: "Kinematic" })] })] }), jsxRuntime.jsx(NumField, { label: "Mass", value: rb.mass, step: 0.1, onChange: v => update(() => { rb.mass = v; }) }), jsxRuntime.jsx(NumField, { label: "Drag", value: rb.drag, step: 0.01, onChange: v => update(() => { rb.drag = v; }) }), jsxRuntime.jsx(NumField, { label: "Bounce", value: rb.restitution, step: 0.05, onChange: v => update(() => { rb.restitution = v; }) }), jsxRuntime.jsx(NumField, { label: "Friction", value: rb.friction, step: 0.05, onChange: v => update(() => { rb.friction = v; }) }), jsxRuntime.jsx(NumField, { label: "Gravity", value: rb.gravityScale, step: 0.1, onChange: v => update(() => { rb.gravityScale = v; }) })] }));
685
+ }
686
+ const update = (fn) => {
687
+ fn();
688
+ dispatch({ type: 'MARK_DIRTY' });
689
+ refreshWorld();
690
+ };
691
+ return (jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', gap: 4, marginBottom: 2 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 10, minWidth: 50, color: '#a6adc8' }, children: "Body" }), jsxRuntime.jsxs("select", { value: rb.bodyType, onChange: (e) => update(() => {
692
+ rb.bodyType = e.target.value;
693
+ }), style: { ...fieldStyle, flex: 1 }, children: [jsxRuntime.jsx("option", { value: "dynamic", children: "Dynamic" }), jsxRuntime.jsx("option", { value: "static", children: "Static" }), jsxRuntime.jsx("option", { value: "kinematic", children: "Kinematic" })] })] }), jsxRuntime.jsx(NumField, { label: "Mass", value: rb.mass, step: 0.1, onChange: (v) => update(() => {
694
+ rb.mass = v;
695
+ }) }), jsxRuntime.jsx(NumField, { label: "Drag", value: rb.drag, step: 0.01, onChange: (v) => update(() => {
696
+ rb.drag = v;
697
+ }) }), jsxRuntime.jsx(NumField, { label: "Bounce", value: rb.restitution, step: 0.05, onChange: (v) => update(() => {
698
+ rb.restitution = v;
699
+ }) }), jsxRuntime.jsx(NumField, { label: "Friction", value: rb.friction, step: 0.05, onChange: (v) => update(() => {
700
+ rb.friction = v;
701
+ }) }), jsxRuntime.jsx(NumField, { label: "Gravity", value: rb.gravityScale, step: 0.1, onChange: (v) => update(() => {
702
+ rb.gravityScale = v;
703
+ }) })] }));
453
704
  };
454
705
  const ADDABLE_COMPONENTS = [
455
- 'Transform', 'Sprite', 'Shape', 'Collider2D', 'RigidBody2D',
456
- 'Text', 'Animator', 'Script', 'AudioSource', 'ParticleEmitter', 'CameraTarget',
706
+ 'Transform',
707
+ 'Sprite',
708
+ 'Shape',
709
+ 'Collider2D',
710
+ 'RigidBody2D',
711
+ 'Text',
712
+ 'Animator',
713
+ 'Script',
714
+ 'AudioSource',
715
+ 'ParticleEmitter',
716
+ 'CameraTarget',
457
717
  ];
458
718
  const InspectorPanel = () => {
459
719
  const { state, world, dispatch, refreshWorld, addLog } = useEditor();
@@ -463,10 +723,11 @@ const InspectorPanel = () => {
463
723
  }
464
724
  const entityId = state.selectedEntities[0];
465
725
  const entity = world.getEntity(entityId);
466
- if (!entity)
726
+ if (!entity) {
467
727
  return null;
728
+ }
468
729
  const compTypes = ADDABLE_COMPONENTS;
469
- const presentComponents = compTypes.filter(type => world.hasComponent(entityId, type));
730
+ const presentComponents = compTypes.filter((type) => world.hasComponent(entityId, type));
470
731
  const componentEditors = {
471
732
  Transform: TransformEditor,
472
733
  Shape: ShapeEditor,
@@ -474,35 +735,87 @@ const InspectorPanel = () => {
474
735
  Text: TextEditor,
475
736
  RigidBody2D: RigidBodyEditor,
476
737
  };
477
- return (jsxRuntime.jsxs("div", { children: [jsxRuntime.jsxs("div", { style: { padding: '4px 0', marginBottom: 4, borderBottom: '1px solid #313244' }, children: [jsxRuntime.jsx("input", { type: "text", value: entity.name, onChange: e => { entity.name = e.target.value; dispatch({ type: 'MARK_DIRTY' }); refreshWorld(); }, style: { ...fieldStyle, fontWeight: 'bold', fontSize: 13 } }), jsxRuntime.jsxs("span", { style: { fontSize: 9, color: '#6c7086' }, children: ["ID: ", entityId] })] }), presentComponents.map(type => {
738
+ return (jsxRuntime.jsxs("div", { children: [jsxRuntime.jsxs("div", { style: { padding: '4px 0', marginBottom: 4, borderBottom: '1px solid #313244' }, children: [jsxRuntime.jsx("input", { type: "text", value: entity.name, onChange: (e) => {
739
+ entity.name = e.target.value;
740
+ dispatch({ type: 'MARK_DIRTY' });
741
+ refreshWorld();
742
+ }, style: { ...fieldStyle, fontWeight: 'bold', fontSize: 13 } }), jsxRuntime.jsxs("span", { style: { fontSize: 9, color: '#6c7086' }, children: ["ID: ", entityId] })] }), presentComponents.map((type) => {
478
743
  const Editor = componentEditors[type];
479
- return (jsxRuntime.jsxs("div", { style: { marginBottom: 8, borderTop: '1px solid #313244', paddingTop: 4 }, children: [jsxRuntime.jsxs("div", { style: { display: 'flex', alignItems: 'center', justifyContent: 'space-between', marginBottom: 4 }, children: [jsxRuntime.jsx("span", { style: { fontSize: 11, fontWeight: 'bold', textTransform: 'uppercase', color: '#89b4fa' }, children: type }), type !== 'Transform' && (jsxRuntime.jsx("button", { onClick: () => {
744
+ return (jsxRuntime.jsxs("div", { style: { marginBottom: 8, borderTop: '1px solid #313244', paddingTop: 4 }, children: [jsxRuntime.jsxs("div", { style: {
745
+ display: 'flex',
746
+ alignItems: 'center',
747
+ justifyContent: 'space-between',
748
+ marginBottom: 4,
749
+ }, children: [jsxRuntime.jsx("span", { style: {
750
+ fontSize: 11,
751
+ fontWeight: 'bold',
752
+ textTransform: 'uppercase',
753
+ color: '#89b4fa',
754
+ }, children: type }), type !== 'Transform' && (jsxRuntime.jsx("button", { onClick: () => {
480
755
  world.removeComponent(entityId, type);
481
756
  dispatch({ type: 'MARK_DIRTY' });
482
757
  refreshWorld();
483
758
  addLog('info', `Removed ${type} from #${entityId}`);
484
- }, style: { background: 'transparent', border: 'none', color: '#f38ba8', cursor: 'pointer', fontSize: 10 }, title: `Remove ${type}`, children: "\u2715" }))] }), Editor ? jsxRuntime.jsx(Editor, { entityId: entityId }) : (jsxRuntime.jsx("pre", { style: { fontSize: 10, opacity: 0.6, whiteSpace: 'pre-wrap', margin: 0 }, children: JSON.stringify(world.getComponent(entityId, type), null, 2) }))] }, type));
485
- }), jsxRuntime.jsxs("div", { style: { marginTop: 8, position: 'relative' }, children: [jsxRuntime.jsx("button", { onClick: () => setShowAddComp(!showAddComp), style: { width: '100%', padding: '4px 8px', background: '#313244', border: '1px solid #45475a', color: '#cdd6f4', borderRadius: 4, cursor: 'pointer', fontSize: 11 }, children: "+ Add Component" }), showAddComp && (jsxRuntime.jsx("div", { style: {
486
- position: 'absolute', bottom: '100%', left: 0, right: 0,
487
- background: '#1e1e2e', border: '1px solid #45475a', borderRadius: 4,
488
- padding: 4, marginBottom: 2, maxHeight: 200, overflow: 'auto', zIndex: 10,
489
- }, children: compTypes.filter(t => !world.hasComponent(entityId, t)).map(type => (jsxRuntime.jsx("button", { onClick: () => {
759
+ }, style: {
760
+ background: 'transparent',
761
+ border: 'none',
762
+ color: '#f38ba8',
763
+ cursor: 'pointer',
764
+ fontSize: 10,
765
+ }, title: `Remove ${type}`, children: "\u2715" }))] }), Editor ? (jsxRuntime.jsx(Editor, { entityId: entityId })) : (jsxRuntime.jsx("pre", { style: { fontSize: 10, opacity: 0.6, whiteSpace: 'pre-wrap', margin: 0 }, children: JSON.stringify(world.getComponent(entityId, type), null, 2) }))] }, type));
766
+ }), jsxRuntime.jsxs("div", { style: { marginTop: 8, position: 'relative' }, children: [jsxRuntime.jsx("button", { onClick: () => setShowAddComp(!showAddComp), style: {
767
+ width: '100%',
768
+ padding: '4px 8px',
769
+ background: '#313244',
770
+ border: '1px solid #45475a',
771
+ color: '#cdd6f4',
772
+ borderRadius: 4,
773
+ cursor: 'pointer',
774
+ fontSize: 11,
775
+ }, children: "+ Add Component" }), showAddComp && (jsxRuntime.jsx("div", { style: {
776
+ position: 'absolute',
777
+ bottom: '100%',
778
+ left: 0,
779
+ right: 0,
780
+ background: '#1e1e2e',
781
+ border: '1px solid #45475a',
782
+ borderRadius: 4,
783
+ padding: 4,
784
+ marginBottom: 2,
785
+ maxHeight: 200,
786
+ overflow: 'auto',
787
+ zIndex: 10,
788
+ }, children: compTypes
789
+ .filter((t) => !world.hasComponent(entityId, t))
790
+ .map((type) => (jsxRuntime.jsx("button", { onClick: () => {
490
791
  world.addComponent(entityId, type);
491
792
  dispatch({ type: 'MARK_DIRTY' });
492
793
  refreshWorld();
493
794
  addLog('info', `Added ${type} to #${entityId}`);
494
795
  setShowAddComp(false);
495
796
  }, style: {
496
- display: 'block', width: '100%', textAlign: 'left',
497
- background: 'transparent', border: 'none', color: '#cdd6f4',
498
- padding: '3px 6px', cursor: 'pointer', borderRadius: 2, fontSize: 11,
797
+ display: 'block',
798
+ width: '100%',
799
+ textAlign: 'left',
800
+ background: 'transparent',
801
+ border: 'none',
802
+ color: '#cdd6f4',
803
+ padding: '3px 6px',
804
+ cursor: 'pointer',
805
+ borderRadius: 2,
806
+ fontSize: 11,
499
807
  }, children: type }, type))) }))] })] }));
500
808
  };
501
809
  const AssetPanel = () => {
502
810
  const [tab, setTab] = React.useState('all');
503
811
  const tabBtnStyle = (active) => ({
504
- padding: '2px 8px', border: 'none', borderRadius: '3px 3px 0 0', cursor: 'pointer', fontSize: 10,
505
- background: active ? '#313244' : 'transparent', color: active ? '#cdd6f4' : '#6c7086',
812
+ padding: '2px 8px',
813
+ border: 'none',
814
+ borderRadius: '3px 3px 0 0',
815
+ cursor: 'pointer',
816
+ fontSize: 10,
817
+ background: active ? '#313244' : 'transparent',
818
+ color: active ? '#cdd6f4' : '#6c7086',
506
819
  });
507
820
  const assets = [
508
821
  { name: 'player.png', type: 'sprites', icon: '🖼' },
@@ -515,11 +828,31 @@ const AssetPanel = () => {
515
828
  { name: 'playerController.ts', type: 'scripts', icon: '📜' },
516
829
  { name: 'enemyAI.ts', type: 'scripts', icon: '📜' },
517
830
  ];
518
- const filtered = tab === 'all' ? assets : assets.filter(a => a.type === tab);
519
- return (jsxRuntime.jsxs("div", { style: { display: 'flex', flexDirection: 'column', height: '100%' }, children: [jsxRuntime.jsxs("div", { style: { display: 'flex', gap: 2, padding: '2px 4px', borderBottom: '1px solid #313244' }, children: [['all', 'sprites', 'audio', 'scripts'].map(t => (jsxRuntime.jsx("button", { onClick: () => setTab(t), style: tabBtnStyle(tab === t), children: t.charAt(0).toUpperCase() + t.slice(1) }, t))), jsxRuntime.jsx("span", { style: { flex: 1 } }), jsxRuntime.jsx("button", { style: { background: 'transparent', border: 'none', color: '#a6adc8', cursor: 'pointer', fontSize: 10 }, title: "Import Asset", children: "+ Import" })] }), jsxRuntime.jsx("div", { style: { flex: 1, overflow: 'auto', display: 'grid', gridTemplateColumns: 'repeat(auto-fill, 80px)', gap: 4, padding: 4 }, children: filtered.map(a => (jsxRuntime.jsxs("div", { draggable: true, style: {
520
- display: 'flex', flexDirection: 'column', alignItems: 'center', gap: 2,
521
- padding: 4, background: '#313244', borderRadius: 4, cursor: 'grab',
522
- fontSize: 9, textAlign: 'center',
831
+ const filtered = tab === 'all' ? assets : assets.filter((a) => a.type === tab);
832
+ return (jsxRuntime.jsxs("div", { style: { display: 'flex', flexDirection: 'column', height: '100%' }, children: [jsxRuntime.jsxs("div", { style: { display: 'flex', gap: 2, padding: '2px 4px', borderBottom: '1px solid #313244' }, children: [['all', 'sprites', 'audio', 'scripts'].map((t) => (jsxRuntime.jsx("button", { onClick: () => setTab(t), style: tabBtnStyle(tab === t), children: t.charAt(0).toUpperCase() + t.slice(1) }, t))), jsxRuntime.jsx("span", { style: { flex: 1 } }), jsxRuntime.jsx("button", { style: {
833
+ background: 'transparent',
834
+ border: 'none',
835
+ color: '#a6adc8',
836
+ cursor: 'pointer',
837
+ fontSize: 10,
838
+ }, title: "Import Asset", children: "+ Import" })] }), jsxRuntime.jsx("div", { style: {
839
+ flex: 1,
840
+ overflow: 'auto',
841
+ display: 'grid',
842
+ gridTemplateColumns: 'repeat(auto-fill, 80px)',
843
+ gap: 4,
844
+ padding: 4,
845
+ }, children: filtered.map((a) => (jsxRuntime.jsxs("div", { draggable: true, style: {
846
+ display: 'flex',
847
+ flexDirection: 'column',
848
+ alignItems: 'center',
849
+ gap: 2,
850
+ padding: 4,
851
+ background: '#313244',
852
+ borderRadius: 4,
853
+ cursor: 'grab',
854
+ fontSize: 9,
855
+ textAlign: 'center',
523
856
  }, children: [jsxRuntime.jsx("span", { style: { fontSize: 20 }, children: a.icon }), jsxRuntime.jsx("span", { style: { wordBreak: 'break-all' }, children: a.name })] }, a.name))) })] }));
524
857
  };
525
858
  const ConsolePanel = () => {
@@ -527,14 +860,27 @@ const ConsolePanel = () => {
527
860
  const [filter, setFilter] = React.useState('all');
528
861
  const scrollRef = React.useRef(null);
529
862
  React.useEffect(() => {
530
- if (scrollRef.current)
863
+ if (scrollRef.current) {
531
864
  scrollRef.current.scrollTop = scrollRef.current.scrollHeight;
865
+ }
532
866
  }, [consoleLogs.length]);
533
- const filtered = filter === 'all' ? consoleLogs : consoleLogs.filter(l => l.level === filter);
867
+ const filtered = filter === 'all' ? consoleLogs : consoleLogs.filter((l) => l.level === filter);
534
868
  const levelColors = { info: '#89b4fa', warn: '#f9e2af', error: '#f38ba8' };
535
- const filterBtn = (f, label) => (jsxRuntime.jsx("button", { onClick: () => setFilter(f), style: { padding: '1px 6px', border: 'none', borderRadius: 2, cursor: 'pointer', fontSize: 9,
536
- background: filter === f ? '#313244' : 'transparent', color: filter === f ? '#cdd6f4' : '#6c7086' }, children: label }));
537
- return (jsxRuntime.jsxs("div", { style: { display: 'flex', flexDirection: 'column', height: '100%', fontFamily: 'monospace' }, children: [jsxRuntime.jsxs("div", { style: { display: 'flex', gap: 2, padding: '2px 4px', borderBottom: '1px solid #313244' }, children: [filterBtn('all', 'All'), " ", filterBtn('info', 'Info'), " ", filterBtn('warn', 'Warn'), " ", filterBtn('error', 'Error'), jsxRuntime.jsx("span", { style: { flex: 1 } }), jsxRuntime.jsxs("span", { style: { fontSize: 9, color: '#6c7086' }, children: [filtered.length, " entries"] })] }), jsxRuntime.jsx("div", { ref: scrollRef, style: { flex: 1, overflow: 'auto', fontSize: 10 }, children: filtered.length === 0 ? (jsxRuntime.jsx("div", { style: { opacity: 0.4, padding: 8 }, children: "Console ready." })) : filtered.map(log => (jsxRuntime.jsxs("div", { style: { padding: '1px 4px', borderBottom: '1px solid #181825', display: 'flex', gap: 6 }, children: [jsxRuntime.jsx("span", { style: { color: '#6c7086', minWidth: 55 }, children: new Date(log.timestamp).toLocaleTimeString([], { hour12: false }) }), jsxRuntime.jsxs("span", { style: { color: levelColors[log.level], fontWeight: 'bold', minWidth: 35 }, children: ["[", log.level.toUpperCase(), "]"] }), jsxRuntime.jsx("span", { children: log.message })] }, log.id))) })] }));
869
+ const filterBtn = (f, label) => (jsxRuntime.jsx("button", { onClick: () => setFilter(f), style: {
870
+ padding: '1px 6px',
871
+ border: 'none',
872
+ borderRadius: 2,
873
+ cursor: 'pointer',
874
+ fontSize: 9,
875
+ background: filter === f ? '#313244' : 'transparent',
876
+ color: filter === f ? '#cdd6f4' : '#6c7086',
877
+ }, children: label }));
878
+ return (jsxRuntime.jsxs("div", { style: { display: 'flex', flexDirection: 'column', height: '100%', fontFamily: 'monospace' }, children: [jsxRuntime.jsxs("div", { style: { display: 'flex', gap: 2, padding: '2px 4px', borderBottom: '1px solid #313244' }, children: [filterBtn('all', 'All'), " ", filterBtn('info', 'Info'), " ", filterBtn('warn', 'Warn'), ' ', filterBtn('error', 'Error'), jsxRuntime.jsx("span", { style: { flex: 1 } }), jsxRuntime.jsxs("span", { style: { fontSize: 9, color: '#6c7086' }, children: [filtered.length, " entries"] })] }), jsxRuntime.jsx("div", { ref: scrollRef, style: { flex: 1, overflow: 'auto', fontSize: 10 }, children: filtered.length === 0 ? (jsxRuntime.jsx("div", { style: { opacity: 0.4, padding: 8 }, children: "Console ready." })) : (filtered.map((log) => (jsxRuntime.jsxs("div", { style: {
879
+ padding: '1px 4px',
880
+ borderBottom: '1px solid #181825',
881
+ display: 'flex',
882
+ gap: 6,
883
+ }, children: [jsxRuntime.jsx("span", { style: { color: '#6c7086', minWidth: 55 }, children: new Date(log.timestamp).toLocaleTimeString([], { hour12: false }) }), jsxRuntime.jsxs("span", { style: { color: levelColors[log.level], fontWeight: 'bold', minWidth: 35 }, children: ["[", log.level.toUpperCase(), "]"] }), jsxRuntime.jsx("span", { children: log.message })] }, log.id)))) })] }));
538
884
  };
539
885
  const EditorViewport = () => {
540
886
  const canvasRef = React.useRef(null);
@@ -547,11 +893,13 @@ const EditorViewport = () => {
547
893
  // Draw loop
548
894
  React.useEffect(() => {
549
895
  const canvas = canvasRef.current;
550
- if (!canvas)
896
+ if (!canvas) {
551
897
  return;
898
+ }
552
899
  const ctx = canvas.getContext('2d');
553
- if (!ctx)
900
+ if (!ctx) {
554
901
  return;
902
+ }
555
903
  const { width, height } = canvas.getBoundingClientRect();
556
904
  canvas.width = width;
557
905
  canvas.height = height;
@@ -563,8 +911,8 @@ const EditorViewport = () => {
563
911
  // Grid
564
912
  if (state.showGrid) {
565
913
  const gs = state.gridSize;
566
- const halfW = (width / 2) / state.zoom;
567
- const halfH = (height / 2) / state.zoom;
914
+ const halfW = width / 2 / state.zoom;
915
+ const halfH = height / 2 / state.zoom;
568
916
  const offX = -state.panOffset.x / state.zoom;
569
917
  const offY = -state.panOffset.y / state.zoom;
570
918
  const startX = Math.floor((offX - halfW) / gs) * gs;
@@ -600,11 +948,13 @@ const EditorViewport = () => {
600
948
  // Render entities
601
949
  const entities = world.allEntities();
602
950
  for (const entity of entities) {
603
- if (!entity.enabled)
951
+ if (!entity.enabled) {
604
952
  continue;
953
+ }
605
954
  const transform = world.getComponent(entity.id, 'Transform');
606
- if (!transform)
955
+ if (!transform) {
607
956
  continue;
957
+ }
608
958
  ctx.save();
609
959
  ctx.translate(transform.position.x, transform.position.y);
610
960
  ctx.rotate(transform.rotation);
@@ -619,15 +969,17 @@ const EditorViewport = () => {
619
969
  switch (shape.shape) {
620
970
  case 'rect':
621
971
  ctx.fillRect(-shape.width / 2, -shape.height / 2, shape.width, shape.height);
622
- if (shape.strokeWidth > 0)
972
+ if (shape.strokeWidth > 0) {
623
973
  ctx.strokeRect(-shape.width / 2, -shape.height / 2, shape.width, shape.height);
974
+ }
624
975
  break;
625
976
  case 'circle':
626
977
  ctx.beginPath();
627
978
  ctx.arc(0, 0, shape.radius, 0, Math.PI * 2);
628
979
  ctx.fill();
629
- if (shape.strokeWidth > 0)
980
+ if (shape.strokeWidth > 0) {
630
981
  ctx.stroke();
982
+ }
631
983
  break;
632
984
  case 'line':
633
985
  ctx.beginPath();
@@ -639,12 +991,14 @@ const EditorViewport = () => {
639
991
  if (shape.points.length > 2) {
640
992
  ctx.beginPath();
641
993
  ctx.moveTo(shape.points[0].x, shape.points[0].y);
642
- for (let i = 1; i < shape.points.length; i++)
994
+ for (let i = 1; i < shape.points.length; i++) {
643
995
  ctx.lineTo(shape.points[i].x, shape.points[i].y);
996
+ }
644
997
  ctx.closePath();
645
998
  ctx.fill();
646
- if (shape.strokeWidth > 0)
999
+ if (shape.strokeWidth > 0) {
647
1000
  ctx.stroke();
1001
+ }
648
1002
  }
649
1003
  break;
650
1004
  }
@@ -722,7 +1076,7 @@ const EditorViewport = () => {
722
1076
  ctx.fillRect(b.x + b.w - hs, b.y + b.h - hs, hs * 2, hs * 2);
723
1077
  }
724
1078
  // Entity name label (only if no visible shape/sprite/text OR selected)
725
- if (isSelected || (!(shape === null || shape === void 0 ? void 0 : shape.visible) && !(sprite === null || sprite === void 0 ? void 0 : sprite.visible) && !(text === null || text === void 0 ? void 0 : text.visible))) {
1079
+ if (isSelected || (!shape?.visible && !sprite?.visible && !text?.visible)) {
726
1080
  ctx.font = `${10 / state.zoom}px monospace`;
727
1081
  ctx.fillStyle = isSelected ? '#89b4fa' : 'rgba(205, 214, 244, 0.4)';
728
1082
  ctx.textAlign = 'center';
@@ -730,7 +1084,7 @@ const EditorViewport = () => {
730
1084
  const b = getEntityBounds(entity.id, world);
731
1085
  ctx.fillText(entity.name, 0, b.y - 4 / state.zoom);
732
1086
  // Draw small dot for invisible/empty entities
733
- if (!(shape === null || shape === void 0 ? void 0 : shape.visible) && !(sprite === null || sprite === void 0 ? void 0 : sprite.visible) && !(text === null || text === void 0 ? void 0 : text.visible)) {
1087
+ if (!shape?.visible && !sprite?.visible && !text?.visible) {
734
1088
  ctx.fillStyle = isSelected ? '#89b4fa' : 'rgba(205, 214, 244, 0.3)';
735
1089
  ctx.beginPath();
736
1090
  ctx.arc(0, 0, 3 / state.zoom, 0, Math.PI * 2);
@@ -762,22 +1116,33 @@ const EditorViewport = () => {
762
1116
  // Middle click or pan tool = pan
763
1117
  if (e.button === 1 || (e.button === 0 && state.tool === 'pan')) {
764
1118
  panning.current = true;
765
- panStart.current = { x: e.clientX, y: e.clientY, px: state.panOffset.x, py: state.panOffset.y };
1119
+ panStart.current = {
1120
+ x: e.clientX,
1121
+ y: e.clientY,
1122
+ px: state.panOffset.x,
1123
+ py: state.panOffset.y,
1124
+ };
766
1125
  e.target.setPointerCapture(e.pointerId);
767
1126
  return;
768
1127
  }
769
- if (e.button !== 0)
1128
+ if (e.button !== 0) {
770
1129
  return;
1130
+ }
771
1131
  const { x: wx, y: wy } = screenToWorld(e);
772
1132
  // Hit test (reverse order for top-most)
773
1133
  const entities = world.allEntities();
774
1134
  let hitId = null;
775
1135
  for (let i = entities.length - 1; i >= 0; i--) {
776
1136
  const ent = entities[i];
777
- if (!ent.enabled)
1137
+ if (!ent.enabled) {
778
1138
  continue;
1139
+ }
779
1140
  const bounds = getEntityWorldBounds(ent.id, world);
780
- if (bounds && wx >= bounds.x && wx <= bounds.x + bounds.w && wy >= bounds.y && wy <= bounds.y + bounds.h) {
1141
+ if (bounds &&
1142
+ wx >= bounds.x &&
1143
+ wx <= bounds.x + bounds.w &&
1144
+ wy >= bounds.y &&
1145
+ wy <= bounds.y + bounds.h) {
781
1146
  hitId = ent.id;
782
1147
  break;
783
1148
  }
@@ -788,8 +1153,9 @@ const EditorViewport = () => {
788
1153
  dragging.current = true;
789
1154
  dragStart.current = { x: wx, y: wy };
790
1155
  const t = world.getComponent(hitId, 'Transform');
791
- if (t)
1156
+ if (t) {
792
1157
  dragEntityStart.current = { x: t.position.x, y: t.position.y };
1158
+ }
793
1159
  }
794
1160
  }
795
1161
  else {
@@ -801,7 +1167,10 @@ const EditorViewport = () => {
801
1167
  if (panning.current) {
802
1168
  const dx = e.clientX - panStart.current.x;
803
1169
  const dy = e.clientY - panStart.current.y;
804
- dispatch({ type: 'SET_PAN', offset: { x: panStart.current.px + dx, y: panStart.current.py + dy } });
1170
+ dispatch({
1171
+ type: 'SET_PAN',
1172
+ offset: { x: panStart.current.px + dx, y: panStart.current.py + dy },
1173
+ });
805
1174
  return;
806
1175
  }
807
1176
  if (dragging.current && state.selectedEntities.length > 0) {
@@ -823,7 +1192,15 @@ const EditorViewport = () => {
823
1192
  refreshWorld();
824
1193
  }
825
1194
  }
826
- }, [state.selectedEntities, state.snapToGrid, state.gridSize, screenToWorld, world, dispatch, refreshWorld]);
1195
+ }, [
1196
+ state.selectedEntities,
1197
+ state.snapToGrid,
1198
+ state.gridSize,
1199
+ screenToWorld,
1200
+ world,
1201
+ dispatch,
1202
+ refreshWorld,
1203
+ ]);
827
1204
  const handlePointerUp = React.useCallback(() => {
828
1205
  if (dragging.current && state.selectedEntities.length > 0) {
829
1206
  const entityId = state.selectedEntities[0];
@@ -834,8 +1211,16 @@ const EditorViewport = () => {
834
1211
  if (finalPos.x !== startPos.x || finalPos.y !== startPos.y) {
835
1212
  undoStack.push({
836
1213
  description: 'Move entity',
837
- undo: () => { t.position.x = startPos.x; t.position.y = startPos.y; refreshWorld(); },
838
- redo: () => { t.position.x = finalPos.x; t.position.y = finalPos.y; refreshWorld(); },
1214
+ undo: () => {
1215
+ t.position.x = startPos.x;
1216
+ t.position.y = startPos.y;
1217
+ refreshWorld();
1218
+ },
1219
+ redo: () => {
1220
+ t.position.x = finalPos.x;
1221
+ t.position.y = finalPos.y;
1222
+ refreshWorld();
1223
+ },
839
1224
  });
840
1225
  }
841
1226
  }
@@ -848,7 +1233,12 @@ const EditorViewport = () => {
848
1233
  const factor = e.deltaY > 0 ? 0.9 : 1.1;
849
1234
  dispatch({ type: 'SET_ZOOM', zoom: state.zoom * factor });
850
1235
  }, [state.zoom, dispatch]);
851
- return (jsxRuntime.jsx("canvas", { ref: canvasRef, style: { width: '100%', height: '100%', display: 'block', cursor: getCursorForTool(state.tool) }, onPointerDown: handlePointerDown, onPointerMove: handlePointerMove, onPointerUp: handlePointerUp, onWheel: handleWheel, onContextMenu: e => e.preventDefault() }));
1236
+ return (jsxRuntime.jsx("canvas", { ref: canvasRef, style: {
1237
+ width: '100%',
1238
+ height: '100%',
1239
+ display: 'block',
1240
+ cursor: getCursorForTool(state.tool),
1241
+ }, onPointerDown: handlePointerDown, onPointerMove: handlePointerMove, onPointerUp: handlePointerUp, onWheel: handleWheel, onContextMenu: (e) => e.preventDefault() }));
852
1242
  };
853
1243
 
854
1244
  exports.NiceGameEditor = NiceGameEditor;