C++通过COM接口操作PPT

 更新时间:2017年05月15日 14:09:33   作者:DoubleLi  
这篇文章主要为大家详细介绍了C++通过COM接口操作PPT的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

一、 背景说明

在VS环境下,开发C++代码操作PPT,支持对PPT模板的修改。包括修改文本标签、图表、表格。满足大多数软件生成PPT报告的要求,先手工创建好PPT模板,在程序中修改模板数据。

二、 开发环境构建

通过VS2012的Class Wizard创建PowerPoint和Excel的COM接口;由于需要操作PPT中的图表,而图表的数据使用Excel存储的,需要修改图表的数据就得生成Excel的COM接口。

1.1 进入类向导

1.2 添加PowerPoint COM接口

1.3 添加Excel COM接口

选中所有的COM接口,生成接口文件。

三、 定义PPT文件基础操作函数(头文件略)

3.1 定义PPT应用基础对象

class CPPTObject
{
public:
  CApplication m_PPTApp;
  CSlides m_Slides;
  CSlide m_curSlide;
  CPresentation m_Presentaion;
};

3.2 启动PowerPoint软件,调用COM接口需要安装Office

// 创建PPT应用,启动powerpoint程序。
bool CPPTUtil::CreatePPTApplication()
{
  COleException exception;

  LPCSTR str = "Powerpoint.Application";
  if(!m_pPPTObject->m_PPTApp.CreateDispatch(str, &exception))
  {
    AfxMessageBox(exception.m_sc, MB_SETFOREGROUND);
    return false;
  }

  m_pPPTObject->m_PPTApp.put_Visible(true);
  return true;
}

3.3 打开PPT模板文件。修改PPT内容前,先打开PPT。

// 打开模板ppt。
bool CPPTUtil::OpenPPT(const std::string& pptPath)
{
  CPresentations presentations = m_pPPTObject->m_PPTApp.get_Presentations();
  m_pPPTObject->m_Presentaion = presentations.Open(CString(pptPath.c_str()), 0, 0, 1);

  m_pPPTObject->m_Slides = m_pPPTObject->m_Presentaion.get_Slides();

  return true;
}

3.4 保存PPT文件内容,关闭文件,退出PowerPoint程序。

// 关闭PPT,保存数据关闭。
bool CPPTUtil::ClosePPT()
{
  m_pPPTObject->m_Presentaion.Save();
  m_pPPTObject->m_Presentaion.Close();
  m_pPPTObject->m_PPTApp.Quit();
  return true;
}

3.5 选中具体的PPT幻灯片。

// 选中PPT指定索引的幻灯片。
bool CPPTUtil::SelectSlide(long slideIndex)
{
  if (slideIndex > m_pPPTObject->m_Slides.get_Count())
  {
    return false;
  }

  m_pPPTObject->m_curSlide = m_pPPTObject->m_Slides.Range(COleVariant(slideIndex));

  return true;
}

四、 修改文本编辑框函数

// 修改文本框
bool CPPTUtil::ModifyTextBox(const std::string& boxName, const std::string& strValue)
{
  CShapes shapes = m_pPPTObject->m_curSlide.get_Shapes();
  for(long i = 1; i <= shapes.get_Count(); ++i)
  {
    CShape shape(shapes.Item(COleVariant(i)));
    CString name = shape.get_Name();

    if(shape.get_Type() == (long)Office::msoTextBox
      && name.Compare(CString(boxName.c_str())) == 0) 
    {
      CTextFrame textFrame = shape.get_TextFrame();
      CTextRange textRange = textFrame.get_TextRange();
      CString txt = textRange.get_Text();

      textRange.put_Text(strValue.c_str());
    }
  }

  return true;
}

boxName对应于PPT中的Shape Name。这个Shape Name貌似在PowerPoint中没有地方能看到,也没有办法修改。只能在调试时记录下来。

五、 修改PPT中的图表函数。先在PPT中定义图表模板,通过COM接口修改图表数据。

5.1 定义图表数据结构。图表的数据都是用Excel存储的。

5.1.1 定义单元格数据结构

CCellDataCom::CCellDataCom(const CellValueType valueType, const std::string& strValue, 
      const int iRow, const int iCol)
{
  m_ValueType = valueType;
  m_strValue = strValue;

  m_strPos = indexToString(iRow, iCol);
}

// 获取单元格值类型
CellValueType CCellDataCom::getValueType()
{
  return m_ValueType;
}

// 获取字符串类型值
const std::string& CCellDataCom::getStringValue()
{
  return m_strValue;
}

// 获取整型值
long CCellDataCom::getLongValue()
{
  return atol(m_strValue.c_str());
}

// 获取浮点类型值
double CCellDataCom::getDoubleValue()
{
  return atof(m_strValue.c_str());
}

// 获取单元格位置名称
const std::string& CCellDataCom::getPos()
{
  return m_strPos;
}

// 将单元格坐标转换名称字符串
CString CCellDataCom::indexToString( int row, int col )  
{  
  CString strResult; 
  if( col > 26 )  
  {  
    strResult.Format(_T("%c%c%d"),'A' + (col-1)/26-1,'A' + (col-1)%26,row); 
  }  
  else  
  {  
  strResult.Format(_T("%c%d"), 'A' + (col-1)%26,row); 
  } 

  return strResult; 
}  

5.1.2   定义图表数据结构

// 插入一行记录
void CChartDataCom::insertRowData(const std::list<CCellDataCom>& lstRowData)
{
  m_lstValue.push_back(lstRowData);
}

// 获取图表数据
const std::list<std::list<CCellDataCom> >& CChartDataCom::getValue()const
{
  return m_lstValue;
}

5.2 修改图表数据函数

// 修改图表

bool CPPTUtil::ModifyChart(const std::string& chartName, const CChartDataCom& chartData)
{
  CShapes shapes = m_pPPTObject->m_curSlide.get_Shapes();
  for(long i = 1; i <= shapes.get_Count(); ++i)
  {
    CShape shape(shapes.Item(COleVariant(i)));
    if(shape.get_Type() != (long)Office::msoChart
      || chartName != std::string(shape.get_Name().GetBuffer()))
    {
      continue;
    }

    // 修改图表数据
    return ModifyChartData(shape.get_Chart(), chartData);
  }

  return false;
}

// 修改图表数据
bool CPPTUtil::ModifyChartData(CChart chart, const CChartDataCom& chartData)
{
  // 激活图表组件的excel数据表格,打开内嵌的excel.
  CChartData chartDataModel = chart.get_ChartData();
  chartDataModel.Activate();

  CWorkbook workBook = chartDataModel.get_Workbook();
  CWorksheets sheets = workBook.get_Worksheets();
  if(sheets.get_Count() == 0)
  {
    return false;
  }

  // 获取第一个sheet, 图表组件的数据都在内嵌excel的第一个sheet页。
  VARIANT vaSheetIndex;
  vaSheetIndex.vt = VT_I4;
  vaSheetIndex.lVal = 1;
  CWorksheet sheet = sheets.get_Item(vaSheetIndex);

  bool bRet = true;

  // 循环修改单元格的数据
  const std::list<std::list<CCellDataCom> >& lstValue = chartData.getValue();
  std::list<std::list<CCellDataCom> >::const_iterator iterAllData = lstValue.begin();
  for(; iterAllData != lstValue.end(); ++iterAllData)
  {
    std::list<CCellDataCom>::const_iterator iterRowData = iterAllData->begin();
    for(; iterRowData != iterAllData->end(); ++iterRowData)
    {
      bRet = ModifyCellData(sheet, *iterRowData);
      if(bRet == false)
      {
        break;
      }
    }

    if(bRet == false)
    {
      break;
    }
  }

  // 关闭Excel
  CApplication0 app0 = workBook.get_Application();
  app0.Quit();
  Sleep(2000);

  return bRet;
}

// 修改单元格数据
bool CPPTUtil::ModifyCellData(CWorksheet sheet, CCellDataCom cellData)
{
  const std::string& cellPos = cellData.getPos();
  CRange range = sheet.get_Range(COleVariant(cellPos.c_str()), COleVariant(cellPos.c_str()));

  COleVariant* pOleVar = NULL;
  if(cellData.getValueType() == CELL_STRING_TYPE)
  {
    pOleVar = new COleVariant(CString(cellData.getStringValue().c_str()));
  }
  else if(cellData.getValueType() == CELL_LONG_TYPE)
  {
    pOleVar = new COleVariant(cellData.getLongValue());
  }
  else if(cellData.getValueType() == CELL_DOUBLE_TYPE)
  {
    pOleVar = new COleVariant(cellData.getDoubleValue());
  }
  else
  {
    return false;
  }

  range.put_Value2(*pOleVar);
  delete pOleVar;

  return true;
}

六、 合并多个PPT文件函数

// 合并PPT
bool CPPTUtil::MergePPT(const std::string& outputPPTPath, const std::list<std::string>& lstMergePPTPath)
{
  CApplication pptApp;
  COleException exception;

  // 打开PowerPoint程序
  LPCSTR str = "Powerpoint.Application";
  if(!pptApp.CreateDispatch(str, &exception))
  {
    AfxMessageBox(exception.m_sc, MB_SETFOREGROUND);
    return false;
  }

  pptApp.put_Visible(true);

  // 打开输出文件
  CPresentations presentations = pptApp.get_Presentations();
  CPresentation outPresention = presentations.Open(CString(outputPPTPath.c_str()), 0, 0, 1);

  // 循环打开合并文件插入PPT页面
  std::list<std::string>::const_iterator iterMergeFile = lstMergePPTPath.begin();
  for(; iterMergeFile != lstMergePPTPath.end(); ++iterMergeFile)
  {
    CPresentation mergePresention = presentations.Open(CString(iterMergeFile->c_str()), 0, 0, 1);
    CSlides mergeSlides = mergePresention.get_Slides();
    int pageNum = mergeSlides.get_Count();
    mergePresention.Close();

    // 合并PPT页签
    CSlides outSlides = outPresention.get_Slides();
    outSlides.InsertFromFile(CString(iterMergeFile->c_str()), outSlides.get_Count(), 1, pageNum);
  }

  outPresention.Save();
  outPresention.Close();
  pptApp.Quit();

  return true;
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • tinyxml 常用的C++ XML解析器非常优秀

    tinyxml 常用的C++ XML解析器非常优秀

    读取和设置xml配置文件是最常用的操作,试用了几个C++的XML解析器,个人感觉TinyXML是使用起来最舒服的,因为它的API接口和Java的十分类似,面向对象性很好
    2012-11-11
  • C++如何调用opencv完成运动目标捕捉详解

    C++如何调用opencv完成运动目标捕捉详解

    OpenCV作为机器视觉开源库,使用起来非常不错,这篇文章主要给大家介绍了关于C++如何调用opencv完成运动目标捕捉的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-05-05
  • C语言实现linux网卡检测精简版

    C语言实现linux网卡检测精简版

    这篇文章主要为大家详细介绍了C语言实现linux网卡检测的精简版,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-06-06
  • C语言超细致讲解循环语句

    C语言超细致讲解循环语句

    我们说到当满足特定条件时,就会执行if语句或者switch语句后面的语句,否则不执行,但是这只能执行一次,在日常生活中,有些事情是需要重复去做的,C语句就为此引入了循环语句。所以今天继续为大家分享C语言循环家族
    2022-05-05
  • 详解C语言中双指针算法的使用

    详解C语言中双指针算法的使用

    双指针,指的是在遍历对象的过程中,不是普通的使用单个指针进行访问,而是使用两个相同方向(快慢指针)或者相反方向(对撞指针)的指针进行扫描,从而达到相应的目的。本文将通过示例带大家深入了解双指针算法的使用
    2022-08-08
  • C++中常量与指针的示例详解

    C++中常量与指针的示例详解

    在C++学习使用过程中,每个人都不可避免地使用指针,而且都或多或少的接触过常量指针或指针常量,但是对这两个的概念还是很容易搞糊涂的,所以这篇文章主要给大家介绍了关于C++中常量与指针的相关资料,需要的朋友可以参考下
    2021-06-06
  • C++中delete指针后最好将其置空的操作方法

    C++中delete指针后最好将其置空的操作方法

    C++编程中,当你使用delete运算符释放指针所指向的内存后,通常将该指针置空,如果一个指针在被删除后没有置空,而你在代码的其他部分再次尝试删除同一个指针,可能会导致程序崩溃或产生未定义行为,本文介绍C++中delete指针后最好将其置空的操作方法,感兴趣的朋友一起看看吧
    2024-06-06
  • C语言递归实现扫雷游戏

    C语言递归实现扫雷游戏

    这篇文章主要为大家详细介绍了C语言递归实现扫雷游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-07-07
  • C语言实现文件内容按行随机排列的算法示例

    C语言实现文件内容按行随机排列的算法示例

    这篇文章主要介绍了C语言实现文件内容按行随机排列的算法,涉及C语言字符串、数组遍历与随机数相关算法实现技巧,需要的朋友可以参考下
    2017-09-09
  • Ubuntu18.04下QT开发Android无法连接设备问题解决实现

    Ubuntu18.04下QT开发Android无法连接设备问题解决实现

    本文主要介绍了Ubuntu18.04下QT开发Android无法连接设备问题解决实现,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-06-06

最新评论