mirror of
https://github.com/ONLYOFFICE/core.git
synced 2026-04-07 13:55:33 +08:00
453 lines
13 KiB
C++
453 lines
13 KiB
C++
//#include "./stdafx.h"
|
|
|
|
#include "GraphicFrame.h"
|
|
#include "SpTree.h"
|
|
#include "../Slide.h"
|
|
#include "../SlideLayout.h"
|
|
#include "../SlideMaster.h"
|
|
|
|
#include "../../ASCOfficeDrawingConverter.h"
|
|
|
|
namespace PPTX
|
|
{
|
|
namespace Logic
|
|
{
|
|
GraphicFrame::GraphicFrame()
|
|
{
|
|
}
|
|
|
|
GraphicFrame::~GraphicFrame()
|
|
{
|
|
}
|
|
|
|
GraphicFrame::GraphicFrame(XmlUtils::CXmlNode& node)
|
|
{
|
|
fromXML(node);
|
|
}
|
|
|
|
const GraphicFrame& GraphicFrame::operator =(XmlUtils::CXmlNode& node)
|
|
{
|
|
fromXML(node);
|
|
return *this;
|
|
}
|
|
|
|
void GraphicFrame::fromXML(XmlUtils::CXmlNode& node)
|
|
{
|
|
XmlUtils::CXmlNodes oNodes;
|
|
if (node.GetNodes(_T("*"), oNodes))
|
|
{
|
|
int count = oNodes.GetCount();
|
|
for (int i = 0; i < count; ++i)
|
|
{
|
|
XmlUtils::CXmlNode oNode;
|
|
oNodes.GetAt(i, oNode);
|
|
CString strName = XmlUtils::GetNameNoNS(oNode.GetName());
|
|
|
|
if (_T("xfrm") == strName)
|
|
xfrm = oNode;
|
|
else if (_T("nvGraphicFramePr") == strName)
|
|
nvGraphicFramePr = oNode;
|
|
|
|
if (_T("graphic") == strName)
|
|
{
|
|
XmlUtils::CXmlNode oNodeData;
|
|
if (oNode.GetNode(_T("a:graphicData"), oNodeData))
|
|
{
|
|
XmlUtils::CXmlNode oNode1 = oNodeData.ReadNodeNoNS(_T("tbl"));
|
|
if (oNode1.IsValid())
|
|
{
|
|
table = oNode1;
|
|
return;
|
|
}
|
|
XmlUtils::CXmlNode oNode2 = oNodeData.ReadNodeNoNS(_T("oleObj"));
|
|
if (oNode2.IsValid())
|
|
{
|
|
fromXMLOle(oNode2);
|
|
oNode2.ReadAttributeBase(L"spid", spid);
|
|
pic = oNode2.ReadNode(_T("p:pic"));
|
|
|
|
if (pic.is_init())
|
|
{
|
|
xfrm.Merge(pic->spPr.xfrm);
|
|
}
|
|
return;
|
|
}
|
|
XmlUtils::CXmlNode oNode3 = oNodeData.ReadNodeNoNS(_T("AlternateContent"));
|
|
if (oNode3.IsValid())
|
|
{
|
|
XmlUtils::CXmlNode oNodeC;
|
|
if (oNode3.GetNode(_T("mc:Choice"), oNodeC))
|
|
{
|
|
XmlUtils::CXmlNode oNodeO;
|
|
if (oNodeC.GetNode(_T("p:oleObj"), oNodeO))
|
|
{
|
|
oNodeO.ReadAttributeBase(L"spid", spid);
|
|
}
|
|
}
|
|
|
|
XmlUtils::CXmlNode oNodeFallback;
|
|
if (oNode3.GetNode(_T("mc:Fallback"), oNodeFallback))
|
|
{
|
|
XmlUtils::CXmlNode oNodeO;
|
|
if (oNodeFallback.GetNode(_T("p:oleObj"), oNodeO))
|
|
{
|
|
fromXMLOle(oNodeO);
|
|
pic = oNodeO.ReadNode(_T("p:pic"));
|
|
|
|
if (pic.is_init())
|
|
{
|
|
xfrm.Merge(pic->spPr.xfrm);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
XmlUtils::CXmlNode oNode4 = oNodeData.ReadNode(_T("dgm:relIds"));
|
|
if (oNode4.IsValid())
|
|
{
|
|
smartArt = oNode4;
|
|
}
|
|
XmlUtils::CXmlNode oNode5 = oNodeData.ReadNode(_T("c:chart"));
|
|
if (oNode5.IsValid())
|
|
{
|
|
chartRec = oNode5;
|
|
}
|
|
XmlUtils::CXmlNode oNode6 = oNodeData.ReadNode(_T("com:legacyDrawing"));
|
|
if (oNode6.IsValid())
|
|
{
|
|
oNode6.ReadAttributeBase(L"spid", spid);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if(pic.IsInit() && oleObject.IsInit())
|
|
{
|
|
pic->oleObject = oleObject;
|
|
pic->blipFill.blip->oleRid = oleObject->m_oId.get().ToString();
|
|
}
|
|
|
|
FillParentPointersForChilds();
|
|
}
|
|
void GraphicFrame::fromXMLOle(XmlUtils::CXmlNode& node)
|
|
{
|
|
oleObject.Init();
|
|
node.ReadAttributeBase(L"progId", oleObject->m_sProgId);
|
|
node.ReadAttributeBase(L"r:id", oleObject->m_oId);
|
|
int imgW = node.GetAttributeInt(L"imgW", 0);
|
|
if(imgW > 0)
|
|
{
|
|
oleObject->m_oDxaOrig = Emu_To_Twips(imgW);
|
|
}
|
|
int imgH = node.GetAttributeInt(L"imgH", 0);
|
|
if(imgH > 0)
|
|
{
|
|
oleObject->m_oDyaOrig = Emu_To_Twips(imgH);
|
|
}
|
|
}
|
|
|
|
void GraphicFrame::toXmlWriter(NSBinPptxRW::CXmlWriter* pWriter) const
|
|
{
|
|
CString strNS = _T("");
|
|
if (pWriter->m_lDocType == XMLWRITER_DOC_TYPE_DOCX && pWriter->m_lGroupIndex >= 0)
|
|
{
|
|
pWriter->StartNode(_T("wpg:graphicFrame"));
|
|
strNS = _T("wpg");
|
|
}
|
|
else if (pWriter->m_lDocType == XMLWRITER_DOC_TYPE_XLSX && pWriter->m_lGroupIndex >= 0)
|
|
{
|
|
pWriter->StartNode(_T("xdr:graphicFrame"));
|
|
strNS = _T("xdr");
|
|
}
|
|
else
|
|
{
|
|
pWriter->StartNode(_T("p:graphicFrame"));
|
|
}
|
|
|
|
pWriter->EndAttributes();
|
|
|
|
nvGraphicFramePr.toXmlWriter(pWriter);
|
|
|
|
if (strNS == _T(""))
|
|
xfrm.toXmlWriter(pWriter);
|
|
else
|
|
xfrm.toXmlWriter2(strNS, pWriter);
|
|
|
|
if (table.is_init())
|
|
{
|
|
pWriter->WriteString(_T("<a:graphic><a:graphicData uri=\"http://schemas.openxmlformats.org/drawingml/2006/table\">"));
|
|
table->toXmlWriter(pWriter);
|
|
pWriter->WriteString(_T("</a:graphicData></a:graphic>"));
|
|
}
|
|
else if (chartRec.is_init())
|
|
{
|
|
pWriter->WriteString(_T("<a:graphic><a:graphicData uri=\"http://schemas.openxmlformats.org/drawingml/2006/chart\">"));
|
|
chartRec->toXmlWriter(pWriter);
|
|
pWriter->WriteString(_T("</a:graphicData></a:graphic>"));
|
|
}
|
|
|
|
if (pWriter->m_lDocType == XMLWRITER_DOC_TYPE_DOCX && pWriter->m_lGroupIndex >= 0)
|
|
pWriter->EndNode(_T("wpg:graphicFrame"));
|
|
else if (pWriter->m_lDocType == XMLWRITER_DOC_TYPE_XLSX && pWriter->m_lGroupIndex >= 0)
|
|
pWriter->EndNode(_T("xdr:graphicFrame"));
|
|
else
|
|
pWriter->EndNode(_T("p:graphicFrame"));
|
|
}
|
|
|
|
void GraphicFrame::toPPTY(NSBinPptxRW::CBinaryFileWriter* pWriter) const
|
|
{
|
|
if (pic.is_init())
|
|
{
|
|
pic->toPPTY(pWriter);
|
|
return;
|
|
}
|
|
if (!smartArt.is_init() && !table.is_init() && !chartRec.is_init() && !spid.is_init() )
|
|
return;
|
|
|
|
CString xml_object_vml;
|
|
CString xml_object_rels;
|
|
|
|
if (spid.is_init())
|
|
{
|
|
xml_object_vml = GetVmlXmlBySpid(xml_object_rels);
|
|
}
|
|
if (smartArt.is_init() && !table.is_init() && !chartRec.is_init() && !spid.is_init())
|
|
{
|
|
smartArt->LoadDrawing(pWriter);
|
|
if (smartArt->m_diag.is_init())
|
|
{
|
|
smartArt->m_diag->nvGrpSpPr.cNvPr = nvGraphicFramePr.cNvPr;
|
|
smartArt->m_diag->nvGrpSpPr.nvPr = nvGraphicFramePr.nvPr;
|
|
|
|
bool bIsInitCoords = false;
|
|
if (smartArt->m_diag->grpSpPr.xfrm.is_init())
|
|
bIsInitCoords = true;
|
|
else
|
|
smartArt->m_diag->grpSpPr.xfrm = new PPTX::Logic::Xfrm();
|
|
|
|
PPTX::Logic::Xfrm* dst = smartArt->m_diag->grpSpPr.xfrm.GetPointer();
|
|
const PPTX::Logic::Xfrm* src = &xfrm;
|
|
|
|
dst->offX = src->offX;
|
|
dst->offY = src->offY;
|
|
dst->extX = src->extX;
|
|
dst->extY = src->extY;
|
|
|
|
if (!bIsInitCoords || !dst->chOffX.is_init() || !dst->chOffY.is_init() || !dst->chExtX.is_init() || !dst->chExtY.is_init())
|
|
{
|
|
dst->chOffX = 0;
|
|
dst->chOffY = 0;
|
|
dst->chExtX = src->extX;
|
|
dst->chExtY = src->extY;
|
|
}
|
|
|
|
dst->flipH = src->flipH;
|
|
dst->flipV = src->flipV;
|
|
dst->rot = src->rot;
|
|
|
|
//удалим индекс плейсхолдера если он есть(p:nvPr) - он будет лишний так как будет имплементация объекта
|
|
if (smartArt->m_diag->nvGrpSpPr.nvPr.ph.IsInit())
|
|
{
|
|
if(smartArt->m_diag->nvGrpSpPr.nvPr.ph->idx.IsInit())
|
|
{
|
|
smartArt->m_diag->nvGrpSpPr.nvPr.ph.reset();
|
|
}
|
|
}
|
|
smartArt->toPPTY(pWriter);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (!table.is_init() && !chartRec.is_init() && xml_object_vml.IsEmpty() == false)
|
|
{
|
|
CString temp = _T("<v:object>");
|
|
temp.Append(xml_object_vml);
|
|
temp.Append(_T("</v:object>"));
|
|
NSBinPptxRW::CDrawingConverter oDrawingConverter;
|
|
//oDrawingConverter.SetFontManager(pFontManager);
|
|
|
|
RELEASEOBJECT(oDrawingConverter.m_pBinaryWriter->m_pCommon->m_pImageManager);
|
|
oDrawingConverter.m_pBinaryWriter->m_pCommon->m_pImageManager = pWriter->m_pCommon->m_pImageManager;
|
|
|
|
CString *main_props = NULL;
|
|
|
|
oDrawingConverter.SetRelsPath(xml_object_rels);
|
|
oDrawingConverter.SetAdditionalParam(_T("xfrm_override"), (BYTE*)&xfrm, sizeof(xfrm));
|
|
|
|
HRESULT hRes = oDrawingConverter.AddObject(temp, &main_props);
|
|
if (hRes == S_OK && oDrawingConverter.m_pBinaryWriter->GetPosition() > 10)
|
|
{
|
|
pWriter->WriteBYTEArray(oDrawingConverter.m_pBinaryWriter->GetBuffer()+10,oDrawingConverter.m_pBinaryWriter->GetPosition()-10);
|
|
}
|
|
oDrawingConverter.m_pBinaryWriter->m_pCommon->m_pImageManager = NULL;
|
|
return;
|
|
}
|
|
pWriter->StartRecord(SPTREE_TYPE_GRFRAME);
|
|
|
|
pWriter->WriteBYTE(NSBinPptxRW::g_nodeAttributeStart);
|
|
pWriter->WriteString2(0, spid);
|
|
pWriter->WriteBYTE(NSBinPptxRW::g_nodeAttributeEnd);
|
|
|
|
pWriter->WriteRecord1(0, nvGraphicFramePr);
|
|
pWriter->WriteRecord1(1, xfrm);
|
|
pWriter->WriteRecord2(2, table);
|
|
|
|
if (chartRec.is_init())
|
|
{
|
|
pWriter->WriteRecord2(3, chartRec);
|
|
}
|
|
else if (xml_object_vml.IsEmpty() == false)
|
|
{
|
|
}
|
|
|
|
pWriter->EndRecord();
|
|
}
|
|
|
|
void GraphicFrame::fromPPTY(NSBinPptxRW::CBinaryFileReader* pReader)
|
|
{
|
|
LONG _end_rec = pReader->GetPos() + pReader->GetLong() + 4;
|
|
pReader->Skip(1); // start attributes
|
|
|
|
while (true)
|
|
{
|
|
BYTE _at = pReader->GetUChar_TypeNode();
|
|
if (_at == NSBinPptxRW::g_nodeAttributeEnd)
|
|
break;
|
|
|
|
switch (_at)
|
|
{
|
|
case 0:
|
|
{
|
|
spid = pReader->GetString2();
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
nvGraphicFramePr.cNvPr.id = -1;
|
|
|
|
while (pReader->GetPos() < _end_rec)
|
|
{
|
|
BYTE _at = pReader->GetUChar();
|
|
switch (_at)
|
|
{
|
|
case 0:
|
|
{
|
|
nvGraphicFramePr.fromPPTY(pReader);
|
|
break;
|
|
}
|
|
case 1:
|
|
{
|
|
xfrm.fromPPTY(pReader);
|
|
xfrm.m_ns = _T("p");
|
|
break;
|
|
}
|
|
case 2:
|
|
{
|
|
table = new Logic::Table();
|
|
table->fromPPTY(pReader);
|
|
break;
|
|
}
|
|
case 3:
|
|
{
|
|
chartRec = new Logic::ChartRec();
|
|
chartRec->fromPPTY(pReader);
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
pReader->Seek(_end_rec);
|
|
}
|
|
|
|
void GraphicFrame::GetRect(Aggplus::RECT& pRect)const
|
|
{
|
|
pRect.left = xfrm.offX.get();
|
|
pRect.top = xfrm.offY.get();
|
|
pRect.right = pRect.left + xfrm.extX.get();
|
|
pRect.bottom = pRect.top + xfrm.extY.get();
|
|
|
|
if(parentIs<Logic::SpTree>())
|
|
parentAs<Logic::SpTree>().NormalizeRect(pRect);
|
|
}
|
|
|
|
CString GraphicFrame::toXML() const
|
|
{
|
|
//XML::XNode node;
|
|
//if(dm.is_init())
|
|
//{
|
|
// return XML::XElement(ns.p + "graphicFrame",
|
|
// XML::Write(nvGraphicFramePr) +
|
|
// XML::Write(xfrm) +
|
|
// XML::XElement(ns.a + "graphic",
|
|
// XML::XElement(ns.a + "graphicData", //Возможно, здесь надо добавить ури
|
|
// XML::XElement(ns.dgm + "relIds",
|
|
// XML::XNamespace(ns.dgm) +
|
|
// XML::XNamespace(ns.r) +
|
|
// XML::XAttribute(ns.r + "dm", dm) +
|
|
// XML::XAttribute(ns.r + "lo", lo) +
|
|
// XML::XAttribute(ns.r + "qs", qs) +
|
|
// XML::XAttribute(ns.r + "cs", cs)
|
|
// )
|
|
// )
|
|
// )
|
|
// );
|
|
//}
|
|
//return node;
|
|
return _T("");
|
|
}
|
|
|
|
void GraphicFrame::FillParentPointersForChilds()
|
|
{
|
|
xfrm.SetParentPointer(this);
|
|
nvGraphicFramePr.SetParentPointer(this);
|
|
if(table.IsInit())
|
|
table->SetParentPointer(this);
|
|
if (smartArt.is_init())
|
|
smartArt->SetParentPointer(this);
|
|
if (chartRec.is_init())
|
|
chartRec->SetParentPointer(this);
|
|
if (pic.is_init())
|
|
pic->SetParentPointer(this);
|
|
}
|
|
|
|
//OOX::CPath GraphicFrame::GetPathBySpid()const
|
|
//{
|
|
// OOX::CPath filename;
|
|
// if(parentFileIs<PPTX::Slide>())
|
|
// filename = parentFileAs<PPTX::Slide>().GetPathBySpid(spid.get_value_or(_T("")));
|
|
// else if(parentFileIs<PPTX::SlideLayout>())
|
|
// filename = parentFileAs<PPTX::SlideLayout>().GetPathBySpid(spid.get_value_or(_T("")));
|
|
// else if(parentFileIs<PPTX::SlideMaster>())
|
|
// filename = parentFileAs<PPTX::SlideMaster>().GetPathBySpid(spid.get_value_or(_T("")));
|
|
|
|
// return filename;
|
|
//}
|
|
CString GraphicFrame::GetVmlXmlBySpid(CString & rels)const
|
|
{
|
|
CString xml;
|
|
rels = _T("");
|
|
if(parentFileIs<PPTX::Slide>() && parentFileAs<PPTX::Slide>().Vml.IsInit())
|
|
{
|
|
xml = parentFileAs<PPTX::Slide>().GetVmlXmlBySpid(spid.get_value_or(_T("")));
|
|
rels = parentFileAs<PPTX::Slide>().Vml->m_sFilename;
|
|
}
|
|
else if(parentFileIs<PPTX::SlideLayout>() && parentFileAs<PPTX::SlideLayout>().Vml.IsInit())
|
|
{
|
|
xml= parentFileAs<PPTX::SlideLayout>().GetVmlXmlBySpid(spid.get_value_or(_T("")));
|
|
rels = parentFileAs<PPTX::SlideLayout>().Vml->m_sFilename;
|
|
}
|
|
else if(parentFileIs<PPTX::SlideMaster>() && parentFileAs<PPTX::SlideMaster>().Vml.IsInit())
|
|
{
|
|
xml = parentFileAs<PPTX::SlideMaster>().GetVmlXmlBySpid(spid.get_value_or(_T("")));
|
|
rels = parentFileAs<PPTX::SlideMaster>().Vml->m_sFilename;
|
|
}
|
|
|
|
return xml;
|
|
}
|
|
} // namespace Logic
|
|
} // namespace PPTX
|