Browse Source

Merge pull request #8 from interlegis/master

30/08/2017
pull/1440/head
Rogério Frá 7 years ago
committed by GitHub
parent
commit
45d84ec808
  1. 5
      .gitignore
  2. 1
      .travis.yml
  3. 26
      check_migrations.sh
  4. 16
      create_admin.py
  5. 2
      docker-compose.yml
  6. 3
      fix_qa.sh
  7. 4
      genkey.py
  8. 0
      media/.gitkeep
  9. 1
      requirements/requirements.txt
  10. 54
      sapl/api/forms.py
  11. 14
      sapl/api/serializers.py
  12. 6
      sapl/api/urls.py
  13. 8
      sapl/api/views.py
  14. 73
      sapl/base/forms.py
  15. 20
      sapl/base/migrations/0008_auto_20170814_1409.py
  16. 4
      sapl/base/models.py
  17. 12
      sapl/base/templatetags/common_tags.py
  18. 3
      sapl/base/tests/teststub_urls.py
  19. 3
      sapl/base/urls.py
  20. 145
      sapl/base/views.py
  21. 20
      sapl/comissoes/migrations/0002_auto_20170809_1236.py
  22. 2
      sapl/comissoes/models.py
  23. 6
      sapl/comissoes/views.py
  24. 2
      sapl/compilacao/compilacao_data_tables.sql
  25. 65
      sapl/compilacao/forms.py
  26. 21
      sapl/compilacao/migrations/0002_auto_20170825_1108.py
  27. 33
      sapl/compilacao/migrations/0003_auto_20170825_1136.py
  28. 18
      sapl/compilacao/models.py
  29. 5
      sapl/compilacao/templatetags/compilacao_filters.py
  30. 41
      sapl/compilacao/tests/test_tipo_texto_articulado_form.py
  31. 470
      sapl/compilacao/views.py
  32. 2
      sapl/crispy_layout_mixin.py
  33. 1
      sapl/crud/base.py
  34. 40
      sapl/decorators.py
  35. 4
      sapl/legacy/migracao_documentos.py
  36. 20
      sapl/legacy/migration.py
  37. 10
      sapl/legacy/scripts/street_sweeper.py
  38. 1
      sapl/materia/admin.py
  39. 3
      sapl/materia/apps.py
  40. 3
      sapl/materia/email_utils.py
  41. 192
      sapl/materia/forms.py
  42. 19
      sapl/materia/migrations/0012_auto_20170829_1321.py
  43. 5
      sapl/materia/models.py
  44. 5
      sapl/materia/signals.py
  45. 3
      sapl/materia/tests/test_materia.py
  46. 28
      sapl/materia/urls.py
  47. 144
      sapl/materia/views.py
  48. 3
      sapl/norma/apps.py
  49. 11
      sapl/norma/forms.py
  50. 2
      sapl/norma/models.py
  51. 3
      sapl/norma/signals.py
  52. 78
      sapl/norma/tests/test_norma.py
  53. 4
      sapl/painel/urls.py
  54. 11
      sapl/painel/views.py
  55. 123
      sapl/parlamentares/forms.py
  56. 19
      sapl/parlamentares/migrations/0005_auto_20170814_1615.py
  57. 16
      sapl/parlamentares/models.py
  58. 123
      sapl/parlamentares/tests/test_parlamentares.py
  59. 3
      sapl/parlamentares/urls.py
  60. 106
      sapl/parlamentares/views.py
  61. 60
      sapl/protocoloadm/forms.py
  62. 166
      sapl/protocoloadm/tests/test_protocoloadm.py
  63. 70
      sapl/protocoloadm/views.py
  64. 1
      sapl/redireciona_urls/tests.py
  65. 20
      sapl/redireciona_urls/urls.py
  66. 9
      sapl/redireciona_urls/views.py
  67. 33
      sapl/relatorios/templates/pdf_capa_processo_preparar_pysc.py
  68. 200
      sapl/relatorios/templates/pdf_detalhe_materia_gerar.py
  69. 59
      sapl/relatorios/templates/pdf_detalhe_materia_preparar_pysc.py
  70. 31
      sapl/relatorios/templates/pdf_documento_administrativo_preparar_pysc.py
  71. 3
      sapl/relatorios/templates/pdf_espelho_gerar.py
  72. 28
      sapl/relatorios/templates/pdf_espelho_preparar_pysc.py
  73. 30
      sapl/relatorios/templates/pdf_etiqueta_protocolo_preparar_pysc.py
  74. 32
      sapl/relatorios/templates/pdf_materia_gerar.py
  75. 42
      sapl/relatorios/templates/pdf_materia_preparar_pysc.py
  76. 24
      sapl/relatorios/templates/pdf_norma_gerar.py
  77. 29
      sapl/relatorios/templates/pdf_norma_preparar_pysc.py
  78. 66
      sapl/relatorios/templates/pdf_ordem_dia_preparar_pysc.py
  79. 4
      sapl/relatorios/templates/pdf_pauta_sessao_gerar.py
  80. 81
      sapl/relatorios/templates/pdf_pauta_sessao_preparar_pysc.py
  81. 39
      sapl/relatorios/templates/pdf_protocolo_gerar.py
  82. 27
      sapl/relatorios/templates/pdf_protocolo_preparar_pysc.py
  83. 10
      sapl/relatorios/templates/pdf_sessao_plenaria_gerar.py
  84. 125
      sapl/relatorios/templates/pdf_sessao_plenaria_preparar_pysc.py
  85. 23
      sapl/relatorios/views.py
  86. 5
      sapl/rules/map_rules.py
  87. 70
      sapl/sessao/forms.py
  88. 25
      sapl/sessao/migrations/0010_auto_20170810_1033.py
  89. 25
      sapl/sessao/migrations/0010_auto_20170814_1804.py
  90. 20
      sapl/sessao/migrations/0011_auto_20170814_1409.py
  91. 20
      sapl/sessao/migrations/0011_auto_20170814_1849.py
  92. 23
      sapl/sessao/migrations/0012_auto_20170814_1615.py
  93. 21
      sapl/sessao/migrations/0012_auto_20170815_1244.py
  94. 16
      sapl/sessao/migrations/0013_merge.py
  95. 27
      sapl/sessao/models.py
  96. 1
      sapl/sessao/serializers.py
  97. 68
      sapl/sessao/tests/test_sessao.py
  98. 106
      sapl/sessao/views.py
  99. 4
      sapl/settings.py
  100. 2
      sapl/static/js/app.js

5
.gitignore

@ -91,9 +91,12 @@ whoosh_index
collected_static collected_static
bower bower
bower_components bower_components
media
whoosh/ whoosh/
solr-4.10.2/ solr-4.10.2/
postgres-data/ postgres-data/
data/ data/
solr-*/ solr-*/
# ignora tudo dentro de media, mas cria a pasta no checkout
media/*
!media/.gitkeep

1
.travis.yml

@ -14,6 +14,7 @@ before_script:
- cp sapl/.env_test sapl/.env - cp sapl/.env_test sapl/.env
- psql -c "CREATE USER sapl WITH PASSWORD 'sapl'" -U postgres; - psql -c "CREATE USER sapl WITH PASSWORD 'sapl'" -U postgres;
- psql -c "CREATE DATABASE sapl OWNER sapl;" -U postgres - psql -c "CREATE DATABASE sapl OWNER sapl;" -U postgres
- ./check_migrations.sh
script: script:
- ./manage.py migrate - ./manage.py migrate

26
check_migrations.sh

@ -0,0 +1,26 @@
#!/bin/bash
# TODO: Após migrar para Django 1.10 usar
#
# ./manage.py makemigrations --check --dry-run
#
# ATENÇÃO: a chamada atual termina com exit 1 se NÃO HÁ migração a ser aplicada
# ou seja, termina com "erro" se está tudo bem!
# A chamada do django 1.10 INVERTE ISSO.
#
# https://docs.djangoproject.com/en/1.10/ref/django-admin/#cmdoption-makemigrations-check
python manage.py makemigrations --dry-run --exit
MIGRATIONS=$?
NC='\033[0m'
if [ $MIGRATIONS -eq 0 ]; then
RED='\033[0;31m'
echo
echo -e "${RED}ALGUMAS ALTERAÇÕES EXIGEM MIGRAÇÃO.${NC}"
echo -e "${RED}RODE 'python manage.py makemigrations' ANTES DE SUBMETER SEU CÓDIGO...${NC}"
echo
exit 1
fi

16
create_admin.py

@ -1,28 +1,34 @@
import os import os
import sys import sys
import django import django
from sapl import settings
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "sapl.settings") os.environ.setdefault("DJANGO_SETTINGS_MODULE", "sapl.settings")
def create_superuser(): def create_superuser():
from django.contrib.auth.models import User from django.contrib.auth.models import User
username = "admin" username = "admin"
password = os.environ['ADMIN_PASSWORD'] if 'ADMIN_PASSWORD' in os.environ else None password = os.environ[
'ADMIN_PASSWORD'] if 'ADMIN_PASSWORD' in os.environ else None
email = os.environ['ADMIN_EMAIL'] if 'ADMIN_EMAIL' in os.environ else '' email = os.environ['ADMIN_EMAIL'] if 'ADMIN_EMAIL' in os.environ else ''
if User.objects.filter(username=username).exists(): if User.objects.filter(username=username).exists():
print("[SUPERUSER] User %s already exists. Exiting without change." % username) print("[SUPERUSER] User %s already exists."
" Exiting without change." % username)
sys.exit('ADMIN_USER_EXISTS') sys.exit('ADMIN_USER_EXISTS')
else: else:
if not password: if not password:
print("[SUPERUSER] Environment variable $ADMIN_PASSWORD for user %s was not set. Leaving..." % username) print(
"[SUPERUSER] Environment variable $ADMIN_PASSWORD"
" for user %s was not set. Leaving..." % username)
sys.exit('MISSING_ADMIN_PASSWORD') sys.exit('MISSING_ADMIN_PASSWORD')
print("[SUPERUSER] Creating superuser...") print("[SUPERUSER] Creating superuser...")
u = User.objects.create_superuser(username=username, password=password, email=email) u = User.objects.create_superuser(
username=username, password=password, email=email)
u.save() u.save()
print("[SUPERUSER] Done.") print("[SUPERUSER] Done.")

2
docker-compose.yml

@ -10,7 +10,7 @@ sapldb:
ports: ports:
- "5532:5432" - "5532:5432"
sapl: sapl:
image: interlegis/sapl:3.1.18-BETA image: interlegis/sapl:3.1.19-BETA
volumes: volumes:
- sapl_data:/var/interlegis/sapl/data - sapl_data:/var/interlegis/sapl/data
- sapl_media:/var/interlegis/sapl/media - sapl_media:/var/interlegis/sapl/media

3
fix_qa.sh

@ -8,4 +8,5 @@
# Uma forma simples de fazer isso é adicionando antes suas mudanças à # Uma forma simples de fazer isso é adicionando antes suas mudanças à
# "staging area" do git, com `git add .` e após usar o script `git diff`. # "staging area" do git, com `git add .` e após usar o script `git diff`.
isort --recursive --skip='migrations' --skip='templates' --skip='ipython_log.py' . isort --recursive --skip='migrations' --skip='templates' --skip='ipython_log.py*' .
autopep8 --in-place --recursive . --exclude='migrations,ipython_log.py*'

4
genkey.py

@ -2,7 +2,9 @@ import random
def generate_secret(): def generate_secret():
return ''.join([random.SystemRandom().choice('abcdefghijklmnopqrstuvwxyz0123456789!@#$%^&*(-_=+)') for i in range(50)]) return (''.join([random.SystemRandom().choice(
'abcdefghijklmnopqrst'
'uvwxyz0123456789!@#$%^&*(-_=+)') for i in range(50)]))
if __name__ == '__main__': if __name__ == '__main__':
print(generate_secret()) print(generate_secret())

0
media/.gitkeep

1
requirements/requirements.txt

@ -32,4 +32,5 @@ pysolr==3.6.0
python-magic==0.4.12 python-magic==0.4.12
gunicorn==19.6.0 gunicorn==19.6.0
django-reversion==2.0.8 django-reversion==2.0.8
WeasyPrint==0.30
whoosh==2.7.4 whoosh==2.7.4

54
sapl/api/forms.py

@ -1,9 +1,9 @@
from django.db.models import Q, F from django.db.models import Q
from django.forms.fields import CharField, MultiValueField from django.forms.fields import CharField, MultiValueField
from django.forms.widgets import MultiWidget, TextInput from django.forms.widgets import MultiWidget, TextInput
from django.utils import timezone from django.utils import timezone
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from django_filters.filters import MethodFilter, ModelChoiceFilter, DateFilter from django_filters.filters import DateFilter, MethodFilter, ModelChoiceFilter
from rest_framework import serializers from rest_framework import serializers
from rest_framework.compat import django_filters from rest_framework.compat import django_filters
from rest_framework.filters import FilterSet from rest_framework.filters import FilterSet
@ -168,13 +168,13 @@ class AutoresPossiveisFilterSet(FilterSet):
if not hasattr(self, filter_for_model): if not hasattr(self, filter_for_model):
return qs return qs
return getattr(self, filter_for_model)(qs, data_relativa)
def filter_parlamentar(self, queryset, data_relativa):
# não leva em conta afastamentos
if not data_relativa: if not data_relativa:
data_relativa = timezone.now() data_relativa = timezone.now()
return getattr(self, filter_for_model)(qs, data_relativa).distinct()
def filter_parlamentar(self, queryset, data_relativa):
# não leva em conta afastamentos
legislatura_relativa = Legislatura.objects.filter( legislatura_relativa = Legislatura.objects.filter(
data_inicio__lte=data_relativa, data_inicio__lte=data_relativa,
data_fim__gte=data_relativa).first() data_fim__gte=data_relativa).first()
@ -188,26 +188,42 @@ class AutoresPossiveisFilterSet(FilterSet):
if legislatura_relativa.atual(): if legislatura_relativa.atual():
params['parlamentar_set__ativo'] = True params['parlamentar_set__ativo'] = True
qs = queryset.filter(**params).distinct() qs = queryset.filter(**params)
return qs return qs
def filter_frente(self, queryset, data_relativa):
# implementar regras específicas para frente
return queryset
def filter_comissao(self, queryset, data_relativa): def filter_comissao(self, queryset, data_relativa):
# implementar regras específicas para comissao return queryset.filter(
return queryset Q(comissao_set__data_extincao__isnull=True,
comissao_set__data_fim_comissao__isnull=True) |
Q(comissao_set__data_extincao__gte=data_relativa,
comissao_set__data_fim_comissao__isnull=True) |
Q(comissao_set__data_extincao__gte=data_relativa,
comissao_set__data_fim_comissao__isnull=True) |
Q(comissao_set__data_extincao__isnull=True,
comissao_set__data_fim_comissao__gte=data_relativa) |
Q(comissao_set__data_extincao__gte=data_relativa,
comissao_set__data_fim_comissao__gte=data_relativa),
comissao_set__data_criacao__lte=data_relativa)
def filter_orgao(self, queryset, data_relativa): def filter_frente(self, queryset, data_relativa):
# implementar regras específicas para orgao return queryset.filter(
return queryset Q(frente_set__data_extincao__isnull=True) |
Q(frente_set__data_extincao__gte=data_relativa),
frente_set__data_criacao__lte=data_relativa)
def filter_bancada(self, queryset, data_relativa): def filter_bancada(self, queryset, data_relativa):
# implementar regras específicas para bancada return queryset.filter(
return queryset Q(bancada_set__data_extincao__isnull=True) |
Q(bancada_set__data_extincao__gte=data_relativa),
bancada_set__data_criacao__lte=data_relativa)
def filter_bloco(self, queryset, data_relativa): def filter_bloco(self, queryset, data_relativa):
# implementar regras específicas para bloco return queryset.filter(
Q(bloco_set__data_extincao__isnull=True) |
Q(bloco_set__data_extincao__gte=data_relativa),
bloco_set__data_criacao__lte=data_relativa)
def filter_orgao(self, queryset, data_relativa):
# na implementação, não havia regras a implementar para orgao
return queryset return queryset

14
sapl/api/serializers.py

@ -64,14 +64,18 @@ class SessaoPlenariaSerializer(serializers.ModelSerializer):
txtSiglaOrgao = serializers.SerializerMethodField('get_sigla_orgao') txtSiglaOrgao = serializers.SerializerMethodField('get_sigla_orgao')
txtApelido = serializers.SerializerMethodField('get_name') txtApelido = serializers.SerializerMethodField('get_name')
txtNomeOrgao = serializers.SerializerMethodField('get_nome_orgao') txtNomeOrgao = serializers.SerializerMethodField('get_nome_orgao')
codEstadoReuniao = serializers.SerializerMethodField('get_estadoSessaoPlenaria') codEstadoReuniao = serializers.SerializerMethodField(
'get_estadoSessaoPlenaria')
txtTipoReuniao = serializers.SerializerMethodField('get_tipo_sessao') txtTipoReuniao = serializers.SerializerMethodField('get_tipo_sessao')
txtObjeto = serializers.SerializerMethodField('get_assunto_sessao') txtObjeto = serializers.SerializerMethodField('get_assunto_sessao')
txtLocal = serializers.SerializerMethodField('get_endereco_orgao') txtLocal = serializers.SerializerMethodField('get_endereco_orgao')
bolReuniaoConjunta = serializers.SerializerMethodField('get_reuniao_conjunta') bolReuniaoConjunta = serializers.SerializerMethodField(
bolHabilitarEventoInterativo = serializers.SerializerMethodField('get_iterativo') 'get_reuniao_conjunta')
bolHabilitarEventoInterativo = serializers.SerializerMethodField(
'get_iterativo')
idYoutube = serializers.SerializerMethodField('get_url') idYoutube = serializers.SerializerMethodField('get_url')
codEstadoTransmissaoYoutube = serializers.SerializerMethodField('get_estadoTransmissaoYoutube') codEstadoTransmissaoYoutube = serializers.SerializerMethodField(
'get_estadoTransmissaoYoutube')
datReuniaoString = serializers.SerializerMethodField('get_date') datReuniaoString = serializers.SerializerMethodField('get_date')
# Constantes SessaoPlenaria (de 1-9) (apenas 3 serão usados) # Constantes SessaoPlenaria (de 1-9) (apenas 3 serão usados)
@ -106,7 +110,6 @@ class SessaoPlenariaSerializer(serializers.ModelSerializer):
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
super(SessaoPlenariaSerializer, self).__init__(args, kwargs) super(SessaoPlenariaSerializer, self).__init__(args, kwargs)
casa = CasaLegislativa.objects.first()
def get_pk_sessao(self, obj): def get_pk_sessao(self, obj):
return obj.pk return obj.pk
@ -146,6 +149,7 @@ class SessaoPlenariaSerializer(serializers.ModelSerializer):
return self.TRANSMISSAO_EM_ANDAMENTO return self.TRANSMISSAO_EM_ANDAMENTO
else: else:
return self.SEM_TRANSMISSAO return self.SEM_TRANSMISSAO
def get_assunto_sessao(self, obj): def get_assunto_sessao(self, obj):
pauta_sessao = '' pauta_sessao = ''
ordem_dia = OrdemDia.objects.filter(sessao_plenaria=obj.pk) ordem_dia = OrdemDia.objects.filter(sessao_plenaria=obj.pk)

6
sapl/api/urls.py

@ -2,9 +2,9 @@ from django.conf import settings
from django.conf.urls import include, url from django.conf.urls import include, url
from rest_framework.routers import DefaultRouter from rest_framework.routers import DefaultRouter
from sapl.api.views import (AutorListView, MateriaLegislativaViewSet, from sapl.api.views import (AutoresPossiveisListView, AutoresProvaveisListView,
ModelChoiceView, SessaoPlenariaViewSet, AutorListView, MateriaLegislativaViewSet,
AutoresPossiveisListView, AutoresProvaveisListView) ModelChoiceView, SessaoPlenariaViewSet)
from .apps import AppConfig from .apps import AppConfig

8
sapl/api/views.py

@ -7,10 +7,10 @@ from rest_framework.generics import ListAPIView
from rest_framework.mixins import ListModelMixin, RetrieveModelMixin from rest_framework.mixins import ListModelMixin, RetrieveModelMixin
from rest_framework.permissions import (AllowAny, IsAuthenticated, from rest_framework.permissions import (AllowAny, IsAuthenticated,
IsAuthenticatedOrReadOnly) IsAuthenticatedOrReadOnly)
from rest_framework.viewsets import GenericViewSet, ModelViewSet from rest_framework.viewsets import GenericViewSet
from sapl.api.forms import AutorChoiceFilterSet, AutorSearchForFieldFilterSet,\ from sapl.api.forms import (AutorChoiceFilterSet, AutoresPossiveisFilterSet,
AutoresPossiveisFilterSet AutorSearchForFieldFilterSet)
from sapl.api.serializers import (AutorChoiceSerializer, AutorSerializer, from sapl.api.serializers import (AutorChoiceSerializer, AutorSerializer,
ChoiceSerializer, ChoiceSerializer,
MateriaLegislativaSerializer, MateriaLegislativaSerializer,
@ -173,8 +173,6 @@ class AutoresProvaveisListView(ListAPIView):
serializer_class = ChoiceSerializer serializer_class = ChoiceSerializer
def get_queryset(self): def get_queryset(self):
queryset = ListAPIView.get_queryset(self)
params = {'content_type__isnull': False} params = {'content_type__isnull': False}
tipo = '' tipo = ''

73
sapl/base/forms.py

@ -1,3 +1,4 @@
import django_filters
from crispy_forms.bootstrap import FieldWithButtons, InlineRadios, StrictButton from crispy_forms.bootstrap import FieldWithButtons, InlineRadios, StrictButton
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import HTML, Button, Div, Field, Fieldset, Layout, Row from crispy_forms.layout import HTML, Button, Div, Field, Fieldset, Layout, Row
@ -7,14 +8,12 @@ from django.contrib.auth import get_user_model
from django.contrib.auth.forms import (AuthenticationForm, PasswordResetForm, from django.contrib.auth.forms import (AuthenticationForm, PasswordResetForm,
SetPasswordForm) SetPasswordForm)
from django.contrib.auth.models import Group, User from django.contrib.auth.models import Group, User
from django.contrib.auth.password_validation import validate_password
from django.contrib.contenttypes.models import ContentType from django.contrib.contenttypes.models import ContentType
from django.core.exceptions import ValidationError from django.core.exceptions import ValidationError
from django.db import models, transaction from django.db import models, transaction
from django.forms import ModelForm from django.forms import ModelForm
from django.utils.translation import string_concat
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
import django_filters from django.utils.translation import string_concat
from sapl.base.models import Autor, TipoAutor from sapl.base.models import Autor, TipoAutor
from sapl.crispy_layout_mixin import (SaplFormLayout, form_actions, to_column, from sapl.crispy_layout_mixin import (SaplFormLayout, form_actions, to_column,
@ -24,20 +23,12 @@ from sapl.sessao.models import SessaoPlenaria
from sapl.settings import MAX_IMAGE_UPLOAD_SIZE from sapl.settings import MAX_IMAGE_UPLOAD_SIZE
from sapl.utils import (RANGE_ANOS, ChoiceWithoutValidationField, from sapl.utils import (RANGE_ANOS, ChoiceWithoutValidationField,
ImageThumbnailFileInput, RangeWidgetOverride, ImageThumbnailFileInput, RangeWidgetOverride,
autor_label, autor_modal, models_with_gr_for_model) autor_label, autor_modal, models_with_gr_for_model,
qs_override_django_filter)
from .models import AppConfig, CasaLegislativa from .models import AppConfig, CasaLegislativa
ACTION_CREATE_USERS_AUTOR_CHOICE = [ ACTION_CREATE_USERS_AUTOR_CHOICE = [
('C', _('Criar novo Usuário')),
('A', _('Associar um usuário existente')),
('N', _('Autor sem Usuário de Acesso ao Sapl')),
]
ACTION_CREATE_USERS_AUTOR_CHOICE = [
('C', _('Criar novo Usuário')),
('A', _('Associar um usuário existente')), ('A', _('Associar um usuário existente')),
('N', _('Autor sem Usuário de Acesso ao Sapl')), ('N', _('Autor sem Usuário de Acesso ao Sapl')),
] ]
@ -256,42 +247,7 @@ class AutorForm(ModelForm):
if self.instance.user: if self.instance.user:
qs_user = qs_user.exclude(pk=self.instance.user.pk) qs_user = qs_user.exclude(pk=self.instance.user.pk)
if cd['action_user'] == 'C': if cd['action_user'] == 'A':
param_username = {get_user_model().USERNAME_FIELD: cd['username']}
if User.objects.filter(**param_username).exists():
raise ValidationError(
_('Já existe usuário com o username "%s". '
'Para utilizar esse username você deve selecionar '
'"Associar um usuário existente".') % cd['username'])
if ('senha' not in cd or 'senha_confirma' not in cd or
not cd['senha'] or not cd['senha_confirma']):
raise ValidationError(_(
'A senha e sua confirmação devem ser informadas.'))
msg = _('As senhas não conferem.')
self.valida_igualdade(cd['senha'], cd['senha_confirma'], msg)
try:
validate_password(self.cleaned_data['senha'])
except ValidationError as error:
raise ValidationError(error)
if ('email' not in cd or 'confirma_email' not in cd or
not cd['email'] or not cd['confirma_email']):
raise ValidationError(_(
'O email e sua confirmação devem ser informados.'))
msg = _('Os emails não conferem.')
self.valida_igualdade(cd['email'], cd['confirma_email'], msg)
if not settings.DEBUG:
if qs_user.filter(email=cd['email']).exists():
raise ValidationError(_('Este email já foi cadastrado.'))
if qs_autor.filter(user__email=cd['email']).exists():
raise ValidationError(
_('Já existe um Autor com este email.'))
elif cd['action_user'] == 'A':
param_username = {get_user_model().USERNAME_FIELD: cd['username']} param_username = {get_user_model().USERNAME_FIELD: cd['username']}
if not User.objects.filter(**param_username).exists(): if not User.objects.filter(**param_username).exists():
raise ValidationError( raise ValidationError(
@ -362,22 +318,7 @@ class AutorForm(ModelForm):
if not u.is_active: if not u.is_active:
u.is_active = settings.DEBUG u.is_active = settings.DEBUG
u.save() u.save()
elif self.cleaned_data['action_user'] == 'C':
param_username = {
get_user_model().USERNAME_FIELD: self.cleaned_data['username']}
if get_user_model().USERNAME_FIELD != 'email':
param_username['email'] = self.cleaned_data['email']
u = get_user_model().objects.create(**param_username)
u.set_password(self.cleaned_data['senha'])
# Define usuário como ativo em ambiente de desenvolvimento
# pode logar sem a necessidade de passar pela validação de email
# troque par False para testar o envio de email em desenvolvimento
u.is_active = settings.DEBUG
u.save()
autor.user = u autor.user = u
if not autor.tipo.content_type: if not autor.tipo.content_type:
@ -501,6 +442,10 @@ class RelatorioPresencaSessaoFilterSet(django_filters.FilterSet):
row1, form_actions(save_label='Pesquisar')) row1, form_actions(save_label='Pesquisar'))
) )
@property
def qs(self):
return qs_override_django_filter(self)
class RelatorioHistoricoTramitacaoFilterSet(django_filters.FilterSet): class RelatorioHistoricoTramitacaoFilterSet(django_filters.FilterSet):

20
sapl/base/migrations/0008_auto_20170814_1409.py

@ -0,0 +1,20 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.13 on 2017-08-14 14:09
from __future__ import unicode_literals
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('base', '0007_auto_20170808_0850'),
]
operations = [
migrations.AlterField(
model_name='appconfig',
name='sequencia_numeracao',
field=models.CharField(choices=[('A', 'Sequencial por ano'), ('L', 'Sequencial por legislatura'), ('U', 'Sequencial único')], default='A', max_length=1, verbose_name='Sequência de numeração'),
),
]

4
sapl/base/models.py

@ -1,16 +1,16 @@
import reversion
from django.contrib.contenttypes.fields import GenericForeignKey from django.contrib.contenttypes.fields import GenericForeignKey
from django.contrib.contenttypes.models import ContentType from django.contrib.contenttypes.models import ContentType
from django.db import models from django.db import models
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
import reversion
from sapl.utils import UF, YES_NO_CHOICES, get_settings_auth_user_model from sapl.utils import UF, YES_NO_CHOICES, get_settings_auth_user_model
TIPO_DOCUMENTO_ADMINISTRATIVO = (('O', _('Ostensivo')), TIPO_DOCUMENTO_ADMINISTRATIVO = (('O', _('Ostensivo')),
('R', _('Restritivo'))) ('R', _('Restritivo')))
SEQUENCIA_NUMERACAO = (('A', _('Sequencial por ano')), SEQUENCIA_NUMERACAO = (('A', _('Sequencial por ano')),
('L', _('Sequencial por legislatura')),
('U', _('Sequencial único'))) ('U', _('Sequencial único')))

12
sapl/base/templatetags/common_tags.py

@ -5,6 +5,7 @@ from sapl.base.models import AppConfig
from sapl.materia.models import DocumentoAcessorio, MateriaLegislativa from sapl.materia.models import DocumentoAcessorio, MateriaLegislativa
from sapl.norma.models import NormaJuridica from sapl.norma.models import NormaJuridica
from sapl.parlamentares.models import Filiacao from sapl.parlamentares.models import Filiacao
from sapl.utils import filiacao_data
register = template.Library() register = template.Library()
@ -117,6 +118,7 @@ def str2intabs(value):
except: except:
return '' return ''
@register.filter @register.filter
def has_iframe(request): def has_iframe(request):
@ -180,3 +182,13 @@ def urldetail_content_type(obj, value):
def urldetail(obj): def urldetail(obj):
return '%s:%s_detail' % ( return '%s:%s_detail' % (
obj._meta.app_config.name, obj._meta.model_name) obj._meta.app_config.name, obj._meta.model_name)
@register.filter
def filiacao_data_filter(parlamentar, data_inicio):
return filiacao_data(parlamentar, data_inicio)
@register.filter
def filiacao_intervalo_filter(parlamentar, date_range):
return filiacao_data(parlamentar, date_range[0], date_range[1])

3
sapl/base/tests/teststub_urls.py

@ -6,5 +6,4 @@ from sapl.urls import urlpatterns as original_patterns
urlpatterns = original_patterns + patterns('', url(r'^zzzz$', urlpatterns = original_patterns + patterns('', url(r'^zzzz$',
TemplateView.as_view( TemplateView.as_view(
template_name='index.html'), template_name='index.html'),
name='zzzz') name='zzzz'))
)

3
sapl/base/urls.py

@ -16,8 +16,7 @@ from .views import (AppConfigCrud, CasaLegislativaCrud, HelpView,
RelatorioMateriasPorAnoAutorTipoView, RelatorioMateriasPorAnoAutorTipoView,
RelatorioMateriasPorAutorView, RelatorioMateriasPorAutorView,
RelatorioMateriasTramitacaoView, RelatorioMateriasTramitacaoView,
RelatorioPresencaSessaoView, RelatorioPresencaSessaoView, SaplSearchView)
SaplSearchView)
app_name = AppConfig.name app_name = AppConfig.name

145
sapl/base/views.py

@ -4,25 +4,26 @@ from django.contrib.auth import get_user_model
from django.contrib.auth.mixins import PermissionRequiredMixin from django.contrib.auth.mixins import PermissionRequiredMixin
from django.contrib.auth.models import Group from django.contrib.auth.models import Group
from django.contrib.auth.tokens import default_token_generator from django.contrib.auth.tokens import default_token_generator
from django.core.exceptions import ObjectDoesNotExist
from django.core.mail import send_mail from django.core.mail import send_mail
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from django.db.models import Count, Q from django.db.models import Count
from django.http import HttpResponseRedirect from django.http import HttpResponseRedirect
from django.utils.encoding import force_bytes from django.utils.encoding import force_bytes
from django.utils.http import urlsafe_base64_decode, urlsafe_base64_encode from django.utils.http import urlsafe_base64_decode, urlsafe_base64_encode
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from django.views.generic.base import TemplateView from django.views.generic.base import TemplateView
from django_filters.views import FilterView from django_filters.views import FilterView
from haystack.views import SearchView from haystack.views import SearchView
from sapl.base.forms import AutorForm, AutorFormForAdmin, TipoAutorForm from sapl.base.forms import AutorForm, AutorFormForAdmin, TipoAutorForm
from sapl.base.models import Autor, TipoAutor from sapl.base.models import Autor, TipoAutor
from sapl.crud.base import CrudAux from sapl.crud.base import CrudAux
from sapl.materia.models import MateriaLegislativa, TipoMateriaLegislativa from sapl.materia.models import (Autoria, MateriaLegislativa,
from sapl.parlamentares.models import Parlamentar TipoMateriaLegislativa)
from sapl.sessao.models import PresencaOrdemDia, SessaoPlenaria from sapl.sessao.models import (PresencaOrdemDia, SessaoPlenaria,
from sapl.utils import sapl_logger SessaoPlenariaPresenca)
from sapl.utils import parlamentares_ativos, sapl_logger
from .forms import (CasaLegislativaForm, ConfiguracoesAppForm, from .forms import (CasaLegislativaForm, ConfiguracoesAppForm,
RelatorioAtasFilterSet, RelatorioAtasFilterSet,
@ -209,15 +210,6 @@ class RelatorioPresencaSessaoView(FilterView):
filterset_class = RelatorioPresencaSessaoFilterSet filterset_class = RelatorioPresencaSessaoFilterSet
template_name = 'base/RelatorioPresencaSessao_filter.html' template_name = 'base/RelatorioPresencaSessao_filter.html'
def calcular_porcentagem_presenca(self,
parlamentares,
total_sessao,
total_ordemdia):
for p in parlamentares:
p.sessao_porc = round(p.sessao_count * 100 / total_sessao, 1)
p.ordemdia_porc = round(p.ordemdia_count * 100 / total_ordemdia, 1)
return parlamentares
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
context = super(RelatorioPresencaSessaoView, context = super(RelatorioPresencaSessaoView,
self).get_context_data(**kwargs) self).get_context_data(**kwargs)
@ -234,37 +226,68 @@ class RelatorioPresencaSessaoView(FilterView):
sufixo = 'sessao_plenaria__data_inicio__range' sufixo = 'sessao_plenaria__data_inicio__range'
param0 = {'%s' % sufixo: _range} param0 = {'%s' % sufixo: _range}
param1 = {'presencaordemdia__%s' % sufixo: _range}
param2 = {'sessaoplenariapresenca__%s' % sufixo: _range} # Parlamentares com Mandato no intervalo de tempo (Ativos)
parlamentares_qs = parlamentares_ativos(
pls = Parlamentar.objects.filter( _range[0], _range[1]).order_by('nome_parlamentar')
(Q(**param1) | Q(presencaordemdia__isnull=True)) & parlamentares_id = parlamentares_qs.values_list(
(Q(**param2) | Q(sessaoplenariapresenca__isnull=True)) 'id', flat=True)
).annotate(
sessao_count=Count( # Presenças de cada Parlamentar em Sessões
'sessaoplenariapresenca__sessao_plenaria', presenca_sessao = SessaoPlenariaPresenca.objects.filter(
distinct=True), parlamentar_id__in=parlamentares_id,
ordemdia_count=Count( sessao_plenaria__data_inicio__range=_range).values_list(
'presencaordemdia__sessao_plenaria', 'parlamentar_id').annotate(
distinct=True), sessao_count=Count('id'))
sessao_porc=Count(0),
ordemdia_porc=Count(0) # Presenças de cada Ordem do Dia
).exclude( presenca_ordem = PresencaOrdemDia.objects.filter(
sessao_count=0, parlamentar_id__in=parlamentares_id,
ordemdia_count=0) sessao_plenaria__data_inicio__range=_range).values_list(
'parlamentar_id').annotate(
sessao_count=Count('id'))
total_ordemdia = PresencaOrdemDia.objects.filter( total_ordemdia = PresencaOrdemDia.objects.filter(
**param0).distinct('sessao_plenaria__id').order_by( **param0).distinct('sessao_plenaria__id').order_by(
'sessao_plenaria__id').count() 'sessao_plenaria__id').count()
self.calcular_porcentagem_presenca( total_sessao = context['object_list'].count()
pls,
context['object_list'].count(),
total_ordemdia)
# Completa o dicionario as informacoes parlamentar/sessao/ordem
parlamentares_presencas = []
for i, p in enumerate(parlamentares_qs):
parlamentares_presencas.append({
'parlamentar': p,
'sessao_porc': 0,
'ordemdia_porc': 0
})
try:
sessao_count = presenca_sessao.get(parlamentar_id=p.id)[1]
except ObjectDoesNotExist:
sessao_count = 0
try:
ordemdia_count = presenca_ordem.get(parlamentar_id=p.id)[1]
except ObjectDoesNotExist:
ordemdia_count = 0
parlamentares_presencas[i].update({
'sessao_count': sessao_count,
'ordemdia_count': ordemdia_count
})
if total_sessao != 0:
parlamentares_presencas[i].update(
{'sessao_porc': round(
sessao_count * 100 / total_sessao, 2)})
if total_ordemdia != 0:
parlamentares_presencas[i].update(
{'ordemdia_porc': round(
ordemdia_count * 100 / total_ordemdia, 2)})
context['date_range'] = _range
context['total_ordemdia'] = total_ordemdia context['total_ordemdia'] = total_ordemdia
context['total_sessao'] = context['object_list'].count() context['total_sessao'] = context['object_list'].count()
context['parlamentares'] = pls context['parlamentares'] = parlamentares_presencas
context['periodo'] = ( context['periodo'] = (
self.request.GET['data_inicio_0'] + self.request.GET['data_inicio_0'] +
' - ' + self.request.GET['data_inicio_1']) ' - ' + self.request.GET['data_inicio_1'])
@ -322,6 +345,46 @@ class RelatorioMateriasPorAnoAutorTipoView(FilterView):
filterset_class = RelatorioMateriasPorAnoAutorTipoFilterSet filterset_class = RelatorioMateriasPorAnoAutorTipoFilterSet
template_name = 'base/RelatorioMateriasPorAnoAutorTipo_filter.html' template_name = 'base/RelatorioMateriasPorAnoAutorTipo_filter.html'
def get_materias_autor_ano(self, ano):
autorias = Autoria.objects.filter(materia__ano=ano).values(
'autor',
'materia__tipo__sigla',
'materia__tipo__descricao').annotate(
total=Count('materia__tipo')).order_by(
'autor',
'materia__tipo')
autores_ids = set([i['autor'] for i in autorias])
autores = dict((a.id, a) for a in Autor.objects.filter(
id__in=autores_ids))
relatorio = []
visitados = set()
curr = None
for a in autorias:
# se mudou autor, salva atual, caso existente, e reinicia `curr`
if a['autor'] not in visitados:
if curr:
relatorio.append(curr)
curr = {}
curr['autor'] = autores[a['autor']]
curr['materia'] = []
curr['total'] = 0
visitados.add(a['autor'])
# atualiza valores
curr['materia'].append((a['materia__tipo__descricao'], a['total']))
curr['total'] += a['total']
# adiciona o ultimo
relatorio.append(curr)
return relatorio
def get_filterset_kwargs(self, filterset_class): def get_filterset_kwargs(self, filterset_class):
super(RelatorioMateriasPorAnoAutorTipoView, super(RelatorioMateriasPorAnoAutorTipoView,
self).get_filterset_kwargs(filterset_class) self).get_filterset_kwargs(filterset_class)
@ -346,6 +409,12 @@ class RelatorioMateriasPorAnoAutorTipoView(FilterView):
qr = self.request.GET.copy() qr = self.request.GET.copy()
context['filter_url'] = ('&' + qr.urlencode()) if len(qr) > 0 else '' context['filter_url'] = ('&' + qr.urlencode()) if len(qr) > 0 else ''
if 'ano' in self.request.GET and self.request.GET['ano']:
ano = int(self.request.GET['ano'])
context['relatorio'] = self.get_materias_autor_ano(ano)
else:
context['relatorio'] = []
return context return context

20
sapl/comissoes/migrations/0002_auto_20170809_1236.py

@ -0,0 +1,20 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.12 on 2017-08-09 12:36
from __future__ import unicode_literals
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('comissoes', '0001_initial'),
]
operations = [
migrations.AlterField(
model_name='comissao',
name='nome',
field=models.CharField(max_length=100, verbose_name='Nome'),
),
]

2
sapl/comissoes/models.py

@ -35,7 +35,7 @@ class Comissao(models.Model):
tipo = models.ForeignKey(TipoComissao, tipo = models.ForeignKey(TipoComissao,
on_delete=models.PROTECT, on_delete=models.PROTECT,
verbose_name=_('Tipo')) verbose_name=_('Tipo'))
nome = models.CharField(max_length=60, verbose_name=_('Nome')) nome = models.CharField(max_length=100, verbose_name=_('Nome'))
sigla = models.CharField(max_length=10, verbose_name=_('Sigla')) sigla = models.CharField(max_length=10, verbose_name=_('Sigla'))
data_criacao = models.DateField(verbose_name=_('Data de Criação')) data_criacao = models.DateField(verbose_name=_('Data de Criação'))
data_extincao = models.DateField( data_extincao = models.DateField(

6
sapl/comissoes/views.py

@ -82,21 +82,23 @@ class ComissaoCrud(Crud):
public = [RP_LIST, RP_DETAIL, ] public = [RP_LIST, RP_DETAIL, ]
class BaseMixin(Crud.BaseMixin): class BaseMixin(Crud.BaseMixin):
list_field_names = ['nome', 'sigla', 'tipo', 'data_criacao', 'data_extincao', 'ativa'] list_field_names = ['nome', 'sigla', 'tipo',
'data_criacao', 'data_extincao', 'ativa']
ordering = '-ativa', 'sigla' ordering = '-ativa', 'sigla'
class ListView(Crud.ListView): class ListView(Crud.ListView):
@xframe_options_exempt @xframe_options_exempt
def get(self, request, *args, **kwargs): def get(self, request, *args, **kwargs):
return super().get(request, *args, **kwargs) return super().get(request, *args, **kwargs)
class DetailView(Crud.DetailView): class DetailView(Crud.DetailView):
@xframe_options_exempt @xframe_options_exempt
def get(self, request, *args, **kwargs): def get(self, request, *args, **kwargs):
return super().get(request, *args, **kwargs) return super().get(request, *args, **kwargs)
class MateriasTramitacaoListView(ListView): class MateriasTramitacaoListView(ListView):
template_name = "comissoes/materias_em_tramitacao.html" template_name = "comissoes/materias_em_tramitacao.html"
paginate_by = 10 paginate_by = 10

2
sapl/compilacao/compilacao_data_tables.sql

@ -33,7 +33,7 @@ INSERT INTO compilacao_tipodispositivo (id, nome, class_css, rotulo_prefixo_html
SELECT pg_catalog.setval('compilacao_tipodispositivo_id_seq', 126, true); SELECT pg_catalog.setval('compilacao_tipodispositivo_id_seq', 126, true);
delete from compilacao_tipodispositivorelationship; #delete from compilacao_tipodispositivorelationship;
SELECT pg_catalog.setval('compilacao_tipodispositivorelationship_id_seq', 1, false); SELECT pg_catalog.setval('compilacao_tipodispositivorelationship_id_seq', 1, false);
INSERT INTO compilacao_tipodispositivorelationship (filho_permitido_id, pai_id, filho_de_insercao_automatica, perfil_id, quantidade_permitida, permitir_variacao) VALUES (112, 1, false, 2, -1, true); INSERT INTO compilacao_tipodispositivorelationship (filho_permitido_id, pai_id, filho_de_insercao_automatica, perfil_id, quantidade_permitida, permitir_variacao) VALUES (112, 1, false, 2, -1, true);
INSERT INTO compilacao_tipodispositivorelationship (filho_permitido_id, pai_id, filho_de_insercao_automatica, perfil_id, quantidade_permitida, permitir_variacao) VALUES (114, 1, false, 2, -1, true); INSERT INTO compilacao_tipodispositivorelationship (filho_permitido_id, pai_id, filho_de_insercao_automatica, perfil_id, quantidade_permitida, permitir_variacao) VALUES (114, 1, false, 2, -1, true);

65
sapl/compilacao/forms.py

@ -1,7 +1,8 @@
from datetime import timedelta from datetime import timedelta
from crispy_forms.bootstrap import (Alert, FieldWithButtons, FormActions, from crispy_forms.bootstrap import (Alert, FieldWithButtons, FormActions,
InlineRadios, StrictButton) InlineCheckboxes, InlineRadios,
StrictButton)
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import (HTML, Button, Column, Div, Field, Fieldset, from crispy_forms.layout import (HTML, Button, Column, Div, Field, Fieldset,
Layout, Row, Submit) Layout, Row, Submit)
@ -859,6 +860,21 @@ class DispositivoEdicaoVigenciaForm(ModelForm):
('fim_vigencia', 3), ('fim_vigencia', 3),
('inicio_eficacia', 3), ('inicio_eficacia', 3),
('fim_eficacia', 3), ]) ('fim_eficacia', 3), ])
inst = kwargs['instance']
while inst.auto_inserido and inst.dispositivo_pai:
inst = inst.dispositivo_pai
if (inst.dispositivos_vigencias_set.exists()):
row_datas.fields.append(
Alert(
css_class='alert-info col-md-12',
content='<strong>%s</strong> %s' % (
_('Atenção!'),
_('O Dispositivo em edição define vigência de outros '
'dispositivos. Alterar as datas de vigência '
'alterará as datas de vigência dos dispositivos '
'vigêntes por este em edição.'))))
layout.append( layout.append(
Fieldset(_('Datas de Controle de Vigência'), Fieldset(_('Datas de Controle de Vigência'),
row_datas, row_datas,
@ -909,17 +925,26 @@ class DispositivoEdicaoVigenciaForm(ModelForm):
super(DispositivoEdicaoVigenciaForm, self).save() super(DispositivoEdicaoVigenciaForm, self).save()
data = self.cleaned_data data = self.cleaned_data
instance = self.instance
inst = instance
extensao = 'extensao' in data and data['extensao'] == 'True' while instance.auto_inserido and instance.dispositivo_pai:
dp = instance.dispositivo_pai
dp.inicio_vigencia = instance.inicio_vigencia
dp.inicio_eficacia = instance.inicio_eficacia
dp.fim_vigencia = instance.fim_vigencia
dp.fim_eficacia = instance.fim_vigencia
dp.save()
if extensao: instance = dp
dv = data['dispositivo_vigencia']
dv = data['dispositivo_vigencia']
if dv and dv.auto_inserido: if dv and dv.auto_inserido:
dv = dv.dispositivo_pai dv = dv.dispositivo_pai
extensao = 'extensao' in data and data['extensao'] == 'True'
if extensao:
dv_pk = dv.pk if dv else None dv_pk = dv.pk if dv else None
instance = self.instance
def extenderPara(dpt_pk): def extenderPara(dpt_pk):
@ -936,6 +961,26 @@ class DispositivoEdicaoVigenciaForm(ModelForm):
extenderPara(instance.pk) extenderPara(instance.pk)
inst = instance
while instance.auto_inserido and instance.dispositivo_pai:
instance = instance.dispositivo_pai
inst.dispositivos_vigencias_set.filter(
ta_publicado__isnull=True).update(
inicio_vigencia=inst.inicio_vigencia,
inicio_eficacia=inst.inicio_eficacia,
fim_vigencia=inst.fim_vigencia,
fim_eficacia=inst.fim_eficacia)
inst.dispositivos_vigencias_set.filter(
ta_publicado__isnull=False).update(
inicio_vigencia=inst.inicio_eficacia,
inicio_eficacia=inst.inicio_eficacia,
fim_vigencia=inst.fim_eficacia,
fim_eficacia=inst.fim_eficacia)
return inst
class MultipleChoiceWithoutValidationField(forms.MultipleChoiceField): class MultipleChoiceWithoutValidationField(forms.MultipleChoiceField):
@ -1249,20 +1294,20 @@ class DispositivoEdicaoAlteracaoForm(ModelForm):
class TextNotificacoesForm(Form): class TextNotificacoesForm(Form):
type_notificacoes = forms.ChoiceField( type_notificacoes = forms.MultipleChoiceField(
label=_('Níveis de Notificações'), label=_('Níveis de Notificações'),
choices=[('default', _('Dispositivos sem Notificações!')), choices=[('default', _('Dispositivos sem Notificações!')),
('success', _('Informações!')), ('success', _('Informações!')),
('info', _('Boas Práticas!')), ('info', _('Boas Práticas!')),
('warning', _('Alertas!')), ('warning', _('Alertas!')),
('danger', _('Erros!'))], ('danger', _('Erros!'))],
required=False) required=False,
widget=widgets.CheckboxSelectMultiple())
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
field_type_notificacoes = to_row([(Field( field_type_notificacoes = to_row([(InlineCheckboxes(
'type_notificacoes', 'type_notificacoes'), 10),
template="compilacao/layout/bootstrap_btn_checkbox.html"), 10),
(Submit('submit-form', _('Filtrar'), (Submit('submit-form', _('Filtrar'),
css_class='btn btn-primary pull-right'), 2)]) css_class='btn btn-primary pull-right'), 2)])

21
sapl/compilacao/migrations/0002_auto_20170825_1108.py

@ -0,0 +1,21 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.13 on 2017-08-25 11:08
from __future__ import unicode_literals
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
('compilacao', '0001_initial'),
]
operations = [
migrations.AlterField(
model_name='tipotextoarticulado',
name='content_type',
field=models.OneToOneField(null=True, on_delete=django.db.models.deletion.SET_NULL, to='contenttypes.ContentType', verbose_name='Modelo Integrado'),
),
]

33
sapl/compilacao/migrations/0003_auto_20170825_1136.py

@ -0,0 +1,33 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.13 on 2017-08-25 11:36
from __future__ import unicode_literals
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
('compilacao', '0002_auto_20170825_1108'),
]
operations = [
migrations.AlterField(
model_name='tipotextoarticulado',
name='content_type',
field=models.OneToOneField(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to='contenttypes.ContentType', verbose_name='Modelo Integrado'),
),
migrations.AlterField(
model_name='tipotextoarticulado',
name='participacao_social',
field=models.BooleanField(choices=[(True, 'Sim'), (False, 'Não')], default=False, verbose_name='Participação Social'),
preserve_default=False,
),
migrations.AlterField(
model_name='tipotextoarticulado',
name='publicacao_func',
field=models.BooleanField(choices=[(True, 'Sim'), (False, 'Não')], default=False, verbose_name='Histórico de Publicação'),
preserve_default=False,
),
]

18
sapl/compilacao/models.py

@ -1,6 +1,5 @@
from datetime import datetime from datetime import datetime
import reversion
from django.contrib import messages from django.contrib import messages
from django.contrib.contenttypes.fields import GenericForeignKey from django.contrib.contenttypes.fields import GenericForeignKey
from django.contrib.contenttypes.models import ContentType from django.contrib.contenttypes.models import ContentType
@ -11,7 +10,9 @@ from django.db.models.deletion import PROTECT
from django.http.response import Http404 from django.http.response import Http404
from django.template import defaultfilters from django.template import defaultfilters
from django.utils.decorators import classonlymethod from django.utils.decorators import classonlymethod
from django.utils.encoding import force_text
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
import reversion
from sapl.compilacao.utils import (get_integrations_view_names, int_to_letter, from sapl.compilacao.utils import (get_integrations_view_names, int_to_letter,
int_to_roman) int_to_roman)
@ -110,18 +111,17 @@ class TipoTextoArticulado(models.Model):
descricao = models.CharField(max_length=50, verbose_name=_('Descrição')) descricao = models.CharField(max_length=50, verbose_name=_('Descrição'))
content_type = models.OneToOneField( content_type = models.OneToOneField(
ContentType, ContentType,
null=True, default=None,
verbose_name=_('Modelo Integrado'))
participacao_social = models.NullBooleanField(
default=False,
blank=True, null=True, blank=True, null=True,
on_delete=models.SET_NULL,
verbose_name=_('Modelo Integrado'))
participacao_social = models.BooleanField(
blank=False,
choices=YES_NO_CHOICES, choices=YES_NO_CHOICES,
verbose_name=_('Participação Social')) verbose_name=_('Participação Social'))
publicacao_func = models.NullBooleanField( publicacao_func = models.BooleanField(
default=False,
blank=True, null=True,
choices=YES_NO_CHOICES, choices=YES_NO_CHOICES,
blank=False,
verbose_name=_('Histórico de Publicação')) verbose_name=_('Histórico de Publicação'))
perfis = models.ManyToManyField( perfis = models.ManyToManyField(
@ -843,7 +843,7 @@ class Publicacao(TimestampedMixin):
@reversion.register() @reversion.register()
class Dispositivo(BaseModel, TimestampedMixin): class Dispositivo(BaseModel, TimestampedMixin):
TEXTO_PADRAO_DISPOSITIVO_REVOGADO = _('(Revogado)') TEXTO_PADRAO_DISPOSITIVO_REVOGADO = force_text(_('(Revogado)'))
INTERVALO_ORDEM = 1000 INTERVALO_ORDEM = 1000
ordem = models.PositiveIntegerField( ordem = models.PositiveIntegerField(
default=0, default=0,

5
sapl/compilacao/templatetags/compilacao_filters.py

@ -92,6 +92,10 @@ def nota_automatica(dispositivo, ta_pub_list):
elif not dispositivo.dispositivo_substituido_id: elif not dispositivo.dispositivo_substituido_id:
return _('Inclusão feita pelo %s - %s.') % ( return _('Inclusão feita pelo %s - %s.') % (
d, ta_publicado) d, ta_publicado)
else:
if dispositivo.tipo_dispositivo.dispositivo_de_articulacao:
return _('Alteração de rótulo feita pelo %s - %s.') % (
d, ta_publicado)
else: else:
return _('Alteração feita pelo %s - %s.') % ( return _('Alteração feita pelo %s - %s.') % (
d, ta_publicado) d, ta_publicado)
@ -286,7 +290,6 @@ def nomenclatura_heranca(d, ignore_ultimo=0, ignore_primeiro=0):
return result return result
@register.filter @register.filter
def list(obj): def list(obj):
return [obj, ] return [obj, ]

41
sapl/compilacao/tests/test_tipo_texto_articulado_form.py

@ -0,0 +1,41 @@
from django.utils.translation import ugettext_lazy as _
from model_mommy import mommy
import pytest
from sapl.compilacao import forms
from sapl.compilacao.models import PerfilEstruturalTextoArticulado
from sapl.compilacao.views import choice_models_in_extenal_views
def test_valida_campos_obrigatorios_tipo_texto_articulado_form():
form = forms.TipoTaForm(data={})
assert not form.is_valid()
errors = form.errors
assert errors['sigla'] == [_('Este campo é obrigatório.')]
assert errors['descricao'] == [_('Este campo é obrigatório.')]
assert errors['participacao_social'] == [_('Este campo é obrigatório.')]
assert errors['publicacao_func'] == [_('Este campo é obrigatório.')]
assert len(errors) == 4
_content_types = choice_models_in_extenal_views()
@pytest.mark.parametrize('content_type', _content_types)
@pytest.mark.django_db(transaction=False)
def test_tipo_texto_articulado_form_valid(content_type):
perfil = mommy.make(PerfilEstruturalTextoArticulado)
form = forms.TipoTaForm(data={'sigla': 'si',
'descricao': 'teste',
'content_type': content_type[0],
'participacao_social': True,
'publicacao_func': True,
'perfis': [perfil.pk, ]
})
assert form.is_valid(), form.errors

470
sapl/compilacao/views.py

@ -1,7 +1,7 @@
import logging
import sys
from collections import OrderedDict from collections import OrderedDict
from datetime import timedelta from datetime import timedelta
import logging
import sys
from braces.views import FormMessagesMixin from braces.views import FormMessagesMixin
from django import forms from django import forms
@ -19,8 +19,8 @@ from django.http.response import (HttpResponse, HttpResponseRedirect,
from django.shortcuts import get_object_or_404, redirect from django.shortcuts import get_object_or_404, redirect
from django.utils.dateparse import parse_date from django.utils.dateparse import parse_date
from django.utils.encoding import force_text from django.utils.encoding import force_text
from django.utils.translation import ugettext_lazy as _
from django.utils.translation import string_concat from django.utils.translation import string_concat
from django.utils.translation import ugettext_lazy as _
from django.views.generic.base import TemplateView from django.views.generic.base import TemplateView
from django.views.generic.detail import DetailView from django.views.generic.detail import DetailView
from django.views.generic.edit import (CreateView, DeleteView, FormView, from django.views.generic.edit import (CreateView, DeleteView, FormView,
@ -50,6 +50,7 @@ from sapl.compilacao.utils import (DISPOSITIVO_SELECT_RELATED,
from sapl.crud.base import Crud, CrudListView, make_pagination from sapl.crud.base import Crud, CrudListView, make_pagination
from sapl.settings import BASE_DIR from sapl.settings import BASE_DIR
TipoNotaCrud = Crud.build(TipoNota, 'tipo_nota') TipoNotaCrud = Crud.build(TipoNota, 'tipo_nota')
TipoVideCrud = Crud.build(TipoVide, 'tipo_vide') TipoVideCrud = Crud.build(TipoVide, 'tipo_vide')
TipoPublicacaoCrud = Crud.build(TipoPublicacao, 'tipo_publicacao') TipoPublicacaoCrud = Crud.build(TipoPublicacao, 'tipo_publicacao')
@ -142,7 +143,6 @@ class IntegracaoTaView(TemplateView):
implemente, ou passe `None` para as chaves que são fixas. implemente, ou passe `None` para as chaves que são fixas.
""") """)
map_fields = self.map_fields
ta_values = getattr(self, 'ta_values', {}) ta_values = getattr(self, 'ta_values', {})
item = get_object_or_404(self.model, pk=kwargs['pk']) item = get_object_or_404(self.model, pk=kwargs['pk'])
@ -287,6 +287,203 @@ class CompMixin(PermissionRequiredMixin):
context['NO_ENTRIES_MSG'] = CrudListView.no_entries_msg context['NO_ENTRIES_MSG'] = CrudListView.no_entries_msg
return context return context
def get_notificacoes(self, object_list=None, type_notificacoes=None):
p = []
def padd(r, type_notificacao, reverse_url=None, test=True, msg='',
kwargs=None, to_position=None):
if not test:
return
r.contextual_class = type_notificacao
if not kwargs:
kwargs = {'ta_id': r.ta_id, 'pk': r.pk}
if reverse_url:
p.append((type_notificacao, msg,
reverse_lazy(reverse_url, kwargs=kwargs),
to_position))
else:
p.append((type_notificacao, msg, None, to_position))
def success(r):
type_notificacao = 'success'
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.inconstitucionalidade,
_('Declarado Inconstitucional.'))
padd(r, type_notificacao, 'sapl.compilacao:ta_text_edit',
r.ta_publicado and r.dispositivo_atualizador,
_('Dispositivo alterado em %s' % r.ta_publicado),
{'ta_id': r.ta_publicado_id}, r.dispositivo_atualizador_id)
def info(r):
type_notificacao = 'info'
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.publicacao and
r.dispositivo_vigencia and
r.publicacao.data != r.dispositivo_vigencia.inicio_vigencia,
_('Data da publicação associada ao Dispositivo difere da data'
' de inicio de vigência do Dispositivo de vigência.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.publicacao and r.publicacao.data != r.inicio_vigencia,
_('Data da publicação associada ao Dispositivo difere '
'da data de inicio de vigência.'))
padd(r, type_notificacao, 'sapl.compilacao:dispositivo_edit',
r.rotulo != r.rotulo_padrao(local_insert=1),
_('Rótulo Diferente do Padrão'))
padd(r, type_notificacao, 'sapl.compilacao:dispositivo_edit',
r.texto_atualizador and r.texto_atualizador != r.texto,
_('Texto do Dispositivo para o Documento '
'está diferente do texto para o Documento Alterador.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.texto_atualizador and r.texto_atualizador == r.texto,
_('Texto do Dispositivo no Documento Alterador '
'está igual ao Texto no Documento Original. '
'Não é necessário manter armazenado o texto no Documento '
'Alterador.'))
def warning(r):
type_notificacao = 'warning'
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.dispositivo_vigencia and r.inicio_vigencia !=
r.dispositivo_vigencia.inicio_vigencia,
_('Data de início de Vigência difere da data início de '
'Vigência do Dispositivo de Vigência'))
padd(r, type_notificacao, 'sapl.compilacao:ta_text',
r.inconstitucionalidade and not r.notas.exists(),
_('Dispositivo está definido como inconstitucional. É '
'aconcelhavel inserir uma Nota informando esta condição.'),
kwargs={'ta_id': r.ta_id},
to_position=r.pk)
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.inconstitucionalidade and not (
r.inicio_vigencia == r.fim_vigencia and
r.fim_vigencia == r.inicio_eficacia and
r.inicio_eficacia == r.fim_eficacia),
_('Dispositivo está definido como inconstitucional porém '
'existe diferença entre as datas início e fim de '
'vigência e eficácia.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.publicacao and
r.ta_publicado and r.ta_publicado != r.publicacao.ta,
_('A Publicação associada a este Dispositivo não é '
'uma publicação do Texto Articulado Alterador.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
not r.publicacao,
_('Dispositivo sem registro de publicação.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.texto and r.tipo_dispositivo.dispositivo_de_articulacao,
_('Dispositivos de Articulação não '
'deveriam armazenar texto.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
not r.texto and
not r.tipo_dispositivo.dispositivo_de_articulacao,
_('Dispositivo está sem texto.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.texto_atualizador and not r.ta_publicado,
_('Existe Texto Atualizador, porém este Dispositivo não '
'está associado a nenhum Documento Atualizador.'))
def danger(r):
type_notificacao = 'danger'
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
not r.dispositivo_vigencia,
_('Dispositivo sem definição de Dispositivo de Vigência.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.inconstitucionalidade and
r.inicio_vigencia != r.fim_vigencia,
_('Dispositivo está definido como inconstitucional porém '
'existe período de vigência.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.ta_publicado and not r.dispositivo_atualizador,
_('Dispositivo está associado a um Texto Articulado '
'Atualizador mas, a nenhum Dispositivo Atualizador.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
not r.dispositivo_atualizador and
r.dispositivo_substituido,
_('Dispositivo está substituindo outro mas não foi informado '
'o Dispositivo Atualizador.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.dispositivo_substituido and
r.dispositivo_substituido.tipo_dispositivo !=
r.tipo_dispositivo,
_('Dispositivo está substituindo um Dispositivo '
'de outro tipo.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.dispositivo_substituido and
r.dispositivo_substituido.ta != r.ta,
_('Dispositivo está substituindo um Dispositivo de outro '
'Texto Articulado.'))
padd(r.dispositivo_substituido, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.dispositivo_substituido and
r.dispositivo_substituido.dispositivo_subsequente != r,
_('Dispositivo está substituindo um Dispositivo que não '
'possui este como seu Dispositivo Subsequente.'))
padd(r.dispositivo_subsequente, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.dispositivo_subsequente and
r.dispositivo_subsequente.dispositivo_substituido != r,
_('Dispositivo foi substituído por outro que não '
'possui este como seu Dispositivo Substituído.'))
rr = []
for r in object_list:
p = []
r.contextual_class = ""
for f in type_notificacoes:
if f != 'default':
locals()[f](r)
r.notificacoes = p
if p or 'default' in type_notificacoes:
rr.append(r)
if r.dispositivos_alterados_set.exists():
rr += self.get_notificacoes(
r.dispositivos_alterados_set.all(), type_notificacoes)
return rr
class TipoTaListView(CompMixin, ListView): class TipoTaListView(CompMixin, ListView):
model = TipoTextoArticulado model = TipoTextoArticulado
@ -964,6 +1161,23 @@ class TextEditView(CompMixin, TemplateView):
else: else:
if 'lock' in request.GET: if 'lock' in request.GET:
# TODO - implementar logging de ação de usuário # TODO - implementar logging de ação de usuário
notificacoes = self.get_notificacoes(
object_list=self.object.dispositivos_set.all(),
type_notificacoes=['danger', ])
if notificacoes:
messages.error(
request,
_('Existem erros a serem corrigidos que impedem a '
'publicação deste Texto Articulado. '
'Corrija os erros apontados nas notificações.'))
self.request.session[
'type_notificacoes'] = ['danger', ]
return redirect(to=reverse_lazy(
'sapl.compilacao:ta_text_notificacoes', kwargs={
'ta_id': self.object.id}))
self.object.editing_locked = True self.object.editing_locked = True
self.object.privacidade = STATUS_TA_PUBLIC self.object.privacidade = STATUS_TA_PUBLIC
self.object.save() self.object.save()
@ -1564,9 +1778,7 @@ class ActionDeleteDispositivoMixin(ActionsCommonsMixin):
ordem__gt=base_ordem, ordem__gt=base_ordem,
nivel__lte=base.nivel).first() nivel__lte=base.nivel).first()
if not pbi: if pbi:
base.delete()
else:
dcc_a_excluir = Dispositivo.objects.order_by( dcc_a_excluir = Dispositivo.objects.order_by(
'ordem').filter( 'ordem').filter(
ta_id=base.ta_id, ta_id=base.ta_id,
@ -1605,8 +1817,14 @@ class ActionDeleteDispositivoMixin(ActionsCommonsMixin):
dr.dispositivo0 - dr.dispositivo0 -
primeiro_a_religar + d.dispositivo0) primeiro_a_religar + d.dispositivo0)
dr.rotulo = dr.rotulo_padrao() dr.rotulo = dr.rotulo_padrao()
dr.save(clean=base != dr) dr.save(clean=base != dr)
if base.tipo_dispositivo.dispositivo_de_alteracao:
dpts = base.dispositivos_alterados_set.all().order_by(
'-ordem_bloco_atualizador')
for dpt in dpts:
self.remover_dispositivo(dpt, False)
if base.pk: if base.pk:
base.delete() base.delete()
@ -1882,22 +2100,25 @@ class ActionDispositivoCreateMixin(ActionsCommonsMixin):
try: try:
Dispositivo.objects.filter( Dispositivo.objects.filter(
(Q(ta=dvt.ta) & Q(ta_publicado__isnull=True)) | ta=dvt.ta, ta_publicado__isnull=True
Q(ta_publicado=dvt.ta)
).update( ).update(
dispositivo_vigencia=dvt, dispositivo_vigencia=dvt,
inicio_vigencia=dvt.inicio_vigencia, inicio_vigencia=dvt.inicio_vigencia,
inicio_eficacia=dvt.inicio_eficacia) inicio_eficacia=dvt.inicio_eficacia)
dps = Dispositivo.objects.filter(dispositivo_vigencia_id=dvt.pk, Dispositivo.objects.filter(ta_publicado=dvt.ta
ta_publicado_id=dvt.ta_id) ).update(
with transaction.atomic(): dispositivo_vigencia=dvt,
inicio_vigencia=dvt.inicio_eficacia,
inicio_eficacia=dvt.inicio_eficacia)
dps = Dispositivo.objects.filter(dispositivo_vigencia=dvt)
for d in dps: for d in dps:
if d.dispositivo_substituido: if d.dispositivo_substituido:
ds = d.dispositivo_substituido ds = d.dispositivo_substituido
ds.fim_vigencia = d.inicio_vigencia - timedelta(days=1) ds.fim_vigencia = d.inicio_vigencia - timedelta(days=1)
ds.fim_eficacia = d.inicio_eficacia - timedelta(days=1) ds.fim_eficacia = d.inicio_eficacia - timedelta(days=1)
d.save() ds.save()
if d.dispositivo_subsequente: if d.dispositivo_subsequente:
ds = d.dispositivo_subsequente ds = d.dispositivo_subsequente
@ -2302,7 +2523,7 @@ class ActionsEditMixin(ActionDragAndMoveDispositivoAlteradoMixin,
dispositivos_do_bloco = \ dispositivos_do_bloco = \
bloco_alteracao.dispositivos_alterados_set.order_by( bloco_alteracao.dispositivos_alterados_set.order_by(
'ordem_bloco_atualizador') 'ordem_bloco_atualizador')
if dispositivos_do_bloco.exists: if dispositivos_do_bloco.exists():
ndp.ordem_bloco_atualizador = dispositivos_do_bloco.last( ndp.ordem_bloco_atualizador = dispositivos_do_bloco.last(
).ordem_bloco_atualizador + Dispositivo.INTERVALO_ORDEM ).ordem_bloco_atualizador + Dispositivo.INTERVALO_ORDEM
@ -2376,13 +2597,21 @@ class ActionsEditMixin(ActionDragAndMoveDispositivoAlteradoMixin,
'Alterador!'), time=10000) 'Alterador!'), time=10000)
return data return data
if dispositivo_a_alterar.tipo_dispositivo.dispositivo_de_articulacao\
and not revogacao:
self.set_message(
data, 'warning',
_('Registrar alteração de um dispositivo de articulação '
'só é relevante para o caso de alterações de rótulo. '
'Se não é este o caso, a alteração deve ser específica '
'para o dispositivo que se quer alterar.'), modal=True)
ndp = Dispositivo.new_instance_based_on( ndp = Dispositivo.new_instance_based_on(
dispositivo_a_alterar, dispositivo_a_alterar.tipo_dispositivo) dispositivo_a_alterar, dispositivo_a_alterar.tipo_dispositivo)
ndp.auto_inserido = dispositivo_a_alterar.auto_inserido ndp.auto_inserido = dispositivo_a_alterar.auto_inserido
ndp.rotulo = dispositivo_a_alterar.rotulo ndp.rotulo = dispositivo_a_alterar.rotulo
ndp.publicacao = bloco_alteracao.publicacao ndp.publicacao = bloco_alteracao.publicacao
if not revogacao: if not revogacao:
ndp.texto = dispositivo_a_alterar.texto ndp.texto = dispositivo_a_alterar.texto
else: else:
@ -2390,12 +2619,13 @@ class ActionsEditMixin(ActionDragAndMoveDispositivoAlteradoMixin,
ndp.dispositivo_de_revogacao = True ndp.dispositivo_de_revogacao = True
ndp.dispositivo_vigencia = bloco_alteracao.dispositivo_vigencia ndp.dispositivo_vigencia = bloco_alteracao.dispositivo_vigencia
if ndp.dispositivo_vigencia: if ndp.dispositivo_vigencia:
ndp.inicio_eficacia = ndp.dispositivo_vigencia.inicio_eficacia ndp.inicio_eficacia = ndp.dispositivo_vigencia.inicio_eficacia
ndp.inicio_vigencia = ndp.dispositivo_vigencia.inicio_vigencia ndp.inicio_vigencia = ndp.dispositivo_vigencia.inicio_eficacia
else: else:
ndp.inicio_eficacia = bloco_alteracao.inicio_eficacia ndp.inicio_eficacia = bloco_alteracao.inicio_eficacia
ndp.inicio_vigencia = bloco_alteracao.inicio_vigencia ndp.inicio_vigencia = bloco_alteracao.inicio_eficacia
try: try:
ordem = dispositivo_a_alterar.criar_espaco( ordem = dispositivo_a_alterar.criar_espaco(
@ -2421,6 +2651,7 @@ class ActionsEditMixin(ActionDragAndMoveDispositivoAlteradoMixin,
dispositivos_do_bloco = \ dispositivos_do_bloco = \
bloco_alteracao.dispositivos_alterados_set.order_by( bloco_alteracao.dispositivos_alterados_set.order_by(
'ordem_bloco_atualizador') 'ordem_bloco_atualizador')
if dispositivos_do_bloco.exists(): if dispositivos_do_bloco.exists():
ndp.ordem_bloco_atualizador = dispositivos_do_bloco.last( ndp.ordem_bloco_atualizador = dispositivos_do_bloco.last(
).ordem_bloco_atualizador + Dispositivo.INTERVALO_ORDEM ).ordem_bloco_atualizador + Dispositivo.INTERVALO_ORDEM
@ -2449,6 +2680,7 @@ class ActionsEditMixin(ActionDragAndMoveDispositivoAlteradoMixin,
bloco_alteracao.ordenar_bloco_alteracao() bloco_alteracao.ordenar_bloco_alteracao()
if not revogacao: if not revogacao:
if 'message' not in data:
self.set_message( self.set_message(
data, 'success', data, 'success',
_('Dispositivo de Alteração adicionado com sucesso.')) _('Dispositivo de Alteração adicionado com sucesso.'))
@ -3038,10 +3270,14 @@ class TextNotificacoesView(CompMixin, ListView, FormView):
self.object = TextoArticulado.objects.get(pk=self.kwargs['ta_id']) self.object = TextoArticulado.objects.get(pk=self.kwargs['ta_id'])
return super(TextNotificacoesView, self).get(request, *args, **kwargs) return super(TextNotificacoesView, self).get(request, *args, **kwargs)
def post(self, request, *args, **kwargs):
self.object = TextoArticulado.objects.get(pk=self.kwargs['ta_id'])
return FormView.post(self, request, *args, **kwargs)
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
if 'object' not in kwargs: if 'object' not in kwargs:
kwargs['object'] = self.object kwargs['object'] = self.object
return super(TextNotificacoesView, self).get_context_data(**kwargs) return ListView.get_context_data(self, **kwargs)
def get_success_url(self): def get_success_url(self):
return reverse_lazy('sapl.compilacao:ta_text_notificacoes', return reverse_lazy('sapl.compilacao:ta_text_notificacoes',
@ -3071,199 +3307,11 @@ class TextNotificacoesView(CompMixin, ListView, FormView):
ta_id=self.kwargs['ta_id'] ta_id=self.kwargs['ta_id']
).select_related(*DISPOSITIVO_SELECT_RELATED) ).select_related(*DISPOSITIVO_SELECT_RELATED)
p = []
def padd(r, type_notificacao, reverse_url=None, test=True, msg='',
kwargs=None, to_position=None):
if not test:
return
r.contextual_class = type_notificacao
if not kwargs:
kwargs = {'ta_id': r.ta_id, 'pk': r.pk}
if reverse_url:
p.append((type_notificacao, msg,
reverse_lazy(reverse_url, kwargs=kwargs),
to_position))
else:
p.append((type_notificacao, msg, None, to_position))
def success(r):
type_notificacao = 'success'
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.inconstitucionalidade,
_('Declarado Inconstitucional.'))
padd(r, type_notificacao, 'sapl.compilacao:ta_text_edit',
r.ta_publicado and r.dispositivo_atualizador,
_('Dispositivo alterado em %s' % r.ta_publicado),
{'ta_id': r.ta_publicado_id}, r.dispositivo_atualizador_id)
def info(r):
type_notificacao = 'info'
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.publicacao and
r.dispositivo_vigencia and
r.publicacao.data != r.dispositivo_vigencia.inicio_vigencia,
_('Data da publicação associada ao Dispositivo difere da data'
' de inicio de vigência do Dispositivo de vigência.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.publicacao and r.publicacao.data != r.inicio_vigencia,
_('Data da publicação associada ao Dispositivo difere '
'da data de inicio de vigência.'))
padd(r, type_notificacao, 'sapl.compilacao:dispositivo_edit',
r.rotulo != r.rotulo_padrao(local_insert=1),
_('Rótulo Diferente do Padrão'))
padd(r, type_notificacao, 'sapl.compilacao:dispositivo_edit',
r.texto_atualizador and r.texto_atualizador != r.texto,
_('Texto do Dispositivo para o Documento '
'está diferente do texto para o Documento Alterador.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.texto_atualizador and r.texto_atualizador == r.texto,
_('Texto do Dispositivo no Documento Alterador '
'está igual ao Texto no Documento Original. '
'Não é necessário manter armazenado o texto no Documento '
'Alterador.'))
def warning(r):
type_notificacao = 'warning'
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.dispositivo_vigencia and r.inicio_vigencia !=
r.dispositivo_vigencia.inicio_vigencia,
_('Data de início de Vigência difere da data início de '
'Vigência do Dispositivo de Vigência'))
padd(r, type_notificacao, 'sapl.compilacao:ta_text',
r.inconstitucionalidade and not r.notas.exists(),
_('Dispositivo está definido como inconstitucional. É '
'aconcelhavel inserir uma Nota informando esta condição.'),
kwargs={'ta_id': r.ta_id},
to_position=r.pk)
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.inconstitucionalidade and not (
r.inicio_vigencia == r.fim_vigencia and
r.fim_vigencia == r.inicio_eficacia and
r.inicio_eficacia == r.fim_eficacia),
_('Dispositivo está definido como inconstitucional porém '
'existe diferença entre as datas início e fim de '
'vigência e eficácia.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.publicacao and
r.ta_publicado and r.ta_publicado != r.publicacao.ta,
_('A Publicação associada a este Dispositivo não é '
'uma publicação do Texto Articulado Alterador.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
not r.publicacao,
_('Dispositivo sem registro de publicação.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.texto and r.tipo_dispositivo.dispositivo_de_articulacao,
_('Dispositivos de Articulação não '
'deveriam armazenar texto.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
not r.texto and
not r.tipo_dispositivo.dispositivo_de_articulacao,
_('Dispositivo está sem texto.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.texto_atualizador and not r.ta_publicado,
_('Existe Texto Atualizador, porém este Dispositivo não '
'está associado a nenhum Documento Atualizador.'))
def danger(r):
type_notificacao = 'danger'
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
not r.dispositivo_vigencia,
_('Dispositivo sem definição de Dispositivo de Vigência.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_vigencia',
r.inconstitucionalidade and
r.inicio_vigencia != r.fim_vigencia,
_('Dispositivo está definido como inconstitucional porém '
'existe período de vigência.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.ta_publicado and not r.dispositivo_atualizador,
_('Dispositivo está associado a um Texto Articulado '
'Atualizador mas, a nenhum Dispositivo Atualizador.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
not r.dispositivo_atualizador and
r.dispositivo_substituido,
_('Dispositivo está substituindo outro mas não foi informado '
'o Dispositivo Atualizador.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.dispositivo_substituido and
r.dispositivo_substituido.tipo_dispositivo !=
r.tipo_dispositivo,
_('Dispositivo está substituindo um Dispositivo '
'de outro tipo.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.dispositivo_substituido and
r.dispositivo_substituido.ta != r.ta,
_('Dispositivo está substituindo um Dispositivo de outro '
'Texto Articulado.'))
padd(r, type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.dispositivo_substituido and
r.dispositivo_substituido.dispositivo_subsequente != r,
_('Dispositivo está substituindo um Dispositivo que não '
'possui este como seu Dispositivo Subsequente.'))
padd(r,
type_notificacao,
'sapl.compilacao:dispositivo_edit_alteracao',
r.dispositivo_subsequente and
r.dispositivo_subsequente.dispositivo_substituido != r,
_('Dispositivo foi substituído por outro que não '
'possui este como seu Dispositivo Substituído.'))
rr = []
for r in result:
p = []
r.contextual_class = ""
type_notificacoes = [] type_notificacoes = []
if 'type_notificacoes' in self.request.session: if 'type_notificacoes' in self.request.session:
type_notificacoes = self.request.session['type_notificacoes'] type_notificacoes = self.request.session['type_notificacoes']
if isinstance(type_notificacoes, list): if type_notificacoes and not isinstance(type_notificacoes, list):
for f in type_notificacoes: type_notificacoes = [type_notificacoes, ]
if f != 'default':
locals()[f](r)
r.notificacoes = p
if p or 'default' in type_notificacoes:
rr.append(r)
return rr return self.get_notificacoes(result, type_notificacoes)

2
sapl/crispy_layout_mixin.py

@ -1,5 +1,6 @@
from math import ceil from math import ceil
import rtyaml
from crispy_forms.bootstrap import FormActions from crispy_forms.bootstrap import FormActions
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import HTML, Div, Fieldset, Layout, Submit from crispy_forms.layout import HTML, Div, Fieldset, Layout, Submit
@ -7,7 +8,6 @@ from django import template
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from django.utils import formats from django.utils import formats
from django.utils.translation import ugettext as _ from django.utils.translation import ugettext as _
import rtyaml
def heads_and_tails(list_of_lists): def heads_and_tails(list_of_lists):

1
sapl/crud/base.py

@ -955,6 +955,7 @@ class CrudAux(Crud):
Se o valor de subnav_template_name é nulo faz o filter subnav Se o valor de subnav_template_name é nulo faz o filter subnav
não abrir o padrão e nem um outro arquivo. não abrir o padrão e nem um outro arquivo.
""" """
if 'subnav_template_name' not in context:
context['subnav_template_name'] = self.subnav_template_name context['subnav_template_name'] = self.subnav_template_name
return context return context

40
sapl/decorators.py

@ -0,0 +1,40 @@
from datetime import date
from functools import wraps
def vigencia_atual(decorated_method):
"""
concatena a string ' (Atual)' caso a model instancia estiver
em vigência na data atual do servidor
Premissas:
* A classe precisa conter os atributos 'data_inicio' e 'data_fim'.
* 'data_inicio' e 'data_fim' precisam ser do tipo models.DateField
"""
@wraps(decorated_method)
def display_atual(self):
try:
string_displayed = decorated_method(self)
except TypeError:
string_displayed = ""
if hasattr(self, 'data_inicio') and hasattr(self, 'data_fim'):
today = date.today()
e_atual = self.data_inicio <= today <= self.data_fim
string_displayed = "{} {}".format(
string_displayed, "(Atual)" if e_atual else "")
else:
print('{} {}'.format(
"Instance does not have the attributes [{}, {}].".format(
'data_inicio',
'data_fim'
),
"Decorator @{} has been disabled.".format(
vigencia_atual.__name__()
)
)
)
return string_displayed
return display_atual

4
sapl/legacy/migracao_documentos.py

@ -3,7 +3,6 @@ import os
import re import re
import magic import magic
from django.db.models.signals import post_delete, post_save
from sapl.base.models import CasaLegislativa from sapl.base.models import CasaLegislativa
from sapl.materia.models import (DocumentoAcessorio, MateriaLegislativa, from sapl.materia.models import (DocumentoAcessorio, MateriaLegislativa,
@ -86,7 +85,8 @@ DOCS = {
DocumentoAcessorioAdministrativo: [( DocumentoAcessorioAdministrativo: [(
'arquivo', 'arquivo',
'administrativo/{}', 'administrativo/{}',
'private/documentoacessorioadministrativo/{0}/{0}_acessorio_administrativo{1}') 'private/documentoacessorioadministrativo/{0}/'
'{0}_acessorio_administrativo{1}')
], ],
} }

20
sapl/legacy/migration.py

@ -12,22 +12,19 @@ from django.contrib.auth.models import Group
from django.contrib.contenttypes.models import ContentType from django.contrib.contenttypes.models import ContentType
from django.core.exceptions import ObjectDoesNotExist from django.core.exceptions import ObjectDoesNotExist
from django.db import OperationalError, ProgrammingError, connections, models from django.db import OperationalError, ProgrammingError, connections, models
from django.db.models import CharField, Max, ProtectedError, TextField, Count from django.db.models import CharField, Count, Max, ProtectedError, TextField
from django.db.models.base import ModelBase from django.db.models.base import ModelBase
from django.db.models.signals import post_delete, post_save
from model_mommy import mommy from model_mommy import mommy
from model_mommy.mommy import foreign_key_required, make from model_mommy.mommy import foreign_key_required, make
from sapl.base.models import Argumento, Autor, Constraint, ProblemaMigracao from sapl.base.models import Argumento, Autor, Constraint, ProblemaMigracao
from sapl.comissoes.models import Comissao, Composicao, Participacao from sapl.comissoes.models import Comissao, Composicao, Participacao
from sapl.legacy.models import Protocolo as ProtocoloLegado from sapl.materia.models import (AcompanhamentoMateria, Proposicao,
from sapl.materia.models import (AcompanhamentoMateria, DocumentoAcessorio,
MateriaLegislativa, Proposicao,
StatusTramitacao, TipoDocumento, StatusTramitacao, TipoDocumento,
TipoMateriaLegislativa, TipoProposicao, TipoMateriaLegislativa, TipoProposicao,
Tramitacao) Tramitacao)
from sapl.norma.models import (AssuntoNorma, NormaJuridica, from sapl.norma.models import (AssuntoNorma, NormaJuridica, NormaRelacionada,
TipoVinculoNormaJuridica, NormaRelacionada) TipoVinculoNormaJuridica)
from sapl.parlamentares.models import (Legislatura, Mandato, Parlamentar, from sapl.parlamentares.models import (Legislatura, Mandato, Parlamentar,
TipoAfastamento) TipoAfastamento)
from sapl.protocoloadm.models import (DocumentoAdministrativo, Protocolo, from sapl.protocoloadm.models import (DocumentoAdministrativo, Protocolo,
@ -816,7 +813,6 @@ def adjust_tipoafastamento(new, old):
new.indicador = 'A' new.indicador = 'A'
def adjust_tipoproposicao(new, old): def adjust_tipoproposicao(new, old):
if old.ind_mat_ou_doc == 'M': if old.ind_mat_ou_doc == 'M':
new.tipo_conteudo_related = TipoMateriaLegislativa.objects.get( new.tipo_conteudo_related = TipoMateriaLegislativa.objects.get(
@ -898,13 +894,13 @@ def adjust_autor(new, old):
def adjust_comissao(new, old): def adjust_comissao(new, old):
if old.dat_extincao: if not old.dat_extincao and not old.dat_fim_comissao:
if date.today() < new.data_extincao: new.ativa = True
elif old.dat_extincao and date.today() < new.data_extincao or \
old.dat_fim_comissao and date.today() < new.data_fim_comissao:
new.ativa = True new.ativa = True
else: else:
new.ativa = False new.ativa = False
if not old.dat_extincao:
new.ativa = True
AJUSTE_ANTES_SALVAR = { AJUSTE_ANTES_SALVAR = {

10
sapl/legacy/scripts/street_sweeper.py

@ -12,7 +12,8 @@ DB = ''
SELECT_EXCLUIDOS = "SELECT %s FROM %s WHERE ind_excluido = 1 ORDER BY %s" SELECT_EXCLUIDOS = "SELECT %s FROM %s WHERE ind_excluido = 1 ORDER BY %s"
REGISTROS_INCONSISTENTES = "DELETE FROM %s WHERE %s in (%s) AND ind_excluido = 0 " REGISTROS_INCONSISTENTES = "DELETE FROM %s WHERE %s "
"in (%s) AND ind_excluido = 0 "
EXCLUI_REGISTRO = "DELETE FROM %s WHERE ind_excluido=1" EXCLUI_REGISTRO = "DELETE FROM %s WHERE ind_excluido=1"
@ -43,6 +44,7 @@ mapa['parlamentar'] = ['autor', 'autoria', 'composicao_comissao',
'sessao_plenaria_presenca', 'unidade_tramitacao'] 'sessao_plenaria_presenca', 'unidade_tramitacao']
""" """
def get_ids_excluidos(cursor, query): def get_ids_excluidos(cursor, query):
""" """
recupera as PKs de registros com ind_excluido = 1 da tabela principal recupera as PKs de registros com ind_excluido = 1 da tabela principal
@ -58,7 +60,8 @@ def remove_tabelas(cursor, tabela_principal, pk, query_dependentes=None):
QUERY = SELECT_EXCLUIDOS % (pk, tabela_principal, pk) QUERY = SELECT_EXCLUIDOS % (pk, tabela_principal, pk)
ids_excluidos = get_ids_excluidos(cursor, QUERY) ids_excluidos = get_ids_excluidos(cursor, QUERY)
print("\nRegistros da tabela '%s' com ind_excluido = 1: %s" % (tabela_principal.upper(), len(ids_excluidos))) print("\nRegistros da tabela '%s' com ind_excluido = 1: %s" %
(tabela_principal.upper(), len(ids_excluidos)))
""" """
Remove registros de tabelas que dependem da tabela principal, Remove registros de tabelas que dependem da tabela principal,
@ -69,7 +72,8 @@ def remove_tabelas(cursor, tabela_principal, pk, query_dependentes=None):
print("Dependencias inconsistentes") print("Dependencias inconsistentes")
for tabela in mapa[tabela_principal]: for tabela in mapa[tabela_principal]:
QUERY_DEP = REGISTROS_INCONSISTENTES % (tabela, pk, ','.join(ids_excluidos)) QUERY_DEP = REGISTROS_INCONSISTENTES % (
tabela, pk, ','.join(ids_excluidos))
# Trata caso especifico de norma_juridica # Trata caso especifico de norma_juridica
if query_dependentes: if query_dependentes:

1
sapl/materia/admin.py

@ -11,6 +11,7 @@ if not DEBUG:
admin.site.unregister(Proposicao) admin.site.unregister(Proposicao)
class ProposicaoAdmin(admin.ModelAdmin): class ProposicaoAdmin(admin.ModelAdmin):
def has_add_permission(self, request, obj=None): def has_add_permission(self, request, obj=None):
return False return False

3
sapl/materia/apps.py

@ -6,6 +6,3 @@ class AppConfig(apps.AppConfig):
name = 'sapl.materia' name = 'sapl.materia'
label = 'materia' label = 'materia'
verbose_name = _('Matéria') verbose_name = _('Matéria')
def ready(self):
from . import signals

3
sapl/materia/email_utils.py

@ -206,6 +206,7 @@ def do_envia_email_tramitacao(base_url, materia, status, unidade_destino):
# a conexão será fechada # a conexão será fechada
except Exception: except Exception:
connection.close() connection.close()
raise Exception('Erro ao enviar e-mail de acompanhamento de matéria.') raise Exception(
'Erro ao enviar e-mail de acompanhamento de matéria.')
connection.close() connection.close()

192
sapl/materia/forms.py

@ -1,7 +1,8 @@
from datetime import date, datetime
import os import os
from datetime import date, datetime
import django_filters
from crispy_forms.bootstrap import Alert, FormActions, InlineRadios from crispy_forms.bootstrap import Alert, FormActions, InlineRadios
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import (HTML, Button, Column, Div, Field, Fieldset, from crispy_forms.layout import (HTML, Button, Column, Div, Field, Fieldset,
@ -13,21 +14,18 @@ from django.core.files.base import File
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from django.db import models, transaction from django.db import models, transaction
from django.db.models import Max from django.db.models import Max
from django.forms import ModelForm, ModelChoiceField, widgets from django.forms import ModelChoiceField, ModelForm, widgets
from django.forms.fields import BooleanField
from django.forms.forms import Form from django.forms.forms import Form
from django.forms.widgets import Select, HiddenInput from django.forms.models import ModelMultipleChoiceField
from django.utils import six from django.forms.widgets import CheckboxSelectMultiple, HiddenInput, Select
from django.utils.encoding import force_text from django.utils.encoding import force_text
from django.utils.html import format_html from django.utils.html import format_html
from django.utils.safestring import mark_safe from django.utils.safestring import mark_safe
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from django_filters.filterset import STRICTNESS
import django_filters
import sapl
from sapl.base.models import Autor, TipoAutor from sapl.base.models import Autor, TipoAutor
from sapl.comissoes.models import Comissao from sapl.comissoes.models import Comissao
from sapl.compilacao.forms import error_messages
from sapl.compilacao.models import (STATUS_TA_IMMUTABLE_PUBLIC, from sapl.compilacao.models import (STATUS_TA_IMMUTABLE_PUBLIC,
STATUS_TA_PRIVATE) STATUS_TA_PRIVATE)
from sapl.crispy_layout_mixin import (SaplFormLayout, form_actions, to_column, from sapl.crispy_layout_mixin import (SaplFormLayout, form_actions, to_column,
@ -42,8 +40,8 @@ from sapl.settings import MAX_DOC_UPLOAD_SIZE
from sapl.utils import (RANGE_ANOS, YES_NO_CHOICES, from sapl.utils import (RANGE_ANOS, YES_NO_CHOICES,
ChoiceWithoutValidationField, ChoiceWithoutValidationField,
MateriaPesquisaOrderingFilter, RangeWidgetOverride, MateriaPesquisaOrderingFilter, RangeWidgetOverride,
autor_label, autor_modal, models_with_gr_for_model) autor_label, autor_modal, models_with_gr_for_model,
import sapl qs_override_django_filter)
from .models import (AcompanhamentoMateria, Anexada, Autoria, DespachoInicial, from .models import (AcompanhamentoMateria, Anexada, Autoria, DespachoInicial,
DocumentoAcessorio, Numeracao, Proposicao, Relatoria, DocumentoAcessorio, Numeracao, Proposicao, Relatoria,
@ -166,6 +164,7 @@ class AcompanhamentoMateriaForm(ModelForm):
class DocumentoAcessorioForm(ModelForm): class DocumentoAcessorioForm(ModelForm):
data = forms.DateField(required=True)
class Meta: class Meta:
model = DocumentoAcessorio model = DocumentoAcessorio
@ -494,7 +493,8 @@ class MateriaLegislativaFilterSet(django_filters.FilterSet):
autoria__autor = django_filters.CharFilter(widget=forms.HiddenInput()) autoria__autor = django_filters.CharFilter(widget=forms.HiddenInput())
autoria__primeiro_autor = django_filters.BooleanFilter(required=False, autoria__primeiro_autor = django_filters.BooleanFilter(
required=False,
label='Primeiro Autor', label='Primeiro Autor',
widget=forms.HiddenInput()) widget=forms.HiddenInput())
@ -592,42 +592,7 @@ class MateriaLegislativaFilterSet(django_filters.FilterSet):
@property @property
def qs(self): def qs(self):
if not hasattr(self, '_qs'): return qs_override_django_filter(self)
valid = self.is_bound and self.form.is_valid()
if self.is_bound and not valid:
if self.strict == STRICTNESS.RAISE_VALIDATION_ERROR:
raise forms.ValidationError(self.form.errors)
elif bool(self.strict) == STRICTNESS.RETURN_NO_RESULTS:
self._qs = self.queryset.none()
return self._qs
# else STRICTNESS.IGNORE... ignoring
# start with all the results and filter from there
qs = self.queryset.all()
for name, filter_ in six.iteritems(self.filters):
value = None
if valid:
value = self.form.cleaned_data[name]
else:
raw_value = self.form[name].value()
try:
value = self.form.fields[name].clean(raw_value)
except forms.ValidationError:
if self.strict == STRICTNESS.RAISE_VALIDATION_ERROR:
raise
elif bool(self.strict) == STRICTNESS.RETURN_NO_RESULTS:
self._qs = self.queryset.none()
return self._qs
# else STRICTNESS.IGNORE... ignoring
if value is not None: # valid & clean data
qs = qs._next_is_sticky()
qs = filter_.filter(qs, value)
self._qs = qs
return self._qs
def pega_ultima_tramitacao(): def pega_ultima_tramitacao():
@ -735,6 +700,55 @@ class AutoriaForm(ModelForm):
return cd return cd
class AutoriaMultiCreateForm(Form):
tipo_autor = ModelChoiceField(label=_('Tipo Autor'),
required=False,
queryset=TipoAutor.objects.all(),
empty_label=_('Selecione'),)
data_relativa = forms.DateField(
widget=forms.HiddenInput(), required=False)
autor = ModelMultipleChoiceField(
queryset=Autor.objects.all(),
label=_('Possiveis Autores'),
required=False,
widget=CheckboxSelectMultiple)
autores = ModelMultipleChoiceField(
queryset=Autor.objects.all(),
required=False,
widget=HiddenInput)
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
row1 = to_row([('tipo_autor', 12), ])
row2 = to_row([('autor', 12), ])
self.helper = FormHelper()
self.helper.layout = Layout(
Fieldset(
_('Autorias'), row1, row2, 'data_relativa', 'autores',
form_actions(save_label='Incluir Autores Selecionados')))
self.fields['autor'].choices = []
def clean(self):
cd = super().clean()
if 'autores' in self.errors:
del self.errors['autores']
if 'autor' not in cd or not cd['autor'].exists():
raise ValidationError(
_('Ao menos um autor deve ser selecionado para inclusão'))
return cd
class AcessorioEmLoteFilterSet(django_filters.FilterSet): class AcessorioEmLoteFilterSet(django_filters.FilterSet):
filter_overrides = {models.DateField: { filter_overrides = {models.DateField: {
@ -1581,3 +1595,87 @@ class MateriaAssuntoForm(ModelForm):
fields = ['materia', 'assunto'] fields = ['materia', 'assunto']
widgets = {'materia': forms.HiddenInput()} widgets = {'materia': forms.HiddenInput()}
class EtiquetaPesquisaForm(forms.Form):
tipo_materia = forms.ModelChoiceField(
label=TipoMateriaLegislativa._meta.verbose_name,
queryset=TipoMateriaLegislativa.objects.all(),
required=False,
empty_label='Selecione')
data_inicial = forms.DateField(
label='Data Inicial',
required=False,
widget=forms.DateInput(format='%d/%m/%Y')
)
data_final = forms.DateField(
label='Data Final',
required=False,
widget=forms.DateInput(format='%d/%m/%Y')
)
processo_inicial = forms.IntegerField(
label='Processo Inicial',
required=False)
processo_final = forms.IntegerField(
label='Processo Final',
required=False)
def __init__(self, *args, **kwargs):
super(EtiquetaPesquisaForm, self).__init__(*args, **kwargs)
row1 = to_row(
[('tipo_materia', 6),
('data_inicial', 3),
('data_final', 3)])
row2 = to_row(
[('processo_inicial', 6),
('processo_final', 6)])
self.helper = FormHelper()
self.helper.layout = Layout(
Fieldset(
('Formulário de Etiqueta'),
row1, row2,
form_actions(save_label='Pesquisar')
)
)
def clean(self):
cleaned_data = self.cleaned_data
# Verifica se algum campo de data foi preenchido
if cleaned_data['data_inicial'] or cleaned_data['data_final']:
# Então verifica se o usuário preencheu o Incial e mas não
# preencheu o Final, ou vice-versa
if (not cleaned_data['data_inicial'] or
not cleaned_data['data_final']):
raise ValidationError(_(
'Caso pesquise por data, os campos de Data Incial e ' +
'Data Final devem ser preenchidos obrigatoriamente'))
# Caso tenha preenchido, verifica se a data final é maior que
# a inicial
elif cleaned_data['data_final'] < cleaned_data['data_inicial']:
raise ValidationError(_(
'A Data Final não pode ser menor que a Data Inicial'))
# O mesmo processo anterior é feito com o processo
if (cleaned_data['processo_inicial'] or
cleaned_data['processo_final']):
if (not cleaned_data['processo_inicial'] or
not cleaned_data['processo_final']):
raise ValidationError(_(
'Caso pesquise por número de processo, os campos de ' +
'Processo Inicial e Processo Final ' +
'devem ser preenchidos obrigatoriamente'))
elif (cleaned_data['processo_final'] <
cleaned_data['processo_inicial']):
raise ValidationError(_(
'O processo final não pode ser menor que o inicial'))
return cleaned_data

19
sapl/materia/migrations/0012_auto_20170829_1321.py

@ -0,0 +1,19 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.3 on 2017-08-29 13:21
from __future__ import unicode_literals
from django.db import migrations
class Migration(migrations.Migration):
dependencies = [
('materia', '0011_auto_20170808_1034'),
]
operations = [
migrations.AlterModelOptions(
name='materialegislativa',
options={'permissions': (('can_access_impressos', 'Can access impressos'),), 'verbose_name': 'Matéria Legislativa', 'verbose_name_plural': 'Matérias Legislativas'},
),
]

5
sapl/materia/models.py

@ -1,5 +1,6 @@
from datetime import datetime from datetime import datetime
import reversion
from django.contrib.auth.models import Group from django.contrib.auth.models import Group
from django.contrib.contenttypes.fields import GenericRelation from django.contrib.contenttypes.fields import GenericRelation
from django.contrib.contenttypes.models import ContentType from django.contrib.contenttypes.models import ContentType
@ -8,7 +9,6 @@ from django.db import models
from django.utils import formats from django.utils import formats
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from model_utils import Choices from model_utils import Choices
import reversion
from sapl.base.models import Autor from sapl.base.models import Autor
from sapl.comissoes.models import Comissao from sapl.comissoes.models import Comissao
@ -19,7 +19,6 @@ from sapl.utils import (RANGE_ANOS, YES_NO_CHOICES, SaplGenericForeignKey,
SaplGenericRelation, restringe_tipos_de_arquivo_txt, SaplGenericRelation, restringe_tipos_de_arquivo_txt,
texto_upload_path) texto_upload_path)
EM_TRAMITACAO = [(1, 'Sim'), EM_TRAMITACAO = [(1, 'Sim'),
(0, 'Não')] (0, 'Não')]
@ -235,6 +234,8 @@ class MateriaLegislativa(models.Model):
verbose_name_plural = _('Matérias Legislativas') verbose_name_plural = _('Matérias Legislativas')
unique_together = (("tipo", "numero", "ano"),) unique_together = (("tipo", "numero", "ano"),)
permissions = (("can_access_impressos", "Can access impressos"),)
def __str__(self): def __str__(self):
return _('%(tipo)s%(numero)s de %(ano)s') % { return _('%(tipo)s%(numero)s de %(ano)s') % {
'tipo': self.tipo, 'numero': self.numero, 'ano': self.ano} 'tipo': self.tipo, 'numero': self.numero, 'ano': self.ano}

5
sapl/materia/signals.py

@ -1,8 +1,3 @@
from django.db.models.signals import post_delete, post_save
import django.dispatch import django.dispatch
from .models import DocumentoAcessorio, MateriaLegislativa
tramitacao_signal = django.dispatch.Signal(providing_args=['post', 'request']) tramitacao_signal = django.dispatch.Signal(providing_args=['post', 'request'])

3
sapl/materia/tests/test_materia.py

@ -1,9 +1,9 @@
import pytest
from django.contrib.auth import get_user_model from django.contrib.auth import get_user_model
from django.contrib.contenttypes.models import ContentType from django.contrib.contenttypes.models import ContentType
from django.core.files.uploadedfile import SimpleUploadedFile from django.core.files.uploadedfile import SimpleUploadedFile
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from model_mommy import mommy from model_mommy import mommy
import pytest
from sapl.base.models import Autor, TipoAutor from sapl.base.models import Autor, TipoAutor
from sapl.comissoes.models import Comissao, TipoComissao from sapl.comissoes.models import Comissao, TipoComissao
@ -226,6 +226,7 @@ def test_documento_acessorio_submit(admin_client):
'data_materia': '2016-03-21', 'data_materia': '2016-03-21',
'autor': autor, 'autor': autor,
'ementa': 'teste_ementa', 'ementa': 'teste_ementa',
'data': '2016-03-21',
'salvar': 'salvar'}, 'salvar': 'salvar'},
follow=True) follow=True)

28
sapl/materia/urls.py

@ -2,12 +2,13 @@ from django.conf.urls import include, url
from sapl.materia.views import (AcompanhamentoConfirmarView, from sapl.materia.views import (AcompanhamentoConfirmarView,
AcompanhamentoExcluirView, AcompanhamentoExcluirView,
AcompanhamentoMateriaView, AcompanhamentoMateriaView, AnexadaCrud,
AdicionarVariasAutorias, AnexadaCrud,
AssuntoMateriaCrud, AutoriaCrud, AssuntoMateriaCrud, AutoriaCrud,
ConfirmarProposicao, CriarProtocoloMateriaView, AutoriaMultiCreateView, ConfirmarProposicao,
DespachoInicialCrud, DocumentoAcessorioCrud, CriarProtocoloMateriaView, DespachoInicialCrud,
DocumentoAcessorioCrud,
DocumentoAcessorioEmLoteView, DocumentoAcessorioEmLoteView,
ImpressosView, EtiquetaPesquisaView,
LegislacaoCitadaCrud, MateriaAssuntoCrud, LegislacaoCitadaCrud, MateriaAssuntoCrud,
MateriaLegislativaCrud, MateriaLegislativaCrud,
MateriaLegislativaPesquisaView, MateriaTaView, MateriaLegislativaPesquisaView, MateriaTaView,
@ -25,10 +26,17 @@ from sapl.materia.views import (AcompanhamentoConfirmarView,
from .apps import AppConfig from .apps import AppConfig
from . import receivers
app_name = AppConfig.name app_name = AppConfig.name
urlpatterns_impressos = [
url(r'^materia/impressos/$',
ImpressosView.as_view(),
name='impressos'),
url(r'^materia/impressos/etiqueta-pesquisa/$',
EtiquetaPesquisaView.as_view(),
name='impressos_etiqueta'),
]
urlpatterns_materia = [ urlpatterns_materia = [
url(r'^materia/', include(MateriaLegislativaCrud.get_urls() + url(r'^materia/', include(MateriaLegislativaCrud.get_urls() +
AnexadaCrud.get_urls() + AnexadaCrud.get_urls() +
@ -60,9 +68,9 @@ urlpatterns_materia = [
AcompanhamentoExcluirView.as_view(), AcompanhamentoExcluirView.as_view(),
name='acompanhar_excluir'), name='acompanhar_excluir'),
url(r'^materia/(?P<pk>\d+)/adicionar-varias-autorias/', url(r'^materia/(?P<pk>\d+)/autoria/multicreate',
AdicionarVariasAutorias.as_view(), AutoriaMultiCreateView.as_view(),
name='adicionar_varias_autorias'), name='autoria_multicreate'),
url(r'^materia/acessorio-em-lote', DocumentoAcessorioEmLoteView.as_view(), url(r'^materia/acessorio-em-lote', DocumentoAcessorioEmLoteView.as_view(),
name='acessorio_em_lote'), name='acessorio_em_lote'),
@ -120,5 +128,5 @@ urlpatterns_sistema = [
url(r'^sistema/materia/orgao/', include(OrgaoCrud.get_urls())), url(r'^sistema/materia/orgao/', include(OrgaoCrud.get_urls())),
] ]
urlpatterns = urlpatterns_materia + \ urlpatterns = urlpatterns_impressos + urlpatterns_materia + \
urlpatterns_proposicao + urlpatterns_sistema urlpatterns_proposicao + urlpatterns_sistema

144
sapl/materia/views.py

@ -1,21 +1,18 @@
from datetime import datetime, date from datetime import datetime
from random import choice from random import choice
from string import ascii_letters, digits from string import ascii_letters, digits
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import HTML from crispy_forms.layout import HTML
from django import forms
from django.contrib import messages from django.contrib import messages
from django.contrib.auth.decorators import permission_required from django.contrib.auth.decorators import permission_required
from django.contrib.auth.mixins import PermissionRequiredMixin from django.contrib.auth.mixins import PermissionRequiredMixin
from django.contrib.contenttypes.models import ContentType from django.core.exceptions import ObjectDoesNotExist
from django.core.exceptions import (ObjectDoesNotExist,
MultipleObjectsReturned)
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from django.db.models import Q
from django.http import HttpResponse, JsonResponse from django.http import HttpResponse, JsonResponse
from django.http.response import Http404, HttpResponseRedirect from django.http.response import Http404, HttpResponseRedirect
from django.shortcuts import get_object_or_404, redirect from django.shortcuts import get_object_or_404, redirect
from django.template import Context, loader, RequestContext
from django.utils import formats from django.utils import formats
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from django.views.generic import CreateView, ListView, TemplateView, UpdateView from django.views.generic import CreateView, ListView, TemplateView, UpdateView
@ -23,8 +20,8 @@ from django.views.generic.base import RedirectView
from django.views.generic.edit import FormView from django.views.generic.edit import FormView
from django_filters.views import FilterView from django_filters.views import FilterView
import sapl
from sapl.base.models import Autor, CasaLegislativa from sapl.base.models import Autor, CasaLegislativa
from sapl.comissoes.models import Comissao
from sapl.comissoes.models import Comissao, Participacao from sapl.comissoes.models import Comissao, Participacao
from sapl.compilacao.models import (STATUS_TA_IMMUTABLE_RESTRICT, from sapl.compilacao.models import (STATUS_TA_IMMUTABLE_RESTRICT,
STATUS_TA_PRIVATE) STATUS_TA_PRIVATE)
@ -34,23 +31,22 @@ from sapl.crud.base import (ACTION_CREATE, ACTION_DELETE, ACTION_DETAIL,
ACTION_LIST, ACTION_UPDATE, RP_DETAIL, RP_LIST, ACTION_LIST, ACTION_UPDATE, RP_DETAIL, RP_LIST,
Crud, CrudAux, MasterDetailCrud, Crud, CrudAux, MasterDetailCrud,
PermissionRequiredForAppCrudMixin, make_pagination) PermissionRequiredForAppCrudMixin, make_pagination)
from sapl.materia.forms import (AnexadaForm, ConfirmarProposicaoForm, from sapl.materia.forms import (AnexadaForm, AutoriaForm,
LegislacaoCitadaForm, AutoriaForm, ProposicaoForm, AutoriaMultiCreateForm,
TipoProposicaoForm, TramitacaoForm, ConfirmarProposicaoForm, LegislacaoCitadaForm,
TramitacaoUpdateForm) ProposicaoForm, TipoProposicaoForm,
from sapl.materia.models import Autor TramitacaoForm, TramitacaoUpdateForm)
from sapl.norma.models import LegislacaoCitada from sapl.norma.models import LegislacaoCitada
from sapl.parlamentares.models import Parlamentar
from sapl.protocoloadm.models import Protocolo from sapl.protocoloadm.models import Protocolo
from sapl.utils import (TURNO_TRAMITACAO_CHOICES, YES_NO_CHOICES, autor_label, from sapl.utils import (TURNO_TRAMITACAO_CHOICES, YES_NO_CHOICES, autor_label,
autor_modal, gerar_hash_arquivo, get_base_url, autor_modal, gerar_hash_arquivo, get_base_url,
montar_row_autor) montar_row_autor)
import sapl
from .email_utils import do_envia_email_confirmacao from .email_utils import do_envia_email_confirmacao
from .forms import (AcessorioEmLoteFilterSet, AcompanhamentoMateriaForm, from .forms import (AcessorioEmLoteFilterSet, AcompanhamentoMateriaForm,
AdicionarVariasAutoriasFilterSet, DespachoInicialForm, AdicionarVariasAutoriasFilterSet, DespachoInicialForm,
DocumentoAcessorioForm, MateriaAssuntoForm, DocumentoAcessorioForm, EtiquetaPesquisaForm,
MateriaAssuntoForm,
MateriaLegislativaFilterSet, MateriaSimplificadaForm, MateriaLegislativaFilterSet, MateriaSimplificadaForm,
PrimeiraTramitacaoEmLoteFilterSet, ReceberProposicaoForm, PrimeiraTramitacaoEmLoteFilterSet, ReceberProposicaoForm,
RelatoriaForm, TramitacaoEmLoteFilterSet, RelatoriaForm, TramitacaoEmLoteFilterSet,
@ -65,6 +61,7 @@ from .models import (AcompanhamentoMateria, Anexada, AssuntoMateria, Autoria,
TipoProposicao, Tramitacao, UnidadeTramitacao) TipoProposicao, Tramitacao, UnidadeTramitacao)
from .signals import tramitacao_signal from .signals import tramitacao_signal
import weasyprint
AssuntoMateriaCrud = Crud.build(AssuntoMateria, 'assunto_materia') AssuntoMateriaCrud = Crud.build(AssuntoMateria, 'assunto_materia')
@ -1065,6 +1062,11 @@ class DocumentoAcessorioCrud(MasterDetailCrud):
def __init__(self, **kwargs): def __init__(self, **kwargs):
super(MasterDetailCrud.CreateView, self).__init__(**kwargs) super(MasterDetailCrud.CreateView, self).__init__(**kwargs)
def get_initial(self):
self.initial['data'] = datetime.now().date()
return self.initial
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
context = super( context = super(
MasterDetailCrud.CreateView, self).get_context_data(**kwargs) MasterDetailCrud.CreateView, self).get_context_data(**kwargs)
@ -1086,13 +1088,16 @@ class AutoriaCrud(MasterDetailCrud):
parent_field = 'materia' parent_field = 'materia'
help_path = '' help_path = ''
public = [RP_LIST, RP_DETAIL] public = [RP_LIST, RP_DETAIL]
list_field_names = ['autor', 'autor__tipo__descricao', 'primeiro_autor']
class CreateView(MasterDetailCrud.CreateView): class LocalBaseMixin():
form_class = AutoriaForm form_class = AutoriaForm
@property @property
def layout_key(self): def layout_key(self):
return 'AutoriaCreate' return None
class CreateView(LocalBaseMixin, MasterDetailCrud.CreateView):
def get_initial(self): def get_initial(self):
initial = super().get_initial() initial = super().get_initial()
@ -1101,12 +1106,7 @@ class AutoriaCrud(MasterDetailCrud):
initial['autor'] = [] initial['autor'] = []
return initial return initial
class UpdateView(MasterDetailCrud.UpdateView): class UpdateView(LocalBaseMixin, MasterDetailCrud.UpdateView):
form_class = AutoriaForm
@property
def layout_key(self):
return 'AutoriaUpdate'
def get_initial(self): def get_initial(self):
initial = super().get_initial() initial = super().get_initial()
@ -1117,6 +1117,47 @@ class AutoriaCrud(MasterDetailCrud):
return initial return initial
class AutoriaMultiCreateView(PermissionRequiredForAppCrudMixin, FormView):
app_label = sapl.materia.apps.AppConfig.label
form_class = AutoriaMultiCreateForm
template_name = 'materia/autoria_multicreate_form.html'
@classmethod
def get_url_regex(cls):
return r'^(?P<pk>\d+)/%s/multicreate' % cls.model._meta.model_name
@property
def layout_key(self):
return None
def get_initial(self):
initial = super().get_initial()
self.materia = MateriaLegislativa.objects.get(id=self.kwargs['pk'])
initial['data_relativa'] = self.materia.data_apresentacao
initial['autores'] = self.materia.autores.all()
return initial
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
context['title'] = '%s <small>(%s)</small>' % (
_('Adicionar Várias Autorias'), self.materia)
return context
def get_success_url(self):
messages.add_message(
self.request, messages.SUCCESS,
_('Autorias adicionadas com sucesso.'))
return reverse(
'sapl.materia:autoria_list', kwargs={'pk': self.materia.pk})
def form_valid(self, form):
autores_selecionados = form.cleaned_data['autor']
for autor in autores_selecionados:
Autoria.objects.create(materia=self.materia, autor=autor)
return FormView.form_valid(self, form)
class DespachoInicialCrud(MasterDetailCrud): class DespachoInicialCrud(MasterDetailCrud):
model = DespachoInicial model = DespachoInicial
parent_field = 'materia' parent_field = 'materia'
@ -1696,8 +1737,7 @@ class TramitacaoEmLoteView(PrimeiraTramitacaoEmLoteView):
if ('tramitacao__status' in qr and if ('tramitacao__status' in qr and
'tramitacao__unidade_tramitacao_destino' in qr and 'tramitacao__unidade_tramitacao_destino' in qr and
qr['tramitacao__status'] and qr['tramitacao__status'] and
qr['tramitacao__unidade_tramitacao_destino'] qr['tramitacao__unidade_tramitacao_destino']):
):
lista = filtra_tramitacao_destino_and_status( lista = filtra_tramitacao_destino_and_status(
qr['tramitacao__status'], qr['tramitacao__status'],
qr['tramitacao__unidade_tramitacao_destino']) qr['tramitacao__unidade_tramitacao_destino'])
@ -1705,3 +1745,57 @@ class TramitacaoEmLoteView(PrimeiraTramitacaoEmLoteView):
id__in=lista).distinct() id__in=lista).distinct()
return context return context
class ImpressosView(PermissionRequiredMixin, TemplateView):
template_name = 'materia/impressos/impressos.html'
permission_required = ('materia.can_access_impressos', )
def gerar_pdf_impressos(request, context):
template = loader.get_template('materia/impressos/pdf.html')
html = template.render(RequestContext(request, context))
response = HttpResponse(content_type="application/pdf")
weasyprint.HTML(
string=html,
base_url=request.build_absolute_uri()).write_pdf(
response)
return response
class EtiquetaPesquisaView(PermissionRequiredMixin, FormView):
form_class = EtiquetaPesquisaForm
template_name = 'materia/impressos/etiqueta.html'
permission_required = ('materia.can_access_impressos', )
def form_valid(self, form):
context = {}
materias = MateriaLegislativa.objects.all().order_by(
'-data_apresentacao')
if form.cleaned_data['tipo_materia']:
materias = materias.filter(tipo=form.cleaned_data['tipo_materia'])
if form.cleaned_data['data_inicial']:
materias = materias.filter(
data_apresentacao__gte=form.cleaned_data['data_inicial'],
data_apresentacao__lte=form.cleaned_data['data_final'])
if form.cleaned_data['processo_inicial']:
materias = materias.filter(
numeracao__numero_materia__gte=form.cleaned_data[
'processo_inicial'],
numeracao__numero_materia__lte=form.cleaned_data[
'processo_final'])
context['quantidade'] = len(materias)
if context['quantidade'] > 20:
materias = materias[:20]
context['materias'] = materias
return gerar_pdf_impressos(self.request, context)

3
sapl/norma/apps.py

@ -6,6 +6,3 @@ class AppConfig(apps.AppConfig):
name = 'sapl.norma' name = 'sapl.norma'
label = 'norma' label = 'norma'
verbose_name = _('Norma Jurídica') verbose_name = _('Norma Jurídica')
def ready(self):
from . import signals

11
sapl/norma/forms.py

@ -118,21 +118,24 @@ class NormaJuridicaForm(ModelForm):
widgets = {'assuntos': widgets.CheckboxSelectMultiple} widgets = {'assuntos': widgets.CheckboxSelectMultiple}
def clean(self): def clean(self):
super(NormaJuridicaForm, self).clean() cleaned_data = super(NormaJuridicaForm, self).clean()
cleaned_data = self.cleaned_data if not self.is_valid():
return cleaned_data
if (cleaned_data['tipo_materia'] and if (cleaned_data['tipo_materia'] and
cleaned_data['numero_materia'] and cleaned_data['numero_materia'] and
cleaned_data['ano_materia']): cleaned_data['ano_materia']):
try: try:
materia = MateriaLegislativa.objects.get( materia = MateriaLegislativa.objects.get(
tipo_id=cleaned_data['tipo_materia'], tipo_id=cleaned_data['tipo_materia'],
numero=cleaned_data['numero_materia'], numero=cleaned_data['numero_materia'],
ano=cleaned_data['ano_materia']) ano=cleaned_data['ano_materia'])
except ObjectDoesNotExist: except ObjectDoesNotExist:
raise forms.ValidationError("Matéria escolhida não existe!") raise forms.ValidationError(
_("Matéria %s/%s é inexistente." % (
self.cleaned_data['numero_materia'],
self.cleaned_data['ano_materia'])))
else: else:
cleaned_data['materia'] = materia cleaned_data['materia'] = materia

2
sapl/norma/models.py

@ -59,9 +59,11 @@ class TipoNormaJuridica(models.Model):
def __str__(self): def __str__(self):
return self.descricao return self.descricao
def norma_upload_path(instance, filename): def norma_upload_path(instance, filename):
return texto_upload_path(instance, filename, subpath=instance.ano) return texto_upload_path(instance, filename, subpath=instance.ano)
@reversion.register() @reversion.register()
class NormaJuridica(models.Model): class NormaJuridica(models.Model):
ESFERA_FEDERACAO_CHOICES = Choices( ESFERA_FEDERACAO_CHOICES = Choices(

3
sapl/norma/signals.py

@ -1,3 +0,0 @@
from django.db.models.signals import post_delete, post_save
from .models import NormaJuridica

78
sapl/norma/tests/test_norma.py

@ -1,7 +1,10 @@
import pytest import pytest
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from django.utils.translation import ugettext_lazy as _
from model_mommy import mommy from model_mommy import mommy
from sapl.materia.models import MateriaLegislativa, TipoMateriaLegislativa
from sapl.norma.forms import (NormaJuridicaForm,)
from sapl.norma.models import NormaJuridica, TipoNormaJuridica from sapl.norma.models import NormaJuridica, TipoNormaJuridica
@ -38,14 +41,77 @@ def test_incluir_norma_errors(admin_client):
follow=True) follow=True)
assert (response.context_data['form'].errors['tipo'] == assert (response.context_data['form'].errors['tipo'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
assert (response.context_data['form'].errors['numero'] == assert (response.context_data['form'].errors['numero'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
assert (response.context_data['form'].errors['ano'] == assert (response.context_data['form'].errors['ano'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
assert (response.context_data['form'].errors['data'] == assert (response.context_data['form'].errors['data'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
assert (response.context_data['form'].errors['esfera_federacao'] == assert (response.context_data['form'].errors['esfera_federacao'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
assert (response.context_data['form'].errors['ementa'] == assert (response.context_data['form'].errors['ementa'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
# TODO esse teste repete o teste acima (test_incluir_norma_errors)
# mas a granularidade certa para testar campos obrigatórios seria
# no nível de form ou então de model, não de client...
def test_norma_form_invalida():
form = NormaJuridicaForm(data={})
assert not form.is_valid()
errors = form.errors
assert errors['tipo'] == [_('Este campo é obrigatório.')]
assert errors['numero'] == [_('Este campo é obrigatório.')]
assert errors['ano'] == [_('Este campo é obrigatório.')]
assert errors['data'] == [_('Este campo é obrigatório.')]
assert errors['esfera_federacao'] == [_('Este campo é obrigatório.')]
assert errors['ementa'] == [_('Este campo é obrigatório.')]
@pytest.mark.django_db(transaction=False)
def test_norma_juridica_materia_inexistente():
tipo = mommy.make(TipoNormaJuridica)
tipo_materia = mommy.make(TipoMateriaLegislativa)
form = NormaJuridicaForm(data={'tipo': str(tipo.pk),
'numero': '1',
'ano': '2017',
'data': '2017-12-12',
'esfera_federacao': 'F',
'ementa': 'teste norma',
'tipo_materia': str(tipo_materia.pk),
'numero_materia': '2',
'ano_materia': '2017'
})
assert not form.is_valid()
assert form.errors['__all__'] == [_("Matéria 2/2017 é inexistente.")]
@pytest.mark.django_db(transaction=False)
def test_norma_juridica_materia_existente():
tipo = mommy.make(TipoNormaJuridica)
tipo_materia = mommy.make(TipoMateriaLegislativa)
materia = mommy.make(MateriaLegislativa,
numero=2,
ano=2017,
tipo=tipo_materia)
form = NormaJuridicaForm(data={'tipo': str(tipo.pk),
'numero': '1',
'ano': '2017',
'data': '2017-12-12',
'esfera_federacao': 'F',
'ementa': 'teste norma',
'tipo_materia': str(tipo_materia.pk),
'numero_materia': '2',
'ano_materia': '2017'
})
assert form.is_valid()

4
sapl/painel/urls.py

@ -2,8 +2,8 @@ from django.conf.urls import url
from .apps import AppConfig from .apps import AppConfig
from .views import (cronometro_painel, get_dados_painel, painel_mensagem_view, from .views import (cronometro_painel, get_dados_painel, painel_mensagem_view,
painel_parlamentar_view, painel_view, painel_parlamentar_view, painel_view, painel_votacao_view,
painel_votacao_view, votante_view) votante_view)
app_name = AppConfig.name app_name = AppConfig.name

11
sapl/painel/views.py

@ -1,10 +1,10 @@
from datetime import date
from django.contrib import messages from django.contrib import messages
from django.contrib.auth.decorators import user_passes_test from django.contrib.auth.decorators import user_passes_test
from django.core.exceptions import ObjectDoesNotExist, MultipleObjectsReturned from django.core.exceptions import ObjectDoesNotExist
from django.db.models import Q
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from django.db.models import Q
from django.http import HttpResponse, JsonResponse from django.http import HttpResponse, JsonResponse
from django.http.response import Http404, HttpResponseRedirect from django.http.response import Http404, HttpResponseRedirect
from django.shortcuts import render from django.shortcuts import render
@ -12,7 +12,6 @@ from django.utils.translation import ugettext_lazy as _
from sapl.crud.base import Crud from sapl.crud.base import Crud
from sapl.painel.apps import AppConfig from sapl.painel.apps import AppConfig
from sapl.painel.models import Painel
from sapl.parlamentares.models import Filiacao, Votante from sapl.parlamentares.models import Filiacao, Votante
from sapl.sessao.models import (ExpedienteMateria, OrdemDia, PresencaOrdemDia, from sapl.sessao.models import (ExpedienteMateria, OrdemDia, PresencaOrdemDia,
RegistroVotacao, SessaoPlenaria, RegistroVotacao, SessaoPlenaria,
@ -425,6 +424,7 @@ def get_votos(response, materia):
'total_votos': total, 'total_votos': total,
'tipo_votacao': tipo_votacao, 'tipo_votacao': tipo_votacao,
'tipo_resultado': registro.tipo_resultado_votacao.nome, 'tipo_resultado': registro.tipo_resultado_votacao.nome,
'natureza_resultado': registro.tipo_resultado_votacao.natureza,
}) })
else: else:
response.update({ response.update({
@ -434,6 +434,7 @@ def get_votos(response, materia):
'total_votos': 0, 'total_votos': 0,
'tipo_votacao': tipo_votacao, 'tipo_votacao': tipo_votacao,
'tipo_resultado': 'Ainda não foi votada.', 'tipo_resultado': 'Ainda não foi votada.',
'natureza_resultado': None,
}) })
return response return response
@ -463,6 +464,7 @@ def get_votos_nominal(response, materia):
'total_votos': 0, 'total_votos': 0,
'tipo_votacao': tipo_votacao, 'tipo_votacao': tipo_votacao,
'tipo_resultado': 'Não foi votado ainda', 'tipo_resultado': 'Não foi votado ainda',
'natureza_resultado': None,
'votos': None 'votos': None
}) })
@ -505,6 +507,7 @@ def get_votos_nominal(response, materia):
'total_votos': total, 'total_votos': total,
'tipo_votacao': tipo_votacao, 'tipo_votacao': tipo_votacao,
'tipo_resultado': registro.tipo_resultado_votacao.nome, 'tipo_resultado': registro.tipo_resultado_votacao.nome,
'natureza_resultado': registro.tipo_resultado_votacao.natureza,
'votos': votos 'votos': votos
}) })

123
sapl/parlamentares/forms.py

@ -5,7 +5,7 @@ from crispy_forms.layout import Fieldset, Layout
from django import forms from django import forms
from django.contrib.auth import get_user_model from django.contrib.auth import get_user_model
from django.contrib.auth.models import Group, User from django.contrib.auth.models import Group, User
from django.core.exceptions import ObjectDoesNotExist, ValidationError from django.core.exceptions import ValidationError
from django.db import transaction from django.db import transaction
from django.db.models import Q from django.db.models import Q
from django.forms import ModelForm from django.forms import ModelForm
@ -62,17 +62,34 @@ class MandatoForm(ModelForm):
def clean(self): def clean(self):
super(MandatoForm, self).clean() super(MandatoForm, self).clean()
if not self.is_valid():
return self.cleaned_data
data = self.cleaned_data data = self.cleaned_data
try:
if 'legislatura' in data and 'parlamentar' in data: legislatura = data['legislatura']
Mandato.objects.get(
parlamentar__pk=self.initial['parlamentar'].pk, data_inicio_mandato = data['data_inicio_mandato']
legislatura__pk=data['legislatura'].pk) if data_inicio_mandato:
except ObjectDoesNotExist: if (data_inicio_mandato < legislatura.data_inicio or
pass data_inicio_mandato > legislatura.data_fim):
else: raise ValidationError(_("Data início mandato fora do intervalo "
raise ValidationError('Mandato nesta legislatura já existe.') "de legislatura informada"))
return data
data_fim_mandato = data['data_fim_mandato']
if data_fim_mandato:
if (data_fim_mandato < legislatura.data_inicio or
data_fim_mandato > legislatura.data_fim):
raise ValidationError(_("Data fim mandato fora do intervalo de "
"legislatura informada"))
existe_mandato = Mandato.objects.filter(
parlamentar=data['parlamentar'],
legislatura=data['legislatura']).exists()
if existe_mandato:
raise ValidationError(_('Mandato nesta legislatura já existe.'))
return self.cleaned_data
class LegislaturaForm(ModelForm): class LegislaturaForm(ModelForm):
@ -81,6 +98,26 @@ class LegislaturaForm(ModelForm):
model = Legislatura model = Legislatura
exclude = [] exclude = []
def clean(self):
data = super(LegislaturaForm, self).clean()
if not self.is_valid():
return self.cleaned_data
data_inicio = data['data_inicio']
data_fim = data['data_fim']
data_eleicao = data['data_eleicao']
if data_eleicao.year >= data_inicio.year:
raise ValidationError(_("Data eleição não pode ser inferior a "
"data início da legislatura"))
if data_inicio > data_fim or (data_fim.year - data_inicio.year != 4):
raise ValidationError(_("Intervalo de início e fim inválido para "
"legislatura."))
return data
class LegislaturaCreateForm(LegislaturaForm): class LegislaturaCreateForm(LegislaturaForm):
@ -88,6 +125,10 @@ class LegislaturaCreateForm(LegislaturaForm):
super(LegislaturaCreateForm, self).clean() super(LegislaturaCreateForm, self).clean()
cleaned_data = self.cleaned_data cleaned_data = self.cleaned_data
if not self.is_valid():
return cleaned_data
eleicao = cleaned_data['data_eleicao'] eleicao = cleaned_data['data_eleicao']
inicio = cleaned_data['data_inicio'] inicio = cleaned_data['data_inicio']
fim = cleaned_data['data_fim'] fim = cleaned_data['data_fim']
@ -275,44 +316,23 @@ class FrenteForm(ModelForm):
class VotanteForm(ModelForm): class VotanteForm(ModelForm):
senha = forms.CharField(
max_length=20,
label=_('Senha'),
required=True,
widget=forms.PasswordInput())
senha_confirma = forms.CharField(
max_length=20,
label=_('Confirmar Senha'),
required=True,
widget=forms.PasswordInput())
username = forms.CharField( username = forms.CharField(
label=_('Usuário'), label=_('Usuário'),
required=True, required=True,
max_length=30) max_length=30)
email = forms.EmailField(
required=True,
label=_('Email'))
email_confirma = forms.EmailField(
required=True,
label=_('Confirmar Email'))
class Meta: class Meta:
model = Votante model = Votante
fields = ['username', 'senha', 'senha_confirma', 'email', fields = ['username']
'email_confirma']
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
row1 = to_row([('username', 4), ('senha', 4), ('senha_confirma', 4)]) row1 = to_row([('username', 4)])
row2 = to_row([('email', 6), ('email_confirma', 6)])
self.helper = FormHelper() self.helper = FormHelper()
self.helper.layout = Layout( self.helper.layout = Layout(
Fieldset(_('Votante'), Fieldset(_('Votante'),
row1, row2, form_actions(save_label='Salvar')) row1, form_actions(save_label='Salvar'))
) )
super(VotanteForm, self).__init__(*args, **kwargs) super(VotanteForm, self).__init__(*args, **kwargs)
@ -327,23 +347,19 @@ class VotanteForm(ModelForm):
cd = self.cleaned_data cd = self.cleaned_data
username = cd['username'] username = cd['username']
if get_user_model().objects.filter(username=username).exists(): user = get_user_model().objects.filter(username=username)
raise ValidationError(_('Não foi possível salvar registro,\ if not user.exists():
pois usuário existente'))
if ('senha' not in cd or 'senha_confirma' not in cd or
not cd['senha'] or not cd['senha_confirma']):
raise ValidationError(_( raise ValidationError(_(
'A senha e sua confirmação devem ser informadas.')) "{} [{}] {}".format(
msg = _('As senhas não conferem.') 'Não foi possível vincular usuário. Usuário',
self.valida_igualdade(cd['senha'], cd['senha_confirma'], msg) username,
'não existe')))
if ('email' not in cd or 'email_confirma' not in cd or if Votante.objects.filter(user=user[0].pk).exists():
not cd['email'] or not cd['email_confirma']):
raise ValidationError(_( raise ValidationError(_(
'O email e sua confirmação devem ser informados.')) "{} [{}] {}".format(
msg = _('Os emails não conferem.') 'Não foi possível vincular usuário. Usuário',
self.valida_igualdade(cd['email'], cd['email_confirma'], msg) username,
'já esta vinculado à outro parlamentar')))
return self.cleaned_data return self.cleaned_data
@ -352,12 +368,7 @@ class VotanteForm(ModelForm):
votante = super(VotanteForm, self).save(commit) votante = super(VotanteForm, self).save(commit)
# Cria user # Cria user
u = User.objects.create( u = User.objects.get(username=self.cleaned_data['username'])
username=self.cleaned_data['username'],
email=self.cleaned_data['email'])
u.set_password(self.cleaned_data['senha'])
u.save()
# Adiciona user ao grupo # Adiciona user ao grupo
g = Group.objects.filter(name=SAPL_GROUP_VOTANTE)[0] g = Group.objects.filter(name=SAPL_GROUP_VOTANTE)[0]
u.groups.add(g) u.groups.add(g)

19
sapl/parlamentares/migrations/0005_auto_20170814_1615.py

@ -0,0 +1,19 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.11 on 2017-08-14 16:15
from __future__ import unicode_literals
from django.db import migrations
class Migration(migrations.Migration):
dependencies = [
('parlamentares', '0004_auto_20170711_1305'),
]
operations = [
migrations.AlterModelOptions(
name='frente',
options={'verbose_name': 'Frente Parlamentar', 'verbose_name_plural': 'Frentes Parlamentares'},
),
]

16
sapl/parlamentares/models.py

@ -1,11 +1,12 @@
from datetime import datetime from datetime import datetime
import reversion
from django.db import models from django.db import models
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from model_utils import Choices from model_utils import Choices
import reversion
from sapl.base.models import Autor from sapl.base.models import Autor
from sapl.decorators import vigencia_atual
from sapl.utils import (INDICADOR_AFASTAMENTO, UF, YES_NO_CHOICES, from sapl.utils import (INDICADOR_AFASTAMENTO, UF, YES_NO_CHOICES,
SaplGenericRelation, get_settings_auth_user_model, SaplGenericRelation, get_settings_auth_user_model,
intervalos_tem_intersecao, intervalos_tem_intersecao,
@ -28,14 +29,12 @@ class Legislatura(models.Model):
current_year = datetime.now().year current_year = datetime.now().year
return self.data_inicio.year <= current_year <= self.data_fim.year return self.data_inicio.year <= current_year <= self.data_fim.year
@vigencia_atual
def __str__(self): def __str__(self):
current = ' (%s)' % _('Atual') if self.atual() else '' return _('%(numero)sª (%(start)s - %(end)s)') % {
return _('%(numero)sª (%(start)s - %(end)s)%(current)s') % {
'numero': self.numero, 'numero': self.numero,
'start': self.data_inicio.year, 'start': self.data_inicio.year,
'end': self.data_fim.year, 'end': self.data_fim.year}
'current': current}
@reversion.register() @reversion.register()
@ -64,6 +63,7 @@ class SessaoLegislativa(models.Model):
verbose_name_plural = _('Sessões Legislativas') verbose_name_plural = _('Sessões Legislativas')
ordering = ['-data_inicio', '-data_fim'] ordering = ['-data_inicio', '-data_fim']
@vigencia_atual
def __str__(self): def __str__(self):
return _('%(numero)sº (%(inicio)s - %(fim)s)') % { return _('%(numero)sº (%(inicio)s - %(fim)s)') % {
'numero': self.numero, 'numero': self.numero,
@ -554,8 +554,8 @@ class Frente(models.Model):
)) ))
class Meta: class Meta:
verbose_name = _('Frente') verbose_name = _('Frente Parlamentar')
verbose_name_plural = _('Frentes') verbose_name_plural = _('Frentes Parlamentares')
def get_parlamentares(self): def get_parlamentares(self):
return Parlamentar.objects.filter(ativo=True) return Parlamentar.objects.filter(ativo=True)

123
sapl/parlamentares/tests/test_parlamentares.py

@ -1,7 +1,9 @@
import pytest import pytest
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from django.utils.translation import ugettext_lazy as _
from model_mommy import mommy from model_mommy import mommy
from sapl.parlamentares.forms import (LegislaturaForm, MandatoForm)
from sapl.parlamentares.models import (Dependente, Filiacao, Legislatura, from sapl.parlamentares.models import (Dependente, Filiacao, Legislatura,
Mandato, Parlamentar, Partido, Mandato, Parlamentar, Partido,
TipoDependente) TipoDependente)
@ -94,11 +96,11 @@ def test_form_errors_dependente(admin_client):
follow=True) follow=True)
assert (response.context_data['form'].errors['nome'] == assert (response.context_data['form'].errors['nome'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
assert (response.context_data['form'].errors['tipo'] == assert (response.context_data['form'].errors['tipo'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
assert (response.context_data['form'].errors['sexo'] == assert (response.context_data['form'].errors['sexo'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
@pytest.mark.django_db(transaction=False) @pytest.mark.django_db(transaction=False)
@ -112,9 +114,9 @@ def test_form_errors_filiacao(admin_client):
follow=True) follow=True)
assert (response.context_data['form'].errors['partido'] == assert (response.context_data['form'].errors['partido'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
assert (response.context_data['form'].errors['data'] == assert (response.context_data['form'].errors['data'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
@pytest.mark.django_db(transaction=False) @pytest.mark.django_db(transaction=False)
@ -126,14 +128,13 @@ def test_mandato_submit(admin_client):
kwargs={'pk': 14}), kwargs={'pk': 14}),
{'parlamentar': 14, # hidden field {'parlamentar': 14, # hidden field
'legislatura': 5, 'legislatura': 5,
'data_fim_mandato': '2016-01-01',
'data_expedicao_diploma': '2016-03-22', 'data_expedicao_diploma': '2016-03-22',
'observacao': 'Observação do mandato', 'observacao': 'Observação do mandato',
'salvar': 'salvar'}, 'salvar': 'salvar'},
follow=True) follow=True)
mandato = Mandato.objects.first() mandato = Mandato.objects.first()
assert 'Observação do mandato' == mandato.observacao assert str(_('Observação do mandato')) == str(_(mandato.observacao))
@pytest.mark.django_db(transaction=False) @pytest.mark.django_db(transaction=False)
@ -146,6 +147,110 @@ def test_form_errors_mandato(admin_client):
follow=True) follow=True)
assert (response.context_data['form'].errors['legislatura'] == assert (response.context_data['form'].errors['legislatura'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
assert (response.context_data['form'].errors['data_expedicao_diploma'] == assert (response.context_data['form'].errors['data_expedicao_diploma'] ==
['Este campo é obrigatório.']) [_('Este campo é obrigatório.')])
def test_mandato_form_invalido():
form = MandatoForm(data={})
assert not form.is_valid()
errors = form.errors
assert errors['legislatura'] == [_('Este campo é obrigatório.')]
assert errors['parlamentar'] == [_('Este campo é obrigatório.')]
assert errors['data_expedicao_diploma'] == [_('Este campo é obrigatório.')]
@pytest.mark.django_db(transaction=False)
def test_mandato_form_duplicado():
parlamentar = mommy.make(Parlamentar, pk=1)
legislatura = mommy.make(Legislatura, pk=1)
Mandato.objects.create(parlamentar=parlamentar,
legislatura=legislatura,
data_expedicao_diploma='2017-07-25')
form = MandatoForm(data={
'parlamentar': str(parlamentar.pk),
'legislatura': str(legislatura.pk),
'data_expedicao_diploma': '01/07/2015'
})
assert not form.is_valid()
assert form.errors['__all__'] == [_('Mandato nesta legislatura já existe.')]
@pytest.mark.django_db(transaction=False)
def test_mandato_form_datas_invalidas():
parlamentar = mommy.make(Parlamentar, pk=1)
legislatura = mommy.make(Legislatura, pk=1,
data_inicio='2017-01-01',
data_fim='2021-12-31')
form = MandatoForm(data={
'parlamentar': str(parlamentar.pk),
'legislatura': str(legislatura.pk),
'data_expedicao_diploma': '2016-11-01',
'data_inicio_mandato': '2016-12-12',
'data_fim_mandato': '2019-10-09'
})
assert not form.is_valid()
assert form.errors['__all__'] == \
["Data início mandato fora do intervalo de legislatura informada"]
form = MandatoForm(data={
'parlamentar': str(parlamentar.pk),
'legislatura': str(legislatura.pk),
'data_expedicao_diploma': '2016-11-01',
'data_inicio_mandato': '2017-02-02',
'data_fim_mandato': '2022-01-01'
})
assert not form.is_valid()
assert form.errors['__all__'] == \
["Data fim mandato fora do intervalo de legislatura informada"]
def test_legislatura_form_invalido():
legislatura_form = LegislaturaForm(data={})
assert not legislatura_form.is_valid()
errors = legislatura_form.errors
errors['numero'] == [_('Este campo é obrigatório.')]
errors['data_inicio'] == [_('Este campo é obrigatório.')]
errors['data_fim'] == [_('Este campo é obrigatório.')]
errors['data_eleicao'] == [_('Este campo é obrigatório.')]
assert len(errors) == 4
def test_legislatura_form_datas_invalidas():
legislatura_form = LegislaturaForm(data={'numero': '1',
'data_inicio': '2017-02-01',
'data_fim': '2021-12-31',
'data_eleicao': '2017-02-01'
})
assert not legislatura_form.is_valid()
expected = \
_("Data eleição não pode ser inferior a data início da legislatura")
assert legislatura_form.errors['__all__'] == [expected]
legislatura_form = LegislaturaForm(data={'numero': '1',
'data_inicio': '2017-02-01',
'data_fim': '2017-01-01',
'data_eleicao': '2016-11-01'
})
assert not legislatura_form.is_valid()
assert legislatura_form.errors['__all__'] == \
[_("Intervalo de início e fim inválido para legislatura.")]

3
sapl/parlamentares/urls.py

@ -5,8 +5,7 @@ from sapl.parlamentares.views import (CargoMesaCrud, ColigacaoCrud,
FiliacaoCrud, FrenteCrud, FrenteList, FiliacaoCrud, FrenteCrud, FrenteList,
LegislaturaCrud, MandatoCrud, LegislaturaCrud, MandatoCrud,
MesaDiretoraView, NivelInstrucaoCrud, MesaDiretoraView, NivelInstrucaoCrud,
ParlamentarCrud, ParlamentarCrud, ParlamentarMateriasView,
ParlamentarMateriasView,
ParticipacaoParlamentarCrud, PartidoCrud, ParticipacaoParlamentarCrud, PartidoCrud,
ProposicaoParlamentarCrud, ProposicaoParlamentarCrud,
RelatoriaParlamentarCrud, RelatoriaParlamentarCrud,

106
sapl/parlamentares/views.py

@ -1,7 +1,12 @@
from datetime import datetime
import json
from django.contrib import messages from django.contrib import messages
from django.contrib.contenttypes.models import ContentType
from django.core.exceptions import MultipleObjectsReturned, ObjectDoesNotExist from django.core.exceptions import MultipleObjectsReturned, ObjectDoesNotExist
from django.core.urlresolvers import reverse, reverse_lazy from django.core.urlresolvers import reverse, reverse_lazy
from django.db.models import F, Q from django.db.models import F, Q
from django.db.models.aggregates import Count
from django.http import JsonResponse from django.http import JsonResponse
from django.http.response import HttpResponseRedirect from django.http.response import HttpResponseRedirect
from django.templatetags.static import static from django.templatetags.static import static
@ -9,13 +14,16 @@ from django.utils.datastructures import MultiValueDictKeyError
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from django.views.decorators.clickjacking import xframe_options_exempt from django.views.decorators.clickjacking import xframe_options_exempt
from django.views.generic import FormView from django.views.generic import FormView
from django.views.generic.edit import UpdateView
from sapl.base.models import Autor
from sapl.comissoes.models import Participacao from sapl.comissoes.models import Participacao
from sapl.crud.base import (RP_CHANGE, RP_DETAIL, RP_LIST, Crud, CrudAux, from sapl.crud.base import (RP_CHANGE, RP_DETAIL, RP_LIST, Crud, CrudAux,
CrudBaseForListAndDetailExternalAppView, CrudBaseForListAndDetailExternalAppView,
MasterDetailCrud) MasterDetailCrud)
from sapl.materia.models import Proposicao, Relatoria from sapl.materia.models import Autoria, Proposicao, Relatoria
from sapl.parlamentares.apps import AppConfig from sapl.parlamentares.apps import AppConfig
from sapl.utils import parlamentares_ativos
from .forms import (FiliacaoForm, LegislaturaCreateForm, LegislaturaUpdateForm, from .forms import (FiliacaoForm, LegislaturaCreateForm, LegislaturaUpdateForm,
MandatoForm, ParlamentarCreateForm, ParlamentarForm, MandatoForm, ParlamentarCreateForm, ParlamentarForm,
@ -25,15 +33,6 @@ from .models import (CargoMesa, Coligacao, ComposicaoColigacao, ComposicaoMesa,
NivelInstrucao, Parlamentar, Partido, SessaoLegislativa, NivelInstrucao, Parlamentar, Partido, SessaoLegislativa,
SituacaoMilitar, TipoAfastamento, TipoDependente, Votante) SituacaoMilitar, TipoAfastamento, TipoDependente, Votante)
from sapl.base.models import Autor
from sapl.materia.models import Autoria
from django.contrib.contenttypes.models import ContentType
from django.db.models.aggregates import Count
import datetime
import json
CargoMesaCrud = CrudAux.build(CargoMesa, 'cargo_mesa') CargoMesaCrud = CrudAux.build(CargoMesa, 'cargo_mesa')
PartidoCrud = CrudAux.build(Partido, 'partidos') PartidoCrud = CrudAux.build(Partido, 'partidos')
SessaoLegislativaCrud = CrudAux.build(SessaoLegislativa, 'sessao_legislativa') SessaoLegislativaCrud = CrudAux.build(SessaoLegislativa, 'sessao_legislativa')
@ -67,8 +66,7 @@ class VotanteView(MasterDetailCrud):
def delete(self, *args, **kwargs): def delete(self, *args, **kwargs):
obj = self.get_object() obj = self.get_object()
if obj.user: obj.delete()
obj.user.delete()
return HttpResponseRedirect( return HttpResponseRedirect(
reverse('sapl.parlamentares:votante_list', reverse('sapl.parlamentares:votante_list',
kwargs={'pk': obj.parlamentar.pk})) kwargs={'pk': obj.parlamentar.pk}))
@ -189,7 +187,8 @@ class ColigacaoCrud(CrudAux):
ordering = ('-numero_votos', 'nome') ordering = ('-numero_votos', 'nome')
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
context = super(ColigacaoCrud.ListView, self).get_context_data(kwargs=kwargs) context = super(ColigacaoCrud.ListView, self).get_context_data(
kwargs=kwargs)
rows = context['rows'] rows = context['rows']
coluna_votos_recebidos = 2 coluna_votos_recebidos = 2
for row in rows: for row in rows:
@ -201,15 +200,25 @@ class ColigacaoCrud(CrudAux):
class DetailView(CrudAux.DetailView): class DetailView(CrudAux.DetailView):
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
context = super(ColigacaoCrud.DetailView, self).get_context_data(kwargs=kwargs) context = super().get_context_data(kwargs=kwargs)
coligacao = context['coligacao'] coligacao = context['coligacao']
if not coligacao.numero_votos: if not coligacao.numero_votos:
coligacao.numero_votos = '0' coligacao.numero_votos = '0'
context['subnav_template_name'] = \
'parlamentares/subnav_coligacao.yaml'
return context return context
class BaseMixin(CrudAux.BaseMixin): class UpdateView(CrudAux.UpdateView):
subnav_template_name = 'parlamentares/subnav_coligacao.yaml'
def get_context_data(self, **kwargs):
context = super(UpdateView, self).get_context_data(kwargs=kwargs)
context['subnav_template_name'] = \
'parlamentares/subnav_coligacao.yaml'
return context
def json_date_convert(date): def json_date_convert(date):
@ -224,32 +233,6 @@ def json_date_convert(date):
year=int(ano)) year=int(ano))
def parlamentares_ativos(data_inicio, data_fim=None):
'''
:param data_inicio: define a data de inicial do período desejado
:param data_fim: define a data final do período desejado
:return: queryset dos parlamentares ativos naquele período
'''
mandatos_ativos = Mandato.objects.filter(Q(
data_inicio_mandato__lte=data_inicio,
data_fim_mandato__isnull=True) | Q(
data_inicio_mandato__lte=data_inicio,
data_fim_mandato__gte=data_inicio))
if data_fim:
mandatos_ativos = mandatos_ativos | Mandato.objects.filter(
data_inicio_mandato__gte=data_inicio,
data_inicio_mandato__lte=data_fim)
else:
mandatos_ativos = mandatos_ativos | Mandato.objects.filter(
data_inicio_mandato__gte=data_inicio)
parlamentares_id = mandatos_ativos.values_list(
'parlamentar_id',
flat=True).distinct('parlamentar_id')
return Parlamentar.objects.filter(id__in=parlamentares_id)
def frente_atualiza_lista_parlamentares(request): def frente_atualiza_lista_parlamentares(request):
''' '''
:param request: recebe os parâmetros do GET da chamada Ajax :param request: recebe os parâmetros do GET da chamada Ajax
@ -295,6 +278,7 @@ class FrenteCrud(CrudAux):
list_field_names = ['nome', 'data_criacao', 'parlamentares'] list_field_names = ['nome', 'data_criacao', 'parlamentares']
class CreateView(CrudAux.CreateView): class CreateView(CrudAux.CreateView):
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs) context = super().get_context_data(**kwargs)
@ -305,6 +289,7 @@ class FrenteCrud(CrudAux):
return context return context
class UpdateView(CrudAux.UpdateView): class UpdateView(CrudAux.UpdateView):
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs) context = super().get_context_data(**kwargs)
@ -343,7 +328,6 @@ class MandatoCrud(MasterDetailCrud):
return context return context
class CreateView(MasterDetailCrud.CreateView): class CreateView(MasterDetailCrud.CreateView):
form_class = MandatoForm form_class = MandatoForm
@ -388,6 +372,7 @@ class LegislaturaCrud(CrudAux):
form_class = LegislaturaUpdateForm form_class = LegislaturaUpdateForm
class DetailView(CrudAux.DetailView): class DetailView(CrudAux.DetailView):
def has_permission(self): def has_permission(self):
return True return True
@ -396,6 +381,7 @@ class LegislaturaCrud(CrudAux):
return super().get(request, *args, **kwargs) return super().get(request, *args, **kwargs)
class ListView(CrudAux.ListView): class ListView(CrudAux.ListView):
def has_permission(self): def has_permission(self):
return True return True
@ -671,17 +657,20 @@ class MesaDiretoraView(FormView):
not SessaoLegislativa.objects.exists()): not SessaoLegislativa.objects.exists()):
return self.validation(request) return self.validation(request)
sessao = SessaoLegislativa.objects.filter( legislatura = Legislatura.objects.first()
legislatura=Legislatura.objects.first()).first( sessoes = SessaoLegislativa.objects.filter(
) legislatura=legislatura).order_by("data_inicio")
today = datetime.now()
sessao_atual = sessoes.filter(data_inicio__year=today.year).first()
mesa = sessao.composicaomesa_set.all() if sessao else [] mesa = sessao_atual.composicaomesa_set.all() if sessao_atual else []
cargos_ocupados = [m.cargo for m in mesa] cargos_ocupados = [m.cargo for m in mesa]
cargos = CargoMesa.objects.all() cargos = CargoMesa.objects.all()
cargos_vagos = list(set(cargos) - set(cargos_ocupados)) cargos_vagos = list(set(cargos) - set(cargos_ocupados))
parlamentares = Legislatura.objects.first().mandato_set.all() parlamentares = legislatura.mandato_set.all()
parlamentares_ocupados = [m.parlamentar for m in mesa] parlamentares_ocupados = [m.parlamentar for m in mesa]
parlamentares_vagos = list( parlamentares_vagos = list(
set( set(
@ -696,11 +685,9 @@ class MesaDiretoraView(FormView):
return self.render_to_response( return self.render_to_response(
{'legislaturas': Legislatura.objects.all( {'legislaturas': Legislatura.objects.all(
).order_by('-numero'), ).order_by('-numero'),
'legislatura_selecionada': Legislatura.objects.first(), 'legislatura_selecionada': legislatura,
'sessoes': SessaoLegislativa.objects.filter( 'sessoes': sessoes,
legislatura=Legislatura.objects.first()), 'sessao_selecionada': sessao_atual,
'sessao_selecionada': SessaoLegislativa.objects.filter(
legislatura=Legislatura.objects.first()).first(),
'composicao_mesa': mesa, 'composicao_mesa': mesa,
'parlamentares': parlamentares_vagos, 'parlamentares': parlamentares_vagos,
'cargos_vagos': cargos_vagos 'cargos_vagos': cargos_vagos
@ -714,7 +701,6 @@ def altera_field_mesa(request):
operação (Legislatura/Sessão/Inclusão/Remoção), operação (Legislatura/Sessão/Inclusão/Remoção),
atualizando os campos após cada alteração atualizando os campos após cada alteração
""" """
legislatura = request.GET['legislatura'] legislatura = request.GET['legislatura']
sessoes = SessaoLegislativa.objects.filter( sessoes = SessaoLegislativa.objects.filter(
legislatura=legislatura).order_by('-data_inicio') legislatura=legislatura).order_by('-data_inicio')
@ -730,9 +716,11 @@ def altera_field_mesa(request):
# Caso a mudança tenha sido no campo legislatura, a sessão # Caso a mudança tenha sido no campo legislatura, a sessão
# atual deve ser a primeira daquela legislatura # atual deve ser a primeira daquela legislatura
else: else:
sessao_selecionada = SessaoLegislativa.objects.filter( today = datetime.now()
legislatura=legislatura).order_by( try:
'-data_inicio').first().id sessao_selecionada = sessoes.get(data_inicio__year=today.year).id
except ObjectDoesNotExist:
sessao_selecionada = sessoes.first().id
# Atualiza os componentes da view após a mudança # Atualiza os componentes da view após a mudança
composicao_mesa = ComposicaoMesa.objects.filter( composicao_mesa = ComposicaoMesa.objects.filter(
@ -895,6 +883,10 @@ def altera_field_mesa_public_view(request):
# Caso a mudança tenha sido no campo legislatura, a sessão # Caso a mudança tenha sido no campo legislatura, a sessão
# atual deve ser a primeira daquela legislatura # atual deve ser a primeira daquela legislatura
else: else:
try:
today = datetime.now()
sessao_selecionada = sessoes.get(data_inicio__year=today.year).id
except ObjectDoesNotExist as e:
sessao_selecionada = sessoes.first().id sessao_selecionada = sessoes.first().id
# Atualiza os componentes da view após a mudança # Atualiza os componentes da view após a mudança

60
sapl/protocoloadm/forms.py

@ -1,6 +1,5 @@
from datetime import datetime from datetime import datetime
import django_filters
from crispy_forms.bootstrap import InlineRadios from crispy_forms.bootstrap import InlineRadios
from crispy_forms.helper import FormHelper from crispy_forms.helper import FormHelper
from crispy_forms.layout import HTML, Button, Fieldset, Layout, Submit from crispy_forms.layout import HTML, Button, Fieldset, Layout, Submit
@ -9,10 +8,12 @@ from django.core.exceptions import ObjectDoesNotExist, ValidationError
from django.db import models from django.db import models
from django.forms import ModelForm from django.forms import ModelForm
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
import django_filters
from sapl.base.models import Autor from sapl.base.models import Autor, TipoAutor
from sapl.crispy_layout_mixin import form_actions, to_row from sapl.crispy_layout_mixin import form_actions, to_row, SaplFormLayout
from sapl.materia.models import MateriaLegislativa, TipoMateriaLegislativa, UnidadeTramitacao from sapl.materia.models import (MateriaLegislativa, TipoMateriaLegislativa,
UnidadeTramitacao)
from sapl.utils import (RANGE_ANOS, AnoNumeroOrderingFilter, from sapl.utils import (RANGE_ANOS, AnoNumeroOrderingFilter,
RangeWidgetOverride, autor_label, autor_modal) RangeWidgetOverride, autor_label, autor_modal)
@ -20,6 +21,7 @@ from .models import (DocumentoAcessorioAdministrativo, DocumentoAdministrativo,
Protocolo, TipoDocumentoAdministrativo, Protocolo, TipoDocumentoAdministrativo,
TramitacaoAdministrativo) TramitacaoAdministrativo)
TIPOS_PROTOCOLO = [('0', 'Recebido'), ('1', 'Enviado'), ('', 'Ambos')] TIPOS_PROTOCOLO = [('0', 'Recebido'), ('1', 'Enviado'), ('', 'Ambos')]
TIPOS_PROTOCOLO_CREATE = [('0', 'Recebido'), ('1', 'Enviado')] TIPOS_PROTOCOLO_CREATE = [('0', 'Recebido'), ('1', 'Enviado')]
@ -115,7 +117,7 @@ class ProtocoloFilterSet(django_filters.FilterSet):
self.form.helper = FormHelper() self.form.helper = FormHelper()
self.form.helper.form_method = 'GET' self.form.helper.form_method = 'GET'
self.form.helper.layout = Layout( self.form.helper.layout = Layout(
Fieldset('', Fieldset(_('Pesquisar Protocolo'),
row1, row2, row1, row2,
row3, row3,
HTML(autor_label), HTML(autor_label),
@ -217,13 +219,12 @@ class AnularProcoloAdmForm(ModelForm):
cleaned_data = super(AnularProcoloAdmForm, self).clean() cleaned_data = super(AnularProcoloAdmForm, self).clean()
numero = cleaned_data.get("numero") if not self.is_valid():
ano = cleaned_data.get("ano") return cleaned_data
numero = cleaned_data['numero']
ano = cleaned_data['ano']
# se não inserido numero ou ano não prosseguir
# (e ele vai falhar pq numero e ano são obrigatórios)
if not numero or not ano:
return
try: try:
protocolo = Protocolo.objects.get(numero=numero, ano=ano) protocolo = Protocolo.objects.get(numero=numero, ano=ano)
if protocolo.anulado: if protocolo.anulado:
@ -233,6 +234,7 @@ class AnularProcoloAdmForm(ModelForm):
except ObjectDoesNotExist: except ObjectDoesNotExist:
raise forms.ValidationError( raise forms.ValidationError(
_("Protocolo %s/%s não existe" % (numero, ano))) _("Protocolo %s/%s não existe" % (numero, ano)))
exists = False exists = False
if protocolo.tipo_materia: if protocolo.tipo_materia:
exists = MateriaLegislativa.objects.filter( exists = MateriaLegislativa.objects.filter(
@ -245,6 +247,7 @@ class AnularProcoloAdmForm(ModelForm):
raise forms.ValidationError( raise forms.ValidationError(
_("Protocolo %s/%s não pode ser removido pois existem " _("Protocolo %s/%s não pode ser removido pois existem "
"documentos vinculados a ele." % (numero, ano))) "documentos vinculados a ele." % (numero, ano)))
class Meta: class Meta:
model = Protocolo model = Protocolo
fields = ['numero', fields = ['numero',
@ -349,6 +352,11 @@ class ProtocoloMateriaForm(ModelForm):
queryset=Autor.objects.all() queryset=Autor.objects.all()
) )
tipo_autor = forms.ModelChoiceField(required=True,
empty_label='------',
queryset=TipoAutor.objects.all()
)
tipo_materia = forms.ModelChoiceField( tipo_materia = forms.ModelChoiceField(
label=_('Tipo de Matéria'), label=_('Tipo de Matéria'),
required=True, required=True,
@ -364,12 +372,12 @@ class ProtocoloMateriaForm(ModelForm):
assunto_ementa = forms.CharField(required=True, assunto_ementa = forms.CharField(required=True,
widget=forms.Textarea, label='Ementa') widget=forms.Textarea, label='Ementa')
class Meta: class Meta:
model = Protocolo model = Protocolo
fields = ['tipo_materia', fields = ['tipo_materia',
'numero_paginas', 'numero_paginas',
'autor', 'autor',
'tipo_autor',
'assunto_ementa', 'assunto_ementa',
'observacao'] 'observacao']
@ -387,9 +395,9 @@ class ProtocoloMateriaForm(ModelForm):
row1 = to_row( row1 = to_row(
[('tipo_materia', 4), [('tipo_materia', 4),
('numero_paginas', 4)]) ('numero_paginas', 2),
row2 = to_row( ('tipo_autor', 3),
[('autor', 4)]) ('autor', 3)])
row3 = to_row( row3 = to_row(
[('assunto_ementa', 12)]) [('assunto_ementa', 12)])
row4 = to_row( row4 = to_row(
@ -398,7 +406,7 @@ class ProtocoloMateriaForm(ModelForm):
self.helper = FormHelper() self.helper = FormHelper()
self.helper.layout = Layout( self.helper.layout = Layout(
Fieldset(_('Identificação da Matéria'), Fieldset(_('Identificação da Matéria'),
row1, row2, row3, row1, row3,
row4, form_actions(save_label='Protocolar Matéria'))) row4, form_actions(save_label='Protocolar Matéria')))
super(ProtocoloMateriaForm, self).__init__( super(ProtocoloMateriaForm, self).__init__(
@ -559,18 +567,24 @@ class DocumentoAdministrativoForm(ModelForm):
def clean(self): def clean(self):
super(DocumentoAdministrativoForm, self).clean() super(DocumentoAdministrativoForm, self).clean()
numero_protocolo = self.data['numero_protocolo'] cleaned_data = self.cleaned_data
ano_protocolo = self.data['ano_protocolo']
if not self.is_valid():
return cleaned_data
numero_protocolo = cleaned_data['numero_protocolo']
ano_protocolo = cleaned_data['ano_protocolo']
# campos opcionais, mas que se informados devem ser válidos
if numero_protocolo and ano_protocolo: if numero_protocolo and ano_protocolo:
try: try:
self.fields['protocolo'].initial = Protocolo.objects.get( self.fields['protocolo'].initial = Protocolo.objects.get(
numero=numero_protocolo, numero=numero_protocolo,
ano=ano_protocolo).pk ano=ano_protocolo).pk
except ObjectDoesNotExist: except ObjectDoesNotExist:
msg = _('Protocolo %s/%s inexistente' % ( msg = _('Protocolo %s/%s inexistente.' % (
numero_protocolo, ano_protocolo)) numero_protocolo, ano_protocolo))
raise ValidationError(str(msg)) raise ValidationError(msg)
return self.cleaned_data return self.cleaned_data
@ -608,12 +622,10 @@ class DocumentoAdministrativoForm(ModelForm):
[('observacao', 12)]) [('observacao', 12)])
self.helper = FormHelper() self.helper = FormHelper()
self.helper.layout = Layout( self.helper.layout = SaplFormLayout(
Fieldset(_('Identificação Básica'), Fieldset(_('Identificação Básica'),
row1, row2, row3, row4, row5), row1, row2, row3, row4, row5),
Fieldset(_('Outras Informações'), Fieldset(_('Outras Informações'),
row6, row7), row6, row7))
form_actions(more=[Submit('Excluir', 'Excluir')]),
)
super(DocumentoAdministrativoForm, self).__init__( super(DocumentoAdministrativoForm, self).__init__(
*args, **kwargs) *args, **kwargs)

166
sapl/protocoloadm/tests/test_protocoloadm.py

@ -6,11 +6,15 @@ from django.utils.encoding import force_text
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from model_mommy import mommy from model_mommy import mommy
from sapl.materia.models import UnidadeTramitacao from sapl.materia.models import (MateriaLegislativa, UnidadeTramitacao)
from sapl.protocoloadm.forms import AnularProcoloAdmForm from sapl.protocoloadm.forms import (AnularProcoloAdmForm,
DocumentoAdministrativoForm,
ProtocoloDocumentForm,
ProtocoloMateriaForm,)
from sapl.protocoloadm.models import (DocumentoAdministrativo, Protocolo, from sapl.protocoloadm.models import (DocumentoAdministrativo, Protocolo,
StatusTramitacaoAdministrativo, StatusTramitacaoAdministrativo,
TipoDocumentoAdministrativo, TipoDocumentoAdministrativo,
TipoMateriaLegislativa,
TramitacaoAdministrativo) TramitacaoAdministrativo)
@ -250,3 +254,161 @@ def test_create_tramitacao(admin_client):
tramitacao = TramitacaoAdministrativo.objects.last() tramitacao = TramitacaoAdministrativo.objects.last()
# Verifica se a tramitacao que obedece as regras de negócios é criada # Verifica se a tramitacao que obedece as regras de negócios é criada
assert tramitacao.data_tramitacao == datetime.date(2016, 8, 21) assert tramitacao.data_tramitacao == datetime.date(2016, 8, 21)
@pytest.mark.django_db(transaction=False)
def test_anular_protocolo_dados_invalidos():
protocolo = mommy.make(Protocolo, pk=1, numero=1, ano=2017)
form = AnularProcoloAdmForm(data={})
assert not form.is_valid()
errors = form.errors
assert errors['numero'] == [_('Este campo é obrigatório.')]
assert errors['ano'] == [_('Este campo é obrigatório.')]
assert errors['justificativa_anulacao'] == [_('Este campo é obrigatório.')]
assert len(errors) == 3
@pytest.mark.django_db(transaction=False)
def test_anular_protocolo_form_anula_protocolo_inexistente():
form = AnularProcoloAdmForm(data={'numero': '1',
'ano': '2017',
'justificativa_anulacao': 'teste'
})
assert not form.is_valid()
assert form.errors['__all__'] == [_(
'Protocolo 1/2017 não existe')]
@pytest.mark.django_db(transaction=False)
def test_anular_protocolo_form_anula_protocolo_anulado():
protocolo = mommy.make(Protocolo, numero=1, ano=2017, anulado=True)
form = AnularProcoloAdmForm(data={'numero': '1',
'ano': '2017',
'justificativa_anulacao': 'teste'
})
assert not form.is_valid()
assert form.errors['__all__'] == [_(
'Protocolo 1/2017 já encontra-se anulado')]
@pytest.mark.django_db(transaction=False)
def test_anular_protocolo_form_anula_protocolo_com_doc_vinculado():
tipo_materia = mommy.make(TipoMateriaLegislativa)
protocolo_materia = mommy.make(Protocolo,
numero=1,
ano=2017,
tipo_materia=tipo_materia,
anulado=False)
materia_legislativa = mommy.make(MateriaLegislativa,
ano=2017,
numero_protocolo=1)
form = AnularProcoloAdmForm(data={'numero': '1',
'ano': '2017',
'justificativa_anulacao': 'teste'
})
assert not form.is_valid()
assert form.errors['__all__'] == \
[_("Protocolo 1/2017 não pode ser removido pois existem "
"documentos vinculados a ele.")]
tipo_documento = mommy.make(TipoDocumentoAdministrativo)
protocolo_documento = mommy.make(Protocolo,
numero=2,
ano=2017,
tipo_documento=tipo_documento,
anulado=False)
documento_administrativo = mommy.make(DocumentoAdministrativo,
protocolo=protocolo_documento)
form = AnularProcoloAdmForm(data={'numero': '2',
'ano': '2017',
'justificativa_anulacao': 'teste'
})
assert not form.is_valid()
assert form.errors['__all__'] == \
[_("Protocolo 2/2017 não pode ser removido pois existem "
"documentos vinculados a ele.")]
def test_documento_administrativo_invalido():
form = DocumentoAdministrativoForm(data={})
assert not form.is_valid()
errors = form.errors
assert errors['ano'] == [_('Este campo é obrigatório.')]
assert errors['tipo'] == [_('Este campo é obrigatório.')]
assert errors['assunto'] == [_('Este campo é obrigatório.')]
assert errors['numero'] == [_('Este campo é obrigatório.')]
assert errors['data'] == [_('Este campo é obrigatório.')]
assert len(errors) == 5
@pytest.mark.django_db(transaction=False)
def test_documento_administrativo_protocolo_inexistente():
tipo = mommy.make(TipoDocumentoAdministrativo)
form = DocumentoAdministrativoForm(data={'ano': '2017',
'tipo': str(tipo.pk),
'assunto': 'teste',
'numero': '1',
'data': '2017-10-10',
'numero_protocolo': '11',
'ano_protocolo': '2017'
})
assert not form.is_valid()
assert form.errors['__all__'] == [_('Protocolo 11/2017 inexistente.')]
def test_protocolo_documento_form_invalido():
form = ProtocoloDocumentForm(data={})
assert not form.is_valid()
errors = form.errors
assert errors['tipo_protocolo'] == [_('Este campo é obrigatório.')]
assert errors['interessado'] == [_('Este campo é obrigatório.')]
assert errors['tipo_documento'] == [_('Este campo é obrigatório.')]
assert errors['numero_paginas'] == [_('Este campo é obrigatório.')]
assert errors['assunto'] == [_('Este campo é obrigatório.')]
assert len(errors) == 5
def test_protocolo_materia_invalido():
form = ProtocoloMateriaForm(data={})
assert not form.is_valid()
errors = form.errors
assert errors['assunto_ementa'] == [_('Este campo é obrigatório.')]
assert errors['tipo_autor'] == [_('Este campo é obrigatório.')]
assert errors['tipo_materia'] == [_('Este campo é obrigatório.')]
assert errors['numero_paginas'] == [_('Este campo é obrigatório.')]
assert errors['autor'] == [_('Este campo é obrigatório.')]
assert len(errors) == 5

70
sapl/protocoloadm/views.py

@ -1,26 +1,27 @@
from datetime import date, datetime from datetime import date, datetime
from braces.views import FormValidMessageMixin from braces.views import FormValidMessageMixin
from django.contrib import messages from django.contrib import messages
from django.contrib.auth.mixins import PermissionRequiredMixin from django.contrib.auth.mixins import PermissionRequiredMixin
from django.db.models import Q from django.contrib.contenttypes.models import ContentType
from django.core.exceptions import ObjectDoesNotExist from django.core.exceptions import ObjectDoesNotExist
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from django.db.models import Max from django.db.models import Max, Q
from django.http import (Http404, HttpResponse, HttpResponseRedirect, from django.http import Http404, HttpResponse, JsonResponse
JsonResponse)
from django.shortcuts import redirect from django.shortcuts import redirect
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from django.views.generic import CreateView, DetailView, FormView, ListView from django.views.generic import CreateView, ListView
from django.views.generic.base import TemplateView from django.views.generic.base import TemplateView, RedirectView
from django_filters.views import FilterView from django_filters.views import FilterView
import sapl
from sapl.base.models import Autor from sapl.base.models import Autor
from sapl.comissoes.models import Comissao
from sapl.crud.base import Crud, CrudAux, MasterDetailCrud, make_pagination from sapl.crud.base import Crud, CrudAux, MasterDetailCrud, make_pagination
from sapl.materia.models import MateriaLegislativa, TipoMateriaLegislativa from sapl.materia.models import MateriaLegislativa, TipoMateriaLegislativa
from sapl.parlamentares.models import Legislatura, Parlamentar
from sapl.protocoloadm.models import Protocolo
from sapl.utils import create_barcode, get_client_ip from sapl.utils import create_barcode, get_client_ip
import sapl
from .forms import (AnularProcoloAdmForm, DocumentoAcessorioAdministrativoForm, from .forms import (AnularProcoloAdmForm, DocumentoAcessorioAdministrativoForm,
DocumentoAdministrativoFilterSet, DocumentoAdministrativoFilterSet,
@ -28,12 +29,9 @@ from .forms import (AnularProcoloAdmForm, DocumentoAcessorioAdministrativoForm,
ProtocoloFilterSet, ProtocoloMateriaForm, ProtocoloFilterSet, ProtocoloMateriaForm,
TramitacaoAdmEditForm, TramitacaoAdmForm) TramitacaoAdmEditForm, TramitacaoAdmForm)
from .models import (DocumentoAcessorioAdministrativo, DocumentoAdministrativo, from .models import (DocumentoAcessorioAdministrativo, DocumentoAdministrativo,
Protocolo, StatusTramitacaoAdministrativo, StatusTramitacaoAdministrativo,
TipoDocumentoAdministrativo, TramitacaoAdministrativo) TipoDocumentoAdministrativo, TramitacaoAdministrativo)
from sapl.parlamentares.models import Parlamentar
from sapl.protocoloadm.models import Protocolo
from sapl.comissoes.models import Comissao
from django.contrib.contenttypes.models import ContentType
TipoDocumentoAdministrativoCrud = CrudAux.build( TipoDocumentoAdministrativoCrud = CrudAux.build(
TipoDocumentoAdministrativo, '') TipoDocumentoAdministrativo, '')
@ -102,14 +100,21 @@ class DocumentoAdministrativoCrud(Crud):
def list_url(self): def list_url(self):
return '' return ''
class ListView(DocumentoAdministrativoMixin, Crud.ListView): class ListView(RedirectView, DocumentoAdministrativoMixin, Crud.ListView):
pass
class CreateView(DocumentoAdministrativoMixin, Crud.CreateView): def get_redirect_url(self, *args, **kwargs):
namespace = self.model._meta.app_config.name
return reverse('%s:%s' % (namespace, 'pesq_doc_adm'))
class CreateView(Crud.CreateView):
form_class = DocumentoAdministrativoForm form_class = DocumentoAdministrativoForm
layout_key = None layout_key = None
class UpdateView(DocumentoAdministrativoMixin, Crud.UpdateView): @property
def cancel_url(self):
return self.search_url
class UpdateView(Crud.UpdateView):
form_class = DocumentoAdministrativoForm form_class = DocumentoAdministrativoForm
layout_key = None layout_key = None
@ -120,6 +125,7 @@ class DocumentoAdministrativoCrud(Crud):
'numero_protocolo': p.numero} 'numero_protocolo': p.numero}
class DetailView(DocumentoAdministrativoMixin, Crud.DetailView): class DetailView(DocumentoAdministrativoMixin, Crud.DetailView):
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs) context = super().get_context_data(**kwargs)
self.layout_display[0]['rows'][-1][0]['text'] = ( self.layout_display[0]['rows'][-1][0]['text'] = (
@ -128,9 +134,10 @@ class DocumentoAdministrativoCrud(Crud):
kwargs={'pk': self.object.pk})) kwargs={'pk': self.object.pk}))
return context return context
class DeleteView(DocumentoAdministrativoMixin, Crud.DeleteView): class DeleteView(Crud.DeleteView):
def get_success_url(self): def get_success_url(self):
return reverse('sapl.protocoloadm:pesq_doc_adm', kwargs={}) return self.search_url
class StatusTramitacaoAdministrativoCrud(CrudAux): class StatusTramitacaoAdministrativoCrud(CrudAux):
@ -178,7 +185,7 @@ class ProtocoloPesquisaView(PermissionRequiredMixin, FilterView):
context['page_range'] = make_pagination( context['page_range'] = make_pagination(
page_obj.number, paginator.num_pages) page_obj.number, paginator.num_pages)
context['title'] = _('Pesquisa de Protocolos') context['title'] = _('Protocolo')
return context return context
@ -287,6 +294,13 @@ class ProtocoloDocumentoView(PermissionRequiredMixin,
if numeracao == 'A': if numeracao == 'A':
numero = Protocolo.objects.filter( numero = Protocolo.objects.filter(
ano=date.today().year).aggregate(Max('numero')) ano=date.today().year).aggregate(Max('numero'))
elif numeracao == 'L':
legislatura = Legislatura.objects.last()
data_inicio = legislatura.data_inicio
data_fim = legislatura.data_fim
numero = Protocolo.objects.filter(
data__gte=data_inicio, data__lte=data_fim).aggregate(
Max('numero'))
elif numeracao == 'U': elif numeracao == 'U':
numero = Protocolo.objects.all().aggregate(Max('numero')) numero = Protocolo.objects.all().aggregate(Max('numero'))
@ -458,14 +472,20 @@ class ProtocoloMateriaView(PermissionRequiredMixin, CreateView):
return context return context
def autores_ativos(self): def autores_ativos(self):
lista_parlamentares = Parlamentar.objects.filter(ativo=True).values_list('id', flat=True) lista_parlamentares = Parlamentar.objects.filter(
ativo=True).values_list('id', flat=True)
model_parlamentar = ContentType.objects.get_for_model(Parlamentar) model_parlamentar = ContentType.objects.get_for_model(Parlamentar)
autor_parlamentar = Autor.objects.filter(content_type=model_parlamentar, object_id__in=lista_parlamentares) autor_parlamentar = Autor.objects.filter(
content_type=model_parlamentar, object_id__in=lista_parlamentares)
lista_comissoes = Comissao.objects.filter(Q(data_extincao__isnull=True)|Q(data_extincao__gt=date.today())).values_list('id', flat=True) lista_comissoes = Comissao.objects.filter(Q(
data_extincao__isnull=True) | Q(
data_extincao__gt=date.today())).values_list('id', flat=True)
model_comissao = ContentType.objects.get_for_model(Comissao) model_comissao = ContentType.objects.get_for_model(Comissao)
autor_comissoes = Autor.objects.filter(content_type=model_comissao, object_id__in=lista_comissoes) autor_comissoes = Autor.objects.filter(
autores_outros = Autor.objects.exclude(content_type__in=[model_parlamentar, model_comissao]) content_type=model_comissao, object_id__in=lista_comissoes)
autores_outros = Autor.objects.exclude(
content_type__in=[model_parlamentar, model_comissao])
q = autor_parlamentar | autor_comissoes | autores_outros q = autor_parlamentar | autor_comissoes | autores_outros
return q return q

1
sapl/redireciona_urls/tests.py

@ -6,6 +6,7 @@ EMPTY_STRING = ''
class RedirecionaURLsTests(TestCase): class RedirecionaURLsTests(TestCase):
def test_redireciona_index_SAPL(self): def test_redireciona_index_SAPL(self):
response = self.client.get(reverse( response = self.client.get(reverse(
'sapl.redireciona_urls:redireciona_sapl_index') 'sapl.redireciona_urls:redireciona_sapl_index')

20
sapl/redireciona_urls/urls.py

@ -1,24 +1,18 @@
from django.conf.urls import url
from .apps import AppConfig from .apps import AppConfig
from .views import ( from .views import (RedirecionaAtasList, RedirecionaComissao,
RedirecionaAtasList,
RedirecionaComissao,
RedirecionaHistoricoTramitacoesList, RedirecionaHistoricoTramitacoesList,
RedirecionaMateriaLegislativaDetail, RedirecionaMateriaLegislativaDetail,
RedirecionaMateriaLegislativaList, RedirecionaMateriaLegislativaList,
RedirecionaMateriasPorAnoAutorTipo, RedirecionaMateriasPorAnoAutorTipo,
RedirecionaMateriasPorAutor, RedirecionaMateriasPorAutor, RedirecionaMesaDiretoraView,
RedirecionaMesaDiretoraView,
RedirecionaNormasJuridicasDetail, RedirecionaNormasJuridicasDetail,
RedirecionaNormasJuridicasList, RedirecionaNormasJuridicasList, RedirecionaParlamentar,
RedirecionaParlamentar, RedirecionaPautaSessao, RedirecionaPresencaParlamentares,
RedirecionaPautaSessao,
RedirecionaPresencaParlamentares,
RedirecionaRelatoriosList, RedirecionaRelatoriosList,
RedirecionaRelatoriosMateriasEmTramitacaoList, RedirecionaRelatoriosMateriasEmTramitacaoList,
RedirecionaSessaoPlenaria, RedirecionaSAPLIndex, RedirecionaSessaoPlenaria)
RedirecionaSAPLIndex)
from django.conf.urls import url
app_name = AppConfig.name app_name = AppConfig.name

9
sapl/redireciona_urls/views.py

@ -1,17 +1,19 @@
from .exceptions import UnknownUrlNameError
from django.core.urlresolvers import NoReverseMatch, reverse from django.core.urlresolvers import NoReverseMatch, reverse
from django.views.generic import RedirectView from django.views.generic import RedirectView
from sapl.base.apps import AppConfig as atasConfig from sapl.base.apps import AppConfig as atasConfig
from sapl.base.apps import AppConfig as presenca_sessaoConfig
from sapl.base.apps import AppConfig as relatoriosConfig
from sapl.comissoes.apps import AppConfig as comissoesConfig from sapl.comissoes.apps import AppConfig as comissoesConfig
from sapl.materia.apps import AppConfig as materiaConfig from sapl.materia.apps import AppConfig as materiaConfig
from sapl.norma.apps import AppConfig as normaConfig from sapl.norma.apps import AppConfig as normaConfig
from sapl.parlamentares.apps import AppConfig as parlamentaresConfig from sapl.parlamentares.apps import AppConfig as parlamentaresConfig
from sapl.sessao.apps import AppConfig as sessaoConfig from sapl.sessao.apps import AppConfig as sessaoConfig
from .exceptions import UnknownUrlNameError
EMPTY_STRING = '' EMPTY_STRING = ''
presenca_sessaoConfig = relatoriosConfig = atasConfig
app_parlamentares = parlamentaresConfig.name app_parlamentares = parlamentaresConfig.name
app_atas = atasConfig.name app_atas = atasConfig.name
app_presenca_sessao = presenca_sessaoConfig.name app_presenca_sessao = presenca_sessaoConfig.name
@ -50,7 +52,6 @@ relatorio_materia_por_ano_autor_tipo = (
historico_tramitacoes = (app_relatorios + ':historico_tramitacoes') historico_tramitacoes = (app_relatorios + ':historico_tramitacoes')
def has_iframe(url, request): def has_iframe(url, request):
iframe = request.GET.get( iframe = request.GET.get(

33
sapl/relatorios/templates/pdf_capa_processo_preparar_pysc.py

@ -11,7 +11,8 @@ casa={}
aux = context.sapl_documentos.props_sapl.propertyItems() aux = context.sapl_documentos.props_sapl.propertyItems()
for item in aux: for item in aux:
casa[item[0]] = item[1] casa[item[0]] = item[1]
localidade=context.zsql.localidade_obter_zsql(cod_localidade=casa["cod_localidade"]) localidade = context.zsql.localidade_obter_zsql(
cod_localidade=casa["cod_localidade"])
if len(casa["num_cep"]) == 8: if len(casa["num_cep"]) == 8:
cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:] cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:]
else: else:
@ -23,7 +24,8 @@ if cep!="":
linha1 = linha1 + " - " linha1 = linha1 + " - "
linha1 = linha1 + "CEP " + cep linha1 = linha1 + "CEP " + cep
if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None: if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None:
linha1 = linha1 + " - "+localidade[0].nom_localidade+" "+localidade[0].sgl_uf linha1 = linha1 + " - " + \
localidade[0].nom_localidade + " " + localidade[0].sgl_uf
if casa["num_tel"] != "" and casa["num_tel"] != None: if casa["num_tel"] != "" and casa["num_tel"] != None:
linha1 = linha1 + " Tel.: " + casa["num_tel"] linha1 = linha1 + " Tel.: " + casa["num_tel"]
@ -57,10 +59,14 @@ else:
protocolos = [] protocolos = []
REQUEST = context.REQUEST REQUEST = context.REQUEST
for protocolo in context.zsql.protocolo_pesquisar_zsql(tip_protocolo=REQUEST['rad_tip_protocolo'], for protocolo in context.zsql.protocolo_pesquisar_zsql(tip_protocolo=REQUEST['rad_tip_protocolo'],
cod_protocolo=REQUEST['txt_num_protocolo'], ano_protocolo=REQUEST['txt_ano_protocolo'], cod_protocolo=REQUEST['txt_num_protocolo'], ano_protocolo=REQUEST[
tip_documento=REQUEST['lst_tip_documento'], tip_processo=REQUEST['rad_tip_processo'], 'txt_ano_protocolo'],
tip_materia=REQUEST['lst_tip_materia'], des_assunto=REQUEST['txt_assunto'], tip_documento=REQUEST['lst_tip_documento'], tip_processo=REQUEST[
cod_autor=REQUEST['hdn_cod_autor'], des_interessado=REQUEST['txa_txt_interessado'], 'rad_tip_processo'],
tip_materia=REQUEST[
'lst_tip_materia'], des_assunto=REQUEST['txt_assunto'],
cod_autor=REQUEST['hdn_cod_autor'], des_interessado=REQUEST[
'txa_txt_interessado'],
dat_apres=REQUEST['dt_apres'], dat_apres2=REQUEST['dt_apres2']): dat_apres=REQUEST['dt_apres'], dat_apres2=REQUEST['dt_apres2']):
dic = {} dic = {}
@ -68,7 +74,8 @@ for protocolo in context.zsql.protocolo_pesquisar_zsql(tip_protocolo=REQUEST['ra
dic['ano'] = str(protocolo.ano_protocolo) dic['ano'] = str(protocolo.ano_protocolo)
dic['data']=context.pysc.iso_to_port_pysc(protocolo.dat_protocolo)+' - '+protocolo.hor_protocolo dic['data'] = context.pysc.iso_to_port_pysc(
protocolo.dat_protocolo) + ' - ' + protocolo.hor_protocolo
dic['txt_assunto'] = protocolo.txt_assunto_ementa dic['txt_assunto'] = protocolo.txt_assunto_ementa
@ -100,18 +107,21 @@ for protocolo in context.zsql.protocolo_pesquisar_zsql(tip_protocolo=REQUEST['ra
dic['num_materia'] = '' dic['num_materia'] = ''
for materia in context.zsql.materia_obter_zsql(num_protocolo=protocolo.cod_protocolo, ano_ident_basica=protocolo.ano_protocolo): for materia in context.zsql.materia_obter_zsql(num_protocolo=protocolo.cod_protocolo, ano_ident_basica=protocolo.ano_protocolo):
dic['num_materia']=str(materia.num_ident_basica)+'/'+ str(materia.ano_ident_basica) dic['num_materia'] = str(materia.num_ident_basica) + \
'/' + str(materia.ano_ident_basica)
dic['num_documento'] = '' dic['num_documento'] = ''
for documento in context.zsql.documento_administrativo_obter_zsql(num_protocolo=protocolo.cod_protocolo): for documento in context.zsql.documento_administrativo_obter_zsql(num_protocolo=protocolo.cod_protocolo):
dic['num_documento']=str(documento.num_documento)+'/'+ str(documento.ano_documento) dic['num_documento'] = str(
documento.num_documento) + '/' + str(documento.ano_documento)
dic['num_processo'] = dic['num_materia'] or dic['num_documento'] dic['num_processo'] = dic['num_materia'] or dic['num_documento']
dic['numeracao'] = '' dic['numeracao'] = ''
for materia_num in context.zsql.materia_obter_zsql(num_protocolo=protocolo.cod_protocolo, ano_ident_basica=protocolo.ano_protocolo): for materia_num in context.zsql.materia_obter_zsql(num_protocolo=protocolo.cod_protocolo, ano_ident_basica=protocolo.ano_protocolo):
for numera in context.zsql.numeracao_obter_zsql(cod_materia=materia_num.cod_materia, ind_excluido=0): for numera in context.zsql.numeracao_obter_zsql(cod_materia=materia_num.cod_materia, ind_excluido=0):
dic['numeracao']='PROCESSO N&#176; ' +str(numera.num_materia)+'/'+ str(numera.ano_materia) dic['numeracao'] = 'PROCESSO N&#176; ' + \
str(numera.num_materia) + '/' + str(numera.ano_materia)
dic['anulado'] = '' dic['anulado'] = ''
if protocolo.ind_anulado == 1: if protocolo.ind_anulado == 1:
@ -131,7 +141,8 @@ filtro['autor']=REQUEST.hdn_cod_autor
filtro['interessado'] = REQUEST.txa_txt_interessado filtro['interessado'] = REQUEST.txa_txt_interessado
sessao = session.id sessao = session.id
caminho = context.pdf_capa_processo_gerar(sessao,imagem,data,protocolos,cabecalho,rodape,filtro) caminho = context.pdf_capa_processo_gerar(
sessao, imagem, data, protocolos, cabecalho, rodape, filtro)
if caminho == 'aviso': if caminho == 'aviso':
return response.redirect('mensagem_emitir_proc') return response.redirect('mensagem_emitir_proc')
else: else:

200
sapl/relatorios/templates/pdf_detalhe_materia_gerar.py

@ -21,17 +21,21 @@ def cabecalho(dic_inf_basicas,imagem):
tmp += '\t\t\t\t<lines>2cm 24.5cm 19cm 24.5cm</lines>\n' tmp += '\t\t\t\t<lines>2cm 24.5cm 19cm 24.5cm</lines>\n'
if dic_inf_basicas['nom_camara'] != "" and dic_inf_basicas['nom_camara'] != None: if dic_inf_basicas['nom_camara'] != "" and dic_inf_basicas['nom_camara'] != None:
tmp += '\t\t\t\t<setFont name="Helvetica" size="16"/>\n' tmp += '\t\t\t\t<setFont name="Helvetica" size="16"/>\n'
tmp+='\t\t\t\t<drawString x="5cm" y="27.2cm">' + dic_inf_basicas['nom_camara'] + '</drawString>\n' tmp += '\t\t\t\t<drawString x="5cm" y="27.2cm">' + \
dic_inf_basicas['nom_camara'] + '</drawString>\n'
tmp += '\t\t\t\t<setFont name="Helvetica" size="14"/>\n' tmp += '\t\t\t\t<setFont name="Helvetica" size="14"/>\n'
tmp += '\t\t\t\t<drawString x="5cm" y="26.5cm">Sistema de Apoio ao Processo Legislativo</drawString>\n' tmp += '\t\t\t\t<drawString x="5cm" y="26.5cm">Sistema de Apoio ao Processo Legislativo</drawString>\n'
if str(dic_inf_basicas['nom_projeto']) != "" and str(dic_inf_basicas['nom_projeto']) != None: if str(dic_inf_basicas['nom_projeto']) != "" and str(dic_inf_basicas['nom_projeto']) != None:
tmp += '\t\t\t\t<setFont name="Helvetica" size="15"/>\n' tmp += '\t\t\t\t<setFont name="Helvetica" size="15"/>\n'
tmp+='\t\t\t\t<drawCentredString x="10.5cm" y="25.2cm">' + str(dic_inf_basicas['nom_projeto']) + '</drawCentredString>\n' tmp += '\t\t\t\t<drawCentredString x="10.5cm" y="25.2cm">' + \
str(dic_inf_basicas['nom_projeto']) + '</drawCentredString>\n'
if str(dic_inf_basicas['cod_projeto']) != "" and str(dic_inf_basicas['cod_projeto']) != None: if str(dic_inf_basicas['cod_projeto']) != "" and str(dic_inf_basicas['cod_projeto']) != None:
tmp += '\t\t\t\t<setFont name="Helvetica" size="15"/>\n' tmp += '\t\t\t\t<setFont name="Helvetica" size="15"/>\n'
tmp+='\t\t\t\t<drawCentredString x="10.5cm" y="24.7cm">' + str(dic_inf_basicas['cod_projeto']) + '</drawCentredString>\n' tmp += '\t\t\t\t<drawCentredString x="10.5cm" y="24.7cm">' + \
str(dic_inf_basicas['cod_projeto']) + '</drawCentredString>\n'
return tmp return tmp
def rodape(dic_rodape): def rodape(dic_rodape):
""" """
Função que gera o codigo rml do rodape da pagina. Função que gera o codigo rml do rodape da pagina.
@ -60,11 +64,14 @@ def rodape(dic_rodape):
tmp += '\t\t\t\t<setFont name="Helvetica" size="8"/>\n' tmp += '\t\t\t\t<setFont name="Helvetica" size="8"/>\n'
tmp += '\t\t\t\t<drawString x="2cm" y="3.3cm">' + data_emissao + '</drawString>\n' tmp += '\t\t\t\t<drawString x="2cm" y="3.3cm">' + data_emissao + '</drawString>\n'
tmp += '\t\t\t\t<drawString x="17.9cm" y="3.3cm">Página <pageNumber/></drawString>\n' tmp += '\t\t\t\t<drawString x="17.9cm" y="3.3cm">Página <pageNumber/></drawString>\n'
tmp+='\t\t\t\t<drawCentredString x="10.5cm" y="2.7cm">' + linha1 + '</drawCentredString>\n' tmp += '\t\t\t\t<drawCentredString x="10.5cm" y="2.7cm">' + \
tmp+='\t\t\t\t<drawCentredString x="10.5cm" y="2.3cm">' + linha2 + '</drawCentredString>\n' linha1 + '</drawCentredString>\n'
tmp += '\t\t\t\t<drawCentredString x="10.5cm" y="2.3cm">' + \
linha2 + '</drawCentredString>\n'
return tmp return tmp
def paraStyle(): def paraStyle():
"""Função que gera o código rml que define o estilo dos parágrafos""" """Função que gera o código rml que define o estilo dos parágrafos"""
@ -86,6 +93,7 @@ def paraStyle():
return tmp return tmp
def inf_basicas(dic_inf_basicas): def inf_basicas(dic_inf_basicas):
""" """
Função que gera o código rml das funções básicas do relatório Função que gera o código rml das funções básicas do relatório
@ -95,15 +103,18 @@ def inf_basicas(dic_inf_basicas):
# Texto do projeto # Texto do projeto
texto_projeto = str(dic_inf_basicas['texto_projeto']) texto_projeto = str(dic_inf_basicas['texto_projeto'])
if texto_projeto != "" and texto_projeto != None: if texto_projeto != "" and texto_projeto != None:
tmp+='\t\t<para style="texto_projeto">' + texto_projeto.replace('&','&amp;') + '</para>\n' tmp += '\t\t<para style="texto_projeto">' + \
texto_projeto.replace('&', '&amp;') + '</para>\n'
# início das informações básicas # início das informações básicas
tmp += '\t\t<para style="P1">Informações Básicas</para>\n' tmp += '\t\t<para style="P1">Informações Básicas</para>\n'
if str(dic_inf_basicas['apresentada']) != "" and str(dic_inf_basicas['apresentada']) != None: if str(dic_inf_basicas['apresentada']) != "" and str(dic_inf_basicas['apresentada']) != None:
tmp+='\t\t<para style="P2"><b>Apresentada em: </b> ' + str(dic_inf_basicas['apresentada']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Apresentada em: </b> ' + \
str(dic_inf_basicas['apresentada']) + '</para>\n'
if str(dic_inf_basicas['formato']) != "" and str(dic_inf_basicas['formato']) != None: if str(dic_inf_basicas['formato']) != "" and str(dic_inf_basicas['formato']) != None:
tmp+='\t\t<para style="P2"><b>Formato: </b> ' + str(dic_inf_basicas['formato']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Formato: </b> ' + \
str(dic_inf_basicas['formato']) + '</para>\n'
if dic_inf_basicas['publicada'] == 0: if dic_inf_basicas['publicada'] == 0:
tmp += '\t\t<para style="P2"><b>Publicada:</b> Não</para>\n' tmp += '\t\t<para style="P2"><b>Publicada:</b> Não</para>\n'
@ -111,7 +122,8 @@ def inf_basicas(dic_inf_basicas):
tmp += '\t\t<para style="P2"><b>Publicada:</b> Sim</para>\n' tmp += '\t\t<para style="P2"><b>Publicada:</b> Sim</para>\n'
if str(dic_inf_basicas['objeto']) != "" and str(dic_inf_basicas['objeto']) != None: if str(dic_inf_basicas['objeto']) != "" and str(dic_inf_basicas['objeto']) != None:
tmp+='\t\t<para style="P2"><b>Objeto: </b> ' + str(dic_inf_basicas['objeto']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Objeto: </b> ' + \
str(dic_inf_basicas['objeto']) + '</para>\n'
if dic_inf_basicas['tramitacao'] == 0: if dic_inf_basicas['tramitacao'] == 0:
tmp += '\t\t<para style="P2"><b>Tramitação:</b> Não</para>\n' tmp += '\t\t<para style="P2"><b>Tramitação:</b> Não</para>\n'
@ -119,13 +131,16 @@ def inf_basicas(dic_inf_basicas):
tmp += '\t\t<para style="P2"><b>Tramitação:</b> Sim</para>\n' tmp += '\t\t<para style="P2"><b>Tramitação:</b> Sim</para>\n'
if str(dic_inf_basicas['reg_tramitacao']) != "" and str(dic_inf_basicas['reg_tramitacao']) != None: if str(dic_inf_basicas['reg_tramitacao']) != "" and str(dic_inf_basicas['reg_tramitacao']) != None:
tmp+='\t\t<para style="P2"><b>Regime: </b> ' + str(dic_inf_basicas['reg_tramitacao']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Regime: </b> ' + \
str(dic_inf_basicas['reg_tramitacao']) + '</para>\n'
if str(dic_inf_basicas['prazo']) != "" and str(dic_inf_basicas['prazo']) != None: if str(dic_inf_basicas['prazo']) != "" and str(dic_inf_basicas['prazo']) != None:
tmp+='\t\t<para style="P2"><b>Dias de prazo: </b> ' + str(dic_inf_basicas['prazo']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Dias de prazo: </b> ' + \
str(dic_inf_basicas['prazo']) + '</para>\n'
if str(dic_inf_basicas['fim_prazo']) != "" and str(dic_inf_basicas['fim_prazo']) != None: if str(dic_inf_basicas['fim_prazo']) != "" and str(dic_inf_basicas['fim_prazo']) != None:
tmp+='\t\t<para style="P2"><b>Data do fim do prazo: </b> ' + str(dic_inf_basicas['fim_prazo']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Data do fim do prazo: </b> ' + \
str(dic_inf_basicas['fim_prazo']) + '</para>\n'
if dic_inf_basicas['mat_complementar'] == 0: if dic_inf_basicas['mat_complementar'] == 0:
tmp += '\t\t<para style="P2"><b>Matéria Complementar:</b> Não</para>\n' tmp += '\t\t<para style="P2"><b>Matéria Complementar:</b> Não</para>\n'
@ -139,18 +154,22 @@ def inf_basicas(dic_inf_basicas):
apelido = dic_inf_basicas['apelido'] apelido = dic_inf_basicas['apelido']
if apelido != "" and apelido != None: if apelido != "" and apelido != None:
tmp+='\t\t<para style="P2"><b>Apelido: </b> ' + apelido.replace('&','&amp;') + '</para>\n' tmp += '\t\t<para style="P2"><b>Apelido: </b> ' + \
apelido.replace('&', '&amp;') + '</para>\n'
indexacao = dic_inf_basicas['indexacao'] indexacao = dic_inf_basicas['indexacao']
if indexacao != "" and indexacao != None: if indexacao != "" and indexacao != None:
tmp+='\t\t<para style="P2"><b>Indexação: </b> ' + indexacao.replace('&','&amp;') + '</para>\n' tmp += '\t\t<para style="P2"><b>Indexação: </b> ' + \
indexacao.replace('&', '&amp;') + '</para>\n'
observacao = dic_inf_basicas['observacao'] observacao = dic_inf_basicas['observacao']
if observacao != "" and observacao != None: if observacao != "" and observacao != None:
tmp+='\t\t<para style="P2"><b>Observação: </b> ' + observacao.replace('&','&amp;') + '</para>\n' tmp += '\t\t<para style="P2"><b>Observação: </b> ' + \
observacao.replace('&', '&amp;') + '</para>\n'
return tmp return tmp
def orig_externa(dic_orig_externa): def orig_externa(dic_orig_externa):
""" """
Função que gera o código rml da origem externa Função que gera o código rml da origem externa
@ -160,46 +179,60 @@ def orig_externa(dic_orig_externa):
tmp += '\t\t<para style="P1">Origem Externa</para>\n' tmp += '\t\t<para style="P1">Origem Externa</para>\n'
try: try:
if dic_orig_externa['local'] != "" and dic_orig_externa['local'] != None: if dic_orig_externa['local'] != "" and dic_orig_externa['local'] != None:
tmp+='\t\t<para style="P2"><b>Local:</b> ' + dic_orig_externa['local'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Local:</b> ' + \
dic_orig_externa['local'] + '</para>\n'
if dic_orig_externa['data'] != "" and dic_orig_externa['data'] != None: if dic_orig_externa['data'] != "" and dic_orig_externa['data'] != None:
tmp+='\t\t<para style="P2"><b>Data:</b> ' + dic_orig_externa['data'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Data:</b> ' + \
dic_orig_externa['data'] + '</para>\n'
if dic_orig_externa['tipo'] != "" and dic_orig_externa['tipo'] != None: if dic_orig_externa['tipo'] != "" and dic_orig_externa['tipo'] != None:
tmp+='\t\t<para style="P2"><b>Tipo:</b> ' + dic_orig_externa['tipo'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Tipo:</b> ' + \
dic_orig_externa['tipo'] + '</para>\n'
if dic_orig_externa['numero_ano'] != "" and dic_orig_externa['numero_ano'] != None: if dic_orig_externa['numero_ano'] != "" and dic_orig_externa['numero_ano'] != None:
tmp+='\t\t<para style="P2"><b>Número/Ano:</b> ' + dic_orig_externa['numero_ano'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Número/Ano:</b> ' + \
except: pass dic_orig_externa['numero_ano'] + '</para>\n'
except:
pass
return tmp return tmp
def mat_anexadas(lst_mat_anexadas): def mat_anexadas(lst_mat_anexadas):
tmp = '' tmp = ''
tmp += '\t\t<para style="P1">Matérias Anexadas</para>\n' tmp += '\t\t<para style="P1">Matérias Anexadas</para>\n'
for dic_mat in lst_mat_anexadas: for dic_mat in lst_mat_anexadas:
if dic_mat['nom_mat'] != "" and dic_mat['nom_mat'] != None: if dic_mat['nom_mat'] != "" and dic_mat['nom_mat'] != None:
tmp+='\t\t<para style="P2"><b>Nome da matéria:</b> ' + dic_mat['nom_mat'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Nome da matéria:</b> ' + \
tmp+='\t\t<para style="P2"><b>Data:</b> ' + dic_mat['data'] + '</para>\n' dic_mat['nom_mat'] + '</para>\n'
tmp+='\t\t<para style="P2"><b>Data final:</b> ' + str(dic_mat['data_fim']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Data:</b> ' + \
dic_mat['data'] + '</para>\n'
tmp += '\t\t<para style="P2"><b>Data final:</b> ' + \
str(dic_mat['data_fim']) + '</para>\n'
return tmp return tmp
def autoria(lst_autoria): def autoria(lst_autoria):
tmp = '' tmp = ''
tmp += '\t\t<para style="P1">Autores</para>\n' tmp += '\t\t<para style="P1">Autores</para>\n'
for dic_autor in lst_autoria: for dic_autor in lst_autoria:
if dic_autor['nom_autor'] != "" and dic_autor['nom_autor'] != None: if dic_autor['nom_autor'] != "" and dic_autor['nom_autor'] != None:
tmp+='\t\t<para style="P2"><b>Nome do Autor:</b> ' + dic_autor['nom_autor'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Nome do Autor:</b> ' + \
dic_autor['nom_autor'] + '</para>\n'
if dic_autor['nom_autor'] != "" and dic_autor['cargo'] != None: if dic_autor['nom_autor'] != "" and dic_autor['cargo'] != None:
tmp+='\t\t<para style="P2"><b>Cargo:</b> ' + dic_autor['cargo'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Cargo:</b> ' + \
dic_autor['cargo'] + '</para>\n'
if dic_autor['nom_autor'] != "" and dic_autor['tipo'] != None: if dic_autor['nom_autor'] != "" and dic_autor['tipo'] != None:
tmp+='\t\t<para style="P2"><b>Tipo:</b> ' + dic_autor['tipo'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Tipo:</b> ' + \
dic_autor['tipo'] + '</para>\n'
return tmp return tmp
def despachos_iniciais(lst_des_iniciais): def despachos_iniciais(lst_des_iniciais):
tmp = '' tmp = ''
@ -207,94 +240,137 @@ def despachos_iniciais(lst_des_iniciais):
for dic_dados in lst_des_iniciais: for dic_dados in lst_des_iniciais:
if dic_dados['nom_comissao'] == None: if dic_dados['nom_comissao'] == None:
dic_dados['nom_comissao'] = " " dic_dados['nom_comissao'] = " "
tmp+='\t\t<para style="P2"><b>Nome da comissão:</b> ' + dic_dados['nom_comissao'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Nome da comissão:</b> ' + \
dic_dados['nom_comissao'] + '</para>\n'
return tmp return tmp
def tramitacoes(dic_tramitacoes): def tramitacoes(dic_tramitacoes):
tmp = '' tmp = ''
tmp += '\t\t<para style="P1">Última Tramitação</para>\n' tmp += '\t\t<para style="P1">Última Tramitação</para>\n'
try: try:
tmp+='\t\t<para style="P2"><b>Data Ação:</b> ' + str(dic_tramitacoes['data']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Data Ação:</b> ' + \
tmp+='\t\t<para style="P2"><b>Unidade Local:</b> ' + dic_tramitacoes['unidade'] + '</para>\n' str(dic_tramitacoes['data']) + '</para>\n'
tmp+='\t\t<para style="P2"><b>Encaminhada em:</b> ' + str(dic_tramitacoes['data_enc']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Unidade Local:</b> ' + \
tmp+='\t\t<para style="P2"><b>Destino:</b> ' + dic_tramitacoes['destino'] + '</para>\n' dic_tramitacoes['unidade'] + '</para>\n'
tmp+='\t\t<para style="P2"><b>Turno:</b> ' + dic_tramitacoes['turno'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Encaminhada em:</b> ' + \
tmp+='\t\t<para style="P2"><b>Status:</b> ' + dic_tramitacoes['status'] + '</para>\n' str(dic_tramitacoes['data_enc']) + '</para>\n'
tmp += '\t\t<para style="P2"><b>Destino:</b> ' + \
dic_tramitacoes['destino'] + '</para>\n'
tmp += '\t\t<para style="P2"><b>Turno:</b> ' + \
dic_tramitacoes['turno'] + '</para>\n'
tmp += '\t\t<para style="P2"><b>Status:</b> ' + \
dic_tramitacoes['status'] + '</para>\n'
if dic_tramitacoes['urgente'] == 0: if dic_tramitacoes['urgente'] == 0:
tmp += '\t\t<para style="P2"><b>Urgente:</b> Não</para>\n' tmp += '\t\t<para style="P2"><b>Urgente:</b> Não</para>\n'
else: else:
tmp += '\t\t<para style="P2"><b>Urgente:</b> Sim</para>\n' tmp += '\t\t<para style="P2"><b>Urgente:</b> Sim</para>\n'
tmp+='\t\t<para style="P2"><b>Data do fim do prazo:</b> ' + str(dic_tramitacoes['data_fim']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Data do fim do prazo:</b> ' + \
str(dic_tramitacoes['data_fim']) + '</para>\n'
if dic_tramitacoes['texto_acao'] != "" and dic_tramitacoes['texto_acao'] != None: if dic_tramitacoes['texto_acao'] != "" and dic_tramitacoes['texto_acao'] != None:
tmp+='\t\t<para style="P2"><b>Texto da Ação:</b> ' + dic_tramitacoes['texto_acao'].replace('&','&amp;') + '</para>\n' tmp += '\t\t<para style="P2"><b>Texto da Ação:</b> ' + \
dic_tramitacoes['texto_acao'].replace(
'&', '&amp;') + '</para>\n'
except: pass except:
pass
return tmp return tmp
def relatorias(lst_relatorias): def relatorias(lst_relatorias):
tmp = '' tmp = ''
tmp += '\t\t<para style="P1">Relatorias</para>\n' tmp += '\t\t<para style="P1">Relatorias</para>\n'
for dic_comissao in lst_relatorias: for dic_comissao in lst_relatorias:
tmp+='\t\t<para style="P2"><b>Comissão:</b> ' + dic_comissao['nom_comissao'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Comissão:</b> ' + \
tmp+='\t\t<para style="P2"><b>Data Designação:</b> ' + str(dic_comissao['data_desig']) + '</para>\n' dic_comissao['nom_comissao'] + '</para>\n'
tmp+='\t\t<para style="P2"><b>Parlamentar:</b> ' + dic_comissao['parlamentar'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Data Designação:</b> ' + \
tmp+='\t\t<para style="P2"><b>Data Destituição:</b> ' + str(dic_comissao['data_dest']) + '</para>\n' str(dic_comissao['data_desig']) + '</para>\n'
tmp+='\t\t<para style="P2"><b>Motivo Fim Relatoria:</b> ' + dic_comissao['motivo'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Parlamentar:</b> ' + \
dic_comissao['parlamentar'] + '</para>\n'
tmp += '\t\t<para style="P2"><b>Data Destituição:</b> ' + \
str(dic_comissao['data_dest']) + '</para>\n'
tmp += '\t\t<para style="P2"><b>Motivo Fim Relatoria:</b> ' + \
dic_comissao['motivo'] + '</para>\n'
return tmp return tmp
def numeracoes(lst_numeracoes): def numeracoes(lst_numeracoes):
tmp = '' tmp = ''
tmp += '\t\t<para style="P1">Numerações</para>\n' tmp += '\t\t<para style="P1">Numerações</para>\n'
for dic_dados in lst_numeracoes: for dic_dados in lst_numeracoes:
tmp+='\t\t<para style="P2"><b>Nome:</b> ' + dic_dados['nome'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Nome:</b> ' + \
tmp+='\t\t<para style="P2"><b>Ano:</b> ' + str(dic_dados['ano']) + '</para>\n' dic_dados['nome'] + '</para>\n'
tmp += '\t\t<para style="P2"><b>Ano:</b> ' + \
str(dic_dados['ano']) + '</para>\n'
return tmp return tmp
def legislacoes_citadas(lst_leg_citadas): def legislacoes_citadas(lst_leg_citadas):
tmp = '' tmp = ''
tmp += '\t\t<para style="P1">Legislações Citadas</para>\n' tmp += '\t\t<para style="P1">Legislações Citadas</para>\n'
for dic_dados in lst_leg_citadas: for dic_dados in lst_leg_citadas:
tmp+='\t\t<para style="P2"><b>Tipo Norma:</b> ' + str(dic_dados['nome_lei']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Tipo Norma:</b> ' + \
tmp+='\t\t<para style="P2"><b>Disposição:</b> ' + str(dic_dados['disposicao']) + '</para>\n' str(dic_dados['nome_lei']) + '</para>\n'
tmp+='\t\t<para style="P2"><b>Parte:</b> ' + str(dic_dados['parte']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Disposição:</b> ' + \
tmp+='\t\t<para style="P2"><b>Livro:</b> ' + str(dic_dados['livro']) + '</para>\n' str(dic_dados['disposicao']) + '</para>\n'
tmp+='\t\t<para style="P2"><b>Tí­tulo:</b> ' + str(dic_dados['titulo']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Parte:</b> ' + \
tmp+='\t\t<para style="P2"><b>Capí­tulo:</b> ' + str(dic_dados['capitulo']) + '</para>\n' str(dic_dados['parte']) + '</para>\n'
tmp+='\t\t<para style="P2"><b>Seção:</b> ' + str(dic_dados['secao']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Livro:</b> ' + \
tmp+='\t\t<para style="P2"><b>Subseção:</b> ' + str(dic_dados['subsecao']) + '</para>\n' str(dic_dados['livro']) + '</para>\n'
tmp+='\t\t<para style="P2"><b>Artigo:</b> ' + str(dic_dados['artigo']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Tí­tulo:</b> ' + \
tmp+='\t\t<para style="P2"><b>Parágrafo:</b> ' + str(dic_dados['paragrafo']) + '</para>\n' str(dic_dados['titulo']) + '</para>\n'
tmp+='\t\t<para style="P2"><b>Inciso:</b> ' + str(dic_dados['inciso']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Capí­tulo:</b> ' + \
tmp+='\t\t<para style="P2"><b>Alí­nea:</b> ' + str(dic_dados['alinea']) + '</para>\n' str(dic_dados['capitulo']) + '</para>\n'
tmp+='\t\t<para style="P2"><b>Item:</b> ' + str(dic_dados['item']) + '</para>\n' tmp += '\t\t<para style="P2"><b>Seção:</b> ' + \
str(dic_dados['secao']) + '</para>\n'
tmp += '\t\t<para style="P2"><b>Subseção:</b> ' + \
str(dic_dados['subsecao']) + '</para>\n'
tmp += '\t\t<para style="P2"><b>Artigo:</b> ' + \
str(dic_dados['artigo']) + '</para>\n'
tmp += '\t\t<para style="P2"><b>Parágrafo:</b> ' + \
str(dic_dados['paragrafo']) + '</para>\n'
tmp += '\t\t<para style="P2"><b>Inciso:</b> ' + \
str(dic_dados['inciso']) + '</para>\n'
tmp += '\t\t<para style="P2"><b>Alí­nea:</b> ' + \
str(dic_dados['alinea']) + '</para>\n'
tmp += '\t\t<para style="P2"><b>Item:</b> ' + \
str(dic_dados['item']) + '</para>\n'
return tmp return tmp
def documentos_acessorios(lst_acessorios): def documentos_acessorios(lst_acessorios):
tmp = '' tmp = ''
tmp += '\t\t<para style="P1">Documentos Acessórios</para>\n' tmp += '\t\t<para style="P1">Documentos Acessórios</para>\n'
for dic_dados in lst_acessorios: for dic_dados in lst_acessorios:
if dic_dados['tipo'] != None: if dic_dados['tipo'] != None:
tmp+='\t\t<para style="P2"><b>Tipo:</b> ' + dic_dados['tipo'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Tipo:</b> ' + \
dic_dados['tipo'] + '</para>\n'
if dic_dados['nome'] != None: if dic_dados['nome'] != None:
tmp+='\t\t<para style="P2"><b>Nome:</b> ' + dic_dados['nome'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Nome:</b> ' + \
dic_dados['nome'] + '</para>\n'
tmp+='\t\t<para style="P2"><b>Data:</b> ' + dic_dados['data'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Data:</b> ' + \
dic_dados['data'] + '</para>\n'
if dic_dados['autor'] != None: if dic_dados['autor'] != None:
tmp+='\t\t<para style="P2"><b>Autor:</b> ' + dic_dados['autor'] + '</para>\n' tmp += '\t\t<para style="P2"><b>Autor:</b> ' + \
dic_dados['autor'] + '</para>\n'
if dic_dados['ementa'] != None: if dic_dados['ementa'] != None:
tmp+='\t\t<para style="P2"><b>Ementa:</b> ' + dic_dados['ementa'].replace('&','&amp;') + '</para>\n' tmp += '\t\t<para style="P2"><b>Ementa:</b> ' + \
dic_dados['ementa'].replace('&', '&amp;') + '</para>\n'
if dic_dados['indexacao'] != None: if dic_dados['indexacao'] != None:
tmp+='\t\t<para style="P2"><b>Ementa:</b> ' + dic_dados['indexacao'].replace('&','&amp;') + '</para>\n' tmp += '\t\t<para style="P2"><b>Ementa:</b> ' + \
dic_dados['indexacao'].replace('&', '&amp;') + '</para>\n'
return tmp return tmp
def principal(imagem, dic_rodape, dic_inf_basicas, dic_orig_externa, lst_mat_anexadas, lst_autoria, lst_des_iniciais, def principal(imagem, dic_rodape, dic_inf_basicas, dic_orig_externa, lst_mat_anexadas, lst_autoria, lst_des_iniciais,
dic_tramitacoes, lst_relatorias, lst_numeracoes, lst_leg_citadas, lst_acessorios, sessao=''): dic_tramitacoes, lst_relatorias, lst_numeracoes, lst_leg_citadas, lst_acessorios, sessao=''):
""" """

59
sapl/relatorios/templates/pdf_detalhe_materia_preparar_pysc.py

@ -17,7 +17,8 @@ casa={}
aux = context.sapl_documentos.props_sapl.propertyItems() aux = context.sapl_documentos.props_sapl.propertyItems()
for item in aux: for item in aux:
casa[item[0]] = item[1] casa[item[0]] = item[1]
localidade=context.zsql.localidade_obter_zsql(cod_localidade=casa["cod_localidade"]) localidade = context.zsql.localidade_obter_zsql(
cod_localidade=casa["cod_localidade"])
data_emissao = DateTime().strftime("%d/%m/%Y") data_emissao = DateTime().strftime("%d/%m/%Y")
rodape = casa rodape = casa
rodape['data_emissao'] = data_emissao rodape['data_emissao'] = data_emissao
@ -39,7 +40,8 @@ for materia in context.zsql.materia_obter_zsql(cod_materia=REQUEST['cod_materia'
inf_basicas_dic['publicada'] = materia.dat_publicacao inf_basicas_dic['publicada'] = materia.dat_publicacao
inf_basicas_dic['objeto'] = materia.des_objeto inf_basicas_dic['objeto'] = materia.des_objeto
inf_basicas_dic['tramitacao'] = materia.ind_tramitacao inf_basicas_dic['tramitacao'] = materia.ind_tramitacao
inf_basicas_dic['cod_projeto']= materia.sgl_tipo_materia+" "+ str(materia.num_ident_basica)+" de "+ str(materia.ano_ident_basica) inf_basicas_dic['cod_projeto'] = materia.sgl_tipo_materia + " " + \
str(materia.num_ident_basica) + " de " + str(materia.ano_ident_basica)
inf_basicas_dic['nom_projeto'] = materia.des_tipo_materia inf_basicas_dic['nom_projeto'] = materia.des_tipo_materia
for tramitacao in context.zsql.regime_tramitacao_obter_zsql(cod_regime_tramitacao=materia.cod_regime_tramitacao): for tramitacao in context.zsql.regime_tramitacao_obter_zsql(cod_regime_tramitacao=materia.cod_regime_tramitacao):
@ -76,14 +78,17 @@ for materia in context.zsql.materia_obter_zsql(cod_materia=REQUEST['cod_materia'
orig_externa_dic['local'] = origem.sgl_origem + "-" + origem.nom_origem orig_externa_dic['local'] = origem.sgl_origem + "-" + origem.nom_origem
orig_externa_dic['tipo'] = materia.tip_origem_externa orig_externa_dic['tipo'] = materia.tip_origem_externa
orig_externa_dic['data'] = materia.dat_origem_externa orig_externa_dic['data'] = materia.dat_origem_externa
orig_externa_dic['numero_ano']= str(materia.num_origem_externa)+ "/"+ str(materia.ano_origem_externa) orig_externa_dic['numero_ano'] = str(
materia.num_origem_externa) + "/" + str(materia.ano_origem_externa)
# #o bloco abaixo gera o dicionario das materias anexadas (ln 55) # #o bloco abaixo gera o dicionario das materias anexadas (ln 55)
lst_mat_anexadas = [] lst_mat_anexadas = []
dic_mat = {} dic_mat = {}
for anexada in context.zsql.anexada_obter_zsql(cod_materia_principal=materia.cod_materia): for anexada in context.zsql.anexada_obter_zsql(cod_materia_principal=materia.cod_materia):
aux1 = context.zsql.materia_obter_zsql(cod_materia = anexada.cod_materia_anexada) aux1 = context.zsql.materia_obter_zsql(
aux2 = context.zsql.tipo_materia_legislativa_obter_zsql(tip_materia = aux1[0].tip_id_basica) cod_materia=anexada.cod_materia_anexada)
aux2 = context.zsql.tipo_materia_legislativa_obter_zsql(
tip_materia=aux1[0].tip_id_basica)
# """#tratando possíveis erros""" # """#tratando possíveis erros"""
# if aux2.sgl_tipo_materia==None: aux2.sgl_tipo_materia="" # if aux2.sgl_tipo_materia==None: aux2.sgl_tipo_materia=""
# if aux2.num_ident_basica==None: aux2.num_ident_basica="" # if aux2.num_ident_basica==None: aux2.num_ident_basica=""
@ -91,7 +96,8 @@ for materia in context.zsql.materia_obter_zsql(cod_materia=REQUEST['cod_materia'
# if anexadas.dat_anexacao==None: anexadas.dat_anexacao="" # if anexadas.dat_anexacao==None: anexadas.dat_anexacao=""
# if anexadas.dat_desanexacao==None: anexadas.dat_desanexacao="" # if anexadas.dat_desanexacao==None: anexadas.dat_desanexacao=""
# """#""" # """#"""
dic_mat['nom_mat']= aux2[0].sgl_tipo_materia+ "/"+ str(aux1[0].num_ident_basica)+ "/"+ str(aux1[0].ano_ident_basica) dic_mat['nom_mat'] = aux2[0].sgl_tipo_materia + "/" + \
str(aux1[0].num_ident_basica) + "/" + str(aux1[0].ano_ident_basica)
dic_mat['data'] = anexada.dat_anexacao dic_mat['data'] = anexada.dat_anexacao
dic_mat['data_fim'] = anexada.dat_desanexacao dic_mat['data_fim'] = anexada.dat_desanexacao
lst_mat_anexadas.append(dic_mat) lst_mat_anexadas.append(dic_mat)
@ -139,7 +145,8 @@ for materia in context.zsql.materia_obter_zsql(cod_materia=REQUEST['cod_materia'
comissao.nom_comissao = '' comissao.nom_comissao = ''
if comissao.sgl_comissao == None: if comissao.sgl_comissao == None:
comissao.sgl_comissao = '' comissao.sgl_comissao = ''
dic_dados['nom_comissao']=comissao.nom_comissao+ " - "+ comissao.sgl_comissao dic_dados['nom_comissao'] = comissao.nom_comissao + \
" - " + comissao.sgl_comissao
lst_des_iniciais.append(dic_dados) lst_des_iniciais.append(dic_dados)
# #o bloco abaixo gera o dicionário de Tramitacoes(ln 87) # #o bloco abaixo gera o dicionário de Tramitacoes(ln 87)
@ -153,9 +160,8 @@ for materia in context.zsql.materia_obter_zsql(cod_materia=REQUEST['cod_materia'
dic_tramitacoes['data_fim'] = tramitacao.dat_fim_prazo dic_tramitacoes['data_fim'] = tramitacao.dat_fim_prazo
dic_tramitacoes['texto_acao'] = tramitacao.txt_tramitacao dic_tramitacoes['texto_acao'] = tramitacao.txt_tramitacao
for unidade in context.zsql.unidade_tramitacao_obter_zsql(cod_unid_tramitacao=tramitacao.cod_unid_tram_local): for unidade in context.zsql.unidade_tramitacao_obter_zsql(cod_unid_tramitacao=tramitacao.cod_unid_tram_local):
#-----------------se unidade for comissao--------------------------------- #-----------------se unidade for comissao--------------------------
if unidade.cod_orgao == None: if unidade.cod_orgao == None:
for comissao in context.zsql.comissao_obter_zsql(cod_comissao=unidade.cod_comissao): for comissao in context.zsql.comissao_obter_zsql(cod_comissao=unidade.cod_comissao):
if tramitacao.cod_unid_tram_dest != None: if tramitacao.cod_unid_tram_dest != None:
@ -163,17 +169,21 @@ for materia in context.zsql.materia_obter_zsql(cod_materia=REQUEST['cod_materia'
# se unidade destino for comissao # se unidade destino for comissao
if unidade_dest.cod_orgao == None: if unidade_dest.cod_orgao == None:
for comissao_dest in context.zsql.comissao_obter_zsql(cod_comissao=unidade_dest.cod_comissao): for comissao_dest in context.zsql.comissao_obter_zsql(cod_comissao=unidade_dest.cod_comissao):
dic_tramitacoes['unidade']= comissao.nom_comissao dic_tramitacoes[
dic_tramitacoes['destino']= comissao_dest.nom_comissao 'unidade'] = comissao.nom_comissao
dic_tramitacoes[
'destino'] = comissao_dest.nom_comissao
# se unidade destino for orgao # se unidade destino for orgao
if unidade_dest.cod_comissao == None: if unidade_dest.cod_comissao == None:
for orgao_dest in context.zsql.orgao_obter_zsql(cod_orgao=unidade_dest.cod_orgao): for orgao_dest in context.zsql.orgao_obter_zsql(cod_orgao=unidade_dest.cod_orgao):
dic_tramitacoes['unidade']= comissao.nom_comissao dic_tramitacoes[
dic_tramitacoes['destino']= orgao_dest.nom_orgao 'unidade'] = comissao.nom_comissao
dic_tramitacoes[
'destino'] = orgao_dest.nom_orgao
else: else:
dic_tramitacoes['unidade'] = comissao.nom_comissao dic_tramitacoes['unidade'] = comissao.nom_comissao
dic_tramitacoes['destino'] = "None" dic_tramitacoes['destino'] = "None"
#---------------se unidade for orgao----------------------------------------- #---------------se unidade for orgao-------------------------------
if unidade.cod_comissao == None: if unidade.cod_comissao == None:
for orgao in context.zsql.orgao_obter_zsql(cod_orgao=unidade.cod_orgao): for orgao in context.zsql.orgao_obter_zsql(cod_orgao=unidade.cod_orgao):
if tramitacao.cod_unid_tram_dest != None: if tramitacao.cod_unid_tram_dest != None:
@ -181,13 +191,17 @@ for materia in context.zsql.materia_obter_zsql(cod_materia=REQUEST['cod_materia'
# se unidade destino for comissao # se unidade destino for comissao
if unidade_dest.cod_orgao == None: if unidade_dest.cod_orgao == None:
for comissao_dest in context.zsql.comissao_obter_zsql(cod_comissao=unidade_dest.cod_comissao): for comissao_dest in context.zsql.comissao_obter_zsql(cod_comissao=unidade_dest.cod_comissao):
dic_tramitacoes['unidade']= orgao.nom_orgao dic_tramitacoes[
dic_tramitacoes['destino']= comissao_dest.nom_comissao 'unidade'] = orgao.nom_orgao
dic_tramitacoes[
'destino'] = comissao_dest.nom_comissao
# se unidade destino for orgao # se unidade destino for orgao
if unidade_dest.cod_comissao == None: if unidade_dest.cod_comissao == None:
for orgao_dest in context.zsql.orgao_obter_zsql(cod_orgao=unidade_dest.cod_orgao): for orgao_dest in context.zsql.orgao_obter_zsql(cod_orgao=unidade_dest.cod_orgao):
dic_tramitacoes['unidade']= orgao.nom_orgao dic_tramitacoes[
dic_tramitacoes['destino']= orgao_dest.nom_orgao 'unidade'] = orgao.nom_orgao
dic_tramitacoes[
'destino'] = orgao_dest.nom_orgao
else: else:
dic_tramitacoes['unidade'] = orgao.nom_orgao dic_tramitacoes['unidade'] = orgao.nom_orgao
dic_tramitacoes['destino'] = "None" dic_tramitacoes['destino'] = "None"
@ -218,7 +232,8 @@ for materia in context.zsql.materia_obter_zsql(cod_materia=REQUEST['cod_materia'
dic_dados = {} dic_dados = {}
for numeracao in context.zsql.numeracao_obter_zsql(cod_materia=materia.cod_materia): for numeracao in context.zsql.numeracao_obter_zsql(cod_materia=materia.cod_materia):
for tipo_materia in context.zsql.tipo_materia_legislativa_obter_zsql(tip_materia=numeracao.tip_materia): for tipo_materia in context.zsql.tipo_materia_legislativa_obter_zsql(tip_materia=numeracao.tip_materia):
dic_dados['nome']= tipo_materia.sgl_tipo_materia+ "-"+ tipo_materia.des_tipo_materia+ ""+ numeracao.num_materia dic_dados['nome'] = tipo_materia.sgl_tipo_materia + "-" + \
tipo_materia.des_tipo_materia + "" + numeracao.num_materia
dic_dados['ano'] = numeracao.ano_materia dic_dados['ano'] = numeracao.ano_materia
lst_numeracoes.append(dic_dados) lst_numeracoes.append(dic_dados)
@ -228,8 +243,10 @@ for materia in context.zsql.materia_obter_zsql(cod_materia=REQUEST['cod_materia'
lst_legis_citadas = [] lst_legis_citadas = []
dic_dados = {} dic_dados = {}
for legislacao in context.zsql.legislacao_citada_obter_zsql(cod_materia=materia.cod_materia): for legislacao in context.zsql.legislacao_citada_obter_zsql(cod_materia=materia.cod_materia):
norma = context.zsql.norma_juridica_obter_zsql(cod_norma = legislacao.cod_norma_sel) norma = context.zsql.norma_juridica_obter_zsql(
dic_dados['nome_lei']= str(norma[0].tip_norma_sel) + ""+ str(norma[0].num_norma) + " de"+ str(norma[0].ano_norma) cod_norma=legislacao.cod_norma_sel)
dic_dados['nome_lei'] = str(norma[0].tip_norma_sel) + "" + \
str(norma[0].num_norma) + " de" + str(norma[0].ano_norma)
dic_dados['disposicao'] = legislacao.des_disposicoes dic_dados['disposicao'] = legislacao.des_disposicoes
dic_dados['parte'] = legislacao.des_parte dic_dados['parte'] = legislacao.des_parte
dic_dados['livro'] = legislacao.des_livro dic_dados['livro'] = legislacao.des_livro

31
sapl/relatorios/templates/pdf_documento_administrativo_preparar_pysc.py

@ -11,7 +11,8 @@ casa={}
aux = context.sapl_documentos.props_sapl.propertyItems() aux = context.sapl_documentos.props_sapl.propertyItems()
for item in aux: for item in aux:
casa[item[0]] = item[1] casa[item[0]] = item[1]
localidade=context.zsql.localidade_obter_zsql(cod_localidade=casa["cod_localidade"]) localidade = context.zsql.localidade_obter_zsql(
cod_localidade=casa["cod_localidade"])
if len(casa["num_cep"]) == 8: if len(casa["num_cep"]) == 8:
cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:] cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:]
else: else:
@ -23,7 +24,8 @@ if cep!="":
linha1 = linha1 + " - " linha1 = linha1 + " - "
linha1 = linha1 + "CEP " + cep linha1 = linha1 + "CEP " + cep
if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None: if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None:
linha1 = linha1 + " - "+localidade[0].nom_localidade+" "+localidade[0].sgl_uf linha1 = linha1 + " - " + \
localidade[0].nom_localidade + " " + localidade[0].sgl_uf
if casa["num_tel"] != "" and casa["num_tel"] != None: if casa["num_tel"] != "" and casa["num_tel"] != None:
linha1 = linha1 + " Tel.: " + casa["num_tel"] linha1 = linha1 + " Tel.: " + casa["num_tel"]
@ -57,14 +59,20 @@ else:
documentos = [] documentos = []
REQUEST = context.REQUEST REQUEST = context.REQUEST
for documento in context.zsql.documento_administrativo_pesquisar_zsql(tip_documento=REQUEST['lst_tip_documento'], for documento in context.zsql.documento_administrativo_pesquisar_zsql(tip_documento=REQUEST['lst_tip_documento'],
num_documento=REQUEST['txt_num_documento'], ano_documento=REQUEST['txt_ano_documento'], num_documento=REQUEST['txt_num_documento'], ano_documento=REQUEST[
num_protocolo=REQUEST['txt_num_protocolo'], ind_tramitacao=REQUEST['rad_tramitando'], 'txt_ano_documento'],
des_assunto=REQUEST['txa_txt_assunto'], cod_status=REQUEST['lst_status'], num_protocolo=REQUEST[
txt_interessado=REQUEST['txa_txt_interessado'], dat_apres1=REQUEST['dt_apres1'], 'txt_num_protocolo'], ind_tramitacao=REQUEST['rad_tramitando'],
des_assunto=REQUEST[
'txa_txt_assunto'], cod_status=REQUEST['lst_status'],
txt_interessado=REQUEST[
'txa_txt_interessado'], dat_apres1=REQUEST['dt_apres1'],
dat_apres2=REQUEST['dt_apres2'], rd_ordem=REQUEST['rd_ordenacao']): dat_apres2=REQUEST['dt_apres2'], rd_ordem=REQUEST['rd_ordenacao']):
dic = {} dic = {}
dic['titulo']=documento.sgl_tipo_documento+" "+str(documento.num_documento)+" "+str(documento.ano_documento)+" - "+documento.des_tipo_documento dic['titulo'] = documento.sgl_tipo_documento + " " + \
str(documento.num_documento) + " " + \
str(documento.ano_documento) + " - " + documento.des_tipo_documento
dic['txt_assunto'] = documento.txt_assunto dic['txt_assunto'] = documento.txt_assunto
dic['txt_interessado'] = documento.txt_interessado dic['txt_interessado'] = documento.txt_interessado
@ -103,7 +111,8 @@ filtro['assunto']=REQUEST.txa_txt_assunto
filtro['tipo_documento'] = '' filtro['tipo_documento'] = ''
if REQUEST.lst_tip_documento != '': if REQUEST.lst_tip_documento != '':
for tipo_documento in context.zsql.tipo_documento_administrativo_obter_zsql(ind_excluido=0, tip_documento=REQUEST.lst_tip_documento): for tipo_documento in context.zsql.tipo_documento_administrativo_obter_zsql(ind_excluido=0, tip_documento=REQUEST.lst_tip_documento):
filtro['tipo_documento']= tipo_documento.sgl_tipo_documento + ' - ' + tipo_documento.des_tipo_documento filtro['tipo_documento'] = tipo_documento.sgl_tipo_documento + \
' - ' + tipo_documento.des_tipo_documento
filtro['tramitando'] = '' filtro['tramitando'] = ''
if REQUEST.rad_tramitando == '1': if REQUEST.rad_tramitando == '1':
@ -114,10 +123,12 @@ elif REQUEST['rad_tramitando']=='0':
filtro['situacao_atual'] = '' filtro['situacao_atual'] = ''
if REQUEST.lst_status != '': if REQUEST.lst_status != '':
for status in context.zsql.status_tramitacao_administrativo_obter_zsql(ind_exluido=0, cod_status=REQUEST.lst_status): for status in context.zsql.status_tramitacao_administrativo_obter_zsql(ind_exluido=0, cod_status=REQUEST.lst_status):
filtro['situacao_atual']=status.sgl_status + ' - ' + status.des_status filtro['situacao_atual'] = status.sgl_status + \
' - ' + status.des_status
sessao = session.id sessao = session.id
caminho = context.pdf_documento_administrativo_gerar(sessao,imagem,data,documentos,cabecalho,rodape,filtro) caminho = context.pdf_documento_administrativo_gerar(
sessao, imagem, data, documentos, cabecalho, rodape, filtro)
if caminho == 'aviso': if caminho == 'aviso':
return response.redirect('mensagem_emitir_proc') return response.redirect('mensagem_emitir_proc')
else: else:

3
sapl/relatorios/templates/pdf_espelho_gerar.py

@ -163,4 +163,5 @@ def principal(imagem, lst_materias, dic_cabecalho, lst_rodape):
# return "/temp_folder/"+arquivoPdf # return "/temp_folder/"+arquivoPdf
# return principal(sessao,imagem,data,lst_materias,dic_cabecalho,lst_rodape,dic_filtro) # return
# principal(sessao,imagem,data,lst_materias,dic_cabecalho,lst_rodape,dic_filtro)

28
sapl/relatorios/templates/pdf_espelho_preparar_pysc.py

@ -11,7 +11,8 @@ casa={}
aux = context.sapl_documentos.props_sapl.propertyItems() aux = context.sapl_documentos.props_sapl.propertyItems()
for item in aux: for item in aux:
casa[item[0]] = item[1] casa[item[0]] = item[1]
localidade=context.zsql.localidade_obter_zsql(cod_localidade=casa["cod_localidade"]) localidade = context.zsql.localidade_obter_zsql(
cod_localidade=casa["cod_localidade"])
if len(casa["num_cep"]) == 8: if len(casa["num_cep"]) == 8:
cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:] cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:]
else: else:
@ -23,7 +24,8 @@ if cep!="":
linha1 = linha1 + " - " linha1 = linha1 + " - "
linha1 = linha1 + "CEP " + cep linha1 = linha1 + "CEP " + cep
if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None: if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None:
linha1 = linha1 + " - "+localidade[0].nom_localidade+" "+localidade[0].sgl_uf linha1 = linha1 + " - " + \
localidade[0].nom_localidade + " " + localidade[0].sgl_uf
if casa["num_tel"] != "" and casa["num_tel"] != None: if casa["num_tel"] != "" and casa["num_tel"] != None:
linha1 = linha1 + " Tel.: " + casa["num_tel"] linha1 = linha1 + " Tel.: " + casa["num_tel"]
@ -60,8 +62,10 @@ if REQUEST.txt_check=='1':
REQUEST = context.REQUEST REQUEST = context.REQUEST
for materia in context.zsql.materia_obter_zsql(cod_materia=cod_mat): for materia in context.zsql.materia_obter_zsql(cod_materia=cod_mat):
dic = {} dic = {}
dic['titulo']="INDICAÇÃO: "+str(materia.num_ident_basica)+" "+str(materia.ano_ident_basica) dic['titulo'] = "INDICAÇÃO: " + \
dic['materia']=str(materia.num_ident_basica)+"/"+str(materia.ano_ident_basica) str(materia.num_ident_basica) + " " + str(materia.ano_ident_basica)
dic['materia'] = str(materia.num_ident_basica) + \
"/" + str(materia.ano_ident_basica)
dic['dat_apresentacao'] = materia.dat_apresentacao dic['dat_apresentacao'] = materia.dat_apresentacao
dic['txt_ementa'] = materia.txt_ementa dic['txt_ementa'] = materia.txt_ementa
@ -106,7 +110,8 @@ if REQUEST.txt_check=='1':
dic['norma_juridica_vinculada'] = "Não há nenhuma norma jurídica vinculada" dic['norma_juridica_vinculada'] = "Não há nenhuma norma jurídica vinculada"
for norma in context.zsql.materia_buscar_norma_juridica_zsql(cod_materia=materia.cod_materia): for norma in context.zsql.materia_buscar_norma_juridica_zsql(cod_materia=materia.cod_materia):
dic['norma_juridica_vinculada']=norma.des_norma+" "+str(norma.num_norma)+"/"+str(norma.ano_norma) dic['norma_juridica_vinculada'] = norma.des_norma + " " + \
str(norma.num_norma) + "/" + str(norma.ano_norma)
materias.append(dic) materias.append(dic)
@ -117,8 +122,11 @@ else:
for cod_mat in codigo: for cod_mat in codigo:
for materia in context.zsql.materia_obter_zsql(cod_materia=cod_mat): for materia in context.zsql.materia_obter_zsql(cod_materia=cod_mat):
dic = {} dic = {}
dic['titulo']="INDICAÇÃO: "+str(materia.num_ident_basica)+" "+str(materia.ano_ident_basica) dic['titulo'] = "INDICAÇÃO: " + \
dic['materia']=str(materia.num_ident_basica)+"/"+str(materia.ano_ident_basica) str(materia.num_ident_basica) + " " + \
str(materia.ano_ident_basica)
dic['materia'] = str(materia.num_ident_basica) + \
"/" + str(materia.ano_ident_basica)
dic['dat_apresentacao'] = materia.dat_apresentacao dic['dat_apresentacao'] = materia.dat_apresentacao
dic['txt_ementa'] = materia.txt_ementa dic['txt_ementa'] = materia.txt_ementa
@ -163,7 +171,8 @@ else:
dic['norma_juridica_vinculada'] = "Não há nenhuma norma jurídica vinculada" dic['norma_juridica_vinculada'] = "Não há nenhuma norma jurídica vinculada"
for norma in context.zsql.materia_buscar_norma_juridica_zsql(cod_materia=materia.cod_materia): for norma in context.zsql.materia_buscar_norma_juridica_zsql(cod_materia=materia.cod_materia):
dic['norma_juridica_vinculada']=norma.des_norma+" "+str(norma.num_norma)+"/"+str(norma.ano_norma) dic['norma_juridica_vinculada'] = norma.des_norma + " " + \
str(norma.num_norma) + "/" + str(norma.ano_norma)
materias.append(dic) materias.append(dic)
@ -193,7 +202,8 @@ filtro={} # Dicionário que conterá os dados do filtro
# filtro['situacao_atual']=status.sgl_status + ' - ' + status.des_status # filtro['situacao_atual']=status.sgl_status + ' - ' + status.des_status
sessao = session.id sessao = session.id
caminho = context.pdf_espelho_gerar(sessao,imagem,data,materias,cabecalho,rodape,filtro) caminho = context.pdf_espelho_gerar(
sessao, imagem, data, materias, cabecalho, rodape, filtro)
if caminho == 'aviso': if caminho == 'aviso':
return response.redirect('mensagem_emitir_proc') return response.redirect('mensagem_emitir_proc')
else: else:

30
sapl/relatorios/templates/pdf_etiqueta_protocolo_preparar_pysc.py

@ -11,7 +11,8 @@ casa={}
aux = context.sapl_documentos.props_sapl.propertyItems() aux = context.sapl_documentos.props_sapl.propertyItems()
for item in aux: for item in aux:
casa[item[0]] = item[1] casa[item[0]] = item[1]
localidade=context.zsql.localidade_obter_zsql(cod_localidade=casa["cod_localidade"]) localidade = context.zsql.localidade_obter_zsql(
cod_localidade=casa["cod_localidade"])
if len(casa["num_cep"]) == 8: if len(casa["num_cep"]) == 8:
cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:] cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:]
else: else:
@ -23,7 +24,8 @@ if cep!="":
linha1 = linha1 + " - " linha1 = linha1 + " - "
linha1 = linha1 + "CEP " + cep linha1 = linha1 + "CEP " + cep
if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None: if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None:
linha1 = linha1 + " - "+localidade[0].nom_localidade+" "+localidade[0].sgl_uf linha1 = linha1 + " - " + \
localidade[0].nom_localidade + " " + localidade[0].sgl_uf
if casa["num_tel"] != "" and casa["num_tel"] != None: if casa["num_tel"] != "" and casa["num_tel"] != None:
linha1 = linha1 + " Tel.: " + casa["num_tel"] linha1 = linha1 + " Tel.: " + casa["num_tel"]
@ -57,16 +59,21 @@ else:
protocolos = [] protocolos = []
REQUEST = context.REQUEST REQUEST = context.REQUEST
for protocolo in context.zsql.protocolo_pesquisar_zsql(tip_protocolo=REQUEST['rad_tip_protocolo'], for protocolo in context.zsql.protocolo_pesquisar_zsql(tip_protocolo=REQUEST['rad_tip_protocolo'],
cod_protocolo=REQUEST['txt_num_protocolo'], ano_protocolo=REQUEST['txt_ano_protocolo'], cod_protocolo=REQUEST['txt_num_protocolo'], ano_protocolo=REQUEST[
tip_documento=REQUEST['lst_tip_documento'], tip_processo=REQUEST['rad_tip_processo'], 'txt_ano_protocolo'],
tip_materia=REQUEST['lst_tip_materia'], des_assunto=REQUEST['txt_assunto'], tip_documento=REQUEST['lst_tip_documento'], tip_processo=REQUEST[
cod_autor=REQUEST['hdn_cod_autor'], des_interessado=REQUEST['txa_txt_interessado'], 'rad_tip_processo'],
tip_materia=REQUEST[
'lst_tip_materia'], des_assunto=REQUEST['txt_assunto'],
cod_autor=REQUEST['hdn_cod_autor'], des_interessado=REQUEST[
'txa_txt_interessado'],
dat_apres=REQUEST['dt_apres'], dat_apres2=REQUEST['dt_apres2']): dat_apres=REQUEST['dt_apres'], dat_apres2=REQUEST['dt_apres2']):
dic = {} dic = {}
dic['titulo'] = str(protocolo.cod_protocolo) dic['titulo'] = str(protocolo.cod_protocolo)
dic['data']='Data: '+context.pysc.iso_to_port_pysc(protocolo.dat_protocolo)+' Horário: '+protocolo.hor_protocolo[0:2]+':'+protocolo.hor_protocolo[3:5] dic['data'] = 'Data: ' + context.pysc.iso_to_port_pysc(
protocolo.dat_protocolo) + ' Horário: ' + protocolo.hor_protocolo[0:2] + ':' + protocolo.hor_protocolo[3:5]
dic['txt_assunto'] = protocolo.txt_assunto_ementa dic['txt_assunto'] = protocolo.txt_assunto_ementa
@ -92,11 +99,13 @@ for protocolo in context.zsql.protocolo_pesquisar_zsql(tip_protocolo=REQUEST['ra
dic['num_materia'] = '' dic['num_materia'] = ''
for materia in context.zsql.materia_obter_zsql(num_protocolo=protocolo.cod_protocolo, ano_ident_basica=protocolo.ano_protocolo): for materia in context.zsql.materia_obter_zsql(num_protocolo=protocolo.cod_protocolo, ano_ident_basica=protocolo.ano_protocolo):
dic['num_materia']=materia.sgl_tipo_materia+' '+str(materia.num_ident_basica)+'/'+str(materia.ano_ident_basica) dic['num_materia'] = materia.sgl_tipo_materia + ' ' + \
str(materia.num_ident_basica) + '/' + str(materia.ano_ident_basica)
dic['num_documento'] = '' dic['num_documento'] = ''
for documento in context.zsql.documento_administrativo_obter_zsql(num_protocolo=protocolo.cod_protocolo): for documento in context.zsql.documento_administrativo_obter_zsql(num_protocolo=protocolo.cod_protocolo):
dic['num_documento']=documento.sgl_tipo_documento+' '+str(documento.num_documento)+'/'+ str(documento.ano_documento) dic['num_documento'] = documento.sgl_tipo_documento + ' ' + \
str(documento.num_documento) + '/' + str(documento.ano_documento)
dic['ident_processo'] = dic['num_materia'] or dic['num_documento'] dic['ident_processo'] = dic['num_materia'] or dic['num_documento']
@ -120,7 +129,8 @@ filtro['autor']=REQUEST.hdn_cod_autor
filtro['interessado'] = REQUEST.txa_txt_interessado filtro['interessado'] = REQUEST.txa_txt_interessado
sessao = session.id sessao = session.id
caminho = context.pdf_etiqueta_protocolo_gerar(sessao,imagem,data,protocolos,cabecalho,rodape,filtro) caminho = context.pdf_etiqueta_protocolo_gerar(
sessao, imagem, data, protocolos, cabecalho, rodape, filtro)
if caminho == 'aviso': if caminho == 'aviso':
return response.redirect('mensagem_emitir_proc') return response.redirect('mensagem_emitir_proc')
else: else:

32
sapl/relatorios/templates/pdf_materia_gerar.py

@ -1,4 +1,4 @@
##parameters=sessao,imagem,data,lst_materias,dic_cabecalho,lst_rodape,dic_filtro # parameters=sessao,imagem,data,lst_materias,dic_cabecalho,lst_rodape,dic_filtro
"""relatorio_materia.py """relatorio_materia.py
External method para gerar o arquivo rml do resultado de uma pesquisa de matérias External method para gerar o arquivo rml do resultado de uma pesquisa de matérias
@ -17,27 +17,34 @@ def cabecalho(dic_cabecalho,imagem):
tmp_data += '\t\t\t\t<image x="2.1cm" y="25.7cm" width="59" height="62" file="' + imagem + '"/>\n' tmp_data += '\t\t\t\t<image x="2.1cm" y="25.7cm" width="59" height="62" file="' + imagem + '"/>\n'
tmp_data += '\t\t\t\t<lines>2cm 25cm 19cm 25cm</lines>\n' tmp_data += '\t\t\t\t<lines>2cm 25cm 19cm 25cm</lines>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="16"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="16"/>\n'
tmp_data+='\t\t\t\t<drawString x="5cm" y="27.1cm">' + dic_cabecalho['nom_casa'] + '</drawString>\n' tmp_data += '\t\t\t\t<drawString x="5cm" y="27.1cm">' + \
dic_cabecalho['nom_casa'] + '</drawString>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica" size="13"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica" size="13"/>\n'
tmp_data+='\t\t\t\t<drawString x="5cm" y="26.5cm">' + dic_cabecalho['nom_estado'] + '</drawString>\n' tmp_data += '\t\t\t\t<drawString x="5cm" y="26.5cm">' + \
dic_cabecalho['nom_estado'] + '</drawString>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="13"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="13"/>\n'
tmp_data += '\t\t\t\t<drawCentredString x="10.5cm" y="25.2cm">Relatório de Matérias Legislativas</drawCentredString>\n' tmp_data += '\t\t\t\t<drawCentredString x="10.5cm" y="25.2cm">Relatório de Matérias Legislativas</drawCentredString>\n'
return tmp_data return tmp_data
def rodape(lst_rodape): def rodape(lst_rodape):
"""Gera o codigo rml do rodape""" """Gera o codigo rml do rodape"""
tmp_data = '' tmp_data = ''
tmp_data += '\t\t\t\t<lines>2cm 3.2cm 19cm 3.2cm</lines>\n' tmp_data += '\t\t\t\t<lines>2cm 3.2cm 19cm 3.2cm</lines>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica" size="8"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica" size="8"/>\n'
tmp_data+='\t\t\t\t<drawString x="2cm" y="3.3cm">' + lst_rodape[2] + '</drawString>\n' tmp_data += '\t\t\t\t<drawString x="2cm" y="3.3cm">' + \
lst_rodape[2] + '</drawString>\n'
tmp_data += '\t\t\t\t<drawString x="17.9cm" y="3.3cm">Página <pageNumber/></drawString>\n' tmp_data += '\t\t\t\t<drawString x="17.9cm" y="3.3cm">Página <pageNumber/></drawString>\n'
tmp_data+='\t\t\t\t<drawCentredString x="10.5cm" y="2.7cm">' + lst_rodape[0] + '</drawCentredString>\n' tmp_data += '\t\t\t\t<drawCentredString x="10.5cm" y="2.7cm">' + \
tmp_data+='\t\t\t\t<drawCentredString x="10.5cm" y="2.3cm">' + lst_rodape[1] + '</drawCentredString>\n' lst_rodape[0] + '</drawCentredString>\n'
tmp_data += '\t\t\t\t<drawCentredString x="10.5cm" y="2.3cm">' + \
lst_rodape[1] + '</drawCentredString>\n'
return tmp_data return tmp_data
def paraStyle(): def paraStyle():
"""Gera o codigo rml que define o estilo dos paragrafos""" """Gera o codigo rml que define o estilo dos paragrafos"""
@ -56,6 +63,7 @@ def paraStyle():
return tmp_data return tmp_data
def materias(lst_materias): def materias(lst_materias):
"""Gera o codigo rml do conteudo da pesquisa de materias""" """Gera o codigo rml do conteudo da pesquisa de materias"""
@ -74,15 +82,20 @@ def materias(lst_materias):
tmp_data += '\t\t<condPageBreak height="1.5cm"/>\n' tmp_data += '\t\t<condPageBreak height="1.5cm"/>\n'
# materias # materias
tmp_data+='\t\t<para style="P1"> <b>'+ dic['titulo'] +'</b> - <b>Autor: </b>' + dic['nom_autor'] + ' </para>\n' tmp_data += '\t\t<para style="P1"> <b>' + \
dic['titulo'] + '</b> - <b>Autor: </b>' + \
dic['nom_autor'] + ' </para>\n'
if dic['txt_ementa'] != None: if dic['txt_ementa'] != None:
txt_ementa = dic['txt_ementa'].replace('&', '&amp;') txt_ementa = dic['txt_ementa'].replace('&', '&amp;')
tmp_data += '\t\t<para style="P2"> ' + txt_ementa + ' </para>\n' tmp_data += '\t\t<para style="P2"> ' + txt_ementa + ' </para>\n'
tmp_data+='\t\t<para style="P2"><b>Situação:</b> ' + dic['des_situacao'] + '/ <b>Norma Jurídica Vinculada:</b> ' + dic['norma_vinculada'] + '</para>\n' tmp_data += '\t\t<para style="P2"><b>Situação:</b> ' + \
dic['des_situacao'] + '/ <b>Norma Jurídica Vinculada:</b> ' + \
dic['norma_vinculada'] + '</para>\n'
tmp_data += '\t</story>\n' tmp_data += '\t</story>\n'
return tmp_data return tmp_data
def principal(imagem, lst_materias, dic_cabecalho, lst_rodape): def principal(imagem, lst_materias, dic_cabecalho, lst_rodape):
"""Funcao pricipal que gera a estrutura global do arquivo rml""" """Funcao pricipal que gera a estrutura global do arquivo rml"""
@ -116,4 +129,5 @@ def principal(imagem, lst_materias, dic_cabecalho, lst_rodape):
# return "/temp_folder/"+arquivoPdf # return "/temp_folder/"+arquivoPdf
# return principal(sessao,imagem,data,lst_materias,dic_cabecalho,lst_rodape,dic_filtro) # return
# principal(sessao,imagem,data,lst_materias,dic_cabecalho,lst_rodape,dic_filtro)

42
sapl/relatorios/templates/pdf_materia_preparar_pysc.py

@ -13,7 +13,8 @@ casa={}
aux = context.sapl_documentos.props_sapl.propertyItems() aux = context.sapl_documentos.props_sapl.propertyItems()
for item in aux: for item in aux:
casa[item[0]] = item[1] casa[item[0]] = item[1]
localidade=context.zsql.localidade_obter_zsql(cod_localidade=casa["cod_localidade"]) localidade = context.zsql.localidade_obter_zsql(
cod_localidade=casa["cod_localidade"])
if len(casa["num_cep"]) == 8: if len(casa["num_cep"]) == 8:
cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:] cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:]
else: else:
@ -25,7 +26,8 @@ if cep!="":
linha1 = linha1 + " - " linha1 = linha1 + " - "
linha1 = linha1 + "CEP " + cep linha1 = linha1 + "CEP " + cep
if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None: if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None:
linha1 = linha1 + " - "+localidade[0].nom_localidade+" "+localidade[0].sgl_uf linha1 = linha1 + " - " + \
localidade[0].nom_localidade + " " + localidade[0].sgl_uf
if casa["num_tel"] != "" and casa["num_tel"] != None: if casa["num_tel"] != "" and casa["num_tel"] != None:
linha1 = linha1 + " Tel.: " + casa["num_tel"] linha1 = linha1 + " Tel.: " + casa["num_tel"]
@ -59,17 +61,25 @@ else:
materias = [] materias = []
REQUEST = context.REQUEST REQUEST = context.REQUEST
for materia in context.zsql.materia_pesquisar_zsql(tip_id_basica=REQUEST['lst_tip_materia'], num_ident_basica=REQUEST['txt_numero'], for materia in context.zsql.materia_pesquisar_zsql(tip_id_basica=REQUEST['lst_tip_materia'], num_ident_basica=REQUEST['txt_numero'],
ano_ident_basica=REQUEST['txt_ano'], ind_tramitacao=REQUEST['rad_tramitando'], ano_ident_basica=REQUEST[
des_assunto=REQUEST['txt_assunto'], nom_relator=REQUEST['txt_relator'], 'txt_ano'], ind_tramitacao=REQUEST['rad_tramitando'],
cod_status=REQUEST['lst_status'], des_tipo_autor=REQUEST['lst_tip_autor'], des_assunto=REQUEST[
dat_apresentacao=REQUEST['dt_apres'], dat_apresentacao2=REQUEST['dt_apres2'], 'txt_assunto'], nom_relator=REQUEST['txt_relator'],
dat_publicacao=REQUEST['dt_public'], dat_publicacao2=REQUEST['dt_public2'], cod_status=REQUEST['lst_status'], des_tipo_autor=REQUEST[
cod_partido=REQUEST['lst_cod_partido'],cod_autor=REQUEST['hdn_cod_autor'], 'lst_tip_autor'],
dat_apresentacao=REQUEST[
'dt_apres'], dat_apresentacao2=REQUEST['dt_apres2'],
dat_publicacao=REQUEST[
'dt_public'], dat_publicacao2=REQUEST['dt_public2'],
cod_partido=REQUEST['lst_cod_partido'], cod_autor=REQUEST[
'hdn_cod_autor'],
rd_ordem=REQUEST['rd_ordenacao'], rd_ordem_td=REQUEST['rd_ordem_td']): rd_ordem=REQUEST['rd_ordenacao'], rd_ordem_td=REQUEST['rd_ordem_td']):
dic = {} dic = {}
dic['titulo']=materia.sgl_tipo_materia+" "+materia.des_tipo_materia+" "+str(materia.num_ident_basica)+"/"+str(materia.ano_ident_basica) dic['titulo'] = materia.sgl_tipo_materia + " " + materia.des_tipo_materia + \
" " + str(materia.num_ident_basica) + "/" + \
str(materia.ano_ident_basica)
dic['txt_ementa'] = materia.txt_ementa dic['txt_ementa'] = materia.txt_ementa
dic['nom_autor'] = " " dic['nom_autor'] = " "
for autoria in context.zsql.autoria_obter_zsql(cod_materia=materia.cod_materia): for autoria in context.zsql.autoria_obter_zsql(cod_materia=materia.cod_materia):
@ -105,10 +115,11 @@ for materia in context.zsql.materia_pesquisar_zsql(tip_id_basica=REQUEST['lst_ti
dic['des_situacao'] = des_status dic['des_situacao'] = des_status
dic['ultima_acao'] = txt_tramitacao dic['ultima_acao'] = txt_tramitacao
dic['norma_vinculada'] = " " dic['norma_vinculada'] = " "
for norma_vinculada in context.zsql.materia_buscar_norma_juridica_zsql(cod_materia=materia.cod_materia): for norma_vinculada in context.zsql.materia_buscar_norma_juridica_zsql(cod_materia=materia.cod_materia):
dic['norma_vinculada']=norma_vinculada.des_norma+" "+str(norma_vinculada.num_norma)+"/"+str(norma_vinculada.ano_norma) dic['norma_vinculada'] = norma_vinculada.des_norma + " " + \
str(norma_vinculada.num_norma) + "/" + \
str(norma_vinculada.ano_norma)
materias.append(dic) materias.append(dic)
@ -129,7 +140,8 @@ if REQUEST.hdn_txt_autor==' ': # Corrige bug do Netscape
filtro['tipo_materia'] = '' filtro['tipo_materia'] = ''
if REQUEST.lst_tip_materia != '': if REQUEST.lst_tip_materia != '':
for tipo_materia in context.zsql.tipo_materia_legislativa_obter_zsql(ind_excluido=0, tip_materia=REQUEST.lst_tip_materia): for tipo_materia in context.zsql.tipo_materia_legislativa_obter_zsql(ind_excluido=0, tip_materia=REQUEST.lst_tip_materia):
filtro['tipo_materia']= tipo_materia.sgl_tipo_materia + ' - ' + tipo_materia.des_tipo_materia filtro['tipo_materia'] = tipo_materia.sgl_tipo_materia + \
' - ' + tipo_materia.des_tipo_materia
filtro['partido'] = '' filtro['partido'] = ''
if REQUEST.lst_cod_partido != '': if REQUEST.lst_cod_partido != '':
@ -145,10 +157,12 @@ elif REQUEST['rad_tramitando']=='0':
filtro['situacao_atual'] = '' filtro['situacao_atual'] = ''
if REQUEST.lst_status != '': if REQUEST.lst_status != '':
for status in context.zsql.status_tramitacao_obter_zsql(ind_excluido=0, cod_status=REQUEST.lst_status): for status in context.zsql.status_tramitacao_obter_zsql(ind_excluido=0, cod_status=REQUEST.lst_status):
filtro['situacao_atual']=status.sgl_status + ' - ' + status.des_status filtro['situacao_atual'] = status.sgl_status + \
' - ' + status.des_status
sessao = session.id sessao = session.id
caminho = context.pdf_materia_gerar(sessao,imagem,data,materias,cabecalho,rodape,filtro) caminho = context.pdf_materia_gerar(
sessao, imagem, data, materias, cabecalho, rodape, filtro)
if caminho == 'aviso': if caminho == 'aviso':
return response.redirect('mensagem_emitir_proc') return response.redirect('mensagem_emitir_proc')
else: else:

24
sapl/relatorios/templates/pdf_norma_gerar.py

@ -1,4 +1,4 @@
##parameters=sessao,imagem,data,lst_normas,dic_cabecalho,lst_rodape,dic_filtro # parameters=sessao,imagem,data,lst_normas,dic_cabecalho,lst_rodape,dic_filtro
"""relatorio_norma.py """relatorio_norma.py
External method para gerar o arquivo rml do resultado de uma pesquisa de normas External method para gerar o arquivo rml do resultado de uma pesquisa de normas
@ -17,7 +17,8 @@ def cabecalho(inf_basicas_dic,imagem):
tmp_data += '\t\t\t\t<image x="2.1cm" y="25.7cm" width="59" height="62" file="' + imagem + '"/>\n' tmp_data += '\t\t\t\t<image x="2.1cm" y="25.7cm" width="59" height="62" file="' + imagem + '"/>\n'
tmp_data += '\t\t\t\t<lines>2cm 25.4cm 19cm 25.4cm</lines>\n' tmp_data += '\t\t\t\t<lines>2cm 25.4cm 19cm 25.4cm</lines>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="15"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="15"/>\n'
tmp_data+='\t\t\t\t<drawString x="5cm" y="27.2cm">' + dic_cabecalho['nom_casa'] + '</drawString>\n' tmp_data += '\t\t\t\t<drawString x="5cm" y="27.2cm">' + \
dic_cabecalho['nom_casa'] + '</drawString>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica" size="12"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica" size="12"/>\n'
tmp_data += '\t\t\t\t<drawString x="5cm" y="26.6cm">Sistema de Apoio ao Processo Legislativo</drawString>\n' tmp_data += '\t\t\t\t<drawString x="5cm" y="26.6cm">Sistema de Apoio ao Processo Legislativo</drawString>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="13"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="13"/>\n'
@ -25,19 +26,24 @@ def cabecalho(inf_basicas_dic,imagem):
return tmp_data return tmp_data
def rodape(lst_rodape): def rodape(lst_rodape):
"""Gera o codigo rml do rodape""" """Gera o codigo rml do rodape"""
tmp_data = '' tmp_data = ''
tmp_data += '\t\t\t\t<lines>2cm 3.2cm 19cm 3.2cm</lines>\n' tmp_data += '\t\t\t\t<lines>2cm 3.2cm 19cm 3.2cm</lines>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica" size="8"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica" size="8"/>\n'
tmp_data+='\t\t\t\t<drawString x="2cm" y="3.3cm">' + lst_rodape[2] + '</drawString>\n' tmp_data += '\t\t\t\t<drawString x="2cm" y="3.3cm">' + \
lst_rodape[2] + '</drawString>\n'
tmp_data += '\t\t\t\t<drawString x="17.9cm" y="3.3cm">Página <pageNumber/></drawString>\n' tmp_data += '\t\t\t\t<drawString x="17.9cm" y="3.3cm">Página <pageNumber/></drawString>\n'
tmp_data+='\t\t\t\t<drawCentredString x="10.5cm" y="2.7cm">' + lst_rodape[0] + '</drawCentredString>\n' tmp_data += '\t\t\t\t<drawCentredString x="10.5cm" y="2.7cm">' + \
tmp_data+='\t\t\t\t<drawCentredString x="10.5cm" y="2.3cm">' + lst_rodape[1] + '</drawCentredString>\n' lst_rodape[0] + '</drawCentredString>\n'
tmp_data += '\t\t\t\t<drawCentredString x="10.5cm" y="2.3cm">' + \
lst_rodape[1] + '</drawCentredString>\n'
return tmp_data return tmp_data
def paraStyle(): def paraStyle():
"""Gera o codigo rml que define o estilo dos paragrafos""" """Gera o codigo rml que define o estilo dos paragrafos"""
@ -56,6 +62,7 @@ def paraStyle():
return tmp_data return tmp_data
def normas(lst_normas): def normas(lst_normas):
"""Gera o codigo rml do conteudo da pesquisa de normas""" """Gera o codigo rml do conteudo da pesquisa de normas"""
@ -86,11 +93,13 @@ def normas(lst_normas):
txt_ementa = dic['txt_ementa'].replace('&', '&amp;') txt_ementa = dic['txt_ementa'].replace('&', '&amp;')
tmp_data += '\t\t<para style="P2">' + txt_ementa + '</para>\n' tmp_data += '\t\t<para style="P2">' + txt_ementa + '</para>\n'
if dic['materia_vinculada'] != None: if dic['materia_vinculada'] != None:
tmp_data+='\t\t<para style="P2"><b>Matéria Legislativa:</b> ' + dic['materia_vinculada'] + '</para>\n' tmp_data += '\t\t<para style="P2"><b>Matéria Legislativa:</b> ' + \
dic['materia_vinculada'] + '</para>\n'
tmp_data += '\t</story>\n' tmp_data += '\t</story>\n'
return tmp_data return tmp_data
def principal(imagem, lst_normas, dic_cabecalho, lst_rodape): def principal(imagem, lst_normas, dic_cabecalho, lst_rodape):
"""Funcao pricipal que gera a estrutura global do arquivo rml""" """Funcao pricipal que gera a estrutura global do arquivo rml"""
@ -124,4 +133,5 @@ def principal(imagem, lst_normas, dic_cabecalho, lst_rodape):
# return "/temp_folder/"+arquivoPdf # return "/temp_folder/"+arquivoPdf
# return principal(sessao,imagem,data,lst_normas,dic_cabecalho,lst_rodape,dic_filtro) # return
# principal(sessao,imagem,data,lst_normas,dic_cabecalho,lst_rodape,dic_filtro)

29
sapl/relatorios/templates/pdf_norma_preparar_pysc.py

@ -11,7 +11,8 @@ casa={}
aux = context.sapl_documentos.props_sapl.propertyItems() aux = context.sapl_documentos.props_sapl.propertyItems()
for item in aux: for item in aux:
casa[item[0]] = item[1] casa[item[0]] = item[1]
localidade=context.zsql.localidade_obter_zsql(cod_localidade=casa["cod_localidade"]) localidade = context.zsql.localidade_obter_zsql(
cod_localidade=casa["cod_localidade"])
if len(casa["num_cep"]) == 8: if len(casa["num_cep"]) == 8:
cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:] cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:]
else: else:
@ -23,7 +24,8 @@ if cep!="":
linha1 = linha1 + " - " linha1 = linha1 + " - "
linha1 = linha1 + "CEP " + cep linha1 = linha1 + "CEP " + cep
if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None: if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None:
linha1 = linha1 + " - "+localidade[0].nom_localidade+" "+localidade[0].sgl_uf linha1 = linha1 + " - " + \
localidade[0].nom_localidade + " " + localidade[0].sgl_uf
if casa["num_tel"] != "" and casa["num_tel"] != None: if casa["num_tel"] != "" and casa["num_tel"] != None:
linha1 = linha1 + " Tel.: " + casa["num_tel"] linha1 = linha1 + " Tel.: " + casa["num_tel"]
@ -57,20 +59,27 @@ else:
normas = [] normas = []
REQUEST = context.REQUEST REQUEST = context.REQUEST
for norma in context.zsql.norma_juridica_obter_zsql(tip_norma=REQUEST['lst_tip_norma'], num_norma=REQUEST['txt_numero'], for norma in context.zsql.norma_juridica_obter_zsql(tip_norma=REQUEST['lst_tip_norma'], num_norma=REQUEST['txt_numero'],
ano_norma=REQUEST['txt_ano'], des_assunto=REQUEST['txt_assunto'], ano_norma=REQUEST['txt_ano'], des_assunto=REQUEST[
cod_assunto=REQUEST['lst_assunto_norma'], dat_norma=REQUEST['dt_norma'], 'txt_assunto'],
dat_norma2=REQUEST['dt_norma2'], dat_publicacao=REQUEST['dt_public'], cod_assunto=REQUEST[
'lst_assunto_norma'], dat_norma=REQUEST['dt_norma'],
dat_norma2=REQUEST[
'dt_norma2'], dat_publicacao=REQUEST['dt_public'],
dat_publicacao2=REQUEST['dt_public2'], rd_ordem=REQUEST['rd_ordenacao']): dat_publicacao2=REQUEST['dt_public2'], rd_ordem=REQUEST['rd_ordenacao']):
dic = {} dic = {}
dic['titulo']=norma.sgl_tipo_norma+""+str(norma.num_norma)+" de "+str(norma.dat_norma)+" - "+norma.des_tipo_norma dic['titulo'] = norma.sgl_tipo_norma + "" + \
str(norma.num_norma) + " de " + \
str(norma.dat_norma) + " - " + norma.des_tipo_norma
dic['txt_ementa'] = norma.txt_ementa dic['txt_ementa'] = norma.txt_ementa
dic['materia_vinculada'] = " " dic['materia_vinculada'] = " "
if norma.cod_materia != None: if norma.cod_materia != None:
for materia_vinculada in context.zsql.materia_obter_zsql(cod_materia=str(norma.cod_materia)): for materia_vinculada in context.zsql.materia_obter_zsql(cod_materia=str(norma.cod_materia)):
dic['materia_vinculada']=materia_vinculada.sgl_tipo_materia+" "+str(materia_vinculada.num_ident_basica)+"/"+str(materia_vinculada.ano_ident_basica) dic['materia_vinculada'] = materia_vinculada.sgl_tipo_materia + " " + \
str(materia_vinculada.num_ident_basica) + "/" + \
str(materia_vinculada.ano_ident_basica)
normas.append(dic) normas.append(dic)
@ -84,10 +93,12 @@ filtro['assunto']=REQUEST.txt_assunto
filtro['tipo_norma'] = '' filtro['tipo_norma'] = ''
if REQUEST.lst_tip_norma != '': if REQUEST.lst_tip_norma != '':
for tipo_norma in context.zsql.tipo_norma_juridica_obter_zsql(ind_excluido=0, tip_norma=REQUEST.lst_tip_norma): for tipo_norma in context.zsql.tipo_norma_juridica_obter_zsql(ind_excluido=0, tip_norma=REQUEST.lst_tip_norma):
filtro['tipo_norma']= tipo_norma.sgl_tipo_norma + ' - ' + tipo_norma.des_tipo_norma filtro['tipo_norma'] = tipo_norma.sgl_tipo_norma + \
' - ' + tipo_norma.des_tipo_norma
sessao = session.id sessao = session.id
caminho = context.pdf_norma_gerar(sessao,imagem,data,normas,cabecalho,rodape,filtro) caminho = context.pdf_norma_gerar(
sessao, imagem, data, normas, cabecalho, rodape, filtro)
if caminho == 'aviso': if caminho == 'aviso':
return response.redirect('mensagem_emitir_proc') return response.redirect('mensagem_emitir_proc')
else: else:

66
sapl/relatorios/templates/pdf_ordem_dia_preparar_pysc.py

@ -9,35 +9,46 @@ if context.REQUEST['cod_sessao_plen']!='':
pauta = [] # lista contendo a pauta da ordem do dia a ser impressa pauta = [] # lista contendo a pauta da ordem do dia a ser impressa
data = "" data = ""
for dat_sessao in context.zsql.sessao_plenaria_obter_zsql(cod_sessao_plen=cod_sessao_plen, ind_excluido=0): for dat_sessao in context.zsql.sessao_plenaria_obter_zsql(cod_sessao_plen=cod_sessao_plen, ind_excluido=0):
data = context.pysc.data_converter_pysc(dat_sessao.dat_inicio_sessao) # converte data para formato yyyy/mm/dd # converte data para formato yyyy/mm/dd
dat_ordem = context.pysc.data_converter_pysc(dat_sessao.dat_inicio_sessao) # converte data para formato yyyy/mm/dd data = context.pysc.data_converter_pysc(dat_sessao.dat_inicio_sessao)
dat_ordem = context.pysc.data_converter_pysc(
dat_sessao.dat_inicio_sessao) # converte data para formato yyyy/mm/dd
# seleciona dados da sessão plenária # seleciona dados da sessão plenária
for sp in context.zsql.sessao_plenaria_obter_zsql(dat_inicio_sessao=data, ind_excluido=0): for sp in context.zsql.sessao_plenaria_obter_zsql(dat_inicio_sessao=data, ind_excluido=0):
dicsp = {} # dicionário que armazenará os dados a serem impressos de uma sessão plenária dicsp = {} # dicionário que armazenará os dados a serem impressos de uma sessão plenária
ts = context.zsql.tipo_sessao_plenaria_obter_zsql(tip_sessao=sp.tip_sessao)[0] ts = context.zsql.tipo_sessao_plenaria_obter_zsql(
dicsp["sessao"] = str(sp.num_sessao_plen)+"ª Sessao "+ts.nom_sessao+" da "+str(sp.num_sessao_leg)+"ª Sessao Legislativa da "+str(sp.num_legislatura)+"ª Legislatura" tip_sessao=sp.tip_sessao)[0]
dia = context.pysc.data_converter_por_extenso_pysc(data=sp.dat_inicio_sessao) dicsp["sessao"] = str(sp.num_sessao_plen) + "ª Sessao " + ts.nom_sessao + " da " + str(
sp.num_sessao_leg) + "ª Sessao Legislativa da " + str(sp.num_legislatura) + "ª Legislatura"
dia = context.pysc.data_converter_por_extenso_pysc(
data=sp.dat_inicio_sessao)
hora = context.pysc.hora_formatar_pysc(hora=sp.hr_inicio_sessao) hora = context.pysc.hora_formatar_pysc(hora=sp.hr_inicio_sessao)
dicsp["datasessao"] = "Dia "+str(dia)+" ("+str(sp.dia_sessao)+") - Inicio as "+hora dicsp["datasessao"] = "Dia " + \
str(dia) + " (" + str(sp.dia_sessao) + ") - Inicio as " + hora
splen.append(dicsp) splen.append(dicsp)
# seleciona as matérias que compõem a pauta na data escolhida # seleciona as matérias que compõem a pauta na data escolhida
for ordem in context.zsql.ordem_dia_obter_zsql(dat_ordem=data, ind_excluido=0): for ordem in context.zsql.ordem_dia_obter_zsql(dat_ordem=data, ind_excluido=0):
# seleciona os detalhes de uma matéria # seleciona os detalhes de uma matéria
materia = context.zsql.materia_obter_zsql(cod_materia=ordem.cod_materia)[0] materia = context.zsql.materia_obter_zsql(
cod_materia=ordem.cod_materia)[0]
dic = {} # dicionário que armazenará os dados a serem impressos de uma matéria dic = {} # dicionário que armazenará os dados a serem impressos de uma matéria
dic["num_ordem"] = ordem.num_ordem dic["num_ordem"] = ordem.num_ordem
dic["id_materia"] = materia.des_tipo_materia+" - Nº "+str(materia.num_ident_basica)+"/"+str(materia.ano_ident_basica) dic["id_materia"] = materia.des_tipo_materia + " - Nº " + \
str(materia.num_ident_basica) + "/" + str(materia.ano_ident_basica)
# dic["id_materia"] = materia.sgl_tipo_materia+" - "+str(materia.num_ident_basica)+"/"+str(materia.ano_ident_basica)+" - "+materia.des_tipo_materia # dic["id_materia"] = materia.sgl_tipo_materia+" - "+str(materia.num_ident_basica)+"/"+str(materia.ano_ident_basica)+" - "+materia.des_tipo_materia
dic["txt_ementa"] = ordem.txt_observacao dic["txt_ementa"] = ordem.txt_observacao
# numeracao do processo 26/02/2011 # numeracao do processo 26/02/2011
dic["des_numeracao"] = "" dic["des_numeracao"] = ""
numeracao = context.zsql.numeracao_obter_zsql(cod_materia=ordem.cod_materia) numeracao = context.zsql.numeracao_obter_zsql(
cod_materia=ordem.cod_materia)
if len(numeracao): if len(numeracao):
numeracao = numeracao[0] numeracao = numeracao[0]
dic["des_numeracao"] = str(numeracao.num_materia)+"/"+str(numeracao.ano_materia) dic["des_numeracao"] = str(
numeracao.num_materia) + "/" + str(numeracao.ano_materia)
dic["des_turno"] = "" dic["des_turno"] = ""
dic["des_situacao"] = "" dic["des_situacao"] = ""
tramitacao = context.zsql.tramitacao_obter_zsql(cod_materia=ordem.cod_materia, ind_ult_tramitacao=1) tramitacao = context.zsql.tramitacao_obter_zsql(
cod_materia=ordem.cod_materia, ind_ult_tramitacao=1)
if len(tramitacao): if len(tramitacao):
tramitacao = tramitacao[0] tramitacao = tramitacao[0]
if tramitacao.sgl_turno != "": if tramitacao.sgl_turno != "":
@ -50,7 +61,8 @@ if context.REQUEST['cod_sessao_plen']!='':
dic["des_situacao"] = " " dic["des_situacao"] = " "
dic["nom_autor"] = '' dic["nom_autor"] = ''
autoria = context.zsql.autoria_obter_zsql(cod_materia=ordem.cod_materia, ind_primeiro_autor=1) autoria = context.zsql.autoria_obter_zsql(
cod_materia=ordem.cod_materia, ind_primeiro_autor=1)
if len(autoria): # se existe autor if len(autoria): # se existe autor
autoria = autoria[0] autoria = autoria[0]
try: try:
@ -59,11 +71,13 @@ if context.REQUEST['cod_sessao_plen']!='':
autor = autor[0] autor = autor[0]
if autor.des_tipo_autor == "Parlamentar": if autor.des_tipo_autor == "Parlamentar":
parlamentar = context.zsql.parlamentar_obter_zsql(cod_parlamentar=autor.cod_parlamentar)[0] parlamentar = context.zsql.parlamentar_obter_zsql(
cod_parlamentar=autor.cod_parlamentar)[0]
dic["nom_autor"] = parlamentar.nom_parlamentar dic["nom_autor"] = parlamentar.nom_parlamentar
elif autor.des_tipo_autor == "Comissao": elif autor.des_tipo_autor == "Comissao":
comissao = context.zsql.comissao_obter_zsql(cod_comissao=autor.cod_comissao)[0] comissao = context.zsql.comissao_obter_zsql(
cod_comissao=autor.cod_comissao)[0]
dic["nom_autor"] = comissao.nom_comissao dic["nom_autor"] = comissao.nom_comissao
else: else:
dic["nom_autor"] = autor.nom_autor dic["nom_autor"] = autor.nom_autor
@ -71,9 +85,12 @@ if context.REQUEST['cod_sessao_plen']!='':
pass pass
lst_relator = [] # lista contendo os relatores da matéria lst_relator = [] # lista contendo os relatores da matéria
for relatoria in context.zsql.relatoria_obter_zsql(cod_materia=ordem.cod_materia): for relatoria in context.zsql.relatoria_obter_zsql(cod_materia=ordem.cod_materia):
parlamentar = context.zsql.parlamentar_obter_zsql(cod_parlamentar=relatoria.cod_parlamentar)[0] parlamentar = context.zsql.parlamentar_obter_zsql(
comissao = context.zsql.comissao_obter_zsql(cod_comissao=relatoria.cod_comissao)[0] cod_parlamentar=relatoria.cod_parlamentar)[0]
lst_relator.append(parlamentar.nom_parlamentar+" - "+comissao.nom_comissao) comissao = context.zsql.comissao_obter_zsql(
cod_comissao=relatoria.cod_comissao)[0]
lst_relator.append(parlamentar.nom_parlamentar +
" - " + comissao.nom_comissao)
if not len(lst_relator): if not len(lst_relator):
lst_relator = [''] lst_relator = ['']
dic["lst_relator"] = lst_relator dic["lst_relator"] = lst_relator
@ -81,14 +98,16 @@ if context.REQUEST['cod_sessao_plen']!='':
# adiciona o dicionário na pauta # adiciona o dicionário na pauta
pauta.append(dic) pauta.append(dic)
# obtém as propriedades da casa legislativa para montar o cabeçalho e o rodapé da página # obtém as propriedades da casa legislativa para montar o cabeçalho e o
# rodapé da página
casa = {} casa = {}
aux = context.sapl_documentos.props_sapl.propertyItems() aux = context.sapl_documentos.props_sapl.propertyItems()
for item in aux: for item in aux:
casa[item[0]] = item[1] casa[item[0]] = item[1]
# obtém a localidade # obtém a localidade
localidade = context.zsql.localidade_obter_zsql(cod_localidade=casa["cod_localidade"]) localidade = context.zsql.localidade_obter_zsql(
cod_localidade=casa["cod_localidade"])
# monta o cabeçalho da página # monta o cabeçalho da página
cabecalho = {} cabecalho = {}
@ -103,7 +122,8 @@ if context.REQUEST['cod_sessao_plen']!='':
# tenta buscar o logotipo da casa LOGO_CASA # tenta buscar o logotipo da casa LOGO_CASA
if hasattr(context.sapl_documentos.props_sapl, 'logo_casa.gif'): if hasattr(context.sapl_documentos.props_sapl, 'logo_casa.gif'):
imagem = context.sapl_documentos.props_sapl['logo_casa.gif'].absolute_url() imagem = context.sapl_documentos.props_sapl[
'logo_casa.gif'].absolute_url()
else: else:
imagem = context.imagens.absolute_url() + "/brasao_transp.gif" imagem = context.imagens.absolute_url() + "/brasao_transp.gif"
@ -118,7 +138,8 @@ if context.REQUEST['cod_sessao_plen']!='':
linha1 = linha1 + " " linha1 = linha1 + " "
linha1 = linha1 + " CEP: " + num_cep linha1 = linha1 + " CEP: " + num_cep
if localidade[0].nom_localidade != None and localidade[0].nom_localidade != "": if localidade[0].nom_localidade != None and localidade[0].nom_localidade != "":
linha1 = linha1 +" "+localidade[0].nom_localidade +" - "+localidade[0].sgl_uf linha1 = linha1 + " " + \
localidade[0].nom_localidade + " - " + localidade[0].sgl_uf
if casa["num_tel"] != None and casa["num_tel"] != "": if casa["num_tel"] != None and casa["num_tel"] != "":
linha1 = linha1 + " Tel.: " + casa["num_tel"] linha1 = linha1 + " Tel.: " + casa["num_tel"]
@ -131,7 +152,8 @@ if context.REQUEST['cod_sessao_plen']!='':
rodape = [linha1, linha2, dat_emissao] rodape = [linha1, linha2, dat_emissao]
sessao = session.id sessao = session.id
caminho = context.pdf_ordem_dia_gerar( sessao, imagem, dat_ordem, splen, pauta, cabecalho, rodape) caminho = context.pdf_ordem_dia_gerar(
sessao, imagem, dat_ordem, splen, pauta, cabecalho, rodape)
if caminho == 'aviso': if caminho == 'aviso':
return response.redirect('mensagem_emitir_proc') return response.redirect('mensagem_emitir_proc')
else: else:

4
sapl/relatorios/templates/pdf_pauta_sessao_gerar.py

@ -23,7 +23,9 @@ def cabecalho(inf_basicas_dic, imagem):
tmp += '\t\t\t\t<setFont name="Helvetica-Bold" size="12"/>\n' tmp += '\t\t\t\t<setFont name="Helvetica-Bold" size="12"/>\n'
tmp += '\t\t\t\t<drawString x="4.2cm" y="25cm">Pauta da ' + str(inf_basicas_dic['num_sessao_plen']) + 'ª Reunião ' + str(inf_basicas_dic['nom_sessao']) + ' da ' + str( tmp += '\t\t\t\t<drawString x="4.2cm" y="25cm">Pauta da ' + str(inf_basicas_dic['num_sessao_plen']) + 'ª Reunião ' + str(inf_basicas_dic['nom_sessao']) + ' da ' + str(
inf_basicas_dic['num_sessao_leg']) + 'ª Sessão Legislativa da </drawString>\n' inf_basicas_dic['num_sessao_leg']) + 'ª Sessão Legislativa da </drawString>\n'
tmp += '\t\t\t\t<drawString x="6.7cm" y="24.5cm">' + str(inf_basicas_dic['num_legislatura']) + ' Legislatura </drawString>\n' tmp += '\t\t\t\t<drawString x="6.7cm" y="24.5cm">' + \
str(inf_basicas_dic['num_legislatura']) + \
' Legislatura </drawString>\n'
return tmp return tmp

81
sapl/relatorios/templates/pdf_pauta_sessao_preparar_pysc.py

@ -7,14 +7,16 @@ session= request.SESSION
if context.REQUEST['data'] != '': if context.REQUEST['data'] != '':
dat_inicio_sessao = context.REQUEST['data'] dat_inicio_sessao = context.REQUEST['data']
pauta = [] # lista contendo a pauta da ordem do dia a ser impressa pauta = [] # lista contendo a pauta da ordem do dia a ser impressa
data = context.pysc.data_converter_pysc(dat_inicio_sessao) # converte data para formato yyyy/mm/dd # converte data para formato yyyy/mm/dd
data = context.pysc.data_converter_pysc(dat_inicio_sessao)
codigo = context.REQUEST['cod_sessao_plen'] codigo = context.REQUEST['cod_sessao_plen']
# seleciona as matérias que compõem a pauta na data escolhida # seleciona as matérias que compõem a pauta na data escolhida
for sessao in context.zsql.sessao_plenaria_obter_zsql(dat_inicio_sessao=data, cod_sessao_plen=codigo, ind_excluido=0): for sessao in context.zsql.sessao_plenaria_obter_zsql(dat_inicio_sessao=data, cod_sessao_plen=codigo, ind_excluido=0):
inf_basicas_dic = {} # dicionário que armazenará as informacoes basicas da sessao plenaria inf_basicas_dic = {} # dicionário que armazenará as informacoes basicas da sessao plenaria
# seleciona o tipo da sessao plenaria # seleciona o tipo da sessao plenaria
tipo_sessao = context.zsql.tipo_sessao_plenaria_obter_zsql(tip_sessao=sessao.tip_sessao,ind_excluido=0)[0] tipo_sessao = context.zsql.tipo_sessao_plenaria_obter_zsql(
tip_sessao=sessao.tip_sessao, ind_excluido=0)[0]
inf_basicas_dic["nom_sessao"] = tipo_sessao.nom_sessao inf_basicas_dic["nom_sessao"] = tipo_sessao.nom_sessao
inf_basicas_dic["num_sessao_plen"] = sessao.num_sessao_plen inf_basicas_dic["num_sessao_plen"] = sessao.num_sessao_plen
inf_basicas_dic["nom_sessao"] = tipo_sessao.nom_sessao inf_basicas_dic["nom_sessao"] = tipo_sessao.nom_sessao
@ -30,40 +32,52 @@ if context.REQUEST['data']!='':
for expediente_materia in context.zsql.votacao_expediente_materia_obter_zsql(dat_ordem=data, cod_sessao_plen=codigo, ind_excluido=0): for expediente_materia in context.zsql.votacao_expediente_materia_obter_zsql(dat_ordem=data, cod_sessao_plen=codigo, ind_excluido=0):
# seleciona os detalhes de uma matéria # seleciona os detalhes de uma matéria
materia = context.zsql.materia_obter_zsql(cod_materia=expediente_materia.cod_materia)[0] materia = context.zsql.materia_obter_zsql(
cod_materia=expediente_materia.cod_materia)[0]
dic_expediente_materia = {} dic_expediente_materia = {}
dic_expediente_materia["num_ordem"] = expediente_materia.num_ordem dic_expediente_materia["num_ordem"] = expediente_materia.num_ordem
dic_expediente_materia["id_materia"] = materia.sgl_tipo_materia+" - "+materia.des_tipo_materia+" No. "+str(materia.num_ident_basica)+"/"+str(materia.ano_ident_basica) dic_expediente_materia["id_materia"] = materia.sgl_tipo_materia + " - " + materia.des_tipo_materia + \
" No. " + str(materia.num_ident_basica) + "/" + \
str(materia.ano_ident_basica)
dic_expediente_materia["txt_ementa"] = materia.txt_ementa dic_expediente_materia["txt_ementa"] = materia.txt_ementa
dic_expediente_materia["ordem_observacao"] = expediente_materia.ordem_observacao dic_expediente_materia[
"ordem_observacao"] = expediente_materia.ordem_observacao
dic_expediente_materia["des_numeracao"] = "" dic_expediente_materia["des_numeracao"] = ""
numeracao = context.zsql.numeracao_obter_zsql(cod_materia=expediente_materia.cod_materia) numeracao = context.zsql.numeracao_obter_zsql(
cod_materia=expediente_materia.cod_materia)
if len(numeracao): if len(numeracao):
numeracao = numeracao[0] numeracao = numeracao[0]
dic_expediente_materia["des_numeracao"] = str(numeracao.num_materia)+"/"+str(numeracao.ano_materia) dic_expediente_materia["des_numeracao"] = str(
numeracao.num_materia) + "/" + str(numeracao.ano_materia)
dic_expediente_materia["nom_autor"] = '' dic_expediente_materia["nom_autor"] = ''
autoria = context.zsql.autoria_obter_zsql(cod_materia=expediente_materia.cod_materia, ind_primeiro_autor=1) autoria = context.zsql.autoria_obter_zsql(
cod_materia=expediente_materia.cod_materia, ind_primeiro_autor=1)
if len(autoria) > 0: # se existe autor if len(autoria) > 0: # se existe autor
autoria = autoria[0] autoria = autoria[0]
autor = context.zsql.autor_obter_zsql(cod_autor=autoria.cod_autor) autor = context.zsql.autor_obter_zsql(
cod_autor=autoria.cod_autor)
if len(autor) > 0: if len(autor) > 0:
autor = autor[0] autor = autor[0]
if autor.des_tipo_autor == "Parlamentar": if autor.des_tipo_autor == "Parlamentar":
parlamentar = context.zsql.parlamentar_obter_zsql(cod_parlamentar=autor.cod_parlamentar)[0] parlamentar = context.zsql.parlamentar_obter_zsql(
dic_expediente_materia["nom_autor"] = parlamentar.nom_parlamentar cod_parlamentar=autor.cod_parlamentar)[0]
dic_expediente_materia[
"nom_autor"] = parlamentar.nom_parlamentar
elif autor.des_tipo_autor == "Comissao": elif autor.des_tipo_autor == "Comissao":
comissao = context.zsql.comissao_obter_zsql(cod_comissao=autor.cod_comissao)[0] comissao = context.zsql.comissao_obter_zsql(
cod_comissao=autor.cod_comissao)[0]
dic_expediente_materia["nom_autor"] = comissao.nom_comissao dic_expediente_materia["nom_autor"] = comissao.nom_comissao
else: else:
dic_expediente_materia["nom_autor"] = autor.nom_autor dic_expediente_materia["nom_autor"] = autor.nom_autor
dic_expediente_materia["des_turno"] = "" dic_expediente_materia["des_turno"] = ""
dic_expediente_materia["des_situacao"] = "" dic_expediente_materia["des_situacao"] = ""
tramitacao = context.zsql.tramitacao_obter_zsql(cod_materia=expediente_materia.cod_materia, ind_ult_tramitacao=1) tramitacao = context.zsql.tramitacao_obter_zsql(
cod_materia=expediente_materia.cod_materia, ind_ult_tramitacao=1)
if len(tramitacao): if len(tramitacao):
tramitacao = tramitacao[0] tramitacao = tramitacao[0]
if tramitacao.sgl_turno != "": if tramitacao.sgl_turno != "":
@ -76,46 +90,55 @@ if context.REQUEST['data']!='':
dic_expediente_materia["des_situacao"] = " " dic_expediente_materia["des_situacao"] = " "
lst_expediente_materia.append(dic_expediente_materia) lst_expediente_materia.append(dic_expediente_materia)
# Lista das matérias da Ordem do Dia, incluindo o status da tramitação # Lista das matérias da Ordem do Dia, incluindo o status da tramitação
lst_votacao = [] lst_votacao = []
for votacao in context.zsql.votacao_ordem_dia_obter_zsql(dat_ordem=data, cod_sessao_plen=codigo, ind_excluido=0): for votacao in context.zsql.votacao_ordem_dia_obter_zsql(dat_ordem=data, cod_sessao_plen=codigo, ind_excluido=0):
# seleciona os detalhes de uma matéria # seleciona os detalhes de uma matéria
materia = context.zsql.materia_obter_zsql(cod_materia=votacao.cod_materia)[0] materia = context.zsql.materia_obter_zsql(
cod_materia=votacao.cod_materia)[0]
dic_votacao = {} dic_votacao = {}
dic_votacao["num_ordem"] = votacao.num_ordem dic_votacao["num_ordem"] = votacao.num_ordem
dic_votacao["id_materia"] = materia.sgl_tipo_materia+" - "+materia.des_tipo_materia+" No. "+str(materia.num_ident_basica)+"/"+str(materia.ano_ident_basica) dic_votacao["id_materia"] = materia.sgl_tipo_materia + " - " + materia.des_tipo_materia + \
" No. " + str(materia.num_ident_basica) + "/" + \
str(materia.ano_ident_basica)
dic_votacao["txt_ementa"] = materia.txt_ementa dic_votacao["txt_ementa"] = materia.txt_ementa
dic_votacao["ordem_observacao"] = votacao.ordem_observacao dic_votacao["ordem_observacao"] = votacao.ordem_observacao
dic_votacao["des_numeracao"] = "" dic_votacao["des_numeracao"] = ""
numeracao = context.zsql.numeracao_obter_zsql(cod_materia=votacao.cod_materia) numeracao = context.zsql.numeracao_obter_zsql(
cod_materia=votacao.cod_materia)
if len(numeracao): if len(numeracao):
numeracao = numeracao[0] numeracao = numeracao[0]
dic_votacao["des_numeracao"] = str(numeracao.num_materia)+"/"+str(numeracao.ano_materia) dic_votacao["des_numeracao"] = str(
numeracao.num_materia) + "/" + str(numeracao.ano_materia)
dic_votacao["nom_autor"] = '' dic_votacao["nom_autor"] = ''
autoria = context.zsql.autoria_obter_zsql(cod_materia=votacao.cod_materia, ind_primeiro_autor=1) autoria = context.zsql.autoria_obter_zsql(
cod_materia=votacao.cod_materia, ind_primeiro_autor=1)
if len(autoria) > 0: # se existe autor if len(autoria) > 0: # se existe autor
autoria = autoria[0] autoria = autoria[0]
autor = context.zsql.autor_obter_zsql(cod_autor=autoria.cod_autor) autor = context.zsql.autor_obter_zsql(
cod_autor=autoria.cod_autor)
if len(autor) > 0: if len(autor) > 0:
autor = autor[0] autor = autor[0]
if autor.des_tipo_autor == "Parlamentar": if autor.des_tipo_autor == "Parlamentar":
parlamentar = context.zsql.parlamentar_obter_zsql(cod_parlamentar=autor.cod_parlamentar)[0] parlamentar = context.zsql.parlamentar_obter_zsql(
cod_parlamentar=autor.cod_parlamentar)[0]
dic_votacao["nom_autor"] = parlamentar.nom_parlamentar dic_votacao["nom_autor"] = parlamentar.nom_parlamentar
elif autor.des_tipo_autor == "Comissao": elif autor.des_tipo_autor == "Comissao":
comissao = context.zsql.comissao_obter_zsql(cod_comissao=autor.cod_comissao)[0] comissao = context.zsql.comissao_obter_zsql(
cod_comissao=autor.cod_comissao)[0]
dic_votacao["nom_autor"] = comissao.nom_comissao dic_votacao["nom_autor"] = comissao.nom_comissao
else: else:
dic_votacao["nom_autor"] = autor.nom_autor dic_votacao["nom_autor"] = autor.nom_autor
dic_votacao["des_turno"] = "" dic_votacao["des_turno"] = ""
dic_votacao["des_situacao"] = "" dic_votacao["des_situacao"] = ""
tramitacao = context.zsql.tramitacao_obter_zsql(cod_materia=votacao.cod_materia, ind_ult_tramitacao=1) tramitacao = context.zsql.tramitacao_obter_zsql(
cod_materia=votacao.cod_materia, ind_ult_tramitacao=1)
if len(tramitacao): if len(tramitacao):
tramitacao = tramitacao[0] tramitacao = tramitacao[0]
if tramitacao.sgl_turno != "": if tramitacao.sgl_turno != "":
@ -128,12 +151,14 @@ if context.REQUEST['data']!='':
dic_votacao["des_situacao"] = " " dic_votacao["des_situacao"] = " "
lst_votacao.append(dic_votacao) lst_votacao.append(dic_votacao)
# obtém as propriedades da casa legislativa para montar o cabeçalho e o rodapé da página # obtém as propriedades da casa legislativa para montar o cabeçalho e o
# rodapé da página
cabecalho = {} cabecalho = {}
# tenta buscar o logotipo da casa LOGO_CASA # tenta buscar o logotipo da casa LOGO_CASA
if hasattr(context.sapl_documentos.props_sapl, 'logo_casa.gif'): if hasattr(context.sapl_documentos.props_sapl, 'logo_casa.gif'):
imagem = context.sapl_documentos.props_sapl['logo_casa.gif'].absolute_url() imagem = context.sapl_documentos.props_sapl[
'logo_casa.gif'].absolute_url()
else: else:
imagem = context.imagens.absolute_url() + "/brasao_transp.gif" imagem = context.imagens.absolute_url() + "/brasao_transp.gif"
@ -142,7 +167,8 @@ if context.REQUEST['data']!='':
aux = context.sapl_documentos.props_sapl.propertyItems() aux = context.sapl_documentos.props_sapl.propertyItems()
for item in aux: for item in aux:
casa[item[0]] = item[1] casa[item[0]] = item[1]
localidade=context.zsql.localidade_obter_zsql(cod_localidade=casa["cod_localidade"]) localidade = context.zsql.localidade_obter_zsql(
cod_localidade=casa["cod_localidade"])
data_emissao = DateTime().strftime("%d/%m/%Y") data_emissao = DateTime().strftime("%d/%m/%Y")
rodape = casa rodape = casa
rodape['data_emissao'] = data_emissao rodape['data_emissao'] = data_emissao
@ -155,7 +181,8 @@ if context.REQUEST['data']!='':
# return lst_votacao # return lst_votacao
sessao = session.id sessao = session.id
caminho = context.pdf_pauta_sessao_gerar(rodape, sessao, imagem, inf_basicas_dic, lst_votacao, lst_expediente_materia) caminho = context.pdf_pauta_sessao_gerar(
rodape, sessao, imagem, inf_basicas_dic, lst_votacao, lst_expediente_materia)
if caminho == 'aviso': if caminho == 'aviso':
return response.redirect('mensagem_emitir_proc') return response.redirect('mensagem_emitir_proc')
else: else:

39
sapl/relatorios/templates/pdf_protocolo_gerar.py

@ -1,4 +1,4 @@
##parameters=sessao,imagem,data,lst_protocolos,dic_cabecalho,lst_rodape,dic_filtro # parameters=sessao,imagem,data,lst_protocolos,dic_cabecalho,lst_rodape,dic_filtro
"""relatorio_protocolo.py """relatorio_protocolo.py
External method para gerar o arquivo rml do resultado de uma pesquisa de protocolos External method para gerar o arquivo rml do resultado de uma pesquisa de protocolos
@ -17,7 +17,8 @@ def cabecalho(dic_cabecalho,imagem):
tmp_data += '\t\t\t\t<image x="2.1cm" y="25.7cm" width="59" height="62" file="' + imagem + '"/>\n' tmp_data += '\t\t\t\t<image x="2.1cm" y="25.7cm" width="59" height="62" file="' + imagem + '"/>\n'
tmp_data += '\t\t\t\t<lines>2cm 25.4cm 19cm 25.4cm</lines>\n' tmp_data += '\t\t\t\t<lines>2cm 25.4cm 19cm 25.4cm</lines>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="15"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="15"/>\n'
tmp_data+='\t\t\t\t<drawString x="5cm" y="27.2cm">' + dic_cabecalho['nom_casa'] + '</drawString>\n' tmp_data += '\t\t\t\t<drawString x="5cm" y="27.2cm">' + \
dic_cabecalho['nom_casa'] + '</drawString>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica" size="12"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica" size="12"/>\n'
tmp_data += '\t\t\t\t<drawString x="5cm" y="26.6cm">Sistema de Apoio ao Processo Legislativo</drawString>\n' tmp_data += '\t\t\t\t<drawString x="5cm" y="26.6cm">Sistema de Apoio ao Processo Legislativo</drawString>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="13"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica-Bold" size="13"/>\n'
@ -25,19 +26,24 @@ def cabecalho(dic_cabecalho,imagem):
return tmp_data return tmp_data
def rodape(lst_rodape): def rodape(lst_rodape):
"""Gera o codigo rml do rodape""" """Gera o codigo rml do rodape"""
tmp_data = '' tmp_data = ''
tmp_data += '\t\t\t\t<lines>2cm 3.2cm 19cm 3.2cm</lines>\n' tmp_data += '\t\t\t\t<lines>2cm 3.2cm 19cm 3.2cm</lines>\n'
tmp_data += '\t\t\t\t<setFont name="Helvetica" size="8"/>\n' tmp_data += '\t\t\t\t<setFont name="Helvetica" size="8"/>\n'
tmp_data+='\t\t\t\t<drawString x="2cm" y="3.3cm">' + lst_rodape[2] + '</drawString>\n' tmp_data += '\t\t\t\t<drawString x="2cm" y="3.3cm">' + \
lst_rodape[2] + '</drawString>\n'
tmp_data += '\t\t\t\t<drawString x="17.9cm" y="3.3cm">Página <pageNumber/></drawString>\n' tmp_data += '\t\t\t\t<drawString x="17.9cm" y="3.3cm">Página <pageNumber/></drawString>\n'
tmp_data+='\t\t\t\t<drawCentredString x="10.5cm" y="2.7cm">' + lst_rodape[0] + '</drawCentredString>\n' tmp_data += '\t\t\t\t<drawCentredString x="10.5cm" y="2.7cm">' + \
tmp_data+='\t\t\t\t<drawCentredString x="10.5cm" y="2.3cm">' + lst_rodape[1] + '</drawCentredString>\n' lst_rodape[0] + '</drawCentredString>\n'
tmp_data += '\t\t\t\t<drawCentredString x="10.5cm" y="2.3cm">' + \
lst_rodape[1] + '</drawCentredString>\n'
return tmp_data return tmp_data
def paraStyle(): def paraStyle():
"""Gera o codigo rml que define o estilo dos paragrafos""" """Gera o codigo rml que define o estilo dos paragrafos"""
@ -56,6 +62,7 @@ def paraStyle():
return tmp_data return tmp_data
def protocolos(lst_protocolos): def protocolos(lst_protocolos):
"""Gera o codigo rml do conteudo da pesquisa de protocolos""" """Gera o codigo rml do conteudo da pesquisa de protocolos"""
@ -78,7 +85,8 @@ def protocolos(lst_protocolos):
# protocolos # protocolos
if dic['titulo'] != None: if dic['titulo'] != None:
tmp_data+='\t\t<para style="P1">Protocolo ' + dic['titulo'] + '</para>\n' tmp_data += '\t\t<para style="P1">Protocolo ' + \
dic['titulo'] + '</para>\n'
tmp_data += '\t\t<para style="P1">\n' tmp_data += '\t\t<para style="P1">\n'
tmp_data += '\t\t\t<font color="white"> </font>\n' tmp_data += '\t\t\t<font color="white"> </font>\n'
tmp_data += '\t\t</para>\n' tmp_data += '\t\t</para>\n'
@ -86,21 +94,27 @@ def protocolos(lst_protocolos):
txt_assunto = dic['txt_assunto'].replace('&', '&amp;') txt_assunto = dic['txt_assunto'].replace('&', '&amp;')
tmp_data += '\t\t<para style="P2">' + txt_assunto + '</para>\n' tmp_data += '\t\t<para style="P2">' + txt_assunto + '</para>\n'
if dic['txt_interessado'] != None: if dic['txt_interessado'] != None:
tmp_data+='\t\t<para style="P2"><b>Interessado:</b> ' + dic['txt_interessado'] + '</para>\n' tmp_data += '\t\t<para style="P2"><b>Interessado:</b> ' + \
dic['txt_interessado'] + '</para>\n'
elif dic['nom_autor'] != None: elif dic['nom_autor'] != None:
tmp_data+='\t\t<para style="P2"><b>Autor:</b> ' + dic['nom_autor'] + '</para>\n' tmp_data += '\t\t<para style="P2"><b>Autor:</b> ' + \
dic['nom_autor'] + '</para>\n'
if dic['natureza'] != None: if dic['natureza'] != None:
tmp_data+='\t\t<para style="P2"><b>Natureza Processo:</b> ' + dic['natureza'] + '</para>\n' tmp_data += '\t\t<para style="P2"><b>Natureza Processo:</b> ' + \
dic['natureza'] + '</para>\n'
if dic['processo'] != None: if dic['processo'] != None:
tmp_data+='\t\t<para style="P2"><b>Classificação:</b> ' + dic['processo'] + '</para>\n' tmp_data += '\t\t<para style="P2"><b>Classificação:</b> ' + \
dic['processo'] + '</para>\n'
if dic['data'] != None: if dic['data'] != None:
tmp_data+='\t\t<para style="P2"><b>Data Protocolo:</b> ' + dic['data'] + '</para>\n' tmp_data += '\t\t<para style="P2"><b>Data Protocolo:</b> ' + \
dic['data'] + '</para>\n'
if dic['anulado'] != "": if dic['anulado'] != "":
tmp_data += '\t\t<para style="P2"><b>** PROTOCOLO ANULADO **</b> ' '</para>\n' tmp_data += '\t\t<para style="P2"><b>** PROTOCOLO ANULADO **</b> ' '</para>\n'
tmp_data += '\t</story>\n' tmp_data += '\t</story>\n'
return tmp_data return tmp_data
def principal(imagem, lst_protocolos, dic_cabecalho, lst_rodape): def principal(imagem, lst_protocolos, dic_cabecalho, lst_rodape):
"""Funcao pricipal que gera a estrutura global do arquivo rml""" """Funcao pricipal que gera a estrutura global do arquivo rml"""
@ -134,4 +148,5 @@ def principal(imagem, lst_protocolos, dic_cabecalho, lst_rodape):
# return "/temp_folder/"+arquivoPdf # return "/temp_folder/"+arquivoPdf
# return principal(sessao,imagem,data,lst_protocolos,dic_cabecalho,lst_rodape,dic_filtro) # return
# principal(sessao,imagem,data,lst_protocolos,dic_cabecalho,lst_rodape,dic_filtro)

27
sapl/relatorios/templates/pdf_protocolo_preparar_pysc.py

@ -11,7 +11,8 @@ casa={}
aux = context.sapl_documentos.props_sapl.propertyItems() aux = context.sapl_documentos.props_sapl.propertyItems()
for item in aux: for item in aux:
casa[item[0]] = item[1] casa[item[0]] = item[1]
localidade=context.zsql.localidade_obter_zsql(cod_localidade=casa["cod_localidade"]) localidade = context.zsql.localidade_obter_zsql(
cod_localidade=casa["cod_localidade"])
if len(casa["num_cep"]) == 8: if len(casa["num_cep"]) == 8:
cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:] cep = casa["num_cep"][:4] + "-" + casa["num_cep"][5:]
else: else:
@ -23,7 +24,8 @@ if cep!="":
linha1 = linha1 + " - " linha1 = linha1 + " - "
linha1 = linha1 + "CEP " + cep linha1 = linha1 + "CEP " + cep
if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None: if localidade[0].nom_localidade != "" and localidade[0].nom_localidade != None:
linha1 = linha1 + " - "+localidade[0].nom_localidade+" "+localidade[0].sgl_uf linha1 = linha1 + " - " + \
localidade[0].nom_localidade + " " + localidade[0].sgl_uf
if casa["num_tel"] != "" and casa["num_tel"] != None: if casa["num_tel"] != "" and casa["num_tel"] != None:
linha1 = linha1 + " Tel.: " + casa["num_tel"] linha1 = linha1 + " Tel.: " + casa["num_tel"]
@ -57,16 +59,22 @@ else:
protocolos = [] protocolos = []
REQUEST = context.REQUEST REQUEST = context.REQUEST
for protocolo in context.zsql.protocolo_pesquisar_zsql(tip_protocolo=REQUEST['rad_tip_protocolo'], for protocolo in context.zsql.protocolo_pesquisar_zsql(tip_protocolo=REQUEST['rad_tip_protocolo'],
cod_protocolo=REQUEST['txt_num_protocolo'], ano_protocolo=REQUEST['txt_ano_protocolo'], cod_protocolo=REQUEST['txt_num_protocolo'], ano_protocolo=REQUEST[
tip_documento=REQUEST['lst_tip_documento'], tip_processo=REQUEST['rad_tip_processo'], 'txt_ano_protocolo'],
tip_materia=REQUEST['lst_tip_materia'], des_assunto=REQUEST['txt_assunto'], tip_documento=REQUEST['lst_tip_documento'], tip_processo=REQUEST[
cod_autor=REQUEST['hdn_cod_autor'], des_interessado=REQUEST['txa_txt_interessado'], 'rad_tip_processo'],
tip_materia=REQUEST[
'lst_tip_materia'], des_assunto=REQUEST['txt_assunto'],
cod_autor=REQUEST['hdn_cod_autor'], des_interessado=REQUEST[
'txa_txt_interessado'],
dat_apres=REQUEST['dt_apres'], dat_apres2=REQUEST['dt_apres2']): dat_apres=REQUEST['dt_apres'], dat_apres2=REQUEST['dt_apres2']):
dic = {} dic = {}
dic['titulo']=str(protocolo.cod_protocolo)+'/'+str(protocolo.ano_protocolo) dic['titulo'] = str(protocolo.cod_protocolo) + '/' + \
str(protocolo.ano_protocolo)
dic['data']=context.pysc.iso_to_port_pysc(protocolo.dat_protocolo)+' - <b>Horário:</b>'+protocolo.hor_protocolo dic['data'] = context.pysc.iso_to_port_pysc(
protocolo.dat_protocolo) + ' - <b>Horário:</b>' + protocolo.hor_protocolo
dic['txt_assunto'] = protocolo.txt_assunto_ementa dic['txt_assunto'] = protocolo.txt_assunto_ementa
@ -110,7 +118,8 @@ filtro['autor']=REQUEST.hdn_cod_autor
filtro['interessado'] = REQUEST.txa_txt_interessado filtro['interessado'] = REQUEST.txa_txt_interessado
sessao = session.id sessao = session.id
caminho = context.pdf_protocolo_gerar(sessao,imagem,data,protocolos,cabecalho,rodape,filtro) caminho = context.pdf_protocolo_gerar(
sessao, imagem, data, protocolos, cabecalho, rodape, filtro)
if caminho == 'aviso': if caminho == 'aviso':
return response.redirect('mensagem_emitir_proc') return response.redirect('mensagem_emitir_proc')
else: else:

10
sapl/relatorios/templates/pdf_sessao_plenaria_gerar.py

@ -26,7 +26,9 @@ def cabecalho(inf_basicas_dic, imagem):
tmp += '\t\t\t\t<setFont name="Helvetica-Bold" size="12"/>\n' tmp += '\t\t\t\t<setFont name="Helvetica-Bold" size="12"/>\n'
tmp += '\t\t\t\t<drawString x="4.2cm" y="25cm">Resumo da ' + str(inf_basicas_dic['num_sessao_plen']) + 'ª Reunião ' + str(inf_basicas_dic['nom_sessao']) + ' da ' + str( tmp += '\t\t\t\t<drawString x="4.2cm" y="25cm">Resumo da ' + str(inf_basicas_dic['num_sessao_plen']) + 'ª Reunião ' + str(inf_basicas_dic['nom_sessao']) + ' da ' + str(
inf_basicas_dic['num_sessao_leg']) + 'ª Sessão Legislativa da </drawString>\n' inf_basicas_dic['num_sessao_leg']) + 'ª Sessão Legislativa da </drawString>\n'
tmp += '\t\t\t\t<drawString x="6.7cm" y="24.5cm">' + str(inf_basicas_dic['num_legislatura']) + ' Legislatura </drawString>\n' tmp += '\t\t\t\t<drawString x="6.7cm" y="24.5cm">' + \
str(inf_basicas_dic['num_legislatura']) + \
' Legislatura </drawString>\n'
return tmp return tmp
@ -183,7 +185,8 @@ def expediente_materia(lst_expediente_materia):
txt_ementa = expediente_materia['txt_ementa'].replace('&', '&amp;') txt_ementa = expediente_materia['txt_ementa'].replace('&', '&amp;')
tmp += '<td><para style="P4">' + txt_ementa + '</para></td>\n' tmp += '<td><para style="P4">' + txt_ementa + '</para></td>\n'
tmp += '<td><para style="P3"><b>' + \ tmp += '<td><para style="P3"><b>' + \
str(expediente_materia['nom_resultado']) + '</b></para>\n' + '<para style="P3">' str(expediente_materia['nom_resultado']) + \
'</b></para>\n' + '<para style="P3">'
if expediente_materia['votacao_observacao'] != txt_ementa: if expediente_materia['votacao_observacao'] != txt_ementa:
tmp += str(expediente_materia['votacao_observacao']) tmp += str(expediente_materia['votacao_observacao'])
else: else:
@ -242,7 +245,8 @@ def votacao(lst_votacao):
txt_ementa = votacao['txt_ementa'].replace('&', '&amp;') txt_ementa = votacao['txt_ementa'].replace('&', '&amp;')
tmp += '<td><para style="P4">' + txt_ementa + '</para></td>\n' tmp += '<td><para style="P4">' + txt_ementa + '</para></td>\n'
tmp += '<td><para style="P3"><b>' + \ tmp += '<td><para style="P3"><b>' + \
str(votacao['nom_resultado']) + '</b></para>\n' + '<para style="P3">' str(votacao['nom_resultado']) + \
'</b></para>\n' + '<para style="P3">'
if votacao['votacao_observacao'] != txt_ementa: if votacao['votacao_observacao'] != txt_ementa:
tmp += str(votacao['votacao_observacao']) tmp += str(votacao['votacao_observacao'])
else: else:

125
sapl/relatorios/templates/pdf_sessao_plenaria_preparar_pysc.py

@ -7,14 +7,16 @@ session= request.SESSION
if context.REQUEST['data'] != '': if context.REQUEST['data'] != '':
dat_inicio_sessao = context.REQUEST['data'] dat_inicio_sessao = context.REQUEST['data']
pauta = [] # lista contendo a pauta da ordem do dia a ser impressa pauta = [] # lista contendo a pauta da ordem do dia a ser impressa
data = context.pysc.data_converter_pysc(dat_inicio_sessao) # converte data para formato yyyy/mm/dd # converte data para formato yyyy/mm/dd
data = context.pysc.data_converter_pysc(dat_inicio_sessao)
codigo = context.REQUEST['cod_sessao_plen'] codigo = context.REQUEST['cod_sessao_plen']
# seleciona as matérias que compõem a pauta na data escolhida # seleciona as matérias que compõem a pauta na data escolhida
for sessao in context.zsql.sessao_plenaria_obter_zsql(dat_inicio_sessao=data, cod_sessao_plen=codigo, ind_excluido=0): for sessao in context.zsql.sessao_plenaria_obter_zsql(dat_inicio_sessao=data, cod_sessao_plen=codigo, ind_excluido=0):
inf_basicas_dic = {} # dicionário que armazenará as informacoes basicas da sessao plenaria inf_basicas_dic = {} # dicionário que armazenará as informacoes basicas da sessao plenaria
# seleciona o tipo da sessao plenaria # seleciona o tipo da sessao plenaria
tipo_sessao = context.zsql.tipo_sessao_plenaria_obter_zsql(tip_sessao=sessao.tip_sessao,ind_excluido=0)[0] tipo_sessao = context.zsql.tipo_sessao_plenaria_obter_zsql(
tip_sessao=sessao.tip_sessao, ind_excluido=0)[0]
inf_basicas_dic["num_sessao_plen"] = sessao.num_sessao_plen inf_basicas_dic["num_sessao_plen"] = sessao.num_sessao_plen
inf_basicas_dic["nom_sessao"] = tipo_sessao.nom_sessao inf_basicas_dic["nom_sessao"] = tipo_sessao.nom_sessao
inf_basicas_dic["num_legislatura"] = sessao.num_legislatura inf_basicas_dic["num_legislatura"] = sessao.num_legislatura
@ -50,7 +52,8 @@ if context.REQUEST['data']!='':
for tip_expediente in context.zsql.tipo_expediente_obter_zsql(): for tip_expediente in context.zsql.tipo_expediente_obter_zsql():
for expediente in context.zsql.expediente_obter_zsql(cod_sessao_plen=sessao.cod_sessao_plen, cod_expediente=tip_expediente.cod_expediente, ind_excluido=0): for expediente in context.zsql.expediente_obter_zsql(cod_sessao_plen=sessao.cod_sessao_plen, cod_expediente=tip_expediente.cod_expediente, ind_excluido=0):
dic_expedientes = {} dic_expedientes = {}
dic_expedientes["nom_expediente"] = tip_expediente.nom_expediente dic_expedientes[
"nom_expediente"] = tip_expediente.nom_expediente
dic_expedientes["txt_expediente"] = expediente.txt_expediente dic_expedientes["txt_expediente"] = expediente.txt_expediente
if dic_expedientes: if dic_expedientes:
@ -61,19 +64,25 @@ if context.REQUEST['data']!='':
for expediente_materia in context.zsql.votacao_expediente_materia_obter_zsql(dat_ordem=data, cod_sessao_plen=sessao.cod_sessao_plen, ind_excluido=0): for expediente_materia in context.zsql.votacao_expediente_materia_obter_zsql(dat_ordem=data, cod_sessao_plen=sessao.cod_sessao_plen, ind_excluido=0):
# seleciona os detalhes de uma matéria # seleciona os detalhes de uma matéria
materia = context.zsql.materia_obter_zsql(cod_materia=expediente_materia.cod_materia)[0] materia = context.zsql.materia_obter_zsql(
cod_materia=expediente_materia.cod_materia)[0]
dic_expediente_materia = {} dic_expediente_materia = {}
dic_expediente_materia["num_ordem"] = expediente_materia.num_ordem dic_expediente_materia["num_ordem"] = expediente_materia.num_ordem
dic_expediente_materia["id_materia"] = materia.sgl_tipo_materia+" "+materia.des_tipo_materia+" "+str(materia.num_ident_basica)+"/"+str(materia.ano_ident_basica) dic_expediente_materia["id_materia"] = materia.sgl_tipo_materia + " " + materia.des_tipo_materia + \
" " + str(materia.num_ident_basica) + "/" + \
str(materia.ano_ident_basica)
dic_expediente_materia["des_numeracao"] = "" dic_expediente_materia["des_numeracao"] = ""
numeracao = context.zsql.numeracao_obter_zsql(cod_materia=expediente_materia.cod_materia) numeracao = context.zsql.numeracao_obter_zsql(
cod_materia=expediente_materia.cod_materia)
if len(numeracao): if len(numeracao):
numeracao = numeracao[0] numeracao = numeracao[0]
dic_expediente_materia["des_numeracao"] = str(numeracao.num_materia)+"/"+str(numeracao.ano_materia) dic_expediente_materia["des_numeracao"] = str(
numeracao.num_materia) + "/" + str(numeracao.ano_materia)
tram = context.zsql.tramitacao_turno_obter_zsql(cod_materia=materia.cod_materia, dat_inicio_sessao=data) tram = context.zsql.tramitacao_turno_obter_zsql(
cod_materia=materia.cod_materia, dat_inicio_sessao=data)
dic_expediente_materia["des_turno"] = "" dic_expediente_materia["des_turno"] = ""
if len(tram): if len(tram):
tram_turno = tram[0] tram_turno = tram[0]
@ -83,35 +92,46 @@ if context.REQUEST['data']!='':
dic_expediente_materia["des_turno"] = turno[1] dic_expediente_materia["des_turno"] = turno[1]
dic_expediente_materia["txt_ementa"] = materia.txt_ementa dic_expediente_materia["txt_ementa"] = materia.txt_ementa
dic_expediente_materia["ordem_observacao"] = expediente_materia.ordem_observacao dic_expediente_materia[
"ordem_observacao"] = expediente_materia.ordem_observacao
dic_expediente_materia["nom_autor"] = "" dic_expediente_materia["nom_autor"] = ""
autoria = context.zsql.autoria_obter_zsql(cod_materia=expediente_materia.cod_materia, ind_primeiro_autor=1) autoria = context.zsql.autoria_obter_zsql(
cod_materia=expediente_materia.cod_materia, ind_primeiro_autor=1)
if len(autoria) > 0: # se existe autor if len(autoria) > 0: # se existe autor
autoria = autoria[0] autoria = autoria[0]
autor = context.zsql.autor_obter_zsql(cod_autor=autoria.cod_autor) autor = context.zsql.autor_obter_zsql(
cod_autor=autoria.cod_autor)
if len(autor) > 0: if len(autor) > 0:
autor = autor[0] autor = autor[0]
try: try:
if autor.des_tipo_autor == "Parlamentar": if autor.des_tipo_autor == "Parlamentar":
parlamentar = context.zsql.parlamentar_obter_zsql(cod_parlamentar=autor.cod_parlamentar)[0] parlamentar = context.zsql.parlamentar_obter_zsql(
dic_expediente_materia["nom_autor"] = parlamentar.nom_parlamentar cod_parlamentar=autor.cod_parlamentar)[0]
dic_expediente_materia[
"nom_autor"] = parlamentar.nom_parlamentar
elif autor.des_tipo_autor == "Comissao": elif autor.des_tipo_autor == "Comissao":
comissao = context.zsql.comissao_obter_zsql(cod_comissao=autor.cod_comissao)[0] comissao = context.zsql.comissao_obter_zsql(
dic_expediente_materia["nom_autor"] = comissao.nom_comissao cod_comissao=autor.cod_comissao)[0]
dic_expediente_materia[
"nom_autor"] = comissao.nom_comissao
elif autor.nom_autor != "": elif autor.nom_autor != "":
dic_expediente_materia["nom_autor"] = autor.nom_autor dic_expediente_materia[
"nom_autor"] = autor.nom_autor
else: else:
dic_expediente_materia["nom_autor"] = autor.des_tipo_autor dic_expediente_materia[
"nom_autor"] = autor.des_tipo_autor
except: except:
dic_expediente_materia["nom_autor"] = "NC-em" dic_expediente_materia["nom_autor"] = "NC-em"
dic_expediente_materia["votacao_observacao"] = "" dic_expediente_materia["votacao_observacao"] = ""
if expediente_materia.tip_resultado_votacao: if expediente_materia.tip_resultado_votacao:
resultado = context.zsql.tipo_resultado_votacao_obter_zsql(tip_resultado_votacao=expediente_materia.tip_resultado_votacao, ind_excluido=0) resultado = context.zsql.tipo_resultado_votacao_obter_zsql(
tip_resultado_votacao=expediente_materia.tip_resultado_votacao, ind_excluido=0)
for i in resultado: for i in resultado:
dic_expediente_materia["nom_resultado"] = i.nom_resultado dic_expediente_materia["nom_resultado"] = i.nom_resultado
if expediente_materia.votacao_observacao: if expediente_materia.votacao_observacao:
dic_expediente_materia["votacao_observacao"] = expediente_materia.votacao_observacao dic_expediente_materia[
"votacao_observacao"] = expediente_materia.votacao_observacao
else: else:
dic_expediente_materia["nom_resultado"] = "Matéria não votada" dic_expediente_materia["nom_resultado"] = "Matéria não votada"
dic_expediente_materia["votacao_observacao"] = "Vazio" dic_expediente_materia["votacao_observacao"] = "Vazio"
@ -122,9 +142,12 @@ if context.REQUEST['data']!='':
for orador_expediente in context.zsql.oradores_expediente_obter_zsql(cod_sessao_plen=sessao.cod_sessao_plen, ind_excluido=0): for orador_expediente in context.zsql.oradores_expediente_obter_zsql(cod_sessao_plen=sessao.cod_sessao_plen, ind_excluido=0):
for parlamentar in context.zsql.parlamentar_obter_zsql(cod_parlamentar=orador_expediente.cod_parlamentar, ind_excluido=0): for parlamentar in context.zsql.parlamentar_obter_zsql(cod_parlamentar=orador_expediente.cod_parlamentar, ind_excluido=0):
dic_oradores_expediente = {} dic_oradores_expediente = {}
dic_oradores_expediente["num_ordem"] = orador_expediente.num_ordem dic_oradores_expediente[
dic_oradores_expediente["nom_parlamentar"] = parlamentar.nom_parlamentar "num_ordem"] = orador_expediente.num_ordem
dic_oradores_expediente['sgl_partido'] = parlamentar.sgl_partido dic_oradores_expediente[
"nom_parlamentar"] = parlamentar.nom_parlamentar
dic_oradores_expediente[
'sgl_partido'] = parlamentar.sgl_partido
lst_oradores_expediente.append(dic_oradores_expediente) lst_oradores_expediente.append(dic_oradores_expediente)
# Lista presença na ordem do dia # Lista presença na ordem do dia
@ -132,30 +155,39 @@ if context.REQUEST['data']!='':
for presenca_ordem_dia in context.zsql.presenca_ordem_dia_obter_zsql(cod_sessao_plen=sessao.cod_sessao_plen, ind_excluido=0): for presenca_ordem_dia in context.zsql.presenca_ordem_dia_obter_zsql(cod_sessao_plen=sessao.cod_sessao_plen, ind_excluido=0):
for parlamentar in context.zsql.parlamentar_obter_zsql(cod_parlamentar=presenca_ordem_dia.cod_parlamentar, ind_excluido=0): for parlamentar in context.zsql.parlamentar_obter_zsql(cod_parlamentar=presenca_ordem_dia.cod_parlamentar, ind_excluido=0):
dic_presenca_ordem_dia = {} dic_presenca_ordem_dia = {}
dic_presenca_ordem_dia['nom_parlamentar'] = parlamentar.nom_parlamentar dic_presenca_ordem_dia[
'nom_parlamentar'] = parlamentar.nom_parlamentar
dic_presenca_ordem_dia['sgl_partido'] = parlamentar.sgl_partido dic_presenca_ordem_dia['sgl_partido'] = parlamentar.sgl_partido
lst_presenca_ordem_dia.append(dic_presenca_ordem_dia) lst_presenca_ordem_dia.append(dic_presenca_ordem_dia)
# Lista das matérias da Ordem do Dia, incluindo o resultado das votacoes # Lista das matérias da Ordem do Dia, incluindo o resultado das
# votacoes
lst_votacao = [] lst_votacao = []
for votacao in context.zsql.votacao_ordem_dia_obter_zsql(dat_ordem=data, cod_sessao_plen=sessao.cod_sessao_plen, ind_excluido=0): for votacao in context.zsql.votacao_ordem_dia_obter_zsql(dat_ordem=data, cod_sessao_plen=sessao.cod_sessao_plen, ind_excluido=0):
# seleciona os detalhes de uma matéria # seleciona os detalhes de uma matéria
materia = context.zsql.materia_obter_zsql(cod_materia=votacao.cod_materia)[0] materia = context.zsql.materia_obter_zsql(
cod_materia=votacao.cod_materia)[0]
dic_votacao = {} dic_votacao = {}
dic_votacao["num_ordem"] = votacao.num_ordem dic_votacao["num_ordem"] = votacao.num_ordem
dic_votacao["id_materia"] = materia.sgl_tipo_materia+" "+materia.des_tipo_materia+" "+str(materia.num_ident_basica)+"/"+str(materia.ano_ident_basica) dic_votacao["id_materia"] = materia.sgl_tipo_materia + " " + materia.des_tipo_materia + \
" " + str(materia.num_ident_basica) + "/" + \
str(materia.ano_ident_basica)
dic_votacao["des_numeracao"] = "" dic_votacao["des_numeracao"] = ""
numeracao = context.zsql.numeracao_obter_zsql(cod_materia=votacao.cod_materia) numeracao = context.zsql.numeracao_obter_zsql(
cod_materia=votacao.cod_materia)
if len(numeracao): if len(numeracao):
numeracao = numeracao[0] numeracao = numeracao[0]
dic_votacao["des_numeracao"] = str(numeracao.num_materia)+"/"+str(numeracao.ano_materia) dic_votacao["des_numeracao"] = str(
numeracao.num_materia) + "/" + str(numeracao.ano_materia)
dic_votacao["des_turno"] = "" dic_votacao["des_turno"] = ""
tramitacao = context.zsql.tramitacao_obter_zsql(cod_materia=materia.cod_materia, ind_ult_tramitacao=1) tramitacao = context.zsql.tramitacao_obter_zsql(
cod_materia=materia.cod_materia, ind_ult_tramitacao=1)
if len(tramitacao): if len(tramitacao):
tramitacao = tramitacao[0] tramitacao = tramitacao[0]
tram = context.zsql.tramitacao_turno_obter_zsql(cod_materia=materia.cod_materia, dat_inicio_sessao=data) tram = context.zsql.tramitacao_turno_obter_zsql(
cod_materia=materia.cod_materia, dat_inicio_sessao=data)
if len(tram): if len(tram):
tram_turno = tram[0] tram_turno = tram[0]
if tram_turno.sgl_turno != "": if tram_turno.sgl_turno != "":
@ -165,18 +197,23 @@ if context.REQUEST['data']!='':
dic_votacao["txt_ementa"] = materia.txt_ementa dic_votacao["txt_ementa"] = materia.txt_ementa
dic_votacao["ordem_observacao"] = votacao.ordem_observacao dic_votacao["ordem_observacao"] = votacao.ordem_observacao
dic_votacao["nom_autor"] = "" dic_votacao["nom_autor"] = ""
autoria = context.zsql.autoria_obter_zsql(cod_materia=votacao.cod_materia, ind_primeiro_autor=1) autoria = context.zsql.autoria_obter_zsql(
cod_materia=votacao.cod_materia, ind_primeiro_autor=1)
if len(autoria) > 0: # se existe autor if len(autoria) > 0: # se existe autor
autoria = autoria[0] autoria = autoria[0]
autor = context.zsql.autor_obter_zsql(cod_autor=autoria.cod_autor) autor = context.zsql.autor_obter_zsql(
cod_autor=autoria.cod_autor)
if len(autor) > 0: if len(autor) > 0:
autor = autor[0] autor = autor[0]
try: try:
if autor.des_tipo_autor == "Parlamentar": if autor.des_tipo_autor == "Parlamentar":
parlamentar = context.zsql.parlamentar_obter_zsql(cod_parlamentar=autor.cod_parlamentar)[0] parlamentar = context.zsql.parlamentar_obter_zsql(
dic_votacao["nom_autor"] = parlamentar.nom_parlamentar cod_parlamentar=autor.cod_parlamentar)[0]
dic_votacao[
"nom_autor"] = parlamentar.nom_parlamentar
elif autor.des_tipo_autor == "Comissao": elif autor.des_tipo_autor == "Comissao":
comissao = context.zsql.comissao_obter_zsql(cod_comissao=autor.cod_comissao)[0] comissao = context.zsql.comissao_obter_zsql(
cod_comissao=autor.cod_comissao)[0]
dic_votacao["nom_autor"] = comissao.nom_comissao dic_votacao["nom_autor"] = comissao.nom_comissao
elif autor.nom_autor != "": elif autor.nom_autor != "":
dic_votacao["nom_autor"] = autor.nom_autor dic_votacao["nom_autor"] = autor.nom_autor
@ -187,11 +224,13 @@ if context.REQUEST['data']!='':
dic_votacao["votacao_observacao"] = "" dic_votacao["votacao_observacao"] = ""
if votacao.tip_resultado_votacao: if votacao.tip_resultado_votacao:
resultado = context.zsql.tipo_resultado_votacao_obter_zsql(tip_resultado_votacao=votacao.tip_resultado_votacao, ind_excluido=0) resultado = context.zsql.tipo_resultado_votacao_obter_zsql(
tip_resultado_votacao=votacao.tip_resultado_votacao, ind_excluido=0)
for i in resultado: for i in resultado:
dic_votacao["nom_resultado"] = i.nom_resultado dic_votacao["nom_resultado"] = i.nom_resultado
if votacao.votacao_observacao: if votacao.votacao_observacao:
dic_votacao["votacao_observacao"] = votacao.votacao_observacao dic_votacao[
"votacao_observacao"] = votacao.votacao_observacao
else: else:
dic_votacao["nom_resultado"] = "Matéria não votada" dic_votacao["nom_resultado"] = "Matéria não votada"
dic_votacao["votacao_observacao"] = "Vazio" dic_votacao["votacao_observacao"] = "Vazio"
@ -207,12 +246,14 @@ if context.REQUEST['data']!='':
dic_oradores['sgl_partido'] = parlamentar.sgl_partido dic_oradores['sgl_partido'] = parlamentar.sgl_partido
lst_oradores.append(dic_oradores) lst_oradores.append(dic_oradores)
# obtém as propriedades da casa legislativa para montar o cabeçalho e o rodapé da página # obtém as propriedades da casa legislativa para montar o cabeçalho e o
# rodapé da página
cabecalho = {} cabecalho = {}
# tenta buscar o logotipo da casa LOGO_CASA # tenta buscar o logotipo da casa LOGO_CASA
if hasattr(context.sapl_documentos.props_sapl, 'logo_casa.gif'): if hasattr(context.sapl_documentos.props_sapl, 'logo_casa.gif'):
imagem = context.sapl_documentos.props_sapl['logo_casa.gif'].absolute_url() imagem = context.sapl_documentos.props_sapl[
'logo_casa.gif'].absolute_url()
else: else:
imagem = context.imagens.absolute_url() + "/brasao_transp.gif" imagem = context.imagens.absolute_url() + "/brasao_transp.gif"
@ -221,7 +262,8 @@ if context.REQUEST['data']!='':
aux = context.sapl_documentos.props_sapl.propertyItems() aux = context.sapl_documentos.props_sapl.propertyItems()
for item in aux: for item in aux:
casa[item[0]] = item[1] casa[item[0]] = item[1]
localidade=context.zsql.localidade_obter_zsql(cod_localidade=casa["cod_localidade"]) localidade = context.zsql.localidade_obter_zsql(
cod_localidade=casa["cod_localidade"])
data_emissao = DateTime().strftime("%d/%m/%Y") data_emissao = DateTime().strftime("%d/%m/%Y")
rodape = casa rodape = casa
rodape['data_emissao'] = data_emissao rodape['data_emissao'] = data_emissao
@ -234,7 +276,8 @@ if context.REQUEST['data']!='':
# return lst_votacao # return lst_votacao
sessao = session.id sessao = session.id
caminho = context.pdf_sessao_plenaria_gerar(rodape, sessao, imagem, inf_basicas_dic, lst_mesa, lst_presenca_sessao, lst_expedientes, lst_expediente_materia, lst_oradores_expediente, lst_presenca_ordem_dia, lst_votacao, lst_oradores) caminho = context.pdf_sessao_plenaria_gerar(rodape, sessao, imagem, inf_basicas_dic, lst_mesa, lst_presenca_sessao,
lst_expedientes, lst_expediente_materia, lst_oradores_expediente, lst_presenca_ordem_dia, lst_votacao, lst_oradores)
if caminho == 'aviso': if caminho == 'aviso':
return response.redirect('mensagem_emitir_proc') return response.redirect('mensagem_emitir_proc')
else: else:

23
sapl/relatorios/views.py

@ -15,11 +15,10 @@ from sapl.protocoloadm.models import (DocumentoAdministrativo, Protocolo,
TramitacaoAdministrativo) TramitacaoAdministrativo)
from sapl.sessao.models import (ExpedienteMateria, ExpedienteSessao, from sapl.sessao.models import (ExpedienteMateria, ExpedienteSessao,
IntegranteMesa, Orador, OradorExpediente, IntegranteMesa, Orador, OradorExpediente,
OrdemDia, PresencaOrdemDia, RegistroVotacao, OrdemDia, PresencaOrdemDia, SessaoPlenaria,
SessaoPlenaria, SessaoPlenariaPresenca, SessaoPlenariaPresenca)
TipoExpediente)
from sapl.settings import STATIC_ROOT from sapl.settings import STATIC_ROOT
from sapl.utils import UF from sapl.utils import UF, filiacao_data
from .templates import (pdf_capa_processo_gerar, from .templates import (pdf_capa_processo_gerar,
pdf_documento_administrativo_gerar, pdf_espelho_gerar, pdf_documento_administrativo_gerar, pdf_espelho_gerar,
@ -510,12 +509,8 @@ def get_sessao_plenaria(sessao, casa):
for parlamentar in [p.parlamentar for p in presenca]: for parlamentar in [p.parlamentar for p in presenca]:
dic_presenca = {} dic_presenca = {}
dic_presenca["nom_parlamentar"] = parlamentar.nome_parlamentar dic_presenca["nom_parlamentar"] = parlamentar.nome_parlamentar
partido = Filiacao.objects.filter( partido_sigla = filiacao_data(parlamentar, sessao.data_inicio)
parlamentar=parlamentar).first()
if partido:
partido_sigla = partido.partido.sigla
else:
partido_sigla = ''
dic_presenca['sgl_partido'] = partido_sigla dic_presenca['sgl_partido'] = partido_sigla
lst_presenca_sessao.append(dic_presenca) lst_presenca_sessao.append(dic_presenca)
@ -651,12 +646,8 @@ def get_sessao_plenaria(sessao, casa):
dic_presenca_ordem_dia = {} dic_presenca_ordem_dia = {}
dic_presenca_ordem_dia['nom_parlamentar'] = ( dic_presenca_ordem_dia['nom_parlamentar'] = (
parlamentar.nome_parlamentar) parlamentar.nome_parlamentar)
partido_sigla = Filiacao.objects.filter( sigla = filiacao_data(parlamentar, sessao.data_inicio)
parlamentar=parlamentar).first()
if not partido_sigla:
sigla = ''
else:
sigla = partido_sigla.partido.sigla
dic_presenca_ordem_dia['sgl_partido'] = sigla dic_presenca_ordem_dia['sgl_partido'] = sigla
lst_presenca_ordem_dia.append(dic_presenca_ordem_dia) lst_presenca_ordem_dia.append(dic_presenca_ordem_dia)

5
sapl/rules/map_rules.py

@ -53,6 +53,7 @@ __listdetailchange__ = [RP_LIST, RP_DETAIL, RP_CHANGE]
rules_group_administrativo = { rules_group_administrativo = {
'group': SAPL_GROUP_ADMINISTRATIVO, 'group': SAPL_GROUP_ADMINISTRATIVO,
'rules': [ 'rules': [
(materia.MateriaLegislativa, ['can_access_impressos']),
(protocoloadm.DocumentoAdministrativo, __base__), (protocoloadm.DocumentoAdministrativo, __base__),
(protocoloadm.DocumentoAcessorioAdministrativo, __base__), (protocoloadm.DocumentoAcessorioAdministrativo, __base__),
(protocoloadm.TramitacaoAdministrativo, __base__), (protocoloadm.TramitacaoAdministrativo, __base__),
@ -69,6 +70,7 @@ rules_group_protocolo = {
(protocoloadm.DocumentoAcessorioAdministrativo, __listdetailchange__), (protocoloadm.DocumentoAcessorioAdministrativo, __listdetailchange__),
(materia.MateriaLegislativa, __listdetailchange__), (materia.MateriaLegislativa, __listdetailchange__),
(materia.MateriaLegislativa, ['can_access_impressos']),
(materia.DocumentoAcessorio, __listdetailchange__), (materia.DocumentoAcessorio, __listdetailchange__),
(materia.Anexada, __base__), (materia.Anexada, __base__),
(materia.Autoria, __base__), (materia.Autoria, __base__),
@ -98,7 +100,7 @@ rules_group_materia = {
(materia.DespachoInicial, __base__), (materia.DespachoInicial, __base__),
(materia.DocumentoAcessorio, __base__), (materia.DocumentoAcessorio, __base__),
(materia.MateriaLegislativa, __base__), (materia.MateriaLegislativa, __base__ + ['can_access_impressos']),
(materia.Numeracao, __base__), (materia.Numeracao, __base__),
(materia.Tramitacao, __base__), (materia.Tramitacao, __base__),
(norma.LegislacaoCitada, __base__), (norma.LegislacaoCitada, __base__),
@ -209,6 +211,7 @@ rules_group_geral = {
(materia.AssuntoMateria, __base__), # não há implementação (materia.AssuntoMateria, __base__), # não há implementação
(materia.MateriaAssunto, __base__), # não há implementação (materia.MateriaAssunto, __base__), # não há implementação
(materia.MateriaLegislativa, ['can_access_impressos']),
(materia.TipoProposicao, __base__), (materia.TipoProposicao, __base__),
(materia.TipoMateriaLegislativa, __base__), (materia.TipoMateriaLegislativa, __base__),
(materia.RegimeTramitacao, __base__), (materia.RegimeTramitacao, __base__),

70
sapl/sessao/forms.py

@ -10,7 +10,8 @@ from django.utils.translation import ugettext_lazy as _
from sapl.crispy_layout_mixin import form_actions, to_row from sapl.crispy_layout_mixin import form_actions, to_row
from sapl.materia.forms import MateriaLegislativaFilterSet from sapl.materia.forms import MateriaLegislativaFilterSet
from sapl.materia.models import MateriaLegislativa, TipoMateriaLegislativa from sapl.materia.models import (MateriaLegislativa, StatusTramitacao,
TipoMateriaLegislativa)
from sapl.parlamentares.models import Parlamentar from sapl.parlamentares.models import Parlamentar
from sapl.utils import (RANGE_DIAS_MES, RANGE_MESES, from sapl.utils import (RANGE_DIAS_MES, RANGE_MESES,
MateriaPesquisaOrderingFilter, autor_label, MateriaPesquisaOrderingFilter, autor_label,
@ -52,6 +53,48 @@ ORDENACAO_RESUMO = [('cont_mult', 'Conteúdo Multimídia'),
('oradores_expli', 'Oradores das Explicações Pessoais')] ('oradores_expli', 'Oradores das Explicações Pessoais')]
class SessaoPlenariaForm(ModelForm):
class Meta:
model = SessaoPlenaria
exclude = ['cod_andamento_sessao']
def clean(self):
super(SessaoPlenariaForm, self).clean()
if not self.is_valid():
return self.cleaned_data
instance = self.instance
num = self.cleaned_data['numero']
sl = self.cleaned_data['sessao_legislativa']
leg = self.cleaned_data['legislatura']
tipo = self.cleaned_data['tipo']
error = ValidationError(
"Número de Sessão Plenária já existente "
"para a Legislatura, Sessão Legislativa e Tipo informados. "
"Favor escolher um número distinto.")
sessoes = SessaoPlenaria.objects.filter(numero=num,
sessao_legislativa=sl,
legislatura=leg,
tipo=tipo).\
values_list('id', flat=True)
qtd_sessoes = len(sessoes)
if qtd_sessoes > 0:
if instance.pk: # update
if instance.pk not in sessoes or qtd_sessoes > 1:
raise error
else: # create
raise error
return self.cleaned_data
class BancadaForm(ModelForm): class BancadaForm(ModelForm):
class Meta: class Meta:
@ -255,10 +298,15 @@ class SessaoPlenariaFilterSet(django_filters.FilterSet):
class AdicionarVariasMateriasFilterSet(MateriaLegislativaFilterSet): class AdicionarVariasMateriasFilterSet(MateriaLegislativaFilterSet):
o = MateriaPesquisaOrderingFilter() o = MateriaPesquisaOrderingFilter()
tramitacao__status = django_filters.ModelChoiceFilter(
required=True,
queryset=StatusTramitacao.objects.all(),
label=_('Status da Matéria'))
class Meta: class Meta:
model = MateriaLegislativa model = MateriaLegislativa
fields = ['numero', fields = ['tramitacao__status',
'numero',
'numero_protocolo', 'numero_protocolo',
'ano', 'ano',
'tipo', 'tipo',
@ -280,15 +328,17 @@ class AdicionarVariasMateriasFilterSet(MateriaLegislativaFilterSet):
self.filters['relatoria__parlamentar_id'].label = 'Relatoria' self.filters['relatoria__parlamentar_id'].label = 'Relatoria'
row1 = to_row( row1 = to_row(
[('tipo', 12)]) [('tramitacao__status', 12)])
row2 = to_row( row2 = to_row(
[('tipo', 12)])
row3 = to_row(
[('numero', 4), [('numero', 4),
('ano', 4), ('ano', 4),
('numero_protocolo', 4)]) ('numero_protocolo', 4)])
row3 = to_row( row4 = to_row(
[('data_apresentacao', 6), [('data_apresentacao', 6),
('data_publicacao', 6)]) ('data_publicacao', 6)])
row4 = to_row( row5 = to_row(
[('autoria__autor', 0), [('autoria__autor', 0),
(Button('pesquisar', (Button('pesquisar',
'Pesquisar Autor', 'Pesquisar Autor',
@ -296,17 +346,17 @@ class AdicionarVariasMateriasFilterSet(MateriaLegislativaFilterSet):
(Button('limpar', (Button('limpar',
'limpar Autor', 'limpar Autor',
css_class='btn btn-primary btn-sm'), 10)]) css_class='btn btn-primary btn-sm'), 10)])
row5 = to_row( row6 = to_row(
[('autoria__autor__tipo', 6), [('autoria__autor__tipo', 6),
# ('autoria__autor__partido', 6) # ('autoria__autor__partido', 6)
]) ])
row6 = to_row( row7 = to_row(
[('relatoria__parlamentar_id', 6), [('relatoria__parlamentar_id', 6),
('local_origem_externa', 6)]) ('local_origem_externa', 6)])
row7 = to_row( row8 = to_row(
[('em_tramitacao', 6), [('em_tramitacao', 6),
('o', 6)]) ('o', 6)])
row8 = to_row( row9 = to_row(
[('ementa', 12)]) [('ementa', 12)])
self.form.helper = FormHelper() self.form.helper = FormHelper()
@ -316,7 +366,7 @@ class AdicionarVariasMateriasFilterSet(MateriaLegislativaFilterSet):
row1, row2, row3, row1, row2, row3,
HTML(autor_label), HTML(autor_label),
HTML(autor_modal), HTML(autor_modal),
row4, row5, row6, row7, row8, row4, row5, row6, row7, row8, row9,
form_actions(save_label='Pesquisar')) form_actions(save_label='Pesquisar'))
) )

25
sapl/sessao/migrations/0010_auto_20170810_1033.py

@ -0,0 +1,25 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.11 on 2017-08-10 10:33
from __future__ import unicode_literals
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('sessao', '0009_auto_20170619_1441'),
]
operations = [
migrations.AddField(
model_name='registrovotacao',
name='data_hora_atualizacao',
field=models.DateTimeField(auto_now=True, null=True, verbose_name='Data'),
),
migrations.AddField(
model_name='registrovotacao',
name='data_hora_criacao',
field=models.DateTimeField(auto_now_add=True, null=True, verbose_name='Data Criação'),
),
]

25
sapl/sessao/migrations/0010_auto_20170814_1804.py

@ -0,0 +1,25 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.13 on 2017-08-14 18:04
from __future__ import unicode_literals
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('sessao', '0009_auto_20170619_1441'),
]
operations = [
migrations.AddField(
model_name='tiporesultadovotacao',
name='natureza',
field=models.CharField(choices=[('A', 'Aprovado'), ('R', 'Rejeitado')], max_length=100, null=True, verbose_name='Natureza do Tipo'),
),
migrations.AlterField(
model_name='tiporesultadovotacao',
name='nome',
field=models.CharField(max_length=100, verbose_name='Nome do Tipo'),
),
]

20
sapl/sessao/migrations/0011_auto_20170814_1409.py

@ -0,0 +1,20 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.13 on 2017-08-14 14:09
from __future__ import unicode_literals
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('sessao', '0010_auto_20170810_1033'),
]
operations = [
migrations.AlterField(
model_name='bloco',
name='partidos',
field=models.ManyToManyField(blank=True, to='parlamentares.Partido', verbose_name='Partidos'),
),
]

20
sapl/sessao/migrations/0011_auto_20170814_1849.py

@ -0,0 +1,20 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.13 on 2017-08-14 18:49
from __future__ import unicode_literals
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('sessao', '0010_auto_20170814_1804'),
]
operations = [
migrations.AlterField(
model_name='tiporesultadovotacao',
name='natureza',
field=models.CharField(blank=True, choices=[('A', 'Aprovado'), ('R', 'Rejeitado')], max_length=100, null=True, verbose_name='Natureza do Tipo'),
),
]

23
sapl/sessao/migrations/0012_auto_20170814_1615.py

@ -0,0 +1,23 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.11 on 2017-08-14 16:15
from __future__ import unicode_literals
from django.db import migrations
class Migration(migrations.Migration):
dependencies = [
('sessao', '0011_auto_20170814_1409'),
]
operations = [
migrations.AlterModelOptions(
name='bancada',
options={'ordering': ('-legislatura__numero',), 'verbose_name': 'Bancada Parlamentar', 'verbose_name_plural': 'Bancadas Parlamentares'},
),
migrations.AlterModelOptions(
name='bloco',
options={'verbose_name': 'Bloco Parlamentar', 'verbose_name_plural': 'Blocos Parlamentares'},
),
]

21
sapl/sessao/migrations/0012_auto_20170815_1244.py

@ -0,0 +1,21 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.13 on 2017-08-15 12:44
from __future__ import unicode_literals
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('sessao', '0011_auto_20170814_1849'),
]
operations = [
migrations.AlterField(
model_name='tiporesultadovotacao',
name='natureza',
field=models.CharField(blank=True, choices=[('A', 'Aprovado'), ('R', 'Rejeitado')], default='', max_length=100, verbose_name='Natureza do Tipo'),
preserve_default=False,
),
]

16
sapl/sessao/migrations/0013_merge.py

@ -0,0 +1,16 @@
# -*- coding: utf-8 -*-
# Generated by Django 1.9.13 on 2017-08-15 13:42
from __future__ import unicode_literals
from django.db import migrations
class Migration(migrations.Migration):
dependencies = [
('sessao', '0012_auto_20170814_1615'),
('sessao', '0012_auto_20170815_1244'),
]
operations = [
]

27
sapl/sessao/models.py

@ -59,8 +59,8 @@ class Bancada(models.Model):
)) ))
class Meta: class Meta:
verbose_name = _('Bancada') verbose_name = _('Bancada Parlamentar')
verbose_name_plural = _('Bancadas') verbose_name_plural = _('Bancadas Parlamentares')
ordering = ('-legislatura__numero', ) ordering = ('-legislatura__numero', )
def __str__(self): def __str__(self):
@ -368,7 +368,12 @@ class PresencaOrdemDia(models.Model): # OrdemDiaPresenca
@reversion.register() @reversion.register()
class TipoResultadoVotacao(models.Model): class TipoResultadoVotacao(models.Model):
nome = models.CharField(max_length=100, verbose_name=_('Tipo')) nome = models.CharField(max_length=100, verbose_name=_('Nome do Tipo'))
natureza = models.CharField(max_length=100,
blank=True,
choices=(('A', 'Aprovado'),
('R', 'Rejeitado')),
verbose_name=_('Natureza do Tipo'))
class Meta: class Meta:
verbose_name = _('Tipo de Resultado de Votação') verbose_name = _('Tipo de Resultado de Votação')
@ -400,6 +405,16 @@ class RegistroVotacao(models.Model):
observacao = models.TextField( observacao = models.TextField(
blank=True, verbose_name=_('Observações')) blank=True, verbose_name=_('Observações'))
data_hora_criacao = models.DateTimeField(
blank=True, null=True,
auto_now_add=True,
verbose_name=_('Data Criação'))
data_hora_atualizacao = models.DateTimeField(
blank=True, null=True,
auto_now=True,
verbose_name=_('Data'))
class Meta: class Meta:
verbose_name = _('Votação') verbose_name = _('Votação')
verbose_name_plural = _('Votações') verbose_name_plural = _('Votações')
@ -477,7 +492,7 @@ class Bloco(models.Model):
nome = models.CharField( nome = models.CharField(
max_length=80, verbose_name=_('Nome do Bloco')) max_length=80, verbose_name=_('Nome do Bloco'))
partidos = models.ManyToManyField( partidos = models.ManyToManyField(
Partido, blank=True, verbose_name=_('Bancadas')) Partido, blank=True, verbose_name=_('Partidos'))
data_criacao = models.DateField( data_criacao = models.DateField(
blank=True, null=True, verbose_name=_('Data Criação')) blank=True, null=True, verbose_name=_('Data Criação'))
data_extincao = models.DateField( data_extincao = models.DateField(
@ -496,8 +511,8 @@ class Bloco(models.Model):
)) ))
class Meta: class Meta:
verbose_name = _('Bloco') verbose_name = _('Bloco Parlamentar')
verbose_name_plural = _('Blocos') verbose_name_plural = _('Blocos Parlamentares')
def __str__(self): def __str__(self):
return self.nome return self.nome

1
sapl/sessao/serializers.py

@ -4,6 +4,7 @@ from .models import SessaoPlenaria
class SessaoPlenariaSerializer(serializers.Serializer): class SessaoPlenariaSerializer(serializers.Serializer):
class Meta: class Meta:
model = SessaoPlenaria model = SessaoPlenaria
fields = ('tipo', fields = ('tipo',

68
sapl/sessao/tests/test_sessao.py

@ -0,0 +1,68 @@
import pytest
from django.utils.translation import ugettext_lazy as _
from model_mommy import mommy
from sapl.parlamentares.models import Legislatura, SessaoLegislativa
from sapl.sessao import forms
from sapl.sessao.models import SessaoPlenaria, TipoSessaoPlenaria
def test_valida_campos_obrigatorios_sessao_plenaria_form():
form = forms.SessaoPlenariaForm(data={})
assert not form.is_valid()
errors = form.errors
assert errors['legislatura'] == ['Este campo é obrigatório.']
assert errors['sessao_legislativa'] == ['Este campo é obrigatório.']
assert errors['tipo'] == ['Este campo é obrigatório.']
assert errors['numero'] == ['Este campo é obrigatório.']
assert errors['data_inicio'] == ['Este campo é obrigatório.']
assert errors['hora_inicio'] == ['Este campo é obrigatório.']
assert len(errors) == 6
@pytest.mark.django_db(transaction=False)
def test_sessao_plenaria_form_valido():
legislatura = mommy.make(Legislatura)
sessao = mommy.make(SessaoLegislativa)
tipo = mommy.make(TipoSessaoPlenaria)
form = forms.SessaoPlenariaForm(data={'legislatura': str(legislatura.pk),
'numero': '1',
'tipo': str(tipo.pk),
'sessao_legislativa': str(sessao.pk),
'data_inicio': '10/11/2017',
'hora_inicio': '10:10'
})
assert form.is_valid()
@pytest.mark.django_db(transaction=False)
def test_numero_duplicado_sessao_plenaria_form():
legislatura = mommy.make(Legislatura)
sessao = mommy.make(SessaoLegislativa)
tipo = mommy.make(TipoSessaoPlenaria)
sessao_plenaria = mommy.make(SessaoPlenaria,
legislatura=legislatura,
sessao_legislativa=sessao,
tipo=tipo,
numero=1)
form = forms.SessaoPlenariaForm(data={'legislatura': str(legislatura.pk),
'numero': '1',
'tipo': str(tipo.pk),
'sessao_legislativa': str(sessao.pk),
'data_inicio': '10/11/2017',
'hora_inicio': '10:10'
})
assert not form.is_valid()
assert form.errors['__all__'] == ["Número de Sessão Plenária já existente "
"para a Legislatura, Sessão Legislativa "
"e Tipo informados. Favor escolher um "
"número distinto."]

106
sapl/sessao/views.py

@ -25,13 +25,13 @@ from sapl.base.models import AppConfig as AppsAppConfig
from sapl.crud.base import (RP_DETAIL, RP_LIST, Crud, CrudAux, from sapl.crud.base import (RP_DETAIL, RP_LIST, Crud, CrudAux,
MasterDetailCrud, MasterDetailCrud,
PermissionRequiredForAppCrudMixin, make_pagination) PermissionRequiredForAppCrudMixin, make_pagination)
from sapl.materia.forms import pega_ultima_tramitacao from sapl.materia.forms import filtra_tramitacao_status
from sapl.materia.models import (Autoria, DocumentoAcessorio, from sapl.materia.models import (Autoria, DocumentoAcessorio,
TipoMateriaLegislativa, Tramitacao) TipoMateriaLegislativa, Tramitacao)
from sapl.materia.views import MateriaLegislativaPesquisaView from sapl.materia.views import MateriaLegislativaPesquisaView
from sapl.norma.models import NormaJuridica from sapl.norma.models import NormaJuridica
from sapl.parlamentares.models import (Filiacao, Legislatura, Parlamentar, from sapl.parlamentares.models import (Filiacao, Legislatura, Mandato,
SessaoLegislativa, Mandato) Parlamentar, SessaoLegislativa)
from sapl.sessao.apps import AppConfig from sapl.sessao.apps import AppConfig
from sapl.sessao.forms import ExpedienteMateriaForm, OrdemDiaForm from sapl.sessao.forms import ExpedienteMateriaForm, OrdemDiaForm
@ -39,14 +39,14 @@ from .forms import (AdicionarVariasMateriasFilterSet, ExpedienteForm,
ListMateriaForm, MesaForm, OradorExpedienteForm, ListMateriaForm, MesaForm, OradorExpedienteForm,
OradorForm, PautaSessaoFilterSet, PresencaForm, OradorForm, PautaSessaoFilterSet, PresencaForm,
ResumoOrdenacaoForm, SessaoPlenariaFilterSet, ResumoOrdenacaoForm, SessaoPlenariaFilterSet,
VotacaoEditForm, VotacaoForm, VotacaoNominalForm) SessaoPlenariaForm, VotacaoEditForm, VotacaoForm,
VotacaoNominalForm)
from .models import (Bancada, Bloco, CargoBancada, CargoMesa, from .models import (Bancada, Bloco, CargoBancada, CargoMesa,
ExpedienteMateria, ExpedienteSessao, IntegranteMesa, ExpedienteMateria, ExpedienteSessao, IntegranteMesa,
MateriaLegislativa, Orador, OradorExpediente, OrdemDia, MateriaLegislativa, Orador, OradorExpediente, OrdemDia,
PresencaOrdemDia, RegistroVotacao, ResumoOrdenacao, PresencaOrdemDia, RegistroVotacao, ResumoOrdenacao,
SessaoPlenaria, SessaoPlenariaPresenca, TipoExpediente, SessaoPlenaria, SessaoPlenariaPresenca, TipoExpediente,
TipoResultadoVotacao, TipoSessaoPlenaria, TipoResultadoVotacao, TipoSessaoPlenaria, VotoParlamentar)
VotoParlamentar)
TipoSessaoCrud = CrudAux.build(TipoSessaoPlenaria, 'tipo_sessao_plenaria') TipoSessaoCrud = CrudAux.build(TipoSessaoPlenaria, 'tipo_sessao_plenaria')
TipoExpedienteCrud = CrudAux.build(TipoExpediente, 'tipo_expediente') TipoExpedienteCrud = CrudAux.build(TipoExpediente, 'tipo_expediente')
@ -81,6 +81,7 @@ def reordernar_materias_ordem(request, pk):
return HttpResponseRedirect( return HttpResponseRedirect(
reverse('sapl.sessao:ordemdia_list', kwargs={'pk': pk})) reverse('sapl.sessao:ordemdia_list', kwargs={'pk': pk}))
def verifica_presenca(request, model, spk): def verifica_presenca(request, model, spk):
if not model.objects.filter(sessao_plenaria_id=spk).exists(): if not model.objects.filter(sessao_plenaria_id=spk).exists():
msg = _('Votação não pode ser aberta sem presenças') msg = _('Votação não pode ser aberta sem presenças')
@ -162,7 +163,7 @@ class MateriaOrdemDiaCrud(MasterDetailCrud):
public = [RP_LIST, RP_DETAIL] public = [RP_LIST, RP_DETAIL]
class BaseMixin(MasterDetailCrud.BaseMixin): class BaseMixin(MasterDetailCrud.BaseMixin):
list_field_names = ['numero_ordem', 'materia', 'observacao', list_field_names = ['numero_ordem', 'materia', 'materia__ementa',
'resultado'] 'resultado']
class CreateView(MasterDetailCrud.CreateView): class CreateView(MasterDetailCrud.CreateView):
@ -172,8 +173,10 @@ class MateriaOrdemDiaCrud(MasterDetailCrud):
self.initial['data_ordem'] = SessaoPlenaria.objects.get( self.initial['data_ordem'] = SessaoPlenaria.objects.get(
pk=self.kwargs['pk']).data_inicio.strftime('%d/%m/%Y') pk=self.kwargs['pk']).data_inicio.strftime('%d/%m/%Y')
max_numero_ordem = OrdemDia.objects.filter( max_numero_ordem = OrdemDia.objects.filter(
sessao_plenaria=self.kwargs['pk']).aggregate(Max('numero_ordem'))['numero_ordem__max'] sessao_plenaria=self.kwargs['pk']).aggregate(
self.initial['numero_ordem'] = (max_numero_ordem if max_numero_ordem else 0) + 1 Max('numero_ordem'))['numero_ordem__max']
self.initial['numero_ordem'] = (
max_numero_ordem if max_numero_ordem else 0) + 1
return self.initial return self.initial
def get_success_url(self): def get_success_url(self):
@ -196,6 +199,7 @@ class MateriaOrdemDiaCrud(MasterDetailCrud):
return 'OrdemDiaDetail' return 'OrdemDiaDetail'
class ListView(MasterDetailCrud.ListView): class ListView(MasterDetailCrud.ListView):
paginate_by = None
ordering = ['numero_ordem', 'materia', 'resultado'] ordering = ['numero_ordem', 'materia', 'resultado']
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
@ -313,9 +317,10 @@ class ExpedienteMateriaCrud(MasterDetailCrud):
class BaseMixin(MasterDetailCrud.BaseMixin): class BaseMixin(MasterDetailCrud.BaseMixin):
list_field_names = ['numero_ordem', 'materia', list_field_names = ['numero_ordem', 'materia',
'observacao', 'resultado'] 'materia__ementa', 'resultado']
class ListView(MasterDetailCrud.ListView): class ListView(MasterDetailCrud.ListView):
paginate_by = None
ordering = ['numero_ordem', 'materia', 'resultado'] ordering = ['numero_ordem', 'materia', 'resultado']
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
@ -422,8 +427,10 @@ class ExpedienteMateriaCrud(MasterDetailCrud):
self.initial['data_ordem'] = SessaoPlenaria.objects.get( self.initial['data_ordem'] = SessaoPlenaria.objects.get(
pk=self.kwargs['pk']).data_inicio.strftime('%d/%m/%Y') pk=self.kwargs['pk']).data_inicio.strftime('%d/%m/%Y')
max_numero_ordem = ExpedienteMateria.objects.filter( max_numero_ordem = ExpedienteMateria.objects.filter(
sessao_plenaria=self.kwargs['pk']).aggregate(Max('numero_ordem'))['numero_ordem__max'] sessao_plenaria=self.kwargs['pk']).aggregate(
self.initial['numero_ordem'] = (max_numero_ordem if max_numero_ordem else 0) + 1 Max('numero_ordem'))['numero_ordem__max']
self.initial['numero_ordem'] = (
max_numero_ordem if max_numero_ordem else 0) + 1
return self.initial return self.initial
def get_success_url(self): def get_success_url(self):
@ -543,11 +550,15 @@ class SessaoCrud(Crud):
class UpdateView(Crud.UpdateView): class UpdateView(Crud.UpdateView):
form_class = SessaoPlenariaForm
def get_initial(self): def get_initial(self):
return {'sessao_legislativa': self.object.sessao_legislativa} return {'sessao_legislativa': self.object.sessao_legislativa}
class CreateView(Crud.CreateView): class CreateView(Crud.CreateView):
form_class = SessaoPlenariaForm
@property @property
def cancel_url(self): def cancel_url(self):
return self.search_url return self.search_url
@ -719,8 +730,6 @@ class PresencaOrdemDiaView(FormMixin, PresencaMixin, DetailView):
self.object = self.get_object() self.object = self.get_object()
form = self.get_form() form = self.get_form()
pk = kwargs['pk']
if form.is_valid(): if form.is_valid():
# Pegar os presentes salvos no banco # Pegar os presentes salvos no banco
presentes_banco = PresencaOrdemDia.objects.filter( presentes_banco = PresencaOrdemDia.objects.filter(
@ -769,7 +778,7 @@ class ListMateriaOrdemDiaView(FormMixin, DetailView):
materias_ordem = [] materias_ordem = []
for o in ordem: for o in ordem:
ementa = o.observacao ementa = o.materia.ementa
titulo = o.materia titulo = o.materia
numero = o.numero_ordem numero = o.numero_ordem
@ -829,7 +838,7 @@ class ListMateriaOrdemDiaView(FormMixin, DetailView):
materias_ordem = [] materias_ordem = []
for o in ordem: for o in ordem:
ementa = o.observacao ementa = o.materia.ementa
titulo = o.materia titulo = o.materia
numero = o.numero_ordem numero = o.numero_ordem
@ -1159,7 +1168,7 @@ class ResumoView(DetailView):
materias_expediente = [] materias_expediente = []
for m in materias: for m in materias:
ementa = m.observacao ementa = m.materia.ementa
titulo = m.materia titulo = m.materia
numero = m.numero_ordem numero = m.numero_ordem
@ -1219,7 +1228,7 @@ class ResumoView(DetailView):
sessao_plenaria_id=self.object.id) sessao_plenaria_id=self.object.id)
materias_ordem = [] materias_ordem = []
for o in ordem: for o in ordem:
ementa = o.observacao ementa = o.materia.ementa
titulo = o.materia titulo = o.materia
numero = o.numero_ordem numero = o.numero_ordem
@ -1250,7 +1259,8 @@ class ResumoView(DetailView):
# ===================================================================== # =====================================================================
# Oradores nas Explicações Pessoais # Oradores nas Explicações Pessoais
oradores_explicacoes = [] oradores_explicacoes = []
for orador in Orador.objects.filter(sessao_plenaria_id=self.object.id): for orador in Orador.objects.filter(
sessao_plenaria_id=self.object.id).order_by('numero_ordem'):
for parlamentar in Parlamentar.objects.filter( for parlamentar in Parlamentar.objects.filter(
id=orador.parlamentar.id): id=orador.parlamentar.id):
partido_sigla = Filiacao.objects.filter( partido_sigla = Filiacao.objects.filter(
@ -1429,7 +1439,7 @@ class VotacaoEditView(SessaoPermissionMixin):
ordem = OrdemDia.objects.get(id=ordem_id) ordem = OrdemDia.objects.get(id=ordem_id)
materia = {'materia': ordem.materia, 'ementa': ordem.observacao} materia = {'materia': ordem.materia, 'ementa': ordem.materia.ementa}
context.update({'materia': materia}) context.update({'materia': materia})
votacao = RegistroVotacao.objects.filter( votacao = RegistroVotacao.objects.filter(
@ -1458,9 +1468,9 @@ class VotacaoEditView(SessaoPermissionMixin):
class VotacaoView(SessaoPermissionMixin): class VotacaoView(SessaoPermissionMixin):
''' """
Votação Simbólica e Secreta Votação Simbólica e Secreta
''' """
template_name = 'sessao/votacao/votacao.html' template_name = 'sessao/votacao/votacao.html'
form_class = VotacaoForm form_class = VotacaoForm
@ -1483,7 +1493,7 @@ class VotacaoView(SessaoPermissionMixin):
qtde_presentes = PresencaOrdemDia.objects.filter( qtde_presentes = PresencaOrdemDia.objects.filter(
sessao_plenaria_id=self.object.id).count() sessao_plenaria_id=self.object.id).count()
materia = {'materia': ordem.materia, 'ementa': ordem.observacao} materia = {'materia': ordem.materia, 'ementa': ordem.materia.ementa}
context.update({'votacao_titulo': titulo, context.update({'votacao_titulo': titulo,
'materia': materia, 'materia': materia,
'total_presentes': qtde_presentes}) 'total_presentes': qtde_presentes})
@ -1509,7 +1519,7 @@ class VotacaoView(SessaoPermissionMixin):
qtde_presentes = PresencaOrdemDia.objects.filter( qtde_presentes = PresencaOrdemDia.objects.filter(
sessao_plenaria_id=self.object.id).count() sessao_plenaria_id=self.object.id).count()
materia = {'materia': ordem.materia, 'ementa': ordem.observacao} materia = {'materia': ordem.materia, 'ementa': ordem.materia.ementa}
context.update({'votacao_titulo': titulo, context.update({'votacao_titulo': titulo,
'materia': materia, 'materia': materia,
'total_presentes': qtde_presentes}) 'total_presentes': qtde_presentes})
@ -1646,7 +1656,7 @@ class VotacaoNominalAbstract(SessaoPermissionMixin):
materia = {'materia': materia_votacao.materia, materia = {'materia': materia_votacao.materia,
'ementa': sub( 'ementa': sub(
'&nbsp;', ' ', strip_tags( '&nbsp;', ' ', strip_tags(
materia_votacao.observacao))} materia_votacao.materia.ementa))}
context = {'materia': materia, 'object': self.get_object(), context = {'materia': materia, 'object': self.get_object(),
'parlamentares': self.get_parlamentares(presentes), 'parlamentares': self.get_parlamentares(presentes),
'tipos': self.get_tipos_votacao(), 'tipos': self.get_tipos_votacao(),
@ -1830,7 +1840,7 @@ class VotacaoNominalEditAbstract(SessaoPermissionMixin):
raise Http404() raise Http404()
materia = ordem.materia materia = ordem.materia
observacao = ordem.observacao ementa = ordem.materia.ementa
elif self.expediente: elif self.expediente:
expediente_id = kwargs['oid'] expediente_id = kwargs['oid']
@ -1843,7 +1853,7 @@ class VotacaoNominalEditAbstract(SessaoPermissionMixin):
raise Http404() raise Http404()
materia = expediente.materia materia = expediente.materia
observacao = expediente.observacao ementa = expediente.materia.ementa
votos = VotoParlamentar.objects.filter(votacao_id=votacao.id) votos = VotoParlamentar.objects.filter(votacao_id=votacao.id)
@ -1856,7 +1866,7 @@ class VotacaoNominalEditAbstract(SessaoPermissionMixin):
materia = {'materia': materia, materia = {'materia': materia,
'ementa': sub( 'ementa': sub(
'&nbsp;', ' ', strip_tags(observacao))} '&nbsp;', ' ', strip_tags(ementa))}
context.update({'materia': materia}) context.update({'materia': materia})
votacao_existente = {'observacao': sub( votacao_existente = {'observacao': sub(
@ -1953,7 +1963,7 @@ class VotacaoNominalExpedienteDetailView(DetailView):
materia = {'materia': expediente.materia, materia = {'materia': expediente.materia,
'ementa': sub( 'ementa': sub(
'&nbsp;', ' ', strip_tags(expediente.observacao))} '&nbsp;', ' ', strip_tags(expediente.materia.ementa))}
context.update({'materia': materia}) context.update({'materia': materia})
votacao_existente = {'observacao': sub( votacao_existente = {'observacao': sub(
@ -2004,7 +2014,7 @@ class VotacaoExpedienteView(SessaoPermissionMixin):
sessao_plenaria_id=self.object.id).count() sessao_plenaria_id=self.object.id).count()
materia = {'materia': expediente.materia, materia = {'materia': expediente.materia,
'ementa': expediente.observacao} 'ementa': expediente.materia.ementa}
context.update({'votacao_titulo': titulo, context.update({'votacao_titulo': titulo,
'materia': materia, 'materia': materia,
'total_presentes': qtde_presentes}) 'total_presentes': qtde_presentes})
@ -2031,7 +2041,7 @@ class VotacaoExpedienteView(SessaoPermissionMixin):
sessao_plenaria_id=self.object.id).count() sessao_plenaria_id=self.object.id).count()
materia = {'materia': expediente.materia, materia = {'materia': expediente.materia,
'ementa': expediente.observacao} 'ementa': expediente.materia.ementa}
context.update({'votacao_titulo': titulo, context.update({'votacao_titulo': titulo,
'materia': materia, 'materia': materia,
'total_presentes': qtde_presentes}) 'total_presentes': qtde_presentes})
@ -2134,7 +2144,7 @@ class VotacaoExpedienteEditView(SessaoPermissionMixin):
expediente = ExpedienteMateria.objects.get(id=expediente_id) expediente = ExpedienteMateria.objects.get(id=expediente_id)
materia = {'materia': expediente.materia, materia = {'materia': expediente.materia,
'ementa': expediente.observacao} 'ementa': expediente.materia.ementa}
context.update({'materia': materia}) context.update({'materia': materia})
try: try:
@ -2164,7 +2174,8 @@ class VotacaoExpedienteEditView(SessaoPermissionMixin):
expediente_id = kwargs['oid'] expediente_id = kwargs['oid']
if(int(request.POST['anular_votacao']) == 1): if(int(request.POST['anular_votacao']) == 1):
for r in RegistroVotacao.objects.filter(expediente_id=expediente_id): for r in RegistroVotacao.objects.filter(
expediente_id=expediente_id):
r.delete() r.delete()
expediente = ExpedienteMateria.objects.get( expediente = ExpedienteMateria.objects.get(
@ -2229,7 +2240,7 @@ class PautaSessaoDetailView(DetailView):
materias_expediente = [] materias_expediente = []
for m in materias: for m in materias:
ementa = m.observacao ementa = m.materia.ementa
titulo = m.materia titulo = m.materia
numero = m.numero_ordem numero = m.numero_ordem
situacao = m.materia.tramitacao_set.last().status situacao = m.materia.tramitacao_set.last().status
@ -2286,7 +2297,7 @@ class PautaSessaoDetailView(DetailView):
materias_ordem = [] materias_ordem = []
for o in ordem: for o in ordem:
ementa = o.observacao ementa = o.materia.ementa
titulo = o.materia titulo = o.materia
numero = o.numero_ordem numero = o.numero_ordem
situacao = o.materia.tramitacao_set.last().status situacao = o.materia.tramitacao_set.last().status
@ -2431,14 +2442,6 @@ class PesquisarPautaSessaoView(PesquisarSessaoPlenariaView):
return context return context
def filtra_tramitacao_ordem_dia():
lista = pega_ultima_tramitacao()
return Tramitacao.objects.filter(
id__in=lista,
status__descricao='Ordem do Dia').distinct().values_list(
'materia_id', flat=True)
def retira_materias_ja_adicionadas(id_sessao, model): def retira_materias_ja_adicionadas(id_sessao, model):
lista = model.objects.filter( lista = model.objects.filter(
sessao_plenaria_id=id_sessao) sessao_plenaria_id=id_sessao)
@ -2460,17 +2463,21 @@ class AdicionarVariasMateriasExpediente(PermissionRequiredForAppCrudMixin,
qs = self.get_queryset() qs = self.get_queryset()
lista_ordem_dia = filtra_tramitacao_ordem_dia() if 'tramitacao__status' in self.request.GET:
if self.request.GET['tramitacao__status']:
lista_status = filtra_tramitacao_status(
self.request.GET['tramitacao__status'])
lista_materias_adicionadas = retira_materias_ja_adicionadas( lista_materias_adicionadas = retira_materias_ja_adicionadas(
self.kwargs['pk'], ExpedienteMateria) self.kwargs['pk'], ExpedienteMateria)
qs = qs.filter(id__in=lista_ordem_dia).exclude( qs = qs.filter(id__in=lista_status).exclude(
id__in=lista_materias_adicionadas).distinct() id__in=lista_materias_adicionadas).distinct()
kwargs.update({ kwargs.update({
'queryset': qs, 'queryset': qs,
}) })
return kwargs return kwargs
def get_context_data(self, **kwargs): def get_context_data(self, **kwargs):
@ -2511,6 +2518,7 @@ class AdicionarVariasMateriasExpediente(PermissionRequiredForAppCrudMixin,
expediente = ExpedienteMateria() expediente = ExpedienteMateria()
expediente.sessao_plenaria_id = self.kwargs['pk'] expediente.sessao_plenaria_id = self.kwargs['pk']
expediente.materia_id = materia.id expediente.materia_id = materia.id
# TODO: o campo observacao deve ser uma copia de ML.ementa?
expediente.observacao = MateriaLegislativa.objects.get( expediente.observacao = MateriaLegislativa.objects.get(
pk=materia.id).ementa pk=materia.id).ementa
if lista_materias_expediente: if lista_materias_expediente:
@ -2540,12 +2548,15 @@ class AdicionarVariasMateriasOrdemDia(AdicionarVariasMateriasExpediente):
qs = self.get_queryset() qs = self.get_queryset()
lista_ordem_dia = filtra_tramitacao_ordem_dia() if 'tramitacao__status' in self.request.GET:
if self.request.GET['tramitacao__status']:
lista_status = filtra_tramitacao_status(
self.request.GET['tramitacao__status'])
lista_materias_adicionadas = retira_materias_ja_adicionadas( lista_materias_adicionadas = retira_materias_ja_adicionadas(
self.kwargs['pk'], OrdemDia) self.kwargs['pk'], OrdemDia)
qs = qs.filter(id__in=lista_ordem_dia).exclude( qs = qs.filter(id__in=lista_status).exclude(
id__in=lista_materias_adicionadas).distinct() id__in=lista_materias_adicionadas).distinct()
kwargs.update({ kwargs.update({
@ -2576,6 +2587,7 @@ class AdicionarVariasMateriasOrdemDia(AdicionarVariasMateriasExpediente):
ordem_dia = OrdemDia() ordem_dia = OrdemDia()
ordem_dia.sessao_plenaria_id = self.kwargs['pk'] ordem_dia.sessao_plenaria_id = self.kwargs['pk']
ordem_dia.materia_id = materia.id ordem_dia.materia_id = materia.id
# TODO: o campo observacao deve ser uma copia de ML.ementa?
ordem_dia.observacao = MateriaLegislativa.objects.get( ordem_dia.observacao = MateriaLegislativa.objects.get(
pk=materia.id).ementa pk=materia.id).ementa
if lista_materias_ordem_dia: if lista_materias_ordem_dia:

4
sapl/settings.py

@ -22,7 +22,6 @@ from unipath import Path
from .temp_suppress_crispy_form_warnings import \ from .temp_suppress_crispy_form_warnings import \
SUPRESS_CRISPY_FORM_WARNINGS_LOGGING SUPRESS_CRISPY_FORM_WARNINGS_LOGGING
BASE_DIR = Path(__file__).ancestor(1) BASE_DIR = Path(__file__).ancestor(1)
PROJECT_DIR = Path(__file__).ancestor(2) PROJECT_DIR = Path(__file__).ancestor(2)
@ -39,6 +38,9 @@ ALLOWED_HOSTS = ['*']
LOGIN_REDIRECT_URL = '/' LOGIN_REDIRECT_URL = '/'
LOGIN_URL = '/login/?next=' LOGIN_URL = '/login/?next='
if DEBUG:
EMAIL_BACKEND = 'django.core.mail.backends.console.EmailBackend'
else:
EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend' EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend'

2
sapl/static/js/app.js

@ -8,8 +8,6 @@ function initTinymce(elements, readonly=false) {
menubar: "edit format table tools", menubar: "edit format table tools",
toolbar: "undo redo | styleselect | bold italic | alignleft aligncenter alignright alignjustify | bullist numlist outdent indent", toolbar: "undo redo | styleselect | bold italic | alignleft aligncenter alignright alignjustify | bullist numlist outdent indent",
tools: "inserttable", tools: "inserttable",
border_css: "/static/styles/style_tinymce.css",
content_css: "/static/styles/style_tinymce.css",
} }
if (readonly) { if (readonly) {

Some files were not shown because too many files changed in this diff

Loading…
Cancel
Save