define_constants(); register_activation_hook( WC_ADMIN_PLUGIN_FILE, array( $this, 'on_activation' ) ); add_action( 'plugins_loaded', array( $this, 'on_plugins_loaded' ) ); add_filter( 'action_scheduler_store_class', array( $this, 'replace_actionscheduler_store_class' ) ); } /** * Install DB and create cron events when activated. * * @return void */ public function on_activation() { require_once WC_ADMIN_ABSPATH . 'includes/class-wc-admin-install.php'; WC_Admin_Install::create_tables(); WC_Admin_Install::create_events(); } /** * Setup plugin once all other plugins are loaded. * * @return void */ public function on_plugins_loaded() { $this->load_plugin_textdomain(); if ( ! $this->check_dependencies() ) { add_action( 'admin_init', array( $this, 'deactivate_self' ) ); add_action( 'admin_notices', array( $this, 'render_dependencies_notice' ) ); return; } if ( ! $this->check_build() ) { add_action( 'admin_notices', array( $this, 'render_build_notice' ) ); } $this->includes(); $this->hooks(); } /** * Define Constants. */ protected function define_constants() { $this->define( 'WC_ADMIN_APP', 'wc-admin-app' ); $this->define( 'WC_ADMIN_ABSPATH', dirname( __FILE__ ) . '/' ); $this->define( 'WC_ADMIN_DIST_JS_FOLDER', 'dist/' ); $this->define( 'WC_ADMIN_DIST_CSS_FOLDER', 'dist/' ); $this->define( 'WC_ADMIN_FEATURES_PATH', WC_ADMIN_ABSPATH . 'includes/features/' ); $this->define( 'WC_ADMIN_PLUGIN_FILE', __FILE__ ); $this->define( 'WC_ADMIN_VERSION_NUMBER', '0.12.0' ); } /** * Load Localisation files. */ protected function load_plugin_textdomain() { load_plugin_textdomain( 'woocommerce-admin', false, basename( dirname( __FILE__ ) ) . '/languages' ); } /** * Include WC Admin classes. */ public function includes() { require_once WC_ADMIN_ABSPATH . 'includes/core-functions.php'; // Initialize the WC API extensions. require_once WC_ADMIN_ABSPATH . 'includes/class-wc-admin-reports-sync.php'; require_once WC_ADMIN_ABSPATH . 'includes/class-wc-admin-install.php'; require_once WC_ADMIN_ABSPATH . 'includes/class-wc-admin-events.php'; require_once WC_ADMIN_ABSPATH . 'includes/class-wc-admin-api-init.php'; // Data triggers. require_once WC_ADMIN_ABSPATH . 'includes/data-stores/class-wc-admin-notes-data-store.php'; // CRUD classes. require_once WC_ADMIN_ABSPATH . 'includes/notes/class-wc-admin-note.php'; require_once WC_ADMIN_ABSPATH . 'includes/notes/class-wc-admin-notes.php'; // Admin note providers. // @todo These should be bundled in the features/ folder, but loading them from there currently has a load order issue. require_once WC_ADMIN_ABSPATH . 'includes/notes/class-wc-admin-notes-new-sales-record.php'; require_once WC_ADMIN_ABSPATH . 'includes/notes/class-wc-admin-notes-settings-notes.php'; require_once WC_ADMIN_ABSPATH . 'includes/notes/class-wc-admin-notes-giving-feedback-notes.php'; require_once WC_ADMIN_ABSPATH . 'includes/notes/class-wc-admin-notes-woo-subscriptions-notes.php'; require_once WC_ADMIN_ABSPATH . 'includes/notes/class-wc-admin-notes-historical-data.php'; require_once WC_ADMIN_ABSPATH . 'includes/notes/class-wc-admin-notes-order-milestones.php'; require_once WC_ADMIN_ABSPATH . 'includes/notes/class-wc-admin-notes-mobile-app.php'; require_once WC_ADMIN_ABSPATH . 'includes/notes/class-wc-admin-notes-welcome-message.php'; } /** * Filter in our ActionScheduler Store class. * * @param string $store_class ActionScheduler Store class name. * @return string ActionScheduler Store class name. */ public function replace_actionscheduler_store_class( $store_class ) { // Don't override any other overrides. if ( 'ActionScheduler_wpPostStore' !== $store_class ) { return $store_class; } // Include our store class here instead of wc_admin_plugins_loaded() // because ActionScheduler is hooked into `plugins_loaded` at a // much higher priority. require_once WC_ADMIN_ABSPATH . '/includes/class-wc-admin-actionscheduler-wppoststore.php'; return 'WC_Admin_ActionScheduler_WPPostStore'; } /** * Removes core hooks in favor of our local feature plugin handlers. * * @see WC_Admin_Library::__construct() */ protected function hooks() { remove_action( 'init', array( 'WC_Admin_Library', 'load_features' ) ); remove_action( 'admin_enqueue_scripts', array( 'WC_Admin_Library', 'register_scripts' ) ); remove_action( 'admin_enqueue_scripts', array( 'WC_Admin_Library', 'load_scripts' ), 15 ); remove_action( 'woocommerce_components_settings', array( 'WC_Admin_Library', 'add_component_settings' ) ); remove_filter( 'admin_body_class', array( 'WC_Admin_Library', 'add_admin_body_classes' ) ); remove_action( 'admin_menu', array( 'WC_Admin_Library', 'register_page_handler' ) ); remove_filter( 'admin_title', array( 'WC_Admin_Library', 'update_admin_title' ) ); remove_action( 'rest_api_init', array( 'WC_Admin_Library', 'register_user_data' ) ); remove_action( 'in_admin_header', array( 'WC_Admin_Library', 'embed_page_header' ) ); remove_filter( 'woocommerce_settings_groups', array( 'WC_Admin_Library', 'add_settings_group' ) ); remove_filter( 'woocommerce_settings-wc_admin', array( 'WC_Admin_Library', 'add_settings' ) ); remove_action( 'admin_head', array( 'WC_Admin_Library', 'update_link_structure' ), 20 ); require_once WC_ADMIN_ABSPATH . 'includes/class-wc-admin-loader.php'; add_filter( 'woocommerce_admin_features', array( $this, 'replace_supported_features' ) ); add_action( 'admin_menu', array( $this, 'register_devdocs_page' ) ); } /** * Returns true if all dependencies for the wc-admin plugin are loaded. * * @return bool */ protected function check_dependencies() { $woocommerce_minimum_met = class_exists( 'WooCommerce' ) && version_compare( WC_VERSION, '3.6', '>=' ); if ( ! $woocommerce_minimum_met ) { return false; } $wordpress_version = get_bloginfo( 'version' ); return version_compare( $wordpress_version, '4.9.9', '>' ); } /** * Returns true if build file exists. * * @return bool */ protected function check_build() { return file_exists( plugin_dir_path( __FILE__ ) . '/dist/app/index.js' ); } /** * Deactivates this plugin. */ public function deactivate_self() { deactivate_plugins( plugin_basename( WC_ADMIN_PLUGIN_FILE ) ); unset( $_GET['activate'] ); } /** * Notify users of the plugin requirements. */ public function render_dependencies_notice() { // The notice varies by WordPress version. $wordpress_version = get_bloginfo( 'version' ); $wordpress_includes_gutenberg = version_compare( $wordpress_version, '4.9.9', '>' ); if ( $wordpress_includes_gutenberg ) { $message = sprintf( /* translators: URL of WooCommerce plugin */ __( 'The WooCommerce Admin feature plugin requires WooCommerce 3.6 or greater to be installed and active.', 'woocommerce-admin' ), 'https://wordpress.org/plugins/woocommerce/' ); } else { $message = sprintf( /* translators: 1: URL of WordPress.org, 2: URL of WooCommerce plugin */ __( 'The WooCommerce Admin feature plugin requires both WordPress 5.0 or greater and WooCommerce 3.6 or greater to be installed and active.', 'woocommerce-admin' ), 'https://wordpress.org/', 'https://wordpress.org/plugins/woocommerce/' ); } printf( '

%s

', $message ); /* WPCS: xss ok. */ } /** * Notify users that the plugin needs to be built. */ public function render_build_notice() { $message_one = __( 'You have installed a development version of WooCommerce Admin which requires files to be built. From the plugin directory, run npm install to install dependencies, npm run build to build the files.', 'woocommerce-admin' ); $message_two = sprintf( /* translators: 1: URL of GitHub Repository build page */ __( 'Or you can download a pre-built version of the plugin by visiting the releases page in the repository.', 'woocommerce-admin' ), 'https://github.com/woocommerce/woocommerce-admin/releases' ); printf( '

%s %s

', $message_one, $message_two ); /* WPCS: xss ok. */ } /** * Overwrites the allowed features array using a local `feature-config.php` file. * * @param array $features Array of feature slugs. */ public function replace_supported_features( $features ) { if ( ! function_exists( 'wc_admin_get_feature_config' ) ) { require_once WC_ADMIN_ABSPATH . '/includes/feature-config.php'; } $feature_config = wc_admin_get_feature_config(); $features = array_keys( array_filter( $feature_config ) ); return $features; } /** * Adds a menu item for the wc-admin devdocs. */ public function register_devdocs_page() { if ( WC_Admin_Loader::is_feature_enabled( 'devdocs' ) && defined( 'WP_DEBUG' ) && WP_DEBUG && defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ) { wc_admin_register_page( array( 'title' => 'DevDocs', 'parent' => 'woocommerce', 'path' => '/devdocs', ) ); } } /** * Define constant if not already set. * * @param string $name Constant name. * @param string|bool $value Constant value. */ protected function define( $name, $value ) { if ( ! defined( $name ) ) { define( $name, $value ); } } /** * Prevent cloning. */ private function __clone() {} /** * Prevent unserializing. */ private function __wakeup() {} } WC_Admin_Feature_Plugin::instance()->init();