]> git.evergreen-ils.org Git - working/Evergreen.git/blob - Open-ILS/xul/staff_client/server/patron/ue.js
fixed bug in profile editing when user does not have perm to edit that profile
[working/Evergreen.git] / Open-ILS / xul / staff_client / server / patron / ue.js
1 var cgi                                                 = null;
2 var clone                                               = false;
3 var patron                                              = null;
4 var counter                                             = 0;
5 var identTypesCache                     = {};
6 var statCatsCache                               = {};
7 var surveysCache                                = {};
8 var surveyQuestionsCache        = {};
9 var surveyAnswersCache          = {};
10 var userCache                                   = {};
11 var groupsCache                         = {};
12 var netLevelsCache                      = {};
13 var guardianNote                                = null; 
14
15
16 /* fetch the necessary data to start off */
17 function uEditInit() {
18
19         _debug('uEditInit(): ' + location.search);
20
21         cgi             = new CGI();
22         session = cgi.param('ses');
23         clone           = cgi.param('clone');
24         if(!session) throw "User session is not defined";
25
26         fetchUser(session);
27         $('uedit_user').appendChild(text(USER.usrname()));
28
29         setTimeout( function() { 
30                 uEditBuild(); uEditShowPage('uedit_userid'); }, 20 );
31 }
32
33 /* ------------------------------------------------------------------------------ */
34 /* Fetch code
35 /* ------------------------------------------------------------------------------ */
36 function uEditFetchIdentTypes() {
37         var req = new Request(FETCH_ID_TYPES);
38         req.send(true);
39         return req.result();
40 }
41
42 function uEditFetchStatCats() {
43         var req = new Request(SC_FETCH_ALL, SESSION);
44         req.send(true);
45         return req.result();
46 }
47
48 function uEditFetchSurveys() {
49         var req = new Request(SV_FETCH_ALL, SESSION);
50         req.send(true);
51         return req.result();
52 }
53
54 function uEditFetchGroups() {
55         var req = new Request(FETCH_GROUPS);
56         req.send(true);
57         return req.result();
58 }
59
60 function uEditFetchNetLevels() {
61         var req = new Request(FETCH_NET_LEVELS, SESSION);
62         req.send(true);
63         return req.result();
64 }
65
66 /* ------------------------------------------------------------------------------ */
67
68
69
70 /* fetches necessary and builds the UI */
71 function uEditBuild() {
72         fetchHighestPermOrgs( SESSION, USER.id(), myPerms );
73
74         uEditBuildLibSelector();
75         patron = fetchFleshedUser(cgi.param('usr'));
76         if(!patron) patron = uEditNewPatron(); 
77
78
79         uEditDraw( 
80                 uEditFetchIdentTypes(),
81                 uEditFetchGroups(),
82                 uEditFetchStatCats(),
83                 uEditFetchSurveys(),
84                 uEditFetchNetLevels()
85                 );
86
87         if(patron.isnew()) {
88                 if(clone) uEditClone(clone);
89                 else uEditCreateNewAddr();
90
91         } else {
92
93                 $('ue_barcode').disabled = true;
94                 unHideMe($('ue_mark_card_lost'));
95                 unHideMe($('ue_reset_pw'));
96                 uEditCheckEditPerm();
97         }
98
99         if(PERMS['BAR_PATRON'] == -1) 
100                 $('ue_barred').disabled = true;
101 }
102
103
104 /* if this user does not have permission to put users into
105         the edited users group, they do not have permission to 
106         edit this user */
107 function uEditCheckEditPerm() {
108
109         var perm = uEditFindGroupPerm(groupsCache[patron.profile()]);   
110         _debug("editing user with group app perm "+patron.profile()+' : '+
111                 groupsCache[patron.profile()].name() +', and perm = ' + perm);
112
113         if(PERMS[perm] != -1) return;
114
115         /* we can edit our own account, but not others in our group */
116         if( patron.id() != USER.id() ){
117                 _debug("we are not allowed to edit this user");
118         
119                 $('ue_save').disabled = true;
120                 $('ue_save_clone').disabled = true;
121         
122                 uEditIterateFields(
123                         function(f) {
124                                 if( f && f.widget && f.widget.node )
125                                         f.widget.node.disabled = true;
126                         }       
127                 );      
128
129         }
130
131         var node = $('ue_profile').parentNode;
132         node.removeChild($('ue_profile'));
133         node.appendChild(elem('span',null,groupsCache[patron.profile()].name()));
134
135         var field = uEditFindFieldByKey('profile');
136         field.required = false;
137         uEditCheckErrors();
138 }
139
140
141 /* creates a new patron object with card attached */
142 var uEditCardVirtId = -1;
143 function uEditNewPatron() {
144         var patron = new au(); 
145         patron.isnew(1);
146         patron.id(-1);
147         card = new ac();
148         card.id(uEditCardVirtId--);
149         card.isnew(1);
150         patron.card(card);
151         patron.cards([card]);
152         patron.stat_cat_entries([]);
153         patron.survey_responses([]);
154         patron.addresses([]);
155         patron.home_ou(USER.ws_ou());
156         uEditMakeRandomPw(patron);
157         return patron;
158 }
159
160 function uEditMakeRandomPw(patron) {
161         var rand  = Math.random();
162         rand = parseInt(rand * 10000) + '';
163         while(rand.length < 4) rand += '0';
164         appendClear($('ue_password_plain'),text(rand));
165         unHideMe($('ue_password_gen'));
166         patron.passwd(rand);
167         return rand;
168 }
169
170 function uEditResetPw() { 
171         var pw = uEditMakeRandomPw(patron);     
172         $('ue_password1').value = pw;
173         $('ue_password2').value = pw;
174 }
175
176 function uEditClone(clone) {
177
178         var cloneUser = fetchFleshedUser(clone);
179         patron.usrgroup(cloneUser.usrgroup());
180
181         if( cloneUser.day_phone() )
182                 $('ue_day_phone').value = cloneUser.day_phone();
183         if( cloneUser.evening_phone() )
184                 $('ue_night_phone').value = cloneUser.evening_phone();
185         if( cloneUser.other_phone() )
186                 $('ue_other_phone').value = cloneUser.other_phone();
187         setSelector($('ue_org_selector'), cloneUser.home_ou());
188
189
190         setSelector($('ue_profile'), cloneUser.profile());
191
192         /* force the expire date to be set */
193         $('ue_profile').onchange();
194
195         for( var a in cloneUser.addresses() ) {
196                 var addr = cloneUser.addresses()[a];
197                 if( cloneUser.mailing_address && 
198                                 addr.id() == cloneUser.mailing_address().id() )
199                         patron.mailing_address(addr);
200                 if( cloneUser.billing_address() &&
201                                 addr.id() == cloneUser.billing_address().id() )
202                         patron.billing_address(addr);
203                 patron.addresses().push(addr);
204         }
205
206         uEditBuildAddrs(patron);
207 }
208
209
210 /* Creates a new blank address, 
211         adds it to the user and the fields array */
212 var uEditVirtualAddrId = -1;
213 function uEditCreateNewAddr() {
214         var addr = new aua();
215
216         addr.id(uEditVirtualAddrId--);
217         addr.isnew(1);
218         addr.usr(patron.id());
219         addr.country(defaultCountry);
220
221         if(!patron.addresses()) 
222                 patron.addresses([]);
223
224         if(patron.addresses().length == 0) {
225                 patron.mailing_address(addr);
226                 patron.billing_address(addr);
227         }
228
229         addr.valid(1);
230         addr.within_city_limits(1);
231
232         uEditBuildAddrFields(patron, addr);
233         patron.addresses().push(addr);
234         uEditIterateFields(function(f) { uEditCheckValid(f); });
235         uEditCheckErrors();
236 }
237
238
239 /* kicks off the UI drawing */
240 function uEditDraw(identTypes, groups, statCats, surveys, netLevels ) {
241         hideMe($('uedit_loading'));
242         unHideMe($('ue_maintd'));
243
244         dataFields = [];
245         uEditDrawIDTypes(identTypes);
246         uEditDrawGroups(groups);
247         uEditDrawStatCats(statCats);
248         uEditDrawSurveys(surveys);
249         uEditDrawNetLevels(netLevels);
250         uEditDefineData(patron);
251
252         uEditIterateFields(function(f) { uEditActivateField(f) });
253         uEditIterateFields(function(f) { uEditCheckValid(f); });
254         uEditCheckErrors();
255 }
256
257
258 /** Applies the event handlers and sets the data for the field */
259 function uEditActivateField(field) {
260
261         if( field.widget.id ) {
262                 field.widget.node = $(field.widget.id);
263
264         } else {
265                 field.widget.node = 
266                         $n(field.widget.base, field.widget.name);
267         }
268
269         uEditSetOnchange(field);
270
271         if(field.widget.onblur) {
272                 field.widget.node.onblur = 
273                         function() { field.widget.onblur(field); };
274         }
275
276         field.widget.node.disabled = field.widget.disabled;
277         if(field.object == null) return;
278         var val = field.object[field.key]();
279         if(val == null) return;
280
281         if( field.widget.type == 'input' )
282                 field.widget.node.value = val;
283
284         if( field.widget.type == 'select' )
285                 setSelector(field.widget.node, val);
286
287         if( field.widget.type == 'checkbox' )
288                 field.widget.node.checked = 
289                         (val && val != 'f') ? true : false;
290
291         if( field.widget.onload ) 
292                 field.widget.onload(val);
293 }
294
295
296 /* set up the onchange event for the field */
297 function uEditSetOnchange(field) {
298         var func = function() {uEditOnChange( field );}
299         field.widget.node.onchange = func;
300
301         if(field.widget.type != 'select')
302                 field.widget.node.onkeyup = func;
303 }
304
305 /* find the current value of the field object's widget */
306 function uEditNodeVal(field) {
307         if(field.widget.type == 'input')
308                 return field.widget.node.value;
309
310         if(field.widget.type == 'checkbox')
311                 return field.widget.node.checked;
312
313         if(field.widget.type == 'select')
314                 return getSelectorVal(field.widget.node);
315 }
316
317
318 /* update a field value */
319 function uEditOnChange(field) {
320
321         var newval = uEditNodeVal(field);
322         field.object[field.key](newval);
323         field.object.ischanged(1);
324
325         if(field.widget.onpostchange)
326                 field.widget.onpostchange(field, newval);
327
328
329         uEditIterateFields(function(f) { uEditCheckValid(f); });
330         uEditCheckErrors();
331 }
332
333
334 function uEditCheckValid(field) {
335         var newval = uEditNodeVal(field);
336
337         if(newval) {
338
339                 if(field.widget.regex) { 
340                         if(newval.match(field.widget.regex)) 
341                                 removeCSSClass(field.widget.node, CSS_INVALID_DATA);
342                         else
343                                 addCSSClass(field.widget.node, CSS_INVALID_DATA);
344
345                 } else {
346                         removeCSSClass(field.widget.node, CSS_INVALID_DATA);
347                 }
348
349         } else {
350
351                 if(field.required) {
352                         addCSSClass(field.widget.node, CSS_INVALID_DATA);
353
354                 } else {
355                         removeCSSClass(field.widget.node, CSS_INVALID_DATA);
356                 }
357         }
358 }
359
360 /* find a field object by object key */
361 function uEditFindFieldByKey(key) {
362         var fields = grep( dataFields,
363                 function(item) { return (item.key == key); });
364         return (fields) ? fields[0] : null;
365 }
366
367 /* find a list of fields by object key */
368 function uEditFindFieldsByKey(key) {
369         return grep( dataFields,
370                 function(item) { return (item.key == key); });
371 }
372
373 /* find a field object by widget id */
374 function uEditFindFieldByWId(id) {
375         var fields = grep( dataFields,
376                 function(item) { return (item.widget.id == id); });
377         return (fields) ? fields[0] : null;
378 }
379
380
381 function uEditIterateFields(callback) {
382         for( var f in dataFields ) 
383                 callback(dataFields[f]);
384 }
385
386
387 function uEditGetErrorStrings() {
388         var errors = [];
389         uEditIterateFields(
390                 function(field) { 
391                         if(field.errkey) {
392                                 if( field.widget.node.className.indexOf(CSS_INVALID_DATA) != -1) {
393                                         var str = $(field.errkey).innerHTML;
394                                         if(str) errors.push(str);
395                                 }
396                         }
397                 }
398         );
399
400         /* munge up something for all of the required surveys 
401                 (which are not registered with the fields) */
402         if( patron.isnew() ) {
403                 var sel = $('ue_survey_table');
404
405                 if( sel ) {
406                         var rows = sel.getElementsByTagName('tr');
407
408                         for( var r in rows ) {
409                 
410                                 var row = rows[r];
411                                 var sel = $n(row, 'ue_survey_answer');
412                                 if(!sel) continue;
413                                 var qstn = row.getAttribute('question');
414                 
415                                 if(qstn) {
416                                         qstn            = surveyQuestionsCache[qstn];
417                                         survey  = surveysCache[qstn.survey()];
418                                         var val = getSelectorVal(sel);
419                                         if(!val && isTrue(survey.required()))
420                                                 errors.push($('ue_bad_survey').innerHTML + ' : ' + qstn.question());
421                                 }
422                         }
423                 }
424         }
425
426         /* ------------------------------------------------------------ */
427
428         if(errors[0]) return errors;
429         return null;
430 }
431
432 function uEditAlertErrors() {
433         var errors = uEditGetErrorStrings();
434         if(!errors) return false;
435         alert(errors.join("\n"));
436         return true;
437 }
438
439
440 /* send the user to the database */
441 function uEditSaveUser(cloneme) {
442
443         if(uEditGetErrorStrings()) {
444                 uEditAlertErrors();
445                 return;
446         }
447
448         /* null is unique in the db, but '' is not */
449         if( ! patron.ident_value() ) patron.ident_value(null);
450         if( ! patron.ident_type2() ) patron.ident_type2(null);
451         if( ! patron.ident_value2() ) patron.ident_value2(null);
452
453         if(! patron.dob() ) patron.dob(null);
454
455         var req = new Request(UPDATE_PATRON, SESSION, patron);
456         req.alertEvent = false;
457         req.send(true);
458         var newuser = req.result();
459
460         var evt;
461         if( (evt = checkILSEvent(newuser)) || ! newuser ) {
462                 if(evt) alert(js2JSON(newuser));
463                 return;
464
465         } else {
466                 
467                 /* if it's a new user and a guardian note was created for this user,
468                         create the note after we have the new user's id */
469                 if( guardianNote ) {
470                         guardianNote.usr( newuser.id() );
471                         var req = new Request(CREATE_USER_NOTE, SESSION, guardianNote);
472                         req.alertEvent = false;
473                         req.send(true);
474                         var resp = req.result();
475                         if( checkILSEvent(resp) ) {
476                                 alertILSEvent(resp, 
477                                         "Error creating patron guardian/parent note");
478                                 return;
479                         }
480                 }
481
482                 alert($('ue_success').innerHTML);
483         }
484
485         if(cloneme) {
486                 /* if the user we just created was a clone, and we want to clone it,
487                 we really want to clone the original */
488                 if( clone ) cloneme = clone;
489                 else cloneme = newuser.id();
490         }
491
492         if (window.xulG && typeof window.xulG.on_save == 'function') {
493                 window.xulG.on_save(newuser, cloneme); 
494
495         } else {
496
497                 var href = location.href;
498
499                 href = href.replace(/\&?usr=\d+/, '');
500                 href = href.replace(/\&?clone=\d+/, '');
501
502                 if( cloneme ) href += '&clone=' + cloneme;
503                 location.href = href;
504         }
505 }
506
507 function uEditCancel() {
508         var href = location.href;
509         href = href.replace(/\&?usr=\d+/, '');
510         href = href.replace(/\&?clone=\d+/, '');
511         var id = cgi.param('usr')
512         /* reload the current user if available */
513         if( id ) href += "&usr=" + id;
514         location.href = href;
515 }
516
517
518 var uEditDupHashes = {};
519 var uEditDupTemplate;
520
521 function uEditRunDupeSearch(type, search_hash) {
522
523         if(!patron.isnew()) return;
524
525         _debug('dup search: ' + js2JSON(search_hash));
526
527         var req = new Request(PATRON_SEARCH, SESSION, search_hash);
528
529         var container = $('dup_div_container');
530         if(!uEditDupTemplate)
531                 uEditDupTemplate = container.removeChild($('dup_div'));
532
533         /* clear any existing dups for this type */
534         iterate( container.getElementsByTagName('div'),
535                 function(d) {
536                         if( d.getAttribute('type') == type ) {
537                                 container.removeChild(d)
538                                 return;
539                         }
540                 }
541         );
542
543         req.callback(
544                 function(r) {
545                         uEditHandleDupResults( r.getResultObject(), search_hash, type, container );
546                 }
547         );
548         req.send();
549 }
550
551
552 function uEditHandleDupResults(ids, search_hash, type, container) {
553
554         _debug('dup search results: ' + js2JSON(ids));
555
556         if(!(ids && ids[0]))  /* no results */
557                 return uEditDupHashes[type] = null;
558
559         /* add a dup link to the UI and plug in the data */
560         var node = uEditDupTemplate.cloneNode(true);
561         container.appendChild(node);
562         node.setAttribute('type', type);
563
564         var link = $n(node, 'link');
565         link.setAttribute('type', type);
566         unHideMe(link);
567         $n(node,'count').appendChild(text(ids.length));
568
569         for( var o in search_hash ) 
570                 $n(node, 'data').appendChild(
571                         text(search_hash[o].value + ' '));
572
573         uEditDupHashes[type] = search_hash;
574
575         switch(type) {
576                 case 'ident' :
577                         if(confirm($('ue_dup_ident1').innerHTML)) 
578                                 uEditShowSearch(null, type);
579                         break;
580         }
581 }
582
583
584 function uEditShowSearch(link,type) {
585         if(!type) type = link.getAttribute('type');
586         if(window.xulG)
587                 window.xulG.spawn_search(uEditDupHashes[type]); 
588         else alert('Search would be:\n' + js2JSON(uEditDupHashes[type]));
589 }
590
591 function uEditMarkCardLost() {
592
593         for( var c in patron.cards() ) {
594
595                 var card = patron.cards()[c];
596                 if( patron.card().id() == card.id() ) {
597
598                         /* de-activite the current card */
599                         card.ischanged(1);
600                         card.active(0);
601
602                         /* create a new card for the patron */
603                         var newcard = new ac();
604                         newcard.id(uEditCardVirtId--);
605                         newcard.isnew(1);
606                         patron.card(newcard);
607                         patron.cards().push(newcard);
608
609
610                         /* reset the widget */
611                         var field = uEditFindFieldByWId('ue_barcode');
612                         field.widget.node.disabled = false;
613                         field.widget.node.value = "";
614                         field.widget.node.onchange();
615                         field.object = newcard;
616                 }
617         }
618 }
619
620
621 function compactArray(arr) {
622         var a = [];
623         for( var i = 0; arr && i < arr.length; i++ ) {
624                 if( arr[i] != null )
625                         a.push(arr[i]);
626         }
627         return a;
628 }