]> git.donarmstrong.com Git - roundcube.git/blob - program/js/tiny_mce/plugins/paste/editor_plugin_src.js
Imported Upstream version 0.2~stable
[roundcube.git] / program / js / tiny_mce / plugins / paste / editor_plugin_src.js
1 /**\r
2  * $Id: editor_plugin_src.js 862 2008-06-02 20:09:06Z 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                                 if (bLinebreaks) { \r
111                                         // Special paragraph treatment \r
112                                         if (this.editor.getParam("paste_create_paragraphs", true)) {\r
113                                                 var rl = this.editor.getParam("paste_replace_list", '\u2122,<sup>TM</sup>,\u2026,...,\u201c|\u201d,",\u2019,\',\u2013|\u2014|\u2015|\u2212,-').split(',');\r
114                                                 for (var i=0; i<rl.length; i+=2)\r
115                                                         content = content.replace(new RegExp(rl[i], 'gi'), rl[i+1]);\r
116 \r
117                                                 content = content.replace(/\r\n\r\n/g, '</p><p>');\r
118                                                 content = content.replace(/\r\r/g, '</p><p>');\r
119                                                 content = content.replace(/\n\n/g, '</p><p>');\r
120 \r
121                                                 // Has paragraphs \r
122                                                 if ((pos = content.indexOf('</p><p>')) != -1) { \r
123                                                         this.editor.execCommand("Delete"); \r
124 \r
125                                                         var node = this.editor.selection.getNode(); \r
126 \r
127                                                         // Get list of elements to break \r
128                                                         var breakElms = [];\r
129 \r
130                                                         do { \r
131                                                                 if (node.nodeType == 1) { \r
132                                                                         // Don't break tables and break at body \r
133                                                                         if (node.nodeName == "TD" || node.nodeName == "BODY") \r
134                                                                                 break; \r
135                         \r
136                                                                         breakElms[breakElms.length] = node; \r
137                                                                 } \r
138                                                         } while(node = node.parentNode); \r
139 \r
140                                                         var before = "", after = "</p>"; \r
141                                                         before += content.substring(0, pos); \r
142 \r
143                                                         for (var i=0; i<breakElms.length; i++) { \r
144                                                                 before += "</" + breakElms[i].nodeName + ">"; \r
145                                                                 after += "<" + breakElms[(breakElms.length-1)-i].nodeName + ">"; \r
146                                                         } \r
147 \r
148                                                         before += "<p>"; \r
149                                                         content = before + content.substring(pos+7) + after; \r
150                                                 } \r
151                                         } \r
152 \r
153                                         if (this.editor.getParam("paste_create_linebreaks", true)) {\r
154                                                 content = content.replace(/\r\n/g, '<br />');\r
155                                                 content = content.replace(/\r/g, '<br />');\r
156                                                 content = content.replace(/\n/g, '<br />');\r
157                                         }\r
158                                 } \r
159 \r
160                                 this.editor.execCommand("mceInsertRawHTML", false, content); \r
161                         }\r
162                 },\r
163 \r
164                 _insertWordContent : function(content) { \r
165                         var t = this, ed = t.editor;\r
166 \r
167                         if (content && content.length > 0) {\r
168                                 // Cleanup Word content\r
169                                 var bull = String.fromCharCode(8226);\r
170                                 var middot = String.fromCharCode(183);\r
171 \r
172                                 if (ed.getParam('paste_insert_word_content_callback'))\r
173                                         content = ed.execCallback('paste_insert_word_content_callback', 'before', content);\r
174 \r
175                                 var rl = ed.getParam("paste_replace_list", '\u2122,<sup>TM</sup>,\u2026,...,\u201c|\u201d,",\u2019,\',\u2013|\u2014|\u2015|\u2212,-').split(',');\r
176                                 for (var i=0; i<rl.length; i+=2)\r
177                                         content = content.replace(new RegExp(rl[i], 'gi'), rl[i+1]);\r
178 \r
179                                 if (this.editor.getParam("paste_convert_headers_to_strong", false)) {\r
180                                         content = content.replace(new RegExp('<p class=MsoHeading.*?>(.*?)<\/p>', 'gi'), '<p><b>$1</b></p>');\r
181                                 }\r
182 \r
183                                 content = content.replace(new RegExp('tab-stops: list [0-9]+.0pt">', 'gi'), '">' + "--list--");\r
184                                 content = content.replace(new RegExp(bull + "(.*?)<BR>", "gi"), "<p>" + middot + "$1</p>");\r
185                                 content = content.replace(new RegExp('<SPAN style="mso-list: Ignore">', 'gi'), "<span>" + bull); // Covert to bull list\r
186                                 content = content.replace(/<o:p><\/o:p>/gi, "");\r
187                                 content = content.replace(new RegExp('<br style="page-break-before: always;.*>', 'gi'), '-- page break --'); // Replace pagebreaks\r
188                                 content = content.replace(new RegExp('<(!--)([^>]*)(--)>', 'g'), "");  // Word comments\r
189 \r
190                                 if (this.editor.getParam("paste_remove_spans", true))\r
191                                         content = content.replace(/<\/?span[^>]*>/gi, "");\r
192 \r
193                                 if (this.editor.getParam("paste_remove_styles", true))\r
194                                         content = content.replace(new RegExp('<(\\w[^>]*) style="([^"]*)"([^>]*)', 'gi'), "<$1$3");\r
195 \r
196                                 content = content.replace(/<\/?font[^>]*>/gi, "");\r
197 \r
198                                 // Strips class attributes.\r
199                                 switch (this.editor.getParam("paste_strip_class_attributes", "all")) {\r
200                                         case "all":\r
201                                                 content = content.replace(/<(\w[^>]*) class=([^ |>]*)([^>]*)/gi, "<$1$3");\r
202                                                 break;\r
203 \r
204                                         case "mso":\r
205                                                 content = content.replace(new RegExp('<(\\w[^>]*) class="?mso([^ |>]*)([^>]*)', 'gi'), "<$1$3");\r
206                                                 break;\r
207                                 }\r
208 \r
209                                 content = content.replace(new RegExp('href="?' + this._reEscape("" + document.location) + '', 'gi'), 'href="' + this.editor.documentBaseURI.getURI());\r
210                                 content = content.replace(/<(\w[^>]*) lang=([^ |>]*)([^>]*)/gi, "<$1$3");\r
211                                 content = content.replace(/<\\?\?xml[^>]*>/gi, "");\r
212                                 content = content.replace(/<\/?\w+:[^>]*>/gi, "");\r
213                                 content = content.replace(/-- page break --\s*<p>&nbsp;<\/p>/gi, ""); // Remove pagebreaks\r
214                                 content = content.replace(/-- page break --/gi, ""); // Remove pagebreaks\r
215 \r
216                 //              content = content.replace(/\/?&nbsp;*/gi, ""); &nbsp;\r
217                 //              content = content.replace(/<p>&nbsp;<\/p>/gi, '');\r
218 \r
219                                 if (!this.editor.getParam('force_p_newlines')) {\r
220                                         content = content.replace('', '' ,'gi');\r
221                                         content = content.replace('</p>', '<br /><br />' ,'gi');\r
222                                 }\r
223 \r
224                                 if (!tinymce.isIE && !this.editor.getParam('force_p_newlines')) {\r
225                                         content = content.replace(/<\/?p[^>]*>/gi, "");\r
226                                 }\r
227 \r
228                                 content = content.replace(/<\/?div[^>]*>/gi, "");\r
229 \r
230                                 // Convert all middlot lists to UL lists\r
231                                 if (this.editor.getParam("paste_convert_middot_lists", true)) {\r
232                                         var div = ed.dom.create("div", null, content);\r
233 \r
234                                         // Convert all middot paragraphs to li elements\r
235                                         var className = this.editor.getParam("paste_unindented_list_class", "unIndentedList");\r
236 \r
237                                         while (this._convertMiddots(div, "--list--")) ; // bull\r
238                                         while (this._convertMiddots(div, middot, className)) ; // Middot\r
239                                         while (this._convertMiddots(div, bull)) ; // bull\r
240 \r
241                                         content = div.innerHTML;\r
242                                 }\r
243 \r
244                                 // Replace all headers with strong and fix some other issues\r
245                                 if (this.editor.getParam("paste_convert_headers_to_strong", false)) {\r
246                                         content = content.replace(/<h[1-6]>&nbsp;<\/h[1-6]>/gi, '<p>&nbsp;&nbsp;</p>');\r
247                                         content = content.replace(/<h[1-6]>/gi, '<p><b>');\r
248                                         content = content.replace(/<\/h[1-6]>/gi, '</b></p>');\r
249                                         content = content.replace(/<b>&nbsp;<\/b>/gi, '<b>&nbsp;&nbsp;</b>');\r
250                                         content = content.replace(/^(&nbsp;)*/gi, '');\r
251                                 }\r
252 \r
253                                 content = content.replace(/--list--/gi, ""); // Remove --list--\r
254 \r
255                                 if (ed.getParam('paste_insert_word_content_callback'))\r
256                                         content = ed.execCallback('paste_insert_word_content_callback', 'after', content);\r
257 \r
258                                 // Insert cleaned content\r
259                                 this.editor.execCommand("mceInsertContent", false, content);\r
260 \r
261                                 if (this.editor.getParam('paste_force_cleanup_wordpaste', true)) {\r
262                                         var ed = this.editor;\r
263 \r
264                                         window.setTimeout(function() {\r
265                                                 ed.execCommand("mceCleanup");\r
266                                         }, 1); // Do normal cleanup detached from this thread\r
267                                 }\r
268                         }\r
269                 },\r
270 \r
271                 _reEscape : function(s) {\r
272                         var l = "?.\\*[](){}+^$:";\r
273                         var o = "";\r
274 \r
275                         for (var i=0; i<s.length; i++) {\r
276                                 var c = s.charAt(i);\r
277 \r
278                                 if (l.indexOf(c) != -1)\r
279                                         o += '\\' + c;\r
280                                 else\r
281                                         o += c;\r
282                         }\r
283 \r
284                         return o;\r
285                 },\r
286 \r
287                 _convertMiddots : function(div, search, class_name) {\r
288                         var ed = this.editor, mdot = String.fromCharCode(183), bull = String.fromCharCode(8226);\r
289                         var nodes, prevul, i, p, ul, li, np, cp, li;\r
290 \r
291                         nodes = div.getElementsByTagName("p");\r
292                         for (i=0; i<nodes.length; i++) {\r
293                                 p = nodes[i];\r
294 \r
295                                 // Is middot\r
296                                 if (p.innerHTML.indexOf(search) == 0) {\r
297                                         ul = ed.dom.create("ul");\r
298 \r
299                                         if (class_name)\r
300                                                 ul.className = class_name;\r
301 \r
302                                         // Add the first one\r
303                                         li = ed.dom.create("li");\r
304                                         li.innerHTML = p.innerHTML.replace(new RegExp('' + mdot + '|' + bull + '|--list--|&nbsp;', "gi"), '');\r
305                                         ul.appendChild(li);\r
306 \r
307                                         // Add the rest\r
308                                         np = p.nextSibling;\r
309                                         while (np) {\r
310                                                 // If the node is whitespace, then\r
311                                                 // ignore it and continue on.\r
312                                                 if (np.nodeType == 3 && new RegExp('^\\s$', 'm').test(np.nodeValue)) {\r
313                                                                 np = np.nextSibling;\r
314                                                                 continue;\r
315                                                 }\r
316 \r
317                                                 if (search == mdot) {\r
318                                                                 if (np.nodeType == 1 && new RegExp('^o(\\s+|&nbsp;)').test(np.innerHTML)) {\r
319                                                                                 // Second level of nesting\r
320                                                                                 if (!prevul) {\r
321                                                                                                 prevul = ul;\r
322                                                                                                 ul = ed.dom.create("ul");\r
323                                                                                                 prevul.appendChild(ul);\r
324                                                                                 }\r
325                                                                                 np.innerHTML = np.innerHTML.replace(/^o/, '');\r
326                                                                 } else {\r
327                                                                                 // Pop the stack if we're going back up to the first level\r
328                                                                                 if (prevul) {\r
329                                                                                                 ul = prevul;\r
330                                                                                                 prevul = null;\r
331                                                                                 }\r
332                                                                                 // Not element or middot paragraph\r
333                                                                                 if (np.nodeType != 1 || np.innerHTML.indexOf(search) != 0)\r
334                                                                                                 break;\r
335                                                                 }\r
336                                                 } else {\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 \r
342                                                 cp = np.nextSibling;\r
343                                                 li = ed.dom.create("li");\r
344                                                 li.innerHTML = np.innerHTML.replace(new RegExp('' + mdot + '|' + bull + '|--list--|&nbsp;', "gi"), '');\r
345                                                 np.parentNode.removeChild(np);\r
346                                                 ul.appendChild(li);\r
347                                                 np = cp;\r
348                                         }\r
349 \r
350                                         p.parentNode.replaceChild(ul, p);\r
351 \r
352                                         return true;\r
353                                 }\r
354                         }\r
355 \r
356                         return false;\r
357                 },\r
358 \r
359                 _clipboardHTML : function() {\r
360                         var div = document.getElementById('_TinyMCE_clipboardHTML');\r
361 \r
362                         if (!div) {\r
363                                 var div = document.createElement('DIV');\r
364                                 div.id = '_TinyMCE_clipboardHTML';\r
365 \r
366                                 with (div.style) {\r
367                                         visibility = 'hidden';\r
368                                         overflow = 'hidden';\r
369                                         position = 'absolute';\r
370                                         width = 1;\r
371                                         height = 1;\r
372                                 }\r
373 \r
374                                 document.body.appendChild(div);\r
375                         }\r
376 \r
377                         div.innerHTML = '';\r
378                         var rng = document.body.createTextRange();\r
379                         rng.moveToElementText(div);\r
380                         rng.execCommand('Paste');\r
381                         var html = div.innerHTML;\r
382                         div.innerHTML = '';\r
383                         return html;\r
384                 }\r
385         });\r
386 \r
387         // Register plugin\r
388         tinymce.PluginManager.add('paste', tinymce.plugins.PastePlugin);\r
389 })();