-
Notifications
You must be signed in to change notification settings - Fork 0
/
premium.php
442 lines (373 loc) · 13.1 KB
/
premium.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
<?php
/**
* WPSEO Premium plugin file.
*
* @package WPSEO\Premium
*/
use Yoast\WP\SEO\Integrations\Blocks\Siblings_Block;
use Yoast\WP\SEO\Integrations\Blocks\Subpages_Block;
use Yoast\WP\SEO\Premium\Addon_Installer;
use Yoast\WP\SEO\Premium\Helpers\Prominent_Words_Helper;
use Yoast\WP\SEO\Presenters\Admin\Help_Link_Presenter;
use Yoast\WP\SEO\Repositories\Indexable_Repository;
/**
* Class WPSEO_Premium
*/
class WPSEO_Premium {
/**
* Option that stores the current version.
*
* @var string
*/
const OPTION_CURRENT_VERSION = 'wpseo_current_version';
/**
* Human readable version of the current version.
*
* @var string
*/
const PLUGIN_VERSION_NAME = '17.3';
/**
* Machine readable version for determining whether an upgrade is needed.
*
* @var string
*/
const PLUGIN_VERSION_CODE = '16';
/**
* Instance of the WPSEO_Redirect_Page class.
*
* @var WPSEO_Redirect_Page
*/
private $redirects;
/**
* List of registered classes implementing the WPSEO_WordPress_Integration interface.
*
* @var WPSEO_WordPress_Integration[]
*/
private $integrations = [];
/**
* Function that will be executed when plugin is activated.
*/
public static function install() {
if ( ! Addon_Installer::is_yoast_seo_up_to_date() ) {
delete_option( Addon_Installer::OPTION_KEY );
}
$wpseo_addon_installer = new Addon_Installer( __DIR__ );
$wpseo_addon_installer->install_or_load_yoast_seo_from_vendor_directory();
// Load the Redirect File Manager.
require_once WPSEO_PREMIUM_PATH . 'classes/redirect/redirect-file-util.php';
// Create the upload directory.
WPSEO_Redirect_File_Util::create_upload_dir();
// Enable tracking.
if ( class_exists( WPSEO_Options::class ) ) {
WPSEO_Premium_Option::register_option();
WPSEO_Options::set( 'tracking', true );
WPSEO_Options::set( 'should_redirect_after_install', true );
}
\do_action( 'wpseo_register_capabilities_premium' );
\WPSEO_Capability_Manager_Factory::get( 'premium' )->add();
}
/**
* WPSEO_Premium Constructor
*/
public function __construct() {
$this->integrations = [
'premium-metabox' => new WPSEO_Premium_Metabox(
YoastSEOPremium()->classes->get( Prominent_Words_Helper::class )
),
'premium-assets' => new WPSEO_Premium_Assets(),
'link-suggestions' => new WPSEO_Metabox_Link_Suggestions(),
'redirects-endpoint' => new WPSEO_Premium_Redirect_EndPoint( new WPSEO_Premium_Redirect_Service() ),
'redirects-undo-endpoint' => new WPSEO_Premium_Redirect_Undo_EndPoint( new WPSEO_Redirect_Manager() ),
'redirect-export-manager' => new WPSEO_Premium_Redirect_Export_Manager(),
'keyword-export-manager' => new WPSEO_Premium_Keyword_Export_Manager(),
'orphaned-post-filter' => new WPSEO_Premium_Orphaned_Post_Filter(),
'expose-javascript-shortlinks' => new WPSEO_Premium_Expose_Shortlinks(),
'multi-keyword' => new WPSEO_Multi_Keyword(),
'siblings-block' => new Siblings_Block( YoastSEO()->classes->get( Indexable_Repository::class ) ),
'subpages-block' => new Subpages_Block( YoastSEO()->classes->get( Indexable_Repository::class ) ),
];
if ( WPSEO_Options::get( 'enable_cornerstone_content' ) ) {
$this->integrations['stale-cornerstone-content-filter'] = new WPSEO_Premium_Stale_Cornerstone_Content_Filter();
}
$this->setup();
}
/**
* Sets up the Yoast SEO premium plugin.
*
* @return void
*/
private function setup() {
$this->load_textdomain();
$this->redirect_setup();
add_action( 'init', [ 'WPSEO_Premium_Option', 'register_option' ] );
add_action( 'init', [ 'WPSEO_Premium_Redirect_Option', 'register_option' ] );
if ( is_admin() ) {
// Make sure priority is below registration of other implementations of the beacon in News, Video, etc.
add_filter( 'wpseo_helpscout_beacon_settings', [ $this, 'filter_helpscout_beacon' ], 1 );
// Only register the yoast i18n when the page is a Yoast SEO page.
if ( $this->is_yoast_seo_premium_page( filter_input( INPUT_GET, 'page' ) ) ) {
$this->register_i18n_promo_class();
}
add_filter( 'wpseo_enable_tracking', '__return_true', 1 );
// Add Sub Menu page and add redirect page to admin page array.
// This should be possible in one method in the future, see #535.
add_filter( 'wpseo_submenu_pages', [ $this, 'add_submenu_pages' ], 9 );
// Add input fields to page meta post types.
add_action(
'Yoast\WP\SEO\admin_post_types_beforearchive',
[
$this,
'admin_page_meta_post_types_checkboxes',
],
10,
2
);
// Add page analysis fields to variable array key patterns.
add_filter(
'wpseo_option_titles_variable_array_key_patterns',
[ $this, 'add_variable_array_key_pattern' ]
);
// Settings.
add_action( 'admin_init', [ $this, 'register_settings' ] );
// Add Premium imports.
$this->integrations[] = new WPSEO_Premium_Import_Manager();
}
// Only activate post and term watcher if permalink structure is enabled.
if ( get_option( 'permalink_structure' ) ) {
add_action( 'admin_init', [ $this, 'init_watchers' ] );
add_action( 'rest_api_init', [ $this, 'init_watchers' ] );
}
if ( ! is_admin() ) {
// Add 404 redirect link to WordPress toolbar.
add_action( 'admin_bar_menu', [ $this, 'admin_bar_menu' ], 96 );
add_filter( 'redirect_canonical', [ $this, 'redirect_canonical_fix' ], 1, 2 );
}
add_action( 'wpseo_premium_indicator_classes', [ $this, 'change_premium_indicator' ] );
add_action( 'wpseo_premium_indicator_text', [ $this, 'change_premium_indicator_text' ] );
// Only initialize the AJAX for all tabs except settings.
$facebook_name = new WPSEO_Facebook_Profile();
$facebook_name->set_hooks();
foreach ( $this->integrations as $integration ) {
$integration->register_hooks();
}
}
/**
* Checks if the page is a premium page.
*
* @param string $page The page to check.
*
* @return bool
*/
private function is_yoast_seo_premium_page( $page ) {
$premium_pages = [ 'wpseo_redirects' ];
return in_array( $page, $premium_pages, true );
}
/**
* Registers the promotion class for our GlotPress instance.
*
* @link https://github.com/Yoast/i18n-module
*/
private function register_i18n_promo_class() {
new Yoast_I18n_v3(
[
'textdomain' => 'wordpress-seo-premium',
'project_slug' => 'wordpress-seo-premium',
'plugin_name' => 'Yoast SEO premium',
'hook' => 'wpseo_admin_promo_footer',
'api_url' => 'https://translationspress.com/app/api/yoast/wordpress-seo-premium/',
'glotpress_name' => 'Yoast Translate',
'glotpress_logo' => 'https://yoast.com/app/uploads/yoast/Yoast_Translate.svg',
'register_url' => 'https://yoa.st/translationspress',
]
);
}
/**
* Sets the autoloader for the redirects and instantiates the redirect page object.
*
* @return void
*/
private function redirect_setup() {
$this->redirects = new WPSEO_Redirect_Page();
// Adds integration that filters redirected entries from the sitemap.
$this->integrations['redirect-sitemap-filter'] = new WPSEO_Redirect_Sitemap_Filter( home_url() );
}
/**
* Initialize the watchers for the posts and the terms
*/
public function init_watchers() {
// The Post Watcher.
$post_watcher = new WPSEO_Post_Watcher();
$post_watcher->register_hooks();
// The Term Watcher.
$term_watcher = new WPSEO_Term_Watcher();
$term_watcher->register_hooks();
}
/**
* Hooks into the `redirect_canonical` filter to catch ongoing redirects and move them to the correct spot
*
* @param string $redirect_url The target url where the requested URL will be redirected to.
* @param string $requested_url The current requested URL.
*
* @return string
*/
public function redirect_canonical_fix( $redirect_url, $requested_url ) {
$redirects = new WPSEO_Redirect_Option( false );
$path = wp_parse_url( $requested_url, PHP_URL_PATH );
$redirect = $redirects->get( $path );
if ( $redirect === false ) {
return $redirect_url;
}
$redirect_url = $redirect->get_origin();
if ( substr( $redirect_url, 0, 1 ) === '/' ) {
$redirect_url = home_url( $redirect_url );
}
wp_redirect( $redirect_url, $redirect->get_type(), 'Yoast SEO Premium' );
exit;
}
/**
* Add 'Create Redirect' option to admin bar menu on 404 pages
*/
public function admin_bar_menu() {
// Prevent function from running if the page is not a 404 page or the user has not the right capabilities to create redirects.
if ( ! is_404() || ! WPSEO_Capability_Utils::current_user_can( 'wpseo_manage_options' ) ) {
return;
}
global $wp, $wp_admin_bar;
$parsed_url = wp_parse_url( home_url( $wp->request ) );
if ( ! is_array( $parsed_url ) || empty( $parsed_url['path'] ) ) {
return;
}
$old_url = WPSEO_Redirect_Util::strip_base_url_path_from_url( home_url(), $parsed_url['path'] );
if ( isset( $parsed_url['query'] ) && $parsed_url['query'] !== '' ) {
$old_url .= '?' . $parsed_url['query'];
}
$old_url = rawurlencode( $old_url );
$node = [
'id' => 'wpseo-premium-create-redirect',
'title' => __( 'Create Redirect', 'wordpress-seo-premium' ),
'href' => admin_url( 'admin.php?page=wpseo_redirects&old_url=' . $old_url ),
];
$wp_admin_bar->add_menu( $node );
}
/**
* Add page analysis to array with variable array key patterns
*
* @param array $patterns Array with patterns for page analysis.
*
* @return array
*/
public function add_variable_array_key_pattern( $patterns ) {
if ( in_array( 'page-analyse-extra-', $patterns, true ) === false ) {
$patterns[] = 'page-analyse-extra-';
}
return $patterns;
}
/**
* This hook will add an input-field for specifying custom fields for page analysis.
*
* The values will be comma-separated and will target the belonging field in the post_meta. Page analysis will
* use the content of it by sticking it to the post_content.
*
* @param Yoast_Form $yform The Yoast_Form object.
* @param string $name The post type name.
*/
public function admin_page_meta_post_types_checkboxes( $yform, $name ) {
$custom_fields_help_link = new Help_Link_Presenter(
WPSEO_Shortlinker::get( 'https://yoa.st/4cr' ),
__( 'Learn more about including custom fields in the page analysis', 'wordpress-seo-premium' )
);
echo '<div class="yoast-settings-section yoast-settings-section--last">';
$yform->textinput_extra_content(
'page-analyse-extra-' . $name,
esc_html__( 'Custom fields to include in page analysis', 'wordpress-seo-premium' ),
[
'extra_content' => $custom_fields_help_link,
]
);
echo '</div>';
}
/**
* Function adds the premium pages to the Yoast SEO menu
*
* @param array $submenu_pages Array with the configuration for the submenu pages.
*
* @return array
*/
public function add_submenu_pages( $submenu_pages ) {
/**
* Filter: 'wpseo_premium_manage_redirects_role' - Change the minimum rule to access and change the site redirects
*
* @api string wpseo_manage_redirects
*/
$submenu_pages[] = [
'wpseo_dashboard',
'',
__( 'Redirects', 'wordpress-seo-premium' ) . ' <span class="yoast-badge yoast-premium-badge"></span>',
'wpseo_manage_redirects',
'wpseo_redirects',
[ $this->redirects, 'display' ],
];
return $submenu_pages;
}
/**
* Change premium indicator to green when premium is enabled
*
* @param string[] $classes The current classes for the indicator.
*
* @return string[] The new classes for the indicator.
*/
public function change_premium_indicator( $classes ) {
$class_no = array_search( 'wpseo-premium-indicator--no', $classes, true );
if ( $class_no !== false ) {
unset( $classes[ $class_no ] );
$classes[] = 'wpseo-premium-indicator--yes';
}
return $classes;
}
/**
* Replaces the screen reader text for the premium indicator.
*
* @param string $text The original text.
*
* @return string The new text.
*/
public function change_premium_indicator_text( $text ) {
return __( 'Enabled', 'wordpress-seo-premium' );
}
/**
* Register the premium settings
*/
public function register_settings() {
register_setting( 'yoast_wpseo_redirect_options', 'wpseo_redirect' );
}
/**
* Output admin css in admin head
*/
public function admin_css() {
echo "<style type='text/css'>#wpseo_content_top{ padding-left: 0; margin-left: 0; }</style>";
}
/**
* Load textdomain
*/
private function load_textdomain() {
load_plugin_textdomain( 'wordpress-seo-premium', false, dirname( WPSEO_PREMIUM_BASENAME ) . '/languages/' );
}
/**
* Initializes the HelpScout support modal for WPSEO settings pages.
*
* @param array $helpscout_settings The helpscout settings.
*
* @return array The HelpScout beacon settings array.
*/
public function filter_helpscout_beacon( $helpscout_settings ) {
$beacon_id = '1ae02e91-5865-4f13-b220-7daed946ba25';
$helpscout_settings['products'][] = WPSEO_Addon_Manager::PREMIUM_SLUG;
// Set the beacon to the premium beacon for all pages.
foreach ( $helpscout_settings['pages_ids'] as $page => $beacon ) {
$helpscout_settings['pages_ids'][ $page ] = $beacon_id;
}
// Add the redirects page.
$helpscout_settings['pages_ids']['wpseo_redirects'] = $beacon_id;
return $helpscout_settings;
}
}