os/graphics/graphicsdeviceinterface/directgdiadaptation/swsrc/swdirectgdipiearc.cpp
Update contrib.
1 // Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
16 #include "swdirectgdiengine.h"
17 #include "swdirectgdiellipse.h"
21 A utility class to efficiently draw arcs.
22 @see CSwDirectGdiEngine::PieArcOutline
26 class TSwDirectGdiArc : public TSwDirectGdiEllipse
29 void Construct(const TRect& aRect,const TPoint& aStart,const TPoint& aEnd);
30 TBool SingleStep(TPoint& aTl,TBool& aDoTl,TPoint& aTr,TBool& aDoTr,TPoint& aBl,TBool& aDoBl,TPoint& aBr,TBool& aDoBr);
31 void Step(TPoint& aTl,TBool& aDoTl,TPoint& aTr,TBool& aDoTr,TPoint& aBl,TBool& aDoBl,TPoint& aBr,TBool& aDoBr);
40 TBool iStartquadenabled;
41 TBool iEndquadenabled;
50 Constructs a TSwDirectGdiArc.
51 @param aRect The bounding rectangle.
52 @param aStart The arc start point.
53 @param aEnd The arc end point.
54 @see CSwDirectGdiEngine::PieArcOutline
56 void TSwDirectGdiArc::Construct(const TRect& aRect,const TPoint& aStart,const TPoint& aEnd)
58 iStart=Intersection(aRect,aStart);
59 iEnd=Intersection(aRect,aEnd);
60 TSwDirectGdiEllipse::Construct(aRect);
65 iStartquadenabled=EFalse;
66 iEndquadenabled=EFalse;
70 TPoint center=aRect.Center();
71 if (iStart.iX>=center.iX) iStartquad=1;
72 if (iStart.iY>=center.iY) iStartquad+=2;
73 if (iEnd.iX>=center.iX) iEndquad=1;
74 if (iEnd.iY>=center.iY) iEndquad+=2;
75 if (iStartquad!=iEndquad)
77 if (iStartquad==0 && iEndquad==1)
82 else if (iStartquad==0 && iEndquad==3)
84 else if (iStartquad==1 && iEndquad==2)
86 else if (iStartquad==1 && iEndquad==3)
91 else if (iStartquad==2 && iEndquad==0)
96 else if (iStartquad==2 && iEndquad==1)
98 else if (iStartquad==3 && iEndquad==0)
100 else if (iStartquad==3 && iEndquad==2)
106 else if (iStart==iEnd)
116 if (iStartquad==0 && (iStart.iX<iEnd.iX || iStart.iY>iEnd.iY))
123 else if (iStartquad==1 && (iStart.iX<iEnd.iX || iStart.iY<iEnd.iY))
130 else if (iStartquad==2 && (iStart.iX>iEnd.iX || iStart.iY>iEnd.iY))
137 else if (iStartquad==3 && (iStart.iX>iEnd.iX || iStart.iY<iEnd.iY))
145 if (iStartquad==1 || iStartquad==2)
146 iStartquadenabled=ETrue;
147 if (iEndquad==0 || iEndquad==3)
148 iEndquadenabled=ETrue;
149 iStartquaddone=EFalse;
154 Produces the next stage creating an arc, taking four points (the corners of
155 the rectangle the arc should fill) as parameters.
157 @param aTopLeft Top left corner of rectangle.
158 @param aDoTl Span the top left quadrant.
159 @param aTopRight Top right corner of rectangle.
160 @param aDoTr Span the top right quadrant.
161 @param aBottomLeft Bottom left corner of rectangle.
162 @param aDoBl Span the bottom left quadrant.
163 @param aBottomRight Bottom right corner of rectangle.
164 @param aDoBr Span the bottom right quadrant.
165 @return TBool ETrue if step completed successfully.
166 @see TSwDirectGdiEllipse::SingleStep
168 TBool TSwDirectGdiArc::SingleStep(TPoint& aTopLeft,TBool& aDoTl,TPoint& aTopRight,
169 TBool& aDoTr,TPoint& aBottomLeft,TBool& aDoBl,
170 TPoint& aBottomRight,TBool& aDoBr)
172 TBool finished=TSwDirectGdiEllipse::SingleStep(aTopLeft,aTopRight,aBottomLeft,aBottomRight);
173 Step(aTopLeft,aDoTl,aTopRight,aDoTr,aBottomLeft,aDoBl,aBottomRight,aDoBr);
179 Determines how many quadrants are left to draw.
181 @param aTopLeft Top left corner of rectangle.
182 @param aDoTl Span the top left quadrant.
183 @param aTopRight Top right corner of rectangle.
184 @param aDoTr Span the top right quadrant.
185 @param aBottomLeft Bottom left corner of rectangle.
186 @param aDoBl Span the bottom left quadrant.
187 @param aBottomRight Bottom right corner of rectangle.
188 @param aDoBr Span the bottom right quadrant.
189 @see TSwDirectGdiArc::SingleStep
190 @see TSwDirectGdiArc::Construct
192 void TSwDirectGdiArc::Step(TPoint& aTopLeft,TBool& aDoTl,TPoint& aTopRight,TBool& aDoTr,
193 TPoint& aBottomLeft,TBool& aDoBl,TPoint& aBottomRight,TBool& aDoBr)
201 if (!iStartquadenabled)
203 if (iStartquad==0 && aTopLeft.iX<=iStart.iX && aTopLeft.iY>=iStart.iY)
205 iStartquadenabled=ETrue;
206 iStartquaddone=ETrue;
208 if (iStartquad==3 && aBottomRight.iX>=iStart.iX && aBottomRight.iY<=iStart.iY)
210 iStartquadenabled=ETrue;
211 iStartquaddone=ETrue;
216 if (iStartquad==1 && (aTopRight.iX>iStart.iX || aTopRight.iY>iStart.iY))
218 iStartquadenabled=EFalse;
219 iStartquaddone=ETrue;
221 if (iStartquad==2 && (aBottomLeft.iX<iStart.iX || aBottomLeft.iY<iStart.iY))
223 iStartquadenabled=EFalse;
224 iStartquaddone=ETrue;
232 if (iEndquad==0 && (aTopLeft.iX<iEnd.iX || aTopLeft.iY>iEnd.iY))
234 iEndquadenabled=EFalse;
237 if (iEndquad==3 && (aBottomRight.iX>iEnd.iX || aBottomRight.iY<iEnd.iY))
239 iEndquadenabled=EFalse;
245 if (iEndquad==1 && aTopRight.iX>=iEnd.iX && aTopRight.iY>=iEnd.iY)
247 iEndquadenabled=ETrue;
250 if (iEndquad==2 && aBottomLeft.iX<=iEnd.iX && aBottomLeft.iY<=iEnd.iY)
252 iEndquadenabled=ETrue;
257 if (iStartquad!=iEndquad)
259 if (iStartquadenabled)
261 if (iStartquad==0) aDoTl=ETrue;
262 else if (iStartquad==1) aDoTr=ETrue;
263 else if (iStartquad==2) aDoBl=ETrue;
264 else if (iStartquad==3) aDoBr=ETrue;
268 if (iEndquad==0) aDoTl=ETrue;
269 else if (iEndquad==1) aDoTr=ETrue;
270 else if (iEndquad==2) aDoBl=ETrue;
271 else if (iEndquad==3) aDoBr=ETrue;
278 if (iStartquadenabled && iEndquadenabled)
280 if (iStartquad==0) aDoTl=ETrue;
281 else if (iStartquad==1) aDoTr=ETrue;
282 else if (iStartquad==2) aDoBl=ETrue;
283 else if (iStartquad==3) aDoBr=ETrue;
288 if (iStartquadenabled || iEndquadenabled)
290 if (iStartquad==0) aDoTl=ETrue;
291 else if (iStartquad==1) aDoTr=ETrue;
292 else if (iStartquad==2) aDoBl=ETrue;
293 else if (iStartquad==3) aDoBr=ETrue;
297 if (aTopLeft.iX==aTopRight.iX)
299 if (aDoTl && aDoTr) aDoTr=EFalse;
300 if (aDoBl && aDoBr) aDoBr=EFalse;
305 // Pie and Arc drawing functions
309 @see MDirectGdiEngine::DrawArc()
311 void CSwDirectGdiEngine::DrawArc(const TRect& aRect,const TPoint& aStart,
317 TRect targetRect(rcpy);
318 targetRect.Grow((iPenSize.iWidth>>1)+1,(iPenSize.iHeight>>1)+1);
319 PieArcOutline(rcpy,aStart+iOrigin,aEnd+iOrigin,EFalse);
323 @see MDirectGdiEngine::DrawPie()
325 void CSwDirectGdiEngine::DrawPie(const TRect& aRect,const TPoint& aStart,
331 TPoint startIntersect = aStart + iOrigin, endIntersect = aEnd + iOrigin;
332 TInt startQuadrant, endQuadrant;
334 const TBool isEllipse = AnalyseEllipse(
335 rcpy, startIntersect, endIntersect, startQuadrant, endQuadrant, quadrants);
337 if (iBrushStyle!=DirectGdi::ENullBrush)
342 PieFill(rcpy, startIntersect, endIntersect, startQuadrant, endQuadrant, quadrants);
344 if ((iPenStyle!=DirectGdi::ENullPen) && (iPenSize.iWidth>0) && (iPenSize.iHeight>0))
345 PieArcOutline(rcpy,aStart+iOrigin,aEnd+iOrigin,ETrue);
349 Calculates which quadrants are completely filled and which quadrants contain the
350 start and end points.
352 @param aRect The bounding rectangle.
353 @param aStart The arc start point.
354 @param aEnd The arc end point.
355 @param aStartQuad On return, contains the quadrant which contains aStart.
356 @param aEndQuad On return, contains the quadrant which contains aEnd.
357 @param aQuads On return, populates an array of full quadrants to fill.
358 @pre Input params aRect, aStart, aEnd are to be given.
359 @post Output params aStart, aEnd, aStartQuad, aEndQuad, aQuads will be populated.
360 @return ETrue if the pie is an ellipse, otherwise EFalse.
362 TBool CSwDirectGdiEngine::AnalyseEllipse(const TRect& aRect,TPoint& aStart,TPoint& aEnd,
363 TInt& aStartQuad,TInt& aEndQuad,TBool* aQuads)
367 const TPoint center = aRect.Center();
368 TSwDirectGdiEllipse ellipse;
369 aStart=ellipse.Intersection(aRect,aStart);
370 aEnd=ellipse.Intersection(aRect,aEnd);
380 const TInt startx = aStart.iX - center.iX, starty = aStart.iY - center.iY;
381 const TInt endx = aEnd.iX - center.iX, endy = aEnd.iY - center.iY;
382 if (startx>=0) aStartQuad=1;
383 if (starty>=0) aStartQuad+=2;
384 if (endx>=0) aEndQuad=1;
385 if (endy>=0) aEndQuad+=2;
386 aQuads[1]=EFalse,aQuads[2]=EFalse,aQuads[3]=EFalse,aQuads[4]=EFalse; // complete quadrants to draw
387 aQuads[0]=EFalse; // ellipse is a sliver completely within a quadrant
388 if (aStartQuad==aEndQuad)
390 if (aStartQuad==0 && (startx<endx || starty>endy))
396 else if (aStartQuad==1 && (startx<endx || starty<endy))
402 else if (aStartQuad==2 && (startx>endx || starty>endy))
408 else if (aStartQuad==3 && (startx>endx || starty<endy))
414 else aQuads[0]=ETrue; // "slice"
418 if (aStartQuad==0 && aEndQuad==1)
423 else if (aStartQuad==0 && aEndQuad==3)
425 else if (aStartQuad==1 && aEndQuad==2)
427 else if (aStartQuad==1 && aEndQuad==3)
432 else if (aStartQuad==2 && aEndQuad==0)
437 else if (aStartQuad==2 && aEndQuad==1)
439 else if (aStartQuad==3 && aEndQuad==0)
441 else if (aStartQuad==3 && aEndQuad==2)
451 Draws the outline of a pie or arc.
453 @param aRect The bounding rectangle.
454 @param aStart The arc start point.
455 @param aEnd The arc end point.
456 @param aPie If this is the outline for a pie ETrue, else EFalse
458 void CSwDirectGdiEngine::PieArcOutline(const TRect& aRect,const TPoint& aStart,
459 const TPoint& aEnd,TBool aPie)
461 // arc runs counterclockwise, from startradius-center/ellipse intersect to endradius-center/ellipse intersect
462 TInt dotparam=iDotParam;
465 TInt halfpenwidth=iPenSize.iWidth>>1;
466 TInt halfpenheight=iPenSize.iHeight>>1;
467 TInt otherhalfwidth=(iPenSize.iWidth+1)>>1;
468 TInt otherhalfheight=(iPenSize.iHeight+1)>>1;
469 rcpy.iTl.iX-=halfpenwidth;
470 rcpy.iTl.iY-=halfpenheight;
471 rcpy.iBr.iX+=otherhalfwidth;
472 rcpy.iBr.iY+=otherhalfheight;
473 TRect clipRect(0,0,0,0);
474 for(TInt count=0;count<iDefaultRegionPtr->Count();count++)
476 clipRect=(*iDefaultRegionPtr)[count];
477 if (!clipRect.Intersects(rcpy))
479 clipRect.Intersection(rcpy);
481 arc.Construct(aRect,aStart,aEnd);
482 iDotParam=Max(iPenSize.iWidth>>1,iPenSize.iHeight>>1);
483 TBool dotl,dotr,dobl,dobr;
484 while(!arc.SingleStep(pt[0],dotl,pt[1],dotr,pt[2],dobl,pt[3],dobr))
486 if (dotl) PenDrawClipped(pt[0], clipRect);
487 if (dotr) PenDrawClipped(pt[1], clipRect);
488 if (dobl) PenDrawClipped(pt[2], clipRect);
489 if (dobr) PenDrawClipped(pt[3], clipRect);
490 iDotParam+=iDotDirection;
492 if (pt[0].iY==pt[2].iY)
494 if (dotl) PenDrawClipped(pt[0], clipRect);
495 if (dotr) PenDrawClipped(pt[1], clipRect);
500 const TPoint center = aRect.Center();
502 line.Construct(arc.iStart,center);
503 line.SingleStep(temp);
504 while(!line.SingleStep(temp))
506 PenDrawClipped(temp, clipRect);
507 iDotParam+=iDotDirection;
509 line.Construct(arc.iEnd,center);
510 line.SingleStep(temp);
511 while(!line.SingleStep(temp))
513 PenDrawClipped(temp, clipRect);
514 iDotParam+=iDotDirection;
516 PenDrawClipped(center, clipRect);
518 iDrawDevice->UpdateRegion(clipRect);
526 @param aRect The bounding rectangle.
527 @param aStart The arc start point.
528 @param aEnd The arc end point.
529 @param aStartQuad The quadrant which contains aStart.
530 @param aEndQuad The quadrant which contains aEnd.
531 @param aQuads An array of full quadrants to fill.
532 @see CSwDirectGdiEngine::DrawPie
533 @see CSwDirectGdiEngine::AnalyseEllipse
535 void CSwDirectGdiEngine::PieFill(const TRect& aRect, const TPoint& aStart, const TPoint& aEnd,
536 TInt aStartQuad, TInt aEndQuad, const TBool* aQuads)
538 // arc runs counterclockwise, from startradius-center/ellipse intersect to endradius-center/ellipse intersect
539 const TInt limit = iDefaultRegionPtr->Count();
540 TRect clipRect(0,0,0,0);
541 for(TInt count=0;count<limit;count++)
543 clipRect=(*iDefaultRegionPtr)[count];
544 if (!clipRect.Intersects(aRect))
548 clipRect.Intersection(aRect);
551 PieShell(aRect, aStart, aEnd, aQuads, aStartQuad, aEndQuad, clipRect);
555 PieSliver(aRect, aStart, aEnd, aStartQuad, clipRect);
557 iDrawDevice->UpdateRegion(clipRect);
562 Draws a pie spanning one or more quadrants.
564 @param aRect The bounding rectangle.
565 @param aStart The arc start point.
566 @param aEnd The arc end point.
567 @param aQuads Array of full quadrants to fill.
568 @param aStartQuad The quadrant which contains aStart.
569 @param aEndQuad The quadrant which contains aEnd.
570 @param aClipRect The rectangle to which the pie is clipped.
572 void CSwDirectGdiEngine::PieShell(const TRect& aRect,const TPoint& aStart,
573 const TPoint& aEnd, const TBool* aQuads, TInt aStartQuad,
574 TInt aEndQuad, TRect aClipRect)
576 TSwDirectGdiEllipse ellipse;
577 ellipse.Construct(aRect);
578 TInt c=aRect.Center().iX;
581 TBool donestart=EFalse;
582 TBool doneend=EFalse;
583 TBool todostart=EFalse;
584 TBool todoend=EFalse;
585 while(!ellipse.NextStep(pt[0],pt[1],pt[2],pt[3]))
587 if (pt[aStartQuad].iY==aStart.iY)
589 if (pt[aEndQuad].iY==aEnd.iY)
599 if (aQuads[1] || (aEndQuad==0 && !doneend)) // tl quadrant, half top end chord
602 if (aQuads[2] || (aStartQuad==1 && !donestart)) // tr quadrant, half top start chord
605 if (aQuads[3] || (aStartQuad==2 && !donestart)) // bl quadrant, half top start chord
608 if (aQuads[4] || (aEndQuad==3 && !doneend)) // br quadrant, half top end chord
611 ClipFillLine(tl,tr,aClipRect);
612 ClipFillLine(bl,br,aClipRect);
613 // do partial quadrants
620 ClipFillLine(tl,tr,aClipRect);
622 else if (aStartQuad==3)
626 ClipFillLine(bl,br,aClipRect);
635 ClipFillLine(bl,br,aClipRect);
637 else if (aEndQuad==1)
641 ClipFillLine(tl,tr,aClipRect);
649 if (pt[0].iY==pt[2].iY) // congruent mid lines
651 if (pt[aStartQuad].iY==aStart.iY)
653 if (pt[aEndQuad].iY==aEnd.iY)
659 TBool leftflag=EFalse;
660 TBool rightflag=EFalse;
661 if (aQuads[1] || (aEndQuad==0 && !doneend) ||
662 aQuads[3] || (aStartQuad==2 && !donestart) ||
663 (todostart && aStartQuad==0) || (todoend && aEndQuad==2))
665 if (aQuads[2] || (aStartQuad==1 && !donestart) ||
666 aQuads[4] || (aEndQuad==3 && !doneend) ||
667 (todostart && aStartQuad==3) || (todoend && aEndQuad==1))
669 if (leftflag) tl.iX=pt[0].iX;
670 if (rightflag) tr.iX=pt[1].iX;
671 ClipFillLine(tl,tr,aClipRect);
675 tl.iY=aRect.Center().iY;
677 if (aStartQuad==3) tr.iX=aStart.iX-1;
678 if (aEndQuad==2) tl.iX=aEnd.iX+1;
679 ClipFillLine(tl,tr,aClipRect);
681 PieTriangles(aStartQuad==1 || aStartQuad==2,aStart,aRect.Center(), aClipRect);
682 PieTriangles(aEndQuad==0 || aEndQuad==3,aEnd,aRect.Center(), aClipRect);
686 Draws a filled triangle which forms part of a partially filled quadrant.
688 @param aInside A boolean value indicating which side of the line to fill.
689 @param aStart The start of the line which forms the hypotenuse of the triangle.
690 @param aEnd The end of the line which forms the hypotenuse of the triangle.
691 @param aClipRect The rectangle to which the pie is clipped.
692 @see CSwDirectGdiEngine::PieShell
694 void CSwDirectGdiEngine::PieTriangles(TBool aInside,const TPoint& aStart,const TPoint& aEnd, TRect aClipRect)
696 TInt x=aInside?aEnd.iX:aStart.iX;
697 if (aStart.iX>aEnd.iX)
716 TPoint pt,left,right;
717 line.Construct(aStart,aEnd);
719 while(!line.NextStep(pt))
721 if (pt.iY==aEnd.iY) break;
722 left.iX=Min(pt.iX,x);
723 right.iX=Max(pt.iX,x);
724 left.iY=right.iY=pt.iY;
725 ClipFillLine(left,right,aClipRect);
730 Draws a pie entirely contained in one quadrant.
732 @param aRect The rectangle in which to draw the ellipse.
733 @param aStart The arc start point.
734 @param aEnd The arc end point.
735 @param aQuad The quadrant containing the pie.
736 @param aClipRect The rectangle to which the pie is clipped.
738 void CSwDirectGdiEngine::PieSliver(const TRect& aRect,const TPoint& aStart,
739 const TPoint& aEnd, TInt aQuad, TRect aClipRect)
741 TPoint center=aRect.Center(),left,right;
742 TPoint nearinter(aStart),farinter(aEnd);
743 if (Abs(nearinter.iY-center.iY)>Abs(farinter.iY-center.iY))
748 TBool ellipseComplete = EFalse;
750 TSwDirectGdiEllipse ellipse;
751 ellipse.Construct(aRect);
753 mainline.Construct(farinter,center);
754 ellipseComplete = ellipse.SingleStep(pt[0],pt[1],pt[2],pt[3]);
755 mainline.SingleStep(right);
757 while(!ellipseComplete && pt[aQuad].iY!=right.iY)
758 ellipseComplete = ellipse.NextStep(pt[0],pt[1],pt[2],pt[3]);
760 while(!ellipseComplete && pt[aQuad].iY==right.iY)
763 ellipseComplete = ellipse.NextStep(pt[0],pt[1],pt[2],pt[3]);
765 if (right.iY==nearinter.iY || (ellipseComplete && (pt[0].iY != pt[2].iY)))
767 if (left.iX>right.iX)
773 if(right==farinter && left.iX<=right.iX)
779 if (left.iX<=right.iX)
780 ClipFillLine(left,right,aClipRect);
782 while(!mainline.NextStep(right));
785 line.Construct(nearinter,center);
786 TBool linestat=EFalse;
788 linestat=line.SingleStep(temppt);
789 while(temppt.iY!=right.iY && !linestat);
793 linestat=line.SingleStep(temppt);
795 while(temppt.iY==right.iY && !linestat);
798 if (left.iX>right.iX)
804 if(right==farinter && left.iX<=right.iX)
810 if (left.iX<=right.iX)
811 ClipFillLine(left,right,aClipRect);
813 while(!mainline.NextStep(right));