Add a New Appearance to the Product Content-Type for the Page Builder on Magento 2 EE
The recent addition of Page Builder (formely known as BlueFoot when it was available as a separate extension), allows for a more user-friendly way to create CMS Pages. However, when using a Products
item to generate a list of products, only one template is available. Let me show you how to create new templates and make them available on the creation form.
Short Explanation (+ Download)
Behind the scene, a Products
item from the PageBuilder is just a wrapper for the ProductsList
widget. Therefore, the process is fairly simple, we just need to:
- Tweak some JavaScript (
widget-directive.js
) to allow new templates to be used. - Create a valid .phtml file using the methods available for
ProductsList
widget (Magento\CatalogWidget\Block\Product\ProductsList). - Add a new
Appearance
with some XML.
Download the module from this article
You can download an archive of the module here. Just drop it at the root of your website and modify what you need based on the tutorial below.
Full explanation
Create a module
First thing we have to do is create a new module. Here's a link to the official documentation if you don't know how to do it.
For the example, we'll name the module Thedotwriter_PageBuilderExtensionProducts
(following the naming convention used in the documentation for extending a content type).
List of files
Here's a tree view of all files and folders we're going to create:
app
└───code
└───Thedotwriter
└───PageBuilderExtensionProducts
│ composer.json
│ registration.php
│
├───etc
│ │ module.xml
│ │
│ └───adminhtml
│ di.xml
│
└───view
├───adminhtml
│ ├───pagebuilder
│ │ └───content_type
│ │ products.xml
│ │
│ └───web
│ ├───css
│ │ └───images
│ │ └───content-type
│ │ └───products
│ │ └───appearance
│ │ menu-list-complex.svg
│ │ menu-list-simple.svg
│ │
│ └───js
│ └───content-type
│ └───products
│ └───mass-converter
│ widget-directive.js
│
└───frontend
└───templates
└───product
└───widget
└───content
widget-menu-list-complex.phtml
widget-menu-list-simple.phtml
Now, let's explain them one by one.
Add a dependency to Magento_PageBuilder
Let's make sure to load our module components after Magento_PageBuilder. For that, add a <sequence>
to your module.xml file:
etc/module.xml
:
<?xml version="1.0"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:Module/etc/module.xsd">
<module name="Thedotwriter_PageBuilderExtensionProducts" setup_version="0.1.0">
<sequence>
<module name="Magento_PageBuilder"/>
</sequence>
</module>
</config>
Create a new widget convertor
Remember, the Products
PageBuilder item is only a wrapper for the already existing ProductsList
widget ("Catalog Product List" in the backend). Therefore, the PageBuilder item configuraton needs to be passed to a ProductsList
widget at some point. That's the job of the widget-directive.js
convertor.
There's one issue with it though, the path to the phtml file is hard coded into the original file. So, we're gonna create another JS convertor that lets us set the phtml file as a setting in the products.xml file. It's quite simple (once you spent enough time looking around!):
Copy:
vendor/magento/module-page-builder/view/adminhtml/web/js/content-type/products/mass-converter/widget-directive.js
Into our module :
view/web/js/content-type/products/mass-converter/widget-directive.js
Then in the toDom()
function, change the line 53:
template: "Magento_CatalogWidget::product/widget/content/grid.phtml",
Into this:
template: config.template_path,
Create a new template file
Now, create an empty .phtml
file in the following path in our module:
view/frontend/templates/product/widget/content/
This example being based on a recent work of mine, the file is called widget-menu-list-simple.phtml
, but call it however you want. Just remember to update the name in the products.xml
file I'll make you create later. That's where the template file will be specified.
In the ZIP file provided in this article, you'll find an example of how to work with this template. But if you want more, you can check out the original grid.phtml
file in vendor/magento/module-catalog-widget/view/frontend/templates/product/widget/content/
.
Add a new Appearance
To setup a new Appearance, start by creating a products.xml
file in
view/adminhtml/pagebuilder/content_type/
. Then add the following:
<?xml version="1.0"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:module:Magento_PageBuilder:etc/content_type.xsd">
<type name="products">
<appearances>
<appearance name="menu-list-simple"
preview_template="Magento_PageBuilder/content-type/products/grid/preview"
master_template="Magento_PageBuilder/content-type/products/grid/master"
reader="Magento_PageBuilder/js/master-format/read/configurable">
<elements>
<element name="main">
<style name="text_align" source="text_align"/>
<style converter="Magento_PageBuilder/js/converter/style/border-style" name="border" source="border_style"/>
<style converter="Magento_PageBuilder/js/converter/style/color" name="border_color" source="border_color"/>
<style converter="Magento_PageBuilder/js/converter/style/border-width" name="border_width" source="border_width"/>
<style converter="Magento_PageBuilder/js/converter/style/remove-px" name="border_radius" source="border_radius"/>
<style name="display" source="display" converter="Magento_PageBuilder/js/converter/style/display" preview_converter="Magento_PageBuilder/js/converter/style/preview/display"/>
<style name="margins" storage_key="margins_and_padding" reader="Magento_PageBuilder/js/property/margins" converter="Magento_PageBuilder/js/converter/style/margins"/>
<style name="padding" storage_key="margins_and_padding" reader="Magento_PageBuilder/js/property/paddings" converter="Magento_PageBuilder/js/converter/style/paddings"/>
<attribute source="data-content-type" name="name"/>
<attribute source="data-appearance" name="appearance"/>
<html name="html" preview_converter="Magento_PageBuilder/js/converter/attribute/preview/store-id"/>
<css name="css_classes"/>
</element>
</elements>
<converters>
<converter component="Thedotwriter_PageBuilderExtensionProducts/js/content-type/products/mass-converter/widget-directive" name="widget_directive">
<config>
<item name="html_variable" value="html"/>
<item name="template_path" value="Thedotwriter_PageBuilderExtensionProducts::product/widget/content/widget-menu-list-simple.phtml"/>
</config>
</converter>
</converters>
</appearance>
</appearances>
</type>
</config>
What we've done above is copy the default "grid" <appearance>
(originally set in vendor/magento/module-page-builder/view/adminhtml/pagebuilder/content_type/products.xml), put it in our XML file and change the name of the appearance to menu-list-simple
.
Next, we update the path of the widget_directive converter to Thedotwriter_PageBuilderExtensionProducts/js/content-type/products/mass-converter/widget-directive
which will make Magento use our javascript file created earlier:
<converter component="Thedotwriter_PageBuilderExtensionProducts/js/content-type/products/mass-converter/widget-directive" name="widget_directive">
And finally, we add a new config item named "template_path" to the converter settings. That's where you specify the template you want to use:
<item name="template_path" value="Thedotwriter_PageBuilderExtensionProducts::product/widget/content/widget-menu-list-simple.phtml"/>
Make the new Appearance available in the backend
Once the Appearance is created, you have to tell Magento to make it available for the PageBuilder Products
item.
This is done by adding this to your module di.xml
file:
<?xml version="1.0"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:ObjectManager/etc/config.xsd">
<virtualType name="AppearanceSourceProducts">
<arguments>
<argument name="optionsData">
<item name="1" xsi:type="array">
<item name="value" xsi:type="string">menu-list-simple</item>
<item name="title" xsi:type="string" translate="true">Menu List Simple</item>
<item name="icon" xsi:type="string">Thedotwriter_PageBuilderExtensionProducts/css/images/content-type/products/appearance/menu-list-simple.svg</item>
</item>
</argument>
</arguments>
</virtualType>
</config>
Update the value
, title
and icon
of the item according to the appearance name, the label you want to show on the edit form and the SVG icon you want to use.
Don't forget to enable your module and clear the cache and you should be good to go!