Вложение системы вкладок - PullRequest
       2

Вложение системы вкладок

2 голосов
/ 01 октября 2019

Когда мы щелкаем на другой вкладке в главном меню и возвращаемся к первой, второе меню (вторые вкладки) теряет свое активное состояние (никакая вкладка не открыта)

Кроме того, состояние is-activeна ссылки удаляются.

Я не знаю, как не повлиять на дочерние вкладки.

Вот ссылка , чтобы увидеть проблему.

<div class="container js-tabs-container">
    <div class="tabs">
        <ul>
            <li class="is-active" data-tab="tab-1"><a>Link 1</a></li>
            <li data-tab="tab-2"><a>Link 2</a></li>
        </ul>
    </div>

    <div class="js-tab-content" id="tab-1">
        <p>Tab 1</p>

        <!-- Nesting tabs -->
        <div class="container js-tabs-container">
            <div class="tabs">
                <ul>
                    <li class="is-active" data-tab="tab-2-1"><a>Link 2-1</a></li>
                    <li data-tab="tab-2-2"><a>Link 2-2</a></li>
                </ul>
            </div>

            <div class="js-tab-content" id="tab-2-1">
              <p>Tab 2-1</p>
            </div>
            <div class="js-tab-content" id="tab-2-2">
              <p>Tab 2-2</p>
            </div>
        </div>
    </div>

    <div class="js-tab-content" id="tab-2">
        <p>Tab 2</p>
    </div>


</div>
document.addEventListener('DOMContentLoaded', function () {

  var tabs = document.getElementsByClassName('tabs');

  if (tabs) {
    var _loop = function _loop() {
      var tabListItems = tabs[i].querySelectorAll('.tabs-menu-item');
      tabListItems.forEach(function(tabListItem) {

        // création d'un écouteur d'évènements sur le clic d'une tab
        tabListItem.addEventListener('click', function () {

          // suppression de la classe is-active sur chacune des tabs avant de la rajouter sur la tab qui a été cliquée
          tabListItems.forEach(function(tabListItem) {
            tabListItem.classList.remove('is-active');
          });
          tabListItem.classList.add('is-active');

          // tabName correspond à la valeur de l'attribut data-tab
          var tabName = tabListItem.dataset.tab;

          // on identifie tous les contenus possibles puis on applique la classe has-display-none si l'ID du contenu ne correspond pas à la valeur de l'attribut data-tab
          tabListItem.closest('.js-tabs').querySelectorAll('.js-tab-content').forEach(function(tabContent) {
            if (tabContent.id !== tabName) {
              tabContent.classList.add('has-display-none');
            } else {
              tabContent.classList.remove('has-display-none');
            }
          });
        }, false);
      });
    };

    for (var i = 0; i < tabs.length; i++) {
      _loop();
    }
  }
});

Спасибо

1 Ответ

1 голос
/ 01 октября 2019

Проблема, с которой вы столкнулись, вызвана использованием querySelectorAll, которое не просто выбирает первый слой дочерних узлов, но детализирует его непосредственно в DOM и извлекает ВСЕ соответствующие узлы. Простое решение может заключаться в том, чтобы отличать дочерние вкладки от родительских вкладок, добавив атрибут для их дифференциации и используя отдельный обработчик для дочерних элементов. Более комплексный подход может включать в себя работу только с вкладками верхнего уровня и содержимым вкладок.

ссылка CodePen

(я добавил английские комментарии к коду и удалилчужие для удобства чтения другими пользователями)

'use strict';

document.addEventListener('DOMContentLoaded', function () {

  var tabs = document.getElementsByClassName('tabs');
  if (tabs) {
    var _loop = function _loop() {
      var tabListItems = tabs[i].querySelectorAll('li');

      tabListItems.forEach(function (tabListItem) {

        tabListItem.addEventListener('click', function (e) {

          // Select any siblings of the current tab.
          let siblings = Array.from(tabListItem.parentElement.children);

          // Remove the is-active status from all siblings
          siblings.forEach(function (element) {
            element.classList.remove('is-active');
          });

          // Add the is-active status to the selected tab.
          tabListItem.classList.add('is-active');

          var tabName = tabListItem.dataset.tab;

          // Same as above, rather than selecting all of the js-tab-content
          // elements, we only select those which are at the same level as
          // the selected tab.
          let tabsContainer = tabListItem.closest('.js-tabs-container');
          let tabsContainerChildren = Array.from(tabsContainer.children);

          // Filter out other children that aren't js-tab-content elements.
          let tabs = tabsContainerChildren.filter(function(el) {
            return el.className.includes('js-tab-content')
          });

          tabs.forEach(function(tabContent) {
            if (tabContent.id !== tabName) {
              tabContent.classList.add('has-display-none');
            } else {
              tabContent.classList.remove('has-display-none');
            }
          })

        }, false);
      });
    };

    for (var i = 0; i < tabs.length; i++) {
      _loop();
    }
  }
});

Если ваша структура в целом остается неизменной, это должно работать правильно для любого уровня вложенных вкладок, так как при выборе вкладки влияют только элементы - экраныи ссылки вокруг него.


Рефакторинг

Использование функций со стрелками и назначение меньшего числа переменных там, где они не нужны, позволят вам писать более короткий и лаконичный код. Вот пример, не обязательно оптимальный способ его написания, но он может дать вам некоторые идеи для извлечения функций и создания цепочек итераций массива.

document.addEventListener('DOMContentLoaded', () => {
  Array.from(document.getElementsByClassName('tabs')).forEach(tab => {
    tab.querySelectorAll('li').forEach(listItem => {
      listItem.addEventListener('click', tabClickHandler(listItem), false);
    });
  });
});

function tabClickHandler (listItem) {
  return () => {
    let siblings = Array.from(listItem.parentElement.children);

    siblings.forEach(el => el.classList.remove('is-active'));
    listItem.classList.add('is-active');

    let tabName = listItem.dataset.tab;

    Array.from(listItem.closest('.js-tabs-container').children)
      .filter(el => el.classList.contains('js-tab-content'))
      .forEach(tab => {
        if (tab.id !== tabName) {
          tab.classList.add('has-display-none');
        } else {
          tab.classList.remove('has-display-none');
        }
      });
  }
}

Попробуйте здесь

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...