create-2d-composition
关于
This skill enables programmatic creation of 2D graphics through SVG generation, diagram layout algorithms, and image compositing. Use it for generating diagrams, flowcharts, or custom visual assets, especially when you need batch processing with parameter variations or reproducible scientific figures. It's ideal for building custom chart types not available in standard libraries.
快速安装
Claude Code
推荐npx skills add pjt222/agent-almanac -a claude-code/plugin add https://github.com/pjt222/agent-almanacgit clone https://github.com/pjt222/agent-almanac.git ~/.claude/skills/create-2d-composition在 Claude Code 中复制并粘贴此命令以安装该技能
技能文档
造二維構圖
以 SVG 建、圖排、像合、批作程生二維像。
用
- 程生圖、流、信息圖
- 可重科圖或刊像
- 自生徽、標、視資
- 合多像或數視
- 築自定圖型
- 批生含參變之像
- 為網或印建 SVG 模
入
| 入 | 型 | 述 | 例 |
|---|---|---|---|
| 排規 | 設 | 寸、邊、格排 | 畫 800x600px、20px 邊 |
| 視素 | 數/資 | 形、字、像、點 | 矩坐、標、標 |
| 風參 | CSS/屬 | 色、體、筆寬、透 | fill="#3366cc"、stroke-width="2" |
| 數源 | 檔/陣 | 所視或注之值 | CSV 數、JSON 設 |
| 出式 | 串 | SVG、PNG、PDF、合式 | output.svg、300 DPI PNG |
行
一、設 Python 境
裝二維構所需庫:
# Core libraries
pip install svgwrite pillow cairosvg
# Optional: advanced features
pip install drawsvg reportlab pycairo
# For data-driven graphics
pip install matplotlib numpy pandas
得: 庫裝成 敗: 察 Python 版(3.7+)、用虛境避衝
二、建基 SVG 像
以 svgwrite 生 SVG:
import svgwrite
from svgwrite import cm, mm
def create_basic_svg(output_path):
"""Create a simple SVG graphic."""
# Initialize drawing (use mm for precise dimensions)
dwg = svgwrite.Drawing(output_path, size=('180mm', '120mm'), profile='full')
# Add background rectangle
dwg.add(dwg.rect(
insert=(0, 0),
size=('100%', '100%'),
fill='white'
))
# Add shapes
dwg.add(dwg.circle(
center=(90*mm, 60*mm),
r=30*mm,
fill='lightblue',
stroke='navy',
stroke_width=2
))
dwg.add(dwg.rect(
insert=(30*mm, 30*mm),
size=(60*mm, 40*mm),
fill='lightgreen',
stroke='darkgreen',
stroke_width=2,
rx=5, # Rounded corners
ry=5
))
# Add text
dwg.add(dwg.text(
'Example Graphic',
insert=(90*mm, 20*mm),
text_anchor='middle',
font_size='18pt',
font_family='Arial',
fill='black'
))
dwg.save()
print(f"Saved: {output_path}")
得: SVG 檔生含形與字 敗: 察 svgwrite 版、驗出目可寫
三、以排輯築圖解
建算位之結構圖解:
def create_flowchart(steps, output_path):
"""Generate a flowchart from list of steps."""
dwg = svgwrite.Drawing(output_path, size=('800px', '600px'))
# Layout parameters
box_width = 120
box_height = 60
spacing_y = 100
start_x = 340
start_y = 50
for i, step in enumerate(steps):
y_pos = start_y + i * spacing_y
# Draw box
box = dwg.add(dwg.g(id=f'step_{i}'))
box.add(dwg.rect(
insert=(start_x, y_pos),
size=(box_width, box_height),
fill='lightblue',
stroke='navy',
stroke_width=2,
rx=5,
ry=5
))
# Add text (wrapped if needed)
text_lines = wrap_text(step, max_width=16)
text_y = y_pos + box_height/2 - (len(text_lines)-1) * 7
for j, line in enumerate(text_lines):
box.add(dwg.text(
line,
insert=(start_x + box_width/2, text_y + j*14),
text_anchor='middle',
font_size='12pt',
font_family='Arial',
fill='black'
))
# Draw arrow to next step
if i < len(steps) - 1:
arrow_start_y = y_pos + box_height
arrow_end_y = y_pos + spacing_y
dwg.add(dwg.line(
start=(start_x + box_width/2, arrow_start_y),
end=(start_x + box_width/2, arrow_end_y),
stroke='black',
stroke_width=2,
marker_end=dwg.marker(
id='arrow',
viewBox='0 0 10 10',
refX=5,
refY=5,
markerWidth=6,
markerHeight=6,
orient='auto'
)
))
dwg.save()
def wrap_text(text, max_width=20):
"""Simple text wrapping."""
words = text.split()
lines = []
current_line = []
for word in words:
test_line = ' '.join(current_line + [word])
if len(test_line) <= max_width:
current_line.append(word)
else:
if current_line:
lines.append(' '.join(current_line))
current_line = [word]
if current_line:
lines.append(' '.join(current_line))
return lines
得: 流圖含連匣與箭 敗: 調排算、驗箭標定
四、合柵像
以 Pillow 合多像:
from PIL import Image, ImageDraw, ImageFont, ImageFilter
import os
def composite_images(image_paths, output_path, layout='grid'):
"""Composite multiple images into single output."""
# Load images
images = [Image.open(path) for path in image_paths]
if layout == 'grid':
# Calculate grid dimensions
n = len(images)
cols = int(n ** 0.5)
rows = (n + cols - 1) // cols
# Get max dimensions
max_width = max(img.width for img in images)
max_height = max(img.height for img in images)
# Create composite canvas
canvas_width = cols * max_width
canvas_height = rows * max_height
composite = Image.new('RGB', (canvas_width, canvas_height), 'white')
# Paste images
for i, img in enumerate(images):
row = i // cols
col = i % cols
x = col * max_width
y = row * max_height
composite.paste(img, (x, y))
elif layout == 'horizontal':
# Horizontal concatenation
total_width = sum(img.width for img in images)
max_height = max(img.height for img in images)
composite = Image.new('RGB', (total_width, max_height), 'white')
x_offset = 0
for img in images:
composite.paste(img, (x_offset, 0))
x_offset += img.width
elif layout == 'vertical':
# Vertical concatenation
max_width = max(img.width for img in images)
total_height = sum(img.height for img in images)
composite = Image.new('RGB', (max_width, total_height), 'white')
y_offset = 0
for img in images:
composite.paste(img, (0, y_offset))
y_offset += img.height
composite.save(output_path)
print(f"Saved composite: {output_path}")
def add_annotations(image_path, annotations, output_path):
"""Add text annotations to image."""
img = Image.open(image_path)
draw = ImageDraw.Draw(img)
# Load font
try:
font = ImageFont.truetype("Arial.ttf", 24)
except:
font = ImageFont.load_default()
for annotation in annotations:
text = annotation['text']
position = annotation['position']
color = annotation.get('color', 'black')
# Add text shadow for readability
shadow_offset = 2
draw.text(
(position[0] + shadow_offset, position[1] + shadow_offset),
text,
font=font,
fill='white'
)
draw.text(position, text, font=font, fill=color)
img.save(output_path)
得: 合像成含正排 敗: 察諸入像存、驗像模相容
五、生數驅像
依數建視:
import numpy as np
def create_bar_chart_svg(data, labels, output_path):
"""Generate SVG bar chart from data."""
dwg = svgwrite.Drawing(output_path, size=('600px', '400px'))
# Chart area
margin = 50
chart_width = 500
chart_height = 300
bar_spacing = 10
# Calculate bar dimensions
n_bars = len(data)
bar_width = (chart_width - (n_bars - 1) * bar_spacing) / n_bars
# Scale data to fit chart
max_value = max(data)
scale = chart_height / max_value
# Draw axes
dwg.add(dwg.line(
start=(margin, margin),
end=(margin, margin + chart_height),
stroke='black',
stroke_width=2
))
dwg.add(dwg.line(
start=(margin, margin + chart_height),
end=(margin + chart_width, margin + chart_height),
stroke='black',
stroke_width=2
))
# Draw bars
for i, (value, label) in enumerate(zip(data, labels)):
x = margin + i * (bar_width + bar_spacing)
bar_height = value * scale
y = margin + chart_height - bar_height
# Bar
dwg.add(dwg.rect(
insert=(x, y),
size=(bar_width, bar_height),
fill='steelblue',
stroke='navy',
stroke_width=1
))
# Value label
dwg.add(dwg.text(
f'{value:.1f}',
insert=(x + bar_width/2, y - 5),
text_anchor='middle',
font_size='10pt',
fill='black'
))
# X-axis label
dwg.add(dwg.text(
label,
insert=(x + bar_width/2, margin + chart_height + 20),
text_anchor='middle',
font_size='10pt',
fill='black'
))
dwg.save()
得: SVG 柱圖含按比之數 敗: 理邊例(空數、負值)、加驗
六、批生像
自多像建:
def batch_generate_badges(users, template_path, output_dir):
"""Generate badge for each user."""
os.makedirs(output_dir, exist_ok=True)
for user in users:
output_path = os.path.join(output_dir, f"{user['id']}_badge.svg")
dwg = svgwrite.Drawing(output_path, size=('300px', '100px'))
# Background
dwg.add(dwg.rect(
insert=(0, 0),
size=('100%', '100%'),
fill='#3366cc',
rx=10,
ry=10
))
# User name
dwg.add(dwg.text(
user['name'],
insert=(150, 40),
text_anchor='middle',
font_size='20pt',
font_weight='bold',
fill='white'
))
# User role
dwg.add(dwg.text(
user['role'],
insert=(150, 70),
text_anchor='middle',
font_size='14pt',
fill='lightblue'
))
dwg.save()
print(f"Generated badge: {output_path}")
得: 各數項生一像 敗: 察數構、以默處缺欄
七、SVG 轉柵
為諸用出 SVG 至 PNG/PDF:
import cairosvg
def svg_to_png(svg_path, png_path, dpi=300):
"""Convert SVG to PNG with specified DPI."""
# Calculate pixel dimensions from DPI
# Assuming A4 size as example
width_inches = 8.27
height_inches = 11.69
width_px = int(width_inches * dpi)
height_px = int(height_inches * dpi)
cairosvg.svg2png(
url=svg_path,
write_to=png_path,
output_width=width_px,
output_height=height_px
)
print(f"Converted to PNG: {png_path}")
def svg_to_pdf(svg_path, pdf_path):
"""Convert SVG to PDF."""
cairosvg.svg2pdf(url=svg_path, write_to=pdf_path)
print(f"Converted to PDF: {pdf_path}")
得: 柵出於定解 敗: 缺則裝 cairo 系庫、察 SVG 有效
驗
- 像於標應中正渲
- 字可讀位正
- 色合規、寸合用
- SVG 合準、柵出 DPI 正
- 排適數變、批畢無誤
- 出檔理組、碼含誤理
忌
- 單位惑:SVG 單與屏素印 DPI
- 字溢:過界→換行
- 體可用:系異→嵌或網安
- 坐算:格差一誤
- 色式:hex 串、非元
- SVG 有效:XML 構、閉諸標
- 檔路:特符、檔名空格
- 存用:大批需塊、比例保、PNG 支 alpha 而 JPEG 否
參
- render-publication-graphic:刊特出需
- create-3d-scene:三維之類程法
- generate-quarto-report:像融於報
GitHub 仓库
相关推荐技能
content-collections
元Content Collections 是一个 TypeScript 优先的构建工具,可将本地 Markdown/MDX 文件转换为类型安全的数据集合。它专为构建博客、文档站和内容密集型 Vite+React 应用而设计,提供基于 Zod 的自动模式验证。该工具涵盖从 Vite 插件配置、MDX 编译到生产环境部署的完整工作流。
polymarket
元这个Claude Skill为开发者提供完整的Polymarket预测市场开发支持,涵盖API调用、交易执行和市场数据分析。关键特性包括实时WebSocket数据流,可监控实时交易、订单和市场动态。开发者可用它构建预测市场应用、实施交易策略并集成实时市场预测功能。
creating-opencode-plugins
元该Skill帮助开发者创建OpenCode插件,用于接入命令、文件、LSP等25+种事件。它提供了插件结构、事件API规范和JavaScript/TypeScript实现模式,适合需要拦截操作、扩展功能或自定义事件处理的场景。开发者可通过它快速构建响应式模块来增强OpenCode AI助手的能力。
sglang
元SGLang是一个专为LLM设计的高性能推理框架,特别适用于需要结构化输出的场景。它通过RadixAttention前缀缓存技术,在处理JSON、正则表达式、工具调用等具有重复前缀的复杂工作流时,能实现极速生成。如果你正在构建智能体或多轮对话系统,并追求远超vLLM的推理性能,SGLang是理想选择。
