var dumpErrors = false; var container; function getViewBox(path) { let bounds = path.getBounds(); return `${(bounds.fLeft-2)*.95} ${(bounds.fTop-2)*.95} ${(bounds.fRight+2)*1.05} ${(bounds.fBottom+2)*1.05}`; } function addSVG(testName, expectedPath, actualPath, message) { if (!dumpErrors) { return; } if (!container) { let styleEl = document.createElement('style'); document.head.appendChild(styleEl); let sheet = styleEl.sheet; sheet.insertRule(`svg { border: 1px solid #DDD; max-width: 45%; vertical-align: top; }`, 0); container = document.createElement('div'); document.body.appendChild(container); } let thisTest = document.createElement('div'); thisTest.innerHTML = `

Failed test ${testName}

${message}
`; container.appendChild(thisTest); } const TOLERANCE = 0.0001; function diffPaths(expected, actual) { // Look through commands and see if they are within tolerance. let eCmds = expected.toCmds(), aCmds = actual.toCmds(); if (eCmds.length !== aCmds.length) { //console.log(`Expected: ${JSON.stringify(eCmds)} and Actual: ${JSON.stringify(aCmds)}`); return `Different amount of verbs. Expected had ${eCmds.length}, Actual had ${aCmds.length}`; } for(let idx = 0; idx < eCmds.length; idx++){ let eCmd = eCmds[idx], aCmd = aCmds[idx]; if (eCmd.length !== aCmd.length) { // Should never happen, means WASM code is returning bad ops. return `Command index ${idx} differs in num arguments. Expected had ${eCmd.length}, Actual had ${aCmd.length}`; } let eVerb = eCmd[0], aVerb = aCmd[0]; if (eVerb !== aVerb) { return `Command index ${idx} differs. Expected had ${eVerb}, Actual had ${aVerb}`; } for (let arg = 1; arg < eCmd.length; arg++) { if (Math.abs(eCmd[arg] - aCmd[arg]) > TOLERANCE) { return `Command index ${idx} has different argument for verb ${eVerb} at position ${arg}. Expected had ${eCmd[arg]}, Actual had ${aCmd[arg]}` } } } return null; } describe('PathKit\'s PathOps Behavior', function() { var PATHOP_MAP = {}; var FILLTYPE_MAP = {}; function init() { if (PathKit && !PATHOP_MAP['kIntersect_SkPathOp']) { PATHOP_MAP = { 'kIntersect_SkPathOp': PathKit.PathOp.INTERSECT, 'kDifference_SkPathOp': PathKit.PathOp.DIFFERENCE, 'kUnion_SkPathOp': PathKit.PathOp.UNION, 'kXOR_SkPathOp': PathKit.PathOp.XOR, 'kXOR_PathOp': PathKit.PathOp.XOR, 'kReverseDifference_SkPathOp': PathKit.PathOp.REVERSE_DIFFERENCE, }; FILLTYPE_MAP = { 'kWinding_FillType': PathKit.FillType.WINDING, 'kEvenOdd_FillType': PathKit.FillType.EVENODD, 'kInverseWinding_FillType': PathKit.FillType.INVERSE_WINDING, 'kInverseEvenOdd_FillType': PathKit.FillType.INVERSE_EVENODD, }; } } function getFillType(str) { let e = FILLTYPE_MAP[str]; expect(e).toBeTruthy(`Could not find FillType Enum for ${str}`); return e; } function getPathOp(str) { let e = PATHOP_MAP[str]; expect(e).toBeTruthy(`Could not find PathOp Enum for ${str}`); return e; } it('combines two paths with .op() and matches what we see from C++', function(done) { LoadPathKit.then(catchException(done, () => { init(); // Test JSON created with: // ./out/Clang/pathops_unittest -J ./modules/pathkit/tests/PathOpsOp.json -m PathOpsOp$ fetch('/base/tests/PathOpsOp.json').then((r) => { r.json().then((json) => { expect(json).toBeTruthy(); let testNames = Object.keys(json); // Assert we loaded a non-zero amount of tests, i.e. the JSON is valid. expect(testNames.length > 0).toBeTruthy(); testNames.sort(); for (testName of testNames) { let test = json[testName]; let path1 = PathKit.FromCmds(test.p1); expect(path1).not.toBeNull(`path1 error when loading cmds '${test.p1}'`); path1.setFillType(getFillType(test.fillType1)); let path2 = PathKit.FromCmds(test.p2); expect(path2).not.toBeNull(`path2 error when loading cmds '${test.p2}'`); path2.setFillType(getFillType(test.fillType2)); let combined = path1.op(path2, getPathOp(test.op)); if (test.expectSuccess === 'no') { expect(combined).toBeNull(`Test ${testName} should have not created output, but did`); } else { expect(combined).not.toBeNull(); let expected = PathKit.FromCmds(test.out); // Do a tolerant match. let diff = diffPaths(expected, combined); if (test.expectMatch === 'yes'){ // Check fill type expect(combined.getFillType().value).toEqual(getFillType(test.fillTypeOut).value); // diff should be null if the paths are identical (modulo rounding) if (diff) { expect(`[${testName}] ${diff}`).toBe(''); addSVG('[PathOps] ' + testName, expected, combined, diff); } } else if (test.expectMatch === 'flaky') { // Don't worry about it, at least it didn't crash. } else { if (!diff) { expect(`[${testName}] was expected to have paths that differed`).not.toBe(''); } } expected.delete(); } // combined === path1, so we only have to delete one. path1.delete(); path2.delete(); } done(); }); }); })); }); it('simplifies a path with .simplify() and matches what we see from C++', function(done) { LoadPathKit.then(catchException(done, () => { init(); // Test JSON created with: // ./out/Clang/pathops_unittest -J ./modules/pathkit/tests/PathOpsSimplify.json -m PathOpsSimplify$ fetch('/base/tests/PathOpsSimplify.json').then((r) => { r.json().then((json) => { expect(json).toBeTruthy(); let testNames = Object.keys(json); // Assert we loaded a non-zero amount of tests, i.e. the JSON is valid. expect(testNames.length > 0).toBeTruthy(); testNames.sort(); for (testName of testNames) { let test = json[testName]; let path = PathKit.FromCmds(test.path); expect(path).not.toBeNull(`path1 error when loading cmds '${test.path}'`); path.setFillType(getFillType(test.fillType)); let simplified = path.simplify(); if (test.expectSuccess === 'no') { expect(simplified).toBeNull(`Test ${testName} should have not created output, but did`); } else { expect(simplified).not.toBeNull(); let expected = PathKit.FromCmds(test.out); // Do a tolerant match. let diff = diffPaths(expected, simplified); if (test.expectMatch === 'yes'){ // Check fill type expect(simplified.getFillType().value).toEqual(getFillType(test.fillTypeOut).value); // diff should be null if the paths are identical (modulo rounding) if (diff) { expect(`[${testName}] ${diff}`).toBe(''); addSVG('[Simplify] ' + testName, expected, simplified, diff); } } else if (test.expectMatch === 'flaky') { // Don't worry about it, at least it didn't crash. } else { if (!diff) { expect(`[${testName}] was expected to not match output`).not.toBe(''); } } expected.delete(); } // simplified === path, so we only have to delete one. path.delete(); } done(); }); }); })); }); });