TerminalSettingsWidget.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302
  1. /*
  2. * Copyright (c) 2018-2021, the SerenityOS developers.
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include "TerminalSettingsWidget.h"
  7. #include <AK/Assertions.h>
  8. #include <AK/JsonObject.h>
  9. #include <AK/QuickSort.h>
  10. #include <Applications/TerminalSettings/TerminalSettingsMainGML.h>
  11. #include <Applications/TerminalSettings/TerminalSettingsViewGML.h>
  12. #include <LibConfig/Client.h>
  13. #include <LibCore/DirIterator.h>
  14. #include <LibCore/File.h>
  15. #include <LibGUI/Application.h>
  16. #include <LibGUI/Button.h>
  17. #include <LibGUI/CheckBox.h>
  18. #include <LibGUI/FontPicker.h>
  19. #include <LibGUI/ItemListModel.h>
  20. #include <LibGUI/Label.h>
  21. #include <LibGUI/MessageBox.h>
  22. #include <LibGUI/OpacitySlider.h>
  23. #include <LibGUI/RadioButton.h>
  24. #include <LibGUI/SpinBox.h>
  25. #include <LibGUI/Widget.h>
  26. #include <LibGfx/Font/Font.h>
  27. #include <LibGfx/Font/FontDatabase.h>
  28. #include <LibKeyboard/CharacterMap.h>
  29. #include <LibVT/TerminalWidget.h>
  30. #include <spawn.h>
  31. TerminalSettingsMainWidget::TerminalSettingsMainWidget()
  32. {
  33. load_from_gml(terminal_settings_main_gml);
  34. auto& beep_bell_radio = *find_descendant_of_type_named<GUI::RadioButton>("beep_bell_radio");
  35. auto& visual_bell_radio = *find_descendant_of_type_named<GUI::RadioButton>("visual_bell_radio");
  36. auto& no_bell_radio = *find_descendant_of_type_named<GUI::RadioButton>("no_bell_radio");
  37. m_bell_mode = parse_bell(Config::read_string("Terminal", "Window", "Bell"));
  38. m_original_bell_mode = m_bell_mode;
  39. switch (m_bell_mode) {
  40. case VT::TerminalWidget::BellMode::Visible:
  41. visual_bell_radio.set_checked(true, GUI::AllowCallback::No);
  42. break;
  43. case VT::TerminalWidget::BellMode::AudibleBeep:
  44. beep_bell_radio.set_checked(true, GUI::AllowCallback::No);
  45. break;
  46. case VT::TerminalWidget::BellMode::Disabled:
  47. no_bell_radio.set_checked(true, GUI::AllowCallback::No);
  48. break;
  49. }
  50. beep_bell_radio.on_checked = [this](bool) {
  51. m_bell_mode = VT::TerminalWidget::BellMode::AudibleBeep;
  52. Config::write_string("Terminal", "Window", "Bell", stringify_bell(m_bell_mode));
  53. set_modified(true);
  54. };
  55. visual_bell_radio.on_checked = [this](bool) {
  56. m_bell_mode = VT::TerminalWidget::BellMode::Visible;
  57. Config::write_string("Terminal", "Window", "Bell", stringify_bell(m_bell_mode));
  58. set_modified(true);
  59. };
  60. no_bell_radio.on_checked = [this](bool) {
  61. m_bell_mode = VT::TerminalWidget::BellMode::Disabled;
  62. Config::write_string("Terminal", "Window", "Bell", stringify_bell(m_bell_mode));
  63. set_modified(true);
  64. };
  65. m_max_history_size = Config::read_i32("Terminal", "Terminal", "MaxHistorySize");
  66. m_original_max_history_size = m_max_history_size;
  67. auto& history_size_spinbox = *find_descendant_of_type_named<GUI::SpinBox>("history_size_spinbox");
  68. history_size_spinbox.set_value(m_max_history_size, GUI::AllowCallback::No);
  69. history_size_spinbox.on_change = [this](int value) {
  70. m_max_history_size = value;
  71. Config::write_i32("Terminal", "Terminal", "MaxHistorySize", static_cast<i32>(m_max_history_size));
  72. set_modified(true);
  73. };
  74. m_show_scrollbar = Config::read_bool("Terminal", "Terminal", "ShowScrollBar", true);
  75. m_orignal_show_scrollbar = m_show_scrollbar;
  76. auto& show_scrollbar_checkbox = *find_descendant_of_type_named<GUI::CheckBox>("terminal_show_scrollbar");
  77. show_scrollbar_checkbox.on_checked = [&](bool show_scrollbar) {
  78. m_show_scrollbar = show_scrollbar;
  79. Config::write_bool("Terminal", "Terminal", "ShowScrollBar", show_scrollbar);
  80. set_modified(true);
  81. };
  82. show_scrollbar_checkbox.set_checked(m_show_scrollbar, GUI::AllowCallback::No);
  83. m_confirm_close = Config::read_bool("Terminal", "Terminal", "ConfirmClose", true);
  84. m_orignal_confirm_close = m_confirm_close;
  85. auto& confirm_close_checkbox = *find_descendant_of_type_named<GUI::CheckBox>("terminal_confirm_close");
  86. confirm_close_checkbox.on_checked = [&](bool confirm_close) {
  87. m_confirm_close = confirm_close;
  88. Config::write_bool("Terminal", "Terminal", "ConfirmClose", confirm_close);
  89. set_modified(true);
  90. };
  91. confirm_close_checkbox.set_checked(m_confirm_close, GUI::AllowCallback::No);
  92. }
  93. TerminalSettingsViewWidget::TerminalSettingsViewWidget()
  94. {
  95. load_from_gml(terminal_settings_view_gml);
  96. auto& slider = *find_descendant_of_type_named<GUI::OpacitySlider>("background_opacity_slider");
  97. m_opacity = Config::read_i32("Terminal", "Window", "Opacity");
  98. m_original_opacity = m_opacity;
  99. slider.set_value(m_opacity);
  100. slider.on_change = [this](int value) {
  101. m_opacity = value;
  102. Config::write_i32("Terminal", "Window", "Opacity", static_cast<i32>(m_opacity));
  103. set_modified(true);
  104. };
  105. m_color_scheme = Config::read_string("Terminal", "Window", "ColorScheme");
  106. m_original_color_scheme = m_color_scheme;
  107. // The settings window takes a reference to this vector, so it needs to outlive this scope.
  108. // As long as we ensure that only one settings window may be open at a time (which we do),
  109. // this should cause no problems.
  110. static Vector<String> color_scheme_names;
  111. color_scheme_names.clear();
  112. Core::DirIterator iterator("/res/terminal-colors", Core::DirIterator::SkipParentAndBaseDir);
  113. while (iterator.has_next()) {
  114. auto path = iterator.next_path();
  115. color_scheme_names.append(path.replace(".ini", "", ReplaceMode::FirstOnly));
  116. }
  117. quick_sort(color_scheme_names);
  118. auto& color_scheme_combo = *find_descendant_of_type_named<GUI::ComboBox>("color_scheme_combo");
  119. color_scheme_combo.set_only_allow_values_from_model(true);
  120. color_scheme_combo.set_model(*GUI::ItemListModel<String>::create(color_scheme_names));
  121. color_scheme_combo.set_selected_index(color_scheme_names.find_first_index(m_color_scheme).value());
  122. color_scheme_combo.set_enabled(color_scheme_names.size() > 1);
  123. color_scheme_combo.on_change = [&](auto&, const GUI::ModelIndex& index) {
  124. m_color_scheme = index.data().as_string();
  125. Config::write_string("Terminal", "Window", "ColorScheme", m_color_scheme);
  126. set_modified(true);
  127. };
  128. auto& font_button = *find_descendant_of_type_named<GUI::Button>("terminal_font_button");
  129. auto& font_text = *find_descendant_of_type_named<GUI::Label>("terminal_font_label");
  130. auto font_name = Config::read_string("Terminal", "Text", "Font");
  131. if (font_name.is_empty())
  132. m_font = Gfx::FontDatabase::the().default_fixed_width_font();
  133. else
  134. m_font = Gfx::FontDatabase::the().get_by_name(font_name);
  135. m_original_font = m_font;
  136. font_text.set_text(m_font->human_readable_name());
  137. font_text.set_font(m_font);
  138. font_button.on_click = [&](auto) {
  139. auto picker = GUI::FontPicker::construct(window(), m_font.ptr(), true);
  140. if (picker->exec() == GUI::Dialog::ExecResult::OK) {
  141. m_font = picker->font();
  142. font_text.set_text(m_font->human_readable_name());
  143. font_text.set_font(m_font);
  144. Config::write_string("Terminal", "Text", "Font", m_font->qualified_name());
  145. set_modified(true);
  146. }
  147. };
  148. auto& font_selection = *find_descendant_of_type_named<GUI::Widget>("terminal_font_selection");
  149. auto& use_default_font_button = *find_descendant_of_type_named<GUI::CheckBox>("terminal_font_defaulted");
  150. use_default_font_button.on_checked = [&, font_name](bool use_default_font) {
  151. if (use_default_font) {
  152. font_selection.set_enabled(false);
  153. m_font = Gfx::FontDatabase::the().default_fixed_width_font();
  154. font_text.set_text(m_font->human_readable_name());
  155. font_text.set_font(m_font);
  156. Config::write_string("Terminal", "Text", "Font", m_font->qualified_name());
  157. } else {
  158. font_selection.set_enabled(true);
  159. m_font = font_name.is_empty()
  160. ? Gfx::FontDatabase::the().default_fixed_width_font()
  161. : Gfx::FontDatabase::the().get_by_name(font_name);
  162. Config::write_string("Terminal", "Text", "Font", m_font->qualified_name());
  163. }
  164. set_modified(true);
  165. };
  166. // The "use default font" setting is not stored itself - we automatically set it if the actually present font is the default,
  167. // whether that was filled in by the above defaulting code or by the user.
  168. use_default_font_button.set_checked(m_font == Gfx::FontDatabase::the().default_fixed_width_font(), GUI::AllowCallback::No);
  169. font_selection.set_enabled(!use_default_font_button.is_checked());
  170. auto& terminal_cursor_block = *find_descendant_of_type_named<GUI::RadioButton>("terminal_cursor_block");
  171. auto& terminal_cursor_underline = *find_descendant_of_type_named<GUI::RadioButton>("terminal_cursor_underline");
  172. auto& terminal_cursor_bar = *find_descendant_of_type_named<GUI::RadioButton>("terminal_cursor_bar");
  173. auto& terminal_cursor_blinking = *find_descendant_of_type_named<GUI::CheckBox>("terminal_cursor_blinking");
  174. m_cursor_shape = VT::TerminalWidget::parse_cursor_shape(Config::read_string("Terminal", "Cursor", "Shape")).value_or(VT::CursorShape::Block);
  175. m_original_cursor_shape = m_cursor_shape;
  176. m_cursor_is_blinking_set = Config::read_bool("Terminal", "Cursor", "Blinking", true);
  177. m_original_cursor_is_blinking_set = m_cursor_is_blinking_set;
  178. switch (m_cursor_shape) {
  179. case VT::CursorShape::Underline:
  180. terminal_cursor_underline.set_checked(true);
  181. break;
  182. case VT::CursorShape::Bar:
  183. terminal_cursor_bar.set_checked(true);
  184. break;
  185. default:
  186. terminal_cursor_block.set_checked(true);
  187. }
  188. terminal_cursor_blinking.on_checked = [&](bool is_checked) {
  189. set_modified(true);
  190. m_cursor_is_blinking_set = is_checked;
  191. Config::write_bool("Terminal", "Cursor", "Blinking", is_checked);
  192. };
  193. terminal_cursor_blinking.set_checked(Config::read_bool("Terminal", "Cursor", "Blinking", true));
  194. terminal_cursor_block.on_checked = [&](bool) {
  195. set_modified(true);
  196. m_cursor_shape = VT::CursorShape::Block;
  197. Config::write_string("Terminal", "Cursor", "Shape", "Block");
  198. };
  199. terminal_cursor_block.set_checked(Config::read_string("Terminal", "Cursor", "Shape") == "Block");
  200. terminal_cursor_underline.on_checked = [&](bool) {
  201. set_modified(true);
  202. m_cursor_shape = VT::CursorShape::Underline;
  203. Config::write_string("Terminal", "Cursor", "Shape", "Underline");
  204. };
  205. terminal_cursor_underline.set_checked(Config::read_string("Terminal", "Cursor", "Shape") == "Underline");
  206. terminal_cursor_bar.on_checked = [&](bool) {
  207. set_modified(true);
  208. m_cursor_shape = VT::CursorShape::Bar;
  209. Config::write_string("Terminal", "Cursor", "Shape", "Bar");
  210. };
  211. terminal_cursor_bar.set_checked(Config::read_string("Terminal", "Cursor", "Shape") == "Bar");
  212. }
  213. VT::TerminalWidget::BellMode TerminalSettingsMainWidget::parse_bell(StringView bell_string)
  214. {
  215. if (bell_string == "AudibleBeep")
  216. return VT::TerminalWidget::BellMode::AudibleBeep;
  217. if (bell_string == "Visible")
  218. return VT::TerminalWidget::BellMode::Visible;
  219. if (bell_string == "Disabled")
  220. return VT::TerminalWidget::BellMode::Disabled;
  221. VERIFY_NOT_REACHED();
  222. }
  223. String TerminalSettingsMainWidget::stringify_bell(VT::TerminalWidget::BellMode bell_mode)
  224. {
  225. if (bell_mode == VT::TerminalWidget::BellMode::AudibleBeep)
  226. return "AudibleBeep";
  227. if (bell_mode == VT::TerminalWidget::BellMode::Disabled)
  228. return "Disabled";
  229. if (bell_mode == VT::TerminalWidget::BellMode::Visible)
  230. return "Visible";
  231. VERIFY_NOT_REACHED();
  232. }
  233. void TerminalSettingsMainWidget::apply_settings()
  234. {
  235. m_original_max_history_size = m_max_history_size;
  236. m_orignal_show_scrollbar = m_show_scrollbar;
  237. m_original_bell_mode = m_bell_mode;
  238. m_orignal_confirm_close = m_confirm_close;
  239. write_back_settings();
  240. }
  241. void TerminalSettingsMainWidget::write_back_settings() const
  242. {
  243. Config::write_i32("Terminal", "Terminal", "MaxHistorySize", static_cast<i32>(m_original_max_history_size));
  244. Config::write_bool("Terminal", "Terminal", "ShowScrollBar", m_orignal_show_scrollbar);
  245. Config::write_bool("Terminal", "Terminal", "ConfirmClose", m_orignal_confirm_close);
  246. Config::write_string("Terminal", "Window", "Bell", stringify_bell(m_original_bell_mode));
  247. }
  248. void TerminalSettingsMainWidget::cancel_settings()
  249. {
  250. write_back_settings();
  251. }
  252. void TerminalSettingsViewWidget::apply_settings()
  253. {
  254. m_original_opacity = m_opacity;
  255. m_original_font = m_font;
  256. m_original_color_scheme = m_color_scheme;
  257. m_original_cursor_shape = m_cursor_shape;
  258. m_original_cursor_is_blinking_set = m_cursor_is_blinking_set;
  259. write_back_settings();
  260. }
  261. void TerminalSettingsViewWidget::write_back_settings() const
  262. {
  263. Config::write_i32("Terminal", "Window", "Opacity", static_cast<i32>(m_original_opacity));
  264. Config::write_string("Terminal", "Text", "Font", m_original_font->qualified_name());
  265. Config::write_string("Terminal", "Window", "ColorScheme", m_original_color_scheme);
  266. Config::write_string("Terminal", "Cursor", "Shape", VT::TerminalWidget::stringify_cursor_shape(m_original_cursor_shape));
  267. Config::write_bool("Terminal", "Cursor", "Blinking", m_original_cursor_is_blinking_set);
  268. }
  269. void TerminalSettingsViewWidget::cancel_settings()
  270. {
  271. write_back_settings();
  272. }