760bf60f98a840e1865f0163ed87690973f6f442
[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     var container = $(this.content.find('#' + options.inside.getId()).get(0));\r
69     \r
70     var containerContent = container.contents();\r
71     var idx1 = Math.min.apply(Math, options.textNodeIdx);\r
72     var idx2 = Math.max.apply(Math, options.textNodeIdx);\r
73     var textNode1 = $(containerContent.get(idx1));\r
74     var textNode2 = $(containerContent.get(idx2));\r
75     var sameNode = textNode1.get(0) === textNode2.get(0);\r
76     textNode1.after(options._with.dom);\r
77     textNode1.detach();\r
78     if(!sameNode)\r
79         textNode2.detach();\r
80     \r
81     var prefixOutside = textNode1.text().substr(0, options.offsetStart);\r
82     var prefixInside = textNode1.text().substr(options.offsetStart)\r
83     var suffixInside = textNode2.text().substr(0, options.offsetEnd)\r
84     var suffixOutside = textNode2.text().substr(options.offsetEnd);\r
85     var core;\r
86     if(sameNode)\r
87         core = textNode1.text().substr(options.offsetStart, options.offsetEnd - options.offsetStart);\r
88     \r
89     options._with.dom.before(prefixOutside);\r
90     if(sameNode) {\r
91         options._with.setContent(core);\r
92     } else {\r
93         options._with.dom.append(prefixInside);\r
94         for(var i = idx1 + 1; i < idx2; i++) {\r
95             options._with.dom.append(containerContent[i]);\r
96         }\r
97         options._with.dom.append(suffixInside);\r
98     }\r
99     options._with.dom.after(suffixOutside);\r
100 };\r
101 \r
102 Canvas.prototype.nodeSplit = function(options) {\r
103     options = _.extend({textNodeIdx: 0}, options);\r
104     \r
105     var nodeToSplit = $(this.content.find('#' + options.node.getId()).get(0));\r
106     \r
107     var nodeContents = nodeToSplit.contents();\r
108     if(nodeContents.length === 0 || \r
109        nodeContents.length - 1 < options.textNodeIdx || \r
110        nodeContents.get(options.textNodeIdx).nodeType != 3)\r
111         return false;\r
112     \r
113     var textNode = $(nodeContents.get(options.textNodeIdx));\r
114 \r
115     var succeedingNodes = [];\r
116     var passed = false;\r
117     nodeContents.each(function() {\r
118         var node = this;\r
119         if(passed)\r
120             succeedingNodes.push(node);\r
121         if(node === textNode.get(0))\r
122             passed = true;\r
123     });\r
124     \r
125     var prefix = $.trim(textNode.text().substr(0, options.offset));\r
126     var suffix = $.trim(textNode.text().substr(options.offset));\r
127     \r
128     textNode.before(prefix);\r
129     textNode.remove();\r
130     \r
131     var newNode = canvasNode.create({tag: nodeToSplit.attr('wlxml-tag'), klass: nodeToSplit.attr('wlxml-class')});\r
132     newNode.dom.append(suffix);\r
133     succeedingNodes.forEach(function(node) {\r
134         newNode.dom.append(node)\r
135     });\r
136     nodeToSplit.after(newNode.dom);\r
137     return newNode;\r
138 };\r
139 \r
140 Canvas.prototype.nodeRemove = function(options) {\r
141     var toRemove = $(this.content.find('#' + options.node.getId()).get(0));\r
142     toRemove.remove();\r
143 };\r
144 \r
145 Canvas.prototype.listCreate = function(options) {\r
146     var element1 = $(this.content.find('#' + options.start.getId()).get(0));\r
147     var element2 = $(this.content.find('#' + options.end.getId()).get(0));\r
148     if(!element1.parent().get(0) === element2.parent().get(0))\r
149         return false;\r
150         \r
151     var parent = element1.parent();\r
152     \r
153     if(parent.contents().index(element1) > parent.contents().index(element2)) {\r
154         var tmp = element1;\r
155         element1 = element2;\r
156         element2 = tmp;\r
157     }\r
158     \r
159     var nodesToWrap = [];\r
160     \r
161     var place = 'before';\r
162     var canvas = this;\r
163     parent.contents().each(function() {\r
164         var node = this;\r
165         if(node === element1.get(0))\r
166             place = 'inside';\r
167         if(place === 'inside') {\r
168             var $node;\r
169             if(node.nodeType === 3) {\r
170                 $node = canvasNode.create({tag: 'div', content: $.trim(node.data)}).dom; //canvas._createNode('div').text(node.data);\r
171                 $(node).remove();\r
172             }\r
173             else {\r
174                 $node = $(node);\r
175             }\r
176             $node.attr('wlxml-class', 'item');\r
177             nodesToWrap.push($node);\r
178         }\r
179         if(node === element2.get(0))\r
180             return false;\r
181     });\r
182     \r
183     var list = canvasNode.create({tag: 'div', klass: 'list-items' + (options.type === 'enum' ? '-enum' : '')}).dom; //this._createNode('div', 'list-items');\r
184     \r
185     var parentNode = options.start.parent();\r
186     \r
187     var toret;\r
188     if(parentNode && parentNode.isOfClass('list-items')) {\r
189         list.wrap('<div wlxml-tag="div" wlxml-class="item" class="canvas-silent-item">');\r
190         toret = list.parent();\r
191     } else {\r
192         toret = list;\r
193     }\r
194         \r
195     \r
196     element1.before(toret);\r
197     \r
198     nodesToWrap.forEach(function(node) {\r
199         node.remove();\r
200         list.append(node);\r
201     });\r
202 };\r
203 \r
204 Canvas.prototype.listRemove = function(options) {\r
205     var pointerElement = $(this.content.find('#' + options.pointer.getId()));\r
206     var listElement = options.pointer.getClass() === 'list-items' ? pointerElement : \r
207         pointerElement.parents('[wlxml-class|="list-items"][wlxml-tag]');\r
208     \r
209     var nested = false;\r
210     if(listElement.length > 1) {\r
211         listElement = $(listElement[0]);\r
212         nested = true;\r
213     }\r
214     \r
215     if(nested) {\r
216         listElement.unwrap();\r
217     } else {\r
218         listElement.find('[wlxml-class=item]').each(function() {\r
219             $(this).removeAttr('wlxml-class');\r
220         });\r
221     }\r
222     listElement.children().unwrap();\r
223 };\r
224 \r
225 Canvas.prototype.getPrecedingNode = function(options) {\r
226     var element = $(this.content.find('#' + options.node.getId()).get(0));\r
227     var prev = element.prev()\r
228     if(prev.length === 0)\r
229         prev = element.parent();\r
230     return canvasNode.create(prev);\r
231 };\r
232 \r
233 Canvas.prototype.nodeInsideList = function(options) {\r
234     if(options.node) {\r
235         if(options.node.isOfClass('list-items') || options.node.isOfClass('item'))\r
236             return true;\r
237         var pointerElement = $(this.content.find('#' + options.node.getId()));\r
238         return pointerElement.parents('[wlxml-class=list-items], [wlxml-class=item]').length > 0;\r
239     }\r
240     return false;\r
241 };\r
242 \r
243 \r
244 return {\r
245     create: function(desc) { return new Canvas(desc); }\r
246 };\r
247 \r
248 });