2011-03-23 10:51:00 +01:00
|
|
|
/**
|
|
|
|
* File: LeftAndMain.BatchActions.js
|
|
|
|
*/
|
|
|
|
(function($) {
|
|
|
|
$.entwine('ss', function($){
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class: #Form_BatchActionsForm
|
|
|
|
*
|
|
|
|
* Batch actions which take a bunch of selected pages,
|
|
|
|
* usually from the CMS tree implementation, and perform serverside
|
|
|
|
* callbacks on the whole set. We make the tree selectable when the jQuery.UI tab
|
|
|
|
* enclosing this form is opened.
|
|
|
|
*
|
|
|
|
* Events:
|
|
|
|
* register - Called before an action is added.
|
|
|
|
* unregister - Called before an action is removed.
|
|
|
|
*/
|
|
|
|
$('#Form_BatchActionsForm').entwine({
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Variable: Tree
|
|
|
|
* (DOMElement)
|
|
|
|
*/
|
|
|
|
Tree: null,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Variable: Actions
|
|
|
|
* (Array) Stores all actions that can be performed on the collected IDs as
|
|
|
|
* function closures. This might trigger filtering of the selected IDs,
|
|
|
|
* a confirmation message, etc.
|
|
|
|
*/
|
|
|
|
Actions: [],
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor: onmatch
|
|
|
|
*/
|
|
|
|
onmatch: function() {
|
2011-04-17 09:55:23 +02:00
|
|
|
var self = this, tree = $('.cms-tree');
|
2011-03-23 10:51:00 +01:00
|
|
|
|
|
|
|
this.setTree(tree);
|
|
|
|
|
|
|
|
tree.bind('check_node.jstree', function(e, data) {
|
|
|
|
self.serializeFromTree();
|
|
|
|
});
|
|
|
|
|
2011-04-24 01:05:59 +02:00
|
|
|
$('.cms-tree-view-modes :input[name=view-mode]').bind('click', function(e) {
|
2011-04-22 13:34:01 +02:00
|
|
|
if($(e.target).val() == 'multiselect') {
|
2011-03-23 10:51:00 +01:00
|
|
|
tree.addClass('multiple');
|
|
|
|
self.serializeFromTree();
|
2011-04-22 13:34:01 +02:00
|
|
|
} else {
|
|
|
|
tree.removeClass('multiple');
|
2011-03-23 10:51:00 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this._super();
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function: register
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
*
|
|
|
|
* (String) type - ...
|
|
|
|
* (Function) callback - ...
|
|
|
|
*/
|
|
|
|
register: function(type, callback) {
|
|
|
|
this.trigger('register', {type: type, callback: callback});
|
|
|
|
var actions = this.getActions();
|
|
|
|
actions[type] = callback;
|
|
|
|
this.setActions(actions);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function: unregister
|
|
|
|
*
|
|
|
|
* Remove an existing action.
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
*
|
|
|
|
* {String} type
|
|
|
|
*/
|
|
|
|
unregister: function(type) {
|
|
|
|
this.trigger('unregister', {type: type});
|
|
|
|
|
|
|
|
var actions = this.getActions();
|
|
|
|
if(actions[type]) delete actions[type];
|
|
|
|
this.setActions(actions);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function: _isActive
|
|
|
|
*
|
|
|
|
* Determines if we should allow and track tree selections.
|
|
|
|
*
|
|
|
|
* Todo:
|
|
|
|
* Too much coupling with tabset
|
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
* (boolean)
|
|
|
|
*/
|
|
|
|
_isActive: function() {
|
2011-04-24 01:05:59 +02:00
|
|
|
return $('.cms-content-batchactions').is(':visible');
|
2011-03-23 10:51:00 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function: refreshSelected
|
|
|
|
*
|
|
|
|
* Ajax callbacks determine which pages is selectable in a certain batch action.
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* {Object} rootNode
|
|
|
|
*/
|
|
|
|
refreshSelected : function(rootNode) {
|
|
|
|
var self = this, st = this.getTree(), ids = this.getIDs(), allIds = [];
|
|
|
|
// Default to refreshing the entire tree
|
|
|
|
if(rootNode == null) rootNode = st;
|
|
|
|
|
|
|
|
for(var idx in ids) {
|
|
|
|
$($(st).getNodeByID(idx)).addClass('selected').attr('selected', 'selected');
|
|
|
|
}
|
|
|
|
|
|
|
|
$(rootNode).find('li').each(function() {
|
|
|
|
allIds.push($(this).data('id'));
|
|
|
|
|
|
|
|
// Disable the nodes while the ajax request is being processed
|
|
|
|
$(this).addClass('treeloading').setEnabled(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Post to the server to ask which pages can have this batch action applied
|
|
|
|
var applicablePagesURL = this.find(':input[name=Action]').val() + '/applicablepages/?csvIDs=' + allIds.join(',');
|
|
|
|
jQuery.getJSON(applicablePagesURL, function(applicableIDs) {
|
|
|
|
// Set a CSS class on each tree node indicating which can be batch-actioned and which can't
|
|
|
|
jQuery(rootNode).find('li').each(function() {
|
|
|
|
$(this).removeClass('treeloading');
|
|
|
|
|
|
|
|
var id = $(this).data('id');
|
|
|
|
if(id == 0 || $.inArray(id, applicableIDs) >= 0) {
|
|
|
|
$(this).setEnabled(true);
|
|
|
|
} else {
|
|
|
|
// De-select the node if it's non-applicable
|
|
|
|
$(this).removeClass('selected').setEnabled(false);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
self.serializeFromTree();
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function: serializeFromTree
|
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
* (boolean)
|
|
|
|
*/
|
|
|
|
serializeFromTree: function() {
|
|
|
|
var tree = this.getTree(), ids = tree.getSelectedIDs();
|
|
|
|
|
|
|
|
// if no IDs are selected, stop here. This is an implict way for the
|
|
|
|
// callback to cancel the actions
|
|
|
|
if(!ids || !ids.length) return false;
|
|
|
|
|
|
|
|
// write IDs to the hidden field
|
|
|
|
this.setIDs(ids);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function: setIDS
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* {Array} ids
|
|
|
|
*/
|
|
|
|
setIDs: function(ids) {
|
|
|
|
if(ids) this.find(':input[name=csvIDs]').val(ids.join(','));
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function: getIDS
|
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
* {Array}
|
|
|
|
*/
|
|
|
|
getIDs: function() {
|
|
|
|
return this.find(':input[name=csvIDs]').val().split(',');
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function: onsubmit
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* (Event) e
|
|
|
|
*/
|
|
|
|
onsubmit: function(e) {
|
|
|
|
var ids = this.getIDs();
|
|
|
|
var tree = this.getTree();
|
|
|
|
|
|
|
|
// if no nodes are selected, return with an error
|
|
|
|
if(!ids || !ids.length) {
|
|
|
|
alert(ss.i18n._t('CMSMAIN.SELECTONEPAGE'));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// apply callback, which might modify the IDs
|
|
|
|
var type = this.find(':input[name=Action]').val();
|
|
|
|
if(this.getActions()[type]) ids = this.getActions()[type].apply(this, [ids]);
|
|
|
|
|
|
|
|
// write (possibly modified) IDs back into to the hidden field
|
|
|
|
this.setIDs(ids);
|
|
|
|
|
|
|
|
// Reset failure states
|
|
|
|
tree.find('li').removeClass('failed');
|
|
|
|
|
|
|
|
var button = this.find(':submit:first');
|
|
|
|
button.addClass('loading');
|
|
|
|
|
|
|
|
jQuery.ajax({
|
|
|
|
// don't use original form url
|
|
|
|
url: type,
|
|
|
|
type: 'POST',
|
|
|
|
data: this.serializeArray(),
|
|
|
|
complete: function(xmlhttp, status) {
|
|
|
|
button.removeClass('loading');
|
|
|
|
|
|
|
|
// status message
|
2012-03-06 21:36:16 +01:00
|
|
|
var msg = xmlhttp.getResponseHeader('X-Status');
|
|
|
|
if(msg) statusMessage(msg, (status == 'success') ? 'good' : 'bad');
|
2011-03-23 10:51:00 +01:00
|
|
|
},
|
|
|
|
success: function(data, status) {
|
|
|
|
var id;
|
|
|
|
|
|
|
|
// TODO This should use a more common serialization in a new tree library
|
|
|
|
if(data.modified) {
|
|
|
|
for(id in data.modified) {
|
2012-03-06 21:37:08 +01:00
|
|
|
var node = tree.getNodeByID(id);
|
|
|
|
// Can't use built-in set_title() as it sanitized the HTML (removing tags)
|
|
|
|
// $(node).find('.text').html(data.modified[id]['TreeTitle']);
|
|
|
|
tree.jstree('set_text', node, data.modified[id]['TreeTitle']);
|
2011-03-23 10:51:00 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if(data.deleted) {
|
|
|
|
for(id in data.deleted) {
|
|
|
|
var node = tree.getNodeByID(id);
|
|
|
|
// TODO Remove node
|
|
|
|
// if(node && node.parentTreeNode) node.parentTreeNode.removeTreeNode(node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(data.error) {
|
|
|
|
for(id in data.error) {
|
|
|
|
var node = tree.getNodeByID(id);
|
|
|
|
$(node).addClass('failed');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Deselect all nodes
|
|
|
|
tree.find('li').removeClass('selected');
|
|
|
|
|
|
|
|
// TODO Fix up to work properly with jstree - unclear if state setting is still required in new design
|
|
|
|
// // Check if current page still exists, and refresh it.
|
|
|
|
// // Otherwise remove the current form
|
|
|
|
// var selectedNode = tree.jstree('get_selected');
|
|
|
|
// if(selectedNode) {
|
|
|
|
// var selectedNodeId = selectedNode.getID();
|
|
|
|
// if(data.modified[selectedNodeId]) {
|
|
|
|
// // only if the current page was modified
|
|
|
|
// tree.jstree('select_node', selectedNode);
|
|
|
|
// } else if(data.deleted[selectedNodeId]) {
|
2011-03-31 10:52:29 +02:00
|
|
|
// jQuery('.cms-edit-form').entwine('ss').removeForm();
|
2011-03-23 10:51:00 +01:00
|
|
|
// }
|
|
|
|
// } else {
|
2011-03-31 10:52:29 +02:00
|
|
|
// jQuery('.cms-edit-form').entwine('ss').removeForm();
|
2011-03-23 10:51:00 +01:00
|
|
|
// }
|
|
|
|
|
|
|
|
// close panel
|
|
|
|
// TODO Coupling with tabs
|
|
|
|
// jQuery('#TreeActions').tabs('select', -1);
|
|
|
|
},
|
|
|
|
dataType: 'json'
|
|
|
|
});
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class: #Form_BatchActionsForm :select[name=Action]
|
|
|
|
*/
|
|
|
|
$('#Form_BatchActionsForm select[name=Action]').entwine({
|
|
|
|
|
2011-04-22 13:34:01 +02:00
|
|
|
onmatch: function() {
|
|
|
|
this.trigger('change');
|
|
|
|
this._super();
|
|
|
|
},
|
|
|
|
|
2011-03-23 10:51:00 +01:00
|
|
|
/**
|
|
|
|
* Function: onchange
|
|
|
|
*
|
|
|
|
* Parameters:
|
|
|
|
* (Event) e
|
|
|
|
*/
|
|
|
|
onchange: function(e) {
|
2011-04-22 13:34:01 +02:00
|
|
|
var form = $(e.target.form), btn = form.find(':submit');
|
|
|
|
if($(e.target).val() == -1) {
|
|
|
|
btn.attr('disabled', 'disabled');
|
|
|
|
} else {
|
|
|
|
btn.removeAttr('disabled');
|
2012-03-06 21:37:58 +01:00
|
|
|
form.submit();
|
2011-04-22 13:34:01 +02:00
|
|
|
}
|
2011-03-23 10:51:00 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
$(document).ready(function() {
|
|
|
|
/**
|
|
|
|
* Publish selected pages action
|
|
|
|
*/
|
|
|
|
$('#Form_BatchActionsForm').entwine('ss').register('admin/batchactions/publish', function(ids) {
|
|
|
|
var confirmed = confirm(
|
|
|
|
"You have " + ids.length + " pages selected.\n\n"
|
|
|
|
+ "Do your really want to publish?"
|
|
|
|
);
|
|
|
|
return (confirmed) ? ids : false;
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unpublish selected pages action
|
|
|
|
*/
|
|
|
|
$('#Form_BatchActionsForm').entwine('ss').register('admin/batchactions/unpublish', function(ids) {
|
|
|
|
var confirmed = confirm(
|
|
|
|
"You have " + ids.length + " pages selected.\n\n"
|
|
|
|
+ "Do your really want to unpublish?"
|
|
|
|
);
|
|
|
|
return (confirmed) ? ids : false;
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Delete selected pages action
|
|
|
|
*/
|
|
|
|
$('#Form_BatchActionsForm').entwine('ss').register('admin/batchactions/delete', function(ids) {
|
|
|
|
var confirmed = confirm(
|
|
|
|
"You have " + ids.length + " pages selected.\n\n"
|
|
|
|
+ "Do your really want to delete?"
|
|
|
|
);
|
|
|
|
return (confirmed) ? ids : false;
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Delete selected pages from live action
|
|
|
|
*/
|
|
|
|
$('#Form_BatchActionsForm').entwine('ss').register('admin/batchactions/deletefromlive', function(ids) {
|
|
|
|
var confirmed = confirm(
|
|
|
|
"You have " + ids.length + " pages selected.\n\n"
|
|
|
|
+ "Do your really want to delete these pages from live?"
|
|
|
|
);
|
|
|
|
return (confirmed) ? ids : false;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
})(jQuery);
|