Ubercart: modifying checkout panes
The Ubercart module is one of the best e-commerce options for Drupal currently. It is very user friendly and highly flexible with administrators having control over the product catalogue, payment gateways and email notifications. Site administrators also have control over which checkout panes are displayed during checkout and the order in which they appear. However, as I found out recently, while it is easy to control checkout pane visibility, and even add your own, there's no simple way of modifying the forms contained within a checkout pane. This article will cover one solution on how to overcome this.
For a site I'm working on, I needed to be able to add a new field to contain the user's title (Mr, Mrs, Miss, etc) to the "billing information" ubercart checkout pane. I first implemented hook_form_alter()
to add the field and set a custom submit handler. However, while I was able to modify the form, the submit function was never called and I wasn't able to save the value of the new field to the customer's order. This is because the ubercart checkout panes don't fully utilise the FAPI.
My solution was to create a new checkout pane and to entirely recreate the billing pane by pulling in the pane contents from the ubercart module, adding my own changes and returning the merged version.
So first I created a new checkout pane by implementing the hook, hook_checkout_pane()
:
<span style="color: #000000"><span style="color: #0000BB"><?php<br></span><span style="color: #007700">function </span><span style="color: #0000BB">mymodule_checkout_pane</span><span style="color: #007700">() {<br> </span><span style="color: #FF8000">// Replacement for standard billing address pane.<br> </span><span style="color: #0000BB">$panes</span><span style="color: #007700">[] = array(<br> </span><span style="color: #DD0000">'id' </span><span style="color: #007700">=> </span><span style="color: #DD0000">'mymodule_billing'</span><span style="color: #007700">,<br> </span><span style="color: #DD0000">'callback' </span><span style="color: #007700">=> </span><span style="color: #DD0000">'mymodule_checkout_pane_mymodule_billing'</span><span style="color: #007700">,<br> </span><span style="color: #DD0000">'title' </span><span style="color: #007700">=> </span><span style="color: #0000BB">t</span><span style="color: #007700">(</span><span style="color: #DD0000">'Billing Address'</span><span style="color: #007700">),<br> </span><span style="color: #DD0000">'desc' </span><span style="color: #007700">=> </span><span style="color: #0000BB">t</span><span style="color: #007700">(</span><span style="color: #DD0000">'Custom billing address fields.'</span><span style="color: #007700">),<br> </span><span style="color: #DD0000">'weight' </span><span style="color: #007700">=> </span><span style="color: #0000BB">2</span><span style="color: #007700">,<br> </span><span style="color: #DD0000">'process' </span><span style="color: #007700">=> </span><span style="color: #0000BB">TRUE</span><span style="color: #007700">,<br> </span><span style="color: #DD0000">'collapsible' </span><span style="color: #007700">=> </span><span style="color: #0000BB">FALSE</span><span style="color: #007700">,<br> );<br> return </span><span style="color: #0000BB">$panes</span><span style="color: #007700">;<br>}<br></span><span style="color: #0000BB">?></span></span>
Each pane needs a unique id. I originally used the same id as the ubercart billing information pane in the hope that I could override it, but that didn't work unfortunately. The other two fields to pay particular attention to are callback
- the function to call to build the pane and process it, and process
- this should be set to TRUE, so your callback function is called with the 'process' operation.
For the callback function, I didn't want to just copy and paste in Ubercart's uc_checkout_pane_billing()
function as then I wouldn't be able to avail of any modifications made to the original "billing information" pane by simply upgrading the module - I would have to modify my custom module each time. So for each operation, my checkout pane calls Ubercart's one, makes my custom changes and then returns the merged result. The final result is as follows:
<span style="color: #000000"><span style="color: #0000BB"><?php<br></span><span style="color: #007700">function </span><span style="color: #0000BB">mymodule_checkout_pane_mymodule_billing</span><span style="color: #007700">(</span><span style="color: #0000BB">$op</span><span style="color: #007700">, &</span><span style="color: #0000BB">$arg1</span><span style="color: #007700">, </span><span style="color: #0000BB">$arg2</span><span style="color: #007700">) {<br> require_once(</span><span style="color: #0000BB">drupal_get_path</span><span style="color: #007700">(</span><span style="color: #DD0000">'module'</span><span style="color: #007700">, </span><span style="color: #DD0000">'uc_cart'</span><span style="color: #007700">) . </span><span style="color: #DD0000">'/uc_cart_checkout_pane.inc'</span><span style="color: #007700">);<br><br> switch (</span><span style="color: #0000BB">$op</span><span style="color: #007700">) {<br> case </span><span style="color: #DD0000">'view'</span><span style="color: #007700">:<br> </span><span style="color: #FF8000">// This is needed to avoid 'an illegal choice has been made' error.<br> </span><span style="color: #007700">if (isset(</span><span style="color: #0000BB">$_POST</span><span style="color: #007700">[</span><span style="color: #DD0000">'panes'</span><span style="color: #007700">][</span><span style="color: #DD0000">'mymodule_billing'</span><span style="color: #007700">][</span><span style="color: #DD0000">'billing_country'</span><span style="color: #007700">])) {<br> </span><span style="color: #0000BB">$_POST</span><span style="color: #007700">[</span><span style="color: #DD0000">'panes'</span><span style="color: #007700">][</span><span style="color: #DD0000">'billing'</span><span style="color: #007700">][</span><span style="color: #DD0000">'billing_country'</span><span style="color: #007700">] = </span><span style="color: #0000BB">$_POST</span><span style="color: #007700">[</span><span style="color: #DD0000">'panes'</span><span style="color: #007700">][</span><span style="color: #DD0000">'mymodule_billing'</span><span style="color: #007700">][</span><span style="color: #DD0000">'billing_country'</span><span style="color: #007700">];<br> }<br> </span><span style="color: #0000BB">$contents </span><span style="color: #007700">= </span><span style="color: #0000BB">uc_checkout_pane_billing</span><span style="color: #007700">(</span><span style="color: #0000BB">$op</span><span style="color: #007700">, </span><span style="color: #0000BB">$arg1</span><span style="color: #007700">, </span><span style="color: #0000BB">$arg2</span><span style="color: #007700">);<br><br> </span><span style="color: #FF8000">// Add 'title' or 'salutation' to billing address details.<br> </span><span style="color: #0000BB">$contents</span><span style="color: #007700">[</span><span style="color: #DD0000">'contents'</span><span style="color: #007700">][</span><span style="color: #DD0000">'billing_title'</span><span style="color: #007700">] = array(<br> </span><span style="color: #DD0000">'#type' </span><span style="color: #007700">=> </span><span style="color: #DD0000">'select'</span><span style="color: #007700">,<br> </span><span style="color: #DD0000">'#title' </span><span style="color: #007700">=> </span><span style="color: #0000BB">t</span><span style="color: #007700">(</span><span style="color: #DD0000">'Title'</span><span style="color: #007700">),<br> </span><span style="color: #DD0000">'#options' </span><span style="color: #007700">=> array(</span><span style="color: #DD0000">'Mr'</span><span style="color: #007700">, </span><span style="color: #DD0000">'Mrs'</span><span style="color: #007700">, </span><span style="color: #DD0000">'Ms'</span><span style="color: #007700">, </span><span style="color: #DD0000">'Miss'</span><span style="color: #007700">, </span><span style="color: #DD0000">'Dr'</span><span style="color: #007700">, </span><span style="color: #DD0000">'Fr'</span><span style="color: #007700">, </span><span style="color: #DD0000">'Rev'</span><span style="color: #007700">, </span><span style="color: #DD0000">'Sr'</span><span style="color: #007700">),<br> </span><span style="color: #DD0000">'#required' </span><span style="color: #007700">=> </span><span style="color: #0000BB">TRUE</span><span style="color: #007700">,<br> </span><span style="color: #DD0000">'#weight' </span><span style="color: #007700">=> </span><span style="color: #0000BB">0</span><span style="color: #007700">,<br> </span><span style="color: #DD0000">'#default_value' </span><span style="color: #007700">=> </span><span style="color: #0000BB">$arg1</span><span style="color: #007700">-></span><span style="color: #0000BB">data</span><span style="color: #007700">[</span><span style="color: #DD0000">'billing_title'</span><span style="color: #007700">],<br> );<br> </span><span style="color: #FF8000">// Address history selector doesn't work for this solution, so remove it.<br> </span><span style="color: #007700">unset(</span><span style="color: #0000BB">$contents</span><span style="color: #007700">[</span><span style="color: #DD0000">'contents'</span><span style="color: #007700">][</span><span style="color: #DD0000">'billing_address_select'</span><span style="color: #007700">]); <br><br> return </span><span style="color: #0000BB">$contents</span><span style="color: #007700">;<br><br> case </span><span style="color: #DD0000">'review'</span><span style="color: #007700">:<br> return </span><span style="color: #0000BB">uc_checkout_pane_billing</span><span style="color: #007700">(</span><span style="color: #0000BB">$op</span><span style="color: #007700">, </span><span style="color: #0000BB">$arg1</span><span style="color: #007700">, </span><span style="color: #0000BB">$arg2</span><span style="color: #007700">);<br><br> case </span><span style="color: #DD0000">'process'</span><span style="color: #007700">:<br> </span><span style="color: #0000BB">$arg1</span><span style="color: #007700">-></span><span style="color: #0000BB">billing_title </span><span style="color: #007700">= </span><span style="color: #0000BB">$arg2</span><span style="color: #007700">[</span><span style="color: #DD0000">'billing_title'</span><span style="color: #007700">]; </span><span style="color: #FF8000">// Save our custom field.<br> </span><span style="color: #007700">return </span><span style="color: #0000BB">uc_checkout_pane_billing</span><span style="color: #007700">(</span><span style="color: #0000BB">$op</span><span style="color: #007700">, </span><span style="color: #0000BB">$arg1</span><span style="color: #007700">, </span><span style="color: #0000BB">$arg2</span><span style="color: #007700">);<br> }<br>}<br></span><span style="color: #0000BB">?></span></span>