Completed
Push — whitelist_exclusions ( e8d560...32f270 )
by
unknown
45:28 queued 38:35
created

class.jetpack-gutenberg.php (1 issue)

Upgrade to new PHP Analysis Engine

These results are based on our legacy PHP analysis, consider migrating to our new PHP analysis engine instead. Learn more

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 Jetpack
7
 */
8
9
/**
10
 * Wrapper function to safely register a gutenberg block type
11
 *
12
 * @param string $slug Slug of the block.
13
 * @param array  $args Arguments that are passed into register_block_type.
14
 *
15
 * @see register_block_type
16
 *
17
 * @since 6.7.0
18
 *
19
 * @return WP_Block_Type|false The registered block type on success, or false on failure.
20
 */
21
function jetpack_register_block( $slug, $args = array() ) {
22
	if ( 0 !== strpos( $slug, 'jetpack/' ) && ! strpos( $slug, '/' ) ) {
23
		_doing_it_wrong( 'jetpack_register_block', 'Prefix the block with jetpack/ ', '7.1.0' );
24
		$slug = 'jetpack/' . $slug;
25
	}
26
27
	// Checking whether block is registered to ensure it isn't registered twice.
28
	if ( Jetpack_Gutenberg::is_registered( $slug ) ) {
29
		return false;
30
	}
31
32
	return register_block_type( $slug, $args );
33
}
34
35
/**
36
 * Helper function to register a Jetpack Gutenberg plugin
37
 *
38
 * @deprecated 7.1.0 Use Jetpack_Gutenberg::set_extension_available() instead
39
 *
40
 * @param string $slug Slug of the plugin.
41
 *
42
 * @since 6.9.0
43
 *
44
 * @return void
45
 */
46
function jetpack_register_plugin( $slug ) {
47
	_deprecated_function( __FUNCTION__, '7.1', 'Jetpack_Gutenberg::set_extension_available' );
48
49
	Jetpack_Gutenberg::register_plugin( $slug );
50
}
51
52
/**
53
 * Set the reason why an extension (block or plugin) is unavailable
54
 *
55
 * @deprecated 7.1.0 Use Jetpack_Gutenberg::set_extension_unavailable() instead
56
 *
57
 * @param string $slug Slug of the block.
58
 * @param string $reason A string representation of why the extension is unavailable.
59
 *
60
 * @since 7.0.0
61
 *
62
 * @return void
63
 */
64
function jetpack_set_extension_unavailability_reason( $slug, $reason ) {
65
	_deprecated_function( __FUNCTION__, '7.1', 'Jetpack_Gutenberg::set_extension_unavailable' );
66
67
	Jetpack_Gutenberg::set_extension_unavailability_reason( $slug, $reason );
68
}
69
70
/**
71
 * General Gutenberg editor specific functionality
72
 */
73
class Jetpack_Gutenberg {
74
75
	/**
76
	 * Only these extensions can be registered. Used to control availability of beta blocks.
77
	 *
78
	 * @var array Extensions whitelist
79
	 */
80
	private static $extensions = array();
81
82
	/**
83
	 * Keeps track of the reasons why a given extension is unavailable.
84
	 *
85
	 * @var array Extensions availability information
86
	 */
87
	private static $availability = array();
88
89
	/**
90
	 * Prepend the 'jetpack/' prefix to a block name
91
	 *
92
	 * @param string $block_name The block name.
93
	 *
94
	 * @return string The prefixed block name.
95
	 */
96
	private static function prepend_block_prefix( $block_name ) {
97
		return 'jetpack/' . $block_name;
98
	}
99
100
	/**
101
	 * Remove the 'jetpack/' or jetpack-' prefix from an extension name
102
	 *
103
	 * @param string $extension_name The extension name.
104
	 *
105
	 * @return string The unprefixed extension name.
106
	 */
107
	private static function remove_extension_prefix( $extension_name ) {
108
		if ( wp_startswith( $extension_name, 'jetpack/' ) || wp_startswith( $extension_name, 'jetpack-' ) ) {
109
			return substr( $extension_name, strlen( 'jetpack/' ) );
110
		}
111
		return $extension_name;
112
	}
113
114
	/**
115
	 * Whether two arrays share at least one item
116
	 *
117
	 * @param array $a An array.
118
	 * @param array $b Another array.
119
	 *
120
	 * @return boolean True if $a and $b share at least one item
121
	 */
122
	protected static function share_items( $a, $b ) {
123
		return count( array_intersect( $a, $b ) ) > 0;
124
	}
125
126
	/**
127
	 * Register a block
128
	 *
129
	 * @deprecated 7.1.0 Use jetpack_register_block() instead
130
	 *
131
	 * @param string $slug Slug of the block.
132
	 * @param array  $args Arguments that are passed into register_block_type().
133
	 */
134
	public static function register_block( $slug, $args ) {
135
		_deprecated_function( __METHOD__, '7.1', 'jetpack_register_block' );
136
137
		jetpack_register_block( 'jetpack/' . $slug, $args );
138
	}
139
140
	/**
141
	 * Register a plugin
142
	 *
143
	 * @deprecated 7.1.0 Use Jetpack_Gutenberg::set_extension_available() instead
144
	 *
145
	 * @param string $slug Slug of the plugin.
146
	 */
147
	public static function register_plugin( $slug ) {
148
		_deprecated_function( __METHOD__, '7.1', 'Jetpack_Gutenberg::set_extension_available' );
149
150
		self::set_extension_available( $slug );
151
	}
152
153
	/**
154
	 * Register a block
155
	 *
156
	 * @deprecated 7.0.0 Use jetpack_register_block() instead
157
	 *
158
	 * @param string $slug Slug of the block.
159
	 * @param array  $args Arguments that are passed into the register_block_type.
160
	 * @param array  $availability array containing if a block is available and the reason when it is not.
161
	 */
162
	public static function register( $slug, $args, $availability ) {
163
		_deprecated_function( __METHOD__, '7.0', 'jetpack_register_block' );
164
165
		if ( isset( $availability['available'] ) && ! $availability['available'] ) {
166
			self::set_extension_unavailability_reason( $slug, $availability['unavailable_reason'] );
167
		} else {
168
			self::register_block( $slug, $args );
169
		}
170
	}
171
172
	/**
173
	 * Set a (non-block) extension as available
174
	 *
175
	 * @param string $slug Slug of the extension.
176
	 */
177
	public static function set_extension_available( $slug ) {
178
		self::$availability[ self::remove_extension_prefix( $slug ) ] = true;
179
	}
180
181
	/**
182
	 * Set the reason why an extension (block or plugin) is unavailable
183
	 *
184
	 * @param string $slug Slug of the extension.
185
	 * @param string $reason A string representation of why the extension is unavailable.
186
	 */
187
	public static function set_extension_unavailable( $slug, $reason ) {
188
		self::$availability[ self::remove_extension_prefix( $slug ) ] = $reason;
189
	}
190
191
	/**
192
	 * Set the reason why an extension (block or plugin) is unavailable
193
	 *
194
	 * @deprecated 7.1.0 Use set_extension_unavailable() instead
195
	 *
196
	 * @param string $slug Slug of the extension.
197
	 * @param string $reason A string representation of why the extension is unavailable.
198
	 */
199
	public static function set_extension_unavailability_reason( $slug, $reason ) {
200
		_deprecated_function( __METHOD__, '7.1', 'Jetpack_Gutenberg::set_extension_unavailable' );
201
202
		self::set_extension_unavailable( $slug, $reason );
203
	}
204
205
	/**
206
	 * Set up a whitelist of allowed block editor extensions
207
	 *
208
	 * @return void
209
	 */
210
	public static function init() {
211
		if ( ! self::should_load() ) {
212
			return;
213
		}
214
215
		/**
216
		 * Alternative to `JETPACK_BETA_BLOCKS`, set to `true` to load Beta Blocks.
217
		 *
218
		 * @since 6.9.0
219
		 *
220
		 * @param boolean
221
		 */
222
		if ( apply_filters( 'jetpack_load_beta_blocks', false ) ) {
223
			Jetpack_Constants::set_constant( 'JETPACK_BETA_BLOCKS', true );
224
		}
225
226
		/**
227
		 * Filter the whitelist of block editor extensions that are available through Jetpack.
228
		 *
229
		 * @since 7.0.0
230
		 *
231
		 * @param array
232
		 */
233
		self::$extensions = apply_filters( 'jetpack_set_available_extensions', self::get_available_extensions() );
234
235
		/**
236
		 * Filter the whitelist of block editor plugins that are available through Jetpack.
237
		 *
238
		 * @deprecated 7.0.0 Use jetpack_set_available_extensions instead
239
		 *
240
		 * @since 6.8.0
241
		 *
242
		 * @param array
243
		 */
244
		self::$extensions = apply_filters( 'jetpack_set_available_blocks', self::$extensions );
245
246
		/**
247
		 * Filter the whitelist of block editor plugins that are available through Jetpack.
248
		 *
249
		 * @deprecated 7.0.0 Use jetpack_set_available_extensions instead
250
		 *
251
		 * @since 6.9.0
252
		 *
253
		 * @param array
254
		 */
255
		self::$extensions = apply_filters( 'jetpack_set_available_plugins', self::$extensions );
256
	}
257
258
	/**
259
	 * Resets the class to its original state
260
	 *
261
	 * Used in unit tests
262
	 *
263
	 * @return void
264
	 */
265
	public static function reset() {
266
		self::$extensions   = array();
267
		self::$availability = array();
268
	}
269
270
	/**
271
	 * Return the Gutenberg extensions (blocks and plugins) directory
272
	 *
273
	 * @return string The Gutenberg extensions directory
274
	 */
275
	public static function get_blocks_directory() {
276
		/**
277
		 * Filter to select Gutenberg blocks directory
278
		 *
279
		 * @since 6.9.0
280
		 *
281
		 * @param string default: '_inc/blocks/'
282
		 */
283
		return apply_filters( 'jetpack_blocks_directory', '_inc/blocks/' );
284
	}
285
286
	/**
287
	 * Checks for a given .json file in the blocks folder.
288
	 *
289
	 * @param string $preset The name of the .json file to look for.
290
	 *
291
	 * @return bool True if the file is found.
292
	 */
293
	public static function preset_exists( $preset ) {
294
		return file_exists( JETPACK__PLUGIN_DIR . self::get_blocks_directory() . $preset . '.json' );
295
	}
296
297
	/**
298
	 * Decodes JSON loaded from a preset file in the blocks folder
299
	 *
300
	 * @param string $preset The name of the .json file to load.
301
	 *
302
	 * @return mixed Returns an object if the file is present, or false if a valid .json file is not present.
303
	 */
304
	public static function get_preset( $preset ) {
305
		return json_decode(
306
			// phpcs:ignore WordPress.WP.AlternativeFunctions.file_get_contents_file_get_contents
307
			file_get_contents( JETPACK__PLUGIN_DIR . self::get_blocks_directory() . $preset . '.json' )
308
		);
309
	}
310
311
	/**
312
	 * Returns a whitelist of Jetpack Gutenberg extensions (blocks and plugins), based on index.json
313
	 *
314
	 * @return array A list of blocks: eg [ 'publicize', 'markdown' ]
315
	 */
316
	public static function get_jetpack_gutenberg_extensions_whitelist() {
317
		$preset_extensions_manifest = self::preset_exists( 'index' ) ? self::get_preset( 'index' ) : (object) array();
318
319
		$preset_extensions = isset( $preset_extensions_manifest->production ) ? (array) $preset_extensions_manifest->production : array();
320
321
		if ( Jetpack_Constants::is_true( 'JETPACK_BETA_BLOCKS' ) ) {
322
			$beta_extensions = isset( $preset_extensions_manifest->beta ) ? (array) $preset_extensions_manifest->beta : array();
323
			return array_unique( array_merge( $preset_extensions, $beta_extensions ) );
324
		}
325
326
		return $preset_extensions;
327
	}
328
329
	/**
330
	 * Returns a diff from a combined list of whitelisted extensions and extensions determined to be excluded
331
	 *
332
	 * @param  array $whitelisted_extensions An array of whitelisted extensions.
0 ignored issues
show
Should the type for parameter $whitelisted_extensions not be array|null?

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

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

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

Loading history...
333
	 *
334
	 * @return array A list of blocks: eg [ 'publicize', 'markdown' ]
335
	 */
336
	public static function get_available_extensions( $whitelisted_extensions = null ) {
337
		$exclusions             = get_option( 'jetpack_extensions_exclusions', array() );
338
		$whitelisted_extensions = is_null( $whitelisted_extensions ) ? self::get_jetpack_gutenberg_extensions_whitelist() : $whitelisted_extensions;
339
340
		$combined_extension_list = array_diff( $whitelisted_extensions, $exclusions );
341
342
		return $combined_extension_list;
343
	}
344
345
	/**
346
	 * Get availability of each block / plugin.
347
	 *
348
	 * @return array A list of block and plugins and their availablity status
349
	 */
350
	public static function get_availability() {
351
		/**
352
		 * Fires before Gutenberg extensions availability is computed.
353
		 *
354
		 * In the function call you supply, use `jetpack_register_block()` to set a block as available.
355
		 * Alternatively, use `Jetpack_Gutenberg::set_extension_available()` (for a non-block plugin), and
356
		 * `Jetpack_Gutenberg::set_extension_unavailable()` (if the block or plugin should not be registered
357
		 * but marked as unavailable).
358
		 *
359
		 * @since 7.0.0
360
		 */
361
		do_action( 'jetpack_register_gutenberg_extensions' );
362
363
		$available_extensions = array();
364
365
		foreach ( self::$extensions as $extension ) {
366
			$is_available = self::is_registered( 'jetpack/' . $extension ) ||
367
			( isset( self::$availability[ $extension ] ) && true === self::$availability[ $extension ] );
368
369
			$available_extensions[ $extension ] = array(
370
				'available' => $is_available,
371
			);
372
373
			if ( ! $is_available ) {
374
				$reason = isset( self::$availability[ $extension ] ) ? self::$availability[ $extension ] : 'missing_module';
375
				$available_extensions[ $extension ]['unavailable_reason'] = $reason;
376
			}
377
		}
378
379
		return $available_extensions;
380
	}
381
382
	/**
383
	 * Check if an extension/block is already registered
384
	 *
385
	 * @since 7.2
386
	 *
387
	 * @param string $slug Name of extension/block to check.
388
	 *
389
	 * @return bool
390
	 */
391
	public static function is_registered( $slug ) {
392
		return WP_Block_Type_Registry::get_instance()->is_registered( $slug );
393
	}
394
395
	/**
396
	 * Check if Gutenberg editor is available
397
	 *
398
	 * @since 6.7.0
399
	 *
400
	 * @return bool
401
	 */
402
	public static function is_gutenberg_available() {
403
		return true;
404
	}
405
406
	/**
407
	 * Check whether conditions indicate Gutenberg Extensions (blocks and plugins) should be loaded
408
	 *
409
	 * Loading blocks and plugins is enabled by default and may be disabled via filter:
410
	 *   add_filter( 'jetpack_gutenberg', '__return_false' );
411
	 *
412
	 * @since 6.9.0
413
	 *
414
	 * @return bool
415
	 */
416
	public static function should_load() {
417
		if ( ! Jetpack::is_active() && ! Jetpack::is_development_mode() ) {
418
			return false;
419
		}
420
421
		/**
422
		 * Filter to disable Gutenberg blocks
423
		 *
424
		 * @since 6.5.0
425
		 *
426
		 * @param bool true Whether to load Gutenberg blocks
427
		 */
428
		return (bool) apply_filters( 'jetpack_gutenberg', true );
429
	}
430
431
	/**
432
	 * Only enqueue block assets when needed.
433
	 *
434
	 * @param string $type Slug of the block.
435
	 * @param array  $script_dependencies Script dependencies. Will be merged with automatically
436
	 *                                    detected script dependencies from the webpack build.
437
	 *
438
	 * @return void
439
	 */
440
	public static function load_assets_as_required( $type, $script_dependencies = array() ) {
441
		if ( is_admin() ) {
442
			// A block's view assets will not be required in wp-admin.
443
			return;
444
		}
445
446
		$type = sanitize_title_with_dashes( $type );
447
		self::load_styles_as_required( $type );
448
		self::load_scripts_as_required( $type, $script_dependencies );
449
	}
450
451
	/**
452
	 * Only enqueue block sytles when needed.
453
	 *
454
	 * @param string $type Slug of the block.
455
	 *
456
	 * @since 7.2.0
457
	 *
458
	 * @return void
459
	 */
460
	public static function load_styles_as_required( $type ) {
461
		if ( is_admin() ) {
462
			// A block's view assets will not be required in wp-admin.
463
			return;
464
		}
465
466
		// Enqueue styles.
467
		$style_relative_path = self::get_blocks_directory() . $type . '/view' . ( is_rtl() ? '.rtl' : '' ) . '.css';
468 View Code Duplication
		if ( self::block_has_asset( $style_relative_path ) ) {
469
			$style_version = self::get_asset_version( $style_relative_path );
470
			$view_style    = plugins_url( $style_relative_path, JETPACK__PLUGIN_FILE );
471
			wp_enqueue_style( 'jetpack-block-' . $type, $view_style, array(), $style_version );
472
		}
473
474
	}
475
476
	/**
477
	 * Only enqueue block scripts when needed.
478
	 *
479
	 * @param string $type Slug of the block.
480
	 * @param array  $dependencies Script dependencies. Will be merged with automatically
481
	 *                             detected script dependencies from the webpack build.
482
	 *
483
	 * @since 7.2.0
484
	 *
485
	 * @return void
486
	 */
487
	public static function load_scripts_as_required( $type, $dependencies = array() ) {
488
		if ( is_admin() ) {
489
			// A block's view assets will not be required in wp-admin.
490
			return;
491
		}
492
493
		// Enqueue script.
494
		$script_relative_path = self::get_blocks_directory() . $type . '/view.js';
495
		$script_deps_path     = JETPACK__PLUGIN_DIR . self::get_blocks_directory() . $type . '/view.deps.json';
496
497
		$script_dependencies = file_exists( $script_deps_path )
498
			? json_decode( file_get_contents( $script_deps_path ) )
499
			: array();
500
		$script_dependencies = array_merge( $script_dependencies, $dependencies, array( 'wp-polyfill' ) );
501
502 View Code Duplication
		if ( self::block_has_asset( $script_relative_path ) ) {
503
			$script_version = self::get_asset_version( $script_relative_path );
504
			$view_script    = plugins_url( $script_relative_path, JETPACK__PLUGIN_FILE );
505
			wp_enqueue_script( 'jetpack-block-' . $type, $view_script, $script_dependencies, $script_version, false );
506
		}
507
508
		wp_localize_script(
509
			'jetpack-block-' . $type,
510
			'Jetpack_Block_Assets_Base_Url',
511
			plugins_url( self::get_blocks_directory(), JETPACK__PLUGIN_FILE )
512
		);
513
	}
514
515
	/**
516
	 * Check if an asset exists for a block.
517
	 *
518
	 * @param string $file Path of the file we are looking for.
519
	 *
520
	 * @return bool $block_has_asset Does the file exist.
521
	 */
522
	public static function block_has_asset( $file ) {
523
		return file_exists( JETPACK__PLUGIN_DIR . $file );
524
	}
525
526
	/**
527
	 * Get the version number to use when loading the file. Allows us to bypass cache when developing.
528
	 *
529
	 * @param string $file Path of the file we are looking for.
530
	 *
531
	 * @return string $script_version Version number.
532
	 */
533
	public static function get_asset_version( $file ) {
534
		return Jetpack::is_development_version() && self::block_has_asset( $file )
535
			? filemtime( JETPACK__PLUGIN_DIR . $file )
536
			: JETPACK__VERSION;
537
	}
538
539
	/**
540
	 * Load Gutenberg editor assets
541
	 *
542
	 * @since 6.7.0
543
	 *
544
	 * @return void
545
	 */
546
	public static function enqueue_block_editor_assets() {
547
		if ( ! self::should_load() ) {
548
			return;
549
		}
550
551
		$rtl        = is_rtl() ? '.rtl' : '';
552
		$beta       = Jetpack_Constants::is_true( 'JETPACK_BETA_BLOCKS' ) ? '-beta' : '';
553
		$blocks_dir = self::get_blocks_directory();
554
555
		$editor_script = plugins_url( "{$blocks_dir}editor{$beta}.js", JETPACK__PLUGIN_FILE );
556
		$editor_style  = plugins_url( "{$blocks_dir}editor{$beta}{$rtl}.css", JETPACK__PLUGIN_FILE );
557
558
		$editor_deps_path = JETPACK__PLUGIN_DIR . $blocks_dir . "editor{$beta}.deps.json";
559
		$editor_deps      = file_exists( $editor_deps_path )
560
			? json_decode( file_get_contents( $editor_deps_path ) )
561
			: array();
562
		$editor_deps[]    = 'wp-polyfill';
563
564
		$version = Jetpack::is_development_version() && file_exists( JETPACK__PLUGIN_DIR . $blocks_dir . 'editor.js' )
565
			? filemtime( JETPACK__PLUGIN_DIR . $blocks_dir . 'editor.js' )
566
			: JETPACK__VERSION;
567
568 View Code Duplication
		if ( method_exists( 'Jetpack', 'build_raw_urls' ) ) {
569
			$site_fragment = Jetpack::build_raw_urls( home_url() );
570
		} elseif ( class_exists( 'WPCOM_Masterbar' ) && method_exists( 'WPCOM_Masterbar', 'get_calypso_site_slug' ) ) {
571
			$site_fragment = WPCOM_Masterbar::get_calypso_site_slug( get_current_blog_id() );
572
		} else {
573
			$site_fragment = '';
574
		}
575
576
		wp_enqueue_script(
577
			'jetpack-blocks-editor',
578
			$editor_script,
579
			$editor_deps,
580
			$version,
581
			false
582
		);
583
584
		wp_localize_script(
585
			'jetpack-blocks-editor',
586
			'Jetpack_Block_Assets_Base_Url',
587
			plugins_url( $blocks_dir . '/', JETPACK__PLUGIN_FILE )
588
		);
589
590
		wp_localize_script(
591
			'jetpack-blocks-editor',
592
			'Jetpack_Editor_Initial_State',
593
			array(
594
				'available_blocks' => self::get_availability(),
595
				'jetpack'          => array( 'is_active' => Jetpack::is_active() ),
596
				'siteFragment'     => $site_fragment,
597
			)
598
		);
599
600
		wp_set_script_translations( 'jetpack-blocks-editor', 'jetpack', plugins_url( 'languages/json', JETPACK__PLUGIN_FILE ) );
601
602
		// Adding a filter late to allow every other filter to process the path, including the CDN.
603
		add_filter( 'pre_load_script_translations', array( __CLASS__, 'filter_pre_load_script_translations' ), 1000, 3 );
604
605
		wp_enqueue_style( 'jetpack-blocks-editor', $editor_style, array(), $version );
606
	}
607
608
	/**
609
	 * A workaround for setting i18n data for WordPress client-side i18n mechanism.
610
	 * We are not yet using dotorg language packs for the editor file, so this short-circuits
611
	 * the translation loading and feeds our JSON data directly into the translation getter.
612
	 *
613
	 * @param NULL   $null     not used.
614
	 * @param String $file     the file path that is being loaded, ignored.
615
	 * @param String $handle   the script handle.
616
	 * @return NULL|String the translation data only if we're working with our handle.
617
	 */
618
	public static function filter_pre_load_script_translations( $null, $file, $handle ) {
619
		if ( 'jetpack-blocks-editor' !== $handle ) {
620
			return null;
621
		}
622
623
		return Jetpack::get_i18n_data_json();
624
	}
625
626
	/**
627
	 * Some blocks do not depend on a specific module,
628
	 * and can consequently be loaded outside of the usual modules.
629
	 * We will look for such modules in the extensions/ directory.
630
	 *
631
	 * @since 7.1.0
632
	 */
633
	public static function load_independent_blocks() {
634
		if ( self::should_load() ) {
635
			/**
636
			 * Look for files that match our list of available Jetpack Gutenberg extensions (blocks and plugins).
637
			 * If available, load them.
638
			 */
639
			foreach ( self::$extensions as $extension ) {
640
				$extension_file_glob = glob( JETPACK__PLUGIN_DIR . 'extensions/*/' . $extension . '/' . $extension . '.php' );
641
				if ( ! empty( $extension_file_glob ) ) {
642
					include_once $extension_file_glob[0];
643
				}
644
			}
645
		}
646
	}
647
}
648