2014-08-14 05:04:47 +09:00
|
|
|
/* Any copyright is dedicated to the Public Domain.
|
|
|
|
* http://creativecommons.org/publicdomain/zero/1.0/ */
|
|
|
|
|
2018-12-06 21:55:15 +09:00
|
|
|
function xmlEncode(s) {
|
2014-08-14 05:04:47 +09:00
|
|
|
var i = 0, ch;
|
2014-10-22 23:59:20 +09:00
|
|
|
s = String(s);
|
2014-08-14 05:04:47 +09:00
|
|
|
while (i < s.length && (ch = s[i]) !== '&' && ch !== '<' &&
|
|
|
|
ch !== '\"' && ch !== '\n' && ch !== '\r' && ch !== '\t') {
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
if (i >= s.length) {
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
var buf = s.substring(0, i);
|
|
|
|
while (i < s.length) {
|
|
|
|
ch = s[i++];
|
|
|
|
switch (ch) {
|
|
|
|
case '&':
|
|
|
|
buf += '&';
|
|
|
|
break;
|
|
|
|
case '<':
|
|
|
|
buf += '<';
|
|
|
|
break;
|
|
|
|
case '\"':
|
|
|
|
buf += '"';
|
|
|
|
break;
|
|
|
|
case '\n':
|
|
|
|
buf += '
';
|
|
|
|
break;
|
|
|
|
case '\r':
|
|
|
|
buf += '
';
|
|
|
|
break;
|
|
|
|
case '\t':
|
|
|
|
buf += '	';
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
buf += ch;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
function DOMElement(name) {
|
|
|
|
this.nodeName = name;
|
|
|
|
this.childNodes = [];
|
|
|
|
this.attributes = {};
|
|
|
|
this.textContent = '';
|
2015-10-28 07:48:10 +09:00
|
|
|
|
|
|
|
if (name === 'style') {
|
|
|
|
this.sheet = {
|
|
|
|
cssRules: [],
|
2018-12-06 21:55:15 +09:00
|
|
|
insertRule: function(rule) {
|
2015-10-28 07:48:10 +09:00
|
|
|
this.cssRules.push(rule);
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
2014-08-14 05:04:47 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
DOMElement.prototype = {
|
2017-09-13 06:23:41 +09:00
|
|
|
getAttribute: function DOMElement_getAttribute(name) {
|
|
|
|
if (name in this.attributes) {
|
|
|
|
return this.attributes[name];
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
},
|
2014-08-14 05:04:47 +09:00
|
|
|
|
2017-06-19 21:11:13 +09:00
|
|
|
getAttributeNS: function DOMElement_getAttributeNS(NS, name) {
|
2017-07-15 01:55:17 +09:00
|
|
|
// Fast path
|
|
|
|
if (name in this.attributes) {
|
|
|
|
return this.attributes[name];
|
|
|
|
}
|
|
|
|
// Slow path - used by test/unit/display_svg_spec.js
|
|
|
|
// Assuming that there is only one matching attribute for a given name,
|
|
|
|
// across all namespaces.
|
|
|
|
if (NS) {
|
|
|
|
var suffix = ':' + name;
|
|
|
|
for (var fullName in this.attributes) {
|
|
|
|
if (fullName.slice(-suffix.length) === suffix) {
|
|
|
|
return this.attributes[fullName];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
2017-06-19 21:11:13 +09:00
|
|
|
},
|
|
|
|
|
2017-09-13 06:23:41 +09:00
|
|
|
setAttribute: function DOMElement_setAttribute(name, value) {
|
2014-08-14 05:04:47 +09:00
|
|
|
value = value || '';
|
|
|
|
value = xmlEncode(value);
|
|
|
|
this.attributes[name] = value;
|
|
|
|
},
|
|
|
|
|
2017-09-13 06:23:41 +09:00
|
|
|
setAttributeNS: function DOMElement_setAttributeNS(NS, name, value) {
|
|
|
|
this.setAttribute(name, value);
|
|
|
|
},
|
|
|
|
|
2014-08-14 05:04:47 +09:00
|
|
|
appendChild: function DOMElement_appendChild(element) {
|
|
|
|
var childNodes = this.childNodes;
|
2018-12-06 22:02:39 +09:00
|
|
|
if (!childNodes.includes(element)) {
|
2014-08-14 05:04:47 +09:00
|
|
|
childNodes.push(element);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2019-04-02 06:23:50 +09:00
|
|
|
hasChildNodes: function DOMElement_hasChildNodes() {
|
|
|
|
return this.childNodes.length !== 0;
|
|
|
|
},
|
|
|
|
|
2014-08-14 05:04:47 +09:00
|
|
|
cloneNode: function DOMElement_cloneNode() {
|
|
|
|
var newNode = new DOMElement(this.nodeName);
|
|
|
|
newNode.childNodes = this.childNodes;
|
|
|
|
newNode.attributes = this.attributes;
|
|
|
|
newNode.textContent = this.textContent;
|
|
|
|
return newNode;
|
|
|
|
},
|
2017-06-21 20:12:34 +09:00
|
|
|
|
|
|
|
// This method is offered for convenience. It is recommended to directly use
|
|
|
|
// getSerializer because that allows you to process the chunks as they come
|
|
|
|
// instead of requiring the whole image to fit in memory.
|
|
|
|
toString: function DOMElement_toString() {
|
|
|
|
var buf = [];
|
|
|
|
var serializer = this.getSerializer();
|
|
|
|
var chunk;
|
|
|
|
while ((chunk = serializer.getNext()) !== null) {
|
|
|
|
buf.push(chunk);
|
|
|
|
}
|
|
|
|
return buf.join('');
|
|
|
|
},
|
|
|
|
|
|
|
|
getSerializer: function DOMElement_getSerializer() {
|
|
|
|
return new DOMElementSerializer(this);
|
2018-12-06 21:55:15 +09:00
|
|
|
},
|
|
|
|
};
|
2015-02-03 00:12:52 +09:00
|
|
|
|
2017-06-21 20:12:34 +09:00
|
|
|
function DOMElementSerializer(node) {
|
|
|
|
this._node = node;
|
|
|
|
this._state = 0;
|
|
|
|
this._loopIndex = 0;
|
|
|
|
this._attributeKeys = null;
|
|
|
|
this._childSerializer = null;
|
|
|
|
}
|
|
|
|
DOMElementSerializer.prototype = {
|
|
|
|
/**
|
|
|
|
* Yields the next chunk in the serialization of the element.
|
|
|
|
*
|
|
|
|
* @returns {string|null} null if the element has fully been serialized.
|
|
|
|
*/
|
|
|
|
getNext: function DOMElementSerializer_getNext() {
|
|
|
|
var node = this._node;
|
|
|
|
switch (this._state) {
|
|
|
|
case 0: // Start opening tag.
|
|
|
|
++this._state;
|
|
|
|
return '<' + node.nodeName;
|
|
|
|
case 1: // Add SVG namespace if this is the root element.
|
|
|
|
++this._state;
|
|
|
|
if (node.nodeName === 'svg:svg') {
|
|
|
|
return ' xmlns:xlink="http://www.w3.org/1999/xlink"' +
|
|
|
|
' xmlns:svg="http://www.w3.org/2000/svg"';
|
|
|
|
}
|
2018-12-06 22:02:39 +09:00
|
|
|
/* falls through */
|
2017-06-21 20:12:34 +09:00
|
|
|
case 2: // Initialize variables for looping over attributes.
|
|
|
|
++this._state;
|
|
|
|
this._loopIndex = 0;
|
|
|
|
this._attributeKeys = Object.keys(node.attributes);
|
2018-12-06 22:02:39 +09:00
|
|
|
/* falls through */
|
2017-06-21 20:12:34 +09:00
|
|
|
case 3: // Serialize any attributes and end opening tag.
|
|
|
|
if (this._loopIndex < this._attributeKeys.length) {
|
|
|
|
var name = this._attributeKeys[this._loopIndex++];
|
|
|
|
return ' ' + name + '="' + xmlEncode(node.attributes[name]) + '"';
|
|
|
|
}
|
|
|
|
++this._state;
|
|
|
|
return '>';
|
|
|
|
case 4: // Serialize textContent for tspan/style elements.
|
|
|
|
if (node.nodeName === 'svg:tspan' || node.nodeName === 'svg:style') {
|
|
|
|
this._state = 6;
|
|
|
|
return xmlEncode(node.textContent);
|
|
|
|
}
|
|
|
|
++this._state;
|
|
|
|
this._loopIndex = 0;
|
2018-12-06 22:02:39 +09:00
|
|
|
/* falls through */
|
2017-06-21 20:12:34 +09:00
|
|
|
case 5: // Serialize child nodes (only for non-tspan/style elements).
|
|
|
|
var value;
|
|
|
|
while (true) {
|
|
|
|
value = this._childSerializer && this._childSerializer.getNext();
|
|
|
|
if (value !== null) {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
var nextChild = node.childNodes[this._loopIndex++];
|
|
|
|
if (nextChild) {
|
|
|
|
this._childSerializer = new DOMElementSerializer(nextChild);
|
|
|
|
} else {
|
|
|
|
this._childSerializer = null;
|
|
|
|
++this._state;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-12-06 22:02:39 +09:00
|
|
|
/* falls through */
|
2017-06-21 20:12:34 +09:00
|
|
|
case 6: // Ending tag.
|
|
|
|
++this._state;
|
|
|
|
return '</' + node.nodeName + '>';
|
|
|
|
case 7: // Done.
|
|
|
|
return null;
|
|
|
|
default:
|
|
|
|
throw new Error('Unexpected serialization state: ' + this._state);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2017-07-06 18:55:18 +09:00
|
|
|
const document = {
|
2018-12-06 21:55:15 +09:00
|
|
|
childNodes: [],
|
2014-08-14 05:04:47 +09:00
|
|
|
|
2015-12-22 04:46:50 +09:00
|
|
|
get currentScript() {
|
2018-12-06 21:55:15 +09:00
|
|
|
return { src: '', };
|
2015-12-22 04:46:50 +09:00
|
|
|
},
|
|
|
|
|
2015-10-28 07:48:10 +09:00
|
|
|
get documentElement() {
|
|
|
|
return this;
|
2014-08-14 05:04:47 +09:00
|
|
|
},
|
|
|
|
|
2018-12-06 21:55:15 +09:00
|
|
|
createElementNS: function(NS, element) {
|
2014-08-14 05:04:47 +09:00
|
|
|
var elObject = new DOMElement(element);
|
|
|
|
return elObject;
|
|
|
|
},
|
2015-10-28 07:48:10 +09:00
|
|
|
|
2018-12-06 21:55:15 +09:00
|
|
|
createElement: function(element) {
|
2015-10-28 07:48:10 +09:00
|
|
|
return this.createElementNS('', element);
|
|
|
|
},
|
|
|
|
|
2018-12-06 21:55:15 +09:00
|
|
|
getElementsByTagName: function(element) {
|
2015-10-28 07:48:10 +09:00
|
|
|
if (element === 'head') {
|
|
|
|
return [this.head || (this.head = new DOMElement('head'))];
|
|
|
|
}
|
|
|
|
return [];
|
2018-12-06 21:55:15 +09:00
|
|
|
},
|
2014-08-14 05:04:47 +09:00
|
|
|
};
|
2017-05-08 12:32:44 +09:00
|
|
|
|
2018-12-06 21:55:15 +09:00
|
|
|
function Image() {
|
2017-05-08 12:32:44 +09:00
|
|
|
this._src = null;
|
|
|
|
this.onload = null;
|
|
|
|
}
|
|
|
|
Image.prototype = {
|
2018-12-06 21:55:15 +09:00
|
|
|
get src() {
|
2017-05-08 12:32:44 +09:00
|
|
|
return this._src;
|
|
|
|
},
|
2018-12-06 21:55:15 +09:00
|
|
|
set src(value) {
|
2017-05-08 12:32:44 +09:00
|
|
|
this._src = value;
|
|
|
|
if (this.onload) {
|
|
|
|
this.onload();
|
|
|
|
}
|
2018-12-06 21:55:15 +09:00
|
|
|
},
|
|
|
|
};
|
2017-05-08 12:32:44 +09:00
|
|
|
|
2017-07-06 18:55:18 +09:00
|
|
|
exports.document = document;
|
|
|
|
exports.Image = Image;
|
|
|
|
|
|
|
|
var exported_symbols = Object.keys(exports);
|
|
|
|
|
|
|
|
exports.setStubs = function(namespace) {
|
|
|
|
exported_symbols.forEach(function(key) {
|
|
|
|
console.assert(!(key in namespace), 'property should not be set: ' + key);
|
|
|
|
namespace[key] = exports[key];
|
|
|
|
});
|
|
|
|
};
|
|
|
|
exports.unsetStubs = function(namespace) {
|
|
|
|
exported_symbols.forEach(function(key) {
|
|
|
|
console.assert(key in namespace, 'property should be set: ' + key);
|
|
|
|
delete namespace[key];
|
|
|
|
});
|
|
|
|
};
|