Получить ссылку меню


11

Я пытаюсь создать меню в Drupal 8, которое просто ссылки на текущую страницу. Так что, если меню:

  • Дом
  • Родитель 1
    • Подросток 1
      • Ребенок 1
    • Подросток 2
      • Ребенок 2
      • Ребенок 3
      • Ребенок 4
  • Родитель 2

Когда я нахожусь на странице «Ребенок 3», я хочу, чтобы блок меню со ссылкой выглядел так:

  • Ребенок 2
  • Ребенок 3
  • Ребенок 4

Думаю, я знаю, как это сделать в D7, но мне трудно переводить эти знания в D8. Это даже что-то выполнимое в D8? И если это так, может кто-нибудь указать мне правильное направление, как это сделать?

Спасибо!

Пояснение: дочерний уровень должен быть переменным, чтобы элементы меню с различной глубиной могли отображать своих братьев и сестер. Так, например, в дополнение к желанию меню для детей, мне нужно меню для вспомогательных родителей и меню для родителей и так далее. Я также не контролирую / не знаю, насколько глубоко заходит меню и насколько глубоко оно распространяется на все разделы.

Ответы:


19

Итак, я в конечном итоге выяснил некоторый код, который позволил бы мне сделать это, создав собственный блок и, в методе сборки, вывел меню с добавленными в него преобразователями. Это ссылка, которую я использовал, чтобы выяснить, как получить меню в блоке и добавить к нему преобразователи: http://alexrayu.com/blog/drupal-8-display-submenu-block . Моя в build()итоге выглядела так:

$menu_tree = \Drupal::menuTree();
$menu_name = 'main';

// Build the typical default set of menu tree parameters.
$parameters = $menu_tree->getCurrentRouteMenuTreeParameters($menu_name);

// Load the tree based on this set of parameters.
$tree = $menu_tree->load($menu_name, $parameters);

// Transform the tree using the manipulators you want.
$manipulators = array(
  // Only show links that are accessible for the current user.
  array('callable' => 'menu.default_tree_manipulators:checkAccess'),
  // Use the default sorting of menu links.
  array('callable' => 'menu.default_tree_manipulators:generateIndexAndSort'),
  // Remove all links outside of siblings and active trail
  array('callable' => 'intranet.menu_transformers:removeInactiveTrail'),
);
$tree = $menu_tree->transform($tree, $manipulators);

// Finally, build a renderable array from the transformed tree.
$menu = $menu_tree->build($tree);

return array(
  '#markup' => \Drupal::service('renderer')->render($menu),
  '#cache' => array(
    'contexts' => array('url.path'),
  ),
);

Преобразователь является службой, поэтому я добавил intranet.services.ymlв свой модуль интрасети указатель на класс, который я определил. У класса было три метода:, removeInactiveTrail()который вызывал getCurrentParent()для получения родительского элемента страницы, на которой в данный момент находился пользователь, и stripChildren()который урезал меню до только дочерних элементов текущего элемента меню и его родных элементов (то есть: удалил все подменю, которые не были ' т в активном следе).

Вот как это выглядело:

/**
 * Removes all link trails that are not siblings to the active trail.
 *
 * For a menu such as:
 * Parent 1
 *  - Child 1
 *  -- Child 2
 *  -- Child 3
 *  -- Child 4
 *  - Child 5
 * Parent 2
 *  - Child 6
 * with current page being Child 3, Parent 2, Child 6, and Child 5 would be
 * removed.
 *
 * @param \Drupal\Core\Menu\MenuLinkTreeElement[] $tree
 *   The menu link tree to manipulate.
 *
 * @return \Drupal\Core\Menu\MenuLinkTreeElement[]
 *   The manipulated menu link tree.
 */
public function removeInactiveTrail(array $tree) {
  // Get the current item's parent ID
  $current_item_parent = IntranetMenuTransformers::getCurrentParent($tree);

  // Tree becomes the current item parent's children if the current item
  // parent is not empty. Otherwise, it's already the "parent's" children
  // since they are all top level links.
  if (!empty($current_item_parent)) {
    $tree = $current_item_parent->subtree;
  }

  // Strip children from everything but the current item, and strip children
  // from the current item's children.
  $tree = IntranetMenuTransformers::stripChildren($tree);

  // Return the tree.
  return $tree;
}

/**
 * Get the parent of the current active menu link, or return NULL if the
 * current active menu link is a top-level link.
 *
 * @param \Drupal\Core\Menu\MenuLinkTreeElement[] $tree
 *   The tree to pull the parent link out of.
 * @param \Drupal\Core\Menu\MenuLinkTreeElement|null $prev_parent
 *   The previous parent's parent, or NULL if no previous parent exists.
 * @param \Drupal\Core\Menu\MenuLinkTreeElement|null $parent
 *   The parent of the current active link, or NULL if not parent exists.
 *
 * @return \Drupal\Core\Menu\MenuLinkTreeElement|null
 *   The parent of the current active menu link, or NULL if no parent exists.
 */
private function getCurrentParent($tree, $prev_parent = NULL, $parent = NULL) {
  // Get active item
  foreach ($tree as $leaf) {
    if ($leaf->inActiveTrail) {
      $active_item = $leaf;
      break;
    }
  }

  // If the active item is set and has children
  if (!empty($active_item) && !empty($active_item->subtree)) {
    // run getCurrentParent with the parent ID as the $active_item ID.
    return IntranetMenuTransformers::getCurrentParent($active_item->subtree, $parent, $active_item);
  }

  // If the active item is not set, we know there was no active item on this
  // level therefore the active item parent is the previous level's parent
  if (empty($active_item)) {
    return $prev_parent;
  }

  // Otherwise, the current active item has no children to check, so it is
  // the bottommost and its parent is the correct parent.
  return $parent;
}


/**
 * Remove the children from all MenuLinkTreeElements that aren't active. If
 * it is active, remove its children's children.
 *
 * @param \Drupal\Core\Menu\MenuLinkTreeElement[] $tree
 *   The menu links to strip children from non-active leafs.
 *
 * @return \Drupal\Core\Menu\MenuLinkTreeElement[]
 *   A menu tree with no children of non-active leafs.
 */
private function stripChildren($tree) {
  // For each item in the tree, if the item isn't active, strip its children
  // and return the tree.
  foreach ($tree as &$leaf) {
    // Check if active and if has children
    if ($leaf->inActiveTrail && !empty($leaf->subtree)) {
      // Then recurse on the children.
      $leaf->subtree = IntranetMenuTransformers::stripChildren($leaf->subtree);
    }
    // Otherwise, if not the active menu
    elseif (!$leaf->inActiveTrail) {
      // Otherwise, it's not active, so we don't want to display any children
      // so strip them.
      $leaf->subtree = array();
    }
  }

  return $tree;
}

Это лучший способ сделать это? Возможно нет. Но это, по крайней мере, обеспечивает отправную точку для людей, которые должны сделать что-то подобное.


Это в значительной степени то, что делает footermap. +1 за использование сервиса menu.tree.
Мрадклифф

2
Не могли бы вы сказать, какой код должен быть размещен в файле service.yml? Как указать класс из файла service.yml?
siddiq

Как исключить ссылку родительского меню?
Пермана

3

Drupal 8 имеет функциональность «Блок меню», встроенную в ядро, единственное, что вам нужно сделать, это создать новый блок меню в пользовательском интерфейсе блока и настроить его.

Это происходит путем:

  • Поместите новый блок и затем выберите меню, для которого вы хотите создать блок.
  • В конфигурации блока вы должны выбрать «Начальный уровень меню», чтобы быть 3.
  • Вы также можете установить «Максимальное количество отображаемых уровней меню» на 1, если вы хотите печатать только пункты меню с третьего уровня.

К сожалению, я не уверен, на каком уровне будет страница, поэтому я не могу просто создать для нее блок меню. Также существует вероятность того, что это могут быть переменные уровни, в зависимости от того, чем в итоге будет структура сайта.
Эрин Маклафлин

menu_block для Drupal 8 в настоящее время не включает в себя функциональность для отслеживания текущего узла, патчи в обзоре здесь; drupal.org/node/2756675
Кристиан

ОК для статического использования. Но не для динамического использования, как в «Поместите блок на каждую страницу и покажите братьев и сестер на текущей странице, независимо от того, на каком уровне вы находитесь в данный момент».
leymannx

3

Установка рута для текущей ссылки может помочь:

$menu_tree = \Drupal::menuTree();
$menu_name = 'main';

$parameters = $menu_tree->getCurrentRouteMenuTreeParameters($menu_name);
$currentLinkId = reset($parameters->activeTrail);
$parameters->setRoot($currentLinkId);
$tree = $menu_tree->load($menu_name, $parameters);

// Transform the tree using the manipulators you want.
$manipulators = array(
  // Only show links that are accessible for the current user.
  array('callable' => 'menu.default_tree_manipulators:checkAccess'),
  // Use the default sorting of menu links.
  array('callable' => 'menu.default_tree_manipulators:generateIndexAndSort'),
);
$tree = $menu_tree->transform($tree, $manipulators);

Нет, к сожалению, это показывает только детей. Но не братья и сестры. ОП хочет братьев и сестер.
leymannx

3

Блок меню братьев и сестер

С помощью ответа @Icubes MenuLinkTreeInterface::getCurrentRouteMenuTreeParametersмы можем просто получить активную трассу текущего маршрута. Имея это, у нас также есть родительский пункт меню. Установка этого в качестве отправной точки MenuTreeParameters::setRootдля создания нового дерева дает вам желаемое меню братьев и сестер.

// Enable url-wise caching.
$build = [
  '#cache' => [
    'contexts' => ['url'],
  ],
];

$menu_name = 'main';
$menu_tree = \Drupal::menuTree();

// This one will give us the active trail in *reverse order*.
// Our current active link always will be the first array element.
$parameters   = $menu_tree->getCurrentRouteMenuTreeParameters($menu_name);
$active_trail = array_keys($parameters->activeTrail);

// But actually we need its parent.
// Except for <front>. Which has no parent.
$parent_link_id = isset($active_trail[1]) ? $active_trail[1] : $active_trail[0];

// Having the parent now we set it as starting point to build our custom
// tree.
$parameters->setRoot($parent_link_id);
$parameters->setMaxDepth(1);
$parameters->excludeRoot();
$tree = $menu_tree->load($menu_name, $parameters);

// Optional: Native sort and access checks.
$manipulators = [
  ['callable' => 'menu.default_tree_manipulators:checkNodeAccess'],
  ['callable' => 'menu.default_tree_manipulators:checkAccess'],
  ['callable' => 'menu.default_tree_manipulators:generateIndexAndSort'],
];
$tree = $menu_tree->transform($tree, $manipulators);

// Finally, build a renderable array.
$menu = $menu_tree->build($tree);

$build['#markup'] = \Drupal::service('renderer')->render($menu);

return $build;

Это решение работает как шарм. :)
Панкадж Сачдева
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.