首頁 >web前端 >js教程 >Ajax保留瀏覽器歷史的兩個解決方案

Ajax保留瀏覽器歷史的兩個解決方案

巴扎黑
巴扎黑原創
2017-04-29 15:44:161513瀏覽

總是在github down點東西,github整個介面做的不錯,體驗也很好~對於其中的源代碼滑動的特效最為喜歡了~剛開始以為這個只是普通的ajax請求效果,但是發現這個特效能夠導致瀏覽器網址列跟著變化,再點選前進後退按鈕後又可以將程式碼滑回滑出~~於是乎就來研究下吧~

#

#   一、透過錨點Hash實現:

  在這方面其實國內很早就有做了,比如淘寶畫報,通過的是在地址欄後面加#錨點實現的,瀏覽器是可以識別錨點為單位的歷史記錄的。但不是說頁面本身有這個錨點,錨點的Hash只是起到一個引導瀏覽器將這次的記錄推入歷史記錄棧頂的作用。

  來做一個小小的demo:

    <style type="text/css">
        #tab1_header,#tab2_header{
            cursor:pointer;
            border:1px solid;
            width:50px;
        }
        #tab1,#tab2{
            width:90%;
            height:200px;
            border:1px solid;
        }
    </style>
    <p id="tab_header">
    	<span id="tab1_header">Tab1</span>
    	<span id="tab2_header">Tab2</span>
    </p>
    <p id="tab1">1</p>
    <p id="tab2">2</p>

  一個很簡單的Tab切換如果一般情況下就直接:

$("#tab1_header").click(function() {
            $("#tab2").hide();
            $("#tab1").show();
});
$("#tab2_header").click(function() {
            $("#tab1").hide();
            $("#tab2").show();
});

但如果點擊到tab2時想透過後退按鈕退到tab1時就不行了,如果刷新的話瀏覽器的行為完全不是出於用戶的想法,這樣的話,我們可以加入#錨點來模擬新頁面,為什麼要說模擬呢,如果直接透過js改變window.location瀏覽器會重新載入頁面,但加#就不會重新載入並且能保存在歷史中。 JS透過window.location.hash來控制URL後面的錨點#。

  我們把程式碼改為這樣:

$(function(){
			showTab();
			$(window).bind(&#39;hashchange&#39;, function(e){
				showTab();
			});
			$("#tab1_header").click(showTab1);
			$("#tab2_header").click(showTab2);
		});

        function showTab() {
            if (window.location.hash == "#tab2"){
				showTab2();
			} else {
				showTab1();
			}
        }
        function showTab1() {
            $("#tab2").hide();
            $("#tab1").show();
            window.location.hash = "#tab1";
        };
        function showTab2() {
            $("#tab1").hide();
            $("#tab2").show();
            window.location.hash = "#tab2";
        };

加上window.location.hash = "#tab1"這段程式碼就行了,點擊tab後,網址列後面就會加上#tab1,點擊tab2後就會改成#tab2,當瀏覽器偵測到url變化時就會觸發hashchange這一事件,就是用戶在點擊後退時能夠得到的事件就能夠透過window.location.hash進行判斷並進行ajax操作了,但是haschange這個事件並不是每個瀏覽器都有的,只有現代高階瀏覽器才有,所以在低階的瀏覽器中需要用輪詢來偵測URL是否在變化,這個這裡就不具體說了。

  二、透過HTML5加強型的History物件實作(類別Pjax)

可以透過window.history.pushState這個方法無刷新的更新瀏覽器地址欄,這個方法在更新地址欄的同時將地址壓入歷史記錄堆棧裡,而要取出這個棧頂頁面則可以用popstate這個事件來捕獲~

  來模擬一下github的環境,github中每個url是對應一個完整的實際頁面的,所以在ajax請求頁面時需要異步獲取target頁面中指定id容器中的內容:

#   例如有這樣兩個頁面:

  index.html

<!DOCTYPE HTML>
<html>
    <head>
        <meta http-equiv="Content-Type" content="text/html; charset=gbk">
        <title>index</title>
    </head>
    <body>
    	<script>document.write(new Date());</script>
    	<p id="cn">
        	<a href="second.html">加载前</a>
        </p>
    </body>
</html>

  second.html  

<!DOCTYPE HTML>
<html>
    <head>
        <meta http-equiv="Content-Type" content="text/html; charset=gbk">
        <title>second</title>
    </head>
    <body>
    	<script>document.write(new Date());</script>
        <p id="cn">
        	<a href="index.html">加载后</a>
        </p>
    </body>
</html>

假如用同步的http請求打開的話完全是兩個頁面,兩個頁面加入很多地方一樣的話我們完全可以用這種方法來實現ajax請求變更DOM,我在這裡加了3f1c4e4b6b16bbbd69b2ee476dc4f83adocument.write(new Date());2cacc6d41bbb37262a98f745aa00fbf0語句透過它的變化能得知是否取自兩個http請求,局

  部的ajax請求是不會改變這個時間顯示的。  

$(function() {
    var state = {
        title: "index",
        url: "index.html"
    };
    $("#cn").click(function() {
        window.history.pushState(state, "index", "second.html");
        var $self = $(this);
        $.ajax({
            url: "second.html",
            dataType: "html",
            complete: function(jqXHR, status, responseText) {
                responseText = jqXHR.responseText;
                if (jqXHR.isResolved()) {
                    jqXHR.done(function(r) {
                        responseText = r;
                    });
                    $self.html($("<p>").append(responseText.replace(/<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi, "")).find("#cn"));
                }
            }
        });
        document.title = "second";
        return false;
    });
    $(window).bind(&#39;popstate&#39;, function(e) {
        var st = e.state;
        //$("#cn").load(st.url + " #cn");
        $.ajax({
            url: "index.html",
            dataType: "html",
            complete: function(jqXHR, status, responseText) {
                responseText = jqXHR.responseText;
                if (jqXHR.isResolved()) {
                    jqXHR.done(function(r) {
                        responseText = r;
                    });
                    $("#cn").html($("<p>").append(responseText.replace(/<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi, "")).find("#cn"));
                }
            }
        });
        document.title = e.state.title;
    });
});

上面語句中當#cn元素被點擊時將state透過pushState方法壓入歷史記錄棧,並在第三個參數中將瀏覽器URL框中指向second頁面,並透過ajax將second頁面異步載入,將相應的部分加入容器中,這樣就實現了異步加載並改變地址欄url了,同樣用戶點擊後退時,觸發popstate,剛才pushState方法中的第一個參數state便是這裡傳入的形參e中的state屬性,透過var st = e.state取出供開發使用。同時載入index頁面中對應內容。時間有限這個js沒有進行重構,直接寫$.ajax了,其實假如不需要任何特效單純的異步載入在jQ中可以直接用$("#cn").load(st.url + " #cn ");將請求的html對應的#cn放到本頁的#cn容器中,但加入要更加炫的特效的話就要直接操作ajax傳回的資料了。

$("#cn").html($("<p>").append(responseText.replace(/<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi, "")).find("#cn"));

先創建一個p容器在將經過script過濾過的程式碼裝入這個容器在透過find方法找到裡面對應的選擇器容器插入本身的頁面中,這裡可以不用html來填充,可以根據自己的專案需要用slideUp, show什麼的特效進行內容顯示~~

# 另外這裡要推薦一個jQuery元件叫pjax(https://github.com/defunkt/jquery-pjax),比較牛叉的一個元件,異步的部分load進來另外一個頁面對應容器中的內容,實現的機理和我上面的第二種方案一致。 pushState + ajax = pjax 感覺這個應用會熱起來的。

稍微總結下,兩種方案其實對於想IE6或FF3.6等比較低階的瀏覽器支援不是很好,前者若要相容低階瀏覽器要用輪詢來監聽瀏覽器網址列行為,而後者的話是完全的HTML5應用,對於非HTML5瀏覽器只能做判斷跳轉了。 ###

  如pjax最后的一段无奈的兼容处理:  

$.support.pjax = window.history && window.history.pushState


// Fall back to normalcy for older browsers.
if ( !$.support.pjax ) {
  $.pjax = function( options ) {
    window.location = $.isFunction(options.url) ? options.url() : options.url
  }
  $.fn.pjax = function() { return this }
}

               

以上是Ajax保留瀏覽器歷史的兩個解決方案的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn