1 /************************************************************** 2 * 3 * Licensed to the Apache Software Foundation (ASF) under one 4 * or more contributor license agreements. See the NOTICE file 5 * distributed with this work for additional information 6 * regarding copyright ownership. The ASF licenses this file 7 * to you under the Apache License, Version 2.0 (the 8 * "License"); you may not use this file except in compliance 9 * with the License. You may obtain a copy of the License at 10 * 11 * http://www.apache.org/licenses/LICENSE-2.0 12 * 13 * Unless required by applicable law or agreed to in writing, 14 * software distributed under the License is distributed on an 15 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 16 * KIND, either express or implied. See the License for the 17 * specific language governing permissions and limitations 18 * under the License. 19 * 20 *************************************************************/ 21 22 23 24 // MARKER(update_precomp.py): autogen include statement, do not remove 25 #include "precompiled_sw.hxx" 26 27 #include <tools/resid.hxx> 28 #include <hintids.hxx> 29 #include <swtypes.hxx> 30 #include <errhdl.hxx> 31 #include <txtatr.hxx> 32 #include <ndtxt.hxx> 33 #include <txttxmrk.hxx> 34 #include <tox.hxx> 35 #include <poolfmt.hrc> 36 #include <doc.hxx> 37 #include <docary.hxx> 38 #include <paratr.hxx> 39 #include <editeng/tstpitem.hxx> 40 #include <SwStyleNameMapper.hxx> 41 #include <hints.hxx> // SwPtrMsgPoolItem 42 #include <algorithm> 43 #include <functional> 44 #include <switerator.hxx> 45 46 using namespace std; 47 48 const sal_Char* SwForm::aFormEntry = "<E>"; 49 const sal_Char* SwForm::aFormTab = "<T>"; 50 const sal_Char* SwForm::aFormPageNums = "<#>"; 51 const sal_Char* SwForm::aFormLinkStt = "<LS>"; 52 const sal_Char* SwForm::aFormLinkEnd = "<LE>"; 53 const sal_Char* SwForm::aFormEntryNum = "<E#>"; 54 const sal_Char* SwForm::aFormEntryTxt = "<ET>"; 55 const sal_Char* SwForm::aFormChapterMark= "<C>"; 56 const sal_Char* SwForm::aFormText = "<X>"; 57 const sal_Char* SwForm::aFormAuth = "<A>"; 58 sal_uInt8 SwForm::nFormTabLen = 3; 59 sal_uInt8 SwForm::nFormEntryLen = 3; 60 sal_uInt8 SwForm::nFormPageNumsLen = 3; 61 sal_uInt8 SwForm::nFormLinkSttLen = 4; 62 sal_uInt8 SwForm::nFormLinkEndLen = 4; 63 sal_uInt8 SwForm::nFormEntryNumLen = 4; 64 sal_uInt8 SwForm::nFormEntryTxtLen = 4; 65 sal_uInt8 SwForm::nFormChapterMarkLen = 3; 66 sal_uInt8 SwForm::nFormTextLen = 3; 67 sal_uInt8 SwForm::nFormAuthLen = 5; 68 69 SV_IMPL_PTRARR(SwTOXMarks, SwTOXMark*) 70 71 TYPEINIT2( SwTOXMark, SfxPoolItem, SwClient ); // fuers rtti 72 73 struct PatternIni 74 { 75 sal_uInt16 n1; 76 sal_uInt16 n2; 77 sal_uInt16 n3; 78 sal_uInt16 n4; 79 sal_uInt16 n5; 80 }; 81 const PatternIni aPatternIni[] = 82 { 83 {USHRT_MAX, USHRT_MAX, USHRT_MAX, USHRT_MAX, USHRT_MAX}, //Header - no pattern 84 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_ARTICLE, 85 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_BOOK, 86 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_BOOKLET, 87 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_CONFERENCE, 88 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_INBOOK, 89 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_INCOLLECTION, 90 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_INPROCEEDINGS, 91 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_JOURNAL, 92 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_MANUAL, 93 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_MASTERSTHESIS, 94 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_MISC, 95 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_PHDTHESIS, 96 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_PROCEEDINGS, 97 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_TECHREPORT, 98 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_UNPUBLISHED, 99 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_EMAIL, 100 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, AUTH_FIELD_URL, USHRT_MAX},//AUTH_TYPE_WWW, 101 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_CUSTOM1, 102 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_CUSTOM2, 103 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_CUSTOM3, 104 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_CUSTOM4, 105 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_TYPE_CUSTOM5, 106 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_FIELD_YEAR, 107 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_FIELD_URL, 108 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_FIELD_CUSTOM1, 109 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_FIELD_CUSTOM2, 110 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_FIELD_CUSTOM3, 111 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_FIELD_CUSTOM4, 112 {AUTH_FIELD_AUTHOR, AUTH_FIELD_TITLE, AUTH_FIELD_YEAR, USHRT_MAX, USHRT_MAX}, //AUTH_FIELD_CUSTOM5, 113 {USHRT_MAX, USHRT_MAX, USHRT_MAX, USHRT_MAX, USHRT_MAX} 114 }; 115 116 SwFormTokens lcl_GetAuthPattern(sal_uInt16 nTypeId) 117 { 118 SwFormTokens aRet; 119 120 PatternIni aIni = aPatternIni[nTypeId]; 121 sal_uInt16 nVals[5]; 122 nVals[0] = aIni.n1; 123 nVals[1] = aIni.n2; 124 nVals[2] = aIni.n3; 125 nVals[3] = aIni.n4; 126 nVals[4] = aIni.n5; 127 128 SwFormToken aStartToken( TOKEN_AUTHORITY ); 129 aStartToken.nAuthorityField = AUTH_FIELD_IDENTIFIER; 130 aRet.push_back( aStartToken ); 131 SwFormToken aSeparatorToken( TOKEN_TEXT ); 132 aSeparatorToken.sText = String::CreateFromAscii( ": " ); 133 aRet.push_back( aSeparatorToken ); 134 SwFormToken aTextToken( TOKEN_TEXT ); 135 aTextToken.sText = String::CreateFromAscii( ", " ); 136 137 for(sal_uInt16 i = 0; i < 5 ; i++) 138 { 139 if(nVals[i] == USHRT_MAX) 140 break; 141 if( i > 0 ) 142 aRet.push_back( aTextToken ); 143 144 // -> #i21237# 145 SwFormToken aToken(TOKEN_AUTHORITY); 146 147 aToken.nAuthorityField = nVals[i]; 148 aRet.push_back(aToken); 149 // <- #i21237# 150 } 151 152 return aRet; 153 } 154 /*-------------------------------------------------------------------- 155 Beschreibung: Verzeichnis-Markierungen D/Ctor 156 --------------------------------------------------------------------*/ 157 158 159 /// pool default constructor 160 SwTOXMark::SwTOXMark() 161 : SfxPoolItem( RES_TXTATR_TOXMARK ) 162 , SwModify( 0 ) 163 , 164 pTxtAttr( 0 ), 165 bAutoGenerated(sal_False), 166 bMainEntry(sal_False) 167 { 168 } 169 170 171 SwTOXMark::SwTOXMark( const SwTOXType* pTyp ) 172 : SfxPoolItem( RES_TXTATR_TOXMARK ) 173 , SwModify( const_cast<SwTOXType*>(pTyp) ) 174 , 175 pTxtAttr( 0 ), nLevel( 0 ), 176 bAutoGenerated(sal_False), 177 bMainEntry(sal_False) 178 { 179 } 180 181 182 SwTOXMark::SwTOXMark( const SwTOXMark& rCopy ) 183 : SfxPoolItem( RES_TXTATR_TOXMARK ) 184 , SwModify(rCopy.GetRegisteredInNonConst()) 185 , 186 aPrimaryKey( rCopy.aPrimaryKey ), aSecondaryKey( rCopy.aSecondaryKey ), 187 aTextReading( rCopy.aTextReading ), 188 aPrimaryKeyReading( rCopy.aPrimaryKeyReading ), 189 aSecondaryKeyReading( rCopy.aSecondaryKeyReading ), 190 pTxtAttr( 0 ), nLevel( rCopy.nLevel ), 191 bAutoGenerated( rCopy.bAutoGenerated), 192 bMainEntry(rCopy.bMainEntry) 193 { 194 // AlternativString kopieren 195 aAltText = rCopy.aAltText; 196 } 197 198 199 SwTOXMark::~SwTOXMark() 200 { 201 } 202 203 204 void SwTOXMark::RegisterToTOXType( SwTOXType& rMark ) 205 { 206 rMark.Add(this); 207 } 208 209 int SwTOXMark::operator==( const SfxPoolItem& rAttr ) const 210 { 211 ASSERT( SfxPoolItem::operator==( rAttr ), "keine gleichen Attribute" ); 212 return GetRegisteredIn() == ((SwTOXMark&)rAttr).GetRegisteredIn(); 213 } 214 215 216 SfxPoolItem* SwTOXMark::Clone( SfxItemPool* ) const 217 { 218 return new SwTOXMark( *this ); 219 } 220 221 void SwTOXMark::Modify( const SfxPoolItem* pOld, const SfxPoolItem* pNew) 222 { 223 NotifyClients(pOld, pNew); 224 if (pOld && (RES_REMOVE_UNO_OBJECT == pOld->Which())) 225 { // invalidate cached uno object 226 SetXTOXMark(::com::sun::star::uno::Reference< 227 ::com::sun::star::text::XDocumentIndexMark>(0)); 228 } 229 } 230 231 void SwTOXMark::InvalidateTOXMark() 232 { 233 SwPtrMsgPoolItem aMsgHint( RES_REMOVE_UNO_OBJECT, 234 &static_cast<SwModify&>(*this) ); // cast to base class! 235 NotifyClients(&aMsgHint, &aMsgHint); 236 } 237 238 String SwTOXMark::GetText() const 239 { 240 String aStr; 241 if( aAltText.Len() ) 242 aStr = aAltText; 243 else if( pTxtAttr && pTxtAttr->GetpTxtNd() ) 244 { 245 const xub_StrLen* pEndIdx = pTxtAttr->GetEnd(); 246 ASSERT( pEndIdx, "TOXMark ohne Mark!!"); 247 if( pEndIdx ) 248 { 249 const xub_StrLen nStt = *pTxtAttr->GetStart(); 250 aStr = pTxtAttr->GetpTxtNd()->GetExpandTxt( nStt, *pEndIdx-nStt ); 251 } 252 } 253 return aStr; 254 } 255 256 void SwTOXMark::InsertTOXMarks( SwTOXMarks& aMarks, const SwTOXType& rType ) 257 { 258 SwIterator<SwTOXMark,SwTOXType> aIter(rType); 259 SwTOXMark* pMark = aIter.First(); 260 while( pMark ) 261 { 262 if(pMark->GetTxtTOXMark()) 263 aMarks.C40_INSERT(SwTOXMark, pMark, aMarks.Count()); 264 pMark = aIter.Next(); 265 } 266 } 267 268 /*-------------------------------------------------------------------- 269 Beschreibung: Typen von Verzeichnissen verwalten 270 --------------------------------------------------------------------*/ 271 272 SwTOXType::SwTOXType( TOXTypes eTyp, const String& rName ) 273 : SwModify(0), 274 aName(rName), 275 eType(eTyp) 276 { 277 } 278 279 280 SwTOXType::SwTOXType(const SwTOXType& rCopy) 281 : SwModify( (SwModify*)rCopy.GetRegisteredIn() ), 282 aName(rCopy.aName), 283 eType(rCopy.eType) 284 { 285 } 286 287 /*-------------------------------------------------------------------- 288 Beschreibung: Formen bearbeiten 289 --------------------------------------------------------------------*/ 290 291 SwForm::SwForm( TOXTypes eTyp ) // #i21237# 292 : eType( eTyp ), nFormMaxLevel( SwForm::GetFormMaxLevel( eTyp )), 293 // nFirstTabPos( lNumIndent ), 294 bCommaSeparated(sal_False) 295 { 296 //bHasFirstTabPos = 297 bGenerateTabPos = sal_False; 298 bIsRelTabPos = sal_True; 299 300 // Inhaltsverzeichnis hat entsprechend Anzahl Headlines + Ueberschrift 301 // Benutzer hat 10 Ebenen + Ueberschrift 302 // Stichwort hat 3 Ebenen + Ueberschrift + Trenner 303 // indexes of tables, objects illustrations and authorities consist of a heading and one level 304 305 sal_uInt16 nPoolId; 306 switch( eType ) 307 { 308 case TOX_INDEX: nPoolId = STR_POOLCOLL_TOX_IDXH; break; 309 case TOX_USER: nPoolId = STR_POOLCOLL_TOX_USERH; break; 310 case TOX_CONTENT: nPoolId = STR_POOLCOLL_TOX_CNTNTH; break; 311 case TOX_ILLUSTRATIONS: nPoolId = STR_POOLCOLL_TOX_ILLUSH; break; 312 case TOX_OBJECTS : nPoolId = STR_POOLCOLL_TOX_OBJECTH; break; 313 case TOX_TABLES : nPoolId = STR_POOLCOLL_TOX_TABLESH; break; 314 case TOX_AUTHORITIES : nPoolId = STR_POOLCOLL_TOX_AUTHORITIESH; break; 315 default: 316 ASSERT( !this, "ungueltiger TOXTyp"); 317 return ; 318 } 319 320 SwFormTokens aTokens; 321 if (TOX_CONTENT == eType) 322 { 323 aTokens.push_back(SwFormToken(TOKEN_ENTRY_NO)); 324 aTokens.push_back(SwFormToken(TOKEN_ENTRY_TEXT)); 325 } 326 else 327 aTokens.push_back(SwFormToken(TOKEN_ENTRY)); 328 329 if (TOX_AUTHORITIES != eType) 330 { 331 SwFormToken aToken(TOKEN_TAB_STOP); 332 aToken.nTabStopPosition = 0; 333 334 // --> FME 2004-12-10 #i36870# right aligned tab for all 335 aToken.cTabFillChar = '.'; 336 aToken.eTabAlign = SVX_TAB_ADJUST_END; 337 // <-- 338 339 aTokens.push_back(aToken); 340 aTokens.push_back(SwFormToken(TOKEN_PAGE_NUMS)); 341 } 342 343 SetTemplate( 0, SW_RESSTR( nPoolId++ )); 344 345 if(TOX_INDEX == eType) 346 { 347 for( sal_uInt16 i = 1; i < 5; ++i ) 348 { 349 if(1 == i) 350 { 351 SwFormTokens aTmpTokens; 352 SwFormToken aTmpToken(TOKEN_ENTRY); 353 aTmpTokens.push_back(aTmpToken); 354 355 SetPattern( i, aTmpTokens ); 356 SetTemplate( i, SW_RESSTR( STR_POOLCOLL_TOX_IDXBREAK )); 357 } 358 else 359 { 360 SetPattern( i, aTokens ); 361 SetTemplate( i, SW_RESSTR( STR_POOLCOLL_TOX_IDX1 + i - 2 )); 362 } 363 } 364 } 365 else 366 for( sal_uInt16 i = 1; i < GetFormMax(); ++i, ++nPoolId ) // Nr 0 ist der Titel 367 { 368 if(TOX_AUTHORITIES == eType) 369 SetPattern(i, lcl_GetAuthPattern(i)); 370 else 371 SetPattern( i, aTokens ); 372 373 if( TOX_CONTENT == eType && 6 == i ) 374 nPoolId = STR_POOLCOLL_TOX_CNTNT6; 375 else if( TOX_USER == eType && 6 == i ) 376 nPoolId = STR_POOLCOLL_TOX_USER6; 377 else if( TOX_AUTHORITIES == eType ) 378 nPoolId = STR_POOLCOLL_TOX_AUTHORITIES1; 379 SetTemplate( i, SW_RESSTR( nPoolId ) ); 380 } 381 } 382 383 384 SwForm::SwForm(const SwForm& rForm) 385 : eType( rForm.eType ) 386 { 387 *this = rForm; 388 } 389 390 391 SwForm& SwForm::operator=(const SwForm& rForm) 392 { 393 eType = rForm.eType; 394 nFormMaxLevel = rForm.nFormMaxLevel; 395 // nFirstTabPos = rForm.nFirstTabPos; 396 // bHasFirstTabPos = rForm.bHasFirstTabPos; 397 bGenerateTabPos = rForm.bGenerateTabPos; 398 bIsRelTabPos = rForm.bIsRelTabPos; 399 bCommaSeparated = rForm.bCommaSeparated; 400 for(sal_uInt16 i=0; i < nFormMaxLevel; ++i) 401 { 402 aPattern[i] = rForm.aPattern[i]; 403 aTemplate[i] = rForm.aTemplate[i]; 404 } 405 return *this; 406 } 407 408 sal_uInt16 SwForm::GetFormMaxLevel( TOXTypes eTOXType ) 409 { 410 sal_uInt16 nRet = 0; 411 switch( eTOXType ) 412 { 413 case TOX_INDEX: nRet = 5; break; 414 case TOX_USER: nRet = MAXLEVEL+1; break; 415 case TOX_CONTENT: nRet = MAXLEVEL+1; break; 416 case TOX_ILLUSTRATIONS: 417 case TOX_OBJECTS : 418 case TOX_TABLES : nRet = 2; break; 419 case TOX_AUTHORITIES : nRet = AUTH_TYPE_END + 1; break; 420 } 421 return nRet; 422 } 423 424 // #i21237# 425 bool operator == (const SwFormToken & rToken, FormTokenType eType) 426 { 427 return rToken.eTokenType == eType; 428 } 429 430 //----------------------------------------------------------------------------- 431 void SwForm::AdjustTabStops(SwDoc& rDoc, sal_Bool bInsertNewTapStops) // #i21237# 432 { 433 for(sal_uInt16 nLevel = 1; nLevel < GetFormMax(); nLevel++) 434 { 435 const String& sTemplateName = GetTemplate(nLevel); 436 437 SwTxtFmtColl* pColl = rDoc.FindTxtFmtCollByName( sTemplateName ); 438 if( !pColl ) 439 { 440 sal_uInt16 nId = SwStyleNameMapper::GetPoolIdFromUIName 441 ( sTemplateName, nsSwGetPoolIdFromName::GET_POOLID_TXTCOLL ); // #i21237# 442 if( USHRT_MAX != nId ) 443 pColl = rDoc.GetTxtCollFromPool( nId ); 444 } 445 446 const SvxTabStopItem* pTabStops = 0; 447 sal_uInt16 nTabCount = 0; 448 if( pColl && 449 0 != ( pTabStops = &pColl->GetTabStops(sal_False) ) && 450 0 != ( nTabCount = pTabStops->Count() ) ) 451 { 452 // #i21237# 453 SwFormTokens aCurrentPattern = GetPattern(nLevel); 454 SwFormTokens::iterator aIt = aCurrentPattern.begin(); 455 456 sal_Bool bChanged = sal_False; 457 458 for(sal_uInt16 nTab = 0; nTab < nTabCount; ++nTab) 459 { 460 const SvxTabStop& rTab = (*pTabStops)[nTab]; 461 462 // --> FME 2004-12-16 #i29178# 463 // For Word import, we do not want to replace exising tokens, 464 // we insert new tabstop tokens without a tabstop character: 465 if ( bInsertNewTapStops ) 466 { 467 if ( SVX_TAB_ADJUST_DEFAULT != rTab.GetAdjustment() ) 468 { 469 bChanged = sal_True; 470 SwFormToken aToken(TOKEN_TAB_STOP); 471 aToken.bWithTab = sal_False; 472 aToken.nTabStopPosition = rTab.GetTabPos(); 473 aToken.eTabAlign = rTab.GetAdjustment(); 474 aToken.cTabFillChar = rTab.GetFill(); 475 aCurrentPattern.push_back(aToken); 476 } 477 } 478 // <-- 479 else 480 { 481 aIt = find_if(aIt, aCurrentPattern.end(), 482 SwFormTokenEqualToFormTokenType 483 (TOKEN_TAB_STOP)); 484 if ( aIt != aCurrentPattern.end() ) 485 { 486 bChanged = sal_True; 487 aIt->nTabStopPosition = rTab.GetTabPos(); 488 aIt->eTabAlign = nTab == nTabCount - 1 && 489 SVX_TAB_ADJUST_RIGHT == rTab.GetAdjustment() ? 490 SVX_TAB_ADJUST_END : 491 rTab.GetAdjustment(); 492 aIt->cTabFillChar = rTab.GetFill(); 493 ++aIt; 494 } 495 else 496 break; // no more tokens to replace 497 } 498 } 499 // <-- 500 501 if(bChanged) 502 SetPattern(nLevel, aCurrentPattern); // #i21237# 503 } 504 } 505 } 506 /*-------------------------------------------------------------------- 507 Beschreibung: Ctor TOXBase 508 --------------------------------------------------------------------*/ 509 510 511 SwTOXBase::SwTOXBase(const SwTOXType* pTyp, const SwForm& rForm, 512 sal_uInt16 nCreaType, const String& rTitle ) 513 : SwClient((SwModify*)pTyp) 514 , aForm(rForm) 515 , aTitle(rTitle) 516 , eLanguage((LanguageType)::GetAppLanguage()) 517 , nCreateType(nCreaType) 518 , nOLEOptions(0) 519 , eCaptionDisplay(CAPTION_COMPLETE) 520 , bProtected( sal_True ) 521 , bFromChapter(sal_False) 522 , bFromObjectNames(sal_False) 523 , bLevelFromChapter(sal_False) 524 , maMSTOCExpression() 525 , mbKeepExpression(sal_True) 526 { 527 aData.nOptions = 0; 528 } 529 530 531 SwTOXBase::SwTOXBase( const SwTOXBase& rSource, SwDoc* pDoc ) 532 : SwClient( rSource.GetRegisteredInNonConst() ) 533 , mbKeepExpression(sal_True) 534 { 535 CopyTOXBase( pDoc, rSource ); 536 } 537 538 void SwTOXBase::RegisterToTOXType( SwTOXType& rType ) 539 { 540 rType.Add( this ); 541 } 542 543 SwTOXBase& SwTOXBase::CopyTOXBase( SwDoc* pDoc, const SwTOXBase& rSource ) 544 { 545 maMSTOCExpression = rSource.maMSTOCExpression; 546 SwTOXType* pType = (SwTOXType*)rSource.GetTOXType(); 547 if( pDoc && USHRT_MAX == pDoc->GetTOXTypes().GetPos( pType )) 548 { 549 // type not in pDoc, so create it now 550 const SwTOXTypes& rTypes = pDoc->GetTOXTypes(); 551 sal_Bool bFound = sal_False; 552 for( sal_uInt16 n = rTypes.Count(); n; ) 553 { 554 const SwTOXType* pCmp = rTypes[ --n ]; 555 if( pCmp->GetType() == pType->GetType() && 556 pCmp->GetTypeName() == pType->GetTypeName() ) 557 { 558 pType = (SwTOXType*)pCmp; 559 bFound = sal_True; 560 break; 561 } 562 } 563 564 if( !bFound ) 565 pType = (SwTOXType*)pDoc->InsertTOXType( *pType ); 566 } 567 pType->Add( this ); 568 569 nCreateType = rSource.nCreateType; 570 aTitle = rSource.aTitle; 571 aForm = rSource.aForm; 572 bProtected = rSource.bProtected; 573 bFromChapter = rSource.bFromChapter; 574 bFromObjectNames = rSource.bFromObjectNames; 575 sMainEntryCharStyle = rSource.sMainEntryCharStyle; 576 sSequenceName = rSource.sSequenceName; 577 eCaptionDisplay = rSource.eCaptionDisplay; 578 nOLEOptions = rSource.nOLEOptions; 579 eLanguage = rSource.eLanguage; 580 sSortAlgorithm = rSource.sSortAlgorithm; 581 582 for( sal_uInt16 i = 0; i < MAXLEVEL; ++i ) 583 aStyleNames[i] = rSource.aStyleNames[i]; 584 585 // its the same data type! 586 aData.nOptions = rSource.aData.nOptions; 587 588 if( !pDoc || pDoc->IsCopyIsMove() ) 589 aName = rSource.GetTOXName(); 590 else 591 aName = pDoc->GetUniqueTOXBaseName( *pType, &rSource.GetTOXName() ); 592 593 return *this; 594 } 595 596 /*-------------------------------------------------------------------- 597 Beschreibung: Verzeichnisspezifische Funktionen 598 --------------------------------------------------------------------*/ 599 600 SwTOXBase::~SwTOXBase() 601 { 602 // if( GetTOXType()->GetType() == TOX_USER ) 603 // delete aData.pTemplateName; 604 } 605 606 void SwTOXBase::SetTitle(const String& rTitle) 607 { aTitle = rTitle; } 608 609 610 SwTOXBase & SwTOXBase::operator = (const SwTOXBase & rSource) 611 { 612 ByteString aTmpStr(aTitle, RTL_TEXTENCODING_ASCII_US); 613 ByteString aTmpStr1(rSource.aTitle, RTL_TEXTENCODING_ASCII_US); 614 615 aForm = rSource.aForm; 616 aName = rSource.aName; 617 aTitle = rSource.aTitle; 618 sMainEntryCharStyle = rSource.sMainEntryCharStyle; 619 for(sal_uInt16 nLevel = 0; nLevel < MAXLEVEL; nLevel++) 620 aStyleNames[nLevel] = rSource.aStyleNames[nLevel]; 621 sSequenceName = rSource.sSequenceName; 622 eLanguage = rSource.eLanguage; 623 sSortAlgorithm = rSource.sSortAlgorithm; 624 aData = rSource.aData; 625 nCreateType = rSource.nCreateType; 626 nOLEOptions = rSource.nOLEOptions; 627 eCaptionDisplay = rSource.eCaptionDisplay; 628 bProtected = rSource.bProtected; 629 bFromChapter = rSource.bFromChapter; 630 bFromObjectNames = rSource.bFromObjectNames; 631 bLevelFromChapter = rSource.bLevelFromChapter; 632 633 if (rSource.GetAttrSet()) 634 SetAttrSet(*rSource.GetAttrSet()); 635 636 return *this; 637 } 638 639 /* -----------------16.07.99 16:02------------------- 640 641 SwTOXBase & SwTOXBase::operator = (const SwTOXBase & rSource) 642 { 643 aForm = rSource.aForm; 644 aName = rSource.aName; 645 aTitle = rSource.aTitle; 646 sMainEntryCharStyle = rSource.sMainEntryCharStyle; 647 sSequenceName = rSource.sSequenceName; 648 eLanguage = rSource.eLanguage; 649 sSortAlgorithm = rSource.sSortAlgorithm; 650 aData = rSource.aData; 651 nCreateType = rSource.nCreateType; 652 nOLEOptions = rSource.nOLEOptions; 653 eCaptionDisplay = rSource.eCaptionDisplay; 654 bProtected = rSource.bProtected; 655 bFromChapter = rSource.bFromChapter; 656 bFromObjectNames = rSource.bFromObjectNames; 657 bLevelFromChapter = rSource.bLevelFromChapter; 658 659 if (rSource.GetAttrSet()) 660 SetAttrSet(*rSource.GetAttrSet()); 661 662 return *this; 663 } 664 665 --------------------------------------------------*/ 666 667 String SwFormToken::GetString() const 668 { 669 String sRet; 670 671 sal_Bool bAppend = sal_True; 672 switch( eTokenType ) 673 { 674 case TOKEN_ENTRY_NO: 675 sRet.AssignAscii( SwForm::aFormEntryNum ); 676 break; 677 case TOKEN_ENTRY_TEXT: 678 sRet.AssignAscii( SwForm::aFormEntryTxt ); 679 break; 680 case TOKEN_ENTRY: 681 sRet.AssignAscii( SwForm::aFormEntry ); 682 break; 683 case TOKEN_TAB_STOP: 684 sRet.AssignAscii( SwForm::aFormTab ); 685 break; 686 case TOKEN_TEXT: 687 sRet.AssignAscii( SwForm::aFormText ); 688 break; 689 case TOKEN_PAGE_NUMS: 690 sRet.AssignAscii( SwForm::aFormPageNums ); 691 break; 692 case TOKEN_CHAPTER_INFO: 693 sRet.AssignAscii( SwForm::aFormChapterMark ); 694 break; 695 case TOKEN_LINK_START: 696 sRet.AssignAscii( SwForm::aFormLinkStt ); 697 break; 698 case TOKEN_LINK_END: 699 sRet.AssignAscii( SwForm::aFormLinkEnd ); 700 break; 701 case TOKEN_AUTHORITY: 702 { 703 sRet.AssignAscii( SwForm::aFormAuth ); 704 String sTmp( String::CreateFromInt32( nAuthorityField )); 705 if( sTmp.Len() < 2 ) 706 sTmp.Insert('0', 0); 707 sRet.Insert( sTmp, 2 ); 708 } 709 break; 710 case TOKEN_END: 711 break; 712 } 713 sRet.Erase( sRet.Len() - 1 ); 714 sRet += ' '; 715 sRet += sCharStyleName; 716 sRet += ','; 717 sRet += String::CreateFromInt32( nPoolId ); 718 sRet += ','; 719 720 // TabStopPosition and TabAlign or ChapterInfoFormat 721 if(TOKEN_TAB_STOP == eTokenType) 722 { 723 sRet += String::CreateFromInt32( nTabStopPosition ); 724 sRet += ','; 725 sRet += String::CreateFromInt32( static_cast< sal_Int32 >(eTabAlign) ); 726 sRet += ','; 727 sRet += cTabFillChar; 728 sRet += ','; 729 sRet += String::CreateFromInt32( bWithTab ); 730 } 731 else if(TOKEN_CHAPTER_INFO == eTokenType) 732 { 733 sRet += String::CreateFromInt32( nChapterFormat ); 734 //add maximum permetted level 735 sRet += ','; 736 sRet += String::CreateFromInt32( nOutlineLevel ); 737 } 738 else if(TOKEN_TEXT == eTokenType) 739 { 740 //append Text if Len() > 0 only! 741 if( sText.Len() ) 742 { 743 sRet += TOX_STYLE_DELIMITER; 744 String sTmp( sText ); 745 sTmp.EraseAllChars( TOX_STYLE_DELIMITER ); 746 sRet += sTmp; 747 sRet += TOX_STYLE_DELIMITER; 748 } 749 else 750 bAppend = sal_False; 751 } 752 else if(TOKEN_ENTRY_NO == eTokenType) 753 { 754 sRet += String::CreateFromInt32( nChapterFormat ); 755 //add maximum permitted level 756 sRet += ','; 757 sRet += String::CreateFromInt32( nOutlineLevel ); 758 } 759 760 if(bAppend) 761 { 762 sRet += '>'; 763 } 764 else 765 { 766 // don't append empty text tokens 767 sRet.Erase(); 768 } 769 770 return sRet; 771 } 772 773 // -> #i21237# 774 SwFormTokensHelper::SwFormTokensHelper(const String & rPattern) 775 { 776 xub_StrLen nCurPatternPos = 0; 777 xub_StrLen nCurPatternLen = 0; 778 779 while (nCurPatternPos < rPattern.Len()) 780 { 781 nCurPatternPos = nCurPatternPos + nCurPatternLen; 782 783 SwFormToken aToken = BuildToken(rPattern, nCurPatternPos); 784 aTokens.push_back(aToken); 785 } 786 } 787 788 SwFormToken SwFormTokensHelper::BuildToken( const String & sPattern, 789 xub_StrLen & nCurPatternPos ) const 790 { 791 String sToken( SearchNextToken(sPattern, nCurPatternPos) ); 792 nCurPatternPos = nCurPatternPos + sToken.Len(); 793 xub_StrLen nTokenLen; 794 FormTokenType eTokenType = GetTokenType(sToken, &nTokenLen); 795 796 // at this point sPattern contains the 797 // character style name, the PoolId, tab stop position, tab stop alignment, chapter info format 798 // the form is: CharStyleName, PoolId[, TabStopPosition|ChapterInfoFormat[, TabStopAlignment[, TabFillChar]]] 799 // in text tokens the form differs from the others: CharStyleName, PoolId[,\0xffinserted text\0xff] 800 SwFormToken eRet( eTokenType ); 801 String sAuthFieldEnum = sToken.Copy( 2, 2 ); 802 sToken = sToken.Copy( nTokenLen, sToken.Len() - nTokenLen - 1); 803 804 eRet.sCharStyleName = sToken.GetToken( 0, ','); 805 String sTmp( sToken.GetToken( 1, ',' )); 806 if( sTmp.Len() ) 807 eRet.nPoolId = static_cast<sal_uInt16>(sTmp.ToInt32()); 808 809 switch( eTokenType ) 810 { 811 //i53420 812 case TOKEN_ENTRY_NO: 813 if( (sTmp = sToken.GetToken( 2, ',' ) ).Len() ) 814 eRet.nChapterFormat = static_cast<sal_uInt16>(sTmp.ToInt32()); 815 if( (sTmp = sToken.GetToken( 3, ',' ) ).Len() ) 816 eRet.nOutlineLevel = static_cast<sal_uInt16>(sTmp.ToInt32()); //the maximum outline level to examine 817 break; 818 819 case TOKEN_TEXT: 820 { 821 xub_StrLen nStartText = sToken.Search( TOX_STYLE_DELIMITER ); 822 if( STRING_NOTFOUND != nStartText ) 823 { 824 xub_StrLen nEndText = sToken.Search( TOX_STYLE_DELIMITER, 825 nStartText + 1); 826 if( STRING_NOTFOUND != nEndText ) 827 { 828 eRet.sText = sToken.Copy( nStartText + 1, 829 nEndText - nStartText - 1); 830 } 831 } 832 } 833 break; 834 835 case TOKEN_TAB_STOP: 836 if( (sTmp = sToken.GetToken( 2, ',' ) ).Len() ) 837 eRet.nTabStopPosition = sTmp.ToInt32(); 838 839 if( (sTmp = sToken.GetToken( 3, ',' ) ).Len() ) 840 eRet.eTabAlign = static_cast<SvxTabAdjust>(sTmp.ToInt32()); 841 842 if( (sTmp = sToken.GetToken( 4, ',' ) ).Len() ) 843 eRet.cTabFillChar = sTmp.GetChar(0); 844 845 if( (sTmp = sToken.GetToken( 5, ',' ) ).Len() ) 846 eRet.bWithTab = 0 != sTmp.ToInt32(); 847 break; 848 849 case TOKEN_CHAPTER_INFO: 850 if( (sTmp = sToken.GetToken( 2, ',' ) ).Len() ) 851 eRet.nChapterFormat = static_cast<sal_uInt16>(sTmp.ToInt32()); //SwChapterFormat; 852 //i53420 853 if( (sTmp = sToken.GetToken( 3, ',' ) ).Len() ) 854 eRet.nOutlineLevel = static_cast<sal_uInt16>(sTmp.ToInt32()); //the maximum outline level to examine 855 856 break; 857 858 case TOKEN_AUTHORITY: 859 eRet.nAuthorityField = static_cast<sal_uInt16>(sAuthFieldEnum.ToInt32()); 860 break; 861 default: break; 862 } 863 return eRet; 864 } 865 866 String SwFormTokensHelper::SearchNextToken( const String & sPattern, 867 xub_StrLen nStt ) const 868 { 869 //it's not so easy - it doesn't work if the text part contains a '>' 870 //sal_uInt16 nTokenEnd = sPattern.Search('>'); 871 872 String aResult; 873 874 xub_StrLen nEnd = sPattern.Search( '>', nStt ); 875 if( STRING_NOTFOUND == nEnd ) 876 { 877 nEnd = sPattern.Len(); 878 } 879 else 880 { 881 xub_StrLen nTextSeparatorFirst = sPattern.Search( TOX_STYLE_DELIMITER, nStt ); 882 if( STRING_NOTFOUND != nTextSeparatorFirst ) 883 { 884 xub_StrLen nTextSeparatorSecond = sPattern.Search( TOX_STYLE_DELIMITER, 885 nTextSeparatorFirst + 1 ); 886 if( STRING_NOTFOUND != nTextSeparatorSecond && 887 nEnd > nTextSeparatorFirst ) 888 nEnd = sPattern.Search( '>', nTextSeparatorSecond ); 889 } 890 891 ++nEnd; 892 893 aResult = sPattern.Copy( nStt, nEnd - nStt ); 894 } 895 896 return aResult; 897 } 898 899 FormTokenType SwFormTokensHelper::GetTokenType(const String & sToken, 900 xub_StrLen * pTokenLen) const 901 { 902 static struct 903 { 904 const sal_Char* pNm; 905 sal_uInt16 nLen; 906 sal_uInt16 nOffset; 907 FormTokenType eToken; 908 } __READONLY_DATA aTokenArr[] = { 909 { SwForm::aFormTab, SwForm::nFormEntryLen, 1, TOKEN_TAB_STOP }, 910 { SwForm::aFormPageNums, SwForm::nFormPageNumsLen, 1, TOKEN_PAGE_NUMS }, 911 { SwForm::aFormLinkStt, SwForm::nFormLinkSttLen, 1, TOKEN_LINK_START }, 912 { SwForm::aFormLinkEnd, SwForm::nFormLinkEndLen, 1, TOKEN_LINK_END }, 913 { SwForm::aFormEntryNum, SwForm::nFormEntryNumLen, 1, TOKEN_ENTRY_NO }, 914 { SwForm::aFormEntryTxt, SwForm::nFormEntryTxtLen, 1, TOKEN_ENTRY_TEXT }, 915 { SwForm::aFormChapterMark,SwForm::nFormChapterMarkLen,1,TOKEN_CHAPTER_INFO }, 916 { SwForm::aFormText, SwForm::nFormTextLen, 1, TOKEN_TEXT }, 917 { SwForm::aFormEntry, SwForm::nFormEntryLen, 1, TOKEN_ENTRY }, 918 { SwForm::aFormAuth, SwForm::nFormAuthLen, 3, TOKEN_AUTHORITY }, 919 { 0, 0, 0, TOKEN_END } 920 }; 921 922 FormTokenType eTokenType = TOKEN_TEXT; 923 xub_StrLen nTokenLen = 0; 924 const sal_Char* pNm; 925 for( int i = 0; 0 != ( pNm = aTokenArr[ i ].pNm ); ++i ) 926 if( COMPARE_EQUAL == sToken.CompareToAscii( pNm, 927 aTokenArr[ i ].nLen - aTokenArr[ i ].nOffset )) 928 { 929 eTokenType = aTokenArr[ i ].eToken; 930 nTokenLen = aTokenArr[ i ].nLen; 931 break; 932 } 933 934 ASSERT( pNm, "wrong token" ); 935 if (pTokenLen) 936 *pTokenLen = nTokenLen; 937 938 return eTokenType; 939 } 940 941 // <- #i21237# 942 943 void SwForm::SetPattern(sal_uInt16 nLevel, const SwFormTokens& rTokens) 944 { 945 ASSERT(nLevel < GetFormMax(), "Index >= FORM_MAX"); 946 aPattern[nLevel] = rTokens; 947 } 948 949 void SwForm::SetPattern(sal_uInt16 nLevel, const String & rStr) 950 { 951 ASSERT(nLevel < GetFormMax(), "Index >= FORM_MAX"); 952 953 SwFormTokensHelper aHelper(rStr); 954 aPattern[nLevel] = aHelper.GetTokens(); 955 } 956 957 const SwFormTokens& SwForm::GetPattern(sal_uInt16 nLevel) const 958 { 959 ASSERT(nLevel < GetFormMax(), "Index >= FORM_MAX"); 960 return aPattern[nLevel]; 961 } 962 963