前端上传图片压缩预览上传进度例子

女生学前端好还是后台开发
web前端和后台开发
前端,后台开发必备技能

html 代码

    <head>
        <meta charset=”utf-8″ />
        <meta
            name=”viewport”
            content=”width=device-width,initial-scale=1, minimum-scale=1, maximum-scale=1, user-scalable=no”
        />
        <meta name=”format-detection” content=”telephone=no” />
        <title>前端上传图片压缩预览上传进度例子</title>
        <style>
            body {
                font-family: Microsoft Yahei, helvetica;
                position: relative;
            }
            body,
            button,
            dd,
            dl,
            dt,
            form,
            h1,
            h2,
            h3,
            h4,
            iframe,
            img,
            input,
            li,
            p,
            td,
            textarea,
            ul {
                margin: 0;
                padding: 0;
                border: 0;
                resize: none;
                outline: 0;
            }
            table {
                border-spacing: 0;
                border-collapse: collapse;
            }
            canvas,
            iframe {
                display: block;
            }
            img {
                border: none;
                vertical-align: middle;
            }
            li {
                list-style-type: none;
            }
            a:active,
            a:hover,
            a:link,
            a:visited {
                text-decoration: none;
                color: inherit;
            }
            /*常用类*/
            .hide {
                display: none !important;
            }
            .notEvent {
                pointer-events: none;
            }
            /*清除浮动*/
            .clearfix:after {
                content: ‘.’;
                display: block;
                height: 0;
                visibility: hidden;
                clear: both;
                overflow: hidden;
            }
            /*图片列表*/
            .imagelist {
            }
            .imagelist ul {
                padding: 0;
                margin: 0;
            }
            .imagelist li,
            .imagelist .newbtn {
                width: 80px;
                height: 80px;
                position: relative;
                float: left;
                margin: 10px 10px 0 0;
            }
            .imagelist li {
                background-color: #eee;
                background-position: 50% 50%;
                background-repeat: no-repeat;
                background-size: cover;
            }
            .imagelist li i {
                position: absolute;
                z-index: 2;
                top: -8px;
                right: -8px;
                cursor: pointer;
                width: 14px;
                height: 14px;
                line-height: 14px;
                font-size: 12px;
                padding: 4px;
                font-style: initial;
                text-align: center;
                color: #fff;
                border-radius: 50%;
                background: rgba(0, 0, 0, 0.6);
            }
            .imagelist li i:active {
                transform: scale3d(0.92, 0.92, 1);
                -webkit-transform: scale3d(0.92, 0.92, 1);
            }
            .imagelist li[data-load=’1′] {
                background: rgba(0, 0, 0, 0.4);
            }
            .imagelist li[data-load=’1′]:after {
                content: ”;
                position: absolute;
                left: 50%;
                top: 50%;
                margin: -15px 0 0 -15px;
                width: 30px;
                height: 30px;
                background: url()
                    no-repeat;
                background-size: 100%;
                animation: rotate_cw 2s linear infinite;
                -webkit-animation: rotate_cw 2s linear infinite;
            }
            .imagelist .newbtn {
                background: #eee;
                box-sizing: border-box;
                padding: 15px;
                border-radius: 2px;
            }
            .imagelist .newbtn:active {
                background: #ddd;
            }
            .imagelist .newbtn img {
                width: 50px;
                height: 50px;
            }
            .imagelist .newbtn input {
                position: absolute;
                left: 0;
                top: 0;
                width: 100%;
                height: 100%;
                opacity: 0;
            }
            /*form*/
            .formwrap {
                padding: 10px;
            }
            .formwrap button[type=’submit’] {
                display: block;
                margin: 30px 0;
                background: #1b82d2;
                font-size: 15px;
                color: #fff;
                line-height: 24px;
                width: 600px;
                max-width: 100%;
                padding: 10px 0;
                border-radius: 3px;
            }
            .formwrap button[type=’submit’]:active {
                background: #1275c1;
            }
            /*上传提示*/
            .uploadPrompt {
                position: fixed;
                width: 80px;
                padding: 10px;
                text-align: center;
                left: 50%;
                top: 50%;
                z-index: 500;
                background: rgba(0, 0, 0, 0.6);
                border-radius: 5px;
                transform: translate3d(-50%, -50%, 0);
                -webkit-transform: translate3d(-50%, -50%, 0);
            }
            .uploadPrompt .ico {
                width: 30px;
                height: 30px;
                margin-top: 5px;
                animation: rotate_cw 1.5s linear infinite;
                -webkit-animation: rotate_cw 1.5s linear infinite;
            }
            .uploadPrompt .text {
                margin: 0;
                color: #fff;
                line-height: 20px;
                font-size: 14px;
                margin-top: 8px;
            }
            @keyframes rotate_cw {
                0% {
                    transform: rotateZ(0);
                }
                100% {
                    transform: rotateZ(360deg);
                }
            }
            @-webkit-keyframes rotate_cw {
                0% {
                    -webkit-transform: rotateZ(0);
                }
                100% {
                    -webkit-transform: rotateZ(360deg);
                }
            }
        </style>
    </head>
    <body>
        <form
            method=”post”
            action=””
            enctype=”multipart/form-data”
            class=”formwrap”
            id=”FormWrap”
        >
            <div class=”imagelist clearfix” id=”UploadingImages”>
                <ul id=”UploadingContainer”></ul>
                <div class=”newbtn” id=”UploadingBtn”>
                    <img
                        src=””
                    />
                    <input
                        data-role=”none”
                        type=”file”
                        multiple=”multiple”
                        accept=”image/gif,image/jpeg,image/png”
                        id=”UploadControl”
                    />
                </div>
            </div>
            <button type=”submit”>提交</button>
        </form>
        <div class=”uploadPrompt hide” id=”UploadPrompt”>
            <img
                class=”ico”
                src=””
            />
            <p class=”text” id=”UploadPromptText”></p>
        </div>
        <script>
            /*
可以删掉用jquery代替
*/
            //判断是否具有指定样式类
            HTMLElement.prototype.hasClass = function(name) {
                var c = this.className.split(‘ ‘)
                for (var i = c.length – 1; i >= 0; i–) {
                    if (c[i].toLowerCase() == name.toLowerCase()) {
                        return true
                    }
                }
                return false
            }
            //添加样式类
            HTMLElement.prototype.addClass = function(name) {
                var list1 = name.split(‘ ‘)
                var list2 = this.className.split(‘ ‘)
                list1.forEach(function(item, i) {
                    var index = list2.indexOf(item)
                    if (index === -1) {
                        list2.push(item)
                    }
                })
                this.className = list2.join(‘ ‘)
                return this
            }
            //删除样式类
            HTMLElement.prototype.removeClass = function(name) {
                var list1 = name.split(‘ ‘)
                var list2 = this.className.split(‘ ‘)
                list1.forEach(function(item) {
                    var index = list2.indexOf(item)
                    if (index > -1) {
                        list2.splice(index, 1)
                    }
                })
                this.className = list2.join(‘ ‘)
                return this
            }
            //删除自己
            HTMLElement.prototype.remove = function() {
                this.parentNode.removeChild(this)
                return this
            }
            ;(function() {
                /*
解决安卓机上input file accept不为image/*时,不触发onchange事件
但是在PC上input file设置accept=”image/*”时,响应又非常慢
所以需要判断是移动端还是PC端
*/
                if (!IsPC()) {
                    ;[].forEach.call(
                        document.querySelectorAll(‘input[type=file]’),
                        function(item) {
                            item.setAttribute(‘accept’, ‘image/*’)
                        },
                    )
                }
                function IsPC() {
                    var userAgentInfo = navigator.userAgent
                    var Agents = [
                        ‘Android’,
                        ‘iPhone’,
                        ‘SymbianOS’,
                        ‘Windows Phone’,
                        ‘iPad’,
                        ‘iPod’,
                    ]
                    var flag = true
                    for (var v = 0; v < Agents.length; v++) {
                        if (userAgentInfo.indexOf(Agents[v]) > 0) {
                            flag = false
                            break
                        }
                    }
                    return flag
                }
                //压缩画布
                var CompressCanvas = document.createElement(‘canvas’)
                var CompressContext = CompressCanvas.getContext(‘2d’)
                //瓦片画布
                var TileCanvas = document.createElement(‘canvas’)
                var TileContext = TileCanvas.getContext(‘2d’)
                //文件最大大小
                var MaxSize = 200 * 1024
                //压缩率
                var CompressRatio = 0.5
                //图片列表
                var SelectImageList = {}
                //是否有图像正在压缩中
                var IsImageCompress = false
                //元素
                var $_FormWrap = document.querySelector(‘#FormWrap’)
                var $_UploadControl = document.querySelector(‘#UploadControl’)
                var $_UploadingImages = document.querySelector(
                    ‘#UploadingImages’,
                )
                var $_UploadingContainer = document.querySelector(
                    ‘#UploadingContainer’,
                )
                var $_UploadPrompt = document.querySelector(‘#UploadPrompt’)
                var $_UploadPromptText = document.querySelector(
                    ‘#UploadPromptText’,
                )
                //初始化
                //事件
                //预览图删除按钮事件
                $_UploadingContainer.addEventListener(‘click’, function() {
                    var s = this
                    var _this = isEventAgencyTarget.call(
                        $_UploadingContainer,
                        ‘i’,
                    )
                    if (_this) {
                        var _parent = _this.parentElement
                        var _id = _parent.getAttribute(‘data-id’)
                        SelectImageList[_id] && delete SelectImageList[_id]
                        _parent.parentElement.removeChild(_parent)
                    }
                })
                //判断给定选择器元素是否在事件冒泡路径中
                function isEventAgencyTarget(exp) {
                    var s = this
                    var target = null
                    var exp = arguments[0] || ”
                    var agencylist = [].slice.call(s.querySelectorAll(exp))
                    agencylist.forEach(function(item) {
                        if (window.event.path.indexOf(item) > -1) {
                            target = item
                        }
                    })
                    return target || false
                }
                //提交事件
                $_FormWrap.addEventListener(‘submit’, function(e) {
                    e.preventDefault()
                    if (IsImageCompress) {
                        alert(‘图像压缩中,请稍等片刻!’)
                        return false
                    }
                    //显示提示
                    document.body.addClass(‘notEvent’)
                    $_UploadPrompt.removeClass(‘hide’)
                    $_UploadPromptText.innerText = ‘提交中…’
                    //实例化
                    var formdata = getFormData()
                    //循环遍历表单元素并填入FormData
                    ;[].forEach.call($_FormWrap, function(item) {
                        var name = item.getAttribute(‘name’)
                        name && formdata.append(name, item.value)
                    })
                    //循环遍历待上传图片二进制数据并填入FormData
                    var list = Object.keys(SelectImageList)
                    list.forEach(function(item, i) {
                        formdata.append(
                            ‘upload_’ + i,
                            SelectImageList[item].data,
                            SelectImageList[item].name,
                        )
                    })
                    formdata.append(‘upload_length’, list.length)
                    //提交
                    var xhr = new XMLHttpRequest()
                    xhr.open(
                        $_FormWrap.getAttribute(‘method’),
                        $_FormWrap.getAttribute(‘action’),
                    )
                    xhr.onreadystatechange = function() {
                        if (xhr.readyState == 4 && xhr.status == 200) {
                            /*上传成功*/
                            location.reload()
                            alert(‘提交成功!’)
                            document.body.removeClass(‘notEvent’)
                            $_UploadPrompt.addClass(‘hide’)
                        }
                    }
                    xhr.onerror = function() {
                        alert(‘提交失败,请稍后再试!’)
                    }
                    //数据发送进度
                    xhr.upload.addEventListener(
                        ‘progress’,
                        function(e) {
                            pecent = ~~((100 * e.loaded) / e.total)
                            $_UploadPromptText.innerText =
                                ‘上传进度(’ + pecent + ‘%)’
                        },
                        false,
                    )
                    xhr.send(formdata)
                    return false
                })
                //选择文件事件
                $_UploadControl.addEventListener(‘change’, function() {
                    if (!this.files.length) return
                    var index = 0
                    var Fragment = document.createDocumentFragment()
                    //文件列表
                    var files = Array.prototype.slice.call(this.files)
                    //标记图像压缩中
                    IsImageCompress = true
                    //遍历文件列表
                    files.forEach(function(file, i) {
                        if (!/(jpeg|png|gif)/i.test(file.type)) return
                        var img = new Image()
                        //创建dom
                        var li = document.createElement(‘li’)
                        li.innerHTML = ‘<i>X</i>’
                        li.setAttribute(‘data-load’, 1)
                        Fragment.appendChild(li)
                        //初始化
                        var reader = new FileReader()
                        reader.onload = function() {
                            var result = this.result
                            img.src = result
                            //如果图片大小小于200kb,则不进行处理
                            if (result.length <= MaxSize) {
                                li.setAttribute(‘data-load’, 0)
                                li.setAttribute(‘data-id’, Date.now())
                                li.style.backgroundImage =
                                    ‘url(“‘ + result + ‘”)’
                                SelectImageList[li.getAttribute(‘data-id’)] = {
                                    name: file.name,
                                    data: data64toBinary(result, file.type),
                                }
                                if (++index === files.length) {
                                    IsImageCompress = false
                                }
                            } else {
                                //图片加载完毕之后进行压缩
                                img.onload = function() {
                                    var data = ImageCompress(this)
                                    li.setAttribute(‘data-load’, 0)
                                    li.setAttribute(‘data-id’, Date.now())
                                    li.style.backgroundImage =
                                        ‘url(“‘ + data + ‘”)’
                                    SelectImageList[
                                        li.getAttribute(‘data-id’)
                                    ] = {
                                        name: file.name,
                                        data: data64toBinary(data, file.type),
                                    }
                                    if (++index === files.length) {
                                        IsImageCompress = false
                                    }
                                }
                            }
                        }
                        reader.readAsDataURL(file)
                    })
                    $_UploadingContainer.appendChild(Fragment)
                    $_UploadControl.value = ”
                })
                //使用canvas对大图片进行压缩
                function ImageCompress(img) {
                    var width = img.width
                    var height = img.height
                    var initSize = img.src.length
                    //如果图片大于四百万像素,计算压缩比并将大小压至400万以下
                    var ratio = (width * height) / 4000000
                    if (ratio > 1) {
                        ratio = Math.sqrt(ratio)
                        width /= ratio
                        height /= ratio
                    } else {
                        ratio = 1
                    }
                    CompressCanvas.width = width
                    CompressCanvas.height = height
                    //绘制底色
                    CompressContext.fillStyle = ‘#fff’
                    CompressContext.fillRect(0, 0, width, height)
                    //如果图片像素大于100万则使用瓦片绘制
                    var count = (width * height) / 1000000
                    if (count > 1) {
                        //计算要分成多少块瓦片
                        count = ~~(Math.sqrt(count) + 1)
                        //计算每块瓦片的宽和高
                        var nw = ~~(width / count)
                        var nh = ~~(height / count)
                        TileCanvas.width = nw
                        TileCanvas.height = nh
                        for (var i = 0; i < count; i++) {
                            for (var j = 0; j < count; j++) {
                                TileContext.drawImage(
                                    img,
                                    i * nw * ratio,
                                    j * nh * ratio,
                                    nw * ratio,
                                    nh * ratio,
                                    0,
                                    0,
                                    nw,
                                    nh,
                                )
                                CompressContext.drawImage(
                                    TileCanvas,
                                    i * nw,
                                    j * nh,
                                    nw,
                                    nh,
                                )
                            }
                        }
                    } else {
                        CompressContext.drawImage(img, 0, 0, width, height)
                    }
                    //进行最小压缩
                    var ndata = CompressCanvas.toDataURL(
                        ‘image/jpeg’,
                        CompressRatio,
                    )
                    CompressCanvas.width = CompressCanvas.height = TileCanvas.width = TileCanvas.height = 0
                    return ndata
                }
                //data64转二进制
                function data64toBinary(basestr, type) {
                    var text = window.atob(basestr.split(‘,’)[1])
                    var buffer = new Uint8Array(text.length)
                    var pecent = 0,
                        loop = null
                    for (var i = 0; i < text.length; i++) {
                        buffer[i] = text.charCodeAt(i)
                    }
                    return getBlob([buffer], type)
                }
                /**
                 * 获取blob对象的兼容性写法
                 * @param buffer
                 * @param format
                 * @returns {*}
                 */
                function getBlob(buffer, format) {
                    try {
                        return new Blob(buffer, { type: format })
                    } catch (e) {
                        var bb = new (window.BlobBuilder ||
                            window.WebKitBlobBuilder ||
                            window.MSBlobBuilder)()
                        buffer.forEach(function(buf) {
                            bb.append(buf)
                        })
                        return bb.getBlob(format)
                    }
                }
                /**
                 * 获取formdata
                 */
                function getFormData() {
                    var isNeedShim =
                        ~navigator.userAgent.indexOf(‘Android’) &&
                        ~navigator.vendor.indexOf(‘Google’) &&
                        !~navigator.userAgent.indexOf(‘Chrome’) &&
                        navigator.userAgent.match(/AppleWebKit\/(\d+)/).pop() <=
                            534
                    return isNeedShim ? new FormDataShim() : new FormData()
                }
                /**
                 * formdata 补丁, 给不支持formdata上传blob的android机打补丁
                 * @constructor
                 */
                function FormDataShim() {
                    console.warn(‘using formdata shim’)
                    var o = this,
                        parts = [],
                        boundary =
                            Array(21).join(‘-‘) +
                            (+new Date() * (1e16 * Math.random())).toString(36),
                        oldSend = XMLHttpRequest.prototype.send
                    this.append = function(name, value, filename) {
                        parts.push(
                            ‘–‘ +
                                boundary +
                                ‘\r\nContent-Disposition: form-data; name=”‘ +
                                name +
                                ‘”‘,
                        )
                        if (value instanceof Blob) {
                            parts.push(
                                ‘; filename=”‘ +
                                    (filename || ‘blob’) +
                                    ‘”\r\nContent-Type: ‘ +
                                    value.type +
                                    ‘\r\n\r\n’,
                            )
                            parts.push(value)
                        } else {
                            parts.push(‘\r\n\r\n’ + value)
                        }
                        parts.push(‘\r\n’)
                    }
                    // Override XHR send()
                    XMLHttpRequest.prototype.send = function(val) {
                        var fr,
                            data,
                            oXHR = this
                        if (val === o) {
                            // Append the final boundary string
                            parts.push(‘–‘ + boundary + ‘–\r\n’)
                            // Create the blob
                            data = getBlob(parts)
                            // Set up and read the blob into an array to be sent
                            fr = new FileReader()
                            fr.onload = function() {
                                oldSend.call(oXHR, fr.result)
                            }
                            fr.onerror = function(err) {
                                throw err
                            }
                            fr.readAsArrayBuffer(data)
                            // Set the multipart content type and boudary
                            this.setRequestHeader(
                                ‘Content-Type’,
                                ‘multipart/form-data; boundary=’ + boundary,
                            )
                            XMLHttpRequest.prototype.send = oldSend
                        } else {
                            oldSend.call(this, val)
                        }
                    }
                }
            })()
        </script>
    </body>
</html>
后台开发好还是前端开发好
前端后台开发
后台开发 前端开发
赞(0)
前端开发者 » 前端上传图片压缩预览上传进度例子
64K

评论 抢沙发

评论前必须登录!