FontPicker.cpp 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  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. m_size = index.data().to_i32();
  121. m_size_spin_box->set_value(m_size.value());
  122. update_font();
  123. };
  124. m_size_spin_box->on_change = [this](int value) {
  125. m_size = value;
  126. Optional<size_t> index_of_new_size_in_list = m_sizes.find_first_index(m_size.value());
  127. if (index_of_new_size_in_list.has_value()) {
  128. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::SingleSelection);
  129. m_size_list_view->set_cursor(m_size_list_view->model()->index(index_of_new_size_in_list.value()), GUI::AbstractView::SelectionUpdate::Set);
  130. } else {
  131. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::NoSelection);
  132. }
  133. update_font();
  134. };
  135. auto& ok_button = *widget.find_descendant_of_type_named<GUI::Button>("ok_button");
  136. ok_button.on_click = [this](auto) {
  137. done(ExecOK);
  138. };
  139. auto& cancel_button = *widget.find_descendant_of_type_named<GUI::Button>("cancel_button");
  140. cancel_button.on_click = [this](auto) {
  141. done(ExecCancel);
  142. };
  143. set_font(current_font);
  144. }
  145. FontPicker::~FontPicker()
  146. {
  147. }
  148. void FontPicker::set_font(const Gfx::Font* font)
  149. {
  150. if (m_font == font)
  151. return;
  152. m_font = font;
  153. m_sample_text_label->set_font(m_font);
  154. if (!m_font) {
  155. m_family = {};
  156. m_weight = {};
  157. m_size = {};
  158. m_weights.clear();
  159. m_sizes.clear();
  160. m_weight_list_view->model()->update();
  161. m_size_list_view->model()->update();
  162. return;
  163. }
  164. m_family = font->family();
  165. m_weight = font->weight();
  166. m_size = font->presentation_size();
  167. auto family_index = m_families.find_first_index(m_font->family());
  168. if (family_index.has_value())
  169. m_family_list_view->set_cursor(m_family_list_view->model()->index(family_index.value()), GUI::AbstractView::SelectionUpdate::Set);
  170. auto weight_index = m_weights.find_first_index(m_font->weight());
  171. if (weight_index.has_value()) {
  172. m_weight_list_view->set_cursor(m_weight_list_view->model()->index(weight_index.value()), GUI::AbstractView::SelectionUpdate::Set);
  173. }
  174. auto size_index = m_sizes.find_first_index(m_font->presentation_size());
  175. if (size_index.has_value())
  176. m_size_list_view->set_cursor(m_size_list_view->model()->index(size_index.value()), GUI::AbstractView::SelectionUpdate::Set);
  177. }
  178. void FontPicker::update_font()
  179. {
  180. if (m_family.has_value() && m_size.has_value() && m_weight.has_value()) {
  181. m_font = Gfx::FontDatabase::the().get(m_family.value(), m_size.value(), m_weight.value());
  182. m_sample_text_label->set_font(m_font);
  183. }
  184. }
  185. }