os/graphics/graphicsdeviceinterface/directgdiadaptation/swsrc/swdirectgdiline.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 "swdirectgdipolygon.h"
20 @see MDirectGdiEngine::DrawLine()
22 void CSwDirectGdiEngine::DrawLine(const TPoint& aPt1,const TPoint& aPt2)
24 DoDrawLine(aPt1,aPt2,ETrue);
28 @see MDirectGdiEngine::DrawLineTo()
30 void CSwDirectGdiEngine::DrawLineTo(const TPoint& aPoint)
32 DrawLine(iLinePosition,aPoint);
37 @see MDirectGdiEngine::DrawLineBy()
39 void CSwDirectGdiEngine::DrawLineBy(const TPoint& aVector)
41 DrawLine(iLinePosition,iLinePosition + aVector);
45 @see MDirectGdiEngine::DrawPolyLine()
47 @panic DGDIAdapter 27, if the passed point list has too few points (debug only).
49 void CSwDirectGdiEngine::DrawPolyLine(const TArray<TPoint>& aPointList)
51 DrawPolyLineNoEndPoint(aPointList);
53 if (iPenStyle == DirectGdi::ESolidPen)
55 Plot(aPointList[aPointList.Count()-1]);
60 @see MDirectGdiEngine::DrawPolyLineNoEndPoint()
62 @panic DGDIAdapter 27, if the passed point list has too few points (debug only).
64 void CSwDirectGdiEngine::DrawPolyLineNoEndPoint(const TArray<TPoint>& aPointList)
66 GRAPHICS_ASSERT_DEBUG(aPointList.Count() > 0, EDirectGdiPanicInvalidPointArray);
68 const TInt vertexes = aPointList.Count()-1;
70 for (TInt count = 0; count < vertexes; count++)
72 DrawLine(aPointList[count], aPointList[count + 1]);
77 @see MDirectGdiEngine::DrawPolygon()
79 void CSwDirectGdiEngine::DrawPolygon(const TArray<TPoint>& aPointList, DirectGdi::TFillRule aFillRule)
81 const TInt numpoints = aPointList.Count();
83 if (iBrushStyle != DirectGdi::ENullBrush)
85 TRect pointrect(0,0,0,0);
86 TRect truncrect(0,0,0,0);
87 TBool largepolygon = EFalse;
89 for (TInt count = 0; count < numpoints; count++)
91 pointrect.iTl = aPointList[count] + iOrigin;
92 truncrect.iTl = pointrect.iTl;
93 TruncateRect(truncrect);
95 if (pointrect.iTl != truncrect.iTl)
104 PolyFillLarge(&aPointList, aFillRule);
108 PolyFill(&aPointList, aFillRule);
112 if (iPenStyle != DirectGdi::ENullPen)
114 if (iPenSize.iWidth > 0 && iPenSize.iHeight > 0)
116 PolyOutline(&aPointList);
122 Draws a straight line from the start to the end position using current pen size, colour and style.
124 @param aPt1 Start position.
125 @param aPt2 End position.
126 @param aDrawStartPoint If ETrue, draws the first pixel of the line.
128 @post The internal drawing position is set to the line's endpoint.
129 @see CSwDirectGdiEngine::DrawLine()
131 void CSwDirectGdiEngine::DoDrawLine(TPoint aPt1, TPoint aPt2, TBool aDrawStartPoint)
133 iLinePosition = aPt2;
143 TRect temp(aPt1,aPt2);
145 temp.Grow(iPenSize.iWidth, iPenSize.iHeight);
148 iDrawDevice->GetDrawRect(screenRect);
149 screenRect.Grow(iPenSize.iWidth, iPenSize.iHeight);
151 const TInt dotParam = iDotParam;
153 const CGraphicsContext::TDrawMode drawMode = GcDrawMode(iDrawMode);
155 TRect clipRect(0,0,0,0);
156 for (TInt count = 0; count < iDefaultRegionPtr->Count(); count++)
158 iDotParam = dotParam;
159 clipRect = (*iDefaultRegionPtr)[count];
161 if (!clipRect.Intersects(temp))
164 line.Construct(aPt1,aPt2);
165 line.JumpToRect(screenRect);
166 if (iPenStyle != DirectGdi::ESolidPen)
168 while (!line.SingleStep(plotpt))
170 iDotParam += iDotDirection;
176 clipRect.Intersection(temp);
178 if ((iPenSize.iWidth > 1 || iPenSize.iHeight > 1) && (iPenStyle == DirectGdi::ESolidPen)) // wide solid line
180 DoDrawSolidWideLine(aPt1, aPt2, aDrawStartPoint, screenRect, clipRect);
182 else if (iPenSize.iWidth > 1 || iPenSize.iHeight > 1) // dotted line
184 DoDrawDottedWideLine(aPt1, aPt2, aDrawStartPoint, screenRect, clipRect);
186 else if (iPenStyle != DirectGdi::ESolidPen) // single pixel dotted line
189 line.Construct(aPt1,aPt2);
190 line.JumpToRect(screenRect);
192 iDotParam = dotParam;
193 if (!aDrawStartPoint)
195 line.SingleStep(plotpt);
196 iDotParam += iDotDirection;
199 while (!line.SingleStep(plotpt))
201 PenDrawClipped(plotpt, clipRect);
202 iDotParam += iDotDirection;
205 else if (aPt1.iY == aPt2.iY &&
206 (aPt1.iY >= clipRect.iTl.iY &&
207 aPt1.iY < clipRect.iBr.iY))
208 { // single pixel solid horizontal line
209 TInt start = Min(aPt1.iX,aPt2.iX + 1);
210 TInt length = Abs(aPt2.iX - aPt1.iX);
212 if (!aDrawStartPoint)
214 if (aPt1.iX < aPt2.iX)
223 if (start < clipRect.iTl.iX)
225 length += start - clipRect.iTl.iX;
226 start = clipRect.iTl.iX;
228 if ( (start + length) > clipRect.iBr.iX)
230 length = clipRect.iBr.iX - start;
235 iDrawDevice->WriteRgbMulti(start, aPt1.iY, length, 1, iPenColor, drawMode);
238 else if (aPt1.iX == aPt2.iX && (aPt1.iX >= clipRect.iTl.iX && aPt1.iX < clipRect.iBr.iX))
239 { // single pixel solid vertical line
240 TInt start = Min(aPt1.iY,aPt2.iY + 1);
241 TInt length = Abs(aPt2.iY - aPt1.iY);
243 if (!aDrawStartPoint)
245 if (aPt1.iY < aPt2.iY)
255 if (start < clipRect.iTl.iY)
257 length += start - clipRect.iTl.iY;
258 start = clipRect.iTl.iY;
260 if (start + length > clipRect.iBr.iY)
262 length = clipRect.iBr.iY - start;
267 iDrawDevice->WriteRgbMulti(aPt1.iX,start,1,length,iPenColor, drawMode);
271 { // single pixel solid diagonal line
273 line.Construct(aPt1,aPt2);
275 line.JumpToRect(screenRect);
277 if (!aDrawStartPoint)
279 line.SingleStep(plotpt);
282 while (!line.SingleStep(plotpt))
284 if (clipRect.Contains(plotpt))
286 iDrawDevice->WriteRgb(plotpt.iX, plotpt.iY, iPenColor, drawMode);
291 iDrawDevice->UpdateRegion(clipRect);
296 Draws a straight line from the start to the end position using pen sizes larger than 1x1 pixel.
298 @param aPt1 Start position.
299 @param aPt2 End position.
300 @param aDrawStartPoint If ETrue, draws the first pixel of the line.
301 @param aScreenRect Rectangle representing the screen boundary.
302 @param aClipRect The rectangle to which the line is clipped.
303 @see CSwDirectGdiEngine::DrawLine()
305 void CSwDirectGdiEngine::DoDrawSolidWideLine(const TPoint& aPt1,
307 TBool aDrawStartPoint,
308 const TRect& aScreenRect,
311 CFbsDrawDevice* drawDevice = iDrawDevice;
314 line.Construct(aPt1,aPt2);
317 line.JumpToRect(aScreenRect);
318 if (!aDrawStartPoint)
319 line.SingleStep(plotpt);
321 TInt* deferred = NULL;
322 const TInt doubleheight = iPenSize.iHeight << 1;
326 deferred = new TInt[doubleheight];
329 if (!iPenArray || !deferred)
331 while (!line.SingleStep(plotpt))
332 PenDrawClipped(plotpt, aClipRect);
336 const TBool down = (aPt2.iY >= aPt1.iY);
338 for (TInt fillcount = 0; fillcount < doubleheight; )
340 deferred[fillcount++] = KMaxTInt;
341 deferred[fillcount++] = KMinTInt;
345 TInt nexty = plotpt.iY;
348 nexty -= ((iPenSize.iHeight - 1) >> 1);
352 nexty += (iPenSize.iHeight >> 1);
355 TInt lasty = plotpt.iY;
357 while (!line.SingleStep(plotpt))
359 if (plotpt.iY != lasty)
361 if (nexty >= aClipRect.iTl.iY && nexty < aClipRect.iBr.iY)
363 TInt left = deferred[nextline];
364 TInt right = deferred[nextline + 1];
365 if (left < aClipRect.iTl.iX)
367 left = aClipRect.iTl.iX;
369 if (right >= aClipRect.iBr.iX)
371 right = aClipRect.iBr.iX - 1;
376 drawDevice->WriteRgbMulti(left,nexty,right - left + 1,1,iPenColor,CGraphicsContext::EDrawModePEN);
390 deferred[nextline++] = KMaxTInt;
391 deferred[nextline++] = KMinTInt;
392 if (nextline == doubleheight)
398 PenDrawDeferred(plotpt,deferred,nextline);
401 for (TInt restofline = 0; restofline < doubleheight; restofline += 2,nextline += 2)
403 if (nextline == doubleheight)
406 if (nexty >= aClipRect.iTl.iY && nexty < aClipRect.iBr.iY)
408 TInt left = deferred[nextline];
409 TInt right = deferred[nextline+1];
410 if (left < aClipRect.iTl.iX)
412 left = aClipRect.iTl.iX;
414 if (right >= aClipRect.iBr.iX)
416 right = aClipRect.iBr.iX-1;
421 drawDevice->WriteRgbMulti(left,nexty,right - left + 1,1,iPenColor,CGraphicsContext::EDrawModePEN);
440 Draws a dotted straight line from the start to the end position using pen sizes larger than 1x1 pixel.
442 @param aPt1 Start position.
443 @param aPt2 End position.
444 @param aDrawStartPoint If ETrue, draws the first pixel of the line.
445 @param aScreenRect Rectangle representing the screen boundary.
446 @param aClipRect The rectangle to which the line is clipped.
447 @see CSwDirectGdiEngine::DrawLine()
449 void CSwDirectGdiEngine::DoDrawDottedWideLine(const TPoint& aPt1,
451 TBool aDrawStartPoint,
452 const TRect& aScreenRect,
456 line.Construct(aPt1,aPt2);
459 line.JumpToRect(aScreenRect);
460 if (!aDrawStartPoint)
462 line.SingleStep(plotpt);
463 iDotParam += iDotDirection;
466 const TInt maxdim = Max(iPenSize.iWidth, iPenSize.iHeight);
471 while (!done && !(iDotMask & (1 << ((iDotParam / maxdim) % iDotLength))))
473 done = line.SingleStep(plotpt);
474 iDotParam += iDotDirection;
477 TPoint startdash(plotpt);
478 TPoint enddash(plotpt);
480 while (!done && (iDotMask & (1 << ((iDotParam / maxdim) % iDotLength))))
483 done = line.SingleStep(plotpt);
484 iDotParam += iDotDirection;
487 DoDrawSolidWideLine(startdash,enddash,ETrue,aScreenRect,aClipRect);
492 Fills a polygon defined using an array of points. The first point in the array defines the
493 start of the first side of the polygon. The final side of the polygon is drawn using the last point
494 from the array. The area is filled with the current brush settings.
496 Self-crossing polygons are filled according to the specified fill rule.
498 @param aPointList Array of points specifying the vertices of the polygon.
499 @param aFillRule Polygon filling rule.
501 void CSwDirectGdiEngine::PolyFill(const TArray<TPoint>* aPointList, DirectGdi::TFillRule aFillRule)
508 TRect clipRect(0,0,0,0);
509 const TInt limit = iDefaultRegionPtr->Count();
510 for (TInt count = 0; count < limit; count++)
512 clipRect = (*iDefaultRegionPtr)[count];
513 CSwDirectGdiPolygonFiller polyfill;
514 polyfill.Construct(aPointList,aFillRule);
516 for(polyfill.GetNextPixelRun(exists,scanline,pixelRunStart,pixelRunEnd);exists;
517 polyfill.GetNextPixelRun(exists,scanline,pixelRunStart,pixelRunEnd))
519 TPoint start(pixelRunStart, scanline);
520 TPoint end(pixelRunEnd, scanline);
523 ClipFillLine(start,end,clipRect);
527 iDrawDevice->UpdateRegion(clipRect);
533 Fills a polygon defined using an array of points. The first point in the array defines the
534 start of the first side of the polygon. The final side of the polygon is drawn using the last point
535 from the array. The area is filled with the current brush settings. Optimized for polygons that are
536 much larger than the screen.
538 Self-crossing polygons are filled according to the specified fill rule.
540 @param aPointList Array of points specifying the vertices of the polygon.
541 @param aFillRule Polygon filling rule.
543 void CSwDirectGdiEngine::PolyFillLarge(const TArray<TPoint>* aPointList, DirectGdi::TFillRule aFillRule)
549 TRect clipRect(0,0,0,0);
550 const TInt limit = iDefaultRegionPtr->Count();
551 for (TInt count = 0; count < limit; count++)
553 clipRect = (*iDefaultRegionPtr)[count];
554 CSwDirectGdiPolygonFiller polyfill;
555 polyfill.Construct(aPointList,aFillRule,CSwDirectGdiPolygonFiller::EGetPixelRunsSequentiallyForSpecifiedScanLines);
556 TInt clipRectOffsetStart = clipRect.iTl.iY - iOrigin.iY;
557 TInt clipRectOffsetEnd = clipRect.iBr.iY - iOrigin.iY;
559 for (TInt scanline = clipRectOffsetStart; scanline < clipRectOffsetEnd; scanline++)
561 polyfill.GetNextPixelRunOnSpecifiedScanLine(exists,scanline,pixelRunStart,pixelRunEnd);
564 TPoint start(pixelRunStart,scanline);
565 TPoint end(pixelRunEnd,scanline);
568 ClipFillLine(start,end,clipRect);
569 polyfill.GetNextPixelRunOnSpecifiedScanLine(exists,scanline,pixelRunStart,pixelRunEnd);
574 iDrawDevice->UpdateRegion(clipRect);
579 Draws a polygon defined by an array of points using the current pen settings. The first point in the array defines the
580 start of the first side of the polygon. The final side of the polygon is drawn using the last point
581 from the array, and the line is drawn to the start point of the first side.
583 @param aPointList List of points specifying the vertices of the polygon.
585 void CSwDirectGdiEngine::PolyOutline(const TArray<TPoint>* aPointList)
587 const TInt vertexes = aPointList->Count();
589 for (TInt count = 0; count < vertexes; count++)
591 TPoint point1((*aPointList)[count]);
592 TPoint point2((*aPointList)[(count + 1) % vertexes]);
594 if (point1.iY < point2.iY)
596 DoDrawLine(point1,point2,ETrue);
601 iDotParam += Max(Abs(point2.iX - point1.iX),Abs(point2.iY - point1.iY));
602 const TInt dotParam = iDotParam;
603 DoDrawLine(point2,point1,EFalse);
605 if (Abs(point2.iX - point1.iX) > Abs(point2.iY - point1.iY))
607 if (iPenStyle == DirectGdi::ESolidPen || (iDotMask & (1 << ((iDotParam / iPenSize.iWidth) % iDotLength))))
608 DoPlot((*aPointList)[count]);
612 if (iPenStyle == DirectGdi::ESolidPen || (iDotMask & (1 << ((iDotParam / iPenSize.iHeight) % iDotLength))))
613 DoPlot((*aPointList)[count]);
617 iDotParam = dotParam;