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

9

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

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 состояла в том, чтобы включить файл, в котором класс объявлен в functions.php:

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

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

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

Игорь Сколдин
источник

Ответы:

8

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

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' ) );

}
passatgt
источник
1
есть функция remove_action: codex.wordpress.org/Function_Reference/remove_action
Alex Older
Да, это я пропустил, у этого плагина есть переменная, к которой можно обращаться как к глобальной. Моя глупость Спасибо за ваш ответ, это работает в данном конкретном случае (для этого плагина).
Игорь Сколдин
Алекс Олдер связан с местом, которое объясняет, почему его ответ работает. Remove_action принимает массив со статическим классом или классом экземпляра, внутри которого вы хотите удалить метод.
ninja08,
11

Если ваш плагин зарегистрирован так:

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

  function test_method() {
    echo 'Echoed from the parent';
  }
}
$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 'Echoed from the child';
  }
}
$c = new Test_Class_Child();

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

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 'Echoed from the child';
  }
}
$c = new Test_Class_Child();

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

s_ha_dum
источник
2
Это должен быть принятый ответ. Это в целом полезно, и не ограничивается только конкретным случаем OP.
Дэвид Р.
1

Этот плагин делает свою функцию инициализации wcpgsk_init()подключаемой, поэтому другой способ переопределить его - сначала определить его в плагине, который необходимо использовать (так как это слишком поздно в "functions.php" вашей темы). Таким образом, вы можете поместить свое переопределение в «wp-content / mu-plugins / functions.php»:

function wcpgsk_init() {
    global $wcpgsk, $wcpgsk_about, $wcpgsk_options, $wcpgsk_session, $wcpgsk_woocommerce_active;    
    //only continue loading
    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'; // Fake __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;

        // Your override.
        class My_WCPGSK_Main extends WCPGSK_Main {
            public function wcpgsk_email_after_order_table($order) {
                echo "O la la";
            }
        }
        define( 'WCRGSK_DOMAIN', WCPGSK_DOMAIN ); // Fix typo! (WooCommerce Rich Guys Swiss Knife?)

        //load into our global
        $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
    );
} );

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

Бонгер
источник