From 8463311870b96c9a5c5db43d5509ba8eb3f18cdb Mon Sep 17 00:00:00 2001 From: Dave Pagurek Date: Sun, 15 Dec 2024 13:38:12 -0500 Subject: [PATCH 1/2] Match perf of textToContours and p5 1.x --- src/type/p5.Font.js | 17 +++++++++-------- 1 file changed, 9 insertions(+), 8 deletions(-) diff --git a/src/type/p5.Font.js b/src/type/p5.Font.js index ab6f23d1f5..cd53abef0c 100644 --- a/src/type/p5.Font.js +++ b/src/type/p5.Font.js @@ -130,16 +130,17 @@ function font(p5, fn) { } textToPoints(str, x, y, width, height, options) { - ({ width, height, options } = this._parseArgs(width, height, options)); - - // lineate and get the glyphs for each line - let commands = this.textToPaths(str, x, y, width, height, options); - - // convert glyphs to points array with {sampleFactor, simplifyThreshold} - return pathToPoints(commands, options); + // By segmenting per contour, pointAtLength becomes much faster + const contourPoints = this.textToContours(str, x, y, width, height, options); + return contourPoints.reduce((acc, next) => { + acc.push(...next); + return acc; + }, []); } textToContours(str, x, y, width, height, options) { + ({ width, height, options } = this._parseArgs(width, height, options)); + const cmds = this.textToPaths(str, x, y, width, height, options); const cmdContours = []; for (const cmd of cmds) { @@ -1135,7 +1136,7 @@ function font(p5, fn) { } let opts = parseOpts(options, { - sampleFactor: 0.25, + sampleFactor: 0.1, simplifyThreshold: 0 }); From 3d42b1fc3cad3a4684b6f07391b77d609086077d Mon Sep 17 00:00:00 2001 From: Dave Pagurek Date: Sun, 15 Dec 2024 14:04:04 -0500 Subject: [PATCH 2/2] Use a library for pointAtLength --- package-lock.json | 6 + package.json | 5 +- src/type/p5.Font.js | 518 ++---------------- src/webgl/text.js | 26 +- test/unit/visual/cases/typography.js | 2 +- .../000.png | Bin 1865 -> 1764 bytes .../Fonts can be converted to points/000.png | Bin 1479 -> 1069 bytes .../Sampling density can be changed/000.png | Bin 1893 -> 1869 bytes 8 files changed, 45 insertions(+), 512 deletions(-) diff --git a/package-lock.json b/package-lock.json index a168d6b02d..361236241e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -9,6 +9,7 @@ "version": "1.9.4", "license": "LGPL-2.1", "dependencies": { + "@davepagurek/bezier-path": "^0.0.2", "acorn": "^8.12.1", "acorn-walk": "^8.3.4", "colorjs.io": "^0.5.2", @@ -415,6 +416,11 @@ "tough-cookie": "^4.1.4" } }, + "node_modules/@davepagurek/bezier-path": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/@davepagurek/bezier-path/-/bezier-path-0.0.2.tgz", + "integrity": "sha512-4L9ddgzZc9DRGyl1RrS3z5nwnVJoyjsAelVG4X1jh4tVxryEHr4H9QavhxW/my6Rn3669Qz6mhv8gd5O/WeFTA==" + }, "node_modules/@esbuild/aix-ppc64": { "version": "0.21.5", "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz", diff --git a/package.json b/package.json index e34e3facd0..15ce6387ea 100644 --- a/package.json +++ b/package.json @@ -22,6 +22,7 @@ }, "version": "1.9.4", "dependencies": { + "@davepagurek/bezier-path": "^0.0.2", "acorn": "^8.12.1", "acorn-walk": "^8.3.4", "colorjs.io": "^0.5.2", @@ -81,9 +82,7 @@ }, "author": "", "husky": { - "hooks": { - - } + "hooks": {} }, "msw": { "workerDirectory": [ diff --git a/src/type/p5.Font.js b/src/type/p5.Font.js index cd53abef0c..7d8008de87 100644 --- a/src/type/p5.Font.js +++ b/src/type/p5.Font.js @@ -30,6 +30,7 @@ * loading fonts from files and urls, and extracting points from their paths. */ import Typr from './lib/Typr.js'; +import { createFromCommands } from '@davepagurek/bezier-path'; function unquote(name) { // Unquote name from CSS @@ -655,497 +656,20 @@ function font(p5, fn) { return num; } - const findDotsAtSegment = (p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t) => { - const t1 = 1 - t; - const t13 = Math.pow(t1, 3); - const t12 = Math.pow(t1, 2); - const t2 = t * t; - const t3 = t2 * t; - const x = t13 * p1x + t12 * 3 * t * c1x + t1 * 3 * t * t * c2x + t3 * p2x; - const y = t13 * p1y + t12 * 3 * t * c1y + t1 * 3 * t * t * c2y + t3 * p2y; - const mx = p1x + 2 * t * (c1x - p1x) + t2 * (c2x - 2 * c1x + p1x); - const my = p1y + 2 * t * (c1y - p1y) + t2 * (c2y - 2 * c1y + p1y); - const nx = c1x + 2 * t * (c2x - c1x) + t2 * (p2x - 2 * c2x + c1x); - const ny = c1y + 2 * t * (c2y - c1y) + t2 * (p2y - 2 * c2y + c1y); - const ax = t1 * p1x + t * c1x; - const ay = t1 * p1y + t * c1y; - const cx = t1 * c2x + t * p2x; - const cy = t1 * c2y + t * p2y; - let alpha = 90 - Math.atan2(mx - nx, my - ny) * 180 / Math.PI; - if (mx > nx || my < ny) { - alpha += 180; - } - return { - x, y, m: { x: mx, y: my }, n: { x: nx, y: ny }, - start: { x: ax, y: ay }, end: { x: cx, y: cy }, alpha - }; - } - - const getPointAtSegmentLength = (p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, length) => { - return length == null ? bezlen(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y) : - findDotsAtSegment(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, - getTatLen(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, length)); - } - - const pointAtLength = (path, length, isTotal) => { - path = path2curve(path); - let x, y, p, l, point; - let sp = '', len = 0, subpaths = {} - for (let i = 0, ii = path.length; i < ii; i++) { - p = path[i]; - if (p[0] === 'M') { - x = +p[1]; - y = +p[2]; - } else { - l = getPointAtSegmentLength(x, y, p[1], p[2], p[3], p[4], p[5], p[6]); - if (len + l > length) { - if (!isTotal) { - point = getPointAtSegmentLength(x, y, p[1], p[2], p[3], p[4], p[5], p[6], length - len); - return { x: point.x, y: point.y, alpha: point.alpha }; - } - } - len += l; - x = +p[5]; - y = +p[6]; - } - sp += p.shift() + p; - } - subpaths.end = sp; - - point = isTotal ? len : findDotsAtSegment - (x, y, p[0], p[1], p[2], p[3], p[4], p[5], 1); - - if (point.alpha) { - point = { x: point.x, y: point.y, alpha: point.alpha }; - } - - return point; - } - - const pathToAbsolute = (pathArray) => { - let res = [], x = 0, y = 0, mx = 0, my = 0, start = 0; - if (!pathArray) { - // console.warn("Unexpected state: undefined pathArray"); // shouldn't happen - return res; - } - if (pathArray[0][0] === 'M') { - x = +pathArray[0][1]; - y = +pathArray[0][2]; - mx = x; - my = y; - start++; - res[0] = ['M', x, y]; - } - - let dots, crz = - pathArray.length === 3 && - pathArray[0][0] === 'M' && - pathArray[1][0].toUpperCase() === 'R' && - pathArray[2][0].toUpperCase() === 'Z'; - - for (let r, pa, i = start, ii = pathArray.length; i < ii; i++) { - res.push((r = [])); - pa = pathArray[i]; - if (pa[0] !== pa[0].toUpperCase()) { - r[0] = pa[0].toUpperCase(); - switch (r[0]) { - case 'A': - r[1] = pa[1]; - r[2] = pa[2]; - r[3] = pa[3]; - r[4] = pa[4]; - r[5] = pa[5]; - r[6] = +(pa[6] + x); - r[7] = +(pa[7] + y); - break; - case 'V': - r[1] = +pa[1] + y; - break; - case 'H': - r[1] = +pa[1] + x; - break; - case 'R': - dots = [x, y].concat(pa.slice(1)); - for (let j = 2, jj = dots.length; j < jj; j++) { - dots[j] = +dots[j] + x; - dots[++j] = +dots[j] + y; - } - res.pop(); - res = res.concat(catmullRom2bezier(dots, crz)); - break; - case 'M': - mx = +pa[1] + x; - my = +pa[2] + y; - break; - default: - for (let j = 1, jj = pa.length; j < jj; j++) { - r[j] = +pa[j] + (j % 2 ? x : y); - } - } - } else if (pa[0] === 'R') { - dots = [x, y].concat(pa.slice(1)); - res.pop(); - res = res.concat(catmullRom2bezier(dots, crz)); - r = ['R'].concat(pa.slice(-2)); - } else { - for (let k = 0, kk = pa.length; k < kk; k++) { - r[k] = pa[k]; - } - } - switch (r[0]) { - case 'Z': - x = mx; - y = my; - break; - case 'H': - x = r[1]; - break; - case 'V': - y = r[1]; - break; - case 'M': - mx = r[r.length - 2]; - my = r[r.length - 1]; - break; - default: - x = r[r.length - 2]; - y = r[r.length - 1]; - } - } - return res; - } - - const path2curve = (path, path2) => { - const p = pathToAbsolute(path), p2 = path2 && pathToAbsolute(path2); - const attrs = { x: 0, y: 0, bx: 0, by: 0, X: 0, Y: 0, qx: null, qy: null }; - const attrs2 = { x: 0, y: 0, bx: 0, by: 0, X: 0, Y: 0, qx: null, qy: null }; - const pcoms1 = []; // path commands of original path p - const pcoms2 = []; // path commands of original path p2 - let ii; - const processPath = (path, d, pcom) => { - let nx, ny, tq = { T: 1, Q: 1 }; - if (!path) { - return ['C', d.x, d.y, d.x, d.y, d.x, d.y]; - } - if (!(path[0] in tq)) { - d.qx = d.qy = null; - } - switch (path[0]) { - case 'M': - d.X = path[1]; - d.Y = path[2]; - break; - case 'A': - path = ['C'].concat(a2c.apply(0, [d.x, d.y].concat(path.slice(1)))); - break; - case 'S': - if (pcom === 'C' || pcom === 'S') { - nx = d.x * 2 - d.bx; - ny = d.y * 2 - d.by; - } else { - nx = d.x; - ny = d.y; - } - path = ['C', nx, ny].concat(path.slice(1)); - break; - case 'T': - if (pcom === 'Q' || pcom === 'T') { - d.qx = d.x * 2 - d.qx; - d.qy = d.y * 2 - d.qy; - } else { - d.qx = d.x; - d.qy = d.y; - } - path = ['C'].concat(q2c(d.x, d.y, d.qx, d.qy, path[1], path[2])); - break; - case 'Q': - d.qx = path[1]; - d.qy = path[2]; - path = ['C'].concat( - q2c(d.x, d.y, path[1], path[2], path[3], path[4]) - ); - break; - case 'L': - path = ['C'].concat(l2c(d.x, d.y, path[1], path[2])); - break; - case 'H': - path = ['C'].concat(l2c(d.x, d.y, path[1], d.y)); - break; - case 'V': - path = ['C'].concat(l2c(d.x, d.y, d.x, path[1])); - break; - case 'Z': - path = ['C'].concat(l2c(d.x, d.y, d.X, d.Y)); - break; - } - return path; - }, - fixArc = (pp, i) => { - if (pp[i].length > 7) { - pp[i].shift(); - const pi = pp[i]; - while (pi.length) { - pcoms1[i] = 'A'; - if (p2) { - pcoms2[i] = 'A'; - } - pp.splice(i++, 0, ['C'].concat(pi.splice(0, 6))); - } - pp.splice(i, 1); - ii = Math.max(p.length, (p2 && p2.length) || 0); - } - }, - fixM = (path1, path2, a1, a2, i) => { - if (path1 && path2 && path1[i][0] === 'M' && path2[i][0] !== 'M') { - path2.splice(i, 0, ['M', a2.x, a2.y]); - a1.bx = 0; - a1.by = 0; - a1.x = path1[i][1]; - a1.y = path1[i][2]; - ii = Math.max(p.length, (p2 && p2.length) || 0); - } - }; - - let pfirst = ''; // temporary holder for original path command - let pcom = ''; // holder for previous path command of original path - - ii = Math.max(p.length, (p2 && p2.length) || 0); - for (let i = 0; i < ii; i++) { - if (p[i]) { - pfirst = p[i][0]; - } // save current path command - if (pfirst !== 'C') { - pcoms1[i] = pfirst; // Save current path command - if (i) { - pcom = pcoms1[i - 1]; - } // Get previous path command pcom - } - p[i] = processPath(p[i], attrs, pcom); - if (pcoms1[i] !== 'A' && pfirst === 'C') { - pcoms1[i] = 'C'; - } - fixArc(p, i); // fixArc adds also the right amount of A:s to pcoms1 - if (p2) { - // the same procedures is done to p2 - if (p2[i]) { - pfirst = p2[i][0]; - } - if (pfirst !== 'C') { - pcoms2[i] = pfirst; - if (i) { - pcom = pcoms2[i - 1]; - } - } - p2[i] = processPath(p2[i], attrs2, pcom); - if (pcoms2[i] !== 'A' && pfirst === 'C') { - pcoms2[i] = 'C'; - } - fixArc(p2, i); - } - fixM(p, p2, attrs, attrs2, i); - fixM(p2, p, attrs2, attrs, i); - const seg = p[i], seg2 = p2 && p2[i], seglen = seg.length, seg2len = p2 && seg2.length; - attrs.x = seg[seglen - 2]; - attrs.y = seg[seglen - 1]; - attrs.bx = parseFloat(seg[seglen - 4]) || attrs.x; - attrs.by = parseFloat(seg[seglen - 3]) || attrs.y; - attrs2.bx = p2 && (parseFloat(seg2[seg2len - 4]) || attrs2.x); - attrs2.by = p2 && (parseFloat(seg2[seg2len - 3]) || attrs2.y); - attrs2.x = p2 && seg2[seg2len - 2]; - attrs2.y = p2 && seg2[seg2len - 1]; - } - - return p2 ? [p, p2] : p; - } - - const a2c = (x1, y1, rx, ry, angle, lac, sweep_flag, x2, y2, recursive) => { - // see: http://www.w3.org/TR/SVG11/implnote.html#ArcImplementationNotes - const PI = Math.PI, _120 = PI * 120 / 180; - let f1, f2, cx, cy, xy; - const rad = PI / 180 * (+angle || 0); - let res = []; - const rotate = (x, y, rad) => { - const X = x * Math.cos(rad) - y * Math.sin(rad), - Y = x * Math.sin(rad) + y * Math.cos(rad); - return { x: X, y: Y }; - }; - - if (!recursive) { - xy = rotate(x1, y1, -rad); - x1 = xy.x; - y1 = xy.y; - xy = rotate(x2, y2, -rad); - x2 = xy.x; - y2 = xy.y; - const x = (x1 - x2) / 2; - const y = (y1 - y2) / 2; - let h = x * x / (rx * rx) + y * y / (ry * ry); - if (h > 1) { - h = Math.sqrt(h); - rx = h * rx; - ry = h * ry; - } - const rx2 = rx * rx, ry2 = ry * ry; - const k = (lac === sweep_flag ? -1 : 1) * Math.sqrt(Math.abs( - (rx2 * ry2 - rx2 * y * y - ry2 * x * x) / (rx2 * y * y + ry2 * x * x))); - - cx = k * rx * y / ry + (x1 + x2) / 2; - cy = k * -ry * x / rx + (y1 + y2) / 2; - f1 = Math.asin(((y1 - cy) / ry).toFixed(9)); - f2 = Math.asin(((y2 - cy) / ry).toFixed(9)); - - f1 = x1 < cx ? PI - f1 : f1; - f2 = x2 < cx ? PI - f2 : f2; - - if (f1 < 0) { - f1 = PI * 2 + f1; - } - if (f2 < 0) { - f2 = PI * 2 + f2; - } - - if (sweep_flag && f1 > f2) { - f1 = f1 - PI * 2; - } - if (!sweep_flag && f2 > f1) { - f2 = f2 - PI * 2; - } - } else { - f1 = recursive[0]; - f2 = recursive[1]; - cx = recursive[2]; - cy = recursive[3]; - } - let df = f2 - f1; - if (Math.abs(df) > _120) { - const f2old = f2, x2old = x2, y2old = y2; - f2 = f1 + _120 * (sweep_flag && f2 > f1 ? 1 : -1); - x2 = cx + rx * Math.cos(f2); - y2 = cy + ry * Math.sin(f2); - res = a2c(x2, y2, rx, ry, angle, 0, sweep_flag, - x2old, y2old, [f2, f2old, cx, cy]); - } - df = f2 - f1; - const c1 = Math.cos(f1), - s1 = Math.sin(f1), - c2 = Math.cos(f2), - s2 = Math.sin(f2), - t = Math.tan(df / 4), - hx = 4 / 3 * rx * t, - hy = 4 / 3 * ry * t, - m1 = [x1, y1], - m2 = [x1 + hx * s1, y1 - hy * c1], - m3 = [x2 + hx * s2, y2 - hy * c2], - m4 = [x2, y2]; - m2[0] = 2 * m1[0] - m2[0]; - m2[1] = 2 * m1[1] - m2[1]; - if (recursive) { - return [m2, m3, m4].concat(res); - } else { - res = [m2, m3, m4].concat(res).join().split(','); - const newres = []; - for (let i = 0, ii = res.length; i < ii; i++) { - newres[i] = i % 2 ? rotate(res[i - 1], res[i], rad).y - : rotate(res[i], res[i + 1], rad).x; - } - return newres; - } - } - - // http://schepers.cc/getting-to-the-point - function catmullRom2bezier(crp, z) { - const d = []; - for (let i = 0, iLen = crp.length; iLen - 2 * !z > i; i += 2) { - const p = [ - { x: +crp[i - 2], y: +crp[i - 1] }, - { x: +crp[i], y: +crp[i + 1] }, - { x: +crp[i + 2], y: +crp[i + 3] }, - { x: +crp[i + 4], y: +crp[i + 5] } - ]; - if (z) { - if (!i) { - p[0] = { x: +crp[iLen - 2], y: +crp[iLen - 1] }; - } else if (iLen - 4 === i) { - p[3] = { x: +crp[0], y: +crp[1] }; - } else if (iLen - 2 === i) { - p[2] = { x: +crp[0], y: +crp[1] }; - p[3] = { x: +crp[2], y: +crp[3] }; - } - } else { - if (iLen - 4 === i) { - p[3] = p[2]; - } else if (!i) { - p[0] = { x: +crp[i], y: +crp[i + 1] }; - } - } - d.push(['C', - (-p[0].x + 6 * p[1].x + p[2].x) / 6, - (-p[0].y + 6 * p[1].y + p[2].y) / 6, - (p[1].x + 6 * p[2].x - p[3].x) / 6, - (p[1].y + 6 * p[2].y - p[3].y) / 6, - p[2].x, p[2].y - ]); - } - return d; - } - - function l2c(x1, y1, x2, y2) { - return [x1, y1, x2, y2, x2, y2]; - } - - function q2c(x1, y1, ax, ay, x2, y2) { - const _13 = 1 / 3, _23 = 2 / 3; - return [_13 * x1 + _23 * ax, _13 * y1 + _23 * ay, - _13 * x2 + _23 * ax, _13 * y2 + _23 * ay, x2, y2]; - } - - const bezlen = (x1, y1, x2, y2, x3, y3, x4, y4, z) => { - z = z ?? 1; - z = z > 1 ? 1 : z < 0 ? 0 : z; - const z2 = z / 2, n = 12; - let sum = 0; - const Tvalues = [-0.1252, 0.1252, -0.3678, 0.3678, -0.5873, 0.5873, -0.7699, 0.7699, -0.9041, 0.9041, -0.9816, 0.9816]; - const Cvalues = [0.2491, 0.2491, 0.2335, 0.2335, 0.2032, 0.2032, 0.1601, 0.1601, 0.1069, 0.1069, 0.0472, 0.0472]; - for (let i = 0; i < n; i++) { - const ct = z2 * Tvalues[i] + z2, xbase = base3(ct, x1, x2, x3, x4), - ybase = base3(ct, y1, y2, y3, y4), comb = xbase * xbase + ybase * ybase; - sum += Cvalues[i] * Math.sqrt(comb); - } - return z2 * sum; - } - - const getTatLen = (x1, y1, x2, y2, x3, y3, x4, y4, ll) => { - if (ll < 0 || bezlen(x1, y1, x2, y2, x3, y3, x4, y4) < ll) { - return; - } - const t = 1, e = 0.01; - let step = t / 2, t2 = t - step; - let l = bezlen(x1, y1, x2, y2, x3, y3, x4, y4, t2); - while (Math.abs(l - ll) > e) { - step /= 2; - t2 += (l < ll ? 1 : -1) * step; - l = bezlen(x1, y1, x2, y2, x3, y3, x4, y4, t2); - } - return t2; - } - - const base3 = (t, p1, p2, p3, p4) => { - const t1 = -3 * p1 + 9 * p2 - 9 * p3 + 3 * p4, - t2 = t * t1 + 6 * p1 - 12 * p2 + 6 * p3; - return t * t2 - 3 * p1 + 3 * p2; - } - + const path = createFromCommands(arrayCommandsToObjects(cmds)); let opts = parseOpts(options, { sampleFactor: 0.1, simplifyThreshold: 0 }); + + const totalPoints = Math.ceil(path.getTotalLength() * opts.sampleFactor); let points = []; - let len = pointAtLength(cmds, 0, 1); - let t = len / (len * opts.sampleFactor); - for (let i = 0; i < len; i += t) { - points.push(pointAtLength(cmds, i)); + for (let i = 0; i < totalPoints; i++) { + points.push( + path.getPointAtLength(path.getTotalLength() * (i / (totalPoints - 1))) + ); } if (opts.simplifyThreshold) { @@ -1156,6 +680,32 @@ function font(p5, fn) { } }; +// Convert arrays to named objects +export const arrayCommandsToObjects = (commands) => commands.map((command) => { + const type = command[0]; + switch (type) { + case 'Z': { + return { type }; + } + case 'M': + case 'L': { + const [, x, y] = command; + return { type, x, y }; + } + case 'Q': { + const [, x1, y1, x, y] = command; + return { type, x1, y1, x, y }; + } + case 'C': { + const [, x1, y1, x2, y2, x, y] = command; + return { type, x1, y1, x2, y2, x, y }; + } + default: { + throw new Error(`Unexpected path command: ${type}`); + } + } +}); + export default font; if (typeof p5 !== 'undefined') { diff --git a/src/webgl/text.js b/src/webgl/text.js index 2e418b82c8..0f75bcb42d 100644 --- a/src/webgl/text.js +++ b/src/webgl/text.js @@ -3,6 +3,7 @@ import * as constants from '../core/constants'; import { RendererGL } from './p5.RendererGL'; import { Vector } from '../math/p5.Vector'; import { Geometry } from './p5.Geometry'; +import { arrayCommandsToObjects } from '../type/p5.Font'; function text(p5, fn){ // Text/Typography @@ -194,30 +195,7 @@ function text(p5, fn){ const gHeight = yMax - yMin; // Convert arrays to named objects - const cmds = commands.map((command) => { - const type = command[0]; - switch (type) { - case 'Z': { - return { type }; - } - case 'M': - case 'L': { - const [, x, y] = command; - return { type, x, y }; - } - case 'Q': { - const [, x1, y1, x, y] = command; - return { type, x1, y1, x, y }; - } - case 'C': { - const [, x1, y1, x2, y2, x, y] = command; - return { type, x1, y1, x2, y2, x, y }; - } - default: { - throw new Error(`Unexpected path command: ${type}`); - } - } - }) + const cmds = arrayCommandsToObjects(commands); let i; const strokes = []; // the strokes in this glyph diff --git a/test/unit/visual/cases/typography.js b/test/unit/visual/cases/typography.js index fb19910e5f..e9aa7a26be 100644 --- a/test/unit/visual/cases/typography.js +++ b/test/unit/visual/cases/typography.js @@ -451,7 +451,7 @@ visualSuite("Typography", function () { p5.background(200); p5.strokeWeight(2); p5.textSize(50); - const contours = font.textToContours('p5*js', 0, 50, { samplingDensity: 0.5 }) + const contours = font.textToContours('p5*js', 0, 50, { sampleFactor: 0.5 }) p5.beginShape(); for (const pts of contours) { p5.beginContour(); diff --git a/test/unit/visual/screenshots/Typography/textToContours/Fonts can be converted to points grouped by contour/000.png b/test/unit/visual/screenshots/Typography/textToContours/Fonts can be converted to points grouped by contour/000.png index 0e53ac0a0f8c7ca58f7611eb8d8a52bb7a093bae..b00b2470d58c2d06ee74baa00ddc2231921ff019 100644 GIT binary patch literal 1764 zcmd6o|3A}-1II^~mhot7tXZV@u5-j(ri3QEZ49&GL)Sj@S*-auO=!%_mkRUwY$97& zrX~53I9nR>^>bF?e2Y^TrulO7EnhB&J15td>wdpK;Pb=l`FuQ{|G-Ohn(AY0h%tmf zAjZCAV$dhE{uFTgYsH~^j;`E7ljd4m%OmLBtWSZj!0Or>AXZ*(INM-3kRH^8;6PQJMks6t`=~47c*~@}?N^JRUFU$`yBHS>X~^AeQv^0Z+xyVx#?{94GVIqW5;PNmB|>HaZ++ZG zH+_0q<2EC)&Per0BBq*wVm^_SZ;(7sGlPK%_&^)aEq`I;U>MSFBfdHE?Gti84)%7R zr>lxGR24Z=+~OIV`p@c=Mm_cBVQPC^f+u5h?JKCKc#fs7q~wsGc1S$^nMYoMHT_qzr^Vh!RDzflq*tuaKXX1rL&^TSj&Hg`mKJjTT;lw@ zyJ`E|7%`DY0BvD}huh9JC;&IQ_bkIZ4RdNonKVP3VP!t)ID=BO;L31d109vs-@yMd zf>yZAgdsd-MlzuzEWmFBBZT&*#?GVk#B&V3-ME>Zt*-phJ8O+_3B!QxQ=9dM8>RPy zZD@P14?{GWRTkTDf_VTS3N|r$2%FXrVC3F{WDVaTwsQaJ##@)S9cQhYT+nPMx6&? z$b;?PF@nQJyll*W(|&NwS1RTy zy=V^QqqS)Z?kB0bRy_OW1v#x*ZpDcdDypg-ThNpp2z#9rfheR+-0zB}_6Vg99*T#v z_|=>OJr$kR--V`{JD()k{BgfM=?zwpRK02~`}~^xi%x%ogGf8I>fm6zG~p`W?G_TR ztE!(2HmQ|JIgg_dN4>}6ARuLh;)0)DGBEpa)9CK#LZY;X~NVdCce_!fm&OHY} z-%yJPEj$Ci_)}=$cSyb5mUQtrIH7hdt{F#*!AIl6QJJ2}0i= z!0sLE0)OyUyI@C@_ixUncTEIV?@BOM|I7_2u{Uzdja0mpqD|AZ9NPX9iLj+2=kagr zk^>H$WgK^U1sh&PYk27^=2h5zJ4PE#&Dv;m-S(yb;yS+%^5wPC;|d)Lt#hCS{OAhhH5GI$?Fq0Pd34TvfRmeI%VUwl z--lhY-UG9!9=;4`-aU#Ren8@7O_y_jq&VEVoar~K2;Nbvj%~U-S-+8=>opR6` z%RYwbtSZo=05USjD4nTm$T|vi!L6;R^#8N)P7m8tq{;MU+JCw&h%bpsRDxj{e*#;b BB2xeW literal 1865 zcmdUw`6CmI1IFi?Ij0OUW^P`ZE3YHdWQ^2$qhnLfx!xplWH#4axpGC4+?{ewycOon z9L<@?5i*t)F~=Oa-uC)_|BLsB&+|M#JU>0ZJgI+RF%qH*q5uFu!oeQxbm-py0|Y$O zU{?OcL*ol|!kh-w4k|7J0Kx(eXzPnaH%^hYlNwSyY3@w*1Yd+_mqwgpJqz&;pigw-flNUlj8Af{BN`Jxw*8Sn>he&vv z2+1y;uTzomHarYivb_NQETyY?1U8J4b@rgHG~PLp-37r_pXB_%lGd`Zw?|sv-F?e@ z+uS_Kn`Saw&If{ZfFaS*(QATyENeV4rN3+$<2%)>ye_%>mPVry$>bJdOiUY*MA}>7 z_VxA65OzXk;lYGOh9P|3Nug3M-(p6jsCl(B~FTUL-m7;Yk!dle)tqR!9P&mC%vS= zLz(q?hMc06DbD(|W z3nhNfTV`Ucp(hcOMHFwx409~Z=vrqUP|2(o8DV+q_iwv9ZO*IYT}Ly1k-Umg{;!HY63 z8?Q{@#68W>Tm@BFMq(lo-B#){PJd-8Y}9+dqOe{@Dl)SPYfF8wx9ggWh@q2Ps{A|JnpNcd5%b~{~!lYtoPB)aD zFWqG!e~JYp9gIP*Nn|0_En0SN!MX1)oZVt=GB`AeBH=Daa%DE-w9T6Zc|C*GRMk1! zQV|&x68<MT^PR-e@G9ToVPMc$*ZIwI8p#m!l3mK%%HeI=&6D61O zi`O%&i87qSrG87eDlof|_WanN0|)jHM0(Jo(#Oy&TAR5PNRgKC@&mZyVu4sjPN41$ zJN#!V|IKK3d8x3xtee!?ji6mvd)#r?Ep~{dSONi;*u$Kaue3$>O5R~dU?Va5@pGKa zWoXh2CD+vXN&;oLEgwV_>DNRVENwYl%R4ie44Cj)m}n2JLI%_|7h4gisvTK?7qEpO z>W8`eynaAQJjA8D65Lr4TJR#R>3|O3rX9dAp%Lh zyicfHRO_URx9|3&E+INQLlV#H85m729H^P8v@K-0qvok*o8KeuDKMBD?pLa}+=c7| zmSiRhEk5%}#e|`aLq=l%X^zM%dtdzV){fl8XRYenbse+o_f3p4y Yyv$7aTvJ=RefW(42U{$<_OwU*zst>iiU0rr diff --git a/test/unit/visual/screenshots/Typography/textToPoints/Fonts can be converted to points/000.png b/test/unit/visual/screenshots/Typography/textToPoints/Fonts can be converted to points/000.png index f616ac1350f856cc496bb22c8e98e50d2def4758..863932ae3b5a730aaefd47844f2afae6b26374da 100644 GIT binary patch literal 1069 zcmeAS@N?(olHy`uVBq!ia0vp^DImV1DlD;uum9 z_jXR=x<>{)Ee}%;oNSoPsGgh}wjeg-(GBCwyIH@spH-Q8zHjoh@88Zp|Ge{Z=&{DV z4O0?7N_8|`d8DXtw1T6fA*6W0jfxOwmL)TokG3u55uA|3CnDWv5V7Xae2I7B2>ql zpJaNep~CxZR>Pj|O8+6+X>DKfl&T&W&5vX`F-5Z2 zUFWh(+Jl;g1sB`5-9J?(78@1saKZT8oU|5a&6 z{B(JilyjbT;k$KA`u3Hg#I5%MLM5jXV|3cx3jE^}lDvN_@7Nbo0#=2BTM@o?mRIY-c{j=C&kO!o#0s zYh1YhU!D@}`T5Gn{-1eu!hZhCPKEcy8}9b+$(b!x@ucOPL8f2tGf!LV2^Wu=1nX=r zmb|#`b$8-2hCYcm*G%_@s>L|Fx&Ob`c;iph>|TQj7wx}Cde=&nM({Fs=BX|g+&}Ht zhq$U6^%hT6M6iGNz?4!S$=niGzXc;egA%M z1bXLXeC(MA{A;@f=kKt0C<*E>e#I&jSoU8=QE6FvA$y2(0sl%aCa0~!;=%%VoL4n9 z1iU?Rl!fK{gjEU(A=ZV4j0bI(XmM~{^^5Uw7&=Pd@V}^NWGcMNvJ;s989ZJ6T-G@y GGywn<^y@|d literal 1479 zcmcJP?LQL;0LSOTjZwBMspM%0na7WcY#Pf`PGKm+?L5tv6`6+}YM%0JkVr~lyI+>77u`|kJo{N8-CuDLrYE1pyY007D^&M40V zhI|cB{-7flXy^mTM0+~f1DITO5da{U?1Dm~;{(?*>+VoX<#>NPb#}n@D!L2-6rMHgxoQ$!cGceemWXg+Q5Fj^X5skf7e` z&0Rynm*t~K;L6j%~)RdNXy-Kw@mOu+G;u$xheL1 zGIQOk0osR53>nFZS5Lqb5M|(g%oDogERi|$5$=13OUY3=|EuBVPZrX4d!@Fi`Y+*8 z;ayidj%aV1nwoZp27U1dqa5`mS2nzokqPO>c$#@Vu?9B6DmMZWx;yMZeFhdSE8e>| zN@-vlzV5!FeSOFa)0en=eXfGKx-@v+Wr`r%e@wv}64x>#B#LL>XFRL1=$+2af<)=9 zx%eN=sjaZWe0wKi`JwE0obxOq|_%ai~fkPJ_Xk%hKWxHMv5S!isTk-JONzoOYx zk$(;`FPSGy22-m>o85_QX#m|$bn$iJDLw3KY&GuF981!n1;$oT)FX@EIX??v`$#PP zXtV8@mn5_kkKUdh+Jr;a3xnjb_kqM4uTn}+57n9YN;#dU6sx>>Bv)eC){*Yx0W0xI zA2%8xbEOW0vCZnaeAse9ZOVkkXg}w%5s|vjP0-l(?pk^uKn!(_uwLTs6|_X^(-C&j^u+v&Kf#^yUrX#wFA) zsLQ%TNY#%=SRGNUQs6SsAp2btrGEFFZ7Ux>_NBaM%asu~4B+6qgHG#_*kC6GyUM^Lb?w zT4<}|=j);=Y&KBrV>NIDXNkBoWm#pjU2)xQ2WOHVK{Ej%aZ7Zs>p8y|*u)h{SN7ki z>kv^zm5j4@8;N5%%tATDqlSP*B3oNFZ0Oh%D?_5gw53AWIzF}#y`FJ=F+6oBQS{vs z`fz0l3^blf2o)tr*Dm~B2wor>)%TyC3gqu(rk=nY-f%D?|53|^N-%5|ot$Q^`LIDfmnvvj1xi!9N;m$kCyUtQjfYxCIu>uQ19VRp)=*w0F&!*qd3v;il; zm$O}|h4bgS-HGPh0cj*s%NWXMyk6HOP><$gm}C!yuG&FnRzOuS{X=Y1}68v01H zpr<6zSLFcYYoL$iWP$-o?oQ62Zw_2lmS3PMn=4~>;q-rU*8eh~Vusw8NwIxrTFEU9 QIxsW9<+3}9X^&0$7l=Z*iU0rr diff --git a/test/unit/visual/screenshots/Typography/textToPoints/Sampling density can be changed/000.png b/test/unit/visual/screenshots/Typography/textToPoints/Sampling density can be changed/000.png index f7b8b755bb670d848785648427dad65c623ed9d9..b9ee9088a4ea69493e531673ea3f2dc45e1eb62a 100644 GIT binary patch literal 1869 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