Skip to content

Latest commit

 

History

History
482 lines (365 loc) · 14.1 KB

README.md

File metadata and controls

482 lines (365 loc) · 14.1 KB

BlockX

This plugin provides an easier way for developers to add new block types to WordPress Gutenberg Editor. You can add configuratable boxes and templates without any javascript code. Everything can be configured and rendered in PHP.

Filters

Some data filters to change blockx block behavoirs from outside.

Block attributes

Gutenberg block attributes can be changed before blockx prepares the $attributes['content'] values.

add_filter(
    Palasthotel\WordPress\BlockX\Plugin::FILTER_BLOCK_ATTRIBUTES, 
    function($attributes, $block, $originalAttributes){
        /**
         * @var array $attributes
         * @var \Palasthotel\WordPress\BlockX\Blocks\_BlockType $block
         * @var array $originalAttributes
         */
      return $attributes;
    }, 
    10, 
    3
);

Prepare contents

BlockX contents object can be changed after it was processed by the blockx classes prepare method and right before it will be provided to the template.

add_filter(
    Palasthotel\WordPress\BlockX\Plugin::FILTER_PREPARE_CONTENT, 
    function($content, $block, $originalContent){
        /**
         * @var stdClass $content
         * @var \Palasthotel\WordPress\BlockX\Blocks\_BlockType $block
         * @var stdClass $originalAttributes
         */
      return $content;
    }, 
    10, 
    3
);

Create new block class

Blocks need to extend _BlockType abstract class which comes with the interface _IBlockType that forces you to add some required functions. The minimal setup looks like this:

// MyBlock.php
namespace MyNamespace;

use Palasthotel\WordPress\BlockX\Blocks\_BlockType;
use Palasthotel\WordPress\BlockX\Model\BlockId;
use Palasthotel\WordPress\BlockX\Model\ContentStructure;

class MyBlock extends _BlockType{

    public function id(): BlockId{
        // have a look at the Block Name character restrictions
        // https://developer.wordpress.org/block-editor/developers/block-api/block-registration/#block-name
      	// the block id corresponds to the namespace and block-name parts of the first argument of the registerBlockType function
        return BlockId::build("my-namespace","my-block");
    }
    
	public function category(): string{
        // you can use the default categories
        // https://developer.wordpress.org/block-editor/developers/block-api/block-registration/#category
        // OR
        // you can add custom categories
        // https://developer.wordpress.org/block-editor/developers/filters/block-filters/#managing-block-categories
	    	return "common";
    }
  
		public function title(): string{
	    	return "My Block";
    }
    
		public function contentStructure(): ContentStructure{
      	// the ContentStructure instance will define the inputs of our block
	  		return new ContentStructure([
         		// use Palasthotel\WordPress\BlockX\Widgets\* classes here
      	]);
    }

}

There are some more optional functions that can be overwritten:

// MyBlock.php
...
class MyBlock extends _BlockType{
    ...
    public function registerBlockTypeArgs(): array{
      	$args = parent::registerBlockTypeArgs();
      	// all other arguments for the second argument of registerBlockType call can be injected via this $args object
      	// https://developer.wordpress.org/block-editor/developers/block-api/block-registration/
      	
      	$args["icon"] = "wordpress"; // https://developer.wordpress.org/resource/dashicons
        $args["description"] = "This is my custom block.";
      	$args["supports"] = [
          	"align" => true,
            "multiple" => false,
        ];
        ...
        
	      return $args;
    }
  	
  	public function prepare( stdClass $content ): stdClass {
      $content = parent::prepare($content);
      // extend $content with some more data that is important for rendering
      if($this->isEditor()){
        // extend $content with data that is important for editor preview rendering only
      } else {
        // extend $content with data that is important for frontend rendering only
      }
      return $content;
    }
}

Add a block

Use the action blockx_collect to add your custom block classes:

add_action('blockx_collect', function(Gutenberg $gutenberg){
  $gutenberg->addBlockType(new MyBlock());
});

Your brand new block will now be available in Gutenberg.

Widgets

In the namespace Palasthotel\WordPress\BlockX\Widgets there are some classes that should be used to build the array for the ContentStructure classes constructor call.

Text

Simple text input.

\Palasthotel\WordPress\BlockX\Widgets\Text::build(string $key, string $label, string $defaultValue = "")
->rows(1)
->help("Description text");

With rows larger than 1 this widget is equal to Textarea.

The help function lets you provide a description.

Textarea

Is technically the same as Text.

\Palasthotel\WordPress\BlockX\Widgets\Textarea::build(string $key, string $label, string $defaultValue = "")
->rows(10)
->help("Description text");

With rows equals 1 this widget is equal to Text.

The help function lets you provide a description.

Number

Simple integer input.

\Palasthotel\WordPress\BlockX\Widgets\Number::build(string $key, string $label, int $defaultValue = 0);

Toggle

Simple boolean input.

\Palasthotel\WordPress\BlockX\Widgets\Toggle::build(string $key, string $label, boolean $defaultValue = false);

You can additionally provide some help text.

\Palasthotel\WordPress\BlockX\Widgets\Toggle::build(string $key, string $label, boolean $defaultValue = false)
  	->help("Helping description text.");

The label as well as the help can also be configured as \Palasthotel\WordPress\BlockX\Model\StateLabel instances.

\Palasthotel\WordPress\BlockX\Widgets\Toggle::build(
		string $key, 
		StateLabel::build("This is on", "This is off"), 
		boolean $defaultValue = false
)->help(StateLabel::build("This happens when it's on", "This happens when it's off."));

Select

Choose from a list of options.

\Palasthotel\WordPress\BlockX\Widgets\Select::build(
	string $key, 
	string $label,
	[
		\Palasthotel\WordPress\BlockX\Model\Option::build($value, $label),
		...
	],
	string|null $defaultValue = null
)->multiple(true);

If $defaultValue is null the first Option will be selected when instanciating a new block of this type.

If you use multiple(true) please provide an array of strings as default value. At the moment there is a known visual bug with multiple select control.

Media

Since plugin Version: 1.0.6

Use media contents.

\Palasthotel\WordPress\BlockX\Widgets\Media::build(
	string $key, 
	string $label,
	string $defaultValue = ""
)
->multiple(true)
->setMediaTypes(["image", "audio", "application/pdf"])
->setMediaUploadTitle("My Title");

Depending on multiple set to true or false the content value of the media widget field will be a single ID or an array of IDs. Default value is false.

If setMediaTypes expects a list of media_type or mime_type strings.

You can also change the title of the media overlay with setMediaUploadTitle.

Post

Select one specific post content.

\Palasthotel\WordPress\BlockX\Widgets\Post::build(string $key, string $label, $defaultValue)
->postTypes(["post", "page"]) // default is ["post"]
->postStatus(["publish", "future"]); // default is ["publish"]

TaxonomyTerm

Choose a term of a taxonomy from a select field.

\Palasthotel\WordPress\BlockX\Widgets\TaxonomyTerm::build(string $key, string $label, WP_Taxonomy|string $taxonomy, int|null $defaultValue);

Important:

  • When using custom taxonomies they need be configured with show_in_rest => true.
  • $defaultValue needs to be a termId of the given taxonomy.

TaxQuery

For much more complex taxonomy conditions this widget provides a way to build tax_query arguments.

\Palasthotel\WordPress\BlockX\Widgets\TaxQuery::build(
	string $key,
	string $label,
	WP_Taxonomy[]|string[] $taxonomies,
	array $defaultValue = []
);

Panel

Gutenberg uses collapsible panels to group configuration sections. It's possible to use this ui pattern with the Panel Widget type.

\Palasthotel\WordPress\BlockX\Widgets\Panel::build(
  string $label, 
  $contentStructure: ContentStructure
)->opened(bool $isOpened)

Important:

  • Panels can only be used as first level ContentStructure Items. It's not possible to use a Panel in a Panel or in other Widgets that open a new level of content structure.

ListOf

ListOf wraps a list of other widgets and lets you build a list of these widgets instances.

\Palasthotel\WordPress\BlockX\Widgets\ListOf::build(
  string $label, 
  $contentStructure: ContentStructure
)

Templates

Every Block needs two templates to render pretty content in the editor and in frontend. Naming conventions are blockx__%namespace%--%block-name%.php and blockx__%namespace%--%block-name%__editor.php. So for our block example this would be blockx__my-namespace--my-block.php and blockx__my-namespace--my-block__editor.php.

Tempaltes can be overwritten in theme path your-theme/plugin-parts/* or any first level subfolder like your-theme/plugins-parts/my-blockx/*. You can also add new template paths that will be scanned on the templates search. To add a plugin template folder simply use the filter blockx_add_templates_paths like so:

add_filter('blockx_add_templates_paths', function($paths){
  $paths[] = "/my/absolute/path/to/my-plugin/templates/";
  return $paths;
})

React Editor Template

There is another way to provide content rendering of BlockX in Gutenberg Editor. This is a little more tricky than the php editor template variant but it might help preventing server performance issues and can help build better and faster user experiance.

Use the enqueueEditorAssets function to provide custom javascript files..

// MyBlock.php
...
class MyBlock extends _BlockType{
  	...
    public function enqueueEditorAssets( Dependencies $depenencies ){
      wp_enqueue_script(
        "my-blockx-components-script", 
        plugin_dir_url(__FILE__)."/my-blockx-components.js"
      );
      $dependencies->addHandle("my-blockx-components-script");
    }
  	...
}

The my-blockx-components.js file will be enqueue as a dependency of blockx.js.

// my-blockx-components.js
window.BlockXComponents = {
    ...(window.BlockXComponents || {}),
    ["my-namespace/my-block"]: ({id, content})=> <MyBlockEditorCompontent {...content} />,
};

If the BlockXComponents object holds a component for the block id this will be used in Gutenberg content view instead of server side rendering.

Important:

  • This is JSX syntax so you need to use a javascript bundler to transpile it to browser readable javascript code
  • Best practice is to use the @wordpress/scripts npm package to transpile React comoponents

Usage

After building the skeleton of the block data will be collected and used like the following example:

// MyBlock.php
use \Palasthotel\WordPress\BlockX\Model\ContentStructure;
use \Palasthotel\WordPress\BlockX\Model\Option;
use \Palasthotel\WordPress\BlockX\Model\StateLabel;
use \Palasthotel\WordPress\BlockX\Widgets\Text;
use \Palasthotel\WordPress\BlockX\Widgets\Number;
use \Palasthotel\WordPress\BlockX\Widgets\TaxQuery;
use \Palasthotel\WordPress\BlockX\Widgets\Toggle;

class MyBlock extends _BlockType{
	...
	
	// build the inputs
	public function contentStructure(): ContentStructure{
		return new ContentStructure([
			Text::build( "headline", "Headline" ),
			Number::build( "number_of_posts", "Number of posts", 5 ),
			Number::build( "offset", "Offset", 0 ),
			TaxQuery::build( "tquery", "Taxonomies", ["category", "post_tag"]),
			Toggle::build("show_autor", "Show author", false)->help(
				StateLabel::build(
					"Author name will be visible.",
					"No author name be visible."
				)
			)
		]);
	}
	
	// prepare the data to use it in templates
	public function prepare(\stdClass $content): \stdClass{
		$content = parent::prepare($content);
		
		$tax_query = [];
		// build wp tax_query with data provided by $content->tquery
		
		$content->wp_query_args = [
			"posts_per_page" => $content->number_of_posts,
			"offset" => $content->offset,
			"tax_query" => $tax_query,
			"post_type" => "my-custom-post-type",
		];
		
		return $content;
	}
}

Now the block will provide input widgets and parpare the $content data that will be delivered to the templates.

// blockx__my-namespace--my-block__editor.php
/**
 * @var MyBlock $this 
 * @var stdClass $content
 */
 echo "<h3>$content->headline</h3>";
 echo "<div>";
 echo "<strong>Number of posts:</strong> $content->number_of_posts";
 echo "<br/>";
 echo "<strong>Offset:</strong> $content->offset";
... // some more data info
 echo "</div>";

It is possible to render WP_Query data in the editor template. But beware of peformance issues.


Alternatively use React Component editor rendering.

// my-blockx-components.js
const MyBlockEditorComponent = ({headline, number_of_posts, offset})=>{
 return <>
 	<hr>{headline}</hr>
 	<div>
   	<strong>Number of posts:</strong> {number_of_posts}<br/>
   	<strong>Offset:</strong> {offset}
 	</div>
 </>
}
window.BlockXComponents = {
   ...(window.BlockXComponents || {}),
   ["my-namespace/my-block"]: ({id, content})=> <MyBlockEditorCompontent {...content} />,
};

It is possible to use wp.data and the REST API to query for posts and display an even better preview. This has better performance than the php server side rendering version but can nevertheless also lead to server performance issues.


// blockx__my-namespace--my-block.php
/**
* @var MyBlock $this 
* @var stdClass $content
*/
$query = new WP_Query($content->wp_query_args);
echo "<ul class='teaser-list'>";
while($query->have_posts()){
   $query->the_post();
   echo "<li class='teaser-item'>".get_the_title()."</li>";
}
echo "</ul>";
wp_reset_postdata();

Future

  • More Widgets
    • custom autocomplete
  • Edit content in place and with markup