2022-07-05 23:18:21 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2022, Andreas Kling <kling@serenityos.org>
|
|
|
|
* Copyright (c) 2022, Matthew Costa <ucosty@gmail.com>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
2022-07-03 21:26:51 +02:00
|
|
|
#include "BrowserWindow.h"
|
|
|
|
#include "WebView.h"
|
2022-07-05 18:42:45 +01:00
|
|
|
#include <LibCore/EventLoop.h>
|
2022-07-05 21:10:50 +02:00
|
|
|
#include <QAction>
|
2022-07-09 17:46:11 +02:00
|
|
|
#include <QPlainTextEdit>
|
2022-07-03 21:26:51 +02:00
|
|
|
#include <QStatusBar>
|
|
|
|
|
2022-07-08 14:14:40 +02:00
|
|
|
extern String s_serenity_resource_root;
|
|
|
|
|
2022-07-05 18:42:45 +01:00
|
|
|
BrowserWindow::BrowserWindow(Core::EventLoop& event_loop)
|
|
|
|
: m_event_loop(event_loop)
|
2022-07-03 21:26:51 +02:00
|
|
|
{
|
2022-07-05 23:18:21 +01:00
|
|
|
m_tabs_container = new QTabWidget;
|
|
|
|
m_tabs_container->setElideMode(Qt::TextElideMode::ElideRight);
|
|
|
|
m_tabs_container->setMovable(true);
|
|
|
|
m_tabs_container->setTabsClosable(true);
|
2022-07-05 21:10:50 +02:00
|
|
|
|
2022-07-12 11:41:11 +02:00
|
|
|
m_tabs_bar = m_tabs_container->findChild<QTabBar*>();
|
|
|
|
m_tabs_bar->hide();
|
|
|
|
|
2022-07-08 14:14:40 +02:00
|
|
|
auto* menu = menuBar()->addMenu("&File");
|
2022-07-07 02:55:21 +02:00
|
|
|
|
2022-07-08 14:14:40 +02:00
|
|
|
auto* new_tab_action = new QAction("New &Tab");
|
2022-07-07 02:55:21 +02:00
|
|
|
new_tab_action->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_T));
|
|
|
|
menu->addAction(new_tab_action);
|
|
|
|
|
2022-07-08 14:14:40 +02:00
|
|
|
auto* quit_action = new QAction("&Quit");
|
2022-07-07 02:55:21 +02:00
|
|
|
quit_action->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_Q));
|
|
|
|
menu->addAction(quit_action);
|
2022-07-05 21:10:50 +02:00
|
|
|
|
2022-07-09 17:46:11 +02:00
|
|
|
auto* inspect_menu = menuBar()->addMenu("&Inspect");
|
|
|
|
|
|
|
|
auto* view_source_action = new QAction("View &Source");
|
|
|
|
view_source_action->setIcon(QIcon(QString("%1/res/icons/16x16/filetype-html.png").arg(s_serenity_resource_root.characters())));
|
|
|
|
view_source_action->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_U));
|
|
|
|
inspect_menu->addAction(view_source_action);
|
|
|
|
QObject::connect(view_source_action, &QAction::triggered, this, [this] {
|
|
|
|
if (m_current_tab) {
|
|
|
|
auto source = m_current_tab->view().source();
|
|
|
|
|
|
|
|
auto* text_edit = new QPlainTextEdit;
|
|
|
|
text_edit->setFont(QFontDatabase::systemFont(QFontDatabase::SystemFont::FixedFont));
|
|
|
|
text_edit->resize(800, 600);
|
|
|
|
text_edit->setPlainText(source.characters());
|
|
|
|
text_edit->show();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2022-07-08 14:14:40 +02:00
|
|
|
auto* debug_menu = menuBar()->addMenu("&Debug");
|
|
|
|
|
|
|
|
auto* dump_dom_tree_action = new QAction("Dump DOM Tree");
|
|
|
|
dump_dom_tree_action->setIcon(QIcon(QString("%1/res/icons/browser/dom-tree.png").arg(s_serenity_resource_root.characters())));
|
|
|
|
debug_menu->addAction(dump_dom_tree_action);
|
|
|
|
QObject::connect(dump_dom_tree_action, &QAction::triggered, this, [this] {
|
|
|
|
debug_request("dump-dom-tree");
|
|
|
|
});
|
|
|
|
|
|
|
|
auto* dump_layout_tree_action = new QAction("Dump Layout Tree");
|
|
|
|
dump_layout_tree_action->setIcon(QIcon(QString("%1/res/icons/16x16/layout.png").arg(s_serenity_resource_root.characters())));
|
|
|
|
debug_menu->addAction(dump_layout_tree_action);
|
|
|
|
QObject::connect(dump_layout_tree_action, &QAction::triggered, this, [this] {
|
|
|
|
debug_request("dump-layout-tree");
|
|
|
|
});
|
|
|
|
|
|
|
|
auto* dump_stacking_context_tree_action = new QAction("Dump Stacking Context Tree");
|
|
|
|
dump_stacking_context_tree_action->setIcon(QIcon(QString("%1/res/icons/16x16/layers.png").arg(s_serenity_resource_root.characters())));
|
|
|
|
debug_menu->addAction(dump_stacking_context_tree_action);
|
|
|
|
QObject::connect(dump_stacking_context_tree_action, &QAction::triggered, this, [this] {
|
|
|
|
debug_request("dump-stacking-context-tree");
|
|
|
|
});
|
|
|
|
|
|
|
|
auto* dump_style_sheets_action = new QAction("Dump Style Sheets");
|
|
|
|
dump_style_sheets_action->setIcon(QIcon(QString("%1/res/icons/16x16/filetype-css.png").arg(s_serenity_resource_root.characters())));
|
|
|
|
debug_menu->addAction(dump_style_sheets_action);
|
|
|
|
QObject::connect(dump_style_sheets_action, &QAction::triggered, this, [this] {
|
|
|
|
debug_request("dump-style-sheets");
|
|
|
|
});
|
|
|
|
|
|
|
|
auto* dump_history_action = new QAction("Dump History");
|
|
|
|
dump_history_action->setIcon(QIcon(QString("%1/res/icons/16x16/history.png").arg(s_serenity_resource_root.characters())));
|
|
|
|
debug_menu->addAction(dump_history_action);
|
|
|
|
QObject::connect(dump_history_action, &QAction::triggered, this, [this] {
|
|
|
|
debug_request("dump-history");
|
|
|
|
});
|
|
|
|
|
|
|
|
auto* dump_cookies_action = new QAction("Dump Cookies");
|
|
|
|
dump_cookies_action->setIcon(QIcon(QString("%1/res/icons/browser/cookie.png").arg(s_serenity_resource_root.characters())));
|
|
|
|
debug_menu->addAction(dump_cookies_action);
|
|
|
|
QObject::connect(dump_cookies_action, &QAction::triggered, this, [this] {
|
|
|
|
debug_request("dump-cookies");
|
|
|
|
});
|
|
|
|
|
|
|
|
auto* dump_local_storage_action = new QAction("Dump Local Storage");
|
|
|
|
dump_local_storage_action->setIcon(QIcon(QString("%1/res/icons/browser/local-storage.png").arg(s_serenity_resource_root.characters())));
|
|
|
|
debug_menu->addAction(dump_local_storage_action);
|
|
|
|
QObject::connect(dump_local_storage_action, &QAction::triggered, this, [this] {
|
|
|
|
debug_request("dump-local-storage");
|
|
|
|
});
|
|
|
|
|
|
|
|
debug_menu->addSeparator();
|
|
|
|
|
|
|
|
auto* show_line_box_borders_action = new QAction("Show Line Box Borders");
|
|
|
|
show_line_box_borders_action->setCheckable(true);
|
|
|
|
debug_menu->addAction(show_line_box_borders_action);
|
|
|
|
QObject::connect(show_line_box_borders_action, &QAction::triggered, this, [this, show_line_box_borders_action] {
|
|
|
|
bool state = show_line_box_borders_action->isChecked();
|
|
|
|
debug_request("set-line-box-borders", state ? "on" : "off");
|
|
|
|
});
|
|
|
|
|
|
|
|
debug_menu->addSeparator();
|
|
|
|
|
|
|
|
auto* collect_garbage_action = new QAction("Collect Garbage");
|
|
|
|
collect_garbage_action->setIcon(QIcon(QString("%1/res/icons/16x16/trash-can.png").arg(s_serenity_resource_root.characters())));
|
|
|
|
debug_menu->addAction(collect_garbage_action);
|
|
|
|
QObject::connect(collect_garbage_action, &QAction::triggered, this, [this] {
|
|
|
|
debug_request("collect-garbage");
|
|
|
|
});
|
|
|
|
|
|
|
|
auto* clear_cache_action = new QAction("Clear Cache");
|
|
|
|
clear_cache_action->setIcon(QIcon(QString("%1/res/icons/browser/clear-cache.png").arg(s_serenity_resource_root.characters())));
|
|
|
|
debug_menu->addAction(clear_cache_action);
|
|
|
|
QObject::connect(clear_cache_action, &QAction::triggered, this, [this] {
|
|
|
|
debug_request("clear-cache");
|
|
|
|
});
|
|
|
|
|
|
|
|
debug_menu->addSeparator();
|
|
|
|
|
|
|
|
auto* enable_scripting_action = new QAction("Enable Scripting");
|
|
|
|
enable_scripting_action->setCheckable(true);
|
|
|
|
enable_scripting_action->setChecked(true);
|
|
|
|
debug_menu->addAction(enable_scripting_action);
|
|
|
|
QObject::connect(enable_scripting_action, &QAction::triggered, this, [this, enable_scripting_action] {
|
|
|
|
bool state = enable_scripting_action->isChecked();
|
|
|
|
debug_request("scripting", state ? "on" : "off");
|
|
|
|
});
|
|
|
|
|
|
|
|
auto* enable_same_origin_policy_action = new QAction("Enable Same-Origin Policy");
|
|
|
|
enable_same_origin_policy_action->setCheckable(true);
|
|
|
|
debug_menu->addAction(enable_same_origin_policy_action);
|
|
|
|
QObject::connect(enable_same_origin_policy_action, &QAction::triggered, this, [this, enable_same_origin_policy_action] {
|
|
|
|
bool state = enable_same_origin_policy_action->isChecked();
|
|
|
|
debug_request("same-origin-policy", state ? "on" : "off");
|
|
|
|
});
|
|
|
|
|
2022-07-05 23:18:21 +01:00
|
|
|
QObject::connect(new_tab_action, &QAction::triggered, this, &BrowserWindow::new_tab);
|
|
|
|
QObject::connect(quit_action, &QAction::triggered, this, &QMainWindow::close);
|
|
|
|
QObject::connect(m_tabs_container, &QTabWidget::currentChanged, [this](int index) {
|
|
|
|
setWindowTitle(m_tabs_container->tabText(index));
|
|
|
|
setWindowIcon(m_tabs_container->tabIcon(index));
|
|
|
|
});
|
2022-07-06 14:36:49 +01:00
|
|
|
QObject::connect(m_tabs_container, &QTabWidget::tabCloseRequested, this, &BrowserWindow::close_tab);
|
2022-07-03 21:26:51 +02:00
|
|
|
|
2022-07-05 23:18:21 +01:00
|
|
|
new_tab();
|
|
|
|
|
|
|
|
setCentralWidget(m_tabs_container);
|
|
|
|
}
|
2022-07-03 21:26:51 +02:00
|
|
|
|
2022-07-08 14:14:40 +02:00
|
|
|
void BrowserWindow::debug_request(String const& request, String const& argument)
|
|
|
|
{
|
|
|
|
if (!m_current_tab)
|
|
|
|
return;
|
|
|
|
m_current_tab->debug_request(request, argument);
|
|
|
|
}
|
|
|
|
|
2022-07-05 23:18:21 +01:00
|
|
|
void BrowserWindow::new_tab()
|
|
|
|
{
|
|
|
|
auto tab = make<Tab>(this);
|
|
|
|
auto tab_ptr = tab.ptr();
|
|
|
|
m_tabs.append(std::move(tab));
|
2022-07-03 21:26:51 +02:00
|
|
|
|
2022-07-05 23:18:21 +01:00
|
|
|
if (m_current_tab == nullptr) {
|
|
|
|
m_current_tab = tab_ptr;
|
|
|
|
}
|
2022-07-03 21:26:51 +02:00
|
|
|
|
2022-07-05 23:18:21 +01:00
|
|
|
m_tabs_container->addTab(tab_ptr, "New Tab");
|
2022-07-06 14:36:49 +01:00
|
|
|
m_tabs_container->setCurrentWidget(tab_ptr);
|
2022-07-05 21:10:50 +02:00
|
|
|
|
2022-07-05 23:18:21 +01:00
|
|
|
QObject::connect(tab_ptr, &Tab::title_changed, this, &BrowserWindow::tab_title_changed);
|
|
|
|
QObject::connect(tab_ptr, &Tab::favicon_changed, this, &BrowserWindow::tab_favicon_changed);
|
2022-07-12 11:41:11 +02:00
|
|
|
|
|
|
|
if (m_tabs_container->count() > 1)
|
|
|
|
m_tabs_bar->show();
|
2022-07-03 21:26:51 +02:00
|
|
|
}
|
|
|
|
|
2022-07-06 14:36:49 +01:00
|
|
|
void BrowserWindow::close_tab(int index)
|
|
|
|
{
|
|
|
|
auto* tab = m_tabs_container->widget(index);
|
|
|
|
m_tabs_container->removeTab(index);
|
|
|
|
m_tabs.remove_first_matching([&](auto& entry) {
|
|
|
|
return entry == tab;
|
|
|
|
});
|
2022-07-12 11:41:11 +02:00
|
|
|
|
|
|
|
if (m_tabs_container->count() <= 1)
|
|
|
|
m_tabs_bar->hide();
|
2022-07-06 14:36:49 +01:00
|
|
|
}
|
|
|
|
|
2022-07-05 23:18:21 +01:00
|
|
|
int BrowserWindow::tab_index(Tab* tab)
|
2022-07-03 21:26:51 +02:00
|
|
|
{
|
2022-07-05 23:18:21 +01:00
|
|
|
return m_tabs_container->indexOf(tab);
|
2022-07-03 21:26:51 +02:00
|
|
|
}
|
2022-07-04 02:11:58 +02:00
|
|
|
|
2022-07-05 23:18:21 +01:00
|
|
|
void BrowserWindow::tab_title_changed(int index, QString const& title)
|
2022-07-04 02:11:58 +02:00
|
|
|
{
|
2022-07-05 23:18:21 +01:00
|
|
|
if (title.isEmpty()) {
|
|
|
|
m_tabs_container->setTabText(index, "...");
|
2022-07-04 02:11:58 +02:00
|
|
|
setWindowTitle("Ladybird");
|
2022-07-05 23:18:21 +01:00
|
|
|
} else {
|
|
|
|
m_tabs_container->setTabText(index, title);
|
2022-07-04 02:11:58 +02:00
|
|
|
setWindowTitle(QString("%1 - Ladybird").arg(title));
|
2022-07-05 23:18:21 +01:00
|
|
|
}
|
2022-07-04 02:11:58 +02:00
|
|
|
}
|
2022-07-05 05:53:11 +01:00
|
|
|
|
2022-07-05 23:18:21 +01:00
|
|
|
void BrowserWindow::tab_favicon_changed(int index, QIcon icon)
|
2022-07-05 05:53:11 +01:00
|
|
|
{
|
2022-07-05 23:18:21 +01:00
|
|
|
m_tabs_container->setTabIcon(index, icon);
|
2022-07-05 05:53:11 +01:00
|
|
|
setWindowIcon(icon);
|
|
|
|
}
|
2022-07-05 18:42:45 +01:00
|
|
|
|
|
|
|
void BrowserWindow::closeEvent(QCloseEvent* event)
|
|
|
|
{
|
|
|
|
QWidget::closeEvent(event);
|
|
|
|
|
|
|
|
// FIXME: Ladybird only supports one window at the moment. When we support
|
|
|
|
// multiple windows, we'll only want to fire off the quit event when
|
|
|
|
// all of the browser windows have closed.
|
|
|
|
m_event_loop.quit(0);
|
|
|
|
}
|