PDFViewer.cpp 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. /*
  2. * Copyright (c) 2021, Matthew Olsson <mattco@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include "PDFViewer.h"
  7. #include <LibGUI/Action.h>
  8. #include <LibGUI/Painter.h>
  9. #include <LibPDF/Renderer.h>
  10. static constexpr int PAGE_PADDING = 25;
  11. PDFViewer::PDFViewer()
  12. {
  13. set_should_hide_unnecessary_scrollbars(true);
  14. set_focus_policy(GUI::FocusPolicy::StrongFocus);
  15. set_scrollbars_enabled(true);
  16. start_timer(30'000);
  17. }
  18. void PDFViewer::set_document(RefPtr<PDF::Document> document)
  19. {
  20. m_document = document;
  21. m_current_page_index = document->get_first_page_index();
  22. m_zoom_level = initial_zoom_level;
  23. m_rendered_page_list.clear();
  24. m_rendered_page_list.ensure_capacity(document->get_page_count());
  25. for (u32 i = 0; i < document->get_page_count(); i++)
  26. m_rendered_page_list.unchecked_append(HashMap<u32, RefPtr<Gfx::Bitmap>>());
  27. update();
  28. }
  29. RefPtr<Gfx::Bitmap> PDFViewer::get_rendered_page(u32 index)
  30. {
  31. auto& rendered_page_map = m_rendered_page_list[index];
  32. auto existing_rendered_page = rendered_page_map.get(m_zoom_level);
  33. if (existing_rendered_page.has_value())
  34. return existing_rendered_page.value();
  35. auto rendered_page = render_page(m_document->get_page(index));
  36. rendered_page_map.set(m_zoom_level, rendered_page);
  37. return rendered_page;
  38. }
  39. void PDFViewer::paint_event(GUI::PaintEvent& event)
  40. {
  41. GUI::Frame::paint_event(event);
  42. GUI::Painter painter(*this);
  43. painter.add_clip_rect(widget_inner_rect());
  44. painter.add_clip_rect(event.rect());
  45. painter.fill_rect(event.rect(), Color(0x80, 0x80, 0x80));
  46. if (!m_document)
  47. return;
  48. auto page = get_rendered_page(m_current_page_index);
  49. set_content_size(page->size());
  50. painter.translate(frame_thickness(), frame_thickness());
  51. painter.translate(-horizontal_scrollbar().value(), -vertical_scrollbar().value());
  52. int x = max(0, (width() - page->width()) / 2);
  53. int y = max(0, (height() - page->height()) / 2);
  54. painter.blit({ x, y }, *page, page->rect());
  55. }
  56. void PDFViewer::mousewheel_event(GUI::MouseEvent& event)
  57. {
  58. if (!m_document)
  59. return;
  60. bool scrolled_down = event.wheel_delta() > 0;
  61. if (event.ctrl()) {
  62. if (scrolled_down) {
  63. zoom_out();
  64. } else {
  65. zoom_in();
  66. }
  67. } else {
  68. auto& scrollbar = event.shift() ? horizontal_scrollbar() : vertical_scrollbar();
  69. if (scrolled_down) {
  70. if (scrollbar.value() == scrollbar.max()) {
  71. if (m_current_page_index < m_document->get_page_count() - 1) {
  72. m_current_page_index++;
  73. if (on_page_change)
  74. on_page_change(m_current_page_index);
  75. scrollbar.set_value(0);
  76. }
  77. } else {
  78. scrollbar.increase_slider_by(20);
  79. }
  80. } else {
  81. if (scrollbar.value() == 0) {
  82. if (m_current_page_index > 0) {
  83. m_current_page_index--;
  84. if (on_page_change)
  85. on_page_change(m_current_page_index);
  86. scrollbar.set_value(scrollbar.max());
  87. }
  88. } else {
  89. scrollbar.decrease_slider_by(20);
  90. }
  91. }
  92. update();
  93. }
  94. }
  95. void PDFViewer::mousedown_event(GUI::MouseEvent& event)
  96. {
  97. if (event.button() == GUI::MouseButton::Middle) {
  98. m_pan_starting_position = to_content_position(event.position());
  99. set_override_cursor(Gfx::StandardCursor::Drag);
  100. }
  101. }
  102. void PDFViewer::mouseup_event(GUI::MouseEvent&)
  103. {
  104. set_override_cursor(Gfx::StandardCursor::None);
  105. }
  106. void PDFViewer::mousemove_event(GUI::MouseEvent& event)
  107. {
  108. if (event.buttons() & GUI::MouseButton::Middle) {
  109. auto delta = to_content_position(event.position()) - m_pan_starting_position;
  110. horizontal_scrollbar().decrease_slider_by(delta.x());
  111. vertical_scrollbar().decrease_slider_by(delta.y());
  112. update();
  113. }
  114. }
  115. void PDFViewer::timer_event(Core::TimerEvent&)
  116. {
  117. // Clear the bitmap vector of all pages except the current page
  118. for (size_t i = 0; i < m_rendered_page_list.size(); i++) {
  119. if (i != m_current_page_index)
  120. m_rendered_page_list[i].clear();
  121. }
  122. }
  123. void PDFViewer::zoom_in()
  124. {
  125. if (m_zoom_level < number_of_zoom_levels - 1) {
  126. m_zoom_level++;
  127. update();
  128. }
  129. }
  130. void PDFViewer::zoom_out()
  131. {
  132. if (m_zoom_level > 0) {
  133. m_zoom_level--;
  134. update();
  135. }
  136. }
  137. void PDFViewer::reset_zoom()
  138. {
  139. m_zoom_level = initial_zoom_level;
  140. update();
  141. }
  142. RefPtr<Gfx::Bitmap> PDFViewer::render_page(const PDF::Page& page)
  143. {
  144. auto zoom_scale_factor = static_cast<float>(zoom_levels[m_zoom_level]) / 100.0f;
  145. auto page_width = page.media_box.upper_right_x - page.media_box.lower_left_x;
  146. auto page_height = page.media_box.upper_right_y - page.media_box.lower_left_y;
  147. auto page_scale_factor = page_height / page_width;
  148. auto height = static_cast<float>(this->height() - 2 * frame_thickness() - PAGE_PADDING * 2) * zoom_scale_factor;
  149. auto width = height / page_scale_factor;
  150. auto bitmap = Gfx::Bitmap::try_create(Gfx::BitmapFormat::BGRA8888, { width, height }).release_value_but_fixme_should_propagate_errors();
  151. PDF::Renderer::render(*m_document, page, bitmap);
  152. if (page.rotate != 0) {
  153. int rotation_count = (page.rotate / 90) % 4;
  154. if (rotation_count == 3) {
  155. bitmap = bitmap->rotated(Gfx::RotationDirection::CounterClockwise).release_value_but_fixme_should_propagate_errors();
  156. } else {
  157. for (int i = 0; i < rotation_count; i++)
  158. bitmap = bitmap->rotated(Gfx::RotationDirection::Clockwise).release_value_but_fixme_should_propagate_errors();
  159. }
  160. }
  161. return bitmap;
  162. }