xref: /aoo42x/main/svx/source/svdraw/svdhdl.cxx (revision 953ac054)
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 <algorithm>
28 
29 #include <svx/svdhdl.hxx>
30 #include <svx/svdpagv.hxx>
31 #include <svx/svdetc.hxx>
32 #include <svx/svdmrkv.hxx>
33 #include <vcl/window.hxx>
34 
35 #include <vcl/virdev.hxx>
36 #include <tools/poly.hxx>
37 #include <vcl/bmpacc.hxx>
38 
39 #include <svx/sxekitm.hxx>
40 #include "svx/svdstr.hrc"
41 #include "svx/svdglob.hxx"
42 
43 #include <svx/svdmodel.hxx>
44 #include "gradtrns.hxx"
45 #include <svx/xflgrit.hxx>
46 #include <svx/svdundo.hxx>
47 #include <svx/dialmgr.hxx>
48 #include <svx/xflftrit.hxx>
49 
50 // #105678#
51 #include <svx/svdopath.hxx>
52 #include <basegfx/vector/b2dvector.hxx>
53 #include <basegfx/polygon/b2dpolygon.hxx>
54 #include <svx/sdr/overlay/overlaymanager.hxx>
55 #include <svx/sdr/overlay/overlayanimatedbitmapex.hxx>
56 #include <svx/sdr/overlay/overlaybitmapex.hxx>
57 #include <svx/sdr/overlay/overlayline.hxx>
58 #include <svx/sdr/overlay/overlaytriangle.hxx>
59 #include <svx/sdr/overlay/overlayrectangle.hxx>
60 #include <svx/sdrpagewindow.hxx>
61 #include <svx/sdrpaintwindow.hxx>
62 #include <vcl/svapp.hxx>
63 #include <svx/sdr/overlay/overlaypolypolygon.hxx>
64 #include <vcl/lazydelete.hxx>
65 
66 #include <basegfx/polygon/b2dpolygontools.hxx>
67 #include <drawinglayer/primitive2d/polypolygonprimitive2d.hxx>
68 #include <svx/sdr/overlay/overlayprimitive2dsequenceobject.hxx>
69 #include <drawinglayer/primitive2d/graphicprimitive2d.hxx>
70 #include <drawinglayer/primitive2d/maskprimitive2d.hxx>
71 #include <drawinglayer/primitive2d/unifiedtransparenceprimitive2d.hxx>
72 #include <drawinglayer/primitive2d/polygonprimitive2d.hxx>
73 
74 ////////////////////////////////////////////////////////////////////////////////////////////////////
75 // #i15222#
76 // Due to the resource problems in Win95/98 with bitmap resources I
77 // will change this handle bitmap providing class. Old version was splitting
78 // and preparing all small handle bitmaps in device bitmap format, now this will
79 // be done on the fly. Thus, there is only the one big bitmap remembered. With
80 // three source bitmaps, this will be 3 system bitmap resources instead of hundreds.
81 // The price for that needs to be evaluated. Maybe we will need another change here
82 // if this is too expensive.
83 class SdrHdlBitmapSet
84 {
85 	// the bitmap holding all infos
86 	BitmapEx					maMarkersBitmap;
87 
88 	// the cropped Bitmaps for reusage
89 	::std::vector< BitmapEx >	maRealMarkers;
90 
91 	// elpers
92 	BitmapEx& impGetOrCreateTargetBitmap(sal_uInt16 nIndex, const Rectangle& rRectangle);
93 
94 public:
95 	SdrHdlBitmapSet(sal_uInt16 nResId);
96 	~SdrHdlBitmapSet();
97 
98 	const BitmapEx& GetBitmapEx(BitmapMarkerKind eKindOfMarker, sal_uInt16 nInd=0);
99 };
100 
101 ////////////////////////////////////////////////////////////////////////////////////////////////////
102 #define KIND_COUNT			(14)
103 #define INDEX_COUNT			(6)
104 #define INDIVIDUAL_COUNT	(6)
105 
106 SdrHdlBitmapSet::SdrHdlBitmapSet(sal_uInt16 nResId)
107 :	maMarkersBitmap(ResId(nResId, *ImpGetResMgr())), // just use resource with alpha channel
108 	// 14 kinds (BitmapMarkerKind) use index [0..5], 6 extra
109 	maRealMarkers((KIND_COUNT * INDEX_COUNT) + INDIVIDUAL_COUNT)
110 {
111 }
112 
113 SdrHdlBitmapSet::~SdrHdlBitmapSet()
114 {
115 }
116 
117 BitmapEx& SdrHdlBitmapSet::impGetOrCreateTargetBitmap(sal_uInt16 nIndex, const Rectangle& rRectangle)
118 {
119 	BitmapEx& rTargetBitmap = maRealMarkers[nIndex];
120 
121 	if(rTargetBitmap.IsEmpty())
122 	{
123 		rTargetBitmap = maMarkersBitmap;
124 		rTargetBitmap.Crop(rRectangle);
125 	}
126 
127 	return rTargetBitmap;
128 }
129 
130 // change getting of bitmap to use the big resource bitmap
131 const BitmapEx& SdrHdlBitmapSet::GetBitmapEx(BitmapMarkerKind eKindOfMarker, sal_uInt16 nInd)
132 {
133 	// fill in size and source position in maMarkersBitmap
134 	const sal_uInt16 nYPos(nInd * 11);
135 
136 	switch(eKindOfMarker)
137 	{
138 		default:
139 		{
140 			DBG_ERROR( "unknown kind of marker" );
141 			// no break here, return Rect_7x7 as default
142 		}
143 		case Rect_7x7:
144 		{
145 			return impGetOrCreateTargetBitmap((0 * INDEX_COUNT) + nInd, Rectangle(Point(0, nYPos), Size(7, 7)));
146 		}
147 
148 		case Rect_9x9:
149 		{
150 			return impGetOrCreateTargetBitmap((1 * INDEX_COUNT) + nInd, Rectangle(Point(7, nYPos), Size(9, 9)));
151 		}
152 
153 		case Rect_11x11:
154 		{
155 			return impGetOrCreateTargetBitmap((2 * INDEX_COUNT) + nInd, Rectangle(Point(16, nYPos), Size(11, 11)));
156 		}
157 
158 		case Rect_13x13:
159 		{
160 			const sal_uInt16 nIndex((3 * INDEX_COUNT) + nInd);
161 
162 			switch(nInd)
163 			{
164 				case 0:
165 				{
166 					return impGetOrCreateTargetBitmap(nIndex, Rectangle(Point(72, 66), Size(13, 13)));
167 				}
168 				case 1:
169 				{
170 					return impGetOrCreateTargetBitmap(nIndex, Rectangle(Point(85, 66), Size(13, 13)));
171 				}
172 				case 2:
173 				{
174 					return impGetOrCreateTargetBitmap(nIndex, Rectangle(Point(72, 79), Size(13, 13)));
175 				}
176 				case 3:
177 				{
178 					return impGetOrCreateTargetBitmap(nIndex, Rectangle(Point(85, 79), Size(13, 13)));
179 				}
180 				case 4:
181 				{
182 					return impGetOrCreateTargetBitmap(nIndex, Rectangle(Point(98, 79), Size(13, 13)));
183 				}
184 				default: // case 5:
185 				{
186 					return impGetOrCreateTargetBitmap(nIndex, Rectangle(Point(98, 66), Size(13, 13)));
187 				}
188 			}
189 		}
190 
191 		case Circ_7x7:
192 		case Customshape_7x7:
193 		{
194 			return impGetOrCreateTargetBitmap((4 * INDEX_COUNT) + nInd, Rectangle(Point(27, nYPos), Size(7, 7)));
195 		}
196 
197 		case Circ_9x9:
198 		case Customshape_9x9:
199 		{
200 			return impGetOrCreateTargetBitmap((5 * INDEX_COUNT) + nInd, Rectangle(Point(34, nYPos), Size(9, 9)));
201 		}
202 
203 		case Circ_11x11:
204 		case Customshape_11x11:
205 		{
206 			return impGetOrCreateTargetBitmap((6 * INDEX_COUNT) + nInd, Rectangle(Point(43, nYPos), Size(11, 11)));
207 		}
208 
209 		case Elli_7x9:
210 		{
211 			return impGetOrCreateTargetBitmap((7 * INDEX_COUNT) + nInd, Rectangle(Point(54, nYPos), Size(7, 9)));
212 		}
213 
214 		case Elli_9x11:
215 		{
216 			return impGetOrCreateTargetBitmap((8 * INDEX_COUNT) + nInd, Rectangle(Point(61, nYPos), Size(9, 11)));
217 		}
218 
219 		case Elli_9x7:
220 		{
221 			return impGetOrCreateTargetBitmap((9 * INDEX_COUNT) + nInd, Rectangle(Point(70, nYPos), Size(9, 7)));
222 		}
223 
224 		case Elli_11x9:
225 		{
226 			return impGetOrCreateTargetBitmap((10 * INDEX_COUNT) + nInd, Rectangle(Point(79, nYPos), Size(11, 9)));
227 		}
228 
229 		case RectPlus_7x7:
230 		{
231 			return impGetOrCreateTargetBitmap((11 * INDEX_COUNT) + nInd, Rectangle(Point(90, nYPos), Size(7, 7)));
232 		}
233 
234 		case RectPlus_9x9:
235 		{
236 			return impGetOrCreateTargetBitmap((12 * INDEX_COUNT) + nInd, Rectangle(Point(97, nYPos), Size(9, 9)));
237 		}
238 
239 		case RectPlus_11x11:
240 		{
241 			return impGetOrCreateTargetBitmap((13 * INDEX_COUNT) + nInd, Rectangle(Point(106, nYPos), Size(11, 11)));
242 		}
243 
244 		case Crosshair:
245 		{
246 			return impGetOrCreateTargetBitmap((KIND_COUNT * INDEX_COUNT) + 0, Rectangle(Point(0, 66), Size(13, 13)));
247 		}
248 
249 		case Crosshair_Unselected:
250 		{
251 			return impGetOrCreateTargetBitmap((KIND_COUNT * INDEX_COUNT) + 1, Rectangle(Point(0, 79), Size(13, 13)));
252 		}
253 
254 		case Glue:
255 		{
256 			return impGetOrCreateTargetBitmap((KIND_COUNT * INDEX_COUNT) + 2, Rectangle(Point(13, 70), Size(11, 11)));
257 		}
258 
259 		case Glue_Unselected:
260 		{
261 			return impGetOrCreateTargetBitmap((KIND_COUNT * INDEX_COUNT) + 3, Rectangle(Point(13, 81), Size(11, 11)));
262 		}
263 
264 		case Anchor: // #101688# AnchorTR for SW
265 		case AnchorTR:
266 		{
267 			return impGetOrCreateTargetBitmap((KIND_COUNT * INDEX_COUNT) + 4, Rectangle(Point(24, 68), Size(24, 24)));
268 		}
269 
270 		// #98388# add AnchorPressed to be able to animate anchor control
271 		case AnchorPressed:
272 		case AnchorPressedTR:
273 		{
274 			return impGetOrCreateTargetBitmap((KIND_COUNT * INDEX_COUNT) + 5, Rectangle(Point(48, 68), Size(24, 24)));
275 		}
276 	}
277 
278 	// cannot happen since all paths return something; return Rect_7x7 as default (see switch)
279 	return maRealMarkers[0];
280 }
281 
282 ////////////////////////////////////////////////////////////////////////////////////////////////////
283 
284 SdrHdlBitmapSet& getSimpleSet()
285 {
286 	static vcl::DeleteOnDeinit< SdrHdlBitmapSet > aSimpleSet(new SdrHdlBitmapSet(SIP_SA_MARKERS));
287 	return *aSimpleSet.get();
288 }
289 
290 SdrHdlBitmapSet& getModernSet()
291 {
292 	static vcl::DeleteOnDeinit< SdrHdlBitmapSet > aModernSet(new SdrHdlBitmapSet(SIP_SA_FINE_MARKERS));
293 	return *aModernSet.get();
294 }
295 
296 SdrHdlBitmapSet& getHighContrastSet()
297 {
298 	static vcl::DeleteOnDeinit< SdrHdlBitmapSet > aHighContrastSet(new SdrHdlBitmapSet(SIP_SA_ACCESSIBILITY_MARKERS));
299 	return *aHighContrastSet.get();
300 }
301 
302 ////////////////////////////////////////////////////////////////////////////////////////////////////
303 
304 SdrHdl::SdrHdl():
305 	pObj(NULL),
306 	pPV(NULL),
307 	pHdlList(NULL),
308 	eKind(HDL_MOVE),
309 	nDrehWink(0),
310 	nObjHdlNum(0),
311 	nPolyNum(0),
312 	nPPntNum(0),
313 	nSourceHdlNum(0),
314 	bSelect(sal_False),
315 	b1PixMore(sal_False),
316 	bPlusHdl(sal_False),
317 	mbMoveOutside(false),
318 	mbMouseOver(false)
319 {
320 }
321 
322 SdrHdl::SdrHdl(const Point& rPnt, SdrHdlKind eNewKind):
323 	pObj(NULL),
324 	pPV(NULL),
325 	pHdlList(NULL),
326 	aPos(rPnt),
327 	eKind(eNewKind),
328 	nDrehWink(0),
329 	nObjHdlNum(0),
330 	nPolyNum(0),
331 	nPPntNum(0),
332 	nSourceHdlNum(0),
333 	bSelect(sal_False),
334 	b1PixMore(sal_False),
335 	bPlusHdl(sal_False),
336 	mbMoveOutside(false),
337 	mbMouseOver(false)
338 {
339 }
340 
341 SdrHdl::~SdrHdl()
342 {
343 	GetRidOfIAObject();
344 }
345 
346 void SdrHdl::Set1PixMore(sal_Bool bJa)
347 {
348 	if(b1PixMore != bJa)
349 	{
350 		b1PixMore = bJa;
351 
352 		// create new display
353 		Touch();
354 	}
355 }
356 
357 void SdrHdl::SetMoveOutside( bool bMoveOutside )
358 {
359 	if(mbMoveOutside != bMoveOutside)
360 	{
361 		mbMoveOutside = bMoveOutside;
362 
363 		// create new display
364 		Touch();
365 	}
366 }
367 
368 void SdrHdl::SetDrehWink(long n)
369 {
370 	if(nDrehWink != n)
371 	{
372 		nDrehWink = n;
373 
374 		// create new display
375 		Touch();
376 	}
377 }
378 
379 void SdrHdl::SetPos(const Point& rPnt)
380 {
381 	if(aPos != rPnt)
382 	{
383 		// remember new position
384 		aPos = rPnt;
385 
386 		// create new display
387 		Touch();
388 	}
389 }
390 
391 void SdrHdl::SetSelected(sal_Bool bJa)
392 {
393 	if(bSelect != bJa)
394 	{
395 		// remember new value
396 		bSelect = bJa;
397 
398 		// create new display
399 		Touch();
400 	}
401 }
402 
403 void SdrHdl::SetHdlList(SdrHdlList* pList)
404 {
405 	if(pHdlList != pList)
406 	{
407 		// remember list
408 		pHdlList = pList;
409 
410 		// now it's possible to create graphic representation
411 		Touch();
412 	}
413 }
414 
415 void SdrHdl::SetObj(SdrObject* pNewObj)
416 {
417 	if(pObj != pNewObj)
418 	{
419 		// remember new object
420 		pObj = pNewObj;
421 
422 		// graphic representation may have changed
423 		Touch();
424 	}
425 }
426 
427 void SdrHdl::Touch()
428 {
429 	// force update of graphic representation
430 	CreateB2dIAObject();
431 }
432 
433 void SdrHdl::GetRidOfIAObject()
434 {
435 	//OLMaIAOGroup.Delete();
436 
437 	// OVERLAYMANAGER
438 	maOverlayGroup.clear();
439 }
440 
441 void SdrHdl::CreateB2dIAObject()
442 {
443 	// first throw away old one
444 	GetRidOfIAObject();
445 
446 	if(pHdlList && pHdlList->GetView() && !pHdlList->GetView()->areMarkHandlesHidden())
447 	{
448 		BitmapColorIndex eColIndex = LightGreen;
449 		BitmapMarkerKind eKindOfMarker = Rect_7x7;
450 
451 		sal_Bool bRot = pHdlList->IsRotateShear();
452 		if(pObj)
453 			eColIndex = (bSelect) ? Cyan : LightCyan;
454 		if(bRot)
455 		{
456 			// Rotation handles in red
457 			if(pObj && bSelect)
458 				eColIndex = Red;
459 			else
460 				eColIndex = LightRed;
461 		}
462 
463 		switch(eKind)
464 		{
465 			case HDL_MOVE:
466 			{
467 				eKindOfMarker = (b1PixMore) ? Rect_9x9 : Rect_7x7;
468 				break;
469 			}
470 			case HDL_UPLFT:
471 			case HDL_UPRGT:
472 			case HDL_LWLFT:
473 			case HDL_LWRGT:
474 			{
475 				// Corner handles
476 				if(bRot)
477 				{
478 					eKindOfMarker = Circ_7x7;
479 				}
480 				else
481 				{
482 					eKindOfMarker = Rect_7x7;
483 				}
484 				break;
485 			}
486 			case HDL_UPPER:
487 			case HDL_LOWER:
488 			{
489 				// Upper/Lower handles
490 				if(bRot)
491 				{
492 					eKindOfMarker = Elli_9x7;
493 				}
494 				else
495 				{
496 					eKindOfMarker = Rect_7x7;
497 				}
498 				break;
499 			}
500 			case HDL_LEFT:
501 			case HDL_RIGHT:
502 			{
503 				// Left/Right handles
504 				if(bRot)
505 				{
506 					eKindOfMarker = Elli_7x9;
507 				}
508 				else
509 				{
510 					eKindOfMarker = Rect_7x7;
511 				}
512 				break;
513 			}
514 			case HDL_POLY:
515 			{
516 				if(bRot)
517 				{
518 					eKindOfMarker = (b1PixMore) ? Circ_9x9 : Circ_7x7;
519 				}
520 				else
521 				{
522 					eKindOfMarker = (b1PixMore) ? Rect_9x9 : Rect_7x7;
523 				}
524 				break;
525 			}
526 			case HDL_BWGT: // weight at poly
527 			{
528 				eKindOfMarker = Circ_7x7;
529 				break;
530 			}
531 			case HDL_CIRC:
532 			{
533 				eKindOfMarker = Rect_11x11;
534 				break;
535 			}
536 			case HDL_REF1:
537 			case HDL_REF2:
538 			{
539 				eKindOfMarker = Crosshair;
540 				break;
541 			}
542 			{
543 				eKindOfMarker = Crosshair_Unselected;
544 				break;
545 			}
546 			case HDL_GLUE:
547 			{
548 				eKindOfMarker = Glue;
549 				break;
550 			}
551 			case HDL_GLUE_UNSEL:
552 			{
553 				eKindOfMarker = Glue_Unselected;
554 				break;
555 			}
556 			case HDL_ANCHOR:
557 			{
558 				eKindOfMarker = Anchor;
559 				break;
560 			}
561 			case HDL_USER:
562 			{
563 				break;
564 			}
565 			// #101688# top right anchor for SW
566 			case HDL_ANCHOR_TR:
567 			{
568 				eKindOfMarker = AnchorTR;
569 				break;
570 			}
571 
572 			// for SJ and the CustomShapeHandles:
573 			case HDL_CUSTOMSHAPE1:
574 			{
575 				eKindOfMarker = (b1PixMore) ? Customshape_9x9 : Customshape_7x7;
576 				eColIndex = Yellow;
577 				break;
578 			}
579 			default:
580 				break;
581 		}
582 
583 		SdrMarkView* pView = pHdlList->GetView();
584 		SdrPageView* pPageView = pView->GetSdrPageView();
585 
586 		if(pPageView)
587 		{
588 			for(sal_uInt32 b(0L); b < pPageView->PageWindowCount(); b++)
589 			{
590 				// const SdrPageViewWinRec& rPageViewWinRec = rPageViewWinList[b];
591 				const SdrPageWindow& rPageWindow = *pPageView->GetPageWindow(b);
592 
593 				if(rPageWindow.GetPaintWindow().OutputToWindow())
594 				{
595 					Point aMoveOutsideOffset(0, 0);
596 
597 					// add offset if necessary
598 					if(pHdlList->IsMoveOutside() || mbMoveOutside)
599 					{
600 						OutputDevice& rOutDev = rPageWindow.GetPaintWindow().GetOutputDevice();
601 						Size aOffset = rOutDev.PixelToLogic(Size(4, 4));
602 
603 						if(eKind == HDL_UPLFT || eKind == HDL_UPPER || eKind == HDL_UPRGT)
604 							aMoveOutsideOffset.Y() -= aOffset.Width();
605 						if(eKind == HDL_LWLFT || eKind == HDL_LOWER || eKind == HDL_LWRGT)
606 							aMoveOutsideOffset.Y() += aOffset.Height();
607 						if(eKind == HDL_UPLFT || eKind == HDL_LEFT  || eKind == HDL_LWLFT)
608 							aMoveOutsideOffset.X() -= aOffset.Width();
609 						if(eKind == HDL_UPRGT || eKind == HDL_RIGHT || eKind == HDL_LWRGT)
610 							aMoveOutsideOffset.X() += aOffset.Height();
611 					}
612 
613 					if(rPageWindow.GetOverlayManager())
614 					{
615 						basegfx::B2DPoint aPosition(aPos.X(), aPos.Y());
616 						::sdr::overlay::OverlayObject* pNewOverlayObject = CreateOverlayObject(
617 							aPosition,
618 							eColIndex,
619 							eKindOfMarker,
620 							aMoveOutsideOffset);
621 
622 						// OVERLAYMANAGER
623 						if(pNewOverlayObject)
624 						{
625 							rPageWindow.GetOverlayManager()->add(*pNewOverlayObject);
626 							maOverlayGroup.append(*pNewOverlayObject);
627 						}
628 					}
629 				}
630 			}
631 		}
632 	}
633 }
634 
635 BitmapMarkerKind SdrHdl::GetNextBigger(BitmapMarkerKind eKnd) const
636 {
637 	BitmapMarkerKind eRetval(eKnd);
638 
639 	switch(eKnd)
640 	{
641 		case Rect_7x7:			eRetval = Rect_9x9;			break;
642 		case Rect_9x9:			eRetval = Rect_11x11;		break;
643 		case Rect_11x11:		eRetval = Rect_13x13;		break;
644 		//case Rect_13x13:		eRetval = ;	break;
645 
646 		case Circ_7x7:			eRetval = Circ_9x9;			break;
647 		case Circ_9x9:			eRetval = Circ_11x11;		break;
648 		//case Circ_11x11:		eRetval = ;	break;
649 
650 		case Customshape_7x7:		eRetval = Customshape_9x9;		break;
651 		case Customshape_9x9:		eRetval = Customshape_11x11;	break;
652 		//case Customshape_11x11:	eRetval = ;	break;
653 
654 		case Elli_7x9:			eRetval = Elli_9x11;		break;
655 		//case Elli_9x11:			eRetval = ;	break;
656 
657 		case Elli_9x7:			eRetval = Elli_11x9;		break;
658 		//case Elli_11x9:			eRetval = ;	break;
659 
660 		case RectPlus_7x7:		eRetval = RectPlus_9x9;		break;
661 		case RectPlus_9x9:		eRetval = RectPlus_11x11;	break;
662 		//case RectPlus_11x11:	eRetval = ;	break;
663 
664 		//case Crosshair:			eRetval = ;	break;
665 		//case Glue:				eRetval = ;	break;
666 
667 		// #98388# let anchor blink with its pressed state
668 		case Anchor:			eRetval = AnchorPressed;	break;
669 
670 		// #101688# same for AnchorTR
671 		case AnchorTR:			eRetval = AnchorPressedTR;	break;
672 		default:
673 			break;
674 	}
675 
676 	return eRetval;
677 }
678 
679 // #101928#
680 BitmapEx SdrHdl::ImpGetBitmapEx(BitmapMarkerKind eKindOfMarker, sal_uInt16 nInd, sal_Bool bFine, sal_Bool bIsHighContrast)
681 {
682 	if(bIsHighContrast)
683 	{
684 		return getHighContrastSet().GetBitmapEx(eKindOfMarker, nInd);
685 	}
686 	else
687 	{
688 		if(bFine)
689 		{
690 			return getModernSet().GetBitmapEx(eKindOfMarker, nInd);
691 		}
692 		else
693 		{
694 			return getSimpleSet().GetBitmapEx(eKindOfMarker, nInd);
695 		}
696 	}
697 }
698 
699 ::sdr::overlay::OverlayObject* SdrHdl::CreateOverlayObject(
700 	const basegfx::B2DPoint& rPos,
701 	BitmapColorIndex eColIndex, BitmapMarkerKind eKindOfMarker, Point aMoveOutsideOffset)
702 {
703 	::sdr::overlay::OverlayObject* pRetval = 0L;
704 	sal_Bool bIsFineHdl(pHdlList->IsFineHdl());
705 	const StyleSettings& rStyleSettings = Application::GetSettings().GetStyleSettings();
706 	sal_Bool bIsHighContrast(rStyleSettings.GetHighContrastMode());
707 
708 	// support bigger sizes
709 	sal_Bool bForceBiggerSize(sal_False);
710 
711 	if(pHdlList->GetHdlSize() > 3)
712 	{
713 		switch(eKindOfMarker)
714 		{
715 			case Anchor:
716 			case AnchorPressed:
717 			case AnchorTR:
718 			case AnchorPressedTR:
719 			{
720 				// #121463# For anchor, do not simply make bigger because of HdlSize,
721 				// do it dependent of IsSelected() which Writer can set in drag mode
722 				if(IsSelected())
723 				{
724 					bForceBiggerSize = sal_True;
725 				}
726 				break;
727 			}
728 			default:
729 			{
730 				bForceBiggerSize = sal_True;
731 				break;
732 			}
733 		}
734 	}
735 
736 	// #101928# ...for high contrast, too.
737 	if(!bForceBiggerSize && bIsHighContrast)
738 	{
739 		// #107925#
740 		// ...but not for anchors, else they will not blink when activated
741 		if(Anchor != eKindOfMarker && AnchorTR != eKindOfMarker)
742 		{
743 			bForceBiggerSize = sal_True;
744 		}
745 	}
746 
747 	if(bForceBiggerSize)
748 	{
749 		eKindOfMarker = GetNextBigger(eKindOfMarker);
750 	}
751 
752 	// #97016# II This handle has the focus, visualize it
753 	if(IsFocusHdl() && pHdlList && pHdlList->GetFocusHdl() == this)
754 	{
755 		// create animated handle
756 		BitmapMarkerKind eNextBigger = GetNextBigger(eKindOfMarker);
757 
758 		if(eNextBigger == eKindOfMarker)
759 		{
760 			// this may happen for the not supported getting-bigger types.
761 			// Choose an alternative here
762 			switch(eKindOfMarker)
763 			{
764 				case Rect_13x13:		eNextBigger = Rect_11x11;	break;
765 				case Circ_11x11:		eNextBigger = Elli_11x9;	break;
766 				case Elli_9x11:			eNextBigger = Elli_11x9;	break;
767 				case Elli_11x9:			eNextBigger = Elli_9x11;	break;
768 				case RectPlus_11x11:	eNextBigger = Rect_13x13;	break;
769 
770 				case Crosshair:
771 					eNextBigger = Crosshair_Unselected;
772 					break;
773 
774 				case Glue:
775 					eNextBigger = Glue_Unselected;
776 					break;
777 				default:
778 					break;
779 			}
780 		}
781 
782 		// create animated hdl
783 		// #101928# use ImpGetBitmapEx(...) now
784 		BitmapEx aBmpEx1 = ImpGetBitmapEx(eKindOfMarker, (sal_uInt16)eColIndex, bIsFineHdl, bIsHighContrast);
785 		BitmapEx aBmpEx2 = ImpGetBitmapEx(eNextBigger, (sal_uInt16)eColIndex, bIsFineHdl, bIsHighContrast);
786 
787 		// #i53216# Use system cursor blink time. Use the unsigned value.
788 		const sal_uInt32 nBlinkTime((sal_uInt32)Application::GetSettings().GetStyleSettings().GetCursorBlinkTime());
789 
790 		if(eKindOfMarker == Anchor || eKindOfMarker == AnchorPressed)
791 		{
792 			// #98388# when anchor is used take upper left as reference point inside the handle
793 			pRetval = new ::sdr::overlay::OverlayAnimatedBitmapEx(rPos, aBmpEx1, aBmpEx2, nBlinkTime);
794 		}
795 		else if(eKindOfMarker == AnchorTR || eKindOfMarker == AnchorPressedTR)
796 		{
797 			// #101688# AnchorTR for SW, take top right as (0,0)
798 			pRetval = new ::sdr::overlay::OverlayAnimatedBitmapEx(rPos, aBmpEx1, aBmpEx2, nBlinkTime,
799 				(sal_uInt16)(aBmpEx1.GetSizePixel().Width() - 1), 0,
800 				(sal_uInt16)(aBmpEx2.GetSizePixel().Width() - 1), 0);
801 		}
802 		else
803 		{
804 			// create centered handle as default
805 			pRetval = new ::sdr::overlay::OverlayAnimatedBitmapEx(rPos, aBmpEx1, aBmpEx2, nBlinkTime,
806 				(sal_uInt16)(aBmpEx1.GetSizePixel().Width() - 1) >> 1,
807 				(sal_uInt16)(aBmpEx1.GetSizePixel().Height() - 1) >> 1,
808 				(sal_uInt16)(aBmpEx2.GetSizePixel().Width() - 1) >> 1,
809 				(sal_uInt16)(aBmpEx2.GetSizePixel().Height() - 1) >> 1);
810 		}
811 	}
812 	else
813 	{
814 		// create normal handle
815 		// #101928# use ImpGetBitmapEx(...) now
816 		BitmapEx aBmpEx = ImpGetBitmapEx(eKindOfMarker, (sal_uInt16)eColIndex, bIsFineHdl, bIsHighContrast);
817 
818 		if(eKindOfMarker == Anchor || eKindOfMarker == AnchorPressed)
819 		{
820 			// #98388# upper left as reference point inside the handle for AnchorPressed, too
821 			pRetval = new ::sdr::overlay::OverlayBitmapEx(rPos, aBmpEx);
822 		}
823 		else if(eKindOfMarker == AnchorTR || eKindOfMarker == AnchorPressedTR)
824 		{
825 			// #101688# AnchorTR for SW, take top right as (0,0)
826 			pRetval = new ::sdr::overlay::OverlayBitmapEx(rPos, aBmpEx,
827 				(sal_uInt16)(aBmpEx.GetSizePixel().Width() - 1), 0);
828 		}
829 		else
830 		{
831 			sal_uInt16 nCenX((sal_uInt16)(aBmpEx.GetSizePixel().Width() - 1L) >> 1);
832 			sal_uInt16 nCenY((sal_uInt16)(aBmpEx.GetSizePixel().Height() - 1L) >> 1);
833 
834 			if(aMoveOutsideOffset.X() > 0)
835 			{
836 				nCenX = 0;
837 			}
838 			else if(aMoveOutsideOffset.X() < 0)
839 			{
840 				nCenX = (sal_uInt16)(aBmpEx.GetSizePixel().Width() - 1);
841 			}
842 
843 			if(aMoveOutsideOffset.Y() > 0)
844 			{
845 				nCenY = 0;
846 			}
847 			else if(aMoveOutsideOffset.Y() < 0)
848 			{
849 				nCenY = (sal_uInt16)(aBmpEx.GetSizePixel().Height() - 1);
850 			}
851 
852 			// create centered handle as default
853 			pRetval = new ::sdr::overlay::OverlayBitmapEx(rPos, aBmpEx, nCenX, nCenY);
854 		}
855 	}
856 
857 	return pRetval;
858 }
859 
860 bool SdrHdl::IsHdlHit(const Point& rPnt) const
861 {
862 	// OVERLAYMANAGER
863 	basegfx::B2DPoint aPosition(rPnt.X(), rPnt.Y());
864 	return maOverlayGroup.isHitLogic(aPosition);
865 }
866 
867 Pointer SdrHdl::GetPointer() const
868 {
869 	PointerStyle ePtr=POINTER_MOVE;
870 	const sal_Bool bSize=eKind>=HDL_UPLFT && eKind<=HDL_LWRGT;
871 	const sal_Bool bRot=pHdlList!=NULL && pHdlList->IsRotateShear();
872 	const sal_Bool bDis=pHdlList!=NULL && pHdlList->IsDistortShear();
873 	if (bSize && pHdlList!=NULL && (bRot || bDis)) {
874 		switch (eKind) {
875 			case HDL_UPLFT: case HDL_UPRGT:
876 			case HDL_LWLFT: case HDL_LWRGT: ePtr=bRot ? POINTER_ROTATE : POINTER_REFHAND; break;
877 			case HDL_LEFT : case HDL_RIGHT: ePtr=POINTER_VSHEAR; break;
878 			case HDL_UPPER: case HDL_LOWER: ePtr=POINTER_HSHEAR; break;
879 			default:
880 				break;
881 		}
882 	} else {
883 		// Fuer Resize von gedrehten Rechtecken die Mauszeiger etwas mitdrehen
884 		if (bSize && nDrehWink!=0) {
885 			long nHdlWink=0;
886 			switch (eKind) {
887 				case HDL_LWRGT: nHdlWink=31500;	break;
888 				case HDL_LOWER: nHdlWink=27000;	break;
889 				case HDL_LWLFT: nHdlWink=22500;	break;
890 				case HDL_LEFT : nHdlWink=18000;	break;
891 				case HDL_UPLFT: nHdlWink=13500;	break;
892 				case HDL_UPPER: nHdlWink=9000;	break;
893 				case HDL_UPRGT: nHdlWink=4500;	break;
894 				case HDL_RIGHT: nHdlWink=0;		break;
895 				default:
896 					break;
897 			}
898 			nHdlWink+=nDrehWink+2249; // und etwas drauf (zum runden)
899 			while (nHdlWink<0) nHdlWink+=36000;
900 			while (nHdlWink>=36000) nHdlWink-=36000;
901 			nHdlWink/=4500;
902 			switch ((sal_uInt8)nHdlWink) {
903 				case 0: ePtr=POINTER_ESIZE;		break;
904 				case 1: ePtr=POINTER_NESIZE;	break;
905 				case 2: ePtr=POINTER_NSIZE;		break;
906 				case 3: ePtr=POINTER_NWSIZE;	break;
907 				case 4: ePtr=POINTER_WSIZE;		break;
908 				case 5: ePtr=POINTER_SWSIZE;	break;
909 				case 6: ePtr=POINTER_SSIZE;		break;
910 				case 7: ePtr=POINTER_SESIZE;	break;
911 			} // switch
912 		} else {
913 			switch (eKind) {
914 				case HDL_UPLFT: ePtr=POINTER_NWSIZE;	break;
915 				case HDL_UPPER: ePtr=POINTER_NSIZE;		break;
916 				case HDL_UPRGT: ePtr=POINTER_NESIZE;	break;
917 				case HDL_LEFT : ePtr=POINTER_WSIZE;		break;
918 				case HDL_RIGHT: ePtr=POINTER_ESIZE;		break;
919 				case HDL_LWLFT: ePtr=POINTER_SWSIZE;	break;
920 				case HDL_LOWER: ePtr=POINTER_SSIZE;		break;
921 				case HDL_LWRGT: ePtr=POINTER_SESIZE;	break;
922 				case HDL_POLY : ePtr=POINTER_MOVEPOINT;	break;
923 				case HDL_CIRC : ePtr=POINTER_HAND;		break;
924 				case HDL_REF1 : ePtr=POINTER_REFHAND;	break;
925 				case HDL_REF2 : ePtr=POINTER_REFHAND;	break;
926 				case HDL_BWGT : ePtr=POINTER_MOVEBEZIERWEIGHT;	break;
927 				case HDL_GLUE : ePtr=POINTER_MOVEPOINT;	break;
928 				case HDL_GLUE_UNSEL : ePtr=POINTER_MOVEPOINT;	break;
929 				case HDL_CUSTOMSHAPE1 : ePtr=POINTER_HAND;	break;
930 				default:
931 					break;
932 			}
933 		}
934 	}
935 	return Pointer(ePtr);
936 }
937 
938 // #97016# II
939 sal_Bool SdrHdl::IsFocusHdl() const
940 {
941 	switch(eKind)
942 	{
943 		case HDL_UPLFT:		// top left
944 		case HDL_UPPER:		// top
945 		case HDL_UPRGT:		// top right
946 		case HDL_LEFT:		// left
947 		case HDL_RIGHT:		// right
948 		case HDL_LWLFT:		// bottom left
949 		case HDL_LOWER:		// bottom
950 		case HDL_LWRGT:		// bottom right
951 		{
952 			// if it's an activated TextEdit, it's moved to extended points
953 			if(pHdlList && pHdlList->IsMoveOutside())
954 				return sal_False;
955 			else
956 				return sal_True;
957 		}
958 
959 		case HDL_MOVE:		// Handle zum Verschieben des Objekts
960 		case HDL_POLY:		// Punktselektion an Polygon oder Bezierkurve
961 		case HDL_BWGT:		// Gewicht an einer Bezierkurve
962 		case HDL_CIRC:		// Winkel an Kreissegmenten, Eckenradius am Rect
963 		case HDL_REF1:		// Referenzpunkt 1, z.B. Rotationsmitte
964 		case HDL_REF2:		// Referenzpunkt 2, z.B. Endpunkt der Spiegelachse
965 		//case HDL_MIRX:		// Die Spiegelachse selbst
966 		case HDL_GLUE:		// glue point
967 		case HDL_GLUE_UNSEL: // glue point unselected
968 
969 		// #98388# do NOT activate here, let SW implement their own SdrHdl and
970 		// overload IsFocusHdl() there to make the anchor accessible
971 		//case HDL_ANCHOR:		// anchor symbol (SD, SW)
972 		// #101688# same for AnchorTR
973 		//case HDL_ANCHOR_TR:	// anchor symbol (SD, SW)
974 
975 		//case HDL_TRNS:		// interactive transparency
976 		//case HDL_GRAD:		// interactive gradient
977 		//case HDL_COLR:		// interactive color
978 
979 		// for SJ and the CustomShapeHandles:
980 		case HDL_CUSTOMSHAPE1:
981 
982 		case HDL_USER:
983 		{
984 			return sal_True;
985 		}
986 
987 		default:
988 		{
989 			return sal_False;
990 		}
991 	}
992 }
993 
994 void SdrHdl::onMouseEnter(const MouseEvent& /*rMEvt*/)
995 {
996 }
997 
998 void SdrHdl::onMouseLeave()
999 {
1000 }
1001 
1002 bool SdrHdl::isMouseOver() const
1003 {
1004 	return mbMouseOver;
1005 }
1006 
1007 ////////////////////////////////////////////////////////////////////////////////////////////////////
1008 // class SdrHdlColor
1009 
1010 SdrHdlColor::SdrHdlColor(const Point& rRef, Color aCol, const Size& rSize, sal_Bool bLum)
1011 :	SdrHdl(rRef, HDL_COLR),
1012 	aMarkerSize(rSize),
1013 	bUseLuminance(bLum)
1014 {
1015 	if(IsUseLuminance())
1016 		aCol = GetLuminance(aCol);
1017 
1018 	// remember color
1019 	aMarkerColor = aCol;
1020 }
1021 
1022 SdrHdlColor::~SdrHdlColor()
1023 {
1024 }
1025 
1026 void SdrHdlColor::CreateB2dIAObject()
1027 {
1028 	// first throw away old one
1029 	GetRidOfIAObject();
1030 
1031 	if(pHdlList)
1032 	{
1033 		SdrMarkView* pView = pHdlList->GetView();
1034 
1035 		if(pView && !pView->areMarkHandlesHidden())
1036 		{
1037 			SdrPageView* pPageView = pView->GetSdrPageView();
1038 
1039 			if(pPageView)
1040 			{
1041 				for(sal_uInt32 b(0L); b < pPageView->PageWindowCount(); b++)
1042 				{
1043 					// const SdrPageViewWinRec& rPageViewWinRec = rPageViewWinList[b];
1044 					const SdrPageWindow& rPageWindow = *pPageView->GetPageWindow(b);
1045 
1046 					if(rPageWindow.GetPaintWindow().OutputToWindow())
1047 					{
1048 						if(rPageWindow.GetOverlayManager())
1049 						{
1050 							Bitmap aBmpCol(CreateColorDropper(aMarkerColor));
1051 							basegfx::B2DPoint aPosition(aPos.X(), aPos.Y());
1052 							::sdr::overlay::OverlayObject* pNewOverlayObject = new
1053 								::sdr::overlay::OverlayBitmapEx(
1054 									aPosition,
1055 									BitmapEx(aBmpCol),
1056 									(sal_uInt16)(aBmpCol.GetSizePixel().Width() - 1) >> 1,
1057 									(sal_uInt16)(aBmpCol.GetSizePixel().Height() - 1) >> 1
1058 								);
1059 							DBG_ASSERT(pNewOverlayObject, "Got NO new IAO!");
1060 
1061 							// OVERLAYMANAGER
1062 							if(pNewOverlayObject)
1063 							{
1064 								rPageWindow.GetOverlayManager()->add(*pNewOverlayObject);
1065 								maOverlayGroup.append(*pNewOverlayObject);
1066 							}
1067 						}
1068 					}
1069 				}
1070 			}
1071 		}
1072 	}
1073 }
1074 
1075 Bitmap SdrHdlColor::CreateColorDropper(Color aCol)
1076 {
1077 	// get the Bitmap
1078 	Bitmap aRetval(aMarkerSize, 24);
1079 	aRetval.Erase(aCol);
1080 
1081 	// get write access
1082 	BitmapWriteAccess* pWrite = aRetval.AcquireWriteAccess();
1083 	DBG_ASSERT(pWrite, "Got NO write access to a new Bitmap !!!");
1084 
1085 	if(pWrite)
1086 	{
1087 		// draw outer border
1088 		sal_Int32 nWidth = aMarkerSize.Width();
1089 		sal_Int32 nHeight = aMarkerSize.Height();
1090 
1091 		pWrite->SetLineColor(Color(COL_LIGHTGRAY));
1092 		pWrite->DrawLine(Point(0, 0), Point(0, nHeight - 1));
1093 		pWrite->DrawLine(Point(1, 0), Point(nWidth - 1, 0));
1094 		pWrite->SetLineColor(Color(COL_GRAY));
1095 		pWrite->DrawLine(Point(1, nHeight - 1), Point(nWidth - 1, nHeight - 1));
1096 		pWrite->DrawLine(Point(nWidth - 1, 1), Point(nWidth - 1, nHeight - 2));
1097 
1098 		// draw lighter UpperLeft
1099 		const Color aLightColor(
1100 			(sal_uInt8)(::std::min((sal_Int16)((sal_Int16)aCol.GetRed() + (sal_Int16)0x0040), (sal_Int16)0x00ff)),
1101 			(sal_uInt8)(::std::min((sal_Int16)((sal_Int16)aCol.GetGreen() + (sal_Int16)0x0040), (sal_Int16)0x00ff)),
1102 			(sal_uInt8)(::std::min((sal_Int16)((sal_Int16)aCol.GetBlue() + (sal_Int16)0x0040), (sal_Int16)0x00ff)));
1103 		pWrite->SetLineColor(aLightColor);
1104 		pWrite->DrawLine(Point(1, 1), Point(1, nHeight - 2));
1105 		pWrite->DrawLine(Point(2, 1), Point(nWidth - 2, 1));
1106 
1107 		// draw darker LowerRight
1108 		const Color aDarkColor(
1109 			(sal_uInt8)(::std::max((sal_Int16)((sal_Int16)aCol.GetRed() - (sal_Int16)0x0040), (sal_Int16)0x0000)),
1110 			(sal_uInt8)(::std::max((sal_Int16)((sal_Int16)aCol.GetGreen() - (sal_Int16)0x0040), (sal_Int16)0x0000)),
1111 			(sal_uInt8)(::std::max((sal_Int16)((sal_Int16)aCol.GetBlue() - (sal_Int16)0x0040), (sal_Int16)0x0000)));
1112 		pWrite->SetLineColor(aDarkColor);
1113 		pWrite->DrawLine(Point(2, nHeight - 2), Point(nWidth - 2, nHeight - 2));
1114 		pWrite->DrawLine(Point(nWidth - 2, 2), Point(nWidth - 2, nHeight - 3));
1115 
1116 		// get rid of write access
1117 		delete pWrite;
1118 	}
1119 
1120 	return aRetval;
1121 }
1122 
1123 Color SdrHdlColor::GetLuminance(const Color& rCol)
1124 {
1125 	sal_uInt8 aLum = rCol.GetLuminance();
1126 	Color aRetval(aLum, aLum, aLum);
1127 	return aRetval;
1128 }
1129 
1130 void SdrHdlColor::CallColorChangeLink()
1131 {
1132 	aColorChangeHdl.Call(this);
1133 }
1134 
1135 void SdrHdlColor::SetColor(Color aNew, sal_Bool bCallLink)
1136 {
1137 	if(IsUseLuminance())
1138 		aNew = GetLuminance(aNew);
1139 
1140 	if(aMarkerColor != aNew)
1141 	{
1142 		// remember new color
1143 		aMarkerColor = aNew;
1144 
1145 		// create new display
1146 		Touch();
1147 
1148 		// tell about change
1149 		if(bCallLink)
1150 			CallColorChangeLink();
1151 	}
1152 }
1153 
1154 void SdrHdlColor::SetSize(const Size& rNew)
1155 {
1156 	if(rNew != aMarkerSize)
1157 	{
1158 		// remember new size
1159 		aMarkerSize = rNew;
1160 
1161 		// create new display
1162 		Touch();
1163 	}
1164 }
1165 
1166 ////////////////////////////////////////////////////////////////////////////////////////////////////
1167 // class SdrHdlGradient
1168 
1169 SdrHdlGradient::SdrHdlGradient(const Point& rRef1, const Point& rRef2, sal_Bool bGrad)
1170 :	SdrHdl(rRef1, bGrad ? HDL_GRAD : HDL_TRNS),
1171 	pColHdl1(NULL),
1172 	pColHdl2(NULL),
1173 	a2ndPos(rRef2),
1174 	bGradient(bGrad)
1175 {
1176 }
1177 
1178 SdrHdlGradient::~SdrHdlGradient()
1179 {
1180 }
1181 
1182 void SdrHdlGradient::Set2ndPos(const Point& rPnt)
1183 {
1184 	if(a2ndPos != rPnt)
1185 	{
1186 		// remember new position
1187 		a2ndPos = rPnt;
1188 
1189 		// create new display
1190 		Touch();
1191 	}
1192 }
1193 
1194 void SdrHdlGradient::CreateB2dIAObject()
1195 {
1196 	// first throw away old one
1197 	GetRidOfIAObject();
1198 
1199 	if(pHdlList)
1200 	{
1201 		SdrMarkView* pView = pHdlList->GetView();
1202 
1203 		if(pView && !pView->areMarkHandlesHidden())
1204 		{
1205 			SdrPageView* pPageView = pView->GetSdrPageView();
1206 
1207 			if(pPageView)
1208 			{
1209 				for(sal_uInt32 b(0L); b < pPageView->PageWindowCount(); b++)
1210 				{
1211 					const SdrPageWindow& rPageWindow = *pPageView->GetPageWindow(b);
1212 
1213 					if(rPageWindow.GetPaintWindow().OutputToWindow())
1214 					{
1215 						if(rPageWindow.GetOverlayManager())
1216 						{
1217 							// striped line in between
1218 							basegfx::B2DVector aVec(a2ndPos.X() - aPos.X(), a2ndPos.Y() - aPos.Y());
1219 							double fVecLen = aVec.getLength();
1220 							double fLongPercentArrow = (1.0 - 0.05) * fVecLen;
1221 							double fHalfArrowWidth = (0.05 * 0.5) * fVecLen;
1222 							aVec.normalize();
1223 							basegfx::B2DVector aPerpend(-aVec.getY(), aVec.getX());
1224 							sal_Int32 nMidX = (sal_Int32)(aPos.X() + aVec.getX() * fLongPercentArrow);
1225 							sal_Int32 nMidY = (sal_Int32)(aPos.Y() + aVec.getY() * fLongPercentArrow);
1226 							Point aMidPoint(nMidX, nMidY);
1227 
1228 							basegfx::B2DPoint aPosition(aPos.X(), aPos.Y());
1229 							basegfx::B2DPoint aMidPos(aMidPoint.X(), aMidPoint.Y());
1230 
1231 							::sdr::overlay::OverlayObject* pNewOverlayObject = new
1232 								::sdr::overlay::OverlayLineStriped(
1233 									aPosition, aMidPos
1234 								);
1235 							DBG_ASSERT(pNewOverlayObject, "Got NO new IAO!");
1236 
1237 							pNewOverlayObject->setBaseColor(IsGradient() ? Color(COL_BLACK) : Color(COL_BLUE));
1238 							rPageWindow.GetOverlayManager()->add(*pNewOverlayObject);
1239 							maOverlayGroup.append(*pNewOverlayObject);
1240 
1241 							// arrowhead
1242 							Point aLeft(aMidPoint.X() + (sal_Int32)(aPerpend.getX() * fHalfArrowWidth),
1243 										aMidPoint.Y() + (sal_Int32)(aPerpend.getY() * fHalfArrowWidth));
1244 							Point aRight(aMidPoint.X() - (sal_Int32)(aPerpend.getX() * fHalfArrowWidth),
1245 										aMidPoint.Y() - (sal_Int32)(aPerpend.getY() * fHalfArrowWidth));
1246 
1247 							basegfx::B2DPoint aPositionLeft(aLeft.X(), aLeft.Y());
1248 							basegfx::B2DPoint aPositionRight(aRight.X(), aRight.Y());
1249 							basegfx::B2DPoint aPosition2(a2ndPos.X(), a2ndPos.Y());
1250 
1251 							pNewOverlayObject = new
1252 								::sdr::overlay::OverlayTriangle(
1253 									aPositionLeft,
1254 									aPosition2,
1255 									aPositionRight,
1256 									IsGradient() ? Color(COL_BLACK) : Color(COL_BLUE)
1257 								);
1258 							DBG_ASSERT(pNewOverlayObject, "Got NO new IAO!");
1259 
1260 							rPageWindow.GetOverlayManager()->add(*pNewOverlayObject);
1261 							maOverlayGroup.append(*pNewOverlayObject);
1262 						}
1263 					}
1264 				}
1265 			}
1266 		}
1267 	}
1268 }
1269 
1270 IMPL_LINK(SdrHdlGradient, ColorChangeHdl, SdrHdl*, /*pHdl*/)
1271 {
1272 	if(GetObj())
1273 		FromIAOToItem(GetObj(), sal_True, sal_True);
1274 	return 0;
1275 }
1276 
1277 void SdrHdlGradient::FromIAOToItem(SdrObject* _pObj, sal_Bool bSetItemOnObject, sal_Bool bUndo)
1278 {
1279 	// from IAO positions and colors to gradient
1280 	const SfxItemSet& rSet = _pObj->GetMergedItemSet();
1281 
1282 	GradTransformer aGradTransformer;
1283 	GradTransGradient aOldGradTransGradient;
1284 	GradTransGradient aGradTransGradient;
1285 	GradTransVector aGradTransVector;
1286 
1287 	String aString;
1288 
1289 	aGradTransVector.maPositionA = basegfx::B2DPoint(GetPos().X(), GetPos().Y());
1290 	aGradTransVector.maPositionB = basegfx::B2DPoint(Get2ndPos().X(), Get2ndPos().Y());
1291 	if(pColHdl1)
1292 		aGradTransVector.aCol1 = pColHdl1->GetColor();
1293 	if(pColHdl2)
1294 		aGradTransVector.aCol2 = pColHdl2->GetColor();
1295 
1296 	if(IsGradient())
1297 		aOldGradTransGradient.aGradient = ((XFillGradientItem&)rSet.Get(XATTR_FILLGRADIENT)).GetGradientValue();
1298 	else
1299 		aOldGradTransGradient.aGradient = ((XFillFloatTransparenceItem&)rSet.Get(XATTR_FILLFLOATTRANSPARENCE)).GetGradientValue();
1300 
1301 	// transform vector data to gradient
1302 	aGradTransformer.VecToGrad(aGradTransVector, aGradTransGradient, aOldGradTransGradient, _pObj, bMoveSingleHandle, bMoveFirstHandle);
1303 
1304 	if(bSetItemOnObject)
1305 	{
1306 		SdrModel* pModel = _pObj->GetModel();
1307 		SfxItemSet aNewSet(pModel->GetItemPool());
1308 
1309 		if(IsGradient())
1310 		{
1311 			aString = String();
1312 			XFillGradientItem aNewGradItem(aString, aGradTransGradient.aGradient);
1313 			aNewSet.Put(aNewGradItem);
1314 		}
1315 		else
1316 		{
1317 			aString = String();
1318 			XFillFloatTransparenceItem aNewTransItem(aString, aGradTransGradient.aGradient);
1319 			aNewSet.Put(aNewTransItem);
1320 		}
1321 
1322 		if(bUndo && pModel->IsUndoEnabled())
1323 		{
1324 			pModel->BegUndo(SVX_RESSTR(IsGradient() ? SIP_XA_FILLGRADIENT : SIP_XA_FILLTRANSPARENCE));
1325 			pModel->AddUndo(pModel->GetSdrUndoFactory().CreateUndoAttrObject(*_pObj));
1326 			pModel->EndUndo();
1327 		}
1328 
1329 		pObj->SetMergedItemSetAndBroadcast(aNewSet);
1330 	}
1331 
1332 	// back transformation, set values on pIAOHandle
1333 	aGradTransformer.GradToVec(aGradTransGradient, aGradTransVector, _pObj);
1334 
1335 	SetPos(Point(FRound(aGradTransVector.maPositionA.getX()), FRound(aGradTransVector.maPositionA.getY())));
1336 	Set2ndPos(Point(FRound(aGradTransVector.maPositionB.getX()), FRound(aGradTransVector.maPositionB.getY())));
1337 	if(pColHdl1)
1338 	{
1339 		pColHdl1->SetPos(Point(FRound(aGradTransVector.maPositionA.getX()), FRound(aGradTransVector.maPositionA.getY())));
1340 		pColHdl1->SetColor(aGradTransVector.aCol1);
1341 	}
1342 	if(pColHdl2)
1343 	{
1344 		pColHdl2->SetPos(Point(FRound(aGradTransVector.maPositionB.getX()), FRound(aGradTransVector.maPositionB.getY())));
1345 		pColHdl2->SetColor(aGradTransVector.aCol2);
1346 	}
1347 }
1348 
1349 ////////////////////////////////////////////////////////////////////////////////////////////////////
1350 
1351 SdrHdlLine::~SdrHdlLine() {}
1352 
1353 void SdrHdlLine::CreateB2dIAObject()
1354 {
1355 	// first throw away old one
1356 	GetRidOfIAObject();
1357 
1358 	if(pHdlList)
1359 	{
1360 		SdrMarkView* pView = pHdlList->GetView();
1361 
1362 		if(pView && !pView->areMarkHandlesHidden() && pHdl1 && pHdl2)
1363 		{
1364 			SdrPageView* pPageView = pView->GetSdrPageView();
1365 
1366 			if(pPageView)
1367 			{
1368 				for(sal_uInt32 b(0L); b < pPageView->PageWindowCount(); b++)
1369 				{
1370 					const SdrPageWindow& rPageWindow = *pPageView->GetPageWindow(b);
1371 
1372 					if(rPageWindow.GetPaintWindow().OutputToWindow())
1373 					{
1374 						if(rPageWindow.GetOverlayManager())
1375 						{
1376 							basegfx::B2DPoint aPosition1(pHdl1->GetPos().X(), pHdl1->GetPos().Y());
1377 							basegfx::B2DPoint aPosition2(pHdl2->GetPos().X(), pHdl2->GetPos().Y());
1378 
1379 							::sdr::overlay::OverlayObject* pNewOverlayObject = new
1380 								::sdr::overlay::OverlayLineStriped(
1381 									aPosition1,
1382 									aPosition2
1383 								);
1384 							DBG_ASSERT(pNewOverlayObject, "Got NO new IAO!");
1385 
1386 							// OVERLAYMANAGER
1387 							if(pNewOverlayObject)
1388 							{
1389 								// color(?)
1390 								pNewOverlayObject->setBaseColor(Color(COL_LIGHTRED));
1391 
1392 								rPageWindow.GetOverlayManager()->add(*pNewOverlayObject);
1393 								maOverlayGroup.append(*pNewOverlayObject);
1394 							}
1395 						}
1396 					}
1397 				}
1398 			}
1399 		}
1400 	}
1401 }
1402 
1403 Pointer SdrHdlLine::GetPointer() const
1404 {
1405 	return Pointer(POINTER_REFHAND);
1406 }
1407 
1408 ////////////////////////////////////////////////////////////////////////////////////////////////////
1409 
1410 SdrHdlBezWgt::~SdrHdlBezWgt() {}
1411 
1412 void SdrHdlBezWgt::CreateB2dIAObject()
1413 {
1414 	// call parent
1415 	SdrHdl::CreateB2dIAObject();
1416 
1417 	// create lines
1418 	if(pHdlList)
1419 	{
1420 		SdrMarkView* pView = pHdlList->GetView();
1421 
1422 		if(pView && !pView->areMarkHandlesHidden())
1423 		{
1424 			SdrPageView* pPageView = pView->GetSdrPageView();
1425 
1426 			if(pPageView)
1427 			{
1428 				for(sal_uInt32 b(0L); b < pPageView->PageWindowCount(); b++)
1429 				{
1430 					const SdrPageWindow& rPageWindow = *pPageView->GetPageWindow(b);
1431 
1432 					if(rPageWindow.GetPaintWindow().OutputToWindow())
1433 					{
1434 						if(rPageWindow.GetOverlayManager())
1435 						{
1436 							basegfx::B2DPoint aPosition1(pHdl1->GetPos().X(), pHdl1->GetPos().Y());
1437 							basegfx::B2DPoint aPosition2(aPos.X(), aPos.Y());
1438 
1439 							if(!aPosition1.equal(aPosition2))
1440 							{
1441 								::sdr::overlay::OverlayObject* pNewOverlayObject = new
1442 									::sdr::overlay::OverlayLineStriped(
1443 										aPosition1,
1444 										aPosition2
1445 									);
1446 								DBG_ASSERT(pNewOverlayObject, "Got NO new IAO!");
1447 
1448 								// OVERLAYMANAGER
1449 								if(pNewOverlayObject)
1450 								{
1451 									// line part is not hittable
1452 									pNewOverlayObject->setHittable(sal_False);
1453 
1454 									// color(?)
1455 									pNewOverlayObject->setBaseColor(Color(COL_LIGHTBLUE));
1456 
1457 									rPageWindow.GetOverlayManager()->add(*pNewOverlayObject);
1458 									maOverlayGroup.append(*pNewOverlayObject);
1459 								}
1460 							}
1461 						}
1462 					}
1463 				}
1464 			}
1465 		}
1466 	}
1467 }
1468 
1469 ////////////////////////////////////////////////////////////////////////////////////////////////////
1470 
1471 E3dVolumeMarker::E3dVolumeMarker(const basegfx::B2DPolyPolygon& rWireframePoly)
1472 {
1473 	aWireframePoly = rWireframePoly;
1474 }
1475 
1476 void E3dVolumeMarker::CreateB2dIAObject()
1477 {
1478 	// create lines
1479 	if(pHdlList)
1480 	{
1481 		SdrMarkView* pView = pHdlList->GetView();
1482 
1483 		if(pView && !pView->areMarkHandlesHidden())
1484 		{
1485 			SdrPageView* pPageView = pView->GetSdrPageView();
1486 
1487 			if(pPageView)
1488 			{
1489 				for(sal_uInt32 b(0L); b < pPageView->PageWindowCount(); b++)
1490 				{
1491 					const SdrPageWindow& rPageWindow = *pPageView->GetPageWindow(b);
1492 
1493 					if(rPageWindow.GetPaintWindow().OutputToWindow())
1494 					{
1495 						if(rPageWindow.GetOverlayManager() && aWireframePoly.count())
1496 							{
1497 								::sdr::overlay::OverlayObject* pNewOverlayObject = new
1498 								::sdr::overlay::OverlayPolyPolygonStripedAndFilled(
1499 									aWireframePoly);
1500 								DBG_ASSERT(pNewOverlayObject, "Got NO new IAO!");
1501 
1502 								// OVERLAYMANAGER
1503 								if(pNewOverlayObject)
1504 								{
1505 									pNewOverlayObject->setBaseColor(Color(COL_BLACK));
1506 
1507 									rPageWindow.GetOverlayManager()->add(*pNewOverlayObject);
1508 									maOverlayGroup.append(*pNewOverlayObject);
1509 								}
1510 							}
1511 						}
1512 					}
1513 				}
1514 			}
1515 		}
1516 	}
1517 
1518 ////////////////////////////////////////////////////////////////////////////////////////////////////
1519 
1520 ImpEdgeHdl::~ImpEdgeHdl()
1521 {
1522 }
1523 
1524 void ImpEdgeHdl::CreateB2dIAObject()
1525 {
1526 	if(nObjHdlNum <= 1 && pObj)
1527 	{
1528 		// first throw away old one
1529 		GetRidOfIAObject();
1530 
1531 		BitmapColorIndex eColIndex = LightCyan;
1532 		BitmapMarkerKind eKindOfMarker = Rect_7x7;
1533 
1534 		if(pHdlList)
1535 		{
1536 			SdrMarkView* pView = pHdlList->GetView();
1537 
1538 			if(pView && !pView->areMarkHandlesHidden())
1539 			{
1540 				const SdrEdgeObj* pEdge = (SdrEdgeObj*)pObj;
1541 
1542 				if(pEdge->GetConnectedNode(nObjHdlNum == 0) != NULL)
1543 					eColIndex = LightRed;
1544 
1545 				if(nPPntNum < 2)
1546 				{
1547 					// Handle with plus sign inside
1548 					eKindOfMarker = Circ_7x7;
1549 				}
1550 
1551 				SdrPageView* pPageView = pView->GetSdrPageView();
1552 
1553 				if(pPageView)
1554 				{
1555 					for(sal_uInt32 b(0); b < pPageView->PageWindowCount(); b++)
1556 					{
1557 						const SdrPageWindow& rPageWindow = *pPageView->GetPageWindow(b);
1558 
1559 						if(rPageWindow.GetPaintWindow().OutputToWindow())
1560 						{
1561 							if(rPageWindow.GetOverlayManager())
1562 							{
1563 								basegfx::B2DPoint aPosition(aPos.X(), aPos.Y());
1564 
1565 								::sdr::overlay::OverlayObject* pNewOverlayObject = CreateOverlayObject(
1566 									aPosition,
1567 									eColIndex,
1568 									eKindOfMarker);
1569 
1570 								// OVERLAYMANAGER
1571 								if(pNewOverlayObject)
1572 								{
1573 									rPageWindow.GetOverlayManager()->add(*pNewOverlayObject);
1574 									maOverlayGroup.append(*pNewOverlayObject);
1575 								}
1576 							}
1577 						}
1578 					}
1579 				}
1580 			}
1581 		}
1582 	}
1583 	else
1584 	{
1585 		// call parent
1586 		SdrHdl::CreateB2dIAObject();
1587 	}
1588 }
1589 
1590 void ImpEdgeHdl::SetLineCode(SdrEdgeLineCode eCode)
1591 {
1592 	if(eLineCode != eCode)
1593 	{
1594 		// remember new value
1595 		eLineCode = eCode;
1596 
1597 		// create new display
1598 		Touch();
1599 	}
1600 }
1601 
1602 Pointer ImpEdgeHdl::GetPointer() const
1603 {
1604 	SdrEdgeObj* pEdge=PTR_CAST(SdrEdgeObj,pObj);
1605 	if (pEdge==NULL)
1606 		return SdrHdl::GetPointer();
1607 	if (nObjHdlNum<=1)
1608 		return Pointer(POINTER_MOVEPOINT); //Pointer(POINTER_DRAW_CONNECT);
1609 	if (IsHorzDrag())
1610 		return Pointer(POINTER_ESIZE);
1611 	else
1612 		return Pointer(POINTER_SSIZE);
1613 }
1614 
1615 sal_Bool ImpEdgeHdl::IsHorzDrag() const
1616 {
1617 	SdrEdgeObj* pEdge=PTR_CAST(SdrEdgeObj,pObj);
1618 	if (pEdge==NULL)
1619 		return sal_False;
1620 	if (nObjHdlNum<=1)
1621 		return sal_False;
1622 
1623 	SdrEdgeKind eEdgeKind = ((SdrEdgeKindItem&)(pEdge->GetObjectItem(SDRATTR_EDGEKIND))).GetValue();
1624 
1625 	const SdrEdgeInfoRec& rInfo=pEdge->aEdgeInfo;
1626 	if (eEdgeKind==SDREDGE_ORTHOLINES || eEdgeKind==SDREDGE_BEZIER)
1627 	{
1628 		return !rInfo.ImpIsHorzLine(eLineCode,*pEdge->pEdgeTrack);
1629 	}
1630 	else if (eEdgeKind==SDREDGE_THREELINES)
1631 	{
1632 		long nWink=nObjHdlNum==2 ? rInfo.nAngle1 : rInfo.nAngle2;
1633 		if (nWink==0 || nWink==18000)
1634 			return sal_True;
1635 		else
1636 			return sal_False;
1637 	}
1638 	return sal_False;
1639 }
1640 
1641 ////////////////////////////////////////////////////////////////////////////////////////////////////
1642 
1643 ImpMeasureHdl::~ImpMeasureHdl()
1644 {
1645 }
1646 
1647 void ImpMeasureHdl::CreateB2dIAObject()
1648 {
1649 	// first throw away old one
1650 	GetRidOfIAObject();
1651 
1652 	if(pHdlList)
1653 	{
1654 		SdrMarkView* pView = pHdlList->GetView();
1655 
1656 		if(pView && !pView->areMarkHandlesHidden())
1657 		{
1658 			BitmapColorIndex eColIndex = LightCyan;
1659 			BitmapMarkerKind eKindOfMarker = Rect_9x9;
1660 
1661 			if(nObjHdlNum > 1)
1662 			{
1663 				eKindOfMarker = Rect_7x7;
1664 			}
1665 
1666 			if(bSelect)
1667 			{
1668 				eColIndex = Cyan;
1669 			}
1670 
1671 			SdrPageView* pPageView = pView->GetSdrPageView();
1672 
1673 			if(pPageView)
1674 			{
1675 				for(sal_uInt32 b(0L); b < pPageView->PageWindowCount(); b++)
1676 				{
1677 					const SdrPageWindow& rPageWindow = *pPageView->GetPageWindow(b);
1678 
1679 					if(rPageWindow.GetPaintWindow().OutputToWindow())
1680 					{
1681 						if(rPageWindow.GetOverlayManager())
1682 						{
1683 							basegfx::B2DPoint aPosition(aPos.X(), aPos.Y());
1684 
1685 							::sdr::overlay::OverlayObject* pNewOverlayObject = CreateOverlayObject(
1686 								aPosition,
1687 								eColIndex,
1688 								eKindOfMarker);
1689 
1690 							// OVERLAYMANAGER
1691 							if(pNewOverlayObject)
1692 							{
1693 								rPageWindow.GetOverlayManager()->add(*pNewOverlayObject);
1694 								maOverlayGroup.append(*pNewOverlayObject);
1695 							}
1696 						}
1697 					}
1698 				}
1699 			}
1700 		}
1701 	}
1702 }
1703 
1704 Pointer ImpMeasureHdl::GetPointer() const
1705 {
1706 	switch (nObjHdlNum)
1707 	{
1708 		case 0: case 1: return Pointer(POINTER_HAND);
1709 		case 2: case 3: return Pointer(POINTER_MOVEPOINT);
1710 		case 4: case 5: return SdrHdl::GetPointer(); // will be rotated accordingly
1711 	} // switch
1712 	return Pointer(POINTER_NOTALLOWED);
1713 }
1714 
1715 ////////////////////////////////////////////////////////////////////////////////////////////////////
1716 
1717 ImpTextframeHdl::ImpTextframeHdl(const Rectangle& rRect) :
1718 	SdrHdl(rRect.TopLeft(),HDL_MOVE),
1719 	maRect(rRect)
1720 {
1721 }
1722 
1723 void ImpTextframeHdl::CreateB2dIAObject()
1724 {
1725 	// first throw away old one
1726 	GetRidOfIAObject();
1727 
1728 	if(pHdlList)
1729 	{
1730 		SdrMarkView* pView = pHdlList->GetView();
1731 
1732 		if(pView && !pView->areMarkHandlesHidden())
1733 		{
1734 			SdrPageView* pPageView = pView->GetSdrPageView();
1735 
1736 			if(pPageView)
1737 			{
1738 				for(sal_uInt32 b(0L); b < pPageView->PageWindowCount(); b++)
1739 				{
1740 					const SdrPageWindow& rPageWindow = *pPageView->GetPageWindow(b);
1741 
1742 					if(rPageWindow.GetPaintWindow().OutputToWindow())
1743 					{
1744 						if(rPageWindow.GetOverlayManager())
1745 						{
1746 							const basegfx::B2DPoint aTopLeft(maRect.Left(), maRect.Top());
1747 							const basegfx::B2DPoint aBottomRight(maRect.Right(), maRect.Bottom());
1748 							const SvtOptionsDrawinglayer aSvtOptionsDrawinglayer;
1749 							const Color aHilightColor(aSvtOptionsDrawinglayer.getHilightColor());
1750 							const double fTransparence(aSvtOptionsDrawinglayer.GetTransparentSelectionPercent() * 0.01);
1751 
1752 							::sdr::overlay::OverlayRectangle* pNewOverlayObject = new ::sdr::overlay::OverlayRectangle(
1753 								aTopLeft,
1754 								aBottomRight,
1755 								aHilightColor,
1756 								fTransparence,
1757 								3.0,
1758 								3.0,
1759 								nDrehWink * -F_PI18000,
1760 								500,
1761 								true); // allow animation; the Handle is not shown at text edit time
1762 
1763 							pNewOverlayObject->setHittable(false);
1764 
1765 							// OVERLAYMANAGER
1766 							if(pNewOverlayObject)
1767 							{
1768 								rPageWindow.GetOverlayManager()->add(*pNewOverlayObject);
1769 								maOverlayGroup.append(*pNewOverlayObject);
1770 							}
1771 						}
1772 					}
1773 				}
1774 			}
1775 		}
1776 	}
1777 }
1778 
1779 ////////////////////////////////////////////////////////////////////////////////////////////////////
1780 
1781 class ImpSdrHdlListSorter: public ContainerSorter {
1782 public:
1783 	ImpSdrHdlListSorter(Container& rNewCont): ContainerSorter(rNewCont) {}
1784 	virtual int Compare(const void* pElem1, const void* pElem2) const;
1785 };
1786 
1787 int ImpSdrHdlListSorter::Compare(const void* pElem1, const void* pElem2) const
1788 {
1789 	SdrHdlKind eKind1=((SdrHdl*)pElem1)->GetKind();
1790 	SdrHdlKind eKind2=((SdrHdl*)pElem2)->GetKind();
1791 	// Level 1: Erst normale Handles, dann Glue, dann User, dann Plushandles, dann Retpunkt-Handles
1792 	unsigned n1=1;
1793 	unsigned n2=1;
1794 	if (eKind1!=eKind2)
1795 	{
1796 		if (eKind1==HDL_REF1 || eKind1==HDL_REF2 || eKind1==HDL_MIRX) n1=5;
1797 		else if (eKind1==HDL_GLUE || eKind1==HDL_GLUE_UNSEL) n1=2;
1798 		else if (eKind1==HDL_USER) n1=3;
1799 		else if (eKind1==HDL_SMARTTAG) n1=0;
1800 		if (eKind2==HDL_REF1 || eKind2==HDL_REF2 || eKind2==HDL_MIRX) n2=5;
1801 		else if (eKind2==HDL_GLUE || eKind1==HDL_GLUE_UNSEL) n2=2;
1802 		else if (eKind2==HDL_USER) n2=3;
1803 		else if (eKind2==HDL_SMARTTAG) n2=0;
1804 	}
1805 	if (((SdrHdl*)pElem1)->IsPlusHdl()) n1=4;
1806 	if (((SdrHdl*)pElem2)->IsPlusHdl()) n2=4;
1807 	if (n1==n2)
1808 	{
1809 		// Level 2: PageView (Pointer)
1810 		SdrPageView* pPV1=((SdrHdl*)pElem1)->GetPageView();
1811 		SdrPageView* pPV2=((SdrHdl*)pElem2)->GetPageView();
1812 		if (pPV1==pPV2)
1813 		{
1814 			// Level 3: Position (x+y)
1815 			SdrObject* pObj1=((SdrHdl*)pElem1)->GetObj();
1816 			SdrObject* pObj2=((SdrHdl*)pElem2)->GetObj();
1817 			if (pObj1==pObj2)
1818 			{
1819 				sal_uInt32 nNum1=((SdrHdl*)pElem1)->GetObjHdlNum();
1820 				sal_uInt32 nNum2=((SdrHdl*)pElem2)->GetObjHdlNum();
1821 				if (nNum1==nNum2)
1822 				{ // #48763#
1823 					if (eKind1==eKind2)
1824 						return (long)pElem1<(long)pElem2 ? -1 : 1; // Notloesung, um immer die gleiche Sortierung zu haben
1825 					return (sal_uInt16)eKind1<(sal_uInt16)eKind2 ? -1 : 1;
1826 				}
1827 				else
1828 					return nNum1<nNum2 ? -1 : 1;
1829 			}
1830 			else
1831 			{
1832 				return (long)pObj1<(long)pObj2 ? -1 : 1;
1833 			}
1834 		}
1835 		else
1836 		{
1837 			return (long)pPV1<(long)pPV2 ? -1 : 1;
1838 		}
1839 	}
1840 	else
1841 	{
1842 		return n1<n2 ? -1 : 1;
1843 	}
1844 }
1845 
1846 SdrMarkView* SdrHdlList::GetView() const
1847 {
1848 	return pView;
1849 }
1850 
1851 // #105678# Help struct for re-sorting handles
1852 struct ImplHdlAndIndex
1853 {
1854 	SdrHdl*						mpHdl;
1855 	sal_uInt32					mnIndex;
1856 };
1857 
1858 // #105678# Help method for sorting handles taking care of OrdNums, keeping order in
1859 // single objects and re-sorting polygon handles intuitively
1860 extern "C" int __LOADONCALLAPI ImplSortHdlFunc( const void* pVoid1, const void* pVoid2 )
1861 {
1862 	const ImplHdlAndIndex* p1 = (ImplHdlAndIndex*)pVoid1;
1863 	const ImplHdlAndIndex* p2 = (ImplHdlAndIndex*)pVoid2;
1864 
1865 	if(p1->mpHdl->GetObj() == p2->mpHdl->GetObj())
1866 	{
1867 		if(p1->mpHdl->GetObj() && p1->mpHdl->GetObj()->ISA(SdrPathObj))
1868 		{
1869 			// same object and a path object
1870 			if((p1->mpHdl->GetKind() == HDL_POLY || p1->mpHdl->GetKind() == HDL_BWGT)
1871 				&& (p2->mpHdl->GetKind() == HDL_POLY || p2->mpHdl->GetKind() == HDL_BWGT))
1872 			{
1873 				// both handles are point or control handles
1874 				if(p1->mpHdl->GetPolyNum() == p2->mpHdl->GetPolyNum())
1875 				{
1876 					if(p1->mpHdl->GetPointNum() < p2->mpHdl->GetPointNum())
1877 					{
1878 						return -1;
1879 					}
1880 					else
1881 					{
1882 						return 1;
1883 					}
1884 				}
1885 				else if(p1->mpHdl->GetPolyNum() < p2->mpHdl->GetPolyNum())
1886 				{
1887 					return -1;
1888 				}
1889 				else
1890 				{
1891 					return 1;
1892 				}
1893 			}
1894 		}
1895 	}
1896 	else
1897 	{
1898 		if(!p1->mpHdl->GetObj())
1899 		{
1900 			return -1;
1901 		}
1902 		else if(!p2->mpHdl->GetObj())
1903 		{
1904 			return 1;
1905 		}
1906 		else
1907 		{
1908 			// different objects, use OrdNum for sort
1909 			const sal_uInt32 nOrdNum1 = p1->mpHdl->GetObj()->GetOrdNum();
1910 			const sal_uInt32 nOrdNum2 = p2->mpHdl->GetObj()->GetOrdNum();
1911 
1912 			if(nOrdNum1 < nOrdNum2)
1913 			{
1914 				return -1;
1915 			}
1916 			else
1917 			{
1918 				return 1;
1919 			}
1920 		}
1921 	}
1922 
1923 	// fallback to indices
1924 	if(p1->mnIndex < p2->mnIndex)
1925 	{
1926 		return -1;
1927 	}
1928 	else
1929 	{
1930 		return 1;
1931 	}
1932 }
1933 
1934 ////////////////////////////////////////////////////////////////////////////////////////////////////
1935 // #97016# II
1936 
1937 void SdrHdlList::TravelFocusHdl(sal_Bool bForward)
1938 {
1939 	// security correction
1940 	if(mnFocusIndex != CONTAINER_ENTRY_NOTFOUND && mnFocusIndex >= GetHdlCount())
1941 		mnFocusIndex = CONTAINER_ENTRY_NOTFOUND;
1942 
1943 	if(aList.Count())
1944 	{
1945 		// take care of old handle
1946 		const sal_uIntPtr nOldHdlNum(mnFocusIndex);
1947 		SdrHdl* pOld = GetHdl(nOldHdlNum);
1948 		//SDOsal_Bool bRefresh(sal_False);
1949 
1950 		if(pOld)
1951 		{
1952 			// switch off old handle
1953 			mnFocusIndex = CONTAINER_ENTRY_NOTFOUND;
1954 			pOld->Touch();
1955 			//SDObRefresh = sal_True;
1956 		}
1957 
1958 		// #105678# Alloc pointer array for sorted handle list
1959 		ImplHdlAndIndex* pHdlAndIndex = new ImplHdlAndIndex[aList.Count()];
1960 
1961 		// #105678# build sorted handle list
1962 		sal_uInt32 a;
1963 		for( a = 0; a < aList.Count(); a++)
1964 		{
1965 			pHdlAndIndex[a].mpHdl = (SdrHdl*)aList.GetObject(a);
1966 			pHdlAndIndex[a].mnIndex = a;
1967 		}
1968 
1969 		// #105678# qsort all entries
1970 		qsort(pHdlAndIndex, aList.Count(), sizeof(ImplHdlAndIndex), ImplSortHdlFunc);
1971 
1972 		// #105678# look for old num in sorted array
1973 		sal_uIntPtr nOldHdl(nOldHdlNum);
1974 
1975 		if(nOldHdlNum != CONTAINER_ENTRY_NOTFOUND)
1976 		{
1977 			for(a = 0; a < aList.Count(); a++)
1978 			{
1979 				if(pHdlAndIndex[a].mpHdl == pOld)
1980 				{
1981 					nOldHdl = a;
1982 					break;
1983 				}
1984 			}
1985 		}
1986 
1987 		// #105678# build new HdlNum
1988 		sal_uIntPtr nNewHdl(nOldHdl);
1989 
1990 		// #105678# do the focus travel
1991 		if(bForward)
1992 		{
1993 			if(nOldHdl != CONTAINER_ENTRY_NOTFOUND)
1994 			{
1995 				if(nOldHdl == aList.Count() - 1)
1996 				{
1997 					// end forward run
1998 					nNewHdl = CONTAINER_ENTRY_NOTFOUND;
1999 				}
2000 				else
2001 				{
2002 					// simply the next handle
2003 					nNewHdl++;
2004 				}
2005 			}
2006 			else
2007 			{
2008 				// start forward run at first entry
2009 				nNewHdl = 0;
2010 			}
2011 		}
2012 		else
2013 		{
2014 			if(nOldHdl == CONTAINER_ENTRY_NOTFOUND)
2015 			{
2016 				// start backward run at last entry
2017 				nNewHdl = aList.Count() - 1;
2018 
2019 			}
2020 			else
2021 			{
2022 				if(nOldHdl == 0)
2023 				{
2024 					// end backward run
2025 					nNewHdl = CONTAINER_ENTRY_NOTFOUND;
2026 				}
2027 				else
2028 				{
2029 					// simply the previous handle
2030 					nNewHdl--;
2031 				}
2032 			}
2033 		}
2034 
2035 		// #105678# build new HdlNum
2036 		sal_uInt32 nNewHdlNum(nNewHdl);
2037 
2038 		// look for old num in sorted array
2039 		if(nNewHdl != CONTAINER_ENTRY_NOTFOUND)
2040 		{
2041 			SdrHdl* pNew = pHdlAndIndex[nNewHdl].mpHdl;
2042 
2043 			for(a = 0; a < aList.Count(); a++)
2044 			{
2045 				if((SdrHdl*)aList.GetObject(a) == pNew)
2046 				{
2047 					nNewHdlNum = a;
2048 					break;
2049 				}
2050 			}
2051 		}
2052 
2053 		// take care of next handle
2054 		if(nOldHdlNum != nNewHdlNum)
2055 		{
2056 			mnFocusIndex = nNewHdlNum;
2057 			SdrHdl* pNew = GetHdl(mnFocusIndex);
2058 
2059 			if(pNew)
2060 			{
2061 				pNew->Touch();
2062 				//SDObRefresh = sal_True;
2063 			}
2064 		}
2065 
2066 		// #105678# free mem again
2067 		delete [] pHdlAndIndex;
2068 	}
2069 }
2070 
2071 SdrHdl* SdrHdlList::GetFocusHdl() const
2072 {
2073 	if(mnFocusIndex != CONTAINER_ENTRY_NOTFOUND && mnFocusIndex < GetHdlCount())
2074 		return GetHdl(mnFocusIndex);
2075 	else
2076 		return 0L;
2077 }
2078 
2079 void SdrHdlList::SetFocusHdl(SdrHdl* pNew)
2080 {
2081 	if(pNew)
2082 	{
2083 		SdrHdl* pActual = GetFocusHdl();
2084 
2085 		if(!pActual || pActual != pNew)
2086 		{
2087 			sal_uIntPtr nNewHdlNum = GetHdlNum(pNew);
2088 
2089 			if(nNewHdlNum != CONTAINER_ENTRY_NOTFOUND)
2090 			{
2091 				//SDOsal_Bool bRefresh(sal_False);
2092 				mnFocusIndex = nNewHdlNum;
2093 
2094 				if(pActual)
2095 				{
2096 					pActual->Touch();
2097 					//SDObRefresh = sal_True;
2098 				}
2099 
2100 				if(pNew)
2101 				{
2102 					pNew->Touch();
2103 					//SDObRefresh = sal_True;
2104 				}
2105 
2106 				//OLMif(bRefresh)
2107 				//OLM{
2108 				//OLM	if(pView)
2109 				//OLM		pView->RefreshAllIAOManagers();
2110 				//OLM}
2111 			}
2112 		}
2113 	}
2114 }
2115 
2116 void SdrHdlList::ResetFocusHdl()
2117 {
2118 	SdrHdl* pHdl = GetFocusHdl();
2119 
2120 	mnFocusIndex = CONTAINER_ENTRY_NOTFOUND;
2121 
2122 	if(pHdl)
2123 	{
2124 		pHdl->Touch();
2125 	}
2126 }
2127 
2128 ////////////////////////////////////////////////////////////////////////////////////////////////////
2129 
2130 SdrHdlList::SdrHdlList(SdrMarkView* pV)
2131 :	mnFocusIndex(CONTAINER_ENTRY_NOTFOUND),
2132 	pView(pV),
2133 	aList(1024,32,32)
2134 {
2135 	nHdlSize = 3;
2136 	bRotateShear = sal_False;
2137 	bMoveOutside = sal_False;
2138 	bDistortShear = sal_False;
2139 	bFineHandles = sal_True; // new default: Handles are fine handles
2140 }
2141 
2142 SdrHdlList::~SdrHdlList()
2143 {
2144 	Clear();
2145 }
2146 
2147 void SdrHdlList::SetHdlSize(sal_uInt16 nSiz)
2148 {
2149 	if(nHdlSize != nSiz)
2150 	{
2151 		// remember new value
2152 		nHdlSize = nSiz;
2153 
2154 		// propagate change to IAOs
2155 		for(sal_uInt32 i=0; i<GetHdlCount(); i++)
2156 		{
2157 			SdrHdl* pHdl = GetHdl(i);
2158 			pHdl->Touch();
2159 		}
2160 	}
2161 }
2162 
2163 void SdrHdlList::SetMoveOutside(sal_Bool bOn)
2164 {
2165 	if(bMoveOutside != bOn)
2166 	{
2167 		// remember new value
2168 		bMoveOutside = bOn;
2169 
2170 		// propagate change to IAOs
2171 		for(sal_uInt32 i=0; i<GetHdlCount(); i++)
2172 		{
2173 			SdrHdl* pHdl = GetHdl(i);
2174 			pHdl->Touch();
2175 		}
2176 	}
2177 }
2178 
2179 void SdrHdlList::SetRotateShear(sal_Bool bOn)
2180 {
2181 	bRotateShear = bOn;
2182 }
2183 
2184 void SdrHdlList::SetDistortShear(sal_Bool bOn)
2185 {
2186 	bDistortShear = bOn;
2187 }
2188 
2189 void SdrHdlList::SetFineHdl(sal_Bool bOn)
2190 {
2191 	if(bFineHandles != bOn)
2192 	{
2193 		// remember new state
2194 		bFineHandles = bOn;
2195 
2196 		// propagate change to IAOs
2197 		for(sal_uInt32 i=0; i<GetHdlCount(); i++)
2198 		{
2199 			SdrHdl* pHdl = GetHdl(i);
2200 			pHdl->Touch();
2201 		}
2202 	}
2203 }
2204 
2205 SdrHdl* SdrHdlList::RemoveHdl(sal_uIntPtr nNum)
2206 {
2207 	SdrHdl* pRetval = (SdrHdl*)aList.Remove(nNum);
2208 
2209 	return pRetval;
2210 }
2211 
2212 void SdrHdlList::Clear()
2213 {
2214 	for (sal_uIntPtr i=0; i<GetHdlCount(); i++)
2215 	{
2216 		SdrHdl* pHdl=GetHdl(i);
2217 		delete pHdl;
2218 	}
2219 	aList.Clear();
2220 
2221 	bRotateShear=sal_False;
2222 	bDistortShear=sal_False;
2223 }
2224 
2225 void SdrHdlList::Sort()
2226 {
2227 	// #97016# II: remember current focused handle
2228 	SdrHdl* pPrev = GetFocusHdl();
2229 
2230 	ImpSdrHdlListSorter aSort(aList);
2231 	aSort.DoSort();
2232 
2233 	// #97016# II: get now and compare
2234 	SdrHdl* pNow = GetFocusHdl();
2235 
2236 	if(pPrev != pNow)
2237 	{
2238 		//SDOsal_Bool bRefresh(sal_False);
2239 
2240 		if(pPrev)
2241 		{
2242 			pPrev->Touch();
2243 			//SDObRefresh = sal_True;
2244 		}
2245 
2246 		if(pNow)
2247 		{
2248 			pNow->Touch();
2249 			//SDObRefresh = sal_True;
2250 		}
2251 	}
2252 }
2253 
2254 sal_uIntPtr SdrHdlList::GetHdlNum(const SdrHdl* pHdl) const
2255 {
2256 	if (pHdl==NULL)
2257 		return CONTAINER_ENTRY_NOTFOUND;
2258 	sal_uIntPtr nPos=aList.GetPos(pHdl);
2259 	return nPos;
2260 }
2261 
2262 void SdrHdlList::AddHdl(SdrHdl* pHdl, sal_Bool bAtBegin)
2263 {
2264 	if (pHdl!=NULL)
2265 	{
2266 		if (bAtBegin)
2267 		{
2268 			aList.Insert(pHdl,sal_uIntPtr(0));
2269 		}
2270 		else
2271 		{
2272 			aList.Insert(pHdl,CONTAINER_APPEND);
2273 		}
2274 		pHdl->SetHdlList(this);
2275 	}
2276 }
2277 
2278 SdrHdl* SdrHdlList::IsHdlListHit(const Point& rPnt, sal_Bool bBack, sal_Bool bNext, SdrHdl* pHdl0) const
2279 {
2280 	SdrHdl* pRet=NULL;
2281 	sal_uIntPtr nAnz=GetHdlCount();
2282 	sal_uIntPtr nNum=bBack ? 0 : nAnz;
2283 	while ((bBack ? nNum<nAnz : nNum>0) && pRet==NULL)
2284 	{
2285 		if (!bBack)
2286 			nNum--;
2287 		SdrHdl* pHdl=GetHdl(nNum);
2288 		if (bNext)
2289 		{
2290 			if (pHdl==pHdl0)
2291 				bNext=sal_False;
2292 		}
2293 		else
2294 		{
2295 			if (pHdl->IsHdlHit(rPnt))
2296 				pRet=pHdl;
2297 		}
2298 		if (bBack)
2299 			nNum++;
2300 	}
2301 	return pRet;
2302 }
2303 
2304 SdrHdl* SdrHdlList::GetHdl(SdrHdlKind eKind1) const
2305 {
2306 	SdrHdl* pRet=NULL;
2307 	for (sal_uIntPtr i=0; i<GetHdlCount() && pRet==NULL; i++)
2308 	{
2309 		SdrHdl* pHdl=GetHdl(i);
2310 		if (pHdl->GetKind()==eKind1)
2311 			pRet=pHdl;
2312 	}
2313 	return pRet;
2314 }
2315 
2316 // --------------------------------------------------------------------
2317 // SdrCropHdl
2318 // --------------------------------------------------------------------
2319 
2320 SdrCropHdl::SdrCropHdl(
2321 	const Point& rPnt,
2322 	SdrHdlKind eNewKind,
2323 	double fShearX,
2324 	double fRotation)
2325 :	SdrHdl(rPnt, eNewKind),
2326 	mfShearX(fShearX),
2327 	mfRotation(fRotation)
2328 {
2329 }
2330 
2331 // --------------------------------------------------------------------
2332 
2333 BitmapEx SdrCropHdl::GetHandlesBitmap( bool bIsFineHdl, bool bIsHighContrast )
2334 {
2335 	if( bIsHighContrast )
2336 	{
2337 		static BitmapEx* pHighContrastBitmap = 0;
2338 		if( pHighContrastBitmap == 0 )
2339 			pHighContrastBitmap = new BitmapEx(ResId(SIP_SA_ACCESSIBILITY_CROP_MARKERS, *ImpGetResMgr()));
2340 		return *pHighContrastBitmap;
2341 	}
2342 	else if( bIsFineHdl )
2343 	{
2344 		static BitmapEx* pModernBitmap = 0;
2345 		if( pModernBitmap == 0 )
2346 			pModernBitmap = new BitmapEx(ResId(SIP_SA_CROP_FINE_MARKERS, *ImpGetResMgr()));
2347 		return *pModernBitmap;
2348 	}
2349 	else
2350 	{
2351 		static BitmapEx* pSimpleBitmap = 0;
2352 		if( pSimpleBitmap == 0 )
2353 			pSimpleBitmap = new BitmapEx(ResId(SIP_SA_CROP_MARKERS, *ImpGetResMgr()));
2354 		return *pSimpleBitmap;
2355 	}
2356 }
2357 
2358 // --------------------------------------------------------------------
2359 
2360 BitmapEx SdrCropHdl::GetBitmapForHandle( const BitmapEx& rBitmap, int nSize )
2361 {
2362 	int nPixelSize = 0, nX = 0, nY = 0, nOffset = 0;
2363 
2364 	if( nSize <= 3 )
2365 	{
2366 		nPixelSize = 13;
2367 		nOffset = 0;
2368 	}
2369 	else if( nSize <=4 )
2370 	{
2371 		nPixelSize = 17;
2372 		nOffset = 39;
2373 	}
2374 	else
2375 	{
2376 		nPixelSize = 21;
2377 		nOffset = 90;
2378 	}
2379 
2380 	switch( eKind )
2381 	{
2382 		case HDL_UPLFT: nX = 0; nY = 0; break;
2383 		case HDL_UPPER: nX = 1; nY = 0; break;
2384 		case HDL_UPRGT: nX = 2; nY = 0; break;
2385 		case HDL_LEFT:  nX = 0; nY = 1; break;
2386 		case HDL_RIGHT: nX = 2; nY = 1; break;
2387 		case HDL_LWLFT: nX = 0; nY = 2; break;
2388 		case HDL_LOWER: nX = 1; nY = 2; break;
2389 		case HDL_LWRGT: nX = 2; nY = 2; break;
2390 		default: break;
2391 	}
2392 
2393 	Rectangle aSourceRect( Point( nX * (nPixelSize) + nOffset, nY * (nPixelSize)), Size(nPixelSize, nPixelSize) );
2394 
2395 	BitmapEx aRetval(rBitmap);
2396 	aRetval.Crop(aSourceRect);
2397 	return aRetval;
2398 }
2399 
2400 // --------------------------------------------------------------------
2401 
2402 void SdrCropHdl::CreateB2dIAObject()
2403 {
2404 	// first throw away old one
2405 	GetRidOfIAObject();
2406 
2407 	SdrMarkView* pView = pHdlList ? pHdlList->GetView() : 0;
2408 	SdrPageView* pPageView = pView ? pView->GetSdrPageView() : 0;
2409 
2410 	if( pPageView && !pView->areMarkHandlesHidden() )
2411 	{
2412 		sal_Bool bIsFineHdl(pHdlList->IsFineHdl());
2413 		const StyleSettings& rStyleSettings = Application::GetSettings().GetStyleSettings();
2414 		sal_Bool bIsHighContrast(rStyleSettings.GetHighContrastMode());
2415 		int nHdlSize = pHdlList->GetHdlSize();
2416 		if( bIsHighContrast )
2417 			nHdlSize = 4;
2418 
2419 		const BitmapEx aHandlesBitmap( GetHandlesBitmap( bIsFineHdl, bIsHighContrast ) );
2420 		BitmapEx aBmpEx1( GetBitmapForHandle( aHandlesBitmap, nHdlSize ) );
2421 
2422 		for(sal_uInt32 b(0L); b < pPageView->PageWindowCount(); b++)
2423 		{
2424 			// const SdrPageViewWinRec& rPageViewWinRec = rPageViewWinList[b];
2425 			const SdrPageWindow& rPageWindow = *pPageView->GetPageWindow(b);
2426 
2427 			if(rPageWindow.GetPaintWindow().OutputToWindow())
2428 			{
2429 				if(rPageWindow.GetOverlayManager())
2430 				{
2431 					basegfx::B2DPoint aPosition(aPos.X(), aPos.Y());
2432 
2433 					::sdr::overlay::OverlayObject* pOverlayObject = 0L;
2434 
2435 					// animate focused handles
2436 					if(IsFocusHdl() && (pHdlList->GetFocusHdl() == this))
2437 					{
2438 						if( nHdlSize >= 2 )
2439 							nHdlSize = 1;
2440 
2441 						BitmapEx aBmpEx2( GetBitmapForHandle( aHandlesBitmap, nHdlSize + 1 ) );
2442 
2443 						const sal_uInt32 nBlinkTime = sal::static_int_cast<sal_uInt32>(rStyleSettings.GetCursorBlinkTime());
2444 
2445 						pOverlayObject = new ::sdr::overlay::OverlayAnimatedBitmapEx(
2446 							aPosition,
2447 							aBmpEx1,
2448 							aBmpEx2,
2449 							nBlinkTime,
2450 							(sal_uInt16)(aBmpEx1.GetSizePixel().Width() - 1) >> 1,
2451 							(sal_uInt16)(aBmpEx1.GetSizePixel().Height() - 1) >> 1,
2452 							(sal_uInt16)(aBmpEx2.GetSizePixel().Width() - 1) >> 1,
2453 							(sal_uInt16)(aBmpEx2.GetSizePixel().Height() - 1) >> 1,
2454 							mfShearX,
2455 							mfRotation);
2456 					}
2457 					else
2458 					{
2459 						// create centered handle as default
2460 						pOverlayObject = new ::sdr::overlay::OverlayBitmapEx(
2461 							aPosition,
2462 							aBmpEx1,
2463 							(sal_uInt16)(aBmpEx1.GetSizePixel().Width() - 1) >> 1,
2464 							(sal_uInt16)(aBmpEx1.GetSizePixel().Height() - 1) >> 1,
2465 							0.0,
2466 							mfShearX,
2467 							mfRotation);
2468 					}
2469 
2470 					// OVERLAYMANAGER
2471 					if(pOverlayObject)
2472 					{
2473 						rPageWindow.GetOverlayManager()->add(*pOverlayObject);
2474 						maOverlayGroup.append(*pOverlayObject);
2475 					}
2476 				}
2477 			}
2478 		}
2479 	}
2480 }
2481 
2482 ////////////////////////////////////////////////////////////////////////////////////////////////////
2483 // with the correction of crop handling I could get rid of the extra mirroring flag, adapted stuff
2484 // accordingly
2485 
2486 SdrCropViewHdl::SdrCropViewHdl(
2487 	const basegfx::B2DHomMatrix& rObjectTransform,
2488 	const Graphic& rGraphic,
2489 	double fCropLeft,
2490 	double fCropTop,
2491 	double fCropRight,
2492 	double fCropBottom)
2493 :	SdrHdl(Point(), HDL_USER),
2494 	maObjectTransform(rObjectTransform),
2495 	maGraphic(rGraphic),
2496 	mfCropLeft(fCropLeft),
2497 	mfCropTop(fCropTop),
2498 	mfCropRight(fCropRight),
2499 	mfCropBottom(fCropBottom)
2500 {
2501 }
2502 
2503 void SdrCropViewHdl::CreateB2dIAObject()
2504 {
2505 	GetRidOfIAObject();
2506 	SdrMarkView* pView = pHdlList ? pHdlList->GetView() : 0;
2507 	SdrPageView* pPageView = pView ? pView->GetSdrPageView() : 0;
2508 
2509 	if(pPageView && pView->areMarkHandlesHidden())
2510 	{
2511 		return;
2512 	}
2513 
2514 	// decompose to have current translate and scale
2515 	basegfx::B2DVector aScale, aTranslate;
2516 	double fRotate, fShearX;
2517 
2518 	maObjectTransform.decompose(aScale, aTranslate, fRotate, fShearX);
2519 
2520 	if(aScale.equalZero())
2521 	{
2522 		return;
2523 	}
2524 
2525 	// detect 180 degree rotation, this is the same as mirrored in X and Y,
2526 	// thus change to mirroring. Prefer mirroring here. Use the equal call
2527 	// with getSmallValue here, the original which uses rtl::math::approxEqual
2528 	// is too correct here. Maybe this changes with enhanced precision in aw080
2529 	// to the better so that this can be reduced to the more precise call again
2530 	if(basegfx::fTools::equal(fabs(fRotate), F_PI, 0.000000001))
2531 	{
2532 		aScale.setX(aScale.getX() * -1.0);
2533 		aScale.setY(aScale.getY() * -1.0);
2534 		fRotate = 0.0;
2535 	}
2536 
2537 	// remember mirroring, reset at Scale and adapt crop values for usage;
2538 	// mirroring can stay in the object transformation, so do not have to
2539 	// cope with it here (except later for the CroppedImage transformation,
2540 	// see below)
2541 	const bool bMirroredX(aScale.getX() < 0.0);
2542 	const bool bMirroredY(aScale.getY() < 0.0);
2543 	double fCropLeft(mfCropLeft);
2544 	double fCropTop(mfCropTop);
2545 	double fCropRight(mfCropRight);
2546 	double fCropBottom(mfCropBottom);
2547 
2548 	if(bMirroredX)
2549 	{
2550 		aScale.setX(-aScale.getX());
2551 	}
2552 
2553 	if(bMirroredY)
2554 	{
2555 		aScale.setY(-aScale.getY());
2556 	}
2557 
2558 	// create target translate and scale
2559 	const basegfx::B2DVector aTargetScale(
2560 		aScale.getX() + fCropRight + fCropLeft,
2561 		aScale.getY() + fCropBottom + fCropTop);
2562 	const basegfx::B2DVector aTargetTranslate(
2563 		aTranslate.getX() - fCropLeft,
2564 		aTranslate.getY() - fCropTop);
2565 
2566 	// create ranges to make comparisons
2567 	const basegfx::B2DRange aCurrentForCompare(
2568 		aTranslate.getX(), aTranslate.getY(),
2569 		aTranslate.getX() + aScale.getX(), aTranslate.getY() + aScale.getY());
2570 	basegfx::B2DRange aCropped(
2571 		aTargetTranslate.getX(), aTargetTranslate.getY(),
2572 		aTargetTranslate.getX() + aTargetScale.getX(), aTargetTranslate.getY() + aTargetScale.getY());
2573 
2574 	if(aCropped.isEmpty())
2575 	{
2576 		// nothing to return since cropped content is completely empty
2577 		return;
2578 	}
2579 
2580 	if(aCurrentForCompare.equal(aCropped))
2581 	{
2582 		// no crop at all
2583 		return;
2584 	}
2585 
2586 	// back-transform to have values in unit coordinates
2587 	basegfx::B2DHomMatrix aBackToUnit;
2588 	aBackToUnit.translate(-aTranslate.getX(), -aTranslate.getY());
2589 	aBackToUnit.scale(
2590 		basegfx::fTools::equalZero(aScale.getX()) ? 1.0 : 1.0 / aScale.getX(),
2591 		basegfx::fTools::equalZero(aScale.getY()) ? 1.0 : 1.0 / aScale.getY());
2592 
2593 	// transform cropped back to unit coordinates
2594 	aCropped.transform(aBackToUnit);
2595 
2596 	// prepare crop PolyPolygon
2597 	basegfx::B2DPolygon aGraphicOutlinePolygon(
2598 		basegfx::tools::createPolygonFromRect(
2599 			aCropped));
2600 	basegfx::B2DPolyPolygon aCropPolyPolygon(aGraphicOutlinePolygon);
2601 
2602 	// current range is unit range
2603 	basegfx::B2DRange aOverlap(0.0, 0.0, 1.0, 1.0);
2604 
2605 	aOverlap.intersect(aCropped);
2606 
2607 	if(!aOverlap.isEmpty())
2608 	{
2609 		aCropPolyPolygon.append(
2610 			basegfx::tools::createPolygonFromRect(
2611 				aOverlap));
2612 	}
2613 
2614 	// transform to object coordinates to prepare for clip
2615 	aCropPolyPolygon.transform(maObjectTransform);
2616 	aGraphicOutlinePolygon.transform(maObjectTransform);
2617 
2618 	// create cropped transformation
2619 	basegfx::B2DHomMatrix aCroppedTransform;
2620 	const bool bCombinedMirrorX(bMirroredX);
2621 
2622 	aCroppedTransform.scale(
2623 		aCropped.getWidth(),
2624 		aCropped.getHeight());
2625 	aCroppedTransform.translate(
2626 		aCropped.getMinX(),
2627 		aCropped.getMinY());
2628 	aCroppedTransform = maObjectTransform * aCroppedTransform;
2629 
2630 	// prepare graphic primitive (transformed)
2631 	const drawinglayer::primitive2d::Primitive2DReference aGraphic(
2632 		new drawinglayer::primitive2d::GraphicPrimitive2D(
2633 			aCroppedTransform,
2634 			maGraphic));
2635 
2636 	// prepare outline polygon for whole graphic
2637 	const SvtOptionsDrawinglayer aSvtOptionsDrawinglayer;
2638 	const basegfx::BColor aHilightColor(aSvtOptionsDrawinglayer.getHilightColor().getBColor());
2639 	const drawinglayer::primitive2d::Primitive2DReference aGraphicOutline(
2640 		new drawinglayer::primitive2d::PolygonHairlinePrimitive2D(
2641 		aGraphicOutlinePolygon,
2642 		aHilightColor));
2643 
2644 	// combine these
2645 	drawinglayer::primitive2d::Primitive2DSequence aCombination(2);
2646 	aCombination[0] = aGraphic;
2647 	aCombination[1] = aGraphicOutline;
2648 
2649 	// embed to MaskPrimitive2D
2650 	const drawinglayer::primitive2d::Primitive2DReference aMaskedGraphic(
2651 		new drawinglayer::primitive2d::MaskPrimitive2D(
2652 			aCropPolyPolygon,
2653 			aCombination));
2654 
2655 	// embed to UnifiedTransparencePrimitive2D
2656 	const drawinglayer::primitive2d::Primitive2DReference aTransparenceMaskedGraphic(
2657 		new drawinglayer::primitive2d::UnifiedTransparencePrimitive2D(
2658 			drawinglayer::primitive2d::Primitive2DSequence(&aMaskedGraphic, 1),
2659 			0.8));
2660 
2661 	const drawinglayer::primitive2d::Primitive2DSequence aSequence(&aTransparenceMaskedGraphic, 1);
2662 
2663 	for(sal_uInt32 b(0L); b < pPageView->PageWindowCount(); b++)
2664 	{
2665 		// const SdrPageViewWinRec& rPageViewWinRec = rPageViewWinList[b];
2666 		const SdrPageWindow& rPageWindow = *pPageView->GetPageWindow(b);
2667 
2668 		if(rPageWindow.GetPaintWindow().OutputToWindow())
2669 		{
2670 			if(rPageWindow.GetOverlayManager())
2671 			{
2672 				::sdr::overlay::OverlayObject* pNew = new sdr::overlay::OverlayPrimitive2DSequenceObject(aSequence);
2673 				DBG_ASSERT(pNew, "Got NO new IAO!");
2674 
2675 				if(pNew)
2676 				{
2677 					// only informative object, no hit
2678 					pNew->setHittable(false);
2679 
2680 					rPageWindow.GetOverlayManager()->add(*pNew);
2681 					maOverlayGroup.append(*pNew);
2682 				}
2683 			}
2684 		}
2685 	}
2686 }
2687 
2688 ////////////////////////////////////////////////////////////////////////////////////////////////////
2689 // eof
2690