HTML Attributes
HTML attributes in Markup templates are just HTML attributes. One specific behavior change is related to boolean attributes and besides that, your knowledge about HTML attributes transfer as is.
Boolean attributes
Boolean attributes in HTML are attributes that represent true
or false
values.
html`
<p hidden="false">hidden text</p>
<button disabled>click me</button>
<input type="checkbox" checked="false" />
`.render(document.body)
// <p hidden="false">hidden text</p> <- still hidden
// <button disabled="">click me</button>
// <input type="checkbox" checked="false"> <- still checked
The issue with boolean attributes in HTML is that giving them the value of false
does not stop their effect on the element. If they are present in the tag they work as having the value of true
no matter their value.
Markup honors the true
or false
values and allows you to add or remove these attributes just by specifying their boolean values.
const hidden = false
const disabled = true
const checked = false
html`
<p hidden="${hidden}">hidden text</p>
<button disabled="${disabled}">click me</button>
<input type="checkbox" checked="${checked}" />
`.render(document.body)
// <p>hidden text</p>
// <button disabled="true">click me</button>
// <input type="checkbox">
Setting an attribute value to false
(as string or boolean) or nil (undefined
or null
) will remove the attribute.
const disabled = false
const checked = null
html`
<p hidden="false">hidden text</p>
<button disabled="${disabled}">click me</button>
<input type="checkbox" checked="${checked}" />
`.render(document.body)
// <p>hidden text</p>
// <button>click me</button>
// <input type="checkbox">
Value attributes
Markup is aware of value you inject in the template as attribute values and will track and update them accordingly. There is no extra syntax necessary to make this happen.
const type = 'button'
const active = 'active'
const style = 'color: white; background: black'
html`
<button
type="${type}"
class="btn ${active} common"
style="border: none; ${style}"
>
click me
</button>
`.render(document.body)
// <button type="button" class="btn active common" style="border: none; color: white; background: black">click me</button>
Event attributes
HTML allows you to declare inline event attributes and they work the same with markup. No extra syntax is needed to add attach events but Markup does additional things in the background you can learn more about by reading the events docs.
const handleClick = (event) => {
console.log(event)
}
html`<button onclick="${handleClick}">click me</button>`
Reference attribute
One thing that exists in Markup and not in HTML is the ref
attribute that allows you to create a reference to an element you can use to access the rendered DOM element for whatever operation you need. You can read about references for more details.
html`<button ref="btn">click me</button>`
Attributes as object
It is good to know all the possible attributes in advance but sometimes that's not possible. For that, you can collect attributes as objects and just inject them as is overriding as needed.
const [count, setCount] = state(0)
const btn = ({ text = 'click me', ...props }) =>
html` <button ${props} type="button">${text}</button>`
const countUp = () => {
setCount((prev) => prev + 1)
}
const temp = html`
<p>${count}</p>
${button({ text: '+', ariaLabel: 'count up button', onClick: countUp })}
`
/* renders:
<p>0</p>
<button type="button" aria-label="count up button">+</button>
*/
temp.render(document.body)
Any attribute you set after injecting the attribute object will override the object attribute name value. In the example above, we ensure that the type
of the button is always button
by setting it after injecting the props
object.
Also, you can use camelcase attribute name to changed to kebab-case. In the example above ariaLabel
will become aria-label
when the button renders. The onClick
will change to onclick
and handled like an event attributes.