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_svx.hxx" 26 27 #include "svddrgm1.hxx" 28 #include <math.h> 29 30 #ifndef _MATH_H 31 #define _MATH_H 32 #endif 33 #include <tools/bigint.hxx> 34 #include <vcl/svapp.hxx> 35 #include "svx/xattr.hxx" 36 #include <svx/xpoly.hxx> 37 #include <svx/svdetc.hxx> 38 #include <svx/svdtrans.hxx> 39 #include <svx/svdundo.hxx> 40 #include <svx/svdmark.hxx> 41 #include <svx/svdocapt.hxx> 42 #include <svx/svdpagv.hxx> 43 #include "svx/svdstr.hrc" // Namen aus der Resource 44 #include "svx/svdglob.hxx" // StringCache 45 #include <svx/svddrgv.hxx> 46 #include <svx/svdundo.hxx> 47 #include <svx/svdograf.hxx> 48 #include <svx/dialogs.hrc> 49 #include <svx/dialmgr.hxx> 50 #include <svx/sdgcpitm.hxx> 51 #include <basegfx/polygon/b2dpolygon.hxx> 52 #include <basegfx/polygon/b2dpolygontools.hxx> 53 #include <svx/sdr/overlay/overlaypolypolygon.hxx> 54 #include <svx/sdr/overlay/overlaymanager.hxx> 55 #include <svx/sdr/overlay/overlayrollingrectangle.hxx> 56 #include <svx/sdrpagewindow.hxx> 57 #include <svx/sdrpaintwindow.hxx> 58 #include <basegfx/matrix/b2dhommatrix.hxx> 59 #include <basegfx/polygon/b2dpolypolygontools.hxx> 60 #include <svx/sdr/contact/viewobjectcontact.hxx> 61 #include <svx/sdr/contact/viewcontact.hxx> 62 #include <svx/sdr/contact/displayinfo.hxx> 63 #include <svx/sdr/overlay/overlayprimitive2dsequenceobject.hxx> 64 #include <drawinglayer/primitive2d/unifiedtransparenceprimitive2d.hxx> 65 #include <svx/sdr/contact/objectcontact.hxx> 66 #include "svx/svditer.hxx" 67 #include <svx/svdopath.hxx> 68 #include <svx/polypolygoneditor.hxx> 69 #include <drawinglayer/primitive2d/polypolygonprimitive2d.hxx> 70 #include <drawinglayer/primitive2d/polygonprimitive2d.hxx> 71 #include <drawinglayer/primitive2d/transformprimitive2d.hxx> 72 #include <drawinglayer/primitive2d/markerarrayprimitive2d.hxx> 73 #include <svx/sdr/primitive2d/sdrattributecreator.hxx> 74 #include <svx/sdr/primitive2d/sdrdecompositiontools.hxx> 75 #include <svx/svdoole2.hxx> 76 #include <svx/svdovirt.hxx> 77 #include <svx/svdouno.hxx> 78 #include <svx/sdr/primitive2d/sdrprimitivetools.hxx> 79 #include <basegfx/matrix/b2dhommatrixtools.hxx> 80 #include <drawinglayer/attribute/sdrlineattribute.hxx> 81 #include <drawinglayer/attribute/sdrlinestartendattribute.hxx> 82 #include <map> 83 #include <vector> 84 85 //////////////////////////////////////////////////////////////////////////////////////////////////// 86 87 SdrDragEntry::SdrDragEntry() 88 : mbAddToTransparent(false) 89 { 90 } 91 92 SdrDragEntry::~SdrDragEntry() 93 { 94 } 95 96 //////////////////////////////////////////////////////////////////////////////////////////////////// 97 98 SdrDragEntryPolyPolygon::SdrDragEntryPolyPolygon(const basegfx::B2DPolyPolygon& rOriginalPolyPolygon) 99 : SdrDragEntry(), 100 maOriginalPolyPolygon(rOriginalPolyPolygon) 101 { 102 } 103 104 SdrDragEntryPolyPolygon::~SdrDragEntryPolyPolygon() 105 { 106 } 107 108 drawinglayer::primitive2d::Primitive2DSequence SdrDragEntryPolyPolygon::createPrimitive2DSequenceInCurrentState(SdrDragMethod& rDragMethod) 109 { 110 drawinglayer::primitive2d::Primitive2DSequence aRetval; 111 112 if(maOriginalPolyPolygon.count()) 113 { 114 basegfx::B2DPolyPolygon aCopy(maOriginalPolyPolygon); 115 const SvtOptionsDrawinglayer aSvtOptionsDrawinglayer; 116 117 rDragMethod.applyCurrentTransformationToPolyPolygon(aCopy); 118 basegfx::BColor aColA(aSvtOptionsDrawinglayer.GetStripeColorA().getBColor()); 119 basegfx::BColor aColB(aSvtOptionsDrawinglayer.GetStripeColorB().getBColor()); 120 const double fStripeLength(aSvtOptionsDrawinglayer.GetStripeLength()); 121 122 if(Application::GetSettings().GetStyleSettings().GetHighContrastMode()) 123 { 124 aColA = aColB = Application::GetSettings().GetStyleSettings().GetHighlightColor().getBColor(); 125 aColB.invert(); 126 } 127 128 aRetval.realloc(2); 129 aRetval[0] = new drawinglayer::primitive2d::PolyPolygonMarkerPrimitive2D( 130 aCopy, 131 aColA, 132 aColB, 133 fStripeLength); 134 135 const basegfx::BColor aHilightColor(aSvtOptionsDrawinglayer.getHilightColor().getBColor()); 136 const double fTransparence(aSvtOptionsDrawinglayer.GetTransparentSelectionPercent() * 0.01); 137 138 aRetval[1] = new drawinglayer::primitive2d::PolyPolygonSelectionPrimitive2D( 139 aCopy, 140 aHilightColor, 141 fTransparence, 142 3.0, 143 false); 144 } 145 146 return aRetval; 147 } 148 149 //////////////////////////////////////////////////////////////////////////////////////////////////// 150 151 SdrDragEntrySdrObject::SdrDragEntrySdrObject(const SdrObject& rOriginal, sdr::contact::ObjectContact& rObjectContact, bool bModify) 152 : SdrDragEntry(), 153 maOriginal(rOriginal), 154 mpClone(0), 155 mrObjectContact(rObjectContact), 156 mbModify(bModify) 157 { 158 // add SdrObject parts to transparent overlay stuff 159 setAddToTransparent(true); 160 } 161 162 SdrDragEntrySdrObject::~SdrDragEntrySdrObject() 163 { 164 if(mpClone) 165 { 166 SdrObject::Free(mpClone); 167 } 168 } 169 170 void SdrDragEntrySdrObject::prepareCurrentState(SdrDragMethod& rDragMethod) 171 { 172 // for the moment, i need to re-create the clone in all cases. I need to figure 173 // out when clone and original have the same class, so that i can use operator= 174 // in those cases 175 176 // // copy all other needed stuff 177 // basegfx::B2DHomMatrix aMatrix; 178 // basegfx::B2DPolyPolygon aPolyPolygon; 179 // pOleObject->TRGetBaseGeometry(aMatrix, aPolyPolygon); 180 // pClone->TRSetBaseGeometry(aMatrix, aPolyPolygon); 181 182 if(mpClone) 183 { 184 SdrObject::Free(mpClone); 185 mpClone = 0; 186 } 187 188 if(mbModify) 189 { 190 if(!mpClone) 191 { 192 mpClone = maOriginal.getFullDragClone(); 193 } 194 195 // apply original transformation, implemented at the DragMethods 196 rDragMethod.applyCurrentTransformationToSdrObject(*mpClone); 197 } 198 } 199 200 drawinglayer::primitive2d::Primitive2DSequence SdrDragEntrySdrObject::createPrimitive2DSequenceInCurrentState(SdrDragMethod& /* rDragMethod */) 201 { 202 const SdrObject* pSource = &maOriginal; 203 204 if(mbModify && mpClone) 205 { 206 // choose source for geometry data 207 pSource = mpClone; 208 } 209 210 // get VOC and Primitive2DSequence 211 sdr::contact::ViewContact& rVC = pSource->GetViewContact(); 212 sdr::contact::ViewObjectContact& rVOC = rVC.GetViewObjectContact(mrObjectContact); 213 sdr::contact::DisplayInfo aDisplayInfo; 214 215 // Do not use the last ViewPort set at the OC from the last ProcessDisplay(), 216 // here we want the complete primitive sequence without visibility clippings 217 mrObjectContact.resetViewPort(); 218 219 return rVOC.getPrimitive2DSequenceHierarchy(aDisplayInfo); 220 } 221 222 //////////////////////////////////////////////////////////////////////////////////////////////////// 223 224 SdrDragEntryPrimitive2DSequence::SdrDragEntryPrimitive2DSequence( 225 const drawinglayer::primitive2d::Primitive2DSequence& rSequence, 226 bool bAddToTransparent) 227 : SdrDragEntry(), 228 maPrimitive2DSequence(rSequence) 229 { 230 // add parts to transparent overlay stuff eventually 231 setAddToTransparent(bAddToTransparent); 232 } 233 234 SdrDragEntryPrimitive2DSequence::~SdrDragEntryPrimitive2DSequence() 235 { 236 } 237 238 drawinglayer::primitive2d::Primitive2DSequence SdrDragEntryPrimitive2DSequence::createPrimitive2DSequenceInCurrentState(SdrDragMethod& rDragMethod) 239 { 240 drawinglayer::primitive2d::Primitive2DReference aTransformPrimitive2D( 241 new drawinglayer::primitive2d::TransformPrimitive2D( 242 rDragMethod.getCurrentTransformation(), 243 maPrimitive2DSequence)); 244 245 return drawinglayer::primitive2d::Primitive2DSequence(&aTransformPrimitive2D, 1); 246 } 247 248 //////////////////////////////////////////////////////////////////////////////////////////////////// 249 250 SdrDragEntryPointGlueDrag::SdrDragEntryPointGlueDrag(const std::vector< basegfx::B2DPoint >& rPositions, bool bIsPointDrag) 251 : maPositions(rPositions), 252 mbIsPointDrag(bIsPointDrag) 253 { 254 // add SdrObject parts to transparent overlay stuff 255 setAddToTransparent(true); 256 } 257 258 SdrDragEntryPointGlueDrag::~SdrDragEntryPointGlueDrag() 259 { 260 } 261 262 drawinglayer::primitive2d::Primitive2DSequence SdrDragEntryPointGlueDrag::createPrimitive2DSequenceInCurrentState(SdrDragMethod& rDragMethod) 263 { 264 drawinglayer::primitive2d::Primitive2DSequence aRetval; 265 266 if(!maPositions.empty()) 267 { 268 basegfx::B2DPolygon aPolygon; 269 sal_uInt32 a(0); 270 271 for(a = 0; a < maPositions.size(); a++) 272 { 273 aPolygon.append(maPositions[a]); 274 } 275 276 basegfx::B2DPolyPolygon aPolyPolygon(aPolygon); 277 278 rDragMethod.applyCurrentTransformationToPolyPolygon(aPolyPolygon); 279 280 const basegfx::B2DPolygon aTransformed(aPolyPolygon.getB2DPolygon(0)); 281 std::vector< basegfx::B2DPoint > aTransformedPositions; 282 283 aTransformedPositions.reserve(aTransformed.count()); 284 285 for(a = 0; a < aTransformed.count(); a++) 286 { 287 aTransformedPositions.push_back(aTransformed.getB2DPoint(a)); 288 } 289 290 if(mbIsPointDrag) 291 { 292 const SvtOptionsDrawinglayer aSvtOptionsDrawinglayer; 293 basegfx::BColor aColor(aSvtOptionsDrawinglayer.GetStripeColorA().getBColor()); 294 295 if(Application::GetSettings().GetStyleSettings().GetHighContrastMode()) 296 { 297 aColor = Application::GetSettings().GetStyleSettings().GetHighlightColor().getBColor(); 298 } 299 300 drawinglayer::primitive2d::Primitive2DReference aMarkerArrayPrimitive2D( 301 new drawinglayer::primitive2d::MarkerArrayPrimitive2D(aTransformedPositions, 302 drawinglayer::primitive2d::createDefaultCross_3x3(aColor))); 303 304 aRetval = drawinglayer::primitive2d::Primitive2DSequence(&aMarkerArrayPrimitive2D, 1); 305 } 306 else 307 { 308 const basegfx::BColor aBackPen(1.0, 1.0, 1.0); 309 const basegfx::BColor aRGBFrontColor(0.0, 0.0, 1.0); // COL_LIGHTBLUE 310 drawinglayer::primitive2d::Primitive2DReference aMarkerArrayPrimitive2D( 311 new drawinglayer::primitive2d::MarkerArrayPrimitive2D(aTransformedPositions, 312 drawinglayer::primitive2d::createDefaultGluepoint_7x7(aBackPen, aRGBFrontColor))); 313 314 aRetval = drawinglayer::primitive2d::Primitive2DSequence(&aMarkerArrayPrimitive2D, 1); 315 } 316 } 317 318 return aRetval; 319 } 320 321 //////////////////////////////////////////////////////////////////////////////////////////////////// 322 323 TYPEINIT0(SdrDragMethod); 324 325 void SdrDragMethod::resetSdrDragEntries() 326 { 327 // clear entries; creation is on demand 328 clearSdrDragEntries(); 329 } 330 331 basegfx::B2DRange SdrDragMethod::getCurrentRange() const 332 { 333 return getB2DRangeFromOverlayObjectList(); 334 } 335 336 void SdrDragMethod::clearSdrDragEntries() 337 { 338 for(sal_uInt32 a(0); a < maSdrDragEntries.size(); a++) 339 { 340 delete maSdrDragEntries[a]; 341 } 342 343 maSdrDragEntries.clear(); 344 } 345 346 void SdrDragMethod::addSdrDragEntry(SdrDragEntry* pNew) 347 { 348 if(pNew) 349 { 350 maSdrDragEntries.push_back(pNew); 351 } 352 } 353 354 void SdrDragMethod::createSdrDragEntries() 355 { 356 if(getSdrDragView().GetSdrPageView() && getSdrDragView().GetSdrPageView()->HasMarkedObjPageView()) 357 { 358 if(getSdrDragView().IsDraggingPoints()) 359 { 360 createSdrDragEntries_PointDrag(); 361 } 362 else if(getSdrDragView().IsDraggingGluePoints()) 363 { 364 createSdrDragEntries_GlueDrag(); 365 } 366 else 367 { 368 if(getSolidDraggingActive()) 369 { 370 createSdrDragEntries_SolidDrag(); 371 } 372 else 373 { 374 createSdrDragEntries_PolygonDrag(); 375 } 376 } 377 } 378 } 379 380 void SdrDragMethod::createSdrDragEntryForSdrObject(const SdrObject& rOriginal, sdr::contact::ObjectContact& rObjectContact, bool bModify) 381 { 382 // add full obejct drag; Clone() at the object has to work 383 // for this 384 addSdrDragEntry(new SdrDragEntrySdrObject(rOriginal, rObjectContact, bModify)); 385 } 386 387 void SdrDragMethod::createSdrDragEntries_SolidDrag() 388 { 389 const sal_uInt32 nMarkAnz(getSdrDragView().GetMarkedObjectCount()); 390 SdrPageView* pPV = getSdrDragView().GetSdrPageView(); 391 392 if(pPV) 393 { 394 for(sal_uInt32 a(0); a < nMarkAnz; a++) 395 { 396 SdrMark* pM = getSdrDragView().GetSdrMarkByIndex(a); 397 398 if(pM->GetPageView() == pPV) 399 { 400 const SdrObject* pObject = pM->GetMarkedSdrObj(); 401 402 if(pObject) 403 { 404 if(pPV->PageWindowCount()) 405 { 406 sdr::contact::ObjectContact& rOC = pPV->GetPageWindow(0)->GetObjectContact(); 407 SdrObjListIter aIter(*pObject); 408 409 while(aIter.IsMore()) 410 { 411 SdrObject* pCandidate = aIter.Next(); 412 413 if(pCandidate) 414 { 415 const bool bSuppressFullDrag(!pCandidate->supportsFullDrag()); 416 bool bAddWireframe(bSuppressFullDrag); 417 418 if(!bAddWireframe && !pCandidate->HasLineStyle()) 419 { 420 // add wireframe for objects without outline 421 bAddWireframe = true; 422 } 423 424 if(!bSuppressFullDrag) 425 { 426 // add full obejct drag; Clone() at the object has to work 427 // for this 428 createSdrDragEntryForSdrObject(*pCandidate, rOC, true); 429 } 430 431 if(bAddWireframe) 432 { 433 // when dragging a 50% transparent copy of a filled or not filled object without 434 // outline, this is normally hard to see. Add extra wireframe in that case. This 435 // works nice e.g. with thext frames etc. 436 addSdrDragEntry(new SdrDragEntryPolyPolygon(pCandidate->TakeXorPoly())); 437 } 438 } 439 } 440 } 441 } 442 } 443 } 444 } 445 } 446 447 void SdrDragMethod::createSdrDragEntries_PolygonDrag() 448 { 449 const sal_uInt32 nMarkAnz(getSdrDragView().GetMarkedObjectCount()); 450 bool bNoPolygons(getSdrDragView().IsNoDragXorPolys() || nMarkAnz > getSdrDragView().GetDragXorPolyLimit()); 451 basegfx::B2DPolyPolygon aResult; 452 sal_uInt32 nPointCount(0); 453 454 for(sal_uInt32 a(0); !bNoPolygons && a < nMarkAnz; a++) 455 { 456 SdrMark* pM = getSdrDragView().GetSdrMarkByIndex(a); 457 458 if(pM->GetPageView() == getSdrDragView().GetSdrPageView()) 459 { 460 const basegfx::B2DPolyPolygon aNewPolyPolygon(pM->GetMarkedSdrObj()->TakeXorPoly()); 461 462 for(sal_uInt32 b(0); b < aNewPolyPolygon.count(); b++) 463 { 464 nPointCount += aNewPolyPolygon.getB2DPolygon(b).count(); 465 } 466 467 if(nPointCount > getSdrDragView().GetDragXorPointLimit()) 468 { 469 bNoPolygons = true; 470 } 471 472 if(!bNoPolygons) 473 { 474 aResult.append(aNewPolyPolygon); 475 } 476 } 477 } 478 479 if(bNoPolygons) 480 { 481 const Rectangle aR(getSdrDragView().GetSdrPageView()->MarkSnap()); 482 const basegfx::B2DRange aNewRectangle(aR.Left(), aR.Top(), aR.Right(), aR.Bottom()); 483 basegfx::B2DPolygon aNewPolygon(basegfx::tools::createPolygonFromRect(aNewRectangle)); 484 485 aResult = basegfx::B2DPolyPolygon(basegfx::tools::expandToCurve(aNewPolygon)); 486 } 487 488 if(aResult.count()) 489 { 490 addSdrDragEntry(new SdrDragEntryPolyPolygon(aResult)); 491 } 492 } 493 494 void SdrDragMethod::createSdrDragEntries_PointDrag() 495 { 496 const sal_uInt32 nMarkAnz(getSdrDragView().GetMarkedObjectCount()); 497 std::vector< basegfx::B2DPoint > aPositions; 498 499 for(sal_uInt32 nm(0); nm < nMarkAnz; nm++) 500 { 501 SdrMark* pM = getSdrDragView().GetSdrMarkByIndex(nm); 502 503 if(pM->GetPageView() == getSdrDragView().GetSdrPageView()) 504 { 505 const SdrUShortCont* pPts = pM->GetMarkedPoints(); 506 507 if(pPts && pPts->GetCount()) 508 { 509 const SdrObject* pObj = pM->GetMarkedSdrObj(); 510 const SdrPathObj* pPath = dynamic_cast< const SdrPathObj* >(pObj); 511 512 if(pPath) 513 { 514 const basegfx::B2DPolyPolygon aPathXPP = pPath->GetPathPoly(); 515 516 if(aPathXPP.count()) 517 { 518 const sal_uInt32 nPtAnz(pPts->GetCount()); 519 520 for(sal_uInt32 nPtNum(0); nPtNum < nPtAnz; nPtNum++) 521 { 522 sal_uInt32 nPolyNum, nPointNum; 523 const sal_uInt16 nObjPt(pPts->GetObject(nPtNum)); 524 525 if(sdr::PolyPolygonEditor::GetRelativePolyPoint(aPathXPP, nObjPt, nPolyNum, nPointNum)) 526 { 527 aPositions.push_back(aPathXPP.getB2DPolygon(nPolyNum).getB2DPoint(nPointNum)); 528 } 529 } 530 } 531 } 532 } 533 } 534 } 535 536 if(!aPositions.empty()) 537 { 538 addSdrDragEntry(new SdrDragEntryPointGlueDrag(aPositions, true)); 539 } 540 } 541 542 void SdrDragMethod::createSdrDragEntries_GlueDrag() 543 { 544 const sal_uInt32 nMarkAnz(getSdrDragView().GetMarkedObjectCount()); 545 std::vector< basegfx::B2DPoint > aPositions; 546 547 for(sal_uInt32 nm(0); nm < nMarkAnz; nm++) 548 { 549 SdrMark* pM = getSdrDragView().GetSdrMarkByIndex(nm); 550 551 if(pM->GetPageView() == getSdrDragView().GetSdrPageView()) 552 { 553 const SdrUShortCont* pPts = pM->GetMarkedGluePoints(); 554 555 if(pPts && pPts->GetCount()) 556 { 557 const SdrObject* pObj = pM->GetMarkedSdrObj(); 558 const SdrGluePointList* pGPL = pObj->GetGluePointList(); 559 560 if(pGPL) 561 { 562 const sal_uInt32 nPtAnz(pPts->GetCount()); 563 564 for(sal_uInt32 nPtNum(0); nPtNum < nPtAnz; nPtNum++) 565 { 566 const sal_uInt16 nObjPt(pPts->GetObject(nPtNum)); 567 const sal_uInt16 nGlueNum(pGPL->FindGluePoint(nObjPt)); 568 569 if(SDRGLUEPOINT_NOTFOUND != nGlueNum) 570 { 571 const Point aPoint((*pGPL)[nGlueNum].GetAbsolutePos(*pObj)); 572 aPositions.push_back(basegfx::B2DPoint(aPoint.X(), aPoint.Y())); 573 } 574 } 575 } 576 } 577 } 578 } 579 580 if(!aPositions.empty()) 581 { 582 addSdrDragEntry(new SdrDragEntryPointGlueDrag(aPositions, false)); 583 } 584 } 585 586 void SdrDragMethod::ImpTakeDescriptionStr(sal_uInt16 nStrCacheID, XubString& rStr, sal_uInt16 nVal) const 587 { 588 sal_uInt16 nOpt=0; 589 if (IsDraggingPoints()) { 590 nOpt=IMPSDR_POINTSDESCRIPTION; 591 } else if (IsDraggingGluePoints()) { 592 nOpt=IMPSDR_GLUEPOINTSDESCRIPTION; 593 } 594 getSdrDragView().ImpTakeDescriptionStr(nStrCacheID,rStr,nVal,nOpt); 595 } 596 597 SdrObject* SdrDragMethod::GetDragObj() const 598 { 599 SdrObject* pObj=NULL; 600 if (getSdrDragView().pDragHdl!=NULL) pObj=getSdrDragView().pDragHdl->GetObj(); 601 if (pObj==NULL) pObj=getSdrDragView().pMarkedObj; 602 return pObj; 603 } 604 605 SdrPageView* SdrDragMethod::GetDragPV() const 606 { 607 SdrPageView* pPV=NULL; 608 if (getSdrDragView().pDragHdl!=NULL) pPV=getSdrDragView().pDragHdl->GetPageView(); 609 if (pPV==NULL) pPV=getSdrDragView().pMarkedPV; 610 return pPV; 611 } 612 613 void SdrDragMethod::applyCurrentTransformationToSdrObject(SdrObject& rTarget) 614 { 615 // the original applies the transformation using TRGetBaseGeometry/TRSetBaseGeometry. 616 // Later this should be the only needed one for linear transforms (not for SdrDragCrook and 617 // SdrDragDistort, those are NOT linear). Currently, this can not yet be used since the 618 // special handling of rotate/mirror due to the not-being-able to handle it in the old 619 // drawinglayer stuff. Text would currently not correctly be mirrored in the preview. 620 basegfx::B2DHomMatrix aObjectTransform; 621 basegfx::B2DPolyPolygon aObjectPolyPolygon; 622 bool bPolyUsed(rTarget.TRGetBaseGeometry(aObjectTransform, aObjectPolyPolygon)); 623 624 // apply transform to object transform 625 aObjectTransform *= getCurrentTransformation(); 626 627 if(bPolyUsed) 628 { 629 // do something special since the object size is in the polygon 630 // break up matrix to get the scale 631 basegfx::B2DTuple aScale; 632 basegfx::B2DTuple aTranslate; 633 double fRotate, fShearX; 634 aObjectTransform.decompose(aScale, aTranslate, fRotate, fShearX); 635 636 // get polygon's pos and size 637 const basegfx::B2DRange aPolyRange(aObjectPolyPolygon.getB2DRange()); 638 639 // get the scaling factors (do not mirror, this is in the object transformation) 640 const double fScaleX(fabs(aScale.getX()) / (basegfx::fTools::equalZero(aPolyRange.getWidth()) ? 1.0 : aPolyRange.getWidth())); 641 const double fScaleY(fabs(aScale.getY()) / (basegfx::fTools::equalZero(aPolyRange.getHeight()) ? 1.0 : aPolyRange.getHeight())); 642 643 // prepare transform matrix for polygon 644 basegfx::B2DHomMatrix aPolyTransform(basegfx::tools::createTranslateB2DHomMatrix( 645 -aPolyRange.getMinX(), -aPolyRange.getMinY())); 646 aPolyTransform.scale(fScaleX, fScaleY); 647 648 // normally the poly should be moved back, but the translation is in the object 649 // transformation and thus does not need to be done 650 // aPolyTransform.translate(-aPolyRange.getMinX(), -aPolyRange.getMinY()); 651 652 // transform the polygon 653 aObjectPolyPolygon.transform(aPolyTransform); 654 } 655 656 rTarget.TRSetBaseGeometry(getCurrentTransformation() * aObjectTransform, aObjectPolyPolygon); 657 } 658 659 void SdrDragMethod::applyCurrentTransformationToPolyPolygon(basegfx::B2DPolyPolygon& rTarget) 660 { 661 // original uses CurrentTransformation 662 rTarget.transform(getCurrentTransformation()); 663 } 664 665 SdrDragMethod::SdrDragMethod(SdrDragView& rNewView) 666 : maSdrDragEntries(), 667 maOverlayObjectList(), 668 mrSdrDragView(rNewView), 669 mbMoveOnly(false), 670 mbSolidDraggingActive(getSdrDragView().IsSolidDragging()) 671 { 672 if(mbSolidDraggingActive && Application::GetSettings().GetStyleSettings().GetHighContrastMode()) 673 { 674 // fallback to wireframe when high contrast is used 675 mbSolidDraggingActive = false; 676 } 677 } 678 679 SdrDragMethod::~SdrDragMethod() 680 { 681 clearSdrDragEntries(); 682 } 683 684 void SdrDragMethod::Show() 685 { 686 getSdrDragView().ShowDragObj(); 687 } 688 689 void SdrDragMethod::Hide() 690 { 691 getSdrDragView().HideDragObj(); 692 } 693 694 basegfx::B2DHomMatrix SdrDragMethod::getCurrentTransformation() 695 { 696 return basegfx::B2DHomMatrix(); 697 } 698 699 void SdrDragMethod::CancelSdrDrag() 700 { 701 Hide(); 702 } 703 704 struct compareConstSdrObjectRefs 705 { 706 bool operator()(const SdrObject* p1, const SdrObject* p2) const 707 { 708 return (p1 < p2); 709 } 710 }; 711 712 typedef std::map< const SdrObject*, SdrObject*, compareConstSdrObjectRefs> SdrObjectAndCloneMap; 713 714 void SdrDragMethod::CreateOverlayGeometry(sdr::overlay::OverlayManager& rOverlayManager) 715 { 716 // create SdrDragEntries on demand 717 if(maSdrDragEntries.empty()) 718 { 719 createSdrDragEntries(); 720 } 721 722 // if there are entries, derive OverlayObjects from the entries, including 723 // modification from current interactive state 724 if(!maSdrDragEntries.empty()) 725 { 726 // #54102# SdrDragEntrySdrObject creates clones of SdrObjects as base for creating the needed 727 // primitives, holding the original and the clone. If connectors (Edges) are involved, 728 // the cloned connectors need to be connected to the cloned SdrObjects (after cloning 729 // they are connected to the original SdrObjects). To do so, trigger the preparation 730 // steps for SdrDragEntrySdrObject, save an association of (orig, clone) in a helper 731 // and evtl. remember if it was an edge 732 SdrObjectAndCloneMap aOriginalAndClones; 733 std::vector< SdrEdgeObj* > aEdges; 734 sal_uInt32 a; 735 736 // #54102# execute prepareCurrentState for all SdrDragEntrySdrObject, register pair of original and 737 // clone, remember edges 738 for(a = 0; a < maSdrDragEntries.size(); a++) 739 { 740 SdrDragEntrySdrObject* pSdrDragEntrySdrObject = dynamic_cast< SdrDragEntrySdrObject*>(maSdrDragEntries[a]); 741 742 if(pSdrDragEntrySdrObject) 743 { 744 pSdrDragEntrySdrObject->prepareCurrentState(*this); 745 746 SdrEdgeObj* pSdrEdgeObj = dynamic_cast< SdrEdgeObj* >(pSdrDragEntrySdrObject->getClone()); 747 748 if(pSdrEdgeObj) 749 { 750 aEdges.push_back(pSdrEdgeObj); 751 } 752 753 if(pSdrDragEntrySdrObject->getClone()) 754 { 755 aOriginalAndClones[&pSdrDragEntrySdrObject->getOriginal()] = pSdrDragEntrySdrObject->getClone(); 756 } 757 } 758 } 759 760 // #54102# if there are edges, reconnect their ends to the corresponding clones (if found) 761 if(aEdges.size()) 762 { 763 for(a = 0; a < aEdges.size(); a++) 764 { 765 SdrEdgeObj* pSdrEdgeObj = aEdges[a]; 766 SdrObject* pConnectedTo = pSdrEdgeObj->GetConnectedNode(true); 767 768 if(pConnectedTo) 769 { 770 SdrObjectAndCloneMap::iterator aEntry = aOriginalAndClones.find(pConnectedTo); 771 772 if(aEntry != aOriginalAndClones.end()) 773 { 774 pSdrEdgeObj->ConnectToNode(true, aEntry->second); 775 } 776 } 777 778 pConnectedTo = pSdrEdgeObj->GetConnectedNode(false); 779 780 if(pConnectedTo) 781 { 782 SdrObjectAndCloneMap::iterator aEntry = aOriginalAndClones.find(pConnectedTo); 783 784 if(aEntry != aOriginalAndClones.end()) 785 { 786 pSdrEdgeObj->ConnectToNode(false, aEntry->second); 787 } 788 } 789 } 790 } 791 792 // collect primitives for visualisation 793 drawinglayer::primitive2d::Primitive2DSequence aResult; 794 drawinglayer::primitive2d::Primitive2DSequence aResultTransparent; 795 796 for(a = 0; a < maSdrDragEntries.size(); a++) 797 { 798 SdrDragEntry* pCandidate = maSdrDragEntries[a]; 799 800 if(pCandidate) 801 { 802 const drawinglayer::primitive2d::Primitive2DSequence aCandidateResult(pCandidate->createPrimitive2DSequenceInCurrentState(*this)); 803 804 if(aCandidateResult.hasElements()) 805 { 806 if(pCandidate->getAddToTransparent()) 807 { 808 drawinglayer::primitive2d::appendPrimitive2DSequenceToPrimitive2DSequence(aResultTransparent, aCandidateResult); 809 } 810 else 811 { 812 drawinglayer::primitive2d::appendPrimitive2DSequenceToPrimitive2DSequence(aResult, aCandidateResult); 813 } 814 } 815 } 816 } 817 818 if(DoAddConnectorOverlays()) 819 { 820 const drawinglayer::primitive2d::Primitive2DSequence aConnectorOverlays(AddConnectorOverlays()); 821 822 if(aConnectorOverlays.hasElements()) 823 { 824 // add connector overlays to transparent part 825 drawinglayer::primitive2d::appendPrimitive2DSequenceToPrimitive2DSequence(aResultTransparent, aConnectorOverlays); 826 } 827 } 828 829 if(aResult.hasElements()) 830 { 831 sdr::overlay::OverlayObject* pNewOverlayObject = new sdr::overlay::OverlayPrimitive2DSequenceObject(aResult); 832 rOverlayManager.add(*pNewOverlayObject); 833 addToOverlayObjectList(*pNewOverlayObject); 834 } 835 836 if(aResultTransparent.hasElements()) 837 { 838 drawinglayer::primitive2d::Primitive2DReference aUnifiedTransparencePrimitive2D(new drawinglayer::primitive2d::UnifiedTransparencePrimitive2D(aResultTransparent, 0.5)); 839 aResultTransparent = drawinglayer::primitive2d::Primitive2DSequence(&aUnifiedTransparencePrimitive2D, 1); 840 841 sdr::overlay::OverlayObject* pNewOverlayObject = new sdr::overlay::OverlayPrimitive2DSequenceObject(aResultTransparent); 842 rOverlayManager.add(*pNewOverlayObject); 843 addToOverlayObjectList(*pNewOverlayObject); 844 } 845 } 846 847 // evtl add DragStripes (help lines cross the page when dragging) 848 if(getSdrDragView().IsDragStripes()) 849 { 850 Rectangle aActionRectangle; 851 getSdrDragView().TakeActionRect(aActionRectangle); 852 853 const basegfx::B2DPoint aTopLeft(aActionRectangle.Left(), aActionRectangle.Top()); 854 const basegfx::B2DPoint aBottomRight(aActionRectangle.Right(), aActionRectangle.Bottom()); 855 sdr::overlay::OverlayRollingRectangleStriped* pNew = new sdr::overlay::OverlayRollingRectangleStriped( 856 aTopLeft, aBottomRight, true, false); 857 858 rOverlayManager.add(*pNew); 859 addToOverlayObjectList(*pNew); 860 } 861 } 862 863 void SdrDragMethod::destroyOverlayGeometry() 864 { 865 clearOverlayObjectList(); 866 } 867 868 bool SdrDragMethod::DoAddConnectorOverlays() 869 { 870 // these conditions are translated from SdrDragView::ImpDrawEdgeXor 871 const SdrMarkList& rMarkedNodes = getSdrDragView().GetEdgesOfMarkedNodes(); 872 873 if(!rMarkedNodes.GetMarkCount()) 874 { 875 return false; 876 } 877 878 if(!getSdrDragView().IsRubberEdgeDragging() && !getSdrDragView().IsDetailedEdgeDragging()) 879 { 880 return false; 881 } 882 883 if(getSdrDragView().IsDraggingPoints() || getSdrDragView().IsDraggingGluePoints()) 884 { 885 return false; 886 } 887 888 if(!getMoveOnly() && !( 889 IS_TYPE(SdrDragMove, this) || IS_TYPE(SdrDragResize, this) || 890 IS_TYPE(SdrDragRotate,this) || IS_TYPE(SdrDragMirror,this))) 891 { 892 return false; 893 } 894 895 const bool bDetail(getSdrDragView().IsDetailedEdgeDragging() && getMoveOnly()); 896 897 if(!bDetail && !getSdrDragView().IsRubberEdgeDragging()) 898 { 899 return false; 900 } 901 902 // one more migrated from SdrEdgeObj::NspToggleEdgeXor 903 if(IS_TYPE(SdrDragObjOwn, this) || IS_TYPE(SdrDragMovHdl, this)) 904 { 905 return false; 906 } 907 908 return true; 909 } 910 911 drawinglayer::primitive2d::Primitive2DSequence SdrDragMethod::AddConnectorOverlays() 912 { 913 drawinglayer::primitive2d::Primitive2DSequence aRetval; 914 const bool bDetail(getSdrDragView().IsDetailedEdgeDragging() && getMoveOnly()); 915 const SdrMarkList& rMarkedNodes = getSdrDragView().GetEdgesOfMarkedNodes(); 916 917 for(sal_uInt16 a(0); a < rMarkedNodes.GetMarkCount(); a++) 918 { 919 SdrMark* pEM = rMarkedNodes.GetMark(a); 920 921 if(pEM && pEM->GetMarkedSdrObj()) 922 { 923 SdrEdgeObj* pEdge = dynamic_cast< SdrEdgeObj* >(pEM->GetMarkedSdrObj()); 924 925 if(pEdge) 926 { 927 const basegfx::B2DPolygon aEdgePolygon(pEdge->ImplAddConnectorOverlay(*this, pEM->IsCon1(), pEM->IsCon2(), bDetail)); 928 929 if(aEdgePolygon.count()) 930 { 931 // this polygon is a temporary calculated connector path, so it is not possible to fetch 932 // the needed primitives directly from the pEdge object which does not get changed. If full 933 // drag is on, use the SdrObjects ItemSet to create a adequate representation 934 bool bUseSolidDragging(getSolidDraggingActive()); 935 936 if(bUseSolidDragging) 937 { 938 // switch off solid dragging if connector is not visible 939 if(!pEdge->HasLineStyle()) 940 { 941 bUseSolidDragging = false; 942 } 943 } 944 945 if(bUseSolidDragging) 946 { 947 const SfxItemSet& rItemSet = pEdge->GetMergedItemSet(); 948 const drawinglayer::attribute::SdrLineAttribute aLine( 949 drawinglayer::primitive2d::createNewSdrLineAttribute(rItemSet)); 950 951 if(!aLine.isDefault()) 952 { 953 const drawinglayer::attribute::SdrLineStartEndAttribute aLineStartEnd( 954 drawinglayer::primitive2d::createNewSdrLineStartEndAttribute( 955 rItemSet, 956 aLine.getWidth())); 957 958 drawinglayer::primitive2d::appendPrimitive2DReferenceToPrimitive2DSequence( 959 aRetval, drawinglayer::primitive2d::createPolygonLinePrimitive( 960 aEdgePolygon, 961 basegfx::B2DHomMatrix(), 962 aLine, 963 aLineStartEnd)); 964 } 965 } 966 else 967 { 968 const SvtOptionsDrawinglayer aSvtOptionsDrawinglayer; 969 basegfx::BColor aColA(aSvtOptionsDrawinglayer.GetStripeColorA().getBColor()); 970 basegfx::BColor aColB(aSvtOptionsDrawinglayer.GetStripeColorB().getBColor()); 971 const double fStripeLength(aSvtOptionsDrawinglayer.GetStripeLength()); 972 973 if(Application::GetSettings().GetStyleSettings().GetHighContrastMode()) 974 { 975 aColA = aColB = Application::GetSettings().GetStyleSettings().GetHighlightColor().getBColor(); 976 aColB.invert(); 977 } 978 979 drawinglayer::primitive2d::Primitive2DReference aPolyPolygonMarkerPrimitive2D( 980 new drawinglayer::primitive2d::PolygonMarkerPrimitive2D( 981 aEdgePolygon, aColA, aColB, fStripeLength)); 982 drawinglayer::primitive2d::appendPrimitive2DReferenceToPrimitive2DSequence(aRetval, aPolyPolygonMarkerPrimitive2D); 983 } 984 } 985 } 986 } 987 } 988 989 return aRetval; 990 } 991 992 //////////////////////////////////////////////////////////////////////////////////////////////////// 993 994 TYPEINIT1(SdrDragMovHdl,SdrDragMethod); 995 996 SdrDragMovHdl::SdrDragMovHdl(SdrDragView& rNewView) 997 : SdrDragMethod(rNewView), 998 bMirrObjShown(false) 999 { 1000 } 1001 1002 void SdrDragMovHdl::createSdrDragEntries() 1003 { 1004 // SdrDragMovHdl does not use the default drags, 1005 // but creates nothing 1006 } 1007 1008 void SdrDragMovHdl::TakeSdrDragComment(XubString& rStr) const 1009 { 1010 rStr=ImpGetResStr(STR_DragMethMovHdl); 1011 if (getSdrDragView().IsDragWithCopy()) rStr+=ImpGetResStr(STR_EditWithCopy); 1012 } 1013 1014 bool SdrDragMovHdl::BeginSdrDrag() 1015 { 1016 if( !GetDragHdl() ) 1017 return false; 1018 1019 DragStat().Ref1()=GetDragHdl()->GetPos(); 1020 DragStat().SetShown(!DragStat().IsShown()); 1021 SdrHdlKind eKind=GetDragHdl()->GetKind(); 1022 SdrHdl* pH1=GetHdlList().GetHdl(HDL_REF1); 1023 SdrHdl* pH2=GetHdlList().GetHdl(HDL_REF2); 1024 1025 if (eKind==HDL_MIRX) 1026 { 1027 if (pH1==NULL || pH2==NULL) 1028 { 1029 DBG_ERROR("SdrDragMovHdl::BeginSdrDrag(): Verschieben der Spiegelachse: Referenzhandles nicht gefunden"); 1030 return false; 1031 } 1032 1033 DragStat().SetActionRect(Rectangle(pH1->GetPos(),pH2->GetPos())); 1034 } 1035 else 1036 { 1037 Point aPt(GetDragHdl()->GetPos()); 1038 DragStat().SetActionRect(Rectangle(aPt,aPt)); 1039 } 1040 1041 return true; 1042 } 1043 1044 void SdrDragMovHdl::MoveSdrDrag(const Point& rNoSnapPnt) 1045 { 1046 Point aPnt(rNoSnapPnt); 1047 1048 if ( GetDragHdl() && DragStat().CheckMinMoved(rNoSnapPnt)) 1049 { 1050 if (GetDragHdl()->GetKind()==HDL_MIRX) 1051 { 1052 SdrHdl* pH1=GetHdlList().GetHdl(HDL_REF1); 1053 SdrHdl* pH2=GetHdlList().GetHdl(HDL_REF2); 1054 1055 if (pH1==NULL || pH2==NULL) 1056 return; 1057 1058 if (!DragStat().IsNoSnap()) 1059 { 1060 long nBestXSnap=0; 1061 long nBestYSnap=0; 1062 bool bXSnapped=false; 1063 bool bYSnapped=false; 1064 Point aDif(aPnt-DragStat().GetStart()); 1065 getSdrDragView().CheckSnap(Ref1()+aDif,NULL,nBestXSnap,nBestYSnap,bXSnapped,bYSnapped); 1066 getSdrDragView().CheckSnap(Ref2()+aDif,NULL,nBestXSnap,nBestYSnap,bXSnapped,bYSnapped); 1067 aPnt.X()+=nBestXSnap; 1068 aPnt.Y()+=nBestYSnap; 1069 } 1070 1071 if (aPnt!=DragStat().GetNow()) 1072 { 1073 Hide(); 1074 DragStat().NextMove(aPnt); 1075 Point aDif(DragStat().GetNow()-DragStat().GetStart()); 1076 pH1->SetPos(Ref1()+aDif); 1077 pH2->SetPos(Ref2()+aDif); 1078 1079 SdrHdl* pHM = GetHdlList().GetHdl(HDL_MIRX); 1080 1081 if(pHM) 1082 pHM->Touch(); 1083 1084 Show(); 1085 DragStat().SetActionRect(Rectangle(pH1->GetPos(),pH2->GetPos())); 1086 } 1087 } 1088 else 1089 { 1090 if (!DragStat().IsNoSnap()) SnapPos(aPnt); 1091 long nSA=0; 1092 1093 if (getSdrDragView().IsAngleSnapEnabled()) 1094 nSA=getSdrDragView().GetSnapAngle(); 1095 1096 if (getSdrDragView().IsMirrorAllowed(true,true)) 1097 { // eingeschraenkt 1098 if (!getSdrDragView().IsMirrorAllowed(false,false)) nSA=4500; 1099 if (!getSdrDragView().IsMirrorAllowed(true,false)) nSA=9000; 1100 } 1101 1102 if (getSdrDragView().IsOrtho() && nSA!=9000) 1103 nSA=4500; 1104 1105 if (nSA!=0) 1106 { // Winkelfang 1107 SdrHdlKind eRef=HDL_REF1; 1108 1109 if (GetDragHdl()->GetKind()==HDL_REF1) 1110 eRef=HDL_REF2; 1111 1112 SdrHdl* pH=GetHdlList().GetHdl(eRef); 1113 1114 if (pH!=NULL) 1115 { 1116 Point aRef(pH->GetPos()); 1117 long nWink=NormAngle360(GetAngle(aPnt-aRef)); 1118 long nNeuWink=nWink; 1119 nNeuWink+=nSA/2; 1120 nNeuWink/=nSA; 1121 nNeuWink*=nSA; 1122 nNeuWink=NormAngle360(nNeuWink); 1123 double a=(nNeuWink-nWink)*nPi180; 1124 double nSin=sin(a); 1125 double nCos=cos(a); 1126 RotatePoint(aPnt,aRef,nSin,nCos); 1127 1128 // Bei bestimmten Werten Rundungsfehler ausschliessen: 1129 if (nSA==9000) 1130 { 1131 if (nNeuWink==0 || nNeuWink==18000) aPnt.Y()=aRef.Y(); 1132 if (nNeuWink==9000 || nNeuWink==27000) aPnt.X()=aRef.X(); 1133 } 1134 1135 if (nSA==4500) 1136 OrthoDistance8(aRef,aPnt,true); 1137 } 1138 } 1139 1140 if (aPnt!=DragStat().GetNow()) 1141 { 1142 Hide(); 1143 DragStat().NextMove(aPnt); 1144 GetDragHdl()->SetPos(DragStat().GetNow()); 1145 SdrHdl* pHM = GetHdlList().GetHdl(HDL_MIRX); 1146 1147 if(pHM) 1148 pHM->Touch(); 1149 1150 Show(); 1151 DragStat().SetActionRect(Rectangle(aPnt,aPnt)); 1152 } 1153 } 1154 } 1155 } 1156 1157 bool SdrDragMovHdl::EndSdrDrag(bool /*bCopy*/) 1158 { 1159 if( GetDragHdl() ) 1160 { 1161 switch (GetDragHdl()->GetKind()) 1162 { 1163 case HDL_REF1: 1164 Ref1()=DragStat().GetNow(); 1165 break; 1166 1167 case HDL_REF2: 1168 Ref2()=DragStat().GetNow(); 1169 break; 1170 1171 case HDL_MIRX: 1172 Ref1()+=DragStat().GetNow()-DragStat().GetStart(); 1173 Ref2()+=DragStat().GetNow()-DragStat().GetStart(); 1174 break; 1175 1176 default: break; 1177 } 1178 } 1179 1180 return true; 1181 } 1182 1183 void SdrDragMovHdl::CancelSdrDrag() 1184 { 1185 Hide(); 1186 1187 SdrHdl* pHdl = GetDragHdl(); 1188 if( pHdl ) 1189 pHdl->SetPos(DragStat().GetRef1()); 1190 1191 SdrHdl* pHM = GetHdlList().GetHdl(HDL_MIRX); 1192 1193 if(pHM) 1194 pHM->Touch(); 1195 } 1196 1197 Pointer SdrDragMovHdl::GetSdrDragPointer() const 1198 { 1199 const SdrHdl* pHdl = GetDragHdl(); 1200 1201 if (pHdl!=NULL) 1202 { 1203 return pHdl->GetPointer(); 1204 } 1205 1206 return Pointer(POINTER_REFHAND); 1207 } 1208 1209 //////////////////////////////////////////////////////////////////////////////////////////////////// 1210 1211 TYPEINIT1(SdrDragObjOwn,SdrDragMethod); 1212 1213 SdrDragObjOwn::SdrDragObjOwn(SdrDragView& rNewView) 1214 : SdrDragMethod(rNewView), 1215 mpClone(0) 1216 { 1217 const SdrObject* pObj = GetDragObj(); 1218 1219 if(pObj) 1220 { 1221 // suppress full drag for some object types 1222 setSolidDraggingActive(pObj->supportsFullDrag()); 1223 } 1224 } 1225 1226 SdrDragObjOwn::~SdrDragObjOwn() 1227 { 1228 if(mpClone) 1229 { 1230 SdrObject::Free(mpClone); 1231 } 1232 } 1233 1234 void SdrDragObjOwn::createSdrDragEntries() 1235 { 1236 if(mpClone) 1237 { 1238 basegfx::B2DPolyPolygon aDragPolyPolygon; 1239 bool bAddWireframe(true); 1240 1241 if(getSolidDraggingActive()) 1242 { 1243 SdrPageView* pPV = getSdrDragView().GetSdrPageView(); 1244 1245 if(pPV && pPV->PageWindowCount()) 1246 { 1247 sdr::contact::ObjectContact& rOC = pPV->GetPageWindow(0)->GetObjectContact(); 1248 addSdrDragEntry(new SdrDragEntrySdrObject(*mpClone, rOC, false)); 1249 1250 // potentially no wireframe needed, full drag works 1251 bAddWireframe = false; 1252 } 1253 } 1254 1255 if(!bAddWireframe) 1256 { 1257 // check for extra conditions for wireframe, e.g. no border at 1258 // objects 1259 if(!mpClone->HasLineStyle()) 1260 { 1261 bAddWireframe = true; 1262 } 1263 } 1264 1265 if(bAddWireframe) 1266 { 1267 // use wireframe poly when full drag is off or did not work 1268 aDragPolyPolygon = mpClone->TakeXorPoly(); 1269 } 1270 1271 // add evtl. extra DragPolyPolygon 1272 const basegfx::B2DPolyPolygon aSpecialDragPolyPolygon(mpClone->getSpecialDragPoly(DragStat())); 1273 1274 if(aSpecialDragPolyPolygon.count()) 1275 { 1276 aDragPolyPolygon.append(aSpecialDragPolyPolygon); 1277 } 1278 1279 if(aDragPolyPolygon.count()) 1280 { 1281 addSdrDragEntry(new SdrDragEntryPolyPolygon(aDragPolyPolygon)); 1282 } 1283 } 1284 } 1285 1286 void SdrDragObjOwn::TakeSdrDragComment(XubString& rStr) const 1287 { 1288 // #i103058# get info string from the clone preferred, the original will 1289 // not be changed. For security, use original as fallback 1290 if(mpClone) 1291 { 1292 rStr = mpClone->getSpecialDragComment(DragStat()); 1293 } 1294 else 1295 { 1296 const SdrObject* pObj = GetDragObj(); 1297 1298 if(pObj) 1299 { 1300 rStr = pObj->getSpecialDragComment(DragStat()); 1301 } 1302 } 1303 } 1304 1305 bool SdrDragObjOwn::BeginSdrDrag() 1306 { 1307 if(!mpClone) 1308 { 1309 const SdrObject* pObj = GetDragObj(); 1310 1311 if(pObj && !pObj->IsResizeProtect()) 1312 { 1313 if(pObj->beginSpecialDrag(DragStat())) 1314 { 1315 // create nitial clone to have a start visualisation 1316 mpClone = pObj->getFullDragClone(); 1317 mpClone->applySpecialDrag(DragStat()); 1318 1319 return true; 1320 } 1321 } 1322 } 1323 1324 return false; 1325 } 1326 1327 void SdrDragObjOwn::MoveSdrDrag(const Point& rNoSnapPnt) 1328 { 1329 const SdrObject* pObj = GetDragObj(); 1330 1331 if(pObj) 1332 { 1333 Point aPnt(rNoSnapPnt); 1334 SdrPageView* pPV = GetDragPV(); 1335 1336 if(pPV) 1337 { 1338 if(!DragStat().IsNoSnap()) 1339 { 1340 SnapPos(aPnt); 1341 } 1342 1343 if(getSdrDragView().IsOrtho()) 1344 { 1345 if (DragStat().IsOrtho8Possible()) 1346 { 1347 OrthoDistance8(DragStat().GetStart(),aPnt,getSdrDragView().IsBigOrtho()); 1348 } 1349 else if (DragStat().IsOrtho4Possible()) 1350 { 1351 OrthoDistance4(DragStat().GetStart(),aPnt,getSdrDragView().IsBigOrtho()); 1352 } 1353 } 1354 1355 if(DragStat().CheckMinMoved(rNoSnapPnt)) 1356 { 1357 if(aPnt != DragStat().GetNow()) 1358 { 1359 Hide(); 1360 DragStat().NextMove(aPnt); 1361 1362 // since SdrDragObjOwn currently supports no transformation of 1363 // existing SdrDragEntries but only their recreation, a recreation 1364 // after every move is needed in this mode. Delete existing 1365 // SdrDragEntries here to force their recreation in the following Show(). 1366 clearSdrDragEntries(); 1367 1368 // delete current clone (after the last reference to it is deleted above) 1369 if(mpClone) 1370 { 1371 SdrObject::Free(mpClone); 1372 mpClone = 0; 1373 } 1374 1375 // create a new clone and modify to current drag state 1376 if(!mpClone) 1377 { 1378 mpClone = pObj->getFullDragClone(); 1379 mpClone->applySpecialDrag(DragStat()); 1380 1381 // #120999# AutoGrowWidth may change for SdrTextObj due to the automatism used 1382 // with bDisableAutoWidthOnDragging, so not only geometry changes but 1383 // also this (pretty indirect) property change is possible. If it gets 1384 // changed, it needs to be copied to the original since nothing will 1385 // happen when it only changes in the drag clone 1386 const bool bOldAutoGrowWidth(((SdrTextAutoGrowWidthItem&)pObj->GetMergedItem(SDRATTR_TEXT_AUTOGROWWIDTH)).GetValue()); 1387 const bool bNewAutoGrowWidth(((SdrTextAutoGrowWidthItem&)mpClone->GetMergedItem(SDRATTR_TEXT_AUTOGROWWIDTH)).GetValue()); 1388 1389 if(bOldAutoGrowWidth != bNewAutoGrowWidth) 1390 { 1391 GetDragObj()->SetMergedItem(SdrTextAutoGrowWidthItem(bNewAutoGrowWidth)); 1392 } 1393 } 1394 1395 Show(); 1396 } 1397 } 1398 } 1399 } 1400 } 1401 1402 bool SdrDragObjOwn::EndSdrDrag(bool /*bCopy*/) 1403 { 1404 Hide(); 1405 SdrUndoAction* pUndo = NULL; 1406 SdrUndoAction* pUndo2 = NULL; 1407 std::vector< SdrUndoAction* > vConnectorUndoActions; 1408 bool bRet = false; 1409 SdrObject* pObj = GetDragObj(); 1410 1411 if(pObj) 1412 { 1413 const bool bUndo = getSdrDragView().IsUndoEnabled(); 1414 1415 if( bUndo ) 1416 { 1417 if(!getSdrDragView().IsInsObjPoint() && pObj->IsInserted() ) 1418 { 1419 if (DragStat().IsEndDragChangesAttributes()) 1420 { 1421 pUndo=getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoAttrObject(*pObj); 1422 1423 if (DragStat().IsEndDragChangesGeoAndAttributes()) 1424 { 1425 vConnectorUndoActions = getSdrDragView().CreateConnectorUndo( *pObj ); 1426 pUndo2 = getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoGeoObject(*pObj); 1427 } 1428 } 1429 else 1430 { 1431 vConnectorUndoActions = getSdrDragView().CreateConnectorUndo( *pObj ); 1432 pUndo= getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoGeoObject(*pObj); 1433 } 1434 } 1435 1436 if( pUndo ) 1437 { 1438 getSdrDragView().BegUndo( pUndo->GetComment() ); 1439 } 1440 else 1441 { 1442 getSdrDragView().BegUndo(); 1443 } 1444 } 1445 1446 // evtl. use opertator= for setting changed object data (do not change selection in 1447 // view, this will destroy the interactor). This is possible since a clone is now 1448 // directly modified by the modifiers. Only SdrTableObj is adding own UNDOs 1449 // in it's SdrTableObj::endSpecialDrag, so currently not possible. OTOH it uses 1450 // a CreateUndoGeoObject() so maybe setting SetEndDragChangesAttributes is okay. I 1451 // will test this now 1452 Rectangle aBoundRect0; 1453 1454 if(pObj->GetUserCall()) 1455 { 1456 aBoundRect0 = pObj->GetLastBoundRect(); 1457 } 1458 1459 bRet = pObj->applySpecialDrag(DragStat()); 1460 1461 if(bRet) 1462 { 1463 pObj->SetChanged(); 1464 pObj->BroadcastObjectChange(); 1465 pObj->SendUserCall( SDRUSERCALL_RESIZE, aBoundRect0 ); 1466 } 1467 1468 if(bRet) 1469 { 1470 if( bUndo ) 1471 { 1472 getSdrDragView().AddUndoActions( vConnectorUndoActions ); 1473 1474 if ( pUndo ) 1475 { 1476 getSdrDragView().AddUndo(pUndo); 1477 } 1478 1479 if ( pUndo2 ) 1480 { 1481 getSdrDragView().AddUndo(pUndo2); 1482 } 1483 } 1484 } 1485 else 1486 { 1487 if( bUndo ) 1488 { 1489 std::vector< SdrUndoAction* >::iterator vConnectorUndoIter( vConnectorUndoActions.begin() ); 1490 1491 while( vConnectorUndoIter != vConnectorUndoActions.end() ) 1492 { 1493 delete *vConnectorUndoIter++; 1494 } 1495 1496 delete pUndo; 1497 delete pUndo2; 1498 } 1499 } 1500 1501 if( bUndo ) 1502 getSdrDragView().EndUndo(); 1503 } 1504 1505 return bRet; 1506 } 1507 1508 Pointer SdrDragObjOwn::GetSdrDragPointer() const 1509 { 1510 const SdrHdl* pHdl=GetDragHdl(); 1511 1512 if (pHdl) 1513 { 1514 return pHdl->GetPointer(); 1515 } 1516 1517 return Pointer(POINTER_MOVE); 1518 } 1519 1520 //////////////////////////////////////////////////////////////////////////////////////////////////// 1521 1522 TYPEINIT1(SdrDragMove,SdrDragMethod); 1523 1524 void SdrDragMove::createSdrDragEntryForSdrObject(const SdrObject& rOriginal, sdr::contact::ObjectContact& rObjectContact, bool /*bModify*/) 1525 { 1526 // for SdrDragMove, use current Primitive2DSequence of SdrObject visualisation 1527 // in given ObjectContact directly 1528 sdr::contact::ViewContact& rVC = rOriginal.GetViewContact(); 1529 sdr::contact::ViewObjectContact& rVOC = rVC.GetViewObjectContact(rObjectContact); 1530 sdr::contact::DisplayInfo aDisplayInfo; 1531 1532 // Do not use the last ViewPort set at the OC from the last ProcessDisplay(), 1533 // here we want the complete primitive sequence without visibility clippings 1534 rObjectContact.resetViewPort(); 1535 1536 addSdrDragEntry(new SdrDragEntryPrimitive2DSequence(rVOC.getPrimitive2DSequenceHierarchy(aDisplayInfo), true)); 1537 } 1538 1539 void SdrDragMove::applyCurrentTransformationToSdrObject(SdrObject& rTarget) 1540 { 1541 rTarget.Move(Size(DragStat().GetDX(), DragStat().GetDY())); 1542 } 1543 1544 SdrDragMove::SdrDragMove(SdrDragView& rNewView) 1545 : SdrDragMethod(rNewView) 1546 { 1547 setMoveOnly(true); 1548 } 1549 1550 void SdrDragMove::TakeSdrDragComment(XubString& rStr) const 1551 { 1552 XubString aStr; 1553 1554 ImpTakeDescriptionStr(STR_DragMethMove, rStr); 1555 rStr.AppendAscii(" (x="); 1556 getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDX(), aStr); 1557 rStr += aStr; 1558 rStr.AppendAscii(" y="); 1559 getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDY(), aStr); 1560 rStr += aStr; 1561 rStr += sal_Unicode(')'); 1562 1563 if(getSdrDragView().IsDragWithCopy()) 1564 { 1565 if(!getSdrDragView().IsInsObjPoint() && !getSdrDragView().IsInsGluePoint()) 1566 { 1567 rStr += ImpGetResStr(STR_EditWithCopy); 1568 } 1569 } 1570 } 1571 1572 bool SdrDragMove::BeginSdrDrag() 1573 { 1574 DragStat().SetActionRect(GetMarkedRect()); 1575 Show(); 1576 1577 return true; 1578 } 1579 1580 basegfx::B2DHomMatrix SdrDragMove::getCurrentTransformation() 1581 { 1582 return basegfx::tools::createTranslateB2DHomMatrix(DragStat().GetDX(), DragStat().GetDY()); 1583 } 1584 1585 void SdrDragMove::ImpCheckSnap(const Point& rPt) 1586 { 1587 Point aPt(rPt); 1588 sal_uInt16 nRet=SnapPos(aPt); 1589 aPt-=rPt; 1590 1591 if ((nRet & SDRSNAP_XSNAPPED) !=0) 1592 { 1593 if (bXSnapped) 1594 { 1595 if (Abs(aPt.X())<Abs(nBestXSnap)) 1596 { 1597 nBestXSnap=aPt.X(); 1598 } 1599 } 1600 else 1601 { 1602 nBestXSnap=aPt.X(); 1603 bXSnapped=true; 1604 } 1605 } 1606 1607 if ((nRet & SDRSNAP_YSNAPPED) !=0) 1608 { 1609 if (bYSnapped) 1610 { 1611 if (Abs(aPt.Y())<Abs(nBestYSnap)) 1612 { 1613 nBestYSnap=aPt.Y(); 1614 } 1615 } 1616 else 1617 { 1618 nBestYSnap=aPt.Y(); 1619 bYSnapped=true; 1620 } 1621 } 1622 } 1623 1624 void SdrDragMove::MoveSdrDrag(const Point& rNoSnapPnt_) 1625 { 1626 nBestXSnap=0; 1627 nBestYSnap=0; 1628 bXSnapped=false; 1629 bYSnapped=false; 1630 Point aNoSnapPnt(rNoSnapPnt_); 1631 const Rectangle& aSR=GetMarkedRect(); 1632 long nMovedx=aNoSnapPnt.X()-DragStat().GetStart().X(); 1633 long nMovedy=aNoSnapPnt.Y()-DragStat().GetStart().Y(); 1634 Point aLO(aSR.TopLeft()); aLO.X()+=nMovedx; aLO.Y()+=nMovedy; 1635 Point aRU(aSR.BottomRight()); aRU.X()+=nMovedx; aRU.Y()+=nMovedy; 1636 Point aLU(aLO.X(),aRU.Y()); 1637 Point aRO(aRU.X(),aLO.Y()); 1638 ImpCheckSnap(aLO); 1639 1640 if (!getSdrDragView().IsMoveSnapOnlyTopLeft()) 1641 { 1642 ImpCheckSnap(aRO); 1643 ImpCheckSnap(aLU); 1644 ImpCheckSnap(aRU); 1645 } 1646 1647 Point aPnt(aNoSnapPnt.X()+nBestXSnap,aNoSnapPnt.Y()+nBestYSnap); 1648 bool bOrtho=getSdrDragView().IsOrtho(); 1649 1650 if (bOrtho) 1651 OrthoDistance8(DragStat().GetStart(),aPnt,getSdrDragView().IsBigOrtho()); 1652 1653 if (DragStat().CheckMinMoved(aNoSnapPnt)) 1654 { 1655 Point aPt1(aPnt); 1656 Rectangle aLR(getSdrDragView().GetWorkArea()); 1657 bool bWorkArea=!aLR.IsEmpty(); 1658 bool bDragLimit=IsDragLimit(); 1659 1660 if (bDragLimit || bWorkArea) 1661 { 1662 Rectangle aSR2(GetMarkedRect()); 1663 Point aD(aPt1-DragStat().GetStart()); 1664 1665 if (bDragLimit) 1666 { 1667 Rectangle aR2(GetDragLimitRect()); 1668 1669 if (bWorkArea) 1670 aLR.Intersection(aR2); 1671 else 1672 aLR=aR2; 1673 } 1674 1675 if (aSR2.Left()>aLR.Left() || aSR2.Right()<aLR.Right()) 1676 { // ist ueberhaupt Platz zum verschieben? 1677 aSR2.Move(aD.X(),0); 1678 1679 if (aSR2.Left()<aLR.Left()) 1680 { 1681 aPt1.X()-=aSR2.Left()-aLR.Left(); 1682 } 1683 else if (aSR2.Right()>aLR.Right()) 1684 { 1685 aPt1.X()-=aSR2.Right()-aLR.Right(); 1686 } 1687 } 1688 else 1689 aPt1.X()=DragStat().GetStart().X(); // kein Platz zum verschieben 1690 1691 if (aSR2.Top()>aLR.Top() || aSR2.Bottom()<aLR.Bottom()) 1692 { // ist ueberhaupt Platz zum verschieben? 1693 aSR2.Move(0,aD.Y()); 1694 1695 if (aSR2.Top()<aLR.Top()) 1696 { 1697 aPt1.Y()-=aSR2.Top()-aLR.Top(); 1698 } 1699 else if (aSR2.Bottom()>aLR.Bottom()) 1700 { 1701 aPt1.Y()-=aSR2.Bottom()-aLR.Bottom(); 1702 } 1703 } 1704 else 1705 aPt1.Y()=DragStat().GetStart().Y(); // kein Platz zum verschieben 1706 } 1707 1708 if (getSdrDragView().IsDraggingGluePoints()) 1709 { // Klebepunkte aufs BoundRect des Obj limitieren 1710 aPt1-=DragStat().GetStart(); 1711 const SdrMarkList& rML=GetMarkedObjectList(); 1712 sal_uLong nMarkAnz=rML.GetMarkCount(); 1713 1714 for (sal_uLong nMarkNum=0; nMarkNum<nMarkAnz; nMarkNum++) 1715 { 1716 const SdrMark* pM=rML.GetMark(nMarkNum); 1717 const SdrUShortCont* pPts=pM->GetMarkedGluePoints(); 1718 sal_uLong nPtAnz=pPts==NULL ? 0 : pPts->GetCount(); 1719 1720 if (nPtAnz!=0) 1721 { 1722 const SdrObject* pObj=pM->GetMarkedSdrObj(); 1723 const SdrGluePointList* pGPL=pObj->GetGluePointList(); 1724 Rectangle aBound(pObj->GetCurrentBoundRect()); 1725 1726 for (sal_uLong nPtNum=0; nPtNum<nPtAnz; nPtNum++) 1727 { 1728 sal_uInt16 nId=pPts->GetObject(nPtNum); 1729 sal_uInt16 nGlueNum=pGPL->FindGluePoint(nId); 1730 1731 if (nGlueNum!=SDRGLUEPOINT_NOTFOUND) 1732 { 1733 Point aPt((*pGPL)[nGlueNum].GetAbsolutePos(*pObj)); 1734 aPt+=aPt1; // soviel soll verschoben werden 1735 if (aPt.X()<aBound.Left() ) aPt1.X()-=aPt.X()-aBound.Left() ; 1736 if (aPt.X()>aBound.Right() ) aPt1.X()-=aPt.X()-aBound.Right() ; 1737 if (aPt.Y()<aBound.Top() ) aPt1.Y()-=aPt.Y()-aBound.Top() ; 1738 if (aPt.Y()>aBound.Bottom()) aPt1.Y()-=aPt.Y()-aBound.Bottom(); 1739 } 1740 } 1741 } 1742 } 1743 1744 aPt1+=DragStat().GetStart(); 1745 } 1746 1747 if (bOrtho) 1748 OrthoDistance8(DragStat().GetStart(),aPt1,false); 1749 1750 if (aPt1!=DragStat().GetNow()) 1751 { 1752 Hide(); 1753 DragStat().NextMove(aPt1); 1754 Rectangle aAction(GetMarkedRect()); 1755 aAction.Move(DragStat().GetDX(),DragStat().GetDY()); 1756 DragStat().SetActionRect(aAction); 1757 Show(); 1758 } 1759 } 1760 } 1761 1762 bool SdrDragMove::EndSdrDrag(bool bCopy) 1763 { 1764 Hide(); 1765 1766 if (getSdrDragView().IsInsObjPoint() || getSdrDragView().IsInsGluePoint()) 1767 bCopy=false; 1768 1769 if (IsDraggingPoints()) 1770 { 1771 getSdrDragView().MoveMarkedPoints(Size(DragStat().GetDX(),DragStat().GetDY()),bCopy); 1772 } 1773 else if (IsDraggingGluePoints()) 1774 { 1775 getSdrDragView().MoveMarkedGluePoints(Size(DragStat().GetDX(),DragStat().GetDY()),bCopy); 1776 } 1777 else 1778 { 1779 getSdrDragView().MoveMarkedObj(Size(DragStat().GetDX(),DragStat().GetDY()),bCopy); 1780 } 1781 1782 return true; 1783 } 1784 1785 Pointer SdrDragMove::GetSdrDragPointer() const 1786 { 1787 if (IsDraggingPoints() || IsDraggingGluePoints()) 1788 { 1789 return Pointer(POINTER_MOVEPOINT); 1790 } 1791 else 1792 { 1793 return Pointer(POINTER_MOVE); 1794 } 1795 } 1796 1797 //////////////////////////////////////////////////////////////////////////////////////////////////// 1798 1799 TYPEINIT1(SdrDragResize,SdrDragMethod); 1800 1801 SdrDragResize::SdrDragResize(SdrDragView& rNewView) 1802 : SdrDragMethod(rNewView), 1803 aXFact(1,1), 1804 aYFact(1,1) 1805 { 1806 } 1807 1808 void SdrDragResize::TakeSdrDragComment(XubString& rStr) const 1809 { 1810 ImpTakeDescriptionStr(STR_DragMethResize, rStr); 1811 bool bEqual(aXFact == aYFact); 1812 Fraction aFact1(1,1); 1813 Point aStart(DragStat().GetStart()); 1814 Point aRef(DragStat().GetRef1()); 1815 sal_Int32 nXDiv(aStart.X() - aRef.X()); 1816 1817 if(!nXDiv) 1818 nXDiv = 1; 1819 1820 sal_Int32 nYDiv(aStart.Y() - aRef.Y()); 1821 1822 if(!nYDiv) 1823 nYDiv = 1; 1824 1825 bool bX(aXFact != aFact1 && Abs(nXDiv) > 1); 1826 bool bY(aYFact != aFact1 && Abs(nYDiv) > 1); 1827 1828 if(bX || bY) 1829 { 1830 XubString aStr; 1831 1832 rStr.AppendAscii(" ("); 1833 1834 if(bX) 1835 { 1836 if(!bEqual) 1837 rStr.AppendAscii("x="); 1838 1839 getSdrDragView().GetModel()->TakePercentStr(aXFact, aStr); 1840 rStr += aStr; 1841 } 1842 1843 if(bY && !bEqual) 1844 { 1845 if(bX) 1846 rStr += sal_Unicode(' '); 1847 1848 rStr.AppendAscii("y="); 1849 getSdrDragView().GetModel()->TakePercentStr(aYFact, aStr); 1850 rStr += aStr; 1851 } 1852 1853 rStr += sal_Unicode(')'); 1854 } 1855 1856 if(getSdrDragView().IsDragWithCopy()) 1857 rStr += ImpGetResStr(STR_EditWithCopy); 1858 } 1859 1860 bool SdrDragResize::BeginSdrDrag() 1861 { 1862 SdrHdlKind eRefHdl=HDL_MOVE; 1863 SdrHdl* pRefHdl=NULL; 1864 1865 switch (GetDragHdlKind()) 1866 { 1867 case HDL_UPLFT: eRefHdl=HDL_LWRGT; break; 1868 case HDL_UPPER: eRefHdl=HDL_LOWER; DragStat().SetHorFixed(true); break; 1869 case HDL_UPRGT: eRefHdl=HDL_LWLFT; break; 1870 case HDL_LEFT : eRefHdl=HDL_RIGHT; DragStat().SetVerFixed(true); break; 1871 case HDL_RIGHT: eRefHdl=HDL_LEFT ; DragStat().SetVerFixed(true); break; 1872 case HDL_LWLFT: eRefHdl=HDL_UPRGT; break; 1873 case HDL_LOWER: eRefHdl=HDL_UPPER; DragStat().SetHorFixed(true); break; 1874 case HDL_LWRGT: eRefHdl=HDL_UPLFT; break; 1875 default: break; 1876 } 1877 1878 if (eRefHdl!=HDL_MOVE) 1879 pRefHdl=GetHdlList().GetHdl(eRefHdl); 1880 1881 if (pRefHdl!=NULL && !getSdrDragView().IsResizeAtCenter()) 1882 { 1883 DragStat().Ref1()=pRefHdl->GetPos(); 1884 } 1885 else 1886 { 1887 SdrHdl* pRef1=GetHdlList().GetHdl(HDL_UPLFT); 1888 SdrHdl* pRef2=GetHdlList().GetHdl(HDL_LWRGT); 1889 1890 if (pRef1!=NULL && pRef2!=NULL) 1891 { 1892 DragStat().Ref1()=Rectangle(pRef1->GetPos(),pRef2->GetPos()).Center(); 1893 } 1894 else 1895 { 1896 DragStat().Ref1()=GetMarkedRect().Center(); 1897 } 1898 } 1899 1900 Show(); 1901 1902 return true; 1903 } 1904 1905 basegfx::B2DHomMatrix SdrDragResize::getCurrentTransformation() 1906 { 1907 basegfx::B2DHomMatrix aRetval(basegfx::tools::createTranslateB2DHomMatrix( 1908 -DragStat().Ref1().X(), -DragStat().Ref1().Y())); 1909 aRetval.scale(aXFact, aYFact); 1910 aRetval.translate(DragStat().Ref1().X(), DragStat().Ref1().Y()); 1911 1912 return aRetval; 1913 } 1914 1915 void SdrDragResize::MoveSdrDrag(const Point& rNoSnapPnt) 1916 { 1917 Point aPnt(GetSnapPos(rNoSnapPnt)); 1918 Point aStart(DragStat().GetStart()); 1919 Point aRef(DragStat().GetRef1()); 1920 Fraction aMaxFact(0x7FFFFFFF,1); 1921 Rectangle aLR(getSdrDragView().GetWorkArea()); 1922 bool bWorkArea=!aLR.IsEmpty(); 1923 bool bDragLimit=IsDragLimit(); 1924 1925 if (bDragLimit || bWorkArea) 1926 { 1927 Rectangle aSR(GetMarkedRect()); 1928 1929 if (bDragLimit) 1930 { 1931 Rectangle aR2(GetDragLimitRect()); 1932 1933 if (bWorkArea) 1934 aLR.Intersection(aR2); 1935 else 1936 aLR=aR2; 1937 } 1938 1939 if (aPnt.X()<aLR.Left()) 1940 aPnt.X()=aLR.Left(); 1941 else if (aPnt.X()>aLR.Right()) 1942 aPnt.X()=aLR.Right(); 1943 1944 if (aPnt.Y()<aLR.Top()) 1945 aPnt.Y()=aLR.Top(); 1946 else if (aPnt.Y()>aLR.Bottom()) 1947 aPnt.Y()=aLR.Bottom(); 1948 1949 if (aRef.X()>aSR.Left()) 1950 { 1951 Fraction aMax(aRef.X()-aLR.Left(),aRef.X()-aSR.Left()); 1952 1953 if (aMax<aMaxFact) 1954 aMaxFact=aMax; 1955 } 1956 1957 if (aRef.X()<aSR.Right()) 1958 { 1959 Fraction aMax(aLR.Right()-aRef.X(),aSR.Right()-aRef.X()); 1960 1961 if (aMax<aMaxFact) 1962 aMaxFact=aMax; 1963 } 1964 1965 if (aRef.Y()>aSR.Top()) 1966 { 1967 Fraction aMax(aRef.Y()-aLR.Top(),aRef.Y()-aSR.Top()); 1968 1969 if (aMax<aMaxFact) 1970 aMaxFact=aMax; 1971 } 1972 1973 if (aRef.Y()<aSR.Bottom()) 1974 { 1975 Fraction aMax(aLR.Bottom()-aRef.Y(),aSR.Bottom()-aRef.Y()); 1976 1977 if (aMax<aMaxFact) 1978 aMaxFact=aMax; 1979 } 1980 } 1981 1982 long nXDiv=aStart.X()-aRef.X(); if (nXDiv==0) nXDiv=1; 1983 long nYDiv=aStart.Y()-aRef.Y(); if (nYDiv==0) nYDiv=1; 1984 long nXMul=aPnt.X()-aRef.X(); 1985 long nYMul=aPnt.Y()-aRef.Y(); 1986 1987 if (nXDiv<0) 1988 { 1989 nXDiv=-nXDiv; 1990 nXMul=-nXMul; 1991 } 1992 1993 if (nYDiv<0) 1994 { 1995 nYDiv=-nYDiv; 1996 nYMul=-nYMul; 1997 } 1998 1999 bool bXNeg=nXMul<0; if (bXNeg) nXMul=-nXMul; 2000 bool bYNeg=nYMul<0; if (bYNeg) nYMul=-nYMul; 2001 bool bOrtho=getSdrDragView().IsOrtho() || !getSdrDragView().IsResizeAllowed(false); 2002 2003 if (!DragStat().IsHorFixed() && !DragStat().IsVerFixed()) 2004 { 2005 if (Abs(nXDiv)<=1 || Abs(nYDiv)<=1) 2006 bOrtho=false; 2007 2008 if (bOrtho) 2009 { 2010 if ((Fraction(nXMul,nXDiv)>Fraction(nYMul,nYDiv)) !=getSdrDragView().IsBigOrtho()) 2011 { 2012 nXMul=nYMul; 2013 nXDiv=nYDiv; 2014 } 2015 else 2016 { 2017 nYMul=nXMul; 2018 nYDiv=nXDiv; 2019 } 2020 } 2021 } 2022 else 2023 { 2024 if (bOrtho) 2025 { 2026 if (DragStat().IsHorFixed()) 2027 { 2028 bXNeg=false; 2029 nXMul=nYMul; 2030 nXDiv=nYDiv; 2031 } 2032 2033 if (DragStat().IsVerFixed()) 2034 { 2035 bYNeg=false; 2036 nYMul=nXMul; 2037 nYDiv=nXDiv; 2038 } 2039 } 2040 else 2041 { 2042 if (DragStat().IsHorFixed()) 2043 { 2044 bXNeg=false; 2045 nXMul=1; 2046 nXDiv=1; 2047 } 2048 2049 if (DragStat().IsVerFixed()) 2050 { 2051 bYNeg=false; 2052 nYMul=1; 2053 nYDiv=1; 2054 } 2055 } 2056 } 2057 2058 Fraction aNeuXFact(nXMul,nXDiv); 2059 Fraction aNeuYFact(nYMul,nYDiv); 2060 2061 if (bOrtho) 2062 { 2063 if (aNeuXFact>aMaxFact) 2064 { 2065 aNeuXFact=aMaxFact; 2066 aNeuYFact=aMaxFact; 2067 } 2068 2069 if (aNeuYFact>aMaxFact) 2070 { 2071 aNeuXFact=aMaxFact; 2072 aNeuYFact=aMaxFact; 2073 } 2074 } 2075 2076 if (bXNeg) 2077 aNeuXFact=Fraction(-aNeuXFact.GetNumerator(),aNeuXFact.GetDenominator()); 2078 2079 if (bYNeg) 2080 aNeuYFact=Fraction(-aNeuYFact.GetNumerator(),aNeuYFact.GetDenominator()); 2081 2082 if (DragStat().CheckMinMoved(aPnt)) 2083 { 2084 if ((!DragStat().IsHorFixed() && aPnt.X()!=DragStat().GetNow().X()) || 2085 (!DragStat().IsVerFixed() && aPnt.Y()!=DragStat().GetNow().Y())) 2086 { 2087 Hide(); 2088 DragStat().NextMove(aPnt); 2089 aXFact=aNeuXFact; 2090 aYFact=aNeuYFact; 2091 Show(); 2092 } 2093 } 2094 } 2095 2096 void SdrDragResize::applyCurrentTransformationToSdrObject(SdrObject& rTarget) 2097 { 2098 rTarget.Resize(DragStat().Ref1(),aXFact,aYFact); 2099 } 2100 2101 bool SdrDragResize::EndSdrDrag(bool bCopy) 2102 { 2103 Hide(); 2104 2105 if (IsDraggingPoints()) 2106 { 2107 getSdrDragView().ResizeMarkedPoints(DragStat().Ref1(),aXFact,aYFact,bCopy); 2108 } 2109 else if (IsDraggingGluePoints()) 2110 { 2111 getSdrDragView().ResizeMarkedGluePoints(DragStat().Ref1(),aXFact,aYFact,bCopy); 2112 } 2113 else 2114 { 2115 getSdrDragView().ResizeMarkedObj(DragStat().Ref1(),aXFact,aYFact,bCopy); 2116 } 2117 2118 return true; 2119 } 2120 2121 Pointer SdrDragResize::GetSdrDragPointer() const 2122 { 2123 const SdrHdl* pHdl=GetDragHdl(); 2124 2125 if (pHdl!=NULL) 2126 { 2127 return pHdl->GetPointer(); 2128 } 2129 2130 return Pointer(POINTER_MOVE); 2131 } 2132 2133 //////////////////////////////////////////////////////////////////////////////////////////////////// 2134 2135 TYPEINIT1(SdrDragRotate,SdrDragMethod); 2136 2137 void SdrDragRotate::applyCurrentTransformationToSdrObject(SdrObject& rTarget) 2138 { 2139 rTarget.Rotate(DragStat().GetRef1(), nWink, sin(nWink*nPi180), cos(nWink*nPi180)); 2140 } 2141 2142 SdrDragRotate::SdrDragRotate(SdrDragView& rNewView) 2143 : SdrDragMethod(rNewView), 2144 nSin(0.0), 2145 nCos(1.0), 2146 nWink0(0), 2147 nWink(0), 2148 bRight(false) 2149 { 2150 } 2151 2152 void SdrDragRotate::TakeSdrDragComment(XubString& rStr) const 2153 { 2154 ImpTakeDescriptionStr(STR_DragMethRotate, rStr); 2155 rStr.AppendAscii(" ("); 2156 XubString aStr; 2157 sal_Int32 nTmpWink(NormAngle360(nWink)); 2158 2159 if(bRight && nWink) 2160 { 2161 nTmpWink -= 36000; 2162 } 2163 2164 getSdrDragView().GetModel()->TakeWinkStr(nTmpWink, aStr); 2165 rStr += aStr; 2166 rStr += sal_Unicode(')'); 2167 2168 if(getSdrDragView().IsDragWithCopy()) 2169 rStr += ImpGetResStr(STR_EditWithCopy); 2170 } 2171 2172 bool SdrDragRotate::BeginSdrDrag() 2173 { 2174 SdrHdl* pH=GetHdlList().GetHdl(HDL_REF1); 2175 2176 if (pH!=NULL) 2177 { 2178 Show(); 2179 DragStat().Ref1()=pH->GetPos(); 2180 nWink0=GetAngle(DragStat().GetStart()-DragStat().GetRef1()); 2181 return true; 2182 } 2183 else 2184 { 2185 DBG_ERROR("SdrDragRotate::BeginSdrDrag(): Kein Referenzpunkt-Handle gefunden"); 2186 return false; 2187 } 2188 } 2189 2190 basegfx::B2DHomMatrix SdrDragRotate::getCurrentTransformation() 2191 { 2192 return basegfx::tools::createRotateAroundPoint( 2193 DragStat().GetRef1().X(), DragStat().GetRef1().Y(), 2194 -atan2(nSin, nCos)); 2195 } 2196 2197 void SdrDragRotate::MoveSdrDrag(const Point& rPnt_) 2198 { 2199 Point aPnt(rPnt_); 2200 if (DragStat().CheckMinMoved(aPnt)) 2201 { 2202 long nNeuWink=NormAngle360(GetAngle(aPnt-DragStat().GetRef1())-nWink0); 2203 long nSA=0; 2204 2205 if (getSdrDragView().IsAngleSnapEnabled()) 2206 nSA=getSdrDragView().GetSnapAngle(); 2207 2208 if (!getSdrDragView().IsRotateAllowed(false)) 2209 nSA=9000; 2210 2211 if (nSA!=0) 2212 { // Winkelfang 2213 nNeuWink+=nSA/2; 2214 nNeuWink/=nSA; 2215 nNeuWink*=nSA; 2216 } 2217 2218 nNeuWink=NormAngle180(nNeuWink); 2219 2220 if (nWink!=nNeuWink) 2221 { 2222 sal_uInt16 nSekt0=GetAngleSector(nWink); 2223 sal_uInt16 nSekt1=GetAngleSector(nNeuWink); 2224 2225 if (nSekt0==0 && nSekt1==3) 2226 bRight=true; 2227 2228 if (nSekt0==3 && nSekt1==0) 2229 bRight=false; 2230 2231 nWink=nNeuWink; 2232 double a=nWink*nPi180; 2233 double nSin1=sin(a); // schonmal berechnen, damit mgl. wenig Zeit 2234 double nCos1=cos(a); // zwischen Hide() und Show() vergeht 2235 Hide(); 2236 nSin=nSin1; 2237 nCos=nCos1; 2238 DragStat().NextMove(aPnt); 2239 Show(); 2240 } 2241 } 2242 } 2243 2244 bool SdrDragRotate::EndSdrDrag(bool bCopy) 2245 { 2246 Hide(); 2247 2248 if (nWink!=0) 2249 { 2250 if (IsDraggingPoints()) 2251 { 2252 getSdrDragView().RotateMarkedPoints(DragStat().GetRef1(),nWink,bCopy); 2253 } 2254 else if (IsDraggingGluePoints()) 2255 { 2256 getSdrDragView().RotateMarkedGluePoints(DragStat().GetRef1(),nWink,bCopy); 2257 } 2258 else 2259 { 2260 getSdrDragView().RotateMarkedObj(DragStat().GetRef1(),nWink,bCopy); 2261 } 2262 } 2263 return true; 2264 } 2265 2266 Pointer SdrDragRotate::GetSdrDragPointer() const 2267 { 2268 return Pointer(POINTER_ROTATE); 2269 } 2270 2271 //////////////////////////////////////////////////////////////////////////////////////////////////// 2272 2273 TYPEINIT1(SdrDragShear,SdrDragMethod); 2274 2275 SdrDragShear::SdrDragShear(SdrDragView& rNewView, bool bSlant1) 2276 : SdrDragMethod(rNewView), 2277 aFact(1,1), 2278 nWink0(0), 2279 nWink(0), 2280 nTan(0.0), 2281 bVertical(false), 2282 bResize(false), 2283 bUpSideDown(false), 2284 bSlant(bSlant1) 2285 { 2286 } 2287 2288 void SdrDragShear::TakeSdrDragComment(XubString& rStr) const 2289 { 2290 ImpTakeDescriptionStr(STR_DragMethShear, rStr); 2291 rStr.AppendAscii(" ("); 2292 2293 sal_Int32 nTmpWink(nWink); 2294 2295 if(bUpSideDown) 2296 nTmpWink += 18000; 2297 2298 nTmpWink = NormAngle180(nTmpWink); 2299 2300 XubString aStr; 2301 2302 getSdrDragView().GetModel()->TakeWinkStr(nTmpWink, aStr); 2303 rStr += aStr; 2304 rStr += sal_Unicode(')'); 2305 2306 if(getSdrDragView().IsDragWithCopy()) 2307 rStr += ImpGetResStr(STR_EditWithCopy); 2308 } 2309 2310 bool SdrDragShear::BeginSdrDrag() 2311 { 2312 SdrHdlKind eRefHdl=HDL_MOVE; 2313 SdrHdl* pRefHdl=NULL; 2314 2315 switch (GetDragHdlKind()) 2316 { 2317 case HDL_UPPER: eRefHdl=HDL_LOWER; break; 2318 case HDL_LOWER: eRefHdl=HDL_UPPER; break; 2319 case HDL_LEFT : eRefHdl=HDL_RIGHT; bVertical=true; break; 2320 case HDL_RIGHT: eRefHdl=HDL_LEFT ; bVertical=true; break; 2321 default: break; 2322 } 2323 2324 if (eRefHdl!=HDL_MOVE) 2325 pRefHdl=GetHdlList().GetHdl(eRefHdl); 2326 2327 if (pRefHdl!=NULL) 2328 { 2329 DragStat().Ref1()=pRefHdl->GetPos(); 2330 nWink0=GetAngle(DragStat().GetStart()-DragStat().GetRef1()); 2331 } 2332 else 2333 { 2334 DBG_ERROR("SdrDragShear::BeginSdrDrag(): Kein Referenzpunkt-Handle fuer Shear gefunden"); 2335 return false; 2336 } 2337 2338 Show(); 2339 return true; 2340 } 2341 2342 basegfx::B2DHomMatrix SdrDragShear::getCurrentTransformation() 2343 { 2344 basegfx::B2DHomMatrix aRetval(basegfx::tools::createTranslateB2DHomMatrix( 2345 -DragStat().GetRef1().X(), -DragStat().GetRef1().Y())); 2346 2347 if (bResize) 2348 { 2349 if (bVertical) 2350 { 2351 aRetval.scale(aFact, 1.0); 2352 aRetval.shearY(-nTan); 2353 } 2354 else 2355 { 2356 aRetval.scale(1.0, aFact); 2357 aRetval.shearX(-nTan); 2358 } 2359 } 2360 2361 aRetval.translate(DragStat().GetRef1().X(), DragStat().GetRef1().Y()); 2362 2363 return aRetval; 2364 } 2365 2366 void SdrDragShear::MoveSdrDrag(const Point& rPnt) 2367 { 2368 if (DragStat().CheckMinMoved(rPnt)) 2369 { 2370 bResize=!getSdrDragView().IsOrtho(); 2371 long nSA=0; 2372 2373 if (getSdrDragView().IsAngleSnapEnabled()) 2374 nSA=getSdrDragView().GetSnapAngle(); 2375 2376 Point aP0(DragStat().GetStart()); 2377 Point aPnt(rPnt); 2378 Fraction aNeuFact(1,1); 2379 2380 // Wenn kein Winkelfang, dann ggf. Rasterfang (ausser bei Slant) 2381 if (nSA==0 && !bSlant) 2382 aPnt=GetSnapPos(aPnt); 2383 2384 if (!bSlant && !bResize) 2385 { // Shear ohne Resize 2386 if (bVertical) 2387 aPnt.X()=aP0.X(); 2388 else 2389 aPnt.Y()=aP0.Y(); 2390 } 2391 2392 Point aRef(DragStat().GetRef1()); 2393 Point aDif(aPnt-aRef); 2394 2395 long nNeuWink=0; 2396 2397 if (bSlant) 2398 { 2399 nNeuWink=NormAngle180(-(GetAngle(aDif)-nWink0)); 2400 2401 if (bVertical) 2402 nNeuWink=NormAngle180(-nNeuWink); 2403 } 2404 else 2405 { 2406 if (bVertical) 2407 nNeuWink=NormAngle180(GetAngle(aDif)); 2408 else 2409 nNeuWink=NormAngle180(-(GetAngle(aDif)-9000)); 2410 2411 if (nNeuWink<-9000 || nNeuWink>9000) 2412 nNeuWink=NormAngle180(nNeuWink+18000); 2413 2414 if (bResize) 2415 { 2416 Point aPt2(aPnt); 2417 2418 if (nSA!=0) 2419 aPt2=GetSnapPos(aPnt); // den also in jedem Falle fangen 2420 2421 if (bVertical) 2422 { 2423 aNeuFact=Fraction(aPt2.X()-aRef.X(),aP0.X()-aRef.X()); 2424 } 2425 else 2426 { 2427 aNeuFact=Fraction(aPt2.Y()-aRef.Y(),aP0.Y()-aRef.Y()); 2428 } 2429 } 2430 } 2431 2432 bool bNeg=nNeuWink<0; 2433 2434 if (bNeg) 2435 nNeuWink=-nNeuWink; 2436 2437 if (nSA!=0) 2438 { // Winkelfang 2439 nNeuWink+=nSA/2; 2440 nNeuWink/=nSA; 2441 nNeuWink*=nSA; 2442 } 2443 2444 nNeuWink=NormAngle360(nNeuWink); 2445 bUpSideDown=nNeuWink>9000 && nNeuWink<27000; 2446 2447 if (bSlant) 2448 { // Resize fuer Slant berechnen 2449 // Mit Winkelfang jedoch ohne 89deg Begrenzung 2450 long nTmpWink=nNeuWink; 2451 if (bUpSideDown) nNeuWink-=18000; 2452 if (bNeg) nTmpWink=-nTmpWink; 2453 bResize=true; 2454 double nCos=cos(nTmpWink*nPi180); 2455 aNeuFact=nCos; 2456 Kuerzen(aFact,10); // 3 Dezimalstellen sollten reichen 2457 } 2458 2459 if (nNeuWink>8900) 2460 nNeuWink=8900; 2461 2462 if (bNeg) 2463 nNeuWink=-nNeuWink; 2464 2465 if (nWink!=nNeuWink || aFact!=aNeuFact) 2466 { 2467 nWink=nNeuWink; 2468 aFact=aNeuFact; 2469 double a=nWink*nPi180; 2470 double nTan1=0.0; 2471 nTan1=tan(a); // schonmal berechnen, damit mgl. wenig Zeit zwischen Hide() und Show() vergeht 2472 Hide(); 2473 nTan=nTan1; 2474 DragStat().NextMove(rPnt); 2475 Show(); 2476 } 2477 } 2478 } 2479 2480 void SdrDragShear::applyCurrentTransformationToSdrObject(SdrObject& rTarget) 2481 { 2482 if (bResize) 2483 { 2484 if (bVertical) 2485 { 2486 rTarget.Resize(DragStat().GetRef1(),aFact,Fraction(1,1)); 2487 } 2488 else 2489 { 2490 rTarget.Resize(DragStat().GetRef1(),Fraction(1,1),aFact); 2491 } 2492 } 2493 2494 if (nWink!=0) 2495 { 2496 rTarget.Shear(DragStat().GetRef1(),nWink,tan(nWink*nPi180),bVertical); 2497 } 2498 } 2499 2500 bool SdrDragShear::EndSdrDrag(bool bCopy) 2501 { 2502 Hide(); 2503 2504 if (bResize && aFact==Fraction(1,1)) 2505 bResize=false; 2506 2507 if (nWink!=0 || bResize) 2508 { 2509 if (nWink!=0 && bResize) 2510 { 2511 XubString aStr; 2512 ImpTakeDescriptionStr(STR_EditShear,aStr); 2513 2514 if (bCopy) 2515 aStr+=ImpGetResStr(STR_EditWithCopy); 2516 2517 getSdrDragView().BegUndo(aStr); 2518 } 2519 2520 if (bResize) 2521 { 2522 if (bVertical) 2523 { 2524 getSdrDragView().ResizeMarkedObj(DragStat().GetRef1(),aFact,Fraction(1,1),bCopy); 2525 } 2526 else 2527 { 2528 getSdrDragView().ResizeMarkedObj(DragStat().GetRef1(),Fraction(1,1),aFact,bCopy); 2529 } 2530 2531 bCopy=false; 2532 } 2533 2534 if (nWink!=0) 2535 { 2536 getSdrDragView().ShearMarkedObj(DragStat().GetRef1(),nWink,bVertical,bCopy); 2537 } 2538 2539 if (nWink!=0 && bResize) 2540 getSdrDragView().EndUndo(); 2541 2542 return true; 2543 } 2544 2545 return false; 2546 } 2547 2548 Pointer SdrDragShear::GetSdrDragPointer() const 2549 { 2550 if (bVertical) 2551 return Pointer(POINTER_VSHEAR); 2552 else 2553 return Pointer(POINTER_HSHEAR); 2554 } 2555 2556 //////////////////////////////////////////////////////////////////////////////////////////////////// 2557 2558 TYPEINIT1(SdrDragMirror,SdrDragMethod); 2559 2560 void SdrDragMirror::applyCurrentTransformationToSdrObject(SdrObject& rTarget) 2561 { 2562 if(bMirrored) 2563 { 2564 rTarget.Mirror(DragStat().GetRef1(), DragStat().GetRef2()); 2565 } 2566 } 2567 2568 SdrDragMirror::SdrDragMirror(SdrDragView& rNewView) 2569 : SdrDragMethod(rNewView), 2570 nWink(0), 2571 bMirrored(false), 2572 bSide0(false) 2573 { 2574 } 2575 2576 bool SdrDragMirror::ImpCheckSide(const Point& rPnt) const 2577 { 2578 long nWink1=GetAngle(rPnt-DragStat().GetRef1()); 2579 nWink1-=nWink; 2580 nWink1=NormAngle360(nWink1); 2581 2582 return nWink1<18000; 2583 } 2584 2585 void SdrDragMirror::TakeSdrDragComment(XubString& rStr) const 2586 { 2587 if (aDif.X()==0) 2588 ImpTakeDescriptionStr(STR_DragMethMirrorHori,rStr); 2589 else if (aDif.Y()==0) 2590 ImpTakeDescriptionStr(STR_DragMethMirrorVert,rStr); 2591 else if (Abs(aDif.X())==Abs(aDif.Y())) 2592 ImpTakeDescriptionStr(STR_DragMethMirrorDiag,rStr); 2593 else 2594 ImpTakeDescriptionStr(STR_DragMethMirrorFree,rStr); 2595 2596 if (getSdrDragView().IsDragWithCopy()) 2597 rStr+=ImpGetResStr(STR_EditWithCopy); 2598 } 2599 2600 bool SdrDragMirror::BeginSdrDrag() 2601 { 2602 SdrHdl* pH1=GetHdlList().GetHdl(HDL_REF1); 2603 SdrHdl* pH2=GetHdlList().GetHdl(HDL_REF2); 2604 2605 if (pH1!=NULL && pH2!=NULL) 2606 { 2607 DragStat().Ref1()=pH1->GetPos(); 2608 DragStat().Ref2()=pH2->GetPos(); 2609 Ref1()=pH1->GetPos(); 2610 Ref2()=pH2->GetPos(); 2611 aDif=pH2->GetPos()-pH1->GetPos(); 2612 bool b90=(aDif.X()==0) || aDif.Y()==0; 2613 bool b45=b90 || (Abs(aDif.X())==Abs(aDif.Y())); 2614 nWink=NormAngle360(GetAngle(aDif)); 2615 2616 if (!getSdrDragView().IsMirrorAllowed(false,false) && !b45) 2617 return false; // freier Achsenwinkel nicht erlaubt 2618 2619 if (!getSdrDragView().IsMirrorAllowed(true,false) && !b90) 2620 return false; // 45deg auch nicht erlaubt 2621 2622 bSide0=ImpCheckSide(DragStat().GetStart()); 2623 Show(); 2624 return true; 2625 } 2626 else 2627 { 2628 DBG_ERROR("SdrDragMirror::BeginSdrDrag(): Spiegelachse nicht gefunden"); 2629 return false; 2630 } 2631 } 2632 2633 basegfx::B2DHomMatrix SdrDragMirror::getCurrentTransformation() 2634 { 2635 basegfx::B2DHomMatrix aRetval; 2636 2637 if (bMirrored) 2638 { 2639 const double fDeltaX(DragStat().GetRef2().X() - DragStat().GetRef1().X()); 2640 const double fDeltaY(DragStat().GetRef2().Y() - DragStat().GetRef1().Y()); 2641 const double fRotation(atan2(fDeltaY, fDeltaX)); 2642 2643 aRetval = basegfx::tools::createTranslateB2DHomMatrix(-DragStat().GetRef1().X(), -DragStat().GetRef1().Y()); 2644 aRetval.rotate(-fRotation); 2645 aRetval.scale(1.0, -1.0); 2646 aRetval.rotate(fRotation); 2647 aRetval.translate(DragStat().GetRef1().X(), DragStat().GetRef1().Y()); 2648 } 2649 2650 return aRetval; 2651 } 2652 2653 void SdrDragMirror::MoveSdrDrag(const Point& rPnt) 2654 { 2655 if (DragStat().CheckMinMoved(rPnt)) 2656 { 2657 bool bNeuSide=ImpCheckSide(rPnt); 2658 bool bNeuMirr=bSide0!=bNeuSide; 2659 2660 if (bMirrored!=bNeuMirr) 2661 { 2662 Hide(); 2663 bMirrored=bNeuMirr; 2664 DragStat().NextMove(rPnt); 2665 Show(); 2666 } 2667 } 2668 } 2669 2670 bool SdrDragMirror::EndSdrDrag(bool bCopy) 2671 { 2672 Hide(); 2673 2674 if (bMirrored) 2675 { 2676 getSdrDragView().MirrorMarkedObj(DragStat().GetRef1(),DragStat().GetRef2(),bCopy); 2677 } 2678 2679 return true; 2680 } 2681 2682 Pointer SdrDragMirror::GetSdrDragPointer() const 2683 { 2684 return Pointer(POINTER_MIRROR); 2685 } 2686 2687 //////////////////////////////////////////////////////////////////////////////////////////////////// 2688 2689 TYPEINIT1(SdrDragGradient, SdrDragMethod); 2690 2691 SdrDragGradient::SdrDragGradient(SdrDragView& rNewView, bool bGrad) 2692 : SdrDragMethod(rNewView), 2693 pIAOHandle(NULL), 2694 bIsGradient(bGrad) 2695 { 2696 } 2697 2698 void SdrDragGradient::TakeSdrDragComment(XubString& rStr) const 2699 { 2700 if(IsGradient()) 2701 ImpTakeDescriptionStr(STR_DragMethGradient, rStr); 2702 else 2703 ImpTakeDescriptionStr(STR_DragMethTransparence, rStr); 2704 } 2705 2706 bool SdrDragGradient::BeginSdrDrag() 2707 { 2708 bool bRetval(false); 2709 2710 pIAOHandle = (SdrHdlGradient*)GetHdlList().GetHdl(IsGradient() ? HDL_GRAD : HDL_TRNS); 2711 2712 if(pIAOHandle) 2713 { 2714 // save old values 2715 DragStat().Ref1() = pIAOHandle->GetPos(); 2716 DragStat().Ref2() = pIAOHandle->Get2ndPos(); 2717 2718 // what was hit? 2719 bool bHit(false); 2720 SdrHdlColor* pColHdl = pIAOHandle->GetColorHdl1(); 2721 2722 // init handling flags 2723 pIAOHandle->SetMoveSingleHandle(false); 2724 pIAOHandle->SetMoveFirstHandle(false); 2725 2726 // test first color handle 2727 if(pColHdl) 2728 { 2729 basegfx::B2DPoint aPosition(DragStat().GetStart().X(), DragStat().GetStart().Y()); 2730 2731 if(pColHdl->getOverlayObjectList().isHitLogic(aPosition)) 2732 { 2733 bHit = true; 2734 pIAOHandle->SetMoveSingleHandle(true); 2735 pIAOHandle->SetMoveFirstHandle(true); 2736 } 2737 } 2738 2739 // test second color handle 2740 pColHdl = pIAOHandle->GetColorHdl2(); 2741 2742 if(!bHit && pColHdl) 2743 { 2744 basegfx::B2DPoint aPosition(DragStat().GetStart().X(), DragStat().GetStart().Y()); 2745 2746 if(pColHdl->getOverlayObjectList().isHitLogic(aPosition)) 2747 { 2748 bHit = true; 2749 pIAOHandle->SetMoveSingleHandle(true); 2750 } 2751 } 2752 2753 // test gradient handle itself 2754 if(!bHit) 2755 { 2756 basegfx::B2DPoint aPosition(DragStat().GetStart().X(), DragStat().GetStart().Y()); 2757 2758 if(pIAOHandle->getOverlayObjectList().isHitLogic(aPosition)) 2759 { 2760 bHit = true; 2761 } 2762 } 2763 2764 // everything up and running :o} 2765 bRetval = bHit; 2766 } 2767 else 2768 { 2769 DBG_ERROR("SdrDragGradient::BeginSdrDrag(): IAOGradient nicht gefunden"); 2770 } 2771 2772 return bRetval; 2773 } 2774 2775 void SdrDragGradient::MoveSdrDrag(const Point& rPnt) 2776 { 2777 if(pIAOHandle && DragStat().CheckMinMoved(rPnt)) 2778 { 2779 DragStat().NextMove(rPnt); 2780 2781 // Do the Move here!!! DragStat().GetStart() 2782 Point aMoveDiff = rPnt - DragStat().GetStart(); 2783 2784 if(pIAOHandle->IsMoveSingleHandle()) 2785 { 2786 if(pIAOHandle->IsMoveFirstHandle()) 2787 { 2788 pIAOHandle->SetPos(DragStat().Ref1() + aMoveDiff); 2789 if(pIAOHandle->GetColorHdl1()) 2790 pIAOHandle->GetColorHdl1()->SetPos(DragStat().Ref1() + aMoveDiff); 2791 } 2792 else 2793 { 2794 pIAOHandle->Set2ndPos(DragStat().Ref2() + aMoveDiff); 2795 if(pIAOHandle->GetColorHdl2()) 2796 pIAOHandle->GetColorHdl2()->SetPos(DragStat().Ref2() + aMoveDiff); 2797 } 2798 } 2799 else 2800 { 2801 pIAOHandle->SetPos(DragStat().Ref1() + aMoveDiff); 2802 pIAOHandle->Set2ndPos(DragStat().Ref2() + aMoveDiff); 2803 2804 if(pIAOHandle->GetColorHdl1()) 2805 pIAOHandle->GetColorHdl1()->SetPos(DragStat().Ref1() + aMoveDiff); 2806 2807 if(pIAOHandle->GetColorHdl2()) 2808 pIAOHandle->GetColorHdl2()->SetPos(DragStat().Ref2() + aMoveDiff); 2809 } 2810 2811 // new state 2812 pIAOHandle->FromIAOToItem(getSdrDragView().GetMarkedObjectList().GetMark(0)->GetMarkedSdrObj(), false, false); 2813 } 2814 } 2815 2816 bool SdrDragGradient::EndSdrDrag(bool /*bCopy*/) 2817 { 2818 // here the result is clear, do something with the values 2819 Ref1() = pIAOHandle->GetPos(); 2820 Ref2() = pIAOHandle->Get2ndPos(); 2821 2822 // new state 2823 pIAOHandle->FromIAOToItem(getSdrDragView().GetMarkedObjectList().GetMark(0)->GetMarkedSdrObj(), true, true); 2824 2825 return true; 2826 } 2827 2828 void SdrDragGradient::CancelSdrDrag() 2829 { 2830 // restore old values 2831 pIAOHandle->SetPos(DragStat().Ref1()); 2832 pIAOHandle->Set2ndPos(DragStat().Ref2()); 2833 2834 if(pIAOHandle->GetColorHdl1()) 2835 pIAOHandle->GetColorHdl1()->SetPos(DragStat().Ref1()); 2836 2837 if(pIAOHandle->GetColorHdl2()) 2838 pIAOHandle->GetColorHdl2()->SetPos(DragStat().Ref2()); 2839 2840 // new state 2841 pIAOHandle->FromIAOToItem(getSdrDragView().GetMarkedObjectList().GetMark(0)->GetMarkedSdrObj(), true, false); 2842 } 2843 2844 Pointer SdrDragGradient::GetSdrDragPointer() const 2845 { 2846 return Pointer(POINTER_REFHAND); 2847 } 2848 2849 //////////////////////////////////////////////////////////////////////////////////////////////////// 2850 2851 TYPEINIT1(SdrDragCrook,SdrDragMethod); 2852 2853 SdrDragCrook::SdrDragCrook(SdrDragView& rNewView) 2854 : SdrDragMethod(rNewView), 2855 aFact(1,1), 2856 bContortionAllowed(false), 2857 bNoContortionAllowed(false), 2858 bContortion(false), 2859 bResizeAllowed(false), 2860 bResize(false), 2861 bRotateAllowed(false), 2862 bRotate(false), 2863 bVertical(false), 2864 bValid(false), 2865 bLft(false), 2866 bRgt(false), 2867 bUpr(false), 2868 bLwr(false), 2869 bAtCenter(false), 2870 nWink(0), 2871 nMarkSize(0), 2872 eMode(SDRCROOK_ROTATE) 2873 { 2874 } 2875 2876 void SdrDragCrook::TakeSdrDragComment(XubString& rStr) const 2877 { 2878 ImpTakeDescriptionStr(!bContortion ? STR_DragMethCrook : STR_DragMethCrookContortion, rStr); 2879 2880 if(bValid) 2881 { 2882 rStr.AppendAscii(" ("); 2883 2884 XubString aStr; 2885 sal_Int32 nVal(nWink); 2886 2887 if(bAtCenter) 2888 nVal *= 2; 2889 2890 nVal = Abs(nVal); 2891 getSdrDragView().GetModel()->TakeWinkStr(nVal, aStr); 2892 rStr += aStr; 2893 rStr += sal_Unicode(')'); 2894 } 2895 2896 if(getSdrDragView().IsDragWithCopy()) 2897 rStr += ImpGetResStr(STR_EditWithCopy); 2898 } 2899 2900 // #96920# These defines parametrise the created raster 2901 // for interactions 2902 #define DRAG_CROOK_RASTER_MINIMUM (4) 2903 #define DRAG_CROOK_RASTER_MAXIMUM (15) 2904 #define DRAG_CROOK_RASTER_DISTANCE (30) 2905 2906 basegfx::B2DPolyPolygon impCreateDragRaster(SdrPageView& rPageView, const Rectangle& rMarkRect) 2907 { 2908 basegfx::B2DPolyPolygon aRetval; 2909 2910 if(rPageView.PageWindowCount()) 2911 { 2912 OutputDevice& rOut = (rPageView.GetPageWindow(0)->GetPaintWindow().GetOutputDevice()); 2913 Rectangle aPixelSize = rOut.LogicToPixel(rMarkRect); 2914 sal_uInt32 nHorDiv(aPixelSize.GetWidth() / DRAG_CROOK_RASTER_DISTANCE); 2915 sal_uInt32 nVerDiv(aPixelSize.GetHeight() / DRAG_CROOK_RASTER_DISTANCE); 2916 2917 if(nHorDiv > DRAG_CROOK_RASTER_MAXIMUM) 2918 nHorDiv = DRAG_CROOK_RASTER_MAXIMUM; 2919 if(nHorDiv < DRAG_CROOK_RASTER_MINIMUM) 2920 nHorDiv = DRAG_CROOK_RASTER_MINIMUM; 2921 2922 if(nVerDiv > DRAG_CROOK_RASTER_MAXIMUM) 2923 nVerDiv = DRAG_CROOK_RASTER_MAXIMUM; 2924 if(nVerDiv < DRAG_CROOK_RASTER_MINIMUM) 2925 nVerDiv = DRAG_CROOK_RASTER_MINIMUM; 2926 2927 const double fXLen(rMarkRect.GetWidth() / (double)nHorDiv); 2928 const double fYLen(rMarkRect.GetHeight() / (double)nVerDiv); 2929 double fYPos(rMarkRect.Top()); 2930 sal_uInt32 a, b; 2931 2932 for(a = 0; a <= nVerDiv; a++) 2933 { 2934 // hor lines 2935 for(b = 0; b < nHorDiv; b++) 2936 { 2937 basegfx::B2DPolygon aHorLineSegment; 2938 2939 const double fNewX(rMarkRect.Left() + (b * fXLen)); 2940 aHorLineSegment.append(basegfx::B2DPoint(fNewX, fYPos)); 2941 aHorLineSegment.appendBezierSegment( 2942 basegfx::B2DPoint(fNewX + (fXLen * (1.0 / 3.0)), fYPos), 2943 basegfx::B2DPoint(fNewX + (fXLen * (2.0 / 3.0)), fYPos), 2944 basegfx::B2DPoint(fNewX + fXLen, fYPos)); 2945 aRetval.append(aHorLineSegment); 2946 } 2947 2948 // increments 2949 fYPos += fYLen; 2950 } 2951 2952 double fXPos(rMarkRect.Left()); 2953 2954 for(a = 0; a <= nHorDiv; a++) 2955 { 2956 // ver lines 2957 for(b = 0; b < nVerDiv; b++) 2958 { 2959 basegfx::B2DPolygon aVerLineSegment; 2960 2961 const double fNewY(rMarkRect.Top() + (b * fYLen)); 2962 aVerLineSegment.append(basegfx::B2DPoint(fXPos, fNewY)); 2963 aVerLineSegment.appendBezierSegment( 2964 basegfx::B2DPoint(fXPos, fNewY + (fYLen * (1.0 / 3.0))), 2965 basegfx::B2DPoint(fXPos, fNewY + (fYLen * (2.0 / 3.0))), 2966 basegfx::B2DPoint(fXPos, fNewY + fYLen)); 2967 aRetval.append(aVerLineSegment); 2968 } 2969 2970 // increments 2971 fXPos += fXLen; 2972 } 2973 } 2974 2975 return aRetval; 2976 } 2977 2978 void SdrDragCrook::createSdrDragEntries() 2979 { 2980 // Add extended frame raster first, so it will be behind objects 2981 if(getSdrDragView().GetSdrPageView()) 2982 { 2983 const basegfx::B2DPolyPolygon aDragRaster(impCreateDragRaster(*getSdrDragView().GetSdrPageView(), GetMarkedRect())); 2984 2985 if(aDragRaster.count()) 2986 { 2987 addSdrDragEntry(new SdrDragEntryPolyPolygon(aDragRaster)); 2988 } 2989 } 2990 2991 // call parent 2992 SdrDragMethod::createSdrDragEntries(); 2993 } 2994 2995 bool SdrDragCrook::BeginSdrDrag() 2996 { 2997 bContortionAllowed=getSdrDragView().IsCrookAllowed(false); 2998 bNoContortionAllowed=getSdrDragView().IsCrookAllowed(true); 2999 bResizeAllowed=getSdrDragView().IsResizeAllowed(false); 3000 bRotateAllowed=getSdrDragView().IsRotateAllowed(false); 3001 3002 if (bContortionAllowed || bNoContortionAllowed) 3003 { 3004 bVertical=(GetDragHdlKind()==HDL_LOWER || GetDragHdlKind()==HDL_UPPER); 3005 aMarkRect=GetMarkedRect(); 3006 aMarkCenter=aMarkRect.Center(); 3007 nMarkSize=bVertical ? (aMarkRect.GetHeight()-1) : (aMarkRect.GetWidth()-1); 3008 aCenter=aMarkCenter; 3009 aStart=DragStat().GetStart(); 3010 Show(); 3011 return true; 3012 } 3013 else 3014 { 3015 return false; 3016 } 3017 } 3018 3019 void SdrDragCrook::_MovAllPoints(basegfx::B2DPolyPolygon& rTarget) 3020 { 3021 SdrPageView* pPV = getSdrDragView().GetSdrPageView(); 3022 3023 if(pPV) 3024 { 3025 XPolyPolygon aTempPolyPoly(rTarget); 3026 3027 if (pPV->HasMarkedObjPageView()) 3028 { 3029 sal_uInt16 nPolyAnz=aTempPolyPoly.Count(); 3030 3031 if (!bContortion && !getSdrDragView().IsNoDragXorPolys()) 3032 { 3033 sal_uInt16 n1st=0,nLast=0; 3034 Point aC(aCenter); 3035 3036 while (n1st<nPolyAnz) 3037 { 3038 nLast=n1st; 3039 while (nLast<nPolyAnz && aTempPolyPoly[nLast].GetPointCount()!=0) nLast++; 3040 Rectangle aBound(aTempPolyPoly[n1st].GetBoundRect()); 3041 sal_uInt16 i; 3042 3043 for (i=n1st+1; i<nLast; i++) 3044 { 3045 aBound.Union(aTempPolyPoly[n1st].GetBoundRect()); 3046 } 3047 3048 Point aCtr0(aBound.Center()); 3049 Point aCtr1(aCtr0); 3050 3051 if (bResize) 3052 { 3053 Fraction aFact1(1,1); 3054 3055 if (bVertical) 3056 { 3057 ResizePoint(aCtr1,aC,aFact1,aFact); 3058 } 3059 else 3060 { 3061 ResizePoint(aCtr1,aC,aFact,aFact1); 3062 } 3063 } 3064 3065 bool bRotOk=false; 3066 double nSin=0,nCos=0; 3067 3068 if (aRad.X()!=0 && aRad.Y()!=0) 3069 { 3070 bRotOk=bRotate; 3071 3072 switch (eMode) 3073 { 3074 case SDRCROOK_ROTATE : CrookRotateXPoint (aCtr1,NULL,NULL,aC,aRad,nSin,nCos,bVertical); break; 3075 case SDRCROOK_SLANT : CrookSlantXPoint (aCtr1,NULL,NULL,aC,aRad,nSin,nCos,bVertical); break; 3076 case SDRCROOK_STRETCH: CrookStretchXPoint(aCtr1,NULL,NULL,aC,aRad,nSin,nCos,bVertical,aMarkRect); break; 3077 } // switch 3078 } 3079 3080 aCtr1-=aCtr0; 3081 3082 for (i=n1st; i<nLast; i++) 3083 { 3084 if (bRotOk) 3085 { 3086 RotateXPoly(aTempPolyPoly[i],aCtr0,nSin,nCos); 3087 } 3088 3089 aTempPolyPoly[i].Move(aCtr1.X(),aCtr1.Y()); 3090 } 3091 3092 n1st=nLast+1; 3093 } 3094 } 3095 else 3096 { 3097 sal_uInt16 i,j; 3098 3099 for (j=0; j<nPolyAnz; j++) 3100 { 3101 XPolygon& aPol=aTempPolyPoly[j]; 3102 sal_uInt16 nPtAnz=aPol.GetPointCount(); 3103 i=0; 3104 3105 while (i<nPtAnz) 3106 { 3107 Point* pPnt=&aPol[i]; 3108 Point* pC1=NULL; 3109 Point* pC2=NULL; 3110 3111 if (i+1<nPtAnz && aPol.IsControl(i)) 3112 { // Kontrollpunkt links 3113 pC1=pPnt; 3114 i++; 3115 pPnt=&aPol[i]; 3116 } 3117 3118 i++; 3119 3120 if (i<nPtAnz && aPol.IsControl(i)) 3121 { // Kontrollpunkt rechts 3122 pC2=&aPol[i]; 3123 i++; 3124 } 3125 3126 _MovCrookPoint(*pPnt,pC1,pC2); 3127 } 3128 } 3129 } 3130 } 3131 3132 rTarget = aTempPolyPoly.getB2DPolyPolygon(); 3133 } 3134 } 3135 3136 void SdrDragCrook::_MovCrookPoint(Point& rPnt, Point* pC1, Point* pC2) 3137 { 3138 bool bVert=bVertical; 3139 bool bC1=pC1!=NULL; 3140 bool bC2=pC2!=NULL; 3141 Point aC(aCenter); 3142 3143 if (bResize) 3144 { 3145 Fraction aFact1(1,1); 3146 3147 if (bVert) 3148 { 3149 ResizePoint(rPnt,aC,aFact1,aFact); 3150 3151 if (bC1) 3152 ResizePoint(*pC1,aC,aFact1,aFact); 3153 3154 if (bC2) 3155 ResizePoint(*pC2,aC,aFact1,aFact); 3156 } 3157 else 3158 { 3159 ResizePoint(rPnt,aC,aFact,aFact1); 3160 3161 if (bC1) 3162 ResizePoint(*pC1,aC,aFact,aFact1); 3163 3164 if (bC2) 3165 ResizePoint(*pC2,aC,aFact,aFact1); 3166 } 3167 } 3168 3169 if (aRad.X()!=0 && aRad.Y()!=0) 3170 { 3171 double nSin,nCos; 3172 3173 switch (eMode) 3174 { 3175 case SDRCROOK_ROTATE : CrookRotateXPoint (rPnt,pC1,pC2,aC,aRad,nSin,nCos,bVert); break; 3176 case SDRCROOK_SLANT : CrookSlantXPoint (rPnt,pC1,pC2,aC,aRad,nSin,nCos,bVert); break; 3177 case SDRCROOK_STRETCH: CrookStretchXPoint(rPnt,pC1,pC2,aC,aRad,nSin,nCos,bVert,aMarkRect); break; 3178 } // switch 3179 } 3180 } 3181 3182 void SdrDragCrook::MoveSdrDrag(const Point& rPnt) 3183 { 3184 if (DragStat().CheckMinMoved(rPnt)) 3185 { 3186 Point aPnt(rPnt); 3187 bool bNeuMoveOnly=getSdrDragView().IsMoveOnlyDragging(); 3188 bAtCenter=false; 3189 SdrCrookMode eNeuMode=getSdrDragView().GetCrookMode(); 3190 bool bNeuContortion=!bNeuMoveOnly && ((bContortionAllowed && !getSdrDragView().IsCrookNoContortion()) || !bNoContortionAllowed); 3191 bResize=!getSdrDragView().IsOrtho() && bResizeAllowed && !bNeuMoveOnly; 3192 bool bNeuRotate=bRotateAllowed && !bNeuContortion && !bNeuMoveOnly && eNeuMode==SDRCROOK_ROTATE; 3193 long nSA=0; 3194 3195 if (nSA==0) 3196 aPnt=GetSnapPos(aPnt); 3197 3198 Point aNeuCenter(aMarkCenter.X(),aStart.Y()); 3199 3200 if (bVertical) 3201 { 3202 aNeuCenter.X()=aStart.X(); 3203 aNeuCenter.Y()=aMarkCenter.Y(); 3204 } 3205 3206 if (!getSdrDragView().IsCrookAtCenter()) 3207 { 3208 switch (GetDragHdlKind()) 3209 { 3210 case HDL_UPLFT: aNeuCenter.X()=aMarkRect.Right(); bLft=true; break; 3211 case HDL_UPPER: aNeuCenter.Y()=aMarkRect.Bottom(); bUpr=true; break; 3212 case HDL_UPRGT: aNeuCenter.X()=aMarkRect.Left(); bRgt=true; break; 3213 case HDL_LEFT : aNeuCenter.X()=aMarkRect.Right(); bLft=true; break; 3214 case HDL_RIGHT: aNeuCenter.X()=aMarkRect.Left(); bRgt=true; break; 3215 case HDL_LWLFT: aNeuCenter.X()=aMarkRect.Right(); bLft=true; break; 3216 case HDL_LOWER: aNeuCenter.Y()=aMarkRect.Top(); bLwr=true; break; 3217 case HDL_LWRGT: aNeuCenter.X()=aMarkRect.Left(); bRgt=true; break; 3218 default: bAtCenter=true; 3219 } 3220 } 3221 else 3222 bAtCenter=true; 3223 3224 Fraction aNeuFact(1,1); 3225 long dx1=aPnt.X()-aNeuCenter.X(); 3226 long dy1=aPnt.Y()-aNeuCenter.Y(); 3227 bValid=bVertical ? dx1!=0 : dy1!=0; 3228 3229 if (bValid) 3230 { 3231 if (bVertical) 3232 bValid=Abs(dx1)*100>Abs(dy1); 3233 else 3234 bValid=Abs(dy1)*100>Abs(dx1); 3235 } 3236 3237 long nNeuRad=0; 3238 nWink=0; 3239 3240 if (bValid) 3241 { 3242 double a=0; // Steigung des Radius 3243 long nPntWink=0; 3244 3245 if (bVertical) 3246 { 3247 a=((double)dy1)/((double)dx1); // Steigung des Radius 3248 nNeuRad=((long)(dy1*a)+dx1) /2; 3249 aNeuCenter.X()+=nNeuRad; 3250 nPntWink=GetAngle(aPnt-aNeuCenter); 3251 } 3252 else 3253 { 3254 a=((double)dx1)/((double)dy1); // Steigung des Radius 3255 nNeuRad=((long)(dx1*a)+dy1) /2; 3256 aNeuCenter.Y()+=nNeuRad; 3257 nPntWink=GetAngle(aPnt-aNeuCenter)-9000; 3258 } 3259 3260 if (!bAtCenter) 3261 { 3262 if (nNeuRad<0) 3263 { 3264 if (bRgt) nPntWink+=18000; 3265 if (bLft) nPntWink=18000-nPntWink; 3266 if (bLwr) nPntWink=-nPntWink; 3267 } 3268 else 3269 { 3270 if (bRgt) nPntWink=-nPntWink; 3271 if (bUpr) nPntWink=18000-nPntWink; 3272 if (bLwr) nPntWink+=18000; 3273 } 3274 3275 nPntWink=NormAngle360(nPntWink); 3276 } 3277 else 3278 { 3279 if (nNeuRad<0) nPntWink+=18000; 3280 if (bVertical) nPntWink=18000-nPntWink; 3281 nPntWink=NormAngle180(nPntWink); 3282 nPntWink=Abs(nPntWink); 3283 } 3284 3285 double nUmfang=2*Abs(nNeuRad)*nPi; 3286 3287 if (bResize) 3288 { 3289 if (nSA!=0) 3290 { // Winkelfang 3291 long nWink0=nPntWink; 3292 nPntWink+=nSA/2; 3293 nPntWink/=nSA; 3294 nPntWink*=nSA; 3295 BigInt a2(nNeuRad); 3296 a2*=BigInt(nWink); 3297 a2/=BigInt(nWink0); 3298 nNeuRad=long(a2); 3299 3300 if (bVertical) 3301 aNeuCenter.X()=aStart.X()+nNeuRad; 3302 else 3303 aNeuCenter.Y()=aStart.Y()+nNeuRad; 3304 } 3305 3306 long nMul=(long)(nUmfang*NormAngle360(nPntWink)/36000); 3307 3308 if (bAtCenter) 3309 nMul*=2; 3310 3311 aNeuFact=Fraction(nMul,nMarkSize); 3312 nWink=nPntWink; 3313 } 3314 else 3315 { 3316 nWink=(long)((nMarkSize*360/nUmfang)*100)/2; 3317 3318 if (nWink==0) 3319 bValid=false; 3320 3321 if (bValid && nSA!=0) 3322 { // Winkelfang 3323 long nWink0=nWink; 3324 nWink+=nSA/2; 3325 nWink/=nSA; 3326 nWink*=nSA; 3327 BigInt a2(nNeuRad); 3328 a2*=BigInt(nWink); 3329 a2/=BigInt(nWink0); 3330 nNeuRad=long(a2); 3331 3332 if (bVertical) 3333 aNeuCenter.X()=aStart.X()+nNeuRad; 3334 else 3335 aNeuCenter.Y()=aStart.Y()+nNeuRad; 3336 } 3337 } 3338 } 3339 3340 if (nWink==0 || nNeuRad==0) 3341 bValid=false; 3342 3343 if (!bValid) 3344 nNeuRad=0; 3345 3346 if (!bValid && bResize) 3347 { 3348 long nMul=bVertical ? dy1 : dx1; 3349 3350 if (bLft || bUpr) 3351 nMul=-nMul; 3352 3353 long nDiv=nMarkSize; 3354 3355 if (bAtCenter) 3356 { 3357 nMul*=2; 3358 nMul=Abs(nMul); 3359 } 3360 3361 aNeuFact=Fraction(nMul,nDiv); 3362 } 3363 3364 if (aNeuCenter!=aCenter || bNeuContortion!=bContortion || aNeuFact!=aFact || 3365 bNeuMoveOnly != getMoveOnly() || bNeuRotate!=bRotate || eNeuMode!=eMode) 3366 { 3367 Hide(); 3368 setMoveOnly(bNeuMoveOnly); 3369 bRotate=bNeuRotate; 3370 eMode=eNeuMode; 3371 bContortion=bNeuContortion; 3372 aCenter=aNeuCenter; 3373 aFact=aNeuFact; 3374 aRad=Point(nNeuRad,nNeuRad); 3375 bResize=aFact!=Fraction(1,1) && aFact.GetDenominator()!=0 && aFact.IsValid(); 3376 DragStat().NextMove(aPnt); 3377 Show(); 3378 } 3379 } 3380 } 3381 3382 void SdrDragCrook::applyCurrentTransformationToSdrObject(SdrObject& rTarget) 3383 { 3384 const bool bDoResize(aFact!=Fraction(1,1)); 3385 const bool bDoCrook(aCenter!=aMarkCenter && aRad.X()!=0 && aRad.Y()!=0); 3386 3387 if (bDoCrook || bDoResize) 3388 { 3389 if (bDoResize) 3390 { 3391 Fraction aFact1(1,1); 3392 3393 if (bContortion) 3394 { 3395 if (bVertical) 3396 { 3397 rTarget.Resize(aCenter,aFact1,aFact); 3398 } 3399 else 3400 { 3401 rTarget.Resize(aCenter,aFact,aFact1); 3402 } 3403 } 3404 else 3405 { 3406 Point aCtr0(rTarget.GetSnapRect().Center()); 3407 Point aCtr1(aCtr0); 3408 3409 if (bVertical) 3410 { 3411 ResizePoint(aCtr1,aCenter,aFact1,aFact); 3412 } 3413 else 3414 { 3415 ResizePoint(aCtr1,aCenter,aFact,aFact1); 3416 } 3417 3418 Size aSiz(aCtr1.X()-aCtr0.X(),aCtr1.Y()-aCtr0.Y()); 3419 3420 rTarget.Move(aSiz); 3421 } 3422 } 3423 3424 if (bDoCrook) 3425 { 3426 const Rectangle aLocalMarkRect(getSdrDragView().GetMarkedObjRect()); 3427 const bool bLocalRotate(!bContortion && eMode == SDRCROOK_ROTATE && getSdrDragView().IsRotateAllowed(false)); 3428 3429 getSdrDragView().ImpCrookObj(&rTarget,aCenter,aRad,eMode,bVertical,!bContortion,bLocalRotate,aLocalMarkRect); 3430 } 3431 } 3432 } 3433 3434 void SdrDragCrook::applyCurrentTransformationToPolyPolygon(basegfx::B2DPolyPolygon& rTarget) 3435 { 3436 // use helper derived from old stuff 3437 _MovAllPoints(rTarget); 3438 } 3439 3440 bool SdrDragCrook::EndSdrDrag(bool bCopy) 3441 { 3442 Hide(); 3443 3444 if (bResize && aFact==Fraction(1,1)) 3445 bResize=false; 3446 3447 const bool bUndo = getSdrDragView().IsUndoEnabled(); 3448 3449 bool bDoCrook=aCenter!=aMarkCenter && aRad.X()!=0 && aRad.Y()!=0; 3450 3451 if (bDoCrook || bResize) 3452 { 3453 if (bResize && bUndo) 3454 { 3455 XubString aStr; 3456 ImpTakeDescriptionStr(!bContortion?STR_EditCrook:STR_EditCrookContortion,aStr); 3457 3458 if (bCopy) 3459 aStr+=ImpGetResStr(STR_EditWithCopy); 3460 3461 getSdrDragView().BegUndo(aStr); 3462 } 3463 3464 if (bResize) 3465 { 3466 Fraction aFact1(1,1); 3467 3468 if (bContortion) 3469 { 3470 if (bVertical) 3471 getSdrDragView().ResizeMarkedObj(aCenter,aFact1,aFact,bCopy); 3472 else 3473 getSdrDragView().ResizeMarkedObj(aCenter,aFact,aFact1,bCopy); 3474 } 3475 else 3476 { 3477 if (bCopy) 3478 getSdrDragView().CopyMarkedObj(); 3479 3480 sal_uLong nMarkAnz=getSdrDragView().GetMarkedObjectList().GetMarkCount(); 3481 3482 for (sal_uLong nm=0; nm<nMarkAnz; nm++) 3483 { 3484 SdrMark* pM=getSdrDragView().GetMarkedObjectList().GetMark(nm); 3485 SdrObject* pO=pM->GetMarkedSdrObj(); 3486 Point aCtr0(pO->GetSnapRect().Center()); 3487 Point aCtr1(aCtr0); 3488 3489 if (bVertical) 3490 ResizePoint(aCtr1,aCenter,aFact1,aFact); 3491 else 3492 ResizePoint(aCtr1,aCenter,aFact,aFact1); 3493 3494 Size aSiz(aCtr1.X()-aCtr0.X(),aCtr1.Y()-aCtr0.Y()); 3495 if( bUndo ) 3496 AddUndo(getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoMoveObject(*pO,aSiz)); 3497 pO->Move(aSiz); 3498 } 3499 } 3500 3501 bCopy=false; 3502 } 3503 3504 if (bDoCrook) 3505 { 3506 getSdrDragView().CrookMarkedObj(aCenter,aRad,eMode,bVertical,!bContortion,bCopy); 3507 getSdrDragView().SetLastCrookCenter(aCenter); 3508 } 3509 3510 if (bResize && bUndo) 3511 getSdrDragView().EndUndo(); 3512 3513 return true; 3514 } 3515 3516 return false; 3517 } 3518 3519 Pointer SdrDragCrook::GetSdrDragPointer() const 3520 { 3521 return Pointer(POINTER_CROOK); 3522 } 3523 3524 //////////////////////////////////////////////////////////////////////////////////////////////////// 3525 3526 TYPEINIT1(SdrDragDistort,SdrDragMethod); 3527 3528 SdrDragDistort::SdrDragDistort(SdrDragView& rNewView) 3529 : SdrDragMethod(rNewView), 3530 nPolyPt(0), 3531 bContortionAllowed(false), 3532 bNoContortionAllowed(false), 3533 bContortion(false) 3534 { 3535 } 3536 3537 void SdrDragDistort::TakeSdrDragComment(XubString& rStr) const 3538 { 3539 ImpTakeDescriptionStr(STR_DragMethDistort, rStr); 3540 3541 XubString aStr; 3542 3543 rStr.AppendAscii(" (x="); 3544 getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDX(), aStr); 3545 rStr += aStr; 3546 rStr.AppendAscii(" y="); 3547 getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDY(), aStr); 3548 rStr += aStr; 3549 rStr += sal_Unicode(')'); 3550 3551 if(getSdrDragView().IsDragWithCopy()) 3552 rStr += ImpGetResStr(STR_EditWithCopy); 3553 } 3554 3555 void SdrDragDistort::createSdrDragEntries() 3556 { 3557 // Add extended frame raster first, so it will be behind objects 3558 if(getSdrDragView().GetSdrPageView()) 3559 { 3560 const basegfx::B2DPolyPolygon aDragRaster(impCreateDragRaster(*getSdrDragView().GetSdrPageView(), GetMarkedRect())); 3561 3562 if(aDragRaster.count()) 3563 { 3564 addSdrDragEntry(new SdrDragEntryPolyPolygon(aDragRaster)); 3565 } 3566 } 3567 3568 // call parent 3569 SdrDragMethod::createSdrDragEntries(); 3570 } 3571 3572 bool SdrDragDistort::BeginSdrDrag() 3573 { 3574 bContortionAllowed=getSdrDragView().IsDistortAllowed(false); 3575 bNoContortionAllowed=getSdrDragView().IsDistortAllowed(true); 3576 3577 if (bContortionAllowed || bNoContortionAllowed) 3578 { 3579 SdrHdlKind eKind=GetDragHdlKind(); 3580 nPolyPt=0xFFFF; 3581 3582 if (eKind==HDL_UPLFT) nPolyPt=0; 3583 if (eKind==HDL_UPRGT) nPolyPt=1; 3584 if (eKind==HDL_LWRGT) nPolyPt=2; 3585 if (eKind==HDL_LWLFT) nPolyPt=3; 3586 if (nPolyPt>3) return false; 3587 3588 aMarkRect=GetMarkedRect(); 3589 aDistortedRect=XPolygon(aMarkRect); 3590 Show(); 3591 return true; 3592 } 3593 else 3594 { 3595 return false; 3596 } 3597 } 3598 3599 void SdrDragDistort::_MovAllPoints(basegfx::B2DPolyPolygon& rTarget) 3600 { 3601 if (bContortion) 3602 { 3603 SdrPageView* pPV = getSdrDragView().GetSdrPageView(); 3604 3605 if(pPV) 3606 { 3607 if (pPV->HasMarkedObjPageView()) 3608 { 3609 basegfx::B2DPolyPolygon aDragPolygon(rTarget); 3610 const basegfx::B2DRange aOriginalRange(aMarkRect.Left(), aMarkRect.Top(), aMarkRect.Right(), aMarkRect.Bottom()); 3611 const basegfx::B2DPoint aTopLeft(aDistortedRect[0].X(), aDistortedRect[0].Y()); 3612 const basegfx::B2DPoint aTopRight(aDistortedRect[1].X(), aDistortedRect[1].Y()); 3613 const basegfx::B2DPoint aBottomLeft(aDistortedRect[3].X(), aDistortedRect[3].Y()); 3614 const basegfx::B2DPoint aBottomRight(aDistortedRect[2].X(), aDistortedRect[2].Y()); 3615 3616 aDragPolygon = basegfx::tools::distort(aDragPolygon, aOriginalRange, aTopLeft, aTopRight, aBottomLeft, aBottomRight); 3617 rTarget = aDragPolygon; 3618 } 3619 } 3620 } 3621 } 3622 3623 void SdrDragDistort::MoveSdrDrag(const Point& rPnt) 3624 { 3625 if (DragStat().CheckMinMoved(rPnt)) 3626 { 3627 Point aPnt(GetSnapPos(rPnt)); 3628 3629 if (getSdrDragView().IsOrtho()) 3630 OrthoDistance8(DragStat().GetStart(),aPnt,getSdrDragView().IsBigOrtho()); 3631 3632 bool bNeuContortion=(bContortionAllowed && !getSdrDragView().IsCrookNoContortion()) || !bNoContortionAllowed; 3633 3634 if (bNeuContortion!=bContortion || aDistortedRect[nPolyPt]!=aPnt) 3635 { 3636 Hide(); 3637 aDistortedRect[nPolyPt]=aPnt; 3638 bContortion=bNeuContortion; 3639 DragStat().NextMove(aPnt); 3640 Show(); 3641 } 3642 } 3643 } 3644 3645 bool SdrDragDistort::EndSdrDrag(bool bCopy) 3646 { 3647 Hide(); 3648 bool bDoDistort=DragStat().GetDX()!=0 || DragStat().GetDY()!=0; 3649 3650 if (bDoDistort) 3651 { 3652 getSdrDragView().DistortMarkedObj(aMarkRect,aDistortedRect,!bContortion,bCopy); 3653 return true; 3654 } 3655 3656 return false; 3657 } 3658 3659 Pointer SdrDragDistort::GetSdrDragPointer() const 3660 { 3661 return Pointer(POINTER_REFHAND); 3662 } 3663 3664 void SdrDragDistort::applyCurrentTransformationToSdrObject(SdrObject& rTarget) 3665 { 3666 const bool bDoDistort(DragStat().GetDX()!=0 || DragStat().GetDY()!=0); 3667 3668 if (bDoDistort) 3669 { 3670 getSdrDragView().ImpDistortObj(&rTarget, aMarkRect, aDistortedRect, !bContortion); 3671 } 3672 } 3673 3674 void SdrDragDistort::applyCurrentTransformationToPolyPolygon(basegfx::B2DPolyPolygon& rTarget) 3675 { 3676 // use helper derived from old stuff 3677 _MovAllPoints(rTarget); 3678 } 3679 3680 //////////////////////////////////////////////////////////////////////////////////////////////////// 3681 3682 TYPEINIT1(SdrDragCrop,SdrDragResize); 3683 3684 SdrDragCrop::SdrDragCrop(SdrDragView& rNewView) 3685 : SdrDragObjOwn(rNewView) 3686 { 3687 // switch off solid dragging for crop; it just makes no sense since showing 3688 // a 50% transparent object above the original will not be visible 3689 setSolidDraggingActive(false); 3690 } 3691 3692 void SdrDragCrop::TakeSdrDragComment(XubString& rStr) const 3693 { 3694 ImpTakeDescriptionStr(STR_DragMethCrop, rStr); 3695 3696 XubString aStr; 3697 3698 rStr.AppendAscii(" (x="); 3699 getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDX(), aStr); 3700 rStr += aStr; 3701 rStr.AppendAscii(" y="); 3702 getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDY(), aStr); 3703 rStr += aStr; 3704 rStr += sal_Unicode(')'); 3705 3706 if(getSdrDragView().IsDragWithCopy()) 3707 rStr += ImpGetResStr(STR_EditWithCopy); 3708 } 3709 3710 bool SdrDragCrop::BeginSdrDrag() 3711 { 3712 // call parent 3713 bool bRetval(SdrDragObjOwn::BeginSdrDrag()); 3714 3715 if(!GetDragHdl()) 3716 { 3717 // we need the DragHdl, break if not there 3718 bRetval = false; 3719 } 3720 3721 return bRetval; 3722 } 3723 3724 bool SdrDragCrop::EndSdrDrag(bool /*bCopy*/) 3725 { 3726 Hide(); 3727 3728 if( DragStat().GetDX()==0 && DragStat().GetDY()==0 ) 3729 return false; 3730 3731 const SdrMarkList& rMarkList = getSdrDragView().GetMarkedObjectList(); 3732 3733 if( rMarkList.GetMarkCount() != 1 ) 3734 return false; 3735 3736 SdrGrafObj* pObj = dynamic_cast<SdrGrafObj*>( rMarkList.GetMark( 0 )->GetMarkedSdrObj() ); 3737 3738 if( !pObj || (pObj->GetGraphicType() == GRAPHIC_NONE) || (pObj->GetGraphicType() == GRAPHIC_DEFAULT) ) 3739 return false; 3740 3741 const GraphicObject& rGraphicObject = pObj->GetGraphicObject(); 3742 const MapMode aMapMode100thmm(MAP_100TH_MM); 3743 Size aGraphicSize(rGraphicObject.GetPrefSize()); 3744 3745 if( MAP_PIXEL == rGraphicObject.GetPrefMapMode().GetMapUnit() ) 3746 aGraphicSize = Application::GetDefaultDevice()->PixelToLogic( aGraphicSize, aMapMode100thmm ); 3747 else 3748 aGraphicSize = Application::GetDefaultDevice()->LogicToLogic( aGraphicSize, rGraphicObject.GetPrefMapMode(), aMapMode100thmm); 3749 3750 if( aGraphicSize.nA == 0 || aGraphicSize.nB == 0 ) 3751 return false; 3752 3753 const SdrGrafCropItem& rOldCrop = (const SdrGrafCropItem&)pObj->GetMergedItem(SDRATTR_GRAFCROP); 3754 3755 const bool bUndo = getSdrDragView().IsUndoEnabled(); 3756 3757 if( bUndo ) 3758 { 3759 String aUndoStr; 3760 ImpTakeDescriptionStr(STR_DragMethCrop, aUndoStr); 3761 3762 getSdrDragView().BegUndo( aUndoStr ); 3763 getSdrDragView().AddUndo( getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoGeoObject(*pObj)); 3764 // also need attr undo, the SdrGrafCropItem will be changed 3765 getSdrDragView().AddUndo( getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoAttrObject(*pObj)); 3766 } 3767 3768 // new part to comute the user's drag activities 3769 // get the original objects transformation 3770 basegfx::B2DHomMatrix aOriginalMatrix; 3771 basegfx::B2DPolyPolygon aPolyPolygon; 3772 bool bShearCorrected(false); 3773 3774 // get transformation from object 3775 pObj->TRGetBaseGeometry(aOriginalMatrix, aPolyPolygon); 3776 3777 { // TTTT correct shear, it comes currently mirrored from TRGetBaseGeometry, can be removed with aw080 3778 basegfx::B2DTuple aScale; 3779 basegfx::B2DTuple aTranslate; 3780 double fRotate(0.0), fShearX(0.0); 3781 3782 aOriginalMatrix.decompose(aScale, aTranslate, fRotate, fShearX); 3783 3784 if(!basegfx::fTools::equalZero(fShearX)) 3785 { 3786 bShearCorrected = true; 3787 aOriginalMatrix = basegfx::tools::createScaleShearXRotateTranslateB2DHomMatrix( 3788 aScale, 3789 -fShearX, 3790 fRotate, 3791 aTranslate); 3792 } 3793 } 3794 3795 // invert it to be able to work on unit coordinates 3796 basegfx::B2DHomMatrix aInverse(aOriginalMatrix); 3797 3798 aInverse.invert(); 3799 3800 // gererate start point of original drag vector in unit coordinates (the 3801 // vis-a-vis of the drag point) 3802 basegfx::B2DPoint aLocalStart(0.0, 0.0); 3803 bool bOnAxis(false); 3804 3805 switch(GetDragHdlKind()) 3806 { 3807 case HDL_UPLFT: aLocalStart.setX(1.0); aLocalStart.setY(1.0); break; 3808 case HDL_UPPER: aLocalStart.setX(0.5); aLocalStart.setY(1.0); bOnAxis = true; break; 3809 case HDL_UPRGT: aLocalStart.setX(0.0); aLocalStart.setY(1.0); break; 3810 case HDL_LEFT : aLocalStart.setX(1.0); aLocalStart.setY(0.5); bOnAxis = true; break; 3811 case HDL_RIGHT: aLocalStart.setX(0.0); aLocalStart.setY(0.5); bOnAxis = true; break; 3812 case HDL_LWLFT: aLocalStart.setX(1.0); aLocalStart.setY(0.0); break; 3813 case HDL_LOWER: aLocalStart.setX(0.5); aLocalStart.setY(0.0); bOnAxis = true; break; 3814 case HDL_LWRGT: aLocalStart.setX(0.0); aLocalStart.setY(0.0); break; 3815 default: break; 3816 } 3817 3818 // create the current drag position in unit coordinates 3819 basegfx::B2DPoint aLocalCurrent(aInverse * basegfx::B2DPoint(DragStat().GetNow().X(), DragStat().GetNow().Y())); 3820 3821 // if one of the edge handles is used, limit to X or Y drag only 3822 if(bOnAxis) 3823 { 3824 if(basegfx::fTools::equal(aLocalStart.getX(), 0.5)) 3825 { 3826 aLocalCurrent.setX(aLocalStart.getX()); 3827 } 3828 else 3829 { 3830 aLocalCurrent.setY(aLocalStart.getY()); 3831 } 3832 } 3833 3834 // create internal change in unit coordinates 3835 basegfx::B2DHomMatrix aDiscreteChangeMatrix; 3836 3837 if(!basegfx::fTools::equal(aLocalCurrent.getX(), aLocalStart.getX())) 3838 { 3839 if(aLocalStart.getX() < 0.5) 3840 { 3841 aDiscreteChangeMatrix.scale(aLocalCurrent.getX(), 1.0); 3842 } 3843 else 3844 { 3845 aDiscreteChangeMatrix.scale(1.0 - aLocalCurrent.getX(), 1.0); 3846 aDiscreteChangeMatrix.translate(aLocalCurrent.getX(), 0.0); 3847 } 3848 } 3849 3850 if(!basegfx::fTools::equal(aLocalCurrent.getY(), aLocalStart.getY())) 3851 { 3852 if(aLocalStart.getY() < 0.5) 3853 { 3854 aDiscreteChangeMatrix.scale(1.0, aLocalCurrent.getY()); 3855 } 3856 else 3857 { 3858 aDiscreteChangeMatrix.scale(1.0, 1.0 - aLocalCurrent.getY()); 3859 aDiscreteChangeMatrix.translate(0.0, aLocalCurrent.getY()); 3860 } 3861 } 3862 3863 // preparematrix to apply to object; evtl. back-correct shear 3864 basegfx::B2DHomMatrix aNewObjectMatrix(aOriginalMatrix * aDiscreteChangeMatrix); 3865 3866 if(bShearCorrected) 3867 { 3868 // TTTT back-correct shear 3869 basegfx::B2DTuple aScale; 3870 basegfx::B2DTuple aTranslate; 3871 double fRotate(0.0), fShearX(0.0); 3872 3873 aNewObjectMatrix.decompose(aScale, aTranslate, fRotate, fShearX); 3874 aNewObjectMatrix = basegfx::tools::createScaleShearXRotateTranslateB2DHomMatrix( 3875 aScale, 3876 -fShearX, 3877 fRotate, 3878 aTranslate); 3879 } 3880 3881 // apply change to object by applying the unit coordinate change followed 3882 // by the original change 3883 pObj->TRSetBaseGeometry(aNewObjectMatrix, aPolyPolygon); 3884 3885 // the following old code uses aOldRect/aNewRect to calculate the crop change for 3886 // the crop item. It implies unrotated objects, so create the unrotated original 3887 // erctangle and the unrotated modified rectangle. Latter can in case of shear and/or 3888 // rotation not be fetched by using 3889 // 3890 //Rectangle aNewRect( pObj->GetLogicRect() ); 3891 // 3892 // as it was done before because the top-left of that new rect *will* have an offset 3893 // caused by the evtl. existing shear and/or rotation, so calculate a unrotated 3894 // rectangle how it would be as a result when appling the unit coordinate change 3895 // to the unrotated original transformation. 3896 basegfx::B2DTuple aScale; 3897 basegfx::B2DTuple aTranslate; 3898 double fRotate, fShearX; 3899 3900 // get access to scale and translate 3901 aOriginalMatrix.decompose(aScale, aTranslate, fRotate, fShearX); 3902 3903 // prepare unsheared/unrotated versions of the old and new transformation 3904 const basegfx::B2DHomMatrix aMatrixOriginalNoShearNoRotate( 3905 basegfx::tools::createScaleTranslateB2DHomMatrix( 3906 basegfx::absolute(aScale), 3907 aTranslate)); 3908 3909 // create the ranges for these 3910 basegfx::B2DRange aRangeOriginalNoShearNoRotate(0.0, 0.0, 1.0, 1.0); 3911 basegfx::B2DRange aRangeNewNoShearNoRotate(0.0, 0.0, 1.0, 1.0); 3912 3913 aRangeOriginalNoShearNoRotate.transform(aMatrixOriginalNoShearNoRotate); 3914 aRangeNewNoShearNoRotate.transform(aMatrixOriginalNoShearNoRotate * aDiscreteChangeMatrix); 3915 3916 // extract the old Rectangle structures 3917 Rectangle aOldRect( 3918 basegfx::fround(aRangeOriginalNoShearNoRotate.getMinX()), 3919 basegfx::fround(aRangeOriginalNoShearNoRotate.getMinY()), 3920 basegfx::fround(aRangeOriginalNoShearNoRotate.getMaxX()), 3921 basegfx::fround(aRangeOriginalNoShearNoRotate.getMaxY())); 3922 Rectangle aNewRect( 3923 basegfx::fround(aRangeNewNoShearNoRotate.getMinX()), 3924 basegfx::fround(aRangeNewNoShearNoRotate.getMinY()), 3925 basegfx::fround(aRangeNewNoShearNoRotate.getMaxX()), 3926 basegfx::fround(aRangeNewNoShearNoRotate.getMaxY())); 3927 3928 // continue with the old original stuff 3929 double fScaleX = ( aGraphicSize.Width() - rOldCrop.GetLeft() - rOldCrop.GetRight() ) / (double)aOldRect.GetWidth(); 3930 double fScaleY = ( aGraphicSize.Height() - rOldCrop.GetTop() - rOldCrop.GetBottom() ) / (double)aOldRect.GetHeight(); 3931 3932 // not needed since the modification is done in unit coordinates, free from shear/rotate and mirror 3933 // // TTTT may be removed or exhanged by other stuff in aw080 3934 // // to correct the never working combination of cropped images and mirroring 3935 // // I have to correct the rectangles the calculation is based on here. In the current 3936 // // core geometry stuff a vertical mirror is expressed as 180 degree rotation. All 3937 // // this can be removed again when aw080 will have cleaned up the old 3938 // // (non-)transformation mess in the core. 3939 // if(18000 == pObj->GetGeoStat().nDrehWink) 3940 // { 3941 // // old notation of vertical mirror, need to correct diffs since both rects 3942 // // are rotated by 180 degrees 3943 // aOldRect = Rectangle(aOldRect.TopLeft() - (aOldRect.BottomRight() - aOldRect.TopLeft()), aOldRect.TopLeft()); 3944 // aNewRect = Rectangle(aNewRect.TopLeft() - (aNewRect.BottomRight() - aNewRect.TopLeft()), aNewRect.TopLeft()); 3945 // } 3946 3947 sal_Int32 nDiffLeft = aNewRect.nLeft - aOldRect.nLeft; 3948 sal_Int32 nDiffTop = aNewRect.nTop - aOldRect.nTop; 3949 sal_Int32 nDiffRight = aNewRect.nRight - aOldRect.nRight; 3950 sal_Int32 nDiffBottom = aNewRect.nBottom - aOldRect.nBottom; 3951 3952 if(pObj->IsMirrored()) 3953 { 3954 // mirrored X or Y, for old stuff, exchange X 3955 // TTTT: check for aw080 3956 sal_Int32 nTmp(nDiffLeft); 3957 nDiffLeft = -nDiffRight; 3958 nDiffRight = -nTmp; 3959 } 3960 3961 sal_Int32 nLeftCrop = static_cast<sal_Int32>( rOldCrop.GetLeft() + nDiffLeft * fScaleX ); 3962 sal_Int32 nTopCrop = static_cast<sal_Int32>( rOldCrop.GetTop() + nDiffTop * fScaleY ); 3963 sal_Int32 nRightCrop = static_cast<sal_Int32>( rOldCrop.GetRight() - nDiffRight * fScaleX ); 3964 sal_Int32 nBottomCrop = static_cast<sal_Int32>( rOldCrop.GetBottom() - nDiffBottom * fScaleY ); 3965 3966 SfxItemPool& rPool = getSdrDragView().GetModel()->GetItemPool(); 3967 SfxItemSet aSet( rPool, SDRATTR_GRAFCROP, SDRATTR_GRAFCROP ); 3968 aSet.Put( SdrGrafCropItem( nLeftCrop, nTopCrop, nRightCrop, nBottomCrop ) ); 3969 getSdrDragView().SetAttributes( aSet, false ); 3970 3971 if( bUndo ) 3972 getSdrDragView().EndUndo(); 3973 3974 return true; 3975 } 3976 3977 Pointer SdrDragCrop::GetSdrDragPointer() const 3978 { 3979 return Pointer(POINTER_CROP); 3980 } 3981 3982 //////////////////////////////////////////////////////////////////////////////////////////////////// 3983 // eof 3984