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 <hintids.hxx> 28 #include <rtl/logfile.hxx> 29 #include <vcl/outdev.hxx> 30 #include <sfx2/printer.hxx> 31 #include <editeng/eeitem.hxx> 32 #include <editeng/flditem.hxx> 33 #include <editeng/editeng.hxx> 34 #include <svx/svdoutl.hxx> 35 #include <editeng/colritem.hxx> 36 #include <svx/svdpage.hxx> 37 #include <svx/svdogrp.hxx> 38 #include <editeng/langitem.hxx> 39 #include <editeng/unolingu.hxx> 40 #include <editeng/measfld.hxx> 41 #include <svx/svdpool.hxx> 42 #include <fmtanchr.hxx> 43 #include <charatr.hxx> 44 #include <frmfmt.hxx> 45 #include <charfmt.hxx> 46 #include <viewimp.hxx> 47 #include <swhints.hxx> 48 #include <doc.hxx> 49 #include <IDocumentUndoRedo.hxx> 50 #include <docsh.hxx> 51 #include <rootfrm.hxx> //Damit der RootDtor gerufen wird. 52 #include <poolfmt.hxx> 53 #include <viewsh.hxx> // fuer MakeDrawView 54 #include <drawdoc.hxx> 55 #include <UndoDraw.hxx> 56 #include <swundo.hxx> // fuer die UndoIds 57 #include <dcontact.hxx> 58 #include <dview.hxx> 59 #include <mvsave.hxx> 60 #include <flyfrm.hxx> 61 #include <dflyobj.hxx> 62 #include <svx/svdetc.hxx> 63 #include <editeng/fhgtitem.hxx> 64 #include <svx/svdpagv.hxx> 65 #include <dcontact.hxx> 66 #include <txtfrm.hxx> 67 #include <frmfmt.hxx> 68 #include <editeng/frmdiritem.hxx> 69 #include <fmtornt.hxx> 70 #include <svx/svditer.hxx> 71 #include <vector> 72 #include <switerator.hxx> 73 74 using namespace ::com::sun::star; 75 using namespace ::com::sun::star::linguistic2; 76 77 78 SV_IMPL_VARARR_SORT( _ZSortFlys, _ZSortFly ) 79 80 /************************************************************************* 81 |* 82 |* SwDoc::GroupSelection / SwDoc::UnGroupSelection 83 |* 84 |* Ersterstellung JP 21.08.95 85 |* Letzte Aenderung JP 21.08.95 86 |* 87 |*************************************************************************/ 88 // OD 2004-04-01 #i26791# - local method to determine positioning and 89 // alignment attributes for a drawing object, which is newly connected to 90 // the layout. Used for a newly formed group object <SwDoc::GroupSelection(..)> 91 // and the members of a destroyed group <SwDoc::UnGroupSelection(..)> 92 void lcl_AdjustPositioningAttr( SwDrawFrmFmt* _pFrmFmt, 93 const SdrObject& _rSdrObj ) 94 { 95 const SwContact* pContact = GetUserCall( &_rSdrObj ); 96 ASSERT( pContact, "<lcl_AdjustPositioningAttr(..)> - missing contact object." ); 97 98 // determine position of new group object relative to its anchor frame position 99 SwTwips nHoriRelPos = 0; 100 SwTwips nVertRelPos = 0; 101 { 102 const SwFrm* pAnchorFrm = pContact->GetAnchoredObj( &_rSdrObj )->GetAnchorFrm(); 103 ASSERT( !pAnchorFrm || 104 !pAnchorFrm->IsTxtFrm() || 105 !static_cast<const SwTxtFrm*>(pAnchorFrm)->IsFollow(), 106 "<lcl_AdjustPositioningAttr(..)> - anchor frame is a follow. Please inform OD." ); 107 bool bVert = false; 108 bool bR2L = false; 109 // --> OD 2005-05-10 #i45952# - use anchor position of 110 // anchor frame, if it exist. 111 Point aAnchorPos; 112 if ( pAnchorFrm ) 113 { 114 // --> OD 2005-05-10 #i45952# 115 aAnchorPos = pAnchorFrm->GetFrmAnchorPos( ::HasWrap( &_rSdrObj ) ); 116 // <-- 117 bVert = pAnchorFrm->IsVertical(); 118 bR2L = pAnchorFrm->IsRightToLeft(); 119 } 120 else 121 { 122 // --> OD 2005-05-10 #i45952# 123 aAnchorPos = _rSdrObj.GetAnchorPos(); 124 // <-- 125 // If no anchor frame exist - e.g. because no layout exists - the 126 // default layout direction is taken. 127 const SvxFrameDirectionItem* pDirItem = 128 static_cast<const SvxFrameDirectionItem*>(&(_pFrmFmt->GetAttrSet().GetPool()->GetDefaultItem( RES_FRAMEDIR ))); 129 switch ( pDirItem->GetValue() ) 130 { 131 case FRMDIR_VERT_TOP_LEFT: 132 { 133 // vertical from left-to-right - Badaa: supported now! 134 bVert = true; 135 bR2L = true; 136 //Badaa: 2008-04-18 * Support for Classical Mongolian Script (SCMS) joint with Jiayanmin 137 //ASSERT( false, "<lcl_AdjustPositioningAttr(..)> - vertical from left-to-right not supported." ); 138 //End 139 } 140 break; 141 case FRMDIR_VERT_TOP_RIGHT: 142 { 143 // vertical from right-to-left 144 bVert = true; 145 bR2L = false; 146 } 147 break; 148 case FRMDIR_HORI_RIGHT_TOP: 149 { 150 // horizontal from right-to-left 151 bVert = false; 152 bR2L = true; 153 } 154 break; 155 case FRMDIR_HORI_LEFT_TOP: 156 { 157 // horizontal from left-to-right 158 bVert = false; 159 bR2L = false; 160 } 161 break; 162 } 163 164 } 165 // use geometry of drawing object 166 const SwRect aObjRect = _rSdrObj.GetSnapRect(); 167 168 if ( bVert ) 169 { 170 if ( bR2L ) { 171 //FRMDIR_VERT_TOP_LEFT 172 nHoriRelPos = aObjRect.Left() - aAnchorPos.X(); 173 nVertRelPos = aObjRect.Top() - aAnchorPos.Y(); 174 } else { 175 //FRMDIR_VERT_TOP_RIGHT 176 nHoriRelPos = aObjRect.Top() - aAnchorPos.Y(); 177 nVertRelPos = aAnchorPos.X() - aObjRect.Right(); 178 } 179 } 180 else if ( bR2L ) 181 { 182 nHoriRelPos = aAnchorPos.X() - aObjRect.Right(); 183 nVertRelPos = aObjRect.Top() - aAnchorPos.Y(); 184 } 185 else 186 { 187 nHoriRelPos = aObjRect.Left() - aAnchorPos.X(); 188 nVertRelPos = aObjRect.Top() - aAnchorPos.Y(); 189 } 190 //End of SCMS 191 } 192 193 _pFrmFmt->SetFmtAttr( SwFmtHoriOrient( nHoriRelPos, text::HoriOrientation::NONE, text::RelOrientation::FRAME ) ); 194 _pFrmFmt->SetFmtAttr( SwFmtVertOrient( nVertRelPos, text::VertOrientation::NONE, text::RelOrientation::FRAME ) ); 195 // --> OD 2005-03-11 #i44334#, #i44681# - positioning attributes already set 196 _pFrmFmt->PosAttrSet(); 197 // <-- 198 // --> OD 2004-10-01 #i34750# - keep current object rectangle for drawing 199 // objects. The object rectangle is used on events from the drawing layer 200 // to adjust the positioning attributes - see <SwDrawContact::_Changed(..)>. 201 { 202 const SwAnchoredObject* pAnchoredObj = pContact->GetAnchoredObj( &_rSdrObj ); 203 if ( pAnchoredObj->ISA(SwAnchoredDrawObject) ) 204 { 205 const SwAnchoredDrawObject* pAnchoredDrawObj = 206 static_cast<const SwAnchoredDrawObject*>(pAnchoredObj); 207 const SwRect aObjRect = _rSdrObj.GetSnapRect(); 208 const_cast<SwAnchoredDrawObject*>(pAnchoredDrawObj) 209 ->SetLastObjRect( aObjRect.SVRect() ); 210 } 211 } 212 // <-- 213 } 214 215 SwDrawContact* SwDoc::GroupSelection( SdrView& rDrawView ) 216 { 217 // OD 30.06.2003 #108784# - replace marked 'virtual' drawing objects by 218 // the corresponding 'master' drawing objects. 219 SwDrawView::ReplaceMarkedDrawVirtObjs( rDrawView ); 220 221 const SdrMarkList &rMrkList = rDrawView.GetMarkedObjectList(); 222 SwDrawFrmFmt *pFmt = 0L; 223 SdrObject *pObj = rMrkList.GetMark( 0 )->GetMarkedSdrObj(); 224 sal_Bool bNoGroup = ( 0 == pObj->GetUpGroup() ); 225 SwDrawContact* pNewContact = 0; 226 if( bNoGroup ) 227 { 228 //Ankerattribut aufheben. 229 SwDrawContact *pMyContact = (SwDrawContact*)GetUserCall(pObj); 230 const SwFmtAnchor aAnch( pMyContact->GetFmt()->GetAnchor() ); 231 232 SwUndoDrawGroup *const pUndo = (!GetIDocumentUndoRedo().DoesUndo()) 233 ? 0 234 : new SwUndoDrawGroup( (sal_uInt16)rMrkList.GetMarkCount() ); 235 236 // --> OD 2005-08-16 #i53320# 237 bool bGroupMembersNotPositioned( false ); 238 { 239 SwAnchoredDrawObject* pAnchoredDrawObj = 240 static_cast<SwAnchoredDrawObject*>(pMyContact->GetAnchoredObj( pObj )); 241 bGroupMembersNotPositioned = pAnchoredDrawObj->NotYetPositioned(); 242 } 243 // <-- 244 //ContactObjekte und Formate vernichten. 245 for( sal_uInt16 i = 0; i < rMrkList.GetMarkCount(); ++i ) 246 { 247 pObj = rMrkList.GetMark( i )->GetMarkedSdrObj(); 248 SwDrawContact *pContact = (SwDrawContact*)GetUserCall(pObj); 249 250 // --> OD 2005-08-16 #i53320# 251 #ifdef DBG_UTIL 252 SwAnchoredDrawObject* pAnchoredDrawObj = 253 static_cast<SwAnchoredDrawObject*>(pContact->GetAnchoredObj( pObj )); 254 ASSERT( bGroupMembersNotPositioned == pAnchoredDrawObj->NotYetPositioned(), 255 "<SwDoc::GroupSelection(..)> - group members have different positioning status!" ); 256 #endif 257 // <-- 258 259 pFmt = (SwDrawFrmFmt*)pContact->GetFmt(); 260 //loescht sich selbst! 261 pContact->Changed(*pObj, SDRUSERCALL_DELETE, pObj->GetLastBoundRect() ); 262 pObj->SetUserCall( 0 ); 263 264 if( pUndo ) 265 pUndo->AddObj( i, pFmt, pObj ); 266 else 267 DelFrmFmt( pFmt ); 268 269 // --> OD 2005-05-10 #i45952# - re-introduce position 270 // normalization of group member objects, because its anchor position 271 // is cleared, when they are grouped. 272 Point aAnchorPos( pObj->GetAnchorPos() ); 273 pObj->NbcSetAnchorPos( Point( 0, 0 ) ); 274 pObj->NbcMove( Size( aAnchorPos.X(), aAnchorPos.Y() ) ); 275 // <-- 276 } 277 278 pFmt = MakeDrawFrmFmt( String::CreateFromAscii( 279 RTL_CONSTASCII_STRINGPARAM( "DrawObject" )), 280 GetDfltFrmFmt() ); 281 pFmt->SetFmtAttr( aAnch ); 282 // --> OD 2004-10-25 #i36010# - set layout direction of the position 283 pFmt->SetPositionLayoutDir( 284 text::PositionLayoutDir::PositionInLayoutDirOfAnchor ); 285 // <-- 286 287 rDrawView.GroupMarked(); 288 ASSERT( rMrkList.GetMarkCount() == 1, "GroupMarked more or none groups." ); 289 290 SdrObject* pNewGroupObj = rMrkList.GetMark( 0 )->GetMarkedSdrObj(); 291 pNewContact = new SwDrawContact( pFmt, pNewGroupObj ); 292 // --> OD 2004-11-22 #i35635# 293 pNewContact->MoveObjToVisibleLayer( pNewGroupObj ); 294 // <-- 295 pNewContact->ConnectToLayout(); 296 // --> OD 2005-08-16 #i53320# - No adjustment of the positioning and 297 // alignment attributes, if group members aren't positioned yet. 298 if ( !bGroupMembersNotPositioned ) 299 { 300 // OD 2004-04-01 #i26791# - Adjust positioning and alignment attributes. 301 lcl_AdjustPositioningAttr( pFmt, *pNewGroupObj ); 302 } 303 // <-- 304 305 if( pUndo ) 306 { 307 pUndo->SetGroupFmt( pFmt ); 308 GetIDocumentUndoRedo().AppendUndo( pUndo ); 309 } 310 } 311 else 312 { 313 if (GetIDocumentUndoRedo().DoesUndo()) 314 { 315 GetIDocumentUndoRedo().ClearRedo(); 316 } 317 318 rDrawView.GroupMarked(); 319 ASSERT( rMrkList.GetMarkCount() == 1, "GroupMarked more or none groups." ); 320 } 321 322 return pNewContact; 323 } 324 325 326 void SwDoc::UnGroupSelection( SdrView& rDrawView ) 327 { 328 bool const bUndo = GetIDocumentUndoRedo().DoesUndo(); 329 if( bUndo ) 330 { 331 GetIDocumentUndoRedo().ClearRedo(); 332 } 333 334 // OD 30.06.2003 #108784# - replace marked 'virtual' drawing objects by 335 // the corresponding 'master' drawing objects. 336 SwDrawView::ReplaceMarkedDrawVirtObjs( rDrawView ); 337 338 const SdrMarkList &rMrkList = rDrawView.GetMarkedObjectList(); 339 // --> OD 2006-11-01 #130889# 340 std::vector< std::pair< SwDrawFrmFmt*, SdrObject* > >* pFmtsAndObjs( 0L ); 341 const sal_uInt32 nMarkCount( rMrkList.GetMarkCount() ); 342 // <-- 343 if ( nMarkCount ) 344 { 345 // --> OD 2006-11-01 #130889# 346 pFmtsAndObjs = new std::vector< std::pair< SwDrawFrmFmt*, SdrObject* > >[nMarkCount]; 347 // <-- 348 SdrObject *pMyObj = rMrkList.GetMark( 0 )->GetMarkedSdrObj(); 349 if( !pMyObj->GetUpGroup() ) 350 { 351 String sDrwFmtNm( String::CreateFromAscii( 352 RTL_CONSTASCII_STRINGPARAM("DrawObject" ))); 353 for ( sal_uInt16 i = 0; i < nMarkCount; ++i ) 354 { 355 SdrObject *pObj = rMrkList.GetMark( i )->GetMarkedSdrObj(); 356 if ( pObj->IsA( TYPE(SdrObjGroup) ) ) 357 { 358 SwDrawContact *pContact = (SwDrawContact*)GetUserCall(pObj); 359 SwFmtAnchor aAnch( pContact->GetFmt()->GetAnchor() ); 360 SdrObjList *pLst = ((SdrObjGroup*)pObj)->GetSubList(); 361 362 SwUndoDrawUnGroup* pUndo = 0; 363 if( bUndo ) 364 { 365 pUndo = new SwUndoDrawUnGroup( (SdrObjGroup*)pObj ); 366 GetIDocumentUndoRedo().AppendUndo(pUndo); 367 } 368 369 for ( sal_uInt16 i2 = 0; i2 < pLst->GetObjCount(); ++i2 ) 370 { 371 SdrObject* pSubObj = pLst->GetObj( i2 ); 372 SwDrawFrmFmt *pFmt = MakeDrawFrmFmt( sDrwFmtNm, 373 GetDfltFrmFmt() ); 374 pFmt->SetFmtAttr( aAnch ); 375 // --> OD 2004-10-25 #i36010# - set layout direction of the position 376 pFmt->SetPositionLayoutDir( 377 text::PositionLayoutDir::PositionInLayoutDirOfAnchor ); 378 // <-- 379 // --> OD 2006-11-01 #130889# 380 // creation of <SwDrawContact> instances for the group 381 // members and its connection to the Writer layout is 382 // done after intrinsic ungrouping. 383 // SwDrawContact* pContact = new SwDrawContact( pFmt, pSubObj ); 384 // // --> OD 2004-11-22 #i35635# 385 // pContact->MoveObjToVisibleLayer( pSubObj ); 386 // // <-- 387 // pContact->ConnectToLayout(); 388 // // OD 2004-04-07 #i26791# - Adjust positioning and 389 // // alignment attributes. 390 // lcl_AdjustPositioningAttr( pFmt, *pSubObj ); 391 pFmtsAndObjs[i].push_back( std::pair< SwDrawFrmFmt*, SdrObject* >( pFmt, pSubObj ) ); 392 // <-- 393 394 if( bUndo ) 395 pUndo->AddObj( i2, pFmt ); 396 } 397 } 398 } 399 } 400 } 401 rDrawView.UnGroupMarked(); 402 // --> OD 2006-11-01 #130889# 403 // creation of <SwDrawContact> instances for the former group members and 404 // its connection to the Writer layout. 405 for ( sal_uInt32 i = 0; i < nMarkCount; ++i ) 406 { 407 SwUndoDrawUnGroupConnectToLayout* pUndo = 0; 408 if( bUndo ) 409 { 410 pUndo = new SwUndoDrawUnGroupConnectToLayout(); 411 GetIDocumentUndoRedo().AppendUndo(pUndo); 412 } 413 414 while ( pFmtsAndObjs[i].size() > 0 ) 415 { 416 SwDrawFrmFmt* pFmt( pFmtsAndObjs[i].back().first ); 417 SdrObject* pObj( pFmtsAndObjs[i].back().second ); 418 pFmtsAndObjs[i].pop_back(); 419 420 SwDrawContact* pContact = new SwDrawContact( pFmt, pObj ); 421 pContact->MoveObjToVisibleLayer( pObj ); 422 pContact->ConnectToLayout(); 423 lcl_AdjustPositioningAttr( pFmt, *pObj ); 424 425 if ( bUndo ) 426 { 427 pUndo->AddFmtAndObj( pFmt, pObj ); 428 } 429 } 430 } 431 delete [] pFmtsAndObjs; 432 // <-- 433 } 434 435 /************************************************************************* 436 |* 437 |* SwDoc::DeleteSelection() 438 |* 439 |* Ersterstellung MA 14. Nov. 95 440 |* Letzte Aenderung MA 14. Nov. 95 441 |* 442 |*************************************************************************/ 443 444 sal_Bool SwDoc::DeleteSelection( SwDrawView& rDrawView ) 445 { 446 sal_Bool bCallBase = sal_False; 447 const SdrMarkList &rMrkList = rDrawView.GetMarkedObjectList(); 448 if( rMrkList.GetMarkCount() ) 449 { 450 GetIDocumentUndoRedo().StartUndo(UNDO_EMPTY, NULL); 451 sal_uInt16 i; 452 sal_Bool bDelMarked = sal_True; 453 454 if( 1 == rMrkList.GetMarkCount() ) 455 { 456 SdrObject *pObj = rMrkList.GetMark( 0 )->GetMarkedSdrObj(); 457 if( pObj->ISA(SwVirtFlyDrawObj) ) 458 { 459 SwFlyFrmFmt* pFrmFmt = (SwFlyFrmFmt*) 460 ((SwVirtFlyDrawObj*)pObj)->GetFlyFrm()->GetFmt(); 461 if( pFrmFmt ) 462 { 463 DelLayoutFmt( pFrmFmt ); 464 bDelMarked = sal_False; 465 } 466 } 467 } 468 469 for( i = 0; i < rMrkList.GetMarkCount(); ++i ) 470 { 471 SdrObject *pObj = rMrkList.GetMark( i )->GetMarkedSdrObj(); 472 if( !pObj->ISA(SwVirtFlyDrawObj) ) 473 { 474 SwDrawContact *pC = (SwDrawContact*)GetUserCall(pObj); 475 SwDrawFrmFmt *pFrmFmt = (SwDrawFrmFmt*)pC->GetFmt(); 476 if( pFrmFmt && 477 FLY_AS_CHAR == pFrmFmt->GetAnchor().GetAnchorId() ) 478 { 479 rDrawView.MarkObj( pObj, rDrawView.Imp().GetPageView(), sal_True ); 480 --i; 481 DelLayoutFmt( pFrmFmt ); 482 } 483 } 484 } 485 486 if( rMrkList.GetMarkCount() && bDelMarked ) 487 { 488 SdrObject *pObj = rMrkList.GetMark( 0 )->GetMarkedSdrObj(); 489 if( !pObj->GetUpGroup() ) 490 { 491 SwUndoDrawDelete *const pUndo = 492 (!GetIDocumentUndoRedo().DoesUndo()) 493 ? 0 494 : new SwUndoDrawDelete( (sal_uInt16)rMrkList.GetMarkCount() ); 495 496 //ContactObjekte vernichten, Formate sicherstellen. 497 for( i = 0; i < rMrkList.GetMarkCount(); ++i ) 498 { 499 const SdrMark& rMark = *rMrkList.GetMark( i ); 500 pObj = rMark.GetMarkedSdrObj(); 501 SwDrawContact *pContact = (SwDrawContact*)pObj->GetUserCall(); 502 if( pContact ) // natuerlich nicht bei gruppierten Objekten 503 { 504 SwDrawFrmFmt *pFmt = (SwDrawFrmFmt*)pContact->GetFmt(); 505 // OD 18.06.2003 #108784# - before delete of selection 506 // is performed, marked <SwDrawVirtObj>-objects have to 507 // be replaced by its reference objects. 508 // Thus, assert, if a <SwDrawVirt>-object is found in the mark list. 509 if ( pObj->ISA(SwDrawVirtObj) ) 510 { 511 ASSERT( false, 512 "<SwDrawVirtObj> is still marked for delete. application will crash!" ); 513 } 514 //loescht sich selbst! 515 pContact->Changed(*pObj, SDRUSERCALL_DELETE, pObj->GetLastBoundRect() ); 516 pObj->SetUserCall( 0 ); 517 518 if( pUndo ) 519 pUndo->AddObj( i, pFmt, rMark ); 520 else 521 DelFrmFmt( pFmt ); 522 } 523 } 524 525 if( pUndo ) 526 { 527 GetIDocumentUndoRedo().AppendUndo( pUndo ); 528 } 529 } 530 bCallBase = sal_True; 531 } 532 SetModified(); 533 534 GetIDocumentUndoRedo().EndUndo(UNDO_EMPTY, NULL); 535 } 536 537 return bCallBase; 538 } 539 540 /************************************************************************* 541 |* 542 |* SwDoc::DeleteSelection() 543 |* 544 |* Ersterstellung JP 11.01.96 545 |* Letzte Aenderung JP 11.01.96 546 |* 547 |*************************************************************************/ 548 549 _ZSortFly::_ZSortFly( const SwFrmFmt* pFrmFmt, const SwFmtAnchor* pFlyAn, 550 sal_uInt32 nArrOrdNum ) 551 : pFmt( pFrmFmt ), pAnchor( pFlyAn ), nOrdNum( nArrOrdNum ) 552 { 553 // #i11176# 554 // This also needs to work when no layout exists. Thus, for 555 // FlyFrames an alternative method is used now in that case. 556 if( RES_FLYFRMFMT == pFmt->Which() ) 557 { 558 if( pFmt->getIDocumentLayoutAccess()->GetCurrentViewShell() ) //swmod 071107//swmod 071225 559 { 560 // Schauen, ob es ein SdrObject dafuer gibt 561 SwFlyFrm* pFly = SwIterator<SwFlyFrm,SwFmt>::FirstElement( *pFrmFmt ); 562 if( pFly ) 563 nOrdNum = pFly->GetVirtDrawObj()->GetOrdNum(); 564 } 565 else 566 { 567 // Schauen, ob es ein SdrObject dafuer gibt 568 SwFlyDrawContact* pContact = SwIterator<SwFlyDrawContact,SwFmt>::FirstElement( *pFrmFmt ); 569 if( pContact ) 570 nOrdNum = pContact->GetMaster()->GetOrdNum(); 571 } 572 } 573 else if( RES_DRAWFRMFMT == pFmt->Which() ) 574 { 575 // Schauen, ob es ein SdrObject dafuer gibt 576 SwDrawContact* pContact = SwIterator<SwDrawContact,SwFmt>::FirstElement( *pFrmFmt ); 577 if( pContact ) 578 nOrdNum = pContact->GetMaster()->GetOrdNum(); 579 } 580 else { 581 ASSERT( !this, "was ist das fuer ein Format?" ); 582 } 583 } 584 585 /*************************************************************************/ 586 // Wird auch vom Sw3-Reader gerufen, wenn ein Fehler beim Einlesen 587 // des Drawing Layers auftrat. In diesem Fall wird der Layer komplett 588 // neu aufgebaut. 589 590 // #75371# 591 #include <svx/sxenditm.hxx> 592 593 void SwDoc::InitDrawModel() 594 { 595 RTL_LOGFILE_CONTEXT_AUTHOR( aLog, "SW", "JP93722", "SwDoc::InitDrawModel" ); 596 597 //!!Achtung im sw3-Reader (sw3imp.cxx) gibt es aehnlichen Code, der 598 //mitgepfelgt werden muss. 599 if ( pDrawModel ) 600 ReleaseDrawModel(); 601 602 //DrawPool und EditEnginePool anlegen, diese gehoeren uns und werden 603 //dem Drawing nur mitgegeben. Im ReleaseDrawModel werden die Pools wieder 604 //zerstoert. 605 // 17.2.99: for Bug 73110 - for loading the drawing items. This must 606 // be loaded without RefCounts! 607 SfxItemPool *pSdrPool = new SdrItemPool( &GetAttrPool() ); 608 // #75371# change DefaultItems for the SdrEdgeObj distance items 609 // to TWIPS. 610 if(pSdrPool) 611 { 612 const long nDefEdgeDist = ((500 * 72) / 127); // 1/100th mm in twips 613 pSdrPool->SetPoolDefaultItem(SdrEdgeNode1HorzDistItem(nDefEdgeDist)); 614 pSdrPool->SetPoolDefaultItem(SdrEdgeNode1VertDistItem(nDefEdgeDist)); 615 pSdrPool->SetPoolDefaultItem(SdrEdgeNode2HorzDistItem(nDefEdgeDist)); 616 pSdrPool->SetPoolDefaultItem(SdrEdgeNode2VertDistItem(nDefEdgeDist)); 617 618 // #i33700# 619 // Set shadow distance defaults as PoolDefaultItems. Details see bug. 620 pSdrPool->SetPoolDefaultItem(SdrShadowXDistItem((300 * 72) / 127)); 621 pSdrPool->SetPoolDefaultItem(SdrShadowYDistItem((300 * 72) / 127)); 622 } 623 SfxItemPool *pEEgPool = EditEngine::CreatePool( sal_False ); 624 pSdrPool->SetSecondaryPool( pEEgPool ); 625 if ( !GetAttrPool().GetFrozenIdRanges () ) 626 GetAttrPool().FreezeIdRanges(); 627 else 628 pSdrPool->FreezeIdRanges(); 629 630 // SJ: #95129# set FontHeight pool defaults without changing static SdrEngineDefaults 631 GetAttrPool().SetPoolDefaultItem(SvxFontHeightItem( 240, 100, EE_CHAR_FONTHEIGHT )); 632 633 RTL_LOGFILE_CONTEXT_TRACE( aLog, "before create DrawDocument" ); 634 //Das SdrModel gehoert dem Dokument, wir haben immer zwei Layer und eine 635 //Seite. 636 pDrawModel = new SwDrawDocument( this ); 637 638 pDrawModel->EnableUndo( GetIDocumentUndoRedo().DoesUndo() ); 639 640 String sLayerNm; 641 sLayerNm.AssignAscii(RTL_CONSTASCII_STRINGPARAM("Hell" )); 642 nHell = pDrawModel->GetLayerAdmin().NewLayer( sLayerNm )->GetID(); 643 644 sLayerNm.AssignAscii(RTL_CONSTASCII_STRINGPARAM("Heaven" )); 645 nHeaven = pDrawModel->GetLayerAdmin().NewLayer( sLayerNm )->GetID(); 646 647 sLayerNm.AssignAscii(RTL_CONSTASCII_STRINGPARAM("Controls" )); 648 nControls = pDrawModel->GetLayerAdmin().NewLayer( sLayerNm )->GetID(); 649 650 // OD 25.06.2003 #108784# - add invisible layers corresponding to the 651 // visible ones. 652 { 653 sLayerNm.AssignAscii(RTL_CONSTASCII_STRINGPARAM("InvisibleHell" )); 654 nInvisibleHell = pDrawModel->GetLayerAdmin().NewLayer( sLayerNm )->GetID(); 655 656 sLayerNm.AssignAscii(RTL_CONSTASCII_STRINGPARAM("InvisibleHeaven" )); 657 nInvisibleHeaven = pDrawModel->GetLayerAdmin().NewLayer( sLayerNm )->GetID(); 658 659 sLayerNm.AssignAscii(RTL_CONSTASCII_STRINGPARAM("InvisibleControls" )); 660 nInvisibleControls = pDrawModel->GetLayerAdmin().NewLayer( sLayerNm )->GetID(); 661 } 662 663 SdrPage* pMasterPage = pDrawModel->AllocPage( sal_False ); 664 pDrawModel->InsertPage( pMasterPage ); 665 RTL_LOGFILE_CONTEXT_TRACE( aLog, "after create DrawDocument" ); 666 667 RTL_LOGFILE_CONTEXT_TRACE( aLog, "before create Spellchecker/Hyphenator" ); 668 SdrOutliner& rOutliner = pDrawModel->GetDrawOutliner(); 669 uno::Reference< XSpellChecker1 > xSpell = ::GetSpellChecker(); 670 rOutliner.SetSpeller( xSpell ); 671 uno::Reference<XHyphenator> xHyphenator( ::GetHyphenator() ); 672 rOutliner.SetHyphenator( xHyphenator ); 673 RTL_LOGFILE_CONTEXT_TRACE( aLog, "after create Spellchecker/Hyphenator" ); 674 675 SetCalcFieldValueHdl(&rOutliner); 676 SetCalcFieldValueHdl(&pDrawModel->GetHitTestOutliner()); 677 678 //JP 16.07.98: Bug 50193 - Linkmanager am Model setzen, damit 679 // dort ggfs. verlinkte Grafiken eingefuegt werden koennen 680 //JP 28.01.99: der WinWord Import benoetigt ihn auch 681 pDrawModel->SetLinkManager( &GetLinkManager() ); 682 pDrawModel->SetAddExtLeading( get(IDocumentSettingAccess::ADD_EXT_LEADING) ); 683 684 OutputDevice* pRefDev = getReferenceDevice( false ); 685 if ( pRefDev ) 686 pDrawModel->SetRefDevice( pRefDev ); 687 688 pDrawModel->SetNotifyUndoActionHdl( LINK( this, SwDoc, AddDrawUndo )); 689 if ( pCurrentView ) 690 { 691 ViewShell* pViewSh = pCurrentView; 692 do 693 { 694 SwRootFrm* pRoot = pViewSh->GetLayout(); 695 if( pRoot && !pRoot->GetDrawPage() ) 696 { 697 // Disable "multiple layout" for the moment: 698 // use pMasterPage instead of a new created SdrPage 699 // pDrawModel->AllocPage( FALSE ); 700 // pDrawModel->InsertPage( pDrawPage ); 701 SdrPage* pDrawPage = pMasterPage; 702 pRoot->SetDrawPage( pDrawPage ); 703 pDrawPage->SetSize( pRoot->Frm().SSize() ); 704 } 705 pViewSh = (ViewShell*)pViewSh->GetNext(); 706 }while( pViewSh != pCurrentView ); 707 } 708 } 709 710 /** method to notify drawing page view about the invisible layers 711 712 OD 26.06.2003 #108784# 713 714 @author OD 715 */ 716 void SwDoc::NotifyInvisibleLayers( SdrPageView& _rSdrPageView ) 717 { 718 String sLayerNm; 719 sLayerNm.AssignAscii(RTL_CONSTASCII_STRINGPARAM("InvisibleHell" )); 720 _rSdrPageView.SetLayerVisible( sLayerNm, sal_False ); 721 722 sLayerNm.AssignAscii(RTL_CONSTASCII_STRINGPARAM("InvisibleHeaven" )); 723 _rSdrPageView.SetLayerVisible( sLayerNm, sal_False ); 724 725 sLayerNm.AssignAscii(RTL_CONSTASCII_STRINGPARAM("InvisibleControls" )); 726 _rSdrPageView.SetLayerVisible( sLayerNm, sal_False ); 727 } 728 729 /** method to determine, if a layer ID belongs to the visible ones. 730 731 OD 25.06.2003 #108784# 732 Note: If given layer ID is unknown, method asserts and returns <false>. 733 734 @author OD 735 */ 736 bool SwDoc::IsVisibleLayerId( const SdrLayerID& _nLayerId ) const 737 { 738 bool bRetVal; 739 740 if ( _nLayerId == GetHeavenId() || 741 _nLayerId == GetHellId() || 742 _nLayerId == GetControlsId() ) 743 { 744 bRetVal = true; 745 } 746 else if ( _nLayerId == GetInvisibleHeavenId() || 747 _nLayerId == GetInvisibleHellId() || 748 _nLayerId == GetInvisibleControlsId() ) 749 { 750 bRetVal = false; 751 } 752 else 753 { 754 ASSERT( false, "<SwDoc::IsVisibleLayerId(..)> - unknown layer ID." ); 755 bRetVal = false; 756 } 757 758 return bRetVal; 759 } 760 761 /** method to determine, if the corresponding visible layer ID for a invisible one. 762 763 OD 25.06.2003 #108784# 764 Note: If given layer ID is a visible one, method returns given layer ID. 765 Note: If given layer ID is unknown, method returns given layer ID. 766 767 @author OD 768 */ 769 SdrLayerID SwDoc::GetVisibleLayerIdByInvisibleOne( const SdrLayerID& _nInvisibleLayerId ) 770 { 771 SdrLayerID nVisibleLayerId; 772 773 if ( _nInvisibleLayerId == GetInvisibleHeavenId() ) 774 { 775 nVisibleLayerId = GetHeavenId(); 776 } 777 else if ( _nInvisibleLayerId == GetInvisibleHellId() ) 778 { 779 nVisibleLayerId = GetHellId(); 780 } 781 else if ( _nInvisibleLayerId == GetInvisibleControlsId() ) 782 { 783 nVisibleLayerId = GetControlsId(); 784 } 785 else if ( _nInvisibleLayerId == GetHeavenId() || 786 _nInvisibleLayerId == GetHellId() || 787 _nInvisibleLayerId == GetControlsId() ) 788 { 789 ASSERT( false, "<SwDoc::GetVisibleLayerIdByInvisibleOne(..)> - given layer ID already an invisible one." ); 790 nVisibleLayerId = _nInvisibleLayerId; 791 } 792 else 793 { 794 ASSERT( false, "<SwDoc::GetVisibleLayerIdByInvisibleOne(..)> - given layer ID is unknown." ); 795 nVisibleLayerId = _nInvisibleLayerId; 796 } 797 798 return nVisibleLayerId; 799 } 800 801 /** method to determine, if the corresponding invisible layer ID for a visible one. 802 803 OD 25.06.2003 #108784# 804 Note: If given layer ID is a invisible one, method returns given layer ID. 805 Note: If given layer ID is unknown, method returns given layer ID. 806 807 @author OD 808 */ 809 SdrLayerID SwDoc::GetInvisibleLayerIdByVisibleOne( const SdrLayerID& _nVisibleLayerId ) 810 { 811 SdrLayerID nInvisibleLayerId; 812 813 if ( _nVisibleLayerId == GetHeavenId() ) 814 { 815 nInvisibleLayerId = GetInvisibleHeavenId(); 816 } 817 else if ( _nVisibleLayerId == GetHellId() ) 818 { 819 nInvisibleLayerId = GetInvisibleHellId(); 820 } 821 else if ( _nVisibleLayerId == GetControlsId() ) 822 { 823 nInvisibleLayerId = GetInvisibleControlsId(); 824 } 825 else if ( _nVisibleLayerId == GetInvisibleHeavenId() || 826 _nVisibleLayerId == GetInvisibleHellId() || 827 _nVisibleLayerId == GetInvisibleControlsId() ) 828 { 829 ASSERT( false, "<SwDoc::GetInvisibleLayerIdByVisibleOne(..)> - given layer ID already an invisible one." ); 830 nInvisibleLayerId = _nVisibleLayerId; 831 } 832 else 833 { 834 ASSERT( false, "<SwDoc::GetInvisibleLayerIdByVisibleOne(..)> - given layer ID is unknown." ); 835 nInvisibleLayerId = _nVisibleLayerId; 836 } 837 838 return nInvisibleLayerId; 839 } 840 841 /*************************************************************************/ 842 843 844 void SwDoc::ReleaseDrawModel() 845 { 846 if ( pDrawModel ) 847 { 848 //!!Den code im sw3io fuer Einfuegen Dokument mitpflegen!! 849 850 delete pDrawModel; pDrawModel = 0; 851 SfxItemPool *pSdrPool = GetAttrPool().GetSecondaryPool(); 852 853 ASSERT( pSdrPool, "missing Pool" ); 854 SfxItemPool *pEEgPool = pSdrPool->GetSecondaryPool(); 855 ASSERT( !pEEgPool->GetSecondaryPool(), "i don't accept additional pools"); 856 pSdrPool->Delete(); //Erst die Items vernichten lassen, 857 //dann erst die Verkettung loesen 858 GetAttrPool().SetSecondaryPool( 0 ); //Der ist ein muss! 859 pSdrPool->SetSecondaryPool( 0 ); //Der ist sicherer 860 SfxItemPool::Free(pSdrPool); 861 SfxItemPool::Free(pEEgPool); 862 } 863 } 864 865 /*************************************************************************/ 866 867 868 SdrModel* SwDoc::_MakeDrawModel() 869 { 870 ASSERT( !pDrawModel, "_MakeDrawModel: Why?" ); 871 InitDrawModel(); 872 if ( pCurrentView ) 873 { 874 ViewShell* pTmp = pCurrentView; 875 do 876 { 877 pTmp->MakeDrawView(); 878 pTmp = (ViewShell*) pTmp->GetNext(); 879 } while ( pTmp != pCurrentView ); 880 881 //Broadcast, damit die FormShell mit der DrawView verbunden werden kann 882 if( GetDocShell() ) 883 { 884 SfxSimpleHint aHnt( SW_BROADCAST_DRAWVIEWS_CREATED ); 885 GetDocShell()->Broadcast( aHnt ); 886 } 887 } //swmod 071029//swmod 071225 888 return pDrawModel; 889 } 890 891 /*************************************************************************/ 892 893 void SwDoc::DrawNotifyUndoHdl() 894 { 895 pDrawModel->SetNotifyUndoActionHdl( Link() ); 896 } 897 898 /************************************************************************* 899 * 900 * Am Outliner Link auf Methode fuer Felddarstellung in Editobjekten setzen 901 * 902 *************************************************************************/ 903 904 void SwDoc::SetCalcFieldValueHdl(Outliner* pOutliner) 905 { 906 pOutliner->SetCalcFieldValueHdl(LINK(this, SwDoc, CalcFieldValueHdl)); 907 } 908 909 /************************************************************************* 910 |* 911 |* Felder bzw URLs im Outliner erkennen und Darstellung festlegen 912 |* 913 \************************************************************************/ 914 915 IMPL_LINK(SwDoc, CalcFieldValueHdl, EditFieldInfo*, pInfo) 916 { 917 if (pInfo) 918 { 919 const SvxFieldItem& rField = pInfo->GetField(); 920 const SvxFieldData* pField = rField.GetField(); 921 922 if (pField && pField->ISA(SvxDateField)) 923 { 924 /****************************************************************** 925 * Date-Field 926 ******************************************************************/ 927 pInfo->SetRepresentation( 928 ((const SvxDateField*) pField)->GetFormatted( 929 *GetNumberFormatter( sal_True ), LANGUAGE_SYSTEM) ); 930 } 931 else if (pField && pField->ISA(SvxURLField)) 932 { 933 /****************************************************************** 934 * URL-Field 935 ******************************************************************/ 936 937 switch ( ((const SvxURLField*) pField)->GetFormat() ) 938 { 939 case SVXURLFORMAT_APPDEFAULT: //!!! einstellbar an App??? 940 case SVXURLFORMAT_REPR: 941 { 942 pInfo->SetRepresentation( 943 ((const SvxURLField*)pField)->GetRepresentation()); 944 } 945 break; 946 947 case SVXURLFORMAT_URL: 948 { 949 pInfo->SetRepresentation( 950 ((const SvxURLField*)pField)->GetURL()); 951 } 952 break; 953 } 954 955 sal_uInt16 nChrFmt; 956 957 if (IsVisitedURL(((const SvxURLField*)pField)->GetURL())) 958 nChrFmt = RES_POOLCHR_INET_VISIT; 959 else 960 nChrFmt = RES_POOLCHR_INET_NORMAL; 961 962 SwFmt *pFmt = GetCharFmtFromPool(nChrFmt); 963 964 Color aColor(COL_LIGHTBLUE); 965 if (pFmt) 966 aColor = pFmt->GetColor().GetValue(); 967 968 pInfo->SetTxtColor(aColor); 969 } 970 else if (pField && pField->ISA(SdrMeasureField)) 971 { 972 /****************************************************************** 973 * Measure-Field 974 ******************************************************************/ 975 pInfo->ClearFldColor(); 976 } 977 else if ( pField && pField->ISA(SvxExtTimeField)) 978 { 979 /****************************************************************** 980 * Time-Field 981 ******************************************************************/ 982 pInfo->SetRepresentation( 983 ((const SvxExtTimeField*) pField)->GetFormatted( 984 *GetNumberFormatter( sal_True ), LANGUAGE_SYSTEM) ); 985 } 986 else 987 { 988 DBG_ERROR("unbekannter Feldbefehl"); 989 pInfo->SetRepresentation( String( '?' ) ); 990 } 991 } 992 993 return(0); 994 } 995 996 /* TFFDI: The functions formerly declared 'inline' 997 */ 998 const SdrModel* SwDoc::GetDrawModel() const { return pDrawModel; } 999 SdrModel* SwDoc::GetDrawModel() { return pDrawModel; } 1000 SdrLayerID SwDoc::GetHeavenId() const { return nHeaven; } 1001 SdrLayerID SwDoc::GetHellId() const { return nHell; } 1002 SdrLayerID SwDoc::GetControlsId() const { return nControls; } 1003 SdrLayerID SwDoc::GetInvisibleHeavenId() const { return nInvisibleHeaven; } 1004 SdrLayerID SwDoc::GetInvisibleHellId() const { return nInvisibleHell; } 1005 SdrLayerID SwDoc::GetInvisibleControlsId() const { return nInvisibleControls; } 1006 SdrModel* SwDoc::GetOrCreateDrawModel() { return GetDrawModel() ? GetDrawModel() : _MakeDrawModel(); } 1007 1008 // --> OD 2006-03-14 #i62875# 1009 namespace docfunc 1010 { 1011 bool ExistsDrawObjs( SwDoc& p_rDoc ) 1012 { 1013 bool bExistsDrawObjs( false ); 1014 1015 if ( p_rDoc.GetDrawModel() && 1016 p_rDoc.GetDrawModel()->GetPage( 0 ) ) 1017 { 1018 const SdrPage& rSdrPage( *(p_rDoc.GetDrawModel()->GetPage( 0 )) ); 1019 1020 SdrObjListIter aIter( rSdrPage, IM_FLAT ); 1021 while( aIter.IsMore() ) 1022 { 1023 SdrObject* pObj( aIter.Next() ); 1024 if ( !dynamic_cast<SwVirtFlyDrawObj*>(pObj) && 1025 !dynamic_cast<SwFlyDrawObj*>(pObj) ) 1026 { 1027 bExistsDrawObjs = true; 1028 break; 1029 } 1030 } 1031 } 1032 1033 return bExistsDrawObjs; 1034 } 1035 1036 bool AllDrawObjsOnPage( SwDoc& p_rDoc ) 1037 { 1038 bool bAllDrawObjsOnPage( true ); 1039 1040 if ( p_rDoc.GetDrawModel() && 1041 p_rDoc.GetDrawModel()->GetPage( 0 ) ) 1042 { 1043 const SdrPage& rSdrPage( *(p_rDoc.GetDrawModel()->GetPage( 0 )) ); 1044 1045 SdrObjListIter aIter( rSdrPage, IM_FLAT ); 1046 while( aIter.IsMore() ) 1047 { 1048 SdrObject* pObj( aIter.Next() ); 1049 if ( !dynamic_cast<SwVirtFlyDrawObj*>(pObj) && 1050 !dynamic_cast<SwFlyDrawObj*>(pObj) ) 1051 { 1052 SwDrawContact* pDrawContact = 1053 dynamic_cast<SwDrawContact*>(::GetUserCall( pObj )); 1054 if ( pDrawContact ) 1055 { 1056 SwAnchoredDrawObject* pAnchoredDrawObj = 1057 dynamic_cast<SwAnchoredDrawObject*>(pDrawContact->GetAnchoredObj( pObj )); 1058 1059 // error handling 1060 { 1061 if ( !pAnchoredDrawObj ) 1062 { 1063 ASSERT( false, 1064 "<docfunc::AllDrawObjsOnPage() - missing anchored draw object" ); 1065 bAllDrawObjsOnPage = false; 1066 break; 1067 } 1068 } 1069 1070 if ( pAnchoredDrawObj->NotYetPositioned() ) 1071 { 1072 // The drawing object isn't yet layouted. 1073 // Thus, it isn't known, if all drawing objects are on page. 1074 bAllDrawObjsOnPage = false; 1075 break; 1076 } 1077 else if ( pAnchoredDrawObj->IsOutsidePage() ) 1078 { 1079 bAllDrawObjsOnPage = false; 1080 break; 1081 } 1082 } 1083 else 1084 { 1085 // contact object of drawing object doesn't exists. 1086 // Thus, the drawing object isn't yet positioned. 1087 // Thus, it isn't known, if all drawing objects are on page. 1088 bAllDrawObjsOnPage = false; 1089 break; 1090 } 1091 } 1092 } 1093 } 1094 1095 return bAllDrawObjsOnPage; 1096 } 1097 } 1098 // <-- 1099 1100 // eof 1101