2024-03-07 23:29:25 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2024, Aliaksandr Kalenik <kalenik.aliaksandr@gmail.com>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <LibWeb/Bindings/ExceptionOrUtils.h>
|
|
|
|
#include <LibWeb/WebIDL/ObservableArray.h>
|
|
|
|
|
|
|
|
namespace Web::WebIDL {
|
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
GC_DEFINE_ALLOCATOR(ObservableArray);
|
2024-04-06 10:16:04 -07:00
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
GC::Ref<ObservableArray> ObservableArray::create(JS::Realm& realm)
|
2024-03-07 23:29:25 +01:00
|
|
|
{
|
|
|
|
auto prototype = realm.intrinsics().array_prototype();
|
2025-05-22 19:46:08 +03:00
|
|
|
return realm.create<ObservableArray>(realm, prototype);
|
2024-03-07 23:29:25 +01:00
|
|
|
}
|
|
|
|
|
2025-05-22 19:46:08 +03:00
|
|
|
ObservableArray::ObservableArray(JS::Realm& realm, Object& prototype)
|
|
|
|
: JS::Array(realm, prototype)
|
2024-03-07 23:29:25 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void ObservableArray::visit_edges(JS::Cell::Visitor& visitor)
|
|
|
|
{
|
|
|
|
Base::visit_edges(visitor);
|
|
|
|
visitor.visit(m_on_set_an_indexed_value);
|
|
|
|
visitor.visit(m_on_delete_an_indexed_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ObservableArray::set_on_set_an_indexed_value_callback(SetAnIndexedValueCallbackFunction&& callback)
|
|
|
|
{
|
2024-11-15 04:01:23 +13:00
|
|
|
m_on_set_an_indexed_value = GC::create_function(heap(), move(callback));
|
2024-03-07 23:29:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void ObservableArray::set_on_delete_an_indexed_value_callback(DeleteAnIndexedValueCallbackFunction&& callback)
|
|
|
|
{
|
2024-11-15 04:01:23 +13:00
|
|
|
m_on_delete_an_indexed_value = GC::create_function(heap(), move(callback));
|
2024-03-07 23:29:25 +01:00
|
|
|
}
|
|
|
|
|
2025-09-15 17:23:39 +02:00
|
|
|
JS::ThrowCompletionOr<bool> ObservableArray::internal_set(JS::PropertyKey const& property_key, JS::Value value, JS::Value receiver, JS::CacheableSetPropertyMetadata* metadata, PropertyLookupPhase phase)
|
2024-03-07 23:29:25 +01:00
|
|
|
{
|
|
|
|
if (property_key.is_number() && m_on_set_an_indexed_value)
|
|
|
|
TRY(Bindings::throw_dom_exception_if_needed(vm(), [&] { return m_on_set_an_indexed_value->function()(value); }));
|
2025-05-04 16:00:13 +02:00
|
|
|
return TRY(Base::internal_set(property_key, value, receiver, metadata, phase));
|
2024-03-07 23:29:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
JS::ThrowCompletionOr<bool> ObservableArray::internal_delete(JS::PropertyKey const& property_key)
|
|
|
|
{
|
2025-01-10 17:25:42 +03:00
|
|
|
if (property_key.is_number() && m_on_delete_an_indexed_value) {
|
|
|
|
auto maybe_value_and_attributes = indexed_properties().get(property_key.as_number());
|
|
|
|
JS::Value deleted_value;
|
|
|
|
if (maybe_value_and_attributes.has_value())
|
|
|
|
deleted_value = maybe_value_and_attributes->value;
|
|
|
|
TRY(Bindings::throw_dom_exception_if_needed(vm(), [&] { return m_on_delete_an_indexed_value->function()(deleted_value); }));
|
|
|
|
}
|
2024-03-07 23:29:25 +01:00
|
|
|
return JS::Array::internal_delete(property_key);
|
|
|
|
}
|
|
|
|
|
|
|
|
JS::ThrowCompletionOr<void> ObservableArray::append(JS::Value value)
|
|
|
|
{
|
|
|
|
if (m_on_set_an_indexed_value)
|
|
|
|
TRY(Bindings::throw_dom_exception_if_needed(vm(), [&] { return m_on_set_an_indexed_value->function()(value); }));
|
|
|
|
indexed_properties().append(value);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
void ObservableArray::clear()
|
|
|
|
{
|
|
|
|
while (!indexed_properties().is_empty()) {
|
2025-04-24 14:36:34 +02:00
|
|
|
auto deleted_value = indexed_properties().storage()->take_first().value;
|
|
|
|
MUST(m_on_delete_an_indexed_value->function()(deleted_value));
|
2024-03-07 23:29:25 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|