-
Notifications
You must be signed in to change notification settings - Fork 237
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add Service Navigation component, add new Mobile Navigation component #4368
base: main
Are you sure you want to change the base?
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,128 @@ | ||
import { Component } from 'govuk-frontend' | ||
|
||
/** | ||
* Mobile Navigation enhancement for Service Navigation component | ||
*/ | ||
class MobileNavigation extends Component { | ||
static moduleName = 'app-mobile-navigation' | ||
|
||
/** | ||
* @param {Element} $root - HTML element | ||
*/ | ||
constructor($root) { | ||
super($root) | ||
|
||
this.templates = this.$root.querySelectorAll( | ||
'.app-mobile-navigation__template' | ||
) | ||
this.links = this.$root.querySelectorAll('a') | ||
|
||
Array.from(this.templates).forEach((template) => { | ||
const templateClone = template.content.cloneNode(true) | ||
let link | ||
|
||
if (template.parentNode.tagName === 'A') { | ||
link = template.parentNode | ||
link.removeChild(template) | ||
} else { | ||
link = template.parentNode.parentNode | ||
template.parentNode.removeChild(template) | ||
} | ||
|
||
const button = document.createElement('button') | ||
button.classList.add('govuk-service-navigation__link') | ||
button.classList.add('app-mobile-navigation__toggle-button') | ||
button.setAttribute( | ||
'aria-expanded', | ||
String( | ||
link.parentNode.classList.contains( | ||
'govuk-service-navigation__item--active' | ||
) | ||
) | ||
) | ||
button.textContent = link.textContent | ||
|
||
link.insertAdjacentElement('afterend', templateClone.firstElementChild) | ||
link.insertAdjacentElement('afterend', button) | ||
}) | ||
|
||
const currentLink = Array.from( | ||
this.$root.querySelectorAll( | ||
`.app-navigation__subnav-item a[href="${window.location.pathname.slice(0, -1)}"]` | ||
) | ||
).pop() | ||
|
||
if (currentLink) { | ||
currentLink.classList.add('app-mobile-navigation__link--active') | ||
} | ||
|
||
this.subNavs = $root.querySelectorAll('.app-mobile-navigation__list') | ||
|
||
// A global const for storing a matchMedia instance which we'll use to detect when a screen size change happens | ||
// Set the matchMedia to the govuk-frontend tablet breakpoint | ||
|
||
const breakPoint = getComputedStyle( | ||
document.documentElement | ||
).getPropertyValue('--govuk-frontend-breakpoint-tablet') | ||
|
||
this.mql = window.matchMedia(`(min-width: ${breakPoint})`) | ||
|
||
// MediaQueryList.addEventListener isn't supported by Safari < 14 so we need | ||
// to be able to fall back to the deprecated MediaQueryList.addListener | ||
if ('addEventListener' in this.mql) { | ||
this.mql.addEventListener('change', () => this.setHiddenStates()) | ||
} else { | ||
// @ts-expect-error Property 'addListener' does not exist | ||
this.mql.addListener(() => this.setHiddenStates()) | ||
} | ||
|
||
this.setHiddenStates() | ||
this.setEventListener() | ||
} | ||
|
||
/** | ||
* Set up event delegation for button clicks | ||
*/ | ||
setEventListener() { | ||
this.$root.addEventListener('click', (e) => { | ||
if (e.target.tagName === 'BUTTON') { | ||
const ariaExpanded = e.target.getAttribute('aria-expanded') === 'true' | ||
|
||
e.target.setAttribute('aria-expanded', !ariaExpanded) | ||
|
||
const subNav = e.target.nextSibling | ||
|
||
if ( | ||
subNav && | ||
subNav.classList && | ||
subNav.classList.contains('app-mobile-navigation__list') | ||
) { | ||
if (ariaExpanded) { | ||
subNav.setAttribute('hidden', '') | ||
} else { | ||
subNav.removeAttribute('hidden') | ||
} | ||
} | ||
} | ||
}) | ||
} | ||
|
||
/** | ||
* Hide links if viewport is below tablet | ||
*/ | ||
setHiddenStates() { | ||
if (!this.mql.matches) { | ||
this.links.forEach((a) => a.setAttribute('hidden', '')) | ||
this.subNavs.forEach((subNav) => { | ||
if (subNav.previousSibling.getAttribute('aria-expanded') === 'true') { | ||
subNav.removeAttribute('hidden') | ||
} | ||
}) | ||
} else { | ||
this.subNavs.forEach((x) => x.setAttribute('hidden', '')) | ||
this.links.forEach((a) => a.removeAttribute('hidden')) | ||
} | ||
} | ||
} | ||
|
||
export default MobileNavigation |
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
@@ -0,0 +1,101 @@ | ||||||||||||||||||||||||||||
$navigation-height: 50px; | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-mobile-navigation__theme { | ||||||||||||||||||||||||||||
@include govuk-typography-common; | ||||||||||||||||||||||||||||
position: relative; // this is to get around the artificial click area generated by the :after of the parent button | ||||||||||||||||||||||||||||
margin: 0; | ||||||||||||||||||||||||||||
padding: 0; | ||||||||||||||||||||||||||||
color: govuk-colour("dark-grey"); | ||||||||||||||||||||||||||||
// Font is defined as a hard 19px so | ||||||||||||||||||||||||||||
// it does not re-size on mobile viewport | ||||||||||||||||||||||||||||
font-size: 19px; | ||||||||||||||||||||||||||||
font-size: govuk-px-to-rem(19px); | ||||||||||||||||||||||||||||
font-weight: normal; | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-mobile-navigation__list { | ||||||||||||||||||||||||||||
margin: 0; | ||||||||||||||||||||||||||||
padding: 0; | ||||||||||||||||||||||||||||
list-style: none; | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
@include govuk-media-query($from: tablet) { | ||||||||||||||||||||||||||||
position: relative; | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
// Offset gutter by tablet list-item padding | ||||||||||||||||||||||||||||
left: $govuk-gutter-half * -1; | ||||||||||||||||||||||||||||
width: calc(100% - $govuk-gutter); | ||||||||||||||||||||||||||||
min-height: $navigation-height; | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-mobile-navigation__list-item--current { | ||||||||||||||||||||||||||||
@include govuk-media-query($from: tablet) { | ||||||||||||||||||||||||||||
border-bottom: 4px solid govuk-colour("blue"); | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-navigation__subnav-item { | ||||||||||||||||||||||||||||
display: block; | ||||||||||||||||||||||||||||
position: relative; | ||||||||||||||||||||||||||||
padding: govuk-spacing(3); | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-navigation__subnav-item--current { | ||||||||||||||||||||||||||||
$_current-indicator-width: 4px; | ||||||||||||||||||||||||||||
padding-left: govuk-spacing(4) - $_current-indicator-width; | ||||||||||||||||||||||||||||
border-left: $_current-indicator-width solid govuk-colour("blue"); | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-mobile-navigation .govuk-service-navigation__item--active { | ||||||||||||||||||||||||||||
@include govuk-media-query($until: tablet) { | ||||||||||||||||||||||||||||
border-color: transparent; | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-mobile-navigation__link--active { | ||||||||||||||||||||||||||||
@include govuk-media-query($until: tablet) { | ||||||||||||||||||||||||||||
border-left: 5px solid rgb(26.1, 100.8, 165.6); | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-mobile-navigation__toggle-button { | ||||||||||||||||||||||||||||
@include govuk-media-query($from: tablet) { | ||||||||||||||||||||||||||||
display: none; | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
position: relative; | ||||||||||||||||||||||||||||
padding: 0; | ||||||||||||||||||||||||||||
border: 0; | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
background: none; | ||||||||||||||||||||||||||||
font-size: inherit; | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-mobile-navigation__toggle-button::after { | ||||||||||||||||||||||||||||
content: ""; | ||||||||||||||||||||||||||||
box-sizing: border-box; | ||||||||||||||||||||||||||||
display: block; | ||||||||||||||||||||||||||||
position: absolute; | ||||||||||||||||||||||||||||
right: govuk-px-to-rem(-14px); | ||||||||||||||||||||||||||||
bottom: govuk-px-to-rem(10px); | ||||||||||||||||||||||||||||
width: govuk-px-to-rem(6px); | ||||||||||||||||||||||||||||
height: govuk-px-to-rem(6px); | ||||||||||||||||||||||||||||
transform: rotate(135deg); | ||||||||||||||||||||||||||||
border-top: govuk-px-to-rem(2px) solid; | ||||||||||||||||||||||||||||
border-right: govuk-px-to-rem(2px) solid; | ||||||||||||||||||||||||||||
Comment on lines
+83
to
+84
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Solid way to have the chevron visible for people using high contrast mode 🙌🏻 |
||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-mobile-navigation__toggle-button[aria-expanded="false"] ~ .app-navigation__list { | ||||||||||||||||||||||||||||
display: none; | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
Comment on lines
+87
to
+89
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We may be able to let that selector go when we re-implement the current navigation feature for using |
||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-mobile-navigation__toggle-button[aria-expanded="true"]::after { | ||||||||||||||||||||||||||||
transform: rotate(-45deg); | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
Comment on lines
+91
to
+93
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. suggestion This can likely be the default state of the button and save a selector. |
||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
.app-mobile-navigation__toggle-button[aria-expanded="true"] ~ .app-navigation__list { | ||||||||||||||||||||||||||||
@include govuk-media-query($until: tablet) { | ||||||||||||||||||||||||||||
display: block; | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||
display: none; | ||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||
Comment on lines
+95
to
+101
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. suggestion Hiding the navigation list and toggle would gain at being regrouped in a single block of code and a single media query for easier maintainability rather than split in their own selector, in the lines of:
Suggested change
This would allow the file to describe first how the different parts look, then when they're visible or not. That media query may also not be necessary once the JavaScript handles using |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -26,6 +26,7 @@ $govuk-new-typography-scale: true; | |
@import "govuk/components/panel"; | ||
@import "govuk/components/phase-banner"; | ||
@import "govuk/components/radios"; | ||
@import "govuk/components/service-navigation"; | ||
@import "govuk/components/skip-link"; | ||
@import "govuk/components/summary-list"; | ||
@import "govuk/components/table"; | ||
|
@@ -58,7 +59,7 @@ $app-code-color: #d13118; | |
@import "components/highlight"; | ||
@import "components/image-card"; | ||
@import "components/masthead"; | ||
@import "components/navigation"; | ||
@import "components/mobile-navigation"; | ||
@import "components/options"; | ||
@import "components/page-navigation"; | ||
@import "components/phase-banner"; | ||
|
@@ -68,8 +69,17 @@ $app-code-color: #d13118; | |
@import "components/tabs"; | ||
@import "components/table"; | ||
|
||
.app-nav-subitems-mobile { | ||
@include govuk-media-query($from: tablet) { | ||
display: none; | ||
} | ||
|
||
background-color: govuk-colour("mid-grey"); | ||
} | ||
|
||
// We don't change the global width container width so that examples are the current width. | ||
.app-width-container { | ||
.app-width-container, | ||
.govuk-service-navigation .govuk-width-container { | ||
Comment on lines
+81
to
+82
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This may be an indication that we may want to update the CSS variable in this stylesheet and add a custom width container to the examples 🤔 This is something we can explore in a future PR, though 😊 |
||
@include govuk-width-container(1100px); | ||
} | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
issue This has a risk to collect elements more broadly than we intend, if one day we add some content to the navigation through slots, for example. Would be good to limit this to links from the actual navigation using their class from the Service Navigation (unless we can set up our own
.js-
prefixed class).