item_list.cpp 45 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523
  1. /*************************************************************************/
  2. /* item_list.cpp */
  3. /*************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /*************************************************************************/
  8. /* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
  9. /* Copyright (c) 2014-2022 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 "item_list.h"
  31. #include "core/os/os.h"
  32. #include "core/project_settings.h"
  33. void ItemList::add_item(const String &p_item, const Ref<Texture> &p_texture, bool p_selectable) {
  34. Item item;
  35. item.icon = p_texture;
  36. item.icon_transposed = false;
  37. item.icon_region = Rect2i();
  38. item.icon_modulate = Color(1, 1, 1, 1);
  39. item.text = p_item;
  40. item.selectable = p_selectable;
  41. item.selected = false;
  42. item.disabled = false;
  43. item.tooltip_enabled = true;
  44. item.custom_bg = Color(0, 0, 0, 0);
  45. items.push_back(item);
  46. update();
  47. shape_changed = true;
  48. }
  49. void ItemList::add_icon_item(const Ref<Texture> &p_item, bool p_selectable) {
  50. Item item;
  51. item.icon = p_item;
  52. item.icon_transposed = false;
  53. item.icon_region = Rect2i();
  54. item.icon_modulate = Color(1, 1, 1, 1);
  55. //item.text=p_item;
  56. item.selectable = p_selectable;
  57. item.selected = false;
  58. item.disabled = false;
  59. item.tooltip_enabled = true;
  60. item.custom_bg = Color(0, 0, 0, 0);
  61. items.push_back(item);
  62. update();
  63. shape_changed = true;
  64. }
  65. void ItemList::set_item_text(int p_idx, const String &p_text) {
  66. ERR_FAIL_INDEX(p_idx, items.size());
  67. items.write[p_idx].text = p_text;
  68. update();
  69. shape_changed = true;
  70. }
  71. String ItemList::get_item_text(int p_idx) const {
  72. ERR_FAIL_INDEX_V(p_idx, items.size(), String());
  73. return items[p_idx].text;
  74. }
  75. void ItemList::set_item_tooltip_enabled(int p_idx, const bool p_enabled) {
  76. ERR_FAIL_INDEX(p_idx, items.size());
  77. items.write[p_idx].tooltip_enabled = p_enabled;
  78. }
  79. bool ItemList::is_item_tooltip_enabled(int p_idx) const {
  80. ERR_FAIL_INDEX_V(p_idx, items.size(), false);
  81. return items[p_idx].tooltip_enabled;
  82. }
  83. void ItemList::set_item_tooltip(int p_idx, const String &p_tooltip) {
  84. ERR_FAIL_INDEX(p_idx, items.size());
  85. items.write[p_idx].tooltip = p_tooltip;
  86. update();
  87. shape_changed = true;
  88. }
  89. String ItemList::get_item_tooltip(int p_idx) const {
  90. ERR_FAIL_INDEX_V(p_idx, items.size(), String());
  91. return items[p_idx].tooltip;
  92. }
  93. void ItemList::set_item_icon(int p_idx, const Ref<Texture> &p_icon) {
  94. ERR_FAIL_INDEX(p_idx, items.size());
  95. items.write[p_idx].icon = p_icon;
  96. update();
  97. shape_changed = true;
  98. }
  99. Ref<Texture> ItemList::get_item_icon(int p_idx) const {
  100. ERR_FAIL_INDEX_V(p_idx, items.size(), Ref<Texture>());
  101. return items[p_idx].icon;
  102. }
  103. void ItemList::set_item_icon_transposed(int p_idx, const bool p_transposed) {
  104. ERR_FAIL_INDEX(p_idx, items.size());
  105. items.write[p_idx].icon_transposed = p_transposed;
  106. update();
  107. shape_changed = true;
  108. }
  109. bool ItemList::is_item_icon_transposed(int p_idx) const {
  110. ERR_FAIL_INDEX_V(p_idx, items.size(), false);
  111. return items[p_idx].icon_transposed;
  112. }
  113. void ItemList::set_item_icon_region(int p_idx, const Rect2 &p_region) {
  114. ERR_FAIL_INDEX(p_idx, items.size());
  115. items.write[p_idx].icon_region = p_region;
  116. update();
  117. shape_changed = true;
  118. }
  119. Rect2 ItemList::get_item_icon_region(int p_idx) const {
  120. ERR_FAIL_INDEX_V(p_idx, items.size(), Rect2());
  121. return items[p_idx].icon_region;
  122. }
  123. void ItemList::set_item_icon_modulate(int p_idx, const Color &p_modulate) {
  124. ERR_FAIL_INDEX(p_idx, items.size());
  125. items.write[p_idx].icon_modulate = p_modulate;
  126. update();
  127. }
  128. Color ItemList::get_item_icon_modulate(int p_idx) const {
  129. ERR_FAIL_INDEX_V(p_idx, items.size(), Color());
  130. return items[p_idx].icon_modulate;
  131. }
  132. void ItemList::set_item_custom_bg_color(int p_idx, const Color &p_custom_bg_color) {
  133. ERR_FAIL_INDEX(p_idx, items.size());
  134. items.write[p_idx].custom_bg = p_custom_bg_color;
  135. update();
  136. }
  137. Color ItemList::get_item_custom_bg_color(int p_idx) const {
  138. ERR_FAIL_INDEX_V(p_idx, items.size(), Color());
  139. return items[p_idx].custom_bg;
  140. }
  141. void ItemList::set_item_custom_fg_color(int p_idx, const Color &p_custom_fg_color) {
  142. ERR_FAIL_INDEX(p_idx, items.size());
  143. items.write[p_idx].custom_fg = p_custom_fg_color;
  144. update();
  145. }
  146. Color ItemList::get_item_custom_fg_color(int p_idx) const {
  147. ERR_FAIL_INDEX_V(p_idx, items.size(), Color());
  148. return items[p_idx].custom_fg;
  149. }
  150. void ItemList::set_item_tag_icon(int p_idx, const Ref<Texture> &p_tag_icon) {
  151. ERR_FAIL_INDEX(p_idx, items.size());
  152. items.write[p_idx].tag_icon = p_tag_icon;
  153. update();
  154. shape_changed = true;
  155. }
  156. Ref<Texture> ItemList::get_item_tag_icon(int p_idx) const {
  157. ERR_FAIL_INDEX_V(p_idx, items.size(), Ref<Texture>());
  158. return items[p_idx].tag_icon;
  159. }
  160. void ItemList::set_item_selectable(int p_idx, bool p_selectable) {
  161. ERR_FAIL_INDEX(p_idx, items.size());
  162. items.write[p_idx].selectable = p_selectable;
  163. }
  164. bool ItemList::is_item_selectable(int p_idx) const {
  165. ERR_FAIL_INDEX_V(p_idx, items.size(), false);
  166. return items[p_idx].selectable;
  167. }
  168. void ItemList::set_item_disabled(int p_idx, bool p_disabled) {
  169. ERR_FAIL_INDEX(p_idx, items.size());
  170. items.write[p_idx].disabled = p_disabled;
  171. update();
  172. }
  173. bool ItemList::is_item_disabled(int p_idx) const {
  174. ERR_FAIL_INDEX_V(p_idx, items.size(), false);
  175. return items[p_idx].disabled;
  176. }
  177. void ItemList::set_item_metadata(int p_idx, const Variant &p_metadata) {
  178. ERR_FAIL_INDEX(p_idx, items.size());
  179. items.write[p_idx].metadata = p_metadata;
  180. update();
  181. shape_changed = true;
  182. }
  183. Variant ItemList::get_item_metadata(int p_idx) const {
  184. ERR_FAIL_INDEX_V(p_idx, items.size(), Variant());
  185. return items[p_idx].metadata;
  186. }
  187. void ItemList::select(int p_idx, bool p_single) {
  188. ERR_FAIL_INDEX(p_idx, items.size());
  189. if (p_single || select_mode == SELECT_SINGLE) {
  190. if (!items[p_idx].selectable || items[p_idx].disabled) {
  191. return;
  192. }
  193. for (int i = 0; i < items.size(); i++) {
  194. items.write[i].selected = p_idx == i;
  195. }
  196. current = p_idx;
  197. ensure_selected_visible = false;
  198. } else {
  199. if (items[p_idx].selectable && !items[p_idx].disabled) {
  200. items.write[p_idx].selected = true;
  201. }
  202. }
  203. update();
  204. }
  205. void ItemList::unselect(int p_idx) {
  206. ERR_FAIL_INDEX(p_idx, items.size());
  207. if (select_mode != SELECT_MULTI) {
  208. items.write[p_idx].selected = false;
  209. current = -1;
  210. } else {
  211. items.write[p_idx].selected = false;
  212. }
  213. update();
  214. }
  215. void ItemList::unselect_all() {
  216. if (items.size() < 1) {
  217. return;
  218. }
  219. for (int i = 0; i < items.size(); i++) {
  220. items.write[i].selected = false;
  221. }
  222. current = -1;
  223. update();
  224. }
  225. bool ItemList::is_selected(int p_idx) const {
  226. ERR_FAIL_INDEX_V(p_idx, items.size(), false);
  227. return items[p_idx].selected;
  228. }
  229. void ItemList::set_current(int p_current) {
  230. ERR_FAIL_INDEX(p_current, items.size());
  231. if (select_mode == SELECT_SINGLE) {
  232. select(p_current, true);
  233. } else {
  234. current = p_current;
  235. update();
  236. }
  237. }
  238. int ItemList::get_current() const {
  239. return current;
  240. }
  241. void ItemList::move_item(int p_from_idx, int p_to_idx) {
  242. ERR_FAIL_INDEX(p_from_idx, items.size());
  243. ERR_FAIL_INDEX(p_to_idx, items.size());
  244. if (is_anything_selected() && get_selected_items()[0] == p_from_idx) {
  245. current = p_to_idx;
  246. }
  247. Item item = items[p_from_idx];
  248. items.remove(p_from_idx);
  249. items.insert(p_to_idx, item);
  250. update();
  251. shape_changed = true;
  252. }
  253. int ItemList::get_item_count() const {
  254. return items.size();
  255. }
  256. void ItemList::remove_item(int p_idx) {
  257. ERR_FAIL_INDEX(p_idx, items.size());
  258. items.remove(p_idx);
  259. update();
  260. shape_changed = true;
  261. defer_select_single = -1;
  262. }
  263. void ItemList::clear() {
  264. items.clear();
  265. current = -1;
  266. ensure_selected_visible = false;
  267. update();
  268. shape_changed = true;
  269. defer_select_single = -1;
  270. }
  271. void ItemList::set_fixed_column_width(int p_size) {
  272. ERR_FAIL_COND(p_size < 0);
  273. fixed_column_width = p_size;
  274. update();
  275. shape_changed = true;
  276. }
  277. int ItemList::get_fixed_column_width() const {
  278. return fixed_column_width;
  279. }
  280. void ItemList::set_same_column_width(bool p_enable) {
  281. same_column_width = p_enable;
  282. update();
  283. shape_changed = true;
  284. }
  285. bool ItemList::is_same_column_width() const {
  286. return same_column_width;
  287. }
  288. void ItemList::set_max_text_lines(int p_lines) {
  289. ERR_FAIL_COND(p_lines < 1);
  290. max_text_lines = p_lines;
  291. update();
  292. shape_changed = true;
  293. }
  294. int ItemList::get_max_text_lines() const {
  295. return max_text_lines;
  296. }
  297. void ItemList::set_max_columns(int p_amount) {
  298. ERR_FAIL_COND(p_amount < 0);
  299. max_columns = p_amount;
  300. update();
  301. shape_changed = true;
  302. }
  303. int ItemList::get_max_columns() const {
  304. return max_columns;
  305. }
  306. void ItemList::set_select_mode(SelectMode p_mode) {
  307. select_mode = p_mode;
  308. update();
  309. }
  310. ItemList::SelectMode ItemList::get_select_mode() const {
  311. return select_mode;
  312. }
  313. void ItemList::set_icon_mode(IconMode p_mode) {
  314. ERR_FAIL_INDEX((int)p_mode, 2);
  315. icon_mode = p_mode;
  316. update();
  317. shape_changed = true;
  318. }
  319. ItemList::IconMode ItemList::get_icon_mode() const {
  320. return icon_mode;
  321. }
  322. void ItemList::set_fixed_icon_size(const Size2 &p_size) {
  323. fixed_icon_size = p_size;
  324. update();
  325. }
  326. Size2 ItemList::get_fixed_icon_size() const {
  327. return fixed_icon_size;
  328. }
  329. Size2 ItemList::Item::get_icon_size() const {
  330. if (icon.is_null()) {
  331. return Size2();
  332. }
  333. Size2 size_result = Size2(icon_region.size).abs();
  334. if (icon_region.size.x == 0 || icon_region.size.y == 0) {
  335. size_result = icon->get_size();
  336. }
  337. if (icon_transposed) {
  338. Size2 size_tmp = size_result;
  339. size_result.x = size_tmp.y;
  340. size_result.y = size_tmp.x;
  341. }
  342. return size_result;
  343. }
  344. void ItemList::_gui_input(const Ref<InputEvent> &p_event) {
  345. ERR_FAIL_COND(p_event.is_null());
  346. double prev_scroll = scroll_bar->get_value();
  347. Ref<InputEventMouseMotion> mm = p_event;
  348. if (defer_select_single >= 0 && mm.is_valid()) {
  349. defer_select_single = -1;
  350. return;
  351. }
  352. Ref<InputEventMouseButton> mb = p_event;
  353. if (defer_select_single >= 0 && mb.is_valid() && mb->get_button_index() == BUTTON_LEFT && !mb->is_pressed()) {
  354. select(defer_select_single, true);
  355. emit_signal("multi_selected", defer_select_single, true);
  356. defer_select_single = -1;
  357. return;
  358. }
  359. if (mb.is_valid() && (mb->get_button_index() == BUTTON_LEFT || (allow_rmb_select && mb->get_button_index() == BUTTON_RIGHT)) && mb->is_pressed()) {
  360. search_string = ""; //any mousepress cancels
  361. Vector2 pos = mb->get_position();
  362. Ref<StyleBox> bg = get_stylebox("bg");
  363. pos -= bg->get_offset();
  364. pos.y += scroll_bar->get_value();
  365. int closest = -1;
  366. for (int i = 0; i < items.size(); i++) {
  367. Rect2 rc = items[i].rect_cache;
  368. if (i % current_columns == current_columns - 1) {
  369. rc.size.width = get_size().width; //not right but works
  370. }
  371. if (rc.has_point(pos)) {
  372. closest = i;
  373. break;
  374. }
  375. }
  376. if (closest != -1) {
  377. int i = closest;
  378. if (select_mode == SELECT_MULTI && items[i].selected && mb->get_command()) {
  379. unselect(i);
  380. emit_signal("multi_selected", i, false);
  381. } else if (select_mode == SELECT_MULTI && mb->get_shift() && current >= 0 && current < items.size() && current != i) {
  382. int from = current;
  383. int to = i;
  384. if (i < current) {
  385. SWAP(from, to);
  386. }
  387. for (int j = from; j <= to; j++) {
  388. bool selected = !items[j].selected;
  389. select(j, false);
  390. if (selected) {
  391. emit_signal("multi_selected", j, true);
  392. }
  393. }
  394. if (mb->get_button_index() == BUTTON_RIGHT) {
  395. emit_signal("item_rmb_selected", i, get_local_mouse_position());
  396. }
  397. } else {
  398. if (!mb->is_doubleclick() && !mb->get_command() && select_mode == SELECT_MULTI && items[i].selectable && !items[i].disabled && items[i].selected && mb->get_button_index() == BUTTON_LEFT) {
  399. defer_select_single = i;
  400. return;
  401. }
  402. if (items[i].selected && mb->get_button_index() == BUTTON_RIGHT) {
  403. emit_signal("item_rmb_selected", i, get_local_mouse_position());
  404. } else {
  405. bool selected = items[i].selected;
  406. select(i, select_mode == SELECT_SINGLE || !mb->get_command());
  407. if (!selected || allow_reselect) {
  408. if (select_mode == SELECT_SINGLE) {
  409. emit_signal("item_selected", i);
  410. } else {
  411. emit_signal("multi_selected", i, true);
  412. }
  413. }
  414. if (mb->get_button_index() == BUTTON_RIGHT) {
  415. emit_signal("item_rmb_selected", i, get_local_mouse_position());
  416. } else if (/*select_mode==SELECT_SINGLE &&*/ mb->is_doubleclick()) {
  417. emit_signal("item_activated", i);
  418. }
  419. }
  420. }
  421. return;
  422. }
  423. if (mb->get_button_index() == BUTTON_RIGHT) {
  424. emit_signal("rmb_clicked", mb->get_position());
  425. return;
  426. }
  427. // Since closest is null, more likely we clicked on empty space, so send signal to interested controls. Allows, for example, implement items deselecting.
  428. emit_signal("nothing_selected");
  429. }
  430. if (mb.is_valid() && mb->get_button_index() == BUTTON_WHEEL_UP && mb->is_pressed()) {
  431. scroll_bar->set_value(scroll_bar->get_value() - scroll_bar->get_page() * mb->get_factor() / 8);
  432. }
  433. if (mb.is_valid() && mb->get_button_index() == BUTTON_WHEEL_DOWN && mb->is_pressed()) {
  434. scroll_bar->set_value(scroll_bar->get_value() + scroll_bar->get_page() * mb->get_factor() / 8);
  435. }
  436. if (p_event->is_pressed() && items.size() > 0) {
  437. if (p_event->is_action("ui_up")) {
  438. if (search_string != "") {
  439. uint64_t now = OS::get_singleton()->get_ticks_msec();
  440. uint64_t diff = now - search_time_msec;
  441. if (diff < uint64_t(ProjectSettings::get_singleton()->get("gui/timers/incremental_search_max_interval_msec")) * 2) {
  442. for (int i = current - 1; i >= 0; i--) {
  443. if (items[i].text.begins_with(search_string)) {
  444. set_current(i);
  445. ensure_current_is_visible();
  446. if (select_mode == SELECT_SINGLE) {
  447. emit_signal("item_selected", current);
  448. }
  449. break;
  450. }
  451. }
  452. accept_event();
  453. return;
  454. }
  455. }
  456. if (current >= current_columns) {
  457. set_current(current - current_columns);
  458. ensure_current_is_visible();
  459. if (select_mode == SELECT_SINGLE) {
  460. emit_signal("item_selected", current);
  461. }
  462. accept_event();
  463. }
  464. } else if (p_event->is_action("ui_down")) {
  465. if (search_string != "") {
  466. uint64_t now = OS::get_singleton()->get_ticks_msec();
  467. uint64_t diff = now - search_time_msec;
  468. if (diff < uint64_t(ProjectSettings::get_singleton()->get("gui/timers/incremental_search_max_interval_msec")) * 2) {
  469. for (int i = current + 1; i < items.size(); i++) {
  470. if (items[i].text.begins_with(search_string)) {
  471. set_current(i);
  472. ensure_current_is_visible();
  473. if (select_mode == SELECT_SINGLE) {
  474. emit_signal("item_selected", current);
  475. }
  476. break;
  477. }
  478. }
  479. accept_event();
  480. return;
  481. }
  482. }
  483. if (current < items.size() - current_columns) {
  484. set_current(current + current_columns);
  485. ensure_current_is_visible();
  486. if (select_mode == SELECT_SINGLE) {
  487. emit_signal("item_selected", current);
  488. }
  489. accept_event();
  490. }
  491. } else if (p_event->is_action("ui_page_up")) {
  492. search_string = ""; //any mousepress cancels
  493. for (int i = 4; i > 0; i--) {
  494. if (current - current_columns * i >= 0) {
  495. set_current(current - current_columns * i);
  496. ensure_current_is_visible();
  497. if (select_mode == SELECT_SINGLE) {
  498. emit_signal("item_selected", current);
  499. }
  500. accept_event();
  501. break;
  502. }
  503. }
  504. } else if (p_event->is_action("ui_page_down")) {
  505. search_string = ""; //any mousepress cancels
  506. for (int i = 4; i > 0; i--) {
  507. if (current + current_columns * i < items.size()) {
  508. set_current(current + current_columns * i);
  509. ensure_current_is_visible();
  510. if (select_mode == SELECT_SINGLE) {
  511. emit_signal("item_selected", current);
  512. }
  513. accept_event();
  514. break;
  515. }
  516. }
  517. } else if (p_event->is_action("ui_left")) {
  518. search_string = ""; //any mousepress cancels
  519. if (current % current_columns != 0) {
  520. set_current(current - 1);
  521. ensure_current_is_visible();
  522. if (select_mode == SELECT_SINGLE) {
  523. emit_signal("item_selected", current);
  524. }
  525. accept_event();
  526. }
  527. } else if (p_event->is_action("ui_right")) {
  528. search_string = ""; //any mousepress cancels
  529. if (current % current_columns != (current_columns - 1) && current + 1 < items.size()) {
  530. set_current(current + 1);
  531. ensure_current_is_visible();
  532. if (select_mode == SELECT_SINGLE) {
  533. emit_signal("item_selected", current);
  534. }
  535. accept_event();
  536. }
  537. } else if (p_event->is_action("ui_cancel")) {
  538. search_string = "";
  539. } else if (p_event->is_action("ui_select") && select_mode == SELECT_MULTI) {
  540. if (current >= 0 && current < items.size()) {
  541. if (items[current].selectable && !items[current].disabled && !items[current].selected) {
  542. select(current, false);
  543. emit_signal("multi_selected", current, true);
  544. } else if (items[current].selected) {
  545. unselect(current);
  546. emit_signal("multi_selected", current, false);
  547. }
  548. }
  549. } else if (p_event->is_action("ui_accept")) {
  550. search_string = ""; //any mousepress cance
  551. if (current >= 0 && current < items.size()) {
  552. emit_signal("item_activated", current);
  553. }
  554. } else {
  555. Ref<InputEventKey> k = p_event;
  556. if (k.is_valid() && k->get_unicode()) {
  557. uint64_t now = OS::get_singleton()->get_ticks_msec();
  558. uint64_t diff = now - search_time_msec;
  559. uint64_t max_interval = uint64_t(GLOBAL_DEF("gui/timers/incremental_search_max_interval_msec", 2000));
  560. search_time_msec = now;
  561. if (diff > max_interval) {
  562. search_string = "";
  563. }
  564. if (String::chr(k->get_unicode()) != search_string) {
  565. search_string += String::chr(k->get_unicode());
  566. }
  567. for (int i = current + 1; i <= items.size(); i++) {
  568. if (i == items.size()) {
  569. if (current == 0 || current == -1) {
  570. break;
  571. } else {
  572. i = 0;
  573. }
  574. }
  575. if (i == current) {
  576. break;
  577. }
  578. if (items[i].text.findn(search_string) == 0) {
  579. set_current(i);
  580. ensure_current_is_visible();
  581. if (select_mode == SELECT_SINGLE) {
  582. emit_signal("item_selected", current);
  583. }
  584. break;
  585. }
  586. }
  587. }
  588. }
  589. }
  590. Ref<InputEventPanGesture> pan_gesture = p_event;
  591. if (pan_gesture.is_valid()) {
  592. scroll_bar->set_value(scroll_bar->get_value() + scroll_bar->get_page() * pan_gesture->get_delta().y / 8);
  593. }
  594. if (scroll_bar->get_value() != prev_scroll) {
  595. accept_event(); //accept event if scroll changed
  596. }
  597. }
  598. void ItemList::ensure_current_is_visible() {
  599. ensure_selected_visible = true;
  600. update();
  601. }
  602. static Rect2 _adjust_to_max_size(Size2 p_size, Size2 p_max_size) {
  603. Size2 size = p_max_size;
  604. int tex_width = p_size.width * size.height / p_size.height;
  605. int tex_height = size.height;
  606. if (tex_width > size.width) {
  607. tex_width = size.width;
  608. tex_height = p_size.height * tex_width / p_size.width;
  609. }
  610. int ofs_x = (size.width - tex_width) / 2;
  611. int ofs_y = (size.height - tex_height) / 2;
  612. return Rect2(ofs_x, ofs_y, tex_width, tex_height);
  613. }
  614. void ItemList::_notification(int p_what) {
  615. if (p_what == NOTIFICATION_RESIZED) {
  616. shape_changed = true;
  617. update();
  618. }
  619. if (p_what == NOTIFICATION_DRAW) {
  620. Ref<StyleBox> bg = get_stylebox("bg");
  621. int mw = scroll_bar->get_minimum_size().x;
  622. scroll_bar->set_anchor_and_margin(MARGIN_LEFT, ANCHOR_END, -mw);
  623. scroll_bar->set_anchor_and_margin(MARGIN_RIGHT, ANCHOR_END, 0);
  624. scroll_bar->set_anchor_and_margin(MARGIN_TOP, ANCHOR_BEGIN, bg->get_margin(MARGIN_TOP));
  625. scroll_bar->set_anchor_and_margin(MARGIN_BOTTOM, ANCHOR_END, -bg->get_margin(MARGIN_BOTTOM));
  626. Size2 size = get_size();
  627. int width = size.width - bg->get_minimum_size().width;
  628. if (scroll_bar->is_visible()) {
  629. width -= mw;
  630. }
  631. draw_style_box(bg, Rect2(Point2(), size));
  632. int hseparation = get_constant("hseparation");
  633. int vseparation = get_constant("vseparation");
  634. int icon_margin = get_constant("icon_margin");
  635. int line_separation = get_constant("line_separation");
  636. Ref<StyleBox> sbsel = has_focus() ? get_stylebox("selected_focus") : get_stylebox("selected");
  637. Ref<StyleBox> cursor = has_focus() ? get_stylebox("cursor") : get_stylebox("cursor_unfocused");
  638. Ref<Font> font = get_font("font");
  639. Color guide_color = get_color("guide_color");
  640. Color font_color = get_color("font_color");
  641. Color font_color_selected = get_color("font_color_selected");
  642. int font_height = font->get_height();
  643. Vector<int> line_size_cache;
  644. Vector<int> line_limit_cache;
  645. if (max_text_lines) {
  646. line_size_cache.resize(max_text_lines);
  647. line_limit_cache.resize(max_text_lines);
  648. }
  649. if (has_focus()) {
  650. VisualServer::get_singleton()->canvas_item_add_clip_ignore(get_canvas_item(), true);
  651. draw_style_box(get_stylebox("bg_focus"), Rect2(Point2(), size));
  652. VisualServer::get_singleton()->canvas_item_add_clip_ignore(get_canvas_item(), false);
  653. }
  654. if (shape_changed) {
  655. float max_column_width = 0;
  656. //1- compute item minimum sizes
  657. for (int i = 0; i < items.size(); i++) {
  658. Size2 minsize;
  659. if (items[i].icon.is_valid()) {
  660. if (fixed_icon_size.x > 0 && fixed_icon_size.y > 0) {
  661. minsize = fixed_icon_size * icon_scale;
  662. } else {
  663. minsize = items[i].get_icon_size() * icon_scale;
  664. }
  665. if (items[i].text != "") {
  666. if (icon_mode == ICON_MODE_TOP) {
  667. minsize.y += icon_margin;
  668. } else {
  669. minsize.x += icon_margin;
  670. }
  671. }
  672. }
  673. if (items[i].text != "") {
  674. Size2 s = font->get_string_size(items[i].text);
  675. //s.width=MIN(s.width,fixed_column_width);
  676. if (icon_mode == ICON_MODE_TOP) {
  677. minsize.x = MAX(minsize.x, s.width);
  678. if (max_text_lines > 0) {
  679. minsize.y += (font_height + line_separation) * max_text_lines;
  680. } else {
  681. minsize.y += s.height;
  682. }
  683. } else {
  684. minsize.y = MAX(minsize.y, s.height);
  685. minsize.x += s.width;
  686. }
  687. }
  688. if (fixed_column_width > 0) {
  689. minsize.x = fixed_column_width;
  690. }
  691. max_column_width = MAX(max_column_width, minsize.x);
  692. // elements need to adapt to the selected size
  693. minsize.y += vseparation;
  694. minsize.x += hseparation;
  695. items.write[i].rect_cache.size = minsize;
  696. items.write[i].min_rect_cache.size = minsize;
  697. }
  698. int fit_size = size.x - bg->get_minimum_size().width - mw;
  699. //2-attempt best fit
  700. current_columns = 0x7FFFFFFF;
  701. if (max_columns > 0) {
  702. current_columns = max_columns;
  703. }
  704. while (true) {
  705. //repeat until all fits
  706. bool all_fit = true;
  707. Vector2 ofs;
  708. int col = 0;
  709. int max_h = 0;
  710. separators.clear();
  711. for (int i = 0; i < items.size(); i++) {
  712. if (current_columns > 1 && items[i].rect_cache.size.width + ofs.x > fit_size) {
  713. //went past
  714. current_columns = MAX(col, 1);
  715. all_fit = false;
  716. break;
  717. }
  718. if (same_column_width) {
  719. items.write[i].rect_cache.size.x = max_column_width;
  720. }
  721. items.write[i].rect_cache.position = ofs;
  722. max_h = MAX(max_h, items[i].rect_cache.size.y);
  723. ofs.x += items[i].rect_cache.size.x + hseparation;
  724. col++;
  725. if (col == current_columns) {
  726. if (i < items.size() - 1) {
  727. separators.push_back(ofs.y + max_h + vseparation / 2);
  728. }
  729. for (int j = i; j >= 0 && col > 0; j--, col--) {
  730. items.write[j].rect_cache.size.y = max_h;
  731. }
  732. ofs.x = 0;
  733. ofs.y += max_h + vseparation;
  734. col = 0;
  735. max_h = 0;
  736. }
  737. }
  738. for (int j = items.size() - 1; j >= 0 && col > 0; j--, col--) {
  739. items.write[j].rect_cache.size.y = max_h;
  740. }
  741. if (all_fit) {
  742. float page = MAX(0, size.height - bg->get_minimum_size().height);
  743. float max = MAX(page, ofs.y + max_h);
  744. if (auto_height) {
  745. auto_height_value = ofs.y + max_h + bg->get_minimum_size().height;
  746. }
  747. scroll_bar->set_max(max);
  748. scroll_bar->set_page(page);
  749. if (max <= page) {
  750. scroll_bar->set_value(0);
  751. scroll_bar->hide();
  752. } else {
  753. scroll_bar->show();
  754. if (do_autoscroll_to_bottom) {
  755. scroll_bar->set_value(max);
  756. }
  757. }
  758. break;
  759. }
  760. }
  761. minimum_size_changed();
  762. shape_changed = false;
  763. }
  764. //ensure_selected_visible needs to be checked before we draw the list.
  765. if (ensure_selected_visible && current >= 0 && current < items.size()) {
  766. Rect2 r = items[current].rect_cache;
  767. int from = scroll_bar->get_value();
  768. int to = from + scroll_bar->get_page();
  769. if (r.position.y < from) {
  770. scroll_bar->set_value(r.position.y);
  771. } else if (r.position.y + r.size.y > to) {
  772. scroll_bar->set_value(r.position.y + r.size.y - (to - from));
  773. }
  774. }
  775. ensure_selected_visible = false;
  776. Vector2 base_ofs = bg->get_offset();
  777. base_ofs.y -= int(scroll_bar->get_value());
  778. const Rect2 clip(-base_ofs, size); // visible frame, don't need to draw outside of there
  779. int first_item_visible;
  780. {
  781. // do a binary search to find the first item whose rect reaches below clip.position.y
  782. int lo = 0;
  783. int hi = items.size();
  784. while (lo < hi) {
  785. const int mid = (lo + hi) / 2;
  786. const Rect2 &rcache = items[mid].rect_cache;
  787. if (rcache.position.y + rcache.size.y < clip.position.y) {
  788. lo = mid + 1;
  789. } else {
  790. hi = mid;
  791. }
  792. }
  793. // we might have ended up with column 2, or 3, ..., so let's find the first column
  794. while (lo > 0 && items[lo - 1].rect_cache.position.y == items[lo].rect_cache.position.y) {
  795. lo -= 1;
  796. }
  797. first_item_visible = lo;
  798. }
  799. for (int i = first_item_visible; i < items.size(); i++) {
  800. Rect2 rcache = items[i].rect_cache;
  801. if (rcache.position.y > clip.position.y + clip.size.y) {
  802. break; // done
  803. }
  804. if (!clip.intersects(rcache)) {
  805. continue;
  806. }
  807. if (current_columns == 1) {
  808. rcache.size.width = width - rcache.position.x;
  809. }
  810. if (items[i].selected) {
  811. Rect2 r = rcache;
  812. r.position += base_ofs;
  813. r.position.y -= vseparation / 2;
  814. r.size.y += vseparation;
  815. r.position.x -= hseparation / 2;
  816. r.size.x += hseparation;
  817. draw_style_box(sbsel, r);
  818. }
  819. if (items[i].custom_bg.a > 0.001) {
  820. Rect2 r = rcache;
  821. r.position += base_ofs;
  822. // Size rect to make the align the temperature colors
  823. r.position.y -= vseparation / 2;
  824. r.size.y += vseparation;
  825. r.position.x -= hseparation / 2;
  826. r.size.x += hseparation;
  827. draw_rect(r, items[i].custom_bg);
  828. }
  829. Vector2 text_ofs;
  830. if (items[i].icon.is_valid()) {
  831. Size2 icon_size;
  832. //= _adjust_to_max_size(items[i].get_icon_size(),fixed_icon_size) * icon_scale;
  833. if (fixed_icon_size.x > 0 && fixed_icon_size.y > 0) {
  834. icon_size = fixed_icon_size * icon_scale;
  835. } else {
  836. icon_size = items[i].get_icon_size() * icon_scale;
  837. }
  838. Vector2 icon_ofs;
  839. Point2 pos = items[i].rect_cache.position + icon_ofs + base_ofs;
  840. if (icon_mode == ICON_MODE_TOP) {
  841. pos.x += Math::floor((items[i].rect_cache.size.width - icon_size.width) / 2);
  842. pos.y += MIN(
  843. Math::floor((items[i].rect_cache.size.height - icon_size.height) / 2),
  844. items[i].rect_cache.size.height - items[i].min_rect_cache.size.height);
  845. text_ofs.y = icon_size.height + icon_margin;
  846. text_ofs.y += items[i].rect_cache.size.height - items[i].min_rect_cache.size.height;
  847. } else {
  848. pos.y += Math::floor((items[i].rect_cache.size.height - icon_size.height) / 2);
  849. text_ofs.x = icon_size.width + icon_margin;
  850. }
  851. Rect2 draw_rect = Rect2(pos, icon_size);
  852. if (fixed_icon_size.x > 0 && fixed_icon_size.y > 0) {
  853. Rect2 adj = _adjust_to_max_size(items[i].get_icon_size() * icon_scale, icon_size);
  854. draw_rect.position += adj.position;
  855. draw_rect.size = adj.size;
  856. }
  857. Color modulate = items[i].icon_modulate;
  858. if (items[i].disabled) {
  859. modulate.a *= 0.5;
  860. }
  861. // If the icon is transposed, we have to switch the size so that it is drawn correctly
  862. if (items[i].icon_transposed) {
  863. Size2 size_tmp = draw_rect.size;
  864. draw_rect.size.x = size_tmp.y;
  865. draw_rect.size.y = size_tmp.x;
  866. }
  867. Rect2 region = (items[i].icon_region.size.x == 0 || items[i].icon_region.size.y == 0) ? Rect2(Vector2(), items[i].icon->get_size()) : Rect2(items[i].icon_region);
  868. draw_texture_rect_region(items[i].icon, draw_rect, region, modulate, items[i].icon_transposed);
  869. }
  870. if (items[i].tag_icon.is_valid()) {
  871. draw_texture(items[i].tag_icon, items[i].rect_cache.position + base_ofs);
  872. }
  873. if (items[i].text != "") {
  874. int max_len = -1;
  875. Vector2 size2 = font->get_string_size(items[i].text);
  876. if (fixed_column_width) {
  877. max_len = fixed_column_width;
  878. } else if (same_column_width) {
  879. max_len = items[i].rect_cache.size.x;
  880. } else {
  881. max_len = size2.x;
  882. }
  883. Color modulate = items[i].selected ? font_color_selected : (items[i].custom_fg != Color() ? items[i].custom_fg : font_color);
  884. if (items[i].disabled) {
  885. modulate.a *= 0.5;
  886. }
  887. if (icon_mode == ICON_MODE_TOP && max_text_lines > 0) {
  888. int ss = items[i].text.length();
  889. float ofs = 0;
  890. int line = 0;
  891. for (int j = 0; j <= ss; j++) {
  892. int cs = j < ss ? font->get_char_size(items[i].text[j], items[i].text[j + 1]).x : 0;
  893. if (ofs + cs > max_len || j == ss) {
  894. line_limit_cache.write[line] = j;
  895. line_size_cache.write[line] = ofs;
  896. line++;
  897. ofs = 0;
  898. if (line >= max_text_lines) {
  899. break;
  900. }
  901. } else {
  902. ofs += cs;
  903. }
  904. }
  905. line = 0;
  906. ofs = 0;
  907. text_ofs.y += font->get_ascent();
  908. text_ofs = text_ofs.floor();
  909. text_ofs += base_ofs;
  910. text_ofs += items[i].rect_cache.position;
  911. FontDrawer drawer(font, Color(1, 1, 1));
  912. for (int j = 0; j < ss; j++) {
  913. if (j == line_limit_cache[line]) {
  914. line++;
  915. ofs = 0;
  916. if (line >= max_text_lines) {
  917. break;
  918. }
  919. }
  920. ofs += drawer.draw_char(get_canvas_item(), text_ofs + Vector2(ofs + (max_len - line_size_cache[line]) / 2, line * (font_height + line_separation)).floor(), items[i].text[j], items[i].text[j + 1], modulate);
  921. }
  922. //special multiline mode
  923. } else {
  924. if (fixed_column_width > 0) {
  925. size2.x = MIN(size2.x, fixed_column_width);
  926. }
  927. if (icon_mode == ICON_MODE_TOP) {
  928. text_ofs.x += (items[i].rect_cache.size.width - size2.x) / 2;
  929. } else {
  930. text_ofs.y += (items[i].rect_cache.size.height - size2.y) / 2;
  931. }
  932. text_ofs.y += font->get_ascent();
  933. text_ofs = text_ofs.floor();
  934. text_ofs += base_ofs;
  935. text_ofs += items[i].rect_cache.position;
  936. draw_string(font, text_ofs, items[i].text, modulate, max_len + 1);
  937. }
  938. }
  939. if (select_mode == SELECT_MULTI && i == current) {
  940. Rect2 r = rcache;
  941. r.position += base_ofs;
  942. r.position.y -= vseparation / 2;
  943. r.size.y += vseparation;
  944. r.position.x -= hseparation / 2;
  945. r.size.x += hseparation;
  946. draw_style_box(cursor, r);
  947. }
  948. }
  949. int first_visible_separator = 0;
  950. {
  951. // do a binary search to find the first separator that is below clip_position.y
  952. int lo = 0;
  953. int hi = separators.size();
  954. while (lo < hi) {
  955. const int mid = (lo + hi) / 2;
  956. if (separators[mid] < clip.position.y) {
  957. lo = mid + 1;
  958. } else {
  959. hi = mid;
  960. }
  961. }
  962. first_visible_separator = lo;
  963. }
  964. for (int i = first_visible_separator; i < separators.size(); i++) {
  965. if (separators[i] > clip.position.y + clip.size.y) {
  966. break; // done
  967. }
  968. const int y = base_ofs.y + separators[i];
  969. draw_line(Vector2(bg->get_margin(MARGIN_LEFT), y), Vector2(width, y), guide_color);
  970. }
  971. }
  972. }
  973. void ItemList::_scroll_changed(double) {
  974. update();
  975. }
  976. int ItemList::get_item_at_position(const Point2 &p_pos, bool p_exact) const {
  977. Vector2 pos = p_pos;
  978. Ref<StyleBox> bg = get_stylebox("bg");
  979. pos -= bg->get_offset();
  980. pos.y += scroll_bar->get_value();
  981. int closest = -1;
  982. int closest_dist = 0x7FFFFFFF;
  983. for (int i = 0; i < items.size(); i++) {
  984. Rect2 rc = items[i].rect_cache;
  985. if (i % current_columns == current_columns - 1) {
  986. rc.size.width = get_size().width - rc.position.x; //make sure you can still select the last item when clicking past the column
  987. }
  988. if (rc.has_point(pos)) {
  989. closest = i;
  990. break;
  991. }
  992. float dist = rc.distance_to(pos);
  993. if (!p_exact && dist < closest_dist) {
  994. closest = i;
  995. closest_dist = dist;
  996. }
  997. }
  998. return closest;
  999. }
  1000. bool ItemList::is_pos_at_end_of_items(const Point2 &p_pos) const {
  1001. if (items.empty()) {
  1002. return true;
  1003. }
  1004. Vector2 pos = p_pos;
  1005. Ref<StyleBox> bg = get_stylebox("bg");
  1006. pos -= bg->get_offset();
  1007. pos.y += scroll_bar->get_value();
  1008. Rect2 endrect = items[items.size() - 1].rect_cache;
  1009. return (pos.y > endrect.position.y + endrect.size.y);
  1010. }
  1011. String ItemList::get_tooltip(const Point2 &p_pos) const {
  1012. int closest = get_item_at_position(p_pos, true);
  1013. if (closest != -1) {
  1014. if (!items[closest].tooltip_enabled) {
  1015. return "";
  1016. }
  1017. if (items[closest].tooltip != "") {
  1018. return items[closest].tooltip;
  1019. }
  1020. if (items[closest].text != "") {
  1021. return items[closest].text;
  1022. }
  1023. }
  1024. return Control::get_tooltip(p_pos);
  1025. }
  1026. void ItemList::sort_items_by_text() {
  1027. items.sort();
  1028. update();
  1029. shape_changed = true;
  1030. if (select_mode == SELECT_SINGLE) {
  1031. for (int i = 0; i < items.size(); i++) {
  1032. if (items[i].selected) {
  1033. select(i);
  1034. return;
  1035. }
  1036. }
  1037. }
  1038. }
  1039. int ItemList::find_metadata(const Variant &p_metadata) const {
  1040. for (int i = 0; i < items.size(); i++) {
  1041. if (items[i].metadata == p_metadata) {
  1042. return i;
  1043. }
  1044. }
  1045. return -1;
  1046. }
  1047. void ItemList::set_allow_rmb_select(bool p_allow) {
  1048. allow_rmb_select = p_allow;
  1049. }
  1050. bool ItemList::get_allow_rmb_select() const {
  1051. return allow_rmb_select;
  1052. }
  1053. void ItemList::set_allow_reselect(bool p_allow) {
  1054. allow_reselect = p_allow;
  1055. }
  1056. bool ItemList::get_allow_reselect() const {
  1057. return allow_reselect;
  1058. }
  1059. void ItemList::set_icon_scale(real_t p_scale) {
  1060. icon_scale = p_scale;
  1061. }
  1062. real_t ItemList::get_icon_scale() const {
  1063. return icon_scale;
  1064. }
  1065. Vector<int> ItemList::get_selected_items() {
  1066. Vector<int> selected;
  1067. for (int i = 0; i < items.size(); i++) {
  1068. if (items[i].selected) {
  1069. selected.push_back(i);
  1070. if (this->select_mode == SELECT_SINGLE) {
  1071. break;
  1072. }
  1073. }
  1074. }
  1075. return selected;
  1076. }
  1077. bool ItemList::is_anything_selected() {
  1078. for (int i = 0; i < items.size(); i++) {
  1079. if (items[i].selected) {
  1080. return true;
  1081. }
  1082. }
  1083. return false;
  1084. }
  1085. void ItemList::_set_items(const Array &p_items) {
  1086. ERR_FAIL_COND(p_items.size() % 3);
  1087. clear();
  1088. for (int i = 0; i < p_items.size(); i += 3) {
  1089. String text = p_items[i + 0];
  1090. Ref<Texture> icon = p_items[i + 1];
  1091. bool disabled = p_items[i + 2];
  1092. int idx = get_item_count();
  1093. add_item(text, icon);
  1094. set_item_disabled(idx, disabled);
  1095. }
  1096. }
  1097. Array ItemList::_get_items() const {
  1098. Array items;
  1099. for (int i = 0; i < get_item_count(); i++) {
  1100. items.push_back(get_item_text(i));
  1101. items.push_back(get_item_icon(i));
  1102. items.push_back(is_item_disabled(i));
  1103. }
  1104. return items;
  1105. }
  1106. Size2 ItemList::get_minimum_size() const {
  1107. if (auto_height) {
  1108. return Size2(0, auto_height_value);
  1109. }
  1110. return Size2();
  1111. }
  1112. void ItemList::set_autoscroll_to_bottom(const bool p_enable) {
  1113. do_autoscroll_to_bottom = p_enable;
  1114. }
  1115. void ItemList::set_auto_height(bool p_enable) {
  1116. auto_height = p_enable;
  1117. shape_changed = true;
  1118. update();
  1119. }
  1120. bool ItemList::has_auto_height() const {
  1121. return auto_height;
  1122. }
  1123. void ItemList::_bind_methods() {
  1124. ClassDB::bind_method(D_METHOD("add_item", "text", "icon", "selectable"), &ItemList::add_item, DEFVAL(Variant()), DEFVAL(true));
  1125. ClassDB::bind_method(D_METHOD("add_icon_item", "icon", "selectable"), &ItemList::add_icon_item, DEFVAL(true));
  1126. ClassDB::bind_method(D_METHOD("set_item_text", "idx", "text"), &ItemList::set_item_text);
  1127. ClassDB::bind_method(D_METHOD("get_item_text", "idx"), &ItemList::get_item_text);
  1128. ClassDB::bind_method(D_METHOD("set_item_icon", "idx", "icon"), &ItemList::set_item_icon);
  1129. ClassDB::bind_method(D_METHOD("get_item_icon", "idx"), &ItemList::get_item_icon);
  1130. ClassDB::bind_method(D_METHOD("set_item_icon_transposed", "idx", "transposed"), &ItemList::set_item_icon_transposed);
  1131. ClassDB::bind_method(D_METHOD("is_item_icon_transposed", "idx"), &ItemList::is_item_icon_transposed);
  1132. ClassDB::bind_method(D_METHOD("set_item_icon_region", "idx", "rect"), &ItemList::set_item_icon_region);
  1133. ClassDB::bind_method(D_METHOD("get_item_icon_region", "idx"), &ItemList::get_item_icon_region);
  1134. ClassDB::bind_method(D_METHOD("set_item_icon_modulate", "idx", "modulate"), &ItemList::set_item_icon_modulate);
  1135. ClassDB::bind_method(D_METHOD("get_item_icon_modulate", "idx"), &ItemList::get_item_icon_modulate);
  1136. ClassDB::bind_method(D_METHOD("set_item_selectable", "idx", "selectable"), &ItemList::set_item_selectable);
  1137. ClassDB::bind_method(D_METHOD("is_item_selectable", "idx"), &ItemList::is_item_selectable);
  1138. ClassDB::bind_method(D_METHOD("set_item_disabled", "idx", "disabled"), &ItemList::set_item_disabled);
  1139. ClassDB::bind_method(D_METHOD("is_item_disabled", "idx"), &ItemList::is_item_disabled);
  1140. ClassDB::bind_method(D_METHOD("set_item_metadata", "idx", "metadata"), &ItemList::set_item_metadata);
  1141. ClassDB::bind_method(D_METHOD("get_item_metadata", "idx"), &ItemList::get_item_metadata);
  1142. ClassDB::bind_method(D_METHOD("set_item_custom_bg_color", "idx", "custom_bg_color"), &ItemList::set_item_custom_bg_color);
  1143. ClassDB::bind_method(D_METHOD("get_item_custom_bg_color", "idx"), &ItemList::get_item_custom_bg_color);
  1144. ClassDB::bind_method(D_METHOD("set_item_custom_fg_color", "idx", "custom_fg_color"), &ItemList::set_item_custom_fg_color);
  1145. ClassDB::bind_method(D_METHOD("get_item_custom_fg_color", "idx"), &ItemList::get_item_custom_fg_color);
  1146. ClassDB::bind_method(D_METHOD("set_item_tooltip_enabled", "idx", "enable"), &ItemList::set_item_tooltip_enabled);
  1147. ClassDB::bind_method(D_METHOD("is_item_tooltip_enabled", "idx"), &ItemList::is_item_tooltip_enabled);
  1148. ClassDB::bind_method(D_METHOD("set_item_tooltip", "idx", "tooltip"), &ItemList::set_item_tooltip);
  1149. ClassDB::bind_method(D_METHOD("get_item_tooltip", "idx"), &ItemList::get_item_tooltip);
  1150. ClassDB::bind_method(D_METHOD("select", "idx", "single"), &ItemList::select, DEFVAL(true));
  1151. ClassDB::bind_method(D_METHOD("unselect", "idx"), &ItemList::unselect);
  1152. ClassDB::bind_method(D_METHOD("unselect_all"), &ItemList::unselect_all);
  1153. ClassDB::bind_method(D_METHOD("is_selected", "idx"), &ItemList::is_selected);
  1154. ClassDB::bind_method(D_METHOD("get_selected_items"), &ItemList::get_selected_items);
  1155. ClassDB::bind_method(D_METHOD("move_item", "from_idx", "to_idx"), &ItemList::move_item);
  1156. ClassDB::bind_method(D_METHOD("get_item_count"), &ItemList::get_item_count);
  1157. ClassDB::bind_method(D_METHOD("remove_item", "idx"), &ItemList::remove_item);
  1158. ClassDB::bind_method(D_METHOD("clear"), &ItemList::clear);
  1159. ClassDB::bind_method(D_METHOD("sort_items_by_text"), &ItemList::sort_items_by_text);
  1160. ClassDB::bind_method(D_METHOD("set_fixed_column_width", "width"), &ItemList::set_fixed_column_width);
  1161. ClassDB::bind_method(D_METHOD("get_fixed_column_width"), &ItemList::get_fixed_column_width);
  1162. ClassDB::bind_method(D_METHOD("set_same_column_width", "enable"), &ItemList::set_same_column_width);
  1163. ClassDB::bind_method(D_METHOD("is_same_column_width"), &ItemList::is_same_column_width);
  1164. ClassDB::bind_method(D_METHOD("set_max_text_lines", "lines"), &ItemList::set_max_text_lines);
  1165. ClassDB::bind_method(D_METHOD("get_max_text_lines"), &ItemList::get_max_text_lines);
  1166. ClassDB::bind_method(D_METHOD("set_max_columns", "amount"), &ItemList::set_max_columns);
  1167. ClassDB::bind_method(D_METHOD("get_max_columns"), &ItemList::get_max_columns);
  1168. ClassDB::bind_method(D_METHOD("set_select_mode", "mode"), &ItemList::set_select_mode);
  1169. ClassDB::bind_method(D_METHOD("get_select_mode"), &ItemList::get_select_mode);
  1170. ClassDB::bind_method(D_METHOD("set_icon_mode", "mode"), &ItemList::set_icon_mode);
  1171. ClassDB::bind_method(D_METHOD("get_icon_mode"), &ItemList::get_icon_mode);
  1172. ClassDB::bind_method(D_METHOD("set_fixed_icon_size", "size"), &ItemList::set_fixed_icon_size);
  1173. ClassDB::bind_method(D_METHOD("get_fixed_icon_size"), &ItemList::get_fixed_icon_size);
  1174. ClassDB::bind_method(D_METHOD("set_icon_scale", "scale"), &ItemList::set_icon_scale);
  1175. ClassDB::bind_method(D_METHOD("get_icon_scale"), &ItemList::get_icon_scale);
  1176. ClassDB::bind_method(D_METHOD("set_allow_rmb_select", "allow"), &ItemList::set_allow_rmb_select);
  1177. ClassDB::bind_method(D_METHOD("get_allow_rmb_select"), &ItemList::get_allow_rmb_select);
  1178. ClassDB::bind_method(D_METHOD("set_allow_reselect", "allow"), &ItemList::set_allow_reselect);
  1179. ClassDB::bind_method(D_METHOD("get_allow_reselect"), &ItemList::get_allow_reselect);
  1180. ClassDB::bind_method(D_METHOD("set_auto_height", "enable"), &ItemList::set_auto_height);
  1181. ClassDB::bind_method(D_METHOD("has_auto_height"), &ItemList::has_auto_height);
  1182. ClassDB::bind_method(D_METHOD("is_anything_selected"), &ItemList::is_anything_selected);
  1183. ClassDB::bind_method(D_METHOD("get_item_at_position", "position", "exact"), &ItemList::get_item_at_position, DEFVAL(false));
  1184. ClassDB::bind_method(D_METHOD("ensure_current_is_visible"), &ItemList::ensure_current_is_visible);
  1185. ClassDB::bind_method(D_METHOD("get_v_scroll"), &ItemList::get_v_scroll);
  1186. ClassDB::bind_method(D_METHOD("_scroll_changed"), &ItemList::_scroll_changed);
  1187. ClassDB::bind_method(D_METHOD("_gui_input"), &ItemList::_gui_input);
  1188. ClassDB::bind_method(D_METHOD("_set_items"), &ItemList::_set_items);
  1189. ClassDB::bind_method(D_METHOD("_get_items"), &ItemList::_get_items);
  1190. ADD_PROPERTY(PropertyInfo(Variant::ARRAY, "items", PROPERTY_HINT_NONE, "", PROPERTY_USAGE_NOEDITOR | PROPERTY_USAGE_INTERNAL), "_set_items", "_get_items");
  1191. ADD_PROPERTY(PropertyInfo(Variant::INT, "select_mode", PROPERTY_HINT_ENUM, "Single,Multi"), "set_select_mode", "get_select_mode");
  1192. ADD_PROPERTY(PropertyInfo(Variant::BOOL, "allow_reselect"), "set_allow_reselect", "get_allow_reselect");
  1193. ADD_PROPERTY(PropertyInfo(Variant::BOOL, "allow_rmb_select"), "set_allow_rmb_select", "get_allow_rmb_select");
  1194. ADD_PROPERTY(PropertyInfo(Variant::INT, "max_text_lines", PROPERTY_HINT_RANGE, "1,10,1,or_greater"), "set_max_text_lines", "get_max_text_lines");
  1195. ADD_PROPERTY(PropertyInfo(Variant::BOOL, "auto_height"), "set_auto_height", "has_auto_height");
  1196. ADD_GROUP("Columns", "");
  1197. ADD_PROPERTY(PropertyInfo(Variant::INT, "max_columns", PROPERTY_HINT_RANGE, "0,10,1,or_greater"), "set_max_columns", "get_max_columns");
  1198. ADD_PROPERTY(PropertyInfo(Variant::BOOL, "same_column_width"), "set_same_column_width", "is_same_column_width");
  1199. ADD_PROPERTY(PropertyInfo(Variant::INT, "fixed_column_width", PROPERTY_HINT_RANGE, "0,100,1,or_greater"), "set_fixed_column_width", "get_fixed_column_width");
  1200. ADD_GROUP("Icon", "");
  1201. ADD_PROPERTY(PropertyInfo(Variant::INT, "icon_mode", PROPERTY_HINT_ENUM, "Top,Left"), "set_icon_mode", "get_icon_mode");
  1202. ADD_PROPERTY(PropertyInfo(Variant::REAL, "icon_scale"), "set_icon_scale", "get_icon_scale");
  1203. ADD_PROPERTY(PropertyInfo(Variant::VECTOR2, "fixed_icon_size"), "set_fixed_icon_size", "get_fixed_icon_size");
  1204. BIND_ENUM_CONSTANT(ICON_MODE_TOP);
  1205. BIND_ENUM_CONSTANT(ICON_MODE_LEFT);
  1206. BIND_ENUM_CONSTANT(SELECT_SINGLE);
  1207. BIND_ENUM_CONSTANT(SELECT_MULTI);
  1208. ADD_SIGNAL(MethodInfo("item_selected", PropertyInfo(Variant::INT, "index")));
  1209. ADD_SIGNAL(MethodInfo("item_rmb_selected", PropertyInfo(Variant::INT, "index"), PropertyInfo(Variant::VECTOR2, "at_position")));
  1210. ADD_SIGNAL(MethodInfo("multi_selected", PropertyInfo(Variant::INT, "index"), PropertyInfo(Variant::BOOL, "selected")));
  1211. ADD_SIGNAL(MethodInfo("item_activated", PropertyInfo(Variant::INT, "index")));
  1212. ADD_SIGNAL(MethodInfo("rmb_clicked", PropertyInfo(Variant::VECTOR2, "at_position")));
  1213. ADD_SIGNAL(MethodInfo("nothing_selected"));
  1214. GLOBAL_DEF("gui/timers/incremental_search_max_interval_msec", 2000);
  1215. ProjectSettings::get_singleton()->set_custom_property_info("gui/timers/incremental_search_max_interval_msec", PropertyInfo(Variant::INT, "gui/timers/incremental_search_max_interval_msec", PROPERTY_HINT_RANGE, "0,10000,1,or_greater")); // No negative numbers
  1216. }
  1217. ItemList::ItemList() {
  1218. current = -1;
  1219. select_mode = SELECT_SINGLE;
  1220. icon_mode = ICON_MODE_LEFT;
  1221. fixed_column_width = 0;
  1222. same_column_width = false;
  1223. max_text_lines = 1;
  1224. max_columns = 1;
  1225. auto_height = false;
  1226. auto_height_value = 0.0f;
  1227. scroll_bar = memnew(VScrollBar);
  1228. add_child(scroll_bar);
  1229. shape_changed = true;
  1230. scroll_bar->connect("value_changed", this, "_scroll_changed");
  1231. set_focus_mode(FOCUS_ALL);
  1232. current_columns = 1;
  1233. search_time_msec = 0;
  1234. ensure_selected_visible = false;
  1235. defer_select_single = -1;
  1236. allow_rmb_select = false;
  1237. allow_reselect = false;
  1238. do_autoscroll_to_bottom = false;
  1239. icon_scale = 1.0f;
  1240. set_clip_contents(true);
  1241. }
  1242. ItemList::~ItemList() {
  1243. }