Files
DocumentServer-v-9.2.0/core/RtfFile/Format/RtfField.cpp
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

490 lines
15 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 <boost/algorithm/string.hpp>
#include "RtfField.h"
#include "../OOXml/Writer/OOXWriter.h"
#include "../OOXml/Writer/OOXRelsWriter.h"
OOXFieldBegin::OOXFieldBegin()
{
SetDefault();
}
int OOXFieldBegin::GetType()
{
return TYPE_OOX_FIELD;
}
void OOXFieldBegin::SetDefault()
{
m_bDirty = PROP_DEF;
m_bLock = PROP_DEF;
m_oCharProperty.SetDefault();
}
std::wstring OOXFieldBegin::RenderToRtf(RenderParameter oRenderParameter)
{
std::wstring sResult;
sResult += L"{\\field ";
RENDER_RTF_BOOL( m_bDirty, sResult, L"flddirty" )
RENDER_RTF_BOOL( m_bLock, sResult, L"fldlock" )
sResult += L"{\\*\\fldinst";
sResult += m_oCharProperty.RenderToRtf(oRenderParameter);
return sResult;
}
std::wstring OOXFieldBegin::RenderToOOX(RenderParameter oRenderParameter)
{
std::wstring sResult;
sResult += L"<w:r>";
std::wstring props = m_oCharProperty.RenderToOOX(oRenderParameter);
if (props.empty())
{
sResult += L"<w:rPr>";
sResult += props;
sResult += L"</w:rPr>";
}
sResult += L"<w:fldChar w:fldCharType=\"separate\"";
RENDER_OOX_INT_ATTRIBUTE( m_bDirty, sResult, L"dirty" )
RENDER_OOX_INT_ATTRIBUTE( m_bLock, sResult, L"fldLock" )
sResult += L"/>";
sResult += L"</w:r>";
return sResult;
}
int OOXFieldInsertText::GetType()
{
return TYPE_OOX_FIELD;
}
std::wstring OOXFieldInsertText::RenderToRtf(RenderParameter oRenderParameter)
{
if( NULL != m_oText )
return m_oText->RenderToRtf( oRenderParameter );
else
return L"";
}
std::wstring OOXFieldInsertText::RenderToOOX(RenderParameter oRenderParameter)
{
if( NULL != m_oText )
{
std::wstring sResult;
sResult += L"<w:r>";
sResult += L"<w:instrText>";
oRenderParameter.nType = RENDER_TO_RTF_PARAM_CHAR;
oRenderParameter.nValue = RENDER_TO_RTF_PARAM_NO_PAR;
sResult += m_oText->RenderToOOX( oRenderParameter );
sResult += L"</w:instrText>";
sResult += L"</w:r>";
return sResult;
}
else
return L"";
}
int OOXFieldSeparate::GetType()
{
return TYPE_OOX_FIELD;
}
std::wstring OOXFieldSeparate::RenderToRtf(RenderParameter oRenderParameter)
{
std::wstring sResult;
return L"}{\\fldrslt";
}
std::wstring OOXFieldSeparate::RenderToOOX(RenderParameter oRenderParameter)
{
return L"<w:fldChar w:fldCharType=\"separate\"/>";
}
int OOXFieldEnd::GetType()
{
return TYPE_OOX_FIELD;
}
std::wstring OOXFieldEnd::RenderToRtf(RenderParameter oRenderParameter)
{
std::wstring sResult;
return L"}}";
}
std::wstring OOXFieldEnd::RenderToOOX(RenderParameter oRenderParameter)
{
return L"<w:fldChar w:fldCharType=\"end\"/>";
}
RtfFormField::RtfFormField()
{
}
int RtfFormField::GetType()
{
return TYPE_OOX_FORMFIELD;
}
std::wstring RtfFormField::RenderToRtf(RenderParameter oRenderParameter)
{
return L"";
}
std::wstring RtfFormField::RenderToOOX(RenderParameter oRenderParameter)
{
return L"";
}
RtfFieldInst::RtfFieldInst()
{
SetDefault();
}
void RtfFieldInst::SetDefaultRtf()
{
SetDefault();
}
void RtfFieldInst::SetDefaultOOX()
{
SetDefault();
}
void RtfFieldInst::SetDefault()
{
m_pTextItems = TextItemContainerPtr( new TextItemContainer() );
}
std::wstring RtfFieldInst::RenderToRtf(RenderParameter oRenderParameter)
{
return L"";
}
std::wstring RtfFieldInst::RenderToOOX(RenderParameter oRenderParameter)
{
RtfDocument* pRtfDocument = static_cast<RtfDocument*> (oRenderParameter.poDocument);
OOXWriter* pOOXWriter = static_cast<OOXWriter*> (oRenderParameter.poWriter);
if (m_pTextItems)
return m_pTextItems->RenderToOOX(oRenderParameter);
else
return L"";
}
RtfField::RtfField()
{
SetDefault();
}
int RtfField::GetType()
{
return TYPE_RTF_FIELD;
}
bool RtfField::IsValid()
{
return true;
//return false == m_oInsert.IsValid() && false == m_oResult.IsValid();
}
void RtfField::SetDefaultRtf()
{
SetDefault();
}
void RtfField::SetDefaultOOX()
{
SetDefault();
}
void RtfField::SetDefault()
{
m_eMode = fm_none;
m_bReferenceToEndnote = false;
m_bTextOnly = false;
m_sData = L"";
m_pResult = RtfFieldInstPtr(new RtfFieldInst());
m_pInsert = RtfFieldInstPtr(new RtfFieldInst());
m_oCharProperty.SetDefault();
}
std::wstring RtfField::RenderToRtf(RenderParameter oRenderParameter)
{
std::wstring sResult;
sResult += L"{\\field ";
if( fm_none != m_eMode )
{
switch( m_eMode )
{
case fm_flddirty: sResult += L"{\\flddirty "; break;
case fm_fldedit: sResult += L"{\\fldedit "; break;
case fm_fldlock: sResult += L"{\\fldlock "; break;
case fm_fldpriv: sResult += L"{\\fldpriv "; break;
}
}
sResult += L"{\\*\\fldinst ";
RenderParameter oNewParam = oRenderParameter;
oNewParam.nType = RENDER_TO_RTF_PARAM_PLAIN;//RENDER_TO_RTF_PARAM_CHAR;
oNewParam.nValue = RENDER_TO_RTF_PARAM_NO_PAR;
sResult += m_pInsert->m_pTextItems->RenderToRtf( oNewParam );
if( true == m_bReferenceToEndnote )
sResult += L"\\fldalt";
if( !m_sData.empty() )
sResult += L"{\\*\\datafield " + m_sData + L"}";
sResult += L"}";
std::wstring str = m_pResult->m_pTextItems->RenderToRtf( oRenderParameter ) ;
sResult += L"{\\fldrslt " + str + L"}";
sResult += L"}";
return sResult;
}
std::wstring RtfField::RenderToOOX(RenderParameter oRenderParameter)
{
RtfDocument* pRtfDocument = static_cast<RtfDocument*> (oRenderParameter.poDocument);
OOXWriter* pOOXWriter = static_cast<OOXWriter*> (oRenderParameter.poWriter);
std::wstring sResult;
if( true == m_bTextOnly )
{
RenderParameter oNewParam = oRenderParameter;
oNewParam.nType = RENDER_TO_OOX_PARAM_RUN;
sResult += m_pResult->m_pTextItems->RenderToOOX(oNewParam);
}
else
{
bool bInsert = false;
bool bDelete = false;
std::wstring sAuthor, sDate;
if (m_pInsert->m_oCharProperty.m_nRevised != PROP_DEF)
{
bInsert = true;
sAuthor = pRtfDocument->m_oRevisionTable.GetAuthor(m_pInsert->m_oCharProperty.m_nRevauth);
sDate = std::wstring(RtfUtility::convertDateTime(m_pInsert->m_oCharProperty.m_nRevdttm).c_str());
sResult += L"<w:ins w:date=\"" + sDate + L"\" w:author=\"" + XmlUtils::EncodeXmlString(sAuthor) + L"\" w:id=\"" + std::to_wstring(pOOXWriter->m_nCurTrackChangesId++).c_str() + L"\">";
m_pInsert->m_oCharProperty.m_nRevised = PROP_DEF;
}
if (m_pInsert->m_oCharProperty.m_nDeleted != PROP_DEF)
{
bDelete = true;
sAuthor = pRtfDocument->m_oRevisionTable.GetAuthor(m_pInsert->m_oCharProperty.m_nRevauthDel);
sDate = std::wstring(RtfUtility::convertDateTime(m_pInsert->m_oCharProperty.m_nRevdttmDel).c_str());
sResult += L"<w:del w:date=\"" + sDate + L"\" w:author=\"" + XmlUtils::EncodeXmlString(sAuthor) + L"\" w:id=\"" + std::to_wstring(pOOXWriter->m_nCurTrackChangesId++).c_str() + L"\">";
m_pInsert->m_oCharProperty.m_nDeleted = PROP_DEF;
}
//поверяем на наличие гиперссылки
RenderParameter oNewParam = oRenderParameter;
oNewParam.nType = RENDER_TO_OOX_PARAM_PLAIN;
std::wstring sInsertText = m_pInsert->m_pTextItems->RenderToOOX( oNewParam );
size_t nIndex = sInsertText.find( L"HYPERLINK" );
if ( std::wstring::npos != nIndex && (m_pResult) && (m_pResult->m_pTextItems) && m_pResult->m_pTextItems->GetCount() < 2)
{
std::wstring sHyperlink = sInsertText;
sHyperlink.erase( nIndex, 9/*(int)_tcslen( L"HYPERLINK" )*/ );
size_t nSplash = sHyperlink.find( L"\\" );
if (std::wstring::npos != nSplash)
{
sHyperlink = sHyperlink.substr(0, nSplash);
}
//оставляем только одну ссылку
XmlUtils::replace_all(sHyperlink, L"\"", L"" );
boost::algorithm::trim(sHyperlink);
//заменяем пробелы на %20
XmlUtils::replace_all(sHyperlink, L" ", L"%20" );
//добавляем в rels
OOXRelsWriter* poRelsWriter = static_cast<OOXRelsWriter*>( oRenderParameter.poRels );
std::wstring sId = poRelsWriter->AddRelationship( L"http://schemas.openxmlformats.org/officeDocument/2006/relationships/hyperlink", XmlUtils::EncodeXmlString( sHyperlink ), false );
//добавляем гиперссылку в документ
sResult += L"<w:hyperlink r:id=\"" + sId + L"\" >";
oNewParam.nType = RENDER_TO_OOX_PARAM_RUN;
sResult += m_pResult->RenderToOOX(oNewParam);
sResult += L"</w:hyperlink>";
}
else if ((m_pInsert->m_pFormField) && (m_pInsert->m_pFormField->type == 2))
{
_UINT32 id = 0x7fff + pOOXWriter->m_nFormFieldId++;
std::wstring text_ = m_pInsert->m_pFormField->defres < m_pInsert->m_pFormField->list.size() ?
m_pInsert->m_pFormField->list[m_pInsert->m_pFormField->defres] : L"";
sResult += L"<w:sdt><w:sdtPr><w:alias w:val=\"\"/><w:id w:val=\"" + std::to_wstring(id) + L"\"/><w:dropDownList>";
for (size_t i = 0; i < m_pInsert->m_pFormField->list.size(); i++)
{
sResult += L"<w:listItem w:value=\"" + XmlUtils::EncodeXmlString(m_pInsert->m_pFormField->list[i]) + L"\" w:displayText=\"" + m_pInsert->m_pFormField->list[i] + L"\"/>";
}
sResult += L"</w:dropDownList></w:sdtPr><w:sdtEndPr/><w:sdtContent>";
sResult += L"<w:r><w:t>" + XmlUtils::EncodeXmlString(text_) + L"</w:t></w:r>";
sResult += L"</w:sdtContent></w:sdt>";
}
else
{
RenderParameter oNewParametr = oRenderParameter;
oNewParametr.nType = RENDER_TO_OOX_PARAM_FIELD;
std::wstring props = m_pResult->m_oCharProperty.RenderToOOX(oRenderParameter);
if (!props.empty()) props = L"<w:rPr>" + props + L"</w:rPr>";
//начинаем Field
sResult += L"<w:r>";
if (!props.empty())
sResult += props;
sResult += L"<w:fldChar w:fldCharType=\"begin\"";
if (m_pInsert->m_pFormField)
{
sResult += L"><w:ffData>";
sResult += L"<w:name w:val=\"" + XmlUtils::EncodeXmlString(m_pInsert->m_pFormField->name) + L"\"/>";
sResult += L"<w:enabled/>";
if (m_pInsert->m_pFormField->recalc == 1)
sResult += L"<w:calcOnExit/>";
else
sResult += L"<w:calcOnExit w:val=\"0\"/>";
if (false == m_pInsert->m_pFormField->entrymcr.empty())
sResult += L"<w:entryMacro w:val=\"" + m_pInsert->m_pFormField->entrymcr + L"\"/>";
if (false == m_pInsert->m_pFormField->exitmcr.empty())
sResult += L"<w:exitMacro w:val=\"" + m_pInsert->m_pFormField->exitmcr + L"\"/>";
if (false == m_pInsert->m_pFormField->helptext.empty())
sResult += L"<w:helpText w:type=\"text\" w:val=\"" + XmlUtils::EncodeXmlString(m_pInsert->m_pFormField->helptext) + L"\"/>";
if (false == m_pInsert->m_pFormField->stattext.empty())
sResult += L"<w:statusText w:type=\"text\" w:val=\"" + XmlUtils::EncodeXmlString(m_pInsert->m_pFormField->stattext) + L"\"/>";
if (m_pInsert->m_pFormField->type == 0)
{
sResult += L"<w:textInput>";
switch (m_pInsert->m_pFormField->typetx)
{
case 1: sResult += L"<w:type w:val=\"number\"/>"; break;
case 2: sResult += L"<w:type w:val=\"date\"/>"; break;
case 3: sResult += L"<w:type w:val=\"currentDate\"/>"; break;
case 4: sResult += L"<w:type w:val=\"currentTime\"/>"; break;
case 5: sResult += L"<w:type w:val=\"calculated\"/>"; break;
}
if (false == m_pInsert->m_pFormField->deftext.empty())
sResult += L"<w:default w:val=\"" + XmlUtils::EncodeXmlString(m_pInsert->m_pFormField->deftext) + L"\"/>";
if (false == m_pInsert->m_pFormField->format.empty())
sResult += L"<w:format w:val=\"" + XmlUtils::EncodeXmlString(m_pInsert->m_pFormField->format) + L"\"/>";
if (m_pInsert->m_pFormField->maxlen > 0)
sResult += L"<w:maxLength w:val=\"" + std::to_wstring(m_pInsert->m_pFormField->maxlen) + L"\"/>";
sResult += L"</w:textInput>";
}
else if (m_pInsert->m_pFormField->type == 1)
{
sResult += L"<w:checkBox>";
if (m_pInsert->m_pFormField->sizeCheckBox == 1)
sResult += L"<w:size w:val=\"" + std::to_wstring(m_pInsert->m_pFormField->hps) + L"\"/>";
else
sResult += L"<w:sizeAuto/>";
sResult += L"<w:default w:val=\"0\"/>";
if (m_pInsert->m_pFormField->res == 1)
{
sResult += L"<w:checked/>";
}
else
{
sResult += L"<w:checked w:val=\"0\"/>";
}
sResult += L"</w:checkBox>";
}
sResult += L"</w:ffData></w:fldChar>";
}
else sResult += L"/>";
sResult += L"</w:r>";
//-----------
if ((m_pInsert->m_pTextItems) && (m_pInsert->m_pTextItems->GetCount() > 0))
{
for (int i = 0; i < m_pInsert->m_pTextItems->GetCount(); i++)
{
sResult += m_pInsert->m_pTextItems->m_aArray[i]->RenderToOOX(oNewParametr);
}
}
oNewParametr = oRenderParameter;
// разделитель
sResult += L"<w:r>";
sResult += L"<w:fldChar w:fldCharType=\"separate\"/></w:r>";
//пишем содержание-кэш
if ((m_pResult->m_pTextItems) && (m_pResult->m_pTextItems->GetCount() > 0))
{
sResult += m_pResult->m_pTextItems->m_aArray[0]->RenderToOOX(oNewParametr);
for (int i = 1; i < m_pResult->m_pTextItems->GetCount(); i++)
{
RtfParagraph *paragraph = dynamic_cast<RtfParagraph *>(m_pResult->m_pTextItems->m_aArray[i].get());
if (paragraph)
{
if (bDelete) sResult += L"</w:del>";
if (bInsert) sResult += L"</w:ins>";
bDelete = bInsert = false;
sResult += L"</w:p>";
sResult += L"<w:p>";
sResult += L"<w:pPr>";
sResult += paragraph->m_oProperty.RenderToOOX(oRenderParameter);
sResult += L"</w:pPr>";
}
sResult += m_pResult->m_pTextItems->m_aArray[i]->RenderToOOX(oNewParametr);
}
}
else
{
sResult += L"<w:r>";
if (!props.empty())
sResult += props;
sResult += L"</w:r>";
}
//заканчиваем Field
sResult += L"<w:r>";
if (!props.empty())
sResult += props;
sResult += L"<w:fldChar w:fldCharType=\"end\"/></w:r>";
//sResult += L"<w:r><w:fldChar w:fldCharType=\"end\"/></w:r>";
}
if (bDelete) sResult += L"</w:del>";
if (bInsert) sResult += L"</w:ins>";
}
return sResult;
}