| 
									
										
										
										
											2022-08-12 14:00:00 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2022, Sam Atkins <atkinssj@serenityos.org> | 
					
						
							| 
									
										
										
										
											2023-01-18 20:10:00 +01:00
										 |  |  |  * Copyright (c) 2023, MacDue <macdue@dueutil.tech> | 
					
						
							| 
									
										
										
										
											2022-08-12 14:00:00 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-18 20:10:00 +01:00
										 |  |  | #include <AK/Variant.h>
 | 
					
						
							| 
									
										
										
										
											2022-08-12 14:00:00 +01:00
										 |  |  | #include <AK/Vector.h>
 | 
					
						
							|  |  |  | #include <LibGfx/AffineTransform.h>
 | 
					
						
							|  |  |  | #include <LibGfx/Color.h>
 | 
					
						
							| 
									
										
										
										
											2025-01-28 18:19:30 +01:00
										 |  |  | #include <LibGfx/CompositingAndBlendingOperator.h>
 | 
					
						
							| 
									
										
										
										
											2024-12-18 12:26:37 +01:00
										 |  |  | #include <LibGfx/Filter.h>
 | 
					
						
							| 
									
										
										
										
											2024-06-05 10:33:18 +02:00
										 |  |  | #include <LibGfx/Font/Font.h>
 | 
					
						
							| 
									
										
										
										
											2025-04-19 18:34:53 +02:00
										 |  |  | #include <LibGfx/FontCascadeList.h>
 | 
					
						
							| 
									
										
										
										
											2023-01-18 20:10:00 +01:00
										 |  |  | #include <LibGfx/PaintStyle.h>
 | 
					
						
							| 
									
										
										
										
											2024-08-08 15:12:29 +02:00
										 |  |  | #include <LibGfx/Path.h>
 | 
					
						
							|  |  |  | #include <LibGfx/WindingRule.h>
 | 
					
						
							| 
									
										
										
										
											2023-03-29 18:35:02 +02:00
										 |  |  | #include <LibWeb/Bindings/CanvasRenderingContext2DPrototype.h>
 | 
					
						
							| 
									
										
										
										
											2023-01-18 20:10:00 +01:00
										 |  |  | #include <LibWeb/HTML/CanvasGradient.h>
 | 
					
						
							| 
									
										
										
										
											2023-02-02 20:47:46 +00:00
										 |  |  | #include <LibWeb/HTML/CanvasPattern.h>
 | 
					
						
							| 
									
										
										
										
											2022-08-12 14:00:00 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Web::HTML { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // https://html.spec.whatwg.org/multipage/canvas.html#canvasstate
 | 
					
						
							|  |  |  | class CanvasState { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     virtual ~CanvasState() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-15 14:52:40 +02:00
										 |  |  |     virtual Gfx::Painter* painter_for_canvas_state() = 0; | 
					
						
							|  |  |  |     virtual Gfx::Path& path_for_canvas_state() = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-12 14:00:00 +01:00
										 |  |  |     void save(); | 
					
						
							|  |  |  |     void restore(); | 
					
						
							|  |  |  |     void reset(); | 
					
						
							|  |  |  |     bool is_context_lost(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |     using FillOrStrokeVariant = Variant<Gfx::Color, GC::Root<CanvasGradient>, GC::Root<CanvasPattern>>; | 
					
						
							| 
									
										
										
										
											2023-01-18 20:10:00 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     struct FillOrStrokeStyle { | 
					
						
							|  |  |  |         FillOrStrokeStyle(Gfx::Color color) | 
					
						
							| 
									
										
										
										
											2023-02-02 20:41:03 +00:00
										 |  |  |             : m_fill_or_stroke_style(color) | 
					
						
							| 
									
										
										
										
											2023-01-18 20:10:00 +01:00
										 |  |  |         { | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |         FillOrStrokeStyle(GC::Root<CanvasGradient> gradient) | 
					
						
							| 
									
										
										
										
											2023-02-02 20:41:03 +00:00
										 |  |  |             : m_fill_or_stroke_style(gradient) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |         FillOrStrokeStyle(GC::Root<CanvasPattern> pattern) | 
					
						
							| 
									
										
										
										
											2023-02-02 20:41:03 +00:00
										 |  |  |             : m_fill_or_stroke_style(pattern) | 
					
						
							| 
									
										
										
										
											2023-01-18 20:10:00 +01:00
										 |  |  |         { | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         NonnullRefPtr<Gfx::PaintStyle> to_gfx_paint_style(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Optional<Gfx::Color> as_color() const; | 
					
						
							|  |  |  |         Gfx::Color to_color_but_fixme_should_accept_any_paint_style() const; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |         using JsFillOrStrokeStyle = Variant<String, GC::Root<CanvasGradient>, GC::Root<CanvasPattern>>; | 
					
						
							| 
									
										
										
										
											2023-02-02 20:41:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         JsFillOrStrokeStyle to_js_fill_or_stroke_style() const | 
					
						
							| 
									
										
										
										
											2023-01-18 20:10:00 +01:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-02-02 20:41:03 +00:00
										 |  |  |             return m_fill_or_stroke_style.visit( | 
					
						
							|  |  |  |                 [&](Gfx::Color color) -> JsFillOrStrokeStyle { | 
					
						
							| 
									
										
										
										
											2024-10-04 14:10:07 +02:00
										 |  |  |                     return color.to_string(Gfx::Color::HTMLCompatibleSerialization::Yes); | 
					
						
							| 
									
										
										
										
											2023-02-02 20:41:03 +00:00
										 |  |  |                 }, | 
					
						
							|  |  |  |                 [&](auto handle) -> JsFillOrStrokeStyle { | 
					
						
							|  |  |  |                     return handle; | 
					
						
							|  |  |  |                 }); | 
					
						
							| 
									
										
										
										
											2023-01-18 20:10:00 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private: | 
					
						
							| 
									
										
										
										
											2023-02-02 20:41:03 +00:00
										 |  |  |         FillOrStrokeVariant m_fill_or_stroke_style; | 
					
						
							| 
									
										
										
										
											2023-01-19 19:10:00 +01:00
										 |  |  |         RefPtr<Gfx::PaintStyle> m_color_paint_style { nullptr }; | 
					
						
							| 
									
										
										
										
											2023-01-18 20:10:00 +01:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-12 14:00:00 +01:00
										 |  |  |     // https://html.spec.whatwg.org/multipage/canvas.html#drawing-state
 | 
					
						
							|  |  |  |     struct DrawingState { | 
					
						
							|  |  |  |         Gfx::AffineTransform transform; | 
					
						
							| 
									
										
										
										
											2023-01-18 20:10:00 +01:00
										 |  |  |         FillOrStrokeStyle fill_style { Gfx::Color::Black }; | 
					
						
							|  |  |  |         FillOrStrokeStyle stroke_style { Gfx::Color::Black }; | 
					
						
							| 
									
										
										
										
											2024-10-18 01:56:58 +03:00
										 |  |  |         float shadow_offset_x { 0.0f }; | 
					
						
							|  |  |  |         float shadow_offset_y { 0.0f }; | 
					
						
							| 
									
										
										
										
											2024-12-03 21:22:44 +01:00
										 |  |  |         float shadow_blur { 0.0f }; | 
					
						
							| 
									
										
										
										
											2024-10-18 01:56:58 +03:00
										 |  |  |         Gfx::Color shadow_color { Gfx::Color::Transparent }; | 
					
						
							| 
									
										
										
										
											2024-12-18 12:26:37 +01:00
										 |  |  |         Vector<Gfx::Filter> filters; | 
					
						
							|  |  |  |         Optional<String> filters_string; | 
					
						
							| 
									
										
										
										
											2022-08-12 14:00:00 +01:00
										 |  |  |         float line_width { 1 }; | 
					
						
							| 
									
										
										
										
											2024-11-07 09:27:55 -05:00
										 |  |  |         Bindings::CanvasLineCap line_cap { Bindings::CanvasLineCap::Butt }; | 
					
						
							|  |  |  |         Bindings::CanvasLineJoin line_join { Bindings::CanvasLineJoin::Miter }; | 
					
						
							|  |  |  |         float miter_limit { 10 }; | 
					
						
							| 
									
										
										
										
											2024-06-24 19:02:30 -03:00
										 |  |  |         Vector<double> dash_list; | 
					
						
							| 
									
										
										
										
											2024-11-07 09:27:55 -05:00
										 |  |  |         float line_dash_offset { 0 }; | 
					
						
							| 
									
										
										
										
											2023-03-29 18:35:02 +02:00
										 |  |  |         bool image_smoothing_enabled { true }; | 
					
						
							|  |  |  |         Bindings::ImageSmoothingQuality image_smoothing_quality { Bindings::ImageSmoothingQuality::Low }; | 
					
						
							| 
									
										
										
										
											2023-05-19 22:42:47 +02:00
										 |  |  |         float global_alpha = { 1 }; | 
					
						
							| 
									
										
										
										
											2025-01-28 18:19:30 +01:00
										 |  |  |         Gfx::CompositingAndBlendingOperator current_compositing_and_blending_operator = Gfx::CompositingAndBlendingOperator::SourceOver; | 
					
						
							| 
									
										
										
										
											2025-04-15 15:18:27 -06:00
										 |  |  |         RefPtr<CSS::CSSStyleValue const> font_style_value { nullptr }; | 
					
						
							| 
									
										
										
										
											2025-04-19 18:34:53 +02:00
										 |  |  |         RefPtr<Gfx::FontCascadeList const> current_font_cascade_list { nullptr }; | 
					
						
							| 
									
										
										
										
											2023-08-03 12:18:17 +02:00
										 |  |  |         Bindings::CanvasTextAlign text_align { Bindings::CanvasTextAlign::Start }; | 
					
						
							|  |  |  |         Bindings::CanvasTextBaseline text_baseline { Bindings::CanvasTextBaseline::Alphabetic }; | 
					
						
							| 
									
										
										
										
											2022-08-12 14:00:00 +01:00
										 |  |  |     }; | 
					
						
							|  |  |  |     DrawingState& drawing_state() { return m_drawing_state; } | 
					
						
							|  |  |  |     DrawingState const& drawing_state() const { return m_drawing_state; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void clear_drawing_state_stack() { m_drawing_state_stack.clear(); } | 
					
						
							|  |  |  |     void reset_drawing_state() { m_drawing_state = {}; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual void reset_to_default_state() = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protected: | 
					
						
							|  |  |  |     CanvasState() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     DrawingState m_drawing_state; | 
					
						
							|  |  |  |     Vector<DrawingState> m_drawing_state_stack; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // https://html.spec.whatwg.org/multipage/canvas.html#concept-canvas-context-lost
 | 
					
						
							|  |  |  |     bool m_context_lost { false }; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |