Drupal Module Development Tutorial

On my Drupal 7 site here, I wanted to create a method of automatically injecting my Google Ad banners on all content pages before the Disqus comments thread. You could do this manually, say by adding a text field to your basic page content_type with a default value of all the Google Ads JavaScript, but this is clumsy: it clutters up your content pages, and it doesn’t retroactively update any existing pages. No, this is a good time to make use of Drupal’s ability to modify content dynamically using its hooks API in a simple custom module.

We should note here that you control the order of elements on a Drupal node by setting the “Weight” property (an integer typically ranging from -100 to 100) of each piece of content; the higher the weight, the lower the priority, and the further down the page it’s likely to appear. A field or node with a value of 0 will appear before 25, which will appear before 30, and so forth. Disqus comments in Drupal are injected into the main content area by the Disqus module with weight = 50.

Furthermore, hook_node_view() is the Drupal 7 API hook that makes it possible to modify the properties and contents of nodes before they’re displayed.

But first of all, we need to ensure our module is recognized by Drupal 7. So we create a directory in /sites/all/modules/ — I called it ‘googleads’ — and then inside that directory we create a googleads.info file:

name = Google Ad Injector
description = "Inject the Google Ads banners at the bottom of the content area."
core = 7.x

files[] = googleads.module
configure = admin/config/content/googleads

The entries here are all important. The “name” and “description” provide a meaningful entry in the Modules listing (admin/modules), “core” tells Drupal 7 your module is compatible, and adding a “configure” directive gives you a “Configure” link in the modules listing. Finally, the “files” directive tells Drupal which other files comprise the module.

Now we should build the googleads.module file itself. We’ll get started by writing the code to modify the content pages:

/**
 * Implements hook_node_view().
 */
function googleads_node_view($node, $view_mode, $langcode) {
  $google_ad = '<!-- Google ads -->
                <script async src="//pagead2.googlesyndication.com/pagead/js/adsbygoogle.js"></script>
                rest of google ad code';

  // inject google_ad into content
  if ($view_mode == 'full' || ($view_mode == 'teaser' && variable_get('googleads_teasers', 0) == 1)) {
       $node->content['google_ads'] = array(
         '#markup' => $google_ad,
         '#weight' => variable_get('googleads_weight', 50)
       );
  }
}

This uses Drupal’s hook_node_view() hook to set up a method of modifying every node’s content before it’s rendered by render($page[‘content’]) in page.tpl.php. When writing custom implementations of hooks, you should name them after your module, both for clarity and to avoid PHP’s cannot redeclare error. So in this case the function is called googleads_node_view(), and I’ve introduced a couple of custom variables: ‘googleads_weight’ and ‘googleads_teasers’. The ‘googleads_weight’ variable gives the user the option of defining a custom weight, giving them some control over where the ads appear on the page. Meanwhile, calling hook_node_view() will modify every instance of every node, but I don’t necessarily want ads appearing in teasers like my category view: so this is user-definable in googleads_teasers. We’ll look at how to set these options in a moment.

It’s worth noting that there are other view modes besides ‘full’ and ‘teaser’. You can certainly make these configurable as well, although I saw no need.

Also, Drupal’s variable_get() function here is important: it checks Drupal’s $conf global variables array for googleads_teasers and googleads_weight, and if neither variable is found, variable_get() supplies default values (0 and 50). If the weight and teaser options have been set by the site administrator, they’d already exist in the $conf array when referenced by our module, and we want to honor those values.

Now, let’s set up an administration form so a site admin can easily define the weight and the appearance of ads on teaser pages. We add some additional code to googleads.module:

/**
 * Implements hook_menu().
 */
function googleads_menu() {
  $items = array();

  $items['admin/config/content/googleads'] = array(
     'title' => 'Google Ads',
     'description' => 'Configuration for the Google Ads injector',
     'page callback' => 'drupal_get_form',
     'page arguments' => array('googleads_form'),
     'access arguments' => array('access administration pages'),
     'type' => MENU_NORMAL_ITEM,
  );

  return $items;
}

In a large module with thousands of lines of code, it’s wise to begin separating out the different components: the admin page(s) are frequently placed in a separate file called admin.inc.php, which you’d then have to specify in googleads.info. But this is a small module, so I’m leaving them in googleads.module.

hook_menu() is important for several reasons: it specifies an URL for the administration page, provides a title and description to appear in the administration menu, calls the drupal_get_form() function to render the admin form, and sets the permission level needed to view the form. We want to ensure only a site admin can access the admin page. Next, we’ll set up the admin form itself:

/**
 * Page callback: Google Ads settings
 *
 * @see googleads_menu()
 */
function googleads_form($form, &$form_state) {

  $form['googleads_weight'] = array(
     '#type' => 'select',
     '#title' => t('Weight'),
     '#description' => t('When the ads are shown in the content area, you can set the position at which they will be shown.'),
     '#options' => drupal_map_assoc(array(0, 10, 20, 30, 40, 50)),
     '#default_value' => variable_get('googleads_weight', 50)
  );

  $form['googleads_teasers'] = array(
     '#type' => 'checkbox',
     '#title' => t('Teasers?'),
     '#description' => t('Select this option to show Google Ads on teaser pages as well as full content pages.'),
     '#default_value' => variable_get('googleads_teasers', 0)
  );

  return system_settings_form($form);
}

This is pretty self-explanatory: you’re using hook_form() to create the form widgets and specify the values that will be submitted (googleads_weight, googleads_teasers). In this case, we get a select box with the values 10, 20, 30, 40, 50 for the weight, and a checkbox to toggle the teaser ads on and all (values 0 and 1). hook_form() also provides a basic submit button.

Finally, you want to validate the form data before it goes into your database:

/**
 * Implements validation from the Form API.
 *
 * @param $form
 *   A structured array containing the elements and properties of the form.
 * @param $form_state
 *   An array that stores information about the form's current state
 *   during processing.
 */
function googleads_form_validate($form, &$form_state){
  $gads_weight = $form_state['values']['googleads_weight'];
  if (!is_numeric($gads_weight)){
    form_set_error('current_pos', t('You must enter a number for the weight!'));
  }
  elseif ($gads_weight < -50){
    form_set_error('googleads_weight', t('The weight must be greater than or equal to -50.'));
  }
  elseif ($gads_weight > 50) {
    form_set_error('googleads_weight'. t('The weight must be less than or equal to 50.'));
  }
  
  $gads_teaser = $form_state['values']['googleads_teasers'];
  if ($gads_teaser != 0 && $gads_teaser != 1){
    form_set_error('current_pos', t('The teasers option should be yes or no!'));
  }
}

Using _form_validate() here. It might seem unnecessary to validate a form that doesn’t permit users to do much: they can only select options from a drop-down and toggle a checkbox. But it’s good practice never to trust user-submitted values. Who knows, maybe some joker is using the Tamper Data Firefox plugin to change the form submission data.

The final googleads.module file looks like this:

<?php

/**
 * @file
 * A module that displays Google ad banners in the node content.
 */

/**
 * Implements hook_node_view().
 */
function googleads_node_view($node, $view_mode, $langcode) {
  $google_ad = '<!-- Google ads -->
                <script async src="//pagead2.googlesyndication.com/pagead/js/adsbygoogle.js"></script>
               more google ad code';

  // inject google_ad into content
  if ($view_mode == 'full' || ($view_mode == 'teaser' && variable_get('googleads_teasers', 0) == 1)) {
       $node->content['google_ads'] = array(
         '#markup' => $google_ad,
         '#weight' => variable_get('googleads_weight', 50)
       );
  }
}

/**
 * Implements hook_menu().
 */
function googleads_menu() {
  $items = array();

  $items['admin/config/content/googleads'] = array(
     'title' => 'Google Ads',
     'description' => 'Configuration for the Google Ads injector.',
     'page callback' => 'drupal_get_form',
     'page arguments' => array('googleads_form'),
     'access arguments' => array('access administration pages'),
     'type' => MENU_NORMAL_ITEM,
  );
  return $items;
}

/**
 * Page callback: Google Ads settings
 *
 * @see googleads_menu()
 */
function googleads_form($form, &$form_state) {
  $form['googleads_weight'] = array(
     '#type' => 'select',
     '#title' => t('Weight'),
     '#description' => t('When the ads are shown in the content area, you can set the position at which they will be shown.'),
     '#options' => drupal_map_assoc(array(0, 10, 20, 30, 40, 50)),
     '#default_value' => variable_get('googleads_weight', 50)
  );

  $form['googleads_teasers'] = array(
     '#type' => 'checkbox',
     '#title' => t('Teasers?'),
     '#description' => t('Select this option to show Google Ads on teaser pages as well as full content pages.'),
     '#default_value' => variable_get('googleads_teasers', 0)
  );
  return system_settings_form($form);
}

/**
 * Implements validation from the Form API.
 *
 * @param $form
 *   A structured array containing the elements and properties of the form.
 * @param $form_state
 *   An array that stores information about the form's current state
 *   during processing.
 */
function googleads_form_validate($form, &$form_state){
  $gads_weight = $form_state['values']['googleads_weight'];
  if (!is_numeric($gads_weight)){
    form_set_error('current_pos', t('You must enter a number for the weight!'));
  }
  elseif ($gads_weight < -50){
    form_set_error('googleads_weight', t('The weight must be greater than or equal to -50.'));
  }
  elseif ($gads_weight > 50) {
    form_set_error('googleads_weight'. t('The weight must be less than or equal to 50.'));
  }
  $gads_teaser = $form_state['values']['googleads_teasers'];
  if ($gads_teaser != 0 && $gads_teaser != 1){
    form_set_error('current_pos', t('The teasers option should be yes or no (1, 0)!'));
  }
}

?>

So there it is. Now we have a simple Google ads injector that allows us to set the weight and select whether or not your ads appear in the teaser for each page. If you save your module file, and flush your Drupal caches, you’ll have a “Google Ad Injector” module in your modules list that you can toggle on and off. If you inspect the variables table in your database after submitting the form for the first time, you’ll see that ‘googleads_weight’ and ‘googleads_teasers’ entries are now extant there.

If I wanted to distribute this module publicly, I’d add a few additional features:

  • Make the Google ads code a text field in the admin form so users could paste in their own Google ads code
  • Provide additional options to configure ads for RSS feeds and Search results ($view_mode == ‘rss’ or ‘search’)
  • Provide the option to make it a block: thus you could insert ads in the header, footer, sidebars, and so on
  • Provide a googleads.install file so your module can easily be added to, removed from, and updated on other web sites

Loading

Leave a Reply

Your email address will not be published. Required fields are marked *