plugins/aem/edge-delivery-services/skills/ue-component-model/SKILL.md
Create or edit the Universal Editor component configuration (component-definition.json, component-models.json, component-filters.json) for AEM Edge Delivery Services blocks. Use this skill whenever the user mentions component models, component definitions, component filters, block configuration for the Universal Editor, UE block setup, adding a new block to UE, configuring block properties, block authoring fields, or any task involving the three JSON config files that control how blocks appear in the Universal Editor. Also trigger when the user wants to create a new EDS/Franklin block with UE support, modify block fields, add a block to the section filter, or asks about how blocks connect to the Universal Editor.
npx skillsauth add adobe/skills ue-component-modelInstall 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 helps you create or edit the three JSON configuration files that control how AEM Edge Delivery Services (EDS) blocks appear and behave in the Universal Editor (UE):
Before generating any configuration, read and analyze:
The block's JS file (blocks/<name>/<name>.js) — understand what content the decorate(block) function expects:
block.querySelector('a') (links/URLs), block.querySelector('picture') (images), etc.?The block's CSS file (blocks/<name>/<name>.css) — look for variant-specific styles.
Existing config — check if entries already exist:
component-definition.json for the block IDcomponent-models.json for the model IDcomponent-filters.json for the block in the section components listblocks/<name>/_<name>.json distributed config fileBased on the JS analysis:
Simple block: One component with its own fields. Most blocks are this type.
Container block: Has repeatable child items (cards, slides, tabs).
block.children or creates items from rowsKey-value block: Configuration-style block (2-column key-value pairs).
"key-value": true in templateMap the block's content expectations to component model fields. Read references/field-types.md for the full field type reference.
Common field mappings:
| Block expects... | Use component type | Notes |
|-----------------|-------------------|-------|
| An image | reference (name: image) | Pair with text field named imageAlt |
| A URL/link | aem-content (name: link or url) | For page links and external URLs |
| Rich text content | richtext | For formatted text with headings, lists, links |
| Plain text (single line) | text | For titles, labels, short strings |
| Plain text (multi-line) | textarea | For descriptions, notes, long text without formatting |
| Heading level choice | select with h1-h6 options | Name it titleType to auto-collapse with title |
| Style variants | multiselect (name: classes) | Values become CSS classes on block div |
| Multiple toggles | checkbox-group | For multiple independent boolean options |
| Boolean toggle | boolean | For show/hide options |
| Number value | number | For counts, limits |
| Content Fragment | aem-content-fragment | For CF-driven blocks |
| Experience Fragment | aem-experience-fragment | For reusable content+layout fragments |
| Content tags | aem-tag | For categorization via AEM tag picker |
Field naming rules (semantic collapsing):
image + imageAlt → collapsed into <picture><img alt="...">link + linkText + linkTitle + linkType → collapsed into <a href="..." title="...">text</a> with optional classtitle + titleType → collapsed into <h2>title</h2> (level from titleType)group_ (underscore separator) are grouped into a single cellGenerate entries for all three files. The approach depends on whether the project uses centralized or distributed config.
Check for distributed config pattern: If the block directory contains _<blockname>.json files (e.g., blocks/hero/_hero.json), create a distributed config file instead of editing the central files.
component-definition.json — Add to the "Blocks" group's components array:
{
"title": "<Block Display Name>",
"id": "<block-id>",
"plugins": {
"xwalk": {
"page": {
"resourceType": "core/franklin/components/block/v1/block",
"template": {
"name": "<Block Name>",
"model": "<model-id>"
}
}
}
}
}
For container blocks, add both the container AND item definitions. The container gets "filter" instead of "model", and the item uses "core/franklin/components/block/v1/block/item" as resourceType.
For key-value blocks, add "key-value": true to the template.
Template can include default values for any model field (e.g., "titleType": "h3", "classes": ["light"]).
component-models.json — Add a new model entry:
{
"id": "<model-id>",
"fields": [
{
"component": "<field-type>",
"name": "<property-name>",
"label": "<Display Label>",
"valueType": "string"
}
]
}
component-filters.json — Add the block ID to the section filter's components array. For container blocks, also add a new filter entry defining allowed children.
blocks/<name>/_<name>.json):Create a single file with all three configs:
{
"definitions": [ ... ],
"models": [ ... ],
"filters": [ ... ]
}
Still add the block to the section filter in the central component-filters.json.
After generating the config, verify:
id in the definition matches what's used in component-filters.json. The template.model value matches the id in component-models.json.section filter's components array (otherwise authors can't add it to pages).name properties in the model fields should produce HTML that the block's decorate() function can consume.image/imageAlt, not image/altText unless intentional).For detailed information, read these reference files as needed:
text, textarea, richtext, reference, aem-content, aem-content-fragment, aem-experience-fragment, aem-tag, select, multiselect, checkbox-group, radio-group, boolean, number, date-time, container, tab), valueType constraints, required properties, field properties, validation types, conditional fields, and option formatssection filter's components list.core/franklin/components/block/v1/block. Don't invent custom resource types.template.model must exactly match the model id, and template.filter must exactly match the filter id.text for single-line strings, textarea for multi-line plain text, and richtext for formatted content. For URLs and page links, use aem-content so authors get the content picker.valueType (e.g., boolean must use "boolean", number must use "number", checkbox-group must use "string[]"). Always include valueType and check the field-types reference for the enforced value.filter (not a model) in their template, and a corresponding filter entry in component-filters.json.tools
Identifies which items (pages, campaigns, products, channels, regions) had the biggest increases or decreases for a key metric between two time periods. Use this skill when someone asks "what's up and what's down," "which campaigns moved the most," "top gainers and losers," "what pages are trending," "show me what changed by channel," or any variation of identifying the biggest movers and decliners for a metric.
tools
Compares the performance of two or more audience segments across key metrics side by side. Use this skill when someone wants to compare audiences, cohorts, or groups — for example, "how do mobile users compare to desktop users on conversion," "compare new vs. returning visitors," "show me the difference between these two segments," "compare these audiences on our KPIs," or "which segment performs better." Also trigger for "segment comparison," "audience comparison," or "cohort comparison."
business
Produces a compact KPI digest showing how key metrics changed over a period and what's driving the movement. Use this skill when someone asks for a performance summary, a weekly recap, a morning briefing, a KPI update, or any variation of "how did we do this week/month." Also trigger for requests like "give me a performance overview," "what moved in the last 7 days," "pull our KPI report," or "summarize our metrics."
testing
Analyzes a multi-step conversion funnel to find where users drop off and which steps have the worst leakage. Use this skill when someone describes a journey or funnel and asks about conversion rates, drop-off, fallout, or step completion. Trigger for phrases like "analyze our onboarding funnel," "where are users dropping off," "what's our checkout conversion rate," "funnel analysis," "show me fallout between these steps," or "which step loses the most users."