import { _ } from "@/core"; import { _global } from "@/core/0.foundation"; const Events = { // Bind an event to a `callback` function. Passing `"all"` will bind // the callback to all events fired. on(name, callback, context) { if (!eventsApi(this, "on", name, [callback, context]) || !callback) { return this; } this._events || (this._events = {}); const events = this._events[name] || (this._events[name] = []); events.push({ callback, context, ctx: context || this }); return this; }, // Bind an event to only be triggered a single time. After the first time // the callback is invoked, it will be removed. once(name, callback, context) { if (!eventsApi(this, "once", name, [callback, context]) || !callback) { return this; } const self = this; var once = _.once(function () { self.off(name, once); callback.apply(this, arguments); }); once._callback = callback; return this.on(name, once, context); }, // Remove one or many callbacks. If `context` is null, removes all // callbacks with that function. If `callback` is null, removes all // callbacks for the event. If `name` is null, removes all bound // callbacks for all events. off(name, callback, context) { if ( !this._events || !eventsApi(this, "off", name, [callback, context]) ) { return this; } // Remove all callbacks for all events. if (!name && !callback && !context) { this._events = void 0; return this; } const names = name ? [name] : _.keys(this._events); for (let i = 0, length = names.length; i < length; i++) { name = names[i]; // Bail out if there are no events stored. const events = this._events[name]; if (!events) continue; // Remove all callbacks for this event. if (!callback && !context) { delete this._events[name]; continue; } // Find any remaining events. const remaining = []; for (let j = 0, k = events.length; j < k; j++) { const event = events[j]; if ( (callback && callback !== event.callback && callback !== event.callback._callback) || (context && context !== event.context) ) { remaining.push(event); } } // Replace events if there are any remaining. Otherwise, clean up. if (remaining.length) { this._events[name] = remaining; } else { delete this._events[name]; } } return this; }, un() { this.off.apply(this, arguments); }, // Trigger one or many events, firing all bound callbacks. Callbacks are // passed the same arguments as `trigger` is, apart from the event name // (unless you're listening on `"all"`, which will cause your callback to // receive the true name of the event as the first argument). trigger(name) { if (!this._events) return this; const args = slice.call(arguments, 1); if (!eventsApi(this, "trigger", name, args)) return this; const events = this._events[name]; const allEvents = this._events.all; if (events) triggerEvents(events, args); if (allEvents) triggerEvents(allEvents, arguments); return this; }, fireEvent() { this.trigger.apply(this, arguments); }, // Inversion-of-control versions of `on` and `once`. Tell *this* object to // listen to an event in another object ... keeping track of what it's // listening to. listenTo(obj, name, callback) { const listeningTo = this._listeningTo || (this._listeningTo = {}); const id = obj._listenId || (obj._listenId = _.uniqueId("l")); listeningTo[id] = obj; if (!callback && typeof name === "object") callback = this; obj.on(name, callback, this); return this; }, listenToOnce(obj, name, callback) { if (typeof name === "object") { for (const event in name) { this.listenToOnce(obj, event, name[event]); } return this; } if (eventSplitter.test(name)) { const names = name.split(eventSplitter); for (let i = 0, length = names.length; i < length; i++) { this.listenToOnce(obj, names[i], callback); } return this; } if (!callback) return this; var once = _.once(function () { this.stopListening(obj, name, once); callback.apply(this, arguments); }); once._callback = callback; return this.listenTo(obj, name, once); }, // Tell this object to stop listening to either specific events ... or // to every object it's currently listening to. stopListening(obj, name, callback) { let listeningTo = this._listeningTo; if (!listeningTo) return this; const remove = !name && !callback; if (!callback && typeof name === "object") callback = this; if (obj) (listeningTo = {})[obj._listenId] = obj; for (const id in listeningTo) { obj = listeningTo[id]; obj.off(name, callback, this); if (remove || _.isEmpty(obj._events)) { delete this._listeningTo[id]; } } return this; } }; // Regular expression used to split event strings. var eventSplitter = /\s+/; // Implement fancy features of the Events API such as multiple event // names `"change blur"` and jQuery-style event maps `{change: action}` // in terms of the existing API. var eventsApi = function (obj, action, name, rest) { if (!name) return true; // Handle event maps. if (typeof name === "object") { for (const key in name) { obj[action].apply(obj, [key, name[key]].concat(rest)); } return false; } // Handle space separated event names. if (eventSplitter.test(name)) { const names = name.split(eventSplitter); for (let i = 0, length = names.length; i < length; i++) { obj[action].apply(obj, [names[i]].concat(rest)); } return false; } return true; }; // A difficult-to-believe, but optimized internal dispatch function for // triggering events. Tries to keep the usual cases speedy (most internal // BI events have 3 arguments). var triggerEvents = function (events, args) { let ev, i = -1, l = events.length, a1 = args[0], a2 = args[1], a3 = args[2]; switch (args.length) { case 0: while (++i < l) (ev = events[i]).callback.call(ev.ctx); return; case 1: while (++i < l) (ev = events[i]).callback.call(ev.ctx, a1); return; case 2: while (++i < l) (ev = events[i]).callback.call(ev.ctx, a1, a2); return; case 3: while (++i < l) { (ev = events[i]).callback.call(ev.ctx, a1, a2, a3); } return; default: while (++i < l) (ev = events[i]).callback.apply(ev.ctx, args); return; } }; // BI.Router // --------------- // Routers map faux-URLs to actions, and fire events when routes are // matched. Creating a new one sets its `routes` hash, if not set statically. export const Router = function (options) { options || (options = {}); if (options.routes) this.routes = options.routes; this._bindRoutes(); this._init.apply(this, arguments); }; // Cached regular expressions for matching named param parts and splatted // parts of route strings. const optionalParam = /\((.*?)\)/g; const namedParam = /(\(\?)?:\w+/g; const splatParam = /\*\w+/g; const escapeRegExp = /[\-{}\[\]+?.,\\\^$|#\s]/g; // Set up all inheritable **BI.Router** properties and methods. _.extend(Router.prototype, Events, { // _init is an empty function by default. Override it with your own // initialization logic. _init() {}, // Manually bind a single named route to a callback. For example: // // this.route('search/:query/p:num', 'search', function(query, num) { // ... // }); // route(route, name, callback) { if (!_.isRegExp(route)) route = this._routeToRegExp(route); if (_.isFunction(name)) { callback = name; name = ""; } if (!callback) callback = this[name]; const router = this; history.route(route, (fragment) => { const args = router._extractParameters(route, fragment); if (router.execute(callback, args, name) !== false) { router.trigger.apply(router, [`route:${name}`].concat(args)); router.trigger("route", name, args); history.trigger("route", router, name, args); } }); return this; }, // Execute a route handler with the provided parameters. This is an // excellent place to do pre-route setup or post-route cleanup. execute(callback, args, name) { if (callback) callback.apply(this, args); }, // Simple proxy to `BI.history` to save a fragment into the history. navigate(fragment, options) { history.navigate(fragment, options); return this; }, // Bind all defined routes to `BI.history`. We have to reverse the // order of the routes here to support behavior where the most general // routes can be defined at the bottom of the route map. _bindRoutes() { if (!this.routes) return; this.routes = _.result(this, "routes"); let route, routes = _.keys(this.routes); while ((route = routes.pop()) != null) { this.route(route, this.routes[route]); } }, // Convert a route string into a regular expression, suitable for matching // against the current location hash. _routeToRegExp(route) { route = route .replace(escapeRegExp, "\\$&") .replace(optionalParam, "(?:$1)?") .replace(namedParam, (match, optional) => optional ? match : "([^/?]+)" ) .replace(splatParam, "([^?]*?)"); return new RegExp(`^${route}(?:\\?([\\s\\S]*))?$`); }, // Given a route, and a URL fragment that it matches, return the array of // extracted decoded parameters. Empty or unmatched parameters will be // treated as `null` to normalize cross-browser behavior. _extractParameters(route, fragment) { const params = route.exec(fragment).slice(1); return _.map(params, (param, i) => { // Don't decode the search params. if (i === params.length - 1) return param || null; let resultParam = null; if (param) { try { resultParam = decodeURIComponent(param); } catch (e) { resultParam = param; } } return resultParam; }); } }); // History // ---------------- // Handles cross-browser history management, based on either // [pushState](http://diveintohtml5.info/history.html) and real URLs, or // [onhashchange](https://developer.mozilla.org/en-US/docs/DOM/window.onhashchange) // and URL fragments. If the browser supports neither (old IE, natch), // falls back to polling. const History = function () { this.handlers = []; this.checkUrl = _.bind(this.checkUrl, this); // Ensure that `History` can be used outside of the browser. if (typeof window !== "undefined") { this.location = _global.location; this.history = _global.history; } }; // Cached regex for stripping a leading hash/slash and trailing space. const routeStripper = /^[#\/]|\s+$/g; // Cached regex for stripping leading and trailing slashes. const rootStripper = /^\/+|\/+$/g; // Cached regex for stripping urls of hash. const pathStripper = /#.*$/; // Has the history handling already been started? History.started = false; // Set up all inheritable **BI.History** properties and methods. _.extend(History.prototype, Events, { // The default interval to poll for hash changes, if necessary, is // twenty times a second. interval: 50, // Are we at the app root? atRoot() { const path = this.location.pathname.replace(/[^\/]$/, "$&/"); return path === this.root && !this.getSearch(); }, // In IE6, the hash fragment and search params are incorrect if the // fragment contains `?`. getSearch() { const match = this.location.href.replace(/#.*/, "").match(/\?.+/); return match ? match[0] : ""; }, // Gets the true hash value. Cannot use location.hash directly due to bug // in Firefox where location.hash will always be decoded. getHash(window) { const match = (window || this).location.href.match(/#(.*)$/); return match ? match[1] : ""; }, // Get the pathname and search params, without the root. getPath() { let path = this.location.pathname + this.getSearch(); try { path = decodeURI(path); } catch (e) {} const root = this.root.slice(0, -1); if (!path.indexOf(root)) path = path.slice(root.length); return path.charAt(0) === "/" ? path.slice(1) : path; }, // Get the cross-browser normalized URL fragment from the path or hash. getFragment(fragment) { if (fragment == null) { if (this._hasPushState || !this._wantsHashChange) { fragment = this.getPath(); } else { fragment = this.getHash(); } } return fragment.replace(routeStripper, ""); }, // Start the hash change handling, returning `true` if the current URL matches // an existing route, and `false` otherwise. start(options) { if (History.started) { throw new Error("BI.history has already been started"); } History.started = true; // Figure out the initial configuration. Do we need an iframe? // Is pushState desired ... is it available? this.options = _.extend({ root: "/" }, this.options, options); this.root = this.options.root; this._wantsHashChange = this.options.hashChange !== false; this._hasHashChange = "onhashchange" in window; this._wantsPushState = !!this.options.pushState; this._hasPushState = !!( this.options.pushState && this.history && this.history.pushState ); this.fragment = this.getFragment(); // Normalize root to always include a leading and trailing slash. this.root = `/${this.root}/`.replace(rootStripper, "/"); // Transition from hashChange to pushState or vice versa if both are // requested. if (this._wantsHashChange && this._wantsPushState) { // If we've started off with a route from a `pushState`-enabled // browser, but we're currently in a browser that doesn't support it... if (!this._hasPushState && !this.atRoot()) { const root = this.root.slice(0, -1) || "/"; this.location.replace(`${root}#${this.getPath()}`); // Return immediately as browser will do redirect to new url return true; // Or if we've started out with a hash-based route, but we're currently // in a browser where it could be `pushState`-based instead... } else if (this._hasPushState && this.atRoot()) { this.navigate(this.getHash(), { replace: true }); } } // Proxy an iframe to handle location events if the browser doesn't // support the `hashchange` event, HTML5 history, or the user wants // `hashChange` but not `pushState`. if ( !this._hasHashChange && this._wantsHashChange && (!this._wantsPushState || !this._hasPushState) ) { const iframe = document.createElement("iframe"); iframe.src = "javascript:0"; iframe.style.display = "none"; iframe.tabIndex = -1; const body = document.body; // Using `appendChild` will throw on IE < 9 if the document is not ready. this.iframe = body.insertBefore( iframe, body.firstChild ).contentWindow; this.iframe.document.open().close(); this.iframe.location.hash = `#${this.fragment}`; } // Add a cross-platform `addEventListener` shim for older browsers. const addEventListener = _global.addEventListener || function (eventName, listener) { return attachEvent(`on${eventName}`, listener); }; // Depending on whether we're using pushState or hashes, and whether // 'onhashchange' is supported, determine how we check the URL state. if (this._hasPushState) { addEventListener("popstate", this.checkUrl, false); } else if ( this._wantsHashChange && this._hasHashChange && !this.iframe ) { addEventListener("hashchange", this.checkUrl, false); } else if (this._wantsHashChange) { this._checkUrlInterval = setInterval(this.checkUrl, this.interval); } if (!this.options.silent) return this.loadUrl(); }, // Disable BI.history, perhaps temporarily. Not useful in a real app, // but possibly useful for unit testing Routers. stop() { // Add a cross-platform `removeEventListener` shim for older browsers. const removeEventListener = _global.removeEventListener || function (eventName, listener) { return detachEvent(`on${eventName}`, listener); }; // Remove window listeners. if (this._hasPushState) { removeEventListener("popstate", this.checkUrl, false); } else if ( this._wantsHashChange && this._hasHashChange && !this.iframe ) { removeEventListener("hashchange", this.checkUrl, false); } // Clean up the iframe if necessary. if (this.iframe) { document.body.removeChild(this.iframe.frameElement); this.iframe = null; } // Some environments will throw when clearing an undefined interval. if (this._checkUrlInterval) clearInterval(this._checkUrlInterval); History.started = false; }, // Add a route to be tested when the fragment changes. Routes added later // may override previous routes. route(route, callback) { this.handlers.unshift({ route, callback }); }, // check route is Exist. if exist, return the route checkRoute(route) { for (let i = 0; i < this.handlers.length; i++) { if ( this.handlers[i].route.toString() === Router.prototype._routeToRegExp(route).toString() ) { return this.handlers[i]; } } return null; }, // remove a route match in routes unRoute(route) { const index = _.findIndex(this.handlers, (handler) => handler.route.test(route) ); if (index > -1) { this.handlers.splice(index, 1); } }, // Checks the current URL to see if it has changed, and if it has, // calls `loadUrl`, normalizing across the hidden iframe. checkUrl(e) { let current = this.getFragment(); try { // getFragment 得到的值是编码过的,而this.fragment是没有编码过的 // 英文路径没有问题,遇上中文和空格有问题了 current = decodeURIComponent(current); } catch (e) {} // If the user pressed the back button, the iframe's hash will have // changed and we should use that for comparison. if (current === this.fragment && this.iframe) { current = this.getHash(this.iframe); } if (current === this.fragment) return false; if (this.iframe) this.navigate(current); this.loadUrl(); }, // Attempt to load the current URL fragment. If a route succeeds with a // match, returns `true`. If no defined routes matches the fragment, // returns `false`. loadUrl(fragment) { fragment = this.fragment = this.getFragment(fragment); return _.some(this.handlers, (handler) => { if (handler.route.test(fragment)) { handler.callback(fragment); return true; } }); }, // Save a fragment into the hash history, or replace the URL state if the // 'replace' option is passed. You are responsible for properly URL-encoding // the fragment in advance. // // The options object can contain `trigger: true` if you wish to have the // route callback be fired (not usually desirable), or `replace: true`, if // you wish to modify the current URL without adding an entry to the history. navigate(fragment, options) { if (!History.started) return false; if (!options || options === true) options = { trigger: !!options }; // Normalize the fragment. fragment = this.getFragment(fragment || ""); // Don't include a trailing slash on the root. let root = this.root; if (fragment === "" || fragment.charAt(0) === "?") { root = root.slice(0, -1) || "/"; } const url = root + fragment; // Strip the hash and decode for matching. fragment = fragment.replace(pathStripper, ""); try { fragment = decodeURI(fragment); } catch (e) {} if (this.fragment === fragment) return; this.fragment = fragment; // If pushState is available, we use it to set the fragment as a real URL. if (this._hasPushState) { this.history[options.replace ? "replaceState" : "pushState"]( {}, document.title, url ); // If hash changes haven't been explicitly disabled, update the hash // fragment to store history. } else if (this._wantsHashChange) { this._updateHash(this.location, fragment, options.replace); if (this.iframe && fragment !== this.getHash(this.iframe)) { // Opening and closing the iframe tricks IE7 and earlier to push a // history entry on hash-tag change. When replace is true, we don't // want this. if (!options.replace) this.iframe.document.open().close(); this._updateHash( this.iframe.location, fragment, options.replace ); } // If you've told us that you explicitly don't want fallback hashchange- // based history, then `navigate` becomes a page refresh. } else { return this.location.assign(url); } if (options.trigger) return this.loadUrl(fragment); }, // Update the hash location, either replacing the current entry, or adding // a new one to the browser history. _updateHash(location, fragment, replace) { if (replace) { const href = location.href.replace(/(javascript:|#).*$/, ""); location.replace(`${href}#${fragment}`); } else { // Some browsers require that `hash` contains a leading #. location.hash = `#${fragment}`; } } }); // Create the default BI.history. export const history = new History();