2024-05-09 09:59:27 -05:00
|
|
|
import { detectMimeType, rgbToHex } from '@plugin/utils';
|
2024-04-17 08:27:53 -05:00
|
|
|
import { calculateLinearGradient } from '@plugin/utils/calculateLinearGradient';
|
2024-04-12 09:52:36 -05:00
|
|
|
|
2024-04-17 08:27:53 -05:00
|
|
|
import { Fill } from '@ui/lib/types/utils/fill';
|
2024-05-09 09:59:27 -05:00
|
|
|
import { ImageColor } from '@ui/lib/types/utils/imageColor';
|
2024-04-12 06:55:42 -05:00
|
|
|
|
2024-05-09 09:59:27 -05:00
|
|
|
export const translateFill = async (
|
|
|
|
fill: Paint,
|
|
|
|
width: number,
|
|
|
|
height: number
|
|
|
|
): Promise<Fill | undefined> => {
|
2024-04-12 06:55:42 -05:00
|
|
|
switch (fill.type) {
|
|
|
|
case 'SOLID':
|
|
|
|
return translateSolidFill(fill);
|
2024-04-15 03:06:19 -05:00
|
|
|
case 'GRADIENT_LINEAR':
|
|
|
|
return translateGradientLinearFill(fill, width, height);
|
2024-05-09 09:59:27 -05:00
|
|
|
case 'IMAGE':
|
|
|
|
return await translateImageFill(fill);
|
2024-04-12 06:55:42 -05:00
|
|
|
}
|
|
|
|
|
2024-04-19 02:48:19 -05:00
|
|
|
console.error(`Unsupported fill type: ${fill.type}`);
|
2024-04-12 06:55:42 -05:00
|
|
|
};
|
|
|
|
|
2024-05-09 09:59:27 -05:00
|
|
|
export const translateFills = async (
|
2024-04-15 03:06:19 -05:00
|
|
|
fills: readonly Paint[] | typeof figma.mixed,
|
|
|
|
width: number,
|
|
|
|
height: number
|
2024-05-09 09:59:27 -05:00
|
|
|
): Promise<Fill[]> => {
|
2024-04-17 08:27:53 -05:00
|
|
|
const figmaFills = fills === figma.mixed ? [] : fills;
|
|
|
|
const penpotFills: Fill[] = [];
|
2024-04-12 06:55:42 -05:00
|
|
|
|
2024-04-17 08:27:53 -05:00
|
|
|
for (const fill of figmaFills) {
|
2024-05-09 09:59:27 -05:00
|
|
|
const penpotFill = await translateFill(fill, width, height);
|
2024-04-12 06:55:42 -05:00
|
|
|
if (penpotFill) {
|
2024-05-09 09:59:27 -05:00
|
|
|
// fills are applied in reverse order in Figma, that's why we unshift
|
2024-04-12 06:55:42 -05:00
|
|
|
penpotFills.unshift(penpotFill);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return penpotFills;
|
|
|
|
};
|
2024-04-17 08:27:53 -05:00
|
|
|
|
2024-04-22 03:52:25 -05:00
|
|
|
export const translatePageFill = (fill: Paint): string | undefined => {
|
|
|
|
switch (fill.type) {
|
|
|
|
case 'SOLID':
|
|
|
|
return rgbToHex(fill.color);
|
|
|
|
}
|
|
|
|
|
|
|
|
console.error(`Unsupported page fill type: ${fill.type}`);
|
|
|
|
};
|
|
|
|
|
2024-05-09 09:59:27 -05:00
|
|
|
const translateImage = async (imageHash: string | null): Promise<ImageColor | undefined> => {
|
|
|
|
if (!imageHash) return;
|
|
|
|
|
|
|
|
const image = figma.getImageByHash(imageHash);
|
|
|
|
if (!image) return;
|
|
|
|
|
|
|
|
const bytes = await image.getBytesAsync();
|
|
|
|
const size = await image.getSizeAsync();
|
|
|
|
const b64 = figma.base64Encode(bytes);
|
|
|
|
const mimeType = detectMimeType(b64);
|
|
|
|
const dataUri = `data:${mimeType};base64,${b64}`;
|
|
|
|
|
|
|
|
return {
|
|
|
|
width: size.width,
|
|
|
|
height: size.height,
|
|
|
|
mtype: mimeType,
|
|
|
|
keepAspectRatio: true,
|
|
|
|
dataUri: dataUri
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
const translateImageFill = async (fill: ImagePaint): Promise<Fill | undefined> => {
|
|
|
|
const fillImage = await translateImage(fill.imageHash);
|
|
|
|
if (!fillImage) return;
|
|
|
|
|
|
|
|
return {
|
|
|
|
fillOpacity: !fill.visible ? 0 : fill.opacity,
|
|
|
|
fillImage: fillImage
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-04-17 08:27:53 -05:00
|
|
|
const translateSolidFill = (fill: SolidPaint): Fill => {
|
|
|
|
return {
|
|
|
|
fillColor: rgbToHex(fill.color),
|
|
|
|
fillOpacity: !fill.visible ? 0 : fill.opacity
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
const translateGradientLinearFill = (fill: GradientPaint, width: number, height: number): Fill => {
|
|
|
|
const points = calculateLinearGradient(width, height, fill.gradientTransform);
|
|
|
|
|
|
|
|
return {
|
|
|
|
fillColorGradient: {
|
|
|
|
type: 'linear',
|
|
|
|
startX: points.start[0] / width,
|
|
|
|
startY: points.start[1] / height,
|
|
|
|
endX: points.end[0] / width,
|
|
|
|
endY: points.end[1] / height,
|
|
|
|
width: 1,
|
|
|
|
stops: [
|
|
|
|
{
|
|
|
|
color: rgbToHex(fill.gradientStops[0].color),
|
|
|
|
offset: fill.gradientStops[0].position,
|
|
|
|
opacity: fill.gradientStops[0].color.a * (fill.opacity ?? 1)
|
|
|
|
},
|
|
|
|
{
|
|
|
|
color: rgbToHex(fill.gradientStops[1].color),
|
|
|
|
offset: fill.gradientStops[1].position,
|
|
|
|
opacity: fill.gradientStops[1].color.a * (fill.opacity ?? 1)
|
|
|
|
}
|
|
|
|
]
|
|
|
|
},
|
2024-05-09 09:59:27 -05:00
|
|
|
fillOpacity: !fill.visible ? 0 : fill.opacity
|
2024-04-17 08:27:53 -05:00
|
|
|
};
|
|
|
|
};
|