`
jelly
  • 浏览: 299410 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

Dojo和Ajax 程序的浏览器History 控制

    博客分类:
  • AJAX
阅读更多

web应用程序都面临浏览器上的history功能的困扰,前进,后退,刷新,将是你的程序失控,重复提交,丢失数据等等场景让你焦头烂额,运行的好好的程序,被客户随便摆弄一下就错误百出。这些就是客户心智和程序员心智不同产生的冲突。

最初的web程序只是一些静态的网页,一般都通过地址栏的URL地址来进行定位,所以,浏览器集成了很方便的History功能,让你在浏览的历史记录中来回切换。可是,随着复杂应用程序开始越来越多的通过浏览器来作为客户端,噩梦就开始了。

近年来,Ajax技术的兴起为程序设计者和客户提供了不同的技术实现和用户体验,Ajax火了起来,也开始颠覆了传统的web操作和应用传统。但是浏览器并没有相应的来迎合这些变化,这样的话, Ajax 程序也同样的面临浏览器的干扰,而History功能就是其中让人头疼的一个。

越来越多的程序员面临这个问题,开始考虑解决之道,解决的方案五花八门,不过都这能解决问题的一部分,无法从根本上解决这个问题。要根本解决这个问题,只怕需要浏览器的厂商和标准部门针对Ajax一起来想出一个标准化的处理方式。

当然我们没有时间来等待别人来帮助我们解决问题,我这里介绍一下Dojo 解决这个问题的方法,同时研究一下它的实现原理。

需要解决的问题

Ajax程序进行页面内容变化的时候,并没有刷新整个页面也没有改变浏览器地址,所以如果进行以下的操作将会出现无法意料的问题:

行为案例:1search 页面--2search-->3显示列表在本页面--4点击列表中的某条记录--5页面切换-->6所选记录的详细内容页面-->7编辑并保存-->8停留在原页面,提示保存成功-->9返回--.10页面切换-->11search页面

上面的步骤都不会导致浏览器地址变化,因为我们的应用是单页面的frame。

这个案例涉及到两个页面(黑体 1,6),4 个动作(红色 2,4,7,9)

后退/ 前进: 如果在step6的时候,如果浏览器的后退按钮可以点击的话(一般如果你之前没有访问过别的url,那么将不会有后退地址),将不会回到3或者1,而是1之前你访问过的地址。

刷新:如果在step6的时候,你点击刷新,将不会刷新step6所在的页面,因为浏览器的地址没有发生变化,你会发现,页面刷新之后来到了step1.

收藏: 同上,你收藏的step1所在的页面。

客户看到他花钱买的程序却是这样的行为,客户岂不是要疯掉!!!

解决之道

1. dojo

 dojo1.0 提供dojo.back 对象来处理这里问题,可以记录页面状态到一个堆栈对象(historyStack)中。但是这个historyStack开始是空的,当我们进入的首页的时候,可以通过setInitialState 方法设置他的第一个状态,如

js 代码
  1. dojo.back.setInitialState(currentState);  

之后每一次页面状态变化,你都可以将这个状态用书签记录下来,如

js 代码
  1. dojo.back.addToHistor(currentState);  

state 代表你需要记录的页面状态,一般包括下面几个属性:

back 函数  //当用户点击后退按钮,并且跳转到这个状态的时候将被调用。
forward 函数  //  当用户点击前进按钮,并且跳转到这个状态的时候将被调用。
changeUrl 布尔类型或者字符串 布尔类型指示是否为当前状态创建一个唯一的hash值 如果是一个字符串,那么将使用这个字符串而不是hash值。

  1. var state = {   
  2.     back: function() { alert("Back was clicked!"); 
  3.         dojo.back.addToHistory(repeatState); },   
  4.     forward: function() { alert("Forward was clicked!"); }   
  5.  };  

dojo 的back对象将会在用户点击浏览器后退按钮的时候自动将当前state从historyStack中Pop出来。同时进入到前一个state。

下面摘选自dojo库的back.js

js 代码
  1. function handleBackButton(){   
  2.         //summary: private method. Do not call this directly.   
  3.   
  4.         //The "current" page is always at the top of the history stack.   
  5.         //console.debug("handlingBackButton");   
  6.         var current = historyStack.pop();   
  7.         if(!current){ return; }   
  8.         var last = historyStack[historyStack.length-1];   
  9.         if(!last && historyStack.length == 0){   
  10.             last = initialState;   
  11.         }   
  12.         if(last){   
  13.             if(last.kwArgs["back"]){   
  14.                 last.kwArgs["back"]();   
  15.             }else if(last.kwArgs["backButton"]){   
  16.                 last.kwArgs["backButton"]();   
  17.             }else if(last.kwArgs["handle"]){   
  18.                 last.kwArgs.handle("back");   
  19.             }   
  20.         }   
  21.         forwardStack.push(current);   
  22.         //console.debug("done handling back");   
  23.     }  

完整的例子,请看dojo1.0的发行包里面的back.html(dojo/tests/back.html),里面有完成的例子。

 

dojo怎么控制到后退按钮的呢?怎么记录书签,其中的奥妙何在呢?那就要看iframe在这里起到什么作用了。

 

2.  利用iframe 了控制每个页面的书签(bookmark)

 在页面地址没有变化的情况下,怎么记录书签呢?怎么样让用户能够收藏呢?如果要能够收藏,地址一定要变化,但是地址变化的时候又要确保页面的内容不发生变化。How to do?

Yes, Use # , 增加书签到地址后面,如: http://yourdomain/yourpage#bookmark 这样的话,地址栏变化了,那么后退按钮也就有了历史记录了。别高兴的太早,Ie并不是这样。Ie并不会将http://yourdomain/yourpage#bookmarkhttp://yourdomain/yourpage认为是两个页面,所以也就不会加入到History中,怎么处理这个问题呢?iframe在这里派上作用拉。通过设置iframe的src属性可以让Ie将这些状态记录到history中。

 

参考:http://www.contentwithstyle.co.uk/Articles/38/fixing-the-back-button-and-enabling-bookmarking-for-ajax-apps

 http://codinginparadise.org/weblog/2005/09/ajax-history-libraries.html

js 代码
分享到:
评论
14 楼 冯冀川 2008-10-21  
http://code.google.com/p/reallysimplehistory/

RSH似乎对AJAX的历史记录解决的不错,并且是一个很轻量级的独立的框架
13 楼 internetafei 2008-10-20  
dojo.back 和 base href   一起用会有问题,怎么解决
12 楼 tsenfine 2008-08-10  
FF3对hash的处理似乎不像对普通url的处理那样,并不重新载入网页
11 楼 jelly 2008-02-04  
night2006 写道
是挺不错,哪里可以提供dojo api和教程

www.dojotoolkit.com
10 楼 jessdy 2008-01-30  
zhaoxin 写道
allenBen 写道
最好就是尽量少用Ajax 因为还不成熟,框架的Bug也太多

[url=http://www.gwteasy.cn][/url]小蜗牛娱乐平台
javascript 写的相信吗?纯的ajax有bug吗?

好好的EXT被弄得又卡又难看
9 楼 zhaoxin 2008-01-24  
allenBen 写道
最好就是尽量少用Ajax 因为还不成熟,框架的Bug也太多

[url=http://www.gwteasy.cn][/url]小蜗牛娱乐平台
javascript 写的相信吗?纯的ajax有bug吗?
8 楼 night2006 2008-01-16  
是挺不错,哪里可以提供dojo api和教程
7 楼 Tin 2008-01-13  
Dojo这个框架的一大亮点就是它的IO设计,可以享受集中控制的好处。
它的io进行了统一封装,使用bind接口通讯,并进行了排队。这样的封装也就支持了它通过iframe来模拟前进后退,非常方便。
这一点它比Prototype和JQuery这样的轻封装占有了一些优势,也方便你扩展一些Js上的AOP机制上去。


但是它还有不适用场合:使用iframe通讯的场合(如跨子域),或者scriptio的时候,不能用history。
6 楼 dbwang 2008-01-08  
<div class='quote_title'>jelly 写道</div><div class='quote_div'><p>web应用程序都面临浏览器上的history功能的困扰,前进,后退,刷新,将是你的程序失控,重复提交,丢失数据等等场景让你焦头烂额,运行的好好的程序,被客户随便摆弄一下就错误百出。这些就是客户心智和程序员心智不同产生的冲突。</p>
<p>最初的web程序只是一些静态的网页,一般都通过地址栏的URL地址来进行定位,所以,浏览器集成了很方便的History功能,让你在浏览的历史记录中来回切换。可是,随着复杂应用程序开始越来越多的通过浏览器来作为客户端,噩梦就开始了。</p>
<p>近年来,Ajax技术的兴起为程序设计者和客户提供了不同的技术实现和用户体验,Ajax火了起来,也开始颠覆了传统的web操作和应用传统。但是浏览器并没有相应的来迎合这些变化,这样的话, Ajax 程序也同样的面临浏览器的干扰,而History功能就是其中让人头疼的一个。</p>
<p>越来越多的程序员面临这个问题,开始考虑解决之道,解决的方案五花八门,不过都这能解决问题的一部分,无法从根本上解决这个问题。要根本解决这个问题,只怕需要浏览器的厂商和标准部门针对Ajax一起来想出一个标准化的处理方式。</p>
<p>当然我们没有时间来等待别人来帮助我们解决问题,我这里介绍一下Dojo 解决这个问题的方法,同时研究一下它的实现原理。</p>
<h4>需要解决的问题</h4>
<p>Ajax程序进行页面内容变化的时候,并没有刷新整个页面也没有改变浏览器地址,所以如果进行以下的操作将会出现无法意料的问题:</p>
<p>行为案例:1<strong>search 页面</strong>--2<span style='color: #ff0000;'>search</span>--&gt;3显示列表在本页面--4<span style='color: #ff0000;'>点击列表中的某条记录</span>--5页面切换--&gt;6<strong>所选记录的详细内容页面</strong>--&gt;7<span style='color: #ff0000;'>编辑并保存</span>--&gt;8停留在原页面,提示保存成功--&gt;9<span style='color: #ff0000;'>返回</span>--.10页面切换--&gt;11<strong>search页面</strong></p>
<p><strong>上面的步骤都不会导致浏览器地址变化,因为我们的应用是单页面的frame。</strong></p>
<p>这个案例涉及到两个页面(<strong>黑体 1,6</strong>),4 个动作(<span style='color: #ff0000;'>红色 2,4,7,9</span>)</p>
<p>后退/ 前进: 如果在step6的时候,如果浏览器的后退按钮可以点击的话(一般如果你之前没有访问过别的url,那么将不会有后退地址),将不会回到3或者1,而是1之前你访问过的地址。</p>
<p>刷新:如果在step6的时候,你点击刷新,将不会刷新step6所在的页面,因为浏览器的地址没有发生变化,你会发现,页面刷新之后来到了step1.</p>
<p>收藏: 同上,你收藏的step1所在的页面。</p>
<p>客户看到他花钱买的程序却是这样的行为,客户岂不是要疯掉!!!</p>
<h4>解决之道</h4>
<p>1. dojo</p>
<span><span><span><span><span><span><span><span><span><span><span><span><span><span><span>
<p> dojo1.0 提供dojo.back 对象来处理这里问题,可以记录页面状态到一个堆栈对象(<span style='font-family: Arial;'>historyStack</span>)中。但是这个<span style='font-family: Arial;'>historyStack开始是空的,当我们进入的首页的时候,可以通过<span style='font-family: Arial;'>setInitialState 方法设置他的第一个状态,如 </span></span><span style='font-family: Arial;'/></p>
<div class='code_title'>js 代码</div>
<div class='dp-highlighter'>
<ol class='dp-c'>
    <li class='alt'><span><span>dojo.back.setInitialState(currentState);  </span></span> </li>
</ol>
</div>
<p>之后每一次页面状态变化,你都可以将这个状态用书签记录下来,如</p>
<div class='code_title'>js 代码</div>
<div class='dp-highlighter'>
<ol class='dp-c'>
    <li class='alt'><span><span>dojo.back.addToHistor(currentState);  </span></span> </li>
</ol>
</div>
<p>state 代表你需要记录的页面状态,一般包括下面几个属性:</p>
<p><span style='font-family: Arial;'>back 函数  //当用户点击后退按钮,并且跳转到这个状态的时候将被调用。<br/>
forward 函数  //  当用户点击前进按钮,并且跳转到这个状态的时候将被调用。<br/>
changeUrl 布尔类型或者字符串 布尔类型指示是否为当前状态创建一个唯一的hash值 如果是一个字符串,那么将使用这个字符串而不是hash值。</span> </p>
<div class='dp-highlighter'>

<ol class='dp-c'>
    <li class='alt'><span><span class='keyword'>var</span><span> state = {   </span></span> </li>
    <li><span>    back: </span><span class='keyword'>function</span><span>() { alert(</span><span class='string'>"Back was clicked!"</span><span>); </span> </li>
    <li><span>        dojo.back.addToHistory(repeatState); },   </span> </li>
    <li class='alt'><span>    forward: </span><span class='keyword'>function</span><span>() { alert(</span><span class='string'>"Forward was clicked!"</span><span>); }   </span> </li>
    <li><span> };  </span> </li>
</ol>
</div>
<p>dojo 的back对象将会在用户点击浏览器后退按钮的时候自动将当前state从<span style='font-family: Arial;'>historyStack中Pop出来。同时进入到前一个state。</span></p>
<p>下面摘选自dojo库的back.js</p>
<div class='code_title'>js 代码</div>
<div class='dp-highlighter'>

<ol class='dp-c'>
    <li class='alt'><span><span class='keyword'>function</span><span> handleBackButton(){   </span></span> </li>
    <li><span>        </span><span class='comment'>//summary: private method. Do not call this directly. </span><span>  </span> </li>
    <li class='alt'><span>  </span> </li>
    <li><span>        </span><span class='comment'>//The "current" page is always at the top of the history stack. </span><span>  </span> </li>
    <li class='alt'><span>        </span><span class='comment'>//console.debug("handlingBackButton"); </span><span>  </span> </li>
    <li><span>        </span><span class='keyword'>var</span><span> current = historyStack.pop();   </span> </li>
    <li class='alt'><span>        </span><span class='keyword'>if</span><span>(!current){ </span><span class='keyword'>return</span><span>; }   </span> </li>
    <li><span>        </span><span class='keyword'>var</span><span> last = historyStack[historyStack.length-1];   </span> </li>
    <li class='alt'><span>        </span><span class='keyword'>if</span><span>(!last &amp;&amp; historyStack.length == 0){   </span> </li>
    <li><span>            last = initialState;   </span> </li>
    <li class='alt'><span>        }   </span> </li>
    <li><span>        </span><span class='keyword'>if</span><span>(last){   </span> </li>
    <li class='alt'><span>            </span><span class='keyword'>if</span><span>(last.kwArgs[</span><span class='string'>"back"</span><span>]){   </span> </li>
    <li><span>                last.kwArgs[</span><span class='string'>"back"</span><span>]();   </span> </li>
    <li class='alt'><span>            }</span><span class='keyword'>else</span><span> </span><span class='keyword'>if</span><span>(last.kwArgs[</span><span class='string'>"backButton"</span><span>]){   </span> </li>
    <li><span>                last.kwArgs[</span><span class='string'>"backButton"</span><span>]();   </span> </li>
    <li class='alt'><span>            }</span><span class='keyword'>else</span><span> </span><span class='keyword'>if</span><span>(last.kwArgs[</span><span class='string'>"handle"</span><span>]){   </span> </li>
    <li><span>                last.kwArgs.handle(</span><span class='string'>"back"</span><span>);   </span> </li>
    <li class='alt'><span>            }   </span> </li>
    <li><span>        }   </span> </li>
    <li class='alt'><span>        forwardStack.push(current);   </span> </li>
    <li><span>        </span><span class='comment'>//console.debug("done handling back"); </span><span>  </span> </li>
    <li class='alt'><span>    }  </span> </li>
</ol>
</div>
<p>完整的例子,请看dojo1.0的发行包里面的back.html(<span style='font-family: Arial;'>dojo/tests/back.html</span>),里面有完成的例子。</p>
<p> </p>
<p>dojo怎么控制到后退按钮的呢?怎么记录书签,其中的奥妙何在呢?那就要看iframe在这里起到什么作用了。</p>
<p> </p>

<p>2.  利用iframe 了控制每个页面的书签(<span style='font-family: Arial;'>bookmark</span>)</p>
<p> 在页面地址没有变化的情况下,怎么记录书签呢?怎么样让用户能够收藏呢?如果要能够收藏,地址一定要变化,但是地址变化的时候又要确保页面的内容不发生变化。How to do? </p>
<p>Yes, Use # , 增加书签到地址后面,如: <a href='http://yourdomain/yourpage#bookmark'>http://yourdomain/yourpage#bookmark</a> 这样的话,地址栏变化了,那么后退按钮也就有了历史记录了。别高兴的太早,Ie并不是这样。Ie并不会将<a href='http://yourdomain/yourpage#bookmark'>http://yourdomain/yourpage#bookmark</a>和<a href='http://yourdomain/yourpage'>http://yourdomain/yourpage</a>认为是两个页面,所以也就不会加入到History中,怎么处理这个问题呢?iframe在这里派上作用拉。通过设置iframe的src属性可以让Ie将这些状态记录到history中。</p>
<p> </p>
<p>参考:<span style='font-family: Arial;'><a href='http://www.contentwithstyle.co.uk/Articles/38/fixing-the-back-button-and-enabling-bookmarking-for-ajax-apps'>http://www.contentwithstyle.co.uk/Articles/38/fixing-the-back-button-and-enabling-bookmarking-for-ajax-apps</a></span></p>
<p> <span style='font-family: Arial;'>http://codinginparadise.org/weblog/2005/09/ajax-history-libraries.html</span></p>
<div class='code_title'>js 代码</div>
</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></div>
5 楼 allenBen 2007-12-27  
最好就是尽量少用Ajax 因为还不成熟,框架的Bug也太多
4 楼 jelly 2007-12-27  
前进后退我们是无法控制的。只能针对性的使用一些诡计解决我们面对的问题。
3 楼 han0332217 2007-12-26  
如果不用dojo,也不用Ajax,怎么控制浏览器的前进和后退
2 楼 FGhost 2007-12-12  
不错。正被这个问题困扰呢。
1 楼 prophet 2007-12-11  
写得不错,正好也在看DOJO

相关推荐

Global site tag (gtag.js) - Google Analytics