-
-
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
fix(runtime-dom): map kebab attributes to camelCase props (fix #13055) #13085
base: main
Are you sure you want to change the base?
Conversation
Size ReportBundles
Usages
|
@vue/compiler-core
@vue/compiler-dom
@vue/compiler-ssr
@vue/compiler-sfc
@vue/reactivity
@vue/runtime-core
@vue/runtime-dom
@vue/server-renderer
@vue/shared
vue
@vue/compat
commit: |
/ecosystem-ci run |
📝 Ran ecosystem CI: Open
|
@@ -140,5 +142,5 @@ function shouldSetAsProp( | |||
return false | |||
} | |||
|
|||
return key in el | |||
return camelize(key) in el || key in el |
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 cache camelKey on el like this:
const camelKey = camelize(key)
if (camelKey in el) {
;(el as any)[`$${key}`] = camelKey
return true
}
patchProp
changes as follows
patchDOMProp(el, el[`$${key}`] || key, nextValue, parentComponent)
Call camelize
only once, check in
only once
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.
Yes, we can cache the check result in the element itself. I would just like to point that there are 2 situations where we need to perform the check: for .prop
bindings and at the end shouldSetAsProp
.
For .prop
bindings, check would always be performed and only once.
But if we are coming from shouldSetAsProp
, to also avoid performing the in
check when we don't need to, I would also need to add the $${key}
property on the places where shouldSetAsProp
would return true - or alternatively set its default value in the top of the function.
I have a question though: is it possible that el
would already have a property with the name we are setting ($${key}
)? Should we actually add a check for that and what to do if it already exists?
function shouldSetAsProp(...) {
...
const camelKey = camelize(key)
if (camelKey in el) {
if ((el as any)[`$${key}`] && (el as any)[`$${key}`] !== camelKey) {
// What to do? Throw?
}
;(el as any)[`$${key}`] = camelKey
return true
}
...
}
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.
We could use a symbol
as key too... either to set directly the value of the current prop being set (would get overwritten by every new attribute being set and not cached across renders) or use it to create an object where properties could then be referenced by their specific key and thus caching the results. Or we leave it as you suggested above. Thoughts?
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.
Should we actually add a check for that and what to do if it already exists?
We just need to use a key that is as unlikely to duplicate as possible, such as $_v_prop_${key}
or symbol as key.
Description
This PR fixes an issue where kebab-case attributes are not correctly being set as DOM properties.
The problem is happening because there is no conversion from kebab-case to camelCase in the patchProp function when checking if a property exists on the given element when using the
in
operator.The problem also happens when using the
.prop
modifier, since thepatchDOMProp
function is called with the givenkey
argument, without checking first for the existence of the camelCase DOM property:core/packages/runtime-dom/src/patchProp.ts
Line 50 in e16e9a7
The changes in this PR addresses the issue in these two places, by checking first if the camelCase version of the
key
exists in the Element and using the originalkey
argument as fallback.Fixes
Notes
Because of how the current version of
patchProp
andshouldSetAsProp
are structured, it was not possible to avoid making the check with thein
operator twice for the cases whereshouldSetAsProp
was called and executed entirely (or at least I couldn't figure out a way).I understand this is performance sensitive part of the code so I refactored those functions in a separate branch to show-case a different version where the check is performed at most once per code branch.
I decided to use my first solution for this PR since it was smaller and it didn't introduce style changes, which are discouraged according to the contribution guidelines. But since the same guidelines also suggests attention to performance impacts, I decided to create the second version, which would theoretically be more performant (no benchmarks).
I'm happy to update this PR with the changes from my other branch, if that would so be preferred.