2011-03-23 22:51:00 +13:00
/ * *
* File : LeftAndMain . BatchActions . js
* /
( function ( $ ) {
2012-05-18 12:07:40 +12:00
$ . entwine ( 'ss.tree' , function ( $ ) {
2011-03-23 22:51:00 +13:00
/ * *
* 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 : 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 : [ ] ,
2012-06-12 22:48:08 +12:00
getTree : function ( ) {
return $ ( '.cms-tree' ) ;
} ,
fromTree : {
oncheck _node : function ( e , data ) {
this . serializeFromTree ( ) ;
2012-06-18 00:31:27 +02:00
} ,
onuncheck _node : function ( e , data ) {
this . serializeFromTree ( ) ;
2012-06-12 22:48:08 +12:00
}
} ,
2015-05-15 11:50:23 +12:00
/ * *
2015-06-16 10:52:42 +12:00
* @ func registerDefault
* @ desc Register default bulk confirmation dialogs
2015-05-15 11:50:23 +12:00
* /
registerDefault : function ( ) {
// Publish selected pages action
this . register ( 'admin/pages/batchactions/publish' , function ( ids ) {
var confirmed = confirm (
ss . i18n . inject (
ss . i18n . _t (
"CMSMAIN.BATCH_PUBLISH_PROMPT" ,
"You have {num} page(s) selected.\n\nDo you really want to publish?"
) ,
{ 'num' : ids . length }
)
) ;
return ( confirmed ) ? ids : false ;
} ) ;
// Unpublish selected pages action
this . register ( 'admin/pages/batchactions/unpublish' , function ( ids ) {
var confirmed = confirm (
ss . i18n . inject (
ss . i18n . _t (
"CMSMAIN.BATCH_UNPUBLISH_PROMPT" ,
"You have {num} page(s) selected.\n\nDo you really want to unpublish"
) ,
{ 'num' : ids . length }
)
) ;
return ( confirmed ) ? ids : false ;
} ) ;
// Delete selected pages action
// @deprecated since 4.0 Use archive instead
this . register ( 'admin/pages/batchactions/delete' , function ( ids ) {
var confirmed = confirm (
ss . i18n . inject (
ss . i18n . _t (
"CMSMAIN.BATCH_DELETE_PROMPT" ,
"You have {num} page(s) selected.\n\nDo you really want to delete?"
) ,
{ 'num' : ids . length }
)
) ;
return ( confirmed ) ? ids : false ;
} ) ;
// Delete selected pages action
this . register ( 'admin/pages/batchactions/archive' , function ( ids ) {
var confirmed = confirm (
ss . i18n . inject (
ss . i18n . _t (
"CMSMAIN.BATCH_ARCHIVE_PROMPT" ,
"You have {num} page(s) selected.\n\nDo you really want to archive?\n\nThese pages will be removed from both the draft and published sites without discarding the history."
) ,
{ 'num' : ids . length }
)
) ;
return ( confirmed ) ? ids : false ;
} ) ;
// Delete selected pages from live action
this . register ( 'admin/pages/batchactions/deletefromlive' , function ( ids ) {
var confirmed = confirm (
ss . i18n . inject (
ss . i18n . _t (
"CMSMAIN.BATCH_DELETELIVE_PROMPT" ,
"You have {num} page(s) selected.\n\nDo you really want to delete these pages from live?"
) ,
{ 'num' : ids . length }
)
) ;
return ( confirmed ) ? ids : false ;
} ) ;
} ,
2012-06-12 22:48:08 +12:00
onadd : function ( ) {
2015-05-15 11:50:23 +12:00
this . registerDefault ( ) ;
2012-06-12 22:48:08 +12:00
this . _super ( ) ;
} ,
2012-05-11 12:53:40 +12:00
/ * *
2015-06-16 10:52:42 +12:00
* @ func register
* @ param { string } type
* @ param { function } callback
2011-03-23 22:51:00 +13:00
* /
register : function ( type , callback ) {
this . trigger ( 'register' , { type : type , callback : callback } ) ;
var actions = this . getActions ( ) ;
actions [ type ] = callback ;
this . setActions ( actions ) ;
} ,
2015-06-16 10:52:42 +12:00
2011-03-23 22:51:00 +13:00
/ * *
2015-06-16 10:52:42 +12:00
* @ func unregister
* @ param { string } type
* @ desc Remove an existing action .
2011-03-23 22:51:00 +13:00
* /
unregister : function ( type ) {
this . trigger ( 'unregister' , { type : type } ) ;
2015-06-16 10:52:42 +12:00
2011-03-23 22:51:00 +13:00
var actions = this . getActions ( ) ;
if ( actions [ type ] ) delete actions [ type ] ;
this . setActions ( actions ) ;
} ,
2015-06-16 10:52:42 +12:00
2011-03-23 22:51:00 +13:00
/ * *
2015-06-16 10:52:42 +12:00
* @ func refreshSelected
* @ param { object } rootNode
* @ desc Ajax callbacks determine which pages is selectable in a certain batch action .
2011-03-23 22:51:00 +13:00
* /
refreshSelected : function ( rootNode ) {
2015-05-15 11:50:23 +12:00
var self = this ,
st = this . getTree ( ) ,
ids = this . getIDs ( ) ,
allIds = [ ] ,
selectedAction = this . find ( ':input[name=Action]' ) . val ( ) ;
2011-03-23 22:51:00 +13:00
// Default to refreshing the entire tree
if ( rootNode == null ) rootNode = st ;
for ( var idx in ids ) {
$ ( $ ( st ) . getNodeByID ( idx ) ) . addClass ( 'selected' ) . attr ( 'selected' , 'selected' ) ;
}
2015-05-15 11:50:23 +12:00
// If no action is selected, enable all nodes
2015-06-16 10:52:42 +12:00
if ( ! selectedAction || selectedAction == - 1 ) {
2015-05-15 11:50:23 +12:00
$ ( rootNode ) . find ( 'li' ) . each ( function ( ) {
$ ( this ) . setEnabled ( true ) ;
} ) ;
return ;
}
// Disable the nodes while the ajax request is being processed
2011-03-23 22:51:00 +13:00
$ ( rootNode ) . find ( 'li' ) . each ( function ( ) {
allIds . push ( $ ( this ) . data ( 'id' ) ) ;
$ ( this ) . addClass ( 'treeloading' ) . setEnabled ( false ) ;
} ) ;
2015-05-15 11:50:23 +12:00
2011-03-23 22:51:00 +13:00
// Post to the server to ask which pages can have this batch action applied
2015-05-15 11:50:23 +12:00
var applicablePagesURL = selectedAction + '/applicablepages/?csvIDs=' + allIds . join ( ',' ) ;
2011-03-23 22:51:00 +13:00
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 ) ;
2015-05-15 11:50:23 +12:00
$ ( this ) . prop ( 'selected' , false ) ;
2011-03-23 22:51:00 +13:00
}
} ) ;
self . serializeFromTree ( ) ;
} ) ;
} ,
/ * *
2015-06-16 10:52:42 +12:00
* @ func serializeFromTree
* @ return { boolean }
2011-03-23 22:51:00 +13:00
* /
serializeFromTree : function ( ) {
var tree = this . getTree ( ) , ids = tree . getSelectedIDs ( ) ;
// write IDs to the hidden field
this . setIDs ( ids ) ;
return true ;
} ,
/ * *
2015-06-16 10:52:42 +12:00
* @ func setIDS
* @ param { array } ids
2011-03-23 22:51:00 +13:00
* /
setIDs : function ( ids ) {
2012-03-08 01:27:47 +01:00
this . find ( ':input[name=csvIDs]' ) . val ( ids ? ids . join ( ',' ) : null ) ;
2011-03-23 22:51:00 +13:00
} ,
/ * *
2015-06-16 10:52:42 +12:00
* @ func getIDS
* @ return { array }
2011-03-23 22:51:00 +13:00
* /
getIDs : function ( ) {
2015-05-15 11:50:23 +12:00
// Map empty value to empty array
var value = this . find ( ':input[name=csvIDs]' ) . val ( ) ;
return value
? value . split ( ',' )
: [ ] ;
2011-03-23 22:51:00 +13:00
} ,
2015-06-16 10:52:42 +12:00
2011-03-23 22:51:00 +13:00
onsubmit : function ( e ) {
2015-05-15 11:50:23 +12:00
var self = this , ids = this . getIDs ( ) , tree = this . getTree ( ) , actions = this . getActions ( ) ;
2011-03-23 22:51:00 +13:00
// if no nodes are selected, return with an error
if ( ! ids || ! ids . length ) {
2015-05-15 11:50:23 +12:00
alert ( ss . i18n . _t ( 'CMSMAIN.SELECTONEPAGE' , 'Please select at least one page' ) ) ;
e . preventDefault ( ) ;
2011-03-23 22:51:00 +13:00
return false ;
}
// apply callback, which might modify the IDs
var type = this . find ( ':input[name=Action]' ) . val ( ) ;
2015-05-15 11:50:23 +12:00
if ( actions [ type ] ) {
ids = this . getActions ( ) [ type ] . apply ( this , [ ids ] ) ;
}
// Discontinue processing if there are no further items
if ( ! ids || ! ids . length ) {
e . preventDefault ( ) ;
return false ;
}
2011-03-23 22:51:00 +13:00
// 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' ) ;
2012-03-08 01:27:47 +01:00
// Deselect all nodes
tree . jstree ( 'uncheck_all' ) ;
self . setIDs ( [ ] ) ;
// Reset action
self . find ( ':input[name=Action]' ) . val ( '' ) . change ( ) ;
2011-03-23 22:51:00 +13:00
2012-05-14 15:13:49 +02:00
// status message (decode into UTF-8, HTTP headers don't allow multibyte)
2012-03-06 21:36:16 +01:00
var msg = xmlhttp . getResponseHeader ( 'X-Status' ) ;
2012-05-14 15:13:49 +02:00
if ( msg ) statusMessage ( decodeURIComponent ( msg ) , ( status == 'success' ) ? 'good' : 'bad' ) ;
2011-03-23 22:51:00 +13:00
} ,
success : function ( data , status ) {
2012-03-08 01:27:47 +01:00
var id , node ;
2011-03-23 22:51:00 +13:00
if ( data . modified ) {
2012-03-08 01:47:26 +01:00
var modifiedNodes = [ ] ;
2011-03-23 22:51:00 +13:00
for ( id in data . modified ) {
2012-03-08 01:27:47 +01:00
node = tree . getNodeByID ( id ) ;
2012-03-06 21:37:08 +01:00
tree . jstree ( 'set_text' , node , data . modified [ id ] [ 'TreeTitle' ] ) ;
2012-03-08 01:47:26 +01:00
modifiedNodes . push ( node ) ;
2011-03-23 22:51:00 +13:00
}
2012-03-08 01:47:26 +01:00
$ ( modifiedNodes ) . effect ( 'highlight' ) ;
2011-03-23 22:51:00 +13:00
}
if ( data . deleted ) {
for ( id in data . deleted ) {
2012-03-08 01:27:47 +01:00
node = tree . getNodeByID ( id ) ;
if ( node . length ) tree . jstree ( 'delete_node' , node ) ;
2011-03-23 22:51:00 +13:00
}
}
if ( data . error ) {
for ( id in data . error ) {
2012-03-08 01:27:47 +01:00
node = tree . getNodeByID ( id ) ;
2011-03-23 22:51:00 +13:00
$ ( node ) . addClass ( 'failed' ) ;
}
}
} ,
dataType : 'json'
} ) ;
2015-05-15 11:50:23 +12:00
// Never process this action; Only invoke via ajax
e . preventDefault ( ) ;
2011-03-23 22:51:00 +13:00
return false ;
}
} ) ;
2015-06-16 10:52:42 +12:00
$ ( '.cms-content-batchactions-button' ) . entwine ( {
onmatch : function ( ) {
this . _super ( ) ;
this . updateTree ( ) ;
} ,
onunmatch : function ( ) {
this . _super ( ) ;
} ,
onclick : function ( e ) {
this . updateTree ( ) ;
} ,
updateTree : function ( ) {
var tree = $ ( '.cms-tree' ) ,
form = $ ( '#Form_BatchActionsForm' ) ;
this . _super ( ) ;
if ( this . data ( 'active' ) ) {
tree . addClass ( 'multiple' ) ;
tree . removeClass ( 'draggable' ) ;
form . serializeFromTree ( ) ;
2015-06-16 10:52:42 +12:00
$ ( '#Form_BatchActionsForm' ) . refreshSelected ( ) ;
2015-06-16 10:52:42 +12:00
} else {
tree . removeClass ( 'multiple' ) ;
tree . addClass ( 'draggable' ) ;
}
}
} ) ;
2011-03-23 22:51:00 +13:00
/ * *
2012-05-11 15:31:12 +12:00
* Class : # Form _BatchActionsForm : select [ name = Action ]
2011-03-23 22:51:00 +13:00
* /
2012-05-11 15:31:12 +12:00
$ ( '#Form_BatchActionsForm select[name=Action]' ) . entwine ( {
onchange : function ( e ) {
2015-06-16 10:52:42 +12:00
var form = $ ( e . target . form ) ,
btn = form . find ( ':submit' ) ,
selected = $ ( e . target ) . val ( ) ;
if ( ! selected || selected == - 1 ) {
2012-05-11 15:31:12 +12:00
btn . attr ( 'disabled' , 'disabled' ) . button ( 'refresh' ) ;
} else {
btn . removeAttr ( 'disabled' ) . button ( 'refresh' ) ;
2015-05-15 11:50:23 +12:00
}
// Refresh selected / enabled nodes
$ ( '#Form_BatchActionsForm' ) . refreshSelected ( ) ;
2012-03-08 01:03:49 +01:00
2012-05-11 15:31:12 +12:00
// TODO Should work by triggering change() along, but doesn't - entwine event bubbling?
this . trigger ( "liszt:updated" ) ;
2012-03-08 01:03:49 +01:00
2012-05-11 15:31:12 +12:00
this . _super ( e ) ;
}
2011-03-23 22:51:00 +13:00
} ) ;
} ) ;
2012-05-11 12:53:40 +12:00
} ) ( jQuery ) ;