| (function () { | |
| var image = (function (domGlobals) { | |
| 'use strict'; | |
| var global = tinymce.util.Tools.resolve('tinymce.PluginManager'); | |
| var hasDimensions = function (editor) { | |
| return editor.settings.image_dimensions === false ? false : true; | |
| }; | |
| var hasAdvTab = function (editor) { | |
| return editor.settings.image_advtab === true ? true : false; | |
| }; | |
| var getPrependUrl = function (editor) { | |
| return editor.getParam('image_prepend_url', ''); | |
| }; | |
| var getClassList = function (editor) { | |
| return editor.getParam('image_class_list'); | |
| }; | |
| var hasDescription = function (editor) { | |
| return editor.settings.image_description === false ? false : true; | |
| }; | |
| var hasImageTitle = function (editor) { | |
| return editor.settings.image_title === true ? true : false; | |
| }; | |
| var hasImageCaption = function (editor) { | |
| return editor.settings.image_caption === true ? true : false; | |
| }; | |
| var getImageList = function (editor) { | |
| return editor.getParam('image_list', false); | |
| }; | |
| var hasUploadUrl = function (editor) { | |
| return editor.getParam('images_upload_url', false); | |
| }; | |
| var hasUploadHandler = function (editor) { | |
| return editor.getParam('images_upload_handler', false); | |
| }; | |
| var getUploadUrl = function (editor) { | |
| return editor.getParam('images_upload_url'); | |
| }; | |
| var getUploadHandler = function (editor) { | |
| return editor.getParam('images_upload_handler'); | |
| }; | |
| var getUploadBasePath = function (editor) { | |
| return editor.getParam('images_upload_base_path'); | |
| }; | |
| var getUploadCredentials = function (editor) { | |
| return editor.getParam('images_upload_credentials'); | |
| }; | |
| var Settings = { | |
| hasDimensions: hasDimensions, | |
| hasAdvTab: hasAdvTab, | |
| getPrependUrl: getPrependUrl, | |
| getClassList: getClassList, | |
| hasDescription: hasDescription, | |
| hasImageTitle: hasImageTitle, | |
| hasImageCaption: hasImageCaption, | |
| getImageList: getImageList, | |
| hasUploadUrl: hasUploadUrl, | |
| hasUploadHandler: hasUploadHandler, | |
| getUploadUrl: getUploadUrl, | |
| getUploadHandler: getUploadHandler, | |
| getUploadBasePath: getUploadBasePath, | |
| getUploadCredentials: getUploadCredentials | |
| }; | |
| var Global = typeof domGlobals.window !== 'undefined' ? domGlobals.window : Function('return this;')(); | |
| var path = function (parts, scope) { | |
| var o = scope !== undefined && scope !== null ? scope : Global; | |
| for (var i = 0; i < parts.length && o !== undefined && o !== null; ++i) { | |
| o = o[parts[i]]; | |
| } | |
| return o; | |
| }; | |
| var resolve = function (p, scope) { | |
| var parts = p.split('.'); | |
| return path(parts, scope); | |
| }; | |
| var unsafe = function (name, scope) { | |
| return resolve(name, scope); | |
| }; | |
| var getOrDie = function (name, scope) { | |
| var actual = unsafe(name, scope); | |
| if (actual === undefined || actual === null) { | |
| throw new Error(name + ' not available on this browser'); | |
| } | |
| return actual; | |
| }; | |
| var Global$1 = { getOrDie: getOrDie }; | |
| function FileReader () { | |
| var f = Global$1.getOrDie('FileReader'); | |
| return new f(); | |
| } | |
| var global$1 = tinymce.util.Tools.resolve('tinymce.util.Promise'); | |
| var global$2 = tinymce.util.Tools.resolve('tinymce.util.Tools'); | |
| var global$3 = tinymce.util.Tools.resolve('tinymce.util.XHR'); | |
| var parseIntAndGetMax = function (val1, val2) { | |
| return Math.max(parseInt(val1, 10), parseInt(val2, 10)); | |
| }; | |
| var getImageSize = function (url, callback) { | |
| var img = domGlobals.document.createElement('img'); | |
| function done(width, height) { | |
| if (img.parentNode) { | |
| img.parentNode.removeChild(img); | |
| } | |
| callback({ | |
| width: width, | |
| height: height | |
| }); | |
| } | |
| img.onload = function () { | |
| var width = parseIntAndGetMax(img.width, img.clientWidth); | |
| var height = parseIntAndGetMax(img.height, img.clientHeight); | |
| done(width, height); | |
| }; | |
| img.onerror = function () { | |
| done(0, 0); | |
| }; | |
| var style = img.style; | |
| style.visibility = 'hidden'; | |
| style.position = 'fixed'; | |
| style.bottom = style.left = '0px'; | |
| style.width = style.height = 'auto'; | |
| domGlobals.document.body.appendChild(img); | |
| img.src = url; | |
| }; | |
| var buildListItems = function (inputList, itemCallback, startItems) { | |
| function appendItems(values, output) { | |
| output = output || []; | |
| global$2.each(values, function (item) { | |
| var menuItem = { text: item.text || item.title }; | |
| if (item.menu) { | |
| menuItem.menu = appendItems(item.menu); | |
| } else { | |
| menuItem.value = item.value; | |
| itemCallback(menuItem); | |
| } | |
| output.push(menuItem); | |
| }); | |
| return output; | |
| } | |
| return appendItems(inputList, startItems || []); | |
| }; | |
| var removePixelSuffix = function (value) { | |
| if (value) { | |
| value = value.replace(/px$/, ''); | |
| } | |
| return value; | |
| }; | |
| var addPixelSuffix = function (value) { | |
| if (value.length > 0 && /^[0-9]+$/.test(value)) { | |
| value += 'px'; | |
| } | |
| return value; | |
| }; | |
| var mergeMargins = function (css) { | |
| if (css.margin) { | |
| var splitMargin = css.margin.split(' '); | |
| switch (splitMargin.length) { | |
| case 1: | |
| css['margin-top'] = css['margin-top'] || splitMargin[0]; | |
| css['margin-right'] = css['margin-right'] || splitMargin[0]; | |
| css['margin-bottom'] = css['margin-bottom'] || splitMargin[0]; | |
| css['margin-left'] = css['margin-left'] || splitMargin[0]; | |
| break; | |
| case 2: | |
| css['margin-top'] = css['margin-top'] || splitMargin[0]; | |
| css['margin-right'] = css['margin-right'] || splitMargin[1]; | |
| css['margin-bottom'] = css['margin-bottom'] || splitMargin[0]; | |
| css['margin-left'] = css['margin-left'] || splitMargin[1]; | |
| break; | |
| case 3: | |
| css['margin-top'] = css['margin-top'] || splitMargin[0]; | |
| css['margin-right'] = css['margin-right'] || splitMargin[1]; | |
| css['margin-bottom'] = css['margin-bottom'] || splitMargin[2]; | |
| css['margin-left'] = css['margin-left'] || splitMargin[1]; | |
| break; | |
| case 4: | |
| css['margin-top'] = css['margin-top'] || splitMargin[0]; | |
| css['margin-right'] = css['margin-right'] || splitMargin[1]; | |
| css['margin-bottom'] = css['margin-bottom'] || splitMargin[2]; | |
| css['margin-left'] = css['margin-left'] || splitMargin[3]; | |
| } | |
| delete css.margin; | |
| } | |
| return css; | |
| }; | |
| var createImageList = function (editor, callback) { | |
| var imageList = Settings.getImageList(editor); | |
| if (typeof imageList === 'string') { | |
| global$3.send({ | |
| url: imageList, | |
| success: function (text) { | |
| callback(JSON.parse(text)); | |
| } | |
| }); | |
| } else if (typeof imageList === 'function') { | |
| imageList(callback); | |
| } else { | |
| callback(imageList); | |
| } | |
| }; | |
| var waitLoadImage = function (editor, data, imgElm) { | |
| function selectImage() { | |
| imgElm.onload = imgElm.onerror = null; | |
| if (editor.selection) { | |
| editor.selection.select(imgElm); | |
| editor.nodeChanged(); | |
| } | |
| } | |
| imgElm.onload = function () { | |
| if (!data.width && !data.height && Settings.hasDimensions(editor)) { | |
| editor.dom.setAttribs(imgElm, { | |
| width: imgElm.clientWidth, | |
| height: imgElm.clientHeight | |
| }); | |
| } | |
| selectImage(); | |
| }; | |
| imgElm.onerror = selectImage; | |
| }; | |
| var blobToDataUri = function (blob) { | |
| return new global$1(function (resolve, reject) { | |
| var reader = FileReader(); | |
| reader.onload = function () { | |
| resolve(reader.result); | |
| }; | |
| reader.onerror = function () { | |
| reject(reader.error.message); | |
| }; | |
| reader.readAsDataURL(blob); | |
| }); | |
| }; | |
| var Utils = { | |
| getImageSize: getImageSize, | |
| buildListItems: buildListItems, | |
| removePixelSuffix: removePixelSuffix, | |
| addPixelSuffix: addPixelSuffix, | |
| mergeMargins: mergeMargins, | |
| createImageList: createImageList, | |
| waitLoadImage: waitLoadImage, | |
| blobToDataUri: blobToDataUri | |
| }; | |
| var global$4 = tinymce.util.Tools.resolve('tinymce.dom.DOMUtils'); | |
| var hasOwnProperty = Object.prototype.hasOwnProperty; | |
| var shallow = function (old, nu) { | |
| return nu; | |
| }; | |
| var baseMerge = function (merger) { | |
| return function () { | |
| var objects = new Array(arguments.length); | |
| for (var i = 0; i < objects.length; i++) { | |
| objects[i] = arguments[i]; | |
| } | |
| if (objects.length === 0) { | |
| throw new Error('Can\'t merge zero objects'); | |
| } | |
| var ret = {}; | |
| for (var j = 0; j < objects.length; j++) { | |
| var curObject = objects[j]; | |
| for (var key in curObject) { | |
| if (hasOwnProperty.call(curObject, key)) { | |
| ret[key] = merger(ret[key], curObject[key]); | |
| } | |
| } | |
| } | |
| return ret; | |
| }; | |
| }; | |
| var merge = baseMerge(shallow); | |
| var DOM = global$4.DOM; | |
| var getHspace = function (image) { | |
| if (image.style.marginLeft && image.style.marginRight && image.style.marginLeft === image.style.marginRight) { | |
| return Utils.removePixelSuffix(image.style.marginLeft); | |
| } else { | |
| return ''; | |
| } | |
| }; | |
| var getVspace = function (image) { | |
| if (image.style.marginTop && image.style.marginBottom && image.style.marginTop === image.style.marginBottom) { | |
| return Utils.removePixelSuffix(image.style.marginTop); | |
| } else { | |
| return ''; | |
| } | |
| }; | |
| var getBorder = function (image) { | |
| if (image.style.borderWidth) { | |
| return Utils.removePixelSuffix(image.style.borderWidth); | |
| } else { | |
| return ''; | |
| } | |
| }; | |
| var getAttrib = function (image, name) { | |
| if (image.hasAttribute(name)) { | |
| return image.getAttribute(name); | |
| } else { | |
| return ''; | |
| } | |
| }; | |
| var getStyle = function (image, name) { | |
| return image.style[name] ? image.style[name] : ''; | |
| }; | |
| var hasCaption = function (image) { | |
| return image.parentNode !== null && image.parentNode.nodeName === 'FIGURE'; | |
| }; | |
| var setAttrib = function (image, name, value) { | |
| image.setAttribute(name, value); | |
| }; | |
| var wrapInFigure = function (image) { | |
| var figureElm = DOM.create('figure', { class: 'image' }); | |
| DOM.insertAfter(figureElm, image); | |
| figureElm.appendChild(image); | |
| figureElm.appendChild(DOM.create('figcaption', { contentEditable: true }, 'Caption')); | |
| figureElm.contentEditable = 'false'; | |
| }; | |
| var removeFigure = function (image) { | |
| var figureElm = image.parentNode; | |
| DOM.insertAfter(image, figureElm); | |
| DOM.remove(figureElm); | |
| }; | |
| var toggleCaption = function (image) { | |
| if (hasCaption(image)) { | |
| removeFigure(image); | |
| } else { | |
| wrapInFigure(image); | |
| } | |
| }; | |
| var normalizeStyle = function (image, normalizeCss) { | |
| var attrValue = image.getAttribute('style'); | |
| var value = normalizeCss(attrValue !== null ? attrValue : ''); | |
| if (value.length > 0) { | |
| image.setAttribute('style', value); | |
| image.setAttribute('data-mce-style', value); | |
| } else { | |
| image.removeAttribute('style'); | |
| } | |
| }; | |
| var setSize = function (name, normalizeCss) { | |
| return function (image, name, value) { | |
| if (image.style[name]) { | |
| image.style[name] = Utils.addPixelSuffix(value); | |
| normalizeStyle(image, normalizeCss); | |
| } else { | |
| setAttrib(image, name, value); | |
| } | |
| }; | |
| }; | |
| var getSize = function (image, name) { | |
| if (image.style[name]) { | |
| return Utils.removePixelSuffix(image.style[name]); | |
| } else { | |
| return getAttrib(image, name); | |
| } | |
| }; | |
| var setHspace = function (image, value) { | |
| var pxValue = Utils.addPixelSuffix(value); | |
| image.style.marginLeft = pxValue; | |
| image.style.marginRight = pxValue; | |
| }; | |
| var setVspace = function (image, value) { | |
| var pxValue = Utils.addPixelSuffix(value); | |
| image.style.marginTop = pxValue; | |
| image.style.marginBottom = pxValue; | |
| }; | |
| var setBorder = function (image, value) { | |
| var pxValue = Utils.addPixelSuffix(value); | |
| image.style.borderWidth = pxValue; | |
| }; | |
| var setBorderStyle = function (image, value) { | |
| image.style.borderStyle = value; | |
| }; | |
| var getBorderStyle = function (image) { | |
| return getStyle(image, 'borderStyle'); | |
| }; | |
| var isFigure = function (elm) { | |
| return elm.nodeName === 'FIGURE'; | |
| }; | |
| var defaultData = function () { | |
| return { | |
| src: '', | |
| alt: '', | |
| title: '', | |
| width: '', | |
| height: '', | |
| class: '', | |
| style: '', | |
| caption: false, | |
| hspace: '', | |
| vspace: '', | |
| border: '', | |
| borderStyle: '' | |
| }; | |
| }; | |
| var getStyleValue = function (normalizeCss, data) { | |
| var image = domGlobals.document.createElement('img'); | |
| setAttrib(image, 'style', data.style); | |
| if (getHspace(image) || data.hspace !== '') { | |
| setHspace(image, data.hspace); | |
| } | |
| if (getVspace(image) || data.vspace !== '') { | |
| setVspace(image, data.vspace); | |
| } | |
| if (getBorder(image) || data.border !== '') { | |
| setBorder(image, data.border); | |
| } | |
| if (getBorderStyle(image) || data.borderStyle !== '') { | |
| setBorderStyle(image, data.borderStyle); | |
| } | |
| return normalizeCss(image.getAttribute('style')); | |
| }; | |
| var create = function (normalizeCss, data) { | |
| var image = domGlobals.document.createElement('img'); | |
| write(normalizeCss, merge(data, { caption: false }), image); | |
| setAttrib(image, 'alt', data.alt); | |
| if (data.caption) { | |
| var figure = DOM.create('figure', { class: 'image' }); | |
| figure.appendChild(image); | |
| figure.appendChild(DOM.create('figcaption', { contentEditable: true }, 'Caption')); | |
| figure.contentEditable = 'false'; | |
| return figure; | |
| } else { | |
| return image; | |
| } | |
| }; | |
| var read = function (normalizeCss, image) { | |
| return { | |
| src: getAttrib(image, 'src'), | |
| alt: getAttrib(image, 'alt'), | |
| title: getAttrib(image, 'title'), | |
| width: getSize(image, 'width'), | |
| height: getSize(image, 'height'), | |
| class: getAttrib(image, 'class'), | |
| style: normalizeCss(getAttrib(image, 'style')), | |
| caption: hasCaption(image), | |
| hspace: getHspace(image), | |
| vspace: getVspace(image), | |
| border: getBorder(image), | |
| borderStyle: getStyle(image, 'borderStyle') | |
| }; | |
| }; | |
| var updateProp = function (image, oldData, newData, name, set) { | |
| if (newData[name] !== oldData[name]) { | |
| set(image, name, newData[name]); | |
| } | |
| }; | |
| var normalized = function (set, normalizeCss) { | |
| return function (image, name, value) { | |
| set(image, value); | |
| normalizeStyle(image, normalizeCss); | |
| }; | |
| }; | |
| var write = function (normalizeCss, newData, image) { | |
| var oldData = read(normalizeCss, image); | |
| updateProp(image, oldData, newData, 'caption', function (image, _name, _value) { | |
| return toggleCaption(image); | |
| }); | |
| updateProp(image, oldData, newData, 'src', setAttrib); | |
| updateProp(image, oldData, newData, 'alt', setAttrib); | |
| updateProp(image, oldData, newData, 'title', setAttrib); | |
| updateProp(image, oldData, newData, 'width', setSize('width', normalizeCss)); | |
| updateProp(image, oldData, newData, 'height', setSize('height', normalizeCss)); | |
| updateProp(image, oldData, newData, 'class', setAttrib); | |
| updateProp(image, oldData, newData, 'style', normalized(function (image, value) { | |
| return setAttrib(image, 'style', value); | |
| }, normalizeCss)); | |
| updateProp(image, oldData, newData, 'hspace', normalized(setHspace, normalizeCss)); | |
| updateProp(image, oldData, newData, 'vspace', normalized(setVspace, normalizeCss)); | |
| updateProp(image, oldData, newData, 'border', normalized(setBorder, normalizeCss)); | |
| updateProp(image, oldData, newData, 'borderStyle', normalized(setBorderStyle, normalizeCss)); | |
| }; | |
| var normalizeCss = function (editor, cssText) { | |
| var css = editor.dom.styles.parse(cssText); | |
| var mergedCss = Utils.mergeMargins(css); | |
| var compressed = editor.dom.styles.parse(editor.dom.styles.serialize(mergedCss)); | |
| return editor.dom.styles.serialize(compressed); | |
| }; | |
| var getSelectedImage = function (editor) { | |
| var imgElm = editor.selection.getNode(); | |
| var figureElm = editor.dom.getParent(imgElm, 'figure.image'); | |
| if (figureElm) { | |
| return editor.dom.select('img', figureElm)[0]; | |
| } | |
| if (imgElm && (imgElm.nodeName !== 'IMG' || imgElm.getAttribute('data-mce-object') || imgElm.getAttribute('data-mce-placeholder'))) { | |
| return null; | |
| } | |
| return imgElm; | |
| }; | |
| var splitTextBlock = function (editor, figure) { | |
| var dom = editor.dom; | |
| var textBlock = dom.getParent(figure.parentNode, function (node) { | |
| return editor.schema.getTextBlockElements()[node.nodeName]; | |
| }, editor.getBody()); | |
| if (textBlock) { | |
| return dom.split(textBlock, figure); | |
| } else { | |
| return figure; | |
| } | |
| }; | |
| var readImageDataFromSelection = function (editor) { | |
| var image = getSelectedImage(editor); | |
| return image ? read(function (css) { | |
| return normalizeCss(editor, css); | |
| }, image) : defaultData(); | |
| }; | |
| var insertImageAtCaret = function (editor, data) { | |
| var elm = create(function (css) { | |
| return normalizeCss(editor, css); | |
| }, data); | |
| editor.dom.setAttrib(elm, 'data-mce-id', '__mcenew'); | |
| editor.focus(); | |
| editor.selection.setContent(elm.outerHTML); | |
| var insertedElm = editor.dom.select('*[data-mce-id="__mcenew"]')[0]; | |
| editor.dom.setAttrib(insertedElm, 'data-mce-id', null); | |
| if (isFigure(insertedElm)) { | |
| var figure = splitTextBlock(editor, insertedElm); | |
| editor.selection.select(figure); | |
| } else { | |
| editor.selection.select(insertedElm); | |
| } | |
| }; | |
| var syncSrcAttr = function (editor, image) { | |
| editor.dom.setAttrib(image, 'src', image.getAttribute('src')); | |
| }; | |
| var deleteImage = function (editor, image) { | |
| if (image) { | |
| var elm = editor.dom.is(image.parentNode, 'figure.image') ? image.parentNode : image; | |
| editor.dom.remove(elm); | |
| editor.focus(); | |
| editor.nodeChanged(); | |
| if (editor.dom.isEmpty(editor.getBody())) { | |
| editor.setContent(''); | |
| editor.selection.setCursorLocation(); | |
| } | |
| } | |
| }; | |
| var writeImageDataToSelection = function (editor, data) { | |
| var image = getSelectedImage(editor); | |
| write(function (css) { | |
| return normalizeCss(editor, css); | |
| }, data, image); | |
| syncSrcAttr(editor, image); | |
| if (isFigure(image.parentNode)) { | |
| var figure = image.parentNode; | |
| splitTextBlock(editor, figure); | |
| editor.selection.select(image.parentNode); | |
| } else { | |
| editor.selection.select(image); | |
| Utils.waitLoadImage(editor, data, image); | |
| } | |
| }; | |
| var insertOrUpdateImage = function (editor, data) { | |
| var image = getSelectedImage(editor); | |
| if (image) { | |
| if (data.src) { | |
| writeImageDataToSelection(editor, data); | |
| } else { | |
| deleteImage(editor, image); | |
| } | |
| } else if (data.src) { | |
| insertImageAtCaret(editor, data); | |
| } | |
| }; | |
| var updateVSpaceHSpaceBorder = function (editor) { | |
| return function (evt) { | |
| var dom = editor.dom; | |
| var rootControl = evt.control.rootControl; | |
| if (!Settings.hasAdvTab(editor)) { | |
| return; | |
| } | |
| var data = rootControl.toJSON(); | |
| var css = dom.parseStyle(data.style); | |
| rootControl.find('#vspace').value(''); | |
| rootControl.find('#hspace').value(''); | |
| css = Utils.mergeMargins(css); | |
| if (css['margin-top'] && css['margin-bottom'] || css['margin-right'] && css['margin-left']) { | |
| if (css['margin-top'] === css['margin-bottom']) { | |
| rootControl.find('#vspace').value(Utils.removePixelSuffix(css['margin-top'])); | |
| } else { | |
| rootControl.find('#vspace').value(''); | |
| } | |
| if (css['margin-right'] === css['margin-left']) { | |
| rootControl.find('#hspace').value(Utils.removePixelSuffix(css['margin-right'])); | |
| } else { | |
| rootControl.find('#hspace').value(''); | |
| } | |
| } | |
| if (css['border-width']) { | |
| rootControl.find('#border').value(Utils.removePixelSuffix(css['border-width'])); | |
| } else { | |
| rootControl.find('#border').value(''); | |
| } | |
| if (css['border-style']) { | |
| rootControl.find('#borderStyle').value(css['border-style']); | |
| } else { | |
| rootControl.find('#borderStyle').value(''); | |
| } | |
| rootControl.find('#style').value(dom.serializeStyle(dom.parseStyle(dom.serializeStyle(css)))); | |
| }; | |
| }; | |
| var updateStyle = function (editor, win) { | |
| win.find('#style').each(function (ctrl) { | |
| var value = getStyleValue(function (css) { | |
| return normalizeCss(editor, css); | |
| }, merge(defaultData(), win.toJSON())); | |
| ctrl.value(value); | |
| }); | |
| }; | |
| var makeTab = function (editor) { | |
| return { | |
| title: 'Advanced', | |
| type: 'form', | |
| pack: 'start', | |
| items: [ | |
| { | |
| label: 'Style', | |
| name: 'style', | |
| type: 'textbox', | |
| onchange: updateVSpaceHSpaceBorder(editor) | |
| }, | |
| { | |
| type: 'form', | |
| layout: 'grid', | |
| packV: 'start', | |
| columns: 2, | |
| padding: 0, | |
| defaults: { | |
| type: 'textbox', | |
| maxWidth: 50, | |
| onchange: function (evt) { | |
| updateStyle(editor, evt.control.rootControl); | |
| } | |
| }, | |
| items: [ | |
| { | |
| label: 'Vertical space', | |
| name: 'vspace' | |
| }, | |
| { | |
| label: 'Border width', | |
| name: 'border' | |
| }, | |
| { | |
| label: 'Horizontal space', | |
| name: 'hspace' | |
| }, | |
| { | |
| label: 'Border style', | |
| type: 'listbox', | |
| name: 'borderStyle', | |
| width: 90, | |
| maxWidth: 90, | |
| onselect: function (evt) { | |
| updateStyle(editor, evt.control.rootControl); | |
| }, | |
| values: [ | |
| { | |
| text: 'Select...', | |
| value: '' | |
| }, | |
| { | |
| text: 'Solid', | |
| value: 'solid' | |
| }, | |
| { | |
| text: 'Dotted', | |
| value: 'dotted' | |
| }, | |
| { | |
| text: 'Dashed', | |
| value: 'dashed' | |
| }, | |
| { | |
| text: 'Double', | |
| value: 'double' | |
| }, | |
| { | |
| text: 'Groove', | |
| value: 'groove' | |
| }, | |
| { | |
| text: 'Ridge', | |
| value: 'ridge' | |
| }, | |
| { | |
| text: 'Inset', | |
| value: 'inset' | |
| }, | |
| { | |
| text: 'Outset', | |
| value: 'outset' | |
| }, | |
| { | |
| text: 'None', | |
| value: 'none' | |
| }, | |
| { | |
| text: 'Hidden', | |
| value: 'hidden' | |
| } | |
| ] | |
| } | |
| ] | |
| } | |
| ] | |
| }; | |
| }; | |
| var AdvTab = { makeTab: makeTab }; | |
| var doSyncSize = function (widthCtrl, heightCtrl) { | |
| widthCtrl.state.set('oldVal', widthCtrl.value()); | |
| heightCtrl.state.set('oldVal', heightCtrl.value()); | |
| }; | |
| var doSizeControls = function (win, f) { | |
| var widthCtrl = win.find('#width')[0]; | |
| var heightCtrl = win.find('#height')[0]; | |
| var constrained = win.find('#constrain')[0]; | |
| if (widthCtrl && heightCtrl && constrained) { | |
| f(widthCtrl, heightCtrl, constrained.checked()); | |
| } | |
| }; | |
| var doUpdateSize = function (widthCtrl, heightCtrl, isContrained) { | |
| var oldWidth = widthCtrl.state.get('oldVal'); | |
| var oldHeight = heightCtrl.state.get('oldVal'); | |
| var newWidth = widthCtrl.value(); | |
| var newHeight = heightCtrl.value(); | |
| if (isContrained && oldWidth && oldHeight && newWidth && newHeight) { | |
| if (newWidth !== oldWidth) { | |
| newHeight = Math.round(newWidth / oldWidth * newHeight); | |
| if (!isNaN(newHeight)) { | |
| heightCtrl.value(newHeight); | |
| } | |
| } else { | |
| newWidth = Math.round(newHeight / oldHeight * newWidth); | |
| if (!isNaN(newWidth)) { | |
| widthCtrl.value(newWidth); | |
| } | |
| } | |
| } | |
| doSyncSize(widthCtrl, heightCtrl); | |
| }; | |
| var syncSize = function (win) { | |
| doSizeControls(win, doSyncSize); | |
| }; | |
| var updateSize = function (win) { | |
| doSizeControls(win, doUpdateSize); | |
| }; | |
| var createUi = function () { | |
| var recalcSize = function (evt) { | |
| updateSize(evt.control.rootControl); | |
| }; | |
| return { | |
| type: 'container', | |
| label: 'Dimensions', | |
| layout: 'flex', | |
| align: 'center', | |
| spacing: 5, | |
| items: [ | |
| { | |
| name: 'width', | |
| type: 'textbox', | |
| maxLength: 5, | |
| size: 5, | |
| onchange: recalcSize, | |
| ariaLabel: 'Width' | |
| }, | |
| { | |
| type: 'label', | |
| text: 'x' | |
| }, | |
| { | |
| name: 'height', | |
| type: 'textbox', | |
| maxLength: 5, | |
| size: 5, | |
| onchange: recalcSize, | |
| ariaLabel: 'Height' | |
| }, | |
| { | |
| name: 'constrain', | |
| type: 'checkbox', | |
| checked: true, | |
| text: 'Constrain proportions' | |
| } | |
| ] | |
| }; | |
| }; | |
| var SizeManager = { | |
| createUi: createUi, | |
| syncSize: syncSize, | |
| updateSize: updateSize | |
| }; | |
| var onSrcChange = function (evt, editor) { | |
| var srcURL, prependURL, absoluteURLPattern; | |
| var meta = evt.meta || {}; | |
| var control = evt.control; | |
| var rootControl = control.rootControl; | |
| var imageListCtrl = rootControl.find('#image-list')[0]; | |
| if (imageListCtrl) { | |
| imageListCtrl.value(editor.convertURL(control.value(), 'src')); | |
| } | |
| global$2.each(meta, function (value, key) { | |
| rootControl.find('#' + key).value(value); | |
| }); | |
| if (!meta.width && !meta.height) { | |
| srcURL = editor.convertURL(control.value(), 'src'); | |
| prependURL = Settings.getPrependUrl(editor); | |
| absoluteURLPattern = new RegExp('^(?:[a-z]+:)?//', 'i'); | |
| if (prependURL && !absoluteURLPattern.test(srcURL) && srcURL.substring(0, prependURL.length) !== prependURL) { | |
| srcURL = prependURL + srcURL; | |
| } | |
| control.value(srcURL); | |
| Utils.getImageSize(editor.documentBaseURI.toAbsolute(control.value()), function (data) { | |
| if (data.width && data.height && Settings.hasDimensions(editor)) { | |
| rootControl.find('#width').value(data.width); | |
| rootControl.find('#height').value(data.height); | |
| SizeManager.syncSize(rootControl); | |
| } | |
| }); | |
| } | |
| }; | |
| var onBeforeCall = function (evt) { | |
| evt.meta = evt.control.rootControl.toJSON(); | |
| }; | |
| var getGeneralItems = function (editor, imageListCtrl) { | |
| var generalFormItems = [ | |
| { | |
| name: 'src', | |
| type: 'filepicker', | |
| filetype: 'image', | |
| label: 'Source', | |
| autofocus: true, | |
| onchange: function (evt) { | |
| onSrcChange(evt, editor); | |
| }, | |
| onbeforecall: onBeforeCall | |
| }, | |
| imageListCtrl | |
| ]; | |
| if (Settings.hasDescription(editor)) { | |
| generalFormItems.push({ | |
| name: 'alt', | |
| type: 'textbox', | |
| label: 'Image description' | |
| }); | |
| } | |
| if (Settings.hasImageTitle(editor)) { | |
| generalFormItems.push({ | |
| name: 'title', | |
| type: 'textbox', | |
| label: 'Image Title' | |
| }); | |
| } | |
| if (Settings.hasDimensions(editor)) { | |
| generalFormItems.push(SizeManager.createUi()); | |
| } | |
| if (Settings.getClassList(editor)) { | |
| generalFormItems.push({ | |
| name: 'class', | |
| type: 'listbox', | |
| label: 'Class', | |
| values: Utils.buildListItems(Settings.getClassList(editor), function (item) { | |
| if (item.value) { | |
| item.textStyle = function () { | |
| return editor.formatter.getCssText({ | |
| inline: 'img', | |
| classes: [item.value] | |
| }); | |
| }; | |
| } | |
| }) | |
| }); | |
| } | |
| if (Settings.hasImageCaption(editor)) { | |
| generalFormItems.push({ | |
| name: 'caption', | |
| type: 'checkbox', | |
| label: 'Caption' | |
| }); | |
| } | |
| return generalFormItems; | |
| }; | |
| var makeTab$1 = function (editor, imageListCtrl) { | |
| return { | |
| title: 'General', | |
| type: 'form', | |
| items: getGeneralItems(editor, imageListCtrl) | |
| }; | |
| }; | |
| var MainTab = { | |
| makeTab: makeTab$1, | |
| getGeneralItems: getGeneralItems | |
| }; | |
| var url = function () { | |
| return Global$1.getOrDie('URL'); | |
| }; | |
| var createObjectURL = function (blob) { | |
| return url().createObjectURL(blob); | |
| }; | |
| var revokeObjectURL = function (u) { | |
| url().revokeObjectURL(u); | |
| }; | |
| var URL = { | |
| createObjectURL: createObjectURL, | |
| revokeObjectURL: revokeObjectURL | |
| }; | |
| var global$5 = tinymce.util.Tools.resolve('tinymce.ui.Factory'); | |
| function XMLHttpRequest () { | |
| var f = Global$1.getOrDie('XMLHttpRequest'); | |
| return new f(); | |
| } | |
| var noop = function () { | |
| }; | |
| var pathJoin = function (path1, path2) { | |
| if (path1) { | |
| return path1.replace(/\/$/, '') + '/' + path2.replace(/^\//, ''); | |
| } | |
| return path2; | |
| }; | |
| function Uploader (settings) { | |
| var defaultHandler = function (blobInfo, success, failure, progress) { | |
| var xhr, formData; | |
| xhr = XMLHttpRequest(); | |
| xhr.open('POST', settings.url); | |
| xhr.withCredentials = settings.credentials; | |
| xhr.upload.onprogress = function (e) { | |
| progress(e.loaded / e.total * 100); | |
| }; | |
| xhr.onerror = function () { | |
| failure('Image upload failed due to a XHR Transport error. Code: ' + xhr.status); | |
| }; | |
| xhr.onload = function () { | |
| var json; | |
| if (xhr.status < 200 || xhr.status >= 300) { | |
| failure('HTTP Error: ' + xhr.status); | |
| return; | |
| } | |
| json = JSON.parse(xhr.responseText); | |
| if (!json || typeof json.location !== 'string') { | |
| failure('Invalid JSON: ' + xhr.responseText); | |
| return; | |
| } | |
| success(pathJoin(settings.basePath, json.location)); | |
| }; | |
| formData = new domGlobals.FormData(); | |
| formData.append('file', blobInfo.blob(), blobInfo.filename()); | |
| xhr.send(formData); | |
| }; | |
| var uploadBlob = function (blobInfo, handler) { | |
| return new global$1(function (resolve, reject) { | |
| try { | |
| handler(blobInfo, resolve, reject, noop); | |
| } catch (ex) { | |
| reject(ex.message); | |
| } | |
| }); | |
| }; | |
| var isDefaultHandler = function (handler) { | |
| return handler === defaultHandler; | |
| }; | |
| var upload = function (blobInfo) { | |
| return !settings.url && isDefaultHandler(settings.handler) ? global$1.reject('Upload url missing from the settings.') : uploadBlob(blobInfo, settings.handler); | |
| }; | |
| settings = global$2.extend({ | |
| credentials: false, | |
| handler: defaultHandler | |
| }, settings); | |
| return { upload: upload }; | |
| } | |
| var onFileInput = function (editor) { | |
| return function (evt) { | |
| var Throbber = global$5.get('Throbber'); | |
| var rootControl = evt.control.rootControl; | |
| var throbber = new Throbber(rootControl.getEl()); | |
| var file = evt.control.value(); | |
| var blobUri = URL.createObjectURL(file); | |
| var uploader = Uploader({ | |
| url: Settings.getUploadUrl(editor), | |
| basePath: Settings.getUploadBasePath(editor), | |
| credentials: Settings.getUploadCredentials(editor), | |
| handler: Settings.getUploadHandler(editor) | |
| }); | |
| var finalize = function () { | |
| throbber.hide(); | |
| URL.revokeObjectURL(blobUri); | |
| }; | |
| throbber.show(); | |
| return Utils.blobToDataUri(file).then(function (dataUrl) { | |
| var blobInfo = editor.editorUpload.blobCache.create({ | |
| blob: file, | |
| blobUri: blobUri, | |
| name: file.name ? file.name.replace(/\.[^\.]+$/, '') : null, | |
| base64: dataUrl.split(',')[1] | |
| }); | |
| return uploader.upload(blobInfo).then(function (url) { | |
| var src = rootControl.find('#src'); | |
| src.value(url); | |
| rootControl.find('tabpanel')[0].activateTab(0); | |
| src.fire('change'); | |
| finalize(); | |
| return url; | |
| }); | |
| }).catch(function (err) { | |
| editor.windowManager.alert(err); | |
| finalize(); | |
| }); | |
| }; | |
| }; | |
| var acceptExts = '.jpg,.jpeg,.png,.gif'; | |
| var makeTab$2 = function (editor) { | |
| return { | |
| title: 'Upload', | |
| type: 'form', | |
| layout: 'flex', | |
| direction: 'column', | |
| align: 'stretch', | |
| padding: '20 20 20 20', | |
| items: [ | |
| { | |
| type: 'container', | |
| layout: 'flex', | |
| direction: 'column', | |
| align: 'center', | |
| spacing: 10, | |
| items: [ | |
| { | |
| text: 'Browse for an image', | |
| type: 'browsebutton', | |
| accept: acceptExts, | |
| onchange: onFileInput(editor) | |
| }, | |
| { | |
| text: 'OR', | |
| type: 'label' | |
| } | |
| ] | |
| }, | |
| { | |
| text: 'Drop an image here', | |
| type: 'dropzone', | |
| accept: acceptExts, | |
| height: 100, | |
| onchange: onFileInput(editor) | |
| } | |
| ] | |
| }; | |
| }; | |
| var UploadTab = { makeTab: makeTab$2 }; | |
| function curry(fn) { | |
| var initialArgs = []; | |
| for (var _i = 1; _i < arguments.length; _i++) { | |
| initialArgs[_i - 1] = arguments[_i]; | |
| } | |
| return function () { | |
| var restArgs = []; | |
| for (var _i = 0; _i < arguments.length; _i++) { | |
| restArgs[_i] = arguments[_i]; | |
| } | |
| var all = initialArgs.concat(restArgs); | |
| return fn.apply(null, all); | |
| }; | |
| } | |
| var submitForm = function (editor, evt) { | |
| var win = evt.control.getRoot(); | |
| SizeManager.updateSize(win); | |
| editor.undoManager.transact(function () { | |
| var data = merge(readImageDataFromSelection(editor), win.toJSON()); | |
| insertOrUpdateImage(editor, data); | |
| }); | |
| editor.editorUpload.uploadImagesAuto(); | |
| }; | |
| function Dialog (editor) { | |
| function showDialog(imageList) { | |
| var data = readImageDataFromSelection(editor); | |
| var win, imageListCtrl; | |
| if (imageList) { | |
| imageListCtrl = { | |
| type: 'listbox', | |
| label: 'Image list', | |
| name: 'image-list', | |
| values: Utils.buildListItems(imageList, function (item) { | |
| item.value = editor.convertURL(item.value || item.url, 'src'); | |
| }, [{ | |
| text: 'None', | |
| value: '' | |
| }]), | |
| value: data.src && editor.convertURL(data.src, 'src'), | |
| onselect: function (e) { | |
| var altCtrl = win.find('#alt'); | |
| if (!altCtrl.value() || e.lastControl && altCtrl.value() === e.lastControl.text()) { | |
| altCtrl.value(e.control.text()); | |
| } | |
| win.find('#src').value(e.control.value()).fire('change'); | |
| }, | |
| onPostRender: function () { | |
| imageListCtrl = this; | |
| } | |
| }; | |
| } | |
| if (Settings.hasAdvTab(editor) || Settings.hasUploadUrl(editor) || Settings.hasUploadHandler(editor)) { | |
| var body = [MainTab.makeTab(editor, imageListCtrl)]; | |
| if (Settings.hasAdvTab(editor)) { | |
| body.push(AdvTab.makeTab(editor)); | |
| } | |
| if (Settings.hasUploadUrl(editor) || Settings.hasUploadHandler(editor)) { | |
| body.push(UploadTab.makeTab(editor)); | |
| } | |
| win = editor.windowManager.open({ | |
| title: 'Insert/edit image', | |
| data: data, | |
| bodyType: 'tabpanel', | |
| body: body, | |
| onSubmit: curry(submitForm, editor) | |
| }); | |
| } else { | |
| win = editor.windowManager.open({ | |
| title: 'Insert/edit image', | |
| data: data, | |
| body: MainTab.getGeneralItems(editor, imageListCtrl), | |
| onSubmit: curry(submitForm, editor) | |
| }); | |
| } | |
| SizeManager.syncSize(win); | |
| } | |
| function open() { | |
| Utils.createImageList(editor, showDialog); | |
| } | |
| return { open: open }; | |
| } | |
| var register = function (editor) { | |
| editor.addCommand('mceImage', Dialog(editor).open); | |
| }; | |
| var Commands = { register: register }; | |
| var hasImageClass = function (node) { | |
| var className = node.attr('class'); | |
| return className && /\bimage\b/.test(className); | |
| }; | |
| var toggleContentEditableState = function (state) { | |
| return function (nodes) { | |
| var i = nodes.length, node; | |
| var toggleContentEditable = function (node) { | |
| node.attr('contenteditable', state ? 'true' : null); | |
| }; | |
| while (i--) { | |
| node = nodes[i]; | |
| if (hasImageClass(node)) { | |
| node.attr('contenteditable', state ? 'false' : null); | |
| global$2.each(node.getAll('figcaption'), toggleContentEditable); | |
| } | |
| } | |
| }; | |
| }; | |
| var setup = function (editor) { | |
| editor.on('preInit', function () { | |
| editor.parser.addNodeFilter('figure', toggleContentEditableState(true)); | |
| editor.serializer.addNodeFilter('figure', toggleContentEditableState(false)); | |
| }); | |
| }; | |
| var FilterContent = { setup: setup }; | |
| var register$1 = function (editor) { | |
| editor.addButton('image', { | |
| icon: 'image', | |
| tooltip: 'Insert/edit image', | |
| onclick: Dialog(editor).open, | |
| stateSelector: 'img:not([data-mce-object],[data-mce-placeholder]),figure.image' | |
| }); | |
| editor.addMenuItem('image', { | |
| icon: 'image', | |
| text: 'Image', | |
| onclick: Dialog(editor).open, | |
| context: 'insert', | |
| prependToContext: true | |
| }); | |
| }; | |
| var Buttons = { register: register$1 }; | |
| global.add('image', function (editor) { | |
| FilterContent.setup(editor); | |
| Buttons.register(editor); | |
| Commands.register(editor); | |
| }); | |
| function Plugin () { | |
| } | |
| return Plugin; | |
| }(window)); | |
| })(); | |