1 # vim: set et sw=4 sts=4 encoding=utf-8 foldmethod=marker :
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 *
16 from sercom.widgets import *
27 fknamepl = fkname + 's'
31 def validate_fk(data):
32 fk = data.get(fkname + 'ID', None)
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)
46 return val.validate_get(cls, name, id)
48 def validate_set(id, data):
50 return val.validate_set(cls, name, id, data)
52 def validate_new(data):
54 return val.validate_new(cls, name, data)
57 return val.validate_del(cls, name, id)
62 return [(fk1.id, fk1.shortrepr()) for fk1 in Docente.select()]
65 return [(0, u'---')] + [(fk1.id, fk1.shortrepr()) for fk1 in Curso.select()]
68 function alumnos_agregar_a_la_lista(texto, lista)
70 t = MochiKit.DOM.getElement(texto);
72 curso = MochiKit.DOM.getElement('form_cursoID');
74 alert("No deberias ver esto, y quiere decir que tu form esta roto.\\nTe falta un combo de curso");
77 if (curso.options[curso.selectedIndex].value <= 0) {
78 alert('Debes seleccionar un curso primero');
81 url = "/grupo/get_inscripto?cursoid="+curso.options[curso.selectedIndex].value+"&padron="+t.value;
88 alert("The metadata for MochiKit.Async could not be fetched :(");
91 function procesar(result)
93 l = MochiKit.DOM.getElement('form_responsable_info');
95 l.innerHTML = result.msg;
97 l.innerHTML = result.msg.value;
100 function buscar_alumno()
102 /* Obtengo el padron ingresado */
103 p = MochiKit.DOM.getElement('form_responsable');
108 /* Obtengo el curso */
109 l = MochiKit.DOM.getElement('form_cursoID');
110 cursoid = l.options[l.selectedIndex].value;
112 alert('Debe seleccionar un curso');
115 url = "/grupo/get_inscripto?cursoid="+cursoid+'&padron='+padron;
116 var d = loadJSONDoc(url);
117 d.addCallbacks(procesar, err);
122 connect('form_responsable', 'onblur', buscar_alumno);
127 /* TODO : Validar datos y evitar el submit si no esta completo */
129 /* Selecciono todos los miembros si no, no llegan al controllere*/
130 l = MochiKit.DOM.getElement('form_miembros');
131 for (i=0; i<l.options.length; i++) {
132 l.options[i].selected = true;
134 return true; // Dejo hacer el submit
137 MochiKit.DOM.addLoadEvent(prepare)
141 return [(fk1.id, fk1.shortrepr()) for fk1 in Docente.select()]
143 class GrupoForm(W.TableForm):
144 class Fields(W.WidgetsList):
145 cursoID = W.SingleSelectField(label=_(u'Curso'), options = get_cursos, validator = V.Int(not_empty=True))
146 nombre = W.TextField(label=_(u'Nombre'), validator=V.UnicodeString(not_empty=True,strip=True))
147 responsable = CustomTextField(label=_(u'Responsable'), validator=V.UnicodeString(), attrs=dict(size='8'))
148 miembros = AjaxMultiSelect(label=_(u'Miembros'), validator=V.Int(), on_add="alumnos_agregar_a_la_lista")
149 tutores = W.MultipleSelectField(label=_(u'Tutores'), validator=V.Int(), options=get_docentes)
152 javascript = [W.JSSource("MochiKit.DOM.focusOnLoad('curso');"), W.JSSource(ajax)]
153 form_attrs = dict(onsubmit='return doSubmit()')
156 return [(0, u'---')] + [(fk1.id, fk1.shortrepr()) for fk1 in Curso.select()]
158 class GrupoFiltros(W.TableForm):
159 class Fields(W.WidgetsList):
160 cursoID = W.SingleSelectField(label=_(u'Curso'), options = get_cursos, validator = V.Int(not_empty=True))
163 filtro = GrupoFiltros()
169 class GrupoController(controllers.Controller, identity.SecureResource):
170 """Basic model admin interface"""
171 require = identity.has_permission('admin')
174 def default(self, tg_errors=None):
175 """handle non exist urls"""
176 raise redirect('list')
180 raise redirect('list')
182 @expose(template='kid:%s.templates.list' % __name__)
184 def list(self, cursoID = 0):
185 """List records in model"""
186 vfilter = dict(cursoID = cursoID)
187 if int(cursoID) == 0:
190 r = cls.select(cls.q.cursoID == cursoID)
191 return dict(records=r, name=name, namepl=namepl, form=filtro, vfilter=vfilter)
194 def activate(self, id, activo):
195 """Save or create record to model"""
197 raise redirect('../../list')
199 @expose(template='kid:%s.templates.new' % __name__)
201 """Create new records in model"""
202 return dict(name=name, namepl=namepl, form=form, values=kw)
207 def create(self, **kw):
208 """Save or create record to model"""
209 resp = kw['responsable']
211 # Busco el alumno inscripto
212 resp = AlumnoInscripto.selectBy(cursoID=kw['cursoID'],
213 alumno=Alumno.byPadron(kw['responsable'])).getOne()
214 except SQLObjectNotFound:
216 kw['responsable'] = resp
219 flash(_(u'Se creó un nuevo %s.') % name)
220 raise redirect('list')
222 @expose(template='kid:%s.templates.edit' % __name__)
223 def edit(self, id, **kw):
224 """Edit record in model"""
226 # TODO : No encontre mejor forma de pasar cosas al form
227 # de manera comoda y facil de formatear segun lo que espera la UI (que
228 # en este caso es super particular). Ese EmptyClass no se si hay algo estandar
229 # en python que usar, puse {} y [] pero cuando quiero hacer values.id = XX explota.
232 values = EmptyClass()
234 values.cursoID = r.cursoID
235 values.nombre = r.nombre
236 # TODO : Ver como llenar la lista primero :S
238 values.responsable = r.responsable.alumno.padron
239 values.miembros = [{"id":i.alumno.id, "label":i.alumno.alumno.nombre} for i in filter(lambda x: x.baja is None, r.miembros)]
240 values.tutores = [a.docenteID for a in r.tutores]
241 return dict(name=name, namepl=namepl, record=values, form=form)
246 def update(self, id, **kw):
247 """Save or create record to model"""
248 responsable = kw['responsable']
249 curso = kw['cursoID']
250 resp = kw['responsable']
252 # Busco el alumno inscripto
253 resp = AlumnoInscripto.selectBy(cursoID=kw['cursoID'],
254 alumno=Alumno.byPadron(kw['responsable'])).getOne()
255 except SQLObjectNotFound:
257 kw['responsable'] = resp
258 r = validate_set(id, kw)
259 flash(_(u'El %s fue actualizado.') % name)
260 raise redirect('../list')
262 @expose(template='kid:%s.templates.show' % __name__)
263 def show(self,id, **kw):
264 """Show record in model"""
266 return dict(name=name, namepl=namepl, record=r)
269 def delete(self, id):
270 """Destroy record in model"""
272 flash(_(u'El %s fue eliminado permanentemente.') % name)
273 raise redirect('../list')
276 def get_inscripto(self, cursoid, padron):
280 # Busco el alumno inscripto
281 alumno = AlumnoInscripto.selectBy(curso=cursoid, alumno=Alumno.byUsuario(padron)).getOne()
283 msg['id'] = alumno.id
284 msg['value'] = alumno.alumno.nombre
285 except SQLObjectNotFound:
286 msg = 'No existe el alumno %s en el curso seleccionado.' % padron
288 except Exception, (inst):
289 msg = u"""Se ha producido un error inesperado al buscar el registro:\n %s""" % str(inst)
291 return dict(msg=msg, error=error)
294 def get_alumnos(self, grupoid):
298 # Busco los alumnos del grupo
299 grupo = Grupo.get(int(grupoid))
300 miembros = Miembro.selectBy(baja=None, grupo=grupo)
305 alumnoInscripto = AlumnoInscripto.get(m.alumno.id)
306 msg['id'] = alumnoInscripto.id
307 msg['label'] = alumnoInscripto.shortrepr()
308 integrantes.append(msg)
309 except Exception, (inst):
310 msg = u"""Se ha producido un error inesperado al buscar el registro:\n %s""" % str(inst)
313 integrantes.append(msg)
314 return dict(msg=integrantes, error=error)