4
* Copyright 2009, Moxiecode Systems AB
5
* Released under LGPL License.
7
* License: http://tinymce.moxiecode.com/license
8
* Contributing: http://tinymce.moxiecode.com/contributing
12
var TreeWalker = tinymce.dom.TreeWalker;
13
var externalName = 'contenteditable', internalName = 'data-mce-' + externalName;
16
function handleContentEditableSelection(ed) {
17
var dom = ed.dom, selection = ed.selection, invisibleChar, caretContainerId = 'mce_noneditablecaret', invisibleChar = '\uFEFF';
19
// Returns the content editable state of a node "true/false" or null
20
function getContentEditable(node) {
23
// Ignore non elements
24
if (node.nodeType === 1) {
25
// Check for fake content editable
26
contentEditable = node.getAttribute(internalName);
27
if (contentEditable && contentEditable !== "inherit") {
28
return contentEditable;
31
// Check for real content editable
32
contentEditable = node.contentEditable;
33
if (contentEditable !== "inherit") {
34
return contentEditable;
41
// Returns the noneditable parent or null if there is a editable before it or if it wasn't found
42
function getNonEditableParent(node) {
46
state = getContentEditable(node);
48
return state === "false" ? node : null;
51
node = node.parentNode;
55
// Get caret container parent for the specified node
56
function getParentCaretContainer(node) {
58
if (node.id === caretContainerId) {
62
node = node.parentNode;
66
// Finds the first text node in the specified node
67
function findFirstTextNode(node) {
71
walker = new TreeWalker(node, node);
73
for (node = walker.current(); node; node = walker.next()) {
74
if (node.nodeType === 3) {
81
// Insert caret container before/after target or expand selection to include block
82
function insertCaretContainerOrExpandToBlock(target, before) {
83
var caretContainer, rng;
86
if (getContentEditable(target) === "false") {
87
if (dom.isBlock(target)) {
88
selection.select(target);
93
rng = dom.createRng();
95
if (getContentEditable(target) === "true") {
96
if (!target.firstChild) {
97
target.appendChild(ed.getDoc().createTextNode('\u00a0'));
100
target = target.firstChild;
104
//caretContainer = dom.create('span', {id: caretContainerId, 'data-mce-bogus': true, style:'border: 1px solid red'}, invisibleChar);
105
caretContainer = dom.create('span', {id: caretContainerId, 'data-mce-bogus': true}, invisibleChar);
108
target.parentNode.insertBefore(caretContainer, target);
110
dom.insertAfter(caretContainer, target);
113
rng.setStart(caretContainer.firstChild, 1);
115
selection.setRng(rng);
117
return caretContainer;
120
// Removes any caret container except the one we might be in
121
function removeCaretContainer(caretContainer) {
122
var child, currentCaretContainer, lastContainer;
124
if (caretContainer) {
125
rng = selection.getRng(true);
126
rng.setStartBefore(caretContainer);
127
rng.setEndBefore(caretContainer);
129
child = findFirstTextNode(caretContainer);
130
if (child && child.nodeValue.charAt(0) == invisibleChar) {
131
child = child.deleteData(0, 1);
134
dom.remove(caretContainer, true);
136
selection.setRng(rng);
138
currentCaretContainer = getParentCaretContainer(selection.getStart());
139
while ((caretContainer = dom.get(caretContainerId)) && caretContainer !== lastContainer) {
140
if (currentCaretContainer !== caretContainer) {
141
child = findFirstTextNode(caretContainer);
142
if (child && child.nodeValue.charAt(0) == invisibleChar) {
143
child = child.deleteData(0, 1);
146
dom.remove(caretContainer, true);
149
lastContainer = caretContainer;
154
// Modifies the selection to include contentEditable false elements or insert caret containers
155
function moveSelection() {
156
var nonEditableStart, nonEditableEnd, isCollapsed, rng, element;
158
// 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
159
function hasSideContent(element, left) {
160
var container, offset, walker, node, len;
162
container = rng.startContainer;
163
offset = rng.startOffset;
165
// If endpoint is in middle of text node then expand to beginning/end of element
166
if (container.nodeType == 3) {
167
len = container.nodeValue.length;
168
if ((offset > 0 && offset < len) || (left ? offset == len : offset == 0)) {
172
// Can we resolve the node by index
173
if (offset < container.childNodes.length) {
174
// Browser represents caret position as the offset at the start of an element. When moving right
175
// this is the element we are moving into so we consider our container to be child node at offset-1
176
var pos = !left && offset > 0 ? offset-1 : offset;
177
container = container.childNodes[pos];
178
if (container.hasChildNodes()) {
179
container = container.firstChild;
182
// 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
183
return !left ? element : null;
187
// Walk left/right to look for contents
188
walker = new TreeWalker(container, element);
189
while (node = walker[left ? 'prev' : 'next']()) {
190
if (node.nodeType === 3 && node.nodeValue.length > 0) {
192
} else if (getContentEditable(node) === "true") {
193
// Found contentEditable=true element return this one to we can move the caret inside it
201
// Remove any existing caret containers
202
removeCaretContainer();
204
// Get noneditable start/end elements
205
isCollapsed = selection.isCollapsed();
206
nonEditableStart = getNonEditableParent(selection.getStart());
207
nonEditableEnd = getNonEditableParent(selection.getEnd());
209
// Is any fo the range endpoints noneditable
210
if (nonEditableStart || nonEditableEnd) {
211
rng = selection.getRng(true);
213
// If it's a caret selection then look left/right to see if we need to move the caret out side or expand
215
nonEditableStart = nonEditableStart || nonEditableEnd;
216
var start = selection.getStart();
217
if (element = hasSideContent(nonEditableStart, true)) {
218
// We have no contents to the left of the caret then insert a caret container before the noneditable element
219
insertCaretContainerOrExpandToBlock(element, true);
220
} else if (element = hasSideContent(nonEditableStart, false)) {
221
// We have no contents to the right of the caret then insert a caret container after the noneditable element
222
insertCaretContainerOrExpandToBlock(element, false);
224
// We are in the middle of a noneditable so expand to select it
225
selection.select(nonEditableStart);
228
rng = selection.getRng(true);
230
// Expand selection to include start non editable element
231
if (nonEditableStart) {
232
rng.setStartBefore(nonEditableStart);
235
// Expand selection to include end non editable element
236
if (nonEditableEnd) {
237
rng.setEndAfter(nonEditableEnd);
240
selection.setRng(rng);
245
function handleKey(ed, e) {
246
var keyCode = e.keyCode, nonEditableParent, caretContainer, startElement, endElement;
248
function getNonEmptyTextNodeSibling(node, prev) {
249
while (node = node[prev ? 'previousSibling' : 'nextSibling']) {
250
if (node.nodeType !== 3 || node.nodeValue.length > 0) {
256
function positionCaretOnElement(element, start) {
257
selection.select(element);
258
selection.collapse(start);
261
function canDelete(backspace) {
262
var rng, container, offset, nonEditableParent;
264
function removeNodeIfNotParent(node) {
265
var parent = container;
268
if (parent === node) {
272
parent = parent.parentNode;
279
function isNextPrevTreeNodeNonEditable() {
280
var node, walker, nonEmptyElements = ed.schema.getNonEmptyElements();
282
walker = new tinymce.dom.TreeWalker(container, ed.getBody());
283
while (node = (backspace ? walker.prev() : walker.next())) {
284
// Found IMG/INPUT etc
285
if (nonEmptyElements[node.nodeName.toLowerCase()]) {
289
// Found text node with contents
290
if (node.nodeType === 3 && tinymce.trim(node.nodeValue).length > 0) {
294
// Found non editable node
295
if (getContentEditable(node) === "false") {
296
removeNodeIfNotParent(node);
301
// Check if the content node is within a non editable parent
302
if (getNonEditableParent(node)) {
309
if (selection.isCollapsed()) {
310
rng = selection.getRng(true);
311
container = rng.startContainer;
312
offset = rng.startOffset;
313
container = getParentCaretContainer(container) || container;
315
// Is in noneditable parent
316
if (nonEditableParent = getNonEditableParent(container)) {
317
removeNodeIfNotParent(nonEditableParent);
321
// Check if the caret is in the middle of a text node
322
if (container.nodeType == 3 && (backspace ? offset > 0 : offset < container.nodeValue.length)) {
326
// Resolve container index
327
if (container.nodeType == 1) {
328
container = container.childNodes[offset] || container;
331
// Check if previous or next tree node is non editable then block the event
332
if (isNextPrevTreeNodeNonEditable()) {
340
startElement = selection.getStart()
341
endElement = selection.getEnd();
343
// Disable all key presses in contentEditable=false except delete or backspace
344
nonEditableParent = getNonEditableParent(startElement) || getNonEditableParent(endElement);
345
if (nonEditableParent && (keyCode < 112 || keyCode > 124) && keyCode != VK.DELETE && keyCode != VK.BACKSPACE) {
346
// Is Ctrl+c, Ctrl+v or Ctrl+x then use default browser behavior
347
if ((tinymce.isMac ? e.metaKey : e.ctrlKey) && (keyCode == 67 || keyCode == 88 || keyCode == 86)) {
353
// Arrow left/right select the element and collapse left/right
354
if (keyCode == VK.LEFT || keyCode == VK.RIGHT) {
355
var left = keyCode == VK.LEFT;
356
// If a block element find previous or next element to position the caret
357
if (ed.dom.isBlock(nonEditableParent)) {
358
var targetElement = left ? nonEditableParent.previousSibling : nonEditableParent.nextSibling;
359
var walker = new TreeWalker(targetElement, targetElement);
360
var caretElement = left ? walker.prev() : walker.next();
361
positionCaretOnElement(caretElement, !left);
363
positionCaretOnElement(nonEditableParent, left);
367
// Is arrow left/right, backspace or delete
368
if (keyCode == VK.LEFT || keyCode == VK.RIGHT || keyCode == VK.BACKSPACE || keyCode == VK.DELETE) {
369
caretContainer = getParentCaretContainer(startElement);
370
if (caretContainer) {
371
// Arrow left or backspace
372
if (keyCode == VK.LEFT || keyCode == VK.BACKSPACE) {
373
nonEditableParent = getNonEmptyTextNodeSibling(caretContainer, true);
375
if (nonEditableParent && getContentEditable(nonEditableParent) === "false") {
378
if (keyCode == VK.LEFT) {
379
positionCaretOnElement(nonEditableParent, true);
381
dom.remove(nonEditableParent);
385
removeCaretContainer(caretContainer);
389
// Arrow right or delete
390
if (keyCode == VK.RIGHT || keyCode == VK.DELETE) {
391
nonEditableParent = getNonEmptyTextNodeSibling(caretContainer);
393
if (nonEditableParent && getContentEditable(nonEditableParent) === "false") {
396
if (keyCode == VK.RIGHT) {
397
positionCaretOnElement(nonEditableParent, false);
399
dom.remove(nonEditableParent);
403
removeCaretContainer(caretContainer);
408
if ((keyCode == VK.BACKSPACE || keyCode == VK.DELETE) && !canDelete(keyCode == VK.BACKSPACE)) {
416
ed.onMouseDown.addToTop(function(ed, e) {
417
var node = ed.selection.getNode();
419
if (getContentEditable(node) === "false" && node == e.target) {
420
// Expand selection on mouse down we can't block the default event since it's used for drag/drop
425
ed.onMouseUp.addToTop(moveSelection);
426
ed.onKeyDown.addToTop(handleKey);
427
ed.onKeyUp.addToTop(moveSelection);
430
tinymce.create('tinymce.plugins.NonEditablePlugin', {
431
init : function(ed, url) {
432
var editClass, nonEditClass, nonEditableRegExps;
434
// Converts configured regexps to noneditable span items
435
function convertRegExpsToNonEditable(ed, args) {
436
var i = nonEditableRegExps.length, content = args.content, cls = tinymce.trim(nonEditClass);
438
// Don't replace the variables when raw is used for example on undo/redo
439
if (args.format == "raw") {
444
content = content.replace(nonEditableRegExps[i], function(match) {
445
var args = arguments, index = args[args.length - 2];
447
// Is value inside an attribute then don't replace
448
if (index > 0 && content.charAt(index - 1) == '"') {
452
return '<span class="' + cls + '" data-mce-content="' + ed.dom.encode(args[0]) + '">' + ed.dom.encode(typeof(args[1]) === "string" ? args[1] : args[0]) + '</span>';
456
args.content = content;
459
editClass = " " + tinymce.trim(ed.getParam("noneditable_editable_class", "mceEditable")) + " ";
460
nonEditClass = " " + tinymce.trim(ed.getParam("noneditable_noneditable_class", "mceNonEditable")) + " ";
462
// Setup noneditable regexps array
463
nonEditableRegExps = ed.getParam("noneditable_regexp");
464
if (nonEditableRegExps && !nonEditableRegExps.length) {
465
nonEditableRegExps = [nonEditableRegExps];
468
ed.onPreInit.add(function() {
469
handleContentEditableSelection(ed);
471
if (nonEditableRegExps) {
472
ed.selection.onBeforeSetContent.add(convertRegExpsToNonEditable);
473
ed.onBeforeSetContent.add(convertRegExpsToNonEditable);
476
// Apply contentEditable true/false on elements with the noneditable/editable classes
477
ed.parser.addAttributeFilter('class', function(nodes) {
478
var i = nodes.length, className, node;
482
className = " " + node.attr("class") + " ";
484
if (className.indexOf(editClass) !== -1) {
485
node.attr(internalName, "true");
486
} else if (className.indexOf(nonEditClass) !== -1) {
487
node.attr(internalName, "false");
492
// Remove internal name
493
ed.serializer.addAttributeFilter(internalName, function(nodes, name) {
494
var i = nodes.length, node;
499
if (nonEditableRegExps && node.attr('data-mce-content')) {
503
node.value = node.attr('data-mce-content');
505
node.attr(externalName, null);
506
node.attr(internalName, null);
511
// Convert external name into internal name
512
ed.parser.addAttributeFilter(externalName, function(nodes, name) {
513
var i = nodes.length, node;
517
node.attr(internalName, node.attr(externalName));
518
node.attr(externalName, null);
524
getInfo : function() {
526
longname : 'Non editable elements',
527
author : 'Moxiecode Systems AB',
528
authorurl : 'http://tinymce.moxiecode.com',
529
infourl : 'http://wiki.moxiecode.com/index.php/TinyMCE:Plugins/noneditable',
530
version : tinymce.majorVersion + "." + tinymce.minorVersion
536
tinymce.PluginManager.add('noneditable', tinymce.plugins.NonEditablePlugin);
b'\\ No newline at end of file'