2 // $Id: wclib.js,v 625.2 2005/09/18 04:04:32 ajc Exp $
4 // JavaScript function library for WebCit.
10 var room_is_trash = 0;
12 if (document.layers) {browserType = "nn4"}
13 if (document.all) {browserType = "ie"}
14 if (window.navigator.userAgent.toLowerCase().match("gecko")) {
18 var ns6=document.getElementById&&!document.all;
22 // This code handles the popups for instant messages.
26 function hide_page_popup() {
27 if (browserType == "gecko" )
28 document.poppedLayer = eval('document.getElementById(\'page_popup\')');
29 else if (browserType == "ie")
30 document.poppedLayer = eval('document.all[\'page_popup\']');
32 document.poppedLayer = eval('document.layers[\'`page_popup\']');
34 document.poppedLayer.style.visibility = "hidden";
37 function hide_imsg_popup() {
38 if (browserType == "gecko" )
39 document.poppedLayer = eval('document.getElementById(\'important_message\')');
40 else if (browserType == "ie")
41 document.poppedLayer = eval('document.all[\'important_message\']');
43 document.poppedLayer = eval('document.layers[\'`important_message\']');
45 document.poppedLayer.style.visibility = "hidden";
48 // This function activates the ajax-powered recipient autocompleters on the message entry screen.
49 function activate_entmsg_autocompleters() {
50 new Ajax.Autocompleter('cc_id', 'cc_name_choices', 'cc_autocomplete', {} );
51 new Ajax.Autocompleter('bcc_id', 'bcc_name_choices', 'bcc_autocomplete', {} );
52 new Ajax.Autocompleter('recp_id', 'recp_name_choices', 'recp_autocomplete', {} );
57 // Toggle the icon bar between menu/roomlist...
58 var which_div_expanded = null;
59 var num_drop_targets = 0;
60 var drop_targets_elements = new Array();
61 var drop_targets_roomnames = new Array();
63 function switch_to_room_list() {
64 $('iconbar').innerHTML = $('iconbar').innerHTML.substr(0, $('iconbar').innerHTML.indexOf('switch'));
65 new Ajax.Updater('iconbar', 'iconbar_ajax_rooms', { method: 'get' } );
68 function expand_floor(floor_div) {
69 if (which_div_expanded != null) {
70 if ($(which_div_expanded) != null) {
71 $(which_div_expanded).style.display = 'none' ;
75 // clicking on the already-expanded floor causes the whole list to collapse
76 if (which_div_expanded == floor_div) {
77 which_div_expanded = null;
79 // notify the server that no floors are expanded
81 'set_floordiv_expanded/-1', {
88 // expand the requested floor
89 $(floor_div).style.display = 'block';
90 which_div_expanded = floor_div;
92 // notify the server of which floor is expanded
94 'set_floordiv_expanded/'+floor_div, {
100 function switch_to_menu_buttons() {
101 which_div_expanded = null;
102 num_drop_targets = 0;
103 new Ajax.Updater('iconbar', 'iconbar_ajax_menu', { method: 'get' } );
107 // Static variables for mailbox view...
109 var CtdlNumMsgsSelected = 0;
110 var CtdlMsgsSelected = new Array();
112 // This gets called when you single click on a message in the mailbox view.
113 // We know that the element id of the table row will be the letter 'm' plus the message number.
115 function CtdlSingleClickMsg(evt, msgnum) {
117 // Clear the preview pane until we load the new message
118 $('preview_pane').innerHTML = '';
120 // De-select any messages that were already selected, *unless* the Ctrl key
121 // is being pressed, in which case the user wants multi select.
123 if (CtdlNumMsgsSelected > 0) {
124 for (i=0; i<CtdlNumMsgsSelected; ++i) {
125 $('m'+CtdlMsgsSelected[i]).style.backgroundColor = '#fff';
126 $('m'+CtdlMsgsSelected[i]).style.color = '#000';
128 CtdlNumMsgsSelected = 0;
132 // For multi select ... is the message being clicked already selected?
133 already_selected = 0;
134 if ( (evt.ctrlKey) && (CtdlNumMsgsSelected > 0) ) {
135 for (i=0; i<CtdlNumMsgsSelected; ++i) {
136 if (CtdlMsgsSelected[i] == msgnum) {
137 already_selected = 1;
142 // Now select (or de-select) the message
143 if ( (evt.ctrlKey) && (already_selected == 1) ) {
144 $('m'+msgnum).style.backgroundColor = '#fff';
145 $('m'+msgnum).style.color = '#000';
148 $('m'+msgnum).style.backgroundColor='#69aaff';
149 $('m'+msgnum).style.color='#fff';
150 CtdlNumMsgsSelected = CtdlNumMsgsSelected + 1;
151 CtdlMsgsSelected[CtdlNumMsgsSelected-1] = msgnum;
154 // Update the preview pane
155 new Ajax.Updater('preview_pane', 'msg/'+msgnum, { method: 'get' } );
157 // Mark the message as read
161 parameters: 'g_cmd=SEEN '+msgnum+'|1',
162 onComplete: CtdlRemoveTheUnseenBold(msgnum)
166 return false; // try to defeat the default click behavior
169 // Delete selected messages.
170 function CtdlDeleteSelectedMessages(evt) {
172 if (CtdlNumMsgsSelected < 1) {
173 // Nothing to delete, so exit silently.
176 for (i=0; i<CtdlNumMsgsSelected; ++i) {
177 if (parseInt(room_is_trash) > 0) {
181 parameters: 'g_cmd=DELE ' + CtdlMsgsSelected[i],
182 onComplete: CtdlClearDeletedMsg(CtdlMsgsSelected[i])
190 parameters: 'g_cmd=MOVE ' + CtdlMsgsSelected[i] + '|_TRASH_|0',
191 onComplete: CtdlClearDeletedMsg(CtdlMsgsSelected[i])
196 CtdlNumMsgsSelected = 0;
198 // Clear the preview pane too.
199 $('preview_pane').innerHTML = '';
203 // Move selected messages.
204 function CtdlMoveSelectedMessages(evt, target_roomname) {
206 if (CtdlNumMsgsSelected < 1) {
207 // Nothing to delete, so exit silently.
210 for (i=0; i<CtdlNumMsgsSelected; ++i) {
214 parameters:'g_cmd=MOVE ' + CtdlMsgsSelected[i] + '|' + target_roomname + '|0',
215 onComplete:CtdlClearDeletedMsg(CtdlMsgsSelected[i])
219 CtdlNumMsgsSelected = 0;
221 // Clear the preview pane too.
222 $('preview_pane').innerHTML = '';
227 // This gets called when the user touches the keyboard after selecting messages...
228 function CtdlMsgListKeyPress(evt) {
229 if(document.all) { // aIEeee
230 var whichKey = window.event.keyCode;
232 else { // non-sux0r browsers
233 var whichKey = evt.which;
235 if (whichKey == 46) { // DELETE key
236 CtdlDeleteSelectedMessages(evt);
241 // Take the boldface away from a message to indicate that it has been seen.
242 function CtdlRemoveTheUnseenBold(msgnum) {
243 $('m'+msgnum).style.fontWeight='normal';
246 // A message has been deleted, so yank it from the list.
247 // (IE barfs on m9999.innerHTML='' so we use a script.aculo.us effect instead.)
248 function CtdlClearDeletedMsg(msgnum) {
249 new Effect.Squish('m'+msgnum);
253 // These functions called when the user down-clicks on the message list resizer bar
258 function CtdlResizeMsgListMouseUp(evt) {
259 document.onmouseup = null;
260 document.onmousemove = null;
261 if (document.layers) {
262 document.releaseEvents(Event.MOUSEUP | Event.MOUSEMOVE);
267 function CtdlResizeMsgListMouseMove(evt) {
268 y = (ns6 ? evt.clientY : event.clientY);
269 increment = y - saved_y;
271 // First move the bottom of the message list...
272 d = $('message_list');
274 divHeight = d.offsetHeight;
276 else if (d.style.pixelHeight) {
277 divHeight = d.style.pixelHeight;
279 d.style.height = (divHeight + increment) + 'px';
281 // Then move the top of the preview pane...
282 d = $('preview_pane');
284 divTop = d.offsetTop;
286 else if (d.style.pixelTop) {
287 divTop = d.style.pixelTop;
289 d.style.top = (divTop + increment) + 'px';
291 // Resize the bottom of the preview pane...
292 d = $('preview_pane');
294 divHeight = d.offsetHeight;
296 else if (d.style.pixelHeight) {
297 divHeight = d.style.pixelHeight;
299 d.style.height = (divHeight - increment) + 'px';
301 // Then move the top of the slider bar.
302 d = $('resize_msglist');
304 divTop = d.offsetTop;
306 else if (d.style.pixelTop) {
307 divTop = d.style.pixelTop;
309 d.style.top = (divTop + increment) + 'px';
315 function CtdlResizeMsgListMouseDown(evt) {
316 saved_y = (ns6 ? evt.clientY : event.clientY);
317 document.onmouseup = CtdlResizeMsgListMouseUp;
318 document.onmousemove = CtdlResizeMsgListMouseMove;
319 if (document.layers) {
320 document.captureEvents(Event.MOUSEUP | Event.MOUSEMOVE);
322 return false; // disable the default action
327 // These functions handle drag and drop message moving
331 function CtdlMoveMsgMouseDown(evt, msgnum) {
333 // do the highlight first
334 CtdlSingleClickMsg(evt, msgnum);
336 // Now handle the possibility of dragging
337 saved_x = (ns6 ? evt.clientX : event.clientX);
338 saved_y = (ns6 ? evt.clientY : event.clientY);
339 document.onmouseup = CtdlMoveMsgMouseUp;
340 document.onmousemove = CtdlMoveMsgMouseMove;
341 if (document.layers) {
342 document.captureEvents(Event.MOUSEUP | Event.MOUSEMOVE);
348 function CtdlMoveMsgMouseMove(evt) {
349 x = (ns6 ? evt.clientX : event.clientX);
350 y = (ns6 ? evt.clientY : event.clientY);
352 if ( (x == saved_x) && (y == saved_y) ) {
356 if (CtdlNumMsgsSelected < 1) {
363 drag_o_text = "<div style=\"overflow:none; background-color:#fff; color:#000; border: 1px solid black; filter:alpha(opacity=75); -moz-opacity:.75; opacity:.75;\"><tr><td>";
364 for (i=0; i<CtdlNumMsgsSelected; ++i) {
365 drag_o_text = drag_o_text +
366 ctdl_ts_getInnerText(
367 $('m'+CtdlMsgsSelected[i]).cells[0]
370 drag_o_text = drag_o_text + "<div>";
372 mm_div = document.createElement("DIV");
373 mm_div.style.position='absolute';
374 mm_div.style.top = y + 'px';
375 mm_div.style.left = x + 'px';
376 mm_div.style.pixelHeight = '300';
377 mm_div.style.pixelWidth = '300';
378 mm_div.innerHTML = drag_o_text;
379 document.body.appendChild(mm_div);
382 mm_div.style.top = y + 'px';
383 mm_div.style.left = x + 'px';
386 return false; // prevent the default mouse action from happening?
389 function CtdlMoveMsgMouseUp(evt) {
390 document.onmouseup = null;
391 document.onmousemove = null;
392 if (document.layers) {
393 document.releaseEvents(Event.MOUSEUP | Event.MOUSEMOVE);
397 document.body.removeChild(mm_div);
401 if (num_drop_targets < 1) { // nowhere to drop
405 // Did we release the mouse button while hovering over a drop target?
406 // NOTE: this only works cross-browser because the iconbar div is always
407 // positioned at 0,0. Browsers differ in whether the 'offset'
408 // functions return pos relative to the document or parent.
410 for (i=0; i<num_drop_targets; ++i) {
412 x = (ns6 ? evt.clientX : event.clientX);
413 y = (ns6 ? evt.clientY : event.clientY);
415 l = parseInt(drop_targets_elements[i].offsetLeft);
416 t = parseInt(drop_targets_elements[i].offsetTop);
417 r = parseInt(drop_targets_elements[i].offsetLeft)
418 + parseInt(drop_targets_elements[i].offsetWidth);
419 b = parseInt(drop_targets_elements[i].offsetTop)
420 + parseInt(drop_targets_elements[i].offsetHeight);
422 /* alert('Offsets are: ' + l + ' ' + t + ' ' + r + ' ' + b + '.'); */
424 if ( (x >= l) && (x <= r) && (y >= t) && (y <= b) ) {
425 // Yes, we dropped it on a hotspot.
426 CtdlMoveSelectedMessages(evt, drop_targets_roomnames[i]);
435 function ctdl_ts_getInnerText(el) {
436 if (typeof el == "string") return el;
437 if (typeof el == "undefined") { return el };
438 if (el.innerText) return el.innerText; //Not needed but it is faster
441 var cs = el.childNodes;
443 for (var i = 0; i < l; i++) {
444 switch (cs[i].nodeType) {
445 case 1: //ELEMENT_NODE
446 str += ts_getInnerText(cs[i]);
449 str += cs[i].nodeValue;