forked from mia/Aegisub
523 lines
16 KiB
C++
523 lines
16 KiB
C++
// Copyright (c) 2006, Niels Martin Hansen
|
|
// 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/
|
|
|
|
/// @file auto4_base.cpp
|
|
/// @brief Baseclasses for Automation 4 scripting framework
|
|
/// @ingroup scripting
|
|
///
|
|
|
|
#include "config.h"
|
|
|
|
#include "auto4_base.h"
|
|
|
|
#include "ass_file.h"
|
|
#include "ass_style.h"
|
|
#include "command/command.h"
|
|
#include "dialog_progress.h"
|
|
#include "include/aegisub/context.h"
|
|
#include "options.h"
|
|
#include "string_codec.h"
|
|
#include "subs_controller.h"
|
|
#include "subtitle_format.h"
|
|
#include "utils.h"
|
|
|
|
#include <libaegisub/dispatch.h>
|
|
#include <libaegisub/fs.h>
|
|
#include <libaegisub/path.h>
|
|
#include <libaegisub/util.h>
|
|
|
|
#include <boost/algorithm/string/replace.hpp>
|
|
#include <boost/algorithm/string/trim.hpp>
|
|
#include <boost/format.hpp>
|
|
#include <boost/tokenizer.hpp>
|
|
#include <future>
|
|
|
|
#include <wx/dcmemory.h>
|
|
#include <wx/log.h>
|
|
#include <wx/sizer.h>
|
|
#include <wx/msgdlg.h>
|
|
|
|
#ifdef __WINDOWS__
|
|
#define WIN32_LEAN_AND_MEAN
|
|
#include <windows.h>
|
|
|
|
#include <libaegisub/charset_conv_win.h>
|
|
#endif
|
|
|
|
namespace Automation4 {
|
|
bool CalculateTextExtents(AssStyle *style, std::string const& text, double &width, double &height, double &descent, double &extlead)
|
|
{
|
|
width = height = descent = extlead = 0;
|
|
|
|
double fontsize = style->fontsize * 64;
|
|
double spacing = style->spacing * 64;
|
|
|
|
#ifdef WIN32
|
|
// This is almost copypasta from TextSub
|
|
HDC thedc = CreateCompatibleDC(0);
|
|
if (!thedc) return false;
|
|
SetMapMode(thedc, MM_TEXT);
|
|
|
|
LOGFONTW lf;
|
|
ZeroMemory(&lf, sizeof(lf));
|
|
lf.lfHeight = (LONG)fontsize;
|
|
lf.lfWeight = style->bold ? FW_BOLD : FW_NORMAL;
|
|
lf.lfItalic = style->italic;
|
|
lf.lfUnderline = style->underline;
|
|
lf.lfStrikeOut = style->strikeout;
|
|
lf.lfCharSet = style->encoding;
|
|
lf.lfOutPrecision = OUT_TT_PRECIS;
|
|
lf.lfClipPrecision = CLIP_DEFAULT_PRECIS;
|
|
lf.lfQuality = ANTIALIASED_QUALITY;
|
|
lf.lfPitchAndFamily = DEFAULT_PITCH|FF_DONTCARE;
|
|
wcsncpy(lf.lfFaceName, agi::charset::ConvertW(style->font).c_str(), 32);
|
|
|
|
HFONT thefont = CreateFontIndirect(&lf);
|
|
if (!thefont) return false;
|
|
SelectObject(thedc, thefont);
|
|
|
|
std::wstring wtext(agi::charset::ConvertW(text));
|
|
SIZE sz;
|
|
if (spacing != 0 ) {
|
|
width = 0;
|
|
for (auto c : wtext) {
|
|
GetTextExtentPoint32(thedc, &c, 1, &sz);
|
|
width += sz.cx + spacing;
|
|
height = sz.cy;
|
|
}
|
|
}
|
|
else {
|
|
GetTextExtentPoint32(thedc, &wtext[0], (int)wtext.size(), &sz);
|
|
width = sz.cx;
|
|
height = sz.cy;
|
|
}
|
|
|
|
TEXTMETRIC tm;
|
|
GetTextMetrics(thedc, &tm);
|
|
descent = tm.tmDescent;
|
|
extlead= tm.tmExternalLeading;
|
|
|
|
DeleteObject(thedc);
|
|
DeleteObject(thefont);
|
|
|
|
#else // not WIN32
|
|
wxMemoryDC thedc;
|
|
|
|
// fix fontsize to be 72 DPI
|
|
//fontsize = -FT_MulDiv((int)(fontsize+0.5), 72, thedc.GetPPI().y);
|
|
|
|
// now try to get a font!
|
|
// use the font list to get some caching... (chance is the script will need the same font very often)
|
|
// USING wxTheFontList SEEMS TO CAUSE BAD LEAKS!
|
|
//wxFont *thefont = wxTheFontList->FindOrCreateFont(
|
|
wxFont thefont(
|
|
(int)fontsize,
|
|
wxFONTFAMILY_DEFAULT,
|
|
style->italic ? wxFONTSTYLE_ITALIC : wxFONTSTYLE_NORMAL,
|
|
style->bold ? wxFONTWEIGHT_BOLD : wxFONTWEIGHT_NORMAL,
|
|
style->underline,
|
|
to_wx(style->font),
|
|
wxFONTENCODING_SYSTEM); // FIXME! make sure to get the right encoding here, make some translation table between windows and wx encodings
|
|
thedc.SetFont(thefont);
|
|
|
|
wxString wtext(to_wx(text));
|
|
if (spacing) {
|
|
// If there's inter-character spacing, kerning info must not be used, so calculate width per character
|
|
// NOTE: Is kerning actually done either way?!
|
|
for (auto const& wc : wtext) {
|
|
int a, b, c, d;
|
|
thedc.GetTextExtent(wc, &a, &b, &c, &d);
|
|
double scaling = fontsize / (double)(b > 0 ? b : 1); // semi-workaround for missing OS/2 table data for scaling
|
|
width += (a + spacing)*scaling;
|
|
height = b > height ? b*scaling : height;
|
|
descent = c > descent ? c*scaling : descent;
|
|
extlead = d > extlead ? d*scaling : extlead;
|
|
}
|
|
} else {
|
|
// If the inter-character spacing should be zero, kerning info can (and must) be used, so calculate everything in one go
|
|
wxCoord lwidth, lheight, ldescent, lextlead;
|
|
thedc.GetTextExtent(wtext, &lwidth, &lheight, &ldescent, &lextlead);
|
|
double scaling = fontsize / (double)(lheight > 0 ? lheight : 1); // semi-workaround for missing OS/2 table data for scaling
|
|
width = lwidth*scaling; height = lheight*scaling; descent = ldescent*scaling; extlead = lextlead*scaling;
|
|
}
|
|
#endif
|
|
|
|
// Compensate for scaling
|
|
width = style->scalex / 100 * width / 64;
|
|
height = style->scaley / 100 * height / 64;
|
|
descent = style->scaley / 100 * descent / 64;
|
|
extlead = style->scaley / 100 * extlead / 64;
|
|
|
|
return true;
|
|
}
|
|
|
|
ExportFilter::ExportFilter(std::string const& name, std::string const& description, int priority)
|
|
: AssExportFilter(name, description, priority)
|
|
{
|
|
}
|
|
|
|
std::string ExportFilter::GetScriptSettingsIdentifier()
|
|
{
|
|
return inline_string_encode("Automation Settings " + GetName());
|
|
}
|
|
|
|
wxWindow* ExportFilter::GetConfigDialogWindow(wxWindow *parent, agi::Context *c) {
|
|
config_dialog.reset(GenerateConfigDialog(parent, c));
|
|
|
|
if (config_dialog) {
|
|
std::string val = c->ass->GetScriptInfo(GetScriptSettingsIdentifier());
|
|
if (!val.empty())
|
|
config_dialog->Unserialise(val);
|
|
return config_dialog->CreateWindow(parent);
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
void ExportFilter::LoadSettings(bool is_default, agi::Context *c) {
|
|
if (config_dialog) {
|
|
std::string val = config_dialog->Serialise();
|
|
if (!val.empty())
|
|
c->ass->SetScriptInfo(GetScriptSettingsIdentifier(), val);
|
|
}
|
|
}
|
|
|
|
// ProgressSink
|
|
ProgressSink::ProgressSink(agi::ProgressSink *impl, BackgroundScriptRunner *bsr)
|
|
: impl(impl)
|
|
, bsr(bsr)
|
|
, trace_level(OPT_GET("Automation/Trace Level")->GetInt())
|
|
{
|
|
}
|
|
|
|
void ProgressSink::ShowDialog(ScriptDialog *config_dialog)
|
|
{
|
|
agi::dispatch::Main().Sync([=] {
|
|
wxDialog w; // container dialog box
|
|
w.SetExtraStyle(wxWS_EX_VALIDATE_RECURSIVELY);
|
|
w.Create(bsr->GetParentWindow(), -1, to_wx(bsr->GetTitle()));
|
|
auto s = new wxBoxSizer(wxHORIZONTAL); // sizer for putting contents in
|
|
wxWindow *ww = config_dialog->CreateWindow(&w); // generate actual dialog contents
|
|
s->Add(ww, 0, wxALL, 5); // add contents to dialog
|
|
w.SetSizerAndFit(s);
|
|
w.CenterOnParent();
|
|
w.ShowModal();
|
|
});
|
|
}
|
|
|
|
int ProgressSink::ShowDialog(wxDialog *dialog)
|
|
{
|
|
int ret = 0;
|
|
agi::dispatch::Main().Sync([&] { ret = dialog->ShowModal(); });
|
|
return ret;
|
|
}
|
|
|
|
BackgroundScriptRunner::BackgroundScriptRunner(wxWindow *parent, std::string const& title)
|
|
: impl(new DialogProgress(parent, to_wx(title)))
|
|
{
|
|
}
|
|
|
|
BackgroundScriptRunner::~BackgroundScriptRunner()
|
|
{
|
|
}
|
|
|
|
void BackgroundScriptRunner::Run(std::function<void (ProgressSink*)> task)
|
|
{
|
|
int prio = OPT_GET("Automation/Thread Priority")->GetInt();
|
|
if (prio == 0) prio = 50; // normal
|
|
else if (prio == 1) prio = 30; // below normal
|
|
else if (prio == 2) prio = 10; // lowest
|
|
else prio = 50; // fallback normal
|
|
|
|
impl->Run([&](agi::ProgressSink *ps) {
|
|
ProgressSink aps(ps, this);
|
|
task(&aps);
|
|
}, prio);
|
|
}
|
|
|
|
wxWindow *BackgroundScriptRunner::GetParentWindow() const
|
|
{
|
|
return impl.get();
|
|
}
|
|
|
|
std::string BackgroundScriptRunner::GetTitle() const
|
|
{
|
|
return from_wx(impl->GetTitle());
|
|
}
|
|
|
|
// Script
|
|
Script::Script(agi::fs::path const& filename)
|
|
: filename(filename)
|
|
{
|
|
include_path.emplace_back(filename.parent_path());
|
|
|
|
std::string include_paths = OPT_GET("Path/Automation/Include")->GetString();
|
|
boost::char_separator<char> sep("|");
|
|
for (auto const& tok : boost::tokenizer<boost::char_separator<char>>(include_paths, sep)) {
|
|
auto path = config::path->Decode(tok);
|
|
if (path.is_absolute() && agi::fs::DirectoryExists(path))
|
|
include_path.emplace_back(std::move(path));
|
|
}
|
|
}
|
|
|
|
// ScriptManager
|
|
ScriptManager::~ScriptManager()
|
|
{
|
|
}
|
|
|
|
void ScriptManager::Add(std::unique_ptr<Script> script)
|
|
{
|
|
if (find(scripts.begin(), scripts.end(), script) == scripts.end())
|
|
scripts.emplace_back(std::move(script));
|
|
|
|
ScriptsChanged();
|
|
}
|
|
|
|
void ScriptManager::Remove(Script *script)
|
|
{
|
|
auto i = find_if(scripts.begin(), scripts.end(), [&](std::unique_ptr<Script> const& s) { return s.get() == script; });
|
|
if (i != scripts.end())
|
|
scripts.erase(i);
|
|
|
|
ScriptsChanged();
|
|
}
|
|
|
|
void ScriptManager::RemoveAll()
|
|
{
|
|
scripts.clear();
|
|
ScriptsChanged();
|
|
}
|
|
|
|
void ScriptManager::Reload(Script *script)
|
|
{
|
|
script->Reload();
|
|
ScriptsChanged();
|
|
}
|
|
|
|
const std::vector<cmd::Command*>& ScriptManager::GetMacros()
|
|
{
|
|
macros.clear();
|
|
for (auto& script : scripts) {
|
|
std::vector<cmd::Command*> sfs = script->GetMacros();
|
|
copy(sfs.begin(), sfs.end(), back_inserter(macros));
|
|
}
|
|
return macros;
|
|
}
|
|
|
|
// AutoloadScriptManager
|
|
AutoloadScriptManager::AutoloadScriptManager(std::string path)
|
|
: path(std::move(path))
|
|
{
|
|
Reload();
|
|
}
|
|
|
|
void AutoloadScriptManager::Reload()
|
|
{
|
|
scripts.clear();
|
|
|
|
std::vector<std::future<std::unique_ptr<Script>>> script_futures;
|
|
|
|
boost::char_separator<char> sep("|");
|
|
for (auto const& tok : boost::tokenizer<boost::char_separator<char>>(path, sep)) {
|
|
auto dirname = config::path->Decode(tok);
|
|
if (!agi::fs::DirectoryExists(dirname)) continue;
|
|
|
|
for (auto filename : agi::fs::DirectoryIterator(dirname, "*.*"))
|
|
script_futures.emplace_back(std::async(std::launch::async, [=] {
|
|
return ScriptFactory::CreateFromFile(dirname/filename, false, false);
|
|
}));
|
|
}
|
|
|
|
int error_count = 0;
|
|
for (auto& future : script_futures) {
|
|
auto s = future.get();
|
|
if (s) {
|
|
if (!s->GetLoadedState()) ++error_count;
|
|
scripts.emplace_back(std::move(s));
|
|
}
|
|
}
|
|
|
|
if (error_count == 1) {
|
|
wxLogWarning("A script in the Automation autoload directory failed to load.\nPlease review the errors, fix them and use the Rescan Autoload Dir button in Automation Manager to load the scripts again.");
|
|
}
|
|
else if (error_count > 1) {
|
|
wxLogWarning("Multiple scripts in the Automation autoload directory failed to load.\nPlease review the errors, fix them and use the Rescan Autoload Dir button in Automation Manager to load the scripts again.");
|
|
}
|
|
|
|
ScriptsChanged();
|
|
}
|
|
|
|
LocalScriptManager::LocalScriptManager(agi::Context *c)
|
|
: context(c)
|
|
{
|
|
slots.push_back(c->subsController->AddFileSaveListener(&LocalScriptManager::OnSubtitlesSave, this));
|
|
slots.push_back(c->subsController->AddFileOpenListener(&LocalScriptManager::Reload, this));
|
|
}
|
|
|
|
void LocalScriptManager::Reload()
|
|
{
|
|
scripts.clear();
|
|
|
|
auto local_scripts = context->ass->GetScriptInfo("Automation Scripts");
|
|
if (local_scripts.empty()) {
|
|
ScriptsChanged();
|
|
return;
|
|
}
|
|
|
|
auto autobasefn(OPT_GET("Path/Automation/Base")->GetString());
|
|
|
|
boost::char_separator<char> sep("|");
|
|
for (auto const& cur : boost::tokenizer<boost::char_separator<char>>(local_scripts, sep)) {
|
|
auto trimmed = boost::trim_copy(cur);
|
|
char first_char = trimmed[0];
|
|
trimmed.erase(0);
|
|
|
|
agi::fs::path basepath;
|
|
if (first_char == '~') {
|
|
basepath = context->subsController->Filename().parent_path();
|
|
} else if (first_char == '$') {
|
|
basepath = autobasefn;
|
|
} else if (first_char == '/') {
|
|
} else {
|
|
wxLogWarning("Automation Script referenced with unknown location specifier character.\nLocation specifier found: %c\nFilename specified: %s",
|
|
first_char, to_wx(trimmed));
|
|
continue;
|
|
}
|
|
auto sfname = basepath/trimmed;
|
|
if (agi::fs::FileExists(sfname))
|
|
scripts.emplace_back(Automation4::ScriptFactory::CreateFromFile(sfname, true));
|
|
else {
|
|
wxLogWarning("Automation Script referenced could not be found.\nFilename specified: %c%s\nSearched relative to: %s\nResolved filename: %s",
|
|
first_char, to_wx(trimmed), basepath.wstring(), sfname.wstring());
|
|
}
|
|
}
|
|
|
|
ScriptsChanged();
|
|
}
|
|
|
|
void LocalScriptManager::OnSubtitlesSave()
|
|
{
|
|
// Store Automation script data
|
|
// Algorithm:
|
|
// 1. If script filename has Automation Base Path as a prefix, the path is relative to that (ie. "$")
|
|
// 2. Otherwise try making it relative to the ass filename
|
|
// 3. If step 2 failed, or absolute path is shorter than path relative to ass, use absolute path ("/")
|
|
// 4. Otherwise, use path relative to ass ("~")
|
|
std::string scripts_string;
|
|
agi::fs::path autobasefn(OPT_GET("Path/Automation/Base")->GetString());
|
|
|
|
for (auto& script : GetScripts()) {
|
|
if (!scripts_string.empty())
|
|
scripts_string += "|";
|
|
|
|
auto scriptfn(script->GetFilename().string());
|
|
auto autobase_rel = config::path->MakeRelative(scriptfn, autobasefn);
|
|
auto assfile_rel = config::path->MakeRelative(scriptfn, "?script");
|
|
|
|
if (autobase_rel.string().size() <= scriptfn.size() && autobase_rel.string().size() <= assfile_rel.string().size()) {
|
|
scriptfn = "$" + autobase_rel.generic_string();
|
|
} else if (assfile_rel.string().size() <= scriptfn.size() && assfile_rel.string().size() <= autobase_rel.string().size()) {
|
|
scriptfn = "~" + assfile_rel.generic_string();
|
|
} else {
|
|
scriptfn = "/" + script->GetFilename().generic_string();
|
|
}
|
|
|
|
scripts_string += scriptfn;
|
|
}
|
|
context->ass->SetScriptInfo("Automation Scripts", scripts_string);
|
|
}
|
|
|
|
// ScriptFactory
|
|
ScriptFactory::ScriptFactory(std::string engine_name, std::string filename_pattern)
|
|
: engine_name(std::move(engine_name))
|
|
, filename_pattern(std::move(filename_pattern))
|
|
{
|
|
}
|
|
|
|
void ScriptFactory::Register(std::unique_ptr<ScriptFactory> factory)
|
|
{
|
|
if (find(Factories().begin(), Factories().end(), factory) != Factories().end())
|
|
throw agi::InternalError("Automation 4: Attempt to register the same script factory multiple times. This should never happen.", nullptr);
|
|
|
|
Factories().emplace_back(std::move(factory));
|
|
}
|
|
|
|
std::unique_ptr<Script> ScriptFactory::CreateFromFile(agi::fs::path const& filename, bool complain_about_unrecognised, bool create_unknown)
|
|
{
|
|
for (auto& factory : Factories()) {
|
|
auto s = factory->Produce(filename);
|
|
if (s) {
|
|
if (!s->GetLoadedState()) {
|
|
wxLogError(_("Failed to load Automation script '%s':\n%s"), filename.wstring(), s->GetDescription());
|
|
}
|
|
return s;
|
|
}
|
|
}
|
|
|
|
if (complain_about_unrecognised) {
|
|
wxLogError(_("The file was not recognised as an Automation script: %s"), filename.wstring());
|
|
}
|
|
|
|
return create_unknown ? agi::util::make_unique<UnknownScript>(filename) : nullptr;
|
|
}
|
|
|
|
std::vector<std::unique_ptr<ScriptFactory>>& ScriptFactory::Factories()
|
|
{
|
|
static std::vector<std::unique_ptr<ScriptFactory>> factories;
|
|
return factories;
|
|
}
|
|
|
|
const std::vector<std::unique_ptr<ScriptFactory>>& ScriptFactory::GetFactories()
|
|
{
|
|
return Factories();
|
|
}
|
|
|
|
std::string ScriptFactory::GetWildcardStr()
|
|
{
|
|
std::string fnfilter, catchall;
|
|
for (auto& fact : Factories()) {
|
|
if (fact->GetEngineName().empty() || fact->GetFilenamePattern().empty())
|
|
continue;
|
|
|
|
std::string filter(fact->GetFilenamePattern());
|
|
boost::replace_all(filter, ",", ";");
|
|
fnfilter += str(boost::format("%s scripts (%s)|%s|") % fact->GetEngineName() % fact->GetFilenamePattern() % filter);
|
|
catchall += filter + ";";
|
|
}
|
|
fnfilter += from_wx(_("All Files")) + " (*.*)|*.*";
|
|
|
|
if (!catchall.empty())
|
|
catchall.pop_back();
|
|
|
|
if (Factories().size() > 1)
|
|
fnfilter = from_wx(_("All Supported Formats")) + "|" + catchall + "|" + fnfilter;
|
|
|
|
return fnfilter;
|
|
}
|
|
}
|