使用PhpOffice\PhpSpreadsheet 实现Excel多sheet导出 (2023-08-05)


use PhpOffice\PhpSpreadsheet\IOFactory;
use PhpOffice\PhpSpreadsheet\Spreadsheet;
use PhpOffice\PhpSpreadsheet\Writer\Xlsx;

class ExportService
{
   /**
     * 多sheet的导出
     * @author chaihao <158533752@qq.com>
     * @param [array] $data_array
     */
    public  function xtExport(array $dataArray): string
    {
        $spreadsheet = new Spreadsheet();
        foreach ($dataArray as $key => $data) {
            $this->opSheet($spreadsheet, $key, $data);
        }
        header('Content-Type: application/vnd.ms-excel');
        header('Content-Disposition: attachment;filename="' . $name . '.xlsx"');
        header('Cache-Control: max-age=0');
        // $writer = new Xlsx($spreadsheet);
        // $writer->save('php://output');
        // //删除清空:
        // $spreadsheet->disconnectWorksheets();
        // unset($spreadsheet);
        // exit;

       if (!file_exists(BASE_PATH . '/exportFile/')) {
            mkdir(BASE_PATH . '/exportFile/', 0777, true);
        }
        $path = BASE_PATH . '/exportFile/' . time() . uniqid() . '.xlsx';
        // 导出Excel文件
        $writer = IOFactory::createWriter($spreadsheet, 'Xlsx');
        $writer->save($path);
        return $path;
    }


    /**
     *  设定EXCEL表头
     * @param integer $num
     * @return array
     * @author: chaihao <158533752@qq.com>
     */
    public function getExcelHeader($num = 100)
    {
        $col = range('A', 'Z');
        $newCol = [];

        for ($j = 0; $j < ceil($num / 26); $j++) {
            for ($i = 0; $i < 26; $i++) {
                if (count($newCol) >= $num) {
                    break 2;
                }
                $newCol[] = ($j == 0 ? '' : $col[$j - 1]) . $col[$i];
            }
        }
        return $newCol;
    }


    /**
     * 处理多sheet
     * @param [type] $spreadsheet
     * @param [type] $n
     * @param [type] $data
     */
    public function opSheet($spreadsheet, $n, $data)
    {
        $spreadsheet->createSheet(); //创建sheet
        $objActSheet = $spreadsheet->setActiveSheetIndex($n); //设置当前的活动sheet
        // $keys = $data['rows'][0]; //这是你的数据键名
        // $count = count($keys); //计算你所占的列数
        $count = count($data['rows'][0]); //计算你所占的列数
        $infoNum = count($data['info'] ?? []); //求k-v值的所占行数, 多表头占用行
        $infoStart = $infoNum + 1; //下面的详细信息的开始行数
        $cellName = $this->getExcelHeader($count);

        $sheet = $spreadsheet->getActiveSheet($n)->setTitle($data['title']); //设置sheet的名称

        // $spreadsheet->getActiveSheet($n)->mergeCells('A1:' . $cellName[$count - 1] . '1'); //合并单元格
        // $spreadsheet->getActiveSheet($n)->getStyle('A1')->getFont()->setSize(20); //设置title的字体大小
        $spreadsheet->getActiveSheet($n)->getStyle($infoStart)->getFont()->setBold(true); //标题栏加粗
        // $objActSheet->setCellValue('A1', $data['title']); //设置每个sheet中的名称title


        foreach ($data['rows'] as $key => $item) {
            //循环设置单元格:
            for ($i = 0; $i < $count; $i++) {
                $sheet->setCellValue($cellName[$i] . ($key + $infoStart), $item[$i]);
                $spreadsheet->getActiveSheet($n)->getColumnDimension($cellName[$i])->setWidth(20); //固定列宽
            }
        }
    }
}
    /**
     * 导出数据
     * @return \Psr\Http\Message\ResponseInterface
     */
    public function exportLink($data = [])
    {
        if ($data) {
            //  设置 sheet 名
            $export['title'] = '链接';
            $export['rows'] = [];
            foreach ($data as $item) {
                $result = [
                    '标题' => $item['title'] ?? '',
                    '链接' => $item['url'],
                    '开始时间' => $item['start_date'],
                    '结束时间' => $item['end_date'],
                    'IP (总)' => $item['ip'],
                ];
                if (empty($export['rows'])) {
                    // 设置表头
                    $export['rows'][] = array_keys($result);
                }
                // 内容
                $export['rows'][] = array_values($result);
            }
            // 每个$export_sheet 都是 一个 sheet 表格
           //  [
           //   $export_sheet_1,
           //   $export_sheet_2,
           //   $export_sheet_3
           //   ...
           //  ]
            $path = (new ExportService())->xtExport([$export]);
            return $this->response->download($path);
        } else {
            return $this->failed('数据为空');
        }
    }

多表头处理

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,839评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,543评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,116评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,371评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,384评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,111评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,416评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,053评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,558评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,007评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,117评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,756评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,324评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,315评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,539评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,578评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,877评论 2 345