Fixing splitting at the very beginning of a text element
[fnpeditor.git] / modules / documentCanvas / canvas.js
index 760bf60..ab8a578 100644 (file)
-define([\r
-'libs/jquery-1.9.1.min',\r
-'libs/underscore-min',\r
-'modules/documentCanvas/transformations',\r
-'modules/documentCanvas/canvasNode',\r
-'libs/text!./template.html'\r
-], function($, _, transformations, canvasNode, template) {\r
-\r
-'use strict';\r
-\r
-var Canvas = function(html) {\r
-    this.dom = $(template);\r
-    this.content = this.dom.find('#rng-module-documentCanvas-content');\r
-    this.setHTML(html);\r
-};\r
-\r
-Canvas.prototype.setHTML = function(html) {\r
-    if(html) {\r
-        this.content.html(html);\r
-    }\r
-};\r
-\r
-Canvas.prototype.getContent = function() {\r
-    return this.content.contents();\r
-};\r
-\r
-Canvas.prototype.findNodes = function(desc) {\r
-    var selector = '';\r
-    if(typeof desc === 'string') {\r
-        selector = desc;\r
-    }\r
-    else {\r
-        if(desc.klass)\r
-            selector += '[wlxml-class=' + desc.klass + ']';\r
-        if(desc.tag)\r
-            selector += '[wlxml-tag=' + desc.tag + ']';\r
-    }\r
-    var toret = [];\r
-    this.content.find(selector).each(function() {\r
-        toret.push(canvasNode.create($(this)));\r
-    });\r
-    return toret;\r
-};\r
-\r
-Canvas.prototype.getNodeById = function(id) {\r
-    return canvasNode.create($(this.content.find('#' +id)));\r
-}\r
-\r
-Canvas.prototype.nodeAppend = function(options) {\r
-    var element; // = $(this.content.find('#' + options.context.id).get(0));\r
-    if(options.to === 'root') {\r
-        element = this.content;\r
-    } else {\r
-        element = $(this.content.find('#' + options.to.getId()).get(0));\r
-    }\r
-    element.append(options.node.dom);\r
-};\r
-\r
-Canvas.prototype.nodeInsertAfter = function(options) {\r
-    var element = $(this.content.find('#' + options.after.getId()).get(0));\r
-    element.after(options.node.dom);\r
-};\r
-\r
-Canvas.prototype.nodeWrap = function(options) {\r
-    options = _.extend({textNodeIdx: 0}, options);\r
-    if(typeof options.textNodeIdx === 'number')\r
-        options.textNodeIdx = [options.textNodeIdx];\r
-    var container = $(this.content.find('#' + options.inside.getId()).get(0));\r
-    \r
-    var containerContent = container.contents();\r
-    var idx1 = Math.min.apply(Math, options.textNodeIdx);\r
-    var idx2 = Math.max.apply(Math, options.textNodeIdx);\r
-    var textNode1 = $(containerContent.get(idx1));\r
-    var textNode2 = $(containerContent.get(idx2));\r
-    var sameNode = textNode1.get(0) === textNode2.get(0);\r
-    textNode1.after(options._with.dom);\r
-    textNode1.detach();\r
-    if(!sameNode)\r
-        textNode2.detach();\r
-    \r
-    var prefixOutside = textNode1.text().substr(0, options.offsetStart);\r
-    var prefixInside = textNode1.text().substr(options.offsetStart)\r
-    var suffixInside = textNode2.text().substr(0, options.offsetEnd)\r
-    var suffixOutside = textNode2.text().substr(options.offsetEnd);\r
-    var core;\r
-    if(sameNode)\r
-        core = textNode1.text().substr(options.offsetStart, options.offsetEnd - options.offsetStart);\r
-    \r
-    options._with.dom.before(prefixOutside);\r
-    if(sameNode) {\r
-        options._with.setContent(core);\r
-    } else {\r
-        options._with.dom.append(prefixInside);\r
-        for(var i = idx1 + 1; i < idx2; i++) {\r
-            options._with.dom.append(containerContent[i]);\r
-        }\r
-        options._with.dom.append(suffixInside);\r
-    }\r
-    options._with.dom.after(suffixOutside);\r
-};\r
-\r
-Canvas.prototype.nodeSplit = function(options) {\r
-    options = _.extend({textNodeIdx: 0}, options);\r
-    \r
-    var nodeToSplit = $(this.content.find('#' + options.node.getId()).get(0));\r
-    \r
-    var nodeContents = nodeToSplit.contents();\r
-    if(nodeContents.length === 0 || \r
-       nodeContents.length - 1 < options.textNodeIdx || \r
-       nodeContents.get(options.textNodeIdx).nodeType != 3)\r
-        return false;\r
-    \r
-    var textNode = $(nodeContents.get(options.textNodeIdx));\r
-\r
-    var succeedingNodes = [];\r
-    var passed = false;\r
-    nodeContents.each(function() {\r
-        var node = this;\r
-        if(passed)\r
-            succeedingNodes.push(node);\r
-        if(node === textNode.get(0))\r
-            passed = true;\r
-    });\r
-    \r
-    var prefix = $.trim(textNode.text().substr(0, options.offset));\r
-    var suffix = $.trim(textNode.text().substr(options.offset));\r
-    \r
-    textNode.before(prefix);\r
-    textNode.remove();\r
-    \r
-    var newNode = canvasNode.create({tag: nodeToSplit.attr('wlxml-tag'), klass: nodeToSplit.attr('wlxml-class')});\r
-    newNode.dom.append(suffix);\r
-    succeedingNodes.forEach(function(node) {\r
-        newNode.dom.append(node)\r
-    });\r
-    nodeToSplit.after(newNode.dom);\r
-    return newNode;\r
-};\r
-\r
-Canvas.prototype.nodeRemove = function(options) {\r
-    var toRemove = $(this.content.find('#' + options.node.getId()).get(0));\r
-    toRemove.remove();\r
-};\r
-\r
-Canvas.prototype.listCreate = function(options) {\r
-    var element1 = $(this.content.find('#' + options.start.getId()).get(0));\r
-    var element2 = $(this.content.find('#' + options.end.getId()).get(0));\r
-    if(!element1.parent().get(0) === element2.parent().get(0))\r
-        return false;\r
-        \r
-    var parent = element1.parent();\r
-    \r
-    if(parent.contents().index(element1) > parent.contents().index(element2)) {\r
-        var tmp = element1;\r
-        element1 = element2;\r
-        element2 = tmp;\r
-    }\r
-    \r
-    var nodesToWrap = [];\r
-    \r
-    var place = 'before';\r
-    var canvas = this;\r
-    parent.contents().each(function() {\r
-        var node = this;\r
-        if(node === element1.get(0))\r
-            place = 'inside';\r
-        if(place === 'inside') {\r
-            var $node;\r
-            if(node.nodeType === 3) {\r
-                $node = canvasNode.create({tag: 'div', content: $.trim(node.data)}).dom; //canvas._createNode('div').text(node.data);\r
-                $(node).remove();\r
-            }\r
-            else {\r
-                $node = $(node);\r
-            }\r
-            $node.attr('wlxml-class', 'item');\r
-            nodesToWrap.push($node);\r
-        }\r
-        if(node === element2.get(0))\r
-            return false;\r
-    });\r
-    \r
-    var list = canvasNode.create({tag: 'div', klass: 'list-items' + (options.type === 'enum' ? '-enum' : '')}).dom; //this._createNode('div', 'list-items');\r
-    \r
-    var parentNode = options.start.parent();\r
-    \r
-    var toret;\r
-    if(parentNode && parentNode.isOfClass('list-items')) {\r
-        list.wrap('<div wlxml-tag="div" wlxml-class="item" class="canvas-silent-item">');\r
-        toret = list.parent();\r
-    } else {\r
-        toret = list;\r
-    }\r
-        \r
-    \r
-    element1.before(toret);\r
-    \r
-    nodesToWrap.forEach(function(node) {\r
-        node.remove();\r
-        list.append(node);\r
-    });\r
-};\r
-\r
-Canvas.prototype.listRemove = function(options) {\r
-    var pointerElement = $(this.content.find('#' + options.pointer.getId()));\r
-    var listElement = options.pointer.getClass() === 'list-items' ? pointerElement : \r
-        pointerElement.parents('[wlxml-class|="list-items"][wlxml-tag]');\r
-    \r
-    var nested = false;\r
-    if(listElement.length > 1) {\r
-        listElement = $(listElement[0]);\r
-        nested = true;\r
-    }\r
-    \r
-    if(nested) {\r
-        listElement.unwrap();\r
-    } else {\r
-        listElement.find('[wlxml-class=item]').each(function() {\r
-            $(this).removeAttr('wlxml-class');\r
-        });\r
-    }\r
-    listElement.children().unwrap();\r
-};\r
-\r
-Canvas.prototype.getPrecedingNode = function(options) {\r
-    var element = $(this.content.find('#' + options.node.getId()).get(0));\r
-    var prev = element.prev()\r
-    if(prev.length === 0)\r
-        prev = element.parent();\r
-    return canvasNode.create(prev);\r
-};\r
-\r
-Canvas.prototype.nodeInsideList = function(options) {\r
-    if(options.node) {\r
-        if(options.node.isOfClass('list-items') || options.node.isOfClass('item'))\r
-            return true;\r
-        var pointerElement = $(this.content.find('#' + options.node.getId()));\r
-        return pointerElement.parents('[wlxml-class=list-items], [wlxml-class=item]').length > 0;\r
-    }\r
-    return false;\r
-};\r
-\r
-\r
-return {\r
-    create: function(desc) { return new Canvas(desc); }\r
-};\r
-\r
+define([
+'libs/jquery-1.9.1.min',
+'libs/underscore-min',
+'modules/documentCanvas/transformations',
+'modules/documentCanvas/canvasNode',
+'libs/text!./template.html'
+], function($, _, transformations, canvasNode, template) {
+
+'use strict';
+
+var Canvas = function(html) {
+    this.dom = $(template);
+    this.content = this.dom.find('#rng-module-documentCanvas-content');
+    this.setHTML(html);
+};
+
+Canvas.prototype.setHTML = function(html) {
+    if(html) {
+        this.content.html(html);
+    }
+};
+
+Canvas.prototype.getContent = function() {
+    return this.content.contents();
+};
+
+Canvas.prototype.findNodes = function(desc) {
+    var selector = '';
+    if(typeof desc === 'string') {
+        selector = desc;
+    }
+    else {
+        if(desc.klass)
+            selector += '[wlxml-class=' + desc.klass + ']';
+        if(desc.tag)
+            selector += '[wlxml-tag=' + desc.tag + ']';
+    }
+    var toret = [];
+    this.content.find(selector).each(function() {
+        toret.push(canvasNode.create($(this)));
+    });
+    return toret;
+};
+
+Canvas.prototype.getNodeById = function(id) {
+    return canvasNode.create($(this.content.find('#' +id)));
+};
+
+Canvas.prototype.nodeAppend = function(options) {
+    var element; // = $(this.content.find('#' + options.context.id).get(0));
+    if(options.to === 'root') {
+        element = this.content;
+    } else {
+        element = $(this.content.find('#' + options.to.getId()).get(0));
+    }
+    element.append(options.node.dom);
+};
+
+Canvas.prototype.nodeInsertAfter = function(options) {
+    var element = $(this.content.find('#' + options.after.getId()).get(0));
+    element.after(options.node.dom);
+};
+
+Canvas.prototype.nodeWrap = function(options) {
+    options = _.extend({textNodeIdx: 0}, options);
+    if(typeof options.textNodeIdx === 'number')
+        options.textNodeIdx = [options.textNodeIdx];
+    
+    var container = $(this.content.find('#' + options.inside.getId()).get(0)),
+        containerContent = container.contents(),
+        idx1 = Math.min.apply(Math, options.textNodeIdx),
+        idx2 = Math.max.apply(Math, options.textNodeIdx),
+        textNode1 = $(containerContent.get(idx1)),
+        textNode2 = $(containerContent.get(idx2)),
+        sameNode = textNode1.get(0) === textNode2.get(0),
+        prefixOutside = textNode1.text().substr(0, options.offsetStart),
+        prefixInside = textNode1.text().substr(options.offsetStart),
+        suffixInside = textNode2.text().substr(0, options.offsetEnd),
+        suffixOutside = textNode2.text().substr(options.offsetEnd)
+    ;
+    
+    textNode1.after(options._with.dom);
+    textNode1.detach();
+    
+    options._with.dom.before(prefixOutside);
+    if(sameNode) {
+        var core = textNode1.text().substr(options.offsetStart, options.offsetEnd - options.offsetStart);
+        options._with.setContent(core);
+    } else {
+        textNode2.detach();
+        options._with.dom.append(prefixInside);
+        for(var i = idx1 + 1; i < idx2; i++) {
+            options._with.dom.append(containerContent[i]);
+        }
+        options._with.dom.append(suffixInside);
+    }
+    options._with.dom.after(suffixOutside);
+};
+
+Canvas.prototype.nodeUnwrap = function(options) {
+
+    var removeWithJoin = function(node) {
+        var contents = node.parent().contents(),
+            idx = contents.index(node),
+            prev = idx > 0 ? contents[idx-1] : null,
+            next = idx + 1 < contents.length ? contents[idx+1] : null;
+
+        if(prev && prev.nodeType === 3 && next && next.nodeType === 3) {
+            prev.data = prev.data + next.data;
+            $(next).remove();
+        }
+        node.remove();
+    };
+
+    var toUnwrap = $(this.content.find('#' + options.node.getId()).get(0));
+
+
+    var parent = toUnwrap.parent();
+    var parentContents = parent.contents();
+
+    if(toUnwrap.contents().length !== 1 || toUnwrap.contents()[0].nodeType !== 3)
+        return false;
+
+    var idx = parentContents.index(toUnwrap);
+
+    var combineWith,
+        action;
+
+    if(idx > 0 && parentContents[idx-1].nodeType === 3) {
+        combineWith = parentContents[idx-1];
+        action = 'append';
+    } else if(idx + 1 < parentContents.length && parentContents[idx+1].nodeType === 3) {
+        combineWith = parentContents[idx+1];
+        action = 'prepend';
+    }
+
+    if(combineWith) {
+        var text = 
+                (action === 'prepend' ? toUnwrap.text() : '') +
+                combineWith.data +
+                (action === 'append' ? toUnwrap.text() : '')
+        ;
+        combineWith.data = text;
+        removeWithJoin(toUnwrap);
+    } else {
+        if(parentContents.length === 1 || idx === 0) {
+            parent.append(toUnwrap.text());
+        } else {
+            toUnwrap.prev().after(toUnwrap.text());
+        }
+        toUnwrap.remove();
+    }
+};
+
+Canvas.prototype.nodeSplit = function(options) {
+    options = _.extend({textNodeIdx: 0}, options);
+    
+    var nodeToSplit = $(this.content.find('#' + options.node.getId()).get(0));
+    
+    var nodeContents = nodeToSplit.contents();
+    if(nodeContents.length === 0 || 
+       nodeContents.length - 1 < options.textNodeIdx || 
+       nodeContents.get(options.textNodeIdx).nodeType != 3)
+        return false;
+    
+    var textNode = $(nodeContents.get(options.textNodeIdx));
+
+    var succeedingNodes = [];
+    var passed = false;
+    nodeContents.each(function() {
+        var node = this;
+        if(passed)
+            succeedingNodes.push(node);
+        if(node === textNode.get(0))
+            passed = true;
+    });
+    
+    var prefix = $.trim(textNode.text().substr(0, options.offset));
+    var suffix = $.trim(textNode.text().substr(options.offset));
+    
+    textNode.before(prefix);
+    textNode.remove();
+    
+    var newNode = canvasNode.create({tag: nodeToSplit.attr('wlxml-tag'), klass: nodeToSplit.attr('wlxml-class')});
+    newNode.dom.append(suffix);
+    succeedingNodes.forEach(function(node) {
+        newNode.dom.append(node);
+    });
+    nodeToSplit.after(newNode.dom);
+    return newNode;
+};
+
+Canvas.prototype.nodeRemove = function(options) {
+    var toRemove = $(this.content.find('#' + options.node.getId()).get(0));
+    toRemove.remove();
+};
+
+Canvas.prototype.listCreate = function(options) {
+    var element1 = $(this.content.find('#' + options.start.getId()).get(0));
+    var element2 = $(this.content.find('#' + options.end.getId()).get(0));
+    if(element1.parent().get(0) !== element2.parent().get(0))
+        return false;
+        
+    var parent = element1.parent();
+    
+    if(parent.contents().index(element1) > parent.contents().index(element2)) {
+        var tmp = element1;
+        element1 = element2;
+        element2 = tmp;
+    }
+    
+    var nodesToWrap = [];
+    
+    var place = 'before';
+    var canvas = this;
+    parent.contents().each(function() {
+        var node = this;
+        if(node === element1.get(0))
+            place = 'inside';
+        if(place === 'inside') {
+            var $node;
+            if(node.nodeType === 3) {
+                $node = canvasNode.create({tag: 'div', content: $.trim(node.data)}).dom; //canvas._createNode('div').text(node.data);
+                $(node).remove();
+            }
+            else {
+                $node = $(node);
+            }
+            $node.attr('wlxml-class', 'item');
+            nodesToWrap.push($node);
+        }
+        if(node === element2.get(0))
+            return false;
+    });
+    
+    var list = canvasNode.create({tag: 'div', klass: 'list-items' + (options.type === 'enum' ? '-enum' : '')}).dom; //this._createNode('div', 'list-items');
+    
+    var parentNode = options.start.parent();
+    
+    var toret;
+    if(parentNode && parentNode.isOfClass('list-items')) {
+        list.wrap('<div wlxml-tag="div" wlxml-class="item" class="canvas-silent-item">');
+        toret = list.parent();
+    } else {
+        toret = list;
+    }
+        
+    
+    element1.before(toret);
+    
+    nodesToWrap.forEach(function(node) {
+        node.remove();
+        list.append(node);
+    });
+};
+
+Canvas.prototype.listRemove = function(options) {
+    var pointerElement = $(this.content.find('#' + options.pointer.getId()));
+    var listElement = options.pointer.isOfClass('list-items') ? pointerElement : 
+        pointerElement.parents('[wlxml-class|="list-items"][wlxml-tag]');
+    
+    var nested = false,
+        nestedLists;
+    if(listElement.length > 1) {
+        listElement = $(listElement[0]);
+        nested = true;
+    }
+    
+    if(nested) {
+        // We are only moving one level up
+        listElement.unwrap();
+    } else {
+        // We are removing the whole list
+        nestedLists = listElement.find('[wlxml-class=item] > [wlxml-class|=list-items]');
+        nestedLists.unwrap();
+        listElement.find('[wlxml-class=item]').each(function() {
+            $(this).removeAttr('wlxml-class');
+        });
+    }
+
+    listElement.children().unwrap();
+
+    var c = this;
+    if(nestedLists) {
+        nestedLists.each(function() {
+            c.listRemove({pointer: canvasNode.create($(this))});
+        });
+    }
+};
+
+Canvas.prototype.getPrecedingNode = function(options) {
+    var element = $(this.content.find('#' + options.node.getId()).get(0));
+    var prev = element.prev();
+    if(prev.length === 0)
+        prev = element.parent();
+    return canvasNode.create(prev);
+};
+
+Canvas.prototype.nodeInsideList = function(options) {
+    if(options.node) {
+        if(options.node.isOfClass('list-items') || options.node.isOfClass('item'))
+            return true;
+        var pointerElement = $(this.content.find('#' + options.node.getId()));
+        return pointerElement.parents('[wlxml-class=list-items], [wlxml-class=item]').length > 0;
+    }
+    return false;
+};
+
+
+return {
+    create: function(desc) { return new Canvas(desc); }
+};
+
 });
\ No newline at end of file
 });
\ No newline at end of file