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.

javascript
                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.

javascript
                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.

javascript
                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.

javascript
                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.

javascript
                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.

javascript
                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.

edit this doc