+ nodeType: Node.TEXT_NODE,
+
+ getText: function() {
+ return this.nativeNode.data;
+ },
+
+ setText: function(text) {
+ //console.log('smartxml: ' + text);
+ this.nativeNode.data = text;
+ this.triggerTextChangeEvent();
+ },
+
+ appendText: function(text) {
+ this.nativeNode.data = this.nativeNode.data + text;
+ this.triggerTextChangeEvent();
+ },
+
+ prependText: function(text) {
+ this.nativeNode.data = text + this.nativeNode.data;
+ this.triggerTextChangeEvent();
+ },
+
+ wrapWith: function(desc) {
+ if(typeof desc.start === 'number' && typeof desc.end === 'number') {
+ return this.document._wrapText({
+ inside: this.parent(),
+ textNodeIdx: this.parent().indexOf(this),
+ offsetStart: Math.min(desc.start, desc.end),
+ offsetEnd: Math.max(desc.start, desc.end),
+ _with: {tagName: desc.tagName, attrs: desc.attrs}
+ });
+ } else {
+ return DocumentNode.prototype.wrapWith.call(this, desc);
+ }
+ },
+
+ split: function(params) {
+ var parentElement = this.parent(),
+ passed = false,
+ succeedingChildren = [],
+ prefix = this.getText().substr(0, params.offset),
+ suffix = this.getText().substr(params.offset);
+
+ parentElement.contents().forEach(function(child) {
+ if(passed) {
+ succeedingChildren.push(child);
+ }
+ if(child.sameNode(this)) {
+ passed = true;
+ }
+ }.bind(this));
+
+ if(prefix.length > 0) {
+ this.setText(prefix);
+ }
+ else {
+ this.detach();
+ }
+
+ var attrs = {};
+ parentElement.getAttrs().forEach(function(attr) {attrs[attr.name] = attr.value; });
+ var newElement = this.document.createDocumentNode({tagName: parentElement.getTagName(), attrs: attrs});
+ parentElement.after(newElement);
+
+ if(suffix.length > 0) {
+ newElement.append({text: suffix});
+ }
+ succeedingChildren.forEach(function(child) {
+ newElement.append(child);
+ });
+
+ return {first: parentElement, second: newElement};
+ },
+
+ triggerTextChangeEvent: function() {
+ var event = new events.ChangeEvent('nodeTextChange', {node: this});
+ this.document.trigger('change', event);
+ }
+});
+
+
+var parseXML = function(xml) {
+ return $($.trim(xml))[0];
+};
+
+var registerTransformation = function(desc, name, target) {
+ var Transformation = transformations.createContextTransformation(desc, name);
+ target[name] = function(args) {
+ var instance = this;
+ return instance.transform(Transformation, args);
+ }
+};
+
+var registerMethod = function(methodName, method, target) {
+ if(target[methodName]) {
+ throw new Error('Cannot extend {target} with method name {methodName}. Name already exists.'
+ .replace('{target}', target)
+ .replace('{methodName}', methodName)
+ );
+ }
+ target[methodName] = method;
+};
+
+
+var Document = function(xml) {
+ this.loadXML(xml);
+ this.undoStack = [];
+ this.redoStack = [];
+ this._transformationLevel = 0;
+
+ this._nodeMethods = {};
+ this._nodeTransformations = {};
+};
+
+$.extend(Document.prototype, Backbone.Events, {
+ ElementNodeFactory: ElementNode,
+ TextNodeFactory: TextNode,
+
+ createDocumentNode: function(from) {
+ if(!(from instanceof Node)) {
+ if(from.text !== undefined) {
+ /* globals document */
+ from = document.createTextNode(from.text);
+ } else {
+ var node = $('<' + from.tagName + '>');
+
+ _.keys(from.attrs || {}).forEach(function(key) {
+ node.attr(key, from.attrs[key]);
+ });
+
+ from = node[0];
+ }
+ }
+ var Factory;
+ if(from.nodeType === Node.TEXT_NODE) {
+ Factory = this.TextNodeFactory;
+ } else if(from.nodeType === Node.ELEMENT_NODE) {
+ Factory = this.ElementNodeFactory;
+ }
+ var toret = new Factory(from, this);
+ _.extend(toret, this._nodeMethods);
+ _.extend(toret, this._nodeTransformations);
+ return toret;
+ },
+
+ loadXML: function(xml, options) {
+ options = options || {};
+ defineDocumentProperties(this, $(parseXML(xml)));
+ if(!options.silent) {
+ this.trigger('contentSet');
+ }
+ },
+
+ toXML: function() {
+ return this.root.toXML();
+ },
+
+ containsNode: function(node) {
+ return this.root && (node.nativeNode === this.root.nativeNode || node._$.parents().index(this.root._$) !== -1);
+ },
+
+ wrapNodes: function(params) {
+ if(!(params.node1.parent().sameNode(params.node2.parent()))) {
+ throw new Error('Wrapping non-sibling nodes not supported.');
+ }
+
+ var parent = params.node1.parent(),
+ parentContents = parent.contents(),
+ wrapper = this.createDocumentNode({
+ tagName: params._with.tagName,
+ attrs: params._with.attrs}),
+ idx1 = parent.indexOf(params.node1),
+ idx2 = parent.indexOf(params.node2);
+
+ if(idx1 > idx2) {
+ var tmp = idx1;
+ idx1 = idx2;
+ idx2 = tmp;
+ }
+
+ var insertingMethod, insertingTarget;
+ if(idx1 === 0) {
+ insertingMethod = 'prepend';
+ insertingTarget = parent;
+ } else {
+ insertingMethod = 'after';
+ insertingTarget = parentContents[idx1-1];
+ }
+
+ for(var i = idx1; i <= idx2; i++) {
+ wrapper.append(parentContents[i].detach());
+ }
+
+ insertingTarget[insertingMethod](wrapper);
+ return wrapper;
+ },
+
+ getSiblingParents: function(params) {
+ var parents1 = [params.node1].concat(params.node1.parents()).reverse(),
+ parents2 = [params.node2].concat(params.node2.parents()).reverse(),
+ noSiblingParents = null;
+
+ if(parents1.length === 0 || parents2.length === 0 || !(parents1[0].sameNode(parents2[0]))) {
+ return noSiblingParents;
+ }
+
+ var i;
+ for(i = 0; i < Math.min(parents1.length, parents2.length); i++) {
+ if(parents1[i].sameNode(parents2[i])) {
+ continue;
+ }
+ break;
+ }
+ return {node1: parents1[i], node2: parents2[i]};
+ },
+
+ _wrapText: function(params) {
+ params = _.extend({textNodeIdx: 0}, params);
+ if(typeof params.textNodeIdx === 'number') {
+ params.textNodeIdx = [params.textNodeIdx];
+ }
+
+ var contentsInside = params.inside.contents(),
+ idx1 = Math.min.apply(Math, params.textNodeIdx),
+ idx2 = Math.max.apply(Math, params.textNodeIdx),
+ textNode1 = contentsInside[idx1],
+ textNode2 = contentsInside[idx2],
+ sameNode = textNode1.sameNode(textNode2),
+ prefixOutside = textNode1.getText().substr(0, params.offsetStart),
+ prefixInside = textNode1.getText().substr(params.offsetStart),
+ suffixInside = textNode2.getText().substr(0, params.offsetEnd),
+ suffixOutside = textNode2.getText().substr(params.offsetEnd)
+ ;
+
+ if(!(textNode1.parent().sameNode(textNode2.parent()))) {
+ throw new Error('Wrapping text in non-sibling text nodes not supported.');
+ }
+
+ var wrapperElement = this.createDocumentNode({tagName: params._with.tagName, attrs: params._with.attrs});
+ textNode1.after(wrapperElement);
+ textNode1.detach();
+
+ if(prefixOutside.length > 0) {
+ wrapperElement.before({text:prefixOutside});
+ }
+ if(sameNode) {
+ var core = textNode1.getText().substr(params.offsetStart, params.offsetEnd - params.offsetStart);
+ wrapperElement.append({text: core});
+ } else {
+ textNode2.detach();
+ if(prefixInside.length > 0) {
+ wrapperElement.append({text: prefixInside});
+ }
+ for(var i = idx1 + 1; i < idx2; i++) {
+ wrapperElement.append(contentsInside[i]);
+ }
+ if(suffixInside.length > 0) {
+ wrapperElement.append({text: suffixInside});
+ }
+ }
+ if(suffixOutside.length > 0) {
+ wrapperElement.after({text: suffixOutside});
+ }
+ return wrapperElement;
+ },
+
+ trigger: function() {
+ //console.log('trigger: ' + arguments[0] + (arguments[1] ? ', ' + arguments[1].type : ''));
+ Backbone.Events.trigger.apply(this, arguments);
+ },
+
+ getNodeInsertion: function(node) {
+ var insertion = {};
+ if(node instanceof DocumentNode) {
+ insertion.ofNode = node;
+ insertion.insertsNew = !this.containsNode(node);
+ } else {
+ insertion.ofNode = this.createDocumentNode(node);
+ insertion.insertsNew = true;
+ }
+ return insertion;
+ },
+
+ replaceRoot: function(node) {
+ var insertion = this.getNodeInsertion(node);
+ this.root.detach();
+ defineDocumentProperties(this, insertion.ofNode._$);
+ insertion.ofNode.triggerChangeEvent('nodeAdded');
+ return insertion.ofNode;
+ },
+
+ registerMethod: function(methodName, method) {
+ registerMethod(methodName, method, this);
+ },
+
+ registerNodeMethod: function(methodName, method) {
+ registerMethod(methodName, method, this._nodeMethods);
+ },
+
+ registerDocumentTransformation: function(desc, name) {
+ registerTransformation(desc, name, this);
+ },
+
+ registerNodeTransformation: function(desc, name) {
+ registerTransformation(desc, name, this._nodeTransformations);
+ },
+
+ registerExtension: function(extension) {
+ //debugger;
+ var doc = this,
+ existingPropertyNames = _.values(this);
+
+ ['document', 'documentNode'].forEach(function(dstName) {
+ var dstExtension = extension[dstName];
+ if(dstExtension) {
+ if(dstExtension.methods) {
+ _.pairs(dstExtension.methods).forEach(function(pair) {
+ var methodName = pair[0],
+ method = pair[1],
+ operation;
+ operation = {document: 'registerMethod', documentNode: 'registerNodeMethod'}[dstName];
+ doc[operation](methodName, method);
+
+ });
+ }
+
+ if(dstExtension.transformations) {
+ _.pairs(dstExtension.transformations).forEach(function(pair) {
+ var name = pair[0],
+ desc = pair[1],
+ operation;
+ operation = {document: 'registerDocumentTransformation', documentNode: 'registerNodeTransformation'}[dstName];
+ doc[operation](desc, name);
+ });
+ }
+ }
+ });
+ },
+
+ transform: function(Transformation, args) {
+ //console.log('transform');
+ var toret, transformation;
+
+ if(typeof Transformation === 'function') {
+ transformation = new Transformation(this, this, args);
+ } else {
+ transformation = Transformation;
+ }
+ if(transformation) {
+ this._transformationLevel++;
+ toret = transformation.run();
+ if(this._transformationLevel === 1) {
+ this.undoStack.push(transformation);
+ }
+ this._transformationLevel--;
+ //console.log('clearing redo stack');
+ this.redoStack = [];
+ return toret;
+ } else {
+ throw new Error('Transformation ' + transformation + ' doesn\'t exist!');
+ }
+ },
+ undo: function() {
+ var transformation = this.undoStack.pop();
+ if(transformation) {
+ transformation.undo();
+ this.redoStack.push(transformation);
+ }
+ },
+ redo: function() {
+ var transformation = this.redoStack.pop();
+ if(transformation) {
+ transformation.run();
+ this.undoStack.push(transformation);
+ }
+ },
+
+ getNodeByPath: function(path) {
+ var toret = this.root;
+ path.forEach(function(idx) {
+ toret = toret.contents()[idx];
+ });
+ return toret;
+ }
+});
+
+var defineDocumentProperties = function(doc, $document) {
+ Object.defineProperty(doc, 'root', {get: function() {
+ return doc.createDocumentNode($document[0]);
+ }, configurable: true});
+ Object.defineProperty(doc, 'dom', {get: function() {
+ return $document[0];
+ }, configurable: true});
+};