HTML では、ファイル アップロード プロセス中にプロンプトが表示されないことが多く、ユーザーはその時点でアップロードしているかどうかがわかりません。アップロードが成功したので、今日紹介するのは、Web ベクトル用の HTML5 ファイルのアップロード プログレス バーを実装することです。 setCompType() については、「ベクトル チャート グラフ埋め込みの応用」の記事で既に説明しています。 HTML5 ネットワーク トポロジ マップ」メソッドのアプリケーションです。今日は、setImage() メソッドを使用して、システムで定義されたベクトル リソースを最大限に活用して、ファイル アップロードの進行状況バーを実装します。まずレンダリングを見てみましょう:
レンダリングからわかるように、サーバーは mp4 ファイルをアップロードし、現在のアップロードの進行状況が下部に表示されます。
次に、具体的な実装について説明します:
まず、進行状況バーの構造を分析しましょう:
1. 背景が必要です
2. 現在の進行状況の値が必要です
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 という名前のベクトル オブジェクトを定義しました。 , ベクター オブジェクトは 2 つの部分で構成され、1 つは背景、もう 1 つは前景です。
背景はデータの背景属性をバインドするデータ バインディングを使用して描画され、前景は setCompType() メソッドの略称であるカスタム タイプ メソッドを使用して描画されます。描画はデータの値に基づいて計算されます。描画幅。
ベクターの大まかなデザインが完成したので、それを使ってどのように動作するかを見てみましょう。
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});
ノードを作成し、そのノードのイメージ属性を定義したベクトルに設定してから、graphView ネットワーク トポロジ図にノードを表示するために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() メソッドを通じてノードの値を継続的に設定しますが、コードを実行すると、graphView をズームするとプログレス バーがまったく動かず、初期状態になっていることがわかります。 、進行状況バーが変化していることがわかります。これはなぜですか?実際、理由は非常に単純です。値を変更するときに、graphView に更新を通知しないため、ノードの値が変化してもプログレス バーは変化しません。では、どのようにして chartView に更新を通知するのでしょうか。メソッドは非常に簡単です。ノードの値を変更した後、ノード コードを作成した後、次のコードを追加するだけです。
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. 首先,我们需要有个服务器来接收文件,服务器中除了使用常规的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');});
2. 其次,我们需要设计一个文件上传的表单:
3. 再者,我们需要结合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);
このようにして、HTML5 ファイルアップロードプログレスバーのコード設計が完了しました。スペースの都合上、fromidable については省略させていただきます。完全なコードは以下にあります。興味のある方はダウンロードして学習してください。
クリックしてソースコードをダウンロード