/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
* This file is part of the LibreOffice project.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* This file incorporates work covered by the following license notice:
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
*/
#include <hintids.hxx>
#include <com/sun/star/i18n/ScriptType.hpp>
#include <com/sun/star/i18n/XBreakIterator.hpp>
#include <comphelper/string.hxx>
#include <utility>
#include <vcl/svapp.hxx>
#include <svtools/htmlout.hxx>
#include <svtools/htmlkywd.hxx>
#include <svtools/htmltokn.h>
#include <svl/whiter.hxx>
#include <sfx2/event.hxx>
#include <sfx2/htmlmode.hxx>
#include <editeng/escapementitem.hxx>
#include <editeng/formatbreakitem.hxx>
#include <editeng/boxitem.hxx>
#include <editeng/ulspitem.hxx>
#include <editeng/udlnitem.hxx>
#include <editeng/crossedoutitem.hxx>
#include <editeng/blinkitem.hxx>
#include <editeng/colritem.hxx>
#include <editeng/fontitem.hxx>
#include <editeng/fhgtitem.hxx>
#include <editeng/postitem.hxx>
#include <editeng/wghtitem.hxx>
#include <editeng/adjustitem.hxx>
#include <editeng/lrspitem.hxx>
#include <editeng/langitem.hxx>
#include <editeng/frmdiritem.hxx>
#include <fchrfmt.hxx>
#include <fmtautofmt.hxx>
#include <fmtfsize.hxx>
#include <fmtclds.hxx>
#include <fmtpdsc.hxx>
#include <fmtflcnt.hxx>
#include <fmtinfmt.hxx>
#include <txatbase.hxx>
#include <frmatr.hxx>
#include <charfmt.hxx>
#include <fmtfld.hxx>
#include <doc.hxx>
#include <IDocumentStylePoolAccess.hxx>
#include <pam.hxx>
#include <ndtxt.hxx>
#include <paratr.hxx>
#include <poolfmt.hxx>
#include <pagedesc.hxx>
#include <swtable.hxx>
#include <fldbas.hxx>
#include <breakit.hxx>
#include "htmlatr.hxx"
#include "htmlnum.hxx"
#include "wrthtml.hxx"
#include "htmlfly.hxx"
#include <numrule.hxx>
#include <rtl/character.hxx>
#include <osl/diagnose.h>
#include <deque>
#include <svtools/HtmlWriter.hxx>
#include <o3tl/string_view.hxx>
#include <memory>
#include <algorithm>
using namespace css;
HTMLOutEvent const aAnchorEventTable[] =
{
{ OOO_STRING_SVTOOLS_HTML_O_SDonclick, OOO_STRING_SVTOOLS_HTML_O_onclick, SvMacroItemId::OnClick },
{ OOO_STRING_SVTOOLS_HTML_O_SDonmouseover, OOO_STRING_SVTOOLS_HTML_O_onmouseover, SvMacroItemId::OnMouseOver },
{ OOO_STRING_SVTOOLS_HTML_O_SDonmouseout, OOO_STRING_SVTOOLS_HTML_O_onmouseout, SvMacroItemId::OnMouseOut },
{ nullptr, nullptr, SvMacroItemId::NONE }
};
static SwHTMLWriter& OutHTML_SvxAdjust( SwHTMLWriter& rWrt, const SfxPoolItem& rHt );
sal_uInt16 SwHTMLWriter::GetDefListLvl( std::u16string_view rNm, sal_uInt16 nPoolId )
{
if( nPoolId == RES_POOLCOLL_HTML_DD )
{
return 1 | HTML_DLCOLL_DD;
}
else if( nPoolId == RES_POOLCOLL_HTML_DT )
{
return 1 | HTML_DLCOLL_DT;
}
OUString sDTDD = u"" OOO_STRING_SVTOOLS_HTML_dt " "_ustr;
if( o3tl::starts_with(rNm, sDTDD) )
// DefinitionList - term
return o3tl::narrowing<sal_uInt16>(o3tl::toInt32(rNm.substr( sDTDD.getLength() ))) | HTML_DLCOLL_DT;
sDTDD = OOO_STRING_SVTOOLS_HTML_dd " ";
if( o3tl::starts_with(rNm, sDTDD) )
// DefinitionList - definition
return o3tl::narrowing<sal_uInt16>(o3tl::toInt32(rNm.substr( sDTDD.getLength() ))) | HTML_DLCOLL_DD;
return 0;
}
void SwHTMLWriter::OutAndSetDefList( sal_uInt16 nNewLvl )
{
// possibly, we first need to start a new list
if( m_nDefListLvl < nNewLvl )
{
// output </pre> for the previous(!) paragraph, if required.
// Preferable, the <pre> is exported by OutHTML_SwFormatOff for the
// previous paragraph already, but that's not possible, because a very
// deep look at the next paragraph (this one) is required to figure
// out that a def list starts here.
ChangeParaToken( HtmlTokenId::NONE );
// write according to the level difference
for( sal_uInt16 i=m_nDefListLvl; i<nNewLvl; i++ )
{
if (IsLFPossible())
OutNewLine();
HTMLOutFuncs::Out_AsciiTag( Strm(), Concat2View(GetNamespace() + OOO_STRING_SVTOOLS_HTML_deflist) );
HTMLOutFuncs::Out_AsciiTag( Strm(), Concat2View(GetNamespace() + OOO_STRING_SVTOOLS_HTML_dd) );
IncIndentLevel();
SetLFPossible(true);
}
}
else if( m_nDefListLvl > nNewLvl )
{
for( sal_uInt16 i=nNewLvl ; i < m_nDefListLvl; i++ )
{
DecIndentLevel();
if (IsLFPossible())
OutNewLine();
HTMLOutFuncs::Out_AsciiTag( Strm(), Concat2View(GetNamespace() + OOO_STRING_SVTOOLS_HTML_dd), false );
HTMLOutFuncs::Out_AsciiTag( Strm(), Concat2View(GetNamespace() + OOO_STRING_SVTOOLS_HTML_deflist), false );
SetLFPossible(true);
}
}
m_nDefListLvl = nNewLvl;
}
void SwHTMLWriter::ChangeParaToken( HtmlTokenId nNew )
{
if( nNew != m_nLastParaToken && HtmlTokenId::PREFORMTXT_ON == m_nLastParaToken )
{
HTMLOutFuncs::Out_AsciiTag( Strm(), Concat2View(GetNamespace() + OOO_STRING_SVTOOLS_HTML_preformtxt), false );
SetLFPossible(true);
}
m_nLastParaToken = nNew;
}
sal_uInt16 SwHTMLWriter::GetCSS1ScriptForScriptType( sal_uInt16 nScriptType )
{
sal_uInt16 nRet = CSS1_OUTMODE_ANY_SCRIPT;
switch( nScriptType )
{
case i18n::ScriptType::LATIN:
nRet = CSS1_OUTMODE_WESTERN;
break;
case i18n::ScriptType::ASIAN:
nRet = CSS1_OUTMODE_CJK;
break;
case i18n::ScriptType::COMPLEX:
nRet = CSS1_OUTMODE_CTL;
break;
}
return nRet;
}
// a single output function should be enough for all formats
/*
* Output the formats as follows
* - output the tag for formats for which a corresponding HTML tag exist
* - for all the other formats, output a paragraph tag <P> and set bUserFormat
* - if a paragraph alignment is set for the supplied ItemSet of the node or
* for the ItemSet of the format, output an ALIGN=xxx if HTML allows it
* - In all cases, hard attribute is written as STYLE option.
* If bUserFormat is not set, only the supplied ItemSet is considered.
* Otherwise, attributes of the format are output as well.
*/
namespace {
struct SwHTMLTextCollOutputInfo
{
OString aToken; // End token to be output
std::optional<SfxItemSet> moItemSet; // hard attribute
bool bInNumberBulletList; // in an enumerated list;
bool bParaPossible; // a </P> may be output additionally
bool bOutPara; // a </P> is supposed to be output
bool bOutDiv; // write a </DIV>
SwHTMLTextCollOutputInfo() :
bInNumberBulletList( false ),
bParaPossible( false ),
bOutPara( false ),
bOutDiv( false )
{}
bool HasParaToken() const { return aToken.getLength()==1 && aToken[0]=='P'; }
bool ShouldOutputToken() const { return bOutPara || !HasParaToken(); }
};
}
SwHTMLFormatInfo::SwHTMLFormatInfo( const SwFormat *pF, SwDoc *pDoc, SwDoc *pTemplate,
bool bOutStyles,
LanguageType eDfltLang,
sal_uInt16 nCSS1Script )
: pFormat(pF)
, nLeftMargin(0)
, nRightMargin(0)
, nFirstLineIndent(0)
, nTopMargin(0)
, nBottomMargin(0)
, bScriptDependent( false )
{
sal_uInt16 nRefPoolId = 0;
// Get the selector of the format
sal_uInt16 nDeep = SwHTMLWriter::GetCSS1Selector( pFormat, aToken, aClass,
nRefPoolId );
OSL_ENSURE( nDeep ? !aToken.isEmpty() : aToken.isEmpty(),
"Something seems to be wrong with this token!" );
OSL_ENSURE( nDeep ? nRefPoolId != 0 : nRefPoolId == 0,
"Something seems to be wrong with the comparison style!" );
bool bTextColl = pFormat->Which() == RES_TXTFMTCOLL ||
pFormat->Which() == RES_CONDTXTFMTCOLL;
const SwFormat *pReferenceFormat = nullptr; // Comparison format
if( nDeep != 0 )
{
// It's an HTML-tag style or this style is derived from such
// a style.
if( !bOutStyles )
{
// if no styles are exported, it may be necessary to additionally
// write hard attribute
switch( nDeep )
{
case CSS1_FMT_ISTAG:
case CSS1_FMT_CMPREF:
// for HTML-tag styles the differences to the original
// (if available)
pReferenceFormat = SwHTMLWriter::GetTemplateFormat( nRefPoolId,
&pTemplate->getIDocumentStylePoolAccess() );
break;
default:
// otherwise, the differences to the HTML-tag style of the
// original or the ones to the current document, if it the
// HTML-tag style is not available
if( pTemplate )
pReferenceFormat = SwHTMLWriter::GetTemplateFormat( nRefPoolId,
&pTemplate->getIDocumentStylePoolAccess() );
else
pReferenceFormat = SwHTMLWriter::GetParentFormat( *pFormat, nDeep );
break;
}
}
}
else if( bTextColl )
{
// HTML-tag styles that are not derived from a paragraph style
// must be exported as hard attribute relative to the text-body
// style. For a 'not-styles' export, the one of the HTML style
// should be used as a reference
if( !bOutStyles && pTemplate )
pReferenceFormat = pTemplate->getIDocumentStylePoolAccess().GetTextCollFromPool( RES_POOLCOLL_TEXT, false );
else
pReferenceFormat = pDoc->getIDocumentStylePoolAccess().GetTextCollFromPool( RES_POOLCOLL_TEXT, false );
}
if( pReferenceFormat || nDeep==0 )
{
moItemSet.emplace( *pFormat->GetAttrSet().GetPool(),
pFormat->GetAttrSet().GetRanges() );
// if the differences to a different style are supposed to be
// written, hard attribute is necessary. This is always true
// for styles that are not derived from HTML-tag styles.
moItemSet->Set( pFormat->GetAttrSet() );
if( pReferenceFormat )
SwHTMLWriter::SubtractItemSet( *moItemSet, pReferenceFormat->GetAttrSet(), true );
// delete ItemSet that is empty straight away. This will save work
// later on
if( !moItemSet->Count() )
{
moItemSet.reset();
}
}
if( !bTextColl )
return;
if( bOutStyles )
{
// We have to add hard attributes for any script dependent
// item that is not accessed by the style
static const sal_uInt16 aWhichIds[3][4] =
{
{ RES_CHRATR_FONT, RES_CHRATR_FONTSIZE,
RES_CHRATR_POSTURE, RES_CHRATR_WEIGHT },
{ RES_CHRATR_CJK_FONT, RES_CHRATR_CJK_FONTSIZE,
RES_CHRATR_CJK_POSTURE, RES_CHRATR_CJK_WEIGHT },
{ RES_CHRATR_CTL_FONT, RES_CHRATR_CTL_FONTSIZE,
RES_CHRATR_CTL_POSTURE, RES_CHRATR_CTL_WEIGHT }
};
sal_uInt16 nRef = 0;
sal_uInt16 aSets[2] = {0,0};
switch( nCSS1Script )
{
case CSS1_OUTMODE_WESTERN:
nRef = 0;
aSets[0] = 1;
aSets[1] = 2;
break;
case CSS1_OUTMODE_CJK:
nRef = 1;
aSets[0] = 0;
aSets[1] = 2;
break;
case CSS1_OUTMODE_CTL:
nRef = 2;
aSets[0] = 0;
aSets[1] = 1;
break;
}
for( int i=0; i<4; ++i )
{
const SfxPoolItem& rRef = pFormat->GetFormatAttr( aWhichIds[nRef][i] );
for(sal_uInt16 nSet : aSets)
{
const SfxPoolItem& rSet = pFormat->GetFormatAttr( aWhichIds[nSet][i] );
if( rSet != rRef )
{
if( !moItemSet )
moItemSet.emplace( *pFormat->GetAttrSet().GetPool(),
pFormat->GetAttrSet().GetRanges() );
moItemSet->Put( rSet );
}
}
}
}
// remember all the different default spacings from the style or
// the comparison style.
SvxFirstLineIndentItem const& rFirstLine(
(pReferenceFormat ? pReferenceFormat : pFormat)->GetFirstLineIndent());
SvxTextLeftMarginItem const& rTextLeftMargin(
(pReferenceFormat ? pReferenceFormat : pFormat)->GetTextLeftMargin());
SvxRightMarginItem const& rRightMargin(
(pReferenceFormat ? pReferenceFormat : pFormat)->GetRightMargin());
nLeftMargin = rTextLeftMargin.GetTextLeft();
nRightMargin = rRightMargin.GetRight();
nFirstLineIndent = rFirstLine.ResolveTextFirstLineOffset({});
const SvxULSpaceItem &rULSpace =
(pReferenceFormat ? pReferenceFormat : pFormat)->GetULSpace();
nTopMargin = rULSpace.GetUpper();
nBottomMargin = rULSpace.GetLower();
// export language if it differs from the default language
TypedWhichId<SvxLanguageItem> nWhichId =
SwHTMLWriter::GetLangWhichIdFromScript( nCSS1Script );
const SvxLanguageItem& rLang = pFormat->GetFormatAttr( nWhichId );
LanguageType eLang = rLang.GetLanguage();
if( eLang != eDfltLang )
{
if( !moItemSet )
moItemSet.emplace( *pFormat->GetAttrSet().GetPool(),
pFormat->GetAttrSet().GetRanges() );
moItemSet->Put( rLang );
}
static const TypedWhichId<SvxLanguageItem> aWhichIds[3] =
{ RES_CHRATR_LANGUAGE, RES_CHRATR_CJK_LANGUAGE,
RES_CHRATR_CTL_LANGUAGE };
for(const TypedWhichId<SvxLanguageItem>& i : aWhichIds)
{
if( i != nWhichId )
{
const SvxLanguageItem& rTmpLang = pFormat->GetFormatAttr(i);
if( rTmpLang.GetLanguage() != eLang )
{
if( !moItemSet )
moItemSet.emplace( *pFormat->GetAttrSet().GetPool(),
pFormat->GetAttrSet().GetRanges() );
moItemSet->Put( rTmpLang );
}
}
}
}
SwHTMLFormatInfo::~SwHTMLFormatInfo()
{
}
static void OutHTML_SwFormat( SwHTMLWriter& rWrt, const SwFormat& rFormat,
const SfxItemSet *pNodeItemSet,
SwHTMLTextCollOutputInfo& rInfo )
{
OSL_ENSURE( RES_CONDTXTFMTCOLL==rFormat.Which() || RES_TXTFMTCOLL==rFormat.Which(),
"not a paragraph style" );
// First, some flags
sal_uInt16 nNewDefListLvl = 0;
sal_uInt16 nNumStart = USHRT_MAX;
bool bForceDL = false;
bool bDT = false;
rInfo.bInNumberBulletList = false; // Are we in a list?
bool bNumbered = false; // The current paragraph is numbered
bool bPara = false; // the current token is <P>
bool bHeading = false; // the current token is <H1> .. <H6>
rInfo.bParaPossible = false; // a <P> may be additionally output
bool bNoEndTag = false; // don't output an end tag
rWrt.m_bNoAlign = false; // no ALIGN=... possible
if (rWrt.mbXHTML)
{
rWrt.m_bNoAlign = true;
}
sal_uInt8 nBulletGrfLvl = 255; // The bullet graphic we want to output
// Are we in a bulleted or numbered list?
const SwTextNode* pTextNd = rWrt.m_pCurrentPam->GetPointNode().GetTextNode();
SwHTMLNumRuleInfo aNumInfo;
if( rWrt.GetNextNumInfo() )
{
aNumInfo = *rWrt.GetNextNumInfo();
rWrt.ClearNextNumInfo();
}
else
{
aNumInfo.Set( *pTextNd );
}
if( aNumInfo.GetNumRule() )
{
rInfo.bInNumberBulletList = true;
nNewDefListLvl = 0;
// is the current paragraph numbered?
bNumbered = aNumInfo.IsNumbered();
sal_uInt8 nLvl = aNumInfo.GetLevel();
OSL_ENSURE( pTextNd->GetActualListLevel() == nLvl,
"Remembered Num level is wrong" );
OSL_ENSURE( bNumbered == pTextNd->IsCountedInList(),
"Remembered numbering state is wrong" );
if( bNumbered )
{
nBulletGrfLvl = nLvl; // only temporarily!!!
// #i57919#
// correction of re-factoring done by cws swnumtree:
// - <nNumStart> has to contain the restart value, if the
// numbering is restarted at this text node. Value <USHRT_MAX>
// indicates, that no additional restart value has to be written.
if ( pTextNd->IsListRestart() )
{
nNumStart = static_cast< sal_uInt16 >(pTextNd->GetActualListStartValue());
}
OSL_ENSURE( rWrt.m_nLastParaToken == HtmlTokenId::NONE,
"<PRE> was not closed before <LI>." );
}
}
// Now, we're getting the token and, if necessary, the class
std::unique_ptr<SwHTMLFormatInfo> pTmpInfo(new SwHTMLFormatInfo(&rFormat));
SwHTMLFormatInfo *pFormatInfo;
SwHTMLFormatInfos::iterator it = rWrt.m_TextCollInfos.find( pTmpInfo );
if (it != rWrt.m_TextCollInfos.end())
{
pFormatInfo = it->get();
}
else
{
pFormatInfo = new SwHTMLFormatInfo( &rFormat, rWrt.m_pDoc, rWrt.m_xTemplate.get(),
rWrt.m_bCfgOutStyles, rWrt.m_eLang,
rWrt.m_nCSS1Script );
rWrt.m_TextCollInfos.insert(std::unique_ptr<SwHTMLFormatInfo>(pFormatInfo));
if( rWrt.m_aScriptParaStyles.count( rFormat.GetName() ) )
pFormatInfo->bScriptDependent = true;
}
// Now, we define what is possible due to the token
HtmlTokenId nToken = HtmlTokenId::NONE; // token for tag change
bool bOutNewLine = false; // only output a single LF?
if( !pFormatInfo->aToken.isEmpty() )
{
// It is an HTML-tag style or the style is derived from such a
// style.
rInfo.aToken = pFormatInfo->aToken;
if (rInfo.aToken == OOO_STRING_SVTOOLS_HTML_address)
{
rInfo.bParaPossible = true;
rWrt.m_bNoAlign = true;
}
else if (rInfo.aToken == OOO_STRING_SVTOOLS_HTML_blockquote)
{
rInfo.bParaPossible = true;
rWrt.m_bNoAlign = true;
}
else if (rInfo.aToken == OOO_STRING_SVTOOLS_HTML_parabreak)
{
bPara = true;
}
else if (rInfo.aToken == OOO_STRING_SVTOOLS_HTML_preformtxt)
{
if (HtmlTokenId::PREFORMTXT_ON == rWrt.m_nLastParaToken)
{
bOutNewLine = true;
}
else
{
nToken = HtmlTokenId::PREFORMTXT_ON;
rWrt.m_bNoAlign = true;
bNoEndTag = true;
}
}
else if (rInfo.aToken == OOO_STRING_SVTOOLS_HTML_dt || rInfo.aToken == OOO_STRING_SVTOOLS_HTML_dd)
{
bDT = rInfo.aToken == OOO_STRING_SVTOOLS_HTML_dt;
rInfo.bParaPossible = !bDT;
rWrt.m_bNoAlign = true;
bForceDL = true;
}
else if (rInfo.aToken == OOO_STRING_SVTOOLS_HTML_head1 ||
rInfo.aToken == OOO_STRING_SVTOOLS_HTML_head2 ||
rInfo.aToken == OOO_STRING_SVTOOLS_HTML_head3 ||
rInfo.aToken == OOO_STRING_SVTOOLS_HTML_head4 ||
rInfo.aToken == OOO_STRING_SVTOOLS_HTML_head5 ||
rInfo.aToken == OOO_STRING_SVTOOLS_HTML_head6)
{
bHeading = true;
}
}
else
{
// all styles that do not correspond to an HTML tag, or that are
// not derived from it, are exported as <P>
rInfo.aToken = OOO_STRING_SVTOOLS_HTML_parabreak ""_ostr;
bPara = true;
}
// If necessary, take the hard attribute from the style
if( pFormatInfo->moItemSet )
{
OSL_ENSURE(!rInfo.moItemSet, "Where does this ItemSet come from?");
rInfo.moItemSet.emplace( *pFormatInfo->moItemSet );
}
// additionally, add the hard attribute from the paragraph
if( pNodeItemSet )
{
if (rInfo.moItemSet)
rInfo.moItemSet->Put( *pNodeItemSet );
else
rInfo.moItemSet.emplace( *pNodeItemSet );
}
// we will need the lower spacing of the paragraph later on
const SvxULSpaceItem& rULSpace =
pNodeItemSet ? pNodeItemSet->Get(RES_UL_SPACE)
: rFormat.GetULSpace();
if( (rWrt.m_bOutHeader &&
rWrt.m_pCurrentPam->GetPoint()->GetNodeIndex() ==
rWrt.m_pCurrentPam->GetMark()->GetNodeIndex()) ||
rWrt.m_bOutFooter )
{
if( rWrt.m_bCfgOutStyles )
{
SvxULSpaceItem aULSpaceItem( rULSpace );
if( rWrt.m_bOutHeader )
aULSpaceItem.SetLower( rWrt.m_nHeaderFooterSpace );
else
aULSpaceItem.SetUpper( rWrt.m_nHeaderFooterSpace );
if (!rInfo.moItemSet)
{
rInfo.moItemSet.emplace(*rFormat.GetAttrSet().GetPool(), svl::Items<RES_UL_SPACE, RES_UL_SPACE>);
}
rInfo.moItemSet->Put( aULSpaceItem );
}
rWrt.m_bOutHeader = false;
rWrt.m_bOutFooter = false;
}
if( bOutNewLine )
{
// output a line break (without indentation) at the beginning of the
// paragraph, only
rInfo.aToken.clear(); // don't output an end tag
rWrt.Strm().WriteOString( SAL_NEWLINE_STRING );
return;
}
// should an ALIGN=... be written?
const SvxAdjustItem* pAdjItem = nullptr;
if( rInfo.moItemSet )
pAdjItem = rInfo.moItemSet->GetItemIfSet( RES_PARATR_ADJUST, false );
// Consider the lower spacing of the paragraph? (never in the last
// paragraph of tables)
bool bUseParSpace = !rWrt.m_bOutTable ||
(rWrt.m_pCurrentPam->GetPoint()->GetNodeIndex() !=
rWrt.m_pCurrentPam->GetMark()->GetNodeIndex());
// If styles are exported, indented paragraphs become definition lists
SvxFirstLineIndentItem const& rFirstLine(
pNodeItemSet ? pNodeItemSet->Get(RES_MARGIN_FIRSTLINE)
: rFormat.GetFirstLineIndent());
SvxTextLeftMarginItem const& rTextLeftMargin(
pNodeItemSet ? pNodeItemSet->Get(RES_MARGIN_TEXTLEFT)
: rFormat.GetTextLeftMargin());
if( (!rWrt.m_bCfgOutStyles || bForceDL) && !rInfo.bInNumberBulletList )
{
sal_Int32 nLeftMargin;
if( bForceDL )
nLeftMargin = rTextLeftMargin.GetTextLeft();
else
nLeftMargin = rTextLeftMargin.GetTextLeft() > pFormatInfo->nLeftMargin
? rTextLeftMargin.GetTextLeft() - pFormatInfo->nLeftMargin
: 0;
if( nLeftMargin > 0 && rWrt.m_nDefListMargin > 0 )
{
nNewDefListLvl = static_cast< sal_uInt16 >((nLeftMargin + (rWrt.m_nDefListMargin/2)) /
rWrt.m_nDefListMargin);
if( nNewDefListLvl == 0 && bForceDL && !bDT )
nNewDefListLvl = 1;
}
else
{
// If the left margin is 0 or negative, emulating indent
// with <dd> does not work. We then set a def list only if
// the dd style is used.
nNewDefListLvl = (bForceDL&& !bDT) ? 1 : 0;
}
bool bIsNextTextNode =
rWrt.m_pDoc->GetNodes()[rWrt.m_pCurrentPam->GetPoint()->GetNodeIndex()+1]
->IsTextNode();
if( bForceDL && bDT )
{
// Instead of a DD we must use a DT from the level above this one.
nNewDefListLvl++;
}
else if( !nNewDefListLvl && !rWrt.m_bCfgOutStyles && bPara &&
rULSpace.GetLower()==0 &&
((bUseParSpace && bIsNextTextNode) || rWrt.m_nDefListLvl==1) &&
(!pAdjItem || SvxAdjust::Left==pAdjItem->GetAdjust()) )
{
// Export paragraphs without a lower spacing as DT
nNewDefListLvl = 1;
bDT = true;
rInfo.bParaPossible = false;
rWrt.m_bNoAlign = true;
}
}
if( nNewDefListLvl != rWrt.m_nDefListLvl )
rWrt.OutAndSetDefList( nNewDefListLvl );
// if necessary, start a bulleted or numbered list
if( rInfo.bInNumberBulletList )
{
OSL_ENSURE( !rWrt.m_nDefListLvl, "DL cannot be inside OL!" );
OutHTML_NumberBulletListStart( rWrt, aNumInfo );
if( bNumbered )
{
// disable PVS False positive 557 "Array underrun is possible"
// we know here that nBulletGrfLvl < 10
// we're in the case bInNumberBulletList && bNumbered
// so we retrieved nLvl which comes from SwHTMLNumRuleInfo::GetLevel()
if( !rWrt.m_aBulletGrfs[nBulletGrfLvl].isEmpty() ) //-V557
bNumbered = false;
else
nBulletGrfLvl = 255;
}
}
// Take the defaults of the style, because they don't need to be
// exported
rWrt.m_nDfltLeftMargin = pFormatInfo->nLeftMargin;
rWrt.m_nDfltRightMargin = pFormatInfo->nRightMargin;
rWrt.m_nDfltFirstLineIndent = pFormatInfo->nFirstLineIndent;
if( rInfo.bInNumberBulletList )
{
if( !rWrt.IsHTMLMode( HTMLMODE_LSPACE_IN_NUMBER_BULLET ) )
rWrt.m_nDfltLeftMargin = rTextLeftMargin.GetTextLeft();
// In numbered lists, don't output a first line indent.
rWrt.m_nFirstLineIndent = rFirstLine.ResolveTextFirstLineOffset({});
}
if( rInfo.bInNumberBulletList && bNumbered && bPara && !rWrt.m_bCfgOutStyles )
{
// a single LI doesn't have spacing
rWrt.m_nDfltTopMargin = 0;
rWrt.m_nDfltBottomMargin = 0;
}
else if( rWrt.m_nDefListLvl && bPara )
{
// a single DD doesn't have spacing, as well
rWrt.m_nDfltTopMargin = 0;
rWrt.m_nDfltBottomMargin = 0;
}
else
{
rWrt.m_nDfltTopMargin = pFormatInfo->nTopMargin;
// if in the last paragraph of a table the lower paragraph spacing
// is changed, Netscape doesn't get it. That's why we don't
// export anything here for now, by setting this spacing to the
// default value.
if( rWrt.m_bCfgNetscape4 && !bUseParSpace )
rWrt.m_nDfltBottomMargin = rULSpace.GetLower();
else
rWrt.m_nDfltBottomMargin = pFormatInfo->nBottomMargin;
}
if( rWrt.m_nDefListLvl )
{
rWrt.m_nLeftMargin =
(rWrt.m_nDefListLvl-1) * rWrt.m_nDefListMargin;
}
if (rWrt.IsLFPossible() && !rWrt.m_bFirstLine)
rWrt.OutNewLine(); // paragraph tag on a new line
rInfo.bOutPara = false;
// this is now our new token
rWrt.ChangeParaToken( nToken );
bool bHasParSpace = bUseParSpace && rULSpace.GetLower() > 0;
// XHTML doesn't allow character children for <blockquote>.
bool bXhtmlBlockQuote = rWrt.mbXHTML && rInfo.aToken == OOO_STRING_SVTOOLS_HTML_blockquote;
// if necessary, start a new list item
bool bNumberedForListItem = bNumbered;
if (!bNumberedForListItem)
{
// Open a list also for the leading unnumbered nodes (= list headers in ODF terminology);
// to do that, detect if this unnumbered node is the first in this list
const auto& rPrevListInfo = rWrt.GetNumInfo();
if (rPrevListInfo.GetNumRule() != aNumInfo.GetNumRule() || aNumInfo.IsRestart(rPrevListInfo)
|| rPrevListInfo.GetDepth() < aNumInfo.GetDepth())
bNumberedForListItem = true;
}
if( rInfo.bInNumberBulletList && bNumberedForListItem )
{
OStringBuffer sOut(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_li);
if (!bNumbered)
{
// Handles list headers (<text:list-header> ODF element)
sOut.append(" " OOO_STRING_SVTOOLS_HTML_O_style "=\"display: block\"");
}
else if (USHRT_MAX != nNumStart)
sOut.append(" " OOO_STRING_SVTOOLS_HTML_O_value "=\"" + OString::number(nNumStart)
+ "\"");
HTMLOutFuncs::Out_AsciiTag(rWrt.Strm(), sOut);
}
if( rWrt.m_nDefListLvl > 0 && !bForceDL )
{
OString aTag = bDT ? OOO_STRING_SVTOOLS_HTML_dt : OOO_STRING_SVTOOLS_HTML_dd;
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + aTag) );
}
if( pAdjItem &&
rWrt.IsHTMLMode( HTMLMODE_NO_CONTROL_CENTERING ) &&
rWrt.HasControls() )
{
// The align=... attribute does behave strange in netscape
// if there are controls in a paragraph, because the control and
// all text behind the control does not recognize this attribute.
OString sOut = "<" + rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_division;
rWrt.Strm().WriteOString( sOut );
rWrt.m_bTextAttr = false;
rWrt.m_bOutOpts = true;
OutHTML_SvxAdjust( rWrt, *pAdjItem );
rWrt.Strm().WriteChar( '>' );
pAdjItem = nullptr;
rWrt.m_bNoAlign = false;
rInfo.bOutDiv = true;
rWrt.IncIndentLevel();
rWrt.SetLFPossible(true);
rWrt.OutNewLine();
}
// for BLOCKQUOTE, ADDRESS and DD we output another paragraph token, if
// - no styles are written and
// - a lower spacing or a paragraph alignment exists
// Also, XHTML does not allow character children in this context.
OString aToken = rInfo.aToken;
if( (!rWrt.m_bCfgOutStyles || rWrt.mbXHTML) && rInfo.bParaPossible && !bPara &&
(bHasParSpace || bXhtmlBlockQuote || pAdjItem) )
{
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + rInfo.aToken) );
aToken = OOO_STRING_SVTOOLS_HTML_parabreak ""_ostr;
bPara = true;
rWrt.m_bNoAlign = false;
}
LanguageType eLang;
if (rInfo.moItemSet)
{
const SvxLanguageItem& rLangItem = rInfo.moItemSet->Get(SwHTMLWriter::GetLangWhichIdFromScript(rWrt.m_nCSS1Script));
eLang = rLangItem.GetLanguage();
}
else
eLang = rWrt.m_eLang;
if( rInfo.moItemSet )
{
static const TypedWhichId<SvxLanguageItem> aWhichIds[3] = { RES_CHRATR_LANGUAGE, RES_CHRATR_CJK_LANGUAGE, RES_CHRATR_CTL_LANGUAGE };
for(auto const & i : aWhichIds)
{
// export language if it differs from the default language only.
const SvxLanguageItem* pTmpItem = rInfo.moItemSet->GetItemIfSet( i );
if( pTmpItem && pTmpItem->GetLanguage() == eLang )
rInfo.moItemSet->ClearItem( i );
}
}
// and the text direction
SvxFrameDirection nDir = rWrt.GetHTMLDirection(
(pNodeItemSet ? pNodeItemSet->Get( RES_FRAMEDIR )
: rFormat.GetFrameDir() ).GetValue() );
// We only write a <P>, if
// - we are not inside OL/UL/DL, or
// - the paragraph of an OL/UL is not numbered or
// - styles are not exported and
// - a lower spacing, or
// - a paragraph alignment exists, or
// - styles are exported and
// - the text body style was changed, or
// - a user format is exported, or
// - a paragraph attribute exists
if( !bPara ||
(!rInfo.bInNumberBulletList && !rWrt.m_nDefListLvl) ||
(rInfo.bInNumberBulletList && !bNumbered) ||
(!rWrt.m_bCfgOutStyles &&
(bHasParSpace || bXhtmlBlockQuote || pAdjItem ||
(eLang != LANGUAGE_DONTKNOW && eLang != rWrt.m_eLang))) ||
nDir != rWrt.m_nDirection ||
rWrt.m_bCfgOutStyles )
{
// now, options are output
rWrt.m_bTextAttr = false;
rWrt.m_bOutOpts = true;
OString sOut = "<" + rWrt.GetNamespace() + aToken;
if( eLang != LANGUAGE_DONTKNOW && eLang != rWrt.m_eLang )
{
rWrt.Strm().WriteOString( sOut );
sOut = ""_ostr;
rWrt.OutLanguage( eLang );
}
if( nDir != rWrt.m_nDirection )
{
if( !sOut.isEmpty() )
{
rWrt.Strm().WriteOString( sOut );
sOut = ""_ostr;
}
rWrt.OutDirection( nDir );
}
if( rWrt.m_bCfgOutStyles &&
(!pFormatInfo->aClass.isEmpty() || pFormatInfo->bScriptDependent) )
{
sOut += " " OOO_STRING_SVTOOLS_HTML_O_class "=\"";
rWrt.Strm().WriteOString( sOut );
sOut = ""_ostr;
OUString aClass( pFormatInfo->aClass );
if( pFormatInfo->bScriptDependent )
{
if( !aClass.isEmpty() )
aClass += "-";
switch( rWrt.m_nCSS1Script )
{
case CSS1_OUTMODE_WESTERN:
aClass += "western";
break;
case CSS1_OUTMODE_CJK:
aClass += "cjk";
break;
case CSS1_OUTMODE_CTL:
aClass += "ctl";
break;
}
}
HTMLOutFuncs::Out_String( rWrt.Strm(), aClass );
sOut += "\"";
}
// set inline heading (heading in a text frame anchored as character and
// formatted with frame style "Inline Heading")
if( bHeading && rWrt.IsInlineHeading() )
sOut += " " OOO_STRING_SVTOOLS_HTML_O_style "=\"display:inline;\"";
rWrt.Strm().WriteOString( sOut );
sOut = ""_ostr;
std::string_view sStyle;
if (rWrt.IsSpacePreserve())
{
if (rWrt.mbXHTML)
rWrt.Strm().WriteOString(" xml:space=\"preserve\"");
else
sStyle = "white-space: pre-wrap";
}
// if necessary, output alignment
if( !rWrt.m_bNoAlign && pAdjItem )
OutHTML_SvxAdjust( rWrt, *pAdjItem );
rWrt.m_bParaDotLeaders = bPara && rWrt.m_bCfgPrintLayout && rWrt.indexOfDotLeaders(
pTextNd->GetAnyFormatColl().GetPoolFormatId(), pTextNd->GetText()) > -1;
// and now, if necessary, the STYLE options
if (rWrt.m_bCfgOutStyles && rInfo.moItemSet)
{
OutCSS1_ParaTagStyleOpt(rWrt, *rInfo.moItemSet, sStyle);
}
if (rWrt.m_bParaDotLeaders) {
sOut += " " OOO_STRING_SVTOOLS_HTML_O_class "=\""
sCSS2_P_CLASS_leaders "\"><"
OOO_STRING_SVTOOLS_HTML_O_span;
rWrt.Strm().WriteOString( sOut );
sOut = ""_ostr;
}
rWrt.Strm().WriteChar( '>' );
// is a </P> supposed to be written?
rInfo.bOutPara =
bPara &&
( rWrt.m_bCfgOutStyles || bHasParSpace );
// if no end tag is supposed to be written, delete it
if( bNoEndTag )
rInfo.aToken.clear();
}
if( nBulletGrfLvl != 255 )
{
OSL_ENSURE( aNumInfo.GetNumRule(), "Where is the numbering gone???" );
OSL_ENSURE( nBulletGrfLvl < MAXLEVEL, "There are not this many layers." );
const SwNumFormat& rNumFormat = aNumInfo.GetNumRule()->Get(nBulletGrfLvl);
OutHTML_BulletImage( rWrt, OOO_STRING_SVTOOLS_HTML_image, rNumFormat.GetBrush(),
rWrt.m_aBulletGrfs[nBulletGrfLvl]);
}
rWrt.GetNumInfo() = aNumInfo;
// reset the defaults
rWrt.m_nDfltLeftMargin = 0;
rWrt.m_nDfltRightMargin = 0;
rWrt.m_nDfltFirstLineIndent = 0;
rWrt.m_nDfltTopMargin = 0;
rWrt.m_nDfltBottomMargin = 0;
rWrt.m_nLeftMargin = 0;
rWrt.m_nFirstLineIndent = 0;
}
static void OutHTML_SwFormatOff( SwHTMLWriter& rWrt, const SwHTMLTextCollOutputInfo& rInfo )
{
// if there is no token, we don't need to output anything
if( rInfo.aToken.isEmpty() )
{
rWrt.FillNextNumInfo();
const SwHTMLNumRuleInfo& rNextInfo = *rWrt.GetNextNumInfo();
// a bulleted list must be closed in PRE as well
if( rInfo.bInNumberBulletList )
{
const SwHTMLNumRuleInfo& rNRInfo = rWrt.GetNumInfo();
if( rNextInfo.GetNumRule() != rNRInfo.GetNumRule() ||
rNextInfo.GetDepth() != rNRInfo.GetDepth() ||
rNextInfo.IsNumbered() || rNextInfo.IsRestart(rNRInfo) )
rWrt.ChangeParaToken( HtmlTokenId::NONE );
OutHTML_NumberBulletListEnd( rWrt, rNextInfo );
}
else if( rNextInfo.GetNumRule() != nullptr )
rWrt.ChangeParaToken( HtmlTokenId::NONE );
return;
}
if( rInfo.ShouldOutputToken() )
{
if (rWrt.IsPrettyPrint() && rWrt.IsLFPossible())
rWrt.OutNewLine( true );
// if necessary, for BLOCKQUOTE, ADDRESS and DD another paragraph token
// is output, if
// - no styles are written and
// - a lower spacing exists
if( rInfo.bParaPossible && rInfo.bOutPara )
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_parabreak), false );
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + rInfo.aToken), false );
rWrt.SetLFPossible(
rInfo.aToken != OOO_STRING_SVTOOLS_HTML_dt &&
rInfo.aToken != OOO_STRING_SVTOOLS_HTML_dd &&
rInfo.aToken != OOO_STRING_SVTOOLS_HTML_li);
}
if( rInfo.bOutDiv )
{
rWrt.DecIndentLevel();
if (rWrt.IsLFPossible())
rWrt.OutNewLine();
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_division), false );
rWrt.SetLFPossible(true);
}
// if necessary, close the list item, then close a bulleted or numbered list
if( rInfo.bInNumberBulletList )
{
rWrt.FillNextNumInfo();
OutHTML_NumberBulletListEnd( rWrt, *rWrt.GetNextNumInfo() );
}
}
namespace {
class HTMLStartEndPos
{
sal_Int32 m_nStart;
sal_Int32 m_nEnd;
std::unique_ptr<SfxPoolItem> m_pItem;
public:
HTMLStartEndPos( const SfxPoolItem& rItem, sal_Int32 nStt, sal_Int32 nE );
const SfxPoolItem& GetItem() const { return *m_pItem; }
void SetStart(sal_Int32 nStt) { m_nStart = nStt; }
sal_Int32 GetStart() const { return m_nStart; }
sal_Int32 GetEnd() const { return m_nEnd; }
void SetEnd(sal_Int32 nE) { m_nEnd = nE; }
};
}
HTMLStartEndPos::HTMLStartEndPos(const SfxPoolItem& rItem, sal_Int32 nStt, sal_Int32 nE)
: m_nStart(nStt)
, m_nEnd(nE)
, m_pItem(rItem.Clone())
{}
typedef std::map<sal_Int32, std::vector<HTMLStartEndPos*>> HTMLStartEndPositions;
namespace {
enum HTMLOnOffState { HTML_NOT_SUPPORTED, // unsupported Attribute
HTML_REAL_VALUE, // Attribute with value
HTML_ON_VALUE, // Attribute is On-Tag
HTML_OFF_VALUE, // Attribute is Off-Tag
HTML_CHRFMT_VALUE, // Attribute for character format
HTML_COLOR_VALUE, // Attribute for foreground color
HTML_STYLE_VALUE, // Attribute must be exported as style
HTML_DROPCAP_VALUE, // DropCap-Attribute
HTML_AUTOFMT_VALUE }; // Attribute for automatic character styles
class HTMLEndPosLst
{
HTMLStartEndPositions m_aStartLst; // list, each position's elements sorted by appearance order
HTMLStartEndPositions m_aEndLst; // list, no sort of elements in position
std::deque<sal_Int32> m_aScriptChgLst; // positions where script changes
// 0 is not contained in this list,
// but the text length
// the script that is valid up to the position
// contained in aScriptChgList at the same index
std::vector<sal_uInt16> m_aScriptLst;
SwDoc* m_pDoc; // the current document
SwDoc* m_pTemplate; // the HTML template (or 0)
std::optional<Color> m_xDefaultColor; // the default foreground colors
std::set<OUString>& m_rScriptTextStyles;
sal_uLong m_nHTMLMode;
bool m_bOutStyles : 1; // are styles exported
// Insert/remove a SttEndPos in/from the Start and End lists.
// The end position is known.
void InsertItem_(HTMLStartEndPos* pPos);
// determine the 'type' of the attribute
HTMLOnOffState GetHTMLItemState( const SfxPoolItem& rItem );
// does a specific OnTag item exist
bool ExistsOnTagItem( sal_uInt16 nWhich, sal_Int32 nPos );
// does an item exist that can be used to disable an attribute that
// is exported the same way as the supplied item in the same range?
bool ExistsOffTagItem( sal_uInt16 nWhich, sal_Int32 nStartPos,
sal_Int32 nEndPos );
// adapt the end of a split item
void FixSplittedItem(HTMLStartEndPos* pPos, sal_Int32 nNewEnd);
// insert an attribute in the lists and, if necessary, split it
void InsertItem( const SfxPoolItem& rItem, sal_Int32 nStart,
sal_Int32 nEnd );
// split an already existing attribute
void SplitItem( const SfxPoolItem& rItem, sal_Int32 nStart,
sal_Int32 nEnd );
// Insert without taking care of script
void InsertNoScript( const SfxPoolItem& rItem, sal_Int32 nStart,
sal_Int32 nEnd, SwHTMLFormatInfos& rFormatInfos,
bool bParaAttrs );
const SwHTMLFormatInfo *GetFormatInfo( const SwFormat& rFormat,
SwHTMLFormatInfos& rFormatInfos );
void OutEndAttrs(SwHTMLWriter& rWrt, std::vector<HTMLStartEndPos*>& posItems);
public:
HTMLEndPosLst( SwDoc *pDoc, SwDoc* pTemplate, std::optional<Color> xDfltColor,
bool bOutStyles, sal_uLong nHTMLMode,
const OUString& rText, std::set<OUString>& rStyles );
~HTMLEndPosLst();
// insert an attribute
void Insert( const SfxPoolItem& rItem, sal_Int32 nStart, sal_Int32 nEnd,
SwHTMLFormatInfos& rFormatInfos, bool bParaAttrs=false );
void Insert( const SfxItemSet& rItemSet, sal_Int32 nStart, sal_Int32 nEnd,
SwHTMLFormatInfos& rFormatInfos, bool bDeep,
bool bParaAttrs=false );
void Insert( const SwDrawFrameFormat& rFormat, sal_Int32 nPos,
SwHTMLFormatInfos& rFormatInfos );
sal_uInt16 GetScriptAtPos( sal_Int32 nPos,
sal_uInt16 nWeak );
void OutStartAttrs( SwHTMLWriter& rWrt, sal_Int32 nPos );
void OutEndAttrs( SwHTMLWriter& rWrt, sal_Int32 nPos );
bool IsHTMLMode(sal_uLong nMode) const { return (m_nHTMLMode & nMode) != 0; }
};
struct SortEnds
{
HTMLStartEndPositions& m_startList;
SortEnds(HTMLStartEndPositions& startList) : m_startList(startList) {}
bool operator()(const HTMLStartEndPos* p1, const HTMLStartEndPos* p2)
{
// if p1 start after p2, then it ends before
if (p1->GetStart() > p2->GetStart())
return true;
if (p1->GetStart() < p2->GetStart())
return false;
for (const auto p : m_startList[p1->GetStart()])
{
if (p == p1)
return false;
if (p == p2)
return true;
}
assert(!"Neither p1 nor p2 found in their start list");
return false;
}
};
#ifndef NDEBUG
bool IsEmpty(const HTMLStartEndPositions& l)
{
return std::find_if(l.begin(), l.end(), [](auto& i) { return !i.second.empty(); }) == l.end();
}
#endif
}
void HTMLEndPosLst::InsertItem_(HTMLStartEndPos* pPos)
{
// Character border attribute must be the first which is written out because of border merge.
auto& posItems1 = m_aStartLst[pPos->GetStart()];
auto it = pPos->GetItem().Which() == RES_CHRATR_BOX ? posItems1.begin() : posItems1.end();
posItems1.insert(it, pPos);
m_aEndLst[pPos->GetEnd()].push_back(pPos);
}
HTMLOnOffState HTMLEndPosLst::GetHTMLItemState( const SfxPoolItem& rItem )
{
HTMLOnOffState eState = HTML_NOT_SUPPORTED;
switch( rItem.Which() )
{
case RES_CHRATR_POSTURE:
case RES_CHRATR_CJK_POSTURE:
case RES_CHRATR_CTL_POSTURE:
switch( static_cast<const SvxPostureItem&>(rItem).GetPosture() )
{
case ITALIC_NORMAL:
eState = HTML_ON_VALUE;
break;
case ITALIC_NONE:
eState = HTML_OFF_VALUE;
break;
default:
if( IsHTMLMode(HTMLMODE_SOME_STYLES) )
eState = HTML_STYLE_VALUE;
break;
}
break;
case RES_CHRATR_CROSSEDOUT:
switch( rItem.StaticWhichCast(RES_CHRATR_CROSSEDOUT).GetStrikeout() )
{
case STRIKEOUT_SINGLE:
case STRIKEOUT_DOUBLE:
eState = HTML_ON_VALUE;
break;
case STRIKEOUT_NONE:
eState = HTML_OFF_VALUE;
break;
default:
;
}
break;
case RES_CHRATR_ESCAPEMENT:
switch( static_cast<SvxEscapement>(rItem.StaticWhichCast(RES_CHRATR_ESCAPEMENT).GetEnumValue()) )
{
case SvxEscapement::Superscript:
case SvxEscapement::Subscript:
eState = HTML_ON_VALUE;
break;
case SvxEscapement::Off:
eState = HTML_OFF_VALUE;
break;
default:
;
}
break;
case RES_CHRATR_UNDERLINE:
switch( rItem.StaticWhichCast(RES_CHRATR_UNDERLINE).GetLineStyle() )
{
case LINESTYLE_SINGLE:
eState = HTML_ON_VALUE;
break;
case LINESTYLE_NONE:
eState = HTML_OFF_VALUE;
break;
default:
if( IsHTMLMode(HTMLMODE_SOME_STYLES) )
eState = HTML_STYLE_VALUE;
break;
}
break;
case RES_CHRATR_OVERLINE:
case RES_CHRATR_HIDDEN:
if( IsHTMLMode(HTMLMODE_SOME_STYLES) )
eState = HTML_STYLE_VALUE;
break;
case RES_CHRATR_WEIGHT:
case RES_CHRATR_CJK_WEIGHT:
case RES_CHRATR_CTL_WEIGHT:
switch( static_cast<const SvxWeightItem&>(rItem).GetWeight() )
{
case WEIGHT_BOLD:
eState = HTML_ON_VALUE;
break;
case WEIGHT_NORMAL:
eState = HTML_OFF_VALUE;
break;
default:
if( IsHTMLMode(HTMLMODE_SOME_STYLES) )
eState = HTML_STYLE_VALUE;
break;
}
break;
case RES_CHRATR_BLINK:
eState = rItem.StaticWhichCast(RES_CHRATR_BLINK).GetValue() ? HTML_ON_VALUE
: HTML_OFF_VALUE;
break;
case RES_CHRATR_COLOR:
eState = HTML_COLOR_VALUE;
break;
case RES_CHRATR_FONT:
case RES_CHRATR_FONTSIZE:
case RES_CHRATR_LANGUAGE:
case RES_CHRATR_CJK_FONT:
case RES_CHRATR_CJK_FONTSIZE:
case RES_CHRATR_CJK_LANGUAGE:
case RES_CHRATR_CTL_FONT:
case RES_CHRATR_CTL_FONTSIZE:
case RES_CHRATR_CTL_LANGUAGE:
case RES_TXTATR_INETFMT:
eState = HTML_REAL_VALUE;
break;
case RES_TXTATR_CHARFMT:
eState = HTML_CHRFMT_VALUE;
break;
case RES_TXTATR_AUTOFMT:
eState = HTML_AUTOFMT_VALUE;
break;
case RES_CHRATR_CASEMAP:
eState = HTML_STYLE_VALUE;
break;
case RES_CHRATR_KERNING:
eState = HTML_STYLE_VALUE;
break;
case RES_CHRATR_BACKGROUND:
if( IsHTMLMode(HTMLMODE_SOME_STYLES) )
eState = HTML_STYLE_VALUE;
break;
case RES_PARATR_DROP:
eState = HTML_DROPCAP_VALUE;
break;
case RES_CHRATR_BOX:
if( IsHTMLMode(HTMLMODE_SOME_STYLES) )
eState = HTML_STYLE_VALUE;
break;
}
return eState;
}
bool HTMLEndPosLst::ExistsOnTagItem( sal_uInt16 nWhich, sal_Int32 nPos )
{
for (const auto& [startPos, items] : m_aStartLst)
{
if (startPos > nPos)
{
// this attribute, and all attributes that follow, start later
break;
}
for (const auto* pTest : items)
{
if (pTest->GetEnd() > nPos)
{
// the attribute starts before, or at, the current position and ends after it
const SfxPoolItem& rItem = pTest->GetItem();
if (rItem.Which() == nWhich && HTML_ON_VALUE == GetHTMLItemState(rItem))
{
// an OnTag attribute was found
return true;
}
}
}
}
return false;
}
bool HTMLEndPosLst::ExistsOffTagItem( sal_uInt16 nWhich, sal_Int32 nStartPos,
sal_Int32 nEndPos )
{
if( nWhich != RES_CHRATR_CROSSEDOUT &&
nWhich != RES_CHRATR_UNDERLINE &&
nWhich != RES_CHRATR_BLINK )
{
return false;
}
for (const auto* pTest : m_aStartLst[nStartPos])
{
if (pTest->GetEnd() == nEndPos)
{
// the attribute starts before or at the current position and ends after it
const SfxPoolItem& rItem = pTest->GetItem();
sal_uInt16 nTstWhich = rItem.Which();
if( (nTstWhich == RES_CHRATR_CROSSEDOUT ||
nTstWhich == RES_CHRATR_UNDERLINE ||
nTstWhich == RES_CHRATR_BLINK) &&
HTML_OFF_VALUE == GetHTMLItemState(rItem) )
{
// an OffTag attribute was found that is exported the same
// way as the current item
return true;
}
}
}
return false;
}
void HTMLEndPosLst::FixSplittedItem(HTMLStartEndPos* pPos, sal_Int32 nNewEnd)
{
// remove the item from the End list
std::erase(m_aEndLst[pPos->GetEnd()], pPos);
// fix the end position accordingly
pPos->SetEnd( nNewEnd );
// from now on, it is closed at the corresponding position
m_aEndLst[nNewEnd].push_back(pPos);
// now, adjust the attributes that got started afterwards
const sal_Int32 nPos = pPos->GetStart();
for (const auto& [startPos, items] : m_aStartLst)
{
if (startPos < nPos)
continue;
if (startPos >= nNewEnd)
break;
auto it = items.begin();
if (startPos == nPos)
{
it = std::find(items.begin(), items.end(), pPos);
if (it != items.end())
++it;
}
for (; it != items.end(); ++it)
{
HTMLStartEndPos* pTest = *it;
const sal_Int32 nTestEnd = pTest->GetEnd();
if (nTestEnd <= nNewEnd)
continue;
// the Test attribute starts before the split attribute
// ends, and ends afterwards, i.e., it must be split, as well
// remove the attribute from the End list
std::erase(m_aEndLst[pTest->GetEnd()], pTest);
// set the new end
pTest->SetEnd( nNewEnd );
// it now ends in the respective position.
m_aEndLst[nNewEnd].push_back(pTest);
// insert the 'rest' of the attribute
InsertItem( pTest->GetItem(), nNewEnd, nTestEnd );
}
}
}
void HTMLEndPosLst::InsertItem( const SfxPoolItem& rItem, sal_Int32 nStart,
sal_Int32 nEnd )
{
assert(nStart < nEnd);
for (auto& [endPos, items] : m_aEndLst)
{
if (endPos <= nStart)
{
// the Test attribute ends, before the new one starts
continue;
}
if (endPos >= nEnd)
{
// the Test attribute (and all that follow) ends, before the new
// one ends
break;
}
std::sort(items.begin(), items.end(), SortEnds(m_aStartLst));
// Iterate over a temporary copy of items, because InsertItem_ may modify the vector
for (HTMLStartEndPos* pTest : std::vector(items))
{
if( pTest->GetStart() < nStart )
{
// the Test attribute ends, before the new one ends. Thus, the
// new attribute must be split.
InsertItem_(new HTMLStartEndPos(rItem, nStart, endPos));
nStart = endPos;
}
}
}
// one attribute must still be inserted
InsertItem_(new HTMLStartEndPos(rItem, nStart, nEnd));
}
void HTMLEndPosLst::SplitItem( const SfxPoolItem& rItem, sal_Int32 nStart,
sal_Int32 nEnd )
{
sal_uInt16 nWhich = rItem.Which();
// first, we must search for the old items by using the start list and
// determine the new item range
for (auto& [nTestStart, items] : m_aStartLst)
{
if( nTestStart >= nEnd )
{
// this attribute, and all that follow, start later
break;
}
for (auto it = items.begin(); it != items.end();)
{
auto itTest = it++; // forward early, allow 'continue', and keep a copy for 'erase'
HTMLStartEndPos* pTest = *itTest;
const sal_Int32 nTestEnd = pTest->GetEnd();
if (nTestEnd <= nStart)
continue;
// the Test attribute ends in the range that must be deleted
const SfxPoolItem& rTestItem = pTest->GetItem();
// only the corresponding OnTag attributes have to be considered
if (rTestItem.Which() != nWhich || HTML_ON_VALUE != GetHTMLItemState(rTestItem))
continue;
// if necessary, insert the second part of the split attribute
if (nTestEnd > nEnd)
InsertItem(rTestItem, nEnd, nTestEnd);
if (nTestStart >= nStart)
{
// the Test item only starts after the new end of the
// attribute. Therefore, it can be completely erased.
it = items.erase(itTest);
std::erase(m_aEndLst[nTestEnd], pTest);
delete pTest;
continue;
}
// the start of the new attribute corresponds to the new end of the attribute
FixSplittedItem(pTest, nStart);
}
}
}
const SwHTMLFormatInfo *HTMLEndPosLst::GetFormatInfo( const SwFormat& rFormat,
SwHTMLFormatInfos& rFormatInfos )
{
SwHTMLFormatInfo *pFormatInfo;
std::unique_ptr<SwHTMLFormatInfo> pTmpInfo(new SwHTMLFormatInfo(&rFormat));
SwHTMLFormatInfos::iterator it = rFormatInfos.find( pTmpInfo );
if (it != rFormatInfos.end())
{
pFormatInfo = it->get();
}
else
{
pFormatInfo = new SwHTMLFormatInfo(&rFormat, m_pDoc, m_pTemplate, m_bOutStyles);
rFormatInfos.insert(std::unique_ptr<SwHTMLFormatInfo>(pFormatInfo));
if (m_rScriptTextStyles.count(rFormat.GetName()))
pFormatInfo->bScriptDependent = true;
}
return pFormatInfo;
}
HTMLEndPosLst::HTMLEndPosLst(SwDoc* pD, SwDoc* pTempl, std::optional<Color> xDfltCol, bool bStyles,
sal_uLong nMode, const OUString& rText, std::set<OUString>& rStyles)
: m_pDoc(pD)
, m_pTemplate(pTempl)
, m_xDefaultColor(std::move(xDfltCol))
, m_rScriptTextStyles(rStyles)
, m_nHTMLMode(nMode)
, m_bOutStyles(bStyles)
{
sal_Int32 nEndPos = rText.getLength();
sal_Int32 nPos = 0;
while( nPos < nEndPos )
{
sal_uInt16 nScript = g_pBreakIt->GetBreakIter()->getScriptType( rText, nPos );
nPos = g_pBreakIt->GetBreakIter()->endOfScript( rText, nPos, nScript );
m_aScriptChgLst.push_back(nPos);
m_aScriptLst.push_back(nScript);
}
}
HTMLEndPosLst::~HTMLEndPosLst()
{
assert(IsEmpty(m_aStartLst) && "Start List not empty in destructor");
assert(IsEmpty(m_aEndLst) && "End List not empty in destructor");
}
void HTMLEndPosLst::InsertNoScript( const SfxPoolItem& rItem,
sal_Int32 nStart, sal_Int32 nEnd,
SwHTMLFormatInfos& rFormatInfos, bool bParaAttrs )
{
// no range ?? in that case, don't take it, it will never take effect !!
if( nStart == nEnd )
return;
bool bSet = false, bSplit = false;
switch( GetHTMLItemState(rItem) )
{
case HTML_ON_VALUE:
// output the attribute, if it isn't 'on', already
if( !ExistsOnTagItem( rItem.Which(), nStart ) )
bSet = true;
break;
case HTML_OFF_VALUE:
// If the corresponding attribute is 'on', split it.
// Additionally, output it as Style, if it is not set for the
// whole paragraph, because in that case it was already output
// together with the paragraph tag.
if( ExistsOnTagItem( rItem.Which(), nStart ) )
bSplit = true;
bSet = m_bOutStyles && !bParaAttrs && !ExistsOffTagItem(rItem.Which(), nStart, nEnd);
break;
case HTML_REAL_VALUE:
// we can always output the attribute
bSet = true;
break;
case HTML_STYLE_VALUE:
// We can only output the attribute as CSS1. If it is set for
// the paragraph, it was already output with the paragraph tag.
// The only exception is the character-background attribute. This
// attribute must always be handled like a Hint.
bSet = m_bOutStyles
&& (!bParaAttrs || rItem.Which() == RES_CHRATR_BACKGROUND
|| rItem.Which() == RES_CHRATR_BOX || rItem.Which() == RES_CHRATR_OVERLINE);
break;
case HTML_CHRFMT_VALUE:
{
OSL_ENSURE( RES_TXTATR_CHARFMT == rItem.Which(),
"Not a character style after all" );
const SwFormatCharFormat& rChrFormat = rItem.StaticWhichCast(RES_TXTATR_CHARFMT);
const SwCharFormat* pFormat = rChrFormat.GetCharFormat();
const SwHTMLFormatInfo *pFormatInfo = GetFormatInfo( *pFormat, rFormatInfos );
if( !pFormatInfo->aToken.isEmpty() )
{
// output the character style tag before the hard
// attributes
InsertItem( rItem, nStart, nEnd );
}
if( pFormatInfo->moItemSet )
{
Insert( *pFormatInfo->moItemSet, nStart, nEnd,
rFormatInfos, true, bParaAttrs );
}
}
break;
case HTML_AUTOFMT_VALUE:
{
OSL_ENSURE( RES_TXTATR_AUTOFMT == rItem.Which(),
"Not an automatic style, after all" );
const SwFormatAutoFormat& rAutoFormat = rItem.StaticWhichCast(RES_TXTATR_AUTOFMT);
const std::shared_ptr<SfxItemSet>& pSet = rAutoFormat.GetStyleHandle();
if( pSet )
Insert( *pSet, nStart, nEnd, rFormatInfos, true, bParaAttrs );
}
break;
case HTML_COLOR_VALUE:
// A foreground color as a paragraph attribute is only exported if
// it is not the same as the default color.
{
OSL_ENSURE( RES_CHRATR_COLOR == rItem.Which(),
"Not a foreground color, after all" );
Color aColor( rItem.StaticWhichCast(RES_CHRATR_COLOR).GetValue() );
if( COL_AUTO == aColor )
aColor = COL_BLACK;
bSet = !bParaAttrs || !m_xDefaultColor || !m_xDefaultColor->IsRGBEqual(aColor);
}
break;
case HTML_DROPCAP_VALUE:
{
OSL_ENSURE( RES_PARATR_DROP == rItem.Which(),
"Not a drop cap, after all" );
const SwFormatDrop& rDrop = rItem.StaticWhichCast(RES_PARATR_DROP);
nEnd = nStart + rDrop.GetChars();
if (!m_bOutStyles)
{
// At least use the attributes of the character style
const SwCharFormat *pCharFormat = rDrop.GetCharFormat();
if( pCharFormat )
{
Insert( pCharFormat->GetAttrSet(), nStart, nEnd,
rFormatInfos, true, bParaAttrs );
}
}
else
{
bSet = true;
}
}
break;
default:
;
}
if( bSet )
InsertItem( rItem, nStart, nEnd );
if( bSplit )
SplitItem( rItem, nStart, nEnd );
}
void HTMLEndPosLst::Insert( const SfxPoolItem& rItem,
sal_Int32 nStart, sal_Int32 nEnd,
SwHTMLFormatInfos& rFormatInfos, bool bParaAttrs )
{
bool bDependsOnScript = false, bDependsOnAnyScript = false;
sal_uInt16 nScript = i18n::ScriptType::LATIN;
switch( rItem.Which() )
{
case RES_CHRATR_FONT:
case RES_CHRATR_FONTSIZE:
case RES_CHRATR_LANGUAGE:
case RES_CHRATR_POSTURE:
case RES_CHRATR_WEIGHT:
bDependsOnScript = true;
nScript = i18n::ScriptType::LATIN;
break;
case RES_CHRATR_CJK_FONT:
case RES_CHRATR_CJK_FONTSIZE:
case RES_CHRATR_CJK_LANGUAGE:
case RES_CHRATR_CJK_POSTURE:
case RES_CHRATR_CJK_WEIGHT:
bDependsOnScript = true;
nScript = i18n::ScriptType::ASIAN;
break;
case RES_CHRATR_CTL_FONT:
case RES_CHRATR_CTL_FONTSIZE:
case RES_CHRATR_CTL_LANGUAGE:
case RES_CHRATR_CTL_POSTURE:
case RES_CHRATR_CTL_WEIGHT:
bDependsOnScript = true;
nScript = i18n::ScriptType::COMPLEX;
break;
case RES_TXTATR_CHARFMT:
{
const SwFormatCharFormat& rChrFormat = rItem.StaticWhichCast(RES_TXTATR_CHARFMT);
const SwCharFormat* pFormat = rChrFormat.GetCharFormat();
const SwHTMLFormatInfo *pFormatInfo = GetFormatInfo( *pFormat, rFormatInfos );
if( pFormatInfo->bScriptDependent )
{
bDependsOnScript = true;
bDependsOnAnyScript = true;
}
}
break;
case RES_TXTATR_INETFMT:
{
if (GetFormatInfo(*m_pDoc->getIDocumentStylePoolAccess().GetCharFormatFromPool(
RES_POOLCHR_INET_NORMAL),
rFormatInfos)
->bScriptDependent
|| GetFormatInfo(*m_pDoc->getIDocumentStylePoolAccess().GetCharFormatFromPool(
RES_POOLCHR_INET_VISIT),
rFormatInfos)
->bScriptDependent)
{
bDependsOnScript = true;
bDependsOnAnyScript = true;
}
}
break;
}
if( bDependsOnScript )
{
sal_Int32 nPos = nStart;
for (size_t i = 0; i < m_aScriptChgLst.size(); i++)
{
sal_Int32 nChgPos = m_aScriptChgLst[i];
if( nPos >= nChgPos )
{
// the hint starts behind or at the next script change,
// so we may continue with this position.
continue;
}
if( nEnd <= nChgPos )
{
// the (rest of) the hint ends before or at the next script
// change, so we can insert it, but only if it belongs
// to the current script.
if (bDependsOnAnyScript || nScript == m_aScriptLst[i])
InsertNoScript( rItem, nPos, nEnd, rFormatInfos,
bParaAttrs );
break;
}
// the hint starts before the next script change and ends behind
// it, so we can insert a hint up to the next script change and
// continue with the rest of the hint.
if (bDependsOnAnyScript || nScript == m_aScriptLst[i])
InsertNoScript( rItem, nPos, nChgPos, rFormatInfos, bParaAttrs );
nPos = nChgPos;
}
}
else
{
InsertNoScript( rItem, nStart, nEnd, rFormatInfos, bParaAttrs );
}
}
void HTMLEndPosLst::Insert( const SfxItemSet& rItemSet,
sal_Int32 nStart, sal_Int32 nEnd,
SwHTMLFormatInfos& rFormatInfos,
bool bDeep, bool bParaAttrs )
{
SfxWhichIter aIter( rItemSet );
sal_uInt16 nWhich = aIter.FirstWhich();
while( nWhich )
{
const SfxPoolItem *pItem;
if( SfxItemState::SET == aIter.GetItemState( bDeep, &pItem ) )
{
Insert( *pItem, nStart, nEnd, rFormatInfos, bParaAttrs );
}
nWhich = aIter.NextWhich();
}
}
void HTMLEndPosLst::Insert( const SwDrawFrameFormat& rFormat, sal_Int32 nPos,
SwHTMLFormatInfos& rFormatInfos )
{
const SdrObject* pTextObj = SwHTMLWriter::GetMarqueeTextObj( rFormat );
if( !pTextObj )
return;
// get the edit engine attributes of the object as SW attributes and
// insert them as hints. Because of the amount of Hints the styles
// are not considered!
const SfxItemSet& rFormatItemSet = rFormat.GetAttrSet();
SfxItemSetFixed<RES_CHRATR_BEGIN, RES_CHRATR_END> aItemSet( *rFormatItemSet.GetPool() );
SwHTMLWriter::GetEEAttrsFromDrwObj( aItemSet, pTextObj );
bool bOutStylesOld = m_bOutStyles;
m_bOutStyles = false;
Insert( aItemSet, nPos, nPos+1, rFormatInfos, false );
m_bOutStyles = bOutStylesOld;
}
sal_uInt16 HTMLEndPosLst::GetScriptAtPos( sal_Int32 nPos, sal_uInt16 nWeak )
{
sal_uInt16 nRet = CSS1_OUTMODE_ANY_SCRIPT;
size_t nScriptChgs = m_aScriptChgLst.size();
size_t i=0;
while (i < nScriptChgs && nPos >= m_aScriptChgLst[i])
i++;
OSL_ENSURE( i < nScriptChgs, "script list is too short" );
if( i < nScriptChgs )
{
if (i18n::ScriptType::WEAK == m_aScriptLst[i])
nRet = nWeak;
else
nRet = SwHTMLWriter::GetCSS1ScriptForScriptType(m_aScriptLst[i]);
}
return nRet;
}
void HTMLEndPosLst::OutStartAttrs( SwHTMLWriter& rWrt, sal_Int32 nPos )
{
rWrt.m_bTagOn = true;
auto it = m_aStartLst.find(nPos);
if (it == m_aStartLst.end())
return;
// the attributes of the start list are sorted in ascending order
for (HTMLStartEndPos* pPos : it->second)
{
// output the attribute
sal_uInt16 nCSS1Script = rWrt.m_nCSS1Script;
sal_uInt16 nWhich = pPos->GetItem().Which();
if( RES_TXTATR_CHARFMT == nWhich ||
RES_TXTATR_INETFMT == nWhich ||
RES_PARATR_DROP == nWhich )
{
rWrt.m_nCSS1Script = GetScriptAtPos( nPos, nCSS1Script );
}
HTMLOutFuncs::FlushToAscii( rWrt.Strm() ); // was one time only - do we still need it?
Out( aHTMLAttrFnTab, pPos->GetItem(), rWrt );
rWrt.maStartedAttributes[pPos->GetItem().Which()]++;
rWrt.m_nCSS1Script = nCSS1Script;
}
}
void HTMLEndPosLst::OutEndAttrs( SwHTMLWriter& rWrt, sal_Int32 nPos )
{
rWrt.m_bTagOn = false;
if (nPos == SAL_MAX_INT32)
{
for (auto& element : m_aEndLst)
OutEndAttrs(rWrt, element.second);
}
else
{
auto it = m_aEndLst.find(nPos);
if (it != m_aEndLst.end())
OutEndAttrs(rWrt, it->second);
}
}
void HTMLEndPosLst::OutEndAttrs(SwHTMLWriter& rWrt, std::vector<HTMLStartEndPos*>& posItems)
{
std::sort(posItems.begin(), posItems.end(), SortEnds(m_aStartLst));
for (auto it = posItems.begin(); it != posItems.end(); it = posItems.erase(it))
{
HTMLStartEndPos* pPos = *it;
HTMLOutFuncs::FlushToAscii( rWrt.Strm() ); // was one time only - do we still need it?
// Skip closing span if next character span has the same border (border merge)
bool bSkipOut = false;
if( pPos->GetItem().Which() == RES_CHRATR_BOX )
{
auto& startPosItems = m_aStartLst[pPos->GetEnd()];
for (auto it2 = startPosItems.begin(); it2 != startPosItems.end(); ++it2)
{
HTMLStartEndPos* pEndPos = *it2;
if( pEndPos->GetItem().Which() == RES_CHRATR_BOX &&
static_cast<const SvxBoxItem&>(pEndPos->GetItem()) ==
static_cast<const SvxBoxItem&>(pPos->GetItem()) )
{
startPosItems.erase(it2);
pEndPos->SetStart(pPos->GetStart());
auto& oldStartPosItems = m_aStartLst[pEndPos->GetStart()];
oldStartPosItems.insert(oldStartPosItems.begin(), pEndPos);
bSkipOut = true;
break;
}
}
}
if( !bSkipOut )
{
Out( aHTMLAttrFnTab, pPos->GetItem(), rWrt );
rWrt.maStartedAttributes[pPos->GetItem().Which()]--;
}
std::erase(m_aStartLst[pPos->GetStart()], pPos);
delete pPos;
}
}
static constexpr bool IsLF(sal_Unicode ch) { return ch == '\n'; }
static constexpr bool IsWhitespaceExcludingLF(sal_Unicode ch)
{
return ch == ' ' || ch == '\t' || ch == '\r';
}
static constexpr bool IsWhitespaceIncludingLF(sal_Unicode ch)
{
return IsWhitespaceExcludingLF(ch) || IsLF(ch);
}
static bool NeedPreserveWhitespace(std::u16string_view str, bool xml)
{
if (str.empty())
return false;
// leading / trailing spaces
// A leading / trailing \n would turn into a leading / trailing <br/>,
// and will not disappear, even without space preserving option
if (IsWhitespaceExcludingLF(str.front()) || IsWhitespaceExcludingLF(str.back()))
return true;
for (size_t i = 0; i < str.size(); ++i)
{
if (xml)
{
// No need to consider \n, which convert to <br/>, when it's after a space
// (but handle it *before* a space)
if (IsWhitespaceIncludingLF(str[i]))
{
do
{
++i;
if (i == str.size())
return false;
} while (IsLF(str[i]));
if (IsWhitespaceExcludingLF(str[i]))
return true; // Second whitespace in a row
}
}
else // html
{
// Only consider \n, when an adjacent space is not \n - which would be eaten
// without a space preserving option
if (IsWhitespaceExcludingLF(str[i]))
{
++i;
if (i == str.size())
return false;
if (IsWhitespaceIncludingLF(str[i]))
return true; // Any whitespace after a non-LF whitespace
}
else if (IsLF(str[i]))
{
do
{
++i;
if (i == str.size())
return false;
}
while (IsLF(str[i]));
if (IsWhitespaceExcludingLF(str[i]))
return true; // A non-LF whitespace after a LF
}
}
}
return false;
}
/* Output of the nodes*/
SwHTMLWriter& OutHTML_SwTextNode( SwHTMLWriter& rWrt, const SwContentNode& rNode )
{
const SwTextNode * pNd = &static_cast<const SwTextNode&>(rNode);
const OUString& rStr = pNd->GetText();
sal_Int32 nEnd = rStr.getLength();
// special case: empty node and HR style (horizontal rule)
// output a <HR>, only
sal_uInt16 nPoolId = pNd->GetAnyFormatColl().GetPoolFormatId();
// Handle horizontal rule <hr>
if (!nEnd &&
(RES_POOLCOLL_HTML_HR==nPoolId || pNd->GetAnyFormatColl().GetName() == OOO_STRING_SVTOOLS_HTML_horzrule))
{
// then, the paragraph-anchored graphics/OLE objects in the paragraph
// MIB 8.7.97: We enclose the line in a <PRE>. This means that the
// spacings are wrong, but otherwise we get an empty paragraph
// after the <HR> which is even uglier.
rWrt.ChangeParaToken( HtmlTokenId::NONE );
// Output all the nodes that are anchored to a frame
rWrt.OutFlyFrame( rNode.GetIndex(), 0, HtmlPosition::Any );
if (rWrt.IsLFPossible())
rWrt.OutNewLine(); // paragraph tag on a new line
rWrt.SetLFPossible(true);
HtmlWriter aHtml(rWrt.Strm(), rWrt.maNamespace);
aHtml.prettyPrint(rWrt.IsPrettyPrint());
aHtml.start(OOO_STRING_SVTOOLS_HTML_horzrule ""_ostr);
const SfxItemSet* pItemSet = pNd->GetpSwAttrSet();
if( !pItemSet )
{
aHtml.end();
return rWrt;
}
if (pItemSet->GetItemIfSet(RES_MARGIN_FIRSTLINE, false)
|| pItemSet->GetItemIfSet(RES_MARGIN_TEXTLEFT, false)
|| pItemSet->GetItemIfSet(RES_MARGIN_RIGHT, false))
{
SvxFirstLineIndentItem const& rFirstLine(pItemSet->Get(RES_MARGIN_FIRSTLINE));
SvxTextLeftMarginItem const& rTextLeftMargin(pItemSet->Get(RES_MARGIN_TEXTLEFT));
SvxRightMarginItem const& rRightMargin(pItemSet->Get(RES_MARGIN_RIGHT));
sal_Int32 const nLeft(rTextLeftMargin.GetLeft(rFirstLine, /*metrics*/ {}));
sal_Int32 const nRight(rRightMargin.GetRight());
if( nLeft || nRight )
{
const SwFrameFormat& rPgFormat =
rWrt.m_pDoc->getIDocumentStylePoolAccess().GetPageDescFromPool
( RES_POOLPAGE_HTML, false )->GetMaster();
const SwFormatFrameSize& rSz = rPgFormat.GetFrameSize();
const SvxLRSpaceItem& rLR = rPgFormat.GetLRSpace();
const SwFormatCol& rCol = rPgFormat.GetCol();
tools::Long nPageWidth = rSz.GetWidth() - rLR.GetLeft() - rLR.GetRight();
if( 1 < rCol.GetNumCols() )
nPageWidth /= rCol.GetNumCols();
const SwTableNode* pTableNd = pNd->FindTableNode();
if( pTableNd )
{
const SwTableBox* pBox = pTableNd->GetTable().GetTableBox(
pNd->StartOfSectionIndex() );
if( pBox )
nPageWidth = pBox->GetFrameFormat()->GetFrameSize().GetWidth();
}
OString sWidth = OString::number(SwHTMLWriter::ToPixel(nPageWidth - nLeft - nRight));
aHtml.attribute(OOO_STRING_SVTOOLS_HTML_O_width, sWidth);
if( !nLeft )
aHtml.attribute(OOO_STRING_SVTOOLS_HTML_O_align, OOO_STRING_SVTOOLS_HTML_AL_left);
else if( !nRight )
aHtml.attribute(OOO_STRING_SVTOOLS_HTML_O_align, OOO_STRING_SVTOOLS_HTML_AL_right);
else
aHtml.attribute(OOO_STRING_SVTOOLS_HTML_O_align, OOO_STRING_SVTOOLS_HTML_AL_center);
}
}
if( const SvxBoxItem* pBoxItem = pItemSet->GetItemIfSet( RES_BOX, false ))
{
const editeng::SvxBorderLine* pBorderLine = pBoxItem->GetBottom();
if( pBorderLine )
{
sal_uInt16 nWidth = pBorderLine->GetScaledWidth();
OString sWidth = OString::number(SwHTMLWriter::ToPixel(nWidth));
aHtml.attribute(OOO_STRING_SVTOOLS_HTML_O_size, sWidth);
const Color& rBorderColor = pBorderLine->GetColor();
if( !rBorderColor.IsRGBEqual( COL_GRAY ) )
{
HtmlWriterHelper::applyColor(aHtml, OOO_STRING_SVTOOLS_HTML_O_color, rBorderColor);
}
if( !pBorderLine->GetInWidth() )
{
aHtml.attribute(OOO_STRING_SVTOOLS_HTML_O_noshade, OOO_STRING_SVTOOLS_HTML_O_noshade);
}
}
}
aHtml.end();
return rWrt;
}
// Do not export the empty nodes with 2pt fonts and standard style that
// are inserted before tables and sections, but do export bookmarks
// and paragraph anchored frames.
if( !nEnd && (nPoolId == RES_POOLCOLL_STANDARD ||
nPoolId == RES_POOLCOLL_TABLE ||
nPoolId == RES_POOLCOLL_TABLE_HDLN) )
{
// The current node is empty and contains the standard style ...
const SvxFontHeightItem* pFontHeightItem;
const SfxItemSet* pItemSet = pNd->GetpSwAttrSet();
if( pItemSet && pItemSet->Count() &&
(pFontHeightItem = pItemSet->GetItemIfSet( RES_CHRATR_FONTSIZE, false )) &&
40 == pFontHeightItem->GetHeight() )
{
// ... moreover, the 2pt font is set ...
SwNodeOffset nNdPos = rWrt.m_pCurrentPam->GetPoint()->GetNodeIndex();
const SwNode *pNextNd = rWrt.m_pDoc->GetNodes()[nNdPos+1];
const SwNode *pPrevNd = rWrt.m_pDoc->GetNodes()[nNdPos-1];
bool bStdColl = nPoolId == RES_POOLCOLL_STANDARD;
if( ( bStdColl && (pNextNd->IsTableNode() || pNextNd->IsSectionNode()) ) ||
( !bStdColl &&
pNextNd->IsEndNode() &&
pPrevNd->IsStartNode() &&
SwTableBoxStartNode == pPrevNd->GetStartNode()->GetStartNodeType() ) )
{
// ... and it is located before a table or a section
rWrt.OutBookmarks();
rWrt.SetLFPossible(rWrt.m_nLastParaToken == HtmlTokenId::NONE);
// Output all frames that are anchored to this node
rWrt.OutFlyFrame( rNode.GetIndex(), 0, HtmlPosition::Any );
rWrt.SetLFPossible(false);
return rWrt;
}
}
}
// catch PageBreaks and PageDescs
bool bPageBreakBehind = false;
if( rWrt.m_bCfgFormFeed &&
!(rWrt.m_bOutTable || rWrt.m_bOutFlyFrame) &&
rWrt.m_pStartNdIdx->GetIndex() != rWrt.m_pCurrentPam->GetPoint()->GetNodeIndex() )
{
bool bPageBreakBefore = false;
const SfxItemSet* pItemSet = pNd->GetpSwAttrSet();
if( pItemSet )
{
const SwFormatPageDesc* pPageDescItem = pItemSet->GetItemIfSet( RES_PAGEDESC );
if( pPageDescItem && pPageDescItem->GetPageDesc() )
{
bPageBreakBefore = true;
}
else if( const SvxFormatBreakItem* pItem = pItemSet->GetItemIfSet( RES_BREAK ) )
{
switch( pItem->GetBreak() )
{
case SvxBreak::PageBefore:
bPageBreakBefore = true;
break;
case SvxBreak::PageAfter:
bPageBreakBehind = true;
break;
case SvxBreak::PageBoth:
bPageBreakBefore = true;
bPageBreakBehind = true;
break;
default:
break;
}
}
}
if( bPageBreakBefore )
rWrt.Strm().WriteChar( '\f' );
}
// if necessary, open a form
rWrt.OutForm();
// Output the page-anchored frames that are 'anchored' to this node
bool bFlysLeft = rWrt.OutFlyFrame( rNode.GetIndex(), 0, HtmlPosition::Prefix );
// Output all frames that are anchored to this node that are supposed to
// be written before the paragraph tag.
if( bFlysLeft )
{
bFlysLeft = rWrt.OutFlyFrame( rNode.GetIndex(), 0, HtmlPosition::Before );
}
if( rWrt.m_pCurrentPam->GetPoint()->GetNode() == rWrt.m_pCurrentPam->GetMark()->GetNode() )
{
nEnd = rWrt.m_pCurrentPam->GetMark()->GetContentIndex();
}
// are there any hard attributes that must be written as options?
rWrt.m_bTagOn = true;
// now, output the tag of the paragraph
const SwFormat& rFormat = pNd->GetAnyFormatColl();
SwHTMLTextCollOutputInfo aFormatInfo;
bool bOldLFPossible = rWrt.IsLFPossible();
bool bOldSpacePreserve = rWrt.IsSpacePreserve();
if (rWrt.IsPreserveSpacesOnWritePrefSet())
rWrt.SetSpacePreserve(NeedPreserveWhitespace(rStr, rWrt.mbReqIF));
OutHTML_SwFormat( rWrt, rFormat, pNd->GetpSwAttrSet(), aFormatInfo );
// If we didn't open a new line before the paragraph tag, we do that now
rWrt.SetLFPossible(rWrt.m_nLastParaToken == HtmlTokenId::NONE);
if (!bOldLFPossible && rWrt.IsLFPossible())
rWrt.OutNewLine();
// then, the bookmarks (including end tag)
rWrt.m_bOutOpts = false;
rWrt.OutBookmarks();
// now it's a good opportunity again for an LF - if it is still allowed
// FIXME: for LOK case we set rWrt.m_nWishLineLen as -1, for now keep old flow
// when LOK side will be fixed - don't insert new line at the beginning
if( rWrt.IsLFPossible() && rWrt.IsPrettyPrint() && rWrt.m_nWishLineLen >= 0 &&
rWrt.GetLineLen() >= rWrt.m_nWishLineLen )
{
rWrt.OutNewLine();
}
rWrt.SetLFPossible(false);
// find text that originates from an outline numbering
sal_Int32 nOffset = 0;
OUString aOutlineText;
OUString aFullText;
// export numbering string as plain text only for the outline numbering,
// because the outline numbering isn't exported as a numbering - see <SwHTMLNumRuleInfo::Set(..)>
if ( pNd->IsOutline() &&
pNd->GetNumRule() == pNd->GetDoc().GetOutlineNumRule() )
{
aOutlineText = pNd->GetNumString();
nOffset = nOffset + aOutlineText.getLength();
aFullText = aOutlineText;
}
OUString aFootEndNoteSym;
if( rWrt.m_pFormatFootnote )
{
aFootEndNoteSym = rWrt.GetFootEndNoteSym( *rWrt.m_pFormatFootnote );
nOffset = nOffset + aFootEndNoteSym.getLength();
aFullText += aFootEndNoteSym;
}
// Table of Contents or other paragraph with dot leaders?
sal_Int32 nIndexTab = rWrt.indexOfDotLeaders( nPoolId, rStr );
if (nIndexTab > -1)
// skip part after the tabulator (page number)
nEnd = nIndexTab;
// are there any hard attributes that must be written as tags?
aFullText += rStr;
HTMLEndPosLst aEndPosLst( rWrt.m_pDoc, rWrt.m_xTemplate.get(),
rWrt.m_xDfltColor, rWrt.m_bCfgOutStyles,
rWrt.GetHTMLMode(), aFullText,
rWrt.m_aScriptTextStyles );
if( aFormatInfo.moItemSet )
{
aEndPosLst.Insert( *aFormatInfo.moItemSet, 0, nEnd + nOffset,
rWrt.m_CharFormatInfos, false, true );
}
if( !aOutlineText.isEmpty() || rWrt.m_pFormatFootnote )
{
// output paragraph attributes, so that the text gets the attributes of
// the paragraph.
aEndPosLst.OutStartAttrs( rWrt, 0 );
// Theoretically, we would have to consider the character style of
// the numbering. Because it cannot be set via the UI, let's ignore
// it for now.
if( !aOutlineText.isEmpty() )
HTMLOutFuncs::Out_String( rWrt.Strm(), aOutlineText );
if( rWrt.m_pFormatFootnote )
{
rWrt.OutFootEndNoteSym( *rWrt.m_pFormatFootnote, aFootEndNoteSym,
aEndPosLst.GetScriptAtPos( aOutlineText.getLength(), rWrt.m_nCSS1Script ) );
rWrt.m_pFormatFootnote = nullptr;
}
}
// for now, correct the start. I.e., if we only output part of the sentence,
// the attributes must be correct there, as well!!
rWrt.m_bTextAttr = true;
size_t nAttrPos = 0;
sal_Int32 nStrPos = rWrt.m_pCurrentPam->GetPoint()->GetContentIndex();
const SwTextAttr * pHt = nullptr;
const size_t nCntAttr = pNd->HasHints() ? pNd->GetSwpHints().Count() : 0;
if( nCntAttr && nStrPos > ( pHt = pNd->GetSwpHints().Get(0) )->GetStart() )
{
// Ok, there are earlier attributes that we must output
do {
aEndPosLst.OutEndAttrs( rWrt, nStrPos + nOffset );
nAttrPos++;
if( pHt->Which() == RES_TXTATR_FIELD
|| pHt->Which() == RES_TXTATR_ANNOTATION )
continue;
if ( pHt->End() && !pHt->HasDummyChar() )
{
const sal_Int32 nHtEnd = *pHt->End(),
nHtStt = pHt->GetStart();
if( !rWrt.m_bWriteAll && nHtEnd <= nStrPos )
continue;
// don't consider empty hints at the beginning - or should we ??
if( nHtEnd == nHtStt )
continue;
// add attribute to the list
if( rWrt.m_bWriteAll )
aEndPosLst.Insert( pHt->GetAttr(), nHtStt + nOffset,
nHtEnd + nOffset,
rWrt.m_CharFormatInfos );
else
{
sal_Int32 nTmpStt = nHtStt < nStrPos ? nStrPos : nHtStt;
sal_Int32 nTmpEnd = std::min(nHtEnd, nEnd);
aEndPosLst.Insert( pHt->GetAttr(), nTmpStt + nOffset,
nTmpEnd + nOffset,
rWrt.m_CharFormatInfos );
}
continue;
// but don't output it, that will be done later !!
}
} while( nAttrPos < nCntAttr && nStrPos >
( pHt = pNd->GetSwpHints().Get( nAttrPos ) )->GetStart() );
// so, let's output all collected attributes from the string pos on
aEndPosLst.OutEndAttrs( rWrt, nStrPos + nOffset );
aEndPosLst.OutStartAttrs( rWrt, nStrPos + nOffset );
}
bool bWriteBreak = (HtmlTokenId::PREFORMTXT_ON != rWrt.m_nLastParaToken);
if (bWriteBreak && (pNd->GetNumRule() || rWrt.mbReqIF))
{
// One line-break is exactly one <br> in the ReqIF case.
bWriteBreak = false;
}
{
// Tabs are leading till there is a non-tab since the start of the paragraph.
bool bLeadingTab = true;
for( ; nStrPos < nEnd; nStrPos++ )
{
// output the frames that are anchored to the current position
if( bFlysLeft )
{
aEndPosLst.OutEndAttrs( rWrt, nStrPos + nOffset );
bFlysLeft = rWrt.OutFlyFrame( rNode.GetIndex(),
nStrPos, HtmlPosition::Inside );
}
bool bOutChar = true;
const SwTextAttr * pTextHt = nullptr;
if (nAttrPos < nCntAttr && pHt->GetStart() == nStrPos)
{
do {
if ( pHt->End() && !pHt->HasDummyChar() )
{
if( *pHt->End() != nStrPos )
{
// insert hints with end, if they don't start
// an empty range (hints that don't start a range
// are ignored)
aEndPosLst.Insert( pHt->GetAttr(), nStrPos + nOffset,
*pHt->End() + nOffset,
rWrt.m_CharFormatInfos );
}
}
else
{
// hints without an end are output last
OSL_ENSURE( !pTextHt, "Why is there already an attribute without an end?" );
if( rWrt.m_nTextAttrsToIgnore>0 )
{
rWrt.m_nTextAttrsToIgnore--;
}
else
{
pTextHt = pHt;
SwFieldIds nFieldWhich;
if( RES_TXTATR_FIELD != pHt->Which()
|| ( SwFieldIds::Postit != (nFieldWhich = static_cast<const SwFormatField&>(pHt->GetAttr()).GetField()->Which())
&& SwFieldIds::Script != nFieldWhich ) )
{
bWriteBreak = false;
}
}
bOutChar = false; // don't output 255
}
} while( ++nAttrPos < nCntAttr && nStrPos ==
( pHt = pNd->GetSwpHints().Get( nAttrPos ) )->GetStart() );
}
// Additionally, some draw formats can bring attributes
if( pTextHt && RES_TXTATR_FLYCNT == pTextHt->Which() )
{
const SwFrameFormat* pFrameFormat =
pTextHt->GetAttr().StaticWhichCast(RES_TXTATR_FLYCNT).GetFrameFormat();
if( RES_DRAWFRMFMT == pFrameFormat->Which() )
aEndPosLst.Insert( *static_cast<const SwDrawFrameFormat *>(pFrameFormat),
nStrPos + nOffset,
rWrt.m_CharFormatInfos );
}
aEndPosLst.OutEndAttrs( rWrt, nStrPos + nOffset );
aEndPosLst.OutStartAttrs( rWrt, nStrPos + nOffset );
if( pTextHt )
{
rWrt.SetLFPossible(rWrt.m_nLastParaToken == HtmlTokenId::NONE &&
nStrPos > 0 &&
rStr[nStrPos-1] == ' ');
sal_uInt16 nCSS1Script = rWrt.m_nCSS1Script;
rWrt.m_nCSS1Script = aEndPosLst.GetScriptAtPos(
nStrPos + nOffset, nCSS1Script );
HTMLOutFuncs::FlushToAscii( rWrt.Strm() );
Out( aHTMLAttrFnTab, pTextHt->GetAttr(), rWrt );
rWrt.m_nCSS1Script = nCSS1Script;
rWrt.SetLFPossible(false);
}
if( bOutChar )
{
sal_uInt32 c = rStr[nStrPos];
if( rtl::isHighSurrogate(c) && nStrPos < nEnd - 1 )
{
const sal_Unicode d = rStr[nStrPos + 1];
if( rtl::isLowSurrogate(d) )
{
c = rtl::combineSurrogates(c, d);
nStrPos++;
}
}
// try to split a line after about 255 characters
// at a space character unless in a PRE-context
if( ' ' == c && rWrt.m_nLastParaToken == HtmlTokenId::NONE && !rWrt.IsSpacePreserve() )
{
sal_Int32 nLineLen;
nLineLen = rWrt.GetLineLen();
sal_Int32 nWordLen = rStr.indexOf( ' ', nStrPos+1 );
if( nWordLen == -1 )
nWordLen = nEnd;
nWordLen -= nStrPos;
if( rWrt.IsPrettyPrint() && rWrt.m_nWishLineLen >= 0 &&
(nLineLen >= rWrt.m_nWishLineLen ||
(nLineLen+nWordLen) >= rWrt.m_nWishLineLen ) )
{
HTMLOutFuncs::FlushToAscii( rWrt.Strm() );
rWrt.OutNewLine();
bOutChar = false;
}
}
if( bOutChar )
{
if( 0x0a == c )
{
HTMLOutFuncs::FlushToAscii( rWrt.Strm() );
HtmlWriter aHtml(rWrt.Strm(), rWrt.maNamespace);
aHtml.prettyPrint(rWrt.IsPrettyPrint());
aHtml.single(OOO_STRING_SVTOOLS_HTML_linebreak ""_ostr);
}
else if (c == CH_TXT_ATR_FORMELEMENT)
{
// Placeholder for a single-point fieldmark.
SwPosition aMarkPos = *rWrt.m_pCurrentPam->GetPoint();
aMarkPos.AdjustContent( nStrPos - aMarkPos.GetContentIndex() );
rWrt.OutPointFieldmarks(aMarkPos);
}
else
{
bool bConsumed = false;
if (c == '\t')
{
if (bLeadingTab && rWrt.m_nLeadingTabWidth.has_value())
{
// Consume a tab if it's leading and we know the number of NBSPs to
// be used as a replacement.
for (sal_Int32 i = 0; i < *rWrt.m_nLeadingTabWidth; ++i)
{
rWrt.Strm().WriteOString(" ");
}
bConsumed = true;
}
}
else
{
// Not a tab -> later tabs are no longer leading.
bLeadingTab = false;
}
if (!bConsumed)
{
HTMLOutFuncs::Out_Char(rWrt.Strm(), c);
}
}
if (!rWrt.mbReqIF)
{
// if a paragraph's last character is a hard line break
// then we need to add an extra <br>
// because browsers like Mozilla wouldn't add a line for the next paragraph
bWriteBreak = (0x0a == c) &&
(HtmlTokenId::PREFORMTXT_ON != rWrt.m_nLastParaToken);
}
}
}
}
HTMLOutFuncs::FlushToAscii( rWrt.Strm() );
}
aEndPosLst.OutEndAttrs( rWrt, SAL_MAX_INT32 );
// Output the frames that are anchored to the last position
if( bFlysLeft )
bFlysLeft = rWrt.OutFlyFrame( rNode.GetIndex(),
nEnd, HtmlPosition::Inside );
OSL_ENSURE( !bFlysLeft, "Not all frames were saved!" );
rWrt.m_bTextAttr = false;
if( bWriteBreak )
{
bool bEndOfCell = rWrt.m_bOutTable &&
rWrt.m_pCurrentPam->GetPoint()->GetNodeIndex() ==
rWrt.m_pCurrentPam->GetMark()->GetNodeIndex();
if( bEndOfCell && !nEnd &&
rWrt.IsHTMLMode(HTMLMODE_NBSP_IN_TABLES) )
{
// If the last paragraph of a table cell is empty and we export
// for the MS-IE, we write a instead of a <BR>
rWrt.Strm().WriteChar( '&' ).WriteOString( OOO_STRING_SVTOOLS_HTML_S_nbsp ).WriteChar( ';' );
}
else
{
HtmlWriter aHtml(rWrt.Strm(), rWrt.maNamespace);
aHtml.prettyPrint(rWrt.IsPrettyPrint());
aHtml.single(OOO_STRING_SVTOOLS_HTML_linebreak ""_ostr);
const SvxULSpaceItem& rULSpace = pNd->GetSwAttrSet().Get(RES_UL_SPACE);
if (rULSpace.GetLower() > 0 && !bEndOfCell)
{
aHtml.single(OOO_STRING_SVTOOLS_HTML_linebreak ""_ostr);
}
rWrt.SetLFPossible(true);
}
}
if( rWrt.m_bClearLeft || rWrt.m_bClearRight )
{
const char* pString;
if( rWrt.m_bClearLeft )
{
if( rWrt.m_bClearRight )
pString = OOO_STRING_SVTOOLS_HTML_AL_all;
else
pString = OOO_STRING_SVTOOLS_HTML_AL_left;
}
else
{
pString = OOO_STRING_SVTOOLS_HTML_AL_right;
}
HtmlWriter aHtml(rWrt.Strm(), rWrt.maNamespace);
aHtml.prettyPrint(rWrt.IsPrettyPrint());
aHtml.start(OOO_STRING_SVTOOLS_HTML_linebreak ""_ostr);
aHtml.attribute(OOO_STRING_SVTOOLS_HTML_O_clear, pString);
aHtml.end();
rWrt.m_bClearLeft = false;
rWrt.m_bClearRight = false;
rWrt.SetLFPossible(true);
}
// if an LF is not allowed already, it is allowed once the paragraphs
// ends with a ' '
if (!rWrt.IsLFPossible() &&
rWrt.m_nLastParaToken == HtmlTokenId::NONE &&
nEnd > 0 && ' ' == rStr[nEnd-1] )
rWrt.SetLFPossible(true);
// dot leaders: print the skipped page number in a different span element
if (nIndexTab > -1) {
OString sOut = OUStringToOString(rStr.subView(nIndexTab + 1), RTL_TEXTENCODING_ASCII_US);
rWrt.Strm().WriteOString( Concat2View("</span><span>" + sOut + "</span>") );
}
rWrt.m_bTagOn = false;
OutHTML_SwFormatOff( rWrt, aFormatInfo );
rWrt.SetSpacePreserve(bOldSpacePreserve);
// if necessary, close a form
rWrt.OutForm( false );
if( bPageBreakBehind )
rWrt.Strm().WriteChar( '\f' );
return rWrt;
}
// In CSS, "px" is 1/96 of inch: https://www.w3.org/TR/css3-values/#absolute-lengths
sal_uInt32 SwHTMLWriter::ToPixel(sal_uInt32 nTwips)
{
// if there is a Twip, there should be a pixel as well
return nTwips
? std::max(o3tl::convert(nTwips, o3tl::Length::twip, o3tl::Length::px), sal_Int64(1))
: 0;
}
Size SwHTMLWriter::ToPixel(Size aTwips)
{
return Size(ToPixel(aTwips.Width()), ToPixel(aTwips.Height()));
}
static SwHTMLWriter& OutHTML_CSS1Attr( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
// if hints are currently written, we try to write the hint as an
// CSS1 attribute
if( rWrt.m_bCfgOutStyles && rWrt.m_bTextAttr )
OutCSS1_HintSpanTag( rWrt, rHt );
return rWrt;
}
/* File CHRATR.HXX: */
static SwHTMLWriter& OutHTML_SvxColor( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
if( !rWrt.m_bTextAttr && rWrt.m_bCfgOutStyles && rWrt.m_bCfgPreferStyles )
{
// don't write the font color as a tag, if styles are preferred to
// normal tags
return rWrt;
}
if( rWrt.m_bTagOn )
{
Color aColor( static_cast<const SvxColorItem&>(rHt).GetValue() );
if( COL_AUTO == aColor )
aColor = COL_BLACK;
if (rWrt.mbXHTML)
{
OString sOut = "<" + rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_span
" " OOO_STRING_SVTOOLS_HTML_O_style "=";
rWrt.Strm().WriteOString(sOut);
HTMLOutFuncs::Out_Color(rWrt.Strm(), aColor, /*bXHTML=*/true).WriteChar('>');
}
else
{
OString sOut = "<" + rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_font " "
OOO_STRING_SVTOOLS_HTML_O_color "=";
rWrt.Strm().WriteOString( sOut );
HTMLOutFuncs::Out_Color( rWrt.Strm(), aColor ).WriteChar( '>' );
}
}
else
{
if (rWrt.mbXHTML)
HTMLOutFuncs::Out_AsciiTag(
rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_span), false);
else
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_font), false );
}
return rWrt;
}
static SwHTMLWriter& OutHTML_SwPosture( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
const FontItalic nPosture = static_cast<const SvxPostureItem&>(rHt).GetPosture();
if( ITALIC_NORMAL == nPosture )
{
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_italic), rWrt.m_bTagOn );
}
else if( rWrt.m_bCfgOutStyles && rWrt.m_bTextAttr )
{
// maybe as CSS1 attribute?
OutCSS1_HintSpanTag( rWrt, rHt );
}
return rWrt;
}
static SwHTMLWriter& OutHTML_SvxFont( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
if (IgnorePropertyForReqIF(rWrt.mbReqIF, "font-family", ""))
{
return rWrt;
}
if( rWrt.m_bTagOn )
{
OUString aNames;
SwHTMLWriter::PrepareFontList( static_cast<const SvxFontItem&>(rHt), aNames, 0,
rWrt.IsHTMLMode(HTMLMODE_FONT_GENERIC) );
if (rWrt.mbXHTML)
{
OString sOut = "<" + rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_span
" " OOO_STRING_SVTOOLS_HTML_O_style "=\"font-family: ";
rWrt.Strm().WriteOString(sOut);
HTMLOutFuncs::Out_String(rWrt.Strm(), aNames)
.WriteOString("\">");
}
else
{
OString sOut = "<" + rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_font " "
OOO_STRING_SVTOOLS_HTML_O_face "=\"";
rWrt.Strm().WriteOString( sOut );
HTMLOutFuncs::Out_String( rWrt.Strm(), aNames )
.WriteOString( "\">" );
}
}
else
{
if (rWrt.mbXHTML)
HTMLOutFuncs::Out_AsciiTag(
rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_span), false);
else
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_font), false );
}
return rWrt;
}
static SwHTMLWriter& OutHTML_SvxFontHeight( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
if (IgnorePropertyForReqIF(rWrt.mbReqIF, "font-size", ""))
{
return rWrt;
}
if( rWrt.m_bTagOn )
{
if (rWrt.mbXHTML)
{
OString sOut = "<" + rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_span;
sal_uInt32 nHeight = static_cast<const SvxFontHeightItem&>(rHt).GetHeight();
// Twips -> points.
sal_uInt16 nSize = nHeight / 20;
sOut += " " OOO_STRING_SVTOOLS_HTML_O_style "=\"font-size: "
+ OString::number(static_cast<sal_Int32>(nSize)) + "pt\"";
rWrt.Strm().WriteOString(sOut);
}
else
{
OString sOut = "<" + rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_font;
sal_uInt32 nHeight = static_cast<const SvxFontHeightItem&>(rHt).GetHeight();
sal_uInt16 nSize = rWrt.GetHTMLFontSize( nHeight );
sOut += " " OOO_STRING_SVTOOLS_HTML_O_size "=\"" +
OString::number(static_cast<sal_Int32>(nSize)) + "\"";
rWrt.Strm().WriteOString( sOut );
if( rWrt.m_bCfgOutStyles && rWrt.m_bTextAttr )
{
// always export font size as CSS option, too
OutCSS1_HintStyleOpt( rWrt, rHt );
}
}
rWrt.Strm().WriteChar( '>' );
}
else
{
if (rWrt.mbXHTML)
HTMLOutFuncs::Out_AsciiTag(
rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_span), false);
else
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_font), false );
}
return rWrt;
}
static SwHTMLWriter& OutHTML_SvxLanguage( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
LanguageType eLang = static_cast<const SvxLanguageItem &>(rHt).GetLanguage();
if( LANGUAGE_DONTKNOW == eLang )
return rWrt;
if( rWrt.m_bTagOn )
{
OString sOut = "<" + rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_span;
rWrt.Strm().WriteOString( sOut );
rWrt.OutLanguage( static_cast<const SvxLanguageItem &>(rHt).GetLanguage() );
rWrt.Strm().WriteChar( '>' );
}
else
{
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_span), false );
}
return rWrt;
}
static SwHTMLWriter& OutHTML_SwWeight( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
const FontWeight nBold = static_cast<const SvxWeightItem&>(rHt).GetWeight();
if( WEIGHT_BOLD == nBold )
{
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_bold), rWrt.m_bTagOn );
}
else if( rWrt.m_bCfgOutStyles && rWrt.m_bTextAttr )
{
// maybe as CSS1 attribute ?
OutCSS1_HintSpanTag( rWrt, rHt );
}
return rWrt;
}
static SwHTMLWriter& OutHTML_SwCrossedOut( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
// Because of Netscape, we output STRIKE and not S!
const FontStrikeout nStrike = static_cast<const SvxCrossedOutItem&>(rHt).GetStrikeout();
if( STRIKEOUT_NONE != nStrike && !rWrt.mbReqIF )
{
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_strike), rWrt.m_bTagOn );
}
else if( rWrt.m_bCfgOutStyles && rWrt.m_bTextAttr )
{
// maybe as CSS1 attribute?
OutCSS1_HintSpanTag( rWrt, rHt );
}
return rWrt;
}
static SwHTMLWriter& OutHTML_SvxEscapement( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
const SvxEscapement eEscape =
static_cast<SvxEscapement>(static_cast<const SvxEscapementItem&>(rHt).GetEnumValue());
OString aTag;
switch( eEscape )
{
case SvxEscapement::Superscript: aTag = OOO_STRING_SVTOOLS_HTML_superscript ""_ostr; break;
case SvxEscapement::Subscript: aTag = OOO_STRING_SVTOOLS_HTML_subscript ""_ostr; break;
default:
;
}
if( !aTag.isEmpty() )
{
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + aTag), rWrt.m_bTagOn );
}
else if( rWrt.m_bCfgOutStyles && rWrt.m_bTextAttr )
{
// maybe as CSS1 attribute?
OutCSS1_HintSpanTag( rWrt, rHt );
}
return rWrt;
}
static SwHTMLWriter& OutHTML_SwUnderline( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
const FontLineStyle eUnder = static_cast<const SvxUnderlineItem&>(rHt).GetLineStyle();
if( LINESTYLE_NONE != eUnder && !rWrt.mbReqIF )
{
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_underline), rWrt.m_bTagOn );
}
else if( rWrt.m_bCfgOutStyles && rWrt.m_bTextAttr )
{
// maybe as CSS1 attribute?
OutCSS1_HintSpanTag( rWrt, rHt );
}
return rWrt;
}
static SwHTMLWriter& OutHTML_SwFlyCnt( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
const SwFormatFlyCnt& rFlyCnt = static_cast<const SwFormatFlyCnt&>(rHt);
const SwFrameFormat& rFormat = *rFlyCnt.GetFrameFormat();
const SdrObject *pSdrObj = nullptr;
SwHTMLFrameType eType = rWrt.GuessFrameType( rFormat, pSdrObj );
AllHtmlFlags nMode = getHTMLOutFrameAsCharTable(eType, rWrt.m_nExportMode);
rWrt.OutFrameFormat( nMode, rFormat, pSdrObj );
return rWrt;
}
// This is now our Blink item. Blinking is activated by setting the item to
// true!
static SwHTMLWriter& OutHTML_SwBlink( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
if( static_cast<const SvxBlinkItem&>(rHt).GetValue() )
{
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_blink), rWrt.m_bTagOn );
}
else if( rWrt.m_bCfgOutStyles && rWrt.m_bTextAttr )
{
// maybe as CSS1 attribute?
OutCSS1_HintSpanTag( rWrt, rHt );
}
return rWrt;
}
SwHTMLWriter& OutHTML_INetFormat( SwHTMLWriter& rWrt, const SwFormatINetFormat& rINetFormat, bool bOn )
{
OUString aURL( rINetFormat.GetValue() );
const SvxMacroTableDtor *pMacTable = rINetFormat.GetMacroTable();
bool bEvents = pMacTable != nullptr && !pMacTable->empty();
// Anything to output at all?
if( aURL.isEmpty() && !bEvents && rINetFormat.GetName().isEmpty() )
return rWrt;
// bOn controls if we are writing the opening or closing tag
if( !bOn )
{
HTMLOutFuncs::Out_AsciiTag( rWrt.Strm(), Concat2View(rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_anchor), false );
return rWrt;
}
OString sOut("<" + rWrt.GetNamespace() + OOO_STRING_SVTOOLS_HTML_anchor);
bool bScriptDependent = false;
{
const SwCharFormat* pFormat = rWrt.m_pDoc->getIDocumentStylePoolAccess().GetCharFormatFromPool(
RES_POOLCHR_INET_NORMAL );
std::unique_ptr<SwHTMLFormatInfo> pFormatInfo(new SwHTMLFormatInfo(pFormat));
auto const it = rWrt.m_CharFormatInfos.find( pFormatInfo );
if (it != rWrt.m_CharFormatInfos.end())
{
bScriptDependent = (*it)->bScriptDependent;
}
}
if( !bScriptDependent )
{
const SwCharFormat* pFormat = rWrt.m_pDoc->getIDocumentStylePoolAccess().GetCharFormatFromPool(
RES_POOLCHR_INET_VISIT );
std::unique_ptr<SwHTMLFormatInfo> pFormatInfo(new SwHTMLFormatInfo(pFormat));
auto const it = rWrt.m_CharFormatInfos.find( pFormatInfo );
if (it != rWrt.m_CharFormatInfos.end())
{
bScriptDependent = (*it)->bScriptDependent;
}
}
if( bScriptDependent )
{
sOut += " " OOO_STRING_SVTOOLS_HTML_O_class "=\"";
const char* pStr = nullptr;
switch( rWrt.m_nCSS1Script )
{
case CSS1_OUTMODE_WESTERN:
pStr = "western";
break;
case CSS1_OUTMODE_CJK:
pStr = "cjk";
break;
case CSS1_OUTMODE_CTL:
pStr = "ctl";
break;
}
sOut += pStr + OString::Concat("\"");
}
rWrt.Strm().WriteOString( sOut );
sOut = ""_ostr;
OUString sRel;
if( !aURL.isEmpty() || bEvents )
{
OUString sTmp( aURL.toAsciiUpperCase() );
sal_Int32 nPos = sTmp.indexOf( "\" REL=" );
if( nPos >= 0 )
{
sRel = aURL.copy( nPos+1 );
aURL = aURL.copy( 0, nPos);
}
aURL = comphelper::string::strip(aURL, ' ');
sOut += " " OOO_STRING_SVTOOLS_HTML_O_href "=\"";
rWrt.Strm().WriteOString( sOut );
rWrt.OutHyperlinkHRefValue( aURL );
sOut = "\""_ostr;
}
if( !rINetFormat.GetName().isEmpty() )
{
sOut += " " OOO_STRING_SVTOOLS_HTML_O_name "=\"";
rWrt.Strm().WriteOString( sOut );
HTMLOutFuncs::Out_String( rWrt.Strm(), rINetFormat.GetName() );
sOut = "\""_ostr;
}
const OUString& rTarget = rINetFormat.GetTargetFrame();
if( !rTarget.isEmpty() )
{
sOut += " " OOO_STRING_SVTOOLS_HTML_O_target "=\"";
rWrt.Strm().WriteOString( sOut );
HTMLOutFuncs::Out_String( rWrt.Strm(), rTarget );
sOut = "\""_ostr;
}
if( !sRel.isEmpty() )
sOut += OUStringToOString(sRel, RTL_TEXTENCODING_ASCII_US);
if( !sOut.isEmpty() )
rWrt.Strm().WriteOString( sOut );
if( bEvents )
HTMLOutFuncs::Out_Events( rWrt.Strm(), *pMacTable, aAnchorEventTable,
rWrt.m_bCfgStarBasic );
rWrt.Strm().WriteOString( ">" );
return rWrt;
}
static SwHTMLWriter& OutHTML_SwFormatINetFormat( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
const SwFormatINetFormat& rINetFormat = static_cast<const SwFormatINetFormat&>(rHt);
if( rWrt.m_bTagOn )
{
// if necessary, temporarily close an attribute that is still open
if( !rWrt.m_aINetFormats.empty() )
{
SwFormatINetFormat *pINetFormat =
rWrt.m_aINetFormats.back();
OutHTML_INetFormat( rWrt, *pINetFormat, false );
}
// now, open the new one
OutHTML_INetFormat( rWrt, rINetFormat, true );
// and remember it
SwFormatINetFormat *pINetFormat = new SwFormatINetFormat( rINetFormat );
rWrt.m_aINetFormats.push_back( pINetFormat );
}
else
{
OutHTML_INetFormat( rWrt, rINetFormat, false );
OSL_ENSURE( rWrt.m_aINetFormats.size(), "there must be a URL attribute missing" );
if( !rWrt.m_aINetFormats.empty() )
{
// get its own attribute from the stack
SwFormatINetFormat *pINetFormat = rWrt.m_aINetFormats.back();
rWrt.m_aINetFormats.pop_back();
delete pINetFormat;
}
if( !rWrt.m_aINetFormats.empty() )
{
// there is still an attribute on the stack that must be reopened
SwFormatINetFormat *pINetFormat = rWrt.m_aINetFormats.back();
OutHTML_INetFormat( rWrt, *pINetFormat, true );
}
}
return rWrt;
}
static SwHTMLWriter& OutHTML_SwTextCharFormat( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( rWrt.m_bOutOpts )
return rWrt;
const SwFormatCharFormat& rChrFormat = static_cast<const SwFormatCharFormat&>(rHt);
const SwCharFormat* pFormat = rChrFormat.GetCharFormat();
if( !pFormat )
{
return rWrt;
}
std::unique_ptr<SwHTMLFormatInfo> pTmpInfo(new SwHTMLFormatInfo(pFormat));
SwHTMLFormatInfos::const_iterator it = rWrt.m_CharFormatInfos.find(pTmpInfo);
if (it == rWrt.m_CharFormatInfos.end())
return rWrt;
const SwHTMLFormatInfo *pFormatInfo = it->get();
assert(pFormatInfo && "Why is there no information about the character style?");
if( rWrt.m_bTagOn )
{
OString sOut = "<" + rWrt.GetNamespace();
if( !pFormatInfo->aToken.isEmpty() )
sOut += pFormatInfo->aToken;
else
sOut += OOO_STRING_SVTOOLS_HTML_span;
if( rWrt.m_bCfgOutStyles &&
(!pFormatInfo->aClass.isEmpty() || pFormatInfo->bScriptDependent) )
{
sOut += " " OOO_STRING_SVTOOLS_HTML_O_class "=\"";
rWrt.Strm().WriteOString( sOut );
OUString aClass( pFormatInfo->aClass );
if( pFormatInfo->bScriptDependent )
{
if( !aClass.isEmpty() )
aClass += "-";
switch( rWrt.m_nCSS1Script )
{
case CSS1_OUTMODE_WESTERN:
aClass += "western";
break;
case CSS1_OUTMODE_CJK:
aClass += "cjk";
break;
case CSS1_OUTMODE_CTL:
aClass += "ctl";
break;
}
}
HTMLOutFuncs::Out_String( rWrt.Strm(), aClass );
sOut = "\""_ostr;
}
sOut += ">";
rWrt.Strm().WriteOString( sOut );
}
else
{
OString aTag = !pFormatInfo->aToken.isEmpty() ? pFormatInfo->aToken.getStr()
: OOO_STRING_SVTOOLS_HTML_span;
HTMLOutFuncs::Out_AsciiTag(rWrt.Strm(), Concat2View(rWrt.GetNamespace() + aTag), false);
}
return rWrt;
}
static SwHTMLWriter& OutHTML_SvxAdjust( SwHTMLWriter& rWrt, const SfxPoolItem& rHt )
{
if( !rWrt.m_bOutOpts || !rWrt.m_bTagOn )
return rWrt;
const SvxAdjustItem& rAdjust = static_cast<const SvxAdjustItem&>(rHt);
const char* pStr = nullptr;
switch( rAdjust.GetAdjust() )
{
case SvxAdjust::Center: pStr = OOO_STRING_SVTOOLS_HTML_AL_center; break;
case SvxAdjust::Left: pStr = OOO_STRING_SVTOOLS_HTML_AL_left; break;
case SvxAdjust::Right: pStr = OOO_STRING_SVTOOLS_HTML_AL_right; break;
case SvxAdjust::Block: pStr = OOO_STRING_SVTOOLS_HTML_AL_justify; break;
default:
;
}
if( pStr )
{
OString sOut = OString::Concat(" " OOO_STRING_SVTOOLS_HTML_O_align "=\"") +
pStr + "\"";
rWrt.Strm().WriteOString( sOut );
}
return rWrt;
}
/*
* here, define the table for the HTML function pointers to the output
* functions.
*/
SwAttrFnTab aHTMLAttrFnTab = {
/* RES_CHRATR_CASEMAP */ OutHTML_CSS1Attr,
/* RES_CHRATR_CHARSETCOLOR */ nullptr,
/* RES_CHRATR_COLOR */ OutHTML_SvxColor,
/* RES_CHRATR_CONTOUR */ nullptr,
/* RES_CHRATR_CROSSEDOUT */ OutHTML_SwCrossedOut,
/* RES_CHRATR_ESCAPEMENT */ OutHTML_SvxEscapement,
/* RES_CHRATR_FONT */ OutHTML_SvxFont,
/* RES_CHRATR_FONTSIZE */ OutHTML_SvxFontHeight,
/* RES_CHRATR_KERNING */ OutHTML_CSS1Attr,
/* RES_CHRATR_LANGUAGE */ OutHTML_SvxLanguage,
/* RES_CHRATR_POSTURE */ OutHTML_SwPosture,
/* RES_CHRATR_UNUSED1*/ nullptr,
/* RES_CHRATR_SHADOWED */ nullptr,
/* RES_CHRATR_UNDERLINE */ OutHTML_SwUnderline,
/* RES_CHRATR_WEIGHT */ OutHTML_SwWeight,
/* RES_CHRATR_WORDLINEMODE */ nullptr,
/* RES_CHRATR_AUTOKERN */ nullptr,
/* RES_CHRATR_BLINK */ OutHTML_SwBlink,
/* RES_CHRATR_NOHYPHEN */ nullptr, // New: don't hyphenate
/* RES_CHRATR_UNUSED2 */ nullptr,
/* RES_CHRATR_BACKGROUND */ OutHTML_CSS1Attr, // New: character background
/* RES_CHRATR_CJK_FONT */ OutHTML_SvxFont,
/* RES_CHRATR_CJK_FONTSIZE */ OutHTML_SvxFontHeight,
/* RES_CHRATR_CJK_LANGUAGE */ OutHTML_SvxLanguage,
/* RES_CHRATR_CJK_POSTURE */ OutHTML_SwPosture,
/* RES_CHRATR_CJK_WEIGHT */ OutHTML_SwWeight,
/* RES_CHRATR_CTL_FONT */ OutHTML_SvxFont,
/* RES_CHRATR_CTL_FONTSIZE */ OutHTML_SvxFontHeight,
/* RES_CHRATR_CTL_LANGUAGE */ OutHTML_SvxLanguage,
/* RES_CHRATR_CTL_POSTURE */ OutHTML_SwPosture,
/* RES_CHRATR_CTL_WEIGHT */ OutHTML_SwWeight,
/* RES_CHRATR_ROTATE */ nullptr,
/* RES_CHRATR_EMPHASIS_MARK */ nullptr,
/* RES_CHRATR_TWO_LINES */ nullptr,
/* RES_CHRATR_SCALEW */ nullptr,
/* RES_CHRATR_RELIEF */ nullptr,
/* RES_CHRATR_HIDDEN */ OutHTML_CSS1Attr,
/* RES_CHRATR_OVERLINE */ OutHTML_CSS1Attr,
/* RES_CHRATR_RSID */ nullptr,
/* RES_CHRATR_BOX */ OutHTML_CSS1Attr,
/* RES_CHRATR_SHADOW */ nullptr,
/* RES_CHRATR_HIGHLIGHT */ nullptr,
/* RES_CHRATR_GRABBAG */ nullptr,
/* RES_CHRATR_BIDIRTL */ nullptr,
/* RES_CHRATR_IDCTHINT */ nullptr,
/* RES_TXTATR_REFMARK */ nullptr,
/* RES_TXTATR_TOXMARK */ nullptr,
/* RES_TXTATR_META */ nullptr,
/* RES_TXTATR_METAFIELD */ nullptr,
/* RES_TXTATR_AUTOFMT */ nullptr,
/* RES_TXTATR_INETFMT */ OutHTML_SwFormatINetFormat,
/* RES_TXTATR_CHARFMT */ OutHTML_SwTextCharFormat,
/* RES_TXTATR_CJK_RUBY */ nullptr,
/* RES_TXTATR_UNKNOWN_CONTAINER */ nullptr,
/* RES_TXTATR_INPUTFIELD */ OutHTML_SwFormatField,
/* RES_TXTATR_CONTENTCONTROL */ nullptr,
/* RES_TXTATR_FIELD */ OutHTML_SwFormatField,
/* RES_TXTATR_FLYCNT */ OutHTML_SwFlyCnt,
/* RES_TXTATR_FTN */ OutHTML_SwFormatFootnote,
/* RES_TXTATR_ANNOTATION */ OutHTML_SwFormatField,
/* RES_TXTATR_LINEBREAK */ OutHTML_SwFormatLineBreak,
/* RES_TXTATR_DUMMY1 */ nullptr, // Dummy:
/* RES_PARATR_LINESPACING */ nullptr,
/* RES_PARATR_ADJUST */ OutHTML_SvxAdjust,
/* RES_PARATR_SPLIT */ nullptr,
/* RES_PARATR_ORPHANS */ nullptr,
/* RES_PARATR_WIDOWS */ nullptr,
/* RES_PARATR_TABSTOP */ nullptr,
/* RES_PARATR_HYPHENZONE*/ nullptr,
/* RES_PARATR_DROP */ OutHTML_CSS1Attr,
/* RES_PARATR_REGISTER */ nullptr, // new: register-true
/* RES_PARATR_NUMRULE */ nullptr, // Dummy:
/* RES_PARATR_SCRIPTSPACE */ nullptr, // Dummy:
/* RES_PARATR_HANGINGPUNCTUATION */ nullptr, // Dummy:
/* RES_PARATR_FORBIDDEN_RULES */ nullptr, // new
/* RES_PARATR_VERTALIGN */ nullptr, // new
/* RES_PARATR_SNAPTOGRID*/ nullptr, // new
/* RES_PARATR_CONNECT_TO_BORDER */ nullptr, // new
/* RES_PARATR_OUTLINELEVEL */ nullptr,
/* RES_PARATR_RSID */ nullptr,
/* RES_PARATR_GRABBAG */ nullptr,
/* RES_PARATR_LIST_ID */ nullptr, // new
/* RES_PARATR_LIST_LEVEL */ nullptr, // new
/* RES_PARATR_LIST_ISRESTART */ nullptr, // new
/* RES_PARATR_LIST_RESTARTVALUE */ nullptr, // new
/* RES_PARATR_LIST_ISCOUNTED */ nullptr, // new
/* RES_FILL_ORDER */ nullptr,
/* RES_FRM_SIZE */ nullptr,
/* RES_PAPER_BIN */ nullptr,
/* RES_MARGIN_FIRSTLINE */ nullptr,
/* RES_MARGIN_TEXTLEFT */ nullptr,
/* RES_MARGIN_RIGHT */ nullptr,
/* RES_MARGIN_LEFT */ nullptr,
/* RES_MARGIN_GUTTER */ nullptr,
/* RES_MARGIN_GUTTER_RIGHT */ nullptr,
/* RES_LR_SPACE */ nullptr,
/* RES_UL_SPACE */ nullptr,
/* RES_PAGEDESC */ nullptr,
/* RES_BREAK */ nullptr,
/* RES_CNTNT */ nullptr,
/* RES_HEADER */ nullptr,
/* RES_FOOTER */ nullptr,
/* RES_PRINT */ nullptr,
/* RES_OPAQUE */ nullptr,
/* RES_PROTECT */ nullptr,
/* RES_SURROUND */ nullptr,
/* RES_VERT_ORIENT */ nullptr,
/* RES_HORI_ORIENT */ nullptr,
/* RES_ANCHOR */ nullptr,
/* RES_BACKGROUND */ nullptr,
/* RES_BOX */ nullptr,
/* RES_SHADOW */ nullptr,
/* RES_FRMMACRO */ nullptr,
/* RES_COL */ nullptr,
/* RES_KEEP */ nullptr,
/* RES_URL */ nullptr,
/* RES_EDIT_IN_READONLY */ nullptr,
/* RES_LAYOUT_SPLIT */ nullptr,
/* RES_CHAIN */ nullptr,
/* RES_TEXTGRID */ nullptr,
/* RES_LINENUMBER */ nullptr,
/* RES_FTN_AT_TXTEND */ nullptr,
/* RES_END_AT_TXTEND */ nullptr,
/* RES_COLUMNBALANCE */ nullptr,
/* RES_FRAMEDIR */ nullptr,
/* RES_HEADER_FOOTER_EAT_SPACING */ nullptr,
/* RES_ROW_SPLIT */ nullptr,
/* RES_FLY_SPLIT */ nullptr,
/* RES_FOLLOW_TEXT_FLOW */ nullptr,
/* RES_COLLAPSING_BORDERS */ nullptr,
/* RES_WRAP_INFLUENCE_ON_OBJPOS */ nullptr,
/* RES_AUTO_STYLE */ nullptr,
/* RES_FRMATR_STYLE_NAME */ nullptr,
/* RES_FRMATR_CONDITIONAL_STYLE_NAME */ nullptr,
/* RES_FRMATR_GRABBAG */ nullptr,
/* RES_TEXT_VERT_ADJUST */ nullptr,
/* RES_BACKGROUND_FULL_SIZE */ nullptr,
/* RES_RTL_GUTTER */ nullptr,
/* RES_DECORATIVE */ nullptr,
/* RES_GRFATR_MIRRORGRF */ nullptr,
/* RES_GRFATR_CROPGRF */ nullptr,
/* RES_GRFATR_ROTATION */ nullptr,
/* RES_GRFATR_LUMINANCE */ nullptr,
/* RES_GRFATR_CONTRAST */ nullptr,
/* RES_GRFATR_CHANNELR */ nullptr,
/* RES_GRFATR_CHANNELG */ nullptr,
/* RES_GRFATR_CHANNELB */ nullptr,
/* RES_GRFATR_GAMMA */ nullptr,
/* RES_GRFATR_INVERT */ nullptr,
/* RES_GRFATR_TRANSPARENCY */ nullptr,
/* RES_GRFATR_DRWAMODE */ nullptr,
/* RES_GRFATR_DUMMY3 */ nullptr,
/* RES_GRFATR_DUMMY4 */ nullptr,
/* RES_GRFATR_DUMMY5 */ nullptr,
/* RES_BOXATR_FORMAT */ nullptr,
/* RES_BOXATR_FORMULA */ nullptr,
/* RES_BOXATR_VALUE */ nullptr
};
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
↑ V547 Expression 'nBulletGrfLvl < MAXLEVEL' is always true.
↑ V547 Expression is always false.
↑ V1037 Two or more case-branches perform the same actions. Check lines: 1305, 1366, 1375
↑ V1037 Two or more case-branches perform the same actions. Check lines: 1358, 1362
↑ V1051 Consider checking for misprints. It's possible that the 'rTmpLang' should be checked here.