FontPicker.cpp 8.7 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"sv).release_value_but_fixme_should_propagate_errors());
  25. auto widget = set_main_widget<GUI::Widget>().release_value_but_fixme_should_propagate_errors();
  26. widget->load_from_gml(font_picker_dialog_gml).release_value_but_fixme_should_propagate_errors();
  27. m_family_list_view = *widget->find_descendant_of_type_named<ListView>("family_list_view");
  28. m_family_list_view->set_model(ItemListModel<DeprecatedString>::create(m_families));
  29. m_family_list_view->horizontal_scrollbar().set_visible(false);
  30. m_variant_list_view = *widget->find_descendant_of_type_named<ListView>("variant_list_view");
  31. m_variant_list_view->set_model(ItemListModel<DeprecatedString>::create(m_variants));
  32. m_variant_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_deprecated_string();
  50. m_variants.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_variants.contains_slow(typeface.variant()))
  55. m_variants.append(typeface.variant());
  56. });
  57. quick_sort(m_variants);
  58. Optional<size_t> index_of_old_variant_in_new_list;
  59. if (m_variant.has_value())
  60. index_of_old_variant_in_new_list = m_variants.find_first_index(m_variant.value());
  61. m_variant_list_view->model()->invalidate();
  62. 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);
  63. update_font();
  64. };
  65. m_variant_list_view->on_selection_change = [this] {
  66. const auto& index = m_variant_list_view->selection().first();
  67. bool font_is_fixed_size = false;
  68. m_variant = index.data().to_deprecated_string();
  69. m_sizes.clear();
  70. Gfx::FontDatabase::the().for_each_typeface([&](auto& typeface) {
  71. if (m_fixed_width_only && !typeface.is_fixed_width())
  72. return;
  73. if (typeface.family() == m_family.value() && typeface.variant() == m_variant.value()) {
  74. font_is_fixed_size = typeface.is_fixed_size();
  75. if (font_is_fixed_size) {
  76. m_size_spin_box->set_visible(false);
  77. typeface.for_each_fixed_size_font([&](auto& font) {
  78. m_sizes.append(font.presentation_size());
  79. });
  80. } else {
  81. m_size_spin_box->set_visible(true);
  82. m_sizes.append(8);
  83. m_sizes.append(10);
  84. m_sizes.append(12);
  85. m_sizes.append(14);
  86. m_sizes.append(16);
  87. m_sizes.append(18);
  88. m_sizes.append(20);
  89. m_sizes.append(22);
  90. m_sizes.append(24);
  91. m_sizes.append(36);
  92. }
  93. }
  94. });
  95. quick_sort(m_sizes);
  96. m_size_list_view->model()->invalidate();
  97. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::SingleSelection);
  98. if (m_size.has_value()) {
  99. Optional<size_t> index_of_old_size_in_new_list = m_sizes.find_first_index(m_size.value());
  100. if (index_of_old_size_in_new_list.has_value()) {
  101. m_size_list_view->set_cursor(m_size_list_view->model()->index(index_of_old_size_in_new_list.value()), GUI::AbstractView::SelectionUpdate::Set);
  102. } else {
  103. if (font_is_fixed_size) {
  104. m_size_list_view->set_cursor(m_size_list_view->model()->index(0), GUI::AbstractView::SelectionUpdate::Set);
  105. } else {
  106. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::NoSelection);
  107. m_size_spin_box->set_value(m_size.value());
  108. }
  109. }
  110. } else {
  111. m_size_list_view->set_cursor(m_size_list_view->model()->index(0), GUI::AbstractView::SelectionUpdate::Set);
  112. }
  113. update_font();
  114. };
  115. m_size_list_view->on_selection_change = [this] {
  116. const auto& index = m_size_list_view->selection().first();
  117. auto size = index.data().to_i32();
  118. Optional<size_t> index_of_new_size_in_list = m_sizes.find_first_index(size);
  119. if (index_of_new_size_in_list.has_value()) {
  120. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::SingleSelection);
  121. m_size = size;
  122. m_size_spin_box->set_value(m_size.value());
  123. }
  124. update_font();
  125. };
  126. m_size_spin_box->on_change = [this](int value) {
  127. m_size = value;
  128. Optional<size_t> index_of_new_size_in_list = m_sizes.find_first_index(m_size.value());
  129. if (index_of_new_size_in_list.has_value()) {
  130. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::SingleSelection);
  131. m_size_list_view->set_cursor(m_size_list_view->model()->index(index_of_new_size_in_list.value()), GUI::AbstractView::SelectionUpdate::Set);
  132. } else {
  133. m_size_list_view->set_selection_mode(GUI::AbstractView::SelectionMode::NoSelection);
  134. }
  135. update_font();
  136. };
  137. auto& ok_button = *widget->find_descendant_of_type_named<GUI::Button>("ok_button");
  138. ok_button.on_click = [this](auto) {
  139. done(ExecResult::OK);
  140. };
  141. ok_button.set_default(true);
  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. }