/*
* (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 "RtfChar.h"
#include "RtfDocument.h"
#include "../OOXml/Writer/OOXWriter.h"
RtfAbsPosTab::RtfAbsPosTab()
{
}
bool RtfAbsPosTab::IsValid()
{
return a_none != m_eAlignment && r_none != m_eRelative;
}
std::wstring RtfAbsPosTab::RenderToRtf(RenderParameter oRenderParameter)
{
std::wstring sResult;
switch( m_eLeader )
{
case l_ptablnone: sResult += L"\\ptablnone"; break;
case l_ptabldot: sResult += L"\\ptabldot"; break;
case l_ptablminus: sResult += L"\\ptablminus"; break;
case l_ptabluscore: sResult += L"\\ptabluscore"; break;
case l_ptablmdot: sResult += L"\\ptablmdo"; break;
}
switch( m_eRelative )
{
case r_margin: sResult += L"\\pmartabq"; break;
case r_indent: sResult += L"\\pindtabq"; break;
}
switch( m_eAlignment )
{
case a_left: sResult += L"l"; break;
case a_center: sResult += L"c"; break;
case a_right: sResult += L"r"; break;
}
if( false == sResult.empty() )
sResult = L"{" + sResult + L"}";
return sResult;
}
std::wstring RtfAbsPosTab::RenderToOOX(RenderParameter oRenderParameter)
{
std::wstring sResult;
switch( m_eLeader )
{
case l_ptablnone: sResult += L" w:leader=\"none\""; break;
case l_ptabldot: sResult += L" w:leader=\"dot\""; break;
case l_ptablminus: sResult += L" w:leader=\"hyphen\""; break;
case l_ptabluscore: sResult += L" w:leader=\"underscore\""; break;
case l_ptablmdot: sResult += L" w:leader=\"middleDot\""; break;
}
switch( m_eRelative )
{
case r_margin: sResult += L" w:relativeTo=\"margin\""; break;
case r_indent: sResult += L" w:relativeTo=\"indent\""; break;
}
switch( m_eAlignment )
{
case a_left: sResult += L" w:alignment=\"left\""; break;
case a_center: sResult += L" w:alignment=\"center\""; break;
case a_right: sResult += L" w:alignment=\"right\""; break;
}
if( !sResult.empty() )
sResult = L"";
return sResult;
}
RtfChar::RtfChar()
{
m_bRtfEncode = true;
}
int RtfChar::GetType()
{
return TYPE_RTF_CHAR;
}
void RtfChar::AddText(std::wstring text)
{
m_sChars += text;
}
void RtfChar::setText(std::wstring text)
{
m_sChars = text;
}
std::wstring RtfChar::GetText()
{
return m_sChars;
}
std::wstring RtfChar::RenderToOOX(RenderParameter oRenderParameter)
{
RtfDocument* poRtfDocument = static_cast (oRenderParameter.poDocument);
OOXWriter* poOOXWriter = static_cast (oRenderParameter.poWriter);
std::wstring sResult;
if (RENDER_TO_OOX_PARAM_RUN == oRenderParameter.nType ||
RENDER_TO_OOX_PARAM_FIELD == oRenderParameter.nType)
{
bool bInsert = false;
bool bDelete = false;
if (m_oProperty.m_nRevised != PROP_DEF)
{
bInsert = true;
std::wstring sAuthor = poRtfDocument->m_oRevisionTable.GetAuthor(m_oProperty.m_nRevauth);
std::wstring sDate(RtfUtility::convertDateTime(m_oProperty.m_nRevdttm).c_str());
sResult += L"m_nCurTrackChangesId++).c_str() + L"\">";
m_oProperty.m_nRevised = PROP_DEF;
}
if (m_oProperty.m_nDeleted != PROP_DEF)
{
bDelete = true;
std::wstring sAuthor = poRtfDocument->m_oRevisionTable.GetAuthor(m_oProperty.m_nRevauthDel);
std::wstring sDate(RtfUtility::convertDateTime(m_oProperty.m_nRevdttmDel).c_str());
sResult += L"m_nCurTrackChangesId++).c_str() + L"\">";
m_oProperty.m_nDeleted = PROP_DEF;
}
sResult += L"";
sResult += L"";
sResult += m_oProperty.RenderToOOX(oRenderParameter);
sResult += L"";
sResult += renderTextToXML((RENDER_TO_OOX_PARAM_FIELD == oRenderParameter.nType ? L"Field" : L"Text"), bDelete );
sResult += L"";
if (bDelete)sResult += L"";
if (bInsert)sResult += L"";
}
else if(RENDER_TO_OOX_PARAM_TEXT == oRenderParameter.nType)
{
sResult = renderTextToXML( L"Text" );
}
else if( RENDER_TO_OOX_PARAM_MATH == oRenderParameter.nType)
{
sResult += L"";
bool bInsert = false;
bool bDelete = false;
if (m_oProperty.m_nRevised != PROP_DEF)
{
bInsert = true;
std::wstring sAuthor = poRtfDocument->m_oRevisionTable.GetAuthor(m_oProperty.m_nRevauth);
std::wstring sDate(RtfUtility::convertDateTime(m_oProperty.m_nRevdttm).c_str());
sResult += L"m_nCurTrackChangesId++).c_str() + L"\">";
m_oProperty.m_nRevised = PROP_DEF;
}
if (m_oProperty.m_nDeleted != PROP_DEF)
{
bDelete = true;
std::wstring sAuthor = poRtfDocument->m_oRevisionTable.GetAuthor(m_oProperty.m_nRevauthDel);
std::wstring sDate(RtfUtility::convertDateTime(m_oProperty.m_nRevdttmDel).c_str());
sResult += L"m_nCurTrackChangesId++).c_str() + L"\">";
m_oProperty.m_nDeleted = PROP_DEF;
}
sResult += m_oProperty.RenderToOOX(oRenderParameter);//w:rPr внутри
sResult += renderTextToXML( L"Math" );
if (bDelete)sResult += L"";
if (bInsert)sResult += L"";
sResult += L"";
}
else if( RENDER_TO_OOX_PARAM_PLAIN == oRenderParameter.nType)
sResult = m_sChars;
return sResult;
}
std::wstring RtfChar::renderTextToXML( std::wstring sParam, bool bDelete )
{
std::wstring sResult;
if( L"Text" == sParam )
{
if (bDelete == false)
{
sResult += L"";
sResult += XmlUtils::EncodeXmlString( m_sChars, true );
sResult += L"";
}
else
{
sResult += L"";
sResult += XmlUtils::EncodeXmlString( m_sChars, true );
sResult += L"";
}
}
else if( L"Math" == sParam && !m_sChars.empty())
{
sResult += L"";
sResult += XmlUtils::EncodeXmlString( m_sChars, true );
sResult += L"";
}
else if (L"Field" == sParam && !m_sChars.empty())
{
sResult += L"";
sResult += XmlUtils::EncodeXmlString(m_sChars, true);
sResult += L"";
}
return sResult;
}
std::wstring RtfChar::renderRtfText( std::wstring& sText, void* poDocument, RtfCharProperty* oCharProperty, bool bMarker)
{
RtfDocument* pDocument = static_cast(poDocument);
int iFont = 0;
if (oCharProperty) iFont = oCharProperty->m_nFont;
int nCodePage = -1;
//применяем параметры codepage от текущего шрифта todo associated fonts.
//todooo разобраться со шрифтами и их подбором
RtfFont oFont;
if( NULL != oCharProperty && true == pDocument->m_oFontTable.GetFont( oCharProperty->m_nFont, oFont ) )
{
if( PROP_DEF != oFont.m_nCharset )
nCodePage = RtfUtility::CharsetToCodepage( oFont.m_nCharset );
else if( PROP_DEF != oFont.m_nCodePage )
nCodePage = oFont.m_nCodePage;
}
return renderRtfText(sText, pDocument, nCodePage, bMarker);
}
std::wstring RtfChar::renderRtfText(const std::wstring& sText)
{
std::wstring sResult;
for (size_t i = 0; i < sText.length(); i++)
{
int nUnicode = (int)sText[i];
if (0 < nUnicode && nUnicode <= 0x8000)
{
sResult += L"\\u" + std::to_wstring(nUnicode) + L"*";
}
else if (0x8000 < nUnicode && nUnicode <= 0xffff) {
sResult += L"\\u" + std::to_wstring(nUnicode - 0x10000) + L"*"; //??? font alt name china ALL FONTS NEW.docx (Mekanik LET)
}
else {
sResult += L"\\u9633*";
}
}
return sResult;
}
std::wstring RtfChar::renderRtfText( std::wstring& sText, void* poDocument, int nCodePage, bool bMarker)
{
RtfDocument* pDocument = static_cast(poDocument);
std::wstring sResult;
if (sText.empty())
{
return sResult;
}
//от настроек документа
if( -1 == nCodePage && RtfDocumentProperty::cp_none != pDocument->m_oProperty.m_eCodePage )
{
switch ( pDocument->m_oProperty.m_eCodePage )
{
case RtfDocumentProperty::cp_ansi:
{
if( PROP_DEF != pDocument->m_oProperty.m_nAnsiCodePage )
nCodePage = pDocument->m_oProperty.m_nAnsiCodePage;
else
nCodePage = CP_ACP;
break;
}
case RtfDocumentProperty::cp_mac: nCodePage = CP_MACCP; break;
case RtfDocumentProperty::cp_pc: nCodePage = 437; break;
case RtfDocumentProperty::cp_pca: nCodePage = 850; break;
}
}
//если ничего нет ставим ANSI или default from user
if( -1 == nCodePage )
{
nCodePage = CP_ACP;
}
if ((nCodePage == CP_ACP || nCodePage == 1252) && pDocument->m_nUserLCID > 0)
{
nCodePage = pDocument->m_lcidConverter.get_codepage(pDocument->m_nUserLCID);
}
if (bMarker)
sResult += L"{\\uc1";
sResult += renderRtfText(sText);
if (bMarker)
sResult += L"}";
return sResult;
}
std::wstring RtfChar::RenderToRtf(RenderParameter oRenderParameter)
{
std::wstring result;
if( RENDER_TO_RTF_PARAM_CHAR == oRenderParameter.nType )
{
if( true == m_bRtfEncode )
result += renderRtfText( m_sChars, oRenderParameter.poDocument, &m_oProperty );
else
result += m_sChars;
}
else
{
std::wstring sText;
if( true == m_bRtfEncode )
sText = renderRtfText( m_sChars, oRenderParameter.poDocument, &m_oProperty );
else
sText = m_sChars;
std::wstring sTextProp = m_oProperty.RenderToRtf( oRenderParameter ) ;
if( !sText.empty() || !sTextProp.empty())
{
if (oRenderParameter.nType != RENDER_TO_RTF_PARAM_NESTED)
result += L"{";
result += sTextProp;
result += L" ";
result += sText;
if (oRenderParameter.nType != RENDER_TO_RTF_PARAM_NESTED)
result += L"}";
}
}
return result;
}
RtfCharSpecial::RtfCharSpecial()
{
m_eType = rsc_none;
m_nTextWrapBreak = PROP_DEF;
m_nSoftHeight = PROP_DEF;
}
std::wstring RtfCharSpecial::_RenderToOOX(RenderParameter oRenderParameter)
{
std::wstring sResult;
switch( m_eType )
{
case rsc_chdate: sResult += L""; break;
case rsc_chdpl: sResult += L""; break;
case rsc_chdpa: sResult += L""; break;
case rsc_chtime: sResult += L""; break;
case rsc_chpgn: sResult += L""; break;
case rsc_sectnum: sResult += L""; break;
case rsc_chftn: sResult += L""; break;
case rsc_chftnEnd: sResult += L""; break;
case rsc_chatn: sResult += L""; break;
case rsc_chftnsep: sResult += L""; break;
case rsc_chftnsepc: sResult += L""; break;
case rsc_page: sResult += L""; break;
case rsc_column: sResult += L""; break;
case rsc_line: sResult += L"";break;
case rsc_softpage: sResult += L""; break;
case rsc_softcol: sResult += L""; break;
case rsc_softline: sResult += L""; break;
case rsc_tab: sResult += L""; break;
case rsc_emspace: sResult += L""; break;
case rsc_qmspace: sResult += L""; break;
case rsc_Formula: sResult += L""; break;
case rsc_zwbo: sResult += L""; break;
case rsc_zwnbo: sResult += L""; break;
case rsc_zwj: sResult += L""; break;
case rsc_zwnj: sResult += L""; break;
case rsc_OptHyphen: sResult += L"-"; break;//
case rsc_NonBrHyphen: sResult += L"-"; break;//
case rsc_NonBrSpace: sResult += L" "; break;
}
switch ( m_nTextWrapBreak )
{
case 0: sResult += L""; break;
case 1: sResult += L""; break;
case 2: sResult += L""; break;
case 3: sResult += L""; break;
}
return sResult;
}
std::wstring RtfCharSpecial::RenderToOOX(RenderParameter oRenderParameter)
{
RtfDocument* poRtfDocument = static_cast (oRenderParameter.poDocument);
OOXWriter* poOOXWriter = static_cast (oRenderParameter.poWriter);
std::wstring sResult;
if (RENDER_TO_OOX_PARAM_RUN == oRenderParameter.nType ||
RENDER_TO_OOX_PARAM_FIELD == oRenderParameter.nType)
{
bool bInsert = false;
bool bDelete = false;
if (m_oProperty.m_nRevised != PROP_DEF)
{
bInsert = true;
std::wstring sAuthor = poRtfDocument->m_oRevisionTable.GetAuthor(m_oProperty.m_nRevauth);
std::wstring sDate(RtfUtility::convertDateTime(m_oProperty.m_nRevdttm).c_str());
sResult += L"m_nCurTrackChangesId++).c_str() + L"\">";
m_oProperty.m_nRevised = PROP_DEF;
}
if (m_oProperty.m_nDeleted != PROP_DEF)
{
bDelete = true;
std::wstring sAuthor = poRtfDocument->m_oRevisionTable.GetAuthor(m_oProperty.m_nRevauthDel);
std::wstring sDate(RtfUtility::convertDateTime(m_oProperty.m_nRevdttmDel).c_str());
sResult += L"m_nCurTrackChangesId++).c_str() + L"\">";
m_oProperty.m_nDeleted = PROP_DEF;
}
sResult += L"";
sResult += L"";
sResult += m_oProperty.RenderToOOX(oRenderParameter);
sResult += L"";
sResult += _RenderToOOX(oRenderParameter);
sResult += L"";
if (bDelete)sResult += L"";
if (bInsert)sResult += L"";
}
else if(RENDER_TO_OOX_PARAM_TEXT == oRenderParameter.nType ||
RENDER_TO_OOX_PARAM_MATH == oRenderParameter.nType ||
RENDER_TO_OOX_PARAM_PLAIN == oRenderParameter.nType)
{
sResult += _RenderToOOX(oRenderParameter);
}
return sResult;
}
std::wstring RtfCharSpecial::RenderToRtf(RenderParameter oRenderParameter)
{
std::wstring sResult;
sResult += L"{";
sResult += m_oProperty.RenderToRtf( oRenderParameter );
switch( m_eType )
{
case rsc_chdate: sResult += L"\\chdate"; break;
case rsc_chdpl: sResult += L"\\chdpl"; break;
case rsc_chdpa: sResult += L"\\chdpa"; break;
case rsc_chtime: sResult += L"\\chtime"; break;
case rsc_chpgn: sResult += L"\\chpgn"; break;
case rsc_sectnum: sResult += L"\\sectnum"; break;
case rsc_chftn: sResult += L"\\chftn"; break;
case rsc_chftnEnd: sResult += L"\\chftn"; break;
case rsc_chatn: sResult += L"\\chatn"; break;
case rsc_chftnsep: sResult += L"\\chftnsep"; break;
case rsc_chftnsepc: sResult += L"\\chftnsepc"; break;
case rsc_page: sResult += L"\\page"; break;
case rsc_column: sResult += L"\\column"; break;
case rsc_line: sResult += L"\\line"; break;
case rsc_softpage: sResult += L"\\softpage"; break;
case rsc_softcol: sResult += L"\\softcol"; break;
case rsc_softline: sResult += L"\\softline"; break;
case rsc_tab: sResult += L"\\tab"; break;
case rsc_Formula: sResult += L"\\|"; break;
case rsc_OptHyphen: sResult += L"\\-"; break;
case rsc_NonBrHyphen: sResult += L"\\_"; break;
case rsc_NonBrSpace: sResult += L"\\~"; break;
case rsc_zwbo: sResult += L"\\zwbo"; break;
case rsc_zwnbo: sResult += L"\\zwnbo"; break;
case rsc_zwj: sResult += L"\\zwj"; break;
case rsc_zwnj: sResult += L"\\zwnj"; break;
}
if( PROP_DEF != m_nTextWrapBreak )
sResult += L"\\par";
//switch ( m_nTextWrapBreak ) //не воспринимается word
//{
// case 0: sResult += L"\\lbr0";break;
// case 1: sResult += L"\\lbr1";break;
// case 2: sResult += L"\\lbr2";break;
// case 3: sResult += L"\\lbr3";break;
//}
if( PROP_DEF != m_nSoftHeight )
{
sResult += L"\\softlheight" + std::to_wstring( m_nSoftHeight );
}
sResult += L"}";
return sResult;
}
std::wstring RtfCharNative::RenderToRtf(RenderParameter oRenderParameter)
{
std::wstring result;
if( RENDER_TO_RTF_PARAM_CHAR == oRenderParameter.nType )
{
result = m_sChars;
}
else
{
std::wstring sText = m_sChars;
if( L"" != sText )
{
result += L"{";
result += m_oProperty.RenderToRtf( oRenderParameter );
result += L" " + sText;
result += L"}";
}
}
return result;
}