CanvasRenderingContext2D.h 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. /*
  2. * Copyright (c) 2020-2022, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2021-2022, Linus Groh <linusg@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #pragma once
  8. #include <AK/String.h>
  9. #include <AK/Variant.h>
  10. #include <LibGfx/AffineTransform.h>
  11. #include <LibGfx/AntiAliasingPainter.h>
  12. #include <LibGfx/Color.h>
  13. #include <LibGfx/Forward.h>
  14. #include <LibGfx/Painter.h>
  15. #include <LibGfx/Path.h>
  16. #include <LibGfx/PathClipper.h>
  17. #include <LibWeb/Bindings/PlatformObject.h>
  18. #include <LibWeb/HTML/Canvas/CanvasCompositing.h>
  19. #include <LibWeb/HTML/Canvas/CanvasDrawImage.h>
  20. #include <LibWeb/HTML/Canvas/CanvasDrawPath.h>
  21. #include <LibWeb/HTML/Canvas/CanvasFillStrokeStyles.h>
  22. #include <LibWeb/HTML/Canvas/CanvasImageData.h>
  23. #include <LibWeb/HTML/Canvas/CanvasImageSmoothing.h>
  24. #include <LibWeb/HTML/Canvas/CanvasPath.h>
  25. #include <LibWeb/HTML/Canvas/CanvasPathDrawingStyles.h>
  26. #include <LibWeb/HTML/Canvas/CanvasRect.h>
  27. #include <LibWeb/HTML/Canvas/CanvasState.h>
  28. #include <LibWeb/HTML/Canvas/CanvasText.h>
  29. #include <LibWeb/HTML/Canvas/CanvasTextDrawingStyles.h>
  30. #include <LibWeb/HTML/Canvas/CanvasTransform.h>
  31. #include <LibWeb/HTML/CanvasGradient.h>
  32. #include <LibWeb/Layout/InlineNode.h>
  33. #include <LibWeb/Layout/LineBox.h>
  34. #include <LibWeb/WebIDL/ExceptionOr.h>
  35. namespace Web::HTML {
  36. // https://html.spec.whatwg.org/multipage/canvas.html#canvasimagesource
  37. // NOTE: This is the Variant created by the IDL wrapper generator, and needs to be updated accordingly.
  38. using CanvasImageSource = Variant<JS::Handle<HTMLImageElement>, JS::Handle<HTMLCanvasElement>>;
  39. class CanvasRenderingContext2D
  40. : public Bindings::PlatformObject
  41. , public CanvasPath
  42. , public CanvasState
  43. , public CanvasTransform<CanvasRenderingContext2D>
  44. , public CanvasFillStrokeStyles<CanvasRenderingContext2D>
  45. , public CanvasRect
  46. , public CanvasDrawPath
  47. , public CanvasText
  48. , public CanvasDrawImage
  49. , public CanvasImageData
  50. , public CanvasImageSmoothing
  51. , public CanvasCompositing
  52. , public CanvasPathDrawingStyles<CanvasRenderingContext2D>
  53. , public CanvasTextDrawingStyles<CanvasRenderingContext2D> {
  54. WEB_PLATFORM_OBJECT(CanvasRenderingContext2D, Bindings::PlatformObject);
  55. JS_DECLARE_ALLOCATOR(CanvasRenderingContext2D);
  56. public:
  57. [[nodiscard]] static JS::NonnullGCPtr<CanvasRenderingContext2D> create(JS::Realm&, HTMLCanvasElement&);
  58. virtual ~CanvasRenderingContext2D() override;
  59. virtual void fill_rect(float x, float y, float width, float height) override;
  60. virtual void stroke_rect(float x, float y, float width, float height) override;
  61. virtual void clear_rect(float x, float y, float width, float height) override;
  62. virtual WebIDL::ExceptionOr<void> draw_image_internal(CanvasImageSource const&, float source_x, float source_y, float source_width, float source_height, float destination_x, float destination_y, float destination_width, float destination_height) override;
  63. virtual void begin_path() override;
  64. virtual void stroke() override;
  65. virtual void stroke(Path2D const& path) override;
  66. virtual void fill_text(StringView, float x, float y, Optional<double> max_width) override;
  67. virtual void stroke_text(StringView, float x, float y, Optional<double> max_width) override;
  68. virtual void fill(StringView fill_rule) override;
  69. virtual void fill(Path2D& path, StringView fill_rule) override;
  70. virtual JS::GCPtr<ImageData> create_image_data(int width, int height) const override;
  71. virtual WebIDL::ExceptionOr<JS::GCPtr<ImageData>> get_image_data(int x, int y, int width, int height) const override;
  72. virtual void put_image_data(ImageData const&, float x, float y) override;
  73. virtual void reset_to_default_state() override;
  74. JS::NonnullGCPtr<HTMLCanvasElement> canvas_for_binding() const;
  75. virtual JS::NonnullGCPtr<TextMetrics> measure_text(StringView text) override;
  76. virtual void clip(StringView fill_rule) override;
  77. virtual void clip(Path2D& path, StringView fill_rule) override;
  78. virtual bool image_smoothing_enabled() const override;
  79. virtual void set_image_smoothing_enabled(bool) override;
  80. virtual Bindings::ImageSmoothingQuality image_smoothing_quality() const override;
  81. virtual void set_image_smoothing_quality(Bindings::ImageSmoothingQuality) override;
  82. virtual float global_alpha() const override;
  83. virtual void set_global_alpha(float) override;
  84. HTMLCanvasElement& canvas_element();
  85. HTMLCanvasElement const& canvas_element() const;
  86. private:
  87. explicit CanvasRenderingContext2D(JS::Realm&, HTMLCanvasElement&);
  88. virtual void initialize(JS::Realm&) override;
  89. virtual void visit_edges(Cell::Visitor&) override;
  90. struct PreparedTextGlyph {
  91. String glyph;
  92. Gfx::IntPoint position;
  93. };
  94. struct PreparedText {
  95. Vector<PreparedTextGlyph> glyphs;
  96. Gfx::TextAlignment physical_alignment;
  97. Gfx::IntRect bounding_box;
  98. };
  99. void did_draw(Gfx::FloatRect const&);
  100. template<typename TDrawFunction>
  101. void draw_clipped(TDrawFunction draw_function)
  102. {
  103. auto painter = this->antialiased_painter();
  104. if (!painter.has_value())
  105. return;
  106. Gfx::ScopedPathClip clipper(painter->underlying_painter(), drawing_state().clip);
  107. auto draw_rect = draw_function(*painter);
  108. if (drawing_state().clip.has_value())
  109. draw_rect.intersect(drawing_state().clip->path.bounding_box());
  110. did_draw(draw_rect);
  111. }
  112. RefPtr<Gfx::Font const> current_font();
  113. PreparedText prepare_text(ByteString const& text, float max_width = INFINITY);
  114. Gfx::Painter* painter();
  115. Optional<Gfx::AntiAliasingPainter> antialiased_painter();
  116. Gfx::Path rect_path(float x, float y, float width, float height);
  117. Gfx::Path text_path(StringView text, float x, float y, Optional<double> max_width);
  118. void bitmap_font_fill_text(StringView text, float x, float y, Optional<double> max_width);
  119. void stroke_internal(Gfx::Path const&);
  120. void fill_internal(Gfx::Path const&, Gfx::Painter::WindingRule);
  121. void clip_internal(Gfx::Path&, Gfx::Painter::WindingRule);
  122. JS::NonnullGCPtr<HTMLCanvasElement> m_element;
  123. OwnPtr<Gfx::Painter> m_painter;
  124. // https://html.spec.whatwg.org/multipage/canvas.html#concept-canvas-origin-clean
  125. bool m_origin_clean { true };
  126. };
  127. enum class CanvasImageSourceUsability {
  128. Bad,
  129. Good,
  130. };
  131. WebIDL::ExceptionOr<CanvasImageSourceUsability> check_usability_of_image(CanvasImageSource const&);
  132. bool image_is_not_origin_clean(CanvasImageSource const&);
  133. }