Apache POI PPT(Java) - 更新PPT图表基础值,保持格式(位置,尺寸,颜色)

时间:2017-10-04 08:13:57

标签: java charts formatting apache-poi powerpoint

我正在尝试根据不同的输入/为我创建的同一个PPTX模板上的不同用户生成多个报告(即N PPTX文件)。

我在PPTX模板上有几个预先格式化的XSLFChart,它包含一个伪底层值。图表已经格式化(即图例,颜色,位置,...形状和相关文本)。 每个图表都包含一个基础数据表作为占位符,我需要用dynimic值替换它。我在不同的数据结构中有这个值。我成功更新了一个PieChart(见下面的例子)但我没有使用barChart barStackedChart和其他人。

public static void updatePieChart(XSLFChart chart, Map<String, String> pieChartValues) {
    // Embedded Excel workbook that holds the chart data
    String pieChartTitle = "MyUpdatedPieChart";
    POIXMLDocumentPart xlsPart = chart.getRelations().get(0);
    try (XSSFWorkbook wb = new XSSFWorkbook()) {
        XSSFSheet sheet = wb.createSheet();

        CTChart ctChart = chart.getCTChart();
        CTPlotArea plotArea = ctChart.getPlotArea();

        CTPieChart pieChart = plotArea.getPieChartArray(0);
        //Pie Chart Series
        CTPieSer ser = pieChart.getSerArray(0);

        // Series Text
        CTSerTx tx = ser.getTx();
        tx.getStrRef().getStrCache().getPtArray(0).setV(pieChartTitle);
        sheet.createRow(0).createCell(1).setCellValue(pieChartTitle);
        String titleRef = new CellReference(sheet.getSheetName(), 0, 1, true, true).formatAsString();
        tx.getStrRef().setF(titleRef);

        // Category Axis Data
        CTAxDataSource cat = ser.getCat();
        CTStrData strData = cat.getStrRef().getStrCache();

        // Values
        CTNumDataSource val = ser.getVal();
        CTNumData numData = val.getNumRef().getNumCache();

        strData.setPtArray(null);  // unset old axis text
        numData.setPtArray(null);  // unset old values

        // set model
        int idx = 0;
        int rownum = 1;
        for (Object key : pieChartValues.keySet()) {
            CTNumVal numVal = numData.addNewPt();
            numVal.setIdx(idx);
            numVal.setV(pieChartValues.get(key));

            CTStrVal sVal = strData.addNewPt();
            sVal.setIdx(idx);
            sVal.setV((String) key);

            idx++;
            XSSFRow row = sheet.createRow(rownum++);
            row.createCell(0).setCellValue((String) key);
            row.createCell(1).setCellValue(Double.valueOf(pieChartValues.get(key)));

        }
        numData.getPtCount().setVal(idx);
        strData.getPtCount().setVal(idx);

        String numDataRange = new CellRangeAddress(1, rownum - 1, 1, 1).formatAsString(sheet.getSheetName(), true);
        val.getNumRef().setF(numDataRange);
        String axisDataRange = new CellRangeAddress(1, rownum - 1, 0, 0).formatAsString(sheet.getSheetName(), true);
        cat.getStrRef().setF(axisDataRange);

        // updated the embedded workbook with the data
        try (OutputStream xlsOut = xlsPart.getPackagePart().getOutputStream()) {
            wb.write(xlsOut);
        }

    } catch (IOException ex) {
        Logger.getLogger(Main.class.getName()).log(Level.SEVERE, null, ex);
    }
}

这里适用于BarChart的相同代码。它编译得很好,但是当我尝试打开PPT时“PowerPoint在FileName.pptx中发现了注释问题”。

public static void updateBarChart(XSLFChart chart, Map<String, String> pieChartValues, String pieChartTitle) {
    // Embedded Excel workbook that holds the chart data
    POIXMLDocumentPart xlsPart = chart.getRelations().get(0);
    try (XSSFWorkbook wb = new XSSFWorkbook()) {
        XSSFSheet sheet = wb.createSheet();

        CTChart ctChart = chart.getCTChart();
        CTPlotArea plotArea = ctChart.getPlotArea();

        CTBarChart barChart = plotArea.getBarChartArray(0);
        //Pie Chart Series
        CTBarSer ser = barChart.getSerArray(0);

        // Series Text
        CTSerTx tx = ser.getTx();
        tx.getStrRef().getStrCache().getPtArray(0).setV(pieChartTitle);
        sheet.createRow(0).createCell(1).setCellValue(pieChartTitle);
        String titleRef = new CellReference(sheet.getSheetName(), 0, 1, true, true).formatAsString();
        tx.getStrRef().setF(titleRef);

        // Category Axis Data
        CTAxDataSource cat = ser.getCat();
        CTStrData strData = cat.getStrRef().getStrCache();

        // Values
        CTNumDataSource val = ser.getVal();
        CTNumData numData = val.getNumRef().getNumCache();

        strData.setPtArray(null);  // unset old axis text
        numData.setPtArray(null);  // unset old values

        // set model
        int idx = 0;
        int rownum = 1;
        for (Object key : pieChartValues.keySet()) {
            CTNumVal numVal = numData.addNewPt();
            numVal.setIdx(idx);
            numVal.setV(pieChartValues.get(key));

            CTStrVal sVal = strData.addNewPt();
            sVal.setIdx(idx);
            sVal.setV((String) key);

            idx++;
            XSSFRow row = sheet.createRow(rownum++);
            row.createCell(0).setCellValue((String) key);
            row.createCell(1).setCellValue(Double.valueOf(pieChartValues.get(key)));

        }
        numData.getPtCount().setVal(idx);
        strData.getPtCount().setVal(idx);

        String numDataRange = new CellRangeAddress(1, rownum - 1, 1, 1).formatAsString(sheet.getSheetName(), true);
        val.getNumRef().setF(numDataRange);
        String axisDataRange = new CellRangeAddress(1, rownum - 1, 0, 0).formatAsString(sheet.getSheetName(), true);
        cat.getStrRef().setF(axisDataRange);

        // updated the embedded workbook with the data
        try (OutputStream xlsOut = xlsPart.getPackagePart().getOutputStream()) {
            wb.write(xlsOut);
        }

    } catch (IOException ex) {
        Logger.getLogger(Main.class.getName()).log(Level.SEVERE, null, ex);
    }
}

我想创建一个给出模板图表(XSLFChart)和数据结构(也可能是List - 因此每个数据表行一个String [])的单个函数,其中包含新值,更新PPT文件上的图表

有任何建议/解决方案吗?谢谢你的帮助!

0 个答案:

没有答案