PDFViewer.cpp 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  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. bool scrolled_down = event.wheel_delta() > 0;
  59. if (event.ctrl()) {
  60. if (scrolled_down) {
  61. zoom_out();
  62. } else {
  63. zoom_in();
  64. }
  65. } else {
  66. auto& scrollbar = event.shift() ? horizontal_scrollbar() : vertical_scrollbar();
  67. if (scrolled_down) {
  68. if (scrollbar.value() == scrollbar.max()) {
  69. if (m_current_page_index < m_document->get_page_count() - 1) {
  70. m_current_page_index++;
  71. scrollbar.set_value(0);
  72. }
  73. } else {
  74. scrollbar.set_value(scrollbar.value() + 20);
  75. }
  76. } else {
  77. if (scrollbar.value() == 0) {
  78. if (m_current_page_index > 0) {
  79. m_current_page_index--;
  80. scrollbar.set_value(scrollbar.max());
  81. }
  82. } else {
  83. scrollbar.set_value(scrollbar.value() - 20);
  84. }
  85. }
  86. }
  87. update();
  88. }
  89. void PDFViewer::timer_event(Core::TimerEvent&)
  90. {
  91. // Clear the bitmap vector of all pages except the current page
  92. for (size_t i = 0; i < m_rendered_page_list.size(); i++) {
  93. if (i != m_current_page_index)
  94. m_rendered_page_list[i].clear();
  95. }
  96. }
  97. void PDFViewer::zoom_in()
  98. {
  99. if (m_zoom_level < number_of_zoom_levels - 1)
  100. m_zoom_level++;
  101. }
  102. void PDFViewer::zoom_out()
  103. {
  104. if (m_zoom_level > 0)
  105. m_zoom_level--;
  106. }
  107. RefPtr<Gfx::Bitmap> PDFViewer::render_page(const PDF::Page& page)
  108. {
  109. auto zoom_scale_factor = static_cast<float>(zoom_levels[m_zoom_level]) / 100.0f;
  110. auto page_width = page.media_box.upper_right_x - page.media_box.lower_left_x;
  111. auto page_height = page.media_box.upper_right_y - page.media_box.lower_left_y;
  112. auto page_scale_factor = page_height / page_width;
  113. auto height = static_cast<float>(this->height() - 2 * frame_thickness() - PAGE_PADDING * 2) * zoom_scale_factor;
  114. auto width = height / page_scale_factor;
  115. auto bitmap = Gfx::Bitmap::create(Gfx::BitmapFormat::BGRA8888, { width, height });
  116. PDF::Renderer::render(*m_document, page, bitmap);
  117. if (page.rotate != 0) {
  118. int rotation_count = (page.rotate / 90) % 4;
  119. if (rotation_count == 3) {
  120. bitmap = bitmap->rotated(Gfx::RotationDirection::CounterClockwise);
  121. } else {
  122. for (int i = 0; i < rotation_count; i++)
  123. bitmap = bitmap->rotated(Gfx::RotationDirection::Clockwise);
  124. }
  125. }
  126. return bitmap;
  127. }