2 * Originally based of this code... - refactored for Roo...
5 * @author Benjamin Arthur Lupton <contact@balupton.com>
6 * @copyright 2010-2011 Benjamin Arthur Lupton <contact@balupton.com>
7 * @license New BSD License <http://creativecommons.org/licenses/BSD/>
13 // ====================================================================
19 * How long should the interval be before hashchange checks
21 thishashChangeInterval : 100,
25 * How long should the interval be before safari poll checks
27 safariPollInterval : 500,
31 * How long should the interval be before we perform a double check
33 doubleCheckInterval : 500,
37 * Force this.not to append suid
43 * How long should we wait between store calls
49 * How long should we wait between busy events
55 * If true will enable debug messages to be logged
61 * What is the title of the initial state
67 * If true, will force HTMl4 mode (hashtags)
73 * Want to override default options and call init manually.
79 * Which bugs are present
83 * Safari 5 and Safari iOS 4 fail to return to the correct state once a hash is replaced by a `replaceState` call
84 * https://bugs.webkit.org/show_bug.cgi?id=56249
89 * Safari 5 and Safari iOS 4 sometimes fail to apply the state change under busy conditions
90 * https://bugs.webkit.org/show_bug.cgi?id=42940
95 * MSIE 6 and 7 sometimes do not apply a hash even it was told to (requiring a second call to the apply function)
100 * MSIE 6 requires the entire hash to be encoded for the hashes to trigger the onHashChange event
105 // ========================================================================
111 sessionStorage : false, // sessionStorage
113 intervalList : false, // array normally.
116 * Is History enabled?
120 // ====================================================================
125 * The store for all session specific data
131 * 1-1: State ID to State Object
137 * 1-1: State String to State ID
143 * 1-1: State URL to State ID
149 * Store the states in an array
151 storedStates : false,
155 * Saved the states in an array
161 * The list of queues to use
162 * First In, First Out
171 // Initialise History
172 init : function(options){
174 initialTitle : window.document.title,
179 this.storedStates=[];
183 Roo.apply(this,options)
185 // Check Load Status of Adapter
186 //if ( typeof this.Adapter === 'undefined' ) {
190 // Check Load Status of Core
191 if ( typeof this.initCore !== 'undefined' ) {
195 // Check Load Status of HTML4 Support
196 if ( typeof this.initHtml4 !== 'undefined' ) {
202 this.enabled = !this.emulated.pushState;
215 // ========================================================================
219 initCore : function(options){
221 this.intervalList = [];
225 this.sessionStorage = window.sessionStorage; // This will throw an exception in some browsers when cookies/localStorage are explicitly disabled (i.e. Chrome)
226 this.sessionStorage.setItem('TEST', '1');
227 this.sessionStorage.removeItem('TEST');
229 this.sessionStorage = false;
233 if ( typeof this.initCore.initialized !== 'undefined' ) {
238 this.initCore.initialized = true;
247 * Clears all setInterval instances.
249 clearAllIntervals: function()
251 var i, il = this.intervalList;
252 if (typeof il !== "undefined" && il !== null) {
253 for (i = 0; i < il.length; i++) {
254 clearInterval(il[i]);
256 this.intervalList = null;
261 // ====================================================================
265 * debugLog(message,...)
266 * Logs the passed arguments if debug enabled
268 debugLog : function()
270 if ( (this.debug||false) ) {
271 Roo.log.apply(this,arguments);
277 // ====================================================================
281 * getInternetExplorerMajorVersion()
282 * Get's the major version of Internet Explorer
284 * @license Public Domain
285 * @author Benjamin Arthur Lupton <contact@balupton.com>
286 * @author James Padolsey <https://gist.github.com/527683>
288 getInternetExplorerMajorVersion : function(){
289 var result = this.getInternetExplorerMajorVersion.cached =
290 (typeof this.getInternetExplorerMajorVersion.cached !== 'undefined')
291 ? this.getInternetExplorerMajorVersion.cached
294 div = window.document.createElement('div'),
295 all = div.getElementsByTagName('i');
296 while ( (div.innerHTML = '<!--[if gt IE ' + (++v) + ']><i></i><![endif]-->') && all[0] ) {}
297 return (v > 4) ? v : false;
304 * isInternetExplorer()
305 * Are we using Internet Explorer?
307 * @license Public Domain
308 * @author Benjamin Arthur Lupton <contact@balupton.com>
310 isInternetExplorer : function(){
312 this.isInternetExplorer.cached =
313 (typeof this.isInternetExplorer.cached !== 'undefined')
314 ? this.isInternetExplorer.cached
315 : Boolean(this.getInternetExplorerMajorVersion())
321 * Which features require emulating?
329 initEmulated : function()
333 if (this.html4Mode) {
339 this.emulated.pushState = !Boolean(
340 window.history && window.history.pushState && window.history.replaceState
342 (/ Mobile\/([1-7][a-z]|(8([abcde]|f(1[0-8]))))/i).test(window.navigator.userAgent) /* disable for versions of iOS before version 4.3 (8F190) */
343 || (/AppleWebKit\/5([0-2]|3[0-2])/i).test(window.navigator.userAgent) /* disable for the mercury iOS browser, or at least older versions of the webkit engine */
346 this.emulated.hashChange = Boolean(
347 !(('onhashchange' in window) || ('onhashchange' in window.document))
349 (this.isInternetExplorer() && this.getInternetExplorerMajorVersion() < 8)
358 * Checks to see if the Object is Empty
359 * @param {Object} obj
362 isEmptyObject = function(obj) {
363 for ( var name in obj ) {
364 if ( obj.hasOwnProperty(name) ) {
373 * Clones a object and eliminate all references to the original contexts
374 * @param {Object} obj
377 cloneObject = function(obj) {
380 hash = JSON.stringify(obj);
381 newObj = JSON.parse(hash);
390 // ====================================================================
395 * Turns "http://mysite.com/dir/page.html?asd" into "http://mysite.com"
396 * @return {String} rootUrl
398 getRootUrl = function(){
400 var rootUrl = window.document.location.protocol+'//'+(window.document.location.hostname||window.document.location.host);
401 if ( window.document.location.port||false ) {
402 rootUrl += ':'+window.document.location.port;
412 * Fetches the `href` attribute of the `<base href="...">` element if it exists
413 * @return {String} baseHref
415 getBaseHref = function(){
418 baseElements = window.document.getElementsByTagName('base'),
422 // Test for Base Element
423 if ( baseElements.length === 1 ) {
424 // Prepare for Base Element
425 baseElement = baseElements[0];
426 baseHref = baseElement.href.replace(/[^\/]+$/,'');
429 // Adjust trailing slash
430 baseHref = baseHref.replace(/\/+$/,'');
431 if ( baseHref ) baseHref += '/';
439 * Fetches the baseHref or basePageUrl or rootUrl (whichever one exists first)
440 * @return {String} baseUrl
442 getBaseUrl = function(){
444 var baseUrl = this.getBaseHref()||this.getBasePageUrl()||this.getRootUrl();
452 * Fetches the URL of the current page
453 * @return {String} pageUrl
455 getPageUrl = function(){
458 State = this.getState(false,false),
459 stateUrl = (State||{}).url||this.getLocationHref(),
463 pageUrl = stateUrl.replace(/\/+$/,'').replace(/[^\/]+$/,function(part,index,string){
464 return (/\./).test(part) ? part : part+'/';
473 * Fetches the Url of the directory of the current page
474 * @return {String} basePageUrl
476 getBasePageUrl = function(){
478 var basePageUrl = (this.getLocationHref()).replace(/[#\?].*/,'').replace(/[^\/]+$/,function(part,index,string){
479 return (/[^\/]$/).test(part) ? '' : part;
480 }).replace(/\/+$/,'')+'/';
488 * Ensures that we have an absolute URL and not a relative URL
489 * @param {string} url
490 * @param {Boolean} allowBaseHref
491 * @return {string} fullUrl
493 getFullUrl = function(url,allowBaseHref){
495 var fullUrl = url, firstChar = url.substring(0,1);
496 allowBaseHref = (typeof allowBaseHref === 'undefined') ? true : allowBaseHref;
499 if ( /[a-z]+\:\/\//.test(url) ) {
502 else if ( firstChar === '/' ) {
504 fullUrl = this.getRootUrl()+url.replace(/^\/+/,'');
506 else if ( firstChar === '#' ) {
508 fullUrl = this.getPageUrl().replace(/#.*/,'')+url;
510 else if ( firstChar === '?' ) {
512 fullUrl = this.getPageUrl().replace(/[\?#].*/,'')+url;
516 if ( allowBaseHref ) {
517 fullUrl = this.getBaseUrl()+url.replace(/^(\.\/)+/,'');
519 fullUrl = this.getBasePageUrl()+url.replace(/^(\.\/)+/,'');
521 // We have an if condition above as we do not want hashes
522 // which are relative to the baseHref in our URLs
523 // as if the baseHref changes, then all our bookmarks
524 // would now point to different locations
525 // whereas the basePageUrl will always stay the same
529 return fullUrl.replace(/\#$/,'');
534 * Ensures that we have a relative URL and not a absolute URL
535 * @param {string} url
536 * @return {string} url
538 getShortUrl = function(url){
540 var shortUrl = url, baseUrl = this.getBaseUrl(), rootUrl = this.getRootUrl();
543 if ( this.emulated.pushState ) {
544 // We are in a if statement as when pushState is not emulated
545 // The actual url these short urls are relative to can change
546 // So within the same session, we the url may end up somewhere different
547 shortUrl = shortUrl.replace(baseUrl,'');
551 shortUrl = shortUrl.replace(rootUrl,'/');
553 // Ensure we can still detect it as a state
554 if ( this.isTraditionalAnchor(shortUrl) ) {
555 shortUrl = './'+shortUrl;
559 shortUrl = shortUrl.replace(/^(\.\/)+/g,'./').replace(/\#$/,'');
566 * getLocationHref(document)
567 * Returns a normalized version of document.location.href
568 * accounting for browser inconsistencies, etc.
570 * This URL will be URI-encoded and will include the hash
572 * @param {object} document
573 * @return {string} url
575 getLocationHref = function(doc) {
576 doc = doc || window.document;
578 // most of the time, this will be true
579 if (doc.URL === doc.location.href)
580 return doc.location.href;
582 // some versions of webkit URI-decode document.location.href
583 // but they leave document.URL in an encoded state
584 if (doc.location.href === decodeURIComponent(doc.URL))
587 // FF 3.6 only updates document.URL when a page is reloaded
588 // document.location.href is updated correctly
589 if (doc.location.hash && decodeURIComponent(doc.location.href.replace(/^[^#]+/, "")) === doc.location.hash)
590 return doc.location.href;
592 if (doc.URL.indexOf('#') == -1 && doc.location.href.indexOf('#') != -1)
593 return doc.location.href;
595 return doc.URL || doc.location.href;
602 * Noramlize the store by adding necessary values
604 normalizeStore = function(){
605 this.store.idToState = this.store.idToState||{};
606 this.store.urlToId = this.store.urlToId||{};
607 this.store.stateToId = this.store.stateToId||{};
612 * Get an object containing the data, title and url of the current state
613 * @param {Boolean} friendly
614 * @param {Boolean} create
615 * @return {Object} State
617 getState : function(friendly,create){
619 if ( typeof friendly === 'undefined' ) { friendly = true; }
620 if ( typeof create === 'undefined' ) { create = true; }
623 var State = this.getLastSavedState();
626 if ( !State && create ) {
627 State = this.createStateObject();
632 State = this.cloneObject(State);
633 State.url = this.cleanUrl||State.url;
641 * getIdByState(State)
642 * Gets a ID for a State
643 * @param {State} newState
644 * @return {String} id
646 getIdByState = function(newState){
649 var id = this.extractId(newState.url),
653 // Find ID via State String
654 str = this.getStateString(newState);
655 if ( typeof this.stateToId[str] !== 'undefined' ) {
656 id = this.stateToId[str];
658 else if ( typeof this.store.stateToId[str] !== 'undefined' ) {
659 id = this.store.stateToId[str];
664 id = (new Date()).getTime() + String(Math.random()).replace(/\D/g,'');
665 if ( typeof this.idToState[id] === 'undefined' && typeof this.store.idToState[id] === 'undefined' ) {
670 // Apply the new State to the ID
671 this.stateToId[str] = id;
672 this.idToState[id] = newState;
681 * normalizeState(State)
682 * Expands a State Object
683 * @param {object} State
686 normalizeState = function(oldState){
688 var newState, dataNotEmpty;
691 if ( !oldState || (typeof oldState !== 'object') ) {
696 if ( typeof oldState.normalized !== 'undefined' ) {
701 if ( !oldState.data || (typeof oldState.data !== 'object') ) {
705 // ----------------------------------------------------------------
709 newState.normalized = true;
710 newState.title = oldState.title||'';
711 newState.url = this.getFullUrl(oldState.url?oldState.url:(this.getLocationHref()));
712 newState.hash = this.getShortUrl(newState.url);
713 newState.data = this.cloneObject(oldState.data);
716 newState.id = this.getIdByState(newState);
718 // ----------------------------------------------------------------
721 newState.cleanUrl = newState.url.replace(/\??\&_suid.*/,'');
722 newState.url = newState.cleanUrl;
724 // Check to see if we have more than just a url
725 dataNotEmpty = !this.isEmptyObject(newState.data);
728 if ( (newState.title || dataNotEmpty) && this.disableSuid !== true ) {
730 newState.hash = this.getShortUrl(newState.url).replace(/\??\&_suid.*/,'');
731 if ( !/\?/.test(newState.hash) ) {
732 newState.hash += '?';
734 newState.hash += '&_suid='+newState.id;
737 // Create the Hashed URL
738 newState.hashedUrl = this.getFullUrl(newState.hash);
740 // ----------------------------------------------------------------
742 // Update the URL if we have a duplicate
743 if ( (this.emulated.pushState || this.bugs.safariPoll) && this.hasUrlDuplicate(newState) ) {
744 newState.url = newState.hashedUrl;
747 // ----------------------------------------------------------------
754 * createStateObject(data,title,url)
755 * Creates a object based on the data, title and url state params
756 * @param {object} data
757 * @param {string} title
758 * @param {string} url
761 createStateObject = function(data,title,url){
770 State = this.normalizeState(State);
778 * Get a state by it's UID
781 getStateById = function(id){
786 var State = this.idToState[id] || this.store.idToState[id] || undefined;
793 * Get a State's String
794 * @param {State} passedState
796 getStateString = function(passedState){
798 var State, cleanedState, str;
801 State = this.normalizeState(passedState);
806 title: passedState.title,
811 str = JSON.stringify(cleanedState);
819 * @param {State} passedState
820 * @return {String} id
822 getStateId = function(passedState){
827 State = this.normalizeState(passedState);
837 * getHashByState(State)
838 * Creates a Hash for the State Object
839 * @param {State} passedState
840 * @return {String} hash
842 getHashByState = function(passedState){
847 State = this.normalizeState(passedState);
857 * extractId(url_or_hash)
858 * Get a State ID by it's URL or Hash
859 * @param {string} url_or_hash
860 * @return {string} id
862 this.extractId = function ( url_or_hash ) {
864 var id,parts,url, tmp;
868 // If the URL has a #, use the id from before the #
869 if (url_or_hash.indexOf('#') != -1)
871 tmp = url_or_hash.split("#")[0];
878 parts = /(.*)\&_suid=([0-9]+)$/.exec(tmp);
879 url = parts ? (parts[1]||url_or_hash) : url_or_hash;
880 id = parts ? String(parts[2]||'') : '';
887 * isTraditionalAnchor
888 * Checks to see if the url is a traditional anchor or not
889 * @param {String} url_or_hash
892 isTraditionalAnchor = function(url_or_hash){
894 var isTraditional = !(/[\/\?\.]/.test(url_or_hash));
897 return isTraditional;
902 * Get a State by it's URL or Hash
903 * @param {String} url_or_hash
904 * @return {State|null}
906 extractState = function(url_or_hash,create){
908 var State = null, id, url;
909 create = create||false;
912 id = this.extractId(url_or_hash);
914 State = this.getStateById(id);
917 // Fetch SUID returned no State
920 url = this.getFullUrl(url_or_hash);
923 id = this.getIdByUrl(url)||false;
925 State = this.getStateById(id);
929 if ( !State && create && !this.isTraditionalAnchor(url_or_hash) ) {
930 State = this.createStateObject(null,null,url);
940 * Get a State ID by a State URL
942 getIdByUrl = function(url){
944 var id = this.urlToId[url] || this.store.urlToId[url] || undefined;
951 * getLastSavedState()
952 * Get an object containing the data, title and url of the current state
953 * @return {Object} State
955 getLastSavedState = function(){
956 return this.savedStates[this.savedStates.length-1]||undefined;
960 * getLastStoredState()
961 * Get an object containing the data, title and url of the current state
962 * @return {Object} State
964 getLastStoredState = function(){
965 return this.storedStates[this.storedStates.length-1]||undefined;
970 * Checks if a Url will have a url conflict
971 * @param {Object} newState
972 * @return {Boolean} hasDuplicate
974 hasUrlDuplicate = function(newState) {
976 var hasDuplicate = false,
980 oldState = this.extractState(newState.url);
983 hasDuplicate = oldState && oldState.id !== newState.id;
992 * @param {Object} newState
993 * @return {Object} newState
995 storeState = function(newState){
997 this.urlToId[newState.url] = newState.id;
1000 this.storedStates.push(this.cloneObject(newState));
1007 * isLastSavedState(newState)
1008 * Tests to see if the state is the last state
1009 * @param {Object} newState
1010 * @return {boolean} isLast
1012 isLastSavedState = function(newState){
1015 newId, oldState, oldId;
1018 if ( this.savedStates.length ) {
1019 newId = newState.id;
1020 oldState = this.getLastSavedState();
1021 oldId = oldState.id;
1024 isLast = (newId === oldId);
1034 * @param {Object} newState
1035 * @return {boolean} changed
1037 saveState = function(newState){
1039 if ( this.isLastSavedState(newState) ) {
1044 this.savedStates.push(this.cloneObject(newState));
1052 * Gets a state by the index
1053 * @param {integer} index
1056 getStateByIndex = function(index){
1061 if ( typeof index === 'undefined' ) {
1062 // Get the last inserted
1063 State = this.savedStates[this.savedStates.length-1];
1065 else if ( index < 0 ) {
1067 State = this.savedStates[this.savedStates.length+index];
1070 // Get from the beginning
1071 State = this.savedStates[index];
1080 * Gets the current index
1083 getCurrentIndex = function(){
1088 if(this.savedStates.length < 1) {
1092 index = this.savedStates.length-1;
1097 // ====================================================================
1102 * @param {Location=} location
1103 * Gets the current document hash
1104 * Note: unlike location.hash, this is guaranteed to return the escaped hash in all browsers
1107 getHash = function(doc){
1108 var url = this.getLocationHref(doc),
1110 hash = this.getHashByUrl(url);
1116 * normalize and Unescape a Hash
1117 * @param {String} hash
1120 unescapeHash = function(hash){
1122 var result = this.normalizeHash(hash);
1125 result = decodeURIComponent(result);
1133 * normalize a hash across browsers
1136 normalizeHash = function(hash){
1138 var result = hash.replace(/[^#]*#/,'').replace(/#.*/, '');
1146 * Sets the document hash
1147 * @param {string} hash
1148 * @return {Roo.History}
1150 setHash = function(hash,queue){
1155 if ( queue !== false && this.busy() ) {
1156 // Wait + Push to Queue
1157 //this.debug('this.setHash: we must wait', arguments);
1160 callback: this.setHash,
1168 //this.debug('this.setHash: called',hash);
1170 // Make Busy + Continue
1173 // Check if hash is a state
1174 State = this.extractState(hash,true);
1175 if ( State && !this.emulated.pushState ) {
1176 // Hash is a state so skip the setHash
1177 //this.debug('this.setHash: Hash is a state so skipping the hash set with a direct pushState call',arguments);
1180 this.pushState(State.data,State.title,State.url,false);
1182 else if ( this.getHash() !== hash ) {
1183 // Hash is a proper hash, so apply it
1185 // Handle browser bugs
1186 if ( this.bugs.setHash ) {
1187 // Fix Safari Bug https://bugs.webkit.org/show_bug.cgi?id=56249
1189 // Fetch the base page
1190 pageUrl = this.getPageUrl();
1192 // Safari hash apply
1193 this.pushState(null,null,pageUrl+'#'+hash,false);
1196 // Normal hash apply
1197 window.document.location.hash = hash;
1207 * normalize and Escape a Hash
1210 escapeHash = function(hash){
1212 var result = normalizeHash(hash);
1215 result = window.encodeURIComponent(result);
1218 if ( !this.bugs.hashEscape ) {
1219 // Restore common parts
1221 .replace(/\%21/g,'!')
1222 .replace(/\%26/g,'&')
1223 .replace(/\%3D/g,'=')
1224 .replace(/\%3F/g,'?');
1233 * Extracts the Hash from a URL
1234 * @param {string} url
1235 * @return {string} url
1237 getHashByUrl = function(url){
1239 var hash = String(url)
1240 .replace(/([^#]*)#?([^#]*)#?(.*)/, '$2')
1244 hash = this.unescapeHash(hash);
1252 * Applies the title to the document
1253 * @param {State} newState
1256 setTitle = function(newState){
1258 var title = newState.title,
1263 firstState = this.getStateByIndex(0);
1264 if ( firstState && firstState.url === newState.url ) {
1265 title = firstState.title||this.initialTitle;
1271 window.document.getElementsByTagName('title')[0].innerHTML = title.replace('<','<').replace('>','>').replace(' & ',' & ');
1273 catch ( Exception ) { }
1274 window.document.title = title;
1281 // ====================================================================
1287 * @param {boolean} value [optional]
1288 * @return {boolean} busy
1290 busy = function(value){
1292 if ( typeof value !== 'undefined' ) {
1293 //this.debug('this.busy: changing ['+(this.busy.flag||false)+'] to ['+(value||false)+']', this.queues.length);
1294 this.busy.flag = value;
1297 else if ( typeof this.busy.flag === 'undefined' ) {
1298 this.busy.flag = false;
1302 if ( !this.busy.flag ) {
1303 // Execute the next item in the queue
1304 window.clearTimeout(this.busy.timeout);
1305 var fireNext = function(){
1307 if ( this.busy.flag ) return;
1308 for ( i=this.queues.length-1; i >= 0; --i ) {
1309 queue = this.queues[i];
1310 if ( queue.length === 0 ) continue;
1311 item = queue.shift();
1312 this.fireQueueItem(item);
1313 this.busy.timeout = window.setTimeout(fireNext,this.busyDelay);
1316 this.busy.timeout = window.setTimeout(fireNext,this.busyDelay);
1320 return this.busy.flag;
1326 * History.fireQueueItem(item)
1328 * @param {Object} item
1329 * @return {Mixed} result
1331 History.fireQueueItem = function(item){
1332 return item.callback.apply(item.scope||History,item.args||[]);
1336 * History.pushQueue(callback,args)
1337 * Add an item to the queue
1338 * @param {Object} item [scope,callback,args,queue]
1340 History.pushQueue = function(item){
1341 // Prepare the queue
1342 History.queues[item.queue||0] = History.queues[item.queue||0]||[];
1345 History.queues[item.queue||0].push(item);
1352 * History.queue (item,queue), (func,queue), (func), (item)
1353 * Either firs the item now if not busy, or adds it to the queue
1355 History.queue = function(item,queue){
1357 if ( typeof item === 'function' ) {
1362 if ( typeof queue !== 'undefined' ) {
1367 if ( History.busy() ) {
1368 History.pushQueue(item);
1370 History.fireQueueItem(item);
1378 * History.clearQueue()
1381 History.clearQueue = function(){
1382 History.busy.flag = false;
1383 History.queues = [];
1388 // ====================================================================
1392 * History.stateChanged
1393 * States whether or not the state has changed since the last double check was initialised
1395 History.stateChanged = false;
1398 * History.doubleChecker
1399 * Contains the timeout used for the double checks
1401 History.doubleChecker = false;
1404 * History.doubleCheckComplete()
1405 * Complete a double check
1408 History.doubleCheckComplete = function(){
1410 History.stateChanged = true;
1413 History.doubleCheckClear();
1420 * History.doubleCheckClear()
1421 * Clear a double check
1424 History.doubleCheckClear = function(){
1426 if ( History.doubleChecker ) {
1427 window.clearTimeout(History.doubleChecker);
1428 History.doubleChecker = false;
1436 * History.doubleCheck()
1437 * Create a double check
1440 History.doubleCheck = function(tryAgain){
1442 History.stateChanged = false;
1443 History.doubleCheckClear();
1445 // Fix IE6,IE7 bug where calling history.back or history.forward does not actually change the hash (whereas doing it manually does)
1446 // Fix Safari 5 bug where sometimes the state does not change: https://bugs.webkit.org/show_bug.cgi?id=42940
1447 if ( History.bugs.ieDoubleCheck ) {
1449 History.doubleChecker = window.setTimeout(
1451 History.doubleCheckClear();
1452 if ( !History.stateChanged ) {
1453 //History.debug('History.doubleCheck: State has not yet changed, trying again', arguments);
1459 this.doubleCheckInterval
1468 // ====================================================================
1472 * History.safariStatePoll()
1473 * Poll the current state
1476 History.safariStatePoll = function(){
1479 // Get the Last State which has the new URL
1481 urlState = History.extractState(History.getLocationHref()),
1484 // Check for a difference
1485 if ( !History.isLastSavedState(urlState) ) {
1486 newState = urlState;
1492 // Check if we have a state with that url
1495 //History.debug('History.safariStatePoll: new');
1496 newState = History.createStateObject();
1499 // Apply the New State
1500 //History.debug('History.safariStatePoll: trigger');
1501 History.Adapter.trigger(window,'popstate');
1508 // ====================================================================
1512 * History.back(queue)
1513 * Send the browser history back one item
1514 * @param {Integer} queue [optional]
1516 History.back = function(queue){
1517 //History.debug('History.back: called', arguments);
1520 if ( queue !== false && History.busy() ) {
1521 // Wait + Push to Queue
1522 //History.debug('History.back: we must wait', arguments);
1525 callback: History.back,
1532 // Make Busy + Continue
1535 // Fix certain browser bugs that prevent the state from changing
1536 History.doubleCheck(function(){
1537 History.back(false);
1548 * History.forward(queue)
1549 * Send the browser history forward one item
1550 * @param {Integer} queue [optional]
1552 History.forward = function(queue){
1553 //History.debug('History.forward: called', arguments);
1556 if ( queue !== false && History.busy() ) {
1557 // Wait + Push to Queue
1558 //History.debug('History.forward: we must wait', arguments);
1561 callback: History.forward,
1568 // Make Busy + Continue
1571 // Fix certain browser bugs that prevent the state from changing
1572 History.doubleCheck(function(){
1573 History.forward(false);
1579 // End forward closure
1584 * History.go(index,queue)
1585 * Send the browser history back or forward index times
1586 * @param {Integer} queue [optional]
1588 History.go = function(index,queue){
1589 //History.debug('History.go: called', arguments);
1597 for ( i=1; i<=index; ++i ) {
1598 History.forward(queue);
1601 else if ( index < 0 ) {
1603 for ( i=-1; i>=index; --i ) {
1604 History.back(queue);
1608 throw new Error('History.go: History.go requires a positive or negative integer passed.');
1616 // ====================================================================
1617 // HTML5 State Support
1619 // Non-Native pushState Implementation
1620 if ( History.emulated.pushState ) {
1622 * Provide Skeleton for HTML4 Browsers
1626 var emptyFunction = function(){};
1627 History.pushState = History.pushState||emptyFunction;
1628 History.replaceState = History.replaceState||emptyFunction;
1629 } // History.emulated.pushState
1631 // Native pushState Implementation
1634 * Use native HTML5 History API Implementation
1638 * History.onPopState(event,extra)
1639 * Refresh the Current State
1641 History.onPopState = function(event,extra){
1643 var stateId = false, newState = false, currentHash, currentState;
1645 // Reset the double check
1646 History.doubleCheckComplete();
1648 // Check for a Hash, and handle apporiatly
1649 currentHash = History.getHash();
1650 if ( currentHash ) {
1652 currentState = History.extractState(currentHash||History.getLocationHref(),true);
1653 if ( currentState ) {
1654 // We were able to parse it, it must be a State!
1655 // Let's forward to replaceState
1656 //History.debug('History.onPopState: state anchor', currentHash, currentState);
1657 History.replaceState(currentState.data, currentState.title, currentState.url, false);
1660 // Traditional Anchor
1661 //History.debug('History.onPopState: traditional anchor', currentHash);
1662 History.Adapter.trigger(window,'anchorchange');
1663 History.busy(false);
1666 // We don't care for hashes
1667 History.expectedStateId = false;
1672 stateId = History.Adapter.extractEventData('state',event,extra) || false;
1676 // Vanilla: Back/forward button was used
1677 newState = History.getStateById(stateId);
1679 else if ( History.expectedStateId ) {
1680 // Vanilla: A new state was pushed, and popstate was called manually
1681 newState = History.getStateById(History.expectedStateId);
1685 newState = History.extractState(History.getLocationHref());
1688 // The State did not exist in our store
1690 // Regenerate the State
1691 newState = History.createStateObject(null,null,History.getLocationHref());
1695 History.expectedStateId = false;
1697 // Check if we are the same state
1698 if ( History.isLastSavedState(newState) ) {
1699 // There has been no change (just the page's hash has finally propagated)
1700 //History.debug('History.onPopState: no change', newState, History.savedStates);
1701 History.busy(false);
1706 History.storeState(newState);
1707 History.saveState(newState);
1709 // Force update of the title
1710 History.setTitle(newState);
1713 History.Adapter.trigger(window,'statechange');
1714 History.busy(false);
1719 History.Adapter.bind(window,'popstate',History.onPopState);
1722 * History.pushState(data,title,url)
1723 * Add a new State to the history object, become it, and trigger onpopstate
1724 * We have to trigger for HTML4 compatibility
1725 * @param {object} data
1726 * @param {string} title
1727 * @param {string} url
1730 History.pushState = function(data,title,url,queue){
1731 //History.debug('History.pushState: called', arguments);
1734 if ( History.getHashByUrl(url) && History.emulated.pushState ) {
1735 throw new Error('History.js does not support states with fragement-identifiers (hashes/anchors).');
1739 if ( queue !== false && History.busy() ) {
1740 // Wait + Push to Queue
1741 //History.debug('History.pushState: we must wait', arguments);
1744 callback: History.pushState,
1751 // Make Busy + Continue
1754 // Create the newState
1755 var newState = History.createStateObject(data,title,url);
1758 if ( History.isLastSavedState(newState) ) {
1759 // Won't be a change
1760 History.busy(false);
1763 // Store the newState
1764 History.storeState(newState);
1765 History.expectedStateId = newState.id;
1767 // Push the newState
1768 history.pushState(newState.id,newState.title,newState.url);
1771 History.Adapter.trigger(window,'popstate');
1774 // End pushState closure
1779 * History.replaceState(data,title,url)
1780 * Replace the State and trigger onpopstate
1781 * We have to trigger for HTML4 compatibility
1782 * @param {object} data
1783 * @param {string} title
1784 * @param {string} url
1787 History.replaceState = function(data,title,url,queue){
1788 //History.debug('History.replaceState: called', arguments);
1791 if ( History.getHashByUrl(url) && History.emulated.pushState ) {
1792 throw new Error('History.js does not support states with fragement-identifiers (hashes/anchors).');
1796 if ( queue !== false && History.busy() ) {
1797 // Wait + Push to Queue
1798 //History.debug('History.replaceState: we must wait', arguments);
1801 callback: History.replaceState,
1808 // Make Busy + Continue
1811 // Create the newState
1812 var newState = History.createStateObject(data,title,url);
1815 if ( History.isLastSavedState(newState) ) {
1816 // Won't be a change
1817 History.busy(false);
1820 // Store the newState
1821 History.storeState(newState);
1822 History.expectedStateId = newState.id;
1824 // Push the newState
1825 history.replaceState(newState.id,newState.title,newState.url);
1828 History.Adapter.trigger(window,'popstate');
1831 // End replaceState closure
1835 } // !History.emulated.pushState
1838 // ====================================================================
1844 if ( sessionStorage ) {
1847 History.store = JSON.parse(sessionStorage.getItem('History.store'))||{};
1854 History.normalizeStore();
1859 History.normalizeStore();
1863 * Clear Intervals on exit to prevent memory leaks
1865 History.Adapter.bind(window,"unload",History.clearAllIntervals);
1868 * Create the initial State
1870 History.saveState(History.storeState(History.extractState(History.getLocationHref(),true)));
1873 * Bind for Saving Store
1875 if ( sessionStorage ) {
1876 // When the page is closed
1877 History.onUnload = function(){
1879 var currentStore, item, currentStoreString;
1883 currentStore = JSON.parse(sessionStorage.getItem('History.store'))||{};
1890 currentStore.idToState = currentStore.idToState || {};
1891 currentStore.urlToId = currentStore.urlToId || {};
1892 currentStore.stateToId = currentStore.stateToId || {};
1895 for ( item in History.idToState ) {
1896 if ( !History.idToState.hasOwnProperty(item) ) {
1899 currentStore.idToState[item] = History.idToState[item];
1901 for ( item in History.urlToId ) {
1902 if ( !History.urlToId.hasOwnProperty(item) ) {
1905 currentStore.urlToId[item] = History.urlToId[item];
1907 for ( item in History.stateToId ) {
1908 if ( !History.stateToId.hasOwnProperty(item) ) {
1911 currentStore.stateToId[item] = History.stateToId[item];
1915 History.store = currentStore;
1916 History.normalizeStore();
1918 // In Safari, going into Private Browsing mode causes the
1919 // Session Storage object to still exist but if you try and use
1920 // or set any property/function of it it throws the exception
1921 // "QUOTA_EXCEEDED_ERR: DOM Exception 22: An attempt was made to
1922 // add something to storage that exceeded the quota." infinitely
1924 currentStoreString = JSON.stringify(currentStore);
1927 sessionStorage.setItem('History.store', currentStoreString);
1930 if (e.code === DOMException.QUOTA_EXCEEDED_ERR) {
1931 if (sessionStorage.length) {
1932 // Workaround for a bug seen on iPads. Sometimes the quota exceeded error comes up and simply
1933 // removing/resetting the storage can work.
1934 sessionStorage.removeItem('History.store');
1935 sessionStorage.setItem('History.store', currentStoreString);
1937 // Otherwise, we're probably private browsing in Safari, so we'll ignore the exception.
1945 // For Internet Explorer
1946 History.intervalList.push(setInterval(History.onUnload,this.storeInterval));
1948 // For Other Browsers
1949 History.Adapter.bind(window,'beforeunload',History.onUnload);
1950 History.Adapter.bind(window,'unload',History.onUnload);
1952 // Both are enabled for consistency
1955 // Non-Native pushState Implementation
1956 if ( !History.emulated.pushState ) {
1957 // Be aware, the following is only for native pushState implementations
1958 // If you are wanting to include something for all browsers
1959 // Then include it above this if block
1964 if ( History.bugs.safariPoll ) {
1965 History.intervalList.push(setInterval(History.safariStatePoll, this.safariPollInterval));
1969 * Ensure Cross Browser Compatibility
1971 if ( window.navigator.vendor === 'Apple Computer, Inc.' || (window.navigator.appCodeName||'') === 'Mozilla' ) {
1973 * Fix Safari HashChange Issue
1977 History.Adapter.bind(window,'hashchange',function(){
1978 History.Adapter.trigger(window,'popstate');
1982 if ( History.getHash() ) {
1983 History.Adapter.onDomLoad(function(){
1984 History.Adapter.trigger(window,'hashchange');
1989 } // !History.emulated.pushState
1992 }; // History.initCore
1994 // Try to Initialise History
1995 if (!History.options || !History.options.delayInit) {