Code Coverage
 
Lines
Functions and Methods
Classes and Traits
Total
0.00% covered (danger)
0.00%
0 / 125
0.00% covered (danger)
0.00%
0 / 5
CRAP
0.00% covered (danger)
0.00%
0 / 1
Jetpack_Stats_Dashboard_Widget
0.00% covered (danger)
0.00%
0 / 125
0.00% covered (danger)
0.00%
0 / 5
306
0.00% covered (danger)
0.00%
0 / 1
 init
0.00% covered (danger)
0.00%
0 / 3
0.00% covered (danger)
0.00%
0 / 1
6
 wp_dashboard_setup
0.00% covered (danger)
0.00%
0 / 12
0.00% covered (danger)
0.00%
0 / 1
30
 admin_head
0.00% covered (danger)
0.00%
0 / 39
0.00% covered (danger)
0.00%
0 / 1
2
 render_widget
0.00% covered (danger)
0.00%
0 / 4
0.00% covered (danger)
0.00%
0 / 1
6
 render_footer
0.00% covered (danger)
0.00%
0 / 67
0.00% covered (danger)
0.00%
0 / 1
56
1<?php
2/**
3 * Adds the Jetpack stats widget to the WordPress admin dashboard.
4 *
5 * @package jetpack
6 */
7
8use Automattic\Jetpack\Assets\Logo as Jetpack_Logo;
9use Automattic\Jetpack\Redirect;
10use Automattic\Jetpack\Stats_Admin\WP_Dashboard_Odyssey_Widget as Dashboard_Stats_Widget;
11use Automattic\Jetpack\Status;
12
13/**
14 * Class that adds the Jetpack stats widget to the WordPress admin dashboard.
15 *
16 * Note that this widget renders whether or not the stats module is active because it currently
17 * displays information about Akismet and Protect.
18 */
19class Jetpack_Stats_Dashboard_Widget {
20
21    /**
22     * Indicates whether the class initialized or not.
23     *
24     * @var bool
25     */
26    private static $initialized = false;
27
28    /**
29     * Initialize the class by calling the setup static function.
30     *
31     * @return void
32     */
33    public static function init() {
34        if ( ! self::$initialized ) {
35            self::$initialized = true;
36            self::wp_dashboard_setup();
37        }
38    }
39
40    /**
41     * Sets up the Jetpack Stats widget in the WordPress admin dashboard.
42     */
43    public static function wp_dashboard_setup() {
44
45        /**
46         * Filter whether the Jetpack Stats dashboard widget should be shown to the current user.
47         * By default, the dashboard widget is shown to users who can view_stats.
48         *
49         * @module stats
50         * @since 11.9
51         *
52         * @param bool Whether to show the widget to the current user.
53         */
54        // Temporarily show the widget to administrators for Simple sites as the view_stats capability is not available.
55        // TODO: Grant the view_stats capability to corresponding users for Simple sites.
56        $can_user_view_stats = current_user_can( 'manage_options' ) || current_user_can( 'view_stats' );
57        if ( ! apply_filters( 'jetpack_stats_dashboard_widget_show_to_user', $can_user_view_stats ) ) {
58            return;
59        }
60
61        if ( Jetpack::is_connection_ready() && Jetpack::is_module_active( 'stats' ) ) {
62            add_action( 'admin_head', array( static::class, 'admin_head' ) );
63
64            // New widget implemented in Odyssey Stats.
65            $stats_widget = new Dashboard_Stats_Widget();
66            wp_add_dashboard_widget(
67                Dashboard_Stats_Widget::DASHBOARD_WIDGET_ID,
68                /** "Stats" is a product name, do not translate. */
69                'Jetpack Stats',
70                array( $stats_widget, 'render' )
71            );
72            // Only load scripts when the widget is not hidden
73            $stats_widget->maybe_load_admin_scripts();
74        }
75    }
76
77    /**
78     * JavaScript and CSS for dashboard widget.
79     *
80     * @access public
81     * @return void
82     */
83    public static function admin_head() {
84        ?>
85            <script type="text/javascript">
86                /* <![CDATA[ */
87                jQuery( function($) {
88                    var dashStats = jQuery( '#dashboard_stats div.inside' );
89
90                    if ( dashStats.find( '.dashboard-widget-control-form' ).length ) {
91                        return;
92                    }
93
94                    if ( ! dashStats.length ) {
95                        dashStats = jQuery( '#dashboard_stats div.dashboard-widget-content' );
96                        var h = parseInt( dashStats.parent().height() ) - parseInt( dashStats.prev().height() );
97                        var args = 'width=' + dashStats.width() + '&height=' + h.toString();
98                    } else {
99                        if ( jQuery('#dashboard_stats' ).hasClass('postbox') ) {
100                            var args = 'width=' + ( dashStats.prev().width() * 2 ).toString();
101                        } else {
102                            var args = 'width=' + ( dashStats.width() * 2 ).toString();
103                        }
104                    }
105
106                    dashStats
107                        .not( '.dashboard-widget-control' )
108                        .load( 'admin.php?page=stats&noheader&dashboard&' + args, function() {
109                            jQuery( '#dashboard_stats' ).removeClass( 'is-loading' );
110                            jQuery( '#stat-chart' ).css( 'width', 'auto' );
111                        } );
112
113                    // Widget settings toggle container.
114                    var toggle = $( '.js-toggle-stats_dashboard_widget_control' );
115
116                    // Move the toggle in the widget header.
117                    toggle.appendTo( '#jetpack_summary_widget .handle-actions' );
118
119                    // Toggle settings when clicking on it.
120                    toggle.show().click( function( e ) {
121                        e.preventDefault();
122                        e.stopImmediatePropagation();
123                        $( this ).parent().toggleClass( 'controlVisible' );
124                        $( '#stats_dashboard_widget_control' ).slideToggle();
125                    } );
126                } );
127                /* ]]> */
128            </script>
129        <?php
130    }
131
132    /**
133     * Renders the widget and fires a dashboard widget action.
134     */
135    public static function render_widget() {
136        // This function won't exist if the stats module is disabled.
137        if ( function_exists( 'stats_jetpack_dashboard_widget' ) ) {
138            stats_jetpack_dashboard_widget();
139        }
140
141        /**
142         * Fires when the dashboard is loaded, but no longer used anywhere in the Jetpack plugin.
143         * The action is still available for backward compatibility.
144         *
145         * @since 3.4.0
146         */
147        do_action( 'jetpack_dashboard_widget' );
148
149        self::render_footer();
150    }
151
152    /**
153     * Load the widget footer showing brute force protection and Akismet stats.
154     */
155    public static function render_footer() {
156        ?>
157        <footer>
158        <div class="blocked-container">
159            <div class="protect">
160                <h3><?php esc_html_e( 'Brute force attack protection', 'jetpack' ); ?></h3>
161                <?php if ( Jetpack::is_module_active( 'protect' ) ) : ?>
162                    <p class="blocked-count">
163                        <?php echo esc_html( number_format_i18n( get_site_option( 'jetpack_protect_blocked_attempts', 0 ) ) ); ?>
164                    </p>
165                    <p><?php echo esc_html_x( 'Blocked malicious login attempts', '{#} Blocked malicious login attempts -- number is on a prior line, text is a caption.', 'jetpack' ); ?></p>
166                <?php elseif ( current_user_can( 'jetpack_activate_modules' ) && ! ( new Status() )->is_offline_mode() ) : ?>
167                    <a href="
168                    <?php
169                    echo esc_url(
170                        wp_nonce_url(
171                            Jetpack::admin_url(
172                                array(
173                                    'action' => 'activate',
174                                    'module' => 'protect',
175                                )
176                            ),
177                            'jetpack_activate-protect'
178                        )
179                    );
180                    ?>
181                                " class="button button-primary" title="<?php esc_attr_e( 'Jetpack helps to keep you secure from brute-force login attacks.', 'jetpack' ); ?>">
182                        <?php esc_html_e( 'Activate', 'jetpack' ); ?>
183                    </a>
184                <?php else : ?>
185                    <?php esc_html_e( 'Brute force attack protection is inactive.', 'jetpack' ); ?>
186                <?php endif; ?>
187            </div>
188
189            <div class="akismet">
190                <h3><?php esc_html_e( 'Akismet Anti-spam', 'jetpack' ); ?></h3>
191                <?php if ( is_plugin_active( 'akismet/akismet.php' ) ) : ?>
192                    <p class="blocked-count">
193                        <?php echo esc_html( number_format_i18n( get_option( 'akismet_spam_count', 0 ) ) ); ?>
194                    </p>
195                    <p><?php echo esc_html_x( 'Blocked spam comments', '{#} Spam comments blocked by Akismet -- number is on a prior line, text is a caption.', 'jetpack' ); ?></p>
196                <?php elseif ( current_user_can( 'activate_plugins' ) && ! is_wp_error( validate_plugin( 'akismet/akismet.php' ) ) ) : ?>
197                    <a href="
198                    <?php
199                    echo esc_url(
200                        wp_nonce_url(
201                            add_query_arg(
202                                array(
203                                    'action' => 'activate',
204                                    'plugin' => 'akismet/akismet.php',
205                                ),
206                                admin_url( 'plugins.php' )
207                            ),
208                            'activate-plugin_akismet/akismet.php'
209                        )
210                    );
211                    ?>
212                                " class="button button-primary">
213                        <?php esc_html_e( 'Activate', 'jetpack' ); ?>
214                    </a>
215                <?php else : ?>
216                    <p><a href="<?php echo esc_url( 'https://akismet.com/?utm_source=jetpack&utm_medium=link&utm_campaign=Jetpack%20Dashboard%20Widget%20Footer%20Link' ); ?>"><?php esc_html_e( 'Anti-spam can help to keep your blog safe from spam!', 'jetpack' ); ?></a></p>
217                <?php endif; ?>
218            </div>
219        </div>
220        <div class="footer-links">
221            <a href="<?php echo esc_url( Redirect::get_url( 'jetpack-support-wordpress-com-stats' ) ); ?>" target="_blank">
222                <?php
223                    $jetpack_logo = new Jetpack_Logo();
224                    echo $jetpack_logo->get_jp_emblem( true );// phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
225                ?>
226            </a>
227        </div>
228        </footer>
229
230        <?php
231    }
232}