Files
libreoffice/editeng/source/rtf/svxrtf.cxx

1515 lines
49 KiB
C++
Raw Normal View History

2000-09-18 16:07:07 +00:00
/*************************************************************************
*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
2000-09-18 16:07:07 +00:00
*
* Copyright 2000, 2010 Oracle and/or its affiliates.
2000-09-18 16:07:07 +00:00
*
* OpenOffice.org - a multi-platform office productivity suite
2000-09-18 16:07:07 +00:00
*
* This file is part of OpenOffice.org.
2000-09-18 16:07:07 +00:00
*
* OpenOffice.org is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License version 3
* only, as published by the Free Software Foundation.
2000-09-18 16:07:07 +00:00
*
* OpenOffice.org is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License version 3 for more details
* (a copy is included in the LICENSE file that accompanied this code).
2000-09-18 16:07:07 +00:00
*
* You should have received a copy of the GNU Lesser General Public License
* version 3 along with OpenOffice.org. If not, see
* <http://www.openoffice.org/license.html>
* for a copy of the LGPLv3 License.
2000-09-18 16:07:07 +00:00
*
************************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_editeng.hxx"
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil -*- */
2001-03-27 20:34:02 +00:00
2000-09-18 16:07:07 +00:00
#include <ctype.h>
#include <tools/datetime.hxx>
#include <rtl/tencinfo.h>
#include <svl/itemiter.hxx>
#include <svl/whiter.hxx>
2000-09-18 16:07:07 +00:00
#include <svtools/rtftoken.h>
#include <svl/itempool.hxx>
2000-09-18 16:07:07 +00:00
#include <comphelper/string.hxx>
#include <com/sun/star/lang/Locale.hpp>
#include <editeng/scriptspaceitem.hxx>
#include <editeng/fontitem.hxx>
#include <editeng/colritem.hxx>
#include <editeng/svxrtf.hxx>
#include <editeng/editids.hrc>
#include <vcl/svapp.hxx>
2000-09-18 16:07:07 +00:00
#include <com/sun/star/document/XDocumentProperties.hpp>
using namespace ::com::sun::star;
2000-09-18 16:07:07 +00:00
SV_IMPL_PTRARR( SvxRTFColorTbl, ColorPtr )
SV_IMPL_PTRARR( SvxRTFItemStackList, SvxRTFItemStackType* )
2008-10-10 13:02:43 +00:00
CharSet lcl_GetDefaultTextEncodingForRTF()
{
::com::sun::star::lang::Locale aLocale;
::rtl::OUString aLangString;
aLocale = Application::GetSettings().GetLocale();
aLangString = aLocale.Language;
if ( aLangString.equals( ::rtl::OUString::createFromAscii( "ru" ) )
|| aLangString.equals( ::rtl::OUString::createFromAscii( "uk" ) ) )
return RTL_TEXTENCODING_MS_1251;
if ( aLangString.equals( ::rtl::OUString::createFromAscii( "tr" ) ) )
return RTL_TEXTENCODING_MS_1254;
else
return RTL_TEXTENCODING_MS_1252;
}
2000-09-18 16:07:07 +00:00
// -------------- Methoden --------------------
SvxRTFParser::SvxRTFParser( SfxItemPool& rPool, SvStream& rIn,
uno::Reference<document::XDocumentProperties> i_xDocProps,
int bReadNewDoc )
2000-09-18 16:07:07 +00:00
: SvRTFParser( rIn, 5 ),
rStrm(rIn),
2000-09-18 16:07:07 +00:00
aColorTbl( 16, 4 ),
aFontTbl( 16, 4 ),
pInsPos( 0 ),
pAttrPool( &rPool ),
m_xDocProps( i_xDocProps ),
pRTFDefaults( 0 ),
nVersionNo( 0 )
2000-09-18 16:07:07 +00:00
{
bNewDoc = bReadNewDoc;
bChkStyleAttr = bCalcValue = bReadDocInfo = bIsInReadStyleTab = FALSE;
2000-11-10 10:34:57 +00:00
bIsLeftToRightDef = TRUE;
2000-09-18 16:07:07 +00:00
{
RTFPlainAttrMapIds aTmp( rPool );
aPlainMap.Insert( (USHORT*)&aTmp,
sizeof( RTFPlainAttrMapIds ) / sizeof(USHORT), 0 );
}
{
RTFPardAttrMapIds aTmp( rPool );
aPardMap.Insert( (USHORT*)&aTmp,
sizeof( RTFPardAttrMapIds ) / sizeof(USHORT), 0 );
}
pDfltFont = new Font;
pDfltColor = new Color;
}
void SvxRTFParser::EnterEnvironment()
{
}
void SvxRTFParser::LeaveEnvironment()
{
}
void SvxRTFParser::ResetPard()
{
}
2000-09-18 16:07:07 +00:00
SvxRTFParser::~SvxRTFParser()
{
if( aColorTbl.Count() )
ClearColorTbl();
if( aFontTbl.Count() )
ClearFontTbl();
if( aStyleTbl.Count() )
ClearStyleTbl();
if( aAttrStack.Count() )
ClearAttrStack();
delete pRTFDefaults;
2000-09-18 16:07:07 +00:00
delete pInsPos;
delete pDfltFont;
delete pDfltColor;
}
void SvxRTFParser::SetInsPos( const SvxPosition& rNew )
{
if( pInsPos )
delete pInsPos;
pInsPos = rNew.Clone();
}
SvParserState SvxRTFParser::CallParser()
{
DBG_ASSERT( pInsPos, "keine Einfuegeposition" );
if( !pInsPos )
return SVPAR_ERROR;
if( aColorTbl.Count() )
ClearColorTbl();
if( aFontTbl.Count() )
ClearFontTbl();
if( aStyleTbl.Count() )
ClearStyleTbl();
if( aAttrStack.Count() )
ClearAttrStack();
bIsSetDfltTab = FALSE;
bNewGroup = FALSE;
nDfltFont = 0;
sBaseURL.Erase();
// erzeuge aus den gesetzten WhichIds die richtige WhichId-Tabelle.
BuildWhichTbl();
return SvRTFParser::CallParser();
}
void SvxRTFParser::Continue( int nToken )
{
SvRTFParser::Continue( nToken );
if( SVPAR_PENDING != GetStatus() )
2001-03-27 20:34:02 +00:00
{
2000-09-18 16:07:07 +00:00
SetAllAttrOfStk();
#if 0
//Regardless of what "color 0" is, word defaults to auto as the default colour.
//e.g. see #i7713#
2001-03-27 20:34:02 +00:00
if( bNewDoc && ((RTFPlainAttrMapIds*)aPlainMap.GetData())->nColor )
pAttrPool->SetPoolDefaultItem( SvxColorItem( GetColor( 0 ),
((RTFPlainAttrMapIds*)aPlainMap.GetData())->nColor ));
#endif
2001-03-27 20:34:02 +00:00
}
2000-09-18 16:07:07 +00:00
}
// wird fuer jedes Token gerufen, das in CallParser erkannt wird
void SvxRTFParser::NextToken( int nToken )
{
sal_Unicode cCh;
switch( nToken )
{
case RTF_COLORTBL: ReadColorTable(); break;
case RTF_FONTTBL: ReadFontTable(); break;
case RTF_STYLESHEET: ReadStyleTable(); break;
case RTF_DEFF:
if( bNewDoc )
{
if( aFontTbl.Count() )
// koennen wir sofort setzen
SetDefault( nToken, nTokenValue );
2000-09-18 16:07:07 +00:00
else
// wird nach einlesen der Fonttabelle gesetzt
nDfltFont = int(nTokenValue);
}
break;
case RTF_DEFTAB:
case RTF_DEFLANG:
if( bNewDoc )
SetDefault( nToken, nTokenValue );
2000-09-18 16:07:07 +00:00
break;
case RTF_PICT: ReadBitmapData(); break;
case RTF_LINE: cCh = '\n'; goto INSINGLECHAR;
case RTF_TAB: cCh = '\t'; goto INSINGLECHAR;
case RTF_SUBENTRYINDEX: cCh = ':'; goto INSINGLECHAR;
case RTF_EMDASH: cCh = 151; goto INSINGLECHAR;
case RTF_ENDASH: cCh = 150; goto INSINGLECHAR;
case RTF_BULLET: cCh = 149; goto INSINGLECHAR;
case RTF_LQUOTE: cCh = 145; goto INSINGLECHAR;
case RTF_RQUOTE: cCh = 146; goto INSINGLECHAR;
case RTF_LDBLQUOTE: cCh = 147; goto INSINGLECHAR;
case RTF_RDBLQUOTE: cCh = 148; goto INSINGLECHAR;
INSINGLECHAR:
2001-03-27 20:34:02 +00:00
aToken = ByteString::ConvertToUnicode( (sal_Char)cCh,
2000-09-18 16:07:07 +00:00
RTL_TEXTENCODING_MS_1252 );
// kein Break, aToken wird als Text gesetzt
case RTF_TEXTTOKEN:
{
InsertText();
// alle angesammelten Attribute setzen
for( USHORT n = aAttrSetList.Count(); n; )
{
SvxRTFItemStackType* pStkSet = aAttrSetList[--n];
SetAttrSet( *pStkSet );
aAttrSetList.DeleteAndDestroy( n );
}
}
break;
case RTF_PAR:
InsertPara();
break;
case '{':
if (bNewGroup) // Verschachtelung !!
2000-09-18 16:07:07 +00:00
_GetAttrSet();
EnterEnvironment();
bNewGroup = true;
2000-09-18 16:07:07 +00:00
break;
case '}':
if( !bNewGroup ) // leere Gruppe ??
AttrGroupEnd();
LeaveEnvironment();
bNewGroup = false;
2000-09-18 16:07:07 +00:00
break;
case RTF_INFO:
#ifndef SVX_LIGHT
if (bReadDocInfo && bNewDoc && m_xDocProps.is())
2000-09-18 16:07:07 +00:00
ReadInfo();
else
#endif
SkipGroup();
break;
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
// erstmal gesamt ueberlesen (muessen alle in einer Gruppe stehen !!)
// Koennen auch ohne dem IGNORE-Flag im RTF-File auftreten; alle Gruppen
// mit IGNORE-Flag werden im default-Zweig ueberlesen.
case RTF_SWG_PRTDATA:
case RTF_FIELD:
case RTF_ATNID:
case RTF_ANNOTATION:
case RTF_BKMKSTART:
case RTF_BKMKEND:
case RTF_BKMK_KEY:
case RTF_XE:
case RTF_TC:
case RTF_NEXTFILE:
case RTF_TEMPLATE:
#if 0
//disabled for #i19718#
2000-09-18 16:07:07 +00:00
case RTF_SHPRSLT: // RTF_SHP fehlt noch !!
#endif
2000-09-18 16:07:07 +00:00
SkipGroup();
break;
// !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
case RTF_PGDSCNO:
case RTF_PGBRK:
case RTF_SHADOW:
if( RTF_IGNOREFLAG != GetStackPtr( -1 )->nTokenId )
break;
nToken = SkipToken( -1 );
if( '{' == GetStackPtr( -1 )->nTokenId )
nToken = SkipToken( -1 );
ReadAttr( nToken, &GetAttrSet() );
break;
default:
switch( nToken & ~(0xff | RTF_SWGDEFS) )
{
case RTF_PARFMT: // hier gibts keine Swg-Defines
ReadAttr( nToken, &GetAttrSet() );
break;
case RTF_CHRFMT:
case RTF_BRDRDEF:
case RTF_TABSTOPDEF:
if( RTF_SWGDEFS & nToken)
{
if( RTF_IGNOREFLAG != GetStackPtr( -1 )->nTokenId )
break;
nToken = SkipToken( -1 );
if( '{' == GetStackPtr( -1 )->nTokenId )
{
nToken = SkipToken( -1 );
}
}
ReadAttr( nToken, &GetAttrSet() );
break;
default:
{
if( /*( '{' == GetStackPtr( -1 )->nTokenId ) ||*/
( RTF_IGNOREFLAG == GetStackPtr( -1 )->nTokenId &&
'{' == GetStackPtr( -2 )->nTokenId ) )
SkipGroup();
}
break;
}
break;
}
}
void SvxRTFParser::ReadStyleTable()
{
int nToken, bSaveChkStyleAttr = bChkStyleAttr;
short nStyleNo = 0;
int _nOpenBrakets = 1; // die erste wurde schon vorher erkannt !!
2000-09-18 16:07:07 +00:00
SvxRTFStyleType* pStyle = new SvxRTFStyleType( *pAttrPool, aWhichMap.GetData() );
pStyle->aAttrSet.Put( GetRTFDefaults() );
2000-09-18 16:07:07 +00:00
bIsInReadStyleTab = TRUE;
2000-09-18 16:07:07 +00:00
bChkStyleAttr = FALSE; // Attribute nicht gegen die Styles checken
while( _nOpenBrakets && IsParserWorking() )
2000-09-18 16:07:07 +00:00
{
switch( nToken = GetNextToken() )
{
case '}': if( --_nOpenBrakets && IsParserWorking() )
2000-09-18 16:07:07 +00:00
// Style konnte vollstaendig gelesen werden,
// also ist das noch ein stabiler Status
SaveState( RTF_STYLESHEET );
break;
case '{':
{
if( RTF_IGNOREFLAG != GetNextToken() )
nToken = SkipToken( -1 );
else if( RTF_UNKNOWNCONTROL != ( nToken = GetNextToken() ) &&
RTF_PN != nToken )
nToken = SkipToken( -2 );
else
{
// gleich herausfiltern
ReadUnknownData();
nToken = GetNextToken();
if( '}' != nToken )
eState = SVPAR_ERROR;
break;
}
++_nOpenBrakets;
2000-09-18 16:07:07 +00:00
}
break;
case RTF_SBASEDON: pStyle->nBasedOn = USHORT(nTokenValue); pStyle->bBasedOnIsSet=TRUE; break;
2000-09-18 16:07:07 +00:00
case RTF_SNEXT: pStyle->nNext = USHORT(nTokenValue); break;
case RTF_OUTLINELEVEL:
case RTF_SOUTLVL: pStyle->nOutlineNo = BYTE(nTokenValue); break;
case RTF_S: nStyleNo = (short)nTokenValue; break;
case RTF_CS: nStyleNo = (short)nTokenValue;
pStyle->bIsCharFmt = TRUE;
break;
case RTF_TEXTTOKEN:
{
pStyle->sName = DelCharAtEnd( aToken, ';' );
/*
??? soll man das umsetzen ???
if( !pStyle->sName.Len() )
pStyle->sName = "Standard";
*/
// sollte die Nummer doppelt vergeben werden ?
if( aStyleTbl.Count() )
{
SvxRTFStyleType* pOldSt = aStyleTbl.Remove( nStyleNo );
if( pOldSt )
delete pOldSt;
}
// alle Daten vom Style vorhanden, also ab in die Tabelle
aStyleTbl.Insert( nStyleNo, pStyle );
pStyle = new SvxRTFStyleType( *pAttrPool, aWhichMap.GetData() );
pStyle->aAttrSet.Put( GetRTFDefaults() );
2000-09-18 16:07:07 +00:00
nStyleNo = 0;
}
break;
default:
switch( nToken & ~(0xff | RTF_SWGDEFS) )
{
case RTF_PARFMT: // hier gibts keine Swg-Defines
ReadAttr( nToken, &pStyle->aAttrSet );
break;
case RTF_CHRFMT:
case RTF_BRDRDEF:
case RTF_TABSTOPDEF:
if( RTF_SWGDEFS & nToken)
{
if( RTF_IGNOREFLAG != GetStackPtr( -1 )->nTokenId )
break;
nToken = SkipToken( -1 );
if( '{' == GetStackPtr( -1 )->nTokenId )
{
nToken = SkipToken( -1 );
#if 0
--_nOpenBrakets; // korrigieren!!
#endif
2000-09-18 16:07:07 +00:00
}
}
ReadAttr( nToken, &pStyle->aAttrSet );
break;
}
break;
}
}
delete pStyle; // loesche das letze Style
SkipToken( -1 ); // die schliesende Klammer wird "oben" ausgewertet
// Flag wieder auf alten Zustand
bChkStyleAttr = bSaveChkStyleAttr;
bIsInReadStyleTab = FALSE;
2000-09-18 16:07:07 +00:00
}
void SvxRTFParser::ReadColorTable()
{
int nToken;
BYTE nRed = 0xff, nGreen = 0xff, nBlue = 0xff;
2000-09-18 16:07:07 +00:00
while( '}' != ( nToken = GetNextToken() ) && IsParserWorking() )
{
switch( nToken )
{
case RTF_RED: nRed = BYTE(nTokenValue); break;
case RTF_GREEN: nGreen = BYTE(nTokenValue); break;
case RTF_BLUE: nBlue = BYTE(nTokenValue); break;
case RTF_TEXTTOKEN: // oder sollte irgendein Unsin darumstehen?
if( 1 == aToken.Len()
? aToken.GetChar( 0 ) != ';'
: STRING_NOTFOUND == aToken.Search( ';' ) )
break; // es muss zumindestens das ';' gefunden werden
// else kein break !!
case ';':
if( IsParserWorking() )
{
// eine Farbe ist Fertig, in die Tabelle eintragen
// versuche die Werte auf SV interne Namen zu mappen
2001-03-27 20:34:02 +00:00
ColorPtr pColor = new Color( nRed, nGreen, nBlue );
if( !aColorTbl.Count() &&
BYTE(-1) == nRed && BYTE(-1) == nGreen && BYTE(-1) == nBlue )
pColor->SetColor( COL_AUTO );
2000-09-18 16:07:07 +00:00
aColorTbl.Insert( pColor, aColorTbl.Count() );
nRed = 0, nGreen = 0, nBlue = 0;
// Color konnte vollstaendig gelesen werden,
// also ist das noch ein stabiler Status
SaveState( RTF_COLORTBL );
}
break;
}
}
SkipToken( -1 ); // die schliesende Klammer wird "oben" ausgewertet
}
void SvxRTFParser::ReadFontTable()
{
int nToken;
int _nOpenBrakets = 1; // die erste wurde schon vorher erkannt !!
2000-09-18 16:07:07 +00:00
Font* pFont = new Font();
short nFontNo(0), nInsFontNo (0);
String sAltNm, sFntNm;
BOOL bIsAltFntNm = FALSE, bCheckNewFont;
2000-09-18 16:07:07 +00:00
CharSet nSystemChar = lcl_GetDefaultTextEncodingForRTF();
pFont->SetCharSet( nSystemChar );
SetEncoding( nSystemChar );
while( _nOpenBrakets && IsParserWorking() )
2000-09-18 16:07:07 +00:00
{
bCheckNewFont = FALSE;
2000-09-18 16:07:07 +00:00
switch( ( nToken = GetNextToken() ))
{
case '}':
bIsAltFntNm = FALSE;
// Style konnte vollstaendig gelesen werden,
// also ist das noch ein stabiler Status
if( --_nOpenBrakets <= 1 && IsParserWorking() )
SaveState( RTF_FONTTBL );
bCheckNewFont = TRUE;
nInsFontNo = nFontNo;
break;
case '{':
2000-09-18 16:07:07 +00:00
if( RTF_IGNOREFLAG != GetNextToken() )
nToken = SkipToken( -1 );
// Unknown und alle bekannten nicht ausgewerteten Gruppen
// sofort ueberspringen
else if( RTF_UNKNOWNCONTROL != ( nToken = GetNextToken() ) &&
RTF_PANOSE != nToken && RTF_FNAME != nToken &&
2000-09-18 16:07:07 +00:00
RTF_FONTEMB != nToken && RTF_FONTFILE != nToken )
nToken = SkipToken( -2 );
else
{
// gleich herausfiltern
ReadUnknownData();
nToken = GetNextToken();
if( '}' != nToken )
eState = SVPAR_ERROR;
break;
}
++_nOpenBrakets;
break;
case RTF_FROMAN:
pFont->SetFamily( FAMILY_ROMAN );
break;
case RTF_FSWISS:
pFont->SetFamily( FAMILY_SWISS );
break;
case RTF_FMODERN:
pFont->SetFamily( FAMILY_MODERN );
break;
case RTF_FSCRIPT:
pFont->SetFamily( FAMILY_SCRIPT );
break;
case RTF_FDECOR:
pFont->SetFamily( FAMILY_DECORATIVE );
break;
// bei technischen/symbolischen Font wird der CharSet ungeschaltet!!
case RTF_FTECH:
pFont->SetCharSet( RTL_TEXTENCODING_SYMBOL );
// deliberate fall through
case RTF_FNIL:
pFont->SetFamily( FAMILY_DONTKNOW );
break;
case RTF_FCHARSET:
if (-1 != nTokenValue)
{
CharSet nCharSet = rtl_getTextEncodingFromWindowsCharset(
(BYTE)nTokenValue);
pFont->SetCharSet(nCharSet);
//When we're in a font, the fontname is in the font
//charset, except for symbol fonts I believe
if (nCharSet == RTL_TEXTENCODING_SYMBOL)
nCharSet = RTL_TEXTENCODING_DONTKNOW;
SetEncoding(nCharSet);
}
break;
case RTF_FPRQ:
switch( nTokenValue )
{
case 1:
pFont->SetPitch( PITCH_FIXED );
break;
case 2:
pFont->SetPitch( PITCH_VARIABLE );
break;
}
break;
case RTF_F:
bCheckNewFont = TRUE;
nInsFontNo = nFontNo;
nFontNo = (short)nTokenValue;
break;
case RTF_FALT:
bIsAltFntNm = TRUE;
break;
case RTF_TEXTTOKEN:
DelCharAtEnd( aToken, ';' );
if ( aToken.Len() )
2000-09-18 16:07:07 +00:00
{
if( bIsAltFntNm )
sAltNm = aToken;
else
sFntNm = aToken;
2000-09-18 16:07:07 +00:00
}
break;
2000-09-18 16:07:07 +00:00
}
if( bCheckNewFont && 1 >= _nOpenBrakets && sFntNm.Len() ) // one font is ready
{
// alle Daten vom Font vorhanden, also ab in die Tabelle
if (sAltNm.Len())
(sFntNm += ';' ) += sAltNm;
pFont->SetName( sFntNm );
aFontTbl.Insert( nInsFontNo, pFont );
pFont = new Font();
pFont->SetCharSet( nSystemChar );
sAltNm.Erase();
sFntNm.Erase();
}
2000-09-18 16:07:07 +00:00
}
// den letzen muessen wir selbst loeschen
delete pFont;
SkipToken( -1 ); // die schliesende Klammer wird "oben" ausgewertet
// setze den default Font am Doc
if( bNewDoc && IsParserWorking() )
SetDefault( RTF_DEFF, nDfltFont );
}
void SvxRTFParser::ReadBitmapData()
{
SvRTFParser::ReadBitmapData();
}
void SvxRTFParser::ReadOLEData()
{
SvRTFParser::ReadOLEData();
}
String& SvxRTFParser::GetTextToEndGroup( String& rStr )
{
rStr.Erase( 0 );
int _nOpenBrakets = 1, nToken; // die erste wurde schon vorher erkannt !!
2000-09-18 16:07:07 +00:00
while( _nOpenBrakets && IsParserWorking() )
2000-09-18 16:07:07 +00:00
{
switch( nToken = GetNextToken() )
{
case '}': --_nOpenBrakets; break;
2000-09-18 16:07:07 +00:00
case '{':
{
2000-11-08 15:00:25 +00:00
if( RTF_IGNOREFLAG != GetNextToken() )
nToken = SkipToken( -1 );
else if( RTF_UNKNOWNCONTROL != GetNextToken() )
nToken = SkipToken( -2 );
else
{
// gleich herausfiltern
ReadUnknownData();
nToken = GetNextToken();
if( '}' != nToken )
eState = SVPAR_ERROR;
break;
}
++_nOpenBrakets;
2000-09-18 16:07:07 +00:00
}
break;
case RTF_TEXTTOKEN:
rStr += aToken;
break;
}
}
SkipToken( -1 ); // die schliesende Klammer wird "oben" ausgewertet
return rStr;
}
util::DateTime SvxRTFParser::GetDateTimeStamp( )
2000-09-18 16:07:07 +00:00
{
util::DateTime aDT;
2000-09-18 16:07:07 +00:00
BOOL bWeiter = TRUE;
int nToken;
while( bWeiter && IsParserWorking() )
{
switch( nToken = GetNextToken() )
{
case RTF_YR: aDT.Year = (USHORT)nTokenValue; break;
case RTF_MO: aDT.Month = (USHORT)nTokenValue; break;
case RTF_DY: aDT.Day = (USHORT)nTokenValue; break;
case RTF_HR: aDT.Hours = (USHORT)nTokenValue; break;
case RTF_MIN: aDT.Minutes = (USHORT)nTokenValue; break;
2000-09-18 16:07:07 +00:00
default:
bWeiter = FALSE;
}
}
SkipToken( -1 ); // die schliesende Klammer wird "oben" ausgewertet
return aDT;
2000-09-18 16:07:07 +00:00
}
void SvxRTFParser::ReadInfo( const sal_Char* pChkForVerNo )
{
#ifndef SVX_LIGHT
int _nOpenBrakets = 1, nToken; // die erste wurde schon vorher erkannt !!
DBG_ASSERT(m_xDocProps.is(),
"SvxRTFParser::ReadInfo: no DocumentProperties");
2000-09-18 16:07:07 +00:00
String sStr, sComment;
long nVersNo = 0;
2000-09-18 16:07:07 +00:00
while( _nOpenBrakets && IsParserWorking() )
2000-09-18 16:07:07 +00:00
{
switch( nToken = GetNextToken() )
{
case '}': --_nOpenBrakets; break;
2000-09-18 16:07:07 +00:00
case '{':
{
if( RTF_IGNOREFLAG != GetNextToken() )
nToken = SkipToken( -1 );
else if( RTF_UNKNOWNCONTROL != GetNextToken() )
nToken = SkipToken( -2 );
else
{
// gleich herausfiltern
ReadUnknownData();
nToken = GetNextToken();
if( '}' != nToken )
eState = SVPAR_ERROR;
break;
}
++_nOpenBrakets;
2000-09-18 16:07:07 +00:00
}
break;
case RTF_TITLE:
m_xDocProps->setTitle( GetTextToEndGroup( sStr ) );
2000-09-18 16:07:07 +00:00
break;
case RTF_SUBJECT:
m_xDocProps->setSubject( GetTextToEndGroup( sStr ) );
2000-09-18 16:07:07 +00:00
break;
case RTF_AUTHOR:
m_xDocProps->setAuthor( GetTextToEndGroup( sStr ) );
2000-09-18 16:07:07 +00:00
break;
case RTF_OPERATOR:
m_xDocProps->setModifiedBy( GetTextToEndGroup( sStr ) );
2000-09-18 16:07:07 +00:00
break;
case RTF_KEYWORDS:
{
::rtl::OUString sTemp = GetTextToEndGroup( sStr );
m_xDocProps->setKeywords(
::comphelper::string::convertCommaSeparated(sTemp) );
break;
}
2000-09-18 16:07:07 +00:00
case RTF_DOCCOMM:
m_xDocProps->setDescription( GetTextToEndGroup( sStr ) );
2000-09-18 16:07:07 +00:00
break;
case RTF_HLINKBASE:
sBaseURL = GetTextToEndGroup( sStr ) ;
break;
case RTF_CREATIM:
m_xDocProps->setCreationDate( GetDateTimeStamp() );
2000-09-18 16:07:07 +00:00
break;
case RTF_REVTIM:
m_xDocProps->setModificationDate( GetDateTimeStamp() );
2000-09-18 16:07:07 +00:00
break;
case RTF_PRINTIM:
m_xDocProps->setPrintDate( GetDateTimeStamp() );
2000-09-18 16:07:07 +00:00
break;
case RTF_COMMENT:
GetTextToEndGroup( sComment );
break;
case RTF_BUPTIM:
SkipGroup();
break;
case RTF_VERN:
nVersNo = nTokenValue;
break;
case RTF_EDMINS:
case RTF_ID:
case RTF_VERSION:
case RTF_NOFPAGES:
case RTF_NOFWORDS:
case RTF_NOFCHARS:
NextToken( nToken );
break;
// default:
}
}
if( pChkForVerNo &&
COMPARE_EQUAL == sComment.CompareToAscii( pChkForVerNo ))
nVersionNo = nVersNo;
SkipToken( -1 ); // die schliesende Klammer wird "oben" ausgewertet
#endif
}
void SvxRTFParser::ClearColorTbl()
{
aColorTbl.DeleteAndDestroy( 0, aColorTbl.Count() );
}
void SvxRTFParser::ClearFontTbl()
{
for( ULONG nCnt = aFontTbl.Count(); nCnt; )
delete aFontTbl.GetObject( --nCnt );
}
void SvxRTFParser::ClearStyleTbl()
{
for( ULONG nCnt = aStyleTbl.Count(); nCnt; )
delete aStyleTbl.GetObject( --nCnt );
}
void SvxRTFParser::ClearAttrStack()
{
SvxRTFItemStackType* pTmp;
for( ULONG nCnt = aAttrStack.Count(); nCnt; --nCnt )
{
pTmp = aAttrStack.Pop();
delete pTmp;
}
}
String& SvxRTFParser::DelCharAtEnd( String& rStr, const sal_Unicode cDel )
{
if( rStr.Len() && ' ' == rStr.GetChar( 0 ))
rStr.EraseLeadingChars();
if( rStr.Len() && ' ' == rStr.GetChar( rStr.Len()-1 ))
rStr.EraseTrailingChars();
if( rStr.Len() && cDel == rStr.GetChar( rStr.Len()-1 ))
rStr.Erase( rStr.Len()-1 );
return rStr;
}
const Font& SvxRTFParser::GetFont( USHORT nId )
{
const Font* pFont = aFontTbl.Get( nId );
if( !pFont )
{
const SvxFontItem& rDfltFont = (const SvxFontItem&)
pAttrPool->GetDefaultItem(
((RTFPlainAttrMapIds*)aPlainMap.GetData())->nFont );
pDfltFont->SetName( rDfltFont.GetStyleName() );
pDfltFont->SetFamily( rDfltFont.GetFamily() );
2000-09-18 16:07:07 +00:00
pFont = pDfltFont;
}
return *pFont;
}
SvxRTFItemStackType* SvxRTFParser::_GetAttrSet( int bCopyAttr )
{
SvxRTFItemStackType* pAkt = aAttrStack.Top();
SvxRTFItemStackType* pNew;
if( pAkt )
pNew = new SvxRTFItemStackType( *pAkt, *pInsPos, bCopyAttr );
else
pNew = new SvxRTFItemStackType( *pAttrPool, aWhichMap.GetData(),
*pInsPos );
pNew->SetRTFDefaults( GetRTFDefaults() );
2000-09-18 16:07:07 +00:00
aAttrStack.Push( pNew );
bNewGroup = FALSE;
return pNew;
}
void SvxRTFParser::_ClearStyleAttr( SvxRTFItemStackType& rStkType )
{
// check attributes to the attributes of the stylesheet or to
// the default attrs of the document
SfxItemSet &rSet = rStkType.GetAttrSet();
const SfxItemPool& rPool = *rSet.GetPool();
const SfxPoolItem* pItem;
SfxWhichIter aIter( rSet );
SvxRTFStyleType* pStyle;
if( !IsChkStyleAttr() ||
!rStkType.GetAttrSet().Count() ||
0 == ( pStyle = aStyleTbl.Get( rStkType.nStyleNo ) ))
{
for( USHORT nWhich = aIter.GetCurWhich(); nWhich; nWhich = aIter.NextWhich() )
{
if( SFX_WHICH_MAX > nWhich &&
SFX_ITEM_SET == rSet.GetItemState( nWhich, FALSE, &pItem ) &&
rPool.GetDefaultItem( nWhich ) == *pItem )
rSet.ClearItem( nWhich ); // loeschen
}
}
else
{
// alle Attribute, die schon vom Style definiert sind, aus dem
// akt. AttrSet entfernen
SfxItemSet &rStyleSet = pStyle->aAttrSet;
const SfxPoolItem* pSItem;
for( USHORT nWhich = aIter.GetCurWhich(); nWhich; nWhich = aIter.NextWhich() )
{
if( SFX_ITEM_SET == rStyleSet.GetItemState( nWhich, TRUE, &pSItem ))
{
// JP 22.06.99: im Style und im Set gleich gesetzt -> loeschen
if( SFX_ITEM_SET == rSet.GetItemState( nWhich, FALSE, &pItem )
&& *pItem == *pSItem )
rSet.ClearItem( nWhich ); // loeschen
}
// Bug 59571 - falls nicht im Style gesetzt und gleich mit
// dem PoolDefault -> auch dann loeschen
else if( SFX_WHICH_MAX > nWhich &&
SFX_ITEM_SET == rSet.GetItemState( nWhich, FALSE, &pItem ) &&
rPool.GetDefaultItem( nWhich ) == *pItem )
rSet.ClearItem( nWhich ); // loeschen
}
}
}
void SvxRTFParser::AttrGroupEnd() // den akt. Bearbeiten, vom Stack loeschen
{
if( aAttrStack.Count() )
{
SvxRTFItemStackType *pOld = aAttrStack.Pop();
SvxRTFItemStackType *pAkt = aAttrStack.Top();
do { // middle check loop
ULONG nOldSttNdIdx = pOld->pSttNd->GetIdx();
if( !pOld->pChildList &&
((!pOld->aAttrSet.Count() && !pOld->nStyleNo ) ||
(nOldSttNdIdx == pInsPos->GetNodeIdx() &&
pOld->nSttCnt == pInsPos->GetCntIdx() )))
break; // keine Attribute oder Bereich
// setze nur die Attribute, die unterschiedlich zum Parent sind
if( pAkt && pOld->aAttrSet.Count() )
{
SfxItemIter aIter( pOld->aAttrSet );
const SfxPoolItem* pItem = aIter.GetCurItem(), *pGet;
while( TRUE )
{
if( SFX_ITEM_SET == pAkt->aAttrSet.GetItemState(
pItem->Which(), FALSE, &pGet ) &&
*pItem == *pGet )
pOld->aAttrSet.ClearItem( pItem->Which() );
if( aIter.IsAtEnd() )
break;
pItem = aIter.NextItem();
}
if( !pOld->aAttrSet.Count() && !pOld->pChildList &&
!pOld->nStyleNo )
break;
}
// setze alle Attribute, die von Start bis hier
// definiert sind.
int bCrsrBack = !pInsPos->GetCntIdx();
if( bCrsrBack )
{
// am Absatzanfang ? eine Position zurueck
ULONG nNd = pInsPos->GetNodeIdx();
MovePos( FALSE );
// if can not move backward then later dont move forward !
bCrsrBack = nNd != pInsPos->GetNodeIdx();
}
//Bug #46608#: ungueltige Bereiche ignorieren!
if( ( pOld->pSttNd->GetIdx() < pInsPos->GetNodeIdx() ||
( pOld->pSttNd->GetIdx() == pInsPos->GetNodeIdx() &&
pOld->nSttCnt <= pInsPos->GetCntIdx() ))
#if 0
//BUG 68555 - dont test for empty paragraph or any range
&& ( nOldSttNdIdx != pInsPos->GetNodeIdx() ||
pOld->nSttCnt != pInsPos->GetCntIdx() ||
!pOld->nSttCnt )
#endif
)
{
if( !bCrsrBack )
{
// alle pard-Attribute gelten nur bis zum vorherigen
// Absatz !!
if( nOldSttNdIdx == pInsPos->GetNodeIdx() )
{
#if 0
//BUG 68555 - dont reset pard attrs, if the group not begins not at start of
// paragraph
// Bereich innerhalb eines Absatzes:
// alle Absatz-Attribute und StyleNo loeschen
// aber nur wenn mitten drin angefangen wurde
if( pOld->nSttCnt )
{
pOld->nStyleNo = 0;
for( USHORT n = 0; n < aPardMap.Count() &&
pOld->aAttrSet.Count(); ++n )
if( aPardMap[n] )
pOld->aAttrSet.ClearItem( aPardMap[n] );
if( !pOld->aAttrSet.Count() && !pOld->pChildList &&
!pOld->nStyleNo )
break; // auch dieser verlaesst uns jetzt
}
#endif
}
else
{
// jetzt wirds kompliziert:
// - alle Zeichen-Attribute behalten den Bereich,
// - alle Absatz-Attribute bekommen den Bereich
// bis zum vorherigen Absatz
SvxRTFItemStackType* pNew = new SvxRTFItemStackType(
*pOld, *pInsPos, TRUE );
pNew->aAttrSet.SetParent( pOld->aAttrSet.GetParent() );
// loesche aus pNew alle Absatz Attribute
for( USHORT n = 0; n < aPardMap.Count() &&
pNew->aAttrSet.Count(); ++n )
if( aPardMap[n] )
pNew->aAttrSet.ClearItem( aPardMap[n] );
pNew->SetRTFDefaults( GetRTFDefaults() );
2000-09-18 16:07:07 +00:00
// gab es ueberhaupt welche ?
if( pNew->aAttrSet.Count() == pOld->aAttrSet.Count() )
delete pNew; // das wars dann
else
{
pNew->nStyleNo = 0;
// spanne jetzt den richtigen Bereich auf
// pNew von alter
SetEndPrevPara( pOld->pEndNd, pOld->nEndCnt );
pNew->nSttCnt = 0;
if( IsChkStyleAttr() )
{
2000-09-18 16:07:07 +00:00
_ClearStyleAttr( *pOld );
_ClearStyleAttr( *pNew ); //#i10381#, methinks.
}
2000-09-18 16:07:07 +00:00
if( pAkt )
{
pAkt->Add( pOld );
pAkt->Add( pNew );
}
else
{
// letzter vom Stack, also zwischenspeichern, bis der
// naechste Text eingelesen wurde. (keine Attribute
// aufspannen!!)
aAttrSetList.Insert( pOld, aAttrSetList.Count() );
aAttrSetList.Insert( pNew, aAttrSetList.Count() );
}
pOld = 0; // pOld nicht loeschen
break; // das wars !!
}
}
}
pOld->pEndNd = pInsPos->MakeNodeIdx();
pOld->nEndCnt = pInsPos->GetCntIdx();
#if 0
2000-09-18 16:07:07 +00:00
if( IsChkStyleAttr() )
_ClearStyleAttr( *pOld );
#else
/*
#i21422#
If the parent (pAkt) sets something e.g. , and the child (pOld)
unsets it and the style both are based on has it unset then
clearing the pOld by looking at the style is clearly a disaster
as the text ends up with pAkts bold and not pOlds no bold, this
should be rethought out. For the moment its safest to just do
the clean if we have no parent, all we suffer is too many
redundant properties.
*/
if (IsChkStyleAttr() && !pAkt)
_ClearStyleAttr( *pOld );
#endif
2000-09-18 16:07:07 +00:00
if( pAkt )
{
2000-09-18 16:07:07 +00:00
pAkt->Add( pOld );
// split up and create new entry, because it make no sense
// to create a "so long" depend list. Bug 95010
if( bCrsrBack && 50 < pAkt->pChildList->Count() )
{
// am Absatzanfang ? eine Position zurueck
MovePos( TRUE );
bCrsrBack = FALSE;
// eine neue Gruppe aufmachen
SvxRTFItemStackType* pNew = new SvxRTFItemStackType(
*pAkt, *pInsPos, TRUE );
pNew->SetRTFDefaults( GetRTFDefaults() );
// alle bis hierher gueltigen Attribute "setzen"
AttrGroupEnd();
pAkt = aAttrStack.Top(); // can be changed after AttrGroupEnd!
pNew->aAttrSet.SetParent( pAkt ? &pAkt->aAttrSet : 0 );
aAttrStack.Push( pNew );
pAkt = pNew;
}
}
2000-09-18 16:07:07 +00:00
else
// letzter vom Stack, also zwischenspeichern, bis der
// naechste Text eingelesen wurde. (keine Attribute
// aufspannen!!)
aAttrSetList.Insert( pOld, aAttrSetList.Count() );
pOld = 0;
}
if( bCrsrBack )
// am Absatzanfang ? eine Position zurueck
MovePos( TRUE );
} while( FALSE );
if( pOld )
delete pOld;
bNewGroup = FALSE;
}
}
void SvxRTFParser::SetAllAttrOfStk() // end all Attr. and set it into doc
{
// noch alle Attrbute vom Stack holen !!
while( aAttrStack.Count() )
AttrGroupEnd();
for( USHORT n = aAttrSetList.Count(); n; )
{
SvxRTFItemStackType* pStkSet = aAttrSetList[--n];
SetAttrSet( *pStkSet );
aAttrSetList.DeleteAndDestroy( n );
}
}
// setzt alle Attribute, die unterschiedlich zum aktuellen sind
void SvxRTFParser::SetAttrSet( SvxRTFItemStackType &rSet )
{
// wurde DefTab nie eingelesen? dann setze auf default
if( !bIsSetDfltTab )
SetDefault( RTF_DEFTAB, 720 );
if( rSet.pChildList )
rSet.Compress( *this );
if( rSet.aAttrSet.Count() || rSet.nStyleNo )
SetAttrInDoc( rSet );
// dann mal alle Childs abarbeiten
if( rSet.pChildList )
for( USHORT n = 0; n < rSet.pChildList->Count(); ++n )
SetAttrSet( *(*rSet.pChildList)[ n ] );
}
// wurde noch kein Text eingefuegt ? (SttPos vom obersten StackEintrag!)
int SvxRTFParser::IsAttrSttPos()
{
SvxRTFItemStackType* pAkt = aAttrStack.Top();
return !pAkt || (pAkt->pSttNd->GetIdx() == pInsPos->GetNodeIdx() &&
pAkt->nSttCnt == pInsPos->GetCntIdx());
}
void SvxRTFParser::SetAttrInDoc( SvxRTFItemStackType & )
{
}
#ifdef USED
void SvxRTFParser::SaveState( int nToken )
{
SvRTFParser::SaveState( nToken );
}
void SvxRTFParser::RestoreState()
{
SvRTFParser::RestoreState();
}
#endif
void SvxRTFParser::BuildWhichTbl()
{
if( aWhichMap.Count() )
aWhichMap.Remove( 0, aWhichMap.Count() );
aWhichMap.Insert( (USHORT)0, (USHORT)0 );
// Aufbau einer Which-Map 'rWhichMap' aus einem Array von
// 'pWhichIds' von Which-Ids. Es hat die Lange 'nWhichIds'.
// Die Which-Map wird nicht geloescht.
SvParser::BuildWhichTbl( aWhichMap, (USHORT*)aPardMap.GetData(), aPardMap.Count() );
SvParser::BuildWhichTbl( aWhichMap, (USHORT*)aPlainMap.GetData(), aPlainMap.Count() );
}
const SfxItemSet& SvxRTFParser::GetRTFDefaults()
{
if( !pRTFDefaults )
{
pRTFDefaults = new SfxItemSet( *pAttrPool, aWhichMap.GetData() );
USHORT nId;
if( 0 != ( nId = ((RTFPardAttrMapIds*)aPardMap.GetData())->nScriptSpace ))
{
SvxScriptSpaceItem aItem( FALSE, nId );
if( bNewDoc )
pAttrPool->SetPoolDefaultItem( aItem );
else
pRTFDefaults->Put( aItem );
}
}
return *pRTFDefaults;
}
2000-09-18 16:07:07 +00:00
/* */
SvxRTFStyleType::SvxRTFStyleType( SfxItemPool& rPool, const USHORT* pWhichRange )
: aAttrSet( rPool, pWhichRange )
{
nOutlineNo = BYTE(-1); // nicht gesetzt
nBasedOn = 0;
bBasedOnIsSet = FALSE; //$flr #117411#
2000-09-18 16:07:07 +00:00
nNext = 0;
bIsCharFmt = FALSE;
}
SvxRTFItemStackType::SvxRTFItemStackType(
SfxItemPool& rPool, const USHORT* pWhichRange,
const SvxPosition& rPos )
: aAttrSet( rPool, pWhichRange ),
pChildList( 0 ),
nStyleNo( 0 )
2000-09-18 16:07:07 +00:00
{
pSttNd = rPos.MakeNodeIdx();
nSttCnt = rPos.GetCntIdx();
pEndNd = pSttNd;
nEndCnt = nSttCnt;
}
SvxRTFItemStackType::SvxRTFItemStackType(
const SvxRTFItemStackType& rCpy,
const SvxPosition& rPos,
int bCopyAttr )
: aAttrSet( *rCpy.aAttrSet.GetPool(), rCpy.aAttrSet.GetRanges() ),
pChildList( 0 ),
nStyleNo( rCpy.nStyleNo )
{
pSttNd = rPos.MakeNodeIdx();
nSttCnt = rPos.GetCntIdx();
pEndNd = pSttNd;
nEndCnt = nSttCnt;
aAttrSet.SetParent( &rCpy.aAttrSet );
if( bCopyAttr )
aAttrSet.Put( rCpy.aAttrSet );
}
SvxRTFItemStackType::~SvxRTFItemStackType()
{
if( pChildList )
delete pChildList;
if( pSttNd != pEndNd )
delete pEndNd;
delete pSttNd;
}
void SvxRTFItemStackType::Add( SvxRTFItemStackType* pIns )
{
if( !pChildList )
pChildList = new SvxRTFItemStackList( 4, 16 );
2000-09-18 16:07:07 +00:00
pChildList->Insert( pIns, pChildList->Count() );
}
#if 0
//cmc: This is the original. nEndCnt is redundantly assigned to itself, and
//pEndNd can leak if not equal to pSttNd.
2000-09-18 16:07:07 +00:00
void SvxRTFItemStackType::SetStartPos( const SvxPosition& rPos )
{
delete pSttNd;
pSttNd = rPos.MakeNodeIdx();
nSttCnt = rPos.GetCntIdx();
pEndNd = pSttNd;
nEndCnt = nEndCnt;
}
#else
void SvxRTFItemStackType::SetStartPos( const SvxPosition& rPos )
{
if (pSttNd != pEndNd)
delete pEndNd;
delete pSttNd;
pSttNd = rPos.MakeNodeIdx();
pEndNd = pSttNd;
nSttCnt = rPos.GetCntIdx();
}
#endif
void SvxRTFItemStackType::MoveFullNode(const SvxNodeIdx &rOldNode,
const SvxNodeIdx &rNewNode)
{
bool bSameEndAsStart = (pSttNd == pEndNd) ? true : false;
if (GetSttNodeIdx() == rOldNode.GetIdx())
{
delete pSttNd;
pSttNd = rNewNode.Clone();
if (bSameEndAsStart)
pEndNd = pSttNd;
}
2000-09-18 16:07:07 +00:00
if (!bSameEndAsStart && GetEndNodeIdx() == rOldNode.GetIdx())
{
delete pEndNd;
pEndNd = rNewNode.Clone();
}
//And the same for all the children
USHORT nCount = pChildList ? pChildList->Count() : 0;
for (USHORT i = 0; i < nCount; ++i)
{
SvxRTFItemStackType* pStk = (*pChildList)[i];
pStk->MoveFullNode(rOldNode, rNewNode);
}
}
2000-09-18 16:07:07 +00:00
bool SvxRTFParser::UncompressableStackEntry(const SvxRTFItemStackType &) const
{
return false;
}
2000-09-18 16:07:07 +00:00
void SvxRTFItemStackType::Compress( const SvxRTFParser& rParser )
{
DBG_ASSERT( pChildList, "es gibt keine ChildListe" );
USHORT n;
SvxRTFItemStackType* pTmp = (*pChildList)[0];
if( !pTmp->aAttrSet.Count() ||
pSttNd->GetIdx() != pTmp->pSttNd->GetIdx() ||
nSttCnt != pTmp->nSttCnt )
return;
SvxNodeIdx* pLastNd = pTmp->pEndNd;
xub_StrLen nLastCnt = pTmp->nEndCnt;
SfxItemSet aMrgSet( pTmp->aAttrSet );
for( n = 1; n < pChildList->Count(); ++n )
{
pTmp = (*pChildList)[n];
if( pTmp->pChildList )
pTmp->Compress( rParser );
if( !pTmp->nSttCnt
? (pLastNd->GetIdx()+1 != pTmp->pSttNd->GetIdx() ||
!rParser.IsEndPara( pLastNd, nLastCnt ) )
: ( pTmp->nSttCnt != nLastCnt ||
pLastNd->GetIdx() != pTmp->pSttNd->GetIdx() ))
{
while( ++n < pChildList->Count() )
if( (pTmp = (*pChildList)[n])->pChildList )
pTmp->Compress( rParser );
return;
}
if (rParser.UncompressableStackEntry(*pTmp))
return;
2000-09-18 16:07:07 +00:00
if( n )
{
// suche alle, die ueber den gesamten Bereich gesetzt sind
SfxItemIter aIter( aMrgSet );
const SfxPoolItem* pItem;
do {
USHORT nWhich = aIter.GetCurItem()->Which();
if( SFX_ITEM_SET != pTmp->aAttrSet.GetItemState( nWhich,
FALSE, &pItem ) || *pItem != *aIter.GetCurItem() )
aMrgSet.ClearItem( nWhich );
if( aIter.IsAtEnd() )
break;
aIter.NextItem();
} while( TRUE );
if( !aMrgSet.Count() )
return;
}
pLastNd = pTmp->pEndNd;
nLastCnt = pTmp->nEndCnt;
}
if( pEndNd->GetIdx() != pLastNd->GetIdx() || nEndCnt != nLastCnt )
return;
// es kann zusammengefasst werden
aAttrSet.Put( aMrgSet );
for( n = 0; n < pChildList->Count(); ++n )
{
pTmp = (*pChildList)[n];
pTmp->aAttrSet.Differentiate( aMrgSet );
if( !pTmp->pChildList && !pTmp->aAttrSet.Count() && !pTmp->nStyleNo )
{
pChildList->Remove( n );
delete pTmp;
--n;
continue;
}
}
if( !pChildList->Count() )
{
delete pChildList;
pChildList = 0;
}
}
void SvxRTFItemStackType::SetRTFDefaults( const SfxItemSet& rDefaults )
{
if( rDefaults.Count() )
{
SfxItemIter aIter( rDefaults );
do {
USHORT nWhich = aIter.GetCurItem()->Which();
if( SFX_ITEM_SET != aAttrSet.GetItemState( nWhich, FALSE ))
aAttrSet.Put( *aIter.GetCurItem() );
if( aIter.IsAtEnd() )
break;
aIter.NextItem();
} while( TRUE );
}
}
2000-09-18 16:07:07 +00:00
/* */
RTFPlainAttrMapIds::RTFPlainAttrMapIds( const SfxItemPool& rPool )
{
nCaseMap = rPool.GetTrueWhich( SID_ATTR_CHAR_CASEMAP, FALSE );
2001-03-27 20:34:02 +00:00
nBgColor = rPool.GetTrueWhich( SID_ATTR_BRUSH_CHAR, FALSE );
2000-09-18 16:07:07 +00:00
nColor = rPool.GetTrueWhich( SID_ATTR_CHAR_COLOR, FALSE );
nContour = rPool.GetTrueWhich( SID_ATTR_CHAR_CONTOUR, FALSE );
nCrossedOut = rPool.GetTrueWhich( SID_ATTR_CHAR_STRIKEOUT, FALSE );
nEscapement = rPool.GetTrueWhich( SID_ATTR_CHAR_ESCAPEMENT, FALSE );
nFont = rPool.GetTrueWhich( SID_ATTR_CHAR_FONT, FALSE );
nFontHeight = rPool.GetTrueWhich( SID_ATTR_CHAR_FONTHEIGHT, FALSE );
nKering = rPool.GetTrueWhich( SID_ATTR_CHAR_KERNING, FALSE );
nLanguage = rPool.GetTrueWhich( SID_ATTR_CHAR_LANGUAGE, FALSE );
nPosture = rPool.GetTrueWhich( SID_ATTR_CHAR_POSTURE, FALSE );
nShadowed = rPool.GetTrueWhich( SID_ATTR_CHAR_SHADOWED, FALSE );
nUnderline = rPool.GetTrueWhich( SID_ATTR_CHAR_UNDERLINE, FALSE );
nOverline = rPool.GetTrueWhich( SID_ATTR_CHAR_OVERLINE, FALSE );
2000-09-18 16:07:07 +00:00
nWeight = rPool.GetTrueWhich( SID_ATTR_CHAR_WEIGHT, FALSE );
nWordlineMode = rPool.GetTrueWhich( SID_ATTR_CHAR_WORDLINEMODE, FALSE );
nAutoKerning = rPool.GetTrueWhich( SID_ATTR_CHAR_AUTOKERN, FALSE );
2000-11-10 10:34:57 +00:00
nCJKFont = rPool.GetTrueWhich( SID_ATTR_CHAR_CJK_FONT, FALSE );
nCJKFontHeight = rPool.GetTrueWhich( SID_ATTR_CHAR_CJK_FONTHEIGHT, FALSE );
nCJKLanguage = rPool.GetTrueWhich( SID_ATTR_CHAR_CJK_LANGUAGE, FALSE );
nCJKPosture = rPool.GetTrueWhich( SID_ATTR_CHAR_CJK_POSTURE, FALSE );
nCJKWeight = rPool.GetTrueWhich( SID_ATTR_CHAR_CJK_WEIGHT, FALSE );
nCTLFont = rPool.GetTrueWhich( SID_ATTR_CHAR_CTL_FONT, FALSE );
nCTLFontHeight = rPool.GetTrueWhich( SID_ATTR_CHAR_CTL_FONTHEIGHT, FALSE );
nCTLLanguage = rPool.GetTrueWhich( SID_ATTR_CHAR_CTL_LANGUAGE, FALSE );
nCTLPosture = rPool.GetTrueWhich( SID_ATTR_CHAR_CTL_POSTURE, FALSE );
nCTLWeight = rPool.GetTrueWhich( SID_ATTR_CHAR_CTL_WEIGHT, FALSE );
nEmphasis = rPool.GetTrueWhich( SID_ATTR_CHAR_EMPHASISMARK, FALSE );
nTwoLines = rPool.GetTrueWhich( SID_ATTR_CHAR_TWO_LINES, FALSE );
nRuby = 0; //rPool.GetTrueWhich( SID_ATTR_CHAR_CJK_RUBY, FALSE );
nCharScaleX = rPool.GetTrueWhich( SID_ATTR_CHAR_SCALEWIDTH, FALSE );
nHorzVert = rPool.GetTrueWhich( SID_ATTR_CHAR_ROTATED, FALSE );
2001-03-12 15:19:49 +00:00
nRelief = rPool.GetTrueWhich( SID_ATTR_CHAR_RELIEF, FALSE );
nHidden = rPool.GetTrueWhich( SID_ATTR_CHAR_HIDDEN, FALSE );
2000-09-18 16:07:07 +00:00
}
RTFPardAttrMapIds ::RTFPardAttrMapIds ( const SfxItemPool& rPool )
{
nLinespacing = rPool.GetTrueWhich( SID_ATTR_PARA_LINESPACE, FALSE );
nAdjust = rPool.GetTrueWhich( SID_ATTR_PARA_ADJUST, FALSE );
nTabStop = rPool.GetTrueWhich( SID_ATTR_TABSTOP, FALSE );
nHyphenzone = rPool.GetTrueWhich( SID_ATTR_PARA_HYPHENZONE, FALSE );
nLRSpace = rPool.GetTrueWhich( SID_ATTR_LRSPACE, FALSE );
nULSpace = rPool.GetTrueWhich( SID_ATTR_ULSPACE, FALSE );
nBrush = rPool.GetTrueWhich( SID_ATTR_BRUSH, FALSE );
nBox = rPool.GetTrueWhich( SID_ATTR_BORDER_OUTER, FALSE );
nShadow = rPool.GetTrueWhich( SID_ATTR_BORDER_SHADOW, FALSE );
nOutlineLvl = rPool.GetTrueWhich( SID_ATTR_PARA_OUTLLEVEL, FALSE );
nSplit = rPool.GetTrueWhich( SID_ATTR_PARA_SPLIT, FALSE );
nKeep = rPool.GetTrueWhich( SID_ATTR_PARA_KEEP, FALSE );
nFontAlign = rPool.GetTrueWhich( SID_PARA_VERTALIGN, FALSE );
nScriptSpace = rPool.GetTrueWhich( SID_ATTR_PARA_SCRIPTSPACE, FALSE );
nHangPunct = rPool.GetTrueWhich( SID_ATTR_PARA_HANGPUNCTUATION, FALSE );
nForbRule = rPool.GetTrueWhich( SID_ATTR_PARA_FORBIDDEN_RULES, FALSE );
nDirection = rPool.GetTrueWhich( SID_ATTR_FRAMEDIRECTION, FALSE );
2000-09-18 16:07:07 +00:00
}
/* vi:set tabstop=4 shiftwidth=4 expandtab: */