networkactive = (is_multisite() && array_key_exists(plugin_basename(__FILE__), (array) get_site_option('active_sitewide_plugins'))); $this->is_CLI = defined('WP_CLI') && WP_CLI; $this->sitewide_settings = get_site_option('disable_comments_sitewide_settings', false); // Load options. if ($this->networkactive && ($this->is_network_admin() || $this->sitewide_settings !== '1')) { $this->options = get_site_option('disable_comments_options', array()); $this->options['disabled_sites'] = $this->get_disabled_sites(); $blog_id = get_current_blog_id(); if( !$this->is_network_admin() && ( empty($this->options['disabled_sites']) || // if site disabled empty($this->options['disabled_sites']["site_$blog_id"]) ) ){ $this->options = [ 'remove_everywhere' => false, 'disabled_post_types' => array(), 'extra_post_types' => array(), 'disabled_sites' => array(), 'remove_xmlrpc_comments' => 0, 'remove_rest_API_comments' => 0, 'settings_saved' => true, 'db_version' => $this->options['db_version'] ]; } } else { $this->options = get_option('disable_comments_options', array()); $not_configured = empty($this->options) || empty($this->options['settings_saved']); if(is_multisite() && $not_configured && $this->sitewide_settings == '1'){ $this->options = get_site_option('disable_comments_options', array()); $this->options['is_network_options'] = true; } } // If it looks like first run, check compat. if (empty($this->options)) { $this->check_compatibility(); } $this->options['sitewide_settings'] = ($this->sitewide_settings == '1'); // Upgrade DB if necessary. $this->check_db_upgrades(); $this->init_filters(); add_action( 'wp_loaded', [ $this, 'start_plugin_usage_tracking'] ); } public function is_network_admin(){ if (is_network_admin() || defined('DOING_AJAX') && DOING_AJAX && is_multisite() && preg_match('#^'.network_admin_url().'#i',$_SERVER['HTTP_REFERER'])) { return true; } return false; } /** * Enable CLI * @since 2.0.0 */ public function enable_cli(){ require_once DC_PLUGIN_ROOT_PATH . "/includes/cli.php"; new Disable_Comment_Command($this); } public function admin_notice(){ if( $this->tracker instanceof DisableComments_Plugin_Tracker ) { if( isset( $this->setup_notice_flag ) && $this->setup_notice_flag === true ) { return; } $current_screen = get_current_screen()->id; $hascaps = $this->networkactive && is_network_admin() ? current_user_can('manage_network_plugins') : current_user_can('manage_options'); if( ! in_array( $current_screen, ['settings_page_disable_comments_settings', 'settings_page_disable_comments_settings-network']) && $hascaps ) { $this->tracker->notice(); } } } public function start_plugin_usage_tracking() { if($this->networkactive && !$this->options['sitewide_settings']){ $this->tracker = null; return; } if (!class_exists('DisableComments_Plugin_Tracker')) { include_once(DC_PLUGIN_ROOT_PATH . '/includes/class-plugin-usage-tracker.php'); } $tracker = $this->tracker = DisableComments_Plugin_Tracker::get_instance(__FILE__, [ 'opt_in' => true, 'goodbye_form' => true, 'item_id' => 'b0112c9030af6ba53de4' ]); $tracker->set_notice_options(array( 'notice' => __('Want to help make Disable Comments even better?', 'disable-comments'), 'extra_notice' => __('We collect non-sensitive diagnostic data and plugin usage information. Your site URL, WordPress & PHP version, plugins & themes and email address to send you the discount coupon. This data lets us make sure this plugin always stays compatible with the most popular plugins and themes. No spam, I promise.', 'disable-comments'), )); $tracker->init(); } private function check_compatibility() { if (version_compare($GLOBALS['wp_version'], '4.7', '<')) { require_once(ABSPATH . 'wp-admin/includes/plugin.php'); deactivate_plugins(__FILE__); if (isset($_GET['action']) && ($_GET['action'] == 'activate' || $_GET['action'] == 'error_scrape')) { // translators: %s: WordPress version no. exit(sprintf(__('Disable Comments requires WordPress version %s or greater.', 'disable-comments'), '4.7')); } } } private function check_db_upgrades() { $old_ver = isset($this->options['db_version']) ? $this->options['db_version'] : 0; if ($old_ver < self::DB_VERSION) { if($this->networkactive){ $this->options['is_network_admin'] = true; } if ($old_ver < 2) { // upgrade options from version 0.2.1 or earlier to 0.3. $this->options['disabled_post_types'] = get_option('disable_comments_post_types', array()); delete_option('disable_comments_post_types'); } if ($old_ver < 5) { // simple is beautiful - remove multiple settings in favour of one. $this->options['remove_everywhere'] = isset($this->options['remove_admin_menu_comments']) ? $this->options['remove_admin_menu_comments'] : false; foreach (array('remove_admin_menu_comments', 'remove_admin_bar_comments', 'remove_recent_comments', 'remove_discussion', 'remove_rc_widget') as $v) { unset($this->options[$v]); } } if ($old_ver < 7 && function_exists( 'get_sites' )) { $this->options['disabled_sites'] = []; $dc_options = get_site_option('disable_comments_options', array()); foreach(get_sites(['number' => 0, 'fields' => 'ids']) as $blog_id){ if(isset($dc_options['disabled_sites'])){ $this->options['disabled_sites']["site_$blog_id"] = in_array($blog_id, $dc_options['disabled_sites']); } else{ $this->options['disabled_sites']["site_$blog_id"] = true; } } $this->options['disabled_sites'] = $this->get_disabled_sites(); } foreach (array('remove_everywhere', 'extra_post_types') as $v) { if (!isset($this->options[$v])) { $this->options[$v] = false; } } $this->options['db_version'] = self::DB_VERSION; $this->update_options(); } } private function update_options() { if ($this->networkactive && !empty($this->options['is_network_admin']) && $this->options['is_network_admin']) { unset($this->options['is_network_admin']); update_site_option('disable_comments_options', $this->options); } else{ update_option('disable_comments_options', $this->options); } } public function get_disabled_sites($default = false){ $disabled_sites = ['all' => true]; foreach(get_sites(['number' => 0, 'fields' => 'ids']) as $blog_id){ $disabled_sites["site_{$blog_id}"] = true; } if($default){ return $disabled_sites; } $this->options['disabled_sites'] = isset($this->options['disabled_sites']) ? $this->options['disabled_sites'] : []; $this->options['disabled_sites'] = wp_parse_args($this->options['disabled_sites'], $disabled_sites); $disabled_sites = $this->options['disabled_sites']; unset($disabled_sites['all']); if(in_array(false, $disabled_sites)){ $this->options['disabled_sites']['all'] = false; } else{ $this->options['disabled_sites']['all'] = true; } return $this->options['disabled_sites']; } // public function get_disabled_count(){ // $disabled_sites = isset($this->options['disabled_sites']) ? $this->options['disabled_sites'] : []; // unset($disabled_sites['all']); // return array_sum($disabled_sites); // } /** * Get an array of disabled post type. */ public function get_disabled_post_types() { $types = $this->options['disabled_post_types']; // Not all extra_post_types might be registered on this particular site. if ($this->networkactive && !empty($this->options['extra_post_types'])) { foreach ((array) $this->options['extra_post_types'] as $extra) { if (post_type_exists($extra)) { $types[] = $extra; } } } return $types; } /** * Check whether comments have been disabled on a given post type. */ private function is_post_type_disabled($type) { return $type && in_array($type, $this->get_disabled_post_types()); } private function init_filters() { // These need to happen now. if ($this->options['remove_everywhere']) { add_action('widgets_init', array($this, 'disable_rc_widget')); add_filter('wp_headers', array($this, 'filter_wp_headers')); add_action('template_redirect', array($this, 'filter_query'), 9); // before redirect_canonical. // Admin bar filtering has to happen here since WP 3.6. add_action('template_redirect', array($this, 'filter_admin_bar')); add_action('admin_init', array($this, 'filter_admin_bar')); // Disable Comments REST API Endpoint add_filter('rest_endpoints', array($this, 'filter_rest_endpoints')); } // remove create comment via xmlrpc if (isset($this->options['remove_xmlrpc_comments']) && intval($this->options['remove_xmlrpc_comments']) === 1) { add_filter('xmlrpc_methods', array($this, 'disable_xmlrc_comments')); } // rest API Comment Block if (isset($this->options['remove_rest_API_comments']) && intval($this->options['remove_rest_API_comments']) === 1) { add_filter('rest_pre_insert_comment', array($this, 'disable_rest_API_comments')); } // These can happen later. add_action('plugins_loaded', array($this, 'register_text_domain')); add_action('wp_loaded', array($this, 'init_wploaded_filters')); // Disable "Latest comments" block in Gutenberg. add_action('enqueue_block_editor_assets', array($this, 'filter_gutenberg_blocks')); // settings page assets add_action('admin_enqueue_scripts', array($this, 'settings_page_assets')); if(!$this->networkactive || $this->options['sitewide_settings']) { add_filter('comment_status_links', function($status_links){ $status_links['disable_comments'] = sprintf("%s", __("Disable Comments", 'disable-comments')); return $status_links; }); } } public function register_text_domain() { load_plugin_textdomain('disable-comments', false, dirname(plugin_basename(__FILE__)) . '/languages'); } public function init_wploaded_filters() { $disabled_post_types = $this->get_disabled_post_types(); if (!empty($disabled_post_types)) { foreach ($disabled_post_types as $type) { // we need to know what native support was for later. if (post_type_supports($type, 'comments')) { $this->modified_types[] = $type; remove_post_type_support($type, 'comments'); remove_post_type_support($type, 'trackbacks'); } } add_filter('comments_array', array($this, 'filter_existing_comments'), 20, 2); add_filter('comments_open', array($this, 'filter_comment_status'), 20, 2); add_filter('pings_open', array($this, 'filter_comment_status'), 20, 2); add_filter('get_comments_number', array($this, 'filter_comments_number'), 20, 2); } elseif (is_admin() && !$this->is_configured()) { /** * It is possible that $disabled_post_types is empty if other * plugins have disabled comments. Hence we also check for * remove_everywhere. If you still get a warning you probably * shouldn't be using this plugin. */ add_action('all_admin_notices', array($this, 'setup_notice')); } // Filters for the admin only. if (is_admin()) { add_action( 'all_admin_notices', array( $this, 'admin_notice' ) ); if ($this->networkactive && is_network_admin()) { add_action('network_admin_menu', array($this, 'settings_menu')); add_action('network_admin_menu', array($this, 'tools_menu')); add_filter('network_admin_plugin_action_links', array($this, 'plugin_actions_links'), 10, 2); } elseif(!$this->networkactive || $this->options['sitewide_settings']) { add_action('admin_menu', array($this, 'settings_menu')); add_action('admin_menu', array($this, 'tools_menu')); add_filter('plugin_action_links', array($this, 'plugin_actions_links'), 10, 2); if (is_multisite()) { // We're on a multisite setup, but the plugin isn't network activated. register_deactivation_hook(__FILE__, array($this, 'single_site_deactivate')); } } add_action('admin_notices', array($this, 'discussion_notice')); add_filter('plugin_row_meta', array($this, 'set_plugin_meta'), 10, 2); if ($this->options['remove_everywhere']) { add_action('admin_menu', array($this, 'filter_admin_menu'), 9999); // do this as late as possible. add_action('admin_print_styles-index.php', array($this, 'admin_css')); add_action('admin_print_styles-profile.php', array($this, 'admin_css')); add_action('wp_dashboard_setup', array($this, 'filter_dashboard')); add_filter('pre_option_default_pingback_flag', '__return_zero'); } } // Filters for front end only. else { add_action('template_redirect', array($this, 'check_comment_template')); if ($this->options['remove_everywhere']) { add_filter('feed_links_show_comments_feed', '__return_false'); } } } // public function get_option( $key, $default = false ){ // return $this->networkactive ? get_site_option( $key, $default ) : get_option( $key, $default ); // } // public function update_option( $option, $value ){ // return $this->networkactive ? update_site_option( $option, $value ) : update_option( $option, $value ); // } // public function delete_option( $option ){ // return $this->networkactive ? delete_site_option( $option ) : delete_option( $option ); // } /** * Replace the theme's comment template with a blank one. * To prevent this, define DISABLE_COMMENTS_REMOVE_COMMENTS_TEMPLATE * and set it to True */ public function check_comment_template() { if (is_singular() && ($this->options['remove_everywhere'] || $this->is_post_type_disabled(get_post_type()))) { if (!defined('DISABLE_COMMENTS_REMOVE_COMMENTS_TEMPLATE') || DISABLE_COMMENTS_REMOVE_COMMENTS_TEMPLATE == true) { // Kill the comments template. add_filter('comments_template', array($this, 'dummy_comments_template'), 20); } // Remove comment-reply script for themes that include it indiscriminately. wp_deregister_script('comment-reply'); // feed_links_extra inserts a comments RSS link. remove_action('wp_head', 'feed_links_extra', 3); } } public function dummy_comments_template() { return dirname(__FILE__) . '/views/comments.php'; } public function is_xmlrpc_rest(){ // remove create comment via xmlrpc if (isset($this->options['remove_xmlrpc_comments']) && intval($this->options['remove_xmlrpc_comments']) === 1) { return true; } // rest API Comment Block if (isset($this->options['remove_rest_API_comments']) && intval($this->options['remove_rest_API_comments']) === 1) { return true; } return false; } /** * Remove the X-Pingback HTTP header */ public function filter_wp_headers($headers) { unset($headers['X-Pingback']); return $headers; } /** * remove method wp.newComment */ public function disable_xmlrc_comments($methods) { unset($methods['wp.newComment']); return $methods; } public function disable_rest_API_comments($prepared_comment, $request) { return; } /** * Issue a 403 for all comment feed requests. */ public function filter_query() { if (is_comment_feed()) { wp_die(__('Comments are closed.', 'disable-comments'), '', array('response' => 403)); } } /** * Remove comment links from the admin bar. */ public function filter_admin_bar() { if (is_admin_bar_showing()) { // Remove comments links from admin bar. remove_action('admin_bar_menu', 'wp_admin_bar_comments_menu', 60); if (is_multisite()) { add_action('admin_bar_menu', array($this, 'remove_network_comment_links'), 500); } } } /** * Remove the comments endpoint for the REST API */ public function filter_rest_endpoints($endpoints) { unset($endpoints['comments']); return $endpoints; } /** * Determines if scripts should be enqueued */ public function filter_gutenberg_blocks($hook) { global $post; if ($this->options['remove_everywhere'] || (isset($post->post_type) && in_array($post->post_type, $this->get_disabled_post_types(), true))) { return $this->disable_comments_script(); } } /** * Enqueues scripts */ public function disable_comments_script() { wp_enqueue_script('disable-comments-gutenberg', plugin_dir_url(__FILE__) . 'assets/js/disable-comments.js', array(), false, true); } /** * Enqueues Scripts for Settings Page */ public function settings_page_assets($hook_suffix) { if ( $hook_suffix === 'settings_page_' . DC_PLUGIN_SLUG || $hook_suffix === 'options-general_' . DC_PLUGIN_SLUG ) { // css wp_enqueue_style('sweetalert2', DC_ASSETS_URI . 'css/sweetalert2.min.css', [], false); // wp_enqueue_style('pagination', DC_ASSETS_URI . 'css/pagination.css', [], false); wp_enqueue_style('disable-comments-style', DC_ASSETS_URI . 'css/style.css', [], false); // js wp_enqueue_script('sweetalert2', DC_ASSETS_URI . 'js/sweetalert2.all.min.js', array('jquery'), false, true); wp_enqueue_script('pagination', DC_ASSETS_URI . 'js/pagination.min.js', array('jquery'), false, true); wp_enqueue_script('disable-comments-scripts', DC_ASSETS_URI . 'js/disable-comments-settings-scripts.js', array('jquery'), false, true); wp_localize_script( 'disable-comments-scripts', 'disableCommentsObj', array( 'save_action' => 'disable_comments_save_settings', 'delete_action' => 'disable_comments_delete_comments', 'settings_URI' => $this->settings_page_url(), '_nonce' => wp_create_nonce('disable_comments_save_settings') ) ); } else { // notice css wp_enqueue_style('disable-comments-notice', DC_ASSETS_URI . 'css/notice.css', [], false); } } /** * Remove comment links from the admin bar in a multisite network. */ public function remove_network_comment_links($wp_admin_bar) { if ($this->networkactive && is_user_logged_in()) { foreach ((array) $wp_admin_bar->user->blogs as $blog) { $wp_admin_bar->remove_menu('blog-' . $blog->userblog_id . '-c'); } } else { // We have no way to know whether the plugin is active on other sites, so only remove this one. $wp_admin_bar->remove_menu('blog-' . get_current_blog_id() . '-c'); } } public function discussion_notice() { $disabled_post_types = $this->get_disabled_post_types(); if (get_current_screen()->id == 'options-discussion' && !empty($disabled_post_types)) { $names = array(); foreach ($disabled_post_types as $type) { $names[$type] = get_post_type_object($type)->labels->name; } // translators: %s: disabled post types. echo '
' . sprintf(__('Note: The Disable Comments plugin is currently active, and comments are completely disabled on: %s. Many of the settings below will not be applicable for those post types.', 'disable-comments'), implode(__(', ', 'disable-comments'), $names)) . '
' . sprintf(__('The Disable Comments plugin is active, but isn\'t configured to do anything yet. Visit the configuration page to choose which post types to disable comments on.', 'disable-comments'), esc_attr($this->settings_page_url())) . '