]> git.llucax.com Git - software/sercom.git/blob - sercom/subcontrollers/grupo/__init__.py
3f092a2c26974ec6e27746361ee851307e61acc2
[software/sercom.git] / sercom / subcontrollers / grupo / __init__.py
1 # vim: set et sw=4 sts=4 encoding=utf-8 foldmethod=marker :
2
3 #{{{ Imports
4 import cherrypy
5 from turbogears import controllers, expose, redirect
6 from turbogears import validate, flash, error_handler
7 from turbogears import validators as V
8 from turbogears import widgets as W
9 from turbogears import identity
10 from turbogears import paginate
11 from docutils.core import publish_parts
12 from sercom.subcontrollers import validate as val
13 from sercom.model import Curso, AlumnoInscripto, Docente, Grupo, Alumno
14 from sqlobject import *
15
16 from sercom.widgets import *
17
18 #}}}
19
20 #{{{ Configuración
21 cls = Grupo
22 name = 'grupo'
23 namepl = 'grupos'
24
25 fkcls = Curso
26 fkname = 'curso'
27 fknamepl = fkname + 's'
28 #}}}
29
30 #{{{ Validación
31 def validate_fk(data):
32     fk = data.get(fkname + 'ID', None)
33     if fk == 0: fk = None
34     if fk is not None:
35         try:
36             fk = fkcls.get(fk)
37         except LookupError:
38             flash(_(u'No se pudo crear el nuevo %s porque el %s con '
39                 'identificador %d no existe.' % (name, fkname, fk)))
40             raise redirect('new', **data)
41     data.pop(fkname + 'ID', None)
42     data[fkname] = fk
43     return fk
44
45 def validate_get(id):
46     return val.validate_get(cls, name, id)
47
48 def validate_set(id, data):
49     validate_fk(data)
50     return val.validate_set(cls, name, id, data)
51
52 def validate_new(data):
53     validate_fk(data)
54     return val.validate_new(cls, name, data)
55 #}}}
56
57 #{{{ Formulario
58 def get_docentes():
59     return [(fk1.id, fk1.shortrepr()) for fk1 in Docente.select()]
60
61 def get_cursos():
62     return [(0, u'---')] + [(fk1.id, fk1.shortrepr()) for fk1 in Curso.select()]
63
64 ajax = u"""
65     function alumnos_agregar_a_la_lista(texto, lista)
66     {
67         t = MochiKit.DOM.getElement(texto);
68
69         curso = MochiKit.DOM.getElement('form_cursoID');
70         if (!curso) {
71             alert("No deberias ver esto, y quiere decir que tu form esta roto.\\nTe falta un combo de curso");
72             return;
73         }
74         if (curso.options[curso.selectedIndex].value <= 0) {
75             alert('Debes seleccionar un curso primero');
76             return;
77         }
78         url = "/grupo/get_inscripto?cursoid="+curso.options[curso.selectedIndex].value+"&padron="+t.value;
79         t.value = "";
80         return url;
81     }
82
83     function err (err)
84     {
85         alert("The metadata for MochiKit.Async could not be fetched :(");
86     }
87
88     function procesar(result)
89     {
90         l = MochiKit.DOM.getElement('form_responsable_info');
91         if (result.error)
92             l.innerHTML = result.msg;
93         else
94             l.innerHTML = result.msg.value;
95     }
96
97     function buscar_alumno()
98     {
99         /* Obtengo el curso */
100         l = MochiKit.DOM.getElement('form_cursoID');
101         cursoid = l.options[l.selectedIndex].value;
102         if (cursoid <= 0) {
103             alert('Debe seleccionar un curso');
104             return;
105         }
106         /* Obtengo el padron ingresado */
107         p = MochiKit.DOM.getElement('form_responsable');
108         padron = p.value;
109         if (padron == '') {
110             alert('Debe ingresar el padrón del alumno responsable');
111             return;
112         }
113         url = "/grupo/get_inscripto?cursoid="+cursoid+'&padron='+padron;
114         var d = loadJSONDoc(url);
115         d.addCallbacks(procesar, err);
116     }
117
118     function prepare()
119     {
120         connect('form_responsable', 'onblur', buscar_alumno);
121     }
122
123     MochiKit.DOM.addLoadEvent(prepare)
124
125 """
126 def get_docentes():
127     return [(fk1.id, fk1.shortrepr()) for fk1 in Docente.select()]
128
129 class GrupoForm(W.TableForm):
130     class Fields(W.WidgetsList):
131         cursoID = W.SingleSelectField(label=_(u'Curso'), options = get_cursos, validator = V.Int(not_empty=True))
132         nombre = W.TextField(label=_(u'Nombre'), validator=V.UnicodeString(not_empty=True,strip=True))
133         responsable = CustomTextField(label=_(u'Responsable'), validator=V.UnicodeString(not_empty=True), attrs=dict(size='8'))
134         miembros = AjaxMultiSelect(label=_(u'Miembros'), validator=V.Int(), on_add="alumnos_agregar_a_la_lista")
135         tutores = W.MultipleSelectField(label=_(u'Tutores'), validator=V.Int(), options=get_docentes)
136
137     fields = Fields()
138     javascript = [W.JSSource("MochiKit.DOM.focusOnLoad('curso');"), W.JSSource(ajax)]
139
140 form = GrupoForm()
141
142 #}}}
143
144 #{{{ Controlador
145 class GrupoController(controllers.Controller, identity.SecureResource):
146     """Basic model admin interface"""
147     require = identity.has_permission('admin')
148
149     @expose()
150     def default(self, tg_errors=None):
151         """handle non exist urls"""
152         raise redirect('list')
153
154     @expose()
155     def index(self):
156         raise redirect('list')
157
158     @expose(template='kid:%s.templates.list' % __name__)
159     @paginate('records')
160     def list(self):
161         """List records in model"""
162         r = cls.select()
163         return dict(records=r, name=name, namepl=namepl)
164
165     @expose()
166     def activate(self, id, activo):
167         """Save or create record to model"""
168         r = validate_get(id)
169         raise redirect('../../list')
170
171     @expose(template='kid:%s.templates.new' % __name__)
172     def new(self, **kw):
173         """Create new records in model"""
174         return dict(name=name, namepl=namepl, form=form, values=kw)
175
176     @validate(form=form)
177     @error_handler(new)
178     @expose()
179     def create(self, **kw):
180         """Save or create record to model"""
181         resp = kw['responsable']
182         try:
183             # Busco el alumno inscripto
184             resp = AlumnoInscripto.selectBy(cursoID=kw['cursoID'],
185                 alumno=Alumno.byPadron(kw['responsable'])).getOne()
186         except SQLObjectNotFound:
187             flash(_(u'El responsable %s no existe') % resp)
188             raise redirect('list')
189         kw['responsable'] = resp
190
191         r = validate_new(kw)
192         flash(_(u'Se creó un nuevo %s.') % name)
193         raise redirect('list')
194
195     @expose(template='kid:%s.templates.edit' % __name__)
196     def edit(self, id, **kw):
197         """Edit record in model"""
198         r = validate_get(id)
199         # TODO : No encontre mejor forma de pasar cosas al form
200         # de manera comoda y facil de formatear segun lo que espera la UI (que
201         # en este caso es super particular). Ese EmptyClass no se si hay algo estandar
202         # en python que usar, puse {} y [] pero cuando quiero hacer values.id = XX explota.
203         class EmptyClass:
204             pass
205         values = EmptyClass()
206         values.id = r.id
207         values.cursoID = r.cursoID
208         values.nombre = r.nombre
209         # TODO : Ver como llenar la lista primero :S
210         if r.responsable:
211             values.responsable = r.responsable.alumno.padron
212         values.miembros = [1]
213         values.tutores = [a.docenteID for a in r.tutores]
214         return dict(name=name, namepl=namepl, record=values, form=form)
215
216     @validate(form=form)
217     @error_handler(edit)
218     @expose()
219     def update(self, id, **kw):
220         """Save or create record to model"""
221         responsable = kw['responsable']
222         curso = kw['cursoID']
223         alumno = None
224         try:
225             # Busco el alumno inscripto
226             alumno = AlumnoInscripto.select(AND(Curso.q.id==curso, Alumno.q.usuario==responsable))
227             if alumno.count() > 0:
228                 alumno = alumno[0]
229             else:
230                 raise Exception
231         except Exception, (inst):
232             flash(_(u'El responsable %s no existe') % responsable)
233             raise redirect('../list')
234
235         r = validate_set(id, kw)
236         r.responsable = alumno
237         flash(_(u'El %s fue actualizado.') % name)
238         raise redirect('../list')
239
240     @expose(template='kid:%s.templates.show' % __name__)
241     def show(self,id, **kw):
242         """Show record in model"""
243         r = validate_get(id)
244         return dict(name=name, namepl=namepl, record=r)
245
246     @expose()
247     def delete(self, id):
248         """Destroy record in model"""
249         r = validate_get(id)
250         r.destroySelf()
251         flash(_(u'El %s fue eliminado permanentemente.') % name)
252         raise redirect('../list')
253
254     @expose('json')
255     def get_inscripto(self, cursoid, padron):
256         msg = u''
257         error=False
258         try:
259             # Busco el alumno inscripto
260             alumno = AlumnoInscripto.selectBy(curso=cursoid, alumno=Alumno.byUsuario(padron)).getOne()
261             msg = {}
262             msg['id'] = alumno.id
263             msg['value'] = alumno.alumno.nombre
264         except SQLObjectNotFound:
265             msg = 'No existe el alumno %s en el curso seleccionado.' % padron
266             error=True
267         except Exception, (inst):
268             msg = u"""Se ha producido un error inesperado al buscar el registro:\n      %s""" % str(inst)
269             error = True
270         return dict(msg=msg, error=error)
271 #}}}
272