mirror of https://github.com/odrling/Aegisub
309 lines
9.1 KiB
C++
309 lines
9.1 KiB
C++
// Copyright (c) 2014, 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.
|
|
//
|
|
// Aegisub Project http://www.aegisub.org/
|
|
|
|
#include "ass_file.h"
|
|
|
|
#include "ass_attachment.h"
|
|
#include "ass_dialogue.h"
|
|
#include "ass_info.h"
|
|
#include "ass_style.h"
|
|
#include "ass_style_storage.h"
|
|
#include "options.h"
|
|
|
|
#include <algorithm>
|
|
#include <boost/algorithm/string/case_conv.hpp>
|
|
#include <boost/algorithm/string/predicate.hpp>
|
|
#include <boost/filesystem/path.hpp>
|
|
#include <cassert>
|
|
#include <unordered_map>
|
|
#include <unordered_set>
|
|
|
|
AssFile::AssFile() { }
|
|
|
|
AssFile::~AssFile() {
|
|
Styles.clear_and_dispose([](AssStyle *e) { delete e; });
|
|
Events.clear_and_dispose([](AssDialogue *e) { delete e; });
|
|
}
|
|
|
|
void AssFile::LoadDefault(bool include_dialogue_line, std::string const& style_catalog) {
|
|
Info.emplace_back("Title", "Default Aegisub file");
|
|
Info.emplace_back("ScriptType", "v4.00+");
|
|
Info.emplace_back("WrapStyle", "0");
|
|
Info.emplace_back("ScaledBorderAndShadow", "yes");
|
|
if (!OPT_GET("Subtitle/Default Resolution/Auto")->GetBool()) {
|
|
Info.emplace_back("PlayResX", std::to_string(OPT_GET("Subtitle/Default Resolution/Width")->GetInt()));
|
|
Info.emplace_back("PlayResY", std::to_string(OPT_GET("Subtitle/Default Resolution/Height")->GetInt()));
|
|
}
|
|
Info.emplace_back("YCbCr Matrix", "None");
|
|
|
|
// Add default style
|
|
Styles.push_back(*new AssStyle);
|
|
|
|
// Add/replace any catalog styles requested
|
|
if (AssStyleStorage::CatalogExists(style_catalog)) {
|
|
AssStyleStorage catalog;
|
|
catalog.LoadCatalog(style_catalog);
|
|
catalog.ReplaceIntoFile(*this);
|
|
}
|
|
|
|
if (include_dialogue_line)
|
|
Events.push_back(*new AssDialogue);
|
|
}
|
|
|
|
AssFile::AssFile(const AssFile &from)
|
|
: Info(from.Info)
|
|
, Attachments(from.Attachments)
|
|
, Extradata(from.Extradata)
|
|
, next_extradata_id(from.next_extradata_id)
|
|
{
|
|
Styles.clone_from(from.Styles,
|
|
[](AssStyle const& e) { return new AssStyle(e); },
|
|
[](AssStyle *e) { delete e; });
|
|
Events.clone_from(from.Events,
|
|
[](AssDialogue const& e) { return new AssDialogue(e); },
|
|
[](AssDialogue *e) { delete e; });
|
|
}
|
|
|
|
void AssFile::swap(AssFile& from) throw() {
|
|
Info.swap(from.Info);
|
|
Styles.swap(from.Styles);
|
|
Events.swap(from.Events);
|
|
Attachments.swap(from.Attachments);
|
|
Extradata.swap(from.Extradata);
|
|
std::swap(Properties, from.Properties);
|
|
std::swap(next_extradata_id, from.next_extradata_id);
|
|
}
|
|
|
|
AssFile& AssFile::operator=(AssFile from) {
|
|
swap(from);
|
|
return *this;
|
|
}
|
|
|
|
EntryList<AssDialogue>::iterator AssFile::iterator_to(AssDialogue& line) {
|
|
using l = EntryList<AssDialogue>;
|
|
bool in_list = !l::node_algorithms::inited(l::value_traits::to_node_ptr(line));
|
|
return in_list ? Events.iterator_to(line) : Events.end();
|
|
}
|
|
|
|
void AssFile::InsertAttachment(agi::fs::path const& filename) {
|
|
AssEntryGroup group = AssEntryGroup::GRAPHIC;
|
|
|
|
auto ext = boost::to_lower_copy(filename.extension().string());
|
|
if (ext == ".ttf" || ext == ".ttc" || ext == ".pfb")
|
|
group = AssEntryGroup::FONT;
|
|
|
|
Attachments.emplace_back(filename, group);
|
|
}
|
|
|
|
std::string AssFile::GetScriptInfo(std::string const& key) const {
|
|
for (auto const& info : Info) {
|
|
if (boost::iequals(key, info.Key()))
|
|
return info.Value();
|
|
}
|
|
|
|
return "";
|
|
}
|
|
|
|
int AssFile::GetScriptInfoAsInt(std::string const& key) const {
|
|
return atoi(GetScriptInfo(key).c_str());
|
|
}
|
|
|
|
void AssFile::SetScriptInfo(std::string const& key, std::string const& value) {
|
|
for (auto it = Info.begin(); it != Info.end(); ++it) {
|
|
if (boost::iequals(key, it->Key())) {
|
|
if (value.empty())
|
|
Info.erase(it);
|
|
else
|
|
it->SetValue(value);
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (!value.empty())
|
|
Info.emplace_back(key, value);
|
|
}
|
|
|
|
void AssFile::GetResolution(int &sw, int &sh) const {
|
|
sw = GetScriptInfoAsInt("PlayResX");
|
|
sh = GetScriptInfoAsInt("PlayResY");
|
|
|
|
// Gabest logic: default is 384x288, assume 1280x1024 if either height or
|
|
// width are that, otherwise assume 4:3 if only heigh or width are set.
|
|
// Why 1280x1024? Who the fuck knows. Clearly just Gabest trolling everyone.
|
|
if (sw == 0 && sh == 0) {
|
|
sw = 384;
|
|
sh = 288;
|
|
}
|
|
else if (sw == 0)
|
|
sw = sh == 1024 ? 1280 : sh * 4 / 3;
|
|
else if (sh == 0)
|
|
sh = sw == 1280 ? 1024 : sw * 3 / 4;
|
|
}
|
|
|
|
std::vector<std::string> AssFile::GetStyles() const {
|
|
std::vector<std::string> styles;
|
|
for (auto& style : Styles)
|
|
styles.push_back(style.name);
|
|
return styles;
|
|
}
|
|
|
|
AssStyle *AssFile::GetStyle(std::string const& name) {
|
|
for (auto& style : Styles) {
|
|
if (boost::iequals(style.name, name))
|
|
return &style;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
int AssFile::Commit(wxString const& desc, int type, int amend_id, AssDialogue *single_line) {
|
|
if (type == COMMIT_NEW || (type & COMMIT_DIAG_ADDREM) || (type & COMMIT_ORDER)) {
|
|
int i = 0;
|
|
for (auto& event : Events)
|
|
event.Row = i++;
|
|
}
|
|
|
|
PushState({desc, &amend_id, single_line});
|
|
|
|
AnnounceCommit(type, single_line);
|
|
|
|
return amend_id;
|
|
}
|
|
|
|
bool AssFile::CompStart(AssDialogue const& lft, AssDialogue const& rgt) {
|
|
return lft.Start < rgt.Start;
|
|
}
|
|
bool AssFile::CompEnd(AssDialogue const& lft, AssDialogue const& rgt) {
|
|
return lft.End < rgt.End;
|
|
}
|
|
bool AssFile::CompStyle(AssDialogue const& lft, AssDialogue const& rgt) {
|
|
return lft.Style < rgt.Style;
|
|
}
|
|
bool AssFile::CompActor(AssDialogue const& lft, AssDialogue const& rgt) {
|
|
return lft.Actor < rgt.Actor;
|
|
}
|
|
bool AssFile::CompEffect(AssDialogue const& lft, AssDialogue const& rgt) {
|
|
return lft.Effect < rgt.Effect;
|
|
}
|
|
bool AssFile::CompLayer(AssDialogue const& lft, AssDialogue const& rgt) {
|
|
return lft.Layer < rgt.Layer;
|
|
}
|
|
|
|
void AssFile::Sort(CompFunc comp, std::set<AssDialogue*> const& limit) {
|
|
Sort(Events, comp, limit);
|
|
}
|
|
|
|
void AssFile::Sort(EntryList<AssDialogue> &lst, CompFunc comp, std::set<AssDialogue*> const& limit) {
|
|
if (limit.empty()) {
|
|
lst.sort(comp);
|
|
return;
|
|
}
|
|
|
|
// Sort each selected block separately, leaving everything else untouched
|
|
for (auto begin = lst.begin(); begin != lst.end(); ++begin) {
|
|
if (!limit.count(&*begin)) continue;
|
|
auto end = begin;
|
|
while (end != lst.end() && limit.count(&*end)) ++end;
|
|
|
|
// sort doesn't support only sorting a sublist, so move them to a temp list
|
|
EntryList<AssDialogue> tmp;
|
|
tmp.splice(tmp.begin(), lst, begin, end);
|
|
tmp.sort(comp);
|
|
lst.splice(end, tmp);
|
|
|
|
begin = --end;
|
|
}
|
|
}
|
|
|
|
uint32_t AssFile::AddExtradata(std::string const& key, std::string const& value) {
|
|
for (auto const& data : Extradata) {
|
|
// perform brute-force deduplication by simple key and value comparison
|
|
if (key == data.key && value == data.value) {
|
|
return data.id;
|
|
}
|
|
}
|
|
Extradata.push_back(ExtradataEntry{next_extradata_id, key, value});
|
|
return next_extradata_id++; // return old value, then post-increment
|
|
}
|
|
|
|
namespace {
|
|
struct extradata_id_cmp {
|
|
bool operator()(ExtradataEntry const& e, uint32_t id) {
|
|
return e.id < id;
|
|
}
|
|
bool operator()(uint32_t id, ExtradataEntry const& e) {
|
|
return id < e.id;
|
|
}
|
|
};
|
|
|
|
template<typename ExtradataType, typename Func>
|
|
void enumerate_extradata(ExtradataType&& extradata, std::vector<uint32_t> const& id_list, Func&& f) {
|
|
auto begin = extradata.begin(), end = extradata.end();
|
|
for (auto id : id_list) {
|
|
auto it = lower_bound(begin, end, id, extradata_id_cmp{});
|
|
if (it != end) {
|
|
f(*it);
|
|
begin = it;
|
|
}
|
|
}
|
|
}
|
|
|
|
template<typename K, typename V>
|
|
using reference_map = std::unordered_map<std::reference_wrapper<const K>, V, std::hash<K>, std::equal_to<K>>;
|
|
}
|
|
|
|
std::vector<ExtradataEntry> AssFile::GetExtradata(std::vector<uint32_t> const& id_list) const {
|
|
std::vector<ExtradataEntry> result;
|
|
enumerate_extradata(Extradata, id_list, [&](ExtradataEntry const& e) {
|
|
result.push_back(e);
|
|
});
|
|
return result;
|
|
}
|
|
|
|
void AssFile::CleanExtradata() {
|
|
if (Extradata.empty()) return;
|
|
|
|
std::unordered_set<uint32_t> ids_used;
|
|
for (auto& line : Events) {
|
|
if (line.ExtradataIds.get().empty()) continue;
|
|
|
|
// Find the ID for each unique key in the line
|
|
reference_map<std::string, uint32_t> keys_used;
|
|
enumerate_extradata(Extradata, line.ExtradataIds.get(), [&](ExtradataEntry const& e) {
|
|
keys_used[e.key] = e.id;
|
|
});
|
|
|
|
for (auto const& used : keys_used)
|
|
ids_used.insert(used.second);
|
|
|
|
// If any keys were duplicated or missing, update the id list
|
|
if (keys_used.size() != line.ExtradataIds.get().size()) {
|
|
std::vector<uint32_t> ids;
|
|
ids.reserve(keys_used.size());
|
|
for (auto const& used : keys_used)
|
|
ids.push_back(used.second);
|
|
std::sort(begin(ids), end(ids));
|
|
line.ExtradataIds = std::move(ids);
|
|
}
|
|
}
|
|
|
|
if (ids_used.size() != Extradata.size()) {
|
|
// Erase all no-longer-used extradata entries
|
|
Extradata.erase(std::remove_if(begin(Extradata), end(Extradata), [&](ExtradataEntry const& e) {
|
|
return !ids_used.count(e.id);
|
|
}), end(Extradata));
|
|
}
|
|
}
|