theme-snippets
Installation
SKILL.md
Odoo Website Snippets Reference
Snippet Architecture
STATIC SNIPPETS (81+ templates) DYNAMIC SNIPPETS (data-driven)
├── Pre-built HTML ├── ir.filters (data source)
├── Drag & drop ├── website.snippet.filter (fetcher)
└── 6 categories └── Display templates (renderer)
SNIPPET OPTIONS SYSTEM
we-select | we-button | we-colorpicker | we-input
data-selector | data-select-class | data-js
Static Snippets Inventory (81+ Templates)
Structure (14)
| ID | Name | Best For |
|---|---|---|
s_banner |
Banner | Landing pages |
s_cover |
Cover | Headers |
s_text_image |
Text - Image | Features |
s_image_text |
Image - Text | Features |
s_title |
Title | Section headers |
s_text_block |
Text Block | Content |
s_numbers |
Numbers | Achievements |
s_three_columns |
Columns | Content grids |
s_features |
Features | Benefits |
s_masonry_block |
Masonry | Galleries |
s_image_gallery |
Image Gallery | Portfolios |
Features (11)
| ID | Name | Best For |
|---|---|---|
s_comparisons |
Comparisons | Product comparison |
s_company_team |
Team | About pages |
s_call_to_action |
CTA | Conversions |
s_references |
References | Social proof |
s_accordion |
Accordion | FAQs |
s_features_grid |
Features Grid | Benefits |
s_pricelist |
Pricelist | Products |
s_faq_collapse |
FAQ | Support |
s_tabs |
Tabs | Organization |
Dynamic Content (15)
| ID | Name | Best For |
|---|---|---|
s_google_map |
Google Map | Contact pages |
s_website_form |
Form | Lead capture |
s_social_media |
Social Media | Engagement |
s_dynamic_snippet |
Dynamic | Data display |
s_countdown |
Countdown | Events |
s_popup |
Popup | Promotions |
s_newsletter_block |
Newsletter | Marketing |
Inner Content (16)
| ID | Name | Best For |
|---|---|---|
s_card |
Card | Items |
s_three_cards / s_four_cards |
Cards | Services/Features |
s_timeline |
Timeline | History |
s_process_steps |
Steps | Tutorials |
s_quotes_carousel |
Testimonials | Social proof |
s_blockquote |
Blockquote | Emphasis |
Mega Menus (9)
| ID | Name |
|---|---|
s_mega_menu_multi_menus |
Multiple menu columns |
s_mega_menu_cards |
Card-style menu |
s_mega_menu_big_icons |
Large icon menu |
s_mega_menu_thumbnails |
Thumbnail gallery |
Creating Custom Snippets
Odoo 14-17: Simple Registration
<!-- 1. Snippet Template -->
<template id="s_my_snippet" name="My Snippet">
<section class="s_my_snippet pt48 pb48 o_cc o_cc1">
<div class="container">
<div class="row">
<div class="col-lg-12 text-center">
<h2>My Custom Snippet</h2>
<p>Content goes here...</p>
</div>
</div>
</div>
</section>
</template>
<!-- 2. Registration -->
<template id="s_my_snippet_insert" inherit_id="website.snippets">
<xpath expr="//div[@id='snippet_structure']//t[@t-snippet][last()]" position="after">
<t t-snippet="my_module.s_my_snippet"
t-thumbnail="/my_module/static/src/img/snippets/s_my_snippet.svg">
<keywords>my, custom, snippet</keywords>
</t>
</xpath>
</template>
Odoo 18-19: With Snippet Groups (Required)
<!-- 1. Snippet Template (same as above) -->
<!-- 2. Custom Group (optional) -->
<template id="snippet_group_custom" inherit_id="website.snippets">
<xpath expr="//div[@id='snippet_groups']" position="inside">
<t snippet-group="custom"
t-snippet="website.s_snippet_group"
string="Custom Snippets"/>
</xpath>
</template>
<!-- 3. Registration with Group -->
<template id="s_my_snippet_insert" inherit_id="website.snippets">
<xpath expr="//div[@id='snippet_structure']/*[1]" position="before">
<t t-snippet="my_module.s_my_snippet"
string="My Snippet"
group="custom"
t-thumbnail="/my_module/static/src/img/snippets/s_my_snippet.svg"/>
</xpath>
</template>
Panel Categories
| Panel ID | Category | XPath |
|---|---|---|
snippet_structure |
Structure | //div[@id='snippet_structure'] |
snippet_media |
Gallery | //div[@id='snippet_media'] |
snippet_feature |
Features | //div[@id='snippet_feature'] |
snippet_dynamic |
Dynamic | //div[@id='snippet_dynamic'] |
snippet_inner_content |
Inner Content | //div[@id='snippet_inner_content'] |
Snippet Options System
We-* Elements
| Element | Purpose | Key Attributes |
|---|---|---|
we-select |
Dropdown | data-name, data-dependencies |
we-button |
Toggle | data-select-class, data-toggle-class |
we-colorpicker |
Color picker | data-css-property |
we-input |
Text/number | data-attribute-name, data-unit |
we-range |
Slider | data-min, data-max, data-step |
we-checkbox |
Toggle | data-select-class |
Data Attributes
| Attribute | Purpose |
|---|---|
data-selector |
Target CSS selector |
data-select-class |
Toggle CSS class |
data-css-property |
CSS property to modify |
data-js |
JavaScript handler class |
data-dependencies |
Conditional visibility |
data-no-preview |
Disable live preview |
Complete Options Example
<template id="s_my_snippet_options" inherit_id="website.snippet_options">
<xpath expr="." position="inside">
<div data-js="MySnippetHandler"
data-selector=".s_my_snippet"
data-drop-in=".oe_structure">
<we-select string="Layout" data-name="layout_opt">
<we-button data-select-class="layout-grid">Grid</we-button>
<we-button data-select-class="layout-list">List</we-button>
</we-select>
<we-colorpicker string="Background"
data-select-style="true"
data-css-property="background-color"/>
<we-range string="Spacing"
data-select-style="true"
data-css-property="gap"
data-min="0" data-max="100" data-step="5" data-unit="px"/>
</div>
</xpath>
</template>
Built-in JS Handlers
BackgroundImage, BackgroundPosition, BackgroundToggler, ColoredLevelBackground, BackgroundShape, ImageTools, Carousel, DynamicSnippet
Dynamic Snippets
Architecture
Data Source (ir.filters) → Filter (website.snippet.filter) → Display Template
Creating a Dynamic Snippet
Step 1: Define Filter
<record id="dynamic_filter_my_items" model="website.snippet.filter">
<field name="name">My Items</field>
<field name="model_name">my.model</field>
<field name="limit">12</field>
<field name="filter_id" ref="ir_filter_my_items"/>
</record>
Step 2: Display Template
<template id="dynamic_filter_template_my_model_card" name="Card">
<t t-foreach="records" t-as="record">
<div class="col-lg-4 mb-4">
<div class="card h-100 shadow-sm">
<img t-att-src="record.image_url" class="card-img-top"/>
<div class="card-body">
<h5 class="card-title" t-esc="record.name"/>
</div>
</div>
</div>
</t>
</template>
Step 3: Dynamic Snippet
<template id="s_dynamic_my_items" name="Dynamic My Items">
<section class="s_dynamic_snippet s_dynamic_my_items pt48 pb48"
data-snippet="s_dynamic_my_items"
data-filter-id="my_module.dynamic_filter_my_items"
data-template-key="my_module.dynamic_filter_template_my_model_card"
data-number-of-elements="6">
<div class="container">
<div class="dynamic_snippet_template row"/>
</div>
</section>
</template>
Product Dynamic Snippets (website_sale)
Filters: dynamic_filter_newest_products, dynamic_filter_recently_sold_products, dynamic_filter_recently_viewed_products
Display templates: card_style, centered, horizontal_card, mini, minimalist_*, banner
Snippet JavaScript
publicWidget for Snippets
/** @odoo-module **/
import publicWidget from "@web/legacy/js/public/public_widget";
publicWidget.registry.MySnippet = publicWidget.Widget.extend({
selector: '.s_my_snippet',
disabledInEditableMode: false,
start: function () {
if (!this.editableMode) {
this._initializeAnimations();
}
return this._super(...arguments);
},
destroy: function () {
$(window).off('.mySnippet');
this._super(...arguments);
},
});
Snippet Options Handler
/** @odoo-module **/
import options from "@web_editor/js/editor/snippets.options";
options.registry.MySnippetOption = options.Class.extend({
selectDataAttribute: function(previewMode, widgetValue, params) {
this._super(...arguments);
if (params.attributeName === 'layout') {
this._applyLayout(widgetValue);
}
},
});
Version Differences
| Feature | Odoo 14-17 | Odoo 18-19 |
|---|---|---|
| Registration | Simple XPath | Groups required |
| JavaScript | @odoo-module |
Plugin-based (19) |
| Bootstrap | 4.x (14-15) / 5.1.3 (16-17) | 5.1.3 |
| Dynamic Snippets | Basic | Full + Categories |
Odoo 19 Plugin Pattern
import { Plugin } from "@html_editor/plugin";
import { registry } from "@web/core/registry";
export class MySnippetPlugin extends Plugin {
static id = "mySnippet";
resources = {
builder_options: [{
template: "my_module.MySnippetOption",
selector: "section",
applyTo: ".s_my_snippet",
}],
};
}
registry.category("website-plugins").add("mySnippet", MySnippetPlugin);
Weekly Installs
4
Repository
ahmed-lakosha/o…de-skillGitHub Stars
36
First Seen
14 days ago
Security Audits
Installed on
opencode4
deepagents4
antigravity4
claude-code4
github-copilot4
codex4