mirror of
https://github.com/RGBCube/serenity
synced 2025-05-23 15:35:08 +00:00
PDFViewer: Add a toolbar
The toolbar has an option to toggle the sidebar, a number input to set the current page, and two buttons to go up and down by one page
This commit is contained in:
parent
cea7dbce42
commit
cf3eb27108
6 changed files with 186 additions and 10 deletions
|
@ -1,4 +1,5 @@
|
||||||
set(SOURCES
|
set(SOURCES
|
||||||
|
NumericInput.cpp
|
||||||
OutlineModel.cpp
|
OutlineModel.cpp
|
||||||
PDFViewer.cpp
|
PDFViewer.cpp
|
||||||
PDFViewerWidget.cpp
|
PDFViewerWidget.cpp
|
||||||
|
|
90
Userland/Applications/PDFViewer/NumericInput.cpp
Normal file
90
Userland/Applications/PDFViewer/NumericInput.cpp
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2021, Matthew Olsson <mattco@serenityos.org>
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "NumericInput.h"
|
||||||
|
#include "ctype.h"
|
||||||
|
|
||||||
|
NumericInput::NumericInput()
|
||||||
|
{
|
||||||
|
set_text("0");
|
||||||
|
|
||||||
|
on_change = [&] {
|
||||||
|
auto number_opt = text().to_int();
|
||||||
|
if (number_opt.has_value()) {
|
||||||
|
set_current_number(number_opt.value(), false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
StringBuilder builder;
|
||||||
|
bool first = true;
|
||||||
|
for (auto& ch : text()) {
|
||||||
|
if (isdigit(ch) || (first && ((ch == '-' && m_min_number < 0) || ch == '+')))
|
||||||
|
builder.append(ch);
|
||||||
|
first = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto new_number_opt = builder.to_string().to_int();
|
||||||
|
if (!new_number_opt.has_value()) {
|
||||||
|
m_needs_text_reset = true;
|
||||||
|
set_text(builder.to_string());
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
m_needs_text_reset = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
set_text(builder.to_string());
|
||||||
|
set_current_number(new_number_opt.value(), false);
|
||||||
|
};
|
||||||
|
|
||||||
|
on_up_pressed = [&] {
|
||||||
|
if (m_current_number < m_max_number)
|
||||||
|
set_current_number(m_current_number + 1);
|
||||||
|
};
|
||||||
|
|
||||||
|
on_down_pressed = [&] {
|
||||||
|
if (m_current_number > m_min_number)
|
||||||
|
set_current_number(m_current_number - 1);
|
||||||
|
};
|
||||||
|
|
||||||
|
on_focusout = [&] { on_focus_lost(); };
|
||||||
|
on_return_pressed = [&] { on_focus_lost(); };
|
||||||
|
on_escape_pressed = [&] { on_focus_lost(); };
|
||||||
|
}
|
||||||
|
|
||||||
|
void NumericInput::set_min_number(i32 number)
|
||||||
|
{
|
||||||
|
m_min_number = number;
|
||||||
|
if (m_current_number < number)
|
||||||
|
set_current_number(number);
|
||||||
|
}
|
||||||
|
|
||||||
|
void NumericInput::set_max_number(i32 number)
|
||||||
|
{
|
||||||
|
m_max_number = number;
|
||||||
|
if (m_current_number > number)
|
||||||
|
set_current_number(number);
|
||||||
|
}
|
||||||
|
|
||||||
|
void NumericInput::on_focus_lost()
|
||||||
|
{
|
||||||
|
if (m_needs_text_reset) {
|
||||||
|
set_text(String::number(m_current_number));
|
||||||
|
m_needs_text_reset = false;
|
||||||
|
}
|
||||||
|
if (on_number_changed)
|
||||||
|
on_number_changed(m_current_number);
|
||||||
|
}
|
||||||
|
|
||||||
|
void NumericInput::set_current_number(i32 number, bool call_change_handler)
|
||||||
|
{
|
||||||
|
if (number == m_current_number)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_current_number = clamp(number, m_min_number, m_max_number);
|
||||||
|
set_text(String::number(m_current_number));
|
||||||
|
if (on_number_changed && call_change_handler)
|
||||||
|
on_number_changed(m_current_number);
|
||||||
|
}
|
31
Userland/Applications/PDFViewer/NumericInput.h
Normal file
31
Userland/Applications/PDFViewer/NumericInput.h
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2021, Matthew Olsson <mattco@serenityos.org>
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <AK/NumericLimits.h>
|
||||||
|
#include <LibGUI/TextBox.h>
|
||||||
|
|
||||||
|
class NumericInput final : public GUI::TextBox {
|
||||||
|
C_OBJECT(NumericInput)
|
||||||
|
public:
|
||||||
|
NumericInput();
|
||||||
|
virtual ~NumericInput() override = default;
|
||||||
|
|
||||||
|
Function<void(i32)> on_number_changed;
|
||||||
|
|
||||||
|
void set_min_number(i32 number);
|
||||||
|
void set_max_number(i32 number);
|
||||||
|
void set_current_number(i32 number, bool call_change_handler = true);
|
||||||
|
|
||||||
|
private:
|
||||||
|
void on_focus_lost();
|
||||||
|
|
||||||
|
bool m_needs_text_reset { false };
|
||||||
|
i32 m_current_number { 0 };
|
||||||
|
i32 m_min_number { NumericLimits<i32>::min() };
|
||||||
|
i32 m_max_number { NumericLimits<i32>::max() };
|
||||||
|
};
|
|
@ -41,6 +41,10 @@ class PDFViewer : public GUI::AbstractScrollableWidget {
|
||||||
public:
|
public:
|
||||||
virtual ~PDFViewer() override = default;
|
virtual ~PDFViewer() override = default;
|
||||||
|
|
||||||
|
ALWAYS_INLINE u32 current_page() const { return m_current_page_index; }
|
||||||
|
ALWAYS_INLINE void set_current_page(u32 current_page) { m_current_page_index = current_page; }
|
||||||
|
|
||||||
|
ALWAYS_INLINE const RefPtr<PDF::Document>& document() const { return m_document; }
|
||||||
void set_document(RefPtr<PDF::Document>);
|
void set_document(RefPtr<PDF::Document>);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
|
@ -9,15 +9,20 @@
|
||||||
#include <LibGUI/Application.h>
|
#include <LibGUI/Application.h>
|
||||||
#include <LibGUI/BoxLayout.h>
|
#include <LibGUI/BoxLayout.h>
|
||||||
#include <LibGUI/FilePicker.h>
|
#include <LibGUI/FilePicker.h>
|
||||||
|
#include <LibGUI/Label.h>
|
||||||
#include <LibGUI/Menu.h>
|
#include <LibGUI/Menu.h>
|
||||||
#include <LibGUI/Menubar.h>
|
#include <LibGUI/Menubar.h>
|
||||||
#include <LibGUI/Splitter.h>
|
#include <LibGUI/Splitter.h>
|
||||||
|
#include <LibGUI/Toolbar.h>
|
||||||
|
#include <LibGUI/ToolbarContainer.h>
|
||||||
|
|
||||||
PDFViewerWidget::PDFViewerWidget()
|
PDFViewerWidget::PDFViewerWidget()
|
||||||
{
|
{
|
||||||
set_fill_with_background_color(true);
|
set_fill_with_background_color(true);
|
||||||
set_layout<GUI::VerticalBoxLayout>();
|
set_layout<GUI::VerticalBoxLayout>();
|
||||||
|
|
||||||
|
create_toolbar();
|
||||||
|
|
||||||
auto& splitter = add<GUI::HorizontalSplitter>();
|
auto& splitter = add<GUI::HorizontalSplitter>();
|
||||||
|
|
||||||
m_sidebar = splitter.add<SidebarWidget>();
|
m_sidebar = splitter.add<SidebarWidget>();
|
||||||
|
@ -37,21 +42,53 @@ void PDFViewerWidget::initialize_menubar(GUI::Menubar& menubar)
|
||||||
file_menu.add_action(GUI::CommonActions::make_quit_action([](auto&) {
|
file_menu.add_action(GUI::CommonActions::make_quit_action([](auto&) {
|
||||||
GUI::Application::the()->quit();
|
GUI::Application::the()->quit();
|
||||||
}));
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
auto& view_menu = menubar.add_menu("&View");
|
void PDFViewerWidget::create_toolbar()
|
||||||
|
{
|
||||||
|
auto& toolbar_container = add<GUI::ToolbarContainer>();
|
||||||
|
auto& toolbar = toolbar_container.add<GUI::Toolbar>();
|
||||||
|
|
||||||
auto open_sidebar_action = GUI::Action::create(
|
auto open_outline_action = GUI::Action::create(
|
||||||
"Open &Sidebar", { Mod_Ctrl, Key_O }, Gfx::Bitmap::load_from_file("/res/icons/16x16/sidebar.png"), [&](auto& action) {
|
"Open &Sidebar", { Mod_Ctrl, Key_O }, Gfx::Bitmap::load_from_file("/res/icons/16x16/sidebar.png"), [&](auto& action) {
|
||||||
m_sidebar_open = !m_sidebar_open;
|
m_sidebar_open = !m_sidebar_open;
|
||||||
m_sidebar->set_fixed_width(m_sidebar_open ? 0 : 200);
|
m_sidebar->set_fixed_width(m_sidebar_open ? 0 : 200);
|
||||||
action.set_text(m_sidebar_open ? "Open &Sidebar" : "Close &Sidebar");
|
action.set_text(m_sidebar_open ? "Open &Sidebar" : "Close &Sidebar");
|
||||||
},
|
},
|
||||||
nullptr);
|
nullptr);
|
||||||
open_sidebar_action->set_enabled(false);
|
open_outline_action->set_enabled(false);
|
||||||
|
m_toggle_sidebar_action = open_outline_action;
|
||||||
|
|
||||||
view_menu.add_action(open_sidebar_action);
|
toolbar.add_action(*open_outline_action);
|
||||||
|
toolbar.add_separator();
|
||||||
|
|
||||||
m_open_outline_action = open_sidebar_action;
|
m_go_to_prev_page_action = GUI::Action::create("Go to &Previous Page", Gfx::Bitmap::load_from_file("/res/icons/16x16/go-up.png"), [&](auto&) {
|
||||||
|
if (m_viewer->current_page() > 0)
|
||||||
|
m_page_text_box->set_current_number(m_viewer->current_page());
|
||||||
|
});
|
||||||
|
m_go_to_prev_page_action->set_enabled(false);
|
||||||
|
|
||||||
|
m_go_to_next_page_action = GUI::Action::create("Go to &Next Page", Gfx::Bitmap::load_from_file("/res/icons/16x16/go-down.png"), [&](auto&) {
|
||||||
|
if (m_viewer->current_page() < m_viewer->document()->get_page_count() - 1)
|
||||||
|
m_page_text_box->set_current_number(m_viewer->current_page() + 2);
|
||||||
|
});
|
||||||
|
m_go_to_next_page_action->set_enabled(false);
|
||||||
|
|
||||||
|
toolbar.add_action(*m_go_to_prev_page_action);
|
||||||
|
toolbar.add_action(*m_go_to_next_page_action);
|
||||||
|
|
||||||
|
m_page_text_box = toolbar.add<NumericInput>();
|
||||||
|
m_page_text_box->set_enabled(false);
|
||||||
|
m_page_text_box->set_fixed_width(30);
|
||||||
|
m_page_text_box->set_min_number(1);
|
||||||
|
|
||||||
|
m_page_text_box->on_number_changed = [&](i32 number) {
|
||||||
|
VERIFY(number >= 1 && static_cast<u32>(number) <= m_viewer->document()->get_page_count());
|
||||||
|
m_viewer->set_current_page(static_cast<u32>(number) - 1);
|
||||||
|
m_viewer->update();
|
||||||
|
};
|
||||||
|
|
||||||
|
m_total_page_label = toolbar.add<GUI::Label>();
|
||||||
}
|
}
|
||||||
|
|
||||||
void PDFViewerWidget::open_file(const String& path)
|
void PDFViewerWidget::open_file(const String& path)
|
||||||
|
@ -62,17 +99,24 @@ void PDFViewerWidget::open_file(const String& path)
|
||||||
m_buffer = file_result.value()->read_all();
|
m_buffer = file_result.value()->read_all();
|
||||||
auto document = adopt_ref(*new PDF::Document(m_buffer));
|
auto document = adopt_ref(*new PDF::Document(m_buffer));
|
||||||
m_viewer->set_document(document);
|
m_viewer->set_document(document);
|
||||||
|
m_total_page_label->set_text(String::formatted("of {}", document->get_page_count()));
|
||||||
|
m_total_page_label->set_fixed_width(30);
|
||||||
|
|
||||||
|
m_page_text_box->set_text(String::number(m_viewer->current_page() + 1));
|
||||||
|
m_page_text_box->set_enabled(true);
|
||||||
|
m_page_text_box->set_max_number(document->get_page_count());
|
||||||
|
m_go_to_prev_page_action->set_enabled(true);
|
||||||
|
m_go_to_next_page_action->set_enabled(true);
|
||||||
|
m_toggle_sidebar_action->set_enabled(true);
|
||||||
|
|
||||||
if (document->outline()) {
|
if (document->outline()) {
|
||||||
auto outline = document->outline();
|
auto outline = document->outline();
|
||||||
m_sidebar->set_outline(outline.release_nonnull());
|
m_sidebar->set_outline(outline.release_nonnull());
|
||||||
m_sidebar->set_fixed_width(200);
|
m_sidebar->set_fixed_width(200);
|
||||||
m_sidebar_open = true;
|
m_sidebar_open = true;
|
||||||
m_open_outline_action->set_enabled(true);
|
|
||||||
} else {
|
} else {
|
||||||
m_sidebar->set_outline({});
|
m_sidebar->set_outline({});
|
||||||
m_sidebar->set_fixed_width(0);
|
m_sidebar->set_fixed_width(0);
|
||||||
m_sidebar_open = false;
|
m_sidebar_open = false;
|
||||||
m_open_outline_action->set_enabled(false);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,9 +6,11 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include "NumericInput.h"
|
||||||
#include "PDFViewer.h"
|
#include "PDFViewer.h"
|
||||||
#include "SidebarWidget.h"
|
#include "SidebarWidget.h"
|
||||||
#include <LibGUI/Action.h>
|
#include <LibGUI/Action.h>
|
||||||
|
#include <LibGUI/TextBox.h>
|
||||||
#include <LibGUI/Widget.h>
|
#include <LibGUI/Widget.h>
|
||||||
|
|
||||||
class PDFViewer;
|
class PDFViewer;
|
||||||
|
@ -19,16 +21,20 @@ class PDFViewerWidget final : public GUI::Widget {
|
||||||
public:
|
public:
|
||||||
~PDFViewerWidget() override = default;
|
~PDFViewerWidget() override = default;
|
||||||
|
|
||||||
void open_file(const String& path);
|
|
||||||
void initialize_menubar(GUI::Menubar&);
|
void initialize_menubar(GUI::Menubar&);
|
||||||
|
void create_toolbar();
|
||||||
|
void open_file(const String& path);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
PDFViewerWidget();
|
PDFViewerWidget();
|
||||||
|
|
||||||
RefPtr<GUI::Action> m_open_outline_action;
|
|
||||||
RefPtr<PDFViewer> m_viewer;
|
RefPtr<PDFViewer> m_viewer;
|
||||||
RefPtr<SidebarWidget> m_sidebar;
|
RefPtr<SidebarWidget> m_sidebar;
|
||||||
|
RefPtr<NumericInput> m_page_text_box;
|
||||||
|
RefPtr<GUI::Label> m_total_page_label;
|
||||||
|
RefPtr<GUI::Action> m_go_to_prev_page_action;
|
||||||
|
RefPtr<GUI::Action> m_go_to_next_page_action;
|
||||||
|
RefPtr<GUI::Action> m_toggle_sidebar_action;
|
||||||
bool m_sidebar_open { false };
|
bool m_sidebar_open { false };
|
||||||
ByteBuffer m_buffer;
|
ByteBuffer m_buffer;
|
||||||
RefPtr<GUI::Action> m_open_action;
|
|
||||||
};
|
};
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue