Upgrade TinyMCE to 3.2.7. Fixes bug #10993.
This commit is contained in:
parent
dae7215922
commit
a2a2bac440
140 changed files with 8023 additions and 3591 deletions
File diff suppressed because one or more lines are too long
|
|
@ -6,7 +6,30 @@
|
|||
*/
|
||||
|
||||
(function() {
|
||||
var Event = tinymce.dom.Event, grep = tinymce.grep, each = tinymce.each, inArray = tinymce.inArray, isOldWebKit = tinymce.isOldWebKit;
|
||||
var Event = tinymce.dom.Event, grep = tinymce.grep, each = tinymce.each, inArray = tinymce.inArray;
|
||||
|
||||
function isEmpty(d, e, f) {
|
||||
var w, n;
|
||||
|
||||
w = d.createTreeWalker(e, NodeFilter.SHOW_ALL, null, false);
|
||||
while (n = w.nextNode()) {
|
||||
// Filter func
|
||||
if (f) {
|
||||
if (!f(n))
|
||||
return false;
|
||||
}
|
||||
|
||||
// Non whitespace text node
|
||||
if (n.nodeType == 3 && n.nodeValue && /[^\s\u00a0]+/.test(n.nodeValue))
|
||||
return false;
|
||||
|
||||
// Is non text element byt still content
|
||||
if (n.nodeType == 1 && /^(HR|IMG|TABLE)$/.test(n.nodeName))
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
};
|
||||
|
||||
tinymce.create('tinymce.plugins.Safari', {
|
||||
init : function(ed) {
|
||||
|
|
@ -20,21 +43,61 @@
|
|||
t.webKitFontSizes = ['x-small', 'small', 'medium', 'large', 'x-large', 'xx-large', '-webkit-xxx-large'];
|
||||
t.namedFontSizes = ['xx-small', 'x-small','small','medium','large','x-large', 'xx-large'];
|
||||
|
||||
// Safari will crash if the build in createlink command is used
|
||||
/* ed.addCommand('CreateLink', function(u, v) {
|
||||
ed.execCommand("mceInsertContent", false, '<a href="' + dom.encode(v) + '">' + ed.selection.getContent() + '</a>');
|
||||
});*/
|
||||
// Safari CreateLink command will not work correctly on images that is aligned
|
||||
ed.addCommand('CreateLink', function(u, v) {
|
||||
var n = ed.selection.getNode(), dom = ed.dom, a;
|
||||
|
||||
if (n && (/^(left|right)$/i.test(dom.getStyle(n, 'float', 1)) || /^(left|right)$/i.test(dom.getAttrib(n, 'align')))) {
|
||||
a = dom.create('a', {href : v}, n.cloneNode());
|
||||
n.parentNode.replaceChild(a, n);
|
||||
ed.selection.select(a);
|
||||
} else
|
||||
ed.getDoc().execCommand("CreateLink", false, v);
|
||||
});
|
||||
|
||||
/*
|
||||
// WebKit generates spans out of thin air this patch used to remove them but it will also remove styles we want so it's disabled for now
|
||||
ed.onPaste.add(function(ed, e) {
|
||||
function removeStyles(e) {
|
||||
e = e.target;
|
||||
|
||||
if (e.nodeType == 1) {
|
||||
e.style.cssText = '';
|
||||
|
||||
each(ed.dom.select('*', e), function(e) {
|
||||
e.style.cssText = '';
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
Event.add(ed.getDoc(), 'DOMNodeInserted', removeStyles);
|
||||
|
||||
window.setTimeout(function() {
|
||||
Event.remove(ed.getDoc(), 'DOMNodeInserted', removeStyles);
|
||||
}, 0);
|
||||
});
|
||||
*/
|
||||
ed.onKeyUp.add(function(ed, e) {
|
||||
var h;
|
||||
var h, b, r, n, s;
|
||||
|
||||
// If backspace or delete key
|
||||
if (e.keyCode == 46 || e.keyCode == 8) {
|
||||
h = ed.getBody().innerHTML;
|
||||
b = ed.getBody();
|
||||
h = b.innerHTML;
|
||||
s = ed.selection;
|
||||
|
||||
// If there is no text content or images or hr elements then remove everything
|
||||
if (!/<(img|hr)/.test(h) && tinymce.trim(h.replace(/<[^>]+>/g, '')).length == 0)
|
||||
ed.setContent('', {format : 'raw'});
|
||||
if (b.childNodes.length == 1 && !/<(img|hr)/.test(h) && tinymce.trim(h.replace(/<[^>]+>/g, '')).length == 0) {
|
||||
// Inject paragrah and bogus br
|
||||
ed.setContent('<p><br mce_bogus="1" /></p>', {format : 'raw'});
|
||||
|
||||
// Move caret before bogus br
|
||||
n = b.firstChild;
|
||||
r = s.getRng();
|
||||
r.setStart(n, 0);
|
||||
r.setEnd(n, 0);
|
||||
s.setRng(r);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
|
|
@ -51,49 +114,96 @@
|
|||
// Workaround for InsertHTML bug, http://bugs.webkit.org/show_bug.cgi?id=16382
|
||||
ed.addCommand('mceInsertContent', function(u, v) {
|
||||
ed.getDoc().execCommand("InsertText", false, 'mce_marker');
|
||||
ed.getBody().innerHTML = ed.getBody().innerHTML.replace(/mce_marker/g, v + '<span id="_mce_tmp">XX</span>');
|
||||
ed.getBody().innerHTML = ed.getBody().innerHTML.replace(/mce_marker/g, ed.dom.processHTML(v) + '<span id="_mce_tmp">XX</span>');
|
||||
ed.selection.select(ed.dom.get('_mce_tmp'));
|
||||
ed.getDoc().execCommand("Delete", false, ' ');
|
||||
});
|
||||
|
||||
/* ed.onKeyDown.add(function(ed, e) {
|
||||
// Ctrl+A select all will fail on WebKit since if you paste the contents you selected it will produce a odd div wrapper
|
||||
if ((e.ctrlKey || e.metaKey) && e.keyCode == 65) {
|
||||
ed.selection.select(ed.getBody(), 1);
|
||||
return Event.cancel(e);
|
||||
}
|
||||
});*/
|
||||
|
||||
// Workaround for missing shift+enter support, http://bugs.webkit.org/show_bug.cgi?id=16973
|
||||
ed.onKeyPress.add(function(ed, e) {
|
||||
if (e.keyCode == 13 && (e.shiftKey || ed.settings.force_br_newlines && ed.selection.getNode().nodeName != 'LI')) {
|
||||
t._insertBR(ed);
|
||||
Event.cancel(e);
|
||||
var se, li, lic, r1, r2, n, sel, doc, be, af, pa;
|
||||
|
||||
if (e.keyCode == 13) {
|
||||
sel = ed.selection;
|
||||
se = sel.getNode();
|
||||
|
||||
// Workaround for missing shift+enter support, http://bugs.webkit.org/show_bug.cgi?id=16973
|
||||
if (e.shiftKey || ed.settings.force_br_newlines && se.nodeName != 'LI') {
|
||||
t._insertBR(ed);
|
||||
Event.cancel(e);
|
||||
}
|
||||
|
||||
// Workaround for DIV elements produced by Safari
|
||||
if (li = dom.getParent(se, 'LI')) {
|
||||
lic = dom.getParent(li, 'OL,UL');
|
||||
doc = ed.getDoc();
|
||||
|
||||
pa = dom.create('p');
|
||||
dom.add(pa, 'br', {mce_bogus : "1"});
|
||||
|
||||
if (isEmpty(doc, li)) {
|
||||
// If list in list then use browser default behavior
|
||||
if (n = dom.getParent(lic.parentNode, 'LI,OL,UL'))
|
||||
return;
|
||||
|
||||
n = dom.getParent(lic, 'p,h1,h2,h3,h4,h5,h6,div') || lic;
|
||||
|
||||
// Create range from the start of block element to the list item
|
||||
r1 = doc.createRange();
|
||||
r1.setStartBefore(n);
|
||||
r1.setEndBefore(li);
|
||||
|
||||
// Create range after the list to the end of block element
|
||||
r2 = doc.createRange();
|
||||
r2.setStartAfter(li);
|
||||
r2.setEndAfter(n);
|
||||
|
||||
be = r1.cloneContents();
|
||||
af = r2.cloneContents();
|
||||
|
||||
if (!isEmpty(doc, af))
|
||||
dom.insertAfter(af, n);
|
||||
|
||||
dom.insertAfter(pa, n);
|
||||
|
||||
if (!isEmpty(doc, be))
|
||||
dom.insertAfter(be, n);
|
||||
|
||||
dom.remove(n);
|
||||
|
||||
n = pa.firstChild;
|
||||
r1 = doc.createRange();
|
||||
r1.setStartBefore(n);
|
||||
r1.setEndBefore(n);
|
||||
sel.setRng(r1);
|
||||
|
||||
return Event.cancel(e);
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
// Safari returns incorrect values
|
||||
ed.addQueryValueHandler('FontSize', function(u, v) {
|
||||
var e, v;
|
||||
// Safari doesn't place lists outside block elements
|
||||
ed.onExecCommand.add(function(ed, cmd) {
|
||||
var sel, dom, bl, bm;
|
||||
|
||||
// Check for the real font size at the start of selection
|
||||
if ((e = ed.dom.getParent(ed.selection.getStart(), 'span')) && (v = e.style.fontSize))
|
||||
return tinymce.inArray(t.namedFontSizes, v) + 1;
|
||||
if (cmd == 'InsertUnorderedList' || cmd == 'InsertOrderedList') {
|
||||
sel = ed.selection;
|
||||
dom = ed.dom;
|
||||
|
||||
// Check for the real font size at the end of selection
|
||||
if ((e = ed.dom.getParent(ed.selection.getEnd(), 'span')) && (v = e.style.fontSize))
|
||||
return tinymce.inArray(t.namedFontSizes, v) + 1;
|
||||
|
||||
// Return default value it's better than nothing right!
|
||||
return ed.getDoc().queryCommandValue('FontSize');
|
||||
});
|
||||
|
||||
// Safari returns incorrect values
|
||||
ed.addQueryValueHandler('FontName', function(u, v) {
|
||||
var e, v;
|
||||
|
||||
// Check for the real font name at the start of selection
|
||||
if ((e = ed.dom.getParent(ed.selection.getStart(), 'span')) && (v = e.style.fontFamily))
|
||||
return v.replace(/, /g, ',');
|
||||
|
||||
// Check for the real font name at the end of selection
|
||||
if ((e = ed.dom.getParent(ed.selection.getEnd(), 'span')) && (v = e.style.fontFamily))
|
||||
return v.replace(/, /g, ',');
|
||||
|
||||
// Return default value it's better than nothing right!
|
||||
return ed.getDoc().queryCommandValue('FontName');
|
||||
if (bl = dom.getParent(sel.getNode(), function(n) {return /^(H[1-6]|P|ADDRESS|PRE)$/.test(n.nodeName);})) {
|
||||
bm = sel.getBookmark();
|
||||
dom.remove(bl, 1);
|
||||
sel.moveToBookmark(bm);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
// Workaround for bug, http://bugs.webkit.org/show_bug.cgi?id=12250
|
||||
|
|
@ -107,36 +217,8 @@
|
|||
t.selElm = null;
|
||||
});
|
||||
|
||||
ed.onBeforeExecCommand.add(function(ed, c, b) {
|
||||
var r = t.bookmarkRng;
|
||||
|
||||
// Restore selection
|
||||
if (r) {
|
||||
ed.selection.setRng(r);
|
||||
t.bookmarkRng = null;
|
||||
//console.debug('restore', r.startContainer, r.startOffset, r.endContainer, r.endOffset);
|
||||
}
|
||||
});
|
||||
|
||||
ed.onInit.add(function() {
|
||||
t._fixWebKitSpans();
|
||||
|
||||
ed.windowManager.onOpen.add(function() {
|
||||
var r = ed.selection.getRng();
|
||||
|
||||
// Store selection if valid
|
||||
if (r.startContainer != ed.getDoc()) {
|
||||
t.bookmarkRng = r.cloneRange();
|
||||
//console.debug('store', r.startContainer, r.startOffset, r.endContainer, r.endOffset);
|
||||
}
|
||||
});
|
||||
|
||||
ed.windowManager.onClose.add(function() {
|
||||
t.bookmarkRng = null;
|
||||
});
|
||||
|
||||
if (isOldWebKit)
|
||||
t._patchSafari2x(ed);
|
||||
});
|
||||
|
||||
ed.onSetContent.add(function() {
|
||||
|
|
@ -247,27 +329,28 @@
|
|||
});
|
||||
},
|
||||
|
||||
getInfo : function() {
|
||||
return {
|
||||
longname : 'Safari compatibility',
|
||||
author : 'Moxiecode Systems AB',
|
||||
authorurl : 'http://tinymce.moxiecode.com',
|
||||
infourl : 'http://wiki.moxiecode.com/index.php/TinyMCE:Plugins/safari',
|
||||
version : tinymce.majorVersion + "." + tinymce.minorVersion
|
||||
};
|
||||
},
|
||||
|
||||
// Internal methods
|
||||
|
||||
_fixWebKitSpans : function() {
|
||||
var t = this, ed = t.editor;
|
||||
|
||||
if (!isOldWebKit) {
|
||||
// Use mutator events on new WebKit
|
||||
Event.add(ed.getDoc(), 'DOMNodeInserted', function(e) {
|
||||
e = e.target;
|
||||
// Use mutator events on new WebKit
|
||||
Event.add(ed.getDoc(), 'DOMNodeInserted', function(e) {
|
||||
e = e.target;
|
||||
|
||||
if (e && e.nodeType == 1)
|
||||
t._fixAppleSpan(e);
|
||||
});
|
||||
} else {
|
||||
// Do post command processing in old WebKit since the browser crashes on Mutator events :(
|
||||
ed.onExecCommand.add(function() {
|
||||
each(ed.dom.select('span'), function(n) {
|
||||
t._fixAppleSpan(n);
|
||||
});
|
||||
|
||||
ed.nodeChanged();
|
||||
});
|
||||
}
|
||||
if (e && e.nodeType == 1)
|
||||
t._fixAppleSpan(e);
|
||||
});
|
||||
},
|
||||
|
||||
_fixAppleSpan : function(e) {
|
||||
|
|
@ -327,78 +410,6 @@
|
|||
}
|
||||
},
|
||||
|
||||
_patchSafari2x : function(ed) {
|
||||
var t = this, setContent, getNode, dom = ed.dom, lr;
|
||||
|
||||
// Inline dialogs
|
||||
if (ed.windowManager.onBeforeOpen) {
|
||||
ed.windowManager.onBeforeOpen.add(function() {
|
||||
r = ed.selection.getRng();
|
||||
});
|
||||
}
|
||||
|
||||
// Fake select on 2.x
|
||||
ed.selection.select = function(n) {
|
||||
this.getSel().setBaseAndExtent(n, 0, n, 1);
|
||||
};
|
||||
|
||||
getNode = ed.selection.getNode;
|
||||
ed.selection.getNode = function() {
|
||||
return t.selElm || getNode.call(this);
|
||||
};
|
||||
|
||||
// Fake range on Safari 2.x
|
||||
ed.selection.getRng = function() {
|
||||
var t = this, s = t.getSel(), d = ed.getDoc(), r, rb, ra, di;
|
||||
|
||||
// Fake range on Safari 2.x
|
||||
if (s.anchorNode) {
|
||||
r = d.createRange();
|
||||
|
||||
try {
|
||||
// Setup before range
|
||||
rb = d.createRange();
|
||||
rb.setStart(s.anchorNode, s.anchorOffset);
|
||||
rb.collapse(1);
|
||||
|
||||
// Setup after range
|
||||
ra = d.createRange();
|
||||
ra.setStart(s.focusNode, s.focusOffset);
|
||||
ra.collapse(1);
|
||||
|
||||
// Setup start/end points by comparing locations
|
||||
di = rb.compareBoundaryPoints(rb.START_TO_END, ra) < 0;
|
||||
r.setStart(di ? s.anchorNode : s.focusNode, di ? s.anchorOffset : s.focusOffset);
|
||||
r.setEnd(di ? s.focusNode : s.anchorNode, di ? s.focusOffset : s.anchorOffset);
|
||||
|
||||
lr = r;
|
||||
} catch (ex) {
|
||||
// Sometimes fails, at least we tried to do it by the book. I hope Safari 2.x will go disappear soooon!!!
|
||||
}
|
||||
}
|
||||
|
||||
return r || lr;
|
||||
};
|
||||
|
||||
// Fix setContent so it works
|
||||
setContent = ed.selection.setContent;
|
||||
ed.selection.setContent = function(h, s) {
|
||||
var r = this.getRng(), b;
|
||||
|
||||
try {
|
||||
setContent.call(this, h, s);
|
||||
} catch (ex) {
|
||||
// Workaround for Safari 2.x
|
||||
b = dom.create('body');
|
||||
b.innerHTML = h;
|
||||
|
||||
each(b.childNodes, function(n) {
|
||||
r.insertNode(n.cloneNode(true));
|
||||
});
|
||||
}
|
||||
};
|
||||
},
|
||||
|
||||
_insertBR : function(ed) {
|
||||
var dom = ed.dom, s = ed.selection, r = s.getRng(), br;
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue