import { remoteComponentLibrary } from '@plugin/RemoteComponentLibrary'; import { transformAutoLayout, transformBlend, transformChildren, transformConstraints, transformCornerRadius, transformDimensionAndPosition, transformEffects, transformFigmaIds, transformFills, transformLayoutSizing, transformProportion, transformSceneNode, transformStrokes } from '@plugin/transformers/partials'; import { ComponentInstance } from '@ui/types'; export const transformInstanceNode = async ( node: InstanceNode, baseX: number, baseY: number ): Promise => { const mainComponent = await node.getMainComponentAsync(); if (mainComponent === null || isUnprocessableComponent(mainComponent)) { return; } if (isExternalComponent(mainComponent)) { await registerExternalComponents(mainComponent); } let reverseChildrenOrder = false; if (node.layoutMode !== 'NONE') { reverseChildrenOrder = true; } return { type: 'instance', name: node.name, mainComponentFigmaId: mainComponent.id, isComponentRoot: isComponentRoot(node), showContent: !node.clipsContent, ...transformFigmaIds(node), ...transformFills(node), ...transformEffects(node), ...transformStrokes(node), ...transformSceneNode(node), ...transformBlend(node), ...transformProportion(node), ...transformLayoutSizing(node), ...transformCornerRadius(node), ...transformDimensionAndPosition(node, baseX, baseY), ...transformConstraints(node), ...transformAutoLayout(node), ...(await transformChildren(node, baseX + node.x, baseY + node.y, reverseChildrenOrder)) }; }; const registerExternalComponents = async (mainComponent: ComponentNode): Promise => { let component: ComponentSetNode | ComponentNode = mainComponent; if (component.parent?.type === 'COMPONENT_SET') { component = component.parent; } if (remoteComponentLibrary.get(component.id) !== undefined) { return; } remoteComponentLibrary.register(component.id, component); }; const isExternalComponent = (mainComponent: ComponentNode): boolean => { return ( mainComponent.remote || (mainComponent.parent?.type === 'COMPONENT_SET' && mainComponent.parent.remote) ); }; /** * We do not want to process component instances in the following scenarios: * * 1. If the component does not have a parent. (it's been removed) * 2. Main component can be in a ComponentSet (the same logic applies to the parent). */ const isUnprocessableComponent = (mainComponent: ComponentNode): boolean => { return ( (mainComponent.parent === null && !mainComponent.remote) || (mainComponent.parent?.type === 'COMPONENT_SET' && mainComponent.parent.parent === null && !mainComponent.parent.remote) ); }; const isComponentRoot = (node: InstanceNode): boolean => { let parent = node.parent; while (parent !== null) { if (parent.type === 'COMPONENT' || parent.type === 'INSTANCE') { return false; } parent = parent.parent; } return true; };