silverstripe-gridfieldexten.../src/GridFieldAddNewMultiClass.php

310 lines
8.0 KiB
PHP
Raw Normal View History

<?php
2017-06-16 06:07:09 +02:00
namespace Symbiote\GridFieldExtensions;
2016-11-29 18:20:15 +01:00
use SilverStripe\Control\Controller;
2017-09-15 06:01:32 +02:00
use SilverStripe\Control\HTTPRequest;
use SilverStripe\Control\HTTPResponse_Exception;
use SilverStripe\Core\ClassInfo;
use SilverStripe\Core\Config\Config;
2017-06-25 21:29:43 +02:00
use SilverStripe\Core\Injector\Injector;
use SilverStripe\Forms\DropdownField;
use SilverStripe\Forms\GridField\AbstractGridFieldComponent;
use SilverStripe\Forms\GridField\GridField;
use SilverStripe\Forms\GridField\GridField_HTMLProvider;
use SilverStripe\Forms\GridField\GridField_URLHandler;
2017-09-15 06:01:32 +02:00
use SilverStripe\Forms\GridField\GridFieldDetailForm;
use SilverStripe\View\ArrayData;
2016-11-29 18:30:01 +01:00
use ReflectionClass;
use Exception;
/**
* A component which lets the user select from a list of classes to create a new record form.
*
* By default the list of classes that are createable is the grid field's model class, and any
* subclasses. This can be customised using {@link setClasses()}.
*/
class GridFieldAddNewMultiClass extends AbstractGridFieldComponent implements
GridField_HTMLProvider,
GridField_URLHandler
2016-12-21 03:34:58 +01:00
{
2017-09-15 06:01:32 +02:00
/**
* @skipUpgrade
*/
const POST_KEY = 'GridFieldAddNewMultiClass';
2016-12-21 03:34:58 +01:00
private static $allowed_actions = array(
'handleAdd'
);
// Should we add an empty string to the add class dropdown?
private static $showEmptyString = true;
private $fragment;
private $title;
2017-09-15 06:01:32 +02:00
/**
* @var array
*/
2016-12-21 03:34:58 +01:00
private $classes;
2017-09-15 06:01:32 +02:00
/**
* @var string
*/
2016-12-21 03:34:58 +01:00
private $defaultClass;
/**
* @var string
*/
2017-06-16 06:07:09 +02:00
protected $itemRequestClass = 'Symbiote\\GridFieldExtensions\\GridFieldAddNewMultiClassHandler';
2016-12-21 03:34:58 +01:00
/**
* @param string $fragment the fragment to render the button in
*/
public function __construct($fragment = 'before')
{
$this->setFragment($fragment);
$this->setTitle(_t('GridFieldExtensions.ADD', 'Add'));
}
/**
* Gets the fragment name this button is rendered into.
*
* @return string
*/
public function getFragment()
{
return $this->fragment;
}
/**
* Sets the fragment name this button is rendered into.
*
* @param string $fragment
* @return GridFieldAddNewMultiClass $this
*/
public function setFragment($fragment)
{
$this->fragment = $fragment;
return $this;
}
/**
* Gets the button title text.
*
* @return string
*/
public function getTitle()
{
return $this->title;
}
/**
* Sets the button title text.
*
* @param string $title
* @return GridFieldAddNewMultiClass $this
*/
public function setTitle($title)
{
$this->title = $title;
return $this;
}
/**
* Gets the classes that can be created using this button, defaulting to the model class and
* its subclasses.
*
* @param GridField $grid
* @return array a map of class name to title
*/
public function getClasses(GridField $grid)
{
$result = array();
if (is_null($this->classes)) {
2022-04-13 07:44:44 +02:00
$classes = array_values(ClassInfo::subclassesFor($grid->getModelClass()) ?? []);
2016-12-21 03:34:58 +01:00
sort($classes);
} else {
$classes = $this->classes;
}
$kill_ancestors = array();
foreach ($classes as $class => $title) {
if (!is_string($class)) {
$class = $title;
}
2022-04-13 07:44:44 +02:00
if (!class_exists($class ?? '')) {
2016-12-21 03:34:58 +01:00
continue;
}
$is_abstract = (($reflection = new ReflectionClass($class)) && $reflection->isAbstract());
if (!$is_abstract && $class === $title) {
$title = singleton($class)->i18n_singular_name();
}
if ($ancestor_to_hide = Config::inst()->get($class, 'hide_ancestor')) {
2016-12-21 03:34:58 +01:00
$kill_ancestors[$ancestor_to_hide] = true;
}
if ($is_abstract || !singleton($class)->canCreate()) {
continue;
}
$result[$class] = $title;
}
if ($kill_ancestors) {
foreach ($kill_ancestors as $class => $bool) {
unset($result[$class]);
}
}
2017-07-05 17:15:20 +02:00
$sanitised = array();
2017-09-01 03:02:18 +02:00
foreach ($result as $class => $title) {
2017-07-05 17:15:20 +02:00
$sanitised[$this->sanitiseClassName($class)] = $title;
}
return $sanitised;
2016-12-21 03:34:58 +01:00
}
/**
* Sets the classes that can be created using this button.
*
* @param array $classes a set of class names, optionally mapped to titles
2017-09-15 06:01:32 +02:00
* @param string $default
2016-12-21 03:34:58 +01:00
* @return GridFieldAddNewMultiClass $this
*/
public function setClasses(array $classes, $default = null)
{
$this->classes = $classes;
if ($default) {
$this->defaultClass = $default;
}
return $this;
}
/**
* Sets the default class that is selected automatically.
*
* @param string $default the class name to use as default
* @return GridFieldAddNewMultiClass $this
*/
public function setDefaultClass($default)
{
$this->defaultClass = $default;
return $this;
}
/**
* Handles adding a new instance of a selected class.
*
* @param GridField $grid
2017-09-15 06:01:32 +02:00
* @param HTTPRequest $request
2016-12-21 03:34:58 +01:00
* @return GridFieldAddNewMultiClassHandler
2017-09-15 06:01:32 +02:00
* @throws Exception
* @throws HTTPResponse_Exception
2016-12-21 03:34:58 +01:00
*/
public function handleAdd($grid, $request)
{
$class = $request->param('ClassName');
$classes = $this->getClasses($grid);
2017-09-15 06:01:32 +02:00
/** @var GridFieldDetailForm $component */
$component = $grid->getConfig()->getComponentByType(GridFieldDetailForm::class);
2016-12-21 03:34:58 +01:00
if (!$component) {
throw new Exception('The add new multi class component requires the detail form component.');
}
2022-04-13 07:44:44 +02:00
if (!$class || !array_key_exists($class, $classes ?? [])) {
2016-12-21 03:34:58 +01:00
throw new HTTPResponse_Exception(400);
}
2017-07-05 17:15:20 +02:00
$unsanitisedClass = $this->unsanitiseClassName($class);
2017-06-25 21:29:43 +02:00
$handler = Injector::inst()->create(
2016-12-21 03:34:58 +01:00
$this->itemRequestClass,
$grid,
$component,
2017-07-05 17:15:20 +02:00
new $unsanitisedClass(),
2016-12-21 03:34:58 +01:00
$grid->getForm()->getController(),
'add-multi-class'
);
$handler->setTemplate($component->getTemplate());
return $handler;
}
/**
* {@inheritDoc}
*/
public function getHTMLFragments($grid)
{
$classes = $this->getClasses($grid);
2022-04-13 07:44:44 +02:00
if (!count($classes ?? [])) {
2016-12-21 03:34:58 +01:00
return array();
}
GridFieldExtensions::include_requirements();
$field = DropdownField::create(
sprintf('%s[%s]', __CLASS__, $grid->getName()),
'',
$classes,
$this->defaultClass
);
2016-12-21 03:34:58 +01:00
if (Config::inst()->get(__CLASS__, 'showEmptyString')) {
$field->setEmptyString(_t('GridFieldExtensions.SELECTTYPETOCREATE', '(Select type to create)'));
}
$field->addExtraClass('no-change-track');
$data = ArrayData::create(array(
2016-12-21 03:34:58 +01:00
'Title' => $this->getTitle(),
'Link' => Controller::join_links($grid->Link(), 'add-multi-class', '{class}'),
'ClassField' => $field
));
return array(
$this->getFragment() => $data->renderWith(__CLASS__)
);
}
/**
* {@inheritDoc}
*/
public function getURLHandlers($grid)
{
return array(
'add-multi-class/$ClassName!' => 'handleAdd'
);
}
public function setItemRequestClass($class)
{
$this->itemRequestClass = $class;
return $this;
}
2017-07-05 17:15:20 +02:00
/**
* Sanitise a model class' name for inclusion in a link
2017-09-15 06:01:32 +02:00
*
* @param string $class
2017-07-05 17:15:20 +02:00
* @return string
*/
protected function sanitiseClassName($class)
{
2022-04-13 07:44:44 +02:00
return str_replace('\\', '-', $class ?? '');
2017-07-05 17:15:20 +02:00
}
/**
* Unsanitise a model class' name from a URL param
2017-09-15 06:01:32 +02:00
*
* @param string $class
2017-07-05 17:15:20 +02:00
* @return string
*/
protected function unsanitiseClassName($class)
{
2022-04-13 07:44:44 +02:00
return str_replace('-', '\\', $class ?? '');
2017-07-05 17:15:20 +02:00
}
}