0
Fork 0
mirror of https://github.com/penpot/penpot-exporter-figma-plugin.git synced 2025-01-03 05:10:13 -05:00

Masked Groups (#114)

* masked groups

* changeset

* refactor

* refactor

* refactor

* fixes

* fixes

* wip

* try to fix

* refactor

* refactor for better undestanding of code

* refactor

* fix name

---------

Co-authored-by: Jordi Sala Morales <jordism91@gmail.com>
This commit is contained in:
Alex Sánchez 2024-05-27 11:50:59 +02:00 committed by GitHub
parent 36afc6da55
commit 761093564b
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 58 additions and 5 deletions

View file

@ -0,0 +1,5 @@
---
"penpot-exporter": minor
---
Added support for masked groups

View file

@ -1,16 +1,22 @@
import { transformSceneNode } from '@plugin/transformers'; import { translateChildren, translateMaskChildren } from '@plugin/translators';
import { PenpotNode } from '@ui/lib/types/penpotNode';
import { Children } from '@ui/lib/types/utils/children'; import { Children } from '@ui/lib/types/utils/children';
const nodeActsAsMask = (node: SceneNode): boolean => {
return 'isMask' in node && node.isMask;
};
export const transformChildren = async ( export const transformChildren = async (
node: ChildrenMixin, node: ChildrenMixin,
baseX: number = 0, baseX: number = 0,
baseY: number = 0 baseY: number = 0
): Promise<Children> => { ): Promise<Children> => {
const maskIndex = node.children.findIndex(nodeActsAsMask);
const containsMask = maskIndex !== -1;
return { return {
children: ( children: containsMask
await Promise.all(node.children.map(child => transformSceneNode(child, baseX, baseY))) ? await translateMaskChildren(node.children, maskIndex, baseX, baseY)
).filter((child): child is PenpotNode => !!child) : await translateChildren(node.children, baseX, baseY)
}; };
}; };

View file

@ -1,5 +1,6 @@
export * from './translateBlendMode'; export * from './translateBlendMode';
export * from './translateBlurEffects'; export * from './translateBlurEffects';
export * from './translateBoolType'; export * from './translateBoolType';
export * from './translateChildren';
export * from './translateShadowEffects'; export * from './translateShadowEffects';
export * from './translateStrokes'; export * from './translateStrokes';

View file

@ -0,0 +1,41 @@
import { transformGroupNodeLike, transformSceneNode } from '@plugin/transformers';
import { PenpotNode } from '@ui/lib/types/penpotNode';
/**
* Translates the children of a node that acts as a mask.
* We need to split the children into two groups: the ones that are masked and the ones that are not.
*
* The masked children will be grouped together in a mask group.
* The unmasked children will be returned as they are.
*
* @maskIndex The index of the mask node in the children array
*/
export const translateMaskChildren = async (
children: readonly SceneNode[],
maskIndex: number,
baseX: number,
baseY: number
): Promise<PenpotNode[]> => {
const maskChild = children[maskIndex];
const unmaskedChildren = await translateChildren(children.slice(0, maskIndex), baseX, baseY);
const maskedChildren = await translateChildren(children.slice(maskIndex), baseX, baseY);
const maskGroup = {
...transformGroupNodeLike(maskChild, baseX, baseY),
children: maskedChildren,
maskedGroup: true
};
return [...unmaskedChildren, maskGroup];
};
export const translateChildren = async (
children: readonly SceneNode[],
baseX: number,
baseY: number
): Promise<PenpotNode[]> => {
return (await Promise.all(children.map(child => transformSceneNode(child, baseX, baseY)))).filter(
(child): child is PenpotNode => !!child
);
};