feat: renderTriggered for forced updates
This commit is contained in:
parent
2ee60e0a03
commit
6027d480f3
@ -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'
|
||||||
|
@ -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(
|
||||||
|
@ -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
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user