level_editor.cpp 38 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486
  1. /*
  2. * Copyright (c) 2012-2018 Daniele Bartolini and individual contributors.
  3. * License: https://github.com/dbartolini/crown/blob/master/LICENSE
  4. */
  5. #if CROWN_TOOLS
  6. #include "core/containers/vector.h"
  7. #include "core/filesystem/file.h"
  8. #include "core/filesystem/filesystem_disk.h"
  9. #include "core/filesystem/path.h"
  10. #include "core/json/json.h"
  11. #include "core/json/json_object.h"
  12. #include "core/json/sjson.h"
  13. #include "core/math/vector2.h"
  14. #include "core/network/socket.h"
  15. #include "core/strings/dynamic_string.h"
  16. #include "device/device.h"
  17. #include "device/device_event_queue.h"
  18. #include "device/device_options.h"
  19. #include "device/input_device.h"
  20. #include "device/input_manager.h"
  21. #include "device/input_types.h"
  22. #include "device/log.h"
  23. #include "device/pipeline.h"
  24. #include "imgui_context.h"
  25. #include "resource/resource_manager.h"
  26. #include "resource/texture_resource.h"
  27. #include "tool_api.h"
  28. #include <iconfontheaders/icons_material_design.h>
  29. #include <imgui.h>
  30. #include <nfd.h>
  31. #include <time.h>
  32. #if CROWN_PLATFORM_POSIX
  33. #include <sys/time.h>
  34. #endif
  35. namespace { const crown::log_internal::System LEVEL_EDITOR = { "LevelEditor" }; }
  36. namespace crown
  37. {
  38. static u16 _width = 1280;
  39. static u16 _height = 720;
  40. struct Pivot
  41. {
  42. enum Enum
  43. {
  44. TOP_LEFT,
  45. TOP_CENTER,
  46. TOP_RIGHT,
  47. LEFT,
  48. CENTER,
  49. RIGHT,
  50. BOTTOM_LEFT,
  51. BOTTOM_CENTER,
  52. BOTTOM_RIGHT,
  53. COUNT
  54. };
  55. };
  56. static const char* pivot_names[] =
  57. {
  58. "Top Left", // Pivot::TOP_LEFT
  59. "Top Right", // Pivot::TOP_CENTER
  60. "Top Center", // Pivot::TOP_RIGHT
  61. "Left", // Pivot::LEFT
  62. "Center", // Pivot::CENTER
  63. "Right", // Pivot::RIGHT
  64. "Bottom Left", // Pivot::BOTTOM_LEFT
  65. "Bottom Center", // Pivot::BOTTOM_CENTER
  66. "Bottom Right" // Pivot::BOTTOM_RIGHT
  67. };
  68. CE_STATIC_ASSERT(countof(pivot_names) == Pivot::COUNT);
  69. Vector2 sprite_cell_xy(int r, int c, int offset_x, int offset_y, int cell_w, int cell_h, int spacing_x, int spacing_y)
  70. {
  71. int x0 = offset_x + c*cell_w + c*spacing_x;
  72. int y0 = offset_y + r*cell_h + r*spacing_y;
  73. return vector2(x0, y0);
  74. }
  75. Vector2 sprite_cell_pivot_xy(int cell_w, int cell_h, int pivot)
  76. {
  77. int pivot_x = 0;
  78. int pivot_y = 0;
  79. switch (pivot)
  80. {
  81. case Pivot::TOP_LEFT:
  82. pivot_x = 0;
  83. pivot_y = 0;
  84. break;
  85. case Pivot::TOP_CENTER:
  86. pivot_x = cell_w / 2;
  87. pivot_y = 0;
  88. break;
  89. case Pivot::TOP_RIGHT:
  90. pivot_x = cell_w;
  91. pivot_y = 0;
  92. break;
  93. case Pivot::BOTTOM_LEFT:
  94. pivot_x = 0;
  95. pivot_y = cell_h;
  96. break;
  97. case Pivot::BOTTOM_CENTER:
  98. pivot_x = cell_w / 2;
  99. pivot_y = cell_h;
  100. break;
  101. case Pivot::BOTTOM_RIGHT:
  102. pivot_x = cell_w;
  103. pivot_y = cell_h;
  104. break;
  105. case Pivot::LEFT:
  106. pivot_x = 0;
  107. pivot_y = cell_h / 2;
  108. break;
  109. case Pivot::CENTER:
  110. pivot_x = cell_w / 2;
  111. pivot_y = cell_h / 2;
  112. break;
  113. case Pivot::RIGHT:
  114. pivot_x = cell_w;
  115. pivot_y = cell_h / 2;
  116. break;
  117. default:
  118. CE_FATAL("Unknown pivot");
  119. break;
  120. }
  121. return vector2(pivot_x, pivot_y);
  122. }
  123. struct SpriteImporter
  124. {
  125. int width;
  126. int height;
  127. int cells_h;
  128. int cells_v;
  129. bool cell_wh_auto;
  130. int cell_w;
  131. int cell_h;
  132. int offset_x;
  133. int offset_y;
  134. int spacing_x;
  135. int spacing_y;
  136. int pivot;
  137. int layer;
  138. int depth;
  139. SpriteImporter()
  140. : width(128)
  141. , height(128)
  142. , cells_h(4)
  143. , cells_v(4)
  144. , cell_wh_auto(false)
  145. , cell_w(16)
  146. , cell_h(16)
  147. , offset_x(0)
  148. , offset_y(0)
  149. , spacing_x(0)
  150. , spacing_y(0)
  151. , pivot(Pivot::CENTER)
  152. , layer(0)
  153. , depth(0)
  154. {
  155. }
  156. void draw()
  157. {
  158. ImGui::Columns(2);
  159. #if 1
  160. ImDrawList* draw_list = ImGui::GetWindowDrawList();
  161. {
  162. // Here we are using InvisibleButton() as a convenience to 1) advance the cursor and 2) allows us to use IsItemHovered()
  163. // However you can draw directly and poll mouse/keyboard by yourself. You can manipulate the cursor using GetCursorPos() and SetCursorPos().
  164. // If you only use the ImDrawList API, you can notify the owner window of its extends by using SetCursorPos(max).
  165. ImVec2 canvas_pos = ImGui::GetCursorScreenPos(); // ImDrawList API uses screen coordinates!
  166. ImVec2 canvas_size = ImGui::GetContentRegionAvail(); // Resize canvas to what's available
  167. if (canvas_size.x < 50.0f) canvas_size.x = 10.0f;
  168. if (canvas_size.y < 50.0f) canvas_size.y = 50.0f;
  169. draw_list->AddRectFilledMultiColor(canvas_pos, ImVec2(canvas_pos.x + canvas_size.x, canvas_pos.y + canvas_size.y), ImColor(50,50,50), ImColor(50,50,60), ImColor(60,60,70), ImColor(50,50,60));
  170. // Pivot is relative to the top-left corner of the cell
  171. Vector2 pivot_xy = sprite_cell_pivot_xy(cell_w
  172. , cell_h
  173. , pivot
  174. );
  175. int num_v = cells_v;
  176. int num_h = cells_h;
  177. for (int h = 0; h < num_v; ++h)
  178. {
  179. for (int w = 0; w < num_h; ++w)
  180. {
  181. Vector2 cell = sprite_cell_xy(h
  182. , w
  183. , offset_x
  184. , offset_y
  185. , cell_w
  186. , cell_h
  187. , spacing_x
  188. , spacing_y
  189. );
  190. const int x0 = (int)cell.x;
  191. const int y0 = (int)cell.y;
  192. const int x1 = x0+(int)cell_w;
  193. const int y1 = y0+(int)cell_h;
  194. draw_list->AddRect(ImVec2(canvas_pos.x + x0, canvas_pos.y + y0)
  195. , ImVec2(canvas_pos.x + x1, canvas_pos.y + y1)
  196. , ImColor(230, 26, 26, 153)
  197. );
  198. draw_list->AddCircleFilled(ImVec2(x0 + canvas_pos.x + pivot_xy.x, y0 + canvas_pos.y + pivot_xy.y)
  199. , 5.0f
  200. , ImColor(26, 26, 230, 153)
  201. );
  202. }
  203. }
  204. ImGui::InvisibleButton("canvas", canvas_size);
  205. ImVec2 mouse_pos_in_canvas = ImVec2(ImGui::GetIO().MousePos.x - canvas_pos.x, ImGui::GetIO().MousePos.y - canvas_pos.y);
  206. draw_list->PushClipRect(canvas_pos, ImVec2(canvas_pos.x+canvas_size.x, canvas_pos.y+canvas_size.y)); // clip lines within the canvas (if we resize it, etc.)
  207. }
  208. #endif
  209. ImGui::NextColumn();
  210. ImGui::BeginGroup();
  211. ImGui::LabelText("Resolution", "%d x %d", width, height);
  212. // FIXME: replace fclamp
  213. ImGui::InputInt("Cells H", &cells_h);
  214. cells_h = (int)fclamp(cells_h, 1, 256);
  215. ImGui::InputInt("Cells V", &cells_v);
  216. cells_v = (int)fclamp(cells_v, 1, 256);
  217. ImGui::Checkbox("Cell WH auto", &cell_wh_auto);
  218. ImGui::InputInt("Cell W", &cell_w);
  219. cell_w = (int)fclamp(cell_w, 1, 4096);
  220. ImGui::InputInt("Cell H", &cell_h);
  221. cell_h = (int)fclamp(cell_h, 1, 4096);
  222. ImGui::InputInt("Offset X", &offset_x);
  223. offset_x = (int)fclamp(offset_x, 0, 128);
  224. ImGui::InputInt("Offset Y", &offset_y);
  225. offset_y = (int)fclamp(offset_y, 0, 128);
  226. ImGui::InputInt("Spacing X", &spacing_x);
  227. spacing_x = (int)fclamp(spacing_x, 0, 128);
  228. ImGui::InputInt("Spacing Y", &spacing_y);
  229. spacing_y = (int)fclamp(spacing_y, 0, 128);
  230. ImGui::Combo("Pivot", &pivot, pivot_names, Pivot::COUNT);
  231. ImGui::InputInt("Layer", &layer);
  232. layer = (int)fclamp(layer, 0, 7);
  233. ImGui::InputInt("Depth", &depth);
  234. depth = (int)fclamp(depth, 0, 9999);
  235. ImGui::EndGroup();
  236. }
  237. };
  238. //-----------------------------------------------------------------------------
  239. struct Inspector
  240. {
  241. // Inspector
  242. char _name[1024];
  243. f32 _position[3];
  244. f32 _rotation[3];
  245. f32 _scale[3];
  246. char _sprite[1024];
  247. char _material[1024];
  248. bool _visible;
  249. char _state_machine[1024];
  250. bool _open;
  251. Inspector()
  252. : _visible(true)
  253. , _open(true)
  254. {
  255. memset(_name, 0, sizeof(_name));
  256. memset(_sprite, 0, sizeof(_sprite));
  257. memset(_material, 0, sizeof(_material));
  258. memset(_position, 0, sizeof(_position));
  259. memset(_rotation, 0, sizeof(_rotation));
  260. memset(_scale, 0, sizeof(_scale));
  261. memset(_state_machine, 0, sizeof(_state_machine));
  262. }
  263. void draw()
  264. {
  265. if (!_open)
  266. return;
  267. if (ImGui::TreeNodeEx("Unit", ImGuiTreeNodeFlags_DefaultOpen))
  268. {
  269. ImGui::InputText("Name", _name, sizeof(_name));
  270. ImGui::TreePop();
  271. }
  272. if (ImGui::TreeNodeEx("Transform", ImGuiTreeNodeFlags_DefaultOpen))
  273. {
  274. ImGui::InputFloat3("Position", _position);
  275. ImGui::InputFloat3("Rotation", _rotation);
  276. ImGui::InputFloat3("Scale", _scale);
  277. ImGui::TreePop();
  278. }
  279. if (ImGui::TreeNodeEx("Renderer", ImGuiTreeNodeFlags_DefaultOpen))
  280. {
  281. ImGui::InputText("Sprite", _sprite, sizeof(_sprite));
  282. ImGui::InputText("Material", _material, sizeof(_material));
  283. ImGui::Checkbox("Visible", &_visible);
  284. ImGui::TreePop();
  285. }
  286. if (ImGui::TreeNodeEx("Animation", ImGuiTreeNodeFlags_DefaultOpen))
  287. {
  288. ImGui::InputText("State Machine", _state_machine, sizeof(_state_machine));
  289. ImGui::TreePop();
  290. }
  291. }
  292. };
  293. //-----------------------------------------------------------------------------
  294. struct SceneView
  295. {
  296. ImVec2 _view_origin;
  297. ImVec2 _view_size;
  298. ImVec2 _mouse_curr;
  299. ImVec2 _mouse_last;
  300. bool _open;
  301. SceneView()
  302. : _open(true)
  303. {
  304. }
  305. void draw()
  306. {
  307. if (!_open)
  308. return;
  309. _view_origin = ImGui::GetCursorScreenPos();
  310. uint16_t w, h;
  311. device()->resolution(w, h);
  312. bgfx::TextureHandle txh = device()->_pipeline->_buffers[0];
  313. CE_ENSURE(bgfx::isValid(txh));
  314. ImGui::Image((void*)(uintptr_t)txh.idx
  315. , ImVec2(w, h)
  316. #if CROWN_PLATFORM_WINDOWS
  317. , ImVec2(0, 0)
  318. , ImVec2(1, 1)
  319. #else
  320. , ImVec2(0, 1)
  321. , ImVec2(1, 0)
  322. #endif // CROWN_PLATFORM_WINDOWS
  323. );
  324. ImVec2 mouse_pos_in_view = ImVec2(ImGui::GetIO().MousePos.x - _view_origin.x
  325. , ImGui::GetIO().MousePos.y - _view_origin.y
  326. );
  327. if (ImGui::IsWindowHovered()
  328. && mouse_pos_in_view.x > 0
  329. && mouse_pos_in_view.x < w
  330. && mouse_pos_in_view.y > 0
  331. && mouse_pos_in_view.y < h
  332. )
  333. {
  334. // Send all input to engine
  335. ImGui::CaptureMouseFromApp(false);
  336. ImGui::CaptureKeyboardFromApp(false);
  337. }
  338. else
  339. {
  340. // Send all input to imgui
  341. ImGui::CaptureMouseFromApp(true);
  342. ImGui::CaptureKeyboardFromApp(true);
  343. }
  344. _view_size = ImGui::GetWindowSize();
  345. _view_size.x -= _view_origin.x;
  346. }
  347. };
  348. //-----------------------------------------------------------------------------
  349. struct SceneTree
  350. {
  351. bool _open;
  352. SceneTree()
  353. : _open(true)
  354. {
  355. }
  356. void draw()
  357. {
  358. if (!_open)
  359. return;
  360. if (ImGui::TreeNodeEx("Units", ImGuiTreeNodeFlags_DefaultOpen))
  361. {
  362. if (ImGui::TreeNodeEx("Objects", ImGuiTreeNodeFlags_DefaultOpen))
  363. {
  364. for (int i = 0; i < 5; i++)
  365. if (ImGui::TreeNode((void*)(intptr_t)i, "Child %d", i))
  366. {
  367. ImGui::Text("blah blah");
  368. ImGui::SameLine();
  369. if (ImGui::SmallButton("print")) printf("Child %d pressed", i);
  370. ImGui::TreePop();
  371. }
  372. ImGui::TreePop();
  373. }
  374. if (ImGui::TreeNodeEx("Lights", ImGuiTreeNodeFlags_DefaultOpen))
  375. {
  376. // ShowHelpMarker("This is a more standard looking tree with selectable nodes.\nClick to select, Ctrl+Click to toggle, click on arrows or double-click to open.");
  377. static bool align_label_with_current_x_position = false;
  378. ImGui::Checkbox("Align label with current X position)", &align_label_with_current_x_position);
  379. ImGui::Text("Hello!");
  380. if (align_label_with_current_x_position)
  381. ImGui::Unindent(ImGui::GetTreeNodeToLabelSpacing());
  382. static int selection_mask = (1 << 2); // Dumb representation of what may be user-side selection state. You may carry selection state inside or outside your objects in whatever format you see fit.
  383. int node_clicked = -1; // Temporary storage of what node we have clicked to process selection at the end of the loop. May be a pointer to your own node type, etc.
  384. ImGui::PushStyleVar(ImGuiStyleVar_IndentSpacing, ImGui::GetFontSize()*3); // Increase spacing to differentiate leaves from expanded contents.
  385. for (int i = 0; i < 6; i++)
  386. {
  387. // Disable the default open on single-click behavior and pass in Selected flag according to our selection state.
  388. ImGuiTreeNodeFlags node_flags = ImGuiTreeNodeFlags_OpenOnArrow | ImGuiTreeNodeFlags_OpenOnDoubleClick | ((selection_mask & (1 << i)) ? ImGuiTreeNodeFlags_Selected : 0);
  389. if (i < 3)
  390. {
  391. // Node
  392. bool node_open = ImGui::TreeNodeEx((void*)(intptr_t)i, node_flags, "Selectable Node %d", i);
  393. if (ImGui::IsItemClicked())
  394. node_clicked = i;
  395. if (node_open)
  396. {
  397. ImGui::Text("Blah blah\nBlah Blah");
  398. ImGui::TreePop();
  399. }
  400. }
  401. else
  402. {
  403. // Leaf: The only reason we have a TreeNode at all is to allow selection of the leaf. Otherwise we can use BulletText() or TreeAdvanceToLabelPos()+Text().
  404. ImGui::TreeNodeEx((void*)(intptr_t)i, node_flags | ImGuiTreeNodeFlags_Leaf | ImGuiTreeNodeFlags_NoTreePushOnOpen, "Selectable Leaf %d", i);
  405. if (ImGui::IsItemClicked())
  406. node_clicked = i;
  407. }
  408. }
  409. if (node_clicked != -1)
  410. {
  411. // Update selection state. Process outside of tree loop to avoid visual inconsistencies during the clicking-frame.
  412. if (ImGui::GetIO().KeyCtrl)
  413. selection_mask ^= (1 << node_clicked); // Ctrl+click to toggle
  414. else //if (!(selection_mask & (1 << node_clicked))) // Depending on selection behavior you want, this commented bit preserve selection when clicking on item that is part of the selection
  415. selection_mask = (1 << node_clicked); // Click to single-select
  416. }
  417. ImGui::PopStyleVar();
  418. if (align_label_with_current_x_position)
  419. ImGui::Indent(ImGui::GetTreeNodeToLabelSpacing());
  420. ImGui::TreePop();
  421. }
  422. ImGui::TreePop();
  423. }
  424. }
  425. };
  426. //-----------------------------------------------------------------------------
  427. struct SpriteAnimator
  428. {
  429. bool _open;
  430. bool _add_animation_popup_open;
  431. Array<const char*> _entities;
  432. s32 _cur_entity;
  433. TextureResource* _texture;
  434. u32 _texture_width;
  435. u32 _texture_height;
  436. struct Frame
  437. {
  438. char name[512];
  439. ImVec2 pivot;
  440. ImVec4 region;
  441. };
  442. Array<Frame> _frames;
  443. char _anim_name[512];
  444. f32 _anim_time;
  445. Array<const char*> _listbox_items;
  446. s32 _listbox_item_current;
  447. Array<Frame> _anim_preview_frames;
  448. f32 _delta;
  449. u32 current;
  450. Vector<DynamicString> file_list_sprites;
  451. FilesystemDisk* _fs;
  452. SpriteAnimator(const DynamicString& src_dir)
  453. : _open(false)
  454. , _add_animation_popup_open(false)
  455. , _entities(default_allocator())
  456. , _cur_entity(0)
  457. , _texture(nullptr)
  458. , _texture_width(0)
  459. , _texture_height(0)
  460. , _frames(default_allocator())
  461. , _anim_time(0.1f)
  462. , _listbox_items(default_allocator())
  463. , _listbox_item_current(0)
  464. , _anim_preview_frames(default_allocator())
  465. , _delta(0.0f)
  466. , current(0)
  467. , file_list_sprites(default_allocator())
  468. {
  469. memset(_anim_name, 0, sizeof(_anim_name));
  470. _fs = CE_NEW(default_allocator(), FilesystemDisk)(default_allocator());
  471. _fs->set_prefix(src_dir.c_str());
  472. get_sprites_list();
  473. }
  474. ~SpriteAnimator()
  475. {
  476. CE_DELETE(default_allocator(), _fs);
  477. }
  478. ImVec2 pixel_to_uv(u32 tex_w, u32 tex_h, f32 x, f32 y)
  479. {
  480. ImVec2 uv;
  481. uv.x = (f32)x / (f32)tex_w;
  482. uv.y = (f32)y / (f32)tex_h;
  483. return uv;
  484. }
  485. void get_directory(DynamicString& dir, const char* path)
  486. {
  487. CE_ENSURE(NULL != path);
  488. const char* ls = strrchr(path, '/');
  489. u32 file_len = strlen(ls+1);
  490. u32 dir_len = strlen(path) - file_len;
  491. char buff[1024];
  492. memcpy(buff, path, dir_len);
  493. buff[dir_len] = '\0';
  494. path::reduce(dir, buff);
  495. }
  496. void save_sprite_animation()
  497. {
  498. TempAllocator4096 ta;
  499. StringStream ss(ta);
  500. ss << "frames = [ ";
  501. for (u32 i = 0; i < array::size(_listbox_items); i++)
  502. {
  503. ss << _listbox_items[i] + 7 << " ";
  504. }
  505. ss << "]\n";
  506. ss << "total_time = ";
  507. ss << _anim_time;
  508. DynamicString dir(ta);
  509. get_directory(dir, _entities[_cur_entity]);
  510. DynamicString file_name(ta);
  511. path::join(file_name, dir.c_str(), _anim_name);
  512. file_name += ".sprite_animation";
  513. File* f = _fs->open(file_name.c_str(), FileOpenMode::WRITE);
  514. f->write(string_stream::c_str(ss), strlen(string_stream::c_str(ss)));
  515. f->close();
  516. }
  517. void get_files_list(Vector<DynamicString>& out, const char* path="")
  518. {
  519. TempAllocator4096 ta;
  520. Vector<DynamicString> files(ta);
  521. _fs->list_files(path, files);
  522. for (u32 i = 0; i < vector::size(files); i++)
  523. {
  524. DynamicString join(ta);
  525. path::join(join, path, files[i].c_str());
  526. join = join.c_str()[0] == '/' ? join.c_str()+1 : join.c_str();
  527. if (_fs->is_directory(join.c_str()))
  528. {
  529. get_files_list(out, join.c_str());
  530. }
  531. else
  532. {
  533. vector::push_back(out, join);
  534. }
  535. }
  536. }
  537. void get_sprites_list()
  538. {
  539. TempAllocator4096 ta;
  540. Vector<DynamicString> files(ta);
  541. get_files_list(files);
  542. for (DynamicString* f = vector::begin(files); f != vector::end(files); f++)
  543. if (f->has_suffix(".sprite"))
  544. array::push_back(_entities, (const char*) strdup(f->c_str()));
  545. }
  546. void draw()
  547. {
  548. if (!_open)
  549. return;
  550. if (_texture)
  551. {
  552. Frame f = _frames[0];
  553. ImVec2 start = pixel_to_uv(_texture_width, _texture_height, f.region.x, f.region.y);
  554. ImVec2 end = pixel_to_uv(_texture_width, _texture_height, f.region.x+f.region.z, f.region.y+f.region.w);
  555. ImGui::Image((void*)(uintptr_t)_texture->handle.idx
  556. , ImVec2(f.region.z, f.region.w)
  557. , start
  558. , end
  559. , ImColor(255, 255, 255, 55)
  560. );
  561. }
  562. if (ImGui::Combo("Entities", &_cur_entity, (const char* const*) array::begin(_entities), array::size(_entities)))
  563. {
  564. array::clear(_frames);
  565. const char* sprite = _entities[_cur_entity];
  566. u32 sprite_len = strlen(sprite);
  567. char entity[1024];
  568. strncpy(entity, sprite, sizeof(entity)-1);
  569. entity[sprite_len-7] = '\0'; // remove ".sprite"
  570. ResourceManager* resman = device()->_resource_manager;
  571. _texture = (TextureResource*)resman->get(RESOURCE_TYPE_TEXTURE, StringId64(entity));
  572. File* file = _fs->open(sprite, FileOpenMode::READ);
  573. const u32 size = file->size();
  574. Buffer buf(default_allocator());
  575. array::resize(buf, size);
  576. file->read(array::begin(buf), size);
  577. _fs->close(*file);
  578. JsonObject obj(default_allocator());
  579. JsonArray list(default_allocator());
  580. sjson::parse(buf, obj);
  581. sjson::parse_array(obj["frames"], list);
  582. _texture_width = sjson::parse_int(obj["width"]);
  583. _texture_height = sjson::parse_int(obj["height"]);
  584. for (u32 i = 0; i < array::size(list); i++)
  585. {
  586. JsonObject frame(default_allocator());
  587. DynamicString name(default_allocator());
  588. JsonArray pivot(default_allocator());
  589. JsonArray region(default_allocator());
  590. sjson::parse_object(list[i], frame);
  591. sjson::parse_array(frame["pivot"], pivot);
  592. sjson::parse_array(frame["region"], region);
  593. Frame f;
  594. sjson::parse_string(frame["name"], name);
  595. strncpy(f.name, name.c_str(), name.length());
  596. f.name[name.length()] = '\0';
  597. f.pivot.x = sjson::parse_float(pivot[0]);
  598. f.pivot.y = sjson::parse_float(pivot[1]);
  599. f.region.x = sjson::parse_float(region[0]);
  600. f.region.y = sjson::parse_float(region[1]);
  601. f.region.z = sjson::parse_float(region[2]);
  602. f.region.w = sjson::parse_float(region[3]);
  603. array::push_back(_frames, f);
  604. }
  605. }
  606. if (ImGui::Button("Add animation", ImVec2(100, 25)))
  607. {
  608. _add_animation_popup_open = true;
  609. }
  610. if (_add_animation_popup_open)
  611. {
  612. ImGui::OpenPopup("Add animation");
  613. }
  614. if (ImGui::BeginPopup("Add animation"))
  615. {
  616. ImGui::InputText("Name", _anim_name, sizeof(_anim_name));
  617. ImGui::InputFloat("Time", &_anim_time, 0.1f, 0.1f);
  618. ImGui::ListBox("Animation Frames", &_listbox_item_current, (const char* const*)array::begin(_listbox_items), array::size(_listbox_items));
  619. if (ImGui::Button("Clear Frames", ImVec2(100.0f, 25.0f)))
  620. {
  621. array::clear(_listbox_items);
  622. array::clear(_anim_preview_frames);
  623. _delta = 0.0f;
  624. current = 0;
  625. }
  626. if (array::size(_anim_preview_frames) > 0)
  627. {
  628. _delta += 1.0f/60.0f;
  629. if (_delta >= _anim_time/array::size(_anim_preview_frames))
  630. {
  631. _delta = 0;
  632. current++;
  633. if (current >= array::size(_anim_preview_frames))
  634. current = 0;
  635. }
  636. Frame f = _anim_preview_frames[current];
  637. ImVec2 start = pixel_to_uv(_texture_width, _texture_height, f.region.x, f.region.y);
  638. ImVec2 end = pixel_to_uv(_texture_width, _texture_height, f.region.x+f.region.z, f.region.y+f.region.w);
  639. ImGui::Image(
  640. (void*)(uintptr_t)_texture->handle.idx
  641. , ImVec2(f.region.z, f.region.w)
  642. , start
  643. , end
  644. , ImColor(255, 255, 255, 55)
  645. );
  646. }
  647. ImGui::Separator();
  648. for (u32 i = 0; i < array::size(_frames); i++)
  649. {
  650. Frame f = _frames[i];
  651. ImVec2 start = pixel_to_uv(_texture_width, _texture_height, f.region.x, f.region.y);
  652. ImVec2 end = pixel_to_uv(_texture_width, _texture_height, f.region.x+f.region.z, f.region.y+f.region.w);
  653. ImGui::SameLine();
  654. if (i % 9 == 0) ImGui::NewLine();
  655. ImGui::BeginGroup();
  656. ImGui::Image(
  657. (void*)(uintptr_t)_texture->handle.idx
  658. , ImVec2(f.region.z, f.region.w)
  659. , start
  660. , end
  661. , ImColor(255, 255, 255, 55)
  662. );
  663. ImGui::NewLine();
  664. if (ImGui::Button(_frames[i].name, ImVec2(100.0f, 25.0f)))
  665. {
  666. array::push_back(_listbox_items, (const char*) strdup(_frames[i].name));
  667. array::push_back(_anim_preview_frames, _frames[i]);
  668. }
  669. ImGui::EndGroup();
  670. }
  671. if (ImGui::Button("Save", ImVec2(100, 25)))
  672. {
  673. save_sprite_animation();
  674. ImGui::CloseCurrentPopup();
  675. _add_animation_popup_open = false;
  676. }
  677. ImGui::SameLine();
  678. if (ImGui::Button("Cancel", ImVec2(100, 25)))
  679. {
  680. ImGui::CloseCurrentPopup();
  681. _add_animation_popup_open = false;
  682. }
  683. ImGui::EndPopup();
  684. }
  685. }
  686. };
  687. //-----------------------------------------------------------------------------
  688. struct LevelEditor
  689. {
  690. DynamicString _source_dir;
  691. // FX
  692. TextureResource* tool_move_texture;
  693. TextureResource* tool_place_texture;
  694. TextureResource* tool_rotate_texture;
  695. TextureResource* tool_scale_texture;
  696. TextureResource* reference_world_texture;
  697. TextureResource* reference_local_texture;
  698. TextureResource* axis_local_texture;
  699. TextureResource* axis_world_texture;
  700. TextureResource* snap_to_grid_texture;
  701. // State
  702. f32 _grid_size;
  703. s32 _rotation_snap;
  704. bool _show_grid;
  705. bool _snap_to_grid;
  706. bool _debug_render_world;
  707. bool _debug_physics_world;
  708. tool::ToolType::Enum _tool_type;
  709. tool::SnapMode::Enum _snap_mode;
  710. tool::ReferenceSystem::Enum _reference_system;
  711. ImVec2 _main_menu_pos;
  712. ImVec2 _main_menu_size;
  713. Console _console;
  714. Inspector _inspector;
  715. SceneView _scene_view;
  716. SceneTree _scene_tree;
  717. SpriteAnimator _animator;
  718. SpriteImporter _sprite_importer;
  719. LevelEditor(const DynamicString& source_dir)
  720. : _source_dir(default_allocator())
  721. , _grid_size(1.0f)
  722. , _rotation_snap(15)
  723. , _show_grid(true)
  724. , _snap_to_grid(true)
  725. , _debug_render_world(false)
  726. , _debug_physics_world(false)
  727. , _tool_type(tool::ToolType::MOVE)
  728. , _snap_mode(tool::SnapMode::RELATIVE)
  729. , _reference_system(tool::ReferenceSystem::LOCAL)
  730. , _main_menu_pos(0, 0)
  731. , _main_menu_size(0, 0)
  732. , _animator(source_dir)
  733. {
  734. ResourceManager* resman = device()->_resource_manager;
  735. tool_move_texture = (TextureResource*)resman->get(RESOURCE_TYPE_TEXTURE, StringId64("core/editors/gui/tool-move"));
  736. tool_place_texture = (TextureResource*)resman->get(RESOURCE_TYPE_TEXTURE, StringId64("core/editors/gui/tool-place"));
  737. tool_rotate_texture = (TextureResource*)resman->get(RESOURCE_TYPE_TEXTURE, StringId64("core/editors/gui/tool-rotate"));
  738. tool_scale_texture = (TextureResource*)resman->get(RESOURCE_TYPE_TEXTURE, StringId64("core/editors/gui/tool-scale"));
  739. reference_world_texture = (TextureResource*)resman->get(RESOURCE_TYPE_TEXTURE, StringId64("core/editors/gui/reference-world"));
  740. reference_local_texture = (TextureResource*)resman->get(RESOURCE_TYPE_TEXTURE, StringId64("core/editors/gui/reference-local"));
  741. axis_local_texture = (TextureResource*)resman->get(RESOURCE_TYPE_TEXTURE, StringId64("core/editors/gui/axis-local"));
  742. axis_world_texture = (TextureResource*)resman->get(RESOURCE_TYPE_TEXTURE, StringId64("core/editors/gui/axis-world"));
  743. snap_to_grid_texture = (TextureResource*)resman->get(RESOURCE_TYPE_TEXTURE, StringId64("core/editors/gui/snap-to-grid"));
  744. imgui_create();
  745. ImGui::LoadDock();
  746. }
  747. ~LevelEditor()
  748. {
  749. ImGui::SaveDock();
  750. imgui_destroy();
  751. }
  752. void update(f32 dt)
  753. {
  754. CE_UNUSED(dt);
  755. static f32 last_w = 0.0f;
  756. static f32 last_h = 0.0f;
  757. if (last_w != _scene_view._view_size.x || last_h != _scene_view._view_size.y)
  758. {
  759. last_w = _scene_view._view_size.x;
  760. last_h = _scene_view._view_size.y;
  761. device()->_width = _scene_view._view_size.x != 0.0f ? _scene_view._view_size.x : 128.0f;
  762. device()->_height = _scene_view._view_size.y != 0.0f ? _scene_view._view_size.y : 128.0f;
  763. }
  764. u32 message_count = 0;
  765. // Receive response from engine
  766. for (;;)
  767. {
  768. u32 msg_len = 0;
  769. ReadResult rr = _console._client.read_nonblock(&msg_len, sizeof(msg_len));
  770. if (rr.error == ReadResult::WOULDBLOCK)
  771. break;
  772. if (ReadResult::SUCCESS == rr.error)
  773. {
  774. char msg[8192];
  775. rr = _console._client.read(msg, msg_len);
  776. msg[msg_len] = '\0';
  777. message_count++;
  778. // logi(LEVEL_EDITOR, "count: %d", message_count);
  779. if (ReadResult::SUCCESS == rr.error)
  780. {
  781. TempAllocator4096 ta;
  782. JsonObject obj(ta);
  783. DynamicString type(ta);
  784. json::parse(msg, obj);
  785. json::parse_string(obj["type"], type);
  786. if (type == "message")
  787. {
  788. DynamicString severity(ta);
  789. DynamicString message(ta);
  790. json::parse_string(obj["severity"], severity);
  791. json::parse_string(obj["message"], message);
  792. LogSeverity::Enum ls = LogSeverity::COUNT;
  793. if (strcmp("info", severity.c_str()) == 0)
  794. ls = LogSeverity::LOG_INFO;
  795. else if (strcmp("warning", severity.c_str()) == 0)
  796. ls = LogSeverity::LOG_WARN;
  797. else if (strcmp("error", severity.c_str()) == 0)
  798. ls = LogSeverity::LOG_ERROR;
  799. else
  800. CE_FATAL("Unknown severity");
  801. ConsoleLog log(ls, message.c_str());
  802. vector::push_back(_console._console_items, log);
  803. }
  804. else
  805. {
  806. ConsoleLog log(LogSeverity::LOG_ERROR, "Unknown message type");
  807. vector::push_back(_console._console_items, log);
  808. }
  809. console_scroll_to_bottom();
  810. }
  811. }
  812. }
  813. imgui_begin_frame(VIEW_IMGUI, _width, _height);
  814. f32 offset_y = _main_menu_size.y;
  815. ImGui::RootDock(ImVec2(0, offset_y), ImVec2(_width, _height-offset_y));
  816. main_menu_bar();
  817. if (ImGui::BeginDock("Scene View"
  818. , &_scene_view._open
  819. , ImGuiWindowFlags_NoScrollbar | ImGuiWindowFlags_NoScrollWithMouse)
  820. )
  821. {
  822. // Draw toolbar
  823. ImGui::BeginGroup();
  824. if (ImGui::ImageButton((void*)(uintptr_t)tool_place_texture->handle.idx, ImVec2(16, 16)))
  825. {
  826. _tool_type = tool::ToolType::PLACE;
  827. tool_send_state();
  828. }
  829. if (ImGui::ImageButton((void*)(uintptr_t)tool_move_texture->handle.idx, ImVec2(16, 16)))
  830. {
  831. _tool_type = tool::ToolType::MOVE;
  832. tool_send_state();
  833. }
  834. if (ImGui::ImageButton((void*)(uintptr_t)tool_rotate_texture->handle.idx, ImVec2(16, 16)))
  835. {
  836. _tool_type = tool::ToolType::ROTATE;
  837. tool_send_state();
  838. }
  839. if (ImGui::ImageButton((void*)(uintptr_t)tool_scale_texture->handle.idx, ImVec2(16, 16)))
  840. {
  841. _tool_type = tool::ToolType::SCALE;
  842. tool_send_state();
  843. }
  844. if (ImGui::ImageButton((void*)(uintptr_t)axis_local_texture->handle.idx, ImVec2(16, 16)))
  845. {
  846. _reference_system = tool::ReferenceSystem::LOCAL;
  847. tool_send_state();
  848. }
  849. if (ImGui::ImageButton((void*)(uintptr_t)axis_world_texture->handle.idx, ImVec2(16, 16)))
  850. {
  851. _reference_system = tool::ReferenceSystem::WORLD;
  852. tool_send_state();
  853. }
  854. if (ImGui::ImageButton((void*)(uintptr_t)reference_world_texture->handle.idx, ImVec2(16, 16)))
  855. {
  856. _snap_mode = tool::SnapMode::RELATIVE;
  857. tool_send_state();
  858. }
  859. if (ImGui::ImageButton((void*)(uintptr_t)reference_local_texture->handle.idx, ImVec2(16, 16)))
  860. {
  861. _snap_mode = tool::SnapMode::ABSOLUTE;
  862. tool_send_state();
  863. }
  864. ImGui::EndGroup();
  865. // Draw scene view
  866. ImGui::SameLine();
  867. _scene_view.draw();
  868. }
  869. ImGui::EndDock();
  870. if (ImGui::BeginDock("Scene Tree", &_scene_tree._open))
  871. {
  872. _scene_tree.draw();
  873. }
  874. ImGui::EndDock();
  875. if (ImGui::BeginDock("Inspector", &_inspector._open))
  876. {
  877. _inspector.draw();
  878. }
  879. ImGui::EndDock();
  880. if (ImGui::BeginDock("Console", &_console._open, ImGuiWindowFlags_NoScrollbar))
  881. {
  882. console_draw(_console);
  883. }
  884. ImGui::EndDock();
  885. if (ImGui::BeginDock("Animator", &_animator._open))
  886. {
  887. _animator.draw();
  888. }
  889. ImGui::EndDock();
  890. // _sprite_importer.draw();
  891. imgui_end_frame();
  892. }
  893. void send_command(StringStream& ss)
  894. {
  895. TempAllocator4096 ta;
  896. StringStream out(ta);
  897. out << "{\"type\":\"script\",\"script\":\"";
  898. out << string_stream::c_str(ss);
  899. out << "\"}";
  900. const char* cmd = string_stream::c_str(out);
  901. const u32 size = strlen32(cmd);
  902. _console._client.write(&size, sizeof(u32));
  903. _console._client.write(cmd, size);
  904. }
  905. void tool_send_state()
  906. {
  907. TempAllocator256 ta;
  908. StringStream ss(ta);
  909. tool::set_grid_size(ss, _grid_size);
  910. tool::set_rotation_snap(ss, _rotation_snap);
  911. tool::enable_show_grid(ss, _show_grid);
  912. tool::enable_snap_to_grid(ss, _snap_to_grid);
  913. tool::enable_debug_render_world(ss, _debug_render_world);
  914. tool::enable_debug_physics_world(ss, _debug_physics_world);
  915. tool::set_tool_type(ss, _tool_type);
  916. tool::set_snap_mode(ss, _snap_mode);
  917. tool::set_reference_system(ss, _reference_system);
  918. send_command(ss);
  919. }
  920. void format_time(uint64_t time, char* str, size_t size)
  921. {
  922. #if CROWN_PLATFORM_WINDOWS
  923. // SYSTEMTIME st;
  924. // FileTimeToSystemTime(&time, &st);
  925. // sprintf(str, "%d/%d/%d %d:%d"
  926. // , st.wDay
  927. // , st.wMonth
  928. // , st.wYear
  929. // , st.wHour
  930. // , st.wMinute);
  931. #else
  932. struct tm lt;
  933. localtime_r((time_t*)&time, &lt);
  934. strftime(str, size, "%d/%m/%Y %H:%M", &lt);
  935. #endif
  936. }
  937. void main_menu_bar()
  938. {
  939. // Main Menu
  940. if (ImGui::BeginMainMenuBar())
  941. {
  942. if (ImGui::BeginMenu("File"))
  943. {
  944. if (ImGui::MenuItem("New"))
  945. {
  946. }
  947. if (ImGui::MenuItem("Open", "Ctrl+O"))
  948. {
  949. nfdchar_t *out_path = NULL;
  950. nfdresult_t result = NFD_OpenDialog(NULL, NULL, &out_path);
  951. if ( result == NFD_OKAY )
  952. {
  953. logi(LEVEL_EDITOR, "Success!");
  954. logi(LEVEL_EDITOR, out_path);
  955. free(out_path);
  956. }
  957. else if ( result == NFD_CANCEL )
  958. {
  959. logi(LEVEL_EDITOR, "User pressed cancel.");
  960. }
  961. else
  962. {
  963. loge(LEVEL_EDITOR, "Error: %s\n", NFD_GetError());
  964. }
  965. }
  966. if (ImGui::MenuItem("Save", "Ctrl+S"))
  967. {
  968. }
  969. if (ImGui::MenuItem("Save As..."))
  970. {
  971. }
  972. ImGui::Separator();
  973. if (ImGui::MenuItem("Quit", "Ctrl+Q", false, true))
  974. {
  975. TempAllocator64 ta;
  976. StringStream ss(ta);
  977. tool::device_quit(ss);
  978. send_command(ss);
  979. }
  980. ImGui::EndMenu();
  981. }
  982. if (ImGui::BeginMenu("Edit"))
  983. {
  984. if (ImGui::MenuItem("Undo", "Ctrl+Z")) {}
  985. if (ImGui::MenuItem("Redo", "Ctrl+Y")) {}
  986. ImGui::EndMenu();
  987. }
  988. if (ImGui::BeginMenu("Create"))
  989. {
  990. TempAllocator256 ta;
  991. StringStream ss(ta);
  992. if (ImGui::BeginMenu("Primitives"))
  993. {
  994. if (ImGui::MenuItem("Cube", NULL, false, true))
  995. {
  996. _tool_type = tool::ToolType::PLACE;
  997. tool_send_state();
  998. tool::set_placeable(ss, "unit", "core/units/primitives/cube");
  999. send_command(ss);
  1000. }
  1001. if (ImGui::MenuItem("Sphere"))
  1002. {
  1003. _tool_type = tool::ToolType::PLACE;
  1004. tool_send_state();
  1005. tool::set_placeable(ss, "unit", "core/units/primitives/sphere");
  1006. send_command(ss);
  1007. }
  1008. if (ImGui::MenuItem("Cone"))
  1009. {
  1010. _tool_type = tool::ToolType::PLACE;
  1011. tool_send_state();
  1012. tool::set_placeable(ss, "unit", "core/units/primitives/cone");
  1013. send_command(ss);
  1014. }
  1015. if (ImGui::MenuItem("Cylinder"))
  1016. {
  1017. _tool_type = tool::ToolType::PLACE;
  1018. tool_send_state();
  1019. tool::set_placeable(ss, "unit", "core/units/primitives/cylinder");
  1020. send_command(ss);
  1021. }
  1022. if (ImGui::MenuItem("Plane"))
  1023. {
  1024. _tool_type = tool::ToolType::PLACE;
  1025. tool_send_state();
  1026. tool::set_placeable(ss, "unit", "core/units/primitives/plane");
  1027. send_command(ss);
  1028. }
  1029. ImGui::EndMenu();
  1030. }
  1031. if (ImGui::MenuItem("Camera"))
  1032. {
  1033. _tool_type = tool::ToolType::PLACE;
  1034. tool_send_state();
  1035. tool::set_placeable(ss, "unit", "core/units/camera");
  1036. send_command(ss);
  1037. }
  1038. if (ImGui::MenuItem("Light"))
  1039. {
  1040. _tool_type = tool::ToolType::PLACE;
  1041. tool_send_state();
  1042. tool::set_placeable(ss, "unit", "core/units/light");
  1043. send_command(ss);
  1044. }
  1045. if (ImGui::MenuItem("Sound"))
  1046. {
  1047. _tool_type = tool::ToolType::PLACE;
  1048. tool_send_state();
  1049. tool::set_placeable(ss, "sound", "core/units/camera");
  1050. send_command(ss);
  1051. }
  1052. ImGui::EndMenu();
  1053. }
  1054. if (ImGui::BeginMenu("Scene View"))
  1055. {
  1056. TempAllocator256 ta;
  1057. StringStream ss(ta);
  1058. if (ImGui::BeginMenu("Camera View"))
  1059. {
  1060. if (ImGui::MenuItem("Perspective"))
  1061. {
  1062. tool::camera_view_perspective(ss);
  1063. send_command(ss);
  1064. }
  1065. if (ImGui::MenuItem("Front"))
  1066. {
  1067. tool::camera_view_front(ss);
  1068. send_command(ss);
  1069. }
  1070. if (ImGui::MenuItem("Back"))
  1071. {
  1072. tool::camera_view_back(ss);
  1073. send_command(ss);
  1074. }
  1075. if (ImGui::MenuItem("Right"))
  1076. {
  1077. tool::camera_view_right(ss);
  1078. send_command(ss);
  1079. }
  1080. if (ImGui::MenuItem("Left"))
  1081. {
  1082. tool::camera_view_left(ss);
  1083. send_command(ss);
  1084. }
  1085. if (ImGui::MenuItem("Top"))
  1086. {
  1087. tool::camera_view_top(ss);
  1088. send_command(ss);
  1089. }
  1090. if (ImGui::MenuItem("Bottom"))
  1091. {
  1092. tool::camera_view_bottom(ss);
  1093. send_command(ss);
  1094. }
  1095. ImGui::EndMenu();
  1096. }
  1097. ImGui::Separator();
  1098. if (ImGui::MenuItem("Show Grid", NULL, _show_grid))
  1099. {
  1100. _show_grid = !_show_grid;
  1101. tool_send_state();
  1102. }
  1103. if (ImGui::SliderFloat("Grid Size", &_grid_size, 0.1f, 5.0f))
  1104. {
  1105. tool_send_state();
  1106. }
  1107. if (ImGui::SliderInt("Snap Rot", &_rotation_snap, 1, 180))
  1108. {
  1109. tool_send_state();
  1110. }
  1111. ImGui::EndMenu();
  1112. }
  1113. if (ImGui::BeginMenu("Windows"))
  1114. {
  1115. if (ImGui::MenuItem("Scene"))
  1116. {
  1117. _scene_view._open = true;
  1118. }
  1119. if (ImGui::MenuItem("Scene Tree"))
  1120. {
  1121. _scene_tree._open = true;
  1122. }
  1123. if (ImGui::MenuItem("Inspector"))
  1124. {
  1125. _inspector._open = true;
  1126. }
  1127. if (ImGui::MenuItem("Console"))
  1128. {
  1129. _console._open = true;
  1130. }
  1131. if (ImGui::MenuItem("Animator"))
  1132. {
  1133. _animator._open = true;
  1134. }
  1135. ImGui::EndMenu();
  1136. }
  1137. if (ImGui::BeginMenu("Help"))
  1138. {
  1139. ImGui::MenuItem("About", "");
  1140. ImGui::EndMenu();
  1141. }
  1142. _main_menu_pos = ImGui::GetWindowPos();
  1143. _main_menu_size = ImGui::GetWindowSize();
  1144. ImGui::EndMainMenuBar();
  1145. }
  1146. }
  1147. };
  1148. LevelEditor* s_editor;
  1149. void tool_init()
  1150. {
  1151. const DeviceOptions& opt = device()->_device_options;
  1152. s_editor = CE_NEW(default_allocator(), LevelEditor)(opt._source_dir);
  1153. }
  1154. void tool_update(f32 dt)
  1155. {
  1156. s_editor->update(dt);
  1157. }
  1158. void tool_shutdown()
  1159. {
  1160. CE_DELETE(default_allocator(), s_editor);
  1161. }
  1162. extern bool next_event(OsEvent& ev);
  1163. bool tool_process_events()
  1164. {
  1165. ImGuiIO& io = ImGui::GetIO();
  1166. bool exit = false;
  1167. bool reset = false;
  1168. TempAllocator4096 ta;
  1169. StringStream ss(ta);
  1170. OsEvent event;
  1171. while (next_event(event))
  1172. {
  1173. if (event.type == OsEventType::NONE)
  1174. continue;
  1175. switch (event.type)
  1176. {
  1177. case OsEventType::BUTTON:
  1178. switch (event.button.device_id)
  1179. {
  1180. case crown::InputDeviceType::KEYBOARD:
  1181. io.KeyCtrl = ((event.button.button_num == crown::KeyboardButton::CTRL_LEFT)
  1182. || (event.button.button_num == crown::KeyboardButton::CTRL_RIGHT)) && event.button.pressed;
  1183. io.KeyShift = ((event.button.button_num == crown::KeyboardButton::SHIFT_LEFT)
  1184. || (event.button.button_num == crown::KeyboardButton::SHIFT_RIGHT)) && event.button.pressed;
  1185. io.KeyAlt = ((event.button.button_num == crown::KeyboardButton::ALT_LEFT)
  1186. || (event.button.button_num == crown::KeyboardButton::ALT_RIGHT)) && event.button.pressed;
  1187. io.KeySuper = ((event.button.button_num == crown::KeyboardButton::SUPER_LEFT)
  1188. || (event.button.button_num == crown::KeyboardButton::SUPER_RIGHT)) && event.button.pressed;
  1189. io.KeysDown[event.button.button_num] = event.button.pressed;
  1190. if (!io.WantCaptureKeyboard)
  1191. {
  1192. if (event.button.pressed)
  1193. {
  1194. if (event.button.button_num == crown::KeyboardButton::W)
  1195. tool::keyboard_pressed(ss, 'w');
  1196. if (event.button.button_num == crown::KeyboardButton::A)
  1197. tool::keyboard_pressed(ss, 'a');
  1198. if (event.button.button_num == crown::KeyboardButton::S)
  1199. tool::keyboard_pressed(ss, 's');
  1200. if (event.button.button_num == crown::KeyboardButton::D)
  1201. tool::keyboard_pressed(ss, 'd');
  1202. }
  1203. else
  1204. {
  1205. if (event.button.button_num == crown::KeyboardButton::W)
  1206. tool::keyboard_released(ss, 'w');
  1207. if (event.button.button_num == crown::KeyboardButton::A)
  1208. tool::keyboard_released(ss, 'a');
  1209. if (event.button.button_num == crown::KeyboardButton::S)
  1210. tool::keyboard_released(ss, 's');
  1211. if (event.button.button_num == crown::KeyboardButton::D)
  1212. tool::keyboard_released(ss, 'd');
  1213. }
  1214. }
  1215. break;
  1216. case crown::InputDeviceType::MOUSE:
  1217. io.MouseDown[0] = (event.button.button_num == crown::MouseButton::LEFT) && event.button.pressed;
  1218. io.MouseDown[1] = (event.button.button_num == crown::MouseButton::RIGHT) && event.button.pressed;
  1219. io.MouseDown[2] = (event.button.button_num == crown::MouseButton::MIDDLE) && event.button.pressed;
  1220. if (!io.WantCaptureMouse)
  1221. {
  1222. ImVec2& mouse_curr = s_editor->_scene_view._mouse_curr;
  1223. mouse_curr.x = io.MousePos.x - s_editor->_scene_view._view_origin.x;
  1224. mouse_curr.y = io.MousePos.y - s_editor->_scene_view._view_origin.y;
  1225. tool::set_mouse_state(ss
  1226. , mouse_curr.x
  1227. , mouse_curr.y
  1228. , io.MouseDown[0]
  1229. , io.MouseDown[2]
  1230. , io.MouseDown[1]
  1231. );
  1232. if (event.button.button_num == crown::MouseButton::LEFT)
  1233. {
  1234. if (event.button.pressed)
  1235. tool::mouse_down(ss, mouse_curr.x, mouse_curr.y);
  1236. else
  1237. tool::mouse_up(ss, mouse_curr.x, mouse_curr.y);
  1238. }
  1239. }
  1240. break;
  1241. }
  1242. break;
  1243. case OsEventType::AXIS:
  1244. switch(event.axis.device_id)
  1245. {
  1246. case InputDeviceType::MOUSE:
  1247. switch(event.axis.axis_num)
  1248. {
  1249. case crown::MouseAxis::CURSOR:
  1250. io.MousePos = ImVec2(event.axis.axis_x, event.axis.axis_y);
  1251. if (!io.WantCaptureMouse)
  1252. {
  1253. ImVec2& mouse_curr = s_editor->_scene_view._mouse_curr;
  1254. ImVec2& mouse_last = s_editor->_scene_view._mouse_last;
  1255. mouse_curr.x = io.MousePos.x - s_editor->_scene_view._view_origin.x;
  1256. mouse_curr.y = io.MousePos.y - s_editor->_scene_view._view_origin.y;
  1257. f32 delta_x = mouse_curr.x - mouse_last.x;
  1258. f32 delta_y = mouse_curr.y - mouse_last.y;
  1259. tool::mouse_move(ss, mouse_curr.x, mouse_curr.y, delta_x, delta_y);
  1260. mouse_last = mouse_curr;
  1261. }
  1262. break;
  1263. case crown::MouseAxis::WHEEL:
  1264. io.MouseWheel += event.axis.axis_y;
  1265. if (!io.WantCaptureMouse)
  1266. tool::mouse_wheel(ss, io.MouseWheel);
  1267. break;
  1268. }
  1269. }
  1270. break;
  1271. case OsEventType::TEXT:
  1272. io.AddInputCharactersUTF8((const char*) event.text.utf8);
  1273. break;
  1274. case OsEventType::RESOLUTION:
  1275. _width = event.resolution.width;
  1276. _height = event.resolution.height;
  1277. reset = true;
  1278. break;
  1279. case OsEventType::EXIT:
  1280. exit = true;
  1281. break;
  1282. default:
  1283. break;
  1284. }
  1285. }
  1286. s_editor->send_command(ss);
  1287. bool vsync = true;
  1288. if (reset)
  1289. bgfx::reset(_width, _height, (vsync ? BGFX_RESET_VSYNC : BGFX_RESET_NONE));
  1290. return exit;
  1291. }
  1292. } // namespace crown
  1293. #endif // CROWN_TOOLS