status bitfield moved from 0x0f00 to 0x00ff (from 16 to 255 possible status)
[brisk.git] / web / room.js
1 /*
2  *  brisk - room.js
3  *
4  *  Copyright (C) 2006-2012 Matteo Nastasi
5  *                          mailto: nastasi@alternativeoutput.it 
6  *                                  matteo.nastasi@milug.org
7  *                          web: http://www.alternativeoutput.it
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 2 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful, but
15  * WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABLILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17  * General Public License for more details. You should have received a
18  * copy of the GNU General Public License along with this program; if
19  * not, write to the Free Software Foundation, Inc, 59 Temple Place -
20  * Suite 330, Boston, MA 02111-1307, USA.
21  *
22  */
23
24
25 /* 
26
27    data = [ [ flags, flags_vlt, name ],  ... ]
28    
29 */
30
31
32 function state_add(flags, flags_vlt, comp)
33 {
34     var content = "", supercont = "";
35     var st, superst, name = "", supername = "", supersfx = "";
36     var tit = "", supertit = "";
37
38
39     if ((flags_vlt & 0xff) != 0) {
40         st = flags_vlt & 0xff;
41         // MLANG 4,12,16,20,24,28
42         switch (st) {
43         case 0x01:
44             name = "st_pau.png";
45             tit = (g_lang == 'en' ? "I'm doing a break" : "sono in pausa");
46             break;
47         case 0x02:
48             name = "st_out.png";
49             tit = (g_lang == 'en' ? "I'm away" : "sono fuori");
50             break;
51         case 0x03:
52             name = "st_dog.png";
53             tit = (g_lang == 'en' ? "Dog time" : "sono a spasso col cane");
54             break;
55         case 0x04:
56             name = "st_eat.png";
57             tit = (g_lang == 'en' ? "I'm eating" : "sto mangiando");
58             break;
59         case 0x05:
60             name = "st_wrk.png";
61             tit = (g_lang == 'en' ? "I'm working" : "sono a lavoro");
62             break;
63         case 0x06:
64             name = "st_smk.png";
65             tit = (g_lang == 'en' ? "I'm smoking a sigarett (and keeping a cancer)" : "sto fumando una sigaretta (e facendomi venire il cancro)");
66             break;
67         case 0x07:
68             name = "st_eye.png";
69             tit = (g_lang == 'en' ? "I'm here!" : "sono presente!");
70             break;
71         case 0x08:
72             name = "st_rabbit.png";
73             tit = (g_lang == 'en' ? "Rabbit time" : "sono a spasso col coniglio");
74             break;
75         case 0x09:
76             name = "st_soccer.png";
77             tit = (g_lang == 'en' ? "Soccer time" : "c'è la partita!!");
78             break;
79         case 0x0a:
80             name = "st_baby.png";
81             tit = (g_lang == 'en' ? "Children time" : "ho il pupo da accudire");
82             break;
83         case 0x0b:
84             name = "st_mop.png";
85             tit = (g_lang == 'en' ? "Mop time" : "sto rassettando");
86             break;
87         case 0x0c:
88             name = "st_babbo.png";
89             tit = (g_lang == 'en' ? "Sto dando i regali" : "sto dando i regali");
90             break;
91         case 0x0d:
92             name = "st_renna.png";
93             tit = (g_lang == 'en' ? "in giro per regali" : "in giro per regali");
94             break;
95         case 0x0e:
96             name = "st_pupaz.png";
97             tit = (g_lang == 'en' ? "Neve a gogò" : "neve a gogò");
98             break;
99         case 0x0f:
100             name = "st_visch.png";
101             tit = (g_lang == 'en' ? "aspettando sotto al vischio" : "aspettando sotto al vischio");
102             break;
103         default:
104             break;
105         }
106     }
107
108     if ((flags & 0xf0000) != 0) {
109         superst = flags & 0xf0000;
110         if (name != "") {
111             supersfx = "_side";                
112         }
113
114         switch (superst) {
115         case 0x20000:
116             if (comp != null) {
117                 supername = "suprend.php?comp="+comp+"&sfx="+supersfx;
118             }
119             else {
120                 supername = "img/superuser"+supersfx+".png";
121             }
122             supertit = (g_lang == 'en' ? "Brisk Supporter" : "Brisk Supporter");
123             break;
124         }
125     }
126
127     if (supername != "") {
128         content += '&nbsp;<img title="'+supertit+'" class="inline" src="'+supername+'">';
129     }
130     
131     if (name != "") {
132         content += '&nbsp;<img title="'+tit+'" class="inline" src="img/'+name+'">';
133     }
134
135     return content;
136 }
137
138 var standup_data_old = null;
139
140 // TODO !!
141 // appendChild , removeChild
142
143 function table_add(curtag, td)
144 {
145     var tbody  = null, tr, ct;
146
147     do {
148         // console.log("wt: "+curtag.tagName);
149
150         if (curtag.tagName.toLowerCase() == "div" || 
151             curtag.tagName.toLowerCase() == "table") {
152             curtag = curtag.firstChild;
153         }
154         else if (curtag.tagName.toLowerCase() == "tbody") {
155             tbody = curtag;
156             break;
157         }
158         else
159             curtag = null;
160     } while (curtag != null);
161     
162     curtag = tbody.firstChild;
163     ct = 0;
164     do {
165         if (curtag.tagName.toLowerCase() == "tr") {
166             if (curtag.firstChild != null) {
167                 curtag = curtag.firstChild;
168                 ct++;
169             }
170             else {
171                 curtag.appendChild(td);
172                 return(true);
173             }
174         }
175         else if (curtag.tagName.toLowerCase() == "td") {
176             if (curtag.nextSibling != null) {
177                 curtag = curtag.nextSibling;
178                 ct++;
179             }
180             else {
181                 if (ct < 4) {
182                     curtag.parentNode.appendChild(td);
183                     return (true);
184                 }
185                 else {
186                     ct = 0;
187                     curtag = curtag.parentNode.nextSibling;
188                 }
189             }
190         }
191         else {
192             curtag = curtag.parentNode;
193         }
194
195     } while (curtag != null);
196
197     tr = document.createElement("tr");
198     tr.appendChild(td);
199     tbody.appendChild(tr);
200
201     return (true);
202 }
203
204 function spcs(c1, c2, n)
205 {
206     var ret = "";
207     var i;
208
209     for (i = 0 ; i < n ; i++) {
210         if ((i % 2) == 0)
211             ret += c1;
212         else
213             ret += c2;
214     }
215
216     return (ret);
217 }
218
219
220 function table_walk(curtag)
221 {
222     do {
223         // console.log("wt: "+curtag.tagName);
224         if (curtag.tagName.toLowerCase() == "div" || 
225             curtag.tagName.toLowerCase() == "table" ||
226             curtag.tagName.toLowerCase() == "tbody") {
227             curtag = curtag.firstChild;
228         }
229         else if (curtag.tagName.toLowerCase() == "tr") {
230             if (curtag.firstChild != null)
231                 curtag = curtag.firstChild;
232             else if (curtag.tagName != '')
233                 curtag = curtag.nextSibling;
234             else
235                 curtag = null;
236         }
237         else if (curtag.tagName.toLowerCase() == "td") {
238             if (curtag.nextSibling != null)
239                 curtag = curtag.nextSibling;
240             else {
241                 if (curtag.parentNode.nextSibling != null && curtag.parentNode.nextSibling.tagName != '')
242                     curtag = curtag.parentNode.nextSibling;
243                 else
244                     curtag = null;
245             }
246         }
247         else
248             curtag = null;
249
250     } while (curtag != null && curtag.tagName.toLowerCase() != "td");
251
252     if (1 == 0) {
253         if (curtag == null)
254             alert("outtag == null"); 
255         else
256             alert("outtag: "+curtag.tagName);
257     }
258     return (curtag);
259 }
260
261 function j_stand_tdcont(el)
262 {
263     return (user_dec_and_state(el));
264 }
265
266 /*
267   ddata = [ [ <flags-int>, <flags-vlt-int>, <nick-str>, <color-str> ], ... ]
268  */
269 function j_stand_cont(ddata)
270 {
271     var i, ii;
272     var content;
273     var st = 0, name = "";
274     var curtag, nextag;
275
276     var data;
277
278     if (g_listen & l_list_isol) {
279         data = new Array();
280
281         for (i = 0, ii = 0 ; ii < ddata.length ; ii++) {
282             if ((ddata[ii][BSK_USER_FLAGS] & 0x02) == 0) {
283                 continue;
284             }
285             data[i++] = ddata[ii];
286         }
287     }
288     else
289         data = ddata;
290
291     // WARNING:
292     //
293     //   managing update needs this branch (for few users and the else!!)
294     //
295     if (standup_data_old == null || data.length < 4) {
296         content = '<table cols="'+(data.length < 4 ? data.length : 4)+'" class="table_standup">';
297         for (i = 0 ; i < data.length ; i++) {
298             if ((i % 4) == 0)
299                 content += '<tr>';
300             content += '<td id="'+i+'" class="room_standup">';
301             content += j_stand_tdcont(data[i]);
302             content += '</td>';
303             
304             if ((i % 4) == 3)
305                 content += '</tr>';
306         }
307         if ((i % 4) < 3)
308             content += '</tr>';
309         content += '</table>';
310         
311         $("standup").innerHTML = content;
312
313         standup_data_old = data;
314     }
315     else {
316         var idx_del, arr_add, idx_mod, arr_mod;
317         var idx_del_n = 0, idx_add_n = 0, idx_mod_n = 0;
318         var i, e;
319         var i_del, i_mod, i_add;
320         var td;
321
322         idx_del = new Array();
323         arr_add = new Array();
324         map_add = new Array();
325         idx_mod = new Array();
326         arr_mod = new Array();
327         map_cur = new Array();
328         
329         // find removed entries
330         for (i = 0 ; i < standup_data_old.length ; i++) {
331             for (e = 0 ; e < data.length ; e++) {
332                 if (standup_data_old[i][BSK_USER_NICK] == data[e][BSK_USER_NICK]) {
333                     break;
334                 }
335             }
336             if (e == data.length) {
337                 idx_del[idx_del_n++] = i;
338                 map_cur[i] = -1;
339             }
340             else {
341                 /* modified entries */
342                 if (standup_data_old[i][BSK_USER_FLAGS] != data[e][BSK_USER_FLAGS] ||
343                     standup_data_old[i][BSK_USER_FLGVL] != data[e][BSK_USER_FLGVL] ||
344                     standup_data_old[i].length != data[e].length ||
345                     (data[e].length == 4 && standup_data_old[i][BSK_USER_SCOL] != data[e][BSK_USER_SCOL])) {
346                     arr_mod[idx_mod_n] = data[e];
347                     idx_mod[idx_mod_n++] = i;
348                 }
349                 map_cur[i] = e;
350             }
351         }
352
353         // find new entries
354         for (e = 0 ; e < data.length ; e++) {
355             for (i = 0 ; i < standup_data_old.length ; i++) {
356                 if (data[e][BSK_USER_NICK] == standup_data_old[i][BSK_USER_NICK] ) {
357                     break;
358                 }
359             }
360             if (i == standup_data_old.length) {
361                 // console.log("ADD: "+data[e][BSK_USER_NICK]);
362                 arr_add[idx_add_n]   = data[e];
363                 map_add[idx_add_n++] = e;
364             }
365         }
366         
367         // TODO: qui travaso add in del
368
369         i_del = 0;
370         // alert("del: ["+j_stand_tdcont(standup_data_old[idx_del[i_del]])+"]");
371         for (i = 0 , i_del = 0, i_mod = 0, i_add = 0, curtag = table_walk($("standup")) ; curtag != null ;  curtag = table_walk(curtag), i++ ) {
372             // console.log("cur.id: "+curtag.id);
373
374             // alert("i: "+i+"  tagname: "+curtag.tagName+"  innerHTML: ["+curtag.innerHTML+"]");
375             // console.log("inloop["+i+"]: "+curtag.tagName+"  ID: "+curtag.id);
376             if (curtag.innerHTML == "") {
377                 // console.log("innerHTML == none");
378                 if (i_add < idx_add_n) {
379                     // console.log("  to be new");
380                     // console.log("  add:   CONT:"+j_stand_tdcont(arr_add[i_add]));
381                     curtag.innerHTML = j_stand_tdcont(arr_add[i_add]);
382                     curtag.id = map_add[i_add];
383                     i_add++
384                 }
385             }
386
387             // else if (i_del < idx_del_n && curtag.innerHTML == j_stand_tdcont(standup_data_old[idx_del[i_del]])) {
388             else if (i_del < idx_del_n && curtag.id == idx_del[i_del]) {
389                 // console.log("to be cancel["+i+"]:  ID: "+curtag.id);
390                 if (i_add < idx_add_n) {
391                     // console.log("  to be new");
392                     // console.log("  add:   CONT:"+j_stand_tdcont(arr_add[i_add]));
393                     curtag.innerHTML = j_stand_tdcont(arr_add[i_add]);
394                     curtag.id = map_add[i_add];
395                     i_add++
396                 }
397                 else {
398                     // console.log("  to be del");
399                     curtag.innerHTML = "";
400                     curtag.id = -1;
401                 }
402                 i_del++;
403             }
404             // else if (i_mod < idx_mod_n && curtag.innerHTML == j_stand_tdcont(standup_data_old[idx_mod[i_mod]])) {
405             else if (i_mod < idx_mod_n && curtag.id == idx_mod[i_mod]) {
406                 // console.log("  to be mod");
407                 // console.log("mod: "+idx_mod[i_mod]+ "  CONT:"+j_stand_tdcont(arr_mod[i_mod]));
408                 curtag.innerHTML = j_stand_tdcont(arr_mod[i_mod]);
409                 curtag.id = map_cur[curtag.id];
410                 i_mod++;
411             }
412             else
413                 curtag.id = map_cur[curtag.id];
414         }
415         // console.log("fineloop");
416
417         for (i ; i_add < idx_add_n ; i_add++, i++) {
418             // console.log("ADD: "+i+" arr_add: "+ arr_add[i_add][BSK_USER_NICK]);
419             td = document.createElement("td");
420             td.className = "room_standup";
421             td.id = map_add[i_add];
422             td.innerHTML = j_stand_tdcont(arr_add[i_add]);
423
424             table_add($("standup"), td);
425         }
426
427         standup_data_old = data;
428         return;
429     }
430     // $("esco").innerHTML =  '<input class="button" name="logout" value="Esco." onclick="esco_cb();" type="button">';
431 }
432
433 function esco_cb() {
434     window.onbeforeunload = null; 
435     window.onunload = null; 
436     // nonunload = true; 
437     act_logout(0);
438  };
439
440
441
442 function j_tab_cont(table_idx, data)
443 {
444     var i;
445     var content = '';
446
447     for (i = 0 ; i < data.length ; i++) {
448         content += j_stand_tdcont(data[i]);
449
450         content += '<br>';
451     }
452     $("table"+table_idx).innerHTML = content;
453 }
454
455 function j_tab_act_cont(idx, act)
456 {
457     if (act == 'sit') {
458         // MLANG 1
459         $("table_act"+idx).innerHTML = '<input type="button" class="button" name="xhenter'+idx+'"  value="'+(g_lang == 'en' ? "Sit down." : "Mi siedo.")+'" onclick="act_sitdown('+idx+');">';
460     }
461     else if (act == 'sitreser') {
462         // <img class="nobo" title="tavolo riservato agli utenti registrati" style="display: inline; margin-right: 80px;" src="img/okauth.png">
463         // MLANG 1
464         $("table_act"+idx).innerHTML = '<input type="button" style="background-repeat: no-repeat; background-position: center; background-image: url(\'img/okauth.png\');" class="button" name="xhenter'+idx+'"  value="'+(g_lang == 'en' ? "Sit down." : "Mi siedo.")+'" onclick="act_sitdown('+idx+');">';
465     }
466     else if (act == 'wake') {
467         // MLANG 1
468         $("table_act"+idx).innerHTML = '<input type="button" class="button" name="xwakeup"  value="'+(g_lang == 'en' ? "Wake up." : "Mi alzo.")+'" onclick="act_wakeup();">';
469     }
470     else if (act == 'reserved') {
471         // MLANG 1
472         $("table_act"+idx).innerHTML = '<img class="nobo" title="'+(g_lang == 'en' ? "reserved table for authenticated users only" : "tavolo riservato agli utenti registrati")+'" style="margin-right: 20px;" src="img/onlyauth.png">';
473     }
474     else {
475         $("table_act"+idx).innerHTML = '';
476     }
477 }
478
479 function j_login_manager(form)
480 {
481     var token;
482
483     if (form.elements['passid'].value == '')
484         return (true);
485
486     else {
487         // console.log("richiesta token");
488         /* richiede token */
489         token = server_request('mesg', 'getchallenge', 'cli_name', encodeURIComponent(form.elements['nameid'].value));
490         tokens = token.split('|');
491         
492         // console.log('XX token: '+token);
493         // console.log(tokens);
494         if (token == null)
495             return (false);
496
497         token = calcMD5(tokens[1]+calcMD5(form.elements['passid'].value));
498         
499         form.elements['passid_private'].value = token;
500         form.elements['passid'].value = ""; // FIXME da sost con la stessa len di A
501
502         return (true);
503     }
504     
505     return (false);
506 }
507
508 function login_formtext_hilite()
509 {
510     formtext_hilite($("nameid"));
511     formtext_hilite($("passid"));
512     formsub_hilite($("sub"));
513 }
514
515 function login_init()
516 {
517     menu_init();
518     login_formtext_hilite();
519 }
520
521 function warrant_formtext_hilite(form)
522 {
523     /*
524     formtext_hilite($("nameid"));
525     formtext_hilite($("emailid"));
526     formsub_hilite($("subid"));
527     formsub_hilite($("cloid"));
528     */
529     formtext_hilite(form.elements['name']);
530     formtext_hilite(form.elements['email']);
531     formsub_hilite(form.elements['sub']);
532     formsub_hilite(form.elements['clo']);
533 }
534
535 function mesgtoadm_formtext_hilite(form)
536 {
537     /*
538     formtext_hilite($("subjid"));
539     formtext_hilite($("mesgid"));
540     formsub_hilite($("subid"));
541     formsub_hilite($("cloid"));
542     */
543     formtext_hilite(form.elements['subj']);
544     formtext_hilite(form.elements['mesg']);
545     formsub_hilite(form.elements['sub']);
546     formsub_hilite(form.elements['clo']);
547 }
548
549
550 function j_check_email(email)
551 {
552     if (/^\w+([\.-]?\w+)*@\w+([\.-]?\w+)*(\.\w{2,3})+$/.test(email))
553         return (true);
554     return (false);
555 }
556
557 function j_authbox(form)
558 {
559     var no; 
560
561     do {
562         if (form.elements['realsub'].value == "chiudi") {
563             $('authbox').style.visibility = "hidden";
564             break;
565         }
566
567         if (form.elements['name'].value == '' || j_check_email(form.elements['email'].value) == false) {
568             // MLANG 2-4
569             no = new notify(gst, 
570                             (g_lang == 'en' ? "<br><b>nickname</b> and/or <b>e-mail</b> fields are invalid;<br>please, fix them." :
571                              "<br>I campi <b>nickname</b> e/o <b>e-mail</b> non sono validi;<br> correggeteli per favore."),
572                             1, (g_lang == 'en' ? "close" : "chiudi"), 280, 100); 
573             break;
574         }
575
576         // submit the request
577         token = server_request('mesg', 'warranty', 
578                                'cli_name', encodeURIComponent(form.elements['name'].value),
579                                'cli_email', encodeURIComponent(form.elements['email'].value) );
580         if (token == "1") {
581             $('authbox').style.visibility = "hidden";
582             form.elements['name'].value = "";
583             form.elements['email'].value = "";
584             break;
585         }
586     } while (0);
587
588     return (false);
589 }
590
591 function authbox(w, h)
592 {
593     var box;
594
595     box = $('authbox');
596
597     box.style.zIndex = 200;
598     box.style.width  = w+"px";
599     box.style.marginLeft  = -parseInt(w/2)+"px";
600     box.style.height = h+"px";
601     box.style.top = parseInt((document.body.clientHeight - h) / 2) + document.body.scrollTop;
602
603     warrant_formtext_hilite($('auth_form'));
604
605     box.style.visibility = "visible";
606     $("nameid").focus();
607 }
608
609 function j_mesgtoadmbox(form)
610 {
611     var no; 
612
613     do {
614         if (form.elements['realsub'].value == "chiudi") {
615             $('mesgtoadmbox').style.visibility = "hidden";
616             break;
617         }
618
619         if (form.elements['mesg'].value == '' || form.elements['subj'].value == '') {
620             // MLANG 1-3
621             no = new notify(gst, (g_lang == 'en' ? "<br><b>subject</b> and the <b>message</b> cannot be void;<br>please, fix them." :
622                                   "<br>Il <b>soggetto</b> e il <b>messaggo</b> non possono essere vuoti;<br>correggeteli per favore."), 1, 
623                                   (g_lang == 'en' ? "close" : "chiudi"), 280, 100); 
624             break;
625         }
626                 
627         // submit the request
628         token = server_request('mesg', 'mesgtoadm', 
629                                'cli_subj', encodeURIComponent(form.elements['subj'].value),
630                                'cli_mesg', encodeURIComponent(form.elements['mesg'].value) );
631         if (token == "1") {
632             $('mesgtoadmbox').style.visibility = "hidden";
633             form.elements['subj'].value = "";
634             form.elements['mesg'].value = "";
635             break;
636         }
637     } while (0);
638
639     return (false);
640 }
641
642 function mesgtoadmbox(w, h)
643 {
644     var box;
645
646     box = $('mesgtoadmbox');
647
648     box.style.zIndex = 200;
649     box.style.width  = w+"px";
650     box.style.marginLeft  = -parseInt(w/2)+"px";
651     box.style.height = h+"px";
652     box.style.top = parseInt((document.body.clientHeight - h) / 2) + document.body.scrollTop;
653
654     mesgtoadm_formtext_hilite($('mesgtoadm_form'));
655
656     box.style.visibility = "visible";
657     $('mesgtoadm_form').elements['subj'].focus();
658 }
659
660 function j_pollbox(form)
661 {
662     var no, i, choose; 
663
664     do {
665         // submit the request
666         
667         for (i = 0 ; i < form.elements.length ; i++) {
668             if (form.elements[i].checked == true)
669                 break;
670         }
671         if (i == form.elements.length) {
672             // MLANG 1-3
673             no = new notify(gst, (g_lang == 'en' ? "<br>You must choose ah item;<br> please, fix it." :
674                                   "<br>Non hai espresso nessuna preferenza;<br> correggi per favore."), 1, 
675                             (g_lang == 'en' ? "close" : "chiudi"), 280, 100); 
676             return false;
677         }
678         else
679             choose = form.elements[i].value;
680
681         token = server_request('mesg', 'poll', 
682                                'cli_choose', encodeURIComponent(choose) );
683
684         if (token == "1") {
685             // TODO: mesg to user
686             // $('mesgtoadmbox').style.visibility = "hidden";
687             break;
688         }
689     } while (0);
690
691     return (false);
692 }
693
694
695 function sideslide(domobj, height, step)
696 {
697     this.st = 'wait';
698     this.twait = 5000;
699
700     this.domobj = domobj;
701     this.height = height;
702     this.step = step;
703
704     this.start();
705 }
706
707 sideslide.prototype = {
708     id: null,
709     st: 'wait',
710     twait: 0,
711     scroll: 0,
712     countdown: 0,
713
714     domobj: null,
715     height: 0,
716     step: 0,
717
718     start: function() {
719         var instant = this;
720         
721         this.st = 'wait';
722         this.id = setTimeout(function () { instant.sideslide_cb(); }, this.twait);
723     },
724
725     sideslide_cb: function() {
726         var instant = this;
727
728         if (this.st == 'wait') {
729             this.st = 'scroll';
730             this.countdown = 10;
731             this.id = setInterval(function () { instant.sideslide_cb(); }, 100);
732         }
733         else if (this.st == 'scroll') {
734             this.scroll += (this.step / 10);
735             if (this.scroll >= this.height - this.step) {
736                 this.scroll = 0;
737             }
738             this.domobj.scrollTop = this.scroll;
739             this.countdown--;
740             if (this.countdown == 0) {
741                 this.stop();
742                 this.st = 'wait';
743                 this.id = setTimeout(function () { instant.sideslide_cb(); }, this.twait);
744             }
745         }
746     },
747
748
749     stop: function() {
750         if (this.id != null) {
751             clearInterval(this.id);
752             this.id = null;
753         }
754     }
755
756 }