Skip to content
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

Tiny relative links solution #177

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Next Next commit
A working relative links implementation (with test) that only account…
…s for links in a document (not `route("./foo")`). See #138.
developit committed Apr 18, 2017
commit 0e684379ea05c7cea098612d620f01be70954b89
27 changes: 9 additions & 18 deletions src/index.js
Original file line number Diff line number Diff line change
@@ -79,17 +79,11 @@ function routeTo(url) {


function routeFromLink(node) {
// only valid elements
if (!node || !node.getAttribute) return;

let href = node.getAttribute('href'),
target = node.getAttribute('target');

// ignore links with targets and non-path URLs
if (!href || !href.match(/^\//g) || (target && !target.match(/^_?self$/i))) return;
// ignore invalid & external links:
if (!node || node.protocol!==location.protocol || node.host!==location.host || (node.target && !node.target.match(/^_?self$/i))) return;

// attempt to route, if no match simply cede control to browser
return route(href);
return route(node.pathname + node.search + node.hash);
}


@@ -117,12 +111,9 @@ function delegateLinkHandler(e) {

let t = e.target;
do {
if (String(t.nodeName).toUpperCase()==='A' && t.getAttribute('href') && isPreactElement(t)) {
if (t.hasAttribute('native')) return;
if (String(t.nodeName).toUpperCase()==='A' && t.pathname && isPreactElement(t) && !t.hasAttribute('native') && routeFromLink(t)) {
// if link is handled by the router, prevent browser defaults
if (routeFromLink(t)) {
return prevent(e);
}
return prevent(e);
}
} while ((t=t.parentNode));
}
@@ -131,13 +122,13 @@ function delegateLinkHandler(e) {
let eventListenersInitialized = false;

function initEventListeners() {
if (eventListenersInitialized){
return;
}
if (eventListenersInitialized) return;

if (typeof addEventListener==='function') {
if (!customHistory) {
addEventListener('popstate', () => routeTo(getCurrentUrl()));
addEventListener('popstate', () => {
routeTo(getCurrentUrl());
});
}
addEventListener('click', delegateLinkHandler);
}
16 changes: 16 additions & 0 deletions test/dom.js
Original file line number Diff line number Diff line change
@@ -170,6 +170,22 @@ describe('dom', () => {
route('/foo');
expect(routerRef.base.outerHTML).to.eql('<p>bar is </p>');
});

it('should support relative links', () => {
class A {
render() {
return <a href="two">go</a>;
}
}
history.replaceState(null, null, '/foo/one');
mount(
<Router>
<A default />
</Router>
);
scratch.querySelector('a').click();
expect(location.pathname).to.equal('/foo/two');
});
});

describe('preact-router/match', () => {