getty.php 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. <?php
  2. /**
  3. * Getty shortcode
  4. *
  5. * [getty src="82278805" width="$width" height="$height"]
  6. * <div class="getty embed image" style="background-color:#fff;display:inline-block;font-family:'Helvetica Neue',Helvetica,Arial,sans-serif;color:#a7a7a7;font-size:11px;width:100%;max-width:462px;"><div style="padding:0;margin:0;text-align:left;"><a href="http://www.gettyimages.com/detail/82278805" target="_blank" style="color:#a7a7a7;text-decoration:none;font-weight:normal !important;border:none;display:inline-block;">Embed from Getty Images</a></div><div style="overflow:hidden;position:relative;height:0;padding:80.086580% 0 0 0;width:100%;"><iframe src="//embed.gettyimages.com/embed/82278805?et=jGiu6FXXSpJDGf1SnwLV2g&sig=TFVNFtqghwNw5iJQ1MFWnI8f4Y40_sfogfZLhai6SfA=" width="462" height="370" scrolling="no" frameborder="0" style="display:inline-block;position:absolute;top:0;left:0;width:100%;height:100%;"></iframe></div><p style="margin:0;"></p></div>
  7. */
  8. if ( defined( 'IS_WPCOM' ) && IS_WPCOM ) {
  9. add_action( 'init', 'jetpack_getty_enable_embeds' );
  10. } else {
  11. jetpack_getty_enable_embeds();
  12. }
  13. /**
  14. * Register Getty as oembed provider. Add filter to reverse iframes to shortcode. Register [getty] shortcode.
  15. *
  16. * @since 4.5.0
  17. * @since 5.8.0 removed string parameter.
  18. */
  19. function jetpack_getty_enable_embeds() {
  20. // Support their oEmbed Endpoint
  21. wp_oembed_add_provider( '#https?://www\.gettyimages\.com/detail/.*#i', "https://embed.gettyimages.com/oembed/", true );
  22. wp_oembed_add_provider( '#https?://(www\.)?gty\.im/.*#i', "https://embed.gettyimages.com/oembed/", true );
  23. // Allow iframes to be filtered to short code (so direct copy+paste can be done)
  24. add_filter( 'pre_kses', 'wpcom_shortcodereverse_getty' );
  25. // Actually display the Getty Embed
  26. add_shortcode( 'getty', 'jetpack_getty_shortcode' );
  27. }
  28. /**
  29. * Filters the oEmbed provider URL for Getty URLs to include site URL host as
  30. * caller if available, falling back to "wordpress.com". Must be applied at
  31. * time of embed in case that `init` is too early (WP.com REST API).
  32. *
  33. * @module shortcodes
  34. *
  35. * @since 5.8.0
  36. *
  37. * @see WP_oEmbed::fetch
  38. *
  39. * @return string oEmbed provider URL
  40. */
  41. add_filter( 'oembed_fetch_url', 'getty_add_oembed_endpoint_caller' );
  42. function getty_add_oembed_endpoint_caller( $provider ) {
  43. // By time filter is called, original provider URL has had url, maxwidth,
  44. // maxheight query parameters added.
  45. if ( 0 !== strpos( $provider, 'https://embed.gettyimages.com/oembed/' ) ) {
  46. return $provider;
  47. }
  48. // Set the caller argument to pass to Getty's oembed provider.
  49. if ( defined( 'IS_WPCOM' ) && IS_WPCOM ) {
  50. // Only include caller for non-private sites
  51. if ( ! function_exists( 'is_private_blog' ) || ! is_private_blog() ) {
  52. $host = parse_url( get_bloginfo( 'url' ), PHP_URL_HOST );
  53. }
  54. // Fall back to WordPress.com
  55. if ( empty( $host ) ) {
  56. $host = 'wordpress.com';
  57. }
  58. } else {
  59. $host = parse_url( get_home_url(), PHP_URL_HOST );
  60. }
  61. return add_query_arg( 'caller', $host, $provider );
  62. }
  63. /**
  64. * Compose shortcode based on Getty iframes.
  65. *
  66. * @since 4.5.0
  67. *
  68. * @param string $content
  69. *
  70. * @return mixed
  71. */
  72. function wpcom_shortcodereverse_getty( $content ) {
  73. if ( ! is_string( $content ) || false === stripos( $content, '.gettyimages.com/' ) ) {
  74. return $content;
  75. }
  76. $regexp = '!<iframe\s+src=[\'"](https?:)?//embed\.gettyimages\.com/embed(/|/?\?assets=)([a-z0-9_-]+(,[a-z0-9_-]+)*)[^\'"]*?[\'"]((?:\s+\w+=[\'"][^\'"]*[\'"])*)((?:[\s\w]*))></iframe>!i';
  77. $regexp_ent = str_replace( '&amp;#0*58;', '&amp;#0*58;|&#0*58;', htmlspecialchars( $regexp, ENT_NOQUOTES ) );
  78. // Markup pattern for 2017 embed syntax with significant differences from
  79. // the prior pattern:
  80. $regexp_2017 = '!<a.+?class=\'gie-(single|slideshow)\'.+?gie\.widgets\.load\({([^}]+)}\).+?embed-cdn\.gettyimages\.com/widgets\.js.+?</script>!';
  81. $regexp_2017_ent = str_replace( '&amp;#0*58;', '&amp;#0*58;|&#0*58;', htmlspecialchars( $regexp_2017, ENT_NOQUOTES ) );
  82. foreach ( array( 'regexp_2017', 'regexp_2017_ent', 'regexp', 'regexp_ent' ) as $reg ) {
  83. if ( ! preg_match_all( $$reg, $content, $matches, PREG_SET_ORDER ) ) {
  84. continue;
  85. }
  86. foreach ( $matches as $match ) {
  87. if ( 'regexp_2017' === $reg || 'regexp_2017_ent' === $reg ) {
  88. // Extract individual keys from the matched JavaScript object
  89. $params = $match[2];
  90. if ( ! preg_match_all( '!(?P<key>\w+)\s*:\s*([\'"](?P<value>[^\'"]*?)(px)?[\'"])!', $params, $key_matches, PREG_SET_ORDER ) ) {
  91. continue;
  92. }
  93. foreach ( $key_matches as $key_match ) {
  94. switch ( $key_match['key'] ) {
  95. case 'items': $ids = $key_match['value']; break;
  96. case 'w': $width = (int) $key_match['value']; break;
  97. case 'h': $height = (int) $key_match['value']; break;
  98. case 'tld': $tld = $key_match['value']; break;
  99. }
  100. }
  101. } else {
  102. $params = $match[5];
  103. if ( 'regexp_ent' === $reg ) {
  104. $params = html_entity_decode( $params );
  105. }
  106. $params = wp_kses_hair( $params, array( 'http' ) );
  107. $ids = esc_html( $match[3] );
  108. $width = isset( $params['width'] ) ? (int) $params['width']['value'] : 0;
  109. $height = isset( $params['height'] ) ? (int) $params['height']['value'] : 0;
  110. }
  111. if ( empty( $ids ) ) {
  112. continue;
  113. }
  114. $shortcode = '[getty src="' . esc_attr( $ids ) . '"';
  115. if ( ! empty( $width ) ) {
  116. $shortcode .= ' width="' . esc_attr( $width ) . '"';
  117. }
  118. if ( ! empty( $height ) ) {
  119. $shortcode .= ' height="' . esc_attr( $height ) . '"';
  120. }
  121. // While it does not appear to have any practical impact, Getty has
  122. // requested that we include TLD in the embed request
  123. if ( ! empty( $tld ) ) {
  124. $shortcode .= ' tld="' . esc_attr( $tld ). '"';
  125. }
  126. $shortcode .= ']';
  127. $content = str_replace( $match[0], $shortcode, $content );
  128. }
  129. }
  130. // strip out enclosing div and any other markup
  131. $regexp = '%<div class="getty\s[^>]*+>.*?<div[^>]*+>(\[getty[^\]]*+\])\s*</div>.*?</div>%is';
  132. $regexp_ent = str_replace( array( '&amp;#0*58;', '[^&gt;]' ), array( '&amp;#0*58;|&#0*58;', '[^&]' ), htmlspecialchars( $regexp, ENT_NOQUOTES ) );
  133. foreach ( array( 'regexp', 'regexp_ent' ) as $reg ) {
  134. if ( ! preg_match_all( $$reg, $content, $matches, PREG_SET_ORDER ) ) {
  135. continue;
  136. }
  137. foreach ( $matches as $match ) {
  138. $content = str_replace( $match[0], $match[1], $content );
  139. }
  140. }
  141. /** This action is documented in modules/widgets/social-media-icons.php */
  142. do_action( 'jetpack_bump_stats_extras', 'html_to_shortcode', 'getty' );
  143. return $content;
  144. }
  145. /**
  146. * Parse shortcode arguments and render its output.
  147. *
  148. * @since 4.5.0
  149. *
  150. * @param array $atts Shortcode parameters.
  151. * @param string $content Content enclosed by shortcode tags.
  152. *
  153. * @return string
  154. */
  155. function jetpack_getty_shortcode( $atts, $content = '' ) {
  156. if ( ! empty( $content ) ) {
  157. $src = $content;
  158. } elseif ( ! empty( $atts['src'] ) ) {
  159. $src = $atts['src'];
  160. } elseif ( ! empty( $atts[0] ) ) {
  161. $src = $atts[0];
  162. } else {
  163. return '<!-- Missing Getty Source ID -->';
  164. }
  165. $src = preg_replace( '/^([\da-z-]+(,[\da-z-]+)*).*$/', '$1', $src );
  166. $params = array(
  167. 'width' => isset( $atts['width'] ) ? (int) $atts['width'] : null,
  168. 'height' => isset( $atts['height'] ) ? (int) $atts['height'] : null
  169. );
  170. if ( ! empty( $atts['tld'] ) ) {
  171. $params['tld'] = $atts['tld'];
  172. }
  173. return wp_oembed_get( 'https://gty.im/' . $src, array_filter( $params ) );
  174. }