Completed
Push — pr/7794 ( fe7444...511b2e )
by George
07:28
created

Jetpack_PostImages::from_html()   C

Complexity

Conditions 8
Paths 7

Size

Total Lines 36
Code Lines 22

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 8
eloc 22
c 0
b 0
f 0
nc 7
nop 1
dl 0
loc 36
rs 5.3846
1
<?php
2
3
/**
4
 * Useful for finding an image to display alongside/in representation of a specific post.
5
 *
6
 * Includes a few different methods, all of which return a similar-format array containing
7
 * details of any images found. Everything can (should) be called statically, it's just a
8
 * function-bucket. You can also call Jetpack_PostImages::get_image() to cycle through all of the methods until
9
 * one of them finds something useful.
10
 *
11
 * This file is included verbatim in Jetpack
12
 */
13
class Jetpack_PostImages {
14
	/**
15
	 * If a slideshow is embedded within a post, then parse out the images involved and return them
16
	 */
17
	static function from_slideshow( $post_id, $width = 200, $height = 200 ) {
18
		$images = array();
19
20
		$post = get_post( $post_id );
21
22
		if ( ! $post ) {
23
			return $images;
24
		}
25
26
		if ( ! empty( $post->post_password ) ) {
27
			return $images;
28
		}
29
30
		if ( false === has_shortcode( $post->post_content, 'slideshow' ) ) {
31
			return $images; // no slideshow - bail
32
		}
33
34
		$permalink = get_permalink( $post->ID );
35
36
		// Mechanic: Somebody set us up the bomb
37
		$old_post = $GLOBALS['post'];
38
		$GLOBALS['post'] = $post;
39
		$old_shortcodes = $GLOBALS['shortcode_tags'];
40
		$GLOBALS['shortcode_tags'] = array( 'slideshow' => $old_shortcodes['slideshow'] );
41
42
		// Find all the slideshows
43
		preg_match_all( '/' . get_shortcode_regex() . '/sx', $post->post_content, $slideshow_matches, PREG_SET_ORDER );
44
45
		ob_start(); // The slideshow shortcode handler calls wp_print_scripts and wp_print_styles... not too happy about that
46
47
		foreach ( $slideshow_matches as $slideshow_match ) {
0 ignored issues
show
Bug introduced by
The expression $slideshow_matches of type null|array<integer,array<integer,string>> is not guaranteed to be traversable. How about adding an additional type check?

There are different options of fixing this problem.

  1. If you want to be on the safe side, you can add an additional type-check:

    $collection = json_decode($data, true);
    if ( ! is_array($collection)) {
        throw new \RuntimeException('$collection must be an array.');
    }
    
    foreach ($collection as $item) { /** ... */ }
    
  2. If you are sure that the expression is traversable, you might want to add a doc comment cast to improve IDE auto-completion and static analysis:

    /** @var array $collection */
    $collection = json_decode($data, true);
    
    foreach ($collection as $item) { /** .. */ }
    
  3. Mark the issue as a false-positive: Just hover the remove button, in the top-right corner of this issue for more options.

Loading history...
48
			$slideshow = do_shortcode_tag( $slideshow_match );
49
			if ( false === $pos = stripos( $slideshow, 'jetpack-slideshow' ) ) // must be something wrong - or we changed the output format in which case none of the following will work
50
				continue;
51
			$start = strpos( $slideshow, '[', $pos );
52
			$end = strpos( $slideshow, ']', $start );
53
			$post_images = json_decode( wp_specialchars_decode( str_replace( "'", '"', substr( $slideshow, $start, $end - $start + 1 ) ), ENT_QUOTES ) ); // parse via JSON
54
			foreach ( $post_images as $post_image ) {
55
				if ( !$post_image_id = absint( $post_image->id ) )
56
					continue;
57
58
				$meta = wp_get_attachment_metadata( $post_image_id );
59
60
				// Must be larger than 200x200 (or user-specified)
61
				if ( !isset( $meta['width'] ) || $meta['width'] < $width )
62
					continue;
63
				if ( !isset( $meta['height'] ) || $meta['height'] < $height )
64
					continue;
65
66
				$url = wp_get_attachment_url( $post_image_id );
67
68
				$images[] = array(
69
					'type'       => 'image',
70
					'from'       => 'slideshow',
71
					'src'        => $url,
72
					'src_width'  => $meta['width'],
73
					'src_height' => $meta['height'],
74
					'href'       => $permalink,
75
				);
76
			}
77
		}
78
		ob_end_clean();
79
80
		// Operator: Main screen turn on
81
		$GLOBALS['shortcode_tags'] = $old_shortcodes;
82
		$GLOBALS['post'] = $old_post;
83
84
		return $images;
85
	}
86
87
	/**
88
	 * If a gallery is detected, then get all the images from it.
89
	 */
90
	static function from_gallery( $post_id ) {
91
		$images = array();
92
93
		$post = get_post( $post_id );
94
95
		if ( ! $post ) {
96
			return $images;
97
		}
98
99
		if ( ! empty( $post->post_password ) ) {
100
			return $images;
101
		}
102
103
		$permalink = get_permalink( $post->ID );
104
105
		/**
106
		 *  Juggle global post object because the gallery shortcode uses the
107
		 *  global object.
108
		 *
109
		 *  See core ticket:
110
		 *  https://core.trac.wordpress.org/ticket/39304
111
		 */
112
		if ( isset( $GLOBALS['post'] ) ) {
113
			$juggle_post = $GLOBALS['post'];
114
			$GLOBALS['post'] = $post;
115
			$galleries = get_post_galleries( $post->ID, false );
116
			$GLOBALS['post'] = $juggle_post;
117
		} else {
118
			$GLOBALS['post'] = $post;
119
			$galleries = get_post_galleries( $post->ID, false );
120
			unset( $GLOBALS['post'] );
121
		}
122
123
		foreach ( $galleries as $gallery ) {
124
			if ( isset( $gallery['type'] ) && 'slideshow' === $gallery['type'] && ! empty( $gallery['ids'] ) ) {
125
				$image_ids = explode( ',', $gallery['ids'] );
126
				$image_size = isset( $gallery['size'] ) ? $gallery['size'] : 'thumbnail';
127
				foreach ( $image_ids as $image_id ) {
128
					$image = wp_get_attachment_image_src( $image_id, $image_size );
129 View Code Duplication
					if ( ! empty( $image[0] ) ) {
130
						list( $raw_src ) = explode( '?', $image[0] ); // pull off any Query string (?w=250)
131
						$raw_src = wp_specialchars_decode( $raw_src ); // rawify it
132
						$raw_src = esc_url_raw( $raw_src ); // clean it
133
						$images[] = array(
134
							'type'  => 'image',
135
							'from'  => 'gallery',
136
							'src'   => $raw_src,
137
							'href'  => $permalink,
138
						);
139
					}
140
				}
141 View Code Duplication
			} elseif ( ! empty( $gallery['src'] ) ) {
142
				foreach ( $gallery['src'] as $src ) {
143
					list( $raw_src ) = explode( '?', $src ); // pull off any Query string (?w=250)
144
					$raw_src = wp_specialchars_decode( $raw_src ); // rawify it
145
					$raw_src = esc_url_raw( $raw_src ); // clean it
146
					$images[] = array(
147
						'type'  => 'image',
148
						'from'  => 'gallery',
149
						'src'   => $raw_src,
150
						'href'  => $permalink,
151
					);
152
				}
153
			}
154
		}
155
156
		return $images;
157
	}
158
159
	/**
160
	 * Get attachment images for a specified post and return them. Also make sure
161
	 * their dimensions are at or above a required minimum.
162
	 */
163
	static function from_attachment( $post_id, $width = 200, $height = 200 ) {
164
		$images = array();
165
166
		$post = get_post( $post_id );
167
168
		if ( ! empty( $post->post_password ) ) {
169
			return $images;
170
		}
171
172
		$post_images = get_posts( array(
173
			'post_parent' => $post_id,   // Must be children of post
174
			'numberposts' => 5,          // No more than 5
175
			'post_type' => 'attachment', // Must be attachments
176
			'post_mime_type' => 'image', // Must be images
177
		) );
178
179
		if ( ! $post_images ) {
180
			return $images;
181
		}
182
183
		$permalink = get_permalink( $post_id );
184
185
		foreach ( $post_images as $post_image ) {
186
			$meta = wp_get_attachment_metadata( $post_image->ID );
187
			// Must be larger than 200x200
188
			if ( !isset( $meta['width'] ) || $meta['width'] < $width )
189
				continue;
190
			if ( !isset( $meta['height'] ) || $meta['height'] < $height )
191
				continue;
192
193
			$url = wp_get_attachment_url( $post_image->ID );
194
195
			$images[] = array(
196
				'type'       => 'image',
197
				'from'       => 'attachment',
198
				'src'        => $url,
199
				'src_width'  => $meta['width'],
200
				'src_height' => $meta['height'],
201
				'href'       => $permalink,
202
			);
203
		}
204
205
		/*
206
		* We only want to pass back attached images that were actually inserted.
207
		* We can load up all the images found in the HTML source and then
208
		* compare URLs to see if an image is attached AND inserted.
209
		*/
210
		$html_images = self::from_html( $post_id );
211
		$inserted_images = array();
212
213
		foreach( $html_images as $html_image ) {
214
			$src = parse_url( $html_image['src'] );
215
			// strip off any query strings from src
216
			if( ! empty( $src['scheme'] ) && ! empty( $src['host'] ) ) {
217
				$inserted_images[] = $src['scheme'] . '://' . $src['host'] . $src['path'];
218
			} elseif( ! empty( $src['host'] ) ) {
219
				$inserted_images[] = set_url_scheme( 'http://' . $src['host'] . $src['path'] );
220
			} else {
221
				$inserted_images[] = site_url( '/' ) . $src['path'];
222
			}
223
		}
224
		foreach( $images as $i => $image ) {
225
			if ( !in_array( $image['src'], $inserted_images ) )
226
				unset( $images[$i] );
227
		}
228
229
		return $images;
230
	}
231
232
	/**
233
	 * Check if a Featured Image is set for this post, and return it in a similar
234
	 * format to the other images?_from_*() methods.
235
	 * @param  int $post_id The post ID to check
236
	 * @return Array containing details of the Featured Image, or empty array if none.
237
	 */
238
	static function from_thumbnail( $post_id, $width = 200, $height = 200 ) {
239
		$images = array();
240
241
		$post = get_post( $post_id );
242
243
		if ( ! empty( $post->post_password ) ) {
244
			return $images;
245
		}
246
247
		if ( ! function_exists( 'get_post_thumbnail_id' ) ) {
248
			return $images;
249
		}
250
251
		$thumb = get_post_thumbnail_id( $post_id );
252
253
		if ( $thumb ) {
254
			$meta = wp_get_attachment_metadata( $thumb );
255
			// Must be larger than requested minimums
256
			if ( !isset( $meta['width'] ) || $meta['width'] < $width )
257
				return $images;
258
			if ( !isset( $meta['height'] ) || $meta['height'] < $height )
259
				return $images;
260
261
			$too_big = ( ( ! empty( $meta['width'] ) && $meta['width'] > 1200 ) || ( ! empty( $meta['height'] ) && $meta['height'] > 1200 ) );
262
263
			if (
264
				$too_big &&
265
				(
266
					( method_exists( 'Jetpack', 'is_module_active' ) && Jetpack::is_module_active( 'photon' ) ) ||
267
					( defined( 'IS_WPCOM' ) && IS_WPCOM )
268
				)
269
			) {
270
				$img_src = wp_get_attachment_image_src( $thumb, array( 1200, 1200 ) );
271
			} else {
272
				$img_src = wp_get_attachment_image_src( $thumb, 'full' );
273
			}
274
			if ( ! is_array( $img_src ) ) {
275
				// If wp_get_attachment_image_src returns false but we know that there should be an image that could be used.
276
				// we try a bit harder and user the data that we have.
277
				$thumb_post_data = get_post( $thumb );
278
				$img_src = array( $thumb_post_data->guid, $meta['width'], $meta['height'] );
279
			}
280
281
			$url = $img_src[0];
282
			$images = array( array( // Other methods below all return an array of arrays
283
				'type'       => 'image',
284
				'from'       => 'thumbnail',
285
				'src'        => $url,
286
				'src_width'  => $img_src[1],
287
				'src_height' => $img_src[2],
288
				'href'       => get_permalink( $thumb ),
289
			) );
290
291
		}
292
293
		if ( empty( $images ) && ( defined( 'IS_WPCOM' ) && IS_WPCOM ) ) {
294
			$meta_thumbnail = get_post_meta( $post_id, '_jetpack_post_thumbnail', true );
295
			if ( ! empty( $meta_thumbnail ) ) {
296
				if ( ! isset( $meta_thumbnail['width'] ) || $meta_thumbnail['width'] < $width ) {
297
					return $images;
298
				}
299
300
				if ( ! isset( $meta_thumbnail['height'] ) || $meta_thumbnail['height'] < $height ) {
301
					return $images;
302
				}
303
304
				$images = array( array( // Other methods below all return an array of arrays
305
					'type'       => 'image',
306
					'from'       => 'thumbnail',
307
					'src'        => $meta_thumbnail['URL'],
308
					'src_width'  => $meta_thumbnail['width'],
309
					'src_height' => $meta_thumbnail['height'],
310
					'href'       => $meta_thumbnail['URL'],
311
				) );
312
			}
313
		}
314
315
		return $images;
316
	}
317
318
	/**
319
	 * Very raw -- just parse the HTML and pull out any/all img tags and return their src
320
	 * @param  mixed $html_or_id The HTML string to parse for images, or a post id
321
	 * @return Array containing images
322
	 */
323
	static function from_html( $html_or_id ) {
324
		$images = array();
325
326
		if ( is_numeric( $html_or_id ) ) {
327
			$post = get_post( $html_or_id );
328
329
			if ( empty( $post ) || ! empty( $post->post_password ) ) {
330
				return $images;
331
			}
332
333
			$html = $post->post_content; // DO NOT apply the_content filters here, it will cause loops
334
		} else {
335
			$html = $html_or_id;
336
		}
337
338
		if ( ! $html ) {
339
			return $images;
340
		}
341
342
		preg_match_all( '!<img.*src=[\'"]([^"]+)[\'"].*/?>!iUs', $html, $matches );
343
		if ( !empty( $matches[1] ) ) {
344
			foreach ( $matches[1] as $match ) {
345
				if ( stristr( $match, '/smilies/' ) )
346
					continue;
347
348
				$images[] = array(
349
					'type'  => 'image',
350
					'from'  => 'html',
351
					'src'   => html_entity_decode( $match ),
352
					'href'  => '', // No link to apply to these. Might potentially parse for that as well, but not for now
353
				);
354
			}
355
		}
356
357
		return $images;
358
	}
359
360
	/**
361
	 * @param    int $post_id The post ID to check
362
	 * @param    int $size
363
	 * @return Array containing details of the image, or empty array if none.
364
	 */
365
	static function from_blavatar( $post_id, $size = 96 ) {
366
367
		$permalink = get_permalink( $post_id );
368
369
		if ( function_exists( 'blavatar_domain' ) && function_exists( 'blavatar_exists' ) && function_exists( 'blavatar_url' ) ) {
370
			$domain = blavatar_domain( $permalink );
371
372
			if ( ! blavatar_exists( $domain ) ) {
373
				return array();
374
			}
375
376
			$url = blavatar_url( $domain, 'img', $size );
377
		} elseif ( function_exists( 'has_site_icon' ) && has_site_icon() ) {
378
			$url = get_site_icon_url( $size );
379
		} else {
380
			return array();
381
		}
382
383
		return array( array(
384
			'type'       => 'image',
385
			'from'       => 'blavatar',
386
			'src'        => $url,
387
			'src_width'  => $size,
388
			'src_height' => $size,
389
			'href'       => $permalink,
390
		) );
391
	}
392
393
	/**
394
	 * Gets a post image from the author avatar.
395
	 *
396
	 * @param int    $post_id The post ID to check.
397
	 * @param int    $size The size of the avatar to get.
398
	 * @param string $default The default image to use.
0 ignored issues
show
Documentation introduced by
Should the type for parameter $default not be false|string?

This check looks for @param annotations where the type inferred by our type inference engine differs from the declared type.

It makes a suggestion as to what type it considers more descriptive.

Most often this is a case of a parameter that can be null in addition to its declared types.

Loading history...
399
	 * @return Array containing details of the image, or empty array if none.
400
	 */
401
	static function from_gravatar( $post_id, $size = 96, $default = false ) {
402
		$post = get_post( $post_id );
403
		$permalink = get_permalink( $post_id );
404
405
		if ( function_exists( 'wpcom_get_avatar_url' ) ) {
406
			$url = wpcom_get_avatar_url( $post->post_author, $size, $default, true );
407
			if ( $url && is_array( $url ) ) {
408
				$url = $url[0];
409
			}
410
		} else {
411
			$url = get_avatar_url( $post->post_author, array(
412
				'size' => $size,
413
				'default' => $default,
414
			) );
415
		}
416
417
		return array(
418
			array(
419
				'type'       => 'image',
420
				'from'       => 'gravatar',
421
				'src'        => $url,
422
				'src_width'  => $size,
423
				'src_height' => $size,
424
				'href'       => $permalink,
425
			),
426
		);
427
	}
428
429
	/**
430
	 * Run through the different methods that we have available to try to find a single good
431
	 * display image for this post.
432
	 * @param  int $post_id
433
	 * @param array $args Other arguments (currently width and height required for images where possible to determine)
434
	 * @return Array containing details of the best image to be used
435
	 */
436
	static function get_image( $post_id, $args = array() ) {
437
		$image = '';
438
439
		/**
440
		 * Fires before we find a single good image for a specific post.
441
		 *
442
		 * @since 2.2.0
443
		 *
444
		 * @param int $post_id Post ID.
445
		 */
446
		do_action( 'jetpack_postimages_pre_get_image', $post_id );
447
		$media = self::get_images( $post_id, $args );
448
449
450
		if ( is_array( $media ) ) {
451
			foreach ( $media as $item ) {
452
				if ( 'image' == $item['type'] ) {
453
					$image = $item;
454
					break;
455
				}
456
			}
457
		}
458
459
		/**
460
		 * Fires after we find a single good image for a specific post.
461
		 *
462
		 * @since 2.2.0
463
		 *
464
		 * @param int $post_id Post ID.
465
		 */
466
		do_action( 'jetpack_postimages_post_get_image', $post_id );
467
468
		return $image;
469
	}
470
471
	/**
472
	 * Get an array containing a collection of possible images for this post, stopping once we hit a method
473
	 * that returns something useful.
474
	 * @param  int $post_id
475
	 * @param  array  $args Optional args, see defaults list for details
476
	 * @return Array containing images that would be good for representing this post
477
	 */
478
	static function get_images( $post_id, $args = array() ) {
479
		// Figure out which image to attach to this post.
480
		$media = false;
481
482
		/**
483
		 * Filters the array of images that would be good for a specific post.
484
		 * This filter is applied before options ($args) filter the original array.
485
		 *
486
		 * @since 2.0.0
487
		 *
488
		 * @param array $media Array of images that would be good for a specific post.
489
		 * @param int $post_id Post ID.
490
		 * @param array $args Array of options to get images.
491
		 */
492
		$media = apply_filters( 'jetpack_images_pre_get_images', $media, $post_id, $args );
493
		if ( $media )
494
			return $media;
495
496
		$defaults = array(
497
			'width'               => 200, // Required minimum width (if possible to determine)
498
			'height'              => 200, // Required minimum height (if possible to determine)
499
500
			'fallback_to_avatars' => false, // Optionally include Blavatar and Gravatar (in that order) in the image stack
501
			'avatar_size'         => 96, // Used for both Grav and Blav
502
			'gravatar_default'    => false, // Default image to use if we end up with no Gravatar
503
504
			'from_thumbnail'      => true, // Use these flags to specify which methods to use to find an image
505
			'from_slideshow'      => true,
506
			'from_gallery'        => true,
507
			'from_attachment'     => true,
508
			'from_html'           => true,
509
510
			'html_content'        => '' // HTML string to pass to from_html()
511
		);
512
		$args = wp_parse_args( $args, $defaults );
513
514
		$media = false;
515
		if ( $args['from_thumbnail'] )
516
			$media = self::from_thumbnail( $post_id, $args['width'], $args['height'] );
517 View Code Duplication
		if ( !$media && $args['from_slideshow'] )
518
			$media = self::from_slideshow( $post_id, $args['width'], $args['height'] );
519
		if ( !$media && $args['from_gallery'] )
520
			$media = self::from_gallery( $post_id );
521 View Code Duplication
		if ( !$media && $args['from_attachment'] )
522
			$media = self::from_attachment( $post_id, $args['width'], $args['height'] );
523
		if ( !$media && $args['from_html'] ) {
524
			if ( empty( $args['html_content'] ) )
525
				$media = self::from_html( $post_id ); // Use the post_id, which will load the content
526
			else
527
				$media = self::from_html( $args['html_content'] ); // If html_content is provided, use that
528
		}
529
530
		if ( !$media && $args['fallback_to_avatars'] ) {
531
			$media = self::from_blavatar( $post_id, $args['avatar_size'] );
532
			if ( !$media )
0 ignored issues
show
Bug Best Practice introduced by
The expression $media of type array is implicitly converted to a boolean; are you sure this is intended? If so, consider using empty($expr) instead to make it clear that you intend to check for an array without elements.

This check marks implicit conversions of arrays to boolean values in a comparison. While in PHP an empty array is considered to be equal (but not identical) to false, this is not always apparent.

Consider making the comparison explicit by using empty(..) or ! empty(...) instead.

Loading history...
533
				$media = self::from_gravatar( $post_id, $args['avatar_size'], $args['gravatar_default'] );
534
		}
535
536
		/**
537
		 * Filters the array of images that would be good for a specific post.
538
		 * This filter is applied after options ($args) filter the original array.
539
		 *
540
		 * @since 2.0.0
541
		 *
542
		 * @param array $media Array of images that would be good for a specific post.
543
		 * @param int $post_id Post ID.
544
		 * @param array $args Array of options to get images.
545
		 */
546
		return apply_filters( 'jetpack_images_get_images', $media, $post_id, $args );
547
	}
548
549
	/**
550
	 * Takes an image URL and pixel dimensions then returns a URL for the
551
	 * resized and cropped image.
552
	 *
553
	 * @param  string $src
554
	 * @param  int    $dimension
0 ignored issues
show
Bug introduced by
There is no parameter named $dimension. Was it maybe removed?

This check looks for PHPDoc comments describing methods or function parameters that do not exist on the corresponding method or function.

Consider the following example. The parameter $italy is not defined by the method finale(...).

/**
 * @param array $germany
 * @param array $island
 * @param array $italy
 */
function finale($germany, $island) {
    return "2:1";
}

The most likely cause is that the parameter was removed, but the annotation was not.

Loading history...
555
	 * @return string            Transformed image URL
556
	 */
557
	static function fit_image_url( $src, $width, $height ) {
558
		$width = (int) $width;
559
		$height = (int) $height;
560
561
		if ( $width < 1 || $height < 1 ) {
562
			return $src;
563
		}
564
565
		// See if we should bypass WordPress.com SaaS resizing
566
		if ( has_filter( 'jetpack_images_fit_image_url_override' ) ) {
567
			/**
568
			 * Filters the image URL used after dimensions are set by Photon.
569
			 *
570
			 * @since 3.3.0
571
			 *
572
			 * @param string $src Image URL.
573
			 * @param int $width Image width.
574
			 * @param int $width Image height.
575
			 */
576
			return apply_filters( 'jetpack_images_fit_image_url_override', $src, $width, $height );
577
		}
578
579
		// If WPCOM hosted image use native transformations
580
		$img_host = parse_url( $src, PHP_URL_HOST );
581
		if ( '.files.wordpress.com' == substr( $img_host, -20 ) ) {
582
			return add_query_arg( array( 'w' => $width, 'h' => $height, 'crop' => 1 ), set_url_scheme( $src ) );
583
		}
584
585
		// Use Photon magic
586
		if( function_exists( 'jetpack_photon_url' ) ) {
587
			return jetpack_photon_url( $src, array( 'resize' => "$width,$height" ) );
588
		}
589
		
590
		// Arg... no way to resize image using WordPress.com infrastructure!
591
		return $src;
592
	}
593
}
594