Skip to content

Commit

Permalink
fix(jsx): props are set as properties
Browse files Browse the repository at this point in the history
  • Loading branch information
kasperskei committed Jan 13, 2025
1 parent 0439318 commit ab42221
Show file tree
Hide file tree
Showing 2 changed files with 61 additions and 46 deletions.
28 changes: 11 additions & 17 deletions packages/jsx/src/index.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -38,27 +38,21 @@ it('static props & children', setup((ctx, h, hf, mount, parent) => {
}))

it('dynamic props', setup((ctx, h, hf, mount, parent) => {
const val = atom('val', 'val')
const prp = atom('prp', 'prp')
const atr = atom('atr', 'atr')

const element = <div id={val} prop:prp={prp} attr:atr={atr} />
const element = <div id={prp} attr:class={atr} />

mount(parent, element)

assert.is(element.id, 'val')
// @ts-expect-error `dunno` can't be inferred
assert.is(element.prp, 'prp')
assert.is(element.getAttribute('atr'), 'atr')
assert.is(element.id, 'prp')
assert.is(element.getAttribute('class'), 'atr')

val(ctx, 'val1')
prp(ctx, 'prp1')
atr(ctx, 'atr1')

assert.is(element.id, 'val1')
// @ts-expect-error `dunno` can't be inferred
assert.is(element.prp, 'prp1')
assert.is(element.getAttribute('atr'), 'atr1')
assert.is(element.id, 'prp1')
assert.is(element.getAttribute('class'), 'atr1')
}))

it('children updates', setup((ctx, h, hf, mount, parent) => {
Expand Down Expand Up @@ -390,12 +384,12 @@ it('same arguments in ref mount and unmount hooks', setup(async (ctx, h, hf, mou
assert.is(unmountArgs[1], component)
}))

it('css property and class attribute', setup(async (ctx, h, hf, mount, parent) => {
it('css property and className property', setup(async (ctx, h, hf, mount, parent) => {
const cls = 'class'
const css = 'color: red;'

const ref1 = (<div css={css} class={cls}></div>)
const ref2 = (<div class={cls} css={css}></div>)
const ref1 = (<div css={css} className={cls}></div>)
const ref2 = (<div className={cls} css={css}></div>)

const component = (
<div>
Expand Down Expand Up @@ -445,7 +439,7 @@ it('css custom property', setup(async (ctx, h, hf, mount, parent) => {
assert.is(component.style.getPropertyValue('--secondProperty'), '')
}))

it('class and className attribute', setup(async (ctx, h, hf, mount, parent) => {
it('class and className properties', setup(async (ctx, h, hf, mount, parent) => {
const classAtom = atom('' as string | undefined)

const ref1 = (<div class={classAtom}></div>)
Expand Down Expand Up @@ -473,8 +467,8 @@ it('class and className attribute', setup(async (ctx, h, hf, mount, parent) => {
classAtom(ctx, undefined)
assert.is(ref1.className, '')
assert.is(ref2.className, '')
assert.ok(!ref1.hasAttribute('class'))
assert.ok(!ref2.hasAttribute('class'))
assert.ok(ref1.hasAttribute('class'))
assert.ok(ref2.hasAttribute('class'))
}))

it('ref mount and unmount callbacks order', setup(async (ctx, h, hf, mount, parent) => {
Expand Down
79 changes: 50 additions & 29 deletions packages/jsx/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,21 @@ type DomApis = Pick<
'document' | 'Node' | 'Text' | 'Element' | 'MutationObserver' | 'HTMLElement' | 'DocumentFragment'
>

const propertiesAsAttribute = [
'width',
'height',
'href',
'list',
'form',
/** Default value in browsers is `-1` and an empty string is cast to `0` instead */
'tabIndex',
'download',
'rowSpan',
'colSpan',
'role',
'popover',
]

const isSkipped = (value: unknown): value is boolean | '' | null | undefined =>
typeof value === 'boolean' || value === '' || value == null

Expand Down Expand Up @@ -83,12 +98,21 @@ const walkLinkedList = (ctx: Ctx, el: JSX.Element, list: Atom<LinkedList<LLNode<
)
}

export const reatomJsx = (ctx: Ctx, DOM: DomApis = globalThis.window) => {
const StylesheetId = 'reatom-jsx-styles'
let styles: Rec<string> = {}
let stylesheet: HTMLStyleElement | undefined
export const reatomJsx = (ctx: Ctx, DOM: DomApis = globalThis.window, options?: {
/**
* Adds a style element containing styles from the `css` property to the document.
* @default true
*/
appendStylesheet?: boolean
}) => {
let name = ''

/** @see https://www.measurethat.net/Benchmarks/Show/11290 */
let styles = new Map<string, string>()
let stylesheet: HTMLStyleElement = DOM.document.createElement('style')
stylesheet.id = 'reatom-jsx-styles'
if (options?.appendStylesheet !== false) DOM.document.head.appendChild(stylesheet)

let set = (element: JSX.Element, key: string, val: any) => {
if (key.startsWith('on:')) {
key = key.slice(3)
Expand All @@ -100,16 +124,9 @@ export const reatomJsx = (ctx: Ctx, DOM: DomApis = globalThis.window) => {
if (val == null) element.style.removeProperty(key)
else element.style.setProperty(key, String(val))
} else if (key === 'css') {
stylesheet ??= DOM.document.getElementById(StylesheetId) as any
if (!stylesheet) {
stylesheet = DOM.document.createElement('style')
stylesheet.id = StylesheetId
DOM.document.head.appendChild(stylesheet)
}

let styleId = styles[val]
let styleId = styles.get(val)
if (!styleId) {
styleId = styles[val] = `${name ? name + '.' : ''}${random(0, 1e6).toString()}`
styles.set(val, styleId = `${name ? name + '.' : ''}${random(0, 1e6).toString()}`)
stylesheet.innerText += '[data-reatom="' + styleId + '"]{' + val + '}\n'
}
/** @see https://measurethat.net/Benchmarks/Show/11819 */
Expand All @@ -119,24 +136,28 @@ export const reatomJsx = (ctx: Ctx, DOM: DomApis = globalThis.window) => {
if (val[key] == null) element.style.removeProperty(key)
else element.style.setProperty(key, val[key])
}
} else if (key.startsWith('prop:')) {
// @ts-expect-error
element[key.slice(5)] = val
} else if (
!propertiesAsAttribute.includes(key)
&& element instanceof DOM.HTMLElement
&& (key in element || key === 'class')
) {
if (key === 'class') key = 'className'
// @ts-ignore
element[key] = val == null ? '' : val
} else {
if (key.startsWith('attr:')) {
key = key.slice(5)
}
if (key.startsWith('attr:')) key = key.slice(5)
if (key === 'className') key = 'class'
if (val == null || val === false) element.removeAttribute(key)
else {
val = val === true ? '' : String(val)
/**
* @see https://measurethat.net/Benchmarks/Show/54
* @see https://measurethat.net/Benchmarks/Show/31249
*/
if (key === 'class' && element instanceof HTMLElement) element.className = val
else element.setAttribute(key, val)
}

/**
* @note aria- and data- attributes have no boolean representation.
* A `false` value is different from the attribute not being
* present, so we can't remove it. For non-boolean aria
* attributes we could treat false as a removal, but the
* amount of exceptions would cost too many bytes. On top of
* that other frameworks generally stringify `false`.
*/
if (val == null || (val === false && key[4] !== '-')) element.removeAttribute(key)
else element.setAttribute(key, key == 'popover' && val == true ? '' : val)
}
}

Expand Down

0 comments on commit ab42221

Please sign in to comment.