bb5f5aadbecace671e471eea14e7ce6fb273f27f
[citadel.git] / webcit / tiny_mce / plugins / paste / editor_plugin_src.js
1 /**\r
2  * $Id: editor_plugin_src.js 919 2008-09-08 20:31:23Z spocke $\r
3  *\r
4  * @author Moxiecode\r
5  * @copyright Copyright © 2004-2008, Moxiecode Systems AB, All rights reserved.\r
6  */\r
7 \r
8 (function() {\r
9         var Event = tinymce.dom.Event;\r
10 \r
11         tinymce.create('tinymce.plugins.PastePlugin', {\r
12                 init : function(ed, url) {\r
13                         var t = this;\r
14 \r
15                         t.editor = ed; \r
16 \r
17                         // Register commands\r
18                         ed.addCommand('mcePasteText', function(ui, v) {\r
19                                 if (ui) {\r
20                                         if ((ed.getParam('paste_use_dialog', true)) || (!tinymce.isIE)) {\r
21                                                 ed.windowManager.open({\r
22                                                         file : url + '/pastetext.htm',\r
23                                                         width : 450,\r
24                                                         height : 400,\r
25                                                         inline : 1\r
26                                                 }, {\r
27                                                         plugin_url : url\r
28                                                 });\r
29                                         } else\r
30                                                 t._insertText(clipboardData.getData("Text"), true);\r
31                                 } else\r
32                                         t._insertText(v.html, v.linebreaks);\r
33                         });\r
34 \r
35                         ed.addCommand('mcePasteWord', function(ui, v) {\r
36                                 if (ui) {\r
37                                         if ((ed.getParam('paste_use_dialog', true)) || (!tinymce.isIE)) {\r
38                                                 ed.windowManager.open({\r
39                                                         file : url + '/pasteword.htm',\r
40                                                         width : 450,\r
41                                                         height : 400,\r
42                                                         inline : 1\r
43                                                 }, {\r
44                                                         plugin_url : url\r
45                                                 });\r
46                                         } else\r
47                                                 t._insertText(t._clipboardHTML());\r
48                                 } else\r
49                                         t._insertWordContent(v);\r
50                         });\r
51 \r
52                         ed.addCommand('mceSelectAll', function() {\r
53                                 ed.execCommand('selectall'); \r
54                         });\r
55 \r
56                         // Register buttons\r
57                         ed.addButton('pastetext', {title : 'paste.paste_text_desc', cmd : 'mcePasteText', ui : true});\r
58                         ed.addButton('pasteword', {title : 'paste.paste_word_desc', cmd : 'mcePasteWord', ui : true});\r
59                         ed.addButton('selectall', {title : 'paste.selectall_desc', cmd : 'mceSelectAll'});\r
60 \r
61                         if (ed.getParam("paste_auto_cleanup_on_paste", false)) {\r
62                                 ed.onPaste.add(function(ed, e) {\r
63                                         return t._handlePasteEvent(e)\r
64                                 });\r
65                         }\r
66 \r
67                         if (!tinymce.isIE && ed.getParam("paste_auto_cleanup_on_paste", false)) {\r
68                                 // Force paste dialog if non IE browser\r
69                                 ed.onKeyDown.add(function(ed, e) {\r
70                                         if (e.ctrlKey && e.keyCode == 86) {\r
71                                                 window.setTimeout(function() {\r
72                                                         ed.execCommand("mcePasteText", true);\r
73                                                 }, 1);\r
74 \r
75                                                 Event.cancel(e);\r
76                                         }\r
77                                 });\r
78                         }\r
79                 },\r
80 \r
81                 getInfo : function() {\r
82                         return {\r
83                                 longname : 'Paste text/word',\r
84                                 author : 'Moxiecode Systems AB',\r
85                                 authorurl : 'http://tinymce.moxiecode.com',\r
86                                 infourl : 'http://wiki.moxiecode.com/index.php/TinyMCE:Plugins/paste',\r
87                                 version : tinymce.majorVersion + "." + tinymce.minorVersion\r
88                         };\r
89                 },\r
90 \r
91                 // Private methods\r
92 \r
93                 _handlePasteEvent : function(e) {\r
94                         var html = this._clipboardHTML(), ed = this.editor, sel = ed.selection, r;\r
95 \r
96                         // Removes italic, strong etc, the if was needed due to bug #1437114\r
97                         if (ed && (r = sel.getRng()) && r.text.length > 0)\r
98                                 ed.execCommand('delete');\r
99 \r
100                         if (html && html.length > 0)\r
101                                 ed.execCommand('mcePasteWord', false, html);\r
102 \r
103                         return Event.cancel(e);\r
104                 },\r
105 \r
106                 _insertText : function(content, bLinebreaks) {\r
107                         content = this.editor.dom.encode(content);\r
108 \r
109                         if (content && content.length > 0) {\r
110                                 // Delete any highlighted text before pasting\r
111                                 if (!this.editor.selection.isCollapsed())\r
112                                         this.editor.execCommand("Delete"); \r
113 \r
114                                 if (bLinebreaks) { \r
115                                         // Special paragraph treatment \r
116                                         if (this.editor.getParam("paste_create_paragraphs", true)) {\r
117                                                 var rl = this.editor.getParam("paste_replace_list", '\u2122,<sup>TM</sup>,\u2026,...,\u201c|\u201d,",\u2019,\',\u2013|\u2014|\u2015|\u2212,-').split(',');\r
118                                                 for (var i=0; i<rl.length; i+=2)\r
119                                                         content = content.replace(new RegExp(rl[i], 'gi'), rl[i+1]);\r
120 \r
121                                                 content = content.replace(/\r\n\r\n/g, '</p><p>');\r
122                                                 content = content.replace(/\r\r/g, '</p><p>');\r
123                                                 content = content.replace(/\n\n/g, '</p><p>');\r
124 \r
125                                                 // Has paragraphs \r
126                                                 if ((pos = content.indexOf('</p><p>')) != -1) { \r
127                                                         this.editor.execCommand("Delete"); \r
128 \r
129                                                         var node = this.editor.selection.getNode(); \r
130 \r
131                                                         // Get list of elements to break \r
132                                                         var breakElms = [];\r
133 \r
134                                                         do { \r
135                                                                 if (node.nodeType == 1) { \r
136                                                                         // Don't break tables and break at body \r
137                                                                         if (node.nodeName == "TD" || node.nodeName == "BODY") \r
138                                                                                 break; \r
139                         \r
140                                                                         breakElms[breakElms.length] = node; \r
141                                                                 } \r
142                                                         } while(node = node.parentNode); \r
143 \r
144                                                         var before = "", after = "</p>"; \r
145                                                         before += content.substring(0, pos); \r
146 \r
147                                                         for (var i=0; i<breakElms.length; i++) { \r
148                                                                 before += "</" + breakElms[i].nodeName + ">"; \r
149                                                                 after += "<" + breakElms[(breakElms.length-1)-i].nodeName + ">"; \r
150                                                         } \r
151 \r
152                                                         before += "<p>"; \r
153                                                         content = before + content.substring(pos+7) + after; \r
154                                                 } \r
155                                         } \r
156 \r
157                                         if (this.editor.getParam("paste_create_linebreaks", true)) {\r
158                                                 content = content.replace(/\r\n/g, '<br />');\r
159                                                 content = content.replace(/\r/g, '<br />');\r
160                                                 content = content.replace(/\n/g, '<br />');\r
161                                         }\r
162                                 } \r
163 \r
164                                 this.editor.execCommand("mceInsertRawHTML", false, content); \r
165                         }\r
166                 },\r
167 \r
168                 _insertWordContent : function(content) { \r
169                         var t = this, ed = t.editor;\r
170 \r
171                         if (content && content.length > 0) {\r
172                                 // Cleanup Word content\r
173                                 var bull = String.fromCharCode(8226);\r
174                                 var middot = String.fromCharCode(183);\r
175 \r
176                                 if (ed.getParam('paste_insert_word_content_callback'))\r
177                                         content = ed.execCallback('paste_insert_word_content_callback', 'before', content);\r
178 \r
179                                 var rl = ed.getParam("paste_replace_list", '\u2122,<sup>TM</sup>,\u2026,...,\x93|\x94|\u201c|\u201d,",\x60|\x91|\x92|\u2018|\u2019,\',\u2013|\u2014|\u2015|\u2212,-').split(',');\r
180                                 for (var i=0; i<rl.length; i+=2)\r
181                                         content = content.replace(new RegExp(rl[i], 'gi'), rl[i+1]);\r
182 \r
183                                 if (this.editor.getParam("paste_convert_headers_to_strong", false)) {\r
184                                         content = content.replace(new RegExp('<p class=MsoHeading.*?>(.*?)<\/p>', 'gi'), '<p><b>$1</b></p>');\r
185                                 }\r
186 \r
187                                 content = content.replace(new RegExp('tab-stops: list [0-9]+.0pt">', 'gi'), '">' + "--list--");\r
188                                 content = content.replace(new RegExp(bull + "(.*?)<BR>", "gi"), "<p>" + middot + "$1</p>");\r
189                                 content = content.replace(new RegExp('<SPAN style="mso-list: Ignore">', 'gi'), "<span>" + bull); // Covert to bull list\r
190                                 content = content.replace(/<o:p><\/o:p>/gi, "");\r
191                                 content = content.replace(new RegExp('<br style="page-break-before: always;.*>', 'gi'), '-- page break --'); // Replace pagebreaks\r
192                                 content = content.replace(/<!--([\s\S]*?)-->|<style>[\s\S]*?<\/style>/g, "");  // Word comments\r
193                                 content = content.replace(/<(meta|link)[^>]+>/g, ""); // Header elements\r
194 \r
195                                 if (this.editor.getParam("paste_remove_spans", true))\r
196                                         content = content.replace(/<\/?span[^>]*>/gi, "");\r
197 \r
198                                 if (this.editor.getParam("paste_remove_styles", true))\r
199                                         content = content.replace(new RegExp('<(\\w[^>]*) style="([^"]*)"([^>]*)', 'gi'), "<$1$3");\r
200 \r
201                                 content = content.replace(/<\/?font[^>]*>/gi, "");\r
202 \r
203                                 // Strips class attributes.\r
204                                 switch (this.editor.getParam("paste_strip_class_attributes", "all")) {\r
205                                         case "all":\r
206                                                 content = content.replace(/<(\w[^>]*) class=([^ |>]*)([^>]*)/gi, "<$1$3");\r
207                                                 break;\r
208 \r
209                                         case "mso":\r
210                                                 content = content.replace(new RegExp('<(\\w[^>]*) class="?mso([^ |>]*)([^>]*)', 'gi'), "<$1$3");\r
211                                                 break;\r
212                                 }\r
213 \r
214                                 content = content.replace(new RegExp('href="?' + this._reEscape("" + document.location) + '', 'gi'), 'href="' + this.editor.documentBaseURI.getURI());\r
215                                 content = content.replace(/<(\w[^>]*) lang=([^ |>]*)([^>]*)/gi, "<$1$3");\r
216                                 content = content.replace(/<\\?\?xml[^>]*>/gi, "");\r
217                                 content = content.replace(/<\/?\w+:[^>]*>/gi, "");\r
218                                 content = content.replace(/-- page break --\s*<p>&nbsp;<\/p>/gi, ""); // Remove pagebreaks\r
219                                 content = content.replace(/-- page break --/gi, ""); // Remove pagebreaks\r
220 \r
221                 //              content = content.replace(/\/?&nbsp;*/gi, ""); &nbsp;\r
222                 //              content = content.replace(/<p>&nbsp;<\/p>/gi, '');\r
223 \r
224                                 if (!this.editor.getParam('force_p_newlines')) {\r
225                                         content = content.replace('', '' ,'gi');\r
226                                         content = content.replace('</p>', '<br /><br />' ,'gi');\r
227                                 }\r
228 \r
229                                 if (!tinymce.isIE && !this.editor.getParam('force_p_newlines')) {\r
230                                         content = content.replace(/<\/?p[^>]*>/gi, "");\r
231                                 }\r
232 \r
233                                 content = content.replace(/<\/?div[^>]*>/gi, "");\r
234 \r
235                                 // Convert all middlot lists to UL lists\r
236                                 if (this.editor.getParam("paste_convert_middot_lists", true)) {\r
237                                         var div = ed.dom.create("div", null, content);\r
238 \r
239                                         // Convert all middot paragraphs to li elements\r
240                                         var className = this.editor.getParam("paste_unindented_list_class", "unIndentedList");\r
241 \r
242                                         while (this._convertMiddots(div, "--list--")) ; // bull\r
243                                         while (this._convertMiddots(div, middot, className)) ; // Middot\r
244                                         while (this._convertMiddots(div, bull)) ; // bull\r
245 \r
246                                         content = div.innerHTML;\r
247                                 }\r
248 \r
249                                 // Replace all headers with strong and fix some other issues\r
250                                 if (this.editor.getParam("paste_convert_headers_to_strong", false)) {\r
251                                         content = content.replace(/<h[1-6]>&nbsp;<\/h[1-6]>/gi, '<p>&nbsp;&nbsp;</p>');\r
252                                         content = content.replace(/<h[1-6]>/gi, '<p><b>');\r
253                                         content = content.replace(/<\/h[1-6]>/gi, '</b></p>');\r
254                                         content = content.replace(/<b>&nbsp;<\/b>/gi, '<b>&nbsp;&nbsp;</b>');\r
255                                         content = content.replace(/^(&nbsp;)*/gi, '');\r
256                                 }\r
257 \r
258                                 content = content.replace(/--list--/gi, ""); // Remove --list--\r
259 \r
260                                 if (ed.getParam('paste_insert_word_content_callback'))\r
261                                         content = ed.execCallback('paste_insert_word_content_callback', 'after', content);\r
262 \r
263                                 // Insert cleaned content\r
264                                 this.editor.execCommand("mceInsertContent", false, content);\r
265 \r
266                                 if (this.editor.getParam('paste_force_cleanup_wordpaste', true)) {\r
267                                         var ed = this.editor;\r
268 \r
269                                         window.setTimeout(function() {\r
270                                                 ed.execCommand("mceCleanup");\r
271                                         }, 1); // Do normal cleanup detached from this thread\r
272                                 }\r
273                         }\r
274                 },\r
275 \r
276                 _reEscape : function(s) {\r
277                         var l = "?.\\*[](){}+^$:";\r
278                         var o = "";\r
279 \r
280                         for (var i=0; i<s.length; i++) {\r
281                                 var c = s.charAt(i);\r
282 \r
283                                 if (l.indexOf(c) != -1)\r
284                                         o += '\\' + c;\r
285                                 else\r
286                                         o += c;\r
287                         }\r
288 \r
289                         return o;\r
290                 },\r
291 \r
292                 _convertMiddots : function(div, search, class_name) {\r
293                         var ed = this.editor, mdot = String.fromCharCode(183), bull = String.fromCharCode(8226);\r
294                         var nodes, prevul, i, p, ul, li, np, cp, li;\r
295 \r
296                         nodes = div.getElementsByTagName("p");\r
297                         for (i=0; i<nodes.length; i++) {\r
298                                 p = nodes[i];\r
299 \r
300                                 // Is middot\r
301                                 if (p.innerHTML.indexOf(search) == 0) {\r
302                                         ul = ed.dom.create("ul");\r
303 \r
304                                         if (class_name)\r
305                                                 ul.className = class_name;\r
306 \r
307                                         // Add the first one\r
308                                         li = ed.dom.create("li");\r
309                                         li.innerHTML = p.innerHTML.replace(new RegExp('' + mdot + '|' + bull + '|--list--|&nbsp;', "gi"), '');\r
310                                         ul.appendChild(li);\r
311 \r
312                                         // Add the rest\r
313                                         np = p.nextSibling;\r
314                                         while (np) {\r
315                                                 // If the node is whitespace, then\r
316                                                 // ignore it and continue on.\r
317                                                 if (np.nodeType == 3 && new RegExp('^\\s$', 'm').test(np.nodeValue)) {\r
318                                                                 np = np.nextSibling;\r
319                                                                 continue;\r
320                                                 }\r
321 \r
322                                                 if (search == mdot) {\r
323                                                                 if (np.nodeType == 1 && new RegExp('^o(\\s+|&nbsp;)').test(np.innerHTML)) {\r
324                                                                                 // Second level of nesting\r
325                                                                                 if (!prevul) {\r
326                                                                                                 prevul = ul;\r
327                                                                                                 ul = ed.dom.create("ul");\r
328                                                                                                 prevul.appendChild(ul);\r
329                                                                                 }\r
330                                                                                 np.innerHTML = np.innerHTML.replace(/^o/, '');\r
331                                                                 } else {\r
332                                                                                 // Pop the stack if we're going back up to the first level\r
333                                                                                 if (prevul) {\r
334                                                                                                 ul = prevul;\r
335                                                                                                 prevul = null;\r
336                                                                                 }\r
337                                                                                 // Not element or middot paragraph\r
338                                                                                 if (np.nodeType != 1 || np.innerHTML.indexOf(search) != 0)\r
339                                                                                                 break;\r
340                                                                 }\r
341                                                 } else {\r
342                                                                 // Not element or middot paragraph\r
343                                                                 if (np.nodeType != 1 || np.innerHTML.indexOf(search) != 0)\r
344                                                                                 break;\r
345                                                         }\r
346 \r
347                                                 cp = np.nextSibling;\r
348                                                 li = ed.dom.create("li");\r
349                                                 li.innerHTML = np.innerHTML.replace(new RegExp('' + mdot + '|' + bull + '|--list--|&nbsp;', "gi"), '');\r
350                                                 np.parentNode.removeChild(np);\r
351                                                 ul.appendChild(li);\r
352                                                 np = cp;\r
353                                         }\r
354 \r
355                                         p.parentNode.replaceChild(ul, p);\r
356 \r
357                                         return true;\r
358                                 }\r
359                         }\r
360 \r
361                         return false;\r
362                 },\r
363 \r
364                 _clipboardHTML : function() {\r
365                         var div = document.getElementById('_TinyMCE_clipboardHTML');\r
366 \r
367                         if (!div) {\r
368                                 var div = document.createElement('DIV');\r
369                                 div.id = '_TinyMCE_clipboardHTML';\r
370 \r
371                                 with (div.style) {\r
372                                         visibility = 'hidden';\r
373                                         overflow = 'hidden';\r
374                                         position = 'absolute';\r
375                                         width = 1;\r
376                                         height = 1;\r
377                                 }\r
378 \r
379                                 document.body.appendChild(div);\r
380                         }\r
381 \r
382                         div.innerHTML = '';\r
383                         var rng = document.body.createTextRange();\r
384                         rng.moveToElementText(div);\r
385                         rng.execCommand('Paste');\r
386                         var html = div.innerHTML;\r
387                         div.innerHTML = '';\r
388                         return html;\r
389                 }\r
390         });\r
391 \r
392         // Register plugin\r
393         tinymce.PluginManager.add('paste', tinymce.plugins.PastePlugin);\r
394 })();