import { EMPTY_OBJ, NOOP } from './utils' import { VNode, Slots, RenderNode, MountedVNode } from './vdom' import { Data, ComponentOptions, ComponentClassOptions, ComponentPropsOptions, WatchOptions } from './componentOptions' import { setupWatcher } from './componentWatch' import { Autorun, DebuggerEvent, ComputedGetter } from '@vue/observer' import { nextTick } from '@vue/scheduler' import { ErrorTypes } from './errorHandling' import { initializeComponentInstance } from './componentUtils' // public component instance type export interface Component

extends PublicInstanceMethods { readonly $el: any readonly $vnode: MountedVNode readonly $parentVNode: MountedVNode readonly $data: D readonly $props: Readonly

readonly $attrs: Readonly readonly $slots: Slots readonly $root: Component readonly $parent: Component readonly $children: Component[] readonly $options: ComponentOptions readonly $refs: Record readonly $proxy: this } interface PublicInstanceMethods { $forceUpdate(): void $nextTick(fn: () => any): Promise $watch( keyOrFn: string | ((this: this) => any), cb: (this: this, newValue: any, oldValue: any) => void, options?: WatchOptions ): () => void $on(event: string, fn: Function): this $once(event: string, fn: Function): this $off(event?: string, fn?: Function): this $emit(name: string, ...payload: any[]): this } interface APIMethods { data?(): Partial render(props: Readonly

, slots: Slots, attrs: Data, parentVNode: VNode): any } interface LifecycleMethods { beforeCreate?(): void created?(): void beforeMount?(): void mounted?(): void beforeUpdate?(vnode: VNode): void updated?(vnode: VNode): void beforeUnmount?(): void unmounted?(): void errorCaptured?(): ( err: Error, type: ErrorTypes, instance: ComponentInstance | null, vnode: VNode ) => boolean | void activated?(): void deactivated?(): void renderTracked?(e: DebuggerEvent): void renderTriggered?(e: DebuggerEvent): void } export interface ComponentClass extends ComponentClassOptions { options?: ComponentOptions new

(): Component & D & P } export interface FunctionalComponent

{ (props: P, slots: Slots, attrs: Data, parentVNode: VNode): any pure?: boolean props?: ComponentPropsOptions

displayName?: string } export type ComponentType = ComponentClass | FunctionalComponent // Internal type that represents a mounted instance. // It extends InternalComponent with mounted instance properties. export interface ComponentInstance

extends InternalComponent, APIMethods, LifecycleMethods { constructor: ComponentClass $vnode: MountedVNode $data: D $props: P $attrs: Data $slots: Slots $root: ComponentInstance $children: ComponentInstance[] $options: ComponentOptions _updateHandle: Autorun _queueJob: ((fn: () => void) => void) _self: ComponentInstance // on proxies only } // actual implementation of the component class InternalComponent implements PublicInstanceMethods { get $el(): any { return this.$vnode && this.$vnode.el } $vnode: VNode | null = null $parentVNode: VNode | null = null $data: Data | null = null $props: Data | null = null $attrs: Data | null = null $slots: Slots | null = null $root: ComponentInstance | null = null $parent: ComponentInstance | null = null $children: ComponentInstance[] = [] $options: ComponentOptions | null = null $refs: Record = {} $proxy: any = null _rawData: Data | null = null _computedGetters: Record | null = null _watchHandles: Set | null = null _mounted: boolean = false _unmounted: boolean = false _events: { [event: string]: Function[] | null } | null = null _updateHandle: Autorun | null = null _queueJob: ((fn: () => void) => void) | null = null _isVue: boolean = true _inactiveRoot: boolean = false constructor() { initializeComponentInstance(this as any) } // to be set by renderer during mount $forceUpdate: () => void = NOOP $nextTick(fn: () => any): Promise { return nextTick(fn) } $watch( keyOrFn: string | ((this: this) => any), cb: (this: this, newValue: any, oldValue: any) => void, options?: WatchOptions ): () => void { return setupWatcher(this as any, keyOrFn, cb, options) } // eventEmitter interface $on(event: string, fn: Function): this { if (Array.isArray(event)) { for (let i = 0; i < event.length; i++) { this.$on(event[i], fn) } } else { const events = this._events || (this._events = Object.create(null)) ;(events[event] || (events[event] = [])).push(fn) } return this } $once(event: string, fn: Function): this { const onceFn = (...args: any[]) => { this.$off(event, onceFn) fn.apply(this, args) } ;(onceFn as any).fn = fn return this.$on(event, onceFn) } $off(event?: string, fn?: Function): this { if (this._events) { if (!event && !fn) { this._events = null } else if (Array.isArray(event)) { for (let i = 0; i < event.length; i++) { this.$off(event[i], fn) } } else if (!fn) { this._events[event as string] = null } else { const fns = this._events[event as string] if (fns) { for (let i = 0; i < fns.length; i++) { const f = fns[i] if (fn === f || fn === (f as any).fn) { fns.splice(i, 1) break } } } } } return this } $emit(name: string, ...payload: any[]): this { const parentData = (this.$parentVNode && this.$parentVNode.data) || EMPTY_OBJ const parentListener = parentData['on' + name] || parentData['on' + name.toLowerCase()] if (parentListener) { invokeListeners(parentListener, payload) } if (this._events) { const handlers = this._events[name] if (handlers) { invokeListeners(handlers, payload) } } return this } } function invokeListeners(value: Function | Function[], payload: any[]) { // TODO handle error if (Array.isArray(value)) { for (let i = 0; i < value.length; i++) { value[i](...payload) } } else { value(...payload) } } // the exported Component has the implementation details of the actual // InternalComponent class but with proper type inference of ComponentClass. export const Component = InternalComponent as ComponentClass