2
0

theme_editor_plugin.cpp 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917
  1. /*************************************************************************/
  2. /* theme_editor_plugin.cpp */
  3. /*************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /*************************************************************************/
  8. /* Copyright (c) 2007-2021 Juan Linietsky, Ariel Manzur. */
  9. /* Copyright (c) 2014-2021 Godot Engine contributors (cf. AUTHORS.md). */
  10. /* */
  11. /* Permission is hereby granted, free of charge, to any person obtaining */
  12. /* a copy of this software and associated documentation files (the */
  13. /* "Software"), to deal in the Software without restriction, including */
  14. /* without limitation the rights to use, copy, modify, merge, publish, */
  15. /* distribute, sublicense, and/or sell copies of the Software, and to */
  16. /* permit persons to whom the Software is furnished to do so, subject to */
  17. /* the following conditions: */
  18. /* */
  19. /* The above copyright notice and this permission notice shall be */
  20. /* included in all copies or substantial portions of the Software. */
  21. /* */
  22. /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
  23. /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
  24. /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
  25. /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
  26. /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
  27. /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
  28. /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
  29. /*************************************************************************/
  30. #include "theme_editor_plugin.h"
  31. #include "core/os/file_access.h"
  32. #include "core/version.h"
  33. #include "editor/editor_scale.h"
  34. #include "scene/gui/progress_bar.h"
  35. void ThemeEditor::edit(const Ref<Theme> &p_theme) {
  36. theme = p_theme;
  37. main_panel->set_theme(p_theme);
  38. main_container->set_theme(p_theme);
  39. }
  40. void ThemeEditor::_propagate_redraw(Control *p_at) {
  41. p_at->notification(NOTIFICATION_THEME_CHANGED);
  42. p_at->minimum_size_changed();
  43. p_at->update();
  44. for (int i = 0; i < p_at->get_child_count(); i++) {
  45. Control *a = Object::cast_to<Control>(p_at->get_child(i));
  46. if (a)
  47. _propagate_redraw(a);
  48. }
  49. }
  50. void ThemeEditor::_refresh_interval() {
  51. _propagate_redraw(main_panel);
  52. _propagate_redraw(main_container);
  53. }
  54. void ThemeEditor::_type_menu_cbk(int p_option) {
  55. type_edit->set_text(type_menu->get_popup()->get_item_text(p_option));
  56. }
  57. void ThemeEditor::_name_menu_about_to_show() {
  58. String fromtype = type_edit->get_text();
  59. List<StringName> names;
  60. if (popup_mode == POPUP_ADD) {
  61. switch (type_select->get_selected()) {
  62. case 0: Theme::get_default()->get_icon_list(fromtype, &names); break;
  63. case 1: Theme::get_default()->get_stylebox_list(fromtype, &names); break;
  64. case 2: Theme::get_default()->get_font_list(fromtype, &names); break;
  65. case 3: Theme::get_default()->get_color_list(fromtype, &names); break;
  66. case 4: Theme::get_default()->get_constant_list(fromtype, &names); break;
  67. }
  68. } else if (popup_mode == POPUP_REMOVE) {
  69. theme->get_icon_list(fromtype, &names);
  70. theme->get_stylebox_list(fromtype, &names);
  71. theme->get_font_list(fromtype, &names);
  72. theme->get_color_list(fromtype, &names);
  73. theme->get_constant_list(fromtype, &names);
  74. }
  75. name_menu->get_popup()->clear();
  76. name_menu->get_popup()->set_size(Size2());
  77. for (List<StringName>::Element *E = names.front(); E; E = E->next()) {
  78. name_menu->get_popup()->add_item(E->get());
  79. }
  80. }
  81. void ThemeEditor::_name_menu_cbk(int p_option) {
  82. name_edit->set_text(name_menu->get_popup()->get_item_text(p_option));
  83. }
  84. struct _TECategory {
  85. template <class T>
  86. struct RefItem {
  87. Ref<T> item;
  88. StringName name;
  89. bool operator<(const RefItem<T> &p) const { return item->get_instance_id() < p.item->get_instance_id(); }
  90. };
  91. template <class T>
  92. struct Item {
  93. T item;
  94. String name;
  95. bool operator<(const Item<T> &p) const { return name < p.name; }
  96. };
  97. Set<RefItem<StyleBox> > stylebox_items;
  98. Set<RefItem<Font> > font_items;
  99. Set<RefItem<Texture> > icon_items;
  100. Set<Item<Color> > color_items;
  101. Set<Item<int> > constant_items;
  102. };
  103. void ThemeEditor::_save_template_cbk(String fname) {
  104. String filename = file_dialog->get_current_path();
  105. Map<String, _TECategory> categories;
  106. // Fill types.
  107. List<StringName> type_list;
  108. Theme::get_default()->get_type_list(&type_list);
  109. for (List<StringName>::Element *E = type_list.front(); E; E = E->next()) {
  110. categories.insert(E->get(), _TECategory());
  111. }
  112. // Fill default theme.
  113. for (Map<String, _TECategory>::Element *E = categories.front(); E; E = E->next()) {
  114. _TECategory &tc = E->get();
  115. List<StringName> stylebox_list;
  116. Theme::get_default()->get_stylebox_list(E->key(), &stylebox_list);
  117. for (List<StringName>::Element *F = stylebox_list.front(); F; F = F->next()) {
  118. _TECategory::RefItem<StyleBox> it;
  119. it.name = F->get();
  120. it.item = Theme::get_default()->get_stylebox(F->get(), E->key());
  121. tc.stylebox_items.insert(it);
  122. }
  123. List<StringName> font_list;
  124. Theme::get_default()->get_font_list(E->key(), &font_list);
  125. for (List<StringName>::Element *F = font_list.front(); F; F = F->next()) {
  126. _TECategory::RefItem<Font> it;
  127. it.name = F->get();
  128. it.item = Theme::get_default()->get_font(F->get(), E->key());
  129. tc.font_items.insert(it);
  130. }
  131. List<StringName> icon_list;
  132. Theme::get_default()->get_icon_list(E->key(), &icon_list);
  133. for (List<StringName>::Element *F = icon_list.front(); F; F = F->next()) {
  134. _TECategory::RefItem<Texture> it;
  135. it.name = F->get();
  136. it.item = Theme::get_default()->get_icon(F->get(), E->key());
  137. tc.icon_items.insert(it);
  138. }
  139. List<StringName> color_list;
  140. Theme::get_default()->get_color_list(E->key(), &color_list);
  141. for (List<StringName>::Element *F = color_list.front(); F; F = F->next()) {
  142. _TECategory::Item<Color> it;
  143. it.name = F->get();
  144. it.item = Theme::get_default()->get_color(F->get(), E->key());
  145. tc.color_items.insert(it);
  146. }
  147. List<StringName> constant_list;
  148. Theme::get_default()->get_constant_list(E->key(), &constant_list);
  149. for (List<StringName>::Element *F = constant_list.front(); F; F = F->next()) {
  150. _TECategory::Item<int> it;
  151. it.name = F->get();
  152. it.item = Theme::get_default()->get_constant(F->get(), E->key());
  153. tc.constant_items.insert(it);
  154. }
  155. }
  156. FileAccess *file = FileAccess::open(filename, FileAccess::WRITE);
  157. ERR_FAIL_COND_MSG(!file, "Can't save theme to file '" + filename + "'.");
  158. file->store_line("; ******************* ");
  159. file->store_line("; Template Theme File ");
  160. file->store_line("; ******************* ");
  161. file->store_line("; ");
  162. file->store_line("; Theme Syntax: ");
  163. file->store_line("; ------------- ");
  164. file->store_line("; ");
  165. file->store_line("; Must be placed in section [theme]");
  166. file->store_line("; ");
  167. file->store_line("; Type.item = [value] ");
  168. file->store_line("; ");
  169. file->store_line("; [value] examples:");
  170. file->store_line("; ");
  171. file->store_line("; Type.item = 6 ; numeric constant. ");
  172. file->store_line("; Type.item = #FF00FF ; HTML color ");
  173. file->store_line("; Type.item = #55FF00FF ; HTML color with alpha 55.");
  174. file->store_line("; Type.item = icon(image.png) ; icon in a png file (relative to theme file).");
  175. file->store_line("; Type.item = font(font.xres) ; font in a resource (relative to theme file).");
  176. file->store_line("; Type.item = sbox(stylebox.xres) ; stylebox in a resource (relative to theme file).");
  177. file->store_line("; Type.item = sboxf(2,#FF00FF) ; flat stylebox with margin 2.");
  178. file->store_line("; Type.item = sboxf(2,#FF00FF,#FFFFFF) ; flat stylebox with margin 2 and border.");
  179. file->store_line("; Type.item = sboxf(2,#FF00FF,#FFFFFF,#000000) ; flat stylebox with margin 2, light & dark borders.");
  180. file->store_line("; Type.item = sboxt(base.png,2,2,2,2) ; textured stylebox with 3x3 stretch and stretch margins.");
  181. file->store_line("; -Additionally, 4 extra integers can be added to sboxf and sboxt to specify custom padding of contents:");
  182. file->store_line("; Type.item = sboxt(base.png,2,2,2,2,5,4,2,4) ;");
  183. file->store_line("; -Order for all is always left, top, right, bottom.");
  184. file->store_line("; ");
  185. file->store_line("; Special values:");
  186. file->store_line("; Type.item = default ; use the value in the default theme (must exist there).");
  187. file->store_line("; Type.item = @somebutton_color ; reference to a library value previously defined.");
  188. file->store_line("; ");
  189. file->store_line("; Library Syntax: ");
  190. file->store_line("; --------------- ");
  191. file->store_line("; ");
  192. file->store_line("; Must be placed in section [library], but usage is optional.");
  193. file->store_line("; ");
  194. file->store_line("; item = [value] ; same as Theme, but assign to library.");
  195. file->store_line("; ");
  196. file->store_line("; examples:");
  197. file->store_line("; ");
  198. file->store_line("; [library]");
  199. file->store_line("; ");
  200. file->store_line("; default_button_color = #FF00FF");
  201. file->store_line("; ");
  202. file->store_line("; [theme]");
  203. file->store_line("; ");
  204. file->store_line("; Button.color = @default_button_color ; used reference.");
  205. file->store_line("; ");
  206. file->store_line("; ******************* ");
  207. file->store_line("; ");
  208. file->store_line("; Template Generated Using: " + String(VERSION_FULL_BUILD));
  209. file->store_line("; ");
  210. file->store_line("; ");
  211. file->store_line("");
  212. file->store_line("[library]");
  213. file->store_line("");
  214. file->store_line("; place library stuff here");
  215. file->store_line("");
  216. file->store_line("[theme]");
  217. file->store_line("");
  218. file->store_line("");
  219. // Write default theme.
  220. for (Map<String, _TECategory>::Element *E = categories.front(); E; E = E->next()) {
  221. _TECategory &tc = E->get();
  222. String underline = "; ";
  223. for (int i = 0; i < E->key().length(); i++)
  224. underline += "*";
  225. file->store_line("");
  226. file->store_line(underline);
  227. file->store_line("; " + E->key());
  228. file->store_line(underline);
  229. if (tc.stylebox_items.size())
  230. file->store_line("\n; StyleBox Items:\n");
  231. for (Set<_TECategory::RefItem<StyleBox> >::Element *F = tc.stylebox_items.front(); F; F = F->next()) {
  232. file->store_line(E->key() + "." + F->get().name + " = default");
  233. }
  234. if (tc.font_items.size())
  235. file->store_line("\n; Font Items:\n");
  236. for (Set<_TECategory::RefItem<Font> >::Element *F = tc.font_items.front(); F; F = F->next()) {
  237. file->store_line(E->key() + "." + F->get().name + " = default");
  238. }
  239. if (tc.icon_items.size())
  240. file->store_line("\n; Icon Items:\n");
  241. for (Set<_TECategory::RefItem<Texture> >::Element *F = tc.icon_items.front(); F; F = F->next()) {
  242. file->store_line(E->key() + "." + F->get().name + " = default");
  243. }
  244. if (tc.color_items.size())
  245. file->store_line("\n; Color Items:\n");
  246. for (Set<_TECategory::Item<Color> >::Element *F = tc.color_items.front(); F; F = F->next()) {
  247. file->store_line(E->key() + "." + F->get().name + " = default");
  248. }
  249. if (tc.constant_items.size())
  250. file->store_line("\n; Constant Items:\n");
  251. for (Set<_TECategory::Item<int> >::Element *F = tc.constant_items.front(); F; F = F->next()) {
  252. file->store_line(E->key() + "." + F->get().name + " = default");
  253. }
  254. }
  255. file->close();
  256. memdelete(file);
  257. }
  258. void ThemeEditor::_dialog_cbk() {
  259. switch (popup_mode) {
  260. case POPUP_ADD: {
  261. switch (type_select->get_selected()) {
  262. case 0: theme->set_icon(name_edit->get_text(), type_edit->get_text(), Ref<Texture>()); break;
  263. case 1: theme->set_stylebox(name_edit->get_text(), type_edit->get_text(), Ref<StyleBox>()); break;
  264. case 2: theme->set_font(name_edit->get_text(), type_edit->get_text(), Ref<Font>()); break;
  265. case 3: theme->set_color(name_edit->get_text(), type_edit->get_text(), Color()); break;
  266. case 4: theme->set_constant(name_edit->get_text(), type_edit->get_text(), 0); break;
  267. }
  268. } break;
  269. case POPUP_CLASS_ADD: {
  270. StringName fromtype = type_edit->get_text();
  271. List<StringName> names;
  272. {
  273. names.clear();
  274. Theme::get_default()->get_icon_list(fromtype, &names);
  275. for (List<StringName>::Element *E = names.front(); E; E = E->next()) {
  276. theme->set_icon(E->get(), fromtype, Ref<Texture>());
  277. }
  278. }
  279. {
  280. names.clear();
  281. Theme::get_default()->get_stylebox_list(fromtype, &names);
  282. for (List<StringName>::Element *E = names.front(); E; E = E->next()) {
  283. theme->set_stylebox(E->get(), fromtype, Ref<StyleBox>());
  284. }
  285. }
  286. {
  287. names.clear();
  288. Theme::get_default()->get_font_list(fromtype, &names);
  289. for (List<StringName>::Element *E = names.front(); E; E = E->next()) {
  290. theme->set_font(E->get(), fromtype, Ref<Font>());
  291. }
  292. }
  293. {
  294. names.clear();
  295. Theme::get_default()->get_color_list(fromtype, &names);
  296. for (List<StringName>::Element *E = names.front(); E; E = E->next()) {
  297. theme->set_color(E->get(), fromtype, Theme::get_default()->get_color(E->get(), fromtype));
  298. }
  299. }
  300. {
  301. names.clear();
  302. Theme::get_default()->get_constant_list(fromtype, &names);
  303. for (List<StringName>::Element *E = names.front(); E; E = E->next()) {
  304. theme->set_constant(E->get(), fromtype, Theme::get_default()->get_constant(E->get(), fromtype));
  305. }
  306. }
  307. } break;
  308. case POPUP_REMOVE: {
  309. switch (type_select->get_selected()) {
  310. case 0: theme->clear_icon(name_edit->get_text(), type_edit->get_text()); break;
  311. case 1: theme->clear_stylebox(name_edit->get_text(), type_edit->get_text()); break;
  312. case 2: theme->clear_font(name_edit->get_text(), type_edit->get_text()); break;
  313. case 3: theme->clear_color(name_edit->get_text(), type_edit->get_text()); break;
  314. case 4: theme->clear_constant(name_edit->get_text(), type_edit->get_text()); break;
  315. }
  316. } break;
  317. case POPUP_CLASS_REMOVE: {
  318. StringName fromtype = type_edit->get_text();
  319. List<StringName> names;
  320. {
  321. names.clear();
  322. Theme::get_default()->get_icon_list(fromtype, &names);
  323. for (List<StringName>::Element *E = names.front(); E; E = E->next()) {
  324. theme->clear_icon(E->get(), fromtype);
  325. }
  326. }
  327. {
  328. names.clear();
  329. Theme::get_default()->get_stylebox_list(fromtype, &names);
  330. for (List<StringName>::Element *E = names.front(); E; E = E->next()) {
  331. theme->clear_stylebox(E->get(), fromtype);
  332. }
  333. }
  334. {
  335. names.clear();
  336. Theme::get_default()->get_font_list(fromtype, &names);
  337. for (List<StringName>::Element *E = names.front(); E; E = E->next()) {
  338. theme->clear_font(E->get(), fromtype);
  339. }
  340. }
  341. {
  342. names.clear();
  343. Theme::get_default()->get_color_list(fromtype, &names);
  344. for (List<StringName>::Element *E = names.front(); E; E = E->next()) {
  345. theme->clear_color(E->get(), fromtype);
  346. }
  347. }
  348. {
  349. names.clear();
  350. Theme::get_default()->get_constant_list(fromtype, &names);
  351. for (List<StringName>::Element *E = names.front(); E; E = E->next()) {
  352. theme->clear_constant(E->get(), fromtype);
  353. }
  354. }
  355. } break;
  356. }
  357. }
  358. void ThemeEditor::_theme_menu_cbk(int p_option) {
  359. if (p_option == POPUP_CREATE_EMPTY || p_option == POPUP_CREATE_EDITOR_EMPTY || p_option == POPUP_IMPORT_EDITOR_THEME) {
  360. bool import = (p_option == POPUP_IMPORT_EDITOR_THEME);
  361. Ref<Theme> base_theme;
  362. if (p_option == POPUP_CREATE_EMPTY) {
  363. base_theme = Theme::get_default();
  364. } else {
  365. base_theme = EditorNode::get_singleton()->get_theme_base()->get_theme();
  366. }
  367. {
  368. List<StringName> types;
  369. base_theme->get_type_list(&types);
  370. for (List<StringName>::Element *T = types.front(); T; T = T->next()) {
  371. StringName type = T->get();
  372. List<StringName> icons;
  373. base_theme->get_icon_list(type, &icons);
  374. for (List<StringName>::Element *E = icons.front(); E; E = E->next()) {
  375. theme->set_icon(E->get(), type, import ? base_theme->get_icon(E->get(), type) : Ref<Texture>());
  376. }
  377. List<StringName> shaders;
  378. base_theme->get_shader_list(type, &shaders);
  379. for (List<StringName>::Element *E = shaders.front(); E; E = E->next()) {
  380. theme->set_shader(E->get(), type, import ? base_theme->get_shader(E->get(), type) : Ref<Shader>());
  381. }
  382. List<StringName> styleboxs;
  383. base_theme->get_stylebox_list(type, &styleboxs);
  384. for (List<StringName>::Element *E = styleboxs.front(); E; E = E->next()) {
  385. theme->set_stylebox(E->get(), type, import ? base_theme->get_stylebox(E->get(), type) : Ref<StyleBox>());
  386. }
  387. List<StringName> fonts;
  388. base_theme->get_font_list(type, &fonts);
  389. for (List<StringName>::Element *E = fonts.front(); E; E = E->next()) {
  390. theme->set_font(E->get(), type, Ref<Font>());
  391. }
  392. List<StringName> colors;
  393. base_theme->get_color_list(type, &colors);
  394. for (List<StringName>::Element *E = colors.front(); E; E = E->next()) {
  395. theme->set_color(E->get(), type, import ? base_theme->get_color(E->get(), type) : Color());
  396. }
  397. List<StringName> constants;
  398. base_theme->get_constant_list(type, &constants);
  399. for (List<StringName>::Element *E = constants.front(); E; E = E->next()) {
  400. theme->set_constant(E->get(), type, base_theme->get_constant(E->get(), type));
  401. }
  402. }
  403. }
  404. return;
  405. }
  406. Ref<Theme> base_theme;
  407. name_select_label->show();
  408. name_hbc->show();
  409. type_select_label->show();
  410. type_select->show();
  411. if (p_option == POPUP_ADD) { // Add.
  412. add_del_dialog->set_title(TTR("Add Item"));
  413. add_del_dialog->get_ok()->set_text(TTR("Add"));
  414. add_del_dialog->popup_centered(Size2(490, 85) * EDSCALE);
  415. base_theme = Theme::get_default();
  416. } else if (p_option == POPUP_CLASS_ADD) { // Add.
  417. add_del_dialog->set_title(TTR("Add All Items"));
  418. add_del_dialog->get_ok()->set_text(TTR("Add All"));
  419. add_del_dialog->popup_centered(Size2(240, 85) * EDSCALE);
  420. base_theme = Theme::get_default();
  421. name_select_label->hide();
  422. name_hbc->hide();
  423. type_select_label->hide();
  424. type_select->hide();
  425. } else if (p_option == POPUP_REMOVE) {
  426. add_del_dialog->set_title(TTR("Remove Item"));
  427. add_del_dialog->get_ok()->set_text(TTR("Remove"));
  428. add_del_dialog->popup_centered(Size2(490, 85) * EDSCALE);
  429. base_theme = theme;
  430. } else if (p_option == POPUP_CLASS_REMOVE) {
  431. add_del_dialog->set_title(TTR("Remove All Items"));
  432. add_del_dialog->get_ok()->set_text(TTR("Remove All"));
  433. add_del_dialog->popup_centered(Size2(240, 85) * EDSCALE);
  434. base_theme = Theme::get_default();
  435. name_select_label->hide();
  436. name_hbc->hide();
  437. type_select_label->hide();
  438. type_select->hide();
  439. }
  440. popup_mode = p_option;
  441. ERR_FAIL_COND(theme.is_null());
  442. List<StringName> types;
  443. base_theme->get_type_list(&types);
  444. type_menu->get_popup()->clear();
  445. if (p_option == 0 || p_option == 1) { // Add.
  446. List<StringName> new_types;
  447. theme->get_type_list(&new_types);
  448. for (List<StringName>::Element *F = new_types.front(); F; F = F->next()) {
  449. bool found = false;
  450. for (List<StringName>::Element *E = types.front(); E; E = E->next()) {
  451. if (E->get() == F->get()) {
  452. found = true;
  453. break;
  454. }
  455. }
  456. if (!found)
  457. types.push_back(F->get());
  458. }
  459. }
  460. types.sort_custom<StringName::AlphCompare>();
  461. for (List<StringName>::Element *E = types.front(); E; E = E->next()) {
  462. type_menu->get_popup()->add_item(E->get());
  463. }
  464. }
  465. void ThemeEditor::_notification(int p_what) {
  466. switch (p_what) {
  467. case NOTIFICATION_PROCESS: {
  468. time_left -= get_process_delta_time();
  469. if (time_left < 0) {
  470. time_left = 1.5;
  471. _refresh_interval();
  472. }
  473. } break;
  474. case NOTIFICATION_THEME_CHANGED: {
  475. theme_menu->set_icon(get_icon("Theme", "EditorIcons"));
  476. } break;
  477. }
  478. }
  479. void ThemeEditor::_bind_methods() {
  480. ClassDB::bind_method("_type_menu_cbk", &ThemeEditor::_type_menu_cbk);
  481. ClassDB::bind_method("_name_menu_about_to_show", &ThemeEditor::_name_menu_about_to_show);
  482. ClassDB::bind_method("_name_menu_cbk", &ThemeEditor::_name_menu_cbk);
  483. ClassDB::bind_method("_theme_menu_cbk", &ThemeEditor::_theme_menu_cbk);
  484. ClassDB::bind_method("_dialog_cbk", &ThemeEditor::_dialog_cbk);
  485. ClassDB::bind_method("_save_template_cbk", &ThemeEditor::_save_template_cbk);
  486. }
  487. ThemeEditor::ThemeEditor() {
  488. time_left = 0;
  489. HBoxContainer *top_menu = memnew(HBoxContainer);
  490. add_child(top_menu);
  491. top_menu->add_child(memnew(Label(TTR("Preview:"))));
  492. top_menu->add_spacer(false);
  493. theme_menu = memnew(MenuButton);
  494. theme_menu->set_text(TTR("Edit Theme"));
  495. theme_menu->set_tooltip(TTR("Theme editing menu."));
  496. theme_menu->get_popup()->add_item(TTR("Add Item"), POPUP_ADD);
  497. theme_menu->get_popup()->add_item(TTR("Add Class Items"), POPUP_CLASS_ADD);
  498. theme_menu->get_popup()->add_item(TTR("Remove Item"), POPUP_REMOVE);
  499. theme_menu->get_popup()->add_item(TTR("Remove Class Items"), POPUP_CLASS_REMOVE);
  500. theme_menu->get_popup()->add_separator();
  501. theme_menu->get_popup()->add_item(TTR("Create Empty Template"), POPUP_CREATE_EMPTY);
  502. theme_menu->get_popup()->add_item(TTR("Create Empty Editor Template"), POPUP_CREATE_EDITOR_EMPTY);
  503. theme_menu->get_popup()->add_item(TTR("Create From Current Editor Theme"), POPUP_IMPORT_EDITOR_THEME);
  504. top_menu->add_child(theme_menu);
  505. theme_menu->get_popup()->connect("id_pressed", this, "_theme_menu_cbk");
  506. ScrollContainer *scroll = memnew(ScrollContainer);
  507. add_child(scroll);
  508. scroll->set_enable_v_scroll(true);
  509. scroll->set_enable_h_scroll(true);
  510. scroll->set_v_size_flags(SIZE_EXPAND_FILL);
  511. MarginContainer *root_container = memnew(MarginContainer);
  512. scroll->add_child(root_container);
  513. root_container->set_theme(Theme::get_default());
  514. root_container->set_clip_contents(true);
  515. root_container->set_custom_minimum_size(Size2(700, 0) * EDSCALE);
  516. root_container->set_v_size_flags(SIZE_EXPAND_FILL);
  517. root_container->set_h_size_flags(SIZE_EXPAND_FILL);
  518. //// Preview Controls ////
  519. main_panel = memnew(Panel);
  520. root_container->add_child(main_panel);
  521. main_container = memnew(MarginContainer);
  522. root_container->add_child(main_container);
  523. main_container->add_constant_override("margin_right", 4 * EDSCALE);
  524. main_container->add_constant_override("margin_top", 4 * EDSCALE);
  525. main_container->add_constant_override("margin_left", 4 * EDSCALE);
  526. main_container->add_constant_override("margin_bottom", 4 * EDSCALE);
  527. HBoxContainer *main_hb = memnew(HBoxContainer);
  528. main_container->add_child(main_hb);
  529. VBoxContainer *first_vb = memnew(VBoxContainer);
  530. main_hb->add_child(first_vb);
  531. first_vb->set_h_size_flags(SIZE_EXPAND_FILL);
  532. first_vb->add_constant_override("separation", 10 * EDSCALE);
  533. first_vb->add_child(memnew(Label("Label")));
  534. first_vb->add_child(memnew(Button("Button")));
  535. Button *bt = memnew(Button);
  536. bt->set_text(TTR("Toggle Button"));
  537. bt->set_toggle_mode(true);
  538. bt->set_pressed(true);
  539. first_vb->add_child(bt);
  540. bt = memnew(Button);
  541. bt->set_text(TTR("Disabled Button"));
  542. bt->set_disabled(true);
  543. first_vb->add_child(bt);
  544. ToolButton *tb = memnew(ToolButton);
  545. tb->set_text("ToolButton");
  546. first_vb->add_child(tb);
  547. CheckButton *cb = memnew(CheckButton);
  548. cb->set_text("CheckButton");
  549. first_vb->add_child(cb);
  550. CheckBox *cbx = memnew(CheckBox);
  551. cbx->set_text("CheckBox");
  552. first_vb->add_child(cbx);
  553. MenuButton *test_menu_button = memnew(MenuButton);
  554. test_menu_button->set_text("MenuButton");
  555. test_menu_button->get_popup()->add_item(TTR("Item"));
  556. test_menu_button->get_popup()->add_item(TTR("Disabled Item"));
  557. test_menu_button->get_popup()->set_item_disabled(1, true);
  558. test_menu_button->get_popup()->add_separator();
  559. test_menu_button->get_popup()->add_check_item(TTR("Check Item"));
  560. test_menu_button->get_popup()->add_check_item(TTR("Checked Item"));
  561. test_menu_button->get_popup()->set_item_checked(4, true);
  562. test_menu_button->get_popup()->add_separator();
  563. test_menu_button->get_popup()->add_radio_check_item(TTR("Radio Item"));
  564. test_menu_button->get_popup()->add_radio_check_item(TTR("Checked Radio Item"));
  565. test_menu_button->get_popup()->set_item_checked(7, true);
  566. test_menu_button->get_popup()->add_separator(TTR("Named Sep."));
  567. PopupMenu *test_submenu = memnew(PopupMenu);
  568. test_menu_button->get_popup()->add_child(test_submenu);
  569. test_submenu->set_name("submenu");
  570. test_menu_button->get_popup()->add_submenu_item(TTR("Submenu"), "submenu");
  571. test_submenu->add_item(TTR("Subitem 1"));
  572. test_submenu->add_item(TTR("Subitem 2"));
  573. first_vb->add_child(test_menu_button);
  574. OptionButton *test_option_button = memnew(OptionButton);
  575. test_option_button->add_item("OptionButton");
  576. test_option_button->add_separator();
  577. test_option_button->add_item(TTR("Has"));
  578. test_option_button->add_item(TTR("Many"));
  579. test_option_button->add_item(TTR("Options"));
  580. first_vb->add_child(test_option_button);
  581. first_vb->add_child(memnew(ColorPickerButton));
  582. VBoxContainer *second_vb = memnew(VBoxContainer);
  583. second_vb->set_h_size_flags(SIZE_EXPAND_FILL);
  584. main_hb->add_child(second_vb);
  585. second_vb->add_constant_override("separation", 10 * EDSCALE);
  586. LineEdit *le = memnew(LineEdit);
  587. le->set_text("LineEdit");
  588. second_vb->add_child(le);
  589. le = memnew(LineEdit);
  590. le->set_text(TTR("Disabled LineEdit"));
  591. le->set_editable(false);
  592. second_vb->add_child(le);
  593. TextEdit *te = memnew(TextEdit);
  594. te->set_text("TextEdit");
  595. te->set_custom_minimum_size(Size2(0, 100) * EDSCALE);
  596. second_vb->add_child(te);
  597. second_vb->add_child(memnew(SpinBox));
  598. HBoxContainer *vhb = memnew(HBoxContainer);
  599. second_vb->add_child(vhb);
  600. vhb->set_custom_minimum_size(Size2(0, 100) * EDSCALE);
  601. vhb->add_child(memnew(VSlider));
  602. VScrollBar *vsb = memnew(VScrollBar);
  603. vsb->set_page(25);
  604. vhb->add_child(vsb);
  605. vhb->add_child(memnew(VSeparator));
  606. VBoxContainer *hvb = memnew(VBoxContainer);
  607. vhb->add_child(hvb);
  608. hvb->set_alignment(ALIGN_CENTER);
  609. hvb->set_h_size_flags(SIZE_EXPAND_FILL);
  610. hvb->add_child(memnew(HSlider));
  611. HScrollBar *hsb = memnew(HScrollBar);
  612. hsb->set_page(25);
  613. hvb->add_child(hsb);
  614. HSlider *hs = memnew(HSlider);
  615. hs->set_editable(false);
  616. hvb->add_child(hs);
  617. hvb->add_child(memnew(HSeparator));
  618. ProgressBar *pb = memnew(ProgressBar);
  619. pb->set_value(50);
  620. hvb->add_child(pb);
  621. VBoxContainer *third_vb = memnew(VBoxContainer);
  622. third_vb->set_h_size_flags(SIZE_EXPAND_FILL);
  623. third_vb->add_constant_override("separation", 10 * EDSCALE);
  624. main_hb->add_child(third_vb);
  625. TabContainer *tc = memnew(TabContainer);
  626. third_vb->add_child(tc);
  627. tc->set_custom_minimum_size(Size2(0, 135) * EDSCALE);
  628. Control *tcc = memnew(Control);
  629. tcc->set_name(TTR("Tab 1"));
  630. tc->add_child(tcc);
  631. tcc = memnew(Control);
  632. tcc->set_name(TTR("Tab 2"));
  633. tc->add_child(tcc);
  634. tcc = memnew(Control);
  635. tcc->set_name(TTR("Tab 3"));
  636. tc->add_child(tcc);
  637. tc->set_tab_disabled(2, true);
  638. Tree *test_tree = memnew(Tree);
  639. third_vb->add_child(test_tree);
  640. test_tree->set_custom_minimum_size(Size2(0, 175) * EDSCALE);
  641. test_tree->add_constant_override("draw_relationship_lines", 1);
  642. TreeItem *item = test_tree->create_item();
  643. item->set_text(0, "Tree");
  644. item = test_tree->create_item(test_tree->get_root());
  645. item->set_text(0, "Item");
  646. item = test_tree->create_item(test_tree->get_root());
  647. item->set_editable(0, true);
  648. item->set_text(0, TTR("Editable Item"));
  649. TreeItem *sub_tree = test_tree->create_item(test_tree->get_root());
  650. sub_tree->set_text(0, TTR("Subtree"));
  651. item = test_tree->create_item(sub_tree);
  652. item->set_cell_mode(0, TreeItem::CELL_MODE_CHECK);
  653. item->set_editable(0, true);
  654. item->set_text(0, "Check Item");
  655. item = test_tree->create_item(sub_tree);
  656. item->set_cell_mode(0, TreeItem::CELL_MODE_RANGE);
  657. item->set_editable(0, true);
  658. item->set_range_config(0, 0, 20, 0.1);
  659. item->set_range(0, 2);
  660. item = test_tree->create_item(sub_tree);
  661. item->set_cell_mode(0, TreeItem::CELL_MODE_RANGE);
  662. item->set_editable(0, true);
  663. item->set_text(0, TTR("Has,Many,Options"));
  664. item->set_range(0, 2);
  665. main_hb->add_constant_override("separation", 20 * EDSCALE);
  666. ////////
  667. add_del_dialog = memnew(ConfirmationDialog);
  668. add_del_dialog->hide();
  669. add_child(add_del_dialog);
  670. VBoxContainer *dialog_vbc = memnew(VBoxContainer);
  671. add_del_dialog->add_child(dialog_vbc);
  672. Label *l = memnew(Label);
  673. l->set_text(TTR("Type:"));
  674. dialog_vbc->add_child(l);
  675. type_hbc = memnew(HBoxContainer);
  676. dialog_vbc->add_child(type_hbc);
  677. type_edit = memnew(LineEdit);
  678. type_edit->set_h_size_flags(SIZE_EXPAND_FILL);
  679. type_hbc->add_child(type_edit);
  680. type_menu = memnew(MenuButton);
  681. type_menu->set_flat(false);
  682. type_menu->set_text("..");
  683. type_hbc->add_child(type_menu);
  684. type_menu->get_popup()->connect("id_pressed", this, "_type_menu_cbk");
  685. l = memnew(Label);
  686. l->set_text(TTR("Name:"));
  687. dialog_vbc->add_child(l);
  688. name_select_label = l;
  689. name_hbc = memnew(HBoxContainer);
  690. dialog_vbc->add_child(name_hbc);
  691. name_edit = memnew(LineEdit);
  692. name_edit->set_h_size_flags(SIZE_EXPAND_FILL);
  693. name_hbc->add_child(name_edit);
  694. name_menu = memnew(MenuButton);
  695. type_menu->set_flat(false);
  696. name_menu->set_text("..");
  697. name_hbc->add_child(name_menu);
  698. name_menu->get_popup()->connect("about_to_show", this, "_name_menu_about_to_show");
  699. name_menu->get_popup()->connect("id_pressed", this, "_name_menu_cbk");
  700. type_select_label = memnew(Label);
  701. type_select_label->set_text(TTR("Data Type:"));
  702. dialog_vbc->add_child(type_select_label);
  703. type_select = memnew(OptionButton);
  704. type_select->add_item(TTR("Icon"));
  705. type_select->add_item(TTR("Style"));
  706. type_select->add_item(TTR("Font"));
  707. type_select->add_item(TTR("Color"));
  708. type_select->add_item(TTR("Constant"));
  709. dialog_vbc->add_child(type_select);
  710. add_del_dialog->get_ok()->connect("pressed", this, "_dialog_cbk");
  711. file_dialog = memnew(EditorFileDialog);
  712. file_dialog->add_filter("*.theme ; " + TTR("Theme File"));
  713. add_child(file_dialog);
  714. file_dialog->connect("file_selected", this, "_save_template_cbk");
  715. }
  716. void ThemeEditorPlugin::edit(Object *p_node) {
  717. if (Object::cast_to<Theme>(p_node)) {
  718. theme_editor->edit(Object::cast_to<Theme>(p_node));
  719. } else {
  720. theme_editor->edit(Ref<Theme>());
  721. }
  722. }
  723. bool ThemeEditorPlugin::handles(Object *p_node) const {
  724. return p_node->is_class("Theme");
  725. }
  726. void ThemeEditorPlugin::make_visible(bool p_visible) {
  727. if (p_visible) {
  728. theme_editor->set_process(true);
  729. button->show();
  730. editor->make_bottom_panel_item_visible(theme_editor);
  731. } else {
  732. theme_editor->set_process(false);
  733. if (theme_editor->is_visible_in_tree())
  734. editor->hide_bottom_panel();
  735. button->hide();
  736. }
  737. }
  738. ThemeEditorPlugin::ThemeEditorPlugin(EditorNode *p_node) {
  739. editor = p_node;
  740. theme_editor = memnew(ThemeEditor);
  741. theme_editor->set_custom_minimum_size(Size2(0, 200) * EDSCALE);
  742. button = editor->add_bottom_panel_item(TTR("Theme"), theme_editor);
  743. button->hide();
  744. }