eBookReaderSwitch/source/pdf/pdf-js.c

976 lines
21 KiB
C

#include "mupdf/fitz.h"
#include "mupdf/pdf.h"
#if FZ_ENABLE_JS
#include "mujs.h"
#include <stdarg.h>
#include <string.h>
struct pdf_js_s
{
fz_context *ctx;
pdf_document *doc;
pdf_obj *form;
js_State *imp;
};
FZ_NORETURN static void rethrow(pdf_js *js)
{
js_newerror(js->imp, fz_caught_message(js->ctx));
js_throw(js->imp);
}
/* Unpack argument object with named arguments into actual parameters. */
static pdf_js *unpack_arguments(js_State *J, ...)
{
if (js_isobject(J, 1))
{
int i = 1;
va_list args;
js_copy(J, 1);
va_start(args, J);
for (;;)
{
const char *s = va_arg(args, const char *);
if (!s)
break;
js_getproperty(J, -1, s);
js_replace(J, i++);
}
va_end(args);
js_pop(J, 1);
}
return js_getcontext(J);
}
static void app_alert(js_State *J)
{
pdf_js *js = unpack_arguments(J, "cMsg", "nIcon", "nType", "cTitle", 0);
pdf_alert_event event;
event.message = js_tostring(J, 1);
event.icon_type = js_tointeger(J, 2);
event.button_group_type = js_tointeger(J, 3);
event.title = js_isdefined(J, 4) ? js_tostring(J, 4) : "PDF Alert";
event.button_pressed = 0; /* WIP WIP WIP IS THIS CORRECT? */
fz_try(js->ctx)
pdf_event_issue_alert(js->ctx, js->doc, &event);
fz_catch(js->ctx)
rethrow(js);
js_pushnumber(J, event.button_pressed);
}
static void app_execMenuItem(js_State *J)
{
pdf_js *js = js_getcontext(J);
const char *cMenuItem = js_tostring(J, 1);
fz_try(js->ctx)
pdf_event_issue_exec_menu_item(js->ctx, js->doc, cMenuItem);
fz_catch(js->ctx)
rethrow(js);
}
static void app_launchURL(js_State *J)
{
pdf_js *js = js_getcontext(J);
const char *cUrl = js_tostring(J, 1);
int bNewFrame = js_toboolean(J, 1);
fz_try(js->ctx)
pdf_event_issue_launch_url(js->ctx, js->doc, cUrl, bNewFrame);
fz_catch(js->ctx)
rethrow(js);
}
static void field_finalize(js_State *J, void *p)
{
pdf_js *js = js_getcontext(J);
pdf_drop_obj(js->ctx, p);
}
static void field_buttonSetCaption(js_State *J)
{
pdf_js *js = js_getcontext(J);
pdf_obj *field = js_touserdata(J, 0, "Field");
const char *cCaption = js_tostring(J, 1);
fz_try(js->ctx)
pdf_field_set_button_caption(js->ctx, field, cCaption);
fz_catch(js->ctx)
rethrow(js);
}
static void field_getName(js_State *J)
{
pdf_js *js = js_getcontext(J);
pdf_obj *field = js_touserdata(J, 0, "Field");
char *name = NULL;
fz_try(js->ctx)
name = pdf_field_name(js->ctx, field);
fz_catch(js->ctx)
rethrow(js);
js_pushstring(J, name);
}
static void field_setName(js_State *J)
{
pdf_js *js = js_getcontext(J);
fz_warn(js->ctx, "Unexpected call to field_setName");
}
static void field_getDisplay(js_State *J)
{
pdf_js *js = js_getcontext(J);
pdf_obj *field = js_touserdata(J, 0, "Field");
int display = 0;
fz_try(js->ctx)
display = pdf_field_display(js->ctx, field);
fz_catch(js->ctx)
rethrow(js);
js_pushnumber(J, display);
}
static void field_setDisplay(js_State *J)
{
pdf_js *js = js_getcontext(J);
pdf_obj *field = js_touserdata(J, 0, "Field");
int display = js_tonumber(J, 1);
fz_try(js->ctx)
pdf_field_set_display(js->ctx, field, display);
fz_catch(js->ctx)
rethrow(js);
}
static pdf_obj *load_color(pdf_js *js, int idx)
{
fz_context *ctx = js->ctx;
pdf_document *doc = js->doc;
js_State *J = js->imp;
pdf_obj *color = NULL;
int i, n;
float c;
n = js_getlength(J, idx);
/* The only legitimate color expressed as an array of length 1
* is [T], meaning transparent. Return a NULL object to represent
* transparent */
if (n <= 1)
return NULL;
fz_var(color);
fz_try(ctx)
{
color = pdf_new_array(ctx, doc, n-1);
for (i = 0; i < n-1; i++)
{
js_getindex(J, idx, i+1);
c = js_tonumber(J, -1);
js_pop(J, 1);
pdf_array_push_real(ctx, color, c);
}
}
fz_catch(ctx)
{
pdf_drop_obj(ctx, color);
rethrow(js);
}
return color;
}
static void field_getFillColor(js_State *J)
{
js_pushundefined(J);
}
static void field_setFillColor(js_State *J)
{
pdf_js *js = js_getcontext(J);
pdf_obj *field = js_touserdata(J, 0, "Field");
pdf_obj *color = load_color(js, 1);
fz_try(js->ctx)
pdf_field_set_fill_color(js->ctx, field, color);
fz_always(js->ctx)
pdf_drop_obj(js->ctx, color);
fz_catch(js->ctx)
rethrow(js);
}
static void field_getTextColor(js_State *J)
{
js_pushundefined(J);
}
static void field_setTextColor(js_State *J)
{
pdf_js *js = js_getcontext(J);
pdf_obj *field = js_touserdata(J, 0, "Field");
pdf_obj *color = load_color(js, 1);
fz_try(js->ctx)
pdf_field_set_text_color(js->ctx, field, color);
fz_always(js->ctx)
pdf_drop_obj(js->ctx, color);
fz_catch(js->ctx)
rethrow(js);
}
static void field_getBorderStyle(js_State *J)
{
pdf_js *js = js_getcontext(J);
pdf_obj *field = js_touserdata(J, 0, "Field");
const char *border_style = NULL;
fz_try(js->ctx)
border_style = pdf_field_border_style(js->ctx, field);
fz_catch(js->ctx)
rethrow(js);
js_pushstring(J, border_style);
}
static void field_setBorderStyle(js_State *J)
{
pdf_js *js = js_getcontext(J);
pdf_obj *field = js_touserdata(J, 0, "Field");
const char *border_style = js_tostring(J, 1);
fz_try(js->ctx)
pdf_field_set_border_style(js->ctx, field, border_style);
fz_catch(js->ctx)
rethrow(js);
}
static void field_getValue(js_State *J)
{
pdf_js *js = js_getcontext(J);
pdf_obj *field = js_touserdata(J, 0, "Field");
const char *str = NULL;
char *end;
double num;
fz_try(js->ctx)
str = pdf_field_value(js->ctx, field);
fz_catch(js->ctx)
rethrow(js);
num = strtod(str, &end);
if (*str && *end == 0)
js_pushnumber(J, num);
else
js_pushstring(J, str);
}
static void field_setValue(js_State *J)
{
pdf_js *js = js_getcontext(J);
pdf_obj *field = js_touserdata(J, 0, "Field");
const char *value = js_tostring(J, 1);
fz_try(js->ctx)
(void)pdf_set_field_value(js->ctx, js->doc, field, value, 0);
fz_catch(js->ctx)
rethrow(js);
}
static void doc_getField(js_State *J)
{
pdf_js *js = js_getcontext(J);
fz_context *ctx = js->ctx;
const char *cName = js_tostring(J, 1);
pdf_obj *dict = NULL;
fz_try(ctx)
dict = pdf_lookup_field(ctx, js->form, cName);
fz_catch(ctx)
rethrow(js);
if (dict)
{
js_getregistry(J, "Field");
js_newuserdata(J, "Field", pdf_keep_obj(js->ctx, dict), field_finalize);
}
else
{
js_pushnull(J);
}
}
static void doc_getNumPages(js_State *J)
{
pdf_js *js = js_getcontext(J);
int pages = pdf_count_pages(js->ctx, js->doc);
js_pushnumber(J, pages);
}
static void doc_setNumPages(js_State *J)
{
pdf_js *js = js_getcontext(J);
fz_warn(js->ctx, "Unexpected call to doc_setNumPages");
}
static void doc_resetForm(js_State *J)
{
pdf_js *js = js_getcontext(J);
pdf_obj *field;
fz_context *ctx = js->ctx;
int i, n;
/* An array of fields has been passed in. Call pdf_reset_field on each item. */
if (js_isarray(J, 1))
{
n = js_getlength(J, 1);
for (i = 0; i < n; ++i)
{
js_getindex(J, 1, i);
field = pdf_lookup_field(ctx, js->form, js_tostring(J, -1));
if (field)
pdf_field_reset(ctx, js->doc, field);
js_pop(J, 1);
}
}
/* No argument or null passed in means reset all. */
else
{
n = pdf_array_len(ctx, js->form);
for (i = 0; i < n; i++)
{
fz_try(ctx)
pdf_field_reset(ctx, js->doc, pdf_array_get(ctx, js->form, i));
fz_catch(ctx)
rethrow(js);
}
}
}
static void doc_print(js_State *J)
{
pdf_js *js = js_getcontext(J);
fz_try(js->ctx)
pdf_event_issue_print(js->ctx, js->doc);
fz_catch(js->ctx)
rethrow(js);
}
static void doc_mailDoc(js_State *J)
{
pdf_js *js = unpack_arguments(J, "bUI", "cTo", "cCc", "cBcc", "cSubject", "cMessage", 0);
pdf_mail_doc_event event;
event.ask_user = js_isdefined(J, 1) ? js_toboolean(J, 1) : 1;
event.to = js_tostring(J, 2);
event.cc = js_tostring(J, 3);
event.bcc = js_tostring(J, 4);
event.subject = js_tostring(J, 5);
event.message = js_tostring(J, 6);
fz_try(js->ctx)
pdf_event_issue_mail_doc(js->ctx, js->doc, &event);
fz_catch(js->ctx)
rethrow(js);
}
static void doc_calculateNow(js_State *J)
{
pdf_js *js = js_getcontext(J);
fz_try(js->ctx)
pdf_calculate_form(js->ctx, js->doc);
fz_catch(js->ctx)
rethrow(js);
}
static void console_println(js_State *J)
{
int i, top = js_gettop(J);
for (i = 1; i < top; ++i) {
const char *s = js_tostring(J, i);
if (i > 1) putchar(' ');
fputs(s, stdout);
}
putchar('\n');
js_pushundefined(J);
}
static void util_printf_d(fz_context *ctx, fz_buffer *out, int ds, int sign, int pad, int w, int base, int value)
{
static const char *digits = "0123456789abcdef";
char buf[50];
unsigned int a;
int i, m = 0;
if (value < 0)
{
sign = '-';
a = -value;
}
else
{
a = value;
}
i = 0;
do
{
buf[i++] = digits[a % base];
a /= base;
if (a > 0 && ++m == 3)
{
if (ds == 0) buf[i++] = ',';
if (ds == 2) buf[i++] = '.';
m = 0;
}
} while (a);
if (sign)
{
if (pad == '0')
while (i < w - 1)
buf[i++] = pad;
buf[i++] = sign;
}
while (i < w)
buf[i++] = pad;
while (i > 0)
fz_append_byte(ctx, out, buf[--i]);
}
static void util_printf_f(fz_context *ctx, fz_buffer *out, int ds, int sign, int pad, int special, int w, int p, double value)
{
char buf[40], *point, *digits = buf;
int n = 0;
int m = 0;
fz_snprintf(buf, sizeof buf, "%.*f", p, value);
if (*digits == '-')
{
sign = '-';
++digits;
}
if (*digits != '.' && (*digits < '0' || *digits > '9'))
{
fz_append_string(ctx, out, "nan");
return;
}
n = strlen(digits);
if (sign)
++n;
point = strchr(digits, '.');
if (point)
m = 3 - (point - digits) % 3;
else
{
m = 3 - n % 3;
if (special)
++n;
}
if (m == 3)
m = 0;
if (pad == '0' && sign)
fz_append_byte(ctx, out, sign);
for (; n < w; ++n)
fz_append_byte(ctx, out, pad);
if (pad == ' ' && sign)
fz_append_byte(ctx, out, sign);
while (*digits && *digits != '.')
{
fz_append_byte(ctx, out, *digits++);
if (++m == 3 && *digits && *digits != '.')
{
if (ds == 0) fz_append_byte(ctx, out, ',');
if (ds == 2) fz_append_byte(ctx, out, '.');
m = 0;
}
}
if (*digits == '.' || special)
{
if (ds == 0 || ds == 1)
fz_append_byte(ctx, out, '.');
else
fz_append_byte(ctx, out, ',');
}
if (*digits == '.')
{
++digits;
while (*digits)
fz_append_byte(ctx, out, *digits++);
}
}
static void util_printf(js_State *J)
{
pdf_js *js = js_getcontext(J);
fz_context *ctx = js->ctx;
const char *fmt = js_tostring(J, 1);
fz_buffer *out = NULL;
int ds, w, p, sign, pad, special;
int c, i = 1;
int failed = 0;
const char *str;
fz_var(out);
fz_try(ctx)
{
out = fz_new_buffer(ctx, 256);
while ((c = *fmt++) != 0)
{
if (c == '%')
{
c = *fmt++;
ds = 1;
if (c == ',')
{
c = *fmt++;
if (!c)
break;
ds = c - '0';
}
special = 0;
sign = 0;
pad = ' ';
while (c == ' ' || c == '+' || c == '0' || c == '#')
{
if (c == '+') sign = '+';
else if (c == ' ') sign = ' ';
else if (c == '0') pad = '0';
else if (c == '#') special = 1;
c = *fmt++;
}
if (!pad)
pad = ' ';
if (!c)
break;
w = 0;
while (c >= '0' && c <= '9')
{
w = w * 10 + (c - '0');
c = *fmt++;
}
if (!c)
break;
p = 0;
if (c == '.')
{
c = *fmt++;
while (c >= '0' && c <= '9')
{
p = p * 10 + (c - '0');
c = *fmt++;
}
}
else
{
special = 1;
}
if (!c)
break;
switch (c)
{
case '%':
fz_append_byte(ctx, out, '%');
break;
case 'x':
util_printf_d(ctx, out, ds, sign, pad, w, 16, js_tryinteger(J, ++i, 0));
break;
case 'd':
util_printf_d(ctx, out, ds, sign, pad, w, 10, js_tryinteger(J, ++i, 0));
break;
case 'f':
util_printf_f(ctx, out, ds, sign, pad, special, w, p, js_trynumber(J, ++i, 0));
break;
case 's':
default:
fz_append_string(ctx, out, js_trystring(J, ++i, ""));
}
}
else
{
fz_append_byte(ctx, out, c);
}
}
str = fz_string_from_buffer(ctx, out);
if (js_try(J))
{
failed = 1;
}
else
{
js_pushstring(J, str);
js_endtry(J);
}
}
fz_always(ctx)
fz_drop_buffer(ctx, out);
fz_catch(ctx)
rethrow(js);
if (failed)
js_throw(J);
}
static void addmethod(js_State *J, const char *name, js_CFunction fun, int n)
{
const char *realname = strchr(name, '.');
realname = realname ? realname + 1 : name;
js_newcfunction(J, fun, name, n);
js_defproperty(J, -2, realname, JS_READONLY | JS_DONTENUM | JS_DONTCONF);
}
static void addproperty(js_State *J, const char *name, js_CFunction getfun, js_CFunction setfun)
{
const char *realname = strchr(name, '.');
realname = realname ? realname + 1 : name;
js_newcfunction(J, getfun, name, 0);
js_newcfunction(J, setfun, name, 1);
js_defaccessor(J, -3, realname, JS_READONLY | JS_DONTENUM | JS_DONTCONF);
}
static void declare_dom(pdf_js *js)
{
js_State *J = js->imp;
/* Allow access to the global environment via the 'global' name */
js_pushglobal(J);
js_defglobal(J, "global", JS_READONLY | JS_DONTCONF | JS_DONTENUM);
/* Create the 'event' object */
js_newobject(J);
js_defglobal(J, "event", JS_READONLY | JS_DONTCONF | JS_DONTENUM);
/* Create the 'util' object */
js_newobject(J);
{
// TODO: util.printd
// TODO: util.printx
addmethod(J, "util.printf", util_printf, 1);
}
js_defglobal(J, "util", JS_READONLY | JS_DONTCONF | JS_DONTENUM);
/* Create the 'app' object */
js_newobject(J);
{
#ifdef _WIN32
js_pushstring(J, "WIN");
#elif defined(__APPLE__)
js_pushstring(J, "MAC");
#else
js_pushstring(J, "UNIX");
#endif
js_defproperty(J, -2, "app.platform", JS_READONLY | JS_DONTENUM | JS_DONTCONF);
addmethod(J, "app.alert", app_alert, 4);
addmethod(J, "app.execMenuItem", app_execMenuItem, 1);
addmethod(J, "app.launchURL", app_launchURL, 2);
}
js_defglobal(J, "app", JS_READONLY | JS_DONTCONF | JS_DONTENUM);
/* Create the Field prototype object */
js_newobject(J);
{
addproperty(J, "Field.value", field_getValue, field_setValue);
addproperty(J, "Field.borderStyle", field_getBorderStyle, field_setBorderStyle);
addproperty(J, "Field.textColor", field_getTextColor, field_setTextColor);
addproperty(J, "Field.fillColor", field_getFillColor, field_setFillColor);
addproperty(J, "Field.display", field_getDisplay, field_setDisplay);
addproperty(J, "Field.name", field_getName, field_setName);
addmethod(J, "Field.buttonSetCaption", field_buttonSetCaption, 1);
}
js_setregistry(J, "Field");
/* Create the console object */
js_newobject(J);
{
addmethod(J, "console.println", console_println, 1);
}
js_defglobal(J, "console", JS_READONLY | JS_DONTCONF | JS_DONTENUM);
/* Put all of the Doc methods in the global object, which is used as
* the 'this' binding for regular non-strict function calls. */
js_pushglobal(J);
{
addproperty(J, "Doc.numPages", doc_getNumPages, doc_setNumPages);
addmethod(J, "Doc.getField", doc_getField, 1);
addmethod(J, "Doc.resetForm", doc_resetForm, 0);
addmethod(J, "Doc.calculateNow", doc_calculateNow, 0);
addmethod(J, "Doc.print", doc_print, 0);
addmethod(J, "Doc.mailDoc", doc_mailDoc, 6);
}
js_pop(J, 1);
}
static void preload_helpers(pdf_js *js)
{
/* When testing on the cluster:
* Use a fixed date for "new Date" and Date.now().
* Sadly, this breaks uses of the Date function without the new keyword.
* Return a fixed number from Math.random().
*/
#ifdef CLUSTER
js_dostring(js->imp,
"var MuPDFOldDate = Date\n"
"Date = function() { return new MuPDFOldDate(298252800000); }\n"
"Date.now = function() { return 298252800000; }\n"
"Date.UTC = function() { return 298252800000; }\n"
"Date.parse = MuPDFOldDate.parse;\n"
"Math.random = function() { return 1/4; }\n"
);
#endif
js_dostring(js->imp,
#include "js/util.js.h"
);
}
void pdf_drop_js(fz_context *ctx, pdf_js *js)
{
if (js)
{
js_freestate(js->imp);
fz_free(ctx, js);
}
}
static void *pdf_js_alloc(void *actx, void *ptr, int n)
{
return fz_realloc_no_throw(actx, ptr, n);
}
static pdf_js *pdf_new_js(fz_context *ctx, pdf_document *doc)
{
pdf_js *js = fz_malloc_struct(ctx, pdf_js);
js->ctx = ctx;
js->doc = doc;
fz_try(ctx)
{
pdf_obj *root, *acroform;
/* Find the form array */
root = pdf_dict_get(ctx, pdf_trailer(ctx, doc), PDF_NAME(Root));
acroform = pdf_dict_get(ctx, root, PDF_NAME(AcroForm));
js->form = pdf_dict_get(ctx, acroform, PDF_NAME(Fields));
/* Initialise the javascript engine, passing the fz_context for use in memory allocation. */
js->imp = js_newstate(pdf_js_alloc, ctx, 0);
if (!js->imp)
fz_throw(ctx, FZ_ERROR_GENERIC, "cannot initialize javascript engine");
/* Also set our pdf_js context, so we can retrieve it in callbacks. */
js_setcontext(js->imp, js);
declare_dom(js);
preload_helpers(js);
}
fz_catch(ctx)
{
pdf_drop_js(ctx, js);
fz_rethrow(ctx);
}
return js;
}
static void pdf_js_load_document_level(pdf_js *js)
{
fz_context *ctx = js->ctx;
pdf_document *doc = js->doc;
pdf_obj *javascript;
int len, i;
javascript = pdf_load_name_tree(ctx, doc, PDF_NAME(JavaScript));
len = pdf_dict_len(ctx, javascript);
fz_try(ctx)
{
for (i = 0; i < len; i++)
{
pdf_obj *fragment = pdf_dict_get_val(ctx, javascript, i);
pdf_obj *code = pdf_dict_get(ctx, fragment, PDF_NAME(JS));
char *codebuf = pdf_load_stream_or_string_as_utf8(ctx, code);
char buf[100];
if (pdf_is_indirect(ctx, code))
fz_snprintf(buf, sizeof buf, "%d", pdf_to_num(ctx, code));
else
fz_snprintf(buf, sizeof buf, "Root/Names/JavaScript/Names/%d/JS", (i+1)*2);
pdf_js_execute(js, buf, codebuf);
fz_free(ctx, codebuf);
}
}
fz_always(ctx)
pdf_drop_obj(ctx, javascript);
fz_catch(ctx)
fz_rethrow(ctx);
}
void pdf_js_event_init(pdf_js *js, pdf_obj *target, const char *value, int willCommit)
{
if (js)
{
js_getglobal(js->imp, "event");
{
js_pushboolean(js->imp, 1);
js_setproperty(js->imp, -2, "rc");
js_pushboolean(js->imp, willCommit);
js_setproperty(js->imp, -2, "willCommit");
js_getregistry(js->imp, "Field");
js_newuserdata(js->imp, "Field", pdf_keep_obj(js->ctx, target), field_finalize);
js_setproperty(js->imp, -2, "target");
js_pushstring(js->imp, value);
js_setproperty(js->imp, -2, "value");
}
js_pop(js->imp, 1);
}
}
int pdf_js_event_result(pdf_js *js)
{
int rc = 1;
if (js)
{
js_getglobal(js->imp, "event");
js_getproperty(js->imp, -1, "rc");
rc = js_tryboolean(js->imp, -1, 1);
js_pop(js->imp, 2);
}
return rc;
}
void pdf_js_event_init_keystroke(pdf_js *js, pdf_obj *target, pdf_keystroke_event *event)
{
if (js)
{
pdf_js_event_init(js, target, event->value, event->willCommit);
js_getglobal(js->imp, "event");
{
js_pushstring(js->imp, event->change);
js_setproperty(js->imp, -2, "change");
js_pushnumber(js->imp, event->selStart);
js_setproperty(js->imp, -2, "selStart");
js_pushnumber(js->imp, event->selEnd);
js_setproperty(js->imp, -2, "selEnd");
}
js_pop(js->imp, 1);
}
}
int pdf_js_event_result_keystroke(pdf_js *js, pdf_keystroke_event *evt)
{
int rc = 1;
if (js)
{
js_getglobal(js->imp, "event");
{
js_getproperty(js->imp, -1, "rc");
rc = js_tryboolean(js->imp, -1, 1);
js_pop(js->imp, 1);
if (rc)
{
js_getproperty(js->imp, -1, "change");
evt->newChange = fz_strdup(js->ctx, js_trystring(js->imp, -1, ""));
js_pop(js->imp, 1);
js_getproperty(js->imp, -1, "selStart");
evt->selStart = js_tryinteger(js->imp, -1, 0);
js_pop(js->imp, 1);
js_getproperty(js->imp, -1, "selEnd");
evt->selEnd = js_tryinteger(js->imp, -1, 0);
js_pop(js->imp, 1);
}
}
js_pop(js->imp, 1);
}
return rc;
}
char *pdf_js_event_value(pdf_js *js)
{
char *value = NULL;
if (js)
{
js_getglobal(js->imp, "event");
js_getproperty(js->imp, -1, "value");
value = fz_strdup(js->ctx, js_trystring(js->imp, -1, "undefined"));
js_pop(js->imp, 2);
}
return value;
}
void pdf_js_execute(pdf_js *js, const char *name, const char *source)
{
if (js)
{
if (js_ploadstring(js->imp, name, source))
{
fz_warn(js->ctx, "%s", js_trystring(js->imp, -1, "Error"));
js_pop(js->imp, 1);
return;
}
js_pushundefined(js->imp);
if (js_pcall(js->imp, 0))
{
fz_warn(js->ctx, "%s", js_trystring(js->imp, -1, "Error"));
js_pop(js->imp, 1);
return;
}
js_pop(js->imp, 1);
}
}
void pdf_enable_js(fz_context *ctx, pdf_document *doc)
{
if (!doc->js)
{
doc->js = pdf_new_js(ctx, doc);
pdf_js_load_document_level(doc->js);
}
}
void pdf_disable_js(fz_context *ctx, pdf_document *doc)
{
pdf_drop_js(ctx, doc->js);
doc->js = NULL;
}
int pdf_js_supported(fz_context *ctx, pdf_document *doc)
{
return doc->js != NULL;
}
#else /* FZ_ENABLE_JS */
void pdf_drop_js(fz_context *ctx, pdf_js *js) { }
void pdf_enable_js(fz_context *ctx, pdf_document *doc) { }
void pdf_disable_js(fz_context *ctx, pdf_document *doc) { }
int pdf_js_supported(fz_context *ctx, pdf_document *doc) { return 0; }
void pdf_js_event_init(pdf_js *js, pdf_obj *target, const char *value, int willCommit) { }
void pdf_js_event_init_keystroke(pdf_js *js, pdf_obj *target, pdf_keystroke_event *event) { }
int pdf_js_event_result_keystroke(pdf_js *js, pdf_keystroke_event *evt) { return 1; }
int pdf_js_event_result(pdf_js *js) { return 1; }
char *pdf_js_event_value(pdf_js *js) { return ""; }
void pdf_js_execute(pdf_js *js, const char *name, const char *source) { }
#endif /* FZ_ENABLE_JS */