mirror of
https://github.com/silverstripe/silverstripe-cms
synced 2024-10-22 08:05:56 +02:00
29cf057e97
(merged from branches/gsoc) git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/cms/trunk@42050 467b73ca-7a2a-4603-9d3b-597d59a354a9
823 lines
21 KiB
JavaScript
823 lines
21 KiB
JavaScript
var _AJAX_LOADING = false;
|
|
|
|
/**
|
|
* Code for the separator bar between the two panes
|
|
*/
|
|
function DraggableSeparator() {
|
|
this.onmousedown = this.onmousedown.bindAsEventListener(this);
|
|
// this.onselectstart = this.onselectstart.bindAsEventListener(this);
|
|
}
|
|
DraggableSeparator.prototype = {
|
|
onmousedown : function(event) {
|
|
this.leftBase = $('left').offsetWidth - Event.pointerX(event);
|
|
this.separatorBase = getDimension($('separator'),'left') - Event.pointerX(event);
|
|
this.rightBase = getDimension($('right'),'left') - Event.pointerX(event);
|
|
|
|
document.onmousemove = this.document_mousemove.bindAsEventListener(this);
|
|
document.onmouseup = this.document_mouseup.bindAsEventListener(this);
|
|
|
|
document.body.onselectstart = this.body_selectstart.bindAsEventListener(this);
|
|
},
|
|
document_mousemove : function(event) {
|
|
$('left').style.width = (this.leftBase + Event.pointerX(event)) + 'px';
|
|
fixRightWidth();
|
|
},
|
|
document_mouseup : function(e) {
|
|
document.onmousemove = null;
|
|
},
|
|
|
|
body_selectstart : function(event) {
|
|
Event.stop(event);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function fixRightWidth() {
|
|
if( !$('right') )
|
|
return;
|
|
|
|
// Absolutely position all the elements
|
|
var sep = getDimension($('left'),'width') + getDimension($('left'),'left');
|
|
$('separator').style.left = (sep + 2) + 'px';
|
|
$('right').style.left = (sep + 6) + 'px';
|
|
|
|
// Give the remaining space to right
|
|
var rightWidth = parseInt(document.body.offsetWidth) - parseInt($('left').offsetWidth) - $('separator').offsetWidth - 8;
|
|
|
|
if( rightWidth >= 0 )
|
|
$('right').style.width = rightWidth + 'px';
|
|
|
|
var rb;
|
|
if(rb = $('rightbottom')) {
|
|
rb.style.left = $('right').style.left;
|
|
rb.style.width = $('right').style.width;
|
|
}
|
|
}
|
|
|
|
Behaviour.register({
|
|
'#separator' : DraggableSeparator,
|
|
|
|
'#left' : {
|
|
hide : function() {
|
|
if(!this.hidden) {
|
|
this.hidden = true;
|
|
this.style.width = null;
|
|
Element.addClassName(this,'hidden');
|
|
Element.addClassName('separator','hidden');
|
|
fixRightWidth();
|
|
}
|
|
},
|
|
show : function() {
|
|
if(this.hidden) {
|
|
this.hidden = false;
|
|
Element.removeClassName(this,'hidden');
|
|
Element.removeClassName('separator','hidden');
|
|
fixRightWidth();
|
|
}
|
|
}
|
|
},
|
|
|
|
'#MainMenu li' : {
|
|
onclick : function(event) {
|
|
return LeftAndMain_window_unload(); // Confirm if there are unsaved changes
|
|
window.location.href = this.getElementsByTagName('a')[0].href;
|
|
Event.stop(event);
|
|
}
|
|
},
|
|
|
|
'#Menu-help' : {
|
|
onclick : function() {
|
|
var w = window.open(this.getElementsByTagName('a')[0].href, 'help');
|
|
w.focus();
|
|
return false;
|
|
}
|
|
},
|
|
|
|
'#Logo' : {
|
|
onclick : function() {
|
|
var w = window.open(this.getElementsByTagName('a')[0].href);
|
|
w.focus();
|
|
return false;
|
|
}
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
window.ontabschanged = function() {
|
|
var formEl = $('Form_EditForm');
|
|
|
|
if( !formEl )
|
|
return;
|
|
|
|
var fs = formEl.getElementsByTagName('fieldset')[0];
|
|
if(fs) fs.style.height = formEl.style.height;
|
|
|
|
// var divs = document.getElementsBySelector('#Form_EditForm div');
|
|
/*for(i=0;i<divs.length;i++) {
|
|
if( ( Element.hasClassName(divs[i],'tab') || Element.hasClassName(divs[i],'tabset') ) && isVisible(divs[i]) ) {
|
|
if(navigator.appName == "Microsoft Internet Explorer")
|
|
fitToParent(divs[i], i == 0 ? 18 : 0);
|
|
else
|
|
fitToParent(divs[i], 3);
|
|
}
|
|
}*/
|
|
|
|
if( _TAB_DIVS_ON_PAGE ) {
|
|
for(i = 0; i < _TAB_DIVS_ON_PAGE.length; i++ ) {
|
|
fitToParent(_TAB_DIVS_ON_PAGE[i], 30);
|
|
}
|
|
}
|
|
}
|
|
|
|
window.onresize = function(init) {
|
|
var right = $('right');
|
|
var rightbottom = $('rightbottom');
|
|
if(rightbottom) rightbottom.style.display = 'none';
|
|
|
|
if(typeof fitToParent == 'function') fitToParent('right', 12);
|
|
|
|
if( $('left') && $('separator') && right )
|
|
$('left').style.height = $('separator').style.height = right.style.height;
|
|
|
|
if(rightbottom) {
|
|
var newHeight = (parseInt(right.style.height) / 3);
|
|
right.style.height = (newHeight*2) + 'px';
|
|
rightbottom.style.height = newHeight + 'px';
|
|
rightbottom.style.display = '';
|
|
rightbottom.style.top = getDimension(right,'top') + (newHeight*2) + 'px';
|
|
}
|
|
|
|
if(typeof fitToParent == 'function') fitToParent('Form_EditForm');
|
|
if(typeof fixHeight_left == 'function') fixHeight_left();
|
|
if(typeof fixRightWidth == 'function') fixRightWidth();
|
|
|
|
window.ontabschanged();
|
|
}
|
|
|
|
appendLoader(function() {
|
|
document.body.style.overflow = 'hidden';
|
|
window.onresize(true);
|
|
});
|
|
|
|
function isVisible(el) {
|
|
// if(typeof el.isVisible != 'undefined') return el.isVisible;
|
|
if(el.tagName == "body" || el.tagName == "BODY") return (el.isVisible = true);
|
|
else if(el.style.display == 'none') return (el.isVisible = false);
|
|
else return (el.isVisible = isVisible(el.parentNode));
|
|
}
|
|
|
|
|
|
LeftAndMain_window_unload = function() {
|
|
window.exiting = true; // this is used by prototype
|
|
if(typeof autoSave == 'function') {
|
|
return autoSave(true);
|
|
}
|
|
}
|
|
|
|
// Event.observe(window, 'beforeunload', LeftAndMain_window_unload);
|
|
|
|
/**
|
|
* Unlock the locked status message.
|
|
* Show a queued message, if one exists
|
|
*/
|
|
function unlockStatusMessage() {
|
|
statusMessage.locked = false;
|
|
if(statusMessage.queued) {
|
|
statusMessage(
|
|
statusMessage.queued.msg,
|
|
statusMessage.queued.type,
|
|
statusMessage.queued.showNetworkActivity);
|
|
|
|
statusMessage.queued = null;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Move form actions to the top and make them ajax
|
|
*/
|
|
function ajaxActionsAtTop(formName, actionHolderName, tabName) {
|
|
var actions = document.getElementsBySelector('#' + formName + ' .Actions')[0];
|
|
var holder;
|
|
|
|
if(actions) {
|
|
if((holder = $(actionHolderName)) && holder != actions) {
|
|
holder.parentNode.removeChild(holder);
|
|
}
|
|
|
|
actions.id = actionHolderName;
|
|
actions.className = 'ajaxActions';
|
|
|
|
$(tabName).appendChild(actions);
|
|
prepareAjaxActions(actions, formName, tabName);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Prepare the ajax actions so that the buttons actually do something
|
|
*/
|
|
function prepareAjaxActions(actions, formName, tabName) {
|
|
var i, button, list = actions.getElementsByTagName('input')
|
|
for (i=0;button=list[i];i++) {
|
|
button.ownerForm = $(formName);
|
|
|
|
button.onclick = function(e) {
|
|
// tries to call a custom method of the format "action_<youraction>_right"
|
|
if(window[this.name + '_' + tabName]) {
|
|
window[this.name + '_' + tabName](e);
|
|
} else {
|
|
statusMessage(ingize(this.value));
|
|
Ajax.SubmitForm(this.ownerForm, this.name, {
|
|
onSuccess: Ajax.Evaluator,
|
|
onFailure: ajaxErrorHandler
|
|
});
|
|
}
|
|
return false;
|
|
}
|
|
// behaveAs(button, StatusTitle);
|
|
}
|
|
}
|
|
|
|
function ingize(val) {
|
|
var ingWord, suffix;
|
|
if(!val) val = "process";
|
|
|
|
if(val.match(/^([^ ]+) +(.*)$/)) {
|
|
ingWord = RegExp.$1;
|
|
suffix = ' ' + RegExp.$2 + '...';
|
|
} else {
|
|
ingWord = val;
|
|
suffix = '...';
|
|
}
|
|
|
|
if(ingWord.match(/^(.*)e$/)) return RegExp.$1 + 'ing' + suffix;
|
|
else return ingWord + 'ing' + suffix;
|
|
}
|
|
|
|
/**
|
|
* Submit the given form and evaluate the Ajax response.
|
|
* Needs to be bound to an object with the following parameters to work:
|
|
* - form
|
|
* - action
|
|
* - verb
|
|
*
|
|
* The bound function can then be called, with the arguments passed
|
|
*/
|
|
|
|
function ajaxSubmitForm(automated, callAfter, form, action, verb) {
|
|
// tinyMCE.triggerSave(true);
|
|
|
|
var alreadySaved = false;
|
|
if($(form).elements.length < 2) alreadySaved = true;
|
|
|
|
if(alreadySaved) {
|
|
if(callAfter) callAfter();
|
|
|
|
} else {
|
|
statusMessage(verb + '...', '', true);
|
|
|
|
var success = function(response) {
|
|
Ajax.Evaluator(response);
|
|
if(callAfter) callAfter();
|
|
}
|
|
|
|
if(callAfter) success = success.bind({callAfter : callAfter});
|
|
Ajax.SubmitForm(form, action, {
|
|
onSuccess : success,
|
|
onFailure : function(response) {
|
|
errorMessage('Error ' + verb, response);
|
|
}
|
|
});
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
/**
|
|
* Post the given fields to the given url
|
|
*/
|
|
function ajaxSubmitFieldSet(href, fieldSet, extraData) {
|
|
// Build data
|
|
var i,field,data = "ajax=1";
|
|
for(i=0;field=fieldSet[i];i++) {
|
|
data += '&' + Form.Element.serialize(field);
|
|
}
|
|
if(extraData){
|
|
data += '&'+extraData;
|
|
}
|
|
// Send request
|
|
new Ajax.Request(href, {
|
|
method : 'post', postBody : data,
|
|
onSuccess : function(response) {
|
|
//alert(response.responseText);
|
|
Ajax.Evaluator(response);
|
|
},
|
|
onFailure : function(response) {
|
|
alert(response.responseText);
|
|
//errorMessage('Error: ', response);
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Post the given fields to the given url
|
|
*/
|
|
function ajaxLink(href) {
|
|
// Send request
|
|
new Ajax.Request(href + (href.indexOf("?") == -1 ? "?" : "&") + "ajax=1", {
|
|
method : 'get',
|
|
onSuccess : Ajax.Evaluator,
|
|
onFailure : ajaxErrorHandler
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Load a URL into the given form
|
|
*/
|
|
function ajaxLoadPage() {
|
|
statusMessage('loading...', 2, true);
|
|
new Ajax.Request(this.URL + '&ajax=1', {
|
|
method : 'get',
|
|
onSuccess : ajaxLoadPage_success.bind(this)
|
|
});
|
|
}
|
|
function ajaxLoadPage_success(response) {
|
|
statusMessage('loaded');
|
|
$(this.form).loadNewPage(response.responseText);
|
|
}
|
|
|
|
/**
|
|
* Behaviour of the statuts message.
|
|
*/
|
|
Behaviour.register({
|
|
'#statusMessage' : {
|
|
showMessage : function(message, type, waitTime, clearManually) {
|
|
if(this.fadeTimer) {
|
|
clearTimeout(this.fadeTimer);
|
|
this.fadeTimer = null;
|
|
}
|
|
if(this.currentEffect) {
|
|
this.currentEffect.cancel();
|
|
this.currentEffect = null;
|
|
}
|
|
|
|
this.innerHTML = message;
|
|
this.className = type;
|
|
Element.setOpacity(this, 1);
|
|
|
|
this.style.position = 'absolute';
|
|
this.style.display = '';
|
|
this.style.visibility = '';
|
|
|
|
if(!clearManually) {
|
|
this.fade(0.5,waitTime ? waitTime : 5);
|
|
}
|
|
},
|
|
clearMessage : function(waitTime) {
|
|
this.fade(0.5, waitTime);
|
|
},
|
|
fade: function(fadeTime, waitTime) {
|
|
if(!fadeTime) fadeTime = 0.5;
|
|
|
|
// Wait a bit before fading
|
|
if(waitTime) {
|
|
this.fadeTimer = setTimeout((function() {
|
|
this.fade(fadeTime);
|
|
}).bind(this), waitTime * 1000);
|
|
|
|
// Fade straight away
|
|
} else {
|
|
this.currentEffect = new Effect.Opacity(this,
|
|
{ duration: 0.5,
|
|
transition: Effect.Transitions.linear,
|
|
from: 1.0, to: 0.0,
|
|
afterFinish : this.afterFade.bind(this) });
|
|
}
|
|
},
|
|
afterFade : function() {
|
|
this.style.visibility = 'hidden';
|
|
this.style.display = 'none';
|
|
this.innerHTML = '';
|
|
}
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Show a status message.
|
|
*
|
|
* @param msg String
|
|
* @param type String (optional) can be 'good' or 'bad'
|
|
* @param clearManually boolean Don't automatically fade message.
|
|
* @param container custom #statusMessage element to show message.
|
|
*/
|
|
function statusMessage(msg, type, clearManually, container) {
|
|
var statusMessageEl = $('statusMessage');
|
|
if(container != null) statusMessageEl = container;
|
|
if(statusMessageEl) {
|
|
if(msg) {
|
|
statusMessageEl.showMessage(msg, type, msg.length / 20, clearManually);
|
|
} else {
|
|
statusMessageEl.clearMessage();
|
|
}
|
|
}
|
|
}
|
|
|
|
function clearStatusMessage() {
|
|
$('statusMessage').clearMessage();
|
|
}
|
|
|
|
/**
|
|
* Called when something goes wrong
|
|
*/
|
|
function errorMessage(msg, fullMessage) {
|
|
// More complex error for developers
|
|
if(fullMessage && window.location.href.indexOf('//dev') != -1) {
|
|
// Get the message from an Ajax response object
|
|
try {
|
|
if(typeof fullMessage == 'object') fullMessage = fullMessage.status + '//' + fullMessage.responseText;
|
|
} catch(er) {
|
|
fullMessage = "";
|
|
}
|
|
msg = msg + '<br>' + fullMessage.replace(/\n/g,'<br>');
|
|
}
|
|
|
|
$('statusMessage').showMessage(msg,'bad',60);
|
|
}
|
|
|
|
function ajaxErrorHandler(response) {
|
|
errorMessage('Error talking to server', response);
|
|
}
|
|
|
|
/**
|
|
* Applying StatusTitle to an element will mean that the title attribute is shown as a statusmessage
|
|
* upon hover
|
|
*/
|
|
/* Commenting out because on IE6, IE7, and Safari 3, the statusmessage becomes
|
|
* 'null' on 2nd hover and because there is not room for long titles when
|
|
* action buttons are on the same line.
|
|
StatusTitle = Class.create();
|
|
StatusTitle.prototype = {
|
|
onmouseover : function() {
|
|
if(this.title) {
|
|
this.message = this.title;
|
|
this.title = null;
|
|
}
|
|
if(this.message) {
|
|
$('statusMessage').showMessage(this.message);
|
|
}
|
|
},
|
|
onmouseout : function() {
|
|
if(this.message) {
|
|
$('statusMessage').fade(0.3,1);
|
|
}
|
|
}
|
|
}
|
|
*/
|
|
|
|
/**
|
|
* BaseForm is the base form class used in the CMS.
|
|
*/
|
|
BaseForm = Class.create();
|
|
BaseForm.prototype = {
|
|
intitialize: function() {
|
|
this.visible = this.style.display == 'none' ? false : true;
|
|
|
|
// Collect all the buttons and attach handlers
|
|
this.buttons = [];
|
|
var i,input,allInputs = this.getElementsByTagName('input');
|
|
for(i=0;input=allInputs[i];i++) {
|
|
if(input.type == 'button' || input.type == 'submit') {
|
|
this.buttons.push(input);
|
|
input.holder = this;
|
|
input.onclick = function() { return this.holder.buttonClicked(this); }
|
|
}
|
|
}
|
|
},
|
|
show: function() {
|
|
this.visible = true;
|
|
Element.hide(show);
|
|
},
|
|
hide: function() {
|
|
this.visible = false;
|
|
Element.hide(this);
|
|
},
|
|
isVisible: function() {
|
|
return this.visible;
|
|
},
|
|
buttonClicked: function(button) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* ChangeTracker is a class that can be applied to forms to support change tracking on forms.
|
|
*/
|
|
ChangeTracker = Class.create();
|
|
ChangeTracker.prototype = {
|
|
initialize: function() {
|
|
this.resetElements();
|
|
},
|
|
|
|
/**
|
|
* Reset all the 'changed field' data.
|
|
*/
|
|
resetElements: function(debug) {
|
|
var elements = Form.getElements(this);
|
|
var i, element;
|
|
for(i=0;element=elements[i];i++) {
|
|
// Initialise each element
|
|
if(element.resetChanged) {
|
|
element.resetChanged();
|
|
} else {
|
|
element.originalSerialized = Form.Element.serialize(element);
|
|
}
|
|
}
|
|
},
|
|
|
|
field_changed: function() {
|
|
return this.originalSerialized != Form.Element.serialize(this);
|
|
},
|
|
|
|
/**
|
|
* Returns true if something in the form has been changed
|
|
*/
|
|
isChanged: function() {
|
|
var elements = Form.getElements(this);
|
|
var i, element;
|
|
for(i=0;element=elements[i];i++) {
|
|
if(!element.isChanged) element.isChanged = this.field_changed;
|
|
if(!this.changeDetection_fieldsToIgnore[element.name] && element.isChanged()) {
|
|
|
|
//if( window.location.href.match( /^https?:\/\/dev/ ) )
|
|
// Debug.log('Changed:'+ element.id + '(' + this.originalSerialized +')->('+Form.Element.serialize(element)+')' );
|
|
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
},
|
|
|
|
changeDetection_fieldsToIgnore : {
|
|
'Sort' : true
|
|
},
|
|
|
|
/**
|
|
* Serialize only the fields to change.
|
|
* You can specify the names of fields that must be included as arguments
|
|
*/
|
|
serializeChangedFields: function() {
|
|
var elements = Form.getElements(this);
|
|
var queryComponent, queryComponents = new Array();
|
|
var i, element;
|
|
|
|
var forceFields = {};
|
|
if(arguments) {for(var i=0;i<arguments.length;i++) forceFields[arguments[i]] = true;}
|
|
|
|
for(i=0;element=elements[i];i++) {
|
|
if(!element.name.match(/^action_(.+)$/i)) // For dropdown those 'action_xxx' fields.
|
|
{ if(!element.isChanged) element.isChanged = this.field_changed;
|
|
if(forceFields[element.name] || (element.isChanged()) || element.name.match(/\[.*\]/g) ) {
|
|
queryComponent = Form.Element.serialize(element);
|
|
if (queryComponent)
|
|
queryComponents.push(queryComponent);
|
|
} else {
|
|
// Used by the Sapphire code to preserve the form field value
|
|
|
|
if( element.name.match( '/\]$/' ) )
|
|
queryComponents.push(element.name.substring( 0, element.name.length - 1 ) + '_unchanged' + ']=1' );
|
|
else
|
|
queryComponents.push(element.name + '_unchanged=1');
|
|
}
|
|
}
|
|
}
|
|
//alert(queryComponents.join('&'));
|
|
return queryComponents.join('&');
|
|
},
|
|
|
|
/**
|
|
* Serialize all the fields on the page
|
|
*/
|
|
serializeAllFields: function() {
|
|
return Form.serializeWithoutButtons(this);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* ModalForm provides a form with the functionality to prevent certian actions from occurring until
|
|
* it's been closed.
|
|
*
|
|
* To use, You should run the blockEvent method as many times as needed.
|
|
*/
|
|
ModalForm = Class.extend('BaseForm');
|
|
ModalForm.prototype = {
|
|
/*
|
|
* Prevent the given event from occurring on the given event while the form is open.
|
|
* These must be CMS-created events, not built-in javascript events.
|
|
* For example, form.blockEvent($('sitetree'), 'SelectionChanged')
|
|
*/
|
|
blockEvent: function(element, event) {
|
|
element.observeMethod(event, (function() { return !this.isVisible();}).bind(this) );
|
|
}
|
|
}
|
|
|
|
|
|
function doYouWantToRollback(handlers) {
|
|
var url = document.getElementsByTagName('base')[0].href + 'admin/canceldraftchangesdialog';
|
|
OpenModalDialog(url, handlers, 'Are you sure?' );
|
|
}
|
|
|
|
function modalDialog(url, handlers) {
|
|
var baseURL = document.getElementsByTagName('base')[0].href;
|
|
if(window.showModalDialog) {
|
|
var result = showModalDialog(baseURL + url + '&Modal=1', null, "status:no;dialogWidth:400px;dialogHeight:150px;edge:sunken");
|
|
if(handlers[result])
|
|
handlers[result]();
|
|
|
|
}
|
|
}
|
|
|
|
|
|
ModalDialog = Class.create();
|
|
ModalDialog.prototype = {
|
|
initialize: function(url, handlers) {
|
|
this.url = url;
|
|
this.handlers = handlers;
|
|
this.timer = setInterval(this.interval.bind(this), 50);
|
|
this.window = window.open(this.url, 'dialog', "status=no,width=400,height=150,edge=sunken");
|
|
this.window.dialogObject = this;
|
|
this.window.linkedObject = this;
|
|
setTimeout( (function(){this.window.linkedObject = this;}).bind(this), 500);
|
|
},
|
|
force: function (val) {
|
|
this.finished = true;
|
|
this.clearInterval(this.time);
|
|
|
|
if(this.handlers[val]) {
|
|
_DO_YOU_WANT_TO_SAVE_IS_OPEN = false;
|
|
(this.handlers[val])();
|
|
} else {
|
|
throw("Couldn't find a handler called '" + this.result + "'");
|
|
}
|
|
},
|
|
interval: function() {
|
|
if(this.finished) {
|
|
clearInterval(this.timer);
|
|
return;
|
|
}
|
|
if(!this.window || this.window.closed) {
|
|
clearInterval(this.timer);
|
|
if(this.handlers) {
|
|
if(this.handlers[this.result]) {
|
|
_DO_YOU_WANT_TO_SAVE_IS_OPEN = false;
|
|
(this.handlers[this.result])();
|
|
|
|
} else {
|
|
throw("Couldn't find a handler called '" + this.result + "'");
|
|
}
|
|
}
|
|
} else {
|
|
this.window.focus();
|
|
}
|
|
}
|
|
}
|
|
|
|
window.top._OPEN_DIALOG = null;
|
|
|
|
OpenModalDialog = function( url, handlers, message ) {
|
|
var dialog = new GBModalDialog( url, handlers, message );
|
|
}
|
|
|
|
GBModalDialog = Class.create();
|
|
GBModalDialog.prototype = {
|
|
|
|
initialize: function( url, handlers, message ) {
|
|
this.url = url;
|
|
this.handlers = handlers;
|
|
this.caption = message;
|
|
|
|
window.top._OPEN_DIALOG = this;
|
|
|
|
GB_show( this.caption, this.url, 110, 450 );
|
|
},
|
|
|
|
execHandler: function( handler ) {
|
|
GB_hide();
|
|
|
|
if( this.handlers[handler] )
|
|
this.handlers[handler]();
|
|
else
|
|
throw( "Unknown handler '" + handler + "'" );
|
|
}
|
|
}
|
|
|
|
function hideLoading() {
|
|
$('Loading').style.display = 'none';
|
|
document.body.className = '';
|
|
}
|
|
function baseHref() {
|
|
var baseTags = document.getElementsByTagName('base');
|
|
if(baseTags) return baseTags[0].href;
|
|
else return "";
|
|
}
|
|
|
|
returnFalse = function() {
|
|
return false;
|
|
}
|
|
showResponseAsSuccessMessage = function(response) {
|
|
statusMessage(response.responseText, 'good');
|
|
}
|
|
|
|
/**
|
|
* This function is called by prototype when it receives notification that the user was logged out.
|
|
* It redirects back to the login form.
|
|
*/
|
|
function onSessionLost() {
|
|
alert("You've been logged out of the server, so we're going to send you back to the log-in screen.");
|
|
window.location.reload(true);
|
|
}
|
|
|
|
var _CURRENT_CONTEXT_MENU = null;
|
|
|
|
/**
|
|
* Create a new context menu
|
|
* @param event The event object
|
|
* @param owner The DOM element that this context-menu was requested from
|
|
* @param menuItems A map of title -> method; context-menu operations to get called
|
|
*/
|
|
function createContextMenu(event, owner, menuItems) {
|
|
if(_CURRENT_CONTEXT_MENU) {
|
|
document.body.removeChild(_CURRENT_CONTEXT_MENU);
|
|
_CURRENT_CONTEXT_MENU = null;
|
|
}
|
|
|
|
var menu = document.createElement("ul");
|
|
menu.className = 'contextMenu';
|
|
menu.style.position = 'absolute';
|
|
menu.style.left = event.clientX + 'px';
|
|
menu.style.top = event.clientY + 'px';
|
|
|
|
var menuItemName, menuItemTag, menuATag;
|
|
for(menuItemName in menuItems) {
|
|
menuItemTag = document.createElement("li");
|
|
|
|
menuATag = document.createElement("a");
|
|
menuATag.href = "#";
|
|
menuATag.onclick = menuATag.oncontextmenu = contextmenu_onclick;
|
|
menuATag.innerHTML = menuItemName;
|
|
menuATag.handler = menuItems[menuItemName];
|
|
menuATag.owner = owner;
|
|
|
|
menuItemTag.appendChild(menuATag);
|
|
menu.appendChild(menuItemTag);
|
|
}
|
|
|
|
document.body.appendChild(menu);
|
|
|
|
document.body.onclick = contextmenu_close;
|
|
|
|
_CURRENT_CONTEXT_MENU = menu;
|
|
|
|
return menu;
|
|
}
|
|
|
|
function contextmenu_close() {
|
|
if(_CURRENT_CONTEXT_MENU) {
|
|
document.body.removeChild(_CURRENT_CONTEXT_MENU);
|
|
_CURRENT_CONTEXT_MENU = null;
|
|
}
|
|
}
|
|
|
|
function contextmenu_onclick() {
|
|
this.handler(this.owner);
|
|
contextmenu_close();
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Shows an ajax loading indicator.
|
|
*
|
|
* @param id String Identifier for the newly created image
|
|
* @param container ID/DOM Element
|
|
* @param imgSrc String (optional)
|
|
* @param insertionType Object (optional) Prototype-style insertion-classes, defaults to Insertion.Bottom
|
|
* @param displayType String (optional) "inline" or "block"
|
|
*/
|
|
function showIndicator(id, container, imgSrc, insertionType, displayType) {
|
|
if(!id || !$(container)) return false;
|
|
if(!imgSrc) imgSrc = "cms/images/network-save.gif";
|
|
if(!displayType) displayType = "inline";
|
|
if(!insertionType) insertionType = Insertion.Bottom;
|
|
|
|
if(!$(id)) {
|
|
var html = '<img src="' + imgSrc + '" class="indicator ' + displayType + '" id="' + id + '" style="display: none" />';
|
|
new insertionType(container, html);
|
|
}
|
|
|
|
Effect.Appear(id);
|
|
}
|
|
|
|
function hideIndicator(id) {
|
|
Effect.Fade(id, {duration: 0.3});
|
|
} |