Hooks : quick start guide
This functionality is only available since version 2.1
If you want to attach your module to some hooks, you can do this pretty easily by following these steps :
Declare hook
in the config.xml file in your module, you have to declare a hooks tag :
<?xml version="1.0" encoding="UTF-8" ?>
<config xmlns="http://thelia.net/schema/dic/config"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xsi:schemaLocation="http://thelia.net/schema/dic/config http://thelia.net/schema/dic/config/thelia-1.0.xsd">
        ....
        <hooks>
            <hook id="module.hook.front" class="Module\Hook\Front" scope="request">
                <tag name="hook.event_listener" event="product.stylesheet" />
                <tag name="hook.event_listener" event="product.after.javascript.initialization" type="front" />
                <tag name="hook.event_listener" event="product.additional" type="front" method="onProductAdditionalContents" />
            </hook>   
        </hooks>
</config>
The hook tag represents a class responsible to handle one or more hooks.
<hook id="module.hook.front" class="Module\Hook\Front" scope="request">
All attributes are mandatory. id must be unique, class is the namespace of the class in which you have your code.
The tag tag indicates a method that will handle a defined hook :
<tag name="hook.event_listener" event="product.additional" type="frontoffice" method="onProductAdditionalContents" active="0" />
Some attributes here are optionals. Attributes :
- name="hook.event_listener"must be defined as well.
- event: represents the hook code for which it wants to respond.
- type: indicate the context of the hook : frontoffice (default), front, fo, backoffice, back, bo, pdf or email.
- method: indicate the method to be called. By default, it will be based on the name of the hook. eg : for- product.additionalhook, the method- onProductAdditionalwill be called (CamelCase prefixed by on).
- active: allow you to activate the hook (set to 1 - default) or not (set to 0) when the module is installed.
Implement the class
Your class must extend Thelia\Core\Hook\BaseHook. This class provides some useful methods to generate code or to retrieve objects from the session : cart, customer, …
When a hook is called from the template, an event for this hook is created and dispatched by Thelia. If your module listens to this hook the method that you indicate in your config.xml is called with as argument the event generated.
This event could be a Thelia\Core\Event\Hook\HookRenderEvent (for hook function) or Thelia\Core\Event\Hook\HookRenderBlockEvent (for hook block).
Example of a hook function
We’re going to display a message in the product page if the module is already in the customer cart.
In config.xml :
<hooks>
    <hook id="module.hook.front" class="Module\Hook\Front" scope="request">
        <tag name="hook.event_listener" event="product.top" />
        <!--
        same as :
        <tag name="hook.event_listener" event="product.top" type="front" method="onProductTop" active="1" />
        -->
    </hook>   
</hooks>
In our class :
<?php
namespace Module\Hook;
use Thelia\Core\Event\Hook\HookRenderEvent;
use Thelia\Core\Hook\BaseHook;
/**
 * Class Front
 * @package Test\Hook
 * @author Julien Chanséaume <jchanseaume@openstudio.fr>
 */
class Front extends BaseHook {
    public function onProductTop(HookRenderEvent $event)
    {
        // retrieve the product id (argument of the smarty hook function)
        $product_id = $event->getArgument('product', null);
        // get the cart from the BaseHook class
        $cart = $this->getCart();
        if (null !== $product_id && null !== $cart){
            foreach($cart->getCartItems() as $cartItem){
                if ($cartItem->getProductId() == $product_id){
                    // Add the content to the event
                    $event->add("<p>This product is already in your cart.</p>");
                    return;
                }
            }
        }
        $event->add("<p>This product is not in your cart.</p>");
    }
}
?>
and that’s it. This is not very useful for now but you can imagine whatever you want.
Other methods like getCart are available thanks to BaseHook class :
- getOrder: the order.
- getCustomer: the customer logged in.
- getLangthe current language.
- getCurrency: the current currency.
- getView: the current view displayed (category, product, contact, …)
- getRequest: the- Symfony\Component\HttpFoundation\Requestobject that allows you to access get, post, server parameters …
You can also notice that the arguments of the smarty hook function/hook are accessible from the event itself thanks to the getArgument method. The event also contains the code (getCode) of the current event. It’s useful when several hooks point to the same method.
Example of a hook block
We’re going to had a new tab to the additional area of the product page. In this new tab we’re going to display related content to the current product.
In config.xml :
<hooks>
    <hook id="module.hook.front" class="Module\Hook\Front" scope="request">
        <tag name="hook.event_listener" event="product.additional" method="onProductAdditionalContents" />
    </hook>   
</hooks>
In our class :
<?php
namespace Module\Hook;
use Thelia\Core\Event\Hook\HookRenderBlockEvent;
use Thelia\Core\Hook\BaseHook;
use Thelia\Model\ContentQuery;
use Thelia\Model\ProductAssociatedContentQuery;
/**
 * Class Front
 * @package Test\Hook
 * @author Julien Chanséaume <jchanseaume@openstudio.fr>
 */
class Front extends BaseHook {
    public function onProductAdditionalContents(HookRenderBlockEvent $event)
    {
        $product_id = intval($event->getArgument('product', null));
        $associatedContents = ProductAssociatedContentQuery::create()->findByProductId($product_id);
        if (null !== $associatedContents){
            $html = array('<ul>');
            foreach($associatedContents as $associatedContent){
                $content = ContentQuery::create()->findPk($associatedContent->getId());
                $html[] = '<li><a href="' . $content->getUrl() . '">' . $content->getTitle() . '</a></li>';
            }
            $html[] = '</ul>';
            $event->add(array(
                "id" => "related-content",
                "title" => $this->trans("related content"),
                "content" => implode($html)
            ));
        }
    }
}
?>
Et voila ! The associated contents are now displayed in a new tab.
The associated template in product page :
    <section id="product-tabs">
        {hookblock name="product.additional" product="{product attr="id"}"}
        <ul class="nav nav-tabs" role="tablist">
            <li class="active" role="presentation"><a id="tab1" href="#description" data-toggle="tab" role="tab">{intl l="Description"}</a></li>
            {forhook rel="product.additional"}
                <li role="presentation"><a id="tab{$id}" href="#{$id}" data-toggle="tab" role="tab">{$title}</a></li>
            {/forhook}
        </ul>
        <div class="tab-content">
            <div class="tab-pane active in" id="description" itemprop="description" role="tabpanel" aria-labelledby="tab1">
                <p>{$DESCRIPTION|default:'N/A' nofilter}</p>
            </div>
            {forhook rel="product.additional"}
            <div class="tab-pane" id="{$id}" role="tabpanel" aria-labelledby="tab{$id}">
                {$content nofilter}
            </div>
            {/forhook}
        </div>
        {/hookblock}
    </section>
It’s great but we can make something easier by using a smarty template in our hook method.
Use smarty template in hooks.
The BaseHook class provides the render method which do the work for you. Its behavior is not complicated.
First, you have to put your templates inside the Template directory. The location of the templates also depends of the type (context) of the hook. If it’s a frontOffice hook then the root path of your templates will be local/modules/MyModule/templates/frontOffice/default/
You can override these smarty templates in the current template. You have to put your templates in this directory (with default template) : template/frontOffice/default/modules/MyModule/
The revisited function :
<?php
    public function onProductAdditionalContents(HookRenderBlockEvent $event)
    {
        $product_id = intval($event->getArgument('product', null));
        if (0 !== $product_id){
            $html = $this->render("related-content.html", array("product" => $product_id));
            if ("" !== $html){
                $event->add(array(
                    "id" => "related-content",
                    "title" => $this->trans("related content"),
                    "content" => $html
                ));
            }
        }
    }
?>
The template (templates/frontOffice/default/related-content.html) :
{ifloop rel="associated_content"}
<ul class="list list-content">
{loop name="associated_content" type="associated_content" product=$product }
    {loop name="content" type="content" id=$CONTENT_ID}
    <li class="list-item">
        <h4 class="list-item-title">{$TITLE}</h4>
        <p class="list-item-chapo">{$CHAPO}</p>
        <p class="list-item-more">
            <a href="{$URL}">
                {images file='img/more.png' source='MyModule'}<img src="{$asset_url}" alt="{intl l="view full article" d="mymodule.fo.default"}" />{/images}
                {intl l="view full article" d="mymodule.fo.default"}
            </a>
        </p>
    </li>
    {/loop}
{/loop}
</ul>
{/ifloop}
As you can see, you can use assets and translations in your smarty template. To keep the module isolated and independent, you have to use the concept of domain in the assets and translate functions.
For translate function (eg: intl) you should use the d attribute with a special code. You can learn more about the intl function on this page : internationalization
For assets functions (eg: image, images, stylesheets, javascripts) you should use the source attribute with your module code.
We’ve added a system of overriding for assets allowing you to redefine the asset in the template you use. For example, if you use the default template, and use the image img/more.png in your smarty template, you could override this image in your template if you put your own image in template/frontOffice/default/modules/MyModule/img/more.png
Adding CSS and JS files.
For some module, you will have to load some specific dependendies like CSS stylesheet or JavaScript.
For CSS stylesheets, you have to add the <link> tag in the <head> of our page. In order to do this, you have to subscribe to specific hook located in this area. The best hook is the main.stylesheet hook if you it’s a stylesheet you want in each pages of your site. If it’s a stylesheet just needed in few pages, you can use hook specific to page : product.stylesheet, category.stylesheet, …
For JavaScript files, you should add them at the bottom of the page and after the inclusion of the default JavaScript files of your template. The hook for global JavaScript is : main.after-javascript-include and for specific page : product.after-javascript-include, category.after-javascript-include, …
Once done, your method have to possibilities to add files to the hook event that is dispatched for these hooks :
Using the BaseHook helper functions
The BaseHook class provides 2 methods to add these files. On the first hand, we have the addCSS method for CSS Stylesheet, on the second hand the addJS method for JavaScript files.
The method supports the same system of overriding as Smarty templates.
For example, this method attached to the main.stylesheet hook
public function onMainStylesheet(HookRenderEvent $event)
{
    $content = $this->addCSS('assets/css/styles.css');
    $content .= $this->addCSS('assets/css/print.css',
                             array("media" => "print"));
    $event->add($content);
}
will render :
<head>
....
    <link rel="stylesheet" type="text/css" href="http://www.myshop.com/assets/assets/css/HookCustomer/assets/css/8a0ca85.css" />
    <link rel="stylesheet" type="text/css" href="http://www.myshop.com/assets/assets/css/HookCustomer/assets/css/8a0ca85.css" media="print" />
...
</head>
Using the Smarty assets block
The smarty assets directives {stylesheets} and {javascripts} can be used to include your module dependecies.
These directives are powerfull and support the same system of overriding. The main advantage is that you can add several files at once :
{stylesheets source="MyModule" file="assets/css/*.css"}
    <link href="{$asset_url}" rel="stylesheet" type="text/css" />
{stylesheet}  
In our method, we just have to call the render function like this :
public function onMainStylesheet(HookRenderEvent $event)
{
    $content = $this->render('main-stylessheet.html');
    $event->add($content);
}
The new module configuration
Before the hooks, modules had the opportunity to interact with backOffice template with the module_include function, especially to create a link in modules page to add a configuration page.
Now, adding shortcuts to your module is simpler as you can add entries in the main navigation bar or elsewhere if needed. You can still add a link in the modules page to the configuration page of your module in attaching your module to the module.configuration hook. The module_include module_configuration is now deprecated but still works as expected.
Improvements since Thelia 2.3
This functionality is only available since version 2.3
The use of hooks has been simplified.
Now, it’s no longer required to implement your own class to handle hooks. If you only want to render a template or add a css or js file you can do it easily. Here is an example :
<hooks>
    <hook id="hooknavigation.hook.test">
      <tag name="hook.event_listener" event="main.navbar-primary" templates="render:myTemplate.html" />
      <tag name="hook.event_listener" event="main.stylesheet" type="front" active="1" templates="css:myCss.css;render:theme.css.html" />
    </hook>
</hooks>
You can notice that the class argument has been omitted as well as the scope argument.
The tag syntax is similar and it operates the same way, except for the new argument templates. This argument is used to define what contents are going to be injected in the hook.
The syntax :
templates="<action>:<file>[;<action>:<file>]*"
- action (what to do with the file) :
    - render : this is the default one (if action is omitted). It will render the file. all parameters affected to the hook will be available in the template.
- dump : will just dump the content of the file.
- css : will create a linkhtml tag and add the CSS file.
- js : will create a scripthtml tag and add the JavaScript file.
 
- file : this is the full path of the file related to the root directory of your template directory. For example, if the hook is a front office hook and want to render the file myTemplate.html. This template should be located in your module inside thetemplates\frontOffice\default\directory. The overriding system is the same as used with classic hook.
A command has been added to cleanup hooks of a module or of all modules. They will be recreated (with default configuration) when the cache of the application will be cleared :
# remove module hooks of the HookNavigation module (without confirmation)
php Thelia hook:clean HookNavigation --assume-yes
# remove hooks of all modules (with confirmation)
php Thelia hook:clean
