2011-04-22 18:11:14 +12:00
|
|
|
# How to extend the CMS interface #
|
|
|
|
|
|
|
|
## Introduction ##
|
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
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" bar to popular pages
|
|
|
|
at the bottom of the CMS. A page can be bookmarked by a CMS author through a
|
|
|
|
simple checkbox.
|
2011-04-22 18:11:14 +12:00
|
|
|
|
2012-01-01 18:24:09 +01:00
|
|
|
For a deeper introduction to the inner workings of the CMS, please refer to our
|
|
|
|
guide on [CMS Architecture](../reference/cms-architecture).
|
|
|
|
|
2011-11-22 11:06:46 +01:00
|
|
|
## Overload a CMS template ##
|
2011-04-22 18:11:14 +12:00
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
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.
|
2011-04-22 18:11:14 +12:00
|
|
|
|
2011-11-22 11:06:46 +01:00
|
|
|
CMS templates are inherited based on their controllers, similar to subclasses of
|
2011-04-22 18:11:14 +12:00
|
|
|
the common `Page` object (a new PHP class `MyPage` will look for a `MyPage.ss` template).
|
2011-11-22 11:06:46 +01:00
|
|
|
We can use this to create a different base template with `LeftAndMain.ss`
|
|
|
|
(which corresponds to the `LeftAndMain` PHP controller class).
|
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
Copy the template markup of the base implementation at `framework/admin/templates/LeftAndMain.ss`
|
|
|
|
into `mysite/templates/LeftAndMain.ss`. It will automatically be picked up by
|
|
|
|
the CMS logic. Add a new section after the `$Content` tag:
|
2011-11-22 11:06:46 +01:00
|
|
|
|
|
|
|
:::ss
|
|
|
|
...
|
2012-11-26 17:26:06 +13:00
|
|
|
<div class="cms-container" data-layout-type="border">
|
2011-11-22 11:06:46 +01:00
|
|
|
$Menu
|
|
|
|
$Content
|
|
|
|
<div class="cms-bottom-bar south">
|
|
|
|
<ul>
|
|
|
|
<li><a href="admin/page/edit/show/1">Edit "My popular page"</a></li>
|
|
|
|
<li><a href="admin/page/edit/show/99">Edit "My other page"</a></li>
|
|
|
|
</ul>
|
2011-04-22 18:11:14 +12:00
|
|
|
</div>
|
2011-11-22 11:06:46 +01:00
|
|
|
</div>
|
|
|
|
...
|
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
Refresh the CMS interface with `admin/?flush=all`, and you should see the new
|
|
|
|
bottom bar with some hardcoded links. We'll make these dynamic further down.
|
|
|
|
|
|
|
|
You might have noticed that we didn't write any JavaScript to add our layout
|
|
|
|
manager. The important piece of information is the `south` class in our new
|
|
|
|
`<div>` structure, plus the height value in our CSS. It instructs the existing
|
|
|
|
parent layout how to render the element. This layout manager
|
|
|
|
([jLayout](http://www.bramstein.com/projects/jlayout/)) allows us to build
|
|
|
|
complex layouts with minimal JavaScript configuration.
|
|
|
|
|
|
|
|
See [layout reference](../reference/layout) for more specific information on
|
|
|
|
CMS layouting.
|
2011-11-22 11:06:46 +01:00
|
|
|
|
|
|
|
## Include custom CSS in the CMS
|
2011-04-22 18:11:14 +12:00
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
In order to show the links in one line, 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`:
|
2011-04-22 18:11:14 +12:00
|
|
|
|
2011-11-22 11:06:46 +01:00
|
|
|
:::css
|
|
|
|
.cms-bottom-bar {height: 20px; padding: 5px; background: #C6D7DF;}
|
|
|
|
.cms-bottom-bar ul {list-style: none; margin: 0; padding: 0;}
|
|
|
|
.cms-bottom-bar ul li {float: left; margin-left: 1em;}
|
|
|
|
.cms-bottom-bar a {color: #444444;}
|
2011-04-22 18:11:14 +12:00
|
|
|
|
2013-03-21 19:48:54 +01:00
|
|
|
Load the new CSS file into the CMS, by setting the `LeftAndMain.extra_requirements_css`
|
2013-04-05 15:34:29 +02:00
|
|
|
[configuration value](/topics/configuration).
|
|
|
|
|
|
|
|
:::yml
|
|
|
|
LeftAndMain:
|
|
|
|
extra_requirements_css:
|
2013-06-02 13:37:34 +12:00
|
|
|
- mysite/css/BookmarkedPages.css:
|
2011-11-22 11:06:46 +01:00
|
|
|
|
|
|
|
## Create a "bookmark" flag on pages ##
|
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
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.
|
2011-11-22 11:06:46 +01:00
|
|
|
|
|
|
|
:::php
|
|
|
|
<?php
|
2013-06-02 13:38:10 +12:00
|
|
|
|
2011-11-22 11:06:46 +01:00
|
|
|
class BookmarkedPageExtension extends DataExtension {
|
2013-06-02 13:38:10 +12:00
|
|
|
|
|
|
|
private static $db = array(
|
|
|
|
'IsBookmarked' => 'Boolean'
|
|
|
|
);
|
2012-06-28 11:43:56 +02:00
|
|
|
|
2012-07-15 17:23:10 +12:00
|
|
|
public function updateCMSFields(FieldList $fields) {
|
2011-11-22 11:06:46 +01:00
|
|
|
$fields->addFieldToTab('Root.Main',
|
|
|
|
new CheckboxField('IsBookmarked', "Show in CMS bookmarks?")
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2011-11-22 11:09:43 +01:00
|
|
|
|
2013-03-27 12:06:57 +01:00
|
|
|
Enable the extension in your [configuration file](/topics/configuration)
|
2011-11-22 11:06:46 +01:00
|
|
|
|
2013-03-27 12:06:57 +01:00
|
|
|
:::yml
|
|
|
|
SiteTree:
|
|
|
|
extensions:
|
|
|
|
- BookmarkedPageExtension
|
2011-11-22 11:09:43 +01:00
|
|
|
|
2011-11-22 11:06:46 +01:00
|
|
|
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
|
2013-06-02 13:38:10 +12:00
|
|
|
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`.
|
2011-11-22 11:09:43 +01:00
|
|
|
|
2012-12-04 10:45:23 +01:00
|
|
|
Add the following code to a new file `mysite/code/BookmarkedLeftAndMainExtension.php`;
|
2011-11-22 11:09:43 +01:00
|
|
|
|
2011-11-22 11:06:46 +01:00
|
|
|
:::php
|
|
|
|
<?php
|
2013-06-02 13:38:10 +12:00
|
|
|
|
2011-11-22 11:06:46 +01:00
|
|
|
class BookmarkedPagesLeftAndMainExtension extends LeftAndMainExtension {
|
2013-06-02 13:38:10 +12:00
|
|
|
|
2012-01-30 23:13:42 +01:00
|
|
|
public function BookmarkedPages() {
|
2012-06-23 00:32:43 +02:00
|
|
|
return Page::get()->filter("IsBookmarked", 1);
|
2011-11-22 11:06:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-27 12:06:57 +01:00
|
|
|
Enable the extension in your [configuration file](/topics/configuration)
|
2011-11-22 11:06:46 +01:00
|
|
|
|
2013-03-27 12:06:57 +01:00
|
|
|
:::yml
|
|
|
|
LeftAndMain:
|
|
|
|
extensions:
|
|
|
|
- BookmarkedPagesLeftAndMainExtension
|
2011-11-22 11:06:46 +01:00
|
|
|
|
|
|
|
As the last step, replace the hardcoded links with our list from the database.
|
2012-12-04 10:45:23 +01:00
|
|
|
Find the `<ul>` you created earlier in `mysite/admin/templates/LeftAndMain.ss`
|
2011-11-22 11:06:46 +01:00
|
|
|
and replace it with the following:
|
|
|
|
|
|
|
|
:::ss
|
|
|
|
<ul>
|
2013-04-26 11:48:59 +02:00
|
|
|
<% loop $BookmarkedPages %>
|
2012-10-08 13:11:33 +13:00
|
|
|
<li><a href="admin/pages/edit/show/$ID">Edit "$Title"</a></li>
|
2012-06-26 17:32:46 +02:00
|
|
|
<% end_loop %>
|
2011-11-22 11:06:46 +01:00
|
|
|
</ul>
|
2011-11-22 11:09:43 +01:00
|
|
|
|
2012-11-22 10:01:02 +13:00
|
|
|
## Extending the CMS actions
|
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
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.
|
2012-11-22 10:01:02 +13:00
|
|
|
|
|
|
|
The following conventions apply:
|
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
* 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.
|
2012-11-22 10:01:02 +13:00
|
|
|
* A `MajorActions` button group is already provided as a default.
|
2013-06-02 13:38:10 +12:00
|
|
|
* 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`.
|
2012-11-22 10:01:02 +13:00
|
|
|
|
|
|
|
Let's walk through a couple of examples of adding new CMS actions in `getCMSActions`.
|
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
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.
|
2012-11-22 10:01:02 +13:00
|
|
|
|
|
|
|
:::php
|
|
|
|
$fields->unshift(FormAction::create('normal', 'Normal button'));
|
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
We can affect the existing button group by manipulating the `CompositeField`
|
|
|
|
already present in the `FieldList`.
|
2012-11-22 10:01:02 +13:00
|
|
|
|
|
|
|
:::php
|
|
|
|
$fields->fieldByName('MajorActions')->push(FormAction::create('grouped', 'New group button'));
|
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
Another option is adding actions into the drop-up - best place for placing
|
|
|
|
infrequently used minor actions.
|
2012-11-22 10:01:02 +13:00
|
|
|
|
|
|
|
:::php
|
|
|
|
$fields->addFieldToTab('ActionMenus.MoreOptions', FormAction::create('minor', 'Minor action'));
|
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
We can also easily create new drop-up menus by defining new tabs within the
|
|
|
|
`TabSet`.
|
2012-11-22 10:01:02 +13:00
|
|
|
|
|
|
|
:::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>
|
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
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.
|
2012-11-22 10:01:02 +13:00
|
|
|
|
2013-06-02 13:38:10 +12:00
|
|
|
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.
|
2012-11-22 10:01:02 +13:00
|
|
|
|
2011-11-22 11:06:46 +01:00
|
|
|
## Summary
|
2011-04-22 18:11:14 +12:00
|
|
|
|
2011-11-22 11:06:46 +01:00
|
|
|
In a few lines of code, we've customized the look and feel of the CMS.
|
2013-06-02 13:38:10 +12:00
|
|
|
|
2011-11-22 11:06:46 +01:00
|
|
|
While this example is only scratching the surface, it includes most building
|
2012-01-01 18:24:09 +01:00
|
|
|
blocks and concepts for more complex extensions as well.
|
|
|
|
|
|
|
|
## Related
|
|
|
|
|
2012-11-26 17:26:06 +13:00
|
|
|
* [Reference: CMS Architecture](../reference/cms-architecture)
|
|
|
|
* [Reference: Layout](../reference/layout)
|
2012-03-25 10:16:59 +13:00
|
|
|
* [Topics: Rich Text Editing](../topics/rich-text-editing)
|
2012-11-22 10:01:02 +13:00
|
|
|
* [CMS Alternating Button](../reference/cms-alternating-button)
|