-
Notifications
You must be signed in to change notification settings - Fork 4.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add: Simple way to register block styles. #16356
Add: Simple way to register block styles. #16356
Conversation
this should be |
Ohhh I understand the proposed API. I think I'd prefer this instead: wp_register_style( 'myguten-style', get_template_directory_uri() . '/custom-style.css' );
gutenberg_register_block_style(
'core/quote',
array(
'name' => 'fancy-quote',
'label' => 'Fancy Quote'
),
'myguten-style'
); |
I also wonder if we should consolidate the handle in the options array instead: gutenberg_register_block_style(
'core/quote',
array(
'name' => 'fancy-quote',
'label' => 'Fancy Quote',
'style' => 'myguten-style'
)
); we could also support the inline variation using the same API gutenberg_register_block_style(
'core/quote',
array(
'name' => 'fancy-quote',
'label' => 'Fancy Quote',
'inline_style' => '.wp-block-quote.is-style-not-fancy-quote { color: blue; }
)
); |
f2d105a
to
759d3aa
Compare
Hi @youknowriad the API was updated to follow your suggestion. "The How has this been tested?" section was updated with the tests done for this API. |
714a30a
to
946e0ab
Compare
lib/blocks.php
Outdated
unset( $style_properties[ 'style' ] ); | ||
add_action( | ||
'enqueue_block_assets', | ||
function() use ( $style_handle ) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is probably not compatible with php 5.2. If we keep it, we should update the minimum version of the plugin to 5.2 (which I think could be considered)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Probably upgrading minimum plugin version is the way to go. As an aside, I'm probably missing something but with PHP 5.2 the only way to express this would be using a class where we pass the $style_handle handle and then that class has a function that calls wp_enqueue_style( $this->style_handle );
or using something like create_function which was deprecated. The PHP 5.2 options that come to my mind all seem verbose.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
PR for WordPress 5.2 minimum (PHP 5.6) #15809
lib/blocks.php
Outdated
add_action( | ||
'enqueue_block_assets', | ||
function() use ( $inline_style ) { | ||
wp_add_inline_style( 'wp-block-library', $inline_style ); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What if it's not a core block?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What problem would that cause?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I mean yeah attaching to wp-block-library
seems weird. Maybe wp-block-editor
instead but I don't see any problem resulting from this.
The question is what does it mean to register a block style. Are we registering this style for all block-editor implementations in core, just the post editor..
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
(also same question about the script down here)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think using wp-block-editor would not work because these styles also need to be enqueued on the website frontend.
Ideally, these styles would be loaded after the block styles because, in case of specificity conflict, block styles would be the used ones. Using wp-block-library ensures that this styles load after the core block styles, it is true that for non-core ones that's not the case, I guess we can increase that probability by changing to a lower priority.
For non-core blocks, an option would be to use WP_Block_Type_Registry get the registered block and add use wp_add_inline_style( $block_type->style, $inline_style );
. The problem is that I verified and for most cases this style is null. I guess it is only not null if the block was registered with register_block_type PHP function and the style was passed as one of the args -- I think that this not a common way to register blocks and its styles.
Another option would be to register an inline styles handle that does not point to a file:
wp_register_style( 'wp-inline-block-styles', false );
wp_enqueue_style( 'wp-inline-block-styles' );
And then we would use wp_add_inline_style( 'wp-inline-block-styles', $inline_style );
The current option ensures that inline-block styles are loaded when wp-block-library styles are loaded if these styles are enqueued inline-block styles would also not be added, that seemed a good option.
Using a handle just for the inline-block styles also does not seems wrong, but using false for src looks a little "hacky".
Regarding the inline script to register the style, I used wp-blocks as its handle because it calls a function part of wp.blocks.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for the details, it seems like there's no perfect solution at the moment. Let's ship the first version as is and iterate.
lib/blocks.php
Outdated
* @param string $block_name Block type name including namespace. | ||
* @param array $style_properties Array containing the properties of the style name, label, style (name of the stylesheet to be enqueued), inline_style (string containing the CSS to be added). | ||
*/ | ||
function gutenberg_register_block_style( $block_name, $style_properties ) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's rename this register_block_style
because it's a function that is meant to land on Core as well.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM 👍
6bf262a
to
7c5bdf7
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
According to the WordPress PHP coding standards, closures must not be passed as filter or action callbacks, as they cannot be removed by remove_action()
/ remove_filter()
.
So if this should ever get into WordPress core, I recommend changing this code accordingly. Perhaps by using a custom class-based solution or by having this function extend WP_Block_Type
in some way.
076c33e
to
667b70e
Compare
This PR was updated and now does not uses anonymous functions, following the recommendation by @swissspidy. It is ready for a new look. |
* @param string $block_name Block type name including namespace. | ||
* @param array $style_properties Array containing the properties of the style name, label, style (name of the stylesheet to be enqueued), inline_style (string containing the CSS to be added). | ||
*/ | ||
function register_block_style( $block_name, $style_properties ) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How can someone unregister a block style using PHP, when it was registered using PHP?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good question @swissspidy. We have three options to implement unregister functionality:
1 - A unregister_block_style function that uses the global wp_filters to get the class instance used in the filter and then calls remove_action. (seems hacky but should work well).
2 - Register block style stores the WP_Block_Styles_Enqueue_Block_Assets_Action in a global array. unregister_block_style function retrieves the instance from the global array using (block name and style name) and calls remove_action.
3 - Plugin developers pass an instance of WP_Block_Styles_Enqueue_Block_Assets_Action (maybe we should find a better name in that case) to register_block_style and they can save the instance to later call unregister_block_style. This does not allow another plugin to remove a style added as they may not have access to the instance.
It seems using the current register API option 2 is the best, but using a global variable is not ideal.
Another possibility is using a new class BlockStyles with a static method to register and another to unregister. The class could follow the same approach specific in 2 but using an instance variable instead of a global variable.
Any ideas about what approach to follow? Is there an alternative approach that I'm missing and is better than the ones referred?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Another possibility is using a new class BlockStyles with a static method to register and another to unregister. The class could follow the same approach specific in 2 but using an instance variable instead of a global variable.
👍
This reminds me a lot of WP_Block_Type_Registry
, which basically does the same thing.
Now that I think about it...
Perhaps – since this is about adding some information for an existing block type - we could even re-use WP_Block_Type_Registry
here. Example (off the top of my head):
function register_block_style( $block_name, $style_properties ) {
$block = null;
if ( WP_Block_Type_Registry::is_registered( $block_name ) ) {
$block = WP_Block_Type_Registry::unregister( $block_name );
} else {
$block = new WP_Block_Type( $block_name );
}
// This property would need to be added to WP_Block_Type in core eventually.
$block->style_properties = isset( $block->style_properties ) ? $block->style_properties : array();
$block->style_properties[] = $style_properties;
WP_Block_Type_Registry::register( $block_name );
}
// This could be added directly to wp_enqueue_registered_block_scripts_and_styles()
function gutenberg_enqueue_additional_styles() {
global $current_screen;
$is_editor = ( ( $current_screen instanceof WP_Screen ) && $current_screen->is_block_editor() );
$block_registry = WP_Block_Type_Registry::get_instance();
foreach ( $block_registry->get_all_registered() as $block_name => $block_type ) {
// Front-end styles.
if ( $is_editor && ! empty( $block_type->style_properties ) ) {
foreach( $block_type->style_properties as $properties ) {
// Do all the sanity checks etc.
wp_add_inline_script( /* ... */ );
}
}
}
}
add_action( 'enqueue_block_assets', 'gutenberg_enqueue_additional_styles' );
Just thinking out loud here 🙂 Perhaps some other voices from core PHP devs would be helpful here, perhaps from @pento.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure if we should reuse WP_Block_Type_Registry as styles and blocks are not exactly the same in the client we use different structures. I agree the problem is the same. In php, it should also be possible to add styles to blocks that the server is not even aware they exist (registered with JS) I will probably use a WP_Block_Styles_Registry as a registry for styles, and follow path 2 instead of using a global I will use the styles registry.
c736397
to
7840638
Compare
686fa1d
to
6541dd5
Compare
This PR was updated to use a block styles registry following some ideas exchanged with @swissspidy, a function that allows unregistering in PHP a previously registered block in PHP was also implemented. |
Amazing work @jorgefilipecosta! 🚀 👏 |
lib/blocks.php
Outdated
) | ||
), | ||
$block_name, | ||
wp_json_encode( $client_style_properties ) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could we pass the right attributes directly array( 'key' => $values )
instead of having to unset properties as it could break if we add new properties in the future.
lib/blocks.php
Outdated
unset( $client_style_properties['inline_style'] ); | ||
} | ||
|
||
wp_add_inline_script( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I feel this should be in a separate action enqueue_block_editor_assets
that is specific to the editor (the styles are good right now though). We could also avoid wp_add_inline_script
and enqueue a separate inline script
that depends on wp-blocks
instead.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @youknowriad, I updated to use another action enqueue_block_editor_assets for the addition of the script as suggested 👍
Regarding the way to avoid wp_add_inline_script, I'm probably missing something -- I'm not seeing how we can avoid a call to wp_add_inline_script (unless printing our own inline script tags).
We can do something similar to:
wp_register_script( 'wp-block-styles', false, array( 'wp-blocks' ) );
wp_add_inline_script( 'wp-block-styles', $inline_script );
wp_enqueue_script( 'wp-block-styles' );
Registering our own script that does not point to any file, but that seems more hacky and I'm not sure if phpcs will not complain of this approach.
In this approach, we still have a call to wp_add_inline_script, was your idea different from what we have or from this approach I commented?
I performed another side update; now, all the scripts are registered together in a single function and a single inline script. This change saves some bytes on every editor page load.
For the inline styles, it is better not to apply a similar change. Inline-styles are not generated by us they are provided by the plugin/theme developers, having an invalid style in a plugin would invalid other styles.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Registering our own script that does not point to any file, but that seems more hacky and I'm not sure if phpcs will not complain of this approach.
This is exactly what I was suggesting. The advantage is that no matter if wp-blocks
is already enqueued or not, it will work (no race conditions).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't see it as crucial though.
9f49ef9
to
8acaf31
Compare
lib/blocks.php
Outdated
$inline_script = '( function() {' . "\n"; | ||
foreach ( $block_styles as $block_name => $styles ) { | ||
foreach ( $styles as $style_properties ) { | ||
$inline_script .= sprintf( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
very minor: when this pattern is needed (concat + wrap) I tend to use an array (here it would be an array of iniline scripts) and just join
and wrap at the end.
Co-Authored-By: Pascal Birchler <pascalb@google.com>
8acaf31
to
a5cce4a
Compare
Thank you for the feedback @swissspidy and @youknowriad, all the feedback was applied before the merge. If there is any other possible improvement that you notice feel free to comment and I will open a follow-up PR. |
It would be good to document this new API somewhere :) |
Added to my to-do list. I should create a follow-up PR soon :) |
This commit backports the block styles functionality added to the block editor in WordPress/gutenberg#16356. Props: youknowriad, aduth, swissspidy. Fixes #48039. git-svn-id: https://develop.svn.wordpress.org/trunk@46111 602fd350-edb4-49c9-b593-d223f7449a82
This commit backports the block styles functionality added to the block editor in WordPress/gutenberg#16356. Props: youknowriad, aduth, swissspidy. Fixes #48039. git-svn-id: https://develop.svn.wordpress.org/trunk@46111 602fd350-edb4-49c9-b593-d223f7449a82
This commit backports the block styles functionality added to the block editor in WordPress/gutenberg#16356. Props: youknowriad, aduth, swissspidy. Fixes #48039. Built from https://develop.svn.wordpress.org/trunk@46111 git-svn-id: http://core.svn.wordpress.org/trunk@45923 1a063a9b-81f0-0310-95a4-ce76da25c4cd
This commit backports the block styles functionality added to the block editor in WordPress/gutenberg#16356. Props: youknowriad, aduth, swissspidy. Fixes #48039. Built from https://develop.svn.wordpress.org/trunk@46111 git-svn-id: https://core.svn.wordpress.org/trunk@45923 1a063a9b-81f0-0310-95a4-ce76da25c4cd
This commit backports the block styles functionality added to the block editor in WordPress/gutenberg#16356. Props: youknowriad, aduth, swissspidy. Fixes #48039. git-svn-id: http://develop.svn.wordpress.org/trunk@46111 602fd350-edb4-49c9-b593-d223f7449a82
Description
Part of: #7551
This PR adds a simple mechanism for theme and plugin developers to register block styles using only PHP function calls to abstract the existing JavaScript mechanism.
How has this been tested?
I added the following code to the functions.php file of the active theme:
I created a custom-style.css file in the theme directory with the following contents:
I loaded the block editor; I verified quote block has Fancy Quote and Not Fancy Quote, and both styles are correctly applied to the editor and the frontend.