(function ()()

in scripts/navigation-loader.js [12:182]


(function () {
  function displayToc() {
    fetch(pathToRoot + 'navigation.html')
      .then((response) => response.text())
      .then((tocHTML) => {
        renderToc(tocHTML);
        updateTocLinks();
        collapseTocParts();
        expandTocPathToCurrentPage();
        restoreTocExpandedState();
        restoreTocScrollTop();
      });
  }

  function renderToc(tocHTML) {
    const containerElement = document.getElementById(TOC_CONTAINER_ID);
    if (containerElement) {
      containerElement.innerHTML = tocHTML;
    }
  }

  function updateTocLinks() {
    document.querySelectorAll(`.${TOC_LINK_CLASS}`).forEach((tocLink) => {
      tocLink.setAttribute('href', `${pathToRoot}${tocLink.getAttribute('href')}`);
      tocLink.addEventListener('keydown', preventScrollBySpaceKey);
    });
    document.querySelectorAll(`.${TOC_SKIP_LINK_CLASS}`).forEach((skipLink) => {
      skipLink.setAttribute('href', `#main`);
      skipLink.addEventListener('keydown', preventScrollBySpaceKey);
    })
  }

  function collapseTocParts() {
    document.querySelectorAll(`.${TOC_PART_CLASS}`).forEach((tocPart) => {
      if (!tocPart.classList.contains(TOC_PART_HIDDEN_CLASS)) {
        tocPart.classList.add(TOC_PART_HIDDEN_CLASS);
        const tocToggleButton = tocPart.querySelector('button');
        if (tocToggleButton) {
          tocToggleButton.setAttribute("aria-expanded", "false");
        }
      }
    });
  }

  const expandTocPathToCurrentPage = () => {
    const tocParts = [...document.querySelectorAll(`.${TOC_PART_CLASS}`)];
    const currentPageId = document.getElementById('content')?.getAttribute('pageIds');
    if (!currentPageId) {
      return;
    }

    let isPartFound = false;
    let currentPageIdPrefix = currentPageId;
    while (!isPartFound && currentPageIdPrefix !== '') {
      tocParts.forEach((part) => {
        const partId = part.getAttribute('pageId');
        if (!isPartFound && partId?.includes(currentPageIdPrefix)) {
          isPartFound = true;
          expandTocPart(part);
          expandTocPathToParent(part);
          part.dataset.active = 'true';
        }
      });
      currentPageIdPrefix = currentPageIdPrefix.substring(0, currentPageIdPrefix.lastIndexOf('/'));
    }
  };

  const expandTocPathToParent = (part) => {
    if (part.classList.contains(TOC_PART_CLASS)) {
      expandTocPart(part);
      expandTocPathToParent(part.parentNode);
    }
  };

  const expandTocPart = (tocPart) => {
    if (tocPart.classList.contains(TOC_PART_HIDDEN_CLASS)) {
      tocPart.classList.remove(TOC_PART_HIDDEN_CLASS);
      const tocToggleButton = tocPart.querySelector('button');
      if (tocToggleButton) {
        tocToggleButton.setAttribute("aria-expanded", "true");
      }
      const tocPartId = tocPart.getAttribute('id');
      safeSessionStorage.setItem(`${TOC_STATE_KEY_PREFIX}${tocPartId}`, 'true');
    }
  };

  /**
   * Restores the state of the navigation tree from the local storage.
   * LocalStorage keys are in the format of `TOC_STATE::${id}` where `id` is the id of the part
   */
  const restoreTocExpandedState = () => {
    const allLocalStorageKeys = safeSessionStorage.getKeys();
    const tocStateKeys = allLocalStorageKeys.filter((key) => key.startsWith(TOC_STATE_KEY_PREFIX));
    tocStateKeys.forEach((key) => {
      const isExpandedTOCPart = safeSessionStorage.getItem(key) === 'true';
      const tocPartId = key.substring(TOC_STATE_KEY_PREFIX.length);
      const tocPart = document.querySelector(`.toc--part[id="${tocPartId}"]`);
      if (tocPart !== null && isExpandedTOCPart) {
        tocPart.classList.remove(TOC_PART_HIDDEN_CLASS);
        const tocToggleButton = tocPart.querySelector('button');
        if (tocToggleButton) {
          tocToggleButton.setAttribute("aria-expanded", "true");
        }
      }
    });
  };

  function saveTocScrollTop() {
    const container = document.getElementById(TOC_SCROLL_CONTAINER_ID);
    if (container) {
      const currentScrollTop = container.scrollTop;
      safeSessionStorage.setItem(`${TOC_STATE_KEY_PREFIX}SCROLL_TOP`, `${currentScrollTop}`);
    }
  }

  function restoreTocScrollTop() {
    const container = document.getElementById(TOC_SCROLL_CONTAINER_ID);
    if (container) {
      const storedScrollTop = safeSessionStorage.getItem(`${TOC_STATE_KEY_PREFIX}SCROLL_TOP`);
      if (storedScrollTop) {
        container.scrollTop = Number(storedScrollTop);
      }
    }
  }

  function initTocScrollListener() {
    const container = document.getElementById(TOC_SCROLL_CONTAINER_ID);
    if (container) {
      container.addEventListener('scroll', saveTocScrollTop);
    }
  }

  function preventScrollBySpaceKey(event) {
    if (event.key === ' ') {
      event.preventDefault();
      event.stopPropagation();
    }
  }

  function resetTocState() {
    const tocKeys = safeSessionStorage.getKeys();
    tocKeys.forEach((key) => {
      if (key.startsWith(TOC_STATE_KEY_PREFIX)) {
        safeSessionStorage.removeItem(key);
      }
    });
  }

  function initLogoClickListener() {
    const logo = document.querySelector('.library-name--link');
    if (logo) {
      logo.addEventListener('click', resetTocState);
    }
  }

  /*
    This is a work-around for safari being IE of our times.
    It doesn't fire a DOMContentLoaded, presumably because eventListener is added after it wants to do it
*/
  if (document.readyState === 'loading') {
    window.addEventListener('DOMContentLoaded', () => {
      displayToc();
      initTocScrollListener();
      initLogoClickListener();
    })
  } else {
    displayToc();
    initTocScrollListener();
    initLogoClickListener();
  }
})();