Как изменить графические атрибуты точки в диаграмме солнечных лучей Excel через Apache POI - PullRequest
0 голосов
/ 14 сентября 2018

У меня есть требование окрашивать различные точки данных в диаграмме солнечных лучей Excel программно.По умолчанию Excel создает диаграмму в следующем виде.enter image description here

Мне нужно сделать что-то подобное.enter image description here

Мне удалось загрузить график и серию, но я не смог понять, как добраться до каждой из точек и изменить цвет заливки.и возможно ли это.

Данные для создания этой диаграммы:

Level 1,Level 2,Level 3,Series 1
A,A.a,A.a.1,5
A,A.a,A.a.2,5
A,A.b,A.b.1,5
A,A.b,A.b.2,5
B,B.a,B.a.1,5
B,B.a,B.a.2,5
B,B.b,B.b.1,5
B,B.b,B.b.2,5
C,C.a,C.a.1,5
C,C.a,C.a.2,5
C,C.b,C.b.1,5
C,C.b,C.b.2,5

Мой код пока

import java.io.IOException;
import java.util.List;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.apache.poi.xddf.usermodel.chart.XDDFChartData;
import org.apache.poi.xssf.usermodel.XSSFChart;
import org.apache.poi.xssf.usermodel.XSSFDrawing;
import org.apache.poi.xssf.usermodel.XSSFSheet;

public class Format {
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        try {
            XSSFWorkbook xwb = new XSSFWorkbook("ChartExample.xlsx");
            XSSFSheet sheet = xwb.getSheetAt(0);
            System.out.println("Loaded sheet is " + sheet.getSheetName());
            XSSFDrawing drawing = sheet.getDrawingPatriarch();
            List <XSSFChart> charts = drawing.getCharts();
            System.out.println("No of Charts " + charts.size());
            XSSFChart chart = charts.get(0);
            List<XDDFChartData> series = chart.getChartSeries();
            System.out.println("No of Data Series " + series.size());
            XDDFChartData data = series.get(0);
            // How do I now get to the data points and then set the fill color for that point?
            xwb.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

Как мне теперь добраться доскажем, указать Cb1 и установить его цвет заливки на красный?

Заранее спасибо.

1 Ответ

0 голосов
/ 23 сентября 2018

Если цель действительно состоит в том, чтобы изменить диаграмму солнечных лучей Excel, то получить XML-диаграмму солнечных лучей можно будет только на очень низком уровне, анализируя XML напрямую.

Вы даже не получите диаграмму солнечных лучей, используя List <XSSFChart> charts = drawing.getCharts();.Диаграмма солнечных лучей не XSSFChart.XSSFChart имеет тип application/vnd.openxmlformats-officedocument.drawingml.chart+xml, в то время как диаграмма солнечных лучей имеет тип application/vnd.ms-office.chartex+xml.Это связано с тем, что диаграмма солнечных лучей - это расширенный тип диаграммы, который недоступен в версиях Office Open XML до 2007 года. Но эти старые версии Office Open XML - это то, для чего разработана apache poi.

Номы можем использовать по крайней мере части apache poi и должны программировать класс XSSFChartEx вместо XSSFChart нашего собственного.К сожалению, также необходим класс XSSFChartExRelation, поскольку такой класс отношений, конечно, также еще не существует.

Пример:

Источник Excel:

enter image description here

Код:

import java.io.IOException;
import java.io.OutputStream;
import java.io.FileOutputStream;
import java.io.FileInputStream;

import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.apache.poi.xssf.usermodel.XSSFSheet;
import org.apache.poi.xssf.usermodel.XSSFDrawing;

import org.apache.poi.ooxml.POIXMLDocumentPart;
import org.apache.poi.ooxml.POIXMLRelation;
import org.apache.poi.openxml4j.opc.PackagePart;

import org.apache.xmlbeans.XmlObject;
import org.apache.xmlbeans.XmlCursor;
import javax.xml.namespace.QName;

public class FormatSunBurstChart {

 private static void setDataPointColor(XmlObject series, int number, String colorHex) {
  XmlCursor cursor = series.newCursor();
  cursor.toLastChild();
  cursor.beginElement(new QName("http://schemas.microsoft.com/office/drawing/2014/chartex", "dataPt", "cx"));
  cursor.insertAttributeWithValue("idx", "" + number);
  cursor.beginElement(new QName("http://schemas.microsoft.com/office/drawing/2014/chartex", "spPr", "cx"));
  cursor.beginElement(new QName("http://schemas.openxmlformats.org/drawingml/2006/main", "solidFill", "a"));
  cursor.beginElement(new QName("http://schemas.openxmlformats.org/drawingml/2006/main", "srgbClr", "a"));
  cursor.insertAttributeWithValue("val", colorHex);

  cursor.dispose();
 }

 public static void main(String[] args) {
  try {
   XSSFWorkbook workbook = new XSSFWorkbook(new FileInputStream("ChartExample.xlsx"));
   XSSFSheet sheet = workbook.getSheetAt(0);
   System.out.println("Loaded sheet is " + sheet.getSheetName());
   XSSFDrawing drawing = sheet.getDrawingPatriarch();
   if (drawing != null) {
    for (POIXMLDocumentPart dpart : drawing.getRelations()) {
     PackagePart ppart = dpart.getPackagePart();
     if ("application/vnd.ms-office.chartex+xml".equals(ppart.getContentType())) {
      XSSFChartEx xssfChartEx = new XSSFChartEx(ppart);
      String rId = drawing.getRelationId(dpart);
      drawing.addRelation(
       rId, 
       new XSSFChartExRelation(
        "application/vnd.ms-office.chartex+xml",
        "http://schemas.microsoft.com/office/2014/relationships/chartEx",
        "/xl/charts/chartEx#.xml"),
       xssfChartEx
      );
      XmlObject series = xssfChartEx.getSeries(0);
      setDataPointColor(series, 1, "FF0000");
      setDataPointColor(series, 2, "FFFF00");
      setDataPointColor(series, 3, "00FF00");
      setDataPointColor(series, 14, "FFFF00");
      setDataPointColor(series, 16, "00FF00");
      setDataPointColor(series, 18, "00FF00");
      setDataPointColor(series, 19, "FF0000");
      setDataPointColor(series, 20, "00FF00");
System.out.println(series);
     }
    }
   }
   FileOutputStream out = new FileOutputStream("ChartExampleChanged.xlsx");
   workbook.write(out);
   workbook.close();
   out.close();
  } catch (Exception e) {
   e.printStackTrace();
  }
 }

 private static class XSSFChartEx extends POIXMLDocumentPart {

  private XmlObject chartExXmlObject;

  private XSSFChartEx(PackagePart part) throws Exception {
   super(part);
   chartExXmlObject = XmlObject.Factory.parse(part.getInputStream());
  }

  private XmlObject getChartExXmlObject() {
   return chartExXmlObject;
  }

  private XmlObject getSeries(int number) {
   XmlObject[] result = chartExXmlObject.selectPath(
    "declare namespace cx='http://schemas.microsoft.com/office/drawing/2014/chartex' " +
    ".//cx:chart/cx:plotArea/cx:plotAreaRegion/cx:series"
   );
   return result[number];
  }

  @Override
  protected void commit() throws IOException {
   PackagePart part = getPackagePart();
   OutputStream out = part.getOutputStream();
   chartExXmlObject.save(out);
   out.close();
  }
 }

 private static class XSSFChartExRelation extends POIXMLRelation {
  private XSSFChartExRelation(String type, String rel, String defaultName) {
   super(type, rel, defaultName);
  }
 }
}

Примечание: здесь используется apache poi версия 4.0.0.

Результат Excel:

enter image description here

...