| |
| |
|
|
| |
| |
| |
| #include "tstrokeoutline.h" |
| #include "tstroke.h" |
| #include "tcurves.h" |
| #include "tmathutil.h" |
| #include "tgl.h" |
| |
| |
| |
| typedef std::pair<TQuadratic *, TQuadratic *> outlineEdge; |
| typedef std::vector<outlineEdge> outlineBoundary; |
| |
| const double infDouble = (std::numeric_limits<double>::max)(); |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| namespace Outline |
| { |
| |
| class infinityCurvature |
| { |
| }; |
| |
| class notValidOutline |
| { |
| }; |
| } |
| |
| namespace |
| { |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| double localComputeStep(const TQuadratic &quad, double pixelSize) |
| { |
| double step = 2; |
| |
| TPointD A = quad.getP0() - 2.0 * quad.getP1() + quad.getP2(); |
| |
| double A_len = norm(A); |
| if (A_len > 0) |
| step = sqrt(2 * pixelSize / A_len); |
| |
| return step; |
| } |
| |
| |
| |
| |
| const int QUARTER_BEGIN = 1; |
| const int QUARTER_END = 0; |
| |
| |
| const int OUTLINE_UP = 1; |
| const int OUTLINE_DOWN = 0; |
| |
| |
| const double ratio_1_3 = 1.0 / 3.0; |
| const double ratio_2_3 = 2.0 / 3.0; |
| |
| |
| |
| |
| template <class T> |
| double curvature_t0(const T *curve) |
| { |
| assert(curve); |
| TPointD v1 = curve->getP1() - curve->getP0(); |
| TPointD v2 = curve->getP2() - curve->getP1(); |
| |
| double v_cross = cross(v1, v2); |
| |
| if (isAlmostZero(v_cross)) |
| return infDouble; |
| |
| return ratio_2_3 * v_cross / pow(norm(v1), ratio_1_3); |
| } |
| |
| |
| |
| |
| double curvature_t1(const TThickQuadratic *curve) |
| { |
| assert(curve); |
| TThickQuadratic tmp; |
| |
| tmp.setThickP0(curve->getThickP2()); |
| tmp.setThickP1(curve->getThickP1()); |
| tmp.setThickP2(curve->getThickP0()); |
| |
| return curvature_t0(&tmp); |
| } |
| |
| |
| |
| |
| |
| TPointD getPointInOutline(const TThickQuadratic *tq, double t, int upOrDown) |
| { |
| assert(tq); |
| const TThickPoint &p = tq->getThickPoint(t); |
| TPointD n = tq->getSpeed(t); |
| if (norm2(n)) { |
| n = normalize(n); |
| n = upOrDown ? rotate90(n) : rotate270(n); |
| } |
| |
| return convert(p) + p.thick * n; |
| } |
| |
| |
| |
| bool checkPointInOutline(const TPointD &pointToTest, |
| const TThickQuadratic *tq, |
| double t, |
| double error) |
| { |
| assert(tq); |
| TThickPoint tpnt = tq->getThickPoint(t); |
| |
| if (fabs(sq(pointToTest.x - tpnt.x) + |
| sq(pointToTest.y - tpnt.y) - |
| sq(tpnt.thick)) < error) |
| return true; |
| |
| return false; |
| } |
| |
| |
| |
| |
| TQuadratic *makeOutlineForThickQuadratic(const TThickQuadratic *tq, int upOrDown) |
| { |
| assert(tq); |
| |
| |
| TThickPoint |
| p0 = tq->getThickP0(), |
| |
| p2 = tq->getThickP2(); |
| |
| TPointD t0 = tq->getP1() - tq->getP0(); |
| TPointD t1 = tq->getP2() - tq->getP1(); |
| |
| if (t0 == t1) |
| return 0; |
| |
| TPointD |
| N0 = tq->getSpeed(0.0), |
| N2 = tq->getSpeed(1.0); |
| |
| if (!norm2(N0) && !norm2(N2)) |
| throw Outline::notValidOutline(); |
| |
| if (norm2(N0)) { |
| N0 = normalize(N0); |
| N0 = upOrDown ? rotate90(N0) : rotate270(N0); |
| } |
| |
| if (norm2(N2)) { |
| N2 = normalize(N2); |
| N2 = upOrDown ? rotate90(N2) : rotate270(N2); |
| } |
| |
| TPointD p0aux = (convert(p0) + p0.thick * N0); |
| TPointD p2aux = (convert(p2) + p2.thick * N2); |
| |
| TQuadratic |
| radius(TPointD(tq->getThickP0().thick, 0.0), |
| TPointD(tq->getThickP1().thick, 0.0), |
| TPointD(tq->getThickP2().thick, 0.0)); |
| |
| TPointD r0 = radius.getSpeed(0.0); |
| TPointD r1 = radius.getSpeed(1.0); |
| |
| TPointD |
| v0, |
| v2; |
| |
| double ct0 = curvature_t0(tq); |
| |
| if (ct0 != infDouble) |
| v0 = (1 + p0.thick * ct0) * t0 + 0.5 * r0.x * N0; |
| else |
| v0 = r0.x * N0; |
| |
| double ct1 = curvature_t1(tq); |
| |
| if (ct1 != infDouble) |
| v2 = (1 + p2.thick * ct1) * t1 + 0.5 * r1.x * N2; |
| else |
| v2 = r1.x * N2; |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| double det = v0.x * v0.y - v2.x * v2.y; |
| if (areAlmostEqual(det, 0.0)) |
| return 0; |
| double xsol; |
| try { |
| xsol = ((p0aux.x - p2aux.x) * v2.y - (p0aux.y - p2aux.y) * v2.x) / det; |
| |
| } catch (TMathException &) { |
| return new TQuadratic((upOrDown) ? p0aux : p2aux, (p0aux + p2aux) * 0.5, (upOrDown) ? p2aux : p0aux); |
| } catch (std::exception &e) { |
| std::string s(e.what()); |
| abort(); |
| } catch (...) { |
| abort(); |
| } |
| |
| return new TQuadratic((upOrDown) ? p0aux : p2aux, p0aux + xsol * v0, (upOrDown) ? p2aux : p0aux); |
| } |
| |
| |
| |
| |
| |
| |
| |
| void makeOutline( |
| outlineBoundary &outl, |
| const TThickQuadratic &t, |
| double error) |
| { |
| outlineEdge edge; |
| const TThickQuadratic *tq = &t; |
| edge.first = edge.second = 0; |
| try { |
| edge.first = makeOutlineForThickQuadratic(tq, OUTLINE_UP); |
| edge.second = makeOutlineForThickQuadratic(tq, OUTLINE_DOWN); |
| } catch (Outline::notValidOutline &) { |
| delete edge.first; |
| delete edge.second; |
| return; |
| } |
| |
| const TQuadratic *q_up = edge.first; |
| const TQuadratic *q_down = edge.second; |
| const double parameterTest = 0.5; |
| |
| |
| bool isAlmostAPoint = |
| areAlmostEqual(tq->getThickP0(), tq->getThickP1(), 1e-2) && |
| areAlmostEqual(tq->getThickP1(), tq->getThickP2(), 1e-2) |
| ; |
| |
| if (isAlmostAPoint || |
| q_up && checkPointInOutline(q_up->getPoint(parameterTest), tq, parameterTest, error) && |
| q_down && checkPointInOutline(q_down->getPoint(parameterTest), tq, parameterTest, error)) { |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| outl.push_back(edge); |
| return; |
| } else { |
| delete edge.first; |
| delete edge.second; |
| } |
| |
| TThickQuadratic |
| tq_left, |
| tq_rigth; |
| |
| tq->split(0.5, tq_left, tq_rigth); |
| |
| makeOutline( outl, tq_left, error); |
| makeOutline( outl, tq_rigth, error); |
| } |
| |
| |
| |
| void splitCircularArcIntoQuadraticCurves(const TPointD &Center, |
| const TPointD &Pstart, |
| const TPointD &Pend, |
| std::vector<TQuadratic *> &quadArray) |
| { |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| const double cos_ang = 0.5 * sqrt(2.0); |
| const double sin_ang = 0.5 * sqrt(2.0); |
| const double tan_semiang = 0.4142135623730950488016887242097; |
| const int N_QUAD = 8; |
| |
| |
| |
| TPointD Rstart = Pstart - Center; |
| TPointD Rend = Pend - Center; |
| double cross_prod = cross(Rstart, Rend); |
| double dot_prod = Rstart * Rend; |
| const double sqr_radius = Rstart * Rstart; |
| TPointD aliasPstart = Pstart; |
| TQuadratic *quad; |
| |
| while ((cross_prod <= 0) || (dot_prod <= cos_ang * sqr_radius)) |
| |
| { |
| if ((int)quadArray.size() == N_QUAD) |
| return; |
| TPointD Rstart_rot_ang(cos_ang * Rstart.x - sin_ang * Rstart.y, |
| sin_ang * Rstart.x + cos_ang * Rstart.y); |
| TPointD Rstart_rot_90(-Rstart.y, Rstart.x); |
| quad = new TQuadratic(aliasPstart, |
| aliasPstart + tan_semiang * Rstart_rot_90, |
| Center + Rstart_rot_ang); |
| quadArray.push_back(quad); |
| |
| |
| |
| |
| Rstart = Rstart_rot_ang; |
| aliasPstart = quad->getP2(); |
| cross_prod = cross(Rstart, Rend); |
| dot_prod = Rstart * Rend; |
| |
| |
| |
| if ((cross_prod <= 0) && (dot_prod > 0.95 * sqr_radius)) |
| return; |
| } |
| |
| if ((cross_prod > 0) && (dot_prod > 0)) |
| { |
| TPointD Rstart_rot_90(-Rstart.y, Rstart.x); |
| |
| double deg_index = (sqr_radius - dot_prod) / (sqr_radius + dot_prod); |
| |
| quad = new TQuadratic(aliasPstart, |
| (deg_index < 0) ? 0.5 * (aliasPstart + Pend) : aliasPstart + sqrt(deg_index) * Rstart_rot_90, |
| Pend); |
| quadArray.push_back(quad); |
| |
| } else |
| quadArray.back()->setP2(Pend); |
| } |
| |
| |
| |
| |
| |
| |
| |
| void copy( |
| const std::vector<TQuadratic *> &arrayUp, |
| const std::vector<TQuadratic *> &arrayDown, |
| outlineBoundary &ob) |
| { |
| int minSize = std::min(arrayUp.size(), arrayDown.size()); |
| |
| assert(minSize > 0); |
| |
| int i; |
| |
| for (i = 0; i < minSize; ++i) { |
| |
| |
| ob.push_back(outlineEdge(arrayUp[i], arrayDown[i])); |
| } |
| if (arrayUp.size() != arrayDown.size()) { |
| const std::vector<TQuadratic *> &vMaxSize = arrayUp.size() > arrayDown.size() ? arrayUp : arrayDown; |
| const std::vector<TQuadratic *> &vMinSize = arrayUp.size() < arrayDown.size() ? arrayUp : arrayDown; |
| |
| int delta = vMaxSize.size() - vMinSize.size(); |
| |
| if (arrayUp.size() > arrayDown.size()) |
| while (i < minSize + delta) { |
| |
| ob.push_back(outlineEdge(arrayUp[i], (TQuadratic *)0)); |
| i++; |
| } |
| else |
| while (i < minSize + delta) { |
| |
| ob.push_back(outlineEdge((TQuadratic *)0, arrayDown[i])); |
| i++; |
| } |
| } |
| } |
| |
| |
| |
| inline void changeQuadraticDirection(TQuadratic *q) |
| { |
| TPointD p = q->getP2(); |
| q->setP2(q->getP0()); |
| q->setP0(p); |
| } |
| |
| |
| |
| |
| void changeDirection(std::vector<TQuadratic *> &array, bool onlyQuads = false) |
| { |
| UINT chunkCount = array.size(); |
| UINT to = tfloor(chunkCount * 0.5); |
| UINT i; |
| |
| if (chunkCount & 1) |
| changeQuadraticDirection(array[to]); |
| |
| --chunkCount; |
| |
| for (i = 0; i < to; ++i) { |
| changeQuadraticDirection(array[i]); |
| changeQuadraticDirection(array[chunkCount - i]); |
| |
| if (!onlyQuads) |
| std::swap(array[i], array[chunkCount - i]); |
| } |
| } |
| |
| |
| |
| |
| |
| TQuadratic getCircleQuarter(const TThickQuadratic *tq, int versus) |
| { |
| TQuadratic out; |
| |
| TPointD v = versus ? -tq->getSpeed(0.0) : tq->getSpeed(1.0); |
| |
| if (norm2(v)) |
| v = normalize(v); |
| |
| TPointD center = versus ? tq->getP0() : tq->getP2(); |
| double radius = versus ? tq->getThickP0().thick : tq->getThickP2().thick; |
| |
| out.setP0(center + (versus ? rotate270(v) : rotate90(v)) * radius); |
| out.setP1(center + v * radius); |
| out.setP2(center + (versus ? rotate90(v) : rotate270(v)) * radius); |
| |
| return out; |
| } |
| |
| |
| |
| void drawQuadratic(const TQuadratic &quad, double pixelSize) |
| { |
| double m_min_step_at_normal_size = localComputeStep(quad, pixelSize); |
| |
| |
| |
| double invSqrtScale = 1.0; |
| |
| TPointD scP0 = quad.getP0(); |
| TPointD scP1 = quad.getP1(); |
| TPointD scP2 = quad.getP2(); |
| |
| TPointD A = scP0 - 2 * scP1 + scP2; |
| TPointD B = scP0 - scP1; |
| |
| double h; |
| h = invSqrtScale * m_min_step_at_normal_size; |
| double h2 = h * h; |
| |
| TPointD P = scP0, D2 = 2 * h2 * A, D1 = A * h2 - 2 * B * h; |
| |
| if (h < 0 || isAlmostZero(h)) |
| return; |
| assert(h > 0); |
| |
| |
| glBegin(GL_LINE_STRIP); |
| glVertex2d(scP0.x, scP0.y); |
| |
| for (double t = h; t < 1; t = t + h) { |
| P = P + D1; |
| D1 = D1 + D2; |
| glVertex2d(P.x, P.y); |
| } |
| |
| glVertex2d(scP2.x, scP2.y); |
| glEnd(); |
| } |
| |
| |
| |
| } |
| |
| |
| |
| void makeOutline(const TStroke *stroke, int startQuad, int endQuad, |
| outlineBoundary &ob, double error2) |
| { |
| |
| |
| assert(stroke); |
| assert(startQuad >= 0); |
| assert(endQuad < stroke->getChunkCount()); |
| assert(startQuad <= endQuad); |
| TThickQuadratic *tq; |
| std::vector<TQuadratic *> arrayUp, arrayDown; |
| TQuadratic arc; |
| |
| if (!stroke->getChunkCount()) |
| return; |
| |
| { |
| const TThickQuadratic *tq = stroke->getChunk(startQuad); |
| |
| |
| |
| |
| TQuadratic |
| arc = getCircleQuarter(tq, QUARTER_BEGIN); |
| |
| |
| splitCircularArcIntoQuadraticCurves(tq->getP0(), arc.getP0(), arc.getP1(), arrayUp); |
| |
| |
| changeDirection(arrayUp); |
| splitCircularArcIntoQuadraticCurves(tq->getP0(), arc.getP1(), arc.getP2(), arrayDown); |
| changeDirection(arrayDown, true); |
| |
| |
| |
| |
| copy( arrayUp, arrayDown, ob); |
| } |
| |
| for (int i = startQuad; i <= endQuad; ++i) { |
| tq = (TThickQuadratic *)stroke->getChunk(i); |
| TThickPoint p0 = tq->getThickP0(); |
| TThickPoint p1 = tq->getThickP1(); |
| TThickPoint p2 = tq->getThickP2(); |
| if (p0.x == p1.x) { |
| if (p1.x == p2.x && ((p1.y > p0.y && p1.y > p2.y) || (p1.y < p0.y && p1.y < p2.y))) |
| tq = new TThickQuadratic(p0, 0.5 * (p0 + p1), p1); |
| } else if (p0.y == p1.y) { |
| if (p0.y == p2.y && ((p1.x > p0.x && p1.x > p2.x) || (p1.x < p0.x && p1.x < p2.x))) |
| tq = new TThickQuadratic(p0, 0.5 * (p0 + p1), p1); |
| } else { |
| double fac1 = 1.0 / (p0.x - p1.x); |
| double fac2 = 1.0 / (p0.y - p1.y); |
| |
| double aux1 = fac1 * (p2.x - p1.x); |
| double aux2 = fac2 * (p2.y - p1.y); |
| double aux3 = fac1 * (p0.x - p2.x); |
| double aux4 = fac2 * (p0.y - p2.y); |
| |
| if ((areAlmostEqual(aux1, aux2) && aux1 >= 0) || |
| (areAlmostEqual(aux3, aux4) && aux3 >= 0 && aux3 <= 1)) |
| tq = new TThickQuadratic(p0, 0.5 * (p0 + p1), p1); |
| } |
| |
| |
| makeOutline( ob, *tq, error2); |
| if (tq != stroke->getChunk(i)) |
| delete tq; |
| } |
| |
| arrayUp.clear(); |
| arrayDown.clear(); |
| |
| |
| |
| { |
| arc = getCircleQuarter(tq, QUARTER_END); |
| splitCircularArcIntoQuadraticCurves(tq->getP2(), arc.getP1(), arc.getP0(), arrayUp); |
| changeDirection(arrayUp); |
| splitCircularArcIntoQuadraticCurves(tq->getP2(), arc.getP2(), arc.getP1(), arrayDown); |
| changeDirection(arrayDown, true); |
| |
| |
| copy( arrayUp, arrayDown, ob); |
| } |
| } |
| |
| |
| |
| void drawOutline(const outlineBoundary &ob, double pixelSize) |
| { |
| for (UINT i = 0; i < ob.size(); ++i) { |
| drawQuadratic(*ob[i].first, pixelSize); |
| drawQuadratic(*ob[i].second, pixelSize); |
| } |
| } |
| |
| void computeOutlines(const TStroke *stroke, int startQuad, int endQuad, |
| std::vector<TQuadratic *> &quadArray, double error2) |
| { |
| outlineBoundary ob; |
| makeOutline(stroke, startQuad, endQuad, ob, error2); |
| |
| assert(quadArray.empty()); |
| quadArray.resize(ob.size() * 2); |
| |
| int i, count = 0; |
| for (i = 0; i < (int)ob.size(); i++) |
| if (ob[i].first) |
| quadArray[count++] = ob[i].first; |
| |
| for (i = (int)ob.size() - 1; i >= 0; i--) |
| if (ob[i].second) |
| quadArray[count++] = ob[i].second; |
| |
| quadArray.resize(count); |
| for (i = 0; i < (int)quadArray.size(); i++) |
| quadArray[i]->reverse(); |
| |
| std::reverse(quadArray.begin(), quadArray.end()); |
| } |
| |
| |
| |
| |
| |