2012-05-09 14:14:34 -03:00
< ? php
/**
2012-05-15 11:04:40 -03:00
* This component provides a checkbox which when checked enables drag - and - drop re - ordering of elements displayed in a { @ link GridField }
*
2012-05-09 14:14:34 -03:00
* @ package forms
*/
2012-05-10 11:27:01 -03:00
class GridFieldSortableRows implements GridField_HTMLProvider , GridField_ActionProvider , GridField_DataManipulator {
2012-05-15 11:04:40 -03:00
protected $sortColumn ;
2015-04-07 10:09:06 +12:00
protected $disable_selection = true ;
2013-10-19 14:34:45 -03:00
protected $append_to_top = false ;
2016-02-06 17:00:23 -04:00
protected $update_versioned_stage = null ;
2016-02-09 13:26:34 -04:00
protected $custom_relation_name = null ;
2015-04-07 10:09:06 +12:00
2012-05-10 11:27:01 -03:00
/**
2016-02-06 17:00:23 -04:00
* @ param string $sortColumn Column that should be used to update the sort information
* @ param bool $disableSelection Disable selection on the GridField when dragging
* @ param string $updateVersionStage Name of the versioned stage to update this disabled by default unless this is set
2016-05-03 12:18:40 -03:00
* @ param string $customRelationName Name of the relationship to use , if left null the name is determined from the GridField ' s name
2012-05-10 11:27:01 -03:00
*/
2016-02-09 13:26:34 -04:00
public function __construct ( $sortColumn , $disableSelection = true , $updateVersionStage = null , $customRelationName = null ) {
2012-05-15 11:04:40 -03:00
$this -> sortColumn = $sortColumn ;
2015-04-07 10:09:06 +12:00
$this -> disable_selection = $disableSelection ;
2016-02-06 17:00:23 -04:00
$this -> update_versioned_stage = $updateVersionStage ;
2016-02-09 13:26:34 -04:00
$this -> custom_relation_name = $customRelationName ;
2012-05-10 11:27:01 -03:00
}
2012-05-15 11:04:40 -03:00
/**
* Returns a map where the keys are fragment names and the values are pieces of HTML to add to these fragments .
* @ param GridField $gridField Grid Field Reference
* @ return Array Map where the keys are fragment names and the values are pieces of HTML to add to these fragments .
*/
public function getHTMLFragments ( $gridField ) {
2013-02-12 15:19:25 +13:00
$dataList = $gridField -> getList ();
2013-02-08 19:47:34 -04:00
if ( class_exists ( 'UnsavedRelationList' ) && $dataList instanceof UnsavedRelationList ) {
return array ();
}
2012-05-15 11:04:40 -03:00
$state = $gridField -> State -> GridFieldSortableRows ;
if ( ! is_bool ( $state -> sortableToggle )) {
$state -> sortableToggle = false ;
}
2012-05-15 12:33:00 -03:00
//Ensure user can edit
2012-07-18 17:16:56 -03:00
if ( ! singleton ( $gridField -> getModelClass ()) -> canEdit ()){
2012-05-15 12:33:00 -03:00
return array ();
}
2012-05-15 11:04:40 -03:00
//Sort order toggle
2013-07-18 20:54:33 -03:00
$sortOrderToggle = GridField_FormAction :: create (
2012-05-15 11:04:40 -03:00
$gridField ,
'sortablerows-toggle' ,
2013-07-18 20:54:33 -03:00
'sorttoggle' ,
'sortableRowsToggle' ,
2012-05-15 11:04:40 -03:00
null
) -> addExtraClass ( 'sortablerows-toggle' );
2013-07-18 20:54:33 -03:00
$sortOrderSave = GridField_FormAction :: create (
2012-05-15 11:04:40 -03:00
$gridField ,
2013-07-18 20:54:33 -03:00
'sortablerows-savesort' ,
'savesort' ,
'saveGridRowSort' ,
2012-05-15 11:04:40 -03:00
null
2013-07-18 20:54:33 -03:00
) -> addExtraClass ( 'sortablerows-savesort' );
2012-05-15 11:04:40 -03:00
2013-07-18 20:54:33 -03:00
//Sort to Page Action
$sortToPage = GridField_FormAction :: create (
2012-05-16 22:37:08 -03:00
$gridField ,
'sortablerows-sorttopage' ,
2013-07-18 20:54:33 -03:00
'sorttopage' ,
2012-05-16 22:37:08 -03:00
'sortToPage' ,
null
) -> addExtraClass ( 'sortablerows-sorttopage' );
2012-05-15 11:04:40 -03:00
$data = array ( 'SortableToggle' => $sortOrderToggle ,
2013-07-18 20:54:33 -03:00
'SortOrderSave' => $sortOrderSave ,
2012-05-16 22:37:08 -03:00
'SortToPage' => $sortToPage ,
2015-09-06 20:45:00 -03:00
'Checked' => ( $state -> sortableToggle == true ? ' checked = "checked"' : '' ),
'List' => $dataList );
2012-05-15 11:04:40 -03:00
$forTemplate = new ArrayData ( $data );
//Inject Requirements
2015-04-13 22:17:21 +01:00
$custom = Config :: inst () -> get ( 'GridFieldSortableRows' , 'Base' );
$base = $custom ? : SORTABLE_GRIDFIELD_BASE ;
Requirements :: css ( $base . '/css/GridFieldSortableRows.css' );
Requirements :: javascript ( $base . '/javascript/GridFieldSortableRows.js' );
2012-05-15 11:04:40 -03:00
2015-04-07 10:09:06 +12:00
$args = array ( 'Colspan' => count ( $gridField -> getColumns ()), 'ID' => $gridField -> ID (), 'DisableSelection' => $this -> disable_selection );
2012-05-15 11:04:40 -03:00
2015-09-06 20:45:00 -03:00
$fragments = array ( 'header' => $forTemplate -> renderWith ( 'GridFieldSortableRows' , $args ));
if ( $gridField -> getConfig () -> getComponentByType ( 'GridFieldPaginator' )) {
$fragments [ 'after' ] = $forTemplate -> renderWith ( 'GridFieldSortableRows_paginator' );
}
return $fragments ;
2012-05-15 11:04:40 -03:00
}
/**
2012-05-10 11:27:01 -03:00
* Manipulate the datalist as needed by this grid modifier .
2012-05-15 11:04:40 -03:00
* @ param GridField $gridField Grid Field Reference
* @ param SS_List $dataList Data List to adjust
* @ return DataList Modified Data List
2012-05-10 11:27:01 -03:00
*/
2012-05-15 11:04:40 -03:00
public function getManipulatedData ( GridField $gridField , SS_List $dataList ) {
2012-08-10 12:50:14 -03:00
//Detect and correct items with a sort column value of 0 (push to bottom)
$this -> fixSortColumn ( $gridField , $dataList );
2012-08-10 12:52:39 -03:00
2012-05-17 16:43:29 -03:00
$headerState = $gridField -> State -> GridFieldSortableHeader ;
2012-05-17 22:05:56 -03:00
$state = $gridField -> State -> GridFieldSortableRows ;
2016-11-24 09:59:18 -04:00
if (( ! is_bool ( $state -> sortableToggle ) || $state -> sortableToggle === false ) && $headerState && is_string ( $headerState -> SortColumn ) && is_string ( $headerState -> SortDirection )) {
return $dataList -> sort ( $headerState -> SortColumn , $headerState -> SortDirection );
2012-05-10 11:42:52 -03:00
}
2012-05-17 22:05:56 -03:00
2016-11-24 09:59:18 -04:00
if ( $state -> sortableToggle === true ) {
2012-05-17 16:43:29 -03:00
$gridField -> getConfig () -> removeComponentsByType ( 'GridFieldFilterHeader' );
$gridField -> getConfig () -> removeComponentsByType ( 'GridFieldSortableHeader' );
}
2012-05-15 11:04:40 -03:00
return $dataList -> sort ( $this -> sortColumn );
}
2013-10-19 14:34:45 -03:00
/**
* Sets if new records should be appended to the top or the bottom of the list
* @ param bool $value Boolean true to append to the top false to append to the bottom
* @ return GridFieldSortableRows Returns the current instance
*/
public function setAppendToTop ( $value ) {
$this -> append_to_top = $value ;
return $this ;
}
2016-02-09 13:26:34 -04:00
2015-04-07 10:09:06 +12:00
/**
* @ param bool $value Boolean true to disable selection of table contents false to enable selection
* @ return GridFieldSortableRows Returns the current instance
*/
public function setDisableSelection ( $value ){
$this -> disable_selection = $value ;
return $this ;
}
2016-02-09 13:26:34 -04:00
2016-02-06 17:00:23 -04:00
/**
* Sets the suffix of the versioned stage that should be updated along side the default stage
* @ param string $value Versioned Stage to update this is disabled by default unless this is set
* @ return GridFieldSortableRows Returns the current instance
*/
public function setUpdateVersionedStage ( $value ) {
$this -> update_versioned_stage = $value ;
return $this ;
}
2013-10-19 14:34:45 -03:00
2016-02-09 13:26:34 -04:00
/**
* Sets the name of the relationship to use , by default the name is determined from the GridField ' s name
* @ param string $value Name of the relationship to use , by default the name is determined from the GridField ' s name
* @ return GridFieldSortableRows Returns the current instance
*/
public function setCustomRelationName ( $value ) {
$this -> custom_relation_name = $value ;
return $this ;
}
2012-05-17 22:05:56 -03:00
/**
* Detects and corrects items with a sort column value of 0 , by appending them to the bottom of the list
2012-05-18 21:04:34 -03:00
* @ param GridField $gridField Grid Field Reference
2012-05-17 22:05:56 -03:00
* @ param SS_List $dataList Data List of items to be checked
*/
2012-05-18 21:04:34 -03:00
protected function fixSortColumn ( $gridField , SS_List $dataList ) {
2013-02-08 19:47:34 -04:00
if ( class_exists ( 'UnsavedRelationList' ) && $dataList instanceof UnsavedRelationList ) {
return ;
}
2012-05-17 22:05:56 -03:00
$list = clone $dataList ;
2012-12-28 14:35:06 -04:00
$list = $list -> alterDataQuery ( function ( $query , SS_List $tmplist ) {
2012-12-30 15:11:33 -04:00
$query -> limit ( array ());
return $query ;
});
2012-10-12 09:36:45 -03:00
$many_many = ( $list instanceof ManyManyList );
if ( ! $many_many ) {
$sng = singleton ( $gridField -> getModelClass ());
$fieldType = $sng -> db ( $this -> sortColumn );
2018-03-01 10:36:02 -04:00
if ( ! $fieldType || ! ( strtolower ( $fieldType ) == 'dbint' || is_subclass_of ( $fieldType , 'DBInt' ))) {
2013-03-08 14:15:22 -04:00
if ( is_array ( $fieldType )) {
user_error ( 'Sort column ' . $this -> sortColumn . ' could not be found in ' . $gridField -> getModelClass () . '\'s ancestry' , E_USER_ERROR );
} else {
2018-03-01 10:36:02 -04:00
user_error ( 'Sort column ' . $this -> sortColumn . ' must be an instance of DBInt, column is of type ' . $fieldType , E_USER_ERROR );
2013-03-08 14:15:22 -04:00
}
2012-10-12 09:36:45 -03:00
exit ;
}
}
2012-05-17 22:05:56 -03:00
$max = $list -> Max ( $this -> sortColumn );
2014-08-06 16:34:25 +01:00
$list = $list -> filter ( $this -> sortColumn , 0 ) -> sort ( " Created,ID " );
2013-02-06 19:38:20 -04:00
if ( $list -> Count () > 0 ) {
2012-05-17 22:05:56 -03:00
$owner = $gridField -> Form -> getRecord ();
$sortColumn = $this -> sortColumn ;
$i = 1 ;
2012-10-12 09:36:45 -03:00
if ( $many_many ) {
2016-02-09 13:26:34 -04:00
list ( $parentClass , $componentClass , $parentField , $componentField , $table ) = $owner -> many_many (( ! empty ( $this -> custom_relation_name ) ? $this -> custom_relation_name : $gridField -> getName ()));
$extraFields = $owner -> many_many_extraFields (( ! empty ( $this -> custom_relation_name ) ? $this -> custom_relation_name : $gridField -> getName ()));
2012-10-12 09:36:45 -03:00
2018-03-01 10:36:02 -04:00
if ( ! $extraFields || ! array_key_exists ( $this -> sortColumn , $extraFields ) || ! ( strtolower ( $extraFields [ $this -> sortColumn ]) == 'dbint' || is_subclass_of ( 'DBInt' , $extraFields [ $this -> sortColumn ]))) {
2014-10-16 10:25:23 -03:00
user_error ( 'Sort column ' . $this -> sortColumn . ' must be an Int, column is of type ' . $extraFields [ $this -> sortColumn ], E_USER_ERROR );
2012-10-12 09:36:45 -03:00
exit ;
}
2013-01-23 20:11:22 -04:00
} else {
//Find table containing the sort column
$table = false ;
$class = $gridField -> getModelClass ();
2013-01-31 13:03:33 -05:00
$db = Config :: inst () -> get ( $class , " db " , CONFIG :: UNINHERITED );
if ( ! empty ( $db ) && array_key_exists ( $sortColumn , $db )) {
2013-01-23 20:11:22 -04:00
$table = $class ;
} else {
$classes = ClassInfo :: ancestry ( $class , true );
foreach ( $classes as $class ) {
2013-01-31 13:03:33 -05:00
$db = Config :: inst () -> get ( $class , " db " , CONFIG :: UNINHERITED );
if ( ! empty ( $db ) && array_key_exists ( $sortColumn , $db )) {
2013-01-23 20:11:22 -04:00
$table = $class ;
break ;
}
}
}
if ( $table === false ) {
user_error ( 'Sort column ' . $this -> sortColumn . ' could not be found in ' . $gridField -> getModelClass () . '\'s ancestry' , E_USER_ERROR );
exit ;
}
2013-05-25 20:42:35 -03:00
$baseDataClass = ClassInfo :: baseDataClass ( $gridField -> getModelClass ());
2013-01-23 20:11:22 -04:00
}
//Start transaction if supported
if ( DB :: getConn () -> supportsTransactions ()) {
DB :: getConn () -> transactionStart ();
2012-10-12 09:36:45 -03:00
}
2013-10-19 14:34:45 -03:00
$idCondition = null ;
if ( $this -> append_to_top && ! ( $list instanceof RelationList )) {
2015-05-07 12:44:55 -03:00
$idCondition = '"ID" IN(\'' . implode ( " ',' " , $dataList -> getIDList ()) . '\')' ;
2013-10-19 14:34:45 -03:00
}
2012-10-12 09:36:45 -03:00
2013-10-22 10:36:08 -03:00
if ( $this -> append_to_top ) {
$topIncremented = array ();
}
2012-05-17 22:05:56 -03:00
foreach ( $list as $obj ) {
if ( $many_many ) {
2013-10-19 14:34:45 -03:00
if ( $this -> append_to_top ) {
2013-10-22 10:36:08 -03:00
//Upgrade all the records (including the last inserted from 0 to 1)
2013-09-04 15:05:28 +02:00
DB :: query ( 'UPDATE "' . $table
. '" SET "' . $sortColumn . '" = "' . $sortColumn . '"+1'
2013-10-22 10:36:08 -03:00
. ' WHERE "' . $parentField . '" = ' . $owner -> ID . ( ! empty ( $topIncremented ) ? ' AND "' . $componentField . '" NOT IN(\'' . implode ( '\',\'' , $topIncremented ) . '\')' : '' ));
$topIncremented [] = $obj -> ID ;
2013-10-19 14:34:45 -03:00
} else {
//Append the last record to the bottom
2013-09-04 15:05:28 +02:00
DB :: query ( 'UPDATE "' . $table
. '" SET "' . $sortColumn . '" = ' . ( $max + $i )
. ' WHERE "' . $componentField . '" = ' . $obj -> ID . ' AND "' . $parentField . '" = ' . $owner -> ID );
}
2013-10-19 14:34:45 -03:00
} else if ( $this -> append_to_top ) {
2013-10-22 10:36:08 -03:00
//Upgrade all the records (including the last inserted from 0 to 1)
2013-10-19 14:34:45 -03:00
DB :: query ( 'UPDATE "' . $table
. '" SET "' . $sortColumn . '" = "' . $sortColumn . '"+1'
2013-10-22 10:36:08 -03:00
. ' WHERE ' . ( $list instanceof RelationList ? '"' . $list -> foreignKey . '" = ' . $owner -> ID : $idCondition ) . ( ! empty ( $topIncremented ) ? ' AND "ID" NOT IN(\'' . implode ( '\',\'' , $topIncremented ) . '\')' : '' ));
2013-10-19 14:34:45 -03:00
2016-02-06 17:00:23 -04:00
if ( $this -> update_versioned_stage && class_exists ( $table ) && Object :: has_extension ( $table , 'Versioned' )) {
DB :: query ( 'UPDATE "' . $table . '_' . $this -> update_versioned_stage
. '" SET "' . $sortColumn . '" = "' . $sortColumn . '"+1'
2016-02-13 13:43:39 -04:00
. ' WHERE ' . ( $list instanceof RelationList ? '"' . $list -> foreignKey . '" = ' . $owner -> ID : $idCondition ) . ( ! empty ( $topIncremented ) ? ' AND "ID" NOT IN(\'' . implode ( '\',\'' , $topIncremented ) . '\')' : '' ));
2016-02-06 17:00:23 -04:00
}
2013-10-22 10:36:08 -03:00
$topIncremented [] = $obj -> ID ;
2013-10-19 14:34:45 -03:00
} else {
//Append the last record to the bottom
DB :: query ( 'UPDATE "' . $table
2013-09-04 15:05:28 +02:00
. '" SET "' . $sortColumn . '" = ' . ( $max + $i )
. ' WHERE "ID" = ' . $obj -> ID );
2013-10-19 14:34:45 -03:00
//LastEdited
DB :: query ( 'UPDATE "' . $baseDataClass
2013-09-04 15:05:28 +02:00
. '" SET "LastEdited" = \'' . date ( 'Y-m-d H:i:s' ) . '\''
. ' WHERE "ID" = ' . $obj -> ID );
2016-02-06 17:00:23 -04:00
if ( $this -> update_versioned_stage && class_exists ( $table ) && Object :: has_extension ( $table , 'Versioned' )) {
DB :: query ( 'UPDATE "' . $table . '_' . $this -> update_versioned_stage
. '" SET "' . $sortColumn . '" = ' . ( $max + $i )
. ' WHERE "ID" = ' . $obj -> ID );
if ( Object :: has_extension ( $baseDataClass , 'Versioned' )) {
DB :: query ( 'UPDATE "' . $baseDataClass . '_' . $this -> update_versioned_stage
. '" SET "LastEdited" = \'' . date ( 'Y-m-d H:i:s' ) . '\''
. ' WHERE "ID" = ' . $obj -> ID );
}
}
2012-05-17 22:05:56 -03:00
}
$i ++ ;
}
2017-04-20 10:45:11 -03:00
//Update LastEdited for affected records when using append to top on a many_many relationship
if ( ! $many_many && $this -> append_to_top && count ( $topIncremented ) > 0 ) {
DB :: query ( 'UPDATE "' . $baseDataClass
. '" SET "LastEdited" = \'' . date ( 'Y-m-d H:i:s' ) . '\''
. ' WHERE "ID" IN(\'' . implode ( '\',\'' , $topIncremented ) . '\')' );
if ( $this -> update_versioned_stage && class_exists ( $table ) && Object :: has_extension ( $table , 'Versioned' ) && Object :: has_extension ( $baseDataClass , 'Versioned' )) {
DB :: query ( 'UPDATE "' . $baseDataClass . '_' . $this -> update_versioned_stage
. '" SET "LastEdited" = \'' . date ( 'Y-m-d H:i:s' ) . '\''
. ' WHERE "ID" IN(\'' . implode ( '\',\'' , $topIncremented ) . '\')' );
}
}
2012-05-17 22:05:56 -03:00
//End transaction if supported
if ( DB :: getConn () -> supportsTransactions ()) {
DB :: getConn () -> transactionEnd ();
}
}
}
2012-05-15 11:04:40 -03:00
/**
* Return a list of the actions handled by this action provider .
* @ param GridField $gridField Grid Field Reference
* @ return Array Array with action identifier strings .
*/
public function getActions ( $gridField ) {
2013-07-18 20:54:33 -03:00
return array ( 'saveGridRowSort' , 'sortableRowsToggle' , 'sortToPage' );
2012-05-15 11:04:40 -03:00
}
/**
* Handle an action on the given grid field .
* @ param GridField $gridField Grid Field Reference
* @ param String $actionName Action identifier , see { @ link getActions ()} .
* @ param Array $arguments Arguments relevant for this
* @ param Array $data All form data
*/
public function handleAction ( GridField $gridField , $actionName , $arguments , $data ) {
$state = $gridField -> State -> GridFieldSortableRows ;
if ( ! is_bool ( $state -> sortableToggle )) {
$state -> sortableToggle = false ;
} else if ( $state -> sortableToggle == true ) {
$gridField -> getConfig () -> removeComponentsByType ( 'GridFieldFilterHeader' );
$gridField -> getConfig () -> removeComponentsByType ( 'GridFieldSortableHeader' );
}
if ( $actionName == 'savegridrowsort' ) {
return $this -> saveGridRowSort ( $gridField , $data );
2012-05-16 22:37:08 -03:00
} else if ( $actionName == 'sorttopage' ) {
2012-05-17 09:56:16 -03:00
return $this -> sortToPage ( $gridField , $data );
}
2012-05-15 11:04:40 -03:00
}
/**
* Handles saving of the row sort order
* @ param GridField $gridField Grid Field Reference
* @ param Array $data Data submitted in the request
*/
2012-05-16 22:37:08 -03:00
protected function saveGridRowSort ( GridField $gridField , $data ) {
2013-02-12 15:19:25 +13:00
$dataList = $gridField -> getList ();
2013-02-08 19:47:34 -04:00
if ( class_exists ( 'UnsavedRelationList' ) && $dataList instanceof UnsavedRelationList ) {
user_error ( 'Cannot sort an UnsavedRelationList' , E_USER_ERROR );
return ;
}
2012-05-15 12:33:00 -03:00
if ( ! singleton ( $gridField -> getModelClass ()) -> canEdit ()){
throw new ValidationException ( _t ( 'GridFieldSortableRows.EditPermissionsFailure' , " No edit permissions " ), 0 );
}
2012-09-10 09:40:54 -03:00
if ( empty ( $data [ 'ItemIDs' ])) {
2012-05-15 11:04:40 -03:00
user_error ( 'No items to sort' , E_USER_ERROR );
}
$className = $gridField -> getModelClass ();
$owner = $gridField -> Form -> getRecord ();
2012-05-17 17:04:53 -03:00
$items = clone $gridField -> getList ();
2012-05-15 11:04:40 -03:00
$many_many = ( $items instanceof ManyManyList );
$sortColumn = $this -> sortColumn ;
2012-05-17 09:56:16 -03:00
$pageOffset = 0 ;
if ( $paginator = $gridField -> getConfig () -> getComponentsByType ( 'GridFieldPaginator' ) -> First ()) {
$pageState = $gridField -> State -> GridFieldPaginator ;
2012-08-19 14:43:20 -03:00
if ( $pageState -> currentPage && is_int ( $pageState -> currentPage ) && $pageState -> currentPage > 1 ) {
2012-05-17 09:56:16 -03:00
$pageOffset = $paginator -> getItemsPerPage () * ( $pageState -> currentPage - 1 );
}
}
2012-05-15 11:04:40 -03:00
if ( $many_many ) {
2016-02-09 13:26:34 -04:00
list ( $parentClass , $componentClass , $parentField , $componentField , $table ) = $owner -> many_many (( ! empty ( $this -> custom_relation_name ) ? $this -> custom_relation_name : $gridField -> getName ()));
2013-01-23 20:11:22 -04:00
} else {
//Find table containing the sort column
$table = false ;
2015-09-06 20:45:00 -03:00
$class = $gridField -> getModelClass ();
2013-01-31 13:03:33 -05:00
$db = Config :: inst () -> get ( $class , " db " , CONFIG :: UNINHERITED );
if ( ! empty ( $db ) && array_key_exists ( $sortColumn , $db )) {
2013-01-23 20:11:22 -04:00
$table = $class ;
} else {
$classes = ClassInfo :: ancestry ( $class , true );
foreach ( $classes as $class ) {
2013-01-31 13:03:33 -05:00
$db = Config :: inst () -> get ( $class , " db " , CONFIG :: UNINHERITED );
if ( ! empty ( $db ) && array_key_exists ( $sortColumn , $db )) {
2013-01-23 20:11:22 -04:00
$table = $class ;
break ;
}
}
}
if ( $table === false ) {
user_error ( 'Sort column ' . $this -> sortColumn . ' could not be found in ' . $gridField -> getModelClass () . '\'s ancestry' , E_USER_ERROR );
exit ;
}
2013-05-25 20:42:35 -03:00
$baseDataClass = ClassInfo :: baseDataClass ( $gridField -> getModelClass ());
2012-05-15 11:04:40 -03:00
}
2013-08-04 12:15:37 -03:00
//Event to notify the Controller or owner DataObject before list sort
if ( $owner && $owner instanceof DataObject && method_exists ( $owner , 'onBeforeGridFieldRowSort' )) {
$owner -> onBeforeGridFieldRowSort ( clone $items );
} else if ( Controller :: has_curr () && Controller :: curr () instanceof ModelAdmin && method_exists ( Controller :: curr (), 'onBeforeGridFieldRowSort' )) {
Controller :: curr () -> onBeforeGridFieldRowSort ( clone $items );
}
2012-05-15 12:33:00 -03:00
//Start transaction if supported
if ( DB :: getConn () -> supportsTransactions ()) {
DB :: getConn () -> transactionStart ();
}
2012-05-17 09:56:16 -03:00
2013-08-04 12:15:37 -03:00
//Perform sorting
2012-09-10 13:38:00 +10:00
$ids = explode ( ',' , $data [ 'ItemIDs' ]);
for ( $sort = 0 ; $sort < count ( $ids ); $sort ++ ) {
$id = intval ( $ids [ $sort ]);
2012-05-15 11:04:40 -03:00
if ( $many_many ) {
DB :: query ( 'UPDATE "' . $table
2012-05-16 22:37:08 -03:00
. '" SET "' . $sortColumn . '" = ' . (( $sort + 1 ) + $pageOffset )
2012-05-15 11:04:40 -03:00
. ' WHERE "' . $componentField . '" = ' . $id . ' AND "' . $parentField . '" = ' . $owner -> ID );
} else {
2013-01-23 20:11:22 -04:00
DB :: query ( 'UPDATE "' . $table
2013-05-25 13:35:55 -03:00
. '" SET "' . $sortColumn . '" = ' . (( $sort + 1 ) + $pageOffset )
. ' WHERE "ID" = ' . $id );
2013-05-25 20:42:35 -03:00
DB :: query ( 'UPDATE "' . $baseDataClass
2013-05-25 13:35:55 -03:00
. '" SET "LastEdited" = \'' . date ( 'Y-m-d H:i:s' ) . '\''
2013-01-23 20:11:22 -04:00
. ' WHERE "ID" = ' . $id );
2016-02-06 17:00:23 -04:00
if ( $this -> update_versioned_stage && class_exists ( $table ) && Object :: has_extension ( $table , 'Versioned' )) {
DB :: query ( 'UPDATE "' . $table . '_' . $this -> update_versioned_stage
. '" SET "' . $sortColumn . '" = ' . (( $sort + 1 ) + $pageOffset )
. ' WHERE "ID" = ' . $id );
if ( Object :: has_extension ( $baseDataClass , 'Versioned' )) {
DB :: query ( 'UPDATE "' . $baseDataClass . '_' . $this -> update_versioned_stage
. '" SET "LastEdited" = \'' . date ( 'Y-m-d H:i:s' ) . '\''
. ' WHERE "ID" = ' . $id );
}
}
2012-05-15 11:04:40 -03:00
}
}
2012-05-15 12:33:00 -03:00
2013-08-04 12:15:37 -03:00
2012-05-15 12:33:00 -03:00
//End transaction if supported
if ( DB :: getConn () -> supportsTransactions ()) {
DB :: getConn () -> transactionEnd ();
}
2013-08-04 12:15:37 -03:00
//Event to notify the Controller or owner DataObject after list sort
if ( $owner && $owner instanceof DataObject && method_exists ( $owner , 'onAfterGridFieldRowSort' )) {
$owner -> onAfterGridFieldRowSort ( clone $items );
} else if ( Controller :: has_curr () && Controller :: curr () instanceof ModelAdmin && method_exists ( Controller :: curr (), 'onAfterGridFieldRowSort' )) {
Controller :: curr () -> onAfterGridFieldRowSort ( clone $items );
}
2012-05-15 11:04:40 -03:00
}
2012-05-16 22:37:08 -03:00
/**
* Handles sorting across pages
* @ param GridField $gridField Grid Field Reference
* @ param Array $data Data submitted in the request
*/
protected function sortToPage ( GridField $gridField , $data ) {
2012-05-17 09:56:16 -03:00
if ( ! $paginator = $gridField -> getConfig () -> getComponentsByType ( 'GridFieldPaginator' ) -> First ()) {
user_error ( 'Paginator not detected' , E_USER_ERROR );
}
if ( empty ( $data [ 'ItemID' ])) {
2012-05-16 22:37:08 -03:00
user_error ( 'No item to sort' , E_USER_ERROR );
}
2012-05-17 09:56:16 -03:00
if ( empty ( $data [ 'Target' ])) {
2012-05-16 22:37:08 -03:00
user_error ( 'No target page' , E_USER_ERROR );
}
2012-05-17 09:56:16 -03:00
$className = $gridField -> getModelClass ();
2012-05-16 22:37:08 -03:00
$owner = $gridField -> Form -> getRecord ();
2012-05-17 17:04:53 -03:00
$items = clone $gridField -> getList ();
2012-05-16 22:37:08 -03:00
$many_many = ( $items instanceof ManyManyList );
$sortColumn = $this -> sortColumn ;
2012-05-17 09:56:16 -03:00
$targetItem = $items -> byID ( intval ( $data [ 'ItemID' ]));
if ( ! $targetItem ) {
user_error ( 'Target item not found' , E_USER_ERROR );
}
$currentPage = 1 ;
$pageState = $gridField -> State -> GridFieldPaginator ;
if ( $pageState -> currentPage && $pageState -> currentPage > 1 ) {
$currentPage = $pageState -> currentPage ;
}
2012-05-16 22:37:08 -03:00
if ( $many_many ) {
2016-02-09 13:26:34 -04:00
list ( $parentClass , $componentClass , $parentField , $componentField , $table ) = $owner -> many_many (( ! empty ( $this -> custom_relation_name ) ? $this -> custom_relation_name : $gridField -> getName ()));
2012-05-16 22:37:08 -03:00
}
2012-05-17 09:56:16 -03:00
2012-05-18 10:50:09 -03:00
if ( $data [ 'Target' ] == 'previouspage' ) {
2013-07-10 14:20:16 -03:00
$items = $items -> limit ( $paginator -> getItemsPerPage () + 1 , ( $paginator -> getItemsPerPage () * ( $currentPage - 1 )) - 1 );
2012-05-17 09:56:16 -03:00
} else if ( $data [ 'Target' ] == 'nextpage' ) {
2013-07-10 14:20:16 -03:00
$items = $items -> limit ( $paginator -> getItemsPerPage () + 1 , $paginator -> getItemsPerPage () * ( $currentPage - 1 ));
2012-05-17 09:56:16 -03:00
} else {
user_error ( 'Not implemented: ' . $data [ 'Target' ], E_USER_ERROR );
}
2013-07-10 14:20:16 -03:00
$sortPositions = $items -> column ( $sortColumn );
2012-05-17 09:56:16 -03:00
2013-08-04 12:15:37 -03:00
//Event to notify the Controller or owner DataObject before list sort
if ( $owner && $owner instanceof DataObject && method_exists ( $owner , 'onBeforeGridFieldPageSort' )) {
$owner -> onBeforeGridFieldPageSort ( clone $items );
} else if ( Controller :: has_curr () && Controller :: curr () instanceof ModelAdmin && method_exists ( Controller :: curr (), 'onBeforeGridFieldPageSort' )) {
Controller :: curr () -> onBeforeGridFieldPageSort ( clone $items );
}
2016-02-06 17:00:23 -04:00
//Find the sort column
if ( $this -> update_versioned_stage && Object :: has_extension ( $className , 'Versioned' )) {
$table = false ;
$classes = ClassInfo :: ancestry ( $className , true );
foreach ( $classes as $class ) {
2017-04-18 09:31:47 -03:00
$db = Config :: inst () -> get ( $class , " db " , CONFIG :: UNINHERITED );
2016-02-06 17:00:23 -04:00
if ( ! empty ( $db ) && array_key_exists ( $sortColumn , $db )) {
$table = $class ;
break ;
}
}
if ( $table === false ) {
user_error ( 'Sort column ' . $this -> sortColumn . ' could not be found in ' . $gridField -> getModelClass () . '\'s ancestry' , E_USER_ERROR );
exit ;
}
}
2012-05-18 10:50:09 -03:00
//Start transaction if supported
if ( DB :: getConn () -> supportsTransactions ()) {
DB :: getConn () -> transactionStart ();
}
if ( $data [ 'Target' ] == 'previouspage' ) {
2012-05-17 09:56:16 -03:00
if ( $many_many ) {
DB :: query ( 'UPDATE "' . $table
2012-05-18 10:50:09 -03:00
. '" SET "' . $sortColumn . '" = ' . $sortPositions [ 0 ]
. ' WHERE "' . $componentField . '" = ' . $targetItem -> ID . ' AND "' . $parentField . '" = ' . $owner -> ID );
2012-05-17 09:56:16 -03:00
} else {
2012-05-18 10:50:09 -03:00
$targetItem -> $sortColumn = $sortPositions [ 0 ];
$targetItem -> write ();
2016-02-06 17:00:23 -04:00
if ( $this -> update_versioned_stage && Object :: has_extension ( $className , 'Versioned' )) {
DB :: query ( 'UPDATE "' . $table . '_' . $this -> update_versioned_stage
. '" SET "' . $sortColumn . '" = ' . $sortPositions [ 0 ]
. ' WHERE "ID" = ' . $targetItem -> ID );
}
2012-05-17 09:56:16 -03:00
}
2012-05-18 10:50:09 -03:00
$i = 1 ;
foreach ( $items as $obj ) {
if ( $obj -> ID == $targetItem -> ID ) {
continue ;
}
if ( $many_many ) {
DB :: query ( 'UPDATE "' . $table
. '" SET "' . $sortColumn . '" = ' . $sortPositions [ $i ]
. ' WHERE "' . $componentField . '" = ' . $obj -> ID . ' AND "' . $parentField . '" = ' . $owner -> ID );
} else {
$obj -> $sortColumn = $sortPositions [ $i ];
$obj -> write ();
2016-02-06 17:00:23 -04:00
if ( $this -> update_versioned_stage && Object :: has_extension ( $className , 'Versioned' )) {
DB :: query ( 'UPDATE "' . $table . '_' . $this -> update_versioned_stage
. '" SET "' . $sortColumn . '" = ' . $sortPositions [ $i ]
. ' WHERE "ID" = ' . $obj -> ID );
}
2012-05-18 10:50:09 -03:00
}
$i ++ ;
}
} else {
2012-05-17 09:56:16 -03:00
if ( $many_many ) {
DB :: query ( 'UPDATE "' . $table
2012-05-18 10:50:09 -03:00
. '" SET "' . $sortColumn . '" = ' . $sortPositions [ count ( $sortPositions ) - 1 ]
2012-05-17 09:56:16 -03:00
. ' WHERE "' . $componentField . '" = ' . $targetItem -> ID . ' AND "' . $parentField . '" = ' . $owner -> ID );
} else {
2012-05-18 10:50:09 -03:00
$targetItem -> $sortColumn = $sortPositions [ count ( $sortPositions ) - 1 ];
2012-05-17 09:56:16 -03:00
$targetItem -> write ();
2016-02-06 17:00:23 -04:00
if ( $this -> update_versioned_stage && Object :: has_extension ( $className , 'Versioned' )) {
DB :: query ( 'UPDATE "' . $table . '_' . $this -> update_versioned_stage
. '" SET "' . $sortColumn . '" = ' . $sortPositions [ count ( $sortPositions ) - 1 ]
. ' WHERE "ID" = ' . $targetItem -> ID );
}
2012-05-17 09:56:16 -03:00
}
2012-05-18 10:50:09 -03:00
$i = 0 ;
foreach ( $items as $obj ) {
if ( $obj -> ID == $targetItem -> ID ) {
continue ;
}
if ( $many_many ) {
DB :: query ( 'UPDATE "' . $table
. '" SET "' . $sortColumn . '" = ' . $sortPositions [ $i ]
. ' WHERE "' . $componentField . '" = ' . $obj -> ID . ' AND "' . $parentField . '" = ' . $owner -> ID );
} else {
$obj -> $sortColumn = $sortPositions [ $i ];
$obj -> write ();
2016-02-06 17:00:23 -04:00
if ( $this -> update_versioned_stage && Object :: has_extension ( $className , 'Versioned' )) {
DB :: query ( 'UPDATE "' . $table . '_' . $this -> update_versioned_stage
. '" SET "' . $sortColumn . '" = ' . $sortPositions [ $i ]
. ' WHERE "ID" = ' . $obj -> ID );
}
2012-05-18 10:50:09 -03:00
}
$i ++ ;
2012-05-17 09:56:16 -03:00
}
}
2012-05-18 10:50:09 -03:00
//End transaction if supported
if ( DB :: getConn () -> supportsTransactions ()) {
DB :: getConn () -> transactionEnd ();
}
2013-08-04 12:15:37 -03:00
//Event to notify the Controller or owner DataObject after list sort
if ( $owner && $owner instanceof DataObject && method_exists ( $owner , 'onAfterGridFieldPageSort' )) {
$owner -> onAfterGridFieldPageSort ( clone $items );
} else if ( Controller :: has_curr () && Controller :: curr () instanceof ModelAdmin && method_exists ( Controller :: curr (), 'onAfterGridFieldPageSort' )) {
Controller :: curr () -> onAfterGridFieldPageSort ( clone $items );
}
2012-05-17 09:56:16 -03:00
}
2012-05-09 14:14:34 -03:00
}