/static/scripts/tiny_mce/plugins/paste/editor_plugin_src.js

http://n23.googlecode.com/ · JavaScript · 387 lines · 352 code · 22 blank · 13 comment · 24 complexity · 779b276f7e43fc535d1a17ba3224c40b MD5 · raw file

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