国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 編程 > JavaScript > 正文

JS實(shí)現(xiàn)表單多文件上傳樣式美化支持選中文件后刪除相關(guān)項(xiàng)

2019-11-20 08:49:58
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

開(kāi)發(fā)中會(huì)經(jīng)常涉及到文件上傳的需求,根據(jù)業(yè)務(wù)不同的需求,有不同的文件上傳情況。

有簡(jiǎn)單的單文件上傳,有多文件上傳,因?yàn)g覽器原生的文件上傳樣式及功能的支持度不算太高,很多時(shí)候我們會(huì)對(duì)樣式進(jìn)行美化,對(duì)功能進(jìn)行完善。

本文根據(jù)一個(gè)例子,對(duì)多文件的上傳樣式做了一些簡(jiǎn)單的美化(其實(shí)也沒(méi)怎么美化。。),同時(shí)支持選擇文件后自定義刪除相關(guān)的文件,最后再上傳

文章篇幅較長(zhǎng),先簡(jiǎn)單看看圖示:

一、文件上傳基礎(chǔ)

1. 單文件上傳

最簡(jiǎn)單的文件上傳,是單文件上傳,form標(biāo)簽中加入enctype="multipart/form-data",form表單中有一個(gè)input[type="file"]項(xiàng)

<form name="form1" method="post" action="/abc.php" enctype="multipart/form-data"><input type="text" name="user" id="user" placeholder="請(qǐng)輸入昵稱"><input type="file" name="userImage" id="userImage"><input type="submit" name="sub" value="提交"></form>

2. 多文件上傳

  1)類似單文件上傳,簡(jiǎn)單的多文件上傳其實(shí)就是多幾個(gè)input[type="file"]項(xiàng)

<form name="form1" method="post" action="/abc.php" enctype="multipart/form-data"><input type="text" name="user" id="user" placeholder="請(qǐng)輸入昵稱"><input type="file" name="userImage1" id="userImage1"><input type="file" name="userImage2" id="userImage2"><input type="file" name="userImage3" id="userImage3"><input type="submit" name="sub" value="提交"></form>

  2) HTML5為表單文件項(xiàng)新增了一個(gè)multiple屬性,可以設(shè)置實(shí)現(xiàn)選擇多個(gè)文件,如

<form name="form1" method="post" action="/abc.php" enctype="multipart/form-data">
<input type="text" name="user" id="user" placeholder="請(qǐng)輸入昵稱">
<input type="file" name="userImage" id="userImage" multiple>
<input type="submit" name="sub" value="提交">
</form>

二、表單文件上傳的美化

看了上面幾個(gè)圖片,可以知道原生的文件選擇項(xiàng)樣式是最基本的,主要體現(xiàn)在三個(gè)點(diǎn):

無(wú)邊框,與其他有邊框的元素不合拍
選擇文件的按鈕樣式太基礎(chǔ)
選擇多個(gè)文件后只顯示總數(shù),未顯示詳細(xì)選擇的文件名
基于幾個(gè)問(wèn)題,可以按需對(duì)其進(jìn)行美化

第一點(diǎn)可以直接添加邊框的樣式

第二點(diǎn)需要增添其他元素,可以新增一個(gè)按鈕(自行按需美化),將原始文件框隱藏,用JS事件綁定,點(diǎn)擊按鈕后模擬文件框的點(diǎn)擊

<input type="file" name="userImage" id="userImage" style="display: none;"><input type="button" id="" value="選擇文件" onclick="document.getElementById('userImage').click()">

第三點(diǎn)與第二點(diǎn)類似,也得添加新的元素,選擇文件后,通過(guò)JS獲取選擇的文件信息,并在新的元素中顯示出來(lái)

想著很簡(jiǎn)單,但隨之而來(lái)的問(wèn)題就是,如果選中的文件數(shù)量很多,新元素占空間的多少就是個(gè)問(wèn)題,可以默認(rèn)顯示幾個(gè)文件,再通過(guò)“查看更多文件”查看到更多的信息

隨之另外的想法是,一次性選中的文件很多,想取消某個(gè)文件時(shí),又得重新選擇。這未免太繁瑣,所以需要提供即時(shí)刪除某個(gè)選中文件的操作

三、選中文件后的刪除

要提供選中文件后可刪除的操作,就必然需要提供相關(guān)入口及腳本操作,下面圍繞這點(diǎn)來(lái)做些解析

1. 界面的處理

選擇文件后,我們可以通過(guò)刪除按鈕刪除選中的文件,因?yàn)闀?huì)出現(xiàn)多文件的情況,所以需要一個(gè)信息模版

<!-- 當(dāng)前選擇的文件列表 文件信息模版 --><script type="text/template" id="file-temp-item-tpl"><span class="file-temp-item" style="{{style}}"><span class="file-temp-name">{{name}}</span><span class="file-temp-btn">×</span></span></script>

選中的文件一多,就得再增添一個(gè)下拉框做輔助,最多顯示5個(gè)文件信息,然后通過(guò)下拉按鈕展開(kāi)下拉框(按鈕樣式自行設(shè)定)

這里5個(gè)文件間的位置計(jì)算的不是很到位,主要是這段代碼,可以自行設(shè)定

// 計(jì)算每一項(xiàng)坐標(biāo)left、占寬widthleft = i === 0 ? 2 : 2 + i * (100 / fileTempLen);width = 100 / fileTempLen - 2;

下拉列表里面的每一項(xiàng)也是一個(gè)模版

  <!-- 查看更多文件 文件信息模版 --><script type="text/template" id="file-more-item-tpl"><li><span class="file-item-more-name">{{name}}</span><span class="file-item-more-btn">×</span></li></script>

以下為初始的HTML結(jié)構(gòu)

 <form name="form" id="form" method="post" action="fileTest.php" enctype="multipart/form-data"><!-- <input type="number" name="numberTest" value="100"> --><input type="file" name="fileTest[]" id="fileTest" multiple><!-- 當(dāng)前選擇的文件列表(最多顯示5條) --><span class="file-temp"></span><!-- 查看更多文件 --><ul class="item-more"></ul><input type="button" class="btn btn-success" id="uploadBtn" value="上傳"><p class="upload-tip">文件上傳成功</p></form>

以下為全部CSS樣式

<link rel="stylesheet" type="text/css" href="bootstrap.min.css">  <style type="text/css">    html {      font-family: Arial;    }    form {      margin: 50px auto;      width: 400px;    }    input {      width: 300px;      padding: 4px;    }    #uploadBtn {      margin-top: -3px;      margin-left: 5px;      width: 60px;      height: 30px;      font-weight: bold;      font-size: 12px;    }    #fileTest {      display: inline-block;      border: 1px solid #ccc;      border-radius: 3px;    }    .file-temp {      position: relative;      display: none;      width: 300px;      height: 31px;    }    .file-temp-item {      position: absolute;      top: 4px;      height: 24px;    }    .item-more-btn {      display: inline-block;      position: absolute;      top: 18px;      right: 0.5%;      width: 10px;      height: 10px;      color: #777;      cursor: pointer;    }    .item-more-btn:hover {      border-top-color: #aaa;    }    .file-temp-name {      display: inline-block;      overflow: hidden;      width: 90%;      height: 26px;      padding: 2px 15px 2px 5px;      border-radius: 2px;      background-color: #eaeaf3;      text-overflow: ellipsis;      white-space: nowrap;    }    .file-temp-btn {      position: absolute;      display: inline-block;      top: 4px;      right: 11%;      width: 18px;      height: 18px;      line-height: 18px;      text-align: center;      border: 1px solid #ddd;      background-color: #ccc;      border-radius: 50%;      color: #fff;      font-size: 18px;      cursor: pointer;    }    .item-more {      position: absolute;      overflow-y: auto;      display: none;      padding-left: 0;      width: 300px;      max-height: 150px;      list-style: none;    }    .item-more li {      position: relative;      padding: 5px;      border: 1px solid #ccc;      border-top: none;    }    .item-more li:hover {      background-color: #f5f5f9;    }    .file-item-more-name {      display: inline-block;      width: 90%;      overflow: hidden;      text-overflow: ellipsis;      white-space: nowrap;    }    .file-item-more-btn {      position: absolute;      display: inline-block;      top: 8px;      right: 2%;      width: 18px;      height: 18px;      line-height: 18px;      text-align: center;      border: 1px solid #ddd;      background-color: #ddd;      border-radius: 50%;      color: #fff;      font-size: 18px;      cursor: pointer;    }    .file-item-more-btn:hover {      background-color: #ccc;    }    .upload-tip {      display: none;      margin: 50px auto;      text-align: center;      font-size: 12px;    }  </style>

2. 腳本的處理

下面,著重介紹JS腳本的處理

要獲取到選中文件的信息,自然想到用value屬性,但通過(guò)文件項(xiàng)的value只能獲取到一個(gè)文件路徑(第一個(gè)),無(wú)論有沒(méi)有multiple

無(wú)multiple

<input type="file" onchange="console.log(this.value);">

有multiple

<input type="file" multiple onchange="console.log(this.value);">

既然直接通過(guò)value獲取不到所有選中的文件信息,只能尋求其他途徑。

  1)FileList

獲取選中的文件信息,還可以用FileList對(duì)象,這是在HTML5中新增的,每個(gè)表單文件項(xiàng)都有個(gè)files屬性,里邊存儲(chǔ)這選中的文件的一些信息

<input type="file" multiple onchange="console.log(this.files);">

選中兩個(gè)文件后,查看文件信息

FileList對(duì)象看起來(lái)是個(gè)類數(shù)組,有l(wèi)ength屬性。所以我們應(yīng)該可以通過(guò)修改或刪除相關(guān)的項(xiàng)來(lái)自定義我們選擇的文件(注意其實(shí)這是不能修改的,且繼續(xù)看下去)

假如我選擇了兩個(gè)文件,想刪除第二項(xiàng)目,使用splice刪除,則

<input type="file" multiple onchange="console.log(Array.prototype.splice.call(this.files, 1, 1));">

報(bào)錯(cuò),由此可知FileList的length屬性是只讀的,那直接修改為可寫可配置呢

Object.defineProperty(FileList.prototype, 'length', {writable: true,configurable: true});

配置之后length能修改了,乍一看還以為splice生效了,然而輸出一看,F(xiàn)ileList對(duì)象內(nèi)容不變,仍為兩項(xiàng)

查閱了一些資料后,了解到瀏覽器為了安全性的考慮,把FileList對(duì)象的內(nèi)容設(shè)為了不可更改,只可以手動(dòng)置空,但不能修改內(nèi)容

所以,解決辦法是,新增一個(gè)數(shù)組,初始復(fù)制FileList對(duì)象的文件內(nèi)容,之后的修改操作則通過(guò)這個(gè)可更改的數(shù)組進(jìn)行

// 存儲(chǔ)更新所選文件var curFiles = []; ...// 選中文件后var files = this.files;if (files && files.length) {// 原始FileList對(duì)象不可更改,所以將其賦予curFiles提供接下來(lái)的修改Array.prototype.push.apply(curFiles, files);}

假如點(diǎn)擊了刪除叉叉,可以直接更新文件信息數(shù)組

var name = $(this).prev().text();// 去除該文件curFiles = curFiles.filter(function(file) {return file.name !== name;});

這樣一來(lái),更新文件信息的問(wèn)題得到解決,然后就可以進(jìn)行文件的上傳了

點(diǎn)擊文件上傳,如果直接調(diào)用$form.submit(); 則上傳的文件信息依然是初始的FileList對(duì)象,達(dá)不到我們自定義的要求,所以需要用Ajax提交

那么,該怎么想后臺(tái)提供一個(gè)文件對(duì)象呢?

  2)FormData

HTML5引入了表單的新對(duì)象FormData, 它可以生成一個(gè)表單對(duì)象,我們可以向其中獲取/設(shè)置鍵值對(duì)信息,再一并提交給后臺(tái)

引用MDN的FormData使用方法,我們可以添加各種類型的數(shù)據(jù),使用ajax提交

var oMyForm = new FormData();oMyForm.append("username", "Groucho");oMyForm.append("accountnum", 123456); // 數(shù)字123456被立即轉(zhuǎn)換成字符串"123456"http:// fileInputElement中已經(jīng)包含了用戶所選擇的文件oMyForm.append("userfile", fileInputElement.files[0]);var oFileBody = '<a id="a"><b id="b">hey!</b></a>'; // Blob對(duì)象包含的文件內(nèi)容var oBlob = new Blob([oFileBody], { type: "text/xml"});oMyForm.append("webmasterfile", oBlob);var oReq = new XMLHttpRequest();oReq.open("POST", "http://foo.com/submitform.php");oReq.send(oMyForm);

也可使用JQ的封裝的ajax,不過(guò)要注意設(shè)置processData和contentType屬性為false,防止JQ胡亂解析文件格式

var fd = new FormData(document.getElementById("fileinfo")); // 使用某個(gè)表單作為初始項(xiàng)fd.append("CustomField", "This is some extra data");$.ajax({url: "stash.php",type: "POST",data: fd,processData: false, // 告訴jQuery不要去處理發(fā)送的數(shù)據(jù)contentType: false // 告訴jQuery不要去設(shè)置Content-Type請(qǐng)求頭});

這里有幾個(gè)要注意的點(diǎn):

1)FormData中的屬性值接受的是單個(gè)文件信息,不能是復(fù)合性的對(duì)象。可能表意不明,且看

var fd = new FormData($('#form')[0]);fd.append('myFileTest', curFiles);$files = $_REQUEST['myFileTest'];var_dump($files);

用PHP接收傳過(guò)來(lái)的數(shù)據(jù),數(shù)據(jù)卻被直接轉(zhuǎn)換成字符串了,非文件對(duì)象

curFiles是文件對(duì)象,那PHP端是不是應(yīng)該用$_FILES來(lái)接收信息呢,試試換成$files = $_FILES['myFileTest'];

直接出問(wèn)題了,說(shuō)明不能這樣處理,需要將curFiles內(nèi)容一項(xiàng)一項(xiàng)拆開(kāi),即單個(gè)文件信息

var fd = new FormData($('#form')[0]);for (var i = 0, j = curFiles.length; i < j; ++i) {fd.append('myFileTest[]', curFiles[i]);}$files = $_FILES['myFileTest'];var_dump($files);

文件接收成功,接下來(lái)就可以按需進(jìn)行文件的操作了

2)后端獲取文件信息的時(shí)候,是直接通過(guò)原始$_FILES獲取的,其他一般的信息才用$_REQUEST獲取

換成$files = $_REQUEST['myFileTest'];試試,直接就是出現(xiàn)找不到myFileTest的問(wèn)題

試試添加一般的文件再提交

var fd = new FormData($('#form')[0]);for (var i = 0, j = curFiles.length; i < j; ++i) {fd.append('myFileTest[]', curFiles[i]);}fd.append('myTest', [1, 2, 3]);$files = $_FILES['myFileTest'];$test = $_REQUEST['myTest'];var_dump($test);var_dump($files);

3)如果需要multiple的多文件上傳,則需要在文件項(xiàng)的文件后添加[]號(hào),表示這是一個(gè)多文件的數(shù)組,以供后端處理解析

fd.append('myFileTest[]', curFiles[i]);

如果沒(méi)有后面的[],則連續(xù)的append會(huì)直接覆蓋原來(lái)的,最后后端獲取到的只是最后append進(jìn)去的項(xiàng)

4)不要直接在JQ的ajax中實(shí)例化出一個(gè)FormData對(duì)象,會(huì)出問(wèn)題

直接在data屬性中生成FormData對(duì)象,會(huì)被JQ忽略,所以后端什么信息也拿不到

混合表單項(xiàng)簡(jiǎn)單的例子:

在表單處理中,很多時(shí)候我們會(huì)進(jìn)行文件上傳和其他基礎(chǔ)項(xiàng)的提交,簡(jiǎn)單地多加一個(gè)input項(xiàng)目,看看是否處理成功

<input type="number" name="numberTest" value="100">

<?php$files = $_FILES['myFileTest'];$test = $_REQUEST['numberTest'];echo json_encode(array('len' => count($files['name']),'num' => $test));?>

以下為全部的JS腳本:

<script type="text/javascript">/*** 向文件列表元素中添加相應(yīng)的文件項(xiàng)* @param {Array} files 當(dāng)前的文件列表數(shù)組對(duì)象*/function addItem(files) {var fileTempItemTpl = $('#file-temp-item-tpl').html(),fileMoreItemTpl = $('#file-more-item-tpl').html()htmlTemp = [],htmlMoreTemp = [],// 文件列表中各文件坐標(biāo)位置及所占空間left = 2,width = 100,// 最多取前5個(gè)文件fileTempLen = files.length > 5 ? 5 : files.length;for (var i = 0, j = files.length; i < j; ++i) {// 當(dāng)i > 4,即第6個(gè)文件開(kāi)始if (i > 4) {htmlMoreTemp.push(fileMoreItemTpl.replace('{{name}}', files[i].name));continue;}// 計(jì)算每一項(xiàng)坐標(biāo)left、占寬widthleft = i === 0 ? 2 : 2 + i * (100 / fileTempLen);width = 100 / fileTempLen - 2;htmlTemp.push(fileTempItemTpl.replace('{{style}}', 'left: ' + left + '%;width: ' + width + '%;').replace('{{name}}', files[i].name));}// 渲染相關(guān)元素內(nèi)容$('.file-temp').html(''+ '<input type="text" style="background-color:#fff;" class="form-control" id="fileTemp" readonly>'+ htmlTemp.join('')+ (files.length > 5? '<span class="item-more-btn" title="查看更多">=</span>': ''));$('.item-more').html(htmlMoreTemp.join(''));}// 保存當(dāng)前選擇的(更新后)文件列表var curFiles = [];// 初始選擇文件時(shí)觸發(fā)$('#fileTest').change(function() {var $this = $(this),$temp = $('.file-temp'),files = this.files;if (files && files.length) {// 原始FileList對(duì)象不可更改,所以將其賦予curFiles提供接下來(lái)的修改Array.prototype.push.apply(curFiles, files);addItem(curFiles);$this.hide();$temp.css('display', 'inline-block');}});$(document)// 取消選擇某個(gè)文件時(shí),在文件列表數(shù)組對(duì)象中刪除這個(gè)值,并更新列表.on('click', '.file-temp-btn, .file-item-more-btn', function() {$('.upload-tip').hide();var name = $(this).prev().text();// 去除該文件curFiles = curFiles.filter(function(file) {return file.name !== name;});// 文件列表數(shù)組對(duì)象長(zhǎng)度大于5才顯示“更多文件列表”下拉項(xiàng)if (curFiles.length <= 5) {$('.item-more').hide();}// 文件列表數(shù)組被清空則重置文件選擇表單項(xiàng)if (!curFiles.length) {$('#fileTest').val('').show();$('.file-temp').css('display', 'none');} else {addItem(curFiles);}console.log(curFiles)})// 顯示“更多文件列表”下拉項(xiàng).on('click', '.item-more-btn', function() {$('.upload-tip').hide();$('.item-more').show('normal');});// 上傳操作$('#uploadBtn').click(function() {$('.upload-tip').hide();// 構(gòu)建FormData對(duì)象var fd = new FormData($('#form')[0]);for (var i = 0, j = curFiles.length; i < j; ++i) {fd.append('myFileTest[]', curFiles[i]);}$.ajax({url: 'fileTest.php',type: 'post',data: fd,processData: false,contentType: false,success: function(rs) {rs = JSON.parse(rs);$('.upload-tip').addClass('text-success').removeClass('text-error').text(rs.len + '個(gè)文件上傳成功, number項(xiàng)值為' + rs.num).show();},error: function(err) {}});});</script>

以上所述是小編給大家介紹的JS實(shí)現(xiàn)表單多文件上傳樣式美化支持選中文件后刪除相關(guān)項(xiàng),希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)武林網(wǎng)網(wǎng)站的支持!

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 木兰县| 宁河县| 岑溪市| 利津县| 南江县| 攀枝花市| 新兴县| 醴陵市| 汽车| 宁阳县| 大理市| 丰原市| 长沙县| 安义县| 乐安县| 隆回县| 朝阳县| 万全县| 静宁县| 陆川县| 焦作市| 长岭县| 洛宁县| 罗源县| 常熟市| 渑池县| 京山县| 龙南县| 华坪县| 萝北县| 甘孜县| 乳源| 大石桥市| 绩溪县| 灯塔市| 贡山| 安远县| 基隆市| 潮安县| 手机| 唐海县|