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.
323 lines
7.8 KiB
PHTML
323 lines
7.8 KiB
PHTML
7 months ago
|
<?php
|
||
|
/**
|
||
|
* The class handles adding of attributes to links to content.
|
||
|
*
|
||
|
* @since 1.0.43.2
|
||
|
* @package RankMath
|
||
|
* @subpackage RankMath\Frontend
|
||
|
* @author Rank Math <support@rankmath.com>
|
||
|
*/
|
||
|
|
||
|
namespace RankMath\Frontend;
|
||
|
|
||
|
use RankMath\Helper;
|
||
|
use RankMath\Traits\Hooker;
|
||
|
use RankMath\Helpers\Str;
|
||
|
use RankMath\Helpers\Url;
|
||
|
use RankMath\Helpers\HTML;
|
||
|
|
||
|
defined( 'ABSPATH' ) || exit;
|
||
|
|
||
|
/**
|
||
|
* Add Link_Attributes class.
|
||
|
*/
|
||
|
class Link_Attributes {
|
||
|
|
||
|
use Hooker;
|
||
|
|
||
|
/**
|
||
|
* Add rel=noopener or not.
|
||
|
*
|
||
|
* @var bool
|
||
|
*/
|
||
|
public $add_noopener;
|
||
|
|
||
|
/**
|
||
|
* Add rel=nofollow to links or not.
|
||
|
*
|
||
|
* @var bool
|
||
|
*/
|
||
|
public $nofollow_link;
|
||
|
|
||
|
/**
|
||
|
* Add rel=nofollow to images or not.
|
||
|
*
|
||
|
* @var bool
|
||
|
*/
|
||
|
public $nofollow_image;
|
||
|
|
||
|
/**
|
||
|
* Open links in a new window or not.
|
||
|
*
|
||
|
* @var bool
|
||
|
*/
|
||
|
public $new_window_link;
|
||
|
|
||
|
/**
|
||
|
* Remove existing CSS class from links or not.
|
||
|
*
|
||
|
* @var bool
|
||
|
*/
|
||
|
public $remove_class;
|
||
|
|
||
|
/**
|
||
|
* Check if the link attributes have been modified or not.
|
||
|
*
|
||
|
* @var bool
|
||
|
*/
|
||
|
public $is_dirty;
|
||
|
|
||
|
/**
|
||
|
* Additional attributes to add to links.
|
||
|
*
|
||
|
* @var array
|
||
|
*/
|
||
|
public $add_attributes;
|
||
|
|
||
|
|
||
|
/**
|
||
|
* The Constructor.
|
||
|
*/
|
||
|
public function __construct() {
|
||
|
$this->action( 'wp', 'add_attributes', 9999 );
|
||
|
$this->action( 'rest_api_init', 'add_attributes' );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Add nofollow, target, title and alt attributes to link and images.
|
||
|
*/
|
||
|
public function add_attributes() {
|
||
|
|
||
|
// Add rel="nofollow" & target="_blank" for external links.
|
||
|
$this->add_noopener = $this->do_filter( 'noopener', true );
|
||
|
$this->nofollow_link = Helper::get_settings( 'general.nofollow_external_links' );
|
||
|
$this->nofollow_image = Helper::get_settings( 'general.nofollow_image_links' );
|
||
|
$this->new_window_link = Helper::get_settings( 'general.new_window_external_links' );
|
||
|
$this->remove_class = $this->do_filter( 'link/remove_class', false );
|
||
|
$this->is_dirty = false;
|
||
|
|
||
|
// Filter to run the link attributes function even when Link options are disabled.
|
||
|
$this->add_attributes = $this->do_filter( 'link/add_attributes', $this->nofollow_link || $this->new_window_link || $this->nofollow_image || $this->add_noopener );
|
||
|
|
||
|
if ( $this->add_attributes || $this->remove_class ) {
|
||
|
$this->filter( 'the_content', 'add_link_attributes', 11 );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Add nofollow and target attributes to link.
|
||
|
*
|
||
|
* @param string $content Post content.
|
||
|
* @return string
|
||
|
*/
|
||
|
public function add_link_attributes( $content ) {
|
||
|
preg_match_all( '/<(a\s[^>]+)>/', $content, $matches );
|
||
|
if ( empty( $matches ) || empty( $matches[0] ) ) {
|
||
|
return $content;
|
||
|
}
|
||
|
|
||
|
foreach ( $matches[0] as $link ) {
|
||
|
$attrs = HTML::extract_attributes( $link );
|
||
|
|
||
|
if ( ! $this->can_add_attributes( $attrs ) ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$attrs = $this->remove_link_class( $attrs );
|
||
|
$attrs = $this->set_external_attrs( $attrs );
|
||
|
|
||
|
if ( $this->is_dirty ) {
|
||
|
$new = '<a' . HTML::attributes_to_string( $attrs ) . '>';
|
||
|
$content = str_replace( $link, $new, $content );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $content;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set rel attribute.
|
||
|
*
|
||
|
* @param array $attrs Array which hold rel attribute.
|
||
|
* @param string $property Property to add.
|
||
|
* @param boolean $append Append or not.
|
||
|
*/
|
||
|
private function set_rel_attribute( &$attrs, $property, $append ) {
|
||
|
if ( empty( $attrs['rel'] ) ) {
|
||
|
$attrs['rel'] = $property;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if ( $append ) {
|
||
|
$attrs['rel'] .= ' ' . $property;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Check if we can do anything
|
||
|
*
|
||
|
* @param array $attrs Array of link attributes.
|
||
|
*
|
||
|
* @return boolean
|
||
|
*/
|
||
|
private function can_add_attributes( $attrs ) {
|
||
|
// If link has no href attribute or if the link is not valid then we don't need to do anything.
|
||
|
if (
|
||
|
empty( $attrs['href'] ) ||
|
||
|
( empty( wp_parse_url( $attrs['href'], PHP_URL_HOST ) ) && ! Url::is_affiliate( $attrs['href'] ) ) ||
|
||
|
( isset( $attrs['role'] ) && 'button' === $attrs['role'] )
|
||
|
) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Remove rank-math-link class.
|
||
|
*
|
||
|
* @since 1.0.44.2
|
||
|
*
|
||
|
* @param array $attrs Array of link attributes.
|
||
|
*
|
||
|
* @return array $attrs
|
||
|
*/
|
||
|
private function remove_link_class( $attrs ) {
|
||
|
if ( ! $this->remove_class || empty( $attrs['class'] ) || strpos( $attrs['class'], 'rank-math-link' ) === false ) {
|
||
|
return $attrs;
|
||
|
}
|
||
|
|
||
|
$this->is_dirty = true;
|
||
|
$attrs['class'] = str_replace( 'rank-math-link', '', $attrs['class'] );
|
||
|
|
||
|
if ( ! trim( $attrs['class'] ) ) {
|
||
|
unset( $attrs['class'] );
|
||
|
}
|
||
|
|
||
|
return $attrs;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set External attributs
|
||
|
*
|
||
|
* @since 1.0.44.2
|
||
|
*
|
||
|
* @param array $attrs Array of link attributes.
|
||
|
*
|
||
|
* @return array $attrs
|
||
|
*/
|
||
|
private function set_external_attrs( $attrs ) {
|
||
|
if ( ! $this->add_attributes ) {
|
||
|
return $attrs;
|
||
|
}
|
||
|
|
||
|
// Skip if there is no href or it's a hash link like "#id".
|
||
|
// Skip if relative link.
|
||
|
// Skip for same domain ignoring sub-domain if any.
|
||
|
if ( ! Url::is_external( $attrs['href'] ) ) {
|
||
|
return $attrs;
|
||
|
}
|
||
|
|
||
|
if ( $this->do_filter( 'nofollow/url', $this->should_add_nofollow( $attrs['href'] ), $attrs['href'] ) ) {
|
||
|
if ( $this->nofollow_link || ( $this->nofollow_image && $this->is_valid_image( $attrs['href'] ) ) ) {
|
||
|
$this->is_dirty = true;
|
||
|
$this->set_rel_attribute( $attrs, 'nofollow', ( isset( $attrs['rel'] ) && ! Str::contains( 'dofollow', $attrs['rel'] ) && ! Str::contains( 'nofollow', $attrs['rel'] ) ) );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $this->new_window_link && ! isset( $attrs['target'] ) ) {
|
||
|
$this->is_dirty = true;
|
||
|
$attrs['target'] = '_blank';
|
||
|
}
|
||
|
|
||
|
if ( $this->add_noopener && $this->do_filter( 'noopener/domain', Url::get_domain( $attrs['href'] ) ) ) {
|
||
|
$this->is_dirty = true;
|
||
|
$this->set_rel_attribute( $attrs, 'noopener', ( isset( $attrs['rel'] ) && ! Str::contains( 'noopener', $attrs['rel'] ) ) );
|
||
|
}
|
||
|
|
||
|
if ( Url::is_affiliate( $attrs['href'] ) ) {
|
||
|
$this->is_dirty = true;
|
||
|
$this->set_rel_attribute( $attrs, 'sponsored', ( isset( $attrs['rel'] ) && ! Str::contains( 'sponsored', $attrs['rel'] ) ) );
|
||
|
}
|
||
|
|
||
|
return $attrs;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Check if we need to add nofollow for this link, based on "nofollow_domains" & "nofollow_exclude_domains"
|
||
|
*
|
||
|
* @param string $url Link URL.
|
||
|
* @return bool
|
||
|
*/
|
||
|
private function should_add_nofollow( $url ) {
|
||
|
if ( ! $this->nofollow_link && ! $this->nofollow_image ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
$include_domains = $this->get_nofollow_domains( 'include' );
|
||
|
$exclude_domains = $this->get_nofollow_domains( 'exclude' );
|
||
|
$parent_domain = Url::get_domain( $url );
|
||
|
$parent_domain = preg_replace( '/^www\./', '', $parent_domain );
|
||
|
|
||
|
// Check if domain is in list.
|
||
|
if ( ! empty( $include_domains ) ) {
|
||
|
return in_array( $parent_domain, $include_domains, true );
|
||
|
}
|
||
|
|
||
|
// Check if domain is NOT in list.
|
||
|
if ( ! empty( $exclude_domains ) && in_array( $parent_domain, $exclude_domains, true ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get domain for nofollow
|
||
|
*
|
||
|
* @param string $type Type either include or exclude.
|
||
|
* @return array
|
||
|
*/
|
||
|
private function get_nofollow_domains( $type ) {
|
||
|
static $rank_math_nofollow_domains;
|
||
|
|
||
|
if ( isset( $rank_math_nofollow_domains[ $type ] ) ) {
|
||
|
return $rank_math_nofollow_domains[ $type ];
|
||
|
}
|
||
|
|
||
|
$setting = 'include' === $type ? 'nofollow_domains' : 'nofollow_exclude_domains';
|
||
|
$domains = Helper::get_settings( "general.{$setting}" );
|
||
|
$domains = Str::to_arr_no_empty( $domains );
|
||
|
|
||
|
// Strip off www. prefixes.
|
||
|
$domains = array_map(
|
||
|
function( $domain ) {
|
||
|
$domain = preg_replace( '#^http(s)?://#', '', trim( $domain, '/' ) );
|
||
|
return preg_replace( '/^www\./', '', $domain );
|
||
|
},
|
||
|
$domains
|
||
|
);
|
||
|
|
||
|
$rank_math_nofollow_domains[ $type ] = $domains;
|
||
|
|
||
|
return $rank_math_nofollow_domains[ $type ];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Is a valid image url.
|
||
|
*
|
||
|
* @param string $url Image url.
|
||
|
*
|
||
|
* @return boolean
|
||
|
*/
|
||
|
private function is_valid_image( $url ) {
|
||
|
foreach ( [ '.jpg', '.jpeg', '.png', '.gif' ] as $ext ) {
|
||
|
if ( Str::contains( $ext, $url ) ) {
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
}
|