arc drawing support

This commit is contained in:
Vadim Lopatin 2016-11-07 17:32:24 +03:00
parent b5658d7811
commit d3724205df
2 changed files with 78 additions and 25 deletions

View File

@ -1112,6 +1112,9 @@ void main()
buf.drawEllipseF(x+300, y+600, 200, 150, 3, 0x80008000, 0x804040FF);
canvas.font.drawText(buf, x + 300, y + 600, "fillEllipseF()"d, 0x208050);
buf.drawEllipseArcF(x+540, y+600, 150, 180, 45, 130, 3, 0x40008000, 0x804040FF);
canvas.font.drawText(buf, x + 540, y + 580, "drawEllipseArcF()"d, 0x208050);
};
tabs.addTab(canvas, "TAB_CANVAS"c);

View File

@ -633,46 +633,65 @@ class DrawBuf : RefCountedObject {
if (points.length < 2)
return;
bool hasInnerArea = !isFullyTransparentColor(innerAreaColour);
if (isFullyTransparentColor(colour)) {
if (hasInnerArea)
fillPolyF(points, innerAreaColour);
return;
}
int len = cast(int)points.length;
if (hasInnerArea) {
PointF[] innerArea;
innerArea.assumeSafeAppend;
//Log.d("fill poly inner: ", points);
for(int i = 0; i < len; i++) {
PointF[4] quad;
int index0 = i - 1;
int index1 = i;
int index2 = (i + 1) % len;
int index3 = (i + 2) % len;
if (index0 < 0)
index0 = len - 1;
calcLineSegmentQuad(points[index0], points[index1], points[index2], points[index3], width, quad);
innerArea ~= quad[3];
}
fillPolyF(innerArea, innerAreaColour);
}
if (!isFullyTransparentColor(colour)) {
for(int i = 0; i + (cycled ? 0 : 1) < len; i++) {
int index0 = i - 1;
int index1 = i;
int index2 = i + 1;
int index3 = i + 2;
if (index0 < 0)
index0 = cycled ? len - 1 : 0;
if (index1 >= len)
index1 %= len; // only can be if cycled
if (index2 >= len) {
if (cycled)
index2 %= len;
else
index2 %= len; // only can be if cycled
index3 %= len; // only can be if cycled
if (!cycled) {
if (index1 == len - 1) {
index0 = index1;
index2 = 0;
index3 = 0;
} else if (index1 == len - 2) {
index2 = len - 1;
}
if (index3 >= len) {
if (cycled)
index3 %= len;
else
index3 = len - 1;
}
}
drawLineSegmentF(points[index0], points[index1], points[index2], points[index3], width, colour);
//Log.d("lineSegment - inner ", index0, ", ", index1, ", ", index2, ", ", index3);
calcLineSegmentQuad(points[index0], points[index1], points[index2], points[index3], width, quad);
innerArea ~= quad[3];
}
fillPolyF(innerArea, innerAreaColour);
}
if (!isFullyTransparentColor(colour)) {
for(int i = 0; i < len; i++) {
int index0 = i - 1;
int index1 = i;
int index2 = i + 1;
int index3 = i + 2;
if (index0 < 0)
index0 = cycled ? len - 1 : 0;
index2 %= len; // only can be if cycled
index3 %= len; // only can be if cycled
if (!cycled) {
if (index1 == len - 1) {
index0 = index1;
index2 = 0;
index3 = 0;
} else if (index1 == len - 2) {
index2 = len - 1;
index3 = len - 1;
}
}
//Log.d("lineSegment - outer ", index0, ", ", index1, ", ", index2, ", ", index3);
if (cycled || i + 1 < len)
drawLineSegmentF(points[index0], points[index1], points[index2], points[index3], width, colour);
}
}
}
@ -739,6 +758,37 @@ class DrawBuf : RefCountedObject {
polyLineF(points, lineWidth, lineColor, true, fillColor);
}
/// draw ellipse arc or filled ellipse arc
void drawEllipseArcF(float centerX, float centerY, float xRadius, float yRadius, float startAngle, float endAngle, float lineWidth, uint lineColor, uint fillColor = COLOR_TRANSPARENT) {
import std.math : sin, cos, PI;
if (xRadius < 0)
xRadius = -xRadius;
if (yRadius < 0)
yRadius = -yRadius;
startAngle = startAngle * 2 * PI / 360;
endAngle = endAngle * 2 * PI / 360;
if (endAngle < startAngle)
endAngle += 2 * PI;
float angleDiff = endAngle - startAngle;
if (angleDiff > 2*PI)
angleDiff %= 2*PI;
int numLines = cast(int)((xRadius + yRadius) / angleDiff);
if (numLines < 3)
numLines = 4;
float step = angleDiff / numLines;
float angle = startAngle;
PointF[] points;
points.assumeSafeAppend;
points ~= PointF(centerX, centerY);
for (int i = 0; i < numLines; i++) {
float x = centerX + cos(angle) * xRadius;
float y = centerY + sin(angle) * yRadius;
angle += step;
points ~= PointF(x, y);
}
polyLineF(points, lineWidth, lineColor, true, fillColor);
}
/// draw poly line of width == 1px; when cycled is true, connect first and last point
void polyLine(Point[] points, uint colour, bool cycled) {
if (points.length < 2)