PageRenderTime 56ms CodeModel.GetById 24ms RepoModel.GetById 1ms app.codeStats 0ms

/htdocs/plugins/tinymce/jscripts/tiny_mce/plugins/noneditable/editor_plugin_src.js

https://github.com/ivebeenlinuxed/Boiler
JavaScript | 537 lines | 399 code | 81 blank | 57 comment | 144 complexity | 273d35e2221889e75f33fd545f77d44e MD5 | raw file
  1. /**
  2. * editor_plugin_src.js
  3. *
  4. * Copyright 2009, Moxiecode Systems AB
  5. * Released under LGPL License.
  6. *
  7. * License: http://tinymce.moxiecode.com/license
  8. * Contributing: http://tinymce.moxiecode.com/contributing
  9. */
  10. (function() {
  11. var TreeWalker = tinymce.dom.TreeWalker;
  12. var externalName = 'contenteditable', internalName = 'data-mce-' + externalName;
  13. var VK = tinymce.VK;
  14. function handleContentEditableSelection(ed) {
  15. var dom = ed.dom, selection = ed.selection, invisibleChar, caretContainerId = 'mce_noneditablecaret', invisibleChar = '\uFEFF';
  16. // Returns the content editable state of a node "true/false" or null
  17. function getContentEditable(node) {
  18. var contentEditable;
  19. // Ignore non elements
  20. if (node.nodeType === 1) {
  21. // Check for fake content editable
  22. contentEditable = node.getAttribute(internalName);
  23. if (contentEditable && contentEditable !== "inherit") {
  24. return contentEditable;
  25. }
  26. // Check for real content editable
  27. contentEditable = node.contentEditable;
  28. if (contentEditable !== "inherit") {
  29. return contentEditable;
  30. }
  31. }
  32. return null;
  33. };
  34. // Returns the noneditable parent or null if there is a editable before it or if it wasn't found
  35. function getNonEditableParent(node) {
  36. var state;
  37. while (node) {
  38. state = getContentEditable(node);
  39. if (state) {
  40. return state === "false" ? node : null;
  41. }
  42. node = node.parentNode;
  43. }
  44. };
  45. // Get caret container parent for the specified node
  46. function getParentCaretContainer(node) {
  47. while (node) {
  48. if (node.id === caretContainerId) {
  49. return node;
  50. }
  51. node = node.parentNode;
  52. }
  53. };
  54. // Finds the first text node in the specified node
  55. function findFirstTextNode(node) {
  56. var walker;
  57. if (node) {
  58. walker = new TreeWalker(node, node);
  59. for (node = walker.current(); node; node = walker.next()) {
  60. if (node.nodeType === 3) {
  61. return node;
  62. }
  63. }
  64. }
  65. };
  66. // Insert caret container before/after target or expand selection to include block
  67. function insertCaretContainerOrExpandToBlock(target, before) {
  68. var caretContainer, rng;
  69. // Select block
  70. if (getContentEditable(target) === "false") {
  71. if (dom.isBlock(target)) {
  72. selection.select(target);
  73. return;
  74. }
  75. }
  76. rng = dom.createRng();
  77. if (getContentEditable(target) === "true") {
  78. if (!target.firstChild) {
  79. target.appendChild(ed.getDoc().createTextNode('\u00a0'));
  80. }
  81. target = target.firstChild;
  82. before = true;
  83. }
  84. //caretContainer = dom.create('span', {id: caretContainerId, 'data-mce-bogus': true, style:'border: 1px solid red'}, invisibleChar);
  85. caretContainer = dom.create('span', {id: caretContainerId, 'data-mce-bogus': true}, invisibleChar);
  86. if (before) {
  87. target.parentNode.insertBefore(caretContainer, target);
  88. } else {
  89. dom.insertAfter(caretContainer, target);
  90. }
  91. rng.setStart(caretContainer.firstChild, 1);
  92. rng.collapse(true);
  93. selection.setRng(rng);
  94. return caretContainer;
  95. };
  96. // Removes any caret container except the one we might be in
  97. function removeCaretContainer(caretContainer) {
  98. var child, currentCaretContainer, lastContainer;
  99. if (caretContainer) {
  100. rng = selection.getRng(true);
  101. rng.setStartBefore(caretContainer);
  102. rng.setEndBefore(caretContainer);
  103. child = findFirstTextNode(caretContainer);
  104. if (child && child.nodeValue.charAt(0) == invisibleChar) {
  105. child = child.deleteData(0, 1);
  106. }
  107. dom.remove(caretContainer, true);
  108. selection.setRng(rng);
  109. } else {
  110. currentCaretContainer = getParentCaretContainer(selection.getStart());
  111. while ((caretContainer = dom.get(caretContainerId)) && caretContainer !== lastContainer) {
  112. if (currentCaretContainer !== caretContainer) {
  113. child = findFirstTextNode(caretContainer);
  114. if (child && child.nodeValue.charAt(0) == invisibleChar) {
  115. child = child.deleteData(0, 1);
  116. }
  117. dom.remove(caretContainer, true);
  118. }
  119. lastContainer = caretContainer;
  120. }
  121. }
  122. };
  123. // Modifies the selection to include contentEditable false elements or insert caret containers
  124. function moveSelection() {
  125. var nonEditableStart, nonEditableEnd, isCollapsed, rng, element;
  126. // Checks if there is any contents to the left/right side of caret returns the noneditable element or any editable element if it finds one inside
  127. function hasSideContent(element, left) {
  128. var container, offset, walker, node, len;
  129. container = rng.startContainer;
  130. offset = rng.startOffset;
  131. // If endpoint is in middle of text node then expand to beginning/end of element
  132. if (container.nodeType == 3) {
  133. len = container.nodeValue.length;
  134. if ((offset > 0 && offset < len) || (left ? offset == len : offset == 0)) {
  135. return;
  136. }
  137. } else {
  138. // Can we resolve the node by index
  139. if (offset < container.childNodes.length) {
  140. // Browser represents caret position as the offset at the start of an element. When moving right
  141. // this is the element we are moving into so we consider our container to be child node at offset-1
  142. var pos = !left && offset > 0 ? offset-1 : offset;
  143. container = container.childNodes[pos];
  144. if (container.hasChildNodes()) {
  145. container = container.firstChild;
  146. }
  147. } else {
  148. // If not then the caret is at the last position in it's container and the caret container should be inserted after the noneditable element
  149. return !left ? element : null;
  150. }
  151. }
  152. // Walk left/right to look for contents
  153. walker = new TreeWalker(container, element);
  154. while (node = walker[left ? 'prev' : 'next']()) {
  155. if (node.nodeType === 3 && node.nodeValue.length > 0) {
  156. return;
  157. } else if (getContentEditable(node) === "true") {
  158. // Found contentEditable=true element return this one to we can move the caret inside it
  159. return node;
  160. }
  161. }
  162. return element;
  163. };
  164. // Remove any existing caret containers
  165. removeCaretContainer();
  166. // Get noneditable start/end elements
  167. isCollapsed = selection.isCollapsed();
  168. nonEditableStart = getNonEditableParent(selection.getStart());
  169. nonEditableEnd = getNonEditableParent(selection.getEnd());
  170. // Is any fo the range endpoints noneditable
  171. if (nonEditableStart || nonEditableEnd) {
  172. rng = selection.getRng(true);
  173. // If it's a caret selection then look left/right to see if we need to move the caret out side or expand
  174. if (isCollapsed) {
  175. nonEditableStart = nonEditableStart || nonEditableEnd;
  176. var start = selection.getStart();
  177. if (element = hasSideContent(nonEditableStart, true)) {
  178. // We have no contents to the left of the caret then insert a caret container before the noneditable element
  179. insertCaretContainerOrExpandToBlock(element, true);
  180. } else if (element = hasSideContent(nonEditableStart, false)) {
  181. // We have no contents to the right of the caret then insert a caret container after the noneditable element
  182. insertCaretContainerOrExpandToBlock(element, false);
  183. } else {
  184. // We are in the middle of a noneditable so expand to select it
  185. selection.select(nonEditableStart);
  186. }
  187. } else {
  188. rng = selection.getRng(true);
  189. // Expand selection to include start non editable element
  190. if (nonEditableStart) {
  191. rng.setStartBefore(nonEditableStart);
  192. }
  193. // Expand selection to include end non editable element
  194. if (nonEditableEnd) {
  195. rng.setEndAfter(nonEditableEnd);
  196. }
  197. selection.setRng(rng);
  198. }
  199. }
  200. };
  201. function handleKey(ed, e) {
  202. var keyCode = e.keyCode, nonEditableParent, caretContainer, startElement, endElement;
  203. function getNonEmptyTextNodeSibling(node, prev) {
  204. while (node = node[prev ? 'previousSibling' : 'nextSibling']) {
  205. if (node.nodeType !== 3 || node.nodeValue.length > 0) {
  206. return node;
  207. }
  208. }
  209. };
  210. function positionCaretOnElement(element, start) {
  211. selection.select(element);
  212. selection.collapse(start);
  213. }
  214. function canDelete(backspace) {
  215. var rng, container, offset, nonEditableParent;
  216. function removeNodeIfNotParent(node) {
  217. var parent = container;
  218. while (parent) {
  219. if (parent === node) {
  220. return;
  221. }
  222. parent = parent.parentNode;
  223. }
  224. dom.remove(node);
  225. moveSelection();
  226. }
  227. function isNextPrevTreeNodeNonEditable() {
  228. var node, walker, nonEmptyElements = ed.schema.getNonEmptyElements();
  229. walker = new tinymce.dom.TreeWalker(container, ed.getBody());
  230. while (node = (backspace ? walker.prev() : walker.next())) {
  231. // Found IMG/INPUT etc
  232. if (nonEmptyElements[node.nodeName.toLowerCase()]) {
  233. break;
  234. }
  235. // Found text node with contents
  236. if (node.nodeType === 3 && tinymce.trim(node.nodeValue).length > 0) {
  237. break;
  238. }
  239. // Found non editable node
  240. if (getContentEditable(node) === "false") {
  241. removeNodeIfNotParent(node);
  242. return true;
  243. }
  244. }
  245. // Check if the content node is within a non editable parent
  246. if (getNonEditableParent(node)) {
  247. return true;
  248. }
  249. return false;
  250. }
  251. if (selection.isCollapsed()) {
  252. rng = selection.getRng(true);
  253. container = rng.startContainer;
  254. offset = rng.startOffset;
  255. container = getParentCaretContainer(container) || container;
  256. // Is in noneditable parent
  257. if (nonEditableParent = getNonEditableParent(container)) {
  258. removeNodeIfNotParent(nonEditableParent);
  259. return false;
  260. }
  261. // Check if the caret is in the middle of a text node
  262. if (container.nodeType == 3 && (backspace ? offset > 0 : offset < container.nodeValue.length)) {
  263. return true;
  264. }
  265. // Resolve container index
  266. if (container.nodeType == 1) {
  267. container = container.childNodes[offset] || container;
  268. }
  269. // Check if previous or next tree node is non editable then block the event
  270. if (isNextPrevTreeNodeNonEditable()) {
  271. return false;
  272. }
  273. }
  274. return true;
  275. }
  276. startElement = selection.getStart()
  277. endElement = selection.getEnd();
  278. // Disable all key presses in contentEditable=false except delete or backspace
  279. nonEditableParent = getNonEditableParent(startElement) || getNonEditableParent(endElement);
  280. if (nonEditableParent && (keyCode < 112 || keyCode > 124) && keyCode != VK.DELETE && keyCode != VK.BACKSPACE) {
  281. // Is Ctrl+c, Ctrl+v or Ctrl+x then use default browser behavior
  282. if ((tinymce.isMac ? e.metaKey : e.ctrlKey) && (keyCode == 67 || keyCode == 88 || keyCode == 86)) {
  283. return;
  284. }
  285. e.preventDefault();
  286. // Arrow left/right select the element and collapse left/right
  287. if (keyCode == VK.LEFT || keyCode == VK.RIGHT) {
  288. var left = keyCode == VK.LEFT;
  289. // If a block element find previous or next element to position the caret
  290. if (ed.dom.isBlock(nonEditableParent)) {
  291. var targetElement = left ? nonEditableParent.previousSibling : nonEditableParent.nextSibling;
  292. var walker = new TreeWalker(targetElement, targetElement);
  293. var caretElement = left ? walker.prev() : walker.next();
  294. positionCaretOnElement(caretElement, !left);
  295. } else {
  296. positionCaretOnElement(nonEditableParent, left);
  297. }
  298. }
  299. } else {
  300. // Is arrow left/right, backspace or delete
  301. if (keyCode == VK.LEFT || keyCode == VK.RIGHT || keyCode == VK.BACKSPACE || keyCode == VK.DELETE) {
  302. caretContainer = getParentCaretContainer(startElement);
  303. if (caretContainer) {
  304. // Arrow left or backspace
  305. if (keyCode == VK.LEFT || keyCode == VK.BACKSPACE) {
  306. nonEditableParent = getNonEmptyTextNodeSibling(caretContainer, true);
  307. if (nonEditableParent && getContentEditable(nonEditableParent) === "false") {
  308. e.preventDefault();
  309. if (keyCode == VK.LEFT) {
  310. positionCaretOnElement(nonEditableParent, true);
  311. } else {
  312. dom.remove(nonEditableParent);
  313. return;
  314. }
  315. } else {
  316. removeCaretContainer(caretContainer);
  317. }
  318. }
  319. // Arrow right or delete
  320. if (keyCode == VK.RIGHT || keyCode == VK.DELETE) {
  321. nonEditableParent = getNonEmptyTextNodeSibling(caretContainer);
  322. if (nonEditableParent && getContentEditable(nonEditableParent) === "false") {
  323. e.preventDefault();
  324. if (keyCode == VK.RIGHT) {
  325. positionCaretOnElement(nonEditableParent, false);
  326. } else {
  327. dom.remove(nonEditableParent);
  328. return;
  329. }
  330. } else {
  331. removeCaretContainer(caretContainer);
  332. }
  333. }
  334. }
  335. if ((keyCode == VK.BACKSPACE || keyCode == VK.DELETE) && !canDelete(keyCode == VK.BACKSPACE)) {
  336. e.preventDefault();
  337. return false;
  338. }
  339. }
  340. }
  341. };
  342. ed.onMouseDown.addToTop(function(ed, e) {
  343. var node = ed.selection.getNode();
  344. if (getContentEditable(node) === "false" && node == e.target) {
  345. // Expand selection on mouse down we can't block the default event since it's used for drag/drop
  346. moveSelection();
  347. }
  348. });
  349. ed.onMouseUp.addToTop(moveSelection);
  350. ed.onKeyDown.addToTop(handleKey);
  351. ed.onKeyUp.addToTop(moveSelection);
  352. };
  353. tinymce.create('tinymce.plugins.NonEditablePlugin', {
  354. init : function(ed, url) {
  355. var editClass, nonEditClass, nonEditableRegExps;
  356. // Converts configured regexps to noneditable span items
  357. function convertRegExpsToNonEditable(ed, args) {
  358. var i = nonEditableRegExps.length, content = args.content, cls = tinymce.trim(nonEditClass);
  359. // Don't replace the variables when raw is used for example on undo/redo
  360. if (args.format == "raw") {
  361. return;
  362. }
  363. while (i--) {
  364. content = content.replace(nonEditableRegExps[i], function(match) {
  365. var args = arguments, index = args[args.length - 2];
  366. // Is value inside an attribute then don't replace
  367. if (index > 0 && content.charAt(index - 1) == '"') {
  368. return match;
  369. }
  370. return '<span class="' + cls + '" data-mce-content="' + ed.dom.encode(args[0]) + '">' + ed.dom.encode(typeof(args[1]) === "string" ? args[1] : args[0]) + '</span>';
  371. });
  372. }
  373. args.content = content;
  374. };
  375. editClass = " " + tinymce.trim(ed.getParam("noneditable_editable_class", "mceEditable")) + " ";
  376. nonEditClass = " " + tinymce.trim(ed.getParam("noneditable_noneditable_class", "mceNonEditable")) + " ";
  377. // Setup noneditable regexps array
  378. nonEditableRegExps = ed.getParam("noneditable_regexp");
  379. if (nonEditableRegExps && !nonEditableRegExps.length) {
  380. nonEditableRegExps = [nonEditableRegExps];
  381. }
  382. ed.onPreInit.add(function() {
  383. handleContentEditableSelection(ed);
  384. if (nonEditableRegExps) {
  385. ed.selection.onBeforeSetContent.add(convertRegExpsToNonEditable);
  386. ed.onBeforeSetContent.add(convertRegExpsToNonEditable);
  387. }
  388. // Apply contentEditable true/false on elements with the noneditable/editable classes
  389. ed.parser.addAttributeFilter('class', function(nodes) {
  390. var i = nodes.length, className, node;
  391. while (i--) {
  392. node = nodes[i];
  393. className = " " + node.attr("class") + " ";
  394. if (className.indexOf(editClass) !== -1) {
  395. node.attr(internalName, "true");
  396. } else if (className.indexOf(nonEditClass) !== -1) {
  397. node.attr(internalName, "false");
  398. }
  399. }
  400. });
  401. // Remove internal name
  402. ed.serializer.addAttributeFilter(internalName, function(nodes, name) {
  403. var i = nodes.length, node;
  404. while (i--) {
  405. node = nodes[i];
  406. if (nonEditableRegExps && node.attr('data-mce-content')) {
  407. node.name = "#text";
  408. node.type = 3;
  409. node.raw = true;
  410. node.value = node.attr('data-mce-content');
  411. } else {
  412. node.attr(externalName, null);
  413. node.attr(internalName, null);
  414. }
  415. }
  416. });
  417. // Convert external name into internal name
  418. ed.parser.addAttributeFilter(externalName, function(nodes, name) {
  419. var i = nodes.length, node;
  420. while (i--) {
  421. node = nodes[i];
  422. node.attr(internalName, node.attr(externalName));
  423. node.attr(externalName, null);
  424. }
  425. });
  426. });
  427. },
  428. getInfo : function() {
  429. return {
  430. longname : 'Non editable elements',
  431. author : 'Moxiecode Systems AB',
  432. authorurl : 'http://tinymce.moxiecode.com',
  433. infourl : 'http://wiki.moxiecode.com/index.php/TinyMCE:Plugins/noneditable',
  434. version : tinymce.majorVersion + "." + tinymce.minorVersion
  435. };
  436. }
  437. });
  438. // Register plugin
  439. tinymce.PluginManager.add('noneditable', tinymce.plugins.NonEditablePlugin);
  440. })();