0
Fork 0
mirror of https://github.com/penpot/penpot-exporter-figma-plugin.git synced 2025-01-04 22:01:08 -05:00
penpot-exporter-figma-plugin/ui-src/context/useFigma.ts
Alex Sánchez a079f168df
Analytics (#228)
* metricts-sentry

* fixes

* refactor

* mixpanel integration

* improvements

* improvements

* fixes

* changeset

* fixes

* fixes

* Update .changeset/few-scissors-sleep.md

Co-authored-by: Jordi Sala Morales <jordism91@gmail.com>

* Update manifest.json

Co-authored-by: Jordi Sala Morales <jordism91@gmail.com>

* Update vite.config.ts

Co-authored-by: Jordi Sala Morales <jordism91@gmail.com>

* fixes

* fixes

* fixes

* lint

---------

Co-authored-by: Jordi Sala Morales <jordism91@gmail.com>
2024-10-28 11:22:50 +01:00

182 lines
4.3 KiB
TypeScript

import { useEffect, useState } from 'react';
import { FormValues } from '@ui/components/ExportForm';
import { identify, track } from '@ui/metrics/mixpanel';
import { parse } from '@ui/parser';
import { MessageData, sendMessage } from '.';
export type UseFigmaHook = {
missingFonts: string[] | undefined;
needsReload: boolean;
loading: boolean;
exporting: boolean;
error: boolean;
step: Steps | undefined;
currentItem: string | undefined;
totalItems: number;
processedItems: number;
reload: () => void;
cancel: () => void;
exportPenpot: (data: FormValues) => void;
};
export type Steps =
| 'processing'
| 'images'
| 'optimization'
| 'building'
| 'components'
| 'exporting'
| 'fills'
| 'format'
| 'libraries'
| 'typographies'
| 'typoFormat'
| 'typoLibraries';
export const useFigma = (): UseFigmaHook => {
const [missingFonts, setMissingFonts] = useState<string[]>();
const [needsReload, setNeedsReload] = useState(false);
const [loading, setLoading] = useState(true);
const [exporting, setExporting] = useState(false);
const [error, setError] = useState(false);
const [step, setStep] = useState<Steps>();
const [currentItem, setCurrentItem] = useState<string | undefined>();
const [totalItems, setTotalItems] = useState<number>(0);
const [processedItems, setProcessedItems] = useState<number>(0);
const postMessage = (type: string, data?: unknown) => {
parent.postMessage({ pluginMessage: { type, data } }, '*');
};
const onMessage = async (event: MessageEvent<MessageData>) => {
if (!event.data.pluginMessage) return;
const { pluginMessage } = event.data;
switch (pluginMessage.type) {
case 'USER_DATA': {
identify({ userId: pluginMessage.data.userId });
track('Plugin Loaded');
break;
}
case 'PENPOT_DOCUMENT': {
const file = await parse(pluginMessage.data);
sendMessage({
type: 'PROGRESS_STEP',
data: 'exporting'
});
const blob = await file.export().catch(error => {
sendMessage({
type: 'ERROR',
data: error.message
});
});
if (blob) {
download(blob, `${pluginMessage.data.name}.zip`);
// get size of the file in Mb rounded to 2 decimal places
const size = Math.round((blob.size / 1024 / 1024) * 100) / 100;
track('File Exported', { 'Exported File Size': size + ' Mb' });
}
setExporting(false);
setStep(undefined);
break;
}
case 'CUSTOM_FONTS': {
setMissingFonts(pluginMessage.data);
setLoading(false);
setNeedsReload(false);
break;
}
case 'CHANGES_DETECTED': {
setNeedsReload(true);
break;
}
case 'PROGRESS_STEP': {
setStep(pluginMessage.data);
setProcessedItems(0);
break;
}
case 'PROGRESS_CURRENT_ITEM': {
setCurrentItem(pluginMessage.data);
break;
}
case 'PROGRESS_TOTAL_ITEMS': {
setTotalItems(pluginMessage.data);
break;
}
case 'PROGRESS_PROCESSED_ITEMS': {
setProcessedItems(pluginMessage.data);
break;
}
case 'ERROR': {
setError(true);
setLoading(false);
setExporting(false);
track('Error', { 'Error Message': pluginMessage.data });
throw new Error(pluginMessage.data);
}
}
};
const download = (blob: Blob, name: string) => {
const url = window.URL.createObjectURL(blob);
const a = document.createElement('a');
a.href = url;
a.download = name;
a.click();
};
const reload = () => {
setLoading(true);
setError(false);
postMessage('reload');
};
const cancel = () => {
postMessage('cancel');
};
const exportPenpot = (data: FormValues) => {
setExporting(true);
setStep('processing');
setProcessedItems(0);
postMessage('export', data);
};
useEffect(() => {
window.addEventListener('message', onMessage);
postMessage('ready');
return () => {
window.removeEventListener('message', onMessage);
};
}, []);
return {
missingFonts,
needsReload,
loading,
exporting,
error,
step,
currentItem,
totalItems,
processedItems,
reload,
cancel,
exportPenpot
};
};