C++通過COM接口操作PPT
一、 背景說明
在VS環(huán)境下,開發(fā)C++代碼操作PPT,支持對(duì)PPT模板的修改。包括修改文本標(biāo)簽、圖表、表格。滿足大多數(shù)軟件生成PPT報(bào)告的要求,先手工創(chuàng)建好PPT模板,在程序中修改模板數(shù)據(jù)。
二、 開發(fā)環(huán)境構(gòu)建
通過VS2012的Class Wizard創(chuàng)建PowerPoint和Excel的COM接口;由于需要操作PPT中的圖表,而圖表的數(shù)據(jù)使用Excel存儲(chǔ)的,需要修改圖表的數(shù)據(jù)就得生成Excel的COM接口。
1.1 進(jìn)入類向?qū)?br />
1.2 添加PowerPoint COM接口
1.3 添加Excel COM接口
選中所有的COM接口,生成接口文件。
三、 定義PPT文件基礎(chǔ)操作函數(shù)(頭文件略)
3.1 定義PPT應(yīng)用基礎(chǔ)對(duì)象
class CPPTObject { public: CApplication m_PPTApp; CSlides m_Slides; CSlide m_curSlide; CPresentation m_Presentaion; };
3.2 啟動(dòng)PowerPoint軟件,調(diào)用COM接口需要安裝Office
// 創(chuàng)建PPT應(yīng)用,啟動(dòng)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內(nèi)容前,先打開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文件內(nèi)容,關(guān)閉文件,退出PowerPoint程序。
// 關(guān)閉PPT,保存數(shù)據(jù)關(guān)閉。 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; }
四、 修改文本編輯框函數(shù)
// 修改文本框 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對(duì)應(yīng)于PPT中的Shape Name。這個(gè)Shape Name貌似在PowerPoint中沒有地方能看到,也沒有辦法修改。只能在調(diào)試時(shí)記錄下來。
五、 修改PPT中的圖表函數(shù)。先在PPT中定義圖表模板,通過COM接口修改圖表數(shù)據(jù)。
5.1 定義圖表數(shù)據(jù)結(jié)構(gòu)。圖表的數(shù)據(jù)都是用Excel存儲(chǔ)的。
5.1.1 定義單元格數(shù)據(jù)結(jié)構(gòu)
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()); } // 獲取浮點(diǎn)類型值 double CCellDataCom::getDoubleValue() { return atof(m_strValue.c_str()); } // 獲取單元格位置名稱 const std::string& CCellDataCom::getPos() { return m_strPos; } // 將單元格坐標(biāo)轉(zhuǎn)換名稱字符串 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 定義圖表數(shù)據(jù)結(jié)構(gòu)
// 插入一行記錄 void CChartDataCom::insertRowData(const std::list<CCellDataCom>& lstRowData) { m_lstValue.push_back(lstRowData); } // 獲取圖表數(shù)據(jù) const std::list<std::list<CCellDataCom> >& CChartDataCom::getValue()const { return m_lstValue; }
5.2 修改圖表數(shù)據(jù)函數(shù)
// 修改圖表 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; } // 修改圖表數(shù)據(jù) return ModifyChartData(shape.get_Chart(), chartData); } return false; } // 修改圖表數(shù)據(jù) bool CPPTUtil::ModifyChartData(CChart chart, const CChartDataCom& chartData) { // 激活圖表組件的excel數(shù)據(jù)表格,打開內(nèi)嵌的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; } // 獲取第一個(gè)sheet, 圖表組件的數(shù)據(jù)都在內(nèi)嵌excel的第一個(gè)sheet頁(yè)。 VARIANT vaSheetIndex; vaSheetIndex.vt = VT_I4; vaSheetIndex.lVal = 1; CWorksheet sheet = sheets.get_Item(vaSheetIndex); bool bRet = true; // 循環(huán)修改單元格的數(shù)據(jù) 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; } } // 關(guān)閉Excel CApplication0 app0 = workBook.get_Application(); app0.Quit(); Sleep(2000); return bRet; } // 修改單元格數(shù)據(jù) 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; }
六、 合并多個(gè)PPT文件函數(shù)
// 合并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); // 循環(huán)打開合并文件插入PPT頁(yè)面 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頁(yè)簽 CSlides outSlides = outPresention.get_Slides(); outSlides.InsertFromFile(CString(iterMergeFile->c_str()), outSlides.get_Count(), 1, pageNum); } outPresention.Save(); outPresention.Close(); pptApp.Quit(); return true; }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
tinyxml 常用的C++ XML解析器非常優(yōu)秀
讀取和設(shè)置xml配置文件是最常用的操作,試用了幾個(gè)C++的XML解析器,個(gè)人感覺TinyXML是使用起來最舒服的,因?yàn)樗腁PI接口和Java的十分類似,面向?qū)ο笮院芎?/div> 2012-11-11C++如何調(diào)用opencv完成運(yùn)動(dòng)目標(biāo)捕捉詳解
OpenCV作為機(jī)器視覺開源庫(kù),使用起來非常不錯(cuò),這篇文章主要給大家介紹了關(guān)于C++如何調(diào)用opencv完成運(yùn)動(dòng)目標(biāo)捕捉的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-05-05C語(yǔ)言實(shí)現(xiàn)linux網(wǎng)卡檢測(cè)精簡(jiǎn)版
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)linux網(wǎng)卡檢測(cè)的精簡(jiǎn)版,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06C語(yǔ)言超細(xì)致講解循環(huán)語(yǔ)句
我們說到當(dāng)滿足特定條件時(shí),就會(huì)執(zhí)行if語(yǔ)句或者switch語(yǔ)句后面的語(yǔ)句,否則不執(zhí)行,但是這只能執(zhí)行一次,在日常生活中,有些事情是需要重復(fù)去做的,C語(yǔ)句就為此引入了循環(huán)語(yǔ)句。所以今天繼續(xù)為大家分享C語(yǔ)言循環(huán)家族2022-05-05C語(yǔ)言實(shí)現(xiàn)文件內(nèi)容按行隨機(jī)排列的算法示例
這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)文件內(nèi)容按行隨機(jī)排列的算法,涉及C語(yǔ)言字符串、數(shù)組遍歷與隨機(jī)數(shù)相關(guān)算法實(shí)現(xiàn)技巧,需要的朋友可以參考下2017-09-09Ubuntu18.04下QT開發(fā)Android無(wú)法連接設(shè)備問題解決實(shí)現(xiàn)
本文主要介紹了Ubuntu18.04下QT開發(fā)Android無(wú)法連接設(shè)備問題解決實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-06-06最新評(píng)論