FontPicker.cpp 8.6 KB

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