mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
Clean up GridField
This commit is contained in:
parent
0653ba9630
commit
4ddc2d231c
@ -3,12 +3,11 @@
|
|||||||
/**
|
/**
|
||||||
* Displays a {@link SS_List} in a grid format.
|
* Displays a {@link SS_List} in a grid format.
|
||||||
*
|
*
|
||||||
* GridField is a field that takes an SS_List and displays it in an table with rows
|
* GridField is a field that takes an SS_List and displays it in an table with rows and columns.
|
||||||
* and columns. It reminds of the old TableFields but works with SS_List types
|
* It reminds of the old TableFields but works with SS_List types and only loads the necessary
|
||||||
* and only loads the necessary rows from the list.
|
* rows from the list.
|
||||||
*
|
*
|
||||||
* The minimum configuration is to pass in name and title of the field and a
|
* The minimum configuration is to pass in name and title of the field and a SS_List.
|
||||||
* SS_List.
|
|
||||||
*
|
*
|
||||||
* <code>
|
* <code>
|
||||||
* $gridField = new GridField('ExampleGrid', 'Example grid', new DataList('Page'));
|
* $gridField = new GridField('ExampleGrid', 'Example grid', new DataList('Page'));
|
||||||
@ -20,45 +19,44 @@
|
|||||||
* @subpackage fields-gridfield
|
* @subpackage fields-gridfield
|
||||||
*/
|
*/
|
||||||
class GridField extends FormField {
|
class GridField extends FormField {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
private static $allowed_actions = array(
|
private static $allowed_actions = array(
|
||||||
'index',
|
'index',
|
||||||
'gridFieldAlterAction'
|
'gridFieldAlterAction',
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The datasource
|
* Data source.
|
||||||
*
|
*
|
||||||
* @var SS_List
|
* @var SS_List
|
||||||
*/
|
*/
|
||||||
protected $list = null;
|
protected $list = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The classname of the DataObject that the GridField will display. Defaults to the value of $this->list->dataClass
|
* Class name of the DataObject that the GridField will display.
|
||||||
|
*
|
||||||
|
* Defaults to the value of $this->list->dataClass.
|
||||||
*
|
*
|
||||||
* @var string
|
* @var string
|
||||||
*/
|
*/
|
||||||
protected $modelClassName = '';
|
protected $modelClassName = '';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* the current state of the GridField
|
* Current state of the GridField.
|
||||||
*
|
*
|
||||||
* @var GridState
|
* @var GridState
|
||||||
*/
|
*/
|
||||||
protected $state = null;
|
protected $state = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
|
||||||
* @var GridFieldConfig
|
* @var GridFieldConfig
|
||||||
*/
|
*/
|
||||||
protected $config = null;
|
protected $config = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The components list
|
* Components list.
|
||||||
*
|
*
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
@ -66,14 +64,15 @@ class GridField extends FormField {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Internal dispatcher for column handlers.
|
* Internal dispatcher for column handlers.
|
||||||
* Keys are column names and values are GridField_ColumnProvider objects
|
*
|
||||||
|
* Keys are column names and values are GridField_ColumnProvider objects.
|
||||||
*
|
*
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
protected $columnDispatch = null;
|
protected $columnDispatch = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Map of callbacks for custom data fields
|
* Map of callbacks for custom data fields.
|
||||||
*
|
*
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
@ -85,8 +84,6 @@ class GridField extends FormField {
|
|||||||
protected $name = '';
|
protected $name = '';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new GridField field
|
|
||||||
*
|
|
||||||
* @param string $name
|
* @param string $name
|
||||||
* @param string $title
|
* @param string $title
|
||||||
* @param SS_List $dataList
|
* @param SS_List $dataList
|
||||||
@ -94,13 +91,18 @@ class GridField extends FormField {
|
|||||||
*/
|
*/
|
||||||
public function __construct($name, $title = null, SS_List $dataList = null, GridFieldConfig $config = null) {
|
public function __construct($name, $title = null, SS_List $dataList = null, GridFieldConfig $config = null) {
|
||||||
parent::__construct($name, $title, null);
|
parent::__construct($name, $title, null);
|
||||||
|
|
||||||
$this->name = $name;
|
$this->name = $name;
|
||||||
|
|
||||||
if($dataList) {
|
if($dataList) {
|
||||||
$this->setList($dataList);
|
$this->setList($dataList);
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->setConfig($config ?: GridFieldConfig_Base::create());
|
if(!$config) {
|
||||||
|
$config = GridFieldConfig_Base::create();
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->setConfig($config);
|
||||||
|
|
||||||
$this->config->addComponent(new GridState_Component());
|
$this->config->addComponent(new GridState_Component());
|
||||||
$this->state = new GridState($this);
|
$this->state = new GridState($this);
|
||||||
@ -108,44 +110,58 @@ class GridField extends FormField {
|
|||||||
$this->addExtraClass('ss-gridfield');
|
$this->addExtraClass('ss-gridfield');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param SS_HTTPRequest $request
|
||||||
|
*
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
public function index($request) {
|
public function index($request) {
|
||||||
return $this->gridFieldAlterAction(array(), $this->getForm(), $request);
|
return $this->gridFieldAlterAction(array(), $this->getForm(), $request);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the modelClass (data object) that this field will get it column headers from.
|
* Set the modelClass (data object) that this field will get it column headers from.
|
||||||
* If no $displayFields has been set, the displayfields will be fetched from
|
*
|
||||||
* this modelclass $summary_fields
|
* If no $displayFields has been set, the display fields will be $summary_fields.
|
||||||
|
*
|
||||||
|
* @see GridFieldDataColumns::getDisplayFields()
|
||||||
*
|
*
|
||||||
* @param string $modelClassName
|
* @param string $modelClassName
|
||||||
*
|
*
|
||||||
* @see GridFieldDataColumns::getDisplayFields()
|
* @return $this
|
||||||
*/
|
*/
|
||||||
public function setModelClass($modelClassName) {
|
public function setModelClass($modelClassName) {
|
||||||
$this->modelClassName = $modelClassName;
|
$this->modelClassName = $modelClassName;
|
||||||
|
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a data class that is a DataObject type that this GridField should look like.
|
* Returns a data class that is a DataObject type that this GridField should look like.
|
||||||
*
|
*
|
||||||
* @throws Exception
|
|
||||||
* @return string
|
* @return string
|
||||||
|
*
|
||||||
|
* @throws LogicException
|
||||||
*/
|
*/
|
||||||
public function getModelClass() {
|
public function getModelClass() {
|
||||||
if($this->modelClassName) return $this->modelClassName;
|
if($this->modelClassName) {
|
||||||
if($this->list && method_exists($this->list, 'dataClass')) {
|
return $this->modelClassName;
|
||||||
$class = $this->list->dataClass();
|
|
||||||
if($class) return $class;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
throw new LogicException('GridField doesn\'t have a modelClassName,'
|
if($this->list && method_exists($this->list, 'dataClass')) {
|
||||||
. ' so it doesn\'t know the columns of this grid.');
|
$class = $this->list->dataClass();
|
||||||
|
|
||||||
|
if($class) {
|
||||||
|
return $class;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new LogicException(
|
||||||
|
'GridField doesn\'t have a modelClassName, so it doesn\'t know the columns of this grid.'
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the GridFieldConfig
|
|
||||||
*
|
|
||||||
* @return GridFieldConfig
|
* @return GridFieldConfig
|
||||||
*/
|
*/
|
||||||
public function getConfig() {
|
public function getConfig() {
|
||||||
@ -155,61 +171,69 @@ class GridField extends FormField {
|
|||||||
/**
|
/**
|
||||||
* @param GridFieldConfig $config
|
* @param GridFieldConfig $config
|
||||||
*
|
*
|
||||||
* @return GridField
|
* @return $this
|
||||||
*/
|
*/
|
||||||
public function setConfig(GridFieldConfig $config) {
|
public function setConfig(GridFieldConfig $config) {
|
||||||
$this->config = $config;
|
$this->config = $config;
|
||||||
|
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return ArrayList
|
||||||
|
*/
|
||||||
public function getComponents() {
|
public function getComponents() {
|
||||||
return $this->config->getComponents();
|
return $this->config->getComponents();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Cast a arbitrary value with the help of a castingDefintion
|
* Cast an arbitrary value with the help of a $castingDefinition.
|
||||||
*
|
|
||||||
* @param $value
|
|
||||||
* @param $castingDefinition
|
|
||||||
*
|
*
|
||||||
* @todo refactor this into GridFieldComponent
|
* @todo refactor this into GridFieldComponent
|
||||||
|
*
|
||||||
|
* @param mixed $value
|
||||||
|
* @param string|array $castingDefinition
|
||||||
|
*
|
||||||
|
* @return mixed
|
||||||
*/
|
*/
|
||||||
public function getCastedValue($value, $castingDefinition) {
|
public function getCastedValue($value, $castingDefinition) {
|
||||||
|
$castingParams = array();
|
||||||
|
|
||||||
if(is_array($castingDefinition)) {
|
if(is_array($castingDefinition)) {
|
||||||
$castingParams = $castingDefinition;
|
$castingParams = $castingDefinition;
|
||||||
array_shift($castingParams);
|
array_shift($castingParams);
|
||||||
$castingDefinition = array_shift($castingDefinition);
|
$castingDefinition = array_shift($castingDefinition);
|
||||||
} else {
|
|
||||||
$castingParams = array();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if(strpos($castingDefinition, '->') === false) {
|
if(strpos($castingDefinition, '->') === false) {
|
||||||
$castingFieldType = $castingDefinition;
|
$castingFieldType = $castingDefinition;
|
||||||
$castingField = DBField::create_field($castingFieldType, $value);
|
$castingField = DBField::create_field($castingFieldType, $value);
|
||||||
$value = call_user_func_array(array($castingField, 'XML'), $castingParams);
|
|
||||||
} else {
|
return call_user_func_array(array($castingField, 'XML'), $castingParams);
|
||||||
$fieldTypeParts = explode('->', $castingDefinition);
|
|
||||||
$castingFieldType = $fieldTypeParts[0];
|
|
||||||
$castingMethod = $fieldTypeParts[1];
|
|
||||||
$castingField = DBField::create_field($castingFieldType, $value);
|
|
||||||
$value = call_user_func_array(array($castingField, $castingMethod), $castingParams);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return $value;
|
list($castingFieldType, $castingMethod) = explode('->', $castingDefinition);
|
||||||
|
|
||||||
|
$castingField = DBField::create_field($castingFieldType, $value);
|
||||||
|
|
||||||
|
return call_user_func_array(array($castingField, $castingMethod), $castingParams);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the datasource
|
* Set the data source.
|
||||||
*
|
*
|
||||||
* @param SS_List $list
|
* @param SS_List $list
|
||||||
|
*
|
||||||
|
* @return $this
|
||||||
*/
|
*/
|
||||||
public function setList(SS_List $list) {
|
public function setList(SS_List $list) {
|
||||||
$this->list = $list;
|
$this->list = $list;
|
||||||
|
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the datasource
|
* Get the data source.
|
||||||
*
|
*
|
||||||
* @return SS_List
|
* @return SS_List
|
||||||
*/
|
*/
|
||||||
@ -218,26 +242,28 @@ class GridField extends FormField {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the datasource after applying the {@link GridField_DataManipulator}s to it.
|
* Get the data source after applying every {@link GridField_DataManipulator} to it.
|
||||||
*
|
*
|
||||||
* @return SS_List
|
* @return SS_List
|
||||||
*/
|
*/
|
||||||
public function getManipulatedList() {
|
public function getManipulatedList() {
|
||||||
$list = $this->getList();
|
$list = $this->getList();
|
||||||
|
|
||||||
foreach($this->getComponents() as $item) {
|
foreach($this->getComponents() as $item) {
|
||||||
if($item instanceof GridField_DataManipulator) {
|
if($item instanceof GridField_DataManipulator) {
|
||||||
$list = $item->getManipulatedData($this, $list);
|
$list = $item->getManipulatedData($this, $list);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return $list;
|
return $list;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the current GridState_Data or the GridState
|
* Get the current GridState_Data or the GridState.
|
||||||
*
|
*
|
||||||
* @param bool $getData - flag for returning the GridState_Data or the GridState
|
* @param bool $getData
|
||||||
*
|
*
|
||||||
* @return GridState_data|GridState
|
* @return GridState_Data|GridState
|
||||||
*/
|
*/
|
||||||
public function getState($getData = true) {
|
public function getState($getData = true) {
|
||||||
if($getData) {
|
if($getData) {
|
||||||
@ -248,7 +274,9 @@ class GridField extends FormField {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the whole gridfield rendered with all the attached components
|
* Returns the whole gridfield rendered with all the attached components.
|
||||||
|
*
|
||||||
|
* @param array $properties
|
||||||
*
|
*
|
||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
@ -264,88 +292,115 @@ class GridField extends FormField {
|
|||||||
Requirements::javascript(THIRDPARTY_DIR . '/jquery-entwine/dist/jquery.entwine-dist.js');
|
Requirements::javascript(THIRDPARTY_DIR . '/jquery-entwine/dist/jquery.entwine-dist.js');
|
||||||
Requirements::javascript(FRAMEWORK_DIR . '/javascript/GridField.js');
|
Requirements::javascript(FRAMEWORK_DIR . '/javascript/GridField.js');
|
||||||
|
|
||||||
// Get columns
|
|
||||||
$columns = $this->getColumns();
|
$columns = $this->getColumns();
|
||||||
|
|
||||||
// Get data
|
|
||||||
$list = $this->getManipulatedList();
|
$list = $this->getManipulatedList();
|
||||||
|
|
||||||
// Render headers, footers, etc
|
|
||||||
$content = array(
|
$content = array(
|
||||||
"before" => "",
|
'before' => '',
|
||||||
"after" => "",
|
'after' => '',
|
||||||
"header" => "",
|
'header' => '',
|
||||||
"footer" => "",
|
'footer' => '',
|
||||||
);
|
);
|
||||||
|
|
||||||
foreach($this->getComponents() as $item) {
|
foreach($this->getComponents() as $item) {
|
||||||
if($item instanceof GridField_HTMLProvider) {
|
if($item instanceof GridField_HTMLProvider) {
|
||||||
$fragments = $item->getHTMLFragments($this);
|
$fragments = $item->getHTMLFragments($this);
|
||||||
if($fragments) foreach($fragments as $k => $v) {
|
|
||||||
$k = strtolower($k);
|
if($fragments) {
|
||||||
if(!isset($content[$k])) $content[$k] = "";
|
foreach($fragments as $fragmentKey => $fragmentValue) {
|
||||||
$content[$k] .= $v . "\n";
|
$fragmentKey = strtolower($fragmentKey);
|
||||||
|
|
||||||
|
if(!isset($content[$fragmentKey])) {
|
||||||
|
$content[$fragmentKey] = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
$content[$fragmentKey] .= $fragmentValue . "\n";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach($content as $k => $v) {
|
foreach($content as $contentKey => $contentValue) {
|
||||||
$content[$k] = trim($v);
|
$content[$contentKey] = trim($contentValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Replace custom fragments and check which fragments are defined
|
// Replace custom fragments and check which fragments are defined. Circular dependencies
|
||||||
// Nested dependencies are handled by deferring the rendering of any content item that
|
// are detected by disallowing any item to be deferred more than 5 times.
|
||||||
// Circular dependencies are detected by disallowing any item to be deferred more than 5 times
|
|
||||||
// It's a fairly crude algorithm but it works
|
$fragmentDefined = array(
|
||||||
|
'header' => true,
|
||||||
|
'footer' => true,
|
||||||
|
'before' => true,
|
||||||
|
'after' => true,
|
||||||
|
);
|
||||||
|
|
||||||
$fragmentDefined = array('header' => true, 'footer' => true, 'before' => true, 'after' => true);
|
|
||||||
reset($content);
|
reset($content);
|
||||||
while(list($k, $v) = each($content)) {
|
|
||||||
if(preg_match_all('/\$DefineFragment\(([a-z0-9\-_]+)\)/i', $v, $matches)) {
|
while(list($contentKey, $contentValue) = each($content)) {
|
||||||
|
if(preg_match_all('/\$DefineFragment\(([a-z0-9\-_]+)\)/i', $contentValue, $matches)) {
|
||||||
foreach($matches[1] as $match) {
|
foreach($matches[1] as $match) {
|
||||||
$fragmentName = strtolower($match);
|
$fragmentName = strtolower($match);
|
||||||
$fragmentDefined[$fragmentName] = true;
|
$fragmentDefined[$fragmentName] = true;
|
||||||
$fragment = isset($content[$fragmentName]) ? $content[$fragmentName] : "";
|
|
||||||
|
|
||||||
// If the fragment still has a fragment definition in it, when we should defer this item until
|
$fragment = '';
|
||||||
// later.
|
|
||||||
|
if(isset($content[$fragmentName])) {
|
||||||
|
$fragment = $content[$fragmentName];
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the fragment still has a fragment definition in it, when we should defer
|
||||||
|
// this item until later.
|
||||||
|
|
||||||
if(preg_match('/\$DefineFragment\(([a-z0-9\-_]+)\)/i', $fragment, $matches)) {
|
if(preg_match('/\$DefineFragment\(([a-z0-9\-_]+)\)/i', $fragment, $matches)) {
|
||||||
// If we've already deferred this fragment, then we have a circular dependency
|
if(isset($fragmentDeferred[$contentKey]) && $fragmentDeferred[$contentKey] > 5) {
|
||||||
if(isset($fragmentDeferred[$k]) && $fragmentDeferred[$k] > 5) {
|
throw new LogicException(sprintf(
|
||||||
throw new LogicException("GridField HTML fragment '$fragmentName' and '$matches[1]' " .
|
'GridField HTML fragment "%s" and "%s" appear to have a circular dependency.',
|
||||||
"appear to have a circular dependency.");
|
$fragmentName,
|
||||||
|
$matches[1]
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise we can push to the end of the content array
|
unset($content[$contentKey]);
|
||||||
unset($content[$k]);
|
|
||||||
$content[$k] = $v;
|
$content[$contentKey] = $contentValue;
|
||||||
if(!isset($fragmentDeferred[$k])) {
|
|
||||||
$fragmentDeferred[$k] = 1;
|
if(!isset($fragmentDeferred[$contentKey])) {
|
||||||
} else {
|
$fragmentDeferred[$contentKey] = 0;
|
||||||
$fragmentDeferred[$k]++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$fragmentDeferred[$contentKey]++;
|
||||||
|
|
||||||
break;
|
break;
|
||||||
} else {
|
} else {
|
||||||
$content[$k] = preg_replace('/\$DefineFragment\(' . $fragmentName . '\)/i', $fragment,
|
$content[$contentKey] = preg_replace(
|
||||||
$content[$k]);
|
sprintf('/\$DefineFragment\(%s\)/i', $fragmentName),
|
||||||
|
$fragment,
|
||||||
|
$content[$contentKey]
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check for any undefined fragments, and if so throw an exception
|
// Check for any undefined fragments, and if so throw an exception.
|
||||||
// While we're at it, trim whitespace off the elements
|
// While we're at it, trim whitespace off the elements.
|
||||||
foreach($content as $k => $v) {
|
|
||||||
if(empty($fragmentDefined[$k])) {
|
foreach($content as $contentKey => $contentValue) {
|
||||||
throw new LogicException("GridField HTML fragment '$k' was given content,"
|
if(empty($fragmentDefined[$contentKey])) {
|
||||||
. " but not defined. Perhaps there is a supporting GridField component you need to add?");
|
throw new LogicException(sprintf(
|
||||||
|
'GridField HTML fragment "%s" was given content, but not defined. Perhaps there is a supporting GridField component you need to add?',
|
||||||
|
$contentKey
|
||||||
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$total = count($list);
|
$total = count($list);
|
||||||
|
|
||||||
if($total > 0) {
|
if($total > 0) {
|
||||||
$rows = array();
|
$rows = array();
|
||||||
foreach($list as $idx => $record) {
|
|
||||||
|
foreach($list as $index => $record) {
|
||||||
if($record->hasMethod('canView') && !$record->canView()) {
|
if($record->hasMethod('canView') && !$record->canView()) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -355,58 +410,80 @@ class GridField extends FormField {
|
|||||||
foreach($this->getColumns() as $column) {
|
foreach($this->getColumns() as $column) {
|
||||||
$colContent = $this->getColumnContent($record, $column);
|
$colContent = $this->getColumnContent($record, $column);
|
||||||
|
|
||||||
// A return value of null means this columns should be skipped altogether.
|
// Null means this columns should be skipped altogether.
|
||||||
|
|
||||||
if($colContent === null) {
|
if($colContent === null) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
$colAttributes = $this->getColumnAttributes($record, $column);
|
$colAttributes = $this->getColumnAttributes($record, $column);
|
||||||
|
|
||||||
$rowContent .= $this->newCell($total, $idx, $record, $colAttributes, $colContent);
|
$rowContent .= $this->newCell(
|
||||||
|
$total,
|
||||||
|
$index,
|
||||||
|
$record,
|
||||||
|
$colAttributes,
|
||||||
|
$colContent
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
$rowAttributes = $this->getRowAttributes($total, $idx, $record);
|
$rowAttributes = $this->getRowAttributes($total, $index, $record);
|
||||||
|
|
||||||
$rows[] = $this->newRow($total, $idx, $record, $rowAttributes, $rowContent);
|
$rows[] = $this->newRow($total, $index, $record, $rowAttributes, $rowContent);
|
||||||
}
|
}
|
||||||
$content['body'] = implode("\n", $rows);
|
$content['body'] = implode("\n", $rows);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Display a message when the grid field is empty
|
// Display a message when the grid field is empty.
|
||||||
if(!(isset($content['body']) && $content['body'])) {
|
|
||||||
$content['body'] = FormField::create_tag(
|
if(empty($content['body'])) {
|
||||||
'tr',
|
$cell = FormField::create_tag(
|
||||||
array("class" => 'ss-gridfield-item ss-gridfield-no-items'),
|
|
||||||
FormField::create_tag(
|
|
||||||
'td',
|
'td',
|
||||||
array('colspan' => count($columns)),
|
array(
|
||||||
|
'colspan' => count($columns),
|
||||||
|
),
|
||||||
_t('GridField.NoItemsFound', 'No items found')
|
_t('GridField.NoItemsFound', 'No items found')
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
|
$row = FormField::create_tag(
|
||||||
|
'tr',
|
||||||
|
array(
|
||||||
|
'class' => 'ss-gridfield-item ss-gridfield-no-items',
|
||||||
|
),
|
||||||
|
$cell
|
||||||
|
);
|
||||||
|
|
||||||
|
$content['body'] = $row;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Turn into the relevant parts of a table
|
$header = $this->getOptionalTableHeader($content);
|
||||||
$head = $content['header']
|
$body = $this->getOptionalTableBody($content);
|
||||||
? FormField::create_tag('thead', array(), $content['header'])
|
$footer = $this->getOptionalTableFooter($content);
|
||||||
: '';
|
|
||||||
$body = $content['body']
|
|
||||||
? FormField::create_tag('tbody', array('class' => 'ss-gridfield-items'), $content['body'])
|
|
||||||
: '';
|
|
||||||
$foot = $content['footer']
|
|
||||||
? FormField::create_tag('tfoot', array(), $content['footer'])
|
|
||||||
: '';
|
|
||||||
|
|
||||||
$this->addExtraClass('ss-gridfield field');
|
$this->addExtraClass('ss-gridfield field');
|
||||||
$attrs = array_diff_key(
|
|
||||||
|
$fieldsetAttributes = array_diff_key(
|
||||||
$this->getAttributes(),
|
$this->getAttributes(),
|
||||||
array('value' => false, 'type' => false, 'name' => false)
|
array(
|
||||||
|
'value' => false,
|
||||||
|
'type' => false,
|
||||||
|
'name' => false,
|
||||||
|
)
|
||||||
);
|
);
|
||||||
$attrs['data-name'] = $this->getName();
|
|
||||||
$tableAttrs = array(
|
$fieldsetAttributes['data-name'] = $this->getName();
|
||||||
'id' => isset($this->id) ? $this->id : null,
|
|
||||||
|
$tableId = null;
|
||||||
|
|
||||||
|
if($this->id) {
|
||||||
|
$tableId = $this->id;
|
||||||
|
}
|
||||||
|
|
||||||
|
$tableAttributes = array(
|
||||||
|
'id' => $tableId,
|
||||||
'class' => 'ss-gridfield-table',
|
'class' => 'ss-gridfield-table',
|
||||||
'cellpadding' => '0',
|
'cellpadding' => '0',
|
||||||
'cellspacing' => '0'
|
'cellspacing' => '0',
|
||||||
);
|
);
|
||||||
|
|
||||||
if($this->getDescription()) {
|
if($this->getDescription()) {
|
||||||
@ -417,11 +494,16 @@ class GridField extends FormField {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return
|
$table = FormField::create_tag(
|
||||||
FormField::create_tag('fieldset', $attrs,
|
'table',
|
||||||
$content['before'] .
|
$tableAttributes,
|
||||||
FormField::create_tag('table', $tableAttrs, $head . "\n" . $foot . "\n" . $body) .
|
$header . "\n" . $footer . "\n" . $body
|
||||||
$content['after']
|
);
|
||||||
|
|
||||||
|
return FormField::create_tag(
|
||||||
|
'fieldset',
|
||||||
|
$fieldsetAttributes,
|
||||||
|
$content['before'] . $table . $content['after']
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -494,27 +576,44 @@ class GridField extends FormField {
|
|||||||
$classes[] = 'last';
|
$classes[] = 'last';
|
||||||
}
|
}
|
||||||
|
|
||||||
$classes[] = ($index % 2) ? 'even' : 'odd';
|
if($index % 2) {
|
||||||
|
$classes[] = 'even';
|
||||||
|
} else {
|
||||||
|
$classes[] = 'odd';
|
||||||
|
}
|
||||||
|
|
||||||
return $classes;
|
return $classes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array $properties
|
||||||
|
*
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
public function Field($properties = array()) {
|
public function Field($properties = array()) {
|
||||||
return $this->FieldHolder($properties);
|
return $this->FieldHolder($properties);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
public function getAttributes() {
|
public function getAttributes() {
|
||||||
return array_merge(parent::getAttributes(), array('data-url' => $this->Link()));
|
return array_merge(
|
||||||
|
parent::getAttributes(),
|
||||||
|
array(
|
||||||
|
'data-url' => $this->Link(),
|
||||||
|
)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the columns of this GridField, they are provided by attached GridField_ColumnProvider
|
* Get the columns of this GridField, they are provided by attached GridField_ColumnProvider.
|
||||||
*
|
*
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function getColumns() {
|
public function getColumns() {
|
||||||
// Get column list
|
|
||||||
$columns = array();
|
$columns = array();
|
||||||
|
|
||||||
foreach($this->getComponents() as $item) {
|
foreach($this->getComponents() as $item) {
|
||||||
if($item instanceof GridField_ColumnProvider) {
|
if($item instanceof GridField_ColumnProvider) {
|
||||||
$item->augmentColumns($this, $columns);
|
$item->augmentColumns($this, $columns);
|
||||||
@ -525,28 +624,36 @@ class GridField extends FormField {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the value from a column
|
* Get the value from a column.
|
||||||
*
|
*
|
||||||
* @param DataObject $record
|
* @param DataObject $record
|
||||||
* @param string $column
|
* @param string $column
|
||||||
*
|
*
|
||||||
* @return string
|
* @return string
|
||||||
|
*
|
||||||
* @throws InvalidArgumentException
|
* @throws InvalidArgumentException
|
||||||
*/
|
*/
|
||||||
public function getColumnContent($record, $column) {
|
public function getColumnContent($record, $column) {
|
||||||
// Build the column dispatch
|
|
||||||
if(!$this->columnDispatch) {
|
if(!$this->columnDispatch) {
|
||||||
$this->buildColumnDispatch();
|
$this->buildColumnDispatch();
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!empty($this->columnDispatch[$column])) {
|
if(!empty($this->columnDispatch[$column])) {
|
||||||
$content = "";
|
$content = '';
|
||||||
|
|
||||||
foreach($this->columnDispatch[$column] as $handler) {
|
foreach($this->columnDispatch[$column] as $handler) {
|
||||||
|
/**
|
||||||
|
* @var GridField_ColumnProvider $handler
|
||||||
|
*/
|
||||||
$content .= $handler->getColumnContent($this, $record, $column);
|
$content .= $handler->getColumnContent($this, $record, $column);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $content;
|
return $content;
|
||||||
} else {
|
} else {
|
||||||
throw new InvalidArgumentException("Bad column '$column'");
|
throw new InvalidArgumentException(sprintf(
|
||||||
|
'Bad column "%s"',
|
||||||
|
$column
|
||||||
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -566,111 +673,139 @@ class GridField extends FormField {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the value of a named field on the given record.
|
* Get the value of a named field on the given record.
|
||||||
* Use of this method ensures that any special rules around the data for this gridfield are followed.
|
*
|
||||||
|
* Use of this method ensures that any special rules around the data for this gridfield are
|
||||||
|
* followed.
|
||||||
|
*
|
||||||
|
* @param DataObject $record
|
||||||
|
* @param string $fieldName
|
||||||
|
*
|
||||||
|
* @return mixed
|
||||||
*/
|
*/
|
||||||
public function getDataFieldValue($record, $fieldName) {
|
public function getDataFieldValue($record, $fieldName) {
|
||||||
// Custom callbacks
|
|
||||||
if(isset($this->customDataFields[$fieldName])) {
|
if(isset($this->customDataFields[$fieldName])) {
|
||||||
$callback = $this->customDataFields[$fieldName];
|
$callback = $this->customDataFields[$fieldName];
|
||||||
|
|
||||||
return $callback($record);
|
return $callback($record);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Default implementation
|
|
||||||
if($record->hasMethod('relField')) {
|
if($record->hasMethod('relField')) {
|
||||||
return $record->relField($fieldName);
|
return $record->relField($fieldName);
|
||||||
} elseif($record->hasMethod($fieldName)) {
|
|
||||||
return $record->$fieldName();
|
|
||||||
} else {
|
|
||||||
return $record->$fieldName;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if($record->hasMethod($fieldName)) {
|
||||||
|
return $record->$fieldName();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $record->$fieldName;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get extra columns attributes used as HTML attributes
|
* Get extra columns attributes used as HTML attributes.
|
||||||
*
|
*
|
||||||
* @param DataObject $record
|
* @param DataObject $record
|
||||||
* @param string $column
|
* @param string $column
|
||||||
*
|
*
|
||||||
* @return array
|
* @return array
|
||||||
|
*
|
||||||
* @throws LogicException
|
* @throws LogicException
|
||||||
* @throws InvalidArgumentException
|
* @throws InvalidArgumentException
|
||||||
*/
|
*/
|
||||||
public function getColumnAttributes($record, $column) {
|
public function getColumnAttributes($record, $column) {
|
||||||
// Build the column dispatch
|
|
||||||
if(!$this->columnDispatch) {
|
if(!$this->columnDispatch) {
|
||||||
$this->buildColumnDispatch();
|
$this->buildColumnDispatch();
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!empty($this->columnDispatch[$column])) {
|
if(!empty($this->columnDispatch[$column])) {
|
||||||
$attrs = array();
|
$attributes = array();
|
||||||
|
|
||||||
foreach($this->columnDispatch[$column] as $handler) {
|
foreach($this->columnDispatch[$column] as $handler) {
|
||||||
$column_attrs = $handler->getColumnAttributes($this, $record, $column);
|
/**
|
||||||
|
* @var GridField_ColumnProvider $handler
|
||||||
|
*/
|
||||||
|
$columnAttributes = $handler->getColumnAttributes($this, $record, $column);
|
||||||
|
|
||||||
if(is_array($column_attrs)) {
|
if(is_array($columnAttributes)) {
|
||||||
$attrs = array_merge($attrs, $column_attrs);
|
$attributes = array_merge($attributes, $columnAttributes);
|
||||||
} elseif($column_attrs) {
|
continue;
|
||||||
$methodSignature = get_class($handler) . "::getColumnAttributes()";
|
|
||||||
throw new LogicException("Non-array response from $methodSignature.");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return $attrs;
|
throw new LogicException(sprintf(
|
||||||
} else {
|
'Non-array response from %s::getColumnAttributes().',
|
||||||
throw new InvalidArgumentException("Bad column '$column'");
|
get_class($handler)
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return $attributes;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new InvalidArgumentException(sprintf(
|
||||||
|
'Bad column "%s"',
|
||||||
|
$column
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get metadata for a column, example array('Title'=>'Email address')
|
* Get metadata for a column.
|
||||||
|
*
|
||||||
|
* @example "array('Title'=>'Email address')"
|
||||||
*
|
*
|
||||||
* @param string $column
|
* @param string $column
|
||||||
*
|
*
|
||||||
* @return array
|
* @return array
|
||||||
|
*
|
||||||
* @throws LogicException
|
* @throws LogicException
|
||||||
* @throws InvalidArgumentException
|
* @throws InvalidArgumentException
|
||||||
*/
|
*/
|
||||||
public function getColumnMetadata($column) {
|
public function getColumnMetadata($column) {
|
||||||
// Build the column dispatch
|
|
||||||
if(!$this->columnDispatch) {
|
if(!$this->columnDispatch) {
|
||||||
$this->buildColumnDispatch();
|
$this->buildColumnDispatch();
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!empty($this->columnDispatch[$column])) {
|
if(!empty($this->columnDispatch[$column])) {
|
||||||
$metadata = array();
|
$metaData = array();
|
||||||
|
|
||||||
foreach($this->columnDispatch[$column] as $handler) {
|
foreach($this->columnDispatch[$column] as $handler) {
|
||||||
$column_metadata = $handler->getColumnMetadata($this, $column);
|
/**
|
||||||
|
* @var GridField_ColumnProvider $handler
|
||||||
|
*/
|
||||||
|
$columnMetaData = $handler->getColumnMetadata($this, $column);
|
||||||
|
|
||||||
if(is_array($column_metadata)) {
|
if(is_array($columnMetaData)) {
|
||||||
$metadata = array_merge($metadata, $column_metadata);
|
$metaData = array_merge($metaData, $columnMetaData);
|
||||||
} else {
|
continue;
|
||||||
$methodSignature = get_class($handler) . "::getColumnMetadata()";
|
|
||||||
throw new LogicException("Non-array response from $methodSignature.");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
throw new LogicException(sprintf(
|
||||||
|
'Non-array response from %s::getColumnMetadata().',
|
||||||
|
get_class($handler)
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
return $metadata;
|
return $metaData;
|
||||||
}
|
}
|
||||||
throw new InvalidArgumentException("Bad column '$column'");
|
|
||||||
|
throw new InvalidArgumentException(sprintf(
|
||||||
|
'Bad column "%s"',
|
||||||
|
$column
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return how many columns the grid will have
|
* Return how many columns the grid will have.
|
||||||
*
|
*
|
||||||
* @return int
|
* @return int
|
||||||
*/
|
*/
|
||||||
public function getColumnCount() {
|
public function getColumnCount() {
|
||||||
// Build the column dispatch
|
if(!$this->columnDispatch) {
|
||||||
if(!$this->columnDispatch) $this->buildColumnDispatch();
|
$this->buildColumnDispatch();
|
||||||
|
}
|
||||||
|
|
||||||
return count($this->columnDispatch);
|
return count($this->columnDispatch);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Build an columnDispatch that maps a GridField_ColumnProvider to a column
|
* Build an columnDispatch that maps a GridField_ColumnProvider to a column for reference later.
|
||||||
* for reference later
|
|
||||||
*
|
|
||||||
*/
|
*/
|
||||||
protected function buildColumnDispatch() {
|
protected function buildColumnDispatch() {
|
||||||
$this->columnDispatch = array();
|
$this->columnDispatch = array();
|
||||||
@ -690,108 +825,137 @@ class GridField extends FormField {
|
|||||||
* This is the action that gets executed when a GridField_AlterAction gets clicked.
|
* This is the action that gets executed when a GridField_AlterAction gets clicked.
|
||||||
*
|
*
|
||||||
* @param array $data
|
* @param array $data
|
||||||
|
* @param Form $form
|
||||||
|
* @param SS_HTTPRequest $request
|
||||||
*
|
*
|
||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
public function gridFieldAlterAction($data, $form, SS_HTTPRequest $request) {
|
public function gridFieldAlterAction($data, $form, SS_HTTPRequest $request) {
|
||||||
$html = '';
|
|
||||||
$data = $request->requestVars();
|
$data = $request->requestVars();
|
||||||
$name = $this->getName();
|
$name = $this->getName();
|
||||||
$fieldData = isset($data[$name]) ? $data[$name] : null;
|
|
||||||
|
|
||||||
// Update state from client
|
$fieldData = null;
|
||||||
|
|
||||||
|
if(isset($data[$name])) {
|
||||||
|
$fieldData = $data[$name];
|
||||||
|
}
|
||||||
|
|
||||||
$state = $this->getState(false);
|
$state = $this->getState(false);
|
||||||
|
|
||||||
if(isset($fieldData['GridState'])) {
|
if(isset($fieldData['GridState'])) {
|
||||||
$state->setValue($fieldData['GridState']);
|
$state->setValue($fieldData['GridState']);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try to execute alter action
|
foreach($data as $dataKey => $dataValue) {
|
||||||
foreach($data as $k => $v) {
|
if(preg_match('/^action_gridFieldAlterAction\?StateID=(.*)/', $dataKey, $matches)) {
|
||||||
if(preg_match('/^action_gridFieldAlterAction\?StateID=(.*)/', $k, $matches)) {
|
$stateChange = Session::get($matches[1]);
|
||||||
$id = $matches[1];
|
|
||||||
$stateChange = Session::get($id);
|
|
||||||
|
|
||||||
$actionName = $stateChange['actionName'];
|
$actionName = $stateChange['actionName'];
|
||||||
|
|
||||||
$args = isset($stateChange['args']) ? $stateChange['args'] : array();
|
$arguments = array();
|
||||||
$html = $this->handleAlterAction($actionName, $args, $data);
|
|
||||||
// A field can optionally return its own HTML
|
if(isset($stateChange['args'])) {
|
||||||
if($html) return $html;
|
$arguments = $stateChange['args'];
|
||||||
|
};
|
||||||
|
|
||||||
|
$html = $this->handleAlterAction($actionName, $arguments, $data);
|
||||||
|
|
||||||
|
if($html) {
|
||||||
|
return $html;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
switch($request->getHeader('X-Pjax')) {
|
if($request->getHeader('X-Pjax') === 'CurrentField') {
|
||||||
case 'CurrentField':
|
|
||||||
return $this->FieldHolder();
|
return $this->FieldHolder();
|
||||||
break;
|
|
||||||
|
|
||||||
case 'CurrentForm':
|
|
||||||
return $form->forTemplate();
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
return $form->forTemplate();
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return $form->forTemplate();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Pass an action on the first GridField_ActionProvider that matches the $actionName
|
* Pass an action on the first GridField_ActionProvider that matches the $actionName.
|
||||||
*
|
*
|
||||||
* @param string $actionName
|
* @param string $actionName
|
||||||
* @param mixed $args
|
* @param mixed $arguments
|
||||||
* @param array $data - send data from a form
|
* @param array $data
|
||||||
|
*
|
||||||
|
* @return mixed
|
||||||
*
|
*
|
||||||
* @return type
|
|
||||||
* @throws InvalidArgumentException
|
* @throws InvalidArgumentException
|
||||||
*/
|
*/
|
||||||
public function handleAlterAction($actionName, $args, $data) {
|
public function handleAlterAction($actionName, $arguments, $data) {
|
||||||
$actionName = strtolower($actionName);
|
$actionName = strtolower($actionName);
|
||||||
|
|
||||||
foreach($this->getComponents() as $component) {
|
foreach($this->getComponents() as $component) {
|
||||||
if(!($component instanceof GridField_ActionProvider)) {
|
if($component instanceof GridField_ActionProvider) {
|
||||||
continue;
|
$actions = array_map('strtolower', (array) $component->getActions($this));
|
||||||
|
|
||||||
|
if(in_array($actionName, $actions)) {
|
||||||
|
return $component->handleAction($this, $actionName, $arguments, $data);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(in_array($actionName, array_map('strtolower', (array) $component->getActions($this)))) {
|
throw new InvalidArgumentException(sprintf(
|
||||||
return $component->handleAction($this, $actionName, $args, $data);
|
'Can\'t handle action "%s"',
|
||||||
}
|
$actionName
|
||||||
}
|
));
|
||||||
throw new InvalidArgumentException("Can't handle action '$actionName'");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Custom request handler that will check component handlers before proceeding to the default implementation.
|
* Custom request handler that will check component handlers before proceeding to the default
|
||||||
|
* implementation.
|
||||||
*
|
*
|
||||||
* @todo There is too much code copied from RequestHandler here.
|
* @todo copy less code from RequestHandler.
|
||||||
|
*
|
||||||
|
* @param SS_HTTPRequest $request
|
||||||
|
* @param DataModel $model
|
||||||
|
*
|
||||||
|
* @return array|RequestHandler|SS_HTTPResponse|string|void
|
||||||
|
*
|
||||||
|
* @throws SS_HTTPResponse_Exception
|
||||||
*/
|
*/
|
||||||
public function handleRequest(SS_HTTPRequest $request, DataModel $model) {
|
public function handleRequest(SS_HTTPRequest $request, DataModel $model) {
|
||||||
if($this->brokenOnConstruct) {
|
if($this->brokenOnConstruct) {
|
||||||
user_error("parent::__construct() needs to be called on {$handlerClass}::__construct()", E_USER_WARNING);
|
user_error(
|
||||||
|
sprintf(
|
||||||
|
"parent::__construct() needs to be called on %s::__construct()",
|
||||||
|
__CLASS__
|
||||||
|
),
|
||||||
|
E_USER_WARNING
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->request = $request;
|
$this->request = $request;
|
||||||
$this->setDataModel($model);
|
$this->setDataModel($model);
|
||||||
|
|
||||||
$fieldData = $this->request->requestVar($this->getName());
|
$fieldData = $this->request->requestVar($this->getName());
|
||||||
if($fieldData && isset($fieldData['GridState'])) $this->getState(false)->setValue($fieldData['GridState']);
|
|
||||||
|
|
||||||
foreach($this->getComponents() as $component) {
|
if($fieldData && isset($fieldData['GridState'])) {
|
||||||
if(!($component instanceof GridField_URLHandler)) {
|
$this->getState(false)->setValue($fieldData['GridState']);
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$urlHandlers = $component->getURLHandlers($this);
|
foreach($this->getComponents() as $component) {
|
||||||
|
if($component instanceof GridField_URLHandler && $urlHandlers = $component->getURLHandlers($this)) {
|
||||||
if($urlHandlers) foreach($urlHandlers as $rule => $action) {
|
foreach($urlHandlers as $rule => $action) {
|
||||||
if($params = $request->match($rule, true)) {
|
if($params = $request->match($rule, true)) {
|
||||||
// Actions can reference URL parameters, eg, '$Action/$ID/$OtherID' => '$Action',
|
// Actions can reference URL parameters.
|
||||||
if($action[0] == '$') $action = $params[substr($action, 1)];
|
// e.g. '$Action/$ID/$OtherID' → '$Action'
|
||||||
|
|
||||||
|
if($action[0] == '$') {
|
||||||
|
$action = $params[substr($action, 1)];
|
||||||
|
}
|
||||||
|
|
||||||
if(!method_exists($component, 'checkAccessAction') || $component->checkAccessAction($action)) {
|
if(!method_exists($component, 'checkAccessAction') || $component->checkAccessAction($action)) {
|
||||||
if(!$action) {
|
if(!$action) {
|
||||||
$action = "index";
|
$action = "index";
|
||||||
} else if(!is_string($action)) {
|
}
|
||||||
throw new LogicException("Non-string method name: " . var_export($action, true));
|
|
||||||
|
if(!is_string($action)) {
|
||||||
|
throw new LogicException(sprintf(
|
||||||
|
'Non-string method name: %s',
|
||||||
|
var_export($action, true)
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@ -804,26 +968,29 @@ class GridField extends FormField {
|
|||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
if($this !== $result && !$request->isEmptyPattern($rule) && is_object($result)
|
if($this !== $result && !$request->isEmptyPattern($rule) && is_object($result) && $result instanceof RequestHandler) {
|
||||||
&& $result instanceof RequestHandler
|
|
||||||
) {
|
|
||||||
|
|
||||||
$returnValue = $result->handleRequest($request, $model);
|
$returnValue = $result->handleRequest($request, $model);
|
||||||
|
|
||||||
if(is_array($returnValue)) {
|
if(is_array($returnValue)) {
|
||||||
throw new LogicException("GridField_URLHandler handlers can't return arrays");
|
throw new LogicException(
|
||||||
|
'GridField_URLHandler handlers can\'t return arrays'
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $returnValue;
|
return $returnValue;
|
||||||
|
}
|
||||||
|
|
||||||
// If we return some other data, and all the URL is parsed, then return that
|
if($request->allParsed()) {
|
||||||
} else if($request->allParsed()) {
|
|
||||||
return $result;
|
return $result;
|
||||||
|
}
|
||||||
|
|
||||||
// But if we have more content on the URL and we don't know what to do with it, return an error
|
return $this->httpError(
|
||||||
} else {
|
404,
|
||||||
return $this->httpError(404,
|
sprintf(
|
||||||
"I can't handle sub-URLs of a " . get_class($result) . " object.");
|
'I can\'t handle sub-URLs of a %s object.',
|
||||||
|
get_class($result)
|
||||||
|
)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -833,6 +1000,9 @@ class GridField extends FormField {
|
|||||||
return parent::handleRequest($request, $model);
|
return parent::handleRequest($request, $model);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritdoc}
|
||||||
|
*/
|
||||||
public function saveInto(DataObjectInterface $record) {
|
public function saveInto(DataObjectInterface $record) {
|
||||||
foreach($this->getComponents() as $component) {
|
foreach($this->getComponents() as $component) {
|
||||||
if($component instanceof GridField_SaveHandler) {
|
if($component instanceof GridField_SaveHandler) {
|
||||||
@ -841,18 +1011,61 @@ class GridField extends FormField {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array $content
|
||||||
|
*
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
protected function getOptionalTableHeader(array $content) {
|
||||||
|
if($content['header']) {
|
||||||
|
return FormField::create_tag(
|
||||||
|
'thead', array(), $content['header']
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class is the base class when you want to have an action that alters
|
* @param array $content
|
||||||
* the state of the {@link GridField}, rendered as a button element.
|
*
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
protected function getOptionalTableBody(array $content) {
|
||||||
|
if($content['body']) {
|
||||||
|
return FormField::create_tag(
|
||||||
|
'tbody', array('class' => 'ss-gridfield-items'), $content['body']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param $content
|
||||||
|
*
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
protected function getOptionalTableFooter($content) {
|
||||||
|
if($content['footer']) {
|
||||||
|
return FormField::create_tag(
|
||||||
|
'tfoot', array(), $content['footer']
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class is the base class when you want to have an action that alters the state of the
|
||||||
|
* {@link GridField}, rendered as a button element.
|
||||||
*
|
*
|
||||||
* @package forms
|
* @package forms
|
||||||
* @subpackage fields-gridfield
|
* @subpackage fields-gridfield
|
||||||
*/
|
*/
|
||||||
class GridField_FormAction extends FormAction {
|
class GridField_FormAction extends FormAction {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var GridField
|
* @var GridField
|
||||||
*/
|
*/
|
||||||
@ -880,10 +1093,10 @@ class GridField_FormAction extends FormAction {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @param GridField $gridField
|
* @param GridField $gridField
|
||||||
* @param type $name
|
* @param string $name
|
||||||
* @param type $label
|
* @param string $title
|
||||||
* @param type $actionName
|
* @param string $actionName
|
||||||
* @param type $args
|
* @param array $args
|
||||||
*/
|
*/
|
||||||
public function __construct(GridField $gridField, $name, $title, $actionName, $args) {
|
public function __construct(GridField $gridField, $name, $title, $actionName, $args) {
|
||||||
$this->gridField = $gridField;
|
$this->gridField = $gridField;
|
||||||
@ -894,19 +1107,20 @@ class GridField_FormAction extends FormAction {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* urlencode encodes less characters in percent form than we need - we
|
* Encode all non-word characters.
|
||||||
* need everything that isn't a \w.
|
|
||||||
*
|
*
|
||||||
* @param string $val
|
* @param string $value
|
||||||
|
*
|
||||||
|
* @return string
|
||||||
*/
|
*/
|
||||||
public function nameEncode($val) {
|
public function nameEncode($value) {
|
||||||
return preg_replace_callback('/[^\w]/', array($this, '_nameEncode'), $val);
|
return (string) preg_replace_callback('/[^\w]/', array($this, '_nameEncode'), $value);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The callback for nameEncode
|
* @param array $match
|
||||||
*
|
*
|
||||||
* @param string $val
|
* @return string
|
||||||
*/
|
*/
|
||||||
public function _nameEncode($match) {
|
public function _nameEncode($match) {
|
||||||
return '%' . dechex(ord($match[0]));
|
return '%' . dechex(ord($match[0]));
|
||||||
@ -916,14 +1130,12 @@ class GridField_FormAction extends FormAction {
|
|||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function getAttributes() {
|
public function getAttributes() {
|
||||||
// Store state in session, and pass ID to client side.
|
|
||||||
$state = array(
|
$state = array(
|
||||||
'grid' => $this->getNameFromParent(),
|
'grid' => $this->getNameFromParent(),
|
||||||
'actionName' => $this->actionName,
|
'actionName' => $this->actionName,
|
||||||
'args' => $this->args,
|
'args' => $this->args,
|
||||||
);
|
);
|
||||||
|
|
||||||
// Ensure $id doesn't contain only numeric characters
|
|
||||||
$id = 'gf_' . substr(md5(serialize($state)), 0, 8);
|
$id = 'gf_' . substr(md5(serialize($state)), 0, 8);
|
||||||
Session::set($id, $state);
|
Session::set($id, $state);
|
||||||
$actionData['StateID'] = $id;
|
$actionData['StateID'] = $id;
|
||||||
@ -931,18 +1143,14 @@ class GridField_FormAction extends FormAction {
|
|||||||
return array_merge(
|
return array_merge(
|
||||||
parent::getAttributes(),
|
parent::getAttributes(),
|
||||||
array(
|
array(
|
||||||
// Note: This field needs to be less than 65 chars, otherwise Suhosin security patch
|
'name' => 'action_gridFieldAlterAction?' . http_build_query($actionData),
|
||||||
// will strip it from the requests
|
|
||||||
'name' => 'action_gridFieldAlterAction' . '?' . http_build_query($actionData),
|
|
||||||
'data-url' => $this->gridField->Link(),
|
'data-url' => $this->gridField->Link(),
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Calculate the name of the gridfield relative to the Form
|
* Calculate the name of the gridfield relative to the form.
|
||||||
*
|
|
||||||
* @param GridField $base
|
|
||||||
*
|
*
|
||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
|
Loading…
Reference in New Issue
Block a user