2017-07-13 11:26:03 +03:00
|
|
|
/*
|
|
|
|
* CVideoHandler.cpp, part of VCMI engine
|
|
|
|
*
|
|
|
|
* Authors: listed in file AUTHORS in main folder
|
|
|
|
*
|
|
|
|
* License: GNU General Public License v2.0 or later
|
|
|
|
* Full text of license available in license.txt file, in main folder
|
|
|
|
*
|
|
|
|
*/
|
2011-12-13 21:23:17 +00:00
|
|
|
#include "StdInc.h"
|
|
|
|
#include <SDL.h>
|
2007-08-08 19:28:56 +00:00
|
|
|
#include "CVideoHandler.h"
|
2011-12-13 21:23:17 +00:00
|
|
|
|
2013-04-07 11:52:07 +00:00
|
|
|
#include "gui/CGuiHandler.h"
|
|
|
|
#include "gui/SDL_Extensions.h"
|
2011-04-07 17:54:08 +00:00
|
|
|
#include "CPlayerInterface.h"
|
2013-07-28 14:49:50 +00:00
|
|
|
#include "../lib/filesystem/Filesystem.h"
|
2009-06-27 23:33:59 +00:00
|
|
|
|
2012-01-12 15:23:00 +00:00
|
|
|
extern CGuiHandler GH; //global gui handler
|
|
|
|
|
2014-05-24 16:52:43 +03:00
|
|
|
#ifndef DISABLE_VIDEO
|
2009-07-27 17:55:56 +00:00
|
|
|
//reads events and returns true on key down
|
2009-06-27 23:33:59 +00:00
|
|
|
static bool keyDown()
|
|
|
|
{
|
|
|
|
SDL_Event ev;
|
|
|
|
while(SDL_PollEvent(&ev))
|
|
|
|
{
|
|
|
|
if(ev.type == SDL_KEYDOWN || ev.type == SDL_MOUSEBUTTONDOWN)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2014-01-06 19:12:21 +00:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma comment(lib, "avcodec.lib")
|
|
|
|
#pragma comment(lib, "avutil.lib")
|
|
|
|
#pragma comment(lib, "avformat.lib")
|
|
|
|
#pragma comment(lib, "swscale.lib")
|
|
|
|
#endif // _MSC_VER
|
|
|
|
|
2012-08-21 17:37:06 +00:00
|
|
|
// Define a set of functions to read data
|
2017-07-20 02:50:47 +03:00
|
|
|
static int lodRead(void * opaque, uint8_t * buf, int size)
|
2012-08-21 17:37:06 +00:00
|
|
|
{
|
|
|
|
auto video = reinterpret_cast<CVideoPlayer *>(opaque);
|
|
|
|
|
2013-02-05 23:11:48 +00:00
|
|
|
return video->data->read(buf, size);
|
2009-06-20 01:34:26 +00:00
|
|
|
}
|
|
|
|
|
2012-08-21 17:37:06 +00:00
|
|
|
static si64 lodSeek(void * opaque, si64 pos, int whence)
|
|
|
|
{
|
|
|
|
auto video = reinterpret_cast<CVideoPlayer *>(opaque);
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(whence & AVSEEK_SIZE)
|
2013-02-05 23:11:48 +00:00
|
|
|
return video->data->getSize();
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2013-02-05 23:11:48 +00:00
|
|
|
return video->data->seek(pos);
|
2009-06-20 01:34:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CVideoPlayer::CVideoPlayer()
|
|
|
|
{
|
2016-11-27 17:48:18 +03:00
|
|
|
stream = -1;
|
2013-06-26 11:18:27 +00:00
|
|
|
format = nullptr;
|
2016-11-27 17:48:18 +03:00
|
|
|
codecContext = nullptr;
|
2013-06-26 11:18:27 +00:00
|
|
|
codec = nullptr;
|
2016-11-27 17:48:18 +03:00
|
|
|
frame = nullptr;
|
2013-06-26 11:18:27 +00:00
|
|
|
sws = nullptr;
|
2016-11-27 17:48:18 +03:00
|
|
|
context = nullptr;
|
2014-05-24 20:12:07 +03:00
|
|
|
texture = nullptr;
|
2013-06-26 11:18:27 +00:00
|
|
|
dest = nullptr;
|
2017-07-20 02:50:47 +03:00
|
|
|
destRect = genRect(0, 0, 0, 0);
|
|
|
|
pos = genRect(0, 0, 0, 0);
|
2016-11-27 17:48:18 +03:00
|
|
|
refreshWait = 0;
|
|
|
|
refreshCount = 0;
|
|
|
|
doLoop = false;
|
2009-06-20 01:34:26 +00:00
|
|
|
|
|
|
|
// Register codecs. TODO: May be overkill. Should call a
|
|
|
|
// combination of av_register_input_format() /
|
|
|
|
// av_register_output_format() / av_register_protocol() instead.
|
|
|
|
av_register_all();
|
|
|
|
}
|
|
|
|
|
2017-07-15 14:08:20 +03:00
|
|
|
bool CVideoPlayer::open(std::string fname, bool scale)
|
2011-08-08 14:20:22 +00:00
|
|
|
{
|
|
|
|
return open(fname, true, false);
|
|
|
|
}
|
|
|
|
|
2009-06-27 23:33:59 +00:00
|
|
|
// loop = to loop through the video
|
|
|
|
// useOverlay = directly write to the screen.
|
2017-07-15 14:08:20 +03:00
|
|
|
bool CVideoPlayer::open(std::string fname, bool loop, bool useOverlay, bool scale)
|
2009-06-20 01:34:26 +00:00
|
|
|
{
|
|
|
|
close();
|
2012-09-15 19:16:16 +00:00
|
|
|
|
2011-04-17 18:02:13 +00:00
|
|
|
this->fname = fname;
|
2009-06-27 23:33:59 +00:00
|
|
|
refreshWait = 3;
|
|
|
|
refreshCount = -1;
|
|
|
|
doLoop = loop;
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2012-08-11 11:34:49 +00:00
|
|
|
ResourceID resource(std::string("Video/") + fname, EResType::VIDEO);
|
2012-08-08 12:17:25 +00:00
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(!CResourceHandler::get()->existsResource(resource))
|
2013-02-27 15:46:14 +00:00
|
|
|
{
|
2014-05-24 20:12:07 +03:00
|
|
|
logGlobal->errorStream() << "Error: video " << resource.getName() << " was not found";
|
2012-08-08 12:17:25 +00:00
|
|
|
return false;
|
2013-02-27 15:46:14 +00:00
|
|
|
}
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2013-02-05 23:11:48 +00:00
|
|
|
data = CResourceHandler::get()->load(resource);
|
2012-08-21 17:37:06 +00:00
|
|
|
|
|
|
|
static const int BUFFER_SIZE = 4096;
|
2013-02-05 23:11:48 +00:00
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
unsigned char * buffer = (unsigned char *)av_malloc(BUFFER_SIZE); // will be freed by ffmpeg
|
|
|
|
context = avio_alloc_context(buffer, BUFFER_SIZE, 0, (void *)this, lodRead, nullptr, lodSeek);
|
2012-08-21 17:37:06 +00:00
|
|
|
|
|
|
|
format = avformat_alloc_context();
|
|
|
|
format->pb = context;
|
|
|
|
// filename is not needed - file was already open and stored in this->data;
|
|
|
|
int avfopen = avformat_open_input(&format, "dummyFilename", nullptr, nullptr);
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(avfopen != 0)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2009-06-20 01:34:26 +00:00
|
|
|
// Retrieve stream information
|
2017-07-20 02:50:47 +03:00
|
|
|
if(avformat_find_stream_info(format, nullptr) < 0)
|
2009-06-20 01:34:26 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// Find the first video stream
|
|
|
|
stream = -1;
|
2017-07-20 02:50:47 +03:00
|
|
|
for(ui32 i = 0; i < format->nb_streams; i++)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2017-07-20 02:50:47 +03:00
|
|
|
if(format->streams[i]->codec->codec_type == AVMEDIA_TYPE_VIDEO)
|
2011-07-05 19:06:47 +00:00
|
|
|
{
|
2009-06-20 01:34:26 +00:00
|
|
|
stream = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(stream < 0)
|
2009-06-20 01:34:26 +00:00
|
|
|
// No video stream in that file
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Get a pointer to the codec context for the video stream
|
|
|
|
codecContext = format->streams[stream]->codec;
|
|
|
|
|
|
|
|
// Find the decoder for the video stream
|
|
|
|
codec = avcodec_find_decoder(codecContext->codec_id);
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(codec == nullptr)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2009-06-20 01:34:26 +00:00
|
|
|
// Unsupported codec
|
|
|
|
return false;
|
|
|
|
}
|
2012-09-15 19:16:16 +00:00
|
|
|
|
2009-06-20 01:34:26 +00:00
|
|
|
// Open codec
|
2017-07-20 02:50:47 +03:00
|
|
|
if(avcodec_open2(codecContext, codec, nullptr) < 0)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2009-06-20 01:34:26 +00:00
|
|
|
// Could not open codec
|
2013-06-26 11:18:27 +00:00
|
|
|
codec = nullptr;
|
2009-06-20 01:34:26 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// Allocate video frame
|
2015-11-11 09:41:28 +01:00
|
|
|
frame = av_frame_alloc();
|
2016-11-27 17:48:18 +03:00
|
|
|
|
2014-07-08 18:20:22 +04:00
|
|
|
//setup scaling
|
2014-07-10 14:18:21 +04:00
|
|
|
if(scale)
|
2014-07-08 18:20:22 +04:00
|
|
|
{
|
2016-11-27 17:48:18 +03:00
|
|
|
pos.w = screen->w;
|
2014-07-08 18:20:22 +04:00
|
|
|
pos.h = screen->h;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-07-20 02:50:47 +03:00
|
|
|
pos.w = codecContext->width;
|
2016-11-27 17:48:18 +03:00
|
|
|
pos.h = codecContext->height;
|
2014-07-08 18:20:22 +04:00
|
|
|
}
|
2009-06-20 01:34:26 +00:00
|
|
|
|
|
|
|
// Allocate a place to put our YUV image on that screen
|
2017-07-20 02:50:47 +03:00
|
|
|
if(useOverlay)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2017-07-20 02:50:47 +03:00
|
|
|
texture = SDL_CreateTexture(mainRenderer, SDL_PIXELFORMAT_IYUV, SDL_TEXTUREACCESS_STATIC, pos.w, pos.h);
|
2012-05-13 15:04:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-07-08 18:20:22 +04:00
|
|
|
dest = CSDL_Ext::newSurface(pos.w, pos.h);
|
2009-06-27 23:33:59 +00:00
|
|
|
destRect.x = destRect.y = 0;
|
2014-07-08 18:20:22 +04:00
|
|
|
destRect.w = pos.w;
|
|
|
|
destRect.h = pos.h;
|
2009-06-27 23:33:59 +00:00
|
|
|
}
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(texture == nullptr && dest == nullptr)
|
2014-05-24 20:12:07 +03:00
|
|
|
return false;
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(texture)
|
2014-05-24 20:12:07 +03:00
|
|
|
{ // Convert the image into YUV format that SDL uses
|
2016-11-27 17:48:18 +03:00
|
|
|
sws = sws_getContext(codecContext->width, codecContext->height, codecContext->pix_fmt,
|
2017-07-20 02:50:47 +03:00
|
|
|
pos.w, pos.h,
|
|
|
|
AV_PIX_FMT_YUV420P,
|
|
|
|
SWS_BICUBIC, nullptr, nullptr, nullptr);
|
2012-05-13 15:04:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-11-05 10:53:44 +01:00
|
|
|
AVPixelFormat screenFormat = AV_PIX_FMT_NONE;
|
2017-07-20 02:50:47 +03:00
|
|
|
if(screen->format->Bshift > screen->format->Rshift)
|
2010-08-26 16:55:42 +00:00
|
|
|
{
|
2012-12-01 06:30:52 +00:00
|
|
|
// this a BGR surface
|
2017-07-20 02:50:47 +03:00
|
|
|
switch(screen->format->BytesPerPixel)
|
2012-12-01 06:30:52 +00:00
|
|
|
{
|
2017-07-20 02:50:47 +03:00
|
|
|
case 2:
|
|
|
|
screenFormat = AV_PIX_FMT_BGR565;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
screenFormat = AV_PIX_FMT_BGR24;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
screenFormat = AV_PIX_FMT_BGR32;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
2012-12-01 06:30:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// this a RGB surface
|
2017-07-20 02:50:47 +03:00
|
|
|
switch(screen->format->BytesPerPixel)
|
2012-12-01 06:30:52 +00:00
|
|
|
{
|
2017-07-20 02:50:47 +03:00
|
|
|
case 2:
|
|
|
|
screenFormat = AV_PIX_FMT_RGB565;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
screenFormat = AV_PIX_FMT_RGB24;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
screenFormat = AV_PIX_FMT_RGB32;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
2012-12-01 06:30:52 +00:00
|
|
|
}
|
2010-08-26 16:55:42 +00:00
|
|
|
}
|
|
|
|
|
2016-11-27 17:48:18 +03:00
|
|
|
sws = sws_getContext(codecContext->width, codecContext->height, codecContext->pix_fmt,
|
2017-07-20 02:50:47 +03:00
|
|
|
pos.w, pos.h, screenFormat,
|
|
|
|
SWS_BICUBIC, nullptr, nullptr, nullptr);
|
2009-06-27 23:33:59 +00:00
|
|
|
}
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(sws == nullptr)
|
2009-06-20 01:34:26 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-06-27 23:33:59 +00:00
|
|
|
// Read the next frame. Return false on error/end of file.
|
2009-06-20 01:34:26 +00:00
|
|
|
bool CVideoPlayer::nextFrame()
|
|
|
|
{
|
|
|
|
AVPacket packet;
|
|
|
|
int frameFinished = 0;
|
2009-06-27 23:33:59 +00:00
|
|
|
bool gotError = false;
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(sws == nullptr)
|
2009-06-27 23:33:59 +00:00
|
|
|
return false;
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2012-05-13 15:04:21 +00:00
|
|
|
while(!frameFinished)
|
|
|
|
{
|
2009-06-27 23:33:59 +00:00
|
|
|
int ret = av_read_frame(format, &packet);
|
2017-07-20 02:50:47 +03:00
|
|
|
if(ret < 0)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2009-06-27 23:33:59 +00:00
|
|
|
// Error. It's probably an end of file.
|
2017-07-20 02:50:47 +03:00
|
|
|
if(doLoop && !gotError)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2009-06-27 23:33:59 +00:00
|
|
|
// Rewind
|
2017-07-20 02:50:47 +03:00
|
|
|
if(av_seek_frame(format, stream, 0, AVSEEK_FLAG_BYTE) < 0)
|
2009-06-27 23:33:59 +00:00
|
|
|
break;
|
|
|
|
gotError = true;
|
2012-05-13 15:04:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-06-27 23:33:59 +00:00
|
|
|
break;
|
|
|
|
}
|
2012-05-13 15:04:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-06-27 23:33:59 +00:00
|
|
|
// Is this a packet from the video stream?
|
2017-07-20 02:50:47 +03:00
|
|
|
if(packet.stream_index == stream)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2009-06-27 23:33:59 +00:00
|
|
|
// Decode video frame
|
2012-07-02 21:51:48 +00:00
|
|
|
|
2011-03-19 22:27:05 +00:00
|
|
|
avcodec_decode_video2(codecContext, frame, &frameFinished, &packet);
|
2009-06-27 23:33:59 +00:00
|
|
|
|
|
|
|
// Did we get a video frame?
|
2017-07-20 02:50:47 +03:00
|
|
|
if(frameFinished)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2009-06-27 23:33:59 +00:00
|
|
|
AVPicture pict;
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(texture)
|
|
|
|
{
|
2014-07-08 18:20:22 +04:00
|
|
|
avpicture_alloc(&pict, AV_PIX_FMT_YUV420P, pos.w, pos.h);
|
2014-05-24 20:12:07 +03:00
|
|
|
|
|
|
|
sws_scale(sws, frame->data, frame->linesize,
|
2017-07-20 02:50:47 +03:00
|
|
|
0, codecContext->height, pict.data, pict.linesize);
|
2014-05-24 20:12:07 +03:00
|
|
|
|
|
|
|
SDL_UpdateYUVTexture(texture, NULL, pict.data[0], pict.linesize[0],
|
2017-07-20 02:50:47 +03:00
|
|
|
pict.data[1], pict.linesize[1],
|
|
|
|
pict.data[2], pict.linesize[2]);
|
2014-05-24 20:12:07 +03:00
|
|
|
avpicture_free(&pict);
|
2012-05-13 15:04:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-12-13 21:23:17 +00:00
|
|
|
pict.data[0] = (ui8 *)dest->pixels;
|
2009-06-27 23:33:59 +00:00
|
|
|
pict.linesize[0] = dest->pitch;
|
|
|
|
|
|
|
|
sws_scale(sws, frame->data, frame->linesize,
|
2017-07-20 02:50:47 +03:00
|
|
|
0, codecContext->height, pict.data, pict.linesize);
|
2009-06-27 23:33:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2009-06-27 23:33:59 +00:00
|
|
|
av_free_packet(&packet);
|
|
|
|
}
|
|
|
|
}
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2009-06-27 23:33:59 +00:00
|
|
|
return frameFinished != 0;
|
|
|
|
}
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
void CVideoPlayer::show(int x, int y, SDL_Surface * dst, bool update)
|
2009-06-27 23:33:59 +00:00
|
|
|
{
|
2017-07-20 02:50:47 +03:00
|
|
|
if(sws == nullptr)
|
2009-06-27 23:33:59 +00:00
|
|
|
return;
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2009-06-27 23:33:59 +00:00
|
|
|
pos.x = x;
|
|
|
|
pos.y = y;
|
2010-08-23 19:00:14 +00:00
|
|
|
CSDL_Ext::blitSurface(dest, &destRect, dst, &pos);
|
2012-09-15 19:16:16 +00:00
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(update)
|
2009-06-27 23:33:59 +00:00
|
|
|
SDL_UpdateRect(dst, pos.x, pos.y, pos.w, pos.h);
|
|
|
|
}
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
void CVideoPlayer::redraw(int x, int y, SDL_Surface * dst, bool update)
|
2009-06-27 23:33:59 +00:00
|
|
|
{
|
|
|
|
show(x, y, dst, update);
|
|
|
|
}
|
2009-06-20 01:34:26 +00:00
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
void CVideoPlayer::update(int x, int y, SDL_Surface * dst, bool forceRedraw, bool update)
|
2009-06-27 23:33:59 +00:00
|
|
|
{
|
2017-07-20 02:50:47 +03:00
|
|
|
if(sws == nullptr)
|
2009-06-27 23:33:59 +00:00
|
|
|
return;
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(refreshCount <= 0)
|
2009-06-27 23:33:59 +00:00
|
|
|
{
|
|
|
|
refreshCount = refreshWait;
|
2017-07-20 02:50:47 +03:00
|
|
|
if(nextFrame())
|
|
|
|
show(x, y, dst, update);
|
2012-05-13 15:04:21 +00:00
|
|
|
else
|
|
|
|
{
|
2011-04-17 18:02:13 +00:00
|
|
|
open(fname);
|
2012-09-15 19:16:16 +00:00
|
|
|
nextFrame();
|
|
|
|
|
2011-04-26 13:30:29 +00:00
|
|
|
// The y position is wrong at the first frame.
|
|
|
|
// Note: either the windows player or the linux player is
|
2012-09-15 19:16:16 +00:00
|
|
|
// broken. Compensate here until the bug is found.
|
2011-04-26 13:30:29 +00:00
|
|
|
show(x, y--, dst, update);
|
|
|
|
}
|
2012-09-15 19:16:16 +00:00
|
|
|
}
|
2012-05-13 15:04:21 +00:00
|
|
|
else
|
|
|
|
{
|
2009-06-27 23:33:59 +00:00
|
|
|
redraw(x, y, dst, update);
|
2009-06-20 01:34:26 +00:00
|
|
|
}
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
refreshCount--;
|
2009-06-20 01:34:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CVideoPlayer::close()
|
|
|
|
{
|
2012-09-15 19:16:16 +00:00
|
|
|
fname = "";
|
2017-07-20 02:50:47 +03:00
|
|
|
if(sws)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2009-06-20 01:34:26 +00:00
|
|
|
sws_freeContext(sws);
|
2013-06-26 11:18:27 +00:00
|
|
|
sws = nullptr;
|
2009-06-20 01:34:26 +00:00
|
|
|
}
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(texture)
|
2014-05-24 20:12:07 +03:00
|
|
|
{
|
|
|
|
SDL_DestroyTexture(texture);
|
|
|
|
texture = nullptr;
|
|
|
|
}
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(dest)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2009-06-27 23:33:59 +00:00
|
|
|
SDL_FreeSurface(dest);
|
2013-06-26 11:18:27 +00:00
|
|
|
dest = nullptr;
|
2009-06-27 23:33:59 +00:00
|
|
|
}
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(frame)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2017-07-20 02:50:47 +03:00
|
|
|
av_frame_free(&frame); //will be set to null
|
2009-06-20 01:34:26 +00:00
|
|
|
}
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(codec)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
2009-06-20 01:34:26 +00:00
|
|
|
avcodec_close(codecContext);
|
2013-06-26 11:18:27 +00:00
|
|
|
codec = nullptr;
|
|
|
|
codecContext = nullptr;
|
2009-06-20 01:34:26 +00:00
|
|
|
}
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(format)
|
2012-05-13 15:04:21 +00:00
|
|
|
{
|
|
|
|
avformat_close_input(&format);
|
2009-06-20 01:34:26 +00:00
|
|
|
}
|
2012-08-21 17:37:06 +00:00
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
if(context)
|
2012-08-21 17:37:06 +00:00
|
|
|
{
|
|
|
|
av_free(context);
|
|
|
|
context = nullptr;
|
|
|
|
}
|
2009-06-20 01:34:26 +00:00
|
|
|
}
|
2009-06-27 23:33:59 +00:00
|
|
|
|
|
|
|
// Plays a video. Only works for overlays.
|
2017-07-20 02:50:47 +03:00
|
|
|
bool CVideoPlayer::playVideo(int x, int y, SDL_Surface * dst, bool stopOnKey)
|
2009-06-27 23:33:59 +00:00
|
|
|
{
|
|
|
|
// Note: either the windows player or the linux player is
|
|
|
|
// broken. Compensate here until the bug is found.
|
|
|
|
y--;
|
|
|
|
|
|
|
|
pos.x = x;
|
|
|
|
pos.y = y;
|
|
|
|
|
2012-05-13 15:04:21 +00:00
|
|
|
while(nextFrame())
|
|
|
|
{
|
2012-09-15 19:16:16 +00:00
|
|
|
|
2009-06-27 23:33:59 +00:00
|
|
|
if(stopOnKey && keyDown())
|
|
|
|
return false;
|
|
|
|
|
2014-07-08 18:20:22 +04:00
|
|
|
SDL_RenderCopy(mainRenderer, texture, NULL, &pos);
|
2014-05-24 20:12:07 +03:00
|
|
|
SDL_RenderPresent(mainRenderer);
|
2009-06-27 23:33:59 +00:00
|
|
|
|
|
|
|
// Wait 3 frames
|
2011-04-07 02:16:32 +00:00
|
|
|
GH.mainFPSmng->framerateDelay();
|
|
|
|
GH.mainFPSmng->framerateDelay();
|
|
|
|
GH.mainFPSmng->framerateDelay();
|
2009-06-27 23:33:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-07-20 02:50:47 +03:00
|
|
|
bool CVideoPlayer::openAndPlayVideo(std::string name, int x, int y, SDL_Surface * dst, bool stopOnKey, bool scale)
|
2009-06-27 23:33:59 +00:00
|
|
|
{
|
2014-07-10 14:18:21 +04:00
|
|
|
open(name, false, true, scale);
|
2009-06-27 23:33:59 +00:00
|
|
|
bool ret = playVideo(x, y, dst, stopOnKey);
|
|
|
|
close();
|
|
|
|
return ret;
|
|
|
|
}
|
2012-09-15 19:16:16 +00:00
|
|
|
|
2009-06-20 01:34:26 +00:00
|
|
|
CVideoPlayer::~CVideoPlayer()
|
|
|
|
{
|
|
|
|
close();
|
|
|
|
}
|
|
|
|
|
2011-12-13 21:23:17 +00:00
|
|
|
#endif
|