Widgets ======= Os widgets são blocos de construção reutilizáveis usados nas [views (visões)](structure-views.md) para criar e configurar complexos elementos de interface do usuário sob uma modelagem orientada a objetos. Por exemplo, um widget datapicker pode gerar um calendário que permite aos usuários selecionarem uma data que desejam inserir em um formulário. Tudo o que você precisa fazer é apenas inserir um código na view (visão) conforme o seguinte: ```php <?php use yii\jui\DatePicker; ?> <?= DatePicker::widget(['name' => 'date']) ?> ``` Existe uma quantidade considerável de widgets empacotados no Yii, como o [[yii\widgets\ActiveForm|active form]], o [[yii\widgets\Menu|menu]], o [jQuery UI widgets](widget-jui.md), o [Twitter Bootstrap widgets](widget-bootstrap.md), etc. A seguir, iremos introduzir os conhecimentos básicos sobre os widgets. Por favor, consulte a documentação de classes da API se você quiser saber mais sobre o uso de um determinado widget. ## Usando Widgets <a name="using-widgets"></a> Os widgets são usados principalmente nas [views (visões)](structure-views.md). Você pode chamar o método [[yii\base\Widget::widget()]] para usar um widget em uma view (visão). O método possui um array de [configuração](concept-configurations.md) para inicializar o widget e retornar o resultado da renderização do widget. Por exemplo, o código a seguir insere um widget datapicker configurado para usar o idioma Russo e manter a data selecionada no atributo `from_date` do `$model`. ```php <?php use yii\jui\DatePicker; ?> <?= DatePicker::widget([ 'model' => $model, 'attribute' => 'from_date', 'language' => 'ru', 'clientOptions' => [ 'dateFormat' => 'yy-mm-dd', ], ]) ?> ``` Alguns widgets podem ter um bloco de conteúdo que deve ser colocado entre as chamadas dos métodos [[yii\base\Widget::begin()]] e [[yii\base\Widget::end()]]. Por exemplo, o código a seguir usa o widget [[yii\widgets\ActiveForm]] para gerar um formulário de login. O widget irá gerar as tags de abertura e de fechamento do `<form>` respectivamente nos lugares onde os métodos `begin()` e `end()` foram chamados. Qualquer conteúdo entre estes métodos serão renderizados entre as tags de abertura e de fechamento do `<form>`. ```php <?php use yii\widgets\ActiveForm; use yii\helpers\Html; ?> <?php $form = ActiveForm::begin(['id' => 'login-form']); ?> <?= $form->field($model, 'username') ?> <?= $form->field($model, 'password')->passwordInput() ?> <div class="form-group"> <?= Html::submitButton('Login') ?> </div> <?php ActiveForm::end(); ?> ``` Observe que ao contrário do [[yii\base\Widget::widget()]] que retorna a renderização de um widget, o método [[yii\base\Widget::begin()]] retorna uma instância do widget que pode ser usado para construir o seu conteúdo. ## Criando Widgets <a name="creating-widgets"></a> Para criar um widget, estenda a classe [[yii\base\Widget]] e sobrescreva os métodos [[yii\base\Widget::init()]] e/ou [[yii\base\Widget::run()]]. Normalmente, o método `init()` deve conter os códigos que normalizam as propriedade do widget, enquanto o método `run()` deve conter o código que gera o resultado da renderização do widget. O resultado da renderização pode ser feito diretamente dando "echo" ou pelo retorno de uma string no método `run()`. No exemplo a seguir, o `HelloWidget` codifica o HTML e exibe o conteúdo atribuído à sua propriedade `message`. Se a propriedade não for definida, será exibido "Hello World" como padrão. ```php namespace app\components; use yii\base\Widget; use yii\helpers\Html; class HelloWidget extends Widget { public $message; public function init() { parent::init(); if ($this->message === null) { $this->message = 'Hello World'; } } public function run() { return Html::encode($this->message); } } ``` Para usar este widget, simplesmente insira o código a seguir em uma view (visão): ```php <?php use app\components\HelloWidget; ?> <?= HelloWidget::widget(['message' => 'Good morning']) ?> ``` O `HelloWidget` abaixo é uma variante que pega o conteúdo entre as chamadas de `begin()` e `end()`, codifica o HTML e em seguida os exibe. ```php namespace app\components; use yii\base\Widget; use yii\helpers\Html; class HelloWidget extends Widget { public function init() { parent::init(); ob_start(); } public function run() { $content = ob_get_clean(); return Html::encode($content); } } ``` Como você pode ver, o buffer de saída do PHP é iniciado no método `init()` para que qualquer conteúdo entre as chamadas de `init()` e `run()` possam ser capturadas, processadas e retornadas em `run()`. > Informação: Ao chamar o [[yii\base\Widget::begin()]], uma nova instância do widget será criada e o método `init()` será chamado logo ao final de seu construtor. Ao chamar o [[yii\base\Widget::end()]], o método `run()` será chamado cujo o resultado da renderização será dado *echo* pelo `end()`. O código a seguir mostra como você pode usar esta nova variante do `HelloWidget`: ```php <?php use app\components\HelloWidget; ?> <?php HelloWidget::begin(); ?> um conteúdo qualquer... <?php HelloWidget::end(); ?> ``` Algumas vezes, um widget pode precisar renderizar um grande conteúdo. Enquanto você pode inserir todo este conteúdo no método `run()`, uma boa prática é colocá-lo em uma [view (visão)](structure-views.md) e chamar o [[yii\base\Widget::render()]] para renderizá-lo. Por exemplo, ```php public function run() { return $this->render('hello'); } ``` Por padrão, as views (visões) para um widget devem ser armazenadas em arquivos sob o diretório `WidgetPath/views`, onde o `WidgetPath` significa o diretório que contém os arquivo da classe do widget. Portanto, o exemplo anterior irá renderizar o arquivo de view (visão) `@app/components/views/hello.php`, assumindo que a classe widget está localizada sob o diretório `@app/components`. Você pode sobrescrever o método [[yii\base\Widget::getViewPath()]] para personalizar o diretório que conterá os arquivos de views (visões) do widget. ## Boas Práticas <a name="best-practices"></a> Os widgets são uma maneira orientada a objetos de reutilizar códigos de view (visão). Ao criar os widgets, você ainda deve seguir o padrão MVC. Em geral, você deve manter a lógica nas classes widgets e manter as apresentações nas [views (visões)](structure-views.md). Os widgets devem ser projetados para serem autossuficientes. Isto é, ao utilizar um widget, você deverá ser capaz de removê-lo de uma view (visão) sem fazer qualquer outra coisa. Isto pode ser complicado se um widget requerer recursos externos, tais como CSS, JavaScript, imagens, etc. Felizmente, o Yii fornece o suporte para [asset bundles](structure-assets.md), que pode ser utilizado para resolver este problema. Quando um widget contiver somente código de view (visão), será bem semelhante a uma [view (visão)](structure-views.md). Na verdade, neste caso, a única diferença é que um widget é uma classe para ser redistribuída, enquanto uma view é apenas um simples script PHP que você preferirá manter em sua aplicação