fix(transition): avoid invoking stale transition end callbacks

fix #2482
This commit is contained in:
Evan You 2020-12-02 12:15:45 -05:00
parent 5ad6ed3d9c
commit eaf8a67c72

View File

@ -126,11 +126,7 @@ export function resolveTransitionProps(
removeTransitionClass(el, isAppear ? appearFromClass : enterFromClass) removeTransitionClass(el, isAppear ? appearFromClass : enterFromClass)
addTransitionClass(el, isAppear ? appearToClass : enterToClass) addTransitionClass(el, isAppear ? appearToClass : enterToClass)
if (!(hook && hook.length > 1)) { if (!(hook && hook.length > 1)) {
if (enterDuration) { whenTransitionEnds(el, type, enterDuration, resolve)
setTimeout(resolve, enterDuration)
} else {
whenTransitionEnds(el, type, resolve)
}
} }
}) })
} }
@ -157,11 +153,7 @@ export function resolveTransitionProps(
removeTransitionClass(el, leaveFromClass) removeTransitionClass(el, leaveFromClass)
addTransitionClass(el, leaveToClass) addTransitionClass(el, leaveToClass)
if (!(onLeave && onLeave.length > 1)) { if (!(onLeave && onLeave.length > 1)) {
if (leaveDuration) { whenTransitionEnds(el, type, leaveDuration, resolve)
setTimeout(resolve, leaveDuration)
} else {
whenTransitionEnds(el, type, resolve)
}
} }
}) })
onLeave && onLeave(el, resolve) onLeave && onLeave(el, resolve)
@ -247,27 +239,39 @@ function nextFrame(cb: () => void) {
}) })
} }
let endId = 0
function whenTransitionEnds( function whenTransitionEnds(
el: Element, el: Element & { _endId?: number },
expectedType: TransitionProps['type'] | undefined, expectedType: TransitionProps['type'] | undefined,
cb: () => void explicitTimeout: number | null,
resolve: () => void
) { ) {
const id = (el._endId = ++endId)
const resolveIfNotStale = () => {
if (id === el._endId) {
resolve()
}
}
if (explicitTimeout) {
return setTimeout(resolveIfNotStale, explicitTimeout)
}
const { type, timeout, propCount } = getTransitionInfo(el, expectedType) const { type, timeout, propCount } = getTransitionInfo(el, expectedType)
if (!type) { if (!type) {
return cb() return resolve()
} }
const endEvent = type + 'end' const endEvent = type + 'end'
let ended = 0 let ended = 0
const end = () => { const end = () => {
el.removeEventListener(endEvent, onEnd) el.removeEventListener(endEvent, onEnd)
cb() resolveIfNotStale()
} }
const onEnd = (e: Event) => { const onEnd = (e: Event) => {
if (e.target === el) { if (e.target === el && ++ended >= propCount) {
if (++ended >= propCount) { end()
end()
}
} }
} }
setTimeout(() => { setTimeout(() => {