* @copyright Copyright (c) 2004 - 2020, Caseproof * @link https://memberpress.com/plugins/members * @license http://www.gnu.org/licenses/old-licenses/gpl-2.0.html */ /** * Singleton class for setting up the plugin. * * @since 1.0.0 * @access public */ final class Members_Plugin { /** * Minimum required PHP version. * * @since 2.0.0 * @access public * @var string */ private $php_version = '5.3.0'; /** * Plugin directory path. * * @since 2.0.0 * @access public * @var string */ public $dir = ''; /** * Plugin directory URI. * * @since 2.0.0 * @access public * @var string */ public $uri = ''; /** * User count of all roles. * * @see members_get_role_user_count() * @since 1.0.0 * @access public * @var array */ public $role_user_count = array(); /** * Returns the instance. * * @since 1.0.0 * @access public * @return object */ public static function get_instance() { static $instance = null; if ( is_null( $instance ) ) { $instance = new self; $instance->setup(); $instance->includes(); $instance->setup_actions(); } return $instance; } /** * Constructor method. * * @since 1.0.0 * @access private * @return void */ private function __construct() {} /** * Magic method to output a string if trying to use the object as a string. * * @since 1.0.0 * @access public * @return void */ public function __toString() { return 'members'; } /** * Magic method to keep the object from being cloned. * * @since 1.0.0 * @access public * @return void */ public function __clone() { _doing_it_wrong( __FUNCTION__, esc_html__( 'Whoah, partner!', 'members' ), '1.0.0' ); } /** * Magic method to keep the object from being unserialized. * * @since 1.0.0 * @access public * @return void */ public function __wakeup() { _doing_it_wrong( __FUNCTION__, esc_html__( 'Whoah, partner!', 'members' ), '1.0.0' ); } /** * Magic method to prevent a fatal error when calling a method that doesn't exist. * * @since 1.0.0 * @access public * @return null */ public function __call( $method = '', $args = array() ) { _doing_it_wrong( "Members_Plugin::{$method}", esc_html__( 'Method does not exist.', 'members' ), '1.0.0' ); unset( $method, $args ); return null; } /** * Sets up globals. * * @since 1.0.0 * @access private * @return void */ private function setup() { // Main plugin directory path and URI. $this->dir = trailingslashit( plugin_dir_path( __FILE__ ) ); $this->uri = trailingslashit( plugin_dir_url( __FILE__ ) ); } /** * Loads files needed by the plugin. * * @since 1.0.0 * @access private * @return void */ private function includes() { // Check if we meet the minimum PHP version. if ( version_compare( PHP_VERSION, $this->php_version, '<' ) ) { // Add admin notice. add_action( 'admin_notices', array( $this, 'php_admin_notice' ) ); // Bail. return; } // Load class files. require_once( $this->dir . 'inc/class-capability.php' ); require_once( $this->dir . 'inc/class-cap-group.php' ); require_once( $this->dir . 'inc/class-registry.php' ); require_once( $this->dir . 'inc/class-role-group.php' ); require_once( $this->dir . 'inc/class-role.php' ); // Load includes files. require_once( $this->dir . 'inc/functions.php' ); require_once( $this->dir . 'inc/functions-admin-bar.php' ); require_once( $this->dir . 'inc/functions-capabilities.php' ); require_once( $this->dir . 'inc/functions-cap-groups.php' ); require_once( $this->dir . 'inc/functions-content-permissions.php' ); require_once( $this->dir . 'inc/functions-deprecated.php' ); require_once( $this->dir . 'inc/functions-options.php' ); require_once( $this->dir . 'inc/functions-private-site.php' ); require_once( $this->dir . 'inc/functions-roles.php' ); require_once( $this->dir . 'inc/functions-role-groups.php' ); require_once( $this->dir . 'inc/functions-shortcodes.php' ); require_once( $this->dir . 'inc/functions-users.php' ); require_once( $this->dir . 'inc/functions-widgets.php' ); // Load template files. require_once( $this->dir . 'inc/template.php' ); // Load admin files. if ( is_admin() ) { // General admin functions. require_once( $this->dir . 'admin/functions-admin.php' ); require_once( $this->dir . 'admin/functions-help.php' ); // Plugin settings. require_once( $this->dir . 'admin/class-settings.php' ); // User management. require_once( $this->dir . 'admin/class-manage-users.php' ); require_once( $this->dir . 'admin/class-user-edit.php' ); require_once( $this->dir . 'admin/class-user-new.php' ); // Edit posts. require_once( $this->dir . 'admin/class-meta-box-content-permissions.php' ); // Role management. require_once( $this->dir . 'admin/class-manage-roles.php' ); require_once( $this->dir . 'admin/class-roles.php' ); require_once( $this->dir . 'admin/class-role-edit.php' ); require_once( $this->dir . 'admin/class-role-new.php' ); require_once( $this->dir . 'admin/class-meta-box-publish-role.php' ); require_once( $this->dir . 'admin/class-meta-box-custom-cap.php' ); // Edit capabilities tabs and groups. require_once( $this->dir . 'admin/class-cap-tabs.php' ); require_once( $this->dir . 'admin/class-cap-section.php' ); require_once( $this->dir . 'admin/class-cap-control.php' ); } $addons = get_option( 'members_active_addons', array() ); if ( ! empty( $addons ) ) { foreach ( $addons as $addon ) { if ( file_exists( __DIR__ . "/addons/{$addon}/addon.php" ) ) { include "addons/{$addon}/addon.php"; } } } } /** * Sets up main plugin actions and filters. * * @since 1.0.0 * @access private * @return void */ private function setup_actions() { // Internationalize the text strings used. add_action( 'plugins_loaded', array( $this, 'i18n' ), 2 ); // Migrate add-ons add_action( 'plugins_loaded', array( $this, 'migrate_addons' ) ); // MemberPress info in block editor add_action( 'enqueue_block_editor_assets', array( $this, 'block_editor_assets' ) ); // Register activation hook. register_activation_hook( __FILE__, array( $this, 'activation' ) ); } /** * Loads the translation files. * * @since 1.0.0 * @access public * @return void */ public function i18n() { load_plugin_textdomain( 'members', false, trailingslashit( dirname( plugin_basename( __FILE__ ) ) ) . 'lang' ); } /** * Method that runs only when the plugin is activated. * * @since 1.0.0 * @access public * @return void */ public function activation() { // Check PHP version requirements. if ( version_compare( PHP_VERSION, $this->php_version, '<' ) ) { // Make sure the plugin is deactivated. deactivate_plugins( plugin_basename( __FILE__ ) ); // Add an error message and die. wp_die( $this->get_min_php_message() ); } // Get the administrator role. $role = get_role( 'administrator' ); // If the administrator role exists, add required capabilities for the plugin. if ( ! empty( $role ) ) { $role->add_cap( 'restrict_content' ); // Edit per-post content permissions. $role->add_cap( 'list_roles' ); // View roles in backend. // Do not allow administrators to edit, create, or delete roles // in a multisite setup. Super admins should assign these manually. if ( ! is_multisite() ) { $role->add_cap( 'create_roles' ); // Create new roles. $role->add_cap( 'delete_roles' ); // Delete existing roles. $role->add_cap( 'edit_roles' ); // Edit existing roles/caps. } } } /** * Returns a message noting the minimum version of PHP required. * * @since 2.0.1 * @access private * @return void */ private function get_min_php_message() { return sprintf( __( 'Members requires PHP version %1$s. You are running version %2$s. Please upgrade and try again.', 'members' ), $this->php_version, PHP_VERSION ); } /** * Outputs the admin notice that the user needs to upgrade their PHP version. It also * auto-deactivates the plugin. * * @since 2.0.1 * @access public * @return void */ public function php_admin_notice() { // Output notice. printf( '

%s

', esc_html( $this->get_min_php_message() ) ); // Make sure the plugin is deactivated. deactivate_plugins( plugin_basename( __FILE__ ) ); } /** * Transition separate add-on plugins into the included add-ons * * @return void */ public function migrate_addons() { // Bail if we've already migrated the add-ons if ( ! empty( get_option( 'members_addons_migrated' ) ) ) { return; } $addons = array(); $plugins = array( 'members-acf-integration' => 'plugin.php', 'members-admin-access' => 'members-admin-access.php', 'members-block-permissions' => 'plugin.php', 'members-category-and-tag-caps' => 'plugin.php', 'members-core-create-caps' => 'members-core-create-caps.php', 'members-edd-integration' => 'plugin.php', 'members-givewp-integration' => 'plugin.php', 'members-meta-box-integration' => 'plugin.php', 'members-privacy-caps' => 'members-privacy-caps.php', 'members-role-hierarchy' => 'members-role-hierarchy.php', 'members-role-levels' => 'members-role-levels.php', 'members-woocommerce-integration' => 'plugin.php' ); require_once ABSPATH . 'wp-admin/includes/file.php'; require_once ABSPATH . 'wp-admin/includes/plugin.php'; foreach ( $plugins as $dir => $file ) { if ( is_plugin_active( "{$dir}/{$file}" ) ) { // Deactive it deactivate_plugins( "{$dir}/{$file}", true ); // Delete it delete_plugins( array( "{$dir}/{$file}" ) ); // Make sure it's stored in our option for active add-ons $addons[] = $dir; } } if ( ! empty( $addons ) ) { update_option( 'members_active_addons', $addons ); } update_option( 'members_addons_migrated', true ); } /** * We need a way to run an add-on's activation hook since the add-ons are no longer separate plugins. * * @param string $addon Add-on directory name * * @return void */ public function run_addon_activator( $addon ) { if ( file_exists( trailingslashit( __DIR__ ) . "addons/{$addon}/src/Activator.php" ) ) { // Require the add-on file include "addons/{$addon}/src/Activator.php"; // Read the file contents into memory, and determine the namespace $contents = file_get_contents( trailingslashit( __DIR__ ) . "addons/{$addon}/src/Activator.php" ); preg_match( '/[\r\n]namespace\W(.+);[\r\n]/', $contents, $matches ); $namespace = $matches[1]; // Run the activator if ( ! empty( $namespace ) ) { $namespace .= '\Activator'; $namespace::activate(); } } } public function block_editor_assets() { $active_addons = get_option( 'members_active_addons', array() ); if ( ! in_array( 'members-block-permissions', $active_addons ) ) { wp_enqueue_script( 'block-editor-mp-upsell', plugin_dir_url( __FILE__ ) . '/addons/members-block-permissions/public/js/upsell.js' , array( 'wp-compose', 'wp-element', 'wp-hooks', 'wp-components' ), null, true ); wp_localize_script( 'block-editor-mp-upsell', 'membersUpsell', array( 'title' => __( 'Permissions', 'members' ), 'message' => __( 'To protect this block by paid membership or centrally with a content protection rule, upgrade to MemberPress.', 'members' ) ) ); } } } /** * Gets the instance of the `Members_Plugin` class. This function is useful for quickly grabbing data * used throughout the plugin. * * @since 1.0.0 * @access public * @return object */ function members_plugin() { return Members_Plugin::get_instance(); } // Let's roll! members_plugin();