Расширение поиска в Drupal 7 - PullRequest
4 голосов
/ 20 декабря 2011

Я хочу расширить поиск узлов по умолчанию в Drupal 7. одним дополнительным полем.

Я изменяю форму поиска следующим новым полем:

function mymodule_form_search_form_alter(&$form, &$form_state, $form_id) {
    $form['basic']['site'] = array(
        '#type' => 'select',
        '#options' => array(
            'KEY1' => 'TITLE1',
            'KEY2' => 'TITLE2',
            'KEY3' => 'TITLE3'
        )
    );
}

У меня есть поле с именем field_data_field_site.field_site_valueкоторый я должен использовать в качестве фильтра в этом поиске.

Я пытался прочитать о hook_search_* функциях, но не понял.

Мой вопрос заключается в следующем.Как я могу расширить форму поиска?У кого-нибудь есть живые примеры?

1 Ответ

11 голосов
/ 21 декабря 2011

Следующее - лучший способ решить эту проблему.

Прежде всего мне нужно изменить блок поиска и форму поиска в Drupal с моим полем и определить новую функцию отправки.

/**
 * Implements hook_form_FORM_ID_alter().
 */
function mymodule_form_search_block_form_alter(&$form, &$form_state, $form_id) {
    $form['#submit'][] = 'search_form_alter_submit';

    $form['site'] = array(
        '#type' => 'select',
        '#options' => _options(),
        '#default_value' => (($_GET['site']) ? $_GET['site'] : '')
    );
}

/**
 * Implements hook_form_FORM_ID_alter().
 */
function mymodule_form_search_form_alter(&$form, &$form_state, $form_id) {
    $form['#submit'][] = 'search_form_alter_submit';

    $form['basic']['site'] = array(
        '#type' => 'select',
        '#options' => _options(),
        '#default_value' => (($_GET['site']) ? $_GET['site'] : '')
    );
}

function _options() {
    return array(
        '' => 'Select site',
        'site-1' => 'Site 1',
        'site-2' => 'Site 2'
    );
}

Функция отправки отправит нас на страницу по умолчанию search/node, но с нашим запросом. Страница будет выглядеть как search/node/Our-query-string?site=Our-option-selected.

function search_form_alter_submit($form, &$form_state) {
    $path = $form_state['redirect'];
    $options = array(
        'query' => array(
            'site' => $form_state['values']['site']
        )
    );
    drupal_goto($path, $options);
}

Следующим шагом будет использование hook_search_info (не забудьте включить его и установить по умолчанию на странице admin/config/search/settings).

/**
 * Implements hook_search_info().
 */
function mymodule_search_info() {
    return array(
        'title' => 'Content',
        'path' => 'node',
        'conditions_callback' => '_conditions_callback',
    );
}

Условия функции обратного вызова определены в hook_search_info. Мы должны предоставить дополнительные запросы для нашего поиска.

function _conditions_callback($keys) {
    $conditions = array();
    if (!empty($_REQUEST['site'])) {
        $conditions['site'] = $_REQUEST['site'];
    }
    return $conditions;
}

Наконец, hook_search_execute отфильтрует наш контент по нашему запросу. Я использовал код по умолчанию из этого хука с нужными мне модификациями.

/**
 * Implements hook_search_execute().
 */
function mymodule_search_execute($keys = NULL, $conditions = NULL) {
    // Build matching conditions
    $query = db_select('search_index', 'i', array('target' => 'slave'))
        ->extend('SearchQuery')
        ->extend('PagerDefault');

    $query->join('node', 'n', 'n.nid = i.sid');

    // Here goes my filter where I joined another table and
    // filter by required field
    $site = (isset($conditions['site'])) ? $conditions['site'] : NULL;
    if ($site) {
        $query->leftJoin('field_data_field_site', 's', 's.entity_id = i.sid');
        $query->condition('s.field_site_value', $site);
    }
    // End of my filter

    $query
        ->condition('n.status', 1)
        ->addTag('node_access')
        ->searchExpression($keys, 'node');

    // Insert special keywords.
    $query->setOption('type', 'n.type');
    $query->setOption('language', 'n.language');
    if ($query->setOption('term', 'ti.tid')) {
        $query->join('taxonomy_index', 'ti', 'n.nid = ti.nid');
    }

    // Only continue if the first pass query matches.
    if (!$query->executeFirstPass()) {
        return array();
    }

    // Add the ranking expressions.
    _node_rankings($query);

    // Load results.
    $find = $query
        ->limit(10)
        ->execute();

    $results = array();
    foreach ($find as $item) {
        // Build the node body.
        $node = node_load($item->sid);
        node_build_content($node, 'search_result');
        $node->body = drupal_render($node->content);

        // Fetch comments for snippet.
        $node->rendered .= ' ' . module_invoke('comment', 'node_update_index', $node);
        // Fetch terms for snippet.
        $node->rendered .= ' ' . module_invoke('taxonomy', 'node_update_index', $node);

        $extra = module_invoke_all('node_search_result', $node);

        $results[] = array(
            'link' => url("node/{$item->sid}", array('absolute' => TRUE)),
            'type' => check_plain(node_type_get_name($node)),
            'title' => $node->title,
            'user' => theme('username', array('account' => $node)),
            'date' => $node->changed,
            'node' => $node,
            'extra' => $extra,
            'score' => $item->calculated_score,
            'snippet' => search_excerpt($keys, $node->body)
        );
    }

    return $results;
}

Я был бы рад, если бы кто-нибудь дал мне лучший ответ.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...