Skip to content

Commit 85b1bff

Browse files
committed
fix(hash): allow base tag different from base parameter
Fix #685
1 parent 51e5f80 commit 85b1bff

File tree

6 files changed

+21
-11
lines changed

6 files changed

+21
-11
lines changed

__tests__/history/html5.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -95,7 +95,7 @@ describe('History HTMl5', () => {
9595

9696
it('calls push with hash part of the url with a base', () => {
9797
dom.reconfigure({ url: 'file:///usr/etc/index.html' })
98-
let history = createWebHistory('/usr/etc/index.html#/')
98+
let history = createWebHistory('#')
9999
let spy = jest.spyOn(window.history, 'pushState')
100100
history.push('/foo')
101101
expect(spy).toHaveBeenCalledWith(

docs/api/index.md

+3-4
Original file line numberDiff line numberDiff line change
@@ -227,10 +227,9 @@ export declare function createWebHashHistory(base?: string): RouterHistory
227227

228228
### Parameters
229229

230-
| Parameter | Type | Description |
231-
| --------- | -------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
232-
| base | `string` | optional base to provide. Defaults to `location.pathname` or `/` if at root. If there is a `<base>` tag in the `head`, its value will be ignored in favor of this parameter **but note it affects all the history.pushState() calls**, meaning that if you use a `<base>` tag, its `href` value **has to match this parameter** (ignoring anything after the |
233-
| `#`) |
230+
| Parameter | Type | Description |
231+
| --------- | -------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
232+
| base | `string` | optional base to provide. Defaults to `location.pathname + location.search`. If there is a `<base>` tag in the `head`, its value will be ignored in favor of this parameter **but note it affects all the history.pushState() calls**, meaning that if you use a `<base>` tag, its `href` value **has to match this parameter** (ignoring anything after the `#`) |
234233

235234
### Examples
236235

e2e/hash/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ const Unicode: RouteComponent = {
2020
const router = createRouter({
2121
// keep a trailing slash in this specific case because we are using a hash
2222
// history
23-
history: createWebHashHistory('/' + __dirname + '/#/'),
23+
history: createWebHashHistory(),
2424
routes: [
2525
{ path: '/', component: Home },
2626
{ path: '/redirect', name: 'redirect', redirect: '/foo' },

e2e/specs/hash.js

+1
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,7 @@ module.exports = {
6262
.execute(function () {
6363
window.history.replaceState(history.state, '', '/hash/#/foo')
6464
})
65+
.assert.urlEquals(baseURL + '/foo')
6566
.click('li:nth-child(3) a')
6667
.assert.urlEquals(baseURL + '/bar')
6768
.back()

src/history/hash.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,8 @@ import { warn } from '../warning'
66
* Creates a hash history. Useful for web applications with no host (e.g.
77
* `file://`) or when configuring a server to handle any URL is not possible.
88
*
9-
* @param base - optional base to provide. Defaults to `location.pathname` or
10-
* `/` if at root. If there is a `<base>` tag in the `head`, its value will be
9+
* @param base - optional base to provide. Defaults to `location.pathname +
10+
* location.search` If there is a `<base>` tag in the `head`, its value will be
1111
* ignored in favor of this parameter **but note it affects all the
1212
* history.pushState() calls**, meaning that if you use a `<base>` tag, it's
1313
* `href` value **has to match this parameter** (ignoring anything after the
@@ -32,7 +32,7 @@ export function createWebHashHistory(base?: string): RouterHistory {
3232
// Make sure this implementation is fine in terms of encoding, specially for IE11
3333
// for `file://`, directly use the pathname and ignore the base
3434
// location.pathname contains an initial `/` even at the root: `https://example.com`
35-
base = location.host ? base || location.pathname : ''
35+
base = location.host ? base || location.pathname + location.search : ''
3636
// allow the user to provide a `#` in the middle: `/base/#/app`
3737
if (base.indexOf('#') < 0) base += '#'
3838

src/history/html5.ts

+12-2
Original file line numberDiff line numberDiff line change
@@ -202,11 +202,21 @@ function useHistoryStateNavigation(base: string) {
202202
state: StateEntry,
203203
replace: boolean
204204
): void {
205-
// when the base has a `#`, only use that for the URL
205+
/**
206+
* if a base tag is provided and we are on a normal domain, we have to
207+
* respect the provided `base` attribute because pushState() will use it and
208+
* potentially erase anything before the `#` like at
209+
* https://github.com/vuejs/vue-router-next/issues/685 where a base of
210+
* `/folder/#` but a base of `/` would erase the `/folder/` section. If
211+
* there is no host, the `<base>` tag makes no sense and if there isn't a
212+
* base tag we can just use everything after the `#`.
213+
*/
206214
const hashIndex = base.indexOf('#')
207215
const url =
208216
hashIndex > -1
209-
? base.slice(hashIndex) + to
217+
? (location.host && document.querySelector('base')
218+
? base
219+
: base.slice(hashIndex)) + to
210220
: createBaseLocation() + base + to
211221
try {
212222
// BROWSER QUIRK

0 commit comments

Comments
 (0)