mirror of https://github.com/interlegis/sapl.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
548 lines
21 KiB
548 lines
21 KiB
import django_filters
|
|
import logging
|
|
|
|
from crispy_forms.layout import Fieldset, Layout
|
|
|
|
from django import forms
|
|
from django.contrib.contenttypes.models import ContentType
|
|
from django.core.exceptions import ValidationError
|
|
from django.db import transaction
|
|
from django.db.models import Q
|
|
from django.forms import ModelForm
|
|
from django.utils.translation import ugettext_lazy as _
|
|
from django.utils import timezone
|
|
|
|
from sapl.base.models import Autor, TipoAutor
|
|
from sapl.comissoes.models import (Comissao, Composicao,
|
|
DocumentoAcessorio, Participacao,
|
|
Periodo, Reuniao)
|
|
from sapl.crispy_layout_mixin import (form_actions, SaplFormHelper,
|
|
to_row)
|
|
from sapl.materia.models import MateriaEmTramitacao, PautaReuniao
|
|
from sapl.parlamentares.models import (Legislatura, Mandato,
|
|
Parlamentar)
|
|
from sapl.utils import (FileFieldCheckMixin,
|
|
FilterOverridesMetaMixin, validar_arquivo)
|
|
|
|
|
|
class ComposicaoForm(forms.ModelForm):
|
|
|
|
comissao = forms.CharField(
|
|
required=False, label='Comissao', widget=forms.HiddenInput())
|
|
logger = logging.getLogger(__name__)
|
|
|
|
class Meta:
|
|
model = Composicao
|
|
exclude = []
|
|
|
|
def __init__(self, user=None, **kwargs):
|
|
super(ComposicaoForm, self).__init__(**kwargs)
|
|
self.fields['comissao'].widget.attrs['disabled'] = 'disabled'
|
|
|
|
def clean(self):
|
|
data = super().clean()
|
|
data['comissao'] = self.initial['comissao']
|
|
comissao_pk = self.initial['comissao'].id
|
|
|
|
if not self.is_valid():
|
|
return data
|
|
|
|
periodo = data['periodo']
|
|
|
|
if periodo.data_fim:
|
|
intersecao_periodo = Composicao.objects.filter(
|
|
Q(periodo__data_inicio__lte=periodo.data_fim, periodo__data_fim__gte=periodo.data_fim) |
|
|
Q(periodo__data_inicio__gte=periodo.data_inicio, periodo__data_fim__lte=periodo.data_inicio),
|
|
comissao_id=comissao_pk)
|
|
else:
|
|
intersecao_periodo = Composicao.objects.filter(
|
|
Q(periodo__data_inicio__gte=periodo.data_inicio, periodo__data_fim__lte=periodo.data_inicio),
|
|
comissao_id=comissao_pk)
|
|
|
|
if intersecao_periodo:
|
|
if periodo.data_fim:
|
|
self.logger.warning(
|
|
'O período informado ({} a {}) choca com períodos já cadastrados para esta comissão'.format(
|
|
periodo.data_inicio, periodo.data_fim
|
|
)
|
|
)
|
|
else:
|
|
self.logger.warning(
|
|
'O período informado ({} - ) choca com períodos já cadastrados para esta comissão'.format(
|
|
periodo.data_inicio
|
|
)
|
|
)
|
|
raise ValidationError('O período informado choca com períodos já cadastrados para esta comissão')
|
|
|
|
return data
|
|
|
|
|
|
class PeriodoForm(forms.ModelForm):
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
class Meta:
|
|
model = Periodo
|
|
exclude = []
|
|
|
|
def clean(self):
|
|
cleaned_data = super(PeriodoForm, self).clean()
|
|
|
|
if not self.is_valid():
|
|
return cleaned_data
|
|
|
|
data_inicio = cleaned_data['data_inicio']
|
|
data_fim = cleaned_data['data_fim']
|
|
|
|
if data_fim and data_fim < data_inicio:
|
|
self.logger.warning(
|
|
'A Data Final ({}) é menor que '
|
|
'a Data Inicial({}).'.format(data_fim, data_inicio)
|
|
)
|
|
raise ValidationError('A Data Final não pode ser menor que '
|
|
'a Data Inicial')
|
|
|
|
# Evita NoneType exception se não preenchida a data_fim
|
|
if not data_fim:
|
|
data_fim = data_inicio
|
|
|
|
legislatura = Legislatura.objects.filter(data_inicio__lte=data_inicio,
|
|
data_fim__gte=data_fim,
|
|
)
|
|
|
|
if not legislatura:
|
|
self.logger.warning(
|
|
'O período informado ({} a {})'
|
|
'não está contido em uma única '
|
|
'legislatura existente'.format(data_inicio, data_fim)
|
|
)
|
|
raise ValidationError('O período informado '
|
|
'deve estar contido em uma única '
|
|
'legislatura existente')
|
|
|
|
return cleaned_data
|
|
|
|
|
|
class ParticipacaoCreateForm(forms.ModelForm):
|
|
|
|
logger = logging.getLogger(__name__)
|
|
parent_pk = forms.CharField(required=False) # widget=forms.HiddenInput())
|
|
|
|
class Meta:
|
|
model = Participacao
|
|
fields = '__all__'
|
|
exclude = ['composicao']
|
|
|
|
def __init__(self, user=None, **kwargs):
|
|
super(ParticipacaoCreateForm, self).__init__(**kwargs)
|
|
|
|
if self.instance:
|
|
comissao = kwargs['initial']
|
|
comissao_pk = int(comissao['parent_pk'])
|
|
composicao = Composicao.objects.get(id=comissao_pk)
|
|
participantes = composicao.participacao_set.all()
|
|
id_part = [p.parlamentar.id for p in participantes]
|
|
else:
|
|
id_part = []
|
|
|
|
qs = self.create_participacao()
|
|
|
|
parlamentares = Mandato.objects.filter(qs,
|
|
parlamentar__ativo=True
|
|
).prefetch_related('parlamentar').\
|
|
values_list('parlamentar',
|
|
flat=True
|
|
).distinct()
|
|
qs = Parlamentar.objects.filter(id__in=parlamentares).distinct().\
|
|
exclude(id__in=id_part)
|
|
eligible = self.verifica()
|
|
result = list(set(qs) & set(eligible))
|
|
if result == eligible:
|
|
self.fields['parlamentar'].queryset = qs
|
|
else:
|
|
ids = [e.id for e in eligible]
|
|
qs = Parlamentar.objects.filter(id__in=ids)
|
|
self.fields['parlamentar'].queryset = qs
|
|
|
|
def clean(self):
|
|
cleaned_data = super(ParticipacaoCreateForm, self).clean()
|
|
|
|
if not self.is_valid():
|
|
return cleaned_data
|
|
|
|
data_designacao = cleaned_data['data_designacao']
|
|
data_desligamento = cleaned_data['data_desligamento']
|
|
|
|
if data_desligamento and \
|
|
data_designacao > data_desligamento:
|
|
self.logger.warning(
|
|
'Data de designação ({}) superior '
|
|
'à data de desligamento ({})'.format(data_designacao, data_desligamento)
|
|
)
|
|
raise ValidationError(_('Data de designação não pode ser superior '
|
|
'à data de desligamento'))
|
|
|
|
composicao = Composicao.objects.get(id=self.initial['parent_pk'])
|
|
cargos_unicos = [
|
|
c.cargo.nome for c in composicao.participacao_set.filter(cargo__unico=True)]
|
|
|
|
if cleaned_data['cargo'].nome in cargos_unicos:
|
|
msg = _('Este cargo é único para esta Comissão.')
|
|
self.logger.warning(
|
|
'Este cargo ({}) é único para esta Comissão.'.format(
|
|
cleaned_data['cargo'].nome
|
|
)
|
|
)
|
|
raise ValidationError(msg)
|
|
return cleaned_data
|
|
|
|
def create_participacao(self):
|
|
composicao = Composicao.objects.get(id=self.initial['parent_pk'])
|
|
data_inicio_comissao = composicao.periodo.data_inicio
|
|
data_fim_comissao = composicao.periodo.data_fim if composicao.periodo.data_fim else timezone.now()
|
|
q1 = Q(data_fim_mandato__isnull=False,
|
|
data_fim_mandato__gte=data_inicio_comissao)
|
|
q2 = Q(data_inicio_mandato__gte=data_inicio_comissao) \
|
|
& Q(data_inicio_mandato__lte=data_fim_comissao)
|
|
q3 = Q(data_fim_mandato__isnull=True,
|
|
data_inicio_mandato__lte=data_inicio_comissao)
|
|
qs = q1 | q2 | q3
|
|
return qs
|
|
|
|
def verifica(self):
|
|
composicao = Composicao.objects.get(id=self.initial['parent_pk'])
|
|
participantes = composicao.participacao_set.all()
|
|
participantes_id = [p.parlamentar.id for p in participantes]
|
|
parlamentares = Parlamentar.objects.all().exclude(
|
|
id__in=participantes_id).order_by('nome_completo')
|
|
parlamentares = [p for p in parlamentares if p.ativo]
|
|
|
|
lista = []
|
|
|
|
for p in parlamentares:
|
|
mandatos = p.mandato_set.all()
|
|
for m in mandatos:
|
|
data_inicio = m.data_inicio_mandato
|
|
data_fim = m.data_fim_mandato
|
|
comp_data_inicio = composicao.periodo.data_inicio
|
|
comp_data_fim = composicao.periodo.data_fim
|
|
if (data_fim and data_fim >= comp_data_inicio)\
|
|
or (data_inicio >= comp_data_inicio and data_inicio <= comp_data_fim)\
|
|
or (data_fim is None and data_inicio <= comp_data_inicio):
|
|
lista.append(p)
|
|
|
|
lista = list(set(lista))
|
|
|
|
return lista
|
|
|
|
|
|
class ParticipacaoEditForm(forms.ModelForm):
|
|
|
|
logger = logging.getLogger(__name__)
|
|
parent_pk = forms.CharField(required=False) # widget=forms.HiddenInput())
|
|
nome_parlamentar = forms.CharField(required=False, label='Parlamentar')
|
|
|
|
class Meta:
|
|
model = Participacao
|
|
fields = ['nome_parlamentar', 'parlamentar', 'cargo', 'titular',
|
|
'data_designacao', 'data_desligamento',
|
|
'motivo_desligamento', 'observacao']
|
|
widgets = {
|
|
'parlamentar': forms.HiddenInput(),
|
|
}
|
|
|
|
def __init__(self, user=None, **kwargs):
|
|
super(ParticipacaoEditForm, self).__init__(**kwargs)
|
|
self.initial['nome_parlamentar'] = Parlamentar.objects.get(
|
|
id=self.initial['parlamentar']).nome_parlamentar
|
|
self.fields['nome_parlamentar'].widget.attrs['disabled'] = 'disabled'
|
|
|
|
def clean(self):
|
|
cleaned_data = super(ParticipacaoEditForm, self).clean()
|
|
|
|
if not self.is_valid():
|
|
return cleaned_data
|
|
|
|
data_designacao = cleaned_data['data_designacao']
|
|
data_desligamento = cleaned_data['data_desligamento']
|
|
|
|
if data_desligamento and \
|
|
data_designacao > data_desligamento:
|
|
self.logger.warning(
|
|
'Data de designação ({}) superior '
|
|
'à data de desligamento ({})'.format(data_designacao, data_desligamento)
|
|
)
|
|
raise ValidationError(_('Data de designação não pode ser superior '
|
|
'à data de desligamento'))
|
|
|
|
composicao_id = self.instance.composicao_id
|
|
|
|
composicao = Composicao.objects.get(id=composicao_id)
|
|
cargos_unicos = [c.cargo.nome for c in
|
|
composicao.participacao_set.filter(cargo__unico=True).exclude(id=self.instance.pk)]
|
|
|
|
if cleaned_data['cargo'].nome in cargos_unicos:
|
|
msg = _('Este cargo é único para esta Comissão.')
|
|
self.logger.warning(
|
|
'Este cargo ({}) é único para esta Comissão (id={}).'.format(
|
|
cleaned_data['cargo'].nome, composicao_id
|
|
)
|
|
)
|
|
raise ValidationError(msg)
|
|
|
|
return cleaned_data
|
|
|
|
|
|
class ComissaoForm(forms.ModelForm):
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
class Meta:
|
|
model = Comissao
|
|
fields = '__all__'
|
|
|
|
def __init__(self, user=None, **kwargs):
|
|
super(ComissaoForm, self).__init__(**kwargs)
|
|
inst = self.instance
|
|
if inst.pk:
|
|
if inst.tipo.natureza == 'P':
|
|
self.fields['apelido_temp'].widget.attrs['disabled'] = 'disabled'
|
|
self.fields['data_instalacao_temp'].widget.attrs['disabled'] = 'disabled'
|
|
self.fields['data_final_prevista_temp'].widget.attrs['disabled'] = 'disabled'
|
|
self.fields['data_prorrogada_temp'].widget.attrs['disabled'] = 'disabled'
|
|
self.fields['data_fim_comissao'].widget.attrs['disabled'] = 'disabled'
|
|
|
|
def clean(self):
|
|
super(ComissaoForm, self).clean()
|
|
|
|
if not self.is_valid():
|
|
return self.cleaned_data
|
|
|
|
if len(self.cleaned_data['nome']) > 100:
|
|
msg = _('Nome da Comissão informado ({}) tem mais de 50 caracteres.'.format(
|
|
self.cleaned_data['nome']))
|
|
self.logger.warning(
|
|
'Nome da Comissão deve ter no máximo 50 caracteres.'
|
|
)
|
|
raise ValidationError(msg)
|
|
if (self.cleaned_data['data_extincao'] and
|
|
self.cleaned_data['data_extincao'] <
|
|
self.cleaned_data['data_criacao']):
|
|
msg = _('Data de extinção não pode ser menor que a de criação')
|
|
self.logger.warning(
|
|
'Data de extinção ({}) não pode ser menor que a de criação ({}).'.format(
|
|
self.cleaned_data['data_extincao'], self.cleaned_data['data_criacao']
|
|
)
|
|
)
|
|
raise ValidationError(msg)
|
|
if (self.cleaned_data['data_final_prevista_temp'] and
|
|
self.cleaned_data['data_final_prevista_temp'] <
|
|
self.cleaned_data['data_criacao']):
|
|
msg = _('Data Prevista para Término não pode ser menor que a de criação')
|
|
self.logger.warning(
|
|
'Data Prevista para Término ({}) não pode ser menor que a de criação ({}).'.format(
|
|
self.cleaned_data['data_final_prevista_temp'], self.cleaned_data['data_criacao']
|
|
)
|
|
)
|
|
raise ValidationError(msg)
|
|
if (self.cleaned_data['data_prorrogada_temp'] and
|
|
self.cleaned_data['data_prorrogada_temp'] <
|
|
self.cleaned_data['data_criacao']):
|
|
msg = _('Data Novo Prazo não pode ser menor que a de criação')
|
|
self.logger.warning(
|
|
'Data Novo Prazo ({}) não pode ser menor que a de criação ({}).'.format(
|
|
self.cleaned_data['data_prorrogada_temp'], self.cleaned_data['data_criacao']
|
|
)
|
|
)
|
|
raise ValidationError(msg)
|
|
if (self.cleaned_data['data_instalacao_temp'] and
|
|
self.cleaned_data['data_instalacao_temp'] <
|
|
self.cleaned_data['data_criacao']):
|
|
msg = _('Data de Instalação não pode ser menor que a de criação')
|
|
self.logger.warning(
|
|
'Data de Instalação ({}) não pode ser menor que a de criação ({}).'.format(
|
|
self.cleaned_data['data_instalacao_temp'], self.cleaned_data['data_criacao']
|
|
)
|
|
)
|
|
raise ValidationError(msg)
|
|
if (self.cleaned_data['data_final_prevista_temp'] and self.cleaned_data['data_instalacao_temp'] and
|
|
self.cleaned_data['data_final_prevista_temp'] <
|
|
self.cleaned_data['data_instalacao_temp']):
|
|
msg = _(
|
|
'Data Prevista para Término não pode ser menor que a de Instalação.')
|
|
self.logger.warning(
|
|
'Data Prevista para Término ({}) não pode ser menor que a de Instalação ({}).'.format(
|
|
self.cleaned_data['data_final_prevista_temp'], self.cleaned_data['data_instalacao_temp']
|
|
)
|
|
)
|
|
raise ValidationError(msg)
|
|
if (self.cleaned_data['data_prorrogada_temp'] and self.cleaned_data['data_instalacao_temp'] and
|
|
self.cleaned_data['data_prorrogada_temp'] <
|
|
self.cleaned_data['data_instalacao_temp']):
|
|
msg = _('Data Novo Prazo não pode ser menor que a de Instalação.')
|
|
self.logger.warning(
|
|
'Data Novo Prazo ({}) não pode ser menor que a de Instalação ({}).'.format(
|
|
self.cleaned_data['data_prorrogada_temp'], self.cleaned_data['data_instalacao_temp']
|
|
)
|
|
)
|
|
raise ValidationError(msg)
|
|
return self.cleaned_data
|
|
|
|
@transaction.atomic
|
|
def save(self, commit=True):
|
|
inst = self.instance
|
|
if not inst.pk:
|
|
comissao = super(ComissaoForm, self).save(commit)
|
|
content_type = ContentType.objects.get_for_model(Comissao)
|
|
object_id = comissao.pk
|
|
tipo = TipoAutor.objects.get(content_type=content_type)
|
|
nome = comissao.sigla + ' - ' + comissao.nome
|
|
Autor.objects.create(
|
|
content_type=content_type,
|
|
object_id=object_id,
|
|
tipo=tipo,
|
|
nome=nome
|
|
)
|
|
return comissao
|
|
else:
|
|
comissao = super(ComissaoForm, self).save(commit)
|
|
return comissao
|
|
|
|
|
|
class ReuniaoForm(ModelForm):
|
|
|
|
logger = logging.getLogger(__name__)
|
|
comissao = forms.ModelChoiceField(queryset=Comissao.objects.all(),
|
|
widget=forms.HiddenInput())
|
|
|
|
class Meta:
|
|
model = Reuniao
|
|
exclude = ['cod_andamento_reuniao']
|
|
|
|
def clean(self):
|
|
super(ReuniaoForm, self).clean()
|
|
|
|
if not self.is_valid():
|
|
return self.cleaned_data
|
|
|
|
if self.cleaned_data['hora_fim']:
|
|
if (self.cleaned_data['hora_fim'] <
|
|
self.cleaned_data['hora_inicio']):
|
|
msg = _(
|
|
'A hora de término da reunião não pode ser menor que a de início')
|
|
self.logger.warning(
|
|
"A hora de término da reunião ({}) não pode ser menor que a de início ({}).".format(
|
|
self.cleaned_data['hora_fim'], self.cleaned_data['hora_inicio']
|
|
)
|
|
)
|
|
raise ValidationError(msg)
|
|
|
|
upload_pauta = self.cleaned_data.get('upload_pauta', False)
|
|
upload_ata = self.cleaned_data.get('upload_ata', False)
|
|
upload_anexo = self.cleaned_data.get('upload_anexo', False)
|
|
|
|
if upload_pauta:
|
|
validar_arquivo(upload_pauta, "Pauta da Reunião")
|
|
|
|
if upload_ata:
|
|
validar_arquivo(upload_ata, "Ata da Reunião")
|
|
|
|
if upload_anexo:
|
|
validar_arquivo(upload_anexo, "Anexo da Reunião")
|
|
|
|
return self.cleaned_data
|
|
|
|
|
|
class PautaReuniaoFilterSet(django_filters.FilterSet):
|
|
|
|
class Meta(FilterOverridesMetaMixin):
|
|
model = MateriaEmTramitacao
|
|
fields = ['materia__tipo', 'materia__ano', 'materia__numero', 'materia__data_apresentacao']
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
super(PautaReuniaoFilterSet, self).__init__(*args, **kwargs)
|
|
|
|
self.filters['materia__tipo'].label = "Tipo da Matéria"
|
|
self.filters['materia__ano'].label = "Ano da Matéria"
|
|
self.filters['materia__numero'].label = "Número da Matéria"
|
|
self.filters['materia__data_apresentacao'].label = "Data (Inicial - Final)"
|
|
|
|
row1 = to_row([('materia__tipo', 4), ('materia__ano', 4), ('materia__numero', 4)])
|
|
row2 = to_row([('materia__data_apresentacao', 12)])
|
|
|
|
self.form.helper = SaplFormHelper()
|
|
self.form.helper.form_method = "GET"
|
|
self.form.helper.layout = Layout(
|
|
Fieldset(
|
|
_("Pesquisa de Matérias"), row1, row2,
|
|
form_actions(label="Pesquisar")
|
|
)
|
|
)
|
|
|
|
|
|
class PautaReuniaoForm(forms.ModelForm):
|
|
|
|
class Meta:
|
|
model = PautaReuniao
|
|
exclude = ['reuniao']
|
|
|
|
|
|
class DocumentoAcessorioCreateForm(FileFieldCheckMixin, forms.ModelForm):
|
|
|
|
parent_pk = forms.CharField(required=False) # widget=forms.HiddenInput())
|
|
|
|
class Meta:
|
|
model = DocumentoAcessorio
|
|
exclude = ['reuniao']
|
|
|
|
def __init__(self, user=None, **kwargs):
|
|
super(DocumentoAcessorioCreateForm, self).__init__(**kwargs)
|
|
|
|
if self.instance:
|
|
reuniao = Reuniao.objects.get(id=self.initial['parent_pk'])
|
|
comissao = reuniao.comissao
|
|
comissao_pk = comissao.id
|
|
documentos = reuniao.documentoacessorio_set.all()
|
|
return self.create_documentoacessorio()
|
|
|
|
def create_documentoacessorio(self):
|
|
reuniao = Reuniao.objects.get(id=self.initial['parent_pk'])
|
|
|
|
def clean(self):
|
|
super(DocumentoAcessorioCreateForm, self).clean()
|
|
|
|
if not self.is_valid():
|
|
return self.cleaned_data
|
|
|
|
arquivo = self.cleaned_data.get('arquivo', False)
|
|
|
|
if arquivo:
|
|
validar_arquivo(arquivo, "Texto Integral")
|
|
|
|
return self.cleaned_data
|
|
|
|
|
|
class DocumentoAcessorioEditForm(FileFieldCheckMixin, forms.ModelForm):
|
|
|
|
parent_pk = forms.CharField(required=False) # widget=forms.HiddenInput())
|
|
|
|
class Meta:
|
|
model = DocumentoAcessorio
|
|
fields = ['nome', 'data', 'autor', 'ementa',
|
|
'indexacao', 'arquivo']
|
|
|
|
def __init__(self, user=None, **kwargs):
|
|
super(DocumentoAcessorioEditForm, self).__init__(**kwargs)
|
|
|
|
def clean(self):
|
|
super(DocumentoAcessorioEditForm, self).clean()
|
|
|
|
if not self.is_valid():
|
|
return self.cleaned_data
|
|
|
|
arquivo = self.cleaned_data.get('arquivo', False)
|
|
|
|
if arquivo:
|
|
validar_arquivo(arquivo, "Texto Integral")
|
|
|
|
return self.cleaned_data
|
|
|