HTML training
- 😸 Author: Marta García
- 🗓️ Last updated: 18/07/2023
The next HTML snippets are examples of an accessible and SEO friendly semantic code, there are other ways to achieve the same results. These are based on the best practices conventions and our dev experience dealing with UI hacks.
This HTML guide is intended to be a quick reference on your development journey, for a deeper understanding we highly recommend MDN Web Docs (opens in a new tab) and the blogs listed in our UI bookmarks.
So you know at Z1 we encourage designers and developers to use native elements as they're meant to be. The Web is accessible by default, let's not break it!
Links
Internal links
The link of an internal page can be set as active with the aria-current="page"
that way can be use as selector to add the active styles and also provide information.
To add interactive actions is actually a button
what you need to use, thought sometimes a
can have side events like analytics triggers because navigate is still its main function.
Using rel="nofollow"
in our internal links not important for SEO can help with pages prioritation because search engines don't trace nofollow links.
<a href="/about/">About us</a>
<a href="/about/" aria-current="page">About us</a>
<a href="/about/" onclick="analyticsEvent()">About us</a>
<a href="/about/" rel="nofollow">Privacy Policy</a>
Anchor links
To quickly navigate along the same page or to link a specific section of another page. Adding an id
to the element and using it behind the page path
will have ready this awesome feature.
<a href="#section-name">Section name</a>
<a href="/about#section-name">Section name</a>
Skip link
Super useful to jump to the main content skiping navigation. Could be visually hidden only when it's not focused, so will be accessible by screen readers and keyboard users, without affect the design.
<a href="#main-content" class="visually-hidden">Skip navigation</a>
External links
There is a debate about whether external links should be opened in a new tab, which in terms of accessibility is an issue. So may is better to give the user the option to choose, this will probably end up being the normal behavior of the Web, indeed browsers already provide that option.
Another concern with external links is that the use of target="_blank"
exposes to more vulnerabilities and for security reasons we must add to the rel
attribute the values noopener
and noreferrer
, although most browsers have already implemented security measures.
Followed links give SEO benefits to the link referred, so Google states that all paid links should be nofollowed. Because of that use rel="nofollow"
in external links can protect against Google penailties.
So with that in mind we should add nofollow
to social media links and big companies for example, don't use it only in links of websites that you want to rewards with organic SEO, like independent blogs or open source projects, those could be followed links.
<a href="https://some-blog.com" rel="noopener noreferrer" target="_blank">
Some blog
</a>
<a href="https://some-blog.com">Some blog</a>
<a href="https://marketing-partner.com" onclick="analyticsEvent()">
Marketing landing
</a>
<a href="https://github.com/" rel="nofollow">GitHub</a>
<a href="https://project.org">Project</a>
Download link
The filename will be the name of the downloaded file, we can specify a value for the download attribute for that to be the new name.
<a href="./doc.pdf" download>Download doc</a>
<a href="./doc.pdf" download="custom-file-name.pdf">Download doc</a>
Image links
If we wrap an image with a link this should have an alt
with information about the link.
<a href="/">
<img src="./header-logo.png" alt="Home" />
</a>
<a href="https://github.com/">
<img src="./github-logo.png" alt="GitHub" />
</a>
<a href="/blog/post-title">
<img src="./post-thumbnail.png" alt="Post title" />
</a>
Link with icon
Any svg
icon that is decorative or his meaning is redundant should have aria-hidden
to no disturb the user experience.
<a href="/post">
Next post
<svg aria-hidden="true"></svg>
</a>
Icon links
If the link doesn't have a text we need to include it as aria-label
.
<a href="#main-content" aria-label="Back to top">
<svg aria-hidden="true" width="100" height="100"></svg>
</a>
Descriptive links
The text should provide information about the link, if not we can add an aria-label
.
Thought that will replace the text inside, so an alternativa can be add the description in a span
visually hidden but useful for screen readers and others assitive technologies.
That cannot be achieved with the hidden
attribute, nether display: none
or visually: hidden
, it requires a CSS class as .visually-hidden
or .sr-only
with special styles for that purpose.
Each link must be descriptive by itself because we can navigate the page between links and not have context about where is being used.
<a href="/features" aria-label="New features available">Learn more</a>
<a href="/features">
Learn more
<span class="visually-hidden">about new features available</span>
</a>
Buttons
Button
Where ever you need and interactive element to add JS events you must use a button
, the appearance is not relevant can be customize it. The native buttons are accesible by default, div
instead will requiere a lot of extra work and accessibility testing.
It's important to know that inside button
the elements will be lost its semantic if for example contain headlines, also interactive elements as links or buttons cannot be nested inside another.
<button type="button" onclick="editItem()">Edit item</button>
Button with icon
Same that links, the icon should have aria-hidden
.
<button type="button" onclick="addItem()">
Add new item
<svg aria-hidden="true"></svg>
</button>
Icon button
As links if a button doesn't have text we need to add it with aria-label
or visually hidden.
<button type="button" onclick="removeItem()" aria-label="Remove item">
<svg aria-hidden="true"></svg>
</button>
<button type="button" onclick="editItem()">
<svg aria-hidden="true"></svg>
<span class="visually-hidden">Edit item details</span>
</button>
Lists
Unordered list
- Item
- Item
- Item
<ul>
<li>Item</li>
<li>Item</li>
<li>Item</li>
</ul>
Ordered list
- A
- B
- C
<ol>
<li>A</li>
<li>B</li>
<li>C</li>
</ol>
Reversed ordered list
- C
- B
- A
<ol reversed>
<li>C</li>
<li>B</li>
<li>A</li>
</ol>
List without bullets
If you use list-style: none
to remove the bullets of a list the element will be considered no longer a list, so you need to add role="list"
to recover its meaning. Be aware that in some cases like this CSS styles may affect the semantic and accessibility of HTML elements.
- Item
- Item
- Item
- Item
- Item
- Item
<ul role="list">
<li>Item</li>
<li>Item</li>
<li>Item</li>
</ul>
Nested lists
We can nest a list inside another and we can have multiple lists nested. The correct markup will be always do it inside a li
, because only li
can be direct child of ul
and ol
. For example we shouldn't use div
to group the list items, that would break the semantic structure.
- Item
- Item
Item list
- Item
- Item
Item list
- Item
- Item
- Item
- Item
- Item
<ul>
<li>Item</li>
<li>Item</li>
<li>
<p>Item list</p>
<ul>
<li>Item</li>
<li>Item</li>
<li>
<p>Item list</p>
<ol>
<li>Item</li>
<li>Item</li>
<li>Item</li>
</ol>
</li>
<li>Item</li>
</ul>
</li>
<li>Item</li>
</ul>
Cards list
The li element can contain any HTML element so we can use ul for many common UI patterns, like a list of features with title and text, could use ul
even if they look like cards, semantically it is a list.
Feature
Description
Feature
Description
Feature
Description
Feature
Description
Feature
Description
Feature
Description
<ul role="list">
<li>
<p><strong>Feature</strong></p>
<p>Description</p>
</li>
<li>
<p><strong>Feature</strong></p>
<p>Description</p>
</li>
<li>
<p><strong>Feature</strong></p>
<p>Description</p>
</li>
</ul>
Steps list
Same for steps of any kind, some have numbers others have arrows, we can use ol
because however they look like regarding to semantics they are ordered lists. If the arrows go in the other direction use the reversed
atribute for this markup to have the same meaning as it does visually.
Step
Instructions
Step
Instructions
Step
Instructions
Step →
Instructions
Step →
Instructions
Step
Instructions
<ol role="list">
<li>
<p><strong>Step</strong></p>
<p>Instructions</p>
</li>
<li>
<p><strong>Step</strong></p>
<p>Instructions</p>
</li>
<li>
<p><strong>Step</strong></p>
<p>Instructions</p>
</li>
</ol>
Forms
Label
Each input
must have a label
connected which should have a text inside. Thought there are anothers ways to make a form field accesible this is the standar. A quick check that a label is asign to the input is that if you click the label the input it should focus.
<label for="peas">Do you like peas?</label>
<input type="text" name="peas" id="peas" />
<label>
<span>Do you like peas?</span>
<input type="text" name="peas" />
</label>
Landmarks
Use the appropriate landmarks to identify the different regions of content on a web page. The most important landmark roles are main
and navigation
, as nearly every page will include at least those regions.
Use HTML5 sectioning elements that have a default ARIA landmark role: main (main)
, nav (navigation)
, aside (complementary)
and in some situations header (banner)
and footer (contentinfo)
. Do not nest landmarks with the same role.
Banner
The element header
has banner role by default, is commonly use for the top navbar that contain the main navigation and is the same on all pages.
Could be a sidebar as well, that's common in web applications, appearance is not as important as semantics in identifying whether it is a header. The main header
should be only one and cannot be nested inside another, but we can use it as a header of an article
.
<header>Site header</header>
<main>
<article>
<header>Article header</header>
</article>
</main>
Main
The main
landmark contain the most important and unique content of each page. Should be only one and cannot be nested inside another.
<main>Main</main>
Complementary
The aside
with complementary role is, as its name indicates, additional to the main content, it contains elements related to the main content but it is something extra, not so important.
Can be use next to the main
or inside an article
. A few examples or aside
can be related posts, shared links, banners, advertising, etc. Again appearance is not decisive, there may be multiple asides.
<main>
<article>
<aside>Share links</aside>
</article>
</main>
<aside>Related posts</aside>
Contentinfo
The footer
contain information about the site or related as navigation, contact, social links.
The main footer
should be only one and cannot be nested inside another, so like header
and aside
could be use inside article
.
<main>
<article>
<footer>Author info</footer>
</article>
</main>
<footer>Footer</footer>
Navigation
The nav
tag should be use when a container has links to pages or links related to the website. The main navigation appears in the header
, some others can be use across the pages in the main
content, like breadcrumb or tabs and listed in the footer
.
The secodary navigations will need an aria-label
, it's not necessary to include descriptive words like menu, navigation or links as the semantic tags will annouce by itself. Shouldn't be nested inside another.
<header>
<nav>Main navigation</nav>
</header>
<main>
<nav aria-label="Breadcrumb">Secondary navigation</nav>
<nav aria-label="Categories">Secondary navigation</nav>
</main>
<footer>
<nav aria-label="Legal">Secondary navigation</nav>
<nav aria-label="Social">Secondary navigation</nav>
</footer>
Region
An alternative landmark can be created using the section
element, which has a default landmark role of region, with an accessible name using aria-labelledby
to reference a heading element
<section aria-labelledby="this-section">
<h2 id="this-section">Section as landmark</h2>
</section>
Keep in mind that too many landmark roles create "noise" in screen readers, making it difficult to understand the overall layout of the page.
So if the section it's not going to be a landmark, what if we use div
instead, is that wrong? not at all, it would be the same, just that section
is semantic so the code look nicer :)
<div>
<h2>Section title</h2>
</div>
<section>
<h2>Section title</h2>
</section>