From f238d6a213a499deed31895d7ebbf139eb9996c5 Mon Sep 17 00:00:00 2001 From: Kendell R Date: Wed, 27 Dec 2023 18:41:09 -0500 Subject: [PATCH] feat(convertPathData): allow converting q to t in more cases (#1889) --- plugins/convertPathData.js | 52 +++++++++++++++++++++++++---- test/plugins/convertPathData.34.svg | 16 +++++++++ 2 files changed, 61 insertions(+), 7 deletions(-) create mode 100644 test/plugins/convertPathData.34.svg diff --git a/plugins/convertPathData.js b/plugins/convertPathData.js index 2a6c66507..0ee7824ea 100644 --- a/plugins/convertPathData.js +++ b/plugins/convertPathData.js @@ -397,8 +397,13 @@ function filters( relSubpoint = [0, 0], pathBase = [0, 0], prev = {}; + /** @type {Point | undefined} */ + let qControlPoint; path = path.filter(function (item, index, path) { + const qPoint = qControlPoint; + qControlPoint = undefined; + let command = item.command; let data = item.args; let next = path[index + 1]; @@ -795,14 +800,24 @@ function filters( // t + q → t + t else if ( // @ts-ignore - prev.command === 't' && - // @ts-ignore - Math.abs(data[2] - prev.args[0]) < error && - // @ts-ignore - Math.abs(data[3] - prev.args[1]) < error + prev.command === 't' ) { - command = 't'; - data = data.slice(2); + // @ts-ignore + const predictedControlPoint = reflectPoint(qPoint, item.base); + const realControlPoint = [ + // @ts-ignore + data[0] + item.base[0], + // @ts-ignore + data[1] + item.base[1], + ]; + if ( + Math.abs(predictedControlPoint[0] - realControlPoint[0]) < + error && + Math.abs(predictedControlPoint[1] - realControlPoint[1]) < error + ) { + command = 't'; + data = data.slice(2); + } } } } @@ -873,6 +888,18 @@ function filters( ) return false; + if (command === 'q') { + // @ts-ignore + qControlPoint = [data[0] + item.base[0], data[1] + item.base[1]]; + } else if (command === 't') { + if (qPoint) { + // @ts-ignore + qControlPoint = reflectPoint(qPoint, item.base); + } else { + // @ts-ignore + qControlPoint = item.coords; + } + } prev = item; return true; }); @@ -1138,6 +1165,17 @@ function getDistance(point1, point2) { return Math.hypot(point1[0] - point2[0], point1[1] - point2[1]); } +/** + * Reflects point across another point + * + * @param {Point} input + * @param {Point} base + * @returns {Point} + */ +function reflectPoint(input, base) { + return [2 * base[0] - input[0], 2 * base[1] - input[1]]; +} + /** * Returns coordinates of the curve point corresponding to the certain t * a·(1 - t)³·p1 + b·(1 - t)²·t·p2 + c·(1 - t)·t²·p3 + d·t³·p4, diff --git a/test/plugins/convertPathData.34.svg b/test/plugins/convertPathData.34.svg new file mode 100644 index 000000000..a6e3eba41 --- /dev/null +++ b/test/plugins/convertPathData.34.svg @@ -0,0 +1,16 @@ +Shouldn't incorrectly convert q to t. Should convert q to t when feasible. + +=== + + + + + + +@@@ + + + + + +