| 
									
										
										
										
											2023-11-06 15:17:19 +00:00
										 |  |  |  | /*
 | 
					
						
							|  |  |  |  |  * Copyright (c) 2023, Matthew Olsson <mattco@serenityos.org>. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include <LibWeb/Animations/Animation.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Animations/AnimationEffect.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Animations/DocumentTimeline.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Bindings/Intrinsics.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/DOM/Document.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/HTML/Window.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/WebIDL/ExceptionOr.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/WebIDL/Promise.h>
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | namespace Web::Animations { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://www.w3.org/TR/web-animations-1/#dom-animation-animation
 | 
					
						
							|  |  |  |  | JS::NonnullGCPtr<Animation> Animation::create(JS::Realm& realm, JS::GCPtr<AnimationEffect> effect, JS::GCPtr<AnimationTimeline> timeline) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     // 1. Let animation be a new Animation object.
 | 
					
						
							|  |  |  |  |     auto animation = realm.heap().allocate<Animation>(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::Window>(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<JS::NonnullGCPtr<Animation>> Animation::construct_impl(JS::Realm& realm, JS::GCPtr<AnimationEffect> effect, JS::GCPtr<AnimationTimeline> 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<AnimationEffect> new_effect) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-11-04 11:24:21 -07:00
										 |  |  |  |     // 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.
 | 
					
						
							| 
									
										
										
										
											2023-11-06 15:17:19 +00:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://www.w3.org/TR/web-animations-1/#animation-set-the-timeline-of-an-animation
 | 
					
						
							|  |  |  |  | void Animation::set_timeline(JS::GCPtr<AnimationTimeline> new_timeline) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-11-04 12:01:11 -07:00
										 |  |  |  |     // Setting this attribute updates the object’s timeline using the procedure to set the timeline of an animation.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 1. Let old timeline be the current timeline of animation, if any.
 | 
					
						
							|  |  |  |  |     auto old_timeline = m_timeline; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 2. If new timeline is the same object as old timeline, abort this procedure.
 | 
					
						
							|  |  |  |  |     if (new_timeline == old_timeline) | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 3. Let the timeline of animation be new timeline.
 | 
					
						
							|  |  |  |  |     if (m_timeline) | 
					
						
							|  |  |  |  |         m_timeline->disassociate_with_animation(*this); | 
					
						
							|  |  |  |  |     m_timeline = new_timeline; | 
					
						
							|  |  |  |  |     m_timeline->associate_with_animation(*this); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 4. If the start time of animation is resolved, make animation’s hold time unresolved.
 | 
					
						
							|  |  |  |  |     if (m_start_time.has_value()) | 
					
						
							|  |  |  |  |         m_hold_time = {}; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // FIXME: 5. 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.
 | 
					
						
							| 
									
										
										
										
											2023-11-06 15:17:19 +00:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // 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<double> const& new_start_time) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     // FIXME: Implement
 | 
					
						
							|  |  |  |  |     (void)new_start_time; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://www.w3.org/TR/web-animations-1/#animation-current-time
 | 
					
						
							|  |  |  |  | Optional<double> Animation::current_time() const | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     // FIXME: Implement
 | 
					
						
							|  |  |  |  |     return {}; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://www.w3.org/TR/web-animations-1/#animation-set-the-current-time
 | 
					
						
							|  |  |  |  | WebIDL::ExceptionOr<void> Animation::set_current_time(Optional<double> 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<void> 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<WebIDL::Promise> 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<WebIDL::Promise> 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<Bindings::AnimationPrototype>(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); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | } |