ImageData.cpp 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. /*
  2. * Copyright (c) 2020-2022, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2024, Kenneth Myhra <kennethmyhra@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <LibGfx/Bitmap.h>
  8. #include <LibJS/Runtime/TypedArray.h>
  9. #include <LibWeb/Bindings/Intrinsics.h>
  10. #include <LibWeb/HTML/ImageData.h>
  11. #include <LibWeb/WebIDL/DOMException.h>
  12. #include <LibWeb/WebIDL/ExceptionOr.h>
  13. namespace Web::HTML {
  14. JS_DEFINE_ALLOCATOR(ImageData);
  15. WebIDL::ExceptionOr<JS::NonnullGCPtr<ImageData>> ImageData::create(JS::Realm& realm, u32 sw, u32 sh, Optional<ImageDataSettings> const&)
  16. {
  17. auto& vm = realm.vm();
  18. // 1. If one or both of sw and sh are zero, then throw an "IndexSizeError" DOMException.
  19. if (sw == 0 || sh == 0)
  20. return WebIDL::IndexSizeError::create(realm, "The source width and height must be greater than zero."_fly_string);
  21. // 2. Initialize this given sw, sh, and settings set to settings.
  22. // 3. Initialize the image data of this to transparent black.
  23. auto data = TRY(JS::Uint8ClampedArray::create(realm, sw * sh * 4));
  24. auto bitmap = TRY_OR_THROW_OOM(vm, Gfx::Bitmap::create_wrapper(Gfx::BitmapFormat::RGBA8888, Gfx::IntSize(sw, sw), 1, sw * sizeof(u32), data->data().data()));
  25. return realm.heap().allocate<ImageData>(realm, realm, bitmap, data);
  26. }
  27. WebIDL::ExceptionOr<JS::NonnullGCPtr<ImageData>> ImageData::construct_impl(JS::Realm& realm, u32 sw, u32 sh, Optional<ImageDataSettings> const& settings)
  28. {
  29. return ImageData::create(realm, sw, sh, settings);
  30. }
  31. JS::GCPtr<ImageData> ImageData::create_with_size(JS::Realm& realm, int width, int height)
  32. {
  33. if (width <= 0 || height <= 0)
  34. return nullptr;
  35. if (width > 16384 || height > 16384)
  36. return nullptr;
  37. auto data_or_error = JS::Uint8ClampedArray::create(realm, width * height * 4);
  38. if (data_or_error.is_error())
  39. return nullptr;
  40. auto data = JS::NonnullGCPtr<JS::Uint8ClampedArray>(*data_or_error.release_value());
  41. auto bitmap_or_error = Gfx::Bitmap::create_wrapper(Gfx::BitmapFormat::RGBA8888, Gfx::IntSize(width, height), 1, width * sizeof(u32), data->data().data());
  42. if (bitmap_or_error.is_error())
  43. return nullptr;
  44. return realm.heap().allocate<ImageData>(realm, realm, bitmap_or_error.release_value(), move(data));
  45. }
  46. ImageData::ImageData(JS::Realm& realm, NonnullRefPtr<Gfx::Bitmap> bitmap, JS::NonnullGCPtr<JS::Uint8ClampedArray> data)
  47. : PlatformObject(realm)
  48. , m_bitmap(move(bitmap))
  49. , m_data(move(data))
  50. {
  51. }
  52. ImageData::~ImageData() = default;
  53. void ImageData::initialize(JS::Realm& realm)
  54. {
  55. Base::initialize(realm);
  56. WEB_SET_PROTOTYPE_FOR_INTERFACE(ImageData);
  57. }
  58. void ImageData::visit_edges(Cell::Visitor& visitor)
  59. {
  60. Base::visit_edges(visitor);
  61. visitor.visit(m_data);
  62. }
  63. unsigned ImageData::width() const
  64. {
  65. return m_bitmap->width();
  66. }
  67. unsigned ImageData::height() const
  68. {
  69. return m_bitmap->height();
  70. }
  71. JS::Uint8ClampedArray* ImageData::data()
  72. {
  73. return m_data;
  74. }
  75. const JS::Uint8ClampedArray* ImageData::data() const
  76. {
  77. return m_data;
  78. }
  79. }