From figma-pack
Provides typed TypeScript clients and patterns for Figma REST/Plugin APIs with error handling, for client wrappers, design tokens, node traversal.
npx claudepluginhub jeremylongshore/claude-code-plugins-plus-skills --plugin figma-packThis skill is limited to using the following tools:
Production patterns for the Figma REST API (external tools) and Plugin API (in-editor plugins). Figma has no official Node.js SDK -- you call `https://api.figma.com` directly with `fetch`. These patterns give you type safety, error handling, and reusable abstractions.
Provides production reference architecture for Figma REST API integrations: design token extraction, asset exports, webhook handlers, with TypeScript project structure and CLI scripts.
Mandatory prerequisite skill for `use_figma` tool calls to execute JS in Figma files. Enables node create/edit/delete, variables/tokens setup, component building, auto-layout changes, property binding, and programmatic file inspection.
Loads mandatory prerequisite context before every use_figma tool call for Figma writes or JS-executed reads like node edits, variable setup, component building, or file inspection.
Share bugs, ideas, or general feedback.
Production patterns for the Figma REST API (external tools) and Plugin API (in-editor plugins). Figma has no official Node.js SDK -- you call https://api.figma.com directly with fetch. These patterns give you type safety, error handling, and reusable abstractions.
FIGMA_PAT environment variable set// src/figma-client.ts
export class FigmaClient {
private baseUrl = 'https://api.figma.com';
constructor(private token: string) {
if (!token) throw new Error('Figma token is required');
}
private async request<T>(path: string, init?: RequestInit): Promise<T> {
const res = await fetch(`${this.baseUrl}${path}`, {
...init,
headers: {
'X-Figma-Token': this.token,
'Content-Type': 'application/json',
...init?.headers,
},
});
if (res.status === 429) {
const retryAfter = parseInt(res.headers.get('Retry-After') || '60');
throw new FigmaRateLimitError(retryAfter);
}
if (res.status === 403) throw new FigmaAuthError('Invalid or expired token');
if (res.status === 404) throw new FigmaNotFoundError(path);
if (!res.ok) throw new FigmaApiError(res.status, await res.text());
return res.json();
}
async getFile(fileKey: string) {
return this.request<FigmaFileResponse>(`/v1/files/${fileKey}`);
}
async getFileNodes(fileKey: string, nodeIds: string[]) {
const ids = encodeURIComponent(nodeIds.join(','));
return this.request<FigmaNodesResponse>(`/v1/files/${fileKey}/nodes?ids=${ids}`);
}
async getImages(fileKey: string, nodeIds: string[], opts?: ImageOptions) {
const params = new URLSearchParams({
ids: nodeIds.join(','),
format: opts?.format ?? 'png',
scale: String(opts?.scale ?? 2),
});
return this.request<FigmaImagesResponse>(`/v1/images/${fileKey}?${params}`);
}
async getComments(fileKey: string) {
return this.request<FigmaCommentsResponse>(`/v1/files/${fileKey}/comments`);
}
async postComment(fileKey: string, message: string, nodeId?: string) {
return this.request(`/v1/files/${fileKey}/comments`, {
method: 'POST',
body: JSON.stringify({
message,
...(nodeId && { client_meta: { node_id: nodeId } }),
}),
});
}
async getLocalVariables(fileKey: string) {
return this.request<FigmaVariablesResponse>(
`/v1/files/${fileKey}/variables/local`
);
}
}
// src/figma-errors.ts
export class FigmaApiError extends Error {
constructor(public status: number, public body: string) {
super(`Figma API error ${status}: ${body}`);
this.name = 'FigmaApiError';
}
}
export class FigmaRateLimitError extends FigmaApiError {
constructor(public retryAfterSeconds: number) {
super(429, `Rate limited. Retry after ${retryAfterSeconds}s`);
this.name = 'FigmaRateLimitError';
}
}
export class FigmaAuthError extends FigmaApiError {
constructor(message: string) {
super(403, message);
this.name = 'FigmaAuthError';
}
}
export class FigmaNotFoundError extends FigmaApiError {
constructor(path: string) {
super(404, `Resource not found: ${path}`);
this.name = 'FigmaNotFoundError';
}
}
// src/figma-types.ts
export interface FigmaNode {
id: string;
name: string;
type: string;
children?: FigmaNode[];
fills?: Paint[];
strokes?: Paint[];
absoluteBoundingBox?: { x: number; y: number; width: number; height: number };
characters?: string; // TEXT nodes
style?: TypeStyle; // TEXT nodes
componentId?: string; // INSTANCE nodes
backgroundColor?: Color; // CANVAS nodes
}
export interface FigmaFileResponse {
name: string;
lastModified: string;
version: string;
thumbnailUrl: string;
document: FigmaNode;
components: Record<string, ComponentMeta>;
styles: Record<string, StyleMeta>;
}
export interface FigmaNodesResponse {
nodes: Record<string, { document: FigmaNode; components: Record<string, ComponentMeta> }>;
}
export interface FigmaImagesResponse {
images: Record<string, string | null>; // nodeId -> URL (null = render failed)
}
export interface ImageOptions {
format?: 'png' | 'svg' | 'jpg' | 'pdf';
scale?: number; // 0.01 to 4. SVG always exports at 1x.
}
interface Paint { type: string; color?: Color; opacity?: number }
interface Color { r: number; g: number; b: number; a?: number }
interface TypeStyle { fontFamily: string; fontSize: number; fontWeight: number }
interface ComponentMeta { key: string; name: string; description: string }
interface StyleMeta { key: string; name: string; style_type: 'FILL' | 'TEXT' | 'EFFECT' | 'GRID' }
// Walk the Figma document tree with a visitor pattern
function walkNodes(node: FigmaNode, visitor: (n: FigmaNode) => void) {
visitor(node);
if (node.children) {
for (const child of node.children) {
walkNodes(child, visitor);
}
}
}
// Example: find all TEXT nodes
function findTextNodes(root: FigmaNode): FigmaNode[] {
const results: FigmaNode[] = [];
walkNodes(root, (n) => {
if (n.type === 'TEXT') results.push(n);
});
return results;
}
// Example: find all COMPONENT nodes
function findComponents(root: FigmaNode): FigmaNode[] {
const results: FigmaNode[] = [];
walkNodes(root, (n) => {
if (n.type === 'COMPONENT') results.push(n);
});
return results;
}
// Singleton instance with automatic retry on transient errors
let client: FigmaClient | null = null;
export function getFigmaClient(): FigmaClient {
if (!client) {
client = new FigmaClient(process.env.FIGMA_PAT!);
}
return client;
}
export async function withRetry<T>(
fn: () => Promise<T>,
maxRetries = 3
): Promise<T> {
for (let attempt = 0; attempt <= maxRetries; attempt++) {
try {
return await fn();
} catch (err) {
if (err instanceof FigmaRateLimitError) {
await new Promise(r => setTimeout(r, err.retryAfterSeconds * 1000));
continue;
}
if (attempt === maxRetries) throw err;
await new Promise(r => setTimeout(r, 1000 * Math.pow(2, attempt)));
}
}
throw new Error('Unreachable');
}
| Pattern | Use Case | Benefit |
|---|---|---|
| Typed errors | catch (e) { if (e instanceof FigmaRateLimitError) } | Targeted recovery |
| Node walker | Traversing arbitrarily deep trees | Handles any file structure |
| Retry wrapper | Transient 429/5xx errors | Automatic recovery |
| Singleton | Shared client across modules | Consistent config, one token |
Apply patterns in figma-core-workflow-a for real-world file inspection.