mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
ef10672364
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@92488 467b73ca-7a2a-4603-9d3b-597d59a354a9
413 lines
12 KiB
JavaScript
413 lines
12 KiB
JavaScript
/**
|
|
* $Id: Popup.js 1075 2009-04-03 19:41:51Z spocke $
|
|
*
|
|
* @author Moxiecode
|
|
* @copyright Copyright © 2004-2008, Moxiecode Systems AB, All rights reserved.
|
|
*/
|
|
|
|
// Some global instances
|
|
var tinymce = null, tinyMCEPopup, tinyMCE;
|
|
|
|
/**#@+
|
|
* @class TinyMCE popup/dialog helper class. This gives you easy access to the
|
|
* parent editor instance and a bunch of other things. It's higly recommended
|
|
* that you load this script into your dialogs.
|
|
*
|
|
* @static
|
|
* @member tinyMCEPopup
|
|
*/
|
|
tinyMCEPopup = {
|
|
/**#@+
|
|
* @method
|
|
*/
|
|
|
|
/**
|
|
* Initializes the popup this will be called automatically.
|
|
*/
|
|
init : function() {
|
|
var t = this, w, ti;
|
|
|
|
// Find window & API
|
|
w = t.getWin();
|
|
tinymce = w.tinymce;
|
|
tinyMCE = w.tinyMCE;
|
|
t.editor = tinymce.EditorManager.activeEditor;
|
|
t.params = t.editor.windowManager.params;
|
|
t.features = t.editor.windowManager.features;
|
|
|
|
// Setup local DOM
|
|
t.dom = t.editor.windowManager.createInstance('tinymce.dom.DOMUtils', document);
|
|
|
|
// Enables you to skip loading the default css
|
|
if (t.features.popup_css !== false)
|
|
t.dom.loadCSS(t.features.popup_css || t.editor.settings.popup_css);
|
|
|
|
// Setup on init listeners
|
|
t.listeners = [];
|
|
t.onInit = {
|
|
add : function(f, s) {
|
|
t.listeners.push({func : f, scope : s});
|
|
}
|
|
};
|
|
|
|
t.isWindow = !t.getWindowArg('mce_inline');
|
|
t.id = t.getWindowArg('mce_window_id');
|
|
t.editor.windowManager.onOpen.dispatch(t.editor.windowManager, window);
|
|
},
|
|
|
|
/**
|
|
* Returns the reference to the parent window that opened the dialog.
|
|
*
|
|
* @return {Window} Reference to the parent window that opened the dialog.
|
|
*/
|
|
getWin : function() {
|
|
return window.dialogArguments || opener || parent || top;
|
|
},
|
|
|
|
/**
|
|
* Returns a window argument/parameter by name.
|
|
*
|
|
* @param {String} n Name of the window argument to retrive.
|
|
* @param {String} dv Optional default value to return.
|
|
* @return {String} Argument value or default value if it wasn't found.
|
|
*/
|
|
getWindowArg : function(n, dv) {
|
|
var v = this.params[n];
|
|
|
|
return tinymce.is(v) ? v : dv;
|
|
},
|
|
|
|
/**
|
|
* Returns a editor parameter/config option value.
|
|
*
|
|
* @param {String} n Name of the editor config option to retrive.
|
|
* @param {String} dv Optional default value to return.
|
|
* @return {String} Parameter value or default value if it wasn't found.
|
|
*/
|
|
getParam : function(n, dv) {
|
|
return this.editor.getParam(n, dv);
|
|
},
|
|
|
|
/**
|
|
* Returns a language item by key.
|
|
*
|
|
* @param {String} n Language item like mydialog.something.
|
|
* @param {String} dv Optional default value to return.
|
|
* @return {String} Language value for the item like "my string" or the default value if it wasn't found.
|
|
*/
|
|
getLang : function(n, dv) {
|
|
return this.editor.getLang(n, dv);
|
|
},
|
|
|
|
/**
|
|
* Executed a command on editor that opened the dialog/popup.
|
|
*
|
|
* @param {String} cmd Command to execute.
|
|
* @param {bool} ui Optional boolean value if the UI for the command should be presented or not.
|
|
* @param {Object} val Optional value to pass with the comman like an URL.
|
|
* @param {Object} a Optional arguments object.
|
|
*/
|
|
execCommand : function(cmd, ui, val, a) {
|
|
a = a || {};
|
|
a.skip_focus = 1;
|
|
|
|
this.restoreSelection();
|
|
return this.editor.execCommand(cmd, ui, val, a);
|
|
},
|
|
|
|
/**
|
|
* Resizes the dialog to the inner size of the window. This is needed since various browsers
|
|
* have different border sizes on windows.
|
|
*/
|
|
resizeToInnerSize : function() {
|
|
var t = this, n, b = document.body, vp = t.dom.getViewPort(window), dw, dh;
|
|
|
|
dw = t.getWindowArg('mce_width') - vp.w;
|
|
dh = t.getWindowArg('mce_height') - vp.h;
|
|
|
|
if (t.isWindow)
|
|
window.resizeBy(dw, dh);
|
|
else
|
|
t.editor.windowManager.resizeBy(dw, dh, t.id);
|
|
},
|
|
|
|
/**
|
|
* Will executed the specified string when the page has been loaded. This function
|
|
* was added for compatibility with the 2.x branch.
|
|
*
|
|
* @param {String} s String to evalutate on init.
|
|
*/
|
|
executeOnLoad : function(s) {
|
|
this.onInit.add(function() {
|
|
eval(s);
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Stores the current editor selection for later restoration. This can be useful since some browsers
|
|
* looses it's selection if a control element is selected/focused inside the dialogs.
|
|
*/
|
|
storeSelection : function() {
|
|
this.editor.windowManager.bookmark = tinyMCEPopup.editor.selection.getBookmark('simple');
|
|
},
|
|
|
|
/**
|
|
* Restores any stored selection. This can be useful since some browsers
|
|
* looses it's selection if a control element is selected/focused inside the dialogs.
|
|
*/
|
|
restoreSelection : function() {
|
|
var t = tinyMCEPopup;
|
|
|
|
if (!t.isWindow && tinymce.isIE)
|
|
t.editor.selection.moveToBookmark(t.editor.windowManager.bookmark);
|
|
},
|
|
|
|
/**
|
|
* Loads a specific dialog language pack. If you pass in plugin_url as a arugment
|
|
* when you open the window it will load the <plugin url>/langs/<code>_dlg.js lang pack file.
|
|
*/
|
|
requireLangPack : function() {
|
|
var t = this, u = t.getWindowArg('plugin_url') || t.getWindowArg('theme_url');
|
|
|
|
if (u && t.editor.settings.language && t.features.translate_i18n !== false) {
|
|
u += '/langs/' + t.editor.settings.language + '_dlg.js';
|
|
|
|
if (!tinymce.ScriptLoader.isDone(u)) {
|
|
document.write('<script type="text/javascript" src="' + tinymce._addVer(u) + '"></script>');
|
|
tinymce.ScriptLoader.markDone(u);
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Executes a color picker on the specified element id. When the user
|
|
* then selects a color it will be set as the value of the specified element.
|
|
*
|
|
* @param {DOMEvent} e DOM event object.
|
|
* @param {string} element_id Element id to be filled with the color value from the picker.
|
|
*/
|
|
pickColor : function(e, element_id) {
|
|
this.execCommand('mceColorPicker', true, {
|
|
color : document.getElementById(element_id).value,
|
|
func : function(c) {
|
|
document.getElementById(element_id).value = c;
|
|
|
|
try {
|
|
document.getElementById(element_id).onchange();
|
|
} catch (ex) {
|
|
// Try fire event, ignore errors
|
|
}
|
|
}
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Opens a filebrowser/imagebrowser this will set the output value from
|
|
* the browser as a value on the specified element.
|
|
*
|
|
* @param {string} element_id Id of the element to set value in.
|
|
* @param {string} type Type of browser to open image/file/flash.
|
|
* @param {string} option Option name to get the file_broswer_callback function name from.
|
|
*/
|
|
openBrowser : function(element_id, type, option) {
|
|
tinyMCEPopup.restoreSelection();
|
|
this.editor.execCallback('file_browser_callback', element_id, document.getElementById(element_id).value, type, window);
|
|
},
|
|
|
|
/**
|
|
* Creates a confirm dialog. Please don't use the blocking behavior of this
|
|
* native version use the callback method instead then it can be extended.
|
|
*
|
|
* @param {String} t Title for the new confirm dialog.
|
|
* @param {function} cb Callback function to be executed after the user has selected ok or cancel.
|
|
* @param {Object} s Optional scope to execute the callback in.
|
|
*/
|
|
confirm : function(t, cb, s) {
|
|
this.editor.windowManager.confirm(t, cb, s, window);
|
|
},
|
|
|
|
/**
|
|
* Creates a alert dialog. Please don't use the blocking behavior of this
|
|
* native version use the callback method instead then it can be extended.
|
|
*
|
|
* @param {String} t Title for the new alert dialog.
|
|
* @param {function} cb Callback function to be executed after the user has selected ok.
|
|
* @param {Object} s Optional scope to execute the callback in.
|
|
*/
|
|
alert : function(tx, cb, s) {
|
|
this.editor.windowManager.alert(tx, cb, s, window);
|
|
},
|
|
|
|
/**
|
|
* Closes the current window.
|
|
*/
|
|
close : function() {
|
|
var t = this;
|
|
|
|
// To avoid domain relaxing issue in Opera
|
|
function close() {
|
|
t.editor.windowManager.close(window);
|
|
tinymce = tinyMCE = t.editor = t.params = t.dom = t.dom.doc = null; // Cleanup
|
|
};
|
|
|
|
if (tinymce.isOpera)
|
|
t.getWin().setTimeout(close, 0);
|
|
else
|
|
close();
|
|
},
|
|
|
|
// Internal functions
|
|
|
|
_restoreSelection : function() {
|
|
var e = window.event.srcElement;
|
|
|
|
if (e.nodeName == 'INPUT' && (e.type == 'submit' || e.type == 'button'))
|
|
tinyMCEPopup.restoreSelection();
|
|
},
|
|
|
|
/* _restoreSelection : function() {
|
|
var e = window.event.srcElement;
|
|
|
|
// If user focus a non text input or textarea
|
|
if ((e.nodeName != 'INPUT' && e.nodeName != 'TEXTAREA') || e.type != 'text')
|
|
tinyMCEPopup.restoreSelection();
|
|
},*/
|
|
|
|
_onDOMLoaded : function() {
|
|
var t = tinyMCEPopup, ti = document.title, bm, h, nv;
|
|
|
|
if (t.domLoaded)
|
|
return;
|
|
|
|
t.domLoaded = 1;
|
|
|
|
// Translate page
|
|
if (t.features.translate_i18n !== false) {
|
|
h = document.body.innerHTML;
|
|
|
|
// Replace a=x with a="x" in IE
|
|
if (tinymce.isIE)
|
|
h = h.replace(/ (value|title|alt)=([^"][^\s>]+)/gi, ' $1="$2"')
|
|
|
|
document.dir = t.editor.getParam('directionality','');
|
|
|
|
if ((nv = t.editor.translate(h)) && nv != h)
|
|
document.body.innerHTML = nv;
|
|
|
|
if ((nv = t.editor.translate(ti)) && nv != ti)
|
|
document.title = ti = nv;
|
|
}
|
|
|
|
document.body.style.display = '';
|
|
|
|
// Restore selection in IE when focus is placed on a non textarea or input element of the type text
|
|
if (tinymce.isIE) {
|
|
document.attachEvent('onmouseup', tinyMCEPopup._restoreSelection);
|
|
|
|
// Add base target element for it since it would fail with modal dialogs
|
|
t.dom.add(t.dom.select('head')[0], 'base', {target : '_self'});
|
|
}
|
|
|
|
t.restoreSelection();
|
|
t.resizeToInnerSize();
|
|
|
|
// Set inline title
|
|
if (!t.isWindow)
|
|
t.editor.windowManager.setTitle(window, ti);
|
|
else
|
|
window.focus();
|
|
|
|
if (!tinymce.isIE && !t.isWindow) {
|
|
tinymce.dom.Event._add(document, 'focus', function() {
|
|
t.editor.windowManager.focus(t.id)
|
|
});
|
|
}
|
|
|
|
// Patch for accessibility
|
|
tinymce.each(t.dom.select('select'), function(e) {
|
|
e.onkeydown = tinyMCEPopup._accessHandler;
|
|
});
|
|
|
|
// Call onInit
|
|
// Init must be called before focus so the selection won't get lost by the focus call
|
|
tinymce.each(t.listeners, function(o) {
|
|
o.func.call(o.scope, t.editor);
|
|
});
|
|
|
|
// Move focus to window
|
|
if (t.getWindowArg('mce_auto_focus', true)) {
|
|
window.focus();
|
|
|
|
// Focus element with mceFocus class
|
|
tinymce.each(document.forms, function(f) {
|
|
tinymce.each(f.elements, function(e) {
|
|
if (t.dom.hasClass(e, 'mceFocus') && !e.disabled) {
|
|
e.focus();
|
|
return false; // Break loop
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
document.onkeyup = tinyMCEPopup._closeWinKeyHandler;
|
|
},
|
|
|
|
_accessHandler : function(e) {
|
|
e = e || window.event;
|
|
|
|
if (e.keyCode == 13 || e.keyCode == 32) {
|
|
e = e.target || e.srcElement;
|
|
|
|
if (e.onchange)
|
|
e.onchange();
|
|
|
|
return tinymce.dom.Event.cancel(e);
|
|
}
|
|
},
|
|
|
|
_closeWinKeyHandler : function(e) {
|
|
e = e || window.event;
|
|
|
|
if (e.keyCode == 27)
|
|
tinyMCEPopup.close();
|
|
},
|
|
|
|
_wait : function() {
|
|
// Use IE method
|
|
if (document.attachEvent) {
|
|
document.attachEvent("onreadystatechange", function() {
|
|
if (document.readyState === "complete") {
|
|
document.detachEvent("onreadystatechange", arguments.callee);
|
|
tinyMCEPopup._onDOMLoaded();
|
|
}
|
|
});
|
|
|
|
if (document.documentElement.doScroll && window == window.top) {
|
|
(function() {
|
|
if (tinyMCEPopup.domLoaded)
|
|
return;
|
|
|
|
try {
|
|
// If IE is used, use the trick by Diego Perini
|
|
// http://javascript.nwbox.com/IEContentLoaded/
|
|
document.documentElement.doScroll("left");
|
|
} catch (ex) {
|
|
setTimeout(arguments.callee, 0);
|
|
return;
|
|
}
|
|
|
|
tinyMCEPopup._onDOMLoaded();
|
|
})();
|
|
}
|
|
|
|
document.attachEvent('onload', tinyMCEPopup._onDOMLoaded);
|
|
} else if (document.addEventListener) {
|
|
window.addEventListener('DOMContentLoaded', tinyMCEPopup._onDOMLoaded, false);
|
|
window.addEventListener('load', tinyMCEPopup._onDOMLoaded, false);
|
|
}
|
|
}
|
|
};
|
|
|
|
tinyMCEPopup.init();
|
|
tinyMCEPopup._wait(); // Wait for DOM Content Loaded
|