mirror of
https://github.com/ONLYOFFICE/core.git
synced 2026-02-10 18:05:41 +08:00
447 lines
17 KiB
C++
447 lines
17 KiB
C++
/*
|
|
* (c) Copyright Ascensio System SIA 2010-2023
|
|
*
|
|
* This program is a free software product. You can redistribute it and/or
|
|
* modify it under the terms of the GNU Affero General Public License (AGPL)
|
|
* version 3 as published by the Free Software Foundation. In accordance with
|
|
* Section 7(a) of the GNU AGPL its Section 15 shall be amended to the effect
|
|
* that Ascensio System SIA expressly excludes the warranty of non-infringement
|
|
* of any third-party rights.
|
|
*
|
|
* This program is distributed WITHOUT ANY WARRANTY; without even the implied
|
|
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. For
|
|
* details, see the GNU AGPL at: http://www.gnu.org/licenses/agpl-3.0.html
|
|
*
|
|
* You can contact Ascensio System SIA at 20A-6 Ernesta Birznieka-Upish
|
|
* street, Riga, Latvia, EU, LV-1050.
|
|
*
|
|
* The interactive user interfaces in modified source and object code versions
|
|
* of the Program must display Appropriate Legal Notices, as required under
|
|
* Section 5 of the GNU AGPL version 3.
|
|
*
|
|
* Pursuant to Section 7(b) of the License you must retain the original Product
|
|
* logo when distributing the program. Pursuant to Section 7(e) we decline to
|
|
* grant you any rights under trademark law for use of our trademarks.
|
|
*
|
|
* All the Product's GUI elements, including illustrations and icon sets, as
|
|
* well as technical writing content are licensed under the terms of the
|
|
* Creative Commons Attribution-ShareAlike 4.0 International. See the License
|
|
* terms at http://creativecommons.org/licenses/by-sa/4.0/legalcode
|
|
*
|
|
*/
|
|
|
|
#include "Drawing.h"
|
|
#include "Pos.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Binary/CFStreamCacheWriter.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/ChartSheetSubstream.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_unions/OBJECTS.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_unions/CHARTFOMATS.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_unions/AXISPARENT.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_unions/AXES.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_unions/ATTACHEDLABEL.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_unions/AI.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_records/MsoDrawing.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_records/Chart.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_records/AxisParent.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_records/Pos.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_records/Text.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_records/ObjectLink.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_records/BRAI.h"
|
|
#include "../../../MsBinaryFile/XlsFile/Format/Logic/Biff_records/SeriesText.h"
|
|
#include "../../PPTXFormat/Logic/Shape.h"
|
|
#include "../Chart/Chart.h"
|
|
|
|
namespace OOX
|
|
{
|
|
using namespace Drawing;
|
|
|
|
namespace Spreadsheet
|
|
{
|
|
CDrawingWorksheet::CDrawingWorksheet()
|
|
{
|
|
}
|
|
CDrawingWorksheet::~CDrawingWorksheet()
|
|
{
|
|
}
|
|
void CDrawingWorksheet::fromXML(XmlUtils::CXmlNode& node)
|
|
{
|
|
}
|
|
std::wstring CDrawingWorksheet::toXML() const
|
|
{
|
|
return (L"");
|
|
}
|
|
void CDrawingWorksheet::toXML(NSStringUtils::CStringBuilder& writer) const
|
|
{
|
|
if(m_oId.IsInit())
|
|
{
|
|
writer.WriteString(L"<drawing r:id=\"");
|
|
writer.WriteString(m_oId->ToString());
|
|
writer.WriteString(L"\"/>");
|
|
}
|
|
|
|
}
|
|
void CDrawingWorksheet::fromXML(XmlUtils::CXmlLiteReader& oReader)
|
|
{
|
|
ReadAttributes( oReader );
|
|
|
|
if ( !oReader.IsEmptyNode() )
|
|
oReader.ReadTillEnd();
|
|
}
|
|
void CDrawingWorksheet::fromBin(XLS::BaseObjectPtr& obj)
|
|
{
|
|
ReadAttributes(obj);
|
|
}
|
|
XLS::BaseObjectPtr CDrawingWorksheet::toBin()
|
|
{
|
|
auto castedPtr(new XLSB::Drawing);
|
|
XLS::BaseObjectPtr ptr(castedPtr);
|
|
if(m_oId.IsInit())
|
|
{
|
|
if(!m_oId->GetValue().empty())
|
|
castedPtr->stRelId.value = m_oId->GetValue();
|
|
}
|
|
return ptr;
|
|
}
|
|
void CDrawingWorksheet::toBin(XLS::StreamCacheWriterPtr& writer)
|
|
{
|
|
auto record = writer->getNextRecord(XLSB::rt_Drawing);
|
|
XLSB::RelID stRelId;
|
|
if(m_oId.IsInit())
|
|
stRelId = m_oId->GetValue();
|
|
else
|
|
stRelId.value.setSize(0xFFFFFFFF);
|
|
*record << stRelId;
|
|
writer->storeNextRecord(record);
|
|
}
|
|
EElementType CDrawingWorksheet::getType () const
|
|
{
|
|
return et_x_FromTo;
|
|
}
|
|
void CDrawingWorksheet::ReadAttributes(XmlUtils::CXmlLiteReader& oReader)
|
|
{
|
|
WritingElement_ReadAttributes_Start_No_NS ( oReader )
|
|
WritingElement_ReadAttributes_Read_if ( oReader, L"id", m_oId )
|
|
WritingElement_ReadAttributes_End_No_NS ( oReader )
|
|
}
|
|
void CDrawingWorksheet::ReadAttributes(XLS::BaseObjectPtr& obj)
|
|
{
|
|
auto ptr = static_cast<XLSB::Drawing*>(obj.get());
|
|
if(ptr != nullptr)
|
|
{
|
|
if(!ptr->stRelId.value.value().empty())
|
|
m_oId = ptr->stRelId.value.value();
|
|
}
|
|
}
|
|
|
|
CDrawing::CDrawing(OOX::Document* pMain) : OOX::FileGlobalEnumerated(pMain), OOX::IFileContainer(pMain)
|
|
{
|
|
m_bSpreadsheets = true;
|
|
}
|
|
CDrawing::CDrawing(OOX::Document* pMain, const CPath& oRootPath, const CPath& oPath) : OOX::FileGlobalEnumerated(pMain), OOX::IFileContainer(pMain)
|
|
{
|
|
m_bSpreadsheets = true;
|
|
read( oRootPath, oPath );
|
|
}
|
|
CDrawing::~CDrawing()
|
|
{
|
|
}
|
|
void CDrawing::read(const CPath& oPath)
|
|
{
|
|
//don't use this. use read(const CPath& oRootPath, const CPath& oFilePath)
|
|
CPath oRootPath;
|
|
read(oRootPath, oPath);
|
|
}
|
|
void CDrawing::read(const CPath& oRootPath, const CPath& oPath)
|
|
{
|
|
m_oReadPath = oPath;
|
|
IFileContainer::Read( oRootPath, oPath );
|
|
|
|
XmlUtils::CXmlLiteReader oReader;
|
|
|
|
if ( !oReader.FromFile( oPath.GetPath() ) )
|
|
return;
|
|
|
|
if ( !oReader.ReadNextNode() )
|
|
return;
|
|
|
|
std::wstring sName = XmlUtils::GetNameNoNS(oReader.GetName());
|
|
if ( (L"wsDr") == sName )
|
|
{
|
|
ReadAttributes( oReader );
|
|
|
|
if ( !oReader.IsEmptyNode() )
|
|
{
|
|
int nCurDepth = oReader.GetDepth();
|
|
while ( oReader.ReadNextSiblingNode( nCurDepth ) )
|
|
{
|
|
sName = XmlUtils::GetNameNoNS(oReader.GetName());
|
|
|
|
CCellAnchor *pItem = NULL;
|
|
|
|
if ( (L"absoluteAnchor") == sName )
|
|
{
|
|
pItem = new CCellAnchor();
|
|
*pItem = oReader;
|
|
pItem->m_oAnchorType.SetValue(SimpleTypes::Spreadsheet::cellanchorAbsolute);
|
|
}
|
|
else if ( (L"oneCellAnchor") == sName )
|
|
{
|
|
pItem = new CCellAnchor();
|
|
*pItem = oReader;
|
|
pItem->m_oAnchorType.SetValue(SimpleTypes::Spreadsheet::cellanchorOneCell);
|
|
}
|
|
else if ( (L"twoCellAnchor") == sName )
|
|
{
|
|
pItem = new CCellAnchor();
|
|
*pItem = oReader;
|
|
pItem->m_oAnchorType.SetValue(SimpleTypes::Spreadsheet::cellanchorTwoCell);
|
|
}
|
|
else if ( (L"AlternateContent") == sName)
|
|
{
|
|
nCurDepth++;
|
|
while( oReader.ReadNextSiblingNode( nCurDepth ) )
|
|
{
|
|
sName = XmlUtils::GetNameNoNS(oReader.GetName());
|
|
if ( (L"Choice") != sName && (L"Fallback") != sName ) continue;
|
|
|
|
nullable<std::wstring> sRequires;
|
|
WritingElement_ReadAttributes_Start( oReader )
|
|
WritingElement_ReadAttributes_Read_if ( oReader, (L"Requires"), sRequires )
|
|
WritingElement_ReadAttributes_End( oReader )
|
|
|
|
if (sRequires.IsInit() == false) continue;
|
|
if (*sRequires != (L"a14"))continue;
|
|
|
|
nCurDepth++;
|
|
while( oReader.ReadNextSiblingNode( nCurDepth ) )
|
|
{
|
|
sName = XmlUtils::GetNameNoNS(oReader.GetName());
|
|
if ( (L"absoluteAnchor") == sName )
|
|
{
|
|
pItem = new CCellAnchor();
|
|
*pItem = oReader;
|
|
pItem->m_oAnchorType.SetValue(SimpleTypes::Spreadsheet::cellanchorAbsolute);
|
|
}
|
|
else if ( (L"oneCellAnchor") == sName )
|
|
{
|
|
pItem = new CCellAnchor();
|
|
*pItem = oReader;
|
|
pItem->m_oAnchorType.SetValue(SimpleTypes::Spreadsheet::cellanchorOneCell);
|
|
}
|
|
else if ( (L"twoCellAnchor") == sName )
|
|
{
|
|
pItem = new CCellAnchor();
|
|
*pItem = oReader;
|
|
pItem->m_oAnchorType.SetValue(SimpleTypes::Spreadsheet::cellanchorTwoCell);
|
|
}
|
|
}nCurDepth--;
|
|
break;
|
|
}
|
|
nCurDepth--;
|
|
}
|
|
|
|
if (pItem)
|
|
{
|
|
m_arrItems.push_back(pItem);
|
|
|
|
if (pItem->m_nId.IsInit() && pItem->m_oElement.is_init())
|
|
{
|
|
m_mapShapes.insert(std::make_pair(*pItem->m_nId, pItem->m_oElement->GetElem().GetPointer()));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
void CDrawing::write(const CPath& oPath, const CPath& oDirectory, CContentTypes& oContent) const
|
|
{
|
|
NSStringUtils::CStringBuilder sXml;
|
|
sXml.WriteString((L"<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?><xdr:wsDr xmlns:xdr=\"http://schemas.openxmlformats.org/drawingml/2006/spreadsheetDrawing\" xmlns:a=\"http://schemas.openxmlformats.org/drawingml/2006/main\" xmlns:r=\"http://schemas.openxmlformats.org/officeDocument/2006/relationships\" xmlns:m=\"http://schemas.openxmlformats.org/officeDocument/2006/math\" xmlns:w=\"http://schemas.openxmlformats.org/wordprocessingml/2006/main\">"));
|
|
for(size_t i = 0, length = m_arrItems.size(); i < length; ++i)
|
|
m_arrItems[i]->toXML(sXml);
|
|
sXml.WriteString((L"</xdr:wsDr>"));
|
|
|
|
std::wstring sPath = oPath.GetPath();
|
|
NSFile::CFileBinary::SaveToFile(sPath.c_str(), sXml.GetData());
|
|
|
|
oContent.Registration( type().OverrideType(), oDirectory, oPath.GetFilename() );
|
|
IFileContainer::Write(oPath, oDirectory, oContent);
|
|
}
|
|
void CDrawing::toXLSChart(XLS::BaseObjectPtr chartStreamPtr)
|
|
{
|
|
auto ptr = static_cast<XLS::ChartSheetSubstream*>(chartStreamPtr.get());
|
|
|
|
auto ChartFormatsPtr = new XLS::CHARTFORMATS;
|
|
ptr->m_CHARTFORMATS = XLS::BaseObjectPtr(ChartFormatsPtr);
|
|
|
|
for(auto anchor : m_arrItems)
|
|
{
|
|
if(anchor->m_oElement.IsInit())
|
|
{
|
|
auto anchorElem = anchor->m_oElement->GetElem();
|
|
auto graphicFrame = static_cast<PPTX::Logic::GraphicFrame*>(anchorElem.GetPointer());
|
|
{
|
|
|
|
auto chartRect = new XLS::Chart;
|
|
if(anchor->m_oPos.IsInit() && anchor->m_oPos->m_oX.IsInit())
|
|
chartRect->x.dVal = anchor->m_oPos->m_oX->GetValue();
|
|
if(anchor->m_oPos.IsInit() && anchor->m_oPos->m_oY.IsInit())
|
|
chartRect->y.dVal = anchor->m_oPos->m_oY->GetValue();
|
|
if(anchor->m_oExt.IsInit() && anchor->m_oExt->m_oCx.IsInit())
|
|
chartRect->dx.dVal = anchor->m_oExt->m_oCx->GetValue();
|
|
if(anchor->m_oExt.IsInit() && anchor->m_oExt->m_oCy->GetValue())
|
|
chartRect->dy.dVal = anchor->m_oExt->m_oCy->GetValue();
|
|
ChartFormatsPtr->m_ChartRect = XLS::BaseObjectPtr(chartRect);
|
|
}
|
|
if(graphicFrame->chartRec.IsInit() && graphicFrame->chartRec->id_data.IsInit())
|
|
{
|
|
auto chartRid = graphicFrame->chartRec->id_data.get();
|
|
auto castedChart = Get<OOX::File>(chartRid);
|
|
auto ChartFile = static_cast<OOX::Spreadsheet::CChartFile*>(castedChart.GetPointer());
|
|
if(ChartFile->m_oChartSpace.m_spPr.IsInit())
|
|
{
|
|
ChartFormatsPtr->m_FRAME = ChartFile->m_oChartSpace.m_spPr->toXLSFrame();
|
|
}
|
|
if(ChartFile->m_oChartSpace.m_chart != nullptr && ChartFile->m_oChartSpace.m_chart->m_plotArea != nullptr)
|
|
{
|
|
auto AxisParentUnion = new XLS::AXISPARENT;
|
|
ChartFormatsPtr->m_arAXISPARENT.push_back(XLS::BaseObjectPtr(AxisParentUnion));
|
|
|
|
for(auto chartIndex = 0; chartIndex < ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items.size(); chartIndex ++)
|
|
{
|
|
if(*ChartFile->m_oChartSpace.m_chart->m_plotArea->m_ItemsElementName0.at(chartIndex) == OOX::Spreadsheet::itemschoicetype5BARCHART)
|
|
{
|
|
auto barChart = static_cast<CT_BarChart*>(ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items.at(chartIndex));
|
|
if(ChartFormatsPtr->m_arAXISPARENT.size() < 2)
|
|
{
|
|
auto axes = new XLS::AXES;
|
|
AxisParentUnion->m_AXES = XLS::BaseObjectPtr(axes);
|
|
if(barChart->m_axId.size() > 0)
|
|
{
|
|
auto vaxId = barChart->m_axId.at(0);
|
|
if(!ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items1.empty())
|
|
{
|
|
auto ivAx = static_cast<CT_CatAx*>(ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items1.at(0));
|
|
if(ivAx->m_axId.IsInit() && ivAx->m_axId.get() == vaxId)
|
|
{
|
|
axes->m_arAxes.push_back(ivAx->toXLS());
|
|
}
|
|
}
|
|
}
|
|
if(barChart->m_axId.size() > 1)
|
|
{
|
|
auto vaxId = barChart->m_axId.at(1);
|
|
if(!ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items1.empty())
|
|
{
|
|
auto dvAx = static_cast<CT_ValAx*>(ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items1.at(1));
|
|
if(dvAx->m_axId.IsInit() && dvAx->m_axId.get() == vaxId)
|
|
{
|
|
axes->m_arAxes.push_back(dvAx->toXLS());
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
AxisParentUnion->m_arCRT.push_back(barChart->toXLS(chartIndex, ptr->m_CHARTFORMATS));
|
|
}
|
|
else if(*ChartFile->m_oChartSpace.m_chart->m_plotArea->m_ItemsElementName0.at(chartIndex) == OOX::Spreadsheet::itemschoicetype5PIECHART)
|
|
{
|
|
auto PieChart = static_cast<CT_PieChart*>(ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items.at(chartIndex));
|
|
AxisParentUnion->m_arCRT.push_back(PieChart->toXLS(chartIndex, ptr->m_CHARTFORMATS));
|
|
}
|
|
else if(*ChartFile->m_oChartSpace.m_chart->m_plotArea->m_ItemsElementName0.at(chartIndex) == OOX::Spreadsheet::itemschoicetype5LINECHART)
|
|
{
|
|
auto LineChart = static_cast<CT_LineChart*>(ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items.at(chartIndex));
|
|
AxisParentUnion->m_arCRT.push_back(LineChart->toXLS(chartIndex, ptr->m_CHARTFORMATS));
|
|
}
|
|
else if(*ChartFile->m_oChartSpace.m_chart->m_plotArea->m_ItemsElementName0.at(chartIndex) == OOX::Spreadsheet::itemschoicetype5AREACHART)
|
|
{
|
|
auto AreaChart = static_cast<CT_AreaChart*>(ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items.at(chartIndex));
|
|
AxisParentUnion->m_arCRT.push_back(AreaChart->toXLS(chartIndex, ptr->m_CHARTFORMATS));
|
|
}
|
|
else if(*ChartFile->m_oChartSpace.m_chart->m_plotArea->m_ItemsElementName0.at(chartIndex) == OOX::Spreadsheet::itemschoicetype5SURFACECHART)
|
|
{
|
|
auto SurfaceChart = static_cast<CT_SurfaceChart*>(ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items.at(chartIndex));
|
|
AxisParentUnion->m_arCRT.push_back(SurfaceChart->toXLS(chartIndex, ptr->m_CHARTFORMATS));
|
|
}
|
|
else if(*ChartFile->m_oChartSpace.m_chart->m_plotArea->m_ItemsElementName0.at(chartIndex) == OOX::Spreadsheet::itemschoicetype5SCATTERCHART)
|
|
{
|
|
auto ScatterChart = static_cast<CT_ScatterChart*>(ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items.at(chartIndex));
|
|
AxisParentUnion->m_arCRT.push_back(ScatterChart->toXLS(chartIndex, ptr->m_CHARTFORMATS));
|
|
}
|
|
else if(*ChartFile->m_oChartSpace.m_chart->m_plotArea->m_ItemsElementName0.at(chartIndex) == OOX::Spreadsheet::itemschoicetype5RADARCHART)
|
|
{
|
|
auto ScatterChart = static_cast<CT_RadarChart*>(ChartFile->m_oChartSpace.m_chart->m_plotArea->m_Items.at(chartIndex));
|
|
AxisParentUnion->m_arCRT.push_back(ScatterChart->toXLS(chartIndex, ptr->m_CHARTFORMATS));
|
|
}
|
|
}
|
|
}
|
|
if(ChartFile->m_oChartSpace.m_chart->m_title != nullptr && ChartFile->m_oChartSpace.m_chart->m_title->m_tx != nullptr)
|
|
{
|
|
auto labelUnion = new XLS::ATTACHEDLABEL;
|
|
auto textRecord = new XLS::Text;
|
|
textRecord->wBkgMode = 1;
|
|
textRecord->at = 2;
|
|
textRecord->vat = 1;
|
|
auto textPos = new XLS::Pos;
|
|
textPos->mdBotRt = 2;
|
|
textPos->mdTopLt = 2;
|
|
labelUnion->m_Pos = XLS::BaseObjectPtr(textPos);
|
|
auto objLink = new XLS::ObjectLink;
|
|
objLink->wLinkObj = 1;
|
|
auto seriesText = new XLS::SeriesText;
|
|
seriesText->stText = ChartFile->m_oChartSpace.m_chart->m_title->m_tx->m_oRich->GetText();
|
|
auto aiUnion = new XLS::AI;
|
|
auto brai = new XLS::BRAI;
|
|
brai->rt = 1;
|
|
aiUnion->m_BRAI = XLS::BaseObjectPtr(brai);
|
|
aiUnion->m_SeriesText = XLS::BaseObjectPtr(seriesText);
|
|
labelUnion->m_AI = XLS::BaseObjectPtr(aiUnion);
|
|
labelUnion->m_ObjectLink = XLS::BaseObjectPtr(objLink);
|
|
labelUnion->m_TextProperties = XLS::BaseObjectPtr(textRecord);
|
|
ChartFormatsPtr->m_arATTACHEDLABEL.push_back(XLS::BaseObjectPtr(labelUnion));
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|
|
const OOX::FileType CDrawing::type() const
|
|
{
|
|
return OOX::Spreadsheet::FileTypes::Drawings;
|
|
}
|
|
const CPath CDrawing::DefaultDirectory() const
|
|
{
|
|
return type().DefaultDirectory();
|
|
}
|
|
const CPath CDrawing::DefaultFileName() const
|
|
{
|
|
return type().DefaultFileName();
|
|
}
|
|
const CPath& CDrawing::GetReadPath()
|
|
{
|
|
return m_oReadPath;
|
|
}
|
|
bool CDrawing::IsEmpty()
|
|
{
|
|
return m_arrItems.empty();
|
|
}
|
|
void CDrawing::ReadAttributes(XmlUtils::CXmlLiteReader& oReader)
|
|
{
|
|
}
|
|
void CDrawing::ClearItems()
|
|
{
|
|
m_mapShapes.clear();
|
|
for ( unsigned int nIndex = 0; nIndex < m_arrItems.size(); nIndex++ )
|
|
{
|
|
if ( m_arrItems[nIndex] )
|
|
delete m_arrItems[nIndex];
|
|
|
|
m_arrItems[nIndex] = NULL;
|
|
}
|
|
m_arrItems.clear();
|
|
}
|
|
|
|
} //Spreadsheet
|
|
} // namespace OOX
|