ca91097b8f
Rather than setting the default script resolution to 640x480, leave it unset in new files. When video is opened, if the user has not already manually set a script resolution, automatically use the video's resolution as the script resolution. If the user has and the set resolution is not a multiple of the video resolution, use the old prompt for updating it. This eliminates the confusing prompt about mismatched resolutions when opening video with a new file while hopefully resulting in users using the correct resolution more often. Originally committed to SVN as r5921.
600 lines
18 KiB
C++
600 lines
18 KiB
C++
// Copyright (c) 2005-2007, Rodrigo Braz Monteiro
|
|
// All rights reserved.
|
|
//
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions are met:
|
|
//
|
|
// * Redistributions of source code must retain the above copyright notice,
|
|
// this list of conditions and the following disclaimer.
|
|
// * Redistributions in binary form must reproduce the above copyright notice,
|
|
// this list of conditions and the following disclaimer in the documentation
|
|
// and/or other materials provided with the distribution.
|
|
// * Neither the name of the Aegisub Group nor the names of its contributors
|
|
// may be used to endorse or promote products derived from this software
|
|
// without specific prior written permission.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
// POSSIBILITY OF SUCH DAMAGE.
|
|
//
|
|
// Aegisub Project http://www.aegisub.org/
|
|
//
|
|
// $Id$
|
|
|
|
/// @file video_context.cpp
|
|
/// @brief Keep track of loaded video
|
|
/// @ingroup video
|
|
///
|
|
|
|
#include "config.h"
|
|
|
|
#ifndef AGI_PRE
|
|
#include <string.h>
|
|
|
|
#include <wx/clipbrd.h>
|
|
#include <wx/config.h>
|
|
#include <wx/filename.h>
|
|
#include <wx/image.h>
|
|
#include <wx/msgdlg.h>
|
|
#endif
|
|
|
|
#include <libaegisub/access.h>
|
|
#include <libaegisub/keyframe.h>
|
|
|
|
#include "ass_dialogue.h"
|
|
#include "ass_file.h"
|
|
#include "ass_style.h"
|
|
#include "ass_time.h"
|
|
#include "audio_controller.h"
|
|
#include "compat.h"
|
|
#include "include/aegisub/audio_player.h"
|
|
#include "include/aegisub/audio_provider.h"
|
|
#include "include/aegisub/context.h"
|
|
#include "include/aegisub/video_provider.h"
|
|
#include "main.h"
|
|
#include "mkv_wrap.h"
|
|
#include "selection_controller.h"
|
|
#include "standard_paths.h"
|
|
#include "threaded_frame_source.h"
|
|
#include "utils.h"
|
|
#include "video_context.h"
|
|
#include "video_frame.h"
|
|
|
|
/// IDs
|
|
enum {
|
|
VIDEO_PLAY_TIMER = 1300
|
|
};
|
|
|
|
BEGIN_EVENT_TABLE(VideoContext, wxEvtHandler)
|
|
EVT_TIMER(VIDEO_PLAY_TIMER,VideoContext::OnPlayTimer)
|
|
END_EVENT_TABLE()
|
|
|
|
/// @brief Constructor
|
|
///
|
|
VideoContext::VideoContext()
|
|
: startFrame(-1)
|
|
, endFrame(-1)
|
|
, playNextFrame(-1)
|
|
, nextFrame(-1)
|
|
, isPlaying(false)
|
|
, keepAudioSync(true)
|
|
, frame_n(0)
|
|
, length(0)
|
|
, arValue(1.)
|
|
, arType(0)
|
|
, hasSubtitles(false)
|
|
, playAudioOnStep(OPT_GET("Audio/Plays When Stepping Video"))
|
|
, VFR_Input(videoFPS)
|
|
, VFR_Output(ovrFPS)
|
|
{
|
|
Bind(EVT_VIDEO_ERROR, &VideoContext::OnVideoError, this);
|
|
Bind(EVT_SUBTITLES_ERROR, &VideoContext::OnSubtitlesError, this);
|
|
|
|
OPT_SUB("Subtitle/Provider", &VideoContext::Reload, this);
|
|
OPT_SUB("Video/Provider", &VideoContext::Reload, this);
|
|
|
|
// It would be nice to find a way to move these to the individual providers
|
|
OPT_SUB("Provider/Avisynth/Allow Ancient", &VideoContext::Reload, this);
|
|
OPT_SUB("Provider/Avisynth/Memory Max", &VideoContext::Reload, this);
|
|
|
|
OPT_SUB("Provider/Video/FFmpegSource/Decoding Threads", &VideoContext::Reload, this);
|
|
OPT_SUB("Provider/Video/FFmpegSource/Unsafe Seeking", &VideoContext::Reload, this);
|
|
}
|
|
|
|
VideoContext::~VideoContext () {
|
|
}
|
|
|
|
VideoContext *VideoContext::Get() {
|
|
static VideoContext instance;
|
|
return &instance;
|
|
}
|
|
|
|
void VideoContext::Reset() {
|
|
StandardPaths::SetPathValue("?video", "");
|
|
|
|
keyFrames.clear();
|
|
keyFramesFilename.clear();
|
|
videoFPS = agi::vfr::Framerate();
|
|
KeyframesOpen(keyFrames);
|
|
if (!ovrFPS.IsLoaded()) TimecodesOpen(videoFPS);
|
|
|
|
// Remove video data
|
|
frame_n = 0;
|
|
length = 0;
|
|
isPlaying = false;
|
|
nextFrame = -1;
|
|
|
|
// Clean up video data
|
|
videoName.clear();
|
|
|
|
// Remove provider
|
|
videoProvider.reset();
|
|
provider.reset();
|
|
}
|
|
|
|
void VideoContext::SetContext(agi::Context *context) {
|
|
this->context = context;
|
|
context->ass->AddCommitListener(&VideoContext::OnSubtitlesCommit, this);
|
|
context->ass->AddFileSaveListener(&VideoContext::OnSubtitlesSave, this);
|
|
}
|
|
|
|
void VideoContext::SetVideo(const wxString &filename) {
|
|
Stop();
|
|
Reset();
|
|
if (filename.empty()) {
|
|
VideoOpen();
|
|
return;
|
|
}
|
|
|
|
try {
|
|
provider.reset(new ThreadedFrameSource(filename, this));
|
|
videoProvider = provider->GetVideoProvider();
|
|
videoFile = filename;
|
|
|
|
// Check that the script resolution matches the video resolution
|
|
int sx = context->ass->GetScriptInfoAsInt("PlayResX");
|
|
int sy = context->ass->GetScriptInfoAsInt("PlayResY");
|
|
int vx = GetWidth();
|
|
int vy = GetHeight();
|
|
|
|
// If the script resolution hasn't been set at all just force it to the
|
|
// video resolution
|
|
if (sx == 0 && sy == 0) {
|
|
context->ass->SetScriptInfo("PlayResX", wxString::Format("%d", vx));
|
|
context->ass->SetScriptInfo("PlayResY", wxString::Format("%d", vy));
|
|
context->ass->Commit(_("Change script resolution"), AssFile::COMMIT_SCRIPTINFO);
|
|
}
|
|
// If it has been set to something other than a multiple of the video
|
|
// resolution, ask the user if they want it to be fixed
|
|
else if (sx % vx != 0 || sy % vy != 0) {
|
|
switch (OPT_GET("Video/Check Script Res")->GetInt()) {
|
|
case 1: // Ask to change on mismatch
|
|
if (wxMessageBox(wxString::Format(_("The resolution of the loaded video and the resolution specified for the subtitles don't match.\n\nVideo resolution:\t%d x %d\nScript resolution:\t%d x %d\n\nChange subtitles resolution to match video?"), vx, vy, sx, sy), _("Resolution mismatch"), wxYES_NO, context->parent) != wxYES)
|
|
break;
|
|
// Fallthrough to case 2
|
|
case 2: // Always change script res
|
|
context->ass->SetScriptInfo("PlayResX", wxString::Format("%d", vx));
|
|
context->ass->SetScriptInfo("PlayResY", wxString::Format("%d", vy));
|
|
context->ass->Commit(_("change script resolution"), AssFile::COMMIT_SCRIPTINFO);
|
|
break;
|
|
default: // Never change
|
|
break;
|
|
}
|
|
}
|
|
|
|
keyFrames = videoProvider->GetKeyFrames();
|
|
|
|
// Set frame rate
|
|
videoFPS = videoProvider->GetFPS();
|
|
if (ovrFPS.IsLoaded()) {
|
|
int ovr = wxMessageBox(_("You already have timecodes loaded. Would you like to replace them with timecodes from the video file?"), _("Replace timecodes?"), wxYES_NO | wxICON_QUESTION);
|
|
if (ovr == wxYES) {
|
|
ovrFPS = agi::vfr::Framerate();
|
|
ovrTimecodeFile.clear();
|
|
}
|
|
}
|
|
|
|
// Gather video parameters
|
|
length = videoProvider->GetFrameCount();
|
|
|
|
// Set filename
|
|
videoName = filename;
|
|
config::mru->Add("Video", STD_STR(filename));
|
|
wxFileName fn(filename);
|
|
StandardPaths::SetPathValue("?video",fn.GetPath());
|
|
|
|
// Get frame
|
|
frame_n = 0;
|
|
|
|
// Show warning
|
|
wxString warning = videoProvider->GetWarning();
|
|
if (!warning.empty()) wxMessageBox(warning,"Warning",wxICON_WARNING | wxOK);
|
|
|
|
hasSubtitles = false;
|
|
if (filename.Right(4).Lower() == ".mkv") {
|
|
hasSubtitles = MatroskaWrapper::HasSubtitles(filename);
|
|
}
|
|
|
|
provider->LoadSubtitles(context->ass);
|
|
VideoOpen();
|
|
KeyframesOpen(keyFrames);
|
|
TimecodesOpen(FPS());
|
|
}
|
|
catch (agi::UserCancelException const&) { }
|
|
catch (agi::FileNotAccessibleError const& err) {
|
|
config::mru->Remove("Video", STD_STR(filename));
|
|
wxMessageBox(lagi_wxString(err.GetMessage()), "Error setting video", wxICON_ERROR | wxOK);
|
|
}
|
|
catch (VideoProviderError const& err) {
|
|
wxMessageBox(lagi_wxString(err.GetMessage()), "Error setting video", wxICON_ERROR | wxOK);
|
|
}
|
|
}
|
|
|
|
void VideoContext::Reload() {
|
|
if (IsLoaded()) {
|
|
int frame = frame_n;
|
|
SetVideo(videoFile);
|
|
JumpToFrame(frame);
|
|
}
|
|
}
|
|
|
|
void VideoContext::OnSubtitlesCommit() {
|
|
if (!IsLoaded()) return;
|
|
|
|
bool wasPlaying = isPlaying;
|
|
Stop();
|
|
|
|
provider->LoadSubtitles(context->ass);
|
|
GetFrameAsync(frame_n);
|
|
|
|
if (wasPlaying) Play();
|
|
}
|
|
|
|
void VideoContext::OnSubtitlesSave() {
|
|
if (!IsLoaded()) {
|
|
context->ass->SetScriptInfo("Video File", "");
|
|
context->ass->SetScriptInfo("Video Aspect Ratio", "");
|
|
context->ass->SetScriptInfo("Video Position", "");
|
|
context->ass->SetScriptInfo("VFR File", "");
|
|
context->ass->SetScriptInfo("Keyframes File", "");
|
|
return;
|
|
}
|
|
|
|
wxString ar;
|
|
if (arType == 4)
|
|
ar = wxString::Format("c%g", arValue);
|
|
else
|
|
ar = wxString::Format("%d", arType);
|
|
|
|
context->ass->SetScriptInfo("Video File", MakeRelativePath(videoName, context->ass->filename));
|
|
context->ass->SetScriptInfo("Video Aspect Ratio", ar);
|
|
context->ass->SetScriptInfo("Video Position", wxString::Format("%d", frame_n));
|
|
context->ass->SetScriptInfo("VFR File", MakeRelativePath(GetTimecodesName(), context->ass->filename));
|
|
context->ass->SetScriptInfo("Keyframes File", MakeRelativePath(GetKeyFramesName(), context->ass->filename));
|
|
}
|
|
|
|
void VideoContext::JumpToFrame(int n) {
|
|
if (!IsLoaded()) return;
|
|
|
|
// Prevent intervention during playback
|
|
if (isPlaying && n != playNextFrame) return;
|
|
|
|
frame_n = mid(0, n, GetLength() - 1);
|
|
|
|
GetFrameAsync(frame_n);
|
|
Seek(frame_n);
|
|
}
|
|
|
|
void VideoContext::JumpToTime(int ms, agi::vfr::Time end) {
|
|
JumpToFrame(FrameAtTime(ms, end));
|
|
}
|
|
|
|
void VideoContext::GetFrameAsync(int n) {
|
|
provider->RequestFrame(n, videoFPS.TimeAtFrame(n) / 1000.0);
|
|
}
|
|
|
|
std::tr1::shared_ptr<AegiVideoFrame> VideoContext::GetFrame(int n, bool raw) {
|
|
return provider->GetFrame(n, videoFPS.TimeAtFrame(n) / 1000.0, raw);
|
|
}
|
|
|
|
int VideoContext::GetWidth() const {
|
|
return videoProvider->GetWidth();
|
|
}
|
|
int VideoContext::GetHeight() const {
|
|
return videoProvider->GetHeight();
|
|
}
|
|
|
|
void VideoContext::SaveSnapshot(bool raw) {
|
|
// Get folder
|
|
static const agi::OptionValue* ssPath = OPT_GET("Path/Screenshot");
|
|
wxString option = lagi_wxString(ssPath->GetString());
|
|
wxFileName videoFile(videoName);
|
|
wxString basepath;
|
|
// Is it a path specifier and not an actual fixed path?
|
|
if (option[0] == '?') {
|
|
// If dummy video is loaded, we can't save to the video location
|
|
if (option.StartsWith("?video") && (videoName.Find("?dummy") != wxNOT_FOUND)) {
|
|
// So try the script location instead
|
|
option = "?script";
|
|
}
|
|
// Find out where the ?specifier points to
|
|
basepath = StandardPaths::DecodePath(option);
|
|
// If where ever that is isn't defined, we can't save there
|
|
if ((basepath == "\\") || (basepath == "/")) {
|
|
// So save to the current user's home dir instead
|
|
basepath = wxGetHomeDir();
|
|
}
|
|
}
|
|
// Actual fixed (possibly relative) path, decode it
|
|
else basepath = DecodeRelativePath(option,StandardPaths::DecodePath("?user/"));
|
|
basepath += "/" + videoFile.GetName();
|
|
|
|
// Get full path
|
|
int session_shot_count = 1;
|
|
wxString path;
|
|
while (1) {
|
|
path = basepath + wxString::Format("_%03i_%i.png",session_shot_count,frame_n);
|
|
++session_shot_count;
|
|
wxFileName tryPath(path);
|
|
if (!tryPath.FileExists()) break;
|
|
}
|
|
|
|
GetFrame(frame_n,raw)->GetImage().SaveFile(path,wxBITMAP_TYPE_PNG);
|
|
}
|
|
|
|
void VideoContext::NextFrame() {
|
|
if (!videoProvider.get() || isPlaying || frame_n == videoProvider->GetFrameCount())
|
|
return;
|
|
|
|
JumpToFrame(frame_n + 1);
|
|
// Start playing audio
|
|
if (playAudioOnStep->GetBool()) {
|
|
context->audioController->PlayRange(SampleRange(
|
|
context->audioController->SamplesFromMilliseconds(TimeAtFrame(frame_n - 1)),
|
|
context->audioController->SamplesFromMilliseconds(TimeAtFrame(frame_n))));
|
|
}
|
|
}
|
|
|
|
void VideoContext::PrevFrame() {
|
|
if (!videoProvider.get() || isPlaying || frame_n == 0)
|
|
return;
|
|
|
|
JumpToFrame(frame_n - 1);
|
|
// Start playing audio
|
|
if (playAudioOnStep->GetBool()) {
|
|
context->audioController->PlayRange(SampleRange(
|
|
context->audioController->SamplesFromMilliseconds(TimeAtFrame(frame_n)),
|
|
context->audioController->SamplesFromMilliseconds(TimeAtFrame(frame_n + 1))));
|
|
}
|
|
}
|
|
|
|
void VideoContext::Play() {
|
|
if (isPlaying) {
|
|
Stop();
|
|
return;
|
|
}
|
|
|
|
if (!IsLoaded()) return;
|
|
|
|
// Set variables
|
|
startFrame = frame_n;
|
|
endFrame = -1;
|
|
|
|
// Start playing audio
|
|
context->audioController->PlayToEnd(context->audioController->SamplesFromMilliseconds(TimeAtFrame(startFrame)));
|
|
|
|
//audio->Play will override this if we put it before, so put it after.
|
|
isPlaying = true;
|
|
|
|
// Start timer
|
|
playTime.Start();
|
|
playback.SetOwner(this,VIDEO_PLAY_TIMER);
|
|
playback.Start(10);
|
|
}
|
|
|
|
void VideoContext::PlayLine() {
|
|
AssDialogue *curline = context->selectionController->GetActiveLine();
|
|
if (!curline) return;
|
|
|
|
// Start playing audio
|
|
context->audioController->PlayRange(SampleRange(
|
|
context->audioController->SamplesFromMilliseconds(curline->Start.GetMS()),
|
|
context->audioController->SamplesFromMilliseconds(curline->End.GetMS())));
|
|
|
|
// Set variables
|
|
isPlaying = true;
|
|
startFrame = FrameAtTime(context->selectionController->GetActiveLine()->Start.GetMS(),agi::vfr::START);
|
|
endFrame = FrameAtTime(context->selectionController->GetActiveLine()->End.GetMS(),agi::vfr::END);
|
|
|
|
// Jump to start
|
|
playNextFrame = startFrame;
|
|
JumpToFrame(startFrame);
|
|
|
|
// Set other variables
|
|
playTime.Start();
|
|
|
|
// Start timer
|
|
playback.SetOwner(this,VIDEO_PLAY_TIMER);
|
|
playback.Start(10);
|
|
}
|
|
|
|
void VideoContext::Stop() {
|
|
if (isPlaying) {
|
|
playback.Stop();
|
|
isPlaying = false;
|
|
context->audioController->Stop();
|
|
}
|
|
}
|
|
|
|
void VideoContext::OnPlayTimer(wxTimerEvent &event) {
|
|
// Lock
|
|
wxMutexError res = playMutex.TryLock();
|
|
if (res == wxMUTEX_BUSY) return;
|
|
playMutex.Unlock();
|
|
wxMutexLocker lock(playMutex);
|
|
|
|
// Get time difference
|
|
int dif = playTime.Time();
|
|
|
|
// Find next frame
|
|
int startMs = TimeAtFrame(startFrame);
|
|
int nextFrame = frame_n;
|
|
int i=0;
|
|
for (i=0;i<10;i++) {
|
|
if (nextFrame >= length) break;
|
|
if (dif < TimeAtFrame(nextFrame) - startMs) {
|
|
break;
|
|
}
|
|
nextFrame++;
|
|
}
|
|
|
|
// End
|
|
if (nextFrame >= length || (endFrame != -1 && nextFrame > endFrame)) {
|
|
Stop();
|
|
return;
|
|
}
|
|
|
|
// Same frame
|
|
if (nextFrame == frame_n) return;
|
|
|
|
// Next frame is before or over 2 frames ahead, so force audio resync
|
|
if (context->audioController->IsPlaying() && keepAudioSync && (nextFrame < frame_n || nextFrame > frame_n + 2)) {
|
|
context->audioController->ResyncPlaybackPosition(context->audioController->SamplesFromMilliseconds(TimeAtFrame(nextFrame)));
|
|
}
|
|
|
|
// Jump to next frame
|
|
playNextFrame = nextFrame;
|
|
frame_n = nextFrame;
|
|
JumpToFrame(nextFrame);
|
|
|
|
// Sync audio
|
|
if (keepAudioSync && nextFrame % 10 == 0 && context->audioController->IsPlaying()) {
|
|
int64_t audPos = context->audioController->SamplesFromMilliseconds(TimeAtFrame(nextFrame));
|
|
int64_t curPos = context->audioController->GetPlaybackPosition();
|
|
int delta = int(audPos-curPos);
|
|
if (delta < 0) delta = -delta;
|
|
int maxDelta = context->audioController->SamplesFromMilliseconds(1000);
|
|
if (delta > maxDelta) context->audioController->ResyncPlaybackPosition(audPos);
|
|
}
|
|
}
|
|
|
|
double VideoContext::GetARFromType(int type) const {
|
|
if (type == 0) return (double)GetWidth()/(double)GetHeight();
|
|
if (type == 1) return 4.0/3.0;
|
|
if (type == 2) return 16.0/9.0;
|
|
if (type == 3) return 2.35;
|
|
return 1.0; //error
|
|
}
|
|
|
|
void VideoContext::SetAspectRatio(int type, double value) {
|
|
if (type != 4) value = GetARFromType(type);
|
|
|
|
arType = type;
|
|
arValue = mid(.5, value, 5.);
|
|
ARChange(arType, arValue);
|
|
}
|
|
|
|
void VideoContext::LoadKeyframes(wxString filename) {
|
|
if (filename == keyFramesFilename || filename.empty()) return;
|
|
try {
|
|
keyFrames = agi::keyframe::Load(STD_STR(filename));
|
|
keyFramesFilename = filename;
|
|
KeyframesOpen(keyFrames);
|
|
config::mru->Add("Keyframes", STD_STR(filename));
|
|
}
|
|
catch (agi::keyframe::Error const& err) {
|
|
wxMessageBox(err.GetMessage(), "Error opening keyframes file", wxOK | wxICON_ERROR, NULL);
|
|
config::mru->Remove("Keyframes", STD_STR(filename));
|
|
}
|
|
catch (agi::acs::AcsError const&) {
|
|
wxLogError("Could not open file " + filename);
|
|
config::mru->Remove("Keyframes", STD_STR(filename));
|
|
}
|
|
}
|
|
|
|
void VideoContext::SaveKeyframes(wxString filename) {
|
|
agi::keyframe::Save(STD_STR(filename), GetKeyFrames());
|
|
config::mru->Add("Keyframes", STD_STR(filename));
|
|
}
|
|
|
|
void VideoContext::CloseKeyframes() {
|
|
keyFramesFilename.clear();
|
|
if (videoProvider)
|
|
keyFrames = videoProvider->GetKeyFrames();
|
|
else
|
|
keyFrames.clear();
|
|
KeyframesOpen(keyFrames);
|
|
}
|
|
|
|
void VideoContext::LoadTimecodes(wxString filename) {
|
|
if (filename == ovrTimecodeFile || filename.empty()) return;
|
|
try {
|
|
ovrFPS = agi::vfr::Framerate(STD_STR(filename));
|
|
ovrTimecodeFile = filename;
|
|
config::mru->Add("Timecodes", STD_STR(filename));
|
|
OnSubtitlesCommit();
|
|
TimecodesOpen(ovrFPS);
|
|
}
|
|
catch (const agi::acs::AcsError&) {
|
|
wxLogError("Could not open file " + filename);
|
|
config::mru->Remove("Timecodes", STD_STR(filename));
|
|
}
|
|
catch (const agi::vfr::Error& e) {
|
|
wxLogError("Timecode file parse error: %s", e.GetMessage());
|
|
}
|
|
}
|
|
void VideoContext::SaveTimecodes(wxString filename) {
|
|
try {
|
|
FPS().Save(STD_STR(filename), IsLoaded() ? length : -1);
|
|
config::mru->Add("Timecodes", STD_STR(filename));
|
|
}
|
|
catch(const agi::acs::AcsError&) {
|
|
wxLogError("Could not write to " + filename);
|
|
}
|
|
}
|
|
void VideoContext::CloseTimecodes() {
|
|
ovrFPS = agi::vfr::Framerate();
|
|
ovrTimecodeFile.clear();
|
|
OnSubtitlesCommit();
|
|
TimecodesOpen(videoFPS);
|
|
}
|
|
|
|
int VideoContext::TimeAtFrame(int frame, agi::vfr::Time type) const {
|
|
if (ovrFPS.IsLoaded()) {
|
|
return ovrFPS.TimeAtFrame(frame, type);
|
|
}
|
|
return videoFPS.TimeAtFrame(frame, type);
|
|
}
|
|
int VideoContext::FrameAtTime(int time, agi::vfr::Time type) const {
|
|
if (ovrFPS.IsLoaded()) {
|
|
return ovrFPS.FrameAtTime(time, type);
|
|
}
|
|
return videoFPS.FrameAtTime(time, type);
|
|
}
|
|
|
|
void VideoContext::OnVideoError(VideoProviderErrorEvent const& err) {
|
|
wxLogError(
|
|
"Failed seeking video. The video file may be corrupt or incomplete.\n"
|
|
"Error message reported: %s",
|
|
lagi_wxString(err.GetMessage()));
|
|
}
|
|
void VideoContext::OnSubtitlesError(SubtitlesProviderErrorEvent const& err) {
|
|
wxLogError(
|
|
"Failed rendering subtitles. Error message reported: %s",
|
|
lagi_wxString(err.GetMessage()));
|
|
}
|
|
|
|
void VideoContext::OnExit() {
|
|
// On unix wxThreadModule will shut down any still-running threads (and
|
|
// display a warning that it's doing so) before the destructor for
|
|
// VideoContext runs, so manually kill the thread
|
|
Get()->provider.reset();
|
|
}
|