2012-05-18 20:50:33 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2012 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Simplify.h"
|
|
|
|
|
|
|
|
namespace SimplifyAngleTest {
|
|
|
|
|
|
|
|
#include "Simplify.cpp"
|
|
|
|
|
|
|
|
} // end of SimplifyAngleTest namespace
|
|
|
|
|
|
|
|
#include "Intersection_Tests.h"
|
|
|
|
|
|
|
|
static const SkPoint lines[][2] = {
|
|
|
|
{ { 10, 10}, { 10, 20} },
|
|
|
|
{ { 10, 10}, { 20, 10} },
|
|
|
|
{ { 10, 10}, {-20, 10} },
|
|
|
|
{ { 10, 10}, { 10, -20} },
|
|
|
|
{ { 10, 10}, { 20, 20} },
|
|
|
|
{ { 10, 10}, {-20, -20} },
|
|
|
|
{ { 10, 10}, {-20, 40} },
|
|
|
|
{ { 10, 10}, { 40, -20} }
|
|
|
|
};
|
|
|
|
|
|
|
|
static const size_t lineCount = sizeof(lines) / sizeof(lines[0]);
|
|
|
|
|
|
|
|
static const SkPoint quads[][3] = {
|
|
|
|
{{ 1, 1}, { 2, 2}, { 1, 3}}, // 0
|
|
|
|
{{ 1, 1}, { 3, 3}, { 1, 5}}, // 1
|
|
|
|
{{ 1, 1}, { 4, 4}, { 1, 7}}, // 2
|
|
|
|
{{ 1, 1}, { 5, 5}, { 9, 9}}, // 3
|
|
|
|
{{ 1, 1}, { 4, 4}, { 7, 1}}, // 4
|
|
|
|
{{ 1, 1}, { 3, 3}, { 5, 1}}, // 5
|
|
|
|
{{ 1, 1}, { 2, 2}, { 3, 1}}, // 6
|
|
|
|
};
|
|
|
|
|
|
|
|
static const size_t quadCount = sizeof(quads) / sizeof(quads[0]);
|
|
|
|
|
|
|
|
static const SkPoint cubics[][4] = {
|
|
|
|
{{ 1, 1}, { 2, 2}, { 2, 3}, { 1, 4}},
|
|
|
|
{{ 1, 1}, { 3, 3}, { 3, 5}, { 1, 7}},
|
|
|
|
{{ 1, 1}, { 4, 4}, { 4, 7}, { 1, 10}},
|
|
|
|
{{ 1, 1}, { 5, 5}, { 8, 8}, { 9, 9}},
|
|
|
|
{{ 1, 1}, { 4, 4}, { 7, 4}, { 10, 1}},
|
|
|
|
{{ 1, 1}, { 3, 3}, { 5, 3}, { 7, 1}},
|
|
|
|
{{ 1, 1}, { 2, 2}, { 3, 2}, { 4, 1}},
|
|
|
|
};
|
|
|
|
|
|
|
|
static const size_t cubicCount = sizeof(cubics) / sizeof(cubics[0]);
|
|
|
|
|
2012-08-23 15:24:42 +00:00
|
|
|
struct segment {
|
|
|
|
SkPath::Verb verb;
|
|
|
|
SkPoint pts[4];
|
|
|
|
};
|
|
|
|
|
|
|
|
static const segment segmentTest1[] = {
|
|
|
|
{SkPath::kLine_Verb, {{2, 1}, {1, 0} }},
|
|
|
|
{SkPath::kQuad_Verb, {{2, 1}, {1, 0}, {0, 0}}},
|
|
|
|
{SkPath::kQuad_Verb, {{2, 1}, {3, 2}, {2, 3}}},
|
|
|
|
{SkPath::kLine_Verb, {{2, 1}, {3, 2} }},
|
|
|
|
{SkPath::kMove_Verb }
|
|
|
|
};
|
|
|
|
|
2012-08-24 15:24:36 +00:00
|
|
|
static const segment segmentTest2[] = {
|
|
|
|
{SkPath::kLine_Verb, {{1, 0}, {0, 0} }},
|
|
|
|
{SkPath::kQuad_Verb, {{1, 0}, {1.89897954f, 0.898979485f}, {2.39387703f, 1.59591794f}}},
|
|
|
|
{SkPath::kLine_Verb, {{1, 0}, {3, 2} }},
|
|
|
|
{SkPath::kMove_Verb }
|
|
|
|
};
|
|
|
|
|
2012-08-31 20:55:07 +00:00
|
|
|
static const segment segmentTest3[] = {
|
|
|
|
{SkPath::kQuad_Verb, {{0, 0}, {2, 0}, {0, 1}}},
|
|
|
|
{SkPath::kQuad_Verb, {{0, 0}, {1, 0}, {0, 1}}},
|
|
|
|
{SkPath::kMove_Verb }
|
|
|
|
};
|
|
|
|
|
2012-08-23 15:24:42 +00:00
|
|
|
static const segment* segmentTests[] = {
|
2012-08-31 20:55:07 +00:00
|
|
|
segmentTest3,
|
2012-08-24 15:24:36 +00:00
|
|
|
segmentTest2,
|
|
|
|
segmentTest1,
|
2012-08-23 15:24:42 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static const size_t segmentTestCount = sizeof(segmentTests) / sizeof(segmentTests[0]);
|
|
|
|
|
|
|
|
static void testSegments(bool testFlat) {
|
|
|
|
for (size_t testIndex = 0; testIndex < segmentTestCount; ++testIndex) {
|
|
|
|
const segment* segPtr = segmentTests[testIndex];
|
|
|
|
SimplifyAngleTest::Angle lesser, greater;
|
|
|
|
int index = 0;
|
|
|
|
do {
|
|
|
|
int next = index + 1;
|
2012-08-24 15:24:36 +00:00
|
|
|
#if HIGH_DEF_ANGLES==0
|
2012-08-23 15:24:42 +00:00
|
|
|
if (testFlat) {
|
|
|
|
lesser.setFlat(segPtr[index].pts, segPtr[index].verb, 0, index, next);
|
|
|
|
} else {
|
|
|
|
lesser.set(segPtr[index].pts, segPtr[index].verb, 0, index, next);
|
|
|
|
}
|
2012-08-24 15:24:36 +00:00
|
|
|
#else
|
|
|
|
lesser.set(segPtr[index].pts, segPtr[index].verb, 0, index, next, 0, 1);
|
|
|
|
#endif
|
2012-08-23 15:24:42 +00:00
|
|
|
if (segPtr[next].verb == SkPath::kMove_Verb) {
|
|
|
|
break;
|
|
|
|
}
|
2012-08-24 15:24:36 +00:00
|
|
|
#if HIGH_DEF_ANGLES==0
|
2012-08-23 15:24:42 +00:00
|
|
|
if (testFlat) {
|
|
|
|
greater.setFlat(segPtr[next].pts, segPtr[next].verb, 0, index, next);
|
|
|
|
} else {
|
|
|
|
greater.set(segPtr[next].pts, segPtr[next].verb, 0, index, next);
|
|
|
|
}
|
2012-08-24 15:24:36 +00:00
|
|
|
#else
|
|
|
|
greater.set(segPtr[next].pts, segPtr[next].verb, 0, index, next, 0, 1);
|
|
|
|
#endif
|
2012-08-23 15:24:42 +00:00
|
|
|
bool result = lesser < greater;
|
|
|
|
SkASSERT(result);
|
|
|
|
index = next;
|
|
|
|
} while (true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-18 20:50:33 +00:00
|
|
|
static void testLines(bool testFlat) {
|
|
|
|
// create angles in a circle
|
|
|
|
SkTDArray<SimplifyAngleTest::Angle> angles;
|
|
|
|
SkTDArray<SimplifyAngleTest::Angle* > angleList;
|
|
|
|
SkTDArray<double> arcTans;
|
|
|
|
size_t x;
|
|
|
|
for (x = 0; x < lineCount; ++x) {
|
|
|
|
SimplifyAngleTest::Angle* angle = angles.append();
|
2012-08-24 15:24:36 +00:00
|
|
|
#if HIGH_DEF_ANGLES==0
|
2012-05-18 20:50:33 +00:00
|
|
|
if (testFlat) {
|
2012-06-01 17:44:28 +00:00
|
|
|
angle->setFlat(lines[x], SkPath::kLine_Verb, 0, x, x + 1);
|
2012-05-18 20:50:33 +00:00
|
|
|
} else {
|
2012-06-01 17:44:28 +00:00
|
|
|
angle->set(lines[x], SkPath::kLine_Verb, 0, x, x + 1);
|
2012-05-18 20:50:33 +00:00
|
|
|
}
|
2012-08-24 15:24:36 +00:00
|
|
|
#else
|
|
|
|
angle->set(lines[x], SkPath::kLine_Verb, 0, x, x + 1, 0, 1);
|
|
|
|
#endif
|
2012-05-18 20:50:33 +00:00
|
|
|
double arcTan = atan2(lines[x][0].fX - lines[x][1].fX,
|
|
|
|
lines[x][0].fY - lines[x][1].fY);
|
|
|
|
arcTans.push(arcTan);
|
|
|
|
}
|
|
|
|
for (x = 0; x < lineCount; ++x) {
|
|
|
|
angleList.push(&angles[x]);
|
|
|
|
}
|
|
|
|
QSort<SimplifyAngleTest::Angle>(angleList.begin(), angleList.end() - 1);
|
|
|
|
bool first = true;
|
|
|
|
bool wrap = false;
|
|
|
|
double base, last;
|
|
|
|
for (size_t x = 0; x < lineCount; ++x) {
|
|
|
|
const SimplifyAngleTest::Angle* angle = angleList[x];
|
|
|
|
int span = angle->start();
|
|
|
|
// SkDebugf("%s [%d] %1.9g (%1.9g,%1.9g %1.9g,%1.9g)\n", __FUNCTION__,
|
|
|
|
// span, arcTans[span], lines[span][0].fX, lines[span][0].fY,
|
|
|
|
// lines[span][1].fX, lines[span][1].fY);
|
|
|
|
if (first) {
|
|
|
|
base = last = arcTans[span];
|
|
|
|
first = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (last < arcTans[span]) {
|
|
|
|
last = arcTans[span];
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!wrap) {
|
|
|
|
if (base < arcTans[span]) {
|
|
|
|
SkDebugf("%s !wrap [%d] %g\n", __FUNCTION__, span, arcTans[span]);
|
|
|
|
SkASSERT(0);
|
|
|
|
}
|
|
|
|
last = arcTans[span];
|
|
|
|
wrap = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
SkDebugf("%s wrap [%d] %g\n", __FUNCTION__, span, arcTans[span]);
|
|
|
|
SkASSERT(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void testQuads(bool testFlat) {
|
|
|
|
SkTDArray<SimplifyAngleTest::Angle> angles;
|
|
|
|
SkTDArray<SimplifyAngleTest::Angle* > angleList;
|
|
|
|
size_t x;
|
|
|
|
for (x = 0; x < quadCount; ++x) {
|
|
|
|
SimplifyAngleTest::Angle* angle = angles.append();
|
2012-08-24 15:24:36 +00:00
|
|
|
#if HIGH_DEF_ANGLES==0
|
2012-05-18 20:50:33 +00:00
|
|
|
if (testFlat) {
|
2012-06-01 17:44:28 +00:00
|
|
|
angle->setFlat(quads[x], SkPath::kQuad_Verb, 0, x, x + 1);
|
2012-05-18 20:50:33 +00:00
|
|
|
} else {
|
2012-06-01 17:44:28 +00:00
|
|
|
angle->set(quads[x], SkPath::kQuad_Verb, 0, x, x + 1);
|
2012-05-18 20:50:33 +00:00
|
|
|
}
|
2012-08-24 15:24:36 +00:00
|
|
|
#else
|
|
|
|
angle->set(quads[x], SkPath::kQuad_Verb, 0, x, x + 1, 0, 1);
|
|
|
|
#endif
|
|
|
|
}
|
2012-05-18 20:50:33 +00:00
|
|
|
for (x = 0; x < quadCount; ++x) {
|
|
|
|
angleList.push(&angles[x]);
|
|
|
|
}
|
|
|
|
QSort<SimplifyAngleTest::Angle>(angleList.begin(), angleList.end() - 1);
|
|
|
|
for (size_t x = 0; x < quadCount; ++x) {
|
|
|
|
*angleList[x] < *angleList[x + 1];
|
|
|
|
SkASSERT(x == quadCount - 1 || *angleList[x] < *angleList[x + 1]);
|
|
|
|
const SimplifyAngleTest::Angle* angle = angleList[x];
|
2012-06-01 18:20:10 +00:00
|
|
|
if ((int) x != angle->start()) {
|
2012-05-18 20:50:33 +00:00
|
|
|
SkDebugf("%s [%d] [%d]\n", __FUNCTION__, x, angle->start());
|
|
|
|
SkASSERT(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void testCubics(bool testFlat) {
|
|
|
|
SkTDArray<SimplifyAngleTest::Angle> angles;
|
|
|
|
SkTDArray<SimplifyAngleTest::Angle* > angleList;
|
|
|
|
for (size_t x = 0; x < cubicCount; ++x) {
|
|
|
|
SimplifyAngleTest::Angle* angle = angles.append();
|
2012-08-24 15:24:36 +00:00
|
|
|
#if HIGH_DEF_ANGLES==0
|
2012-05-18 20:50:33 +00:00
|
|
|
if (testFlat) {
|
2012-06-01 17:44:28 +00:00
|
|
|
angle->setFlat(cubics[x], SkPath::kCubic_Verb, 0, x, x + 1);
|
2012-05-18 20:50:33 +00:00
|
|
|
} else {
|
2012-06-01 17:44:28 +00:00
|
|
|
angle->set(cubics[x], SkPath::kCubic_Verb, 0, x, x + 1);
|
2012-05-18 20:50:33 +00:00
|
|
|
}
|
2012-08-24 15:24:36 +00:00
|
|
|
#else
|
|
|
|
angle->set(cubics[x], SkPath::kCubic_Verb, 0, x, x + 1, 0, 1);
|
|
|
|
#endif
|
2012-05-18 20:50:33 +00:00
|
|
|
angleList.push(angle);
|
|
|
|
}
|
|
|
|
QSort<SimplifyAngleTest::Angle>(angleList.begin(), angleList.end() - 1);
|
|
|
|
for (size_t x = 0; x < cubicCount; ++x) {
|
|
|
|
const SimplifyAngleTest::Angle* angle = angleList[x];
|
2012-06-01 18:20:10 +00:00
|
|
|
if ((int) x != angle->start()) {
|
2012-05-18 20:50:33 +00:00
|
|
|
SkDebugf("%s [%d] [%d]\n", __FUNCTION__, x, angle->start());
|
|
|
|
SkASSERT(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void (*tests[])(bool) = {
|
2012-08-23 15:24:42 +00:00
|
|
|
testSegments,
|
2012-05-18 20:50:33 +00:00
|
|
|
testLines,
|
|
|
|
testQuads,
|
|
|
|
testCubics
|
|
|
|
};
|
|
|
|
|
|
|
|
static const size_t testCount = sizeof(tests) / sizeof(tests[0]);
|
|
|
|
|
|
|
|
static void (*firstTest)(bool) = 0;
|
|
|
|
static bool skipAll = false;
|
|
|
|
|
|
|
|
void SimplifyAngle_Test() {
|
|
|
|
if (skipAll) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
size_t index = 0;
|
|
|
|
if (firstTest) {
|
|
|
|
while (index < testCount && tests[index] != firstTest) {
|
|
|
|
++index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bool firstTestComplete = false;
|
|
|
|
for ( ; index < testCount; ++index) {
|
2012-08-23 15:24:42 +00:00
|
|
|
(*tests[index])(false); // set to true to exercise setFlat
|
2012-05-18 20:50:33 +00:00
|
|
|
firstTestComplete = true;
|
|
|
|
}
|
2012-06-01 17:44:28 +00:00
|
|
|
}
|