开发网站后台管理时,经常遇到页面返回的情况,如果采用history.go(-1)的方式返回上一页面,则有时在页面回发之后会停留在当前页面,需要点击多次;如果用ReturnUrl参数,则感觉使用场合不是很合适,而且需要在每个连接上面都指定ReturnUrl参数,虽然可以用Request.Url.PathAndQuery来获取当前页面的完整链接,但实现起来较为繁琐;我在参考了vs2005ASP.NET网站配置的代码之后,发现它是返回上一页的方式是使用Stack的方式来处理,即后进先出的方式;这样我整理了一下,发现通过这种方式来处理返回的应用很便捷。
这里主要解决两个问题:一个是视图控制,即要显示那个视图(上一视图会自动隐藏)和返回上一视图;二是页面跳转控制,返回上一页面,无需传送参数,也不必指定url。
解决这两个问题都使用到了Stack,对于视图控制,该Stack存储于ViewState中,对于页面控制,则存储与Session中;下面是编码部分:
-
页面跳转,首先在页面基类中设计属性:Urls和方法ShowPreviouseUrl();
- 代码
Stack Urls { get { return Session["Urls"] as Stack ?? new Stack(); } set { Session["Urls"] = value; } }
- 代码
protected void ShowPreviouseUrl() { if (Urls.Count > 1) { Stack tempUrls = Urls; tempUrls.Pop(); string url = tempUrls.Peek() as string; Urls = tempUrls; Response.Redirect(url); } }
- 在页面基类中的OnLoad事件加入:
当页面载入触发OnLoad事件时,程序会获取当前页面完整的url,当Urls属性中不包含该url时,则将url插入Urls的顶部,然后存储在session中;if(!IsPostBack) { string url = Request.Url.PathAndQuery; if (Urls.Count == 0 || !url.Equals(Urls.Peek() as string)) { Stack tempUrls = Urls; tempUrls.Push(url); Urls = tempUrls; } }
- 在具体的一个页面中放置一个返回链接,并加入事件即可:
当触发该事件时,程序会从session中拿取Urls属性值,并判断Urls个数大于1个时,先移除处于Urls顶部的Url,然后在获取下一顶部的Url,该Url就是要返回页面地址;最后通过Response.Redirect实现页面跳转。protected void LinkButtonBack_Click(object sender, EventArgs e) { ShowPreviouseUrl(); }
- 代码
-
页面中视图控制,同样在页面基类中设计属性Views和方法ShowPreviouseView()、ShowView():
- 代码
Stack Views { get { return ViewState["Views"] as Stack ?? new Stack(); } set { ViewState["Views"] = value; } }
- 代码:
该函数执行时,首先判断是否存在可返回的视图,如果存在,则显示上一视图,同时将当前视图从Views中移除;如果不存在,则检测是否存在上一页面,如果存在,则返回上一页面,否则不作相应。protected void ShowPreviousePanel() { Stack tempStatck = Views; if (tempStatck.Count == 0 || tempStatck.Count == 1) { if (Urls.Count > 1) { Stack tempUrls = Urls; tempUrls.Pop(); string url = tempUrls.Peek() as string; Urls = tempUrls; Response.Redirect(url); } return; } FindControl(Page, tempStatck.Pop() as string).Visible = false; FindControl(Page, tempStatck.Peek() as string).Visible = true; Panels = tempStatck; }
- 代码:
protected void ShowPanel(string viewId) { Stack tempStatck = Views; if (tempStatck.Count > 0) { FindControl(Page, tempStatck.Peek() as string).Visible = false; } Control targetView = FindControl(Page, viewId); if (targetView == null) { throw new LogicException("没有找到要显示的视图"); } targetView .Visible = true; tempStatck.Push(viewId); Views= tempStatck; }
输入的参数为某个视图的ID,程序通过递归的方式获取该视图控件,如果找不到该控件,则抛出异常;如果找到该视图,则显示之,而之前的视图会被隐藏(如果存在),同时程序会把即将显示的视图放入Views对象的顶部;最后将Views对象存储入ViewStatus中。protected static Control FindControl(Control container, string ControlId) { if (container != null && container.Controls.Count > 0) { if (container.FindControl(ControlId) == null) { foreach (Control item in container.Controls) { Control target = FindControl(item, ControlId); if (target != null) { return target; } } } else { return container.FindControl(ControlId); } } return null; }
- 当编写某个应用时并需要多个操作时,可以考虑把该应用编写为一个aspx页面,而多个操作则编写为对应视图的操作;(这里暂不考虑rest规则)加入存在两个Panel,则将其Visible属性为false:
然后再On_Load事件中确定显示某个视图:<asp:Panel ID="PanelFileList" runat="server" Visible="false"></asp:Panel> <asp:Panel ID="PanelFielUpload" runat="server" Visible="false"></asp:Panel>
ShowPanel(PanelFileList.ID)
- 返回上一视图,在返回控件加入onclick事件,并添加调用方法:
该方法调用之后,程序会尝试显示上一视图,如果不存在上一视图,则尝试跳转页面至上一页面,如果不存在上一页面,则不作其他响应。ShowPreviousePanel();
- 代码