mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
e6011f3aae
Hopefully this commit can be reverted once we fix the layout manager to work with all four directions (north, south, east, west). A "bookmark bar" makes more sense as an example than having the links in the menu, and it allows us to illustrate the CMS layout techniques.
214 lines
7.3 KiB
Markdown
214 lines
7.3 KiB
Markdown
# How to extend the CMS interface #
|
|
|
|
## Introduction ##
|
|
|
|
The CMS interface works just like any other part of your website: It consists of
|
|
PHP controllers, templates, CSS stylesheets and JavaScript. Because it uses the
|
|
same base elements, it is relatively easy to extend.
|
|
|
|
As an example, we're going to add a permanent "bookmarks" link list to popular pages
|
|
into the main CMS menu. A page can be bookmarked by a CMS author through a
|
|
simple checkbox.
|
|
|
|
For a deeper introduction to the inner workings of the CMS, please refer to our
|
|
guide on [CMS Architecture](../reference/cms-architecture).
|
|
|
|
## Overload a CMS template ##
|
|
|
|
If you place a template with an identical name into your application template
|
|
directory (usually `mysite/templates/`), it'll take priority over the built-in
|
|
one.
|
|
|
|
CMS templates are inherited based on their controllers, similar to subclasses of
|
|
the common `Page` object (a new PHP class `MyPage` will look for a `MyPage.ss` template).
|
|
We can use this to create a different base template with `LeftAndMain.ss`
|
|
(which corresponds to the `LeftAndMain` PHP controller class).
|
|
|
|
Copy the template markup of the base implementation at `framework/admin/templates/Includes/LeftAndMain_Menu.ss`
|
|
into `mysite/templates/Includes/LeftAndMain_Menu.ss`. It will automatically be picked up by
|
|
the CMS logic. Add a new section into the `<ul class="cms-menu-list">`
|
|
|
|
:::ss
|
|
...
|
|
<ul class="cms-menu-list">
|
|
<!-- ... -->
|
|
<li class="bookmarked-link first">
|
|
<a href="admin/pages/edit/show/1">Edit "My popular page"</a>
|
|
</li>
|
|
<li class="bookmarked-link last">
|
|
<a href="admin/pages/edit/show/99">Edit "My other page"</a>
|
|
</li>
|
|
</ul>
|
|
...
|
|
|
|
Refresh the CMS interface with `admin/?flush=all`, and you should see those
|
|
hardcoded links underneath the left-hand menu. We'll make these dynamic further down.
|
|
|
|
## Include custom CSS in the CMS
|
|
|
|
In order to show the links a bit separated from the other menu entries,
|
|
we'll add some CSS, and get it to load
|
|
with the CMS interface. Paste the following content into a new file called
|
|
`mysite/css/BookmarkedPages.css`:
|
|
|
|
:::css
|
|
.bookmarked-link.first {margin-top: 1em;}
|
|
|
|
Load the new CSS file into the CMS, by setting the `LeftAndMain.extra_requirements_css`
|
|
[configuration value](/topics/configuration).
|
|
|
|
:::yml
|
|
LeftAndMain:
|
|
extra_requirements_css:
|
|
- mysite/css/BookmarkedPages.css:
|
|
|
|
## Create a "bookmark" flag on pages ##
|
|
|
|
Now we'll define which pages are actually bookmarked, a flag that is stored in
|
|
the database. For this we need to decorate the page record with a
|
|
`DataExtension`. Create a new file called `mysite/code/BookmarkedPageExtension.php`
|
|
and insert the following code.
|
|
|
|
:::php
|
|
<?php
|
|
|
|
class BookmarkedPageExtension extends DataExtension {
|
|
|
|
private static $db = array(
|
|
'IsBookmarked' => 'Boolean'
|
|
);
|
|
|
|
public function updateCMSFields(FieldList $fields) {
|
|
$fields->addFieldToTab('Root.Main',
|
|
new CheckboxField('IsBookmarked', "Show in CMS bookmarks?")
|
|
);
|
|
}
|
|
}
|
|
|
|
Enable the extension in your [configuration file](/topics/configuration)
|
|
|
|
:::yml
|
|
SiteTree:
|
|
extensions:
|
|
- BookmarkedPageExtension
|
|
|
|
In order to add the field to the database, run a `dev/build/?flush=all`.
|
|
Refresh the CMS, open a page for editing and you should see the new checkbox.
|
|
|
|
## Retrieve the list of bookmarks from the database
|
|
|
|
One piece in the puzzle is still missing: How do we get the list of bookmarked
|
|
pages from the database into the template we've already created (with hardcoded
|
|
links)? Again, we extend a core class: The main CMS controller called
|
|
`LeftAndMain`.
|
|
|
|
Add the following code to a new file `mysite/code/BookmarkedLeftAndMainExtension.php`;
|
|
|
|
:::php
|
|
<?php
|
|
|
|
class BookmarkedPagesLeftAndMainExtension extends LeftAndMainExtension {
|
|
|
|
public function BookmarkedPages() {
|
|
return Page::get()->filter("IsBookmarked", 1);
|
|
}
|
|
}
|
|
|
|
Enable the extension in your [configuration file](/topics/configuration)
|
|
|
|
:::yml
|
|
LeftAndMain:
|
|
extensions:
|
|
- BookmarkedPagesLeftAndMainExtension
|
|
|
|
As the last step, replace the hardcoded links with our list from the database.
|
|
Find the `<ul>` you created earlier in `mysite/admin/templates/LeftAndMain.ss`
|
|
and replace it with the following:
|
|
|
|
:::ss
|
|
<ul class="cms-menu-list">
|
|
<!-- ... -->
|
|
<% loop $BookmarkedPages %>
|
|
<li class="bookmarked-link $FirstLast">
|
|
<li><a href="admin/pages/edit/show/$ID">Edit "$Title"</a></li>
|
|
</li>
|
|
<% end_loop %>
|
|
</ul>
|
|
|
|
## Extending the CMS actions
|
|
|
|
CMS actions follow a principle similar to the CMS fields: they are built in the
|
|
backend with the help of `FormFields` and `FormActions`, and the frontend is
|
|
responsible for applying a consistent styling.
|
|
|
|
The following conventions apply:
|
|
|
|
* New actions can be added by redefining `getCMSActions`, or adding an extension
|
|
with `updateCMSActions`.
|
|
* It is required the actions are contained in a `FieldSet` (`getCMSActions`
|
|
returns this already).
|
|
* Standalone buttons are created by adding a top-level `FormAction` (no such
|
|
button is added by default).
|
|
* Button groups are created by adding a top-level `CompositeField` with
|
|
`FormActions` in it.
|
|
* A `MajorActions` button group is already provided as a default.
|
|
* Drop ups with additional actions that appear as links are created via a
|
|
`TabSet` and `Tabs` with `FormActions` inside.
|
|
* A `ActionMenus.MoreOptions` tab is already provided as a default and contains
|
|
some minor actions.
|
|
* You can override the actions completely by providing your own
|
|
`getAllCMSFields`.
|
|
|
|
Let's walk through a couple of examples of adding new CMS actions in `getCMSActions`.
|
|
|
|
First of all we can add a regular standalone button anywhere in the set. Here
|
|
we are inserting it in the front of all other actions. We could also add a
|
|
button group (`CompositeField`) in a similar fashion.
|
|
|
|
:::php
|
|
$fields->unshift(FormAction::create('normal', 'Normal button'));
|
|
|
|
We can affect the existing button group by manipulating the `CompositeField`
|
|
already present in the `FieldList`.
|
|
|
|
:::php
|
|
$fields->fieldByName('MajorActions')->push(FormAction::create('grouped', 'New group button'));
|
|
|
|
Another option is adding actions into the drop-up - best place for placing
|
|
infrequently used minor actions.
|
|
|
|
:::php
|
|
$fields->addFieldToTab('ActionMenus.MoreOptions', FormAction::create('minor', 'Minor action'));
|
|
|
|
We can also easily create new drop-up menus by defining new tabs within the
|
|
`TabSet`.
|
|
|
|
:::php
|
|
$fields->addFieldToTab('ActionMenus.MyDropUp', FormAction::create('minor', 'Minor action in a new drop-up'));
|
|
|
|
<div class="hint" markdown='1'>
|
|
Empty tabs will be automatically removed from the `FieldList` to prevent clutter.
|
|
</div>
|
|
|
|
New actions will need associated controller handlers to work. You can use a
|
|
`LeftAndMainExtension` to provide one. Refer to [Controller documentation](../topics/controller)
|
|
for instructions on setting up handlers.
|
|
|
|
To make the actions more user-friendly you can also use alternating buttons as
|
|
detailed in the [CMS Alternating Button](../reference/cms-alternating-button)
|
|
how-to.
|
|
|
|
## Summary
|
|
|
|
In a few lines of code, we've customized the look and feel of the CMS.
|
|
|
|
While this example is only scratching the surface, it includes most building
|
|
blocks and concepts for more complex extensions as well.
|
|
|
|
## Related
|
|
|
|
* [Reference: CMS Architecture](../reference/cms-architecture)
|
|
* [Reference: Layout](../reference/layout)
|
|
* [Topics: Rich Text Editing](../topics/rich-text-editing)
|
|
* [CMS Alternating Button](../reference/cms-alternating-button)
|