2024-05-13 17:04:13 +02:00
|
|
|
import { translateFill } from '@plugin/translators/fills';
|
2024-04-16 16:08:39 +02:00
|
|
|
|
2024-04-18 09:42:45 +02:00
|
|
|
import { Stroke, StrokeAlignment, StrokeCaps } from '@ui/lib/types/utils/stroke';
|
2024-04-16 16:08:39 +02:00
|
|
|
|
2024-05-09 16:59:27 +02:00
|
|
|
export const translateStrokes = async (
|
2024-04-18 09:42:45 +02:00
|
|
|
nodeStrokes: MinimalStrokesMixin,
|
2024-04-17 10:53:38 +02:00
|
|
|
hasFillGeometry?: boolean,
|
2024-04-18 09:42:45 +02:00
|
|
|
vectorNetwork?: VectorNetwork,
|
|
|
|
individualStrokes?: IndividualStrokesMixin
|
2024-05-09 16:59:27 +02:00
|
|
|
): Promise<Stroke[]> => {
|
|
|
|
return await Promise.all(
|
|
|
|
nodeStrokes.strokes.map(async (paint, index) => {
|
2024-05-13 16:44:00 +02:00
|
|
|
const fill = await translateFill(paint);
|
2024-05-09 16:59:27 +02:00
|
|
|
const stroke: Stroke = {
|
|
|
|
strokeColor: fill?.fillColor,
|
|
|
|
strokeOpacity: fill?.fillOpacity,
|
|
|
|
strokeWidth: translateStrokeWeight(nodeStrokes.strokeWeight, individualStrokes),
|
|
|
|
strokeAlignment: translateStrokeAlignment(nodeStrokes.strokeAlign),
|
|
|
|
strokeStyle: nodeStrokes.dashPattern.length ? 'dashed' : 'solid',
|
|
|
|
strokeImage: fill?.fillImage
|
|
|
|
};
|
2024-04-17 10:53:38 +02:00
|
|
|
|
2024-05-09 16:59:27 +02:00
|
|
|
if (!hasFillGeometry && index === 0 && vectorNetwork && vectorNetwork.vertices.length > 0) {
|
|
|
|
stroke.strokeCapStart = translateStrokeCap(vectorNetwork.vertices[0]);
|
|
|
|
stroke.strokeCapEnd = translateStrokeCap(
|
|
|
|
vectorNetwork.vertices[vectorNetwork.vertices.length - 1]
|
|
|
|
);
|
|
|
|
}
|
2024-04-17 10:53:38 +02:00
|
|
|
|
2024-05-09 16:59:27 +02:00
|
|
|
return stroke;
|
|
|
|
})
|
|
|
|
);
|
2024-04-16 16:08:39 +02:00
|
|
|
};
|
2024-04-17 10:53:38 +02:00
|
|
|
|
2024-04-18 09:42:45 +02:00
|
|
|
const translateStrokeWeight = (
|
|
|
|
strokeWeight: number | typeof figma.mixed,
|
|
|
|
individualStrokes?: IndividualStrokesMixin
|
|
|
|
): number => {
|
|
|
|
if (strokeWeight !== figma.mixed) {
|
|
|
|
return strokeWeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!individualStrokes) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Math.max(
|
|
|
|
individualStrokes.strokeTopWeight,
|
|
|
|
individualStrokes.strokeRightWeight,
|
|
|
|
individualStrokes.strokeBottomWeight,
|
|
|
|
individualStrokes.strokeLeftWeight
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
const translateStrokeAlignment = (
|
|
|
|
strokeAlign: 'CENTER' | 'INSIDE' | 'OUTSIDE'
|
|
|
|
): StrokeAlignment => {
|
|
|
|
switch (strokeAlign) {
|
|
|
|
case 'CENTER':
|
|
|
|
return 'center';
|
|
|
|
case 'INSIDE':
|
|
|
|
return 'inner';
|
|
|
|
case 'OUTSIDE':
|
|
|
|
return 'outer';
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-04-17 10:53:38 +02:00
|
|
|
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';
|
2024-04-24 16:29:24 +02:00
|
|
|
case 'NONE':
|
|
|
|
default:
|
|
|
|
return;
|
2024-04-17 10:53:38 +02:00
|
|
|
}
|
|
|
|
};
|