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, url
11 from docutils.core import publish_parts
12 from sercom.subcontrollers import validate as val
13 from sercom.model import Curso, AlumnoInscripto, Docente, DocenteInscripto, Grupo, Alumno, Miembro
14 from sqlobject import *
15 from sqlobject.dberrors import *
17 from sercom.widgets import *
21 log = logging.getLogger('sercom.curso.grupo.admin')
33 return val.validate_get(cls, name, id)
35 def validate_set(id, data):
36 return val.validate_set(cls, name, id, data)
38 def validate_new(data):
39 return val.validate_new(cls, name, data)
42 return val.validate_del(cls, name, id)
47 return [(fk1.id, fk1.shortrepr()) for fk1 in Docente.select()]
49 def get_docentes_inscriptos(id):
50 return [(fk1.id, fk1.shortrepr()) for fk1 in DocenteInscripto.select(DocenteInscripto.q.cursoID==id)]
53 function alumnos_agregar_a_la_lista(texto, lista)
55 t = MochiKit.DOM.getElement(texto);
57 curso = MochiKit.DOM.getElement('form_cursoID');
59 alert("No deberias ver esto, y quiere decir que tu form esta roto.\\nTe falta un combo de curso");
62 if (curso.value <= 0) {
63 alert('Debes seleccionar un curso primero');
66 url = "/curso/grupo/get_inscripto?cursoid="+curso.value+"&padron="+t.value;
73 alert("The metadata for MochiKit.Async could not be fetched :(");
76 function procesar(result)
78 l = MochiKit.DOM.getElement('form_responsable_info');
80 l.innerHTML = result.msg;
82 l.innerHTML = result.msg.value;
85 function buscar_alumno()
87 /* Obtengo el padron ingresado */
88 p = MochiKit.DOM.getElement('form_responsable');
93 /* Obtengo el curso */
94 l = MochiKit.DOM.getElement('form_cursoID');
97 alert('Debe seleccionar un curso');
100 url = "/curso/grupo/get_inscripto?cursoid="+cursoid+'&padron='+padron;
101 var d = loadJSONDoc(url);
102 d.addCallbacks(procesar, err);
107 connect('form_responsable', 'onblur', buscar_alumno);
112 /* TODO : Validar datos y evitar el submit si no esta completo */
114 /* Selecciono todos los miembros si no, no llegan al controllere*/
115 l = MochiKit.DOM.getElement('form_miembros');
116 for (i=0; i<l.options.length; i++) {
117 l.options[i].selected = true;
119 return true; // Dejo hacer el submit
122 MochiKit.DOM.addLoadEvent(prepare)
126 class GrupoForm(W.TableForm):
127 class Fields(W.WidgetsList):
128 cursoID = W.HiddenField(validator=V.Int)
129 nombre = W.TextField(label=_(u'Nombre'), validator=V.UnicodeString(not_empty=True,strip=True))
130 responsable = CustomTextField(label=_(u'Responsable'), validator=V.UnicodeString(), attrs=dict(size='8'))
131 miembros = AjaxMultiSelect(label=_(u'Miembros'), validator=V.Int(), on_add="alumnos_agregar_a_la_lista")
132 tutores = W.MultipleSelectField(label=_(u'Tutores'), validator=V.Int)
135 javascript = [W.JSSource("MochiKit.DOM.focusOnLoad('form_nombre');"), W.JSSource(ajax)]
136 form_attrs = dict(onsubmit='return doSubmit()')
140 def get_gruposA(cursoID):
141 return [(0, u'---')] + [(g.id, g.shortrepr()) for g in Grupo.select(Grupo.q.cursoID==cursoID)]
143 def get_gruposB(cursoID):
144 return [(0, u'Nuevo Grupo')] + [(g.id, g.shortrepr()) for g in Grupo.select(Grupo.q.cursoID==cursoID)]
149 alert("The metadata for MochiKit.Async could not be fetched :(");
154 /* TODO : Validar datos y evitar el submit si no esta completo */
156 /* Selecciono todos los miembros si no, no llegan al controllere*/
157 l = MochiKit.DOM.getElement('form_grupos_to');
158 for (i=0; i<l.options.length; i++) {
159 l.options[i].selected = true;
161 /* Selecciono todos los miembros si no, no llegan al controllere*/
162 l = MochiKit.DOM.getElement('form_grupos_from');
163 for (i=0; i<l.options.length; i++) {
164 l.options[i].selected = true;
167 return true; // Dejo hacer el submit
170 function initWidgets(disabled) {
172 MochiKit.DOM.getElement('form_listaGrupoA').selectedIndex = 0;
174 MochiKit.DOM.getElement('form_listaGrupoB').selectedIndex = 0;
175 MochiKit.DOM.getElement('form_grupos_to').options.length = 0;
176 MochiKit.DOM.getElement('form_grupos_from').options.length = 0;
177 MochiKit.DOM.getElement('form_listaGrupoB').disabled = disabled;
178 MochiKit.DOM.getElement('form_grupos_to').disabled = disabled;
179 MochiKit.DOM.getElement('form_grupos_from').disabled = disabled;
180 MochiKit.DOM.getElement('form_tutoresA').disabled = true;
181 MochiKit.DOM.getElement('form_tutoresB').disabled = true;
182 MochiKit.DOM.getElement('form_responsableA').disabled = true;
183 MochiKit.DOM.getElement('form_responsableB').disabled = true;
186 function onListaAChange() {
187 lista = MochiKit.DOM.getElement('form_listaGrupoA');
188 if ( lista.selectedIndex != '0' ) {
194 // carga el grupo en el multiple select
195 grupoA = MochiKit.DOM.getElement('form_grupos_from');
196 id = lista.options[lista.selectedIndex].value
197 cargarGrupo(id, grupoA);
200 function onListaBChange() {
201 lista = MochiKit.DOM.getElement('form_listaGrupoB');
202 listaA = MochiKit.DOM.getElement('form_listaGrupoA');
203 MochiKit.DOM.getElement('form_grupos_to').options.length = 0;
204 if ( lista.selectedIndex == 0 ) {
207 if ( lista.selectedIndex != '0' ) {
208 if ( lista.selectedIndex == listaA.selectedIndex ) {
209 window.alert('Debe seleccionar 2 grupos distintos');
210 MochiKit.DOM.getElement('form_grupos_to').options.length = 0;
213 grupoB = MochiKit.DOM.getElement('form_grupos_to');
214 id = lista.options[lista.selectedIndex].value
215 cargarGrupo(id, grupoB);
219 function makeOption(option) {
220 return OPTION({"value": option.value}, option.text);
223 function cargarGrupo(grupoid, lista) {
224 var result = loadJSONDoc('/curso/grupo/get_alumnos?grupoid='+id);
225 result.addCallbacks(partial(cargarLista, lista), err)
230 alert("The metadata for MochiKit.Async could not be fetched :(");
233 function cargarLista(lista, result) {
234 var alumnos = result.msg;
236 window.alert(result.msg);
241 label = alumnos[i].label;
242 MochiKit.DOM.appendChildNodes(lista, OPTION({"value":id}, label))
244 ActualizarResponsables();
247 function ActualizarResponsables()
249 replaceChildNodes('form_responsableA', '');
250 replaceChildNodes('form_responsableB', '');
251 appendChildNodes('form_responsableA', map(makeOption, $('form_grupos_from').options));
252 appendChildNodes('form_responsableB', map(makeOption, $('form_grupos_to').options));
254 if (getElement('form_grupos_from').options.length == 0) {
255 getElement('form_tutoresA').disabled = true;
256 getElement('form_responsableA').disabled = true;
258 getElement('form_tutoresA').disabled = false;
259 getElement('form_responsableA').disabled = false;
261 if (getElement('form_grupos_to').options.length == 0) {
262 getElement('form_tutoresB').disabled = true;
263 getElement('form_responsableB').disabled = true;
265 getElement('form_tutoresB').disabled = false;
266 getElement('form_responsableB').disabled = false;
271 class GrupoAdminForm(W.TableForm):
272 class Fields(W.WidgetsList):
273 cursoID = W.HiddenField()
274 listaGrupoA = W.SingleSelectField(label=_(u'Grupo A'), attrs = dict(onChange='onListaAChange()'), validator = V.Int(not_empty=True))
275 listaGrupoB = W.SingleSelectField(label=_(u'Grupo B'), attrs = dict(onChange='onListaBChange()'), validator = V.Int(not_empty=True))
276 grupos = AjaxDosListasSelect(label=_(u'Grupos'),title_from=u"Grupo A", size=8, move_signal="ActualizarResponsables();", title_to=u"Grupo B", validator=V.Int(not_empty=True))
277 responsableA = W.SingleSelectField(label=_(u'Responsable A'), validator = V.Int())
278 responsableB = W.SingleSelectField(label=_(u'Responsable B'), validator = V.Int())
279 tutoresA = W.MultipleSelectField(label=_(u'Tutores A'), validator = V.Int(not_empty=True))
280 tutoresB = W.MultipleSelectField(label=_(u'Tutores B'), validator = V.Int(not_empty=True))
283 javascript = [W.JSSource("MochiKit.DOM.focusOnLoad('listaGrupoA');"), W.JSSource(ajaxadmin)]
284 form_attrs = dict(onsubmit='return doSubmit();')
286 formadmin = GrupoAdminForm()
291 class GrupoController(controllers.Controller, identity.SecureResource):
292 """Basic model admin interface"""
293 require = identity.has_permission('admin')
296 def default(self, tg_errors=None):
297 """handle non exist urls"""
298 raise redirect(tg.url('/curso/list'))
302 raise redirect(tg.url('/curso/list'))
304 @expose(template='kid:%s.templates.list' % __name__)
305 @validate(validators=dict(curso=V.Int))
307 def list(self, curso):
308 """List records in model"""
309 r = cls.selectBy(cursoID=curso)
310 return dict(records=r, name=name, namepl=namepl.capitalize(), curso=Curso.get(curso))
312 @expose(template='kid:%s.templates.new' % __name__)
313 @validate(validators=dict(curso=V.Int))
314 def new(self, curso, **kw):
315 """Create new records in model"""
316 kw['cursoID'] = curso # FIXME esto está roto porque los widgets son stateless
317 options = dict(tutores=get_docentes_inscriptos(curso))
318 return dict(name=name, namepl=namepl, form=form, options=options, values=kw)
323 def create(self, **kw):
324 """Save or create record to model"""
325 resp = kw['responsable']
327 # Busco el alumno inscripto
328 resp = AlumnoInscripto.selectBy(cursoID=kw['cursoID'],
329 alumno=Alumno.byPadron(kw['responsable'])).getOne()
330 except SQLObjectNotFound:
332 kw['responsable'] = resp
335 flash(_(u'Se creó un nuevo %s.') % name)
336 raise redirect('list/%d' % int(kw['cursoID']))
338 @expose(template='kid:%s.templates.edit' % __name__)
339 def edit(self, id, **kw):
340 """Edit record in model"""
342 # TODO : No encontre mejor forma de pasar cosas al form
343 # de manera comoda y facil de formatear segun lo que espera la UI (que
344 # en este caso es super particular). Ese EmptyClass no se si hay algo estandar
345 # en python que usar, puse {} y [] pero cuando quiero hacer values.id = XX explota.
346 options = dict(tutores=get_docentes_inscriptos(r.curso.id))
349 values = EmptyClass()
351 values.cursoID = r.cursoID
352 values.nombre = r.nombre
353 # TODO : Ver como llenar la lista primero :S
355 values.responsable = r.responsable.alumno.padron
356 values.miembros = [{"id":i.alumno.id, "label":i.alumno.alumno.nombre} for i in filter(lambda x: x.baja is None, r.miembros)]
357 values.tutores = [a.docenteID for a in r.tutores]
358 return dict(name=name, namepl=namepl, record=values, options=options, form=form)
363 def update(self, id, **kw):
364 """Save or create record to model"""
365 responsable = kw['responsable']
366 curso = kw['cursoID']
367 resp = kw['responsable']
369 # Busco el alumno inscripto
370 resp = AlumnoInscripto.selectBy(cursoID=kw['cursoID'],
371 alumno=Alumno.byPadron(kw['responsable'])).getOne()
372 except SQLObjectNotFound:
374 kw['responsable'] = resp
375 r = validate_set(id, kw)
376 flash(_(u'El %s fue actualizado.') % name)
377 raise redirect('../list/%d' % r.curso.id)
379 @expose(template='kid:%s.templates.show' % __name__)
380 def show(self,id, **kw):
381 """Show record in model"""
383 return dict(name=name, namepl=namepl, record=r)
386 def delete(self, cursoID, id):
387 """Destroy record in model"""
389 flash(_(u'El %s fue eliminado permanentemente.') % name)
390 raise redirect('../../list/%d' % int(cursoID))
393 def get_inscripto(self, cursoid, padron):
397 # Busco el alumno inscripto
398 alumno = AlumnoInscripto.selectBy(curso=cursoid, alumno=Alumno.byUsuario(padron)).getOne()
400 msg['id'] = alumno.id
401 msg['value'] = alumno.alumno.nombre
402 except SQLObjectNotFound:
403 msg = 'No existe el alumno %s en el curso seleccionado.' % padron
405 except Exception, (inst):
406 msg = u"""Se ha producido un error inesperado al buscar el registro:\n %s""" % str(inst)
408 return dict(msg=msg, error=error)
411 def get_alumnos(self, grupoid):
415 # Busco los alumnos del grupo
416 grupo = Grupo.get(int(grupoid))
417 miembros = Miembro.selectBy(baja=None, grupo=grupo)
422 alumnoInscripto = AlumnoInscripto.get(m.alumno.id)
423 msg['id'] = alumnoInscripto.id
424 msg['label'] = alumnoInscripto.shortrepr()
425 integrantes.append(msg)
426 except Exception, (inst):
427 msg = u"""Se ha producido un error inesperado al buscar el registro:\n %s""" % str(inst)
430 integrantes.append(msg)
431 return dict(msg=integrantes, error=error)
433 @expose(template='kid:%s.templates.admin' % __name__)
434 def admin(self, cursoID, **kw):
435 """Create new records in model"""
437 listaGrupoA=get_gruposA(cursoID),
438 listaGrupoB=get_gruposB(cursoID),
439 tutoresA=get_docentes_inscriptos(cursoID),
440 tutoresB=get_docentes_inscriptos(cursoID),
442 kw['cursoID'] = cursoID
443 return dict(name=name, namepl=namepl, options=options, form=formadmin, values=kw, cursoID=int(cursoID))
445 @validate(form=formadmin)
446 @error_handler(admin)
448 def adminupdate(self, **kw):
449 """Save or create record to model"""
450 cursoID = int(kw['cursoID'])
452 grupoAId = kw['listaGrupoA']
453 grupoBId = kw['listaGrupoB']
454 miembrosA = kw.get('grupos_from', [])
455 miembrosB = kw.get('grupos_to', [])
456 responsableA = kw['responsableA']
457 responsableB = kw['responsableB']
458 tutoresA = kw.get('tutoresA', [])
459 tutoresB = kw.get('tutoresB', [])
461 # por las dudas de que no sea una lista
462 if not isinstance(miembrosA, list):
463 miembrosA = [miembrosA]
464 if not isinstance(miembrosB, list):
465 miembrosB = [miembrosB]
466 if not isinstance(tutoresA, list):
467 tutoresA = [tutoresA]
468 if not isinstance(tutoresB, list):
469 tutoresB = [tutoresB]
472 """ levanto los grupos originales """
473 grupoAorig = validate_get(int(grupoAId))
475 log.debug(Miembro.selectBy(grupo=grupoAorig, baja=None))
476 """ Si el grupo A quedo vacio deberia eliminarlo (primero
477 genero los otros para que no elimine los alumnos)"""
478 for mA in Miembro.selectBy(grupo=grupoAorig, baja=None):
479 if str(mA.alumno.id) not in miembrosA:
480 grupoAorig.remove_miembro(mA.alumno.id)
483 grupoA = validate_get(grupoAId)
486 grupoA.add_miembro(a, baja=None)
487 except DuplicateEntryError:
491 flash(_(u'Error A %s.' % e))
492 raise redirect(url('/curso/grupo/list' % int(cursoID)))
493 # seteo el reponsable del grupo
494 if responsableA and int(responsableA) != 0:
495 grupoA.responsable = AlumnoInscripto.get(int(responsableA))
499 grupoA.add_tutor(int(t))
501 #TODO ver por que no anda el duplicate error, por ahora cacheo silencioso
505 #Elimino el grupo si quedo vacio
506 if len(miembrosA) == 0:
508 validate_del(grupoAId)
512 # si se selecciono un grupo nuevo
513 if int(grupoBId) == 0:
514 # creo un grupo nuevo
517 nuevosMiembros.append(AlumnoInscripto.get(int(m)))
520 nuevosTutores.append(DocenteInscripto.get(t))
522 Grupo(miembros = nuevosMiembros, tutores = nuevosTutores, cursoID=cursoID, nombre='NuevoGrupo'+str(cursoID))
524 grupoBorig = validate_get(int(grupoBId))
526 b = list(Miembro.selectBy(grupo=grupoBorig, baja=None))
528 #borro todos y los vuelvo a agregar
529 for mB in Miembro.selectBy(grupo=grupoBorig, baja=None):
530 if str(mB.alumno.id) not in miembrosB:
531 grupoBorig.remove_miembro(mB.alumno.id)
533 grupoB = validate_get(grupoBId)
536 grupoB.add_miembro(b, baja=None)
537 except DuplicateEntryError:
541 flash(_(u'Error B %s.' % e))
542 raise redirect(url('/curso/grupo/list/%d' % int(cursoID)))
543 # seteo el reponsable del grupo
544 if responsableB and int(responsableB) != 0:
545 grupoB.responsable = AlumnoInscripto.get(int(responsableB))
547 #Elimino el grupo si quedo vacio
548 if len(miembrosB) == 0:
550 validate_del(grupoBId)
556 grupoB.add_tutor(int(t))
558 #TODO ver por que no anda el duplicate error, por ahora cahceo silencioso
560 flash(_(u'Los grupos fueron actualizado.'))
561 raise redirect(url('/curso/grupo/list/%d' % int(cursoID)))