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

DrupalИзменить расположение public, private и temporary директорий из инсталляционного профиля

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

Задача — изменить расположение public://, private:// и temporary:// директорий при инсталляции друпала с помощью своего профиля.

Сложность — эти директории теперь прописываются только в settings.php и менять их нужно на самой ранней стадии установки друпала.

Решение — в hook_install_tasks_alter() добавляем свой таск, копируем файл настроек при необходимости, изменяем его, сбрасываем кэш:

DrupalУстановка Drupal 8 с помощью Composer и бубна

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

Всё чаще стали предлагать работу на Drupal 8, а я ещё толком за него и не брался. Дай думаю для начала создам свой профиль и переведу блог на восьмёрку, благо совсем недавно вышла версия 8.4.

DrupalУстановить дефолтное число значений для unlimited поля

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

Код позволяет в настройках поля выбрать первоначальное число значений для unlimited поля:

/**
 * Implements hook_form_FORM_ID_alter(): field_ui_field_edit_form.
 */
function hook_form_field_ui_field_edit_form_alter(&$form, $form_state) {
  $field_instance = $form['#instance'];
 
  $form['instance']['settings']['default_cardinality'] = array(
    '#type' => 'textfield',
    '#title' => t('Default cardinality'),
    '#default_value' => isset($field_instance['settings']['default_cardinality']) ? $field_instance['settings']['default_cardinality'] : 1,
    '#weight' => 10,

DrupalФиксим вывод reCaptcha на AJAX формах

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

Разработчики модуля reCAPTCHA третий год не могут заставить работать капчу на ajax формах. Проблема в следующем — при перезагрузке формы с помощью AJAX капча пропадает, поскольку модуль никак не реагирует на подгрузку динамического контента. Фиксим самостоятельно:

(function ($) {
  Drupal.behaviors.mymodule = {
    attach: function (context, settings) {
      if ('grecaptcha' in window && context !== document) {

DrupalПоследовательно запустить два Batch-а

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

Если нужно выполнить последовательно два Batch-а и операции второго Batch-а не зависят от результатов первого, то достаточно подряд вызвать batch_set(), а если зависят, то:

function mymodule_myform_submit($form, &$form_state) {
  // Создаём операции для первого batch-а
  $operations = array();
  for ($i = 0; $i < 100; $i++) {
    $operations[] = array('mymodule_first_batch_operation', array());
  }
 
  // В конец добавляем операцию для запуска второго batch-а
  $operations[] = array('mymodule_start_second_batch', array());
 
  // Запускаем первый batch

DrupalКак передать переменную из конструктора Batch в его finished callback

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

Пример передачи значения переменной foo в mymofule_batch_finished():

/**
 * Form submit.
 */
function mymodule_myform_submit($form, &$form_state) {
  ...
  $batch = array(
    'operations' => $operations,
    'finished' => 'mymodule_batch_finished',
    'foo' => 'bar',
  );
 
  batch_set($batch);
}
 
/**
 * Batch finished callback.
 */
function mymodule_batch_finished($success, $results, $operations) {
  $batch_set = _batch_current_set();
  debug($batch_set['foo']);
}

DrupalАвтоматическое удаление устаревших анонимных заказов в статусе "корзина"

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

Drupal Commerce самостоятельно не удаляет устаревшие анонимные заказы в статусе "корзина", что приводит к бессмысленному распуханию таблицы commerce_order.

"Устаревшими" я считаю заказы анонимных пользователей, чьи сессии были удалены сборщиком мусора (по умолчанию это сессии старше 3 дней).

Решаем проблему:

/**
 * Implements hook_cron().
 */
function hook_cron() {
  if (module_exists('commerce_order')) {
    $order_ids = db_select('commerce_order', 'o')
      ->fields('o', array('order_id'))
      ->condition('o.uid', 0)

DrupalДобавить класс определённому элементу перед выполнением ajax запроса

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

Очень часто перед выполнением ajax запроса нужно добавить какому-то элементу определённый класс, а после получения ответа убрать его, чтобы например показать какую-нибудь анимашу, мол keep calm и всё такое.

Реализовать можно с помощью создания собственного типа ajax-прогресса:

(function ($) {
  if (Drupal.ajax) {
    var ajaxBeforeSend = Drupal.ajax.prototype.beforeSend;
    var ajaxSuccess = Drupal.ajax.prototype.success;
    var ajaxError = Drupal.ajax.prototype.error;
 
    Drupal.ajax.prototype.beforeSend = function(xmlhttprequest, options) {

DrupalВысылать письмо при входе администратора с нового браузера

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

Небольшой код, который уведомляет о входе администратора с нового браузера:

/**
 * Implements hook_user_login().
 */
function modulename_user_login(&$edit, $account) {
  if (
    $account->uid == 1 &&
    (empty($account->data['useragent']) || $account->data['useragent'] != md5($_SERVER['HTTP_USER_AGENT']))
  ) {
    $message = 'Дата: ' . date('r') . "\n";
    $message .= 'IP: ' . ip_address() . "\n";
    $message .= 'Host: ' . @gethostbyaddr(ip_address()) . "\n";
    $message .= 'User Agent: ' . $_SERVER['HTTP_USER_AGENT'] . "\n";