moving smartxml transformations into object properties name space
[fnpeditor.git] / src / smartxml / smartxml.js
index 41e8e19..831a96a 100644 (file)
@@ -38,12 +38,8 @@ var DocumentNode = function(nativeNode, document) {
 
 $.extend(DocumentNode.prototype, {
 
-    transform: function(name, args) {
-        var Transformation = this.transformations.get(name),
-            transformation;
-        if(Transformation) {
-            transformation = new Transformation(this.document, this, args);
-        }
+    transform: function(Transformation, args) {
+        var transformation = new Transformation(this.document, this, args);
         return this.document.transform(transformation);
     },
 
@@ -380,63 +376,6 @@ $.extend(ElementNode.prototype, {
     }
 });
 
-// trans
-
-// todo - split+append
-
-// ElementNode.prototype.transformations.register(transformations.createContextTransformation({
-//     name: 'smartxml.setAttr',
-//     impl: function(args) {
-//         this.setAttr(args.name, args.value);
-//     },
-//     getChangeRoot: function() {
-//         return this.context;
-//     }
-// }));
-
-// ElementNode.prototype.transformations.register(transformations.createContextTransformation({
-//     name: 'smartxml.setAttr2',
-//     impl: function(args) {
-//         this.prevAttr = this.getAttr(args.name);
-//         this.setAttr(args.name, args.value);
-//     },
-//     undo: function(args) {
-//         this.setAttr(args.name, this.prevAttr);
-//     }
-// }));
-
-// DocumentNode.prototype.transformations.register(transformations.createContextTransformation({
-//     name: 'smartxml.wrapWith',
-//     getChangeRoot: function() {
-//         return this.context.parent();
-//     },
-//     impl: function(args) {
-//         return this.wrapWith(args);
-//     }
-// }));
-
-// DocumentNode.prototype.transformations.register(transformations.createContextTransformation({
-//     name: 'smartxml.wrapText',
-//     getChangeRoot: function() {
-//         return this.context;
-//     },
-//     impl: function(args) {
-//         return this.wrapText(args);
-//     }
-// }));
-
-// DocumentNode.prototype.transformations.register(transformations.createContextTransformation({
-//     name: 'smartxml.detach',
-//     getChangeRoot: function() {
-//         return this.context.parent();
-//     },
-//     impl: function(args) {
-//         return this.detach();
-//     }
-// }));
-
-///
-
 var TextNode = function(nativeNode, document) {
     DocumentNode.call(this, nativeNode, document);
 };
@@ -524,66 +463,37 @@ $.extend(TextNode.prototype, {
 });
 
 
-// TextNode.prototype.transformations.register(transformations.createContextTransformation({
-//     name: 'rng.breakContent',
-//     // impl: function(args) {
-//     //     var node = this.context,
-//     //         newNodes, emptyNode, emptyText;
-//     //     newNodes = node.transform('smartxml.split', {offset: args.offset});
-//     //     [newNodes.first, newNodes.second].some(function(newNode) {
-//     //         if(!(newNode.contents().length)) {
-//     //             newNode.transform('smartxml.append', {text: ''});
-//     //             return true; // break
-//     //         }
-//     //     });
-//     //     return _.extend(newNodes, {emptyText: emptyText});
-//     // },
-//     impl: function(args) {
-//         var node = this,
-//             newNodes, emptyNode, emptyText;
-//         newNodes = node.split({offset: args.offset});
-//         [newNodes.first, newNodes.second].some(function(newNode) {
-//             if(!(newNode.contents().length)) {
-//                 newNode.append({text: ''});
-//                 return true; // break
-//             }
-//         });
-//         return _.extend(newNodes, {emptyText: emptyText});
-//     },
-//     getChangeRoot: function() {
-//         return this.context.parent().parent();
-//     },
-//     isAllowed: function(args) {
-//         var parent = this.parent();
-//         return !!(parent && parent.parent());
-//     }
-// }));
-
-
-// ElementNode.prototype.transformations.register(transformations.createContextTransformation({
-//     name: 'smartxml.setText',
-//     impl: function(args) {
-//         this.setText(args.text);
-//     },
-//     getChangeRoot: function() {
-//         return this.context;
-//     }
-// }));
-
-
 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.transformations = new transformations.TransformationStorage();
     
     this._nodeMethods = {};
-    this._nodeTransformations = new transformations.TransformationStorage();
+    this._nodeTransformations = {};
 };
 
 $.extend(Document.prototype, Backbone.Events, {
@@ -613,7 +523,7 @@ $.extend(Document.prototype, Backbone.Events, {
         }
         var toret = new Factory(from, this);
         _.extend(toret, this._nodeMethods);
-        toret.transformations = this._nodeTransformations;
+        _.extend(toret, this._nodeTransformations);
         return toret;
     },
 
@@ -764,31 +674,62 @@ $.extend(Document.prototype, Backbone.Events, {
     },
 
     registerMethod: function(methodName, method) {
-        this[methodName] = method;
+        registerMethod(methodName, method, this);
     },
 
-    registerTransformation: function(Transformation) {
-        return this.transformations.register(Transformation);
+    registerNodeMethod: function(methodName, method) {
+        registerMethod(methodName, method, this._nodeMethods);
     },
 
-    registerNodeMethod: function(methodName, method) {
-        this._nodeMethods[methodName] = method;
+    registerDocumentTransformation: function(desc, name) {
+        registerTransformation(desc, name, this);
     },
 
-    registerNodeTransformation: function(Transformation) {
-        this._nodeTransformations.register(Transformation);
+    registerNodeTransformation: function(desc, name) {
+        registerTransformation(desc, name, this._nodeTransformations);
     },
 
+    registerExtension: function(extension) {
+        //debugger;
+        var doc = this,
+            existingPropertyNames = _.values(this);
 
-    transform: function(transformation, args) {
-        //console.log('transform');
-        var Transformation, toret;
-        if(typeof transformation === 'string') {
-            Transformation = this.transformations.get(transformation);
-            if(Transformation) {
-                transformation = new Transformation(this, this, args);
+        ['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();
@@ -836,20 +777,6 @@ var defineDocumentProperties = function(doc, $document) {
     }, configurable: true});
 };
 
-// Document.prototype.transformations.register(transformations.createContextTransformation({
-//     name: 'smartxml.wrapNodes',
-//     // init: function() {
-
-//     // },
-//     // getChangeRoot: function() {
-//     //     return this.context;
-//     // },
-//     impl: function(args) {
-//         this.wrapNodes(args);
-//     },
-
-// }));
-
 
 return {
     documentFromXML: function(xml) {