-
Notifications
You must be signed in to change notification settings - Fork 89
/
Copy pathUserExperience_LazyLoad_Plugin.php
339 lines (290 loc) · 9.17 KB
/
UserExperience_LazyLoad_Plugin.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
<?php
/**
* File: UserExperience_LazyLoad_Plugin.php
*
* @package W3TC
*/
namespace W3TC;
/**
* Class UserExperience_LazyLoad_Plugin
*/
class UserExperience_LazyLoad_Plugin {
/**
* Configuration object for lazy loading plugin.
*
* @var Config
*/
private $config;
/**
* Mapping of attachment URLs to post IDs.
*
* @var array
*/
private $posts_by_url = array();
/**
* Constructor to initialize the UserExperience_LazyLoad_Plugin class.
*
* Sets up the configuration object for managing plugin settings.
*
* @return void
*/
public function __construct() {
$this->config = Dispatcher::config();
}
/**
* Runs the lazy loading plugin.
*
* Initializes output buffer callbacks, hooks for various Google Maps plugins,
* and registers necessary WordPress filters.
*
* @return void
*/
public function run() {
Util_Bus::add_ob_callback( 'lazyload', array( $this, 'ob_callback' ) );
$this->metaslider_hooks();
if ( $this->config->get_boolean( 'lazyload.googlemaps.google_maps_easy' ) ) {
$p = new UserExperience_LazyLoad_GoogleMaps_GoogleMapsEasy();
add_filter( 'w3tc_lazyload_mutator_before', array( $p, 'w3tc_lazyload_mutator_before' ) );
}
if ( $this->config->get_boolean( 'lazyload.googlemaps.wp_google_maps' ) ) {
add_filter(
'w3tc_lazyload_mutator_before',
array(
new UserExperience_LazyLoad_GoogleMaps_WPGoogleMaps(),
'w3tc_lazyload_mutator_before',
)
);
}
if ( $this->config->get_boolean( 'lazyload.googlemaps.wp_google_map_plugin' ) ) {
$p = new UserExperience_LazyLoad_GoogleMaps_WPGoogleMapPlugin();
add_filter( 'w3tc_lazyload_mutator_before', array( $p, 'w3tc_lazyload_mutator_before' ) );
}
add_filter( 'wp_get_attachment_url', array( $this, 'wp_get_attachment_url' ), 10, 2 );
add_filter( 'w3tc_footer_comment', array( $this, 'w3tc_footer_comment' ) );
}
/**
* Output buffer callback for processing HTML content.
*
* Modifies the HTML buffer to include lazy loading functionality, embeds the
* lazy loading script, and processes content through a mutator.
*
* @param string $buffer The output buffer content.
*
* @return string The modified or unmodified buffer content.
*/
public function ob_callback( $buffer ) {
if ( '' === $buffer || ! \W3TC\Util_Content::is_html_xml( $buffer ) ) {
return $buffer;
}
$can_process = array(
'enabled' => true,
'buffer' => $buffer,
'reason' => null,
);
$can_process = $this->can_process( $can_process );
$can_process = apply_filters( 'w3tc_lazyload_can_process', $can_process );
// set reject reason in comment.
if ( $can_process['enabled'] ) {
$reject_reason = '';
} else {
$reject_reason = empty( $can_process['reason'] ) ? ' (not specified)' : ' (' . $can_process['reason'] . ')';
}
$buffer = str_replace( '{w3tc_lazyload_reject_reason}', $reject_reason, $buffer );
// processing.
if ( ! $can_process['enabled'] ) {
return $buffer;
}
$mutator = new UserExperience_LazyLoad_Mutator( $this->config, $this->posts_by_url );
$buffer = $mutator->run( $buffer );
// embed lazyload script.
if ( $mutator->content_modified() ) {
$buffer = apply_filters( 'w3tc_lazyload_embed_script', $buffer );
$is_embed_script = apply_filters( 'w3tc_lazyload_is_embed_script', true );
if ( $is_embed_script ) {
$buffer = $this->embed_script( $buffer );
}
}
return $buffer;
}
/**
* Checks if lazy loading can process the current request.
*
* Determines if lazy loading should be enabled based on context such as
* admin area, feed, or short initialization.
*
* @param array $can_process Array with processing status, buffer content, and reason.
*
* @return array Updated array with processing status and reason.
*/
private function can_process( $can_process ) {
if ( defined( 'WP_ADMIN' ) ) {
$can_process['enabled'] = false;
$can_process['reason'] = 'WP_ADMIN';
return $can_process;
}
if ( defined( 'SHORTINIT' ) && SHORTINIT ) {
$can_process['enabled'] = false;
$can_process['reason'] = 'SHORTINIT';
return $can_process;
}
if ( function_exists( 'is_feed' ) && is_feed() ) {
$can_process['enabled'] = false;
$can_process['reason'] = 'feed';
return $can_process;
}
return $can_process;
}
/**
* Appends a lazy loading footer comment to strings.
*
* Adds a footer comment indicating the use of lazy loading and any reject reasons.
*
* @param array $strings Existing footer comments.
*
* @return array Modified footer comments.
*/
public function w3tc_footer_comment( $strings ) {
$strings[] = __( 'Lazy Loading', 'w3-total-cache' ) . '{w3tc_lazyload_reject_reason}';
return $strings;
}
/**
* Embeds the lazy loading script into the HTML content.
*
* Adds the lazy loading JavaScript code and configuration to the appropriate
* section of the HTML buffer.
*
* phpcs:disable WordPress.WP.EnqueuedResources.NonEnqueuedScript
* phpcs:disable WordPress.WP.AlternativeFunctions
*
* @param string $buffer The HTML content buffer.
*
* @return string The modified buffer with the embedded script.
*/
private function embed_script( $buffer ) {
$js_url = plugins_url( 'pub/js/lazyload.min.js', W3TC_FILE );
$method = $this->config->get_string( 'lazyload.embed_method' );
$fire_event = 'function(t){var e;try{e=new CustomEvent("w3tc_lazyload_loaded",{detail:{e:t}})}catch(a){(e=document.createEvent("CustomEvent")).initCustomEvent("w3tc_lazyload_loaded",!1,!1,{e:t})}window.dispatchEvent(e)}';
$thresholds = '';
$config_threshold = $this->config->get_string( 'lazyload.threshold' );
if ( ! empty( $config_threshold ) ) {
$thresholds = 'thresholds:' . wp_json_encode( $config_threshold ) . ',';
}
$config = '{elements_selector:".lazy",' . $thresholds . 'callback_loaded:' . $fire_event . '}';
$on_initialized_javascript = apply_filters( 'w3tc_lazyload_on_initialized_javascript', '' );
if ( 'async_head' === $method ) {
$on_initialized_javascript_wrapped = '';
if ( ! empty( $on_initialized_javascript ) ) {
// LazyLoad::Initialized fired just before making LazyLoad global so next execution cycle have it.
$on_initialized_javascript_wrapped =
'window.addEventListener("LazyLoad::Initialized", function(){' .
'setTimeout(function() {' .
$on_initialized_javascript .
'}, 1);' .
'});';
}
$embed_script =
'<style>img.lazy{min-height:1px}</style>' .
'<link href="' . esc_url( $js_url ) . '" as="script">';
$buffer = preg_replace(
'~<head(\s+[^>]*)*>~Ui',
'\\0' . $embed_script,
$buffer,
1
);
// load lazyload in footer to make sure DOM is ready at the moment of initialization.
$footer_script =
'<script>' .
$on_initialized_javascript_wrapped .
'window.w3tc_lazyload=1,' .
'window.lazyLoadOptions=' . $config .
'</script>' .
'<script async src="' . esc_url( $js_url ) . '"></script>';
$buffer = preg_replace(
'~</body(\s+[^>]*)*>~Ui',
$footer_script . '\\0',
$buffer,
1
);
} elseif ( 'inline_footer' === $method ) {
$footer_script =
'<style>img.lazy{min-height:1px}</style>' .
'<script>' .
file_get_contents( W3TC_DIR . '/pub/js/lazyload.min.js' ) .
'window.w3tc_lazyload=new LazyLoad(' . $config . ');' .
$on_initialized_javascript .
'</script>';
$buffer = preg_replace(
'~</body(\s+[^>]*)*>~Ui',
$footer_script . '\\0',
$buffer,
1
);
} else { // 'sync_head'
$head_script =
'<style>img.lazy{min-height:1px}</style>' .
'<script src="' . esc_url( $js_url ) . '"></script>';
$buffer = preg_replace(
'~<head(\s+[^>]*)*>~Ui',
'\\0' . $head_script,
$buffer,
1
);
$footer_script =
'<script>' .
'window.w3tc_lazyload=new LazyLoad(' . $config . ');' .
$on_initialized_javascript .
'</script>';
$buffer = preg_replace(
'~</body(\s+[^>]*)*>~Ui',
$footer_script . '\\0',
$buffer,
1
);
}
return $buffer;
}
/**
* Maps attachment URLs to their corresponding post IDs.
*
* Updates the internal posts-by-URL mapping for tracking purposes.
*
* @param string $url The attachment URL.
* @param int $post_id The post ID.
*
* @return string The unmodified attachment URL.
*/
public function wp_get_attachment_url( $url, $post_id ) {
$this->posts_by_url[ $url ] = $post_id;
return $url;
}
/**
* Adds hooks specific to MetaSlider plugin compatibility.
*
* Modifies MetaSlider content to work seamlessly with lazy loading.
*
* @return void
*/
private function metaslider_hooks() {
add_filter( 'metaslider_nivo_slider_get_html', array( $this, 'metaslider_nivo_slider_get_html' ) );
}
/**
* Filters MetaSlider HTML output for compatibility.
*
* Prevents lazy loading from interfering with MetaSlider's image loading
* by adding a `no-lazy` class to images.
*
* @param string $content The MetaSlider HTML content.
*
* @return string Modified HTML content.
*/
public function metaslider_nivo_slider_get_html( $content ) {
// nivo slider use "src" attr of <img> tags to populate
// own image via JS, i.e. cant be replaced by lazyloading.
$content = preg_replace(
'~(\s+)(class=)([\"\'])(.*?)([\"\'])~',
'$1$2$3$4 no-lazy$5',
$content
);
return $content;
}
}