manage list update (remove too old lines)
[brisk.git] / web / commons.js
1 /*
2  *  brisk - commons.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 var PLAYERS_N = 3;
25 var EXIT_BAN_TIME = 3600;
26 var cookiepath = "/brisk/";
27
28 var BSK_USER_FLAGS = 0;
29 var BSK_USER_FLGVL = 1;
30 var BSK_USER_NICK  = 2;
31 var BSK_USER_SCOL  = 3;
32
33 var mlang_commons = { 'imgload_a' : { 'it' : 'Immagine caricate ', 
34                                       'en' : 'Loaded images ' },
35                       'imgload_b' : { 'it' : '%.', 
36                                       'en' : '%.' },
37                       'gamleav'   : { 'it' : 'Sei sicuro di volere lasciare questa mano?' ,
38                                       'en' : 'Are you sure to leave this game?' },
39                       'brileav'   : { 'it' : '    Vuoi veramente abbandonare la briscola ?\n(clicca annulla o cancel se vuoi ricaricare la briscola)',
40                                       'en' : '    Are you really sure to leave briscola ?\n(click cancel yo reload it)' },
41                       'brireco'   : { 'it' : 'Ripristino della briscola fallito, per non perdere la sessione ricaricare la pagina manualmente.',
42                                       'en' : 'Recovery of briscola failed, to keep the current session reload the page manually.' },
43                       'btn_sit'   : { 'it' : 'Mi siedo.',
44                                       'en' : 'Sit down.' },
45                       'btn_exit'  : { 'it' : 'Esco.',
46                                       'en' : 'Exit.' },
47                       'tit_list'  : { '0'  : { 'it' : '',
48                                                'en' : '' },
49                                       '1'  : { 'it' : '(solo aut.)',
50                                                'en' : '(only aut.)' },
51                                       '2'  : { 'it' : '(isolam.to)',
52                                                'en' : '(isolation)' } }
53                     };
54
55 function $() {
56     if (arguments.length == 1)
57         return document.getElementById(arguments[0]);
58     else
59         return (arguments[0]).document.getElementById((arguments[1]));
60 }
61
62 function dec2hex(d, padding)
63 {
64     var hex = Number(d).toString(16);
65     padding = typeof (padding) === "undefined" || padding === null ? padding = 2 : padding;
66
67     while (hex.length < padding) {
68         hex = "0" + hex;
69     }
70
71     return hex;
72 }
73
74 function getStyle(x,IEstyleProp, MozStyleProp) 
75 {
76     if (x.currentStyle) {
77         var y = x.currentStyle[IEstyleProp];
78     } else if (window.getComputedStyle) {
79         var y = document.defaultView.getComputedStyle(x,null).getPropertyValue(MozStyleProp);
80     }
81     return y;
82 }
83
84 /* replacement of setInterval on IE */
85 (function(){
86     /*if not IE, do nothing*/
87     if(!document.uniqueID){return;};
88
89     /*Copy the default setInterval behavior*/
90     var nativeSetInterval = window.setInterval;
91     window.setInterval = function(fn,ms) {              
92         var param = [];
93         if(arguments.length <= 2)       {
94             return nativeSetInterval(fn,ms);
95         }
96         else {
97             for(var i=2;i<arguments.length;i+=1) {
98                 param[i-2] =  arguments[i];
99             }   
100         }
101         
102         if(typeof(fn)=='function') {
103             
104             return (function (fn,ms,param) {
105                 var fo = function () {                                                          
106                     fn.apply(window,param);
107                 };                      
108                 return nativeSetInterval(fo,ms); 
109             })(fn,ms,param);
110         }
111         else if(typeof(fn)=='string')
112         {
113             return  nativeSetInterval(fn,ms);
114         }
115         else
116         {
117             throw Error('setInterval Error\nInvalid function type');
118         };
119     };
120
121     /*Copy the default setTimeout behavior*/
122     var nativeSetTimeout = window.setTimeout;
123     window.setTimeout = function(fn,ms) {               
124         var param = [];
125         if(arguments.length <= 2)       {
126             return nativeSetTimeout(fn,ms);
127         }
128         else {
129             for(var i=2;i<arguments.length;i+=1) {
130                 param[i-2] =  arguments[i];
131             }   
132         }
133         
134         if(typeof(fn)=='function') {
135             
136             return (function (fn,ms,param) {
137                 var fo = function () {                                                          
138                     fn.apply(window,param);
139                 };                      
140                 return nativeSetTimeout(fo,ms); 
141             })(fn,ms,param);
142         }
143         else if(typeof(fn)=='string')
144         {
145             return  nativeSetTimeout(fn,ms);
146         }
147         else
148         {
149             throw Error('setTimeout Error\nInvalid function type');
150         };
151     };
152
153 })()
154
155 function addEvent(obj,type,fn)
156 {
157     if (obj.addEventListener) {
158         obj.addEventListener( type, fn, false);
159     }
160     else if (obj.attachEvent) {
161         obj["e"+type+fn] = fn;
162         obj[type+fn] = function() { obj["e"+type+fn]( window.event ); }
163         obj.attachEvent( "on"+type, obj[type+fn] );
164     }
165     else
166         throw new Error("Event registration not supported");
167 }
168
169 function removeEvent(obj,type,fn)
170 {
171     if (obj.removeEventListener) {
172         obj.removeEventListener( type, fn, false );
173     }
174     else if (obj.detachEvent) {
175         obj.detachEvent( "on"+type, obj[type+fn] );
176         obj[type+fn] = null;
177         obj["e"+type+fn] = null;
178     }
179 }
180
181     // var card_pos = RANGE 0 <= x < cards_ea_n
182
183 function show_bigpict(obj, act, x, y)
184 {
185    var big, sfx;
186
187    if (arguments.length > 4)
188        sfx = arguments[4];
189    else
190        sfx = '';
191
192    big = $(obj.id+"_big"+sfx);
193    if (act == "over") {
194        big.style.left = obj.offsetLeft + x+"px";
195        big.style.top  = obj.offsetTop  + y+"px";
196        big.style.visibility = "visible";
197        }
198    else {
199        big.style.visibility = "hidden";
200        }
201 }
202
203 function rnd_int(min, max) {
204   return Math.floor(Math.random() * (max - min + 1) + min);
205 }
206
207 function error_images()
208 {
209     // alert("GHESEMU!");
210     setTimeout(preload_images, 2000, g_preload_img_arr, g_imgct-1);
211 }
212
213 function abort_images()
214 {
215     // alert("ABORTAIMAGES");
216     setTimeout(preload_images, 2000, g_preload_img_arr, g_imgct-1);
217 }
218
219 function unload_images()
220 {
221     // alert("ABORTAIMAGES");
222     setTimeout(preload_images, 2000, g_preload_img_arr, g_imgct-1);
223 }
224
225 function reset_images()
226 {
227     // alert("ABORTAIMAGES");
228     setTimeout(preload_images, 2000, g_preload_img_arr, g_imgct-1);
229 }
230
231 function update_images()
232 {
233     // MLANG "Immagine caricate" + g_preload_imgsz_arr[g_imgct] + "%."
234     $("imgct").innerHTML = mlang_commons['imgload_a'][g_lang]+g_preload_imgsz_arr[g_imgct]+"%.";
235     if (g_imgct+1 < g_preload_img_arr.length) {
236         g_imgct++;
237         setTimeout(preload_images, 100, g_preload_img_arr, g_imgct-1);
238     }
239     // $("imgct").innerHTML += "U";
240 }
241
242 function preload_images(arr,idx)
243 {
244     var im = new Image;
245     
246     // $("imgct").innerHTML = "Stiamo caricando "+arr[idx]+"%.<br>";
247     im.onload =   update_images;
248     im.onerror =  error_images;
249     im.onabort =  abort_images;
250     im.onunload = unload_images;
251     im.onreset =  reset_images;
252     im.src =      arr[idx];
253     // $("imgct").innerHTML += "P";
254 }
255
256 function safestatus(a)
257 {
258     try{
259         return (a.status);
260     } catch(b)
261         { return (-1); }
262 }
263
264 function createXMLHttpRequest() {
265     if (typeof(ActiveXObject) != 'undefined') { // Konqueror complain as unknown object
266         try { return new ActiveXObject("Msxml2.XMLHTTP");    } catch(e) {}
267         try { return new ActiveXObject("Microsoft.XMLHTTP"); } catch(e) {}
268     }
269     try { return new XMLHttpRequest();                   } catch(e) {}
270     alert("XMLHttpRequest not supported");
271     return null;
272 }
273
274 function send_mesg(mesg)
275 {
276     var xhr_wr = createXMLHttpRequest();
277     var is_conn = (sess == "not_connected" ? false : true);
278     
279     // alert("xhr_wr: "+xhr_wr+"  is_conn: "+is_conn);
280     xhr_wr.open('GET', 'index_wr.php?'+(is_conn ? 'sess='+sess+'&' : '')+'mesg='+mesg, (is_conn ? true : false));
281     xhr_wr.setRequestHeader("If-Modified-Since", new Date().toUTCString());
282     xhr_wr.onreadystatechange = function() { return; };
283     if (typeof(g_debug) == 'number' && g_debug > 0
284         && typeof(console) == 'object' && typeof(console.log) == 'function') { // OK
285             var ldate = new Date();
286             console.log(ldate.getTime()+':MESG:'+mesg); // OK
287     }
288     xhr_wr.send(null);
289
290     if (!is_conn) {
291         if (xhr_wr.responseText != null) {
292             eval(xhr_wr.responseText);
293         }
294     }
295 }
296
297 /*
298   sync request to server
299   server_request([arg0=arg1[, arg2=arg3[, ...]]])
300   if var name == '__POST__' than all other vars will be managed as POST content
301                                  and the call will be a POST
302  */
303 function server_request()
304 {
305     var xhr_wr = createXMLHttpRequest();
306     var i, collect = "", post_collect = null, is_post = false;
307
308     if (arguments.length > 0) {
309         for (i = 0 ; i < arguments.length ; i+= 2) {
310             if (arguments[i] == "__POST__") {
311                 is_post = true;
312                 post_collect = "";
313                 i -= 1;
314                 continue;
315             }
316             if (is_post)
317                 post_collect += (post_collect == "" ? "" : "&") + arguments[i] + "=" + encodeURIComponent(arguments[i+1]);
318             else
319                 collect += (i == 0 ? "" : "&") + arguments[i] + "=" + encodeURIComponent(arguments[i+1]);
320         }
321     }
322     // alert("Args: "+arguments.length);
323
324     var is_conn = (sess == "not_connected" ? false : true);
325     
326     // console.log("server_request:preresp: "+xhr_wr.responseText);
327
328     if (is_post) {
329         xhr_wr.open('POST', 'index_wr.php?'+(is_conn ? 'sess='+sess+'&' : '')+collect, false);
330         xhr_wr.setRequestHeader("Content-type","application/x-www-form-urlencoded");
331     }
332     else {
333         xhr_wr.open('GET', 'index_wr.php?'+(is_conn ? 'sess='+sess+'&' : '')+collect, false);
334     }
335     xhr_wr.onreadystatechange = function() { return; };
336     xhr_wr.send(post_collect);
337     
338     if (xhr_wr.responseText != null) {
339         // console.log("server_request:resp: "+xhr_wr.responseText);
340         return (xhr_wr.responseText);
341     } 
342     else
343         return (null);
344 }
345
346 /* Stat: CHAT and TABLE */
347
348 function chatt_checksend(obj,e)
349 {
350     var keynum;
351     var keychar;
352     var numcheck;
353
354     if(window.event) { // IE
355         keynum = e.keyCode;
356     }
357     else if(e.which) { // Netscape/Firefox/Opera
358         keynum = e.which;
359     }
360     // alert("OBJ: "+obj);
361     if (keynum == 13 && obj.value != "") { // Enter
362         act_chatt(obj.value);
363         obj.value = "";
364     }
365 }
366 function act_chatt(value)
367 {
368     send_mesg("chatt|"+encodeURIComponent(value));
369     /*
370     obj.disabled = true;
371     obj.value = "";
372     obj.disabled = false;
373     obj.focus();
374     */
375     return false;
376 }
377
378 /* Stat: ROOM */
379 function act_ping()
380 {
381     send_mesg("ping");
382 }
383
384 function act_sitdown(table)
385 {
386     send_mesg("sitdown|"+table);
387 }
388
389 function act_wakeup()
390 {
391     send_mesg("wakeup");
392 }
393
394 function act_splash()
395 {
396     send_mesg("splash");
397 }
398
399 function act_help()
400 {
401     send_mesg("help");
402 }
403
404 function act_passwdhowto()
405 {
406     send_mesg("passwdhowto");
407 }
408
409 function act_mesgtoadm()
410 {
411     send_mesg("mesgtoadm");
412 }
413
414 function act_tav()
415 {
416     act_chatt('/tav '+$('txt_in').value); 
417     $('txt_in').value = '';
418 }
419
420 function act_about()
421 {
422     send_mesg("about");
423 }
424
425 function act_placing()
426 {
427     send_mesg("placing");
428 }
429
430 function act_roadmap()
431 {
432     send_mesg("roadmap");
433 }
434
435 function act_whysupport()
436 {
437     send_mesg("whysupport");
438 }
439
440 function act_lascio()
441 {
442     send_mesg("lascio");
443 }
444
445 function safelascio()
446 {
447     var res;
448     // MLANG "Sei sicuro di volere lasciare questa mano?"
449     res = window.confirm(mlang_commons['gamleav'][g_lang]);
450     if (res)
451         act_lascio();
452 }
453
454 function act_logout(exitlock)
455 {
456     send_mesg("logout|"+exitlock);
457 }
458
459 function ModerateItem(item_ar)
460 {
461     var tr, td, date;
462
463     date = new Date();
464
465     this.time  = item_ar[0];
466     this.usrid = item_ar[1];
467     this.table = item_ar[2];
468     this.name  = item_ar[3];
469     this.cont  = item_ar[4];
470
471     this.loctm = date.getTime();
472
473     date.setTime(this.time * 1000);
474
475     tr = document.createElement("tr");
476     td = document.createElement("td");
477     // FIXME a more readable date here
478     // td.innerHTML = date.xxxx;
479     td.innerHTML = this.time % 100000;
480     tr.appendChild(td);
481
482     td = document.createElement("td");
483     td.innerHTML = this.table;
484     tr.appendChild(td);
485
486     td = document.createElement("td");
487     td.innerHTML = this.usrid;
488     tr.appendChild(td);
489
490     td = document.createElement("td");
491     td.innerHTML = this.name;
492     tr.appendChild(td);
493
494     td = document.createElement("td");
495     td.innerHTML = this.cont;
496     tr.appendChild(td);
497
498     this.tr = tr;
499 }
500
501 ModerateItem.prototype = {
502     time: 0,  // (sec)
503     loctm: 0, // (msec)
504     usrid: 0,
505     table: -1,
506     name: "",
507     cont: "",
508
509     tr: null,
510     hide: false,
511     sel: false,
512
513     tr_get: function () {
514         return this.tr;
515     },
516
517     sel_get: function () {
518         return this.sel;
519     },
520
521     sel_set: function (v) {
522         if (this.sel != v) {
523             this.sel = v;
524             this.tr.className = (v ? 'selected' : 'normal');
525         }
526     }
527 }
528
529 function Moderate()
530 {
531     this.item = new Array();
532 }
533
534 Moderate.prototype = {
535     win:  null,
536     table: null,
537     enabled: false,
538     item: null,
539
540     room_show: true,
541     table_show: -1,
542
543     // max_dt: 1800000, // (msec) maximum delta between current and line time
544     max_dt: 15000, // (msec) FIXME: DEV VERSION maximum delta between current and line time
545
546     cur: -1,
547
548     disable: function () {
549         if (this.tout) {
550             clearTimeout(this.tout);
551             this.tout = 0;
552         }
553         if (this.win) {
554             this.win.onbeforeunload = null;
555             this.win.close();
556             this.win = null;
557         }
558     },
559
560     activate: function (enable) {
561         if (this.enabled == enable) {
562             return true;
563         }
564         if (enable) {
565             this.disable();
566
567             // FIXME: remove scrollbars, only for devel reason
568             // this.win = window.open("moderation.php", "_blank", "width=800,height=600,toolbar=no,location=no,menubar=no,status=no");
569             this.win = window.open("moderation.php", "_blank", "width=800,height=600,toolbar=no,location=no,menubar=no,status=no,scrollbars=yes");
570             if (this.win == null) {
571                 this.disable();
572                 return false;
573             }
574             // to finish initialization we wait for popup page onload event ...
575             this.win_waitonload();
576         }
577         else {
578             this.disable();
579             this.enabled = false;
580         }
581
582     },
583
584     win_waitonload: function () {
585         if (typeof(this.win.is_loaded)  == 'undefined' || this.win.is_loaded != true) {
586             this.tout = setTimeout(function (obj) { obj.win_waitonload(); }, 250, this);
587         }
588         else {
589             this.post_onload();
590         }
591     },
592
593     post_onload: function() {
594         var tr, td, remtr;
595
596         this.win.anc = this;
597         this.table = $(this.win, 'moder_tab');
598
599         for (i = 0 ; i < this.item.length ; i++) {
600             this.table.appendChild(this.item[i].tr_get());
601             this.item[i].hide = false;
602         }
603
604         this.enabled = true;
605     },
606
607     onunload: function() {
608         act_moderate();
609     },
610
611     is_enabled: function() {
612         return (this.enabled);
613     },
614
615     add: function(item) {
616         var mi;
617
618         this.item_gc();
619
620         mi = new ModerateItem(item);
621         mi.tr.className = 'normal';
622
623         var self;
624         self = this;
625         mi.tr.onclick = function () { self.row_select(mi); };
626
627         this.item.push(mi);
628         this.table.appendChild(mi.tr_get());
629     },
630
631     item_remove: function(idx) {
632         var old;
633
634         old = this.item.splice(idx,1);
635
636         if (!old[0].hide)
637             this.table.removeChild(old[0].tr_get());
638
639         delete old;
640     }
641
642     // moderation items garbage collector: after this.max_dt a line is removed
643     item_gc: function() {
644         var date, time;
645
646         date = new Date();
647         time = date.getTime();
648
649         for (i = 0 ; i < this.item.length ; i++) {
650             if (time - this.item[i].loctm > this.max_dt) {
651                 this.item_remove(i);
652                 i--;
653             }
654         }
655     },
656
657     row_select: function(mi) {
658         for (i = 0 ; i < this.item.length ; i++) {
659             if (this.item[i] == mi) {
660                 this.item[i].sel_set(!this.item[i].sel_get());
661             }
662             else {
663                 this.item[i].sel_set(false);
664             }
665         }
666         // mi.tr.className = "selected";
667     },
668
669     room_show_update: function(obj) {
670         this.tab_update(obj.checked, this.table_show);
671     },
672
673     //
674     table_show_update: function(obj) {
675         this.tab_update(this.room_show, obj.options[obj.selectedIndex].value );
676     },
677
678     tab_update: function(room_new, table_new)
679     {
680         // remove all and add all valid
681         for (i = 0 ; i < this.item.length ; i++) {
682             if (this.item[i].hide)
683                 continue;
684             this.table.removeChild(this.item[i].tr_get());
685             this.item[i].hide = true;
686         }
687
688         for (i = 0 ; i < this.item.length ; i++) {
689             var app = false;
690
691             if (room_new && table_new == -1) {
692                 app = true;
693             }
694             else if (room_new && table_new != -1) {
695                 if (this.item[i].table == table_new || this.item[i].table == -1) {
696                     app = true;
697                 }
698             }
699             else if (!room_new && table_new == -1) {
700                 if (this.item[i].table != -1) {
701                     app = true;
702                 }
703             }
704             else if (!room_new && table_new != -1) {
705                 if (this.item[i].table == table_new) {
706                     app = true;
707                 }
708             }
709             if (app) {
710                 this.table.appendChild(this.item[i].tr_get());
711                 this.item[i].hide = false;
712             }
713         }
714         this.room_show  = room_new;
715         this.table_show = table_new;
716     }
717 }
718
719 // function AddBefore(rowId){
720 //     var target = document.getElementById(rowId);
721 //     var newElement = document.createElement('tr');
722 //     target.parentNode.insertBefore(newElement, target);
723 //     return newElement;
724 // }
725
726 // function AddAfter(rowId){
727 //     var target = document.getElementById(rowId);
728 //     var newElement = document.createElement('tr');
729
730 //     target.parentNode.insertBefore(newElement, target.nextSibling );
731 //     return newElement;
732 // }
733
734 function moderate(enable)
735 {
736     return (g_moder.activate(enable));
737 }
738
739 var g_moder = new Moderate();
740
741 function act_moderate()
742 {
743     send_mesg("moderate|"+(g_moder.is_enabled() ? "false" : "true"));
744 }
745
746
747
748 //         // build table with js
749         
750 //         g_moder.item = new Array;
751 //         g_moder.table = xxx;
752 //     }
753 //     else {
754 //         if (g_moder == null)
755 //             return true;
756
757 //         if (g_moder.win != null) {
758 //             g_moder.win.close();
759 //             g_moder.win = null;
760 //         }
761
762 //         if (g_moder.item != null) {
763 //             // TODO CLEANUP
764 //             ;
765 //         }
766 //         g_moder.cur = -1;
767 //     }
768 // }
769
770
771 function act_reloadroom()
772 {
773     if (g_moder.is_enabled()) {
774         g_moder.disable();
775     }
776     window.onunload = null;
777     window.onbeforeunload = null;
778     document.location.assign("index.php");
779 }
780
781
782 function act_shutdown()
783 {
784     var c = 0;
785
786     send_mesg("shutdown");
787     // while (xhr_wr.readyState != 4)
788     //  c++;
789 }
790
791 function postact_logout()
792 {
793     // alert("postact_logout");
794
795     if (g_moder.is_enabled()) {
796         g_moder.disable();
797     }
798
799     try { 
800         hstm.abort();
801     } catch (e) {}
802
803     // eraseCookie("sess");
804     document.location.assign("index.php");
805 }
806
807 /*
808   function slowimg(img,x1,y1,deltat,free,action,srcend)
809   img    - image to move
810   x1,y1  - destination coords
811   deltat - time for each frame (in msec)
812   free   - when the release the local block for other operations (range: 0 - 1)
813   action - function to run when the image is moved
814   srcend - image to switch when the image is moved
815 */
816
817 function sleep(st, delay)
818 {
819     // alert("LOC_NEW PRE: "+st.st_loc_new);
820
821     st.st_loc_new++;
822
823     setTimeout(function(obj){ if (obj.st_loc_new > obj.st_loc) { obj.st_loc++; }},
824                delay, st);
825 }
826
827 function slowimg(img,x1,y1,deltat,free,action,srcend) {
828     this.img = img;
829
830     // this.x0  = parseInt(document.defaultView.getComputedStyle(this.img, "").getPropertyValue("left"));
831     this.x0 = parseInt(getStyle(this.img,"left", "left"));
832 // alert("img.x0 = "+this.x0);
833     // this.y0  = parseInt(document.defaultView.getComputedStyle(this.img, "").getPropertyValue("top"));
834     this.y0  = parseInt(getStyle(this.img,"top", "top"));
835     this.x1  = x1;
836     this.y1  = y1;
837     this.deltat = deltat;
838     this.free = free;
839     this.action = action;
840     this.srcend = srcend;
841 }
842
843 slowimg.prototype = {
844     img: null, 
845     st: null,
846     x0: 0,
847     y0: 0,
848     x1: 0,
849     y1: 0,
850     dx: 0,
851     dy: 0,
852     free: 0,
853     step_n:    0,
854     step_cur:  0,
855     step_free: 0,
856     time:      0,
857     deltat:   40,
858     tout: 0,
859     action: null,
860     srcend: null,
861     
862     setstart: function(x0,y0)
863     {
864         this.x0 = x0;
865         this.y0 = y0;
866     },
867     
868     setaction: function(act)
869     {
870         this.action = act;
871     },
872     
873
874     settime: function(time) 
875     {
876         this.time = (time < this.deltat ? this.deltat : time);
877         this.step_n = parseInt(this.time / this.deltat);
878         this.dx = (this.x1 - this.x0) / this.step_n;
879         this.dy = (this.y1 - this.y0) / this.step_n;
880         if (this.step_n * this.deltat == this.time) {
881             this.step_n--;
882         }
883         if (this.free < 1) {
884             this.step_free = parseInt(this.step_n * this.free);
885         }
886     },
887     
888     start: function(st)
889     {
890         // $("logz").innerHTML += "               xxxxxxxxxxxxxxxxxxxxxSTART<br>";
891         this.st = st;
892         this.st.st_loc_new++;
893         
894         this.img.style.visibility = "visible";
895         setTimeout(function(obj){ obj.animate(); }, this.deltat, this);
896     },
897     
898     animate: function()
899     {
900         // $("log").innerHTML = "Val " + this.step_cur + " N: " + this.step_n + "<br>";
901         if (this.step_cur == 0) {
902             var date = new Date();
903             // $("logz").innerHTML = "Timestart: " + date + "<br>";
904         }
905         if (this.step_cur <= this.step_n) {
906             this.img.style.left = this.x0 + this.dx * this.step_cur;
907             this.img.style.top  = this.y0 + this.dy * this.step_cur;
908             this.step_cur++;
909             setTimeout(function(obj){ obj.animate(); }, this.deltat, this);
910             if (this.step_cur == this.step_free && this.st != null) {
911                 if (this.st.st_loc < this.st.st_loc_new) {
912                     // alert("QUI1  " + this.step_cur + "  ZZ  "+  this.step_free);
913                     this.st.st_loc++;
914                     this.st = null;
915                 }
916             }
917         }
918         else {
919             this.img.style.left = this.x1;
920             this.img.style.top  = this.y1;
921             // $("logz").innerHTML += "xxxxxxxxxxxxxxxCLEAR<br>";
922             var date = new Date();
923             // $("logz").innerHTML += "Timestop: " + date + "<br>";
924
925             if (this.action != null) {
926                 eval(this.action);
927             }
928
929             if (this.st != null && this.st.st_loc < this.st.st_loc_new) {
930                 // alert("QUI2");
931                 this.st.st_loc++;
932                 this.st = null;
933             }
934             if (this.srcend != null) {
935                 this.img.src = this.srcend;
936             }
937         }
938     }
939 }
940
941 function div_show(div)
942 {
943     div.style.top = parseInt((document.body.clientHeight - parseInt(getStyle(div,"height", "height"))) / 2) + document.body.scrollTop;
944     div.style.visibility = "visible";
945 }
946
947 function notify_ex(st, text, tout, butt, w, h, is_opa, block_time)
948 {
949     var clo, box;
950     var t = this;
951     
952     this.st = st;
953
954     this.ancestor = document.body;
955     
956     this.st.st_loc_new++;
957
958     clo = document.createElement("input");
959     clo.type = "submit";
960     clo.className = "button";
961     clo.style.bottom = "4px";
962     clo.obj = this;
963     if (block_time > 0) {
964         clo.value = "leggere, prego.";
965         this.butt = butt;
966     }
967     else {
968         clo.value = butt;
969         clo.onclick = this.input_hide;
970     }
971
972     clodiv = document.createElement("div");
973     clodiv.className = "notify_clo";
974     this.clo = clo;
975     this.clodiv = clodiv;
976
977     clodiv.appendChild(clo);
978
979     cont = document.createElement("div");
980
981     cont.style.borderBottomStyle = "solid";
982     cont.style.borderBottomWidth = "1px";
983     cont.style.borderBottomColor = "gray";
984     cont.style.height = (h - 30)+"px";
985     cont.style.overflow = "auto";
986     cont.innerHTML = text;
987
988     box =  document.createElement("div");
989     if (is_opa)
990         box.className = "notify_opaque";
991     else
992         box.className = "notify";
993
994     box.style.zIndex = 200;
995     box.style.width  = w+"px";
996     box.style.marginLeft  = -parseInt(w/2)+"px";
997     box.style.height = h+"px";
998     box.style.top = parseInt((document.body.clientHeight - h) / 2) + document.body.scrollTop;
999     box.appendChild(cont);
1000     box.appendChild(clodiv);
1001     box.style.visibility = "visible";
1002
1003     this.notitag = box;
1004     
1005     this.ancestor.appendChild(box);
1006     
1007     this.toutid = setTimeout(function(obj){ obj.unblock(); }, tout, this);
1008
1009     if (block_time != 0) {
1010         this.tblkid = setTimeout(function(obj){ obj.clo.value = obj.butt; obj.clo.onclick = obj.input_hide; formsub_hilite(obj.clo); obj.clo.focus(); }, block_time, this);
1011     }
1012     else {
1013         formsub_hilite(clo);
1014         clo.focus();
1015     }
1016
1017 }
1018
1019
1020 notify_ex.prototype = {
1021     ancestor: null,
1022     st: null,
1023     notitag: null,
1024     toutid: null,
1025     clo: null,
1026     clodiv: null, 
1027     butt: null,
1028     tblkid: null,
1029
1030     unblock: function()
1031     {
1032         if (this.st.st_loc < this.st.st_loc_new) {
1033             this.st.st_loc++;
1034         }
1035     },
1036     
1037     hide: function()
1038     {
1039         clearTimeout(this.toutid);
1040         this.ancestor.removeChild(this.notitag);
1041         this.unblock();
1042     },
1043
1044     input_hide: function()
1045     {
1046         clearTimeout(this.obj.toutid);
1047         this.obj.ancestor.removeChild(this.obj.notitag);
1048         this.obj.unblock();
1049     }
1050 }
1051
1052
1053 notify.prototype = notify_ex.prototype;                // Define sub-class
1054 notify.prototype.constructor = notify;
1055 notify.baseConstructor = notify_ex;
1056 notify.superClass = notify_ex.prototype;
1057
1058 function notify(st, text, tout, butt, w, h)
1059 {
1060     notify_ex.call(this, st, text, tout, butt, w, h, false, 0);
1061 }
1062
1063 function globst() {
1064     this.st = -1;
1065     this.st_loc = -1;
1066     this.st_loc_new = -1;
1067     this.comms  = new Array;
1068 }
1069
1070 globst.prototype = {
1071     st: -1,
1072     st_loc: -1,
1073     st_loc_new: -1,
1074     comms: null,
1075     sleep_hdl: null,
1076
1077     sleep: function(delay) {
1078         st.st_loc_new++;
1079
1080         if (!this.the_end) {
1081             this.sleep_hdl = setTimeout(function(obj){ if (obj.st_loc_new > obj.st_loc) { obj.st_loc++; obj.sleep_hdl = null; }},
1082                                         delay, this);
1083         }
1084     },
1085
1086     abort: function() {
1087         if (this.sleep_hdl != null) {
1088             clearTimeout(this.sleep_hdl);
1089             this.sleep_hdl = null;
1090         }
1091     }
1092 }
1093
1094 function remark_step()
1095 {
1096     var ct = $("remark").l_remct;
1097     
1098     if (ct != 0) {
1099         ct++;
1100         if (ct > 2)
1101             ct = 1;
1102         $("remark").className = "remark"+ct;
1103         $("remark").l_remct = ct;
1104         setTimeout(remark_step,500);
1105     }
1106     else
1107         $("remark").className = "remark0";
1108     
1109     return;
1110 }
1111
1112 function remark_on()
1113 {
1114     if ($("remark").l_remct == 0) {
1115         $("remark").l_remct = 1;
1116         setTimeout(remark_step,500);
1117     }
1118 }
1119
1120 function remark_off()
1121 {
1122     $("remark").l_remct = 0;
1123     $("remark").className = "remark0";
1124 }
1125
1126
1127 function italizer(ga)
1128 {
1129     var pre, pos;
1130     if (ga[0] & 2) 
1131         return "<i>"+ga[1]+"</i>";
1132     else
1133         return ga[1];
1134 }
1135
1136
1137 function exitlock_show(num, islock)
1138 {
1139     g_exitlock = num;
1140
1141     num = (num < 3 ? num : 3);
1142     $("exitlock").src = "img/exitlock"+num+(islock ? "n" : "y")+".png";
1143     // alert("EXITLOCK: "+$("exitlock").src);
1144     $("exitlock").style.visibility = "visible";
1145 }
1146
1147 var fin = 0;
1148
1149 //    exitlock_show(0, true);
1150
1151
1152 var chatt_lines = new Array();
1153 var chatt_lines_n = 0;
1154
1155 var CHATT_MAXLINES = 40;
1156
1157 function user_decorator(user)
1158 {
1159     var name;
1160     var flags = user[BSK_USER_FLAGS];
1161     var flags_vlt = user[BSK_USER_FLGVL];
1162     if ((flags & 0x03) != 0)
1163         name = "<span class='au" + (flags & 0x03) + "'>"+user[BSK_USER_NICK]+"</span>";
1164     else
1165         name = user[BSK_USER_NICK];
1166
1167     return (name);
1168 }
1169
1170 function user_dec_and_state(el)
1171 {
1172     var content = "";
1173     var val_el;
1174
1175     content = user_decorator(el);
1176     content += state_add(el[BSK_USER_FLAGS], el[BSK_USER_FLGVL],
1177                          (typeof(el[BSK_USER_SCOL]) != 'undefined' ? el[BSK_USER_SCOL] : null));
1178     
1179     return (content);
1180 }
1181
1182
1183 /* PRO CHATT */
1184 function chatt_sub(dt,data,str)
1185 {
1186     var must_scroll = false;
1187     var name;
1188     var flags;
1189     var isauth;
1190     var bolder = [ (data[BSK_USER_FLAGS] | 1), data[BSK_USER_FLGVL], data[BSK_USER_NICK] ];
1191     name = user_decorator(bolder);
1192
1193     if ($("txt").scrollTop + parseInt(getStyle($("txt"),"height", "height")) -  $("txt").scrollHeight >= 0)
1194         must_scroll = true;
1195
1196     // alert("ARRIVA NAME: "+ name + "  STR:"+str);
1197     if (chatt_lines_n == CHATT_MAXLINES) {
1198         $("txt").innerHTML = "";
1199         for (i = 0 ; i < (CHATT_MAXLINES - 1) ; i++) {
1200             chatt_lines[i] = chatt_lines[i+1];
1201             $("txt").innerHTML += chatt_lines[i];
1202         }
1203         chatt_lines[i] = dt+name+": "+str+ "<br>";
1204         $("txt").innerHTML += chatt_lines[i];
1205     }
1206     else {
1207         chatt_lines[chatt_lines_n] = dt+name+": "+str+ "<br>";
1208         $("txt").innerHTML += chatt_lines[chatt_lines_n];
1209         chatt_lines_n++;
1210     }
1211     // $("txt").innerHTML;
1212
1213     
1214     if (must_scroll) {
1215         $("txt").scrollTop = 10000000;
1216     }
1217     // alert("scTOP "+$("txt").scrollTop+"  scHEIGHT: "+$("txt").scrollHeight+" HEIGHT: "+getStyle($("txt"),"height", "height") );
1218 }
1219
1220 /*
1221  *  GESTIONE DEI COOKIES
1222  */
1223 function createCookie(name,value,hours,path) {
1224         if (hours) {
1225                 var date = new Date();
1226                 date.setTime(date.getTime()+(hours*60*60*1000));
1227                 var expires = "; expires="+date.toGMTString();
1228         }
1229         else var expires = "";
1230         document.cookie = name+"="+value+expires+"; path="+path;
1231 }
1232
1233 function readCookie(name) {
1234         var nameEQ = name + "=";
1235         var ca = document.cookie.split(';');
1236         for(var i=0;i < ca.length;i++) {
1237                 var c = ca[i];
1238                 while (c.charAt(0)==' ') c = c.substring(1,c.length);
1239                 if (c.indexOf(nameEQ) == 0) return c.substring(nameEQ.length,c.length);
1240         }
1241         return null;
1242 }
1243
1244 function eraseCookie(name) {
1245         createCookie(name,"",-1);
1246 }
1247
1248 function onbeforeunload_cb () {
1249     return("");
1250 }
1251
1252 function onunload_cb () {
1253     
1254     if (typeof(hstm) != "undefined")
1255         hstm.the_end = true; 
1256
1257     act_shutdown();
1258     
1259     return(false);
1260 }
1261
1262 function room_checkspace(emme,tables,inpe)
1263 {
1264     nome = "<b>";
1265     for (i = 0 ; i < emme ; i++) 
1266         nome += "m";
1267     nome += "</b>";
1268
1269     alta = "";
1270     for (i = 0 ; i < 5 ; i++) 
1271         alta += nome+"<br>";
1272
1273     for (i = 0 ; i < tables ; i++) {
1274         $("table"+i).innerHTML = alta;
1275         // MLANG Mi siedo.
1276         $("table_act"+i).innerHTML = "<input type=\"button\" class=\"button\" name=\"xhenter"+i+"\"  value=\""+mlang_commons['btn_sit'][g_lang]+"\" onclick=\"act_sitdown(1);\">";
1277         }
1278
1279     stand = "<table class=\"table_standup\"><tbody><tr>";
1280     for (i = 0 ; i < inpe ; i++) {
1281         stand += "<td>"+nome+"</td>";
1282         if ((i+1) % 4 == 0) {
1283             stand += "</tr><tr>";
1284         }
1285     }
1286     stand += "</tr>";
1287     $("standup").innerHTML = stand;
1288
1289     // VERIFY: what is this button ?
1290     // MLANG Esco.
1291     $("esco").innerHTML = "<input class=\"button\" name=\"logout\" type=\"button\" value=\""+mlang_commons['btn_exit'][g_lang]+"\" onclick=\"act_logout();\" type=\"button\">";
1292 }
1293
1294 function  unescapeHTML(cont) {
1295     var div = document.createElement('div');
1296     var memo = "";
1297     var i;
1298
1299     div.innerHTML = cont;
1300     if (div.childNodes[0]) {
1301         if (div.childNodes.length > 1) {
1302             if (div.childNodes.toArray)
1303                 alert("si puo");
1304             else {
1305                 var length = div.childNodes.length, results = new Array(length);
1306             while (length--)
1307                 results[length] = div.childNodes[length];
1308                 
1309             for (i=0 ; i<results.length ; i++)
1310                 memo = memo + results[i].nodeValue;
1311             }
1312
1313             return (memo);
1314         }
1315         else {
1316             return (div.childNodes[0].nodeValue);
1317         }
1318     }
1319     else {
1320         return ('');
1321     }
1322 }
1323
1324 function playsound(tag, sound) {
1325    // g_withflash is a global var
1326    if (g_withflash) {
1327       $(tag).innerHTML = '<OBJECT classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000" '+
1328 'codebase="http://active.macromedia.com/flash2/cabs/swflash.cab#version=4,0,0,0" id="mysound" WIDTH=1 HEIGHT=1>' +
1329 '<PARAM NAME="movie" VALUE="../playsound.swf"><PARAM NAME="PLAY" VALUE="true"><PARAM NAME="LOOP" VALUE="false">' +
1330 '<PARAM NAME=FlashVars VALUE="streamUrl='+sound+'">' +
1331 '<EMBED swliveconnect="true" name="mysound" src="../playsound.swf" FlashVars="streamUrl='+sound+'" PLAY="true" LOOP="false" '+
1332 ' WIDTH=1 HEIGHT=1 TYPE="application/x-shockwave-flash" PLUGINSPAGE="http://www.macromedia.com/shockwave/download/index.cgi?P1_Prod_Version=ShockwaveFlash"></OBJECT>';
1333    }
1334 }
1335
1336 function topbanner_init()
1337 {
1338     setInterval(topbanner_cb, 666);
1339 ;
1340 }
1341
1342 function topbanner_cb()
1343 {
1344     var a, b;
1345
1346     a = $('topbanner').style.backgroundColor;
1347     b = $('topbanner').style.borderLeftColor;
1348
1349     $('topbanner').style.backgroundColor = b;
1350     $('topbanner').style.borderColor = a+" "+a+" "+a+" "+a;
1351
1352     // console.log("A: "+a+"  B: "+b);
1353 }
1354
1355 function sidebanner_init()
1356 {
1357     setInterval(sidebanner_cb, 666);
1358 }
1359
1360 function sidebanner2_init()
1361 {
1362     setInterval(sidebanner2_cb, 666);
1363 }
1364
1365 function sidebanner_cb()
1366 {
1367     var a, b;
1368
1369     a = $('sidebanner').style.backgroundColor;
1370     b = $('sidebanner').style.borderLeftColor;
1371
1372     $('sidebanner').style.backgroundColor = b;
1373     $('sidebanner').style.borderColor = a+" "+a+" "+a+" "+a;
1374
1375     // console.log("A: "+a+"  B: "+b);
1376 }
1377
1378 function sidebanner2_cb()
1379 {
1380     var a, b;
1381
1382     a = $('sidebanner2').style.backgroundColor;
1383     b = $('sidebanner2').style.borderLeftColor;
1384
1385     $('sidebanner2').style.backgroundColor = b;
1386     $('sidebanner2').style.borderColor = a+" "+a+" "+a+" "+a;
1387
1388     // console.log("A: "+a+"  B: "+b);
1389 }
1390
1391
1392 function langtolng(lang)
1393 {
1394     if (lang == "en")
1395         return ("-en");
1396     else
1397         return ("");
1398 }
1399
1400 function formtext_hilite(obj)
1401 {
1402     obj.className = 'input_text';
1403     addEvent(obj, "focus", function () { this.className = 'input_text_hi'; });
1404     addEvent(obj, "blur",  function () { this.className = 'input_text'; });
1405 }
1406
1407 function formsub_hilite(obj)
1408 {
1409     obj.className = 'input_sub';
1410     addEvent(obj, "focus", function () { this.className = 'input_sub_hi'; });
1411     addEvent(obj, "blur",  function () { this.className = 'input_sub'; });
1412 }
1413
1414 // return the value of the radio button that is checked
1415 // return an empty string if none are checked, or
1416 // there are no radio buttons
1417 function get_checked_value(radioObj) {
1418         if(!radioObj)
1419                 return "";
1420         var radioLength = radioObj.length;
1421         if(radioLength == undefined)
1422                 if(radioObj.checked)
1423                         return radioObj.value;
1424                 else
1425                         return "";
1426         for(var i = 0; i < radioLength; i++) {
1427                 if(radioObj[i].checked) {
1428                         return radioObj[i].value;
1429                 }
1430         }
1431         return "";
1432 }
1433
1434 // set the radio button with the given value as being checked
1435 // do nothing if there are no radio buttons
1436 // if the given value does not exist, all the radio buttons
1437 // are reset to unchecked
1438 function set_checked_value(radioObj, newValue) {
1439         if(!radioObj)
1440                 return;
1441         var radioLength = radioObj.length;
1442         if(radioLength == undefined) {
1443                 radioObj.checked = (radioObj.value == newValue.toString());
1444                 return;
1445         }
1446         for(var i = 0; i < radioLength; i++) {
1447                 radioObj[i].checked = false;
1448                 if(radioObj[i].value == newValue.toString()) {
1449                         radioObj[i].checked = true;
1450                 }
1451         }
1452 }
1453
1454 function url_append_arg(url, name, value)
1455 {
1456     var pos, sep, pref, rest;
1457
1458     if ((pos = url.indexOf('?'+name+'=')) == -1) {
1459         pos = url.indexOf('&'+name+'=');
1460     }
1461     if (pos == -1) {
1462         if ((pos = url.indexOf('?')) != -1)
1463             sep = '&';
1464         else
1465             sep = '?';
1466
1467         return (url+sep+name+"="+encodeURIComponent(value));
1468     }
1469     else {
1470         pref = url.substring(0, pos+1);
1471         rest = url.substring(pos+1);
1472         // alert("rest: "+rest+"  pos: "+pos);
1473         if ((pos = rest.indexOf('&')) != -1) {
1474             rest = rest.substring(pos);
1475         }
1476         else {
1477             rest = "";
1478         }
1479         return (pref+name+"="+encodeURIComponent(value)+rest);
1480     }
1481 }
1482
1483 function url_append_args(url)
1484 {
1485     var i, ret;
1486
1487     ret = url;
1488     for (i = 1 ; i < arguments.length-1 ; i+= 2) {
1489         ret = url_append_arg(ret, arguments[i], arguments[i+1]);
1490     }
1491
1492     return (ret);
1493 }
1494
1495 function url_complete(parent, url)
1496 {
1497     var p, p2, rest;
1498     var host = "", path = "";
1499
1500     // host extraction
1501     p = parent.indexOf("://");
1502     if (p > -1) {
1503         rest = parent.substring(p+3);
1504         p2 = rest.indexOf("/");
1505         if (p2 > -1) {
1506             host = parent.substring(0, p+3+p2);
1507             rest = parent.substring(p+3+p2);
1508         }
1509         else {
1510             host = rest;
1511             rest = "";
1512         }
1513     }
1514     else {
1515         rest = parent;
1516     }
1517
1518     // path extraction
1519     p = rest.lastIndexOf("/");
1520     if (p > -1) {
1521         path = rest.substring(0, p+1);
1522     }
1523
1524     // alert("host: ["+host+"]  path: ["+path+"]");
1525     if (url.substring(0,6) == 'http:/' || url.substring(0,7) == 'https:/') {
1526         return (url);
1527     }
1528     else if (url.substring(0,1) == '/') {
1529         return (host+url);
1530     }
1531     else {
1532         return (host+path+url);
1533     }
1534 }