thinkphp6+elementui实现多图片压缩包下载

在日常开发中,有时会遇到需要批量下载多个用户的文件(如身份证正反面照片)的需求。为了提升用户体验,减少人工操作,我们可以通过前端和后端的配合来实现这一功能。本文将详细介绍如何通过 thinkPHP6 + Element ui 实现批量下载图片的功能。刚好最近客户想要实现批量下载多个用户的身份证正反面照片,于是网上找了资料整理了下。

前端

1.写个下载按钮

js 复制代码
<el-button class="filter-item" type="success" icon="el-icon-set-up" @click="downIdcard" size="small">下载身份证件</el-button>

2.下载事件函数

js 复制代码
 if (this.selectedCodes.length == 0) {
        this.$message.error("至少选择一条数据");
        return
      }
      this.$http.get('downloadIdcardImgs', { params: { code: this.selectedCodes }, responseType: 'blob' }).then(res => {
        console.log(res);

        // 创建一个URL对象
        const url = window.URL.createObjectURL(new Blob([res.data]));

        // 创建一个隐藏的a标签并模拟点击
        const link = document.createElement('a');
        link.href = url;
        link.setAttribute('download',  this.selectedCodes[0]+'-'+this.selectedCodes[this.selectedCodes.length-1]+ '.zip'); // 文件名自定义
        document.body.appendChild(link);
        link.click();

        // 清理
        document.body.removeChild(link);
        window.URL.revokeObjectURL(url);
      })

后端

1.定义好路由,保证前端能正常访问到后端接口

2.extend扩展类库目录中新建zipfile目录,里面新建zipfile.php文件,文件代码如下:

php 复制代码
<?php
declare(strict_types=1);

namespace zipfile;
class zipfile {
    var $datasec = array ();
    var $ctrl_dir = array ();
    var $eof_ctrl_dir = "\x50\x4b\x05\x06\x00\x00\x00\x00";
    var $old_offset = 0;
    
    function unix2_dostime($unixtime = 0){
        $timearray = ($unixtime == 0) ? getdate () : getdate($unixtime);        
        if ($timearray ['year'] < 1980){
            $timearray ['year'] = 1980;
            $timearray ['mon'] = 1;
            $timearray ['mday'] = 1;
            $timearray ['hours'] = 0;
            $timearray ['minutes'] = 0;
            $timearray ['seconds'] = 0;
        }
        return (($timearray ['year'] - 1980) << 25) | ($timearray ['mon'] << 21) | ($timearray ['mday'] << 16) | ($timearray ['hours'] << 11) | ($timearray ['minutes'] << 5) | ($timearray ['seconds'] >> 1);
    }
    function add_file($data, $name, $time = 0){
        $name = str_replace('\\', '/', $name);
        
        $dtime = dechex($this->unix2_dostime($time));
        $hexdtime = '\x' . $dtime [6] . $dtime [7] . '\x' . $dtime [4] . $dtime [5] . '\x' . $dtime [2] . $dtime [3] . '\x' . $dtime [0] . $dtime [1];
        eval('$hexdtime = "' . $hexdtime . '";');
        
        $fr = "\x50\x4b\x03\x04";
        $fr .= "\x14\x00";
        $fr .= "\x00\x00";
        $fr .= "\x08\x00";
        $fr .= $hexdtime;
        
        $unc_len = strlen($data);
        $crc = crc32($data);
        $zdata = gzcompress($data);
        $zdata = substr(substr($zdata, 0, strlen($zdata)- 4), 2);
        $c_len = strlen($zdata);
        $fr .= pack('V', $crc);
        $fr .= pack('V', $c_len);
        $fr .= pack('V', $unc_len);
        $fr .= pack('v', strlen($name));
        $fr .= pack('v', 0);
        $fr .= $name;
        
        $fr .= $zdata;
        $fr .= pack('V', $crc);
        $fr .= pack('V', $c_len);
        $fr .= pack('V', $unc_len);
        
        $this->datasec [] = $fr;
        
        $cdrec = "\x50\x4b\x01\x02";
        $cdrec .= "\x00\x00";
        $cdrec .= "\x14\x00";
        $cdrec .= "\x00\x00";
        $cdrec .= "\x08\x00";
        $cdrec .= $hexdtime;
        $cdrec .= pack('V', $crc);
        $cdrec .= pack('V', $c_len);
        $cdrec .= pack('V', $unc_len);
        $cdrec .= pack('v', strlen($name));
        $cdrec .= pack('v', 0);
        $cdrec .= pack('v', 0);
        $cdrec .= pack('v', 0);
        $cdrec .= pack('v', 0);
        $cdrec .= pack('V', 32);
        
        $cdrec .= pack('V', $this->old_offset);
        $this->old_offset += strlen($fr);
        
        $cdrec .= $name;
        
        $this->ctrl_dir[] = $cdrec;
    }
    function add_path($path, $l = 0){
        $d = @opendir($path);
        $l = $l > 0 ? $l : strlen($path) + 1;
        while($v = @readdir($d)){
            if($v == '.' || $v == '..'){
                continue;
            }
            $v = $path . '/' . $v;
            if(is_dir($v)){
                $this->add_path($v, $l);
            } else {
                $this->add_file(file_get_contents($v), substr($v, $l));
            }
        }
    }
    function file(){
        $data = implode('', $this->datasec);
        $ctrldir = implode('', $this->ctrl_dir);
        return $data . $ctrldir . $this->eof_ctrl_dir . pack('v', sizeof($this->ctrl_dir)) . pack('v', sizeof($this->ctrl_dir)) . pack('V', strlen($ctrldir)) . pack('V', strlen($data)) . "\x00\x00";
    }
    
    function add_files($files){
        foreach($files as $file){
            if (is_file($file)){
                $data = implode("", file($file));
                $this->add_file($data, $file);
            }
        }
    }
    function output($file){
        $fp = fopen($file, "w");
        fwrite($fp, $this->file ());
        fclose($fp);
    }
}

3.编写路由中定义的函数,实现图片压缩文件下载

php 复制代码
 public function downloadIdcardImgs()
    {

        $dfile =  tempnam('/tmp', 'tmp'); //产生一个临时文件,用于缓存下载文件
        $zip = new zipfile();
        //----------------------
        $filename = 'image.zip'; //下载的默认文件名
        //以下是需要下载的图片数组信息,将需要下载的图片信息转化为类似即可
        // 需要下载的图片数组信息
        $image = array(
            array('image_src' => 'https://xxx.com/images/20250206/2025020611551410003.jpg', 'image_name' => '1.jpg'),
            array('image_src' => 'https://xxx.com/images/20250206/2025020611550849112.jpg', 'image_name' => '2.jpg'),
        );

        foreach ($image as $v) {
            $zip->add_file(file_get_contents($v['image_src']),  $v['image_name']);
            // 添加打包的图片,第一个参数是图片内容,第二个参数是压缩包里面的显示的名称, 可包含路径
            // 或是想打包整个目录 用 $zip->add_path($image_path);
        }
        //----------------------
        $zip->output($dfile);

        // 下载文件
        ob_clean();
        header('Access-Control-Allow-Origin: *'); // 允许所有来源的跨域请求
        header('Access-Control-Allow-Methods: POST, GET, OPTIONS'); // 允许的方法
        header('Access-Control-Allow-Headers: Content-Type, Authorization'); // 允许的请求头
        header('Pragma: public');
        header('Last-Modified:' . gmdate('D, d M Y H:i:s') . 'GMT');
        header('Cache-Control:no-store, no-cache, must-revalidate');
        header('Cache-Control:pre-check=0, post-check=0, max-age=0');
        header('Content-Transfer-Encoding:binary');
        header('Content-Encoding:none');
        header('Content-type: application/zip');
        header('Content-Description: File Transfer');
        // header('Content-Encoding: gzip');
        header('Content-Disposition:attachment; filename="' . $filename . '"'); //设置下载的默认文件名
        // header('Content-length:' . filesize($dfile));
        header('transfer-encoding: chunked');
        $fp = fopen($dfile, 'r');
        while (connection_status() == 0 && $buf = @fread($fp, 8192)) {
            echo $buf;
        }
        fclose($fp);
        @unlink($dfile);
        @flush();
        @ob_flush();
        exit();
    }

4.这样点击下载按钮,就可以下载一个压缩文件,解压后就是所需要的图片了。

总结

通过上面的实现方案,我们能成功实现批量下载用户所需图片的功能。前端通过简单的按钮点击即可触发下载操作,后端则负责图片的获取、打包和返回。整个过程高效且易于维护,适用于多种类似的批量文件下载场景。

相关推荐
清岚_lxn4 小时前
原生SSE实现AI智能问答+Vue3前端打字机流效果
前端·javascript·人工智能·vue·ai问答
ZoeLandia4 小时前
Element UI 设置 el-table-column 宽度 width 为百分比无效
前端·ui·element-ui
橘子味的冰淇淋~5 小时前
解决 vite.config.ts 引入scss 预处理报错
前端·vue·scss
小小小小宇6 小时前
V8 引擎垃圾回收机制详解
前端
lauo7 小时前
智体知识库:ai-docs对分布式智体编程语言Poplang和javascript的语法的比较(知识库问答)
开发语言·前端·javascript·分布式·机器人·开源
拉不动的猪7 小时前
设计模式之------单例模式
前端·javascript·面试
一袋米扛几楼987 小时前
【React框架】什么是 Vite?如何使用vite自动生成react的目录?
前端·react.js·前端框架
Alt.97 小时前
SpringMVC基础二(RestFul、接收数据、视图跳转)
java·开发语言·前端·mvc
lorogy7 小时前
【VSCode配置】运行springboot项目和vue项目
vue.js·spring boot·vscode
进取星辰8 小时前
1、从零搭建魔法工坊:React 19 新手村生存指南
前端·react.js·前端框架