Update TinyMCE to 4.5.2

This commit is contained in:
Dan Stillman 2017-02-02 00:23:44 -05:00
parent 0b9b758029
commit 0998c1d1da
5 changed files with 359 additions and 75 deletions

View File

@ -58,23 +58,27 @@ tinymce.PluginManager.add('link', function(editor) {
return false; return false;
} }
function openDetachedWindow(url) { function appendClickRemove(link, evt) {
// Added by Zotero document.body.appendChild(link);
editor.execCommand('ZoteroLinkClick', false, url); link.dispatchEvent(evt);
document.body.removeChild(link);
}
function openDetachedWindow(url) { /* Added by Zotero */ editor.execCommand("ZoteroLinkClick", false, url); return;
// Chrome and Webkit has implemented noopener and works correctly with/without popup blocker // Chrome and Webkit has implemented noopener and works correctly with/without popup blocker
// Firefox has it implemented noopener but when the popup blocker is activated it doesn't work // Firefox has it implemented noopener but when the popup blocker is activated it doesn't work
// Edge has only implemented noreferrer and it seems to remove opener as well // Edge has only implemented noreferrer and it seems to remove opener as well
// Older IE versions pre IE 11 falls back to a window.open approach // Older IE versions pre IE 11 falls back to a window.open approach
/*if (!tinymce.Env.ie || tinymce.Env.ie > 10) { if (!tinymce.Env.ie || tinymce.Env.ie > 10) {
var link = document.createElement('a'); var link = document.createElement('a');
link.target = '_blank'; link.target = '_blank';
link.href = url; link.href = url;
link.rel = 'noreferrer noopener'; link.rel = 'noreferrer noopener';
var evt = document.createEvent('MouseEvents'); var evt = document.createEvent('MouseEvents');
evt.initMouseEvent('click', true, true, window, true, 0, 0, 0, 0, false, false, false, false, 0, null); evt.initMouseEvent('click', true, true, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
link.dispatchEvent(evt);
appendClickRemove(link, evt);
} else { } else {
var win = window.open('', '_blank'); var win = window.open('', '_blank');
if (win) { if (win) {
@ -84,7 +88,7 @@ tinymce.PluginManager.add('link', function(editor) {
doc.write('<meta http-equiv="refresh" content="0; url=' + tinymce.DOM.encode(url) + '">'); doc.write('<meta http-equiv="refresh" content="0; url=' + tinymce.DOM.encode(url) + '">');
doc.close(); doc.close();
} }
}*/ }
} }
function gotoLink(a) { function gotoLink(a) {

View File

@ -1853,4 +1853,4 @@ define("tinymce/pasteplugin/Plugin", [
}); });
expose(["tinymce/pasteplugin/Utils"]); expose(["tinymce/pasteplugin/Utils"]);
})(this); })(window);

View File

@ -1 +1 @@
body{background-color:#FFFFFF;color:#000000;font-family:Verdana,Arial,Helvetica,sans-serif;font-size:14px;scrollbar-3dlight-color:#F0F0EE;scrollbar-arrow-color:#676662;scrollbar-base-color:#F0F0EE;scrollbar-darkshadow-color:#DDDDDD;scrollbar-face-color:#E0E0DD;scrollbar-highlight-color:#F0F0EE;scrollbar-shadow-color:#F0F0EE;scrollbar-track-color:#F5F5F5}td,th{font-family:Verdana,Arial,Helvetica,sans-serif;font-size:14px}.mce-content-body .mce-reset{margin:0;padding:0;border:0;outline:0;vertical-align:top;background:transparent;text-decoration:none;color:black;font-family:Arial;font-size:11px;text-shadow:none;float:none;position:static;width:auto;height:auto;white-space:nowrap;cursor:inherit;line-height:normal;font-weight:normal;text-align:left;-webkit-tap-highlight-color:transparent;-moz-box-sizing:content-box;-webkit-box-sizing:content-box;box-sizing:content-box;direction:ltr;max-width:none}.mce-object{border:1px dotted #3A3A3A;background:#D5D5D5 url(img/object.gif) no-repeat center}.mce-preview-object{display:inline-block;position:relative;margin:0 2px 0 2px;line-height:0;border:1px solid gray}.mce-preview-object .mce-shim{position:absolute;top:0;left:0;width:100%;height:100%;background:url()}figure.align-left{float:left}figure.align-right{float:right}figure.image.align-center{display:table;margin-left:auto;margin-right:auto}figure.image{display:inline-block;border:1px solid gray;margin:0 2px 0 1px;background:#f5f2f0}figure.image img{margin:8px 8px 0 8px}figure.image figcaption{margin:6px 8px 6px 8px;text-align:center}.mce-toc{border:1px solid gray}.mce-toc h2{margin:4px}.mce-toc li{list-style-type:none}.mce-preview-object[data-mce-selected="2"] .mce-shim{display:none}.mce-pagebreak{cursor:default;display:block;border:0;width:100%;height:5px;border:1px dashed #666;margin-top:15px;page-break-before:always}@media print{.mce-pagebreak{border:0}}.mce-item-anchor{cursor:default;display:inline-block;-webkit-user-select:all;-webkit-user-modify:read-only;-moz-user-select:all;-moz-user-modify:read-only;user-select:all;user-modify:read-only;width:9px !important;height:9px !important;border:1px dotted #3A3A3A;background:#D5D5D5 url(img/anchor.gif) no-repeat center}.mce-nbsp,.mce-shy{background:#AAA}.mce-shy::after{content:'-'}hr{cursor:default}.mce-match-marker{background:#AAA;color:#fff}.mce-match-marker-selected{background:#3399ff;color:#fff}.mce-spellchecker-word{border-bottom:2px solid #F00;cursor:default}.mce-spellchecker-grammar{border-bottom:2px solid #008000;cursor:default}.mce-item-table,.mce-item-table td,.mce-item-table th,.mce-item-table caption{border:1px dashed #BBB}td[data-mce-selected],th[data-mce-selected]{background-color:#3399ff !important}.mce-edit-focus{outline:1px dotted #333}.mce-content-body *[contentEditable=false] *[contentEditable=true]:focus{outline:2px solid #2d8ac7}.mce-content-body *[contentEditable=false] *[contentEditable=true]:hover{outline:2px solid #7ACAFF}.mce-content-body *[contentEditable=false][data-mce-selected]{outline:2px solid #2d8ac7}.mce-resize-bar-dragging{background-color:blue;opacity:.25;filter:alpha(opacity=25);zoom:1} body{background-color:#FFFFFF;color:#000000;font-family:Verdana,Arial,Helvetica,sans-serif;font-size:14px;scrollbar-3dlight-color:#F0F0EE;scrollbar-arrow-color:#676662;scrollbar-base-color:#F0F0EE;scrollbar-darkshadow-color:#DDDDDD;scrollbar-face-color:#E0E0DD;scrollbar-highlight-color:#F0F0EE;scrollbar-shadow-color:#F0F0EE;scrollbar-track-color:#F5F5F5}td,th{font-family:Verdana,Arial,Helvetica,sans-serif;font-size:14px}.mce-content-body .mce-reset{margin:0;padding:0;border:0;outline:0;vertical-align:top;background:transparent;text-decoration:none;color:black;font-family:Arial;font-size:11px;text-shadow:none;float:none;position:static;width:auto;height:auto;white-space:nowrap;cursor:inherit;line-height:normal;font-weight:normal;text-align:left;-webkit-tap-highlight-color:transparent;-moz-box-sizing:content-box;-webkit-box-sizing:content-box;box-sizing:content-box;direction:ltr;max-width:none}.mce-object{border:1px dotted #3A3A3A;background:#D5D5D5 url(img/object.gif) no-repeat center}.mce-preview-object{display:inline-block;position:relative;margin:0 2px 0 2px;line-height:0;border:1px solid gray}.mce-preview-object[data-mce-selected="2"] .mce-shim{display:none}.mce-preview-object .mce-shim{position:absolute;top:0;left:0;width:100%;height:100%;background:url()}figure.align-left{float:left}figure.align-right{float:right}figure.image.align-center{display:table;margin-left:auto;margin-right:auto}figure.image{display:inline-block;border:1px solid gray;margin:0 2px 0 1px;background:#f5f2f0}figure.image img{margin:8px 8px 0 8px}figure.image figcaption{margin:6px 8px 6px 8px;text-align:center}.mce-toc{border:1px solid gray}.mce-toc h2{margin:4px}.mce-toc li{list-style-type:none}.mce-pagebreak{cursor:default;display:block;border:0;width:100%;height:5px;border:1px dashed #666;margin-top:15px;page-break-before:always}@media print{.mce-pagebreak{border:0}}.mce-item-anchor{cursor:default;display:inline-block;-webkit-user-select:all;-webkit-user-modify:read-only;-moz-user-select:all;-moz-user-modify:read-only;user-select:all;user-modify:read-only;width:9px !important;height:9px !important;border:1px dotted #3A3A3A;background:#D5D5D5 url(img/anchor.gif) no-repeat center}.mce-nbsp,.mce-shy{background:#AAA}.mce-shy::after{content:'-'}hr{cursor:default}.mce-match-marker{background:#AAA;color:#fff}.mce-match-marker-selected{background:#3399ff;color:#fff}.mce-spellchecker-word{border-bottom:2px solid #F00;cursor:default}.mce-spellchecker-grammar{border-bottom:2px solid #008000;cursor:default}.mce-item-table,.mce-item-table td,.mce-item-table th,.mce-item-table caption{border:1px dashed #BBB}td[data-mce-selected],th[data-mce-selected]{background-color:#3399ff !important}.mce-edit-focus{outline:1px dotted #333}.mce-content-body *[contentEditable=false] *[contentEditable=true]:focus{outline:2px solid #2d8ac7}.mce-content-body *[contentEditable=false] *[contentEditable=true]:hover{outline:2px solid #7ACAFF}.mce-content-body *[contentEditable=false][data-mce-selected]{outline:2px solid #2d8ac7}.mce-resize-bar-dragging{background-color:blue;opacity:.25;filter:alpha(opacity=25);zoom:1}

View File

@ -875,13 +875,16 @@ define('tinymce.modern.ui.Sidebar', [
define('tinymce.modern.ui.SkinLoaded', [ define('tinymce.modern.ui.SkinLoaded', [
], function () { ], function () {
var fireSkinLoaded = function (editor) { var fireSkinLoaded = function (editor) {
var done = function () {
editor._skinLoaded = true;
editor.fire('SkinLoaded');
};
return function() { return function() {
if (editor.initialized) { if (editor.initialized) {
editor.fire('SkinLoaded'); done();
} else { } else {
editor.on('init', function() { editor.on('init', done);
editor.fire('SkinLoaded');
});
} }
}; };
}; };

View File

@ -1,4 +1,4 @@
// 4.5.1 (2016-12-07) // 4.5.2 (2017-01-04)
/** /**
* Compiled inline version. (Library mode) * Compiled inline version. (Library mode)
@ -9425,10 +9425,15 @@ define("tinymce/dom/ScriptLoader", [
var DOM = DOMUtils.DOM; var DOM = DOMUtils.DOM;
var each = Tools.each, grep = Tools.grep; var each = Tools.each, grep = Tools.grep;
var isFunction = function (f) {
return typeof f === 'function';
};
function ScriptLoader() { function ScriptLoader() {
var QUEUED = 0, var QUEUED = 0,
LOADING = 1, LOADING = 1,
LOADED = 2, LOADED = 2,
FAILED = 3,
states = {}, states = {},
queue = [], queue = [],
scriptLoadedCallbacks = {}, scriptLoadedCallbacks = {},
@ -9441,9 +9446,10 @@ define("tinymce/dom/ScriptLoader", [
* *
* @method load * @method load
* @param {String} url Absolute URL to script to add. * @param {String} url Absolute URL to script to add.
* @param {function} callback Optional callback function to execute ones this script gets loaded. * @param {function} callback Optional success callback function when the script loaded successfully.
* @param {function} callback Optional failure callback function when the script failed to load.
*/ */
function loadScript(url, callback) { function loadScript(url, success, failure) {
var dom = DOM, elm, id; var dom = DOM, elm, id;
// Execute callback when script is loaded // Execute callback when script is loaded
@ -9454,21 +9460,25 @@ define("tinymce/dom/ScriptLoader", [
elm.onreadystatechange = elm.onload = elm = null; elm.onreadystatechange = elm.onload = elm = null;
} }
callback(); success();
} }
function error() { function error() {
/*eslint no-console:0 */ /*eslint no-console:0 */
// Report the error so it's easier for people to spot loading errors
if (typeof console !== "undefined" && console.log) {
console.log("Failed to load: " + url);
}
// We can't mark it as done if there is a load error since // We can't mark it as done if there is a load error since
// A) We don't want to produce 404 errors on the server and // A) We don't want to produce 404 errors on the server and
// B) the onerror event won't fire on all browsers. // B) the onerror event won't fire on all browsers.
// done(); // done();
if (isFunction(failure)) {
failure();
} else {
// Report the error so it's easier for people to spot loading errors
if (typeof console !== "undefined" && console.log) {
console.log("Failed to load script: " + url);
}
}
} }
id = dom.uniqueId(); id = dom.uniqueId();
@ -9524,10 +9534,11 @@ define("tinymce/dom/ScriptLoader", [
* *
* @method add * @method add
* @param {String} url Absolute URL to script to add. * @param {String} url Absolute URL to script to add.
* @param {function} callback Optional callback function to execute ones this script gets loaded. * @param {function} success Optional success callback function to execute when the script loades successfully.
* @param {Object} scope Optional scope to execute callback in. * @param {Object} scope Optional scope to execute callback in.
* @param {function} failure Optional failure callback function to execute when the script failed to load.
*/ */
this.add = this.load = function(url, callback, scope) { this.add = this.load = function(url, success, scope, failure) {
var state = states[url]; var state = states[url];
// Add url to load queue // Add url to load queue
@ -9536,14 +9547,15 @@ define("tinymce/dom/ScriptLoader", [
states[url] = QUEUED; states[url] = QUEUED;
} }
if (callback) { if (success) {
// Store away callback for later execution // Store away callback for later execution
if (!scriptLoadedCallbacks[url]) { if (!scriptLoadedCallbacks[url]) {
scriptLoadedCallbacks[url] = []; scriptLoadedCallbacks[url] = [];
} }
scriptLoadedCallbacks[url].push({ scriptLoadedCallbacks[url].push({
func: callback, success: success,
failure: failure,
scope: scope || this scope: scope || this
}); });
} }
@ -9558,11 +9570,12 @@ define("tinymce/dom/ScriptLoader", [
* Starts the loading of the queue. * Starts the loading of the queue.
* *
* @method loadQueue * @method loadQueue
* @param {function} callback Optional callback to execute when all queued items are loaded. * @param {function} success Optional callback to execute when all queued items are loaded.
* @param {function} failure Optional callback to execute when queued items failed to load.
* @param {Object} scope Optional scope to execute the callback in. * @param {Object} scope Optional scope to execute the callback in.
*/ */
this.loadQueue = function(callback, scope) { this.loadQueue = function(success, scope, failure) {
this.loadScripts(queue, callback, scope); this.loadScripts(queue, success, scope, failure);
}; };
/** /**
@ -9571,23 +9584,27 @@ define("tinymce/dom/ScriptLoader", [
* *
* @method loadScripts * @method loadScripts
* @param {Array} scripts Array of queue items to load. * @param {Array} scripts Array of queue items to load.
* @param {function} callback Optional callback to execute ones all items are loaded. * @param {function} callback Optional callback to execute when scripts is loaded successfully.
* @param {Object} scope Optional scope to execute callback in. * @param {Object} scope Optional scope to execute callback in.
* @param {function} callback Optional callback to execute if scripts failed to load.
*/ */
this.loadScripts = function(scripts, callback, scope) { this.loadScripts = function(scripts, success, scope, failure) {
var loadScripts; var loadScripts, failures = [];
function execScriptLoadedCallbacks(url) { function execCallbacks(name, url) {
// Execute URL callback functions // Execute URL callback functions
each(scriptLoadedCallbacks[url], function(callback) { each(scriptLoadedCallbacks[url], function(callback) {
callback.func.call(callback.scope); if (isFunction(callback[name])) {
callback[name].call(callback.scope);
}
}); });
scriptLoadedCallbacks[url] = undef; scriptLoadedCallbacks[url] = undef;
} }
queueLoadedCallbacks.push({ queueLoadedCallbacks.push({
func: callback, success: success,
failure: failure,
scope: scope || this scope: scope || this
}); });
@ -9600,13 +9617,18 @@ define("tinymce/dom/ScriptLoader", [
// Load scripts that needs to be loaded // Load scripts that needs to be loaded
each(loadingScripts, function(url) { each(loadingScripts, function(url) {
// Script is already loaded then execute script callbacks directly // Script is already loaded then execute script callbacks directly
if (states[url] == LOADED) { if (states[url] === LOADED) {
execScriptLoadedCallbacks(url); execCallbacks('success', url);
return;
}
if (states[url] === FAILED) {
execCallbacks('failure', url);
return; return;
} }
// Is script not loading then start loading it // Is script not loading then start loading it
if (states[url] != LOADING) { if (states[url] !== LOADING) {
states[url] = LOADING; states[url] = LOADING;
loading++; loading++;
@ -9614,7 +9636,16 @@ define("tinymce/dom/ScriptLoader", [
states[url] = LOADED; states[url] = LOADED;
loading--; loading--;
execScriptLoadedCallbacks(url); execCallbacks('success', url);
// Load more scripts if they where added by the recently loaded script
loadScripts();
}, function () {
states[url] = FAILED;
loading--;
failures.push(url);
execCallbacks('failure', url);
// Load more scripts if they where added by the recently loaded script // Load more scripts if they where added by the recently loaded script
loadScripts(); loadScripts();
@ -9625,7 +9656,15 @@ define("tinymce/dom/ScriptLoader", [
// No scripts are currently loading then execute all pending queue loaded callbacks // No scripts are currently loading then execute all pending queue loaded callbacks
if (!loading) { if (!loading) {
each(queueLoadedCallbacks, function(callback) { each(queueLoadedCallbacks, function(callback) {
callback.func.call(callback.scope); if (failures.length === 0) {
if (isFunction(callback.success)) {
callback.success.call(callback.scope);
}
} else {
if (isFunction(callback.failure)) {
callback.failure.call(callback.scope, failures);
}
}
}); });
queueLoadedCallbacks.length = 0; queueLoadedCallbacks.length = 0;
@ -9793,8 +9832,9 @@ define("tinymce/AddOnManager", [
* @method load * @method load
* @param {String} name Short name of the add-on that gets loaded. * @param {String} name Short name of the add-on that gets loaded.
* @param {String} addOnUrl URL to the add-on that will get loaded. * @param {String} addOnUrl URL to the add-on that will get loaded.
* @param {function} callback Optional callback to execute ones the add-on is loaded. * @param {function} success Optional success callback to execute when an add-on is loaded.
* @param {Object} scope Optional scope to execute the callback in. * @param {Object} scope Optional scope to execute the callback in.
* @param {function} failure Optional failure callback to execute when an add-on failed to load.
* @example * @example
* // Loads a plugin from an external URL * // Loads a plugin from an external URL
* tinymce.PluginManager.load('myplugin', '/some/dir/someplugin/plugin.js'); * tinymce.PluginManager.load('myplugin', '/some/dir/someplugin/plugin.js');
@ -9805,7 +9845,7 @@ define("tinymce/AddOnManager", [
* plugins: '-myplugin' // Don't try to load it again * plugins: '-myplugin' // Don't try to load it again
* }); * });
*/ */
load: function(name, addOnUrl, callback, scope) { load: function(name, addOnUrl, success, scope, failure) {
var self = this, url = addOnUrl; var self = this, url = addOnUrl;
function loadDependencies() { function loadDependencies() {
@ -9817,11 +9857,11 @@ define("tinymce/AddOnManager", [
self.load(newUrl.resource, newUrl, undefined, undefined); self.load(newUrl.resource, newUrl, undefined, undefined);
}); });
if (callback) { if (success) {
if (scope) { if (scope) {
callback.call(scope); success.call(scope);
} else { } else {
callback.call(ScriptLoader); success.call(ScriptLoader);
} }
} }
} }
@ -9843,7 +9883,7 @@ define("tinymce/AddOnManager", [
if (self.lookup[name]) { if (self.lookup[name]) {
loadDependencies(); loadDependencies();
} else { } else {
ScriptLoader.ScriptLoader.add(url, loadDependencies, scope); ScriptLoader.ScriptLoader.add(url, loadDependencies, scope, failure);
} }
} }
}; };
@ -18811,7 +18851,7 @@ define("tinymce/fmt/Preview", [
function selectorToHtml(selector, editor) { function selectorToHtml(selector, editor) {
return parsedSelectorToHtml(parseSelector(selector, editor)); return parsedSelectorToHtml(parseSelector(selector), editor);
} }
@ -18934,9 +18974,9 @@ define("tinymce/fmt/Preview", [
items[0].name = name; items[0].name = name;
} }
name = format.selector; name = format.selector;
previewFrag = parsedSelectorToHtml(items); previewFrag = parsedSelectorToHtml(items, editor);
} else { } else {
previewFrag = parsedSelectorToHtml([name]); previewFrag = parsedSelectorToHtml([name], editor);
} }
previewElm = dom.select(name, previewFrag)[0] || previewFrag.firstChild; previewElm = dom.select(name, previewFrag)[0] || previewFrag.firstChild;
@ -21364,8 +21404,10 @@ define("tinymce/Formatter", [
textNode = findFirstTextNode(caretContainer); textNode = findFirstTextNode(caretContainer);
} }
// Expand to word is caret is in the middle of a text node and the char before/after is a alpha numeric character // Expand to word if caret is in the middle of a text node and the char before/after is a alpha numeric character
if (text && offset > 0 && offset < text.length && /\w/.test(text.charAt(offset)) && /\w/.test(text.charAt(offset - 1))) { var wordcharRegex = /[^\s\u00a0\u00ad\u200b\ufeff]/;
if (text && offset > 0 && offset < text.length &&
wordcharRegex.test(text.charAt(offset)) && wordcharRegex.test(text.charAt(offset - 1))) {
// Get bookmark of caret position // Get bookmark of caret position
bookmark = selection.getBookmark(); bookmark = selection.getBookmark();
@ -22952,6 +22994,10 @@ define("tinymce/EnterKey", [
// Insert new block before/after the parent block depending on caret location // Insert new block before/after the parent block depending on caret location
if (CaretContainer.isCaretContainerBlock(parentBlock)) { if (CaretContainer.isCaretContainerBlock(parentBlock)) {
newBlock = CaretContainer.showCaretContainerBlock(parentBlock); newBlock = CaretContainer.showCaretContainerBlock(parentBlock);
if (dom.isEmpty(parentBlock)) {
emptyBlock(parentBlock);
}
moveToCaretPosition(newBlock);
} else if (isCaretAtStartOrEndOfBlock()) { } else if (isCaretAtStartOrEndOfBlock()) {
insertNewBlockAfter(); insertNewBlockAfter();
} else if (isCaretAtStartOrEndOfBlock(true)) { } else if (isCaretAtStartOrEndOfBlock(true)) {
@ -26051,7 +26097,7 @@ define("tinymce/util/Observable", [
return { return {
/** /**
* Fires the specified event by name. Consult the * Fires the specified event by name. Consult the
* <a href="/advanced/events">event reference</a> for more details on each event. * <a href="/docs/advanced/events">event reference</a> for more details on each event.
* *
* @method fire * @method fire
* @param {String} name Name of the event to fire. * @param {String} name Name of the event to fire.
@ -26085,7 +26131,7 @@ define("tinymce/util/Observable", [
/** /**
* Binds an event listener to a specific event by name. Consult the * Binds an event listener to a specific event by name. Consult the
* <a href="/advanced/events">event reference</a> for more details on each event. * <a href="/docs/advanced/events">event reference</a> for more details on each event.
* *
* @method on * @method on
* @param {String} name Event name or space separated list of events to bind. * @param {String} name Event name or space separated list of events to bind.
@ -26103,7 +26149,7 @@ define("tinymce/util/Observable", [
/** /**
* Unbinds an event listener to a specific event by name. Consult the * Unbinds an event listener to a specific event by name. Consult the
* <a href="/advanced/events">event reference</a> for more details on each event. * <a href="/docs/advanced/events">event reference</a> for more details on each event.
* *
* @method off * @method off
* @param {String?} name Name of the event to unbind. * @param {String?} name Name of the event to unbind.
@ -26125,7 +26171,7 @@ define("tinymce/util/Observable", [
/** /**
* Bind the event callback and once it fires the callback is removed. Consult the * Bind the event callback and once it fires the callback is removed. Consult the
* <a href="/advanced/events">event reference</a> for more details on each event. * <a href="/docs/advanced/events">event reference</a> for more details on each event.
* *
* @method once * @method once
* @param {String} name Name of the event to bind. * @param {String} name Name of the event to bind.
@ -34409,8 +34455,12 @@ define("tinymce/util/Quirks", [
var rng = editor.selection.getRng(); var rng = editor.selection.getRng();
var startCaretPos = CaretPosition.fromRangeStart(rng); var startCaretPos = CaretPosition.fromRangeStart(rng);
var endCaretPos = CaretPosition.fromRangeEnd(rng); var endCaretPos = CaretPosition.fromRangeEnd(rng);
var prev = caretWalker.prev(startCaretPos);
var next = caretWalker.next(endCaretPos);
return !editor.selection.isCollapsed() && !caretWalker.prev(startCaretPos) && !caretWalker.next(endCaretPos); return !editor.selection.isCollapsed() &&
(!prev || prev.isAtStart()) &&
(!next || (next.isAtEnd() && startCaretPos.getNode() !== next.getNode()));
} }
// Type over case delete and insert this won't cover typeover with a IME but at least it covers the common case // Type over case delete and insert this won't cover typeover with a IME but at least it covers the common case
@ -35199,11 +35249,11 @@ define("tinymce/file/Uploader", [
resolve(handlerSuccess(blobInfo, url)); resolve(handlerSuccess(blobInfo, url));
}; };
var failure = function() { var failure = function(error) {
closeNotification(); closeNotification();
uploadStatus.removeFailed(blobInfo.blobUri()); uploadStatus.removeFailed(blobInfo.blobUri());
resolvePending(blobInfo.blobUri(), handlerFailure(blobInfo, failure)); resolvePending(blobInfo.blobUri(), handlerFailure(blobInfo, error));
resolve(handlerFailure(blobInfo, failure)); resolve(handlerFailure(blobInfo, error));
}; };
progress = function(percent) { progress = function(percent) {
@ -35401,6 +35451,10 @@ define("tinymce/file/ImageScanner", [
], function(Promise, Arr, Fun, Conversions, Env) { ], function(Promise, Arr, Fun, Conversions, Env) {
var count = 0; var count = 0;
var uniqueId = function(prefix) {
return (prefix || 'blobid') + (count++);
};
return function(uploadStatus, blobCache) { return function(uploadStatus, blobCache) {
var cachedPromises = {}; var cachedPromises = {};
@ -35418,6 +35472,19 @@ define("tinymce/file/ImageScanner", [
image: img, image: img,
blobInfo: blobInfo blobInfo: blobInfo
}); });
} else {
Conversions.uriToBlob(img.src).then(function (blob) {
Conversions.blobToDataUri(blob).then(function (dataUri) {
base64 = Conversions.parseDataUri(dataUri).data;
blobInfo = blobCache.create(uniqueId(), blob, base64);
blobCache.add(blobInfo);
resolve({
image: img,
blobInfo: blobInfo
});
});
});
} }
return; return;
@ -35435,9 +35502,7 @@ define("tinymce/file/ImageScanner", [
}); });
} else { } else {
Conversions.uriToBlob(img.src).then(function(blob) { Conversions.uriToBlob(img.src).then(function(blob) {
var blobInfoId = 'blobid' + (count++), blobInfo = blobCache.create(uniqueId(), blob, base64);
blobInfo = blobCache.create(blobInfoId, blob, base64);
blobCache.add(blobInfo); blobCache.add(blobInfo);
resolve({ resolve({
@ -35688,6 +35753,78 @@ define("tinymce/file/UploadStatus", [
}; };
}); });
// Included from: js/tinymce/classes/ErrorReporter.js
/**
* ErrorReporter.js
*
* Released under LGPL License.
* Copyright (c) 1999-2016 Ephox Corp. All rights reserved
*
* License: http://www.tinymce.com/license
* Contributing: http://www.tinymce.com/contributing
*/
/**
* Various error reporting helper functions.
*
* @class tinymce.ErrorReporter
* @private
*/
define("tinymce/ErrorReporter", [
"tinymce/AddOnManager"
], function (AddOnManager) {
var PluginManager = AddOnManager.PluginManager;
var resolvePluginName = function (targetUrl, suffix) {
for (var name in PluginManager.urls) {
var matchUrl = PluginManager.urls[name] + '/plugin' + suffix + '.js';
if (matchUrl === targetUrl) {
return name;
}
}
return null;
};
var pluginUrlToMessage = function (editor, url) {
var plugin = resolvePluginName(url, editor.suffix);
return plugin ?
'Failed to load plugin: ' + plugin + ' from url ' + url :
'Failed to load plugin url: ' + url;
};
var displayNotification = function (editor, message) {
editor.notificationManager.open({
type: 'error',
text: message
});
};
var displayError = function (editor, message) {
if (editor._skinLoaded) {
displayNotification(editor, message);
} else {
editor.on('SkinLoaded', function () {
displayNotification(editor, message);
});
}
};
var uploadError = function (editor, message) {
displayError(editor, 'Failed to upload image: ' + message);
};
var pluginLoadError = function (editor, url) {
displayError(editor, pluginUrlToMessage(editor, url));
};
return {
pluginLoadError: pluginLoadError,
uploadError: uploadError
};
});
// Included from: js/tinymce/classes/EditorUpload.js // Included from: js/tinymce/classes/EditorUpload.js
/** /**
@ -35711,8 +35848,9 @@ define("tinymce/EditorUpload", [
"tinymce/file/Uploader", "tinymce/file/Uploader",
"tinymce/file/ImageScanner", "tinymce/file/ImageScanner",
"tinymce/file/BlobCache", "tinymce/file/BlobCache",
"tinymce/file/UploadStatus" "tinymce/file/UploadStatus",
], function(Arr, Uploader, ImageScanner, BlobCache, UploadStatus) { "tinymce/ErrorReporter"
], function(Arr, Uploader, ImageScanner, BlobCache, UploadStatus, ErrorReporter) {
return function(editor) { return function(editor) {
var blobCache = new BlobCache(), uploader, imageScanner, settings = editor.settings; var blobCache = new BlobCache(), uploader, imageScanner, settings = editor.settings;
var uploadStatus = new UploadStatus(); var uploadStatus = new UploadStatus();
@ -35808,6 +35946,8 @@ define("tinymce/EditorUpload", [
if (uploadInfo.status && editor.settings.images_replace_blob_uris !== false) { if (uploadInfo.status && editor.settings.images_replace_blob_uris !== false) {
replaceImageUri(image, uploadInfo.url); replaceImageUri(image, uploadInfo.url);
} else if (uploadInfo.error) {
ErrorReporter.uploadError(editor, uploadInfo.error);
} }
return { return {
@ -37822,6 +37962,7 @@ define("tinymce/SelectionOverrides", [
editor.$('*[data-mce-selected]').removeAttr('data-mce-selected'); editor.$('*[data-mce-selected]').removeAttr('data-mce-selected');
node.setAttribute('data-mce-selected', 1); node.setAttribute('data-mce-selected', 1);
selectedContentEditableNode = node; selectedContentEditableNode = node;
hideFakeCaret();
return range; return range;
} }
@ -38005,13 +38146,14 @@ define("tinymce/Editor", [
"tinymce/EditorUpload", "tinymce/EditorUpload",
"tinymce/SelectionOverrides", "tinymce/SelectionOverrides",
"tinymce/util/Uuid", "tinymce/util/Uuid",
"tinymce/ui/Sidebar" "tinymce/ui/Sidebar",
"tinymce/ErrorReporter"
], function( ], function(
DOMUtils, DomQuery, AddOnManager, NodeChange, Node, DomSerializer, Serializer, DOMUtils, DomQuery, AddOnManager, NodeChange, Node, DomSerializer, Serializer,
Selection, Formatter, UndoManager, EnterKey, ForceBlocks, EditorCommands, Selection, Formatter, UndoManager, EnterKey, ForceBlocks, EditorCommands,
URI, ScriptLoader, EventUtils, WindowManager, NotificationManager, URI, ScriptLoader, EventUtils, WindowManager, NotificationManager,
Schema, DomParser, Quirks, Env, Tools, Delay, EditorObservable, Mode, Shortcuts, EditorUpload, Schema, DomParser, Quirks, Env, Tools, Delay, EditorObservable, Mode, Shortcuts, EditorUpload,
SelectionOverrides, Uuid, Sidebar SelectionOverrides, Uuid, Sidebar, ErrorReporter
) { ) {
// Shorten these names // Shorten these names
var DOM = DOMUtils.DOM, ThemeManager = AddOnManager.ThemeManager, PluginManager = AddOnManager.PluginManager; var DOM = DOMUtils.DOM, ThemeManager = AddOnManager.ThemeManager, PluginManager = AddOnManager.PluginManager;
@ -38407,6 +38549,12 @@ define("tinymce/Editor", [
}); });
scriptLoader.loadQueue(function() { scriptLoader.loadQueue(function() {
if (!self.removed) {
self.init();
}
}, self, function (urls) {
ErrorReporter.pluginLoadError(self, urls[0]);
if (!self.removed) { if (!self.removed) {
self.init(); self.init();
} }
@ -40695,7 +40843,7 @@ define("tinymce/EditorManager", [
* @property minorVersion * @property minorVersion
* @type String * @type String
*/ */
minorVersion: '5.1', minorVersion: '5.2',
/** /**
* Release date of TinyMCE build. * Release date of TinyMCE build.
@ -40703,7 +40851,7 @@ define("tinymce/EditorManager", [
* @property releaseDate * @property releaseDate
* @type String * @type String
*/ */
releaseDate: '2016-12-07', releaseDate: '2017-01-04',
/** /**
* Collection of editor instances. * Collection of editor instances.
@ -45239,6 +45387,73 @@ define("tinymce/ui/FlowLayout", [
}); });
}); });
// Included from: js/tinymce/classes/fmt/FontInfo.js
/**
* FontInfo.js
*
* Released under LGPL License.
* Copyright (c) 1999-2016 Ephox Corp. All rights reserved
*
* License: http://www.tinymce.com/license
* Contributing: http://www.tinymce.com/contributing
*/
/**
* Internal class for computing font size for elements.
*
* @private
* @class tinymce.fmt.FontInfo
*/
define("tinymce/fmt/FontInfo", [
"tinymce/dom/DOMUtils"
], function(DOMUtils) {
var getSpecifiedFontProp = function (propName, rootElm, elm) {
while (elm !== rootElm) {
if (elm.style[propName]) {
return elm.style[propName];
}
elm = elm.parentNode;
}
return 0;
};
var toPt = function (fontSize) {
if (/[0-9.]+px$/.test(fontSize)) {
return Math.round(parseInt(fontSize, 10) * 72 / 96) + 'pt';
}
return fontSize;
};
var normalizeFontFamily = function (fontFamily) {
// 'Font name', Font -> Font name,Font
return fontFamily.replace(/[\'\"]/g, '').replace(/,\s+/g, ',');
};
var getComputedFontProp = function (propName, elm) {
return DOMUtils.DOM.getStyle(elm, propName, true);
};
var getFontSize = function (rootElm, elm) {
var specifiedFontSize = getSpecifiedFontProp('fontSize', rootElm, elm);
return specifiedFontSize ? specifiedFontSize : getComputedFontProp('fontSize', elm);
};
var getFontFamily = function (rootElm, elm) {
var specifiedFontSize = getSpecifiedFontProp('fontFamily', rootElm, elm);
return normalizeFontFamily(specifiedFontSize ? specifiedFontSize : getComputedFontProp('fontFamily', elm));
};
return {
getFontSize: getFontSize,
getFontFamily: getFontFamily,
toPt: toPt
};
});
// Included from: js/tinymce/classes/ui/FormatControls.js // Included from: js/tinymce/classes/ui/FormatControls.js
/** /**
@ -45265,8 +45480,9 @@ define("tinymce/ui/FormatControls", [
"tinymce/util/Arr", "tinymce/util/Arr",
"tinymce/dom/DOMUtils", "tinymce/dom/DOMUtils",
"tinymce/EditorManager", "tinymce/EditorManager",
"tinymce/Env" "tinymce/Env",
], function(Control, Widget, FloatPanel, Tools, Arr, DOMUtils, EditorManager, Env) { "tinymce/fmt/FontInfo"
], function(Control, Widget, FloatPanel, Tools, Arr, DOMUtils, EditorManager, Env, FontInfo) {
var each = Tools.each; var each = Tools.each;
var flatten = function (ar) { var flatten = function (ar) {
@ -45341,6 +45557,67 @@ define("tinymce/ui/FormatControls", [
}; };
} }
function createFontNameListBoxChangeHandler(items) {
return function() {
var self = this;
var getFirstFont = function (fontFamily) {
return fontFamily ? fontFamily.split(',')[0] : '';
};
editor.on('nodeChange', function(e) {
var fontFamily, value = null;
fontFamily = FontInfo.getFontFamily(editor.getBody(), e.element);
each(items, function(item) {
if (item.value.toLowerCase() === fontFamily.toLowerCase()) {
value = item.value;
}
});
each(items, function(item) {
if (!value && getFirstFont(item.value).toLowerCase() === getFirstFont(fontFamily).toLowerCase()) {
value = item.value;
}
});
self.value(value);
if (!value && fontFamily) {
self.text(getFirstFont(fontFamily));
}
});
};
}
function createFontSizeListBoxChangeHandler(items) {
return function() {
var self = this;
editor.on('nodeChange', function(e) {
var px, pt, value = null;
px = FontInfo.getFontSize(editor.getBody(), e.element);
pt = FontInfo.toPt(px);
each(items, function(item) {
if (item.value === px) {
value = px;
} else if (item.value === pt) {
value = pt;
}
});
self.value(value);
if (!value) {
self.text(pt);
}
});
};
}
function createFormats(formats) { function createFormats(formats) {
formats = formats.replace(/;$/, '').split(';'); formats = formats.replace(/;$/, '').split(';');
@ -45724,7 +46001,7 @@ define("tinymce/ui/FormatControls", [
selectall: ['Select all', 'SelectAll', 'Meta+A'], selectall: ['Select all', 'SelectAll', 'Meta+A'],
bold: ['Bold', 'Bold', 'Meta+B'], bold: ['Bold', 'Bold', 'Meta+B'],
italic: ['Italic', 'Italic', 'Meta+I'], italic: ['Italic', 'Italic', 'Meta+I'],
underline: ['Underline', 'Underline'], underline: ['Underline', 'Underline', 'Meta+U'],
strikethrough: ['Strikethrough', 'Strikethrough'], strikethrough: ['Strikethrough', 'Strikethrough'],
subscript: ['Subscript', 'Subscript'], subscript: ['Subscript', 'Subscript'],
superscript: ['Superscript', 'Superscript'], superscript: ['Superscript', 'Superscript'],
@ -45877,7 +46154,7 @@ define("tinymce/ui/FormatControls", [
tooltip: 'Font Family', tooltip: 'Font Family',
values: items, values: items,
fixedWidth: true, fixedWidth: true,
onPostRender: createListBoxChangeHandler(items, 'fontname'), onPostRender: createFontNameListBoxChangeHandler(items),
onselect: function(e) { onselect: function(e) {
if (e.control.settings.value) { if (e.control.settings.value) {
editor.execCommand('FontName', false, e.control.settings.value); editor.execCommand('FontName', false, e.control.settings.value);
@ -45907,7 +46184,7 @@ define("tinymce/ui/FormatControls", [
tooltip: 'Font Sizes', tooltip: 'Font Sizes',
values: items, values: items,
fixedWidth: true, fixedWidth: true,
onPostRender: createListBoxChangeHandler(items, 'fontsize'), onPostRender: createFontSizeListBoxChangeHandler(items),
onclick: function(e) { onclick: function(e) {
if (e.control.settings.value) { if (e.control.settings.value) {
editor.execCommand('FontSize', false, e.control.settings.value); editor.execCommand('FontSize', false, e.control.settings.value);