2020-01-18 11:38:21 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
2022-03-04 19:16:57 +03:00
|
|
|
* Copyright (c) 2021-2022, Sam Atkins <atkinssj@serenityos.org>
|
2022-02-10 22:28:48 +03:00
|
|
|
* Copyright (c) 2022, the SerenityOS developers.
|
2020-01-18 11:38:21 +03:00
|
|
|
*
|
2021-04-22 11:24:48 +03:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-01-18 11:38:21 +03:00
|
|
|
*/
|
|
|
|
|
2020-01-02 16:55:19 +03:00
|
|
|
#include "InspectorWidget.h"
|
2023-12-06 01:31:00 +03:00
|
|
|
#include <LibGUI/Action.h>
|
2020-02-06 22:33:02 +03:00
|
|
|
#include <LibGUI/BoxLayout.h>
|
2023-12-06 01:31:00 +03:00
|
|
|
#include <LibGUI/Menu.h>
|
2023-12-06 17:34:53 +03:00
|
|
|
#include <LibWebView/Attribute.h>
|
2023-11-24 03:50:16 +03:00
|
|
|
#include <LibWebView/InspectorClient.h>
|
2022-04-30 11:46:33 +03:00
|
|
|
#include <LibWebView/OutOfProcessWebView.h>
|
2020-01-02 16:55:19 +03:00
|
|
|
|
2020-05-08 22:38:30 +03:00
|
|
|
namespace Browser {
|
|
|
|
|
2023-11-24 03:50:16 +03:00
|
|
|
NonnullRefPtr<InspectorWidget> InspectorWidget::create(WebView::OutOfProcessWebView& content_view)
|
2021-08-27 19:40:25 +03:00
|
|
|
{
|
2023-11-24 03:50:16 +03:00
|
|
|
return adopt_ref(*new (nothrow) InspectorWidget(content_view));
|
2021-08-17 19:00:27 +03:00
|
|
|
}
|
|
|
|
|
2023-11-24 03:50:16 +03:00
|
|
|
InspectorWidget::InspectorWidget(WebView::OutOfProcessWebView& content_view)
|
2020-01-02 16:55:19 +03:00
|
|
|
{
|
2023-11-24 03:50:16 +03:00
|
|
|
set_layout<GUI::VerticalBoxLayout>(4);
|
2021-09-13 18:17:43 +03:00
|
|
|
set_fill_with_background_color(true);
|
|
|
|
|
2023-11-28 00:18:32 +03:00
|
|
|
m_inspector_view = add<WebView::OutOfProcessWebView>();
|
2023-11-24 03:50:16 +03:00
|
|
|
m_inspector_client = make<WebView::InspectorClient>(content_view, *m_inspector_view);
|
2020-06-12 23:30:11 +03:00
|
|
|
|
2023-12-06 01:31:00 +03:00
|
|
|
m_edit_node_action = GUI::Action::create("&Edit node"sv, [this](auto&) { m_inspector_client->context_menu_edit_dom_node(); });
|
2023-12-06 19:16:28 +03:00
|
|
|
m_copy_node_action = GUI::Action::create("&Copy HTML"sv, [this](auto&) { m_inspector_client->context_menu_copy_dom_node(); });
|
2023-12-06 19:56:16 +03:00
|
|
|
m_screenshot_node_action = GUI::Action::create("Take node &screenshot"sv, [this](auto&) { m_inspector_client->context_menu_screenshot_dom_node(); });
|
2023-12-07 19:05:34 +03:00
|
|
|
m_create_child_element_action = GUI::Action::create("Create child &element"sv, [this](auto&) { m_inspector_client->context_menu_create_child_element(); });
|
|
|
|
m_create_child_text_node_action = GUI::Action::create("Create child &text node"sv, [this](auto&) { m_inspector_client->context_menu_create_child_text_node(); });
|
2023-12-07 19:13:15 +03:00
|
|
|
m_clone_node_action = GUI::Action::create("C&lone node"sv, [this](auto&) { m_inspector_client->context_menu_clone_dom_node(); });
|
2023-12-06 01:31:00 +03:00
|
|
|
m_delete_node_action = GUI::Action::create("&Delete node"sv, [this](auto&) { m_inspector_client->context_menu_remove_dom_node(); });
|
|
|
|
m_add_attribute_action = GUI::Action::create("&Add attribute"sv, [this](auto&) { m_inspector_client->context_menu_add_dom_node_attribute(); });
|
|
|
|
m_remove_attribute_action = GUI::Action::create("&Remove attribute"sv, [this](auto&) { m_inspector_client->context_menu_remove_dom_node_attribute(); });
|
2023-12-06 18:12:31 +03:00
|
|
|
m_copy_attribute_value_action = GUI::Action::create("Copy attribute &value"sv, [this](auto&) { m_inspector_client->context_menu_copy_dom_node_attribute_value(); });
|
2023-12-06 01:31:00 +03:00
|
|
|
|
2023-12-07 19:05:34 +03:00
|
|
|
auto add_create_child_menu = [&](auto& menu) {
|
|
|
|
auto create_child_menu = menu.add_submenu("Create child"_string);
|
|
|
|
create_child_menu->add_action(*m_create_child_element_action);
|
|
|
|
create_child_menu->add_action(*m_create_child_text_node_action);
|
|
|
|
};
|
|
|
|
|
2023-12-06 01:31:00 +03:00
|
|
|
m_dom_node_text_context_menu = GUI::Menu::construct();
|
|
|
|
m_dom_node_text_context_menu->add_action(*m_edit_node_action);
|
2023-12-06 19:16:28 +03:00
|
|
|
m_dom_node_text_context_menu->add_action(*m_copy_node_action);
|
2023-12-06 01:31:00 +03:00
|
|
|
m_dom_node_text_context_menu->add_separator();
|
|
|
|
m_dom_node_text_context_menu->add_action(*m_delete_node_action);
|
|
|
|
|
|
|
|
m_dom_node_tag_context_menu = GUI::Menu::construct();
|
|
|
|
m_dom_node_tag_context_menu->add_action(*m_edit_node_action);
|
|
|
|
m_dom_node_tag_context_menu->add_separator();
|
|
|
|
m_dom_node_tag_context_menu->add_action(*m_add_attribute_action);
|
2023-12-07 19:05:34 +03:00
|
|
|
add_create_child_menu(*m_dom_node_tag_context_menu);
|
2023-12-07 19:13:15 +03:00
|
|
|
m_dom_node_tag_context_menu->add_action(*m_clone_node_action);
|
2023-12-06 01:31:00 +03:00
|
|
|
m_dom_node_tag_context_menu->add_action(*m_delete_node_action);
|
2023-12-06 19:16:28 +03:00
|
|
|
m_dom_node_tag_context_menu->add_separator();
|
|
|
|
m_dom_node_tag_context_menu->add_action(*m_copy_node_action);
|
2023-12-06 19:56:16 +03:00
|
|
|
m_dom_node_tag_context_menu->add_action(*m_screenshot_node_action);
|
2023-12-06 01:31:00 +03:00
|
|
|
|
|
|
|
m_dom_node_attribute_context_menu = GUI::Menu::construct();
|
|
|
|
m_dom_node_attribute_context_menu->add_action(*m_edit_node_action);
|
2023-12-06 18:12:31 +03:00
|
|
|
m_dom_node_attribute_context_menu->add_action(*m_copy_attribute_value_action);
|
2023-12-06 01:31:00 +03:00
|
|
|
m_dom_node_attribute_context_menu->add_action(*m_remove_attribute_action);
|
|
|
|
m_dom_node_attribute_context_menu->add_separator();
|
|
|
|
m_dom_node_attribute_context_menu->add_action(*m_add_attribute_action);
|
2023-12-07 19:05:34 +03:00
|
|
|
add_create_child_menu(*m_dom_node_attribute_context_menu);
|
2023-12-07 19:13:15 +03:00
|
|
|
m_dom_node_attribute_context_menu->add_action(*m_clone_node_action);
|
2023-12-06 01:31:00 +03:00
|
|
|
m_dom_node_attribute_context_menu->add_action(*m_delete_node_action);
|
2023-12-06 19:16:28 +03:00
|
|
|
m_dom_node_attribute_context_menu->add_separator();
|
|
|
|
m_dom_node_attribute_context_menu->add_action(*m_copy_node_action);
|
2023-12-06 19:56:16 +03:00
|
|
|
m_dom_node_attribute_context_menu->add_action(*m_screenshot_node_action);
|
2023-12-06 01:31:00 +03:00
|
|
|
|
|
|
|
m_inspector_client->on_requested_dom_node_text_context_menu = [this](auto position) {
|
|
|
|
m_edit_node_action->set_text("&Edit text");
|
2023-12-06 19:16:28 +03:00
|
|
|
m_copy_node_action->set_text("&Copy text");
|
2023-12-06 01:31:00 +03:00
|
|
|
|
|
|
|
m_dom_node_text_context_menu->popup(to_widget_position(position));
|
|
|
|
};
|
|
|
|
|
|
|
|
m_inspector_client->on_requested_dom_node_tag_context_menu = [this](auto position, auto const& tag) {
|
2023-12-16 17:19:34 +03:00
|
|
|
m_edit_node_action->set_text(ByteString::formatted("&Edit \"{}\"", tag));
|
2023-12-06 19:16:28 +03:00
|
|
|
m_copy_node_action->set_text("&Copy HTML");
|
2023-12-06 01:31:00 +03:00
|
|
|
|
|
|
|
m_dom_node_tag_context_menu->popup(to_widget_position(position));
|
|
|
|
};
|
|
|
|
|
2023-12-06 17:34:53 +03:00
|
|
|
m_inspector_client->on_requested_dom_node_attribute_context_menu = [this](auto position, auto const&, auto const& attribute) {
|
2023-12-06 18:12:31 +03:00
|
|
|
static constexpr size_t MAX_ATTRIBUTE_VALUE_LENGTH = 32;
|
|
|
|
|
2023-12-06 19:16:28 +03:00
|
|
|
m_copy_node_action->set_text("&Copy HTML");
|
2023-12-16 17:19:34 +03:00
|
|
|
m_edit_node_action->set_text(ByteString::formatted("&Edit attribute \"{}\"", attribute.name));
|
|
|
|
m_remove_attribute_action->set_text(ByteString::formatted("&Remove attribute \"{}\"", attribute.name));
|
|
|
|
m_copy_attribute_value_action->set_text(ByteString::formatted("Copy attribute &value \"{:.{}}{}\"",
|
2023-12-06 18:12:31 +03:00
|
|
|
attribute.value, MAX_ATTRIBUTE_VALUE_LENGTH,
|
|
|
|
attribute.value.bytes_as_string_view().length() > MAX_ATTRIBUTE_VALUE_LENGTH ? "..."sv : ""sv));
|
2023-12-06 01:31:00 +03:00
|
|
|
|
|
|
|
m_dom_node_attribute_context_menu->popup(to_widget_position(position));
|
|
|
|
};
|
|
|
|
|
2023-11-24 03:50:16 +03:00
|
|
|
m_inspector_view->set_focus(true);
|
2020-01-02 16:55:19 +03:00
|
|
|
}
|
|
|
|
|
2023-11-24 03:50:16 +03:00
|
|
|
InspectorWidget::~InspectorWidget() = default;
|
2021-08-27 22:21:30 +03:00
|
|
|
|
2023-11-24 03:50:16 +03:00
|
|
|
void InspectorWidget::inspect()
|
2021-06-08 00:21:16 +03:00
|
|
|
{
|
2023-11-24 03:50:16 +03:00
|
|
|
m_inspector_client->inspect();
|
2021-06-08 00:21:16 +03:00
|
|
|
}
|
|
|
|
|
2023-11-24 03:50:16 +03:00
|
|
|
void InspectorWidget::reset()
|
2021-09-01 14:50:47 +03:00
|
|
|
{
|
2023-11-24 03:50:16 +03:00
|
|
|
m_inspector_client->reset();
|
2021-09-01 14:50:47 +03:00
|
|
|
}
|
|
|
|
|
2023-11-24 03:50:16 +03:00
|
|
|
void InspectorWidget::select_default_node()
|
2021-09-02 14:05:32 +03:00
|
|
|
{
|
2023-11-24 03:50:16 +03:00
|
|
|
m_inspector_client->select_default_node();
|
|
|
|
}
|
2021-09-02 14:05:32 +03:00
|
|
|
|
2023-11-24 03:50:16 +03:00
|
|
|
void InspectorWidget::select_hovered_node()
|
|
|
|
{
|
|
|
|
m_inspector_client->select_hovered_node();
|
2021-08-27 19:40:25 +03:00
|
|
|
}
|
|
|
|
|
2023-12-06 01:31:00 +03:00
|
|
|
Gfx::IntPoint InspectorWidget::to_widget_position(Gfx::IntPoint position) const
|
|
|
|
{
|
|
|
|
return m_inspector_view->screen_relative_rect().location().translated(position);
|
|
|
|
}
|
|
|
|
|
2020-05-08 22:38:30 +03:00
|
|
|
}
|