Files
Yajbir Singh f1b860b25c
Some checks failed
check / markdownlint (push) Has been cancelled
check / spellchecker (push) Has been cancelled
updated
2025-12-11 19:03:17 +05:30

190 lines
3.9 KiB
C++

#include "HWPXFile.h"
#include <cwctype>
namespace HWP
{
CHWPXFile::CHWPXFile(const HWP_STRING& sFileName)
: m_sFileName(sFileName), m_pZipFolder(nullptr), m_oDocInfo(this)
{}
CHWPXFile::~CHWPXFile()
{
Close();
CLEAR_ARRAY(CHWPSection, m_arSections);
}
VECTOR<const CHWPSection*> CHWPXFile::GetSections() const
{
RETURN_VECTOR_CONST_PTR(CHWPSection, m_arSections);
}
bool CHWPXFile::Detect()
{
BYTE *pBuffer = NULL;
DWORD dwBufferSize = 0;
if (!NSFile::CFileBinary::ReadAllBytes(m_sFileName, &pBuffer, dwBufferSize))
return false;
m_pZipFolder = new CZipFolderMemory(pBuffer, dwBufferSize);
return GetFileHeader();
}
bool CHWPXFile::Open()
{
if (!NSFile::CFileBinary::Exists(m_sFileName) || !Detect() || !ReadDocInfo())
return false;
for (const std::wstring& wsPath : GetPathsToSections())
ReadSection(wsPath);
return true;
}
void CHWPXFile::Close()
{
if (nullptr != m_pZipFolder)
{
delete m_pZipFolder;
m_pZipFolder = nullptr;
}
}
VECTOR<HWP_STRING> CHWPXFile::GetPathsToSections() const
{
VECTOR<std::wstring> arPaths{m_pZipFolder->getFiles(L"Contents", false)};
if (arPaths.empty())
return VECTOR<HWP_STRING>{};
struct TPathInfo
{
std::wstring m_wsPath;
unsigned short m_ushIndex;
bool operator<(const TPathInfo& oPathInfo) const
{
return m_ushIndex < oPathInfo.m_ushIndex;
}
};
VECTOR<TPathInfo> arPathsToSectors;
size_t unFoundPos, unNumberStart, unNumberEnd;
for (const std::wstring& wsPath : arPaths)
{
unFoundPos = wsPath.find(L"/Contents/section");
if (std::wstring::npos ==unFoundPos)
continue;
unNumberStart = unFoundPos + 17;
unNumberEnd = unNumberStart;
while (unNumberEnd < wsPath.length() && iswdigit(wsPath[unNumberEnd]))
++unNumberEnd;
if (unNumberEnd == unNumberStart ||
unNumberEnd >= wsPath.length() ||
L".xml" != wsPath.substr(unNumberEnd, 4))
continue;
arPathsToSectors.push_back({wsPath, static_cast<unsigned short>(std::stoi(wsPath.substr(unNumberStart, unNumberEnd - unNumberStart)))});
}
std::sort(arPathsToSectors.begin(), arPathsToSectors.end());
VECTOR<HWP_STRING> arResult{arPathsToSectors.size()};
for (unsigned int unIndex = 0; unIndex < arPathsToSectors.size(); ++unIndex)
arResult[unIndex] = arPathsToSectors[unIndex].m_wsPath;
return arResult;
}
bool CHWPXFile::GetFileHeader()
{
CXMLReader oReader;
if (!GetDocument(L"version.xml", oReader))
return false;
return m_oFileHeader.Parse(oReader);
}
const CHWPDocInfo* CHWPXFile::GetDocInfo() const
{
return &m_oDocInfo;
}
bool CHWPXFile::GetChildStream(const HWP_STRING& sFileName, CHWPStream& oBuffer)
{
if (nullptr == m_pZipFolder || !m_pZipFolder->exists(sFileName))
return false;
CZipFolderMemory::CBuffer *pFileBuffer = nullptr;
m_pZipFolder->read(sFileName, pFileBuffer);
if (nullptr == pFileBuffer)
return false;
pFileBuffer->UnsetDestroy();
oBuffer.SetStream((HWP_BYTE*)pFileBuffer->Buffer, pFileBuffer->Size);
delete pFileBuffer;
return true;
}
bool CHWPXFile::ReadDocInfo()
{
CXMLReader oContentReader;
if (!GetDocument(L"Contents/content.hpf", oContentReader))
return false;
if (m_oDocInfo.ReadContentHpf(oContentReader))
{
CXMLReader oHeaderReader;
if (!GetDocument(L"Contents/header.xml", oHeaderReader))
return false;
return m_oDocInfo.ParseHWPX(oHeaderReader);
}
return false;
}
bool CHWPXFile::ReadSection(const HWP_STRING& sName)
{
CXMLReader oReader;
if (!GetDocument(sName, oReader))
return false;
CHWPSection* pSection = new CHWPSection();
const bool bResult = pSection->Parse(oReader, EHanType::HWPX);
if (bResult)
m_arSections.push_back(pSection);
else
delete pSection;
return bResult;
}
bool CHWPXFile::GetDocument(const HWP_STRING& sEntryName, CXMLReader& oReader)
{
if (nullptr == m_pZipFolder || nullptr == oReader.GetReader())
return false;
if (m_pZipFolder->getReaderFromFile(sEntryName, *oReader.GetReader()))
return oReader.ReadNextNode();
return false;
}
}