ignite-html-router/ignite-router.js

201 lines
6.6 KiB
JavaScript
Raw Normal View History

2020-09-08 15:44:26 -07:00
import { IgniteElement } from "../ignite-html/ignite-element.js";
import { IgniteTemplate, slot, div, html } from "../ignite-html/ignite-template.js";
class RouterLink extends IgniteElement {
constructor() {
super();
this.pushStateListener = () => this.update();
this.popStateListener = () => this.update();
window.addEventListener("popstate", this.popStateListener);
window.addEventListener("pushstate", this.pushStateListener);
2020-09-08 15:44:26 -07:00
}
get properties() {
return {
active: false,
route: null
2020-09-08 15:44:26 -07:00
};
}
render() {
return this.template
.onClick((event) => this.onClick(event))
.child(
new slot(this)
.class(this.active, (value) => { return value ? "active" : null })
);
}
update() {
var routeMatches = RouteMatcher.matches(this.route);
if (routeMatches && !this.active) {
this.active = true;
} else if (!routeMatches && this.active) {
this.active = false;
}
}
2020-09-08 15:44:26 -07:00
onClick(event) {
event.preventDefault();
window.history.pushState(this.route, this.route, this.route);
window.dispatchEvent(new Event("pushstate"));
}
cleanup() {
window.removeEventListener("popstate", this.popStateListener);
window.removeEventListener("pushstate", this.pushStateListener);
2020-09-08 15:44:26 -07:00
}
}
class RouterView extends IgniteElement {
constructor() {
super();
this.pushStateListener = () => this.update();
this.popStateListener = () => this.update();
window.addEventListener("popstate", this.popStateListener);
window.addEventListener("pushstate", this.pushStateListener);
}
get properties() {
return {
show: false,
route: null
};
2020-09-08 15:44:26 -07:00
}
render() {
return this.template.child(
new slot(this)
).style("display", this.show, null, (value) => { return value ? null : "none"; });
}
update() {
var routeMatches = RouteMatcher.matches(this.route);
if (routeMatches && !this.show) {
this.show = true;
} else if (!routeMatches && this.show) {
this.show = false;
}
2020-09-08 15:44:26 -07:00
}
cleanup() {
window.removeEventListener("popstate", this.popStateListener);
window.removeEventListener("pushstate", this.pushStateListener);
}
}
class RouteMatcher {
static matches(route) {
//Get the path parts from the window
var pathParts = window.location.pathname.split("/").splice(1);
//Get the route parts
var fromRoot = (route.trim().startsWith("/"));
var routeParts = (fromRoot ? route.trim().split("/").splice(1) : route.trim().split("/"));
//Check to see if we have a trailing route part, if so, remove it.
if (pathParts.length > 0 && pathParts[pathParts.length - 1] == "") {
pathParts.pop();
}
if (routeParts.length > 0 && routeParts[routeParts.length - 1] == "") {
routeParts.pop();
}
//If path parts is 0 and route parts is 0 we have a match.
if (pathParts.length == 0 && routeParts.length == 0) {
return true;
}
//If path parts is 0, and the route part is ** then this is a match.
else if (pathParts.length == 0 && routeParts.length == 1 && (routeParts[0] == "**" || routeParts[0] == "*")) {
return true;
}
//If path parts is 0 and the route starts with ! and is a length of 1 then this is a match.
else if (pathParts.length == 0 && routeParts.length == 1 && routeParts[0].startsWith("!")) {
return true;
}
//If the path parts is 0 and the route is !*/** then this is a match
else if (pathParts.length == 0 && routeParts.length == 2 && routeParts[0].startsWith("!") && routeParts[1] == "**") {
return true;
}
//Check the route parts against the path parts.
var max = Math.min(pathParts.length, routeParts.length);
if (fromRoot) {
for (var i = 0; i < max; i++) {
if (routeParts[i].startsWith("!") && pathParts[i] == routeParts[i].substring(1)) {
return false;
} else if (routeParts[i] == "**") {
return true;
} else if (routeParts[i] != pathParts[i] && routeParts[i] != "*" && !routeParts[i].startsWith("!")) {
return false;
} else if (i + 2 == routeParts.length && i + 1 == pathParts.length && routeParts[i + 1] == "**") {
return true;
}
}
if (routeParts.length > pathParts.length) {
return false;
} else if (pathParts.length > routeParts.length && routeParts[routeParts.length - 1] != "**") {
return false;
} else {
return true;
}
} else {
for (var offset = 0; offset < pathParts.length; offset++) {
for (var i = 0; i < max; i++) {
if (i + offset >= pathParts.length) {
return false;
}
if (routeParts[i].startsWith("!") && pathParts[i + offset] == routeParts[i].substring(1)) {
break;
} else if (routeParts[i] == "**") {
return true;
}
else if (routeParts[i] != pathParts[i + offset] && routeParts[i] != "*" && !routeParts[i].startsWith("!")) {
break;
} else if (i + 1 == routeParts.length && offset + routeParts.length == pathParts.length) {
return true;
} else if (i + 2 == routeParts.length && offset + i + 1 == pathParts.length && routeParts[i + 1] == "**") {
return true;
}
}
}
return false;
}
}
2020-09-08 15:44:26 -07:00
}
window.RouteMatcher = RouteMatcher;
2020-09-08 15:44:26 -07:00
class RouterLinkTemplate extends IgniteTemplate {
constructor(route, element) {
super("router-link", [element]);
this.property("route", route);
2020-09-08 15:44:26 -07:00
}
}
class RouterViewTemplate extends IgniteTemplate {
constructor(route, element) {
super("router-view", [element]);
this.property("route", route);
2020-09-08 15:44:26 -07:00
}
}
customElements.define("router-link", RouterLink);
customElements.define("router-view", RouterView);
export {
RouterLinkTemplate as RouterLink,
RouterViewTemplate as RouterView
}