成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

基于Vue + Node.js + MongoDB的圖片上傳組件,實(shí)現(xiàn)圖片的預(yù)覽和刪除

hzx / 2775人閱讀

摘要:網(wǎng)址功能單圖多圖上傳圖片上傳預(yù)覽上傳進(jìn)度條分組上傳,分組查詢新建分組,刪除分組刪除圖片選擇圖片目錄結(jié)構(gòu)前端利用搭建,中引入子組件。實(shí)現(xiàn)分組的新增查詢刪除。利用模塊實(shí)現(xiàn)刪除文件功能。

公司要寫一些為自身業(yè)務(wù)量身定制的的組件,要基于Vue,寫完后擴(kuò)展了一下功能,選擇寫圖片上傳是因?yàn)樽约褐耙恢睂@個功能比較迷糊,所以這次好好了解了一下。演示在網(wǎng)址打開后的show.gif中。

使用技術(shù):Vue.js | node.js | express | MongoDB。
github網(wǎng)址:https://github.com/neroneroff...

功能

單圖多圖上傳

圖片上傳預(yù)覽

上傳進(jìn)度條

分組上傳,分組查詢

新建分組,刪除分組

刪除圖片

選擇圖片

目錄結(jié)構(gòu)

前端利用Vue搭建,Entry.vue中引入子組件Upload.vue。在Upload.vue中,使用input標(biāo)簽,上傳圖片,form表單提交數(shù)據(jù),但是from讓人很頭疼,提交后刷新頁面,所以給它綁定了一個隱藏的iframe標(biāo)簽來實(shí)現(xiàn)無刷新提交表單。

Dom中:

調(diào)用上傳函數(shù)提交完數(shù)據(jù)后:

upload();
document.forms[0].target="rfFrame";

圖片預(yù)覽
利用html5的fileReader對象

   let count = 0;//上傳函數(shù)外定義變量,記錄文件的數(shù)量,即遞歸的次數(shù)

/*-----------------------此段代碼在上傳函數(shù)中-------------------------------*/
   let fileReader = new FileReader();
   //解析圖片路徑,實(shí)現(xiàn)預(yù)覽
   fileReader.readAsDataURL(file.files[count]);
   fileReader.onload=()=>{
      previewData = {
         url:fileReader.result,//圖片預(yù)覽的img標(biāo)簽的src
         name:file.files[count].name,
         size:file.files[count].size,
      };
      //這段代碼在上傳函數(shù)中,所以在外面定義了一個_this = this,fileList為vue的data中定義的狀態(tài)
    _this.fileList.push(previewData);
   };

進(jìn)度條實(shí)現(xiàn)
在axios的配置中定義onUploadProgress函數(shù),接收參數(shù):progressEvent,利用它的兩個屬性:progressEvent.total和progressEvent.loaded(上傳的文件總字節(jié)數(shù)和已上傳的字節(jié)數(shù))
node寫接口,實(shí)現(xiàn)圖片的接收、查詢、刪除。實(shí)現(xiàn)分組的新增、查詢、刪除。利用Formidable模塊接收并處理前端傳過來的表單數(shù)據(jù)。利用fs模塊實(shí)現(xiàn)刪除文件功能。

let progress = 0;
let config = {
  headers: {"Content-Type": "multipart/form-data"},
  onUploadProgress (progressEvent){

    if(progressEvent.lengthComputable){
      progress = progressEvent.total/progressEvent.loaded;
      _this.$refs.progress[_this.$refs.progress.length-1].style.width = Number(progress).toFixed(2)*100+"%";
    }
  }
};

向formData中插入文件

formData = new FormData();
if(file.files[count]){
formData.append("file",file.files[count],file.files[count].name);

對于上傳方式,我這里統(tǒng)一采用依次上傳,無論是單圖多圖,單圖上傳一次就好,多圖則遞歸調(diào)用上傳函數(shù),直到遞歸次數(shù)等于圖片數(shù)量,停止遞歸。

上傳函數(shù)

      let file=$event.target,
      formData = new FormData();
      //遞歸調(diào)用自身,實(shí)現(xiàn)多文件依次上傳
      let _this = this;
      let count = 0;
      let previewData = {};


    uploadImage($event){
      let file=$event.target,
      formData = new FormData();
      //遞歸調(diào)用自身,實(shí)現(xiàn)多文件依次上傳
      let _this = this;
      let count = 0;
      let previewData = {};

      function upload(){
        //開始上傳時,滾到底部
        _this.$refs.picWrapper.scrollTop = _this.$refs.picWrapper.scrollHeight;
        //定義axios配置信息
        let progress = 0;
        let config = {
          headers: {"Content-Type": "multipart/form-data"},
          onUploadProgress (progressEvent){
            console.log(`進(jìn)度條的數(shù)量${_this.$refs.progress.length -1}`);
            if(progressEvent.lengthComputable){
              progress = progressEvent.total/progressEvent.loaded;
              //進(jìn)度條
              _this.$refs.progress[_this.$refs.progress.length-1].style.width = Number(progress).toFixed(2)*100+"%";
            }
          }
        };
        //向formData中插入文件
        if(file.files[count]){
        formData.append("file",file.files[count],file.files[count].name);
        let fileReader = new FileReader();
        //解析圖片路徑,實(shí)現(xiàn)預(yù)覽
        fileReader.readAsDataURL(file.files[count]);
        fileReader.onload=()=>{
          previewData = {
            url:fileReader.result,
            name:file.files[count].name,
            size:file.files[count].size,
          };
          _this.fileList.push(previewData);
          _this.progressShow = true
        };
        fileReader.onloadend=()=>{
          //檢測圖片大小是否超出限制
          if(formData.get("file").size>_this.maxSize){
            formData.delete("file");
            //當(dāng)圖片全部上傳完畢,停止遞歸
            count++;
            if(count > file.files.length-1){
              return
            }
            upload()
          }else{
              //發(fā)送數(shù)據(jù)
              axios.post(`/upload?mark=${_this.group}`,formData,config).then((response)=>{
                formData.delete("file");
                let res = response.data;
                console.log(res);
                if(res.result){
                  //如果是新建上傳
                  if(_this.group === "new"){
                    _this.fileList.push(res.data);
                      _this.fileList.forEach((item,index)=>{
                          if(!item.newName){
                            _this.fileList.splice(index,1)
                          }
                      })

                    }else{
                    //如果是選擇其他組上傳,直接把返回數(shù)據(jù)賦值到文件數(shù)組
                      _this.fileList = res.data;
                    }

                  _this.newUpload = false
                }else{
                  alert("上傳失敗");
                  return;
                }
                _this.noPic = false;
                count++;
                if(count > file.files.length-1){
                  return
                }
                upload()
              }).catch((err)=>{
                alert("上傳失敗123");
              });
            }
        };
        }
      }
      //第一次調(diào)用
      upload();
      document.forms[0].target="rfFrame";

}

node.js寫后端
//引入表單處理模塊
let Formidable = require("formidable");

一系列定義....

form.encoding = "utf-8";
form.uploadDir = "/project/vue/vue_uploader/my-server/public/images";//定義文件存放地址
form.keepExtensions = true;
form.multiples = false;//以單文件依次上傳的方式,實(shí)現(xiàn)多文件上傳
form.maxFieldsSize = 1*1024;
//解析圖片,重命名圖片名稱,返回給前端。
let fileData = "";
let fileDir = "images";//定義文件的存放路徑
let route = "upload_";//定義路由
let serverIp = "http://localhost:3002/";//定義服務(wù)器IP

對文件數(shù)據(jù)進(jìn)行處理,存入本地并存入數(shù)據(jù)庫(由于涉及到分組上傳。。。所以比較復(fù)雜)

解析文件函數(shù):

function handleFile (file){
    let filename = file.name;
    let nameArray = filename.split(".");
    let type = nameArray[nameArray.length-1];
    let name = "";
    for (let i = 0;i{
        if(err){
            res.json({
                result:false,
                msg:err.message
            })
        }else{
            if(doc){
                doc.picList.push(fileData);
                doc.save((err,saveResult)=>{

                    if(err){
                        return res.json({
                            result:false,
                        });
                    }else{
                        let length= doc.picList.length;
                        console.log(doc.picList.length)
                        if(groupMark === "all"){
                            UploadData.find({},(err,queryResult)=>{
                                if(err){
                                    res.json({
                                        result:false,
                                        mgs:"發(fā)生錯誤了"
                                    })
                                }else{
                                    let allPic = [];
                                    queryResult.forEach((item)=>{
                                        if(item.group !=="default"){
                                            allPic = allPic.concat(item.picList)
                                        }
                                    });
                                        res.json({
                                            result:true,
                                            data:allPic.concat(queryResult[1].picList)
                                        })

                                }
                            })
                        }else if(groupMark === "new"){

                            UploadData.findOne({group:"default"},(err,queryResult)=>{
                                if(err){
                                    return res.json({
                                        result:false,
                                        msg:err.message
                                    });
                                }else{
                                    return res.json({
                                        result:true,
                                        data:queryResult.picList[queryResult.picList.length-1]
                                    })
                                }
                            });

                        }else{
                            UploadData.findOne({group:group},(err,queryResult)=>{
                                if(err){
                                    return res.json({
                                        result:false,
                                        msg:err.message
                                    });
                                }else{
                                    return res.json({
                                        result:true,
                                        data:queryResult.picList
                                    })
                                }
                            });
                        }
                    }
                })

            }

        }

    })
}

最后,調(diào)用解析文件函數(shù)

form.parse(req,(err,fields,files)=>{
    //傳多個文件
    if(files.file instanceof Array){
        return
    }else{
     //傳單個文件
        handleFile(files.file)
    }
});  

數(shù)據(jù)庫結(jié)構(gòu):

剩下的還有文件刪除,新增分組,刪除分組,分組查詢的功能,由于篇幅有限,這些功能可以去看源碼

第一次用node和mongoDB寫后臺業(yè)務(wù),還有很多地方需要完善,代碼會繼續(xù)更新~

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/19150.html

相關(guān)文章

  • 一些基于React、VueNode.js、MongoDB技術(shù)棧實(shí)踐項(xiàng)目

    摘要:利用中間件實(shí)現(xiàn)異步請求,實(shí)現(xiàn)兩個用戶角色實(shí)時通信。目前還未深入了解的一些概念。往后會寫更多的前后臺聯(lián)通的項(xiàng)目。刪除分組會連同組內(nèi)的所有圖片一起刪除。算是對自己上次用寫后臺的一個強(qiáng)化,項(xiàng)目文章在這里。后來一直沒動,前些日子才把后續(xù)的完善。 歡迎訪問我的個人網(wǎng)站:http://www.neroht.com/? 剛學(xué)vue和react時,利用業(yè)余時間寫的關(guān)于這兩個框架的訓(xùn)練,都相對簡單,有的...

    tangr206 評論0 收藏0
  • 2017年3月份前端資源分享

    平日學(xué)習(xí)接觸過的網(wǎng)站積累,以每月的形式發(fā)布。2017年以前看這個網(wǎng)址:http://www.kancloud.cn/jsfron... 03月份前端資源分享 1. Javascript 175453545 Redux compose and middleware 源碼分析 深入 Promise(二)——進(jìn)擊的 Promise Effective JavaScript leeheys blog -...

    ermaoL 評論0 收藏0
  • 2017年3月份前端資源分享

    平日學(xué)習(xí)接觸過的網(wǎng)站積累,以每月的形式發(fā)布。2017年以前看這個網(wǎng)址:http://www.kancloud.cn/jsfron... 03月份前端資源分享 1. Javascript 175453545 Redux compose and middleware 源碼分析 深入 Promise(二)——進(jìn)擊的 Promise Effective JavaScript leeheys blog -...

    kamushin233 評論0 收藏0
  • 2017年3月份前端資源分享

    平日學(xué)習(xí)接觸過的網(wǎng)站積累,以每月的形式發(fā)布。2017年以前看這個網(wǎng)址:http://www.kancloud.cn/jsfron... 03月份前端資源分享 1. Javascript 175453545 Redux compose and middleware 源碼分析 深入 Promise(二)——進(jìn)擊的 Promise Effective JavaScript leeheys blog -...

    yy736044583 評論0 收藏0

發(fā)表評論

0條評論

hzx

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<