brandcom / cakephp-content-blocks
ContentBlocks plugin for CakePHP
Installs: 128
Dependents: 0
Suggesters: 0
Security: 0
Stars: 0
Watchers: 1
Forks: 0
Open Issues: 0
Type:cakephp-plugin
pkg:composer/brandcom/cakephp-content-blocks
Requires
- cakephp/cakephp: ^3.5
Requires (Dev)
- phpunit/phpunit: ^5.7.14|^6.0
README
Installation
You can install this plugin into your CakePHP application using composer.
The recommended way to install composer packages is:
composer require brandcom/cakephp-content-blocks
Load the plugin:
bin/cake plugin load ContentBlocks
... and run the migrations:
bin/cake migrations migrate --plugin ContentBlocks
Getting Started
1. Create Blocks
The plugin does not come with any blocks, so you will have to create them on your own.
Let's create a simple TextContentBlock with a title and a content field.
Note: Every ContentBlock must end with
*ContentBlock
-
In your database, create a table
text_content_blockswith the fieldstitle(Varchar 255) andcontent(Text). Note: You will also need anidfield and a fieldcontent_blocks_block_idasint(11), unsigned. -
Then, run the
bin/cake bake model text_content_blockscommand. You won't need any templates or controllers. -
Edit the baked
TextContentBlock.phpfile and make the class extendContentBlocks\Model\Entity\Blockinstead ofEntity. -
Modify your
TextContentBlocksTable.phpso that yourTextContentBlocksTableextendsContentBlocks\Model\Entity\BlocksTable, and modify the class as follows:
Set the relation:
$this->belongsTo('Blocks', [
'foreignKey' => 'content_blocks_block_id',
'className' => 'ContentBlocks.Blocks',
]);
Fix the buildRules() method, set table to Blocks:
$rules->add($rules->existsIn(['content_blocks_block_id'], 'Blocks'));
You will find more on customizing your Block below, but let's now add it to one of your pages.
2. Admin interface
To add your TextContentBlock to an Entity, e.g., Article, add the BlocksAdmin cell to your edit template:
<?= $this->cell("ContentBlocks.BlocksAdmin", ['entityOrKey' => $article]) ?>
This will render a table representing the BlockArea for the respective entity.
New in version 0.3: You can also create Areas for custom keys, e.g. on other places on your site or for index pages:
<?= $this->cell("ContentBlocks.BlocksAdmin", ['entityOrKey' => 'my_custom_key']) ?>
At the bottom of the table, there will be a button for each of your ContentBlocks. You should already find a button with the title Text.
If you click on the button, a new Block will be added to the Block Area. You can enter content and save.
3. Create a template for your block
ContentBlock templates are elements. In your Template/Element folder, create the folder content_blocks/ and the file text.ctp.
The template file name is always the lower_case_underscore version of your model name, omitting the content_block.
So the template for MyCoolHeroHeaderContentBlock will be in /Template/Element/content_blocks/my_cool_hero_header.ctp.
Your TextContentBlock Entity will be available as the $block variable.
Now you can create your template:
<?php
/**
* @var \App\Model\Entity\TextContentBlock $block
*/
?>
<section class="my-12">
<h2>
<?= $block->title ?>
</h2>
<div class="content">
<?= $block->content ?>
</div>
</section>
Note: You can change the rendering logic in your
TextContentBlockEntity by overriding therender()method.
4. Display the blocks
To render your block area for an entity, add this cell to your template:
<?= $this->cell("ContentBlocks.BlocksArea", ['entity' => $article]) ?>
5. Modifying the Admin interface
If you want to change how fields are displayed, you can override the getFields() method in your TextContentBlock Entity.
The method should return an array of all editable fields with the field names as keys and an array as value which is passed to FormHelper::control as $options:
public function getFields(): array
{
return array_merge(
parent::getFields(),
[
'title' => [
'label' => __("Block Title"),
],
'style' => [
'label' => __("Choose a style for this block."),
'options' => [
'default' => __("Default Style"),
'funky' => __("Other cool Style"),
],
],
]
);
}
Special field options:
beforeControl: Will be rendered before the respective control.afterControl: Will be rendered after the respective control.
To change the hidden fields, override Block::getHiddenFields():
public function getHiddenFields(): array
{
return array_merge(
parent::getHiddenFields(),
[
'some_field',
'another_hidden_field',
]
);
}
E.g., content_blocks_block_id is hidden by default, and you may want to make it editable.
6. Containing associated Models
Define a beforeFind() method in your TextContenBlocksTable
public function beforeFind(Event $event, Query $query): Query
{
return $query->contain([
'Images',
]);
}
7. Edit related models
The plugin supports an admin interface even for related models.
HasMany relations
Say you have a SliderContentBlock with several slides. This means, you will have e.g. a SliderBlockSlide entity and
a SliderBlockSlidesTable.
- Let your Slide entity use the
ContentBlocks\Model\Entity\Traits\BelongsToBlockTrait - Contain the Slide as shown in 6. Containing associated Models
- In your
SliderContentBlock, override theBlock's methodgetManagedModels(). This should return an array of all related models which shall be editable in the admin form.
public function getManagedModels(): array
{
return [
"SliderBlockSlides,"
];
}
- To customize the appearance in the admin form, you can override the methods from the
BelongsToBlockTrait, e.g. to define a nicer title or control what fields are available for editing (similarly as in 5. Modifying the Admin interface).
8. Pass custom data to the block template
By default, the containing model of the block area is passed to the template as $owner,
so, e.g., if you have an Article entity which has a TextContentBlock, your article entity will be accessible
in the text.ctp as $owner.
You can override this - and add more view variables - by overriding the getViewVariables() method in your *ContentBlocksTable.
If you want to rename $owner and add more data, you can do that like so:
public function getViewVariables($entity): array
{
$vars = parent::getViewVariables($entity);
return [
'article' => $vars['owner'],
'random' => "This is just a string",
'someOtherVariable' => $this->getSomeOtherVariable($entity),
];
}
$entity, the instance of your *ContentBlock, will be passed to the method.
9. Allow or disallow the ContentBlock on specific Entities
If an Entity's model is listed in ContentBlock::getDisallowedEntities(), it will never be visible in the Block list.
protected function getDisallowedEntities(): array
{
return [
"Articles",
"Jobs",
];
}
You can also define a list of allowed Entities:
protected function getAllowedEntities(): array
{
return [
"BlogPosts",
];
}
Note: If a model is listed in both, it will be disallowed.
10. Block HTML-Anchors and (custom) preview-link
New in 0.2.0: Every Block has a field for an HTML-Anchor. A link to the anchor is displayed below.
To override the default route, you can define a url on your $owner entity which holds the Area, e.g. Articles.
This can be useful, if your URL uses a slug or other parameters. The $block instance will be passed to the method:
public function getContentBlocksViewUrl(Block $block): array
{
$anchor = $block->html_anchor ?: "content-block-" . $block->id;
return [
'prefix' => false,
'plugin' => false,
'controller' => 'Articles',
'action' => 'view',
$this->slug,
'#' => $anchor,
];
}
Contribution
You can contribute to this project via pull requests or issues.