Adding jshint grunt task
[fnpeditor.git] / modules / documentCanvas / canvas.js
1 define([\r
2 'libs/jquery-1.9.1.min',\r
3 'libs/underscore-min',\r
4 'modules/documentCanvas/transformations',\r
5 'modules/documentCanvas/canvasNode',\r
6 'libs/text!./template.html'\r
7 ], function($, _, transformations, canvasNode, template) {\r
8 \r
9 'use strict';\r
10 \r
11 var Canvas = function(html) {\r
12     this.dom = $(template);\r
13     this.content = this.dom.find('#rng-module-documentCanvas-content');\r
14     this.setHTML(html);\r
15 };\r
16 \r
17 Canvas.prototype.setHTML = function(html) {\r
18     if(html) {\r
19         this.content.html(html);\r
20     }\r
21 };\r
22 \r
23 Canvas.prototype.getContent = function() {\r
24     return this.content.contents();\r
25 };\r
26 \r
27 Canvas.prototype.findNodes = function(desc) {\r
28     var selector = '';\r
29     if(typeof desc === 'string') {\r
30         selector = desc;\r
31     }\r
32     else {\r
33         if(desc.klass)\r
34             selector += '[wlxml-class=' + desc.klass + ']';\r
35         if(desc.tag)\r
36             selector += '[wlxml-tag=' + desc.tag + ']';\r
37     }\r
38     var toret = [];\r
39     this.content.find(selector).each(function() {\r
40         toret.push(canvasNode.create($(this)));\r
41     });\r
42     return toret;\r
43 };\r
44 \r
45 Canvas.prototype.getNodeById = function(id) {\r
46     return canvasNode.create($(this.content.find('#' +id)));\r
47 }\r
48 \r
49 Canvas.prototype.nodeAppend = function(options) {\r
50     var element; // = $(this.content.find('#' + options.context.id).get(0));\r
51     if(options.to === 'root') {\r
52         element = this.content;\r
53     } else {\r
54         element = $(this.content.find('#' + options.to.getId()).get(0));\r
55     }\r
56     element.append(options.node.dom);\r
57 };\r
58 \r
59 Canvas.prototype.nodeInsertAfter = function(options) {\r
60     var element = $(this.content.find('#' + options.after.getId()).get(0));\r
61     element.after(options.node.dom);\r
62 };\r
63 \r
64 Canvas.prototype.nodeWrap = function(options) {\r
65     options = _.extend({textNodeIdx: 0}, options);\r
66     if(typeof options.textNodeIdx === 'number')\r
67         options.textNodeIdx = [options.textNodeIdx];\r
68     \r
69     var container = $(this.content.find('#' + options.inside.getId()).get(0)),\r
70         containerContent = container.contents(),\r
71         idx1 = Math.min.apply(Math, options.textNodeIdx),\r
72         idx2 = Math.max.apply(Math, options.textNodeIdx),\r
73         textNode1 = $(containerContent.get(idx1)),\r
74         textNode2 = $(containerContent.get(idx2)),\r
75         sameNode = textNode1.get(0) === textNode2.get(0),\r
76         prefixOutside = textNode1.text().substr(0, options.offsetStart),\r
77         prefixInside = textNode1.text().substr(options.offsetStart),\r
78         suffixInside = textNode2.text().substr(0, options.offsetEnd),\r
79         suffixOutside = textNode2.text().substr(options.offsetEnd)\r
80     ;\r
81     \r
82     textNode1.after(options._with.dom);\r
83     textNode1.detach();\r
84     \r
85     options._with.dom.before(prefixOutside);\r
86     if(sameNode) {\r
87         var core = textNode1.text().substr(options.offsetStart, options.offsetEnd - options.offsetStart);\r
88         options._with.setContent(core);\r
89     } else {\r
90         textNode2.detach();\r
91         options._with.dom.append(prefixInside);\r
92         for(var i = idx1 + 1; i < idx2; i++) {\r
93             options._with.dom.append(containerContent[i]);\r
94         }\r
95         options._with.dom.append(suffixInside);\r
96     }\r
97     options._with.dom.after(suffixOutside);\r
98 };\r
99 \r
100 Canvas.prototype.nodeSplit = function(options) {\r
101     options = _.extend({textNodeIdx: 0}, options);\r
102     \r
103     var nodeToSplit = $(this.content.find('#' + options.node.getId()).get(0));\r
104     \r
105     var nodeContents = nodeToSplit.contents();\r
106     if(nodeContents.length === 0 || \r
107        nodeContents.length - 1 < options.textNodeIdx || \r
108        nodeContents.get(options.textNodeIdx).nodeType != 3)\r
109         return false;\r
110     \r
111     var textNode = $(nodeContents.get(options.textNodeIdx));\r
112 \r
113     var succeedingNodes = [];\r
114     var passed = false;\r
115     nodeContents.each(function() {\r
116         var node = this;\r
117         if(passed)\r
118             succeedingNodes.push(node);\r
119         if(node === textNode.get(0))\r
120             passed = true;\r
121     });\r
122     \r
123     var prefix = $.trim(textNode.text().substr(0, options.offset));\r
124     var suffix = $.trim(textNode.text().substr(options.offset));\r
125     \r
126     textNode.before(prefix);\r
127     textNode.remove();\r
128     \r
129     var newNode = canvasNode.create({tag: nodeToSplit.attr('wlxml-tag'), klass: nodeToSplit.attr('wlxml-class')});\r
130     newNode.dom.append(suffix);\r
131     succeedingNodes.forEach(function(node) {\r
132         newNode.dom.append(node)\r
133     });\r
134     nodeToSplit.after(newNode.dom);\r
135     return newNode;\r
136 };\r
137 \r
138 Canvas.prototype.nodeRemove = function(options) {\r
139     var toRemove = $(this.content.find('#' + options.node.getId()).get(0));\r
140     toRemove.remove();\r
141 };\r
142 \r
143 Canvas.prototype.listCreate = function(options) {\r
144     var element1 = $(this.content.find('#' + options.start.getId()).get(0));\r
145     var element2 = $(this.content.find('#' + options.end.getId()).get(0));\r
146     if(!element1.parent().get(0) === element2.parent().get(0))\r
147         return false;\r
148         \r
149     var parent = element1.parent();\r
150     \r
151     if(parent.contents().index(element1) > parent.contents().index(element2)) {\r
152         var tmp = element1;\r
153         element1 = element2;\r
154         element2 = tmp;\r
155     }\r
156     \r
157     var nodesToWrap = [];\r
158     \r
159     var place = 'before';\r
160     var canvas = this;\r
161     parent.contents().each(function() {\r
162         var node = this;\r
163         if(node === element1.get(0))\r
164             place = 'inside';\r
165         if(place === 'inside') {\r
166             var $node;\r
167             if(node.nodeType === 3) {\r
168                 $node = canvasNode.create({tag: 'div', content: $.trim(node.data)}).dom; //canvas._createNode('div').text(node.data);\r
169                 $(node).remove();\r
170             }\r
171             else {\r
172                 $node = $(node);\r
173             }\r
174             $node.attr('wlxml-class', 'item');\r
175             nodesToWrap.push($node);\r
176         }\r
177         if(node === element2.get(0))\r
178             return false;\r
179     });\r
180     \r
181     var list = canvasNode.create({tag: 'div', klass: 'list-items' + (options.type === 'enum' ? '-enum' : '')}).dom; //this._createNode('div', 'list-items');\r
182     \r
183     var parentNode = options.start.parent();\r
184     \r
185     var toret;\r
186     if(parentNode && parentNode.isOfClass('list-items')) {\r
187         list.wrap('<div wlxml-tag="div" wlxml-class="item" class="canvas-silent-item">');\r
188         toret = list.parent();\r
189     } else {\r
190         toret = list;\r
191     }\r
192         \r
193     \r
194     element1.before(toret);\r
195     \r
196     nodesToWrap.forEach(function(node) {\r
197         node.remove();\r
198         list.append(node);\r
199     });\r
200 };\r
201 \r
202 Canvas.prototype.listRemove = function(options) {\r
203     var pointerElement = $(this.content.find('#' + options.pointer.getId()));\r
204     var listElement = options.pointer.getClass() === 'list-items' ? pointerElement : \r
205         pointerElement.parents('[wlxml-class|="list-items"][wlxml-tag]');\r
206     \r
207     var nested = false;\r
208     if(listElement.length > 1) {\r
209         listElement = $(listElement[0]);\r
210         nested = true;\r
211     }\r
212     \r
213     if(nested) {\r
214         listElement.unwrap();\r
215     } else {\r
216         listElement.find('[wlxml-class=item]').each(function() {\r
217             $(this).removeAttr('wlxml-class');\r
218         });\r
219     }\r
220     listElement.children().unwrap();\r
221 };\r
222 \r
223 Canvas.prototype.getPrecedingNode = function(options) {\r
224     var element = $(this.content.find('#' + options.node.getId()).get(0));\r
225     var prev = element.prev()\r
226     if(prev.length === 0)\r
227         prev = element.parent();\r
228     return canvasNode.create(prev);\r
229 };\r
230 \r
231 Canvas.prototype.nodeInsideList = function(options) {\r
232     if(options.node) {\r
233         if(options.node.isOfClass('list-items') || options.node.isOfClass('item'))\r
234             return true;\r
235         var pointerElement = $(this.content.find('#' + options.node.getId()));\r
236         return pointerElement.parents('[wlxml-class=list-items], [wlxml-class=item]').length > 0;\r
237     }\r
238     return false;\r
239 };\r
240 \r
241 \r
242 return {\r
243     create: function(desc) { return new Canvas(desc); }\r
244 };\r
245 \r
246 });