Core Toggle
@nrk/core-toggle
makes a <button>
toggle the visibility of next element sibling. Toggles can be nested and easily extended with custom animations or behavior through the toggle event.
Example
<button>Popup VanillaJS</button>
<core-toggle class="my-dropdown" popup hidden>
<ul>
<li><a>Link</a></li>
<li>
<button>Can also be nested</button>
<core-toggle class="my-dropdown" popup hidden>
<ul>
<li><a>Sub-link</a></li>
<li><input type="text" autofocus aria-label="Skriv her"></li>
</ul>
</core-toggle>
</li>
</ul>
</core-toggle>
<div id="jsx-toggle-popup"></div>
<script type="text/jsx">
ReactDOM.render(<>
<button>Popup JSX</button>
<CoreToggle className='my-dropdown' hidden popup onToggleSelect={console.warn}>
<ul>
<li><button>Select</button></li>
<li><a href='#'>Link</a></li>
<li>
<button>Can also be nested</button>
<CoreToggle className='my-dropdown' hidden popup>
<ul>
<li><a href='#'>Sub-link</a></li>
</ul>
</CoreToggle>
</li>
</ul>
</CoreToggle>
</>, document.getElementById('jsx-toggle-popup'))
</script>
Installation
Using NPM provides own element namespace and extensibility.
Recommended:
npm install @nrk/core-toggle
Using static registers the custom element with default name automatically:
<script src="https://static.nrk.no/core-components/major/7/core-toggle/core-toggle.min.js"></script>
Remember to polyfill custom elements if needed.
Usage
HTML / JavaScript
<button>Toggle VanillaJS</button>
<core-toggle
hidden <!-- Set hidden attribute to prevent FOUC -->
popup="{Boolean|String}">
<div>Content</div>
</core-toggle>
import CoreToggle from '@nrk/core-toggle'
window.customElements.define('core-toggle', CoreToggle)
const myToggle = document.querySelector('core-toggle')
myToggle.button
myToggle.popup
myToggle.hidden
myToggle.value
myToggle.popup = true
myToggle.hidden = true
myToggle.value = 'Velg'
React / Preact
import CoreToggle from '@nrk/core-toggle/jsx'
<CoreToggle
hidden
popup={Boolean|String}
ref={(comp) => {}}
forwardRef={(el) => {}}
onToggle={Function}
onToggleSelect={Function}>
<button>Use with JSX</button>
<div>Content</div>
</CoreToggle>
Markup
With for
Putting the toggle button directly before the content is highly recommended, as this fulfills all accessibility requirements by default. There might be scenarios though, where styling makes this DOM structure impractical. In such cases, give the <button>
a for
attribute, and the <core-toggle>
an id
with corresponding value. Make sure there is no text between the button and toggle content, as this will break the experience for screen reader users:
<div>
<button for="my-toggle">Toggle VanillaJS</button>
</div>
<core-toggle id="my-toggle" hidden>Content</core-toggle>
Autofocus
If you have form elements inside a <core-toggle>
, you can optionally add a autofocus
attribute to the most prominent form element. This helps the user navigate quickly when toggle is opened.
Events
toggle
Fired after open state changes:
document.addEventListener('toggle', (event) => {
event.target
})
toggle.select
Fired whenever an <a>
or <button>
element is selected inside a toggle with the popup
option enabled.
Useful for setting the value of the toggle button with the selected value.
document.addEventListener('toggle.select', (event) => {
event.target
event.detail
event.target.value = event.detail
})
Styling
Note: <core-toggle>
is display: inline
by default. Change this by for instance setting core-tabs:not([hidden]) { display: block | flex | grid }
or similar in your app. Not needed when position
or float
is used. All styling in documentation is example only. Both the <button>
and <core-toggle>
element receive attributes reflecting the current toggle state:
.my-button {}
.my-button[aria-expanded="true"] {}
.my-button[aria-expanded="false"] {}
.my-toggle-content {}
.my-toggle-content:not([hidden]) {}
.my-toggle-content[hidden] {}
Example: Expand
Content is only toggled when clicking the button. Great for accordions and expand/collapse panels.
<button>Toggle VanillaJS</button>
<core-toggle hidden>Content</core-toggle>
<div id="jsx-toggle-default"></div>
<script type="text/jsx">
ReactDOM.render(<>
<button>Toggle JSX</button>
<CoreToggle hidden onToggle={console.log}>Content</CoreToggle>
</>, document.getElementById('jsx-toggle-default'))
</script>
Example: Select
Listen to the toggle.select
event and update the button's value from the selected item
to create a component that behaves like a <select>
:
<button>Episode 1</button>
<core-toggle class="my-select my-dropdown" hidden popup="Choose episode">
<ul>
<li><button>Episode 1</button></li>
<li><button>Episode 2</button></li>
<li><button>Episode 3</button></li>
</ul>
</core-toggle>
<script>
document.addEventListener('toggle.select', (event) => {
if (!event.target.classList.contains('my-select')) return
event.target.value = event.detail
event.target.hidden = true
event.target.button.focus()
})
</script>
<div id="jsx-toggle-select"></div>
<script type="text/jsx">
class MyToggleSelect extends React.Component {
constructor (props) {
super(props)
this.state = { value: 'Select number' }
this.onSelect = this.onSelect.bind(this)
}
onSelect (event) {
event.target.hidden = true
this.setState({ value: event.detail.textContent })
}
render () {
return <>
<button>{this.state.value}</button>
<CoreToggle className='my-dropdown' popup='Example picker' hidden onToggleSelect={this.onSelect}>
<ul>
<li><button>One</button></li>
<li><button>Two</button></li>
<li><button>Three</button></li>
</ul>
</CoreToggle>
</>
}
}
ReactDOM.render(<MyToggleSelect/>, document.getElementById('jsx-toggle-select'))
</script>
FAQ
Why not use <details>
instead?
Despite having a native <details>
element for expanding/collapsing content, there are several issues regarding browser support, styling, accessibility. Furthermore, polyfills often conflict with other standards such as <dialog>
.
The menu role is mainly inteded for context menues and toolbars in application interfaces, and has quite complex keyboard navigation requirements. As most end users will not expect application behavior in websites and internal web based systems, (implemented) attributes like aria-controls
and aria-labelledby
is sufficient for a good user experience.
Why does dropdowns not open on hover?
Both touch devices and screen readers will have trouble properly interacting with hoverable interfaces (unless more complex fallback logic is implemented). To achieve a consistent and accessible interface, <core-toggle>
is designed around click interactions.
Why is there no group-option to achieve a single open toggle?
Some expand/collapse interfaces like accordions behaves like a group - allowing only one expanded area at the time. This pattern however requires more logic and carefully designed animations to avoid confusion over expected scroll position.
Example: The user first opens "Toggle-1", and then "Toggle-2" (which closes "Toggle-1"). Since "Toggle-1" is placed above, the position "Toggle-2" now changes - potentially outside the viewport on smaller devices.
Note: If you do need to implement grouping, you can achieve this by reacting to the toggle event.