BookmarksBarWidget.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388
  1. /*
  2. * Copyright (c) 2020, Emanuel Sprung <emanuel.sprung@gmail.com>
  3. * Copyright (c) 2022, networkException <networkexception@serenityos.org>
  4. * Copyright (c) 2023, Cameron Youell <cameronyouell@gmail.com>
  5. *
  6. * SPDX-License-Identifier: BSD-2-Clause
  7. */
  8. #include <Applications/Browser/BookmarksBarWidget.h>
  9. #include <Applications/Browser/Browser.h>
  10. #include <Applications/Browser/EditBookmarkGML.h>
  11. #include <LibGUI/Action.h>
  12. #include <LibGUI/BoxLayout.h>
  13. #include <LibGUI/Button.h>
  14. #include <LibGUI/Dialog.h>
  15. #include <LibGUI/Event.h>
  16. #include <LibGUI/JsonArrayModel.h>
  17. #include <LibGUI/Menu.h>
  18. #include <LibGUI/MessageBox.h>
  19. #include <LibGUI/Model.h>
  20. #include <LibGUI/TextBox.h>
  21. #include <LibGUI/Widget.h>
  22. #include <LibGUI/Window.h>
  23. #include <LibGfx/Palette.h>
  24. namespace Browser {
  25. namespace {
  26. class BookmarkEditor final : public GUI::Dialog {
  27. C_OBJECT(BookmarkEditor)
  28. public:
  29. static Vector<JsonValue>
  30. edit_bookmark(Window* parent_window, StringView title, StringView url, BookmarksBarWidget::PerformEditOn perform_edit_on)
  31. {
  32. auto editor = BookmarkEditor::construct(parent_window, title, url);
  33. if (perform_edit_on == BookmarksBarWidget::PerformEditOn::NewBookmark) {
  34. editor->set_title("Add Bookmark");
  35. } else {
  36. editor->set_title("Edit Bookmark");
  37. }
  38. editor->set_icon(g_icon_bag.bookmark_filled);
  39. if (editor->exec() == ExecResult::OK) {
  40. return Vector<JsonValue> { editor->title(), editor->url() };
  41. }
  42. return {};
  43. }
  44. private:
  45. BookmarkEditor(Window* parent_window, StringView title, StringView url)
  46. : Dialog(parent_window)
  47. {
  48. auto widget = set_main_widget<GUI::Widget>().release_value_but_fixme_should_propagate_errors();
  49. widget->load_from_gml(edit_bookmark_gml).release_value_but_fixme_should_propagate_errors();
  50. set_resizable(false);
  51. resize(260, 85);
  52. m_title_textbox = *widget->find_descendant_of_type_named<GUI::TextBox>("title_textbox");
  53. m_title_textbox->set_text(title);
  54. m_title_textbox->set_focus(true);
  55. m_title_textbox->select_all();
  56. auto& ok_button = *widget->find_descendant_of_type_named<GUI::Button>("ok_button");
  57. ok_button.on_click = [this](auto) {
  58. done(ExecResult::OK);
  59. };
  60. ok_button.set_default(true);
  61. m_url_textbox = *widget->find_descendant_of_type_named<GUI::TextBox>("url_textbox");
  62. m_url_textbox->set_text(url);
  63. m_url_textbox->on_change = [this, &ok_button]() {
  64. auto has_url = !m_url_textbox->text().is_empty();
  65. ok_button.set_enabled(has_url);
  66. };
  67. auto& cancel_button = *widget->find_descendant_of_type_named<GUI::Button>("cancel_button");
  68. cancel_button.on_click = [this](auto) {
  69. done(ExecResult::Cancel);
  70. };
  71. }
  72. DeprecatedString title() const
  73. {
  74. return m_title_textbox->text();
  75. }
  76. DeprecatedString url() const
  77. {
  78. return m_url_textbox->text();
  79. }
  80. RefPtr<GUI::TextBox> m_title_textbox;
  81. RefPtr<GUI::TextBox> m_url_textbox;
  82. };
  83. }
  84. static BookmarksBarWidget* s_the;
  85. BookmarksBarWidget& BookmarksBarWidget::the()
  86. {
  87. return *s_the;
  88. }
  89. BookmarksBarWidget::BookmarksBarWidget(DeprecatedString const& bookmarks_file, bool enabled)
  90. {
  91. s_the = this;
  92. set_layout<GUI::HorizontalBoxLayout>(2, 0);
  93. set_fixed_height(20);
  94. if (!enabled)
  95. set_visible(false);
  96. m_additional = GUI::Button::construct();
  97. m_additional->set_tooltip("Show hidden bookmarks");
  98. m_additional->set_menu(m_additional_menu);
  99. auto bitmap_or_error = Gfx::Bitmap::load_from_file("/res/icons/16x16/overflow-menu.png"sv);
  100. if (!bitmap_or_error.is_error())
  101. m_additional->set_icon(bitmap_or_error.release_value());
  102. m_additional->set_button_style(Gfx::ButtonStyle::Coolbar);
  103. m_additional->set_fixed_size(22, 20);
  104. m_additional->set_focus_policy(GUI::FocusPolicy::TabFocus);
  105. m_separator = GUI::Widget::construct();
  106. m_context_menu = GUI::Menu::construct();
  107. auto default_action = GUI::Action::create(
  108. "&Open", g_icon_bag.go_to, [this](auto&) {
  109. if (on_bookmark_click)
  110. on_bookmark_click(m_context_menu_url, Open::InSameTab);
  111. },
  112. this);
  113. m_context_menu_default_action = default_action;
  114. m_context_menu->add_action(default_action);
  115. m_context_menu->add_action(GUI::Action::create(
  116. "Open in New &Tab", g_icon_bag.new_tab, [this](auto&) {
  117. if (on_bookmark_click)
  118. on_bookmark_click(m_context_menu_url, Open::InNewTab);
  119. },
  120. this));
  121. m_context_menu->add_action(GUI::Action::create(
  122. "Open in New Window", g_icon_bag.new_window, [this](auto&) {
  123. if (on_bookmark_click) {
  124. on_bookmark_click(m_context_menu_url, Open::InNewWindow);
  125. }
  126. },
  127. this));
  128. m_context_menu->add_separator();
  129. m_context_menu->add_action(GUI::Action::create(
  130. "&Edit...", g_icon_bag.rename, [this](auto&) {
  131. if (auto result = edit_bookmark(m_context_menu_url); result.is_error())
  132. GUI::MessageBox::show_error(this->window(), MUST(String::formatted("Failed to edit bookmark: {}", result.error())));
  133. },
  134. this));
  135. m_context_menu->add_action(GUI::CommonActions::make_delete_action(
  136. [this](auto&) {
  137. if (auto result = remove_bookmark(m_context_menu_url); result.is_error())
  138. GUI::MessageBox::show_error(this->window(), MUST(String::formatted("Failed to remove bookmark: {}", result.error())));
  139. },
  140. this));
  141. Vector<GUI::JsonArrayModel::FieldSpec> fields;
  142. fields.empend("title", "Title"_short_string, Gfx::TextAlignment::CenterLeft);
  143. fields.empend("url", "Url"_short_string, Gfx::TextAlignment::CenterRight);
  144. set_model(GUI::JsonArrayModel::create(bookmarks_file, move(fields)));
  145. model()->invalidate();
  146. }
  147. BookmarksBarWidget::~BookmarksBarWidget()
  148. {
  149. if (m_model)
  150. m_model->unregister_client(*this);
  151. }
  152. void BookmarksBarWidget::set_model(RefPtr<GUI::Model> model)
  153. {
  154. if (model == m_model)
  155. return;
  156. if (m_model)
  157. m_model->unregister_client(*this);
  158. m_model = move(model);
  159. m_model->register_client(*this);
  160. }
  161. void BookmarksBarWidget::resize_event(GUI::ResizeEvent& event)
  162. {
  163. Widget::resize_event(event);
  164. update_content_size();
  165. }
  166. void BookmarksBarWidget::model_did_update(unsigned)
  167. {
  168. remove_all_children();
  169. m_bookmarks.clear();
  170. int width = 0;
  171. for (int item_index = 0; item_index < model()->row_count(); ++item_index) {
  172. auto title = model()->index(item_index, 0).data().to_deprecated_string();
  173. auto url = model()->index(item_index, 1).data().to_deprecated_string();
  174. Gfx::IntRect rect { width, 0, font().width_rounded_up(title) + 32, height() };
  175. auto& button = add<GUI::Button>();
  176. m_bookmarks.append(button);
  177. button.set_button_style(Gfx::ButtonStyle::Coolbar);
  178. button.set_text(String::from_deprecated_string(title).release_value_but_fixme_should_propagate_errors());
  179. button.set_icon(g_icon_bag.filetype_html);
  180. button.set_fixed_size(font().width(title) + 32, 20);
  181. button.set_relative_rect(rect);
  182. button.set_focus_policy(GUI::FocusPolicy::TabFocus);
  183. button.set_tooltip(url);
  184. button.set_allowed_mouse_buttons_for_pressing(GUI::MouseButton::Primary | GUI::MouseButton::Middle);
  185. button.on_click = [title, url, this](auto) {
  186. if (on_bookmark_click)
  187. on_bookmark_click(url, Open::InSameTab);
  188. };
  189. button.on_middle_mouse_click = [title, url, this](auto) {
  190. if (on_bookmark_click)
  191. on_bookmark_click(url, Open::InNewTab);
  192. };
  193. button.on_context_menu_request = [this, url](auto& context_menu_event) {
  194. m_context_menu_url = url;
  195. m_context_menu->popup(context_menu_event.screen_position(), m_context_menu_default_action);
  196. };
  197. width += rect.width();
  198. }
  199. add_child(*m_separator);
  200. add_child(*m_additional);
  201. update_content_size();
  202. update();
  203. }
  204. void BookmarksBarWidget::update_content_size()
  205. {
  206. int x_position = 0;
  207. m_last_visible_index = -1;
  208. for (size_t i = 0; i < m_bookmarks.size(); ++i) {
  209. auto& bookmark = m_bookmarks.at(i);
  210. if (x_position + bookmark->width() + m_additional->width() > width()) {
  211. m_last_visible_index = i;
  212. break;
  213. }
  214. bookmark->set_x(x_position);
  215. bookmark->set_visible(true);
  216. x_position += bookmark->width();
  217. }
  218. if (m_last_visible_index < 0) {
  219. m_additional->set_visible(false);
  220. } else {
  221. // hide all items > m_last_visible_index and create new bookmarks menu for them
  222. m_additional->set_visible(true);
  223. m_additional_menu = GUI::Menu::construct("Additional Bookmarks"_string.release_value_but_fixme_should_propagate_errors());
  224. m_additional->set_menu(m_additional_menu);
  225. for (size_t i = m_last_visible_index; i < m_bookmarks.size(); ++i) {
  226. auto& bookmark = m_bookmarks.at(i);
  227. bookmark->set_visible(false);
  228. m_additional_menu->add_action(GUI::Action::create(bookmark->text().to_deprecated_string(), g_icon_bag.filetype_html, [&](auto&) { bookmark->on_click(0); }));
  229. }
  230. }
  231. }
  232. bool BookmarksBarWidget::contains_bookmark(StringView url)
  233. {
  234. for (int item_index = 0; item_index < model()->row_count(); ++item_index) {
  235. auto item_url = model()->index(item_index, 1).data().to_deprecated_string();
  236. if (item_url == url) {
  237. return true;
  238. }
  239. }
  240. return false;
  241. }
  242. ErrorOr<void> BookmarksBarWidget::remove_bookmark(StringView url)
  243. {
  244. for (int item_index = 0; item_index < model()->row_count(); ++item_index) {
  245. auto item_url = model()->index(item_index, 1).data().to_deprecated_string();
  246. if (item_url == url) {
  247. auto& json_model = *static_cast<GUI::JsonArrayModel*>(model());
  248. TRY(json_model.remove(item_index));
  249. TRY(json_model.store());
  250. if (on_bookmark_change)
  251. on_bookmark_change();
  252. return {};
  253. }
  254. }
  255. return Error::from_string_view("Bookmark not found"sv);
  256. }
  257. ErrorOr<void> BookmarksBarWidget::add_bookmark(StringView url, StringView title)
  258. {
  259. Vector<JsonValue> values;
  260. TRY(values.try_append(title));
  261. TRY(values.try_append(url));
  262. TRY(update_model(values, [](auto& model, auto&& values) -> ErrorOr<void> {
  263. return TRY(model.add(move(values)));
  264. }));
  265. if (on_bookmark_change)
  266. on_bookmark_change();
  267. values = BookmarkEditor::edit_bookmark(window(), title, url, PerformEditOn::NewBookmark);
  268. if (values.is_empty())
  269. return remove_bookmark(url);
  270. auto model_has_updated = false;
  271. for (int item_index = 0; item_index < model()->row_count(); item_index++) {
  272. auto item_url = model()->index(item_index, 1).data().to_deprecated_string();
  273. if (item_url == url) {
  274. TRY(update_model(values, [item_index](auto& model, auto&& values) {
  275. return model.set(item_index, move(values));
  276. }));
  277. model_has_updated = true;
  278. break;
  279. }
  280. }
  281. if (!model_has_updated)
  282. return Error::from_string_view("Bookmark not found"sv);
  283. if (on_bookmark_change)
  284. on_bookmark_change();
  285. return {};
  286. }
  287. ErrorOr<void> BookmarksBarWidget::edit_bookmark(StringView url)
  288. {
  289. for (int item_index = 0; item_index < model()->row_count(); ++item_index) {
  290. auto item_title = model()->index(item_index, 0).data().to_deprecated_string();
  291. auto item_url = model()->index(item_index, 1).data().to_deprecated_string();
  292. if (item_url == url) {
  293. auto values = BookmarkEditor::edit_bookmark(window(), item_title, item_url, PerformEditOn::ExistingBookmark);
  294. if (values.is_empty())
  295. return {};
  296. TRY(update_model(values, [item_index](auto& model, auto&& values) {
  297. return model.set(item_index, move(values));
  298. }));
  299. if (on_bookmark_change)
  300. on_bookmark_change();
  301. return {};
  302. }
  303. }
  304. return Error::from_string_view("Bookmark not found"sv);
  305. }
  306. ErrorOr<void> BookmarksBarWidget::update_model(Vector<JsonValue>& values, Function<ErrorOr<void>(GUI::JsonArrayModel& model, Vector<JsonValue>&& values)> perform_model_change)
  307. {
  308. if (values.is_empty())
  309. return Error::from_string_view("No values to update model with"sv);
  310. auto& json_model = *static_cast<GUI::JsonArrayModel*>(model());
  311. TRY(perform_model_change(json_model, move(values)));
  312. TRY(json_model.store());
  313. return {};
  314. }
  315. }