<nav class="subnav" aria-label="Sidebar">
    <noscript>
    <button class="subnav__toggle" id="subnav-toggle" aria-controls="subnav-list" aria-haspopup="true" aria-expanded="false"><span>Explore This Section</span></button>
  </noscript>
    <div class="subnav__list" id="subnav-list">
        <ul>
            <li>
                <a href="https://www.iastate.edu/">Parent Page</a>
            </li>
            <li>
                <a href="https://www.iastate.edu/">Sibling Page Title Lorem Ipsum</a>
            </li>
            <li>
                <a href="https://www.iastate.edu/">Sibling Page Title Lorem Ipsum</a>
            </li>
            <li>
                <a href="https://www.iastate.edu/">Sibling Page Title Lorem Ipsum</a>
            </li>
            <li>
                <a href="https://www.iastate.edu/" aria-current="page">Kitchen Sink</a>
                <ul>
                    <li>
                        <a href="https://www.iastate.edu/">Child Page Title Lorem</a>
                    </li>
                    <li>
                        <a href="https://www.iastate.edu/">Child Page Title Lorem</a>
                    </li>

                </ul>
            </li>
            <li>
                <a href="https://www.iastate.edu/">Sibling Page Title Lorem Ipsum</a>
            </li>

        </ul>
    </div>
</nav>
{% macro menu_items(items, level = 0) %}
  {% for item in items %}
    <li>
      {% if item.url %}
        <a href="{{ item.url }}"{% if item.current %} aria-current="page"{% endif %}>{{ item.title }}</a>
      {% else %}
        <span class="lw_separator_title">{{ item.title }}</span>
      {% endif %}
      {% if (level == 0 or item.current) and item.children|length %}
        <ul>
          {{ _self.menu_items(item.children, level + 1) }}
        </ul>
      {% endif %}
    </li>
  {% endfor %}
{% endmacro %}

{% import _self as macros %}

<nav class="subnav" aria-label="Sidebar">
  <noscript>
    <button class="subnav__toggle" id="subnav-toggle" aria-controls="subnav-list" aria-haspopup="true" aria-expanded="false"><span>Explore This Section</span></button>
  </noscript>
  <div class="subnav__list" id="subnav-list">
    <ul>
      {{ macros.menu_items(items) }}
    </ul>
  </div>
</nav>
{
  "items": [
    {
      "url": "https://www.iastate.edu/",
      "title": "Parent Page",
      "current": false,
      "children": []
    },
    {
      "url": "https://www.iastate.edu/",
      "title": "Sibling Page Title Lorem Ipsum",
      "current": false,
      "children": []
    },
    {
      "url": "https://www.iastate.edu/",
      "title": "Sibling Page Title Lorem Ipsum",
      "current": false,
      "children": []
    },
    {
      "url": "https://www.iastate.edu/",
      "title": "Sibling Page Title Lorem Ipsum",
      "current": false,
      "children": []
    },
    {
      "url": "https://www.iastate.edu/",
      "title": "Kitchen Sink",
      "current": true,
      "children": [
        {
          "url": "https://www.iastate.edu/",
          "title": "Child Page Title Lorem",
          "current": false,
          "children": []
        },
        {
          "url": "https://www.iastate.edu/",
          "title": "Child Page Title Lorem",
          "current": false,
          "children": []
        }
      ]
    },
    {
      "url": "https://www.iastate.edu/",
      "title": "Sibling Page Title Lorem Ipsum",
      "current": false,
      "children": []
    }
  ]
}
  • Content:
    .subnav {
      position: relative;
      box-shadow: rem(9) rem(9) rem(20) rgba($black, 0.15);
      &.subnav--horizontal {
        @include media-breakpoint-up(lg) {
          box-shadow: none;
        }
      }
      &:hover {
        html:not(.js) & {
          .subnav__list {
            height: auto;
          }
        }
      }
    }
    
    .subnav__toggle {
      background-color: $iastate-red;
      height: rem(66);
      color: $white;
      cursor: pointer;
      display: block;
      border: none;
      font-family: $typeface-sans-serif;
      font-size: rem(18);
      font-weight: 700;
      line-height: (26 / 18);
      padding: 0 rem(30);
      position: relative;
      text-align: left;
      transition: 0.2s color ease-in-out;
      width: 100%;
      margin-right: rem(13);
      @include media-breakpoint-up(xl) {
        display: none;
      }
      &:before {
        content: "";
        position: absolute;
        right: rem(20);
        top: rem(25);
        width: rem(10);
        height: rem(10);
        border-right: rem(2) solid $iastate-gold;
        border-bottom: rem(2) solid $iastate-gold;
        transform: rotate(45deg);
        transform-origin: center;
        transition: 0.2s transform ease-in-out, 0.2s top ease-in-out;
        .subnav--horizontal & {
          @include media-breakpoint-up(lg) {
            top: rem(15);
          }
        }
      }
      &:hover,
      .user-tabbing &:focus {
        color: $white;
      }
      &[aria-expanded="true"] {
        &:before {
          transform: rotate(225deg);
          top: rem(30);
          .subnav--horizontal & {
            @include media-breakpoint-up(lg) {
              top: rem(21);
            }
          }
        }
      }
      .subnav--horizontal & {
        @include media-breakpoint-up(lg) {
          display: block;
          max-width: rem(215);
          font-size: rem(16);
          line-height: (26 / 16);
          height: rem(46);
          padding: 0 rem(18);
        }
      }
    }
    
    .subnav__list {
      overflow: hidden;
      transition: 0.2s height ease-in-out;
      background-color: $white;
      &[aria-hidden="true"] {
        height: 0;
      }
      html:not(.js) & {
        height: 0;
        @include media-breakpoint-up(xl) {
          height: auto;
        }
      }
      // yellow line for desktop horizontal variant
      .subnav--horizontal & {
        &:before {
          @include media-breakpoint-up(lg) {
            content: "";
            position: absolute;
            border-bottom: rem(2) solid $iastate-gold;
            left: rem(-70);
            right: rem(-70);
            bottom: 0;
            opacity: 0;
            transition: 0.2s opacity ease-in-out;
          }
          @include media-breakpoint-up(xl) {
            left: rem(-190);
            right: rem(-81);
          }
        }
        &[aria-hidden="false"] {
          &:before {
            opacity: 1;
          }
        }
      }
      // sibling pages only
      & > ul {
        padding-bottom: rem(20);
        .subnav--horizontal & {
          @include media-breakpoint-up(lg) {
            display: flex;
            flex-wrap: wrap;
            padding: rem(35) rem(10) rem(25) 0;
            position: relative;
          }
        }
        & > li {
          padding: 0 rem(30);
          .subnav--horizontal & {
            @include media-breakpoint-up(lg) {
              flex: 0 1 auto;
              width: 20%;
              padding: 0 rem(20) 0 0;
              margin-bottom: rem(20);
            }
          }
          a {
            display: block;
            font-family: $typeface-sans-serif;
            font-weight: 700;
            font-size: rem(18);
            line-height: (22 / 18);
            color: $iastate-maroon;
            text-decoration: none;
            padding: rem(15);
            position: relative;
            z-index: 1;
            transition: 0.2s background-color ease-in-out, 0.2s color ease-in-out;
            @include media-breakpoint-up(xl) {
              font-size: rem(16);
              line-height: (19 / 16);
              padding-right: rem(25);
            }
            // current page and hover
            &:hover,
            &[aria-current="page"] {
              background-color: $off-white;
              color: $iastate-red;
              @include media-breakpoint-up(xl) {
                background-color: $white;
              }
              &:after {
                opacity: 0;
              }
              &:before {
                @include media-breakpoint-up(xl) {
                  opacity: 1;
                }
              }
            }
            &[aria-current="page"] {
              pointer-events: none;
            }
            // border under each item
            &:after {
              content: "";
              position: absolute;
              bottom: rem(-1);
              left: rem(14);
              right: 0;
              border-bottom: rem(1) solid $grey2;
              transition: 0.2s opacity ease-in-out;
              @include media-breakpoint-up(xl) {
                right: rem(20);
              }
            }
            // background color for desktop items
            &:before {
              @include media-breakpoint-up(xl) {
                content: "";
                background-color: $off-white;
                position: absolute;
                left: rem(-30);
                right: rem(-30);
                top: 0;
                bottom: 0;
                z-index: -1;
                opacity: 0;
                transition: 0.2s opacity ease-in-out;
              }
            }
            .subnav--horizontal & {
              @include media-breakpoint-up(lg) {
                padding: 0;
              }
              &:before,
              &:after {
                display: none;
              }
            }
          }
          // first child parent page
          &:first-child {
            @include media-breakpoint-up(xl) {
              background-color: $iastate-red;
            }
            a {
              @include media-breakpoint-up(xl) {
                padding: rem(21) rem(15);
                color: $white;
                font-size: rem(18);
                line-height: (25 / 18);
              }
              &:before {
                content: "";
                position: absolute;
                left: 0;
                top: rem(21);
                width: rem(9);
                height: rem(9);
                border-right: rem(2) solid $iastate-maroon;
                border-bottom: rem(2) solid $iastate-maroon;
                transform: rotate(135deg);
                transform-origin: center;
                transition: 0.2s left ease-in-out;
                @include media-breakpoint-up(xl) {
                  border-color: $iastate-gold;
                  top: rem(29);
                  opacity: 1;
                  background-color: transparent;
                }
              }
              &::after {
                @include media-breakpoint-up(xl) {
                  display: none;
                }
              }
              &:hover {
                @include media-breakpoint-up(xl) {
                  background-color: $iastate-red;
                }
                &:before {
                  @include media-breakpoint-up(xl) {
                    left: rem(-8);
                  }
                }
                .subnav--horizontal & {
                  @include media-breakpoint-up(lg) {
                    background-color: transparent;
                    color: $iastate-red;
                  }
                }
              }
              .subnav--horizontal & {
                @include media-breakpoint-up(lg) {
                  padding: 0;
                  color: $iastate-maroon;
                  font-size: rem(16);
                  line-height: (19 / 16);
                }
              }
            }
            .subnav--horizontal & {
              @include media-breakpoint-up(lg) {
                background-color: transparent;
              }
            }
          }
          // last child removing border
          &:last-child {
            a {
              &:after {
                display: none;
              }
            }
          }
          // child page ul hidden on mobile
          > ul {
            margin-bottom: rem(10);
            .subnav--horizontal & {
              @include media-breakpoint-up(lg) {
                display: none;
              }
            }
            li {
              a {
                font-family: $typeface-sans-serif;
                font-weight: 400;
                text-decoration: underline;
                text-decoration-thickness: rem(1);
                text-underline-offset: rem(1);
                padding: rem(8) rem(17) rem(8) rem(30);
                &:after {
                  display: none;
                }
              }
              &:last-child {
                a {
                  &:after {
                    display: block;
                    bottom: rem(-10);
                  }
                }
              }
            }
          }
        }
      }
    }
    
  • URL: /components/raw/subnav/_subnav.scss
  • Filesystem Path: src/components/subnav/_subnav.scss
  • Size: 8 KB

Subnav

This is a custom component (no class prefix in place as it does not conflict with bootstrap)

  • There are two variations of the subnav, the default subnav is collapsed on mobile and opens up on click, the --horizontal variant is used on top-level landing-pages and expands underneath the hero image
  • On mobile (for both variants), the menu requires a click (or touch) to be opened, and for the default version on desktop it is open by default
  • The subnav is setup to use :target when a user has javascript disabled, to allow the nav to open on mobile
  • The custom styles are imported into the index.scss file as @import "../components/subnav/subnav";
  • The .ts file for the javascript is imported into the index.ts file as import subnavInit from "./components/subnav"; and called as subnavInit();