Serenity Operating System
at master 82 lines 3.9 kB view raw
1/* 2 * Copyright (c) 2022, Andreas Kling <kling@serenityos.org> 3 * Copyright (c) 2023, MacDue <macdue@dueutil.tech> 4 * 5 * SPDX-License-Identifier: BSD-2-Clause 6 */ 7 8#include <AK/QuickSort.h> 9#include <LibWeb/Bindings/Intrinsics.h> 10#include <LibWeb/HTML/CanvasGradient.h> 11#include <LibWeb/WebIDL/ExceptionOr.h> 12 13namespace Web::HTML { 14 15// https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-createradialgradient 16WebIDL::ExceptionOr<JS::NonnullGCPtr<CanvasGradient>> CanvasGradient::create_radial(JS::Realm& realm, double x0, double y0, double r0, double x1, double y1, double r1) 17{ 18 // If either of r0 or r1 are negative, then an "IndexSizeError" DOMException must be thrown. 19 if (r0 < 0) 20 return WebIDL::IndexSizeError::create(realm, "The r0 passed is less than 0"); 21 if (r1 < 0) 22 return WebIDL::IndexSizeError::create(realm, "The r1 passed is less than 0"); 23 24 auto radial_gradient = TRY_OR_THROW_OOM(realm.vm(), Gfx::CanvasRadialGradientPaintStyle::create(Gfx::FloatPoint { x0, y0 }, r0, Gfx::FloatPoint { x1, y1 }, r1)); 25 return MUST_OR_THROW_OOM(realm.heap().allocate<CanvasGradient>(realm, realm, *radial_gradient)); 26} 27 28// https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-createlineargradient 29WebIDL::ExceptionOr<JS::NonnullGCPtr<CanvasGradient>> CanvasGradient::create_linear(JS::Realm& realm, double x0, double y0, double x1, double y1) 30{ 31 auto linear_gradient = TRY_OR_THROW_OOM(realm.vm(), Gfx::CanvasLinearGradientPaintStyle::create(Gfx::FloatPoint { x0, y0 }, Gfx::FloatPoint { x1, y1 })); 32 return MUST_OR_THROW_OOM(realm.heap().allocate<CanvasGradient>(realm, realm, *linear_gradient)); 33} 34 35// https://html.spec.whatwg.org/multipage/canvas.html#dom-context-2d-createconicgradient 36WebIDL::ExceptionOr<JS::NonnullGCPtr<CanvasGradient>> CanvasGradient::create_conic(JS::Realm& realm, double start_angle, double x, double y) 37{ 38 auto conic_gradient = TRY_OR_THROW_OOM(realm.vm(), Gfx::CanvasConicGradientPaintStyle::create(Gfx::FloatPoint { x, y }, start_angle)); 39 return MUST_OR_THROW_OOM(realm.heap().allocate<CanvasGradient>(realm, realm, *conic_gradient)); 40} 41 42CanvasGradient::CanvasGradient(JS::Realm& realm, Gfx::GradientPaintStyle& gradient) 43 : PlatformObject(realm) 44 , m_gradient(gradient) 45{ 46} 47 48CanvasGradient::~CanvasGradient() = default; 49 50JS::ThrowCompletionOr<void> CanvasGradient::initialize(JS::Realm& realm) 51{ 52 MUST_OR_THROW_OOM(Base::initialize(realm)); 53 set_prototype(&Bindings::ensure_web_prototype<Bindings::CanvasGradientPrototype>(realm, "CanvasGradient")); 54 55 return {}; 56} 57 58// https://html.spec.whatwg.org/multipage/canvas.html#dom-canvasgradient-addcolorstop 59WebIDL::ExceptionOr<void> CanvasGradient::add_color_stop(double offset, DeprecatedString const& color) 60{ 61 // 1. If the offset is less than 0 or greater than 1, then throw an "IndexSizeError" DOMException. 62 if (offset < 0 || offset > 1) 63 return WebIDL::IndexSizeError::create(realm(), "CanvasGradient color stop offset out of bounds"); 64 65 // 2. Let parsed color be the result of parsing color. 66 auto parsed_color = Color::from_string(color); 67 68 // 3. If parsed color is failure, throw a "SyntaxError" DOMException. 69 if (!parsed_color.has_value()) 70 return WebIDL::SyntaxError::create(realm(), "Could not parse color for CanvasGradient"); 71 72 // 4. Place a new stop on the gradient, at offset offset relative to the whole gradient, and with the color parsed color. 73 TRY_OR_THROW_OOM(realm().vm(), m_gradient->add_color_stop(offset, parsed_color.value())); 74 75 // FIXME: If multiple stops are added at the same offset on a gradient, then they must be placed in the order added, 76 // with the first one closest to the start of the gradient, and each subsequent one infinitesimally further along 77 // towards the end point (in effect causing all but the first and last stop added at each point to be ignored). 78 79 return {}; 80} 81 82}