Chapter 3 - Customisation, Part I¶
Add custom SCSS¶
You’ve adjusted Odoo and Bootstrap variables and set presets, yet you still notice disparities between your website and the client’s design. The only solution is to incorporate custom SCSS.
In theme.scss, reproduce the following design elements:
- Add a green underline on active nav items. 
- Modify the arrow for collapsible nav items. 
- Modify the slider’s arrows by adding a green background and changing their design. 
You will find the various media here.
See also
See reference documentation on how to add your SCSS rules.
 
 
Note
#wrapwrap. This ID is applied to the
div that groups the header, footer, and main content of all
your pages.Find the solution in our Airproof example on header.scss and caroussel.scss.
Add custom JS¶
Now, let’s add a mouse follower to the website. This interactive element will enhance the browsing experience, making it more engaging and visually appealing.
 
Use your JavaScript skills to implement this.
See also
See reference documentation on how to add Javascript code.
Find the solution in our Airproof example on mouse_follower.js and mouse_follower.scss.
Create a custom header¶
With variables, presets, and custom SCSS in place, it’s time to refine the layout and add key cross-page elements, starting with the header.
- Based on the Airproof design, create a custom header with the following elements: - A custom shopping cart icon. 
- A login/user as a button. 
- Navigation text to 14px. 
 - You can find the cart icon and template illustration. 
- In order for the client to find the mega menu you previously created for him, turn it into a template that can be found through the website builder. 
- Place your new header over the content of your homepage. 
See also
See reference documentation on how to:
- create custom headers, 
- do a XPath, 
- create a mega menu template, 
- create a header overlay. 
 
Tip
- Base yourself on the code of existing header templates that you can find in odoo/addons/website/views/website_templates.xml. 
- A good practise should be to create different files to manage your custom views and templates. For example, everything concerning the general layout (header, footer…) in - website_templates.xml, everything related to blog in- website_blog_templates.xml, to event in- website_event_templates.xml, etc.
- To modify the cart icon, you can use anXPath.Since this is linked to eCommerce, place it in a new file calledwebsite_sale_templates.xml.
- Don’t forget to continue making as many modifications as you can through the - Bootstrap variablesand- primary variables(font, colors, size…). You can use them to help you with this exercise.
Note
Ensure that you properly call every template in your header (like the shopping cart, language selector, call to action, etc.), so everything will remain editable via the website builder and all options will be available.
Find the solution in our Airproof example for:
- the xml structure and how to add the header and mega menu template to the options list on website_template.xml. 
- disable the default header: - /website_airproof/data/presets.xml¶- <!-- Disable default header --> <record id="website.template_header_default" model="ir.ui.view"> <field name="active" eval="False"/> </record> 
- declare your - website_templates.xmlfile along with all the new ones in your- manifest.
- disable the options you don’t want in your header via the presets. 
- make the use of primaries like - header-template,- navbar-font,- header-font-size…
- use bootstrap_overridden like - $navbar-light-color,- $navbar-light-hover-color,- $navbar-padding-y…
- add some scss rules. 
- to place the header over the content, add the right field to the home page: - /website_airproof/data/pages/home.xml¶- <!-- Home --> <record id="page_home" model="website.page"> <field name="header_overlay" eval="True"/> 
Create your custom building blocks¶
To allow your client to further customize his website, create tailor-made building blocks that he can freely drag & drop onto different pages.
Based on the Airproof design, create a custom carousel snippet to showcase drones. Then, add it as cover section on your homepage.
- Create the snippet template. Then, add it to the list of building blocks available in the website builder. Place it in its own category. Here you will find the images and snippet illustration. - See also - See reference documentation on how to create a custom building blocks.   
- Add two options in the Website Builder for the bubbles: - Allow users to choose only between a blue or green shadow. 
- Offer a range of possible margins between the bubbles. 
 - See also - See reference documentation on how to add snippet options.   
- Add the snippet on your homepage. 
Tip
Don’t forget to always properly declare your new files in your __manifest__.py and follow
the good folder structure seen previously.
To complete this exercise, you need to:
- Create your template. - You can find all the necessary information in s_airproof_carousel.xml file and s_airproof_carousel/000.scss file from our example module. 
- Record your images in images.xml. 
- Declare your files in the __manifest__.py. 
- Add it to the list of building blocks. In our example, it looks like this: - /website_airproof/views/snippets/options.xml¶- <!-- Add custom snippets to the builder --> <template id="snippets" inherit_id="website.snippets" name="Airproof - Custom Snippets"> <xpath expr="//snippets[@id='snippet_groups']/*[1]" position="before"> <t snippet-group="airproof" t-snippet="website.s_snippet_group" string="Airproof" t-thumbnail="/website_airproof/static/src/img/wbuilder/s-airproof-snippet.svg"/> </xpath> <xpath expr="//snippets[@id='snippet_structure']/*[1]" position="before"> <t t-snippet="website_airproof.s_airproof_carousel" string="Custom snippet" group="airproof"> <keywords>Carousel block</keywords> </t> </xpath> </template> 
 
- Add the option to the Website Builder. In our example, it looks like this: - /website_airproof/views/snippets/s_airproof_carousel.xml¶- <!-- Add options to snippets --> <template id="snippet_options" inherit_id="website.snippet_options" name="Airproof - Snippets Options"> <xpath expr="." position="inside"> <div data-selector=".x_bubble_item"> <!-- Bubble shadow color --> <we-button-group string="Bubble shadow"> <we-button data-select-class="x_bubble1">Blue</we-button> <we-button data-select-class="x_bubble2">Green</we-button> </we-button-group> <!-- Bubble spacing --> <we-range string="Bubble Spacing" data-select-class="mb-1|mb-2|mb-3|mb-4|mb-5"/> </div> </xpath> <xpath expr="//div[@data-js='Box'][@data-selector='section .row > div']" position="attributes"> <!-- Disable standard shadow & borders for bubbles --> <attribute name="t-attf-data-exclude" add=".x_bubble_item" separator=", "/> </xpath> </template> - Additionally, the SCSS related to the bubbles in the s_airproof_carousel/000.scss file. 
- Add your snippet to the homepage. You can find all the necessary information in the home.xml file from our example module. 
Create a new dynamic snippets template¶
Based on the Airproof design, create a custom template that you will apply to a product dynamic snippet on the homepage.
- First, create a custom template that will be added to the list of dynamic products templates. It has to include the following elements: - Add a Discover more link. 
- Add a hover effect on cards. 
- Move the navigation arrows. 
 - You will find the icons here. - See also - See reference documentation on how to create a template for dynamic snippets.   - Tip - You can verify in the Website Builder that your template appears in the list of available templates for the product dynamic snippet. 
- Then, add a product dynamic snippet with the template you just created to the homepage. - See also - See reference documentation on how to call a template. 
To complete this exercise, you need to:
- Create your snippet template. You can find all the necessary information in the options.xml file and caroussel.scss file from our example module. 
- Apply the template to a product dynamic snippet on the homepage. You can find all the necessary information in the home.xml file from our example module. 
