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;
}

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

(0)

相关推荐

  • SQLite教程(二):C/C++接口简介

    一.概述: 在SQLite提供的C/C++接口中,其中5个APIs属于核心接口.在这篇博客中我们将主要介绍它们的用法,以及它们所涉及到的核心SQLite对象,如database_connection和prepared_statement.相比于其它数据库引擎提供的APIs,如OCI.MySQL API等,SQLite提供的接口还是非常易于理解和掌握的.     二.核心对象和接口: 1. 核心对象:     在SQLite中最主要的两个对象是,database_connection和prepar

  • C++调用迅雷接口解析XML下载功能(迅雷下载功能)

    迅雷下载库的网址:http://thunderplatform.xunlei.com 复制代码 代码如下: // FileName: Download.h#pragma once#include "lib\XLDownload.h"#include "lib\XLError.h"#include <vector> // 下载队列的大小,决定同时开启下载线程的数量const int LIMIT = 2; struct Down{    // 解析出来的下载

  • C++实现“隐藏实现,开放接口”的方案

    为什么要有接口? 接口就是一个程序与其它程序交流的窗口.就比如有一个电视机,我并不需要知道它是怎样工作的,我只要知道按电源键就可以开启电视,按节目加(+)减(-)可以切换电视频道就可以了. Java程序员都知道Java中有interface可以实现对外的接口,但C++并没有接口这样的语法,那它要好怎样实现对外提供接口呢?我们可以通过纯虚函数定义一个抽象类,专门用来声明一个类的功能. 我们完成了一个程序模块的开发,要把这个程序模块给别人用,你肯定不会把源代码给他(那别人就完全撑屋你的技术了),你会

  • C++编写非侵入式接口

    终于写到c++的非侵入式接口了,兴奋,开心,失望,解脱,-- .在搞了这么多的面向对象科普之后,本人也已经开始不耐烦,至此,不想做太多阐述. 虽然,很早就清楚怎么在c++下搞非侵入式接口,但是,整个框架代码,重构了十几次之后,才终于满意.支持给基本类型添加接口,好比int,char,const char*,double:支持泛型,好比vector,list:支持继承,基类实现的接口,表示子类也继承了对该接口的实现,而且子类也可以拒绝基类的接口,好比鸭子拒绝基类鸟类"会飞",编译时报错:

  • C++中抽象类和接口的区别介绍

    1. 如果一个类B在语法上继承(extend)了类A, 那么在语义上类B是一个类A.2. 如果一个类B在语法上实现了(implement)接口I, 那么类B遵从接口I制定的协议. 使用abstract class的根本原因在于, 人们希望通过这样的方式, 表现不同层次的抽象. 而interface的本质是一套协议. 在程序设计的发展中, 人们又发现接口可以用来表示对行为的抽象, 不过, 这只是interface的一种用法不是其本质. 理论结合实际才是最好的学习方式, 不过在这里, 我只想举一些我

  • C++ COM编程之接口背后的虚函数表

    前言 学习C++的人,肯定都知道多态机制:多态就是用父类型别的指针指向其子类的实例,然后通过父类的指针调用实际子类的成员函数.对于多态机制是如何实现的,你有没有想过呢?而COM中的接口就将这一机制运用到了极致,所以,不知道多态机制的人,是永运无法明白COM的.所以,在总结COM时,是非常有必要专门总结一下C++的多态机制是如何实现的. 多态 什么是多态?上面也说了,多态就是用父类型别的指针指向其子类的实例,然后通过父类的指针调用实际子类的成员函数.现在通过代码,让大家切身的体会一下多态: 复制代

  • C++ COM编程之什么是接口?

    什么是接口? 说到COM,就不得不说接口了:在进行COM开发的过程中,可以说,我一直都在和各种各样的接口打交道.那接口是什么?对于COM来说,接口是一个包含一个函数指针数组的内存结构,每一个数组元素包含的是一个由组件所实现的函数的地址:所以,对于COM,接口就是这样的一个内存结构,其它东西都是一些COM并不关心的实现细节. 在C++中,可以使用抽象基类来实现COM接口.由于一个COM组件可以支持任意数目的接口,因此对于组件,可以使用抽象基类的多重继承来实现它. 接口的好处 接口提供了两个不同对象

  • 浅谈java的接口和C++虚类的相同和不同之处

    C++虚类相当于java中的抽象类,与接口的不同之处是: 1.一个子类只能继承一个抽象类(虚类),但能实现多个接口 2.一个抽象类可以有构造方法,接口没有构造方法 3.一个抽象类中的方法不一定是抽象方法,即其中的方法可以有实现(有方法体),接口中的方法都是抽象方法,不能有方法体,只有声明 4.一个抽象类可以是public.private.protected.default,接口只有public 5.一个抽象类中的方法可以是public.private.protected.default,接口中的

  • C++访问Redis的mset 二进制数据接口封装方案

    需求 C++中使用hiredis客户端接口访问redis: 需要使用mset一次设置多个二进制数据 以下给出三种封装实现方案: 简单拼接方案 在redis-cli中,mset的语法是这样的: 复制代码 代码如下: /opt/colin$./redis-cli mset a 11 b 22 c 333 OK 按照这样的语法拼接后,直接使用hiredis字符串接口redisCommand传递: void msetNotBinary(redisContext *c, const vector<stri

  • 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 添加E

  • JDBC中resutset接口操作实例详解

    本文主要向大家展示JDBC接口中resutset接口的用法实例,下面我们看看具体内容. 1. ResultSet细节1 功能:封锁结果集数据 操作:如何获得(取出)结果 package com.sjx.a; import java.sql.Connection; import java.sql.DriverManager; import java.sql.ResultSet; import java.sql.Statement; import org.junit.Test; //1. next方

  • Python调用微信公众平台接口操作示例

    本文实例讲述了Python调用微信公众平台接口操作.分享给大家供大家参考,具体如下: 这里使用的是Django,其他类似 # coding=utf-8 from django.http import HttpResponse import hashlib, time, re from xml.etree import ElementTree as ET def weixin(request): token = "your token here" params = request.GET

  • Java接口继承和使用接口操作示例

    本文实例讲述了Java接口继承和使用接口操作.分享给大家供大家参考,具体如下: 一 接口的继承 1 点睛 接口支持多继承,一个接口可以有多个父接口,子接口扩展某个父接口,将会获得父接口里定义的所有抽象方法.常量. 一个接口继承多个父接口时,多个父接口排在extends关键字之后. 2 代码 interface interfaceA { int PROP_A = 5; void testA(); } interface interfaceB { int PROP_B = 6; void testB

  • PHP封装XML和JSON格式数据接口操作示例

    本文实例讲述了PHP封装XML和JSON格式数据接口操作.分享给大家供大家参考,具体如下: 使用PHP开发app后端的接口时,我们需要返回数据,最常用的就是XML和JSON格式的数据,那么样返回呢? 1.JSON json数据是最容制造的,只要使用php自带的json_encode()函数就可以简单的制作出json数据了 2.XML xml数据就要复杂一点了,让我们看一下xml数据的格式 那么怎样构造呢? 3.制作生产接口数据的类 <?php //定义生成接口数据类 class Api { /*

  • vue 调用 RESTful风格接口操作

    首先是简单的java接口代码 写了四个让前端请求的接口,以下为代码 @GetMapping("/v1/user/{username}/{password}") public Result login2(@PathVariable("username") String username, @PathVariable("password") String password){ return Result.succResult(200,username

  • vue组件暴露和.js文件暴露接口操作

    1.将同一类型的组件放在一个文件夹下 2.在此文件夹下创建一个index.js 3.在index.js中导入组件,并把他们暴露出去 1.写法一 import studentCourse1 from './studentCourse.vue' import studentInfo1 from './studentInfo.vue' export var studentCourse=studentCourse1 export var studentInfo=studentInfo1 2.写法二 ex

  • JVM系列之:JIT中的Virtual Call接口操作

    简介 上一篇文章我们讲解了Virtual Call的定义并举例分析了Virtual Call在父类和子类中的优化. JIT对类可以进行优化,那么对于interface可不可以做同样的优化么? 一起来看看吧. 最常用的接口List List应该是大家最最常用的接口了,我想这个大家应该不会反驳. public interface List<E> extends Collection<E> { 今天我们就拿List来做例子,体验一下JIT优化接口的奥秘. 还是上代码,要分析的代码如下:

  • Java接口操作(继承父类并实现多个接口)

    /* 使用接口的时候,需要注意: 1. 接口是没有静态代码块或者构造方法的. 2. 一个类的直接父类是唯一的,但是一个类可以同时实现多个接口. 格式: public class MyInterfaceImpl implements MyInterfaceA, MyInterfaceB { // 覆盖重写所有抽象方法 } 3. 如果实现类所实现的多个接口当中,存在重复的抽象方法,那么只需要覆盖重写一次即可. A接口 package cn.itcast.day10.demo02; public in

  • 在vue中使用jsonp进行跨域请求接口操作

    前言: 这里我们使用的是第三方插件jsonp. github网址:https://github.com/webmodules/jsonp 1.安装 npm install jsonp -S 2.引入 一般新建一个js文件来引入原始jsonp插件,然后对原始插件进行封装,对跨域接口参数的拼接,封装好这个jsonp文件后export出去,之后在哪里用到就再在那里import. 1.新建jsonp.js文件来封装原始jsonp插件 // 引入原始jsonp插件 import originJsonp f

随机推荐