/* * Copyright (c) 2023, Matthew Olsson . * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include namespace Web::Animations { // https://www.w3.org/TR/web-animations-1/#dom-animation-animation JS::NonnullGCPtr Animation::create(JS::Realm& realm, JS::GCPtr effect, JS::GCPtr timeline) { // 1. Let animation be a new Animation object. auto animation = realm.heap().allocate(realm, realm); // 2. Run the procedure to set the timeline of an animation on animation passing timeline as the new timeline or, if // a timeline argument is missing, passing the default document timeline of the Document associated with the // Window that is the current global object. if (!timeline) { auto& window = verify_cast(HTML::current_global_object()); timeline = window.associated_document().timeline(); } animation->set_timeline(timeline); // 3. Run the procedure to set the associated effect of an animation on animation passing source as the new effect. animation->set_effect(effect); return animation; } WebIDL::ExceptionOr> Animation::construct_impl(JS::Realm& realm, JS::GCPtr effect, JS::GCPtr timeline) { return create(realm, effect, timeline); } // https://www.w3.org/TR/web-animations-1/#animation-set-the-associated-effect-of-an-animation void Animation::set_effect(JS::GCPtr new_effect) { // Setting this attribute updates the object’s associated effect using the procedure to set the associated effect of // an animation. // 1. Let old effect be the current associated effect of animation, if any. auto old_effect = m_effect; // 2. If new effect is the same object as old effect, abort this procedure. if (new_effect == old_effect) return; // FIXME: 3. If animation has a pending pause task, reschedule that task to run as soon as animation is ready. // FIXME: 4. If animation has a pending play task, reschedule that task to run as soon as animation is ready to play // new effect. // 5. If new effect is not null and if new effect is the associated effect of another animation, previous animation, // run the procedure to set the associated effect of an animation (this procedure) on previous animation passing // null as new effect. if (new_effect && new_effect->associated_animation() != this) { if (auto animation = new_effect->associated_animation()) animation->set_effect({}); } // 6. Let the associated effect of animation be new effect. if (new_effect) new_effect->set_associated_animation(this); if (m_effect) m_effect->set_associated_animation({}); m_effect = new_effect; // FIXME: 7. Run the procedure to update an animation’s finished state for animation with the did seek flag set to // false, and the synchronously notify flag set to false. } // https://www.w3.org/TR/web-animations-1/#animation-set-the-timeline-of-an-animation void Animation::set_timeline(JS::GCPtr new_timeline) { // FIXME: Implement (void)new_timeline; } // https://www.w3.org/TR/web-animations-1/#dom-animation-starttime // https://www.w3.org/TR/web-animations-1/#set-the-start-time void Animation::set_start_time(Optional const& new_start_time) { // FIXME: Implement (void)new_start_time; } // https://www.w3.org/TR/web-animations-1/#animation-current-time Optional Animation::current_time() const { // FIXME: Implement return {}; } // https://www.w3.org/TR/web-animations-1/#animation-set-the-current-time WebIDL::ExceptionOr Animation::set_current_time(Optional const& seek_time) { // FIXME: Implement (void)seek_time; return {}; } // https://www.w3.org/TR/web-animations-1/#dom-animation-playbackrate // https://www.w3.org/TR/web-animations-1/#set-the-playback-rate WebIDL::ExceptionOr Animation::set_playback_rate(double new_playback_rate) { // FIXME: Implement (void)new_playback_rate; return {}; } // https://www.w3.org/TR/web-animations-1/#animation-play-state Bindings::AnimationPlayState Animation::play_state() const { // FIXME: Implement return Bindings::AnimationPlayState::Idle; } JS::NonnullGCPtr Animation::current_ready_promise() const { if (!m_current_ready_promise) { // The current ready promise is initially a resolved Promise created using the procedure to create a new // resolved Promise with the animation itself as its value and created in the relevant Realm of the animation. m_current_ready_promise = WebIDL::create_resolved_promise(realm(), this); } return *m_current_ready_promise; } JS::NonnullGCPtr Animation::current_finished_promise() const { if (!m_current_finished_promise) { // The current finished promise is initially a pending Promise object. m_current_finished_promise = WebIDL::create_promise(realm()); } return *m_current_finished_promise; } Animation::Animation(JS::Realm& realm) : DOM::EventTarget(realm) { } void Animation::initialize(JS::Realm& realm) { Base::initialize(realm); set_prototype(&Bindings::ensure_web_prototype(realm, "Animation")); } void Animation::visit_edges(Cell::Visitor& visitor) { Base::visit_edges(visitor); visitor.visit(m_effect); visitor.visit(m_timeline); visitor.visit(m_current_ready_promise); visitor.visit(m_current_finished_promise); } }