import { parseSVG } from 'svg-path-parser'; import { transformBlend, transformEffects, transformLayoutAttributes, transformProportion, transformSceneNode, transformStrokesFromVector } from '@plugin/transformers/partials'; import { translateFills } from '@plugin/translators/fills'; import { translateCommandsToSegments, translateWindingRule } from '@plugin/translators/vectors'; import { PathShape } from '@ui/lib/types/shapes/pathShape'; export const transformVectorPaths = (node: VectorNode): PathShape[] => { const pathShapes = node.vectorPaths .filter((vectorPath, index) => { return ( nodeHasFills(node, vectorPath, (node.vectorNetwork.regions ?? [])[index]) || node.strokes.length > 0 ); }) .map((vectorPath, index) => transformVectorPath(node, vectorPath, (node.vectorNetwork.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: translateCommandsToSegments( normalizedPaths, node.absoluteTransform[0][2], node.absoluteTransform[1][2] ), fills: vectorPath.windingRule === 'NONE' ? [] : translateFills(vectorRegion?.fills ?? node.fills), svgAttrs: { fillRule: translateWindingRule(vectorPath.windingRule) }, constraintsH: 'scale', constraintsV: 'scale', ...transformStrokesFromVector(node, normalizedPaths, vectorRegion), ...transformEffects(node), ...transformSceneNode(node), ...transformBlend(node), ...transformProportion(node), ...transformLayoutAttributes(node) }; };