Code Coverage
 
Lines
Functions and Methods
Classes and Traits
Total
35.82% covered (danger)
35.82%
168 / 469
32.56% covered (danger)
32.56%
14 / 43
CRAP
0.00% covered (danger)
0.00%
0 / 1
Jetpack_Gutenberg
36.13% covered (danger)
36.13%
168 / 465
32.56% covered (danger)
32.56%
14 / 43
8433.64
0.00% covered (danger)
0.00%
0 / 1
 is_gutenberg_version_available
87.50% covered (warning)
87.50%
21 / 24
0.00% covered (danger)
0.00%
0 / 1
8.12
 prepend_block_prefix
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 remove_extension_prefix
100.00% covered (success)
100.00%
3 / 3
100.00% covered (success)
100.00%
1 / 1
3
 share_items
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 set_extension_available
100.00% covered (success)
100.00%
2 / 2
100.00% covered (success)
100.00%
1 / 1
1
 set_extension_unavailable
90.00% covered (success)
90.00%
9 / 10
0.00% covered (danger)
0.00%
0 / 1
4.02
 init
n/a
0 / 0
n/a
0 / 0
1
 reset
100.00% covered (success)
100.00%
4 / 4
100.00% covered (success)
100.00%
1 / 1
1
 get_blocks_directory
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 preset_exists
n/a
0 / 0
n/a
0 / 0
1
 get_preset
0.00% covered (danger)
0.00%
0 / 8
0.00% covered (danger)
0.00%
0 / 1
12
 get_jetpack_gutenberg_extensions_allowed_list
100.00% covered (success)
100.00%
3 / 3
100.00% covered (success)
100.00%
1 / 1
3
 get_available_extensions
80.00% covered (warning)
80.00%
4 / 5
0.00% covered (danger)
0.00%
0 / 1
3.07
 is_registered_and_no_entry_in_availability
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
2
 is_available
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
2
 get_cached_availability
0.00% covered (danger)
0.00%
0 / 3
0.00% covered (danger)
0.00%
0 / 1
6
 get_availability
100.00% covered (success)
100.00%
13 / 13
100.00% covered (success)
100.00%
1 / 1
6
 get_extensions
75.00% covered (warning)
75.00%
6 / 8
0.00% covered (danger)
0.00%
0 / 1
4.25
 is_registered
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 is_gutenberg_available
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 should_load
100.00% covered (success)
100.00%
6 / 6
100.00% covered (success)
100.00%
1 / 1
4
 register_blocks_assets_base_url
100.00% covered (success)
100.00%
10 / 10
100.00% covered (success)
100.00%
1 / 1
3
 load_assets_as_required
90.00% covered (success)
90.00%
9 / 10
0.00% covered (danger)
0.00%
0 / 1
4.02
 load_styles_as_required
27.27% covered (danger)
27.27%
3 / 11
0.00% covered (danger)
0.00%
0 / 1
19.85
 load_scripts_as_required
31.82% covered (danger)
31.82%
7 / 22
0.00% covered (danger)
0.00%
0 / 1
28.29
 block_has_asset
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 get_asset_version
0.00% covered (danger)
0.00%
0 / 3
0.00% covered (danger)
0.00%
0 / 1
12
 enqueue_block_editor_assets
0.00% covered (danger)
0.00%
0 / 99
0.00% covered (danger)
0.00%
0 / 1
210
 load_independent_blocks
12.50% covered (danger)
12.50%
1 / 8
0.00% covered (danger)
0.00%
0 / 1
21.75
 load_block_editor_extensions
8.33% covered (danger)
8.33%
1 / 12
0.00% covered (danger)
0.00%
0 / 1
24.26
 blocks_variation
88.00% covered (warning)
88.00%
22 / 25
0.00% covered (danger)
0.00%
0 / 1
9.14
 get_extensions_preset_for_variation
40.00% covered (danger)
40.00%
6 / 15
0.00% covered (danger)
0.00%
0 / 1
17.58
 validate_block_embed_url
95.83% covered (success)
95.83%
23 / 24
0.00% covered (danger)
0.00%
0 / 1
14
 should_show_frontend_preview
0.00% covered (danger)
0.00%
0 / 5
0.00% covered (danger)
0.00%
0 / 1
12
 upgrade_nudge
0.00% covered (danger)
0.00%
0 / 6
0.00% covered (danger)
0.00%
0 / 1
2
 notice
0.00% covered (danger)
0.00%
0 / 34
0.00% covered (danger)
0.00%
0 / 1
72
 get_site_specific_features
0.00% covered (danger)
0.00%
0 / 8
0.00% covered (danger)
0.00%
0 / 1
12
 get_default_plan_for_block
0.00% covered (danger)
0.00%
0 / 5
0.00% covered (danger)
0.00%
0 / 1
6
 set_availability_for_plan
0.00% covered (danger)
0.00%
0 / 29
0.00% covered (danger)
0.00%
0 / 1
110
 get_render_callback_with_availability_check
0.00% covered (danger)
0.00%
0 / 14
0.00% covered (danger)
0.00%
0 / 1
56
 display_deprecated_block_message
0.00% covered (danger)
0.00%
0 / 9
0.00% covered (danger)
0.00%
0 / 1
20
 bypass_block_metadata_doing_it_wrong
0.00% covered (danger)
0.00%
0 / 3
0.00% covered (danger)
0.00%
0 / 1
6
 register_block_metadata_collection
25.00% covered (danger)
25.00%
2 / 8
0.00% covered (danger)
0.00%
0 / 1
6.80
 set_block_js_loading_strategy
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 get_block_js_loading_strategy
100.00% covered (success)
100.00%
7 / 7
100.00% covered (success)
100.00%
1 / 1
2
1<?php //phpcs:ignore WordPress.Files.FileName.InvalidClassFileName
2/**
3 * Handles server-side registration and use of all blocks and plugins available in Jetpack for the block editor, aka Gutenberg.
4 * Works in tandem with client-side block registration via `index.json`
5 *
6 * @package automattic/jetpack
7 */
8
9use Automattic\Jetpack\Assets;
10use Automattic\Jetpack\Blocks;
11use Automattic\Jetpack\Connection\Initial_State as Connection_Initial_State;
12use Automattic\Jetpack\Connection\Manager as Connection_Manager;
13use Automattic\Jetpack\Constants;
14use Automattic\Jetpack\Current_Plan as Jetpack_Plan;
15use Automattic\Jetpack\Modules;
16use Automattic\Jetpack\My_Jetpack\Initializer as My_Jetpack_Initializer;
17use Automattic\Jetpack\Status;
18use Automattic\Jetpack\Status\Host;
19
20if ( ! defined( 'ABSPATH' ) ) {
21    exit( 0 );
22}
23
24// phpcs:disable Universal.Files.SeparateFunctionsFromOO.Mixed -- TODO: Move the functions and such to some other file.
25
26/**
27 * General Gutenberg editor specific functionality
28 */
29class Jetpack_Gutenberg {
30
31    /**
32     * Only these extensions can be registered. Used to control availability of beta blocks.
33     *
34     * @var array|null Extensions allowed list or `null` if not initialized yet.
35     * @see static::get_extensions()
36     */
37    private static $extensions = null;
38
39    /**
40     * Keeps track of the reasons why a given extension is unavailable.
41     *
42     * @var array Extensions availability information
43     */
44    private static $availability = array();
45
46    /**
47     * A cached array of the fully processed availability data. Keeps track of
48     * reasons why an extension is unavailable or missing.
49     *
50     * @var array Extensions availability information.
51     */
52    private static $cached_availability = null;
53
54    /**
55     * Site-specific features available.
56     * Their calculation can be expensive and slow, so we're caching it for the request.
57     *
58     * @var array Site-specific features
59     */
60    private static $site_specific_features = array();
61
62    /**
63     * List of deprecated blocks.
64     *
65     * @var array List of deprecated blocks.
66     */
67    private static $deprecated_blocks = array(
68        'jetpack/revue',
69    );
70
71    /**
72     * Storing the contents of the preset file.
73     *
74     * Already been json_decode.
75     *
76     * @var null|object JSON decoded object after first usage.
77     */
78    private static $preset_cache = null;
79
80    /**
81     * Keep track of JS loading strategies for each block that needs it.
82     *
83     * @var array<string, array|bool>
84     *
85     * @since 15.0
86     */
87    private static $block_js_loading_strategies = array();
88
89    /**
90     * Check to see if a minimum version of Gutenberg is available. Because a Gutenberg version is not available in
91     * php if the Gutenberg plugin is not installed, if we know which minimum WP release has the required version we can
92     * optionally fall back to that.
93     *
94     * @param array  $version_requirements An array containing the required Gutenberg version and, if known, the WordPress version that was released with this minimum version.
95     * @param string $slug The slug of the block or plugin that has the gutenberg version requirement.
96     *
97     * @since 8.3.0
98     *
99     * @return boolean True if the version of gutenberg required by the block or plugin is available.
100     */
101    public static function is_gutenberg_version_available( $version_requirements, $slug ) {
102        global $wp_version;
103
104        // Bail if we don't at least have the gutenberg version requirement, the WP version is optional.
105        if ( empty( $version_requirements['gutenberg'] ) ) {
106            return false;
107        }
108
109        // If running a local dev build of gutenberg plugin GUTENBERG_DEVELOPMENT_MODE is set so assume correct version.
110        if ( defined( 'GUTENBERG_DEVELOPMENT_MODE' ) && GUTENBERG_DEVELOPMENT_MODE ) {
111            return true;
112        }
113
114        $version_available = false;
115
116        // If running a production build of the gutenberg plugin then GUTENBERG_VERSION is set, otherwise if WP version
117        // with required version of Gutenberg is known check that.
118        if ( defined( 'GUTENBERG_VERSION' ) ) {
119            $version_available = version_compare( GUTENBERG_VERSION, $version_requirements['gutenberg'], '>=' );
120        } elseif ( ! empty( $version_requirements['wp'] ) ) {
121            $version_available = version_compare( $wp_version, $version_requirements['wp'], '>=' );
122        }
123
124        if ( ! $version_available ) {
125            $slug = self::remove_extension_prefix( $slug );
126            self::set_extension_unavailable(
127                $slug,
128                'incorrect_gutenberg_version',
129                array(
130                    'required_feature' => $slug,
131                    'required_version' => $version_requirements,
132                    'current_version'  => array(
133                        'wp'        => $wp_version,
134                        'gutenberg' => defined( 'GUTENBERG_VERSION' ) ? GUTENBERG_VERSION : null,
135                    ),
136                )
137            );
138        }
139
140        return $version_available;
141    }
142
143    /**
144     * Prepend the 'jetpack/' prefix to a block name
145     *
146     * @param string $block_name The block name.
147     *
148     * @return string The prefixed block name.
149     */
150    private static function prepend_block_prefix( $block_name ) {
151        return 'jetpack/' . $block_name;
152    }
153
154    /**
155     * Remove the 'jetpack/' or jetpack-' prefix from an extension name
156     *
157     * @param string $extension_name The extension name.
158     *
159     * @return string The unprefixed extension name.
160     */
161    public static function remove_extension_prefix( $extension_name ) {
162        if ( str_starts_with( $extension_name, 'jetpack/' ) || str_starts_with( $extension_name, 'jetpack-' ) ) {
163            return substr( $extension_name, strlen( 'jetpack/' ) );
164        }
165        return $extension_name;
166    }
167
168    /**
169     * Whether two arrays share at least one item
170     *
171     * @param array $a An array.
172     * @param array $b Another array.
173     *
174     * @return boolean True if $a and $b share at least one item
175     */
176    protected static function share_items( $a, $b ) {
177        return array_intersect( $a, $b ) !== array();
178    }
179
180    /**
181     * Set a (non-block) extension as available
182     *
183     * @param string $slug Slug of the extension.
184     */
185    public static function set_extension_available( $slug ) {
186        $slug                        = self::remove_extension_prefix( $slug );
187        self::$availability[ $slug ] = true;
188    }
189
190    /**
191     * Set the reason why an extension (block or plugin) is unavailable
192     *
193     * @param string $slug Slug of the extension.
194     * @param string $reason A string representation of why the extension is unavailable.
195     * @param array  $details A free-form array containing more information on why the extension is unavailable.
196     */
197    public static function set_extension_unavailable( $slug, $reason, $details = array() ) {
198        if (
199            // Extensions that require a plan may be eligible for upgrades.
200            'missing_plan' === $reason
201            && (
202                /**
203                 * Filter 'jetpack_block_editor_enable_upgrade_nudge' with `true` to enable or `false`
204                 * to disable paid feature upgrade nudges in the block editor.
205                 *
206                 * When this is changed to default to `true`, you should also update `modules/memberships/class-jetpack-memberships.php`
207                 * See https://github.com/Automattic/jetpack/pull/13394#pullrequestreview-293063378
208                 *
209                 * @since 7.7.0
210                 *
211                 * @param boolean
212                 */
213                ! apply_filters( 'jetpack_block_editor_enable_upgrade_nudge', false )
214                /** This filter is documented in _inc/lib/admin-pages/class.jetpack-react-page.php */
215                || ! apply_filters( 'jetpack_show_promotions', true )
216            )
217        ) {
218            // The block editor may apply an upgrade nudge if `missing_plan` is the reason.
219            // Add a descriptive suffix to disable behavior but provide informative reason.
220            $reason .= '__nudge_disabled';
221        }
222        $slug                        = self::remove_extension_prefix( $slug );
223        self::$availability[ $slug ] = array(
224            'reason'  => $reason,
225            'details' => $details,
226        );
227    }
228
229    /**
230     * Used to initialize the class, no longer in use.
231     *
232     * @return void
233     * @deprecated 12.2 No longer needed.
234     */
235    public static function init() {
236        _deprecated_function( __METHOD__, '12.2' );
237    }
238
239    /**
240     * Resets the class to its original state
241     *
242     * Used in unit tests
243     *
244     * @return void
245     */
246    public static function reset() {
247        self::$extensions                  = null;
248        self::$availability                = array();
249        self::$cached_availability         = null;
250        self::$block_js_loading_strategies = array();
251    }
252
253    /**
254     * Return the Gutenberg extensions (blocks and plugins) directory
255     *
256     * @return string The Gutenberg extensions directory
257     */
258    public static function get_blocks_directory() {
259        /**
260         * Filter to select Gutenberg blocks directory
261         *
262         * @since 6.9.0
263         *
264         * @param string default: '_inc/blocks/'
265         */
266        return apply_filters( 'jetpack_blocks_directory', '_inc/blocks/' );
267    }
268
269    /**
270     * Checks for a given .json file in the blocks folder.
271     *
272     * @deprecated 14.3
273     *
274     * @param string $preset The name of the .json file to look for.
275     *
276     * @return bool True if the file is found.
277     */
278    public static function preset_exists( $preset ) {
279        _deprecated_function( __METHOD__, '14.3' );
280        return file_exists( JETPACK__PLUGIN_DIR . self::get_blocks_directory() . $preset . '.json' );
281    }
282
283    /**
284     * Decodes JSON loaded from the preset file in the blocks folder
285     *
286     * @since 14.3 Deprecated argument. Only one value is ever used.
287     *
288     * @param null $deprecated No longer used.
289     *
290     * @return mixed Returns an object if the file is present, or false if a valid .json file is not present.
291     */
292    public static function get_preset( $deprecated = null ) {
293        if ( $deprecated ) {
294            _deprecated_argument( __METHOD__, '14.3', 'The $preset argument is no longer needed or used.' );
295        }
296
297        if ( self::$preset_cache ) {
298            return self::$preset_cache;
299        }
300
301        self::$preset_cache = json_decode(
302        // phpcs:ignore WordPress.WP.AlternativeFunctions.file_get_contents_file_get_contents
303            file_get_contents( JETPACK__PLUGIN_DIR . self::get_blocks_directory() . 'index.json' )
304        );
305        return self::$preset_cache;
306    }
307
308    /**
309     * Returns a list of Jetpack Gutenberg extensions (blocks and plugins), based on index.json
310     *
311     * @return array A list of blocks: eg [ 'publicize', 'markdown' ]
312     */
313    public static function get_jetpack_gutenberg_extensions_allowed_list() {
314        $preset_extensions_manifest = ( defined( 'TESTING_IN_JETPACK' ) && TESTING_IN_JETPACK ) ? array() : self::get_preset();
315        $blocks_variation           = self::blocks_variation();
316
317        return self::get_extensions_preset_for_variation( $preset_extensions_manifest, $blocks_variation );
318    }
319
320    /**
321     * Returns a diff from a combined list of allowed extensions and extensions determined to be excluded
322     *
323     * @param  array $allowed_extensions An array of allowed extensions.
324     *
325     * @return array A list of blocks: eg array( 'publicize', 'markdown' )
326     */
327    public static function get_available_extensions( $allowed_extensions = null ) {
328        $exclusions         = get_option( 'jetpack_excluded_extensions', array() );
329        $allowed_extensions = $allowed_extensions === null ? self::get_jetpack_gutenberg_extensions_allowed_list() : $allowed_extensions;
330
331        // Avoid errors if option data is not as expected.
332        if ( ! is_array( $exclusions ) ) {
333            $exclusions = array();
334        }
335
336        return array_diff( $allowed_extensions, $exclusions );
337    }
338
339    /**
340     * Return true if the extension has been registered and there's nothing in the availablilty array.
341     *
342     * @param string $extension The name of the extension.
343     *
344     * @return bool whether the extension has been registered and there's nothing in the availablilty array.
345     */
346    public static function is_registered_and_no_entry_in_availability( $extension ) {
347        return self::is_registered( 'jetpack/' . $extension ) && ! isset( self::$availability[ $extension ] );
348    }
349
350    /**
351     * Return true if the extension has a true entry in the availablilty array.
352     *
353     * @param string $extension The name of the extension.
354     *
355     * @return bool whether the extension has a true entry in the availablilty array.
356     */
357    public static function is_available( $extension ) {
358        return isset( self::$availability[ $extension ] ) && true === self::$availability[ $extension ];
359    }
360
361    /**
362     * Get the availability of each block / plugin, or return the cached availability
363     * if it has already been calculated. Avoids re-registering extensions when not
364     * necessary.
365     *
366     * @return array A list of block and plugins and their availability status.
367     */
368    public static function get_cached_availability() {
369        if ( null === self::$cached_availability ) {
370            self::$cached_availability = self::get_availability();
371        }
372        return self::$cached_availability;
373    }
374
375    /**
376     * Get availability of each block / plugin.
377     *
378     * @return array A list of block and plugins and their availablity status
379     */
380    public static function get_availability() {
381        /**
382         * Fires before Gutenberg extensions availability is computed.
383         *
384         * In the function call you supply, use `Blocks::jetpack_register_block()` to set a block as available.
385         * Alternatively, use `Jetpack_Gutenberg::set_extension_available()` (for a non-block plugin), and
386         * `Jetpack_Gutenberg::set_extension_unavailable()` (if the block or plugin should not be registered
387         * but marked as unavailable).
388         *
389         * @since 7.0.0
390         */
391        do_action( 'jetpack_register_gutenberg_extensions' );
392
393        $available_extensions = array();
394
395        foreach ( static::get_extensions() as $extension ) {
396            $is_available                       = self::is_registered_and_no_entry_in_availability( $extension ) || self::is_available( $extension );
397            $available_extensions[ $extension ] = array(
398                'available' => $is_available,
399            );
400
401            if ( ! $is_available ) {
402                $reason  = isset( self::$availability[ $extension ] ) ? self::$availability[ $extension ]['reason'] : 'missing_module';
403                $details = isset( self::$availability[ $extension ] ) ? self::$availability[ $extension ]['details'] : array();
404                $available_extensions[ $extension ]['unavailable_reason'] = $reason;
405                $available_extensions[ $extension ]['details']            = $details;
406            }
407        }
408
409        return $available_extensions;
410    }
411
412    /**
413     * Return the list of extensions that are available.
414     *
415     * @since 11.9
416     *
417     * @return array A list of block and plugins and their availability status.
418     */
419    public static function get_extensions() {
420        if ( ! static::should_load() ) {
421            return array();
422        }
423
424        if ( null === self::$extensions ) {
425            /**
426             * Filter the list of block editor extensions that are available through Jetpack.
427             *
428             * @since 7.0.0
429             *
430             * @param array
431             */
432            self::$extensions = apply_filters( 'jetpack_set_available_extensions', self::get_available_extensions() );
433
434            if ( ! is_array( self::$extensions ) ) {
435                _doing_it_wrong( __METHOD__, esc_html__( 'The jetpack_set_available_extensions filter must return an array.', 'jetpack' ), '14.9' );
436                self::$extensions = array();
437            }
438        }
439
440        return self::$extensions;
441    }
442
443    /**
444     * Check if an extension/block is already registered
445     *
446     * @since 7.2
447     *
448     * @param string $slug Name of extension/block to check.
449     *
450     * @return bool
451     */
452    public static function is_registered( $slug ) {
453        return WP_Block_Type_Registry::get_instance()->is_registered( $slug );
454    }
455
456    /**
457     * Check if Gutenberg editor is available
458     *
459     * @since 6.7.0
460     *
461     * @return bool
462     */
463    public static function is_gutenberg_available() {
464        return true;
465    }
466
467    /**
468     * Check whether conditions indicate Gutenberg Extensions (blocks and plugins) should be loaded
469     *
470     * Loading blocks and plugins is enabled by default and may be disabled via filter:
471     *   add_filter( 'jetpack_gutenberg', '__return_false' );
472     *
473     * @since 6.9.0
474     *
475     * @return bool
476     */
477    public static function should_load() {
478        if ( ! Jetpack::is_connection_ready() && ! ( new Status() )->is_offline_mode() ) {
479            return false;
480        }
481
482        $return = true;
483
484        if ( ! ( new Modules() )->is_active( 'blocks' ) ) {
485            $return = false;
486        }
487
488        /**
489         * Filter to enable Gutenberg blocks.
490         *
491         * Defaults to true if (connected or in offline mode) and the Blocks module is active.
492         *
493         * @since 6.5.0
494         * @since 13.9 Filter is able to activate or deactivate Gutenberg blocks.
495         *
496         * @param bool true Whether to load Gutenberg blocks
497         */
498        return (bool) apply_filters( 'jetpack_gutenberg', $return );
499    }
500
501    /**
502     * Queue a script to set `Jetpack_Block_Assets_Base_Url`.
503     *
504     * In certain cases Webpack needs to know a base to load additional assets from.
505     * Normally it can determine that itself, but when JS concatenation is involved that tends to confuse it.
506     * We work around that by explicitly outputting a variable with the correct URL.
507     * We set that as its own "script" so we can reliably only output it once.
508     */
509    private static function register_blocks_assets_base_url() {
510        if ( ! wp_script_is( 'jetpack-blocks-assets-base-url', 'registered' ) ) {
511            // phpcs:ignore WordPress.WP.EnqueuedResourceParameters.MissingVersion -- No actual script, so no version needed.
512            wp_register_script( 'jetpack-blocks-assets-base-url', false, array(), null, array( 'in_footer' => false ) );
513            $json_encode_flags = JSON_UNESCAPED_SLASHES | JSON_HEX_TAG | JSON_HEX_AMP;
514            if ( get_option( 'blog_charset' ) === 'UTF-8' ) {
515                $json_encode_flags |= JSON_UNESCAPED_UNICODE;
516            }
517            wp_add_inline_script(
518                'jetpack-blocks-assets-base-url',
519                'var Jetpack_Block_Assets_Base_Url=' . wp_json_encode( plugins_url( self::get_blocks_directory(), JETPACK__PLUGIN_FILE ), $json_encode_flags ) . ';',
520                'before'
521            );
522        }
523    }
524
525    /**
526     * Only enqueue block assets when needed.
527     *
528     * @param string $type Slug of the block or absolute path to the block source code directory.
529     * @param array  $script_dependencies Script dependencies. Will be merged with automatically
530     *                                    detected script dependencies from the webpack build.
531     *
532     * @return void
533     */
534    public static function load_assets_as_required( $type, $script_dependencies = array() ) {
535        if ( is_admin() ) {
536            // A block's view assets will not be required in wp-admin.
537            return;
538        }
539
540        // Retrieve the feature from block.json if a path is passed.
541        if ( path_is_absolute( $type ) ) {
542            $metadata = Blocks::get_block_metadata_from_file( Blocks::get_path_to_block_metadata( $type ) );
543            $feature  = Blocks::get_block_feature_from_metadata( $metadata );
544
545            if ( ! empty( $feature ) ) {
546                $type = $feature;
547            }
548        }
549
550        $type = sanitize_title_with_dashes( $type );
551        self::load_styles_as_required( $type );
552        self::load_scripts_as_required( $type, $script_dependencies );
553    }
554
555    /**
556     * Only enqueue block sytles when needed.
557     *
558     * @param string $type Slug of the block.
559     *
560     * @since 7.2.0
561     *
562     * @return void
563     */
564    public static function load_styles_as_required( $type ) {
565        if ( is_admin() ) {
566            // A block's view assets will not be required in wp-admin.
567            return;
568        }
569
570        // Enqueue styles.
571        $style_relative_path = self::get_blocks_directory() . $type . '/view' . ( is_rtl() ? '.rtl' : '' ) . '.css';
572        if ( self::block_has_asset( $style_relative_path ) ) {
573            $style_version = self::get_asset_version( $style_relative_path );
574            $view_style    = plugins_url( $style_relative_path, JETPACK__PLUGIN_FILE );
575            $view_style    = add_query_arg( 'minify', 'false', $view_style );
576
577            // If this is a customizer preview, render the style directly to the preview after autosave.
578            // phpcs:ignore WordPress.Security.NonceVerification.Recommended
579            if ( is_customize_preview() && ! empty( $_GET['customize_autosaved'] ) ) {
580                // phpcs:ignore WordPress.WP.EnqueuedResources.NonEnqueuedStylesheet
581                echo '<link rel="stylesheet" id="jetpack-block-' . esc_attr( $type ) . '" href="' . esc_attr( $view_style ) . '&amp;ver=' . esc_attr( $style_version ) . '" media="all">';
582            } else {
583                wp_enqueue_style( 'jetpack-block-' . $type, $view_style, array(), $style_version );
584                wp_style_add_data( 'jetpack-block-' . $type, 'path', JETPACK__PLUGIN_DIR . $style_relative_path );
585            }
586        }
587    }
588
589    /**
590     * Only enqueue block scripts when needed.
591     *
592     * @param string $type Slug of the block.
593     * @param array  $script_dependencies Script dependencies. Will be merged with automatically
594     *                             detected script dependencies from the webpack build.
595     *
596     * @since 7.2.0
597     *
598     * @return void
599     */
600    public static function load_scripts_as_required( $type, $script_dependencies = array() ) {
601        if ( is_admin() ) {
602            // A block's view assets will not be required in wp-admin.
603            return;
604        }
605
606        self::register_blocks_assets_base_url();
607
608        // Enqueue script.
609        $script_relative_path  = self::get_blocks_directory() . $type . '/view.js';
610        $script_deps_path      = JETPACK__PLUGIN_DIR . self::get_blocks_directory() . $type . '/view.asset.php';
611        $script_dependencies[] = 'jetpack-blocks-assets-base-url';
612        if ( file_exists( $script_deps_path ) ) {
613            $asset_manifest      = include $script_deps_path;
614            $script_dependencies = array_unique( array_merge( $script_dependencies, $asset_manifest['dependencies'] ) );
615        }
616
617        if ( ! Blocks::is_amp_request() && self::block_has_asset( $script_relative_path ) ) {
618            $script_version = self::get_asset_version( $script_relative_path );
619            $view_script    = plugins_url( $script_relative_path, JETPACK__PLUGIN_FILE );
620            $view_script    = add_query_arg( 'minify', 'false', $view_script );
621            $strategy       = self::get_block_js_loading_strategy( $type );
622
623            // Enqueue dependencies.
624            wp_enqueue_script( 'jetpack-block-' . $type, $view_script, $script_dependencies, $script_version, $strategy );
625
626            // If this is a customizer preview, enqueue the dependencies and render the script directly to the preview after autosave.
627            // phpcs:ignore WordPress.Security.NonceVerification.Recommended
628            if ( is_customize_preview() && ! empty( $_GET['customize_autosaved'] ) ) {
629                // The Map block is dependent on wp-element, and it doesn't appear to to be possible to load
630                // this dynamically into the customizer iframe currently.
631                if ( 'map' === $type ) {
632                    echo '<div>' . esc_html__( 'No map preview available. Publish and refresh to see this widget.', 'jetpack' ) . '</div>';
633                    echo '<script>';
634                    echo 'Array.from(document.getElementsByClassName(\'wp-block-jetpack-map\')).forEach(function(element){element.style.display = \'none\';})';
635                    echo '</script>';
636                } else {
637                    echo '<script id="jetpack-block-' . esc_attr( $type ) . '" src="' . esc_attr( $view_script ) . '&amp;ver=' . esc_attr( $script_version ) . '"></script>';
638                }
639            }
640        }
641    }
642
643    /**
644     * Check if an asset exists for a block.
645     *
646     * @param string $file Path of the file we are looking for.
647     *
648     * @return bool $block_has_asset Does the file exist.
649     */
650    public static function block_has_asset( $file ) {
651        return file_exists( JETPACK__PLUGIN_DIR . $file );
652    }
653
654    /**
655     * Get the version number to use when loading the file. Allows us to bypass cache when developing.
656     *
657     * @param string $file Path of the file we are looking for.
658     *
659     * @return string $script_version Version number.
660     */
661    public static function get_asset_version( $file ) {
662        return Jetpack::is_development_version() && self::block_has_asset( $file )
663            ? filemtime( JETPACK__PLUGIN_DIR . $file )
664            : JETPACK__VERSION;
665    }
666
667    /**
668     * Load Gutenberg editor assets
669     *
670     * @since 6.7.0
671     *
672     * @return void
673     */
674    public static function enqueue_block_editor_assets() {
675        if ( ! self::should_load() ) {
676            return;
677        }
678
679        $status = new Status();
680
681        // Required for Analytics. See _inc/lib/admin-pages/class.jetpack-admin-page.php.
682        if ( ! $status->is_offline_mode() && Jetpack::is_connection_ready() ) {
683            wp_enqueue_script( 'jp-tracks', '//stats.wp.com/w.js', array(), gmdate( 'YW' ), true );
684        }
685
686        $blocks_dir       = self::get_blocks_directory();
687        $blocks_variation = self::blocks_variation();
688
689        if ( 'production' !== $blocks_variation ) {
690            $blocks_env = '-' . esc_attr( $blocks_variation );
691        } else {
692            $blocks_env = '';
693        }
694
695        self::register_blocks_assets_base_url();
696
697        Assets::register_script(
698            'jetpack-blocks-editor',
699            "{$blocks_dir}editor{$blocks_env}.js",
700            JETPACK__PLUGIN_FILE,
701            array(
702                'textdomain'   => 'jetpack',
703                'dependencies' => array( 'jetpack-blocks-assets-base-url' ),
704            )
705        );
706
707        /**
708         * This can be called multiple times per page load in the admin, during the `enqueue_block_assets` action.
709         * These assets are necessary for the admin for editing but are not necessary for each pattern preview.
710         * Therefore we dequeue them, so they don't load for each pattern preview iframe.
711         */
712        if ( ! wp_should_load_block_editor_scripts_and_styles() ) {
713            wp_dequeue_script( 'jp-tracks' );
714            wp_dequeue_script( 'jetpack-blocks-editor' );
715
716            return;
717        }
718
719        // Hack around #20357 (specifically, that the editor bundle depends on
720        // wp-edit-post but wp-edit-post's styles break the Widget Editor and
721        // Site Editor) until a real fix gets unblocked.
722        // @todo Remove this once #20357 is properly fixed.
723        $wp_styles_fix = wp_styles()->query( 'jetpack-blocks-editor', 'registered' );
724        if ( empty( $wp_styles_fix ) ) {
725            wp_die( 'Your installation of Jetpack is incomplete. Please run "jetpack build plugins/jetpack" in your dev env.' );
726        }
727        wp_styles()->query( 'jetpack-blocks-editor', 'registered' )->deps = array();
728
729        Assets::enqueue_script( 'jetpack-blocks-editor' );
730
731        if ( defined( 'IS_WPCOM' ) && IS_WPCOM ) {
732            $user                      = wp_get_current_user();
733            $user_data                 = array(
734                'email'    => $user->user_email,
735                'userid'   => $user->ID,
736                'username' => $user->user_login,
737            );
738            $blog_id                   = get_current_blog_id();
739            $is_current_user_connected = true;
740        } else {
741            $user_data                 = Jetpack_Tracks_Client::get_connected_user_tracks_identity();
742            $blog_id                   = Jetpack_Options::get_option( 'id', 0 );
743            $is_current_user_connected = ( new Connection_Manager( 'jetpack' ) )->is_user_connected();
744        }
745
746        if ( $blocks_variation === 'beta' && $is_current_user_connected ) {
747            wp_enqueue_style( 'recoleta-font', '//s1.wp.com/i/fonts/recoleta/css/400.min.css', array(), Constants::get_constant( 'JETPACK__VERSION' ) );
748        }
749        // AI Assistant
750        $ai_assistant_state = array(
751            'is-enabled' => apply_filters( 'jetpack_ai_enabled', true ),
752        );
753
754        $screen_base = null;
755        if ( function_exists( 'get_current_screen' ) ) {
756            $current_screen = get_current_screen();
757            $screen_base    = $current_screen ? $current_screen->base : null;
758        }
759
760        $modules = array();
761        if ( class_exists( 'Jetpack_Core_API_Module_List_Endpoint' ) ) {
762            $module_list_endpoint = new Jetpack_Core_API_Module_List_Endpoint();
763            $modules              = $module_list_endpoint->get_modules();
764        }
765
766        $jetpack_plan  = Jetpack_Plan::get();
767        $initial_state = array(
768            'available_blocks'        => self::get_availability(),
769            'blocks_variation'        => $blocks_variation,
770            'modules'                 => $modules,
771            'jetpack'                 => array(
772                'is_active'                     => Jetpack::is_connection_ready(),
773                'is_current_user_connected'     => $is_current_user_connected,
774                /** This filter is documented in class.jetpack-gutenberg.php */
775                'enable_upgrade_nudge'          => apply_filters( 'jetpack_block_editor_enable_upgrade_nudge', false ),
776                'is_private_site'               => $status->is_private_site(),
777                'is_coming_soon'                => $status->is_coming_soon(),
778                'is_offline_mode'               => $status->is_offline_mode(),
779                'is_newsletter_feature_enabled' => class_exists( '\Jetpack_Memberships' ),
780                // this is the equivalent of JP initial state siteData.showMyJetpack (class-jetpack-redux-state-helper)
781                // used to determine if we can link to My Jetpack from the block editor
782                'is_my_jetpack_available'       => My_Jetpack_Initializer::should_initialize(),
783                'jetpack_plan'                  => array(
784                    'data' => $jetpack_plan['product_slug'],
785                ),
786                /**
787                 * Enable the RePublicize UI in the block editor context.
788                 *
789                 * @module publicize
790                 *
791                 * @since 10.3.0
792                 * @deprecated 11.5 This is a feature flag that is no longer used.
793                 *
794                 * @param bool true Enable the RePublicize UI in the block editor context. Defaults to true.
795                 */
796                'republicize_enabled'           => apply_filters( 'jetpack_block_editor_republicize_feature', true ),
797            ),
798            'siteFragment'            => $status->get_site_suffix(),
799            'adminUrl'                => esc_url( admin_url() ),
800            'tracksUserData'          => $user_data,
801            'wpcomBlogId'             => $blog_id,
802            'allowedMimeTypes'        => wp_get_mime_types(),
803            'siteLocale'              => str_replace( '_', '-', get_locale() ),
804            'ai-assistant'            => $ai_assistant_state,
805            'screenBase'              => $screen_base,
806            /**
807             * Should all blocks get registered the Jetpack block collection in addition to their own categories?
808             *
809             * @since 15.3
810             *
811             * @param boolean true Enable Jetpack block collection in block categories. Defaults to true.
812             */
813            'registerBlockCollection' => apply_filters( 'jetpack_register_block_collection', true ),
814            /**
815             * Add your own feature flags to the block editor.
816             *
817             * You can access the feature flags in the block editor via hasFeatureFlag( 'your-feature-flag' ) function.
818             *
819             * @since 14.8
820             *
821             * @param array true Enable the RePublicize UI in the block editor context. Defaults to true.
822             */
823            'feature_flags'           => apply_filters( 'jetpack_block_editor_feature_flags', array() ),
824            'pluginBasePath'          => plugins_url( '', Constants::get_constant( 'JETPACK__PLUGIN_FILE' ) ),
825        );
826
827        wp_localize_script(
828            'jetpack-blocks-editor',
829            'Jetpack_Editor_Initial_State',
830            $initial_state
831        );
832
833        // Adds Connection package initial state.
834        Connection_Initial_State::render_script( 'jetpack-blocks-editor' );
835
836        // Register and enqueue the Jetpack Chrome AI token script
837        wp_register_script(
838            'jetpack-chrome-ai-token',
839            'https://widgets.wp.com/jetpack-chrome-ai/v1/3p-token.js',
840            array(),
841            gmdate( 'Ymd' ) . floor( (int) gmdate( 'G' ) / 12 ), // Cache buster: changes twice daily (morning/afternoon) in case we need to rotate the tokens
842            true
843        );
844        wp_enqueue_script( 'jetpack-chrome-ai-token' );
845    }
846
847    /**
848     * Some blocks do not depend on a specific module,
849     * and can consequently be loaded outside of the usual modules.
850     * We will look for such modules in the extensions/ directory.
851     *
852     * @since 7.1.0
853     * @see wp_common_block_scripts_and_styles()
854     */
855    public static function load_independent_blocks() {
856        if ( self::should_load() ) {
857            /**
858             * Look for files that match our list of available Jetpack Gutenberg extensions (blocks and plugins).
859             * If available, load them.
860             */
861            $directories = array( 'blocks', 'plugins', 'extended-blocks' );
862
863            foreach ( static::get_extensions() as $extension ) {
864                foreach ( $directories as $dirname ) {
865                    $path = JETPACK__PLUGIN_DIR . "extensions/{$dirname}/{$extension}/{$extension}.php";
866
867                    if ( file_exists( $path ) ) {
868                        include_once $path;
869                        continue 2;
870                    }
871                }
872            }
873        }
874    }
875
876    /**
877     * Loads PHP components of block editor extensions.
878     *
879     * @since 8.9.0
880     */
881    public static function load_block_editor_extensions() {
882        if ( self::should_load() ) {
883            // Block editor extensions to load.
884            $extensions_to_load = array(
885                'extended-blocks',
886                'plugins',
887            );
888
889            // Collect the extension paths.
890            foreach ( $extensions_to_load as $extension_to_load ) {
891                $extensions_folder = glob( JETPACK__PLUGIN_DIR . 'extensions/' . $extension_to_load . '/*' );
892
893                // Require each of the extension files, in case it exists.
894                foreach ( $extensions_folder as $extension_folder ) {
895                    $name                = basename( $extension_folder );
896                    $extension_file_path = JETPACK__PLUGIN_DIR . 'extensions/' . $extension_to_load . '/' . $name . '/' . $name . '.php';
897
898                    if ( file_exists( $extension_file_path ) ) {
899                        include_once $extension_file_path;
900                    }
901                }
902            }
903        }
904    }
905
906    /**
907     * Determine whether a site should use the default set of blocks, or a custom set.
908     * Possible variations are currently beta, experimental, and production.
909     *
910     * @since 8.1.0
911     *
912     * @return string $block_varation production|beta|experimental
913     */
914    public static function blocks_variation() {
915        // Default to production blocks.
916        $block_varation = 'production';
917
918        /*
919         * Prefer to use this JETPACK_BLOCKS_VARIATION constant
920         * or the jetpack_blocks_variation filter
921         * to set the block variation in your code.
922         */
923        $default = Constants::get_constant( 'JETPACK_BLOCKS_VARIATION' );
924        if ( ! empty( $default ) && in_array( $default, array( 'beta', 'experimental', 'production' ), true ) ) {
925            $block_varation = $default;
926        }
927
928        /**
929        * Alternative to `JETPACK_BETA_BLOCKS`, set to `true` to load Beta Blocks.
930        *
931        * @since 6.9.0
932        * @deprecated 11.8.0 Use jetpack_blocks_variation filter instead.
933        *
934        * @param boolean
935        */
936        $is_beta = apply_filters_deprecated(
937            'jetpack_load_beta_blocks',
938            array( false ),
939            'jetpack-11.8.0',
940            'jetpack_blocks_variation'
941        );
942
943        /*
944         * Switch to beta blocks if you use the JETPACK_BETA_BLOCKS constant
945         * or the deprecated jetpack_load_beta_blocks filter.
946         * This only applies when not using the newer JETPACK_BLOCKS_VARIATION constant.
947         */
948        if (
949            empty( $default )
950            && (
951                $is_beta
952                || Constants::is_true( 'JETPACK_BETA_BLOCKS' )
953            )
954        ) {
955            $block_varation = 'beta';
956        }
957
958        /**
959        * Alternative to `JETPACK_EXPERIMENTAL_BLOCKS`, set to `true` to load Experimental Blocks.
960        *
961        * @since 6.9.0
962        * @deprecated 11.8.0 Use jetpack_blocks_variation filter instead.
963        *
964        * @param boolean
965        */
966        $is_experimental = apply_filters_deprecated(
967            'jetpack_load_experimental_blocks',
968            array( false ),
969            'jetpack-11.8.0',
970            'jetpack_blocks_variation'
971        );
972
973        /*
974         * Switch to experimental blocks if you use the JETPACK_EXPERIMENTAL_BLOCKS constant
975         * or the deprecated jetpack_load_experimental_blocks filter.
976         * This only applies when not using the newer JETPACK_BLOCKS_VARIATION constant.
977         */
978        if (
979            empty( $default )
980            && (
981                $is_experimental
982                || Constants::is_true( 'JETPACK_EXPERIMENTAL_BLOCKS' )
983            )
984        ) {
985            $block_varation = 'experimental';
986        }
987
988        /**
989         * Allow customizing the variation of blocks in use on a site.
990         * Overwrites any previously set values, whether by constant or filter.
991         *
992         * @since 8.1.0
993         *
994         * @param string $block_variation Can be beta, experimental, and production. Defaults to production.
995         */
996        return apply_filters( 'jetpack_blocks_variation', $block_varation );
997    }
998
999    /**
1000     * Get a list of extensions available for the variation you chose.
1001     *
1002     * @since 8.1.0
1003     *
1004     * @param object $preset_extensions_manifest List of extensions available in Jetpack.
1005     * @param string $blocks_variation           Subset of blocks. production|beta|experimental.
1006     *
1007     * @return array $preset_extensions Array of extensions for that variation
1008     */
1009    public static function get_extensions_preset_for_variation( $preset_extensions_manifest, $blocks_variation ) {
1010        $preset_extensions = isset( $preset_extensions_manifest->{ $blocks_variation } )
1011                ? (array) $preset_extensions_manifest->{ $blocks_variation }
1012                : array();
1013
1014        /*
1015         * Experimental and Beta blocks need the production blocks as well.
1016         */
1017        if (
1018            'experimental' === $blocks_variation
1019            || 'beta' === $blocks_variation
1020        ) {
1021            $production_extensions = isset( $preset_extensions_manifest->production )
1022                ? (array) $preset_extensions_manifest->production
1023                : array();
1024
1025            $preset_extensions = array_unique( array_merge( $preset_extensions, $production_extensions ) );
1026        }
1027
1028        /*
1029         * Beta blocks need the experimental blocks as well.
1030         *
1031         * If you've chosen to see Beta blocks,
1032         * we want to make all blocks available to you:
1033         * - Production
1034         * - Experimental
1035         * - Beta
1036         */
1037        if ( 'beta' === $blocks_variation ) {
1038            $production_extensions = isset( $preset_extensions_manifest->experimental )
1039                ? (array) $preset_extensions_manifest->experimental
1040                : array();
1041
1042            $preset_extensions = array_unique( array_merge( $preset_extensions, $production_extensions ) );
1043        }
1044
1045        return $preset_extensions;
1046    }
1047
1048    /**
1049     * Validate a URL used in a SSR block.
1050     *
1051     * @since 8.3.0
1052     *
1053     * @param string $url      URL saved as an attribute in block.
1054     * @param array  $allowed  Array of allowed hosts for that block, or regexes to check against.
1055     * @param bool   $is_regex Array of regexes matching the URL that could be used in block.
1056     *
1057     * @return bool|string
1058     */
1059    public static function validate_block_embed_url( $url, $allowed = array(), $is_regex = false ) {
1060        if (
1061            empty( $url )
1062            || ! is_array( $allowed )
1063            || empty( $allowed )
1064        ) {
1065            return false;
1066        }
1067
1068        $url_components = wp_parse_url( $url );
1069
1070        // Bail early if we cannot find a host.
1071        if ( empty( $url_components['host'] ) ) {
1072            return false;
1073        }
1074
1075        // Normalize URL.
1076        $url = sprintf(
1077            '%s://%s%s%s',
1078            isset( $url_components['scheme'] ) ? $url_components['scheme'] : 'https',
1079            $url_components['host'],
1080            isset( $url_components['path'] ) ? $url_components['path'] : '/',
1081            isset( $url_components['query'] ) ? '?' . $url_components['query'] : ''
1082        );
1083
1084        if ( ! empty( $url_components['fragment'] ) ) {
1085            $url = $url . '#' . rawurlencode( $url_components['fragment'] );
1086        }
1087
1088        /*
1089         * If we're using an allowed list of hosts,
1090         * check if the URL belongs to one of the domains allowed for that block.
1091         */
1092        if (
1093            false === $is_regex
1094            && in_array( $url_components['host'], $allowed, true )
1095        ) {
1096            return $url;
1097        }
1098
1099        /*
1100         * If we are using an array of regexes to check against,
1101         * loop through that.
1102         */
1103        if ( true === $is_regex ) {
1104            foreach ( $allowed as $regex ) {
1105                if ( 1 === preg_match( $regex, $url ) ) {
1106                    return $url;
1107                }
1108            }
1109        }
1110
1111        return false;
1112    }
1113
1114    /**
1115     * Determines whether a preview of the block with an upgrade nudge should
1116     * be displayed for admins on the site frontend.
1117     *
1118     * @since 8.4.0
1119     *
1120     * @param array $availability_for_block The availability for the block.
1121     *
1122     * @return bool
1123     */
1124    public static function should_show_frontend_preview( $availability_for_block ) {
1125        return (
1126            isset( $availability_for_block['details']['required_plan'] )
1127            && current_user_can( 'manage_options' )
1128            && ! is_feed()
1129        );
1130    }
1131
1132    /**
1133     * Output an UpgradeNudge Component on the frontend of a site.
1134     *
1135     * @since 8.4.0
1136     *
1137     * @param string $plan The plan that users need to purchase to make the block work.
1138     *
1139     * @return string
1140     */
1141    public static function upgrade_nudge( $plan ) {
1142        require_once JETPACK__PLUGIN_DIR . '_inc/lib/components.php';
1143        return Jetpack_Components::render_upgrade_nudge(
1144            array(
1145                'plan' => $plan,
1146            )
1147        );
1148    }
1149
1150    /**
1151     * Output a notice within a block.
1152     *
1153     * @since 8.6.0
1154     *
1155     * @param string $message Notice we want to output.
1156     * @param string $status  Status of the notice. Can be one of success, info, warning, error. info by default.
1157     * @param string $classes List of CSS classes.
1158     *
1159     * @return string
1160     */
1161    public static function notice( $message, $status = 'info', $classes = '' ) {
1162        if (
1163            empty( $message )
1164            || ! in_array( $status, array( 'success', 'info', 'warning', 'error' ), true )
1165        ) {
1166            return '';
1167        }
1168
1169        $color = '';
1170        switch ( $status ) {
1171            case 'success':
1172                $color = '#00a32a';
1173                break;
1174            case 'warning':
1175                $color = '#dba617';
1176                break;
1177            case 'error':
1178                $color = '#d63638';
1179                break;
1180            case 'info':
1181            default:
1182                $color = '#72aee6';
1183                break;
1184        }
1185
1186        return sprintf(
1187            '<div class="jetpack-block__notice %1$s %3$s" style="border-left:5px solid %4$s;padding:1em;background-color:#f8f9f9;">%2$s</div>',
1188            esc_attr( $status ),
1189            wp_kses(
1190                $message,
1191                array(
1192                    'br' => array(),
1193                    'p'  => array(),
1194                    'a'  => array(
1195                        'href'   => array(),
1196                        'target' => array(),
1197                        'rel'    => array(),
1198                    ),
1199                )
1200            ),
1201            esc_attr( $classes ),
1202            sanitize_hex_color( $color )
1203        );
1204    }
1205
1206    /**
1207     * Retrieve site-specific features for Simple sites.
1208     *
1209     * We're caching the data for the lifetime of the request, because it can be slow to calculate,
1210     * and it can be called multiple times per single request.
1211     *
1212     * We intentionally don't use object caching or any other type of persistent caching,
1213     * in order to avoid complex cache invalidation on subscription addition or removal.
1214     *
1215     * @since 10.7
1216     *
1217     * @return array
1218     */
1219    private static function get_site_specific_features() {
1220        $current_blog_id = get_current_blog_id();
1221
1222        if ( isset( self::$site_specific_features[ $current_blog_id ] ) ) {
1223            return self::$site_specific_features[ $current_blog_id ];
1224        }
1225
1226        if ( ! class_exists( 'Store_Product_List' ) ) {
1227            require WP_CONTENT_DIR . '/admin-plugins/wpcom-billing/store-product-list.php';
1228        }
1229
1230        $site_specific_features                           = Store_Product_List::get_site_specific_features_data( $current_blog_id );
1231        self::$site_specific_features[ $current_blog_id ] = $site_specific_features;
1232
1233        return $site_specific_features;
1234    }
1235
1236    /**
1237     * Get default required plan for blocks that need a fallback when no plan is found in features data.
1238     * This ensures upgrade banners appear for blocks that are available via WPCOM_ALL_SITES/JETPACK_ALL_SITES
1239     * but should still show upgrade prompts.
1240     *
1241     * @param string $slug Block slug.
1242     * @return string|false Plan slug if a default should be used, false otherwise.
1243     */
1244    private static function get_default_plan_for_block( $slug ) {
1245        $default_plans = array(
1246            'donations'       => 'value_bundle', // Premium plan slug for WordPress.com.
1247            'payment-buttons' => 'value_bundle', // Premium plan slug for WordPress.com.
1248        );
1249
1250        return isset( $default_plans[ $slug ] ) ? $default_plans[ $slug ] : false;
1251    }
1252
1253    /**
1254     * Set the availability of the block as the editor
1255     * is loaded.
1256     *
1257     * @param string $slug Slug of the block.
1258     */
1259    public static function set_availability_for_plan( $slug ) {
1260        $slug = self::remove_extension_prefix( $slug );
1261
1262        if ( Jetpack_Plan::supports( $slug ) ) {
1263            self::set_extension_available( $slug );
1264            return;
1265        }
1266
1267        // Check what's the minimum plan where the feature is available.
1268        $plan           = '';
1269        $features_data  = array();
1270        $is_simple_site = defined( 'IS_WPCOM' ) && IS_WPCOM;
1271        $is_atomic_site = ( new Host() )->is_woa_site();
1272
1273        if ( $is_simple_site || $is_atomic_site ) {
1274            // Simple sites.
1275            if ( $is_simple_site ) {
1276                $features_data = self::get_site_specific_features();
1277            } else {
1278                // Atomic sites.
1279                $option = get_option( 'jetpack_active_plan' );
1280                if ( isset( $option['features'] ) ) {
1281                    $features_data = $option['features'];
1282                }
1283            }
1284
1285            if ( ! empty( $features_data['available'][ $slug ] ) ) {
1286                $plan = $features_data['available'][ $slug ][0];
1287            }
1288
1289            if ( empty( $plan ) ) {
1290                $default_plan = self::get_default_plan_for_block( $slug );
1291                if ( $default_plan ) {
1292                    $plan = $default_plan;
1293                }
1294            }
1295        } else {
1296            // Jetpack sites.
1297            $plan = Jetpack_Plan::get_minimum_plan_for_feature( $slug );
1298        }
1299
1300        self::set_extension_unavailable(
1301            $slug,
1302            'missing_plan',
1303            array(
1304                'required_feature' => $slug,
1305                'required_plan'    => $plan,
1306            )
1307        );
1308    }
1309
1310    /**
1311     * Wraps the suplied render_callback in a function to check
1312     * the availability of the block before rendering it.
1313     *
1314     * @param string   $slug The block slug, used to check for availability.
1315     * @param callable $render_callback The render_callback that will be called if the block is available.
1316     */
1317    public static function get_render_callback_with_availability_check( $slug, $render_callback ) {
1318        return function ( $prepared_attributes, $block_content, $block ) use ( $render_callback, $slug ) {
1319            $availability = self::get_cached_availability();
1320            $bare_slug    = self::remove_extension_prefix( $slug );
1321            if ( isset( $availability[ $bare_slug ] ) && $availability[ $bare_slug ]['available'] ) {
1322                return call_user_func( $render_callback, $prepared_attributes, $block_content, $block );
1323            }
1324
1325            // A preview of the block is rendered for admins on the frontend with an upgrade nudge.
1326            if ( isset( $availability[ $bare_slug ] ) ) {
1327                if ( self::should_show_frontend_preview( $availability[ $bare_slug ] ) ) {
1328                    $block_preview = call_user_func( $render_callback, $prepared_attributes, $block_content, $block );
1329
1330                    // If the upgrade nudge isn't already being displayed by a parent block, display the nudge.
1331                    if ( isset( $block->attributes['shouldDisplayFrontendBanner'] ) && $block->attributes['shouldDisplayFrontendBanner'] ) {
1332                        $upgrade_nudge = self::upgrade_nudge( $availability[ $bare_slug ]['details']['required_plan'] );
1333                        return $upgrade_nudge . $block_preview;
1334                    }
1335
1336                    return $block_preview;
1337                }
1338            }
1339
1340            return null;
1341        };
1342    }
1343
1344    /**
1345     * Display a message to site editors and roles above when a block is no longer supported.
1346     * This is only displayed on the frontend.
1347     *
1348     * @since 12.3
1349     *
1350     * @param string $block_content The block content.
1351     * @param array  $block         The full block, including name and attributes.
1352     *
1353     * @return string
1354     */
1355    public static function display_deprecated_block_message( $block_content, $block ) {
1356        if ( isset( $block['blockName'] ) && in_array( $block['blockName'], self::$deprecated_blocks, true ) ) {
1357            if ( current_user_can( 'edit_posts' ) ) {
1358                $block_content = self::notice(
1359                    __( 'This block is no longer supported. Its contents will no longer be displayed to your visitors and as such this block should be removed.', 'jetpack' ),
1360                    'warning',
1361                    'jetpack-block-deprecated'
1362                );
1363            } else {
1364                $block_content = '';
1365            }
1366        }
1367
1368        return $block_content;
1369    }
1370
1371    /**
1372     * Temporarily bypasses _doing_it_wrong() notices for block metadata collection registration.
1373     *
1374     * WordPress 6.7 introduced block metadata collections (with strict path validation).
1375     * Any sites using symlinks for plugins will fail the validation which causes the metadata
1376     * collection to not be registered. However, the blocks will still fall back to the regular
1377     * registration and no functionality is affected.
1378     * While this validation is being discussed in WordPress Core (#62140),
1379     * this method allows registration to proceed by temporarily disabling
1380     * the relevant notice.
1381     *
1382     * @since 14.2
1383     *
1384     * @param bool   $trigger       Whether to trigger the error.
1385     * @param string $function      The function that was called.
1386     * @param string $message       A message explaining what was done incorrectly.
1387     * @param string $version       The version of WordPress where the message was added.
1388     * @return bool Whether to trigger the error.
1389     */
1390    public static function bypass_block_metadata_doing_it_wrong( $trigger, $function, $message, $version ) { // phpcs:ignore VariableAnalysis.CodeAnalysis.VariableAnalysis.UnusedVariable
1391        if ( $function === 'WP_Block_Metadata_Registry::register_collection' ) {
1392            return false;
1393        }
1394        return $trigger;
1395    }
1396
1397    /**
1398     * Register block metadata collection for Jetpack blocks.
1399     * This allows for more efficient block metadata loading by avoiding
1400     * individual block.json file reads at runtime.
1401     *
1402     * Uses wp_register_block_metadata_collection() if available (WordPress 6.7+)
1403     * and if the manifest file exists. The manifest file is auto-generated
1404     * during the build process.
1405     *
1406     * Runs on plugins_loaded to ensure registration happens before individual
1407     * blocks register themselves on init.
1408     *
1409     * @static
1410     * @since 14.1
1411     * @return void
1412     */
1413    public static function register_block_metadata_collection() {
1414        $meta_file_path = JETPACK__PLUGIN_DIR . '_inc/blocks/blocks-manifest.php';
1415        if ( function_exists( 'wp_register_block_metadata_collection' ) && file_exists( $meta_file_path ) ) {
1416            add_filter( 'doing_it_wrong_trigger_error', array( __CLASS__, 'bypass_block_metadata_doing_it_wrong' ), 10, 4 );
1417
1418            // @phan-suppress-next-line PhanUndeclaredFunction -- New in WP 6.7. We're checking if it exists first. @phan-suppress-current-line UnusedPluginSuppression
1419            wp_register_block_metadata_collection(
1420                JETPACK__PLUGIN_DIR . '_inc/blocks/',
1421                $meta_file_path
1422            );
1423
1424            remove_filter( 'doing_it_wrong_trigger_error', array( __CLASS__, 'bypass_block_metadata_doing_it_wrong' ), 10 );
1425        }
1426    }
1427
1428    /**
1429     * Set the JS loading strategy for a block.
1430     *
1431     * @param string     $block_name The block name.
1432     * @param array|bool $strategy   The JS loading strategy.
1433     *
1434     * @since 15.0
1435     */
1436    public static function set_block_js_loading_strategy( $block_name, $strategy ) {
1437        self::$block_js_loading_strategies[ $block_name ] = $strategy;
1438    }
1439
1440    /**
1441     * Get the JS loading strategy for a block.
1442     *
1443     * @param string $block_name The block name.
1444     *
1445     * @return array|bool The JS loading strategy for the block.
1446     *
1447     * @since 15.0
1448     */
1449    public static function get_block_js_loading_strategy( $block_name ) {
1450        $strategy = array(
1451            'strategy'  => 'defer',
1452            'in_footer' => true,
1453        );
1454
1455        if ( isset( self::$block_js_loading_strategies[ $block_name ] ) ) {
1456            $strategy = self::$block_js_loading_strategies[ $block_name ];
1457        }
1458
1459        return $strategy;
1460    }
1461}
1462
1463if ( ( new Host() )->is_woa_site() ) {
1464    /**
1465    * Enable upgrade nudge for Atomic sites.
1466     * This feature is false as default,
1467     * so let's enable it through this filter.
1468     *
1469     * More doc: https://github.com/Automattic/jetpack/blob/trunk/projects/plugins/jetpack/extensions/README.md#upgrades-for-blocks
1470     */
1471    add_filter( 'jetpack_block_editor_enable_upgrade_nudge', '__return_true' );
1472}