mirror of https://github.com/odrling/Aegisub
At least the auto3 shared lib builds now, still missing the code to call text_extents in the host app though. And completely untested.
Originally committed to SVN as r1157.
This commit is contained in:
parent
2785821a62
commit
e9284e5f88
864
auto3/auto3.c
864
auto3/auto3.c
|
@ -1,4 +1,4 @@
|
||||||
// Copyright (c) 2007, Niels Martin Hansen
|
// Copyright (c) 2005, 2006, 2007, Niels Martin Hansen
|
||||||
// All rights reserved.
|
// All rights reserved.
|
||||||
//
|
//
|
||||||
// Redistribution and use in source and binary forms, with or without
|
// Redistribution and use in source and binary forms, with or without
|
||||||
|
@ -37,12 +37,16 @@
|
||||||
#include "auto3.h"
|
#include "auto3.h"
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
|
||||||
// Win32 DLL entry point
|
// Win32 DLL entry point
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
#define WIN32_LEAN_AND_MEAN
|
#define WIN32_LEAN_AND_MEAN
|
||||||
#include <windows.h>
|
#include <windows.h>
|
||||||
|
#include <wchar.h>
|
||||||
|
|
||||||
BOOL APIENTRY DllMain( HANDLE hModule,
|
BOOL APIENTRY DllMain( HANDLE hModule,
|
||||||
DWORD ul_reason_for_call,
|
DWORD ul_reason_for_call,
|
||||||
LPVOID lpReserved
|
LPVOID lpReserved
|
||||||
|
@ -54,19 +58,463 @@ BOOL APIENTRY DllMain( HANDLE hModule,
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
// Create a new interpreter
|
struct script_reader_data {
|
||||||
// Returns pointer to interpreter object if successful, otherwise NULL
|
FILE *f;
|
||||||
// If this function fails, the error message is filled into the char* pointed to by error (may be NULL)
|
int isfirst;
|
||||||
AUTO3_API struct Auto3Interpreter *CreateAuto3Script(filename_t filename, char **error)
|
#define SCRIPT_READER_BUFSIZE 512
|
||||||
|
char databuf[SCRIPT_READER_BUFSIZE];
|
||||||
|
};
|
||||||
|
static const char *script_reader_func(lua_State *L, void *data, size_t *size)
|
||||||
{
|
{
|
||||||
|
struct script_reader_data *self;
|
||||||
|
char *b;
|
||||||
|
FILE *f;
|
||||||
|
|
||||||
|
self = (struct script_reader_data *)(data);
|
||||||
|
b = self->databuf;
|
||||||
|
f = self->f;
|
||||||
|
|
||||||
|
if (feof(f)) {
|
||||||
|
*size = 0;
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (self->isfirst) {
|
||||||
|
self->isfirst = 0;
|
||||||
|
// check if file is sensible and maybe skip bom
|
||||||
|
if ((*size = fread(b, 1, 4, f)) == 4) {
|
||||||
|
if (b[0] == 0xEF && b[1] == 0xBB && b[2] == 0xBF) {
|
||||||
|
// got an utf8 file with bom
|
||||||
|
// nothing further to do, already skipped the bom
|
||||||
|
fseek(f, -1, SEEK_CUR);
|
||||||
|
} else {
|
||||||
|
// oops, not utf8 with bom
|
||||||
|
// check if there is some other BOM in place and complain if there is...
|
||||||
|
if ((b[0] == 0xFF && b[1] == 0xFE && b[2] == 0x00 && b[3] == 0x00) || // utf32be
|
||||||
|
(b[0] == 0x00 && b[1] == 0x00 && b[2] == 0xFE && b[3] == 0xFF) || // utf32le
|
||||||
|
(b[0] == 0xFF && b[1] == 0xFE) || // utf16be
|
||||||
|
(b[0] == 0xFE && b[1] == 0xFF) || // utf16le
|
||||||
|
(b[0] == 0x2B && b[1] == 0x2F && b[2] == 0x76) || // utf7
|
||||||
|
(b[0] == 0x00 && b[2] == 0x00) || // looks like utf16be
|
||||||
|
(b[1] == 0x00 && b[3] == 0x00)) { // looks like utf16le
|
||||||
|
// can't support these files
|
||||||
|
*size = 0;
|
||||||
|
self->isfirst = -1;
|
||||||
|
strcpy(b, "File is an unsupported UTF");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
// assume utf8 without bom, and rewind file
|
||||||
|
fseek(f, 0, SEEK_SET);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// hmm, rather short file this...
|
||||||
|
// doesn't have a bom, assume it's just ascii/utf8 without bom
|
||||||
|
return self->databuf; // *size is already set
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
*size = fread(b, 1, SCRIPT_READER_BUFSIZE, f);
|
||||||
|
|
||||||
|
return self->databuf;
|
||||||
|
}
|
||||||
|
static int Auto3LuaLoad(lua_State *L, filename_t filename, const char *prettyname, char **error)
|
||||||
|
{
|
||||||
|
struct script_reader_data script_reader;
|
||||||
|
int res;
|
||||||
|
|
||||||
|
script_reader.f =
|
||||||
|
#ifdef WIN32
|
||||||
|
_wfopen(filename, L"rb");
|
||||||
|
#else
|
||||||
|
fopen(filename, "rb");
|
||||||
|
#endif
|
||||||
|
if (!script_reader.f) return -1;
|
||||||
|
|
||||||
|
script_reader.isfirst = 1;
|
||||||
|
|
||||||
|
res = lua_load(L, script_reader_func, &script_reader, prettyname);
|
||||||
|
|
||||||
|
fclose(script_reader.f);
|
||||||
|
|
||||||
|
if (res) {
|
||||||
|
*error = strdup(lua_tostring(L, -1));
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
if (script_reader.isfirst == -1) {
|
||||||
|
// Signals we got a bad UTF
|
||||||
|
*error = strdup(script_reader.databuf);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Read the 'config' global and create config struct
|
||||||
|
static int Auto3ParseConfigData(lua_State *L, struct Auto3Interpreter *script, char **error)
|
||||||
|
{
|
||||||
|
struct Auto3ConfigOption *opt;
|
||||||
|
int i, n;
|
||||||
|
const char *tmp;
|
||||||
|
|
||||||
|
if (!lua_istable(L, -1)) {
|
||||||
|
// No 'config' table at all, just make the sentinel option
|
||||||
|
script->config = calloc(1, sizeof(struct Auto3ConfigOption));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get expected number of elements in table
|
||||||
|
n = luaL_getn(L, -1);
|
||||||
|
// Allocate memory for max number of elements + 1
|
||||||
|
script->config = calloc(n+1, sizeof(struct Auto3ConfigOption));
|
||||||
|
|
||||||
|
// Prepare traversal
|
||||||
|
lua_pushnil(L);
|
||||||
|
opt = script->config;
|
||||||
|
|
||||||
|
// Get at most n options
|
||||||
|
i = 1;
|
||||||
|
while (lua_next(L, -2)) {
|
||||||
|
if (i > n) {
|
||||||
|
// More options than we have space for...
|
||||||
|
// Just ignore the extra options for now
|
||||||
|
lua_pop(L, 2);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Top of stack should be the next option living in a table
|
||||||
|
if (lua_istable(L, -1)) {
|
||||||
|
|
||||||
|
// get the "kind"
|
||||||
|
lua_pushstring(L, "kind");
|
||||||
|
lua_gettable(L, -2);
|
||||||
|
if (lua_isstring(L, -1)) {
|
||||||
|
// use C standard lib functions here, as it's probably faster than messing around with unicode
|
||||||
|
// lua is known to always properly null-terminate strings, and the strings are known to be pure ascii
|
||||||
|
tmp = lua_tostring(L, -1);
|
||||||
|
if (strcmp(tmp, "label") == 0) {
|
||||||
|
opt->kind = COK_LABEL;
|
||||||
|
} else if (strcmp(tmp, "text") == 0) {
|
||||||
|
opt->kind = COK_TEXT;
|
||||||
|
} else if (strcmp(tmp, "int") == 0) {
|
||||||
|
opt->kind = COK_INT;
|
||||||
|
} else if (strcmp(tmp, "float") == 0) {
|
||||||
|
opt->kind = COK_FLOAT;
|
||||||
|
} else if (strcmp(tmp, "bool") == 0) {
|
||||||
|
opt->kind = COK_BOOL;
|
||||||
|
} else if (strcmp(tmp, "colour") == 0) {
|
||||||
|
opt->kind = COK_COLOUR;
|
||||||
|
} else if (strcmp(tmp, "style") == 0) {
|
||||||
|
opt->kind = COK_STYLE;
|
||||||
|
} else {
|
||||||
|
opt->kind = COK_INVALID;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
opt->kind = COK_INVALID;
|
||||||
|
}
|
||||||
|
|
||||||
|
// remove "kind" string from stack again
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
// no need to check for rest if this one is already deemed invalid
|
||||||
|
if (opt->kind != COK_INVALID) {
|
||||||
|
// name
|
||||||
|
lua_pushstring(L, "name");
|
||||||
|
lua_gettable(L, -2);
|
||||||
|
if (lua_isstring(L, -1)) {
|
||||||
|
opt->name = strdup(lua_tostring(L, -1));
|
||||||
|
} else {
|
||||||
|
// name is required to be valid
|
||||||
|
opt->kind = COK_INVALID;
|
||||||
|
}
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
// label
|
||||||
|
lua_pushstring(L, "label");
|
||||||
|
lua_gettable(L, -2);
|
||||||
|
if (lua_isstring(L, -1)) {
|
||||||
|
opt->label = strdup(lua_tostring(L, -1));
|
||||||
|
} else {
|
||||||
|
// label is also required
|
||||||
|
opt->kind = COK_INVALID;
|
||||||
|
}
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
// hint
|
||||||
|
lua_pushstring(L, "hint");
|
||||||
|
lua_gettable(L, -2);
|
||||||
|
if (lua_isstring(L, -1)) {
|
||||||
|
opt->hint = strdup(lua_tostring(L, -1));
|
||||||
|
} else {
|
||||||
|
opt->hint = strdup("");
|
||||||
|
}
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
// min
|
||||||
|
lua_pushstring(L, "min");
|
||||||
|
lua_gettable(L, -2);
|
||||||
|
if (lua_isnumber(L, -1)) {
|
||||||
|
opt->min.valid = 1;
|
||||||
|
opt->min.floatval = (float)lua_tonumber(L, -1);
|
||||||
|
opt->min.intval = (int)opt->min.floatval;
|
||||||
|
} else {
|
||||||
|
opt->min.valid = 0;
|
||||||
|
}
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
// max
|
||||||
|
lua_pushstring(L, "max");
|
||||||
|
lua_gettable(L, -2);
|
||||||
|
if (lua_isnumber(L, -1)) {
|
||||||
|
opt->max.valid = 1;
|
||||||
|
opt->max.floatval = (float)lua_tonumber(L, -1);
|
||||||
|
opt->max.intval = (int)opt->max.floatval;
|
||||||
|
} else {
|
||||||
|
opt->max.valid = 0;
|
||||||
|
}
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
// default
|
||||||
|
lua_pushstring(L, "default");
|
||||||
|
lua_gettable(L, -2);
|
||||||
|
switch (opt->kind) {
|
||||||
|
case COK_LABEL:
|
||||||
|
// nothing to do, nothing expected
|
||||||
|
break;
|
||||||
|
case COK_TEXT:
|
||||||
|
case COK_STYLE:
|
||||||
|
case COK_COLOUR:
|
||||||
|
// expect it to be a string
|
||||||
|
if (lua_isstring(L, -1)) {
|
||||||
|
opt->default_val.stringval = strdup(lua_tostring(L, -1));
|
||||||
|
} else {
|
||||||
|
// not a string, baaaad scripter
|
||||||
|
opt->kind = COK_INVALID;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case COK_INT:
|
||||||
|
// expect it to be a number
|
||||||
|
if (lua_isnumber(L, -1)) {
|
||||||
|
opt->default_val.intval = (int)lua_tonumber(L, -1);
|
||||||
|
} else {
|
||||||
|
opt->kind = COK_INVALID;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case COK_FLOAT:
|
||||||
|
// expect it to be a number
|
||||||
|
if (lua_isnumber(L, -1)) {
|
||||||
|
opt->default_val.floatval = (float)lua_tonumber(L, -1);
|
||||||
|
} else {
|
||||||
|
opt->kind = COK_INVALID;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case COK_BOOL:
|
||||||
|
// expect it to be a bool
|
||||||
|
if (lua_isboolean(L, -1)) {
|
||||||
|
opt->default_val.intval = lua_toboolean(L, -1);
|
||||||
|
} else {
|
||||||
|
opt->kind = COK_INVALID;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case COK_INVALID:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
memcpy(&opt->value, &opt->default_val, sizeof(opt->value));
|
||||||
|
lua_pop(L, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// On to next structure to be filled
|
||||||
|
opt++;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove option table from stack
|
||||||
|
lua_pop(L, 1);
|
||||||
|
// Such that the current key is on top, and we can get the next
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove 'config' table from stack
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Keeping this file a bit shorter: put all functions called from Lua into a separate file
|
||||||
|
#include "callables.c"
|
||||||
|
|
||||||
|
|
||||||
|
// Create a new interpreter
|
||||||
|
AUTO3_API struct Auto3Interpreter *CreateAuto3Script(filename_t filename, char *prettyname, struct Auto3Callbacks *cb, char **error)
|
||||||
|
{
|
||||||
|
struct Auto3Interpreter *script;
|
||||||
|
lua_State *L;
|
||||||
|
|
||||||
|
script = malloc(sizeof(struct Auto3Interpreter));
|
||||||
|
if (!script) return NULL;
|
||||||
|
// Copy in callbacks
|
||||||
|
memcpy(&script->cb, cb, sizeof(struct Auto3Callbacks));
|
||||||
|
|
||||||
|
|
||||||
|
// Init Lua
|
||||||
|
script->L = lua_open();
|
||||||
|
if (!script->L) goto failearly;
|
||||||
|
L = script->L;
|
||||||
|
|
||||||
|
|
||||||
|
// register standard libs
|
||||||
|
lua_pushcfunction(L, luaopen_base); lua_call(L, 0, 0);
|
||||||
|
lua_pushcfunction(L, luaopen_string); lua_call(L, 0, 0);
|
||||||
|
lua_pushcfunction(L, luaopen_table); lua_call(L, 0, 0);
|
||||||
|
lua_pushcfunction(L, luaopen_math); lua_call(L, 0, 0);
|
||||||
|
// dofile and loadfile are replaced with include
|
||||||
|
lua_pushnil(L);
|
||||||
|
lua_setglobal(L, "dofile");
|
||||||
|
lua_pushnil(L);
|
||||||
|
lua_setglobal(L, "loadfile");
|
||||||
|
lua_pushlightuserdata(L, script);
|
||||||
|
lua_pushcclosure(L, LuaInclude, 1);
|
||||||
|
lua_setglobal(L, "include");
|
||||||
|
|
||||||
|
// reference to the script object
|
||||||
|
lua_pushlightuserdata(L, script);
|
||||||
|
|
||||||
|
// make "aegisub" table
|
||||||
|
lua_newtable(L);
|
||||||
|
|
||||||
|
// put helper functions in it
|
||||||
|
// colorstring_to_rgb is moved to utils.auto3
|
||||||
|
lua_pushstring(L, "text_extents");
|
||||||
|
lua_pushvalue(L, -3);
|
||||||
|
lua_pushcclosure(L, LuaTextExtents, 1);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "frame_from_ms");
|
||||||
|
lua_pushvalue(L, -3);
|
||||||
|
lua_pushcclosure(L, LuaFrameFromMs, 1);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "ms_from_frame");
|
||||||
|
lua_pushvalue(L, -3);
|
||||||
|
lua_pushcclosure(L, LuaMsFromFrame, 1);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "report_progress");
|
||||||
|
lua_pushvalue(L, -3);
|
||||||
|
lua_pushcclosure(L, LuaReportProgress, 1);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "output_debug");
|
||||||
|
lua_pushvalue(L, -3);
|
||||||
|
lua_pushcclosure(L, LuaOutputDebug, 1);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "set_status");
|
||||||
|
lua_pushvalue(L, -3);
|
||||||
|
lua_pushcclosure(L, LuaSetStatus, 1);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "lua_automation_version");
|
||||||
|
lua_pushnumber(L, 3);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
// store table
|
||||||
|
lua_setglobal(L, "aegisub");
|
||||||
|
// remove ref to script object
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
|
||||||
|
// Read the script
|
||||||
|
if (Auto3LuaLoad(L, filename, prettyname, error)) {
|
||||||
|
// error is already filled
|
||||||
|
goto faillua;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Execute the script
|
||||||
|
if (lua_pcall(L, 0, 0, 0)) {
|
||||||
|
*error = strdup(lua_tostring(L, -1));
|
||||||
|
goto faillua;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Script has been run, stuff exists in the global environment
|
||||||
|
lua_getglobal(L, "version");
|
||||||
|
if (!lua_isnumber(L, -1)) {
|
||||||
|
*error = strdup("'version' value not found or not a number");
|
||||||
|
goto faillua;
|
||||||
|
}
|
||||||
|
if ((int)lua_tonumber(L, -1) != 3) {
|
||||||
|
// invalid version
|
||||||
|
*error = strdup("'version' must be 3 for Automation 3 scripts");
|
||||||
|
goto faillua;
|
||||||
|
}
|
||||||
|
// skip 'kind', it's useless
|
||||||
|
// name
|
||||||
|
lua_getglobal(L, "name");
|
||||||
|
if (!lua_isstring(L, -1)) {
|
||||||
|
script->name = strdup(prettyname);
|
||||||
|
} else {
|
||||||
|
script->name = strdup(lua_tostring(L, -1));
|
||||||
|
}
|
||||||
|
// description (optional)
|
||||||
|
lua_getglobal(L, "description");
|
||||||
|
if (lua_isstring(L, -1)) {
|
||||||
|
script->description = strdup(lua_tostring(L, -1));
|
||||||
|
} else {
|
||||||
|
script->description = strdup("");
|
||||||
|
}
|
||||||
|
lua_pop(L, 3);
|
||||||
|
|
||||||
|
|
||||||
|
// Parse the config data
|
||||||
|
lua_getglobal(L, "config");
|
||||||
|
if (Auto3ParseConfigData(L, script, error)) {
|
||||||
|
goto faildescription;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
return script;
|
||||||
|
|
||||||
|
|
||||||
|
// Various fail-cases
|
||||||
|
faildescription:
|
||||||
|
free(script->description);
|
||||||
|
free(script->name);
|
||||||
|
faillua:
|
||||||
|
lua_close(script->L);
|
||||||
|
failearly:
|
||||||
|
free(script);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// Release an interpreter
|
// Release an interpreter
|
||||||
AUTO3_API void DestroyAuto3Script(struct Auto3Interpreter *script)
|
AUTO3_API void DestroyAuto3Script(struct Auto3Interpreter *script)
|
||||||
{
|
{
|
||||||
|
struct Auto3ConfigOption *opt;
|
||||||
|
|
||||||
|
// free the config data
|
||||||
|
opt = script->config;
|
||||||
|
while (opt->name) {
|
||||||
|
free(opt->name);
|
||||||
|
free(opt->label);
|
||||||
|
free(opt->hint);
|
||||||
|
if (opt->kind == COK_TEXT || opt->kind == COK_COLOUR || opt->kind == COK_STYLE) {
|
||||||
|
free(opt->default_val.stringval);
|
||||||
|
free(opt->value.stringval);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
opt++;
|
||||||
|
}
|
||||||
|
free(script->config);
|
||||||
|
|
||||||
|
// free the rest
|
||||||
|
free(script->description);
|
||||||
|
free(script->name);
|
||||||
|
lua_close(script->L);
|
||||||
|
free(script);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// Our "malloc" function, allocate memory for strings with this
|
// Our "malloc" function, allocate memory for strings with this
|
||||||
AUTO3_API void *Auto3Malloc(size_t amount)
|
AUTO3_API void *Auto3Malloc(size_t amount)
|
||||||
{
|
{
|
||||||
|
@ -79,16 +527,402 @@ AUTO3_API void Auto3Free(void *ptr)
|
||||||
free(ptr);
|
free(ptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start the script execution
|
|
||||||
// script->logcbdata and log->rwcbdata should be set to sensible values before this call.
|
static void MakeMetaInfoTable(lua_State *L, struct Auto3Interpreter *script)
|
||||||
// The value fields in the config dialog should also be set to values entered by the user here.
|
|
||||||
// This will first call get_meta_info,
|
|
||||||
// then reset_style pointer followed by a number of calls to get_next_style,
|
|
||||||
// then a call to reset_subs_pointer followed by a number of calls to get_next_sub,
|
|
||||||
// then actual processing will take place.
|
|
||||||
// After processing, start_subs_write will be called, followed by a number of calls to write_sub.
|
|
||||||
// Any number of calls to the logging/status functions can take place during script execution
|
|
||||||
AUTO3_API void RunAuto3Script(struct Auto3Interpreter *script)
|
|
||||||
{
|
{
|
||||||
|
int res_x, res_y;
|
||||||
|
|
||||||
|
script->cb.get_meta_info(script->cb.rwdata, &res_x, &res_y);
|
||||||
|
|
||||||
|
lua_newtable(L);
|
||||||
|
|
||||||
|
lua_pushstring(L, "res_x");
|
||||||
|
lua_pushnumber(L, res_x);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "res_y");
|
||||||
|
lua_pushnumber(L, res_y);
|
||||||
|
lua_settable(L, -3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void MakeStylesTable(lua_State *L, struct Auto3Interpreter *script)
|
||||||
|
{
|
||||||
|
char *name, *fontname, *color1, *color2, *color3, *color4;
|
||||||
|
int fontsize, bold, italic, underline, strikeout, borderstyle, align, margin_l, margin_r, margin_v, encoding;
|
||||||
|
float scale_x, scale_y, spacing, angle, outline, shadow;
|
||||||
|
int n;
|
||||||
|
|
||||||
|
lua_newtable(L);
|
||||||
|
n = 0;
|
||||||
|
|
||||||
|
script->cb.reset_style_pointer(script->cb.rwdata);
|
||||||
|
while (script->cb.get_next_style(script->cb.rwdata, &name, &fontname, &fontsize, &color1, &color2, &color3, &color4,
|
||||||
|
&bold, &italic, &underline, &strikeout, &scale_x, &scale_y, &spacing, &angle, &borderstyle, &outline,
|
||||||
|
&shadow, &align, &margin_l, &margin_r, &margin_v, &encoding)) {
|
||||||
|
n++;
|
||||||
|
|
||||||
|
// Got a style...
|
||||||
|
lua_pushstring(L, name); // name for table index
|
||||||
|
lua_newtable(L);
|
||||||
|
|
||||||
|
// Set properties
|
||||||
|
|
||||||
|
lua_pushstring(L, "name");
|
||||||
|
lua_pushstring(L, name);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "fontname");
|
||||||
|
lua_pushstring(L, fontname);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "fontsize");
|
||||||
|
lua_pushnumber(L, fontsize);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "color1");
|
||||||
|
lua_pushstring(L, color1);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "color2");
|
||||||
|
lua_pushstring(L, color2);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "color3");
|
||||||
|
lua_pushstring(L, color3);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "color4");
|
||||||
|
lua_pushstring(L, color4);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "bold");
|
||||||
|
lua_pushboolean(L, bold);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "italic");
|
||||||
|
lua_pushboolean(L, italic);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "underline");
|
||||||
|
lua_pushboolean(L, underline);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "strikeout");
|
||||||
|
lua_pushboolean(L, strikeout);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "scale_x");
|
||||||
|
lua_pushnumber(L, scale_x);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "scale_y");
|
||||||
|
lua_pushnumber(L, scale_y);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "spacing");
|
||||||
|
lua_pushnumber(L, spacing);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "angle");
|
||||||
|
lua_pushnumber(L, angle);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "borderstyle");
|
||||||
|
lua_pushnumber(L, borderstyle);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "outline");
|
||||||
|
lua_pushnumber(L, outline);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "shadow");
|
||||||
|
lua_pushnumber(L, shadow);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "align");
|
||||||
|
lua_pushnumber(L, align);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "margin_l");
|
||||||
|
lua_pushnumber(L, margin_l);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "margin_r");
|
||||||
|
lua_pushnumber(L, margin_r);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "margin_v");
|
||||||
|
lua_pushnumber(L, margin_v);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "encoding");
|
||||||
|
lua_pushnumber(L, encoding);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
// Store to numeric index
|
||||||
|
lua_pushnumber(L, n);
|
||||||
|
lua_pushvalue(L, -2); // extra copy of table
|
||||||
|
lua_settable(L, -5);
|
||||||
|
// And named index
|
||||||
|
lua_settable(L, -3);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Finally, make 'n' key in table
|
||||||
|
lua_pushstring(L, "n");
|
||||||
|
lua_pushnumber(L, n);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void MakeEventsTable(lua_State *L, struct Auto3Interpreter *script)
|
||||||
|
{
|
||||||
|
int layer, start_time, end_time, margin_l, margin_r, margin_v, comment;
|
||||||
|
char *style, *actor, *effect, *text;
|
||||||
|
int n;
|
||||||
|
|
||||||
|
lua_newtable(L);
|
||||||
|
n = 0;
|
||||||
|
|
||||||
|
script->cb.reset_subs_pointer(script->cb.rwdata);
|
||||||
|
while (script->cb.get_next_sub(script->cb.rwdata, &layer, &start_time, &end_time, &style, &actor,
|
||||||
|
&margin_l, &margin_r, &margin_v, &effect, &text, &comment)) {
|
||||||
|
n++;
|
||||||
|
|
||||||
|
// Got a line...
|
||||||
|
lua_pushnumber(L, n);
|
||||||
|
lua_newtable(L);
|
||||||
|
|
||||||
|
lua_pushstring(L, "kind");
|
||||||
|
if (comment)
|
||||||
|
lua_pushstring(L, "comment");
|
||||||
|
else
|
||||||
|
lua_pushstring(L, "dialogue");
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "layer");
|
||||||
|
lua_pushnumber(L, layer);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "start_time");
|
||||||
|
lua_pushnumber(L, start_time);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "end_time");
|
||||||
|
lua_pushnumber(L, end_time);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "style");
|
||||||
|
lua_pushstring(L, style);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "name");
|
||||||
|
lua_pushstring(L, actor);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "margin_l");
|
||||||
|
lua_pushnumber(L, margin_l);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "margin_r");
|
||||||
|
lua_pushnumber(L, margin_r);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "margin_v");
|
||||||
|
lua_pushnumber(L, margin_v);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "effect");
|
||||||
|
lua_pushstring(L, effect);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
lua_pushstring(L, "text");
|
||||||
|
lua_pushstring(L, text);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
// No parsing karaoke data here, that can just as well be done in Lua code
|
||||||
|
|
||||||
|
// Store at numeric index
|
||||||
|
lua_settable(L, -3);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Finally, make 'n' key in table
|
||||||
|
lua_pushstring(L, "n");
|
||||||
|
lua_pushnumber(L, n);
|
||||||
|
lua_settable(L, -3);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void MakeConfigSettingsTable(lua_State *L, struct Auto3Interpreter *script)
|
||||||
|
{
|
||||||
|
struct Auto3ConfigOption *opt;
|
||||||
|
|
||||||
|
lua_newtable(L);
|
||||||
|
|
||||||
|
opt = script->config;
|
||||||
|
while (opt->name) {
|
||||||
|
lua_pushstring(L, opt->name);
|
||||||
|
|
||||||
|
switch (opt->kind) {
|
||||||
|
case COK_TEXT:
|
||||||
|
case COK_STYLE:
|
||||||
|
case COK_COLOUR:
|
||||||
|
lua_pushstring(L, opt->value.stringval);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case COK_INT:
|
||||||
|
lua_pushnumber(L, opt->value.intval);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case COK_FLOAT:
|
||||||
|
lua_pushnumber(L, opt->value.floatval);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case COK_BOOL:
|
||||||
|
lua_pushboolean(L, opt->value.intval);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
lua_pushnil(L);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_settable(L, -3);
|
||||||
|
|
||||||
|
opt++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void ReadBackSubs(lua_State *L, struct Auto3Interpreter *script)
|
||||||
|
{
|
||||||
|
int layer, start_time, end_time, margin_l, margin_r, margin_v, comment;
|
||||||
|
const char *style, *actor, *effect, *text, *kind;
|
||||||
|
int i, n;
|
||||||
|
|
||||||
|
script->cb.start_subs_write(script->cb.rwdata);
|
||||||
|
|
||||||
|
i = 0;
|
||||||
|
n = luaL_getn(L, -1);
|
||||||
|
while (i <= n) {
|
||||||
|
// Assume n is always correct, this is not entirely compatible
|
||||||
|
lua_rawgeti(L, -1, i);
|
||||||
|
if (script->cb.set_progress) script->cb.set_progress(script->cb.logdata, 100.f * i / n);
|
||||||
|
i++;
|
||||||
|
if (!lua_istable(L, -1)) {
|
||||||
|
lua_pop(L, 1);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_pushstring(L, "kind");
|
||||||
|
lua_gettable(L, -2);
|
||||||
|
if (!lua_isstring(L, -1)) {
|
||||||
|
lua_pop(L, 2);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
kind = lua_tostring(L, -1);
|
||||||
|
// leave kind on stack so it won't be gc'd
|
||||||
|
|
||||||
|
// Make comment rather tell if this is a "non-dialogue" line
|
||||||
|
comment = strcmp(kind, "dialogue");
|
||||||
|
// then test if it's a dialogue line (not non-dialogue) or is an actual comment
|
||||||
|
if (!comment || strcmp(kind, "comment")) {
|
||||||
|
|
||||||
|
lua_pushstring(L, "layer");
|
||||||
|
lua_gettable(L, -3);
|
||||||
|
lua_pushstring(L, "start_time");
|
||||||
|
lua_gettable(L, -4);
|
||||||
|
lua_pushstring(L, "end_time");
|
||||||
|
lua_gettable(L, -5);
|
||||||
|
lua_pushstring(L, "style");
|
||||||
|
lua_gettable(L, -6);
|
||||||
|
lua_pushstring(L, "name");
|
||||||
|
lua_gettable(L, -7);
|
||||||
|
lua_pushstring(L, "margin_l");
|
||||||
|
lua_gettable(L, -8);
|
||||||
|
lua_pushstring(L, "margin_r");
|
||||||
|
lua_gettable(L, -9);
|
||||||
|
lua_pushstring(L, "margin_v");
|
||||||
|
lua_gettable(L, -10);
|
||||||
|
lua_pushstring(L, "effect");
|
||||||
|
lua_gettable(L, -11);
|
||||||
|
lua_pushstring(L, "text");
|
||||||
|
lua_gettable(L, -12);
|
||||||
|
|
||||||
|
if (lua_isnumber(L, -10) && lua_isnumber(L, -9) && lua_isnumber(L, -8) &&
|
||||||
|
lua_isstring(L, -7) && lua_isstring(L, -6) && lua_isnumber(L, -5) &&
|
||||||
|
lua_isnumber(L, -4) && lua_isnumber(L, -3) && lua_isstring(L, -2) &&
|
||||||
|
lua_isstring(L, -1)) {
|
||||||
|
layer = (int)lua_tonumber(L, -10);
|
||||||
|
start_time = (int)lua_tonumber(L, -9);
|
||||||
|
end_time = (int)lua_tonumber(L, -8);
|
||||||
|
style = lua_tostring(L, -7);
|
||||||
|
actor = lua_tostring(L, -6);
|
||||||
|
margin_l = (int)lua_tonumber(L, -5);
|
||||||
|
margin_r = (int)lua_tonumber(L, -4);
|
||||||
|
margin_v = (int)lua_tonumber(L, -3);
|
||||||
|
effect = lua_tostring(L, -2);
|
||||||
|
text = lua_tostring(L, -1);
|
||||||
|
|
||||||
|
script->cb.write_sub(script->cb.rwdata, layer, start_time, end_time, style, actor,
|
||||||
|
margin_l, margin_r, margin_v, effect, text, comment);
|
||||||
|
|
||||||
|
} else {
|
||||||
|
if (script->cb.log_error) script->cb.log_error(script->cb.logdata, "Skipping output line with invalid fields");
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_pop(L, 10);
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_pop(L, 2); // pop line and 'kind'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Start the script execution
|
||||||
|
AUTO3_API int RunAuto3Script(struct Auto3Interpreter *script)
|
||||||
|
{
|
||||||
|
lua_State *L;
|
||||||
|
|
||||||
|
L = script->L;
|
||||||
|
|
||||||
|
if (script->cb.set_status) script->cb.set_status(script->cb.logdata, "Preparing subtitle data");
|
||||||
|
if (script->cb.set_progress) script->cb.set_progress(script->cb.logdata, 0);
|
||||||
|
|
||||||
|
// first put the function itself on the stack
|
||||||
|
lua_getglobal(L, "process_lines");
|
||||||
|
|
||||||
|
// now put the four arguments on the stack
|
||||||
|
MakeMetaInfoTable(L, script);
|
||||||
|
MakeStylesTable(L, script);
|
||||||
|
MakeEventsTable(L, script);
|
||||||
|
MakeConfigSettingsTable(L, script);
|
||||||
|
|
||||||
|
// do the actual call
|
||||||
|
if (script->cb.set_status) script->cb.set_status(script->cb.logdata, "Running script");
|
||||||
|
if (lua_pcall(L, 4, 1, 0)) {
|
||||||
|
if (script->cb.log_error) {
|
||||||
|
script->cb.log_error(script->cb.logdata, "Runtime error in script:");
|
||||||
|
script->cb.log_error(script->cb.logdata, lua_tostring(L, -1));
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check for initial sanity
|
||||||
|
if (!lua_istable(L, -1)) {
|
||||||
|
if (script->cb.log_error) script->cb.log_error(script->cb.logdata, "Script did not return a table, unable to process result");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read back subtitles
|
||||||
|
if (script->cb.set_status) script->cb.set_status(script->cb.logdata, "Reading back subtitle data");
|
||||||
|
if (script->cb.set_progress) script->cb.set_progress(script->cb.logdata, 0);
|
||||||
|
ReadBackSubs(L, script);
|
||||||
|
|
||||||
|
// Finished
|
||||||
|
if (script->cb.set_progress) script->cb.set_progress(script->cb.logdata, 100);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
111
auto3/auto3.h
111
auto3/auto3.h
|
@ -37,6 +37,8 @@
|
||||||
|
|
||||||
#ifdef AUTO3LIB
|
#ifdef AUTO3LIB
|
||||||
#include "lua/include/lua.h"
|
#include "lua/include/lua.h"
|
||||||
|
#include "lua/include/lualib.h"
|
||||||
|
#include "lua/include/lauxlib.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
@ -45,7 +47,7 @@ extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
// On Win32, filenames are wide, but UTF-8 everywhere else
|
// On Win32, filenames are wide, but whatever encoding the system uses everywhere else
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
typedef wchar_t* filename_t;
|
typedef wchar_t* filename_t;
|
||||||
#else
|
#else
|
||||||
|
@ -82,9 +84,8 @@ struct Auto3ConfigOption {
|
||||||
enum Auto3ConfigOptionKind kind;
|
enum Auto3ConfigOptionKind kind;
|
||||||
char *label;
|
char *label;
|
||||||
char *hint;
|
char *hint;
|
||||||
int hasmin : 1; // whether there is a min value
|
struct {
|
||||||
int hasmax : 1; // whether there is a max value
|
int valid; // non-zero if the value is present
|
||||||
union {
|
|
||||||
int intval;
|
int intval;
|
||||||
float floatval;
|
float floatval;
|
||||||
} min, max;
|
} min, max;
|
||||||
|
@ -96,10 +97,68 @@ struct Auto3ConfigOption {
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// Callback interface
|
||||||
|
// The application should provide ALL of these functions
|
||||||
|
struct Auto3Callbacks {
|
||||||
|
// Logging and status
|
||||||
|
// pointer passed to logging/status callbacks
|
||||||
|
void *logdata;
|
||||||
|
// log error during script execution
|
||||||
|
void (*log_error)(void *cbdata, const char *msg);
|
||||||
|
// log message during script execution
|
||||||
|
void (*log_message)(void *cbdata, const char *msg);
|
||||||
|
// set progress during script execution
|
||||||
|
void (*set_progress)(void *cbdata, float progress);
|
||||||
|
// set status message during script execution
|
||||||
|
void (*set_status)(void *cbdata, const char* msg);
|
||||||
|
|
||||||
|
// Reading/writing subtitles and related information
|
||||||
|
// pointer passed to read/write data callbacks
|
||||||
|
void *rwdata;
|
||||||
|
// application sets *res_x and *res_y to appropriate values
|
||||||
|
void (*get_meta_info)(void *cbdata, int *res_x, int *res_y);
|
||||||
|
// set style pointer to point at first style
|
||||||
|
void (*reset_style_pointer)(void *cbdata);
|
||||||
|
// Get the next style, the application must fill the data into its own buffers, which it then fill in pointers to
|
||||||
|
// (Ie. the application owns all strings allocated for this callback.)
|
||||||
|
// Return non-zero if a style was found and values filled, otherwise return zero
|
||||||
|
int (*get_next_style)(
|
||||||
|
void *cbdata, char **name, char **fontname, int *fontsize, char **color1, char **color2, char **color3, char **color4,
|
||||||
|
int *bold, int *italic, int *underline, int *strikeout, float *scale_x, float *scale_y, float *spacing, float *angle,
|
||||||
|
int *borderstyle, float *outline, float *shadow, int *align, int *margin_l, int *margin_r, int *margin_v, int *encoding);
|
||||||
|
// set subtitle pointer to point at first subtitle line
|
||||||
|
void (*reset_subs_pointer)(void *cbdata);
|
||||||
|
// Get next subtitle line, the application must fill the data into its own buffers, and then fill in pointers to those
|
||||||
|
// Return non-zero if a style was found and values filled, otherwise return zero
|
||||||
|
int (*get_next_sub)(
|
||||||
|
void *cbdata, int *layer, int *start_time, int *end_time, char **style, char **actor,
|
||||||
|
int *margin_l, int *margin_r, int *margin_v, char **effect, char **text, int *comment);
|
||||||
|
// start writing back new subtitles, application must clear all subtitle lines and be ready to write
|
||||||
|
void (*start_subs_write)(void *cbdata);
|
||||||
|
// Write a subtitle line back to subtitle file, char pointers are owned by the lib
|
||||||
|
void (*write_sub)(void *cbdata, int layer, int start_time, int end_time, const char *style, const char *actor,
|
||||||
|
int margin_l, int margin_r, int margin_v, const char *effect, const char *text, int comment);
|
||||||
|
|
||||||
|
// Getting various environment information during runtime
|
||||||
|
// pointer passed to runtime data callbacks
|
||||||
|
void *rundata;
|
||||||
|
// Resolve a filename passed to the include function
|
||||||
|
// The result must be allocated with Auto3Malloc and will be free'd by the lib
|
||||||
|
filename_t (*resolve_include)(void *cbdata, const char *incname);
|
||||||
|
// Get sizing information for a text string given a style
|
||||||
|
void (*text_extents)(void *cbdata, char *text, char *fontname, int fontsize, int bold, int italic,
|
||||||
|
int spacing, float scale_x, float scale_y, int encoding,
|
||||||
|
float *out_width, float *out_height, float *out_descent, float *out_extlead);
|
||||||
|
// Convert a time in milliseconds to a video frame number
|
||||||
|
int (*frame_from_ms)(void *cbdata, int ms);
|
||||||
|
// Convert a video frame number to a time in milliseconds
|
||||||
|
int (*ms_from_frame)(void *cbdata, int frame);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
// Describes an interpreter
|
// Describes an interpreter
|
||||||
struct Auto3Interpreter {
|
struct Auto3Interpreter {
|
||||||
// Public attributes, treat them as read-only
|
// Public attributes, treat them as read-only
|
||||||
filename_t filename;
|
|
||||||
char *name;
|
char *name;
|
||||||
char *description;
|
char *description;
|
||||||
|
|
||||||
|
@ -108,34 +167,10 @@ struct Auto3Interpreter {
|
||||||
// You may change the "value" field of these (in fact, do so)
|
// You may change the "value" field of these (in fact, do so)
|
||||||
struct Auto3ConfigOption *config;
|
struct Auto3ConfigOption *config;
|
||||||
|
|
||||||
// Callback stuff
|
// Callbacks
|
||||||
// The application should fill these with relevant stuff
|
// This is filled in from the 'cb' argument to the Create function,
|
||||||
// Logging and status
|
// but may be modified later by the application
|
||||||
void *logcbdata; // pointer passed to logging/status callbacks
|
struct Auto3Callbacks cb;
|
||||||
void (*log_error)(void *cbdata, char *msg); // log error during script execution
|
|
||||||
void (*log_message)(void *cbdata, char *msg); // log message during script execution
|
|
||||||
void (*set_progress)(void *cbdata, float progress); // set progress during script execution
|
|
||||||
void (*set_status)(void *cbdata, char* msg); // set status message during script execution
|
|
||||||
// Reading/writing subtitles and related information
|
|
||||||
void *rwcbdata; // pointer passed to read/write data callbacks
|
|
||||||
void (*get_meta_info)(void *cbdata, int *res_x, int *res_y); // application sets *res_x and *res_y to appropriate values
|
|
||||||
void (*reset_style_pointer)(void *cbdata); // set style pointer to point at first style
|
|
||||||
// Get the next style, the application must fill the data into its own buffers, which it then fill in pointers to
|
|
||||||
// When there are no more styles, set *name=NULL
|
|
||||||
void (*get_next_style)(
|
|
||||||
void *cbdata, char **name, char **fontname, int *fontsize, char **color1, char **color2, char **color3, char **color4,
|
|
||||||
int *bold, int *italic, int *underline, int *strikeout, float *scale_x, float *scale_y, float *spacing, float *angle,
|
|
||||||
int *borderstyle, float *outline, float *shadow, int *align, int *margin_l, int *margin_r, int *margin_v, int *encoding);
|
|
||||||
void (*reset_subs_pointer)(void *cbdata); // set subtitle pointer to point at first subtitle line
|
|
||||||
// Get next subtitle line, the application must fill the data into its own buffers, and then fill in pointers to those
|
|
||||||
// When there are no more lines, set *text=NULL
|
|
||||||
void (*get_next_sub)(
|
|
||||||
void *cbdata, int *layer, int *start_time, int *end_time, char **style, char **actor,
|
|
||||||
int *margin_l, int *margin_r, int *margin_v, char **effect, char **text);
|
|
||||||
void (*start_subs_write)(void *cbdata); // start writing back new subtitles, application must clear all subtitle lines and be ready to write
|
|
||||||
// Write a subtitle line back to subtitle file, char pointers are owned by the lib
|
|
||||||
void (*write_sub)(void *cbdata, int layer, int start_time, int end_time, char *style, char *actor,
|
|
||||||
int margin_l, int margin_r, int margin_v, char *effect, char *text);
|
|
||||||
|
|
||||||
#ifdef AUTO3LIB
|
#ifdef AUTO3LIB
|
||||||
// Private data
|
// Private data
|
||||||
|
@ -145,9 +180,12 @@ struct Auto3Interpreter {
|
||||||
|
|
||||||
|
|
||||||
// Create a new interpreter
|
// Create a new interpreter
|
||||||
|
// filename is name of script file
|
||||||
|
// prettyname is a UTF-8 string used as identifier for the script in error messages
|
||||||
|
// cb should point to an Auto3Callbacks struct filled in; a copy of this struct will be made
|
||||||
|
// error will be filled with any error message on fail, the application is responsible for freeing this string (use Auto3Free)
|
||||||
// Returns pointer to interpreter object if successful, otherwise NULL
|
// Returns pointer to interpreter object if successful, otherwise NULL
|
||||||
// If this function fails, the error message is filled into the char* pointed to by error (may be NULL)
|
AUTO3_API struct Auto3Interpreter *CreateAuto3Script(filename_t filename, char *prettyname, struct Auto3Callbacks *cb, char **error);
|
||||||
AUTO3_API struct Auto3Interpreter *CreateAuto3Script(filename_t filename, char **error);
|
|
||||||
// Release an interpreter
|
// Release an interpreter
|
||||||
AUTO3_API void DestroyAuto3Script(struct Auto3Interpreter *script);
|
AUTO3_API void DestroyAuto3Script(struct Auto3Interpreter *script);
|
||||||
|
|
||||||
|
@ -165,7 +203,8 @@ AUTO3_API void Auto3Free(void *ptr);
|
||||||
// then actual processing will take place.
|
// then actual processing will take place.
|
||||||
// After processing, start_subs_write will be called, followed by a number of calls to write_sub.
|
// After processing, start_subs_write will be called, followed by a number of calls to write_sub.
|
||||||
// Any number of calls to the logging/status functions can take place during script execution
|
// Any number of calls to the logging/status functions can take place during script execution
|
||||||
AUTO3_API void RunAuto3Script(struct Auto3Interpreter *script);
|
// Returns non-zero on error
|
||||||
|
AUTO3_API int RunAuto3Script(struct Auto3Interpreter *script);
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
|
|
|
@ -33,12 +33,12 @@
|
||||||
<Tool
|
<Tool
|
||||||
Name="VCLinkerTool"
|
Name="VCLinkerTool"
|
||||||
AdditionalDependencies="lua/lib/luad.lib"
|
AdditionalDependencies="lua/lib/luad.lib"
|
||||||
OutputFile="$(OutDir)/auto3.dll"
|
OutputFile="../bin/aegisub-auto3d.dll"
|
||||||
LinkIncremental="2"
|
LinkIncremental="2"
|
||||||
GenerateDebugInformation="TRUE"
|
GenerateDebugInformation="TRUE"
|
||||||
ProgramDatabaseFile="$(OutDir)/auto3.pdb"
|
ProgramDatabaseFile="../bin/aegisub-auto3d.pdb"
|
||||||
SubSystem="2"
|
SubSystem="2"
|
||||||
ImportLibrary="$(OutDir)/auto3.lib"
|
ImportLibrary="../lib/aegisub-auto3d.lib"
|
||||||
TargetMachine="1"/>
|
TargetMachine="1"/>
|
||||||
<Tool
|
<Tool
|
||||||
Name="VCMIDLTool"/>
|
Name="VCMIDLTool"/>
|
||||||
|
@ -66,9 +66,17 @@
|
||||||
OutputDirectory="Release"
|
OutputDirectory="Release"
|
||||||
IntermediateDirectory="Release"
|
IntermediateDirectory="Release"
|
||||||
ConfigurationType="2"
|
ConfigurationType="2"
|
||||||
CharacterSet="1">
|
CharacterSet="1"
|
||||||
|
WholeProgramOptimization="TRUE">
|
||||||
<Tool
|
<Tool
|
||||||
Name="VCCLCompilerTool"
|
Name="VCCLCompilerTool"
|
||||||
|
Optimization="3"
|
||||||
|
GlobalOptimizations="TRUE"
|
||||||
|
InlineFunctionExpansion="2"
|
||||||
|
EnableIntrinsicFunctions="TRUE"
|
||||||
|
FavorSizeOrSpeed="2"
|
||||||
|
OptimizeForProcessor="2"
|
||||||
|
OptimizeForWindowsApplication="TRUE"
|
||||||
AdditionalIncludeDirectories=""
|
AdditionalIncludeDirectories=""
|
||||||
PreprocessorDefinitions="WIN32;NDEBUG;_WINDOWS;AUTO3LIB;WXUSINGDLL"
|
PreprocessorDefinitions="WIN32;NDEBUG;_WINDOWS;AUTO3LIB;WXUSINGDLL"
|
||||||
RuntimeLibrary="0"
|
RuntimeLibrary="0"
|
||||||
|
@ -81,15 +89,16 @@
|
||||||
Name="VCCustomBuildTool"/>
|
Name="VCCustomBuildTool"/>
|
||||||
<Tool
|
<Tool
|
||||||
Name="VCLinkerTool"
|
Name="VCLinkerTool"
|
||||||
AdditionalDependencies="wxbase28u.lib wxbase28u_net.lib wxmsw28u_adv.lib wxmsw28u_core.lib wxmsw28u_richtext.lib wxpng.lib wxregexu.lib wxzlib.lib lua/lib/lua.lib"
|
AdditionalDependencies="lua/lib/lua.lib"
|
||||||
ShowProgress="2"
|
ShowProgress="2"
|
||||||
OutputFile="$(OutDir)/auto3.dll"
|
OutputFile="../bin/aegisub-auto3.dll"
|
||||||
LinkIncremental="1"
|
LinkIncremental="1"
|
||||||
GenerateDebugInformation="TRUE"
|
GenerateDebugInformation="TRUE"
|
||||||
SubSystem="2"
|
SubSystem="2"
|
||||||
OptimizeReferences="2"
|
OptimizeReferences="2"
|
||||||
EnableCOMDATFolding="2"
|
EnableCOMDATFolding="2"
|
||||||
ImportLibrary="$(OutDir)/auto3.lib"
|
SetChecksum="TRUE"
|
||||||
|
ImportLibrary="../lib/aegisub-auto3.lib"
|
||||||
TargetMachine="1"/>
|
TargetMachine="1"/>
|
||||||
<Tool
|
<Tool
|
||||||
Name="VCMIDLTool"/>
|
Name="VCMIDLTool"/>
|
||||||
|
@ -123,6 +132,21 @@
|
||||||
<File
|
<File
|
||||||
RelativePath=".\auto3.c">
|
RelativePath=".\auto3.c">
|
||||||
</File>
|
</File>
|
||||||
|
<File
|
||||||
|
RelativePath=".\callables.c">
|
||||||
|
<FileConfiguration
|
||||||
|
Name="Debug|Win32"
|
||||||
|
ExcludedFromBuild="TRUE">
|
||||||
|
<Tool
|
||||||
|
Name="VCCLCompilerTool"/>
|
||||||
|
</FileConfiguration>
|
||||||
|
<FileConfiguration
|
||||||
|
Name="Release|Win32"
|
||||||
|
ExcludedFromBuild="TRUE">
|
||||||
|
<Tool
|
||||||
|
Name="VCCLCompilerTool"/>
|
||||||
|
</FileConfiguration>
|
||||||
|
</File>
|
||||||
</Filter>
|
</Filter>
|
||||||
<Filter
|
<Filter
|
||||||
Name="Header Files"
|
Name="Header Files"
|
||||||
|
|
|
@ -0,0 +1,149 @@
|
||||||
|
// Copyright (c) 2005, 2006, 2007, 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
|
||||||
|
//
|
||||||
|
// Website: http://aegisub.cellosoft.com
|
||||||
|
// Contact: mailto:jiifurusu@gmail.com
|
||||||
|
//
|
||||||
|
|
||||||
|
// DO NOT compile this file separately! It's included as part of auto3.c
|
||||||
|
|
||||||
|
|
||||||
|
static struct Auto3Interpreter* GetScriptObject(lua_State *L)
|
||||||
|
{
|
||||||
|
return (struct Auto3Interpreter *)lua_touserdata(L, lua_upvalueindex(1));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int LuaInclude(lua_State *L)
|
||||||
|
{
|
||||||
|
filename_t filename;
|
||||||
|
const char *incname;
|
||||||
|
char *error;
|
||||||
|
struct Auto3Interpreter *script;
|
||||||
|
script = GetScriptObject(L);
|
||||||
|
|
||||||
|
if (!script->cb.resolve_include) {
|
||||||
|
lua_pushstring(L, "Attempt to use include, but not implemented by host application");
|
||||||
|
lua_error(L);
|
||||||
|
}
|
||||||
|
|
||||||
|
incname = luaL_checkstring(L, 1);
|
||||||
|
|
||||||
|
filename = script->cb.resolve_include(script->cb.rundata, incname);
|
||||||
|
|
||||||
|
if (filename) {
|
||||||
|
// Load include
|
||||||
|
if (Auto3LuaLoad(L, filename, incname, &error)) {
|
||||||
|
lua_pushfstring(L, "Failed to include file '%s', error: %s", incname, error);
|
||||||
|
lua_error(L);
|
||||||
|
}
|
||||||
|
// Run include (don't protect, we're already in a protected environment, we'd just propagate it anyway)
|
||||||
|
lua_call(L, 0, 0);
|
||||||
|
} else {
|
||||||
|
lua_pushfstring(L, "Failed to resolve include file '%s'", incname);
|
||||||
|
lua_error(L);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Not really compatible, but I don't think anyone have ever exploited that includes can return stuff
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int LuaTextExtents(lua_State *L)
|
||||||
|
{
|
||||||
|
struct Auto3Interpreter *script;
|
||||||
|
script = GetScriptObject(L);
|
||||||
|
// TODO
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int LuaFrameFromMs(lua_State *L)
|
||||||
|
{
|
||||||
|
int frame;
|
||||||
|
struct Auto3Interpreter *script;
|
||||||
|
script = GetScriptObject(L);
|
||||||
|
|
||||||
|
if (!script->cb.frame_from_ms) return 0;
|
||||||
|
|
||||||
|
frame = script->cb.frame_from_ms(script->cb.rundata, luaL_checkint(L, 1));
|
||||||
|
lua_pushnumber(L, frame);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int LuaMsFromFrame(lua_State *L)
|
||||||
|
{
|
||||||
|
int ms;
|
||||||
|
struct Auto3Interpreter *script;
|
||||||
|
script = GetScriptObject(L);
|
||||||
|
|
||||||
|
if (!script->cb.ms_from_frame) return 0;
|
||||||
|
|
||||||
|
ms = script->cb.ms_from_frame(script->cb.rundata, luaL_checkint(L, 1));
|
||||||
|
lua_pushnumber(L, ms);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int LuaReportProgress(lua_State *L)
|
||||||
|
{
|
||||||
|
struct Auto3Interpreter *script;
|
||||||
|
script = GetScriptObject(L);
|
||||||
|
|
||||||
|
if (script->cb.set_progress) script->cb.set_progress(script->cb.logdata, (float)luaL_checknumber(L, 1));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int LuaOutputDebug(lua_State *L)
|
||||||
|
{
|
||||||
|
struct Auto3Interpreter *script;
|
||||||
|
script = GetScriptObject(L);
|
||||||
|
|
||||||
|
if (script->cb.log_message) script->cb.log_message(script->cb.logdata, luaL_checkstring(L, 1));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int LuaSetStatus(lua_State *L)
|
||||||
|
{
|
||||||
|
struct Auto3Interpreter *script;
|
||||||
|
script = GetScriptObject(L);
|
||||||
|
|
||||||
|
if (script->cb.set_status) script->cb.set_status(script->cb.logdata, luaL_checkstring(L, 1));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -60,6 +60,12 @@
|
||||||
CharacterSet="2">
|
CharacterSet="2">
|
||||||
<Tool
|
<Tool
|
||||||
Name="VCCLCompilerTool"
|
Name="VCCLCompilerTool"
|
||||||
|
Optimization="3"
|
||||||
|
GlobalOptimizations="TRUE"
|
||||||
|
InlineFunctionExpansion="2"
|
||||||
|
EnableIntrinsicFunctions="TRUE"
|
||||||
|
OptimizeForProcessor="2"
|
||||||
|
OptimizeForWindowsApplication="TRUE"
|
||||||
AdditionalIncludeDirectories="include"
|
AdditionalIncludeDirectories="include"
|
||||||
PreprocessorDefinitions="WIN32;NDEBUG;_LIB"
|
PreprocessorDefinitions="WIN32;NDEBUG;_LIB"
|
||||||
RuntimeLibrary="0"
|
RuntimeLibrary="0"
|
||||||
|
|
Loading…
Reference in New Issue