0
Fork 0
mirror of https://github.com/penpot/penpot-exporter-figma-plugin.git synced 2025-01-05 06:10:52 -05:00
penpot-exporter-figma-plugin/plugin-src/translators/translateStrokes.ts

100 lines
2.4 KiB
TypeScript
Raw Normal View History

import { translateFill } from '@plugin/translators/fills';
import { Stroke, StrokeAlignment, StrokeCaps } from '@ui/lib/types/utils/stroke';
export const translateStrokes = (
node: MinimalStrokesMixin | (MinimalStrokesMixin & IndividualStrokesMixin),
strokeCaps: (stroke: Stroke) => Stroke = stroke => stroke
): Stroke[] => {
const sharedStrokeProperties: Stroke = {
strokeWidth: translateStrokeWeight(node),
strokeAlignment: translateStrokeAlignment(node.strokeAlign),
strokeStyle: node.dashPattern.length ? 'dashed' : 'solid'
};
return node.strokes.map((paint, index) =>
translateStroke(paint, sharedStrokeProperties, strokeCaps, index === 0)
);
};
export const translateStroke = (
paint: Paint,
sharedStrokeProperties: Stroke,
strokeCaps: (stroke: Stroke) => Stroke,
firstStroke: boolean
): Stroke => {
const fill = translateFill(paint);
let stroke: Stroke = {
strokeColor: fill?.fillColor,
strokeOpacity: fill?.fillOpacity,
strokeImage: fill?.fillImage,
...sharedStrokeProperties
};
if (firstStroke) {
stroke = strokeCaps(stroke);
}
return stroke;
};
export const translateStrokeCap = (vertex: VectorVertex): StrokeCaps | undefined => {
switch (vertex.strokeCap as StrokeCap | ConnectorStrokeCap) {
case 'ROUND':
return 'round';
case 'ARROW_EQUILATERAL':
case 'TRIANGLE_FILLED':
return 'triangle-arrow';
case 'SQUARE':
return 'square';
case 'CIRCLE_FILLED':
return 'circle-marker';
case 'DIAMOND_FILLED':
return 'diamond-marker';
case 'ARROW_LINES':
return 'line-arrow';
case 'NONE':
default:
return;
}
};
const translateStrokeWeight = (
node: MinimalStrokesMixin | (MinimalStrokesMixin & IndividualStrokesMixin)
): number => {
if (node.strokeWeight !== figma.mixed) {
return node.strokeWeight;
}
if (!isIndividualStrokes(node)) {
return 1;
}
return Math.max(
node.strokeTopWeight,
node.strokeRightWeight,
node.strokeBottomWeight,
node.strokeLeftWeight
);
};
const isIndividualStrokes = (
node: MinimalStrokesMixin | IndividualStrokesMixin
): node is IndividualStrokesMixin => {
return 'strokeTopWeight' in node;
};
const translateStrokeAlignment = (
strokeAlign: 'CENTER' | 'INSIDE' | 'OUTSIDE'
): StrokeAlignment => {
switch (strokeAlign) {
case 'CENTER':
return 'center';
case 'INSIDE':
return 'inner';
case 'OUTSIDE':
return 'outer';
}
};