mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
reactivate api: links
added deleted link back in updated some external links
This commit is contained in:
parent
b55a5435b1
commit
d633cad9ec
@ -144,8 +144,8 @@ Sitemap.php
|
|||||||
* PHP OpenID (http://openidenabled.com/php-openid/)
|
* PHP OpenID (http://openidenabled.com/php-openid/)
|
||||||
* Browscap (http://browscap.org/)
|
* Browscap (http://browscap.org/)
|
||||||
* Plotr (http://www.solutoire.com/plotr/)
|
* Plotr (http://www.solutoire.com/plotr/)
|
||||||
* SWFUpload (http://profandesign.se/swfupload/)
|
* `SWFUpload (http://profandesign.se/swfupload/)`
|
||||||
* Improved classes
|
* Improved classes
|
||||||
* CalendarDateField
|
* CalendarDateField
|
||||||
* Refactored part of Field() into HTMLField() so it can be used in PopupDateTimeField
|
* Refactored part of Field() into HTMLField() so it can be used in PopupDateTimeField
|
||||||
* ComplexTableField
|
* ComplexTableField
|
||||||
|
@ -217,8 +217,8 @@ Right:
|
|||||||
|
|
||||||
### HTMLEditorField (TinyMCE) stores content as UTF8 instead of HTML Entities
|
### HTMLEditorField (TinyMCE) stores content as UTF8 instead of HTML Entities
|
||||||
|
|
||||||
Prior to 2.4.0, the TinyMCE JavaScript library would store a subset of special characters as HTML entities (see [TinyMCE
|
Prior to 2.4.0, the TinyMCE JavaScript library would store a subset of special characters as HTML entities (see `[TinyMCE
|
||||||
Configuration](http://wiki.moxiecode.com/index.php/TinyMCE:Configuration/entities)). SilverStripe expects UTF8 for user
|
Configuration](http://wiki.moxiecode.com/index.php/TinyMCE:Configuration/entities)`). SilverStripe expects UTF8 for user
|
||||||
input in the CMS, database storage as well as output. We have made this behaviour more consistent by converting the
|
input in the CMS, database storage as well as output. We have made this behaviour more consistent by converting the
|
||||||
TinyMCE entities back into UTF8 for database storage.
|
TinyMCE entities back into UTF8 for database storage.
|
||||||
|
|
||||||
|
@ -145,7 +145,7 @@ Affected classes and methods:
|
|||||||
|
|
||||||
### Usage of Controller::join_links() to concatenate links now mandatory
|
### Usage of Controller::join_links() to concatenate links now mandatory
|
||||||
|
|
||||||
The `[api:Controller::join_links()]` method
|
The [api:Controller::join_links()] method
|
||||||
to create links within SilverStripe controllers is now mandatory. This method ensures that links with existing GET
|
to create links within SilverStripe controllers is now mandatory. This method ensures that links with existing GET
|
||||||
parameters don't break through string concatenation.
|
parameters don't break through string concatenation.
|
||||||
|
|
||||||
@ -177,9 +177,9 @@ Full controller example:
|
|||||||
|
|
||||||
|
|
||||||
Using this method is particularly important for any custom
|
Using this method is particularly important for any custom
|
||||||
`[api:TableListField]` or
|
[api:TableListField] or
|
||||||
`[api:ComplexTableField]` subclasses and any
|
[api:ComplexTableField] subclasses and any
|
||||||
`[api:LeftAndMain]` subclass for the CMS UI. These classes in
|
[api:LeftAndMain] subclass for the CMS UI. These classes in
|
||||||
particular were refactored to secure destructive links against Cross Site Request Forgery (CSRF). This is achieved via a
|
particular were refactored to secure destructive links against Cross Site Request Forgery (CSRF). This is achieved via a
|
||||||
mandatory "SecurityID" GET parameter appended to the base link.
|
mandatory "SecurityID" GET parameter appended to the base link.
|
||||||
|
|
||||||
|
@ -15,11 +15,11 @@ Default setting:
|
|||||||
|
|
||||||
Overriding these defaults
|
Overriding these defaults
|
||||||
|
|
||||||
* `[api:HTTP::set_cache_age()]` can be used to set the max-age component of the cache-control line, in seconds.
|
* [api:HTTP::set_cache_age()] can be used to set the max-age component of the cache-control line, in seconds.
|
||||||
Set it to 0 to disable caching; the "no-cache" clause in `Cache-Control` and `Pragma` will be included.
|
Set it to 0 to disable caching; the "no-cache" clause in `Cache-Control` and `Pragma` will be included.
|
||||||
* `[api:HTTP::register_modification_date()]` can be used to set the modification date to something more recent than the default.
|
* [api:HTTP::register_modification_date()] can be used to set the modification date to something more recent than the default.
|
||||||
|
|
||||||
How it works:
|
How it works:
|
||||||
|
|
||||||
* `[api:DataObject::__construct()]` calls `[api:HTTP::register_modification_date()]` whenever a record comes from the database
|
* [api:DataObject::__construct()] calls [api:HTTP::register_modification_date()] whenever a record comes from the database
|
||||||
* `Controller::run()` calls `[api:HTTP::add_cache_headers()]` before outputting the page
|
* [api:Controller::run()] calls [api:HTTP::add_cache_headers()] before outputting the page
|
||||||
|
@ -26,7 +26,7 @@ Feature overview:
|
|||||||
|
|
||||||
You can use the CsvBulkLoader without subclassing or other customizations, if the column names
|
You can use the CsvBulkLoader without subclassing or other customizations, if the column names
|
||||||
in your CSV file match `$db` properties in your dataobject. E.g. a simple import for the
|
in your CSV file match `$db` properties in your dataobject. E.g. a simple import for the
|
||||||
`[api:Member]` class could have this data in a file:
|
[api:Member] class could have this data in a file:
|
||||||
|
|
||||||
FirstName,LastName,Email
|
FirstName,LastName,Email
|
||||||
Donald,Duck,donald@disney.com
|
Donald,Duck,donald@disney.com
|
||||||
@ -38,7 +38,7 @@ The loader would be triggered through the `load()` method:
|
|||||||
$loader = new CsvBulkLoader('Member');
|
$loader = new CsvBulkLoader('Member');
|
||||||
$result = $loader->load('<my-file-path>');
|
$result = $loader->load('<my-file-path>');
|
||||||
|
|
||||||
By the way, you can import `[api:Member]` and `[api:Group]` data through `http://localhost/admin/security`
|
By the way, you can import [api:Member] and [api:Group] data through `http://localhost/admin/security`
|
||||||
interface out of the box.
|
interface out of the box.
|
||||||
|
|
||||||
## Import through ModelAdmin
|
## Import through ModelAdmin
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
The [api:DataObject::$defaults] array allows you to specify simple static values to be the default value for when a
|
The [api:DataObject::$defaults] array allows you to specify simple static values to be the default value for when a
|
||||||
record is created, but in many situations default values needs to be dynamically calculated. In order to do this, the
|
record is created, but in many situations default values needs to be dynamically calculated. In order to do this, the
|
||||||
[api:DataObjectSet->populateDefaults()] method will need to be overloaded.
|
[api:DataObjectSet::populateDefaults()] method will need to be overloaded.
|
||||||
|
|
||||||
This method is called whenever a new record is instantiated, and you must be sure to call the method on the parent
|
This method is called whenever a new record is instantiated, and you must be sure to call the method on the parent
|
||||||
object!
|
object!
|
||||||
|
@ -3,8 +3,8 @@
|
|||||||
The [api:DataObjectSet] class has a number of methods useful for grouping objects by fields. Together with sorting this
|
The [api:DataObjectSet] class has a number of methods useful for grouping objects by fields. Together with sorting this
|
||||||
can be used to break up long lists of data into more manageable sub-sections.
|
can be used to break up long lists of data into more manageable sub-sections.
|
||||||
|
|
||||||
The [api:DataObjectSet->groupBy()] method takes a field name as the single argument, and breaks the set up into a number
|
The [api:DataObjectSet::groupBy()] method takes a field name as the single argument, and breaks the set up into a number
|
||||||
of arrays, where each array contains only objects with the same value of that field. The [api:DataObjectSet->GroupedBy()]
|
of arrays, where each array contains only objects with the same value of that field. The [api:DataObjectSet::GroupedBy()]
|
||||||
method builds on this and returns the same data in a template-friendly format.
|
method builds on this and returns the same data in a template-friendly format.
|
||||||
|
|
||||||
## Grouping Sets By First Letter
|
## Grouping Sets By First Letter
|
||||||
@ -65,7 +65,7 @@ this example this will be a method on the Page class.
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
The final step is to render this into a template. The [api:DataObjectSet->GroupedBy()] method breaks up the set into
|
The final step is to render this into a template. The [api:DataObjectSet::GroupedBy()] method breaks up the set into
|
||||||
a number of sets, grouped by the field that is passed as the parameter. In this case, the getTitleFirstLetter method
|
a number of sets, grouped by the field that is passed as the parameter. In this case, the getTitleFirstLetter method
|
||||||
defined earlier is used to break them up.
|
defined earlier is used to break them up.
|
||||||
|
|
||||||
@ -84,7 +84,7 @@ defined earlier is used to break them up.
|
|||||||
## Grouping Sets By Month
|
## Grouping Sets By Month
|
||||||
|
|
||||||
Grouping a set by month is a very similar process. The only difference would be to sort the records by month name, and
|
Grouping a set by month is a very similar process. The only difference would be to sort the records by month name, and
|
||||||
then create a method on the DataObject that returns the month name, and pass that to the [api:DataObjectSet->GroupedBy()]
|
then create a method on the DataObject that returns the month name, and pass that to the [api:DataObjectSet::GroupedBy()]
|
||||||
call.
|
call.
|
||||||
|
|
||||||
Again, the first step is to create a method on the class in question that will be displayed in a list. For this example,
|
Again, the first step is to create a method on the class in question that will be displayed in a list. For this example,
|
||||||
@ -128,7 +128,7 @@ January to December. This can be accomplshed by sorting by the Date field:
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
The final step is the render this into the template using the [api:DataObjectSet->GroupedBy()] method.
|
The final step is the render this into the template using the [api:DataObjectSet::GroupedBy()] method.
|
||||||
|
|
||||||
:::ss
|
:::ss
|
||||||
// Modules list grouped by the Month Posted
|
// Modules list grouped by the Month Posted
|
||||||
|
@ -11,6 +11,3 @@ the language and functions which are used in the guides.
|
|||||||
* [Grouping DataObjectSets](grouping-dataobjectsets). Group results in a [api:DataObjectSet] to create sub sections.
|
* [Grouping DataObjectSets](grouping-dataobjectsets). Group results in a [api:DataObjectSet] to create sub sections.
|
||||||
* [PHPUnit Configuration](phpunit-configuration). How to setup your testing environment with PHPUnit
|
* [PHPUnit Configuration](phpunit-configuration). How to setup your testing environment with PHPUnit
|
||||||
|
|
||||||
## Feedback
|
|
||||||
|
|
||||||
If you have a topic you would like covered in these how to's please ask for it on our [Bug Tracker] `http://open.silverstripe.org`
|
|
||||||
|
@ -8,7 +8,7 @@ See "[Testing](/topics/testing)" for an overview on how to create unit tests.
|
|||||||
Short answer: Both are valid ways.
|
Short answer: Both are valid ways.
|
||||||
|
|
||||||
The `sake` executable that comes with SilverStripe can trigger a customized
|
The `sake` executable that comes with SilverStripe can trigger a customized
|
||||||
"[api:TestRunner]" class that handles the PHPUnit configuration and output formatting.
|
[api:TestRunner] class that handles the PHPUnit configuration and output formatting.
|
||||||
It's tyically invoked to run all tests through `sake dev/tests/all`,
|
It's tyically invoked to run all tests through `sake dev/tests/all`,
|
||||||
a single test with `sake dev/tests/MyTestClass`, or tests for a module with `sake dev/tests/module/mymodulename`.
|
a single test with `sake dev/tests/MyTestClass`, or tests for a module with `sake dev/tests/module/mymodulename`.
|
||||||
While the custom test runner a handy tool, its also more limited than using `phpunit` directly,
|
While the custom test runner a handy tool, its also more limited than using `phpunit` directly,
|
||||||
|
@ -4,9 +4,12 @@ Please read our [guide to contributing documentation](misc/contributing#writing-
|
|||||||
|
|
||||||
### Overview
|
### Overview
|
||||||
|
|
||||||
* [Getting started](http://silverstripe.org/getting-started/) | [Feature Overview] `http://silverstripe.org/introduction/` | [Demo](http://demo.silverstripe.org/)
|
* [Getting started](http://silverstripe.org/getting-started/) |
|
||||||
* [Download and Installation](installation/) | [Upgrading](/installation/upgrading) | [Requirements](/installation/server-requirements) | [Changelog] `http://open.silverstripe.org/wiki/ChangeLog` | [Roadmap] `http://open.silverstripe.com/roadmap`
|
[CMS Overview](http://www.silverstripe.org/software/cms/) |
|
||||||
* [API documentation](http://api.silverstripe.org/current) | [Official english book] `http://www.silverstripe.org/silverstripe-book` | [Official german book] `http://www.silverstripe.org/das-silverstripe-buch`
|
[Framework Overview](http://www.silverstripe.org/software/framework/) |
|
||||||
|
[Demo](http://demo.silverstripe.org/)
|
||||||
|
* [Download and Installation](installation/) | [Upgrading](/installation/upgrading) | [Requirements](/installation/server-requirements) | [Changelogs](changelogs)
|
||||||
|
* [API documentation](http://api.silverstripe.org/current) | [Books](http://www.silverstripe.org/learn/resources/books)
|
||||||
|
|
||||||
|
|
||||||
### Getting support
|
### Getting support
|
||||||
@ -16,7 +19,7 @@ Please read our [guide to contributing documentation](misc/contributing#writing-
|
|||||||
|
|
||||||
### Level 1: Building your first SilverStripe website
|
### Level 1: Building your first SilverStripe website
|
||||||
|
|
||||||
* [Introduction to PHP5 (zend.com)] `http://devzone.zend.com/node/view/id/627`
|
* [Introduction to PHP (codeacademy.com)](https://www.codecademy.com/learn/php)
|
||||||
* [Tutorials](tutorials)
|
* [Tutorials](tutorials)
|
||||||
* [Building a basic site](tutorials/building-a-basic-site)
|
* [Building a basic site](tutorials/building-a-basic-site)
|
||||||
* [Extending a basic site](tutorials/extending-a-basic-site)
|
* [Extending a basic site](tutorials/extending-a-basic-site)
|
||||||
@ -58,6 +61,3 @@ Please read our [guide to contributing documentation](misc/contributing#writing-
|
|||||||
* [Contributing](http://www.silverstripe.org/contributing-to-silverstripe/)
|
* [Contributing](http://www.silverstripe.org/contributing-to-silverstripe/)
|
||||||
* [Coding Conventions](misc/coding-conventions)
|
* [Coding Conventions](misc/coding-conventions)
|
||||||
|
|
||||||
<div class="hint" markdown="1">
|
|
||||||
Looking for the old DokuWiki installation? See [doc.silverstripe.org/old] `http://doc.silverstripe.org/old`.
|
|
||||||
</div>
|
|
||||||
|
@ -55,7 +55,7 @@ every page on the site, if that's easier.
|
|||||||
|
|
||||||
## I can see unparsed PHP output in my browser
|
## I can see unparsed PHP output in my browser
|
||||||
|
|
||||||
Please make sure all code inside `*.php` files is wrapped in classes. Due to the way `[api:ManifestBuilder]`
|
Please make sure all code inside `*.php` files is wrapped in classes. Due to the way [api:ManifestBuilder]
|
||||||
includes all files with this extension, any **procedural code will be executed on every call**. Most common error here
|
includes all files with this extension, any **procedural code will be executed on every call**. Most common error here
|
||||||
is putting a test.php/phpinfo.php file in the document root. See [datamodel](/topics/datamodel) and [controllers](/topics/controller)
|
is putting a test.php/phpinfo.php file in the document root. See [datamodel](/topics/datamodel) and [controllers](/topics/controller)
|
||||||
for ways how to structure your code.
|
for ways how to structure your code.
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
This topic covers setting up your Mac as a Web Server and installing SilverStripe.
|
This topic covers setting up your Mac as a Web Server and installing SilverStripe.
|
||||||
|
|
||||||
While OSX Comes bundled with PHP and Apache (Thanks Apple!) Its not quite ideal for SilverStripe so for setting up a
|
While OSX Comes bundled with PHP and Apache (Thanks Apple!) Its not quite ideal for SilverStripe so for setting up a
|
||||||
webserver on OSX we suggest using [MAMP](http://www.mamp.info/en/index.php) or using [MacPorts](http://www.macports.org/)
|
webserver on OSX we suggest using [MAMP](https://www.mamp.info/en) or using [MacPorts](http://www.macports.org/)
|
||||||
to manage your packages.
|
to manage your packages.
|
||||||
|
|
||||||
If you want to use the default OSX PHP version then you will need to recompile your own versions of PHP with GD. Providing instructions
|
If you want to use the default OSX PHP version then you will need to recompile your own versions of PHP with GD. Providing instructions
|
||||||
|
@ -122,7 +122,7 @@ When ready, hit **Install SilverStripe**.
|
|||||||
|
|
||||||
SilverStripe should now be installed and you should have a basic site with three pages.
|
SilverStripe should now be installed and you should have a basic site with three pages.
|
||||||
|
|
||||||
However, URLs will not look "nice", like this: http://localhost/index.php/about-us. In order to fix this problem, we need to install a third-party URL rewriting tool, as IIS 6 does not support this natively.
|
However, URLs will not look "nice", like this: `http://localhost/index.php/about-us`. In order to fix this problem, we need to install a third-party URL rewriting tool, as IIS 6 does not support this natively.
|
||||||
|
|
||||||
Proceed to **Install IIRF** below to enable nice URLs.
|
Proceed to **Install IIRF** below to enable nice URLs.
|
||||||
|
|
||||||
|
@ -18,7 +18,7 @@
|
|||||||
After you've installed SilverStripe on XP using WebPI. You'll need to make a configuration change to get it working.
|
After you've installed SilverStripe on XP using WebPI. You'll need to make a configuration change to get it working.
|
||||||
|
|
||||||
Specifically, you need to configure XP not to check that a script file exists before executing the script. This is
|
Specifically, you need to configure XP not to check that a script file exists before executing the script. This is
|
||||||
necessary for URLs of the form http://localhost/silverstripe/index.php/about-us/ to work.
|
necessary for URLs of the form `http://localhost/silverstripe/index.php/about-us/` to work.
|
||||||
|
|
||||||
* Open IIS Administrator
|
* Open IIS Administrator
|
||||||
* Expand the local computer tree node
|
* Expand the local computer tree node
|
||||||
|
@ -41,7 +41,7 @@ step.
|
|||||||
* It's better to submit multiple patches with separate bits of functionality than a big patch containing lots of
|
* It's better to submit multiple patches with separate bits of functionality than a big patch containing lots of
|
||||||
changes
|
changes
|
||||||
* Document your code inline through [PHPDoc](http://en.wikipedia.org/wiki/PHPDoc) syntax. See our
|
* Document your code inline through [PHPDoc](http://en.wikipedia.org/wiki/PHPDoc) syntax. See our
|
||||||
[API documentation] `http://api.silverstripe.org/trunk` for good examples.
|
[API documentation](http://api.silverstripe.org) for examples and [SS Markdown](/misc/ss-markdown).
|
||||||
* Also check and update documentation on [doc.silverstripe.org](http://doc.silverstripe.org). Check for any references to functionality deprecated or extended through your patch. Documentation changes should be included in the patch.
|
* Also check and update documentation on [doc.silverstripe.org](http://doc.silverstripe.org). Check for any references to functionality deprecated or extended through your patch. Documentation changes should be included in the patch.
|
||||||
* We will attribute the change to you whereever possible (git does this automatically for pull requests)
|
* We will attribute the change to you whereever possible (git does this automatically for pull requests)
|
||||||
* If you get stuck, please post to the [forum](http://silverstripe.org/forum) or for deeper core problems, to the [core mailinglist](https://groups.google.com/forum/#!forum/silverstripe-dev)
|
* If you get stuck, please post to the [forum](http://silverstripe.org/forum) or for deeper core problems, to the [core mailinglist](https://groups.google.com/forum/#!forum/silverstripe-dev)
|
||||||
@ -200,7 +200,7 @@ Contributing documentation is the same process as providing any other patch
|
|||||||
should fit elsewhere. *Example: ModelAdmin*
|
should fit elsewhere. *Example: ModelAdmin*
|
||||||
* **Misc**: "Meta" documentation like coding conventions that doesn't directly relate to a feature or API.
|
* **Misc**: "Meta" documentation like coding conventions that doesn't directly relate to a feature or API.
|
||||||
|
|
||||||
See [What to write (jacobian.org)](http://jacobian.org/writing/great-documentation/what-to-write/) for an excellent
|
See [What to write (jacobian.org)](http://jacobian.org/writing/what-to-write/) for an excellent
|
||||||
introduction to the different types of documentation, and [Producing OSS: "Documentation"](http://producingoss.com/en/getting-started.html#documentation)
|
introduction to the different types of documentation, and [Producing OSS: "Documentation"](http://producingoss.com/en/getting-started.html#documentation)
|
||||||
for good rules of thumb for documenting opensource software.
|
for good rules of thumb for documenting opensource software.
|
||||||
|
|
||||||
@ -280,7 +280,7 @@ The [sapphiredocs] `https://github.com/silverstripe/silverstripe-sapphiredocs` m
|
|||||||
|
|
||||||
### Further reading
|
### Further reading
|
||||||
|
|
||||||
* [Writing great documentation (jacobian.org)](http://jacobian.org/writing/great-documentation/)
|
* [Writing great documentation (jacobian.org)](https://jacobian.org/writing/great-documentation/)
|
||||||
* [How tech writing sucks: Five Sins](http://www.slash7.com/articles/2006/11/15/tech-writing-the-five-sins)
|
* [How tech writing sucks: Five Sins](http://www.slash7.com/articles/2006/11/15/tech-writing-the-five-sins)
|
||||||
* [What is good documentation?](http://www.techscribe.co.uk/techw/whatis.htm)
|
* [What is good documentation?](http://www.techscribe.co.uk/techw/whatis.htm)
|
||||||
|
|
||||||
|
@ -75,10 +75,10 @@ In this example, `$A` and `$B` refer to `$obj->Property()->A()` and `$obj->Prope
|
|||||||
<% end_control %>
|
<% end_control %>
|
||||||
|
|
||||||
|
|
||||||
If the method/field returned is an iterator such as a `[api:DataObject]`, then the control block will be repeated for
|
If the method/field returned is an iterator such as a [api:DataObject], then the control block will be repeated for
|
||||||
each element of that iterator. This is the cornerstone of all menu and list generation in SilverStripe.
|
each element of that iterator. This is the cornerstone of all menu and list generation in SilverStripe.
|
||||||
|
|
||||||
In this example, `Menu(1)` returns a `[api:DataObjectSet]` listing each top level main menu item (for more info on `Menu(1)`:
|
In this example, `Menu(1)` returns a [api:DataObjectSet] listing each top level main menu item (for more info on `Menu(1)`:
|
||||||
[Making a Navigation System](/tutorials/building-a-basic-site#Making-a-Navigation-System)). The `<a>`
|
[Making a Navigation System](/tutorials/building-a-basic-site#Making-a-Navigation-System)). The `<a>`
|
||||||
tag is repeated once for each main menu item, and the `$Link` and `$Title` values for each menu item is substituted in.
|
tag is repeated once for each main menu item, and the `$Link` and `$Title` values for each menu item is substituted in.
|
||||||
|
|
||||||
@ -165,7 +165,7 @@ for adding notes for other developers but for things you don't want published in
|
|||||||
|
|
||||||
### Formatting Template Values
|
### Formatting Template Values
|
||||||
|
|
||||||
The following example takes the Title field of our object, casts it to a `[api:Varchar]` object, and then calls
|
The following example takes the Title field of our object, casts it to a [api:Varchar] object, and then calls
|
||||||
the `$XML` object on that Varchar object.
|
the `$XML` object on that Varchar object.
|
||||||
|
|
||||||
:::ss
|
:::ss
|
||||||
@ -241,8 +241,8 @@ Will give you the ability to call `$MyCustomValue` from anywhere in your templat
|
|||||||
// output "I've got one thing to say to you: <i>Hi, this is my site</i>"
|
// output "I've got one thing to say to you: <i>Hi, this is my site</i>"
|
||||||
|
|
||||||
|
|
||||||
Your function could return a single value as above or it could be a subclass of `[api:ArrayData]` for example a
|
Your function could return a single value as above or it could be a subclass of [api:ArrayData] for example a
|
||||||
`[api:DataObject]` with many values then each of these could be accessible via a control loop
|
[api:DataObject] with many values then each of these could be accessible via a control loop
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
..
|
..
|
||||||
@ -298,9 +298,9 @@ This is all very well and good, but how do the templates actually get called?
|
|||||||
|
|
||||||
Templates do nothing on their own. Rather, they are used to render *a particular object*. All of the `<% if %>`, `<%control %>`,
|
Templates do nothing on their own. Rather, they are used to render *a particular object*. All of the `<% if %>`, `<%control %>`,
|
||||||
and variable codes are methods or parameters that are called *on that object*. All that is necessary is
|
and variable codes are methods or parameters that are called *on that object*. All that is necessary is
|
||||||
that the object is an instance of `[api:ViewableData]` (or one of its subclasses).
|
that the object is an instance of [api:ViewableData] (or one of its subclasses).
|
||||||
|
|
||||||
The key is `[api:ViewableData::renderWith()]`. This method is passed a For example, within the controller's default action,
|
The key is [api:ViewableData::renderWith()]. This method is passed a For example, within the controller's default action,
|
||||||
there is an instruction of the following sort:
|
there is an instruction of the following sort:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -310,7 +310,7 @@ there is an instruction of the following sort:
|
|||||||
Here's what this line does:
|
Here's what this line does:
|
||||||
|
|
||||||
* First `renderWith()` constructs a new object: `$template = new SSViewer("TemplateName");`
|
* First `renderWith()` constructs a new object: `$template = new SSViewer("TemplateName");`
|
||||||
* `[api:SSViewer]` will take the content of `TemplateName.ss`, and turn it into PHP code.
|
* [api:SSViewer] will take the content of `TemplateName.ss`, and turn it into PHP code.
|
||||||
* Then `renderWith()` passes the controller to `$template->process($controller);`
|
* Then `renderWith()` passes the controller to `$template->process($controller);`
|
||||||
* `SSViewer::process()` will execute the PHP code generated from `TemplateName.ss` and return the results.
|
* `SSViewer::process()` will execute the PHP code generated from `TemplateName.ss` and return the results.
|
||||||
|
|
||||||
@ -320,7 +320,7 @@ Here's what this line does:
|
|||||||
available template name.
|
available template name.
|
||||||
|
|
||||||
Below is an example of how to implement renderWith. In the example below the page is rendered using the myAjaxTemplate
|
Below is an example of how to implement renderWith. In the example below the page is rendered using the myAjaxTemplate
|
||||||
if the page is called by an ajax function (using `[api:Director::is_ajax()]`). Note that the index function is called by
|
if the page is called by an ajax function (using [api:Director::is_ajax()]). Note that the index function is called by
|
||||||
default if it exists and there is no action in the url parameters.
|
default if it exists and there is no action in the url parameters.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
Ever wonder when you use `$Title` and `<% Control Children %>` what else you can call in the templates?. This page is
|
Ever wonder when you use `$Title` and `<% Control Children %>` what else you can call in the templates?. This page is
|
||||||
here to help with a guide on what template controls you can call.
|
here to help with a guide on what template controls you can call.
|
||||||
|
|
||||||
**Note for advanced users:** These built-in page controls are defined in the `[api:SiteTree]` classes, which are the
|
**Note for advanced users:** These built-in page controls are defined in the [api:SiteTree] classes, which are the
|
||||||
'root' data-object and controller classes for all the sites. So if you're dealing with something that isn't a sub-class
|
'root' data-object and controller classes for all the sites. So if you're dealing with something that isn't a sub-class
|
||||||
of one of these, our handy reference to 'built-in page controls' won't be so relevant.
|
of one of these, our handy reference to 'built-in page controls' won't be so relevant.
|
||||||
|
|
||||||
@ -220,18 +220,18 @@ These controls detect the visitor's previous experience with the site:
|
|||||||
|
|
||||||
#### $Now.Nice, $Now.Year
|
#### $Now.Nice, $Now.Year
|
||||||
|
|
||||||
$Now returns the current date. You can call any of the methods from the `[api:Date]` class on
|
$Now returns the current date. You can call any of the methods from the [api:Date] class on
|
||||||
it.
|
it.
|
||||||
|
|
||||||
#### $Created.Nice, $Created.Ago
|
#### $Created.Nice, $Created.Ago
|
||||||
|
|
||||||
$Created returns the time the page was created, $Created.Ago returns how long ago the page was created. You can also
|
$Created returns the time the page was created, $Created.Ago returns how long ago the page was created. You can also
|
||||||
call any of methods of the `[api:Date]` class on it.
|
call any of methods of the [api:Date] class on it.
|
||||||
|
|
||||||
#### $LastEdited.Nice, $LastEdited.Ago
|
#### $LastEdited.Nice, $LastEdited.Ago
|
||||||
|
|
||||||
$LastEdited returns the time the page was modified, $LastEdited.Ago returns how long ago the page was modified.You can also
|
$LastEdited returns the time the page was modified, $LastEdited.Ago returns how long ago the page was modified.You can also
|
||||||
call any of methods of the `[api:Date]` class on it.
|
call any of methods of the [api:Date] class on it.
|
||||||
|
|
||||||
### DataObjectSet Options
|
### DataObjectSet Options
|
||||||
|
|
||||||
|
@ -2,20 +2,20 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
Shows a group of DataObjects as a (readonly) tabular list (similiar to `[api:TableListField]`.)
|
Shows a group of DataObjects as a (readonly) tabular list (similiar to [api:TableListField].)
|
||||||
|
|
||||||
You can specify limits and filters for the resultset by customizing query-settings (mostly the ID-field on the other
|
You can specify limits and filters for the resultset by customizing query-settings (mostly the ID-field on the other
|
||||||
side of a one-to-many-relationship).
|
side of a one-to-many-relationship).
|
||||||
|
|
||||||
See `[api:TableListField]` for more documentation on the base-class
|
See [api:TableListField] for more documentation on the base-class
|
||||||
|
|
||||||
## Source Input
|
## Source Input
|
||||||
|
|
||||||
See `[api:TableListField]`.
|
See [api:TableListField].
|
||||||
|
|
||||||
## Setting Parent/Child-Relations
|
## Setting Parent/Child-Relations
|
||||||
|
|
||||||
`[api:ComplexTableField]` tries to determine the parent-relation automatically by looking at the $has_one property on the listed
|
[api:ComplexTableField] tries to determine the parent-relation automatically by looking at the $has_one property on the listed
|
||||||
child, or the record loaded into the surrounding form (see getParentClass() and getParentIdName()). You can force a
|
child, or the record loaded into the surrounding form (see getParentClass() and getParentIdName()). You can force a
|
||||||
specific parent relation:
|
specific parent relation:
|
||||||
|
|
||||||
@ -65,7 +65,7 @@ If you don't want several functions to appear (e.g. no add-link), there's severa
|
|||||||
* Use `ComplexTableField->setPermissions(array("show","edit"))` to limit the functionality without touching the template
|
* Use `ComplexTableField->setPermissions(array("show","edit"))` to limit the functionality without touching the template
|
||||||
(more secure). Possible values are "show","edit", "delete" and "add".
|
(more secure). Possible values are "show","edit", "delete" and "add".
|
||||||
|
|
||||||
* Subclass `[api:ComplexTableField]` and override the rendering-mechanism
|
* Subclass [api:ComplexTableField] and override the rendering-mechanism
|
||||||
* Use `ComplexTableField->setTemplate()` and `ComplexTableField->setTemplatePopup()` to provide custom templates
|
* Use `ComplexTableField->setTemplate()` and `ComplexTableField->setTemplatePopup()` to provide custom templates
|
||||||
|
|
||||||
### Customising fields and Requirements in the popup
|
### Customising fields and Requirements in the popup
|
||||||
@ -73,7 +73,7 @@ If you don't want several functions to appear (e.g. no add-link), there's severa
|
|||||||
There are several ways to customise the fields in the popup. Often you would want to display more information in the
|
There are several ways to customise the fields in the popup. Often you would want to display more information in the
|
||||||
popup as there is more real-estate for you to play with.
|
popup as there is more real-estate for you to play with.
|
||||||
|
|
||||||
`[api:ComplexTableField]` gives you several options to do this. You can either
|
[api:ComplexTableField] gives you several options to do this. You can either
|
||||||
|
|
||||||
* Pass a FieldSet in the constructor.
|
* Pass a FieldSet in the constructor.
|
||||||
* Pass a String in the constructor.
|
* Pass a String in the constructor.
|
||||||
@ -83,7 +83,7 @@ The second will call the String as a method on the source class (Which should re
|
|||||||
Popup.
|
Popup.
|
||||||
|
|
||||||
You can also customise Javascript which is loaded for the Lightbox. As Requirements::clear() is called when the popup is
|
You can also customise Javascript which is loaded for the Lightbox. As Requirements::clear() is called when the popup is
|
||||||
instantiated, `[api:ComplexTableField]` will look for a function to gather any specific requirements that you might need on your
|
instantiated, [api:ComplexTableField] will look for a function to gather any specific requirements that you might need on your
|
||||||
source class. (e.g. Inline Javascript or styling).
|
source class. (e.g. Inline Javascript or styling).
|
||||||
|
|
||||||
For this, create a function called "getRequirementsForPopup".
|
For this, create a function called "getRequirementsForPopup".
|
||||||
@ -116,7 +116,7 @@ You'll have to do something like this in your form:
|
|||||||
|
|
||||||
You have to hack in an ID on the form, as the CMS forms have this, and front end forms usually do not.
|
You have to hack in an ID on the form, as the CMS forms have this, and front end forms usually do not.
|
||||||
|
|
||||||
It's not a perfect solution, but it works relatively well to get a simple `[api:ComplexTableField]` up and running on the front
|
It's not a perfect solution, but it works relatively well to get a simple [api:ComplexTableField] up and running on the front
|
||||||
end.
|
end.
|
||||||
|
|
||||||
To come: Make it a lot more flexible so tables can be easily used on the front end. It also needs to be flexible enough
|
To come: Make it a lot more flexible so tables can be easily used on the front end. It also needs to be flexible enough
|
||||||
@ -132,19 +132,19 @@ Most of the time, you need to override the following methods:
|
|||||||
|
|
||||||
### Examples
|
### Examples
|
||||||
|
|
||||||
* `[api:AssetTableField]`
|
* [api:AssetTableField]
|
||||||
* `[api:MemberTableField]`
|
* [api:MemberTableField]
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
`[api:ComplexTableField]`
|
[api:ComplexTableField]
|
||||||
|
|
||||||
## Todo
|
## Todo
|
||||||
|
|
||||||
* Find a less fragile solution for accessing this field through the main controller and ReferencedField, e.g. build a
|
* Find a less fragile solution for accessing this field through the main controller and ReferencedField, e.g. build a
|
||||||
seperate CTF-instance (doesn't necessarly have to be connected to the original by ReferencedField)
|
seperate CTF-instance (doesn't necessarly have to be connected to the original by ReferencedField)
|
||||||
* Control width/height of popup by constructor (hardcoded at the moment)
|
* Control width/height of popup by constructor (hardcoded at the moment)
|
||||||
* Integrate search from MemberTableField.php directly on `[api:ComplexTableField]`
|
* Integrate search from MemberTableField.php directly on [api:ComplexTableField]
|
||||||
* Less performance-hungry implementation of detail-view paging (don't return all items on a single view)
|
* Less performance-hungry implementation of detail-view paging (don't return all items on a single view)
|
||||||
* Use automatic has-many and many-many functions to return a ComponentSet rather than building the join manually
|
* Use automatic has-many and many-many functions to return a ComponentSet rather than building the join manually
|
||||||
* Javascript/Ajax-Sorting (see [http://www.activewidgets.com/grid/](http://www.activewidgets.com/grid/) and [http://openrico.org/rico/livegrid.page](http://openrico.org/rico/livegrid.page))
|
* Javascript/Ajax-Sorting (see [http://www.activewidgets.com/grid/](http://www.activewidgets.com/grid/) and [http://openrico.org/rico/livegrid.page](http://openrico.org/rico/livegrid.page))
|
||||||
|
@ -5,7 +5,7 @@ opting for "convention over configuration". This page details what that databas
|
|||||||
|
|
||||||
## Base tables
|
## Base tables
|
||||||
|
|
||||||
Each direct sub-class of `[api:DataObject]` will have its own table.
|
Each direct sub-class of [api:DataObject] will have its own table.
|
||||||
|
|
||||||
The following fields are always created.
|
The following fields are always created.
|
||||||
|
|
||||||
@ -36,21 +36,21 @@ data sub-classed objects across **multiple tables**.
|
|||||||
|
|
||||||
For example, suppose we have the following set of classes:
|
For example, suppose we have the following set of classes:
|
||||||
|
|
||||||
* Class `[api:SiteTree]` extends `[api:DataObject]`: Title, Content fields
|
* Class [api:SiteTree] extends [api:DataObject]: Title, Content fields
|
||||||
* Class `[api:Page]` extends `[api:SiteTree]`: Abstract field
|
* Class Page extends [api:SiteTree]: Abstract field
|
||||||
* Class NewsSection extends `[api:SiteTree]`: *No special fields*
|
* Class NewsSection extends [api:SiteTree]: *No special fields*
|
||||||
* Class NewsArticle extend `[api:Page]`: ArticleDate field
|
* Class NewsArticle extend Page: ArticleDate field
|
||||||
|
|
||||||
The data for the following classes would be stored across the following tables:
|
The data for the following classes would be stored across the following tables:
|
||||||
|
|
||||||
* `[api:SiteTree]`
|
* [api:SiteTree]
|
||||||
* ID: Int
|
* ID: Int
|
||||||
* ClassName: Enum('SiteTree', 'Page', 'NewsArticle')
|
* ClassName: Enum('SiteTree', 'Page', 'NewsArticle')
|
||||||
* Created: Datetime
|
* Created: Datetime
|
||||||
* LastEdited: Datetime
|
* LastEdited: Datetime
|
||||||
* Title: Varchar
|
* Title: Varchar
|
||||||
* Content: Text
|
* Content: Text
|
||||||
* `[api:Page]`
|
* Page
|
||||||
* ID: Int
|
* ID: Int
|
||||||
* Abstract: Text
|
* Abstract: Text
|
||||||
* NewsArticle
|
* NewsArticle
|
||||||
@ -59,16 +59,16 @@ The data for the following classes would be stored across the following tables:
|
|||||||
|
|
||||||
The way it works is this:
|
The way it works is this:
|
||||||
|
|
||||||
* "Base classes" are direct sub-classes of `[api:DataObject]`. They are always given a table, whether or not they have
|
* "Base classes" are direct sub-classes of [api:DataObject]. They are always given a table, whether or not they have
|
||||||
special fields. This is called the "base table"
|
special fields. This is called the "base table"
|
||||||
* The base table's ClassName field is set to class of the given record. It's an enumeration of all possible
|
* The base table's ClassName field is set to class of the given record. It's an enumeration of all possible
|
||||||
sub-classes of the base class (including the base class itself)
|
sub-classes of the base class (including the base class itself)
|
||||||
* Each sub-class of the base object will also be given its own table, *as long as it has custom fields*. In the
|
* Each sub-class of the base object will also be given its own table, *as long as it has custom fields*. In the
|
||||||
example above, NewsSection didn't have its own data and so an extra table would be redundant.
|
example above, NewsSection didn't have its own data and so an extra table would be redundant.
|
||||||
* In all the tables, ID is the primary key. A matching ID number is used for all parts of a particular record:
|
* In all the tables, ID is the primary key. A matching ID number is used for all parts of a particular record:
|
||||||
record #2 in Page refers to the same object as record #2 in `[api:SiteTree]`.
|
record #2 in Page refers to the same object as record #2 in [api:SiteTree].
|
||||||
|
|
||||||
To retrieve a news article, SilverStripe joins the `[api:SiteTree]`, `[api:Page]` and NewsArticle tables by their ID fields. We use a
|
To retrieve a news article, SilverStripe joins the [api:SiteTree], Page and NewsArticle tables by their ID fields. We use a
|
||||||
left-join for robustness; if there is no matching record in Page, we can return a record with a blank Article field.
|
left-join for robustness; if there is no matching record in Page, we can return a record with a blank Article field.
|
||||||
|
|
||||||
## Staging and versioning
|
## Staging and versioning
|
||||||
@ -102,15 +102,15 @@ It **won't** do any of the following
|
|||||||
|
|
||||||
The information documented in this page is reflected in a few places in the code:
|
The information documented in this page is reflected in a few places in the code:
|
||||||
|
|
||||||
* `[api:DataObject]`
|
* [api:DataObject]
|
||||||
* requireTable() is responsible for specifying the required database schema
|
* requireTable() is responsible for specifying the required database schema
|
||||||
* instance_get() and instance_get_one() are responsible for generating the database queries for selecting data.
|
* instance_get() and instance_get_one() are responsible for generating the database queries for selecting data.
|
||||||
* write() is responsible for generating the database queries for writing data.
|
* write() is responsible for generating the database queries for writing data.
|
||||||
* `[api:Versioned]`
|
* [api:Versioned]
|
||||||
* augmentWrite() is responsible for altering the normal database writing operation to handle versions.
|
* augmentWrite() is responsible for altering the normal database writing operation to handle versions.
|
||||||
* augmentQuery() is responsible for altering the normal data selection queries to support versions.
|
* augmentQuery() is responsible for altering the normal data selection queries to support versions.
|
||||||
* augmentDatabase() is responsible for specifying the altered database schema to support versions.
|
* augmentDatabase() is responsible for specifying the altered database schema to support versions.
|
||||||
* `[api:MySQLDatabase]`: getNextID() is used when creating new objects; it also handles the mechanics of
|
* [api:MySQLDatabase]: getNextID() is used when creating new objects; it also handles the mechanics of
|
||||||
updating the database to have the required schema.
|
updating the database to have the required schema.
|
||||||
|
|
||||||
|
|
||||||
|
@ -8,12 +8,12 @@ A single database record & abstract class for the data-access-model.
|
|||||||
|
|
||||||
* [datamodel](/topics/datamodel): The basic pricinples
|
* [datamodel](/topics/datamodel): The basic pricinples
|
||||||
* [data-types](/topics/data-types): Casting and special property-parsing
|
* [data-types](/topics/data-types): Casting and special property-parsing
|
||||||
* `[api:DataObject]`: A "container" for DataObjects
|
* [api:DataObject]: A "container" for DataObjects
|
||||||
|
|
||||||
## Basics
|
## Basics
|
||||||
|
|
||||||
The call to `DataObject->getCMSFields()` is the centerpiece of every data administration interface in SilverStripe,
|
The call to [api:DataObject::getCMSFields()] is the centerpiece of every data administration interface in SilverStripe,
|
||||||
which returns a `[api:FieldSet]`''.
|
which returns a [api:FieldSet].
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
class MyPage extends Page {
|
class MyPage extends Page {
|
||||||
@ -27,7 +27,7 @@ which returns a `[api:FieldSet]`''.
|
|||||||
|
|
||||||
## Scaffolding Formfields
|
## Scaffolding Formfields
|
||||||
|
|
||||||
These calls retrieve a `[api:FieldSet]` for the area where you intend to work with the scaffolded form.
|
These calls retrieve a [api:FieldSet] for the area where you intend to work with the scaffolded form.
|
||||||
|
|
||||||
### For the CMS
|
### For the CMS
|
||||||
|
|
||||||
@ -39,8 +39,8 @@ These calls retrieve a `[api:FieldSet]` for the area where you intend to work wi
|
|||||||
|
|
||||||
### For the Frontend
|
### For the Frontend
|
||||||
|
|
||||||
Used for simple frontend forms without relation editing or `[api:TabSet] behaviour. Uses `scaffoldFormFields()` by
|
Used for simple frontend forms without relation editing or [api:TabSet] behaviour. Uses `scaffoldFormFields()` by
|
||||||
default. To customize, either overload this method in your subclass, or decorate it by `DataObjectDecorator->updateFormFields()`.
|
default. To customize, either overload this method in your subclass, or decorate it by [api:DataObjectDecorator::updateFormFields()].
|
||||||
|
|
||||||
* Requirements: SilverStripe 2.3.*
|
* Requirements: SilverStripe 2.3.*
|
||||||
|
|
||||||
@ -53,7 +53,7 @@ default. To customize, either overload this method in your subclass, or decorate
|
|||||||
* Requirements: SilverStripe 2.3.*
|
* Requirements: SilverStripe 2.3.*
|
||||||
|
|
||||||
This section covers how to enhance the default scaffolded form fields from above. It is particularly useful when used
|
This section covers how to enhance the default scaffolded form fields from above. It is particularly useful when used
|
||||||
in conjunction with the `[api:ModelAdmin]` in the CMS to make relevant data administration interfaces.
|
in conjunction with the [api:ModelAdmin] in the CMS to make relevant data administration interfaces.
|
||||||
|
|
||||||
|
|
||||||
### Searchable Fields
|
### Searchable Fields
|
||||||
@ -80,8 +80,8 @@ Example: Simple Definition
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Searchable fields will be appear in the search interface with a default form field (usually a `[api:TextField]`) and a default
|
Searchable fields will be appear in the search interface with a default form field (usually a [api:TextField]) and a default
|
||||||
search filter assigned (usually an `[api:ExactMatchFilter]`). To override these defaults, you can specify additional information
|
search filter assigned (usually an [api:ExactMatchFilter]). To override these defaults, you can specify additional information
|
||||||
on `$searchable_fields`:
|
on `$searchable_fields`:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -93,7 +93,7 @@ on `$searchable_fields`:
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
If you assign a single string value, you can set it to be either a `[api:FormField]` or `[api:SearchFilter]`. To specify both, you can
|
If you assign a single string value, you can set it to be either a [api:FormField] or [api:SearchFilter]. To specify both, you can
|
||||||
assign an array:
|
assign an array:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -142,8 +142,8 @@ To include relations (''$has_one'', `$has_many` and `$many_many`) in your search
|
|||||||
|
|
||||||
* Requirements: SilverStripe 2.3.*
|
* Requirements: SilverStripe 2.3.*
|
||||||
|
|
||||||
Summary fields can be used to show a quick overview of the data for a specific `[api:DataObject]` record. Most common use is
|
Summary fields can be used to show a quick overview of the data for a specific [api:DataObject] record. Most common use is
|
||||||
their display as table columns, e.g. in the search results of a `[api:ModelAdmin]` CMS interface.
|
their display as table columns, e.g. in the search results of a [api:ModelAdmin] CMS interface.
|
||||||
|
|
||||||
Example: Getting predefined summary fields
|
Example: Getting predefined summary fields
|
||||||
|
|
||||||
@ -191,4 +191,4 @@ To include relations in your summaries, you can use a dot-notation.
|
|||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
`[api:DataObject]`
|
[api:DataObject]
|
||||||
|
@ -2,14 +2,14 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
Extensions (also referred to as decorators) allow for adding additional functionality to a `[api:DataObject]`.
|
Extensions (also referred to as decorators) allow for adding additional functionality to a [api:DataObject].
|
||||||
|
|
||||||
In some cases, it can be easier to completely replace the used class throughout the core with your custom
|
In some cases, it can be easier to completely replace the used class throughout the core with your custom
|
||||||
implementation. Have a look at `[api:Object->useCustomClass()]`.
|
implementation. Have a look at [api:Object::useCustomClass()].
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
Your Decorator will nee to be a subclass of `[api:DataObjectDecorator]` or the `[api:Extension]` class.
|
Your Decorator will nee to be a subclass of [api:DataObjectDecorator] or the [api:Extension] class.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
<?php
|
<?php
|
||||||
@ -27,7 +27,7 @@ class you want to extend.
|
|||||||
### Adding a decorator to a built-in class
|
### Adding a decorator to a built-in class
|
||||||
|
|
||||||
Sometimes you will want to add decorators to classes that you didn't make. For example, you might want to add the
|
Sometimes you will want to add decorators to classes that you didn't make. For example, you might want to add the
|
||||||
`[api:ForumRole]` decorator to the `[api:Member]` object.
|
[api:ForumRole] decorator to the [api:Member] object.
|
||||||
|
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -98,7 +98,7 @@ The $fields parameter is passed by reference, as it is an object.
|
|||||||
### Custom database generation
|
### Custom database generation
|
||||||
|
|
||||||
Some decorators are designed to transparently add more sophisticated data-collection capabilities to your data object.
|
Some decorators are designed to transparently add more sophisticated data-collection capabilities to your data object.
|
||||||
For example, `[api:Versioned]` adds version tracking and staging to any data object that it is applied to. To do this,
|
For example, [api:Versioned] adds version tracking and staging to any data object that it is applied to. To do this,
|
||||||
you need to be able to create additional database tables and fields to keep your state stored in.
|
you need to be able to create additional database tables and fields to keep your state stored in.
|
||||||
|
|
||||||
To do this, define an **augmentDatabase()** method on your decorator. This will be called when db/build is visited.
|
To do this, define an **augmentDatabase()** method on your decorator. This will be called when db/build is visited.
|
||||||
@ -110,7 +110,7 @@ care of creating, modifying, or leaving tables as required, based on your desire
|
|||||||
### Custom write queries
|
### Custom write queries
|
||||||
|
|
||||||
If you have customised the generated database, then you probably want to change the way that writes happen. This is
|
If you have customised the generated database, then you probably want to change the way that writes happen. This is
|
||||||
used by `[api:Versioned]` to get an entry written in ClassName_versions whenever an insert/update happens.
|
used by [api:Versioned] to get an entry written in ClassName_versions whenever an insert/update happens.
|
||||||
|
|
||||||
To do this, define the **augmentWrite(&$manipulation)** method. This method is passed a manipulation array representing
|
To do this, define the **augmentWrite(&$manipulation)** method. This method is passed a manipulation array representing
|
||||||
the write about to happen, and is able to amend this as desired, since it is passed by reference.
|
the write about to happen, and is able to amend this as desired, since it is passed by reference.
|
||||||
@ -121,21 +121,21 @@ The other queries that you will want to customise are the selection queries, cal
|
|||||||
Versioned object has code to redirect every request to ClassName_live, if you are browsing the live site.
|
Versioned object has code to redirect every request to ClassName_live, if you are browsing the live site.
|
||||||
|
|
||||||
To do this, define the **augmentSQL(SQLQuery &$query)** method. Again, the $query object is passed by reference and can
|
To do this, define the **augmentSQL(SQLQuery &$query)** method. Again, the $query object is passed by reference and can
|
||||||
be modified as needed by your method. Instead of a manipulation array, we have a `[api:SQLQuery]` object.
|
be modified as needed by your method. Instead of a manipulation array, we have a [api:SQLQuery] object.
|
||||||
|
|
||||||
### Additional methods
|
### Additional methods
|
||||||
|
|
||||||
The other thing you may want to do with a decorator is provide a method that can be called on the `[api:DataObject]` that is
|
The other thing you may want to do with a decorator is provide a method that can be called on the [api:DataObject] that is
|
||||||
being decorated. For instance, you may add a publish() method to every `[api:DataObject]` that is decorated with `[api:Versioned]`.
|
being decorated. For instance, you may add a publish() method to every [api:DataObject] that is decorated with [api:Versioned].
|
||||||
|
|
||||||
This is as simple as defining a method called publish() on your decorator. Bear in mind, however, that instead of
|
This is as simple as defining a method called publish() on your decorator. Bear in mind, however, that instead of
|
||||||
$this, you should be referring to $this->owner.
|
$this, you should be referring to $this->owner.
|
||||||
|
|
||||||
* $this = The `[api:DataObjectDecorator]` object.
|
* $this = The [api:DataObjectDecorator] object.
|
||||||
* $this->owner = The related `[api:DataObject]` object.
|
* $this->owner = The related [api:DataObject] object.
|
||||||
|
|
||||||
If you want to add your own internal properties, you can add this to the `[api:DataObjectDecorator]`, and these will be referred
|
If you want to add your own internal properties, you can add this to the [api:DataObjectDecorator], and these will be referred
|
||||||
to as `$this->propertyName`. Every `[api:DataObject]` has an associated `[api:DataObjectDecorator]` instance for each class that it is
|
to as `$this->propertyName`. Every [api:DataObject] has an associated [api:DataObjectDecorator] instance for each class that it is
|
||||||
decorated by.
|
decorated by.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -169,4 +169,5 @@ decorated by.
|
|||||||
|
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:DataObjectDecorator]`
|
|
||||||
|
[api:DataObjectDecorator]
|
||||||
|
@ -2,11 +2,11 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
This class represents a set of `[api:DataObject]`s, such as the results of a query. It is the base for all
|
This class represents a set of [api:DataObject]'s, such as the results of a query. It is the base for all
|
||||||
[datamodel](/topics/datamodel)-related querying. It implements the [Iterator
|
[datamodel](/topics/datamodel)-related querying. It implements the [Iterator
|
||||||
interface](http://php.net/manual/en/language.oop5.iterations.php) introduced in PHP5.
|
interface](http://php.net/manual/en/language.oop5.iterations.php) introduced in PHP5.
|
||||||
|
|
||||||
Relations (`has_many`/`many_many`) are described in `[api:ComponentSet]`, a subclass of `[api:DataObjectSet]`.
|
Relations (`has_many`/`many_many`) are described in [api:ComponentSet], a subclass of [api:DataObjectSet].
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
@ -37,7 +37,7 @@ Relations (`has_many`/`many_many`) are described in `[api:ComponentSet]`, a subc
|
|||||||
|
|
||||||
### Grouping
|
### Grouping
|
||||||
|
|
||||||
You can group a set by a specific column. Consider using `[api:SQLQuery]` with a *GROUP BY* statement for enhanced
|
You can group a set by a specific column. Consider using [api:SQLQuery] with a *GROUP BY* statement for enhanced
|
||||||
performance.
|
performance.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -56,7 +56,7 @@ This works on the object itself, so do NOT do something like this:
|
|||||||
:::php
|
:::php
|
||||||
$sortedSet = $mySet->sort('Lastname'); //ascending
|
$sortedSet = $mySet->sort('Lastname'); //ascending
|
||||||
|
|
||||||
## Merge with other `[api:DataObjectSet]`s
|
## Merge with other [api:DataObjectSet]s
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$myFirstSet->merge($mySecondSet);
|
$myFirstSet->merge($mySecondSet);
|
||||||
@ -65,7 +65,7 @@ This works on the object itself, so do NOT do something like this:
|
|||||||
|
|
||||||
### Mapping for Dropdowns
|
### Mapping for Dropdowns
|
||||||
|
|
||||||
When using `[api:DropdownField]` and its numerous subclasses to select a value from a set, you can easily map
|
When using [api:DropdownField] and its numerous subclasses to select a value from a set, you can easily map
|
||||||
the records to a compatible array:
|
the records to a compatible array:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -88,11 +88,11 @@ It is good practice to check for empty sets before doing any iteration.
|
|||||||
|
|
||||||
### Paging
|
### Paging
|
||||||
|
|
||||||
`[api:DataObject]`s have native support for dealing with **pagination**.
|
[api:DataObject]s have native support for dealing with **pagination**.
|
||||||
See *setPageLimits*, *setPageLength*, etc.
|
See *setPageLimits*, *setPageLength*, etc.
|
||||||
|
|
||||||
FIXME Complete pagination documentation
|
FIXME Complete pagination documentation
|
||||||
|
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:DataObjectSet]`
|
[api:DataObjectSet]
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
`[api:Director]` is the first step in the "execution pipeline". It parses the URL, matching it to one of a number of patterns,
|
[api:Director] is the first step in the "execution pipeline". It parses the URL, matching it to one of a number of patterns,
|
||||||
and determines the controller, action and any argument to be used. It then runs the controller, which will finally run
|
and determines the controller, action and any argument to be used. It then runs the controller, which will finally run
|
||||||
the viewer and/or perform processing steps.
|
the viewer and/or perform processing steps.
|
||||||
|
|
||||||
@ -12,7 +12,7 @@ the viewer and/or perform processing steps.
|
|||||||
|
|
||||||
## Redirection
|
## Redirection
|
||||||
|
|
||||||
The `[api:Director]` class has a number of methods to facilitate 301 and 302 HTTP redirection.
|
The [api:Director] class has a number of methods to facilitate 301 and 302 HTTP redirection.
|
||||||
|
|
||||||
* **Director::redirect("action-name")**: If there's no slash in the URL passed to redirect, then it is assumed that you
|
* **Director::redirect("action-name")**: If there's no slash in the URL passed to redirect, then it is assumed that you
|
||||||
want to go to a different action on the current controller.
|
want to go to a different action on the current controller.
|
||||||
@ -33,8 +33,8 @@ redirectBack().
|
|||||||
|
|
||||||
You can influence the way URLs are resolved one of 2 ways
|
You can influence the way URLs are resolved one of 2 ways
|
||||||
|
|
||||||
1. Adding rules to `[api:Director]` in `<yourproject>/_config.php` (See Default Rewrite Rules below for examples)
|
1. Adding rules to [api:Director] in `<yourproject>/_config.php` (See Default Rewrite Rules below for examples)
|
||||||
2. Adding rules in your extended `[api:Controller]` class via the *$url_handlers* static variable
|
2. Adding rules in your extended [api:Controller] class via the *$url_handlers* static variable
|
||||||
|
|
||||||
See [controller](/topics/controller) for examples and explanations on how the rules get processed for both 1 and 2 above.
|
See [controller](/topics/controller) for examples and explanations on how the rules get processed for both 1 and 2 above.
|
||||||
|
|
||||||
@ -56,8 +56,8 @@ SilverStripe comes with certain rewrite rules (e.g. for *admin/assets*).
|
|||||||
|
|
||||||
## Links
|
## Links
|
||||||
|
|
||||||
* See `[api:ModelAsController]` class for details on controller/model-coupling
|
* See [api:ModelAsController] class for details on controller/model-coupling
|
||||||
* See [execution-pipeline](/reference/execution-pipeline) for custom routing
|
* See [execution-pipeline](/reference/execution-pipeline) for custom routing
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:Director]`
|
[api:Director]
|
||||||
|
@ -50,13 +50,13 @@ mod_rewrite works.
|
|||||||
|
|
||||||
All requests go through main.php, which sets up the environment and then hands control over to Director.
|
All requests go through main.php, which sets up the environment and then hands control over to Director.
|
||||||
|
|
||||||
**See:** The API documentation of `[api:Main]` for information about how main.php processes requests.
|
**See:** The API documentation of [api:Main] for information about how main.php processes requests.
|
||||||
## Director and URL patterns
|
## Director and URL patterns
|
||||||
|
|
||||||
main.php relies on `[api:Director]` to work out which controller should handle this request. `[api:Director]` will instantiate that
|
main.php relies on [api:Director] to work out which controller should handle this request. [api:Director] will instantiate that
|
||||||
controller object and then call `[api:Controller::run()]`.
|
controller object and then call [api:Controller::run()].
|
||||||
|
|
||||||
**See:** The API documentation of `[api:Director]` for information about how Director parses URLs and hands control over to a controller object.
|
**See:** The API documentation of [api:Director] for information about how Director parses URLs and hands control over to a controller object.
|
||||||
|
|
||||||
In general, the URL is build up as follows: page/action/ID/otherID - e.g. `http://www.mysite.com/mypage/addToCart/12`.
|
In general, the URL is build up as follows: page/action/ID/otherID - e.g. `http://www.mysite.com/mypage/addToCart/12`.
|
||||||
This will add an object with ID 12 to the cart.
|
This will add an object with ID 12 to the cart.
|
||||||
@ -73,9 +73,9 @@ When you create a function, you can access the ID like this:
|
|||||||
|
|
||||||
## Controllers and actions
|
## Controllers and actions
|
||||||
|
|
||||||
`[api:Controller]`s are the building blocks of your application.
|
[api:Controller]s are the building blocks of your application.
|
||||||
|
|
||||||
**See:** The API documentation for `[api:Controller]`
|
**See:** The API documentation for [api:Controller]
|
||||||
|
|
||||||
You can access the following controller-method with /team/signup
|
You can access the following controller-method with /team/signup
|
||||||
|
|
||||||
|
@ -1,85 +1,85 @@
|
|||||||
# Form Field Types
|
# Form Field Types
|
||||||
|
|
||||||
This is a highlevel overview of available `[apiFormField]` subclasses. An automatically generated list is available through our [API](api:FormField)
|
This is a highlevel overview of available [api:FormField] subclasses. An automatically generated list is available through our [API](api:FormField)
|
||||||
|
|
||||||
## Formatted Input
|
## Formatted Input
|
||||||
|
|
||||||
* `[AjaxUniqueTextField](api:AjaxUniqueTextField)`: Text field that automatically checks that the value entered is unique for
|
* [api:AjaxUniqueTextField]: Text field that automatically checks that the value entered is unique for
|
||||||
the given set of fields in a given set of tables
|
the given set of fields in a given set of tables
|
||||||
* `[AutocompleteTextField](api:AutocompleteTextField)`
|
* [api:AutocompleteTextField]
|
||||||
* `[ConfirmedPasswordField](api:ConfirmedPasswordField)`: Shows two password-fields, and checks for matching passwords.
|
* [api:ConfirmedPasswordField]: Shows two password-fields, and checks for matching passwords.
|
||||||
* `[CreditCardField](api:CreditCardField)`
|
* [api:CreditCardField]
|
||||||
* `[CurrencyField](api:CurrencyField)`
|
* [api:CurrencyField]
|
||||||
* `[EmailField](api:EmailField)`
|
* [api:EmailField]
|
||||||
* `[HTMLEditorField](api:HTMLEditorField)`: A WYSIWYG editor field, powered by tinymce.
|
* [api:HTMLEditorField]: A WYSIWYG editor field, powered by tinymce.
|
||||||
* `[NumericField](api:NumericField)`: A Single Numeric field extending a typical TextField but with validation.
|
* [api:NumericField]: A Single Numeric field extending a typical TextField but with validation.
|
||||||
* `[PasswordField](api:PasswordField)`
|
* [api:PasswordField]
|
||||||
* `[UniqueRestrictedTextField](api:UniqueRestrictedTextField)`: Text field that automatically checks that the value entered
|
* [api:UniqueRestrictedTextField]: Text field that automatically checks that the value entered
|
||||||
is unique for the given set of fields in a given set of tables
|
is unique for the given set of fields in a given set of tables
|
||||||
* `[UniqueTextField](api:UniqueTextField)`: Text field that automatically checks that the value entered is unique for the
|
* [api:UniqueTextField]: Text field that automatically checks that the value entered is unique for the
|
||||||
given set of fields in a given set of tables
|
given set of fields in a given set of tables
|
||||||
|
|
||||||
## Date/Time
|
## Date/Time
|
||||||
|
|
||||||
* `[DateField](api:DateField)`: Represents a date in a textfield (New Zealand)
|
* [api:DateField]: Represents a date in a textfield (New Zealand)
|
||||||
* `[DatetimeField](api:DatetimeField)`: Combined date- and time field
|
* [api:DatetimeField]: Combined date- and time field
|
||||||
* `[TimeField](api:TimeField)`: Represents time in a textfield (New Zealand)
|
* [api:TimeField]: Represents time in a textfield (New Zealand)
|
||||||
|
|
||||||
## Structure
|
## Structure
|
||||||
|
|
||||||
* `[CompositeField](api:CompositeField)`: Base class for all fields that contain other fields. Uses `<div>` in template, but
|
* [api:CompositeField]: Base class for all fields that contain other fields. Uses `<div>` in template, but
|
||||||
doesn't necessarily have any visible styling.
|
doesn't necessarily have any visible styling.
|
||||||
* `[FieldGroup](api:FieldGroup)`: Same as CompositeField, but has default styling (indentation) attached in CMS-context.
|
* [api:FieldGroup]: Same as CompositeField, but has default styling (indentation) attached in CMS-context.
|
||||||
* `[api:FieldSet]`: Basic container for sequential fields, or nested fields through CompositeField. Does NOT render a
|
* [api:FieldSet]: Basic container for sequential fields, or nested fields through CompositeField. Does NOT render a
|
||||||
`<fieldgroup>`.
|
`<fieldgroup>`.
|
||||||
* `[TabSet](api:TabSet)`
|
* [api:TabSet]
|
||||||
* `[Tab](api:Tab)`
|
* [api:Tab]
|
||||||
|
|
||||||
|
|
||||||
## Actions
|
## Actions
|
||||||
|
|
||||||
* `[api:Form]` for more info
|
* [api:Form] for more info
|
||||||
* `[InlineFormAction](api:InlineFormAction)`: Render a button that will act as If you want to add custom behaviour, please
|
* [api:InlineFormAction]: Render a button that will act as If you want to add custom behaviour, please
|
||||||
set {inlcudeDefaultJS} to false and work with behaviour.js.
|
set {inlcudeDefaultJS} to false and work with behaviour.js.
|
||||||
* `[api:Image]`: Action that uses an image instead of a button
|
* [api:Image]: Action that uses an image instead of a button
|
||||||
* `[InlineFormAction](api:InlineFormAction)`: Prevents placement of a button in the CMS-button-bar.
|
* [api:InlineFormAction]: Prevents placement of a button in the CMS-button-bar.
|
||||||
|
|
||||||
## Files
|
## Files
|
||||||
|
|
||||||
* `[FileField](api:FileField)`: Simple file upload dialog.
|
* [api:FileField]: Simple file upload dialog.
|
||||||
* `[FileIFrameField](api:FileIFrameField)`: File uploads through an iframe
|
* [api:FileIFrameField]: File uploads through an iframe
|
||||||
* `[api:ImageField]`: Image upload through an iframe, with thumbnails and file-selection from existing assets
|
* [api:ImageField]: Image upload through an iframe, with thumbnails and file-selection from existing assets
|
||||||
* `[SimpleImageField](api:SimpleImageField)`: SimpleImageField provides an easy way of uploading images to Image has_one
|
* [api:SimpleImageField]: SimpleImageField provides an easy way of uploading images to Image has_one
|
||||||
relationships. Unlike ImageField, it doesn't use an iframe.
|
relationships. Unlike ImageField, it doesn't use an iframe.
|
||||||
|
|
||||||
|
|
||||||
## Relations
|
## Relations
|
||||||
|
|
||||||
* `[ComplexTableField](api:ComplexTableField)`: Provides a tabuar list in your form with view/edit/add/delete links to modify
|
* [api:ComplexTableField]: Provides a tabuar list in your form with view/edit/add/delete links to modify
|
||||||
records with a "has-one"-relationship (in a lightbox-popup).
|
records with a "has-one"-relationship (in a lightbox-popup).
|
||||||
* `[HasManyComplexTableField](api:HasManyComplexTableField)`
|
* [api:HasManyComplexTableField]
|
||||||
* `[HasOneComplexTableField](api:HasOneComplexTableField)`
|
* [api:HasOneComplexTableField]
|
||||||
* `[LanguageDropdownField](api:LanguageDropdownField)`: An extension to dropdown field, pre-configured to list languages.
|
* [api:LanguageDropdownField]: An extension to dropdown field, pre-configured to list languages.
|
||||||
Tied into i18n.
|
Tied into i18n.
|
||||||
* `[ManyManyComplexTableField](api:ManyManyComplexTableField)`
|
* [api:ManyManyComplexTableField]
|
||||||
* `[TableField](api:TableField)`
|
* [api:TableField]
|
||||||
* `[api:TableListField]`
|
* [api:TableListField]
|
||||||
* `[TreeDropdownField](api:TreeDropdownField)`
|
* [api:TreeDropdownField]
|
||||||
* `[TreeMultiselectField](api:TreeMultiselectField)`: represents many-many joins using a tree selector shown in a
|
* [api:TreeMultiselectField]: represents many-many joins using a tree selector shown in a
|
||||||
dropdown-like element
|
dropdown-like element
|
||||||
* `[api:WidgetArea]`
|
* [api:WidgetArea]
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## Dataless/Utility
|
## Dataless/Utility
|
||||||
|
|
||||||
* `[DatalessField](api:DatalessField)` - Base class for fields which add some HTML to the form but don't submit any data or
|
* [api:DatalessField] - Base class for fields which add some HTML to the form but don't submit any data or
|
||||||
save it to the database
|
save it to the database
|
||||||
* `[HeaderField](api:HeaderField)`: Renders a simple `<h1>`-`<h6>` header
|
* [api:HeaderField]: Renders a simple `<h1>`-`<h6>` header
|
||||||
* `[HiddenField](api:HiddenField)`
|
* [api:HiddenField]
|
||||||
* `[LabelField](api:LabelField)`
|
* [api:LabelField]
|
||||||
* `[LiteralField](api:LiteralField)`: Renders arbitrary HTML into a form.
|
* [api:LiteralField]: Renders arbitrary HTML into a form.
|
||||||
|
|
||||||
## CMS Field Editor
|
## CMS Field Editor
|
||||||
|
|
||||||
Please see `[api:HTMLEditorField]` for in-depth documentation about custom forms created through a GUI in the CMS.
|
Please see [api:HTMLEditorField] for in-depth documentation about custom forms created through a GUI in the CMS.
|
||||||
|
@ -2,15 +2,15 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
Represents an image object, inheriting all base functionality from the [file](api:file) class with extra functionality
|
Represents an image object, inheriting all base functionality from the [api:File] class with extra functionality
|
||||||
including resizing.
|
including resizing.
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
### Form Fields
|
### Form Fields
|
||||||
|
|
||||||
* `[api:Image]`. Designed to provide a complex image uploader for the CMS.
|
* [api:Image]. Designed to provide a complex image uploader for the CMS.
|
||||||
* `[api:SimpleImageField]`. A Simple Image Upload Form
|
* [api:SimpleImageField]. A Simple Image Upload Form
|
||||||
|
|
||||||
### Resizing Images in PHP
|
### Resizing Images in PHP
|
||||||
|
|
||||||
@ -102,9 +102,9 @@ For output of an image tag with the image automatically resized to 80px width, y
|
|||||||
|
|
||||||
### Form Upload
|
### Form Upload
|
||||||
|
|
||||||
For usage on a website form, see `[api:SimpleImageField]`.
|
For usage on a website form, see [api:SimpleImageField].
|
||||||
|
|
||||||
If you want to upload images within the CMS, see `[api:ImageField]`.
|
If you want to upload images within the CMS, see [api:ImageField].
|
||||||
|
|
||||||
### Clearing Thumbnail Cache
|
### Clearing Thumbnail Cache
|
||||||
|
|
||||||
@ -120,4 +120,4 @@ image cache.
|
|||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
`[api:Image]`
|
[api:Image]
|
||||||
|
@ -10,11 +10,11 @@ LeftAndMain is the base class of all the admin area controllers.
|
|||||||
|
|
||||||
Please use LeftAndMain::ForceReload to reload the whole form-area after an Ajax-Request. If you just need to refresh
|
Please use LeftAndMain::ForceReload to reload the whole form-area after an Ajax-Request. If you just need to refresh
|
||||||
parts of the form, please use javascript-replacement in the response of the original Ajax-Request. Consider using
|
parts of the form, please use javascript-replacement in the response of the original Ajax-Request. Consider using
|
||||||
`[api:Form]` for compiling Ajax-Responses and automatic detection of Ajax/Non-Ajax-Calls.
|
[api:Form] for compiling Ajax-Responses and automatic detection of Ajax/Non-Ajax-Calls.
|
||||||
|
|
||||||
### Custom Access Checking
|
### Custom Access Checking
|
||||||
|
|
||||||
You can customize access control in `[api:LeftAndMain]`.
|
You can customize access control in [api:LeftAndMain].
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
// mysite/_config.php
|
// mysite/_config.php
|
||||||
@ -144,7 +144,7 @@ For example:
|
|||||||
static $menu_priority = 60;
|
static $menu_priority = 60;
|
||||||
|
|
||||||
|
|
||||||
See also `[api:CMSMenu]`
|
See also [api:CMSMenu]
|
||||||
|
|
||||||
### Translatable Menu Titles
|
### Translatable Menu Titles
|
||||||
|
|
||||||
@ -181,10 +181,10 @@ See [Javascript in the CMS](/topics/javascript#javascript-cms)
|
|||||||
|
|
||||||
## Related
|
## Related
|
||||||
|
|
||||||
* `[api:CMSMain]`
|
* [api:CMSMain]
|
||||||
* `[api:AssetAdmin]`
|
* [api:AssetAdmin]
|
||||||
* `[api:SecurityAdmin]`
|
* [api:SecurityAdmin]
|
||||||
* `[api:ModelAdmin]`
|
* [api:ModelAdmin]
|
||||||
|
|
||||||
## TODO
|
## TODO
|
||||||
|
|
||||||
|
@ -2,11 +2,11 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
The `[api:Member]` class is used to represent user accounts on a SilverStripe site (including newsletter recipients).
|
The [api:Member] class is used to represent user accounts on a SilverStripe site (including newsletter recipients).
|
||||||
|
|
||||||
## Testing For Logged In Users
|
## Testing For Logged In Users
|
||||||
|
|
||||||
The `[api:Member]` class comes with 2 static methods for getting information about the current logged in user.
|
The [api:Member] class comes with 2 static methods for getting information about the current logged in user.
|
||||||
|
|
||||||
**Member::currentUserID()**
|
**Member::currentUserID()**
|
||||||
|
|
||||||
@ -38,11 +38,11 @@ Returns the full *Member* Object for the current user, returns *null* if user is
|
|||||||
## Subclassing
|
## Subclassing
|
||||||
|
|
||||||
<div class="warning" markdown="1">
|
<div class="warning" markdown="1">
|
||||||
This is the least desirable way of extending the `[api:Member]` class. It's better to use `[api:DataObjectDecorator]`
|
This is the least desirable way of extending the [api:Member] class. It's better to use [api:DataObjectDecorator]
|
||||||
(see below).
|
(see below).
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
You can defined subclasses of `[api:Member]` to add extra fields or functionality to the built-in membership system.
|
You can defined subclasses of [api:Member] to add extra fields or functionality to the built-in membership system.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
class MyMember extends Member {
|
class MyMember extends Member {
|
||||||
@ -53,7 +53,7 @@ You can defined subclasses of `[api:Member]` to add extra fields or functionalit
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
To ensure that all new members are created using this class, put a call to `[api:Object::useCustomClass()]` in
|
To ensure that all new members are created using this class, put a call to [api:Object::useCustomClass()] in
|
||||||
(project)/_config.php:
|
(project)/_config.php:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -64,8 +64,8 @@ Note that if you want to look this class-name up, you can call Object::getCustom
|
|||||||
## Overloading getCMSFields()
|
## Overloading getCMSFields()
|
||||||
|
|
||||||
If you overload the built-in function getCMSFields(), then you can change the form that is used to view & edit member
|
If you overload the built-in function getCMSFields(), then you can change the form that is used to view & edit member
|
||||||
details in the newsletter system. This function returns a `[api:FieldSet]` object. You should generally start by calling
|
details in the newsletter system. This function returns a [api:FieldSet] object. You should generally start by calling
|
||||||
parent::getCMSFields() and manipulate the `[api:FieldSet]` from there.
|
parent::getCMSFields() and manipulate the [api:FieldSet] from there.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
function getCMSFields() {
|
function getCMSFields() {
|
||||||
@ -79,11 +79,11 @@ parent::getCMSFields() and manipulate the `[api:FieldSet]` from there.
|
|||||||
|
|
||||||
## Extending Member or DataObject?
|
## Extending Member or DataObject?
|
||||||
|
|
||||||
Basic rule: Class `[api:Member]` should just be extended for entities who have some kind of login.
|
Basic rule: Class [api:Member] should just be extended for entities who have some kind of login.
|
||||||
If you have different types of `[api:Member]`s in the system, you have to make sure that those with login-capabilities have
|
If you have different types of [api:Member]s in the system, you have to make sure that those with login-capabilities have
|
||||||
unique email-addresses (as this is used for login-credentials).
|
unique email-addresses (as this is used for login-credentials).
|
||||||
For persons without login-capabilities (e.g. for an address-database), you shouldn't extend `[api:Member]` to avoid conflicts
|
For persons without login-capabilities (e.g. for an address-database), you shouldn't extend [api:Member] to avoid conflicts
|
||||||
with the Member-database. This enables us to have a different subclass of `[api:Member]` for an email-address with login-data,
|
with the Member-database. This enables us to have a different subclass of [api:Member] for an email-address with login-data,
|
||||||
and another subclass for the same email-address in the address-database.
|
and another subclass for the same email-address in the address-database.
|
||||||
|
|
||||||
## Member Role Decorator
|
## Member Role Decorator
|
||||||
@ -96,9 +96,9 @@ class. A better way is to use role decorators to add this behaviour.
|
|||||||
// OR
|
// OR
|
||||||
Member::add_role('ForumRole');
|
Member::add_role('ForumRole');
|
||||||
|
|
||||||
A role decorator is simply a subclass of `[api:DataObjectDecorator]` that is designed to be used to add behaviour to `[api:Member]`.
|
A role decorator is simply a subclass of [api:DataObjectDecorator] that is designed to be used to add behaviour to [api:Member].
|
||||||
The roles affect the entire class - all members will get the additional behaviour. However, if you want to restrict
|
The roles affect the entire class - all members will get the additional behaviour. However, if you want to restrict
|
||||||
things, you should add appropriate `[api:Permission::checkMember()]` calls to the role's methods.
|
things, you should add appropriate [api:Permission::checkMember()] calls to the role's methods.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
class ForumRole extends DataObjectDecorator {
|
class ForumRole extends DataObjectDecorator {
|
||||||
@ -125,4 +125,4 @@ things, you should add appropriate `[api:Permission::checkMember()]` calls to th
|
|||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
`[api:Member]`
|
[api:Member]
|
||||||
|
@ -5,10 +5,10 @@
|
|||||||
*Replaces GenericDataAdmin in Silverstripe 2.3*
|
*Replaces GenericDataAdmin in Silverstripe 2.3*
|
||||||
|
|
||||||
The ModelAdmin provides a simple way to utilize the SilverStripe CMS UI with your own custom data models. The
|
The ModelAdmin provides a simple way to utilize the SilverStripe CMS UI with your own custom data models. The
|
||||||
ModelAdmin uses the `[api:DataObject]`'s Scaffolding to create the search fields, forms, and displayed data within the
|
ModelAdmin uses the [api:DataObject]'s Scaffolding to create the search fields, forms, and displayed data within the
|
||||||
CMS.
|
CMS.
|
||||||
|
|
||||||
In order to customize the ModelAdmin CMS interface you will need to understand how `[api:DataObject]` works.
|
In order to customize the ModelAdmin CMS interface you will need to understand how [api:DataObject] works.
|
||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
|
|
||||||
@ -35,11 +35,11 @@ data objects you want to scaffold an interface for:
|
|||||||
|
|
||||||
|
|
||||||
To add the ModelAdmin to your CMS menu, you simply need to define a couple of statics on your ModelAdmin subclass. See
|
To add the ModelAdmin to your CMS menu, you simply need to define a couple of statics on your ModelAdmin subclass. See
|
||||||
`[api:LeftAndMain]` on how to make your menu title translatable.
|
[api:LeftAndMain] on how to make your menu title translatable.
|
||||||
|
|
||||||
|
|
||||||
### Step 2
|
### Step 2
|
||||||
Add a `$searchable_fields` (See `[api:ModelAdmin::$searchable_fields]`) property to your data
|
Add a `$searchable_fields` (See [api:ModelAdmin::$searchable_fields]) property to your data
|
||||||
models, to define the fields and filters for the search interface:
|
models, to define the fields and filters for the search interface:
|
||||||
|
|
||||||
Datamodel `Product`:
|
Datamodel `Product`:
|
||||||
@ -97,19 +97,19 @@ If you are seeing a list of ID#s when creating new objects, ensure you have one
|
|||||||
You can customize the fields which are searchable for each managed DataObject class, as well as the ways in which the
|
You can customize the fields which are searchable for each managed DataObject class, as well as the ways in which the
|
||||||
fields are searched (e.g. "partial match", "fulltext", etc.) using `$searchable_fields`.
|
fields are searched (e.g. "partial match", "fulltext", etc.) using `$searchable_fields`.
|
||||||
|
|
||||||
* See `[api:DataObject]`
|
* See [api:DataObject]
|
||||||
|
|
||||||
![](_images/modeladmin_search.png)
|
![](_images/modeladmin_search.png)
|
||||||
|
|
||||||
## Summary Fields
|
## Summary Fields
|
||||||
|
|
||||||
Summary Fields are the columns which are shown in the `[api:TableListField]` when viewing DataObjects. These can be
|
Summary Fields are the columns which are shown in the [api:TableListField] when viewing DataObjects. These can be
|
||||||
customized for each `[api:DataObject]`'s search results using `$summary_fields`.
|
customized for each [api:DataObject]'s search results using `$summary_fields`.
|
||||||
|
|
||||||
* See `[api:DataObject]`
|
* See [api:DataObject]
|
||||||
|
|
||||||
## Related
|
## Related
|
||||||
|
|
||||||
* `[api:SearchContext]`
|
* [api:SearchContext]
|
||||||
* [genericviews Module] `http://silverstripe.org/generic-views-module`
|
* [genericviews Module] `http://silverstripe.org/generic-views-module`
|
||||||
* [Presentation about ModelAdmin at SupperHappyDevHouse Wellington](http://www.slideshare.net/chillu/modeladmin-in-silverstripe-23)
|
* [Presentation about ModelAdmin at SupperHappyDevHouse Wellington](http://www.slideshare.net/chillu/modeladmin-in-silverstripe-23)
|
||||||
|
@ -55,7 +55,7 @@ From a block that shows a summary of the page edits if administrator, nothing if
|
|||||||
|
|
||||||
Often you want to invalidate a cache when any in a set of objects change, or when the objects in a relationship change.
|
Often you want to invalidate a cache when any in a set of objects change, or when the objects in a relationship change.
|
||||||
To help do this, SilverStripe 2.4 also introduces the concept of Aggregates. These calculate and return SQL aggregates
|
To help do this, SilverStripe 2.4 also introduces the concept of Aggregates. These calculate and return SQL aggregates
|
||||||
on sets of `[api:DataObject]`s - the most useful for us being the Max aggregate.
|
on sets of [api:DataObject]s - the most useful for us being the Max aggregate.
|
||||||
|
|
||||||
For example, if we have a menu, we want that menu to update whenever _any_ page is edited, but would like to cache it
|
For example, if we have a menu, we want that menu to update whenever _any_ page is edited, but would like to cache it
|
||||||
otherwise. By using aggregates, that's easy
|
otherwise. By using aggregates, that's easy
|
||||||
|
@ -22,7 +22,7 @@ You can use whatever codes you like, but for the sanity of developers and users,
|
|||||||
|
|
||||||
## PermissionProvider
|
## PermissionProvider
|
||||||
|
|
||||||
`[api:PermissionProvider]` is an interface which lets you define a method *providePermissions()*. This method should return a
|
[api:PermissionProvider] is an interface which lets you define a method *providePermissions()*. This method should return a
|
||||||
map of permission code names with a human readable explanation of its purpose (see
|
map of permission code names with a human readable explanation of its purpose (see
|
||||||
[permissions:codes](/reference/permission)).
|
[permissions:codes](/reference/permission)).
|
||||||
|
|
||||||
@ -71,4 +71,4 @@ information on request objects.
|
|||||||
|
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:Permission]`
|
[api:Permission]
|
||||||
|
@ -8,7 +8,7 @@ hardcoding any references in the `<head>`-tag of your template, as it enables a
|
|||||||
|
|
||||||
## Including inside PHP Code
|
## Including inside PHP Code
|
||||||
It is common practice to include most Requirements either in the *init()*-method of your [controller](/topics/controller), or
|
It is common practice to include most Requirements either in the *init()*-method of your [controller](/topics/controller), or
|
||||||
as close to rendering as possible (e.g. in `[api:FormField]`
|
as close to rendering as possible (e.g. in [api:FormField]
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
Requirements::javascript("cms/javascript/LeftAndMain.js");
|
Requirements::javascript("cms/javascript/LeftAndMain.js");
|
||||||
@ -133,7 +133,7 @@ In your controller's init() function, add:
|
|||||||
|
|
||||||
## CMS Requirements
|
## CMS Requirements
|
||||||
|
|
||||||
The SilverStripe core includes a lot of Requirements by itself. Most of these are collated in `[api:LeftAndMain]` first.
|
The SilverStripe core includes a lot of Requirements by itself. Most of these are collated in [api:LeftAndMain] first.
|
||||||
|
|
||||||
## Motivation
|
## Motivation
|
||||||
|
|
||||||
@ -189,4 +189,4 @@ slightly different JS/CSS requirements, the whole lot will be refetched.
|
|||||||
nature of an ajax-request. Needs some more research
|
nature of an ajax-request. Needs some more research
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:Requirements]`
|
[api:Requirements]
|
||||||
|
@ -2,11 +2,11 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
`[api:RestfulService]` enables connecting to remote web services which supports REST interface and consume those web services
|
[api:RestfulService] enables connecting to remote web services which supports REST interface and consume those web services
|
||||||
(for example [Flickr](http://www.flickr.com/services/api/), [Youtube](http://code.google.com/apis/youtube/overview.html), Amazon and etc). `[api:RestfulService]` can parse the XML response (sorry no JSON support)
|
(for example [Flickr](http://www.flickr.com/services/api/), [Youtube](http://code.google.com/apis/youtube/overview.html), Amazon and etc). [api:RestfulService] can parse the XML response (sorry no JSON support)
|
||||||
returned from the web service. Further it supports caching of the response, and you can customize the cache interval.
|
returned from the web service. Further it supports caching of the response, and you can customize the cache interval.
|
||||||
|
|
||||||
To gain the functionality you can either create a new `[api:RestfulService]` object or create a class extending the
|
To gain the functionality you can either create a new [api:RestfulService] object or create a class extending the
|
||||||
RestfulService (see [flickrservice] `http://silverstripe.org/flickr-module` and
|
RestfulService (see [flickrservice] `http://silverstripe.org/flickr-module` and
|
||||||
[youtubeservice] `http://silverstripe.org/youtube-gallery-module/` modules).
|
[youtubeservice] `http://silverstripe.org/youtube-gallery-module/` modules).
|
||||||
|
|
||||||
@ -113,7 +113,7 @@ to get the value of entry node with the namespace media, use:
|
|||||||
|
|
||||||
### Handling Errors
|
### Handling Errors
|
||||||
|
|
||||||
If the web service returned an error (for example, API key not available or inadequate parameters) `[api:RestfulService]`
|
If the web service returned an error (for example, API key not available or inadequate parameters) [api:RestfulService]
|
||||||
could delgate the error handling to it's descendant class. To handle the errors define a function called errorCatch
|
could delgate the error handling to it's descendant class. To handle the errors define a function called errorCatch
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -143,9 +143,9 @@ If you want to bypass error handling on your sub-classes you could define that i
|
|||||||
|
|
||||||
## Other Uses
|
## Other Uses
|
||||||
|
|
||||||
### How to use `[api:RestfulService]` to easily embed an RSS feed
|
### How to use [api:RestfulService] to easily embed an RSS feed
|
||||||
|
|
||||||
`[api:RestfulService]` can be used to easily embed an RSS feed (since it's also an xml response) from a site
|
[api:RestfulService] can be used to easily embed an RSS feed (since it's also an xml response) from a site
|
||||||
such as del.icio.us
|
such as del.icio.us
|
||||||
|
|
||||||
Put something like this code in mysite/code/Page.php inside class Page_Controller
|
Put something like this code in mysite/code/Page.php inside class Page_Controller
|
||||||
@ -177,4 +177,4 @@ Put something like this code in mysite/templates/Layout/HomePage.ss:
|
|||||||
|
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:RestfulService]`
|
[api:RestfulService]
|
||||||
|
@ -2,12 +2,12 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
Generating RSS/Atom-feeds is just a matter of rendering a `[api:DataObject]` and the Page Comment Interface.
|
Generating RSS/Atom-feeds is just a matter of rendering a [api:DataObject] and the Page Comment Interface.
|
||||||
Handled through the `[api:RSSFeed]` class.
|
Handled through the [api:RSSFeed] class.
|
||||||
|
|
||||||
`[api:RSSFeed]` doesn't limit you to generating "article-based" feeds, it is just as easy to create a feed of your current
|
[api:RSSFeed] doesn't limit you to generating "article-based" feeds, it is just as easy to create a feed of your current
|
||||||
staff-members. The only logical limitation here is that every item in the RSS-feed should be accessible through a URL on
|
staff-members. The only logical limitation here is that every item in the RSS-feed should be accessible through a URL on
|
||||||
your website, so its advisable to just create feeds from subclasses of `[api:SiteTree]`.
|
your website, so its advisable to just create feeds from subclasses of [api:SiteTree].
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
@ -31,7 +31,7 @@ your website, so its advisable to just create feeds from subclasses of `[api:Sit
|
|||||||
### Example of showing the 10 most recently updated pages
|
### Example of showing the 10 most recently updated pages
|
||||||
|
|
||||||
|
|
||||||
You can use `[api:RSSFeed]` to easily create a feed showing your latest Page updates. Just change mysite/code/Page.php to
|
You can use [api:RSSFeed] to easily create a feed showing your latest Page updates. Just change mysite/code/Page.php to
|
||||||
something like this:
|
something like this:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -72,7 +72,7 @@ can also do http://www.yoursite.com/PageComment/rss?pageid=46 where pageid is th
|
|||||||
|
|
||||||
## External Sources
|
## External Sources
|
||||||
|
|
||||||
`[api:RSSFeed]` only creates feeds from your own data. We've included the [SimplePie](http://simplepie.org) RSS-parser for
|
[api:RSSFeed] only creates feeds from your own data. We've included the [SimplePie](http://simplepie.org) RSS-parser for
|
||||||
accessing feeds from external sources.
|
accessing feeds from external sources.
|
||||||
|
|
||||||
|
|
||||||
@ -81,4 +81,4 @@ accessing feeds from external sources.
|
|||||||
* [blog module](http://silverstripe.org/blog-module)
|
* [blog module](http://silverstripe.org/blog-module)
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:RSSFeed]`
|
[api:RSSFeed]
|
||||||
|
@ -2,17 +2,17 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
Manages searching of properties on one or more `[api:DataObject]` types, based on a given set of input parameters.
|
Manages searching of properties on one or more [api:DataObject] types, based on a given set of input parameters.
|
||||||
`[api:SearchContext]` is intentionally decoupled from any controller-logic,
|
[api:SearchContext] is intentionally decoupled from any controller-logic,
|
||||||
it just receives a set of search parameters and an object class it acts on.
|
it just receives a set of search parameters and an object class it acts on.
|
||||||
|
|
||||||
The default output of a `[api:SearchContext]` is either a `[api:SQLQuery]` object for further refinement, or a
|
The default output of a [api:SearchContext] is either a [api:SQLQuery] object for further refinement, or a
|
||||||
`[api:DataObject]` instance.
|
[api:DataObject] instance.
|
||||||
|
|
||||||
In case you need multiple contexts, consider namespacing your request parameters by using `FieldSet->namespace()` on
|
In case you need multiple contexts, consider namespacing your request parameters by using `FieldSet->namespace()` on
|
||||||
the $fields constructor parameter.
|
the $fields constructor parameter.
|
||||||
|
|
||||||
`[api:SearchContext]` is mainly used by `[api:ModelAdmin]`, our generic data administration interface. Another
|
[api:SearchContext] is mainly used by [api:ModelAdmin], our generic data administration interface. Another
|
||||||
implementation can be found in generic frontend search forms through the [genericviews] `http://silverstripe.org/generic-views-module` module.
|
implementation can be found in generic frontend search forms through the [genericviews] `http://silverstripe.org/generic-views-module` module.
|
||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
@ -29,7 +29,7 @@ Getting results
|
|||||||
|
|
||||||
### Defining fields on your DataObject
|
### Defining fields on your DataObject
|
||||||
|
|
||||||
See `[api:DataObject::$searchable_fields]`.
|
See [api:DataObject::$searchable_fields].
|
||||||
|
|
||||||
### Customizing fields and filters
|
### Customizing fields and filters
|
||||||
|
|
||||||
@ -180,13 +180,13 @@ Results.PaginationSummary(4) defines how many pages the search will show in the
|
|||||||
|
|
||||||
## Available SearchFilters
|
## Available SearchFilters
|
||||||
|
|
||||||
See `[api:SearchFilter]` API Documentation
|
See [api:SearchFilter] API Documentation
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:SearchContext]`
|
[api:SearchContext]
|
||||||
|
|
||||||
## Related
|
## Related
|
||||||
|
|
||||||
* `[api:ModelAdmin]`
|
* [api:ModelAdmin]
|
||||||
* `[api:RestfulServer]`
|
* [api:RestfulServer]
|
||||||
* [Tutorial: Site Search](/tutorials/site-search)
|
* [Tutorial: Site Search](/tutorials/site-search)
|
||||||
|
@ -91,4 +91,4 @@ file and add class's as you need them inside that for each report.
|
|||||||
* More examples.
|
* More examples.
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:ReportAdmin]`
|
[api:ReportAdmin]
|
||||||
|
@ -2,12 +2,12 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
The `[api:SiteConfig]` panel was introduced in 2.4 for providing a generic interface for managing site wide settings or
|
The [api:SiteConfig] panel was introduced in 2.4 for providing a generic interface for managing site wide settings or
|
||||||
functionality which is used throughout the site. Out of the box it provides 2 fields 'Site Name' and 'Site Tagline'.
|
functionality which is used throughout the site. Out of the box it provides 2 fields 'Site Name' and 'Site Tagline'.
|
||||||
|
|
||||||
## Accessing `[api:SiteConfig]` Options
|
## Accessing [api:SiteConfig] Options
|
||||||
|
|
||||||
You can access `[api:SiteConfig]` options from any SS template by using the function $SiteConfig.FieldName
|
You can access [api:SiteConfig] options from any SS template by using the function $SiteConfig.FieldName
|
||||||
|
|
||||||
:::ss
|
:::ss
|
||||||
$SiteConfig.Title
|
$SiteConfig.Title
|
||||||
@ -28,7 +28,7 @@ Or if you want to access variables in the PHP you can do
|
|||||||
$config->Title
|
$config->Title
|
||||||
|
|
||||||
|
|
||||||
## Extending `[api:SiteConfig]`
|
## Extending [api:SiteConfig]
|
||||||
|
|
||||||
To extend the options available in the panel you can define your own fields via an Extension.
|
To extend the options available in the panel you can define your own fields via an Extension.
|
||||||
|
|
||||||
@ -58,13 +58,13 @@ Then add a link to your extension in the _config.php file like below.
|
|||||||
Object::add_extension('SiteConfig', 'CustomSiteConfig');
|
Object::add_extension('SiteConfig', 'CustomSiteConfig');
|
||||||
|
|
||||||
|
|
||||||
This tells SilverStripe to add the CustomSiteConfig extension to the `[api:SiteConfig]` class.
|
This tells SilverStripe to add the CustomSiteConfig extension to the [api:SiteConfig] class.
|
||||||
|
|
||||||
After adding those two pieces of code, rebuild your database by visiting http://yoursite.com/dev/build and then reload
|
After adding those two pieces of code, rebuild your database by visiting http://yoursite.com/dev/build and then reload
|
||||||
the admin interface. You may need to reload it with a ?flush=1 on the end.
|
the admin interface. You may need to reload it with a ?flush=1 on the end.
|
||||||
|
|
||||||
You can define as many extensions for `[api:SiteConfig]` as you need. For example if you are developing a module you can define
|
You can define as many extensions for [api:SiteConfig] as you need. For example if you are developing a module you can define
|
||||||
your own global settings for the dashboard.
|
your own global settings for the dashboard.
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:SiteConfig]`
|
[api:SiteConfig]
|
||||||
|
@ -144,7 +144,7 @@ it is a good starting point, for choosing your customisation.
|
|||||||
|
|
||||||
### Automatic Child Selection
|
### Automatic Child Selection
|
||||||
|
|
||||||
By default, `[api:SiteTree]` class to build a tree using the ParentID field. However, sometimes, you want to change
|
By default, [api:SiteTree] class to build a tree using the ParentID field. However, sometimes, you want to change
|
||||||
this default behaviour.
|
this default behaviour.
|
||||||
|
|
||||||
For example, in our e-commerce module, we use a many-to-many join, Product::Parents, to let you put Products in multiple
|
For example, in our e-commerce module, we use a many-to-many join, Product::Parents, to let you put Products in multiple
|
||||||
@ -154,7 +154,7 @@ groups. Here's how to implement such a change:
|
|||||||
store parents.
|
store parents.
|
||||||
|
|
||||||
* **Define stageChildren method:** This method should return the children of the current page, for the current version.
|
* **Define stageChildren method:** This method should return the children of the current page, for the current version.
|
||||||
If you use DataObject::get, the `[api:Versioned]` class will rewrite your query to access the live site when
|
If you use DataObject::get, the [api:Versioned] class will rewrite your query to access the live site when
|
||||||
appropriate.
|
appropriate.
|
||||||
|
|
||||||
* **Define liveChildren method:** The method should return the children of the current page, for the live site.
|
* **Define liveChildren method:** The method should return the children of the current page, for the live site.
|
||||||
@ -206,7 +206,7 @@ Example:
|
|||||||
|
|
||||||
### Multiple parents in the tree
|
### Multiple parents in the tree
|
||||||
|
|
||||||
The `[api:LeftAndMain]` tree supports multiple parents. We overload CMSTreeClasses and make it include "manyparents" in
|
The [api:LeftAndMain] tree supports multiple parents. We overload CMSTreeClasses and make it include "manyparents" in
|
||||||
the class list.
|
the class list.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -295,4 +295,4 @@ and the specific group.
|
|||||||
Clean up this documentation
|
Clean up this documentation
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:Sitetree]`
|
[api:Sitetree]
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
An object representing a SQL query. It is easier to deal with object-wrappers than string-parsing a raw SQL-query. This
|
An object representing a SQL query. It is easier to deal with object-wrappers than string-parsing a raw SQL-query. This
|
||||||
object is used by `[api:DataObject]`, though...
|
object is used by [api:DataObject], though...
|
||||||
|
|
||||||
A word of caution: Dealing with low-level SQL is not encouraged in the SilverStripe [datamodel](/topics/datamodel) for various
|
A word of caution: Dealing with low-level SQL is not encouraged in the SilverStripe [datamodel](/topics/datamodel) for various
|
||||||
reasons. You'll break the behaviour of:
|
reasons. You'll break the behaviour of:
|
||||||
@ -12,7 +12,7 @@ reasons. You'll break the behaviour of:
|
|||||||
* DataObject::onBeforeWrite/onBeforeDelete
|
* DataObject::onBeforeWrite/onBeforeDelete
|
||||||
* Automatic casting
|
* Automatic casting
|
||||||
* Default-setting through object-model
|
* Default-setting through object-model
|
||||||
* `[api:DataObject]`
|
* [api:DataObject]
|
||||||
* Database abstraction
|
* Database abstraction
|
||||||
|
|
||||||
We'll explain some ways to use *SELECT* with the full power of SQL, but still maintain a connection to the SilverStripe
|
We'll explain some ways to use *SELECT* with the full power of SQL, but still maintain a connection to the SilverStripe
|
||||||
@ -62,7 +62,7 @@ We'll explain some ways to use *SELECT* with the full power of SQL, but still ma
|
|||||||
|
|
||||||
## Working with results
|
## Working with results
|
||||||
|
|
||||||
The result is an array lightly wrapped in a database-specific subclass of `[api:Query]`. This class implements the
|
The result is an array lightly wrapped in a database-specific subclass of [api:Query]. This class implements the
|
||||||
*Iterator*-interface defined in PHP5, and provides convenience-methods for accessing the data.
|
*Iterator*-interface defined in PHP5, and provides convenience-methods for accessing the data.
|
||||||
|
|
||||||
### Iterating
|
### Iterating
|
||||||
@ -87,7 +87,7 @@ Raw SQL is handy for performance-optimized calls.
|
|||||||
return $sqlQuery->execute()->value();
|
return $sqlQuery->execute()->value();
|
||||||
}
|
}
|
||||||
|
|
||||||
Way faster than dealing with `[api:DataObject]`s, but watch out for premature optimisation:
|
Way faster than dealing with [api:DataObject]s, but watch out for premature optimisation:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$players = $myTeam->Players();
|
$players = $myTeam->Players();
|
||||||
@ -142,10 +142,10 @@ This form of building a query has the following advantages:
|
|||||||
* Selection of *ID*, *ClassName*, *RecordClassName*, which are necessary to use *buildDataObjectSet* later on
|
* Selection of *ID*, *ClassName*, *RecordClassName*, which are necessary to use *buildDataObjectSet* later on
|
||||||
* Filtering records for correct *ClassName*
|
* Filtering records for correct *ClassName*
|
||||||
|
|
||||||
### Transforming a result to `[api:DataObjectSet]`
|
### Transforming a result to [api:DataObjectSet]
|
||||||
|
|
||||||
This is a commonly used technique inside SilverStripe: Use raw SQL, but transfer the resulting rows back into
|
This is a commonly used technique inside SilverStripe: Use raw SQL, but transfer the resulting rows back into
|
||||||
`[api:DataObject]`s.
|
[api:DataObject]s.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$sqlQuery = new SQLQuery();
|
$sqlQuery = new SQLQuery();
|
||||||
@ -188,21 +188,21 @@ This is a commonly used technique inside SilverStripe: Use raw SQL, but transfer
|
|||||||
* Not all object-properties accessible: You need to take care of selecting the right stuff yourself
|
* Not all object-properties accessible: You need to take care of selecting the right stuff yourself
|
||||||
* Overlayed object-properties: If you *LEFT JOIN* a table which also has a column 'Birthdate' and do a global select on
|
* Overlayed object-properties: If you *LEFT JOIN* a table which also has a column 'Birthdate' and do a global select on
|
||||||
this table, you might not be able to access original object-properties.
|
this table, you might not be able to access original object-properties.
|
||||||
* You can't create `[api:DataObject]`s where no scalar record-data is available, e.g. when using *GROUP BY*
|
* You can't create [api:DataObject]s where no scalar record-data is available, e.g. when using *GROUP BY*
|
||||||
* Naming conflicts with custom getters: A getter like Player->getName() will overlay the column-data selected in the
|
* Naming conflicts with custom getters: A getter like Player->getName() will overlay the column-data selected in the
|
||||||
above example
|
above example
|
||||||
|
|
||||||
Be careful when saving back `[api:DataObject]`s created through *buildDataObjectSet*, you might get strange side-effects due to
|
Be careful when saving back [api:DataObject]s created through *buildDataObjectSet*, you might get strange side-effects due to
|
||||||
the issues noted above.
|
the issues noted above.
|
||||||
## Using FormFields with custom SQL
|
## Using FormFields with custom SQL
|
||||||
|
|
||||||
Some subclasses of `[api:FormField]` for ways to create sophisticated report-tables based on SQL.
|
Some subclasses of [api:FormField] for ways to create sophisticated report-tables based on SQL.
|
||||||
|
|
||||||
## Related
|
## Related
|
||||||
|
|
||||||
* [datamodel](/topics/datamodel)
|
* [datamodel](/topics/datamodel)
|
||||||
* `[api:DataObject]`
|
* [api:DataObject]
|
||||||
* [database-structure](database-structure)
|
* [database-structure](database-structure)
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:SQLQuery]`
|
[api:SQLQuery]
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
Many sites get too much traffic to justify dynamically sending every request. Caching is needed. Static Publishing
|
Many sites get too much traffic to justify dynamically sending every request. Caching is needed. Static Publishing
|
||||||
will generate static versions of your content (HTML) that can be served without ever hitting PHP or the Database.
|
will generate static versions of your content (HTML) that can be served without ever hitting PHP or the Database.
|
||||||
|
|
||||||
See `[api:StaticExporter]` for a less flexible, but easier way of building a local static cache from all of
|
See [api:StaticExporter] for a less flexible, but easier way of building a local static cache from all of
|
||||||
your pages.
|
your pages.
|
||||||
|
|
||||||
See [Partial-Caching](partial-caching) for a much more flexible way of building in caching without statically delivering
|
See [Partial-Caching](partial-caching) for a much more flexible way of building in caching without statically delivering
|
||||||
@ -262,7 +262,7 @@ example] `http://open.silverstripe.com/browser/modules/cms/trunk/code/staticpubl
|
|||||||
## Deployment
|
## Deployment
|
||||||
|
|
||||||
Once you've set up your rewrite rules and defined which pages need caching, you can build the static HTML files. This is
|
Once you've set up your rewrite rules and defined which pages need caching, you can build the static HTML files. This is
|
||||||
done by the `[api:RebuildStaticCacheTask]`
|
done by the [api:RebuildStaticCacheTask]
|
||||||
|
|
||||||
Execution via URL
|
Execution via URL
|
||||||
|
|
||||||
@ -281,8 +281,8 @@ the cache.
|
|||||||
|
|
||||||
## Related
|
## Related
|
||||||
|
|
||||||
* `[api:StaticExporter]`
|
* [api:StaticExporter]
|
||||||
* [Partial-Caching](partial-caching)
|
* [Partial-Caching](partial-caching)
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
* `[api:StaticPublisher]`
|
* [api:StaticPublisher]
|
||||||
|
@ -2,10 +2,10 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
`[api:TableField]` behaves in the same manner as `[api:TableListField]`, however allows the editing of existing and adding of
|
[api:TableField] behaves in the same manner as [api:TableListField], however allows the editing of existing and adding of
|
||||||
new rows. The data is saved back by the surrounding form-saving (mostly EditForm->save).
|
new rows. The data is saved back by the surrounding form-saving (mostly EditForm->save).
|
||||||
|
|
||||||
See `[api:TableListField]` for more documentation on the base-class
|
See [api:TableListField] for more documentation on the base-class
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
@ -44,7 +44,7 @@ is created.
|
|||||||
|
|
||||||
### Row Transformation
|
### Row Transformation
|
||||||
|
|
||||||
You can apply a `[api:FormTransformation]` to any given field,
|
You can apply a [api:FormTransformation] to any given field,
|
||||||
based on a eval()ed php-rule. You can access all columns on the generated DataObjects here.
|
based on a eval()ed php-rule. You can access all columns on the generated DataObjects here.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -59,23 +59,23 @@ based on a eval()ed php-rule. You can access all columns on the generated DataOb
|
|||||||
### Required Fields
|
### Required Fields
|
||||||
|
|
||||||
Due to the nested nature of this fields dataset, you can't set any required columns as usual with the
|
Due to the nested nature of this fields dataset, you can't set any required columns as usual with the
|
||||||
`[api:RequiredFields]`** on the TableField-instance for this.
|
[api:RequiredFields]** on the TableField-instance for this.
|
||||||
Note: You still have to attach some form of `[api:Validator]` to the form to trigger any validation on this field.
|
Note: You still have to attach some form of [api:Validator] to the form to trigger any validation on this field.
|
||||||
|
|
||||||
|
|
||||||
### Nested Table Fields
|
### Nested Table Fields
|
||||||
|
|
||||||
When you have `[api:TableField]` inside a `[api:ComplexTableField]`, the parent ID may not be known in your
|
When you have [api:TableField] inside a [api:ComplexTableField], the parent ID may not be known in your
|
||||||
getCMSFields() method. In these cases, you can set a value to '$RecordID' in your `[api:TableField]` extra data, and this
|
getCMSFields() method. In these cases, you can set a value to '$RecordID' in your [api:TableField] extra data, and this
|
||||||
will be populated with the newly created record id upon save.
|
will be populated with the newly created record id upon save.
|
||||||
|
|
||||||
## Known Issues
|
## Known Issues
|
||||||
|
|
||||||
* A `[api:TableField]` doesn't reload any submitted form-data if the saving is interrupted by a failed validation. After
|
* A [api:TableField] doesn't reload any submitted form-data if the saving is interrupted by a failed validation. After
|
||||||
refreshing the form with the validation-errors, the `[api:TableField]` will be blank again.
|
refreshing the form with the validation-errors, the [api:TableField] will be blank again.
|
||||||
* You can't add **visible default data** to columns in a `[api:TableField]`, please use *setExtraData*
|
* You can't add **visible default data** to columns in a [api:TableField], please use *setExtraData*
|
||||||
|
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
`[api:TableField]`
|
[api:TableField]
|
||||||
|
@ -2,13 +2,13 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
Form field that embeds a list of `[api:DataObject]`s into a form, such as a member list or a file list.
|
Form field that embeds a list of [api:DataObject]s into a form, such as a member list or a file list.
|
||||||
Provides customizeable columns, record-deletion by ajax, paging, sorting, CSV-export, printing, input by
|
Provides customizeable columns, record-deletion by ajax, paging, sorting, CSV-export, printing, input by
|
||||||
`[api:DataObject]` or raw SQL.
|
[api:DataObject] or raw SQL.
|
||||||
|
|
||||||
## Example
|
## Example
|
||||||
|
|
||||||
Here's an example of a full featured `[api:TableListField]` implementation. It features editing members in the database
|
Here's an example of a full featured [api:TableListField] implementation. It features editing members in the database
|
||||||
directly as a button on each record, as well as filtering, and sorting. It also makes use of the 'export' permission,
|
directly as a button on each record, as well as filtering, and sorting. It also makes use of the 'export' permission,
|
||||||
allowing export of data as a CSV.
|
allowing export of data as a CSV.
|
||||||
|
|
||||||
@ -90,7 +90,7 @@ For more information on each of the features used in the example, you can read b
|
|||||||
$customCsvQuery->select[] = "CONCAT(col1,col2) AS MyCustomSQLColumn";
|
$customCsvQuery->select[] = "CONCAT(col1,col2) AS MyCustomSQLColumn";
|
||||||
$myTableListField->setCustomCsvQuery($customQuery);
|
$myTableListField->setCustomCsvQuery($customQuery);
|
||||||
|
|
||||||
`[api:TableListField]` also tries to resolve Component-relations(has_one, has_many) and custom getters automatically:
|
[api:TableListField] also tries to resolve Component-relations(has_one, has_many) and custom getters automatically:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$myTableListField = new TableListField(
|
$myTableListField = new TableListField(
|
||||||
@ -139,7 +139,7 @@ Example (sorting by "FirstName" column):
|
|||||||
);
|
);
|
||||||
|
|
||||||
|
|
||||||
If you want to sort by custom getters in your `[api:DataObject]`, please reformulate them to a custom SQL column. This
|
If you want to sort by custom getters in your [api:DataObject], please reformulate them to a custom SQL column. This
|
||||||
restriction is needed to avoid performance-hits by caching and sorting potentially large datasets on PHP-level.
|
restriction is needed to avoid performance-hits by caching and sorting potentially large datasets on PHP-level.
|
||||||
|
|
||||||
### Casting
|
### Casting
|
||||||
@ -155,7 +155,7 @@ Column-values can be casted, based on the casting-types available through DBObje
|
|||||||
|
|
||||||
### Permissions
|
### Permissions
|
||||||
|
|
||||||
Permissions vary in different `[api:TableListField]`-implementations, and are evaluated in the template.
|
Permissions vary in different [api:TableListField]-implementations, and are evaluated in the template.
|
||||||
By default, all listed permissions are enabled.
|
By default, all listed permissions are enabled.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -237,7 +237,7 @@ implement averages etc.
|
|||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
In `[api:TableListField]`-implementation, these summaries also react to changes in input-fields by javascript.
|
In [api:TableListField]-implementation, these summaries also react to changes in input-fields by javascript.
|
||||||
Available methods:
|
Available methods:
|
||||||
|
|
||||||
* sum
|
* sum
|
||||||
@ -245,7 +245,7 @@ Available methods:
|
|||||||
|
|
||||||
### Grouping
|
### Grouping
|
||||||
|
|
||||||
Used to group by a specific column in the `[api:DataObject]` and create partial summaries.
|
Used to group by a specific column in the [api:DataObject] and create partial summaries.
|
||||||
Please use only together with addSummary().
|
Please use only together with addSummary().
|
||||||
(Automatically disables sorting).
|
(Automatically disables sorting).
|
||||||
|
|
||||||
@ -257,7 +257,7 @@ Please use only together with addSummary().
|
|||||||
|
|
||||||
### Custom Sorting
|
### Custom Sorting
|
||||||
|
|
||||||
Please subclass `[api:TableListField]` to implement custom sorting, following the naming-convention
|
Please subclass [api:TableListField] to implement custom sorting, following the naming-convention
|
||||||
"`colFunction_<yourFunctionName>`".
|
"`colFunction_<yourFunctionName>`".
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -285,4 +285,4 @@ You can exchange the used template, e.g. to change applied CSS-classes or the HT
|
|||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
`[api:TableListField]`
|
[api:TableListField]
|
||||||
|
@ -88,7 +88,7 @@ OtherContent:
|
|||||||
### Removing the typography class
|
### Removing the typography class
|
||||||
|
|
||||||
Sometimes, it's not enough to add a class, you also want to remove the typography class. You can use the
|
Sometimes, it's not enough to add a class, you also want to remove the typography class. You can use the
|
||||||
`[api:HTMLEditorField]` method setCSSClass.
|
[api:HTMLEditorField] method setCSSClass.
|
||||||
|
|
||||||
This example sets another CSS class typographybis:
|
This example sets another CSS class typographybis:
|
||||||
|
|
||||||
|
@ -29,7 +29,7 @@ Append the option and corresponding value to your URL in your browser's address
|
|||||||
| isDev | | 1 | | Put the site into [development mode](/topics/debugging), enabling debugging messages to the browser on a live server. For security, you'll be asked to log in with an administrator log-in |
|
| isDev | | 1 | | Put the site into [development mode](/topics/debugging), enabling debugging messages to the browser on a live server. For security, you'll be asked to log in with an administrator log-in |
|
||||||
| isTest | | 1 | | Put the site into [test mode](/topics/debugging), enabling debugging messages to the admin email and generic errors to the browser on a live server |
|
| isTest | | 1 | | Put the site into [test mode](/topics/debugging), enabling debugging messages to the admin email and generic errors to the browser on a live server |
|
||||||
| debug | | 1 | | Show a collection of debugging information about the director / controller operation |
|
| debug | | 1 | | Show a collection of debugging information about the director / controller operation |
|
||||||
| debug_request | | 1 | | Show all steps of the request from initial `[api:HTTPRequest]` to `[api:Controller]` to Template Rendering |
|
| debug_request | | 1 | | Show all steps of the request from initial [api:HTTPRequest] to [api:Controller] to Template Rendering |
|
||||||
|
|
||||||
## Classes and Objects
|
## Classes and Objects
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Versioned
|
# Versioned
|
||||||
|
|
||||||
The Versioned class is a `[api:DataObject]` that adds versioning and staging capabilities to the objects.
|
The Versioned class is a [api:DataObject] that adds versioning and staging capabilities to the objects.
|
||||||
|
|
||||||
## Trapping the publication event
|
## Trapping the publication event
|
||||||
|
|
||||||
|
@ -35,7 +35,7 @@ privileges from its parent group.
|
|||||||
|
|
||||||
## Permission checking is at class level
|
## Permission checking is at class level
|
||||||
|
|
||||||
SilverStripe provides a security mechanism via the *Permission::check* method (see `[api:LeftAndMain]` for examples on how
|
SilverStripe provides a security mechanism via the *Permission::check* method (see [api:LeftAndMain] for examples on how
|
||||||
the admin screens work)
|
the admin screens work)
|
||||||
|
|
||||||
(next step -- go from *Permission::checkMember*...)
|
(next step -- go from *Permission::checkMember*...)
|
||||||
@ -48,14 +48,14 @@ works.
|
|||||||
|
|
||||||
### Loading the admin page: looking at security
|
### Loading the admin page: looking at security
|
||||||
|
|
||||||
If you go to [your site]/admin *Director.php* maps the 'admin' URL request through a `[api:Director]` rule to the
|
If you go to [your site]/admin *Director.php* maps the 'admin' URL request through a [api:Director] rule to the
|
||||||
`[api:CMSMain]` controller (see `[api:CMSMain]`, with no arguments).
|
[api:CMSMain] controller.
|
||||||
|
|
||||||
*CMSMain.init()* calls its parent which, of all things is called `[api:LeftAndMain]`. It's in `[api:LeftAndMain]` that the
|
*CMSMain.init()* calls its parent which, of all things is called [api:LeftAndMain]. It's in [api:LeftAndMain] that the
|
||||||
important security checks are made by calling *Permission::check*.
|
important security checks are made by calling *Permission::check*.
|
||||||
|
|
||||||
`[api:Security::permissionFailure]` is the next utility function you can use to redirect to the login form.
|
[api:Security::permissionFailure()] is the next utility function you can use to redirect to the login form.
|
||||||
|
|
||||||
### Customizing Access Checks in CMS Classes
|
### Customizing Access Checks in CMS Classes
|
||||||
|
|
||||||
see `[api:LeftAndMain]`
|
see [api:LeftAndMain]
|
||||||
|
@ -12,7 +12,7 @@ CMS"] `http://www.silverstripe.com/core-team-discussion/flat/2723` for further r
|
|||||||
|
|
||||||
In addition to these principle, some settings are
|
In addition to these principle, some settings are
|
||||||
* Author-level configuration like interface language or date/time formats can be performed in the CMS "My Profile" section (`admin/myprofile`).
|
* Author-level configuration like interface language or date/time formats can be performed in the CMS "My Profile" section (`admin/myprofile`).
|
||||||
* Group-related configuration like `[api:HTMLEditorField]` settings can be found in the "Security" section (`admin/security`).
|
* Group-related configuration like [api:HTMLEditorField] settings can be found in the "Security" section (`admin/security`).
|
||||||
* Site-wide settings like page titles can be set (and extended) on the root tree element in the CMS "Content" section (through the [siteconfig](/reference/siteconfig) API).
|
* Site-wide settings like page titles can be set (and extended) on the root tree element in the CMS "Content" section (through the [siteconfig](/reference/siteconfig) API).
|
||||||
|
|
||||||
## _ss_environment.php
|
## _ss_environment.php
|
||||||
@ -21,7 +21,7 @@ See [environment-management](/topics/environment-management).
|
|||||||
|
|
||||||
## mysite/_config.php
|
## mysite/_config.php
|
||||||
|
|
||||||
This file is detected in each folder by `[api:ManifestBuilder]`. This way, every toplevel-folder (=module)
|
This file is detected in each folder by [api:ManifestBuilder]. This way, every toplevel-folder (=module)
|
||||||
can have independent configuration-rules.
|
can have independent configuration-rules.
|
||||||
|
|
||||||
|
|
||||||
@ -66,8 +66,8 @@ Some constants are user-defineable within *_ss_environment.php*.
|
|||||||
|
|
||||||
## User-level: Member-object
|
## User-level: Member-object
|
||||||
|
|
||||||
All user-related preferences are stored as a property of the `[api:Member]`-class (and as a database-column in the
|
All user-related preferences are stored as a property of the [api:Member]-class (and as a database-column in the
|
||||||
*Member*-table). You can "mix in" your custom preferences by using `[api:DataObject]` for details.
|
*Member*-table). You can "mix in" your custom preferences by using [api:DataObject] for details.
|
||||||
|
|
||||||
## Permissions
|
## Permissions
|
||||||
|
|
||||||
|
@ -130,9 +130,9 @@ permission checks.
|
|||||||
|
|
||||||
## URL Handling
|
## URL Handling
|
||||||
|
|
||||||
In the above example the URLs were configured using the `[api:Director]` rules in the **_config.php** file.
|
In the above example the URLs were configured using the [api:Director] rules in the **_config.php** file.
|
||||||
Alternatively you can specify these in your Controller class via the **$url_handlers** static array (which gets
|
Alternatively you can specify these in your Controller class via the **$url_handlers** static array (which gets
|
||||||
processed by the `[api:RequestHandler]`).
|
processed by the [api:RequestHandler]).
|
||||||
|
|
||||||
This is useful when you want to subvert the fixed action mapping of `fastfood/order/*` to the function **order**. In
|
This is useful when you want to subvert the fixed action mapping of `fastfood/order/*` to the function **order**. In
|
||||||
the case below we also want any orders coming through `/fastfood/drivethrough/` to use the same order function.
|
the case below we also want any orders coming through `/fastfood/drivethrough/` to use the same order function.
|
||||||
@ -150,7 +150,7 @@ the case below we also want any orders coming through `/fastfood/drivethrough/`
|
|||||||
|
|
||||||
## URL Patterns
|
## URL Patterns
|
||||||
|
|
||||||
The `[api:RequestHandler]` class will parse all rules you specify against the following patterns.
|
The [api:RequestHandler] class will parse all rules you specify against the following patterns.
|
||||||
|
|
||||||
**A rule must always start with alphabetical ([A-Za-z]) characters or a $Variable declaration**
|
**A rule must always start with alphabetical ([A-Za-z]) characters or a $Variable declaration**
|
||||||
|
|
||||||
@ -181,10 +181,10 @@ You can use the `debug_request=1` switch from the [urlvariabletools](/reference/
|
|||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
`[api:Controller]`
|
[api:Controller]
|
||||||
|
|
||||||
## Links
|
## Links
|
||||||
|
|
||||||
* `[api:Director]` class
|
* [api:Director] class
|
||||||
* [execution-pipeline](/reference/execution-pipeline)
|
* [execution-pipeline](/reference/execution-pipeline)
|
||||||
* [URL Handling in Controllers](http://maetl.net/silverstripe-url-handling) by maetl
|
* [URL Handling in Controllers](http://maetl.net/silverstripe-url-handling) by maetl
|
||||||
|
@ -1,29 +1,29 @@
|
|||||||
# Data Types
|
# Data Types
|
||||||
|
|
||||||
These are the data-types that you can use when defining your data objects. They are all subclasses of `[api:DBField]`
|
These are the data-types that you can use when defining your data objects. They are all subclasses of [api:DBField]
|
||||||
for introducing their usage.
|
for introducing their usage.
|
||||||
|
|
||||||
|
|
||||||
## Types
|
## Types
|
||||||
|
|
||||||
* `[api:Varchar]`: A variable-length string of up to 255 characters, designed to store raw text
|
* [api:Varchar]: A variable-length string of up to 255 characters, designed to store raw text
|
||||||
* `[api:Text]`: A variable-length string of up to 2 megabytes, designed to store raw text
|
* [api:Text]: A variable-length string of up to 2 megabytes, designed to store raw text
|
||||||
* `[api:HTMLVarchar]`: A variable-length string of up to 255 characters, designed to store HTML
|
* [api:HTMLVarchar]`: A variable-length string of up to 255 characters, designed to store HTML
|
||||||
* `[api:HTMLText]`: A variable-length string of up to 2 megabytes, designed to store HTML
|
* [api:HTMLText]: A variable-length string of up to 2 megabytes, designed to store HTML
|
||||||
* `[api:Enum]`: An enumeration of a set of strings
|
* [api:Enum]: An enumeration of a set of strings
|
||||||
* `[api:Boolean]`: A boolean field.
|
* [api:Boolean]: A boolean field.
|
||||||
* `[api:Int]`: An integer field.
|
* [api:Int]: An integer field.
|
||||||
* `[api:Decimal]`: A decimal number.
|
* [api:Decimal]: A decimal number.
|
||||||
* `[api:Currency]`: A number with 2 decimal points of precision, designed to store currency values.
|
* [api:Currency]: A number with 2 decimal points of precision, designed to store currency values.
|
||||||
* `[api:Percentage]`: A decimal number between 0 and 1 that represents a percentage.
|
* [api:Percentage]: A decimal number between 0 and 1 that represents a percentage.
|
||||||
* `[api:Date]`: A date field
|
* [api:Date]: A date field
|
||||||
* `[api:SS_Datetime]`: A date / time field
|
* [api:SS_Datetime]: A date / time field
|
||||||
* `[api:Time]`: A time field
|
* [api:Time]: A time field
|
||||||
|
|
||||||
## HTMLText vs. Text, and HTMLVarchar vs. Varchar
|
## HTMLText vs. Text, and HTMLVarchar vs. Varchar
|
||||||
|
|
||||||
The database field types `[api:HTMLVarchar]` and `[api:Varchar]` are exactly the same in the database. However, the
|
The database field types [api:HTMLVarchar] and [api:Varchar] are exactly the same in the database. However, the
|
||||||
templating engine knows to escape the `[api:Varchar]` field and not the `[api:HTMLVarchar]` field. So, it's important you
|
templating engine knows to escape the [api:Varchar] field and not the [api:HTMLVarchar] field. So, it's important you
|
||||||
use the right field if you don't want to be putting $FieldType.XML everywhere.
|
use the right field if you don't want to be putting $FieldType.XML everywhere.
|
||||||
|
|
||||||
If you're going to put HTML content into the field, please use the field type with the HTML prefix. Otherwise, you're
|
If you're going to put HTML content into the field, please use the field type with the HTML prefix. Otherwise, you're
|
||||||
|
@ -7,13 +7,13 @@ following connections:
|
|||||||
* Each database-row maps to a php-object
|
* Each database-row maps to a php-object
|
||||||
* Each database-column maps to a property on a php-object
|
* Each database-column maps to a property on a php-object
|
||||||
|
|
||||||
All data tables in SilverStripe are defined as subclasses of `[api:DataObject]`. Inheritance is supported in the data
|
All data tables in SilverStripe are defined as subclasses of [api:DataObject]. Inheritance is supported in the data
|
||||||
model: seperate tables will be linked together, the data spread across these tables. The mapping and saving/loading
|
model: seperate tables will be linked together, the data spread across these tables. The mapping and saving/loading
|
||||||
logic is handled by sapphire, you don't need to worry about writing SQL most of the time.
|
logic is handled by sapphire, you don't need to worry about writing SQL most of the time.
|
||||||
|
|
||||||
The advanced object-relational layer in SilverStripe is one of the main reasons for requiring PHP5. Most of its
|
The advanced object-relational layer in SilverStripe is one of the main reasons for requiring PHP5. Most of its
|
||||||
customizations are possible through [PHP5 Object
|
customizations are possible through [PHP5 Object
|
||||||
Overloading](http://www.onlamp.com/pub/a/php/2005/06/16/overloading.html) handled in the `[api:Object]`-class.
|
Overloading](http://www.onlamp.com/pub/a/php/2005/06/16/overloading.html) handled in the [api:Object]-class.
|
||||||
|
|
||||||
See [database-structure](/reference/database-structure) for in-depth information on the database-schema.
|
See [database-structure](/reference/database-structure) for in-depth information on the database-schema.
|
||||||
|
|
||||||
@ -29,7 +29,7 @@ Note: You need to be logged in as an administrator to perform this command.
|
|||||||
## Querying Data
|
## Querying Data
|
||||||
|
|
||||||
There are static methods available for querying data. They automatically compile the necessary SQL to query the database
|
There are static methods available for querying data. They automatically compile the necessary SQL to query the database
|
||||||
so they are very helpful. In case you need to fall back to plain-jane SQL, have a look at `[api:SQLQuery]`.
|
so they are very helpful. In case you need to fall back to plain-jane SQL, have a look at [api:SQLQuery].
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$records = DataObject::get($obj, $filter, $sort, $join, $limit);
|
$records = DataObject::get($obj, $filter, $sort, $join, $limit);
|
||||||
@ -46,7 +46,7 @@ so they are very helpful. In case you need to fall back to plain-jane SQL, have
|
|||||||
|
|
||||||
Passing a *$join* statement to DataObject::get will filter results further by the JOINs performed against the foreign
|
Passing a *$join* statement to DataObject::get will filter results further by the JOINs performed against the foreign
|
||||||
table. **It will NOT return the additionally joined data.** The returned *$records* will always be a
|
table. **It will NOT return the additionally joined data.** The returned *$records* will always be a
|
||||||
`[api:DataObject]`.
|
[api:DataObject].
|
||||||
|
|
||||||
When using *$join* statements be sure the string is in the proper format for the respective database engine. In MySQL
|
When using *$join* statements be sure the string is in the proper format for the respective database engine. In MySQL
|
||||||
the use of back-ticks may be necessary when referring Table Names and potentially Columns. (see [MySQL
|
the use of back-ticks may be necessary when referring Table Names and potentially Columns. (see [MySQL
|
||||||
@ -188,7 +188,7 @@ on the "Player"-table.
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
SilverStripe's `[api:SiteTree]` base-class for content-pages uses a 1-to-1 relationship to link to its
|
SilverStripe's [api:SiteTree] base-class for content-pages uses a 1-to-1 relationship to link to its
|
||||||
parent element in the tree:
|
parent element in the tree:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -285,7 +285,7 @@ accessors available on both ends.
|
|||||||
### Adding relations
|
### Adding relations
|
||||||
|
|
||||||
Inside sapphire it doesn't matter if you're editing a *has_many*- or a *many_many*-relationship. You need to get a
|
Inside sapphire it doesn't matter if you're editing a *has_many*- or a *many_many*-relationship. You need to get a
|
||||||
`[api:ComponentSet]`.
|
[api:ComponentSet].
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
class Team extends DataObject {
|
class Team extends DataObject {
|
||||||
@ -315,7 +315,7 @@ Inside sapphire it doesn't matter if you're editing a *has_many*- or a *many_man
|
|||||||
### Custom Relation Getters
|
### Custom Relation Getters
|
||||||
|
|
||||||
You can use the flexible datamodel to get a filtered result-list without writing any SQL. For example, this snippet gets
|
You can use the flexible datamodel to get a filtered result-list without writing any SQL. For example, this snippet gets
|
||||||
you the "Players"-relation on a team, but only containing active players. (See `[api:DataObject::$has_many]` for more info on
|
you the "Players"-relation on a team, but only containing active players. (See [api:DataObject::$has_many] for more info on
|
||||||
the described relations).
|
the described relations).
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -448,16 +448,16 @@ See [forms](/topics/forms).
|
|||||||
|
|
||||||
### Saving data with custom SQL
|
### Saving data with custom SQL
|
||||||
|
|
||||||
See `[api:SQLQuery]` for custom *INSERT*, *UPDATE*, *DELETE* queries.
|
See [api:SQLQuery] for custom *INSERT*, *UPDATE*, *DELETE* queries.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## Decorating DataObjects
|
## Decorating DataObjects
|
||||||
|
|
||||||
You can add properties and methods to existing `[api:DataObjects]`s like `[api:Member]` (a core class) without hacking core
|
You can add properties and methods to an existing [api:DataObjects] like [api:Member] (a core class) without hacking core
|
||||||
code or subclassing.
|
code or subclassing.
|
||||||
Please see `[api:DataObjectDecorator]` for a general description, and `[api:Hierarchy]` for our most
|
Please see [api:DataObjectDecorator] for a general description, and [api:Hierarchy] for our most
|
||||||
popular examples.
|
popular examples.
|
||||||
|
|
||||||
|
|
||||||
@ -466,7 +466,7 @@ popular examples.
|
|||||||
|
|
||||||
### Whats the difference between DataObject::get() and a relation-getter?
|
### Whats the difference between DataObject::get() and a relation-getter?
|
||||||
You can work with both in pretty much the same way, but relationship-getters return a special type of collection:
|
You can work with both in pretty much the same way, but relationship-getters return a special type of collection:
|
||||||
A `[api:ComponentSet]` with relation-specific functionality.
|
A [api:ComponentSet] with relation-specific functionality.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$myTeam = DataObject::get_by_id('Team',$myPlayer->TeamID); // returns DataObject
|
$myTeam = DataObject::get_by_id('Team',$myPlayer->TeamID); // returns DataObject
|
||||||
|
@ -32,7 +32,7 @@ use devmode on a public server very very carefully
|
|||||||
Test mode is designed for staging environments or other private collaboration sites before deploying a site live. You do
|
Test mode is designed for staging environments or other private collaboration sites before deploying a site live. You do
|
||||||
not need to use test mode if you do not have a staging environment or a place for testing which is on a public server)
|
not need to use test mode if you do not have a staging environment or a place for testing which is on a public server)
|
||||||
|
|
||||||
In this mode error messages are hidden from the user and it includes `[api:BasicAuth]` integration if you want to password
|
In this mode error messages are hidden from the user and it includes [api:BasicAuth] integration if you want to password
|
||||||
protect the site.
|
protect the site.
|
||||||
|
|
||||||
To set your site to test mode set this in your `mysite/_config.php` file
|
To set your site to test mode set this in your `mysite/_config.php` file
|
||||||
|
@ -55,7 +55,7 @@ Example Forum:
|
|||||||
|
|
||||||
## PHP Include Paths
|
## PHP Include Paths
|
||||||
|
|
||||||
Due to the way `[api:ManifestBuilder]` recursively detects php-files and includes them through PHP5's
|
Due to the way [api:ManifestBuilder] recursively detects php-files and includes them through PHP5's
|
||||||
*__autoload()*-feature, you don't need to worry about include paths. Feel free to structure your php-code into
|
*__autoload()*-feature, you don't need to worry about include paths. Feel free to structure your php-code into
|
||||||
subdirectories inside the *code*-directory.
|
subdirectories inside the *code*-directory.
|
||||||
|
|
||||||
|
@ -150,4 +150,4 @@ The [newsletter module](http://silverstripe.org/newsletter-module) provides a UI
|
|||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
`[api:Email]`
|
[api:Email]
|
||||||
|
@ -36,11 +36,11 @@ warnings.
|
|||||||
An HTTP 500 error will be sent when there has been a fatal error on either a test or production site. You can make this
|
An HTTP 500 error will be sent when there has been a fatal error on either a test or production site. You can make this
|
||||||
friendlier - much like the 404 page, the error content can be edited within the CMS.
|
friendlier - much like the 404 page, the error content can be edited within the CMS.
|
||||||
|
|
||||||
* Create a page of type `[api:ErrorPage]`
|
* Create a page of type [api:ErrorPage]
|
||||||
* Set the error code to 500
|
* Set the error code to 500
|
||||||
* Publish the page.
|
* Publish the page.
|
||||||
|
|
||||||
**HOW IT WORKS: **The publication script for `[api:ErrorPage]` will write the full HTML content, including the template styling,
|
**HOW IT WORKS: **The publication script for [api:ErrorPage] will write the full HTML content, including the template styling,
|
||||||
to assets/error-500.html. The fatal error handler looks for the presence of this file, and if it exists, dumps the
|
to assets/error-500.html. The fatal error handler looks for the presence of this file, and if it exists, dumps the
|
||||||
content. This means that database access isn't required to provide a 500 error page.
|
content. This means that database access isn't required to provide a 500 error page.
|
||||||
|
|
||||||
|
@ -49,12 +49,12 @@ where url is the relative link to the page (eg 'admin/categories'). You can chan
|
|||||||
|
|
||||||
## Overloading EditForm
|
## Overloading EditForm
|
||||||
|
|
||||||
You may need to overload EditForm if your class does not use the `[api:Versioned]` extension.
|
You may need to overload EditForm if your class does not use the [api:Versioned] extension.
|
||||||
|
|
||||||
## Overloading SiteTreeAsUL
|
## Overloading SiteTreeAsUL
|
||||||
|
|
||||||
The tree hints can sometimes cause problems when reorganising the tree, and the CMSMain::SiteTreeAsUL function uses
|
The tree hints can sometimes cause problems when reorganising the tree, and the CMSMain::SiteTreeAsUL function uses
|
||||||
`[api:SiteTree]` explicitly. Use:
|
[api:SiteTree] explicitly. Use:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
public function SiteTreeAsUL() {
|
public function SiteTreeAsUL() {
|
||||||
|
@ -6,8 +6,8 @@ Form validation is a combination of PHP and JavaScript
|
|||||||
|
|
||||||
### Introduction
|
### Introduction
|
||||||
|
|
||||||
Validators are implemented as an argument to the `[api:Form]` constructor. You create a required fields validator like
|
Validators are implemented as an argument to the [api:Form] constructor. You create a required fields validator like
|
||||||
so. In this case, we're creating a `[api:RequiredFields]` validator - the `[api:Validator]` class itself is an abstract
|
so. In this case, we're creating a [api:RequiredFields] validator - the [api:Validator] class itself is an abstract
|
||||||
class.
|
class.
|
||||||
|
|
||||||
|
|
||||||
@ -74,4 +74,4 @@ Disable for a specific form:
|
|||||||
|
|
||||||
## Related
|
## Related
|
||||||
|
|
||||||
* Model Validation with [api:DataObject->validate()]
|
* Model Validation with [api:DataObject::validate()]
|
||||||
|
@ -14,9 +14,9 @@ constructor takes the following arguments:
|
|||||||
* `$name`: This must be the name of the method on that controller that is called to return the form. The first two
|
* `$name`: This must be the name of the method on that controller that is called to return the form. The first two
|
||||||
fields allow the form object to be re-created after submission. **It's vital that they are properly set - if you ever
|
fields allow the form object to be re-created after submission. **It's vital that they are properly set - if you ever
|
||||||
have problems with form action handler not working, check that these values are correct.**
|
have problems with form action handler not working, check that these values are correct.**
|
||||||
* `$fields`: A `[api:FieldSet]`s that make up the editable portion of the form.
|
* `$fields`: A [api:FieldSet] that makes up the editable portion of the form.
|
||||||
* `$actions`: A `[api:FieldSet]`s that make up the control portion of the form - the buttons at the bottom.
|
* `$actions`: A [api:FieldSet] that makes up the control portion of the form - the buttons at the bottom.
|
||||||
* `$validator`: An optional `[api:Validator]` for more information.
|
* `$validator`: An optional [api:Validator] for more information.
|
||||||
|
|
||||||
Example:
|
Example:
|
||||||
|
|
||||||
@ -60,16 +60,16 @@ The real difference, however, is that you can then define your controller method
|
|||||||
|
|
||||||
## Form Field Types
|
## Form Field Types
|
||||||
|
|
||||||
There are many classes extending `[api:FormField]`. Some examples:
|
There are many classes extending [api:FormField]. Some examples:
|
||||||
|
|
||||||
* `[api:TextField]`
|
* [api:TextField]
|
||||||
* `[api:EmailField]`
|
* [api:EmailField]
|
||||||
* `[api:NumericField]`
|
* [api:NumericField]
|
||||||
* `[api:DateField]`
|
* [api:DateField]
|
||||||
* `[api:CheckboxField]`
|
* [api:CheckboxField]
|
||||||
* `[api:DropdownField]`
|
* [api:DropdownField]
|
||||||
* `[api:OptionsetField]`
|
* [api:OptionsetField]
|
||||||
* `[api:CheckboxSetField]`
|
* [api:CheckboxSetField]
|
||||||
|
|
||||||
Full overview at [form-field-types](/reference/form-field-types)
|
Full overview at [form-field-types](/reference/form-field-types)
|
||||||
|
|
||||||
@ -195,7 +195,7 @@ First of all, you need to create your form on it's own class, that way you can d
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
`forTemplate()` tells the `[api:Form]` class to render with a template of return value of `$this->class`, which in this case
|
`forTemplate()` tells the [api:Form] class to render with a template of return value of `$this->class`, which in this case
|
||||||
is *MyForm*, the name of the class. If the template doesn't exist, then it falls back to using Form.ss.
|
is *MyForm*, the name of the class. If the template doesn't exist, then it falls back to using Form.ss.
|
||||||
|
|
||||||
*MyForm.ss* should then be placed into your *templates/Includes* directory for your project. Here is an example of
|
*MyForm.ss* should then be placed into your *templates/Includes* directory for your project. Here is an example of
|
||||||
@ -235,7 +235,7 @@ this case `TextField->Field()` or `EmailField->Field()` which returns an `<input
|
|||||||
for the type of field. Pass in the name of the field as the first parameter, as done above, to render it into the
|
for the type of field. Pass in the name of the field as the first parameter, as done above, to render it into the
|
||||||
template.
|
template.
|
||||||
|
|
||||||
To find more methods, have a look at the `[api:Form]` class, as there is a lot of different methods of customising the form
|
To find more methods, have a look at the [api:Form] class, as there is a lot of different methods of customising the form
|
||||||
templates, for example, you could use `<% control Fields %>` instead of specifying each field manually, as we've done
|
templates, for example, you could use `<% control Fields %>` instead of specifying each field manually, as we've done
|
||||||
above.
|
above.
|
||||||
|
|
||||||
@ -274,9 +274,9 @@ Adds a new text field called FavouriteColour next to the Content field in the CM
|
|||||||
## Related
|
## Related
|
||||||
|
|
||||||
* [form-field-types](/reference/form-field-types)
|
* [form-field-types](/reference/form-field-types)
|
||||||
* `[api:FormField]` class
|
* [api:FormField]
|
||||||
* [multiform module](http://silverstripe.org/multi-form-module)
|
* [multiform module](http://silverstripe.org/multi-form-module)
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
`[api:Form]`
|
[api:Form]
|
||||||
|
@ -7,7 +7,7 @@ different languages based on your global settings and the preferences of your we
|
|||||||
as l10n (short for "localization").
|
as l10n (short for "localization").
|
||||||
|
|
||||||
For translating any content managed through the CMS or stored in the database, please refer to the
|
For translating any content managed through the CMS or stored in the database, please refer to the
|
||||||
[translation](/topics/translation) documentation (which explains the `[api:Translatable]` extension).
|
[translation](/topics/translation) documentation (which explains the [api:Translatable] extension).
|
||||||
|
|
||||||
This page aims to describe the low-level functionality of the i18n-API. It targets developers who:
|
This page aims to describe the low-level functionality of the i18n-API. It targets developers who:
|
||||||
|
|
||||||
@ -28,7 +28,7 @@ The i18n class is enabled by default.
|
|||||||
|
|
||||||
### Setting the locale
|
### Setting the locale
|
||||||
|
|
||||||
To set the locale you just need to call `[api:i18n::set_locale()]` passing, as a parameter, the name of the locale that you
|
To set the locale you just need to call [api:i18n::set_locale()] passing, as a parameter, the name of the locale that you
|
||||||
want to set.
|
want to set.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -44,7 +44,7 @@ available locales.
|
|||||||
|
|
||||||
### Getting the locale
|
### Getting the locale
|
||||||
|
|
||||||
As you set the locale you can also get the current value, just by calling `[api:i18n::get_locale()]`.
|
As you set the locale you can also get the current value, just by calling [api:i18n::get_locale()].
|
||||||
|
|
||||||
### Declaring the content language in HTML {#declaring_the_content_language_in_html}
|
### Declaring the content language in HTML {#declaring_the_content_language_in_html}
|
||||||
|
|
||||||
|
@ -9,9 +9,9 @@ practices can be applied to other libraries as well.
|
|||||||
|
|
||||||
## File Inclusion
|
## File Inclusion
|
||||||
|
|
||||||
SilverStripe-driven code should use the `[api:Requirements]` class to manage clientside dependencies like CSS and JavaScript
|
SilverStripe-driven code should use the [api:Requirements] class to manage clientside dependencies like CSS and JavaScript
|
||||||
files, rather than including `<script>` and `<link>` tags in your templates. This has the advantage that a registry
|
files, rather than including `<script>` and `<link>` tags in your templates. This has the advantage that a registry
|
||||||
of requirements can be built up from different places outside of the main controller, for example included `[api:FormField]`
|
of requirements can be built up from different places outside of the main controller, for example included [api:FormField]
|
||||||
instances.
|
instances.
|
||||||
|
|
||||||
See [requirements](/reference/requirements) documentation.
|
See [requirements](/reference/requirements) documentation.
|
||||||
@ -366,7 +366,7 @@ attributes, or the jQuery.metadata plugin). For returning status messages, pleas
|
|||||||
|
|
||||||
Only return evaluated JavaScript snippets if unavoidable. Most of the time you can just pass data around, and let the
|
Only return evaluated JavaScript snippets if unavoidable. Most of the time you can just pass data around, and let the
|
||||||
clientside react to changes appropriately without telling it directly through JavaScript in AJAX responses. Don't use
|
clientside react to changes appropriately without telling it directly through JavaScript in AJAX responses. Don't use
|
||||||
the `[api:Form]` SilverStripe class, which is built solely around
|
the [api:Form] SilverStripe class, which is built solely around
|
||||||
this inflexible concept.
|
this inflexible concept.
|
||||||
|
|
||||||
Example: Autocomplete input field loading page matches through AJAX
|
Example: Autocomplete input field loading page matches through AJAX
|
||||||
@ -438,7 +438,7 @@ JavaScript:
|
|||||||
|
|
||||||
Although they are the minority of cases, there are times when a simple HTML fragment isn't enough. For example, if you
|
Although they are the minority of cases, there are times when a simple HTML fragment isn't enough. For example, if you
|
||||||
have server side code that needs to trigger the update of a couple of elements in the CMS left-hand tree, it would be
|
have server side code that needs to trigger the update of a couple of elements in the CMS left-hand tree, it would be
|
||||||
inefficient to send back the HTML of entire tree. SilverStripe can serialize to and from JSON (see the `[api:Convert]` class), and jQuery deals very well with it through
|
inefficient to send back the HTML of entire tree. SilverStripe can serialize to and from JSON (see the [api:Convert] class), and jQuery deals very well with it through
|
||||||
[jQuery.getJSON()](http://docs.jquery.com/Ajax/jQuery.getJSON#urldatacallback), as long as the HTTP content-type is
|
[jQuery.getJSON()](http://docs.jquery.com/Ajax/jQuery.getJSON#urldatacallback), as long as the HTTP content-type is
|
||||||
properly set.
|
properly set.
|
||||||
|
|
||||||
|
@ -6,14 +6,14 @@ A module is, quite simply, a collection of classes, templates, and other resourc
|
|||||||
directory. In a default SilverStripe download, even resources in 'sapphire' and 'mysite' are treated in exactly the
|
directory. In a default SilverStripe download, even resources in 'sapphire' and 'mysite' are treated in exactly the
|
||||||
same as every other module.
|
same as every other module.
|
||||||
|
|
||||||
Sapphire's `[api:ManifestBuilder]` will find any class, css or template files anywhere under the site's main
|
Sapphire's [api:ManifestBuilder] will find any class, css or template files anywhere under the site's main
|
||||||
directory. The _config.php file in the module directory can be used to define director rules, calls to
|
directory. The _config.php file in the module directory can be used to define director rules, calls to
|
||||||
Object::useCustomClass(), and the like. So, by unpacking a module into site's main directory and viewing the site with
|
Object::useCustomClass(), and the like. So, by unpacking a module into site's main directory and viewing the site with
|
||||||
?flush=1 on the end of the URL, all the module's new behaviour will be incorporated to your site:
|
?flush=1 on the end of the URL, all the module's new behaviour will be incorporated to your site:
|
||||||
|
|
||||||
* You can create subclasses of base classes such as SiteTree to extend behaviour.
|
* You can create subclasses of base classes such as SiteTree to extend behaviour.
|
||||||
* You can use Object::useCustomClass() to replace a built in class with a class of your own.
|
* You can use [api:Object::useCustomClass()] to replace a built in class with a class of your own.
|
||||||
* You can use [a decorator](api:DataObjectDecorator) to extend or alter the behaviour of a built-in class without replacing
|
* You can use [api:DataObjectDecorator] to extend or alter the behaviour of a built-in class without replacing
|
||||||
it.
|
it.
|
||||||
* You can provide additional director rules to define your own controller for particular URLs.
|
* You can provide additional director rules to define your own controller for particular URLs.
|
||||||
|
|
||||||
|
@ -78,8 +78,8 @@ See [form](/topics/forms) and [tutorial:extending-a-basic-site](/tutorials/exten
|
|||||||
|
|
||||||
### removeFieldFromTab()
|
### removeFieldFromTab()
|
||||||
|
|
||||||
Overloading `getCMSFields()` you can call `removeFieldFromTab()` on a `[api:FieldSet]` object. For example, if you don't
|
Overloading [api:SiteTree::getCMSFields()] you can call `removeFieldFromTab()` on a [api:FieldSet] object. For example, if you don't
|
||||||
want the MenuTitle field to show on your page, which is inherited from `[api:SiteTree]`.
|
want the MenuTitle field to show on your page, which is inherited from [api:SiteTree].
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
class StaffPage extends Page {
|
class StaffPage extends Page {
|
||||||
|
@ -7,23 +7,23 @@ See [Tutorial: Site Search](/tutorials/site-search) for details.
|
|||||||
|
|
||||||
## Searching for DataObject's
|
## Searching for DataObject's
|
||||||
|
|
||||||
The `[api:SearchContext]` class provides a good base implementation that you can hook into your own controllers.
|
The [api:SearchContext] class provides a good base implementation that you can hook into your own controllers.
|
||||||
A working implementation of searchable DataObjects can be seen in the `[api:ModelAdmin]` class.
|
A working implementation of searchable DataObjects can be seen in the [api:ModelAdmin] class.
|
||||||
|
|
||||||
[SearchContext](/reference/searchcontext) goes into more detail about setting up a default search form for `[api:DataObject]`s.
|
[SearchContext](/reference/searchcontext) goes into more detail about setting up a default search form for a [api:DataObject].
|
||||||
|
|
||||||
## Searching for Documents
|
## Searching for Documents
|
||||||
|
|
||||||
SilverStripe does not have a built-in method to search through file content (e.g. in PDF or DOC format).
|
SilverStripe does not have a built-in method to search through file content (e.g. in PDF or DOC format).
|
||||||
You can either extract any textual file content into the `[File](api:File)->Content` property, or use a
|
You can either extract any textual file content into the *Content* property, or use a
|
||||||
dedicated search service like the [sphinx module] `http://silverstripe.org/sphinx-module`.
|
dedicated search service like the [sphinx module](https://github.com/silverstripe-labs/silverstripe-fulltextsearch/tree/2.4).
|
||||||
|
|
||||||
## Related
|
## Related
|
||||||
|
|
||||||
* `[api:ModelAdmin]`
|
* [api:ModelAdmin]
|
||||||
* `[api:RestfulServer]`
|
* [api:RestfulServer]
|
||||||
* [Tutorial: Site Search](/tutorials/site-search)
|
* [Tutorial: Site Search](/tutorials/site-search)
|
||||||
* [SearchContext](/reference/searchcontext)
|
* [SearchContext](/reference/searchcontext)
|
||||||
* [genericviews module] `http://silverstripe.org/generic-views-module`
|
* [genericviews module] `http://silverstripe.org/generic-views-module`
|
||||||
* [sphinx module] `http://silverstripe.org/sphinx-module`
|
* [sphinx module](https://github.com/silverstripe-labs/silverstripe-fulltextsearch/tree/2.4)
|
||||||
* [lucene module] `http://silverstripe.org/lucene-module`
|
* [lucene module](https://code.google.com/archive/p/lucene-silverstripe-plugin)
|
||||||
|
@ -16,9 +16,9 @@ See [http://shiflett.org/articles/sql-injection](http://shiflett.org/articles/sq
|
|||||||
|
|
||||||
### Automatic escaping
|
### Automatic escaping
|
||||||
|
|
||||||
SilverStripe automatically escapes data in `[api:DataObject::write()]` wherever possible,
|
SilverStripe automatically escapes data in [api:DataObject::write()] wherever possible,
|
||||||
through database-specific methods (see `[api:Database->addslashes()]`).
|
through database-specific methods (see [api:MySQLDatabase::addslashes()]).
|
||||||
For `[api:MySQLDatabase]`, this will be `[mysql_real_escape_string()](http://de3.php.net/mysql_real_escape_string)`.
|
For [api:MySQLDatabase], this will be [mysql_real_escape_string()](http://php.net/manual/en/function.mysql-real-escape-string.php).
|
||||||
Data is escaped when saving back to the database, not when writing to object-properties.
|
Data is escaped when saving back to the database, not when writing to object-properties.
|
||||||
|
|
||||||
* DataObject::get_by_id()
|
* DataObject::get_by_id()
|
||||||
@ -119,8 +119,8 @@ for in-depth information about "Cross-Site-Scripting".
|
|||||||
|
|
||||||
### Escaping model properties
|
### Escaping model properties
|
||||||
|
|
||||||
`[api:SSViewer]` (the SilverStripe template engine) automatically takes care of escaping HTML tags from specific
|
[api:SSViewer] (the SilverStripe template engine) automatically takes care of escaping HTML tags from specific
|
||||||
object-properties by [casting](/topics/datamodel#casting) its string value into a `[api:DBField]` object.
|
object-properties by [casting](/topics/datamodel#casting) its string value into a [api:DBField] object.
|
||||||
|
|
||||||
PHP:
|
PHP:
|
||||||
|
|
||||||
@ -203,7 +203,7 @@ presentation from business logic.
|
|||||||
When using *customise()* or *renderWith()* calls in your controller, or otherwise forcing a custom context for your
|
When using *customise()* or *renderWith()* calls in your controller, or otherwise forcing a custom context for your
|
||||||
template, you'll need to take care of casting and escaping yourself in PHP.
|
template, you'll need to take care of casting and escaping yourself in PHP.
|
||||||
|
|
||||||
The `[api:Convert]` class has utilities for this, mainly *Convert::raw2xml()* and *Convert::raw2att()* (which is
|
The [api:Convert] class has utilities for this, mainly [api:Convert::raw2xml()] and [api:Convert::raw2att()] (which are
|
||||||
also used by *XML* and *ATT* in template code).
|
also used by *XML* and *ATT* in template code).
|
||||||
|
|
||||||
PHP:
|
PHP:
|
||||||
@ -260,7 +260,7 @@ Template:
|
|||||||
Some rules of thumb:
|
Some rules of thumb:
|
||||||
|
|
||||||
* Don't concatenate URLs in a template. It only works in extremely simple cases that usually contain bugs.
|
* Don't concatenate URLs in a template. It only works in extremely simple cases that usually contain bugs.
|
||||||
* Use *Controller::join_links()* to concatenate URLs. It deals with query strings and other such edge cases.
|
* Use [api:Controller::join_links()] to concatenate URLs. It deals with query strings and other such edge cases.
|
||||||
|
|
||||||
|
|
||||||
## Cross-Site Request Forgery (CSRF)
|
## Cross-Site Request Forgery (CSRF)
|
||||||
@ -283,7 +283,7 @@ See
|
|||||||
|
|
||||||
## Casting user input
|
## Casting user input
|
||||||
|
|
||||||
When working with `$_GET`, `$_POST` or `Director::urlParams` variables, and you know your variable has to be of a
|
When working with `$_GET`, `$_POST` or [api:Director::urlParams()] variables, and you know your variable has to be of a
|
||||||
certain type, like an integer, then it's essential to cast it as one. *Why?* To be sure that any processing of your
|
certain type, like an integer, then it's essential to cast it as one. *Why?* To be sure that any processing of your
|
||||||
given variable is done safely, with the assumption that it's an integer.
|
given variable is done safely, with the assumption that it's an integer.
|
||||||
|
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
Functional tests test your controllers. The core of these are the same as unit tests:
|
Functional tests test your controllers. The core of these are the same as unit tests:
|
||||||
|
|
||||||
* Create a subclass of `[api:SapphireTest]` in the `mysite/tests` or `(module)/tests` folder.
|
* Create a subclass of [api:SapphireTest] in the `mysite/tests` or `(module)/tests` folder.
|
||||||
* Define static $fixture_file to point to a database YAML file.
|
* Define static $fixture_file to point to a database YAML file.
|
||||||
* Create methods that start with "test" to create your tests.
|
* Create methods that start with "test" to create your tests.
|
||||||
* Assertions are used to work out if a test passed or failed.
|
* Assertions are used to work out if a test passed or failed.
|
||||||
@ -48,7 +48,7 @@ URLs. Here is an example from the subsites module:
|
|||||||
We are using a new static method here: **Director::test($url, $postVars, $sessionObj)**
|
We are using a new static method here: **Director::test($url, $postVars, $sessionObj)**
|
||||||
|
|
||||||
Director::test() lets us execute a URL and see what happens. It bypasses HTTP, instead relying on the cleanly
|
Director::test() lets us execute a URL and see what happens. It bypasses HTTP, instead relying on the cleanly
|
||||||
encapsulated execution model of `[api:Controller]`.
|
encapsulated execution model of [api:Controller].
|
||||||
|
|
||||||
It takes 3 arguments:
|
It takes 3 arguments:
|
||||||
|
|
||||||
@ -56,7 +56,7 @@ It takes 3 arguments:
|
|||||||
* $postVars: Post variables to pass to the URL
|
* $postVars: Post variables to pass to the URL
|
||||||
* $sessionObj: A Session object representing the current session.
|
* $sessionObj: A Session object representing the current session.
|
||||||
|
|
||||||
And it returns an `[api:HTTPResponse]` object, which will give you the response headers (including redirection), status code,
|
And it returns an [api:HTTPResponse] object, which will give you the response headers (including redirection), status code,
|
||||||
and body.
|
and body.
|
||||||
|
|
||||||
We can use string processing on the body of the response to then see if it fits with our expectations.
|
We can use string processing on the body of the response to then see if it fits with our expectations.
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# How To Create a Sapphire Test
|
# How To Create a Sapphire Test
|
||||||
|
|
||||||
A unit test class will test the behaviour of one of your `[api:DataObjects]`. This simple fragment of `[api:SiteTreeTest]`
|
A unit test class will test the behaviour of one of your [api:DataObjects]. This simple fragment of [api:SiteTreeTest]
|
||||||
provides us the basics of creating unit tests.
|
provides us the basics of creating unit tests.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -47,7 +47,7 @@ provides us the basics of creating unit tests.
|
|||||||
|
|
||||||
There are a number of points to note in this code fragment:
|
There are a number of points to note in this code fragment:
|
||||||
|
|
||||||
* Your test is a **subclass of SapphireTest**. Both unit tests and functional tests are a subclass of `[api:SapphireTest]`.
|
* Your test is a **subclass of SapphireTest**. Both unit tests and functional tests are a subclass of [api:SapphireTest].
|
||||||
* **static $fixture_file** is defined. The testing framework will automatically set up a new database for **each** of
|
* **static $fixture_file** is defined. The testing framework will automatically set up a new database for **each** of
|
||||||
your tests. The initial database content will be sourced from the YML file that you list in $fixture_file. You must
|
your tests. The initial database content will be sourced from the YML file that you list in $fixture_file. You must
|
||||||
define this value. Note also that, for the time being, you can only point to one YML file for each test class.
|
define this value. Note also that, for the time being, you can only point to one YML file for each test class.
|
||||||
@ -55,7 +55,7 @@ define this value. Note also that, for the time being, you can only point to on
|
|||||||
database will be rebuilt for each of these.
|
database will be rebuilt for each of these.
|
||||||
* **$this->objFromFixture($className, $identifier)** can be used to select one of the objects named in your fixture
|
* **$this->objFromFixture($className, $identifier)** can be used to select one of the objects named in your fixture
|
||||||
file. To identify to the object, we provide a class name and an identifier. The identifier is specified in the YML
|
file. To identify to the object, we provide a class name and an identifier. The identifier is specified in the YML
|
||||||
file but not saved in the database anywhere. objFromFixture() looks the `[api:DataObject]` up in memory rather than using the
|
file but not saved in the database anywhere. objFromFixture() looks the [api:DataObject] up in memory rather than using the
|
||||||
database. This means that you can use it to test the functions responsible for looking up content in the database.
|
database. This means that you can use it to test the functions responsible for looking up content in the database.
|
||||||
* **$this->assertEquals()** is one of the many assert... functions that PHPUnit provides us. See below for more
|
* **$this->assertEquals()** is one of the many assert... functions that PHPUnit provides us. See below for more
|
||||||
information.
|
information.
|
||||||
@ -78,7 +78,7 @@ if it starts with "/".
|
|||||||
|
|
||||||
## The Database YAML file
|
## The Database YAML file
|
||||||
|
|
||||||
The main feature of `[api:SapphireTest]` over the raw PHPUnit classes is that SapphireTest will prepare a temporary database for
|
The main feature of [api:SapphireTest] over the raw PHPUnit classes is that SapphireTest will prepare a temporary database for
|
||||||
you. The content of that database is provided in a special YAML file. YAML is a simple markup languages that uses tabs
|
you. The content of that database is provided in a special YAML file. YAML is a simple markup languages that uses tabs
|
||||||
and colons instead of the more verbose XML tags, and because of this much better for developers creating files by hand.
|
and colons instead of the more verbose XML tags, and because of this much better for developers creating files by hand.
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@ send emails in your SilverStripe application. Here is a simple example of how y
|
|||||||
$e->send();
|
$e->send();
|
||||||
|
|
||||||
|
|
||||||
Normally, the send() method would send an email using PHP's mail() function. However, if you are running a `[api:SapphireTest]`
|
Normally, the send() method would send an email using PHP's mail() function. However, if you are running a [api:SapphireTest]
|
||||||
test, then it holds off actually sending the email, and instead lets you assert that an email was sent using this method.
|
test, then it holds off actually sending the email, and instead lets you assert that an email was sent using this method.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
|
@ -80,7 +80,7 @@ Skip certain tests
|
|||||||
|
|
||||||
## Writing Tests
|
## Writing Tests
|
||||||
|
|
||||||
Tests are written by creating subclasses of `[api:SapphireTest]`. You should put tests for your site in the
|
Tests are written by creating subclasses of [api:SapphireTest]. You should put tests for your site in the
|
||||||
`mysite/tests` directory. If you are writing tests for a module, put them in the `(modulename)/tests` directory.
|
`mysite/tests` directory. If you are writing tests for a module, put them in the `(modulename)/tests` directory.
|
||||||
|
|
||||||
Generally speaking, there should be one test class for each application class. The name of the test class should be the
|
Generally speaking, there should be one test class for each application class. The name of the test class should be the
|
||||||
|
@ -245,7 +245,7 @@ Note: This only applies for CSS and template files. PHP files **do not** get ove
|
|||||||
|
|
||||||
### Requirements
|
### Requirements
|
||||||
|
|
||||||
The `[api:Requirements::themedCSS()]` function will
|
The [api:Requirements::themedCSS()] function will
|
||||||
do the search specified above. This avoids the need to type a full path to the css file, and also provides better
|
do the search specified above. This avoids the need to type a full path to the css file, and also provides better
|
||||||
ambiguity for themes.
|
ambiguity for themes.
|
||||||
|
|
||||||
|
@ -7,7 +7,7 @@ This page introduces developers to using the CMS for creating content in multipl
|
|||||||
Please see [i18n](/topics/i18n) for a internationalization, globalization and localization support of built-in datatypes as well
|
Please see [i18n](/topics/i18n) for a internationalization, globalization and localization support of built-in datatypes as well
|
||||||
as translating templates and PHP code.
|
as translating templates and PHP code.
|
||||||
|
|
||||||
Translations can be enabled for all subclasses of `[api:DataObject]`, so it can easily be implemented into existing code
|
Translations can be enabled for all subclasses of [api:DataObject], so it can easily be implemented into existing code
|
||||||
with minimal interference.
|
with minimal interference.
|
||||||
|
|
||||||
Warning: If you're upgrading from a SilverStripe version prior to 2.3.2, please migrate your datamodel before using the
|
Warning: If you're upgrading from a SilverStripe version prior to 2.3.2, please migrate your datamodel before using the
|
||||||
@ -60,7 +60,7 @@ Enabling Translatable through *Object::add_extension()* in your *mysite/_config.
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Make sure to rebuild the database through /dev/build after enabling `[api:Translatable]`.
|
Make sure to rebuild the database through /dev/build after enabling [api:Translatable].
|
||||||
Use the correct set_default_locale() before building the database
|
Use the correct set_default_locale() before building the database
|
||||||
for the first time, as this locale will be written on all new records.
|
for the first time, as this locale will be written on all new records.
|
||||||
|
|
||||||
@ -101,7 +101,7 @@ Getting a translation for an existing instance:
|
|||||||
|
|
||||||
|
|
||||||
Getting translations through Translatable::set_reading_locale().
|
Getting translations through Translatable::set_reading_locale().
|
||||||
This is *not* a recommended approach, but sometimes unavoidable (e.g. for `[api:Versioned]` methods).
|
This is *not* a recommended approach, but sometimes unavoidable (e.g. for [api:Versioned] methods).
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$origLocale = Translatable::get_reading_locale();
|
$origLocale = Translatable::get_reading_locale();
|
||||||
@ -120,7 +120,7 @@ Creating a translation:
|
|||||||
|
|
||||||
### Usage for SiteTree
|
### Usage for SiteTree
|
||||||
|
|
||||||
`[api:Translatable]` can be used for subclasses of SiteTree as well.
|
[api:Translatable] can be used for subclasses of SiteTree as well.
|
||||||
If a child page translation is requested without the parent
|
If a child page translation is requested without the parent
|
||||||
page already having a translation in this language, the extension
|
page already having a translation in this language, the extension
|
||||||
will recursively create translations up the tree.
|
will recursively create translations up the tree.
|
||||||
@ -129,7 +129,7 @@ languages by auto-appending the language code at the end.
|
|||||||
You'll need to ensure that the appropriate "reading language" is set
|
You'll need to ensure that the appropriate "reading language" is set
|
||||||
before showing links to other pages on a website through $_GET['locale'].
|
before showing links to other pages on a website through $_GET['locale'].
|
||||||
Pages in different languages can have different publication states
|
Pages in different languages can have different publication states
|
||||||
through the `[api:Versioned]` extension.
|
through the [api:Versioned] extension.
|
||||||
|
|
||||||
Note: You can't get Children() for a parent page in a different language
|
Note: You can't get Children() for a parent page in a different language
|
||||||
through set_reading_locale(). Get the translated parent first.
|
through set_reading_locale(). Get the translated parent first.
|
||||||
@ -148,11 +148,11 @@ through set_reading_locale(). Get the translated parent first.
|
|||||||
|
|
||||||
### Translating custom properties
|
### Translating custom properties
|
||||||
|
|
||||||
Keep in mind that the `[api:Translatable]` extension currently doesn't support the exclusion of properties from being
|
Keep in mind that the [api:Translatable] extension currently doesn't support the exclusion of properties from being
|
||||||
translated - all custom properties will automatically be fetched from their translated record on the database. This means
|
translated - all custom properties will automatically be fetched from their translated record on the database. This means
|
||||||
you don't have to explicitly mark any custom properties as being translatable.
|
you don't have to explicitly mark any custom properties as being translatable.
|
||||||
|
|
||||||
The `[api:Translatable]` decorator applies only to the getCMSFields() method on DataObject or SiteTree, not to any fields
|
The [api:Translatable] decorator applies only to the getCMSFields() method on DataObject or SiteTree, not to any fields
|
||||||
added in overloaded getCMSFields() implementations. See Translatable->updateCMSFields() for details. By default, custom
|
added in overloaded getCMSFields() implementations. See Translatable->updateCMSFields() for details. By default, custom
|
||||||
fields in the CMS won't show an original readonly value on a translated record, although they will save correctly. You can
|
fields in the CMS won't show an original readonly value on a translated record, although they will save correctly. You can
|
||||||
attach this behaviour to custom fields by using Translatable_Transformation as shown below.
|
attach this behaviour to custom fields by using Translatable_Transformation as shown below.
|
||||||
@ -354,11 +354,11 @@ files, you'll need to [set the i18n locale](/topics/translation#setting_the_i18n
|
|||||||
|
|
||||||
(The reasoning is as follows: Translatable doesn't set the i18n locale. Historically these were two separate systems,
|
(The reasoning is as follows: Translatable doesn't set the i18n locale. Historically these were two separate systems,
|
||||||
but they're reasonably interchangeable for a front-end website. The distinction is mainly valid for the CMS, because you
|
but they're reasonably interchangeable for a front-end website. The distinction is mainly valid for the CMS, because you
|
||||||
want the CMS to be in English (`[api:i18n]`), but edit pages in different languages (`[api:Translatable]`).)
|
want the CMS to be in English ([api:i18n]), but edit pages in different languages ([api:Translatable]).)
|
||||||
|
|
||||||
### Migrating from 2.1 datamodel
|
### Migrating from 2.1 datamodel
|
||||||
|
|
||||||
The datamodel of `[api:Translatable]` changed significantly between its original release in SilverStripe 2.1 and SilverStripe
|
The datamodel of [api:Translatable] changed significantly between its original release in SilverStripe 2.1 and SilverStripe
|
||||||
2.3.2. See our [discussion on the
|
2.3.2. See our [discussion on the
|
||||||
mailinglist](http://groups.google.com/group/silverstripe-dev/browse_thread/thread/91e26e1f78d3c1b4/bd276dd5bbc56283?lnk=gst&q=translatable#bd276dd5bbc56283).
|
mailinglist](http://groups.google.com/group/silverstripe-dev/browse_thread/thread/91e26e1f78d3c1b4/bd276dd5bbc56283?lnk=gst&q=translatable#bd276dd5bbc56283).
|
||||||
|
|
||||||
@ -370,13 +370,13 @@ To migrate a database that was built with SilverStripe 2.1.x or 2.2.x, follow th
|
|||||||
* Run `http://mysite.com/dev/build`
|
* Run `http://mysite.com/dev/build`
|
||||||
* Run `http://mysite.com/dev/tasks/MigrateTranslatableTask`
|
* Run `http://mysite.com/dev/tasks/MigrateTranslatableTask`
|
||||||
|
|
||||||
Please see the `[api:MigrateTranslatableTask]` for
|
Please see the [api:MigrateTranslatableTask] for
|
||||||
limitations of this migration task - not all your data will be preserved.
|
limitations of this migration task - not all your data will be preserved.
|
||||||
|
|
||||||
|
|
||||||
### Setting the i18n locale
|
### Setting the i18n locale
|
||||||
|
|
||||||
You can set the `[api:i18n]` locale value which is used to format dates, currencies and other regionally different values to
|
You can set the [api:i18n] locale value which is used to format dates, currencies and other regionally different values to
|
||||||
the same as your current page locale.
|
the same as your current page locale.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -393,7 +393,7 @@ the same as your current page locale.
|
|||||||
|
|
||||||
### Adding a new locale
|
### Adding a new locale
|
||||||
|
|
||||||
The `[api:i18n]` logic has lookup tables for common locales in i18n::$common_locales, which is a subset of i18n::$all_locales.
|
The [api:i18n] logic has lookup tables for common locales in i18n::$common_locales, which is a subset of i18n::$all_locales.
|
||||||
If your locale is not present here, you can simply add it through `mysite/_config.php`:
|
If your locale is not present here, you can simply add it through `mysite/_config.php`:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -406,6 +406,6 @@ This should e.g. enable you to use `$Locale.Nice` in template code.
|
|||||||
|
|
||||||
* [translate.silverstripe.org] `http://translate.silverstripe.org`: Starting point for community-driven translation of the Silverstripe UI
|
* [translate.silverstripe.org] `http://translate.silverstripe.org`: Starting point for community-driven translation of the Silverstripe UI
|
||||||
* [i18n](i18n): Developer-level documentation of Silverstripe's i18n capabilities
|
* [i18n](i18n): Developer-level documentation of Silverstripe's i18n capabilities
|
||||||
* `[api:Translatable]`: DataObject-interface powering the website-content translations
|
* [api:Translatable]: DataObject-interface powering the website-content translations
|
||||||
* ["Translatable ModelAdmin" module] `http://silverstripe.org/translatablemodeladmin-module/`: An extension which allows
|
* ["Translatable ModelAdmin" module] `http://silverstripe.org/translatablemodeladmin-module/`: An extension which allows
|
||||||
translations of `[api:DataObject]`s inside `[api:ModelAdmin]`
|
translations of a [api:DataObject] inside [api:ModelAdmin]
|
||||||
|
@ -231,7 +231,7 @@ This returns the value inputted in the CMS, if it's set or what is in the $title
|
|||||||
|
|
||||||
To implement a form inside a widget, you need to implement a custom controller for your widget to return this form. Make
|
To implement a form inside a widget, you need to implement a custom controller for your widget to return this form. Make
|
||||||
sure that your controller follows the usual naming conventions, and it will be automatically picked up by the
|
sure that your controller follows the usual naming conventions, and it will be automatically picked up by the
|
||||||
`[api:WidgetArea]` rendering in your *Page.ss* template.
|
[api:WidgetArea] rendering in your *Page.ss* template.
|
||||||
|
|
||||||
**mysite/code/MyWidget.php**
|
**mysite/code/MyWidget.php**
|
||||||
|
|
||||||
@ -276,7 +276,7 @@ To output this form, modify your widget template.
|
|||||||
$MyFormName
|
$MyFormName
|
||||||
|
|
||||||
<div class="notice" markdown='1'>
|
<div class="notice" markdown='1'>
|
||||||
**Note:** The necessary controller actions are only present in subclasses of `[api:Page_Controller]`. To use
|
**Note:** The necessary controller actions are only present in subclasses of [api:Page_Controller]. To use
|
||||||
widget forms in other controller subclasses, have a look at *ContentController->handleWidget()* and
|
widget forms in other controller subclasses, have a look at *ContentController->handleWidget()* and
|
||||||
*ContentController::$url_handlers*.
|
*ContentController::$url_handlers*.
|
||||||
</div>
|
</div>
|
||||||
|
@ -116,7 +116,7 @@ quickly between the CMS and the draft and published version of your page.
|
|||||||
**Flushing the cache**
|
**Flushing the cache**
|
||||||
|
|
||||||
Whenever we edit a template file, we need to append *?flush=1* onto the end of the URL, e.g.
|
Whenever we edit a template file, we need to append *?flush=1* onto the end of the URL, e.g.
|
||||||
http://localhost/home/?flush=1. SilverStripe stores template files in a cache for quicker load times. Whenever there are
|
`http://localhost/home/?flush=1`. SilverStripe stores template files in a cache for quicker load times. Whenever there are
|
||||||
changes to the template, we must flush the cache in order for the changes to take effect.
|
changes to the template, we must flush the cache in order for the changes to take effect.
|
||||||
|
|
||||||
## Inserting the page title
|
## Inserting the page title
|
||||||
@ -332,7 +332,7 @@ Create a new file *HomePage.php* in *mysite/code*. Copy the following code into
|
|||||||
|
|
||||||
|
|
||||||
Every page type also has a database table corresponding to it. Every time we modify the database, we need to rebuild it.
|
Every page type also has a database table corresponding to it. Every time we modify the database, we need to rebuild it.
|
||||||
We can do this by going to [http://localhost/dev/build?flush=1](http://localhost/dev/build?flush=1). It may take a
|
We can do this by going to `http://localhost/dev/build?flush=1`. It may take a
|
||||||
moment, so be patient. This add tables and fields needed by your site, and modifies any structures that have changed. It
|
moment, so be patient. This add tables and fields needed by your site, and modifies any structures that have changed. It
|
||||||
does this non-destructively - it will never delete your data.
|
does this non-destructively - it will never delete your data.
|
||||||
|
|
||||||
|
@ -32,9 +32,9 @@ final page. Lets look at each one individually:
|
|||||||
### Model
|
### Model
|
||||||
|
|
||||||
All content on your site is stored in a database. There is a table in the database corresponding for every class that is
|
All content on your site is stored in a database. There is a table in the database corresponding for every class that is
|
||||||
a child of the `[api:DataObject]` class. Every object of that class corresponds to a row in that table -
|
a child of the [api:DataObject] class. Every object of that class corresponds to a row in that table -
|
||||||
this is your "data object", the **"model"** of Model-View-Controller. A page type has a data object that represents all the data for your page - rather than inheriting
|
this is your "data object", the **"model"** of Model-View-Controller. A page type has a data object that represents all the data for your page - rather than inheriting
|
||||||
directly from data object it inherits from `[api:SiteTree]`. We generally create a "Page" data object, and subclass this for
|
directly from data object it inherits from [api:SiteTree]. We generally create a "Page" data object, and subclass this for
|
||||||
the rest of the page types. This allows us to define behavior that is consistent across all pages in our site.
|
the rest of the page types. This allows us to define behavior that is consistent across all pages in our site.
|
||||||
|
|
||||||
### View
|
### View
|
||||||
@ -48,7 +48,7 @@ presentation of your site.
|
|||||||
A page type also has a **"controller"**. A controller contains all the code used to manipulate your data before it is
|
A page type also has a **"controller"**. A controller contains all the code used to manipulate your data before it is
|
||||||
rendered. For example, suppose you were making an auction site, and you only wanted to display the auctions closing in
|
rendered. For example, suppose you were making an auction site, and you only wanted to display the auctions closing in
|
||||||
the next ten minutes. You would implement this in the controller. The controller for a page should inherit from
|
the next ten minutes. You would implement this in the controller. The controller for a page should inherit from
|
||||||
`[api:ContentController]`. Just as we create a "Page" data object and subclass it for the rest of the
|
[api:ContentController]. Just as we create a "Page" data object and subclass it for the rest of the
|
||||||
site, we also create a "Page_Controller" that is subclassed.
|
site, we also create a "Page_Controller" that is subclassed.
|
||||||
|
|
||||||
Creating a new page type simply requires creating these three things. You can then have full control over presentation,
|
Creating a new page type simply requires creating these three things. You can then have full control over presentation,
|
||||||
@ -126,7 +126,7 @@ to be children of the page in the site tree. As we only want news articles in th
|
|||||||
*ArticlePage* pages for children. We can enforce this in the CMS by setting the *$allowed_children* field.
|
*ArticlePage* pages for children. We can enforce this in the CMS by setting the *$allowed_children* field.
|
||||||
|
|
||||||
We will be introducing other fields like this as we progress; there is a full list in the documentation for
|
We will be introducing other fields like this as we progress; there is a full list in the documentation for
|
||||||
`[api:SiteTree]`.
|
[api:SiteTree].
|
||||||
|
|
||||||
Now that we have created our page types, we need to let SilverStripe rebuild the database. If we rebuild the database by
|
Now that we have created our page types, we need to let SilverStripe rebuild the database. If we rebuild the database by
|
||||||
going to [http://localhost/dev/build?flush=1](http://localhost/dev/build?flush=1), SilverStripe will detect that there are two
|
going to [http://localhost/dev/build?flush=1](http://localhost/dev/build?flush=1), SilverStripe will detect that there are two
|
||||||
@ -155,7 +155,7 @@ it. Change the *$db* array in the *ArticlePage* class so it looks like this:
|
|||||||
|
|
||||||
|
|
||||||
Every entry in the array is a key-value pair. The key is the name of the field, and the value is the type. We have a
|
Every entry in the array is a key-value pair. The key is the name of the field, and the value is the type. We have a
|
||||||
`[api:Date]` for a complete list of different data types.
|
[api:Date] for a complete list of different data types.
|
||||||
|
|
||||||
> Note: The names chosen for the fields you add must not already be used. Be careful using field names such as Title,
|
> Note: The names chosen for the fields you add must not already be used. Be careful using field names such as Title,
|
||||||
> Content etc. as these may already be defined in the page types your new page is extending from.
|
> Content etc. as these may already be defined in the page types your new page is extending from.
|
||||||
@ -192,7 +192,7 @@ Let's walk through this method.
|
|||||||
|
|
||||||
|
|
||||||
Firstly, we get the fields from the parent class; we want to add fields, not replace them. The *$fields* variable
|
Firstly, we get the fields from the parent class; we want to add fields, not replace them. The *$fields* variable
|
||||||
returned is a `[api:FieldSet]` object.
|
returned is a [api:FieldSet] object.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$fields->addFieldToTab('Root.Content.Main', new DateField('Date'), 'Content');
|
$fields->addFieldToTab('Root.Content.Main', new DateField('Date'), 'Content');
|
||||||
@ -201,7 +201,7 @@ returned is a `[api:FieldSet]` object.
|
|||||||
|
|
||||||
We can then add our new fields with *addFieldToTab*. The first argument is the tab on which we want to add the field to:
|
We can then add our new fields with *addFieldToTab*. The first argument is the tab on which we want to add the field to:
|
||||||
"Root.Content.Main" is the tab which the content editor is on. The second argument is the field to add; this is not a
|
"Root.Content.Main" is the tab which the content editor is on. The second argument is the field to add; this is not a
|
||||||
database field, but a `[api:FormField]` documentation for more details.
|
database field, but a [api:FormField] documentation for more details.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
return $fields;
|
return $fields;
|
||||||
@ -262,7 +262,7 @@ Set *showCalendar* to true to have a calendar appear underneath the Date field w
|
|||||||
$fields->addFieldToTab('Root.Content.Main', new TextField('Author','Author Name'), 'Content');
|
$fields->addFieldToTab('Root.Content.Main', new TextField('Author','Author Name'), 'Content');
|
||||||
|
|
||||||
By default the first argument *'Date'* or *'Author'* is shown as the title, however this might not be that helpful so to change the title,
|
By default the first argument *'Date'* or *'Author'* is shown as the title, however this might not be that helpful so to change the title,
|
||||||
add the new title as the second argument. See the `[api:DateField]` documentation for more details.
|
add the new title as the second argument. See the [api:DateField] documentation for more details.
|
||||||
|
|
||||||
|
|
||||||
## Creating the templates
|
## Creating the templates
|
||||||
@ -307,7 +307,7 @@ how to remove these blocks of repetitive code in a bit.
|
|||||||
|
|
||||||
We use *$Date* and *$Author* to access the new fields. In fact, all template variables and page controls come from
|
We use *$Date* and *$Author* to access the new fields. In fact, all template variables and page controls come from
|
||||||
either the data object or the controller for the page being displayed. The *$Breadcrumbs* variable comes from the
|
either the data object or the controller for the page being displayed. The *$Breadcrumbs* variable comes from the
|
||||||
*Breadcrumbs()* method of the `[api:SiteTree]` class. *$Date* and *$Author* come from the *Article* table through
|
*Breadcrumbs()* method of the [api:SiteTree] class. *$Date* and *$Author* come from the *Article* table through
|
||||||
your data object. *$Content* comes from the *SiteTree* table through the same data object. The data for your page is
|
your data object. *$Content* comes from the *SiteTree* table through the same data object. The data for your page is
|
||||||
spread across several tables in the database matched by id - e.g. *Content* is in the *SiteTree* table, and *Date* and
|
spread across several tables in the database matched by id - e.g. *Content* is in the *SiteTree* table, and *Date* and
|
||||||
*Author* are in the *Article* table. SilverStripe matches these records by their ids and collates them into the single
|
*Author* are in the *Article* table. SilverStripe matches these records by their ids and collates them into the single
|
||||||
@ -315,7 +315,7 @@ data object.
|
|||||||
|
|
||||||
![](_images/data-collation.jpg)
|
![](_images/data-collation.jpg)
|
||||||
|
|
||||||
Rather than using *$Date* directly, we use *$Date.Nice*. If we look in the `[api:Date]` documentation, we can see
|
Rather than using *$Date* directly, we use *$Date.Nice*. If we look in the [api:Date] documentation, we can see
|
||||||
that the *Nice* function returns the date in *dd/mm/yyyy* format, rather than the *yyyy-mm-dd* format stored in the
|
that the *Nice* function returns the date in *dd/mm/yyyy* format, rather than the *yyyy-mm-dd* format stored in the
|
||||||
database.
|
database.
|
||||||
|
|
||||||
@ -341,7 +341,7 @@ summary.
|
|||||||
|
|
||||||
Here we use the page control *Children*. As the name suggests, this control allows you to iterate over the children of a
|
Here we use the page control *Children*. As the name suggests, this control allows you to iterate over the children of a
|
||||||
page, which in this case is our news articles. The *$Link* variable will give the address of the article which we can
|
page, which in this case is our news articles. The *$Link* variable will give the address of the article which we can
|
||||||
use to create a link, and the *FirstParagraph* function of the `[api:HTMLText]` field gives us a nice summary of the
|
use to create a link, and the *FirstParagraph* function of the [api:HTMLText] field gives us a nice summary of the
|
||||||
article.
|
article.
|
||||||
|
|
||||||
![](_images/articleholder.jpg)
|
![](_images/articleholder.jpg)
|
||||||
@ -480,7 +480,7 @@ control. We can get the data for the news articles by implementing our own funct
|
|||||||
|
|
||||||
|
|
||||||
This function simply runs a database query that gets the latest news articles from the database. By default, this is
|
This function simply runs a database query that gets the latest news articles from the database. By default, this is
|
||||||
five, but you can change it by passing a number to the function. See the `[api:DataObject]` documentation for
|
five, but you can change it by passing a number to the function. See the [api:DataObject] documentation for
|
||||||
details. We can reference this function as a page control in our *HomePage* template:
|
details. We can reference this function as a page control in our *HomePage* template:
|
||||||
|
|
||||||
**themes/tutorial/templates/Layout/Homepage.ss**
|
**themes/tutorial/templates/Layout/Homepage.ss**
|
||||||
@ -514,7 +514,7 @@ the data object.
|
|||||||
## Creating a RSS feed
|
## Creating a RSS feed
|
||||||
|
|
||||||
An RSS feed is something that no news section should be without. SilverStripe makes it easy to create RSS feeds by
|
An RSS feed is something that no news section should be without. SilverStripe makes it easy to create RSS feeds by
|
||||||
providing an `[api:RSSFeed]` class to do all the hard work for you. Create the following function in the
|
providing an [api:RSSFeed] class to do all the hard work for you. Create the following function in the
|
||||||
*ArticleHolder_Controller*:
|
*ArticleHolder_Controller*:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -525,7 +525,7 @@ providing an `[api:RSSFeed]` class to do all the hard work for you. Create the f
|
|||||||
|
|
||||||
|
|
||||||
This function simply creates an RSS feed of all the news articles, and outputs it to the browser. If you go to
|
This function simply creates an RSS feed of all the news articles, and outputs it to the browser. If you go to
|
||||||
[http://localhost/news/rss](http://localhost/news/rss) you will see our RSS feed. What happens here is that
|
`http://localhost/news/rss` you will see our RSS feed. What happens here is that
|
||||||
when there is more to a URL after the page's base URL - "rss" in this case - SilverStripe will call the function with
|
when there is more to a URL after the page's base URL - "rss" in this case - SilverStripe will call the function with
|
||||||
that name on the controller if it exists.
|
that name on the controller if it exists.
|
||||||
|
|
||||||
@ -534,7 +534,7 @@ will most likely see the XML output instead.
|
|||||||
|
|
||||||
![](_images/rss-feed.jpg)
|
![](_images/rss-feed.jpg)
|
||||||
|
|
||||||
Now all we need is to let the user know that our RSS feed exists. The `[api:RSSFeed]` in your controller, it will be
|
Now all we need is to let the user know that our RSS feed exists. The [api:RSSFeed] in your controller, it will be
|
||||||
called when the page is requested. Add this function to *ArticleHolder_Controller*:
|
called when the page is requested. Add this function to *ArticleHolder_Controller*:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -627,7 +627,7 @@ a new *StaffHolder* called "Staff" in the "About Us" section, and create some *S
|
|||||||
|
|
||||||
### Creating the staff section templates
|
### Creating the staff section templates
|
||||||
|
|
||||||
The staff section templates aren't too difficult to create, thanks to the utility methods provided by the `[api:Image]` class.
|
The staff section templates aren't too difficult to create, thanks to the utility methods provided by the [api:Image] class.
|
||||||
|
|
||||||
**themes/tutorial/templates/Layout/StaffHolder.ss**
|
**themes/tutorial/templates/Layout/StaffHolder.ss**
|
||||||
|
|
||||||
@ -650,7 +650,7 @@ The staff section templates aren't too difficult to create, thanks to the utilit
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
||||||
This template is very similar to the *ArticleHolder* template. The *SetWidth* method of the `[api:Image]` class
|
This template is very similar to the *ArticleHolder* template. The *SetWidth* method of the [api:Image] class
|
||||||
will resize the image before sending it to the browser. The resized image is cached, so the server doesn't have to
|
will resize the image before sending it to the browser. The resized image is cached, so the server doesn't have to
|
||||||
resize the image every time the page is viewed.
|
resize the image every time the page is viewed.
|
||||||
|
|
||||||
|
@ -80,13 +80,13 @@ Let's step through this code.
|
|||||||
|
|
||||||
First we create our form fields.
|
First we create our form fields.
|
||||||
|
|
||||||
We do this by creating a `[api:FieldSet]` and passing our fields as arguments. The first field is a new
|
We do this by creating a [api:FieldSet] and passing our fields as arguments. The first field is a new
|
||||||
`[api:TextField]` with the name 'Name'.
|
[api:TextField] with the name 'Name'.
|
||||||
|
|
||||||
There is a second argument when creating a field which specifies the text on the label of the field. If no second
|
There is a second argument when creating a field which specifies the text on the label of the field. If no second
|
||||||
argument is passed, as in this case, it is assumed the label is the same as the name of the field.
|
argument is passed, as in this case, it is assumed the label is the same as the name of the field.
|
||||||
|
|
||||||
The second field we create is an `[api:OptionsetField]`. This is a dropdown, and takes a third argument - an
|
The second field we create is an [api:OptionsetField]. This is a dropdown, and takes a third argument - an
|
||||||
array mapping the values to the options listed in the dropdown.
|
array mapping the values to the options listed in the dropdown.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -102,14 +102,14 @@ button.
|
|||||||
|
|
||||||
Here we create a 'Submit' button which calls the 'doBrowserPoll' method, which we will create later.
|
Here we create a 'Submit' button which calls the 'doBrowserPoll' method, which we will create later.
|
||||||
|
|
||||||
All the form actions (in this case only one) are collected into a `[api:FieldSet]` object the same way we did with
|
All the form actions (in this case only one) are collected into a [api:FieldSet] object the same way we did with
|
||||||
the fields.
|
the fields.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
return new Form($this, 'BrowserPollForm', $fields, $actions);
|
return new Form($this, 'BrowserPollForm', $fields, $actions);
|
||||||
|
|
||||||
|
|
||||||
Finally we create the `[api:Form]` object and return it.
|
Finally we create the [api:Form] object and return it.
|
||||||
|
|
||||||
The first argument is the controller that contains the form, in most cases '$this'. The second is the name of the method
|
The first argument is the controller that contains the form, in most cases '$this'. The second is the name of the method
|
||||||
that returns the form, which is 'BrowserPollForm' in our case. The third and fourth arguments are the
|
that returns the form, which is 'BrowserPollForm' in our case. The third and fourth arguments are the
|
||||||
@ -179,11 +179,11 @@ Great! We now have a browser poll form, but it doesn't actually do anything. In
|
|||||||
implement the 'doBrowserPoll' method that we told it about.
|
implement the 'doBrowserPoll' method that we told it about.
|
||||||
|
|
||||||
First, we need some way of saving the poll submissions to the database, so we can retrieve the results later. We can do
|
First, we need some way of saving the poll submissions to the database, so we can retrieve the results later. We can do
|
||||||
this by creating a new object that extends from `[api:DataObject]`.
|
this by creating a new object that extends from [api:DataObject].
|
||||||
|
|
||||||
If you recall, in tutorial two we said that all objects that inherit from DataObject and that add fields are stored in
|
If you recall, in tutorial two we said that all objects that inherit from DataObject and that add fields are stored in
|
||||||
the database. Also recall that all pages extend DataObject indirectly through `[api:SiteTree]`. Here instead of
|
the database. Also recall that all pages extend DataObject indirectly through [api:SiteTree]. Here instead of
|
||||||
extending SiteTree (or `[api:Page]`) to create a page type, we extend DataObject directly.
|
extending SiteTree (or Page) to create a page type, we extend DataObject directly.
|
||||||
|
|
||||||
*mysite/code/BrowserPollSubmission.php*
|
*mysite/code/BrowserPollSubmission.php*
|
||||||
|
|
||||||
@ -197,7 +197,7 @@ extending SiteTree (or `[api:Page]`) to create a page type, we extend DataObject
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
If we then rebuild the database ([http://localhost/db/build?flush=1](http://localhost/db/build?flush=1)), we will see
|
If we then rebuild the database (`http://localhost/db/build?flush=1`), we will see
|
||||||
that the *BrowserPollSubmission* table is created. Now we just need to define 'doBrowserPoll' on *HomePage_Controller*.
|
that the *BrowserPollSubmission* table is created. Now we just need to define 'doBrowserPoll' on *HomePage_Controller*.
|
||||||
|
|
||||||
*mysite/code/HomePage.php*
|
*mysite/code/HomePage.php*
|
||||||
@ -215,7 +215,7 @@ that the *BrowserPollSubmission* table is created. Now we just need to define 'd
|
|||||||
|
|
||||||
|
|
||||||
A function that processes a form submission takes two arguments - the first is the data in the form, the second is the
|
A function that processes a form submission takes two arguments - the first is the data in the form, the second is the
|
||||||
`[api:Form]` object.
|
[api:Form] object.
|
||||||
|
|
||||||
In our function we create a new *BrowserPollSubmission* object. Since the name of our form fields and the name of the
|
In our function we create a new *BrowserPollSubmission* object. Since the name of our form fields and the name of the
|
||||||
database fields are the same we can save the form directly into the data object.
|
database fields are the same we can save the form directly into the data object.
|
||||||
@ -262,7 +262,7 @@ Now that we have a working form, we need some way of showing the results.
|
|||||||
The first thing to do is make it so a user can only vote once per session. If the user hasn't voted, show the form,
|
The first thing to do is make it so a user can only vote once per session. If the user hasn't voted, show the form,
|
||||||
otherwise show the results.
|
otherwise show the results.
|
||||||
|
|
||||||
We can do this using a session variable. The `[api:Session]` class handles all session variables in SilverStripe.
|
We can do this using a session variable. The [api:Session] class handles all session variables in SilverStripe.
|
||||||
First modify the 'doBrowserPoll' to set the session variable 'BrowserPollVoted' when a user votes.
|
First modify the 'doBrowserPoll' to set the session variable 'BrowserPollVoted' when a user votes.
|
||||||
|
|
||||||
*mysite/code/HomePage.php*
|
*mysite/code/HomePage.php*
|
||||||
@ -311,10 +311,10 @@ We now need some way of getting the data from the database into the template.
|
|||||||
|
|
||||||
In the second tutorial we got the latest news articles for the home page by using the 'DataObject::get' function. We
|
In the second tutorial we got the latest news articles for the home page by using the 'DataObject::get' function. We
|
||||||
can't use the 'DataObject::get' function here directly as we wish to count the total number of votes for each browser.
|
can't use the 'DataObject::get' function here directly as we wish to count the total number of votes for each browser.
|
||||||
By looking at the documentation for 'DataObject::get', we can see that it returns a `[api:DataObjectSet]`
|
By looking at the documentation for 'DataObject::get', we can see that it returns a [api:DataObjectSet]
|
||||||
object. In fact, all data that can be iterated over in a template with a page control is contained in a DataObjectSet.
|
object. In fact, all data that can be iterated over in a template with a page control is contained in a DataObjectSet.
|
||||||
|
|
||||||
A `[api:DataObjectSet]` is a set of not just DataObjects, but of ViewableData, which the majority of
|
A [api:DataObjectSet] is a set of not just DataObjects, but of ViewableData, which the majority of
|
||||||
SilverStripe's classes (including DataObject) inherit from. We can create a DataObjectSet, fill it with our data, and
|
SilverStripe's classes (including DataObject) inherit from. We can create a DataObjectSet, fill it with our data, and
|
||||||
then create our graph using a page control in the template. Create the function 'BrowserPollResults' on the
|
then create our graph using a page control in the template. Create the function 'BrowserPollResults' on the
|
||||||
*HomePage_Controller* class.
|
*HomePage_Controller* class.
|
||||||
@ -370,7 +370,7 @@ We get the total number of submissions, which is needed to calculate the percent
|
|||||||
Now we create an empty DataObjectSet to hold our data and then iterate over the 'Browser' submissions field. The 'groupBy'
|
Now we create an empty DataObjectSet to hold our data and then iterate over the 'Browser' submissions field. The 'groupBy'
|
||||||
method of DataObjectSet splits our DataObjectSet by the 'Browser' field passed to it. The percentage of submissions for each
|
method of DataObjectSet splits our DataObjectSet by the 'Browser' field passed to it. The percentage of submissions for each
|
||||||
browser is calculated using the size of the DataObjectSet. It puts these new DataObjectSets into an array indexed
|
browser is calculated using the size of the DataObjectSet. It puts these new DataObjectSets into an array indexed
|
||||||
by the value of the field. The `[api:ArrayData]` class wraps an array into a ViewableData object, so we finally create a new
|
by the value of the field. The [api:ArrayData] class wraps an array into a ViewableData object, so we finally create a new
|
||||||
ArrayData object, which we can add to our *$doSet* DataObjectSet of results.
|
ArrayData object, which we can add to our *$doSet* DataObjectSet of results.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
|
@ -172,7 +172,7 @@ Next we need to create the *results* function.
|
|||||||
First we populate an array with the data we wish to pass to the template - the search results, query and title of the
|
First we populate an array with the data we wish to pass to the template - the search results, query and title of the
|
||||||
page. The final line is a little more complicated.
|
page. The final line is a little more complicated.
|
||||||
|
|
||||||
When we call a function by its url (eg http://localhost/home/results), SilverStripe will look for a template with the
|
When we call a function by its url (eg `http://localhost/home/results`), SilverStripe will look for a template with the
|
||||||
name `PageType_function.ss`. As we are implementing the *results* function on the *Page* page type, we create our
|
name `PageType_function.ss`. As we are implementing the *results* function on the *Page* page type, we create our
|
||||||
results page template as *Page_results.ss*. Unfortunately this doesn't work when we are using page types that are
|
results page template as *Page_results.ss*. Unfortunately this doesn't work when we are using page types that are
|
||||||
children of the *Page* page type. For example, if someone used the search on the homepage, it would be rendered with
|
children of the *Page* page type. For example, if someone used the search on the homepage, it would be rendered with
|
||||||
@ -193,7 +193,7 @@ function, and then attempt to render it with *Page_results.ss*, falling back to
|
|||||||
## Creating the template
|
## Creating the template
|
||||||
|
|
||||||
Lastly we need to create the template for the search page. This template uses all the same techniques used in previous
|
Lastly we need to create the template for the search page. This template uses all the same techniques used in previous
|
||||||
tutorials. It also uses a number of pagination variables, which are provided by the `[api:DataObjectSet]`
|
tutorials. It also uses a number of pagination variables, which are provided by the [api:DataObjectSet]
|
||||||
class.
|
class.
|
||||||
|
|
||||||
*themes/tutorial/templates/Layout/Page_results.ss*
|
*themes/tutorial/templates/Layout/Page_results.ss*
|
||||||
|
Loading…
x
Reference in New Issue
Block a user