oont-contents/plugins/jetpack/_inc/lib/class.media-summary.php
2025-02-08 15:10:23 +01:00

464 lines
15 KiB
PHP

<?php // phpcs:ignore WordPress.Files.FileName.InvalidClassFileName
/**
* Provides media summary of a post.
*
* @package automattic/jetpack
*/
use Automattic\Jetpack\Image_CDN\Image_CDN_Core;
/**
* Class Jetpack_Media_Summary
*
* Priority: embed [video] > gallery > image > text
*/
class Jetpack_Media_Summary {
/**
* Media cache.
*
* @var array
*/
private static $cache = array();
/**
* Get media summary for a post.
*
* @param int $post_id Post ID.
* @param int $blog_id Blog ID, if applicable.
* @param array $args {
* Optional. An array of arguments.
* @type int $max_words Maximum number of words.
* @type int $max_chars Maximum number of characters.
* }
*
* @return array|mixed|void
*/
public static function get( $post_id, $blog_id = 0, $args = array() ) {
// @todo: Use type hinting in the line above when at PHP 7.0+.
$post_id = (int) $post_id;
$blog_id = (int) $blog_id;
$defaults = array(
'max_words' => 16,
'max_chars' => 256,
);
$args = wp_parse_args( $args, $defaults );
$switched = false;
if ( ! empty( $blog_id ) && get_current_blog_id() !== $blog_id && function_exists( 'switch_to_blog' ) ) {
switch_to_blog( $blog_id );
$switched = true;
} else {
$blog_id = get_current_blog_id();
}
$cache_key = "{$blog_id}_{$post_id}_{$args['max_words']}_{$args['max_chars']}";
if ( isset( self::$cache[ $cache_key ] ) ) {
if ( $switched ) {
restore_current_blog();
}
return self::$cache[ $cache_key ];
}
if ( ! class_exists( 'Jetpack_Media_Meta_Extractor' ) ) {
require_once JETPACK__PLUGIN_DIR . '_inc/lib/class.media-extractor.php';
}
$post = get_post( $post_id );
$permalink = get_permalink( $post_id );
$return = array(
'type' => 'standard',
'permalink' => $permalink,
'image' => '',
'excerpt' => '',
'word_count' => 0,
'secure' => array(
'image' => '',
),
'count' => array(
'image' => 0,
'video' => 0,
'word' => 0,
'link' => 0,
),
);
if ( $post instanceof WP_Post && empty( $post->post_password ) ) {
$return['excerpt'] = self::get_excerpt( $post->post_content, $post->post_excerpt, $args['max_words'], $args['max_chars'], $post );
$return['count']['word'] = self::get_word_count( $post->post_content );
$return['count']['word_remaining'] = self::get_word_remaining_count( $post->post_content, $return['excerpt'] );
$return['count']['link'] = self::get_link_count( $post->post_content );
}
$extract = Jetpack_Media_Meta_Extractor::extract( $blog_id, $post_id, Jetpack_Media_Meta_Extractor::ALL );
if ( empty( $extract['has'] ) ) {
if ( $switched ) {
restore_current_blog();
}
self::$cache[ $cache_key ] = $return;
return $return;
}
// Prioritize [some] video embeds.
if ( ! empty( $extract['has']['shortcode'] ) ) {
foreach ( $extract['shortcode'] as $type => $data ) {
switch ( $type ) {
case 'videopress':
case 'wpvideo':
if ( 0 === $return['count']['video'] ) {
// If there is no id on the video, then let's just skip this.
if ( ! isset( $data['id'][0] ) ) {
break;
}
$guid = $data['id'][0];
$video_info = videopress_get_video_details( $guid );
// Only add the video tags if the guid returns a valid videopress object.
if ( $video_info instanceof stdClass ) {
// Continue early if we can't find a Video slug.
if ( empty( $video_info->files->std->mp4 ) ) {
break;
}
$url = sprintf(
'https://videos.files.wordpress.com/%1$s/%2$s',
$guid,
$video_info->files->std->mp4
);
$thumbnail = $video_info->poster;
if ( ! empty( $thumbnail ) ) {
$return['image'] = $thumbnail;
$return['secure']['image'] = $thumbnail;
}
$return['type'] = 'video';
$return['video'] = esc_url_raw( $url );
$return['video_type'] = 'video/mp4';
$return['secure']['video'] = $return['video'];
}
}
++$return['count']['video'];
break;
case 'youtube':
if ( 0 === $return['count']['video'] ) {
if ( ! isset( $extract['shortcode']['youtube']['id'][0] ) ) {
break;
}
$return['type'] = 'video';
$return['video'] = esc_url_raw( 'http://www.youtube.com/watch?feature=player_embedded&v=' . $extract['shortcode']['youtube']['id'][0] );
$return['image'] = self::get_video_poster( 'youtube', $extract['shortcode']['youtube']['id'][0] );
$return['secure']['video'] = self::https( $return['video'] );
$return['secure']['image'] = self::https( $return['image'] );
}
++$return['count']['video'];
break;
case 'vimeo':
if ( 0 === $return['count']['video'] ) {
$return['type'] = 'video';
$return['video'] = esc_url_raw( 'http://vimeo.com/' . $extract['shortcode']['vimeo']['id'][0] );
$return['secure']['video'] = self::https( $return['video'] );
$poster_image = get_post_meta( $post_id, 'vimeo_poster_image', true );
if ( ! empty( $poster_image ) ) {
$return['image'] = $poster_image;
$poster_url_parts = wp_parse_url( $poster_image );
$return['secure']['image'] = 'https://secure-a.vimeocdn.com' . $poster_url_parts['path'];
}
}
++$return['count']['video'];
break;
}
}
}
if ( ! empty( $extract['has']['embed'] ) ) {
foreach ( $extract['embed']['url'] as $embed ) {
if ( preg_match( '/((youtube|vimeo|dailymotion)\.com|youtu.be)/', $embed ) ) {
if ( 0 === $return['count']['video'] ) {
$return['type'] = 'video';
$return['video'] = 'http://' . $embed;
$return['secure']['video'] = self::https( $return['video'] );
if ( str_contains( $embed, 'youtube' ) ) {
$return['image'] = self::get_video_poster( 'youtube', jetpack_get_youtube_id( $return['video'] ) );
$return['secure']['image'] = self::https( $return['image'] );
} elseif ( str_contains( $embed, 'youtu.be' ) ) {
$youtube_id = jetpack_get_youtube_id( $return['video'] );
$return['video'] = 'http://youtube.com/watch?v=' . $youtube_id . '&feature=youtu.be';
$return['secure']['video'] = self::https( $return['video'] );
$return['image'] = self::get_video_poster( 'youtube', jetpack_get_youtube_id( $return['video'] ) );
$return['secure']['image'] = self::https( $return['image'] );
} elseif ( str_contains( $embed, 'vimeo' ) ) {
$poster_image = get_post_meta( $post_id, 'vimeo_poster_image', true );
if ( ! empty( $poster_image ) ) {
$return['image'] = $poster_image;
$poster_url_parts = wp_parse_url( $poster_image );
$return['secure']['image'] = 'https://secure-a.vimeocdn.com' . $poster_url_parts['path'];
}
} elseif ( str_contains( $embed, 'dailymotion' ) ) {
$return['image'] = str_replace( 'dailymotion.com/video/', 'dailymotion.com/thumbnail/video/', $embed );
$return['image'] = wp_parse_url( $return['image'], PHP_URL_SCHEME ) === null ? 'http://' . $return['image'] : $return['image'];
$return['secure']['image'] = self::https( $return['image'] );
}
}
++$return['count']['video'];
}
}
}
// Do we really want to make the video the primary focus of the post?
if ( 'video' === $return['type'] ) {
$content = wpautop( wp_strip_all_tags( $post->post_content ) );
$paragraphs = explode( '</p>', $content );
$number_of_paragraphs = 0;
foreach ( $paragraphs as $i => $paragraph ) {
// Don't include blank lines as a paragraph.
if ( '' === trim( $paragraph ) ) {
unset( $paragraphs[ $i ] );
continue;
}
++$number_of_paragraphs;
}
$number_of_paragraphs = $number_of_paragraphs - $return['count']['video']; // subtract amount for videos.
// More than 2 paragraph? The video is not the primary focus so we can do some more analysis.
if ( $number_of_paragraphs > 2 ) {
$return['type'] = 'standard';
}
}
// If we don't have any prioritized embed...
if ( 'standard' === $return['type'] ) {
if ( ( ! empty( $extract['has']['gallery'] ) || ! empty( $extract['shortcode']['gallery']['count'] ) ) && ! empty( $extract['image'] ) ) {
// ... Then we prioritize galleries first (multiple images returned)
$return['type'] = 'gallery';
$return['images'] = $extract['image'];
foreach ( $return['images'] as $image ) {
$return['secure']['images'][] = array( 'url' => self::ssl_img( $image['url'] ) );
++$return['count']['image'];
}
} elseif ( ! empty( $extract['has']['image'] ) ) {
// ... Or we try and select a single image that would make sense.
$content = wpautop( wp_strip_all_tags( $post->post_content ) );
$paragraphs = explode( '</p>', $content );
$number_of_paragraphs = 0;
foreach ( $paragraphs as $i => $paragraph ) {
// Don't include 'actual' captions as a paragraph.
if ( str_contains( $paragraph, '[caption' ) ) {
unset( $paragraphs[ $i ] );
continue;
}
// Don't include blank lines as a paragraph.
if ( '' === trim( $paragraph ) ) {
unset( $paragraphs[ $i ] );
continue;
}
++$number_of_paragraphs;
}
$return['image'] = $extract['image'][0]['url'];
$return['secure']['image'] = self::ssl_img( $return['image'] );
++$return['count']['image'];
if ( $number_of_paragraphs <= 2 && is_countable( $extract['image'] ) && 1 === count( $extract['image'] ) ) {
// If we have lots of text or images, let's not treat it as an image post, but return its first image.
$return['type'] = 'image';
}
}
}
if ( $switched ) {
restore_current_blog();
}
/**
* Allow a theme or plugin to inspect and ultimately change the media summary.
*
* @since 4.4.0
*
* @param array $data The calculated media summary data.
* @param int $post_id The id of the post this data applies to.
*/
$return = apply_filters( 'jetpack_media_summary_output', $return, $post_id );
self::$cache[ $cache_key ] = $return;
return $return;
}
/**
* Converts http to https://
*
* @param string $str URL.
*
* @return string URL.
*/
public static function https( $str ) {
return str_replace( 'http://', 'https://', $str );
}
/**
* Returns a Photonized version of the URL.
*
* @param string $url URL.
*
* @return string URL.
*/
public static function ssl_img( $url ) {
if ( str_contains( $url, 'files.wordpress.com' ) ) {
return self::https( $url );
} else {
return self::https( Image_CDN_Core::cdn_url( $url ) );
}
}
/**
* Get the video poster.
*
* @param string $type Video service.
* @param string $id Video ID for the service.
*
* @return string URL of image thumbnail for the video.
*/
public static function get_video_poster( $type, $id ) {
if ( 'videopress' === $type ) {
if ( function_exists( 'video_get_highest_resolution_image_url' ) ) {
return video_get_highest_resolution_image_url( $id );
} elseif ( class_exists( 'VideoPress_Video' ) ) {
$video = new VideoPress_Video( $id );
return $video->poster_frame_uri;
}
} elseif ( 'youtube' === $type ) {
return 'http://img.youtube.com/vi/' . $id . '/0.jpg';
}
}
/**
* Clean text of shortcodes and tags.
*
* @param string $text Dirty text.
*
* @return string Clean text.
*/
public static function clean_text( $text ) {
return trim(
preg_replace(
'/[\s]+/',
' ',
preg_replace(
'@https?://[\S]+@',
'',
strip_shortcodes(
wp_strip_all_tags(
$text
)
)
)
)
);
}
/**
* Retrieve an excerpt for the post summary.
*
* This function works around a suspected problem with Core. If resolved, this function should be simplified.
*
* @link https://github.com/Automattic/jetpack/pull/8510
* @link https://core.trac.wordpress.org/ticket/42814
*
* @param string $post_content The post's content.
* @param string $post_excerpt The post's excerpt. Empty if none was explicitly set.
* @param int $max_words Maximum number of words for the excerpt. Used on wp.com. Default 16.
* @param int $max_chars Maximum characters in the excerpt. Used on wp.com. Default 256.
* @param WP_Post $requested_post The post object.
* @return string Post excerpt.
**/
public static function get_excerpt( $post_content, $post_excerpt, $max_words = 16, $max_chars = 256, $requested_post = null ) {
global $post;
$original_post = $post; // Saving the global for later use.
if ( empty( $post_excerpt ) && function_exists( 'wpcom_enhanced_excerpt_extract_excerpt' ) ) {
return self::clean_text(
wpcom_enhanced_excerpt_extract_excerpt(
array(
'text' => $post_content,
'excerpt_only' => true,
'show_read_more' => false,
'max_words' => $max_words,
'max_chars' => $max_chars,
'read_more_threshold' => 25,
)
)
);
} elseif ( $requested_post instanceof WP_Post ) {
// @todo Refactor to not need to override the global.
// phpcs:ignore: WordPress.WP.GlobalVariablesOverride.Prohibited
$post = $requested_post; // setup_postdata does not set the global.
setup_postdata( $post );
/** This filter is documented in core/src/wp-includes/post-template.php */
$post_excerpt = apply_filters( 'get_the_excerpt', $post_excerpt, $post );
// phpcs:ignore: WordPress.WP.GlobalVariablesOverride.Prohibited
$post = $original_post; // wp_reset_postdata uses the $post global.
wp_reset_postdata();
return self::clean_text( $post_excerpt );
}
return '';
}
/**
* Split a string into an array of words.
*
* @param string $text Post content or excerpt.
*
* @return array Array of words.
*/
public static function split_content_in_words( $text ) {
$words = preg_split( '/[\s!?;,.]+/', $text, -1, PREG_SPLIT_NO_EMPTY );
// Return an empty array if the split above fails.
return $words ? $words : array();
}
/**
* Get the word count.
*
* @param string $post_content Post content.
*
* @return int Word count.
*/
public static function get_word_count( $post_content ) {
return (int) count( self::split_content_in_words( self::clean_text( $post_content ) ) );
}
/**
* Get remainder word count (after the excerpt).
*
* @param string $post_content Post content.
* @param string $excerpt_content Excerpt content.
*
* @return int Number of words after the excerpt.
*/
public static function get_word_remaining_count( $post_content, $excerpt_content ) {
$content_word_count = count( self::split_content_in_words( self::clean_text( $post_content ) ) );
$excerpt_word_count = count( self::split_content_in_words( self::clean_text( $excerpt_content ) ) );
return (int) $content_word_count - $excerpt_word_count;
}
/**
* Counts the number of links in a post.
*
* @param string $post_content Post content.
*
* @return false|int Number of links.
*/
public static function get_link_count( $post_content ) {
return preg_match_all( '/\<a[\> ]/', $post_content, $matches );
}
}