Skip to content

Commit af838c1

Browse files
authoredAug 3, 2024··
feat(custom-element): support for expose on customElement (#6256)
close #5540
1 parent 5a1a89b commit af838c1

File tree

2 files changed

+82
-0
lines changed

2 files changed

+82
-0
lines changed
 

‎packages/runtime-dom/__tests__/customElement.spec.ts

+61
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
import type { MockedFunction } from 'vitest'
12
import {
23
type Ref,
34
type VueElement,
@@ -881,4 +882,64 @@ describe('defineCustomElement', () => {
881882
expect(style.textContent).toBe(`div { color: red; }`)
882883
})
883884
})
885+
886+
describe('expose', () => {
887+
test('expose attributes and callback', async () => {
888+
type SetValue = (value: string) => void
889+
let fn: MockedFunction<SetValue>
890+
891+
const E = defineCustomElement({
892+
setup(_, { expose }) {
893+
const value = ref('hello')
894+
895+
const setValue = (fn = vi.fn((_value: string) => {
896+
value.value = _value
897+
}))
898+
899+
expose({
900+
setValue,
901+
value,
902+
})
903+
904+
return () => h('div', null, [value.value])
905+
},
906+
})
907+
customElements.define('my-el-expose', E)
908+
909+
container.innerHTML = `<my-el-expose></my-el-expose>`
910+
const e = container.childNodes[0] as VueElement & {
911+
value: string
912+
setValue: MockedFunction<SetValue>
913+
}
914+
expect(e.shadowRoot!.innerHTML).toBe(`<div>hello</div>`)
915+
expect(e.value).toBe('hello')
916+
expect(e.setValue).toBe(fn!)
917+
e.setValue('world')
918+
expect(e.value).toBe('world')
919+
await nextTick()
920+
expect(e.shadowRoot!.innerHTML).toBe(`<div>world</div>`)
921+
})
922+
923+
test('warning when exposing an existing property', () => {
924+
const E = defineCustomElement({
925+
props: {
926+
value: String,
927+
},
928+
setup(props, { expose }) {
929+
expose({
930+
value: 'hello',
931+
})
932+
933+
return () => h('div', null, [props.value])
934+
},
935+
})
936+
customElements.define('my-el-expose-two', E)
937+
938+
container.innerHTML = `<my-el-expose-two value="world"></my-el-expose-two>`
939+
940+
expect(
941+
`[Vue warn]: Exposed property "value" already exists on custom element.`,
942+
).toHaveBeenWarned()
943+
})
944+
})
884945
})

‎packages/runtime-dom/src/apiCustomElement.ts

+21
Original file line numberDiff line numberDiff line change
@@ -26,11 +26,13 @@ import {
2626
defineComponent,
2727
getCurrentInstance,
2828
nextTick,
29+
unref,
2930
warn,
3031
} from '@vue/runtime-core'
3132
import {
3233
camelize,
3334
extend,
35+
hasOwn,
3436
hyphenate,
3537
isArray,
3638
isPlainObject,
@@ -308,6 +310,9 @@ export class VueElement extends BaseClass {
308310

309311
// initial render
310312
this._update()
313+
314+
// apply expose
315+
this._applyExpose()
311316
}
312317

313318
const asyncDef = (this._def as ComponentOptions).__asyncLoader
@@ -342,6 +347,22 @@ export class VueElement extends BaseClass {
342347
}
343348
}
344349

350+
private _applyExpose() {
351+
const exposed = this._instance && this._instance.exposed
352+
if (!exposed) return
353+
for (const key in exposed) {
354+
if (!hasOwn(this, key)) {
355+
// exposed properties are readonly
356+
Object.defineProperty(this, key, {
357+
// unwrap ref to be consistent with public instance behavior
358+
get: () => unref(exposed[key]),
359+
})
360+
} else if (__DEV__) {
361+
warn(`Exposed property "${key}" already exists on custom element.`)
362+
}
363+
}
364+
}
365+
345366
protected _setAttr(key: string) {
346367
if (key.startsWith('data-v-')) return
347368
let value = this.hasAttribute(key) ? this.getAttribute(key) : undefined

0 commit comments

Comments
 (0)
Please sign in to comment.