menuStorage = $menu_storage; } /** * {@inheritdoc} */ public static function create(ContainerInterface $container, array $configuration, $plugin_id, $plugin_definition) { return new static( $configuration, $plugin_id, $plugin_definition, $container->get('router.route_provider'), $container->get('state'), $container->get('entity.manager')->getStorage('menu') ); } /** * {@inheritdoc} */ protected function getRoute($view_id, $display_id) { $route = parent::getRoute($view_id, $display_id); // Explicitly set HTML as the format for Page displays. $route->setRequirement('_format', 'html'); return $route; } /** * Sets the current page views render array. * * @param array $element * (optional) A render array. If not specified the previous element is * returned. * * @return array * The page render array. */ public static function &setPageRenderArray(array &$element = NULL) { if (isset($element)) { static::$pageRenderArray = &$element; } return static::$pageRenderArray; } /** * Gets the current views page render array. * * @return array * The page render array. */ public static function &getPageRenderArray() { return static::$pageRenderArray; } /** * {@inheritdoc} */ protected function defineOptions() { $options = parent::defineOptions(); $options['menu'] = [ 'contains' => [ 'type' => ['default' => 'none'], 'title' => ['default' => ''], 'description' => ['default' => ''], 'weight' => ['default' => 0], 'enabled' => ['default' => TRUE], 'menu_name' => ['default' => 'main'], 'parent' => ['default' => ''], 'context' => ['default' => ''], 'expanded' => ['default' => FALSE], ], ]; $options['tab_options'] = [ 'contains' => [ 'type' => ['default' => 'none'], 'title' => ['default' => ''], 'description' => ['default' => ''], 'weight' => ['default' => 0], ], ]; return $options; } /** * {@inheritdoc} */ public static function buildBasicRenderable($view_id, $display_id, array $args = [], Route $route = NULL) { $build = parent::buildBasicRenderable($view_id, $display_id, $args); if ($route) { $build['#view_id'] = $route->getDefault('view_id'); $build['#view_display_plugin_id'] = $route->getOption('_view_display_plugin_id'); $build['#view_display_show_admin_links'] = $route->getOption('_view_display_show_admin_links'); } else { throw new \BadFunctionCallException('Missing route parameters.'); } return $build; } /** * {@inheritdoc} */ public function execute() { parent::execute(); // And now render the view. $render = $this->view->render(); // First execute the view so it's possible to get tokens for the title. // And the title, which is much easier. // @todo Figure out how to support custom response objects. Maybe for pages // it should be dropped. if (is_array($render)) { $render += [ '#title' => ['#markup' => $this->view->getTitle(), '#allowed_tags' => Xss::getHtmlTagList()], ]; } return $render; } /** * {@inheritdoc} */ public function optionsSummary(&$categories, &$options) { parent::optionsSummary($categories, $options); $menu = $this->getOption('menu'); if (!is_array($menu)) { $menu = ['type' => 'none']; } switch ($menu['type']) { case 'none': default: $menu_str = $this->t('No menu'); break; case 'normal': $menu_str = $this->t('Normal: @title', ['@title' => $menu['title']]); break; case 'tab': case 'default tab': $menu_str = $this->t('Tab: @title', ['@title' => $menu['title']]); break; } $options['menu'] = [ 'category' => 'page', 'title' => $this->t('Menu'), 'value' => views_ui_truncate($menu_str, 24), ]; // This adds a 'Settings' link to the style_options setting if the style // has options. if ($menu['type'] == 'default tab') { $options['menu']['setting'] = $this->t('Parent menu item'); $options['menu']['links']['tab_options'] = $this->t('Change settings for the parent menu'); } } /** * {@inheritdoc} */ public function buildOptionsForm(&$form, FormStateInterface $form_state) { parent::buildOptionsForm($form, $form_state); switch ($form_state->get('section')) { case 'menu': $form['#title'] .= $this->t('Menu item entry'); $form['menu'] = [ '#prefix' => '
', '#suffix' => '
', '#tree' => TRUE, ]; $menu = $this->getOption('menu'); if (empty($menu)) { $menu = ['type' => 'none', 'title' => '', 'weight' => 0, 'expanded' => FALSE]; } $form['menu']['type'] = [ '#prefix' => '
', '#suffix' => '
', '#title' => $this->t('Type'), '#type' => 'radios', '#options' => [ 'none' => $this->t('No menu entry'), 'normal' => $this->t('Normal menu entry'), 'tab' => $this->t('Menu tab'), 'default tab' => $this->t('Default menu tab') ], '#default_value' => $menu['type'], ]; $form['menu']['title'] = [ '#prefix' => '
', '#title' => $this->t('Menu link title'), '#type' => 'textfield', '#default_value' => $menu['title'], '#states' => [ 'visible' => [ [ ':input[name="menu[type]"]' => ['value' => 'normal'], ], [ ':input[name="menu[type]"]' => ['value' => 'tab'], ], [ ':input[name="menu[type]"]' => ['value' => 'default tab'], ], ], ], ]; $form['menu']['description'] = [ '#title' => $this->t('Description'), '#type' => 'textfield', '#default_value' => $menu['description'], '#description' => $this->t("Shown when hovering over the menu link."), '#states' => [ 'visible' => [ [ ':input[name="menu[type]"]' => ['value' => 'normal'], ], [ ':input[name="menu[type]"]' => ['value' => 'tab'], ], [ ':input[name="menu[type]"]' => ['value' => 'default tab'], ], ], ], ]; $form['menu']['expanded'] = [ '#title' => $this->t('Show as expanded'), '#type' => 'checkbox', '#default_value' => !empty($menu['expanded']), '#description' => $this->t('If selected and this menu link has children, the menu will always appear expanded.'), ]; // Only display the parent selector if Menu UI module is enabled. $menu_parent = $menu['menu_name'] . ':' . $menu['parent']; if (\Drupal::moduleHandler()->moduleExists('menu_ui')) { $menu_link = 'views_view:views.' . $form_state->get('view')->id() . '.' . $form_state->get('display_id'); $form['menu']['parent'] = \Drupal::service('menu.parent_form_selector')->parentSelectElement($menu_parent, $menu_link); $form['menu']['parent'] += [ '#title' => $this->t('Parent'), '#description' => $this->t('The maximum depth for a link and all its children is fixed. Some menu links may not be available as parents if selecting them would exceed this limit.'), '#attributes' => ['class' => ['menu-title-select']], '#states' => [ 'visible' => [ [ ':input[name="menu[type]"]' => ['value' => 'normal'], ], [ ':input[name="menu[type]"]' => ['value' => 'tab'], ], ], ], ]; } else { $form['menu']['parent'] = [ '#type' => 'value', '#value' => $menu_parent, ]; $form['menu']['markup'] = [ '#markup' => $this->t('Menu selection requires the activation of Menu UI module.'), ]; } $form['menu']['weight'] = [ '#title' => $this->t('Weight'), '#type' => 'textfield', '#default_value' => isset($menu['weight']) ? $menu['weight'] : 0, '#description' => $this->t('In the menu, the heavier links will sink and the lighter links will be positioned nearer the top.'), '#states' => [ 'visible' => [ [ ':input[name="menu[type]"]' => ['value' => 'normal'], ], [ ':input[name="menu[type]"]' => ['value' => 'tab'], ], [ ':input[name="menu[type]"]' => ['value' => 'default tab'], ], ], ], ]; $form['menu']['context'] = [ '#title' => $this->t('Context'), '#suffix' => '
', '#type' => 'checkbox', '#default_value' => !empty($menu['context']), '#description' => $this->t('Displays the link in contextual links'), '#states' => [ 'visible' => [ ':input[name="menu[type]"]' => ['value' => 'tab'], ], ], ]; break; case 'tab_options': $form['#title'] .= $this->t('Default tab options'); $tab_options = $this->getOption('tab_options'); if (empty($tab_options)) { $tab_options = ['type' => 'none', 'title' => '', 'weight' => 0]; } $form['tab_markup'] = [ '#markup' => '
' . $this->t('When providing a menu item as a tab, Drupal needs to know what the parent menu item of that tab will be. Sometimes the parent will already exist, but other times you will need to have one created. The path of a parent item will always be the same path with the last part left off. i.e, if the path to this view is foo/bar/baz, the parent path would be foo/bar.') . '
', ]; $form['tab_options'] = [ '#prefix' => '
', '#suffix' => '
', '#tree' => TRUE, ]; $form['tab_options']['type'] = [ '#prefix' => '
', '#suffix' => '
', '#title' => $this->t('Parent menu item'), '#type' => 'radios', '#options' => ['none' => $this->t('Already exists'), 'normal' => $this->t('Normal menu item'), 'tab' => $this->t('Menu tab')], '#default_value' => $tab_options['type'], ]; $form['tab_options']['title'] = [ '#prefix' => '
', '#title' => $this->t('Title'), '#type' => 'textfield', '#default_value' => $tab_options['title'], '#description' => $this->t('If creating a parent menu item, enter the title of the item.'), '#states' => [ 'visible' => [ [ ':input[name="tab_options[type]"]' => ['value' => 'normal'], ], [ ':input[name="tab_options[type]"]' => ['value' => 'tab'], ], ], ], ]; $form['tab_options']['description'] = [ '#title' => $this->t('Description'), '#type' => 'textfield', '#default_value' => $tab_options['description'], '#description' => $this->t('If creating a parent menu item, enter the description of the item.'), '#states' => [ 'visible' => [ [ ':input[name="tab_options[type]"]' => ['value' => 'normal'], ], [ ':input[name="tab_options[type]"]' => ['value' => 'tab'], ], ], ], ]; $form['tab_options']['weight'] = [ '#suffix' => '
', '#title' => $this->t('Tab weight'), '#type' => 'textfield', '#default_value' => $tab_options['weight'], '#size' => 5, '#description' => $this->t('If the parent menu item is a tab, enter the weight of the tab. Heavier tabs will sink and the lighter tabs will be positioned nearer to the first menu item.'), '#states' => [ 'visible' => [ ':input[name="tab_options[type]"]' => ['value' => 'tab'], ], ], ]; break; } } /** * {@inheritdoc} */ public function validateOptionsForm(&$form, FormStateInterface $form_state) { parent::validateOptionsForm($form, $form_state); if ($form_state->get('section') == 'menu') { $path = $this->getOption('path'); $menu_type = $form_state->getValue(['menu', 'type']); if ($menu_type == 'normal' && strpos($path, '%') !== FALSE) { $form_state->setError($form['menu']['type'], $this->t('Views cannot create normal menu items for paths with a % in them.')); } if ($menu_type == 'default tab' || $menu_type == 'tab') { $bits = explode('/', $path); $last = array_pop($bits); if ($last == '%') { $form_state->setError($form['menu']['type'], $this->t('A display whose path ends with a % cannot be a tab.')); } } if ($menu_type != 'none' && $form_state->isValueEmpty(['menu', 'title'])) { $form_state->setError($form['menu']['title'], $this->t('Title is required for this menu type.')); } } } /** * {@inheritdoc} */ public function submitOptionsForm(&$form, FormStateInterface $form_state) { parent::submitOptionsForm($form, $form_state); switch ($form_state->get('section')) { case 'menu': $menu = $form_state->getValue('menu'); list($menu['menu_name'], $menu['parent']) = explode(':', $menu['parent'], 2); $this->setOption('menu', $menu); // send ajax form to options page if we use it. if ($form_state->getValue(['menu', 'type']) == 'default tab') { $form_state->get('view')->addFormToStack('display', $this->display['id'], 'tab_options'); } break; case 'tab_options': $this->setOption('tab_options', $form_state->getValue('tab_options')); break; } } /** * {@inheritdoc} */ public function validate() { $errors = parent::validate(); $menu = $this->getOption('menu'); if (!empty($menu['type']) && $menu['type'] != 'none' && empty($menu['title'])) { $errors[] = $this->t('Display @display is set to use a menu but the menu link text is not set.', ['@display' => $this->display['display_title']]); } if ($menu['type'] == 'default tab') { $tab_options = $this->getOption('tab_options'); if (!empty($tab_options['type']) && $tab_options['type'] != 'none' && empty($tab_options['title'])) { $errors[] = $this->t('Display @display is set to use a parent menu but the parent menu link text is not set.', ['@display' => $this->display['display_title']]); } } return $errors; } /** * {@inheritdoc} */ public function getArgumentText() { return [ 'filter value not present' => $this->t('When the filter value is NOT in the URL'), 'filter value present' => $this->t('When the filter value IS in the URL or a default is provided'), 'description' => $this->t('The contextual filter values are provided by the URL.'), ]; } /** * {@inheritdoc} */ public function getPagerText() { return [ 'items per page title' => $this->t('Items per page'), 'items per page description' => $this->t('Enter 0 for no limit.') ]; } /** * {@inheritdoc} */ public function calculateDependencies() { $dependencies = parent::calculateDependencies(); $menu = $this->getOption('menu'); if ($menu['type'] === 'normal' && ($menu_entity = $this->menuStorage->load($menu['menu_name']))) { $dependencies[$menu_entity->getConfigDependencyKey()][] = $menu_entity->getConfigDependencyName(); } return $dependencies; } }