/* * Copyright (c) 2020-2022, Andreas Kling * Copyright (c) 2024, Kenneth Myhra * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include namespace Web::HTML { GC_DEFINE_ALLOCATOR(ImageData); // https://html.spec.whatwg.org/multipage/canvas.html#dom-imagedata WebIDL::ExceptionOr> ImageData::create(JS::Realm& realm, u32 sw, u32 sh, Optional const&) { auto& vm = realm.vm(); // 1. If one or both of sw and sh are zero, then throw an "IndexSizeError" DOMException. if (sw == 0 || sh == 0) return WebIDL::IndexSizeError::create(realm, "The source width and height must be greater than zero."_string); // 2. Initialize this given sw, sh, and settings set to settings. // 3. Initialize the image data of this to transparent black. // // If the Canvas Pixel ArrayBuffer cannot be allocated, then rethrow the RangeError thrown by JavaScript, and return. Checked size = sw; size *= sh; size *= sizeof(u32); if (size.has_overflow()) return WebIDL::IndexSizeError::create(realm, "The specified image size could not created"_string); auto data = TRY(JS::Uint8ClampedArray::create(realm, size.value())); auto bitmap = TRY_OR_THROW_OOM(vm, Gfx::Bitmap::create_wrapper(Gfx::BitmapFormat::RGBA8888, Gfx::AlphaType::Unpremultiplied, Gfx::IntSize(sw, sh), sw * sizeof(u32), data->data().data())); return realm.create(realm, bitmap, data); } WebIDL::ExceptionOr> ImageData::construct_impl(JS::Realm& realm, u32 sw, u32 sh, Optional const& settings) { return ImageData::create(realm, sw, sh, settings); } // https://html.spec.whatwg.org/multipage/canvas.html#dom-imagedata-with-data WebIDL::ExceptionOr> ImageData::create(JS::Realm& realm, GC::Root const& data, u32 sw, Optional sh, Optional const&) { auto& vm = realm.vm(); if (!is(*data->raw_object())) return vm.throw_completion(JS::ErrorType::NotAnObjectOfType, "Uint8ClampedArray"); auto& uint8_clamped_array_data = static_cast(*data->raw_object()); // 1. Let length be the number of bytes in data. auto length = uint8_clamped_array_data.byte_length().length(); // 2. If length is not a nonzero integral multiple of four, then throw an "InvalidStateError" DOMException. if (length == 0 || length % 4 != 0) return WebIDL::InvalidStateError::create(realm, "Source data must have a non-sero length that is a multiple of four."_string); // 3. Let length be length divided by four. length = length / 4; // 4. If length is not an integral multiple of sw, then throw an "IndexSizeError" DOMException. // NOTE: At this step, the length is guaranteed to be greater than zero (otherwise the second step above would have aborted the steps), // so if sw is zero, this step will throw the exception and return. if (sw == 0 || length % sw != 0) return WebIDL::IndexSizeError::create(realm, "Source width must be a multiple of source data's length."_string); // 5. Let height be length divided by sw. auto height = length / sw; // 6. If sh was given and its value is not equal to height, then throw an "IndexSizeError" DOMException. if (sh.has_value() && sh.value() != height) return WebIDL::IndexSizeError::create(realm, "Source height must be equal to the calculated height of the data."_string); // 7. Initialize this given sw, sh, settings set to settings, and source set to data. auto bitmap = TRY_OR_THROW_OOM(vm, Gfx::Bitmap::create_wrapper(Gfx::BitmapFormat::RGBA8888, Gfx::AlphaType::Unpremultiplied, Gfx::IntSize(sw, height), sw * sizeof(u32), uint8_clamped_array_data.data().data())); return realm.create(realm, bitmap, uint8_clamped_array_data); } WebIDL::ExceptionOr> ImageData::construct_impl(JS::Realm& realm, GC::Root const& data, u32 sw, Optional sh, Optional const& settings) { return ImageData::create(realm, data, sw, move(sh), settings); } ImageData::ImageData(JS::Realm& realm, NonnullRefPtr bitmap, GC::Ref data) : PlatformObject(realm) , m_bitmap(move(bitmap)) , m_data(move(data)) { } ImageData::~ImageData() = default; void ImageData::initialize(JS::Realm& realm) { Base::initialize(realm); WEB_SET_PROTOTYPE_FOR_INTERFACE(ImageData); } void ImageData::visit_edges(Cell::Visitor& visitor) { Base::visit_edges(visitor); visitor.visit(m_data); } unsigned ImageData::width() const { return m_bitmap->width(); } unsigned ImageData::height() const { return m_bitmap->height(); } JS::Uint8ClampedArray* ImageData::data() { return m_data; } const JS::Uint8ClampedArray* ImageData::data() const { return m_data; } }