Make WordPress Core

Ticket #17817: 17817.10.2.diff

File 17817.10.2.diff, 85.0 KB (added by dougwollison, 9 years ago)

Correct version of previous diff; a number of the new files were missing.

  • src/wp-includes/class-wp-hook.php

     
     1<?php
     2/**
     3 * Plugin API: WP_Hook class
     4 *
     5 * @package WordPress
     6 * @subpackage Plugin
     7 * @since 4.4.0
     8 */
     9
     10/**
     11 * Core class used to implement action and filter hook functionality.
     12 *
     13 * @since 4.4.0
     14 *
     15 * @see Iterator
     16 * @see ArrayAccess
     17 */
     18final class WP_Hook implements Iterator, ArrayAccess {
     19
     20        /**
     21         * Hook callbacks.
     22         *
     23         * @since 4.4.0
     24         * @access public
     25         * @var array
     26         */
     27        public $callbacks = array();
     28
     29        /**
     30         * The priority keys of actively running iterations of a hook.
     31         *
     32         * @since 4.4.0
     33         * @access private
     34         * @var array
     35         */
     36        private $iterations = array();
     37
     38        /**
     39         * Number of levels this hook can be recursively called.
     40         *
     41         * @since 4.4.0
     42         * @access private
     43         * @var int
     44         */
     45        private $nesting_level = 0;
     46
     47        /**
     48         * Hooks a function or method to a specific filter action.
     49         *
     50         * @since 4.4.0
     51         * @access public
     52         *
     53         * @param string   $tag             The name of the filter to hook the $function_to_add callback to.
     54         * @param callable $function_to_add The callback to be run when the filter is applied.
     55         * @param int      $priority        The order in which the functions associated with a
     56         *                                  particular action are executed. Lower numbers correspond with
     57         *                                  earlier execution, and functions with the same priority are executed
     58         *                                  in the order in which they were added to the action.
     59         * @param int      $accepted_args   The number of arguments the function accepts.
     60         */
     61        public function add_filter( $tag, $function_to_add, $priority, $accepted_args ) {
     62                $idx = _wp_filter_build_unique_id( $tag, $function_to_add, $priority );
     63                $priority_existed = isset( $this->callbacks[ $priority ] );
     64
     65                $this->callbacks[ $priority ][ $idx ] = array(
     66                        'function' => $function_to_add,
     67                        'accepted_args' => $accepted_args
     68                );
     69
     70                // if we're adding a new priority to the list, put them back in sorted order
     71                if ( ! $priority_existed && count( $this->callbacks ) > 1 ) {
     72                        ksort( $this->callbacks, SORT_NUMERIC );
     73                }
     74
     75                if ( $this->nesting_level > 0 ) {
     76                        $this->resort_active_iterations();
     77                }
     78        }
     79
     80        /**
     81         * Handles reseting callback priority keys mid-iteration.
     82         *
     83         * @since 4.4.0
     84         * @access private
     85         */
     86        private function resort_active_iterations() {
     87                $new_priorities = array_keys( $this->callbacks );
     88
     89                // If there are no remaining hooks, clear out all running iterations.
     90                if ( ! $new_priorities ) {
     91                        foreach ( $this->iterations as $index => $iteration ) {
     92                                $this->iterations[ $index ] = $new_priorities;
     93                        }
     94                        return;
     95                }
     96
     97                $min = min( $new_priorities );
     98                foreach ( $this->iterations as $index => $iteration ) {
     99                        $current = current( $iteration );
     100                        $this->iterations[ $index ] = $new_priorities;
     101
     102                        if ( $current < $min ) {
     103                                array_unshift( $this->iterations[ $index ], $current );
     104                                continue;
     105                        }
     106
     107                        while ( current( $this->iterations[ $index ] ) < $current ) {
     108                                if ( false === next( $this->iterations[ $index ] ) ) {
     109                                        break;
     110                                }
     111                        }
     112                }
     113        }
     114
     115        /**
     116         * Unhooks a function or method from a specific filter action.
     117         *
     118         * @since 4.4.0
     119         * @access public
     120         *
     121         * @param string   $tag                The filter hook to which the function to be removed is hooked. Used
     122         *                                     for building the callback ID when SPL is not available.
     123         * @param callable $function_to_remove The callback to be removed from running when the filter is applied.
     124         * @param int      $priority           The exact priority used when adding the original filter callback.
     125         * @return bool Whether the callback existed before it was removed.
     126         */
     127        public function remove_filter( $tag, $function_to_remove, $priority ) {
     128                $function_key = _wp_filter_build_unique_id( $tag, $function_to_remove, $priority );
     129
     130                $exists = isset( $this->callbacks[ $priority ][ $function_key ] );
     131                if ( $exists ) {
     132                        unset( $this->callbacks[ $priority ][ $function_key ] );
     133                        if ( ! $this->callbacks[ $priority ] ) {
     134                                unset( $this->callbacks[ $priority ] );
     135                                if ( $this->nesting_level > 0 ) {
     136                                        $this->resort_active_iterations();
     137                                }
     138                        }
     139                }
     140                return $exists;
     141        }
     142
     143        /**
     144         * Checks if a specific action has been registered for this hook.
     145         *
     146         * @since 4.4.0
     147         * @access public
     148         *
     149         * @param callable|bool $function_to_check Optional. The callback to check for. Default false.
     150         * @param string        $tag               Optional. The name of the filter hook. Default empty.
     151         *                                         Used for building the callback ID when SPL is not available.
     152         * @return bool|int The priority of that hook is returned, or false if the function is not attached.
     153         */
     154        public function has_filter( $tag = '', $function_to_check = false ) {
     155                if ( false === $function_to_check ) {
     156                        return $this->has_filters();
     157                }
     158
     159                $function_key =  _wp_filter_build_unique_id( $tag, $function_to_check, false );
     160                if ( ! $function_key ) {
     161                        return false;
     162                }
     163
     164                foreach ( $this->callbacks as $priority => $callbacks ) {
     165                        if ( isset( $callbacks[ $function_key ] ) ) {
     166                                return $priority;
     167                        }
     168                }
     169
     170                return false;
     171        }
     172
     173        /**
     174         * Checks if any callbacks have been registered for this hook.
     175         *
     176         * @since 4.4.0
     177         * @access public
     178         *
     179         * @return bool True if callbacks have been registered for the current hook, false otherwise.
     180         */
     181        public function has_filters() {
     182                foreach ( $this->callbacks as $callbacks ) {
     183                        if ( $callbacks ) {
     184                                return true;
     185                        }
     186                }
     187                return false;
     188        }
     189
     190        /**
     191         * Removes all callbacks from the current filter.
     192         *
     193         * @since 4.4.0
     194         * @access public
     195         *
     196         * @param int|bool $priority Optional. The priority number to remove. Default false.
     197         */
     198        public function remove_all_filters( $priority = false ) {
     199                if ( ! $this->callbacks ) {
     200                        return;
     201                }
     202
     203                if ( false === $priority ) {
     204                        $this->callbacks = array();
     205                } else if ( isset( $this->callbacks[ $priority ] ) ) {
     206                        unset( $this->callbacks[ $priority ] );
     207                }
     208
     209                if ( $this->nesting_level > 0 ) {
     210                        $this->resort_active_iterations();
     211                }
     212        }
     213
     214        /**
     215         * Calls the callback functions added to a filter hook.
     216         *
     217         * @since 4.4.0
     218         * @access public
     219         *
     220         * @param mixed $value The value to filter.
     221         * @param array $args  Arguments to pass to callbacks.
     222         * @return mixed The filtered value after all hooked functions are applied to it.
     223         */
     224        public function apply_filters( $value, $args ) {
     225                if ( ! $this->callbacks ) {
     226                        return $value;
     227                }
     228                $nesting_level = $this->nesting_level++;
     229
     230                $this->iterations[ $nesting_level ] = array_keys( $this->callbacks );
     231                $num_args = count( $args );
     232
     233                do {
     234                        $priority = current( $this->iterations[ $nesting_level ] );
     235
     236                        foreach ( $this->callbacks[ $priority ] as $the_ ) {
     237                                $args[ 0 ] = $value;
     238
     239                                // Avoid the array_slice if possible.
     240                                if ( $the_['accepted_args'] == 0 ) {
     241                                        $value = call_user_func_array( $the_['function'], array() );
     242                                } elseif ( $the_['accepted_args'] >= $num_args ) {
     243                                        $value = call_user_func_array( $the_['function'], $args );
     244                                } else {
     245                                        $value = call_user_func_array( $the_['function'], array_slice( $args, 0, (int)$the_['accepted_args'] ) );
     246                                }
     247                        }
     248                } while ( false !== next( $this->iterations[ $nesting_level ] ) );
     249
     250                unset( $this->iterations[ $nesting_level ] );
     251
     252                $this->nesting_level--;
     253
     254                return $value;
     255        }
     256
     257        /**
     258         * Executes the callback functions hooked on a specific action hook.
     259         *
     260         * @since 4.4.0
     261         * @access public
     262         *
     263         * @param mixed $args Arguments to pass to the hook callbacks.
     264         */
     265        public function do_action( $args ) {
     266                if ( ! $this->callbacks ) {
     267                        return;
     268                }
     269                $nesting_level = $this->nesting_level++;
     270                $this->iterations[ $nesting_level ] = array_keys( $this->callbacks );
     271                $num_args = count( $args );
     272
     273                do {
     274                        $priority = current( $this->iterations[ $nesting_level ] );
     275
     276                        foreach ( $this->callbacks[ $priority ] as $the_ ) {
     277
     278                                // Avoid the array_slice if possible.
     279                                if ( $the_['accepted_args'] == 0 ) {
     280                                        call_user_func_array( $the_['function'], array() );
     281                                } elseif ( $the_['accepted_args'] >= $num_args ) {
     282                                        call_user_func_array( $the_['function'], $args );
     283                                } else {
     284                                        call_user_func_array( $the_['function'], array_slice( $args, 0, (int)$the_['accepted_args'] ) );
     285                                }
     286                        }
     287                } while ( next( $this->iterations[ $nesting_level ] ) !== false );
     288
     289                unset( $this->iterations[ $nesting_level ] );
     290                $this->nesting_level--;
     291        }
     292
     293        /**
     294         * Processes the functions hooked into the 'all' hook.
     295         *
     296         * @since 4.4.0
     297         * @access public
     298         *
     299         * @param array $args Arguments to pass to the hook callbacks. Passed by reference.
     300         */
     301        public function do_all_hook( &$args ) {
     302                $nesting_level = $this->nesting_level++;
     303                $this->iterations[ $nesting_level ] = array_keys( $this->callbacks );
     304
     305                do {
     306                        $priority = current( $this->iterations[ $nesting_level ] );
     307                        foreach ( $this->callbacks[ $priority ] as $the_ ) {
     308                                call_user_func_array( $the_['function'], $args );
     309                        }
     310                } while ( false !== next( $this->iterations[ $nesting_level ] ) );
     311
     312                unset( $this->iterations[ $nesting_level ] );
     313                $this->nesting_level--;
     314        }
     315
     316        /**
     317         * Normalizes filters setup before WordPress has initialized to WP_Hook objects.
     318         *
     319         * @since 4.4.0
     320         * @access public
     321         * @static
     322         *
     323         * @param array $filters Filters to normalize.
     324         * @return WP_Hook[] Array of normalized filters.
     325         */
     326        public static function build_preinitialized_hooks( $filters ) {
     327                /** @var WP_Hook[] $normalized */
     328                $normalized = array();
     329
     330                foreach ( $filters as $tag => $callback_groups ) {
     331                        if ( is_object( $callback_groups ) && $callback_groups instanceof WP_Hook ) {
     332                                $normalized[ $tag ] = $callback_groups;
     333                                continue;
     334                        }
     335                        $hook = new WP_Hook();
     336
     337                        // Loop through callback groups.
     338                        foreach ( $callback_groups as $priority => $callbacks ) {
     339
     340                                // Loop through callbacks.
     341                                foreach ( $callbacks as $cb ) {
     342                                        $hook->add_filter( $tag, $cb['function'], $priority, $cb['accepted_args'] );
     343                                }
     344                        }
     345                        $normalized[ $tag ] = $hook;
     346                }
     347                return $normalized;
     348        }
     349
     350        /**
     351         * Determines whether an offset value exists.
     352         *
     353         * @since 4.4.0
     354         * @access public
     355         *
     356         * @link http://php.net/manual/en/arrayaccess.offsetexists.php
     357         *
     358         * @param mixed $offset An offset to check for.
     359         * @return bool True if the offset exists, false otherwise.
     360         */
     361        public function offsetExists( $offset ) {
     362                return isset( $this->callbacks[ $offset ] );
     363        }
     364
     365        /**
     366         * Retrieves a value at a specified offset.
     367         *
     368         * @since 4.4.0
     369         * @access public
     370         *
     371         * @link http://php.net/manual/en/arrayaccess.offsetget.php
     372         *
     373         * @param mixed $offset The offset to retrieve.
     374         * @return mixed If set, the value at the specified offset, null otherwise.
     375         */
     376        public function offsetGet( $offset ) {
     377                return isset( $this->callbacks[ $offset ] ) ? $this->callbacks[ $offset ] : null;
     378        }
     379
     380        /**
     381         * Sets a value at a specified offset.
     382         *
     383         * @since 4.4.0
     384         * @access public
     385         *
     386         * @link http://php.net/manual/en/arrayaccess.offsetset.php
     387         *
     388         * @param mixed $offset The offset to assign the value to.
     389         * @param mixed $value The value to set.
     390         */
     391        public function offsetSet( $offset, $value ) {
     392                if ( is_null( $offset ) ) {
     393                        $this->callbacks[] = $value;
     394                } else {
     395                        $this->callbacks[ $offset ] = $value;
     396                }
     397        }
     398
     399        /**
     400         * Unsets a specified offset.
     401         *
     402         * @since 4.4.0
     403         * @access public
     404         *
     405         * @link http://php.net/manual/en/arrayaccess.offsetunset.php
     406         *
     407         * @param mixed $offset The offset to unset.
     408         */
     409        public function offsetUnset( $offset ) {
     410                unset( $this->callbacks[ $offset ] );
     411        }
     412
     413        /**
     414         * Return the current element
     415         *
     416         * @since 4.4.0
     417         * @access public
     418         *
     419         * @link http://php.net/manual/en/iterator.current.php
     420         *
     421         * @return mixed
     422         */
     423        public function current() {
     424                return current( $this->callbacks );
     425        }
     426
     427        /**
     428         * Move forward to the next element
     429         *
     430         * @since 4.4.0
     431         * @access public
     432         *
     433         * @link http://php.net/manual/en/iterator.next.php
     434         */
     435        public function next() {
     436                return next( $this->callbacks );
     437        }
     438
     439        /**
     440         * Return the key of the current element
     441         *
     442         * @since 4.4.0
     443         * @access public
     444         *
     445         * @link http://php.net/manual/en/iterator.key.php
     446         *
     447         * @return mixed Returns scalar on success, or NULL on failure
     448         */
     449        public function key() {
     450                return key( $this->callbacks );
     451        }
     452
     453        /**
     454         * Checks if current position is valid
     455         *
     456         * @since 4.4.0
     457         * @access public
     458         *
     459         * @link http://php.net/manual/en/iterator.valid.php
     460         *
     461         * @return boolean
     462         */
     463        public function valid() {
     464                return key( $this->callbacks ) !== null;
     465        }
     466
     467        /**
     468         * Rewind the Iterator to the first element
     469         *
     470         * @since 4.4.0
     471         * @access public
     472         *
     473         * @link http://php.net/manual/en/iterator.rewind.php
     474         */
     475        public function rewind() {
     476                return reset( $this->callbacks );
     477        }
     478
     479
     480}
     481<?php
     482/**
     483 * Plugin API: WP_Hook class
     484 *
     485 * @package WordPress
     486 * @subpackage Plugin
     487 * @since 4.4.0
     488 */
     489
     490/**
     491 * Core class used to implement action and filter hook functionality.
     492 *
     493 * @since 4.4.0
     494 *
     495 * @see Iterator
     496 * @see ArrayAccess
     497 */
     498final class WP_Hook implements Iterator, ArrayAccess {
     499
     500        /**
     501         * Hook callbacks.
     502         *
     503         * @since 4.4.0
     504         * @access public
     505         * @var array
     506         */
     507        public $callbacks = array();
     508
     509        /**
     510         * The priority keys of actively running iterations of a hook.
     511         *
     512         * @since 4.4.0
     513         * @access private
     514         * @var array
     515         */
     516        private $iterations = array();
     517
     518        /**
     519         * Number of levels this hook can be recursively called.
     520         *
     521         * @since 4.4.0
     522         * @access private
     523         * @var int
     524         */
     525        private $nesting_level = 0;
     526
     527        /**
     528         * Hooks a function or method to a specific filter action.
     529         *
     530         * @since 4.4.0
     531         * @access public
     532         *
     533         * @param string   $tag             The name of the filter to hook the $function_to_add callback to.
     534         * @param callable $function_to_add The callback to be run when the filter is applied.
     535         * @param int      $priority        The order in which the functions associated with a
     536         *                                  particular action are executed. Lower numbers correspond with
     537         *                                  earlier execution, and functions with the same priority are executed
     538         *                                  in the order in which they were added to the action.
     539         * @param int      $accepted_args   The number of arguments the function accepts.
     540         */
     541        public function add_filter( $tag, $function_to_add, $priority, $accepted_args ) {
     542                $idx = _wp_filter_build_unique_id( $tag, $function_to_add, $priority );
     543                $priority_existed = isset( $this->callbacks[ $priority ] );
     544
     545                $this->callbacks[ $priority ][ $idx ] = array(
     546                        'function' => $function_to_add,
     547                        'accepted_args' => $accepted_args
     548                );
     549
     550                // if we're adding a new priority to the list, put them back in sorted order
     551                if ( ! $priority_existed && count( $this->callbacks ) > 1 ) {
     552                        ksort( $this->callbacks, SORT_NUMERIC );
     553                }
     554
     555                if ( $this->nesting_level > 0 ) {
     556                        $this->resort_active_iterations();
     557                }
     558        }
     559
     560        /**
     561         * Handles reseting callback priority keys mid-iteration.
     562         *
     563         * @since 4.4.0
     564         * @access private
     565         */
     566        private function resort_active_iterations() {
     567                $new_priorities = array_keys( $this->callbacks );
     568
     569                // If there are no remaining hooks, clear out all running iterations.
     570                if ( ! $new_priorities ) {
     571                        foreach ( $this->iterations as $index => $iteration ) {
     572                                $this->iterations[ $index ] = $new_priorities;
     573                        }
     574                        return;
     575                }
     576
     577                $min = min( $new_priorities );
     578                foreach ( $this->iterations as $index => $iteration ) {
     579                        $current = current( $iteration );
     580                        $this->iterations[ $index ] = $new_priorities;
     581
     582                        if ( $current < $min ) {
     583                                array_unshift( $this->iterations[ $index ], $current );
     584                                continue;
     585                        }
     586
     587                        while ( current( $this->iterations[ $index ] ) < $current ) {
     588                                if ( false === next( $this->iterations[ $index ] ) ) {
     589                                        break;
     590                                }
     591                        }
     592                }
     593        }
     594
     595        /**
     596         * Unhooks a function or method from a specific filter action.
     597         *
     598         * @since 4.4.0
     599         * @access public
     600         *
     601         * @param string   $tag                The filter hook to which the function to be removed is hooked. Used
     602         *                                     for building the callback ID when SPL is not available.
     603         * @param callable $function_to_remove The callback to be removed from running when the filter is applied.
     604         * @param int      $priority           The exact priority used when adding the original filter callback.
     605         * @return bool Whether the callback existed before it was removed.
     606         */
     607        public function remove_filter( $tag, $function_to_remove, $priority ) {
     608                $function_key = _wp_filter_build_unique_id( $tag, $function_to_remove, $priority );
     609
     610                $exists = isset( $this->callbacks[ $priority ][ $function_key ] );
     611                if ( $exists ) {
     612                        unset( $this->callbacks[ $priority ][ $function_key ] );
     613                        if ( ! $this->callbacks[ $priority ] ) {
     614                                unset( $this->callbacks[ $priority ] );
     615                                if ( $this->nesting_level > 0 ) {
     616                                        $this->resort_active_iterations();
     617                                }
     618                        }
     619                }
     620                return $exists;
     621        }
     622
     623        /**
     624         * Checks if a specific action has been registered for this hook.
     625         *
     626         * @since 4.4.0
     627         * @access public
     628         *
     629         * @param callable|bool $function_to_check Optional. The callback to check for. Default false.
     630         * @param string        $tag               Optional. The name of the filter hook. Default empty.
     631         *                                         Used for building the callback ID when SPL is not available.
     632         * @return bool|int The priority of that hook is returned, or false if the function is not attached.
     633         */
     634        public function has_filter( $tag = '', $function_to_check = false ) {
     635                if ( false === $function_to_check ) {
     636                        return $this->has_filters();
     637                }
     638
     639                $function_key =  _wp_filter_build_unique_id( $tag, $function_to_check, false );
     640                if ( ! $function_key ) {
     641                        return false;
     642                }
     643
     644                foreach ( $this->callbacks as $priority => $callbacks ) {
     645                        if ( isset( $callbacks[ $function_key ] ) ) {
     646                                return $priority;
     647                        }
     648                }
     649
     650                return false;
     651        }
     652
     653        /**
     654         * Checks if any callbacks have been registered for this hook.
     655         *
     656         * @since 4.4.0
     657         * @access public
     658         *
     659         * @return bool True if callbacks have been registered for the current hook, false otherwise.
     660         */
     661        public function has_filters() {
     662                foreach ( $this->callbacks as $callbacks ) {
     663                        if ( $callbacks ) {
     664                                return true;
     665                        }
     666                }
     667                return false;
     668        }
     669
     670        /**
     671         * Removes all callbacks from the current filter.
     672         *
     673         * @since 4.4.0
     674         * @access public
     675         *
     676         * @param int|bool $priority Optional. The priority number to remove. Default false.
     677         */
     678        public function remove_all_filters( $priority = false ) {
     679                if ( ! $this->callbacks ) {
     680                        return;
     681                }
     682
     683                if ( false === $priority ) {
     684                        $this->callbacks = array();
     685                } else if ( isset( $this->callbacks[ $priority ] ) ) {
     686                        unset( $this->callbacks[ $priority ] );
     687                }
     688
     689                if ( $this->nesting_level > 0 ) {
     690                        $this->resort_active_iterations();
     691                }
     692        }
     693
     694        /**
     695         * Calls the callback functions added to a filter hook.
     696         *
     697         * @since 4.4.0
     698         * @access public
     699         *
     700         * @param mixed $value The value to filter.
     701         * @param array $args  Arguments to pass to callbacks.
     702         * @return mixed The filtered value after all hooked functions are applied to it.
     703         */
     704        public function apply_filters( $value, $args ) {
     705                if ( ! $this->callbacks ) {
     706                        return $value;
     707                }
     708                $nesting_level = $this->nesting_level++;
     709
     710                $this->iterations[ $nesting_level ] = array_keys( $this->callbacks );
     711                $num_args = count( $args );
     712
     713                do {
     714                        $priority = current( $this->iterations[ $nesting_level ] );
     715
     716                        foreach ( $this->callbacks[ $priority ] as $the_ ) {
     717                                $args[ 0 ] = $value;
     718
     719                                // Avoid the array_slice if possible.
     720                                if ( $the_['accepted_args'] == 0 ) {
     721                                        $value = call_user_func_array( $the_['function'], array() );
     722                                } elseif ( $the_['accepted_args'] >= $num_args ) {
     723                                        $value = call_user_func_array( $the_['function'], $args );
     724                                } else {
     725                                        $value = call_user_func_array( $the_['function'], array_slice( $args, 0, (int)$the_['accepted_args'] ) );
     726                                }
     727                        }
     728                } while ( false !== next( $this->iterations[ $nesting_level ] ) );
     729
     730                unset( $this->iterations[ $nesting_level ] );
     731
     732                $this->nesting_level--;
     733
     734                return $value;
     735        }
     736
     737        /**
     738         * Executes the callback functions hooked on a specific action hook.
     739         *
     740         * @since 4.4.0
     741         * @access public
     742         *
     743         * @param mixed $args Arguments to pass to the hook callbacks.
     744         */
     745        public function do_action( $args ) {
     746                if ( ! $this->callbacks ) {
     747                        return;
     748                }
     749                $nesting_level = $this->nesting_level++;
     750                $this->iterations[ $nesting_level ] = array_keys( $this->callbacks );
     751                $num_args = count( $args );
     752
     753                do {
     754                        $priority = current( $this->iterations[ $nesting_level ] );
     755
     756                        foreach ( $this->callbacks[ $priority ] as $the_ ) {
     757
     758                                // Avoid the array_slice if possible.
     759                                if ( $the_['accepted_args'] == 0 ) {
     760                                        call_user_func_array( $the_['function'], array() );
     761                                } elseif ( $the_['accepted_args'] >= $num_args ) {
     762                                        call_user_func_array( $the_['function'], $args );
     763                                } else {
     764                                        call_user_func_array( $the_['function'], array_slice( $args, 0, (int)$the_['accepted_args'] ) );
     765                                }
     766                        }
     767                } while ( next( $this->iterations[ $nesting_level ] ) !== false );
     768
     769                unset( $this->iterations[ $nesting_level ] );
     770                $this->nesting_level--;
     771        }
     772
     773        /**
     774         * Processes the functions hooked into the 'all' hook.
     775         *
     776         * @since 4.4.0
     777         * @access public
     778         *
     779         * @param array $args Arguments to pass to the hook callbacks. Passed by reference.
     780         */
     781        public function do_all_hook( &$args ) {
     782                $nesting_level = $this->nesting_level++;
     783                $this->iterations[ $nesting_level ] = array_keys( $this->callbacks );
     784
     785                do {
     786                        $priority = current( $this->iterations[ $nesting_level ] );
     787                        foreach ( $this->callbacks[ $priority ] as $the_ ) {
     788                                call_user_func_array( $the_['function'], $args );
     789                        }
     790                } while ( false !== next( $this->iterations[ $nesting_level ] ) );
     791
     792                unset( $this->iterations[ $nesting_level ] );
     793                $this->nesting_level--;
     794        }
     795
     796        /**
     797         * Normalizes filters setup before WordPress has initialized to WP_Hook objects.
     798         *
     799         * @since 4.4.0
     800         * @access public
     801         * @static
     802         *
     803         * @param array $filters Filters to normalize.
     804         * @return WP_Hook[] Array of normalized filters.
     805         */
     806        public static function build_preinitialized_hooks( $filters ) {
     807                /** @var WP_Hook[] $normalized */
     808                $normalized = array();
     809
     810                foreach ( $filters as $tag => $callback_groups ) {
     811                        if ( is_object( $callback_groups ) && $callback_groups instanceof WP_Hook ) {
     812                                $normalized[ $tag ] = $callback_groups;
     813                                continue;
     814                        }
     815                        $hook = new WP_Hook();
     816
     817                        // Loop through callback groups.
     818                        foreach ( $callback_groups as $priority => $callbacks ) {
     819
     820                                // Loop through callbacks.
     821                                foreach ( $callbacks as $cb ) {
     822                                        $hook->add_filter( $tag, $cb['function'], $priority, $cb['accepted_args'] );
     823                                }
     824                        }
     825                        $normalized[ $tag ] = $hook;
     826                }
     827                return $normalized;
     828        }
     829
     830        /**
     831         * Determines whether an offset value exists.
     832         *
     833         * @since 4.4.0
     834         * @access public
     835         *
     836         * @link http://php.net/manual/en/arrayaccess.offsetexists.php
     837         *
     838         * @param mixed $offset An offset to check for.
     839         * @return bool True if the offset exists, false otherwise.
     840         */
     841        public function offsetExists( $offset ) {
     842                return isset( $this->callbacks[ $offset ] );
     843        }
     844
     845        /**
     846         * Retrieves a value at a specified offset.
     847         *
     848         * @since 4.4.0
     849         * @access public
     850         *
     851         * @link http://php.net/manual/en/arrayaccess.offsetget.php
     852         *
     853         * @param mixed $offset The offset to retrieve.
     854         * @return mixed If set, the value at the specified offset, null otherwise.
     855         */
     856        public function offsetGet( $offset ) {
     857                return isset( $this->callbacks[ $offset ] ) ? $this->callbacks[ $offset ] : null;
     858        }
     859
     860        /**
     861         * Sets a value at a specified offset.
     862         *
     863         * @since 4.4.0
     864         * @access public
     865         *
     866         * @link http://php.net/manual/en/arrayaccess.offsetset.php
     867         *
     868         * @param mixed $offset The offset to assign the value to.
     869         * @param mixed $value The value to set.
     870         */
     871        public function offsetSet( $offset, $value ) {
     872                if ( is_null( $offset ) ) {
     873                        $this->callbacks[] = $value;
     874                } else {
     875                        $this->callbacks[ $offset ] = $value;
     876                }
     877        }
     878
     879        /**
     880         * Unsets a specified offset.
     881         *
     882         * @since 4.4.0
     883         * @access public
     884         *
     885         * @link http://php.net/manual/en/arrayaccess.offsetunset.php
     886         *
     887         * @param mixed $offset The offset to unset.
     888         */
     889        public function offsetUnset( $offset ) {
     890                unset( $this->callbacks[ $offset ] );
     891        }
     892
     893        /**
     894         * Return the current element
     895         *
     896         * @since 4.4.0
     897         * @access public
     898         *
     899         * @link http://php.net/manual/en/iterator.current.php
     900         *
     901         * @return mixed
     902         */
     903        public function current() {
     904                return current( $this->callbacks );
     905        }
     906
     907        /**
     908         * Move forward to the next element
     909         *
     910         * @since 4.4.0
     911         * @access public
     912         *
     913         * @link http://php.net/manual/en/iterator.next.php
     914         */
     915        public function next() {
     916                return next( $this->callbacks );
     917        }
     918
     919        /**
     920         * Return the key of the current element
     921         *
     922         * @since 4.4.0
     923         * @access public
     924         *
     925         * @link http://php.net/manual/en/iterator.key.php
     926         *
     927         * @return mixed Returns scalar on success, or NULL on failure
     928         */
     929        public function key() {
     930                return key( $this->callbacks );
     931        }
     932
     933        /**
     934         * Checks if current position is valid
     935         *
     936         * @since 4.4.0
     937         * @access public
     938         *
     939         * @link http://php.net/manual/en/iterator.valid.php
     940         *
     941         * @return boolean
     942         */
     943        public function valid() {
     944                return key( $this->callbacks ) !== null;
     945        }
     946
     947        /**
     948         * Rewind the Iterator to the first element
     949         *
     950         * @since 4.4.0
     951         * @access public
     952         *
     953         * @link http://php.net/manual/en/iterator.rewind.php
     954         */
     955        public function rewind() {
     956                return reset( $this->callbacks );
     957        }
     958
     959
     960}
  • src/wp-includes/plugin.php

     
    2020 */
    2121
    2222// Initialize the filter globals.
    23 global $wp_filter, $wp_actions, $merged_filters, $wp_current_filter;
     23require( ABSPATH . WPINC . '/class-wp-hook.php' );
    2424
    25 if ( ! isset( $wp_filter ) )
     25/** @var WP_Hook[] $wp_filter */
     26global $wp_filter, $wp_actions, $wp_current_filter;
     27
     28if ( $wp_filter ) {
     29        $wp_filter = WP_Hook::build_preinitialized_hooks( $wp_filter );
     30} else {
    2631        $wp_filter = array();
     32}
    2733
    2834if ( ! isset( $wp_actions ) )
    2935        $wp_actions = array();
    3036
    31 if ( ! isset( $merged_filters ) )
    32         $merged_filters = array();
    33 
    3437if ( ! isset( $wp_current_filter ) )
    3538        $wp_current_filter = array();
    3639
     
    8790 * @since 0.71
    8891 *
    8992 * @global array $wp_filter      A multidimensional array of all hooks and the callbacks hooked to them.
    90  * @global array $merged_filters Tracks the tags that need to be merged for later. If the hook is added,
    91  *                               it doesn't need to run through that process.
    9293 *
    9394 * @param string   $tag             The name of the filter to hook the $function_to_add callback to.
    9495 * @param callable $function_to_add The callback to be run when the filter is applied.
     
    101102 * @return true
    102103 */
    103104function add_filter( $tag, $function_to_add, $priority = 10, $accepted_args = 1 ) {
    104         global $wp_filter, $merged_filters;
    105 
    106         $idx = _wp_filter_build_unique_id($tag, $function_to_add, $priority);
    107         $wp_filter[$tag][$priority][$idx] = array('function' => $function_to_add, 'accepted_args' => $accepted_args);
    108         unset( $merged_filters[ $tag ] );
     105        global $wp_filter;
     106        if ( ! isset( $wp_filter[ $tag ] ) ) {
     107                $wp_filter[ $tag ] = new WP_Hook();
     108        }
     109        $wp_filter[ $tag ]->add_filter( $tag, $function_to_add, $priority, $accepted_args );
    109110        return true;
    110111}
    111112
     
    126127 *                   return value.
    127128 */
    128129function has_filter($tag, $function_to_check = false) {
    129         // Don't reset the internal array pointer
    130         $wp_filter = $GLOBALS['wp_filter'];
     130        global $wp_filter;
    131131
    132         $has = ! empty( $wp_filter[ $tag ] );
    133 
    134         // Make sure at least one priority has a filter callback
    135         if ( $has ) {
    136                 $exists = false;
    137                 foreach ( $wp_filter[ $tag ] as $callbacks ) {
    138                         if ( ! empty( $callbacks ) ) {
    139                                 $exists = true;
    140                                 break;
    141                         }
    142                 }
    143 
    144                 if ( ! $exists ) {
    145                         $has = false;
    146                 }
    147         }
    148 
    149         if ( false === $function_to_check || false === $has )
    150                 return $has;
    151 
    152         if ( !$idx = _wp_filter_build_unique_id($tag, $function_to_check, false) )
     132        if ( ! isset( $wp_filter[ $tag ] ) ) {
    153133                return false;
    154 
    155         foreach ( (array) array_keys($wp_filter[$tag]) as $priority ) {
    156                 if ( isset($wp_filter[$tag][$priority][$idx]) )
    157                         return $priority;
    158134        }
    159135
    160         return false;
     136        return $wp_filter[ $tag ]->has_filter( $tag, $function_to_check );
    161137}
    162138
    163139/**
     
    188164 * @since 0.71
    189165 *
    190166 * @global array $wp_filter         Stores all of the filters.
    191  * @global array $merged_filters    Merges the filter hooks using this function.
    192167 * @global array $wp_current_filter Stores the list of current filters with the current one last.
    193168 *
    194169 * @param string $tag     The name of the filter hook.
     
    197172 * @return mixed The filtered value after all hooked functions are applied to it.
    198173 */
    199174function apply_filters( $tag, $value ) {
    200         global $wp_filter, $merged_filters, $wp_current_filter;
     175        global $wp_filter, $wp_current_filter;
    201176
    202177        $args = array();
    203178
     
    217192        if ( !isset($wp_filter['all']) )
    218193                $wp_current_filter[] = $tag;
    219194
    220         // Sort.
    221         if ( !isset( $merged_filters[ $tag ] ) ) {
    222                 ksort($wp_filter[$tag]);
    223                 $merged_filters[ $tag ] = true;
    224         }
    225 
    226         reset( $wp_filter[ $tag ] );
    227 
    228195        if ( empty($args) )
    229196                $args = func_get_args();
    230197
    231         do {
    232                 foreach ( (array) current($wp_filter[$tag]) as $the_ )
    233                         if ( !is_null($the_['function']) ){
    234                                 $args[1] = $value;
    235                                 $value = call_user_func_array($the_['function'], array_slice($args, 1, (int) $the_['accepted_args']));
    236                         }
     198        // don't pass the tag name to WP_Hook
     199        array_shift( $args );
    237200
    238         } while ( next($wp_filter[$tag]) !== false );
     201        $filtered = $wp_filter[ $tag ]->apply_filters( $value, $args );
    239202
    240203        array_pop( $wp_current_filter );
    241204
    242         return $value;
     205        return $filtered;
    243206}
    244207
    245208/**
     
    251214 * functions hooked to `$tag` are supplied using an array.
    252215 *
    253216 * @global array $wp_filter         Stores all of the filters
    254  * @global array $merged_filters    Merges the filter hooks using this function.
    255217 * @global array $wp_current_filter Stores the list of current filters with the current one last
    256218 *
    257219 * @param string $tag  The name of the filter hook.
     
    259221 * @return mixed The filtered value after all hooked functions are applied to it.
    260222 */
    261223function apply_filters_ref_array($tag, $args) {
    262         global $wp_filter, $merged_filters, $wp_current_filter;
     224        global $wp_filter, $wp_current_filter;
    263225
    264226        // Do 'all' actions first
    265227        if ( isset($wp_filter['all']) ) {
     
    277239        if ( !isset($wp_filter['all']) )
    278240                $wp_current_filter[] = $tag;
    279241
    280         // Sort
    281         if ( !isset( $merged_filters[ $tag ] ) ) {
    282                 ksort($wp_filter[$tag]);
    283                 $merged_filters[ $tag ] = true;
    284         }
     242        $filtered = $wp_filter[ $tag ]->apply_filters( $args[0], $args );
    285243
    286         reset( $wp_filter[ $tag ] );
    287 
    288         do {
    289                 foreach ( (array) current($wp_filter[$tag]) as $the_ )
    290                         if ( !is_null($the_['function']) )
    291                                 $args[0] = call_user_func_array($the_['function'], array_slice($args, 0, (int) $the_['accepted_args']));
    292 
    293         } while ( next($wp_filter[$tag]) !== false );
    294 
    295244        array_pop( $wp_current_filter );
    296245
    297         return $args[0];
     246        return $filtered;
    298247}
    299248
    300249/**
     
    319268 * @return bool    Whether the function existed before it was removed.
    320269 */
    321270function remove_filter( $tag, $function_to_remove, $priority = 10 ) {
    322         $function_to_remove = _wp_filter_build_unique_id( $tag, $function_to_remove, $priority );
     271        global $wp_filter;
    323272
    324         $r = isset( $GLOBALS['wp_filter'][ $tag ][ $priority ][ $function_to_remove ] );
    325 
    326         if ( true === $r ) {
    327                 unset( $GLOBALS['wp_filter'][ $tag ][ $priority ][ $function_to_remove ] );
    328                 if ( empty( $GLOBALS['wp_filter'][ $tag ][ $priority ] ) ) {
    329                         unset( $GLOBALS['wp_filter'][ $tag ][ $priority ] );
     273        $r = false;
     274        if ( isset( $wp_filter[ $tag ] ) ) {
     275                $r = $wp_filter[ $tag ]->remove_filter( $tag, $function_to_remove, $priority );
     276                if ( ! $wp_filter[ $tag ]->callbacks ) {
     277                        unset( $wp_filter[ $tag ] );
    330278                }
    331                 if ( empty( $GLOBALS['wp_filter'][ $tag ] ) ) {
    332                         $GLOBALS['wp_filter'][ $tag ] = array();
    333                 }
    334                 unset( $GLOBALS['merged_filters'][ $tag ] );
    335279        }
    336280
    337281        return $r;
     
    342286 *
    343287 * @since 2.7.0
    344288 *
    345  * @global array $wp_filter         Stores all of the filters
    346  * @global array $merged_filters    Merges the filter hooks using this function.
     289 * @global array $wp_filter  Stores all of the filters
    347290 *
    348291 * @param string   $tag      The filter to remove hooks from.
    349292 * @param int|bool $priority Optional. The priority number to remove. Default false.
    350293 * @return true True when finished.
    351294 */
    352295function remove_all_filters( $tag, $priority = false ) {
    353         global $wp_filter, $merged_filters;
     296        global $wp_filter;
    354297
    355298        if ( isset( $wp_filter[ $tag ]) ) {
    356                 if ( false === $priority ) {
    357                         $wp_filter[ $tag ] = array();
    358                 } elseif ( isset( $wp_filter[ $tag ][ $priority ] ) ) {
    359                         $wp_filter[ $tag ][ $priority ] = array();
     299                $wp_filter[ $tag ]->remove_all_filters( $priority );
     300                if ( ! $wp_filter[ $tag ]->has_filters() ) {
     301                        unset( $wp_filter[ $tag ] );
    360302                }
    361303        }
    362304
    363         unset( $merged_filters[ $tag ] );
    364 
    365305        return true;
    366306}
    367307
     
    480420 *                        functions hooked to the action. Default empty.
    481421 */
    482422function do_action($tag, $arg = '') {
    483         global $wp_filter, $wp_actions, $merged_filters, $wp_current_filter;
     423        global $wp_filter, $wp_actions, $wp_current_filter;
    484424
    485425        if ( ! isset($wp_actions[$tag]) )
    486426                $wp_actions[$tag] = 1;
     
    511451        for ( $a = 2, $num = func_num_args(); $a < $num; $a++ )
    512452                $args[] = func_get_arg($a);
    513453
    514         // Sort
    515         if ( !isset( $merged_filters[ $tag ] ) ) {
    516                 ksort($wp_filter[$tag]);
    517                 $merged_filters[ $tag ] = true;
    518         }
     454        $wp_filter[ $tag ]->do_action( $args );
    519455
    520         reset( $wp_filter[ $tag ] );
    521 
    522         do {
    523                 foreach ( (array) current($wp_filter[$tag]) as $the_ )
    524                         if ( !is_null($the_['function']) )
    525                                 call_user_func_array($the_['function'], array_slice($args, 0, (int) $the_['accepted_args']));
    526 
    527         } while ( next($wp_filter[$tag]) !== false );
    528 
    529456        array_pop($wp_current_filter);
    530457}
    531458
     
    564491 * @param array  $args The arguments supplied to the functions hooked to `$tag`.
    565492 */
    566493function do_action_ref_array($tag, $args) {
    567         global $wp_filter, $wp_actions, $merged_filters, $wp_current_filter;
     494        global $wp_filter, $wp_actions, $wp_current_filter;
    568495
    569496        if ( ! isset($wp_actions[$tag]) )
    570497                $wp_actions[$tag] = 1;
     
    587514        if ( !isset($wp_filter['all']) )
    588515                $wp_current_filter[] = $tag;
    589516
    590         // Sort
    591         if ( !isset( $merged_filters[ $tag ] ) ) {
    592                 ksort($wp_filter[$tag]);
    593                 $merged_filters[ $tag ] = true;
    594         }
     517        $wp_filter[ $tag ]->do_action( $args );
    595518
    596         reset( $wp_filter[ $tag ] );
    597 
    598         do {
    599                 foreach ( (array) current($wp_filter[$tag]) as $the_ )
    600                         if ( !is_null($the_['function']) )
    601                                 call_user_func_array($the_['function'], array_slice($args, 0, (int) $the_['accepted_args']));
    602 
    603         } while ( next($wp_filter[$tag]) !== false );
    604 
    605519        array_pop($wp_current_filter);
    606520}
    607521
     
    865779function _wp_call_all_hook($args) {
    866780        global $wp_filter;
    867781
    868         reset( $wp_filter['all'] );
    869         do {
    870                 foreach ( (array) current($wp_filter['all']) as $the_ )
    871                         if ( !is_null($the_['function']) )
    872                                 call_user_func_array($the_['function'], $args);
    873 
    874         } while ( next($wp_filter['all']) !== false );
     782        $wp_filter['all']->do_all_hook( $args );
    875783}
    876784
    877785/**
  • tests/phpunit/includes/functions.php

     
    1818
    1919// For adding hooks before loading WP
    2020function tests_add_filter($tag, $function_to_add, $priority = 10, $accepted_args = 1) {
    21         global $wp_filter, $merged_filters;
     21        global $wp_filter;
    2222
    2323        $idx = _test_filter_build_unique_id($tag, $function_to_add, $priority);
    2424        $wp_filter[$tag][$priority][$idx] = array('function' => $function_to_add, 'accepted_args' => $accepted_args);
    25         unset( $merged_filters[ $tag ] );
    2625        return true;
    2726}
    2827
    2928function _test_filter_build_unique_id($tag, $function, $priority) {
    30         global $wp_filter;
    31         static $filter_id_count = 0;
    32 
    3329        if ( is_string($function) )
    3430                return $function;
    3531
  • tests/phpunit/includes/testcase.php

     
    218218         * @return void
    219219         */
    220220        protected function _backup_hooks() {
    221                 $globals = array( 'merged_filters', 'wp_actions', 'wp_current_filter', 'wp_filter' );
     221                $globals = array( 'wp_actions', 'wp_current_filter' );
    222222                foreach ( $globals as $key ) {
    223223                        self::$hooks_saved[ $key ] = $GLOBALS[ $key ];
    224224                }
     225                self::$hooks_saved['wp_filter'] = array();
     226                foreach ( $GLOBALS['wp_filter'] as $hook_name => $hook_object ) {
     227                        self::$hooks_saved['wp_filter'][ $hook_name ] = clone $hook_object;
     228                }
    225229        }
    226230
    227231        /**
     
    235239         * @return void
    236240         */
    237241        protected function _restore_hooks() {
    238                 $globals = array( 'merged_filters', 'wp_actions', 'wp_current_filter', 'wp_filter' );
     242                $globals = array( 'wp_actions', 'wp_current_filter' );
    239243                foreach ( $globals as $key ) {
    240244                        if ( isset( self::$hooks_saved[ $key ] ) ) {
    241245                                $GLOBALS[ $key ] = self::$hooks_saved[ $key ];
    242246                        }
    243247                }
     248                if ( isset( self::$hooks_saved['wp_filter'] ) ) {
     249                        $GLOBALS['wp_filter'] = array();
     250                        foreach ( self::$hooks_saved['wp_filter'] as $hook_name => $hook_object ) {
     251                                $GLOBALS['wp_filter'][ $hook_name ] = clone $hook_object;
     252                        }
     253                }
    244254        }
    245255
    246256        function flush_cache() {
  • tests/phpunit/tests/actions.php

     
    114114                $this->assertEquals( array( $val1 ), array_pop( $argsvar2 ) );
    115115        }
    116116
     117        /**
     118         * Test that multiple callbacks receive the correct number of args even when the number
     119         * is less than, or greater than previous hooks.
     120         *
     121         * @see https://core.trac.wordpress.org/ticket/17817#comment:72
     122         * @ticket 17817
     123         */
     124        function test_action_args_3() {
     125                $a1 = new MockAction();
     126                $a2 = new MockAction();
     127                $a3 = new MockAction();
     128                $tag = rand_str();
     129                $val1 = rand_str();
     130                $val2 = rand_str();
     131
     132                // a1 accepts two arguments, a2 doesn't, a3 accepts two arguments
     133                add_action( $tag, array( &$a1, 'action' ), 10, 2 );
     134                add_action( $tag, array( &$a2, 'action' ) );
     135                add_action( $tag, array( &$a3, 'action' ), 10, 2 );
     136                // call the action with two arguments
     137                do_action( $tag, $val1, $val2 );
     138
     139                $call_count = $a1->get_call_count();
     140                // a1 should be called with both args
     141                $this->assertEquals( 1, $call_count );
     142                $argsvar1 = $a1->get_args();
     143                $this->assertEquals( array( $val1, $val2 ), array_pop( $argsvar1 ) );
     144
     145                // a2 should be called with one only
     146                $this->assertEquals( 1, $a2->get_call_count() );
     147                $argsvar2 = $a2->get_args();
     148                $this->assertEquals( array( $val1 ), array_pop( $argsvar2 ) );
     149
     150                // a3 should be called with both args
     151                $this->assertEquals( 1, $a3->get_call_count() );
     152                $argsvar3 = $a3->get_args();
     153                $this->assertEquals( array( $val1, $val2 ), array_pop( $argsvar3 ) );
     154        }
     155
    117156        function test_action_priority() {
    118157                $a = new MockAction();
    119158                $tag = rand_str();
     
    258297        }
    259298
    260299        /**
     300         * @ticket 17817
     301         */
     302        function test_action_recursion() {
     303                $tag = rand_str();
     304                $a = new MockAction();
     305                $b = new MockAction();
     306
     307                add_action( $tag, array( $a, 'action' ), 11, 1 );
     308                add_action( $tag, array( $b, 'action' ), 13, 1 );
     309                add_action( $tag, array( $this, 'action_that_causes_recursion' ), 12, 1 );
     310                do_action( $tag, $tag );
     311
     312                $this->assertEquals( 2, $a->get_call_count(), 'recursive actions should call all callbacks with earlier priority' );
     313                $this->assertEquals( 2, $b->get_call_count(), 'recursive actions should call callbacks with later priority' );
     314        }
     315
     316        function action_that_causes_recursion( $tag ) {
     317                static $recursing = false;
     318                if ( ! $recursing ) {
     319                        $recursing = true;
     320                        do_action( $tag, $tag );
     321                }
     322                $recursing = false;
     323        }
     324
     325        /**
     326         * @ticket 9968
     327         * @ticket 17817
     328         */
     329        function test_action_callback_manipulation_while_running() {
     330                $tag = rand_str();
     331                $a = new MockAction();
     332                $b = new MockAction();
     333                $c = new MockAction();
     334                $d = new MockAction();
     335                $e = new MockAction();
     336
     337                add_action( $tag, array( $a, 'action' ), 11, 2 );
     338                add_action( $tag, array( $this, 'action_that_manipulates_a_running_hook' ), 12, 2 );
     339                add_action( $tag, array( $b, 'action' ), 12, 2 );
     340
     341                do_action( $tag, $tag, array( $a, $b, $c, $d, $e ) );
     342                do_action( $tag, $tag, array( $a, $b, $c, $d, $e ) );
     343
     344                $this->assertEquals( 2, $a->get_call_count(), 'callbacks should run unless otherwise instructed' );
     345                $this->assertEquals( 1, $b->get_call_count(), 'callback removed by same priority callback should still get called' );
     346                $this->assertEquals( 1, $c->get_call_count(), 'callback added by same priority callback should not get called' );
     347                $this->assertEquals( 2, $d->get_call_count(), 'callback added by earlier priority callback should get called' );
     348                $this->assertEquals( 1, $e->get_call_count(), 'callback added by later priority callback should not get called' );
     349        }
     350
     351        function action_that_manipulates_a_running_hook( $tag, $mocks ) {
     352                remove_action( $tag, array( $mocks[ 1 ], 'action' ), 12, 2 );
     353                add_action( $tag, array( $mocks[ 2 ], 'action' ), 12, 2 );
     354                add_action( $tag, array( $mocks[ 3 ], 'action' ), 13, 2 );
     355                add_action( $tag, array( $mocks[ 4 ], 'action' ), 10, 2 );
     356        }
     357
     358        /**
     359         * @ticket 17817
     360         *
     361         * This specificaly addresses the concern raised at
     362         * https://core.trac.wordpress.org/ticket/17817#comment:52
     363         */
     364        function test_remove_anonymous_callback() {
     365                $tag = rand_str();
     366                $a = new MockAction();
     367                add_action( $tag, array( $a, 'action' ), 12, 1 );
     368                $this->assertTrue( has_action( $tag ) );
     369
     370                $hook = $GLOBALS['wp_filter'][ $tag ];
     371
     372                // From http://wordpress.stackexchange.com/a/57088/6445
     373                foreach ( $hook as $priority => $filter ) {
     374                        foreach ( $filter as $identifier => $function ) {
     375                                if ( is_array( $function )
     376                                        && is_a( $function['function'][ 0 ], 'MockAction' )
     377                                        && 'action' === $function['function'][ 1 ]
     378                                ) {
     379                                        remove_filter(
     380                                                $tag,
     381                                                array( $function['function'][ 0 ], 'action' ),
     382                                                $priority
     383                                        );
     384                                }
     385                        }
     386                }
     387
     388                $this->assertFalse( has_action( $tag ) );
     389        }
     390
     391
     392        /**
     393         * Test the ArrayAccess methods of WP_Hook
     394         *
     395         * @ticket 17817
     396         */
     397        function test_array_access_of_wp_filter_global() {
     398                global $wp_filter;
     399                $tag = rand_str();
     400
     401                add_action( $tag, '__return_null', 11, 1 );
     402
     403                $this->assertTrue( isset( $wp_filter[ $tag ][ 11 ] ) );
     404                $this->assertArrayHasKey( '__return_null', $wp_filter[ $tag ][ 11 ] );
     405
     406                unset( $wp_filter[ $tag ][ 11 ] );
     407                $this->assertFalse( has_action( $tag, '__return_null' ) );
     408
     409                $wp_filter[ $tag ][ 11 ] = array( '__return_null' => array( 'function' => '__return_null', 'accepted_args' => 1 ) );
     410                $this->assertEquals( 11, has_action( $tag, '__return_null' ) );
     411        }
     412
     413        /**
    261414         * Make sure current_action() behaves as current_filter()
    262415         *
    263416         * @ticket 14994
  • tests/phpunit/tests/filters.php

     
    294294                remove_all_filters( $tag, 12 );
    295295                $this->assertFalse( has_filter( $tag ) );
    296296        }
    297 
    298         /**
    299          * @ticket 29070
    300          */
    301          function test_has_filter_doesnt_reset_wp_filter() {
    302                 add_action( 'action_test_has_filter_doesnt_reset_wp_filter', '__return_null', 1 );
    303                 add_action( 'action_test_has_filter_doesnt_reset_wp_filter', '__return_null', 2 );
    304                 add_action( 'action_test_has_filter_doesnt_reset_wp_filter', '__return_null', 3 );
    305                 add_action( 'action_test_has_filter_doesnt_reset_wp_filter', array( $this, '_action_test_has_filter_doesnt_reset_wp_filter' ), 4 );
    306 
    307                 do_action( 'action_test_has_filter_doesnt_reset_wp_filter' );
    308          }
    309          function _action_test_has_filter_doesnt_reset_wp_filter() {
    310                 global $wp_filter;
    311 
    312                 has_action( 'action_test_has_filter_doesnt_reset_wp_filter', '_function_that_doesnt_exist' );
    313 
    314                 $filters = current( $wp_filter['action_test_has_filter_doesnt_reset_wp_filter'] );
    315                 $the_ = current( $filters );
    316                 $this->assertEquals( $the_['function'], array( $this, '_action_test_has_filter_doesnt_reset_wp_filter' ) );
    317          }
    318297}
  • tests/phpunit/tests/hooks/add_filter.php

     
     1<?php
     2
     3
     4/**
     5 * Test the add_filter method of WP_Hook
     6 *
     7 * @group hooks
     8 */
     9class Tests_WP_Hook_Add_Filter extends WP_UnitTestCase {
     10
     11        public function test_add_filter_with_function() {
     12                $callback = '__return_null';
     13                $hook = new WP_Hook();
     14                $tag = rand_str();
     15                $priority = rand( 1, 100 );
     16                $accepted_args = rand( 1, 100 );
     17
     18                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     19
     20                $function_index = _wp_filter_build_unique_id( $tag, $callback, $priority );
     21                $this->assertEquals( $callback, $hook->callbacks[ $priority ][ $function_index ]['function'] );
     22                $this->assertEquals( $accepted_args, $hook->callbacks[ $priority ][ $function_index ]['accepted_args'] );
     23        }
     24
     25        public function test_add_filter_with_object() {
     26                $a = new MockAction();
     27                $callback = array( $a, 'action' );
     28                $hook = new WP_Hook();
     29                $tag = rand_str();
     30                $priority = rand( 1, 100 );
     31                $accepted_args = rand( 1, 100 );
     32
     33                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     34
     35                $function_index = _wp_filter_build_unique_id( $tag, $callback, $priority );
     36                $this->assertEquals( $callback, $hook->callbacks[ $priority ][ $function_index ]['function'] );
     37                $this->assertEquals( $accepted_args, $hook->callbacks[ $priority ][ $function_index ]['accepted_args'] );
     38        }
     39
     40        public function test_add_filter_with_static_method() {
     41                $callback = array( 'MockAction', 'action' );
     42                $hook = new WP_Hook();
     43                $tag = rand_str();
     44                $priority = rand( 1, 100 );
     45                $accepted_args = rand( 1, 100 );
     46
     47                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     48
     49                $function_index = _wp_filter_build_unique_id( $tag, $callback, $priority );
     50                $this->assertEquals( $callback, $hook->callbacks[ $priority ][ $function_index ]['function'] );
     51                $this->assertEquals( $accepted_args, $hook->callbacks[ $priority ][ $function_index ]['accepted_args'] );
     52        }
     53
     54        public function test_add_two_filters_with_same_priority() {
     55                $callback_one = '__return_null';
     56                $callback_two = '__return_false';
     57                $hook = new WP_Hook();
     58                $tag = rand_str();
     59                $priority = rand( 1, 100 );
     60                $accepted_args = rand( 1, 100 );
     61
     62                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     63                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     64
     65                $hook->add_filter( $tag, $callback_two, $priority, $accepted_args );
     66                $this->assertCount( 2, $hook->callbacks[ $priority ] );
     67        }
     68
     69        public function test_add_two_filters_with_different_priority() {
     70                $callback_one = '__return_null';
     71                $callback_two = '__return_false';
     72                $hook = new WP_Hook();
     73                $tag = rand_str();
     74                $priority = rand( 1, 100 );
     75                $accepted_args = rand( 1, 100 );
     76
     77                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     78                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     79
     80                $hook->add_filter( $tag, $callback_two, $priority + 1, $accepted_args );
     81                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     82                $this->assertCount( 1, $hook->callbacks[ $priority + 1 ] );
     83        }
     84
     85        public function test_readd_filter() {
     86                $callback = '__return_null';
     87                $hook = new WP_Hook();
     88                $tag = rand_str();
     89                $priority = rand( 1, 100 );
     90                $accepted_args = rand( 1, 100 );
     91
     92                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     93                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     94
     95                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     96                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     97        }
     98
     99        public function test_readd_filter_with_different_priority() {
     100                $callback = '__return_null';
     101                $hook = new WP_Hook();
     102                $tag = rand_str();
     103                $priority = rand( 1, 100 );
     104                $accepted_args = rand( 1, 100 );
     105
     106                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     107                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     108
     109                $hook->add_filter( $tag, $callback, $priority + 1, $accepted_args );
     110                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     111                $this->assertCount( 1, $hook->callbacks[ $priority + 1 ] );
     112        }
     113
     114        public function test_sort_after_add_filter() {
     115                $a = new MockAction();
     116                $b = new MockAction();
     117                $c = new MockAction();
     118                $hook = new WP_Hook();
     119                $tag = rand_str();
     120
     121                $hook->add_filter( $tag, array( $a, 'action' ), 10, 1 );
     122                $hook->add_filter( $tag, array( $b, 'action' ), 5, 1 );
     123                $hook->add_filter( $tag, array( $c, 'action' ), 8, 1 );
     124
     125                $this->assertEquals( array( 5, 8, 10 ), array_keys( $hook->callbacks ) );
     126        }
     127}
     128<?php
     129
     130
     131/**
     132 * Test the add_filter method of WP_Hook
     133 *
     134 * @group hooks
     135 */
     136class Tests_WP_Hook_Add_Filter extends WP_UnitTestCase {
     137
     138        public function test_add_filter_with_function() {
     139                $callback = '__return_null';
     140                $hook = new WP_Hook();
     141                $tag = rand_str();
     142                $priority = rand( 1, 100 );
     143                $accepted_args = rand( 1, 100 );
     144
     145                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     146
     147                $function_index = _wp_filter_build_unique_id( $tag, $callback, $priority );
     148                $this->assertEquals( $callback, $hook->callbacks[ $priority ][ $function_index ]['function'] );
     149                $this->assertEquals( $accepted_args, $hook->callbacks[ $priority ][ $function_index ]['accepted_args'] );
     150        }
     151
     152        public function test_add_filter_with_object() {
     153                $a = new MockAction();
     154                $callback = array( $a, 'action' );
     155                $hook = new WP_Hook();
     156                $tag = rand_str();
     157                $priority = rand( 1, 100 );
     158                $accepted_args = rand( 1, 100 );
     159
     160                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     161
     162                $function_index = _wp_filter_build_unique_id( $tag, $callback, $priority );
     163                $this->assertEquals( $callback, $hook->callbacks[ $priority ][ $function_index ]['function'] );
     164                $this->assertEquals( $accepted_args, $hook->callbacks[ $priority ][ $function_index ]['accepted_args'] );
     165        }
     166
     167        public function test_add_filter_with_static_method() {
     168                $callback = array( 'MockAction', 'action' );
     169                $hook = new WP_Hook();
     170                $tag = rand_str();
     171                $priority = rand( 1, 100 );
     172                $accepted_args = rand( 1, 100 );
     173
     174                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     175
     176                $function_index = _wp_filter_build_unique_id( $tag, $callback, $priority );
     177                $this->assertEquals( $callback, $hook->callbacks[ $priority ][ $function_index ]['function'] );
     178                $this->assertEquals( $accepted_args, $hook->callbacks[ $priority ][ $function_index ]['accepted_args'] );
     179        }
     180
     181        public function test_add_two_filters_with_same_priority() {
     182                $callback_one = '__return_null';
     183                $callback_two = '__return_false';
     184                $hook = new WP_Hook();
     185                $tag = rand_str();
     186                $priority = rand( 1, 100 );
     187                $accepted_args = rand( 1, 100 );
     188
     189                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     190                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     191
     192                $hook->add_filter( $tag, $callback_two, $priority, $accepted_args );
     193                $this->assertCount( 2, $hook->callbacks[ $priority ] );
     194        }
     195
     196        public function test_add_two_filters_with_different_priority() {
     197                $callback_one = '__return_null';
     198                $callback_two = '__return_false';
     199                $hook = new WP_Hook();
     200                $tag = rand_str();
     201                $priority = rand( 1, 100 );
     202                $accepted_args = rand( 1, 100 );
     203
     204                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     205                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     206
     207                $hook->add_filter( $tag, $callback_two, $priority + 1, $accepted_args );
     208                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     209                $this->assertCount( 1, $hook->callbacks[ $priority + 1 ] );
     210        }
     211
     212        public function test_readd_filter() {
     213                $callback = '__return_null';
     214                $hook = new WP_Hook();
     215                $tag = rand_str();
     216                $priority = rand( 1, 100 );
     217                $accepted_args = rand( 1, 100 );
     218
     219                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     220                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     221
     222                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     223                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     224        }
     225
     226        public function test_readd_filter_with_different_priority() {
     227                $callback = '__return_null';
     228                $hook = new WP_Hook();
     229                $tag = rand_str();
     230                $priority = rand( 1, 100 );
     231                $accepted_args = rand( 1, 100 );
     232
     233                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     234                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     235
     236                $hook->add_filter( $tag, $callback, $priority + 1, $accepted_args );
     237                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     238                $this->assertCount( 1, $hook->callbacks[ $priority + 1 ] );
     239        }
     240
     241        public function test_sort_after_add_filter() {
     242                $a = new MockAction();
     243                $b = new MockAction();
     244                $c = new MockAction();
     245                $hook = new WP_Hook();
     246                $tag = rand_str();
     247
     248                $hook->add_filter( $tag, array( $a, 'action' ), 10, 1 );
     249                $hook->add_filter( $tag, array( $b, 'action' ), 5, 1 );
     250                $hook->add_filter( $tag, array( $c, 'action' ), 8, 1 );
     251
     252                $this->assertEquals( array( 5, 8, 10 ), array_keys( $hook->callbacks ) );
     253        }
     254}
  • tests/phpunit/tests/hooks/apply_filters.php

     
     1<?php
     2
     3/**
     4 * Test the apply_filters method of WP_Hook
     5 *
     6 * @group hooks
     7 */
     8class Tests_WP_Hook_Apply_Filters extends WP_UnitTestCase {
     9
     10        public function test_apply_filters_with_callback() {
     11                $a = new MockAction();
     12                $callback = array( $a, 'filter' );
     13                $hook = new WP_Hook();
     14                $tag = rand_str();
     15                $priority = rand( 1, 100 );
     16                $accepted_args = rand( 1, 100 );
     17                $arg = rand_str();
     18
     19                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     20
     21                $returned = $hook->apply_filters( $arg, array( $arg ) );
     22
     23                $this->assertEquals( $returned, $arg );
     24                $this->assertEquals( 1, $a->get_call_count() );
     25        }
     26
     27        public function test_apply_filters_with_multiple_calls() {
     28                $a = new MockAction();
     29                $callback = array( $a, 'filter' );
     30                $hook = new WP_Hook();
     31                $tag = rand_str();
     32                $priority = rand( 1, 100 );
     33                $accepted_args = rand( 1, 100 );
     34                $arg = rand_str();
     35
     36                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     37
     38                $returned_one = $hook->apply_filters( $arg, array( $arg ) );
     39                $returned_two = $hook->apply_filters( $returned_one, array( $returned_one ) );
     40
     41                $this->assertEquals( $returned_two, $arg );
     42                $this->assertEquals( 2, $a->get_call_count() );
     43        }
     44
     45}
     46<?php
     47
     48/**
     49 * Test the apply_filters method of WP_Hook
     50 *
     51 * @group hooks
     52 */
     53class Tests_WP_Hook_Apply_Filters extends WP_UnitTestCase {
     54
     55        public function test_apply_filters_with_callback() {
     56                $a = new MockAction();
     57                $callback = array( $a, 'filter' );
     58                $hook = new WP_Hook();
     59                $tag = rand_str();
     60                $priority = rand( 1, 100 );
     61                $accepted_args = rand( 1, 100 );
     62                $arg = rand_str();
     63
     64                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     65
     66                $returned = $hook->apply_filters( $arg, array( $arg ) );
     67
     68                $this->assertEquals( $returned, $arg );
     69                $this->assertEquals( 1, $a->get_call_count() );
     70        }
     71
     72        public function test_apply_filters_with_multiple_calls() {
     73                $a = new MockAction();
     74                $callback = array( $a, 'filter' );
     75                $hook = new WP_Hook();
     76                $tag = rand_str();
     77                $priority = rand( 1, 100 );
     78                $accepted_args = rand( 1, 100 );
     79                $arg = rand_str();
     80
     81                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     82
     83                $returned_one = $hook->apply_filters( $arg, array( $arg ) );
     84                $returned_two = $hook->apply_filters( $returned_one, array( $returned_one ) );
     85
     86                $this->assertEquals( $returned_two, $arg );
     87                $this->assertEquals( 2, $a->get_call_count() );
     88        }
     89
     90}
  • tests/phpunit/tests/hooks/do_action.php

     
     1<?php
     2
     3/**
     4 * Test the do_action method of WP_Hook
     5 *
     6 * @group hooks
     7 */
     8class Tests_WP_Hook_Do_Action extends WP_UnitTestCase {
     9        private $events = array();
     10
     11        public function setUp() {
     12                parent::setUp();
     13                $this->events = array();
     14        }
     15
     16        public function test_do_action_with_callback() {
     17                $a = new MockAction();
     18                $callback = array( $a, 'action' );
     19                $hook = new WP_Hook();
     20                $tag = rand_str();
     21                $priority = rand( 1, 100 );
     22                $accepted_args = rand( 1, 100 );
     23                $arg = rand_str();
     24
     25                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     26                $hook->do_action( array( $arg ) );
     27
     28                $this->assertEquals( 1, $a->get_call_count() );
     29        }
     30
     31        public function test_do_action_with_multiple_calls() {
     32                $a = new MockAction();
     33                $callback = array( $a, 'filter' );
     34                $hook = new WP_Hook();
     35                $tag = rand_str();
     36                $priority = rand( 1, 100 );
     37                $accepted_args = rand( 1, 100 );
     38                $arg = rand_str();
     39
     40                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     41                $hook->do_action( array( $arg ) );
     42                $hook->do_action( array( $arg ) );
     43
     44                $this->assertEquals( 2, $a->get_call_count() );
     45        }
     46
     47        public function test_do_action_with_multiple_callbacks_on_same_priority() {
     48                $a = new MockAction();
     49                $b = new MockAction();
     50                $callback_one = array( $a, 'filter' );
     51                $callback_two = array( $b, 'filter' );
     52                $hook = new WP_Hook();
     53                $tag = rand_str();
     54                $priority = rand( 1, 100 );
     55                $accepted_args = rand( 1, 100 );
     56                $arg = rand_str();
     57
     58                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     59                $hook->add_filter( $tag, $callback_two, $priority, $accepted_args );
     60                $hook->do_action( array( $arg ) );
     61
     62                $this->assertEquals( 1, $a->get_call_count() );
     63                $this->assertEquals( 1, $a->get_call_count() );
     64        }
     65
     66        public function test_do_action_with_multiple_callbacks_on_different_priorities() {
     67                $a = new MockAction();
     68                $b = new MockAction();
     69                $callback_one = array( $a, 'filter' );
     70                $callback_two = array( $b, 'filter' );
     71                $hook = new WP_Hook();
     72                $tag = rand_str();
     73                $priority = rand( 1, 100 );
     74                $accepted_args = rand( 1, 100 );
     75                $arg = rand_str();
     76
     77                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     78                $hook->add_filter( $tag, $callback_two, $priority, $accepted_args );
     79                $hook->do_action( array( $arg ) );
     80
     81                $this->assertEquals( 1, $a->get_call_count() );
     82                $this->assertEquals( 1, $a->get_call_count() );
     83        }
     84
     85        public function test_do_action_with_no_accepted_args() {
     86                $callback = array( $this, '_action_callback' );
     87                $hook = new WP_Hook();
     88                $tag = rand_str();
     89                $priority = rand( 1, 100 );
     90                $accepted_args = 0;
     91                $arg = rand_str();
     92
     93                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     94                $hook->do_action( array( $arg ) );
     95
     96                $this->assertEmpty( $this->events[0]['args'] );
     97        }
     98
     99        public function test_do_action_with_one_accepted_arg() {
     100                $callback = array( $this, '_action_callback' );
     101                $hook = new WP_Hook();
     102                $tag = rand_str();
     103                $priority = rand( 1, 100 );
     104                $accepted_args = 1;
     105                $arg = rand_str();
     106
     107                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     108                $hook->do_action( array( $arg ) );
     109
     110                $this->assertCount( 1, $this->events[0]['args'] );
     111        }
     112
     113        public function test_do_action_with_more_accepted_args() {
     114                $callback = array( $this, '_action_callback' );
     115                $hook = new WP_Hook();
     116                $tag = rand_str();
     117                $priority = rand( 1, 100 );
     118                $accepted_args = 1000;
     119                $arg = rand_str();
     120
     121                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     122                $hook->do_action( array( $arg ) );
     123
     124                $this->assertCount( 1, $this->events[0]['args'] );
     125        }
     126
     127        /**
     128         * Use this rather than MockAction so we can test callbacks with no args
     129         */
     130        public function _action_callback() {
     131                $args = func_get_args();
     132                $this->events[] = array('action' => __FUNCTION__, 'args'=>$args);
     133        }
     134}
     135<?php
     136
     137/**
     138 * Test the do_action method of WP_Hook
     139 *
     140 * @group hooks
     141 */
     142class Tests_WP_Hook_Do_Action extends WP_UnitTestCase {
     143        private $events = array();
     144
     145        public function setUp() {
     146                parent::setUp();
     147                $this->events = array();
     148        }
     149
     150        public function test_do_action_with_callback() {
     151                $a = new MockAction();
     152                $callback = array( $a, 'action' );
     153                $hook = new WP_Hook();
     154                $tag = rand_str();
     155                $priority = rand( 1, 100 );
     156                $accepted_args = rand( 1, 100 );
     157                $arg = rand_str();
     158
     159                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     160                $hook->do_action( array( $arg ) );
     161
     162                $this->assertEquals( 1, $a->get_call_count() );
     163        }
     164
     165        public function test_do_action_with_multiple_calls() {
     166                $a = new MockAction();
     167                $callback = array( $a, 'filter' );
     168                $hook = new WP_Hook();
     169                $tag = rand_str();
     170                $priority = rand( 1, 100 );
     171                $accepted_args = rand( 1, 100 );
     172                $arg = rand_str();
     173
     174                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     175                $hook->do_action( array( $arg ) );
     176                $hook->do_action( array( $arg ) );
     177
     178                $this->assertEquals( 2, $a->get_call_count() );
     179        }
     180
     181        public function test_do_action_with_multiple_callbacks_on_same_priority() {
     182                $a = new MockAction();
     183                $b = new MockAction();
     184                $callback_one = array( $a, 'filter' );
     185                $callback_two = array( $b, 'filter' );
     186                $hook = new WP_Hook();
     187                $tag = rand_str();
     188                $priority = rand( 1, 100 );
     189                $accepted_args = rand( 1, 100 );
     190                $arg = rand_str();
     191
     192                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     193                $hook->add_filter( $tag, $callback_two, $priority, $accepted_args );
     194                $hook->do_action( array( $arg ) );
     195
     196                $this->assertEquals( 1, $a->get_call_count() );
     197                $this->assertEquals( 1, $a->get_call_count() );
     198        }
     199
     200        public function test_do_action_with_multiple_callbacks_on_different_priorities() {
     201                $a = new MockAction();
     202                $b = new MockAction();
     203                $callback_one = array( $a, 'filter' );
     204                $callback_two = array( $b, 'filter' );
     205                $hook = new WP_Hook();
     206                $tag = rand_str();
     207                $priority = rand( 1, 100 );
     208                $accepted_args = rand( 1, 100 );
     209                $arg = rand_str();
     210
     211                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     212                $hook->add_filter( $tag, $callback_two, $priority, $accepted_args );
     213                $hook->do_action( array( $arg ) );
     214
     215                $this->assertEquals( 1, $a->get_call_count() );
     216                $this->assertEquals( 1, $a->get_call_count() );
     217        }
     218
     219        public function test_do_action_with_no_accepted_args() {
     220                $callback = array( $this, '_action_callback' );
     221                $hook = new WP_Hook();
     222                $tag = rand_str();
     223                $priority = rand( 1, 100 );
     224                $accepted_args = 0;
     225                $arg = rand_str();
     226
     227                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     228                $hook->do_action( array( $arg ) );
     229
     230                $this->assertEmpty( $this->events[0]['args'] );
     231        }
     232
     233        public function test_do_action_with_one_accepted_arg() {
     234                $callback = array( $this, '_action_callback' );
     235                $hook = new WP_Hook();
     236                $tag = rand_str();
     237                $priority = rand( 1, 100 );
     238                $accepted_args = 1;
     239                $arg = rand_str();
     240
     241                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     242                $hook->do_action( array( $arg ) );
     243
     244                $this->assertCount( 1, $this->events[0]['args'] );
     245        }
     246
     247        public function test_do_action_with_more_accepted_args() {
     248                $callback = array( $this, '_action_callback' );
     249                $hook = new WP_Hook();
     250                $tag = rand_str();
     251                $priority = rand( 1, 100 );
     252                $accepted_args = 1000;
     253                $arg = rand_str();
     254
     255                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     256                $hook->do_action( array( $arg ) );
     257
     258                $this->assertCount( 1, $this->events[0]['args'] );
     259        }
     260
     261        /**
     262         * Use this rather than MockAction so we can test callbacks with no args
     263         */
     264        public function _action_callback() {
     265                $args = func_get_args();
     266                $this->events[] = array('action' => __FUNCTION__, 'args'=>$args);
     267        }
     268}
  • tests/phpunit/tests/hooks/do_all_hook.php

     
     1<?php
     2
     3/**
     4 * Test the do_all_hook method of WP_Hook
     5 *
     6 * @group hooks
     7 */
     8class Tests_WP_Hook_Do_All_Hook extends WP_UnitTestCase {
     9
     10        public function test_do_all_hook_with_multiple_calls() {
     11                $a = new MockAction();
     12                $callback = array( $a, 'action' );
     13                $hook = new WP_Hook();
     14                $tag = 'all';
     15                $priority = rand( 1, 100 );
     16                $accepted_args = rand( 1, 100 );
     17                $arg = rand_str();
     18
     19                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     20                $args = array( $arg );
     21                $hook->do_all_hook( $args );
     22                $hook->do_all_hook( $args );
     23
     24                $this->assertEquals( 2, $a->get_call_count() );
     25        }
     26}
     27<?php
     28
     29/**
     30 * Test the do_all_hook method of WP_Hook
     31 *
     32 * @group hooks
     33 */
     34class Tests_WP_Hook_Do_All_Hook extends WP_UnitTestCase {
     35
     36        public function test_do_all_hook_with_multiple_calls() {
     37                $a = new MockAction();
     38                $callback = array( $a, 'action' );
     39                $hook = new WP_Hook();
     40                $tag = 'all';
     41                $priority = rand( 1, 100 );
     42                $accepted_args = rand( 1, 100 );
     43                $arg = rand_str();
     44
     45                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     46                $args = array( $arg );
     47                $hook->do_all_hook( $args );
     48                $hook->do_all_hook( $args );
     49
     50                $this->assertEquals( 2, $a->get_call_count() );
     51        }
     52}
  • tests/phpunit/tests/hooks/has_filter.php

     
     1<?php
     2
     3/**
     4 * Test the has_filter method of WP_Hook
     5 *
     6 * @group hooks
     7 */
     8class Tests_WP_Hook_Has_Filter extends WP_UnitTestCase {
     9       
     10        public function test_has_filter_with_function() {
     11                $callback = '__return_null';
     12                $hook = new WP_Hook();
     13                $tag = rand_str();
     14                $priority = rand( 1, 100 );
     15                $accepted_args = rand( 1, 100 );
     16
     17                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     18
     19                $this->assertEquals( $priority, $hook->has_filter( $tag, $callback ) );
     20        }
     21
     22        public function test_has_filter_with_object() {
     23                $a = new MockAction();
     24                $callback = array( $a, 'action' );
     25                $hook = new WP_Hook();
     26                $tag = rand_str();
     27                $priority = rand( 1, 100 );
     28                $accepted_args = rand( 1, 100 );
     29
     30                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     31
     32                $this->assertEquals( $priority, $hook->has_filter( $tag, $callback ) );
     33        }
     34
     35        public function test_has_filter_with_static_method() {
     36                $callback = array( 'MockAction', 'action' );
     37                $hook = new WP_Hook();
     38                $tag = rand_str();
     39                $priority = rand( 1, 100 );
     40                $accepted_args = rand( 1, 100 );
     41
     42                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     43
     44                $this->assertEquals( $priority, $hook->has_filter( $tag, $callback ) );
     45        }
     46
     47        public function test_has_filter_without_callback() {
     48                $callback = '__return_null';
     49                $hook = new WP_Hook();
     50                $tag = rand_str();
     51                $priority = rand( 1, 100 );
     52                $accepted_args = rand( 1, 100 );
     53
     54                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     55
     56                $this->assertTrue( $hook->has_filter() );
     57        }
     58
     59        public function test_not_has_filter_without_callback() {
     60                $hook = new WP_Hook();
     61                $this->assertFalse( $hook->has_filter() );
     62        }
     63
     64        public function test_not_has_filter_with_callback() {
     65                $callback = '__return_null';
     66                $hook = new WP_Hook();
     67                $tag = rand_str();
     68
     69                $this->assertFalse( $hook->has_filter( $tag, $callback ) );
     70        }
     71
     72        public function test_has_filter_with_wrong_callback() {
     73                $callback = '__return_null';
     74                $hook = new WP_Hook();
     75                $tag = rand_str();
     76                $priority = rand( 1, 100 );
     77                $accepted_args = rand( 1, 100 );
     78
     79                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     80
     81                $this->assertFalse( $hook->has_filter( $tag, '__return_false' ) );
     82        }
     83}
     84<?php
     85
     86/**
     87 * Test the has_filter method of WP_Hook
     88 *
     89 * @group hooks
     90 */
     91class Tests_WP_Hook_Has_Filter extends WP_UnitTestCase {
     92       
     93        public function test_has_filter_with_function() {
     94                $callback = '__return_null';
     95                $hook = new WP_Hook();
     96                $tag = rand_str();
     97                $priority = rand( 1, 100 );
     98                $accepted_args = rand( 1, 100 );
     99
     100                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     101
     102                $this->assertEquals( $priority, $hook->has_filter( $tag, $callback ) );
     103        }
     104
     105        public function test_has_filter_with_object() {
     106                $a = new MockAction();
     107                $callback = array( $a, 'action' );
     108                $hook = new WP_Hook();
     109                $tag = rand_str();
     110                $priority = rand( 1, 100 );
     111                $accepted_args = rand( 1, 100 );
     112
     113                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     114
     115                $this->assertEquals( $priority, $hook->has_filter( $tag, $callback ) );
     116        }
     117
     118        public function test_has_filter_with_static_method() {
     119                $callback = array( 'MockAction', 'action' );
     120                $hook = new WP_Hook();
     121                $tag = rand_str();
     122                $priority = rand( 1, 100 );
     123                $accepted_args = rand( 1, 100 );
     124
     125                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     126
     127                $this->assertEquals( $priority, $hook->has_filter( $tag, $callback ) );
     128        }
     129
     130        public function test_has_filter_without_callback() {
     131                $callback = '__return_null';
     132                $hook = new WP_Hook();
     133                $tag = rand_str();
     134                $priority = rand( 1, 100 );
     135                $accepted_args = rand( 1, 100 );
     136
     137                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     138
     139                $this->assertTrue( $hook->has_filter() );
     140        }
     141
     142        public function test_not_has_filter_without_callback() {
     143                $hook = new WP_Hook();
     144                $this->assertFalse( $hook->has_filter() );
     145        }
     146
     147        public function test_not_has_filter_with_callback() {
     148                $callback = '__return_null';
     149                $hook = new WP_Hook();
     150                $tag = rand_str();
     151
     152                $this->assertFalse( $hook->has_filter( $tag, $callback ) );
     153        }
     154
     155        public function test_has_filter_with_wrong_callback() {
     156                $callback = '__return_null';
     157                $hook = new WP_Hook();
     158                $tag = rand_str();
     159                $priority = rand( 1, 100 );
     160                $accepted_args = rand( 1, 100 );
     161
     162                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     163
     164                $this->assertFalse( $hook->has_filter( $tag, '__return_false' ) );
     165        }
     166}
  • tests/phpunit/tests/hooks/has_filters.php

     
     1<?php
     2
     3/**
     4 * Test the has_filters method of WP_Hook
     5 *
     6 * @group hooks
     7 */
     8class Tests_WP_Hook_Has_Filters extends WP_UnitTestCase {
     9       
     10        public function test_has_filters_with_callback() {
     11                $callback = '__return_null';
     12                $hook = new WP_Hook();
     13                $tag = rand_str();
     14                $priority = rand( 1, 100 );
     15                $accepted_args = rand( 1, 100 );
     16
     17                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     18
     19                $this->assertTrue( $hook->has_filters() );
     20        }
     21
     22        public function test_has_filters_without_callback() {
     23                $hook = new WP_Hook();
     24                $this->assertFalse( $hook->has_filters() );
     25        }
     26
     27        public function test_not_has_filters_with_removed_callback() {
     28                $callback = '__return_null';
     29                $hook = new WP_Hook();
     30                $tag = rand_str();
     31                $priority = rand( 1, 100 );
     32                $accepted_args = rand( 1, 100 );
     33
     34                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     35                $hook->remove_filter( $tag, $callback, $priority );
     36                $this->assertFalse( $hook->has_filters() );
     37        }
     38
     39        public function test_not_has_filter_with_directly_removed_callback() {
     40                $callback = '__return_null';
     41                $hook = new WP_Hook();
     42                $tag = rand_str();
     43                $priority = rand( 1, 100 );
     44                $accepted_args = rand( 1, 100 );
     45
     46                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     47                $function_key = _wp_filter_build_unique_id( $tag, $callback, $priority );
     48                unset( $hook->callbacks[ $priority ][ $function_key ] );
     49
     50                $this->assertFalse( $hook->has_filters() );
     51        }
     52}
     53<?php
     54
     55/**
     56 * Test the has_filters method of WP_Hook
     57 *
     58 * @group hooks
     59 */
     60class Tests_WP_Hook_Has_Filters extends WP_UnitTestCase {
     61       
     62        public function test_has_filters_with_callback() {
     63                $callback = '__return_null';
     64                $hook = new WP_Hook();
     65                $tag = rand_str();
     66                $priority = rand( 1, 100 );
     67                $accepted_args = rand( 1, 100 );
     68
     69                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     70
     71                $this->assertTrue( $hook->has_filters() );
     72        }
     73
     74        public function test_has_filters_without_callback() {
     75                $hook = new WP_Hook();
     76                $this->assertFalse( $hook->has_filters() );
     77        }
     78
     79        public function test_not_has_filters_with_removed_callback() {
     80                $callback = '__return_null';
     81                $hook = new WP_Hook();
     82                $tag = rand_str();
     83                $priority = rand( 1, 100 );
     84                $accepted_args = rand( 1, 100 );
     85
     86                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     87                $hook->remove_filter( $tag, $callback, $priority );
     88                $this->assertFalse( $hook->has_filters() );
     89        }
     90
     91        public function test_not_has_filter_with_directly_removed_callback() {
     92                $callback = '__return_null';
     93                $hook = new WP_Hook();
     94                $tag = rand_str();
     95                $priority = rand( 1, 100 );
     96                $accepted_args = rand( 1, 100 );
     97
     98                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     99                $function_key = _wp_filter_build_unique_id( $tag, $callback, $priority );
     100                unset( $hook->callbacks[ $priority ][ $function_key ] );
     101
     102                $this->assertFalse( $hook->has_filters() );
     103        }
     104}
  • tests/phpunit/tests/hooks/iterator.php

     
     1<?php
     2
     3/**
     4 * Test the Iterator implementation of WP_Hook
     5 *
     6 * @group hooks
     7 */
     8class Tests_WP_Hook_Iterator extends WP_UnitTestCase {
     9        public function test_foreach() {
     10                $callback_one = '__return_null';
     11                $callback_two = '__return_false';
     12                $hook = new WP_Hook();
     13                $tag = rand_str();
     14                $priority = rand( 1, 100 );
     15                $accepted_args = rand( 1, 100 );
     16
     17                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     18                $hook->add_filter( $tag, $callback_two, $priority + 1, $accepted_args );
     19
     20                $functions = array();
     21                $priorities = array();
     22                foreach ( $hook as $key => $callbacks ) {
     23                        $priorities[] = $key;
     24                        foreach ( $callbacks as $function_index => $the_ ) {
     25                                $functions[] = $the_['function'];
     26                        }
     27                }
     28                $this->assertEqualSets( array( $priority, $priority + 1 ), $priorities );
     29                $this->assertEqualSets( array( $callback_one, $callback_two ), $functions );
     30        }
     31}
     32<?php
     33
     34/**
     35 * Test the Iterator implementation of WP_Hook
     36 *
     37 * @group hooks
     38 */
     39class Tests_WP_Hook_Iterator extends WP_UnitTestCase {
     40        public function test_foreach() {
     41                $callback_one = '__return_null';
     42                $callback_two = '__return_false';
     43                $hook = new WP_Hook();
     44                $tag = rand_str();
     45                $priority = rand( 1, 100 );
     46                $accepted_args = rand( 1, 100 );
     47
     48                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     49                $hook->add_filter( $tag, $callback_two, $priority + 1, $accepted_args );
     50
     51                $functions = array();
     52                $priorities = array();
     53                foreach ( $hook as $key => $callbacks ) {
     54                        $priorities[] = $key;
     55                        foreach ( $callbacks as $function_index => $the_ ) {
     56                                $functions[] = $the_['function'];
     57                        }
     58                }
     59                $this->assertEqualSets( array( $priority, $priority + 1 ), $priorities );
     60                $this->assertEqualSets( array( $callback_one, $callback_two ), $functions );
     61        }
     62}
  • tests/phpunit/tests/hooks/preinit_hooks.php

     
     1<?php
     2
     3/**
     4 * Test the IteratorAggregate implementation of WP_Hook
     5 *
     6 * @group hooks
     7 */
     8class Tests_WP_Hook_Preinit_Hooks extends WP_UnitTestCase {
     9
     10        public function test_array_to_hooks() {
     11                $tag1 = rand_str();
     12                $priority1 = rand( 1, 100 );
     13                $tag2 = rand_str();
     14                $priority2 = rand( 1, 100 );
     15                $filters = array(
     16                        $tag1 => array(
     17                                $priority1 => array(
     18                                        'test1' => array(
     19                                                'function' => '__return_false',
     20                                                'accepted_args' => 2,
     21                                        ),
     22                                ),
     23                        ),
     24                        $tag2 => array(
     25                                $priority2 => array(
     26                                        'test1' => array(
     27                                                'function' => '__return_null',
     28                                                'accepted_args' => 1,
     29                                        ),
     30                                ),
     31                        ),
     32                );
     33
     34                $hooks = WP_Hook::build_preinitialized_hooks( $filters );
     35
     36                $this->assertEquals( $priority1, $hooks[ $tag1 ]->has_filter( $tag1, '__return_false' ) );
     37                $this->assertEquals( $priority2, $hooks[ $tag2 ]->has_filter( $tag2, '__return_null' ) );
     38        }
     39}
     40<?php
     41
     42/**
     43 * Test the IteratorAggregate implementation of WP_Hook
     44 *
     45 * @group hooks
     46 */
     47class Tests_WP_Hook_Preinit_Hooks extends WP_UnitTestCase {
     48
     49        public function test_array_to_hooks() {
     50                $tag1 = rand_str();
     51                $priority1 = rand( 1, 100 );
     52                $tag2 = rand_str();
     53                $priority2 = rand( 1, 100 );
     54                $filters = array(
     55                        $tag1 => array(
     56                                $priority1 => array(
     57                                        'test1' => array(
     58                                                'function' => '__return_false',
     59                                                'accepted_args' => 2,
     60                                        ),
     61                                ),
     62                        ),
     63                        $tag2 => array(
     64                                $priority2 => array(
     65                                        'test1' => array(
     66                                                'function' => '__return_null',
     67                                                'accepted_args' => 1,
     68                                        ),
     69                                ),
     70                        ),
     71                );
     72
     73                $hooks = WP_Hook::build_preinitialized_hooks( $filters );
     74
     75                $this->assertEquals( $priority1, $hooks[ $tag1 ]->has_filter( $tag1, '__return_false' ) );
     76                $this->assertEquals( $priority2, $hooks[ $tag2 ]->has_filter( $tag2, '__return_null' ) );
     77        }
     78}
  • tests/phpunit/tests/hooks/remove_all_filters.php

     
     1<?php
     2
     3/**
     4 * Test the remove_all_filters method of WP_Hook
     5 *
     6 * @group hooks
     7 */
     8class Tests_WP_Hook_Remove_All_Filters extends WP_UnitTestCase {
     9       
     10        public function test_remove_all_filters() {
     11                $callback = '__return_null';
     12                $hook = new WP_Hook();
     13                $tag = rand_str();
     14                $priority = rand( 1, 100 );
     15                $accepted_args = rand( 1, 100 );
     16
     17                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     18
     19                $hook->remove_all_filters();
     20
     21                $this->assertFalse( $hook->has_filters() );
     22        }
     23
     24        public function test_remove_all_filters_with_priority() {
     25                $callback_one = '__return_null';
     26                $callback_two = '__return_false';
     27                $hook = new WP_Hook();
     28                $tag = rand_str();
     29                $priority = rand( 1, 100 );
     30                $accepted_args = rand( 1, 100 );
     31
     32                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     33                $hook->add_filter( $tag, $callback_two, $priority + 1, $accepted_args );
     34
     35                $hook->remove_all_filters( $priority );
     36
     37                $this->assertFalse( $hook->has_filter( $tag, $callback_one ) );
     38                $this->assertTrue( $hook->has_filters() );
     39                $this->assertEquals( $priority + 1, $hook->has_filter( $tag, $callback_two ) );
     40        }
     41}
     42<?php
     43
     44/**
     45 * Test the remove_all_filters method of WP_Hook
     46 *
     47 * @group hooks
     48 */
     49class Tests_WP_Hook_Remove_All_Filters extends WP_UnitTestCase {
     50       
     51        public function test_remove_all_filters() {
     52                $callback = '__return_null';
     53                $hook = new WP_Hook();
     54                $tag = rand_str();
     55                $priority = rand( 1, 100 );
     56                $accepted_args = rand( 1, 100 );
     57
     58                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     59
     60                $hook->remove_all_filters();
     61
     62                $this->assertFalse( $hook->has_filters() );
     63        }
     64
     65        public function test_remove_all_filters_with_priority() {
     66                $callback_one = '__return_null';
     67                $callback_two = '__return_false';
     68                $hook = new WP_Hook();
     69                $tag = rand_str();
     70                $priority = rand( 1, 100 );
     71                $accepted_args = rand( 1, 100 );
     72
     73                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     74                $hook->add_filter( $tag, $callback_two, $priority + 1, $accepted_args );
     75
     76                $hook->remove_all_filters( $priority );
     77
     78                $this->assertFalse( $hook->has_filter( $tag, $callback_one ) );
     79                $this->assertTrue( $hook->has_filters() );
     80                $this->assertEquals( $priority + 1, $hook->has_filter( $tag, $callback_two ) );
     81        }
     82}
  • tests/phpunit/tests/hooks/remove_filter.php

     
     1<?php
     2
     3/**
     4 * Test the remove_filter method of WP_Hook
     5 *
     6 * @group hooks
     7 */
     8class Tests_WP_Hook_Remove_Filter extends WP_UnitTestCase {
     9       
     10        public function test_remove_filter_with_function() {
     11                $callback = '__return_null';
     12                $hook = new WP_Hook();
     13                $tag = rand_str();
     14                $priority = rand( 1, 100 );
     15                $accepted_args = rand( 1, 100 );
     16
     17                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     18                $hook->remove_filter( $tag, $callback, $priority );
     19
     20                $this->assertFalse( isset( $hook->callbacks[ $priority ] ) );
     21        }
     22
     23        public function test_remove_filter_with_object() {
     24                $a = new MockAction();
     25                $callback = array( $a, 'action' );
     26                $hook = new WP_Hook();
     27                $tag = rand_str();
     28                $priority = rand( 1, 100 );
     29                $accepted_args = rand( 1, 100 );
     30
     31                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     32                $hook->remove_filter( $tag, $callback, $priority );
     33
     34                $this->assertFalse( isset( $hook->callbacks[ $priority ] ) );
     35        }
     36
     37        public function test_remove_filter_with_static_method() {
     38                $callback = array( 'MockAction', 'action' );
     39                $hook = new WP_Hook();
     40                $tag = rand_str();
     41                $priority = rand( 1, 100 );
     42                $accepted_args = rand( 1, 100 );
     43
     44                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     45                $hook->remove_filter( $tag, $callback, $priority );
     46
     47                $this->assertFalse( isset( $hook->callbacks[ $priority ] ) );
     48        }
     49
     50        public function test_remove_filters_with_another_at_same_priority() {
     51                $callback_one = '__return_null';
     52                $callback_two = '__return_false';
     53                $hook = new WP_Hook();
     54                $tag = rand_str();
     55                $priority = rand( 1, 100 );
     56                $accepted_args = rand( 1, 100 );
     57
     58                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     59                $hook->add_filter( $tag, $callback_two, $priority, $accepted_args );
     60
     61                $hook->remove_filter( $tag, $callback_one, $priority );
     62
     63                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     64        }
     65
     66        public function test_remove_filter_with_another_at_different_priority() {
     67                $callback_one = '__return_null';
     68                $callback_two = '__return_false';
     69                $hook = new WP_Hook();
     70                $tag = rand_str();
     71                $priority = rand( 1, 100 );
     72                $accepted_args = rand( 1, 100 );
     73
     74                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     75                $hook->add_filter( $tag, $callback_two, $priority + 1, $accepted_args );
     76
     77                $hook->remove_filter( $tag, $callback_one, $priority );
     78                $this->assertFalse( isset( $hook->callbacks[ $priority ] ) );
     79                $this->assertCount( 1, $hook->callbacks[ $priority + 1 ] );
     80        }
     81}
     82<?php
     83
     84/**
     85 * Test the remove_filter method of WP_Hook
     86 *
     87 * @group hooks
     88 */
     89class Tests_WP_Hook_Remove_Filter extends WP_UnitTestCase {
     90       
     91        public function test_remove_filter_with_function() {
     92                $callback = '__return_null';
     93                $hook = new WP_Hook();
     94                $tag = rand_str();
     95                $priority = rand( 1, 100 );
     96                $accepted_args = rand( 1, 100 );
     97
     98                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     99                $hook->remove_filter( $tag, $callback, $priority );
     100
     101                $this->assertFalse( isset( $hook->callbacks[ $priority ] ) );
     102        }
     103
     104        public function test_remove_filter_with_object() {
     105                $a = new MockAction();
     106                $callback = array( $a, 'action' );
     107                $hook = new WP_Hook();
     108                $tag = rand_str();
     109                $priority = rand( 1, 100 );
     110                $accepted_args = rand( 1, 100 );
     111
     112                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     113                $hook->remove_filter( $tag, $callback, $priority );
     114
     115                $this->assertFalse( isset( $hook->callbacks[ $priority ] ) );
     116        }
     117
     118        public function test_remove_filter_with_static_method() {
     119                $callback = array( 'MockAction', 'action' );
     120                $hook = new WP_Hook();
     121                $tag = rand_str();
     122                $priority = rand( 1, 100 );
     123                $accepted_args = rand( 1, 100 );
     124
     125                $hook->add_filter( $tag, $callback, $priority, $accepted_args );
     126                $hook->remove_filter( $tag, $callback, $priority );
     127
     128                $this->assertFalse( isset( $hook->callbacks[ $priority ] ) );
     129        }
     130
     131        public function test_remove_filters_with_another_at_same_priority() {
     132                $callback_one = '__return_null';
     133                $callback_two = '__return_false';
     134                $hook = new WP_Hook();
     135                $tag = rand_str();
     136                $priority = rand( 1, 100 );
     137                $accepted_args = rand( 1, 100 );
     138
     139                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     140                $hook->add_filter( $tag, $callback_two, $priority, $accepted_args );
     141
     142                $hook->remove_filter( $tag, $callback_one, $priority );
     143
     144                $this->assertCount( 1, $hook->callbacks[ $priority ] );
     145        }
     146
     147        public function test_remove_filter_with_another_at_different_priority() {
     148                $callback_one = '__return_null';
     149                $callback_two = '__return_false';
     150                $hook = new WP_Hook();
     151                $tag = rand_str();
     152                $priority = rand( 1, 100 );
     153                $accepted_args = rand( 1, 100 );
     154
     155                $hook->add_filter( $tag, $callback_one, $priority, $accepted_args );
     156                $hook->add_filter( $tag, $callback_two, $priority + 1, $accepted_args );
     157
     158                $hook->remove_filter( $tag, $callback_one, $priority );
     159                $this->assertFalse( isset( $hook->callbacks[ $priority ] ) );
     160                $this->assertCount( 1, $hook->callbacks[ $priority + 1 ] );
     161        }
     162}