1 |
|
2 | export const RouteConfig = {
|
3 | prefix: '',
|
4 | hasTrailingSlash: false,
|
5 | };
|
6 |
|
7 | export const Route = {
|
8 |
|
9 | Config: RouteConfig,
|
10 |
|
11 | _routes: {},
|
12 | _groupData: null,
|
13 |
|
14 | modifyURI(uri) {
|
15 | if (uri.lastIndexOf('/') === uri.length - 1 && !this.Config.hasTrailingSlash) {
|
16 | uri = uri.substr(0, uri.length - 1);
|
17 | }
|
18 | if (uri === '' && this.Config.prefix === '') {
|
19 | return '/';
|
20 | }
|
21 | return this.Config.prefix + uri;
|
22 | },
|
23 |
|
24 | getCurrentURI() {
|
25 | const pref = this.Config.prefix;
|
26 | let path = window.location.pathname;
|
27 | if (path.indexOf(pref) === 0) {
|
28 | path = path.substr(pref.length);
|
29 | }
|
30 | return path;
|
31 | },
|
32 |
|
33 | |
34 |
|
35 |
|
36 |
|
37 |
|
38 |
|
39 |
|
40 | is(uri, haystack_url = window.location.pathname) {
|
41 | uri = this.modifyURI(uri);
|
42 | const original_segments = haystack_url.split('/');
|
43 | const route_segments = uri.split('/');
|
44 | const original_length = original_segments.length;
|
45 | const route_length = route_segments.length;
|
46 | if (original_length !== route_length) {
|
47 | return false;
|
48 | }
|
49 | for (let i = 1; i < original_length; i++) {
|
50 | if (route_segments[i].indexOf('{') === -1 && route_segments[i] !== original_segments[i]) {
|
51 | return false;
|
52 | }
|
53 | }
|
54 | return true;
|
55 | },
|
56 |
|
57 | |
58 |
|
59 |
|
60 |
|
61 |
|
62 | getNamedParams(uri) {
|
63 | const original_segments = window.location.pathname.split('/');
|
64 | const route_segments = uri.split('/');
|
65 | const route_length = route_segments.length;
|
66 | const params = {};
|
67 | for (let i = 1; i < route_length; i++) {
|
68 | if (route_segments[i].indexOf('{') !== -1) {
|
69 | let name = route_segments[i].substr(1, route_segments[i].length-2);
|
70 | params[name] = original_segments[i];
|
71 | }
|
72 | }
|
73 | return params;
|
74 | },
|
75 |
|
76 | |
77 |
|
78 |
|
79 |
|
80 |
|
81 |
|
82 |
|
83 |
|
84 |
|
85 | matchURIWithRoute(uri) {
|
86 | for (let route in this._routes) {
|
87 | if (this.is(route, uri)) {
|
88 | return route;
|
89 | }
|
90 | }
|
91 | return false;
|
92 | },
|
93 |
|
94 | isDefined(uri) {
|
95 | uri = this.modifyURI(uri);
|
96 | return this._routes[uri] !== undefined;
|
97 | },
|
98 |
|
99 | |
100 |
|
101 |
|
102 |
|
103 |
|
104 |
|
105 | on(uri, callback) {
|
106 | if (this._groupData !== null) {
|
107 | if (this._groupData.prefix !== undefined) {
|
108 | uri = this._groupData.prefix + uri;
|
109 | }
|
110 | }
|
111 | uri = this.modifyURI(uri);
|
112 |
|
113 | if (this._routes[uri] === undefined) {
|
114 | this._routes[uri] = [];
|
115 | }
|
116 | this._routes[uri].push(callback);
|
117 | },
|
118 |
|
119 | |
120 |
|
121 |
|
122 |
|
123 |
|
124 |
|
125 |
|
126 |
|
127 |
|
128 |
|
129 |
|
130 | group(prefix, callback) {
|
131 | this._groupData = {prefix};
|
132 | callback();
|
133 | this._groupData = null;
|
134 | },
|
135 |
|
136 | getAllRouteNames() {
|
137 | return Object.keys(this._routes);
|
138 | },
|
139 |
|
140 | getRouteCallbacks(uri) {
|
141 | if (this._routes[uri] !== undefined) {
|
142 | return this._routes[uri];
|
143 | }
|
144 | return false;
|
145 | },
|
146 |
|
147 | |
148 |
|
149 |
|
150 |
|
151 |
|
152 | run(uri) {
|
153 | uri = this.modifyURI(uri);
|
154 |
|
155 | const callbacks = this.getRouteCallbacks(uri);
|
156 | if (callbacks === false) {
|
157 | console.error('Route "' + uri + '" is not defined.');
|
158 | return false;
|
159 | }
|
160 |
|
161 | const namedParams = this.getNamedParams(uri);
|
162 |
|
163 | callbacks.forEach(cb => {
|
164 | cb(namedParams)
|
165 | });
|
166 |
|
167 | return true;
|
168 | },
|
169 |
|
170 | |
171 |
|
172 |
|
173 |
|
174 |
|
175 | to(uri) {
|
176 | uri = this.modifyURI(uri);
|
177 | history.pushState(null, null, uri);
|
178 | const route = Route.matchURIWithRoute(uri);
|
179 | const event = new CustomEvent('onRouteChange', {detail: uri});
|
180 | event.route = route;
|
181 | document.dispatchEvent(event);
|
182 |
|
183 | if (route) {
|
184 | const callbacks = this.getRouteCallbacks(route);
|
185 | if (callbacks !== false) {
|
186 | const namedParams = this.getNamedParams(uri);
|
187 | callbacks.forEach(cb => {
|
188 | cb(namedParams)
|
189 | });
|
190 | }
|
191 | }
|
192 | }
|
193 |
|
194 | };
|
195 |
|
196 | document.addEventListener('DOMContentLoaded', function() {
|
197 | const curURI = Route.getCurrentURI();
|
198 | const route = Route.matchURIWithRoute(curURI);
|
199 | if (route) {
|
200 | Route.run(route);
|
201 | }
|
202 | }, false);
|