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 (isset($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 = $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); } return implode(' ', $classes); } } /** * Take a value and apply token replacement logic to it. */ public function tokenize_value($value, $row_index) { if (strpos($value, '[') !== FALSE || strpos($value, '!') !== FALSE || strpos($value, '%') !== FALSE) { $fake_item = array( 'alter_text' => TRUE, 'text' => $value, ); // 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']); // 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']) ); } // Render as a record set. 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 $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($group_content)); } } 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]['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 (isset($this->row_plugin)) { $this->row_plugin->query(); } } } /** * @} */