Extending Customer For Heat Clinic Tutorial
In this tutorial, we're going to cover how to extend the actual CustomerImpl object so that we can keep track of a couple of new attributes directly on the object. We're going to create some variables to keep track of the average heat level of sauces that a customer buys so that we can display some special text for our hottest sauce eaters.
Extending CustomerImpl
There are a few steps to extending an entity in Broadleaf:
- Create the extension class
- Notify the persistence unit about your new class
- Set this class as the default for the parent interface in the Broadleaf Entity Configuration
Create the extension class
Since customers are shared between the admin application and the site application, we want to create our objects in the core submodule so that both admin and site can share them.
Let's begin by creating our interface that will extend Customer:
public interface HCCustomer extends Customer {
    public Integer getAverageHeatRatingBought();
    public Integer getNumSaucesBought();
    public void setNumSaucesBought(Integer numSaucesBought);
    public Integer getTotalHeatRating();
    public void setTotalHeatRating(Integer totalHeatRating);
}
and our implementation of that interface:
@Entity
@Table(name = "HC_CUSTOMER")
@PrimaryKeyJoinColumn(name = "CUSTOMER_ID")
public class HCCustomerImpl extends CustomerImpl implements HCCustomer {
    private static final long serialVersionUID = 6545097668293683751L;
    @Column(name = "NUM_SAUCES_BOUGHT")
    protected Integer numSaucesBought = 0;
    @Column(name = "TOTAL_HEAT_RATING")
    protected Integer totalHeatRating = 0;
    @Override
    public Integer getAverageHeatRatingBought() {
        if (numSaucesBought == null || numSaucesBought == 0) {
            return 0;
        }
        return totalHeatRating / numSaucesBought;
    }
    @Override
    public Integer getNumSaucesBought() {
        return numSaucesBought;
    }
    @Override
    public void setNumSaucesBought(Integer numSaucesBought) {
        this.numSaucesBought = numSaucesBought;
    }
    @Override
    public Integer getTotalHeatRating() {
        return totalHeatRating;
    }
    @Override
    public void setTotalHeatRating(Integer totalHeatRating) {
        this.totalHeatRating = totalHeatRating;
    }
}
Note that these two objects should be in the core submodule. I chose the package
com.mycompany.profile.core.domainfor them.
Notify the persistence unit
In core/src/main/resources/META-INF/persistence-core.xml, add the following line to the blPU 
<class>com.mycompany.profile.core.domain.HCCustomerImpl</class>
which will make it look like this:
<persistence-unit name="blPU" transaction-type="RESOURCE_LOCAL">
    <non-jta-data-source>jdbc/web</non-jta-data-source>
    <class>com.mycompany.profile.core.domain.HCCustomerImpl</class>
    <exclude-unlisted-classes/>
</persistence-unit>
Set the default Entity Configuration
We now need to instruct the EntityConfiguration to load our new overriden Customer class when it wants to create a Customer. The first thing to do is create a new file to hold our new entity configurations. Again, since we want this to be shared between admin and site, the most appropriate place to put the new file is in core.
I chose the path core/src/main/resources/applicationContext-entity.xml. Here are the contents of the file:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:jee="http://www.springframework.org/schema/jee"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xmlns:tx="http://www.springframework.org/schema/tx"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="
        http://www.springframework.org/schema/jee
        http://www.springframework.org/schema/jee/spring-jee-3.2.xsd
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context-3.2.xsd
        http://www.springframework.org/schema/tx
        http://www.springframework.org/schema/tx/spring-tx-3.2.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop-3.2.xsd">
        <!-- This file will contain bean overrides for extensions of Broadleaf entities -->
        <!-- For example, if you have a custom class that extends CustomerImpl, you would note that here -->
        <bean id="org.broadleafcommerce.profile.core.domain.Customer"
            class="com.heatclinic.profile.core.domain.HCCustomerImpl"
            scope="prototype"/>
</beans>
Note: If you're using our bundled workspace, we already reference the applicationContext-entity file. You can see this definition in applicationContext.xml.
And there we go! Whenever we create a new Customer via Broadleaf (for example: customerService.createCustomer(), an HCCustomerImpl instance will be generated instead of the typical CustomerImpl. This instance obviously has the extra fields that we added.
Subsequent modifications to other Broadleaf entities would be handled in the same way.
Displaying the new average heat level
Let's show the customer their average hot sauce heat level! For our example, we'll show them this information in the header bar, next to their name.
The welcome message is in site/src/main/webapp/templates/partials/header.html.
All we need to do is change this line:
<span><span th:text="#{home.welcome}">Welcome</span>, <a class="my-account" th:href="@{/account}" th:text="${customer.firstName}"></a></span>
to
<span><span th:text="#{home.welcome}">Welcome</span>, <a class="my-account" th:href="@{/account}" th:text="${customer.firstName}"></a><span th:text=" (${customer.averageHeatRatingBought})"></span></span>
and we'll be in business. Yep, it's really that easy!
Of course, right now, all customers would have an average heat rating of 0 since we're not saving information anywhere. That's demonstrated in our next tutorial, hooking into the order submit workflow, so go check it out!
