mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-24 22:12:07 +00:00 
			
		
		
		
	 4f26b35640
			
		
	
	
		4f26b35640
		
	
	
	
	
		
			
			This allows the logic for keeping track of whether to resume to the paused or the playing state when exiting these states. The new StartingStateHandler also uses the class, since it can also be paused and unpaused while waiting for samples. The pause/play actions on the handlers inheriting from the resuming handler will also now notify the owner that the state has changed so that it can change icons, etc.
		
			
				
	
	
		
			265 lines
		
	
	
	
		
			7.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			265 lines
		
	
	
	
		
			7.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2022, Gregory Bertilson <zaggy1024@gmail.com>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/Atomic.h>
 | |
| #include <AK/Function.h>
 | |
| #include <AK/NonnullOwnPtr.h>
 | |
| #include <AK/Queue.h>
 | |
| #include <AK/Time.h>
 | |
| #include <LibCore/EventLoop.h>
 | |
| #include <LibCore/SharedCircularQueue.h>
 | |
| #include <LibGfx/Bitmap.h>
 | |
| #include <LibThreading/ConditionVariable.h>
 | |
| #include <LibThreading/Mutex.h>
 | |
| #include <LibThreading/Thread.h>
 | |
| #include <LibVideo/Containers/Demuxer.h>
 | |
| #include <LibVideo/Containers/Matroska/Document.h>
 | |
| 
 | |
| #include "VideoDecoder.h"
 | |
| 
 | |
| namespace Video {
 | |
| 
 | |
| struct FrameQueueItem {
 | |
|     static constexpr Time no_timestamp = Time::min();
 | |
| 
 | |
|     enum class Type {
 | |
|         Frame,
 | |
|         Error,
 | |
|     };
 | |
| 
 | |
|     static FrameQueueItem frame(RefPtr<Gfx::Bitmap> bitmap, Time timestamp)
 | |
|     {
 | |
|         return FrameQueueItem(move(bitmap), timestamp);
 | |
|     }
 | |
| 
 | |
|     static FrameQueueItem error_marker(DecoderError&& error, Time timestamp)
 | |
|     {
 | |
|         return FrameQueueItem(move(error), timestamp);
 | |
|     }
 | |
| 
 | |
|     bool is_frame() const { return m_data.has<RefPtr<Gfx::Bitmap>>(); }
 | |
|     RefPtr<Gfx::Bitmap> bitmap() const { return m_data.get<RefPtr<Gfx::Bitmap>>(); }
 | |
|     Time timestamp() const { return m_timestamp; }
 | |
| 
 | |
|     bool is_error() const { return m_data.has<DecoderError>(); }
 | |
|     DecoderError const& error() const { return m_data.get<DecoderError>(); }
 | |
|     DecoderError release_error()
 | |
|     {
 | |
|         auto error = move(m_data.get<DecoderError>());
 | |
|         m_data.set(Empty());
 | |
|         return error;
 | |
|     }
 | |
| 
 | |
|     DeprecatedString debug_string() const
 | |
|     {
 | |
|         if (is_error())
 | |
|             return DeprecatedString::formatted("{} at {}ms", error().string_literal(), timestamp().to_milliseconds());
 | |
|         return DeprecatedString::formatted("frame at {}ms", timestamp().to_milliseconds());
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     FrameQueueItem(RefPtr<Gfx::Bitmap> bitmap, Time timestamp)
 | |
|         : m_data(move(bitmap))
 | |
|         , m_timestamp(timestamp)
 | |
|     {
 | |
|         VERIFY(m_timestamp != no_timestamp);
 | |
|     }
 | |
| 
 | |
|     FrameQueueItem(DecoderError&& error, Time timestamp)
 | |
|         : m_data(move(error))
 | |
|         , m_timestamp(timestamp)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     Variant<Empty, RefPtr<Gfx::Bitmap>, DecoderError> m_data;
 | |
|     Time m_timestamp;
 | |
| };
 | |
| 
 | |
| static constexpr size_t FRAME_BUFFER_COUNT = 4;
 | |
| using VideoFrameQueue = Queue<FrameQueueItem, FRAME_BUFFER_COUNT>;
 | |
| 
 | |
| class PlaybackManager {
 | |
| public:
 | |
|     enum class SeekMode {
 | |
|         Accurate,
 | |
|         Fast,
 | |
|     };
 | |
| 
 | |
|     static constexpr SeekMode DEFAULT_SEEK_MODE = SeekMode::Accurate;
 | |
| 
 | |
|     static DecoderErrorOr<NonnullOwnPtr<PlaybackManager>> from_file(Core::Object& event_handler, StringView file);
 | |
| 
 | |
|     PlaybackManager(Core::Object& event_handler, NonnullOwnPtr<Demuxer>& demuxer, Track video_track, NonnullOwnPtr<VideoDecoder>&& decoder);
 | |
| 
 | |
|     void resume_playback();
 | |
|     void pause_playback();
 | |
|     void restart_playback();
 | |
|     void seek_to_timestamp(Time, SeekMode = DEFAULT_SEEK_MODE);
 | |
|     bool is_playing() const
 | |
|     {
 | |
|         return m_playback_handler->is_playing();
 | |
|     }
 | |
| 
 | |
|     u64 number_of_skipped_frames() const { return m_skipped_frames; }
 | |
| 
 | |
|     Time current_playback_time();
 | |
|     Time duration();
 | |
| 
 | |
|     Function<void(NonnullRefPtr<Gfx::Bitmap>, Time)> on_frame_present;
 | |
| 
 | |
| private:
 | |
|     class PlaybackStateHandler;
 | |
|     // Abstract class to allow resuming play/pause after the state is completed.
 | |
|     class ResumingStateHandler;
 | |
|     class StartingStateHandler;
 | |
|     class PlayingStateHandler;
 | |
|     class PausedStateHandler;
 | |
|     class BufferingStateHandler;
 | |
|     class SeekingStateHandler;
 | |
|     class StoppedStateHandler;
 | |
| 
 | |
|     void start_timer(int milliseconds);
 | |
|     void timer_callback();
 | |
|     Optional<Time> seek_demuxer_to_most_recent_keyframe(Time timestamp, Optional<Time> earliest_available_sample = OptionalNone());
 | |
| 
 | |
|     bool decode_and_queue_one_sample();
 | |
|     void on_decode_timer();
 | |
| 
 | |
|     void dispatch_decoder_error(DecoderError error);
 | |
|     void dispatch_new_frame(RefPtr<Gfx::Bitmap> frame);
 | |
|     // Returns whether we changed playback states. If so, any PlaybackStateHandler processing must cease.
 | |
|     [[nodiscard]] bool dispatch_frame_queue_item(FrameQueueItem&&);
 | |
|     void dispatch_state_change();
 | |
|     void dispatch_fatal_error(Error);
 | |
| 
 | |
|     Core::Object& m_event_handler;
 | |
|     Core::EventLoop& m_main_loop;
 | |
| 
 | |
|     Time m_last_present_in_media_time = Time::zero();
 | |
| 
 | |
|     NonnullOwnPtr<Demuxer> m_demuxer;
 | |
|     Track m_selected_video_track;
 | |
|     NonnullOwnPtr<VideoDecoder> m_decoder;
 | |
| 
 | |
|     NonnullOwnPtr<VideoFrameQueue> m_frame_queue;
 | |
| 
 | |
|     RefPtr<Core::Timer> m_present_timer;
 | |
|     unsigned m_decoding_buffer_time_ms = 16;
 | |
| 
 | |
|     RefPtr<Core::Timer> m_decode_timer;
 | |
| 
 | |
|     NonnullOwnPtr<PlaybackStateHandler> m_playback_handler;
 | |
|     Optional<FrameQueueItem> m_next_frame;
 | |
| 
 | |
|     u64 m_skipped_frames;
 | |
| 
 | |
|     // This is a nested class to allow private access.
 | |
|     class PlaybackStateHandler {
 | |
|     public:
 | |
|         PlaybackStateHandler(PlaybackManager& manager)
 | |
|             : m_manager(manager)
 | |
|         {
 | |
|         }
 | |
|         virtual ~PlaybackStateHandler() = default;
 | |
|         virtual StringView name() = 0;
 | |
| 
 | |
|         virtual ErrorOr<void> on_enter() { return {}; }
 | |
| 
 | |
|         virtual ErrorOr<void> play() { return {}; };
 | |
|         virtual bool is_playing() = 0;
 | |
|         virtual ErrorOr<void> pause() { return {}; };
 | |
|         virtual ErrorOr<void> buffer() { return {}; };
 | |
|         virtual ErrorOr<void> seek(Time target_timestamp, SeekMode);
 | |
|         virtual ErrorOr<void> stop();
 | |
| 
 | |
|         virtual Time current_time() const;
 | |
| 
 | |
|         virtual ErrorOr<void> on_timer_callback() { return {}; };
 | |
|         virtual ErrorOr<void> on_buffer_filled() { return {}; };
 | |
| 
 | |
|     protected:
 | |
|         template<class T, class... Args>
 | |
|         ErrorOr<void> replace_handler_and_delete_this(Args... args);
 | |
| 
 | |
|         PlaybackManager& manager() const;
 | |
| 
 | |
|         PlaybackManager& manager()
 | |
|         {
 | |
|             return const_cast<PlaybackManager&>(const_cast<PlaybackStateHandler const*>(this)->manager());
 | |
|         }
 | |
| 
 | |
|     private:
 | |
|         PlaybackManager& m_manager;
 | |
| #if PLAYBACK_MANAGER_DEBUG
 | |
|         bool m_has_exited { false };
 | |
| #endif
 | |
|     };
 | |
| };
 | |
| 
 | |
| enum EventType : unsigned {
 | |
|     DecoderErrorOccurred = (('v' << 2) | ('i' << 1) | 'd') << 4,
 | |
|     VideoFramePresent,
 | |
|     PlaybackStateChange,
 | |
|     FatalPlaybackError,
 | |
| };
 | |
| 
 | |
| class DecoderErrorEvent : public Core::Event {
 | |
| public:
 | |
|     explicit DecoderErrorEvent(DecoderError error)
 | |
|         : Core::Event(DecoderErrorOccurred)
 | |
|         , m_error(move(error))
 | |
|     {
 | |
|     }
 | |
|     virtual ~DecoderErrorEvent() = default;
 | |
| 
 | |
|     DecoderError const& error() { return m_error; }
 | |
| 
 | |
| private:
 | |
|     DecoderError m_error;
 | |
| };
 | |
| 
 | |
| class VideoFramePresentEvent : public Core::Event {
 | |
| public:
 | |
|     VideoFramePresentEvent() = default;
 | |
|     explicit VideoFramePresentEvent(RefPtr<Gfx::Bitmap> frame)
 | |
|         : Core::Event(VideoFramePresent)
 | |
|         , m_frame(move(frame))
 | |
|     {
 | |
|     }
 | |
|     virtual ~VideoFramePresentEvent() = default;
 | |
| 
 | |
|     RefPtr<Gfx::Bitmap> frame() { return m_frame; }
 | |
| 
 | |
| private:
 | |
|     RefPtr<Gfx::Bitmap> m_frame;
 | |
| };
 | |
| 
 | |
| class PlaybackStateChangeEvent : public Core::Event {
 | |
| public:
 | |
|     explicit PlaybackStateChangeEvent()
 | |
|         : Core::Event(PlaybackStateChange)
 | |
|     {
 | |
|     }
 | |
|     virtual ~PlaybackStateChangeEvent() = default;
 | |
| };
 | |
| 
 | |
| class FatalPlaybackErrorEvent : public Core::Event {
 | |
| public:
 | |
|     explicit FatalPlaybackErrorEvent(Error error)
 | |
|         : Core::Event(FatalPlaybackError)
 | |
|         , m_error(move(error))
 | |
|     {
 | |
|     }
 | |
|     virtual ~FatalPlaybackErrorEvent() = default;
 | |
|     Error const& error() { return m_error; }
 | |
| 
 | |
| private:
 | |
|     Error m_error;
 | |
| };
 | |
| 
 | |
| }
 |