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

发布于 2022-09-11 09:42:42 字数 5671 浏览 11 评论 0

转自 http://www.javaeye.com/topic/146738

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#bookmark和http://yourdomain/yourpage认为是两个页面,所以也就不会加入到History中,怎么处理这个问题呢?iframe在这里派上作用拉。通过设置iframe的src属性可以让Ie将这些状态记录到history中。

参考:http://www.contentwithstyle.co.u ... rking-for-ajax-apps

http://codinginparadise.org/webl ... tory-libraries.html

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文