Commit 05e4fb0b by Alexander Makarov

Revised model guide

parent 8542f60a
Model Model
===== =====
In keeping with the MVC approach, a model in Yii is intended for storing or temporarily representing application data. Yii models have the following basic features: In keeping with the MVC approach, a model in Yii is intended for storing or temporarily representing application data.
Yii models have the following basic features:
- Attribute declaration: a model defines what is considered an attribute. - Attribute declaration: a model defines what is considered an attribute.
- Attribute labels: each attribute may be associated with a label for display purpose. - Attribute labels: each attribute may be associated with a label for display purpose.
- Massive attribute assignment: the ability to populate multiple model attributes in one step. - Massive attribute assignment: the ability to populate multiple model attributes in one step.
- Scenario-based data validation. - Scenario-based data validation.
Models in Yii extend from the [[\yii\base\Model]] class. Models are typically used to both hold data and define the validation rules for that data (aka, the business logic). The business logic greatly simplifies the generation of models from complex web forms by providing validation and error reporting. Models in Yii extend from the [[\yii\base\Model]] class. Models are typically used to both hold data and define
The Model class is also the base class for more advanced models with additional functionality, such as [Active Record](active-record.md). the validation rules for that data (aka, the business logic). The business logic greatly simplifies the generation
of models from complex web forms by providing validation and error reporting.
The Model class is also the base class for more advanced models with additional functionality, such
as [Active Record](active-record.md).
Attributes Attributes
---------- ----------
...@@ -21,7 +26,7 @@ may contain a `title` attribute and a `content` attribute, accessible as follows ...@@ -21,7 +26,7 @@ may contain a `title` attribute and a `content` attribute, accessible as follows
```php ```php
$post = new Post; $post = new Post;
$post->title = 'Hello, world'; $post->title = 'Hello, world';
$post->content = 'Something interesting is happening'; $post->content = 'Something interesting is happening.';
echo $post->title; echo $post->title;
echo $post->content; echo $post->content;
``` ```
...@@ -62,8 +67,11 @@ Attribute labels are mainly used for display purpose. For example, given an attr ...@@ -62,8 +67,11 @@ Attribute labels are mainly used for display purpose. For example, given an attr
a label `First Name` that is more user-friendly when displayed to end users in places such as form labels and a label `First Name` that is more user-friendly when displayed to end users in places such as form labels and
error messages. Given an attribute name, you can obtain its label by calling [[\yii\base\Model::getAttributeLabel()]]. error messages. Given an attribute name, you can obtain its label by calling [[\yii\base\Model::getAttributeLabel()]].
To declare attribute labels, override the [[\yii\base\Model::attributeLabels()]] method. The overridden method returns a mapping of attribute names to attribute labels, as shown in the example below. If an attribute is not found To declare attribute labels, override the [[\yii\base\Model::attributeLabels()]] method. The overridden method returns
in this mapping, its label will be generated using the [[\yii\base\Model::generateAttributeLabel()]] method. In many cases, [[\yii\base\Model::generateAttributeLabel()]] will generate reasonable labels (e.g. `username` to `Username`, `orderNumber` to `Order Number`). a mapping of attribute names to attribute labels, as shown in the example below. If an attribute is not found
in this mapping, its label will be generated using the [[\yii\base\Model::generateAttributeLabel()]] method.
In many cases, [[\yii\base\Model::generateAttributeLabel()]] will generate reasonable labels (e.g. `username` to `Username`,
`orderNumber` to `Order Number`).
```php ```php
// LoginForm has two attributes: username and password // LoginForm has two attributes: username and password
...@@ -86,7 +94,8 @@ Scenarios ...@@ -86,7 +94,8 @@ Scenarios
--------- ---------
A model may be used in different *scenarios*. For example, a `User` model may be used to collect user login inputs, A model may be used in different *scenarios*. For example, a `User` model may be used to collect user login inputs,
but it may also be used for user registration purposes. In the one scenario, every piece of data is required; in the other, only the username and password would be. but it may also be used for user registration purposes. In the one scenario, every piece of data is required;
in the other, only the username and password would be.
To easily implement the business logic for different scenarios, each model has a property named `scenario` To easily implement the business logic for different scenarios, each model has a property named `scenario`
that stores the name of the scenario that the model is currently being used in. As will be explained in the next that stores the name of the scenario that the model is currently being used in. As will be explained in the next
...@@ -112,6 +121,9 @@ class User extends \yii\db\ActiveRecord ...@@ -112,6 +121,9 @@ class User extends \yii\db\ActiveRecord
} }
``` ```
If `scenarios` method is not defined, default scenario is applied. That means attributes with validation rules are
considered *active*.
If you want to keep the default scenario available besides your own scenarios, use inheritance to include it: If you want to keep the default scenario available besides your own scenarios, use inheritance to include it:
```php ```php
class User extends \yii\db\ActiveRecord class User extends \yii\db\ActiveRecord
...@@ -161,9 +173,9 @@ class EmployeeController extends \yii\web\Controller ...@@ -161,9 +173,9 @@ class EmployeeController extends \yii\web\Controller
``` ```
The example above presumes that the model is based upon [Active Record](active-record.md). For basic form models, The example above presumes that the model is based upon [Active Record](active-record.md). For basic form models,
scenarios are rarely needed, as the basic form model is normally tied directly to a single form. scenarios are rarely needed, as the basic form model is normally tied directly to a single form and, as noted above,
The default implementation of the `scenarios()`-method will return all scenarios found in the `rules()` the default implementation of the `scenarios()` returns every property with active validation rule making it always
declaration (explained in the next section) so in simple cases you do not need to define scenarios. available for mass assignment and validation.
Validation Validation
...@@ -211,16 +223,6 @@ When `validate()` is called, the actual validation rules executed are determined ...@@ -211,16 +223,6 @@ When `validate()` is called, the actual validation rules executed are determined
- the rule must be active for the current scenario. - the rule must be active for the current scenario.
### Active Attributes
An attribute is *active* if it is subject to some validations in the current scenario.
### Safe Attributes
An attribute is *safe* if it can be massively assigned in the current scenario.
Massive Attribute Retrieval and Assignment Massive Attribute Retrieval and Assignment
------------------------------------------ ------------------------------------------
...@@ -229,18 +231,23 @@ The following code will return *all* attributes in the `$post` model ...@@ -229,18 +231,23 @@ The following code will return *all* attributes in the `$post` model
as an array of name-value pairs. as an array of name-value pairs.
```php ```php
$attributes = $post->attributes; $post = Post::find(42);
var_dump($attributes); if ($post) {
$attributes = $post->attributes;
var_dump($attributes);
}
``` ```
Using the same `attributes` property you can massively assign data from associative array to model attributes: Using the same `attributes` property you can massively assign data from associative array to model attributes:
```php ```php
$post = new Post();
$attributes = [ $attributes = [
'title' => 'Massive assignment example', 'title' => 'Massive assignment example',
'body' => 'Never allow assigning attributes that are not meant to be assigned.', 'content' => 'Never allow assigning attributes that are not meant to be assigned.',
]; ];
$postForm->attributes = $attributes; $post->attributes = $attributes;
var_dump($attributes);
``` ```
In the code above we're assigning corresponding data to model attributes named as array keys. The key difference from mass In the code above we're assigning corresponding data to model attributes named as array keys. The key difference from mass
...@@ -256,8 +263,10 @@ rules are described in `rules()` method of the model while what's safe for mass ...@@ -256,8 +263,10 @@ rules are described in `rules()` method of the model while what's safe for mass
assignment is described in `scenarios` method: assignment is described in `scenarios` method:
```php ```php
function rules() class User extends ActiveRecord
{ {
function rules()
{
return [ return [
// rule applied when corresponding field is "safe" // rule applied when corresponding field is "safe"
['username', 'string', 'length' => [4, 32]], ['username', 'string', 'length' => [4, 32]],
...@@ -267,21 +276,92 @@ function rules() ...@@ -267,21 +276,92 @@ function rules()
// rule applied when scenario is "signup" no matter if field is "safe" or not // rule applied when scenario is "signup" no matter if field is "safe" or not
['hashcode', 'check', 'on' => 'signup'], ['hashcode', 'check', 'on' => 'signup'],
]; ];
} }
function scenarios() function scenarios()
{ {
return [ return [
// on signup allow mass assignment of username // on signup allow mass assignment of username
'signup' => ['username', 'password'], 'signup' => ['username', 'password'],
'update' => ['username', 'first_name'], 'update' => ['username', 'first_name'],
]; ];
}
}
```
For the code above mass assignment will be allowed stsrictly according to `scenarios()`:
```php
$user = User::find(42);
$data = ['password' => '123'];
$user->attributes = $data;
print_r($data);
```
Will give you empty array because there's no default scenario defined in our `scenarios()`.
```php
$user = User::find(42);
$user->scenario = 'signup';
$data = [
'username' => 'samdark',
'password' => '123',
'hashcode' => 'test',
];
$user->attributes = $data;
print_r($data);
```
Will give you the following:
```php
array(
'username' => 'samdark',
'first_name' => null,
'password' => '123',
'hashcode' => null, // it's not defined in scenarios method
)
```
In case of not defined `scenarios` method like the following:
```php
class User extends ActiveRecord
{
function rules()
{
return [
['username', 'string', 'length' => [4, 32]],
['first_name', 'string', 'max' => 128],
['password', 'required'],
];
}
} }
``` ```
Note that everything is unsafe by default in active record models and you can't make field "safe" without specifying The code above assumes default scenario so mass assignment will be available for all fields with `rules` defined:
scenario. In normal models that are not active record, everything is safe by default.
```php
$user = User::find(42);
$data = [
'username' => 'samdark',
'first_name' => 'Alexander',
'last_name' => 'Makarov',
'password' => '123',
];
$user->attributes = $data;
print_r($data);
```
Will give you the following:
```php
array(
'username' => 'samdark',
'first_name' => 'Alexander',
'password' => '123',
)
```
See also See also
-------- --------
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment