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

Drupal → Кэширование блоков с формой

По умолчанию блоки с формами кэшируются только для анонимных пользователей. Для залогинённых же друпал добавляет во все формы элемент form_token со свойством '#cache' => ['max-age' => 0] (тырк), который запрещает кэшировать все вышестоящие элементы (блок, страница и т.д.). Чтобы этого избежать и включить кэширование блока для всех, нужно отключить генерацию form_token:

public function buildForm(array $form, FormStateInterface $form_state) {
  $form['#token'] = FALSE;
  ...
}

Это не очень безопасно, но других вариантов пока не знаю.

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

public function buildForm(array $form, FormStateInterface $form_state) {
  $form['#token'] = FALSE;
  $form['#action'] = '';
  ...
}

Если в форме будут элементы со свойством #ajax, то появится ещё одна проблема — друпал автоматически добавит для таких элементов кэш-контекст route, который заставит блок с формой генериться заново на каждой новой странице. Например, если блок выводится на 1000 страницах, то в cache_render будет 1000 вариантов этого блока. Решается добавлением в #ajax свойства url с адресом на страницу где точно будет форма, плюс специально сформированное свойство options:

public function buildForm(array $form, FormStateInterface $form_state) {
  $form['submit'] = [
    '#type' => 'submit',
    '#value' => 'Submit',
    '#ajax' => [
      'callback' => '::ajaxSubmit',
      'url' => Url::fromRoute('<front>'), // <-- new
      'options' => ['query' => [FormBuilderInterface::AJAX_FORM_REQUEST => 1]], // <-- new
    ],
    '#id' => 'submit-my-form', // <-- new
  ];
 
  $form['#action'] = '';
  $form['#token'] = FALSE;
  ...
}

Плюс, нужно добавить уникальный #id, чтобы не было конфликта с другими элементами.

Такие вот костыли костылики.

Готовый модуль для проверки.

Написанное актуально для Drupal 8+
Похожие записи

Оставить комментарий

Содержимое этого поля является приватным и не будет отображаться публично. Если у вас есть аккаунт в Gravatar, привязанный к этому e-mail адресу, то он будет использован для отображения аватара.
  • Адреса страниц и электронной почты автоматически преобразуются в ссылки.
  • Доступные HTML теги: <a> <i> <b> <strong> <code> <ul> <ol> <li> <blockquote> <em> <s>
  • Строки и параграфы переносятся автоматически.
  • Подсветка кода осуществляется с помощью тегов: <code>, <css>, <html>, <ini>, <javascript>, <sql>, <php>. Поддерживаемые стили выделения кода: <foo>, [foo].

Подробнее о форматировании