WordPress 插件开发-动作钩子 ( actions hooks )

WordPress 主要有两种类型的钩子:

  • 动作钩子(action hooks) 使得你可以在一个特定时刻上执行一个函数
  • 过滤器钩子(filter hooks )使得你可以操作通过钩子的输出。

钩子不是仅仅针对插件的。WordPress 内部也使用钩子。如果你浏览核心代码,你就能见到很多 WordPress 自己钩自己的例子了。

动作钩子 ( actions )

动作钩子让你可以在 WordPress 加载过程中或者当某个事件发生的特定时刻触发一个函数。例如:你可能希望当 WordPress 第一次加载一个页面或者保存一篇文章时执行一个函数。

你需要理解 do_action() 函数。当钩进 WordPress 中时,你的插件不会直接调用这个函数;但是你的插件几乎都会间接的使用它。

<?php
  do_action($tag,$arg= '');
?>
  • $tag — 动作钩子的名称。
  • $arg — 传递给已注册的动作的值。它看起来像一个单独的参数,但通常都不是这样的。动作钩子可以传递任何个数的参数,或者根本不传参数。对特殊的钩子,你需要查看 WordPress 的源码,因为参数个数在每个钩子的基础改变。

下面是一个具有多参数的动作钩子的例子:

<?php
  do_action($tag,$arg_1,$arg_2,$arg_3);
?>

下面看看一个叫做 wp_head 的钩子是怎么出现在 WordPress 中的。

这个钩子出现在前台的 <head> 里面。 WordPress 和插件经常用这个钩子来添加 meta 信息,样式表,和 js 脚本。

<?php
  do_action('wp_head');
?>

当这段代码在 WordPress 中执行时,它会寻找任何为 wp_head 动作钩子注册的动作。然后按照特殊顺序执行它们。如你所见,它名叫 wp_head 但是没有传递额外的参数。

动作钩子通常都这样,下面是有两个额外参数的动作钩子的例子:

<?php
  do_action('save_post',$post_ID, $post);
?>

这个 save_post 的钩子传递两个参数,一个 $post_ID,一个 $post。

什么是动作?

从技术上讲,一个动作就是一个 PHP 函数。一个函数要成为一个动作,它需要被注册成一个动作钩子。在上面的部分可以看到什么是动作钩子,但是动作钩子要用作任何目的,需要有个动作为它们注册。这就是插件的来源。你开发的自定义的函数(动作)当动作钩子出发之后就可以执行一个特定任务了。要实现这个,就要使用 add_action() 函数。

<?php
add_action( $tag, $function,$priority,$accepted_args );
?>
  • $tag – 你的函数执行时代动作钩子的名称。
  • $function – WordPress 要调用的函数名。
  • $priority – 一个表示动作调用顺序的整数,默认是10。数字越小,这个函数越早被调用
  • $accepted_args – 动作钩子要传递给你的函数的参数个数。默认只有一个参数。

动作钩子并不局限于单个动作。你的插件可以将多个函数添加到一个动作钩子上。其他插件,甚至是 WordPress 核心,经常将多个函数添加到同一个钩子上。

现在是使用动作钩子的时候了。一个常见的动作钩子是 wp_footer 。它提供给前端用户的 WordPress 模板使用。通常它刚好在</body>前调用。在下面的例子中,将要为 wp_footer 注册一个动作并添加一条自定义信息到 footer。

<?php
add_action( 'wp_footer', 'boj_example_footer_message', 100 );
function boj_example_footer_message() {
    echo"基于 <a href="http://wordpress.org" >WordPress </a>架设。;
}
?>

仔细看看上面的代码中如何使用 add_action() 函数的。

<?php
add_action( 'wp_footer', 'boj_example_footer_message', 100 );
?>
  • 第一个参数是钩子的名字( wp_footer )。
  • 第二个参数是要回调的函数 ( boj_example_footer_message )。
  • 第三个参数是优先级 ( 100 )。这个函数比起其他钩到 wp_footer 中的函数,会在比较靠后的次序执行。如果设置成1,就较先执行。

要说明,钩子可能会因为许多原因在 WordPress 执行过程中多次触发。任何添加到这个钩子中的动作每当钩子触发时都会执行一次。

动作钩子函数

你已经知道了最基本的两个动作钩子函数怎么使用;do_action() 和 add_action()。WordPress 还有其他类型的与动作钩子相关的函数用于插件开发。

do_action_ref_array() 函数和 do_action() 函数完成同样的工作,只不过参数传递的方式不同。它并不传递通过附加参数的值来确定的多个参数,而是传递一个包含参数的数组。这个参数数组也是一个必须的参数。这个函数的目的是通过引用传递一个对象给添加到特定钩子的动作(函数)。这意味着这个函数可以不用返回就改变对象本身。

<?php
do_action_ref_arrray($tag,$args);
?>

 

  • $tag – 动作钩子的名字。
  • $args – 要传递给注册到这个钩子的函数的参数的数组。通常,这是一个动作可以改变的对象。

下面看一个 WordPress 如何调用 do_action_ref_array() 的实例。下面的代码展示了 pre_get_posts 动作钩子。WordPress 在从数据库取得 posts 之前执行这个钩子,使得插件可以改变查询 posts 的方式。

<?php
do_action_ref_array('pre_get_posts',array( &$this) );
?>
  • 第一个参数 pre_get_posts 是钩子的名字。
  • 第二个参数是从数据库中查询 posts 的参数的数组。

这个钩子使你可以执行基于那个参数数组的代码。假如你想安装随机的顺序来得到首页的 blog,而不是默认的通过发布时间来得到。你就需要注册一个动作到这个钩子,并改变排序顺序。

<?php
add_action( 'pre_get_posts', 'boj_randomly_order_blog_posts');
function boj_randomly_order_blog_posts( $query ) {
    if($query-> is_home &&empty($query-> query_vars['suppress_filters']))
        $query-> set('rderby','rand');
}
?>

remove_action() 可以删除先前添加到一个钩子的动作。代表性的,你可以删除 WordPress 默认添加的动作。要删除一个动作,这个动作必须已经用 add_action() 函数添加了。如果你在动作注册之前执行 remove_action(),那么动作并不会被从钩子中删除。

如果动作被成功删除,则函数返回 true,否则返回 false。

<?php
remove_action( $tag, $function_to_remove,$priority,$accepted_args);
?>

参数类似于 do_action()。要成功的从一个钩子中删除一个动作, $tag, $function_to_remove, 和 $priority 必须完全的符合 do_action() 中使用的参数。否则动作不会被溢出,同时 remove_action() 返回 flase。

我们看一个叫做 rel_canonical 的 WordPress 默认动作。这个动作在 <head> 和 </head> 元素之间添加一个 canonical 链接。

<?php
add_action( 'wp_head', 'rel_canonical');
?>

要删除这个动作,就要使用 remove_action() 函数。你需要定义 $tag 和 $function_to_rmove 参数。这里你不用添加 $priority 因为先前定义动作的时候没有明确指定优先级。

<?php
remove_action( 'wp_head', 'rel_canonical');
?>

WordPress、plugin 或者 theme 添加的任何动作都可以在插件中删除。通常只删除 WordPress 添加的动作。许多默认的动作都定义在 wp-includes/default-filters.php 文件中。通过浏览这个文件你就会明白 WordPress 是如何使用动作钩子的。

remove_all_actions

在有些插件中,可能需要删除所有特定 tag 或者 特定 tag + 特定优先级的所有钩子。使用 remove_all_actions() 可以一次删除所有符合条件的动作。

<?php
remove_all_actions($tag,$priority);
?>

$priority 参数是可选的,默认是 false。如果你设置了这个参数,那么只有这个优先级的动作会被删除。下面的例子从 wp_head 动作钩子中删除不管任何优先级的动作。

<?php
remove_all_actions('wp_head');
?>

在使用这个函数的时候必须要小心。其他 plugin 或者 theme 可能添加了你不知道动作。这就可能破坏插件应有的功能。通常应该保持你的代码尽可能的特殊。在大多数情况下,你应该使用 remove_action() 函数来代替。

有的时候需要确定一个钩子是否包含一些动作,或者一个特定的动作是否已经添加到了钩子里面。has_action() 提供了这些功能。

<?php
has_action( $tag, $function_to_check);
?>

has_action() 函数的返回值是 Boolean 或者 一个整型值。如果 $function_to_check 参数为空,那么如果有动作已经添加到了钩子中就返回 true,反之,返回 false。而如果 $function_to_check 设置了,而且这个函数已经添加到了钩子里面,则返回该动作的优先级,否则返回 false。

下面的例子中,根据 wp_footer 动作钩子中是否有注册的动作来确定显示的信息。

<?php
if( has_action('wp_footer') )
    echo'<p> footer 中已经注册有动作了。</p>';
else
    echo'<p> footer 中还没有注册动作。</p>';
?>

下面看一个 WordPress 核心添加到 wp_footer 中的动作。wp_print_footer_script() 默认注册给这个钩子。

<?php
add_action( 'wp_footer', 'wp_print_footer_scripts');
?>

did_action() 使你的插件可以检查一个动作钩子是否已经被执行,或者记录执行的次数。这也意味着这一次页面的加载过程中有些动作被执行了多次。

<?php
did_action( $tag );
?>

这个参数返回动作已经执行的次数,如果还未执行,返回 false。这个函数的一般用途是判断一个动作钩子是否已经被触发,并执行基于 did_action() 的返回值的代码。

下面的例子中,如果 plugins_loaded 动作钩子已经被触发,就定义一个 PHP 常量。

<?php
if ( did_action( 'plugins_loaded' ) )
    define('BOJ_MYPLUGIN_READY',true );
?>

register_activation_hook 和 register_deactivation_hook

  • 插件 activation 函数当一个插件在 WordPress 中”activated(启用)”时被触发,这个函数叫做 register_activation_hook()。
  • 函数在插件被禁用时触发,这个函数叫做register_deactivation_hook() 函数。

常用的动作钩子

WordPress 有许多动作钩子,有一些是很常用的。

plugins_loaded

对插件开发者来说,plugins_loaded 动作钩子也许是最重要的动作钩子了。它在大多数 WordPress 文件加载完成之后,并在pluggable 函数和 WordPress 开始执行任何东西之前触发。在大多数的插件中,在这个钩子触发之前,不应该执行其他的代码。

plugins_loaded在所有用户启用的插件都被 WordPress 加载之后执行。这也是在加载过程中插件开发这最早能用到的钩子。

WordPress 的插件应该在这个钩子中执行安装。其他动作也应该添加到这个钩子的回调函数中。

下面的例子中,使用前面部分创建的 boj_example_footer_message 动作。要把它添加到钩到 plugins_loaded 钩子中的安装动作中,而不是单独调用它。

<?php
add_action( 'plugins_loaded', 'boj_footer_message_plugin_setup' );
function boj_footer_message_plugin_setup() {
    /* 添加 footer 信息动作 */
    add_action('wp_footer','boj_example_footer_message', 100 );
}
 
function boj_example_footer_message() {
    echo"基于 <a href="http://wordpress.org" >WordPress </a>架设。;
}
?>

创建一个安装函数并把它钩到 plugins_loaded 中。这样做就可以确保不会由于特定的 WordPress 函数还没有加载而触发错误。

init

init 钩子在大多数的 WordPress 都建立之后。WordPress 同样添加许多内部的功能到这个钩子中,例如 post types 和 taxonomies 的厨厕以及默认 widgets 的初始化。

因为这时几乎 WordPress 中的所有内容都就绪了,当 WordPress 的所有信息都可用时,你的插件使用这个钩子差不多可以做任何需要的事情了。

下面的例子中,为用户添加了给 pages 写摘要的功能。这应该在 init 中执行,因为 “page” post type 在这时使用 add_post_type_support() 函数来创建。( 详见 Part-11, “扩展 posts”)

<?php
add_action( 'init', 'boj_add_excerpts_to_pages');
 
function boj_add_excerpts_to_pages() {
    add_post_type_support('page',array('excerpt') );
}
?>

admin_menu

admin_menu 钩子在管理员页面加载的时候调用。无论何时你的插件直接在管理页面下工作,你都要用这个钩子来执行你的代码。

下面的例子添加了一个内容是 BOJ Settings 的 sub-menu 项到 WordPress 管理页面的设置菜单。(插件设置)

<?php
add_action( 'admin_menu', 'boj_admin_settings_page');
 
function boj_admin_settings_page() {
    add_options_page(
        'BOJ Settings',
        'BOJ Settings',
        'manage_options',
        'boj_admin_settings',
        'boj_admin_settings_page'
    );
}
?>

 

template_redirect

template_redirect 动作钩子很有用,因为它是 WordPress 知道用户正在浏览的页面的关键。它在特定的页面选择 theme template 之前执行。在只在网站的前端触发,并不在管理员页面触发。

当你需要为特定的页面加载代码的时候,这个钩子很有用。

下面的例子中,仅仅为 singular post 加载一个样式表文件。

<?php
add_action( 'template_redirect', 'boj_singular_post_css' );
 
function boj_singular_post_css() {
    if( is_singular('post') ) {
        wp_enqueue_style (
            'boj-singular-post',
            'boj-example.css',
            false,
            0.1,
            'screen'
        );
    }
}
?>

 

wp_head

在网站的前端,WordPress 的模板调用 wp_head() 函数,会触发 wp_head 钩子。插件使用这个钩子在 <head> 和 </head> 标签之间添加 HTML。

下面的例子中在前端添加一个 meta description。

<?php
add_action( 'wp_head', 'boj_front_page_meta_description');
 
function boj_front_page_meta_description() {
    /* 得到站点描述 */
    $description= esc_attr( get_bloginfo('description') );
    /* 如果 description 设置了,显示 meta 元素 */
    if( !empty($description) )
        echo'<meta name="description" content="'.$description.'"/>';
}
?>

有些插件错误的使用了 wp_head 动作钩子来添加 JavaScript 代码,实际上应该使用 wp_enqueue_script() 函数的。( 详见:Part-12,”JavaScript 和 AJAX “)。唯一一种使用这个钩子来添加 JavaScript 的情形是当 JavaScript 代码不在一个单独的文件中时。

 

本文采用「CC BY-SA 4.0 CN」协议转载自互联网、仅供学习交流,内容版权归原作者所有,如涉作品、版权和其他问题请留言处理。

给TA打赏
共{{data.count}}人
人已打赏
玩站

WordPress 如何支持上传中文名文件(不更改名称)

2023-2-23 9:21:35

玩站

WordPress 插件开发-过滤器钩子(filter hooks)

2023-2-23 10:53:22

0 条回复 A文章作者 M管理员
    暂无讨论,说说你的看法吧