Как заменить функцию, объявленную внутри класса плагина, в functions.php?

29 июл. 2015 г., 18:25:43
Просмотры: 33.5K
Голосов: 11

Я хочу изменить функцию в плагине. Она объявлена в главном файле плагина следующим образом:

class WCPGSK_Main {
  ...
  public function wcpgsk_email_after_order_table($order) {
    ...
  }
}

И вызывается оттуда так:

add_action( 'woocommerce_email_after_order_table', array($this, 'wcpgsk_email_after_order_table') );

Я думаю, было бы возможно заменить её, если бы имелся доступ к классу в functions.php. Тогда я мог бы написать что-то вроде:

$wcpgsk = new WCPGSK_Main;

remove_action( 'woocommerce_email_after_order_table', array($wcpgsk, 'wcpgsk_email_after_order_table') );

function customized_wcpgsk_email_after_order_table($order) {
  ...
}
add_action( 'woocommerce_email_after_order_table', array($wcpgsk, 'customized_wcpgsk_email_after_order_table') );

Моя идея получить доступ к классу в файле functions.php заключалась в том, чтобы подключить файл, где объявлен класс:

require_once('/wp-content/plugins/woocommerce-poor-guys-swiss-knife/woocommerce-poor-guys-swiss-knife.php');
$wcpgsk = new WCPGSK_Main;
...

Но это не работает, потому что файл плагина включается при инициализации плагина в WordPress, как я полагаю.

Есть ли способ переписать функцию без изменения файлов плагина?

0
Все ответы на вопрос 3
1
12

Если ваш плагин зарегистрирован следующим образом:

class Test_Class_Parent {
  function __construct() {
    add_action('wp_head',array($this,'test_method'));
  }

  function test_method() {
    echo 'Вывод из родительского класса';
  }
}
$p = new Test_Class_Parent();

Тогда вы можете удалить фильтр, обратившись к глобальной переменной:

class Test_Class_Child extends Test_Class_Parent {
  function __construct() {
    $this->unregister_parent_hook();
    add_action('wp_head',array($this,'test_method'));
  }

  function unregister_parent_hook() {
    global $p;
    remove_action('wp_head',array($p,'test_method'));
  }

  function test_method() {
    echo 'Вывод из дочернего класса';
  }
}
$c = new Test_Class_Child();

В противном случае вам придется анализировать глобальную переменную $wp_filter для поиска ключа регистрации:

class Test_Class_Child extends Test_Class_Parent {
  function __construct() {
    $this->unregister_parent_hook();
    add_action('wp_head',array($this,'test_method'));
  }

  function unregister_parent_hook() {
    global $wp_filter;
    if (!empty($wp_filter['wp_head'])) {
      foreach($wp_filter['wp_head'] as $cb) {
        foreach ($cb as $k => $v) {
          if (
            isset($v['function'])
            && is_a($v['function'][0],'Test_Class_Parent')
            && isset($v['function'][1])
            && 'test_method' == $v['function'][1]
          ) {
            remove_action('wp_head',$k);
          }
        }
      }
    }
  }

  function test_method() {
    echo 'Вывод из дочернего класса';
  }
}
$c = new Test_Class_Child();

Этот метод требует значительных ресурсов, и его следует использовать только в крайнем случае.

29 июл. 2015 г. 21:19:19
Комментарии

Этот ответ должен быть принятым. Он более универсален и полезен в общем случае, а не только в конкретной ситуации автора вопроса.

David R. David R.
23 авг. 2016 г. 13:45:01
3

Это должно работать:

add_action( 'woocommerce_init', 'remove_wcpgsk_email_order_table' );
function remove_wcpgsk_email_order_table() {

    global $wcpgsk;
    remove_action( 'woocommerce_email_after_order_table', array( $wcpgsk, 'wcpgsk_email_after_order_table' ) );

}
29 июл. 2015 г. 18:50:10
Комментарии

есть функция remove_action: https://codex.wordpress.org/Function_Reference/remove_action

Alex Older Alex Older
29 июл. 2015 г. 19:10:47

Да, это я упустил, у этого плагина есть переменная, к которой можно получить доступ как к глобальной. Моя глупость. Спасибо за ответ, это работает в данном конкретном случае (для этого плагина).

Igor Skoldin Igor Skoldin
30 июл. 2015 г. 19:18:10

Alex Older привел ссылку на объяснение, почему его ответ работает. Функция remove_action принимает массив со статическим классом или экземпляром класса, из которого вы хотите удалить метод.

ninja08 ninja08
7 мар. 2017 г. 23:36:57
0

Этот плагин делает свою функцию инициализации wcpgsk_init() заменяемой (pluggable), поэтому другой способ переопределить её — определить её первой в must-use плагине (так как в "functions.php" вашей темы уже слишком поздно). Вы можете поместить ваше переопределение в "wp-content/mu-plugins/functions.php":

function wcpgsk_init() {
    global $wcpgsk, $wcpgsk_about, $wcpgsk_options, $wcpgsk_session, $wcpgsk_woocommerce_active;    
    //продолжаем загрузку только если
    if ( $wcpgsk_woocommerce_active && version_compare( WOOCOMMERCE_VERSION, "2.0" ) >= 0 ) {
        $FILE = WP_PLUGIN_DIR . '/woocommerce-poor-guys-swiss-knife/woocommerce-poor-guys-swiss-knife.php'; // Поддельный __FILE__
        $dirname = dirname( $FILE ) . '/';
        $wcpgsk_options = get_option('wcpgsk_settings', true);
        require_once( $dirname . 'classes/woocommerce-poor-guys-swiss-knife.php' );
        require_once( $dirname . 'classes/woocommerce-poor-guys-swiss-knife-about.php' );   
        require_once( $dirname . 'wcpgsk-af.php' );

        if ( !is_admin() ) :
            add_action( 'plugins_loaded', 'wcpgsk_load_wcsession_helper' );
        endif;

        // Ваше переопределение.
        class My_WCPGSK_Main extends WCPGSK_Main {
            public function wcpgsk_email_after_order_table($order) {
                echo "O la la";
            }
        }
        define( 'WCRGSK_DOMAIN', WCPGSK_DOMAIN ); // Исправление опечатки! (WooCommerce Rich Guys Swiss Knife?)

        //загружаем в нашу глобальную переменную
        $wcpgsk = new My_WCPGSK_Main( $FILE );
        $wcpgsk->version = '2.2.4'; 
        $wcpgsk->wcpgsk_hook_woocommerce_filters();


    } elseif ( version_compare( WOOCOMMERCE_VERSION, "2.0" ) < 0 ) {
        add_action( 'admin_notices', 'wcpgsk_woocommerce_version_message', 0 ) ;    
        return;
    } else {
        return;
    }
}

Но ещё лучший способ переопределить её — установить runkit (https://github.com/padraic/runkit) и затем просто заменить её напрямую в "functions.php" вашей темы:

add_action( 'init', function () {
    $code = <<<'EOD'
echo "O la la";
EOD;
    runkit_method_redefine(
        'WCPGSK_Main',
        'wcpgsk_email_after_order_table',
        '$order',
        $code,
        RUNKIT_ACC_PUBLIC
    );
} );

(Это шутка, кстати.)

30 июл. 2015 г. 10:05:52