2010-12-31 21:02:17 +00:00
|
|
|
// Copyright (c) 2011, Thomas Goyne <plorkyeran@aegisub.org>
|
|
|
|
//
|
|
|
|
// Permission to use, copy, modify, and distribute this software for any
|
|
|
|
// purpose with or without fee is hereby granted, provided that the above
|
|
|
|
// copyright notice and this permission notice appear in all copies.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
// WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
// MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
// ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
// WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
// ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
// OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
|
|
|
|
/// @file keyframe.cpp
|
|
|
|
/// @see keyframe.h
|
|
|
|
/// @ingroup libaegisub
|
|
|
|
///
|
|
|
|
|
2013-01-04 07:01:50 -08:00
|
|
|
#include "libaegisub/keyframe.h"
|
|
|
|
|
2010-12-31 21:02:17 +00:00
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
#include "libaegisub/io.h"
|
|
|
|
#include "libaegisub/line_iterator.h"
|
|
|
|
|
2012-11-13 06:07:39 -08:00
|
|
|
#include <boost/algorithm/string/predicate.hpp>
|
|
|
|
#include <boost/range/algorithm.hpp>
|
|
|
|
|
2011-11-20 17:35:00 +00:00
|
|
|
namespace {
|
|
|
|
std::vector<int> agi_keyframes(std::istream &file) {
|
2010-12-31 21:02:17 +00:00
|
|
|
double fps;
|
|
|
|
std::string fps_str;
|
|
|
|
file >> fps_str;
|
|
|
|
file >> fps;
|
|
|
|
|
2013-05-02 09:20:54 -07:00
|
|
|
return std::vector<int>(agi::line_iterator<int>(file), agi::line_iterator<int>());
|
2010-12-31 21:02:17 +00:00
|
|
|
}
|
|
|
|
|
2011-11-20 17:35:00 +00:00
|
|
|
std::vector<int> other_keyframes(std::istream &file, char (*func)(std::string const&)) {
|
2010-12-31 21:02:17 +00:00
|
|
|
int count = 0;
|
|
|
|
std::vector<int> ret;
|
|
|
|
agi::line_iterator<std::string> end;
|
2013-05-02 09:20:54 -07:00
|
|
|
for (auto line : agi::line_iterator<std::string>(file)) {
|
|
|
|
char c = tolower(func(line));
|
2012-11-13 06:07:39 -08:00
|
|
|
if (c == 'i')
|
2010-12-31 21:02:17 +00:00
|
|
|
ret.push_back(count++);
|
2012-11-13 06:07:39 -08:00
|
|
|
else if (c == 'p' || c == 'b')
|
2010-12-31 21:02:17 +00:00
|
|
|
++count;
|
|
|
|
}
|
2011-11-20 17:35:00 +00:00
|
|
|
return ret;
|
2010-12-31 21:02:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
char xvid(std::string const& line) {
|
|
|
|
return line.empty() ? 0 : line[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
char divx(std::string const& line) {
|
|
|
|
char chrs[] = "IPB";
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
std::string::size_type pos = line.find(chrs[i]);
|
|
|
|
if (pos != line.npos)
|
|
|
|
return line[pos];
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
char x264(std::string const& line) {
|
|
|
|
std::string::size_type pos = line.find("type:");
|
|
|
|
if (pos == line.npos || pos + 5 >= line.size()) return 0;
|
|
|
|
return line[pos + 5];
|
|
|
|
}
|
2011-11-20 17:35:00 +00:00
|
|
|
}
|
2010-12-31 21:02:17 +00:00
|
|
|
|
|
|
|
namespace agi { namespace keyframe {
|
2013-01-04 07:01:50 -08:00
|
|
|
void Save(agi::fs::path const& filename, std::vector<int> const& keyframes) {
|
2010-12-31 21:02:17 +00:00
|
|
|
io::Save file(filename);
|
2014-03-21 14:23:52 -07:00
|
|
|
std::ostream& of = file.Get();
|
2010-12-31 21:02:17 +00:00
|
|
|
of << "# keyframe format v1" << std::endl;
|
2011-11-20 17:35:00 +00:00
|
|
|
of << "fps " << 0 << std::endl;
|
2012-11-13 06:07:39 -08:00
|
|
|
boost::copy(keyframes, std::ostream_iterator<int>(of, "\n"));
|
2010-12-31 21:02:17 +00:00
|
|
|
}
|
|
|
|
|
2013-01-04 07:01:50 -08:00
|
|
|
std::vector<int> Load(agi::fs::path const& filename) {
|
2014-03-21 14:23:52 -07:00
|
|
|
auto file = io::Open(filename);
|
2013-02-16 19:47:31 -08:00
|
|
|
std::istream &is(*file);
|
2010-12-31 21:02:17 +00:00
|
|
|
|
|
|
|
std::string header;
|
2012-11-13 06:07:39 -08:00
|
|
|
getline(is, header);
|
2010-12-31 21:02:17 +00:00
|
|
|
|
|
|
|
if (header == "# keyframe format v1") return agi_keyframes(is);
|
2012-11-13 06:07:39 -08:00
|
|
|
if (boost::starts_with(header, "# XviD 2pass stat file")) return other_keyframes(is, xvid);
|
2013-07-13 08:40:57 -07:00
|
|
|
if (boost::starts_with(header, "# ffmpeg 2-pass log file, using xvid codec")) return other_keyframes(is, xvid);
|
|
|
|
if (boost::starts_with(header, "# avconv 2-pass log file, using xvid codec")) return other_keyframes(is, xvid);
|
2012-11-13 06:07:39 -08:00
|
|
|
if (boost::starts_with(header, "##map version")) return other_keyframes(is, divx);
|
|
|
|
if (boost::starts_with(header, "#options:")) return other_keyframes(is, x264);
|
2010-12-31 21:02:17 +00:00
|
|
|
|
|
|
|
throw Error("Unknown keyframe format");
|
|
|
|
}
|
|
|
|
|
|
|
|
} }
|