2020-04-06 12:09:01 +03:00
|
|
|
/*
|
2022-02-03 22:01:19 +03:00
|
|
|
* Copyright (c) 2020-2022, Andreas Kling <kling@serenityos.org>
|
2022-03-04 21:02:10 +03:00
|
|
|
* Copyright (c) 2021-2022, Linus Groh <linusg@serenityos.org>
|
2020-04-06 12:09:01 +03:00
|
|
|
*
|
2021-04-22 11:24:48 +03:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-04-06 12:09:01 +03:00
|
|
|
*/
|
|
|
|
|
2020-03-19 21:07:56 +03:00
|
|
|
#pragma once
|
|
|
|
|
2022-06-19 21:42:27 +03:00
|
|
|
#include <AK/RefCountForwarder.h>
|
2022-03-04 21:02:10 +03:00
|
|
|
#include <AK/Variant.h>
|
2020-04-12 20:22:42 +03:00
|
|
|
#include <LibGfx/AffineTransform.h>
|
2020-03-19 21:07:56 +03:00
|
|
|
#include <LibGfx/Color.h>
|
|
|
|
#include <LibGfx/Forward.h>
|
2020-05-06 10:26:47 +03:00
|
|
|
#include <LibGfx/Painter.h>
|
2020-04-16 22:06:03 +03:00
|
|
|
#include <LibGfx/Path.h>
|
2020-03-19 21:07:56 +03:00
|
|
|
#include <LibWeb/Bindings/Wrappable.h>
|
2021-04-14 23:29:14 +03:00
|
|
|
#include <LibWeb/DOM/ExceptionOr.h>
|
2022-08-12 17:46:31 +03:00
|
|
|
#include <LibWeb/HTML/Canvas/CanvasFillStrokeStyles.h>
|
2022-08-11 18:10:04 +03:00
|
|
|
#include <LibWeb/HTML/Canvas/CanvasPath.h>
|
2022-08-12 18:06:32 +03:00
|
|
|
#include <LibWeb/HTML/Canvas/CanvasRect.h>
|
2022-08-12 16:00:00 +03:00
|
|
|
#include <LibWeb/HTML/Canvas/CanvasState.h>
|
2022-08-12 16:48:11 +03:00
|
|
|
#include <LibWeb/HTML/Canvas/CanvasTransform.h>
|
2022-02-03 22:08:27 +03:00
|
|
|
#include <LibWeb/HTML/CanvasGradient.h>
|
2021-12-31 01:15:38 +03:00
|
|
|
#include <LibWeb/Layout/InlineNode.h>
|
|
|
|
#include <LibWeb/Layout/LineBox.h>
|
2020-03-19 21:07:56 +03:00
|
|
|
|
2020-07-28 19:20:36 +03:00
|
|
|
namespace Web::HTML {
|
2020-03-19 21:07:56 +03:00
|
|
|
|
2022-03-04 21:02:10 +03:00
|
|
|
// https://html.spec.whatwg.org/multipage/canvas.html#canvasimagesource
|
|
|
|
// NOTE: This is the Variant created by the IDL wrapper generator, and needs to be updated accordingly.
|
|
|
|
using CanvasImageSource = Variant<NonnullRefPtr<HTMLImageElement>, NonnullRefPtr<HTMLCanvasElement>>;
|
|
|
|
|
2020-03-19 21:07:56 +03:00
|
|
|
class CanvasRenderingContext2D
|
2022-06-19 21:42:27 +03:00
|
|
|
: public RefCountForwarder<HTMLCanvasElement>
|
2022-08-11 18:10:04 +03:00
|
|
|
, public Bindings::Wrappable
|
2022-08-12 16:00:00 +03:00
|
|
|
, public CanvasPath
|
2022-08-12 16:48:11 +03:00
|
|
|
, public CanvasState
|
2022-08-12 17:46:31 +03:00
|
|
|
, public CanvasTransform<CanvasRenderingContext2D>
|
2022-08-12 18:06:32 +03:00
|
|
|
, public CanvasFillStrokeStyles<CanvasRenderingContext2D>
|
|
|
|
, public CanvasRect {
|
2020-03-19 21:07:56 +03:00
|
|
|
|
|
|
|
AK_MAKE_NONCOPYABLE(CanvasRenderingContext2D);
|
|
|
|
AK_MAKE_NONMOVABLE(CanvasRenderingContext2D);
|
|
|
|
|
|
|
|
public:
|
|
|
|
using WrapperType = Bindings::CanvasRenderingContext2DWrapper;
|
|
|
|
|
2021-04-23 17:46:57 +03:00
|
|
|
static NonnullRefPtr<CanvasRenderingContext2D> create(HTMLCanvasElement& element) { return adopt_ref(*new CanvasRenderingContext2D(element)); }
|
2020-03-19 21:07:56 +03:00
|
|
|
~CanvasRenderingContext2D();
|
|
|
|
|
2022-08-12 18:06:32 +03:00
|
|
|
virtual void fill_rect(float x, float y, float width, float height) override;
|
|
|
|
virtual void stroke_rect(float x, float y, float width, float height) override;
|
|
|
|
virtual void clear_rect(float x, float y, float width, float height) override;
|
2020-04-14 21:38:44 +03:00
|
|
|
|
2022-03-05 21:54:22 +03:00
|
|
|
DOM::ExceptionOr<void> draw_image(CanvasImageSource const&, float destination_x, float destination_y);
|
|
|
|
DOM::ExceptionOr<void> draw_image(CanvasImageSource const&, float destination_x, float destination_y, float destination_width, float destination_height);
|
|
|
|
DOM::ExceptionOr<void> draw_image(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);
|
2020-04-14 21:38:44 +03:00
|
|
|
|
2022-08-12 16:00:00 +03:00
|
|
|
void set_line_width(float line_width) { drawing_state().line_width = line_width; }
|
|
|
|
float line_width() const { return drawing_state().line_width; }
|
2020-04-16 22:06:03 +03:00
|
|
|
|
|
|
|
void begin_path();
|
|
|
|
void stroke();
|
2022-08-11 19:39:37 +03:00
|
|
|
void stroke(Path2D const& path);
|
2020-06-22 19:39:22 +03:00
|
|
|
|
2022-04-01 20:58:27 +03:00
|
|
|
void fill_text(String const&, float x, float y, Optional<double> max_width);
|
2022-02-03 22:09:57 +03:00
|
|
|
void stroke_text(String const&, float x, float y, Optional<double> max_width);
|
2021-04-15 20:36:10 +03:00
|
|
|
|
2022-04-01 20:58:27 +03:00
|
|
|
void fill(String const& fill_rule);
|
2022-08-11 19:39:37 +03:00
|
|
|
void fill(Path2D& path, String const& fill_rule);
|
2020-04-16 22:06:03 +03:00
|
|
|
|
2020-06-22 19:39:22 +03:00
|
|
|
RefPtr<ImageData> create_image_data(int width, int height) const;
|
2022-03-04 21:07:35 +03:00
|
|
|
DOM::ExceptionOr<RefPtr<ImageData>> get_image_data(int x, int y, int width, int height) const;
|
2022-04-01 20:58:27 +03:00
|
|
|
void put_image_data(ImageData const&, float x, float y);
|
2020-04-22 00:49:51 +03:00
|
|
|
|
2022-08-12 16:00:00 +03:00
|
|
|
virtual void reset_to_default_state() override;
|
2021-12-27 16:31:23 +03:00
|
|
|
|
2022-06-19 21:42:27 +03:00
|
|
|
NonnullRefPtr<HTMLCanvasElement> canvas_for_binding() const;
|
2020-05-21 02:09:31 +03:00
|
|
|
|
2021-12-31 01:15:38 +03:00
|
|
|
RefPtr<TextMetrics> measure_text(String const& text);
|
|
|
|
|
2022-04-10 20:48:11 +03:00
|
|
|
void clip();
|
2022-04-10 20:46:04 +03:00
|
|
|
|
2020-03-19 21:07:56 +03:00
|
|
|
private:
|
|
|
|
explicit CanvasRenderingContext2D(HTMLCanvasElement&);
|
|
|
|
|
2021-12-31 01:15:38 +03:00
|
|
|
struct PreparedTextGlyph {
|
|
|
|
unsigned int c;
|
|
|
|
Gfx::IntPoint position;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct PreparedText {
|
|
|
|
Vector<PreparedTextGlyph> glyphs;
|
|
|
|
Gfx::TextAlignment physical_alignment;
|
|
|
|
Gfx::IntRect bounding_box;
|
|
|
|
};
|
|
|
|
|
2022-04-01 20:58:27 +03:00
|
|
|
void did_draw(Gfx::FloatRect const&);
|
2021-12-31 01:15:38 +03:00
|
|
|
PreparedText prepare_text(String const& text, float max_width = INFINITY);
|
2020-03-22 23:15:49 +03:00
|
|
|
|
2020-03-19 21:07:56 +03:00
|
|
|
OwnPtr<Gfx::Painter> painter();
|
|
|
|
|
2022-06-19 21:42:27 +03:00
|
|
|
HTMLCanvasElement& canvas_element();
|
|
|
|
HTMLCanvasElement const& canvas_element() const;
|
2020-03-19 21:07:56 +03:00
|
|
|
|
2022-08-11 19:39:37 +03:00
|
|
|
void stroke_internal(Gfx::Path const&);
|
|
|
|
void fill_internal(Gfx::Path&, String const& fill_rule);
|
|
|
|
|
2022-03-04 20:57:29 +03:00
|
|
|
// https://html.spec.whatwg.org/multipage/canvas.html#concept-canvas-origin-clean
|
|
|
|
bool m_origin_clean { true };
|
2020-03-19 21:07:56 +03:00
|
|
|
};
|
|
|
|
|
2022-03-04 21:02:10 +03:00
|
|
|
enum class CanvasImageSourceUsability {
|
|
|
|
Bad,
|
|
|
|
Good,
|
|
|
|
};
|
|
|
|
|
|
|
|
DOM::ExceptionOr<CanvasImageSourceUsability> check_usability_of_image(CanvasImageSource const&);
|
|
|
|
bool image_is_not_origin_clean(CanvasImageSource const&);
|
|
|
|
|
2020-03-19 21:07:56 +03:00
|
|
|
}
|