0
Fork 0
mirror of https://github.com/penpot/penpot-exporter-figma-plugin.git synced 2025-01-20 06:22:38 -05:00
penpot-exporter-figma-plugin/plugin-src/transformers/partials/transformVectorPaths.ts
Jordi Sala Morales 2d0b63d5cd
Fix vector network error on invalid access to the property (#225)
* Fix vector network error on invalid access to the property

* Add changeset
2024-10-08 11:27:32 +02:00

87 lines
2.5 KiB
TypeScript

import { parseSVG } from 'svg-path-parser';
import {
transformBlend,
transformEffects,
transformLayoutAttributes,
transformProportion,
transformSceneNode,
transformStrokesFromVector,
transformVectorFills
} from '@plugin/transformers/partials';
import { translateCommands, translateWindingRule } from '@plugin/translators/vectors';
import { PathShape } from '@ui/lib/types/shapes/pathShape';
export const transformVectorPaths = (node: VectorNode): PathShape[] => {
let regions: readonly VectorRegion[] = [];
try {
regions = node.vectorNetwork?.regions ?? [];
} catch (error) {
console.error('Error accessing vector network', node, error);
}
const strokeLength = node.strokes.length;
const pathShapes = node.vectorPaths
.filter((vectorPath, index) => {
return nodeHasFills(node, vectorPath, regions[index]) || strokeLength > 0;
})
.map((vectorPath, index) => transformVectorPath(node, vectorPath, regions[index]));
const geometryShapes = node.fillGeometry
.filter(
geometry =>
!node.vectorPaths.find(
vectorPath => normalizePath(vectorPath.data) === normalizePath(geometry.data)
)
)
.map(geometry => transformVectorPath(node, geometry, undefined));
return [...geometryShapes, ...pathShapes];
};
const normalizePath = (path: string): string => {
// Round to 2 decimal places all numbers
const str = path.replace(/(\d+\.\d+|\d+)/g, (match: string) => {
return parseFloat(match).toFixed(2);
});
// remove spaces
return str.replace(/\s/g, '');
};
const nodeHasFills = (
node: VectorNode,
vectorPath: VectorPath,
vectorRegion: VectorRegion | undefined
): boolean => {
return !!(vectorPath.windingRule !== 'NONE' && (vectorRegion?.fills || node.fills));
};
const transformVectorPath = (
node: VectorNode,
vectorPath: VectorPath,
vectorRegion: VectorRegion | undefined
): PathShape => {
const normalizedPaths = parseSVG(vectorPath.data);
return {
type: 'path',
name: 'svg-path',
content: translateCommands(node, normalizedPaths),
svgAttrs: {
fillRule: translateWindingRule(vectorPath.windingRule)
},
constraintsH: 'scale',
constraintsV: 'scale',
...transformVectorFills(node, vectorPath, vectorRegion),
...transformStrokesFromVector(node, normalizedPaths, vectorRegion),
...transformEffects(node),
...transformSceneNode(node),
...transformBlend(node),
...transformProportion(node),
...transformLayoutAttributes(node)
};
};