zoukankan      html  css  js  c++  java
  • 基于HT for Web矢量实现HTML5文件上传进度条

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


    从效果图能够看到,向server上传了一个mp4文件,并在最下方显示当前上传进度。

    那么接下来我们就来探讨下详细实现:

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

    1. 须要一个背景。background

    2. 须要一个当前进度值。value

    3. 须要一个前景,foreground,依据当前进度值,绘制前景。盖过背景

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

    ht.Default.setImage('progress', {
        width : 150,
        height : 12,
        comps : [
            // 绘制背景
            {
                type : 'rect',
                rect : {x : 0, y : 0, width : 115, height : 12},
                background : {func : function(data) {return data.a('background');}}
            },
            // 绘制前景
            {
                rect : {x : 0, y : 0, width : 115, height : 12},
                type : function(g, rect, comp, data, view) {
                    var width = rect.width, height = rect.height, value = data.getValue(), 
                        foreWidth = width / 100 * value;
                    g.fillStyle = data.a('foreground');
                    g.fillRect(0, 0, foreWidth, height);
                }
            }
        ]
    });

    我们定义了一个名字为progress的矢量对象,矢量对象由两部分组成,一个是背景。一个是前景。

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

    矢量的大体设计已经完毕。那么我们把他用起来。看看效果怎样。

    var dataModel = new ht.DataModel(),
        node = new ht.Node();
    node.setValue(0);
    node.setImage('progress');
    node.a('background', '#5F6870');
    node.a('foreground', '#58B6DA');
    node.p(85, 87);
    dataModel.add(node);
    var graphView = new ht.graph.GraphView(dataModel);
    graphView.addToDOM();
    graphView.layout({x : 0, y : 80, width : 170, height : 30});

    我们创建了一个node,并将node的image属性设置成我们定义的矢量,然后创建一个graphView组件,将node显示在graphView网络拓扑图中。

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

    function setProgressValue(node) {
        var value = node.getValue();
        if (value !== 100) {
            node.setValue(value + 1);
            var second = Math.round(Math.random() * 500);
            setTimeout(setProgressValue, second, node);
        }
    }

    我们通过setTimeout()方法不断设置node的value值。可是,代码执行起来你会发现。进度条根本没有在动。一致处于初始状态,当我们缩放graphView时,能够看到进度条在改变,这是为什么呢?事实上原因非常easy。我们在改动value值时,并没有通知graphView要更新,因此进度条并不会由于node的value值改变而有所改变,那么我们该怎样通知graphView更新呢?方法非常easy,在改动node的value值后,派发一个propertyChange事件就能够了,在创建node代码后加入例如以下代码:

    node.getValue = function() {
        return this._value;
    };
    node.setValue = function(value) {
        var self = this,
            oV = self._value;
        self._value = value;
        self.fp('value', oV, value);
    };

    代码中。通过fp()方法来派发propertyChange事件,如此,进度条就能够正常工作,随着node的value的变化而变化,效果图例如以下:


    可是另一点不足,进度条尽管在跑了,可是我们还是不知道当前进度值是多少,仅仅能通过进度条的比重来大致预计当前进度值,我们是否能在进度条上加入一个文本,用来显示当前进度值呢。答案是肯定的,我们仅仅须要在矢量的comps中加入例如以下代码就能够:

    // 绘制文本
    {
        rect : {x : 118, y : 0, width : 32, height : 12},
        type : 'text',
        text : {func : function(data) {return data.getValue() + '%';}},
        font : '12px arial, sans-ferif',
        color : 'black'
    }

    代码中相同适用了绑定,绑定node当前的value值,详细的效果图例如以下:


    如今的进度条与终于效果就差圆角了,那么圆角要怎样实现呢?事实上也不难,仅仅须要绘制出一个圆角矩形,并结合clip()方法将超出圆角矩形区域的部分截取掉就可以。clip()方法的具体介绍能够參考MDN中的介绍。

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

    /***
     * 绘制圆边矩形
     * @param ctx 画笔
     * @param x 坐标 x
     * @param y 坐标 y
     * @param width 宽度
     * @param height 高度
     * @param radius 圆角半径
     */
    function roundRect(ctx, x, y, width, height, radius) {
        ctx.beginPath();
        ctx.moveTo(x, y + radius);
        ctx.lineTo(x, y + height - radius);
        ctx.quadraticCurveTo(x, y + height, x + radius, y + height);
        ctx.lineTo(x + width - radius, y + height);
        ctx.quadraticCurveTo(x + width, y + height, x + width, y + height - radius);
        ctx.lineTo(x + width, y + radius);
        ctx.quadraticCurveTo(x + width, y, x + width - radius, y);
        ctx.lineTo(x + radius, y);
        ctx.quadraticCurveTo(x, y, x, y + radius);
    }
    

    2. 採用自己定义类型的方法,调用roundRect()方法。绘制一个圆角矩形区域,然后再调用clip()方法,截掉圆角矩形区域外部分。

    有一点须要注意,clip()方法截取的内容仅仅对调用该方法后绘制的内容有效,调用该方法之前绘制的内容并不会被截掉。

    因此下面代码必须放在绘制背景的代码前。

    // 绘制圆角矩形
    {
        rect : {x : 0, y : 0, width : 115, height : 12},
        type : function(g, rect, comp, data, view) {
            var width = rect.width, height = rect.height;
            roundRect(g, 0, 0, width, height, height / 2);
            g.clip();
        }
    }

    看下效果怎样


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

    1. 首先,我们须要有个server来接收文件,server中除了使用常规的webserver外(webserver的简单配置可參考:HT for Web的HTML5树组件延迟载入技术实现),还使用了formidable模块。下面是server的代码:

    var express = require('express'),
        app = express(),
        server = require('http').createServer(app),
        io = require('socket.io')(server),
        path = require('path'),
        root = path.join(__dirname, '../../../'),
        formidable = require('formidable');
    
    // io监听connection事件
    io.on('connection', function(socket){
        // 定义socket名称
        socket.join('upload');
    });
    
    // 设置服务器的工作路径
    app.use(express.static(root));
    
    app.post('/', function(req, res){
        var form = new formidable.IncomingForm();
    
        form.on('end', function(){
            res.end('upload complete!');
        });
        form.on('progress', function(bytesReceived, bytesExpected){
            var percent = Math.floor(bytesReceived / bytesExpected * 100);
            // 获取指定的socket。并派发事件
            io.sockets.in('upload').emit('progress', percent);
        });
        form.parse(req);
    });
    
    // 服务器监听4000端口
    server.listen(3000, function(){
        console.log('server is listening at port 3000');
    });

    2. 其次。我们须要设计一个文件上传的表单:

    <form method="post" action="/" enctype="multipart/form-data" name="fileForm">
        <p><input type="file" name="file"/></p>
        <p><input type="submit" value="Upload"/></p>
    </form>

    3. 再者,我们须要结合ajax无刷新向server上传文件。并结合socket技术监听server事件,在浏览器怎样使用socket能够參考:HT for Web的HTML5树组件延迟载入技术实现

    var fileForm = document.forms.namedItem('fileForm');
    fileForm.addEventListener('submit', function(e) {
        var httpRequest;
        if (window.XMLHttpRequest) { // Mozilla, Safari, IE7+ ...
            httpRequest = new XMLHttpRequest();
        }
        else if (window.ActiveXObject) { // IE 6 and older
            httpRequest = new ActiveXObject("Microsoft.XMLHTTP");
        }
    
        httpRequest.open('POST', '/', true);
        httpRequest.send(new FormData(fileForm));
    
        socket.on('progress', function(val) {
            progress.setValue(val);
        });
        e.preventDefault();
    }, false);

    如此,基于HT for Web自己定义类实现HTML5文件上传进度条的页面设计及代码设计所有完毕,因为篇幅的关系。在fromidable方面讲得比較少,还望见谅,以下我讲附上完整的代码,有兴趣的同学能够下载下来研究研究。

    点击下载源代码


    版权声明:本文博主原创文章,博客,未经同意不得转载。

  • 相关阅读:
    angular中scope的watch用法
    angular中对于no-repeat的优化——track by
    angular筛选器
    为什么我只贴代码不给你们源码?
    新添子节点却无法被之前的删除功能所删除,处理方法,给删除功能延迟,有好办法记得告诉我下哈,感激不进
    IDEA下Maven项目搭建踩坑记----3.最长的bug,最简单的错误。同一类中,部分函数的@AutoWired注入的对象失效
    IDEA下Maven项目搭建踩坑记----2.项目编译之后 在service层运行时找不到 com.dao.CarDao
    IDEA下Maven项目搭建踩坑记----1.pom,xml文件下${spring-version}不能用
    Myeclipse maven 配置有问题 改之后重启还是不好用
    前端 的一些css的写法
  • 原文地址:https://www.cnblogs.com/lcchuguo/p/4809378.html
Copyright © 2011-2022 走看看