showcase/STYLING-GUIDE.md
When building demo pages for integration packages, follow these rules to avoid common pitfalls.
Tailwind v4 aggressively purges CSS classes it can't statically analyze. If your component renders Tailwind classes dynamically (e.g., from state, props, or conditional logic), those classes will be missing from the CSS bundle.
// GOOD — inline styles always work
<div style={{ padding: "32px", borderRadius: "16px", border: "1px solid #e5e5e0" }}>
// BAD — Tailwind may purge these classes
<div className="p-8 rounded-2xl border border-gray-200">
Tailwind classes work fine on:
Tailwind classes are UNSAFE on:
useRenderTool, useHumanInTheLoop, useFrontendTool handlersCopilotKit v2 components use cpk: prefixed Tailwind classes internally. To override them:
/* copilotkit-overrides.css — import in layout.tsx AFTER globals.css */
.copilotKitInput {
border-radius: 0.75rem;
border: 1px solid var(--copilot-kit-separator-color) !important;
}
Do NOT hardcode
.copilotKitChat { background-color: #fff !important; }in these overrides. It overrides demo-level theming (e.g. beautiful-chat's dark mode toggle) and forces white in every theme. Let the v2 styles + per-demoThemeProviderhandle chat backgrounds.
// layout.tsx
import "./globals.css";
import "./copilotkit-overrides.css"; // AFTER globals
Tailwind v4 processes globals.css and purges anything it doesn't recognize as a utility class. CopilotKit class selectors (.copilotKitChat, .copilotKitInput) will be stripped.
To get proper spacing around the CopilotChat component, wrap it like the Dojo does:
// GOOD — matches Dojo spacing
<div className="flex justify-center items-center h-screen w-full">
<div className="h-full w-full md:w-4/5 md:h-4/5 rounded-lg">
<CopilotChat className="h-full rounded-2xl max-w-6xl mx-auto" />
</div>
</div>
The md:w-4/5 gives 80% width on desktop, creating natural side margins. Don't try to add margins via CSS overrides on CopilotKit's internal classes — use the wrapper div.
All integration packages should use the light theme to match the showcase shell:
:root {
--copilot-kit-background-color: #fafaf9;
--copilot-kit-primary-color: #0d6e3f;
--copilot-kit-response-button-background-color: #f5f5f3;
--copilot-kit-response-button-color: #1a1a18;
}
html,
body {
background: #fafaf9;
color: #1a1a18;
}
Don't reference local image files from agent-generated content. The agent may generate image_name values that don't exist on disk. Always add an onError fallback:
{
imageUrl && !imageError && (
setImageError(true)} alt="..." />
);
}
Don't use SVG icons with fill="currentColor" inside CopilotKit chat messages. The currentColor inheritance is unpredictable in the chat context. Use emoji instead:
// GOOD
<span style={{ fontSize: "48px" }}>☀️</span>
// BAD — renders as giant black shapes
<svg fill="currentColor" className="w-14 h-14 text-yellow-200">...</svg>
Always build and test the Docker image locally before pushing:
cd /proj/cpk/CopilotKit
docker build -f showcase/integrations/<slug>/Dockerfile -t <slug>-local showcase/integrations/<slug>/
docker run -d --name <slug>-local -p 4444:10000 -e PORT=10000 <slug>-local
# Verify CSS overrides survived the build
curl -sf http://localhost:4444/_next/static/css/*.css | grep "copilotKitChat"
# Verify inline styles are in rendered HTML
curl -sf http://localhost:4444/demos/<demo> | grep "padding: 32px"
# Clean up
docker rm -f <slug>-local