Models (Modelos)
================

Os models (modelos) fazem parte da arquitetura [MVC](http://en.wikipedia.org/wiki/Model%E2%80%93view%E2%80%93controller).
Eles representam os dados, as regras e a lógica de negócio.

Você pode criar uma classe model estendendo de [[yii\base\Model]] ou de seus filhos.
A classe base [[yii\base\Model]] suporta muitos recursos úteis:

* [Atributos](#attributes): representa os dados de negócio e podem ser acessados 
  normalmente como uma propriedade de objeto ou como um elemento de array;
* [Labels dos atributos](#attribute-labels): especifica os labels de exibição dos 
  atributos;
* [Atribuição em massa](#massive-assignment): suporta popular vários atributos em 
  uma única etapa;
* [Regras de validação](#validation-rules): garante que os dados de entrada sejam 
  baseadas nas regras de validação que foram declaradas;
* [Data Exporting](#data-exporting): permite que os dados de model a serem exportados 
  em array possuam formatos personalizados.

A classe `Model` também é a classe base para models mais avançados, como o [Active Record](db-active-record.md).
Por favor, consulte a documentação relevante para mais detalhes sobre estes models mais avançados.

> Informação: Você não é obrigado basear suas classe model em [[yii\base\Model]]. 
> No entanto, por existir muitos componentes do Yii construídos para suportar o 
> [[yii\base\Model]], normalmente é a classe base preferível para um model.


## Atributos <span id="attributes"></span>

Os models representam dados de negócio por meio de *atributos*. Cada atributo é 
uma propriedade publicamente acessível de um model. O método [[yii\base\Model::attributes()]] 
especifica quais atributos de uma classe model possuirá.

Você pode acessar um atributo como fosse uma propriedade normal de um objeto:

```php
$model = new \app\models\ContactForm;

// "name" é um atributo de ContactForm
$model->name = 'example';
echo $model->name;
```

Você também pode acessar os atributos como elementos de um array, graças ao suporte 
de [ArrayAccess](http://php.net/manual/en/class.arrayaccess.php) e 
[ArrayIterator](http://php.net/manual/en/class.arrayiterator.php) pelo 
[[yii\base\Model]]:

```php
$model = new \app\models\ContactForm;

// acessando atributos como elementos de array
$model['name'] = 'example';
echo $model['name'];

// iterando sobre os atributos
foreach ($model as $name => $value) {
    echo "$name: $value\n";
}
```


### Definindo Atributos <span id="defining-attributes"></span>

Por padrão, se a classe model estender diretamente de [[yii\base\Model]], todas 
as suas variáveis públicas e não estáticas serão atributos. Por exemplo, a classe 
model `ContactForm` a seguir possui quatro atributos: `name`, `email`, `subject` 
e `body`. O model `ContactForm` é usado para representar os dados de entrada obtidos 
a partir de um formulário HTML.

```php
namespace app\models;

use yii\base\Model;

class ContactForm extends Model
{
    public $name;
    public $email;
    public $subject;
    public $body;
}
```


Você pode sobrescrever o método [[yii\base\Model::attributes()]] para definir 
atributos de uma forma diferente. Este método deve retornar os nomes dos atributos 
em um model. Por exemplo, o [[yii\db\ActiveRecord]] faz com que o método retorne 
os nomes das colunas da tabela do banco de dados como nomes de atributos.
Observe que também poderá sobrescrever os métodos mágicos tais como `__get()` e 
`__set()`, para que os atributos poderem ser acessados como propriedades normais 
de objetos.


### Labels dos Atributos <span id="attribute-labels"></span>

Ao exibir valores ou obter dados de entrada dos atributos, muitas vezes é necessário 
exibir alguns labels associados aos atributos. Por exemplo, dado um atributo chamado 
`firstName`, você pode querer exibir um label `First Name` que é mais amigável 
quando exibido aos usuários finais como em formulários e mensagens de erro.

Você pode obter o label de um atributo chamando o método [[yii\base\Model::getAttributeLabel()]]. 
Por exemplo,

```php
$model = new \app\models\ContactForm;

// displays "Name"
echo $model->getAttributeLabel('name');
```

Por padrão, os labels dos atributos automaticamente serão gerados com os nomes dos 
atributos. Isto é feito pelo método [[yii\base\Model::generateAttributeLabel()]]. 
Ele transforma os nomes camel-case das variáveis em várias palavras, colocando em 
caixa alta a primeira letra de cada palavra. Por exemplo, `username` torna-se 
`Username`, enquanto `firstName` torna-se `First Name`.

Se você não quiser usar esta geração automática do labels, poderá sobrescrever o 
método [[yii\base\Model::attributeLabels()]] declarando explicitamente os labels 
dos atributos. Por exemplo,

```php
namespace app\models;

use yii\base\Model;

class ContactForm extends Model
{
    public $name;
    public $email;
    public $subject;
    public $body;

    public function attributeLabels()
    {
        return [
            'name' => 'Your name',
            'email' => 'Your email address',
            'subject' => 'Subject',
            'body' => 'Content',
        ];
    }
}
```

Para aplicações que suportam vários idiomas, você pode querer traduzir os labels 
dos atributos. Isto também é feito no método [[yii\base\Model::attributeLabels()|attributeLabels()]], 
conforme o exemplo a seguir:

```php
public function attributeLabels()
{
    return [
        'name' => \Yii::t('app', 'Your name'),
        'email' => \Yii::t('app', 'Your email address'),
        'subject' => \Yii::t('app', 'Subject'),
        'body' => \Yii::t('app', 'Content'),
    ];
}
```

Você pode até definir condicionalmente os labels dos atributos. Por exemplo, baseado 
no [cenário](#scenarios) que o model estiver utilizando, você pode retornar diferentes 
labels para o mesmo atributo.

> Informação: Estritamente falando, os labels dos atributos fazem parte das 
[views](structure-views.md) (visões). Mas ao declarar os labels em models (modelos), 
frequentemente tornam-se mais convenientes e podem resultar um código mais limpo 
e reutilizável.


## Cenários <span id="scenarios"></span>

Um model (modelo) pode ser usado em diferentes *cenários*. Por exemplo, um model 
`User` pode ser usado para obter dados de entrada de login, mas também pode ser 
usado com a finalidade de registrar o usuário. Em diferentes cenários, um model 
pode usar diferentes regras e lógicas de negócio. Por exemplo, um atributo `email` 
pode ser obrigatório durante o cadastro do usuário, mas não durante ao login.

Um model (modelo) usa a propriedade [[yii\base\Model::scenario]] para identificar 
o cenário que está sendo usado.
Por padrão, um model (modelo) suporta apenas um único cenário chamado `default`. 
O código a seguir mostra duas formas de definir o cenário de um model (modelo):

```php
// o cenário é definido pela propriedade
$model = new User;
$model->scenario = 'login';

// o cenário é definido por meio de configuração
$model = new User(['scenario' => 'login']);
```

Por padrão, os cenários suportados por um model (modelo) são determinados pelas 
[regras de validação](#validation-rules) declaradas no próprio model (modelo).
No entanto, você pode personalizar este comportamento sobrescrevendo o método 
[[yii\base\Model::scenarios()]], conforme o exemplo a seguir:

```php
namespace app\models;

use yii\db\ActiveRecord;

class User extends ActiveRecord
{
    public function scenarios()
    {
        return [
            'login' => ['username', 'password'],
            'register' => ['username', 'email', 'password'],
        ];
    }
}
```

> Informação: Nos exemplos anteriores, as classes model (model) são estendidas de 
[[yii\db\ActiveRecord]] por usarem diversos cenários para auxiliarem as classes 
[Active Record](db-active-record.md) classes.

O método `scenarios()` retorna um array cujas chaves são os nomes dos cenários e 
os valores que correspondem aos *active attributes* (atributo ativo). Um atributo 
ativo podem ser [atribuídos em massa](#massive-assignment) e é sujeito a 
[validação](#validation-rules). No  exemplo anterior, os atributos `username` e 
`password` são ativos no cenário `login`; enquanto no cenário `register`, além 
dos atribitos `username` e `password`, o atributo `email` passará a ser ativo.

A implementação padrão do método `scenarios()` retornará todos os cenários encontrados 
nas regras de validação declaradas no método [[yii\base\Model::rules()]]. Ao 
sobrescrever o método `scenarios()`, se quiser introduzir novos cenários, além 
dos cenários padrão, poderá escrever um código conforme o exemplo a seguir:
 
```php
namespace app\models;

use yii\db\ActiveRecord;

class User extends ActiveRecord
{
    public function scenarios()
    {
        $scenarios = parent::scenarios();
        $scenarios['login'] = ['username', 'password'];
        $scenarios['register'] = ['username', 'email', 'password'];
        return $scenarios;
    }
}
```

O recurso de cenários são usados principalmente para [validação](#validation-rules) 
e para [atribuição em massa](#massive-assignment).
Você pode, no entanto, usá-lo para outros fins. Por exemplo, você pode declarar 
diferentes [labels para os atributos](#attribute-labels) baseados no cenário atual.


## Regras de Validação <span id="validation-rules"></span>

Quando os dados para um model (modelo) são recebidos de usuários finais, devem ser 
validados para garantir que satisfazem as regras (*regras de validação*, também 
conhecidos como *regras de negócio*). Por exemplo, considerando um model (modelo) 
`ContactForm`, você pode querer garantir que todos os atributos não sejam vazios e 
que o atributo `email` contenha um e-mail válido.
Se o valor de algum atributo não satisfizer a regra de negócio correspondente, 
mensagens apropriadas de erros serão exibidas para ajudar o usuário a corrigi-los.

Você pode chamar o método [[yii\base\Model::validate()]] para validar os dados 
recebidos. O método usará as regras de validação declaradas em [[yii\base\Model::rules()]] 
para validar todos os atributos relevantes. Se nenhum erro for encontrado, o método 
retornará true. Caso contrário, o método irá manter os erros na propriedade 
[[yii\base\Model::errors]] e retornará false. Por exemplo,

```php
$model = new \app\models\ContactForm;

// os atributos do model serão populados pelos dados fornecidos pelo usuário
$model->attributes = \Yii::$app->request->post('ContactForm');

if ($model->validate()) {
    // todos os dados estão válidos 
} else {
    // a validação falhou: $errors é um array contendo as mensagens de erro
    $errors = $model->errors;
}
```


Para declarar as regras de validação em um model (modelo), sobrescreva o método 
[[yii\base\Model::rules()]] retornando as regras que os atributos do model (modelo) 
devem satisfazer. O exemplo a seguir mostra as regras de validação sendo declaradas 
no model (modelo) `ContactForm`:

```php
public function rules()
{
    return [
        // os atributos name, email, subject e body são obrigatórios
        [['name', 'email', 'subject', 'body'], 'required'],

        // o atributo email deve ter um e-mail válido
        ['email', 'email'],
    ];
}
```

Uma regra pode ser usada para validar um ou vários atributos e, um atributo pode 
ser validado por uma ou várias regras.
Por favor, consulte a seção [Validação de Dados](input-validation.md) para mais 
detalhes sobre como declarar regras de validação.

Às vezes, você pode querer que uma regra se aplique apenas em determinados 
[cenários](#scenarios). Para fazer isso, você pode especificar a propriedade 
`on` de uma regra, como o seguinte:

```php
public function rules()
{
    return [
        // os atributos username, email e password são obrigatórios no cenario "register"
        [['username', 'email', 'password'], 'required', 'on' => 'register'],

        // os atributos username e password são obrigatórios no cenario "login"
        [['username', 'password'], 'required', 'on' => 'login'],
    ];
}
```

Se você não especificar a propriedade `on`, a regra será aplicada em todos os 
cenários. Uma regra é chamada de *active rule* (regra ativa), se ela puder ser 
aplicada no [[yii\base\Model::scenario|cenário]] atual.

Um atributo será validado, se e somente se, for um atributo ativo declarado no 
método `scenarios()` e estiver associado a uma ou várias regras declaradas no método `rules()`.


## Atribuição em Massa <span id="massive-assignment"></span>

Atribuição em massa é a forma conveniente para popular um model (modelo) com os 
dados de entrada do usuário usando uma única linha de código.
Ele popula os atributos de um model (modelo) atribuindo os dados de entrada diretamente 
na propriedade [[yii\base\Model::$attributes]]. Os dois códigos a seguir são 
equivalentes, ambos tentam atribuir os dados do formulário enviados pelos usuários 
finais para os atributos do model (modelo) `ContactForm`.  Evidentemente, a 
primeira forma, que utiliza a atribuição em massa, é a mais limpa e o menos 
propenso a erros do que a segunda forma:

```php
$model = new \app\models\ContactForm;
$model->attributes = \Yii::$app->request->post('ContactForm');
```

```php
$model = new \app\models\ContactForm;
$data = \Yii::$app->request->post('ContactForm', []);
$model->name = isset($data['name']) ? $data['name'] : null;
$model->email = isset($data['email']) ? $data['email'] : null;
$model->subject = isset($data['subject']) ? $data['subject'] : null;
$model->body = isset($data['body']) ? $data['body'] : null;
```


### Atributos Seguros <span id="safe-attributes"></span>

A atribuição em massa só se aplica aos chamados *safe attributes* (atributos seguros), 
que são os atributos listados no [[yii\base\Model::scenarios()]] para o 
[[yii\base\Model::scenario|cenário]] atual de um model (modelo).
Por exemplo, se o model (modelo) `User` declarar o cenário como o código a seguir, 
quando o cenário atual for `login`, apenas os atributos `username` e `password` 
podem ser atribuídos em massa. Todos os outros atributos permanecerão inalterados.

```php
public function scenarios()
{
    return [
        'login' => ['username', 'password'],
        'register' => ['username', 'email', 'password'],
    ];
}
```

> Informação: A razão da atribuição em massa só se aplicar para os atributos seguros 
é para que você tenha o controle de quais atributos podem ser modificados pelos 
dados dos usuário finais. Por exemplo, se o model (modelo) tiver um atributo 
`permission` que determina a permissão atribuída ao usuário, você gostará que 
apenas os administradores possam modificar este atributo através de uma interface backend.

Como a implementação do método [[yii\base\Model::scenarios()]] retornará todos os 
cenários e atributos encontrados em [[yii\base\Model::rules()]], se não quiser 
sobrescrever este método, isto significa que um atributo é seguro desde que esteja 
mencionado em uma regra de validação ativa.

Por esta razão, uma alias especial de validação chamada `safe`, será fornecida 
para que você possa declarar um atributo seguro, sem ser validado. Por exemplo, 
a declaração da regra a seguir faz com que tanto o atributo `title` quanto o
`description` sejam seguros.

```php
public function rules()
{
    return [
        [['title', 'description'], 'safe'],
    ];
}
```


### Atributos não Seguros <span id="unsafe-attributes"></span>

Como descrito anteriormente, o método [[yii\base\Model::scenarios()]] serve para 
dois propósitos: determinar quais atributos devem ser validados e quais atributos 
são seguros. Em alguns casos raros, você pode quer validar um atributo sem marca-lo 
como seguro. Para fazer isto, acrescente um ponto de exclamação `!` como prefixo 
do nome do atributo ao declarar no método `scenarios()`, como o que foi feito no 
atributo `secret` no exemplo a seguir:

```php
public function scenarios()
{
    return [
        'login' => ['username', 'password', '!secret'],
    ];
}
```

Quando o model (modelo) estiver no cenário `login`, todos os três atributos serão 
validados. No entanto, apenas os atributos `username` e `password` poderão ser 
atribuídos em massa. Para atribuir um valor de entrada no atributo `secret`, terá 
que fazer isto explicitamente da seguinte forma:

```php
$model->secret = $secret;
```


## Exportação de Dados <span id="data-exporting"></span>

Muitas vezes os models (modelos) precisam ser exportados em diferentes tipos de 
formatos. Por exemplo, você pode querer converter um conjunto de models (modelos) 
no formato JSON ou Excel. O processo de exportação pode ser divido em duas etapas independentes.
Na primeira etapa, os models (modelos) serão convertidos em arrays; na segunda 
etapa, os arrays serão convertidos em um determinado formato. Se concentre apenas 
na primeira etapa, uma vez que a segunda etapa pode ser alcançada por formatadores 
de dados genéricos, tais como o [[yii\web\JsonResponseFormatter]].

A maneira mais simples de converter um model (modelo) em um array consiste no uso 
da propriedade [[yii\base\Model::$attributes]].
Por exemplo,

```php
$post = \app\models\Post::findOne(100);
$array = $post->attributes;
```

Por padrão, a propriedade [[yii\base\Model::$attributes]] retornará os valores de 
todos os atributos declarados no método [[yii\base\Model::attributes()]].

Uma maneira mais flexível e poderosa de converter um model (modelo) em um array é 
através do método [[yii\base\Model::toArray()]]. O seu comportamento padrão é o 
mesmo do [[yii\base\Model::$attributes]]. No entanto, ele permite que você escolha 
quais itens de dados, chamados de *fields* (campos), devem ser mostrados no array 
resultante e como eles devem vir formatados.
Na verdade, é a maneira padrão de exportação de models (modelos) no desenvolvimento 
de Web services RESTful, como descrito na seção [Formatando Respostas](rest-response-formatting.md).


### Campos <span id="fields"></span>

Um campo é simplesmente um elemento nomeado no array obtido pela chamada do método 
[[yii\base\Model::toArray()]] de um model (modelo).

Por padrão, os nomes dos campos são iguais aos nomes dos atributos. No entanto, 
você pode alterar este comportamento sobrescrevendo os métodos 
[[yii\base\Model::fields()|fields()]] e/ou [[yii\base\Model::extraFields()|extraFields()]]. 
Ambos os métodos devem retornar uma lista dos campos definidos. Os campos definidos 
pelo método `fields()` são os campos padrão, o que significa que o `toArray()` 
retornará estes campos por padrão. O método `extraFields()` define, de forma adicional, 
os campos disponíveis que também podem ser retornados pelo `toArray()`, contanto 
que sejam especificados através do parâmetro `$expand`. Por exemplo, o código a 
seguir retornará todos os campos definidos em `fields()` incluindo os campos 
`prettyName` e `fullAddress`, a menos que estejam definidos no `extraFields()`.

```php
$array = $model->toArray([], ['prettyName', 'fullAddress']);
```

Você poderá sobrescrever o método `fields()` para adicionar, remover, renomear ou 
redefinir os campos. O valor de retorno do `fields()` deve ser um array. As chaves 
do array não os nomes dos campos e os valores correspondem ao nome do atributo 
definido, na qual, podem ser tanto os nomes de propriedades/atributos quanto funções 
anônimas que retornam o valor dos campos correspondentes. Em um caso especial, 
quando o nome do campo for igual ao nome do atributo definido, você poderá omitir 
a chave do array. Por exemplo,

```php
// usar uma lista explicita de todos os campos lhe garante que qualquer mudança 
// em sua tabela do banco de dados ou atributos do model (modelo) não altere os 
// nomes de seus campos (para manter compatibilidade com versões anterior da API).
public function fields()
{
    return [
        // o nome do campos é igual ao nome do atributo
        'id',

        // o nome do campo é "email", o nome do atributo correspondente é "email_address"
        'email' => 'email_address',

        // o nome do campo é "name", o seu valor é definido por uma função call-back do PHP
        'name' => function () {
            return $this->first_name . ' ' . $this->last_name;
        },
    ];
}

// filtra alguns campos, é bem usado quando você quiser herdar a implementação 
// da classe pai e remover alguns campos delicados.
public function fields()
{
    $fields = parent::fields();

    // remove os campos que contém informações delicadas
    unset($fields['auth_key'], $fields['password_hash'], $fields['password_reset_token']);

    return $fields;
}
```

> Atenção: Como, por padrão, todos os atributos de um model (modelo) serão 
>incluídos no array exportado, você deve examinar seus dados para ter certeza 
>que não possuem informações delicadas. Se existir, deverá sobrescrever o método 
>`fields()` para remove-los. No exemplo anterior, nós decidimos remover os 
>campos `auth_key`, `password_hash` e `password_reset_token`.


## Boas Práticas <span id="best-practices"></span>

A representação dos dados, regras e lógicas de negócios estão centralizados nos 
models (modelos). Muitas vezes precisam ser reutilizadas em lugares diferentes. 
Em um aplicativo bem projetado, models (modelos) geralmente são muitos maiores 
que os [controllers](structure-controllers.md)

Em resumo, os models (modelos):

* podem conter atributos para representar os dados de negócio;
* podem conter regras de validação para garantir a validade e integridade dos dados;
* podem conter métodos para implementar lógicas de negócio;
* NÃO devem acessar diretamente as requisições, sessões ou quaisquer dados do 
ambiente do usuário. Os models (modelos) devem receber estes dados a partir dos 
[controllers (controladores)](structure-controllers.md);
* devem evitar inserir HTML ou outros códigos de apresentação – isto deve ser 
feito nas [views (visões)](structure-views.md);
* devem evitar ter muitos [cenários](#scenarios) em um único model (modelo).

Você deve considerar em utilizar com mais frequência a última recomendação acima 
quando desenvolver sistemas grandes e complexos.
Nestes sistemas, os models (modelos) podem ser bem grandes, pois são usados em 
muitos lugares e podendo, assim, conter muitas regras e lógicas de negócio. 
Nestes casos, a manutenção do código de um model (modelo) pode se transformar 
em um pesadelo, na qual uma simples mudança no código pode afetar vários lugares 
diferentes. Para desenvolver um model (modelo) manutenível, você pode seguir a 
seguinte estratégia: 

* Definir um conjunto de classes model (modelo) base que são compartilhados por 
diferentes [aplicações](structure-applications.md) ou [módulos](structure-modules.md). 
Estas classes model (modelo) base deve contem um conjunto mínimo de regras e lógicas 
de negocio que são comuns entre os locais que as utilizem.
* Em cada [aplicação](structure-applications.md) ou [módulo](structure-modules.md) 
que usa um model (modelo), deve definir uma classe model (modelo) concreta que 
estenderá a classe model (modelo) base que a corresponde. A classe model (modelo) 
concreta irá conter apenas as regras e lógicas que são específicas de uma aplicação 
ou módulo.

Por exemplo, no [Template Avançado de Aplicação](tutorial-advanced-app.md), você 
pode definir uma classe model (modelo) base `common\models\Post`. Em seguida, 
para a aplicação front-end, você define uma classe model (modelo) concreta 
`frontend\models\Post` que estende de `common\models\Post`. E de forma similar 
para a aplicação back-end, você define a `backend\models\Post`. Com essa estratégia, 
você garantirá que o `frontend\models\Post` terá apenas códigos específicos da 
aplicação front-end e, se você fizer qualquer mudança nele, não precisará se 
preocupar se esta mudança causará erros na aplicação back-end.