From figma-pack
Provides typed TypeScript clients and patterns for Figma REST/Plugin APIs with error handling, for client wrappers, design tokens, node traversal.
How this skill is triggered — by the user, by Claude, or both
Slash command
/figma-pack:figma-sdk-patternsThis skill is limited to the following tools:
The summary Claude sees in its skill listing — used to decide when to auto-load this skill
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.
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.
npx claudepluginhub jeremylongshore/claude-code-plugins-plus-skills --plugin figma-packProvides production reference architecture for Figma REST API integrations: design token extraction, asset exports, webhook handlers, with TypeScript project structure and CLI scripts.
Syncs design tokens between code and Figma using Variables API or Tokens Studio plugin. Use when establishing Figma-to-code workflows, exporting Figma tokens, or setting up design-development handoff.
Creates, edits, and optimizes skills for Claude Code, including drafting, evaluating with test prompts, iterating on performance, and improving skill descriptions for better triggering accuracy.