-
-
Notifications
You must be signed in to change notification settings - Fork 8.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
perf(runtime-vapor): refactor apiCreateIf #12629
base: vapor
Are you sure you want to change the base?
Conversation
In the previous implementation, reactive data has an update boundary for each Considering this case <h1 v-if="msg">{{ msg }}</h1>
<span v-else-if="foo">foo</span>
<span v-else-if="foo1">foo1</span>
<span v-else-if="foo2">foo2</span>
<span v-else-if="foo3">foo3</span>
<span v-else-if="foo4">foo4</span>
<span v-else>bar</span> If <span v-else-if="foo4">foo4</span>
<span v-else>bar</span> but now we need re-calc the whole v-if-else chain in this PR. |
import { renderEffect } from './renderEffect' | ||
|
||
export function createIf( | ||
condition: () => any, | ||
b1: BlockFn, | ||
b2?: BlockFn, | ||
ifBlockGetter: () => BlockFn | undefined, | ||
once?: boolean, | ||
// hydrationNode?: Node, | ||
): DynamicFragment { | ||
const frag = __DEV__ ? new DynamicFragment('if') : new DynamicFragment() | ||
if (once) { | ||
frag.update(condition() ? b1 : b2) | ||
frag.update(ifBlockGetter()) | ||
} else { | ||
renderEffect(() => frag.update(condition() ? b1 : b2)) | ||
renderEffect(() => frag.update(ifBlockGetter())) | ||
} | ||
return frag | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe we can use a table-driven approach. Add a feedback parameter, and change the ifBlockGetter part to get an array.
Like this:
export function createIf(
ifBlockGetter: [when: () => any, block: BlockFn][]
fallback?: BlockFn,
once?: boolean
)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This PR provides a very simple way, and I approve of it.
I checked and found that currently solid is also using the method of recomputing the entire chain each time.
Maybe we don't need to add extra complexity due to the performance hit of recomputing the entire if-else-if chain.
createIf
twice, eachv-else-if
generates a newcreateIf
callcreateIf
only once, reduce overheadanalysis
Initial Render
Before
createIf
internally callrenderEffect
and create a newDynamicFragment
instanceAfter
DynamicFragment
instancerenderEffect
callRe-Render(Update)
Before
createIf
structure ensures only necessary conditions are evaluatedDynamicFragment
renderEffect
for dynamic contentEffectScope
stop/create when branches switchAfter
DynamicFragment
needs to be maintainedrenderEffect
Trade-offs:
benchmark
v-else-if
branch (all branches will be evaluated)v-else-if
branch (only up to the third branch will be evaluated)