Completed
Push — add/apps-card ( 6b0d49...a7e330 )
by
unknown
58:41 queued 50:34
created

_inc/lib/admin-pages/class.jetpack-react-page.php (1 issue)

Upgrade to new PHP Analysis Engine

These results are based on our legacy PHP analysis, consider migrating to our new PHP analysis engine instead. Learn more

1
<?php
2
include_once( 'class.jetpack-admin-page.php' );
3
4
// Builds the landing page and its menu
5
class Jetpack_React_Page extends Jetpack_Admin_Page {
6
7
	protected $dont_show_if_not_active = false;
8
9
	protected $is_redirecting = false;
10
11
	function get_page_hook() {
12
		$title = _x( 'Jetpack', 'The menu item label', 'jetpack' );
13
14
		// Add the main admin Jetpack menu
15
		return add_menu_page( 'Jetpack', $title, 'jetpack_admin_page', 'jetpack', array( $this, 'render' ), 'div' );
16
	}
17
18
	function add_page_actions( $hook ) {
19
		/** This action is documented in class.jetpack.php */
20
		do_action( 'jetpack_admin_menu', $hook );
21
22
		// Place the Jetpack menu item on top and others in the order they appear
23
		add_filter( 'custom_menu_order',         '__return_true' );
24
		add_filter( 'menu_order',                array( $this, 'jetpack_menu_order' ) );
25
26
		if ( ! isset( $_GET['page'] ) || 'jetpack' !== $_GET['page'] || ! empty( $_GET['configure'] ) ) {
27
			return; // No need to handle the fallback redirection if we are not on the Jetpack page
28
		}
29
30
		// Adding a redirect meta tag for older WordPress versions or if the REST API is disabled
31
		if ( $this->is_wp_version_too_old() || ! $this->is_rest_api_enabled() ) {
32
			$this->is_redirecting = true;
33
			add_action( 'admin_head', array( $this, 'add_fallback_head_meta' ) );
34
		}
35
36
		// Adding a redirect meta tag wrapped in noscript tags for all browsers in case they have JavaScript disabled
37
		add_action( 'admin_head', array( $this, 'add_noscript_head_meta' ) );
38
39
		// Adding a redirect tag wrapped in browser conditional comments
40
		add_action( 'admin_head', array( $this, 'add_legacy_browsers_head_script' ) );
41
	}
42
43
	/**
44
	 * Add Jetpack Dashboard sub-link and point it to AAG if the user can view stats, manage modules or if Protect is active.
45
	 *
46
	 * Works in Dev Mode or when user is connected.
47
	 *
48
	 * @since 4.3.0
49
	 */
50
	function jetpack_add_dashboard_sub_nav_item() {
51 View Code Duplication
		if ( Jetpack::is_development_mode() || Jetpack::is_active() ) {
52
			global $submenu;
53
			if ( current_user_can( 'jetpack_admin_page' ) ) {
54
				$submenu['jetpack'][] = array( __( 'Dashboard', 'jetpack' ), 'jetpack_admin_page', 'admin.php?page=jetpack#/dashboard' );
55
			}
56
		}
57
	}
58
59
	/**
60
	 * If user is allowed to see the Jetpack Admin, add Settings sub-link.
61
	 *
62
	 * @since 4.3.0
63
	 */
64
	function jetpack_add_settings_sub_nav_item() {
65 View Code Duplication
		if ( ( Jetpack::is_development_mode() || Jetpack::is_active() ) && current_user_can( 'jetpack_admin_page' ) && current_user_can( 'edit_posts' ) ) {
66
			global $submenu;
67
			$submenu['jetpack'][] = array( __( 'Settings', 'jetpack' ), 'jetpack_admin_page', 'admin.php?page=jetpack#/settings' );
68
		}
69
	}
70
71
	function add_fallback_head_meta() {
72
		echo '<meta http-equiv="refresh" content="0; url=?page=jetpack_modules">';
73
	}
74
75
	function add_noscript_head_meta() {
76
		echo '<noscript>';
77
		$this->add_fallback_head_meta();
78
		echo '</noscript>';
79
	}
80
81
	function add_legacy_browsers_head_script() {
82
		echo
83
			"<script type=\"text/javascript\">\n"
84
			. "/*@cc_on\n"
85
			. "if ( @_jscript_version <= 10) {\n"
86
			. "window.location.href = '?page=jetpack_modules';\n"
87
			. "}\n"
88
			. "@*/\n"
89
			. "</script>";
90
	}
91
92 View Code Duplication
	function jetpack_menu_order( $menu_order ) {
93
		$jp_menu_order = array();
94
95
		foreach ( $menu_order as $index => $item ) {
96
			if ( $item != 'jetpack' )
97
				$jp_menu_order[] = $item;
98
99
			if ( $index == 0 )
100
				$jp_menu_order[] = 'jetpack';
101
		}
102
103
		return $jp_menu_order;
104
	}
105
106
	// Render the configuration page for the module if it exists and an error
107
	// screen if the module is not configurable
108
	// @todo remove when real settings are in place
109
	function render_nojs_configurable( $module_name ) {
110
		$module_name = preg_replace( '/[^\da-z\-]+/', '', $_GET['configure'] );
111
112
		include_once( JETPACK__PLUGIN_DIR . '_inc/header.php' );
113
		echo '<div class="wrap configure-module">';
114
115
		if ( Jetpack::is_module( $module_name ) && current_user_can( 'jetpack_configure_modules' ) ) {
116
			Jetpack::admin_screen_configure_module( $module_name );
117
		} else {
118
			echo '<h2>' . esc_html__( 'Error, bad module.', 'jetpack' ) . '</h2>';
119
		}
120
121
		echo '</div><!-- /wrap -->';
122
	}
123
124
	function page_render() {
125
		// Handle redirects to configuration pages
126
		if ( ! empty( $_GET['configure'] ) ) {
127
			return $this->render_nojs_configurable( $_GET['configure'] );
128
		}
129
130
		/** This action is already documented in views/admin/admin-page.php */
131
		do_action( 'jetpack_notices' );
132
133
		// Try fetching by patch
134
		$static_html = @file_get_contents( JETPACK__PLUGIN_DIR . '_inc/build/static.html' );
135
136 View Code Duplication
		if ( false === $static_html ) {
137
138
			// If we still have nothing, display an error
139
			echo '<p>';
140
			esc_html_e( 'Error fetching static.html. Try running: ', 'jetpack' );
141
			echo '<code>yarn distclean && yarn build</code>';
142
			echo '</p>';
143
		} else {
144
145
			// We got the static.html so let's display it
146
			echo $static_html;
147
		}
148
	}
149
150
	function get_i18n_data() {
151
152
		$i18n_json = JETPACK__PLUGIN_DIR . 'languages/json/jetpack-' . jetpack_get_user_locale() . '.json';
153
154
		if ( is_file( $i18n_json ) && is_readable( $i18n_json ) ) {
155
			$locale_data = @file_get_contents( $i18n_json );
156
			if ( $locale_data ) {
157
				return $locale_data;
158
			}
159
		}
160
161
		// Return empty if we have nothing to return so it doesn't fail when parsed in JS
162
		return '{}';
163
	}
164
165
	/**
166
	 * Gets array of any Jetpack notices that have been dismissed.
167
	 *
168
	 * @since 4.0.1
169
	 * @return mixed|void
170
	 */
171
	function get_dismissed_jetpack_notices() {
172
		$jetpack_dismissed_notices = get_option( 'jetpack_dismissed_notices', array() );
173
		/**
174
		 * Array of notices that have been dismissed.
175
		 *
176
		 * @since 4.0.1
177
		 *
178
		 * @param array $jetpack_dismissed_notices If empty, will not show any Jetpack notices.
179
		 */
180
		$dismissed_notices = apply_filters( 'jetpack_dismissed_notices', $jetpack_dismissed_notices );
181
		return $dismissed_notices;
182
	}
183
184
	function additional_styles() {
185
		$rtl = is_rtl() ? '.rtl' : '';
186
187
		wp_enqueue_style( 'dops-css', plugins_url( "_inc/build/admin.dops-style$rtl.css", JETPACK__PLUGIN_FILE ), array(), JETPACK__VERSION );
188
		wp_enqueue_style( 'components-css', plugins_url( "_inc/build/style.min$rtl.css", JETPACK__PLUGIN_FILE ), array(), JETPACK__VERSION );
189
	}
190
191
	function page_admin_scripts() {
192
		if ( $this->is_redirecting ) {
193
			return; // No need for scripts on a fallback page
194
		}
195
196
		$is_dev_mode = Jetpack::is_development_mode();
197
198
		// Enqueue jp.js and localize it
199
		wp_enqueue_script( 'react-plugin', plugins_url( '_inc/build/admin.js', JETPACK__PLUGIN_FILE ), array(), JETPACK__VERSION, true );
200
201
		if ( ! $is_dev_mode ) {
202
			// Required for Analytics
203
			wp_enqueue_script( 'jp-tracks', '//stats.wp.com/w.js', array(), gmdate( 'YW' ), true );
204
		}
205
206
		// Collecting roles that can view site stats
207
		$stats_roles = array();
208
		$enabled_roles = function_exists( 'stats_get_option' ) ? stats_get_option( 'roles' ) : array( 'administrator' );
209
		foreach( get_editable_roles() as $slug => $role ) {
210
			$stats_roles[ $slug ] = array(
211
				'name' => translate_user_role( $role['name'] ),
212
				'canView' => is_array( $enabled_roles ) ? in_array( $slug, $enabled_roles, true ) : false,
213
			);
214
		}
215
216
		$response = rest_do_request( new WP_REST_Request( 'GET', '/jetpack/v4/module/all' ) );
217
		$modules = $response->get_data();
218
219
		// Preparing translated fields for JSON encoding by transforming all HTML entities to
220
		// respective characters.
221
		foreach( $modules as $slug => $data ) {
222
			$modules[ $slug ]['name'] = html_entity_decode( $data['name'] );
223
			$modules[ $slug ]['description'] = html_entity_decode( $data['description'] );
224
			$modules[ $slug ]['short_description'] = html_entity_decode( $data['short_description'] );
225
			$modules[ $slug ]['long_description'] = html_entity_decode( $data['long_description'] );
226
		}
227
228
		// Get last post, to build the link to Customizer in the Related Posts module.
229
		$last_post = get_posts( array( 'posts_per_page' => 1 ) );
230
		$last_post = isset( $last_post[0] ) && $last_post[0] instanceof WP_Post
231
			? get_permalink( $last_post[0]->ID )
232
			: get_home_url();
233
234
		// Get information about current theme.
235
		$current_theme = wp_get_theme();
236
237
		// Get all themes that Infinite Scroll provides support for natively.
238
		$inf_scr_support_themes = array();
239
		foreach ( Jetpack::glob_php( JETPACK__PLUGIN_DIR . 'modules/infinite-scroll/themes' ) as $path ) {
240
			if ( is_readable( $path ) ) {
241
				$inf_scr_support_themes[] = basename( $path, '.php' );
242
			}
243
		}
244
245
		// Add objects to be passed to the initial state of the app
246
		wp_localize_script( 'react-plugin', 'Initial_State', array(
247
			'WP_API_root' => esc_url_raw( rest_url() ),
248
			'WP_API_nonce' => wp_create_nonce( 'wp_rest' ),
249
			'pluginBaseUrl' => plugins_url( '', JETPACK__PLUGIN_FILE ),
250
			'connectionStatus' => array(
251
				'isActive'  => Jetpack::is_active(),
252
				'isStaging' => Jetpack::is_staging_site(),
253
				'devMode'   => array(
254
					'isActive' => $is_dev_mode,
255
					'constant' => defined( 'JETPACK_DEV_DEBUG' ) && JETPACK_DEV_DEBUG,
256
					'url'      => site_url() && false === strpos( site_url(), '.' ),
257
					'filter'   => apply_filters( 'jetpack_development_mode', false ),
258
				),
259
				'isPublic'	=> '1' == get_option( 'blog_public' ),
260
				'isInIdentityCrisis' => Jetpack::validate_sync_error_idc_option(),
261
			),
262
			'dismissedNotices' => $this->get_dismissed_jetpack_notices(),
263
			'isDevVersion' => Jetpack::is_development_version(),
264
			'currentVersion' => JETPACK__VERSION,
265
			'getModules' => $modules,
266
			'showJumpstart' => jetpack_show_jumpstart(),
267
			'showHolidaySnow' => function_exists( 'jetpack_show_holiday_snow_option' ) ? jetpack_show_holiday_snow_option() : false,
268
			'rawUrl' => Jetpack::build_raw_urls( get_home_url() ),
269
			'adminUrl' => esc_url( admin_url() ),
270
			'stats' => array(
271
				// data is populated asynchronously on page load
272
				'data'  => array(
273
					'general' => false,
274
					'day'     => false,
275
					'week'    => false,
276
					'month'   => false,
277
				),
278
				'roles' => $stats_roles,
279
			),
280
			'settings' => $this->get_flattened_settings( $modules ),
281
			'settingNames' => array(
282
				'jetpack_holiday_snow_enabled' => function_exists( 'jetpack_holiday_snow_option_name' ) ? jetpack_holiday_snow_option_name() : false,
283
			),
284
			'userData' => array(
285
//				'othersLinked' => Jetpack::get_other_linked_admins(),
286
				'currentUser'  => jetpack_current_user_data(),
287
			),
288
			'siteData' => array(
289
				'icon' => has_site_icon()
290
					? apply_filters( 'jetpack_photon_url', get_site_icon_url(), array( 'w' => 64 ) )
291
					: '',
292
				'siteVisibleToSearchEngines' => '1' == get_option( 'blog_public' ),
293
				/**
294
				 * Whether promotions are visible or not.
295
				 *
296
				 * @since 4.8.0
297
				 *
298
				 * @param bool $are_promotions_active Status of promotions visibility. True by default.
299
				 */
300
				'showPromotions' => apply_filters( 'jetpack_show_promotions', true ),
301
			),
302
			'themeData' => array(
303
				'name'      => $current_theme->get( 'Name' ),
304
				'hasUpdate' => (bool) get_theme_update_available( $current_theme ),
305
				'support'   => array(
306
					'infinite-scroll' => current_theme_supports( 'infinite-scroll' ) || in_array( $current_theme->get_stylesheet(), $inf_scr_support_themes ),
307
				),
308
			),
309
			'locale' => $this->get_i18n_data(),
310
			'localeSlug' => join( '-', explode( '_', jetpack_get_user_locale() ) ),
311
			'jetpackStateNotices' => array(
312
				'messageCode' => Jetpack::state( 'message' ),
313
				'errorCode' => Jetpack::state( 'error' ),
314
				'errorDescription' => Jetpack::state( 'error_description' ),
315
			),
316
			'tracksUserData' => Jetpack_Tracks_Client::get_connected_user_tracks_identity(),
317
			'currentIp' => function_exists( 'jetpack_protect_get_ip' ) ? jetpack_protect_get_ip() : false,
318
			'lastPostUrl' => esc_url( $last_post ),
319
		) );
320
	}
321
322
	/**
323
	 * Returns an array of modules and settings both as first class members of the object.
324
	 *
325
	 * @param array $modules the result of an API request to get all modules.
326
	 *
327
	 * @return array flattened settings with modules.
328
	 */
329
	function get_flattened_settings( $modules ) {
330
		$core_api_endpoint = new Jetpack_Core_API_Data();
331
		$settings = $core_api_endpoint->get_all_options();
332
		return $settings->data;
333
	}
334
}
335
336
/*
337
 * Only show Jump Start on first activation.
338
 * Any option 'jumpstart' other than 'new connection' will hide it.
339
 *
340
 * The option can be of 4 things, and will be stored as such:
341
 * new_connection      : Brand new connection - Show
342
 * jumpstart_activated : Jump Start has been activated - dismiss
343
 * jetpack_action_taken: Manual activation of a module already happened - dismiss
344
 * jumpstart_dismissed : Manual dismissal of Jump Start - dismiss
345
 *
346
 * @todo move to functions.global.php when available
0 ignored issues
show
Coding Style Best Practice introduced by
Comments for TODO tasks are often forgotten in the code; it might be better to use a dedicated issue tracker.
Loading history...
347
 * @since 3.6
348
 * @return bool | show or hide
349
 */
350
function jetpack_show_jumpstart() {
351
	if ( ! Jetpack::is_active() ) {
352
		return false;
353
	}
354
	$jumpstart_option = Jetpack_Options::get_option( 'jumpstart' );
355
356
	$hide_options = array(
357
		'jumpstart_activated',
358
		'jetpack_action_taken',
359
		'jumpstart_dismissed'
360
	);
361
362
	if ( ! $jumpstart_option || in_array( $jumpstart_option, $hide_options ) ) {
363
		return false;
364
	}
365
366
	return true;
367
}
368
369
/**
370
 * Gather data about the current user.
371
 *
372
 * @since 4.1.0
373
 *
374
 * @return array
375
 */
376
function jetpack_current_user_data() {
377
	$current_user = wp_get_current_user();
378
	$is_master_user = $current_user->ID == Jetpack_Options::get_option( 'master_user' );
379
	$dotcom_data    = Jetpack::get_connected_user_data();
380
	// Add connected user gravatar to the returned dotcom_data.
381
	$dotcom_data['avatar'] = get_avatar_url( $dotcom_data['email'], array( 'size' => 64, 'default' => 'mysteryman' ) );
382
383
	$current_user_data = array(
384
		'isConnected' => Jetpack::is_user_connected( $current_user->ID ),
385
		'isMaster'    => $is_master_user,
386
		'username'    => $current_user->user_login,
387
		'wpcomUser'   => $dotcom_data,
388
		'gravatar'    => get_avatar( $current_user->ID, 40, 'mm', '', array( 'force_display' => true ) ),
389
		'permissions' => array(
390
			'admin_page'         => current_user_can( 'jetpack_admin_page' ),
391
			'connect'            => current_user_can( 'jetpack_connect' ),
392
			'disconnect'         => current_user_can( 'jetpack_disconnect' ),
393
			'manage_modules'     => current_user_can( 'jetpack_manage_modules' ),
394
			'network_admin'      => current_user_can( 'jetpack_network_admin_page' ),
395
			'network_sites_page' => current_user_can( 'jetpack_network_sites_page' ),
396
			'edit_posts'         => current_user_can( 'edit_posts' ),
397
			'publish_posts'      => current_user_can( 'publish_posts' ),
398
			'manage_options'     => current_user_can( 'manage_options' ),
399
			'view_stats'		 => current_user_can( 'view_stats' ),
400
			'manage_plugins'	 => current_user_can( 'install_plugins' )
401
									&& current_user_can( 'activate_plugins' )
402
									&& current_user_can( 'update_plugins' )
403
									&& current_user_can( 'delete_plugins' ),
404
		),
405
	);
406
407
	return $current_user_data;
408
}
409
410
/**
411
 * Set the admin language, based on user language.
412
 *
413
 * @since 4.5.0
414
 *
415
 * @return string
416
 *
417
 * @todo Remove this function when WordPress 4.8 is released
418
 * and replace `jetpack_get_user_locale()` in this file with `get_user_locale()`.
419
 */
420
function jetpack_get_user_locale() {
421
	$locale = get_locale();
422
423
	if ( function_exists( 'get_user_locale' ) ) {
424
		$locale = get_user_locale();
425
	}
426
427
	return $locale;
428
}
429