]> git.llucax.com Git - software/sercom.git/blob - sercom/subcontrollers/grupo/__init__.py
Filtro por Curso para Grupo y linkeo filtrado desde Curso.
[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, Miembro
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 def validate_del(id):
57     return val.validate_del(cls, name, id)
58 #}}}
59
60 #{{{ Formulario
61 def get_docentes():
62     return [(fk1.id, fk1.shortrepr()) for fk1 in Docente.select()]
63
64 def get_cursos():
65     return [(0, u'---')] + [(fk1.id, fk1.shortrepr()) for fk1 in Curso.select()]
66
67 ajax = u"""
68     function alumnos_agregar_a_la_lista(texto, lista)
69     {
70         t = MochiKit.DOM.getElement(texto);
71
72         curso = MochiKit.DOM.getElement('form_cursoID');
73         if (!curso) {
74             alert("No deberias ver esto, y quiere decir que tu form esta roto.\\nTe falta un combo de curso");
75             return;
76         }
77         if (curso.options[curso.selectedIndex].value <= 0) {
78             alert('Debes seleccionar un curso primero');
79             return;
80         }
81         url = "/grupo/get_inscripto?cursoid="+curso.options[curso.selectedIndex].value+"&padron="+t.value;
82         t.value = "";
83         return url;
84     }
85
86     function err (err)
87     {
88         alert("The metadata for MochiKit.Async could not be fetched :(");
89     }
90
91     function procesar(result)
92     {
93         l = MochiKit.DOM.getElement('form_responsable_info');
94         if (result.error)
95             l.innerHTML = result.msg;
96         else
97             l.innerHTML = result.msg.value;
98     }
99
100     function buscar_alumno()
101     {
102         /* Obtengo el curso */
103         l = MochiKit.DOM.getElement('form_cursoID');
104         cursoid = l.options[l.selectedIndex].value;
105         if (cursoid <= 0) {
106             alert('Debe seleccionar un curso');
107             return;
108         }
109         /* Obtengo el padron ingresado */
110         p = MochiKit.DOM.getElement('form_responsable');
111         padron = p.value;
112         if (padron == '') {
113             alert('Debe ingresar el padrón del alumno responsable');
114             return;
115         }
116         url = "/grupo/get_inscripto?cursoid="+cursoid+'&padron='+padron;
117         var d = loadJSONDoc(url);
118         d.addCallbacks(procesar, err);
119     }
120
121     function prepare()
122     {
123         connect('form_responsable', 'onblur', buscar_alumno);
124     }
125
126     function onsubmit()
127     {
128         /* TODO : Validar datos y evitar el submit si no esta completo */
129
130         /* Selecciono todos los miembros si no, no llegan al controllere*/
131         l = MochiKit.DOM.getElement('form_miembros');
132         for (i=0; i<l.options.length; i++) {
133             l.options[i].selected = true;
134         }
135         return true; // Dejo hacer el submit
136     }
137
138     MochiKit.DOM.addLoadEvent(prepare)
139
140 """
141 def get_docentes():
142     return [(fk1.id, fk1.shortrepr()) for fk1 in Docente.select()]
143
144 class GrupoForm(W.TableForm):
145     class Fields(W.WidgetsList):
146         cursoID = W.SingleSelectField(label=_(u'Curso'), options = get_cursos, validator = V.Int(not_empty=True))
147         nombre = W.TextField(label=_(u'Nombre'), validator=V.UnicodeString(not_empty=True,strip=True))
148         responsable = CustomTextField(label=_(u'Responsable'), validator=V.UnicodeString(not_empty=True), attrs=dict(size='8'))
149         miembros = AjaxMultiSelect(label=_(u'Miembros'), validator=V.Int(), on_add="alumnos_agregar_a_la_lista")
150         tutores = W.MultipleSelectField(label=_(u'Tutores'), validator=V.Int(), options=get_docentes)
151
152     fields = Fields()
153     javascript = [W.JSSource("MochiKit.DOM.focusOnLoad('curso');"), W.JSSource(ajax)]
154     form_attrs = dict(onsubmit='return onsubmit()')
155
156 def get_cursos():
157     return [(0, u'---')] + [(fk1.id, fk1.shortrepr()) for fk1 in Curso.select()]
158
159 class GrupoFiltros(W.TableForm):
160     class Fields(W.WidgetsList):
161         cursoID = W.SingleSelectField(label=_(u'Curso'), options = get_cursos, validator = V.Int(not_empty=True))
162     fields = Fields()
163
164 filtro = GrupoFiltros()
165 form = GrupoForm()
166
167 #}}}
168
169 #{{{ Controlador
170 class GrupoController(controllers.Controller, identity.SecureResource):
171     """Basic model admin interface"""
172     require = identity.has_permission('admin')
173
174     @expose()
175     def default(self, tg_errors=None):
176         """handle non exist urls"""
177         raise redirect('list')
178
179     @expose()
180     def index(self):
181         raise redirect('list')
182
183     @expose(template='kid:%s.templates.list' % __name__)
184     @paginate('records')
185     def list(self, cursoID = 0):
186         """List records in model"""
187         vfilter = dict(cursoID = cursoID)
188         if int(cursoID) == 0:
189             r = cls.select()
190         else:
191             r = cls.select(cls.q.cursoID == cursoID)
192         return dict(records=r, name=name, namepl=namepl, form=filtro, vfilter=vfilter)
193
194     @expose()
195     def activate(self, id, activo):
196         """Save or create record to model"""
197         r = validate_get(id)
198         raise redirect('../../list')
199
200     @expose(template='kid:%s.templates.new' % __name__)
201     def new(self, **kw):
202         """Create new records in model"""
203         return dict(name=name, namepl=namepl, form=form, values=kw)
204
205     @validate(form=form)
206     @error_handler(new)
207     @expose()
208     def create(self, **kw):
209         """Save or create record to model"""
210         resp = kw['responsable']
211         try:
212             # Busco el alumno inscripto
213             resp = AlumnoInscripto.selectBy(cursoID=kw['cursoID'],
214                 alumno=Alumno.byPadron(kw['responsable'])).getOne()
215         except SQLObjectNotFound:
216             flash(_(u'El responsable %s no existe') % resp)
217             raise redirect('list')
218         kw['responsable'] = resp
219
220         r = validate_new(kw)
221         flash(_(u'Se creó un nuevo %s.') % name)
222         raise redirect('list')
223
224     @expose(template='kid:%s.templates.edit' % __name__)
225     def edit(self, id, **kw):
226         """Edit record in model"""
227         r = validate_get(id)
228         # TODO : No encontre mejor forma de pasar cosas al form
229         # de manera comoda y facil de formatear segun lo que espera la UI (que
230         # en este caso es super particular). Ese EmptyClass no se si hay algo estandar
231         # en python que usar, puse {} y [] pero cuando quiero hacer values.id = XX explota.
232         class EmptyClass:
233             pass
234         values = EmptyClass()
235         values.id = r.id
236         values.cursoID = r.cursoID
237         values.nombre = r.nombre
238         # TODO : Ver como llenar la lista primero :S
239         if r.responsable:
240             values.responsable = r.responsable.alumno.padron
241         values.miembros = [{"id":i.alumno.id, "label":i.alumno.alumno.nombre} for i in filter(lambda x: x.baja is None, r.miembros)]
242         values.tutores = [a.docenteID for a in r.tutores]
243         return dict(name=name, namepl=namepl, record=values, form=form)
244
245     @validate(form=form)
246     @error_handler(edit)
247     @expose()
248     def update(self, id, **kw):
249         """Save or create record to model"""
250         responsable = kw['responsable']
251         curso = kw['cursoID']
252         resp = kw['responsable']
253         try:
254             # Busco el alumno inscripto
255             resp = AlumnoInscripto.selectBy(cursoID=kw['cursoID'],
256                 alumno=Alumno.byPadron(kw['responsable'])).getOne()
257         except SQLObjectNotFound:
258             flash(_(u'El responsable %s no existe') % resp)
259             raise redirect('../list')
260         kw['responsable'] = resp
261         r = validate_set(id, kw)
262         flash(_(u'El %s fue actualizado.') % name)
263         raise redirect('../list')
264
265     @expose(template='kid:%s.templates.show' % __name__)
266     def show(self,id, **kw):
267         """Show record in model"""
268         r = validate_get(id)
269         return dict(name=name, namepl=namepl, record=r)
270
271     @expose()
272     def delete(self, id):
273         """Destroy record in model"""
274         validate_del(id)
275         flash(_(u'El %s fue eliminado permanentemente.') % name)
276         raise redirect('../list')
277
278     @expose('json')
279     def get_inscripto(self, cursoid, padron):
280         msg = u''
281         error=False
282         try:
283             # Busco el alumno inscripto
284             alumno = AlumnoInscripto.selectBy(curso=cursoid, alumno=Alumno.byUsuario(padron)).getOne()
285             msg = {}
286             msg['id'] = alumno.id
287             msg['value'] = alumno.alumno.nombre
288         except SQLObjectNotFound:
289             msg = 'No existe el alumno %s en el curso seleccionado.' % padron
290             error=True
291         except Exception, (inst):
292             msg = u"""Se ha producido un error inesperado al buscar el registro:\n      %s""" % str(inst)
293             error = True
294         return dict(msg=msg, error=error)
295
296     @expose('json')
297     def get_alumnos(self, grupoid):
298         msg = u''
299         error=False
300         try:
301             # Busco los alumnos del grupo
302             grupo = Grupo.get(int(grupoid))
303             miembros = Miembro.selectBy(baja=None, grupo=grupo)
304             print miembros
305             integrantes = []
306             for m in miembros:
307                 msg = {}
308                 alumnoInscripto = AlumnoInscripto.get(m.alumno.id)
309                 msg['id'] = alumnoInscripto.id
310                 msg['label'] = alumnoInscripto.shortrepr()
311                 integrantes.append(msg)
312         except Exception, (inst):
313             msg = u"""Se ha producido un error inesperado al buscar el registro:\n      %s""" % str(inst)
314             error = True
315             integrantes = []
316             integrantes.append(msg)
317         return dict(msg=integrantes, error=error)
318 #}}}
319