Creating an Observer Class: send a specialized additional email during Create Account
by
, 7 Sep 2012 at 04:42 AM (130229 Views)
Task:
- Storeowner wants to send the customer an email with a unique code when the customer creates an account.
- The code pattern is "ABC00001", where ABC is fixed text, and 00001 is the customer number, with leading zeros up to exactly 5 digits.
- There is no need to build a database of already-issued codes, since the customer ID number is already unique per customer.
- The store is using v1.5.0 code.
The thinking and coding process:
First, since this must happen during account-creation, it makes sense to consider looking at the /includes/modules/create_account.php script which handles those activities.
Since the intention is to see whether an Observer class can be used for this, I started by scanning the file for any notifier hooks. These can be identified in either of two ways: calls to $zco_notifier->notify() for procedural code or $this->notify() from inside any class method.
We want a notifier point that fires sometime after a customer has submitted their user information and it's been validated.
So, at the end of the validation code there's this one:
… but we're not interested in a failure case, so we keep looking:Code:$zco_notifier->notify('NOTIFY_FAILURE_DURING_CREATE_ACCOUNT');
The next one is this:
And, looking closely at the code immediately above it, we see that that part of the code has just finished building the new customer data for the database, inserted that record, and set the customer ID into the session, effectively logging them in. This is perfect! And what's even better is that the $sql_data_array has also been passed, so we can make use of that to customize the email with the customer's name and email address.Code:$zco_notifier->notify('NOTIFY_MODULE_CREATE_ACCOUNT_ADDED_CUSTOMER_RECORD', array_merge(array('customer_id' => $_SESSION['customer_id']), $sql_data_array));
So, now we start with creating an observer class. We start with the basic structure:
You'll see here that we've given the class a name of send_extra_email_at_welcome. This name will be used later in the auto_loader to activate this observer class. The name itself just needs to be unique from any other class used in the store.PHP Code:
<?php
/**
* Observer class for sending an extra email during customer-creation/welcome process
*/
class send_extra_email_at_welcome extends base {
function __construct()
{
}
function update(&$class, $eventID, $paramsArray = array())
{
}
}
Next we need to set up the constructor to watch the NOTIFY_MODULE_CREATE_ACCOUNT_ADDED_CUSTOMER_RECORD hook we identified earlier. Since the notifier hook is activated via a $zco_notifier call, we set the observer up to attach to that $zco_notifier->notify() call:
We have a few main tasks to perform when this notifier hook is called:Code:function __construct() { global $zco_notifier; $zco_notifier->attach($this, array('NOTIFY_MODULE_CREATE_ACCOUNT_ADDED_CUSTOMER_RECORD')); }
a) create a formatted "code" based on the customer number
b) assemble email text with the code
c) send the email
Let's look at these one at a time:
a) create the formatted code from the customer number. We'll use the PHP function str_pad() to add enough zeros to make a 5-digit value from the customer number:
and then add the ABC prefix to the beginning of it:PHP Code:
/**
* Here we take the customer number, and add enough zeros to the left of it to make it be 5 digits long.
*/
$code = str_pad($_SESSION['customer_id'],5,'0',STR_PAD_LEFT);
b) Put together some email text, using the code. The first line here is the plain-text content. The $html_msg array will contain the same content, but formatted as HTML, in case the customer has opted to receive html-formatted messages in their preferences during signup.PHP Code:
/**
* Then the 'ABC' prefix is added to it:
*/
$code = 'ABC' . $code;
We'll need a subject line for the email:PHP Code:
$email_text = 'Here is your unique code: ' . $code;
$html_msg['EMAIL_MESSAGE_HTML'] .= nl2br($email_text);
c) To send the email, we need to get the customer's email address. The call to $zco_notifier->notify() included some extra parameters, which show up in this observer class via the $paramsArray variable:PHP Code:
$email_subject = 'Your unique customer code!';
$html_msg['EMAIL_SUBJECT'] = $email_subject;
Similarly, in this case we happen to also have access to the customer's firstname/lastname, based on whatever they supplied during signup. Here we concatenate both values into a single string:PHP Code:
$customer_email_address = $paramsArray['customers_email_address'];
And now we send the email, using all the data we've prepared:PHP Code:
$customer_name = $paramsArray['customers_firstname'] . ' ' . $paramsArray['customers_lastname'];
When we put it all together, the finished code looks like this:PHP Code:
zen_mail($customer_name, $customer_email_address, $email_subject, $email_text, STORE_NAME, EMAIL_FROM, $html_msg);
We'll store this code into a new file, named /includes/classes/observers/class.send_extra_email_at_welcome.phpPHP Code:
<?php
/**
* Observer class for sending an extra email during customer-creation/welcome process
*/
class send_extra_email_at_welcome extends base {
function __construct()
{
global $zco_notifier;
$zco_notifier->attach($this, array('NOTIFY_MODULE_CREATE_ACCOUNT_ADDED_CUSTOMER_RECORD'));
}
function update(&$class, $eventID, $paramsArray = array())
{
/**
* Here we take the customer number, and add enough zeros to the left of it to make it be 5 digits long.
*/
$code = str_pad($_SESSION['customer_id'],5,'0',STR_PAD_LEFT);
/**
* Then the 'ABC' prefix is added to it:
*/
$code = 'ABC' . $code;
/**
* Now build and send the email to the customer
*/
$email_text = 'Here is your unique code: ' . $code;
$html_msg['EMAIL_MESSAGE_HTML'] .= nl2br($email_text);
$email_subject = 'Your unique customer code!';
$html_msg['EMAIL_SUBJECT'] = $email_subject;
$customer_name = $paramsArray['customers_firstname'] . ' ' . $paramsArray['customers_lastname'];
$customer_email_address = $paramsArray['customers_email_address'];
zen_mail($customer_name, $customer_email_address, $email_subject, $email_text, STORE_NAME, EMAIL_FROM, $html_msg);
}
}
Finally, to turn the whole thing on, we add the auto_loader code to tell Zen Cart to load the observer class and instantiate it. The file will be uploaded to: /includes/auto_loaders/config.send_extra_email_at_welcome.php :
The value of 190 is used to essentially process all of this at the end of the normal autoloader stack.PHP Code:
<?php
$autoLoadConfig[190][] = array('autoType'=>'class',
'loadFile'=>'observers/class.send_extra_email_at_welcome.php');
$autoLoadConfig[190][] = array('autoType'=>'classInstantiate',
'className'=>'send_extra_email_at_welcome',
'objectName'=>'send_extra_email_at_welcome');
The load file points to the class file created earlier. And the className matches the class name in the class file we made, and the objectName is basically the instantiation object built to fire the observer when appropriate … it usually is just the same as the className.
That's it. With both of those files in place, when the customer creates a new account, they will receive an extra email (in addition to the normal Welcome email) in which they are told their unique customer code.
----------
What about making it multi-language capable?
To do this, we must change the language text into defined constants.
So, we'll take these two lines:
and make language defines for them:Code:$email_text = 'Here is your unique code: ' . $code; $email_subject = 'Your unique customer code!';
and change the code to use those constants in place of the text.Code:define('EMAIL_SUBJECT_EXTRA_EMAIL_WELCOME', 'Your unique customer code!'); define('EMAIL_TEXT_EXTRA_EMAIL_MESSAGE', 'Here is your unique code: ');
Now those define statements can be placed into new files in corresponding language folders so they can be translated to suit whatever language the customer has selected while visiting the site. The define statements would go into a file at /includes/languages/NAME_OF_A_LANGUAGE_HERE/extra_definitions/send_extra_email_at_welcome.php.Code:$email_text = EMAIL_TEXT_EXTRA_EMAIL_MESSAGE . $code; $email_subject = EMAIL_SUBJECT_EXTRA_EMAIL_WELCOME;
This blog post was inspired from the more specific use-case discussed on the forum at: http://www.zen-cart.com/showthread.p...01#post1148601