Skip to content

Commit 14b859d

Browse files
committed
fix: allow arbitrary selectors starting with #
BREAKING CHANGE: It is now necessary to escape id selectors like explained at https://mathiasbynens.be/notes/css-escapes. This was necessary to allow selectors like `#container > child`.
1 parent 257f8fe commit 14b859d

File tree

2 files changed

+29
-18
lines changed

2 files changed

+29
-18
lines changed

e2e/scroll-behavior/index.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -36,15 +36,14 @@ const scrollBehavior: ScrollBehavior = async function (
3636

3737
// scroll to anchor by returning the selector
3838
if (to.hash) {
39-
position = { selector: to.hash }
39+
position = { selector: decodeURI(to.hash) }
4040

4141
// specify offset of the element
4242
if (to.hash === '#anchor2') {
4343
position.offset = { y: 100 }
4444
}
4545

46-
// bypass #1number check
47-
if (/^#\d/.test(to.hash) || document.querySelector(to.hash)) {
46+
if (document.querySelector(position.selector)) {
4847
return position
4948
}
5049

@@ -80,6 +79,7 @@ scrollWaiter.add()
8079
const app = createApp({
8180
setup() {
8281
return {
82+
hashWithNumber: { path: '/bar', hash: '#\\31 number' },
8383
flushWaiter: scrollWaiter.flush,
8484
setupWaiter: scrollWaiter.add,
8585
}
@@ -99,7 +99,7 @@ const app = createApp({
9999
<li><router-link to="/bar">/bar</router-link></li>
100100
<li><router-link to="/bar#anchor">/bar#anchor</router-link></li>
101101
<li><router-link to="/bar#anchor2">/bar#anchor2</router-link></li>
102-
<li><router-link to="/bar#1number">/bar#1number</router-link></li>
102+
<li><router-link :to="hashWithNumber">/bar#1number</router-link></li>
103103
</ul>
104104
<router-view class="view" v-slot="{ Component, props }">
105105
<transition

src/scrollBehavior.ts

+25-14
Original file line numberDiff line numberDiff line change
@@ -68,26 +68,37 @@ export function scrollToPosition(position: ScrollPosition): void {
6868

6969
if ('selector' in position) {
7070
/**
71-
* `id`s can accept pretty much any characters, including CSS combinators like >
72-
* or ~. It's still possible to retrieve elements using
71+
* `id`s can accept pretty much any characters, including CSS combinators
72+
* like `>` or `~`. It's still possible to retrieve elements using
7373
* `document.getElementById('~')` but it needs to be escaped when using
74-
* `document.querySelector('#\\~')` for it to be valid. The only requirements
75-
* for `id`s are them to be unique on the page and to not be empty (`id=""`).
76-
* Because of that, when passing an `id` selector, it shouldn't have any other
77-
* selector attached to it (like a class or an attribute) because it wouldn't
78-
* have any effect anyway. We are therefore considering any selector starting
79-
* with a `#` to be an `id` selector so we can directly use `getElementById`
80-
* instead of `querySelector`, allowing users to write simpler selectors like:
81-
* `#1-thing` or `#with~symbols` without having to manually escape them to valid
82-
* CSS selectors: `#\31 -thing` and `#with\\~symbols`.
74+
* `document.querySelector('#\\~')` for it to be valid. The only
75+
* requirements for `id`s are them to be unique on the page and to not be
76+
* empty (`id=""`). Because of that, when passing an id selector, it should
77+
* be properly escaped for it to work with `querySelector`. We could check
78+
* for the id selector to be simple (no CSS combinators `+ >~`) but that
79+
* would make things inconsistent since they are valid characters for an
80+
* `id` but would need to be escaped when using `querySelector`, breaking
81+
* their usage and ending up in no selector returned. Selectors need to be
82+
* escaped:
83+
*
84+
* - `#1-thing` becomes `#\31 -thing`
85+
* - `#with~symbols` becomes `#with\\~symbols`
8386
*
8487
* - More information about the topic can be found at
8588
* https://mathiasbynens.be/notes/html5-id-class.
8689
* - Practical example: https://mathiasbynens.be/demo/html5-id
8790
*/
88-
const el = position.selector.startsWith('#')
89-
? document.getElementById(position.selector.slice(1))
90-
: document.querySelector(position.selector)
91+
if (__DEV__) {
92+
try {
93+
document.querySelector(position.selector)
94+
} catch {
95+
warn(
96+
`The selector "${position.selector}" is invalid. If you are using an id selector, make sure to escape it. You can find more information about escaping characters in selectors at https://mathiasbynens.be/notes/css-escapes.`
97+
)
98+
}
99+
}
100+
101+
const el = document.querySelector(position.selector)
91102

92103
if (!el) {
93104
__DEV__ &&

0 commit comments

Comments
 (0)