Duplicate code is one of the most pungent code smells. A rule that is often used is to re-structure code once it is duplicated in three or more places.
Common duplication problems, and corresponding solutions are:
Complex classes like autoptimizeImages often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes. You can also have a look at the cohesion graph to spot any un-connected, or weakly-connected components.
Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.
While breaking up the class, it is a good idea to analyze how other classes use autoptimizeImages, and based on these observations, apply Extract Interface, too.
1 | <?php |
||
10 | class autoptimizeImages |
||
11 | { |
||
12 | /** |
||
13 | * Options. |
||
14 | * |
||
15 | * @var array |
||
16 | */ |
||
17 | protected $options = array(); |
||
18 | |||
19 | /** |
||
20 | * Singleton instance. |
||
21 | * |
||
22 | * @var self|null |
||
23 | */ |
||
24 | protected static $instance = null; |
||
25 | |||
26 | public function __construct( array $options = array() ) |
||
27 | { |
||
28 | // If options are not provided, fetch them. |
||
29 | if ( empty( $options ) ) { |
||
30 | $options = $this->fetch_options(); |
||
31 | } |
||
32 | |||
33 | $this->set_options( $options ); |
||
34 | } |
||
35 | |||
36 | public function set_options( array $options ) |
||
37 | { |
||
38 | $this->options = $options; |
||
39 | |||
40 | return $this; |
||
41 | } |
||
42 | |||
43 | public static function fetch_options() |
||
44 | { |
||
45 | $value = get_option( 'autoptimize_imgopt_settings' ); |
||
46 | if ( empty( $value ) ) { |
||
47 | // Fallback to returning defaults when no stored option exists yet. |
||
48 | $value = autoptimizeConfig::get_ao_imgopt_default_options(); |
||
49 | } |
||
50 | |||
51 | // get service availability and add it to the options-array. |
||
52 | $value['availabilities'] = get_option( 'autoptimize_service_availablity' ); |
||
53 | |||
54 | if ( empty( $value['availabilities'] ) ) { |
||
55 | $value['availabilities'] = autoptimizeUtils::check_service_availability( true ); |
||
56 | } |
||
57 | |||
58 | return $value; |
||
59 | } |
||
60 | |||
61 | public static function imgopt_active() |
||
62 | { |
||
63 | // function to quickly check if imgopt is active, used below but also in |
||
64 | // autoptimizeMain.php to start ob_ even if no HTML, JS or CSS optimizing is done |
||
65 | // and does not use/ request the availablity data (which could slow things down). |
||
66 | static $imgopt_active = null; |
||
67 | |||
68 | if ( null === $imgopt_active ) { |
||
69 | $opts = get_option( 'autoptimize_imgopt_settings', '' ); |
||
70 | if ( ! empty( $opts ) && is_array( $opts ) && array_key_exists( 'autoptimize_imgopt_checkbox_field_1', $opts ) && ! empty( $opts['autoptimize_imgopt_checkbox_field_1'] ) && '1' === $opts['autoptimize_imgopt_checkbox_field_1'] ) { |
||
71 | $imgopt_active = true; |
||
72 | } else { |
||
73 | $imgopt_active = false; |
||
74 | } |
||
75 | } |
||
76 | |||
77 | return $imgopt_active; |
||
78 | } |
||
79 | |||
80 | /** |
||
81 | * Helper for getting a singleton instance. While being an |
||
82 | * anti-pattern generally, it comes in handy for now from a |
||
83 | * readability/maintainability perspective, until we get some |
||
84 | * proper dependency injection going. |
||
85 | * |
||
86 | * @return self |
||
87 | */ |
||
88 | public static function instance() |
||
89 | { |
||
90 | if ( null === self::$instance ) { |
||
91 | self::$instance = new self(); |
||
92 | } |
||
93 | |||
94 | return self::$instance; |
||
95 | } |
||
96 | |||
97 | public function run() |
||
98 | { |
||
99 | if ( is_admin() ) { |
||
100 | add_action( 'admin_menu', array( $this, 'imgopt_admin_menu' ) ); |
||
101 | add_filter( 'autoptimize_filter_settingsscreen_tabs', array( $this, 'add_imgopt_tab' ), 9 ); |
||
102 | } else { |
||
103 | $this->run_on_frontend(); |
||
104 | } |
||
105 | } |
||
106 | |||
107 | public function run_on_frontend() { |
||
108 | if ( ! $this->should_run() ) { |
||
109 | if ( $this->should_lazyload() ) { |
||
110 | add_filter( |
||
111 | 'autoptimize_html_after_minify', |
||
112 | array( $this, 'filter_lazyload_images' ), |
||
113 | 10, |
||
114 | 1 |
||
115 | ); |
||
116 | add_action( |
||
117 | 'wp_footer', |
||
118 | array( $this, 'add_lazyload_js_footer' ), |
||
119 | 10, |
||
120 | 0 |
||
121 | ); |
||
122 | } |
||
123 | return; |
||
124 | } |
||
125 | |||
126 | $active = false; |
||
127 | |||
128 | if ( apply_filters( 'autoptimize_filter_imgopt_do', true ) ) { |
||
129 | add_filter( |
||
130 | 'autoptimize_html_after_minify', |
||
131 | array( $this, 'filter_optimize_images' ), |
||
132 | 10, |
||
133 | 1 |
||
134 | ); |
||
135 | $active = true; |
||
136 | } |
||
137 | |||
138 | if ( apply_filters( 'autoptimize_filter_imgopt_do_css', true ) ) { |
||
139 | add_filter( |
||
140 | 'autoptimize_filter_base_replace_cdn', |
||
141 | array( $this, 'filter_optimize_css_images' ), |
||
142 | 10, |
||
143 | 1 |
||
144 | ); |
||
145 | $active = true; |
||
146 | } |
||
147 | |||
148 | if ( $active ) { |
||
149 | add_filter( |
||
150 | 'autoptimize_extra_filter_tobepreconn', |
||
151 | array( $this, 'filter_preconnect_imgopt_url' ), |
||
152 | 10, |
||
153 | 1 |
||
154 | ); |
||
155 | } |
||
156 | |||
157 | if ( $this->should_lazyload() ) { |
||
158 | add_action( |
||
159 | 'wp_footer', |
||
160 | array( $this, 'add_lazyload_js_footer' ) |
||
161 | ); |
||
162 | } |
||
163 | } |
||
164 | |||
165 | /** |
||
166 | * Basic checks before we can run. |
||
167 | * |
||
168 | * @return bool |
||
169 | */ |
||
170 | protected function should_run() |
||
171 | { |
||
172 | $opts = $this->options; |
||
173 | $service_not_down = ( 'down' !== $opts['availabilities']['extra_imgopt']['status'] ); |
||
174 | $not_launch_status = ( 'launch' !== $opts['availabilities']['extra_imgopt']['status'] ); |
||
175 | |||
176 | $do_cdn = true; |
||
177 | $_userstatus = $this->get_imgopt_provider_userstatus(); |
||
178 | if ( -2 == $_userstatus['Status'] ) { |
||
179 | $do_cdn = false; |
||
180 | } |
||
181 | |||
182 | if ( |
||
183 | $this->imgopt_active() |
||
184 | && $do_cdn |
||
185 | && $service_not_down |
||
186 | && ( $not_launch_status || $this->launch_ok() ) |
||
187 | ) { |
||
188 | return true; |
||
189 | } |
||
190 | return false; |
||
191 | } |
||
192 | |||
193 | public function get_imgopt_host() |
||
194 | { |
||
195 | static $imgopt_host = null; |
||
196 | |||
197 | if ( null === $imgopt_host ) { |
||
198 | $imgopt_host = 'https://cdn.shortpixel.ai/'; |
||
199 | $avail_imgopt = $this->options['availabilities']['extra_imgopt']; |
||
200 | if ( ! empty( $avail_imgopt ) && array_key_exists( 'hosts', $avail_imgopt ) && is_array( $avail_imgopt['hosts'] ) ) { |
||
201 | $imgopt_host = array_rand( array_flip( $avail_imgopt['hosts'] ) ); |
||
202 | } |
||
203 | $imgopt_host = apply_filters('autoptimize_filter_imgopt_host', $imgopt_host ); |
||
204 | } |
||
205 | |||
206 | return $imgopt_host; |
||
207 | } |
||
208 | |||
209 | public static function get_imgopt_host_wrapper() |
||
210 | { |
||
211 | // needed for CI tests. |
||
212 | $self = new self(); |
||
213 | return $self->get_imgopt_host(); |
||
214 | } |
||
215 | |||
216 | public static function get_service_url_suffix() |
||
217 | { |
||
218 | $suffix = '/af/GWRGFLW109483/' . AUTOPTIMIZE_SITE_DOMAIN; |
||
219 | |||
220 | return $suffix; |
||
221 | } |
||
222 | |||
223 | public function get_img_quality_string() |
||
224 | { |
||
225 | static $quality = null; |
||
226 | |||
227 | if ( null === $quality ) { |
||
228 | $q_array = $this->get_img_quality_array(); |
||
229 | $setting = $this->get_img_quality_setting(); |
||
230 | $quality = apply_filters( |
||
231 | 'autoptimize_filter_imgopt_quality', |
||
232 | 'q_' . $q_array[ $setting ] |
||
233 | ); |
||
234 | } |
||
235 | |||
236 | return $quality; |
||
237 | } |
||
238 | |||
239 | public function get_img_quality_array() |
||
240 | { |
||
241 | static $map = null; |
||
242 | |||
243 | if ( null === $map ) { |
||
244 | $map = array( |
||
245 | '1' => 'lossy', |
||
246 | '2' => 'glossy', |
||
247 | '3' => 'lossless', |
||
248 | ); |
||
249 | $map = apply_filters( |
||
250 | 'autoptimize_filter_imgopt_quality_array', |
||
251 | $map |
||
252 | ); |
||
253 | } |
||
254 | |||
255 | return $map; |
||
256 | } |
||
257 | |||
258 | public function get_img_quality_setting() |
||
259 | { |
||
260 | static $q = null; |
||
261 | |||
262 | if ( null === $q ) { |
||
263 | if ( is_array( $this->options ) && array_key_exists( 'autoptimize_imgopt_select_field_2', $this->options ) ) { |
||
264 | $setting = $this->options['autoptimize_imgopt_select_field_2']; |
||
265 | } |
||
266 | |||
267 | if ( ! isset( $setting ) || empty( $setting ) || ( '1' !== $setting && '3' !== $setting ) ) { |
||
268 | // default image opt. value is 2 ("glossy"). |
||
269 | $q = '2'; |
||
270 | } else { |
||
271 | $q = $setting; |
||
272 | } |
||
273 | } |
||
274 | |||
275 | return $q; |
||
276 | } |
||
277 | |||
278 | public function filter_preconnect_imgopt_url( array $in ) |
||
279 | { |
||
280 | $url_parts = parse_url( $this->get_imgopt_base_url() ); |
||
281 | $in[] = $url_parts['scheme'] . '://' . $url_parts['host']; |
||
282 | |||
283 | return $in; |
||
284 | } |
||
285 | |||
286 | /** |
||
287 | * Makes sure given url contains the full scheme and hostname |
||
288 | * in case they're not present already. |
||
289 | * |
||
290 | * @param string $in Image url to normalize. |
||
291 | * |
||
292 | * @return string |
||
293 | */ |
||
294 | private function normalize_img_url( $in ) |
||
295 | { |
||
296 | // Only parse the site url once. |
||
297 | static $parsed_site_url = null; |
||
298 | if ( null === $parsed_site_url ) { |
||
299 | $parsed_site_url = parse_url( site_url() ); |
||
300 | } |
||
301 | |||
302 | // get CDN domain once. |
||
303 | static $cdn_domain = null; |
||
304 | if ( is_null( $cdn_domain ) ) { |
||
305 | $cdn_url = apply_filters( 'autoptimize_filter_base_cdnurl', get_option( 'autoptimize_cdn_url', '' ) ); |
||
306 | if ( ! empty( $cdn_url ) ) { |
||
307 | $cdn_domain = parse_url( $cdn_url, PHP_URL_HOST ); |
||
308 | } else { |
||
309 | $cdn_domain = ''; |
||
310 | } |
||
311 | } |
||
312 | |||
313 | /** |
||
314 | * This method gets called a lot, often for identical urls it seems. |
||
315 | * `filter_optimize_css_images()` calls us, uses the resulting url and |
||
316 | * gives it to `can_optimize_image()`, and if that returns trueish |
||
317 | * then `build_imgopt_url()` is called (which, again, calls this method). |
||
318 | * Until we dig deeper into whether this all must really happen that |
||
319 | * way, having an internal cache here helps (to avoid doing repeated |
||
320 | * identical string operations). |
||
321 | */ |
||
322 | static $cache = null; |
||
323 | if ( null === $cache ) { |
||
324 | $cache = array(); |
||
325 | } |
||
326 | |||
327 | // Do the work on cache miss only. |
||
328 | if ( ! isset( $cache[ $in ] ) ) { |
||
329 | // Default to what was given to us. |
||
330 | $result = $in; |
||
331 | if ( autoptimizeUtils::is_protocol_relative( $in ) ) { |
||
332 | $result = $parsed_site_url['scheme'] . ':' . $in; |
||
333 | } elseif ( 0 === strpos( $in, '/' ) ) { |
||
334 | // Root-relative... |
||
335 | $result = $parsed_site_url['scheme'] . '://' . $parsed_site_url['host']; |
||
336 | // Add the path for subfolder installs. |
||
337 | if ( isset( $parsed_site_url['path'] ) ) { |
||
338 | $result .= $parsed_site_url['path']; |
||
339 | } |
||
340 | $result .= $in; |
||
341 | } elseif ( ! empty( $cdn_domain ) && strpos( $in, $cdn_domain ) !== 0 ) { |
||
342 | $result = str_replace( $cdn_domain, $parsed_site_url['host'], $in ); |
||
343 | } |
||
344 | |||
345 | $result = apply_filters( 'autoptimize_filter_imgopt_normalized_url', $result ); |
||
346 | |||
347 | // Store in cache. |
||
348 | $cache[ $in ] = $result; |
||
349 | } |
||
350 | |||
351 | return $cache[ $in ]; |
||
352 | } |
||
353 | |||
354 | public function filter_optimize_css_images( $in ) |
||
355 | { |
||
356 | $in = $this->normalize_img_url( $in ); |
||
357 | |||
358 | if ( $this->can_optimize_image( $in ) ) { |
||
359 | return $this->build_imgopt_url( $in, '', '' ); |
||
360 | } else { |
||
361 | return $in; |
||
362 | } |
||
363 | } |
||
364 | |||
365 | private function get_imgopt_base_url() |
||
366 | { |
||
367 | static $imgopt_base_url = null; |
||
368 | |||
369 | if ( null === $imgopt_base_url ) { |
||
370 | $imgopt_host = $this->get_imgopt_host(); |
||
371 | $quality = $this->get_img_quality_string(); |
||
372 | $ret_val = apply_filters( 'autoptimize_filter_imgopt_wait', 'ret_img' ); // values: ret_wait, ret_img, ret_json, ret_blank. |
||
373 | $imgopt_base_url = $imgopt_host . 'client/' . $quality . ',' . $ret_val; |
||
374 | $imgopt_base_url = apply_filters( 'autoptimize_filter_imgopt_base_url', $imgopt_base_url ); |
||
375 | } |
||
376 | |||
377 | return $imgopt_base_url; |
||
378 | } |
||
379 | |||
380 | private function can_optimize_image( $url ) |
||
381 | { |
||
382 | static $cdn_url = null; |
||
383 | static $nopti_images = null; |
||
384 | |||
385 | if ( null === $cdn_url ) { |
||
386 | $cdn_url = apply_filters( |
||
387 | 'autoptimize_filter_base_cdnurl', |
||
388 | get_option( 'autoptimize_cdn_url', '' ) |
||
389 | ); |
||
390 | } |
||
391 | |||
392 | if ( null === $nopti_images ) { |
||
393 | $nopti_images = apply_filters( 'autoptimize_filter_imgopt_noptimize', '' ); |
||
394 | } |
||
395 | |||
396 | $site_host = AUTOPTIMIZE_SITE_DOMAIN; |
||
397 | $url = $this->normalize_img_url( $url ); |
||
398 | $url_parsed = parse_url( $url ); |
||
399 | |||
400 | if ( array_key_exists( 'host', $url_parsed ) && $url_parsed['host'] !== $site_host && empty( $cdn_url ) ) { |
||
401 | return false; |
||
402 | } elseif ( ! empty( $cdn_url ) && strpos( $url, $cdn_url ) === false && array_key_exists( 'host', $url_parsed ) && $url_parsed['host'] !== $site_host ) { |
||
403 | return false; |
||
404 | } elseif ( strpos( $url, '.php' ) !== false ) { |
||
405 | return false; |
||
406 | } elseif ( str_ireplace( array( '.png', '.gif', '.jpg', '.jpeg', '.webp' ), '', $url_parsed['path'] ) === $url_parsed['path'] ) { |
||
407 | // fixme: better check against end of string. |
||
408 | return false; |
||
409 | } elseif ( ! empty( $nopti_images ) ) { |
||
410 | $nopti_images_array = array_filter( array_map( 'trim', explode( ',', $nopti_images ) ) ); |
||
411 | foreach ( $nopti_images_array as $nopti_image ) { |
||
412 | if ( strpos( $url, $nopti_image ) !== false ) { |
||
413 | return false; |
||
414 | } |
||
415 | } |
||
416 | } |
||
417 | return true; |
||
418 | } |
||
419 | |||
420 | private function build_imgopt_url( $orig_url, $width = 0, $height = 0 ) |
||
421 | { |
||
422 | // sanitize width and height. |
||
423 | if ( strpos( $width, '%' ) !== false ) { |
||
424 | $width = 0; |
||
425 | } |
||
426 | if ( strpos( $height, '%' ) !== false ) { |
||
427 | $height = 0; |
||
428 | } |
||
429 | $width = (int) $width; |
||
430 | $height = (int) $height; |
||
431 | |||
432 | $filtered_url = apply_filters( |
||
433 | 'autoptimize_filter_imgopt_build_url', |
||
434 | $orig_url, |
||
435 | $width, |
||
436 | $height |
||
437 | ); |
||
438 | |||
439 | // If filter modified the url, return that. |
||
440 | if ( $filtered_url !== $orig_url ) { |
||
441 | return $filtered_url; |
||
442 | } |
||
443 | |||
444 | $orig_url = $this->normalize_img_url( $orig_url ); |
||
445 | $imgopt_base_url = $this->get_imgopt_base_url(); |
||
446 | $imgopt_size = ''; |
||
447 | |||
448 | if ( $width && 0 !== $width ) { |
||
449 | $imgopt_size = ',w_' . $width; |
||
450 | } |
||
451 | |||
452 | if ( $height && 0 !== $height ) { |
||
453 | $imgopt_size .= ',h_' . $height; |
||
454 | } |
||
455 | |||
456 | $url = $imgopt_base_url . $imgopt_size . '/' . $orig_url; |
||
457 | |||
458 | return $url; |
||
459 | } |
||
460 | |||
461 | public function replace_data_thumbs( $matches ) |
||
462 | { |
||
463 | return $this->replace_img_callback( $matches, 150, 150 ); |
||
464 | } |
||
465 | |||
466 | public function replace_img_callback( $matches, $width = 0, $height = 0 ) |
||
467 | { |
||
468 | if ( $this->can_optimize_image( $matches[1] ) ) { |
||
469 | return str_replace( $matches[1], $this->build_imgopt_url( $matches[1], $width, $height ), $matches[0] ); |
||
470 | } else { |
||
471 | return $matches[0]; |
||
472 | } |
||
473 | } |
||
474 | |||
475 | public function filter_optimize_images( $in ) |
||
476 | { |
||
477 | /* |
||
478 | * potential future functional improvements: |
||
479 | * |
||
480 | * picture element. |
||
481 | * filter for critical CSS. |
||
482 | */ |
||
483 | $to_replace = array(); |
||
484 | |||
485 | // hide noscript tags to avoid nesting noscript tags (as lazyloaded images add noscript). |
||
486 | if ( $this->should_lazyload() ) { |
||
487 | $in = autoptimizeBase::replace_contents_with_marker_if_exists( |
||
488 | 'NOSCRIPT', |
||
489 | '<noscript', |
||
490 | '#<noscript.*?<\/noscript>#is', |
||
491 | $in |
||
492 | ); |
||
493 | } |
||
494 | |||
495 | // extract img tags. |
||
496 | if ( preg_match_all( '#<img[^>]*src[^>]*>#Usmi', $in, $matches ) ) { |
||
497 | foreach ( $matches[0] as $tag ) { |
||
498 | $orig_tag = $tag; |
||
499 | $imgopt_w = ''; |
||
500 | $imgopt_h = ''; |
||
501 | |||
502 | // first do (data-)srcsets. |
||
503 | if ( preg_match_all( '#srcset=("|\')(.*)("|\')#Usmi', $tag, $allsrcsets, PREG_SET_ORDER ) ) { |
||
504 | foreach ( $allsrcsets as $srcset ) { |
||
505 | $srcset = $srcset[2]; |
||
506 | $srcsets = explode( ',', $srcset ); |
||
507 | foreach ( $srcsets as $indiv_srcset ) { |
||
508 | $indiv_srcset_parts = explode( ' ', trim( $indiv_srcset ) ); |
||
509 | if ( isset( $indiv_srcset_parts[1] ) && rtrim( $indiv_srcset_parts[1], 'w' ) !== $indiv_srcset_parts[1] ) { |
||
510 | $imgopt_w = rtrim( $indiv_srcset_parts[1], 'w' ); |
||
511 | } |
||
512 | if ( $this->can_optimize_image( $indiv_srcset_parts[0] ) ) { |
||
513 | $imgopt_url = $this->build_imgopt_url( $indiv_srcset_parts[0], $imgopt_w, '' ); |
||
514 | $tag = str_replace( $indiv_srcset_parts[0], $imgopt_url, $tag ); |
||
515 | } |
||
516 | } |
||
517 | } |
||
518 | } |
||
519 | |||
520 | // proceed with img src. |
||
521 | // get width and height and add to $imgopt_size. |
||
522 | $_get_size = $this->get_size_from_tag( $tag ); |
||
523 | $imgopt_w = $_get_size['width']; |
||
524 | $imgopt_h = $_get_size['height']; |
||
525 | |||
526 | // then start replacing images src. |
||
527 | if ( preg_match_all( '#src=(?:"|\')(?!data)(.*)(?:"|\')#Usmi', $tag, $urls, PREG_SET_ORDER ) ) { |
||
528 | foreach ( $urls as $url ) { |
||
529 | $full_src_orig = $url[0]; |
||
530 | $url = $url[1]; |
||
531 | if ( $this->can_optimize_image( $url ) ) { |
||
532 | $imgopt_url = $this->build_imgopt_url( $url, $imgopt_w, $imgopt_h ); |
||
533 | $full_imgopt_src = str_replace( $url, $imgopt_url, $full_src_orig ); |
||
534 | $tag = str_replace( $full_src_orig, $full_imgopt_src, $tag ); |
||
535 | } |
||
536 | } |
||
537 | } |
||
538 | |||
539 | // do lazyload stuff. |
||
540 | if ( $this->should_lazyload() && str_ireplace( $this->get_lazyload_exclusions(), '', $tag ) === $tag ) { |
||
541 | $noscript_tag = '<noscript>' . $tag . '</noscript>'; |
||
542 | $tag = str_replace( 'srcset=', 'data-srcset=', $tag ); |
||
543 | |||
544 | // add lazyload class. |
||
545 | $tag = $this->inject_classes_in_tag( $tag, 'lazyload ' ); |
||
546 | |||
547 | // set placeholder. |
||
548 | if ( strpos( $url, $this->get_imgopt_host() ) === 0 ) { |
||
549 | // if all img src have been replaced during srcset, we have to extract the |
||
550 | // origin url from the imgopt one to be able to set a lqip placeholder. |
||
551 | $_url = substr( $url, strpos( $url, '/http' ) + 1 ); |
||
552 | } else { |
||
553 | $_url = $url; |
||
554 | } |
||
555 | if ( $this->can_optimize_image( $_url ) && apply_filters( 'autoptimize_filter_imgopt_lazyload_dolqip', true ) ) { |
||
556 | $lqip_w = ''; |
||
557 | $lqip_h = ''; |
||
558 | if isset( $imgopt_w ) { |
||
|
|||
559 | $lqip_w = ',w_' . $imgopt_w; |
||
560 | } |
||
561 | if isset( $imgopt_h ) { |
||
562 | $lqip_h = ',h_' . $imgopt_h; |
||
563 | } |
||
564 | $placeholder = $this->get_imgopt_host() . 'client/q_lqip,ret_wait' . $lqip_w . $lqip_h . '/' . $_url; |
||
565 | } else { |
||
566 | $placeholder = $this->get_default_lazyload_placeholder( $imgopt_w, $imgopt_h ); |
||
567 | } |
||
568 | $placeholder = ' src=\'' . apply_filters( 'autoptimize_filter_imgopt_lazyload_placeholder', $placeholder ); |
||
569 | |||
570 | // add min-heigth off by default as it can deform images, can be enabled with filter. |
||
571 | $min_height = ''; |
||
572 | if ( apply_filters( 'autoptimize_filter_imgopt_lazyload_addminheight', false ) ) { |
||
573 | $min_height = ' style="min-height:' . $imgopt_h . 'px;"'; |
||
574 | } |
||
575 | |||
576 | // add noscript & placeholder. |
||
577 | $tag = $noscript_tag . str_replace( ' src=', $min_height . $placeholder . '\' data-src=', $tag ); |
||
578 | $tag = apply_filters( 'autoptimize_filter_imgopt_lazyloaded_img', $tag ); |
||
579 | } |
||
580 | |||
581 | // add tag to array for later replacement. |
||
582 | if ( $tag !== $orig_tag ) { |
||
583 | $to_replace[ $orig_tag ] = $tag; |
||
584 | } |
||
585 | } |
||
586 | } |
||
587 | |||
588 | // and replace all. |
||
589 | $out = str_replace( array_keys( $to_replace ), array_values( $to_replace ), $in ); |
||
590 | |||
591 | // img thumbnails in e.g. woocommerce. |
||
592 | if ( strpos( $out, 'data-thumb' ) !== false && apply_filters( 'autoptimize_filter_imgopt_datathumbs', true ) ) { |
||
593 | $out = preg_replace_callback( |
||
594 | '/\<div(?:[^>]?)\sdata-thumb\=(?:\"|\')(.+?)(?:\"|\')(?:[^>]*)?\>/s', |
||
595 | array( $this, 'replace_data_thumbs' ), |
||
596 | $out |
||
597 | ); |
||
598 | } |
||
599 | |||
600 | // background-image in inline style. |
||
601 | if ( strpos( $out, 'background-image:' ) !== false && apply_filters( 'autoptimize_filter_imgopt_backgroundimages', true ) ) { |
||
602 | $out = preg_replace_callback( |
||
603 | '/style=(?:"|\').*?background-image:\s?url\((?:"|\')?([^"\')]*)(?:"|\')?\)/', |
||
604 | array( $this, 'replace_img_callback' ), |
||
605 | $out |
||
606 | ); |
||
607 | } |
||
608 | |||
609 | // and restore noscript tags if these were hidden for lazyload purposes. |
||
610 | if ( $this->should_lazyload() ) { |
||
611 | $out = autoptimizeBase::restore_marked_content( |
||
612 | 'NOSCRIPT', |
||
613 | $out |
||
614 | ); |
||
615 | } |
||
616 | |||
617 | return $out; |
||
618 | } |
||
619 | |||
620 | public function get_size_from_tag( $tag ) { |
||
621 | // reusable function to extract widht and height from an image tag |
||
622 | // enforcing a filterable maximum width and height (default 4999X4999). |
||
623 | $width = ''; |
||
624 | $height = ''; |
||
625 | |||
626 | if ( preg_match( '#width=("|\')(.*)("|\')#Usmi', $tag, $_width ) ) { |
||
627 | if ( strpos( $_width[2], '%' ) === false ) { |
||
628 | $width = (int) $_width[2]; |
||
629 | } |
||
630 | } |
||
631 | if ( preg_match( '#height=("|\')(.*)("|\')#Usmi', $tag, $_height ) ) { |
||
632 | if ( strpos( $_height[2], '%' ) === false ) { |
||
633 | $height = (int) $_height[2]; |
||
634 | } |
||
635 | } |
||
636 | |||
637 | // check for and enforce (filterable) max sizes. |
||
638 | $_max_width = apply_filters( 'autoptimize_filter_imgopt_max_width', 4999 ); |
||
639 | if ( $width > $_max_width ) { |
||
640 | $_width = $_max_width; |
||
641 | $height = $_width / $width * $height; |
||
642 | $width = $_width; |
||
643 | } |
||
644 | $_max_height = apply_filters( 'autoptimize_filter_imgopt_max_height', 4999 ); |
||
645 | if ( $height > $_max_height ) { |
||
646 | $_height = $_max_height; |
||
647 | $width = $_height / $height * $width; |
||
648 | $height = $_height; |
||
649 | } |
||
650 | |||
651 | return array( |
||
652 | 'width' => $width, |
||
653 | 'height' => $height, |
||
654 | ); |
||
655 | } |
||
656 | |||
657 | /** |
||
658 | * Lazyload functions |
||
659 | */ |
||
660 | public function should_lazyload() { |
||
661 | if ( ! empty( $this->options['autoptimize_imgopt_checkbox_field_3'] ) ) { |
||
662 | $lazyload_return = true; |
||
663 | } else { |
||
664 | $lazyload_return = false; |
||
665 | } |
||
666 | |||
667 | return $lazyload_return; |
||
668 | } |
||
669 | |||
670 | public static function should_lazyload_wrapper() { |
||
671 | // needed in autoptimizeMain.php. |
||
672 | $self = new self(); |
||
673 | return $self->should_lazyload(); |
||
674 | } |
||
675 | |||
676 | public function filter_lazyload_images( $in ) |
||
677 | { |
||
678 | // only used is image optimization is NOT active but lazyload is. |
||
679 | $to_replace = array(); |
||
680 | |||
681 | // hide noscript tags to avoid nesting noscript tags (as lazyloaded images add noscript). |
||
682 | $out = autoptimizeBase::replace_contents_with_marker_if_exists( |
||
683 | 'NOSCRIPT', |
||
684 | '<noscript', |
||
685 | '#<noscript.*?<\/noscript>#is', |
||
686 | $in |
||
687 | ); |
||
688 | |||
689 | // extract img tags and add lazyload attribs. |
||
690 | if ( preg_match_all( '#<img[^>]*src[^>]*>#Usmi', $out, $matches ) ) { |
||
691 | foreach ( $matches[0] as $tag ) { |
||
692 | $to_replace[ $tag ] = $this->add_lazyload( $tag ); |
||
693 | } |
||
694 | $out = str_replace( array_keys( $to_replace ), array_values( $to_replace ), $out ); |
||
695 | } |
||
696 | |||
697 | // restore noscript tags. |
||
698 | $out = autoptimizeBase::restore_marked_content( |
||
699 | 'NOSCRIPT', |
||
700 | $out |
||
701 | ); |
||
702 | |||
703 | return $out; |
||
704 | } |
||
705 | |||
706 | public function add_lazyload( $tag ) { |
||
707 | // adds actual lazyload-attributes to an image node. |
||
708 | if ( str_ireplace( $this->get_lazyload_exclusions(), '', $tag ) === $tag ) { |
||
709 | // store original tag for use in noscript version. |
||
710 | $noscript_tag = '<noscript>' . $tag . '</noscript>'; |
||
711 | |||
712 | // insert lazyload class. |
||
713 | $tag = $this->inject_classes_in_tag( $tag, 'lazyload ' ); |
||
714 | |||
715 | // get image width & heigth for placeholder fun (and to prevent content reflow). |
||
716 | $_get_size = $this->get_size_from_tag( $tag ); |
||
717 | $width = $_get_size['width']; |
||
718 | $height = $_get_size['height']; |
||
719 | if ( false === $width ) { |
||
720 | $widht = 210; // default width for SVG placeholder. |
||
721 | } |
||
722 | if ( false === $height ) { |
||
723 | $heigth = $width / 3 * 2; // if no height, base it on width using the 3/2 aspect ratio. |
||
724 | } |
||
725 | |||
726 | // insert the actual lazyload stuff. |
||
727 | // see https://css-tricks.com/preventing-content-reflow-from-lazy-loaded-images/ for great read on why we're using empty svg's. |
||
728 | $placeholder = apply_filters( 'autoptimize_filter_imgopt_lazyload_placeholder', $this->get_default_lazyload_placeholder( $width, $height ) ); |
||
729 | $tag = str_replace( ' src=', ' src=\'' . $placeholder . '\' data-src=', $tag ); |
||
730 | $tag = str_replace( ' srcset=', ' data-srcset=', $tag ); |
||
731 | |||
732 | // add the noscript-tag from earlier. |
||
733 | $tag = $noscript_tag . $tag; |
||
734 | $tag = apply_filters( 'autoptimize_filter_imgopt_lazyloaded_img', $tag ); |
||
735 | } |
||
736 | |||
737 | return $tag; |
||
738 | } |
||
739 | |||
740 | public function add_lazyload_js_footer() { |
||
741 | // The JS will by default be excluded form autoptimization but this can be changed with a filter. |
||
742 | $noptimize_flag = ''; |
||
743 | if ( apply_filters( 'autoptimize_filter_imgopt_lazyload_js_noptimize', true ) ) { |
||
744 | $noptimize_flag = ' data-noptimize="1"'; |
||
745 | } |
||
746 | |||
747 | // Adds lazyload CSS & JS to footer, using echo because wp_enqueue_script seems not to support pushing attributes (async). |
||
748 | echo apply_filters( 'autoptimize_filter_imgopt_lazyload_cssoutput', '<style>.lazyload,.lazyloading{opacity:0;}.lazyloaded{opacity:1;transition:opacity 300ms;}</style><noscript><style>.lazyload{display:none;}</style></noscript>' ); |
||
749 | echo apply_filters( 'autoptimize_filter_imgopt_lazyload_jsconfig', '<script' . $noptimize_flag . '>window.lazySizesConfig=window.lazySizesConfig||{};window.lazySizesConfig.loadMode=1;</script>' ); |
||
750 | echo '<script async' . $noptimize_flag . ' src=\'' . plugins_url( 'external/js/lazysizes.min.js', __FILE__ ) . '\'></script>'; |
||
751 | |||
752 | // And add webp detection and loading JS. |
||
753 | if ( $this->should_webp() ) { |
||
754 | $_webp_detect = "function c_webp(A){var n=new Image;n.onload=function(){var e=0<n.width&&0<n.height;A(e)},n.onerror=function(){A(!1)},n.src=''}function s_webp(e){window.supportsWebP=e}c_webp(s_webp);"; |
||
755 | $_webp_load = "document.addEventListener('lazybeforeunveil',function({target:c}){supportsWebP&&['data-src','data-srcset'].forEach(function(a){attr=c.getAttribute(a),null!==attr&&c.setAttribute(a,attr.replace(/\/client\//,'/client/to_webp,'))})});"; |
||
756 | echo apply_filters( 'autoptimize_filter_imgopt_webp_js', '<script' . $noptimize_flag . '>' . $_webp_detect . $_webp_load . '</script>' ); |
||
757 | } |
||
758 | } |
||
759 | |||
760 | public function get_lazyload_exclusions() { |
||
761 | // returns array of strings that if found in an <img tag will stop the img from being lazy-loaded. |
||
762 | static $exclude_lazyload_array = null; |
||
763 | |||
764 | if ( null === $exclude_lazyload_array ) { |
||
765 | $options = $this->options; |
||
766 | |||
767 | // set default exclusions. |
||
768 | $exclude_lazyload_array = array( 'skip-lazy', 'data-no-lazy', 'notlazy', 'data-src', 'data-srcset' ); |
||
769 | |||
770 | // add from setting. |
||
771 | if ( array_key_exists( 'autoptimize_imgopt_text_field_5', $options ) ) { |
||
772 | $exclude_lazyload_option = $options['autoptimize_imgopt_text_field_5']; |
||
773 | if ( ! empty( $exclude_lazyload_option ) ) { |
||
774 | $exclude_lazyload_array = array_merge( $exclude_lazyload_array, array_filter( array_map( 'trim', explode( ',', $options['autoptimize_imgopt_text_field_5'] ) ) ) ); |
||
775 | } |
||
776 | } |
||
777 | |||
778 | // and filter for developer-initiated changes. |
||
779 | $exclude_lazyload_array = apply_filters( 'autoptimize_filter_imgopt_lazyload_exclude_array', $exclude_lazyload_array ); |
||
780 | } |
||
781 | |||
782 | return $exclude_lazyload_array; |
||
783 | } |
||
784 | |||
785 | public function inject_classes_in_tag( $tag, $target_class ) { |
||
786 | if ( strpos( $tag, 'class=' ) !== false ) { |
||
787 | $tag = preg_replace( '/(\sclass\s?=\s?("|\'))/', '$1' . $target_class, $tag ); |
||
788 | } else { |
||
789 | $tag = str_replace( '<img ', '<img class="' . trim( $target_class ) . '" ', $tag ); |
||
790 | } |
||
791 | |||
792 | return $tag; |
||
793 | } |
||
794 | |||
795 | public function get_default_lazyload_placeholder( $imgopt_w, $imgopt_h ) { |
||
796 | return 'data:image/svg+xml,%3Csvg%20xmlns=%22http://www.w3.org/2000/svg%22%20viewBox=%220%200%20' . $imgopt_w . '%20' . $imgopt_h . '%22%3E%3C/svg%3E'; |
||
797 | } |
||
798 | |||
799 | public function should_webp() { |
||
800 | static $webp_return = null; |
||
801 | |||
802 | if ( is_null( $webp_return ) ) { |
||
803 | // webp only works if imgopt and lazyload are also active. |
||
804 | if ( ! empty( $this->options['autoptimize_imgopt_checkbox_field_4'] ) && ! empty( $this->options['autoptimize_imgopt_checkbox_field_3'] ) && $this->imgopt_active() ) { |
||
805 | $webp_return = true; |
||
806 | } else { |
||
807 | $webp_return = false; |
||
808 | } |
||
809 | } |
||
810 | |||
811 | return $webp_return; |
||
812 | } |
||
813 | |||
814 | /** |
||
815 | * Admin page logic and related functions below. |
||
816 | */ |
||
817 | public function imgopt_admin_menu() |
||
818 | { |
||
819 | add_submenu_page( |
||
820 | null, |
||
821 | 'autoptimize_imgopt', |
||
822 | 'autoptimize_imgopt', |
||
823 | 'manage_options', |
||
824 | 'autoptimize_imgopt', |
||
825 | array( $this, 'imgopt_options_page' ) |
||
826 | ); |
||
827 | register_setting( 'autoptimize_imgopt_settings', 'autoptimize_imgopt_settings' ); |
||
828 | } |
||
829 | |||
830 | public function add_imgopt_tab( $in ) |
||
831 | { |
||
832 | $in = array_merge( $in, array( 'autoptimize_imgopt' => __( 'Images', 'autoptimize' ) ) ); |
||
833 | |||
834 | return $in; |
||
835 | } |
||
836 | |||
837 | public function imgopt_options_page() |
||
838 | { |
||
839 | // Check querystring for "refreshCacheChecker" and call cachechecker if so. |
||
840 | if ( array_key_exists( 'refreshImgProvStats', $_GET ) && 1 == $_GET['refreshImgProvStats'] ) { |
||
841 | $this->query_img_provider_stats(); |
||
842 | } |
||
843 | |||
844 | $options = $this->fetch_options(); |
||
845 | $sp_url_suffix = $this->get_service_url_suffix(); |
||
846 | ?> |
||
847 | <style> |
||
848 | #ao_settings_form {background: white;border: 1px solid #ccc;padding: 1px 15px;margin: 15px 10px 10px 0;} |
||
849 | #ao_settings_form .form-table th {font-weight: normal;} |
||
850 | #autoptimize_imgopt_descr{font-size: 120%;} |
||
851 | </style> |
||
852 | <div class="wrap"> |
||
853 | <h1><?php _e( 'Autoptimize Settings', 'autoptimize' ); ?></h1> |
||
854 | <?php echo autoptimizeConfig::ao_admin_tabs(); ?> |
||
855 | <?php if ( 'down' === $options['availabilities']['extra_imgopt']['status'] ) { ?> |
||
856 | <div class="notice-warning notice"><p> |
||
857 | <?php |
||
858 | // translators: "Autoptimize support forum" will appear in a "a href". |
||
859 | echo sprintf( __( 'The image optimization service is currently down, image optimization will be skipped until further notice. Check the %1$sAutoptimize support forum%2$s for more info.', 'autoptimize' ), '<a href="https://wordpress.org/support/plugin/autoptimize/" target="_blank">', '</a>' ); |
||
860 | ?> |
||
861 | </p></div> |
||
862 | <?php } ?> |
||
863 | |||
864 | <?php if ( 'launch' === $options['availabilities']['extra_imgopt']['status'] && ! autoptimizeImages::instance()->launch_ok() ) { ?> |
||
865 | <div class="notice-warning notice"><p> |
||
866 | <?php _e( 'The image optimization service is launching, but not yet available for this domain, it should become available in the next couple of days.', 'autoptimize' ); ?> |
||
867 | </p></div> |
||
868 | <?php } ?> |
||
869 | |||
870 | <?php if ( class_exists( 'Jetpack' ) && method_exists( 'Jetpack', 'get_active_modules' ) && in_array( 'photon', Jetpack::get_active_modules() ) ) { ?> |
||
871 | <div class="notice-warning notice"><p> |
||
872 | <?php |
||
873 | // translators: "disable Jetpack's site accelerator for images" will appear in a "a href" linking to the jetpack settings page. |
||
874 | echo sprintf( __( 'Please %1$sdisable Jetpack\'s site accelerator for images%2$s to be able to use Autoptomize\'s advanced image optimization features below.', 'autoptimize' ), '<a href="admin.php?page=jetpack#/settings">', '</a>' ); |
||
875 | ?> |
||
876 | </p></div> |
||
877 | <?php } ?> |
||
878 | <form id='ao_settings_form' action='options.php' method='post'> |
||
879 | <?php settings_fields( 'autoptimize_imgopt_settings' ); ?> |
||
880 | <h2><?php _e( 'Image optimization', 'autoptimize' ); ?></h2> |
||
881 | <span id='autoptimize_imgopt_descr'><?php _e( 'Make your site significantly faster by just ticking a couple of checkboxes to optimize and lazy load your images, WebP support included!', 'autoptimize' ); ?></span> |
||
882 | <table class="form-table"> |
||
883 | <tr> |
||
884 | <th scope="row"><?php _e( 'Optimize Images', 'autoptimize' ); ?></th> |
||
885 | <td> |
||
886 | <label><input id='autoptimize_imgopt_checkbox' type='checkbox' name='autoptimize_imgopt_settings[autoptimize_imgopt_checkbox_field_1]' <?php if ( ! empty( $options['autoptimize_imgopt_checkbox_field_1'] ) && '1' === $options['autoptimize_imgopt_checkbox_field_1'] ) { echo 'checked="checked"'; } ?> value='1'><?php _e( 'Optimize images on the fly and serve them from Shortpixel\'s global CDN.', 'autoptimize' ); ?></label> |
||
887 | <?php |
||
888 | // show shortpixel status. |
||
889 | $_notice = autoptimizeImages::instance()->get_status_notice(); |
||
890 | if ( $_notice ) { |
||
891 | switch ( $_notice['status'] ) { |
||
892 | case 2: |
||
893 | $_notice_color = 'green'; |
||
894 | break; |
||
895 | case 1: |
||
896 | $_notice_color = 'orange'; |
||
897 | break; |
||
898 | case -1: |
||
899 | $_notice_color = 'red'; |
||
900 | break; |
||
901 | case -2: |
||
902 | $_notice_color = 'red'; |
||
903 | break; |
||
904 | default: |
||
905 | $_notice_color = 'green'; |
||
906 | } |
||
907 | echo apply_filters( 'autoptimize_filter_imgopt_settings_status', '<p><strong><span style="color:' . $_notice_color . ';">' . __( 'Shortpixel status: ', 'autoptimize' ) . '</span></strong>' . $_notice['notice'] . '</p>' ); |
||
908 | } else { |
||
909 | // translators: link points to shortpixel. |
||
910 | $upsell_msg_1 = '<p>' . sprintf( __( 'Get more Google love and improve your website\'s loading speed by having the images optimized on the fly (also in the "next-gen" WebP image format) by %1$sShortPixel%2$s and then cached and served fast from Shortpixel\'s global CDN.', 'autoptimize' ), '<a href="https://shortpixel.com/aospai' . $sp_url_suffix . '" target="_blank">', '</a>' ); |
||
911 | if ( 'launch' === $options['availabilities']['extra_imgopt']['status'] ) { |
||
912 | $upsell_msg_2 = __( 'For a limited time only, this service is offered free for all Autoptimize users, <b>don\'t miss the chance to test it</b> and see how much it could improve your site\'s speed.', 'autoptimize' ); |
||
913 | } else { |
||
914 | // translators: link points to shortpixel. |
||
915 | $upsell_msg_2 = sprintf( __( '%1$sSign-up now%2$s to receive a 1 000 bonus + 50% more image optimization credits regardless of the traffic used. More image optimizations can be purchased starting with $4.99.', 'autoptimize' ), '<a href="https://shortpixel.com/aospai' . $sp_url_suffix . '" target="_blank">', '</a>' ); |
||
916 | } |
||
917 | echo apply_filters( 'autoptimize_imgopt_imgopt_settings_copy', $upsell_msg_1 . ' ' . $upsell_msg_2 . '</p>' ); |
||
918 | } |
||
919 | // translators: link points to shortpixel FAQ. |
||
920 | $faqcopy = sprintf( __( '<strong>Questions</strong>? Have a look at the %1$sShortPixel FAQ%2$s!', 'autoptimize' ), '<strong><a href="https://shortpixel.helpscoutdocs.com/category/60-shortpixel-ai-cdn" target="_blank">', '</strong></a>' ); |
||
921 | // translators: links points to shortpixel TOS & Privacy Policy. |
||
922 | $toscopy = sprintf( __( 'Usage of this feature is subject to Shortpixel\'s %1$sTerms of Use%2$s and %3$sPrivacy policy%4$s.', 'autoptimize' ), '<a href="https://shortpixel.com/tos' . $sp_url_suffix . '" target="_blank">', '</a>', '<a href="https://shortpixel.com/pp' . $sp_url_suffix . '" target="_blank">', '</a>' ); |
||
923 | echo apply_filters( 'autoptimize_imgopt_imgopt_settings_tos', '<p>' . $faqcopy . ' ' . $toscopy . '</p>' ); |
||
924 | ?> |
||
925 | </td> |
||
926 | </tr> |
||
927 | <tr id='autoptimize_imgopt_quality' <?php if ( ! array_key_exists( 'autoptimize_imgopt_checkbox_field_1', $options ) || ( isset( $options['autoptimize_imgopt_checkbox_field_1'] ) && '1' !== $options['autoptimize_imgopt_checkbox_field_1'] ) ) { echo 'class="hidden"'; } ?>> |
||
928 | <th scope="row"><?php _e( 'Image Optimization quality', 'autoptimize' ); ?></th> |
||
929 | <td> |
||
930 | <label> |
||
931 | <select name='autoptimize_imgopt_settings[autoptimize_imgopt_select_field_2]'> |
||
932 | <?php |
||
933 | $_imgopt_array = autoptimizeImages::instance()->get_img_quality_array(); |
||
934 | $_imgopt_val = autoptimizeImages::instance()->get_img_quality_setting(); |
||
935 | |||
936 | foreach ( $_imgopt_array as $key => $value ) { |
||
937 | echo '<option value="' . $key . '"'; |
||
938 | if ( $_imgopt_val == $key ) { |
||
939 | echo ' selected'; |
||
940 | } |
||
941 | echo '>' . ucfirst( $value ) . '</option>'; |
||
942 | } |
||
943 | echo "\n"; |
||
944 | ?> |
||
945 | </select> |
||
946 | </label> |
||
947 | <p> |
||
948 | <?php |
||
949 | // translators: link points to shortpixel image test page. |
||
950 | echo apply_filters( 'autoptimize_imgopt_imgopt_quality_copy', sprintf( __( 'You can %1$stest compression levels here%2$s.', 'autoptimize' ), '<a href="https://shortpixel.com/oic' . $sp_url_suffix . '" target="_blank">', '</a>' ) ); |
||
951 | ?> |
||
952 | </p> |
||
953 | </td> |
||
954 | </tr> |
||
955 | <tr id='autoptimize_imgopt_webp' <?php if ( ! array_key_exists( 'autoptimize_imgopt_checkbox_field_1', $options ) || ( isset( $options['autoptimize_imgopt_checkbox_field_1'] ) && '1' !== $options['autoptimize_imgopt_checkbox_field_1'] ) ) { echo 'class="hidden"'; } ?>> |
||
956 | <th scope="row"><?php _e( 'Load WebP in supported browsers?', 'autoptimize' ); ?></th> |
||
957 | <td> |
||
958 | <label><input type='checkbox' id='autoptimize_imgopt_webp_checkbox' name='autoptimize_imgopt_settings[autoptimize_imgopt_checkbox_field_4]' <?php if ( ! empty( $options['autoptimize_imgopt_checkbox_field_4'] ) && '1' === $options['autoptimize_imgopt_checkbox_field_3'] ) { echo 'checked="checked"'; } ?> value='1'><?php _e( 'Automatically serve "next-gen" WebP image format to any browser that supports it (requires lazy load to be active).', 'autoptimize' ); ?></label> |
||
959 | </td> |
||
960 | </tr> |
||
961 | <tr> |
||
962 | <th scope="row"><?php _e( 'Lazy-load images?', 'autoptimize' ); ?></th> |
||
963 | <td> |
||
964 | <label><input type='checkbox' id='autoptimize_imgopt_lazyload_checkbox' name='autoptimize_imgopt_settings[autoptimize_imgopt_checkbox_field_3]' <?php if ( ! empty( $options['autoptimize_imgopt_checkbox_field_3'] ) && '1' === $options['autoptimize_imgopt_checkbox_field_3'] ) { echo 'checked="checked"'; } ?> value='1'><?php _e( 'Image lazy-loading will delay the loading of non-visible images to allow the browser to optimally load all resources for the "above the fold"-page first.', 'autoptimize' ); ?></label> |
||
965 | </td> |
||
966 | </tr> |
||
967 | <tr id='autoptimize_imgopt_lazyload_exclusions' <?php if ( ! array_key_exists( 'autoptimize_imgopt_checkbox_field_3', $options ) || ( isset( $options['autoptimize_imgopt_checkbox_field_3'] ) && '1' !== $options['autoptimize_imgopt_checkbox_field_3'] ) ) { echo 'class="hidden"'; } ?>> |
||
968 | <th scope="row"><?php _e( 'Lazy-load exclusions', 'autoptimize' ); ?></th> |
||
969 | <td> |
||
970 | <label><input type='text' style='width:80%' id='autoptimize_imgopt_lazyload_exclusions' name='autoptimize_imgopt_settings[autoptimize_imgopt_text_field_5]' value='<?php if ( ! empty( $options['autoptimize_imgopt_text_field_5'] ) ) { echo esc_attr( $options['autoptimize_imgopt_text_field_5'] ); } ?>'><br /><?php _e( 'Comma-separated list of to be excluded image classes or filenames.', 'autoptimize' ); ?></label> |
||
971 | </td> |
||
972 | </tr> |
||
973 | </table> |
||
974 | <p class="submit"><input type="submit" name="submit" id="submit" class="button button-primary" value="<?php _e( 'Save Changes', 'autoptimize' ); ?>" /></p> |
||
975 | </form> |
||
976 | <script> |
||
977 | jQuery(document).ready(function() { |
||
978 | jQuery("#autoptimize_imgopt_checkbox").change(function() { |
||
979 | if (this.checked) { |
||
980 | jQuery("#autoptimize_imgopt_quality").show("slow"); |
||
981 | jQuery("#autoptimize_imgopt_webp").show("slow"); |
||
982 | } else { |
||
983 | jQuery("#autoptimize_imgopt_quality").hide("slow"); |
||
984 | jQuery("#autoptimize_imgopt_webp").hide("slow"); |
||
985 | } |
||
986 | }); |
||
987 | jQuery("#autoptimize_imgopt_webp_checkbox").change(function() { |
||
988 | if (this.checked) { |
||
989 | jQuery("#autoptimize_imgopt_lazyload_checkbox")[0].checked = true; |
||
990 | jQuery("#autoptimize_imgopt_lazyload_exclusions").show("slow"); |
||
991 | } |
||
992 | }); |
||
993 | jQuery("#autoptimize_imgopt_lazyload_checkbox").change(function() { |
||
994 | if (this.checked) { |
||
995 | jQuery("#autoptimize_imgopt_lazyload_exclusions").show("slow"); |
||
996 | } else { |
||
997 | jQuery("#autoptimize_imgopt_lazyload_exclusions").hide("slow"); |
||
998 | jQuery("#autoptimize_imgopt_webp_checkbox")[0].checked = false; |
||
999 | } |
||
1000 | }); |
||
1001 | }); |
||
1002 | </script> |
||
1003 | <?php |
||
1004 | } |
||
1005 | |||
1006 | /** |
||
1007 | * Ïmg opt status as used on dashboard. |
||
1008 | */ |
||
1009 | public function get_imgopt_status_notice() { |
||
1010 | if ( $this->imgopt_active() ) { |
||
1011 | $_imgopt_notice = ''; |
||
1012 | $_stat = get_option( 'autoptimize_imgopt_provider_stat', '' ); |
||
1013 | $_site_host = AUTOPTIMIZE_SITE_DOMAIN; |
||
1014 | $_imgopt_upsell = 'https://shortpixel.com/aospai/af/GWRGFLW109483/' . $_site_host; |
||
1015 | |||
1016 | if ( is_array( $_stat ) ) { |
||
1017 | if ( 1 == $_stat['Status'] ) { |
||
1018 | // translators: "add more credits" will appear in a "a href". |
||
1019 | $_imgopt_notice = sprintf( __( 'Your ShortPixel image optimization and CDN quota is almost used, make sure you %1$sadd more credits%2$s to avoid slowing down your website.', 'autoptimize' ), '<a href="' . $_imgopt_upsell . '" target="_blank">', '</a>' ); |
||
1020 | } elseif ( -1 == $_stat['Status'] || -2 == $_stat['Status'] ) { |
||
1021 | // translators: "add more credits" will appear in a "a href". |
||
1022 | $_imgopt_notice = sprintf( __( 'Your ShortPixel image optimization and CDN quota was used, %1$sadd more credits%2$s to keep fast serving optimized images on your site', 'autoptimize' ), '<a href="' . $_imgopt_upsell . '" target="_blank">', '</a>' ); |
||
1023 | $_imgopt_stats_refresh_url = add_query_arg( array( |
||
1024 | 'page' => 'autoptimize_imgopt', |
||
1025 | 'refreshImgProvStats' => '1', |
||
1026 | ), admin_url( 'options-general.php' ) ); |
||
1027 | if ( $_stat && array_key_exists( 'timestamp', $_stat ) && ! empty( $_stat['timestamp'] ) ) { |
||
1028 | $_imgopt_stats_last_run = __( 'based on status at ', 'autoptimize' ) . date_i18n( get_option( 'time_format' ), $_stat['timestamp'] ); |
||
1029 | } else { |
||
1030 | $_imgopt_stats_last_run = __( 'based on previously fetched data', 'autoptimize' ); |
||
1031 | } |
||
1032 | $_imgopt_notice .= ' (' . $_imgopt_stats_last_run . ', '; |
||
1033 | // translators: "here to refresh" links to the Autoptimize Extra page and forces a refresh of the img opt stats. |
||
1034 | $_imgopt_notice .= sprintf( __( 'click %1$shere to refresh%2$s', 'autoptimize' ), '<a href="' . $_imgopt_stats_refresh_url . '">', '</a>).' ); |
||
1035 | } else { |
||
1036 | $_imgopt_upsell = 'https://shortpixel.com/g/af/GWRGFLW109483'; |
||
1037 | // translators: "log in to check your account" will appear in a "a href". |
||
1038 | $_imgopt_notice = sprintf( __( 'Your ShortPixel image optimization and CDN quota are in good shape, %1$slog in to check your account%2$s.', 'autoptimize' ), '<a href="' . $_imgopt_upsell . '" target="_blank">', '</a>' ); |
||
1039 | } |
||
1040 | $_imgopt_notice = apply_filters( 'autoptimize_filter_imgopt_notice', $_imgopt_notice ); |
||
1041 | |||
1042 | return array( |
||
1043 | 'status' => $_stat['Status'], |
||
1044 | 'notice' => $_imgopt_notice, |
||
1045 | ); |
||
1046 | } |
||
1047 | } |
||
1048 | return false; |
||
1049 | } |
||
1050 | |||
1051 | public static function get_imgopt_status_notice_wrapper() { |
||
1052 | // needed for notice being shown in autoptimizeCacheChecker.php. |
||
1053 | $self = new self(); |
||
1054 | return $self->get_imgopt_status_notice(); |
||
1055 | } |
||
1056 | |||
1057 | /** |
||
1058 | * Get img provider stats (used to display notice). |
||
1059 | */ |
||
1060 | public function query_img_provider_stats() { |
||
1061 | if ( ! empty( $this->options['autoptimize_imgopt_checkbox_field_1'] ) ) { |
||
1062 | $url = ''; |
||
1063 | $endpoint = $this->get_imgopt_host() . 'read-domain/'; |
||
1064 | $domain = AUTOPTIMIZE_SITE_DOMAIN; |
||
1065 | |||
1066 | // make sure parse_url result makes sense, keeping $url empty if not. |
||
1067 | if ( $domain && ! empty( $domain ) ) { |
||
1068 | $url = $endpoint . $domain; |
||
1069 | } |
||
1070 | |||
1071 | $url = apply_filters( |
||
1072 | 'autoptimize_filter_imgopt_stat_url', |
||
1073 | $url |
||
1074 | ); |
||
1075 | |||
1076 | // only do the remote call if $url is not empty to make sure no parse_url |
||
1077 | // weirdness results in useless calls. |
||
1078 | if ( ! empty( $url ) ) { |
||
1079 | $response = wp_remote_get( $url ); |
||
1080 | if ( ! is_wp_error( $response ) ) { |
||
1081 | if ( '200' == wp_remote_retrieve_response_code( $response ) ) { |
||
1082 | $stats = json_decode( wp_remote_retrieve_body( $response ), true ); |
||
1083 | update_option( 'autoptimize_imgopt_provider_stat', $stats ); |
||
1084 | } |
||
1085 | } |
||
1086 | } |
||
1087 | } |
||
1088 | } |
||
1089 | |||
1090 | public static function get_img_provider_stats() |
||
1091 | { |
||
1092 | // wrapper around query_img_provider_stats() so we can get to $this->options from cronjob() in autoptimizeCacheChecker. |
||
1093 | $self = new self(); |
||
1094 | return $self->query_img_provider_stats(); |
||
1095 | } |
||
1096 | |||
1097 | /** |
||
1098 | * Determines and returns the service launch status. |
||
1099 | * |
||
1100 | * @return bool |
||
1101 | */ |
||
1102 | public function launch_ok() |
||
1103 | { |
||
1104 | static $launch_status = null; |
||
1105 | |||
1106 | if ( null === $launch_status ) { |
||
1107 | $avail_imgopt = $this->options['availabilities']['extra_imgopt']; |
||
1108 | $magic_number = intval( substr( md5( parse_url( AUTOPTIMIZE_WP_SITE_URL, PHP_URL_HOST ) ), 0, 3 ), 16 ); |
||
1109 | $has_launched = get_option( 'autoptimize_imgopt_launched', '' ); |
||
1110 | $launch_status = false; |
||
1111 | if ( $has_launched || ( is_array( $avail_imgopt ) && array_key_exists( 'launch-threshold', $avail_imgopt ) && $magic_number < $avail_imgopt['launch-threshold'] ) ) { |
||
1112 | $launch_status = true; |
||
1113 | if ( ! $has_launched ) { |
||
1114 | update_option( 'autoptimize_imgopt_launched', 'on' ); |
||
1115 | } |
||
1116 | } |
||
1117 | } |
||
1118 | |||
1119 | return $launch_status; |
||
1120 | } |
||
1121 | |||
1122 | public static function launch_ok_wrapper() { |
||
1123 | // needed for "plug" notice in autoptimizeMain.php. |
||
1124 | $self = new self(); |
||
1125 | return $self->launch_ok(); |
||
1126 | } |
||
1127 | |||
1128 | public function get_imgopt_provider_userstatus() { |
||
1129 | static $_provider_userstatus = null; |
||
1130 | |||
1131 | if ( is_null( $_provider_userstatus ) ) { |
||
1132 | $_stat = get_option( 'autoptimize_imgopt_provider_stat', '' ); |
||
1133 | if ( is_array( $_stat ) ) { |
||
1134 | if ( array_key_exists( 'Status', $_stat ) ) { |
||
1135 | $_provider_userstatus['Status'] = $_stat['Status']; |
||
1136 | } else { |
||
1137 | // if no stats then we assume all is well. |
||
1138 | $_provider_userstatus['Status'] = 2; |
||
1139 | } |
||
1140 | if ( array_key_exists( 'timestamp', $_stat ) ) { |
||
1141 | $_provider_userstatus['timestamp'] = $_stat['timestamp']; |
||
1142 | } else { |
||
1143 | // if no timestamp then we return "". |
||
1144 | $_provider_userstatus['timestamp'] = ''; |
||
1145 | } |
||
1146 | } |
||
1147 | } |
||
1148 | |||
1149 | return $_provider_userstatus; |
||
1150 | } |
||
1151 | |||
1152 | public function get_status_notice() { |
||
1153 | if ( $this->imgopt_active() ) { |
||
1154 | $notice = ''; |
||
1155 | $stat = $this->get_imgopt_provider_userstatus(); |
||
1156 | $upsell = 'https://shortpixel.com/aospai/af/GWRGFLW109483/' . AUTOPTIMIZE_SITE_DOMAIN; |
||
1157 | $assoc = 'https://shortpixel.helpscoutdocs.com/article/94-how-to-associate-a-domain-to-my-account'; |
||
1158 | |||
1159 | if ( is_array( $stat ) ) { |
||
1160 | if ( 1 == $stat['Status'] ) { |
||
1161 | // translators: "add more credits" will appear in a "a href". |
||
1162 | $notice = sprintf( __( 'Your ShortPixel image optimization and CDN quota is almost used, make sure you %1$sadd more credits%2$s to avoid slowing down your website.', 'autoptimize' ), '<a rel="noopener noreferrer" href="' . $upsell . '" target="_blank">', '</a>' ); |
||
1163 | } elseif ( -1 == $stat['Status'] || -2 == $stat['Status'] ) { |
||
1164 | // translators: "add more credits" will appear in a "a href". |
||
1165 | $notice = sprintf( __( 'Your ShortPixel image optimization and CDN quota was used, %1$sadd more credits%2$s to keep fast serving optimized images on your site.', 'autoptimize' ), '<a rel="noopener noreferrer" href="' . $upsell . '" target="_blank">', '</a>' ); |
||
1166 | // translators: "associate your domain" will appear in a "a href". |
||
1167 | $notice = $notice . ' ' . sprintf( __( 'If you already have enough credits then you may need to %1$sassociate your domain%2$s to your Shortpixel account.', 'autoptimize' ), '<a rel="noopener noreferrer" href="' . $assoc . '" target="_blank">', '</a>' ); |
||
1168 | } else { |
||
1169 | $upsell = 'https://shortpixel.com/g/af/GWRGFLW109483'; |
||
1170 | // translators: "log in to check your account" will appear in a "a href". |
||
1171 | $notice = sprintf( __( 'Your ShortPixel image optimization and CDN quota are in good shape, %1$slog in to check your account%2$s.', 'autoptimize' ), '<a rel="noopener noreferrer" href="' . $upsell . '" target="_blank">', '</a>' ); |
||
1172 | } |
||
1173 | $notice = apply_filters( 'autoptimize_filter_imgopt_notice', $notice ); |
||
1174 | |||
1175 | return array( |
||
1184 |