308 lines
7.5 KiB
PHP
308 lines
7.5 KiB
PHP
|
<?php
|
||
|
/**
|
||
|
* WPSEO plugin file.
|
||
|
*
|
||
|
* @package WPSEO\Admin\Notifications
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* Class Yoast_Notifications.
|
||
|
*/
|
||
|
class Yoast_Notifications {
|
||
|
|
||
|
/**
|
||
|
* Holds the admin page's ID.
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
const ADMIN_PAGE = 'wpseo_dashboard';
|
||
|
|
||
|
/**
|
||
|
* Total notifications count.
|
||
|
*
|
||
|
* @var int
|
||
|
*/
|
||
|
private static $notification_count = 0;
|
||
|
|
||
|
/**
|
||
|
* All error notifications.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
private static $errors = [];
|
||
|
|
||
|
/**
|
||
|
* Active errors.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
private static $active_errors = [];
|
||
|
|
||
|
/**
|
||
|
* Dismissed errors.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
private static $dismissed_errors = [];
|
||
|
|
||
|
/**
|
||
|
* All warning notifications.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
private static $warnings = [];
|
||
|
|
||
|
/**
|
||
|
* Active warnings.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
private static $active_warnings = [];
|
||
|
|
||
|
/**
|
||
|
* Dismissed warnings.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
private static $dismissed_warnings = [];
|
||
|
|
||
|
/**
|
||
|
* Yoast_Notifications constructor.
|
||
|
*/
|
||
|
public function __construct() {
|
||
|
|
||
|
$this->add_hooks();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Add hooks
|
||
|
*/
|
||
|
private function add_hooks() {
|
||
|
// phpcs:ignore WordPress.Security.NonceVerification.Recommended -- Reason: We are not processing form information.
|
||
|
if ( isset( $_GET['page'] ) && is_string( $_GET['page'] ) ) {
|
||
|
// phpcs:ignore WordPress.Security.NonceVerification.Recommended -- Reason: We are not processing form information.
|
||
|
$page = sanitize_text_field( wp_unslash( $_GET['page'] ) );
|
||
|
if ( $page === self::ADMIN_PAGE ) {
|
||
|
add_action( 'admin_enqueue_scripts', [ $this, 'enqueue_assets' ] );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Needed for adminbar and Notifications page.
|
||
|
add_action( 'admin_init', [ __CLASS__, 'collect_notifications' ], 99 );
|
||
|
|
||
|
// Add AJAX hooks.
|
||
|
add_action( 'wp_ajax_yoast_dismiss_notification', [ $this, 'ajax_dismiss_notification' ] );
|
||
|
add_action( 'wp_ajax_yoast_restore_notification', [ $this, 'ajax_restore_notification' ] );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Enqueue assets.
|
||
|
*/
|
||
|
public function enqueue_assets() {
|
||
|
$asset_manager = new WPSEO_Admin_Asset_Manager();
|
||
|
|
||
|
$asset_manager->enqueue_style( 'notifications' );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Handle ajax request to dismiss a notification.
|
||
|
*/
|
||
|
public function ajax_dismiss_notification() {
|
||
|
|
||
|
$notification = $this->get_notification_from_ajax_request();
|
||
|
if ( $notification ) {
|
||
|
$notification_center = Yoast_Notification_Center::get();
|
||
|
$notification_center->maybe_dismiss_notification( $notification );
|
||
|
|
||
|
$this->output_ajax_response( $notification->get_type() );
|
||
|
}
|
||
|
|
||
|
wp_die();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Handle ajax request to restore a notification.
|
||
|
*/
|
||
|
public function ajax_restore_notification() {
|
||
|
|
||
|
$notification = $this->get_notification_from_ajax_request();
|
||
|
if ( $notification ) {
|
||
|
$notification_center = Yoast_Notification_Center::get();
|
||
|
$notification_center->restore_notification( $notification );
|
||
|
|
||
|
$this->output_ajax_response( $notification->get_type() );
|
||
|
}
|
||
|
|
||
|
wp_die();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Create AJAX response data.
|
||
|
*
|
||
|
* @param string $type Notification type.
|
||
|
*/
|
||
|
private function output_ajax_response( $type ) {
|
||
|
|
||
|
$html = $this->get_view_html( $type );
|
||
|
// phpcs:disable WordPress.Security.EscapeOutput -- Reason: WPSEO_Utils::format_json_encode is safe.
|
||
|
echo WPSEO_Utils::format_json_encode(
|
||
|
[
|
||
|
'html' => $html,
|
||
|
'total' => self::get_active_notification_count(),
|
||
|
]
|
||
|
);
|
||
|
// phpcs:enable -- Reason: WPSEO_Utils::format_json_encode is safe.
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the HTML to return in the AJAX request.
|
||
|
*
|
||
|
* @param string $type Notification type.
|
||
|
*
|
||
|
* @return bool|string
|
||
|
*/
|
||
|
private function get_view_html( $type ) {
|
||
|
|
||
|
switch ( $type ) {
|
||
|
case 'error':
|
||
|
$view = 'errors';
|
||
|
break;
|
||
|
|
||
|
case 'warning':
|
||
|
default:
|
||
|
$view = 'warnings';
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// Re-collect notifications.
|
||
|
self::collect_notifications();
|
||
|
|
||
|
/**
|
||
|
* Stops PHPStorm from nagging about this variable being unused. The variable is used in the view.
|
||
|
*
|
||
|
* @noinspection PhpUnusedLocalVariableInspection
|
||
|
*/
|
||
|
$notifications_data = self::get_template_variables();
|
||
|
|
||
|
ob_start();
|
||
|
include WPSEO_PATH . 'admin/views/partial-notifications-' . $view . '.php';
|
||
|
$html = ob_get_clean();
|
||
|
|
||
|
return $html;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Extract the Yoast Notification from the AJAX request.
|
||
|
*
|
||
|
* This function does not handle nonce verification.
|
||
|
*
|
||
|
* @return Yoast_Notification|null A Yoast_Notification on success, null on failure.
|
||
|
*/
|
||
|
private function get_notification_from_ajax_request() {
|
||
|
// phpcs:ignore WordPress.Security.NonceVerification.Missing -- Reason: This function does not handle nonce verification.
|
||
|
if ( ! isset( $_POST['notification'] ) || ! is_string( $_POST['notification'] ) ) {
|
||
|
return null;
|
||
|
}
|
||
|
// phpcs:ignore WordPress.Security.NonceVerification.Missing -- Reason: This function does not handle nonce verification.
|
||
|
$notification_id = sanitize_text_field( wp_unslash( $_POST['notification'] ) );
|
||
|
|
||
|
if ( empty( $notification_id ) ) {
|
||
|
return null;
|
||
|
}
|
||
|
$notification_center = Yoast_Notification_Center::get();
|
||
|
return $notification_center->get_notification_by_id( $notification_id );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Collect the notifications and group them together.
|
||
|
*/
|
||
|
public static function collect_notifications() {
|
||
|
|
||
|
$notification_center = Yoast_Notification_Center::get();
|
||
|
|
||
|
$notifications = $notification_center->get_sorted_notifications();
|
||
|
self::$notification_count = count( $notifications );
|
||
|
|
||
|
self::$errors = array_filter( $notifications, [ __CLASS__, 'filter_error_notifications' ] );
|
||
|
self::$dismissed_errors = array_filter( self::$errors, [ __CLASS__, 'filter_dismissed_notifications' ] );
|
||
|
self::$active_errors = array_diff( self::$errors, self::$dismissed_errors );
|
||
|
|
||
|
self::$warnings = array_filter( $notifications, [ __CLASS__, 'filter_warning_notifications' ] );
|
||
|
self::$dismissed_warnings = array_filter( self::$warnings, [ __CLASS__, 'filter_dismissed_notifications' ] );
|
||
|
self::$active_warnings = array_diff( self::$warnings, self::$dismissed_warnings );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the variables needed in the views.
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
public static function get_template_variables() {
|
||
|
|
||
|
return [
|
||
|
'metrics' => [
|
||
|
'total' => self::$notification_count,
|
||
|
'active' => self::get_active_notification_count(),
|
||
|
'errors' => count( self::$errors ),
|
||
|
'warnings' => count( self::$warnings ),
|
||
|
],
|
||
|
'errors' => [
|
||
|
'dismissed' => self::$dismissed_errors,
|
||
|
'active' => self::$active_errors,
|
||
|
],
|
||
|
'warnings' => [
|
||
|
'dismissed' => self::$dismissed_warnings,
|
||
|
'active' => self::$active_warnings,
|
||
|
],
|
||
|
];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the number of active notifications.
|
||
|
*
|
||
|
* @return int
|
||
|
*/
|
||
|
public static function get_active_notification_count() {
|
||
|
|
||
|
return ( count( self::$active_errors ) + count( self::$active_warnings ) );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Filter out any non-errors.
|
||
|
*
|
||
|
* @param Yoast_Notification $notification Notification to test.
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
private static function filter_error_notifications( Yoast_Notification $notification ) {
|
||
|
|
||
|
return $notification->get_type() === 'error';
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Filter out any non-warnings.
|
||
|
*
|
||
|
* @param Yoast_Notification $notification Notification to test.
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
private static function filter_warning_notifications( Yoast_Notification $notification ) {
|
||
|
|
||
|
return $notification->get_type() !== 'error';
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Filter out any dismissed notifications.
|
||
|
*
|
||
|
* @param Yoast_Notification $notification Notification to test.
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
private static function filter_dismissed_notifications( Yoast_Notification $notification ) {
|
||
|
|
||
|
return Yoast_Notification_Center::is_notification_dismissed( $notification );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
class_alias( Yoast_Notifications::class, 'Yoast_Alerts' );
|