feat: custom element reflection, casting and edge cases
This commit is contained in:
parent
bf4893c17c
commit
00f0b3c465
@ -99,13 +99,66 @@ describe('defineCustomElement', () => {
|
|||||||
container.appendChild(e)
|
container.appendChild(e)
|
||||||
expect(e.shadowRoot!.innerHTML).toBe('<div>one</div><div>two</div>')
|
expect(e.shadowRoot!.innerHTML).toBe('<div>one</div><div>two</div>')
|
||||||
|
|
||||||
|
// reflect
|
||||||
|
// should reflect primitive value
|
||||||
|
expect(e.getAttribute('foo')).toBe('one')
|
||||||
|
// should not reflect rich data
|
||||||
|
expect(e.hasAttribute('bar')).toBe(false)
|
||||||
|
|
||||||
e.foo = 'three'
|
e.foo = 'three'
|
||||||
await nextTick()
|
await nextTick()
|
||||||
expect(e.shadowRoot!.innerHTML).toBe('<div>three</div><div>two</div>')
|
expect(e.shadowRoot!.innerHTML).toBe('<div>three</div><div>two</div>')
|
||||||
|
expect(e.getAttribute('foo')).toBe('three')
|
||||||
|
|
||||||
|
e.foo = null
|
||||||
|
await nextTick()
|
||||||
|
expect(e.shadowRoot!.innerHTML).toBe('<div></div><div>two</div>')
|
||||||
|
expect(e.hasAttribute('foo')).toBe(false)
|
||||||
|
|
||||||
e.bazQux = 'four'
|
e.bazQux = 'four'
|
||||||
await nextTick()
|
await nextTick()
|
||||||
expect(e.shadowRoot!.innerHTML).toBe('<div>three</div><div>four</div>')
|
expect(e.shadowRoot!.innerHTML).toBe('<div></div><div>four</div>')
|
||||||
|
expect(e.getAttribute('baz-qux')).toBe('four')
|
||||||
|
})
|
||||||
|
|
||||||
|
test('attribute -> prop type casting', async () => {
|
||||||
|
const E = defineCustomElement({
|
||||||
|
props: {
|
||||||
|
foo: Number,
|
||||||
|
bar: Boolean
|
||||||
|
},
|
||||||
|
render() {
|
||||||
|
return [this.foo, typeof this.foo, this.bar, typeof this.bar].join(
|
||||||
|
' '
|
||||||
|
)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
customElements.define('my-el-props-cast', E)
|
||||||
|
container.innerHTML = `<my-el-props-cast foo="1"></my-el-props-cast>`
|
||||||
|
const e = container.childNodes[0] as VueElement
|
||||||
|
expect(e.shadowRoot!.innerHTML).toBe(`1 number false boolean`)
|
||||||
|
|
||||||
|
e.setAttribute('bar', '')
|
||||||
|
await nextTick()
|
||||||
|
expect(e.shadowRoot!.innerHTML).toBe(`1 number true boolean`)
|
||||||
|
|
||||||
|
e.setAttribute('foo', '2e1')
|
||||||
|
await nextTick()
|
||||||
|
expect(e.shadowRoot!.innerHTML).toBe(`20 number true boolean`)
|
||||||
|
})
|
||||||
|
|
||||||
|
test('handling properties set before upgrading', () => {
|
||||||
|
const E = defineCustomElement({
|
||||||
|
props: ['foo'],
|
||||||
|
render() {
|
||||||
|
return `foo: ${this.foo}`
|
||||||
|
}
|
||||||
|
})
|
||||||
|
const el = document.createElement('my-el-upgrade') as any
|
||||||
|
el.foo = 'hello'
|
||||||
|
container.appendChild(el)
|
||||||
|
customElements.define('my-el-upgrade', E)
|
||||||
|
expect(el.shadowRoot.innerHTML).toBe(`foo: hello`)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -20,7 +20,7 @@ import {
|
|||||||
nextTick,
|
nextTick,
|
||||||
warn
|
warn
|
||||||
} from '@vue/runtime-core'
|
} from '@vue/runtime-core'
|
||||||
import { camelize, hyphenate, isArray } from '@vue/shared'
|
import { camelize, extend, hyphenate, isArray, toNumber } from '@vue/shared'
|
||||||
import { hydrate, render } from '.'
|
import { hydrate, render } from '.'
|
||||||
|
|
||||||
type VueElementConstructor<P = {}> = {
|
type VueElementConstructor<P = {}> = {
|
||||||
@ -134,7 +134,7 @@ export function defineCustomElement(
|
|||||||
return attrKeys
|
return attrKeys
|
||||||
}
|
}
|
||||||
constructor() {
|
constructor() {
|
||||||
super(Comp, attrKeys, hydate)
|
super(Comp, attrKeys, propKeys, hydate)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -173,12 +173,13 @@ export class VueElement extends HTMLElement {
|
|||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private _def: Component,
|
private _def: Component,
|
||||||
private _attrs: string[],
|
private _attrKeys: string[],
|
||||||
|
private _propKeys: string[],
|
||||||
hydrate?: RootHydrateFunction
|
hydrate?: RootHydrateFunction
|
||||||
) {
|
) {
|
||||||
super()
|
super()
|
||||||
if (this.shadowRoot && hydrate) {
|
if (this.shadowRoot && hydrate) {
|
||||||
hydrate(this._initVNode(), this.shadowRoot)
|
hydrate(this._createVNode(), this.shadowRoot)
|
||||||
} else {
|
} else {
|
||||||
if (__DEV__ && this.shadowRoot) {
|
if (__DEV__ && this.shadowRoot) {
|
||||||
warn(
|
warn(
|
||||||
@ -191,15 +192,23 @@ export class VueElement extends HTMLElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
attributeChangedCallback(name: string, _oldValue: string, newValue: string) {
|
attributeChangedCallback(name: string, _oldValue: string, newValue: string) {
|
||||||
if (this._attrs.includes(name)) {
|
if (this._attrKeys.includes(name)) {
|
||||||
this._setProp(camelize(name), newValue)
|
this._setProp(camelize(name), toNumber(newValue), false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
connectedCallback() {
|
connectedCallback() {
|
||||||
this._connected = true
|
this._connected = true
|
||||||
if (!this._instance) {
|
if (!this._instance) {
|
||||||
render(this._initVNode(), this.shadowRoot!)
|
// check if there are props set pre-upgrade
|
||||||
|
for (const key of this._propKeys) {
|
||||||
|
if (this.hasOwnProperty(key)) {
|
||||||
|
const value = (this as any)[key]
|
||||||
|
delete (this as any)[key]
|
||||||
|
this._setProp(key, value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
render(this._createVNode(), this.shadowRoot!)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -213,41 +222,61 @@ export class VueElement extends HTMLElement {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
protected _getProp(key: string) {
|
protected _getProp(key: string) {
|
||||||
return this._props[key]
|
return this._props[key]
|
||||||
}
|
}
|
||||||
|
|
||||||
protected _setProp(key: string, val: any) {
|
/**
|
||||||
const oldValue = this._props[key]
|
* @internal
|
||||||
this._props[key] = val
|
*/
|
||||||
if (this._instance && val !== oldValue) {
|
protected _setProp(key: string, val: any, shouldReflect = true) {
|
||||||
this._instance.props[key] = val
|
if (val !== this._props[key]) {
|
||||||
|
this._props[key] = val
|
||||||
|
if (this._instance) {
|
||||||
|
render(this._createVNode(), this.shadowRoot!)
|
||||||
|
}
|
||||||
|
// reflect
|
||||||
|
if (shouldReflect) {
|
||||||
|
if (val === true) {
|
||||||
|
this.setAttribute(hyphenate(key), '')
|
||||||
|
} else if (typeof val === 'string' || typeof val === 'number') {
|
||||||
|
this.setAttribute(hyphenate(key), val + '')
|
||||||
|
} else if (!val) {
|
||||||
|
this.removeAttribute(hyphenate(key))
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected _initVNode(): VNode<any, any> {
|
private _createVNode(): VNode<any, any> {
|
||||||
const vnode = createVNode(this._def, this._props)
|
const vnode = createVNode(this._def, extend({}, this._props))
|
||||||
vnode.ce = instance => {
|
if (!this._instance) {
|
||||||
this._instance = instance
|
vnode.ce = instance => {
|
||||||
instance.isCE = true
|
this._instance = instance
|
||||||
|
instance.isCE = true
|
||||||
|
|
||||||
// intercept emit
|
// intercept emit
|
||||||
instance.emit = (event: string, ...args: any[]) => {
|
instance.emit = (event: string, ...args: any[]) => {
|
||||||
this.dispatchEvent(
|
this.dispatchEvent(
|
||||||
new CustomEvent(event, {
|
new CustomEvent(event, {
|
||||||
detail: args
|
detail: args
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// locate nearest Vue custom element parent for provide/inject
|
// locate nearest Vue custom element parent for provide/inject
|
||||||
let parent: Node | null = this
|
let parent: Node | null = this
|
||||||
while (
|
while (
|
||||||
(parent = parent && (parent.parentNode || (parent as ShadowRoot).host))
|
(parent =
|
||||||
) {
|
parent && (parent.parentNode || (parent as ShadowRoot).host))
|
||||||
if (parent instanceof VueElement) {
|
) {
|
||||||
instance.parent = parent._instance
|
if (parent instanceof VueElement) {
|
||||||
break
|
instance.parent = parent._instance
|
||||||
|
break
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user