Это не так просто, как вы думаете. Мало того, что есть много разных единиц измерения, которые нужно учитывать, к сожалению, также есть большие различия между двоичными *.xls
и Office Open XML
*.xlsx
форматами файлов.
Лучшей единицей измерения для работы здесь будет пиксели. Это может быть преобразовано в EMU
, используя Units.EMU_PER_PIXEL
. Поэтому вместо изменения размера изображения в сантиметрах это следует делать в пикселях. Но можно преобразовать сантиметр в пиксели по следующей формуле:
float pixels = cm / 2.54f * 72f * Units.PIXEL_DPI / Units.POINT_DPI
То есть: 1 дюйм = 2,54 см, поэтому см / 2,54 - это дюймы, дюймы * 72 - это точки и точки * пиксель DPI / точки DPI - пикселей.
Если нам нужно работать с использованием пикселей и хотеть разместить изображение по горизонтали и вертикали по центру над ячейкой, конечно, нам также нужно иметь ширину ячейки и высоту строки в пикселях. Имея это, мы можем рассчитать горизонтальное и вертикальное начальное и конечное положение изображения над ячейкой. Затем мы можем установить привязку изображения как начало в верхнем левом углу ячейки плюс dx1
= горизонтальная начальная позиция и плюс dy1
= вертикальная начальная позиция. Кроме того, якорь изображения заканчивается в верхнем левом углу ячейки, плюс dx2
= горизонтальное конечное положение и плюс dy2
= вертикальное конечное положение.
Следующий полный пример помещает изображение размером 3 см x 1,5 см по горизонтали и вертикали по центру над ячейкой B2
, которая имеет высоту 100pt и ширину 50 символов по умолчанию. Работает как для HSSF
, так и для XSSF
.
import java.io.InputStream;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import org.apache.poi.ss.usermodel.*;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.apache.poi.util.IOUtils;
import org.apache.poi.util.Units;
class CreateExcelPictureOverCell {
public static void main(String[] args) throws Exception {
Workbook workbook = new HSSFWorkbook(); String filePath = "./Excel.xls";
//Workbook workbook = new XSSFWorkbook(); String filePath = "./Excel.xlsx";
Sheet sheet = workbook.createSheet();
Row row = sheet.createRow(1); // row 2
float rowHeightInPoints = 100f;
row.setHeightInPoints(rowHeightInPoints);
float rowHeightInPixels = rowHeightInPoints * Units.PIXEL_DPI / Units.POINT_DPI;
Cell cell = row.createCell(1); // col B
sheet.setColumnWidth(1, 50*256); // 50 default characters width
float colWidthInPixels = sheet.getColumnWidthInPixels(1);
InputStream inputStream = new FileInputStream("./logo.png");
byte[] imageBytes = IOUtils.toByteArray(inputStream);
int pictureureIdx = workbook.addPicture(imageBytes, Workbook.PICTURE_TYPE_PNG);
inputStream.close();
CreationHelper helper = workbook.getCreationHelper();
Drawing drawing = sheet.createDrawingPatriarch();
ClientAnchor anchor = helper.createClientAnchor();
//set start position of picture's anchor to top left B2
anchor.setRow1(1);
anchor.setCol1(1);
//create picture
Picture pict = drawing.createPicture(anchor, pictureureIdx);
//get picture's original size
int pictOriginalWidthInPixels = pict.getImageDimension().width;
int pictOriginalHeightInPixels = pict.getImageDimension().height;
//set picture's wanted size
float pictWidthInCm = 3f;
float pictWidthInPixels = pictWidthInCm / 2.54f * 72f * Units.PIXEL_DPI / Units.POINT_DPI;
//want scaling in aspect ratio?
//float scale = pictWidthInPixels / pictOriginalWidthInPixels;
//float pictHeightInPixels = pictOriginalHeightInPixels * scale;
//want explicit set height too?
float pictHeightInCm = 1.5f;
float pictHeightInPixels = pictHeightInCm / 2.54f * 72f * Units.PIXEL_DPI / Units.POINT_DPI;
//calculate the horizontal center position
int horCenterPosInPixels = Math.round(colWidthInPixels/2f - pictWidthInPixels/2f);
//set the horizontal center position as Dx1 of anchor
if (workbook instanceof XSSFWorkbook) {
anchor.setDx1(horCenterPosInPixels * Units.EMU_PER_PIXEL); //in unit EMU for XSSF
} else if (workbook instanceof HSSFWorkbook) {
//see https://stackoverflow.com/questions/48567203/apache-poi-xssfclientanchor-not-positioning-picture-with-respect-to-dx1-dy1-dx/48607117#48607117 for HSSF
int DEFAULT_COL_WIDTH = 10 * 256;
anchor.setDx1(Math.round(horCenterPosInPixels * Units.DEFAULT_CHARACTER_WIDTH / 256f * 14.75f * DEFAULT_COL_WIDTH / colWidthInPixels));
}
//calculate the vertical center position
int vertCenterPosInPixels = Math.round(rowHeightInPixels/2f - pictHeightInPixels/2f);
//set the vertical center position as Dy1 of anchor
if (workbook instanceof XSSFWorkbook) {
anchor.setDy1(Math.round(vertCenterPosInPixels * Units.EMU_PER_PIXEL)); //in unit EMU for XSSF
} else if (workbook instanceof HSSFWorkbook) {
//see https://stackoverflow.com/questions/48567203/apache-poi-xssfclientanchor-not-positioning-picture-with-respect-to-dx1-dy1-dx/48607117#48607117 for HSSF
float DEFAULT_ROW_HEIGHT = 12.75f;
anchor.setDy1(Math.round(vertCenterPosInPixels * Units.PIXEL_DPI / Units.POINT_DPI * 14.75f * DEFAULT_ROW_HEIGHT / rowHeightInPixels));
}
//set end position of picture's anchor to top left B2
anchor.setRow2(1);
anchor.setCol2(1);
//calculate the horizontal end position of picture
int horCenterEndPosInPixels = Math.round(horCenterPosInPixels + pictWidthInPixels);
//set the horizontal end position as Dx2 of anchor
if (workbook instanceof XSSFWorkbook) {
anchor.setDx2(horCenterEndPosInPixels * Units.EMU_PER_PIXEL); //in unit EMU for XSSF
} else if (workbook instanceof HSSFWorkbook) {
//see https://stackoverflow.com/questions/48567203/apache-poi-xssfclientanchor-not-positioning-picture-with-respect-to-dx1-dy1-dx/48607117#48607117 for HSSF
int DEFAULT_COL_WIDTH = 10 * 256;
anchor.setDx2(Math.round(horCenterEndPosInPixels * Units.DEFAULT_CHARACTER_WIDTH / 256f * 14.75f * DEFAULT_COL_WIDTH / colWidthInPixels));
}
//calculate the vertical end position of picture
int vertCenterEndPosInPixels = Math.round(vertCenterPosInPixels + pictHeightInPixels);
//set the vertical end position as Dy2 of anchor
if (workbook instanceof XSSFWorkbook) {
anchor.setDy2(Math.round(vertCenterEndPosInPixels * Units.EMU_PER_PIXEL)); //in unit EMU for XSSF
} else if (workbook instanceof HSSFWorkbook) {
//see https://stackoverflow.com/questions/48567203/apache-poi-xssfclientanchor-not-positioning-picture-with-respect-to-dx1-dy1-dx/48607117#48607117 for HSSF
float DEFAULT_ROW_HEIGHT = 12.75f;
anchor.setDy2(Math.round(vertCenterEndPosInPixels * Units.PIXEL_DPI / Units.POINT_DPI * 14.75f * DEFAULT_ROW_HEIGHT / rowHeightInPixels));
}
FileOutputStream out = new FileOutputStream(filePath);
workbook.write(out);
out.close();
workbook.close();
}
}