|
9 | 9 | #include <LibJS/Runtime/TypedArray.h>
|
10 | 10 | #include <LibWeb/Bindings/Intrinsics.h>
|
11 | 11 | #include <LibWeb/HTML/ImageData.h>
|
| 12 | +#include <LibWeb/WebIDL/Buffers.h> |
12 | 13 | #include <LibWeb/WebIDL/DOMException.h>
|
13 | 14 | #include <LibWeb/WebIDL/ExceptionOr.h>
|
14 | 15 |
|
@@ -37,6 +38,50 @@ WebIDL::ExceptionOr<JS::NonnullGCPtr<ImageData>> ImageData::construct_impl(JS::R
|
37 | 38 | return ImageData::create(realm, sw, sh, settings);
|
38 | 39 | }
|
39 | 40 |
|
| 41 | +// https://html.spec.whatwg.org/multipage/canvas.html#dom-imagedata-with-data |
| 42 | +WebIDL::ExceptionOr<JS::NonnullGCPtr<ImageData>> ImageData::create(JS::Realm& realm, JS::Handle<WebIDL::BufferSource> const& data, u32 sw, Optional<u32> sh, Optional<ImageDataSettings> const&) |
| 43 | +{ |
| 44 | + auto& vm = realm.vm(); |
| 45 | + |
| 46 | + if (!is<JS::Uint8ClampedArray>(*data->raw_object())) |
| 47 | + return vm.throw_completion<JS::TypeError>(JS::ErrorType::NotAnObjectOfType, "Uint8ClampedArray"); |
| 48 | + |
| 49 | + auto& uint8_clamped_array_data = static_cast<JS::Uint8ClampedArray&>(*data->raw_object()); |
| 50 | + |
| 51 | + // 1. Let length be the number of bytes in data. |
| 52 | + auto length = uint8_clamped_array_data.byte_length().length(); |
| 53 | + |
| 54 | + // 2. If length is not a nonzero integral multiple of four, then throw an "InvalidStateError" DOMException. |
| 55 | + if (length == 0 || length % 4 != 0) |
| 56 | + return WebIDL::InvalidStateError::create(realm, "Source data must have a non-sero length that is a multiple of four."_fly_string); |
| 57 | + |
| 58 | + // 3. Let length be length divided by four. |
| 59 | + length = length / 4; |
| 60 | + |
| 61 | + // 4. If length is not an integral multiple of sw, then throw an "IndexSizeError" DOMException. |
| 62 | + // NOTE: At this step, the length is guaranteed to be greater than zero (otherwise the second step above would have aborted the steps), |
| 63 | + // so if sw is zero, this step will throw the exception and return. |
| 64 | + if (sw == 0 || length % sw != 0) |
| 65 | + return WebIDL::IndexSizeError::create(realm, "Source width must be a multiple of source data's length."_fly_string); |
| 66 | + |
| 67 | + // 5. Let height be length divided by sw. |
| 68 | + auto height = length / sw; |
| 69 | + |
| 70 | + // 6. If sh was given and its value is not equal to height, then throw an "IndexSizeError" DOMException. |
| 71 | + if (sh.has_value() && sh.value() != height) |
| 72 | + return WebIDL::IndexSizeError::create(realm, "Source height must be equal to the calculated height of the data."_fly_string); |
| 73 | + |
| 74 | + // 7. Initialize this given sw, sh, settings set to settings, and source set to data. |
| 75 | + auto bitmap = TRY_OR_THROW_OOM(vm, Gfx::Bitmap::create_wrapper(Gfx::BitmapFormat::RGBA8888, Gfx::IntSize(sw, sw), 1, sw * sizeof(u32), uint8_clamped_array_data.data().data())); |
| 76 | + |
| 77 | + return realm.heap().allocate<ImageData>(realm, realm, bitmap, uint8_clamped_array_data); |
| 78 | +} |
| 79 | + |
| 80 | +WebIDL::ExceptionOr<JS::NonnullGCPtr<ImageData>> ImageData::construct_impl(JS::Realm& realm, JS::Handle<WebIDL::BufferSource> const& data, u32 sw, Optional<u32> sh, Optional<ImageDataSettings> const& settings) |
| 81 | +{ |
| 82 | + return ImageData::create(realm, data, sw, move(sh), settings); |
| 83 | +} |
| 84 | + |
40 | 85 | ImageData::ImageData(JS::Realm& realm, NonnullRefPtr<Gfx::Bitmap> bitmap, JS::NonnullGCPtr<JS::Uint8ClampedArray> data)
|
41 | 86 | : PlatformObject(realm)
|
42 | 87 | , m_bitmap(move(bitmap))
|
|
0 commit comments