xandeadx.ru Блог музицирующего веб-девелопера

DrupalМодуль EAV Field — хранение большого числа характеристик сущности в одном поле

Описание

Несколько лет назад я описывал способы реализации каталога с большим количеством атрибутов товара и одним из способов было использование модели EAV, когда все значения атрибутов хранятся в одном составном поле. Есть несколько модулей разной свежести, пытающиеся реализовать это в друпале, но меня ни один не устроил, поэтому родился EAV Field.

DrupalКак расширить функционал фильтр-плагина Views? (добавляем возможность фильтровать числовые поля по нескольким значениям)

Опубликовано в

Пример добавления возможности фильтровать числовые поля по нескольким значениями (оператор "in").

1. В папке своего модуля src/Plugin/views/filter создаём класс и наследуем его от класса, который нужно расширить. В этом классе переопределяем нужные методы и по необходимости добавляем свои. Аннотацию плагина добавлять не нужно.

class ExtendedNumericFilter extends NumericFilter {
 
  /**
   * {@inheritDoc}
   */
  public function operators() {
    $operators = parent::operators();
 
    $operators['in'] = [
      'title' => $this->t('Is one of'),

DrupalAJAX добавление товара в корзину в Commerce 2

Опубликовано в
/**
 * Implements hook_form_BASE_FORM_ID_alter(): commerce_order_item_add_to_cart_form.
 * Alter "Add to cart" form.
 */
function MODULENAME_form_commerce_order_item_add_to_cart_form_alter(array &$form, FormStateInterface $form_state) {
  $product = $form_state->get('product'); /** @var ProductInterface $product */
 
  $form['actions']['submit']['#id'] = 'add-to-cart-button-' . $product->id();
  $form['actions']['submit']['#ajax'] = [
    'callback' => 'MODULENAME_add_to_cart_ajax',
    'event' => 'click',

DrupalВыполнить очередь с помощью Batch API

Опубликовано в

Пример выполнения очереди products_import с помощью Batch API:

class ProductsImportForm extends FormBase {
 
  /**
   * {@inheritDoc}
   */
  public function getFormId() {
    return 'products_import_form';
  }
 
  /**
   * {@inheritDoc}
   */
  public function buildForm(array $form, FormStateInterface $form_state) {
    $form['submit'] = [
      '#type' => 'submit',
      '#value' => $this->t('Start'),
    ];
 
    return $form;
  }
 
  /**
   * {@inheritDoc}
   */
  public function submitForm(array &$form, FormStateInterface $form_state) {

DrupalДобавить сторонней контент сущности своё базовое поле (base field)

Пример добавления терминам базового поля stored_depth:

// MODULENAME.module
 
/**
 * Implements hook_entity_base_field_info().
 */
function MODULENAME_entity_base_field_info(EntityTypeInterface $entity_type) {
  if ($entity_type->id() == 'taxonomy_term') {
    $fields = [];
 
    $fields['stored_depth'] = BaseFieldDefinition::create('integer')
      ->setLabel(t('Depth'))
      ->setSetting('unsigned', TRUE)
      ->setDefaultValue(0);
 
    return $fields;
  }
}

DrupalЗапретить анонимам ставить флаг чаще одного раза в день с одного ip

Опубликовано в

В модуле Flag анонимный юзер может бесконечно флагать одну сущность просто удаляю куку с сессией, что неприемлемо например при реализации лайков. Чтобы это запретить нужно:

1. В админке добавить флагу текстовое поле field_ip.

2. Написать код:

/**
 * Implements hook_ENTITY_TYPE_presave(): flagging.
 */
function MODULENAME_flagging_presave(FlaggingInterface $flagging) {
  // Save user ip
  if ($flagging->hasField('field_ip')) {
    $flagging->set('field_ip', \Drupal::request()->getClientIp());

DrupalСоздание своего текстового фильтра

Опубликовано в

Пример фильтра для оборачивания таблиц в div:

// src/Plugin/Filter/TableWrapperFilter.php
 
/**
 * @Filter(
 *   id = "table_wrapper_filter",
 *   title = @Translation("Table wrapper"),
 *   description = @Translation("Wrap tables to div."),
 *   type = Drupal\filter\Plugin\FilterInterface::TYPE_TRANSFORM_REVERSIBLE,
 *   weight = 20
 * )
 */
class TableWrapperFilter extends FilterBase {
 
  /**
   * {@inheritdoc}
   */
  public function process($text, $langcode) {
    $result = new FilterProcessResult($text);
 
    if (stristr($text, '<table') !== FALSE) {

JavascriptОтличие Debounce от Throttling

Опубликовано в

Debounce — функция будет выполнена только тогда, когда после последней попытки вызова прошло определённое время. Задержка начинает заново отсчитываться с каждой новой попыткой вызова. Например если повесить debounce на onscroll с временем 100ms, то функция выполнится через 100ms после прекращения скрола.

Схематично:

Debounce

DrupalОднократное выполнение ajax запроса

Опубликовано в

Часто возникает необходимость, чтобы ajax ссылка с классом use-ajax выполнила запрос ровно один раз, после чего прекратила реакцию на клики (например это ссылка загрузки контента в таб). Реализовать задуманное можно с помощью небольшого костыля подменяющего Drupal.Ajax.prototype.eventResponse:

(function ($, Drupal) {
  /**
   * Override AJAX eventResponse function.
   */
  var originalAjaxEventResponse = Drupal.Ajax.prototype.eventResponse;
 
  Drupal.Ajax.prototype.eventResponse = function (element, event) {
    var $element = $(element);

DrupalРасширить сторонний форматтер своим функционалом

Опубликовано в

Пример добавления для форматтера text_default функционала обрезки текста:

// MODULENAME.module
 
/**
 * Implements hook_field_formatter_third_party_settings_form().
 */
function MODULENAME_field_formatter_third_party_settings_form(FormatterInterface $plugin, FieldDefinitionInterface $field_definition, $view_mode, array $form, FormStateInterface $form_state) {
  $element = [];
 
  if ($plugin->getPluginId() == 'text_default') {
    $element['max_length'] = [
      '#type' => 'number',
      '#title' => t('Max length'),