EntitySelect.php in Commerce Core 8.2
Namespace
Drupal\commerce\ElementFile
src/Element/EntitySelect.phpView source
<?php
namespace Drupal\commerce\Element;
use Drupal\commerce\EntityHelper;
use Drupal\Core\Form\FormStateInterface;
use Drupal\Core\Render\Element\FormElement;
/**
* Provides a form input element for selecting one or multiple entities.
*
* The element is transformed based on the number of available entities:
* 1..#autocomplete_threshold: Checkboxes/radios element, based on #multiple.
* >#autocomplete_threshold: entity autocomplete element.
* If the element is required, and there's only one available entity, a hidden
* form element can be used instead of checkboxes/radios.
*
* Properties:
* - #target_type: The entity type being selected.
* - #multiple: Whether the user may select more than one item.
* - #default_value: An entity ID or an array of entity IDs.
* - #hide_single_entity: Whether to use a hidden element when there's only one
* available entity and the element is required.
* - #autocomplete_threshold: Determines when to use the autocomplete.
* - #autocomplete_size: The size of the autocomplete element in characters.
* - #autocomplete_placeholder: The placeholder for the autocomplete element.
*
* Example usage:
* @code
* $form['entities'] = [
* '#type' => 'commerce_entity_select',
* '#title' => t('Stores'),
* '#target_type' => 'commerce_store',
* '#multiple' => TRUE,
* ];
*
* @end
*
* @FormElement("commerce_entity_select")
*/
class EntitySelect extends FormElement {
/**
* {@inheritdoc}
*/
public function getInfo() {
$class = get_class($this);
return [
'#input' => TRUE,
'#target_type' => '',
'#multiple' => FALSE,
'#hide_single_entity' => TRUE,
'#autocomplete_threshold' => 7,
'#autocomplete_size' => 60,
'#autocomplete_placeholder' => '',
'#process' => [
[
$class,
'processEntitySelect',
],
[
$class,
'processAjaxForm',
],
],
'#element_validate' => [
[
$class,
'validateEntitySelect',
],
],
];
}
/**
* Process callback.
*/
public static function processEntitySelect(&$element, FormStateInterface $form_state, &$complete_form) {
// Nothing to do if there is no target entity type.
if (empty($element['#target_type'])) {
throw new \InvalidArgumentException('Missing required #target_type parameter.');
}
/** @var \Drupal\Core\Entity\EntityStorageInterface $storage */
$storage = \Drupal::service('entity_type.manager')
->getStorage($element['#target_type']);
$entity_count = $storage
->getQuery()
->accessCheck(TRUE)
->count()
->execute();
$element['#tree'] = TRUE;
// No need to show anything, there's only one possible value.
if ($element['#required'] && $entity_count == 1 && $element['#hide_single_entity']) {
$entity_ids = $storage
->getQuery()
->accessCheck(TRUE)
->execute();
$element['value'] = [
'#type' => 'hidden',
'#value' => reset($entity_ids),
];
return $element;
}
if ($entity_count <= $element['#autocomplete_threshold']) {
// Start with a query to get only access-filtered results.
$entity_ids = $storage
->getQuery()
->accessCheck(TRUE)
->execute();
$entities = $storage
->loadMultiple($entity_ids);
$entity_labels = EntityHelper::extractLabels($entities);
// Radio buttons don't have a None option by default.
if (!$element['#multiple'] && !$element['#required']) {
$entity_labels = [
'' => t('None'),
] + $entity_labels;
}
$element['value'] = [
'#type' => $element['#multiple'] ? 'checkboxes' : 'radios',
'#required' => $element['#required'],
'#options' => $entity_labels,
];
if (!empty($element['#default_value'])) {
$element['value']['#default_value'] = $element['#default_value'];
}
}
else {
$default_value = NULL;
if (!empty($element['#default_value'])) {
// Upcast ids into entities, as expected by entity_autocomplete.
if ($element['#multiple']) {
$default_value = $storage
->loadMultiple($element['#default_value']);
}
else {
$default_value = $storage
->load($element['#default_value']);
}
}
$element['value'] = [
'#type' => 'entity_autocomplete',
'#target_type' => $element['#target_type'],
'#tags' => $element['#multiple'],
'#required' => $element['#required'],
'#default_value' => $default_value,
'#size' => $element['#autocomplete_size'],
'#placeholder' => $element['#autocomplete_placeholder'],
'#maxlength' => NULL,
];
}
// These keys only make sense on the actual input element.
foreach ([
'#title',
'#title_display',
'#description',
'#ajax',
] as $key) {
if (isset($element[$key])) {
$element['value'][$key] = $element[$key];
unset($element[$key]);
}
}
return $element;
}
/**
* Validation callback.
*
* Transforms the subelement value into a consistent format and set it on the
* main element.
*/
public static function validateEntitySelect(&$element, FormStateInterface $form_state, &$complete_form) {
$value_element = $element['value'];
$value = $form_state
->getValue($value_element['#parents']);
if (is_array($value)) {
if ($value_element['#type'] == 'checkboxes') {
// Remove unselected checkboxes.
$value = array_filter($value);
// Non-numeric keys can cause issues when passing values to TypedData.
$value = array_values($value);
}
elseif (!empty($value_element['#tags'])) {
// Extract the entity ids from a multivalue autocomplete.
$value = array_column($value, 'target_id');
}
}
$form_state
->setValueForElement($element, $value);
}
}
Classes
Name![]() |
Description |
---|---|
EntitySelect | Provides a form input element for selecting one or multiple entities. |