PHP使用PhpSpreadsheet实现导出Excel时带下拉框列表 (可支持三级联动)

因项目需要导出Excel表   需要支持下拉   且 还需要支持三级联动功能 

目前应为PHPExcel 不在维护,固采用 PhpSpreadsheet

效果如图:

 第一步:首先 使用composer 获取PhpSpreadsheet

我这里PHP 版本 7.4  命令如下:

composer require phpoffice/phpspreadsheet

 第二步:新建php 文件 代码如下   可根据实际情况 自己改动:

<?phprequire './vendor/autoload.php';
use PhpOffice\PhpSpreadsheet\Cell\DataValidation;
use PhpOffice\PhpSpreadsheet\IOFactory;
use PhpOffice\PhpSpreadsheet\NamedRange;
use PhpOffice\PhpSpreadsheet\Spreadsheet;
use PhpOffice\PhpSpreadsheet\Style\Alignment;
use PhpOffice\PhpSpreadsheet\Worksheet\Worksheet;
use PhpOffice\PhpSpreadsheet\Cell\Coordinate;
use PhpOffice\PhpSpreadsheet\Style\NumberFormat;
use PhpOffice\PhpSpreadsheet\Writer\Xlsx;// 定义表头
$header = [['col' => 'A', 'title' => '姓名', 'field' => 'name', 'width' => 20, 'id_down_type' => 0, 'is_text' => 1],['col' => 'B', 'title' => '证件号', 'field' => 'id_card', 'width' => 30, 'id_down_type' => 0, 'is_text' => 1],['col' => 'C', 'title' => '性别', 'field' => 'gender', 'width' => 20, 'id_down_type' => 1, 'is_text' => 1, 'count' => 2, 'sub_table_name' => 'sex'],['col' => 'D', 'title' => '联系方式', 'field' => 'phone', 'width' => 20, 'id_down_type' => 0, 'is_text' => 1],['col' => 'E', 'title' => '门店', 'field' => 'mendiao', 'width' => 30, 'id_down_type' => 2, 'is_text' => 0],['col' => 'F', 'title' => '公寓', 'field' => 'gongyu', 'width' => 30, 'id_down_type' => 2, 'is_text' => 0],['col' => 'G', 'title' => '门牌号', 'field' => 'menpai', 'width' => 30, 'id_down_type' => 2, 'is_text' => 0],
];// 下拉数据[这里模拟出来数据格式,实际情况从数据库获取数据并整理成下列数据格式]
$oneData = [['id' => 1,'title' => '我是A','text'=>'我是A_1','children' =>[['id' => 2,'title' => '我是A的下级A1','text'=>'我是A的下级A1_2','children' =>[['id' => 3,'title' => '我是A1的下级A11','text'=>'我是A1的下级A11_3',],['id' => 4,'title' => '我是A1的下级A12','text'=>'我是A1的下级A12_4',]]],['id' => 5,'title' => '我是A的下级A2','text'=>'我是A的下级A2_5','children' =>[['id' => 6,'title' => '我是A2的下级A21','text'=>'我是A2的下级A21_6',],['id' => 7,'title' => '我是A2的下级A22','text'=>'我是A2的下级A22_7',]]]]],['id' => 8,'title' => '我是B','text'=>'我是B_8','children' =>[['id' => 9,'title' => '我是B的下级B1','text'=>'我是B的下级B1_9','children' =>[['id' => 10,'title' => '我是B1的下级B11','text'=>'我是B1的下级B11_10',],['id' => 11,'title' => '我是B1的下级B12','text'=>'我是B1的下级B12_11',]]],['id' => 12,'title' => '我是B的下级B2','text' => '我是B的下级B2_12','children' =>[['id' => 13,'title' => '我是B2的下级B21','text' => '我是B2的下级B21_13',],['id' => 14,'title' => '我是B2的下级B22','text' => '我是B2的下级B22_14']]]]],
];//实例化Spreadsheet对象
$spreadsheet = new Spreadsheet();$sex_list = [['id' => 1, 'text' => '男'],['id' => 2, 'text' => '女'],];
$sub_table_num = 1;  //子表顺序  默认1 
//创建子表下拉定义  用于下拉
$sub_table_info_arr = [['title' => 'sex', 'index' => $sub_table_num, 'data' => $sex_list],
];
$sub_table_num++;//创建下拉分表数据
create_excel_sub_table($spreadsheet, $sub_table_info_arr);//创建下拉联动分表数据 
$liandong_name = 'region';  // 这里的region为分表名称
create_excel_drop_down($spreadsheet, $sub_table_num, $liandong_name, $oneData);  //无限极oneData   里面结构为[['text'=>'部门1_00001','children'=>[['text'=>'部门1子集1_00001']]]// 初始化表头
$spreadsheet->setActiveSheetIndex(0);
$spreadsheet->getActiveSheet()->setTitle('导入模板');foreach ($header as $key => $value) {//$spreadsheet->getActiveSheet()->setCellValueByColumnAndRow($key, 1, $value['title']);$spreadsheet->setActiveSheetIndex(0)->setCellValue($value['col'] . "1", $value['title']);$spreadsheet->getActiveSheet()->getStyle($value['col'] . "1", $value['title'])->getFont()->getColor()->setARGB(\PhpOffice\PhpSpreadsheet\Style\Color::COLOR_RED); //设置红色文字// 设置每列宽度$spreadsheet->getActiveSheet()->getColumnDimension($value['col'])->setWidth($value['width']);//设置单元格格式为文本if ($value['is_text'] == 1) {$spreadsheet->getActiveSheet()->getStyle($value['col'])->getNumberFormat()->setFormatCode(NumberFormat::FORMAT_TEXT);}
}
$objActSheet = $spreadsheet->getActiveSheet();// 输出下拉菜单
for ($i = 2; $i < 2000; $i++) {foreach ($header as $key1 => $value1) {//$this->set_select_cell($objActSheet,'A','id_type',$key,$i);if ($value1['id_down_type'] == 1) {//单纯下拉set_select_cell($objActSheet, $value1['col'], $value1['sub_table_name'], $value1['count'], $value1['title'], $i);} elseif ($value1['id_down_type'] == 2) {//联动下拉if ($value1['col'] == 'A' || $value1['col'] == 'E' || $value1['col'] == 'AC') {switch ($value1['col']) {case 'E':$fit_col = ['F','G'];break;case 'AC'://这里为关联 如AC 列  关联AD AE 这个$fit_col = ['AD', 'AE'];break;}$mian_col = $value1['col'];//处理联动下拉set_drop_down_select_cell($objActSheet, $mian_col, $fit_col, $value1['title'], $i,$liandong_name);}}}
}
$path = "./";
$subject = "导出模板".date('YmdHis');
$ext = ".xlsx";  //后缀
$export = $path . $subject . $ext;
$writer = new Xlsx($spreadsheet);
//保存文件
$writer->save($export);
//关闭连接,销毁变量
$spreadsheet->disconnectWorksheets();unset($spreadsheet);
var_dump($export);
die;/*** 创建Excel 分表* @param obpe Excel 对象* @param sub_table_info_arr  需要需要创建的分表二维数组   key title_name 分表表名标题    key index 分表顺序  key data 数据源  且数据源必须为 id text 键值对* @param write_data 写入的数据    二维  包含 id text 两个键值*/
function create_excel_sub_table($obpe, $sub_table_info_arr)
{if (!empty($sub_table_info_arr)) {foreach ($sub_table_info_arr as $key => $value) {$obpe->createSheet();$obpe->setActiveSheetIndex($value['index']);  //设置序号$obpe->getActiveSheet()->setTitle($value['title']); //设置表名$obpe->getActiveSheet()->setSheetState(Worksheet::SHEETSTATE_HIDDEN); //隐藏//密码$obpe->getActiveSheet()->getProtection()->setPassword('PHPExcel');$obpe->getActiveSheet()->getProtection()->setSheet(true);// This should be enabled inorder to enable any of thefollowing!$obpe->getActiveSheet()->getProtection()->setSort(true);$obpe->getActiveSheet()->getProtection()->setInsertRows(true);$obpe->getActiveSheet()->getProtection()->setFormatCells(true);$key = 1;if (!empty($value['data'])) {foreach ($value['data'] as $k => $v) {$str = $v['text'] . '_' . $v['id'];$obpe->getActiveSheet()->setCellValue('A' . $key, $str);$key++;}}}}
}/*** 生成Excel 列* @param int $num 需要的列数* @return array*/function  getExcelIndexRow($num=1){$r = [];$idx = ['idx' => 0];for ($i = 0; $i < $num; $i++) {$r[] =  getNextIdx($idx);}unset($idx);return $r;
}function getNextIdx(&$idx) {// 先转换26进制$num26 = base_convert($idx['idx'], 10, 26);$result = '';// 对每一位进行字符转换for ($i = 0; $i < strlen(strval($num26)); $i++) {if ($i == 0 && strlen(strval($num26)) != 1) {if (is_numeric($num26[$i])) {$result.= chr(ord($num26[$i]) + 16);} else {$result.= chr(ord($num26[$i]) - 23);}} else {if (is_numeric($num26[$i])) {$result.= chr(ord($num26[$i]) + 17);} else {$result.= chr(ord($num26[$i]) - 22);}}}$idx['idx'] = $idx['idx'] + 1;return $result;
}/***生成Excel 联动下拉* @param obpe Excel对象句柄* @param sub_table_num 分表序号* @param table_name 分表名称* @param data_source 需要处理的数据源*/
function create_excel_drop_down($obpe, $sub_table_num, $table_name, $data_source)
{//***********************$obpe->createSheet();$obpe->setActiveSheetIndex($sub_table_num);$obpe->getActiveSheet()->setTitle($table_name);$obpe->getActiveSheet()->setSheetState(\PhpOffice\PhpSpreadsheet\Worksheet\Worksheet::SHEETSTATE_HIDDEN); //隐藏//密码$obpe->getActiveSheet()->getProtection()->setPassword('PHPExcel');$obpe->getActiveSheet()->getProtection()->setSheet(true);// This should be enabled inorder to enable any of thefollowing!$obpe->getActiveSheet()->getProtection()->setSort(true);$obpe->getActiveSheet()->getProtection()->setInsertRows(true);$obpe->getActiveSheet()->getProtection()->setFormatCells(true);$key = 1;//$col = ['A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'AA', 'AB', 'AC', 'AD', 'AE', 'AF', 'AG', 'AH', 'AI', 'AJ', 'AK', 'AL', 'AM', 'AN', 'AO', 'AP', 'AQ', 'AR', 'AS', 'AT', 'AU', 'AV', 'AW', 'AX', 'AY', 'AZ', 'BA'];//这里定义1000个下拉  若超过 1000个下拉 异常输出$col = getExcelIndexRow(1000);$high = 0;//数据源格式必须按照以下格式规整 要不然数据处理出错/*$re = [['name'=>'湖北省','children'=>[['name'=>"武汉市",'children' =>['江夏区','洪山区','青山区','武昌区']],['name'=>"宜昌市",'children' =>['江夏区1','洪山区1','青山区1','武昌区1']],['name'=>"荆州市",'children' =>['江夏区2','洪山区2','青山区2','武昌区2']]]],['name'=>'湖南省','children'=>[['name'=>"长沙市",'children' =>['长沙市','长沙市1','长沙市3','长沙市6']],['name'=>"岳阳市",'children' =>['岳阳市4','岳阳市3','岳阳市2','岳阳市1']],[ 'name'=>"常德市",'children' =>['常德市1','常德市2','常德市3','常德市4']]]],];*/foreach ($data_source as $k => $v) {$str = $v['text'];$obpe->getActiveSheet()->setCellValue($col[0] . ($key + $high), $str);$max = 0; // 重置max$secondNum = count($v['children']);if (!empty($v['children'])) {foreach ($v['children'] as $index => $sendcond) {$obpe->getActiveSheet()->setCellValue($col[$index + 1] . ($key + $high), $sendcond['text']);//这里需要处理  若没有if (isset($sendcond['children']) && $sendcond['children']) {$thirdNum = count($sendcond['children']);} else {$thirdNum = 0;}if ($thirdNum > $max) {$max = $thirdNum;}if (!empty($sendcond['children'])) {foreach ($sendcond['children'] as $id => $third) {//$obpe->getActiveSheet()->setCellValue($col[$index+1].($key+$high+$id+1),$third);$obpe->getActiveSheet()->setCellValue($col[$index + 1] . ($key + $high + $id + 1), $third['text']);}//定义三级名称$obpe->addNamedRange(new NamedRange($sendcond['text'],$obpe->getSheetByName($table_name),$col[$index + 1] . ($key + $high + 1) . ':' . $col[$index + 1] . ($key + $high + 1 + $thirdNum - 1)));}}//定义二级$obpe->addNamedRange(new NamedRange($v['text'],$obpe->getSheetByName($table_name),$col[1] . ($key + $high) . ':' . $col[$secondNum] . ($key + $high)));}$high += $max;$key++;}//移花foreach ($data_source as $var => $cont) {$obpe->getSheetByName($table_name)->setCellValue('UI' . ($var + 1), $cont['text']);}$total = count($data_source);$n = 1;$obpe->addNamedRange(new NamedRange($table_name,$obpe->getSheetByName($table_name),"=\$UI\${$n}:\$UI\${$total}"));
}/***设置Excel  下拉 2020年8月4日16:02:09* @param objActSheet Excel对象句柄* @param col         操作的列* @param table_name  Excel电子薄分表下拉名称*/
function set_select_cell($objActSheet, $col = '', $table_name, $key = '', $setPromptTitle, $i = '', $setErrorTitle = '输入的值有误', $setError = '您输入的值不在下拉列表内')
{$objValidation = $objActSheet->getCell("{$col}" . $i)->getDataValidation();$objValidation->setType(DataValidation::TYPE_LIST)// ->setErrorStyle(DataValidation::STYLE_INFORMATION)->setErrorStyle(DataValidation::STYLE_STOP)->setAllowBlank(true)->setShowInputMessage(true)->setShowErrorMessage(true)->setShowDropDown(true)->setErrorTitle('输入的值有误')->setError('您输入的值不在下拉列表内')->setPromptTitle("{$setPromptTitle}(在列表内选择)")->setPrompt('请从列表中选择一个值')->setFormula1($table_name . '!$A$1:$A$' . $key);
}/*** 设置联动下拉的Excel 渲染 2020年8月8日20:23:39* @param objActSheet Excel对象句柄* @param mian_col    联动下拉第一列* @param fit_col     数组   子下拉节点  setFormula1 依次为下拉第一列名   后续依次关联* @param setPromptTitle       中文字段映射名称*/
function set_drop_down_select_cell($objActSheet, $main_col, $fit_col, $setPromptTitle, $i,$main_table_name='region')
{$indirect_list_arr = $fit_col;  //从主下拉节点开始计算array_unshift($indirect_list_arr, $main_col);//定义主表名//处理主下拉$objValidation = $objActSheet->getCell($main_col . $i)->getDataValidation();$objValidation->setType(\PhpOffice\PhpSpreadsheet\Cell\DataValidation::TYPE_LIST)->setErrorStyle(\PhpOffice\PhpSpreadsheet\Cell\DataValidation::STYLE_STOP)->setAllowBlank(false)->setShowInputMessage(true)->setShowErrorMessage(true)->setShowDropDown(true)->setErrorTitle('输入的值有误')->setError('您输入的值不在下拉列表内')->setPromptTitle("{$setPromptTitle}(在列表内选择)")//->setPromptTitle("type_list(在列表内选择)")->setPrompt('请从列表中选择一个值')->setFormula1("={$main_table_name}");//处理分节点下拉if (!empty($fit_col)) {foreach ($fit_col as $key => $value) {$objValidation = $objActSheet->getCell($value . $i)->getDataValidation();$objValidation->setType(\PhpOffice\PhpSpreadsheet\Cell\DataValidation::TYPE_LIST)->setErrorStyle(\PhpOffice\PhpSpreadsheet\Cell\DataValidation::STYLE_STOP)->setAllowBlank(false)->setShowInputMessage(true)->setShowErrorMessage(true)->setShowDropDown(true)->setErrorTitle('输入的值有误')->setError('您输入的值不在下拉列表内')->setPromptTitle('请选择')->setPrompt('请从列表中选择一个值')->setFormula1('=INDIRECT($' . $indirect_list_arr[$key] . $i . ')');}}return true;
}

步骤三: 运行php   我这里使用命令行 

php index.php

 查看即可

就是如此简单

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/77568.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Charles抓包工具使用(一)(macOS)

Fiddler抓包 | 竟然有这些骚操作&#xff0c;太神奇了&#xff1f; Fiddler响应拦截数据篡改&#xff0c;实现特殊场景深度测试&#xff08;一&#xff09; 利用Fiddler抓包调试工具&#xff0c;实现mock数据特殊场景深度测试&#xff08;二&#xff09; 利用Fiddler抓包调试工…

分布式电网动态电压恢复器模拟装置电子设计大赛

wx供重浩&#xff1a;创享日记 对话框发送&#xff1a;85电网 获取完整论文报告结构框图工程源文件 摘要&#xff1a;本装置采用DC-AC及AC-DC-AC双重结构&#xff0c;前级采用功率因数校正&#xff08;PFC&#xff09;电路完成AC-DC变换&#xff0c;改善输入端电网电能质量。后…

第28天-Kubernetes架构,集群部署,Ingress,项目部署,Dashboard

1.K8S集群部署 1.1.k8s快速入门 1.1.1.简介 Kubernetes简称k8s&#xff0c;是用于自动部署&#xff0c;扩展和管理容器化应用程序的开源系统。 中文官网&#xff1a;https://kubernetes.io/zh/中文社区&#xff1a;https://www.kubernetes.org.cn/官方文档&#xff1a;https…

【期末课程设计】学生成绩管理系统

因其独特&#xff0c;因其始终如一 文章目录 一、学生成绩管理系统介绍 二、学生成绩管理系统设计思路 三、源代码 1. test.c 2. Student Management System.c 3.Stu_System.c 4.Teacher.c 5.Student Management System.h 前言&#xff1a; 学生成绩管理系统含教师…

刷题DAY16

题目一 给定两个字符串str1和str2&#xff0c;再给定三个整数ic、dc和rc&#xff0c;分别代表插入、删除和替换一个字符的代价&#xff0c;返回将str1编辑成str2的最小代价。【举例]str1"abc",str2“adc",ic5,dc3,rc2从“abc“编辑成adc",把b替换成d是代价…

Android学习之路(1) App工程的项目结构

一、App工程的项目结构 1.项目下面有两个分类 一个是app(代表app模块)&#xff0c;另一个是Gradle Scripts。其中app下面又有3个子目录&#xff0c;其功能说明如下&#xff1a; manifests 子目录下面只有一个XML文件&#xff0c;即AndroidManifest.xmljava子目录&#xff0c;…

华为QinQ技术的基本qinq和灵活qinq 2种配置案例

基本qinq配置&#xff1a; 运营商pe设备在收到同一个公司的ce发来的的包&#xff0c;统一打上同样的vlan &#xff0c;如上图&#xff0c;同一个家公司两边统一打上vlan 2&#xff0c;等于在原内网vlan 10或20过来的包再统一打上vlan 2的标签&#xff0c;这样传输就不会和其它…

PostMan调用gitlab接口,OAuth 2.0 身份认证 API ,copy完事~

获取token接口&#xff1a; https://gitlab.***.com/oauth/token &#xff0c;接下来就可以调用其他功能的接口了 例&#xff1a;创建账户&#xff0c;将获取到的access_token放置在接口请求的token中 其他接口调用同上

华为OD机试之报文回路(Java源码)

题目描述 IGMP 协议中响应报文和查询报文&#xff0c;是维系组播通路的两个重要报文&#xff0c;在一条已经建立的组播通路中两个相邻的 HOST 和 ROUTER&#xff0c;ROUTER 会给 HOST 发送查询报文&#xff0c;HOST 收到查询报文后给 ROUTER 回复一个响应报文&#xff0c;以维持…

【前端知识】React 基础巩固(四十六)——自定义Hook的应用

React 基础巩固(四十六)——自定义Hook的应用 一、自定义Hook的应用 自定义Hook本质上只是一种函数代码逻辑的抽取&#xff0c;严格意义上而言&#xff0c;它并不算React的特性。 实现组件创建/销毁时打印日志 import React, { memo, useEffect, useState } from "react…

嵌入式开发学习(STC51-13-温度传感器)

内容 通过DS18B20温度传感器&#xff0c;在数码管显示检测到的温度值&#xff1b; DS18B20介绍 简介 DS18B20是由DALLAS半导体公司推出的一种的“一线总线&#xff08;单总线&#xff09;”接口的温度传感器&#xff1b; 与传统的热敏电阻等测温元件相比&#xff0c;它是一…

神码ai火车头标题伪原创【php源码】

这篇文章主要介绍了如何把python 代码打包成可执行软件&#xff0c;具有一定借鉴价值&#xff0c;需要的朋友可以参考下。希望大家阅读完这篇文章后大有收获&#xff0c;下面让小编带着大家一起了解一下。 火车头采集ai伪原创插件截图&#xff1a; Python 程序封装-打包成exe程…

【Apollo学习笔记】—— 相机仿真

文章目录 前言相关代码整理 测试实践文件目录包管理BUILD文件以及cyberfile.xml文件源程序BUILD运行结果其他参考CameraOutput channels启动camera驱动启动camera video compression驱动 前言 本文是对Cyber RT的学习记录,文章可能存在不严谨、不完善、有缺漏的部分&#xff0…

揭秘程序员最喜欢的5个高薪工作

大家好&#xff0c;这里是程序员晚枫。想了解更多精彩内容&#xff0c;快来关注程序员晚枫 今天给大家推荐5个适合程序员的高薪岗位。 01 推荐岗位 以下是5个工资最高的程序员工作&#xff1a; 数据科学家&#xff1a;数据科学家是负责数据收集、处理、分析和报告的专业人员。…

mssqlmysql数据库忽略大小写

一、mssql -- 创建数据时指定排序集Latin1_General_CI_AS CREATE DATABASE [数据库名] COLLATE Latin1_General_CI_AS 查询效果&#xff1a; 二、mysql

【数学】1、导论、数学归纳法与递归、分治

文章目录 一、数学归纳法与递归1.1 数学归纳法的过程1.2 递归1.2.1 本质就是数学归纳1.2.2 递归的场景1.2.2.1 编程实现数学归纳1.2.2.2 归并排序的分治思想1.2.2.3 分布式系统的分治思想 学习目标&#xff1a; 实用主义&#xff1a;为了解决工作日常的问题&#xff0c;而不是…

S7-200SMART与ET200SP远程IO模块进行PROFINET通信的具体方法

S7-200SMART与ET200SP远程IO模块进行PROFINET通信的具体方法 使用前提: 只有标准型且固件版本为V2.4及以上的S7-200 SMART CPU才支持 PROFINET 控制器功能。 S7-200 SMART 作 PROFINET 控制器最多可带8个 IO 设备(例如:远程 IO、阀岛、变频器、伺服和机器人等)。 本例中以 …

eclipse Java Editor Templates

​ Window - Preferences - Java - Editor - Templates ​ date ${currentDate:date(yyyy.MM.dd)}

九、pig安装

1.上传pig包 2.解压文件 3.改名 4.赋权 5.配置环境变量 export PIG_HOME/usr/local/pig export PATH$PATH:$JAVA_HOME/bin:$HADOOP_HOME/bin:$HADOOP_HOME/sbin:$HIVE_HOME/bin:$HBASE_HOME/bin:$SQOOP_HOME/bin:$PIG_HOME/bin 6.测试

如何在轻量级RTSP服务支持H.264扩展SEI发送接收自定义数据?

为什么开发轻量级RTSP服务&#xff1f; 开发轻量级RTSP服务的目的是为了解决在某些场景下用户或开发者需要单独部署RTSP或RTMP服务的问题。这种服务的优势主要有以下几点&#xff1a; 便利性&#xff1a;通过轻量级RTSP服务&#xff0c;用户无需配置单独的服务器&#xff0c;…