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