说明

在下载大文件时,页面会进入假死状态,于是加上一个进度条以标识后台程序正在运行。

目前,做的进度条并不是实时的,并不会根据程序执行的进度正确显示。

目前是将进度条定时加载到90%,然后停止,等待后台执行完成时,修改标志位以隐藏进度条。

实现代码

js设置cookie

/****************************获取和设置cookie 开始***********************************/
//获取cookie
function getCookie(name) {
var arr, reg = new RegExp("(^| )" + name + "=([^;]*)(;|$)"); //正则匹配
if (arr = document.cookie.match(reg)) {
return unescape(arr[2]);
}
else {
return null;
}
}
//设置自定义过期时间cookie
function setCookie(name, value, time) {
var msec = getMsec(time); //获取毫秒
var exp = new Date();
exp.setTime(exp.getTime() + msec * 1);
document.cookie = name + "=" + escape(value) + ";expires=" + exp.toGMTString();
} //将字符串时间转换为毫秒,1秒=1000毫秒
function getMsec(DateStr) {
var timeNum = str.substring(0, str.length - 1) * 1; //时间数量
var timeStr = str.substring(str.length - 1, str.length); //时间单位前缀,如h表示小时 if (timeStr == "s") //20s表示20秒
{
return timeNum * 1000;
}
else if (timeStr == "h") //12h表示12小时
{
return timeNum * 60 * 60 * 1000;
}
else if (timeStr == "d") {
return timeNum * 24 * 60 * 60 * 1000; //30d表示30天
}
} //删除cookie
function delCookie(name) {
var exp = new Date();
exp.setTime(exp.getTime() - 1);
var cval = getCookie(name);
if (cval != null) {
document.cookie = name + "=" + cval + ";expires=" + exp.toGMTString();
}
}
//检查cookie
function checkCookie() {
username = getCookie('username')
if (username != null && username != "")
{ alert('Welcome again ' + username + '!') }
else
{
username = prompt('Please enter your name:', "")
if (username != null && username != "") {
setCookie('username', username, 365)
}
}
}
/*扩展使用*/ //查看导出结束标志位,是否为指定的值
function checkExportEndSign(ExportEndSignCookieName) {
var ExportCookieValue = getCookie(ExportEndSignCookieName);//获取cookie的值
if (ExportCookieValue != "" && ExportCookieValue != null && ExportCookieValue != undefined && ExportCookieValue == "complete") {
return true;
}
else {
return false;
}
}
/****************************获取和设置cookie 结束***********************************/

C#后台添加cookie

string shid_ExportCookieName = hid_ExportEndSignCookieName.Value;
HttpCookie cookie = HttpContext.Current.Request.Cookies[shid_ExportCookieName];
if (cookie == null)
{
cookie = new HttpCookie(shid_ExportCookieName);
}
cookie.Value = "complete";
HttpContext.Current.Response.AppendCookie(cookie);

需要的js和css

<link href="/script/ext3.4/resources/css/ext-all.css" rel="stylesheet" type="text/css" />
<script src="/script/ext3.4/adapter/ext/ext-base.js" type="text/javascript"></script>
<script src="/script/ext3.4/ext-all.js" type="text/javascript"></script>

ext的进度条,初始化、运行、隐藏

<script type="text/javascript">
/*进度条 相关js*/
//声明需要使用的对象
var progressBarExt;//ext进度条对象
var progressBarExt_isEnd;//ext进度条是否结束(操作执行完成,将标志位置1,进度条循环递增不会再执行)
//进度条对象-初始化
function ProcessBarExt_Init(bar_title) {
progressBarExt = Ext.Msg.show({
title: bar_title,
msg: "数据过大,需要等待一段时间!",
progress: true,
width: 300
});
progressBarExt_isEnd = 0;//初始化为0,未结束
progressBarExt.updateProgress(0, "0%");//初始化为0%,刚开始
Ext.onReady(ProcessBarExt_Run);
}
//进度条对象-开始执行
function ProcessBarExt_Run() {
var count = 0;
var bartext = "";
var curnum = 0;
Ext.TaskMgr.start({
run: function () {
count++;
if (count < 10 && progressBarExt_isEnd == 0) {
curnum = count / 10;
bartext = curnum * 100 + "%";
progressBarExt.updateProgress(curnum, bartext);
}
},
interval: 500
})
}
//进度条对象-进度到100%且隐藏
function ProcessBarExt_Hide() {
progressBarExt_isEnd = 1;
progressBarExt.updateProgress(10, "100%");
setTimeout(function () { progressBarExt.hide(); }, 500);
}
</script>

实现过程

进度条 js使用的是:EXT

核心思路:使用  cookie  标识后台操作执行完成。

js+ajsx实现

这种方式比较简单,直接看代码。

function fun_Login() {
ProcessBarExt_Init("登录");//初始化进度条对象
ProcessBarExt_Run();//进度条对象 开始加载
$.ajax({
type: "POST",
dataTye: "json",
url: "/pages/Login.ashx",
data: { type: 1},
success: function (result) {
if (result == "1") { alert("执行成功");
}
else{
alert("执行失败");
}
ProcessBarExt_Hide();
},
error: function () {
alert("执行出错");
ProcessBarExt_Hide();
}
});
}

aspx实现

这个比较麻烦,经大佬指点 外加调试才实现。

前言,之前做过类似的功能,所以先将 下载文件方法中的curContext.Response.End(); 代码删除掉了。否则,HTTP请求会结束,不会再执行后面的代码。

几种思路

(1)使用cookie

此方法为最终方案,在实际项目中使用。

设置一个cookie以标识后台程序是否执行完成。

一个隐藏域,保存每次点击下载时随机生成的cookie名称,变量ExportEndSignCookieName保存,

后台执行完操作时,根据添加一个cookie,name即为前台随机生成的保存到隐藏域中的值,value为"complete",

前台,使用js的定时器循环获取cookie的值,

如果cookie的值为后台设定的"complete",则隐藏进度条、删除js定时器、删除当前cookie。

js代码

<!--导出文件时 随机生成的cookie名称-->
<asp:HiddenField ID="hid_ExportEndSignCookieName" runat="server" ClientIDMode="Static" /> <script type="text/javascript">
var tExportCookie;//定时器-查看后台是否执行完成
var ExportEndSignCookieName="";//随机生成的cookie名称-查看后台是否执行完成
//导出Excel,添加进度条
function lbtnExport_click(){
if(confirm('是否导出所查询的数据?')){
ExportEndSignCookieName=Math.random();//生成随机的cookie名称
$("#hid_ExportEndSignCookieName").val(ExportEndSignCookieName);//cookie名称赋值给隐藏域
ProcessBarExt_Init("导出Excel");//初始化进度条对象
ProcessBarExt_Run();//进度条对象 开始加载
tExportCookie = window.setInterval(fun_ExportEndSign,20);//定时循环,查看导出操作是否完成
return true;
}
else{
return false;
}
}
//查看导出是否结束,complete
function fun_ExportEndSign(){
if(checkExportEndSign(ExportEndSignCookieName)){
ProcessBarExt_Hide();//进度条隐藏
window.clearInterval(tExportCookie); //删除定时器
delCookie(ExportEndSignCookieName);//删除cookie
}
}
</script>

后台代码

/// <summary>
/// 导出方法
/// </summary>
private void ExportExcelByProc()
{
string tempWhere = "本次查询";
SqlParameter[] sqlParameters =
{
new SqlParameter("@StartIndex",1),
new SqlParameter("@EndIndex",int.MaxValue)
};
DataSet ds = ExecuteStoredProcedureByDSNameEx("PersonInfo_List", "Business",sqlParameters);
if (ds != null && ds.Tables != null && ds.Tables.Count > 0 && ds.Tables[0].Rows.Count > 0)
{
string sheetname = "个人建档管理表";
string[] colsCols = { "sNo"};
string[] colsNames = { "档案编号"};
BF.Web.App_Code.ExcelHelper.ExportHasWhereToWeb(sheetname + ExportExcelRule() + ".xls", ds.Tables[0], sheetname, sheetname, tempWhere, colsCols, colsNames);
string shid_ExportCookieName = hid_ExportEndSignCookieName.Value;
HttpCookie cookie = HttpContext.Current.Request.Cookies[shid_ExportCookieName];
if (cookie == null)
{
cookie = new HttpCookie(shid_ExportCookieName);
}
cookie.Value = "complete";
HttpContext.Current.Response.AppendCookie(cookie);
}
else
{
Messages("当前无查询结果,请重新查询!");
}
}

(2)后台调用js方法

此方法,未实现需求。

在后台注册js,去调用前台的js

在下载文件方法的最后面curContext.Response.BinaryWrite(); 加上了上面的代码,

但是,当HTTP请求执行了下载文件的操作后,就结束了,并没有执行js的方法。

string ExtMsg = @"Ext.MessageBox.show({title: '系统提示',msg: '" + msg + "',buttons: {ok:'确  定'},icon: 'ext-mb-info'});";
ScriptManager.RegisterClientScriptBlock(this, this.GetType(), "click", ExtMsg, true);

(3)使用Hidden隐藏域

此方法,未实现需求。

使用 Hidden做标志位,前台js进行监视,当隐藏域 触发 change事件时,去执行进度条的隐藏操作。

但调试时,当后台代码执行完成后(也执行了隐藏域的赋值),但是前台的js监视,却捕获不到change事件。

是不是这个赋值,将页面进行了刷新,或是这个赋值操作根本没有提交,反正没有起到作用。

(4)iframe下载

此方法,未进行测试,不知道效果如何。

使用iframe进行下载,监视iframe的加载完毕,然后进行进度条的隐藏。

此方法,将隐藏进度条的操作全部放到了js中,但是需要创建一个新的页面。

ASP.NET中二进制流下载文件时进度条的使用的更多相关文章

  1. Android中的常用控件之进度条(ProgressBar)

    ProgressBar的常用属性:style,进度条的样式,默认为圆形,用style="?android:attr/progressBarStyleHorizontal"可以将进度 ...

  2. Asp.net WebAPI 使用流下载文件注意事项

    public HttpResponseMessage Post(string version, string environment, string filetype) { var path = @& ...

  3. (转)客户端触发Asp.net中服务端控件事件

    第一章. Asp.net中服务端控件事件是如何触发的 Asp.net 中在客户端触发服务端事件分为两种情况: 一. WebControls中的Button 和HtmlControls中的Type为su ...

  4. asp.net C# 未能加载文件或程序集或它的某一个依赖项。需要强名称程序集。的解决办法

    asp.net C# 未能加载文件或程序集或它的某一个依赖项.需要强名称程序集.的解决办法 出现这个错误是原因:是有签名的DLL引用了无签名的DLL 如上图所示,就是因为引用Entity.MVCEnt ...

  5. asp.net中的ListBox控件添加双击事件

    问题:在Aspx页里的ListBox A中添加双击事件,将选中项添加到另一个ListBox B中,双击ListBox B中的选中项,删除当前选中项 页面: <asp:ListBox ID=&qu ...

  6. ASP.NET中 WebForm 窗体控件使用及总结【转】

    原文链接:http://www.cnblogs.com/ylbtech/archive/2013/03/06/2944675.html ASP.NET中 WebForm 窗体控件使用及总结. 1.A, ...

  7. MFC读写.txt文件时进度条显示实时进度

    整体实现方式:先获得文件长度,然后用每次读取的长度,计算出完成的百分比,用百分比的值设置进度条. 一.MFC进度条 Progress Control 相关函数 1. create() --创建Prog ...

  8. MFC控件编程进度条编写

    MFC控件编程进度条编写 一丶进度条编程需要用到的方法 进度条MFC已经帮我们封装好类了. 叫做 CProgressCtrl  进度条编程也很简单. 封装的方法也就那个那几个. GetPos()  获 ...

  9. 如何在Vue项目中给路由跳转加上进度条

    1.前言 在平常浏览网页时,我们会注意到在有的网站中,当点击页面中的链接进行路由跳转时,页面顶部会有一个进度条,用来标示页面跳转的进度(如下图所示).虽然实际用处不大,但是对用户来说,有个进度条会大大 ...

随机推荐

  1. Python爬虫实现翻译功能

    前言 学了这么久的python理论知识,需要开始实战来练手巩固了. 准备 首先安装爬虫urllib库 pip install urllib 获取有道翻译的链接url 需要发送的参数在form data ...

  2. Nginx搭建文件共享服务器

    前言 Nginx除了做正反向代理和负载均衡,还能做动静分离服务器,如此便可以当作文件共享服务器使用. 环境 WIN 10 Vmware Workstation 15 Player CentOS Lin ...

  3. Jmeter-记一次AES加密登录实例

    前言 公司有个网站系统,用户名是明文,密码是加密的,所以搞了好久才登录进去,因此记录下艰辛过程. Part 1   了解加密算法 找研发同事去了解这个是怎样一个加密过程,最后得到的结论是:后端会生成一 ...

  4. Linux 学习笔记01丨Ubuntu系统安装、配置及软件教程集合

    1. Ubuntu系统安装 Windows10安装ubuntu18.04双系统教程 Ubuntu 20.04.1 镜像下载 软碟通 机械革命进入BIOS模式 要按F2,注意将Boot Option中将 ...

  5. Java基础教程——正则表达式

    正则表达式·Regular Expression 正则表达式就是一个用于匹配字符串的模板,正则表达式并不仅限于某一种语言,但是在每种语言中有细微的差别. 主要用到的对象: java.util.rege ...

  6. 课时四:Action操作

    参考文档 瑞泰信息技术有限公司 微软MVP(15-18)罗勇 语义说明 核心名词 ​ FetchXML:是基于Xml的查询语言,可以把它简单理解成SQL语句,通过它可以查询Dynamics 365 C ...

  7. 新手上路之如何选择Java版本

    @ 目录 LTS与非LTS LTS 非LTS Java CPU与PSU Java SE.Java EE.Java ME的区别 Java SE Java EE Java ME 每一次JDK上新总有一群人 ...

  8. kali putty远程连接允许以root身份登录

    原文链接:https://blog.csdn.net/long_long_chuang/article/details/70227874 kali linux通过ssh+putty来实现远程登录(亲测 ...

  9. vue获取浏览器地址栏参数(?及/)路由+非路由实现方式

    1.? 参数 浏览器参数形式:http://javam4.com/m4detail?id=1322914793170014208 1.1.路由取参方式 this.$route.query.id 前端跳 ...

  10. C++编程指南续(4-5)

    五.常量 常量是一种标识符,它的值在运行期间恒定不变.C语言用 #define来定义常量(称为宏常量).C++ 语言除了 #define外还可以用const来定义常量(称为const常量). 5.1  ...