【简化】让我们从POI毕业吧
背景-
背景
你用过POI吗?如果你感到遇到了“质量差的难以理解的实现(Poor Obfuscation Implementation)”,那就可以说你抓住了本质^_^。
仔细考虑对Excel文件的操作,可以分为三种类型:
1. 创建数据表
2. 美化数据表
3. 添加气泡、框等动作
换句话说,有三种工作,分别是”数据操作”、”格式操作”和”图形操作”。”数据操作”相对较简单,只需进行get或set操作。而”格式操作”和”图形操作”则比较繁琐,需要设置各种线条颜色、样式、阴影等,如果用POI实现的话,代码会变得非常冗长。
然而,如果我们手动操作Excel,应该没有人会觉得麻烦。因为只要复制粘贴,就可以将所有东西整齐地排列好。如果使用刷子,还可以更方便。如果能将这种方法整合到API中,那将会变成一种易于使用的方法吧?
介紹 efw 的 Excel API
整体形象
样本
我提供以下三种样本。
efw 的 JavaScript 事件示例
var helloExcel_submit={};
helloExcel_submit.paramsFormat={};
helloExcel_submit.fire=function(params){
//ファイル作成時、テンプレートをコピー作成
new Excel("excel/IamExcelTemplate.xlsx")
//シート作成時、テンプレートをコピー作成
.createSheet("newSheet","templateSheet")
//セルに値を代入とともに、書式のコピー元を指定
.setCell("newSheet","B2","helloworld","templateSheet","A1")
//テンプレートの図形をコピーして、新しい場所に貼り付ける
.addShape("newSheet","C2","templateSheet","myCircle")
//保存する
.save("myExcel.xlsx")
//閉じる
.close();
return new Result().attach("myExcel.xlsx").deleteAfterDownload();
}
使用Java中的EFW的Excel API示例。
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@ page import="efw.excel.ExcelManager"%>
<%@ page import="efw.excel.Excel"%>
<%
//ファイル作成時、テンプレートをコピー作成
Excel excel=ExcelManager.open("excel/IamExcelTemplate.xlsx", false);
//シート作成時、テンプレートをコピー作成
excel.createSheet("newSheet","templateSheet");
//セルに値を代入とともに、書式のコピー元を指定
excel.setCellStringValue("newSheet","B2","helloworld");
excel.setCellStyle("newSheet","B2","templateSheet","A1");
//テンプレートの図形をコピーして、新しい場所に貼り付ける
excel.addShapeInCell("newSheet","C2", "templateSheet","myCircle", "", 0, 0, 0, 0);
//保存する
excel.save("myExcel.xlsx", null);
//閉じる
ExcelManager.close(excel.getKey());
%>
原始的POI样本
你认为这个实现很糟糕且难以阅读吗?
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@ page import="efw.file.FileManager"%>
<%@ page import="java.io.File"%>
<%@ page import="java.io.FileOutputStream"%>
<%@ page import="java.io.IOException"%>
<%@ page import="org.apache.poi.EncryptedDocumentException"%>
<%@ page import="org.apache.poi.ss.usermodel.Cell"%>
<%@ page import="org.apache.poi.ss.usermodel.PrintSetup"%>
<%@ page import="org.apache.poi.ss.usermodel.Row"%>
<%@ page import="org.apache.poi.ss.usermodel.Sheet"%>
<%@ page import="org.apache.poi.ss.usermodel.Workbook"%>
<%@ page import="org.apache.poi.ss.usermodel.WorkbookFactory"%>
<%@ page import="org.apache.poi.ss.util.CellReference"%>
<%@ page import="org.apache.poi.xssf.usermodel.XSSFDrawing"%>
<%@ page import="org.apache.poi.xssf.usermodel.XSSFShape"%>
<%@ page import="org.apache.poi.xssf.usermodel.XSSFSheet"%>
<%@ page import="org.apache.poi.xssf.usermodel.XSSFClientAnchor"%>
<%@ page import="org.apache.poi.xssf.usermodel.XSSFSimpleShape"%>
<%@ page import="org.apache.poi.xssf.usermodel.XSSFTextParagraph"%>
<%@ page import="org.apache.poi.xssf.usermodel.XSSFTextRun"%>
<%@ page import="org.apache.poi.xssf.usermodel.XSSFPicture"%>
<%@ page import="org.apache.poi.xssf.usermodel.XSSFPictureData"%>
<%@ page import="org.apache.poi.openxml4j.opc.PackagePart"%>
<%@ page import="org.apache.poi.ss.usermodel.Font"%>
<%@ page import="java.util.List"%>
<%
//テンプレートを一時ファイルにコピーする
File file=FileManager.get("excel/IamExcelTemplate.xlsx");
File tempFile=File.createTempFile("efw", "");
FileManager.duplicate(file, tempFile);
Workbook workbook=WorkbookFactory.create(tempFile);
//新しいシートを作成する
Sheet sheet = workbook.cloneSheet(workbook.getSheetIndex("templateSheet"));
workbook.setSheetName(workbook.getSheetIndex(sheet.getSheetName()), "newSheet");
Sheet tempSheet = workbook.getSheet("templateSheet");
//B2セルに値を設定する
CellReference reference = new CellReference("B2");
Row row = sheet.getRow(reference.getRow());
Cell cell = null;
if (row == null) {
row = sheet.createRow(reference.getRow());
row.setHeightInPoints(sheet.getDefaultRowHeightInPoints());//初期高さを設定
}
cell = row.getCell(reference.getCol());
if (cell==null){
cell=row.createCell(reference.getCol());
}
cell.setCellValue("helloworld");
//B2セルにA1の書式を利用する
CellReference templateReference = new CellReference("A1");
Row templateRow = tempSheet.getRow(templateReference.getRow());
if (templateRow != null) {
Cell templateCell=templateRow.getCell(templateReference.getCol());
cell.setCellStyle(templateCell.getCellStyle());
}
//図形を作成する
CellReference shapeReference = new CellReference("C2");
int cellrow=shapeReference.getRow();
int cellcol=shapeReference.getCol();
XSSFSheet xsheet = (XSSFSheet) workbook.getSheet("newSheet");
XSSFSheet xtemplateSheet=(XSSFSheet) workbook.getSheet("templateSheet");
List<XSSFShape> templateShapes=((XSSFDrawing) xtemplateSheet.getDrawingPatriarch()).getShapes();
for (XSSFShape templateShape : templateShapes) {
if ("myCircle".equals(templateShape.getShapeName())) {
XSSFDrawing patriarch=xsheet.getDrawingPatriarch();
if(patriarch==null) patriarch = xsheet.createDrawingPatriarch();
XSSFShape shape=cloneShape(workbook,patriarch,templateShape,"");
XSSFClientAnchor anchor=(XSSFClientAnchor)(shape.getAnchor());
int x=0;
int y=0;
int dx1=0;
int dy1=0;
int dx2=0;
int dy2=0;
int width=0;
int height=0;
if(x==0){
dx1=anchor.getDx1();
}
if(y==0){
dy1=anchor.getDy1();
}
if(width==0){
width=anchor.getDx2()-anchor.getDx1();
}
if(height==0){
height=anchor.getDy2()-anchor.getDy1();
}
dx2=width+dx1;
dy2=height+dy1;
anchor.setRow1(cellrow);
anchor.setRow2(cellrow);
anchor.setCol1(cellcol);
anchor.setCol2(cellcol);
anchor.setDx1(dx1);
anchor.setDy1(dy1);
anchor.setDx2(dx2);
anchor.setDy2(dy2);
break;
}
}
//目的ファイルとして保存する
FileOutputStream fileOutputStream = new FileOutputStream(FileManager.get("myExcel.xlsx"));
workbook.setForceFormulaRecalculation(true);
workbook.write(fileOutputStream);
fileOutputStream.close();
workbook.close();
//一時ファイルファイルを削除する
tempFile.delete();
%>
<!-- 以下はclone関数 -->
<%!
/**
* XSSFのShapeをコピーする
* @param patriarch
* @param templateShape
* @param value
* @return 作成されたshape
*/
XSSFShape cloneShape(Workbook workbook,XSSFDrawing patriarch,XSSFShape templateShape,String value){
String clsNm=templateShape.getClass().getSimpleName();
if ("XSSFSimpleShape".equals(clsNm)) {
XSSFSimpleShape orgSimpleShape=(XSSFSimpleShape)templateShape;
XSSFSimpleShape simpleShape = patriarch.createSimpleShape((XSSFClientAnchor)orgSimpleShape.getAnchor());
simpleShape.getCTShape().set(orgSimpleShape.getCTShape().copy());
if(orgSimpleShape.getTextParagraphs().size()>0){
XSSFTextParagraph tempParagraph=orgSimpleShape.getTextParagraphs().get(0);
if(tempParagraph.getTextRuns().size()>0){
XSSFTextRun tempRun=tempParagraph.getTextRuns().get(0);
simpleShape.setText(tempRun.getText());
if(simpleShape.getTextParagraphs().size()>0){
XSSFTextParagraph paragraph=simpleShape.getTextParagraphs().get(0);
if(paragraph.getTextRuns().size()>0){
XSSFTextRun textRun= paragraph.getTextRuns().get(0);
if (value!=null) {
textRun.setText(value);
}else {
textRun.setText(tempRun.getText());
}
textRun.setFontSize(tempRun.getFontSize());
textRun.setCharacterSpacing(tempRun.getCharacterSpacing());
textRun.setFontColor(tempRun.getFontColor());
textRun.setFontFamily(tempRun.getFontFamily(), Font.DEFAULT_CHARSET, tempRun.getPitchAndFamily(), false);
paragraph.setTextAlign(tempParagraph.getTextAlign());
paragraph.setTextFontAlign(tempParagraph.getTextFontAlign());
}
}
}
}
return (XSSFShape)simpleShape;
}else if ("XSSFPicture".equals(clsNm)) {
XSSFPicture orgPicture=(XSSFPicture)templateShape;
PackagePart orgPackagePart= orgPicture.getPictureData().getPackagePart();
@SuppressWarnings("unchecked")
List<XSSFPictureData> allpictures=(List<XSSFPictureData>) workbook.getAllPictures();
int pictureIndex=-1;
for(int index=0;index<allpictures.size();index++) {
PackagePart packagePart=allpictures.get(index).getPackagePart();
if (packagePart.equals(orgPackagePart)) {
pictureIndex=index;
break;
}
}
XSSFPicture picture = patriarch.createPicture((XSSFClientAnchor)orgPicture.getAnchor(),pictureIndex);
return (XSSFShape)picture;
}
return null;
}
%>
获取玻璃罐
<dependency>
<groupId>io.github.efwgrp</groupId>
<artifactId>efw</artifactId>
<version>4.07.000</version>
</dependency>
如果使用JDK15及以上版本,则需要相关的jar文件。
<dependency>
<groupId>org.openjdk.nashorn</groupId>
<artifactId>nashorn-core</artifactId>
<version>15.4</version>
</dependency>
这是最关键的兴趣点了。
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi</artifactId>
<version>5.2.3</version>
</dependency>
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml</artifactId>
<version>5.2.3</version>
</dependency>
这次的样本可以通过以下链接进行下载。同时,它包含了efw的Excel操作API的所有测试。如果你有兴趣,可以参考一下。