HTML5矢量實現文件上傳進度條

在HTML中,在文件上傳的過程當中,不少狀況都是沒有任何的提示,這在體驗上很很差,用戶都不知道到時有沒有在上傳、上傳成功了沒有,因此今天給你們介紹的內容是經過HT for Web矢量來實現HTML5文件上傳進度條,矢量在《矢量Chart圖表嵌入HTML5網絡拓撲圖的應用》一文中已經講述了關於setCompType()方法的應用,今天咱們用setImage()方法充分利用系統中定義好的矢量資源來實現文件上傳進度條,咱們先來看下效果圖:html

082017203015232.png

從效果圖能夠看到,向服務器上傳了一個mp4文件,並在最下方顯示當前上傳進度。node

那麼接下來咱們就來探討下具體實現:git

首先,咱們來分析下進度條的結構:github

  1. 須要一個背景,backgroundweb

  2. 須要一個當前進度值,valueajax

  3. 須要一個前景,foreground,根據當前進度值,繪製前景,蓋過背景express

結構就這麼簡單,那麼接下來就是具體的實現了,看碼:瀏覽器

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時,能夠看到進度條在改變,這是爲何呢?其實緣由很簡單,咱們在修改value值時,並無通知graphView要更新,所以進度條並不會由於node的value值改變而有所改變,那麼咱們該如何通知graphView更新呢?方法很簡單,在修改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的變化而變化,效果圖以下:

082019212076335.png

可是還有一點不足,進度條雖然在跑了,可是咱們仍是不知道當前進度值是多少,只能經過進度條的比重來大體估計當前進度值,咱們可否在進度條上添加一個文本,用來顯示當前進度值呢,答案是確定的,咱們只須要在矢量的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);
}
  1. 採用自定義類型的方法,調用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();
    }
}

看下效果如何

082021106132160.png

至此,進度條的設計就結束了,那麼接下來就來看下進度條如何與文件上傳結合起來:

  1. 首先,咱們須要有個服務器來接收文件,服務器中除了使用常規的web服務器外(web服務器的簡單配置可參考:HT for Web的HTML5樹組件延遲加載技術實現),還使用了formidable模塊,如下是服務器的代碼:

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');
});
  1. 其次,咱們須要設計一個文件上傳的表單:

<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>
  1. 再者,咱們須要結合ajax無刷新向服務器上傳文件,並結合socket技術監聽服務器事件,在瀏覽器如何使用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方面講得比較少,還望見諒,下面我講附上完整的代碼,有興趣的同窗能夠下載下來研究研究。

點擊下載源碼

相關文章
相關標籤/搜索