You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
348 lines
9.3 KiB
PHTML
348 lines
9.3 KiB
PHTML
7 months ago
|
<?php
|
||
|
/**
|
||
|
* The Beta Opt-in functionality.
|
||
|
*
|
||
|
* @package RankMath
|
||
|
* @subpackage RankMath\Version_Control
|
||
|
*/
|
||
|
|
||
|
namespace RankMath;
|
||
|
|
||
|
use RankMath\Traits\Hooker;
|
||
|
use RankMath\Helpers\Str;
|
||
|
use RankMath\Helpers\Param;
|
||
|
|
||
|
defined( 'ABSPATH' ) || exit;
|
||
|
|
||
|
/**
|
||
|
* Beta_Optin class.
|
||
|
*/
|
||
|
class Beta_Optin {
|
||
|
|
||
|
use Hooker;
|
||
|
|
||
|
/**
|
||
|
* Beta changelog URL.
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
const BETA_CHANGELOG_URL = 'https://rankmath.com/changelog/beta/';
|
||
|
|
||
|
/**
|
||
|
* Placeholder for opening tag inserted with JS.
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
const NOTICE_START_MARKER = '▷';
|
||
|
|
||
|
/**
|
||
|
* Placeholder for closing tag inserted with JS.
|
||
|
*
|
||
|
* @var string
|
||
|
*/
|
||
|
const NOTICE_END_MARKER = '◁';
|
||
|
|
||
|
/**
|
||
|
* Holds the fetched trunk version in memory to avoid fetching multiple times.
|
||
|
*
|
||
|
* @var mixed
|
||
|
*/
|
||
|
public $trunk_version = false;
|
||
|
|
||
|
/**
|
||
|
* Actions and filters.
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
public function hooks() {
|
||
|
$this->filter( 'site_transient_update_plugins', 'transient_update_plugins' );
|
||
|
$this->action( 'in_plugin_update_message-seo-by-rank-math/rank-math.php', 'plugin_update_message', 10, 2 );
|
||
|
$this->action( 'install_plugins_pre_plugin-information', 'beta_plugin_information' );
|
||
|
$this->action( 'admin_footer', 'beta_changelog_link_js' );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Replace plugin info popup for beta versions.
|
||
|
*/
|
||
|
public function beta_plugin_information() {
|
||
|
if ( 'seo-by-rank-math' !== Param::request( 'plugin' ) ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$transient = get_site_transient( 'update_plugins' );
|
||
|
if ( self::has_beta_update( $transient ) ) {
|
||
|
// No-js fallback.
|
||
|
echo '<html><head></head><body style="margin: 0;"><iframe src="' . esc_attr( self::BETA_CHANGELOG_URL ) . '" style="width: 100%; height: 100%;"></body></html>';
|
||
|
exit;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Check if Rank Math update is a beta update in the transient.
|
||
|
*
|
||
|
* @param mixed $transient Transient value.
|
||
|
* @return boolean If it is a beta update or not.
|
||
|
*/
|
||
|
public static function has_beta_update( $transient ) {
|
||
|
return (
|
||
|
is_object( $transient )
|
||
|
&& ! empty( $transient->response )
|
||
|
&& ! empty( $transient->response['seo-by-rank-math/rank-math.php'] )
|
||
|
&& ! empty( $transient->response['seo-by-rank-math/rank-math.php']->is_beta )
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get all available versions of Rank Math.
|
||
|
*
|
||
|
* @param boolean $beta Include beta versions.
|
||
|
*
|
||
|
* @return array List of versions and download URLs.
|
||
|
*/
|
||
|
public static function get_available_versions( $beta = false ) {
|
||
|
$versions = [];
|
||
|
$plugin_info = Version_Control::get_plugin_info();
|
||
|
|
||
|
foreach ( (array) $plugin_info['versions'] as $version => $url ) {
|
||
|
if ( ! self::is_eligible_version( $version, $beta ) ) {
|
||
|
continue;
|
||
|
}
|
||
|
$versions[ $version ] = $url;
|
||
|
}
|
||
|
|
||
|
uksort( $versions, 'version_compare' );
|
||
|
|
||
|
return $versions;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Check if version should be in the dropdown.
|
||
|
*
|
||
|
* @param string $version Version number.
|
||
|
* @param boolean $beta If beta versions should be included or not.
|
||
|
*
|
||
|
* @return boolean If version should be in the dropdown.
|
||
|
*/
|
||
|
public static function is_eligible_version( $version, $beta ) {
|
||
|
if ( 'trunk' === $version ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if ( ! $beta && Str::contains( 'beta', $version ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get latest version available.
|
||
|
*
|
||
|
* @return string Latest version number.
|
||
|
*/
|
||
|
public static function get_latest_version() {
|
||
|
$plugin_info = Version_Control::get_plugin_info();
|
||
|
return $plugin_info['version'];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get latest beta version available.
|
||
|
*
|
||
|
* @return string Latest beta version number.
|
||
|
*/
|
||
|
public function get_latest_beta_version() {
|
||
|
$version = get_transient( 'rank_math_trunk_version' );
|
||
|
if ( ! $version || $this->is_check_requested() ) {
|
||
|
$version = $this->fetch_trunk_version();
|
||
|
}
|
||
|
|
||
|
$beta = 0;
|
||
|
if ( Str::contains( 'beta', $version ) ) {
|
||
|
$beta = $version;
|
||
|
}
|
||
|
|
||
|
return $beta;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Fetch latest plugin file from public SVN and get version number.
|
||
|
*
|
||
|
* @return string
|
||
|
*/
|
||
|
public function fetch_trunk_version() {
|
||
|
if ( false !== $this->trunk_version ) {
|
||
|
return $this->trunk_version;
|
||
|
}
|
||
|
|
||
|
$this->trunk_version = 0;
|
||
|
|
||
|
$response = wp_remote_get( 'https://plugins.svn.wordpress.org/seo-by-rank-math/trunk/rank-math.php' );
|
||
|
if ( ! is_array( $response ) || is_wp_error( $response ) ) {
|
||
|
return $this->trunk_version;
|
||
|
}
|
||
|
|
||
|
$plugin_file = wp_remote_retrieve_body( $response );
|
||
|
|
||
|
preg_match( '/Version:\s+([0-9a-zA-Z.-]+)\s*$/m', $plugin_file, $matches );
|
||
|
if ( empty( $matches[1] ) ) {
|
||
|
return $this->trunk_version;
|
||
|
}
|
||
|
|
||
|
$this->trunk_version = $matches[1];
|
||
|
set_transient( 'rank_math_trunk_version', $this->trunk_version, ( 12 * HOUR_IN_SECONDS ) );
|
||
|
return $this->trunk_version;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Inject beta in the `update_plugins` transient to be able to update to it.
|
||
|
*
|
||
|
* @param mixed $value Original value.
|
||
|
*
|
||
|
* @return mixed New value.
|
||
|
*/
|
||
|
public function transient_update_plugins( $value ) {
|
||
|
$beta_version = $this->get_latest_beta_version();
|
||
|
$new_version = isset( $value->response['seo-by-rank-math/rank-math.php'] ) && ! empty( $value->response['seo-by-rank-math/rank-math.php']->new_version ) ? $value->response['seo-by-rank-math/rank-math.php']->new_version : 0;
|
||
|
|
||
|
if ( ! $beta_version ) {
|
||
|
return $value;
|
||
|
}
|
||
|
|
||
|
if ( version_compare( $beta_version, rank_math()->version, '>' ) && version_compare( $beta_version, $new_version, '>' ) ) {
|
||
|
$value = $this->inject_beta( $value, $beta_version );
|
||
|
}
|
||
|
|
||
|
return $value;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Inject beta update in the transient value.
|
||
|
*
|
||
|
* @param mixed $value Transient value.
|
||
|
* @param string $beta_version Beta version number.
|
||
|
*
|
||
|
* @return mixed New transient value.
|
||
|
*/
|
||
|
public function inject_beta( $value, $beta_version ) {
|
||
|
if ( empty( $value ) ) {
|
||
|
$value = new \stdClass();
|
||
|
}
|
||
|
|
||
|
if ( empty( $value->response ) ) {
|
||
|
$value->response = [];
|
||
|
}
|
||
|
|
||
|
$value->response['seo-by-rank-math/rank-math.php'] = new \stdClass();
|
||
|
|
||
|
$plugin_data = Version_Control::get_plugin_data( $beta_version, 'https://downloads.wordpress.org/plugin/seo-by-rank-math.zip' );
|
||
|
foreach ( $plugin_data as $prop_key => $prop_value ) {
|
||
|
$value->response['seo-by-rank-math/rank-math.php']->{$prop_key} = $prop_value;
|
||
|
}
|
||
|
|
||
|
$value->response['seo-by-rank-math/rank-math.php']->is_beta = true;
|
||
|
$value->response['seo-by-rank-math/rank-math.php']->upgrade_notice = self::NOTICE_START_MARKER . ' ' . __( 'This update will install a beta version of Rank Math.', 'rank-math' ) . ' ' . self::NOTICE_END_MARKER;
|
||
|
|
||
|
if ( empty( $value->no_update ) ) {
|
||
|
$value->no_update = [];
|
||
|
}
|
||
|
|
||
|
unset( $value->no_update['seo-by-rank-math/rank-math.php'] );
|
||
|
|
||
|
return $value;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Add warning about beta version in the update notice.
|
||
|
*
|
||
|
* @param array $plugin_data An array of plugin metadata.
|
||
|
* @param array $response An array of metadata about the available plugin update.
|
||
|
* @return void
|
||
|
*/
|
||
|
public function plugin_update_message( $plugin_data, $response ) {
|
||
|
if ( empty( $plugin_data['is_beta'] ) ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
printf(
|
||
|
'</p><p class="rank-math-beta-update-notice">%s',
|
||
|
esc_html__( 'This update will install a beta version of Rank Math.', 'rank-math' )
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Add Javascript to open beta changelog link in a new tab instead of the modal.
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
public function beta_changelog_link_js() {
|
||
|
$screen = get_current_screen();
|
||
|
|
||
|
$applicable_screens = [
|
||
|
'update-core',
|
||
|
'plugins',
|
||
|
'update-core-network',
|
||
|
'plugins-network',
|
||
|
];
|
||
|
|
||
|
if ( empty( $screen->base ) || ! in_array( $screen->base, $applicable_screens, true ) ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$transient = get_site_transient( 'update_plugins' );
|
||
|
if ( ! self::has_beta_update( $transient ) ) {
|
||
|
return;
|
||
|
}
|
||
|
?>
|
||
|
<script type="text/javascript">
|
||
|
jQuery( document ).ready( function( $ ) {
|
||
|
// Change our link.
|
||
|
$('.open-plugin-details-modal').each( function( index, element ) {
|
||
|
if ( element.href.indexOf( 'plugin=seo-by-rank-math§ion=changelog' ) !== -1 ) {
|
||
|
// Found our link.
|
||
|
$( element )
|
||
|
.removeClass( 'open-plugin-details-modal thickbox' )
|
||
|
.attr( 'href', '<?php echo esc_js( self::BETA_CHANGELOG_URL ); ?>' )
|
||
|
.attr( 'target', '_blank' );
|
||
|
return false;
|
||
|
}
|
||
|
} );
|
||
|
|
||
|
// Change our notice.
|
||
|
<?php if ( 'update-core' === $screen->base || 'update-core-network' === $screen->base ) { ?>
|
||
|
$('td.plugin-title').each( function( index, element ) {
|
||
|
var contents = $( element ).html();
|
||
|
if ( contents.indexOf( '<?php echo esc_js( html_entity_decode( self::NOTICE_START_MARKER ) ); ?>' ) !== -1 && contents.indexOf( '<?php echo esc_js( html_entity_decode( self::NOTICE_END_MARKER ) ); ?>' ) !== -1 ) {
|
||
|
contents = contents
|
||
|
.replace( '<?php echo esc_js( html_entity_decode( self::NOTICE_START_MARKER ) ); ?>', '</p><div class="update-message notice inline notice-warning notice-alt rank-math-beta-update-notice"><p>' )
|
||
|
.replace( '<?php echo esc_js( html_entity_decode( self::NOTICE_END_MARKER ) ); ?>', '</p></div><p style="display: none;">' );
|
||
|
|
||
|
$( element ).html( contents );
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
} );
|
||
|
<?php } ?>
|
||
|
} );
|
||
|
</script>
|
||
|
<style>
|
||
|
.update-message.rank-math-beta-update-notice {
|
||
|
font-weight: bold;
|
||
|
margin-top: 20px;
|
||
|
}
|
||
|
.update-message.rank-math-beta-update-notice > p:before {
|
||
|
content: "\f534";
|
||
|
}
|
||
|
</style>
|
||
|
<?php
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* If user requested check with force-check parameter.
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function is_check_requested() {
|
||
|
return (bool) Param::get( 'force-check' );
|
||
|
}
|
||
|
}
|