Excel.php 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. <?php
  2. declare(strict_types=1);
  3. namespace catcher\library\excel;
  4. use catcher\CatchUpload;
  5. use catcher\exceptions\FailedException;
  6. use catcher\Utils;
  7. use PhpOffice\PhpSpreadsheet\Exception;
  8. use PhpOffice\PhpSpreadsheet\Spreadsheet;
  9. use PhpOffice\PhpSpreadsheet\Worksheet\Worksheet;
  10. use think\file\UploadedFile;
  11. use think\helper\Str;
  12. class Excel
  13. {
  14. use MacroExcel;
  15. /**
  16. * @var ExcelContract $excel
  17. */
  18. protected $excel;
  19. protected $sheets;
  20. protected $spreadsheet = null;
  21. protected $extension = 'xlsx';
  22. /**
  23. * save
  24. *
  25. * @time 2020年05月25日
  26. * @param ExcelContract $excel
  27. * @param $path
  28. * @param string $disk
  29. * @return mixed
  30. * @throws Exception
  31. */
  32. public function save(ExcelContract $excel, $path, $disk = 'local',$filename=null)
  33. {
  34. $this->excel = $excel;
  35. $this->init();
  36. !is_dir($path) && mkdir($path, 0777, true);
  37. if($filename){
  38. $file = $path . $filename.date('YmdHis').Str::random(2) . '.' .$this->extension;
  39. }else{
  40. $file = $path . date('YmdHis').Str::random(6) . '.' .$this->extension;
  41. }
  42. Factory::make($this->extension)
  43. ->setSpreadsheet($this->spreadsheet)
  44. ->save($file);
  45. if (!file_exists($file)) {
  46. throw new FailedException($file . ' generate failed');
  47. }
  48. if ($disk) {
  49. $file = $this->upload($disk, $file);
  50. }
  51. return ['url' => $file];
  52. }
  53. /**
  54. * set extension
  55. *
  56. * @time 2020年09月08日
  57. * @param $extension
  58. * @return $this
  59. */
  60. public function setExtension($extension)
  61. {
  62. $this->extension = $extension;
  63. return $this;
  64. }
  65. /**
  66. * init excel
  67. *
  68. * @time 2020年05月25日
  69. * @throws Exception
  70. * @return void
  71. */
  72. protected function init()
  73. {
  74. $this->setMemoryLimit();
  75. // register worksheet for current excel
  76. $this->registerWorksheet();
  77. // before save excel
  78. $this->before();
  79. // set excel title
  80. $this->setTitle();
  81. // set excel headers
  82. $this->setExcelHeaders();
  83. // set cell width
  84. $this->setSheetWidth();
  85. // set worksheets
  86. $this->setWorksheets();
  87. }
  88. /**
  89. * 设置 sheets
  90. *
  91. * @time 2020年05月25日
  92. * @throws Exception
  93. * @return void
  94. */
  95. protected function setWorksheets()
  96. {
  97. $keys= $this->getKeys();
  98. $isArray = $this->arrayConfirm();
  99. $worksheet = $this->getWorksheet();
  100. if (empty($keys)) {
  101. if ($isArray) {
  102. foreach ($this->excel->sheets() as $sheet) {
  103. $worksheet->fromArray($sheet, null, $this->start . $this->row);
  104. $this->incRow();
  105. }
  106. } else {
  107. foreach ($this->excel->sheets() as $sheet) {
  108. $worksheet->fromArray($sheet->toArray(), null, $this->start . $this->row);
  109. $this->incRow();
  110. }
  111. }
  112. } else {
  113. if ($isArray) {
  114. foreach ($this->excel->sheets() as $sheet) {
  115. $worksheet->fromArray($this->getValuesByKeys($sheet, $keys), null, $this->start . $this->row);
  116. $this->incRow();
  117. }
  118. } else {
  119. foreach ($this->excel->sheets() as $sheet) {
  120. $worksheet->fromArray($this->getValuesByKeys($sheet->toArray(), $keys), null, $this->start . $this->row);
  121. $this->incRow();
  122. }
  123. }
  124. }
  125. }
  126. /**
  127. * 判断 sheet 是否是 array 类型
  128. *
  129. * @time 2020年05月25日
  130. * @return bool
  131. */
  132. protected function arrayConfirm()
  133. {
  134. $sheets = $this->excel->sheets();
  135. $array = true;
  136. foreach ($sheets as $sheet) {
  137. $array = is_array($sheet);
  138. break;
  139. }
  140. return $array;
  141. }
  142. /**
  143. * 获取 item 特定 key 的值
  144. *
  145. * @time 2020年05月25日
  146. * @param array $item
  147. * @param array $keys
  148. * @return array
  149. */
  150. protected function getValuesByKeys(array $item, array $keys)
  151. {
  152. $array = [];
  153. foreach ($keys as $key) {
  154. $array[] = $item[$key];
  155. }
  156. return $array;
  157. }
  158. /**
  159. * 设置 Excel 头部
  160. *
  161. * @time 2020年05月23日
  162. * @throws Exception
  163. */
  164. protected function setExcelHeaders()
  165. {
  166. $worksheet = $this->getWorksheet();
  167. // get columns
  168. $columns = $this->getSheetColumns();
  169. // get start row
  170. $startRow = $this->getStartRow();
  171. foreach ($this->excel->headers() as $k => $header) {
  172. $worksheet->getCell($columns[$k] . $startRow)->setValue($header);
  173. }
  174. $this->incRow();
  175. }
  176. /**
  177. * get spreadsheet
  178. *
  179. * @time 2020年05月25日
  180. * @return Spreadsheet
  181. */
  182. protected function getSpreadsheet()
  183. {
  184. if (!$this->spreadsheet) {
  185. $this->spreadsheet = new Spreadsheet();
  186. }
  187. return $this->spreadsheet;
  188. }
  189. /**
  190. * 获取 active sheet
  191. *
  192. * @time 2020年05月25日
  193. * @throws Exception
  194. * @return Worksheet
  195. */
  196. protected function getWorksheet()
  197. {
  198. return $this->getSpreadsheet()->getActiveSheet();
  199. }
  200. /**
  201. * upload
  202. *
  203. * @time 2020年05月25日
  204. * @param $disk
  205. * @param $path
  206. * @return string
  207. * @throws \Exception
  208. */
  209. protected function upload($disk, $path)
  210. {
  211. if ($disk == 'local') {
  212. return $this->local($path);
  213. }
  214. $upload = new CatchUpload;
  215. return ($disk ? $upload->setDriver($disk) : $upload)->upload($this->uploadedFile($path));
  216. }
  217. /**
  218. * 返回本地下载地址
  219. *
  220. * @param $path
  221. * @time 2020年09月08日
  222. * @return mixed
  223. */
  224. protected function local($path)
  225. {
  226. // return \config('filesystem.disks.local')['domain'] . '/' . str_replace('\\', '/', str_replace(Utils::publicPath(), '', $path));
  227. return \config('filesystem.disks.local')['domain'] . '/' . str_replace(str_replace('\\', '/', Utils::publicPath()), '', str_replace('\\', '/', $path));
  228. }
  229. /**
  230. * get uploaded file
  231. *
  232. * @time 2020年05月25日
  233. * @param $file
  234. * @return UploadedFile
  235. */
  236. protected function uploadedFile($file)
  237. {
  238. return new UploadedFile($file, pathinfo($file, PATHINFO_BASENAME));
  239. }
  240. }