223 lines
6.6 KiB
PHP
223 lines
6.6 KiB
PHP
<?php
|
|
|
|
namespace Drupal\webform\Controller;
|
|
|
|
use Drupal\Core\Ajax\AjaxResponse;
|
|
use Drupal\Core\Controller\ControllerBase;
|
|
use Drupal\Core\Database\Database;
|
|
use Drupal\webform\Element\WebformMessage;
|
|
use Drupal\webform\Entity\WebformOptions;
|
|
use Drupal\webform\WebformInterface;
|
|
use Symfony\Component\HttpFoundation\JsonResponse;
|
|
use Symfony\Component\HttpFoundation\Request;
|
|
|
|
/**
|
|
* Provides route responses for webform element.
|
|
*/
|
|
class WebformElementController extends ControllerBase {
|
|
|
|
/**
|
|
* Returns response for message close using user or state storage.
|
|
*
|
|
* @param string $storage
|
|
* Mechanism that the message state should be stored in, user or state.
|
|
* @param string $id
|
|
* The unique id of the message.
|
|
*
|
|
* @return \Drupal\Core\Ajax\AjaxResponse
|
|
* An empty Ajax response.
|
|
*
|
|
* @throws \Exception
|
|
* Throws exception is storage is not set to 'user' or 'state'.
|
|
*
|
|
* @see \Drupal\webform\Element\WebformMessage::setClosed
|
|
*/
|
|
public function close($storage, $id) {
|
|
if (!in_array($storage, [WebformMessage::STORAGE_USER, WebformMessage::STORAGE_STATE, WebformMessage::STORAGE_CUSTOM])) {
|
|
throw new \Exception('Undefined storage mechanism for Webform close message.');
|
|
}
|
|
WebformMessage::setClosed($storage, $id);
|
|
return new AjaxResponse();
|
|
}
|
|
|
|
/**
|
|
* Returns response for the element autocomplete route.
|
|
*
|
|
* @param \Symfony\Component\HttpFoundation\Request $request
|
|
* The current request object containing the search string.
|
|
* @param \Drupal\webform\WebformInterface $webform
|
|
* A webform.
|
|
* @param string $key
|
|
* Webform element key.
|
|
*
|
|
* @return \Symfony\Component\HttpFoundation\JsonResponse
|
|
* A JSON response containing the autocomplete suggestions.
|
|
*/
|
|
public function autocomplete(Request $request, WebformInterface $webform, $key) {
|
|
// Get autocomplete query.
|
|
$q = $request->query->get('q') ?: '';
|
|
if ($q == '') {
|
|
return new JsonResponse([]);
|
|
}
|
|
|
|
// Get the initialized webform element.
|
|
$element = $webform->getElement($key);
|
|
if (!$element) {
|
|
return new JsonResponse([]);
|
|
}
|
|
|
|
// Set default autocomplete properties.
|
|
$element += [
|
|
'#autocomplete_existing' => FALSE,
|
|
'#autocomplete_items' => [],
|
|
'#autocomplete_match' => 3,
|
|
'#autocomplete_limit' => 10,
|
|
'#autocomplete_match_operator' => 'CONTAINS',
|
|
];
|
|
|
|
// Check minimum number of characters.
|
|
if (mb_strlen($q) < (int) $element['#autocomplete_match']) {
|
|
return new JsonResponse([]);
|
|
}
|
|
|
|
$matches = [];
|
|
|
|
// Get existing matches.
|
|
if (!empty($element['#autocomplete_existing'])) {
|
|
$matches += $this->getMatchesFromExistingValues($q, $webform->id(), $key, $element['#autocomplete_match_operator'], $element['#autocomplete_limit']);
|
|
}
|
|
|
|
// Get items (aka options) matches.
|
|
if (!empty($element['#autocomplete_items'])) {
|
|
$element['#options'] = $element['#autocomplete_items'];
|
|
$options = WebformOptions::getElementOptions($element);
|
|
$matches += $this->getMatchesFromOptions($q, $options, $element['#autocomplete_match_operator'], $element['#autocomplete_limit']);
|
|
}
|
|
|
|
// Sort matches by label and enforce the limit.
|
|
if ($matches) {
|
|
uasort($matches, function (array $a, array $b) {
|
|
return $a['label'] > $b['label'];
|
|
});
|
|
$matches = array_values($matches);
|
|
$matches = array_slice($matches, 0, $element['#autocomplete_limit']);
|
|
}
|
|
|
|
return new JsonResponse($matches);
|
|
}
|
|
|
|
/**
|
|
* Get matches from existing submission values.
|
|
*
|
|
* @param string $q
|
|
* String to filter option's label by.
|
|
* @param string $webform_id
|
|
* The webform id.
|
|
* @param string $key
|
|
* The element's key.
|
|
* @param string $operator
|
|
* Match operator either CONTAINS or STARTS_WITH.
|
|
* @param int $limit
|
|
* Limit number of matches.
|
|
*
|
|
* @return array
|
|
* An array of matches.
|
|
*/
|
|
protected function getMatchesFromExistingValues($q, $webform_id, $key, $operator = 'CONTAINS', $limit = 10) {
|
|
// Query webform submission for existing values.
|
|
$query = Database::getConnection()->select('webform_submission_data')
|
|
->fields('webform_submission_data', ['value'])
|
|
->condition('webform_id', $webform_id)
|
|
->condition('name', $key)
|
|
->condition('value', ($operator == 'START_WITH') ? "$q%" : "%$q%", 'LIKE')
|
|
->orderBy('value');
|
|
if ($limit) {
|
|
$query->range(0, $limit);
|
|
}
|
|
|
|
// Convert query results values to matches array.
|
|
$values = $query->execute()->fetchCol();
|
|
$matches = [];
|
|
foreach ($values as $value) {
|
|
$matches[$value] = ['value' => $value, 'label' => $value];
|
|
}
|
|
return $matches;
|
|
}
|
|
|
|
/**
|
|
* Get matches from options.
|
|
*
|
|
* @param string $q
|
|
* String to filter option's label by.
|
|
* @param array $options
|
|
* An associative array of webform options.
|
|
* @param string $operator
|
|
* Match operator either CONTAINS or STARTS_WITH.
|
|
* @param int $limit
|
|
* Limit number of matches.
|
|
*
|
|
* @return array
|
|
* An array of matches sorted by label.
|
|
*/
|
|
protected function getMatchesFromOptions($q, array $options, $operator = 'CONTAINS', $limit = 10) {
|
|
// Make sure options are populated.
|
|
if (empty($options)) {
|
|
return [];
|
|
}
|
|
|
|
$matches = [];
|
|
|
|
// Filter and convert options to autocomplete matches.
|
|
$this->getMatchesFromOptionsRecursive($q, $options, $matches, $operator);
|
|
|
|
// Sort matches.
|
|
ksort($matches);
|
|
|
|
// Apply match limit.
|
|
if ($limit) {
|
|
$matches = array_slice($matches, 0, $limit);
|
|
}
|
|
|
|
return array_values($matches);
|
|
}
|
|
|
|
/**
|
|
* Get matches from options recursive.
|
|
*
|
|
* @param string $q
|
|
* String to filter option's label by.
|
|
* @param array $options
|
|
* An associative array of webform options.
|
|
* @param array $matches
|
|
* An associative array of autocomplete matches.
|
|
* @param string $operator
|
|
* Match operator either CONTAINS or STARTS_WITH.
|
|
*/
|
|
protected function getMatchesFromOptionsRecursive($q, array $options, array &$matches, $operator = 'CONTAINS') {
|
|
foreach ($options as $label) {
|
|
if (is_array($label)) {
|
|
$this->getMatchesFromOptionsRecursive($q, $label, $matches, $operator);
|
|
continue;
|
|
}
|
|
|
|
// Cast TranslatableMarkup to string.
|
|
$label = (string) $label;
|
|
|
|
if ($operator == 'STARTS_WITH' && stripos($label, $q) === 0) {
|
|
$matches[$label] = [
|
|
'value' => $label,
|
|
'label' => $label,
|
|
];
|
|
}
|
|
// Default to CONTAINS even when operator is empty.
|
|
elseif (stripos($label, $q) !== FALSE) {
|
|
$matches[$label] = [
|
|
'value' => $label,
|
|
'label' => $label,
|
|
];
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
}
|