FontPicker.cpp 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. /*
  2. * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2022, the SerenityOS developers.
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <AK/QuickSort.h>
  8. #include <LibGUI/Button.h>
  9. #include <LibGUI/FontPicker.h>
  10. #include <LibGUI/FontPickerDialogGML.h>
  11. #include <LibGUI/ItemListModel.h>
  12. #include <LibGUI/Label.h>
  13. #include <LibGUI/ListView.h>
  14. #include <LibGUI/SpinBox.h>
  15. #include <LibGUI/Widget.h>
  16. #include <LibGfx/Font/FontDatabase.h>
  17. namespace GUI {
  18. FontPicker::FontPicker(Window* parent_window, Gfx::Font const* 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::try_load_from_file("/res/icons/16x16/app-font-editor.png").release_value_but_fixme_should_propagate_errors());
  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_variant_list_view = *widget.find_descendant_of_type_named<ListView>("variant_list_view");
  32. m_variant_list_view->set_model(ItemListModel<String>::create(m_variants));
  33. m_variant_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_variants.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_variants.contains_slow(typeface.variant()))
  56. m_variants.append(typeface.variant());
  57. });
  58. quick_sort(m_variants);
  59. Optional<size_t> index_of_old_variant_in_new_list;
  60. if (m_variant.has_value())
  61. index_of_old_variant_in_new_list = m_variants.find_first_index(m_variant.value());
  62. m_variant_list_view->model()->invalidate();
  63. m_variant_list_view->set_cursor(m_variant_list_view->model()->index(index_of_old_variant_in_new_list.value_or(0)), GUI::AbstractView::SelectionUpdate::Set);
  64. update_font();
  65. };
  66. m_variant_list_view->on_selection_change = [this] {
  67. const auto& index = m_variant_list_view->selection().first();
  68. bool font_is_fixed_size = false;
  69. m_variant = index.data().to_string();
  70. m_sizes.clear();
  71. Gfx::FontDatabase::the().for_each_typeface([&](auto& typeface) {
  72. if (m_fixed_width_only && !typeface.is_fixed_width())
  73. return;
  74. if (typeface.family() == m_family.value() && typeface.variant() == m_variant.value()) {
  75. font_is_fixed_size = typeface.is_fixed_size();
  76. if (font_is_fixed_size) {
  77. m_size_spin_box->set_visible(false);
  78. typeface.for_each_fixed_size_font([&](auto& font) {
  79. m_sizes.append(font.presentation_size());
  80. });
  81. } else {
  82. m_size_spin_box->set_visible(true);
  83. m_sizes.append(8);
  84. m_sizes.append(10);
  85. m_sizes.append(12);
  86. m_sizes.append(14);
  87. m_sizes.append(16);
  88. m_sizes.append(18);
  89. m_sizes.append(20);
  90. m_sizes.append(22);
  91. m_sizes.append(24);
  92. m_sizes.append(36);
  93. }
  94. }
  95. });
  96. quick_sort(m_sizes);
  97. m_size_list_view->model()->invalidate();
  98. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::SingleSelection);
  99. if (m_size.has_value()) {
  100. Optional<size_t> index_of_old_size_in_new_list = m_sizes.find_first_index(m_size.value());
  101. if (index_of_old_size_in_new_list.has_value()) {
  102. m_size_list_view->set_cursor(m_size_list_view->model()->index(index_of_old_size_in_new_list.value()), GUI::AbstractView::SelectionUpdate::Set);
  103. } else {
  104. if (font_is_fixed_size) {
  105. m_size_list_view->set_cursor(m_size_list_view->model()->index(0), GUI::AbstractView::SelectionUpdate::Set);
  106. } else {
  107. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::NoSelection);
  108. m_size_spin_box->set_value(m_size.value());
  109. }
  110. }
  111. } else {
  112. m_size_list_view->set_cursor(m_size_list_view->model()->index(0), GUI::AbstractView::SelectionUpdate::Set);
  113. }
  114. update_font();
  115. };
  116. m_size_list_view->on_selection_change = [this] {
  117. const auto& index = m_size_list_view->selection().first();
  118. auto size = index.data().to_i32();
  119. Optional<size_t> index_of_new_size_in_list = m_sizes.find_first_index(size);
  120. if (index_of_new_size_in_list.has_value()) {
  121. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::SingleSelection);
  122. m_size = size;
  123. m_size_spin_box->set_value(m_size.value());
  124. }
  125. update_font();
  126. };
  127. m_size_spin_box->on_change = [this](int value) {
  128. m_size = value;
  129. Optional<size_t> index_of_new_size_in_list = m_sizes.find_first_index(m_size.value());
  130. if (index_of_new_size_in_list.has_value()) {
  131. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::SingleSelection);
  132. m_size_list_view->set_cursor(m_size_list_view->model()->index(index_of_new_size_in_list.value()), GUI::AbstractView::SelectionUpdate::Set);
  133. } else {
  134. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::NoSelection);
  135. }
  136. update_font();
  137. };
  138. auto& ok_button = *widget.find_descendant_of_type_named<GUI::Button>("ok_button");
  139. ok_button.on_click = [this](auto) {
  140. done(ExecResult::OK);
  141. };
  142. auto& cancel_button = *widget.find_descendant_of_type_named<GUI::Button>("cancel_button");
  143. cancel_button.on_click = [this](auto) {
  144. done(ExecResult::Cancel);
  145. };
  146. set_font(current_font);
  147. }
  148. void FontPicker::set_font(Gfx::Font const* 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_variant = {};
  157. m_size = {};
  158. m_variants.clear();
  159. m_sizes.clear();
  160. m_variant_list_view->model()->invalidate();
  161. m_size_list_view->model()->invalidate();
  162. return;
  163. }
  164. m_family = font->family();
  165. m_variant = font->variant();
  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 variant_index = m_variants.find_first_index(m_font->variant());
  171. if (variant_index.has_value())
  172. m_variant_list_view->set_cursor(m_variant_list_view->model()->index(variant_index.value()), GUI::AbstractView::SelectionUpdate::Set);
  173. auto size_index = m_sizes.find_first_index(m_font->presentation_size());
  174. if (size_index.has_value())
  175. m_size_list_view->set_cursor(m_size_list_view->model()->index(size_index.value()), GUI::AbstractView::SelectionUpdate::Set);
  176. }
  177. void FontPicker::update_font()
  178. {
  179. if (m_family.has_value() && m_size.has_value() && m_variant.has_value()) {
  180. m_font = Gfx::FontDatabase::the().get(m_family.value(), m_variant.value(), m_size.value());
  181. m_sample_text_label->set_font(m_font);
  182. }
  183. }
  184. }