Skip to content

Latest commit

 

History

History
406 lines (306 loc) · 20 KB

developer-guide.md

File metadata and controls

406 lines (306 loc) · 20 KB

Guia do Desenvolvedor

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

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.

Bibliotecas PHP Utilizadas

Ver arquivo composer.json

Bibliotecas Javascript Utilizadas

Ver bibliotecas javascript utilizadas no tema.

Arquivo de Configuração

App

Traits

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).

Traits Genéricos

Os traits genéricos podem ser usados em qualquer classe do sistema.

Singleton

Implementa o design pattern singleton. É utilizada nas classes App, GuestUser, ApiOutput, Controller entre outras.

MagicGetter

MagicSetter

MagicCallers

Model

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).

Classe Entity

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.

Traits das Entidades

  • 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.

Verificação de Permissões

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.

Método canUser

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;
    }
}

Método genericPermissionVerification

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;
}

Validações das Entidades

Controller

View

Temas

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.

Bibliotecas Javascript utilizadas no tema

Por enquanto ainda não utilizamos um gerenciador de pacotes para as bibliotecas Javascript. Estas ficam na pasta assets/vendor/.

theme.php

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.

Estrutura de pastas do tema

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

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.

Url da página

Para uma página cujo nome de arquivo é nome-da-pagina.md, a url de acesso será http://mapasculturais/page/site/nome-da-pagina/

Título da página

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 ....

Sidebars

O Conteúdo das sidebars estão nos arquivos _right.md e _left.md

Substituindo uma sidebar

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 ....

Extendendo uma sidebar

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 ....

Layouts

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

Visões

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.

Visões das actions single, create e edit

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; ?>

Partes

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'); ?>

Enviando variáveis para dentro das partes

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>

Assets

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().

Método Asset

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.

Adicionando uma imagem

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'); ?>" />
Criando um link para um asset

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>

Método enqueueStyle

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.

Adicionando um estilo

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');

Método enqueueScript

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.

Adicionando um script

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');
Adicionando um script que depende de outro

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');

Ordem de impressão das tags de estilos e scripts

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

Variáveis Acessíveis

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)

Verificando se um usuário está logado

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; ?>

Exceções

PermissionDenied