Quando uma aplicação Yii começa a processar uma URL requerida, o primeiro passo necessário é obter a rota pela análise da URL. A rota é usada para instanciar o controlador (controller) da ação correspondente para manipular a requisição. Todo este processo é chamado de roteamento.
O processo inverso do roteamento é chamada de criação de URL, onde é criado uma URL a partir de uma determinada rota e seus parâmetros. Quando a URL criada for exigida em outro momento, o processo de roteamento pode resolve-la de volta para a rota original e seus parâmetros.
O ponto central responsável pelo roteamento e pela criação de URL é o
gerenciador de URL, na qual é registrado como o
componente da aplicação urlManager
. O
gerenciador de URL fornece o método
parseRequest() para analisar um requisição
de entrada a fim de obter uma rota e seus parâmetros associados; e o método
createUrl() para criar uma URL a partir de
uma rota e seus parâmetros associados.
Ao configurar o componente urlManager
na configuração da aplicação, poderá
fazer com que sua aplicação reconheça de forma arbitrária diversos formatos de
URL sem modificar o código existente da aplicação. Por exemplo, você pode usar o
código a seguir para criar uma URL a partir da ação post/view
:
use yii\helpers\Url;
// Url::to() chama UrlManager::createUrl() para criar uma URL
$url = Url::to(['post/view', 'id' => 100]);
Dependendo da configuração da propriedade urlManager
, a URL criada pode ser
parecida com um dos formatos a seguir (ou até mesmo com outro formato). E se a
URL criada for requerida, ainda será analisada a fim de obter a rota e os valores
dos parâmetros originais.
/index.php?r=post/view&id=100
/index.php/post/100
/posts/100
O gerenciador de URL suporta dois formatos de URL: o formato de URL padrão e o formato de URL amigável (pretty URL).
O formato de URL padrão usa um parâmetro chamado r
para representar a rota e
os demais parâmetros representam os parâmetros associados a rota. Por exemplo, a
URL /index.php?r=post/view&id=100
representa a rota post/view
e o parâmetro
id
com o valor 100. O formato de URL padrão não exige qualquer tipo de
configuração no gerenciador de URL e trabalha em qualquer
servidor Web.
O formato de URL amigável (pretty URL) usa um caminho adicional após o nome do
script de entrada para representar a rota e seus parâmetros. Por exemplo, o
caminho adicional na URL /index.php/post/100
é /post/100
, onde pode
representar, em uma adequada regra de URL, a rota
post/view
e o parâmetro id
com o valor 100. Para usar o formato de URL
amigável (pretty URL), você precisará escrever um conjunto de
regras de URLs de acordo com a necessidade sobre
como as URLs devem parecer.
Você pode alterar entre os dois formatos de URLs, alternando a propriedade enablePrettyUrl do gerenciador de URL sem alterar qualquer código na aplicação.
O roteamento envolve duas etapas. Na primeira etapa, a requisição de entrada é transformada em uma rota e seus parâmetros. Na segunda etapa, a ação do controller (controlador) correspondente a rota analisada será criada para processar a requisição.
Ao utilizar o formato de URL padrão, a análise de uma requisição para obter uma
rota é tão simples como pegar o valor via GET
do parâmetro r
.
Ao utilizar o formato de URL amigável (pretty URL), o gerenciado de URL examinará as regras de URLs registradas para encontrar alguma correspondência que poderá resolver a requisição em uma rota. Se tal regra não for encontrada, uma exceção yii\web\NotFoundHttpException será lançada.
Uma vez que a requisição analisada apresentar uma rota, é hora de criar a ação
do controller (controlador) identificado pela rota.
A rota é dividida em várias partes pelo separador barra (“/”). Por exemplo, a
rota site/index
será dividida em duas partes: site
e index
. Cada parte é
um ID que pode referenciar a um módulo, um controller (controlador) ou uma ação.
A partir da primeira parte da rota, a aplicação executará as seguintes etapas para
criar o módulo (se existir), o controller (controlador) e a ação:
Nas etapas acima, se ocorrer qualquer erro, uma exceção yii\web\NotFoundHttpException será lançada, indicando a falha no processo de roteamento.
Quando uma requisição analisada apresentar uma rota vazia, a assim chamada
rota padrão será usada em seu lugar. A rota padrão é site/index
é utilizada
como padrão, que referencia a ação index
do controller (controlador) site
.
Você pode personalizar esta configuração pela propriedade
defaultRoute na configuração da aplicação
como mostrado a seguir:
[
// ...
'defaultRoute' => 'main/index',
];
catchAll
¶Às vezes você pode querer colocar sua aplicação Web em modo de manutenção temporariamente e exibir uma mesma página com informações para todas as requisições. Existem muitas maneiras de atingir este objetivo. Mas uma das maneiras mais simples é apenas configurar a propriedade yii\web\Application::$catchAll na configuração da aplicação como mostrado a seguir:
[
// ...
'catchAll' => ['site/offline'],
];
Na configuração acima, a ação site/offline
será utilizado para lidar com todas
as requisições recebidas.
A propriedade catchAll
deve receber um array, o primeiro elemento especifica a
rota e o restante dos elementos (pares de nomes e seus valores) especificam os
parâmetros a serem associados a ação.
O Yii fornece um método de ajuda yii\helpers\Url::to() para criar vários tipos de URLs a partir de determinadas rotas e seus parâmetros. Por exemplo,
use yii\helpers\Url;
// cria uma URL para uma rota: /index.php?r=post/index
echo Url::to(['post/index']);
// cria uma URL para uma rota com parâmetros: /index.php?r=post/view&id=100
echo Url::to(['post/view', 'id' => 100]);
// cria uma URL ancorada: /index.php?r=post/view&id=100#content
echo Url::to(['post/view', 'id' => 100, '#' => 'content']);
// cria uma URL absoluta: https://www.example.com/index.php?r=post/index
echo Url::to(['post/index'], true);
// cria uma URL absoluta usando https: https://www.example.com/index.php?r=post/index
echo Url::to(['post/index'], 'https');
Observe que nos exemplos acima, assumimos o formato de URL padrão. Se o formato de URL amigável (pretty URL) estiver habilitado, as URLs criadas serão diferentes de acordo com as regra de URL em uso.
A rota passada para o método yii\helpers\Url::to() é sensível ao contexto. Ele pode ser tanto uma rota relativa quanto uma rota absoluta que será normalizada de acordo com as regras a seguir:
A partir da versão 2.0.2, você pode especificar uma rota usando alias. Se for este o caso, a alias será a primeira a ser convertida em uma rota real e em seguida será transformada em uma rota absoluta de acordo com as regras informadas anteriormente.
Por exemplo, assumindo que o módulo atual é admin
e o controller (controlador)
atual é post
,
use yii\helpers\Url;
// rota atual requerida: /index.php?r=admin/post/index
echo Url::to(['']);
// uma rota relativa com apenas o ID da ação: /index.php?r=admin/post/index
echo Url::to(['index']);
// uma rota relativa: /index.php?r=admin/post/index
echo Url::to(['post/index']);
// uma rota absoluta: /index.php?r=post/index
echo Url::to(['/post/index']);
// /index.php?r=post/index assumindo que a alias "@posts" foi definida como "/post/index"
echo Url::to(['@posts']);
O método yii\helpers\Url::to() é implementado através das chamadas dos métodos createUrl() e createAbsoluteUrl() do gerenciador de URL. Nas próximas subseções, iremos explicar como configurar o gerenciador de URL para personalizar os formatos das URLs criadas.
O método yii\helpers\Url::to() também suporta a criação de URLs NÃO relacionadas a uma rota em particular. Neste caso, ao invés de passar um array como seu primeiro parâmetro, você pode passar uma string. Por exemplo,
use yii\helpers\Url;
// rota atual requerida: /index.php?r=admin/post/index
echo Url::to();
// uma alias da URL: https://example.com
Yii::setAlias('@example', 'https://example.com/');
echo Url::to('@example');
// uma URL absoluta: https://example.com/images/logo.gif
echo Url::to('/images/logo.gif', true);
Além do método to()
, a classe auxiliar yii\helpers\Url também fornece uma
série de métodos referentes a criação de URLs. Por exemplo,
use yii\helpers\Url;
// URL da página inicial: /index.php?r=site/index
echo Url::home();
// URL base, útil se a aplicação for implementada em uma subpasta da pasta raiz do servidor Web
echo Url::base();
// A URL canônica da requisição atual
// Veja mais detalhes em https://en.wikipedia.org/wiki/Canonical_link_element
echo Url::canonical();
// Obtêm a URL da requisição anterior para reutilizá-la em requisições futuras
Url::remember();
echo Url::previous();
Para utilizar as URLs amigáveis (pretty URLs), configure o componente urlManager
na configuração da aplicação conforme o exemplo a seguir:
[
'components' => [
'urlManager' => [
'enablePrettyUrl' => true,
'showScriptName' => false,
'enableStrictParsing' => false,
'rules' => [
// ...
],
],
],
]
A propriedade enablePrettyUrl é obrigatória para habilitar o formato de URL amigável (pretty URL). O restante das propriedades são opcionais. No entanto, a configuração utilizada acima é a mais utilizada.
/index.php/post/100
, definindo esta propriedade
como false
, a URL /post/100
será gerada.Observação: A fim de esconder o nome do script de entrada nas URLs criadas, além de definir a propriedade showScriptName como
false
, você também pode precisar configurar o seu servidor Web para identificar corretamente o script PHP quando uma URL da requisição não especificar um explicitamente. Caso você estejam utilizando um servidor Web com Apache, você pode consultar a configuração recomendada, conforme descrito na seção Installation.
Uma regra de URL é uma instância de yii\web\UrlRule ou de classes que as estendam. Cada regra de URL consiste de um padrão usado para combinar as partes do caminho das URLs, como uma rota e alguns parâmetros. Uma regra de URL pode ser usado para analisar uma URL da requisição somente se o padrão corresponder com a esta URL. Uma regra de URL pode ser usada para criar uma URL para corresponder a uma determinada rota e seus parâmetros.
Quando uma formato de URL amigável (pretty URL) estiver habilitada, o gerenciador de URL utilizará as regras de URLs declaradas na propriedade rules para analisar as requisições e para criar URLs. Em particular, para analisar uma requisição, o gerenciador de URL verificará as regras na ordem em que foram declaradas e só enxergará a primeira regra que corresponda a URL da requisição. A regra que foi correspondida é então utilizada para obter a rota e seus parâmetros a partir de sua análise. A mesma coisa acontece na criação de URLs, o gerenciador de URL enxergará apenas a primeira regra que corresponda a rota e seus parâmetros para serem utilizados na criação de uma URL.
Você pode configurar a propriedade yii\web\UrlManager::$rules com um array
composto de pares de chave-valor, onde a chave é o padrão da regra e o valor
serão as rotas. Cada par de padrão-rota define uma regra de URL. Por exemplo,
as regras a seguir configuram duas regras de URL.
A primeira regra corresponde a uma URL chamada posts
sendo mapeado para utilizar
a rota post/index
.
A segunda regra corresponde a uma URL que combine com expressão regular
post/(\d+)
seguido de um parâmetro chamado id
sendo mapeado para utilizar a
rota post/view
.
[
'posts' => 'post/index',
'post/<id:\d+>' => 'post/view',
]
Informação: O padrão em uma regra é usado para identificar o caminho de uma URL. Por exemplo, o caminho da URL
/index.php/post/100?source=ad
épost/100
(as barras (“/”) iniciais e finais serão ignoradas) combinando com o padrãopost/(\d+)
.
Além de declarar regras de URL como pares de padrão-rota, você também pode declarar como array. Cada array é utilizado para configurar um único objeto da regra de URL. Isto se faz necessário quando você deseja configurar outras propriedades de uma regra de URL. Por exemplo,
[
// ...outras regras de URL...
[
'pattern' => 'posts',
'route' => 'post/index',
'suffix' => '.json',
],
]
Por padrão, se você não especificar a opção class
na configuração de uma regra,
será utilizado a classe yii\web\UrlRule.
Uma regra de URL pode ser associado a alguns parâmetros nomeados que são
especificados no padrão <ParamName:RegExp>
, onde o ParamName
especifica o
nome do parâmetro e o RegExp
especifica uma expressão regular opcional usada
para corresponder ao valor do parâmetro. Se o RegExp
são for especificado,
significará que o valor do parâmetro será uma string sem barras (“/”).
Observação: Você apenas pode especificar expressões regulares para os parâmetros. As demais partes de um padrão serão considerados como um texto simples.
Quando esta regra for utilizada para analisar uma URL, os parâmetros associados
serão preenchidos com os valores que foram correspondidos pela regra e estes
parâmetros serão disponibilizados logo a seguir no $_GET
pelo componente da
aplicação request
.
Vamos utilizar alguns exemplos para demonstrar como os parâmetros nomeados funcionam. Supondo que declaramos as três regras a seguir:
[
'posts/<year:\d{4}>/<category>' => 'post/index',
'posts' => 'post/index',
'post/<id:\d+>' => 'post/view',
]
Quando as regras forem utilizadas para analisar as URLs:
/index.php/posts
obterá a rota post/index
usando a segunda regra;/index.php/posts/2014/php
obterá a rota post/index
, o parâmetro year
cujo o valor é 2014 e o parâmetro category
cujo valor é php
usando a primeira regra;/index.php/post/100
obterá a rota post/view
e o parâmetro id
cujo valor/index.php/posts/php
causará uma exceção yii\web\NotFoundHttpException
quando a propriedade yii\web\UrlManager::$enableStrictParsing for true
,
por não ter correspondido a nenhum dos padrões. Se a propriedade
yii\web\UrlManager::$enableStrictParsing for false
(o valor padrão), o
caminho posts/php
será retornado como uma rota.E quando as regras fores utilizadas para criar as URLs:
Url::to(['post/index'])
cria /index.php/posts
usando a segunda regra;Url::to(['post/index', 'year' => 2014, 'category' => 'php'])
cria /index.php/posts/2014/php
usando a primeira regra;Url::to(['post/view', 'id' => 100])
cria /index.php/post/100
usando a terceira regra;Url::to(['post/view', 'id' => 100, 'source' => 'ad'])
cria /index.php/post/100?source=ad
usando a terceira regra.
Pela razão do parâmetro source
não foi especificado na regra, ele será acrescentado como uma query string na criação da URL.Url::to(['post/index', 'category' => 'php'])
cria /index.php/post/index?category=php
usando nenhuma das regras.
Observe que, se nenhuma das regras forem aplicadas, a URL será criada simplesmente
como a rota sendo o caminho e todos os parâmetros como query string.Você pode incorporar nomes de parâmetros na rota de uma regra de URL. Isto permite
que uma regra de URL seja utilizada para combinar diversas rotas. Por exemplo, a
regra a seguir incorpora os parâmetros controller
e action
nas rotas.
[
'<controller:(post|comment)>/<id:\d+>/<action:(create|update|delete)>' => '<controller>/<action>',
'<controller:(post|comment)>/<id:\d+>' => '<controller>/view',
'<controller:(post|comment)>s' => '<controller>/index',
]
Para analisar uma URL /index.php/comment/100/create
, a primeira regra será
aplicada, na qual foi definida o parâmetro controller
para ser comment
e o
parâmetro action
para ser create
. Sendo assim, a rota <controller>/<action>
é resolvida como comment/create
.
De forma similar, para criar uma URL com a rota comment/index
, a terceira regra
será aplicada, criando um URL /index.php/comments
.
Informação: Pela parametrização de rotas, é possível reduzir significativamente o número de regras de URL, que também pode melhorar o desempenho do gerenciador de URL.
Por padrão, todos os parâmetros declarados nas regras são obrigatórios. Se uma URL da requisição não contiver um dos parâmetros em particular, ou se a URL está sendo criado sem um dos parâmetros em particular, a regra não será aplicada. Para fazer com que algum parâmetro em particular seja opcional, você pode configurar a propriedade defaults da regra. Os parâmetros listados nesta propriedade são opcionais e serão utilizados quando os mesmos não forem fornecidos.
A declaração da regra a seguir, ambos os parâmetros page
e tag
são opcionais
e utilizarão o valor 1 e a string vazia, respectivamente, quando não forem fornecidos.
[
// ...outras regras...
[
'pattern' => 'posts/<page:\d+>/<tag>',
'route' => 'post/index',
'defaults' => ['page' => 1, 'tag' => ''],
],
]
A regra anterior pode ser usado para analisar ou criar qualquer uma das seguintes URLs:
/index.php/posts
: page
é 1, tag
é ''./index.php/posts/2
: page
é 2, tag
is ''./index.php/posts/2/news
: page
é 2, tag
é 'news'
./index.php/posts/news
: page
é 1, tag
é 'news'
.Sem o uso dos parâmetros opcionais, você deveria criar 4 regras para alcançar o mesmo resultado.
É possível incluir domínios nos padrões das regras de URL. Isto é útil quando sua
aplicação se comporta de forma diferente em diferentes domínios. Por exemplo, a
regra a seguir obtém a rota admin/user/login
pela análise da URL
https://admin.example.com/login
e a rota site/login
pela análise da URL
https://www.example.com/login
.
[
'https://admin.example.com/login' => 'admin/user/login',
'https://www.example.com/login' => 'site/login',
]
Você também pode incorporar parâmetros nos domínios para extrair informações
dinamicamente a partir deles. Por exemplo, a regra a seguir obtém a rota
post/index
e o parâmetro language=en
pela análise da URL https://en.example.com/posts
[
'http://<language:\w+>.example.com/posts' => 'post/index',
]
Observação: Regras com domínios NÃO devem ser incluídos com subpastas do script de entrada em seus padrões. Por exemplo, se a aplicação estiver sob
https://www.example.com/sandbox/blog
, você deve usar o padrãohttps://www.example.com/posts
ao invés dehttps://www.example.com/sandbox/blog/posts
. Isto permite que sua aplicação seja implantado sob qualquer diretório sem a necessidade de alterar o código da aplicação.
Você pode querer adicionar sufixos nas URLs para diversos fins. Por exemplo,
você pode adicionar o .html
nas URLs para que se pareçam com páginas estáticas.
Você também pode adicionar o .json
nas URLs para indicar o tipo de conteúdo
esperado na resposta. Você pode alcançar este objetivo configurando a propriedade
yii\web\UrlManager::$suffix na configuração da aplicação conforme o exemplo a seguir:
[
'components' => [
'urlManager' => [
'enablePrettyUrl' => true,
'showScriptName' => false,
'enableStrictParsing' => true,
'suffix' => '.html',
'rules' => [
// ...
],
],
],
]
A configuração anterior permitirá que o gerenciador de URL
reconheçam as URLs solicitadas e que também criem URLs com o prefixo .html
.
Dica: Você pode definir
/
como o sufixo para que todas as URLs terminem com barra.
Observação: Ao configurar um sufixo da URL e a URL da requisição não conter um, será considerado como uma URL não válida. Isto é uma prática recomendada no SEO (otimização para mecanismos de pesquisa, do inglês search engine optimization).
Ás vezes você poder querer utilizar diferentes sufixos para diferentes URLs.
Isto pode ser alcançado pela configuração da propriedade
suffix individualmente para cada regra de URL.
Quando uma regra de URL possuir esta propriedade definida, sobrescreverá o
sufixo que foi definido da camada do gerenciador de URL.
Por exemplo, a configuração a seguir contém uma regra de URL personalizada que
usa o .json
como sufixo ao invés do sufixo .html
definido globalmente.
[
'components' => [
'urlManager' => [
'enablePrettyUrl' => true,
'showScriptName' => false,
'enableStrictParsing' => true,
'suffix' => '.html',
'rules' => [
// ...
[
'pattern' => 'posts',
'route' => 'post/index',
'suffix' => '.json',
],
],
],
],
]
Ao implementar RESTful API, é necessário que sejam obtidas rotas diferentes pela
análise de uma mesma URL de acordo com o método HTTP utilizado. Isto pode ser
alcançado facilmente adicionando o prefixo do método HTTP suportado, separando
os nomes dos métodos por vírgulas. Por exemplo, a regra a seguir possui o mesmo
padrão post/<id:\d+>
com suporte a diferentes métodos HTTP. A análise de uma
requisição PUT post/100
obterá a rota post/create
, enquanto a requisição
GET post/100
obterá a rota post/view
.
[
'PUT,POST post/<id:\d+>' => 'post/create',
'DELETE post/<id:\d+>' => 'post/delete',
'post/<id:\d+>' => 'post/view',
]
Observação: Se uma regra de URL contiver método(s) HTTP, esta regra só será utilizada para análises de URLs. A regra será ignorada quando o gerenciador de URL for chamado para criar URLs.
Dica: Para simplificar o roteamento do RESTful APIs, o Yii fornece uma classe especial yii\rest\UrlRule de regras que é muito diferente. Esta classe suporta muitos recursos como a pluralização automática de IDs do controller (controlador). Para mais detalhes, por favor, consulte a seção Routing sobre o desenvolvimento de RESTful APIs.
Nos exemplo anteriores, as regras de URL são declaradas principalmente no formato de pares de padrão-rota. Este é um formato de atalho bastante utilizado. Em alguns cenários, você pode querer personalizar uma regra de URL configurando outras propriedades, tais como o yii\web\UrlRule::$suffix. Isto pode ser feito utilizando um array de configuração para especificar uma regra. O exemplo a seguir foi retirado da subseção Sufixos da URL,
[
// ...outras regras de URL...
[
'pattern' => 'posts',
'route' => 'post/index',
'suffix' => '.json',
],
]
Informações: Por padrão, se você não especificar a opção
class
na configuração de uma regra, será usado como padrão a classe yii\web\UrlRule.
As regras de URL podem ser adicionadas dinamicamente ao gerenciador de URL. Esta técnica muitas vezes se faz necessária em módulos que são redistribuídos e que desejam gerenciar as suas próprias regras de URL. Para que estas regras sejam adicionadas dinamicamente e terem efeito durante o processo de roteamento, você pode adiciona-los durante a inicialização (bootstrapping). Para os módulos, significa que deve implementar a interface yii\base\BootstrapInterface e adicionar as regras no método bootstrap() conforme o exemplo a seguir:
public function bootstrap($app)
{
$app->getUrlManager()->addRules([
// declare as regras aqui
], false);
}
Observe que você também deve listar estes módulos no yii\web\Application::bootstrap() para que eles sejam usados no processo de inicialização (bootstrapping)
Apesar do fato que a classe padrão yii\web\UrlRule é flexível o suficiente
para a maior parte dos projetos, há situações em que você terá que criar a sua
própria classe de regra. Por exemplo, em um site de venda de carros, você pode
querer dar suporte a um formato de URL como /Manufacturer/Model
, que tanto o
Manufacturer
quanto o Model
devem coincidir com os dados armazenados em uma
tabela do banco de dados. A classe de regra padrão não vai funcionar nesta
situação pois vão se basear em padrões estaticamente declarados.
Podemos criar uma classe de regra de URL para resolver este formato.
namespace app\components;
use yii\web\UrlRuleInterface;
use yii\base\BaseObject;
class CarUrlRule extends BaseObject implements UrlRuleInterface
{
public function createUrl($manager, $route, $params)
{
if ($route === 'car/index') {
if (isset($params['manufacturer'], $params['model'])) {
return $params['manufacturer'] . '/' . $params['model'];
} elseif (isset($params['manufacturer'])) {
return $params['manufacturer'];
}
}
return false; // esta regra não se aplica
}
public function parseRequest($manager, $request)
{
$pathInfo = $request->getPathInfo();
if (preg_match('%^(\w+)(/(\w+))?$%', $pathInfo, $matches)) {
// checa o $matches[1] e $matches[3] para verificar
// se coincidem com um *fabricante* e um *modelo* no banco de dados.
// Caso coincida, define o $params['manufacturer'] e/ou $params['model']
// e retorna ['car/index', $params]
}
return false; // esta regra não se aplica
}
}
E utilize esta nova classe de regra na configuração yii\web\UrlManager::$rules:
[
// ...outras regras...
[
'class' => 'app\components\CarUrlRule',
// ...configurar outras propriedades...
],
]
Ao desenvolver uma aplicação Web complexa, é importante otimizar as regras de URL para que leve menos tempo na análise de requisições e criação de URLs.
Utilizando rotas parametrizadas, você reduz o número de regras de URL, na qual pode melhorar significativamente o desempenho.
Na análise e criação de URLs, o gerenciador de URL examina as regras de URL na ordem em que foram declaradas. Portanto, você pode considerar ajustar a ordem destas regras, fazendo com que as regras mais específicas e/ou mais comuns sejam colocadas antes que os menos.
Se algumas regras de URL compartilharem o mesmo prefixo em seus padrões ou rotas, você pode considerar utilizar o yii\web\GroupUrlRule para que sejam examinados de forma mais eficiente pelo gerenciador de URL como um grupo. Normalmente é o caso de aplicações compostos por módulos, onde cada módulo possui o seu próprio conjunto de regras de URL utilizando o ID do módulo como prefixo comum.
Found a typo or you think this page needs improvement?
Edit it on github !
Signup or Login in order to comment.