Improved .visually-hidden
Theorically bulletproof CSS class for visually hide anything and keep it accessible to ATs.
<button type="button"> | |
<span aria-hidden="true">×</span> | |
<span class="visually-hidden">Close</span> | |
</button> | |
<a href="/" class="visually-hidden-focusable">Skippy</a> | |
<div dir="rtl"> | |
<button type="button"> | |
<span aria-hidden="true">×</span> | |
<span class="visually-hidden">Close</span> | |
</button> | |
<a href="ffoodd.fr" class="visually-hidden-focusable">ffoodd.fr</a> | |
</div> |
/* | |
Improved screen reader only CSS class | |
@author Gaël Poupard | |
@note Based on Yahoo!'s technique | |
@author Thierry Koblentz | |
@see https://www.cssmojo.com/hide-content-from-sighted-users/ | |
* 1. | |
@note Use to only display content when it's focused, or one of its child elements is focused | |
@see http://www.w3.org/TR/2013/NOTE-WCAG20-TECHS-20130905/G1 | |
@note Based on a HTML5 Boilerplate technique, included in Bootstrap | |
* 2. | |
@note `clip-path` shortest syntax | |
@author Yvain Liechti | |
@see https://twitter.com/ryuran78/status/778943389819604992 | |
* 3. | |
@note preventing text to be condensed | |
author J. Renée Beach | |
@see https://medium.com/@jessebeach/beware-smushed-off-screen-accessible-text-5952a4c2cbfe | |
@note Drupal 8 goes with word-wrap: normal instead | |
@see https://www.drupal.org/node/2045151 | |
@see http://cgit.drupalcode.org/drupal/commit/?id=5b847ea | |
* 4. | |
@note !important is important | |
@note Obviously you wanna hide something | |
@author Harry Roberts | |
@see https://csswizardry.com/2016/05/the-importance-of-important/ | |
*/ | |
.visually-hidden, | |
.visually-hidden-focusable:not(:focus, :focus-within) { | |
border: 0 !important; | |
clip-path: inset(50%) !important; /* 2 */ | |
height: 1px !important; | |
margin: -1px !important; | |
overflow: hidden !important; | |
padding: 0 !important; | |
width: 1px !important; | |
white-space: nowrap !important; /* 3 */ | |
} | |
/* | |
Prevent visually hidden caption from breaking table's collapsing borders | |
@author Louis-Maxime Piton | |
@see https://github.com/twbs/bootstrap/pull/37533 | |
*/ | |
.visually-hidden:not(caption), | |
.visually-hidden-focusable:not(caption):not(:focus, :focus-within) { | |
position: absolute !important; | |
} | |
/* | |
Prevent overflowing children from being focusable. | |
@author Django Janny | |
@see https://github.com/twbs/bootstrap/pull/41286 | |
*/ | |
.visually-hidden *, | |
.visually-hidden-focusable:not(:focus, :focus-within) * { | |
overflow: hidden !important; | |
} |
@keltroth If I sum this case up, since scrollable elements become focusable, any child in . sr-only
that'd become scrollable will also become focusable.
I can't see any other workaround than yours, for now.
FWIW, as @thiemeljiri said, .sr-only
is usually used on text-level elements. Hiding complex content this way should be avoided.
I came to the same conclusion. Thanks !
Hi @keltroth. What's your use case to put a scrollable content into the sr-only area? I cannot imagine a case where that would be correct. Definitely not with
.sr-only
which is also applied to a focused element or an element with focus within like the code in this thread. (I actually always implement this class in a way that it's not applied to a focused element or element with focus within as there is no case where that would be correct. Focused element must be always visible to the user.)Also, scrollable regions should be ideally always focusable (unless you are certain that there always is a focusable element inside it, but ideally even in such case). We're usually doing this manually via setting tabindex="0" to that element, often combined with some role like role="region". Read these resources to understand why:
Thanks to your post, I found out that some browsers actually started making scrollable regions scrollable automatically. As this is crucial for keyboard users (see the links above for explanation), this is something you have to respect. Read this article where this feature is mentioned:
https://cassey.dev/til/2019-11-19-overflow-scroll-gets-focus/
To sum it up, your use case or intention is most likely not OK. Please provide more information to help us guide you to find a correct solution for your case.