views_panes.inc in Chaos Tool Suite (ctools) 7
Same filename and directory in other branches
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.incView 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
Name | Description |
---|---|
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 |