Chinaunix首页 | 论坛 | 博客
  • 博客访问: 955421
  • 博文数量: 110
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 1997
  • 用 户 组: 普通用户
  • 注册时间: 2013-12-15 22:39
个人简介

HT for Web 3D. HT for modern web and mobile app development. www.hightopo.com

文章分类

全部博文(110)

文章存档

2020年(1)

2018年(28)

2017年(21)

2016年(10)

2015年(28)

2014年(19)

2013年(3)

我的朋友

分类: HTML5

2015-06-08 21:26:36

在HTML中,在文件上传的过程中,很多情况都是没有任何的提示,这在体验上很不好,用户都不知道到时有没有在上传、上传成功了没有,所以今天给大家介绍的内容是通过矢量来实现HTML5文件上传进度条,矢量在《矢量Chart图表嵌入HTML5网络拓扑图的应用》一文中已经讲述了关于setCompType()方法的应用,今天我们用setImage()方法充分利用系统中定义好的矢量资源来实现文件上传进度条,我们先来看下效果图:


从效果图可以看到,向服务器上传了一个mp4文件,并在最下方显示当前上传进度。

那么接下来我们就来探讨下具体实现:

首先,我们来分析下进度条的结构:

1. 需要一个背景,background

2. 需要一个当前进度值,value

3. 需要一个前景,foreground,根据当前进度值,绘制前景,盖过背景

结构就这么简单,那么接下来就是具体的实现了,看码:


点击(此处)折叠或打开

  1. ht.Default.setImage('progress', {
  2.     width : 150,
  3.     height : 12,
  4.     comps : [
  5.         // 绘制背景
  6.         {
  7.             type : 'rect',
  8.             rect : {x : 0, y : 0, width : 115, height : 12},
  9.             background : {func : function(data) {return data.a('background');}}
  10.         },
  11.         // 绘制前景
  12.         {
  13.             rect : {x : 0, y : 0, width : 115, height : 12},
  14.             type : function(g, rect, comp, data, view) {
  15.                 var width = rect.width, height = rect.height, value = data.getValue(),
  16.                     foreWidth = width / 100 * value;
  17.                 g.fillStyle = data.a('foreground');
  18.                 g.fillRect(0, 0, foreWidth, height);
  19.             }
  20.         }
  21.     ]
  22. });
我们定义了一个名字为progress的矢量对象,矢量对象由两部分组成,一个是背景,一个是前景。


绘制背景采用了数据绑定的方式,绑定了data的background属性;绘制前景则采用自定义类型的方法绘制,是setCompType()方法的一种缩写,绘制是根据data中的value值计算绘制宽度。

矢量的大体设计已经完成,那么我们把他用起来,看看效果如何。


点击(此处)折叠或打开

  1. var dataModel = new ht.DataModel(),
  2.     node = new ht.Node();
  3. node.setValue(0);
  4. node.setImage('progress');
  5. node.a('background', '#5F6870');
  6. node.a('foreground', '#58B6DA');
  7. node.p(85, 87);
  8. dataModel.add(node);
  9. var graphView = new ht.graph.GraphView(dataModel);
  10. graphView.addToDOM();
  11. graphView.layout({x : 0, y : 80, width : 170, height : 30});
我们创建了一个node,并将node的image属性设置成我们定义的矢量,然后创建一个graphView组件,将node显示在中。


那么接下来我们来模拟文件上传进度,让进度条动起来。


点击(此处)折叠或打开

  1. function setProgressValue(node) {
  2.     var value = node.getValue();
  3.     if (value !== 100) {
  4.         node.setValue(value + 1);
  5.         var second = Math.round(Math.random() * 500);
  6.         setTimeout(setProgressValue, second, node);
  7.     }
  8. }
我们通过setTimeout()方法不断设置node的value值,但是,代码运行起来你会发现,进度条根本没有在动,一致处于初始状态,当我们缩放graphView时,可以看到进度条在改变,这是为什么呢?其实原因很简单,我们在修改value值时,并没有通知graphView要更新,因此进度条并不会因为node的value值改变而有所改变,那么我们该如何通知graphView更新呢?方法很简单,在修改node的value值后,派发一个propertyChange事件就可以了,在创建node代码后添加如下代码:



点击(此处)折叠或打开

  1. node.getValue = function() {
  2.     return this._value;
  3. };
  4. node.setValue = function(value) {
  5.     var self = this,
  6.         oV = self._value;
  7.     self._value = value;
  8.     self.fp('value', oV, value);
  9. };
代码中,通过fp()方法来派发propertyChange事件,如此,进度条就可以正常工作,随着node的value的变化而变化,效果图如下:



但是还有一点不足,进度条虽然在跑了,但是我们还是不知道当前进度值是多少,只能通过进度条的比重来大致估计当前进度值,我们能否在进度条上添加一个文本,用来显示当前进度值呢,答案是肯定的,我们只需要在矢量的comps中添加如下代码就可以:


点击(此处)折叠或打开

  1. // 绘制文本
  2. {
  3.     rect : {x : 118, y : 0, width : 32, height : 12},
  4.     type : 'text',
  5.     text : {func : function(data) {return data.getValue() + '%';}},
  6.     font : '12px arial, sans-ferif',
  7.     color : 'black'
  8. }
代码中同样适用了绑定,绑定node当前的value值,具体的效果图如下:




现在的进度条与最终效果就差圆角了,那么圆角要如何实现呢?其实也不难,只需要绘制出一个圆角矩形,并结合clip()方法将超出圆角矩形区域的部分截取掉即可,clip()方法的详细介绍可以参考MDN中的介绍。

1. 首先,我们需要创建一个挥之圆角矩形的方法:


点击(此处)折叠或打开

  1. /***
  2.  * 绘制圆边矩形
  3.  * @param ctx 画笔
  4.  * @param x 坐标 x
  5.  * @param y 坐标 y
  6.  * @param width 宽度
  7.  * @param height 高度
  8.  * @param radius 圆角半径
  9.  */
  10. function roundRect(ctx, x, y, width, height, radius) {
  11.     ctx.beginPath();
  12.     ctx.moveTo(x, y + radius);
  13.     ctx.lineTo(x, y + height - radius);
  14.     ctx.quadraticCurveTo(x, y + height, x + radius, y + height);
  15.     ctx.lineTo(x + width - radius, y + height);
  16.     ctx.quadraticCurveTo(x + width, y + height, x + width, y + height - radius);
  17.     ctx.lineTo(x + width, y + radius);
  18.     ctx.quadraticCurveTo(x + width, y, x + width - radius, y);
  19.     ctx.lineTo(x + radius, y);
  20.     ctx.quadraticCurveTo(x, y, x, y + radius);
  21. }
2. 采用自定义类型的方法,调用roundRect()方法,绘制一个圆角矩形区域,然后再调用clip()方法,截掉圆角矩形区域外部分。有一点需要注意,clip()方法截取的内容只对调用该方法后绘制的内容有效,调用该方法之前绘制的内容并不会被截掉。因此以下代码必须放在绘制背景的代码前。



点击(此处)折叠或打开

  1. // 绘制圆角矩形
  2. {
  3.     rect : {x : 0, y : 0, width : 115, height : 12},
  4.     type : function(g, rect, comp, data, view) {
  5.         var width = rect.width, height = rect.height;
  6.         roundRect(g, 0, 0, width, height, height / 2);
  7.         g.clip();
  8.     }
  9. }
看下效果如何



至此,进度条的设计就结束了,那么接下来就来看下进度条如何与文件上传结合起来:

1. 首先,我们需要有个服务器来接收文件,服务器中除了使用常规的web服务器外(web服务器的简单配置可参考:HT for Web的HTML5树组件延迟加载技术实现),还使用了,以下是服务器的代码:


点击(此处)折叠或打开

  1. var express = require('express'),
  2.     app = express(),
  3.     server = require('http').createServer(app),
  4.     io = require('socket.io')(server),
  5.     path = require('path'),
  6.     root = path.join(__dirname, '../../../'),
  7.     formidable = require('formidable');
  8. // io监听connection事件
  9. io.on('connection', function(socket){
  10.     // 定义socket名称
  11.     socket.join('upload');
  12. });
  13. // 设置服务器的工作路径
  14. app.use(express.static(root));
  15. app.post('/', function(req, res){
  16.     var form = new formidable.IncomingForm();
  17.     form.on('end', function(){
  18.         res.end('upload complete!');
  19.     });
  20.     form.on('progress', function(bytesReceived, bytesExpected){
  21.         var percent = Math.floor(bytesReceived / bytesExpected * 100);
  22.         // 获取指定的socket,并派发事件
  23.         io.sockets.in('upload').emit('progress', percent);
  24.     });
  25.     form.parse(req);
  26. });
  27. // 服务器监听4000端口
  28. server.listen(3000, function(){
  29.     console.log('server is listening at port 3000');
  30. });
2. 其次,我们需要设计一个文件上传的表单:



点击(此处)折叠或打开

  1. <form method="post" action="/" enctype="multipart/form-data" name="fileForm">
  2.     <p><input type="file" name="file"/></p>
  3.     <p><input type="submit" value="Upload"/></p>
  4. </form>
3. 再者,我们需要结合ajax无刷新向服务器上传文件,并结合socket技术监听服务器事件,在浏览器如何使用socket可以参考:HT for Web的HTML5树组件延迟加载技术实现



点击(此处)折叠或打开

  1. var fileForm = document.forms.namedItem('fileForm');
  2. fileForm.addEventListener('submit', function(e) {
  3.     var httpRequest;
  4.     if (window.XMLHttpRequest) { // Mozilla, Safari, IE7+ ...
  5.         httpRequest = new XMLHttpRequest();
  6.     }
  7.     else if (window.ActiveXObject) { // IE 6 and older
  8.         httpRequest = new ActiveXObject("Microsoft.XMLHTTP");
  9.     }
  10.     httpRequest.open('POST', '/', true);
  11.     httpRequest.send(new FormData(fileForm));
  12.     socket.on('progress', function(val) {
  13.         progress.setValue(val);
  14.     });
  15.     e.preventDefault();
  16. }, false);
如此,基于HT for Web自定义类实现HTML5文件上传进度条的页面设计及代码设计全部完成,由于篇幅的关系,在fromidable方面讲得比较少,还望见谅,下面我讲附上完整的代码,有兴趣的同学可以下载下来研究研究。



阅读(2720) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~