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