#include "ffcpp/Player.h" #include #include #include #include #include namespace ff = ffcpp; #define WINDOW_WIDTH 640 #define WINDOW_HEIGHT 480 class SDLWindow: public ff::IVideoSink { private: template using SDLUniquePtr = std::unique_ptr; using SDLWindowPtr = SDLUniquePtr; using SDLRendererPtr = SDLUniquePtr; using SDLTexturePtr = SDLUniquePtr; private: SDLWindowPtr _wnd; SDLRendererPtr _renderer; SDLTexturePtr _texture; SDL_AudioSpec _audioSpec; SDL_AudioDeviceID _aDevId; std::packaged_task _renderTask; public: SDLWindow(): _wnd(nullptr, SDL_DestroyWindow), _renderer(nullptr, SDL_DestroyRenderer), _texture(nullptr, SDL_DestroyTexture) { int res = SDL_Init(SDL_INIT_VIDEO | SDL_INIT_AUDIO | SDL_INIT_TIMER); if(res < 0) throw std::runtime_error("Error initializing SDL"); _wnd.reset(SDL_CreateWindow("ffPlayer", SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED, WINDOW_WIDTH, WINDOW_HEIGHT, SDL_WINDOW_OPENGL)); if(!_wnd) throw std::runtime_error("Error creating SDL window"); _renderer.reset(SDL_CreateRenderer(_wnd.get(), -1, 0)); if(!_renderer) throw std::runtime_error("Error creating SDL renderer"); _texture.reset(SDL_CreateTexture(_renderer.get(), SDL_PIXELFORMAT_IYUV, SDL_TEXTUREACCESS_STREAMING, WINDOW_WIDTH, WINDOW_HEIGHT)); if(!_texture) throw std::runtime_error("Error creating SDL texture"); SDL_AudioSpec want; SDL_zero(want); want.freq = 44100; want.format = AUDIO_S16; want.channels = 2; want.samples = 4096; want.callback = SDLWindow::audioCallback; _aDevId = SDL_OpenAudioDevice(nullptr, 0, &want, &_audioSpec, SDL_AUDIO_ALLOW_ANY_CHANGE); if(_aDevId == 0) throw std::runtime_error("Error opening audio device"); } void handleEvents() { SDL_Event event; while(true) { SDL_WaitEvent(&event); switch(event.type) { case SDL_QUIT: return; case SDL_USEREVENT: { _renderTask(); break; } } } } private: static void audioCallback(void* userdata, Uint8* stream, int len) { } public: virtual AVPixelFormat getPixelFormat() const noexcept override { return AV_PIX_FMT_YUV420P; } virtual void drawFrame(void* pixelsData, int pitch) override { std::cout << "drawing frame" << std::endl; SDL_UpdateTexture(_texture.get(), nullptr, pixelsData, pitch); SDL_RenderClear(_renderer.get()); SDL_RenderCopy(_renderer.get(), _texture.get(), nullptr, nullptr); SDL_RenderPresent(_renderer.get()); } virtual void drawPlanarYUVFrame(const void *yPlane, const void *uPlane, const void *vPlane, int yPitch, int uPitch, int vPitch) override { _renderTask = std::packaged_task([=]{ SDL_UpdateYUVTexture(_texture.get(), nullptr, (const uint8_t*)yPlane, yPitch, (const uint8_t*)uPlane, uPitch, (const uint8_t*)vPlane, vPitch); SDL_RenderClear(_renderer.get()); SDL_RenderCopy(_renderer.get(), _texture.get(), nullptr, nullptr); SDL_RenderPresent(_renderer.get()); }); auto future = _renderTask.get_future(); SDL_Event event; event.type = SDL_USEREVENT; int res = SDL_PushEvent(&event); future.get(); } }; int main(int argc, char** argv) { auto wnd = std::make_shared(); ff::Player player(wnd); player.setMedia(argv[1]); player.setVideoSize(WINDOW_WIDTH, WINDOW_HEIGHT); player.play(); wnd->handleEvents(); return 0; }