2022-10-24 08:40:30 -07:00
|
|
|
import { IgniteHtml, IgniteRendering } from "../ignite-html/ignite-html.js";
|
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";
|
2021-12-28 22:33:36 -08:00
|
|
|
import { IgniteCallback, IgniteProperty} from "../ignite-html/ignite-html.js";
|
2021-08-11 10:45:06 -07:00
|
|
|
|
|
|
|
/**
|
2021-08-24 13:40:29 -07:00
|
|
|
* Creates a route listener that runs when a route is switched. The listener will run on construct the first time and update the state.
|
2021-08-25 00:01:10 -07:00
|
|
|
* @class IgniteTemplate
|
|
|
|
* @memberof IgniteTemplate
|
2021-08-11 10:45:06 -07:00
|
|
|
* @param {String,String[]} routes A single or multiple set of routes that will invoke the callback if met.
|
2021-12-13 16:49:33 -08:00
|
|
|
* @param {Function(route, data)} showCallback A callback function that is called when the route is shown. Default is null.
|
2021-08-25 00:01:10 -07:00
|
|
|
* @param {Function} hideCallback A callback function that is called when the route is hidden. Default is null.
|
|
|
|
* @param {Boolean} strict If true all routes must match before running the callback. Default is false.
|
2021-08-11 10:45:06 -07:00
|
|
|
* @returns {IgniteTemplate} This ignite template.
|
|
|
|
*/
|
2021-08-25 00:01:10 -07:00
|
|
|
IgniteTemplate.prototype.route = function(routes, showCallback = null, hideCallback = null, strict = false) {
|
2021-08-11 10:45:06 -07:00
|
|
|
//If routes is not an array, wrap it.
|
|
|
|
if (!Array.isArray(routes)) {
|
|
|
|
routes = [routes];
|
|
|
|
}
|
|
|
|
|
2021-08-24 13:40:29 -07:00
|
|
|
//Create an update method that will be used to check and see if the route is met.
|
|
|
|
var update = (event) => {
|
2021-08-11 10:45:06 -07:00
|
|
|
var routeMatches = false;
|
2021-12-13 16:49:33 -08:00
|
|
|
var matchedRoute = null;
|
2021-08-11 10:45:06 -07:00
|
|
|
|
2021-08-24 13:40:29 -07:00
|
|
|
//Create an object to hold any data.
|
|
|
|
var data = {};
|
|
|
|
|
2021-08-11 10:45:06 -07:00
|
|
|
//Based on whether we are strict matching or not check if we have a match.
|
|
|
|
if (!strict) {
|
|
|
|
for (var i = 0; i < routes.length && !routeMatches; i++) {
|
2021-08-24 13:40:29 -07:00
|
|
|
routeMatches = Router.matches(routes[i], data);
|
2021-12-13 16:49:33 -08:00
|
|
|
matchedRoute = routes[i];
|
2021-08-11 10:45:06 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
routeMatches = true;
|
|
|
|
for (var i = 0; i < routes.length && routeMatches; i++) {
|
2021-08-24 13:40:29 -07:00
|
|
|
routeMatches = Router.matches(routes[i], data);
|
2021-12-13 16:49:33 -08:00
|
|
|
matchedRoute = routes[i];
|
2021-08-11 10:45:06 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//Invoke the callback if the route matched.
|
|
|
|
if (routeMatches) {
|
2021-08-25 00:01:10 -07:00
|
|
|
//Show the route element.
|
|
|
|
this.element.style.removeProperty("display");
|
|
|
|
|
|
|
|
if (showCallback) {
|
2021-12-13 16:49:33 -08:00
|
|
|
showCallback(matchedRoute, (event && event.data ? event.data : data));
|
2021-08-25 00:01:10 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
//Hide the route element.
|
2021-12-06 09:27:23 -08:00
|
|
|
this.element.style.setProperty("display", "none", "important");
|
2021-08-25 00:01:10 -07:00
|
|
|
|
|
|
|
if (hideCallback) {
|
|
|
|
hideCallback();
|
2021-08-24 13:40:29 -07:00
|
|
|
}
|
2021-08-11 10:45:06 -07:00
|
|
|
}
|
2021-08-24 13:40:29 -07:00
|
|
|
};
|
2021-08-11 10:45:06 -07:00
|
|
|
|
2021-11-11 11:43:10 -08:00
|
|
|
//Create a managed push & pop callback
|
|
|
|
var managedPush = new IgniteCallback(update, () => {
|
|
|
|
window.removeEventListener("pushstate", managedPush.callback);
|
2021-08-11 10:45:06 -07:00
|
|
|
});
|
|
|
|
|
2021-11-11 11:43:10 -08:00
|
|
|
var managedPop = new IgniteCallback(update, () => {
|
|
|
|
window.removeEventListener("popstate", managedPop.callback);
|
|
|
|
});
|
|
|
|
|
|
|
|
//Register our push & pop callbacks.
|
|
|
|
window.addEventListener("pushstate", managedPush.callback);
|
|
|
|
window.addEventListener("popstate", managedPop.callback);
|
2021-08-11 10:45:06 -07:00
|
|
|
|
2021-11-11 11:43:10 -08:00
|
|
|
//Add the managed callbacks to our template so that upon deconstruction our callback is destroyed correctly.
|
|
|
|
this._callbacks.push(managedPush);
|
|
|
|
this._callbacks.push(managedPop);
|
2021-08-11 10:45:06 -07:00
|
|
|
|
2022-10-24 08:40:30 -07:00
|
|
|
//Upon ready call update so our view is displayed if needed.
|
|
|
|
IgniteRendering.ready(update);
|
2021-08-24 13:40:29 -07:00
|
|
|
|
2021-08-11 10:45:06 -07:00
|
|
|
return this;
|
|
|
|
};
|
2020-09-08 15:44:26 -07:00
|
|
|
|
2021-12-28 22:33:36 -08:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {String|IgniteProperty|Function} route The route to navigate to.
|
|
|
|
* @param {Any|IgniteProperty|Function} data The data to pass along with this navigate, this only applies when not refreshing, default is null.
|
|
|
|
* @param {Boolean|IgniteProperty|Function} refresh Whether or not to refresh the page, default is false.
|
2022-04-22 00:43:36 -07:00
|
|
|
* @param {Boolean|IgniteProperty|Function} keepQuery Whether or not to keep the current url query, default is false.
|
2021-12-28 22:33:36 -08:00
|
|
|
* @returns {IgniteTemplate} This ignite template.
|
|
|
|
*/
|
2022-04-22 00:43:36 -07:00
|
|
|
IgniteTemplate.prototype.navigate = function(route, data = null, refresh = false, keepQuery = false) {
|
2021-12-28 22:33:36 -08:00
|
|
|
return this.onClick(() => {
|
|
|
|
Router.navigate(
|
|
|
|
(route instanceof IgniteProperty ? route.value : (route instanceof Function ? route() : route)),
|
|
|
|
(data instanceof IgniteProperty ? data.value : (data instanceof Function ? data() : data)),
|
2022-04-22 00:43:36 -07:00
|
|
|
(refresh instanceof IgniteProperty ? refresh.value : (refresh instanceof Function ? refresh() : refresh)),
|
|
|
|
(keepQuery instanceof IgniteProperty ? keepQuery.value : (keepQuery instanceof Function ? keepQuery() : keepQuery))
|
2021-12-28 22:33:36 -08:00
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-09-08 15:44:26 -07:00
|
|
|
class RouterLink extends IgniteElement {
|
|
|
|
constructor() {
|
|
|
|
super();
|
2020-09-10 22:45:28 -07:00
|
|
|
|
|
|
|
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 {
|
2020-09-10 22:45:28 -07:00
|
|
|
active: false,
|
2020-11-08 16:14:50 -08:00
|
|
|
routes: [],
|
|
|
|
target: null
|
2020-09-08 15:44:26 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-10-28 10:03:28 -07:00
|
|
|
get styles() {
|
2020-12-24 17:47:24 -08:00
|
|
|
return /*css*/`
|
2020-10-28 10:03:28 -07:00
|
|
|
router-link {
|
|
|
|
display: flex;
|
|
|
|
flex-direction: row;
|
|
|
|
align-items: center;
|
|
|
|
justify-content: center;
|
|
|
|
}
|
|
|
|
`;
|
|
|
|
}
|
|
|
|
|
2020-09-08 15:44:26 -07:00
|
|
|
render() {
|
|
|
|
return this.template
|
|
|
|
.onClick((event) => this.onClick(event))
|
2020-12-12 13:28:32 -08:00
|
|
|
.class(this.active, value => value ? "active" : null)
|
|
|
|
.child(new slot(this).class(this.active, value => value ? "active" : null));
|
2020-09-08 15:44:26 -07:00
|
|
|
}
|
|
|
|
|
2020-09-11 07:54:17 -07:00
|
|
|
ready() {
|
|
|
|
this.update();
|
|
|
|
}
|
|
|
|
|
2020-09-10 22:45:28 -07:00
|
|
|
update() {
|
2021-01-22 15:14:37 -08:00
|
|
|
var routeMatches = false;
|
2020-11-02 08:15:12 -08:00
|
|
|
|
2020-11-08 16:14:50 -08:00
|
|
|
//Check the target first.
|
2021-05-19 10:28:07 -07:00
|
|
|
routeMatches = Router.matches(this.target);
|
2020-11-08 16:14:50 -08:00
|
|
|
|
|
|
|
//Check optional routes next.
|
2021-01-22 15:14:37 -08:00
|
|
|
for (var i = 0; i < this.routes.length && !routeMatches; i++) {
|
2021-05-19 10:28:07 -07:00
|
|
|
routeMatches = Router.matches(this.routes[i]);
|
2020-11-02 08:15:12 -08:00
|
|
|
}
|
2020-09-10 22:45:28 -07:00
|
|
|
|
|
|
|
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) {
|
2020-09-10 22:45:28 -07:00
|
|
|
event.preventDefault();
|
2021-08-11 10:45:06 -07:00
|
|
|
Router.navigate(this.target, null, false);
|
2020-09-10 22:45:28 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
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();
|
2020-09-10 18:55:49 -07:00
|
|
|
|
2020-09-10 22:45:28 -07:00
|
|
|
this.pushStateListener = () => this.update();
|
|
|
|
this.popStateListener = () => this.update();
|
|
|
|
|
|
|
|
window.addEventListener("popstate", this.popStateListener);
|
|
|
|
window.addEventListener("pushstate", this.pushStateListener);
|
|
|
|
}
|
|
|
|
|
|
|
|
get properties() {
|
|
|
|
return {
|
|
|
|
show: false,
|
2021-01-25 00:38:18 -08:00
|
|
|
routes: [],
|
|
|
|
strict: false
|
2020-09-10 22:45:28 -07:00
|
|
|
};
|
2020-09-08 15:44:26 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
2021-05-19 10:28:07 -07:00
|
|
|
return this.template
|
|
|
|
.child(new slot(this))
|
|
|
|
.style("display", this.show, null, value => value ? null : "none");
|
2020-09-10 22:45:28 -07:00
|
|
|
}
|
|
|
|
|
2020-09-11 07:54:17 -07:00
|
|
|
ready() {
|
|
|
|
this.update();
|
|
|
|
}
|
|
|
|
|
2020-09-10 22:45:28 -07:00
|
|
|
update() {
|
2021-01-22 15:14:37 -08:00
|
|
|
var routeMatches = false;
|
2020-11-02 08:15:12 -08:00
|
|
|
|
2021-01-25 00:38:18 -08:00
|
|
|
//Based on whether we are strict matching or not check if we have a match.
|
|
|
|
if (!this.strict) {
|
|
|
|
for (var i = 0; i < this.routes.length && !routeMatches; i++) {
|
2021-05-19 10:28:07 -07:00
|
|
|
routeMatches = Router.matches(this.routes[i]);
|
2021-01-25 00:38:18 -08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
routeMatches = true;
|
|
|
|
for (var i = 0; i < this.routes.length && routeMatches; i++) {
|
2021-05-19 10:28:07 -07:00
|
|
|
routeMatches = Router.matches(this.routes[i]);
|
2021-01-25 00:38:18 -08:00
|
|
|
}
|
2020-11-02 08:15:12 -08:00
|
|
|
}
|
2020-09-10 22:45:28 -07:00
|
|
|
|
2021-01-22 15:14:37 -08:00
|
|
|
//If we found a match show this router view if it's not already visible, otherwise hide it.
|
2020-09-10 22:45:28 -07:00
|
|
|
if (routeMatches && !this.show) {
|
|
|
|
this.show = true;
|
|
|
|
} else if (!routeMatches && this.show) {
|
|
|
|
this.show = false;
|
|
|
|
}
|
2020-09-08 15:44:26 -07:00
|
|
|
}
|
2020-09-10 18:55:49 -07:00
|
|
|
|
2020-09-10 22:45:28 -07:00
|
|
|
cleanup() {
|
|
|
|
window.removeEventListener("popstate", this.popStateListener);
|
|
|
|
window.removeEventListener("pushstate", this.pushStateListener);
|
2020-09-10 18:55:49 -07:00
|
|
|
}
|
2020-09-10 22:45:28 -07:00
|
|
|
}
|
|
|
|
|
2021-05-19 10:28:07 -07:00
|
|
|
class RouterLinkTemplate extends IgniteTemplate {
|
|
|
|
/**
|
|
|
|
* Initializes a new router link template.
|
|
|
|
* @param {String} target The target route when the link is clicked.
|
|
|
|
* @param {String|String[]} routes Optional routes that can be used to control the active state of the link.
|
|
|
|
* @param {...any} elements Elements to render within the link.
|
|
|
|
*/
|
|
|
|
constructor(target, routes, ...elements) {
|
|
|
|
super("router-link", elements);
|
|
|
|
|
|
|
|
if (!routes) {
|
|
|
|
routes = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Array.isArray(routes)) {
|
|
|
|
routes = [routes];
|
|
|
|
}
|
|
|
|
|
|
|
|
this.property("target", target);
|
|
|
|
this.property("routes", routes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class RouterViewTemplate extends IgniteTemplate {
|
|
|
|
/**
|
|
|
|
* Initializes a new router view.
|
|
|
|
* @param {String|String[]} routes Single or multiple routes to trigger this view to render.
|
|
|
|
* @param {any|any[]} elements Elements to render within the view.
|
|
|
|
* @param {Boolean} strict If true all routes must match before this view becomes visible.
|
|
|
|
*/
|
|
|
|
constructor(routes, elements, strict = false) {
|
|
|
|
super("router-view", Array.isArray(elements) ? elements : [elements]);
|
|
|
|
|
|
|
|
if (!Array.isArray(routes)) {
|
|
|
|
routes = [routes];
|
|
|
|
}
|
|
|
|
|
|
|
|
this.property("routes", routes);
|
|
|
|
this.property("strict", strict);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Router {
|
2021-07-28 08:40:52 -07:00
|
|
|
/**
|
|
|
|
* Navigates to a given route and refreshes the page if requested.
|
|
|
|
* @param {String} route The route to navigate to.
|
2021-08-11 10:45:06 -07:00
|
|
|
* @param {Any} data The data to pass along with this navigate, this only applies when not refreshing, default is null.
|
2021-07-28 08:40:52 -07:00
|
|
|
* @param {Boolean} refresh Whether or not to refresh the page, default is false.
|
2022-04-22 00:43:36 -07:00
|
|
|
* @param {Boolean} keepQuery Whether or not to keep the current url query, default is false.
|
2021-07-28 08:40:52 -07:00
|
|
|
*/
|
2022-04-22 00:43:36 -07:00
|
|
|
static navigate(route, data = null, refresh = false, keepQuery = false) {
|
|
|
|
//If keepQuery is true then include the current query.
|
|
|
|
if (keepQuery) {
|
|
|
|
if (route.includes("?")) {
|
|
|
|
route = route.split("?")[0] + window.location.search;
|
|
|
|
} else {
|
|
|
|
route = route + window.location.search;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-19 10:28:07 -07:00
|
|
|
if (refresh) {
|
|
|
|
if (Router.hashMode) {
|
2021-11-25 12:42:21 -08:00
|
|
|
//In hash mode the route can't start with / or #, we have to handle it here.
|
|
|
|
if (route.startsWith("/") || route.startsWith("#")) {
|
2021-10-13 07:47:11 -07:00
|
|
|
route = route.substr(1);
|
|
|
|
}
|
|
|
|
|
2021-05-19 10:28:07 -07:00
|
|
|
window.location.hash = route;
|
|
|
|
window.location.reload();
|
|
|
|
} else {
|
|
|
|
window.location.href = route;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (Router.hashMode) {
|
2021-11-25 12:42:21 -08:00
|
|
|
//In hash mode the route can't start with / or #, we have to handle it here.
|
|
|
|
if (route.startsWith("/") || route.startsWith("#")) {
|
2021-10-13 07:47:11 -07:00
|
|
|
route = route.substr(1);
|
|
|
|
}
|
|
|
|
|
2021-05-19 10:28:07 -07:00
|
|
|
window.location.hash = route;
|
|
|
|
} else {
|
|
|
|
window.history.pushState(route, route, route);
|
|
|
|
}
|
|
|
|
|
2021-10-13 07:47:11 -07:00
|
|
|
//Push the route to our internal states
|
|
|
|
Router.states.push(route);
|
|
|
|
|
2021-08-11 10:45:06 -07:00
|
|
|
//Create a new pushstate event and fire it.
|
|
|
|
var event = new Event("pushstate");
|
|
|
|
event.data = data;
|
|
|
|
window.dispatchEvent(event);
|
2021-05-19 10:28:07 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-28 08:40:52 -07:00
|
|
|
/**
|
|
|
|
* Navigates back one with the ability to have a fallback route or refresh the page.
|
|
|
|
* @param {String} fallback The fallback route incase there is no history, default is null.
|
|
|
|
* @param {Boolean} refresh Whether or not to refresh the page, default is false.
|
|
|
|
*/
|
|
|
|
static back(fallback = null, refresh = false) {
|
|
|
|
if (Router.states && Router.states.length > 1) {
|
|
|
|
Router.states.pop(); //Pop the current state.
|
2021-08-11 10:45:06 -07:00
|
|
|
Router.navigate(Router.states.pop(), null, refresh); //Navigate to the previous state.
|
2021-07-28 08:40:52 -07:00
|
|
|
} else {
|
2021-11-11 11:43:10 -08:00
|
|
|
//Pop the current state, otherwise it can be used to go back but we don't want that.
|
|
|
|
if (Router.states.length > 0) {
|
|
|
|
Router.states.pop();
|
|
|
|
}
|
|
|
|
|
2021-07-28 08:40:52 -07:00
|
|
|
if (fallback) {
|
2021-08-11 10:45:06 -07:00
|
|
|
Router.navigate(fallback, null, refresh);
|
2021-07-28 08:40:52 -07:00
|
|
|
} else {
|
|
|
|
window.history.back();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-11 10:45:06 -07:00
|
|
|
/**
|
|
|
|
* Returns whether or not the current location matches a given route.
|
|
|
|
* @param {String} route The route to check.
|
2021-08-24 13:40:29 -07:00
|
|
|
* @param {Object} data The object to be populated with data from the route.
|
2021-08-11 10:45:06 -07:00
|
|
|
* @returns {Boolean} Returns whether or not the current browser location matches the route.
|
2021-08-24 13:40:29 -07:00
|
|
|
* @example matches('/user/**') //Anything after /user/ is considered a match.
|
|
|
|
* @example matches('/user/{id}/profile') //Anything between /user and /profile is a match. data.id would be set to the value inside {id}
|
|
|
|
* @example matches('/user/!0/profile') //Anything between /user and /profile that is not '0' is a match.
|
2021-08-11 10:45:06 -07:00
|
|
|
*/
|
2021-08-24 13:40:29 -07:00
|
|
|
static matches(route, data = null) {
|
2020-09-10 22:45:28 -07:00
|
|
|
//Get the path parts from the window
|
2021-05-19 10:28:07 -07:00
|
|
|
var pathParts = [];
|
|
|
|
|
|
|
|
//If hash mode is set and we have a hash location, get it and split it.
|
|
|
|
if (Router.hashMode && window.location.hash && window.location.hash.length > 0) {
|
2021-11-25 12:42:21 -08:00
|
|
|
var path = window.location.hash.substr(1);
|
|
|
|
|
|
|
|
//If the path contains ? then remove the query.
|
|
|
|
if (path.includes("?")) {
|
|
|
|
path = path.split("?")[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
//Break the path into path parts.
|
|
|
|
pathParts = path.split("/");
|
2021-05-19 10:28:07 -07:00
|
|
|
if (pathParts.length > 0 && pathParts[0].length == 0) {
|
|
|
|
pathParts.splice(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (!Router.hashMode) {
|
|
|
|
pathParts = window.location.pathname.split("/").splice(1);
|
|
|
|
}
|
2020-09-10 22:45:28 -07:00
|
|
|
|
|
|
|
//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();
|
|
|
|
}
|
2020-09-10 18:55:49 -07:00
|
|
|
|
2020-09-10 22:45:28 -07:00
|
|
|
//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++) {
|
2021-08-24 13:40:29 -07:00
|
|
|
if (routeParts[i].startsWith("{") && routeParts[i].endsWith("}")) {
|
|
|
|
if (data) {
|
|
|
|
data[routeParts[i].substring(1, routeParts[i].length - 1)] = pathParts[i];
|
|
|
|
}
|
|
|
|
} else if (routeParts[i].startsWith("!") && pathParts[i] == routeParts[i].substring(1)) {
|
2020-09-10 22:45:28 -07:00
|
|
|
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;
|
|
|
|
}
|
2021-08-24 13:40:29 -07:00
|
|
|
else if (routeParts[i].startsWith("!") && pathParts[i + offset] == routeParts[i].substring(1)) {
|
2020-09-10 22:45:28 -07:00
|
|
|
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-10 18:55:49 -07:00
|
|
|
}
|
2021-11-25 12:42:21 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a query parameter from the route search if there is one by it's name.
|
|
|
|
* @param {String} name The name of the query parameter to get.
|
|
|
|
* @returns {String} Null if not found, or the value of the query parameter.
|
|
|
|
*/
|
|
|
|
static getParameter(name) {
|
|
|
|
if (Router.hashMode) {
|
|
|
|
var params = new URLSearchParams(window.location.hash.includes("?") ? "?" + window.location.hash.split("?")[1] : "");
|
|
|
|
return (params.has(name) ? params.get(name) : null);
|
|
|
|
} else {
|
|
|
|
var params = new URLSearchParams(window.location.search);
|
|
|
|
return (params.has(name) ? params.get(name) : null);
|
|
|
|
}
|
|
|
|
}
|
2022-05-04 09:03:47 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes a query parameter from the route search if it exists.
|
|
|
|
* @param {String} name
|
|
|
|
*/
|
|
|
|
static removeParameter(name) {
|
|
|
|
if (Router.hashMode) {
|
|
|
|
var params = new URLSearchParams(window.location.hash.includes("?") ? "?" + window.location.hash.split("?")[1] : "");
|
|
|
|
if (params.has(name)) {
|
|
|
|
params.delete(name);
|
|
|
|
window.location.hash = window.location.hash.split("?")[0] + "?" + params.toString();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
var url = new URL(window.location);
|
|
|
|
url.searchParams.delete(name);
|
|
|
|
window.history.pushState(null, '', url.toString());
|
|
|
|
}
|
|
|
|
}
|
2020-09-08 15:44:26 -07:00
|
|
|
}
|
|
|
|
|
2021-07-28 08:40:52 -07:00
|
|
|
Router.states = [];
|
2021-05-19 10:28:07 -07:00
|
|
|
Router.hashMode = false;
|
2020-09-08 15:44:26 -07:00
|
|
|
|
2022-08-26 08:36:21 -07:00
|
|
|
IgniteHtml.register("router-link", RouterLink);
|
|
|
|
IgniteHtml.register("router-view", RouterView);
|
2020-09-08 15:44:26 -07:00
|
|
|
|
2021-05-19 10:28:07 -07:00
|
|
|
window.Router = Router;
|
|
|
|
|
2020-09-08 15:44:26 -07:00
|
|
|
export {
|
|
|
|
RouterLinkTemplate as RouterLink,
|
2020-11-08 16:14:50 -08:00
|
|
|
RouterViewTemplate as RouterView,
|
2021-05-19 10:28:07 -07:00
|
|
|
Router
|
2020-09-08 15:44:26 -07:00
|
|
|
}
|