| |
| |
| #include "tinbetween.h" |
| #include "tcurves.h" |
| #include "tstroke.h" |
| #include "tpalette.h" |
| |
| |
| #include "tmathutil.h" |
| |
| |
| #include <utility> |
| #include <limits> |
| #include <list> |
| |
| #include "../tvectorimage/tvectorimageP.h" |
| |
| |
| |
| |
| double average(std::vector<double> &values, double range = 2.5) |
| { |
| UINT size = values.size(); |
| if (size == 0) |
| return std::numeric_limits<double>::signaling_NaN(); |
| |
| if (size == 1) |
| return values[0]; |
| |
| double sum = 0; |
| UINT j = 0; |
| for (; j < size; j++) { |
| sum += values[j]; |
| } |
| |
| double average = sum / size; |
| |
| double variance = 0; |
| for (j = 0; j < size; j++) { |
| variance += (average - values[j]) * (average - values[j]); |
| } |
| variance /= size; |
| |
| double err; |
| int acceptedNum = 0; |
| sum = 0; |
| for (j = 0; j < size; j++) { |
| err = values[j] - average; |
| err *= err; |
| if (err <= range * variance) { |
| sum += values[j]; |
| acceptedNum++; |
| } |
| } |
| |
| assert(acceptedNum > 0); |
| return (acceptedNum > 0) ? sum / (double)acceptedNum : average; |
| } |
| |
| |
| |
| double weightedAverage(std::vector<double> &values, std::vector<double> &weights, double range = 2.5) |
| { |
| UINT size = values.size(); |
| if (size == 0) |
| return std::numeric_limits<double>::signaling_NaN(); |
| |
| double sum = 0; |
| double totalWeight = 0; |
| UINT j = 0; |
| for (; j < size; j++) { |
| sum += weights[j] * values[j]; |
| totalWeight += weights[j]; |
| } |
| |
| assert(totalWeight > 0); |
| if (totalWeight == 0) |
| return std::numeric_limits<double>::signaling_NaN(); |
| |
| double average = sum / totalWeight; |
| |
| double variance = 0; |
| for (j = 0; j < size; j++) { |
| variance += weights[j] * (average - values[j]) * (average - values[j]); |
| } |
| variance /= totalWeight; |
| |
| double err; |
| totalWeight = 0; |
| sum = 0; |
| for (j = 0; j < size; j++) { |
| err = values[j] - average; |
| err *= err; |
| if (err <= range * variance) { |
| sum += weights[j] * values[j]; |
| totalWeight += weights[j]; |
| } |
| } |
| |
| assert(totalWeight > 0); |
| return (totalWeight != 0) ? sum / totalWeight : average; |
| } |
| |
| |
| |
| inline UINT angleNumber(const std::vector<std::pair<int, double>> &corners, double angle) |
| { |
| UINT count = 0; |
| UINT size = corners.size(); |
| for (UINT j = 0; j < size; j++) |
| if (corners[j].second >= angle) |
| count++; |
| |
| return count; |
| } |
| |
| |
| |
| inline bool isTooComplex(UINT n1, UINT n2, UINT maxSubsetNumber = 100) |
| { |
| UINT n, r; |
| if (n1 > n2) { |
| n = n1; |
| r = n2; |
| } else if (n1 < n2) { |
| n = n2; |
| r = n1; |
| } else { |
| assert(!"equal angle number"); |
| return false; |
| } |
| |
| if (n - r < r) |
| r = n - r; |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| const UINT one = 1; |
| if (n > (one << ((sizeof(UINT) * 8 - 1) / r))) |
| return true; |
| |
| UINT product1 = n; |
| for (UINT i = 1; i < r; i++) |
| product1 *= --n; |
| |
| UINT rFact = r; |
| |
| while (r > 1) |
| rFact *= --r; |
| |
| return (product1 / rFact > maxSubsetNumber); |
| |
| |
| |
| } |
| |
| |
| |
| void eraseSmallAngles(std::vector<std::pair<int, double>> &corners, double angle) |
| { |
| std::vector<std::pair<int, double>>::iterator it = corners.begin(); |
| |
| while (it != corners.end()) { |
| if ((*it).second < angle) |
| it = corners.erase(it); |
| else |
| ++it; |
| } |
| } |
| |
| |
| |
| |
| |
| |
| void detectCorners(const TStroke *stroke, double minDegree, |
| std::vector<std::pair<int, double>> &corners, double &min, double &max) |
| { |
| const double minSin = fabs(sin(minDegree * M_PI_180)); |
| double angle, vectorialProduct, metaCornerLen, partialLen; |
| |
| UINT quadCount1 = stroke->getChunkCount(); |
| TPointD speed1, speed2; |
| int j; |
| |
| TPointD tan1, tan2; |
| min = 180; |
| max = minDegree; |
| for (j = 1; j < (int)quadCount1; j++) { |
| speed1 = stroke->getChunk(j - 1)->getSpeed(1); |
| speed2 = stroke->getChunk(j)->getSpeed(0); |
| if (!(speed1 == TPointD() || speed2 == TPointD())) { |
| tan1 = normalize(speed1); |
| tan2 = normalize(speed2); |
| |
| vectorialProduct = fabs(cross(tan1, tan2)); |
| |
| if (tan1 * tan2 < 0) { |
| angle = 180 - asin(tcrop(vectorialProduct, -1.0, 1.0)) * M_180_PI; |
| corners.push_back(std::make_pair(j, angle)); |
| |
| |
| |
| |
| |
| |
| if (min > angle) |
| min = angle; |
| if (max < angle) |
| max = angle; |
| } else if (vectorialProduct >= minSin) { |
| angle = asin(tcrop(vectorialProduct, -1.0, 1.0)) * M_180_PI; |
| corners.push_back(std::make_pair(j, angle)); |
| |
| |
| |
| |
| |
| |
| if (min > angle) |
| min = angle; |
| if (max < angle) |
| max = angle; |
| } |
| } |
| } |
| |
| const double ratioLen = 2.5; |
| const double ratioAngle = 0.2; |
| std::vector<std::pair<int, double>>::iterator it = corners.begin(); |
| |
| for (j = 1; j < (int)quadCount1; j++) |
| { |
| if (it != corners.end() && j == (*it).first) { |
| ++it; |
| continue; |
| } |
| |
| if (j - 2 >= 0 && (corners.empty() || it == corners.begin() || j - 1 != (*(it - 1)).first) && j + 1 < (int)quadCount1 && (corners.empty() || it == corners.end() || j + 1 != (*it).first)) { |
| speed1 = stroke->getChunk(j - 2)->getSpeed(1); |
| speed2 = stroke->getChunk(j + 1)->getSpeed(0); |
| if (!(speed1 == TPointD() || speed2 == TPointD())) { |
| tan1 = normalize(speed1); |
| tan2 = normalize(speed2); |
| vectorialProduct = fabs(cross(tan1, tan2)); |
| |
| if (tan1 * tan2 < 0) { |
| angle = 180 - asin(tcrop(vectorialProduct, -1.0, 1.0)) * M_180_PI; |
| |
| metaCornerLen = ratioLen * (stroke->getChunk(j - 1)->getLength() + stroke->getChunk(j)->getLength()); |
| partialLen = 0; |
| bool goodAngle = false; |
| |
| for (int i = j - 3; i >= 0 && (corners.empty() || it == corners.begin() || i + 1 != (*(it - 1)).first); i--) { |
| tan1 = stroke->getChunk(i)->getSpeed(1); |
| if (tan1 == TPointD()) |
| continue; |
| tan1 = normalize(tan1); |
| |
| tan2 = stroke->getChunk(i + 1)->getSpeed(1); |
| if (tan2 == TPointD()) |
| continue; |
| tan2 = normalize(tan2); |
| |
| vectorialProduct = fabs(cross(tan1, tan2)); |
| double nearAngle = asin(tcrop(vectorialProduct, -1.0, 1.0)) * M_180_PI; |
| if (tan1 * tan2 < 0) |
| nearAngle = 180 - nearAngle; |
| |
| if (nearAngle > ratioAngle * angle) |
| break; |
| |
| partialLen += stroke->getChunk(i)->getLength(); |
| if (partialLen > metaCornerLen) { |
| goodAngle = true; |
| break; |
| } |
| } |
| if (goodAngle) { |
| partialLen = 0; |
| for (int i = j + 2; i + 1 < (int)quadCount1 && (corners.empty() || it == corners.end() || i + 1 != (*it).first); i++) { |
| tan1 = stroke->getChunk(i)->getSpeed(0); |
| if (tan1 == TPointD()) |
| continue; |
| tan1 = normalize(tan1); |
| |
| tan2 = stroke->getChunk(i + 1)->getSpeed(0); |
| if (tan2 == TPointD()) |
| continue; |
| tan2 = normalize(tan2); |
| |
| vectorialProduct = fabs(cross(tan1, tan2)); |
| double nearAngle = asin(tcrop(vectorialProduct, -1.0, 1.0)) * M_180_PI; |
| if (tan1 * tan2 < 0) |
| nearAngle = 180 - nearAngle; |
| |
| if (nearAngle > 0.1 * angle) |
| break; |
| |
| partialLen += stroke->getChunk(i)->getLength(); |
| if (partialLen > metaCornerLen) { |
| goodAngle = true; |
| break; |
| } |
| } |
| } |
| |
| if (goodAngle) { |
| |
| it = corners.insert(it, std::make_pair(j, angle * 0.7)) + 1; |
| if (min > angle) |
| min = angle; |
| if (max < angle) |
| max = angle; |
| |
| |
| |
| } |
| } |
| } |
| } |
| } |
| } |
| |
| |
| |
| double variance(std::vector<double> &values) |
| { |
| UINT size = values.size(); |
| if (size == 0) |
| return std::numeric_limits<double>::signaling_NaN(); |
| |
| double sum = 0; |
| UINT j = 0; |
| for (; j < size; j++) { |
| sum += values[j]; |
| } |
| |
| double average = sum / size; |
| |
| double variance = 0; |
| for (j = 0; j < size; j++) { |
| variance += (average - values[j]) * (average - values[j]); |
| } |
| |
| return variance / size; |
| } |
| |
| |
| |
| void findBestSolution(const TStroke *stroke1, const TStroke *stroke2, |
| std::pair<int, double> *partialAngles1, UINT partialAngles1Size, |
| const std::vector<std::pair<int, double>> &angles2, |
| UINT r, std::list<std::pair<int, double>> &partialSolution, |
| double &bestValue, |
| std::vector<int> &bestVector) |
| { |
| |
| if (r == partialAngles1Size) { |
| |
| UINT j; |
| std::vector<std::pair<int, double>> angles1; |
| |
| std::list<std::pair<int, double>>::iterator it = partialSolution.begin(); |
| |
| for (; it != partialSolution.end(); ++it) { |
| angles1.push_back(*it); |
| } |
| for (j = 0; j < partialAngles1Size; j++) { |
| angles1.push_back(partialAngles1[j]); |
| } |
| |
| UINT angles1Size = angles1.size(); |
| std::vector<double> rationAngles(angles1Size), ratioLength(angles1Size + 1); |
| std::vector<double> ratioX, ratioY; |
| |
| for (j = 0; j < angles1Size; j++) { |
| rationAngles[j] = fabs(angles1[j].second - angles2[j].second) / (angles1[j].second + angles2[j].second); |
| } |
| |
| UINT firstQuad1 = 0; |
| UINT firstQuad2 = 0; |
| UINT nextQuad1, nextQuad2; |
| |
| TRectD bbox1 = stroke1->getBBox(); |
| TRectD bbox2 = stroke2->getBBox(); |
| |
| double app, div; |
| double invTotalLen1 = stroke1->getLength(); |
| assert(invTotalLen1 > 0); |
| invTotalLen1 = 1.0 / invTotalLen1; |
| |
| double invTotalLen2 = stroke2->getLength(); |
| assert(invTotalLen2 > 0); |
| invTotalLen2 = 1.0 / invTotalLen2; |
| |
| for (j = 0; j <= angles1Size; j++) { |
| |
| if (j < angles1Size) { |
| nextQuad1 = angles1[j].first; |
| nextQuad2 = angles2[j].first; |
| } else { |
| nextQuad1 = stroke1->getChunkCount(); |
| nextQuad2 = stroke2->getChunkCount(); |
| } |
| |
| ratioLength[j] = fabs(stroke1->getLengthAtControlPoint(nextQuad1 * 2) * invTotalLen1 - stroke2->getLengthAtControlPoint(nextQuad2 * 2) * invTotalLen2); |
| |
| TPointD p1(stroke1->getChunk(nextQuad1 - 1)->getP2() - |
| stroke1->getChunk(firstQuad1)->getP0()); |
| p1.x = fabs(p1.x); |
| p1.y = fabs(p1.y); |
| |
| TPointD p2(stroke2->getChunk(nextQuad2 - 1)->getP2() - |
| stroke2->getChunk(firstQuad2)->getP0()); |
| p2.x = fabs(p2.x); |
| p2.y = fabs(p2.y); |
| |
| app = fabs(bbox1.getLx() * p2.x - bbox2.getLx() * p1.x); |
| div = (bbox1.getLx() * p2.x + bbox2.getLx() * p1.x); |
| if (div) |
| ratioX.push_back(app / div); |
| |
| app = fabs(bbox1.getLy() * p2.y - bbox2.getLy() * p1.y); |
| div = (bbox1.getLy() * p2.y + bbox2.getLy() * p1.y); |
| if (div) |
| ratioY.push_back(app / div); |
| |
| firstQuad1 = nextQuad1; |
| firstQuad2 = nextQuad2; |
| } |
| |
| double varAng, varX, varY, varLen; |
| |
| varX = average(ratioX); |
| varY = average(ratioY); |
| varLen = average(ratioLength); |
| varAng = average(rationAngles); |
| |
| double estimate = varX + varY + varAng + varLen; |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| if (estimate < bestValue) { |
| bestValue = estimate; |
| if (bestVector.size() != angles1Size) { |
| assert(!"bad size for bestVector"); |
| bestVector.resize(angles1Size); |
| } |
| |
| for (j = 0; j < angles1Size; j++) { |
| bestVector[j] = angles1[j].first; |
| } |
| } |
| |
| return; |
| } |
| |
| |
| |
| if (r == 1) { |
| for (UINT i = 0; i < partialAngles1Size; i++) { |
| findBestSolution(stroke1, stroke2, |
| partialAngles1 + i, 1, |
| angles2, |
| 1, partialSolution, |
| bestValue, |
| bestVector); |
| } |
| return; |
| } |
| |
| partialSolution.push_back(partialAngles1[0]); |
| findBestSolution(stroke1, stroke2, |
| partialAngles1 + 1, partialAngles1Size - 1, |
| angles2, |
| r - 1, partialSolution, |
| bestValue, |
| bestVector); |
| |
| partialSolution.pop_back(); |
| findBestSolution(stroke1, stroke2, |
| partialAngles1 + 1, partialAngles1Size - 1, |
| angles2, |
| r, partialSolution, |
| bestValue, |
| bestVector); |
| } |
| |
| |
| |
| void findBestSolution(const TStroke *stroke1, const TStroke *stroke2, |
| std::vector<std::pair<int, double>> &angles1, |
| const std::vector<std::pair<int, double>> &angles2, |
| double &bestValue, |
| std::vector<int> &bestVector) |
| { |
| assert(angles1.size() > angles2.size()); |
| |
| std::list<std::pair<int, double>> partialSolution; |
| |
| findBestSolution(stroke1, stroke2, |
| &(angles1[0]), angles1.size(), |
| angles2, |
| angles2.size(), partialSolution, |
| bestValue, bestVector); |
| } |
| |
| |
| |
| void trivialSolution(const TStroke *stroke1, const TStroke *stroke2, |
| const std::vector<std::pair<int, double>> &angles1, |
| const std::vector<std::pair<int, double>> &angles2, |
| std::vector<int> &solution) |
| { |
| assert(angles1.size() > angles2.size()); |
| |
| UINT j; |
| double subStrokeRatio2; |
| |
| double invTotalLen2 = stroke2->getLength(); |
| assert(invTotalLen2); |
| invTotalLen2 = 1.0 / invTotalLen2; |
| |
| double invTotalLen1 = stroke1->getLength(); |
| assert(invTotalLen1 > 0); |
| invTotalLen1 = 1.0 / invTotalLen1; |
| |
| if (solution.size() != angles2.size()) { |
| assert(!"bad size for solution"); |
| solution.resize(angles2.size()); |
| } |
| |
| int toBeErased = angles1.size() - angles2.size(); |
| UINT count = 0; |
| |
| double diff, ratio, oldRatio = 100; |
| |
| subStrokeRatio2 = stroke2->getLengthAtControlPoint(angles2[count].first * 2) * invTotalLen2; |
| |
| for (j = 0; j < angles1.size() && count < solution.size(); j++) { |
| if (toBeErased == 0) { |
| solution[count++] = angles1[j].first; |
| } else { |
| ratio = stroke1->getLengthAtControlPoint(angles1[j].first * 2) * invTotalLen1; |
| assert(ratio > 0 && ratio <= 1); |
| |
| diff = ratio - subStrokeRatio2; |
| if (diff >= 0) { |
| if (fabs(diff) < fabs(oldRatio - subStrokeRatio2)) { |
| solution[count] = angles1[j].first; |
| oldRatio = 100; |
| } else { |
| assert(j > 0); |
| solution[count] = angles1[j - 1].first; |
| } |
| count++; |
| if (angles2.size() < count) |
| subStrokeRatio2 = stroke2->getLengthAtControlPoint(angles2[count].first * 2) * invTotalLen2; |
| else |
| subStrokeRatio2 = 1; |
| } else { |
| toBeErased--; |
| oldRatio = ratio; |
| } |
| } |
| } |
| assert(count == solution.size()); |
| } |
| |
| |
| |
| TStroke *extract(const TStroke *source, UINT firstQuad, UINT lastQuad) |
| { |
| UINT quadCount = source->getChunkCount(); |
| if (firstQuad >= quadCount) { |
| assert(!"bad quadric index"); |
| firstQuad = quadCount - 1; |
| } |
| if (lastQuad < firstQuad) { |
| assert(!"bad quadric index"); |
| lastQuad = firstQuad; |
| } |
| if (lastQuad >= quadCount) { |
| assert(!"bad quadric index"); |
| lastQuad = quadCount - 1; |
| } |
| |
| UINT cpIndex0 = firstQuad * 2; |
| UINT cpIndex1 = lastQuad * 2 + 2; |
| |
| std::vector<TThickPoint> points(cpIndex1 - cpIndex0 + 1); |
| UINT count = 0; |
| for (UINT j = cpIndex0; j <= cpIndex1; j++) { |
| points[count++] = source->getControlPoint(j); |
| } |
| |
| return new TStroke(points); |
| } |
| |
| |
| |
| void sample(const TStroke *stroke, int samplingSize, std::vector<TPointD> &sampledPoint) |
| { |
| double samplingFrequency = 1.0 / (double)samplingSize; |
| sampledPoint.resize(samplingSize); |
| |
| double totalLen = stroke->getLength(); |
| double step = totalLen * samplingFrequency; |
| double len = 0; |
| |
| for (int p = 0; p < samplingSize - 1; p++) { |
| sampledPoint[p] = stroke->getPointAtLength(len); |
| len += step; |
| } |
| sampledPoint.back() = stroke->getControlPoint(stroke->getControlPointCount() - 1); |
| } |
| |
| |
| |
| class TInbetween::Imp |
| { |
| |
| public: |
| |
| |
| struct StrokeTransform { |
| typedef enum { |
| EQUAL, |
| POINT, |
| GENERAL |
| } TransformationType; |
| |
| TPointD m_translate; |
| TPointD m_rotationAndScaleCenter; |
| double m_scaleX, m_scaleY, m_rotation; |
| |
| TransformationType m_type; |
| |
| |
| TAffine m_inverse; |
| |
| std::vector<int> m_firstStrokeCornerIndexes; |
| std::vector<int> m_secondStrokeCornerIndexes; |
| }; |
| |
| |
| |
| TVectorImageP m_firstImage, m_lastImage; |
| |
| std::vector<StrokeTransform> m_transformation; |
| |
| void computeTransformation(); |
| |
| void transferColor(const TVectorImageP &destination) const; |
| |
| TVectorImageP tween(double t) const; |
| |
| Imp(const TVectorImageP firstImage, const TVectorImageP lastImage) |
| : m_firstImage(firstImage), m_lastImage(lastImage) |
| { |
| computeTransformation(); |
| } |
| }; |
| |
| |
| TInbetween::TInbetween(const TVectorImageP firstImage, const TVectorImageP lastImage) |
| : m_imp(new TInbetween::Imp(firstImage, lastImage)) |
| { |
| } |
| |
| |
| |
| TInbetween::~TInbetween() |
| { |
| } |
| |
| |
| |
| void TInbetween::Imp::computeTransformation() |
| { |
| const UINT samplingPointNumber = 10; |
| const UINT bboxSamplingWeight = samplingPointNumber / 2; |
| |
| StrokeTransform transform; |
| double cs, sn, totalLen1, totalLen2, constK, constQ, constB, constD, constA; |
| UINT cpCount1, cpCount2; |
| TPointD stroke1Centroid, stroke2Centroid, stroke1Begin, stroke2Begin, stroke1End, stroke2End, versor1, versor2; |
| std::vector<TPointD> samplingPoint1(samplingPointNumber), samplingPoint2(samplingPointNumber); |
| TStroke *stroke1, *stroke2; |
| std::vector<double> ratioSampling, weigths, subStrokeXScaling, subStrokeYScaling; |
| |
| UINT strokeCount1 = m_firstImage->getStrokeCount(); |
| UINT strokeCount2 = m_lastImage->getStrokeCount(); |
| if (strokeCount1 > strokeCount2) |
| strokeCount1 = strokeCount2; |
| |
| m_transformation.clear(); |
| m_transformation.reserve(strokeCount1); |
| |
| const int maxSubSetNum = (strokeCount1) ? 1000 / strokeCount1 : 1; |
| |
| UINT j, p; |
| |
| for (UINT i = 0; i < strokeCount1; i++) { |
| stroke1 = m_firstImage->getStroke(i); |
| stroke2 = m_lastImage->getStroke(i); |
| |
| |
| cpCount1 = stroke1->getControlPointCount(); |
| cpCount2 = stroke2->getControlPointCount(); |
| |
| transform.m_firstStrokeCornerIndexes.clear(); |
| transform.m_secondStrokeCornerIndexes.clear(); |
| transform.m_translate = TPointD(); |
| transform.m_rotationAndScaleCenter = TPointD(); |
| transform.m_scaleX = 0; |
| transform.m_scaleY = 0; |
| transform.m_rotation = 0; |
| |
| bool isEqual = true; |
| |
| if (cpCount1 == cpCount2) { |
| for (j = 0; j < cpCount1 && isEqual; j++) { |
| isEqual = (stroke1->getControlPoint(j) == stroke2->getControlPoint(j)); |
| } |
| } else |
| isEqual = false; |
| |
| if (isEqual) { |
| transform.m_type = StrokeTransform::EQUAL; |
| } else { |
| totalLen1 = stroke1->getLength(); |
| totalLen2 = stroke2->getLength(); |
| |
| if (totalLen1 == 0 || totalLen2 == 0) { |
| if (totalLen1 == 0 && totalLen2 == 0) { |
| transform.m_type = StrokeTransform::POINT; |
| } else { |
| transform.m_inverse = TAffine(); |
| transform.m_firstStrokeCornerIndexes.resize(2); |
| transform.m_firstStrokeCornerIndexes[0] = 0; |
| transform.m_firstStrokeCornerIndexes[1] = stroke1->getChunkCount(); |
| transform.m_secondStrokeCornerIndexes.resize(2); |
| transform.m_secondStrokeCornerIndexes[0] = 0; |
| transform.m_secondStrokeCornerIndexes[1] = stroke2->getChunkCount(); |
| } |
| } else { |
| |
| const double startMinAngle = 30.0; |
| std::vector<std::pair<int, double>> angles1, angles2; |
| |
| transform.m_type = StrokeTransform::GENERAL; |
| |
| double minAngle, maxAngle; |
| int minAngle1, maxAngle1, minAngle2, maxAngle2; |
| |
| angles1.clear(); |
| angles2.clear(); |
| |
| |
| |
| |
| detectCorners(stroke1, startMinAngle, angles1, minAngle, maxAngle); |
| minAngle1 = (int)minAngle; |
| maxAngle1 = (int)maxAngle; |
| |
| |
| |
| |
| detectCorners(stroke2, startMinAngle, angles2, minAngle, maxAngle); |
| minAngle2 = (int)minAngle; |
| maxAngle2 = (int)maxAngle; |
| |
| if (angles1.empty()) |
| angles2.clear(); |
| if (angles2.empty()) |
| angles1.clear(); |
| |
| |
| |
| |
| |
| |
| |
| double bestValue = (std::numeric_limits<double>::max)(); |
| |
| if (angles1.size() != angles2.size()) { |
| bestValue = (std::numeric_limits<double>::max)(); |
| |
| |
| |
| if (isTooComplex(angles1.size(), angles2.size(), maxSubSetNum)) { |
| |
| int firstAngle = (int)((angles1.size() < angles2.size()) ? minAngle2 : minAngle1); |
| int lastAngle = (int)((angles1.size() < angles2.size()) ? maxAngle1 : maxAngle2); |
| int bestAngle = (int)startMinAngle; |
| |
| if ((int)(angles1.size() + angles2.size()) < lastAngle - firstAngle + 1) { |
| double tempAngle; |
| std::vector<double> sortedAngles1, sortedAngles2; |
| sortedAngles1.reserve(angles1.size()); |
| sortedAngles2.reserve(angles2.size()); |
| for (j = 0; j < angles1.size(); j++) { |
| tempAngle = angles1[j].second; |
| if (tempAngle >= firstAngle && tempAngle <= lastAngle) |
| sortedAngles1.push_back(tempAngle); |
| } |
| for (j = 0; j < angles2.size(); j++) { |
| tempAngle = angles2[j].second; |
| if (tempAngle >= firstAngle && tempAngle <= lastAngle) |
| sortedAngles2.push_back(tempAngle); |
| } |
| std::vector<double> sortedAngles(sortedAngles1.size() + sortedAngles2.size()); |
| |
| std::sort(sortedAngles1.begin(), sortedAngles1.end()); |
| std::sort(sortedAngles2.begin(), sortedAngles2.end()); |
| std::merge(sortedAngles1.begin(), sortedAngles1.end(), sortedAngles2.begin(), sortedAngles2.end(), sortedAngles.begin()); |
| |
| for (j = 0; j < sortedAngles.size(); j++) { |
| int numAng1 = angleNumber(angles1, sortedAngles[j]); |
| int numAng2 = angleNumber(angles2, sortedAngles[j]); |
| double val = (numAng1 == numAng2) ? 0 : fabs((float)(numAng1 - numAng2)) / (numAng1 + numAng2); |
| if (val < bestValue) { |
| bestValue = val; |
| bestAngle = (int)(sortedAngles[j]); |
| if (bestValue == 0 || !isTooComplex(numAng1, numAng2, maxSubSetNum)) |
| break; |
| } |
| } |
| |
| } else |
| { |
| for (int angle = firstAngle; angle <= lastAngle; angle++) { |
| int numAng1 = angleNumber(angles1, angle); |
| int numAng2 = angleNumber(angles2, angle); |
| double val = (numAng1 == numAng2) ? 0 : fabs((float)(numAng1 - numAng2)) / (numAng1 + numAng2); |
| if (val < bestValue) { |
| bestValue = val; |
| bestAngle = angle; |
| if (bestValue == 0 || !isTooComplex(numAng1, numAng2, maxSubSetNum)) |
| break; |
| } |
| } |
| } |
| |
| eraseSmallAngles(angles1, bestAngle); |
| eraseSmallAngles(angles2, bestAngle); |
| |
| |
| |
| |
| |
| |
| } |
| |
| |
| bestValue = (std::numeric_limits<double>::max)(); |
| |
| if (angles1.size() == angles2.size()) { |
| transform.m_firstStrokeCornerIndexes.push_back(0); |
| for (j = 0; j < angles1.size(); j++) |
| transform.m_firstStrokeCornerIndexes.push_back(angles1[j].first); |
| transform.m_firstStrokeCornerIndexes.push_back(stroke1->getChunkCount()); |
| |
| transform.m_secondStrokeCornerIndexes.push_back(0); |
| for (j = 0; j < angles2.size(); j++) |
| transform.m_secondStrokeCornerIndexes.push_back(angles2[j].first); |
| transform.m_secondStrokeCornerIndexes.push_back(stroke2->getChunkCount()); |
| } else { |
| if (isTooComplex(angles1.size(), angles2.size(), maxSubSetNum)) { |
| if (angles1.size() > angles2.size()) { |
| transform.m_firstStrokeCornerIndexes.resize(angles2.size()); |
| trivialSolution(stroke1, stroke2, angles1, angles2, transform.m_firstStrokeCornerIndexes); |
| transform.m_firstStrokeCornerIndexes.insert(transform.m_firstStrokeCornerIndexes.begin(), 0); |
| transform.m_firstStrokeCornerIndexes.push_back(stroke1->getChunkCount()); |
| |
| transform.m_secondStrokeCornerIndexes.push_back(0); |
| for (j = 0; j < angles2.size(); j++) |
| transform.m_secondStrokeCornerIndexes.push_back(angles2[j].first); |
| transform.m_secondStrokeCornerIndexes.push_back(stroke2->getChunkCount()); |
| } else { |
| transform.m_firstStrokeCornerIndexes.push_back(0); |
| for (j = 0; j < angles1.size(); j++) |
| transform.m_firstStrokeCornerIndexes.push_back(angles1[j].first); |
| transform.m_firstStrokeCornerIndexes.push_back(stroke1->getChunkCount()); |
| |
| transform.m_secondStrokeCornerIndexes.resize(angles1.size()); |
| trivialSolution(stroke2, stroke1, angles2, angles1, transform.m_secondStrokeCornerIndexes); |
| transform.m_secondStrokeCornerIndexes.insert(transform.m_secondStrokeCornerIndexes.begin(), 0); |
| transform.m_secondStrokeCornerIndexes.push_back(stroke2->getChunkCount()); |
| } |
| } else { |
| if (angles1.size() > angles2.size()) { |
| transform.m_firstStrokeCornerIndexes.resize(angles2.size()); |
| findBestSolution(stroke1, stroke2, angles1, angles2, bestValue, transform.m_firstStrokeCornerIndexes); |
| transform.m_firstStrokeCornerIndexes.insert(transform.m_firstStrokeCornerIndexes.begin(), 0); |
| transform.m_firstStrokeCornerIndexes.push_back(stroke1->getChunkCount()); |
| |
| transform.m_secondStrokeCornerIndexes.push_back(0); |
| for (j = 0; j < angles2.size(); j++) |
| transform.m_secondStrokeCornerIndexes.push_back(angles2[j].first); |
| transform.m_secondStrokeCornerIndexes.push_back(stroke2->getChunkCount()); |
| } else { |
| transform.m_firstStrokeCornerIndexes.push_back(0); |
| for (j = 0; j < angles1.size(); j++) |
| transform.m_firstStrokeCornerIndexes.push_back(angles1[j].first); |
| transform.m_firstStrokeCornerIndexes.push_back(stroke1->getChunkCount()); |
| |
| transform.m_secondStrokeCornerIndexes.resize(angles1.size()); |
| findBestSolution(stroke2, stroke1, angles2, angles1, bestValue, transform.m_secondStrokeCornerIndexes); |
| transform.m_secondStrokeCornerIndexes.insert(transform.m_secondStrokeCornerIndexes.begin(), 0); |
| transform.m_secondStrokeCornerIndexes.push_back(stroke2->getChunkCount()); |
| } |
| } |
| } |
| } else { |
| transform.m_firstStrokeCornerIndexes.push_back(0); |
| for (j = 0; j < angles1.size(); j++) |
| transform.m_firstStrokeCornerIndexes.push_back(angles1[j].first); |
| transform.m_firstStrokeCornerIndexes.push_back(stroke1->getChunkCount()); |
| |
| transform.m_secondStrokeCornerIndexes.push_back(0); |
| for (j = 0; j < angles2.size(); j++) |
| transform.m_secondStrokeCornerIndexes.push_back(angles2[j].first); |
| transform.m_secondStrokeCornerIndexes.push_back(stroke2->getChunkCount()); |
| } |
| |
| UINT cornerSize = transform.m_firstStrokeCornerIndexes.size(); |
| assert(cornerSize == transform.m_secondStrokeCornerIndexes.size()); |
| assert(cornerSize >= 2); |
| |
| double totalRadRotation = 0; |
| |
| TStroke *subStroke1 = 0; |
| TStroke *subStroke2 = 0; |
| |
| stroke1Centroid = stroke1->getCentroid(); |
| stroke2Centroid = stroke2->getCentroid(); |
| |
| #ifdef _DEBUG |
| assert(transform.m_firstStrokeCornerIndexes[0] == 0); |
| assert(transform.m_secondStrokeCornerIndexes[0] == 0); |
| assert(transform.m_firstStrokeCornerIndexes[cornerSize - 1] == stroke1->getChunkCount()); |
| assert(transform.m_secondStrokeCornerIndexes[cornerSize - 1] == stroke2->getChunkCount()); |
| for (j = 0; j < cornerSize - 1; j++) { |
| assert(transform.m_firstStrokeCornerIndexes[j] < transform.m_firstStrokeCornerIndexes[j + 1]); |
| assert(transform.m_secondStrokeCornerIndexes[j] < transform.m_secondStrokeCornerIndexes[j + 1]); |
| |
| assert(transform.m_firstStrokeCornerIndexes[j] < stroke1->getChunkCount()); |
| assert(transform.m_secondStrokeCornerIndexes[j] < stroke2->getChunkCount()); |
| } |
| #endif |
| |
| for (j = 0; j < cornerSize - 1; j++) { |
| |
| |
| subStroke1 = extract(stroke1, transform.m_firstStrokeCornerIndexes[j], transform.m_firstStrokeCornerIndexes[j + 1] - 1); |
| sample(subStroke1, samplingPointNumber, samplingPoint1); |
| subStroke2 = extract(stroke2, transform.m_secondStrokeCornerIndexes[j], transform.m_secondStrokeCornerIndexes[j + 1] - 1); |
| sample(subStroke2, samplingPointNumber, samplingPoint2); |
| |
| |
| |
| ratioSampling.clear(); |
| ratioSampling.reserve(samplingPointNumber); |
| weigths.clear(); |
| weigths.reserve(samplingPointNumber); |
| |
| TPointD pOld, pNew; |
| |
| |
| for (p = 0; p < samplingPointNumber; p++) { |
| pOld = samplingPoint1[p]; |
| pNew = samplingPoint2[p]; |
| if (pOld == stroke1Centroid) |
| continue; |
| if (pNew == stroke2Centroid) |
| continue; |
| versor1 = normalize(pOld - stroke1Centroid); |
| versor2 = normalize(pNew - stroke2Centroid); |
| weigths.push_back(tdistance(pOld, stroke1Centroid) + tdistance(pNew, stroke2Centroid)); |
| cs = versor1 * versor2; |
| sn = cross(versor1, versor2); |
| double v = atan2(sn, cs); |
| ratioSampling.push_back(v); |
| } |
| delete subStroke1; |
| delete subStroke2; |
| subStroke1 = 0; |
| subStroke2 = 0; |
| |
| double radRotation = weightedAverage(ratioSampling, weigths); |
| |
| totalRadRotation += radRotation; |
| } |
| totalRadRotation /= (cornerSize - 1); |
| transform.m_rotation = totalRadRotation * M_180_PI; |
| |
| if (isAlmostZero(transform.m_rotation, 2)) { |
| transform.m_rotation = 0.0; |
| totalRadRotation = 0.0; |
| } |
| |
| #ifdef _DEBUG |
| |
| |
| #endif |
| |
| |
| |
| if (transform.m_rotation == 0.0) { |
| |
| subStrokeXScaling.clear(); |
| subStrokeXScaling.reserve(cornerSize - 1); |
| subStrokeYScaling.clear(); |
| subStrokeYScaling.reserve(cornerSize - 1); |
| |
| for (j = 0; j < cornerSize - 1; j++) { |
| |
| |
| subStroke1 = extract(stroke1, transform.m_firstStrokeCornerIndexes[j], transform.m_firstStrokeCornerIndexes[j + 1] - 1); |
| sample(subStroke1, samplingPointNumber, samplingPoint1); |
| subStroke2 = extract(stroke2, transform.m_secondStrokeCornerIndexes[j], transform.m_secondStrokeCornerIndexes[j + 1] - 1); |
| sample(subStroke2, samplingPointNumber, samplingPoint2); |
| |
| |
| |
| ratioSampling.clear(); |
| ratioSampling.reserve(samplingPointNumber + bboxSamplingWeight); |
| double appX, appY; |
| |
| TPointD appPoint; |
| double bboxXMin, bboxYMin, bboxXMax, bboxYMax; |
| bboxXMin = bboxYMin = (std::numeric_limits<double>::max)(); |
| bboxXMax = bboxYMax = -(std::numeric_limits<double>::max)(); |
| int h; |
| |
| for (h = 0; h < subStroke1->getControlPointCount(); ++h) { |
| appPoint = subStroke1->getControlPoint(h); |
| if (appPoint.x < bboxXMin) |
| bboxXMin = appPoint.x; |
| if (appPoint.x > bboxXMax) |
| bboxXMax = appPoint.x; |
| if (appPoint.y < bboxYMin) |
| bboxYMin = appPoint.y; |
| if (appPoint.y > bboxYMax) |
| bboxYMax = appPoint.y; |
| } |
| |
| appX = bboxXMax - bboxXMin; |
| appY = bboxYMax - bboxYMin; |
| |
| if (appX) { |
| bboxXMin = (std::numeric_limits<double>::max)(); |
| bboxXMax = -(std::numeric_limits<double>::max)(); |
| for (h = 0; h < subStroke2->getControlPointCount(); ++h) { |
| appPoint = subStroke2->getControlPoint(h); |
| if (appPoint.x < bboxXMin) |
| bboxXMin = appPoint.x; |
| if (appPoint.x > bboxXMax) |
| bboxXMax = appPoint.x; |
| } |
| appX = (isAlmostZero(appX, 1e-01)) ? -1 : (bboxXMax - bboxXMin) / appX; |
| for (UINT tms = 0; tms < bboxSamplingWeight && appX >= 0; tms++) |
| ratioSampling.push_back(appX); |
| |
| for (p = 0; p < samplingPointNumber; p++) { |
| appX = fabs(samplingPoint1[p].x - stroke1Centroid.x); |
| if (appX) |
| ratioSampling.push_back(fabs(samplingPoint2[p].x - stroke2Centroid.x) / appX); |
| } |
| |
| if (!ratioSampling.empty()) { |
| subStrokeXScaling.push_back(average(ratioSampling)); |
| } |
| } |
| |
| |
| |
| ratioSampling.clear(); |
| ratioSampling.reserve(samplingPointNumber + bboxSamplingWeight); |
| |
| if (appY) { |
| bboxYMin = (std::numeric_limits<double>::max)(); |
| bboxYMax = -(std::numeric_limits<double>::max)(); |
| for (h = 0; h < subStroke2->getControlPointCount(); ++h) { |
| appPoint = subStroke2->getControlPoint(h); |
| if (appPoint.y < bboxYMin) |
| bboxYMin = appPoint.y; |
| if (appPoint.y > bboxYMax) |
| bboxYMax = appPoint.y; |
| } |
| |
| appY = (isAlmostZero(appY, 1e-01)) ? -1 : (bboxYMax - bboxYMin) / appY; |
| for (UINT tms = 0; tms < bboxSamplingWeight && appY >= 0; tms++) |
| ratioSampling.push_back(appY); |
| |
| for (p = 0; p < samplingPointNumber; p++) { |
| appY = fabs(samplingPoint1[p].y - stroke1Centroid.y); |
| if (appY) |
| ratioSampling.push_back(fabs(samplingPoint2[p].y - stroke2Centroid.y) / appY); |
| } |
| |
| if (!ratioSampling.empty()) { |
| subStrokeYScaling.push_back(average(ratioSampling)); |
| } |
| } |
| |
| delete subStroke1; |
| delete subStroke2; |
| subStroke1 = 0; |
| subStroke2 = 0; |
| } |
| |
| if (subStrokeXScaling.empty()) { |
| transform.m_scaleX = 1.0; |
| } else { |
| transform.m_scaleX = average(subStrokeXScaling); |
| if (isAlmostZero(transform.m_scaleX - 1.0)) |
| transform.m_scaleX = 1.0; |
| } |
| |
| if (subStrokeYScaling.empty()) { |
| transform.m_scaleY = 1.0; |
| } else { |
| transform.m_scaleY = average(subStrokeYScaling); |
| if (isAlmostZero(transform.m_scaleY - 1.0)) |
| transform.m_scaleY = 1.0; |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| } else { |
| subStrokeXScaling.clear(); |
| subStrokeXScaling.reserve(cornerSize - 1); |
| |
| for (j = 0; j < cornerSize - 1; j++) { |
| |
| |
| subStroke1 = extract(stroke1, transform.m_firstStrokeCornerIndexes[j], transform.m_firstStrokeCornerIndexes[j + 1] - 1); |
| sample(subStroke1, samplingPointNumber, samplingPoint1); |
| subStroke2 = extract(stroke2, transform.m_secondStrokeCornerIndexes[j], transform.m_secondStrokeCornerIndexes[j + 1] - 1); |
| sample(subStroke2, samplingPointNumber, samplingPoint2); |
| |
| |
| |
| ratioSampling.clear(); |
| ratioSampling.reserve(samplingPointNumber + bboxSamplingWeight); |
| |
| TRectD bbox1 = subStroke1->getBBox(); |
| double app = tdistance2(bbox1.getP00(), bbox1.getP11()); |
| if (app) { |
| TRectD bbox2 = TRotation(transform.m_rotation).inv() * subStroke2->getBBox(); |
| app = sqrt(tdistance2(bbox2.getP00(), bbox2.getP11()) / app); |
| |
| for (UINT tms = 0; tms < bboxSamplingWeight; tms++) |
| ratioSampling.push_back(app); |
| |
| double app; |
| for (p = 0; p < samplingPointNumber; p++) { |
| app = tdistance(samplingPoint1[p], stroke1Centroid); |
| if (app) { |
| ratioSampling.push_back(tdistance(samplingPoint2[p], stroke2Centroid) / app); |
| } |
| } |
| } |
| if (!ratioSampling.empty()) { |
| subStrokeXScaling.push_back(average(ratioSampling)); |
| } |
| |
| delete subStroke1; |
| delete subStroke2; |
| subStroke1 = 0; |
| subStroke2 = 0; |
| } |
| |
| if (subStrokeXScaling.empty()) { |
| transform.m_scaleX = transform.m_scaleY = 1.0; |
| } else { |
| transform.m_scaleX = transform.m_scaleY = average(subStrokeXScaling); |
| if (isAlmostZero(transform.m_scaleX - 1.0, 0.00001)) |
| transform.m_scaleX = transform.m_scaleY = 1.0; |
| } |
| |
| |
| |
| |
| |
| |
| |
| } |
| |
| |
| |
| std::vector<TPointD> vpOld(cornerSize), vpNew(cornerSize); |
| TPointD pOld, pNew; |
| |
| for (j = 0; j < cornerSize - 1; j++) { |
| vpOld[j] = stroke1->getChunk(transform.m_firstStrokeCornerIndexes[j])->getP0(); |
| vpNew[j] = stroke2->getChunk(transform.m_secondStrokeCornerIndexes[j])->getP0(); |
| } |
| vpOld[j] = stroke1->getControlPoint(stroke1->getControlPointCount()); |
| vpNew[j] = stroke2->getControlPoint(stroke2->getControlPointCount()); |
| |
| if (transform.m_rotation == 0.0) { |
| if (transform.m_scaleX == 1.0 && transform.m_scaleY == 1.0) { |
| transform.m_translate = stroke2Centroid - stroke1Centroid; |
| transform.m_rotationAndScaleCenter = TPointD(); |
| } else { |
| if (transform.m_scaleX == 1.0) { |
| transform.m_rotationAndScaleCenter.x = 0; |
| transform.m_translate.x = 0; |
| for (j = 0; j < cornerSize; j++) { |
| transform.m_translate.x += vpNew[j].x - vpOld[j].x; |
| } |
| transform.m_translate.x = transform.m_translate.x / cornerSize; |
| } else { |
| transform.m_rotationAndScaleCenter.x = 0; |
| |
| for (j = 0; j < cornerSize; j++) { |
| pOld = vpOld[j]; |
| pNew = vpNew[j]; |
| transform.m_rotationAndScaleCenter.x += (transform.m_scaleX * pOld.x - pNew.x) / (transform.m_scaleX - 1.0); |
| } |
| transform.m_rotationAndScaleCenter.x = transform.m_rotationAndScaleCenter.x / cornerSize; |
| transform.m_translate.x = 0; |
| } |
| |
| if (transform.m_scaleY == 1.0) { |
| transform.m_rotationAndScaleCenter.y = 0; |
| transform.m_translate.y = 0; |
| for (j = 0; j < cornerSize; j++) { |
| transform.m_translate.y += vpNew[j].y - vpOld[j].y; |
| } |
| transform.m_translate.y = transform.m_translate.y / cornerSize; |
| } else { |
| transform.m_rotationAndScaleCenter.y = 0; |
| |
| for (j = 0; j < cornerSize; j++) { |
| pOld = vpOld[j]; |
| pNew = vpNew[j]; |
| transform.m_rotationAndScaleCenter.y += (transform.m_scaleY * pOld.y - pNew.y) / (transform.m_scaleY - 1.0); |
| } |
| transform.m_rotationAndScaleCenter.y = transform.m_rotationAndScaleCenter.y / cornerSize; |
| transform.m_translate.y = 0; |
| } |
| } |
| |
| } else { |
| assert(transform.m_scaleX == transform.m_scaleY); |
| |
| cs = transform.m_scaleX * cos(totalRadRotation); |
| sn = transform.m_scaleX * sin(totalRadRotation); |
| |
| |
| |
| |
| |
| |
| for (j = 0; j < cornerSize; j++) { |
| pOld = vpOld[j]; |
| pNew = vpNew[j]; |
| constK = pNew.x - cs * pOld.x + sn * pOld.y; |
| constQ = pNew.y - sn * pOld.x - cs * pOld.y; |
| constB = 2 * (constK * (cs - 1.) + sn * constQ); |
| constD = 2 * (constQ * (cs - 1.) - sn * constK); |
| constA = transform.m_scaleX * transform.m_scaleX + 1 - 2 * cs; |
| assert(constA > 0); |
| constA = 1.0 / (2 * constA); |
| transform.m_rotationAndScaleCenter.x += -constB * constA; |
| transform.m_rotationAndScaleCenter.y += -constD * constA; |
| } |
| |
| transform.m_rotationAndScaleCenter = transform.m_rotationAndScaleCenter * (1.0 / (double)cornerSize); |
| |
| transform.m_translate.x = (cs - 1.0) * transform.m_rotationAndScaleCenter.x - sn * transform.m_rotationAndScaleCenter.y + constK; |
| |
| transform.m_translate.y = sn * transform.m_rotationAndScaleCenter.x + (cs - 1.0) * transform.m_rotationAndScaleCenter.y + constQ; |
| } |
| |
| |
| |
| transform.m_inverse = (TTranslation(transform.m_translate) * |
| TScale(transform.m_rotationAndScaleCenter, transform.m_scaleX, transform.m_scaleY) * |
| TRotation(transform.m_rotationAndScaleCenter, transform.m_rotation)) |
| .inv(); |
| |
| |
| } |
| |
| } |
| |
| m_transformation.push_back(transform); |
| |
| } |
| } |
| |
| |
| |
| TVectorImageP TInbetween::Imp::tween(double t) const |
| { |
| const double step = 5.0; |
| const double interpolateError = 1.0; |
| |
| TVectorImageP vi = new TVectorImage; |
| |
| UINT strokeCount1 = m_firstImage->getStrokeCount(); |
| UINT strokeCount2 = m_lastImage->getStrokeCount(); |
| vi->setPalette(m_firstImage->getPalette()); |
| if (strokeCount1 > strokeCount2) |
| strokeCount1 = strokeCount2; |
| |
| assert(m_transformation.size() == strokeCount1); |
| |
| double totalLen1, totalLen2, len1, len2, step1, step2; |
| std::vector<TThickPoint> points; |
| TStroke *stroke1, *stroke2, *subStroke1, *subStroke2, *stroke; |
| |
| TAffine mt, invMatrix; |
| TThickPoint point2, finalPoint; |
| UINT i, j, cp, cpSize; |
| |
| for (i = 0; i < strokeCount1; i++) { |
| |
| stroke1 = m_firstImage->getStroke(i); |
| stroke2 = m_lastImage->getStroke(i); |
| |
| if (m_transformation[i].m_type == StrokeTransform::EQUAL) { |
| stroke = new TStroke(*stroke1); |
| } else { |
| points.clear(); |
| totalLen1 = stroke1->getLength(); |
| totalLen2 = stroke2->getLength(); |
| ; |
| |
| if (stroke1->getControlPointCount() == stroke2->getControlPointCount() && |
| stroke1->isSelfLoop() && stroke2->isSelfLoop()) { |
| for (int i = 0; i < stroke1->getControlPointCount(); i++) { |
| TThickPoint p0 = stroke1->getControlPoint(i); |
| TThickPoint p1 = stroke2->getControlPoint(i); |
| points.push_back(p0 * (1 - t) + p1 * t); |
| } |
| stroke = new TStroke(points); |
| } else if (m_transformation[i].m_type == StrokeTransform::POINT) { |
| TThickPoint pOld = stroke1->getThickPointAtLength(0.5 * totalLen1); |
| TThickPoint pNew = stroke2->getThickPointAtLength(0.5 * totalLen2); |
| points.push_back(pOld * (1 - t) + pNew * t); |
| points.push_back(points[0]); |
| points.push_back(points[0]); |
| stroke = new TStroke(points); |
| } else { |
| mt = (m_transformation[i].m_inverse.isIdentity()) |
| ? TAffine() |
| : TTranslation(m_transformation[i].m_translate * t) * |
| TScale(m_transformation[i].m_rotationAndScaleCenter, (1 - t) + t * m_transformation[i].m_scaleX, (1 - t) + t * m_transformation[i].m_scaleY) * |
| TRotation(m_transformation[i].m_rotationAndScaleCenter, m_transformation[i].m_rotation * t); |
| |
| UINT cornerSize = m_transformation[i].m_firstStrokeCornerIndexes.size(); |
| assert(cornerSize == m_transformation[i].m_secondStrokeCornerIndexes.size()); |
| if (cornerSize > m_transformation[i].m_secondStrokeCornerIndexes.size()) |
| cornerSize = m_transformation[i].m_secondStrokeCornerIndexes.size(); |
| |
| assert(cornerSize >= 2); |
| |
| std::vector<TThickPoint> controlPoints; |
| |
| |
| |
| |
| for (j = 0; j < cornerSize - 1; j++) { |
| points.clear(); |
| subStroke1 = extract(stroke1, m_transformation[i].m_firstStrokeCornerIndexes[j], m_transformation[i].m_firstStrokeCornerIndexes[j + 1] - 1); |
| subStroke2 = extract(stroke2, m_transformation[i].m_secondStrokeCornerIndexes[j], m_transformation[i].m_secondStrokeCornerIndexes[j + 1] - 1); |
| |
| totalLen1 = subStroke1->getLength(); |
| totalLen2 = subStroke2->getLength(); |
| |
| if (totalLen1 > totalLen2) { |
| step1 = step; |
| step2 = (totalLen2 / totalLen1) * step; |
| } else { |
| step1 = (totalLen1 / totalLen2) * step; |
| step2 = step; |
| } |
| |
| len1 = 0; |
| len2 = 0; |
| |
| while (len1 <= totalLen1 && len2 <= totalLen2) { |
| point2 = subStroke2->getThickPointAtLength(len2); |
| point2 = TThickPoint(m_transformation[i].m_inverse * subStroke2->getThickPointAtLength(len2), point2.thick); |
| finalPoint = subStroke1->getThickPointAtLength(len1) * (1 - t) + t * point2; |
| |
| points.push_back(TThickPoint(mt * (finalPoint), finalPoint.thick)); |
| len1 += step1; |
| len2 += step2; |
| } |
| point2 = subStroke2->getThickPointAtLength(totalLen2); |
| point2 = TThickPoint(m_transformation[i].m_inverse * subStroke2->getThickPointAtLength(totalLen2), point2.thick); |
| finalPoint = subStroke1->getThickPointAtLength(totalLen1) * (1 - t) + t * point2; |
| |
| points.push_back(TThickPoint(mt * (finalPoint), finalPoint.thick)); |
| |
| stroke = TStroke::interpolate(points, interpolateError, false |
| ); |
| |
| if (j == 0) |
| controlPoints.push_back(stroke->getControlPoint(0)); |
| else |
| controlPoints.back() = (controlPoints.back() + stroke->getControlPoint(0)) * 0.5; |
| |
| cpSize = stroke->getControlPointCount(); |
| for (cp = 1; cp < cpSize; cp++) { |
| controlPoints.push_back(stroke->getControlPoint(cp)); |
| } |
| |
| delete subStroke1; |
| delete subStroke2; |
| delete stroke; |
| subStroke1 = 0; |
| subStroke2 = 0; |
| stroke = 0; |
| } |
| |
| stroke = new TStroke(controlPoints); |
| } |
| } |
| |
| if (stroke1->isSelfLoop() && stroke2->isSelfLoop()) |
| stroke->setSelfLoop(); |
| |
| stroke->setStyle(stroke1->getStyle()); |
| stroke->outlineOptions() = stroke1->outlineOptions(); |
| VIStroke *vs = new VIStroke(stroke, m_firstImage->getVIStroke(i)->m_groupId); |
| vi->m_imp->m_strokes.push_back(vs); |
| |
| } |
| |
| if (m_firstImage->isComputedRegionAlmostOnce()) |
| transferColor(vi); |
| |
| return vi; |
| } |
| |
| |
| |
| void TInbetween::Imp::transferColor(const TVectorImageP &destination) const |
| { |
| const TVectorImageP &original = m_firstImage; |
| destination->setPalette(original->getPalette()); |
| |
| destination->findRegions(); |
| |
| if (destination->getRegionCount()) { |
| UINT strokeCount1 = original->getStrokeCount(); |
| UINT strokeCount2 = destination->getStrokeCount(); |
| if (strokeCount1 > strokeCount2) |
| strokeCount1 = strokeCount2; |
| |
| for (UINT i = 0; i < strokeCount1; i++) { |
| TVectorImage::transferStrokeColors(original, i, destination, i); |
| } |
| } |
| } |
| |
| |
| |
| TVectorImageP TInbetween::tween(double t) const |
| { |
| return m_imp->tween(t); |
| } |
| |
| |
| |