Can you Create your Own Hook in Drupal?
Asked Answered
C

5

34

Is it possible to create your own hook in a Drupal module for other Drupal modules to consume? If not, is there a mechanism in Drupal for third party developers to provide hooks? If everything's been a no so far, where in the core are the list of hooks implemented?

As I understand things, Drupal modules work on a event like system called hooks. When you create a new module, you create functions that implement a hook. For example, there's a hook_delete hook. If you implement a function in your module

function mymodule_delete($node)
{
}

this function will be called whenever a node is deleted.

What I want to know is, is there a way or me, as a third party module developer, to create my own hooks. Say, something like hook_alanskickbutthook so that other module developers could subscribe to this hook.

If this is possible, how do you do it? I've looked around the official docs and haven't found much there, and I still get a little dizzy when I start poking around the Drupal source code (I understand recursion, but don't spend enough time thinking about recursive problems). Full solutions are welcome, but I'm happy to just be pointed in the right direction.

Cudbear answered 14/2, 2011 at 16:22 Comment(0)
B
37

Module_invoke_all() is your ticket to creating your own hooks:

see the API:

http://api.drupal.org/api/drupal/includes--module.inc/function/module_invoke_all

and then look at this great writeup:

http://web.archive.org/web/20101227170201/http://himerus.com/blog/himerus/creating-hooks-your-drupal-modules

(edit: was at http://himerus.com/blog/himerus/creating-hooks-your-drupal-modules but this is now gone)

Once you've made your hook, it can be called in another module using:

/**
 * Implementation of hook_myhookname()
 */

function THISMODULENAME_myhookname(args){
  //do stuff
}
Bunghole answered 14/2, 2011 at 16:28 Comment(7)
In the tutorial you posted, what's calling the "taxonomy_rockstar_generate_layouts" function? (apologies for the newbie-style question. I'm new enough to Druapl that I can't quite tell where things belong in the system).Cudbear
Some corrections/tipps: a) $args is not by reference when using module_invoke_all(). b) If you want to pass something by reference, use either drupal_alter() (forces '_alter' suffix on hook name) or module_implements + a loop and then call the implementations directly. c) The recommended way to document hooks is in a yourmodule.api.php file, see core for examples. d) A common pattern in D7 when collecting information if to use an info suffix and also call drupal_alter on that. Example <?php $stuff = module_invoke_all('yourmodule_stuff_info'); drupal_alter('yourmodule_stuff_info', $stuff); ?>.Nightfall
good catch on the reference...I was thinking of drupal_alter() when I added it, editedBunghole
@Bunghole thanks for the clarification. One more clarifying question. Where should you be calling module_invoke_all from? Am I correct that you'd implement one of the core module's early hooks and put it in there? Or something else?Cudbear
Often you are creating a hook to allow another module modify some data right before you do something with it (and you might use the hook in your own module as well). In that case I would place the module_invoke_all() right before doing something with the data. For examples check out drupal_goto and then Here is another good tutorial/example.Bunghole
first link is dead :( blog has been removed https://mcmap.net/q/451825/-what-ties-a-drupal-hook-to-a-particular-module you may want to look at this link tooHaworth
Thanks Karel-Jan, I've updated the link with the wayback machine copy of the pageBunghole
M
6

For example, say you wanted to create hook_my_custom_goodness() for others to use. Then just place code like this in your module at the point where you want the hook to happen:

$variables['msg'] = 'foo';

// Make sure at least one module implements our hook.
if (sizeof(module_implements('my_custom_goodness')) > 0) {
  // Call modules that implement the hook, and let them change $variables.
  $variables = module_invoke_all('my_custom_goodness', $variables);
}

drupal_set_message($variables['msg']); // Will display 'bar' instead.

Now, if anybody wanted to use your hook, then they could do so in their own module like this:

/**
 * Implements hook_my_custom_goodness().
 */
function SOME_OTHER_MODULE_my_custom_goodness($variables) {
  $variables['msg'] = 'bar';
  return $variables;
}

There is a more complete explanation here:

http://tylerfrankenstein.com/code/drupal-create-custom-hook-for-other-modules

Mielke answered 2/4, 2013 at 19:49 Comment(0)
Z
1

You need to implement two hooks 1. hook_token_info() & 2. hook_tokens() in your module file Below i have given code to create my custom token "query-param-all" and used that token in views- Textarea field.....

/**
* Implements hook_token_info().
*/
function mycustom_token_info() {
  $type = [
      'name' => ('Custom Token'),
      'description' => ('Tokens for custom things.'),
  ];
  $node['query-param-all'] = [
      'name' => ("Get all URL query string"),
      'description' => ('Get all URL query string'),
  ];
  return [
      'types' => ['customtoken' => $type],
      'tokens' => ['customtoken' => $node],
  ];
}

/**
* Implements hook_tokens().
*/
function mycustom_tokens($type, $tokens, array $data, array $options, \Drupal\Core\Render\BubbleableMetadata $bubbleable_metadata) {
  $replacements = [];
  //print '<pre>'; print_r($data);exit;
  $current_path = \Drupal::request()->query->all();
  $query_param = '';
  if( count($current_path) > 0) {
    $amper = '';
    $query_param = '?';
    foreach($current_path as $key => $value){
      $query_param .= $amper.$key.'='.$value;
      $amper = '&';
    }
  }

  if ($type == 'customtoken') {
      foreach ($tokens as $name => $original) {
          switch ($name) {
              case 'query-param-all':
                  $replacements[$original] = $query_param;
              break;
          }
      }
  }
  return $replacements;
}
Zinn answered 6/4, 2022 at 10:0 Comment(0)
H
0

If i recall... http://api.drupal.org/api/drupal/modules--node--node.api.php/function/hook_delete/7

does ths help? been a while since I messed with Drupal.

To create/offer custom Drupal hook, you must implement in a ways such that calling the hook with module_invoke or module_invoke_all does not make any conflicts with other module hooks. The name of the hook should be unique and it should offer all/specific feature in such a general way that it doesn't require any type of adjustments with code. All the configuration must go on admin pages and should store those configurations in a separate table or any existing tables create by Drupal or modules on which your modules depends. The hook should be easy to implment by other modules and it should not be much complex to implement. When you create custom hooks, your module(s) act(s) as API provider.

Hartzke answered 14/2, 2011 at 16:25 Comment(0)
M
0

For Drupal 6 & 7, drupal_alter() is probably the best option.

As stated in the module_invoke_all() documentation,

All arguments are passed by value. Use drupal_alter() if you need to pass arguments by reference.

In Drupal 8, use ModuleHandler::alter.

Passes alterable variables to specific hook_TYPE_alter() implementations.

Movable answered 4/5, 2016 at 20:54 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.