netsurf/javascript/jsapi/window.bnd
2013-06-03 23:10:53 +01:00

403 lines
9.2 KiB
Plaintext

/* Binding to generate window interface
*
* Copyright 2012 Vincent Sanders <vince@netsurf-browser.org>
*
* This file is part of NetSurf, http://www.netsurf-browser.org/
*
* Released under the terms of the MIT License,
* http://www.opensource.org/licenses/mit-license
*/
webidlfile "html.idl";
webidlfile "dom.idl";
webidlfile "console.idl";
hdrcomment "Copyright 2012 Vincent Sanders <vince@netsurf-browser.org>";
hdrcomment "This file is part of NetSurf, http://www.netsurf-browser.org/";
hdrcomment "Released under the terms of the MIT License,";
hdrcomment " http://www.opensource.org/licenses/mit-license";
preamble %{
#include <dom/dom.h>
#include "utils/config.h"
#include "utils/log.h"
#include "utils/corestrings.h"
#include "render/html_internal.h"
#include "javascript/jsapi.h"
#include "console.h"
#include "navigator.h"
#include "event.h"
#include "node.h"
#include "htmlcollection.h"
#include "nodelist.h"
#include "htmldocument.h"
#include "text.h"
#include "comment.h"
#include "htmlelement.h"
#include "window.h"
#include "location.h"
struct browser_window *jsapi_get_browser_window(JSContext *cx);
%}
prologue %{
struct browser_window *jsapi_get_browser_window(JSContext *cx)
{
struct jsclass_private *private;
private = JS_GetInstancePrivate(cx,
JS_GetGlobalObject(cx),
&JSClass_Window,
NULL);
if (private != NULL) {
return private->bw;
}
return NULL;
}
static bool
init_user_prototypes(JSContext *cx,
struct jsclass_private *private,
JSObject *parent)
{
/* Initialises all the user javascript classes to make their
* prototypes available.
*/
/** @todo should we be managing these prototype objects ourselves */
private->prototype_Document = jsapi_InitClass_Document(cx, parent);
if (private->prototype_Document == NULL) {
return false;
}
private->prototype_Navigator = jsapi_InitClass_Navigator(cx, parent);
if (private->prototype_Navigator == NULL) {
return false;
}
private->prototype_Location = jsapi_InitClass_Location(cx, parent);
if (private->prototype_Location == NULL) {
return false;
}
private->prototype_Console = jsapi_InitClass_Console(cx, parent);
if (private->prototype_Console == NULL) {
return false;
}
private->prototype_HTMLElement = jsapi_InitClass_HTMLElement(cx, parent);
if (private->prototype_HTMLElement == NULL) {
return false;
}
private->prototype_HTMLCollection = jsapi_InitClass_HTMLCollection(cx, parent);
if (private->prototype_HTMLCollection == NULL) {
return false;
}
private->prototype_NodeList = jsapi_InitClass_NodeList(cx, parent);
if (private->prototype_NodeList == NULL) {
return false;
}
private->prototype_Text = jsapi_InitClass_Text(cx, parent);
if (private->prototype_Text == NULL) {
return false;
}
private->prototype_Comment = jsapi_InitClass_Comment(cx, parent);
if (private->prototype_Comment == NULL) {
return false;
}
private->prototype_Node = jsapi_InitClass_Node(cx, parent);
if (private->prototype_Node == NULL) {
return false;
}
private->prototype_Event = jsapi_InitClass_Event(cx, parent);
if (private->prototype_Event == NULL) {
return false;
}
return true;
}
%}
binding window {
type js_libdom; /* the binding type */
interface Window; /* Web IDL interface to generate */
private "struct browser_window *" bw;
private "struct html_content *" htmlc;
/* prototypes held in this object */
internal "JSObject *" prototype_Document;
internal "JSObject *" prototype_Navigator;
internal "JSObject *" prototype_Location;
internal "JSObject *" prototype_Console;
internal "JSObject *" prototype_HTMLElement;
internal "JSObject *" prototype_HTMLCollection;
internal "JSObject *" prototype_NodeList;
internal "JSObject *" prototype_Text;
internal "JSObject *" prototype_Comment;
internal "JSObject *" prototype_Node;
internal "JSObject *" prototype_Event;
/** document instantiated on first use */
property unshared document;
/** navigator instantiated on first use */
property unshared navigator;
/** console instantiated on first use */
property unshared console;
/** location is unshared */
property unshared location;
/** @todo instantiate forms, history etc. attributes */
/* events through a single interface */
property unshared type EventHandler;
}
api mark %{
if (private != NULL) {
if (private->prototype_Document != NULL) {
JSAPI_GCMARK(private->prototype_Document);
}
if (private->prototype_Navigator != NULL) {
JSAPI_GCMARK(private->prototype_Navigator);
}
if (private->prototype_Location != NULL) {
JSAPI_GCMARK(private->prototype_Location);
}
if (private->prototype_Console != NULL) {
JSAPI_GCMARK(private->prototype_Console);
}
if (private->prototype_HTMLElement != NULL) {
JSAPI_GCMARK(private->prototype_HTMLElement);
}
if (private->prototype_HTMLCollection != NULL) {
JSAPI_GCMARK(private->prototype_HTMLCollection);
}
if (private->prototype_NodeList != NULL) {
JSAPI_GCMARK(private->prototype_NodeList);
}
if (private->prototype_Text != NULL) {
JSAPI_GCMARK(private->prototype_Text);
}
if (private->prototype_Comment != NULL) {
JSAPI_GCMARK(private->prototype_Comment);
}
if (private->prototype_Node != NULL) {
JSAPI_GCMARK(private->prototype_Node);
}
if (private->prototype_Event != NULL) {
JSAPI_GCMARK(private->prototype_Event);
}
}
%}
api global %{
%}
api init %{
prototype = JS_NewCompartmentAndGlobalObject(cx, &JSClass_Window, NULL);
if (prototype == NULL) {
return NULL;
}
/** @todo reconsider global object handling. future
* editions of spidermonkey appear to be removing the
* idea of a global so we probably need to handle
* global object references internally
*/
/* set the contexts global */
JS_SetGlobalObject(cx, prototype);
/* Populate the global object with the standard globals, like
* Object and Array.
*/
if (!JS_InitStandardClasses(cx, prototype)) {
return NULL;
}
/* add functions to prototype */
if (!JS_DefineFunctions(cx, prototype, jsclass_functions)) {
return NULL;
}
/* add properties to prototype */
if (!JS_DefineProperties(cx, prototype, jsclass_properties))
return NULL;
/* as the global just got changed, force a GC run */
JS_GC(cx);
%}
api new %{
/* @todo sort out windows that are not globals */
assert(parent == NULL);
/* the window object is the global so its prototype *is* the instance */
newobject = prototype;
if (init_user_prototypes(cx, private, prototype) == false) {
/* prototype initialisation failed */
free(private);
return NULL;
}
LOG(("Created new window object %p", newobject));
%}
getter document %{
if (!JSVAL_IS_VOID(JSAPI_PROP_RVAL(cx, vp))) {
/* already created - return it */
return JS_TRUE;
}
/* instantiate the subclasses off the window global */
jsret = jsapi_new_Document(cx,
NULL,
NULL,
(dom_document *)dom_node_ref(private->htmlc->document),
private->htmlc);
%}
getter navigator %{
if (!JSVAL_IS_VOID(JSAPI_PROP_RVAL(cx, vp))) {
/* already created - return it */
return JS_TRUE;
}
jsret = jsapi_new_Navigator(cx, NULL, NULL);
%}
getter console %{
if (!JSVAL_IS_VOID(JSAPI_PROP_RVAL(cx, vp))) {
/* already created - return it */
return JS_TRUE;
}
jsret = jsapi_new_Console(cx, NULL, NULL);
%}
operation confirm %{
warn_user(message, NULL);
%}
operation alert %{
warn_user(message, NULL);
%}
operation prompt %{
warn_user(message, NULL);
%}
/* boolean dispatchEvent(Event event); */
operation dispatchEvent %{
/* this implementation is unique to the window object as it is
* not a "real" dom node.
*/
/* caution, this must match the struct generated from event.bnd */
struct {
dom_event *event;
} *event_private;
dom_string *type_dom = NULL;
dom_exception exc;
jsval eventval = JSVAL_VOID;
jsval event_argv[1];
jsval event_rval;
event_private = JS_GetInstancePrivate(cx, event, &JSClass_Event, NULL);
if (event_private->event == NULL) {
/** @todo type error? */
jsret = JS_FALSE;
} else {
exc = dom_event_get_type(event_private->event, &type_dom);
if (exc == DOM_NO_ERR) {
if (dom_string_isequal(type_dom, corestring_dom_load)) {
JS_GetProperty(cx, JSAPI_THIS_OBJECT(cx, vp), "onload", &eventval);
}
if (!JSVAL_IS_VOID(eventval)) {
event_argv[0] = eventval;
jsret = JS_CallFunctionValue(cx, NULL, eventval, 1, event_argv, &event_rval);
}
}
}
%}
getter location %{
if (!JSVAL_IS_VOID(JSAPI_PROP_RVAL(cx, vp))) {
/* already created - return it */
return JS_TRUE;
}
/* should get the docuemnts location
jsval loc;
JS_GetProperty(cx, private->document, "location", &loc);
jsret = JSVAL_TO_OBJECT(loc);
*/
jsret = jsapi_new_Location(cx,
NULL,
NULL,
llcache_handle_get_url(private->htmlc->base.llcache));
%}
getter window %{
jsret = obj;
%}
getter self %{
jsret = obj;
%}
/* very iffy implementation */
getter top %{
jsret = obj;
%}
getter EventHandler %{
/* this implementation is unique to the window object as it is
* not a dom node.
*/
JSLOG("propname[%d]=\"%s\"",
tinyid,
jsclass_properties[tinyid].name);
%}
setter EventHandler %{
/* this implementation is unique to the window object as it is
* not a dom node.
*/
JSLOG("propname[%d]=\"%s\"",
tinyid,
jsclass_properties[tinyid].name);
%}