import { FunctionalComponent, VNode, defineComponent, h, inject, ref, Ref } from 'vue'; export interface InputProps extends Object { modelValue: string | boolean; } const UPDATE_VALUE_EVENT = 'update:modelValue'; const MODEL_VALUE = 'modelValue'; const ROUTER_LINK_VALUE = 'routerLink'; const NAV_MANAGER = 'navManager'; const ROUTER_PROP_REFIX = 'router'; interface NavManager { navigate: (options: T) => void; } interface ComponentOptions { modelProp?: string; modelUpdateEvent?: string; routerLinkComponent?: boolean; } const getComponentClasses = (classes: unknown) => { return (classes as string)?.split(' ') || []; }; const getElementClasses = (ref: Ref, componentClasses: Set, defaultClasses: string[] = []) => { return [ ...Array.from(ref.value?.classList || []), ...defaultClasses ] .filter((c: string, i, self) => !componentClasses.has(c) && self.indexOf(c) === i); }; /** * Create a callback to define a Vue component wrapper around a Web Component. * * @prop name - The component tag name (i.e. `ion-button`) * @prop componentProps - An array of properties on the * component. These usually match up with the @Prop definitions * in each component's TSX file. * @prop componentOptions - An object that defines additional * options for the component such as router or v-model * integrations. */ export const defineContainer = (name: string, componentProps: string[] = [], componentOptions: ComponentOptions = {}) => { const { modelProp, modelUpdateEvent, routerLinkComponent } = componentOptions; /** * Create a Vue component wrapper around a Web Component. * Note: The `props` here are not all properties on a component. * They refer to whatever properties are set on an instance of a component. */ const Container: FunctionalComponent = defineComponent((props, { attrs, slots, emit }) => { const { modelValue, ...restOfProps } = props; const containerRef = ref(); const classes = new Set(getComponentClasses(attrs.class)); const onVnodeBeforeMount = (vnode: VNode) => { // Add a listener to tell Vue to update the v-model if (vnode.el) { vnode.el.addEventListener(modelUpdateEvent.toLowerCase(), (e: Event) => { emit(UPDATE_VALUE_EVENT, (e?.target as any)[modelProp]); }); } }; let finalProps: any = { ...restOfProps }; if (routerLinkComponent) { const navManager: NavManager = inject(NAV_MANAGER); const handleClick = (ev: Event) => { const routerProps = Object.keys(finalProps).filter(p => p.startsWith(ROUTER_PROP_REFIX)); if (routerProps.length === 0) return; let navigationPayload: any = { event: ev }; routerProps.forEach(prop => { navigationPayload[prop] = finalProps[prop]; }); navManager.navigate(navigationPayload); } if (finalProps.onClick) { const oldClick = finalProps.onClick; finalProps.onClick = (ev: Event) => { oldClick(ev); if (!ev.defaultPrevented) { handleClick(ev); } } } else { finalProps.onClick = handleClick; } } return () => { getComponentClasses(attrs.class).forEach(value => { classes.add(value); }); let propsToAdd = { ...finalProps, ref: containerRef, class: getElementClasses(containerRef, classes) }; if (modelUpdateEvent) { propsToAdd = { ...propsToAdd, onVnodeBeforeMount } } if (modelProp) { propsToAdd = { ...propsToAdd, [modelProp]: props.hasOwnProperty('modelValue') ? props.modelValue : (props as any)[modelProp] } } return h(name, propsToAdd, slots.default && slots.default()); } }); Container.displayName = name; Container.props = componentProps; if (modelProp) { Container.props.push(MODEL_VALUE); Container.emits = [UPDATE_VALUE_EVENT]; } if (routerLinkComponent) { Container.props.push(ROUTER_LINK_VALUE); } return Container; };