]> git.llucax.com Git - software/sercom.git/blob - sercom/subcontrollers/grupo/__init__.py
AjaxMultiSelect widget generico.
[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, loading)
66     {
67         t = MochiKit.DOM.getElement(texto);
68
69         /* Como no se si se puede hacer de otra manera, asumo que tengo en
70          * el form un Combo que se llama curso en el codigo, y tiro error si
71          * no existe
72          */
73         curso = MochiKit.DOM.getElement('form_cursoID');
74         if (!curso) {
75             alert("No deberias ver esto, y quiere decir que tu form esta roto.\\nTe falta un combo de curso");
76             return;
77         }
78         if (curso.options[curso.selectedIndex].value <= 0) {
79             alert('Debes seleccionar un curso primero');
80             return;
81         }
82         load = MochiKit.DOM.getElement(loading);
83         load.style.visibility = 'visible';
84         url = "/grupo/get_inscripto?cursoid="+curso.options[curso.selectedIndex].value+"&padron="+t.value;
85         var d = loadJSONDoc(url);
86         d.addCallbacks(partial(_on_alumno_get_result, lista, loading), partial(_on_alumno_get_error, loading));
87         t.value = "";
88     }
89
90     function err (err)
91     {
92         alert("The metadata for MochiKit.Async could not be fetched :(");
93     }
94
95     function procesar(result)
96     {
97         l = MochiKit.DOM.getElement('form_responsable_info');
98         if (result.error)
99             l.innerHTML = result.msg;
100         else
101             l.innerHTML = result.msg.value;
102     }
103
104     function buscar_alumno()
105     {
106         /* Obtengo el curso */
107         l = MochiKit.DOM.getElement('form_cursoID');
108         cursoid = l.options[l.selectedIndex].value;
109         if (cursoid <= 0) {
110             alert('Debe seleccionar un curso');
111             return;
112         }
113         /* Obtengo el padron ingresado */
114         p = MochiKit.DOM.getElement('form_responsable');
115         padron = p.value;
116         if (padron == '') {
117             alert('Debe ingresar el padrón del alumno responsable');
118             return;
119         }
120         url = "/grupo/get_inscripto?cursoid="+cursoid+'&padron='+padron;
121         var d = loadJSONDoc(url);
122         d.addCallbacks(procesar, err);
123     }
124
125     function prepare()
126     {
127         connect('form_responsable', 'onblur', buscar_alumno);
128     }
129
130     MochiKit.DOM.addLoadEvent(prepare)
131
132 """
133
134 class GrupoForm(W.TableForm):
135     class Fields(W.WidgetsList):
136         curso = W.SingleSelectField(name='cursoID', label=_(u'Curso'), options = get_cursos,
137         validator = V.Int(not_empty=True))
138         nombre = W.TextField(label=_(u'Nombre'), validator=V.UnicodeString(not_empty=True,strip=True))
139         responsable = CustomTextField(label=_(u'Responsable'), validator=V.UnicodeString(not_empty=True), attrs=dict(size='8'))
140         alumnos = AjaxMultiSelect(name='alumnos', label=_(u'Integrantes'), validator=V.Int(), on_add="alumnos_agregar_a_la_lista")
141
142     fields = Fields()
143     javascript = [W.JSSource("MochiKit.DOM.focusOnLoad('curso');"), W.JSSource(ajax)]
144
145 form = GrupoForm()
146
147 #}}}
148
149 #{{{ Controlador
150 class GrupoController(controllers.Controller, identity.SecureResource):
151     """Basic model admin interface"""
152     require = identity.has_permission('admin')
153
154     @expose()
155     def default(self, tg_errors=None):
156         """handle non exist urls"""
157         raise redirect('list')
158
159     @expose()
160     def index(self):
161         raise redirect('list')
162
163     @expose(template='kid:%s.templates.list' % __name__)
164     @paginate('records')
165     def list(self):
166         """List records in model"""
167         r = cls.select()
168         return dict(records=r, name=name, namepl=namepl)
169
170     @expose()
171     def activate(self, id, activo):
172         """Save or create record to model"""
173         r = validate_get(id)
174         raise redirect('../../list')
175
176     @expose(template='kid:%s.templates.new' % __name__)
177     def new(self, **kw):
178         """Create new records in model"""
179         return dict(name=name, namepl=namepl, form=form, values=kw)
180
181     @validate(form=form)
182     @error_handler(new)
183     @expose()
184     def create(self, **kw):
185         """Save or create record to model"""
186         responsable = kw['responsable']
187         curso = kw['cursoID']
188         resp = None
189         try:
190             # Busco el alumno inscripto
191             resp = AlumnoInscripto.select(AND(Curso.q.id==curso, Alumno.q.usuario==responsable))
192             if resp.count() > 0:
193                 resp = resp[0]
194             else:
195                 raise Exception
196         except Exception, (inst):
197             flash(_(u'El responsable %s no existe') % responsable)
198             raise redirect('list')
199
200         kw['responsable'] = resp
201
202         # Busco los alumnos
203         alumnos = []
204         for alumnoid in kw['alumnos']:
205             alumnos.append(Alumno.get(alumnoid))
206         if alumnos == []:
207             flash(_(u'No se pudo crear el grupo. No se han agregado integrantes.'))
208             raise redirect('list')
209
210         del(kw['alumnos'])
211
212         r = validate_new(kw)
213         for a in alumnos:
214             r.add_miembro(a)
215         flash(_(u'Se creó un nuevo %s.') % name)
216         raise redirect('list')
217
218     @expose(template='kid:%s.templates.edit' % __name__)
219     def edit(self, id, **kw):
220         """Edit record in model"""
221         r = validate_get(id)
222         return dict(name=name, namepl=namepl, record=r, form=form)
223
224     @validate(form=form)
225     @error_handler(edit)
226     @expose()
227     def update(self, id, **kw):
228         """Save or create record to model"""
229         responsable = kw['responsable']
230         curso = kw['cursoID']
231         alumno = None
232         try:
233             # Busco el alumno inscripto
234             alumno = AlumnoInscripto.select(AND(Curso.q.id==curso, Alumno.q.usuario==responsable))
235             if alumno.count() > 0:
236                 alumno = alumno[0]
237             else:
238                 raise Exception
239         except Exception, (inst):
240             flash(_(u'El responsable %s no existe') % responsable)
241             raise redirect('../list')
242
243         r = validate_set(id, kw)
244         r.responsable = alumno
245         flash(_(u'El %s fue actualizado.') % name)
246         raise redirect('../list')
247
248     @expose(template='kid:%s.templates.show' % __name__)
249     def show(self,id, **kw):
250         """Show record in model"""
251         r = validate_get(id)
252         return dict(name=name, namepl=namepl, record=r)
253
254     @expose()
255     def delete(self, id):
256         """Destroy record in model"""
257         r = validate_get(id)
258         r.destroySelf()
259         flash(_(u'El %s fue eliminado permanentemente.') % name)
260         raise redirect('../list')
261
262     @expose('json')
263     def get_inscripto(self, cursoid, padron):
264         msg = u''
265         error=False
266         try:
267             # Busco el alumno inscripto
268             alumno = AlumnoInscripto.selectBy(curso=cursoid, alumno=Alumno.byUsuario(padron)).getOne()
269             msg = {}
270             msg['id'] = alumno.id
271             msg['value'] = alumno.alumno.nombre
272         except SQLObjectNotFound:
273             msg = 'No existe el alumno %s en el curso seleccionado.' % padron
274             error=True
275         except Exception, (inst):
276             msg = u"""Se ha producido un error inesperado al buscar el registro:\n      %s""" % str(inst)
277             error = True
278         return dict(msg=msg, error=error)
279 #}}}
280