最近在做一个SL的项目,做完后,遇到一个F5刷新的问题,本人也是第一次接触接触SL项目,记得再ASP.NET浏览器的缓存会自动保存最后一次的浏览记录的。

所以就在网上到处找资料,可惜运气不好,都没找到合适的资料。基本的解决方法都是通过再HTML页面增加JS方法,屏蔽F5刷新按钮的功能,但是这 样的需求并不是我们项目中所要的,还好在BAIDU和群里高手的帮助下,终于大体了解了SL刷新的过程和解决F5刷新返回最后次浏览页面的思想。。

1:SL刷新过程

SL本身就是HTML页面的一个插件程序,在浏览器退出或者F5刷新的时候,SL本身首先调用自身的APP.xaml 的 Application_Exit 方法,然后再次进入浏览器的时候加载Application_Startup方法,一般都是用Application_Startup 方法去设定SL程序的启动初始界面。

知道了SL运行的过程我想实现F5刷新问题就不难了,

2:F5刷新解决思路

首先在本地建立独立存储(COOKIE也可以,不过不知道为啥用COOKIE的话,SL项目用火狐浏览访问数据就有问题,以待以后研究。)

为了简单阐述过程,本文的COOKIE就保存3个值:

UserID:用户账号信息

PageAddress:最后次浏览页面地址

ExitTime:退出时间

1>系统初始化加载的时候,判断本地存储是否有数据,并且判断当前时间与上次退出时间差 是否小于5秒,(我这里利用5秒作为刷新和重新登录的标准,当然因人而异)

2>如果上面条件成立,咋直接根据账号信息,加载系统主界面,并将Navigation 的IFRAM 地址指向本地存储的上次保存地址即可。

3>如果不成立,显示登录界面,并清除本地存储数据,登录后重新赋值账号信息

4>伴随着SL界面的浏览,操作界面的时候同时更新本地存储的页面地址值

5>Application_Exit 事件里,增加 slcookie.ExitTime = DateTime.Now.ToString(); 时间的赋值

OK! 整体思路大致如此,现在我将以一个简短的实例来详细说明

一、新建项目SLF5

打开VS2010,新建Silverlight应用程序,

解决Silverlight F5刷新问题-LMLPHP

并勾选在新网站中承载应用程序

解决Silverlight F5刷新问题-LMLPHP

这样一个SL项目就建立完毕,

接下来需要的就是建立本地存储类,为了便于理解我就取名SLCookie.cs

解决Silverlight F5刷新问题-LMLPHP
 using System;
using System.Net;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Documents;
using System.Windows.Ink;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Animation;
using System.Windows.Shapes;
using System.IO.IsolatedStorage;
using System.Text; namespace SLF5
{
/// <summary>
/// 功能:SilverLight独立存储类
/// 创建人:龚安川
/// 创建时间:2012-08-18
/// </summary>
public class SLCookie
{
/*当前系统Cookie 保存3个字段
* PageAddress :记录最后一次访问的页面地址
* UserID: 用户名
* ExitTime:退出时间
*/ #region 独立存储相关操作函数 #region 设置Cookie
/// <summary>
/// 设置Cookie
/// </summary>
/// <param name="key">the cookie key</param>
/// <param name="value">the cookie value</param>
public static void SetCookie(string key, string value)
{
IsolatedStorageSettings settings = IsolatedStorageSettings.ApplicationSettings;
//判断该键是否存在
if (settings.Contains(key))
{
settings.Remove(key);
settings.Add(key, value);
}
else
{
settings.Add(key, value);
}
settings.Save(); }
#endregion #region 读取一个已经存在的Cookie
/// <summary>
/// 读取一个已经存在的Cookie
/// </summary>
/// <param name="key">cookie key</param>
/// <returns>null if the cookie does not exist, otherwise the cookie value</returns>
public static string GetCookie(string key)
{
IsolatedStorageSettings settings = IsolatedStorageSettings.ApplicationSettings;
//判断该键是否存在
if (settings.Contains(key))
{
return settings[key] != null ? settings[key].ToString() : "";
}
else
{
return string.Empty;
}
}
#endregion #region 删除特定的Cookie(清空它的Value值)
/// <summary>
/// 删除特定的Cookie(清空它的Value值)
/// </summary>
/// <param name="key">the cookie key to delete</param> public static void DeleteCookie(string key)
{
IsolatedStorageSettings settings = IsolatedStorageSettings.ApplicationSettings;
//判断该键是否存在
if (settings.Contains(key))
{
settings.Remove(key);
}
}
#endregion #region 判定指定的key-value对是否在cookie中存在
public static bool Exists(String key, String value)
{
IsolatedStorageSettings settings = IsolatedStorageSettings.ApplicationSettings;
//判断该键是否存在
if (settings.Contains(key))
{
return true;
}
else
{
return false;
}
}
#endregion #region 获取当前cookie内容
public static string getCookieContent()
{
StringBuilder values = new StringBuilder();
IsolatedStorageSettings settings = IsolatedStorageSettings.ApplicationSettings;
if (settings != null && settings.Count > )
{
foreach (string item in settings.Keys)
{
values.Append(settings[item] != null ? settings[item].ToString() : "");
}
return values.ToString();
}
else
{
return string.Empty;
}
}
#endregion #region 清空本地cookie
public static void ClearCookies()
{
IsolatedStorageSettings settings = IsolatedStorageSettings.ApplicationSettings;
settings.Clear();
}
#endregion 清空本地cookie #endregion
}
}
解决Silverlight F5刷新问题-LMLPHP

再为这个基类,写个辅助类,方便操作调用 SLCookieHelp.cs

接下来再APP.XAML.CS的启动事件增加判断,确定进入哪个界面,在退出事件里面修改本地存储登陆时间的更新。

一般情况下,在启动事件里面根据本地存储的数据判断启动不同的用户控件即可,但是本人这个项目正好用到了,基框架界面加载的方式,就顺便在这里也跟大家一起分享了。

现在的代码如下:

基框架界面加载用到了以下几个类:

用户控件:Host.xaml

XAML代码新建后不动即可,cs代码如下:

接口类:IHost.cs

存储全局变量类:Context.cs 以及 AppContext.cs

接下来我们建立几个测试的页面:A.XAML,B.XAML.Default.XAML 以便多页面的切换,每个界面写一些自己标记的信息即可

登陆界面:Login.XAML

就是界面放2个LABEL和2个文本框,默认账号和密码都是:admin ,具体的项目当然要去查询数据库了。

重点来说MainPage界面的构造:

添加程序集:System.Windows.Controls.Navigation 的引用,在项目中添加引用,.NET下面就可以找到该程序集

主界面我利用GRID的2列,

左侧列放一个LISTBOX 来显示项目切换的页面地址信息(类似与菜单吧),

右侧列放置一个navigation 控件,利用此控件的IFRAM指向不同的界面信息。

XAML的源码如下:

好了这样项目所需要的准备工作就已经做完,项目如下:

解决Silverlight F5刷新问题-LMLPHP

接下来开始我们的工作了,

在MainPage.Xaml.cs

定义本地存储辅助类

//本地cookie 类
SLCookieHelp slcookie = new SLCookieHelp();

增加界面加载事件  Loaded += new RoutedEventHandler(MainPage_Loaded);

  string url = @"/Default.xaml";
if (!string.IsNullOrWhiteSpace(slcookie.PageAddress))
{
url = slcookie.PageAddress;
}
Contnetframe.Navigate(new Uri(url, UriKind.Relative));

通过上面来加载显示的页面地址

然后在LISTBOX的选择事件,对本地存储的地址进行更新

这样主界面的工作就完成了,

最后再登陆界面的加载事件增加本地存储的判断以及登陆事件的本地存储账户信息的赋值即可

方法如下:

解决Silverlight F5刷新问题-LMLPHP
   //加载事件
void Login_Loaded(object sender, RoutedEventArgs e)
{
//如果本地COOKIE有值 直接进入界面
if (slcookie != null && !string.IsNullOrWhiteSpace(slcookie.UserID) && !string.IsNullOrWhiteSpace(slcookie.ExitTime))
{
//判断退出时间和登陆时差是否大于5秒
DateTime now = DateTime.Now;
DateTime exit = Convert.ToDateTime(slcookie.ExitTime);
TimeSpan ts = now - exit;
//当操作时间小于5秒默认为刷新
if (ts.TotalSeconds < 5 && ts.TotalSeconds > 0)
{
MainPage mainpage = new MainPage();
Context.Host.SetRootVisual(mainpage);
}
else
{
//清空本地COOKIE
slcookie.ClearCookie();
}
}
}
解决Silverlight F5刷新问题-LMLPHP
解决Silverlight F5刷新问题-LMLPHP
  //登陆
private void btnLogin_Click(object sender, RoutedEventArgs e)
{
if (string.IsNullOrWhiteSpace(txtuid.Text) || string.IsNullOrWhiteSpace(txtpwd.Text))
{
MessageBox.Show("用户名和密码不可为空");
return;
} if (txtuid.Text == "admin" && txtpwd.Text == "admin")
{
slcookie.UserID = "admin";
MainPage mainpage = new MainPage();
Context.Host.SetRootVisual(mainpage);
}
else
{
MessageBox.Show("用户名和密码输入不正确");
}
}
解决Silverlight F5刷新问题-LMLPHP

OK!完工,这样一个基本的SL程序的刷新功能就完成了,因为也是第一次做SL项目,当然肯定有考虑不周或者有更好的办法解决此问题,还我希望论坛的大侠们给予指导,如果写的不对的地方还望大家给予拍砖,虚心接受大家的评论。

05-11 19:33