ac70719d4d
Extends Open Design from web-only to a multi-modal creation tool. The unifying contract is one code-agent loop driven by skills + project metadata + prompt constraints; for non-web surfaces the agent shells out to a single dispatcher (`od media generate`) that the daemon routes per (surface, model). - Types: new Surface union, MediaAspect / AudioKind, image/video/audio ProjectKind + ProjectMetadata fields, video/audio ProjectFileKind. - NewProjectPanel: top-level surface picker + Image / Video / Audio forms with model, aspect, length, duration, voice, audio-kind pickers. - ExamplesTab + DesignSystemsTab: surface filter row that scopes before mode / scenario / category filters. - FileViewer / FileWorkspace: native <video> and <audio> previews and matching tab icons. - Daemon: parses `od.surface` and `> Surface:` blockquotes; recognises mp4 / webm / mov / mp3 / wav / ogg / m4a / flac extensions; spawns agents with OD_BIN / OD_DAEMON_URL / OD_PROJECT_ID / OD_PROJECT_DIR env so any code-agent CLI with shell access can call the dispatcher. - daemon/media.js + daemon/media-models.js: surface-agnostic dispatcher with stub providers that emit deterministic placeholder bytes (1x1 PNG, valid mp4 ftyp, mp3 frame / silent WAV) so the framework works without API keys; real provider integrations slot in later. - daemon/cli.js: `od media generate --surface ... --model ...` subcommand routes to POST /api/projects/:id/media/generate and prints one JSON line for the agent to parse. - prompts/media-contract.ts: hard contract pinned LAST in the system prompt for image/video/audio surfaces — env vars, exact invocation, registered model IDs per surface, six workflow rules. system.ts metadata block updated to point at the contract. - Seed skills: image-poster, video-shortform, audio-jingle each ship a SKILL.md with `mode/surface: image|video|audio` and a stylized example.html preview, and instruct the agent to dispatch via the contract. Made-with: Cursor
565 lines
19 KiB
TypeScript
565 lines
19 KiB
TypeScript
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
|
import { useT } from '../i18n';
|
|
import type { Dict } from '../i18n/types';
|
|
import { fetchSkillExample } from '../providers/registry';
|
|
import { exportAsHtml, exportAsPdf, exportAsZip } from '../runtime/exports';
|
|
import { buildSrcdoc } from '../runtime/srcdoc';
|
|
import type { SkillSummary, Surface } from '../types';
|
|
import { Icon } from './Icon';
|
|
import { PreviewModal } from './PreviewModal';
|
|
|
|
type TranslateFn = (key: keyof Dict, vars?: Record<string, string | number>) => string;
|
|
|
|
interface Props {
|
|
skills: SkillSummary[];
|
|
onUsePrompt: (skill: SkillSummary) => void;
|
|
}
|
|
|
|
type SurfaceFilter = 'all' | Surface;
|
|
type ModeFilter =
|
|
| 'all'
|
|
| 'prototype-desktop'
|
|
| 'prototype-mobile'
|
|
| 'deck'
|
|
| 'document'
|
|
| 'image'
|
|
| 'video'
|
|
| 'audio';
|
|
type ScenarioFilter = string;
|
|
|
|
// Each surface gets its own type pills. We branch on `SURFACE_PILLS` so
|
|
// the mode row reflects what makes sense within the active surface
|
|
// (web has the most granularity; image / video / audio collapse to a
|
|
// single mode pill so the pill count stays reasonable).
|
|
const SURFACE_PILLS: { value: SurfaceFilter; labelKey: keyof Dict; icon: 'grid' | 'image' | 'video' | 'music' | null }[] = [
|
|
{ value: 'all', labelKey: 'examples.modeAll', icon: null },
|
|
{ value: 'web', labelKey: 'examples.surfaceWeb', icon: 'grid' },
|
|
{ value: 'image', labelKey: 'examples.surfaceImage', icon: 'image' },
|
|
{ value: 'video', labelKey: 'examples.surfaceVideo', icon: 'video' },
|
|
{ value: 'audio', labelKey: 'examples.surfaceAudio', icon: 'music' },
|
|
];
|
|
|
|
const WEB_MODE_PILLS: { value: ModeFilter; labelKey: keyof Dict }[] = [
|
|
{ value: 'all', labelKey: 'examples.modeAll' },
|
|
{ value: 'prototype-desktop', labelKey: 'examples.modePrototypeDesktop' },
|
|
{ value: 'prototype-mobile', labelKey: 'examples.modePrototypeMobile' },
|
|
{ value: 'deck', labelKey: 'examples.modeDeck' },
|
|
{ value: 'document', labelKey: 'examples.modeDocument' },
|
|
];
|
|
const IMAGE_MODE_PILLS: { value: ModeFilter; labelKey: keyof Dict }[] = [
|
|
{ value: 'all', labelKey: 'examples.modeAll' },
|
|
{ value: 'image', labelKey: 'examples.modeImage' },
|
|
];
|
|
const VIDEO_MODE_PILLS: { value: ModeFilter; labelKey: keyof Dict }[] = [
|
|
{ value: 'all', labelKey: 'examples.modeAll' },
|
|
{ value: 'video', labelKey: 'examples.modeVideo' },
|
|
];
|
|
const AUDIO_MODE_PILLS: { value: ModeFilter; labelKey: keyof Dict }[] = [
|
|
{ value: 'all', labelKey: 'examples.modeAll' },
|
|
{ value: 'audio', labelKey: 'examples.modeAudio' },
|
|
];
|
|
|
|
// Convenience — the union pill list for the "All surfaces" view.
|
|
const ALL_MODE_PILLS: { value: ModeFilter; labelKey: keyof Dict }[] = [
|
|
...WEB_MODE_PILLS,
|
|
{ value: 'image', labelKey: 'examples.modeImage' },
|
|
{ value: 'video', labelKey: 'examples.modeVideo' },
|
|
{ value: 'audio', labelKey: 'examples.modeAudio' },
|
|
];
|
|
|
|
function surfaceOf(skill: SkillSummary): Surface {
|
|
if (skill.surface) return skill.surface;
|
|
if (skill.mode === 'image') return 'image';
|
|
if (skill.mode === 'video') return 'video';
|
|
if (skill.mode === 'audio') return 'audio';
|
|
return 'web';
|
|
}
|
|
|
|
function pillsForSurface(surface: SurfaceFilter): { value: ModeFilter; labelKey: keyof Dict }[] {
|
|
if (surface === 'web') return WEB_MODE_PILLS;
|
|
if (surface === 'image') return IMAGE_MODE_PILLS;
|
|
if (surface === 'video') return VIDEO_MODE_PILLS;
|
|
if (surface === 'audio') return AUDIO_MODE_PILLS;
|
|
return ALL_MODE_PILLS;
|
|
}
|
|
|
|
const SCENARIO_LABEL_KEY: Record<string, keyof Dict> = {
|
|
general: 'examples.scenarioGeneral',
|
|
engineering: 'examples.scenarioEngineering',
|
|
product: 'examples.scenarioProduct',
|
|
design: 'examples.scenarioDesign',
|
|
marketing: 'examples.scenarioMarketing',
|
|
sales: 'examples.scenarioSales',
|
|
finance: 'examples.scenarioFinance',
|
|
hr: 'examples.scenarioHr',
|
|
operations: 'examples.scenarioOperations',
|
|
support: 'examples.scenarioSupport',
|
|
legal: 'examples.scenarioLegal',
|
|
education: 'examples.scenarioEducation',
|
|
personal: 'examples.scenarioPersonal',
|
|
};
|
|
|
|
function scenarioLabel(t: TranslateFn, tag: string): string {
|
|
const key = SCENARIO_LABEL_KEY[tag];
|
|
if (key) return t(key);
|
|
return tag.charAt(0).toUpperCase() + tag.slice(1);
|
|
}
|
|
|
|
const SCENARIO_ORDER = [
|
|
'engineering',
|
|
'product',
|
|
'design',
|
|
'marketing',
|
|
'sales',
|
|
'finance',
|
|
'hr',
|
|
'operations',
|
|
'support',
|
|
'legal',
|
|
'education',
|
|
'personal',
|
|
'general',
|
|
];
|
|
|
|
function matchesMode(skill: SkillSummary, filter: ModeFilter): boolean {
|
|
if (filter === 'all') return true;
|
|
if (filter === 'deck') return skill.mode === 'deck';
|
|
if (filter === 'prototype-desktop')
|
|
return skill.mode === 'prototype' && (skill.platform ?? 'desktop') === 'desktop';
|
|
if (filter === 'prototype-mobile')
|
|
return skill.mode === 'prototype' && skill.platform === 'mobile';
|
|
if (filter === 'document') return skill.mode === 'template';
|
|
if (filter === 'image') return surfaceOf(skill) === 'image';
|
|
if (filter === 'video') return surfaceOf(skill) === 'video';
|
|
if (filter === 'audio') return surfaceOf(skill) === 'audio';
|
|
return true;
|
|
}
|
|
|
|
function matchesSurface(skill: SkillSummary, filter: SurfaceFilter): boolean {
|
|
if (filter === 'all') return true;
|
|
return surfaceOf(skill) === filter;
|
|
}
|
|
|
|
export function ExamplesTab({ skills, onUsePrompt }: Props) {
|
|
const t = useT();
|
|
// Hold preview HTML per skill across re-renders so cards never re-flicker.
|
|
const [previews, setPreviews] = useState<Record<string, string | null>>({});
|
|
const [surfaceFilter, setSurfaceFilter] = useState<SurfaceFilter>('all');
|
|
const [modeFilter, setModeFilter] = useState<ModeFilter>('all');
|
|
const [scenarioFilter, setScenarioFilter] = useState<ScenarioFilter>('all');
|
|
const [previewSkillId, setPreviewSkillId] = useState<string | null>(null);
|
|
|
|
const loadPreview = useCallback(
|
|
async (id: string) => {
|
|
if (previews[id] !== undefined) return;
|
|
const html = await fetchSkillExample(id);
|
|
setPreviews((prev) => ({ ...prev, [id]: html }));
|
|
},
|
|
[previews],
|
|
);
|
|
|
|
// Open the modal for a card. We always trigger a preview fetch even if
|
|
// the card hasn't been hovered yet — the modal needs the HTML.
|
|
const openPreview = useCallback(
|
|
(id: string) => {
|
|
setPreviewSkillId(id);
|
|
void loadPreview(id);
|
|
},
|
|
[loadPreview],
|
|
);
|
|
|
|
const previewSkill = useMemo(
|
|
() => (previewSkillId ? skills.find((s) => s.id === previewSkillId) ?? null : null),
|
|
[skills, previewSkillId],
|
|
);
|
|
|
|
const surfaceCounts = useMemo(() => {
|
|
const counts: Record<SurfaceFilter, number> = {
|
|
all: skills.length,
|
|
web: 0,
|
|
image: 0,
|
|
video: 0,
|
|
audio: 0,
|
|
};
|
|
for (const s of skills) {
|
|
const sf = surfaceOf(s);
|
|
counts[sf] = (counts[sf] ?? 0) + 1;
|
|
}
|
|
return counts;
|
|
}, [skills]);
|
|
|
|
const surfaceScopedSkills = useMemo(
|
|
() => skills.filter((s) => matchesSurface(s, surfaceFilter)),
|
|
[skills, surfaceFilter],
|
|
);
|
|
|
|
const modePills = useMemo(() => pillsForSurface(surfaceFilter), [surfaceFilter]);
|
|
|
|
const modeCounts = useMemo(() => {
|
|
const c: Record<string, number> = { all: surfaceScopedSkills.length };
|
|
for (const p of modePills) {
|
|
if (p.value === 'all') continue;
|
|
c[p.value] = surfaceScopedSkills.filter((s) => matchesMode(s, p.value)).length;
|
|
}
|
|
return c;
|
|
}, [surfaceScopedSkills, modePills]);
|
|
|
|
const scenarioCounts = useMemo(() => {
|
|
const counts = new Map<string, number>();
|
|
for (const s of surfaceScopedSkills) {
|
|
if (!matchesMode(s, modeFilter)) continue;
|
|
const tag = s.scenario || 'general';
|
|
counts.set(tag, (counts.get(tag) ?? 0) + 1);
|
|
}
|
|
return counts;
|
|
}, [surfaceScopedSkills, modeFilter]);
|
|
|
|
const scenarioOptions = useMemo(() => {
|
|
const have = new Set(scenarioCounts.keys());
|
|
const ordered: string[] = [];
|
|
for (const k of SCENARIO_ORDER) if (have.has(k)) ordered.push(k);
|
|
for (const k of [...have].sort()) if (!ordered.includes(k)) ordered.push(k);
|
|
return ordered;
|
|
}, [scenarioCounts]);
|
|
|
|
const filtered = useMemo(() => {
|
|
const matched = surfaceScopedSkills.filter((s) => {
|
|
if (!matchesMode(s, modeFilter)) return false;
|
|
if (scenarioFilter === 'all') return true;
|
|
return (s.scenario || 'general') === scenarioFilter;
|
|
});
|
|
// Featured magazine-style examples float to the top (lower priority
|
|
// number wins). Non-featured skills keep their server-side order so
|
|
// contributors can still author SKILL.md alphabetically.
|
|
return matched
|
|
.map((s, idx) => ({ s, idx }))
|
|
.sort((a, b) => {
|
|
const aRank = typeof a.s.featured === 'number' ? a.s.featured : Number.POSITIVE_INFINITY;
|
|
const bRank = typeof b.s.featured === 'number' ? b.s.featured : Number.POSITIVE_INFINITY;
|
|
if (aRank !== bRank) return aRank - bRank;
|
|
return a.idx - b.idx;
|
|
})
|
|
.map(({ s }) => s);
|
|
}, [surfaceScopedSkills, modeFilter, scenarioFilter]);
|
|
|
|
if (skills.length === 0) {
|
|
return <div className="tab-empty">{t('examples.emptyNoSkills')}</div>;
|
|
}
|
|
|
|
return (
|
|
<div className="tab-panel examples-panel">
|
|
<div className="examples-toolbar">
|
|
<div
|
|
className="examples-filter-row"
|
|
role="tablist"
|
|
aria-label={t('examples.surfaceLabel')}
|
|
>
|
|
<span className="examples-filter-label">{t('examples.surfaceLabel')}</span>
|
|
{SURFACE_PILLS.map((p) => (
|
|
<button
|
|
key={p.value}
|
|
type="button"
|
|
role="tab"
|
|
aria-selected={surfaceFilter === p.value}
|
|
className={`filter-pill ${surfaceFilter === p.value ? 'active' : ''}`}
|
|
onClick={() => {
|
|
setSurfaceFilter(p.value);
|
|
setModeFilter('all');
|
|
setScenarioFilter('all');
|
|
}}
|
|
>
|
|
{p.icon ? <Icon name={p.icon} size={12} /> : null}
|
|
{t(p.labelKey)}
|
|
<span className="filter-pill-count">{surfaceCounts[p.value]}</span>
|
|
</button>
|
|
))}
|
|
</div>
|
|
<div
|
|
className="examples-filter-row"
|
|
role="tablist"
|
|
aria-label={t('examples.typeLabel')}
|
|
>
|
|
<span className="examples-filter-label">{t('examples.typeLabel')}</span>
|
|
{modePills.map((p) => (
|
|
<button
|
|
key={p.value}
|
|
type="button"
|
|
role="tab"
|
|
aria-selected={modeFilter === p.value}
|
|
className={`filter-pill ${modeFilter === p.value ? 'active' : ''}`}
|
|
onClick={() => {
|
|
setModeFilter(p.value);
|
|
setScenarioFilter('all');
|
|
}}
|
|
>
|
|
{t(p.labelKey)}
|
|
<span className="filter-pill-count">
|
|
{p.value === 'all' ? surfaceScopedSkills.length : (modeCounts[p.value] ?? 0)}
|
|
</span>
|
|
</button>
|
|
))}
|
|
</div>
|
|
{scenarioOptions.length > 1 ? (
|
|
<div
|
|
className="examples-filter-row"
|
|
role="tablist"
|
|
aria-label={t('examples.scenarioLabel')}
|
|
>
|
|
<span className="examples-filter-label">
|
|
{t('examples.scenarioLabel')}
|
|
</span>
|
|
<button
|
|
type="button"
|
|
className={`filter-pill ${scenarioFilter === 'all' ? 'active' : ''}`}
|
|
onClick={() => setScenarioFilter('all')}
|
|
>
|
|
{t('examples.modeAll')}
|
|
<span className="filter-pill-count">{filtered.length}</span>
|
|
</button>
|
|
{scenarioOptions.map((tag) => (
|
|
<button
|
|
key={tag}
|
|
type="button"
|
|
className={`filter-pill ${scenarioFilter === tag ? 'active' : ''}`}
|
|
onClick={() => setScenarioFilter(tag)}
|
|
>
|
|
{scenarioLabel(t, tag)}
|
|
<span className="filter-pill-count">{scenarioCounts.get(tag) ?? 0}</span>
|
|
</button>
|
|
))}
|
|
</div>
|
|
) : null}
|
|
</div>
|
|
|
|
{filtered.length === 0 ? (
|
|
<div className="tab-empty">{t('examples.emptyNoMatch')}</div>
|
|
) : (
|
|
filtered.map((skill) => (
|
|
<ExampleCard
|
|
key={skill.id}
|
|
skill={skill}
|
|
html={previews[skill.id]}
|
|
onLoad={() => void loadPreview(skill.id)}
|
|
onUsePrompt={() => onUsePrompt(skill)}
|
|
onOpenPreview={() => openPreview(skill.id)}
|
|
/>
|
|
))
|
|
)}
|
|
{previewSkill ? (
|
|
<PreviewModal
|
|
title={previewSkill.name}
|
|
subtitle={previewSkill.examplePrompt || previewSkill.description.replace(/\s+/g, ' ').slice(0, 160)}
|
|
views={[
|
|
{
|
|
id: 'preview',
|
|
label: t('examples.previewLabel'),
|
|
html: previews[previewSkill.id],
|
|
},
|
|
]}
|
|
exportTitleFor={() => previewSkill.name}
|
|
onClose={() => setPreviewSkillId(null)}
|
|
/>
|
|
) : null}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function ExampleCard({
|
|
skill,
|
|
html,
|
|
onLoad,
|
|
onUsePrompt,
|
|
onOpenPreview,
|
|
}: {
|
|
skill: SkillSummary;
|
|
html: string | null | undefined;
|
|
onLoad: () => void;
|
|
onUsePrompt: () => void;
|
|
onOpenPreview: () => void;
|
|
}) {
|
|
const t = useT();
|
|
const [hovered, setHovered] = useState(false);
|
|
const [shareOpen, setShareOpen] = useState(false);
|
|
const shareRef = useRef<HTMLDivElement | null>(null);
|
|
|
|
useEffect(() => {
|
|
if (!shareOpen) return;
|
|
const onDoc = (e: MouseEvent) => {
|
|
if (!shareRef.current) return;
|
|
if (!shareRef.current.contains(e.target as Node)) setShareOpen(false);
|
|
};
|
|
const onKey = (e: KeyboardEvent) => {
|
|
if (e.key === 'Escape') setShareOpen(false);
|
|
};
|
|
document.addEventListener('mousedown', onDoc);
|
|
document.addEventListener('keydown', onKey);
|
|
return () => {
|
|
document.removeEventListener('mousedown', onDoc);
|
|
document.removeEventListener('keydown', onKey);
|
|
};
|
|
}, [shareOpen]);
|
|
|
|
const exportTitle = skill.name;
|
|
const isMobile = skill.platform === 'mobile';
|
|
const isDeck = skill.mode === 'deck';
|
|
|
|
return (
|
|
<div
|
|
className="example-card"
|
|
onMouseEnter={() => {
|
|
setHovered(true);
|
|
onLoad();
|
|
}}
|
|
onMouseLeave={() => setHovered(false)}
|
|
>
|
|
<div
|
|
className="example-preview"
|
|
role="button"
|
|
tabIndex={0}
|
|
title={t('common.openPreview')}
|
|
onClick={onOpenPreview}
|
|
onKeyDown={(e) => {
|
|
if (e.key === 'Enter' || e.key === ' ') {
|
|
e.preventDefault();
|
|
onOpenPreview();
|
|
}
|
|
}}
|
|
>
|
|
{html ? (
|
|
<>
|
|
<iframe
|
|
title={`${skill.name} ${t('examples.previewLabel').toLowerCase()}`}
|
|
sandbox="allow-scripts"
|
|
srcDoc={buildSrcdoc(html)}
|
|
tabIndex={-1}
|
|
/>
|
|
<span className="example-preview-overlay" aria-hidden="true">
|
|
{t('examples.openPreview')}
|
|
</span>
|
|
</>
|
|
) : (
|
|
<div className="example-preview-placeholder">
|
|
{hovered
|
|
? t('examples.loadingPreview')
|
|
: t('examples.hoverPreview')}
|
|
</div>
|
|
)}
|
|
</div>
|
|
<div className="example-meta">
|
|
<div className="example-name">{skill.name}</div>
|
|
<div className="example-tags">
|
|
<span className={`example-tag ${isMobile ? 'platform-mobile' : ''} ${isDeck ? 'mode-deck' : ''}`}>
|
|
{tagForSkill(skill, t)}
|
|
</span>
|
|
{skill.scenario && skill.scenario !== 'general' ? (
|
|
<span className="example-tag">
|
|
{scenarioLabel(t, skill.scenario)}
|
|
</span>
|
|
) : null}
|
|
</div>
|
|
<div className="example-prompt">
|
|
{skill.examplePrompt
|
|
? `“${skill.examplePrompt}”`
|
|
: skill.description.replace(/\s+/g, ' ').slice(0, 240)}
|
|
</div>
|
|
<div className="example-card-actions">
|
|
<button className="primary example-cta" onClick={onUsePrompt}>
|
|
{t('examples.usePrompt')}
|
|
</button>
|
|
<button
|
|
className="ghost"
|
|
onClick={onOpenPreview}
|
|
title={t('examples.previewModalTitle')}
|
|
>
|
|
{t('examples.openPreview')}
|
|
</button>
|
|
<div className="share-menu" ref={shareRef}>
|
|
<button
|
|
className="ghost"
|
|
aria-haspopup="menu"
|
|
aria-expanded={shareOpen}
|
|
disabled={!html}
|
|
title={
|
|
html
|
|
? t('examples.shareTitle')
|
|
: t('examples.shareLoadFirst')
|
|
}
|
|
onClick={() => setShareOpen((v) => !v)}
|
|
>
|
|
{t('examples.shareMenu')}
|
|
</button>
|
|
{shareOpen && html ? (
|
|
<div className="share-menu-popover" role="menu">
|
|
<button
|
|
type="button"
|
|
className="share-menu-item"
|
|
role="menuitem"
|
|
onClick={() => {
|
|
setShareOpen(false);
|
|
exportAsPdf(html, exportTitle, { deck: isDeck });
|
|
}}
|
|
>
|
|
<span className="share-menu-icon">📄</span>
|
|
<span>
|
|
{isDeck
|
|
? t('examples.exportPdfAllSlides')
|
|
: t('common.exportPdf')}
|
|
</span>
|
|
</button>
|
|
{isDeck ? (
|
|
<button
|
|
type="button"
|
|
className="share-menu-item"
|
|
role="menuitem"
|
|
title={t('examples.exportPptxLocked')}
|
|
disabled
|
|
>
|
|
<span className="share-menu-icon">📊</span>
|
|
<span>{t('examples.exportPptxLocked')}</span>
|
|
</button>
|
|
) : null}
|
|
<div className="share-menu-divider" />
|
|
<button
|
|
type="button"
|
|
className="share-menu-item"
|
|
role="menuitem"
|
|
onClick={() => {
|
|
setShareOpen(false);
|
|
exportAsZip(html, exportTitle);
|
|
}}
|
|
>
|
|
<span className="share-menu-icon">🗜</span>
|
|
<span>{t('common.exportZip')}</span>
|
|
</button>
|
|
<button
|
|
type="button"
|
|
className="share-menu-item"
|
|
role="menuitem"
|
|
onClick={() => {
|
|
setShareOpen(false);
|
|
exportAsHtml(html, exportTitle);
|
|
}}
|
|
>
|
|
<span className="share-menu-icon">🌐</span>
|
|
<span>{t('common.exportHtml')}</span>
|
|
</button>
|
|
</div>
|
|
) : null}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function tagForSkill(skill: SkillSummary, t: TranslateFn): string {
|
|
if (skill.mode === 'image') return t('examples.tagImage');
|
|
if (skill.mode === 'video') return t('examples.tagVideo');
|
|
if (skill.mode === 'audio') return t('examples.tagAudio');
|
|
if (skill.mode === 'deck') return t('examples.tagSlideDeck');
|
|
if (skill.mode === 'template') return t('examples.tagTemplate');
|
|
if (skill.mode === 'design-system') return t('examples.tagDesignSystem');
|
|
if (skill.platform === 'mobile') return t('examples.tagMobilePrototype');
|
|
return t('examples.tagDesktopPrototype');
|
|
}
|