FontPicker.cpp 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. /*
  2. * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/QuickSort.h>
  7. #include <LibGUI/Button.h>
  8. #include <LibGUI/FontPicker.h>
  9. #include <LibGUI/FontPickerDialogGML.h>
  10. #include <LibGUI/FontPickerWeightModel.h>
  11. #include <LibGUI/Label.h>
  12. #include <LibGUI/ListView.h>
  13. #include <LibGUI/Scrollbar.h>
  14. #include <LibGUI/SpinBox.h>
  15. #include <LibGUI/Widget.h>
  16. #include <LibGfx/FontDatabase.h>
  17. namespace GUI {
  18. FontPicker::FontPicker(Window* parent_window, const Gfx::Font* current_font, bool fixed_width_only)
  19. : Dialog(parent_window)
  20. , m_fixed_width_only(fixed_width_only)
  21. {
  22. set_title("Font picker");
  23. resize(430, 280);
  24. set_icon(Gfx::Bitmap::load_from_file("/res/icons/16x16/app-font-editor.png"));
  25. auto& widget = set_main_widget<GUI::Widget>();
  26. if (!widget.load_from_gml(font_picker_dialog_gml))
  27. VERIFY_NOT_REACHED();
  28. m_family_list_view = *widget.find_descendant_of_type_named<ListView>("family_list_view");
  29. m_family_list_view->set_model(ItemListModel<String>::create(m_families));
  30. m_family_list_view->horizontal_scrollbar().set_visible(false);
  31. m_weight_list_view = *widget.find_descendant_of_type_named<ListView>("weight_list_view");
  32. m_weight_list_view->set_model(adopt_ref(*new FontWeightListModel(m_weights)));
  33. m_weight_list_view->horizontal_scrollbar().set_visible(false);
  34. m_size_spin_box = *widget.find_descendant_of_type_named<SpinBox>("size_spin_box");
  35. m_size_spin_box->set_range(1, 255);
  36. m_size_list_view = *widget.find_descendant_of_type_named<ListView>("size_list_view");
  37. m_size_list_view->set_model(ItemListModel<int>::create(m_sizes));
  38. m_size_list_view->horizontal_scrollbar().set_visible(false);
  39. m_sample_text_label = *widget.find_descendant_of_type_named<Label>("sample_text_label");
  40. m_families.clear();
  41. Gfx::FontDatabase::the().for_each_typeface([&](auto& typeface) {
  42. if (m_fixed_width_only && !typeface.is_fixed_width())
  43. return;
  44. if (!m_families.contains_slow(typeface.family()))
  45. m_families.append(typeface.family());
  46. });
  47. quick_sort(m_families);
  48. m_family_list_view->on_selection_change = [this] {
  49. const auto& index = m_family_list_view->selection().first();
  50. m_family = index.data().to_string();
  51. m_weights.clear();
  52. Gfx::FontDatabase::the().for_each_typeface([&](auto& typeface) {
  53. if (m_fixed_width_only && !typeface.is_fixed_width())
  54. return;
  55. if (typeface.family() == m_family.value() && !m_weights.contains_slow(typeface.weight())) {
  56. m_weights.append(typeface.weight());
  57. }
  58. });
  59. quick_sort(m_weights);
  60. Optional<size_t> index_of_old_weight_in_new_list;
  61. if (m_weight.has_value())
  62. index_of_old_weight_in_new_list = m_weights.find_first_index(m_weight.value());
  63. m_weight_list_view->model()->update();
  64. m_weight_list_view->set_cursor(m_weight_list_view->model()->index(index_of_old_weight_in_new_list.value_or(0)), GUI::AbstractView::SelectionUpdate::Set);
  65. update_font();
  66. };
  67. m_weight_list_view->on_selection_change = [this] {
  68. const auto& index = m_weight_list_view->selection().first();
  69. bool font_is_fixed_size = false;
  70. m_weight = index.data(ModelRole::Custom).to_i32();
  71. m_sizes.clear();
  72. dbgln("Selected weight: {}", m_weight.value());
  73. Gfx::FontDatabase::the().for_each_typeface([&](auto& typeface) {
  74. if (m_fixed_width_only && !typeface.is_fixed_width())
  75. return;
  76. if (typeface.family() == m_family.value() && (int)typeface.weight() == m_weight.value()) {
  77. font_is_fixed_size = typeface.is_fixed_size();
  78. if (font_is_fixed_size) {
  79. m_size_spin_box->set_visible(false);
  80. typeface.for_each_fixed_size_font([&](auto& font) {
  81. m_sizes.append(font.presentation_size());
  82. });
  83. } else {
  84. m_size_spin_box->set_visible(true);
  85. m_sizes.append(8);
  86. m_sizes.append(10);
  87. m_sizes.append(12);
  88. m_sizes.append(14);
  89. m_sizes.append(16);
  90. m_sizes.append(18);
  91. m_sizes.append(20);
  92. m_sizes.append(22);
  93. m_sizes.append(24);
  94. m_sizes.append(36);
  95. }
  96. }
  97. });
  98. quick_sort(m_sizes);
  99. m_size_list_view->model()->update();
  100. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::SingleSelection);
  101. if (m_size.has_value()) {
  102. Optional<size_t> index_of_old_size_in_new_list = m_sizes.find_first_index(m_size.value());
  103. if (index_of_old_size_in_new_list.has_value()) {
  104. m_size_list_view->set_cursor(m_size_list_view->model()->index(index_of_old_size_in_new_list.value()), GUI::AbstractView::SelectionUpdate::Set);
  105. } else {
  106. if (font_is_fixed_size) {
  107. m_size_list_view->set_cursor(m_size_list_view->model()->index(0), GUI::AbstractView::SelectionUpdate::Set);
  108. } else {
  109. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::NoSelection);
  110. m_size_spin_box->set_value(m_size.value());
  111. }
  112. }
  113. } else {
  114. m_size_list_view->set_cursor(m_size_list_view->model()->index(0), GUI::AbstractView::SelectionUpdate::Set);
  115. }
  116. update_font();
  117. };
  118. m_size_list_view->on_selection_change = [this] {
  119. const auto& index = m_size_list_view->selection().first();
  120. auto size = index.data().to_i32();
  121. Optional<size_t> index_of_new_size_in_list = m_sizes.find_first_index(size);
  122. if (index_of_new_size_in_list.has_value()) {
  123. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::SingleSelection);
  124. m_size = size;
  125. m_size_spin_box->set_value(m_size.value());
  126. }
  127. update_font();
  128. };
  129. m_size_spin_box->on_change = [this](int value) {
  130. m_size = value;
  131. Optional<size_t> index_of_new_size_in_list = m_sizes.find_first_index(m_size.value());
  132. if (index_of_new_size_in_list.has_value()) {
  133. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::SingleSelection);
  134. m_size_list_view->set_cursor(m_size_list_view->model()->index(index_of_new_size_in_list.value()), GUI::AbstractView::SelectionUpdate::Set);
  135. } else {
  136. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::NoSelection);
  137. }
  138. update_font();
  139. };
  140. auto& ok_button = *widget.find_descendant_of_type_named<GUI::Button>("ok_button");
  141. ok_button.on_click = [this](auto) {
  142. done(ExecOK);
  143. };
  144. auto& cancel_button = *widget.find_descendant_of_type_named<GUI::Button>("cancel_button");
  145. cancel_button.on_click = [this](auto) {
  146. done(ExecCancel);
  147. };
  148. set_font(current_font);
  149. }
  150. FontPicker::~FontPicker()
  151. {
  152. }
  153. void FontPicker::set_font(const Gfx::Font* font)
  154. {
  155. if (m_font == font)
  156. return;
  157. m_font = font;
  158. m_sample_text_label->set_font(m_font);
  159. if (!m_font) {
  160. m_family = {};
  161. m_weight = {};
  162. m_size = {};
  163. m_weights.clear();
  164. m_sizes.clear();
  165. m_weight_list_view->model()->update();
  166. m_size_list_view->model()->update();
  167. return;
  168. }
  169. m_family = font->family();
  170. m_weight = font->weight();
  171. m_size = font->presentation_size();
  172. auto family_index = m_families.find_first_index(m_font->family());
  173. if (family_index.has_value())
  174. m_family_list_view->set_cursor(m_family_list_view->model()->index(family_index.value()), GUI::AbstractView::SelectionUpdate::Set);
  175. auto weight_index = m_weights.find_first_index(m_font->weight());
  176. if (weight_index.has_value()) {
  177. m_weight_list_view->set_cursor(m_weight_list_view->model()->index(weight_index.value()), GUI::AbstractView::SelectionUpdate::Set);
  178. }
  179. auto size_index = m_sizes.find_first_index(m_font->presentation_size());
  180. if (size_index.has_value())
  181. m_size_list_view->set_cursor(m_size_list_view->model()->index(size_index.value()), GUI::AbstractView::SelectionUpdate::Set);
  182. }
  183. void FontPicker::update_font()
  184. {
  185. if (m_family.has_value() && m_size.has_value() && m_weight.has_value()) {
  186. m_font = Gfx::FontDatabase::the().get(m_family.value(), m_size.value(), m_weight.value());
  187. m_sample_text_label->set_font(m_font);
  188. }
  189. }
  190. }