使用 PHP 在演示文稿中管理图表工作簿

概述

本文解释了如何在 Aspose.Slides 中使用图表工作簿。它展示了如何通过工作簿流读取和写入图表数据,将工作簿单元格用作图表数据标签,访问工作表集合,以及为图表值指定数据源类型。

它还包括将外部工作簿用作图表数据源的操作示例。示例演示了如何创建并分配外部工作簿,检索链接到图表的外部工作簿路径,以及在工作簿可用时编辑图表数据。

从工作簿读取和写入图表数据

Aspose.Slides 提供了 readWorkbookStreamwriteWorkbookStream 方法,允许您读取和写入图表数据工作簿(其中包含使用 Aspose.Cells 编辑的图表数据)。注意,图表数据必须以相同方式组织,或具有与源相似的结构。

下面的 PHP 代码演示了一个示例操作:

  $pres = new Presentation("chart.pptx");
  try {
    $chart = $pres->getSlides()->get_Item(0)->getShapes()->get_Item(0);
    $data = $chart->getChartData();
    $stream = $data->readWorkbookStream();
    $data->getSeries()->clear();
    $data->getCategories()->clear();
    $data->writeWorkbookStream($stream);
  } finally {
    if (!java_is_null($pres)) {
      $pres->dispose();
    }
  }

将工作簿单元格设为图表数据标签

  1. 创建一个 Presentation 类的实例。
  2. 通过索引获取幻灯片的引用。
  3. 添加一个带有数据的气泡图。
  4. 访问图表系列。
  5. 将工作簿单元格设为数据标签。
  6. 保存演示文稿。

下面的 PHP 代码展示了如何将工作簿单元格设为图表数据标签:

  $lbl0 = "Label 0 cell value";
  $lbl1 = "Label 1 cell value";
  $lbl2 = "Label 2 cell value";
  # 实例化表示演示文稿文件的 Presentation 类
  $pres = new Presentation("chart2.pptx");
  try {
    $slide = $pres->getSlides()->get_Item(0);
    $chart = $slide->getShapes()->addChart(ChartType::Bubble, 50, 50, 600, 400, true);
    $series = $chart->getChartData()->getSeries();
    $dataLabelCollection = $series->get_Item(0)->getLabels();
    $dataLabelCollection->getDefaultDataLabelFormat()->setShowLabelValueFromCell(true);
    $wb = $chart->getChartData()->getChartDataWorkbook();
    $dataLabelCollection->get_Item(0)->setValueFromCell($wb->getCell(0, "A10", $lbl0));
    $dataLabelCollection->get_Item(1)->setValueFromCell($wb->getCell(0, "A11", $lbl1));
    $dataLabelCollection->get_Item(2)->setValueFromCell($wb->getCell(0, "A12", $lbl2));
    $pres->save("resultchart.pptx", SaveFormat::Pptx);
  } finally {
    if (!java_is_null($pres)) {
      $pres->dispose();
    }
  }

管理工作表

下面的 PHP 代码演示了使用 ChartDataWorkbook::getWorksheets 方法访问工作表集合的操作:

  $pres = new Presentation();
  try {
    $chart = $pres->getSlides()->get_Item(0)->getShapes()->addChart(ChartType::Pie, 50, 50, 400, 500);
    $wb = $chart->getChartData()->getChartDataWorkbook();
    for($i = 0; $i < java_values($wb->getWorksheets()->size()) ; $i++) {
      echo($wb->getWorksheets()->get_Item($i)->getName());
    }
  } finally {
    if (!java_is_null($pres)) {
      $pres->dispose();
    }
  }

指定数据源类型

下面的 PHP 代码展示了如何为数据源指定类型:

  $pres = new Presentation();
  try {
    $chart = $pres->getSlides()->get_Item(0)->getShapes()->addChart(ChartType::Column3D, 50, 50, 600, 400, true);
    $val = $chart->getChartData()->getSeries()->get_Item(0)->getName();
    $val->setDataSourceType(DataSourceType::StringLiterals);
    $val->setData("LiteralString");
    $val = $chart->getChartData()->getSeries()->get_Item(1)->getName();
    $val->setData($chart->getChartData()->getChartDataWorkbook()->getCell(0, "B1", "NewCell"));
    $pres->save("pres.pptx", SaveFormat::Pptx);
  } finally {
    if (!java_is_null($pres)) {
      $pres->dispose();
    }
  }

检测不受支持的嵌入式工作簿格式

Aspose.Slides 不支持某些图表中可以嵌入的 Excel 二进制工作簿(.xlsb)格式。您可以在 ChartData 上使用 getEmbeddedWorkbookType 方法,并结合 WorkbookType 枚举来检测不受支持的格式并跳过这些图表。

$presentation = new Presentation("sample.pptx");
try {
  $slide = $presentation->getSlides()->get_Item(0);
  $shapes = $slide->getShapes();

  for ($shapeIndex = 0; $shapeIndex < java_values($shapes->size()); $shapeIndex++) {
    $shape = $shapes->get_Item($shapeIndex);

    if (!java_instanceof($shape, new JavaClass("com.aspose.slides.IChart"))) {
      continue;
    }

    $chart = $shape;
    $chartData = $chart->getChartData();

    if (java_values($chartData->getDataSourceType()) == ChartDataSourceType::InternalWorkbook &&
        java_values($chartData->getEmbeddedWorkbookType()) == WorkbookType::WorkbookBinaryMacro) {
      # 嵌入式工作簿为 .xlsb 格式,不受支持。
      continue;
    }

    # 在此读取或修改图表工作簿数据。
  }
} finally {
  $presentation->dispose();
}

外部工作簿

Aspose.Slides 支持将外部工作簿用作图表的数据源。

创建外部工作簿

使用 readWorkbookStreamsetExternalWorkbook 方法,您可以从头创建外部工作簿,或者将内部工作簿设为外部工作簿。

下面的 PHP 代码演示了外部工作簿的创建过程:

  $pres = new Presentation();
  $Array = new java_class("java.lang.reflect.Array");
  try {
    $workbookPath = "externalWorkbook1.xlsx";
    $chart = $pres->getSlides()->get_Item(0)->getShapes()->addChart(ChartType::Pie, 50, 50, 400, 600);
    $fileStream = new Java("java.io.FileOutputStream", $workbookPath);
    $Array = new java_class("java.lang.reflect.Array");
    try {
      $workbookData = $chart->getChartData()->readWorkbookStream();
      $fileStream->write($workbookData, 0, $Array->getLength($workbookData));
    } finally {
      if (!java_is_null($fileStream)) {
        $fileStream->close();
      }
    }
    $chart->getChartData()->setExternalWorkbook($workbookPath);
    $pres->save("externalWorkbook.pptx", SaveFormat::Pptx);
  } catch (JavaException $e) {
  } finally {
    if (!java_is_null($pres)) {
      $pres->dispose();
    }
  }

设置外部工作簿

使用 setExternalWorkbook 方法,您可以将外部工作簿分配给图表作为其数据源。该方法还可用于更新外部工作簿的路径(如果工作簿已被移动)。

虽然无法编辑存放在远程位置或资源中的工作簿数据,但仍然可以将这些工作簿用作外部数据源。如果提供了外部工作簿的相对路径,系统会自动将其转换为完整路径。

下面的 PHP 代码展示了如何设置外部工作簿:

  # 创建 Presentation 类的实例
  $pres = new Presentation("chart.pptx");
  try {
    $chart = $pres->getSlides()->get_Item(0)->getShapes()->addChart(ChartType::Pie, 50, 50, 400, 600, false);
    $chartData = $chart->getChartData();
    $chartData->setExternalWorkbook("externalWorkbook.xlsx");
    $chartData->getSeries()->add($chartData->getChartDataWorkbook()->getCell(0, "B1"), ChartType::Pie);
    $chartData->getSeries()->get_Item(0)->getDataPoints()->addDataPointForPieSeries($chartData->getChartDataWorkbook()->getCell(0, "B2"));
    $chartData->getSeries()->get_Item(0)->getDataPoints()->addDataPointForPieSeries($chartData->getChartDataWorkbook()->getCell(0, "B3"));
    $chartData->getSeries()->get_Item(0)->getDataPoints()->addDataPointForPieSeries($chartData->getChartDataWorkbook()->getCell(0, "B4"));
    $chartData->getCategories()->add($chartData->getChartDataWorkbook()->getCell(0, "A2"));
    $chartData->getCategories()->add($chartData->getChartDataWorkbook()->getCell(0, "A3"));
    $chartData->getCategories()->add($chartData->getChartDataWorkbook()->getCell(0, "A4"));
    $pres->save("Presentation_with_externalWorkbook.pptx", SaveFormat::Pptx);
  } finally {
    if (!java_is_null($pres)) {
      $pres->dispose();
    }
  }

ChartData 参数(位于 setExternalWorkbook 方法下)用于指定是否加载 Excel 工作簿。

  • ChartData 值设为 false 时,仅更新工作簿路径——图表数据不会从目标工作簿加载或更新。当目标工作簿不存在或不可用时,可使用此设置。
  • ChartData 值设为 true 时,图表数据会从目标工作簿更新。
  # 创建 Presentation 类的实例
  $pres = new Presentation("chart.pptx");
  try {
    $chart = $pres->getSlides()->get_Item(0)->getShapes()->addChart(ChartType::Pie, 50, 50, 400, 600, true);
    $chartData = $chart->getChartData();
    $chartData->setExternalWorkbook("http://path/doesnt/exists", false);
    $pres->save("Presentation_with_externalWorkbookWithUpdateChartData.pptx", SaveFormat::Pptx);
  } finally {
    if (!java_is_null($pres)) {
      $pres->dispose();
    }
  }

获取图表的外部数据源工作簿路径

  1. 创建一个 Presentation 类的实例。
  2. 通过索引获取幻灯片的引用。
  3. 创建图表形状的对象。
  4. 创建表示图表数据源的源 (ChartDataSourceType) 类型对象。
  5. 根据源类型与外部工作簿数据源类型相同的情况指定相应条件。

下面的 PHP 代码演示了该操作:

  # 创建 Presentation 类的实例
  $pres = new Presentation("chart.pptx");
  try {
    $slide = $pres->getSlides()->get_Item(1);
    $chart = $slide->getShapes()->get_Item(0);
    $sourceType = $chart->getChartData()->getDataSourceType();
    if ($sourceType == ChartDataSourceType::ExternalWorkbook) {
      $path = $chart->getChartData()->getExternalWorkbookPath();
    }
    # 保存演示文稿
    $pres->save("result.pptx", SaveFormat::Pptx);
  } finally {
    if (!java_is_null($pres)) {
      $pres->dispose();
    }
  }

编辑图表数据

您可以像编辑内部工作簿一样编辑外部工作簿中的数据。当外部工作簿无法加载时,会抛出异常。

下面的 PHP 代码实现了上述过程:

  # 创建 Presentation 类的实例
  $pres = new Presentation("chart.pptx");
  try {
    $chart = $pres->getSlides()->get_Item(0)->getShapes()->get_Item(0);
    $chartData = $chart->getChartData();
    $chartData->getSeries()->get_Item(0)->getDataPoints()->get_Item(0)->getValue()->getAsCell()->setValue(100);
    $pres->save("presentation_out.pptx", SaveFormat::Pptx);
  } finally {
    if (!java_is_null($pres)) {
      $pres->dispose();
    }
  }

常见问题解答

我可以判断特定图表是链接到外部工作簿还是嵌入式工作簿吗?

可以。图表拥有 data source typepath to an external workbook;如果数据源是外部工作簿,您可以读取完整路径以确认使用了外部文件。

是否支持相对路径的外部工作簿,且它们是如何存储的?

支持。如果指定相对路径,系统会自动将其转换为绝对路径。这对项目的可移植性很方便;但请注意,演示文稿会在 PPTX 文件中存储绝对路径。

可以使用位于网络资源/共享上的工作簿吗?

可以,这类工作簿可以作为外部数据源使用。不过,Aspose.Slides 不支持直接编辑远程工作簿——只能将其用作数据源。

保存演示文稿时,Aspose.Slides 会覆盖外部 XLSX 吗?

不会。演示文稿仅存储对外部文件的 link to the external file,并在读取数据时使用该链接。保存演示文稿时不会修改外部文件本身。

如果外部文件受密码保护该怎么办?

Aspose.Slides 在链接时不接受密码。常见做法是事先移除保护或准备一个已解密的副本(例如使用 Aspose.Cells),然后链接到该副本。

多个图表可以引用同一个外部工作簿吗?

可以。每个图表都会存储自己的链接。如果它们指向同一个文件,更新该文件后,下次加载数据时所有图表都会反映更新。