Добавление автозаполнения для текстового поля

10

Я попытался реализовать автозаполнение в текстовом поле для drupal 8 в моем пользовательском модуле

все, что я хотел, это выбрать и отобразить вероятный заголовок, который я набрал с помощью автозаполнения, поэтому объявил открытую функцию автозаполнения в классе в DefaultController.php в каталоге папки -> mymodule / src / Controller / DefaultController.php

<?php

namespace Drupal\mymodule\Controller;

use Drupal\Core\Controller\ControllerBase;
use Symfony\Component\HttpFoundation\JsonResponse;

class DefaultController extends ControllerBase
{
    public function autocomplete($string)
    {
        $matches = array();
        $db = \Drupal::database();
        $result = $db->select('node_field_data', 'n')
        ->fields('n', array('title', 'nid'))
        ->condition('title', '%'.db_like($string).'%', 'LIKE')
        ->addTag('node_access')
        ->execute();

        foreach ($result as $row) {
            $matches[$row->nid] = check_plain($row->title);
        }

        return new JsonResponse($matches);
    }
}

затем создал EditForm.php в каталоге папок -> mymodule / src / Form / EditForm.php

<?php

namespace Drupal\mymodule\Form;

use Drupal\Core\Form\FormBase;
use Drupal\Core\Form\FormStateInterface;

class EditForm extends FormBase
{
    public function getFormId()
    {
        return 'mymodule_edit_form';
    }

    public function buildForm(array $form, FormStateInterface $form_state)
    {
        $form = array();

  $form['input_fields']['nid'] = array(
    '#type' => 'textfield',
    '#title' => t('Name of the referenced node'),
    '#autocomplete_route_name' => 'mymodule.autocomplete',
    '#description' => t('Node Add/Edit type block'),
    '#default' => ($form_state->isValueEmpty('nid')) ? null : ($form_state->getValue('nid')),
    '#required' => true,
  );

        $form['submit'] = array(
    '#type' => 'submit',
    '#value' => t('Create'),
  );

        return $form;
    }
}

также создал mymodule.routing.yml

  mymodule.autocomplete:
  path: '/mymodule/autocomplete'
  defaults:
    _controller: '\Drupal\mymodule\Controller\DefaultController::autocomplete'
  requirements:
    _permission: 'access content'

все еще не реализована функция автозаполнения? Кто-нибудь может указать мне, что мне не хватает ??

макияж меня-живой
источник
вам нужно передать параметры также drupal.org/node/2070985
Шрея Шетти
1
@ShreyaShetty Нет, мне не нужны параметры, так как в d7 я бы использовал '#autocomplete_path' => 'mymodule / autocomplete', поэтому в d8 я использовал '#autocomplete_route_name' => 'mymodule.autocomplete', поэтому я никогда не использовал параметр и при этом я не нуждаюсь ....
сделай меня живым

Ответы:

10

Ваш класс нуждается в некоторой модификации, вам нужно проверить запрос и поместить его в $ string.

<?php

namespace Drupal\mymodule\Controller;

use Symfony\Component\HttpFoundation\JsonResponse;
use Symfony\Component\HttpFoundation\Request;
use Drupal\Component\Utility\Unicode;

class DefaultController extends ControllerBase
{

  /**
   * Returns response for the autocompletion.
   *
   * @param \Symfony\Component\HttpFoundation\Request $request
   *   The current request object containing the search string.
   *
   * @return \Symfony\Component\HttpFoundation\JsonResponse
   *   A JSON response containing the autocomplete suggestions.
   */

  public function autocomplete(request $request) {
    $matches = array();
    $string = $request->query->get('q');
    if ($string) {
      $matches = array();
      $query = \Drupal::entityQuery('node')
      ->condition('status', 1)
      ->condition('title', '%'.db_like($string).'%', 'LIKE');
      //->condition('field_tags.entity.name', 'node_access');
      $nids = $query->execute();
      $result = entity_load_multiple('node', $nids);
      foreach ($result as $row) {
        //$matches[$row->nid->value] = $row->title->value;
        $matches[] = ['value' => $row->nid->value, 'label' => $row->title->value];
      }
    }
    return new JsonResponse($matches);
  }
}
vgoradiya
источник
он все еще НЕ РАБОТАЛ после получения запроса и помещения его в $ string
make-me-alive
Вы проверили, что ваш запрос был распечатан?
вгорадия
Я думаю, что \Drupal::entityQuery('node')было бы предпочтительнее использовать отдельно от выбора.
вгорадия
После просмотра вкладки «Сеть» в инструментах разработчика моего браузера я мог видеть правильные результаты в ответе, но они не отображались в пользовательском интерфейсе. После некоторого поиска я обнаружил, что у меня есть некоторые пользовательские настройки CSS z-indexдля элемента DOM, который содержит форму. Значение было слишком высоким и перекрывало результаты автозаполнения. Понижение моего обычая z-indexисправило это для меня.
tyler.frankenstein
11

Если вы хотите выбрать сущность, то есть более простой способ сделать это. Drupal 8 имеет стандартный тип поля entity_autocomplete, просто укажите свой элемент формы следующим образом:

$form['node'] = [
  '#type' => 'entity_autocomplete',
  '#target_type' => 'node',
];

См. Пользовательское поле автозаполнения для получения дополнительной информации.

Кроме того, никогда не выполняйте запросы базы данных к таблицам узлов / сущностей. Для этого используйте \ Drupal :: entityQuery ().

Berdir
источник
Привет Бердир, Как получить данные из таксономии в моем случае "город", используя приведенный выше код, так как он отлично работает для узла, но не для таксономии ??
Сачин
5
  1. Создайте файл routing.yml и добавьте следующий код: admin_example.autocomplete:

:

  path: '/admin_example/autocomplete'
  defaults:
    _controller: '\Drupal\admin_example\Controller\AdminNotesController::autocomplete'
  requirements:
    _permission: 'access content'
  1. Форма, которую вы создали в mymodule / src / Form / EditForm.php, правильная

Вам нужно изменить код в контроллере. Код ниже:

public function autocomplete(Request $request)
{
 $string = $request->query->get('q');
    $matches = array();
      $query = db_select('node_field_data', 'n')
          ->fields('n', array('title', 'nid'))
          ->condition('title', $string . '%', 'LIKE')
          ->execute()
          ->fetchAll();
    foreach ($query as $row) {
        $matches[] = array('value' => $row->nid, 'label' => $row->title);
    }

    return new JsonResponse($matches);
}
Шрея Шетти
источник
Привет, Шрея, я использовал твое решение, но вместо того, чтобы перейти к форме, он дает мне o / p, например: [{"value": "1", "label": "Patel Residency"}, {"value": " 2 "," label ":" Jain Plaza "}, {" value ":" 3 "," label ":" Kanha Resort "}, {" value ":" 38 "," label ":" Hira Residency "} ]. Я хочу, чтобы он вошел в форму, и поле должно работать как автозаполнение
Sachin
2

Используйте код @vgoradiya, затем в цикле foreach попробуйте это так:

    foreach ($result as $row)
    {
        $matches[] = ['value' => $row->nid, 'label' => check_plain($row->title)];
    }
Спенсер Чанг
источник