-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathHtmRouter.js
154 lines (129 loc) · 5.76 KB
/
HtmRouter.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
// Note that this means you have to push useState into global scope
// in your code. See https://github.com/ruffin--/preact-router-for-htm for more.
(function () {
function isRegExp(obj) {
return Object.prototype.toString.call(obj) === '[object RegExp]';
}
function endsWithViaRegExp(str, re) {
var reString = re.source;
re = reString.endsWith('$') ? re : new RegExp(reString + '$', re.flags);
return re.test(str);
}
function endsWithDeluxe(str, stringOrRegExp) {
return isRegExp(stringOrRegExp)
? endsWithViaRegExp(str, stringOrRegExp)
: str.endsWith(stringOrRegExp);
}
function HtmRouter(props) {
const [url, setUrl] = useState(window.location.href);
var idPropName = props.idPropName || 'id';
var routerRoot = props.root || '';
if (routerRoot && routerRoot.endsWith('/')) {
routerRoot = routerRoot.substring(0, routerRoot.length - 1);
}
if (routerRoot && !routerRoot.startsWith('/')) {
routerRoot = '/' + routerRoot;
}
var haveChildren = Array.isArray(props.children);
// unfortunately if you have only one child, it's not sent to props in an array
var singleChild = !haveChildren && props.children.props; // duck sniff`
var children = haveChildren ? props.children : singleChild ? [props.children] : [];
// #region Router-scoped convenience functions
function processNewUrl(newUrl) {
var prefix = newUrl.indexOf('/') === 0 ? routerRoot : '';
return prefix + newUrl;
}
function addUrlToState(state, title, newUrl) {
setUrl(newUrl);
window.history.pushState(state, title || document.title, newUrl);
}
// =============================================
// jive stolen from
// https://github.com/preactjs/preact-router
// =============================================
function prevent(e) {
if (e.stopImmediatePropagation) {
e.stopImmediatePropagation();
}
if (e.stopPropagation) {
e.stopPropagation();
}
e.preventDefault();
return false;
}
// Handles both delegated and direct-bound link clicks
function delegateLinkHandler(e) {
// ignore events the browser takes care of already:
if (e.ctrlKey || e.metaKey || e.altKey || e.shiftKey || e.button) {
return;
}
let t = e.target;
do {
if (t.localName === 'a' && t.getAttribute('href')) {
if (t.hasAttribute('data-native') || t.hasAttribute('native')) {
return;
}
var newUrl = t.getAttribute('href');
newUrl = processNewUrl(newUrl);
// React apparently makes it hard to pull a "return false", so
// let's ignore any link that just wants to go to "#" to keep things clean.
// https://stackoverflow.com/a/31203399/1028230
if (newUrl === '#') {
prevent(e);
} else {
// if link is handled by the router, prevent browser defaults
// Some of this stolen stuff was edited. -RUF
// Not real sure what "state" should do here
// https://developer.mozilla.org/en-US/docs/Web/API/History/pushState
// I mean I get it, but what would I put here?
// https://stackoverflow.com/a/34178234/1028230
var match = routerHandlesUrl(newUrl);
if (match) {
addUrlToState(match.props, match.props.title, newUrl);
return prevent(e);
}
}
}
} while ((t = t.parentNode));
}
let eventListenersInitialized = false;
function initEventListeners() {
if (eventListenersInitialized) {
return;
}
eventListenersInitialized = true;
addEventListener('click', delegateLinkHandler);
}
// =============================================
// eo stolen jive
// =============================================
function routerHandlesUrl(testUrl) {
var match =
children.find((x) => x.props.path && endsWithDeluxe(testUrl, x.props.path)) ||
children.find((x) => x.props.default);
if (match) {
// We'll do a poor man's version of pushing an ending id into props.
if (testUrl.indexOf('/') > -1 && testUrl.length > 1) {
var possibleIdValue = testUrl.substring(testUrl.lastIndexOf('/') + 1);
// Check for int or guid?
match.props[idPropName] = possibleIdValue;
}
}
console.log('routerHandlesUrl', testUrl, match);
return match;
}
// #endregion
// Okay, this feels like a giant kludge.
// It does mean we can't nest routers.
// Guess we could track multiple & hunt them all for matches by some formula.
window.HtmRouter.route = function (newUrl, state, title) {
newUrl = processNewUrl(newUrl);
addUrlToState(state, title, newUrl);
};
initEventListeners();
var match = routerHandlesUrl(url) || (console.log('no router match') && undefined);
console.warn('router is rerendering ' + +new Date());
return match;
}
window.HtmRouter = HtmRouter;
})();