2 * Simulated browser environment for Rhino
3 * By John Resig <http://ejohn.org/>
4 * Copyright 2007 John Resig, under the MIT License
16 return "Mozilla/5.0 (Macintosh; U; Intel Mac OS X; en-US; rv:1.8.1.3) Gecko/20070309 Firefox/2.0.0.3";
20 var curLocation = (new java.io.File("./")).toURL();
22 window.__defineSetter__("location", function(url){
23 curLocation = new java.net.URL( curLocation, url );
25 window.document = new DOMDocument(
26 new Packages.org.xml.sax.InputSource(
27 new java.io.InputStreamReader(
28 new java.io.FileInputStream( url ))));
31 window.__defineGetter__("location", function(url){
34 return curLocation.getProtocol() + ":";
43 window.setTimeout = function(fn, time){
45 return num = setInterval(function(){
51 window.setInterval = function(fn, time){
52 var num = timers.length;
54 timers[num] = new java.lang.Thread(new java.lang.Runnable({
57 java.lang.Thread.currentThread().sleep(time);
68 window.clearInterval = function(num){
77 window.addEventListener = function(){};
78 window.removeEventListener = function(){};
82 window.DOMDocument = function(file){
84 this._dom = Packages.javax.xml.parsers.
85 DocumentBuilderFactory.newInstance()
86 .newDocumentBuilder().parse(file);
88 if ( !obj_nodes.containsKey( this._dom ) )
89 obj_nodes.put( this._dom, this );
92 DOMDocument.prototype = {
93 createTextNode: function(text){
94 return makeNode( this._dom.createTextNode(
95 text.replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">")) );
97 createElement: function(name){
98 return makeNode( this._dom.createElement(name.toLowerCase()) );
100 getElementsByTagName: function(name){
101 return new DOMNodeList( this._dom.getElementsByTagName(
102 name.toLowerCase()) );
104 getElementById: function(id){
105 var elems = this._dom.getElementsByTagName("*");
107 for ( var i = 0; i < elems.length; i++ ) {
108 var elem = elems.item(i);
109 if ( elem.getAttribute("id") == id )
110 return makeNode(elem);
116 return this.getElementsByTagName("body")[0];
118 get documentElement(){
119 return makeNode( this._dom.getDocumentElement() );
124 addEventListener: function(){},
125 removeEventListener: function(){},
129 importNode: function(node, deep){
130 return makeNode( this._dom.importNode(node._dom, deep) );
132 toString: function(){
133 return "Document" + (typeof this._file == "string" ?
134 ": " + this._file : "");
139 getComputedStyle: function(elem){
141 getPropertyValue: function(prop){
142 prop = prop.replace(/\-(\w)/g,function(m,c){
143 return c.toUpperCase();
145 var val = elem.style[prop];
147 if ( prop == "opacity" && val == "" )
158 function getDocument(node){
159 return obj_nodes.get(node);
164 window.DOMNodeList = function(list){
166 this.length = list.getLength();
168 for ( var i = 0; i < this.length; i++ ) {
169 var node = list.item(i);
170 this[i] = makeNode( node );
174 DOMNodeList.prototype = {
175 toString: function(){
177 Array.prototype.join.call( this, ", " ) + " ]";
180 return Array.prototype.map.call(
181 this, function(node){return node.valueOf();}).join('');
187 window.DOMNode = function(node){
191 DOMNode.prototype = {
193 return this._dom.getNodeType();
196 return this._dom.getNodeValue();
199 return this._dom.getNodeName();
201 cloneNode: function(deep){
202 return makeNode( this._dom.cloneNode(deep) );
205 return getDocument( this._dom.ownerDocument );
207 get documentElement(){
208 return makeNode( this._dom.documentElement );
211 return makeNode( this._dom.getParentNode() );
214 return makeNode( this._dom.getNextSibling() );
216 get previousSibling() {
217 return makeNode( this._dom.getPreviousSibling() );
219 toString: function(){
220 return '"' + this.nodeValue + '"';
223 return this.nodeValue;
229 window.DOMElement = function(elem){
232 get opacity(){ return this._opacity; },
233 set opacity(val){ this._opacity = val + ""; }
237 var styles = (this.getAttribute("style") || "").split(/\s*;\s*/);
239 for ( var i = 0; i < styles.length; i++ ) {
240 var style = styles[i].split(/\s*:\s*/);
241 if ( style.length == 2 )
242 this.style[ style[0] ] = style[1];
246 DOMElement.prototype = extend( new DOMNode(), {
248 return this.tagName.toUpperCase();
251 return this._dom.getTagName();
253 toString: function(){
254 return "<" + this.tagName + (this.id ? "#" + this.id : "" ) + ">";
257 var ret = "<" + this.tagName, attr = this.attributes;
259 for ( var i in attr )
260 ret += " " + i + "='" + attr[i] + "'";
262 if ( this.childNodes.length || this.nodeName == "SCRIPT" )
263 ret += ">" + this.childNodes.valueOf() +
264 "</" + this.tagName + ">";
272 var attr = {}, attrs = this._dom.getAttributes();
274 for ( var i = 0; i < attrs.getLength(); i++ )
275 attr[ attrs.item(i).nodeName ] = attrs.item(i).nodeValue;
281 return this.childNodes.valueOf();
284 html = html.replace(/<\/?([A-Z]+)/g, function(m){
285 return m.toLowerCase();
288 var nodes = this.ownerDocument.importNode(
289 new DOMDocument( new java.io.ByteArrayInputStream(
290 (new java.lang.String("<wrap>" + html + "</wrap>"))
291 .getBytes("UTF8"))).documentElement, true).childNodes;
293 while (this.firstChild)
294 this.removeChild( this.firstChild );
296 for ( var i = 0; i < nodes.length; i++ )
297 this.appendChild( nodes[i] );
301 return nav(this.childNodes);
305 for ( var i = 0; i < nodes.length; i++ )
306 if ( nodes[i].nodeType == 3 )
307 str += nodes[i].nodeValue;
308 else if ( nodes[i].nodeType == 1 )
309 str += nav(nodes[i].childNodes);
313 set textContent(text){
314 while (this.firstChild)
315 this.removeChild( this.firstChild );
316 this.appendChild( this.ownerDocument.createTextNode(text));
326 var val = this.getAttribute("disabled");
327 return val != "false" && !!val;
329 set disabled(val) { return this.setAttribute("disabled",val); },
332 var val = this.getAttribute("checked");
333 return val != "false" && !!val;
335 set checked(val) { return this.setAttribute("checked",val); },
338 if ( !this._selectDone ) {
339 this._selectDone = true;
341 if ( this.nodeName == "OPTION" && !this.parentNode.getAttribute("multiple") ) {
342 var opt = this.parentNode.getElementsByTagName("option");
344 if ( this == opt[0] ) {
347 for ( var i = 1; i < opt.length; i++ )
348 if ( opt[i].selected ) {
354 this.selected = true;
359 var val = this.getAttribute("selected");
360 return val != "false" && !!val;
362 set selected(val) { return this.setAttribute("selected",val); },
364 get className() { return this.getAttribute("class") || ""; },
366 return this.setAttribute("class",
367 val.replace(/(^\s*|\s*$)/g,""));
370 get type() { return this.getAttribute("type") || ""; },
371 set type(val) { return this.setAttribute("type",val); },
373 get value() { return this.getAttribute("value") || ""; },
374 set value(val) { return this.setAttribute("value",val); },
376 get src() { return this.getAttribute("src") || ""; },
377 set src(val) { return this.setAttribute("src",val); },
379 get id() { return this.getAttribute("id") || ""; },
380 set id(val) { return this.setAttribute("id",val); },
382 getAttribute: function(name){
383 return this._dom.hasAttribute(name) ?
384 new String( this._dom.getAttribute(name) ) :
387 setAttribute: function(name,value){
388 this._dom.setAttribute(name,value);
390 removeAttribute: function(name){
391 this._dom.removeAttribute(name);
395 return new DOMNodeList( this._dom.getChildNodes() );
398 return makeNode( this._dom.getFirstChild() );
401 return makeNode( this._dom.getLastChild() );
403 appendChild: function(node){
404 this._dom.appendChild( node._dom );
406 insertBefore: function(node,before){
407 this._dom.insertBefore( node._dom, before ? before._dom : before );
409 removeChild: function(node){
410 this._dom.removeChild( node._dom );
413 getElementsByTagName: DOMDocument.prototype.getElementsByTagName,
414 addEventListener: function(){},
415 removeEventListener: function(){},
417 submit: function(){},
421 return this.getElementsByTagName("*");
424 return this.nodeName == "IFRAME" ? {
425 document: this.contentDocument
428 get contentDocument(){
429 if ( this.nodeName == "IFRAME" ) {
431 this._doc = new DOMDocument(
432 new java.io.ByteArrayInputStream((new java.lang.String(
433 "<html><head><title></title></head><body></body></html>"))
441 // Helper method for extending one object with another
443 function extend(a,b) {
445 var g = b.__lookupGetter__(i), s = b.__lookupSetter__(i);
449 a.__defineGetter__(i, g);
451 a.__defineSetter__(i, s);
458 // Helper method for generating the right
459 // DOM objects based upon the type
461 var obj_nodes = new java.util.HashMap();
463 function makeNode(node){
465 if ( !obj_nodes.containsKey( node ) )
466 obj_nodes.put( node, node.getNodeType() ==
467 Packages.org.w3c.dom.Node.ELEMENT_NODE ?
468 new DOMElement( node ) : new DOMNode( node ) );
470 return obj_nodes.get(node);
476 // Originally implemented by Yehuda Katz
478 window.XMLHttpRequest = function(){
480 this.responseHeaders = {};
483 XMLHttpRequest.prototype = {
484 open: function(method, url, async, user, password){
488 this.method = method || "GET";
490 this.onreadystatechange();
492 setRequestHeader: function(header, value){
493 this.headers[header] = value;
495 getResponseHeader: function(header){ },
496 send: function(data){
499 function makeRequest(){
500 var url = new java.net.URL(curLocation, self.url),
501 connection = url.openConnection();
503 // Add headers to Java connection
504 for (var header in self.headers)
505 connection.addRequestProperty(header, self.headers[header]);
507 connection.connect();
509 // Stick the response headers into responseHeaders
510 for (var i=0; ; i++) {
511 var headerName = connection.getHeaderFieldKey(i);
512 var headerValue = connection.getHeaderField(i);
513 if (!headerName && !headerValue) break;
515 self.responseHeaders[headerName] = headerValue;
519 self.status = parseInt(connection.responseCode);
520 self.statusText = connection.responseMessage;
522 var stream = new java.io.InputStreamReader(
523 connection.getInputStream()),
524 buffer = new java.io.BufferedReader(stream),
527 while ((line = buffer.readLine()) != null)
528 self.responseText += line;
530 self.responseXML = null;
532 if ( self.responseText.match(/^\s*</) ) {
534 self.responseXML = new DOMDocument(
535 new java.io.ByteArrayInputStream(
536 (new java.lang.String(
537 self.responseText)).getBytes("UTF8")));
541 self.onreadystatechange();
545 (new java.lang.Thread(new java.lang.Runnable({
552 onreadystatechange: function(){},
553 getResponseHeader: function(header){
554 if (this.readyState < 3)
555 throw new Error("INVALID_STATE_ERR");
557 var returnedHeaders = [];
558 for (var rHeader in this.responseHeaders) {
559 if (rHeader.match(new Regexp(header, "i")))
560 returnedHeaders.push(this.responseHeaders[rHeader]);
563 if (returnedHeaders.length)
564 return returnedHeaders.join(", ");
569 getAllResponseHeaders: function(header){
570 if (this.readyState < 3)
571 throw new Error("INVALID_STATE_ERR");
573 var returnedHeaders = [];
575 for (var header in this.responseHeaders)
576 returnedHeaders.push( header + ": " + this.responseHeaders[header] );
578 return returnedHeaders.join("\r\n");