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;
}
C++通过COM接口操作PPT的更多相关文章
- JAVA通过COM接口操作PPT
一. 背景说明 在Eclipse环境下,开发JAVA代码操作PPT,支持对PPT模板的修改.包括修改文本标签.图表.表格.满足大多数软件生成PPT报告的要求,即先收工创建好模板,在程序中修改模板数据. ...
- Node.js 中MongoDB的基本接口操作
Node.js 中MongoDB的基本接口操作 连接数据库 安装mongodb模块 导入mongodb模块 调用connect方法 文档的增删改查操作 插入文档 方法: db.collection(& ...
- java poi 操作ppt
java poi 操作ppt 可以参考: https://www.w3cschool.cn/apache_poi_ppt/apache_poi_ppt_installation.html http:/ ...
- /sys/class/gpio 文件接口操作IO端口(s3c2440)
http://blog.csdn.net/mirkerson/article/details/8464231 在嵌入式设备中对GPIO的操作是最基本的操作.一般的做法是写一个单独驱动程序,网上大多数的 ...
- winfrom 操作PPT
///winfrom 操作PPT using System; using System.Collections.Generic; using System.Linq; using System.Tex ...
- Jacob操作ppt
前几天使用Apache 的POI操作ppt,后来发现转成的图片出现乱码,而且处理了之后,还会有遗留 因此决定换一种处理方式 Jacob 是 JAVA-COM Bridge的缩写,是一个中间件,能够提供 ...
- 接口操作XML
接口操作XML 以下代码旨在 脱离TXMLDocument 操作 xml. unit Unit3; interface uses Windows, Messages, SysUtils, Varian ...
- poi 操作 PPT,针对 PPTX--图表篇
poi 操作 PPT,针对 PPTX--图表篇 目录 poi 操作 PPT,针对 PPTX--图表篇 1.读取 PPT 模板 2.替换标题 4.替换图表数据 接下来对 ppt 内的图表进行操作,替换图 ...
- ASP.NET MVC对WebAPI接口操作(添加,更新和删除)
昨天<怎样操作WebAPI接口(显示数据)>http://www.cnblogs.com/insus/p/5670401.html 既有使用jQuery,也有使作HttpClient来从数 ...
随机推荐
- win7无法启用网络发现
1. Windows+R 2. 指令services.msc 3.找到以下服务,设为自动并开启服务 Function Discovery Resource Publication SSDP Disco ...
- 【echarts3】--1 简单入门
echarts3 相信大家都了解吧,是百度研发的 ECharts 特性介绍 ECharts,一个纯 Javascript 的图表库,可以流畅的运行在 PC 和移动设备上,兼容当前绝大部分浏览器(IE8 ...
- Python中高阶函数sorted()用法
在Python中,有内置的排序方法:sorted(iterable, key, reverse). Sorted()函数也是一个高阶函数,它还可以接收一个key函数来实现自定义的排序.key指定的函数 ...
- 批量修改git仓库地址脚本
前言 公司的代码都存放在自己搭建的gitlab上面.之前由于老板升级gitlab.导致下面有个叫做"api"的groups无法访问.通过无所不能的谷歌才知道.在gitlab在某 ...
- Oracle数据库中SCOTT用户下的默认表
①EMP(雇员表): ②DEPT(部门表): ③BONUS(奖金表): ④SALGRADE(工资等级表):
- ubuntu16.04 python3 安装selenium及环境配置
环境 ubuntu16.04 python3 安装selenium sudo pip3 install seleium 默认安装完是支持firefox,但是更新得太慢对于较新的firefox已经不支持 ...
- Ubuntu17.10下启动Rancher
1.安装Docker: 获取最新的docker安装包 wget -qO- https://get.docker.com/ | sh 2.启动docker后台服务: service docker sta ...
- 企业内部DNS跨国配置案例
背景介绍:总公司与北京分公司均由总公司进行统一管理.总公司的主从DNS担任解析总公司服务器与北京分公司的服务器解析任务.总公司DNS委派其他两个公司管理自己域下的服务器解析任务.要求任何一个节点都能解 ...
- java I/O框架 (三)基本流
概述 基本流有字节输入输出流(InputStream,OutputStream),和字符输入输出流(Reader,Writer),它们都是抽象类,作为Java IO API中其他所有流的父类存在. 我 ...
- javascript-深入理解&&和||
先从两个问题看起: 第一个问题 为什么 a && b 返回的是true,b && a 返回的是6 var user = 6; var both = true; cons ...