views_panes.inc

Content type plugin to allow Views to be exposed as a display type, leaving most of the configuration on the view.

File

views_content/plugins/content_types/views_panes.inc

View source
<?php


/**
 * @file
 * Content type plugin to allow Views to be exposed as a display type,
 * leaving most of the configuration on the view.
 */

/**
 * Implements hook_ctools_content_types()
 */
function views_content_views_panes_ctools_content_types() {
    return array(
        'title' => t('View panes'),
        'admin settings' => 'views_content_admin_form',
        'all contexts' => TRUE,
    );
}

/**
 * Return all content types available.
 */
function views_content_views_panes_content_type_content_types($plugin) {
    $types = array();
    // It can be fairly intensive to calculate this, so let's cache this in the
    // cache_views table. The nice thing there is that if views ever change, that
    // table will always be cleared. Except for the occasional default view, so
    // we must use the Views caching functions in order to respect Views caching
    // settings.
    views_include('cache');
    $data = views_cache_get('views_content_panes', TRUE);
    if (!empty($data->data)) {
        $types = $data->data;
    }
    if (empty($types)) {
        $types = array();
        $views = views_get_all_views();
        foreach ($views as $view) {
            if (!empty($view->disabled)) {
                continue;
            }
            $view->init_display();
            foreach ($view->display as $id => $display) {
                if (empty($display->handler->panel_pane_display)) {
                    continue;
                }
                $info = _views_content_panes_content_type($view, $display);
                if ($info) {
                    $types[$view->name . '-' . $id] = $info;
                }
            }
            $view->destroy();
        }
        views_cache_set('views_content_panes', $types, TRUE);
    }
    return $types;
}

/**
 * Return a single content type.
 */
function views_content_views_panes_content_type_content_type($subtype, $plugin) {
    list($name, $display) = explode('-', $subtype);
    $view = views_get_view($name);
    if (empty($view)) {
        return;
    }
    $view->set_display($display);
    $retval = _views_content_panes_content_type($view, $view->display[$display]);
    $view->destroy();
    return $retval;
}
function _views_content_panes_content_type($view, $display) {
    // Ensure the handler is the right type, as Views will fall back to
    // the default display if something is broken:
    if (!is_a($display->handler, 'views_content_plugin_display_panel_pane')) {
        return;
    }
    $title = views_content_get_display_title($view, $display->id);
    $description = $display->handler
        ->get_option('pane_description');
    if (!$description) {
        $description = $view->description;
    }
    $category = $display->handler
        ->get_option('pane_category');
    if (!$category['name']) {
        $category['name'] = t('View panes');
    }
    $icon = 'icon_views_page.png';
    $contexts = array();
    $arguments = $display->handler
        ->get_argument_input();
    ctools_include('views');
    foreach ($arguments as $argument) {
        $contexts[] = ctools_views_get_argument_context($argument);
    }
    $allow = $display->handler
        ->get_option('allow');
    return array(
        'title' => $title,
        'icon' => $icon,
        'description' => filter_xss_admin($description),
        'required context' => $contexts,
        'category' => array(
            $category['name'],
            $category['weight'],
        ),
        'no title override' => empty($allow['title_override']),
    );
}

/**
 * Output function for the 'views' content type.
 *
 * Outputs a view based on the module and delta supplied in the configuration.
 */
function views_content_views_panes_content_type_render($subtype, $conf, $panel_args, $contexts) {
    if (!is_array($contexts)) {
        $contexts = array(
            $contexts,
        );
    }
    list($name, $display) = explode('-', $subtype);
    $view = views_get_view($name);
    if (empty($view)) {
        return;
    }
    $view->set_display($display);
    views_content_views_panes_add_defaults($conf, $view);
    if (!$view->display_handler
        ->access($GLOBALS['user']) || empty($view->display_handler->panel_pane_display)) {
        return;
    }
    $view->display_handler
        ->set_pane_conf($conf);
    $args = array();
    $arguments = $view->display_handler
        ->get_option('arguments');
    $context_keys = isset($conf['context']) ? $conf['context'] : array();
    foreach ($view->display_handler
        ->get_argument_input() as $id => $argument) {
        switch ($argument['type']) {
            case 'context':
                $key = array_shift($context_keys);
                if (isset($contexts[$key])) {
                    if (strpos($argument['context'], '.')) {
                        list($context, $converter) = explode('.', $argument['context'], 2);
                        $args[] = ctools_context_convert_context($contexts[$key], $converter, array(
                            'sanitize' => FALSE,
                        ));
                    }
                    else {
                        $args[] = $contexts[$key]->argument;
                    }
                }
                else {
                    $args[] = isset($arguments[$id]['exception']['value']) ? $arguments[$id]['exception']['value'] : 'all';
                }
                break;
            case 'fixed':
                $args[] = $argument['fixed'];
                break;
            case 'panel':
                $args[] = isset($panel_args[$argument['panel']]) ? $panel_args[$argument['panel']] : NULL;
                break;
            case 'user':
                $args[] = isset($conf['arguments'][$id]) && $conf['arguments'][$id] !== '' ? ctools_context_keyword_substitute($conf['arguments'][$id], array(), $contexts) : NULL;
                break;
            case 'wildcard':
                // Put in the wildcard.
                $args[] = isset($arguments[$id]['wildcard']) ? $arguments[$id]['wildcard'] : '*';
                break;
            case 'none':
            default:
                // Put in NULL.
                // views.module knows what to do with NULL (or missing) arguments.
                $args[] = NULL;
                break;
        }
    }
    // Remove any trailing NULL arguments as these are non-args:
    while (count($args) && end($args) === NULL) {
        array_pop($args);
    }
    $view->set_arguments($args);
    $allow = $view->display_handler
        ->get_option('allow');
    if (!empty($conf['path'])) {
        $conf['path'] = ctools_context_keyword_substitute($conf['path'], array(), $contexts);
    }
    if ($allow['path_override'] && !empty($conf['path'])) {
        $view->override_path = $conf['path'];
    }
    elseif ($path = $view->display_handler
        ->get_option('inherit_panels_path')) {
        if (drupal_is_front_page()) {
            $view->override_path = '';
        }
        else {
            $view->override_path = $_GET['q'];
        }
    }
    $block = new stdClass();
    $block->module = 'views';
    $block->delta = $view->name . '-' . $display;
    if ($allow['link_to_view'] && !empty($conf['link_to_view']) || !$allow['link_to_view'] && $view->display_handler
        ->get_option('link_to_view')) {
        $block->title_link = $view->get_url();
    }
    // More link.
    if ($allow['more_link']) {
        if (empty($conf['more_link'])) {
            $view->display_handler
                ->set_option('use_more', FALSE);
        }
        else {
            $view->display_handler
                ->set_option('use_more', TRUE);
            // Make sure the view runs the count query so we know whether or not the
            // more link applies.
            $view->get_total_rows = TRUE;
        }
    }
    if ($allow['items_per_page'] && isset($conf['items_per_page'])) {
        $view->set_items_per_page($conf['items_per_page']);
    }
    if ($allow['offset']) {
        $view->set_offset($conf['offset']);
    }
    if ($allow['use_pager']) {
        // Only set use_pager if they differ, this way we can avoid overwriting the
        // pager type that Views uses.
        $pager = $view->display_handler
            ->get_option('pager');
        if ($conf['use_pager'] && ($pager['type'] == 'none' || $pager['type'] == 'some')) {
            $pager['type'] = 'full';
        }
        elseif (!$conf['use_pager']) {
            $pager['type'] = $view->get_items_per_page() || !empty($pager['options']['items_per_page']) ? 'some' : 'none';
        }
        if ($conf['use_pager']) {
            if (!isset($pager['options']['id']) || isset($conf['pager_id']) && $pager['options']['id'] != $conf['pager_id']) {
                $pager['options']['id'] = (int) $conf['pager_id'];
            }
        }
        $view->display_handler
            ->set_option('pager', $pager);
    }
    if ($allow['fields_override']) {
        if ($conf['fields_override']) {
            $fields = $view->get_items('field');
            foreach ($fields as $field => $field_display) {
                $fields[$field]['exclude'] = empty($conf['fields_override'][$field]);
            }
            $view->display_handler
                ->set_option('fields', $fields);
        }
    }
    if ($allow['exposed_form'] && !empty($conf['exposed'])) {
        foreach ($conf['exposed'] as $filter_name => $filter_value) {
            if (!is_array($filter_value)) {
                $conf['exposed'][$filter_name] = ctools_context_keyword_substitute($filter_value, array(), $contexts);
            }
        }
        $view->set_exposed_input($conf['exposed']);
    }
    $stored_feeds = drupal_add_feed();
    $block->content = $view->preview();
    if (empty($view->result) && !$view->display_handler
        ->get_option('empty') && empty($view->style_plugin->definition['even empty'])) {
        return;
    }
    // Add contextual links to the output.
    $block = (array) $block;
    views_add_block_contextual_links($block, $view, $display, 'panel_pane');
    $block = (object) $block;
    $block->title = $view->get_title();
    if (empty($view->total_rows) || $view->total_rows <= $view->get_items_per_page()) {
        unset($block->more);
    }
    if (!empty($allow['feed_icons']) && !empty($conf['feed_icons']) || empty($allow['feed_icons']) && $view->display_handler
        ->get_option('feed_icons')) {
        $new_feeds = drupal_add_feed();
        if ($diff = array_diff(array_keys($new_feeds), array_keys($stored_feeds))) {
            foreach ($diff as $url) {
                $block->feeds[$url] = $new_feeds[$url];
            }
        }
    }
    return $block;
}

/**
 * Add defaults to view pane settings.
 * This helps cover us if $allow settings changed and there are no actual
 * settings for a particular item.
 */
function views_content_views_panes_add_defaults(&$conf, $view) {
    $pager = $view->display_handler
        ->get_option('pager');
    if (empty($conf)) {
        $conf = array();
    }
    $conf += array(
        'link_to_view' => $view->display_handler
            ->get_option('link_to_view'),
        'more_link' => $view->display_handler
            ->get_option('use_more'),
        'feed_icons' => FALSE,
        'use_pager' => $pager['type'] != 'none' && $pager['type'] != 'some',
        'pager_id' => isset($pager['options']['id']) ? $pager['options']['id'] : 0,
        'items_per_page' => !empty($pager['options']['items_per_page']) ? $pager['options']['items_per_page'] : 10,
        'offset' => !empty($pager['options']['offset']) ? $pager['options']['offset'] : 0,
        'path_override' => FALSE,
        'path' => $view->get_path(),
        'fields_override' => $view->display_handler
            ->get_option('fields_override'),
    );
}

/**
 * Returns an edit form for a block.
 */
function views_content_views_panes_content_type_edit_form($form, &$form_state) {
    $conf = $form_state['conf'];
    $contexts = $form_state['contexts'];
    // This allows older content to continue to work, where we used to embed
    // the display directly.
    list($name, $display_id) = explode('-', $form_state['subtype_name']);
    $view = views_get_view($name);
    if (empty($view)) {
        $form['markup'] = array(
            '#markup' => t('Broken/missing/deleted view.'),
        );
        return $form;
    }
    $view->set_display($display_id);
    // If it couldn't set the display and we got the default display instead,
    // fail.
    if ($view->current_display == 'default') {
        $form['markup'] = array(
            '#markup' => t('Broken/missing/deleted view display.'),
        );
        return $form;
    }
    $allow = $view->display_handler
        ->get_option('allow');
    // Provide defaults for everything in order to prevent warnings.
    views_content_views_panes_add_defaults($conf, $view);
    $form['arguments']['#tree'] = TRUE;
    foreach ($view->display_handler
        ->get_argument_input() as $id => $argument) {
        if ($argument['type'] == 'user') {
            $form['arguments'][$id] = array(
                '#type' => 'textfield',
                '#default_value' => isset($conf['arguments'][$id]) ? $conf['arguments'][$id] : '',
                '#description' => t('You may use keywords for substitutions.'),
                '#title' => check_plain($argument['label']),
            );
        }
    }
    if ($allow['link_to_view']) {
        $form['link_to_view'] = array(
            '#type' => 'checkbox',
            '#default_value' => isset($conf['link_to_view']) ? $conf['link_to_view'] : $view->display_handler
                ->get_option('link_to_view'),
            '#title' => t('Link title to page'),
        );
    }
    if ($allow['more_link']) {
        $form['more_link'] = array(
            '#type' => 'checkbox',
            '#default_value' => isset($conf['more_link']) ? $conf['more_link'] : $view->display_handler
                ->get_option('use_more'),
            '#description' => t('The text of this link will be "@more". This setting can only be modified on the View configuration.', array(
                '@more' => $view->display_handler
                    ->use_more_text(),
            )),
            '#title' => t('Provide a "more" link.'),
        );
    }
    if (!empty($allow['feed_icons'])) {
        $form['feed_icons'] = array(
            '#type' => 'checkbox',
            '#default_value' => !empty($conf['feed_icons']),
            '#title' => t('Display feed icons'),
        );
    }
    $view->init_style();
    if ($allow['fields_override'] && $view->style_plugin
        ->uses_fields()) {
        $form['fields_override'] = array(
            '#type' => 'fieldset',
            '#title' => 'Fields to display',
            '#collapsible' => TRUE,
            '#tree' => TRUE,
        );
        foreach ($view->display_handler
            ->get_handlers('field') as $field => $handler) {
            $title = $handler->ui_name();
            if ($handler->options['label']) {
                $title .= ' (' . check_plain($handler->options['label']) . ')';
            }
            $form['fields_override'][$field] = array(
                '#type' => 'checkbox',
                '#title' => $title,
                '#default_value' => isset($conf['fields_override'][$field]) ? $conf['fields_override'][$field] : !$handler->options['exclude'],
            );
        }
    }
    ctools_include('dependent');
    if ($allow['use_pager']) {
        $form['use_pager'] = array(
            '#type' => 'checkbox',
            '#title' => t('Use pager'),
            '#default_value' => $conf['use_pager'],
            '#id' => 'use-pager-checkbox',
            '#prefix' => '<div class="container-inline">',
        );
        $form['pager_id'] = array(
            '#type' => 'textfield',
            '#default_value' => $conf['pager_id'],
            '#title' => t('Pager ID'),
            '#size' => 4,
            '#id' => 'use-pager-textfield',
            '#dependency' => array(
                'use-pager-checkbox' => array(
                    1,
                ),
            ),
            '#suffix' => '</div>',
        );
    }
    if ($allow['items_per_page']) {
        $form['items_per_page'] = array(
            '#type' => 'textfield',
            '#default_value' => $conf['items_per_page'],
            '#title' => t('Num items'),
            '#size' => 4,
            '#description' => t('Select the number of items to display, or 0 to display all results.'),
        );
    }
    if ($allow['offset']) {
        $form['offset'] = array(
            '#type' => 'textfield',
            '#default_value' => $conf['offset'],
            '#title' => t('Offset'),
            '#size' => 4,
            '#description' => t('Enter the number of items to skip; enter 0 to skip no items.'),
        );
    }
    if ($allow['path_override']) {
        $form['path'] = array(
            '#type' => 'textfield',
            '#default_value' => isset($conf['path']) ? $conf['path'] : $view->get_path(),
            '#title' => t('Override path'),
            '#size' => 30,
            '#description' => t('If this is set, override the View URL path; this can sometimes be useful to set to the panel URL.'),
        );
        if (!empty($contexts)) {
            $form['path']['#description'] .= ' ' . t('You may use substitutions in this path.');
            $form['contexts'] = array(
                '#type' => 'fieldset',
                '#title' => t('Substitutions'),
                '#collapsible' => TRUE,
                '#collapsed' => TRUE,
            );
            $rows = array();
            foreach ($contexts as $context) {
                foreach (ctools_context_get_converters('%' . check_plain($context->keyword) . ':', $context) as $keyword => $title) {
                    $rows[] = array(
                        check_plain($keyword),
                        t('@identifier: @title', array(
                            '@title' => $title,
                            '@identifier' => $context->identifier,
                        )),
                    );
                }
            }
            $header = array(
                t('Keyword'),
                t('Value'),
            );
            $form['contexts']['context'] = array(
                '#markup' => theme('table', array(
                    'header' => $header,
                    'rows' => $rows,
                )),
            );
        }
    }
    if (empty($conf['exposed'])) {
        $conf['exposed'] = array();
    }
    if ($allow['exposed_form']) {
        // If the exposed form is part of pane configuration, get the exposed
        // form re-tool it for our use.
        $exposed_form_state = array(
            'view' => &$view,
            'display' => &$view->display[$display_id],
        );
        $view->set_exposed_input($conf['exposed']);
        if (version_compare(views_api_version(), '3', '>=')) {
            $exposed_form_state['exposed_form_plugin'] = $view->display_handler
                ->get_plugin('exposed_form');
        }
        $view->init_handlers();
        $exposed_form = array();
        $exposed_form = views_exposed_form($exposed_form, $exposed_form_state);
        $form['exposed'] = array(
            '#tree' => TRUE,
        );
        foreach ($exposed_form['#info'] as $id => $info) {
            $form['exposed'][$id] = array(
                '#type' => 'item',
                '#id' => 'views-exposed-pane',
            );
            if (!empty($info['label'])) {
                $form['exposed'][$id]['#title'] = $info['label'];
            }
            if (!empty($info['operator']) && !empty($exposed_form[$info['operator']])) {
                $form['exposed'][$id][$info['operator']] = $exposed_form[$info['operator']];
                $form['exposed'][$id][$info['operator']]['#parents'] = array(
                    'exposed',
                    $info['operator'],
                );
                $form['exposed'][$id][$info['operator']]['#default_value'] = isset($conf['exposed'][$info['operator']]) ? $conf['exposed'][$info['operator']] : '';
            }
            $form['exposed'][$id][$info['value']] = $exposed_form[$info['value']];
            $form['exposed'][$id][$info['value']]['#parents'] = array(
                'exposed',
                $info['value'],
            );
            $form['exposed'][$id][$info['value']]['#default_value'] = isset($conf['exposed'][$info['value']]) ? $conf['exposed'][$info['value']] : '';
        }
    }
    // The exposed sort stuff doesn't fall into $exposed_form['#info'] so we
    // have to handle it separately.
    if (isset($exposed_form['sort_by'])) {
        $form['exposed']['sort_by'] = $exposed_form['sort_by'];
    }
    if (isset($exposed_form['sort_order'])) {
        $form['exposed']['sort_order'] = $exposed_form['sort_order'];
    }
    // Add the view object to the form to allow additional customization.
    $form_state['view'] = $view;
    return $form;
}

/**
 * Store form values in $conf.
 */
function views_content_views_panes_content_type_edit_form_submit(&$form, &$form_state) {
    // Copy everything from our defaults.
    $keys = array(
        'link_to_view',
        'more_link',
        'feed_icons',
        'use_pager',
        'pager_id',
        'items_per_page',
        'offset',
        'path_override',
        'path',
        'arguments',
        'fields_override',
        'exposed',
    );
    foreach ($keys as $key) {
        if (isset($form_state['values'][$key])) {
            $form_state['conf'][$key] = $form_state['values'][$key];
        }
    }
}

/**
 * Returns the administrative title for a type.
 */
function views_content_views_panes_content_type_admin_title($subtype, $conf, $contexts) {
    list($name, $display) = explode('-', $subtype);
    $view = views_get_view($name);
    if (empty($view) || empty($view->display[$display])) {
        return t('Deleted/missing view @view', array(
            '@view' => $name,
        ));
    }
    $view->set_display($display);
    views_content_views_panes_add_defaults($conf, $view);
    $title = views_content_get_display_title($view, $display);
    return check_plain($title);
}

/**
 * Returns the administrative title for a type.
 */
function views_content_views_panes_content_type_admin_info($subtype, $conf, $contexts) {
    $info = array();
    list($view_name, $display_name) = explode('-', $subtype);
    $view = views_get_view($view_name);
    if (empty($view) || empty($view->display[$display_name])) {
        return;
    }
    $view->set_display($display_name);
    views_content_views_panes_add_defaults($conf, $view);
    // Add arguments first.
    if (!empty($conf['arguments'])) {
        $keys = array_keys($conf['arguments']);
        $values = array_values($conf['arguments']);
        $argument_input = $view->display_handler
            ->get_option('argument_input');
        foreach ($conf['arguments'] as $key => $value) {
            if (!empty($value)) {
                $label = $argument_input[$key]['label'];
                $info[] = $label . ': ' . $value;
            }
        }
    }
    $block = new stdClass();
    if ($info) {
        $block->title = array_shift($info);
        $info[] = $view->display_handler
            ->get_option('pane_description');
        $block->content = theme('item_list', array(
            'items' => $info,
        ));
    }
    else {
        $block->title = $view->display_handler
            ->get_option('pane_description');
        $block->content = '';
    }
    return $block;
}

Functions

Title Deprecated Summary
views_content_views_panes_add_defaults Add defaults to view pane settings. This helps cover us if $allow settings changed and there are no actual settings for a particular item.
views_content_views_panes_content_type_admin_info Returns the administrative title for a type.
views_content_views_panes_content_type_admin_title Returns the administrative title for a type.
views_content_views_panes_content_type_content_type Return a single content type.
views_content_views_panes_content_type_content_types Return all content types available.
views_content_views_panes_content_type_edit_form Returns an edit form for a block.
views_content_views_panes_content_type_edit_form_submit Store form values in $conf.
views_content_views_panes_content_type_render Output function for the 'views' content type.
views_content_views_panes_ctools_content_types Implements hook_ctools_content_types()
_views_content_panes_content_type