傳、預(yù)覽、下載圖片是app常用的功能之一,一般的做法是建立一個(gè)圖片存儲(chǔ)服務(wù)器,再開(kāi)發(fā)一個(gè)接收前端上傳文件的后端服務(wù),接收到前端上傳的圖片文件,然后再調(diào)用存儲(chǔ)服務(wù)器的接口完成圖片文件的存儲(chǔ)。
現(xiàn)在阿里云提供了對(duì)象存儲(chǔ)云服務(wù)(OSS),非常方便APP直接將圖片等文件上傳上去,并提供了多種瀏覽、下載的方式,不僅操作簡(jiǎn)單、而且效率高、速度快,本文介紹三種使用方式,希望能夠讓工程師少走彎路。
這種方式只是把OSS作為文件存儲(chǔ)體替代了自建存儲(chǔ)器,上傳邏輯和路徑還是和傳統(tǒng)的方式一致,后端服務(wù)給前端提供上傳接口,在收到前端上傳的文件后,再調(diào)用阿里云的OSS SDK的API將文件上傳到阿里云的OSS存儲(chǔ)桶中。
上傳流程
詳細(xì)說(shuō)明及相關(guān)代碼請(qǐng)參考《云平臺(tái)對(duì)象存儲(chǔ)OSS》
這種方式更加的便捷、快速,無(wú)需開(kāi)發(fā)接收后端服務(wù),而是前端直接將文件上傳到OSS,這種方式就要求前端按照約定的參數(shù)訪問(wèn)OSS API,為了安全起見(jiàn)要按照阿里云SDK的方式生成相應(yīng)的簽名并提供給前端。
FORM表單直接上傳流程
這里要求提前生成簽名串,生成算法如下:
package aliyun;
import aliyun.pojo.ResultData;
import aliyun.pojo.SignatureData;
import com.alibaba.fastjson.JSONObject;
import org.apache.commons.codec.binary.Base64;
public class Controller {
private final static String POLICY="{\"expiration\":\"2025-12-31T23:59:59Z\",\"conditions\":[[\"content-length-range\",0,104857600]]}";
private String accessKeySecret=System.getenv("ACCESS_SECRET_CODE");
public String signatureCode(int times) throws Exception {
ResultData resultData=new ResultData();
//約束規(guī)則
String encodePolicy=new String(Base64.encodeBase64(POLICY.getBytes()));
// 生成簽名。
String signatureCom=com.aliyun.oss.common.auth.ServiceSignature.create().computeSignature(accessKeySecret, encodePolicy);
SignatureData signatureData=new SignatureData(encodePolicy,signatureCom);
resultData.setData(signatureData);
return JSONObject.toJSONString(resultData);
}
}
其中:POLICY定義了上傳有效時(shí)間和文件大小約束,簽名要用到阿里云賬號(hào)的Access Key Secret。這里的簽名是有時(shí)效性的,有效期內(nèi)不會(huì)改變,如果對(duì)安全要求很高,可以將該簽名過(guò)程作為服務(wù)發(fā)布,讓前端每次在上傳的時(shí)候調(diào)用服務(wù)獲取實(shí)時(shí)的簽名數(shù)據(jù)。
前端form表單的構(gòu)成如下:
<div id="upload">
<form action="http://mybucket.oss-cn-beijing.aliyuncs.com" method="post" enctype="multipart/form-data">
<p>阿里云AccessKeyID:<input type="text" name="OSSAccessKeyId" value="LTAI8N5***XX8cr"></p>
<p>約束規(guī)則base64:<input type="text" name="policy" value='eyJleHBpcmF0aW9uI*********zMVQyMzo1OTo1OVoiLCJjb25kaXRpb25zIjpbWyJjb250ZW50LWxlbmd0aC1yYW5nZSIsMCwxMDQ4NTc2MDBdXX0='></p>
<p>約束規(guī)則簽名:<input type="text" name="Signature" value="pgxE8VnN8kSJEWz*****QV7uBE="></p>
<p>OSS存儲(chǔ)文件名:<input type="text" name="key" value="upload/lee.jpg"></p>
<p>選擇上傳文件:<input name="file" type="file" id="file"></p>
<input name="submit" value="上傳" type="submit">
</form>
</div>
其中:action是你賬號(hào)下OSS存儲(chǔ)桶的endpoint URL,OSSAccessKeyId是你賬號(hào)下的AccessKey ID,policy是對(duì)上傳規(guī)則約束的base64編碼(這里千萬(wàn)不能直接使用你定義的policy字符串,而是該字符串的字節(jié)碼的base64編碼后的結(jié)果,即由上面的計(jì)算結(jié)果提供),Signature就是對(duì)policy的簽名結(jié)果,key是存儲(chǔ)到你指定bucket的目錄和文件名。
通過(guò)這種方式可以不刷新頁(yè)面完成文件直接上傳到OSS,建議采用plupload.js插件實(shí)現(xiàn)。
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="content-type" content="text/html; charset=UTF-8"/>
<title>OSS web直傳</title>
<link rel="stylesheet" type="text/css" href="style.css"/>
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no">
</head>
<body>
<h2>OSS web直傳---直接在JS簽名</h2>
<ol>
<li>基于plupload封裝 </li>
<li>支持html5,flash,silverlight,html4 等協(xié)議上傳</li>
<li>可以運(yùn)行在PC瀏覽器,手機(jī)瀏覽器,微信</li>
<li>可以選擇多文件上傳</li>
<li>顯示上傳進(jìn)度條</li>
<li>可以控制上傳文件的大小</li>
<li>最關(guān)鍵的是,讓你10分鐘之內(nèi)就能移植到你的系統(tǒng),實(shí)現(xiàn)以上牛逼的功能!</li>
<li>注意一點(diǎn),bucket必須設(shè)置了Cors(Post打勾),不然沒(méi)有辦法上傳</li>
<li>注意一點(diǎn),把upload.js 里面的host/accessid/accesskey改成您上傳所需要的信息即可</li>
<li>此方法是直接在前端簽名,有accessid/accesskey泄漏的風(fēng)險(xiǎn), 線上生產(chǎn)請(qǐng)使用后端簽名例子<a href="https://help.aliyun.com/document_detail/oss/practice/pc_web_upload/js_php_upload.html">點(diǎn)擊查看詳細(xì)文檔</a></li>
</ol>
<br>
<form name="theform">
<input type="radio" name="myradio" value="local_name" checked=true/> 上傳文件名字保持本地文件名字
<input type="radio" name="myradio" value="random_name" /> 上傳文件名字是隨機(jī)文件名
<br/>
上傳到指定目錄:<input type="text" id='dirname' placeholder="如果不填,默認(rèn)是上傳到根目錄" size=50>
</form>
<h4>您所選擇的文件列表:</h4>
<div id="ossfile">你的瀏覽器不支持flash,Silverlight或者HTML5!</div>
<br/>
<div id="container">
<a id="selectfiles" href="javascript:void(0);" class='btn'>選擇文件</a>
<a id="postfiles" href="javascript:void(0);" class='btn'>開(kāi)始上傳</a>
</div>
<pre id="console"></pre>
<p> </p>
</body>
<script type="text/javascript" src="lib/plupload-2.1.2/js/plupload.full.min.js"></script>
<script type="text/javascript" src="upload.js"></script>
</html>
upload.js代碼如下:
host='http://mybucket.oss-cn-beijing.aliyuncs.com';
accessid='LTAI8N***mXX8cr';
signature="pgxE8V******Wz1jKv7cQV7uBE=";
policyBase64="eyJleHBpcmF0aW9uIjoiMjAyNS0xMi0zMVQyMzo1OTo1OVoiLCJjb25kaXRpb25zIjpbWyJjb250ZW50LWxlbmd0aC1yYW5nZSIsMCwxMDQ4NTc2MDBdXX0=";
g_dirname=''
g_object_name=''
g_object_name_type=''
now=timestamp=Date.parse(new Date()) / 1000;
function check_object_radio() {
var tt=document.getElementsByName('myradio');
for (var i=0; i < tt.length ; i++ )
{
if(tt[i].checked)
{
g_object_name_type=tt[i].value;
break;
}
}
}
function get_dirname()
{
dir=document.getElementById("dirname").value;
if (dir !='' && dir.indexOf('/') !=dir.length - 1)
{
dir=dir + '/'
}
//alert(dir)
g_dirname=dir
}
function random_string(len) {
len=len || 32;
var chars='ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678';
var maxPos=chars.length;
var pwd='';
for (i=0; i < len; i++) {
pwd +=chars.charAt(Math.floor(Math.random() * maxPos));
}
return pwd;
}
function get_suffix(filename) {
pos=filename.lastIndexOf('.')
suffix=''
if (pos !=-1) {
suffix=filename.substring(pos)
}
return suffix;
}
function calculate_object_name(filename)
{
if (g_object_name_type=='local_name')
{
g_object_name +="${filename}"
}
else if (g_object_name_type=='random_name')
{
suffix=get_suffix(filename)
g_object_name=g_dirname + random_string(10) + suffix
}
return ''
}
function get_uploaded_object_name(filename)
{
if (g_object_name_type=='local_name')
{
tmp_name=g_object_name
tmp_name=tmp_name.replace("${filename}", filename);
return tmp_name
}
else if(g_object_name_type=='random_name')
{
return g_object_name
}
}
function set_upload_param(up, filename, ret)
{
g_object_name=g_dirname;
if (filename !='') {
suffix=get_suffix(filename)
calculate_object_name(filename)
}
new_multipart_params={
'key' : g_object_name,
'policy': policyBase64,
'OSSAccessKeyId': accessid,
'success_action_status' : '200', //讓服務(wù)端返回200,不然,默認(rèn)會(huì)返回204
'signature': signature,
};
up.setOption({
'url': host,
'multipart_params': new_multipart_params
});
up.start();
}
var uploader=new plupload.Uploader({
runtimes : 'html5,flash,silverlight,html4',
browse_button : 'selectfiles',
container: document.getElementById('container'),
flash_swf_url : 'lib/plupload-2.1.2/js/Moxie.swf',
silverlight_xap_url : 'lib/plupload-2.1.2/js/Moxie.xap',
url : 'http://oss.aliyuncs.com',
init: {
PostInit: function() {
document.getElementById('ossfile').innerHTML='';
document.getElementById('postfiles').onclick=function() {
set_upload_param(uploader, '', false);
return false;
};
},
FilesAdded: function(up, files) {
plupload.each(files, function(file) {
document.getElementById('ossfile').innerHTML +='<div id="' + file.id + '">' + file.name + ' (' + plupload.formatSize(file.size) + ')<b></b>'
+'<div class="progress"><div class="progress-bar" style="width: 0%"></div></div>'
+'</div>';
});
},
BeforeUpload: function(up, file) {
check_object_radio();
get_dirname();
set_upload_param(up, file.name, true);
},
UploadProgress: function(up, file) {
var d=document.getElementById(file.id);
d.getElementsByTagName('b')[0].innerHTML='<span>' + file.percent + "%</span>";
var prog=d.getElementsByTagName('div')[0];
var progBar=prog.getElementsByTagName('div')[0]
progBar.style.width=2*file.percent+'px';
progBar.setAttribute('aria-valuenow', file.percent);
},
FileUploaded: function(up, file, info) {
if (info.status==200)
{
document.getElementById(file.id).getElementsByTagName('b')[0].innerHTML='upload to oss success, object name:' + get_uploaded_object_name(file.name);
}
else
{
document.getElementById(file.id).getElementsByTagName('b')[0].innerHTML=info.response;
}
},
Error: function(up, err) {
document.getElementById('console').appendChild(document.createTextNode("\nError xml:" + err.response));
}
}
});
uploader.init();
其中:host為OSS存儲(chǔ)桶endpoint地址、accessid為阿里云賬號(hào)ID、signature為后端生成的簽名、policyBase64為規(guī)則json串的base64編碼
有多種方式可以查看你上傳的文件,詳細(xì)操作見(jiàn)阿里云OSS操作手冊(cè)。這里只說(shuō)明一種直接通過(guò)URL訪問(wèn)的方式。
針對(duì)該bucket設(shè)置為開(kāi)通“靜態(tài)頁(yè)面”、權(quán)限管理設(shè)置為“公共可讀”、防盜鏈設(shè)置“referer”等安全選項(xiàng),然后就可以通過(guò)類似
http://mybucket.oss-cn-beijing.aliyuncs.com/upload/lee
的方式訪問(wèn)你上傳的圖片等文件了。
托管JavaScript庫(kù)并提供用于獲取和打包它們的工具。
構(gòu)架
斷言
覆蓋
Runner
模板引擎允許您執(zhí)行字符串插值。
用于Web的數(shù)據(jù)可視化工具。
用于處理文件的庫(kù)。
函數(shù)式編程庫(kù),用于擴(kuò)展JavaScript的功能。
反應(yīng)式編程庫(kù),以擴(kuò)展JavaScript的功能。
數(shù)據(jù)結(jié)構(gòu)庫(kù)構(gòu)建更復(fù)雜的應(yīng)用程序。
日期庫(kù)。
字符串庫(kù)。
本地化(l10n)和國(guó)際化(i18n)JavaScript庫(kù)。
用于指示負(fù)載狀態(tài)的庫(kù)。
鍵盤包裝
旅游和指南
輸入
日歷
選擇
文件上傳器
其他
模態(tài)和彈出窗口
滾動(dòng)
菜單
表/網(wǎng)格
你不懂JS - 可能是用現(xiàn)代JavaScript編寫(xiě)的最好的書(shū),完全可以免費(fèi)在線閱讀,或者可以買來(lái)支持作者。
原文:https://github.com/sorrycc/awesome-javascript
本文:https://pub.intelligentx.net/collection-awesome-browser-side-javascript-libraries-resources-and-shiny-things
討論:請(qǐng)加入知識(shí)星球或者小紅圈【首席架構(gòu)師圈】
一節(jié)增加推薦圖書(shū)的提交和刪除功能,來(lái)學(xué)習(xí)node的form提交以及node的圖片上傳功能。開(kāi)始之前需要源碼同學(xué)可以先在git上fork:https://github.com/stoneniqiu/ReadingClub
一、form驗(yàn)證
MVC的form驗(yàn)證有三個(gè)地方可以做,第一道關(guān)就是前端提交之前,第二道關(guān)就是在數(shù)據(jù)保存之前,也就是在controller中做驗(yàn)證,第三道關(guān)就是數(shù)據(jù)保存的時(shí)候,也就是如果提交的數(shù)據(jù)模型不符合實(shí)體定義的約束,數(shù)據(jù)是無(wú)法保存的,這是最后一道防線。第一道關(guān)主要是依賴于js或者jquery框架,比較常用的是jquery.validate.js。如果是Asp.net MVC 可以自動(dòng)生成驗(yàn)證規(guī)則,這里就不細(xì)究了,網(wǎng)上有很多文章。第二層和各自的業(yè)務(wù)邏輯有關(guān),也需要做一些必要驗(yàn)證,防止前端禁止JavaScript,而提交不合法數(shù)據(jù),這里是要講基于Mongoose的第三層驗(yàn)證。
我們先回顧一下之前用Mongoose定義的book模型:
var bookSchema=new mongoose.Schema({ title: { type: String, required: true }, rating: { type: Number, required: true, min: 0, max: 5 }, info: { type: String, required: true }, img: String, tags: [String], brief: { type: String, required: true }, ISBN: String, });
每個(gè)屬性定義了類型和是否必須,還可以添加min,max,默認(rèn)值等其他約束。如果提交的模型不滿足這些約束,將不能保存成功。相當(dāng)于Asp.net MVC中的DataAnnotations的作用。后面的form驗(yàn)證就基于此。
我們需要增加4個(gè)路由規(guī)則,2個(gè)用于添加(一個(gè)get,一個(gè)post),一個(gè)用于刪除,一個(gè)用于上傳圖片:
router.get('/book/create', homeController.bookcreateview); router.post('/book/create', homeController.doBookCreate); router.delete('/book/:id', homeController.delete); router.post('/uploadImg', homeController.uploadImg);
基于Express的路由,我們可以創(chuàng)建Restful的路由規(guī)則。路由位于app_server文件夾下。
home.bookcreateview:
module.exports.bookcreateview=function (req, res) { res.render('bookCreate', { title: '新增推薦圖書(shū)' }); };
這里直接是一個(gè)get請(qǐng)求,所以直接用render去渲染視圖,當(dāng)然這個(gè)bookCreate視圖接下來(lái)會(huì)創(chuàng)建。
doBookCreate:
module.exports.doBookCreate=function (req, res) { var requestOptions, path, postdata; path="/api/book"; postdata={ title: req.body.title, info: req.body.info, ISBN: req.body.ISBN, brief: req.body.brief, tags: req.body.tags, img: req.body.img, rating:req.body.rating, }; requestOptions={ url: apiOptions.server + path, method: "POST", json: postdata, }; request(requestOptions, function (err, response, body) { console.log("body.name", body.name, response.statusCode); if (response.statusCode===201) { res.redirect("/detail/"+body._id); } else if (response.statusCode==400 && body.name && body.name=="ValidationError") { res.render('bookCreate', { title: '新增推薦圖書(shū)', error:"val"}); } else { console.log("body.name",body.name); info(res, response.statusCode); } }); };
info:
function info (res, status) { var title, content; if (status===404) { title="404, 頁(yè)面沒(méi)有找到"; content="親,頁(yè)面飛走了..."; } else if (status===500) { title="500, 內(nèi)部錯(cuò)誤"; content="尷尬...,發(fā)生錯(cuò)誤"; } else { title=status + ", 有什么不對(duì)勁"; content="某些地方可能有些錯(cuò)誤"; } res.status(status); res.render('info', { title : title, content : content, status: status, }); };
View Code
在上一節(jié),我們創(chuàng)建了數(shù)據(jù)操作的api部分。代碼的流程就是先從req中獲取到前端傳過(guò)來(lái)的數(shù)據(jù),然后用request模塊調(diào)用api,如果添加成功(狀態(tài)碼是201)就返回到detail頁(yè)面,如果驗(yàn)證失敗,就原路返回,并給出提示。如果錯(cuò)誤,交給info方法去處理。
delete:
module.exports.delete=function (req, res) { var requestOptions, path; path="/api/book/" + req.params.id; requestOptions={ url: apiOptions.server + path, method: "delete", json: {}, }; request(requestOptions, function (err, response, body) { if (response.statusCode==204) { res.json(1); } else { res.json(0); } }); };
如果刪除成功,返回的狀態(tài)碼是204,然后返回json(1)讓前端去處理界面。
1) 先需要在圖書(shū)列表的右側(cè)邊欄增加一個(gè)按鈕:
在books視圖中修改:
.col-md-3 .userinfo p stoneniqiu a(href='/book/create').btn.btn-info 新增推薦
當(dāng)用戶點(diǎn)擊會(huì)跳轉(zhuǎn)到/book/create頁(yè)面
2)新增推薦頁(yè)面:
extends layout include _includes/rating block content .row .col-md-12.page.bookdetail h3 新增推薦書(shū)籍 .row .col-xs-12.col-md-6 form.form-horizontal(action='',method="post",role="form") - if (error=="val") .alert.alert-danger(role="alert") All fields required, please try again .form-group label.control-label(for='title') 書(shū)名 input#name.form-control(name='title') .form-group label.control-label(for='info') 信息 input#name.form-control(name='info') .form-group label.control-label(for='ISBN') ISBN input#name.form-control(name='ISBN') .form-group label.control-label(for='brief') 簡(jiǎn)介 input#name.form-control(name='brief') .form-group label.control-label(for='tags') 標(biāo)簽 input#name.form-control(name='tags') .form-group label.control-label(for='rating') 推薦指數(shù) select#rating.form-control.input-sm(name="rating") option 5 option 4 option 3 option 2 option 1 .form-group p 上傳圖片 a.btn.btn-info(id="upload", name="upload") 上傳圖片 br img(id='img') .form-group button.btn.btn-primary(type='submit') 提交
if語(yǔ)句的地方是用來(lái)顯示錯(cuò)誤提示;圖片上傳,稍后完整介紹;所以提交頁(yè)面基本長(zhǎng)成這樣:
3)Mongoose驗(yàn)證
這個(gè)時(shí)候沒(méi)有加前端驗(yàn)證,form可以直接提交。但是node打印出了錯(cuò)誤日志,Book validation failed,驗(yàn)證失敗。
這是Mongoose給我們返回的驗(yàn)證信息,這時(shí)界面上回顯示一個(gè)提示信息:
這是因?yàn)樵赾ontroller中的處理:
else if (response.statusCode==400 && body.name && body.name=="ValidationError") { res.render('bookCreate', { title: '新增推薦圖書(shū)', error:"val"}); }
以上說(shuō)明了Mongoose會(huì)在數(shù)據(jù)保存的時(shí)候驗(yàn)證實(shí)體,如果實(shí)體不滿足path規(guī)則,將不能保存。但至此有三個(gè)問(wèn)題,第一個(gè)問(wèn)題是提示信息不明確,當(dāng)然我們可以遍歷輸出ValidatorError;第二個(gè)就是,驗(yàn)證錯(cuò)誤之后,頁(yè)面原來(lái)的數(shù)據(jù)沒(méi)有了,需要再輸入一遍,這個(gè)我們可以參考Asp.net MVC將模型數(shù)據(jù)填充到視圖中可以解決;第三個(gè)問(wèn)題就是頁(yè)面前端還沒(méi)有驗(yàn)證,form直接就可以提交了,這個(gè)可以通過(guò)簡(jiǎn)單的Jquery腳本就可以做到;這三點(diǎn)先不細(xì)究。繼續(xù)往下看,如果規(guī)范輸入,這個(gè)時(shí)候是可以提交的,提交之后在books頁(yè)面可以看到:
4)刪除
在標(biāo)題的右側(cè)增加了一個(gè)刪除符號(hào)(books視圖中):
.col-md-10 p a(href="/Detail/#{book._id}")=book.title span.close(data-id='#{book._id}') ×
并添加腳本:
$(".close").click(function { if (confirm("確定刪除?")) { var id=$(this).data("id"); var row=$(this).parents(".booklist"); $.ajax({ url: "/book/" + id, method: "delete", }).done(function(data) { console.log(data); row.fadeOut; }); } });
腳本可以先位于layout視圖下方:
script(src='/javascripts/books.js')
這樣,刪除完成之后會(huì)隱藏當(dāng)前行。下面解決圖片上傳問(wèn)題。
二、圖片上傳
前面我們?cè)诼酚衫锩娑x了一個(gè)uploadimg方法,現(xiàn)在實(shí)現(xiàn)它。一般都涉及兩個(gè)部分,一個(gè)是前臺(tái)圖片的提交,一個(gè)是后端數(shù)據(jù)的處理。
先需要安裝formidable模塊。
然后在Public文件下創(chuàng)建一個(gè)upload/temp文件夾
腳本:
var fs=require('fs'); var formidable=require('formidable'); module.exports.uploadImg=function (req, res) { var form=new formidable.IncomingForm; //創(chuàng)建上傳表單 form.encoding='utf-8'; //設(shè)置編輯 form.uploadDir='./../public/upload/temp/'; //設(shè)置上傳目錄 form.keepExtensions=true; //保留后綴 form.maxFieldsSize=3 * 1024 * 1024; //文件大小 form.parse(req, function(err, fields, files) { console.log(files); if (err) { console.log(err); return res.json(0); } for (var key in files) { console.log(files[key].path); var extName=''; //后綴名 switch (key.type) { case 'image/pjpeg': extName='jpg'; break; case 'image/jpeg': extName='jpg'; break; case 'image/png': case 'image/x-png': default: extName='png'; break; } var avatarName=(new Date).getTime + '.' + extName; var newPath=form.uploadDir + avatarName; fs.renameSync(files[key].path, newPath); //重命名 return res.json("/upload/temp/"+ avatarName); } }); };
這個(gè)form會(huì)自動(dòng)將文件保存到upLoadDir目錄,并以u(píng)pload_xxxx格式重新命名,所以最后使用fs模塊對(duì)文件進(jìn)行重命名。然后返回給前端。
我喜歡用插件,前端我用的是plupload-2.1.8,擁有多種上傳方式,比較方便。放置在Public文件下。在layout.jade中引用js:
script(src='/plupload-2.1.8/js/plupload.full.min.js') script(src='/javascripts/books.js')
而在bookCreate.jade視圖中,修改如下:
a.btn.btn-info(id="upload", name="upload") 上傳圖片 br img(id='img') input#imgvalue(type='hidden',name='img',value='')
a標(biāo)簽用來(lái)觸發(fā)上傳,img用來(lái)預(yù)覽,input用來(lái)存放路徑。在books.js下增加以下代碼:
var uploader=new plupload.Uploader({ runtimes: 'html5,flash,silverlight,html4', browse_button: "upload", url: '/uploadImg', flash_swf_url: '/plupload-2.1.8/js/Moxie.swf', silverlight_xap_url: '/plupload-2.1.8/js/Moxie.xap', filters: { max_file_size: "3mb", mime_types: [ { title: "Image files", extensions: "jpg,gif,png" }, { title: "Zip files", extensions: "zip" } ] }, init: { PostInit: function { }, FilesAdded: function (up, files) { plupload.each(files, function (file) { uploader.start; }); }, UploadProgress: function (up, file) { }, Error: function (up, err) { } } }); uploader.init; uploader.bind('FileUploaded', function (upldr, file, object) { var data=JSON.parse(object.response); console.log(data); $("#img").attr("src", data); $("#imgvalue").val(data); });
提交:
上傳成功后跳轉(zhuǎn)到detail頁(yè)面。
至此,圍繞form的提交這一節(jié)學(xué)習(xí)了Mongoose的數(shù)據(jù)驗(yàn)證,以及使用plupload上傳,以及后端用formidable和fs模塊處理圖片。相對(duì)于Asp.net MVC而言,Asp.net MVC因?yàn)橛凶詣?dòng)化的form相對(duì)快捷一些。下一節(jié)將介紹Angular,作為MEAN中的A,該出場(chǎng)了。
*請(qǐng)認(rèn)真填寫(xiě)需求信息,我們會(huì)在24小時(shí)內(nèi)與您取得聯(lián)系。