O intuíto deste documento é dar uma visão panorâmica da arquitetura e funcionamento do Mapas Culturais para quem quiser colaborar no desenvolvimento da plataforma. Este documento está ainda incompleto e em constante desenvolvimento.
- Introdução
- Arquivo de Configuração
- App
- Traits
- Model
- Controller
- View
- Autenticação
- Roles
- Log
- Cache
- Outputs da API
- Exceções
O mínimo requerido para rodar o Mapas Culturais é PHP >= 5.4, PostgreSQL >= 9.1 com PostGIS >= 2.1.
As seguintes extensões do PHP são requeridas: gd, apc, zip, curl, pgsql, phar, pdo_pgsql.
Ver arquivo composer.json
- Slim - Microframework em cima do qual foi escria a classe App do MapasCulturais.
- Doctrine/ORM - ORM utilizado para o mapeamento das entidades.
- Opauth/OpenId - Utilizado para autenticação via OpenId.
- respect/validation - Utilizado para as validações das propriedades e metadados das entidades.
- smottt/wideimage - Utilizado para transformar imagens (criar thumbnails, por exemplo).
- phpunit/phpunit - Utilizado para testes.
- creof/doctrine2-spatial - Faz o mapeamento de várias procedures do PostGIS para o doctrine.
- mustache/mustache - Utilizado para renderizar alguns templates.
- phpoffice/phpword - Utilizado para criar .docs ou .xls onde necessário.
- michelf/php-markdown - Utilizado para renderizar os markdowns das páginas
Ver bibliotecas javascript utilizadas no tema.
Os traits ficam no namespace MapasCulturais\Traits e seus arquivos na pasta src/protected/application/lib/MapasCulturais/Traits.
Se houver no nome do trait um prefixo (Entity, Controller ou Repository) significa que este trait só deve ser utilizado em classes que estendam a classe com o nome do prefixo dentro do namespace MapasCulturais (ex: o trait EntityAvatar só deve ser utilizado em classes que estendem a classe MapasCulturais\Entity). Já se não houver um prefixo significa que é um trait genérico e que pode ser utilizado em qualquer classe (exemplos: Singleton e MagigGetter).
Os traits genéricos podem ser usados em qualquer classe do sistema.
Implementa o design pattern singleton. É utilizada nas classes App, GuestUser, ApiOutput, Controller entre outras.
As classes de modelo ficam no namespace MapasCulturais\Entities e seus arquivos dentro da pasta src/protected/application/lib/MapasCulturais/Entities.
Estas classes devem estender a classe abstrata MapasCulturais\Entity e usar os Docblock Annotations do Doctrine para fazer o mapeamento com a representação desta entidade no banco de dados (geralmente uma tabela).
Estas podem também usar os traits criados para entidades (os que têm o prefixo Entity no nome, como por exmplo o EntityFiles, que é para ser usado em entidades que têm arquivos anexos).
A classe abstrata MapasCulturais\Entity é a classe que serve de base para todoas as entidades do sistema. Implementa uma série de métodos úteis para, entre outros, verificação de permissões, serialização e validações.
- EntityAgentRelation - Deve ser usado em entidades que podem ter agentes relacionados. Requer uma entidade auxiliar com o mesmo nome da entidade acrescida do sufixo AgentRelation (exemplo: para a entidade Event, uma classe EventAgentRelation).
- EntityFiles - Deve ser usado em entidades que podem ter arquivos anexados.
- EntityAvatar - Deve ser usado em entidades que tenham avatar. Requer o trait EntityFiles.
- EntityGeoLocation - Deve ser usado em entidades georreferenciadas. Requer as propriedades location, do tipo point, e _geoLocation, do tipo geography.
- EntityMetadata - Deve ser usado em entidades que tenham metadados. Requer de uma entidade auxiliar. Se existir no mesmo namespace uma classe com o nome da entidade acrescida do sufixo Meta (exemplo: para a entidade Agent, uma classe AgentMeta), esta será usada, senão a entidade Metadata será usada como auxiliar.
- EntityMetaLists - Deve ser usado em entidades que tenham metadados com múltiplos valores por chave. (exemplo de uso: links).
- EntityNested - Deve ser usado em entidades hierarquicas. Requer as associações autoreferenciadas children e parent.
- EntityOwnerAgent - Deve ser usado em entidades que tenham a associação ManyToOne owner apontando para a entidade MapasCulturais\Entity\Agent. Requer também um mapeamento do tipo int chamado _ownerId que representa o id do agente que é dono desta entidade.
- EntitySoftDelete - Usado em entidades que necessitem de lixeira. Requer um mapeamento do tipo int chamado status.
- EntityTaxonomies - Deve ser usado em entidades que precisem de taxonomias (tags, área de atuação, etc.).
- EntityTypes - Deve ser usado em entidades que tenham tipos. Requer um mapeamento do tipo int chamado _type.
- EntityVerifiable - Deve ser usado em entidades verificáveis, o seja, que podem ser marcadas como oficiais pelos admins ou membros da equipe.
A verificação das permissões são feitas através do método checkPermission passando como parâmetro para este o nome da ação que você deseja checar se o usuário tem ou não permissão para executar. Este método, por ua vez, chama o método canUser que retornará um booleando true se o usuário pode executar a ação ou false se o usuário não pode executar a ação. Caso o usuário não possa executar a ação, o método checkPermission lançará uma exceção do tipo PermissionDenied.
O método canUser recebe como primeiro parâmetro o nome da ação e opcionalmente, como segundo parâmetro, um usuário. Se nenhum usuário for enviado, será usado o usuário logado ou guest. O retorno desta função é um booleano indicando se o usuário pode ou não executar a ação.
Este método procurará por um método auxilar chamado canUser acrescido do nome da ação (exemplo: para a ação remove, um método chamado canUserRemove) e caso não ache será usado o método genericPermissionVerification.
No exemplo a seguir dizemos que somente admins podem alterar o satatus da entidade Exemplo.
class Exemplo extends MapasCulturais\Entity{
use MapasCulturais\Traits\MagicSetter
....
....
protected $_status = 0;
function setStatus($status){
$this->checkPermission('modifyStatus');
$this->_status = $status;
$this->save();
}
protected function canUserModifyStatus($user){
if($user->is("admin"))
return true;
else
return false;
}
}
Este método é utilizado sempre que uma checagem de permissão é feita e o método canUser não encontra um método auxiliar com o nome da ação.
O corpo deste método é o seguinte:
protected function genericPermissionVerification($user){
if($user->is('guest'))
return false;
if($user->is('admin'))
return true;
if($this->getOwnerUser()->id == $user->id)
return true;
if($this->usesAgentRelation() && $this->userHasControl($user))
return true;
return false;
}
Por enquanto ainda não temos resolvida a estrutura para múltiplos temas. O que temos é um tema único dentro da pasta src/protected/application/themes/active, que será modificado para aceitar configurações.
Por enquanto ainda não utilizamos um gerenciador de pacotes para as bibliotecas Javascript. Estas ficam na pasta assets/vendor/.
Este arquivo fica na pasta raíz do tema (src/protected/application/themes/active) e é usado para colocar funções helpers usadas dentro do tema e para estender o sistema utilizando a API de plugins.
dentro da pasta raíz do tema
- assets/ - onde deve ficar tudo que é acessível pelo público dentro da url /public do site
- css/
- fonts/
- img/
- vendor/
- layouts/ - onde ficam os layouts do site
- parts/ - onde ficam os template parts utilizados pelo tema
- views/ - onde ficam as viões dos controles
- pages/ - onde ficam os arquivos de páginas
As páginas do sistema são arquivos .md (Markdown) salvos dentro da pasta pages/ do tema. Para criar uma nova página basta criar um novo arquivo .md dentro desta pasta. Estes arquivos são renderizadas pela biblioteca PHP Markdown Extra.
Para uma página cujo nome de arquivo é nome-da-pagina.md, a url de acesso será http://mapasculturais/page/site/nome-da-pagina/
O texto do primeiro h1 do conteúdo da página será utilizado como título da página (tag title). Isto é feito via javascript.
No exemplo a seguir o título da página será Título da Págna
# Título da Página
Conteúdo da página ....
O Conteúdo das sidebars estão nos arquivos _right.md e _left.md
Você pode substituir uma sidebar envolvendo o conteúdo que você deseja que substitua o conteúdo padrão com as tags <%left left%> para a sidebar da esquerda e <%right right%> para a sidebar da direita.
No exemplo a seguir substituimos a sidebar da direita por um menu com três links:
<%right
- [Primeiro link](#primeiro)
- [Segundo link](#segundo)
- [Terceiro link](#terceiro)
right%>
# Título da Página
Conteúdo da página ....
Você pode extender uma sidebar, adicionando conteúdo antes ou depois do conteúdo padrão, colocando um :after ou :before logo depois da tag de abertura.
No exemplo a seguir extendemos a sidebar da esquerda adicionando um menu com 2 links no final da sidebar:
<%left:after
## submenu da página
- [Primeiro Link](#primeiro)
- [Segundo Link](#segundo)
left%>
# Título da Página
Conteúdo da página ....
O layout é a "moldura" do conteúdo de uma visão. A estrutura mínima de um layout é a seguinte:
<html>
<head>
<title><?php echo isset($entity) ? $this->getTitle($entity) : $this->getTitle() ?></title>
<?php mapasculturais_head(isset($entity) ? $entity : null); ?>
</head>
<body>
<?php body_header(); ?>
<?php echo $TEMPLATE_CONTENT; /* aqui entra o conteúdo da view */ ?>
<?php body_footer(); ?>
</body>
</html>
Por padrão as visões usam o arquivo de layout default.php, mas você pode definir qual layout elas usarão colocando a seguinte linha na primeira linha do seu arquivo de visão:
$this->layout = 'nome-do-layout'; // não precisa do .php no nome do template
As visões são chamadas de dentro das actions do controller através do método render, que inclui o layout definido na view, ou do método partial, que não inclui o layout.
Quando a visão é chamada pelo método render, o conteúdo renderizado da visão é guardado na variável $TEMPLATE_CONTENT e enviado para o layout.
Os arquivos de visão single.php, create.php e edit.php dos controladores agent, space, event e project são, não relidade, o mesmo arquivo. O arquivo real é o single.php e os dois outros são links simbólicos para o primeiro.
Para saber, de dentro de um destes arquivos, em qual action você está, você pode usar a propriedade $this->controller->action:
<?php if($this->controller->action == 'single'): ?>
<p>você está visualizando a entidade</p>
<?php elseif($this->controller->action == 'edit'): ?>
<p>você está editando a entidade</p>
<?php else: ?>
<p>você está criando uma nova entidade<p>
<?php endif; ?>
Se você só deseja saber se está no modo de edição use a função is_editable():
<?php if(is_editable(): ?>
<p> você está em modo de edição (edit ou create). </p>
<?php else: ?>
<p> você está somente visualizando a entidade. <p>
<?php endif; ?>
As partes são blocos de código que podem ser incluidos em diferentes views, layouts ou mesmo dentro de outras partes. Estes blocos de código devem ficar, por padrão, na pasta layouts/parts/ do tema.
Para usar uma parte cujo nome de arquivo é uma-parte.php basta chamar o método part da seguinte forma:
<div> A parte será incluida a seguir: </div>
<?php $this->part('uma-parte'); ?>
Você pode enviar variáveis para usar dentro das partes. Isto é útil em várias situações, por exmplo quando você quer que uma parte seja usada dentro de um loop e você tem que enviar o item atual do loop para usar dentro da parte.
No exemplo a seguir, passamos uma variável chamada user_name, com o valor "Fulano de Tal", para dentro da parte uma-parte.
// dentro de algum arquivo de view, layout ou mesmo outra parte
$this->part('uma-parte', ['user_name' => 'Fulano de Tal']);
<!-- dentro do arquivo layouts/parts/uma-parte.php -->
<span>Nome de usuário: <?php echo $user_name; ?></span>
Os assets são arquivos estáticos (.css, .js, imagens, etc.) utilizados pelo tema.
Para imprimir a url de um asset use a função $this->asset(). Já se você deseja adicionar um js ou css use as funções $app->enqueueScript() e $app->enqueueStyle().
O Método asset do objeto de função serve para imprimir ou somente retornar a url de um asset. Este método aceita dois parâmetros:
O primeiro, $file, é o caminho do arquivo deseja dentro da pasta assets do tema, como exemplo a string "img/uma-image.jpg".
O Segundo, $print, é opcional e tem como padrão true. Se for passado false a função somente retornará a url, mas não imprimirá nada.
O exemplo a seguir usa uma imagem chamada logo.png que está na pasta assets/img/ do tema.
<img src="<?php $this->asset('img/logo.png'); ?>" />
O exemplo a seguir cria um link para o arquivo documento.pdf que está na pasta asset/ do tema.
<a href="<?php $this->asset('documento.pdf'); ?>" >Documento</a>
Este método é utilizado para adicionar arquivos .css que serão utilizados pela visão, layout ou parte. Este método aceitas 5 parâmetros ($group, $script_name, $script_filename, array $dependences, $media), sendo os dois último opcional.
Há três grupos de estilos no sistema: vendor, que são estilos utilizados pelas bibliotecas, fonts que são as fontes utilizadas, e app, que são os estilos escritos exclusivamente para o tema.
O exemplo a seguir adiciona um estilo chamado um-estilo.css escrito para a aplicação.
$app->enqueueStyle('app', 'um-estilo', 'css/um-estilo.css');
Este método é utilizado para adicionar arquivos .js que serão utilizados pela visão, layout ou parte. Este método aceitas 4 parâmetros ($group, $script_name, $script_filename, array $dependences), sendo o último opcional.
Há dois grupos de scripts no sistema: vendor, que são as bibliotecas utilizadas, e app, que são os scripts escritos exclusivamente para o tema.
O exemplo a seguir adiciona um script chamado um-script.js escrito para a aplicação.
$app->enqueueScript('app', 'um-script', 'js/um-script.js');
O exemplo a seguir adiciona uma biblioteca que depende de outra biblioteca.
$app->enqueueScript('vendor', 'jquery-ui-datepicker', '/vendor/jquery-ui.datepicker.js', array('jquery'));
$app->enqueueScript('vendor', 'jquery', '/vendor/jquery/jquery-2.0.3.min.js');
Os grupos de estilos e scripts serão impressos na seguinte ordem e dentro dos grupos os estilos/scripts serão ordenados conforme suas dependências:
- Estilos do grupo vendor
- Estilos do grupo font
- Estilos do grupo app
- Scripts do grupo vendor
- Scripts do grupo app
De dentro dos arquivos das visões (views, layouts e parts) as seguintes variáveis estão acessíveis:
- $this - instância da classe MapasCulturais\View.
- $this->assetUrl - url dos assets.
- $this->baseUrl - url da raíz do site.
- $this->controller - o controller que mandou renderizar a visão.
- $this->controller->action - a action que mandou renderizar a visão.
- $app - instância da classe MapasCulturais\App.
- $app->user - o usuário que estã vendo o site. Este objeto é instância da classe MapasCulturais\Entities\User, se o usuário estiver logado, ou instância da classe MapasCulturais\GuestUser, se o usuário não estiver logado.
- $app->user->profile - o agente padrão do usuário. Instância da classe MapasCulturais\Entities\Agent. (somente para usuários logados)
- $entity - é a entidade que está sendo visualizada, editada ou criada. (somente para as actions single, edit e create dos controladores das entidades agent, space, project e event. Dentro das partes somente se esta foi enviada)
Para saber se um usuário está logado você pode verificar se o usuário não é guest.
<?php if( ! $app->user->is('guest') ): ?>
<p>O usuário está logado e o nome do agente padrão dele é <?php echo $app->user->profile->name; ?> ?></p>
<?php else: ?>
<p>O usuário não está logado</p>
<?php endif; ?>