views_plugin_style.inc

Definition of views_plugin_style.

File

plugins/views_plugin_style.inc

View source
<?php


/**
 * @file
 * Definition of views_plugin_style.
 */

/**
 * @defgroup views_style_plugins Views style plugins
 * @{
 * Style plugins control how a view is rendered. For example, they can choose to
 * display a collection of fields, node_view() output, table output, or any kind
 * of crazy output they want.
 *
 * Many style plugins can have an optional 'row' plugin, that displays a single
 * record. Not all style plugins can utilize this, so it is up to the plugin to
 * set this up and call through to the row plugin.
 *
 * @see hook_views_plugins()
 */

/**
 * Base class to define a style plugin handler.
 */
class views_plugin_style extends views_plugin {
    
    /**
     * Store all available tokens row rows.
     */
    public $row_tokens = array();
    
    /**
     * The row plugin, if it's initialized and the style itself supports it.
     *
     * @var views_plugin_row
     */
    public $row_plugin;
    
    /**
     * Initialize a style plugin.
     *
     * @param view $view
     * @param object $display
     * @param array $options
     *   The style options might come externally as the style can be sourced
     *   from at least two locations. If it's not included, look on the display.
     */
    public function init(&$view, &$display, $options = NULL) {
        $this->view =& $view;
        $this->display =& $display;
        // Overlay incoming options on top of defaults.
        $this->unpack_options($this->options, isset($options) ? $options : $display->handler
            ->get_option('style_options'));
        if ($this->uses_row_plugin() && $display->handler
            ->get_option('row_plugin')) {
            $this->row_plugin = $display->handler
                ->get_plugin('row');
        }
        $this->options += array(
            'grouping' => array(),
        );
        $this->definition += array(
            'uses grouping' => TRUE,
        );
    }
    
    /**
     *
     */
    public function destroy() {
        parent::destroy();
        if ($this->row_plugin) {
            $this->row_plugin
                ->destroy();
        }
    }
    
    /**
     * Return TRUE if this style also uses a row plugin.
     */
    public function uses_row_plugin() {
        return !empty($this->definition['uses row plugin']);
    }
    
    /**
     * Return TRUE if this style also uses a row plugin.
     */
    public function uses_row_class() {
        return !empty($this->definition['uses row class']);
    }
    
    /**
     * Return TRUE if this style also uses fields.
     *
     * @return bool
     */
    public function uses_fields() {
        // If we use a row plugin, ask the row plugin. Chances are, we don't
        // care, it does.
        $row_uses_fields = FALSE;
        if ($this->uses_row_plugin() && !empty($this->row_plugin)) {
            $row_uses_fields = $this->row_plugin
                ->uses_fields();
        }
        // Otherwise, check the definition or the option.
        return $row_uses_fields || !empty($this->definition['uses fields']) || !empty($this->options['uses_fields']);
    }
    
    /**
     * Return TRUE if this style uses tokens.
     *
     * Used to ensure we don't fetch tokens when not needed for performance.
     */
    public function uses_tokens() {
        if ($this->uses_row_class()) {
            $class = '';
            if (isset($this->options['row_class'])) {
                $class = $this->options['row_class'];
            }
            if (strpos($class, '[') !== FALSE || strpos($class, '!') !== FALSE || strpos($class, '%') !== FALSE) {
                return TRUE;
            }
        }
    }
    
    /**
     * Return the token replaced row class for the specified row.
     */
    public function get_row_class($row_index) {
        if ($this->uses_row_class()) {
            $class = $this->options['row_class'];
            if ($this->uses_fields() && $this->view->field) {
                $classes = array();
                // Explode the value by whitespace, this allows the function to handle
                // a single class name and multiple class names that are then tokenized.
                foreach (explode(' ', $class) as $token_class) {
                    $classes = array_merge($classes, explode(' ', strip_tags($this->tokenize_value($token_class, $row_index))));
                }
            }
            else {
                $classes = explode(' ', $class);
            }
            // Convert whatever the result is to a nice clean class name.
            foreach ($classes as &$class) {
                $class = drupal_clean_css_identifier($class);
            }
            unset($class);
            return implode(' ', $classes);
        }
    }
    
    /**
     * Take a value and apply token replacement logic to it.
     */
    public function tokenize_value($value, $row_index) {
        if (!isset($value)) {
            $value = '';
        }
        if (strpos($value, '[') !== FALSE || strpos($value, '!') !== FALSE || strpos($value, '%') !== FALSE) {
            // Row tokens might be empty, for example for node row style.
            $tokens = isset($this->row_tokens[$row_index]) ? $this->row_tokens[$row_index] : array();
            if (!empty($this->view->build_info['substitutions'])) {
                $tokens += $this->view->build_info['substitutions'];
            }
            if ($tokens) {
                $value = strtr($value, $tokens);
            }
        }
        return $value;
    }
    
    /**
     * Should the output of the style plugin be rendered even if it's empty.
     */
    public function even_empty() {
        return !empty($this->definition['even empty']);
    }
    
    /**
     * {@inheritdoc}
     */
    public function option_definition() {
        $options = parent::option_definition();
        $options['grouping'] = array(
            'default' => array(),
        );
        if ($this->uses_row_class()) {
            $options['row_class'] = array(
                'default' => '',
            );
            $options['default_row_class'] = array(
                'default' => TRUE,
                'bool' => TRUE,
            );
            $options['row_class_special'] = array(
                'default' => TRUE,
                'bool' => TRUE,
            );
        }
        $options['uses_fields'] = array(
            'default' => FALSE,
            'bool' => TRUE,
        );
        return $options;
    }
    
    /**
     * {@inheritdoc}
     */
    public function options_form(&$form, &$form_state) {
        parent::options_form($form, $form_state);
        // Only fields-based views can handle grouping. Style plugins can also
        // exclude themselves from being groupable by setting their "use grouping"
        // definition key to FALSE.
        // @todo Document "uses grouping" in docs.php when docs.php is written.
        if ($this->uses_fields() && $this->definition['uses grouping']) {
            $options = array(
                '' => t('- None -'),
            );
            $field_labels = $this->display->handler
                ->get_field_labels(TRUE);
            $options += $field_labels;
            // If there are no fields, we can't group on them.
            if (count($options) > 1) {
                // This is for backward compatibility, when there was just a single
                // select form.
                if (is_string($this->options['grouping'])) {
                    $grouping = $this->options['grouping'];
                    $this->options['grouping'] = array();
                    $this->options['grouping'][0]['field'] = $grouping;
                }
                if (isset($this->options['group_rendered']) && is_string($this->options['group_rendered'])) {
                    $this->options['grouping'][0]['rendered'] = $this->options['group_rendered'];
                    unset($this->options['group_rendered']);
                }
                $c = count($this->options['grouping']);
                // Add a form for every grouping, plus one.
                for ($i = 0; $i <= $c; $i++) {
                    $grouping = !empty($this->options['grouping'][$i]) ? $this->options['grouping'][$i] : array();
                    $grouping += array(
                        'field' => '',
                        'rendered' => TRUE,
                        'rendered_strip' => FALSE,
                    );
                    $form['grouping'][$i]['field'] = array(
                        '#type' => 'select',
                        '#title' => t('Grouping field Nr.@number', array(
                            '@number' => $i + 1,
                        )),
                        '#options' => $options,
                        '#default_value' => $grouping['field'],
                        '#description' => t('You may optionally specify a field by which to group the records. Leave blank to not group.'),
                    );
                    $form['grouping'][$i]['rendered'] = array(
                        '#type' => 'checkbox',
                        '#title' => t('Use rendered output to group rows'),
                        '#default_value' => $grouping['rendered'],
                        '#description' => t('If enabled the rendered output of the grouping field is used to group the rows.'),
                        '#dependency' => array(
                            'edit-style-options-grouping-' . $i . '-field' => array_keys($field_labels),
                        ),
                    );
                    $form['grouping'][$i]['rendered_strip'] = array(
                        '#type' => 'checkbox',
                        '#title' => t('Remove tags from rendered output'),
                        '#default_value' => $grouping['rendered_strip'],
                        '#description' => t('Some modules add HTML to the rendered output and prevent the rows from grouping correctly. Stripping the HTML tags should correct this.'),
                        '#dependency' => array(
                            'edit-style-options-grouping-' . $i . '-field' => array_keys($field_labels),
                        ),
                    );
                }
            }
        }
        if ($this->uses_row_class()) {
            $form['row_class'] = array(
                '#title' => t('Row class'),
                '#description' => t('The class to provide on each row.'),
                '#type' => 'textfield',
                '#default_value' => $this->options['row_class'],
            );
            if ($this->uses_fields()) {
                $form['row_class']['#description'] .= ' ' . t('You may use field tokens from as per the "Replacement patterns" used in "Rewrite the output of this field" for all fields.');
            }
            $form['default_row_class'] = array(
                '#title' => t('Add views row classes'),
                '#description' => t('Add the default row classes like views-row-1 to the output. You can use this to quickly reduce the amount of markup the view provides by default, at the cost of making it more difficult to apply CSS.'),
                '#type' => 'checkbox',
                '#default_value' => $this->options['default_row_class'],
            );
            $form['row_class_special'] = array(
                '#title' => t('Add striping (odd/even), first/last row classes'),
                '#description' => t('Add css classes to the first and last line, as well as odd/even classes for striping.'),
                '#type' => 'checkbox',
                '#default_value' => $this->options['row_class_special'],
            );
        }
        if (!$this->uses_fields() || !empty($this->options['uses_fields'])) {
            $form['uses_fields'] = array(
                '#type' => 'checkbox',
                '#title' => t('Force using fields'),
                '#description' => t('If neither the row nor the style plugin supports fields, this field allows to enable them, so you can for example use groupby.'),
                '#default_value' => $this->options['uses_fields'],
            );
        }
    }
    
    /**
     * {@inheritdoc}
     */
    public function options_validate(&$form, &$form_state) {
        // Don't run validation on style plugins without the grouping setting.
        if (isset($form_state['values']['style_options']['grouping'])) {
            // Don't save grouping if no field is specified.
            foreach ($form_state['values']['style_options']['grouping'] as $index => $grouping) {
                if (empty($grouping['field'])) {
                    unset($form_state['values']['style_options']['grouping'][$index]);
                }
            }
        }
    }
    
    /**
     * Called by the view builder to see if this style handler wants to
     * interfere with the sorts. If so it should build; if it returns
     * any non-TRUE value, normal sorting will NOT be added to the query.
     */
    public function build_sort() {
        return TRUE;
    }
    
    /**
     * Called by the view builder to let the style build a second set of
     * sorts that will come after any other sorts in the view.
     */
    public function build_sort_post() {
    }
    
    /**
     * Allow the style to do stuff before each row is rendered.
     *
     * @param array $result
     *   The full array of results from the query.
     */
    public function pre_render($result) {
        if (!empty($this->row_plugin)) {
            $this->row_plugin
                ->pre_render($result);
        }
    }
    
    /**
     * Render the display in this style.
     */
    public function render() {
        if ($this->uses_row_plugin() && empty($this->row_plugin)) {
            debug('views_plugin_style_default: Missing row plugin');
            return;
        }
        // Group the rows according to the grouping instructions, if specified.
        $sets = $this->render_grouping($this->view->result, $this->options['grouping'], TRUE);
        return $this->render_grouping_sets($sets);
    }
    
    /**
     * Render the grouping sets.
     *
     * Plugins may override this method if they wish some other way of handling
     * grouping.
     *
     * @param array $sets
     *   Array containing the grouping sets to render.
     * @param int $level
     *   Integer indicating the hierarchical level of the grouping.
     *
     * @return string
     *   Rendered output of given grouping sets.
     */
    public function render_grouping_sets($sets, $level = 0) {
        $output = '';
        foreach ($sets as $set) {
            $row = reset($set['rows']);
            $level = isset($set['level']) ? $set['level'] : 0;
            // Render as a grouping set.
            if (is_array($row) && isset($row['group'])) {
                $output .= theme(views_theme_functions('views_view_grouping', $this->view, $this->display), array(
                    'view' => $this->view,
                    'grouping' => $this->options['grouping'][$level],
                    'grouping_level' => $level,
                    'rows' => $set['rows'],
                    'title' => $set['group'],
                ));
            }
            else {
                if ($this->uses_row_plugin()) {
                    foreach ($set['rows'] as $index => $row) {
                        $this->view->row_index = $index;
                        $set['rows'][$index] = $this->row_plugin
                            ->render($row);
                    }
                }
                $output .= theme($this->theme_functions(), array(
                    'view' => $this->view,
                    'options' => $this->options,
                    'grouping_level' => $level,
                    'rows' => $set['rows'],
                    'title' => $set['group'],
                ));
            }
        }
        unset($this->view->row_index);
        return $output;
    }
    
    /**
     * Group records as needed for rendering.
     *
     * @param array $records
     *   An array of records from the view to group.
     * @param array $groupings
     *   An array of grouping instructions on which fields to group. If empty, the
     *   result set will be given a single group with an empty string as a label.
     * @param bool $group_rendered
     *   Boolean value whether to use the rendered or the raw field value for
     *   grouping. If set to NULL the return is structured as before
     *   Views 7.x-3.0-rc2. After Views 7.x-3.0 this boolean is only used if
     *   $groupings is an old-style string or if the rendered option is missing
     *   for a grouping instruction.
     *
     * @return array
     *   The grouped record set.
     *   A nested set structure is generated if multiple grouping fields are used.
     *
     *   @code
     *   array(
     *     'grouping_field_1:grouping_1' => array(
     *       'group' => 'grouping_field_1:content_1',
     *       'rows' => array(
     *         'grouping_field_2:grouping_a' => array(
     *           'group' => 'grouping_field_2:content_a',
     *           'rows' => array(
     *             $row_index_1 => $row_1,
     *             $row_index_2 => $row_2,
     *             // ...
     *           )
     *         ),
     *       ),
     *     ),
     *     'grouping_field_1:grouping_2' => array(
     *       // ...
     *     ),
     *   )
     *   @endcode
     */
    public function render_grouping($records, $groupings = array(), $group_rendered = NULL) {
        // This is for backward compatibility, when $groupings was a string
        // containing the ID of a single field.
        if (is_string($groupings)) {
            $rendered = $group_rendered === NULL ? TRUE : $group_rendered;
            $groupings = array(
                array(
                    'field' => $groupings,
                    'rendered' => $rendered,
                ),
            );
        }
        // Make sure fields are rendered.
        $this->render_fields($this->view->result);
        $sets = array();
        if ($groupings) {
            foreach ($records as $index => $row) {
                // Iterate through configured grouping fields to determine the
                // hierarchically positioned set where the current row belongs to.
                // While iterating, parent groups, that do not exist yet, are added.
                $set =& $sets;
                foreach ($groupings as $level => $info) {
                    $field = $info['field'];
                    $rendered = isset($info['rendered']) ? $info['rendered'] : $group_rendered;
                    $rendered_strip = isset($info['rendered_strip']) ? $info['rendered_strip'] : FALSE;
                    $grouping = '';
                    $group_content = '';
                    // Group on the rendered version of the field, not the raw.  That way,
                    // we can control any special formatting of the grouping field through
                    // the admin or theme layer or anywhere else we'd like.
                    if (isset($this->view->field[$field])) {
                        $group_content = $this->get_field($index, $field);
                        if ($this->view->field[$field]->options['label']) {
                            $group_content = $this->view->field[$field]->options['label'] . ': ' . $group_content;
                        }
                        if ($rendered) {
                            $grouping = $group_content;
                            if ($rendered_strip) {
                                $group_content = $grouping = strip_tags(htmlspecialchars_decode((string) $group_content, ENT_COMPAT));
                            }
                        }
                        else {
                            $grouping = $this->get_field_value($index, $field);
                            // Not all field handlers return a scalar value,
                            // e.g. views_handler_field_field.
                            if (!is_scalar($grouping)) {
                                $grouping = md5(serialize($grouping));
                            }
                        }
                    }
                    // Create the group if it does not exist yet.
                    if (empty($set[$grouping])) {
                        $set[$grouping]['group'] = $group_content;
                        $set[$grouping]['level'] = $level;
                        $set[$grouping]['rows'] = array();
                    }
                    // Move the set reference into the row set of the group we just
                    // determined.
                    $set =& $set[$grouping]['rows'];
                }
                // Add the row to the hierarchically positioned row set we just
                // determined.
                $set[$index] = $row;
            }
        }
        else {
            // Create a single group with an empty grouping field.
            $sets[''] = array(
                'group' => '',
                'rows' => $records,
            );
        }
        // If this parameter isn't explicitly set modify the output to be fully
        // backward compatible to code before Views 7.x-3.0-rc2.
        // @todo Remove this as soon as possible e.g. October 2020
        if ($group_rendered === NULL) {
            $old_style_sets = array();
            foreach ($sets as $group) {
                $old_style_sets[$group['group']] = $group['rows'];
            }
            $sets = $old_style_sets;
        }
        return $sets;
    }
    
    /**
     * Render all of the fields for a given style and store them on the object.
     *
     * @param array $result
     *   The result array from $view->result.
     */
    public function render_fields($result) {
        if (!$this->uses_fields()) {
            return;
        }
        if (!isset($this->rendered_fields)) {
            $this->rendered_fields = array();
            $this->view->row_index = 0;
            $keys = array_keys($this->view->field);
            // If all fields have a field::access FALSE there might be no fields, so
            // there is no reason to execute this code.
            if (!empty($keys)) {
                foreach ($result as $count => $row) {
                    $this->view->row_index = $count;
                    foreach ($keys as $id) {
                        $this->rendered_fields[$count][$id] = $this->view->field[$id]
                            ->theme($row);
                    }
                    $this->row_tokens[$count] = $this->view->field[$id]
                        ->get_render_tokens(array());
                }
            }
            unset($this->view->row_index);
        }
        return $this->rendered_fields;
    }
    
    /**
     * Get a rendered field.
     *
     * @param int $index
     *   The index count of the row.
     * @param string $field
     *   The id of the field.
     */
    public function get_field($index, $field) {
        if (!isset($this->rendered_fields)) {
            $this->render_fields($this->view->result);
        }
        if (isset($this->rendered_fields[$index][$field])) {
            return $this->rendered_fields[$index][$field];
        }
    }
    
    /**
     * Get the raw field value.
     *
     * @param int $index
     *   The index count of the row.
     * @param string $field
     *   The id of the field.
     */
    public function get_field_value($index, $field) {
        $this->view->row_index = $index;
        $value = $this->view->field[$field]
            ->get_value($this->view->result[$index]);
        unset($this->view->row_index);
        return $value;
    }
    
    /**
     * {@inheritdoc}
     */
    public function validate() {
        $errors = parent::validate();
        if ($this->uses_row_plugin()) {
            $plugin = $this->display->handler
                ->get_plugin('row');
            if (empty($plugin)) {
                $errors[] = t('Style @style requires a row style but the row plugin is invalid.', array(
                    '@style' => $this->definition['title'],
                ));
            }
            else {
                $result = $plugin->validate();
                if (!empty($result) && is_array($result)) {
                    $errors = array_merge($errors, $result);
                }
            }
        }
        return $errors;
    }
    
    /**
     * {@inheritdoc}
     */
    public function query() {
        parent::query();
        if ($this->row_plugin) {
            $this->row_plugin
                ->query();
        }
    }

}

/**
 * @}
 */

Classes

Title Deprecated Summary
views_plugin_style Base class to define a style plugin handler.