skills/chart-builder/SKILL.md
Use this skill when creating data visualizations, selecting the right chart type, or generating chart code. Trigger phrases: 'build a chart', 'visualize this data', 'create a graph', 'plot these numbers', 'which chart should I use for'. Not for building interactive dashboards, designing UI components, or creating infographics with design tools like Figma.
npx skillsauth add nickcrew/claude-cortex chart-builderInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
This skill provides a systematic approach to data visualization—from choosing the right chart type for your data and message, to implementing it in Python (matplotlib/seaborn) or JavaScript (Chart.js). It covers chart anatomy, color principles, accessibility requirements, and common pitfalls, so every chart communicates its insight clearly and honestly.
| Chart Type | Best For | Avoid When | |------------|----------|------------| | Bar (vertical) | Comparing categories, rankings | Too many categories (>12) | | Bar (horizontal) | Long category labels, rankings | Showing trends over time | | Line | Trends over time, continuous data | Categorical/unordered x-axis | | Scatter | Correlation between two variables | Fewer than ~20 data points | | Pie / Donut | Part-of-whole (max 5 slices) | Comparing many segments | | Heatmap | Matrix of values, correlation tables | Audiences unfamiliar with the format | | Box plot | Distribution comparison across groups | Presenting to non-technical audience | | Histogram | Distribution of a single variable | Showing trends or comparisons | | Area | Cumulative totals over time | Multiple overlapping series | | Stacked bar | Part-of-whole across categories | Comparing absolute values |
Answer these questions:
A complete chart has:
Bar chart:
import matplotlib.pyplot as plt
import seaborn as sns
import pandas as pd
df = pd.DataFrame({
'Region': ['West', 'East', 'North', 'South'],
'Revenue': [4200, 3100, 1800, 2700]
})
fig, ax = plt.subplots(figsize=(8, 5))
bars = ax.bar(df['Region'], df['Revenue'], color=['#2563EB', '#64748B', '#64748B', '#64748B'])
ax.set_title("West Region Leads Q3 Revenue", fontsize=16, fontweight='bold', pad=12)
ax.set_xlabel("Region", fontsize=12)
ax.set_ylabel("Revenue (USD thousands)", fontsize=12)
ax.yaxis.grid(True, color='#E5E7EB', linewidth=0.8)
ax.set_axisbelow(True)
ax.spines[['top', 'right']].set_visible(False)
# Value labels on bars
for bar in bars:
ax.text(bar.get_x() + bar.get_width()/2, bar.get_height() + 50,
f"${bar.get_height():,.0f}K", ha='center', va='bottom', fontsize=11)
plt.tight_layout()
plt.savefig("revenue_by_region.png", dpi=150, bbox_inches='tight')
plt.show()
Line chart (trend):
import matplotlib.pyplot as plt
import pandas as pd
df = pd.DataFrame({
'Month': ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun'],
'Revenue': [180, 210, 195, 240, 285, 310]
})
fig, ax = plt.subplots(figsize=(9, 5))
ax.plot(df['Month'], df['Revenue'], color='#2563EB', linewidth=2.5, marker='o', markersize=7)
ax.fill_between(range(len(df)), df['Revenue'], alpha=0.08, color='#2563EB')
ax.set_title("Revenue Growing Steadily — Up 72% H1", fontsize=16, fontweight='bold')
ax.set_ylabel("Revenue (USD thousands)", fontsize=12)
ax.yaxis.grid(True, color='#E5E7EB', linewidth=0.8)
ax.set_axisbelow(True)
ax.spines[['top', 'right']].set_visible(False)
plt.tight_layout()
plt.savefig("revenue_trend.png", dpi=150, bbox_inches='tight')
Scatter plot (correlation):
import seaborn as sns
import matplotlib.pyplot as plt
fig, ax = plt.subplots(figsize=(8, 6))
sns.regplot(data=df, x='marketing_spend', y='revenue', ax=ax,
scatter_kws={'alpha': 0.6, 'color': '#2563EB'},
line_kws={'color': '#DC2626', 'linewidth': 1.5})
ax.set_title("Marketing Spend Strongly Correlates with Revenue (r=0.78)", fontsize=14, fontweight='bold')
ax.set_xlabel("Marketing Spend (USD)", fontsize=12)
ax.set_ylabel("Revenue (USD)", fontsize=12)
ax.spines[['top', 'right']].set_visible(False)
plt.tight_layout()
Heatmap (correlation matrix):
import seaborn as sns
import matplotlib.pyplot as plt
corr = df[['revenue', 'units_sold', 'discount_pct', 'marketing_spend']].corr()
fig, ax = plt.subplots(figsize=(7, 6))
sns.heatmap(corr, annot=True, fmt=".2f", cmap='RdBu_r', center=0,
vmin=-1, vmax=1, square=True, linewidths=0.5, ax=ax)
ax.set_title("Correlation Matrix — Sales Variables", fontsize=14, fontweight='bold')
plt.tight_layout()
Bar chart:
<canvas id="revenueChart" width="600" height="400"></canvas>
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
<script>
const ctx = document.getElementById('revenueChart').getContext('2d');
new Chart(ctx, {
type: 'bar',
data: {
labels: ['West', 'East', 'North', 'South'],
datasets: [{
label: 'Revenue (USD K)',
data: [4200, 3100, 1800, 2700],
backgroundColor: ['#2563EB', '#64748B', '#64748B', '#64748B'],
borderRadius: 4,
}]
},
options: {
responsive: true,
plugins: {
title: { display: true, text: 'West Region Leads Q3 Revenue', font: { size: 16 } },
legend: { display: false }
},
scales: {
y: { grid: { color: '#E5E7EB' }, ticks: { callback: v => `$${v}K` } },
x: { grid: { display: false } }
}
}
});
</script>
Line chart (multi-series):
new Chart(ctx, {
type: 'line',
data: {
labels: ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun'],
datasets: [
{
label: 'Revenue',
data: [180, 210, 195, 240, 285, 310],
borderColor: '#2563EB',
backgroundColor: 'rgba(37,99,235,0.08)',
fill: true,
tension: 0.3,
pointRadius: 5
},
{
label: 'Target',
data: [200, 220, 220, 250, 270, 300],
borderColor: '#DC2626',
borderDash: [6, 3],
fill: false,
pointRadius: 0
}
]
},
options: {
plugins: { title: { display: true, text: 'Revenue vs Target — H1 2024' } },
scales: { y: { grid: { color: '#E5E7EB' } } }
}
});
Input: "I have monthly revenue data for 2023 and want to show the growth trend."
Step 1 — Chart choice: Time series data → line chart. Message is trend → line with a subtle fill works well.
Step 2 — Python code:
import matplotlib.pyplot as plt
import pandas as pd
data = {
'month': pd.date_range('2023-01', periods=12, freq='MS'),
'revenue': [210, 195, 230, 255, 280, 310, 295, 340, 365, 390, 410, 445]
}
df = pd.DataFrame(data)
mom_growth = (df['revenue'].iloc[-1] / df['revenue'].iloc[0] - 1) * 100
fig, ax = plt.subplots(figsize=(10, 5))
ax.plot(df['month'], df['revenue'], color='#2563EB', linewidth=2.5, marker='o', markersize=6)
ax.fill_between(df['month'], df['revenue'], alpha=0.07, color='#2563EB')
ax.set_title(f"Revenue Up {mom_growth:.0f}% in 2023", fontsize=16, fontweight='bold')
ax.set_ylabel("Revenue (USD thousands)", fontsize=12)
ax.xaxis.set_major_formatter(plt.matplotlib.dates.DateFormatter('%b'))
ax.yaxis.grid(True, color='#E5E7EB', linewidth=0.8, linestyle='--')
ax.set_axisbelow(True)
ax.spines[['top', 'right']].set_visible(False)
plt.tight_layout()
plt.savefig("2023_revenue_trend.png", dpi=150, bbox_inches='tight')
Output: A clean line chart titled "Revenue Up 112% in 2023" with subtle area fill, gridlines only on y-axis, no top/right spines.
Input: "Compare Q3 revenue across 5 product categories, highlighting which ones are above target."
Step 1 — Chart choice: Comparing categories → horizontal bar chart (better for 5 labels). Highlight above-target bars differently.
Step 2 — Python code:
import matplotlib.pyplot as plt
import pandas as pd
import numpy as np
df = pd.DataFrame({
'Category': ['Analytics', 'Storage', 'Compute', 'Networking', 'Security'],
'Revenue': [4200, 2800, 3600, 1500, 3100],
'Target': [3800, 3000, 3500, 2000, 2900]
})
df['above_target'] = df['Revenue'] >= df['Target']
colors = ['#2563EB' if above else '#94A3B8' for above in df['above_target']]
fig, ax = plt.subplots(figsize=(9, 5))
bars = ax.barh(df['Category'], df['Revenue'], color=colors, height=0.6)
ax.vlines(df['Target'], -0.4, len(df)-0.6, colors='#DC2626', linewidth=1.5,
linestyle='--', label='Target')
ax.set_title("3 of 5 Categories Beat Q3 Revenue Target", fontsize=15, fontweight='bold')
ax.set_xlabel("Revenue (USD thousands)", fontsize=12)
ax.xaxis.grid(True, color='#E5E7EB', linewidth=0.8)
ax.set_axisbelow(True)
ax.spines[['top', 'right']].set_visible(False)
# Labels
for bar, val in zip(bars, df['Revenue']):
ax.text(val + 30, bar.get_y() + bar.get_height()/2,
f"${val:,}K", va='center', fontsize=10)
from matplotlib.patches import Patch
legend_elements = [Patch(color='#2563EB', label='Above Target'),
Patch(color='#94A3B8', label='Below Target')]
ax.legend(handles=legend_elements, loc='lower right')
plt.tight_layout()
plt.savefig("category_vs_target.png", dpi=150, bbox_inches='tight')
Output: Horizontal bar chart with blue bars (above target) and grey bars (below target), red dashed target line, value labels on each bar.
sns.set_style('whitegrid') gives clean defaults in seaborn with one lineax.spines[['top', 'right']].set_visible(False) to immediately make matplotlib charts look modernpalette='colorblind'ax.axhline(0, color='black', linewidth=0.8)tension: 0.3 makes line charts look smoother without being misleadingplt.savefig("chart.png", dpi=150)development
Product vision, roadmap development, and go-to-market execution with structured prioritization frameworks. Use when evaluating features, planning product direction, or assessing market fit.
development
Complete operational workflow for implementer agents (Codex, Gemini, etc.) making code changes and writing tests. Drives all work through atomic commits — each loop operates on the smallest complete, reviewable change. Defines the Code Change Loop, Test Writing Loop, Lint Gate, and Issue Filing process with circuit breakers, severity levels, and escalation rules. Requires `cortex git commit` for all commits. Includes bundled provider-aware review scripts that keep same-model shell-outs as the last resort, plus a fresh-context Codex fallback for code review and test audit. Use this skill when starting any implementation task.
development
Use this skill when writing product requirements documents, prioritizing features, creating user stories, defining acceptance criteria, or setting product metrics. Trigger phrases: 'write a PRD for', 'prioritize this feature backlog', 'write user stories for', 'help me define acceptance criteria', 'what metrics should we track for'. Not for writing code, designing UI mockups, or conducting user research interviews.
tools
Automates browser interactions for web testing, form filling, screenshots, and data extraction. Use when the user needs to navigate websites, interact with web pages, fill forms, take screenshots, test web applications, or extract information from web pages.