feat: renderTriggered for forced updates

This commit is contained in:
Evan You 2018-11-08 14:09:52 -05:00
parent 2ee60e0a03
commit 6027d480f3
4 changed files with 108 additions and 65 deletions

View File

@ -24,7 +24,7 @@ import {
DebuggerEvent DebuggerEvent
} from './autorun' } from './autorun'
export { Autorun, DebuggerEvent } export { Autorun, AutorunOptions, DebuggerEvent }
export { OperationTypes } from './operations' export { OperationTypes } from './operations'
export { computed, ComputedGetter } from './computed' export { computed, ComputedGetter } from './computed'
export { lock, unlock } from './lock' export { lock, unlock } from './lock'

View File

@ -21,7 +21,7 @@ import { createRenderProxy } from './componentProxy'
import { import {
handleError, handleError,
ErrorTypes, ErrorTypes,
callLifecycleHookWithHandle callLifecycleHookWithHandler
} from './errorHandling' } from './errorHandling'
import { warn } from './warning' import { warn } from './warning'
import { setCurrentInstance, unsetCurrentInstance } from './experimental/hooks' import { setCurrentInstance, unsetCurrentInstance } from './experimental/hooks'
@ -56,7 +56,7 @@ export function createComponentInstance<T extends Component>(
instance.$slots = currentVNode.slots || EMPTY_OBJ instance.$slots = currentVNode.slots || EMPTY_OBJ
if (created) { if (created) {
callLifecycleHookWithHandle(created, $proxy, ErrorTypes.CREATED) callLifecycleHookWithHandler(created, $proxy, ErrorTypes.CREATED)
} }
currentVNode = currentContextVNode = null currentVNode = currentContextVNode = null
@ -100,7 +100,7 @@ export function initializeComponentInstance(instance: ComponentInstance) {
// beforeCreate hook is called right in the constructor // beforeCreate hook is called right in the constructor
const { beforeCreate, props } = instance.$options const { beforeCreate, props } = instance.$options
if (beforeCreate) { if (beforeCreate) {
callLifecycleHookWithHandle(beforeCreate, proxy, ErrorTypes.BEFORE_CREATE) callLifecycleHookWithHandler(beforeCreate, proxy, ErrorTypes.BEFORE_CREATE)
} }
initializeProps(instance, props, (currentVNode as VNode).data) initializeProps(instance, props, (currentVNode as VNode).data)
} }
@ -222,18 +222,24 @@ export function shouldUpdateComponent(
if (nextProps === null) { if (nextProps === null) {
return prevProps !== null return prevProps !== null
} }
let shouldUpdate = true
const nextKeys = Object.keys(nextProps) const nextKeys = Object.keys(nextProps)
if (nextKeys.length === Object.keys(prevProps).length) { if (nextKeys.length !== Object.keys(prevProps).length) {
shouldUpdate = false return true
for (let i = 0; i < nextKeys.length; i++) { }
const key = nextKeys[i] for (let i = 0; i < nextKeys.length; i++) {
if (nextProps[key] !== prevProps[key]) { const key = nextKeys[i]
shouldUpdate = true if (nextProps[key] !== prevProps[key]) {
} return true
} }
} }
return shouldUpdate return false
}
export function getReasonForComponentUpdate(
prevVNode: VNode,
nextVNode: VNode
): any {
// TODO: extract more detailed information on why the component is updating
} }
export function createComponentClassFromOptions( export function createComponentClassFromOptions(

View File

@ -1,4 +1,10 @@
import { autorun, stop, Autorun, immutable } from '@vue/observer' import {
autorun,
stop,
Autorun,
immutable,
AutorunOptions
} from '@vue/observer'
import { queueJob, handleSchedulerError, nextTick } from '@vue/scheduler' import { queueJob, handleSchedulerError, nextTick } from '@vue/scheduler'
import { VNodeFlags, ChildrenFlags } from './flags' import { VNodeFlags, ChildrenFlags } from './flags'
import { EMPTY_OBJ, reservedPropRE, isString } from '@vue/shared' import { EMPTY_OBJ, reservedPropRE, isString } from '@vue/shared'
@ -15,7 +21,8 @@ import {
renderFunctionalRoot, renderFunctionalRoot,
createComponentInstance, createComponentInstance,
teardownComponentInstance, teardownComponentInstance,
shouldUpdateComponent shouldUpdateComponent,
getReasonForComponentUpdate
} from './componentUtils' } from './componentUtils'
import { KeepAliveSymbol } from './optional/keepAlive' import { KeepAliveSymbol } from './optional/keepAlive'
import { pushWarningContext, popWarningContext, warn } from './warning' import { pushWarningContext, popWarningContext, warn } from './warning'
@ -23,7 +30,7 @@ import { resolveProps } from './componentProps'
import { import {
handleError, handleError,
ErrorTypes, ErrorTypes,
callLifecycleHookWithHandle callLifecycleHookWithHandler
} from './errorHandling' } from './errorHandling'
export interface NodeOps { export interface NodeOps {
@ -561,6 +568,14 @@ export function createRenderer(options: RendererOptions) {
instance.$parentVNode = nextVNode as MountedVNode instance.$parentVNode = nextVNode as MountedVNode
if (shouldUpdateComponent(prevVNode, nextVNode)) { if (shouldUpdateComponent(prevVNode, nextVNode)) {
if (__DEV__ && instance.$options.renderTriggered) {
callLifecycleHookWithHandler(
instance.$options.renderTriggered,
instance.$proxy,
ErrorTypes.RENDER_TRIGGERED,
getReasonForComponentUpdate(prevVNode, nextVNode)
)
}
instance.$forceUpdate() instance.$forceUpdate()
} else if (instance.$vnode.flags & VNodeFlags.COMPONENT) { } else if (instance.$vnode.flags & VNodeFlags.COMPONENT) {
instance.$vnode.contextVNode = nextVNode instance.$vnode.contextVNode = nextVNode
@ -1194,58 +1209,72 @@ export function createRenderer(options: RendererOptions) {
} = instance } = instance
if (beforeMount) { if (beforeMount) {
callLifecycleHookWithHandle(beforeMount, $proxy, ErrorTypes.BEFORE_MOUNT) callLifecycleHookWithHandler(beforeMount, $proxy, ErrorTypes.BEFORE_MOUNT)
} }
const queueUpdate = (instance.$forceUpdate = () => { const queueUpdate = (instance.$forceUpdate = () => {
queueJob(instance._updateHandle, flushHooks) queueJob(instance._updateHandle, flushHooks)
}) })
instance._updateHandle = autorun( const autorunOptions: AutorunOptions = {
() => { scheduler: queueUpdate
if (instance._unmounted) { }
return
}
if (instance._mounted) {
updateComponentInstance(instance, isSVG)
} else {
// this will be executed synchronously right here
instance.$vnode = renderInstanceRoot(instance) as MountedVNode
queuePostCommitHook(() => { if (__DEV__) {
vnode.el = instance.$vnode.el if (renderTracked) {
if (__COMPAT__) { autorunOptions.onTrack = event => {
// expose __vue__ for devtools callLifecycleHookWithHandler(
;(vnode.el as any).__vue__ = instance renderTracked,
} $proxy,
if (vnode.ref) { ErrorTypes.RENDER_TRACKED,
vnode.ref($proxy) event
}
// retrieve mounted value after initial render so that we get
// to inject effects in hooks
const { mounted } = instance.$options
if (mounted) {
callLifecycleHookWithHandle(mounted, $proxy, ErrorTypes.MOUNTED)
}
})
mount(
instance.$vnode,
container,
vnode as MountedVNode,
isSVG,
endNode
) )
instance._mounted = true
} }
},
{
scheduler: queueUpdate,
onTrack: renderTracked,
onTrigger: renderTriggered
} }
) if (renderTriggered) {
autorunOptions.onTrigger = event => {
callLifecycleHookWithHandler(
renderTriggered,
$proxy,
ErrorTypes.RENDER_TRIGGERED,
event
)
}
}
}
instance._updateHandle = autorun(() => {
if (instance._unmounted) {
return
}
if (instance._mounted) {
updateComponentInstance(instance, isSVG)
} else {
// this will be executed synchronously right here
instance.$vnode = renderInstanceRoot(instance) as MountedVNode
queuePostCommitHook(() => {
vnode.el = instance.$vnode.el
if (__COMPAT__) {
// expose __vue__ for devtools
;(vnode.el as any).__vue__ = instance
}
if (vnode.ref) {
vnode.ref($proxy)
}
// retrieve mounted value after initial render so that we get
// to inject effects in hooks
const { mounted } = instance.$options
if (mounted) {
callLifecycleHookWithHandler(mounted, $proxy, ErrorTypes.MOUNTED)
}
})
mount(instance.$vnode, container, vnode as MountedVNode, isSVG, endNode)
instance._mounted = true
}
}, autorunOptions)
if (__DEV__) { if (__DEV__) {
popWarningContext() popWarningContext()
@ -1267,7 +1296,7 @@ export function createRenderer(options: RendererOptions) {
$options: { beforeUpdate } $options: { beforeUpdate }
} = instance } = instance
if (beforeUpdate) { if (beforeUpdate) {
callLifecycleHookWithHandle( callLifecycleHookWithHandler(
beforeUpdate, beforeUpdate,
$proxy, $proxy,
ErrorTypes.BEFORE_UPDATE, ErrorTypes.BEFORE_UPDATE,
@ -1297,7 +1326,7 @@ export function createRenderer(options: RendererOptions) {
} }
const { updated } = instance.$options const { updated } = instance.$options
if (updated) { if (updated) {
callLifecycleHookWithHandle( callLifecycleHookWithHandler(
updated, updated,
$proxy, $proxy,
ErrorTypes.UPDATED, ErrorTypes.UPDATED,
@ -1334,7 +1363,7 @@ export function createRenderer(options: RendererOptions) {
$options: { beforeUnmount, unmounted } $options: { beforeUnmount, unmounted }
} = instance } = instance
if (beforeUnmount) { if (beforeUnmount) {
callLifecycleHookWithHandle( callLifecycleHookWithHandler(
beforeUnmount, beforeUnmount,
$proxy, $proxy,
ErrorTypes.BEFORE_UNMOUNT ErrorTypes.BEFORE_UNMOUNT
@ -1347,7 +1376,7 @@ export function createRenderer(options: RendererOptions) {
teardownComponentInstance(instance) teardownComponentInstance(instance)
instance._unmounted = true instance._unmounted = true
if (unmounted) { if (unmounted) {
callLifecycleHookWithHandle(unmounted, $proxy, ErrorTypes.UNMOUNTED) callLifecycleHookWithHandler(unmounted, $proxy, ErrorTypes.UNMOUNTED)
} }
} }
@ -1391,7 +1420,7 @@ export function createRenderer(options: RendererOptions) {
callActivatedHook($children[i], false) callActivatedHook($children[i], false)
} }
if (activated) { if (activated) {
callLifecycleHookWithHandle(activated, $proxy, ErrorTypes.ACTIVATED) callLifecycleHookWithHandler(activated, $proxy, ErrorTypes.ACTIVATED)
} }
} }
} }
@ -1416,7 +1445,11 @@ export function createRenderer(options: RendererOptions) {
callDeactivateHook($children[i], false) callDeactivateHook($children[i], false)
} }
if (deactivated) { if (deactivated) {
callLifecycleHookWithHandle(deactivated, $proxy, ErrorTypes.DEACTIVATED) callLifecycleHookWithHandler(
deactivated,
$proxy,
ErrorTypes.DEACTIVATED
)
} }
} }
} }

View File

@ -16,6 +16,8 @@ export const enum ErrorTypes {
DEACTIVATED, DEACTIVATED,
ERROR_CAPTURED, ERROR_CAPTURED,
RENDER, RENDER,
RENDER_TRACKED,
RENDER_TRIGGERED,
WATCH_CALLBACK, WATCH_CALLBACK,
NATIVE_EVENT_HANDLER, NATIVE_EVENT_HANDLER,
COMPONENT_EVENT_HANDLER, COMPONENT_EVENT_HANDLER,
@ -35,6 +37,8 @@ const ErrorTypeStrings: Record<number, string> = {
[ErrorTypes.DEACTIVATED]: 'in deactivated lifecycle hook', [ErrorTypes.DEACTIVATED]: 'in deactivated lifecycle hook',
[ErrorTypes.ERROR_CAPTURED]: 'in errorCaptured lifecycle hook', [ErrorTypes.ERROR_CAPTURED]: 'in errorCaptured lifecycle hook',
[ErrorTypes.RENDER]: 'in render function', [ErrorTypes.RENDER]: 'in render function',
[ErrorTypes.RENDER_TRACKED]: 'in renderTracked debug hook',
[ErrorTypes.RENDER_TRIGGERED]: 'in renderTriggered debug hook',
[ErrorTypes.WATCH_CALLBACK]: 'in watcher callback', [ErrorTypes.WATCH_CALLBACK]: 'in watcher callback',
[ErrorTypes.NATIVE_EVENT_HANDLER]: 'in native event handler', [ErrorTypes.NATIVE_EVENT_HANDLER]: 'in native event handler',
[ErrorTypes.COMPONENT_EVENT_HANDLER]: 'in component event handler', [ErrorTypes.COMPONENT_EVENT_HANDLER]: 'in component event handler',
@ -42,7 +46,7 @@ const ErrorTypeStrings: Record<number, string> = {
'when flushing updates. This may be a Vue internals bug.' 'when flushing updates. This may be a Vue internals bug.'
} }
export function callLifecycleHookWithHandle( export function callLifecycleHookWithHandler(
hook: Function, hook: Function,
instanceProxy: ComponentInstance, instanceProxy: ComponentInstance,
type: ErrorTypes, type: ErrorTypes,