← 返回首页
Javascript基础教程(三十四)
发表时间:2020-04-01 16:07:55
讲解Javascript文件上传

在HTML表单中,可以上传文件的唯一控件就是<input type="file">。

注意:当一个表单包含<input type="file">时,表单的enctype必须指定为multipart/form-data,method必须指定为post,浏览器才能正确编码并以multipart/form-data格式发送表单的数据。

出于安全考虑,浏览器只允许用户点击<input type="file">来选择本地文件,用JavaScript对<input type="file">的value赋值是没有任何效果的。当用户选择了上传某个文件后,JavaScript也无法获得该文件的真实路径:

1.检查上传文件类型

通常,文件上传都是调用后台服务器程序去处理,JavaScript可以在提交表单时对文件扩展名做检查,以便防止用户上传无效格式的文件。

例如:

<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="UTF-8">
    <title>用户注册</title>
</head>
<body>
<h1>用户注册</h1>
<hr>
<div>
    <form name="regForm" action="regok.html" method="post" enctype="multipart/form-data">
        <table>
            <tr>
                <td class="title">上传照片:</td>
                <td><input type="file" id="pic" name="pic"/></td>
            </tr>
            <tr>
                <td colspan="2" align="center"><button type="button" onclick="checkFileType()">检查文件类型</button></td>
            </tr>
        </table>
    </form>
</div>

<script>
    function checkFileType(){
        let f = document.getElementById('pic');
        console.log(f.value);
        let filename = f.value;
        if (!filename || !(filename.endsWith('.jpg') || filename.endsWith('.jpg') ||filename.endsWith('.webp') || filename.endsWith('.png') || filename.endsWith('.gif'))) {
            alert('只能上传图片类型!');
            return false;
        }
        console.log('执行上传程序....');
    }
</script>
</body>
</html>

2.FileAPI

在HTML5的FileReader出现之前,前端的图片预览是这样实现的:把本地图片上传到服务器,服务器把图片地址返回,并把它替换到图片元素的src属性。

这种方法的缺点是:必须要先把图片上传到服务器。那么问题来了,如果上传的图片很大,而网速又很慢,这就需要等待很久预览图片才会显示出来了,而且,如果用户预览图片后发现不太满意,想重新选择一张图片,这时候还要把已经上传到服务器上的图片给删除掉。

HTML5新增FileReader对象以后,预览图片变得简单多了,不再需要后台的配合,并且JS操作本地文件已经成为了可能。下面的例子演示了如何读取用户选取的图片文件,并在一个div中预览图像。

代码如下:

<!doctype html>
<html lang="zh">
<head>
    <meta charset="UTF-8">
    <title>上传文件预览</title>
    <style>
        #preview {
            width: 100%;
            height: 280px;
            background: no-repeat;
        }
    </style>
</head>
<body>
<div id="preview">
</div>
<div id="info">
</div>
<form>
    <input type="file" id="pic"/>
</form>
<script>

    let fileInput = document.getElementById('pic');
    let info = document.getElementById('info');
    let preview = document.getElementById('preview');
    // 监听change事件:
    fileInput.addEventListener('change', function () {
        // 清除背景图片:
        preview.style.backgroundImage = '';
        // 检查文件是否选择:
        if (!fileInput.value) {
            info.innerHTML = '没有选择文件';
            return;
        }
        // 获取File引用:
        var file = fileInput.files[0];
        // 获取File信息:
        info.innerHTML = '文件: ' + file.name + '<br>' +
            '大小: ' + file.size + '<br>' +
            '修改: ' + file.lastModifiedDate;
        if (file.type !== 'image/jpeg' && file.type !== 'image/png' && file.type !== 'image/gif') {
            alert('不是有效的图片文件!');
            return;
        }
        // 读取文件:
        var reader = new FileReader();

        reader.addEventListener("load", function () {
            var data = reader.result;
            console.log("data=" + data); //返回base64编码的图片字符串。
            preview.style.backgroundImage = 'url(' + data + ')';

        }, false);
        // 以DataURL的形式读取文件:
        reader.readAsDataURL(file);
    });
</script>
</body>
</html>

运行效果: