Skip to content

Commit d7c71b5

Browse files
committed
fix(history): allow base with / and base tag
Close #164
1 parent 56d310c commit d7c71b5

File tree

3 files changed

+84
-3
lines changed

3 files changed

+84
-3
lines changed

__tests__/history/html5.spec.ts

+52-1
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,24 @@
1+
import { JSDOM } from 'jsdom'
12
import createWebHistory from '../../src/history/html5'
23
import { createDom } from '../utils'
34

45
// These unit tests are supposed to tests very specific scenarios that are easier to setup
56
// on a unit test than an e2e tests
67
describe('History HTMl5', () => {
8+
let dom: JSDOM
79
beforeAll(() => {
8-
createDom()
10+
dom = createDom()
11+
})
12+
13+
afterAll(() => {
14+
dom.window.close()
15+
})
16+
17+
afterEach(() => {
18+
// ensure no base element is left after a test as only the first is
19+
// respected
20+
for (let element of Array.from(document.getElementsByTagName('base')))
21+
element.remove()
922
})
1023

1124
// this problem is very common on hash history when using a regular link
@@ -17,4 +30,42 @@ describe('History HTMl5', () => {
1730
fullPath: '/',
1831
})
1932
})
33+
34+
it('handles a basic base', () => {
35+
expect(createWebHistory().base).toBe('')
36+
expect(createWebHistory('/').base).toBe('')
37+
})
38+
39+
it('handles a base tag', () => {
40+
const baseEl = document.createElement('base')
41+
baseEl.href = '/foo/'
42+
document.head.appendChild(baseEl)
43+
expect(createWebHistory().base).toBe('/foo')
44+
})
45+
46+
it('handles a base tag with origin', () => {
47+
const baseEl = document.createElement('base')
48+
baseEl.href = 'https://example.com/foo/'
49+
document.head.appendChild(baseEl)
50+
expect(createWebHistory().base).toBe('/foo')
51+
})
52+
53+
it('handles a base tag with origin without trailing slash', () => {
54+
const baseEl = document.createElement('base')
55+
baseEl.href = 'https://example.com/bar'
56+
document.head.appendChild(baseEl)
57+
expect(createWebHistory().base).toBe('/bar')
58+
})
59+
60+
it('ignores base tag if base is provided', () => {
61+
const baseEl = document.createElement('base')
62+
baseEl.href = '/foo/'
63+
document.head.appendChild(baseEl)
64+
expect(createWebHistory('/bar/').base).toBe('/bar')
65+
})
66+
67+
it('handles a non-empty base', () => {
68+
expect(createWebHistory('/foo/').base).toBe('/foo')
69+
expect(createWebHistory('/foo').base).toBe('/foo')
70+
})
2071
})

jest.config.js

+1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ module.exports = {
22
preset: 'ts-jest',
33
globals: {
44
__DEV__: true,
5+
__BROWSER__: true,
56
},
67
coverageDirectory: 'coverage',
78
coverageReporters: ['html', 'lcov', 'text'],

src/history/html5.ts

+31-2
Original file line numberDiff line numberDiff line change
@@ -262,7 +262,36 @@ function useHistoryStateNavigation(base: string) {
262262
}
263263
}
264264

265-
export default function createWebHistory(base: string = ''): RouterHistory {
265+
/**
266+
* Normalizes a base by removing any trailing slash and reading the base tag if
267+
* present.
268+
*
269+
* @param base base to normalize
270+
*/
271+
function normalizeBase(base?: string): string {
272+
if (!base) {
273+
if (__BROWSER__) {
274+
// respect <base> tag
275+
const baseEl = document.querySelector('base')
276+
base = (baseEl && baseEl.getAttribute('href')) || '/'
277+
// strip full URL origin
278+
base = base.replace(/^\w+:\/\/[^\/]+/, '')
279+
} else {
280+
base = '/'
281+
}
282+
}
283+
284+
// ensure leading slash when it was removed by the regex above
285+
if (base.charAt(0) !== '/') base = '/' + base
286+
287+
// remove the trailing slash so all other method can just do `base + fullPath`
288+
// to build an href
289+
return base.replace(/\/$/, '')
290+
}
291+
292+
export default function createWebHistory(base?: string): RouterHistory {
293+
base = normalizeBase(base)
294+
266295
const historyNavigation = useHistoryStateNavigation(base)
267296
const historyListeners = useHistoryListeners(
268297
base,
@@ -283,7 +312,7 @@ export default function createWebHistory(base: string = ''): RouterHistory {
283312
const routerHistory: RouterHistory = {
284313
// it's overridden right after
285314
// @ts-ignore
286-
location: historyNavigation.location.value,
315+
location: '',
287316
base,
288317
back,
289318
forward,

0 commit comments

Comments
 (0)