mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
Merge remote-tracking branch 'origin/3.3' into 3
# Conflicts: # core/Constants.php # dev/DevelopmentAdmin.php
This commit is contained in:
commit
9fed5561f4
@ -204,6 +204,7 @@
|
|||||||
fromContainingPanel: {
|
fromContainingPanel: {
|
||||||
ontoggle: function(e){
|
ontoggle: function(e){
|
||||||
this.toggleClass('collapsed', $(e.target).hasClass('collapsed'));
|
this.toggleClass('collapsed', $(e.target).hasClass('collapsed'));
|
||||||
|
$(window).resize(); //Trigger jLayout
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -507,28 +507,28 @@ class Director implements TemplateGlobalProvider {
|
|||||||
*/
|
*/
|
||||||
public static function is_https() {
|
public static function is_https() {
|
||||||
$return = false;
|
$return = false;
|
||||||
|
|
||||||
|
// See https://en.wikipedia.org/wiki/List_of_HTTP_header_fields
|
||||||
|
// See https://support.microsoft.com/?kbID=307347
|
||||||
|
$headerOverride = false;
|
||||||
|
if(TRUSTED_PROXY) {
|
||||||
|
$headers = (defined('SS_TRUSTED_PROXY_PROTOCOL_HEADER')) ? array(SS_TRUSTED_PROXY_PROTOCOL_HEADER) : null;
|
||||||
|
if(!$headers) {
|
||||||
|
// Backwards compatible defaults
|
||||||
|
$headers = array('HTTP_X_FORWARDED_PROTO', 'HTTP_X_FORWARDED_PROTOCOL', 'HTTP_FRONT_END_HTTPS');
|
||||||
|
}
|
||||||
|
foreach($headers as $header) {
|
||||||
|
$headerCompareVal = ($header === 'HTTP_FRONT_END_HTTPS' ? 'on' : 'https');
|
||||||
|
if(!empty($_SERVER[$header]) && strtolower($_SERVER[$header]) == $headerCompareVal) {
|
||||||
|
$headerOverride = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if ($protocol = Config::inst()->get('Director', 'alternate_protocol')) {
|
if ($protocol = Config::inst()->get('Director', 'alternate_protocol')) {
|
||||||
$return = ($protocol == 'https');
|
$return = ($protocol == 'https');
|
||||||
} else if(
|
} else if($headerOverride) {
|
||||||
TRUSTED_PROXY
|
|
||||||
&& isset($_SERVER['HTTP_X_FORWARDED_PROTO'])
|
|
||||||
&& strtolower($_SERVER['HTTP_X_FORWARDED_PROTO']) == 'https'
|
|
||||||
) {
|
|
||||||
// Convention for (non-standard) proxy signaling a HTTPS forward,
|
|
||||||
// see https://en.wikipedia.org/wiki/List_of_HTTP_header_fields
|
|
||||||
$return = true;
|
|
||||||
} else if(
|
|
||||||
TRUSTED_PROXY
|
|
||||||
&& isset($_SERVER['HTTP_X_FORWARDED_PROTOCOL'])
|
|
||||||
&& strtolower($_SERVER['HTTP_X_FORWARDED_PROTOCOL']) == 'https'
|
|
||||||
) {
|
|
||||||
// Less conventional proxy header
|
|
||||||
$return = true;
|
|
||||||
} else if(
|
|
||||||
isset($_SERVER['HTTP_FRONT_END_HTTPS'])
|
|
||||||
&& strtolower($_SERVER['HTTP_FRONT_END_HTTPS']) == 'on'
|
|
||||||
) {
|
|
||||||
// Microsoft proxy convention: https://support.microsoft.com/?kbID=307347
|
|
||||||
$return = true;
|
$return = true;
|
||||||
} else if((!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off')) {
|
} else if((!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off')) {
|
||||||
$return = true;
|
$return = true;
|
||||||
|
@ -655,14 +655,27 @@ class SS_HTTPRequest implements ArrayAccess {
|
|||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
public function getIP() {
|
public function getIP() {
|
||||||
if (TRUSTED_PROXY && !empty($_SERVER['HTTP_CLIENT_IP'])) {
|
$headerOverrideIP = null;
|
||||||
//check ip from share internet
|
if(TRUSTED_PROXY) {
|
||||||
return $_SERVER['HTTP_CLIENT_IP'];
|
$headers = (defined('SS_TRUSTED_PROXY_IP_HEADER')) ? array(SS_TRUSTED_PROXY_IP_HEADER) : null;
|
||||||
} elseif (TRUSTED_PROXY && !empty($_SERVER['HTTP_X_FORWARDED_FOR'])) {
|
if(!$headers) {
|
||||||
//to check ip is pass from proxy
|
// Backwards compatible defaults
|
||||||
return $_SERVER['HTTP_X_FORWARDED_FOR'];
|
$headers = array('HTTP_CLIENT_IP', 'HTTP_X_FORWARDED_FOR');
|
||||||
|
}
|
||||||
|
foreach($headers as $header) {
|
||||||
|
if(!empty($_SERVER[$header])) {
|
||||||
|
$headerOverrideIP = $_SERVER[$header];
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($headerOverrideIP) {
|
||||||
|
return $headerOverrideIP;
|
||||||
} elseif(isset($_SERVER['REMOTE_ADDR'])) {
|
} elseif(isset($_SERVER['REMOTE_ADDR'])) {
|
||||||
return $_SERVER['REMOTE_ADDR'];
|
return $_SERVER['REMOTE_ADDR'];
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -179,10 +179,13 @@ if(!isset($_SERVER['HTTP_HOST'])) {
|
|||||||
/**
|
/**
|
||||||
* Fix HTTP_HOST from reverse proxies
|
* Fix HTTP_HOST from reverse proxies
|
||||||
*/
|
*/
|
||||||
if (TRUSTED_PROXY && isset($_SERVER['HTTP_X_FORWARDED_HOST'])) {
|
$trustedProxyHeader = (defined('SS_TRUSTED_PROXY_HOST_HEADER'))
|
||||||
|
? SS_TRUSTED_PROXY_HOST_HEADER
|
||||||
|
: 'HTTP_X_FORWARDED_HOST';
|
||||||
|
|
||||||
|
if (TRUSTED_PROXY && !empty($_SERVER[$trustedProxyHeader])) {
|
||||||
// Get the first host, in case there's multiple separated through commas
|
// Get the first host, in case there's multiple separated through commas
|
||||||
$_SERVER['HTTP_HOST'] = strtok($_SERVER['HTTP_X_FORWARDED_HOST'], ',');
|
$_SERVER['HTTP_HOST'] = strtok($_SERVER[SS_TRUSTED_PROXY_HOST_HEADER], ',');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -163,26 +163,25 @@ class ParameterConfirmationToken {
|
|||||||
|
|
||||||
// Are we http or https? Replicates Director::is_https() without its dependencies/
|
// Are we http or https? Replicates Director::is_https() without its dependencies/
|
||||||
$proto = 'http';
|
$proto = 'http';
|
||||||
if(
|
// See https://en.wikipedia.org/wiki/List_of_HTTP_header_fields
|
||||||
TRUSTED_PROXY
|
// See https://support.microsoft.com/?kbID=307347
|
||||||
&& isset($_SERVER['HTTP_X_FORWARDED_PROTO'])
|
$headerOverride = false;
|
||||||
&& strtolower($_SERVER['HTTP_X_FORWARDED_PROTO']) == 'https'
|
if(TRUSTED_PROXY) {
|
||||||
) {
|
$headers = (defined('SS_TRUSTED_PROXY_PROTOCOL_HEADER')) ? array(SS_TRUSTED_PROXY_PROTOCOL_HEADER) : null;
|
||||||
// Convention for (non-standard) proxy signaling a HTTPS forward,
|
if(!$headers) {
|
||||||
// see https://en.wikipedia.org/wiki/List_of_HTTP_header_fields
|
// Backwards compatible defaults
|
||||||
$proto = 'https';
|
$headers = array('HTTP_X_FORWARDED_PROTO', 'HTTP_X_FORWARDED_PROTOCOL', 'HTTP_FRONT_END_HTTPS');
|
||||||
} else if(
|
}
|
||||||
TRUSTED_PROXY
|
foreach($headers as $header) {
|
||||||
&& isset($_SERVER['HTTP_X_FORWARDED_PROTOCOL'])
|
$headerCompareVal = ($header === 'HTTP_FRONT_END_HTTPS' ? 'on' : 'https');
|
||||||
&& strtolower($_SERVER['HTTP_X_FORWARDED_PROTOCOL']) == 'https'
|
if(!empty($_SERVER[$header]) && strtolower($_SERVER[$header]) == $headerCompareVal) {
|
||||||
) {
|
$headerOverride = true;
|
||||||
// Less conventional proxy header
|
break;
|
||||||
$proto = 'https';
|
}
|
||||||
} else if(
|
}
|
||||||
isset($_SERVER['HTTP_FRONT_END_HTTPS'])
|
}
|
||||||
&& strtolower($_SERVER['HTTP_FRONT_END_HTTPS']) == 'on'
|
|
||||||
) {
|
if($headerOverride) {
|
||||||
// Microsoft proxy convention: https://support.microsoft.com/?kbID=307347
|
|
||||||
$proto = 'https';
|
$proto = 'https';
|
||||||
} else if((!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off')) {
|
} else if((!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off')) {
|
||||||
$proto = 'https';
|
$proto = 'https';
|
||||||
@ -190,9 +189,6 @@ class ParameterConfirmationToken {
|
|||||||
$proto = 'https';
|
$proto = 'https';
|
||||||
}
|
}
|
||||||
|
|
||||||
if((!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] != 'off')) $proto = 'https';
|
|
||||||
if(isset($_SERVER['SSL'])) $proto = 'https';
|
|
||||||
|
|
||||||
$parts = array_filter(array(
|
$parts = array_filter(array(
|
||||||
// What's our host
|
// What's our host
|
||||||
$_SERVER['HTTP_HOST'],
|
$_SERVER['HTTP_HOST'],
|
||||||
|
@ -33,7 +33,8 @@ class DevelopmentAdmin extends Controller {
|
|||||||
parent::init();
|
parent::init();
|
||||||
|
|
||||||
// Special case for dev/build: Defer permission checks to DatabaseAdmin->init() (see #4957)
|
// Special case for dev/build: Defer permission checks to DatabaseAdmin->init() (see #4957)
|
||||||
$requestedDevBuild = (stripos($this->getRequest()->getURL(), 'dev/build') === 0 && !Security::database_is_ready());
|
$requestedDevBuild = (stripos($this->getRequest()->getURL(), 'dev/build') === 0)
|
||||||
|
&& (stripos($this->getRequest()->getURL(), 'dev/build/defaults') === false);
|
||||||
|
|
||||||
// We allow access to this controller regardless of live-status or ADMIN permission only
|
// We allow access to this controller regardless of live-status or ADMIN permission only
|
||||||
// if on CLI. Access to this controller is always allowed in "dev-mode", or of the user is ADMIN.
|
// if on CLI. Access to this controller is always allowed in "dev-mode", or of the user is ADMIN.
|
||||||
|
@ -19,7 +19,7 @@
|
|||||||
<div class="left">
|
<div class="left">
|
||||||
<h2>CMS / Framework Installation <?php if($silverstripe_version) echo "<small>Version $silverstripe_version</small>"; ?></h2>
|
<h2>CMS / Framework Installation <?php if($silverstripe_version) echo "<small>Version $silverstripe_version</small>"; ?></h2>
|
||||||
<p>Thanks for choosing to use SilverStripe! Please follow the instructions below and you should be up in running in no time.<br>
|
<p>Thanks for choosing to use SilverStripe! Please follow the instructions below and you should be up in running in no time.<br>
|
||||||
If you get stuck, head over to the <a href="http://silverstripe.org/installing-silverstripe" target="_blank">installation forum</a>, or check out our list of <a href="http://doc.silverstripe.org/doku.php?id=suggested-web-hosts" target="_blank">suggested web hosts</a> known to work with SilverStripe.
|
If you get stuck, head over to the <a href="http://silverstripe.org/community/forums/installing-silverstripe" target="_blank">installation forum</a>, or check out our page of <a href="http://www.silverstripe.org/hosting" target="_blank">suggested web hosting options</a> known to work with SilverStripe.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -32,7 +32,7 @@
|
|||||||
<?php if(isset($hasErrorOtherThanDatabase)) { ?>
|
<?php if(isset($hasErrorOtherThanDatabase)) { ?>
|
||||||
<p class="message error">
|
<p class="message error">
|
||||||
You aren't currently able to install the software. Please <a href="#requirements">see below</a> for details.<br>
|
You aren't currently able to install the software. Please <a href="#requirements">see below</a> for details.<br>
|
||||||
If you are having problems meeting the requirements, see the <a href="http://doc.silverstripe.org/framework/en/installation/server-requirements">server requirements</a>.
|
If you are having problems meeting the requirements, see the <a href="http://doc.silverstripe.org/framework/en/installation/server-requirements" target="_blank">server requirements</a>.
|
||||||
</p>
|
</p>
|
||||||
<?php if (isset($phpIniLocation)) { ?>
|
<?php if (isset($phpIniLocation)) { ?>
|
||||||
<p>Your php.ini file is located at <?php echo $phpIniLocation; ?></p>
|
<p>Your php.ini file is located at <?php echo $phpIniLocation; ?></p>
|
||||||
@ -46,7 +46,7 @@
|
|||||||
<?php } else if($req->hasWarnings()) { ?>
|
<?php } else if($req->hasWarnings()) { ?>
|
||||||
<div class="message warning">
|
<div class="message warning">
|
||||||
<p>There are some issues that we recommend you look at before installing, however, you are still able to install the software.
|
<p>There are some issues that we recommend you look at before installing, however, you are still able to install the software.
|
||||||
<br>Please see below for details. If you are having problems meeting the requirements, see the <a href="http://doc.silverstripe.org/framework/en/installation/server-requirements">server requirements</a>.</p>
|
<br>Please see below for details. If you are having problems meeting the requirements, see the <a href="http://doc.silverstripe.org/framework/en/installation/server-requirements" target="_blank">server requirements</a>.</p>
|
||||||
</div>
|
</div>
|
||||||
<?php } else if(!$dbReq->hasErrors() && !$adminReq->hasErrors()) { ?>
|
<?php } else if(!$dbReq->hasErrors() && !$adminReq->hasErrors()) { ?>
|
||||||
<div class="message goodInstall"><p>You're ready to install! Please confirm the configuration options below. <a href="#install">Install SilverStripe</a></p>
|
<div class="message goodInstall"><p>You're ready to install! Please confirm the configuration options below. <a href="#install">Install SilverStripe</a></p>
|
||||||
@ -243,7 +243,7 @@
|
|||||||
<h3 class="sectionHeading">Theme selection <small>Step 4 of 5</small></h3>
|
<h3 class="sectionHeading">Theme selection <small>Step 4 of 5</small></h3>
|
||||||
<p class="helpText">You can change the theme or <a href="http://addons.silverstripe.org/add-ons?type=theme">download</a> another from the SilverStripe website after installation.</p>
|
<p class="helpText">You can change the theme or <a href="http://addons.silverstripe.org/add-ons?type=theme">download</a> another from the SilverStripe website after installation.</p>
|
||||||
<ul id="Themes">
|
<ul id="Themes">
|
||||||
<li><input type="radio" name="template" value="simple" id="Simple" <?php if(!isset($_POST['template']) || $_POST['template'] == 'simple') {?>checked="checked"<?php }?>><label for="Simple"><a href="https://github.com/silverstripe-themes/silverstripe-simple">Simple</a> - our default theme ready to use, or begin the <a href="http://doc.silverstripe.org/framework/en/tutorials" target="_blank">tutorial</a>.</label></li>
|
<li><input type="radio" name="template" value="simple" id="Simple" <?php if(!isset($_POST['template']) || $_POST['template'] == 'simple') {?>checked="checked"<?php }?>><label for="Simple"><a href="https://github.com/silverstripe-themes/silverstripe-simple">Simple</a> - our default theme ready to use, or begin the <a href="http://www.silverstripe.org/learn/lessons" target="_blank">lessons</a>.</label></li>
|
||||||
<li><input type="radio" name="template" value="tutorial" id="EmptyTemplate" <?php if(isset($_POST['template']) && $_POST['template'] == 'tutorial') {?>checked="checked"<?php }?>><label for="EmptyTemplate">Empty template</label></li>
|
<li><input type="radio" name="template" value="tutorial" id="EmptyTemplate" <?php if(isset($_POST['template']) && $_POST['template'] == 'tutorial') {?>checked="checked"<?php }?>><label for="EmptyTemplate">Empty template</label></li>
|
||||||
</ul>
|
</ul>
|
||||||
<h3 class="sectionHeading" id="install">Confirm Install <small>Step 5 of 5</small></h3>
|
<h3 class="sectionHeading" id="install">Confirm Install <small>Step 5 of 5</small></h3>
|
||||||
|
@ -23,7 +23,7 @@ Our web-based [PHP installer](installation/) can check if you meet the requireme
|
|||||||
* MySQL 5.0+
|
* MySQL 5.0+
|
||||||
* PostgreSQL 8.3+ (requires ["postgresql" module](http://silverstripe.org/postgresql-module))
|
* PostgreSQL 8.3+ (requires ["postgresql" module](http://silverstripe.org/postgresql-module))
|
||||||
* SQL Server 2008+ (requires ["mssql" module](http://silverstripe.org/microsoft-sql-server-database/))
|
* SQL Server 2008+ (requires ["mssql" module](http://silverstripe.org/microsoft-sql-server-database/))
|
||||||
* Support for [Oracle](http://www.silverstripe.org/oracle-database-module/) and [SQLite](http://silverstripe.org/sqlite-database/) is not commercially supported, but is under development by our open source community.
|
* Support for `[Oracle](http://www.silverstripe.org/oracle-database-module/)` and [SQLite](http://silverstripe.org/sqlite-database/) is not commercially supported, but is under development by our open source community.
|
||||||
* One of the following web server products:
|
* One of the following web server products:
|
||||||
* Apache 2.0+ with mod_rewrite and "AllowOverride All" set
|
* Apache 2.0+ with mod_rewrite and "AllowOverride All" set
|
||||||
* IIS 7+
|
* IIS 7+
|
||||||
|
@ -71,7 +71,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**. The most common error here
|
includes all files with this extension, any **procedural code will be executed on every call**. The most common error here
|
||||||
is putting a test.php/phpinfo.php file in the document root. See [datamodel](/developer_guides/model/data_model_and_orm) and [controllers](/developer_guides/controllers)
|
is putting a test.php/phpinfo.php file in the document root. See [datamodel](/developer_guides/model/data_model_and_orm) and [controllers](/developer_guides/controllers)
|
||||||
for ways how to structure your code.
|
for ways how to structure your code.
|
||||||
|
@ -36,11 +36,11 @@ final page. Lets look at each one individually:
|
|||||||
|
|
||||||
### Model
|
### Model
|
||||||
|
|
||||||
All content on our site is stored in a database. Each class that is a child of the `[api:DataObject]` class will have its own table in our database.
|
All content on our site is stored in a database. Each class that is a child of the [api:DataObject] class will have its own table in our database.
|
||||||
|
|
||||||
Every object of such a class will correspond to a row in that table -
|
Every object of such a class will correspond to a row in that table -
|
||||||
this is our "data object", the **"model"** of Model-View-Controller. A page type has a data object that represents all the data for our page. Rather than inheriting
|
this is our "data object", the **"model"** of Model-View-Controller. A page type has a data object that represents all the data for our page. Rather than inheriting
|
||||||
directly from `[api:DataObject]`, it inherits from `[api:SiteTree]`. We generally create a "Page" data object, and subclass this for all other page types. This allows us to define behavior that is consistent across all pages in our site.
|
directly from [api:DataObject], it inherits from [api:SiteTree]. We generally create a "Page" data object, and subclass this for all other page types. This allows us to define behavior that is consistent across all pages in our site.
|
||||||
|
|
||||||
### View
|
### View
|
||||||
|
|
||||||
@ -49,7 +49,7 @@ presentation of our website.
|
|||||||
|
|
||||||
### Controller
|
### Controller
|
||||||
|
|
||||||
Each page type also has a **"controller"**. The controller contains all the code used to manipulate our data before it is rendered. For example, suppose we were making an auction site, and we only wanted to display the auctions closing in the next ten minutes. We would implement this logic 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 site, we also create a "Page_Controller" that is subclassed.
|
Each page type also has a **"controller"**. The controller contains all the code used to manipulate our data before it is rendered. For example, suppose we were making an auction site, and we only wanted to display the auctions closing in the next ten minutes. We would implement this logic 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 site, we also create a "Page_Controller" that is subclassed.
|
||||||
|
|
||||||
|
|
||||||
Creating a new page type requires creating each of these three elements. We will then have full control over presentation, the database, and editable CMS fields.
|
Creating a new page type requires creating each of these three elements. We will then have full control over presentation, the database, and editable CMS fields.
|
||||||
@ -95,7 +95,7 @@ Let's create the *ArticleHolder* page type.
|
|||||||
Here we have done something interesting: the *$allowed_children* field. This is one of a number of static fields we can define to change the properties of a page type. The *$allowed_children* field is an array of page types that are allowed
|
Here we have done something interesting: the *$allowed_children* field. This is one of a number of static fields we can define to change the properties of a page type. The *$allowed_children* field is an array of page types that are allowed
|
||||||
to be children of the page in the site tree. As we only want **news articles** in the news section, we only want pages of the type *ArticlePage* as children. We can enforce this in the CMS by setting the *$allowed_children* field within this class.
|
to be children of the page in the site tree. As we only want **news articles** in the news section, we only want pages of the type *ArticlePage* as children. We can enforce this in the CMS by setting the *$allowed_children* field within this class.
|
||||||
|
|
||||||
We will be introduced to other fields like this as we progress; there is a full list in the documentation for `[api:SiteTree]`.
|
We will be introduced to other fields like this as we progress; there is a full list in the documentation for [api:SiteTree].
|
||||||
|
|
||||||
Now that we have created our page types, we need to let SilverStripe rebuild the database: [http://localhost/your_site_name/dev/build](http://localhost/your_site_name/dev/build). SilverStripe should detect that there are two new page types, and add them to the list of page types in the database.
|
Now that we have created our page types, we need to let SilverStripe rebuild the database: [http://localhost/your_site_name/dev/build](http://localhost/your_site_name/dev/build). SilverStripe should detect that there are two new page types, and add them to the list of page types in the database.
|
||||||
|
|
||||||
@ -162,7 +162,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:FieldList]` object.
|
returned is a [api:FieldList] object.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$fields->addFieldToTab('Root.Main', new TextField('Author'), 'Content');
|
$fields->addFieldToTab('Root.Main', new TextField('Author'), 'Content');
|
||||||
@ -170,14 +170,14 @@ returned is a `[api:FieldList]` 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.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]` - see the documentation for more details.
|
"Root.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] - see the documentation for more details.
|
||||||
|
|
||||||
<div class="hint" markdown="1">
|
<div class="hint" markdown="1">
|
||||||
Note: By default, the CMS only has one tab. Creating new tabs is much like adding to existing tabs. For instance: `$fields->addFieldToTab('Root.NewTab', new TextField('Author'));`
|
Note: By default, the CMS only has one tab. Creating new tabs is much like adding to existing tabs. For instance: `$fields->addFieldToTab('Root.NewTab', new TextField('Author'));`
|
||||||
would create a new tab called "New Tab", and a single "Author" textfield inside.
|
would create a new tab called "New Tab", and a single "Author" textfield inside.
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
We have added two fields: A simple `[api:TextField]` and a `[api:DateField]`.
|
We have added two fields: A simple [api:TextField]` and a [api:DateField].
|
||||||
There are many more fields available in the default installation, listed in ["form field types"](/developer_guides/forms/field_types/common_subclasses).
|
There are many more fields available in the default installation, listed in ["form field types"](/developer_guides/forms/field_types/common_subclasses).
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -232,7 +232,7 @@ By enabling *showCalendar* you show a calendar overlay when clicking on the fiel
|
|||||||
:::php
|
:::php
|
||||||
$dateField->setConfig('dateformat', 'dd/MM/YYYY');
|
$dateField->setConfig('dateformat', 'dd/MM/YYYY');
|
||||||
|
|
||||||
*dateFormat* allows you to specify how you wish the date to be entered and displayed in the CMS field. See the `[api:DateField]` documentation for more configuration options.
|
*dateFormat* allows you to specify how you wish the date to be entered and displayed in the CMS field. See the [api:DateField] documentation for more configuration options.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$fields->addFieldToTab('Root.Main', new TextField('Author', 'Author Name'), 'Content');
|
$fields->addFieldToTab('Root.Main', new TextField('Author', 'Author Name'), 'Content');
|
||||||
@ -269,13 +269,13 @@ First, the template for displaying a single article:
|
|||||||
|
|
||||||
Most of the code is just like the regular Page.ss, we include an informational div with the date and the author of the Article.
|
Most of the code is just like the regular Page.ss, we include an informational div with the date and the author of the Article.
|
||||||
|
|
||||||
To access the new fields, we use *$Date* and *$Author*. In fact, all template variables and page controls come from either the data object or the controller for the page being displayed. The *$Title* variable comes from the *Title* field of the `[api:SiteTree]` class. *$Date* and *$Author* come from the *ArticlePage* table through your custom Page. *$Content* comes from the *SiteTree* table through the same data object. The data for your page is
|
To access the new fields, we use *$Date* and *$Author*. In fact, all template variables and page controls come from either the data object or the controller for the page being displayed. The *$Title* variable comes from the *Title* field of the [api:SiteTree] class. *$Date* and *$Author* come from the *ArticlePage* table through your custom Page. *$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 *ArticlePage* table. SilverStripe matches this data, and collates it into a single data object.
|
*Author* are in the *ArticlePage* table. SilverStripe matches this data, and collates it into a single data object.
|
||||||
|
|
||||||
![](../_images/tutorial2_data-collation.jpg)
|
![](../_images/tutorial2_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.
|
||||||
|
|
||||||
@ -305,7 +305,7 @@ We'll now create a template for the article holder. We want our news section to
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
||||||
Here we use the page control *Children*. As the name suggests, this control allows you to iterate over the children of a page. In this case, the children are 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 article. The function strips all tags from the paragraph extracted.
|
Here we use the page control *Children*. As the name suggests, this control allows you to iterate over the children of a page. In this case, the children are 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 article. The function strips all tags from the paragraph extracted.
|
||||||
|
|
||||||
![](../_images/tutorial2_articleholder.jpg)
|
![](../_images/tutorial2_articleholder.jpg)
|
||||||
|
|
||||||
@ -398,7 +398,7 @@ The controller for a page is only created when page is actually visited, while t
|
|||||||
|
|
||||||
## 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 providing an `[api:RSSFeed]` class to do all the hard work for us. Add the following in the *ArticleHolder_Controller* class:
|
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 us. Add the following in the *ArticleHolder_Controller* class:
|
||||||
|
|
||||||
**mysite/code/ArticleHolder.php**
|
**mysite/code/ArticleHolder.php**
|
||||||
|
|
||||||
@ -418,7 +418,7 @@ Ensure that when you have input the code to implement an RSS feed; flush the web
|
|||||||
|
|
||||||
This function creates an RSS feed of all the news articles, and outputs it to the browser. If we go to [http://localhost/your_site_name/news/rss](http://localhost/your_site_name/news/rss) we should see our RSS feed. When there is more to a URL after a page's base URL, "rss" in this case, SilverStripe will call the function with that name on the controller if it exists.
|
This function creates an RSS feed of all the news articles, and outputs it to the browser. If we go to [http://localhost/your_site_name/news/rss](http://localhost/your_site_name/news/rss) we should see our RSS feed. When there is more to a URL after a page's base URL, "rss" in this case, SilverStripe will call the function with that name on the controller if it exists.
|
||||||
|
|
||||||
Depending on your browser, you should see something like the picture below. If your browser doesn't support RSS, you will most likely see the XML output instead. For more on RSS, see `[api:RSSFeed]`
|
Depending on your browser, you should see something like the picture below. If your browser doesn't support RSS, you will most likely see the XML output instead. For more on RSS, see [api:RSSFeed]
|
||||||
|
|
||||||
![](../_images/tutorial2_rss-feed.jpg)
|
![](../_images/tutorial2_rss-feed.jpg)
|
||||||
|
|
||||||
@ -483,7 +483,7 @@ Nothing here should be new. The *StaffPage* page type is more interesting though
|
|||||||
|
|
||||||
Instead of adding our *Image* as a field in *$db*, we have used the *$has_one* array. This is because an *Image* is not a simple database field like all the fields we have seen so far, but has its own database table. By using the *$has_one* array, we create a relationship between the *StaffPage* table and the *Image* table by storing the id of the respective *Image* in the *StaffPage* table.
|
Instead of adding our *Image* as a field in *$db*, we have used the *$has_one* array. This is because an *Image* is not a simple database field like all the fields we have seen so far, but has its own database table. By using the *$has_one* array, we create a relationship between the *StaffPage* table and the *Image* table by storing the id of the respective *Image* in the *StaffPage* table.
|
||||||
|
|
||||||
We then add an `[api:UploadField]` in the *getCMSFields* function to the tab "Root.Images". Since this tab doesn't exist,
|
We then add an [api:UploadField] in the *getCMSFields* function to the tab "Root.Images". Since this tab doesn't exist,
|
||||||
the *addFieldToTab* function will create it for us. The *UploadField* allows us to select an image or upload a new one in
|
the *addFieldToTab* function will create it for us. The *UploadField* allows us to select an image or upload a new one in
|
||||||
the CMS.
|
the CMS.
|
||||||
|
|
||||||
@ -497,7 +497,7 @@ a new *StaffHolder* called "Staff", and create some *StaffPage*s in it.
|
|||||||
|
|
||||||
### 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/simple/templates/Layout/StaffHolder.ss**
|
**themes/simple/templates/Layout/StaffHolder.ss**
|
||||||
|
|
||||||
@ -521,7 +521,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 *ScaleWidth* method of the `[api:Image]` class
|
This template is very similar to the *ArticleHolder* template. The *ScaleWidth* 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.
|
||||||
|
|
||||||
|
@ -74,11 +74,11 @@ 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:FieldList]` and passing our fields as arguments.
|
We do this by creating a [api:FieldList] and passing our fields as arguments.
|
||||||
The first field is a `[api:TextField]` with the name 'Name'.
|
The first field is a [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
|
||||||
@ -90,14 +90,14 @@ array mapping the values to the options listed in the dropdown.
|
|||||||
After creating the fields, we create the form actions. Form actions appear as buttons at the bottom of the form.
|
After creating the fields, we create the form actions. Form actions appear as buttons at the bottom of the form.
|
||||||
The first argument is the name of the function to call when the button is pressed, and the second is the label of the button.
|
The first argument is the name of the function to call when the button is pressed, and the second is the label of the 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:FieldList]` object the same way we did with
|
All the form actions (in this case only one) are collected into a [api:FieldList] 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
|
||||||
FieldLists containing the fields and form actions respectively.
|
FieldLists containing the fields and form actions respectively.
|
||||||
@ -178,8 +178,8 @@ All going according to plan, if you visit [http://localhost/your_site_name/home/
|
|||||||
|
|
||||||
Great! We now have a browser poll form, but it doesn't actually do anything. In order to make the form work, we have to implement the 'doBrowserPoll()' method that we told it about.
|
Great! We now have a browser poll form, but it doesn't actually do anything. In order to make the form work, we have to 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 this by creating a new object that extends from `[api:DataObject]`.
|
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].
|
||||||
If you recall, in the [second tutorial](/tutorials/extending_a_basic_site) we said that all objects that inherit from DataObject and have their own fields are stored in tables 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 will extend `[api:DataObject]` directly:
|
If you recall, in the [second tutorial](/tutorials/extending_a_basic_site) we said that all objects that inherit from DataObject and have their own fields are stored in tables 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 will extend [api:DataObject] directly:
|
||||||
|
|
||||||
**mysite/code/BrowserPollSubmission.php**
|
**mysite/code/BrowserPollSubmission.php**
|
||||||
|
|
||||||
@ -208,7 +208,7 @@ If we then rebuild the database ([http://localhost/your_site_name/dev/build](htt
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
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.
|
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.
|
||||||
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.
|
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.
|
||||||
We call the 'write' method to write our data to the database, and '$this->redirectBack()' will redirect the user back to the home page.
|
We call the 'write' method to write our data to the database, and '$this->redirectBack()' will redirect the user back to the home page.
|
||||||
|
|
||||||
@ -240,7 +240,7 @@ If we then open the homepage and attempt to submit the form without filling in t
|
|||||||
Now that we have a working form, we need some way of showing the results.
|
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, otherwise show 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, otherwise show the results.
|
||||||
|
|
||||||
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.
|
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.
|
||||||
|
|
||||||
**mysite/code/HomePage.php**
|
**mysite/code/HomePage.php**
|
||||||
|
|
||||||
@ -279,7 +279,7 @@ Although the form is not shown, you'll still see the 'Browser Poll' heading. We'
|
|||||||
|
|
||||||
Now that we're collecting data, it would be nice to show the results on the website as well. We could simply output every vote, but that's boring. Let's group the results by browser, through the SilverStripe data model.
|
Now that we're collecting data, it would be nice to show the results on the website as well. We could simply output every vote, but that's boring. Let's group the results by browser, through the SilverStripe data model.
|
||||||
|
|
||||||
In the [second tutorial](/tutorials/extending_a_basic_site), we got a collection of news articles for the home page by using the 'ArticleHolder::get()' function, which returns a `[api:DataList]`. We can get all submissions in the same fashion, through `BrowserPollSubmission::get()`. This list will be the starting point for our result aggregation.
|
In the [second tutorial](/tutorials/extending_a_basic_site), we got a collection of news articles for the home page by using the 'ArticleHolder::get()' function, which returns a [api:DataList]. We can get all submissions in the same fashion, through `BrowserPollSubmission::get()`. This list will be the starting point for our result aggregation.
|
||||||
|
|
||||||
Create the function 'BrowserPollResults' on the *HomePage_Controller* class.
|
Create the function 'BrowserPollResults' on the *HomePage_Controller* class.
|
||||||
|
|
||||||
@ -306,7 +306,7 @@ This code introduces a few new concepts, so let's step through it.
|
|||||||
```php
|
```php
|
||||||
$submissions = new GroupedList(BrowserPollSubmission::get());
|
$submissions = new GroupedList(BrowserPollSubmission::get());
|
||||||
```
|
```
|
||||||
First we get all of the `BrowserPollSubmission` records from the database. This returns the submissions as a `[api:DataList]`.Then we wrap it inside a `[api:GroupedList]`, which adds the ability to group those records. The resulting object will behave just like the original `DataList`, though (with the addition of a `groupBy()` method).
|
First we get all of the `BrowserPollSubmission` records from the database. This returns the submissions as a [api:DataList]. Then we wrap it inside a [api:GroupedList], which adds the ability to group those records. The resulting object will behave just like the original `DataList`, though (with the addition of a `groupBy()` method).
|
||||||
|
|
||||||
```php
|
```php
|
||||||
$total = $submissions->Count();
|
$total = $submissions->Count();
|
||||||
@ -324,9 +324,9 @@ We get the total number of submissions, which is needed to calculate the percent
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
Now we create an empty `[api:ArrayList]` to hold the data we'll pass to the template. Its similar to `[api:DataList]`, but can hold arbitrary objects rather than just DataObject` instances. Then we iterate over the 'Browser' submissions field.
|
Now we create an empty [api:ArrayList] to hold the data we'll pass to the template. Its similar to [api:DataList], but can hold arbitrary objects rather than just DataObject` instances. Then we iterate over the 'Browser' submissions field.
|
||||||
|
|
||||||
The `groupBy()` method splits our list by the 'Browser' field passed to it, creating new lists with submissions just for a specific browser. Each of those lists is keyed by the browser name. The aggregated result is then contained in an `[api:ArrayData]` object, which behaves much like a standard PHP array, but allows us to use it in SilverStripe templates.
|
The `groupBy()` method splits our list by the 'Browser' field passed to it, creating new lists with submissions just for a specific browser. Each of those lists is keyed by the browser name. The aggregated result is then contained in an [api:ArrayData] object, which behaves much like a standard PHP array, but allows us to use it in SilverStripe templates.
|
||||||
|
|
||||||
|
|
||||||
The final step is to create the template to display our data. Change the 'BrowserPoll' div to the below.
|
The final step is to create the template to display our data. Change the 'BrowserPoll' div to the below.
|
||||||
|
@ -92,7 +92,7 @@ function, and then attempt to render it with *Page_results.ss*, falling back to
|
|||||||
## Creating the template
|
## Creating the template
|
||||||
|
|
||||||
Lastly we need the template for the search page. This template uses all the same techniques used in previous
|
Lastly we need 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:PaginatedList]`
|
tutorials. It also uses a number of pagination variables, which are provided by the [api:PaginatedList]
|
||||||
class.
|
class.
|
||||||
|
|
||||||
*themes/simple/templates/Layout/Page_results.ss*
|
*themes/simple/templates/Layout/Page_results.ss*
|
||||||
|
@ -160,7 +160,7 @@ It's empty by default, but you can add new students as required,
|
|||||||
or relate them to the project by typing in the box above the table.
|
or relate them to the project by typing in the box above the table.
|
||||||
|
|
||||||
In our case, we want to manage those records, edit their details, and add new ones.
|
In our case, we want to manage those records, edit their details, and add new ones.
|
||||||
To accomplish this, we have added a specific `[api:GridFieldConfig]`.
|
To accomplish this, we have added a specific [api:GridFieldConfig].
|
||||||
While we could've built the config from scratch, there's several
|
While we could've built the config from scratch, there's several
|
||||||
preconfigured instances. The `GridFieldConfig_RecordEditor` default configures
|
preconfigured instances. The `GridFieldConfig_RecordEditor` default configures
|
||||||
the field to edit records, rather than just viewing them.
|
the field to edit records, rather than just viewing them.
|
||||||
@ -427,4 +427,4 @@ we suggest some excercises to make the solution more flexible:
|
|||||||
and avoid any duplication between the two subclasses.
|
and avoid any duplication between the two subclasses.
|
||||||
* Render mentor details in their own template
|
* Render mentor details in their own template
|
||||||
* Change the `GridField` to list only five records per page (the default is 20).
|
* Change the `GridField` to list only five records per page (the default is 20).
|
||||||
This configuration is stored in the `[api:GridFieldPaginator]` component
|
This configuration is stored in the [api:GridFieldPaginator] component
|
||||||
|
@ -461,7 +461,7 @@ Occasionally, the system described above won't let you do exactly what you need
|
|||||||
methods that manipulate the SQL query at a lower level. When using these, please ensure that all table and field names
|
methods that manipulate the SQL query at a lower level. When using these, please ensure that all table and field names
|
||||||
are escaped with double quotes, otherwise some DB backends (e.g. PostgreSQL) won't work.
|
are escaped with double quotes, otherwise some DB backends (e.g. PostgreSQL) won't work.
|
||||||
|
|
||||||
Under the hood, query generation is handled by the `[api:DataQuery]` class. This class does provide more direct access
|
Under the hood, query generation is handled by the [api:DataQuery] class. This class does provide more direct access
|
||||||
to certain SQL features that `DataList` abstracts away from you.
|
to certain SQL features that `DataList` abstracts away from you.
|
||||||
|
|
||||||
In general, we advise against using these methods unless it's absolutely necessary. If the ORM doesn't do quite what
|
In general, we advise against using these methods unless it's absolutely necessary. If the ORM doesn't do quite what
|
||||||
@ -573,7 +573,7 @@ sub-classes of the base class (including the base class itself).
|
|||||||
example above, NewsSection didn't have its own data, so an extra table would be redundant.
|
example above, NewsSection didn't have its own data, 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 NewsPage tables by their ID fields.
|
To retrieve a news article, SilverStripe joins the [api:SiteTree], [api:Page] and NewsPage tables by their ID fields.
|
||||||
|
|
||||||
|
@ -295,7 +295,7 @@ and `remove()` method.
|
|||||||
You can use the ORM to get a filtered result list without writing any SQL. For example, this snippet gets you the
|
You can use the ORM 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.
|
"Players"-relation on a team, but only containing active players.
|
||||||
|
|
||||||
See `[api:DataObject::$has_many]` for more info on the described relations.
|
See [api:DataObject::$has_many] for more info on the described relations.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
<?php
|
<?php
|
||||||
|
@ -82,7 +82,7 @@ E.g.
|
|||||||
echo $row['BirthYear'];
|
echo $row['BirthYear'];
|
||||||
}
|
}
|
||||||
|
|
||||||
The result of `SQLSelect::execute()` is an array lightly wrapped in a database-specific subclass of `[api:SS_Query]`.
|
The result of `SQLSelect::execute()` is an array lightly wrapped in a database-specific subclass of [api:SS_Query].
|
||||||
This class implements the *Iterator*-interface, and provides convenience-methods for accessing the data.
|
This class implements the *Iterator*-interface, and provides convenience-methods for accessing the data.
|
||||||
|
|
||||||
### DELETE
|
### DELETE
|
||||||
@ -165,7 +165,7 @@ E.g.
|
|||||||
|
|
||||||
:::php
|
:::php
|
||||||
<?php
|
<?php
|
||||||
$update = SQLUpdate::create('"SiteTree"')->where(array('ID' => 3));
|
$update = SQLUpdate::create('"SiteTree"')->addWhere(array('ID' => 3));
|
||||||
|
|
||||||
// assigning a list of items
|
// assigning a list of items
|
||||||
$update->addAssignments(array(
|
$update->addAssignments(array(
|
||||||
|
@ -9,7 +9,7 @@ While this is a useful approach, it can lead to data inconsistencies if the reco
|
|||||||
controller and form context.
|
controller and form context.
|
||||||
|
|
||||||
Most validation constraints are actually data constraints which belong on the model. SilverStripe provides the
|
Most validation constraints are actually data constraints which belong on the model. SilverStripe provides the
|
||||||
[api:DataObject->validate] method for this purpose.
|
[api:DataObject::validate()] method for this purpose.
|
||||||
|
|
||||||
By default, there is no validation - objects are always valid! However, you can overload this method in your DataObject
|
By default, there is no validation - objects are always valid! However, you can overload this method in your DataObject
|
||||||
sub-classes to specify custom validation, or use the `validate` hook through a [api:DataExtension].
|
sub-classes to specify custom validation, or use the `validate` hook through a [api:DataExtension].
|
||||||
|
@ -10,7 +10,7 @@ It is most commonly applied to pages in the CMS (the `SiteTree` class). Draft co
|
|||||||
from published content shown to your website visitors.
|
from published content shown to your website visitors.
|
||||||
|
|
||||||
Versioning in SilverStripe is handled through the [api:Versioned] class. As a [api:DataExtension] it is possible to
|
Versioning in SilverStripe is handled through the [api:Versioned] class. As a [api:DataExtension] it is possible to
|
||||||
be applied to any `[api:DataObject]` subclass. The extension class will automatically update read and write operations
|
be applied to any [api:DataObject]` subclass. The extension class will automatically update read and write operations
|
||||||
done via the ORM via the `augmentSQL` database hook.
|
done via the ORM via the `augmentSQL` database hook.
|
||||||
|
|
||||||
Adding Versioned to your `DataObject` subclass works the same as any other extension. It accepts two or more arguments
|
Adding Versioned to your `DataObject` subclass works the same as any other extension. It accepts two or more arguments
|
||||||
@ -82,7 +82,7 @@ The record is retrieved as a `DataObject`, but saving back modifications via `wr
|
|||||||
rather than modifying the existing one.
|
rather than modifying the existing one.
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
In order to get a list of all versions for a specific record, we need to generate specialized `[api:Versioned_Version]`
|
In order to get a list of all versions for a specific record, we need to generate specialized [api:Versioned_Version]
|
||||||
objects, which expose the same database information as a `DataObject`, but also include information about when and how
|
objects, which expose the same database information as a `DataObject`, but also include information about when and how
|
||||||
a record was published.
|
a record was published.
|
||||||
|
|
||||||
@ -95,9 +95,9 @@ a record was published.
|
|||||||
|
|
||||||
The usual call to `DataObject->write()` will write to whatever stage is currently active, as defined by the
|
The usual call to `DataObject->write()` will write to whatever stage is currently active, as defined by the
|
||||||
`Versioned::current_stage()` global setting. Each call will automatically create a new version in the
|
`Versioned::current_stage()` global setting. Each call will automatically create a new version in the
|
||||||
`<class>_versions` table. To avoid this, use `[writeWithoutVersion()](api:Versioned->writeWithoutVersion())` instead.
|
`<class>_versions` table. To avoid this, use [writeWithoutVersion()](api:Versioned::writeWithoutVersion()) instead.
|
||||||
|
|
||||||
To move a saved version from one stage to another, call `[writeToStage(<stage>)](api:Versioned->writeToStage())` on the
|
To move a saved version from one stage to another, call [writeToStage(<stage>)](api:Versioned::writeToStage()) on the
|
||||||
object. The process of moving a version to a different stage is also called "publishing", so we've created a shortcut
|
object. The process of moving a version to a different stage is also called "publishing", so we've created a shortcut
|
||||||
for this: `publish(<from-stage>, <to-stage>)`.
|
for this: `publish(<from-stage>, <to-stage>)`.
|
||||||
|
|
||||||
@ -132,7 +132,7 @@ is initialized. But it can also be set and reset temporarily to force a specific
|
|||||||
### Custom SQL
|
### Custom SQL
|
||||||
|
|
||||||
We generally discourage writing `Versioned` queries from scratch, due to the complexities involved through joining
|
We generally discourage writing `Versioned` queries from scratch, due to the complexities involved through joining
|
||||||
multiple tables across an inherited table scheme (see `[api:Versioned->augmentSQL()]`). If possible, try to stick to
|
multiple tables across an inherited table scheme (see [api:Versioned::augmentSQL()]). If possible, try to stick to
|
||||||
smaller modifications of the generated `DataList` objects.
|
smaller modifications of the generated `DataList` objects.
|
||||||
|
|
||||||
Example: Get the first 10 live records, filtered by creation date:
|
Example: Get the first 10 live records, filtered by creation date:
|
||||||
|
@ -146,7 +146,7 @@ To include relations (`$has_one`, `$has_many` and `$many_many`) in your search,
|
|||||||
### Summary Fields
|
### Summary Fields
|
||||||
|
|
||||||
Summary fields can be used to show a quick overview of the data for a specific [api:DataObject] record. The most common use
|
Summary fields can be used to show a quick overview of the data for a specific [api:DataObject] record. The most common use
|
||||||
is their display as table columns, e.g. in the search results of a `[api:ModelAdmin]` CMS interface.
|
is their display as table columns, e.g. in the search results of a [api:ModelAdmin] CMS interface.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
<?php
|
<?php
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
The [api:DataObject::$defaults] array allows you to specify simple static values to be the default values when a
|
The [api:DataObject::$defaults] array allows you to specify simple static values to be the default values when a
|
||||||
record is created, but in many situations default values need to be dynamically calculated. In order to do this, the
|
record is created, but in many situations default values need to be dynamically calculated. In order to do this, the
|
||||||
[api:DataObject->populateDefaults()] method will need to be overloaded.
|
[api:DataObject::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!
|
||||||
|
@ -6,8 +6,8 @@ These lists can get quite long, and hard to present on a single list.
|
|||||||
by splitting up the list into multiple pages.
|
by splitting up the list into multiple pages.
|
||||||
|
|
||||||
In this howto, we present an alternative to pagination:
|
In this howto, we present an alternative to pagination:
|
||||||
Grouping a list by various criteria, through the `[api:GroupedList]` class.
|
Grouping a list by various criteria, through the [api:GroupedList] class.
|
||||||
This class is a `[api:SS_ListDecorator]`, which means it wraps around a list,
|
This class is a [api:SS_ListDecorator], which means it wraps around a list,
|
||||||
adding new functionality.
|
adding new functionality.
|
||||||
|
|
||||||
It provides a `groupBy()` method, which takes a field name, and breaks up the managed list
|
It provides a `groupBy()` method, which takes a field name, and breaks up the managed list
|
||||||
@ -88,7 +88,7 @@ In this case, the `getTitleFirstLetter()` method defined earlier is used to brea
|
|||||||
Grouping a set by month is a very similar process.
|
Grouping a set by month is a very similar process.
|
||||||
The only difference would be to sort the records by month name, and
|
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,
|
then create a method on the DataObject that returns the month name,
|
||||||
and pass that to the [api:GroupedList->GroupedBy()] call.
|
and pass that to the [api:GroupedList::GroupedBy()] call.
|
||||||
|
|
||||||
We're reusing our example `Module` object,
|
We're reusing our example `Module` object,
|
||||||
but grouping by its built-in `Created` property instead,
|
but grouping by its built-in `Created` property instead,
|
||||||
@ -128,7 +128,7 @@ sorted by month name from January to December. This can be accomplshed by sortin
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
The final step is the render this into the template using the [api:GroupedList->GroupedBy()] method.
|
The final step is the render this into the template using the [api:GroupedList::GroupedBy()] method.
|
||||||
|
|
||||||
:::ss
|
:::ss
|
||||||
// Modules list grouped by the Month Posted
|
// Modules list grouped by the Month Posted
|
||||||
|
@ -403,7 +403,7 @@ You can add your own forms by implementing new form instances (see the [Forms tu
|
|||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
* `[api:ContentController]`: The main controller responsible for handling pages.
|
* [api:ContentController]: The main controller responsible for handling pages.
|
||||||
* `[api:Controller]`: Generic controller (not specific to pages.)
|
* [api:Controller]: Generic controller (not specific to pages.)
|
||||||
* `[api:DataObject]`: Underlying model class for page objects.
|
* [api:DataObject]: Underlying model class for page objects.
|
||||||
* `[api:ViewableData]`: Underlying object class for pretty much anything displayable.
|
* [api:ViewableData]: Underlying object class for pretty much anything displayable.
|
||||||
|
@ -23,7 +23,7 @@ Requiring assets from the template is restricted compared to the PHP API.
|
|||||||
## PHP Requirements API
|
## PHP Requirements API
|
||||||
|
|
||||||
It is common practice to include most Requirements either in the *init()*-method of your [controller](../controllers/), or
|
It is common practice to include most Requirements either in the *init()*-method of your [controller](../controllers/), 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
|
||||||
<?php
|
<?php
|
||||||
|
@ -8,7 +8,7 @@ All objects that are being rendered in a template should be a [api:ViewableData]
|
|||||||
[scope](syntax#scope).
|
[scope](syntax#scope).
|
||||||
|
|
||||||
For instance, if we provide a [api:HtmlText] instance to the template we can call the `FirstParagraph` method. This will
|
For instance, if we provide a [api:HtmlText] instance to the template we can call the `FirstParagraph` method. This will
|
||||||
output the result of the [api:HtmlText::FirstParagraph] method to the template.
|
output the result of the [api:HtmlText::FirstParagraph()] method to the template.
|
||||||
|
|
||||||
**mysite/code/Page.ss**
|
**mysite/code/Page.ss**
|
||||||
|
|
||||||
@ -98,7 +98,7 @@ this purpose.
|
|||||||
There's some exceptions to this rule, see the ["security" guide](../security).
|
There's some exceptions to this rule, see the ["security" guide](../security).
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
In case you want to explicitly allow un-escaped HTML input, the property can be cast as `[api:HTMLText]`. The following
|
In case you want to explicitly allow un-escaped HTML input, the property can be cast as [api:HTMLText]. The following
|
||||||
example takes the `Content` field in a `SiteTree` class, which is of this type. It forces the content into an explicitly
|
example takes the `Content` field in a `SiteTree` class, which is of this type. It forces the content into an explicitly
|
||||||
escaped format.
|
escaped format.
|
||||||
|
|
||||||
|
@ -25,8 +25,8 @@ Note that the concept of "pages" used in pagination does not necessarily mean th
|
|||||||
it's just a term to describe a sub-collection of the list.
|
it's just a term to describe a sub-collection of the list.
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
There are two ways to generate pagination controls: [api:PaginatedList->Pages] and
|
There are two ways to generate pagination controls: [api:PaginatedList::Pages()] and
|
||||||
[api:PaginatedList->PaginationSummary]. In this example we will use `PaginationSummary()`.
|
[api:PaginatedList::PaginationSummary()]. In this example we will use `PaginationSummary()`.
|
||||||
|
|
||||||
The first step is to simply list the objects in the template:
|
The first step is to simply list the objects in the template:
|
||||||
|
|
||||||
@ -72,7 +72,7 @@ If there is more than one page, this block will render a set of pagination contr
|
|||||||
|
|
||||||
In some situations where you are generating the list yourself, the underlying list will already contain only the items
|
In some situations where you are generating the list yourself, the underlying list will already contain only the items
|
||||||
that you wish to display on the current page. In this situation the automatic limiting done by [api:PaginatedList]
|
that you wish to display on the current page. In this situation the automatic limiting done by [api:PaginatedList]
|
||||||
will break the pagination. You can disable automatic limiting using the [api:PaginatedList->setLimitItems] method
|
will break the pagination. You can disable automatic limiting using the [api:PaginatedList::setLimitItems()] method
|
||||||
when using custom lists.
|
when using custom lists.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
|
@ -159,7 +159,7 @@ Each controller should define a `Link()` method. This should be used to avoid ha
|
|||||||
}
|
}
|
||||||
|
|
||||||
<div class="info" markdown="1">
|
<div class="info" markdown="1">
|
||||||
The [api:Controller::join_links] is optional, but makes `Link()` more flexible by allowing an `$action` argument, and concatenates the path segments with slashes. The action should map to a method on your controller.
|
The [api:Controller::join_links()] is optional, but makes `Link()` more flexible by allowing an `$action` argument, and concatenates the path segments with slashes. The action should map to a method on your controller.
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
## Related Documentation
|
## Related Documentation
|
||||||
|
@ -95,7 +95,7 @@ You can also fetch one parameter at a time.
|
|||||||
|
|
||||||
## URL Patterns
|
## URL Patterns
|
||||||
|
|
||||||
The `[api:RequestHandler]` class will parse all rules you specify against the following patterns. The most specific rule
|
The [api:RequestHandler] class will parse all rules you specify against the following patterns. The most specific rule
|
||||||
will be the one followed for the response.
|
will be the one followed for the response.
|
||||||
|
|
||||||
<div class="alert">
|
<div class="alert">
|
||||||
|
@ -5,7 +5,7 @@ summary: Validate form data through the server side validation API.
|
|||||||
|
|
||||||
SilverStripe provides server-side form validation out of the box through the [api:Validator] class and its' child class
|
SilverStripe provides server-side form validation out of the box through the [api:Validator] class and its' child class
|
||||||
[api:RequiredFields]. A single `Validator` instance is set on each `Form`. Validators are implemented as an argument to
|
[api:RequiredFields]. A single `Validator` instance is set on each `Form`. Validators are implemented as an argument to
|
||||||
the `[api:Form]` constructor or through the function `setValidator`.
|
the [api:Form] constructor or through the function `setValidator`.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
<?php
|
<?php
|
||||||
@ -50,7 +50,7 @@ In this example we will be required to input a value for `Name` and a valid emai
|
|||||||
|
|
||||||
<div class="info" markdown="1">
|
<div class="info" markdown="1">
|
||||||
Each individual [api:FormField] instance is responsible for validating the submitted content through the
|
Each individual [api:FormField] instance is responsible for validating the submitted content through the
|
||||||
[api:FormField::validate] method. By default, this just checks the value exists. Fields like `EmailField` override
|
[api:FormField::validate()] method. By default, this just checks the value exists. Fields like `EmailField` override
|
||||||
`validate` to check for a specific format.
|
`validate` to check for a specific format.
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -192,7 +192,7 @@ classes added to each input. For Parsley we can structure the form like.
|
|||||||
## Model Validation
|
## Model Validation
|
||||||
|
|
||||||
An alternative (or additional) approach to validation is to place it directly on the database model. SilverStripe
|
An alternative (or additional) approach to validation is to place it directly on the database model. SilverStripe
|
||||||
provides a `[api:DataObject->validate]` method to validate data at the model level. See
|
provides a [api:DataObject::validate()] method to validate data at the model level. See
|
||||||
[Data Model Validation](../model/validation).
|
[Data Model Validation](../model/validation).
|
||||||
|
|
||||||
### Validation in the CMS
|
### Validation in the CMS
|
||||||
|
@ -3,73 +3,73 @@ summary: A table containing a list of the common FormField subclasses.
|
|||||||
|
|
||||||
# Common FormField type subclasses
|
# Common FormField type subclasses
|
||||||
|
|
||||||
This is a high level overview of available `[api:FormField]` subclasses. An automatically generated list is available
|
This is a high level overview of available [api:FormField] subclasses. An automatically generated list is available
|
||||||
on the SilverStripe API documentation.
|
on the SilverStripe API documentation.
|
||||||
|
|
||||||
## Basic
|
## Basic
|
||||||
|
|
||||||
* `[api:CheckboxField]`: Single checkbox field.
|
* [api:CheckboxField]: Single checkbox field.
|
||||||
* `[api:DropdownField]`: A `<select>` tag. Can optionally save into has-one relationships.
|
* [api:DropdownField]: A `<select>` tag. Can optionally save into has-one relationships.
|
||||||
* `[api:ReadonlyField]`: Read-only field to display a non-editable value with a label.
|
* [api:ReadonlyField]: Read-only field to display a non-editable value with a label.
|
||||||
* `[api:TextareaField]`: Multi-line text field.
|
* [api:TextareaField]: Multi-line text field.
|
||||||
* `[api:TextField]`: Single-line text field.
|
* [api:TextField]: Single-line text field.
|
||||||
* `[api:PasswordField]`: Masked input field.
|
* [api:PasswordField]: Masked input field.
|
||||||
|
|
||||||
## Actions
|
## Actions
|
||||||
|
|
||||||
* `[api:FormAction]`: Button element for forms, both for `<input type="submit">` and `<button>`.
|
* [api:FormAction]: Button element for forms, both for `<input type="submit">` and `<button>`.
|
||||||
* `[api:ResetFormAction]`: Action that clears all fields on a form.
|
* [api:ResetFormAction]: Action that clears all fields on a form.
|
||||||
|
|
||||||
## Formatted input
|
## Formatted input
|
||||||
|
|
||||||
* `[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.
|
* [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.
|
||||||
* `[api:ConfirmedPasswordField]`: Two masked input fields, checks for matching passwords.
|
* [api:ConfirmedPasswordField]: Two masked input fields, checks for matching passwords.
|
||||||
* `[api:CountryDropdownField]`: A simple extension to dropdown field, pre-configured to list countries.
|
* [api:CountryDropdownField]: A simple extension to dropdown field, pre-configured to list countries.
|
||||||
* `[api:CreditCardField]`: Allows input of credit card numbers via four separate form fields, including generic validation of its numeric values.
|
* [api:CreditCardField]: Allows input of credit card numbers via four separate form fields, including generic validation of its numeric values.
|
||||||
* `[api:CurrencyField]`: Text field, validating its input as a currency. Limited to US-centric formats, including a hardcoded currency symbol and decimal separators.
|
* [api:CurrencyField]: Text field, validating its input as a currency. Limited to US-centric formats, including a hardcoded currency symbol and decimal separators.
|
||||||
See `[api:MoneyField]` for a more flexible implementation.
|
See [api:MoneyField] for a more flexible implementation.
|
||||||
* `[api:DateField]`: Represents a date in a single input field, or separated into day, month, and year. Can optionally use a calendar popup.
|
* [api:DateField]: Represents a date in a single input field, or separated into day, month, and year. Can optionally use a calendar popup.
|
||||||
* `[api:DatetimeField]`: Combined date- and time field.
|
* [api:DatetimeField]: Combined date- and time field.
|
||||||
* `[api:EmailField]`: Text input field with validation for correct email format according to RFC 2822.
|
* [api:EmailField]: Text input field with validation for correct email format according to RFC 2822.
|
||||||
* `[api:GroupedDropdownField]`: Grouped dropdown, using <optgroup> tags.
|
* [api:GroupedDropdownField]: Grouped dropdown, using <optgroup> tags.
|
||||||
* `[api:HtmlEditorField]`: A WYSIWYG editor interface.
|
* [api:HtmlEditorField]: A WYSIWYG editor interface.
|
||||||
* `[api:MoneyField]`: A form field that can save into a `[api:Money]` database field.
|
* [api:MoneyField]: A form field that can save into a [api:Money] database field.
|
||||||
* `[api:NumericField]`: Text input field with validation for numeric values.
|
* [api:NumericField]: Text input field with validation for numeric values.
|
||||||
* `[api:OptionsetField]`: Set of radio buttons designed to emulate a dropdown.
|
* [api:OptionsetField]: Set of radio buttons designed to emulate a dropdown.
|
||||||
* `[api:PhoneNumberField]`: Field for displaying phone numbers. It separates the number, the area code and optionally the country code and extension.
|
* [api:PhoneNumberField]: Field for displaying phone numbers. It separates the number, the area code and optionally the country code and extension.
|
||||||
* `[api:SelectionGroup]`: SelectionGroup represents a number of fields which are selectable by a radio button that appears at the beginning of each item.
|
* [api:SelectionGroup]: SelectionGroup represents a number of fields which are selectable by a radio button that appears at the beginning of each item.
|
||||||
* `[api:TimeField]`: Input field with time-specific, localised validation.
|
* [api:TimeField]: Input field with time-specific, localised validation.
|
||||||
|
|
||||||
## Structure
|
## Structure
|
||||||
|
|
||||||
* `[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.
|
||||||
* `[api:FieldGroup] attached in CMS-context.
|
* [api:FieldGroup] attached in CMS-context.
|
||||||
* `[api:FieldList]`: Basic container for sequential fields, or nested fields through CompositeField.
|
* [api:FieldList]: Basic container for sequential fields, or nested fields through CompositeField.
|
||||||
* `[api:TabSet]`: Collection of fields which is rendered as separate tabs. Can be nested.
|
* [api:TabSet]: Collection of fields which is rendered as separate tabs. Can be nested.
|
||||||
* `[api:Tab]`: A single tab inside a `TabSet`.
|
* [api:Tab]: A single tab inside a `TabSet`.
|
||||||
* `[api:ToggleCompositeField]`: Allows visibility of a group of fields to be toggled.
|
* [api:ToggleCompositeField]: Allows visibility of a group of fields to be toggled.
|
||||||
|
|
||||||
## Files
|
## Files
|
||||||
|
|
||||||
* `[api:FileField]`: Simple file upload dialog.
|
* [api:FileField]: Simple file upload dialog.
|
||||||
* `[api:UploadField]`: File uploads through HTML5 features, including upload progress, preview and relationship management.
|
* [api:UploadField]: File uploads through HTML5 features, including upload progress, preview and relationship management.
|
||||||
|
|
||||||
## Relations
|
## Relations
|
||||||
|
|
||||||
* `[api:CheckboxSetField]`: Displays a set of checkboxes as a logical group.
|
* [api:CheckboxSetField]: Displays a set of checkboxes as a logical group.
|
||||||
* `[api:TableField]`: In-place editing of tabular data.
|
* [api:TableField]: In-place editing of tabular data.
|
||||||
* `[api:TreeDropdownField]`: Dropdown-like field that allows you to select an item from a hierarchical AJAX-expandable tree.
|
* [api:TreeDropdownField]: Dropdown-like field that allows you to select an item from a hierarchical AJAX-expandable tree.
|
||||||
* `[api:TreeMultiselectField]`: Represents many-many joins using a tree selector shown in a dropdown-like element
|
* [api:TreeMultiselectField]: Represents many-many joins using a tree selector shown in a dropdown-like element
|
||||||
* `[api:GridField]`: Displays a `[api:SS_List]` in a tabular format. Versatile base class which can be configured to allow editing, sorting, etc.
|
* [api:GridField]: Displays a [api:SS_List] in a tabular format. Versatile base class which can be configured to allow editing, sorting, etc.
|
||||||
* `[api:ListboxField]`: Multi-line listbox field, through `<select multiple>`.
|
* [api:ListboxField]: Multi-line listbox field, through `<select multiple>`.
|
||||||
|
|
||||||
|
|
||||||
## Utility
|
## Utility
|
||||||
|
|
||||||
* `[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
|
||||||
* `[api:HeaderField]`: Renders a simple HTML header element.
|
* [api:HeaderField]: Renders a simple HTML header element.
|
||||||
* `[api:HiddenField]` - Renders a hidden input field.
|
* [api:HiddenField] - Renders a hidden input field.
|
||||||
* `[api:LabelField]`: Simple label tag. This can be used to add extra text in your forms.
|
* [api:LabelField]: Simple label tag. This can be used to add extra text in your forms.
|
||||||
* `[api:LiteralField]`: Renders arbitrary HTML into a form.
|
* [api:LiteralField]: Renders arbitrary HTML into a form.
|
||||||
|
@ -122,7 +122,7 @@ field description as an example.
|
|||||||
$dateField->setAttribute('placeholder', $dateField->getConfig('dateformat'));
|
$dateField->setAttribute('placeholder', $dateField->getConfig('dateformat'));
|
||||||
|
|
||||||
<div class="notice" markdown="1">
|
<div class="notice" markdown="1">
|
||||||
Fields scaffolded through [api:DataObject::scaffoldCMSFields] automatically have a description attached to them.
|
Fields scaffolded through [api:DataObject::scaffoldCMSFields()] automatically have a description attached to them.
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
@ -10,8 +10,8 @@ On top of the base functionality, we use our own insertion dialogs to ensure you
|
|||||||
files. In addition to the markup managed by TinyMCE, we use [shortcodes](/developer_guides/extending/shortcodes) to store
|
files. In addition to the markup managed by TinyMCE, we use [shortcodes](/developer_guides/extending/shortcodes) to store
|
||||||
information about inserted images or media elements.
|
information about inserted images or media elements.
|
||||||
|
|
||||||
The framework comes with a `[api:HTMLEditorField]` form field class which encapsulates most of the required
|
The framework comes with a [api:HTMLEditorField] form field class which encapsulates most of the required
|
||||||
functionality. It is usually added through the `[api:DataObject->getCMSFields()]` method:
|
functionality. It is usually added through the [api:DataObject::getCMSFields()] method:
|
||||||
|
|
||||||
**mysite/code/MyObject.php**
|
**mysite/code/MyObject.php**
|
||||||
|
|
||||||
@ -33,14 +33,14 @@ functionality. It is usually added through the `[api:DataObject->getCMSFields()]
|
|||||||
|
|
||||||
### Specify which configuration to use
|
### Specify which configuration to use
|
||||||
|
|
||||||
By default, a config named 'cms' is used in any new `[api:HTMLEditorField]`.
|
By default, a config named 'cms' is used in any new [api:HTMLEditorField].
|
||||||
|
|
||||||
If you have created your own `[api:HtmlEditorConfig]` and would like to use it,
|
If you have created your own [api:HtmlEditorConfig] and would like to use it,
|
||||||
you can call `HtmlEditorConfig::set_active('myConfig')` and all subsequently created `[api:HTMLEditorField]`
|
you can call `HtmlEditorConfig::set_active('myConfig')` and all subsequently created [api:HTMLEditorField]
|
||||||
will use the configuration with the name 'myConfig'.
|
will use the configuration with the name 'myConfig'.
|
||||||
|
|
||||||
You can also specify which `[api:HtmlEditorConfig]` to use on a per field basis via the construct argument.
|
You can also specify which [api:HtmlEditorConfig] to use on a per field basis via the construct argument.
|
||||||
This is particularly useful if you need different configurations for multiple `[api:HTMLEditorField]` on the same page or form.
|
This is particularly useful if you need different configurations for multiple [api:HTMLEditorField] on the same page or form.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
class MyObject extends DataObject {
|
class MyObject extends DataObject {
|
||||||
@ -62,10 +62,10 @@ In the above example, the 'Content' field will use the default 'cms' config whil
|
|||||||
## Configuration
|
## Configuration
|
||||||
|
|
||||||
To keep the JavaScript editor configuration manageable and extensible, we've wrapped it in a PHP class called
|
To keep the JavaScript editor configuration manageable and extensible, we've wrapped it in a PHP class called
|
||||||
`[api:HtmlEditorConfig]`. The class comes with its own defaults, which are extended through the [Configuration API](../../configuration)
|
[api:HtmlEditorConfig]. The class comes with its own defaults, which are extended through the [Configuration API](../../configuration)
|
||||||
in the framework (and the `cms` module in case you've got that installed).
|
in the framework (and the `cms` module in case you've got that installed).
|
||||||
|
|
||||||
There can be multiple configs, which should always be created / accessed using `[api:HtmlEditorConfig::get]`. You can
|
There can be multiple configs, which should always be created / accessed using [api:HtmlEditorConfig::get()]. You can
|
||||||
then set the currently active config using `set_active()`.
|
then set the currently active config using `set_active()`.
|
||||||
|
|
||||||
<div class="info" markdown="1">
|
<div class="info" markdown="1">
|
||||||
@ -81,7 +81,7 @@ order is alphabetical, so if you set a TinyMCE option in the `aardvark/_config.p
|
|||||||
|
|
||||||
In its simplest form, the configuration of the editor includes adding and removing buttons and plugins.
|
In its simplest form, the configuration of the editor includes adding and removing buttons and plugins.
|
||||||
|
|
||||||
You can add plugins to the editor using the Framework's `[api:HtmlEditorConfig::enablePlugins]` method. This will
|
You can add plugins to the editor using the Framework's [api:HtmlEditorConfig::enablePlugins()] method. This will
|
||||||
transparently generate the relevant underlying TinyMCE code.
|
transparently generate the relevant underlying TinyMCE code.
|
||||||
|
|
||||||
**mysite/_config.php**
|
**mysite/_config.php**
|
||||||
@ -108,7 +108,7 @@ Buttons can also be removed:
|
|||||||
HtmlEditorConfig::get('cms')->removeButtons('tablecontrols', 'blockquote', 'hr');
|
HtmlEditorConfig::get('cms')->removeButtons('tablecontrols', 'blockquote', 'hr');
|
||||||
|
|
||||||
<div class="notice" markdown="1">
|
<div class="notice" markdown="1">
|
||||||
Internally `[api:HtmlEditorConfig]` uses the TinyMCE's `theme_advanced_buttons` option to configure these. See the
|
Internally [api:HtmlEditorConfig] uses the TinyMCE's `theme_advanced_buttons` option to configure these. See the
|
||||||
[TinyMCE documentation of this option](http://www.tinymce.com/wiki.php/Configuration:theme_advanced_buttons_1_n)
|
[TinyMCE documentation of this option](http://www.tinymce.com/wiki.php/Configuration:theme_advanced_buttons_1_n)
|
||||||
for more details.
|
for more details.
|
||||||
</div>
|
</div>
|
||||||
@ -182,7 +182,7 @@ plugin that adds a button to the editor:
|
|||||||
tinymce.PluginManager.add('myplugin', tinymce.plugins.myplugin);
|
tinymce.PluginManager.add('myplugin', tinymce.plugins.myplugin);
|
||||||
})();
|
})();
|
||||||
|
|
||||||
You can then enable this plugin through the [api:HtmlEditorConfig::enablePlugins]:
|
You can then enable this plugin through the [api:HtmlEditorConfig::enablePlugins()]:
|
||||||
|
|
||||||
**mysite/_config.php**
|
**mysite/_config.php**
|
||||||
:::php
|
:::php
|
||||||
@ -193,8 +193,8 @@ documentation, or browse through plugins that come with the Framework at `thirdp
|
|||||||
|
|
||||||
## Image and media insertion
|
## Image and media insertion
|
||||||
|
|
||||||
The `[api:HtmlEditorField]` API also handles inserting images and media files into the managed HTML content. It can be
|
The [api:HtmlEditorField] API also handles inserting images and media files into the managed HTML content. It can be
|
||||||
used both for referencing files on the webserver filesystem (through the `[api:File]` and `[api:Image]` APIs), as well
|
used both for referencing files on the webserver filesystem (through the [api:File] and [api:Image] APIs), as well
|
||||||
as hotlinking files from the web.
|
as hotlinking files from the web.
|
||||||
|
|
||||||
We use [shortcodes](/developer_guides/extending/shortcodes) to store information about inserted images or media elements. The
|
We use [shortcodes](/developer_guides/extending/shortcodes) to store information about inserted images or media elements. The
|
||||||
@ -207,7 +207,7 @@ The ["oEmbed" standard](http://www.oembed.com/) is implemented by many media ser
|
|||||||
representation of files just by referencing a website URL. For example, a content author can insert a playable youtube
|
representation of files just by referencing a website URL. For example, a content author can insert a playable youtube
|
||||||
video just by knowing its URL, as opposed to dealing with manual HTML code.
|
video just by knowing its URL, as opposed to dealing with manual HTML code.
|
||||||
|
|
||||||
oEmbed powers the "Insert from web" feature available through `[api:HtmlEditorField]`. Internally, it makes HTTP
|
oEmbed powers the "Insert from web" feature available through [api:HtmlEditorField]. Internally, it makes HTTP
|
||||||
queries to a list of external services if it finds a matching URL. These services are described in the
|
queries to a list of external services if it finds a matching URL. These services are described in the
|
||||||
`Oembed.providers` configuration. Since these requests are performed on page rendering, they typically have a long
|
`Oembed.providers` configuration. Since these requests are performed on page rendering, they typically have a long
|
||||||
cache time (multiple days).
|
cache time (multiple days).
|
||||||
@ -245,7 +245,7 @@ By default, TinyMCE and SilverStripe will generate valid HTML5 markup, but it wi
|
|||||||
`<article>` or `<figure>`. If you plan to use those, add them to the
|
`<article>` or `<figure>`. If you plan to use those, add them to the
|
||||||
[valid_elements](http://www.tinymce.com/wiki.php/Configuration:valid_elements) configuration setting.
|
[valid_elements](http://www.tinymce.com/wiki.php/Configuration:valid_elements) configuration setting.
|
||||||
|
|
||||||
Also, the `[api:SS_HTMLValue]` API underpinning the HTML processing parses the markup into a temporary object tree
|
Also, the [api:SS_HTMLValue] API underpinning the HTML processing parses the markup into a temporary object tree
|
||||||
which can be traversed and modified before saving. The built-in parser only supports HTML4 and XHTML syntax. In order
|
which can be traversed and modified before saving. The built-in parser only supports HTML4 and XHTML syntax. In order
|
||||||
to successfully process HTML5 tags, please use the
|
to successfully process HTML5 tags, please use the
|
||||||
['silverstripe/html5' module](https://github.com/silverstripe/silverstripe-html5).
|
['silverstripe/html5' module](https://github.com/silverstripe/silverstripe-html5).
|
||||||
@ -277,22 +277,22 @@ Adding functionality is a bit more advanced, you'll most likely
|
|||||||
need to add some fields to the PHP forms, as well as write some
|
need to add some fields to the PHP forms, as well as write some
|
||||||
JavaScript to ensure the values from those fields make it into the content
|
JavaScript to ensure the values from those fields make it into the content
|
||||||
elements (and back out in case an existing element gets edited).
|
elements (and back out in case an existing element gets edited).
|
||||||
There's lots of extension points in the `[api:HtmlEditorField_Toolbar]` class
|
There's lots of extension points in the [api:HtmlEditorField_Toolbar] class
|
||||||
to get you started.
|
to get you started.
|
||||||
|
|
||||||
### Security groups with their own editor configuration
|
### Security groups with their own editor configuration
|
||||||
|
|
||||||
Different groups of authors can be assigned their own config,
|
Different groups of authors can be assigned their own config,
|
||||||
e.g. a more restricted rule set for content reviewers (see the "Security" )
|
e.g. a more restricted rule set for content reviewers (see the "Security" )
|
||||||
The config is available on each user record through `[api:Member->getHtmlEditorConfigForCMS()]`.
|
The config is available on each user record through [api:Member::getHtmlEditorConfigForCMS()].
|
||||||
The group assignment is done through the "Security" interface for each `[api:Group]` record.
|
The group assignment is done through the "Security" interface for each [api:Group] record.
|
||||||
Note: The dropdown is only available if more than one config exists.
|
Note: The dropdown is only available if more than one config exists.
|
||||||
|
|
||||||
### Using the editor outside of the CMS
|
### Using the editor outside of the CMS
|
||||||
|
|
||||||
Each interface can have multiple fields of this type, each with their own toolbar to set formatting
|
Each interface can have multiple fields of this type, each with their own toolbar to set formatting
|
||||||
and insert HTML elements. They do share one common set of dialogs for inserting links and other media though,
|
and insert HTML elements. They do share one common set of dialogs for inserting links and other media though,
|
||||||
encapsulated in the `[api:HtmlEditorField_Toolbar]` class.
|
encapsulated in the [api:HtmlEditorField_Toolbar] class.
|
||||||
In the CMS, those dialogs are automatically instantiate, but in your own interfaces outside
|
In the CMS, those dialogs are automatically instantiate, but in your own interfaces outside
|
||||||
of the CMS you have to take care of instantiate yourself:
|
of the CMS you have to take care of instantiate yourself:
|
||||||
|
|
||||||
@ -327,7 +327,7 @@ WYSIWYG editors are complex beasts, so replacing it completely is a difficult ta
|
|||||||
The framework provides a wrapper implementation for the basic required functionality,
|
The framework provides a wrapper implementation for the basic required functionality,
|
||||||
mainly around selecting and inserting content into the editor view.
|
mainly around selecting and inserting content into the editor view.
|
||||||
Have a look in `HtmlEditorField.js` and the `ss.editorWrapper` object to get you started
|
Have a look in `HtmlEditorField.js` and the `ss.editorWrapper` object to get you started
|
||||||
on your own editor wrapper. Note that the `[api:HtmlEditorConfig]` is currently hardwired to support TinyMCE,
|
on your own editor wrapper. Note that the [api:HtmlEditorConfig] is currently hardwired to support TinyMCE,
|
||||||
so its up to you to either convert existing configuration as applicable,
|
so its up to you to either convert existing configuration as applicable,
|
||||||
or start your own configuration.
|
or start your own configuration.
|
||||||
|
|
||||||
|
@ -233,7 +233,7 @@ database.
|
|||||||
|
|
||||||
The `GridFieldDetailForm` component drives the record viewing and editing form. It takes its' fields from
|
The `GridFieldDetailForm` component drives the record viewing and editing form. It takes its' fields from
|
||||||
`DataObject->getCMSFields()` method but can be customized to accept different fields via the
|
`DataObject->getCMSFields()` method but can be customized to accept different fields via the
|
||||||
[api:GridFieldDetailForm->setFields] method.
|
[api:GridFieldDetailForm::setFields()] method.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$form = $gridField->getConfig()->getComponentByType('GridFieldDetailForm');
|
$form = $gridField->getConfig()->getComponentByType('GridFieldDetailForm');
|
||||||
@ -351,7 +351,7 @@ processed placeholders as opposed to native template syntax.
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
Now you can add other components into this area by returning them as an array from your
|
Now you can add other components into this area by returning them as an array from your
|
||||||
[api:GridFieldComponent->getHTMLFragments()] implementation:
|
[api:GridFieldComponent::getHTMLFragments()] implementation:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
<?php
|
<?php
|
||||||
|
@ -321,7 +321,7 @@ The `Upload_Validator` class has configuration options for setting the `default_
|
|||||||
You can specify the file extension or the app category (as specified in the `File` class) in square brackets. It supports setting the file size in bytes or using the syntax supported by `File::ini2bytes()`.
|
You can specify the file extension or the app category (as specified in the `File` class) in square brackets. It supports setting the file size in bytes or using the syntax supported by `File::ini2bytes()`.
|
||||||
|
|
||||||
|
|
||||||
You can also configure the underlying `[api:Upload]` class, by using the YAML config system.
|
You can also configure the underlying [api:Upload] class, by using the YAML config system.
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
Upload:
|
Upload:
|
||||||
|
@ -34,14 +34,14 @@ There's quite a bit in this function, so we'll step through one piece at a time.
|
|||||||
new TextareaField('Message')
|
new TextareaField('Message')
|
||||||
);
|
);
|
||||||
|
|
||||||
First we create all the fields we want in the contact form, and put them inside a FieldList. You can find a list of form fields available on the `[api:FormField]` page.
|
First we create all the fields we want in the contact form, and put them inside a FieldList. You can find a list of form fields available on the [api:FormField] page.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$actions = FieldList(
|
$actions = FieldList(
|
||||||
new FormAction('submit', 'Submit')
|
new FormAction('submit', 'Submit')
|
||||||
);
|
);
|
||||||
|
|
||||||
We then create a `[api:FieldList]` of the form actions, or the buttons that submit the form. Here we add a single form action, with the name 'submit', and the label 'Submit'. We'll use the name of the form action later.
|
We then create a [api:FieldList] of the form actions, or the buttons that submit the form. Here we add a single form action, with the name 'submit', and the label 'Submit'. We'll use the name of the form action later.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
return new Form($this, 'Form', $fields, $actions);
|
return new Form($this, 'Form', $fields, $actions);
|
||||||
@ -104,7 +104,7 @@ The final thing we do is return a 'thank you for your feedback' message to the u
|
|||||||
|
|
||||||
All forms have some basic validation built in – email fields will only let the user enter email addresses, number fields will only accept numbers, and so on. Sometimes you need more complicated validation, so you can define your own validation by extending the Validator class.
|
All forms have some basic validation built in – email fields will only let the user enter email addresses, number fields will only accept numbers, and so on. Sometimes you need more complicated validation, so you can define your own validation by extending the Validator class.
|
||||||
|
|
||||||
The framework comes with a predefined validator called `[api:RequiredFields]`, which performs the common task of making sure particular fields are filled out. Below is the code to add validation to a contact form:
|
The framework comes with a predefined validator called [api:RequiredFields], which performs the common task of making sure particular fields are filled out. Below is the code to add validation to a contact form:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
public function Form() {
|
public function Form() {
|
||||||
|
@ -46,7 +46,7 @@ be marked `private static` and follow the `lower_case_with_underscores` structur
|
|||||||
|
|
||||||
## Accessing and Setting Configuration Properties
|
## Accessing and Setting Configuration Properties
|
||||||
|
|
||||||
This can be done by calling the static method `[api:Config::inst]`, like so:
|
This can be done by calling the static method [api:Config::inst()], like so:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$config = Config::inst()->get('MyClass');
|
$config = Config::inst()->get('MyClass');
|
||||||
|
@ -69,4 +69,4 @@ provide the users a place to configure settings then the `SiteConfig` panel is t
|
|||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
* `[api:SiteConfig]`
|
* [api:SiteConfig]
|
||||||
|
@ -114,7 +114,7 @@ we added a `SayHi` method which is unique to our extension.
|
|||||||
|
|
||||||
If the `Extension` needs to modify an existing method it's a little trickier. It requires that the method you want to
|
If the `Extension` needs to modify an existing method it's a little trickier. It requires that the method you want to
|
||||||
customize has provided an *Extension Hook* in the place where you want to modify the data. An *Extension Hook* is done
|
customize has provided an *Extension Hook* in the place where you want to modify the data. An *Extension Hook* is done
|
||||||
through the `[api:Object->extend]` method.
|
through the [api:Object::extend()] method.
|
||||||
|
|
||||||
**framework/security/Member.php**
|
**framework/security/Member.php**
|
||||||
|
|
||||||
@ -209,8 +209,8 @@ In your [api:Extension] class you can only refer to the source object through th
|
|||||||
|
|
||||||
## Checking to see if an Object has an Extension
|
## Checking to see if an Object has an Extension
|
||||||
|
|
||||||
To see what extensions are currently enabled on an object, use [api:Object->getExtensionInstances] and
|
To see what extensions are currently enabled on an object, use [api:Object::getExtensionInstances()] and
|
||||||
[api:Object->hasExtension]
|
[api:Object::hasExtension()]
|
||||||
|
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
|
@ -33,7 +33,7 @@ Here's some syntax variations:
|
|||||||
|
|
||||||
Shortcodes are automatically parsed on any database field which is declared as [api:HTMLValue] or [api:HTMLText],
|
Shortcodes are automatically parsed on any database field which is declared as [api:HTMLValue] or [api:HTMLText],
|
||||||
when rendered into a template. This means you can use shortcodes on common fields like `SiteTree.Content`, and any
|
when rendered into a template. This means you can use shortcodes on common fields like `SiteTree.Content`, and any
|
||||||
other `[api:DataObject::$db]` definitions of these types.
|
other [api:DataObject::$db] definitions of these types.
|
||||||
|
|
||||||
Other fields can be manually parsed with shortcodes through the `parse` method.
|
Other fields can be manually parsed with shortcodes through the `parse` method.
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@ Some of the goals of dependency injection are:
|
|||||||
* Improve testability of code
|
* Improve testability of code
|
||||||
* Promoting abstraction of logic
|
* Promoting abstraction of logic
|
||||||
|
|
||||||
The following sums up the simplest usage of the `Injector` it creates a new object of type `ClassName` through `create`
|
The following sums up the simplest usage of the `Injector` it creates a new object of type `MyClassName` through `create`
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$object = Injector::inst()->create('MyClassName');
|
$object = Injector::inst()->create('MyClassName');
|
||||||
@ -57,7 +57,7 @@ object instance as the first call.
|
|||||||
|
|
||||||
## Dependencies
|
## Dependencies
|
||||||
|
|
||||||
The `Injector` API can be used to define the types of `$dependancies` that an object requires.
|
The `Injector` API can be used to define the types of `$dependencies` that an object requires.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
<?php
|
<?php
|
||||||
@ -88,7 +88,7 @@ When creating a new instance of `MyController` the dependencies on that class wi
|
|||||||
echo (is_string($object->textProperty));
|
echo (is_string($object->textProperty));
|
||||||
// returns true;
|
// returns true;
|
||||||
|
|
||||||
The [Configuration YAML](../configuration) does the hard work of configuring those `$dependancies` for us.
|
The [Configuration YAML](../configuration) does the hard work of configuring those `$dependencies` for us.
|
||||||
|
|
||||||
**mysite/_config/app.yml**
|
**mysite/_config/app.yml**
|
||||||
|
|
||||||
|
@ -96,7 +96,7 @@ environments. If for some reason you don't have access to the command line, you
|
|||||||
|
|
||||||
### Via the CLI
|
### Via the CLI
|
||||||
|
|
||||||
The [sake](../cli) executable that comes with SilverStripe can trigger a customized `[api:TestRunner]` class that
|
The [sake](../cli) executable that comes with SilverStripe can trigger a customized [api:TestRunner] class that
|
||||||
handles the PHPUnit configuration and output formatting. While the custom test runner a handy tool, it's also more
|
handles the PHPUnit configuration and output formatting. While the custom test runner a handy tool, it's also more
|
||||||
limited than using `phpunit` directly, particularly around formatting test output.
|
limited than using `phpunit` directly, particularly around formatting test output.
|
||||||
|
|
||||||
@ -133,7 +133,7 @@ Unless executing a coverage report there is no need to have xDebug enabled.
|
|||||||
sudo php5enmod xdebug
|
sudo php5enmod xdebug
|
||||||
|
|
||||||
### Use SQLite In Memory
|
### Use SQLite In Memory
|
||||||
SQLIte can be configured to fun in memory as opposed to disk and this makes testing an order of magnitude faster. To effect this change add the following to mysite/_config.php - this enables an optional flag to switch between MySQL and SQLite. Note also that the package silverstripe/sqlite3 will need installed, version will vary depending on which version of SilverStripe is being tested.
|
SQLIte can be configured to run in memory as opposed to disk and this makes testing an order of magnitude faster. To effect this change add the following to mysite/_config.php - this enables an optional flag to switch between MySQL and SQLite. Note also that the package silverstripe/sqlite3 will need installed, version will vary depending on which version of SilverStripe is being tested.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
if(Director::isDev()) {
|
if(Director::isDev()) {
|
||||||
|
@ -6,7 +6,7 @@ summary: Populate test databases with fake seed data.
|
|||||||
To test functionality correctly, we must use consistent data. If we are testing our code with the same data each
|
To test functionality correctly, we must use consistent data. If we are testing our code with the same data each
|
||||||
time, we can trust our tests to yield reliable results and to identify when the logic changes. Each test run in
|
time, we can trust our tests to yield reliable results and to identify when the logic changes. Each test run in
|
||||||
SilverStripe starts with a fresh database containing no records. `Fixtures` provide a way to describe the initial data
|
SilverStripe starts with a fresh database containing no records. `Fixtures` provide a way to describe the initial data
|
||||||
to load into the database. The `[api:SapphireTest]` class takes care of populating a test database with data from
|
to load into the database. The [api:SapphireTest] class takes care of populating a test database with data from
|
||||||
fixtures - all we have to do is define them.
|
fixtures - all we have to do is define them.
|
||||||
|
|
||||||
Fixtures are defined in `YAML`. `YAML` is a markup language which is deliberately simple and easy to read, so it is
|
Fixtures are defined in `YAML`. `YAML` is a markup language which is deliberately simple and easy to read, so it is
|
||||||
@ -192,7 +192,7 @@ To provide the value for the `many_many_extraField` use the YAML list syntax.
|
|||||||
|
|
||||||
While manually defined fixtures provide full flexibility, they offer very little in terms of structure and convention.
|
While manually defined fixtures provide full flexibility, they offer very little in terms of structure and convention.
|
||||||
|
|
||||||
Alternatively, you can use the `[api:FixtureFactory]` class, which allows you to set default values, callbacks on object
|
Alternatively, you can use the [api:FixtureFactory] class, which allows you to set default values, callbacks on object
|
||||||
creation, and dynamic/lazy value setting.
|
creation, and dynamic/lazy value setting.
|
||||||
|
|
||||||
<div class="hint" markdown='1'>
|
<div class="hint" markdown='1'>
|
||||||
|
@ -60,7 +60,7 @@ The test database is rebuilt every time one of these methods is run.
|
|||||||
|
|
||||||
Inside our test method is the `objFromFixture` method that will generate an object for us based on data from our fixture
|
Inside our test method is the `objFromFixture` method that will generate an object for us based on data from our fixture
|
||||||
file. To identify to the object, we provide a class name and an identifier. The identifier is specified in the YAML file
|
file. To identify to the object, we provide a class name and an identifier. The identifier is specified in the YAML file
|
||||||
but not saved in the database anywhere, `objFromFixture` looks the `[api:DataObject]` up in memory rather than using the
|
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.
|
||||||
|
|
||||||
The final part of our test is an assertion command, `assertEquals`. An assertion command allows us to test for something
|
The final part of our test is an assertion command, `assertEquals`. An assertion command allows us to test for something
|
||||||
|
@ -2,8 +2,8 @@ title: How to test emails within unit tests
|
|||||||
|
|
||||||
# Testing Email within Unit Tests
|
# Testing Email within Unit Tests
|
||||||
|
|
||||||
SilverStripe's test system has built-in support for testing emails sent using the `[api:Email]` class. If you are
|
SilverStripe's test system has built-in support for testing emails sent using the [api:Email] class. If you are
|
||||||
running a `[api:SapphireTest]` test, then it holds off actually sending the email, and instead lets you assert that an
|
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.
|
email was sent using this method.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -15,7 +15,7 @@ email was sent using this method.
|
|||||||
$e->send();
|
$e->send();
|
||||||
}
|
}
|
||||||
|
|
||||||
To test that `MyMethod` sends the correct email, use the [api:SapphireTest::assertEmailSent] method.
|
To test that `MyMethod` sends the correct email, use the [api:SapphireTest::assertEmailSent()] method.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
$this->assertEmailSent($to, $from, $subject, $body);
|
$this->assertEmailSent($to, $from, $subject, $body);
|
||||||
|
@ -55,7 +55,7 @@ should have the appropriate permissions to create new databases on your server,
|
|||||||
|
|
||||||
## 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
|
||||||
@ -100,7 +100,7 @@ All command-line arguments are documented on
|
|||||||
### Via the "sake" Wrapper on Command Line
|
### Via the "sake" Wrapper on Command Line
|
||||||
|
|
||||||
The [sake](/developer_guides/cli/) executable that comes with SilverStripe can trigger a customized
|
The [sake](/developer_guides/cli/) 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.
|
||||||
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,
|
||||||
particularly around formatting test output.
|
particularly around formatting test output.
|
||||||
|
|
||||||
|
@ -36,7 +36,7 @@ on a public server very carefully.
|
|||||||
|
|
||||||
Test mode is designed for staging environments or other private collaboration sites before deploying a site live.
|
Test mode is designed for staging environments or other private collaboration sites before deploying a site live.
|
||||||
|
|
||||||
In this mode error messages are hidden from the user and SilverStripe includes `[api:BasicAuth]` integration if you
|
In this mode error messages are hidden from the user and SilverStripe includes [api:BasicAuth] integration if you
|
||||||
want to password protect the site. You can enable that but adding this to your `mysite/_config/app.yml` file:
|
want to password protect the site. You can enable that but adding this to your `mysite/_config/app.yml` file:
|
||||||
|
|
||||||
:::yml
|
:::yml
|
||||||
|
@ -27,7 +27,7 @@ Append the option and corresponding value to your URL in your browser's address
|
|||||||
| isDev | | 1 | | Put the site into [development mode](../), enabling debugging messages to the browser on a live server. For security, you'll be asked to log in with an administrator log-in. Will persist for the current browser session. |
|
| isDev | | 1 | | Put the site into [development mode](../), enabling debugging messages to the browser on a live server. For security, you'll be asked to log in with an administrator log-in. Will persist for the current browser session. |
|
||||||
| isTest | | 1 | | See above. |
|
| isTest | | 1 | | See above. |
|
||||||
| 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
|
||||||
|
|
||||||
|
@ -35,7 +35,7 @@ Here are some more complex examples:
|
|||||||
<% end_cached %>
|
<% end_cached %>
|
||||||
|
|
||||||
An additional global key is incorporated in the cache lookup. The default value for this is
|
An additional global key is incorporated in the cache lookup. The default value for this is
|
||||||
`$CurrentReadingMode, $CurrentUser.ID`. This ensures that the current `[api:Versioned]` state and user ID are used.
|
`$CurrentReadingMode, $CurrentUser.ID`. This ensures that the current [api:Versioned] state and user ID are used.
|
||||||
This may be configured by changing the config value of `SSViewer.global_key`. It is also necessary to flush the
|
This may be configured by changing the config value of `SSViewer.global_key`. It is also necessary to flush the
|
||||||
template caching when modifying this config, as this key is cached within the template itself.
|
template caching when modifying this config, as this key is cached within the template itself.
|
||||||
|
|
||||||
|
@ -4,14 +4,14 @@
|
|||||||
|
|
||||||
The framework uses caches to store infrequently changing values.
|
The framework uses caches to store infrequently changing values.
|
||||||
By default, the storage mechanism is simply the filesystem, although
|
By default, the storage mechanism is simply the filesystem, although
|
||||||
other cache backends can be configured. All caches use the `[api:SS_Cache]` API.
|
other cache backends can be configured. All caches use the [api:SS_Cache] API.
|
||||||
|
|
||||||
The most common caches are manifests of various resources:
|
The most common caches are manifests of various resources:
|
||||||
|
|
||||||
* PHP class locations (`[api:SS_ClassManifest]`)
|
* PHP class locations ([api:SS_ClassManifest])
|
||||||
* Template file locations and compiled templates (`[api:SS_TemplateManifest]`)
|
* Template file locations and compiled templates ([api:SS_TemplateManifest])
|
||||||
* Configuration settings from YAML files (`[api:SS_ConfigManifest]`)
|
* Configuration settings from YAML files ([api:SS_ConfigManifest])
|
||||||
* Language files (`[api:i18n]`)
|
* Language files ([api:i18n])
|
||||||
|
|
||||||
Flushing the various manifests is performed through a GET
|
Flushing the various manifests is performed through a GET
|
||||||
parameter (`flush=1`). Since this action requires more server resources than normal requests,
|
parameter (`flush=1`). Since this action requires more server resources than normal requests,
|
||||||
@ -23,7 +23,7 @@ executing the action is limited to the following cases when performed via a web
|
|||||||
|
|
||||||
## The Cache API
|
## The Cache API
|
||||||
|
|
||||||
The `[api:SS_Cache]` class provides a bunch of static functions wrapping the Zend_Cache system
|
The [api:SS_Cache] class provides a bunch of static functions wrapping the Zend_Cache system
|
||||||
in something a little more easy to use with the SilverStripe config system.
|
in something a little more easy to use with the SilverStripe config system.
|
||||||
|
|
||||||
A `Zend_Cache` has both a frontend (determines how to get the value to cache,
|
A `Zend_Cache` has both a frontend (determines how to get the value to cache,
|
||||||
|
@ -4,11 +4,11 @@ title: Members
|
|||||||
|
|
||||||
## 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()**
|
||||||
|
|
||||||
@ -39,11 +39,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:DataExtension]`
|
This is the least desirable way of extending the [api:Member] class. It's better to use [api:DataExtension]
|
||||||
(see below).
|
(see below).
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
You can define subclasses of `[api:Member]` to add extra fields or functionality to the built-in membership system.
|
You can define 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 {
|
||||||
@ -54,7 +54,7 @@ You can define subclasses of `[api:Member]` to add extra fields or functionality
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
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
|
||||||
@ -65,8 +65,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 public function getCMSFields(), then you can change the form that is used to view & edit member
|
If you overload the built-in public 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:FieldList]` object. You should generally start by calling
|
details in the newsletter system. This function returns a [api:FieldList] object. You should generally start by calling
|
||||||
parent::getCMSFields() and manipulate the `[api:FieldList]` from there.
|
parent::getCMSFields() and manipulate the [api:FieldList] from there.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
public function getCMSFields() {
|
public function getCMSFields() {
|
||||||
@ -80,11 +80,11 @@ parent::getCMSFields() and manipulate the `[api:FieldList]` 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 Extension
|
## Member Role Extension
|
||||||
@ -98,9 +98,9 @@ class. A better way is to use role extensions to add this behaviour. Add the fol
|
|||||||
extensions:
|
extensions:
|
||||||
- MyMemberExtension
|
- MyMemberExtension
|
||||||
|
|
||||||
A role extension is simply a subclass of `[api:DataExtension]` that is designed to be used to add behaviour to `[api:Member]`.
|
A role extension is simply a subclass of [api:DataExtension] 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 MyMemberExtension extends DataExtension {
|
class MyMemberExtension extends DataExtension {
|
||||||
@ -130,4 +130,4 @@ things, you should add appropriate `[api:Permission::checkMember()]` calls to th
|
|||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
`[api:Member]`
|
[api:Member]
|
||||||
|
@ -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 (see [api:CMSMain], with no arguments).
|
||||||
|
|
||||||
*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]
|
||||||
|
@ -19,7 +19,7 @@ The simple usage, Permission::check("PERM_CODE") will detect if the currently lo
|
|||||||
|
|
||||||
## PermissionProvider
|
## PermissionProvider
|
||||||
|
|
||||||
`[api:PermissionProvider]` is an interface which lets you define a method *providePermissions()*.
|
[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.
|
This method should return a map of permission code names with a human readable explanation of its purpose.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -98,4 +98,4 @@ Internally, this checks that the user has any of the defined `CMS_ACCESS_*` perm
|
|||||||
|
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:Permission]`
|
[api:Permission]
|
||||||
|
@ -3,33 +3,33 @@ summary: Explains SilverStripe's Authentication options and custom authenticator
|
|||||||
|
|
||||||
# Authentication
|
# Authentication
|
||||||
|
|
||||||
By default, SilverStripe provides a `[api:MemberAuthenticator]` class which hooks into its own internal
|
By default, SilverStripe provides a [api:MemberAuthenticator] class which hooks into its own internal
|
||||||
authentication system.
|
authentication system.
|
||||||
|
|
||||||
The main login system uses these controllers to handle the various security requests:
|
The main login system uses these controllers to handle the various security requests:
|
||||||
|
|
||||||
`[api:Security]` - Which is the controller which handles most front-end security requests, including logging in, logging out, resetting password, or changing password. This class also provides an interface to allow configured `[api:Authenticator]` classes to each display a custom login form.
|
[api:Security] - Which is the controller which handles most front-end security requests, including logging in, logging out, resetting password, or changing password. This class also provides an interface to allow configured [api:Authenticator] classes to each display a custom login form.
|
||||||
|
|
||||||
`[api:CMSSecurity]` - Which is the controller which handles security requests within the CMS, and allows users to re-login without leaving the CMS.
|
[api:CMSSecurity] - Which is the controller which handles security requests within the CMS, and allows users to re-login without leaving the CMS.
|
||||||
|
|
||||||
## Member Authentication
|
## Member Authentication
|
||||||
|
|
||||||
The default member authentication system is implemented in the following classes:
|
The default member authentication system is implemented in the following classes:
|
||||||
|
|
||||||
`[api:MemberAuthenticator]` - Which is the default member authentication implementation. This uses the email and password stored internally for each member to authenticate them.
|
[api:MemberAuthenticator] - Which is the default member authentication implementation. This uses the email and password stored internally for each member to authenticate them.
|
||||||
|
|
||||||
`[api:MemberLoginForm]` - Is the default form used by `MemberAuthenticator`, and is displayed on the public site at the url `Security/login` by default.
|
[api:MemberLoginForm] - Is the default form used by `MemberAuthenticator`, and is displayed on the public site at the url `Security/login` by default.
|
||||||
|
|
||||||
`[api:CMSMemberLoginForm]` - Is the secondary form used by `MemberAuthenticator`, and will be displayed to the user within the CMS any time their session expires or they are logged out via an action. This form is presented via a popup dialog, and can be used to re-authenticate that user automatically without them having to lose their workspace. E.g. if editing a form, the user can login and continue to publish their content.
|
[api:CMSMemberLoginForm] - Is the secondary form used by `MemberAuthenticator`, and will be displayed to the user within the CMS any time their session expires or they are logged out via an action. This form is presented via a popup dialog, and can be used to re-authenticate that user automatically without them having to lose their workspace. E.g. if editing a form, the user can login and continue to publish their content.
|
||||||
|
|
||||||
## Custom Authentication
|
## Custom Authentication
|
||||||
|
|
||||||
Additional authentication methods (oauth, etc) can be implemented by creating custom implementations of each of the
|
Additional authentication methods (oauth, etc) can be implemented by creating custom implementations of each of the
|
||||||
following base classes:
|
following base classes:
|
||||||
|
|
||||||
`[api:Authenticator]` - The base class for authentication systems. This class also acts as the factory to generate various login forms for parts of the system. If an authenticator supports in-cms reauthentication then it will be necessary to override the `supports_cms` and `get_cms_login_form` methods.
|
[api:Authenticator] - The base class for authentication systems. This class also acts as the factory to generate various login forms for parts of the system. If an authenticator supports in-cms reauthentication then it will be necessary to override the `supports_cms` and `get_cms_login_form` methods.
|
||||||
|
|
||||||
`[api:LoginForm]` - which is the base class for a login form which links to a specific authenticator. At the very least, it will be necessary to implement a form class which provides a default login interface. If in-cms re-authentication is desired, then a specialised subclass of this method may be necessary. For example, this form could be extended to require confirmation of username as well as password.
|
[api:LoginForm] - which is the base class for a login form which links to a specific authenticator. At the very least, it will be necessary to implement a form class which provides a default login interface. If in-cms re-authentication is desired, then a specialised subclass of this method may be necessary. For example, this form could be extended to require confirmation of username as well as password.
|
||||||
|
|
||||||
## Default Admin
|
## Default Admin
|
||||||
|
|
||||||
@ -44,6 +44,6 @@ It is advisable to configure this user in your `_ss_environment.php` file outsid
|
|||||||
define('SS_DEFAULT_ADMIN_USERNAME', 'admin');
|
define('SS_DEFAULT_ADMIN_USERNAME', 'admin');
|
||||||
define('SS_DEFAULT_ADMIN_PASSWORD', 'password');
|
define('SS_DEFAULT_ADMIN_PASSWORD', 'password');
|
||||||
|
|
||||||
When a user logs in with these credentials, then a `[api:Member]` with the Email 'admin' will be generated in
|
When a user logs in with these credentials, then a [api:Member] with the Email 'admin' will be generated in
|
||||||
the database, but without any password information. This means that the password can be reset or changed by simply
|
the database, but without any password information. This means that the password can be reset or changed by simply
|
||||||
updating the `_ss_environment.php` file.
|
updating the `_ss_environment.php` file.
|
||||||
|
@ -58,10 +58,10 @@ Parameterised updates and inserts are also supported, but the syntax is a little
|
|||||||
|
|
||||||
SilverStripe internally will use parameterised queries in SQL statements wherever possible.
|
SilverStripe internally will use parameterised queries in SQL statements wherever possible.
|
||||||
|
|
||||||
If necessary Silverstripe performs any required escaping through database-specific methods (see `[api:Database->addslashes()]`).
|
If necessary Silverstripe performs any required escaping through database-specific methods (see [api:Database::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://de3.php.net/mysql_real_escape_string)`.
|
||||||
|
|
||||||
* Most `[api:DataList]` accessors (see escaping note in method documentation)
|
* Most [api:DataList] accessors (see escaping note in method documentation)
|
||||||
* DataObject::get_by_id()
|
* DataObject::get_by_id()
|
||||||
* DataObject::update()
|
* DataObject::update()
|
||||||
* DataObject::castedUpdate()
|
* DataObject::castedUpdate()
|
||||||
@ -213,8 +213,8 @@ We recommend configuring [shortcodes](/developer_guides/extending/shortcodes) th
|
|||||||
|
|
||||||
### 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](/developer_guides/model/data_types_and_casting) its string value into a `[api:DBField]` object.
|
object-properties by [casting](/developer_guides/model/data_types_and_casting) its string value into a [api:DBField] object.
|
||||||
|
|
||||||
PHP:
|
PHP:
|
||||||
|
|
||||||
@ -297,7 +297,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 *Convert::raw2xml()* and *Convert::raw2att()* (which is
|
||||||
also used by *XML* and *ATT* in template code).
|
also used by *XML* and *ATT* in template code).
|
||||||
|
|
||||||
PHP:
|
PHP:
|
||||||
@ -372,14 +372,14 @@ SilverStripe has built-in countermeasures against [CSRF](http://shiflett.org/art
|
|||||||
will automatically contain a `SecurityID` parameter which is generated as a secure hash on the server, connected to the
|
will automatically contain a `SecurityID` parameter which is generated as a secure hash on the server, connected to the
|
||||||
currently active session of the user. If this form is submitted without this parameter, or if the parameter doesn't
|
currently active session of the user. If this form is submitted without this parameter, or if the parameter doesn't
|
||||||
match the hash stored in the users session, the request is discarded.
|
match the hash stored in the users session, the request is discarded.
|
||||||
You can disable this behaviour through `[api:Form->disableSecurityToken()]`.
|
You can disable this behaviour through [api:Form::disableSecurityToken()].
|
||||||
|
|
||||||
It is also recommended to limit form submissions to the intended HTTP verb (mostly `GET` or `POST`)
|
It is also recommended to limit form submissions to the intended HTTP verb (mostly `GET` or `POST`)
|
||||||
through `[api:Form->setStrictFormMethodCheck()]`.
|
through [api:Form::setStrictFormMethodCheck()].
|
||||||
|
|
||||||
Sometimes you need to handle state-changing HTTP submissions which aren't handled through
|
Sometimes you need to handle state-changing HTTP submissions which aren't handled through
|
||||||
SilverStripe's form system. In this case, you can also check the current HTTP request
|
SilverStripe's form system. In this case, you can also check the current HTTP request
|
||||||
for a valid token through `[api:SecurityToken::checkRequest()]`.
|
for a valid token through [api:SecurityToken::checkRequest()].
|
||||||
|
|
||||||
## Casting user input
|
## Casting user input
|
||||||
|
|
||||||
@ -580,7 +580,11 @@ server IPs using the SS_TRUSTED_PROXY_IPS define in your _ss_environment.php.
|
|||||||
|
|
||||||
:::php
|
:::php
|
||||||
define('SS_TRUSTED_PROXY_IPS', '127.0.0.1,192.168.0.1');
|
define('SS_TRUSTED_PROXY_IPS', '127.0.0.1,192.168.0.1');
|
||||||
|
define('SS_TRUSTED_PROXY_HOST_HEADER', 'HTTP_X_FORWARDED_HOST');
|
||||||
|
define('SS_TRUSTED_PROXY_IP_HEADER', 'HTTP_X_FORWARDED_FOR');
|
||||||
|
define('SS_TRUSTED_PROXY_PROTOCOL_HEADER', 'HTTP_X_FORWARDED_PROTOCOL');
|
||||||
|
|
||||||
|
At the same time, you'll also need to define which headers you trust from these proxy IPs. Since there are multiple ways through which proxies can pass through HTTP information on the original hostname, IP and protocol, these values need to be adjusted for your specific proxy. The header names match their equivalent `$_SERVER` values.
|
||||||
|
|
||||||
If there is no proxy server, 'none' can be used to distrust all clients.
|
If there is no proxy server, 'none' can be used to distrust all clients.
|
||||||
If only trusted servers will make requests then you can use '*' to trust all clients.
|
If only trusted servers will make requests then you can use '*' to trust all clients.
|
||||||
@ -603,7 +607,6 @@ In a future release this behaviour will be changed to be on by default, and this
|
|||||||
variable will be no longer necessary, thus it will be necessary to always set
|
variable will be no longer necessary, thus it will be necessary to always set
|
||||||
SS_TRUSTED_PROXY_IPS if using a proxy.
|
SS_TRUSTED_PROXY_IPS if using a proxy.
|
||||||
|
|
||||||
|
|
||||||
## Related
|
## Related
|
||||||
|
|
||||||
* [http://silverstripe.org/security-releases/](http://silverstripe.org/security-releases/)
|
* [http://silverstripe.org/security-releases/](http://silverstripe.org/security-releases/)
|
||||||
|
@ -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
|
||||||
@ -52,7 +52,7 @@ The simplest way to use [api:CsvBulkLoader] is through a [api:ModelAdmin] interf
|
|||||||
'Player'
|
'Player'
|
||||||
);
|
);
|
||||||
private static $model_importers = array(
|
private static $model_importers = array(
|
||||||
'Player' => 'PlayerCsvBulkLoader',
|
'Player' => 'CsvBulkLoader',
|
||||||
);
|
);
|
||||||
private static $url_segment = 'players';
|
private static $url_segment = 'players';
|
||||||
}
|
}
|
||||||
|
@ -2,7 +2,7 @@ summary: Consume external data through their RESTFul interfaces.
|
|||||||
|
|
||||||
# Restful Service
|
# Restful Service
|
||||||
|
|
||||||
`[api:RestfulService]` is used to enable connections to remote web services through PHP's `curl` command. It provides an
|
[api:RestfulService] is used to enable connections to remote web services through PHP's `curl` command. It provides an
|
||||||
interface and utility functions for generating a valid request and parsing the response returned from the web service.
|
interface and utility functions for generating a valid request and parsing the response returned from the web service.
|
||||||
|
|
||||||
<div class="alert" markdown="1">
|
<div class="alert" markdown="1">
|
||||||
@ -134,7 +134,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),
|
If the web service returned an error (for example, API key not available or inadequate parameters),
|
||||||
`[api:RestfulService]` can delegate the error handling to it's descendant class. To handle the errors, subclass
|
[api:RestfulService] can delegate the error handling to it's descendant class. To handle the errors, subclass
|
||||||
`RestfulService and define a function called errorCatch.
|
`RestfulService and define a function called errorCatch.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -220,4 +220,4 @@ $service->request('service.json', 'GET', null, null, $curlOptions);
|
|||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
* `[api:RestfulService]`
|
* [api:RestfulService]
|
||||||
|
@ -3,15 +3,15 @@ summary: Output records from your database as an RSS Feed.
|
|||||||
|
|
||||||
# RSS Feed
|
# RSS Feed
|
||||||
|
|
||||||
Generating RSS / Atom-feeds is a matter of rendering a `[api:SS_List]` instance through the `[api:RSSFeed]` class.
|
Generating RSS / Atom-feeds is a matter of rendering a [api:SS_List] instance through the [api:RSSFeed] class.
|
||||||
|
|
||||||
The `[api:RSSFeed]` class doesn't limit you to generating article based feeds, it is just as easy to create a feed of
|
The [api:RSSFeed] class doesn't limit you to generating article based feeds, it is just as easy to create a feed of
|
||||||
your current staff members, comments or any other custom `[api:DataObject]` subclasses you have defined. The only
|
your current staff members, comments or any other custom [api:DataObject] subclasses you have defined. The only
|
||||||
logical limitation here is that every item in the RSS-feed should be accessible through a URL on your website, so it's
|
logical limitation here is that every item in the RSS-feed should be accessible through a URL on your website, so it's
|
||||||
advisable to just create feeds from subclasses of `[api:SiteTree]`.
|
advisable to just create feeds from subclasses of [api:SiteTree].
|
||||||
|
|
||||||
<div class="warning" markdown="1">
|
<div class="warning" markdown="1">
|
||||||
If you wish to generate an RSS feed that contains a `[api:DataObject]`, ensure you define a `AbsoluteLink` method on
|
If you wish to generate an RSS feed that contains a [api:DataObject], ensure you define a `AbsoluteLink` method on
|
||||||
the object.
|
the object.
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -47,7 +47,7 @@ will normally go in your `Controllers` `init` method.
|
|||||||
|
|
||||||
### Showing the 10 most recently updated pages
|
### Showing the 10 most recently updated pages
|
||||||
|
|
||||||
You can use `[api:RSSFeed]` to easily create a feed showing your latest Page updates. The following example adds a page
|
You can use [api:RSSFeed] to easily create a feed showing your latest Page updates. The following example adds a page
|
||||||
`/home/rss/` which displays an XML file the latest updated pages.
|
`/home/rss/` which displays an XML file the latest updated pages.
|
||||||
|
|
||||||
**mysite/code/Page.php**
|
**mysite/code/Page.php**
|
||||||
@ -87,7 +87,7 @@ You can use `[api:RSSFeed]` to easily create a feed showing your latest Page upd
|
|||||||
|
|
||||||
### Rendering DataObjects in a RSSFeed
|
### Rendering DataObjects in a RSSFeed
|
||||||
|
|
||||||
DataObjects can be rendered in the feed as well, however, since they aren't explicitly `[api:SiteTree]` subclasses we
|
DataObjects can be rendered in the feed as well, however, since they aren't explicitly [api:SiteTree] subclasses we
|
||||||
need to include a function `AbsoluteLink` to allow the RSS feed to link through to the item.
|
need to include a function `AbsoluteLink` to allow the RSS feed to link through to the item.
|
||||||
|
|
||||||
<div class="info">
|
<div class="info">
|
||||||
@ -193,4 +193,4 @@ As we've added a new template (PlayersRss.ss) make sure you clear your SilverStr
|
|||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
|
|
||||||
* `[api:RSSFeed]`
|
* [api:RSSFeed]
|
||||||
|
@ -2,7 +2,7 @@ title: Embed an RSS Feed
|
|||||||
|
|
||||||
# Embed an RSS Feed
|
# Embed an RSS Feed
|
||||||
|
|
||||||
`[api:RestfulService]` can be used to easily embed an RSS feed from a site. In this How to we'll embed the latest
|
[api:RestfulService] can be used to easily embed an RSS feed from a site. In this How to we'll embed the latest
|
||||||
weather information from the Yahoo Weather API.
|
weather information from the Yahoo Weather API.
|
||||||
|
|
||||||
First, we write the code to query the API feed.
|
First, we write the code to query the API feed.
|
||||||
@ -54,4 +54,4 @@ single field `Description`.
|
|||||||
## Related
|
## Related
|
||||||
|
|
||||||
* [RestfulService Documentation](../restfulservice)
|
* [RestfulService Documentation](../restfulservice)
|
||||||
* `[api:RestfulService]`
|
* [api:RestfulService]
|
||||||
|
@ -204,7 +204,7 @@ 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
|
||||||
|
|
||||||
|
|
||||||
## Related Documentation
|
## Related Documentation
|
||||||
|
@ -41,8 +41,8 @@ default site search, have a look at those extensions and modify as required.
|
|||||||
|
|
||||||
### Fulltext Filter
|
### Fulltext Filter
|
||||||
|
|
||||||
SilverStripe provides a `[api:FulltextFiler]` which you can use to perform custom fulltext searches on
|
SilverStripe provides a [api:FulltextFilter] which you can use to perform custom fulltext searches on
|
||||||
`[api:DataList]`'s.
|
[api:DataList]s.
|
||||||
|
|
||||||
Example DataObject:
|
Example DataObject:
|
||||||
|
|
||||||
|
@ -24,7 +24,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
|
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.
|
you want to set.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
@ -39,7 +39,7 @@ for a complete listing of 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}
|
||||||
|
|
||||||
@ -72,9 +72,9 @@ to write your own logic for any frontend output.
|
|||||||
Config::inst()->update('i18n', 'date_format', 'dd.MM.YYYY');
|
Config::inst()->update('i18n', 'date_format', 'dd.MM.YYYY');
|
||||||
Config::inst()->update('i18n', 'time_format', 'HH:mm');
|
Config::inst()->update('i18n', 'time_format', 'HH:mm');
|
||||||
|
|
||||||
Most localization routines in SilverStripe use the [Zend_Date API](http://framework.zend.com/manual/en/zend.date.html).
|
Most localization routines in SilverStripe use the [Zend_Date API](http://framework.zend.com/manual/1.12/en/zend.date.overview.html).
|
||||||
This means all formats are defined in
|
This means all formats are defined in
|
||||||
[ISO date format](http://framework.zend.com/manual/en/zend.date.constants.html#zend.date.constants.selfdefinedformats),
|
[ISO date format](http://framework.zend.com/manual/1.12/en/zend.date.constants.html),
|
||||||
not PHP's built-in [date()](http://nz.php.net/manual/en/function.date.php).
|
not PHP's built-in [date()](http://nz.php.net/manual/en/function.date.php).
|
||||||
|
|
||||||
### Language Names
|
### Language Names
|
||||||
@ -272,7 +272,7 @@ There are a few special cases:
|
|||||||
## Language definitions
|
## Language definitions
|
||||||
|
|
||||||
Each module can have one language table per locale, stored by convention in the `lang/` subfolder.
|
Each module can have one language table per locale, stored by convention in the `lang/` subfolder.
|
||||||
The translation is powered by [Zend_Translate](http://framework.zend.com/manual/en/zend.translate.html),
|
The translation is powered by [Zend_Translate](http://framework.zend.com/manual/current/en/modules/zend.i18n.translating.html),
|
||||||
which supports different translation adapters, dealing with different storage formats.
|
which supports different translation adapters, dealing with different storage formats.
|
||||||
|
|
||||||
By default, SilverStripe 3.x uses a YAML format (through the [Zend_Translate_RailsYAML adapter](https://github.com/chillu/zend_translate_railsyaml)).
|
By default, SilverStripe 3.x uses a YAML format (through the [Zend_Translate_RailsYAML adapter](https://github.com/chillu/zend_translate_railsyaml)).
|
||||||
@ -411,4 +411,4 @@ The `ss.i18n` object contain a couple functions to help and replace dynamic vari
|
|||||||
* [Help to translate](../../contributing/translations) - Instructions for online collaboration to translate core
|
* [Help to translate](../../contributing/translations) - Instructions for online collaboration to translate core
|
||||||
* [Help to translate](../../contributing/translation_process) - Instructions for adding translation to your own modules
|
* [Help to translate](../../contributing/translation_process) - Instructions for adding translation to your own modules
|
||||||
* [http://www.i18nguy.com/](http://www.i18nguy.com/)
|
* [http://www.i18nguy.com/](http://www.i18nguy.com/)
|
||||||
* [balbus.tk i18n notes](http://www.balbus.tk/internationalize)
|
* [balbus.tk i18n notes](http://www.balbuss.com/internationalize/)
|
||||||
|
@ -37,4 +37,4 @@ You may also notice the 'Sync files' button (highlighted below). This button all
|
|||||||
|
|
||||||
## Upload
|
## Upload
|
||||||
|
|
||||||
Files can be managed through a `FileField` or an `UploadField`. The `[api:FileField]` class provides a simple HTML input with a type of "file", whereas an `[api:UploadField]` provides a much more feature-rich field (including AJAX-based uploads, previews, relationship management and file data management). See [`Reference - UploadField`](/developer_guides/forms/field_types/uploadfield) for more information about how to use the `UploadField` class.
|
Files can be managed through a `FileField` or an `UploadField`. The [api:FileField] class provides a simple HTML input with a type of "file", whereas an [api:UploadField] provides a much more feature-rich field (including AJAX-based uploads, previews, relationship management and file data management). See [`Reference - UploadField`](/developer_guides/forms/field_types/uploadfield) for more information about how to use the `UploadField` class.
|
||||||
|
@ -2,14 +2,14 @@ summary: Learn how to crop and resize images in templates and PHP code
|
|||||||
|
|
||||||
# Image
|
# Image
|
||||||
|
|
||||||
Represents an image object through the `[api:Image]` class, inheriting all base functionality from the `[api:File]` class with extra functionality including resizing.
|
Represents an image object through the [api:Image] class, inheriting all base functionality from the [api:File] class with extra functionality including resizing.
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
### Managing images through form fields
|
### Managing images through form fields
|
||||||
|
|
||||||
Images can be uploaded like any other file, through `[api:FileField]`.
|
Images can be uploaded like any other file, through [api:FileField].
|
||||||
More advanced usage is possible through `[api:UploadField]`,
|
More advanced usage is possible through [api:UploadField],
|
||||||
which provides thumbnails, a detail view of the image properties,
|
which provides thumbnails, a detail view of the image properties,
|
||||||
and management of relationships to other DataObject instances.
|
and management of relationships to other DataObject instances.
|
||||||
Allows upload of images through limiting file extensions with `setAllowedExtensions()`.
|
Allows upload of images through limiting file extensions with `setAllowedExtensions()`.
|
||||||
@ -70,7 +70,7 @@ The image manipulation functions can be used in your code with the same names, e
|
|||||||
|
|
||||||
Some of the MetaData functions need to be prefixed with 'get', example `getHeight()`, `getOrientation()` etc.
|
Some of the MetaData functions need to be prefixed with 'get', example `getHeight()`, `getOrientation()` etc.
|
||||||
|
|
||||||
Please refer to the `[api:Image]` API documentation for specific functions.
|
Please refer to the [api:Image] API documentation for specific functions.
|
||||||
|
|
||||||
### Creating custom image functions
|
### Creating custom image functions
|
||||||
|
|
||||||
@ -117,8 +117,8 @@ You can also create your own functions by extending the image class, for example
|
|||||||
|
|
||||||
### Form Upload
|
### Form Upload
|
||||||
|
|
||||||
For usage on a website form, see `[api:FileField]`.
|
For usage on a website form, see [api:FileField].
|
||||||
If you want to upload images within the CMS, see `[api:UploadField]`.
|
If you want to upload images within the CMS, see [api:UploadField].
|
||||||
|
|
||||||
### Image Quality
|
### Image Quality
|
||||||
|
|
||||||
@ -160,4 +160,4 @@ disappeared, you can try manually flushing the image cache.
|
|||||||
http://localhost/dev/tasks/FlushGeneratedImagesTask
|
http://localhost/dev/tasks/FlushGeneratedImagesTask
|
||||||
|
|
||||||
## API Documentation
|
## API Documentation
|
||||||
`[api:Image]`
|
[api:Image]
|
||||||
|
@ -124,7 +124,7 @@ searched results. Every [api:DataObject] can have its own context, based on the
|
|||||||
class makes a guess at how those fields should be searched, e.g. showing a checkbox for any boolean fields in your
|
class makes a guess at how those fields should be searched, e.g. showing a checkbox for any boolean fields in your
|
||||||
`$db` definition.
|
`$db` definition.
|
||||||
|
|
||||||
To remove, add or modify searchable fields, define a new `[api:DataObject::$searchable_fields]` static on your model
|
To remove, add or modify searchable fields, define a new [api:DataObject::$searchable_fields] static on your model
|
||||||
class (see [SearchContext](../search/searchcontext) docs for details).
|
class (see [SearchContext](../search/searchcontext) docs for details).
|
||||||
|
|
||||||
**mysite/code/Product.php**
|
**mysite/code/Product.php**
|
||||||
@ -167,7 +167,7 @@ model class, where you can add or remove columns. To change the title, use [api:
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
The results list are retrieved from [api:SearchContext->getResults], based on the parameters passed through the search
|
The results list are retrieved from [api:SearchContext::getResults()], based on the parameters passed through the search
|
||||||
form. If no search parameters are given, the results will show every record. Results are a [api:DataList] instance, so
|
form. If no search parameters are given, the results will show every record. Results are a [api:DataList] instance, so
|
||||||
can be customized by additional SQL filters, joins.
|
can be customized by additional SQL filters, joins.
|
||||||
|
|
||||||
@ -225,7 +225,7 @@ checkbox which limits search results to expensive products (over $100).
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
To alter how the results are displayed (via `[api:GridField]`), you can also overload the `getEditForm()` method. For
|
To alter how the results are displayed (via [api:GridField]), you can also overload the `getEditForm()` method. For
|
||||||
example, to add a new component.
|
example, to add a new component.
|
||||||
|
|
||||||
**mysite/code/MyAdmin.php**
|
**mysite/code/MyAdmin.php**
|
||||||
|
@ -3,8 +3,8 @@
|
|||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
A lot can be achieved in SilverStripe by adding properties and form fields
|
A lot can be achieved in SilverStripe by adding properties and form fields
|
||||||
to your own page types (via `[api:SiteTree->getCMSFields()]`), as well as creating
|
to your own page types (via [api:SiteTree::getCMSFields()]), as well as creating
|
||||||
your own data management interfaces through `[api:ModelAdmin]`. But sometimes
|
your own data management interfaces through [api:ModelAdmin]. But sometimes
|
||||||
you'll want to go deeper and tailor the underlying interface to your needs as well.
|
you'll want to go deeper and tailor the underlying interface to your needs as well.
|
||||||
For example, to build a personalized CMS dashboard, or content "slots" where authors
|
For example, to build a personalized CMS dashboard, or content "slots" where authors
|
||||||
can drag their content into. At its core, SilverStripe is a web application
|
can drag their content into. At its core, SilverStripe is a web application
|
||||||
@ -51,14 +51,14 @@ See our [system requirements](/getting_started/server_requirements) for a list o
|
|||||||
|
|
||||||
## Templates and Controllers
|
## Templates and Controllers
|
||||||
|
|
||||||
The CMS backend is handled through the `[api:LeftAndMain]` controller class,
|
The CMS backend is handled through the [api:LeftAndMain] controller class,
|
||||||
which contains base functionality like displaying and saving a record.
|
which contains base functionality like displaying and saving a record.
|
||||||
This is extended through various subclasses, e.g. to add a group hierarchy (`[api:SecurityAdmin]`),
|
This is extended through various subclasses, e.g. to add a group hierarchy ([api:SecurityAdmin]),
|
||||||
a search interface (`[api:ModelAdmin]`) or an "Add Page" form (`[api:CMSPageAddController]`).
|
a search interface ([api:ModelAdmin]) or an "Add Page" form ([api:CMSPageAddController]).
|
||||||
|
|
||||||
The controller structure is too complex to document here, a good starting point
|
The controller structure is too complex to document here, a good starting point
|
||||||
for following the execution path in code are `[api:LeftAndMain->getRecord()]` and `[api:LeftAndMain->getEditForm()]`.
|
for following the execution path in code are [api:LeftAndMain::getRecord()] and [api:LeftAndMain::getEditForm()].
|
||||||
If you have the `cms` module installed, have a look at `[api:CMSMain->getEditForm()]` for a good
|
If you have the `cms` module installed, have a look at [api:CMSMain::getEditForm()] for a good
|
||||||
example on how to extend the base functionality (e.g. by adding page versioning hints to the form).
|
example on how to extend the base functionality (e.g. by adding page versioning hints to the form).
|
||||||
|
|
||||||
CMS templates are inherited based on their controllers, similar to subclasses of
|
CMS templates are inherited based on their controllers, similar to subclasses of
|
||||||
@ -72,13 +72,13 @@ which is in charge of rendering the main content area apart from the CMS menu.
|
|||||||
Depending on the complexity of your layout, you'll also need to overload the
|
Depending on the complexity of your layout, you'll also need to overload the
|
||||||
"EditForm" template (e.g. `MyCMSController_EditForm.ss`), e.g. to implement
|
"EditForm" template (e.g. `MyCMSController_EditForm.ss`), e.g. to implement
|
||||||
a tabbed form which only scrolls the main tab areas, while keeping the buttons at the bottom of the frame.
|
a tabbed form which only scrolls the main tab areas, while keeping the buttons at the bottom of the frame.
|
||||||
This requires manual assignment of the template to your form instance, see `[api:CMSMain->getEditForm()]` for details.
|
This requires manual assignment of the template to your form instance, see [api:CMSMain::getEditForm()] for details.
|
||||||
|
|
||||||
Often its useful to have a "tools" panel in between the menu and your content,
|
Often its useful to have a "tools" panel in between the menu and your content,
|
||||||
usually occupied by a search form or navigational helper.
|
usually occupied by a search form or navigational helper.
|
||||||
In this case, you can either overload the full base template as described above.
|
In this case, you can either overload the full base template as described above.
|
||||||
To avoid duplicating all this template code, you can also use the special `[api:LeftAndMain->Tools()]` and
|
To avoid duplicating all this template code, you can also use the special [api:LeftAndMain::Tools()] and
|
||||||
`[api:LeftAndMain->EditFormTools()]` methods available in `LeftAndMain`.
|
[api:LeftAndMain::EditFormTools()] methods available in `LeftAndMain`.
|
||||||
These placeholders are populated by auto-detected templates,
|
These placeholders are populated by auto-detected templates,
|
||||||
with the naming convention of "<controller classname>_Tools.ss" and "<controller classname>_EditFormTools.ss".
|
with the naming convention of "<controller classname>_Tools.ss" and "<controller classname>_EditFormTools.ss".
|
||||||
So to add or "subclass" a tools panel, simply create this file and it's automatically picked up.
|
So to add or "subclass" a tools panel, simply create this file and it's automatically picked up.
|
||||||
@ -94,7 +94,7 @@ Refer to [Layout reference](/developer_guides/customising_the_admin_interface/cm
|
|||||||
## Forms
|
## Forms
|
||||||
|
|
||||||
SilverStripe constructs forms and its fields within PHP,
|
SilverStripe constructs forms and its fields within PHP,
|
||||||
mainly through the `[getCMSFields()](api:DataObject->getCMSFields())` method.
|
mainly through the [getCMSFields()](api:DataObject::getCMSFields()) method.
|
||||||
This in turn means that the CMS loads these forms as HTML via Ajax calls,
|
This in turn means that the CMS loads these forms as HTML via Ajax calls,
|
||||||
e.g. after saving a record (which requires a form refresh), or switching the section in the CMS.
|
e.g. after saving a record (which requires a form refresh), or switching the section in the CMS.
|
||||||
|
|
||||||
@ -165,10 +165,10 @@ any particular element.
|
|||||||
## JavaScript and CSS dependencies via Requirements and Ajax
|
## JavaScript and CSS dependencies via Requirements and Ajax
|
||||||
|
|
||||||
The JavaScript logic powering the CMS is divided into many files,
|
The JavaScript logic powering the CMS is divided into many files,
|
||||||
which typically are included via the `[api:Requirements]` class, by adding
|
which typically are included via the [api:Requirements] class, by adding
|
||||||
them to `[api:LeftAndMain->init()]` and its subclassed methods.
|
them to [api:LeftAndMain::init()] and its subclassed methods.
|
||||||
This class also takes care of minification and combination of the files,
|
This class also takes care of minification and combination of the files,
|
||||||
which is crucial for the CMS performance (see `[api:Requirements::combine_files()]`).
|
which is crucial for the CMS performance (see [api:Requirements::combine_files()]).
|
||||||
|
|
||||||
Due to the procedural and selector-driven style of UI programming in jQuery.entwine,
|
Due to the procedural and selector-driven style of UI programming in jQuery.entwine,
|
||||||
it can be difficult to find the piece of code responsible for a certain behaviour.
|
it can be difficult to find the piece of code responsible for a certain behaviour.
|
||||||
@ -188,7 +188,7 @@ and [jQuery.delegate](http://api.jquery.com/delegate/), so takes care of dynamic
|
|||||||
Most interfaces will require their own JavaScript and CSS files, so the Ajax loading has
|
Most interfaces will require their own JavaScript and CSS files, so the Ajax loading has
|
||||||
to ensure they're loaded unless already present. A custom-built library called
|
to ensure they're loaded unless already present. A custom-built library called
|
||||||
`jQuery.ondemand` (located in `framework/thirdparty`) takes care of this transparently -
|
`jQuery.ondemand` (located in `framework/thirdparty`) takes care of this transparently -
|
||||||
so as a developer just declare your dependencies through the `[api:Requirements]` API.
|
so as a developer just declare your dependencies through the [api:Requirements] API.
|
||||||
|
|
||||||
## Ajax Loading and Browser History
|
## Ajax Loading and Browser History
|
||||||
|
|
||||||
@ -220,10 +220,10 @@ we often want to update these sections independently from their neighbouring con
|
|||||||
|
|
||||||
In order for this to work, the CMS templates declare certain sections as "PJAX fragments"
|
In order for this to work, the CMS templates declare certain sections as "PJAX fragments"
|
||||||
through a `data-pjax-fragment` attribute. These names correlate to specific
|
through a `data-pjax-fragment` attribute. These names correlate to specific
|
||||||
rendering logic in the PHP controllers, through the `[api:PjaxResponseNegotiator]` class.
|
rendering logic in the PHP controllers, through the [api:PjaxResponseNegotiator] class.
|
||||||
|
|
||||||
Through a custom `X-Pjax` HTTP header, the client can declare which view they're expecting,
|
Through a custom `X-Pjax` HTTP header, the client can declare which view they're expecting,
|
||||||
through identifiers like `CurrentForm` or `Content` (see `[api:LeftAndMain->getResponseNegotiator()]`).
|
through identifiers like `CurrentForm` or `Content` (see [api:LeftAndMain::getResponseNegotiator()]).
|
||||||
These identifiers are passed to `loadPanel()` via the `pjax` data option.
|
These identifiers are passed to `loadPanel()` via the `pjax` data option.
|
||||||
The HTTP response is a JSON object literal, with template replacements keyed by their Pjax fragment.
|
The HTTP response is a JSON object literal, with template replacements keyed by their Pjax fragment.
|
||||||
Through PHP callbacks, we ensure that only the required template parts are actually executed and rendered.
|
Through PHP callbacks, we ensure that only the required template parts are actually executed and rendered.
|
||||||
@ -402,11 +402,11 @@ when using an input of type button, submit or reset, support is limited to plain
|
|||||||
|
|
||||||
## Menu
|
## Menu
|
||||||
|
|
||||||
The navigation menu in the CMS is created through the `[api:CMSMenu]` API,
|
The navigation menu in the CMS is created through the [api:CMSMenu] API,
|
||||||
which auto-detects all subclasses of `LeftAndMain`. This means that your custom
|
which auto-detects all subclasses of `LeftAndMain`. This means that your custom
|
||||||
`ModelAdmin` subclasses will already appear in there without any explicit definition.
|
`ModelAdmin` subclasses will already appear in there without any explicit definition.
|
||||||
To modify existing menu entries or create new ones, see `[api:CMSMenu::add_menu_item()]`
|
To modify existing menu entries or create new ones, see [api:CMSMenu::add_menu_item()]
|
||||||
and `[api:CMSMenu::remove_menu_item()]`.
|
and [api:CMSMenu::remove_menu_item()].
|
||||||
|
|
||||||
New content panels are typically loaded via Ajax, which might change
|
New content panels are typically loaded via Ajax, which might change
|
||||||
the current menu context. For example, a link to edit a file might be clicked
|
the current menu context. For example, a link to edit a file might be clicked
|
||||||
@ -422,7 +422,7 @@ which is picked up by the menu:
|
|||||||
return 'my response';
|
return 'my response';
|
||||||
}
|
}
|
||||||
|
|
||||||
This is usually handled by the existing `[api:LeftAndMain]` logic,
|
This is usually handled by the existing [api:LeftAndMain] logic,
|
||||||
so you don't need to worry about it. The same concept applies for
|
so you don't need to worry about it. The same concept applies for
|
||||||
'X-Title' (change the window title) and 'X-ControllerURL' (change the URL recorded in browser history).
|
'X-Title' (change the window title) and 'X-ControllerURL' (change the URL recorded in browser history).
|
||||||
Note: You can see any additional HTTP headers through the web developer tools in your browser of choice.
|
Note: You can see any additional HTTP headers through the web developer tools in your browser of choice.
|
||||||
@ -437,13 +437,13 @@ For more information, see the [Howto: Customise the CMS tree](/developer_guides/
|
|||||||
|
|
||||||
Note that a similar tree logic is also used for the
|
Note that a similar tree logic is also used for the
|
||||||
form fields to select one or more entries from those hierarchies
|
form fields to select one or more entries from those hierarchies
|
||||||
(`[api:TreeDropdownField]` and `[api:TreeMultiselectField]`).
|
([api:TreeDropdownField] and [api:TreeMultiselectField]).
|
||||||
|
|
||||||
## Tabs
|
## Tabs
|
||||||
|
|
||||||
We're using [jQuery UI tabs](http://jqueryui.com/), but in a customised fashion.
|
We're using [jQuery UI tabs](http://jqueryui.com/), but in a customised fashion.
|
||||||
HTML with tabs can be created either directly through HTML templates in the CMS,
|
HTML with tabs can be created either directly through HTML templates in the CMS,
|
||||||
or indirectly through a `[api:TabSet]` form field. Since tabsets are useable
|
or indirectly through a [api:TabSet] form field. Since tabsets are useable
|
||||||
outside of the CMS as well, the baseline application of tabs happens via
|
outside of the CMS as well, the baseline application of tabs happens via
|
||||||
a small wrapper around `jQuery.tabs()` stored in `TabSet.js`.
|
a small wrapper around `jQuery.tabs()` stored in `TabSet.js`.
|
||||||
|
|
@ -349,7 +349,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
|
||||||
@ -420,7 +420,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.
|
||||||
|
|
||||||
|
@ -2,10 +2,10 @@
|
|||||||
|
|
||||||
## Adding an administration panel
|
## Adding an administration panel
|
||||||
|
|
||||||
Every time you add a new extension of the `[api:LeftAndMain]` class to the CMS,
|
Every time you add a new extension of the [api:LeftAndMain] class to the CMS,
|
||||||
SilverStripe will automatically create a new `[api:CMSMenuItem]` for it
|
SilverStripe will automatically create a new [api:CMSMenuItem] for it
|
||||||
|
|
||||||
The most popular extension of LeftAndMain is a `[api:ModelAdmin]` class, so
|
The most popular extension of LeftAndMain is a [api:ModelAdmin] class, so
|
||||||
for a more detailed introduction to creating new `ModelAdmin` interfaces, read
|
for a more detailed introduction to creating new `ModelAdmin` interfaces, read
|
||||||
the [ModelAdmin reference](../modeladmin).
|
the [ModelAdmin reference](../modeladmin).
|
||||||
|
|
||||||
@ -51,7 +51,7 @@ On top of your administration windows, the menu can also have external links
|
|||||||
(e.g. to external reference). In this example, we're going to add a link to
|
(e.g. to external reference). In this example, we're going to add a link to
|
||||||
Google to the menu.
|
Google to the menu.
|
||||||
|
|
||||||
First, we need to define a `[api:LeftAndMainExtension]` which will contain our
|
First, we need to define a [api:LeftAndMainExtension] which will contain our
|
||||||
button configuration.
|
button configuration.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
|
@ -10,8 +10,8 @@ as well as sort and filter them in a way that would be hard to achieve in a tree
|
|||||||
But sometimes the default behaviour isn't powerful enough, and you want a more
|
But sometimes the default behaviour isn't powerful enough, and you want a more
|
||||||
specific list view for certain page types, for example to sort the list by
|
specific list view for certain page types, for example to sort the list by
|
||||||
a different criteria, or add more columns to filter on. The resulting
|
a different criteria, or add more columns to filter on. The resulting
|
||||||
form is mainly based around a `[GridField](/reference/grid-field)` instance,
|
form is mainly based around a [GridField](/reference/grid-field) instance,
|
||||||
which in turn includes all children in a `[DataList](/topics/datamodel)`.
|
which in turn includes all children in a [DataList](/topics/datamodel).
|
||||||
You can use these two classes as a starting point for your customizations.
|
You can use these two classes as a starting point for your customizations.
|
||||||
|
|
||||||
Here's a brief example on how to add sorting and a new column for a
|
Here's a brief example on how to add sorting and a new column for a
|
||||||
|
@ -7,10 +7,10 @@ by the [jstree](http://jstree.com) library. It is configured through
|
|||||||
`framework/admin/javascript/LeftAndMain.Tree.js`, as well as some
|
`framework/admin/javascript/LeftAndMain.Tree.js`, as well as some
|
||||||
HTML5 metadata generated on its container (see the `data-hints` attribute).
|
HTML5 metadata generated on its container (see the `data-hints` attribute).
|
||||||
|
|
||||||
The tree is rendered through `[api:LeftAndMain->getSiteTreeFor()]`,
|
The tree is rendered through [api:LeftAndMain::getSiteTreeFor()],
|
||||||
which recursively collects all nodes based on various filtering criteria.
|
which recursively collects all nodes based on various filtering criteria.
|
||||||
The node strictly just has to implement the `[api:Hierarchy]` extension,
|
The node strictly just has to implement the [api:Hierarchy] extension,
|
||||||
but in the CMS usually is a `[api:SiteTree]` object.
|
but in the CMS usually is a [api:SiteTree] object.
|
||||||
|
|
||||||
## Add status lozenges to tree nodes
|
## Add status lozenges to tree nodes
|
||||||
|
|
||||||
@ -43,13 +43,13 @@ code like this:
|
|||||||
By applying the proper style sheet, the snippet html above could produce the look of:
|
By applying the proper style sheet, the snippet html above could produce the look of:
|
||||||
![Page Node Screenshot](../../../_images/tree_node.png "Page Node")
|
![Page Node Screenshot](../../../_images/tree_node.png "Page Node")
|
||||||
|
|
||||||
SiteTree is a `[api:DataObject]` which is versioned by `[api:Versioned]` extension.
|
SiteTree is a [api:DataObject] which is versioned by [api:Versioned] extension.
|
||||||
Each node can optionally have publication status flags, e.g. "Removed from draft".
|
Each node can optionally have publication status flags, e.g. "Removed from draft".
|
||||||
Each flag has a unique identifier, which is also used as a CSS class for easier styling.
|
Each flag has a unique identifier, which is also used as a CSS class for easier styling.
|
||||||
|
|
||||||
Developers can easily add a new flag, delete or alter an existing flag on how it is looked
|
Developers can easily add a new flag, delete or alter an existing flag on how it is looked
|
||||||
or changing the flag label. The customization of these lozenges could be done either through
|
or changing the flag label. The customization of these lozenges could be done either through
|
||||||
inherited subclass or `[api:DataExtension]`. It is just really about how we change the return
|
inherited subclass or [api:DataExtension]. It is just really about how we change the return
|
||||||
value of function `SiteTree->getTreeTitle()` by two easily extendable methods
|
value of function `SiteTree->getTreeTitle()` by two easily extendable methods
|
||||||
`SiteTree->getStatusClass()` and `SiteTree->getStatusFlags()`.
|
`SiteTree->getStatusClass()` and `SiteTree->getStatusFlags()`.
|
||||||
|
|
||||||
@ -74,7 +74,7 @@ __Example: using a subclass__
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
The above subclass of `[api:SiteTree]` will add a new flag for indicating its
|
The above subclass of [api:SiteTree] will add a new flag for indicating its
|
||||||
__'Scheduled To Publish'__ status. The look of the page node will be changed
|
__'Scheduled To Publish'__ status. The look of the page node will be changed
|
||||||
from ![Normal Page Node](../../../_images/page_node_normal.png) to ![Scheduled Page Node](../../../_images/page_node_scheduled.png). The getStatusFlags has an `updateStatusFlags()`
|
from ![Normal Page Node](../../../_images/page_node_normal.png) to ![Scheduled Page Node](../../../_images/page_node_scheduled.png). The getStatusFlags has an `updateStatusFlags()`
|
||||||
extension point, so the flags can be modified through `DataExtension` rather than
|
extension point, so the flags can be modified through `DataExtension` rather than
|
||||||
|
@ -71,4 +71,4 @@ More useful reports can be created by changing the `DataList` returned in the `s
|
|||||||
* More examples
|
* More examples
|
||||||
|
|
||||||
## API documentation
|
## API documentation
|
||||||
`[api:ReportAdmin]`
|
[api:ReportAdmin]
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
## Extending existing ModelAdmin
|
## Extending existing ModelAdmin
|
||||||
|
|
||||||
Sometimes you'll work with ModelAdmins from other modules. To customize these interfaces, you can always subclass. But there's
|
Sometimes you'll work with ModelAdmins from other modules. To customize these interfaces, you can always subclass. But there's
|
||||||
also another tool at your disposal: The `[api:Extension]` API.
|
also another tool at your disposal: The [api:Extension] API.
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
class MyAdminExtension extends Extension {
|
class MyAdminExtension extends Extension {
|
||||||
|
@ -6,17 +6,17 @@ summary: Allows a class to define it's own flush functionality.
|
|||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
Allows a class to define it's own flush functionality, which is triggered when `flush=1` is requested in the URL.
|
Allows a class to define it's own flush functionality, which is triggered when `flush=1` is requested in the URL.
|
||||||
`[api:FlushRequestFilter]` is run before a request is made, calling `flush()` statically on all
|
[api:FlushRequestFilter] is run before a request is made, calling `flush()` statically on all
|
||||||
implementors of `[api:Flushable]`.
|
implementors of [api:Flushable].
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
To use this API, you need to make your class implement `[api:Flushable]`, and define a `flush()` static function,
|
To use this API, you need to make your class implement [api:Flushable], and define a `flush()` static function,
|
||||||
this defines the actions that need to be executed on a flush request.
|
this defines the actions that need to be executed on a flush request.
|
||||||
|
|
||||||
### Using with SS_Cache
|
### Using with SS_Cache
|
||||||
|
|
||||||
This example uses `[api:SS_Cache]` in some custom code, and the same cache is cleaned on flush:
|
This example uses [api:SS_Cache] in some custom code, and the same cache is cleaned on flush:
|
||||||
|
|
||||||
:::php
|
:::php
|
||||||
<?php
|
<?php
|
||||||
|
@ -26,15 +26,15 @@ You can write your own adapters by implementing the `ManifestCache` interface.
|
|||||||
## Traversing the Filesystem
|
## Traversing the Filesystem
|
||||||
|
|
||||||
Since manifests usually extract their information from files in the webroot,
|
Since manifests usually extract their information from files in the webroot,
|
||||||
they require a powerful traversal tool: `[api:SS_FileFinder]`.
|
they require a powerful traversal tool: [api:SS_FileFinder].
|
||||||
The class provides filtering abilities for files and folders, as well as
|
The class provides filtering abilities for files and folders, as well as
|
||||||
callbacks for recursive traversal. Each manifest has its own implementation,
|
callbacks for recursive traversal. Each manifest has its own implementation,
|
||||||
for example `[api:ManifestFileFinder]`, adding more domain specific filtering
|
for example [api:ManifestFileFinder], adding more domain specific filtering
|
||||||
like including themes, or excluding testss.
|
like including themes, or excluding testss.
|
||||||
|
|
||||||
## PHP Class Manifest
|
## PHP Class Manifest
|
||||||
|
|
||||||
The `[api:ClassManifest]` builds a manifest of all classes, interfaces and some
|
The [api:ClassManifest] builds a manifest of all classes, interfaces and some
|
||||||
additional items present in a directory, and caches it.
|
additional items present in a directory, and caches it.
|
||||||
|
|
||||||
It finds the following information:
|
It finds the following information:
|
||||||
@ -44,15 +44,15 @@ It finds the following information:
|
|||||||
* All implementors of an interface
|
* All implementors of an interface
|
||||||
* All module configuration files
|
* All module configuration files
|
||||||
|
|
||||||
The gathered information can be accessed through `[api:SS_ClassLoader::instance()]`,
|
The gathered information can be accessed through [api:SS_ClassLoader::instance()],
|
||||||
as well as `[api:ClassInfo]`. Some useful commands of the `ClassInfo` API:
|
as well as [api:ClassInfo]. Some useful commands of the `ClassInfo` API:
|
||||||
|
|
||||||
* `ClassInfo::subclassesFor($class)`: Returns a list of classes that inherit from the given class
|
* `ClassInfo::subclassesFor($class)`: Returns a list of classes that inherit from the given class
|
||||||
* `ClassInfo::ancestry($class)`: Returns the passed class name along with all its parent class names
|
* `ClassInfo::ancestry($class)`: Returns the passed class name along with all its parent class names
|
||||||
* `ClassInfo::implementorsOf($interfaceName)`: Returns all classes implementing the passed in interface
|
* `ClassInfo::implementorsOf($interfaceName)`: Returns all classes implementing the passed in interface
|
||||||
|
|
||||||
In the absence of a generic module API, it is also the primary way to identify
|
In the absence of a generic module API, it is also the primary way to identify
|
||||||
which modules are installed, through `[api:ClassManifest::getModules()]`.
|
which modules are installed, through [api:ClassManifest::getModules()].
|
||||||
A module is defined as a toplevel folder in the webroot which contains
|
A module is defined as a toplevel folder in the webroot which contains
|
||||||
either a `_config.php` file, or a `_config/` folder. Modules can be specifically
|
either a `_config.php` file, or a `_config/` folder. Modules can be specifically
|
||||||
excluded from manifests by creating a blank `_manifest_exclude` file in the module folder.
|
excluded from manifests by creating a blank `_manifest_exclude` file in the module folder.
|
||||||
@ -62,14 +62,14 @@ a `tests/` folder, unless tests are executed.
|
|||||||
|
|
||||||
## Template Manifest
|
## Template Manifest
|
||||||
|
|
||||||
The `[api:SS_TemplateManifest]` class builds a manifest of all templates present in a directory,
|
The [api:SS_TemplateManifest] class builds a manifest of all templates present in a directory,
|
||||||
in both modules and themes. Templates in `tests/` folders are automatically excluded.
|
in both modules and themes. Templates in `tests/` folders are automatically excluded.
|
||||||
The chapter on [template inheritance](../templates/template_inheritance) provides more details
|
The chapter on [template inheritance](../templates/template_inheritance) provides more details
|
||||||
on its operation.
|
on its operation.
|
||||||
|
|
||||||
## Config Manifest
|
## Config Manifest
|
||||||
|
|
||||||
The `[api:SS_ConfigManifest]` loads builds a manifest of configuration items,
|
The [api:SS_ConfigManifest] loads builds a manifest of configuration items,
|
||||||
for both PHP and YAML. It also takes care of ordering and merging configuration fragments.
|
for both PHP and YAML. It also takes care of ordering and merging configuration fragments.
|
||||||
The chapter on [configuration](../configuration) has more details.
|
The chapter on [configuration](../configuration) has more details.
|
||||||
|
|
||||||
|
@ -93,7 +93,7 @@ All requests go through `framework/main.php`, which sets up the execution enviro
|
|||||||
* Optionally regenerates these manifests (if a ["flush" query parameter](flushable) is set)
|
* Optionally regenerates these manifests (if a ["flush" query parameter](flushable) is set)
|
||||||
* Executes all procedural configuration defined through `_config.php` in all discovered modules
|
* Executes all procedural configuration defined through `_config.php` in all discovered modules
|
||||||
* Loads the Composer PHP class autoloader
|
* Loads the Composer PHP class autoloader
|
||||||
* Hands control over to `[api:Director]`
|
* Hands control over to [api:Director]
|
||||||
|
|
||||||
While you usually don't need to modify the bootstrap on this level, some deeper customizations like
|
While you usually don't need to modify the bootstrap on this level, some deeper customizations like
|
||||||
adding your own manifests or a performance-optimized routing might require it.
|
adding your own manifests or a performance-optimized routing might require it.
|
||||||
@ -103,19 +103,19 @@ before handing control off to SilverStripe's own `main.php`.
|
|||||||
|
|
||||||
## Routing and Request Handling
|
## Routing and Request Handling
|
||||||
|
|
||||||
The `main.php` script relies on `[api:Director]` to work out which [controller](../controllers/) should handle this request. It parses the URL, matching it to one of a number of patterns,
|
The `main.php` script relies on [api:Director] to work out which [controller](../controllers/) should handle this request. It parses the URL, matching it to one of a number of patterns,
|
||||||
and determines the controller, action and any argument to be used ([Routing](../controllers/routing)).
|
and determines the controller, action and any argument to be used ([Routing](../controllers/routing)).
|
||||||
|
|
||||||
* Creates a `[api:SS_HTTPRequest]` object containing all request and environment information
|
* Creates a [api:SS_HTTPRequest] object containing all request and environment information
|
||||||
* The [session](../cookies_and_sessions/sessions) holds an abstraction of PHP session
|
* The [session](../cookies_and_sessions/sessions) holds an abstraction of PHP session
|
||||||
* Instantiates a [controller](../controllers/) object
|
* Instantiates a [controller](../controllers/) object
|
||||||
* The `[api:Injector]` is first referenced, and asks the registered
|
* The [api:Injector] is first referenced, and asks the registered
|
||||||
[RequestFilter](../controllers/requestfilters)
|
[RequestFilter](../controllers/requestfilters)
|
||||||
to pre-process the request object (see below)
|
to pre-process the request object (see below)
|
||||||
* The `Controller` executes the actual business logic and populates an `[api:SS_HTTPResponse]`
|
* The `Controller` executes the actual business logic and populates an [api:SS_HTTPResponse]
|
||||||
* The `Controller` can optionally hand off control to further nested controllers
|
* The `Controller` can optionally hand off control to further nested controllers
|
||||||
* The `Controller` optionally renders a response body through `SSViewer` [templates](../templates)
|
* The `Controller` optionally renders a response body through `SSViewer` [templates](../templates)
|
||||||
* The `[api:RequestProcessor]` is called to post-process the request to allow
|
* The [api:RequestProcessor] is called to post-process the request to allow
|
||||||
further filtering before content is sent to the end user
|
further filtering before content is sent to the end user
|
||||||
* The response is output to the client
|
* The response is output to the client
|
||||||
|
|
||||||
|
40
docs/en/04_Changelogs/3.2.2.md
Normal file
40
docs/en/04_Changelogs/3.2.2.md
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
# 3.2.2
|
||||||
|
|
||||||
|
<!--- Changes below this line will be automatically regenerated -->
|
||||||
|
|
||||||
|
## Change Log
|
||||||
|
|
||||||
|
### Security
|
||||||
|
|
||||||
|
* 2016-02-17 [faa94d5](https://github.com/silverstripe/silverstripe-framework/commit/faa94d51d570788dcebc2f2ef6e9de4d179ce1e4) Hostname, IP and Protocol Spoofing through HTTP Headers (Ingo Schommer) - See [ss-2016-003](http://www.silverstripe.org/download/security-releases/ss-2016-003)
|
||||||
|
* 2016-02-17 [15d4db3](https://github.com/silverstripe/silverstripe-framework/commit/15d4db3b4a7dbc9a7e089f9329a396f8408ed7d9) Block unauthenticated access to dev/build/defaults (Damian Mooyman) - See [ss-2015-028](http://www.silverstripe.org/download/security-releases/ss-2015-028)
|
||||||
|
* 2016-02-17 [e2c77c5](https://github.com/silverstripe/silverstripe-framework/commit/e2c77c5a8f13e901c51a3684210811559b592f0c) Ensure Gridfield actions respect CSRF (Damian Mooyman) - See [ss-2016-002](http://www.silverstripe.org/download/security-releases/ss-2016-002)
|
||||||
|
* 2015-11-11 [245e0aa](https://github.com/silverstripe/silverstripe-framework/commit/245e0aae2f5f3eb0acba1d198ad8e196bb224462) Fix FormField error messages not being encoded safely (Damian Mooyman) - See [ss-2015-026](http://www.silverstripe.org/download/security-releases/ss-2015-026)
|
||||||
|
* 2015-11-09 [53b3bc7](https://github.com/silverstripe/silverstripe-framework/commit/53b3bc707bcccb8f5e5060f85ab1398a0975bba2) Dont expose class on error (Hamish Friedlander) - See [ss-2015-025](http://www.silverstripe.org/download/security-releases/ss-2015-025)
|
||||||
|
* 2015-11-01 [ac4342d](https://github.com/silverstripe/silverstripe-framework/commit/ac4342d81d19201bd8d3814f168240db1ac565fe) XML escape RSSFeed $link parameter (Ingo Schommer) - See [ss-2015-022](http://www.silverstripe.org/download/security-releases/ss-2015-022)
|
||||||
|
* 2015-10-28 [97f21fd](https://github.com/silverstripe/silverstripe-framework/commit/97f21fddb3c565052f19ee3b35366f48e1e9a36f) Fix rewrite hash links XSS (Damian Mooyman) - See [ss-2015-021](http://www.silverstripe.org/download/security-releases/ss-2015-021)
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
|
||||||
|
* 2016-02-15 [8771859](https://github.com/silverstripe/silverstripe-framework/commit/87718597e8f04872c285808d0666fbb69c5100ba) "where" method in SQLUpdate Example (Richard Rudy)
|
||||||
|
* 2016-01-28 [3fcf1e2](https://github.com/silverstripe/silverstripe-framework/commit/3fcf1e2c98629dcd0048ff9447bad4cd30b4bf95) edge case on many many extra fields (fixes 4991) (Mark Stephens)
|
||||||
|
* 2016-01-27 [3d0178e](https://github.com/silverstripe/silverstripe-cms/commit/3d0178ebc0b7408442ad2532f998ed47839e7117) Use correct formaction for doRollback exemption (Damian Mooyman)
|
||||||
|
* 2016-01-24 [d8e354d](https://github.com/silverstripe/silverstripe-framework/commit/d8e354d144383fb6459adf92731853d2e54268d6) PHPDocs on DataList::getIDList() and UnsavedRelationList::getIDList() (Damian Mooyman)
|
||||||
|
* 2016-01-22 [bf8bf5e](https://github.com/silverstripe/silverstripe-framework/commit/bf8bf5e4d558126bb99ea63881f1885faafddd3d) Prevent Versioned::doRollbackTo from creating incorrect versions on subclasses of Versioned DataObjects (Damian Mooyman)
|
||||||
|
* 2016-01-21 [cca7129](https://github.com/silverstripe/silverstripe-framework/commit/cca7129385dbb3be1001a8861423c2cf490f02d4) Revert lost documentation (Damian Mooyman)
|
||||||
|
* 2016-01-11 [85ba918](https://github.com/silverstripe/silverstripe-framework/commit/85ba918a54f51dd524d45f2c93172a18421ae3bf) Update field IDs for file link (fixes silverstripe/silverstripe-cms#1307) (Loz Calver)
|
||||||
|
* 2016-01-11 [d637141](https://github.com/silverstripe/silverstripe-cms/commit/d6371414876e32e7369ec0219a57d2186cfe3f0f) preg_quote() anchors in SiteTreeLinkTracking (fixes #1359) (Loz Calver)
|
||||||
|
* 2016-01-05 [00544ff](https://github.com/silverstripe/silverstripe-framework/commit/00544ff100048afdb7ccb1905304dddf8ab3205a) session_regenerate_id uses config system (Daniel Hensby)
|
||||||
|
* 2016-01-05 [4335d8e](https://github.com/silverstripe/silverstripe-framework/commit/4335d8ed221a2b402299b32e31f97fc2956ec161) Members with no ID inherit logged in user permission (Daniel Hensby)
|
||||||
|
* 2015-12-15 [afbb5cf](https://github.com/silverstripe/silverstripe-framework/commit/afbb5cfed4d29aea5868f0f12cd735dc5abe10d3) Vimeo oEmbed endpoint redirecting to no www (UndefinedOffset)
|
||||||
|
* 2015-12-14 [d265c9b](https://github.com/silverstripe/silverstripe-framework/commit/d265c9b733ddac27d6df286ce000b09e1c69b986) Allow omitting a value for OptionsetField submissions (fixes #4824) (Loz Calver)
|
||||||
|
* 2015-12-11 [5a21b2f](https://github.com/silverstripe/silverstripe-framework/commit/5a21b2fb15ed9c675594f0f990765bd4f97155c7) Guard against users being added to all groups on unsaved Group. (Mateusz Uzdowski)
|
||||||
|
* 2015-11-27 [94742fa](https://github.com/silverstripe/silverstripe-framework/commit/94742fa3e2efad8f77f4acd1f9d06bf74916c5e6) Revert method visibility regression (Damian Mooyman)
|
||||||
|
* 2015-11-18 [e9b833f](https://github.com/silverstripe/silverstripe-framework/commit/e9b833f5f0f989af8d611f8cfe71f0b0e2cb0159) ConfirmedPassword field correctly reports mismatching passwords (Christopher Darling)
|
||||||
|
* 2015-11-17 [68d99be](https://github.com/silverstripe/silverstripe-framework/commit/68d99be24b63a933f041cd80a248a7b7fa8d588c) Hidden errors for composite fields nested inside FieldGroups (fixes #4773) (Loz Calver)
|
||||||
|
* 2015-11-17 [97e90b8](https://github.com/silverstripe/silverstripe-cms/commit/97e90b8ebd8078bb60ecea66bdd3761380f93a61) RedirectorPage toggles not working (fixes #1328) (Loz Calver)
|
||||||
|
* 2015-11-17 [b624eb9](https://github.com/silverstripe/silverstripe-cms/commit/b624eb98f1d1ff36811a3294ad29b31a50683d60) Setting target for unwritten VirtualPage breaks write (Loz Calver)
|
||||||
|
* 2015-11-16 [2983d82](https://github.com/silverstripe/silverstripe-cms/commit/2983d823d1eef293ef11aac9e01336e23ed52b59) Ensure VirtualPage forwards request/response data to virtual controllers (fixes #1329) (Loz Calver)
|
||||||
|
* 2015-11-12 [fea1158](https://github.com/silverstripe/silverstripe-framework/commit/fea1158d193ed4d037df94101e3b3f2d24a6ce49) Fix print button only displaying first page (Damian Mooyman)
|
||||||
|
* 2015-11-11 [a40812a](https://github.com/silverstripe/silverstripe-framework/commit/a40812ac3320d27f243ef0ed54aa003fc53720b6) Don’t reuse DBConnector (fixes #4735) (Sam Minnee)
|
||||||
|
* 2015-11-05 [f577ecb](https://github.com/silverstripe/silverstripe-framework/commit/f577ecb81149d0d09dc846204f17b2153a244b5a) prevent use cache on browser back button (Igor Nadj)
|
@ -20,3 +20,64 @@ mode of the site frontend using the `?stage` querystring parameter.
|
|||||||
|
|
||||||
This permission can be customised by altering the `Versioned.non_live_permissions`
|
This permission can be customised by altering the `Versioned.non_live_permissions`
|
||||||
config by assigning a different set of permissions.
|
config by assigning a different set of permissions.
|
||||||
|
<!--- Changes below this line will be automatically regenerated -->
|
||||||
|
|
||||||
|
## Change Log
|
||||||
|
|
||||||
|
### Security
|
||||||
|
|
||||||
|
* 2016-02-17 [893e497](https://github.com/silverstripe/silverstripe-framework/commit/893e49703de4aa1855b5364919cbb0826f754fbf) Hostname, IP and Protocol Spoofing through HTTP Headers (Ingo Schommer) - See [ss-2016-003](http://www.silverstripe.org/download/security-releases/ss-2016-003)
|
||||||
|
* 2016-02-17 [3398f67](https://github.com/silverstripe/silverstripe-framework/commit/3398f670d881447f8777b567f1ead7c0d8d253f5) Block unauthenticated access to dev/build/defaults (Damian Mooyman) - See [ss-2015-028](http://www.silverstripe.org/download/security-releases/ss-2015-028)
|
||||||
|
* 2016-02-17 [56e92f5](https://github.com/silverstripe/silverstripe-framework/commit/56e92f5a32e45849cc9361c8603c31d7010c9d36) Ensure Gridfield actions respect CSRF (Damian Mooyman) - See [ss-2016-002](http://www.silverstripe.org/download/security-releases/ss-2016-002)
|
||||||
|
|
||||||
|
### API Changes
|
||||||
|
|
||||||
|
* 2015-12-07 [38e154a](https://github.com/silverstripe/silverstripe-framework/commit/38e154af0aae89a36f4d3906612ea4bbbf726177) Disable get parameter access to site stage mode (Damian Mooyman)
|
||||||
|
* 2015-12-02 [5353ac5](https://github.com/silverstripe/silverstripe-cms/commit/5353ac5315703240540c9cde0f5c8eeb5571bc19) Refactor versioned security into core module (Damian Mooyman)
|
||||||
|
* 2015-12-02 [6089a7c](https://github.com/silverstripe/silverstripe-framework/commit/6089a7c5bd25d6591deb154f1a34908fa91ac198) Create default security permission model for versioned data objects (Damian Mooyman)
|
||||||
|
* 2015-11-26 [6266f90](https://github.com/silverstripe/silverstripe-framework/commit/6266f909e0c098652582af44ea64f031ea9cdcea) Increased Permission.Code db field to 255 characters (Novusvetus)
|
||||||
|
* 2015-07-20 [ea9434f](https://github.com/silverstripe/silverstripe-framework/commit/ea9434ffeba8d5fbb1dfe38d76f3fed403a9886e) Lazy load template parser (Loz Calver)
|
||||||
|
|
||||||
|
### Features and Enhancements
|
||||||
|
|
||||||
|
* 2015-12-14 [9467ab9](https://github.com/silverstripe/silverstripe-framework/commit/9467ab9a7e717cece3cee1693b16a055b58526ef) Implement unshift() in field list classes (closes #4834) (Loz Calver)
|
||||||
|
* 2015-12-01 [f7c270a](https://github.com/silverstripe/silverstripe-framework/commit/f7c270a3bad984910fa84f552dfa8b99324afb16) Use Config for determining Vary header (Marcus Nyeholt)
|
||||||
|
* 2015-11-10 [603cacc](https://github.com/silverstripe/silverstripe-framework/commit/603caccb90006b3a0592b129687659571112b9a8) CurrencyField to use Currency.currency_symbol (muskie9)
|
||||||
|
* 2015-09-25 [5c04dc5](https://github.com/silverstripe/silverstripe-framework/commit/5c04dc5d673aa11249310bcb6e382db4ee2bff7f) - Added new method to display the number of total items in a paginated list within templates (Marco Kernler)
|
||||||
|
* 2015-08-14 [1b57e0c](https://github.com/silverstripe/silverstripe-framework/commit/1b57e0ca5bdb5d80d6f78686669441ad8b2c9420) implement getter and setter usage for response (Stevie Mayhew)
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
|
||||||
|
* 2016-02-09 [2ad490c](https://github.com/silverstripe/silverstripe-cms/commit/2ad490c3e2d256d8dcd16398631c114aa2a3370e) Prevent folders deleted on the filesystem from breaking asset interface (Damian Mooyman)
|
||||||
|
* 2016-01-22 [f80467a](https://github.com/silverstripe/silverstripe-cms/commit/f80467a74859fba58be835a878ceddbbb4601b42) Don't keep stale treeview data when refreshing Content area (Damian Mooyman)
|
||||||
|
* 2016-01-21 [e364fdb](https://github.com/silverstripe/silverstripe-cms/commit/e364fdb794896b5c6b4810d84c0dfac75d80b53b) Fix incorrect "Add Page" button selector (Damian Mooyman)
|
||||||
|
* 2016-01-20 [abc5556](https://github.com/silverstripe/silverstripe-cms/commit/abc5556520f891d0e3f5cf3d2c3838a194ac5335) Fix legacy breadcrumbs appearing on page save (Damian Mooyman)
|
||||||
|
* 2016-01-20 [df76d78](https://github.com/silverstripe/silverstripe-framework/commit/df76d783fe1f7baaeed67a7c6d63235facd364cd) Fix VersionedTest sometimes failing given certain querystring arguments (Damian Mooyman)
|
||||||
|
* 2016-01-20 [7c4e6f4](https://github.com/silverstripe/silverstripe-cms/commit/7c4e6f4b60567268ed879081823598438c90e729) prevent "Home page" being selected when no selection was made (Damian Mooyman)
|
||||||
|
* 2016-01-02 [b30d335](https://github.com/silverstripe/silverstripe-cms/commit/b30d33585f4640950dc573b9fa283c0db7b5f14c) Adding context parameter to canCreate-check in getClassDropdown of SiteTree (fixes #1334) (Stephan Bauer)
|
||||||
|
* 2016-01-02 [95e96fa](https://github.com/silverstripe/silverstripe-framework/commit/95e96fa2b2d0db9e26f8c716ee3d5e1a26ee09df) jquery.jstree patched to improve drag-and-drop handling (fixes #4881) (Stephan Bauer)
|
||||||
|
* 2015-12-22 [706877d](https://github.com/silverstripe/silverstripe-framework/commit/706877d72e6d64fd1093aa538cebad2311cbeca9) Get locale from <html> element for i18n.js (fixes #4854) (Loz Calver)
|
||||||
|
* 2015-12-22 [54ae002](https://github.com/silverstripe/silverstripe-cms/commit/54ae002d193d7677ff7a99527b37cbb6faa09343) FIx merge regressions in versioned tests (Damian Mooyman)
|
||||||
|
* 2015-12-22 [fce8251](https://github.com/silverstripe/silverstripe-framework/commit/fce82519bd6fcc313677b3687852ce15a3d5d202) Workaround for issues in testing version (Damian Mooyman)
|
||||||
|
* 2015-12-17 [36241d5](https://github.com/silverstripe-labs/silverstripe-reports/commit/36241d52a08ebce841f50fff91f3e4f4ac591be4) Fix regressions is SS_Report::canView (Damian Mooyman)
|
||||||
|
* 2015-12-15 [cd66917](https://github.com/silverstripe/silverstripe-framework/commit/cd66917a867275f3baf4c07efe2513db1ac92822) Vimeo oEmbed endpoint redirecting to no www (UndefinedOffset)
|
||||||
|
* 2015-12-15 [5d0f833](https://github.com/silverstripe-labs/silverstripe-reports/commit/5d0f833a397a2ce937e25b6a7c0350fdabdac63c) SS_Report canView should check permissions (Christopher Darling)
|
||||||
|
* 2015-12-09 [fa0160a](https://github.com/silverstripe/silverstripe-framework/commit/fa0160a874c536528d8300e034a7aa8bb6e23989) Fix regression in canViewStage (Damian Mooyman)
|
||||||
|
* 2015-11-24 [15ae37c](https://github.com/silverstripe/silverstripe-framework/commit/15ae37cf0351b654b5115183ab5a991c316e17e0) Image_Cached record class name (Jonathon Menz)
|
||||||
|
* 2015-10-31 [275ecfd](https://github.com/silverstripe/silverstripe-framework/commit/275ecfd8a95d4f7a025bb5025bb8d729a0e9eb70) Use `Object->hasMethod()` instead of `method_exists()` (madmatt)
|
||||||
|
* 2015-10-07 [71defe7](https://github.com/silverstripe/silverstripe-siteconfig/commit/71defe79b3e4fe7343f892ddf3aa8654725202c4) for #5 to facilitate validation on SiteConfig via DataExtension's. (Patrick Nelson)
|
||||||
|
* 2015-10-06 [a71d99c](https://github.com/silverstripe/silverstripe-framework/commit/a71d99cf8445a906ccd9b13242d36ae1e6a75d74) for #4663 ensuring return values from TabSet are retained from parent. Removing useless override. Cleaning up documentation in TabSet and return types. (Patrick Nelson)
|
||||||
|
* 2015-10-05 [12c4239](https://github.com/silverstripe/silverstripe-framework/commit/12c423909f721c6f5223007ad5e7ba6c162d63a4) (partial) for #3181 where non-submit buttons are being activated on "enter" key press (relates to CMS issue at https://github.com/silverstripe/silverstripe-cms/issues/1288). (Patrick Nelson)
|
||||||
|
* 2015-10-05 [332e490](https://github.com/silverstripe/silverstripe-cms/commit/332e4901478bf76705c7175e4af10b91d4c3b30f) (partial) for #1288 where non-submit buttons are being activated on "enter" key press (relates to framework issue at https://github.com/silverstripe/silverstripe-framework/issues/3181). (Patrick Nelson)
|
||||||
|
* 2015-10-05 [4a70ffe](https://github.com/silverstripe/silverstripe-framework/commit/4a70ffea0687c8c83b6210856e4c10f5aff0a883) Typo in cur methods PHPDoc (Corey Sewell)
|
||||||
|
* 2015-09-29 [5224fc4](https://github.com/silverstripe/silverstripe-framework/commit/5224fc460c6155c4f2253f42d88729b8f31066f6) Permission::checkMember() use of undefined variable $codes (Manuel Teuber)
|
||||||
|
* 2015-09-24 [c0be44d](https://github.com/silverstripe/silverstripe-framework/commit/c0be44d238c45853503fe1550fba0460a9a0f05c) fix response regression in initiation of request handler (Stevie Mayhew)
|
||||||
|
* 2015-09-17 [c9ba6e5](https://github.com/silverstripe/silverstripe-framework/commit/c9ba6e5d0064bfb09ebdb9e5f7054f8c3179f99a) Fix ClassInfo::table_for_object_field (Damian Mooyman)
|
||||||
|
* 2015-09-11 [5cc0878](https://github.com/silverstripe/silverstripe-framework/commit/5cc0878dc1feead47ead82c8f2beca02eefa102b) for #4597: Ensuring GridFieldConfig_RelationEditor is instantiated via Injector, not via "new" keyword. (Patrick Nelson)
|
||||||
|
* 2015-09-02 [2ae5d83](https://github.com/silverstripe/silverstripe-framework/commit/2ae5d83f08b994458aa93625e4ec7cb7f258bbae) Resampled images inherit source properties (Jonathon Menz)
|
||||||
|
* 2015-08-24 [80ce549](https://github.com/silverstripe/silverstripe-framework/commit/80ce5498d84088f8992de3f979071456e7d71746) disable archived pages from being droppable (Damian Mooyman)
|
||||||
|
* 2015-08-21 [b14794b](https://github.com/silverstripe/silverstripe-framework/commit/b14794b780b30d5a6d39df9ed080135ff25045a8) Fix bulk actions making sitetree unclickable (Damian Mooyman)
|
||||||
|
* 2015-08-19 [a19fe39](https://github.com/silverstripe/silverstripe-framework/commit/a19fe39301f8a6a2e80e9a9d294c425b8699dc0c) Avoid PHP 5.6 deprecation with access to HTTP_RAW_POST_DATA. Fixed #4511 (Sam Minnee)
|
||||||
|
* 2015-07-31 [6a45f4a](https://github.com/silverstripe/silverstripe-framework/commit/6a45f4a1e125b1a75d042e59b38824b24fd3cd0f) fix mismatched quotes (Damian Mooyman)
|
||||||
|
* 2015-06-15 [ca039e1](https://github.com/silverstripe/silverstripe-framework/commit/ca039e15ef7306d7b56d64d93892d2fb6173fcf7) Fix regressions in changes to batch action feature (David Craig)
|
||||||
|
* 2015-06-11 [8a4c518](https://github.com/silverstripe/silverstripe-framework/commit/8a4c51893b345f7653e77acdd3667bbe61346784) allow for increase_time_limit_to to work if $_increase_time_limit_max is not yet set (Stevie Mayhew)
|
||||||
|
30
docs/en/04_Changelogs/rc/3.2.2-rc1.md
Normal file
30
docs/en/04_Changelogs/rc/3.2.2-rc1.md
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
# 3.2.2-rc1
|
||||||
|
|
||||||
|
<!--- Changes below this line will be automatically regenerated -->
|
||||||
|
|
||||||
|
## Change Log
|
||||||
|
|
||||||
|
### Bugfixes
|
||||||
|
|
||||||
|
* 2016-02-15 [8771859](https://github.com/silverstripe/silverstripe-framework/commit/87718597e8f04872c285808d0666fbb69c5100ba) "where" method in SQLUpdate Example (Richard Rudy)
|
||||||
|
* 2016-01-28 [3fcf1e2](https://github.com/silverstripe/silverstripe-framework/commit/3fcf1e2c98629dcd0048ff9447bad4cd30b4bf95) edge case on many many extra fields (fixes 4991) (Mark Stephens)
|
||||||
|
* 2016-01-27 [3d0178e](https://github.com/silverstripe/silverstripe-cms/commit/3d0178ebc0b7408442ad2532f998ed47839e7117) Use correct formaction for doRollback exemption (Damian Mooyman)
|
||||||
|
* 2016-01-24 [d8e354d](https://github.com/silverstripe/silverstripe-framework/commit/d8e354d144383fb6459adf92731853d2e54268d6) PHPDocs on DataList::getIDList() and UnsavedRelationList::getIDList() (Damian Mooyman)
|
||||||
|
* 2016-01-22 [bf8bf5e](https://github.com/silverstripe/silverstripe-framework/commit/bf8bf5e4d558126bb99ea63881f1885faafddd3d) Prevent Versioned::doRollbackTo from creating incorrect versions on subclasses of Versioned DataObjects (Damian Mooyman)
|
||||||
|
* 2016-01-21 [cca7129](https://github.com/silverstripe/silverstripe-framework/commit/cca7129385dbb3be1001a8861423c2cf490f02d4) Revert lost documentation (Damian Mooyman)
|
||||||
|
* 2016-01-11 [85ba918](https://github.com/silverstripe/silverstripe-framework/commit/85ba918a54f51dd524d45f2c93172a18421ae3bf) Update field IDs for file link (fixes silverstripe/silverstripe-cms#1307) (Loz Calver)
|
||||||
|
* 2016-01-11 [d637141](https://github.com/silverstripe/silverstripe-cms/commit/d6371414876e32e7369ec0219a57d2186cfe3f0f) preg_quote() anchors in SiteTreeLinkTracking (fixes #1359) (Loz Calver)
|
||||||
|
* 2016-01-05 [00544ff](https://github.com/silverstripe/silverstripe-framework/commit/00544ff100048afdb7ccb1905304dddf8ab3205a) session_regenerate_id uses config system (Daniel Hensby)
|
||||||
|
* 2016-01-05 [4335d8e](https://github.com/silverstripe/silverstripe-framework/commit/4335d8ed221a2b402299b32e31f97fc2956ec161) Members with no ID inherit logged in user permission (Daniel Hensby)
|
||||||
|
* 2015-12-15 [afbb5cf](https://github.com/silverstripe/silverstripe-framework/commit/afbb5cfed4d29aea5868f0f12cd735dc5abe10d3) Vimeo oEmbed endpoint redirecting to no www (UndefinedOffset)
|
||||||
|
* 2015-12-14 [d265c9b](https://github.com/silverstripe/silverstripe-framework/commit/d265c9b733ddac27d6df286ce000b09e1c69b986) Allow omitting a value for OptionsetField submissions (fixes #4824) (Loz Calver)
|
||||||
|
* 2015-12-11 [5a21b2f](https://github.com/silverstripe/silverstripe-framework/commit/5a21b2fb15ed9c675594f0f990765bd4f97155c7) Guard against users being added to all groups on unsaved Group. (Mateusz Uzdowski)
|
||||||
|
* 2015-11-27 [94742fa](https://github.com/silverstripe/silverstripe-framework/commit/94742fa3e2efad8f77f4acd1f9d06bf74916c5e6) Revert method visibility regression (Damian Mooyman)
|
||||||
|
* 2015-11-18 [e9b833f](https://github.com/silverstripe/silverstripe-framework/commit/e9b833f5f0f989af8d611f8cfe71f0b0e2cb0159) ConfirmedPassword field correctly reports mismatching passwords (Christopher Darling)
|
||||||
|
* 2015-11-17 [68d99be](https://github.com/silverstripe/silverstripe-framework/commit/68d99be24b63a933f041cd80a248a7b7fa8d588c) Hidden errors for composite fields nested inside FieldGroups (fixes #4773) (Loz Calver)
|
||||||
|
* 2015-11-17 [97e90b8](https://github.com/silverstripe/silverstripe-cms/commit/97e90b8ebd8078bb60ecea66bdd3761380f93a61) RedirectorPage toggles not working (fixes #1328) (Loz Calver)
|
||||||
|
* 2015-11-17 [b624eb9](https://github.com/silverstripe/silverstripe-cms/commit/b624eb98f1d1ff36811a3294ad29b31a50683d60) Setting target for unwritten VirtualPage breaks write (Loz Calver)
|
||||||
|
* 2015-11-16 [2983d82](https://github.com/silverstripe/silverstripe-cms/commit/2983d823d1eef293ef11aac9e01336e23ed52b59) Ensure VirtualPage forwards request/response data to virtual controllers (fixes #1329) (Loz Calver)
|
||||||
|
* 2015-11-12 [fea1158](https://github.com/silverstripe/silverstripe-framework/commit/fea1158d193ed4d037df94101e3b3f2d24a6ce49) Fix print button only displaying first page (Damian Mooyman)
|
||||||
|
* 2015-11-11 [a40812a](https://github.com/silverstripe/silverstripe-framework/commit/a40812ac3320d27f243ef0ed54aa003fc53720b6) Don’t reuse DBConnector (fixes #4735) (Sam Minnee)
|
||||||
|
* 2015-11-05 [f577ecb](https://github.com/silverstripe/silverstripe-framework/commit/f577ecb81149d0d09dc846204f17b2153a244b5a) prevent use cache on browser back button (Igor Nadj)
|
11
docs/en/04_Changelogs/rc/3.2.2-rc2.md
Normal file
11
docs/en/04_Changelogs/rc/3.2.2-rc2.md
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
# 3.2.2-rc2
|
||||||
|
|
||||||
|
<!--- Changes below this line will be automatically regenerated -->
|
||||||
|
|
||||||
|
## Change Log
|
||||||
|
|
||||||
|
### Security
|
||||||
|
|
||||||
|
* 2016-02-17 [faa94d5](https://github.com/silverstripe/silverstripe-framework/commit/faa94d51d570788dcebc2f2ef6e9de4d179ce1e4) Hostname, IP and Protocol Spoofing through HTTP Headers (Ingo Schommer) - See [ss-2016-003](http://www.silverstripe.org/download/security-releases/ss-2016-003)
|
||||||
|
* 2016-02-17 [15d4db3](https://github.com/silverstripe/silverstripe-framework/commit/15d4db3b4a7dbc9a7e089f9329a396f8408ed7d9) Block unauthenticated access to dev/build/defaults (Damian Mooyman) - See [ss-2015-028](http://www.silverstripe.org/download/security-releases/ss-2015-028)
|
||||||
|
* 2016-02-17 [e2c77c5](https://github.com/silverstripe/silverstripe-framework/commit/e2c77c5a8f13e901c51a3684210811559b592f0c) Ensure Gridfield actions respect CSRF (Damian Mooyman) - See [ss-2016-002](http://www.silverstripe.org/download/security-releases/ss-2016-002)
|
11
docs/en/04_Changelogs/rc/3.3.0-rc3.md
Normal file
11
docs/en/04_Changelogs/rc/3.3.0-rc3.md
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
# 3.3.0-rc3
|
||||||
|
|
||||||
|
<!--- Changes below this line will be automatically regenerated -->
|
||||||
|
|
||||||
|
## Change Log
|
||||||
|
|
||||||
|
### Security
|
||||||
|
|
||||||
|
* 2016-02-17 [893e497](https://github.com/silverstripe/silverstripe-framework/commit/893e49703de4aa1855b5364919cbb0826f754fbf) Hostname, IP and Protocol Spoofing through HTTP Headers (Ingo Schommer) - See [ss-2016-003](http://www.silverstripe.org/download/security-releases/ss-2016-003)
|
||||||
|
* 2016-02-17 [3398f67](https://github.com/silverstripe/silverstripe-framework/commit/3398f670d881447f8777b567f1ead7c0d8d253f5) Block unauthenticated access to dev/build/defaults (Damian Mooyman) - See [ss-2015-028](http://www.silverstripe.org/download/security-releases/ss-2015-028)
|
||||||
|
* 2016-02-17 [56e92f5](https://github.com/silverstripe/silverstripe-framework/commit/56e92f5a32e45849cc9361c8603c31d7010c9d36) Ensure Gridfield actions respect CSRF (Damian Mooyman) - See [ss-2016-002](http://www.silverstripe.org/download/security-releases/ss-2016-002)
|
@ -11,7 +11,7 @@ well written bug reports can be half of the solution already!
|
|||||||
* [Framework Bugtracker](https://github.com/silverstripe/silverstripe-framework/issues)
|
* [Framework Bugtracker](https://github.com/silverstripe/silverstripe-framework/issues)
|
||||||
* [CMS Bugtracker](https://github.com/silverstripe/silverstripe-cms/issues)
|
* [CMS Bugtracker](https://github.com/silverstripe/silverstripe-cms/issues)
|
||||||
* [Documentation Bugtracker](https://github.com/silverstripe/silverstripe-framework/issues)
|
* [Documentation Bugtracker](https://github.com/silverstripe/silverstripe-framework/issues)
|
||||||
* Search on [http://silverstripe.org/modules](http://silverstripe.org/modules) for module-specific bugtrackers
|
* Search on [http://addons.silverstripe.org/](http://addons.silverstripe.org/) for module-specific bugtrackers
|
||||||
* Request features: [UserVoice](http://silverstripe.uservoice.com).
|
* Request features: [UserVoice](http://silverstripe.uservoice.com).
|
||||||
|
|
||||||
Before submitting a bug:
|
Before submitting a bug:
|
||||||
@ -19,7 +19,7 @@ Before submitting a bug:
|
|||||||
* Ask for assistance on the [forums](http://www.silverstripe.org/community/forums/), [core mailinglist](http://groups.google.com/group/silverstripe-dev) or on [IRC](http://irc.silverstripe.org/) if you're unsure if its really a bug.
|
* Ask for assistance on the [forums](http://www.silverstripe.org/community/forums/), [core mailinglist](http://groups.google.com/group/silverstripe-dev) or on [IRC](http://irc.silverstripe.org/) if you're unsure if its really a bug.
|
||||||
* Search for similar, existing tickets
|
* Search for similar, existing tickets
|
||||||
* Is this a security issue? Please follow our separate reporting guidelines below.
|
* Is this a security issue? Please follow our separate reporting guidelines below.
|
||||||
* Is this a issue with the core framework or cms? Modules have their own issue trackers (see [silverstripe.org/modules](http://www.silverstripe.org/modules/))
|
* Is this a issue with the core framework or cms? Modules have their own issue trackers (see [http://addons.silverstripe.org/](http://addons.silverstripe.org/))
|
||||||
* Try to reproduce your issue on a [clean installation](/getting_started/composer#using-development-versions), maybe the bug has already been fixed on an unreleased branch?
|
* Try to reproduce your issue on a [clean installation](/getting_started/composer#using-development-versions), maybe the bug has already been fixed on an unreleased branch?
|
||||||
* The bugtracker is not the place to discuss enhancements, please use the forums or mailinglist.
|
* The bugtracker is not the place to discuss enhancements, please use the forums or mailinglist.
|
||||||
Only log enhancement tickets if they gather a large interest in the community
|
Only log enhancement tickets if they gather a large interest in the community
|
||||||
|
@ -161,7 +161,7 @@ Most importantly: Keep the first line short, and add more detail below.
|
|||||||
This ensures commits are easy to browse, and look nice on github.com
|
This ensures commits are easy to browse, and look nice on github.com
|
||||||
(more info about [proper git commit messages](http://tbaggery.com/2008/04/19/a-note-about-git-commit-messages.html)).
|
(more info about [proper git commit messages](http://tbaggery.com/2008/04/19/a-note-about-git-commit-messages.html)).
|
||||||
|
|
||||||
As we automatically generate [changelogs](http://doc.silverstripe.org/sapphire/en/trunk/changelogs/) from them, we need a way to categorize and filter.
|
As we automatically generate [changelogs](http://localhost/SpiritLevel/SS/doc.silverstripe.org/en/changelogs/) from them, we need a way to categorize and filter.
|
||||||
Please prefix **noteworthy** commit messages with one of the following tags:
|
Please prefix **noteworthy** commit messages with one of the following tags:
|
||||||
|
|
||||||
* `NEW` New feature or major enhancement (both for users and developers)
|
* `NEW` New feature or major enhancement (both for users and developers)
|
||||||
|
@ -51,7 +51,7 @@ so we strive for giving ample warning on any upcoming changes through a "depreca
|
|||||||
How to deprecate an API:
|
How to deprecate an API:
|
||||||
|
|
||||||
* Add a `@deprecated` item to the docblock tag, with a `{@link <class>}` item pointing to the new API to use.
|
* Add a `@deprecated` item to the docblock tag, with a `{@link <class>}` item pointing to the new API to use.
|
||||||
* Update the deprecated code to throw a `[api:Deprecation::notice()]` error.
|
* Update the deprecated code to throw a [api:Deprecation::notice()] error.
|
||||||
* Both the docblock and error message should contain the **target version** where the functionality is removed.
|
* Both the docblock and error message should contain the **target version** where the functionality is removed.
|
||||||
So, if you're committing the change to a *3.1* minor release, the target version will be *4.0*.
|
So, if you're committing the change to a *3.1* minor release, the target version will be *4.0*.
|
||||||
* Deprecations should not be committed to patch releases
|
* Deprecations should not be committed to patch releases
|
||||||
|
@ -73,7 +73,7 @@ the [core committers](core_committers), who will assist with setting up your cre
|
|||||||
* Admin permissions on [transifex](https://www.transifex.com/silverstripe/).
|
* Admin permissions on [transifex](https://www.transifex.com/silverstripe/).
|
||||||
* AWS write permissions on the `silverstripe-ssorg-releases` s3 bucket.
|
* AWS write permissions on the `silverstripe-ssorg-releases` s3 bucket.
|
||||||
* Permission on [silverstripe release announcement](https://groups.google.com/forum/#!forum/silverstripe-announce).
|
* Permission on [silverstripe release announcement](https://groups.google.com/forum/#!forum/silverstripe-announce).
|
||||||
* Moderator permissions in the #silverstripe [IRC channel]((http://www.silverstripe.org/community/contributing-to-silverstripe/irc-channel/))
|
* Moderator permissions in the #silverstripe [IRC channel](http://www.silverstripe.org/community/contributing-to-silverstripe/irc-channel/)
|
||||||
|
|
||||||
### First time setup: Security releases
|
### First time setup: Security releases
|
||||||
|
|
||||||
@ -333,7 +333,7 @@ will need to be regularly updated.
|
|||||||
* Create a release announcement forum sticky on the
|
* Create a release announcement forum sticky on the
|
||||||
[releases and announcements](http://www.silverstripe.org/community/forums/releases-and-announcements/)
|
[releases and announcements](http://www.silverstripe.org/community/forums/releases-and-announcements/)
|
||||||
forum category. Make this a global read-only sticky, and un-sticky any older release.
|
forum category. Make this a global read-only sticky, and un-sticky any older release.
|
||||||
* Update the #silverstripe [IRC](http://www.silverstripe.org/community/contributing-to-silverstripe/irc-channel/) topic to include the new release version.
|
* Update the #silverstripe [IRC](https://www.silverstripe.org/community/contributing-to-silverstripe/irc-channel/) topic to include the new release version.
|
||||||
|
|
||||||
### Stage 4: Web platform installer release
|
### Stage 4: Web platform installer release
|
||||||
|
|
||||||
@ -342,8 +342,9 @@ The web platform installer is available [on the web app gallery](http://www.micr
|
|||||||
In order to update this you will need a Microsoft live account, and have it authorised
|
In order to update this you will need a Microsoft live account, and have it authorised
|
||||||
by SilverStripe staff in order to publish these releases.
|
by SilverStripe staff in order to publish these releases.
|
||||||
|
|
||||||
|
|
||||||
To update this release there is an additional download tool at
|
To update this release there is an additional download tool at
|
||||||
[https://code.platform.silverstripe.com/silverstripe/webpi](https://code.platform.silverstripe.com/silverstripe/webpi)
|
`[https://code.platform.silverstripe.com/silverstripe/webpi](https://code.platform.silverstripe.com/silverstripe/webpi)`
|
||||||
which will guide you through the process of generating a new zip release.
|
which will guide you through the process of generating a new zip release.
|
||||||
|
|
||||||
./make-package 3.2.4 3.2.4
|
./make-package 3.2.4 3.2.4
|
||||||
@ -363,6 +364,7 @@ to submit a new version, including:
|
|||||||
* [Release Process](release_process)
|
* [Release Process](release_process)
|
||||||
* [Translation Process](translation_process)
|
* [Translation Process](translation_process)
|
||||||
* [Core committers](core_committers)
|
* [Core committers](core_committers)
|
||||||
|
* [WebPI Installer](https://docs.silverstripe.org/en/getting_started/installation/other_installation_options/windows_platform_installer/)
|
||||||
|
|
||||||
If at any time a release runs into an unsolveable problem contact the
|
If at any time a release runs into an unsolveable problem contact the
|
||||||
core committers on the [discussion group](https://groups.google.com/forum/#!forum/silverstripe-committers)
|
core committers on the [discussion group](https://groups.google.com/forum/#!forum/silverstripe-committers)
|
||||||
|
@ -37,7 +37,7 @@ If you submit a new feature or an API change, we strongly recommend that your pa
|
|||||||
## Repositories
|
## Repositories
|
||||||
|
|
||||||
* End-user help: [userhelp.silverstripe.org](http://github.com/silverstripe/userhelp.silverstripe.org)
|
* End-user help: [userhelp.silverstripe.org](http://github.com/silverstripe/userhelp.silverstripe.org)
|
||||||
* Developer guides: [docs.silverstripe.org](http://github.com/silverstripe/docs.silverstripe.org)
|
* Developer guides: [docs.silverstripe.org](http://github.com/silverstripe/doc.silverstripe.org)
|
||||||
* Developer API documentation: [api.silverstripe.org](http://github.com/silverstripe/api.silverstripe.org)
|
* Developer API documentation: [api.silverstripe.org](http://github.com/silverstripe/api.silverstripe.org)
|
||||||
|
|
||||||
## Source control
|
## Source control
|
||||||
|
@ -59,7 +59,7 @@ Currently translated entities are not directly factored into code (for security
|
|||||||
you can't see them straight away.
|
you can't see them straight away.
|
||||||
|
|
||||||
It is strongly encouraged that you check your translation this way, as its a good way to double check your translation
|
It is strongly encouraged that you check your translation this way, as its a good way to double check your translation
|
||||||
works in the right context. Please use our [daily-builds](http://www.silverstripe.org/daily-builds/) for your local
|
works in the right context. Please use our `[daily-builds](http://www.silverstripe.org/daily-builds/)` for your local
|
||||||
installation, to ensure you're looking at the most up to date interface. See "Download Translations" above to find out
|
installation, to ensure you're looking at the most up to date interface. See "Download Translations" above to find out
|
||||||
how to retrieve the latest translation files.
|
how to retrieve the latest translation files.
|
||||||
|
|
||||||
@ -135,7 +135,7 @@ This also applies for any modules staying compatible with SilverStripe 2.x.
|
|||||||
## Contact
|
## Contact
|
||||||
|
|
||||||
Translators have their own [mailinglist](https://groups.google.com/forum/#!forum/silverstripe-translators), but you can
|
Translators have their own [mailinglist](https://groups.google.com/forum/#!forum/silverstripe-translators), but you can
|
||||||
also reach a core member on [IRC](http://silverstripe.org/irc). The transifex.com interface has a built-in discussion
|
also reach a core member on [IRC](https://irc.silverstripe.org). The transifex.com interface has a built-in discussion
|
||||||
board if you have specific comments on a translation.
|
board if you have specific comments on a translation.
|
||||||
|
|
||||||
## Related
|
## Related
|
||||||
@ -143,4 +143,4 @@ board if you have specific comments on a translation.
|
|||||||
* [i18n](/developer_guides/i18n): Developer-level documentation of Silverstripe's i18n capabilities
|
* [i18n](/developer_guides/i18n): Developer-level documentation of Silverstripe's i18n capabilities
|
||||||
* [Translation Process](translation_process): Information about managing translations for the core team and/or module maintainers.
|
* [Translation Process](translation_process): Information about managing translations for the core team and/or module maintainers.
|
||||||
* [translatable](https://github.com/silverstripe/silverstripe-translatable): DataObject-interface powering the website-content translations
|
* [translatable](https://github.com/silverstripe/silverstripe-translatable): DataObject-interface powering the website-content translations
|
||||||
* ["Translatable ModelAdmin" module](http://silverstripe.org/translatablemodeladmin-module/): An extension which allows translations of DataObjects inside ModelAdmin
|
* `["Translatable ModelAdmin" module](http://silverstripe.org/translatablemodeladmin-module/)`: An extension which allows translations of DataObjects inside ModelAdmin
|
||||||
|
@ -130,4 +130,4 @@ files back into the JS files SilverStripe can actually read. This requires an in
|
|||||||
* [i18n](/developer_guides/i18n/): Developer-level documentation of Silverstripe's i18n capabilities
|
* [i18n](/developer_guides/i18n/): Developer-level documentation of Silverstripe's i18n capabilities
|
||||||
* [Contributing Translations](/contributing/translations): Information for translators looking to contribute translations of the SilverStripe UI.
|
* [Contributing Translations](/contributing/translations): Information for translators looking to contribute translations of the SilverStripe UI.
|
||||||
* [translatable](https://github.com/silverstripe/silverstripe-translatable): DataObject-interface powering the website-content translations
|
* [translatable](https://github.com/silverstripe/silverstripe-translatable): DataObject-interface powering the website-content translations
|
||||||
* ["Translatable ModelAdmin" module](http://silverstripe.org/translatablemodeladmin-module/): An extension which allows translations of DataObjects inside ModelAdmin
|
* `["Translatable ModelAdmin" module](http://silverstripe.org/translatablemodeladmin-module/)`: An extension which allows translations of DataObjects inside ModelAdmin
|
||||||
|
@ -57,4 +57,4 @@ https://www.djangoproject.com/conduct/
|
|||||||
|
|
||||||
http://web.archive.org/web/20141109123859/http://speakup.io/coc.html
|
http://web.archive.org/web/20141109123859/http://speakup.io/coc.html
|
||||||
|
|
||||||
http://www.crnhq.org/pages.php?pID=10
|
http://www.crnhq.org/files/66138/files/Handouts%20and%20Posters/ResolveTheConflictGuideposter.pdf
|
||||||
|
@ -20,7 +20,7 @@ community. There are also several other websites with SilverStripe documentation
|
|||||||
* The [API Documentation](http://api.silverstripe.org/) contains technical reference and class information.
|
* The [API Documentation](http://api.silverstripe.org/) contains technical reference and class information.
|
||||||
* The [User Help](http://userhelp.silverstripe.com) website contains documentation related to working within the CMS.
|
* The [User Help](http://userhelp.silverstripe.com) website contains documentation related to working within the CMS.
|
||||||
|
|
||||||
New features, API changes and the development [roadmap](http://www.silverstripe.org/roadmap/) for the product are
|
New features, API changes and the development [roadmap](http://www.silverstripe.org/software/roadmap/) for the product are
|
||||||
discussed on the [core mailinglist](https://groups.google.com/forum/#!forum/silverstripe-dev) along with
|
discussed on the [core mailinglist](https://groups.google.com/forum/#!forum/silverstripe-dev) along with
|
||||||
[UserVoice](http://silverstripe.uservoice.com/forums/251266-new-features).
|
[UserVoice](http://silverstripe.uservoice.com/forums/251266-new-features).
|
||||||
|
|
||||||
|
@ -833,6 +833,18 @@ class GridField extends FormField {
|
|||||||
*/
|
*/
|
||||||
public function gridFieldAlterAction($data, $form, SS_HTTPRequest $request) {
|
public function gridFieldAlterAction($data, $form, SS_HTTPRequest $request) {
|
||||||
$data = $request->requestVars();
|
$data = $request->requestVars();
|
||||||
|
|
||||||
|
// Protection against CSRF attacks
|
||||||
|
$token = $this
|
||||||
|
->getForm()
|
||||||
|
->getSecurityToken();
|
||||||
|
if(!$token->checkRequest($request)) {
|
||||||
|
$this->httpError(400, _t("Form.CSRF_FAILED_MESSAGE",
|
||||||
|
"There seems to have been a technical problem. Please click the back button, ".
|
||||||
|
"refresh your browser, and try again."
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
$name = $this->getName();
|
$name = $this->getName();
|
||||||
|
|
||||||
$fieldData = null;
|
$fieldData = null;
|
||||||
|
@ -888,10 +888,11 @@ ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
|||||||
action = "update";
|
action = "update";
|
||||||
}
|
}
|
||||||
|
|
||||||
if(href.match(/^mailto:(.*)$/)) {
|
if(href.match(/^mailto:([^?]*)(\?subject=(.*))?$/)) {
|
||||||
return {
|
return {
|
||||||
LinkType: 'email',
|
LinkType: 'email',
|
||||||
email: RegExp.$1,
|
email: RegExp.$1,
|
||||||
|
Subject: decodeURIComponent(RegExp.$3),
|
||||||
Description: title
|
Description: title
|
||||||
};
|
};
|
||||||
} else if(href.match(/^(assets\/.*)$/) || href.match(/^\[file_link\s*(?:\s*|%20|,)?id=([0-9]+)\]?(#.*)?$/)) {
|
} else if(href.match(/^(assets\/.*)$/) || href.match(/^\[file_link\s*(?:\s*|%20|,)?id=([0-9]+)\]?(#.*)?$/)) {
|
||||||
|
@ -301,7 +301,7 @@ cs:
|
|||||||
FROMWEB: 'Z webu'
|
FROMWEB: 'Z webu'
|
||||||
FindInFolder: 'Hledat ve složce'
|
FindInFolder: 'Hledat ve složce'
|
||||||
IMAGEALT: 'Alternativní text (alt)'
|
IMAGEALT: 'Alternativní text (alt)'
|
||||||
IMAGEALTTEXT: 'Alternativní text (alt) - bude zobrazen, pokud obrázek nemúže být zobrazen'
|
IMAGEALTTEXT: 'Alternativní text (alt) - bude ukázán, když obrázek nemúže být zobrazen'
|
||||||
IMAGEALTTEXTDESC: 'Zobrazeno na obrazovce, když obrázek nemůže být zobrazen'
|
IMAGEALTTEXTDESC: 'Zobrazeno na obrazovce, když obrázek nemůže být zobrazen'
|
||||||
IMAGEDIMENSIONS: Rozměry
|
IMAGEDIMENSIONS: Rozměry
|
||||||
IMAGEHEIGHTPX: Výška
|
IMAGEHEIGHTPX: Výška
|
||||||
@ -336,13 +336,13 @@ cs:
|
|||||||
LeftAndMain:
|
LeftAndMain:
|
||||||
CANT_REORGANISE: 'Nemáte oprávnění měnit stránky nejvyšší úrovně. Vaše změna nebyla uložena.'
|
CANT_REORGANISE: 'Nemáte oprávnění měnit stránky nejvyšší úrovně. Vaše změna nebyla uložena.'
|
||||||
DELETED: Smazáno.
|
DELETED: Smazáno.
|
||||||
DropdownBatchActionsDefault: 'Vyberte akci...'
|
DropdownBatchActionsDefault: Akce
|
||||||
HELP: Nápověda
|
HELP: Nápověda
|
||||||
PAGETYPE: 'Typ stránky'
|
PAGETYPE: 'Typ stránky'
|
||||||
PERMAGAIN: 'Byli jste odhlášeni z CMS. Pokud se chcete znovu přihlásit, zadejte níže své uživatelské jméno a heslo.'
|
PERMAGAIN: 'Byli jste odhlášeni z CMS. Pokud se chcete znovu přihlásit, zadejte níže své uživatelské jméno a heslo.'
|
||||||
PERMALREADY: 'Omlouvám se, ale nemůžete vstoupit do této části CMS. Pokud se chcete přihlásit jako někdo jiný, udělejte tak níže.'
|
PERMALREADY: 'Omlouvám se, ale nemůžete vstoupit do této části CMS. Pokud se chcete přihlásit jako někdo jiný, udělejte tak níže.'
|
||||||
PERMDEFAULT: 'Musíte být přihlášen/a k přístup do oblasti administrace, níže zadejte vaše přihlašovací údaje, prosím.'
|
PERMDEFAULT: 'Musíte být přihlášen/a k přístup do oblasti administrace, níže zadejte vaše přihlašovací údaje, prosím.'
|
||||||
PLEASESAVE: 'Prosím uložte stránku: Tato stránka nemohla být aktualizována, protože ještě nebyla uložena.'
|
PLEASESAVE: 'Prosím uložte stránku: Tato stránka nemohla být aktualizována, ještě nebyla uložena.'
|
||||||
PreviewButton: Náhled
|
PreviewButton: Náhled
|
||||||
REORGANISATIONSUCCESSFUL: 'Strom webu reorganizován úspěšně.'
|
REORGANISATIONSUCCESSFUL: 'Strom webu reorganizován úspěšně.'
|
||||||
SAVEDUP: Uloženo.
|
SAVEDUP: Uloženo.
|
||||||
|
11
lang/de.yml
11
lang/de.yml
@ -1,5 +1,7 @@
|
|||||||
de:
|
de:
|
||||||
AssetAdmin:
|
AssetAdmin:
|
||||||
|
ALLOWEDEXTS: 'Erlaubte Dateiendungen'
|
||||||
|
HIDEALLOWEDEXTS: 'Verberge erlaubte Dateiendungen'
|
||||||
NEWFOLDER: Neuer Ordner
|
NEWFOLDER: Neuer Ordner
|
||||||
SHOWALLOWEDEXTS: 'Erlaubte Dateitypen anzeigen'
|
SHOWALLOWEDEXTS: 'Erlaubte Dateitypen anzeigen'
|
||||||
AssetTableField:
|
AssetTableField:
|
||||||
@ -7,12 +9,14 @@ de:
|
|||||||
DIM: Dimensionen
|
DIM: Dimensionen
|
||||||
FILENAME: Dateiname
|
FILENAME: Dateiname
|
||||||
FOLDER: Ordner
|
FOLDER: Ordner
|
||||||
|
HEIGHT: Höhe
|
||||||
LASTEDIT: 'Letzte Änderung'
|
LASTEDIT: 'Letzte Änderung'
|
||||||
OWNER: Eigentümer
|
OWNER: Eigentümer
|
||||||
SIZE: 'Größe'
|
SIZE: 'Größe'
|
||||||
TITLE: Titel
|
TITLE: Titel
|
||||||
TYPE: 'Typ'
|
TYPE: 'Typ'
|
||||||
URL: URL
|
URL: URL
|
||||||
|
WIDTH: Breite
|
||||||
AssetUploadField:
|
AssetUploadField:
|
||||||
ChooseFiles: 'Dateien auswählen'
|
ChooseFiles: 'Dateien auswählen'
|
||||||
DRAGFILESHERE: 'Dateien hier ablegen'
|
DRAGFILESHERE: 'Dateien hier ablegen'
|
||||||
@ -23,7 +27,9 @@ de:
|
|||||||
FILES: Dateien
|
FILES: Dateien
|
||||||
FROMCOMPUTER: 'Dateien auf Ihrem Computer auswählen'
|
FROMCOMPUTER: 'Dateien auf Ihrem Computer auswählen'
|
||||||
FROMCOMPUTERINFO: 'Von Ihrem Computer hochladen'
|
FROMCOMPUTERINFO: 'Von Ihrem Computer hochladen'
|
||||||
|
REMOVEINFO: 'Entferne diese Datei aus dem Feld'
|
||||||
TOTAL: Insgesamt
|
TOTAL: Insgesamt
|
||||||
|
TOUPLOAD: 'Dateien für den Upload auswählen...'
|
||||||
UPLOADINPROGRESS: 'Bitte warten sie... Upload im Gang'
|
UPLOADINPROGRESS: 'Bitte warten sie... Upload im Gang'
|
||||||
UPLOADOR: oder
|
UPLOADOR: oder
|
||||||
BBCodeParser:
|
BBCodeParser:
|
||||||
@ -146,6 +152,7 @@ de:
|
|||||||
INVALID_REQUEST: 'Ungültige Anfrage'
|
INVALID_REQUEST: 'Ungültige Anfrage'
|
||||||
DropdownField:
|
DropdownField:
|
||||||
CHOOSE: (Auswahl)
|
CHOOSE: (Auswahl)
|
||||||
|
CHOOSESEARCH: '(Auswahl und Suche)'
|
||||||
SOURCE_VALIDATION: 'Bitte wählen Sie aus der Liste. {value} ist kein gültiger Wert'
|
SOURCE_VALIDATION: 'Bitte wählen Sie aus der Liste. {value} ist kein gültiger Wert'
|
||||||
EmailField:
|
EmailField:
|
||||||
VALIDATION: 'Bitte geben Sie eine E-Mail-Adresse ein'
|
VALIDATION: 'Bitte geben Sie eine E-Mail-Adresse ein'
|
||||||
@ -293,8 +300,6 @@ de:
|
|||||||
FROMWEB: 'Aus dem Web'
|
FROMWEB: 'Aus dem Web'
|
||||||
FindInFolder: 'In Ordner suchen'
|
FindInFolder: 'In Ordner suchen'
|
||||||
IMAGEALT: 'Alternativtext (alt)'
|
IMAGEALT: 'Alternativtext (alt)'
|
||||||
IMAGEALTTEXT: 'Alternativer Text (alt) - angezeigt, wenn das Bild nicht dargestellt werden kann'
|
|
||||||
IMAGEALTTEXTDESC: 'Wird von Screenreadern vorgelesen oder angezeigt, falls das Bild nicht dargestellt werden kann'
|
|
||||||
IMAGEDIMENSIONS: Dimensionen
|
IMAGEDIMENSIONS: Dimensionen
|
||||||
IMAGEHEIGHTPX: Höhe (px)
|
IMAGEHEIGHTPX: Höhe (px)
|
||||||
IMAGETITLE: 'Titeltext (Tooltip) - für zusätzliche Informationen über das Bild'
|
IMAGETITLE: 'Titeltext (Tooltip) - für zusätzliche Informationen über das Bild'
|
||||||
@ -326,11 +331,9 @@ de:
|
|||||||
CANT_REORGANISE: 'Sie besitzen nicht die benötigten Zugriffsrechte um Seiten der höchsten Ebene zu bearbeiten. Ihre Änderungen wurden nicht gespeichert.'
|
CANT_REORGANISE: 'Sie besitzen nicht die benötigten Zugriffsrechte um Seiten der höchsten Ebene zu bearbeiten. Ihre Änderungen wurden nicht gespeichert.'
|
||||||
DELETED: Gelöscht.
|
DELETED: Gelöscht.
|
||||||
HELP: Hilfe
|
HELP: Hilfe
|
||||||
PAGETYPE: 'Seitentyp'
|
|
||||||
PERMAGAIN: 'Sie wurden aus dem System ausgeloggt. Falls Sie sich wieder einloggen möchten, geben Sie bitte Benutzernamen und Passwort im untenstehenden Formular an.'
|
PERMAGAIN: 'Sie wurden aus dem System ausgeloggt. Falls Sie sich wieder einloggen möchten, geben Sie bitte Benutzernamen und Passwort im untenstehenden Formular an.'
|
||||||
PERMALREADY: 'Leider dürfen Sie diesen Teil des CMS nicht aufrufen. Wenn Sie sich als jemand anderes einloggen wollen, benutzen Sie bitte das nachstehende Formular.'
|
PERMALREADY: 'Leider dürfen Sie diesen Teil des CMS nicht aufrufen. Wenn Sie sich als jemand anderes einloggen wollen, benutzen Sie bitte das nachstehende Formular.'
|
||||||
PERMDEFAULT: 'Sie müssen angemeldet sein, um auf diesen Bereich zugreifen zu können. Bitte geben Sie Ihre Zugangsdaten ein.'
|
PERMDEFAULT: 'Sie müssen angemeldet sein, um auf diesen Bereich zugreifen zu können. Bitte geben Sie Ihre Zugangsdaten ein.'
|
||||||
PLEASESAVE: 'Diese Seite konnte nicht aktualisiert werden weil sie noch nicht gespeichert wurde - bitte speichern.'
|
|
||||||
PreviewButton: Vorschau
|
PreviewButton: Vorschau
|
||||||
REORGANISATIONSUCCESSFUL: 'Der Seitenbaum wurde erfolgreich sortiert.'
|
REORGANISATIONSUCCESSFUL: 'Der Seitenbaum wurde erfolgreich sortiert.'
|
||||||
SAVEDUP: Gespeichert.
|
SAVEDUP: Gespeichert.
|
||||||
|
@ -293,8 +293,6 @@ eo:
|
|||||||
FROMWEB: 'El la TTT'
|
FROMWEB: 'El la TTT'
|
||||||
FindInFolder: 'Serĉi en dosierujo'
|
FindInFolder: 'Serĉi en dosierujo'
|
||||||
IMAGEALT: 'Alternativa teksto (alt)'
|
IMAGEALT: 'Alternativa teksto (alt)'
|
||||||
IMAGEALTTEXT: 'Alternativa teksto (alt) - vidigi ĝin se ne eblas vidigi bildon'
|
|
||||||
IMAGEALTTEXTDESC: 'Vidigita al ekranlegiloj aŭ se ne eblas vidii bildon'
|
|
||||||
IMAGEDIMENSIONS: Dimensioj
|
IMAGEDIMENSIONS: Dimensioj
|
||||||
IMAGEHEIGHTPX: Alto
|
IMAGEHEIGHTPX: Alto
|
||||||
IMAGETITLE: 'Titola teksto (ŝpruchelpilo) - por plua informo pri la bildo'
|
IMAGETITLE: 'Titola teksto (ŝpruchelpilo) - por plua informo pri la bildo'
|
||||||
@ -326,11 +324,9 @@ eo:
|
|||||||
CANT_REORGANISE: 'Vi ne rajtas ŝanĝi supronivelajn paĝojn. Via ŝanĝo ne konserviĝis.'
|
CANT_REORGANISE: 'Vi ne rajtas ŝanĝi supronivelajn paĝojn. Via ŝanĝo ne konserviĝis.'
|
||||||
DELETED: Forigita.
|
DELETED: Forigita.
|
||||||
HELP: Helpo
|
HELP: Helpo
|
||||||
PAGETYPE: 'Tipo de paĝo:'
|
|
||||||
PERMAGAIN: 'Vin adiaŭis la CMS. Se vi volas denove saluti, enigu salutnomon kaj pasvorton malsupre.'
|
PERMAGAIN: 'Vin adiaŭis la CMS. Se vi volas denove saluti, enigu salutnomon kaj pasvorton malsupre.'
|
||||||
PERMALREADY: 'Bedaŭrinde vi ne povas aliri tiun parton de la CMS. Se vi volas ensaluti kiel aliulo, faru tion sube.'
|
PERMALREADY: 'Bedaŭrinde vi ne povas aliri tiun parton de la CMS. Se vi volas ensaluti kiel aliulo, faru tion sube.'
|
||||||
PERMDEFAULT: 'Necesas ensaluti por aliri la administran zonon; bonvolu enigi viajn akreditaĵoj sube.'
|
PERMDEFAULT: 'Necesas ensaluti por aliri la administran zonon; bonvolu enigi viajn akreditaĵoj sube.'
|
||||||
PLEASESAVE: 'Bonvolu konservi paĝon: Ne eblis ĝisdatigi ĉi tiun paĝon ĉar ĝi ankoraŭ ne estas konservita.'
|
|
||||||
PreviewButton: Antaŭvido
|
PreviewButton: Antaŭvido
|
||||||
REORGANISATIONSUCCESSFUL: 'Sukcese reorganizis la retejan arbon.'
|
REORGANISATIONSUCCESSFUL: 'Sukcese reorganizis la retejan arbon.'
|
||||||
SAVEDUP: Konservita.
|
SAVEDUP: Konservita.
|
||||||
|
@ -301,8 +301,6 @@ es:
|
|||||||
FROMWEB: 'Desde la web'
|
FROMWEB: 'Desde la web'
|
||||||
FindInFolder: 'Buscar en carpeta'
|
FindInFolder: 'Buscar en carpeta'
|
||||||
IMAGEALT: 'Texto alternativo (alt)'
|
IMAGEALT: 'Texto alternativo (alt)'
|
||||||
IMAGEALTTEXT: 'Texto alternativo (alt) - es mostrado si la imagen no puede ser visualizada'
|
|
||||||
IMAGEALTTEXTDESC: 'Mostrar a los lectores de pantalla o si la imagen no se puede visualizar'
|
|
||||||
IMAGEDIMENSIONS: Dimensiones
|
IMAGEDIMENSIONS: Dimensiones
|
||||||
IMAGEHEIGHTPX: Alto
|
IMAGEHEIGHTPX: Alto
|
||||||
IMAGETITLE: 'Texto del título (tooltip) - para obtener más información acerca de la imagen'
|
IMAGETITLE: 'Texto del título (tooltip) - para obtener más información acerca de la imagen'
|
||||||
@ -336,13 +334,10 @@ es:
|
|||||||
LeftAndMain:
|
LeftAndMain:
|
||||||
CANT_REORGANISE: 'Usted no tiene permiso para modificar las páginas de nivel superior. Su modificación no se ha guardado.'
|
CANT_REORGANISE: 'Usted no tiene permiso para modificar las páginas de nivel superior. Su modificación no se ha guardado.'
|
||||||
DELETED: Borrado
|
DELETED: Borrado
|
||||||
DropdownBatchActionsDefault: 'Elegir una acción...'
|
|
||||||
HELP: Ayuda
|
HELP: Ayuda
|
||||||
PAGETYPE: 'Tipo de página'
|
|
||||||
PERMAGAIN: 'Ha sido desconectado del CMS. Si quiere volver a entrar, introduzca su nombre de usuario y contraseña a continuación.'
|
PERMAGAIN: 'Ha sido desconectado del CMS. Si quiere volver a entrar, introduzca su nombre de usuario y contraseña a continuación.'
|
||||||
PERMALREADY: 'Lamentablemente no puede acceder a esta parte del CMS. Si quiere entrar como alguien distinto, hágalo a continuación'
|
PERMALREADY: 'Lamentablemente no puede acceder a esta parte del CMS. Si quiere entrar como alguien distinto, hágalo a continuación'
|
||||||
PERMDEFAULT: 'Debes estar conectado para acceder al área de administración; por favor ingresa tus datos a continuación'
|
PERMDEFAULT: 'Debes estar conectado para acceder al área de administración; por favor ingresa tus datos a continuación'
|
||||||
PLEASESAVE: 'Por favor Guarde la Página: Esta página no se ha podido actualizar porque aún no ha sido salvada.'
|
|
||||||
PreviewButton: Vista previa
|
PreviewButton: Vista previa
|
||||||
REORGANISATIONSUCCESSFUL: 'Reorganizado el árbol del sitio con éxito.'
|
REORGANISATIONSUCCESSFUL: 'Reorganizado el árbol del sitio con éxito.'
|
||||||
SAVEDUP: Guardado
|
SAVEDUP: Guardado
|
||||||
|
@ -318,9 +318,7 @@ fa_IR:
|
|||||||
TITLE: 'آیفریم آپلود عکس'
|
TITLE: 'آیفریم آپلود عکس'
|
||||||
LeftAndMain:
|
LeftAndMain:
|
||||||
DELETED: حذف شده
|
DELETED: حذف شده
|
||||||
DropdownBatchActionsDefault: 'انتخاب یک عملیات...'
|
|
||||||
HELP: کمک
|
HELP: کمک
|
||||||
PAGETYPE: 'نوع صفحه'
|
|
||||||
PERMAGAIN: 'شما از سیستم مدیریت محتوا خارج شده اید.اگر میخواهید دوباره وارد شوید نام کاربری و رمز عبور خود را در قسمت زیر وارد کنید'
|
PERMAGAIN: 'شما از سیستم مدیریت محتوا خارج شده اید.اگر میخواهید دوباره وارد شوید نام کاربری و رمز عبور خود را در قسمت زیر وارد کنید'
|
||||||
PreviewButton: پیشنمایش
|
PreviewButton: پیشنمایش
|
||||||
SAVEDUP: ذخیره شده
|
SAVEDUP: ذخیره شده
|
||||||
|
@ -299,8 +299,6 @@ fi:
|
|||||||
FROMWEB: 'Webistä'
|
FROMWEB: 'Webistä'
|
||||||
FindInFolder: 'Etsi kansiosta'
|
FindInFolder: 'Etsi kansiosta'
|
||||||
IMAGEALT: 'Vaihtoehtoinen teksti (alt)'
|
IMAGEALT: 'Vaihtoehtoinen teksti (alt)'
|
||||||
IMAGEALTTEXT: 'Vaihtoehtoinen teksti (alt) - näytetään jos kuvaa ei voida näyttää'
|
|
||||||
IMAGEALTTEXTDESC: 'Näytetään ruudunlukuohjelmille tai jos kuvaa ei voi näyttää'
|
|
||||||
IMAGEDIMENSIONS: Mitat
|
IMAGEDIMENSIONS: Mitat
|
||||||
IMAGEHEIGHTPX: Korkeus
|
IMAGEHEIGHTPX: Korkeus
|
||||||
IMAGETITLE: 'Otsikko (tooltip) - kuvan lisätietoja varten'
|
IMAGETITLE: 'Otsikko (tooltip) - kuvan lisätietoja varten'
|
||||||
@ -333,11 +331,9 @@ fi:
|
|||||||
CANT_REORGANISE: 'Sinulla ei ole oikeuksia mennä ylemmän tason sivuille. Muutoksiasi ei tallennettu.'
|
CANT_REORGANISE: 'Sinulla ei ole oikeuksia mennä ylemmän tason sivuille. Muutoksiasi ei tallennettu.'
|
||||||
DELETED: Poistettu.
|
DELETED: Poistettu.
|
||||||
HELP: Ohje
|
HELP: Ohje
|
||||||
PAGETYPE: 'Sivutyyppi'
|
|
||||||
PERMAGAIN: 'Olet kirjautunut ulos CMS:stä. Jos haluat kirjautua uudelleen sisään, syötä käyttäjätunnuksesi ja salasanasi alla.'
|
PERMAGAIN: 'Olet kirjautunut ulos CMS:stä. Jos haluat kirjautua uudelleen sisään, syötä käyttäjätunnuksesi ja salasanasi alla.'
|
||||||
PERMALREADY: 'Pahoittelut, mutta et pääse tähän osaan CMS:ää. Jos haluat kirjautua jonain muuna, voit tehdä sen alta.'
|
PERMALREADY: 'Pahoittelut, mutta et pääse tähän osaan CMS:ää. Jos haluat kirjautua jonain muuna, voit tehdä sen alta.'
|
||||||
PERMDEFAULT: 'Sinun tulee olla kirjautuneena ylläpito-osioon; syötä tunnuksesi kenttiin.'
|
PERMDEFAULT: 'Sinun tulee olla kirjautuneena ylläpito-osioon; syötä tunnuksesi kenttiin.'
|
||||||
PLEASESAVE: 'Tätä sivua ei voitu päivittää, koska sitä ei ole vielä tallennettu. Tallenna sivu.'
|
|
||||||
PreviewButton: Esikatselu
|
PreviewButton: Esikatselu
|
||||||
REORGANISATIONSUCCESSFUL: 'Hakemistopuu uudelleenjärjestettiin onnistuneesti.'
|
REORGANISATIONSUCCESSFUL: 'Hakemistopuu uudelleenjärjestettiin onnistuneesti.'
|
||||||
SAVEDUP: Tallennettu.
|
SAVEDUP: Tallennettu.
|
||||||
|
@ -301,8 +301,6 @@ lt:
|
|||||||
FROMWEB: 'Iš interneto'
|
FROMWEB: 'Iš interneto'
|
||||||
FindInFolder: 'Rasti kataloge'
|
FindInFolder: 'Rasti kataloge'
|
||||||
IMAGEALT: 'Alternatyvus tekstas (alt)'
|
IMAGEALT: 'Alternatyvus tekstas (alt)'
|
||||||
IMAGEALTTEXT: 'Alternatyvus tekstas (alt) - rodomas, jeigu nepavyko parodyti paveikslėlio'
|
|
||||||
IMAGEALTTEXTDESC: 'Rodomas, jeigu nepavyko parodyti paveikslėlio'
|
|
||||||
IMAGEDIMENSIONS: Matmenys
|
IMAGEDIMENSIONS: Matmenys
|
||||||
IMAGEHEIGHTPX: Aukštis
|
IMAGEHEIGHTPX: Aukštis
|
||||||
IMAGETITLE: 'Pavadinimo tekstas (tooltip) - papildomai informacijai apie paveikslėlį'
|
IMAGETITLE: 'Pavadinimo tekstas (tooltip) - papildomai informacijai apie paveikslėlį'
|
||||||
@ -336,13 +334,12 @@ lt:
|
|||||||
LeftAndMain:
|
LeftAndMain:
|
||||||
CANT_REORGANISE: 'Jūs neturite leidimo keisti aukščiausio lygio puslapių. Jūsų pakeitimai neišsaugoti.'
|
CANT_REORGANISE: 'Jūs neturite leidimo keisti aukščiausio lygio puslapių. Jūsų pakeitimai neišsaugoti.'
|
||||||
DELETED: Ištrinta.
|
DELETED: Ištrinta.
|
||||||
DropdownBatchActionsDefault: 'Pasirinkite veiksmą...'
|
DropdownBatchActionsDefault: Veiksmai
|
||||||
HELP: Pagalba
|
HELP: Pagalba
|
||||||
PAGETYPE: 'Puslapio tipas'
|
PAGETYPE: 'Puslapio tipas'
|
||||||
PERMAGAIN: 'Jūs atsijungėte. Norėdami vėl prisijungti, įveskite savo duomenis į žemiau esančius laukelius.'
|
PERMAGAIN: 'Jūs atsijungėte. Norėdami vėl prisijungti, įveskite savo duomenis į žemiau esančius laukelius.'
|
||||||
PERMALREADY: 'Deja, bet Jūs negalite patekti į šią TVS dalį. Jeigu norite prisijungti kitu vartotoju, tai atlikite žemiau.'
|
PERMALREADY: 'Deja, bet Jūs negalite patekti į šią TVS dalį. Jeigu norite prisijungti kitu vartotoju, tai atlikite žemiau.'
|
||||||
PERMDEFAULT: 'Jūs turite būti prisijungę, norėdami pasiekti administravimo zoną; prašome suvesti prisijungimo duomenis į žemiau esančius laukelius.'
|
PERMDEFAULT: 'Jūs turite būti prisijungę, norėdami pasiekti administravimo zoną; prašome suvesti prisijungimo duomenis į žemiau esančius laukelius.'
|
||||||
PLEASESAVE: 'Prašome išsaugoti puslapį: Šis puslapis negali būti atnaujintas, nes jis dar nėra išsaugotas.'
|
|
||||||
PreviewButton: Peržiūra
|
PreviewButton: Peržiūra
|
||||||
REORGANISATIONSUCCESSFUL: 'Puslapių medis pertvarkytas sėkmingai.'
|
REORGANISATIONSUCCESSFUL: 'Puslapių medis pertvarkytas sėkmingai.'
|
||||||
SAVEDUP: Išsaugota.
|
SAVEDUP: Išsaugota.
|
||||||
|
10
lang/sk.yml
10
lang/sk.yml
@ -301,8 +301,8 @@ sk:
|
|||||||
FROMWEB: 'Z webu'
|
FROMWEB: 'Z webu'
|
||||||
FindInFolder: 'Vyhľadať v priečinku'
|
FindInFolder: 'Vyhľadať v priečinku'
|
||||||
IMAGEALT: 'Atlernatívny text (alt)'
|
IMAGEALT: 'Atlernatívny text (alt)'
|
||||||
IMAGEALTTEXT: 'Atlernatívny text (alt) - zobrazí sa, ak obrázok nemože byť zobrazený'
|
IMAGEALTTEXT: 'Atlernatívny text (alt) - sa zobrazí, ak nemôže byť zobrazený obrázok'
|
||||||
IMAGEALTTEXTDESC: 'Zobrazí sa na čítačke obrazovky alebo ak obrázok nemôže byť zobrazený'
|
IMAGEALTTEXTDESC: 'Zobrazí sa na obrazovke, keď obrázok nemôže byť zobrazený'
|
||||||
IMAGEDIMENSIONS: Rozmery
|
IMAGEDIMENSIONS: Rozmery
|
||||||
IMAGEHEIGHTPX: Výška
|
IMAGEHEIGHTPX: Výška
|
||||||
IMAGETITLE: 'Text titulky (tooltip) - pre doplňujúce informácie o obrázku'
|
IMAGETITLE: 'Text titulky (tooltip) - pre doplňujúce informácie o obrázku'
|
||||||
@ -336,13 +336,13 @@ sk:
|
|||||||
LeftAndMain:
|
LeftAndMain:
|
||||||
CANT_REORGANISE: 'Nemáte oprávnenie meniť stránky najvyššej úrovne. Vaša zmena nebola uložená.'
|
CANT_REORGANISE: 'Nemáte oprávnenie meniť stránky najvyššej úrovne. Vaša zmena nebola uložená.'
|
||||||
DELETED: Zmazané.
|
DELETED: Zmazané.
|
||||||
DropdownBatchActionsDefault: 'Vybrať akciu...'
|
DropdownBatchActionsDefault: Akcie
|
||||||
HELP: Pomoc
|
HELP: Pomoc
|
||||||
PAGETYPE: 'Typ stránky'
|
PAGETYPE: 'Typ stránky:'
|
||||||
PERMAGAIN: 'Boli ste odhlásený'
|
PERMAGAIN: 'Boli ste odhlásený'
|
||||||
PERMALREADY: 'Je nám ľúto, ale k tejto časti CMS nemáte prístup . Ak sa chcete prihlásiť ako niekto iný, urobte tak nižšie.'
|
PERMALREADY: 'Je nám ľúto, ale k tejto časti CMS nemáte prístup . Ak sa chcete prihlásiť ako niekto iný, urobte tak nižšie.'
|
||||||
PERMDEFAULT: 'Musíte byť prihlásený/á k prístupu do oblasti administrácie, zadajte vaše prihlasovacie údaje dole, prosím.'
|
PERMDEFAULT: 'Musíte byť prihlásený/á k prístupu do oblasti administrácie, zadajte vaše prihlasovacie údaje dole, prosím.'
|
||||||
PLEASESAVE: 'Uložte stránku, prosím. Táto stránka nemôže byť aktualizovaná, pretože ešte nebola uložená.'
|
PLEASESAVE: 'Prosím uložte stránku: Táto stránka nemôže byť aktualizovaná, ešte nebola uložená.'
|
||||||
PreviewButton: Náhľad
|
PreviewButton: Náhľad
|
||||||
REORGANISATIONSUCCESSFUL: 'Strom webu bol reorganizovaný úspešne.'
|
REORGANISATIONSUCCESSFUL: 'Strom webu bol reorganizovaný úspešne.'
|
||||||
SAVEDUP: Uložené.
|
SAVEDUP: Uložené.
|
||||||
|
@ -293,8 +293,6 @@ sv:
|
|||||||
FROMWEB: 'Från webben'
|
FROMWEB: 'Från webben'
|
||||||
FindInFolder: 'Hitta i mapp'
|
FindInFolder: 'Hitta i mapp'
|
||||||
IMAGEALT: 'Alternativ text (alt)'
|
IMAGEALT: 'Alternativ text (alt)'
|
||||||
IMAGEALTTEXT: 'Alternativ text (alt) - visas om bilden inte kan visas'
|
|
||||||
IMAGEALTTEXTDESC: 'Visas för skärmläsare eller om bilden inte kan visas'
|
|
||||||
IMAGEDIMENSIONS: Dimensioner
|
IMAGEDIMENSIONS: Dimensioner
|
||||||
IMAGEHEIGHTPX: Höjd
|
IMAGEHEIGHTPX: Höjd
|
||||||
IMAGETITLE: 'Titel text (tooltip) - för ytterligare information om bilden'
|
IMAGETITLE: 'Titel text (tooltip) - för ytterligare information om bilden'
|
||||||
@ -326,11 +324,9 @@ sv:
|
|||||||
CANT_REORGANISE: 'Du har inte tillstånd att ändra sidor på toppnivå. Dina ändringar har inte sparats.'
|
CANT_REORGANISE: 'Du har inte tillstånd att ändra sidor på toppnivå. Dina ändringar har inte sparats.'
|
||||||
DELETED: Raderad
|
DELETED: Raderad
|
||||||
HELP: Hjälp
|
HELP: Hjälp
|
||||||
PAGETYPE: 'Sidtyp'
|
|
||||||
PERMAGAIN: 'Du har blivit utloggad. Om du vill logga in igen anger du dina uppgifter nedan.'
|
PERMAGAIN: 'Du har blivit utloggad. Om du vill logga in igen anger du dina uppgifter nedan.'
|
||||||
PERMALREADY: 'Tyvärr så har du inte åtkomst till den delen av CMSet. Om du vill logga in med en annan användare kan du göra det nedan'
|
PERMALREADY: 'Tyvärr så har du inte åtkomst till den delen av CMSet. Om du vill logga in med en annan användare kan du göra det nedan'
|
||||||
PERMDEFAULT: 'Du måste vara inloggad för att få åtkomst till administrativa delarna; var vänlig att logga in med dina användaruppgifter nedan.'
|
PERMDEFAULT: 'Du måste vara inloggad för att få åtkomst till administrativa delarna; var vänlig att logga in med dina användaruppgifter nedan.'
|
||||||
PLEASESAVE: 'Var god spara sidan. Den kan inte uppdateras för att den har inte sparats ännu.'
|
|
||||||
PreviewButton: Förhandsgranska
|
PreviewButton: Förhandsgranska
|
||||||
REORGANISATIONSUCCESSFUL: 'Omorganisationen av sidträdet luyckades.'
|
REORGANISATIONSUCCESSFUL: 'Omorganisationen av sidträdet luyckades.'
|
||||||
SAVEDUP: Sparad.
|
SAVEDUP: Sparad.
|
||||||
|
@ -916,7 +916,9 @@ class DataList extends ViewableData implements SS_List, SS_Filterable, SS_Sortab
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns an array with both the keys and values set to the IDs of the records in this list.
|
* Returns an array with both the keys and values set to the IDs of the records in this list.
|
||||||
|
* Does not respect sort order. Use ->column("ID") to get an ID list with the current sort.
|
||||||
*
|
*
|
||||||
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function getIDList() {
|
public function getIDList() {
|
||||||
$ids = $this->column("ID");
|
$ids = $this->column("ID");
|
||||||
|
@ -2077,7 +2077,7 @@ class DataObject extends ViewableData implements DataObjectInterface, i18nEntity
|
|||||||
$candidateManyManys = (array)Config::inst()->get($candidate, 'many_many', Config::UNINHERITED);
|
$candidateManyManys = (array)Config::inst()->get($candidate, 'many_many', Config::UNINHERITED);
|
||||||
|
|
||||||
foreach($candidateManyManys as $relation => $relatedClass) {
|
foreach($candidateManyManys as $relation => $relatedClass) {
|
||||||
if($relatedClass === $this->class) {
|
if (is_a($this, $relatedClass)) {
|
||||||
$relationName = $relation;
|
$relationName = $relation;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -206,8 +206,10 @@ class UnsavedRelationList extends ArrayList {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns an array with both the keys and values set to the IDs of the records in this list.
|
* Returns an array with both the keys and values set to the IDs of the records in this list.
|
||||||
|
* Does not respect sort order. Use ->column("ID") to get an ID list with the current sort.
|
||||||
|
* Does not return the IDs for unsaved DataObjects.
|
||||||
*
|
*
|
||||||
* Does not return the IDs for unsaved DataObjects
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function getIDList() {
|
public function getIDList() {
|
||||||
// Get a list of IDs of our current items - if it's not a number then object then assume it's a DO.
|
// Get a list of IDs of our current items - if it's not a number then object then assume it's a DO.
|
||||||
|
@ -5,6 +5,8 @@
|
|||||||
* allowing you to rollback changes and view history. An example of this is
|
* allowing you to rollback changes and view history. An example of this is
|
||||||
* the pages used in the CMS.
|
* the pages used in the CMS.
|
||||||
*
|
*
|
||||||
|
* @property int $Version
|
||||||
|
*
|
||||||
* @package framework
|
* @package framework
|
||||||
* @subpackage model
|
* @subpackage model
|
||||||
*/
|
*/
|
||||||
@ -637,15 +639,17 @@ class Versioned extends DataExtension implements TemplateGlobalProvider {
|
|||||||
}
|
}
|
||||||
$nextVersion = $nextVersion ?: 1;
|
$nextVersion = $nextVersion ?: 1;
|
||||||
|
|
||||||
// Add the version number to this data
|
|
||||||
$manipulation[$table]['fields']['Version'] = $nextVersion;
|
|
||||||
$newManipulation['fields']['Version'] = $nextVersion;
|
|
||||||
|
|
||||||
// Write AuthorID for baseclass
|
|
||||||
if($table === $baseDataClass) {
|
if($table === $baseDataClass) {
|
||||||
|
// Write AuthorID for baseclass
|
||||||
$userID = (Member::currentUser()) ? Member::currentUser()->ID : 0;
|
$userID = (Member::currentUser()) ? Member::currentUser()->ID : 0;
|
||||||
$newManipulation['fields']['AuthorID'] = $userID;
|
$newManipulation['fields']['AuthorID'] = $userID;
|
||||||
|
|
||||||
|
// Update main table version if not previously known
|
||||||
|
$manipulation[$table]['fields']['Version'] = $nextVersion;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Update _versions table manipulation
|
||||||
|
$newManipulation['fields']['Version'] = $nextVersion;
|
||||||
$manipulation["{$table}_versions"] = $newManipulation;
|
$manipulation["{$table}_versions"] = $newManipulation;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -672,6 +676,19 @@ class Versioned extends DataExtension implements TemplateGlobalProvider {
|
|||||||
|
|
||||||
|
|
||||||
public function augmentWrite(&$manipulation) {
|
public function augmentWrite(&$manipulation) {
|
||||||
|
// get Version number from base data table on write
|
||||||
|
$version = null;
|
||||||
|
$baseDataClass = ClassInfo::baseDataClass($this->owner->class);
|
||||||
|
if(isset($manipulation[$baseDataClass]['fields'])) {
|
||||||
|
if ($this->migratingVersion) {
|
||||||
|
$manipulation[$baseDataClass]['fields']['Version'] = $this->migratingVersion;
|
||||||
|
}
|
||||||
|
if (isset($manipulation[$baseDataClass]['fields']['Version'])) {
|
||||||
|
$version = $manipulation[$baseDataClass]['fields']['Version'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update all tables
|
||||||
$tables = array_keys($manipulation);
|
$tables = array_keys($manipulation);
|
||||||
foreach($tables as $table) {
|
foreach($tables as $table) {
|
||||||
|
|
||||||
@ -682,18 +699,13 @@ class Versioned extends DataExtension implements TemplateGlobalProvider {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Get ID field
|
// Get ID field
|
||||||
$id = $manipulation[$table]['id'] ? $manipulation[$table]['id'] : $manipulation[$table]['fields']['ID'];
|
$id = $manipulation[$table]['id']
|
||||||
|
? $manipulation[$table]['id']
|
||||||
|
: $manipulation[$table]['fields']['ID'];
|
||||||
if(!$id) {
|
if(!$id) {
|
||||||
user_error("Couldn't find ID in " . var_export($manipulation[$table], true), E_USER_ERROR);
|
user_error("Couldn't find ID in " . var_export($manipulation[$table], true), E_USER_ERROR);
|
||||||
}
|
}
|
||||||
|
|
||||||
if($this->migratingVersion) {
|
|
||||||
$manipulation[$table]['fields']['Version'] = $this->migratingVersion;
|
|
||||||
}
|
|
||||||
|
|
||||||
$version = isset($manipulation[$table]['fields']['Version'])
|
|
||||||
? $manipulation[$table]['fields']['Version']
|
|
||||||
: null;
|
|
||||||
if($version < 0 || $this->_nextWriteWithoutVersion) {
|
if($version < 0 || $this->_nextWriteWithoutVersion) {
|
||||||
// Putting a Version of -1 is a signal to leave the version table alone, despite their being no version
|
// Putting a Version of -1 is a signal to leave the version table alone, despite their being no version
|
||||||
unset($manipulation[$table]['fields']['Version']);
|
unset($manipulation[$table]['fields']['Version']);
|
||||||
@ -703,7 +715,7 @@ class Versioned extends DataExtension implements TemplateGlobalProvider {
|
|||||||
$this->augmentWriteVersioned($manipulation, $table, $id);
|
$this->augmentWriteVersioned($manipulation, $table, $id);
|
||||||
}
|
}
|
||||||
|
|
||||||
// For base classes of versioned data objects
|
// Remove "Version" column from subclasses of baseDataClass
|
||||||
if(!$this->hasVersionField($table)) {
|
if(!$this->hasVersionField($table)) {
|
||||||
unset($manipulation[$table]['fields']['Version']);
|
unset($manipulation[$table]['fields']['Version']);
|
||||||
}
|
}
|
||||||
@ -925,56 +937,64 @@ class Versioned extends DataExtension implements TemplateGlobalProvider {
|
|||||||
/**
|
/**
|
||||||
* Move a database record from one stage to the other.
|
* Move a database record from one stage to the other.
|
||||||
*
|
*
|
||||||
* @param fromStage Place to copy from. Can be either a stage name or a version number.
|
* @param int|string $fromStage Place to copy from. Can be either a stage name or a version number.
|
||||||
* @param toStage Place to copy to. Must be a stage name.
|
* @param string $toStage Place to copy to. Must be a stage name.
|
||||||
* @param createNewVersion Set this to true to create a new version number. By default, the existing version
|
* @param bool $createNewVersion Set this to true to create a new version number.
|
||||||
* number will be copied over.
|
* By default, the existing version number will be copied over.
|
||||||
*/
|
*/
|
||||||
public function publish($fromStage, $toStage, $createNewVersion = false) {
|
public function publish($fromStage, $toStage, $createNewVersion = false) {
|
||||||
$this->owner->extend('onBeforeVersionedPublish', $fromStage, $toStage, $createNewVersion);
|
$this->owner->extend('onBeforeVersionedPublish', $fromStage, $toStage, $createNewVersion);
|
||||||
|
|
||||||
$baseClass = $this->owner->class;
|
$baseClass = ClassInfo::baseDataClass($this->owner->class);
|
||||||
while( ($p = get_parent_class($baseClass)) != "DataObject") $baseClass = $p;
|
|
||||||
$extTable = $this->extendWithSuffix($baseClass);
|
$extTable = $this->extendWithSuffix($baseClass);
|
||||||
|
|
||||||
|
/** @var Versioned|DataObject $from */
|
||||||
if(is_numeric($fromStage)) {
|
if(is_numeric($fromStage)) {
|
||||||
$from = Versioned::get_version($baseClass, $this->owner->ID, $fromStage);
|
$from = Versioned::get_version($baseClass, $this->owner->ID, $fromStage);
|
||||||
} else {
|
} else {
|
||||||
$this->owner->flushCache();
|
$this->owner->flushCache();
|
||||||
$from = Versioned::get_one_by_stage($baseClass, $fromStage, "\"{$baseClass}\".\"ID\"={$this->owner->ID}");
|
$from = Versioned::get_one_by_stage($baseClass, $fromStage, array(
|
||||||
|
"\"{$baseClass}\".\"ID\" = ?" => $this->owner->ID
|
||||||
|
));
|
||||||
|
}
|
||||||
|
if(!$from) {
|
||||||
|
user_error("Can't find {$this->owner->class}/{$this->owner->ID} in stage {$fromStage}", E_USER_WARNING);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
$publisherID = isset(Member::currentUser()->ID) ? Member::currentUser()->ID : 0;
|
// Set version of new record
|
||||||
if($from) {
|
|
||||||
$from->forceChange();
|
$from->forceChange();
|
||||||
if($createNewVersion) {
|
if($createNewVersion) {
|
||||||
$latest = self::get_latest_version($baseClass, $this->owner->ID);
|
// Clear version to be automatically created on write
|
||||||
$this->owner->Version = $latest->Version + 1;
|
$from->Version = null;
|
||||||
} else {
|
} else {
|
||||||
$from->migrateVersion($from->Version);
|
$from->migrateVersion($from->Version);
|
||||||
}
|
|
||||||
|
|
||||||
// Mark this version as having been published at some stage
|
// Mark this version as having been published at some stage
|
||||||
|
$publisherID = isset(Member::currentUser()->ID) ? Member::currentUser()->ID : 0;
|
||||||
DB::prepared_query("UPDATE \"{$extTable}_versions\"
|
DB::prepared_query("UPDATE \"{$extTable}_versions\"
|
||||||
SET \"WasPublished\" = ?, \"PublisherID\" = ?
|
SET \"WasPublished\" = ?, \"PublisherID\" = ?
|
||||||
WHERE \"RecordID\" = ? AND \"Version\" = ?",
|
WHERE \"RecordID\" = ? AND \"Version\" = ?",
|
||||||
array(1, $publisherID, $from->ID, $from->Version)
|
array(1, $publisherID, $from->ID, $from->Version)
|
||||||
);
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Change to new stage, write, and revert state
|
||||||
$oldMode = Versioned::get_reading_mode();
|
$oldMode = Versioned::get_reading_mode();
|
||||||
Versioned::reading_stage($toStage);
|
Versioned::reading_stage($toStage);
|
||||||
|
|
||||||
$conn = DB::get_conn();
|
$conn = DB::get_conn();
|
||||||
if(method_exists($conn, 'allowPrimaryKeyEditing')) $conn->allowPrimaryKeyEditing($baseClass, true);
|
if(method_exists($conn, 'allowPrimaryKeyEditing')) {
|
||||||
|
$conn->allowPrimaryKeyEditing($baseClass, true);
|
||||||
$from->write();
|
$from->write();
|
||||||
if(method_exists($conn, 'allowPrimaryKeyEditing')) $conn->allowPrimaryKeyEditing($baseClass, false);
|
$conn->allowPrimaryKeyEditing($baseClass, false);
|
||||||
|
} else {
|
||||||
|
$from->write();
|
||||||
|
}
|
||||||
|
|
||||||
$from->destroy();
|
$from->destroy();
|
||||||
|
|
||||||
Versioned::set_reading_mode($oldMode);
|
Versioned::set_reading_mode($oldMode);
|
||||||
} else {
|
|
||||||
user_error("Can't find {$this->owner->URLSegment}/{$this->owner->ID} in stage $fromStage", E_USER_WARNING);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -42,15 +42,54 @@ class GridFieldDeleteActionTest extends SapphireTest {
|
|||||||
$this->assertEquals(3, count($deleteButtons), 'Delete buttons should show when logged in.');
|
$this->assertEquals(3, count($deleteButtons), 'Delete buttons should show when logged in.');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testActionsRequireCSRF() {
|
||||||
|
$this->logInWithPermission('ADMIN');
|
||||||
|
$this->setExpectedException(
|
||||||
|
'SS_HTTPResponse_Exception',
|
||||||
|
_t("Form.CSRF_FAILED_MESSAGE",
|
||||||
|
"There seems to have been a technical problem. Please click the back button, ".
|
||||||
|
"refresh your browser, and try again."
|
||||||
|
),
|
||||||
|
400
|
||||||
|
);
|
||||||
|
$stateID = 'testGridStateActionField';
|
||||||
|
$request = new SS_HTTPRequest(
|
||||||
|
'POST',
|
||||||
|
'url',
|
||||||
|
array(),
|
||||||
|
array(
|
||||||
|
'action_gridFieldAlterAction?StateID='.$stateID,
|
||||||
|
'SecurityID' => null,
|
||||||
|
)
|
||||||
|
);
|
||||||
|
$this->gridField->gridFieldAlterAction(array('StateID'=>$stateID), $this->form, $request);
|
||||||
|
}
|
||||||
|
|
||||||
public function testDeleteActionWithoutCorrectPermission() {
|
public function testDeleteActionWithoutCorrectPermission() {
|
||||||
if(Member::currentUser()) { Member::currentUser()->logOut(); }
|
if(Member::currentUser()) { Member::currentUser()->logOut(); }
|
||||||
$this->setExpectedException('ValidationException');
|
$this->setExpectedException('ValidationException');
|
||||||
|
|
||||||
$stateID = 'testGridStateActionField';
|
$stateID = 'testGridStateActionField';
|
||||||
Session::set($stateID, array('grid'=>'', 'actionName'=>'deleterecord',
|
Session::set(
|
||||||
'args'=>array('RecordID'=>$this->idFromFixture('GridFieldAction_Delete_Team', 'team1'))));
|
$stateID,
|
||||||
$request = new SS_HTTPRequest('POST', 'url', array(),
|
array(
|
||||||
array('action_gridFieldAlterAction?StateID='.$stateID=>true));
|
'grid' => '',
|
||||||
|
'actionName' => 'deleterecord',
|
||||||
|
'args' => array(
|
||||||
|
'RecordID' => $this->idFromFixture('GridFieldAction_Delete_Team', 'team1')
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
$token = SecurityToken::inst();
|
||||||
|
$request = new SS_HTTPRequest(
|
||||||
|
'POST',
|
||||||
|
'url',
|
||||||
|
array(),
|
||||||
|
array(
|
||||||
|
'action_gridFieldAlterAction?StateID='.$stateID => true,
|
||||||
|
$token->getName() => $token->getValue(),
|
||||||
|
)
|
||||||
|
);
|
||||||
$this->gridField->gridFieldAlterAction(array('StateID'=>$stateID), $this->form, $request);
|
$this->gridField->gridFieldAlterAction(array('StateID'=>$stateID), $this->form, $request);
|
||||||
$this->assertEquals(3, $this->list->count(),
|
$this->assertEquals(3, $this->list->count(),
|
||||||
'User should\'t be able to delete records without correct permissions.');
|
'User should\'t be able to delete records without correct permissions.');
|
||||||
@ -59,10 +98,26 @@ class GridFieldDeleteActionTest extends SapphireTest {
|
|||||||
public function testDeleteActionWithAdminPermission() {
|
public function testDeleteActionWithAdminPermission() {
|
||||||
$this->logInWithPermission('ADMIN');
|
$this->logInWithPermission('ADMIN');
|
||||||
$stateID = 'testGridStateActionField';
|
$stateID = 'testGridStateActionField';
|
||||||
Session::set($stateID, array('grid'=>'', 'actionName'=>'deleterecord',
|
Session::set(
|
||||||
'args'=>array('RecordID'=>$this->idFromFixture('GridFieldAction_Delete_Team', 'team1'))));
|
$stateID,
|
||||||
$request = new SS_HTTPRequest('POST', 'url', array(),
|
array(
|
||||||
array('action_gridFieldAlterAction?StateID='.$stateID=>true));
|
'grid'=>'',
|
||||||
|
'actionName'=>'deleterecord',
|
||||||
|
'args' => array(
|
||||||
|
'RecordID' => $this->idFromFixture('GridFieldAction_Delete_Team', 'team1')
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
$token = SecurityToken::inst();
|
||||||
|
$request = new SS_HTTPRequest(
|
||||||
|
'POST',
|
||||||
|
'url',
|
||||||
|
array(),
|
||||||
|
array(
|
||||||
|
'action_gridFieldAlterAction?StateID='.$stateID=>true,
|
||||||
|
$token->getName() => $token->getValue(),
|
||||||
|
)
|
||||||
|
);
|
||||||
$this->gridField->gridFieldAlterAction(array('StateID'=>$stateID), $this->form, $request);
|
$this->gridField->gridFieldAlterAction(array('StateID'=>$stateID), $this->form, $request);
|
||||||
$this->assertEquals(2, $this->list->count(), 'User should be able to delete records with ADMIN permission.');
|
$this->assertEquals(2, $this->list->count(), 'User should be able to delete records with ADMIN permission.');
|
||||||
}
|
}
|
||||||
@ -76,11 +131,26 @@ class GridFieldDeleteActionTest extends SapphireTest {
|
|||||||
$form = new Form(new Controller(), 'mockform', new FieldList(array($this->gridField)), new FieldList());
|
$form = new Form(new Controller(), 'mockform', new FieldList(array($this->gridField)), new FieldList());
|
||||||
|
|
||||||
$stateID = 'testGridStateActionField';
|
$stateID = 'testGridStateActionField';
|
||||||
Session::set($stateID, array('grid'=>'', 'actionName'=>'deleterecord',
|
Session::set(
|
||||||
'args'=>array('RecordID'=>$this->idFromFixture('GridFieldAction_Delete_Team', 'team1'))));
|
$stateID,
|
||||||
$request = new SS_HTTPRequest('POST', 'url', array(),
|
array(
|
||||||
array('action_gridFieldAlterAction?StateID='.$stateID=>true));
|
'grid'=>'',
|
||||||
|
'actionName'=>'deleterecord',
|
||||||
|
'args' => array(
|
||||||
|
'RecordID' => $this->idFromFixture('GridFieldAction_Delete_Team', 'team1')
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
$token = SecurityToken::inst();
|
||||||
|
$request = new SS_HTTPRequest(
|
||||||
|
'POST',
|
||||||
|
'url',
|
||||||
|
array(),
|
||||||
|
array(
|
||||||
|
'action_gridFieldAlterAction?StateID='.$stateID=>true,
|
||||||
|
$token->getName() => $token->getValue(),
|
||||||
|
)
|
||||||
|
);
|
||||||
$this->gridField->gridFieldAlterAction(array('StateID'=>$stateID), $this->form, $request);
|
$this->gridField->gridFieldAlterAction(array('StateID'=>$stateID), $this->form, $request);
|
||||||
$this->assertEquals(2, $this->list->count(), 'User should be able to delete records with ADMIN permission.');
|
$this->assertEquals(2, $this->list->count(), 'User should be able to delete records with ADMIN permission.');
|
||||||
|
|
||||||
|
119
tests/model/ManyManyListExtensionTest.php
Normal file
119
tests/model/ManyManyListExtensionTest.php
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @package framework
|
||||||
|
* @subpackage tests
|
||||||
|
*/
|
||||||
|
class ManyManyListExtensionTest extends SapphireTest {
|
||||||
|
|
||||||
|
protected static $fixture_file = 'ManyManyListExtensionTest.yml';
|
||||||
|
|
||||||
|
protected $extraDataObjects = array(
|
||||||
|
'ManyManyListTest_IndirectPrimary',
|
||||||
|
'ManyManyListTest_Secondary',
|
||||||
|
'ManyManyListTest_SecondarySub'
|
||||||
|
);
|
||||||
|
|
||||||
|
// Test that when one side of a many-many relationship is added by extension, both
|
||||||
|
// sides still see the extra fields.
|
||||||
|
public function testExtraFieldsViaExtension() {
|
||||||
|
// This extends ManyManyListTest_Secondary with the secondary extension that adds the relationship back
|
||||||
|
// to the primary. The instance from the fixture is ManyManyListTest_SecondarySub, deliberately a sub-class of
|
||||||
|
// the extended class.
|
||||||
|
Object::add_extension('ManyManyListTest_Secondary', 'ManyManyListTest_IndirectSecondaryExtension');
|
||||||
|
|
||||||
|
// Test from the primary (not extended) to the secondary (which is extended)
|
||||||
|
$primary = $this->objFromFixture('ManyManyListTest_IndirectPrimary', 'manymany_extra_primary');
|
||||||
|
$secondaries = $primary->Secondary();
|
||||||
|
$extraFields = $secondaries->getExtraFields();
|
||||||
|
|
||||||
|
$this->assertTrue(count($extraFields) > 0, 'has extra fields');
|
||||||
|
$this->assertTrue(isset($extraFields['DocumentSort']), 'has DocumentSort');
|
||||||
|
|
||||||
|
// Test from the secondary (which is extended) to the primary (not extended)
|
||||||
|
$secondary = $this->objFromFixture('ManyManyListTest_SecondarySub', 'manymany_extra_secondary');
|
||||||
|
|
||||||
|
$primaries = $secondary->Primary();
|
||||||
|
$extraFields = $primaries->getExtraFields();
|
||||||
|
|
||||||
|
$this->assertTrue(count($extraFields) > 0, 'has extra fields');
|
||||||
|
$this->assertTrue(isset($extraFields['DocumentSort']), 'has DocumentSort');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @package framework
|
||||||
|
* @subpackage tests
|
||||||
|
*
|
||||||
|
* A data object that implements the primary side of a many_many (where the extra fields are
|
||||||
|
* defined.) The many-many refers to ManyManyListTest_Secondary rather than ManyManyListTest_SecondarySub
|
||||||
|
* by design, because we're trying to test that a subclass instance picks up the extra fields of it's parent.
|
||||||
|
*/
|
||||||
|
class ManyManyListTest_IndirectPrimary extends DataObject implements TestOnly {
|
||||||
|
|
||||||
|
private static $db = array(
|
||||||
|
'Title' => 'Varchar(255)'
|
||||||
|
);
|
||||||
|
|
||||||
|
private static $many_many = array(
|
||||||
|
'Secondary' => 'ManyManyListTest_Secondary'
|
||||||
|
);
|
||||||
|
|
||||||
|
private static $many_many_extraFields = array(
|
||||||
|
'Secondary' => array(
|
||||||
|
'DocumentSort' => 'Int'
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @package framework
|
||||||
|
* @subpackage tests
|
||||||
|
*
|
||||||
|
* A data object that implements the secondary side of a many_many when extended by
|
||||||
|
* ManyManyListTest_IndirectSecondaryExtension.
|
||||||
|
*/
|
||||||
|
class ManyManyListTest_Secondary extends DataObject implements TestOnly {
|
||||||
|
|
||||||
|
// Possibly not required, but want to simulate a real test failure case where
|
||||||
|
// database tables are present.
|
||||||
|
private static $db = array(
|
||||||
|
'Title' => 'Varchar(255)'
|
||||||
|
);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @package framework
|
||||||
|
* @subpackage tests
|
||||||
|
*
|
||||||
|
* A data object that is a subclass of the secondary side. The test will create an instance of this,
|
||||||
|
* and ensure that the extra fields are available on the instance even though the many many is
|
||||||
|
* defined at the parent level.
|
||||||
|
*/
|
||||||
|
class ManyManyListTest_SecondarySub extends ManyManyListTest_Secondary {
|
||||||
|
|
||||||
|
// private static $db = array(
|
||||||
|
// 'Other' => 'Varchar(255)'
|
||||||
|
// );
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @package framework
|
||||||
|
* @subpackage tests
|
||||||
|
*
|
||||||
|
* An extension that is applied to ManyManyListTest_Secondary that implements the other side of the many-many
|
||||||
|
* relationship.
|
||||||
|
*/
|
||||||
|
class ManyManyListTest_IndirectSecondaryExtension extends DataExtension implements TestOnly {
|
||||||
|
|
||||||
|
private static $db = array(
|
||||||
|
'Title' => 'Varchar(255)'
|
||||||
|
);
|
||||||
|
|
||||||
|
private static $belongs_many_many = array(
|
||||||
|
'Primary' => 'ManyManyListTest_IndirectPrimary'
|
||||||
|
);
|
||||||
|
|
||||||
|
}
|
6
tests/model/ManyManyListExtensionTest.yml
Normal file
6
tests/model/ManyManyListExtensionTest.yml
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
ManyManyListTest_IndirectPrimary:
|
||||||
|
manymany_extra_primary:
|
||||||
|
Title: 'primary'
|
||||||
|
ManyManyListTest_SecondarySub:
|
||||||
|
manymany_extra_secondary:
|
||||||
|
Title: 'secondary'
|
@ -198,15 +198,37 @@ class VersionedTest extends SapphireTest {
|
|||||||
$page1 = $this->objFromFixture('VersionedTest_DataObject', 'page1');
|
$page1 = $this->objFromFixture('VersionedTest_DataObject', 'page1');
|
||||||
$page1->Content = 'orig';
|
$page1->Content = 'orig';
|
||||||
$page1->write();
|
$page1->write();
|
||||||
$oldVersion = $page1->Version;
|
$firstVersion = $page1->Version;
|
||||||
$page1->publish('Stage', 'Live', false);
|
$page1->publish('Stage', 'Live', false);
|
||||||
$this->assertEquals($oldVersion, $page1->Version, 'publish() with $createNewVersion=FALSE');
|
$this->assertEquals(
|
||||||
|
$firstVersion,
|
||||||
|
$page1->Version,
|
||||||
|
'publish() with $createNewVersion=FALSE does not create a new version'
|
||||||
|
);
|
||||||
|
|
||||||
$page1->Content = 'changed';
|
$page1->Content = 'changed';
|
||||||
$page1->write();
|
$page1->write();
|
||||||
$oldVersion = $page1->Version;
|
$secondVersion = $page1->Version;
|
||||||
|
$this->assertTrue($firstVersion < $secondVersion, 'write creates new version');
|
||||||
|
|
||||||
$page1->publish('Stage', 'Live', true);
|
$page1->publish('Stage', 'Live', true);
|
||||||
$this->assertTrue($oldVersion < $page1->Version, 'publish() with $createNewVersion=TRUE');
|
$thirdVersion = Versioned::get_latest_version('VersionedTest_DataObject', $page1->ID)->Version;
|
||||||
|
$liveVersion = Versioned::get_versionnumber_by_stage('VersionedTest_DataObject', 'Live', $page1->ID);
|
||||||
|
$stageVersion = Versioned::get_versionnumber_by_stage('VersionedTest_DataObject', 'Stage', $page1->ID);
|
||||||
|
$this->assertTrue(
|
||||||
|
$secondVersion < $thirdVersion,
|
||||||
|
'publish() with $createNewVersion=TRUE creates a new version'
|
||||||
|
);
|
||||||
|
$this->assertEquals(
|
||||||
|
$liveVersion,
|
||||||
|
$thirdVersion,
|
||||||
|
'publish() with $createNewVersion=TRUE publishes to live'
|
||||||
|
);
|
||||||
|
$this->assertEquals(
|
||||||
|
$stageVersion,
|
||||||
|
$secondVersion,
|
||||||
|
'publish() with $createNewVersion=TRUE does not affect stage'
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testRollbackTo() {
|
public function testRollbackTo() {
|
||||||
@ -222,10 +244,11 @@ class VersionedTest extends SapphireTest {
|
|||||||
$changedVersion = $page1->Version;
|
$changedVersion = $page1->Version;
|
||||||
|
|
||||||
$page1->doRollbackTo($origVersion);
|
$page1->doRollbackTo($origVersion);
|
||||||
$page1 = Versioned::get_one_by_stage('VersionedTest_DataObject', 'Stage',
|
$page1 = Versioned::get_one_by_stage('VersionedTest_DataObject', 'Stage', array(
|
||||||
sprintf('"VersionedTest_DataObject"."ID" = %d', $page1->ID));
|
'"VersionedTest_DataObject"."ID" = ?' => $page1->ID
|
||||||
|
));
|
||||||
|
|
||||||
$this->assertTrue($page1->Version > $changedVersion, 'Create a new higher version number');
|
$this->assertTrue($page1->Version == $changedVersion + 1, 'Create a new higher version number');
|
||||||
$this->assertEquals('orig', $page1->Content, 'Copies the content from the old version');
|
$this->assertEquals('orig', $page1->Content, 'Copies the content from the old version');
|
||||||
|
|
||||||
// check db entries
|
// check db entries
|
||||||
|
Loading…
Reference in New Issue
Block a user