亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發(fā) > XML > 正文

AJAX:如何處理書簽和后退按鈕

2024-09-05 20:56:15
字體:
來源:轉載
供稿:網友
本文將展示一個開源JavaScript庫,該腳本庫給AJAX應用程序帶來了書簽和后退按鈕支持。在學習完這個教程后,開發(fā)人員將能夠獲得對一個AJAX問題的解決方案(甚至連Google Maps和Gmail現在都不提供該解決方案):一個強大的、可用的書簽和后退前進功能,其操作行為如同其他的Web應用程序一樣。

本文將闡述目前AJAX應用程序在使用書簽和后退按鈕方面所面臨的嚴重問題;展示Really Simple History(RSH)庫——一個可以解決以上問題的開源框架,并提供幾個運行中的例子。

本文所展示的這個框架的主要發(fā)明分為兩部分。首先是一個隱藏的HTML表單,用于緩存大量短期會話的客戶端信息;這種緩存功能為頁面導航提供了強大的支持。其次是超鏈接錨點和隱藏Iframe的組合,它們被嵌入后退和前進按鈕,用來截獲和記錄瀏覽器的歷史記錄事件。以上兩種技術都被包裝在一個簡單的JavaScript庫中來簡化開發(fā)。

問題

書簽和后退按鈕在傳統(tǒng)的多頁面Web應用程序中運行得非常好。當用戶瀏覽web站點的時候,其瀏覽器的地址欄記錄隨新的URL而更新,這些記錄可以被粘貼到電子郵件或者書簽中供以后使用。后退和前進按鈕也可以正常操作,使用戶可以在訪問過的頁面中向前或向后翻動。

但是AJAX應用程序卻不一樣,它們是運行在單個Web頁面中的復雜程序。瀏覽器并不是為這類程序而構建的——這類Web應用程序已經過時,它們在每次鼠標點擊的時候都需要重新刷新整個頁面。

在這種類似于Gmail的AJAX軟件中,瀏覽器的地址欄在用戶選擇功能和改變程序狀態(tài)的時候保持不變,這使得無法在特定的應用程序視圖中使用書簽。此外,如果用戶按下“后退”按鈕來“撤銷”上次的操作,他們會驚奇地發(fā)現,瀏覽器會完全離開該應用程序的Web頁面。

解決方案

開源RSH框架可以解決這些問題,它為AJAX應用程序提供了書簽和控制后退、前進按鈕的功能。RSH目前還處于Beta階段,可以在Firefox 1.0、Netscape 7+、Internet Explorer 6+等瀏覽器上運行;目前還不支持Safari。

目前有幾個AJAX框架對書簽和歷史記錄問題有所幫助;但這些框架目前都有幾個由于實現而造成的重大Bug。此外,很多AJAX歷史記錄框架被綁定到較大的庫上,例如Backbase和Dojo;這些框架為AJAX應用程序引入了完全不同的編程模型,迫使開發(fā)人員使用全新的方式來獲得歷史記錄功能。

相較之下,RSH是一個可以包含在現有AJAX系統(tǒng)中的簡單模塊。此外,RSH庫采用了一些技術以避免產生影響其他歷史記錄框架的Bug。

RSH框架由兩個JavaScript類組成:DhtmlHistory和HistoryStorage。

DhtmlHistory類為AJAX應用程序提供歷史記錄抽象。AJAX頁面使用add()方法添加歷史記錄事件到瀏覽器,指定新的地址和相關的歷史記錄數據。DhtmlHistory類使用一個錨散列(如#new-location)更新瀏覽器當前的URL,同時把歷史記錄數據和該新URL關聯(lián)。AJAX應用程序將自己注冊為歷史記錄的監(jiān)聽器,當用戶使用后退和前進按鈕進行瀏覽時,歷史記錄事件被觸發(fā),為瀏覽器提供新的位置以及與add()調用一起保存的任何歷史記錄數據。

第二個類:HistoryStorage,允許開發(fā)人員保存任意數量的已存歷史記錄數據。在普通Web頁面中,當用戶導航到一個新的web站點時,瀏覽器卸載并清除web頁面上的所有應用程序和JavaScript狀態(tài);如果用戶用后退按鈕返回,所有的數據都丟失了。HistoryStorage類通過一個包含簡單散列表方法(例如put()、get()、hasKey())的API來解決這類問題。上面的方法允許開發(fā)人員在用戶離開Web頁面之后保存任意數量的數據;當用戶按后退按鈕重新返回時,歷史記錄數據可以通過HistoryStorage類來訪問。在內部,我們通過使用隱藏的表單字段來實現此功能,這是因為瀏覽器會自動保存表單字段中的值,甚至在用戶離開Web頁面的時候也如此。

例子

讓我們先從一個簡單的例子開始。

首先,任何需要使用RSH框架的頁面都必須包含dhtmlHistory.js腳本:

以下為引用的內容:
<!-- Load the Really Simple
History framework -->
<script type="text/javascript"
src="../../framework/dhtmlHistory.js">
</script>

DHTML歷史記錄應用程序也必須在與AJAX Web頁面相同的目錄下包含blank.html文件;這個文件與RSH框架打包在一起,且對于Internet Explorer來說是必需的。順便提一下,RSH使用一個隱藏Iframe來跟蹤和添加Internet Explorer的歷史記錄變化;這個Iframe需要我們指定一個實際的文件位置才能正常工作,這就是blank.html。

RSH框架創(chuàng)建了一個叫做dhtmlHistory的全局對象,這是操縱瀏覽器歷史記錄的入口點。使用dhtmlHistory的第一步是在Web頁面加載完成后初始化dhtmlHistory對象:

以下為引用的內容:

window.onload = initialize;

function initialize() {
 // initialize the DHTML History
 // framework
 dhtmlHistory.initialize();

然后,開發(fā)人員使用dhtmlHistory.addListener()方法訂閱歷史記錄變化事件。這個方法帶有一個JavaScript回調函數,當DHTML歷史記錄變化事件發(fā)生時,該函數接收兩個參數:新的頁面位置以及任何可與該事件關聯(lián)的可選歷史記錄數據:

以下為引用的內容:

window.onload = initialize;

function initialize() {
 // initialize the DHTML History
 // framework
 dhtmlHistory.initialize();

 // subscribe to DHTML history change
 // events
 dhtmlHistory.addListener(historyChange);
  historyChange()方法很簡單,該函數在用戶導航到一個新位置后接收newLocation以及任何與該事件關聯(lián)的可選historyData。

/** Our callback to receive history change
events. */
function historyChange(newLocation,
historyData) {
 debug("A history change has occurred: "
  + "newLocation="+newLocation
  + ", historyData="+historyData,
  true);
}

上面用到的debug()方法是定義在示例源文件中的一個實用函數,它與完整示例打包在一起供下載。debug()只是用來將消息打印到Web頁面上;第二個布爾型參數(在上述代碼中值為true)控制是否在打印新的調試消息之前清除原有的全部消息。

開發(fā)人員使用add()方法添加歷史記錄事件。添加歷史記錄事件涉及為歷史記錄變化指定一個新地址,例如edit:SomePage,以及提供一個和該事件一起保存的可選historyData值。

以下為引用的內容:

window.onload = initialize;

function initialize() {
 // initialize the DHTML History
 // framework
 dhtmlHistory.initialize();

 // subscribe to DHTML history change
 // events
 dhtmlHistory.addListener(historyChange);

 // if this is the first time we have
 // loaded the page...
 if (dhtmlHistory.isFirstLoad()) {
  debug("Adding values to browser " + "history", false);
  // start adding history
  dhtmlHistory.add("helloworld", "Hello World Data");
  dhtmlHistory.add("foobar", 33);
  dhtmlHistory.add("boobah", true);

  var complexObject = new Object();
  complexObject.value1 = "This is the first value";
  complexObject.value2 = "This is the second data";
  complexObject.value3 = new Array();
  complexObject.value3[0] = "array 1";
  complexObject.value3[1] = "array 2";

  dhtmlHistory.add("complexObject", complexObject);

在add()被調用之后,新的地址將立即作為一個錨值(鏈接地址)顯示在瀏覽器的URL地址欄中。例如,對地址為http://codinginparadise.org/my_ajax_app的AJAX Web頁面調用dhtmlHistory.add("helloworld", "Hello World Data")之后,用戶將會在其瀏覽器URL地址欄中看到如下的地址:

http://codinginparadise.org/my_ajax_app#helloworld

然后用戶可以將這個頁面做成書簽,如果以后用到這個書簽,AJAX應用程序可以讀取#helloworld值,并用它來初始化Web頁面。散列后面的地址值是RSH框架可以透明編碼和解碼的URL地址。

HistoryData非常有用,它保存比簡單的URL更為復雜的AJAX地址變化狀態(tài)。這是一個可選值,可以是任何JavaScript類型,例如Number、String或Object。使用該保存功能的一個例子是在一個富文本編輯器中保存所有文本(比如在用戶離開當前頁面時)。當用戶再回到這個地址時,瀏覽器將會將該對象返回給歷史記錄變化監(jiān)聽器。

開發(fā)人員可以為historyData提供帶有嵌套對象和表示復雜狀態(tài)的數組的完整JavaScript對象;JSON (JavaScript Object Notation)所支持的在歷史記錄數據中都支持,包括簡單數據類型和null類型。然而,DOM對象以及可用腳本編寫的瀏覽器對象(如XMLHttpRequest)不會被保存。請注意,historyData并不隨書簽一起保存,當瀏覽器關閉,瀏覽器緩存被清空,或者用戶清除歷史記錄的時候,它就會消失。

使用dhtmlHistory的最后一步是isFirstLoad()方法。在某些瀏覽器中,如果導航到一個Web頁面,再跳轉到另一個不同的頁面,然后按“后退”按鈕返回到起始的站點,第一頁將完全重新加載,并觸發(fā)onload事件。這樣會對想要在第一次加載頁面時用某種方式對其進行初始化(而其后則不使用這種方式重新加載該頁面)的代碼造成破壞。isFirstLoad()方法可以區(qū)分是第一次加載一個Web頁面還是用戶導航到保存在歷史記錄中的Web頁面時觸發(fā)的“假加載”事件。

在示例代碼中,我們只想在第一次加載頁面的時候添加歷史記錄事件;如果用戶在加載頁面后按后退按鈕返回該頁面,我們就不想重新添加任何歷史記錄事件:

以下為引用的內容:

window.onload = initialize;

function initialize() {
 // initialize the DHTML History
 // framework
 dhtmlHistory.initialize();

 // subscribe to DHTML history change
 // events
 dhtmlHistory.addListener(historyChange);

 // if this is the first time we have
 // loaded the page...
 if (dhtmlHistory.isFirstLoad()) {
  debug("Adding values to browser "+ "history", false);
  // start adding history
  dhtmlHistory.add("helloworld", "Hello World Data");
  dhtmlHistory.add("foobar", 33);
  dhtmlHistory.add("boobah", true);

  var complexObject = new Object();
  complexObject.value1 = "This is the first value";
  complexObject.value2 = "This is the second data";
  complexObject.value3 = new Array();
  complexObject.value3[0] = "array 1";
  complexObject.value3[1] = "array 2";

  dhtmlHistory.add("complexObject", complexObject);

讓我們繼續(xù)使用historyStorage類。類似于dhtmlHistory,historyStorage通過一個叫historyStorage的全局對象來公開它的功能。該對象有幾個模擬散列的方法,比如put(keyName、keyValue)、get(keyName)和hasKey(keyName)。鍵名稱必須是字符串,同時鍵值可以是復雜的JavaScript對象甚至是XML格式的字符串。在我們的源代碼例子中,在第一次加載頁面時,我們使用put()將簡單的XML放入historyStorage:

以下為引用的內容:

window.onload = initialize;

function initialize() {
 // initialize the DHTML History
 // framework
 dhtmlHistory.initialize();

 // subscribe to DHTML history change
 // events
 dhtmlHistory.addListener(historyChange);

 // if this is the first time we have
 // loaded the page...
 if (dhtmlHistory.isFirstLoad()) {
  debug("Adding values to browser " + "history", false);
  // start adding history
  dhtmlHistory.add("helloworld", "Hello World Data");
  dhtmlHistory.add("foobar", 33);
  dhtmlHistory.add("boobah", true);

  var complexObject = new Object();
  complexObject.value1 = "This is the first value";
  complexObject.value2 = "This is the second data";
  complexObject.value3 = new Array();
  complexObject.value3[0] = "array 1";
  complexObject.value3[1] = "array 2";

  dhtmlHistory.add("complexObject", complexObject);

  // cache some values in the history
  // storage
  debug("Storing key 'fakeXML' into " + "history storage", false);
  var fakeXML =
   '<?xml version="1.0" '
   + 'encoding="ISO-8859-1"?>'
   + '<foobar>'
   + '<foo-entry/>'
   + '</foobar>';
  historyStorage.put("fakeXML", fakeXML);
}

然后,如果用戶離開頁面后又通過后退按鈕返回該頁面,我們可以使用get()方法提取保存的值,或者使用hasKey()方法檢查該值是否存在。

以下為引用的內容:

window.onload = initialize;

function initialize() {
 // initialize the DHTML History
 // framework
 dhtmlHistory.initialize();

 // subscribe to DHTML history change
 // events

[1] [2] 下一頁

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
欧美视频在线播放一区| 欧美色视频日本高清在线观看| 欧美一区二区三区婷婷月色| 免费看国产黄色片| 一区二区动漫| 婷婷久久综合网| 桃花视频大全不卡免费观看网站| www.久久色.com| 国产一区二区三区四区五区| 丰满肉嫩西川结衣av| 91精品国产高清一区二区三蜜臀| 青青草免费观看视频| 97视频在线观看视频免费视频| 欧美黑人猛猛猛| 亚洲成av人片在www色猫咪| 精品国产一区二区三区麻豆免费观看完整版| 国产男女猛烈无遮挡在线喷水| 国产精品99久久久久久白浆小说| av日韩亚洲| 91精品视频一区| 亚洲综合欧美综合| 欧美第一页在线观看| 国产最新在线| 爱情岛论坛亚洲入口| 国产欧美日韩在线| 亚洲精品男人天堂| 激情图片在线观看高清国产| 激情小说亚洲色图| 超碰在线公开97| 久久亚洲不卡| 国产高清在线观看| 国产精品久久久久久久久免费丝袜| 日韩精品免费看| 97成人精品视频在线观看| 豆国产96在线|亚洲| 亚洲精品一区二区二区| 色吊丝中文字幕| 久久无码人妻一区二区三区| 久久久亚洲欧洲日产国码aⅴ| 国产精品女人久久久| 色综合男人天堂| 国产乱妇无码大片在线观看| 国产精品青草久久| 亚洲精品乱码久久久久久自慰| 中文字幕在线播放第一页| 日本少妇xxxx软件| shkd中文字幕久久在线观看| 亚洲欧洲综合另类| 欧美三级伦理在线| 黄污网站在线观看| 日本天码aⅴ片在线电影网站| www午夜视频| 精品一区二区三区日韩| 色成人亚洲网| 欧美先锋影音| 日本老师69xxx| 国产91精品入| 亚洲国产精品yw在线观看| 欧美黑人极品猛少妇色xxxxx| fpee性欧美| 国产原创一区二区| 亚洲欧美一区二区三区孕妇| 欧美高清视频在线观看mv| 亚洲激情国产| 日产精品久久久久久久性色| 国产午夜亚洲精品羞羞网站| 日韩欧美综合在线| 成人啪啪免费看| 884aa四虎影成人精品一区| 97超碰国产精品女人人人爽| www.污视频| 日本亚洲欧美成人| 久久中文字幕二区| 国产亚洲一区二区在线观看| 波多野结衣家庭教师在线| heyzo高清中文字幕在线| 亚洲欧美强伦一区二区| 欧美高清videos高潮hd| 手机看片福利在线| 欧美区二区三区| 天堂中文字幕在线观看| 久久精品国产2020观看福利| 粉嫩aⅴ一区二区三区四区| 欧美成人精品午夜一区二区| 久久久久久久高清| 久久久久一区二区三区四区| 久久久久女人精品毛片九一| 国产精品有限公司| 国产情侣高潮对白| 国产美女玉足交| 亚洲国产高清在线观看视频| 亚洲免费影视第一页| 国产午夜精品一区二区三区嫩草| 无码h黄肉3d动漫在线观看| 国产视频一二三| 亚洲欧美一区二区三区不卡| 九色porny自拍视频在线观看| 欧美一区二区三级| 美女又爽又黄视频毛茸茸| h片免费观看| 天天射综合网站| 亚洲成人黄色影院| 高清日韩欧美| 欧美一区二区三区在线| 日韩最新免费不卡| 91在线观看免费高清完整版在线观看| 日韩一区二区三区高清在线观看| 久久久精品久久| 一本一道人人妻人人妻αv| 麻豆视频在线观看免费| 天天综合网久久| 欧美视频中文在线看| 亚洲综合av在线播放| 一级片在线播放| 九色porny91| 亚洲一区中文字幕在线| 亚洲在线色站| 日本猛少妇色xxxxx免费网站| 国产树林野战在线播放| 国产精品女同一区二区| 五月婷婷丁香激情| 在线观看中文字幕的网站| 色综合色综合色综合色综合| 久久蜜臀精品av| 99精产国品一二三产品香蕉| 影音先锋欧美在线| 日韩欧美亚洲国产一区| 亚洲娇小娇小娇小| 性感美女一区二区三区| 国产精品一区免费观看| 中文字幕色呦呦| 男人天堂视频在线观看| 中文字幕日韩av资源站| jizz.日本| 日本理论片2828理论片| 国产精品无码毛片| 亚洲精品第五页| 久久精品99无色码中文字幕| 精品人妻一区二区三区日产乱码| 色素色在线综合| 亚洲第一网站男人都懂| 日韩av成人高清| 久久99精品国产自在现线| 亚洲国产精品yw在线观看| www.浪潮av.com| 蜜桃一区二区三区| 国产成人精品18| 久久精品99北条麻妃| 黄色成人av在线| 在线电影看在线一区二区三区| 日产欧产美韩系列久久99| 性欧美videoshd高清| 粉嫩av性色av蜜臀av网站| 熟女少妇a性色生活片毛片| 水蜜桃久久夜色精品一区| 欧美一级黑人aaaaaaa做受| gogo高清午夜人体在线| 大陆极品少妇内射aaaaaa| 在线观看精品一区二区三区| 精品国产1区2区3区| 久久免费视频精品| 天天做夜夜做人人爱精品| 在线观看日韩av先锋影音电影院| 久久久久国产精品区片区无码| 永久免费精品视频| 91玉足脚交白嫩脚丫在线播放| 亚洲国产成人精品无码区99| 国产99精品在线观看| 影音先锋导航| 国产女人18毛片18精品| 羞羞网站在线看| 啊v视频在线一区二区三区| 91网在线播放| 亚洲综合激情六月婷婷在线观看| 特黄国产免费播放| 国产精品高清亚洲| 别急慢慢来1978如如2| 亚洲欧美日韩国产综合精品二区| 久久久精品在线观看| 午夜精品久久久久99蜜桃最新版| 久久久久久久久久成人| 在线观看一区二区精品视频| 欧美黑人xxxx猛牲大交| ass极品国模人体欣赏| 国产精品视屏| 午夜精品一区二区三区免费视频| 久久综合久久88| 日韩网站在线看片你懂的| 成人精品免费看| 国产福利a级| 97人妻精品一区二区三区动漫| 成人小视频在线播放| 亚洲综合一区二区三区| 久草手机在线视频| 亚洲欧美日韩久久精品| 国产亚洲欧美日韩精品| 一卡二卡三卡四卡| 国产精品无码永久免费不卡| 久久一级电影| 丁香六月天婷婷| 国产精品久久电影观看| 久久亚洲国产成人| 精品视频久久久久久久| 99精品国产一区二区青青牛奶| 韩国中文字幕2020精品| 国产小视频国产精品| 精品三区视频| 国产精品剧情一区二区在线观看| 欧美激情性爽国产精品17p| 国产香蕉尹人视频在线| 日本一区二区在线免费观看| 三级在线观看一区二区| 俺也去精品视频在线观看| 在线欧美激情| 乱插在线www| 潘金莲一级淫片aaaaa免费看| 伊人精品久久久久7777| 2019中文字幕在线免费观看| 最新不卡av| 亚洲国产精品大全| 婷婷综合久久一区二区三区| 麻豆av免费看| 精品国产第一国产综合精品| 国产精品视频最多的网站| 国内精品二区| 视频区小说区图片区| 婷婷中文字幕一区三区| av电影网站在线观看| 免费裸体视频网站| 欧美精品少妇| 欧美性久久久| 可以免费看污视频的网站| 2020日本在线视频中文字幕| 日韩欧美不卡视频| 亚洲国产精品第一页| 动漫成人在线观看| 亚洲人成网站色ww在线| 亚洲成人av中文字幕| av美女在线观看| 久久av在线播放| 日韩精品欧美大片| 国产欧美午夜| 久久婷婷蜜乳一本欲蜜臀| 亚洲美女久久久| 成人免费视频网站在线观看| 草草久视频在线观看电影资源| 中文在线最新版地址| 日韩乱码一区二区| 中文字幕不卡每日更新1区2区| 91精品国产一区二区三区香蕉| 精品日本美女福利在线观看| 国产在线精品一区| 青青在线观看视频中文字| 亚洲精品自拍动漫在线| 久操网在线观看| 国产精品18久久久久久久久久| 国产福利小视频在线观看| 亚洲国产精品www| 欧美日韩专区在线| 91精品91久久久中77777老牛| 亚洲最大最好的私人影剧院| 亚洲小说春色综合另类网蜜桃| 日韩精品欧美在线| 日韩子在线观看| 国产乱人伦真实精品视频| 日本 片 成人 在线| 亚洲男人天堂九九视频| 日本三级一区二区三区| 欧美日韩一级大片| 精品无码国产污污污免费网站| 欧美最新大片在线看| 麻豆自创视频在线观看| 亚洲综合图区| 6—12呦国产精品| 日本一二三区在线观看| 成人羞羞视频播放网站| 欧美88av| 精品人妻一区二区三区潮喷在线| 久久91av| 在线精品视频一区二区| 疯狂欧美牲乱大交777| 日韩av一区二区三区在线| 亚洲国产高清国产精品| 国产午夜久久av| 久久精品在线| 免费人成网站在线观看欧美高清| 在线观看黄色网| 国产免费黄视频在线观看| 国产日产欧美视频| 久久精品国产亚洲精品| 欧美日韩精品在线| 日韩在线观看精品| 久久精品成人| 欧洲免费av| 日本成人中文字幕在线视频| 日韩电影大全免费观看2023年上| 欧美韩日一区二区三区| 国产欧美一二三区| 77成人影视| 999色成人| 欧美成人免费在线观看视频| 久久久久久久久久99| 天天色综合久久| www视频免费看| 三上悠亚在线资源| 欧美大片国产精品| 91中文字幕永久在线| 日韩电影免费观| koreanbj精品视频一区| 色诱色偷偷久久综合| 婷婷激情久久| eeuss鲁片一区| 波多野结衣一区二区三区在线观看| 国产女人18毛片水真多18精品| 8848hh四虎| 国产一区二区三区在线观看网站| 国产+成+人+亚洲欧洲自线| 成人免费毛片在线观看| 99久免费精品视频在线观78| 国产欧美日韩一区二区三区在线观看| 99久久国产综合精品色伊| 国产伦精品一区二区三区视频网站| 亚洲国产一区二区三区a毛片| 精品国产乱码久久久久久夜甘婷婷| 国模私拍视频在线观看| 亚洲人成电影网站色…| 黄色网址网站在线观看| 日韩欧美一二三区|