]> err.no Git - scalable-opengroupware.org/blob - UI/WebServerResources/ContactsUI.js
git-svn-id: http://svn.opengroupware.org/SOGo/inverse/trunk@1080 d1b88da0-ebda-0310...
[scalable-opengroupware.org] / UI / WebServerResources / ContactsUI.js
1 /* JavaScript for SOGoContacts */
2
3 var cachedContacts = new Array();
4 var currentContactFolder = '/personal';
5
6 var usersRightsWindowHeight = 200;
7 var usersRightsWindowWidth = 450;
8
9 function openContactWindow(sender, url) {
10   var msgWin = window.open(url, null, "width=450,height=600,resizable=0");
11   msgWin.focus();
12 }
13
14 function validateEditorInput(sender) {
15   var errortext = "";
16   var field;
17   
18   field = document.pageform.subject;
19   if (field.value == "")
20     errortext = errortext + labels.error_missingsubject + "\n";
21
22   if (!UIxRecipientSelectorHasRecipients())
23     errortext = errortext + labels.error_missingrecipients + "\n";
24   
25   if (errortext.length > 0) {
26     alert(labels.error_validationfailed.decodeEntities() + ":\n"
27           + errortext.decodeEntities());
28     return false;
29   }
30   return true;
31 }
32
33 function openContactsFolder(contactsFolder, params) {
34   if (contactsFolder != currentContactFolder || params) {
35      if (contactsFolder == currentContactFolder) {
36         var contactsList = $("contactsList");
37         if (contactsList)
38            selection = contactsList.getSelectedRowsId();
39         else
40            window.alert("no contactsList");
41      }
42      else
43       selection = null;
44
45     currentContactFolder = contactsFolder;
46     var url = URLForFolderID(currentContactFolder) +
47        "/view?noframe=1&sort=cn&desc=0";
48     if (params)
49       url += '&' + params;
50
51     var selection;
52     if (document.contactsListAjaxRequest) {
53       document.contactsListAjaxRequest.aborted = true;
54       document.contactsListAjaxRequest.abort();
55     }
56     document.contactsListAjaxRequest
57       = triggerAjaxRequest(url, contactsListCallback, selection);
58   }
59 }
60
61 function openContactsFolderAtIndex(element) {
62   var idx = element.getAttribute("idx");
63   var url = URLForFolderID(currentContactFolder) + "/view?noframe=1&idx=" + idx;
64
65   if (document.contactsListAjaxRequest) {
66     document.contactsListAjaxRequest.aborted = true;
67     document.contactsListAjaxRequest.abort();
68   }
69   document.contactsListAjaxRequest
70     = triggerAjaxRequest(url, contactsListCallback);
71 }
72
73 function contactsListCallback(http) {
74   var div = $("contactsListContent");
75
76   if (http.readyState == 4
77       && http.status == 200) {
78     document.contactsListAjaxRequest = null;
79     div.innerHTML = http.responseText;
80     var selected = http.callbackData;
81     if (selected) {
82         for (var i = 0; i < selected.length; i++)
83           $(selected[i]).select();
84     }
85     configureSortableTableHeaders();
86   }
87   else
88     log ("ajax problem 1");
89 }
90
91 function onContactFoldersContextMenu(event) {
92   var menu = $("contactFoldersMenu");
93   Event.observe(menu, "hideMenu", onContactFoldersContextMenuHide, false);
94   popupMenu(event, "contactFoldersMenu", this);
95
96   var topNode = $("contactFolders");
97   var selectedNodes = topNode.getSelectedRows();
98   topNode.menuSelectedRows = selectedNodes;
99   for (var i = 0; i < selectedNodes.length; i++)
100     selectedNodes[i].deselect();
101   topNode.menuSelectedEntry = this;
102   this.select();
103 }
104
105 function onContactContextMenu(event, element) {
106   var menu = $("contactMenu");
107   Event.observe(menu, "hideMenu", onContactContextMenuHide, false);
108   popupMenu(event, "contactMenu", element);
109
110   var topNode = $("contactsList");
111   var selectedNodes = topNode.getSelectedRows();
112   topNode.menuSelectedRows = selectedNodes;
113   for (var i = 0; i < selectedNodes.length; i++)
114     selectedNodes[i].deselect();
115   topNode.menuSelectedEntry = element;
116   element.select();
117 }
118
119 function onContactContextMenuHide(event) {
120   var topNode = $("contactsList");
121
122   if (topNode.menuSelectedEntry) {
123     topNode.menuSelectedEntry.deselect();
124     topNode.menuSelectedEntry = null;
125   }
126   if (topNode.menuSelectedRows) {
127     var nodes = topNode.menuSelectedRows;
128     for (var i = 0; i < nodes.length; i++)
129       nodes[i].select();
130     topNode.menuSelectedRows = null;
131   }
132 }
133
134 function onContactFoldersContextMenuHide(event) {
135   var topNode = $("contactFolders");
136
137   if (topNode.menuSelectedEntry) {
138     topNode.menuSelectedEntry.deselect();
139     topNode.menuSelectedEntry = null;
140   }
141   if (topNode.menuSelectedRows) {
142     var nodes = topNode.menuSelectedRows;
143     for (var i = 0; i < nodes.length; i++)
144       nodes[i].select();
145     topNode.menuSelectedRows = null;
146   }
147 }
148
149 function onFolderMenuHide(event) {
150   var topNode = $('d');
151
152   if (topNode.menuSelectedEntry) {
153     topNode.menuSelectedEntry.deselect();
154     topNode.menuSelectedEntry = null;
155   }
156   if (topNode.selectedEntry)
157     topNode.selectedEntry.select();
158 }
159
160 function loadContact(idx) {
161   if (document.contactAjaxRequest) {
162     document.contactAjaxRequest.aborted = true;
163     document.contactAjaxRequest.abort();
164   }
165
166   if (cachedContacts[currentContactFolder + "/" + idx]) {
167     var div = $('contactView');
168     div.innerHTML = cachedContacts[currentContactFolder + "/" + idx];
169   }
170   else {
171     var url = (URLForFolderID(currentContactFolder)
172                + "/" + idx + "/view?noframe=1");
173     document.contactAjaxRequest
174       = triggerAjaxRequest(url, contactLoadCallback, idx);
175   }
176 }
177
178 function contactLoadCallback(http) {
179   var div = $('contactView');
180
181   if (http.readyState == 4
182       && http.status == 200) {
183     document.contactAjaxRequest = null;
184     var content = http.responseText;
185     cachedContacts[currentContactFolder + "/" + http.callbackData] = content;
186     div.innerHTML = content;
187   }
188   else
189     log ("ajax problem 2: " + http.status);
190 }
191
192 var rowSelectionCount = 0;
193
194 validateControls();
195
196 function showElement(e, shouldShow) {
197   e.style.display = shouldShow ? "" : "none";
198 }
199
200 function enableElement(e, shouldEnable) {
201   if(!e)
202     return;
203   if(shouldEnable) {
204     if(e.hasAttribute("disabled"))
205       e.removeAttribute("disabled");
206   }
207   else {
208     e.setAttribute("disabled", "1");
209   }
210 }
211
212 function validateControls() {
213   var e = $("moveto");
214   this.enableElement(e, rowSelectionCount > 0);
215 }
216
217 function moveTo(uri) {
218   alert("MoveTo: " + uri);
219 }
220
221 /* contact menu entries */
222 function onContactRowClick(event, node) {
223   loadContact(node.getAttribute('id'));
224
225   return onRowClick(event);
226 }
227
228 function onContactRowDblClick(event, node) {
229   var contactId = node.getAttribute('id');
230
231   openContactWindow(null,
232                     URLForFolderID(currentContactFolder)
233                     + "/" + contactId + "/edit");
234
235   return false;
236 }
237
238 function onMenuEditContact(event) {
239   var contactId = document.menuTarget.getAttribute('id');
240
241   openContactWindow(null,
242                     URLForFolderID(currentContactFolder)
243                     + "/" + contactId + "/edit");
244 }
245
246 function onMenuWriteToContact(event) {
247    var contactId = document.menuTarget.getAttribute('id');
248
249    openMailComposeWindow(ApplicationBaseURL + currentContactFolder
250                          + "/" + contactId + "/write");
251 }
252
253 function onMenuDeleteContact(event) {
254   uixDeleteSelectedContacts(this);
255 }
256
257 function onToolbarEditSelectedContacts(event) {
258   var contactsList = $('contactsList');
259   var rows = contactsList.getSelectedRowsId();
260
261   for (var i = 0; i < rows.length; i++) {
262     openContactWindow(null,
263                       URLForFolderID(currentContactFolder)
264                       + "/" + rows[i] + "/edit");
265   }
266
267   return false;
268 }
269
270 function onToolbarWriteToSelectedContacts(event) {
271   var contactsList = $('contactsList');
272   var rows = contactsList.getSelectedRowsId();
273
274   for (var i = 0; i < rows.length; i++)
275     openMailComposeWindow(ApplicationBaseURL + currentContactFolder
276                           + "/" + rows[i] + "/write");
277
278   return false;
279 }
280
281 function uixDeleteSelectedContacts(sender) {
282   var failCount = 0;
283   var contactsList = $('contactsList');
284   var rows = contactsList.getSelectedRowsId();
285
286   var contactView = $('contactView');
287   contactView.innerHTML = '';
288
289   for (var i = 0; i < rows.length; i++) {
290     var url, http, rowElem;
291     
292     /* send AJAX request (synchronously) */
293     
294     url = (URLForFolderID(currentContactFolder) + "/"
295            + rows[i] + "/delete");
296     http = createHTTPClient();
297     http.open("POST", url, false /* not async */);
298     http.send("");
299     if (http.status != 200) { /* request failed */
300       failCount++;
301       http = null;
302       continue;
303     }
304     http = null;
305
306     /* remove from page */
307
308     /* line-through would be nicer, but hiding is OK too */
309     rowElem = $(rows[i]);
310     rowElem.parentNode.removeChild(rowElem);
311   }
312
313   if (failCount > 0)
314     alert("Could not delete the selected contacts!");
315   
316   return false;
317 }
318
319 function newEmailTo(sender) {
320   var mailto = sanitizeMailTo(sender.parentNode.parentNode.menuTarget.innerHTML);
321
322   if (mailto.length > 0)
323     {
324       w = window.open("compose?mailto=" + mailto,
325                       "SOGo_compose",
326                       "width=680,height=520,resizable=1,scrollbars=1,toolbar=0," +
327                       "location=0,directories=0,status=0,menubar=0,copyhistory=0");
328       w.focus();
329     }
330
331   return false; /* stop following the link */
332 }
333
334 function onHeaderClick(event) {
335   if (document.contactsListAjaxRequest) {
336     document.contactsListAjaxRequest.aborted = true;
337     document.contactsListAjaxRequest.abort();
338   }
339   url = URLForFolderID(currentContactFolder);
340 // //   log("url: " + url);
341 //   var url = "" + this.href;
342   if (url.indexOf("noframe=", 0) == -1)
343      url += "&noframe=1";
344   document.contactsListAjaxRequest
345      = triggerAjaxRequest(url, contactsListCallback);
346
347   event.preventDefault();
348 }
349
350 function registerDraggableMessageNodes() {
351   log ("can we drag...");
352 }
353
354 function newContact(sender) {
355   openContactWindow(sender,
356                     URLForFolderID(currentContactFolder) + "/new");
357
358   return false; /* stop following the link */
359 }
360
361 function onFolderSelectionChange() {
362    var folderList = $("contactFolders");
363    var nodes = folderList.getSelectedNodes();
364    $("contactView").innerHTML = '';
365   
366    if (nodes[0].hasClassName("denied")) {
367       var div = $("contactsListContent");
368       div.innerHTML = "";
369    }
370    else
371       openContactsFolder(nodes[0].getAttribute("id"), null);
372 }
373
374 function onSearchFormSubmit() {
375   var searchValue = $("searchValue");
376
377   openContactsFolder(currentContactFolder,
378                      "search=" + searchValue.value);
379
380   return false;
381 }
382
383 function onConfirmContactSelection(event) {
384    var tag = this.getAttribute("name");
385    var folderLi = $(currentContactFolder);
386    var currentContactFolderName = folderLi.innerHTML;
387    var selectorList = null;
388    var initialValues = null;
389
390    if (selector) {
391       var selectorId = selector.getAttribute("id");
392       selectorList = opener.window.document.getElementById('uixselector-'
393                                                            + selectorId
394                                                            + '-uidList');
395       initialValues = selectorList.value;
396    }
397
398    var contactsList = $("contactsList");
399    var rows = contactsList.getSelectedRows();
400    for (i = 0; i < rows.length; i++) {
401       var cid = rows[i].getAttribute("contactid");
402       var cname = '' + rows[i].getAttribute("contactname");
403       var email = '' + rows[i].cells[1].innerHTML;
404       opener.window.addContact(tag, currentContactFolderName + '/' + cname,
405                                cid, cname, email);
406    }
407
408    if (selector && selector.changeNotification
409        && selectorList.value != initialValues)
410       selector.changeNotification("addition");
411
412    event.preventDefault();
413 }
414
415 function onContactMailTo(node) {
416   return openMailTo(node.innerHTML);
417 }
418
419 function refreshContacts(contactId) {
420   openContactsFolder(currentContactFolder, "reload=true");
421   cachedContacts[currentContactFolder + "/" + contactId] = null;
422   loadContact(contactId);
423
424   return false;
425 }
426
427 function onAddressBookNew(event) {
428   var name = window.prompt(labels["Name of the Address Book"].decodeEntities());
429   if (name) {
430     if (document.newAbAjaxRequest) {
431       document.newAbAjaxRequest.aborted = true;
432       document.newAbAjaxRequest.abort();
433     }
434     var url = ApplicationBaseURL + "/newAb?name=" + name;
435     document.newAbAjaxRequest
436        = triggerAjaxRequest(url, newAbCallback, name);
437   }
438   preventDefault(event);
439 }
440
441 function appendAddressBook(name, folder) {
442    var li = document.createElement("li");
443    $("contactFolders").appendChild(li);
444    li.setAttribute("id", folder);
445    li.appendChild(document.createTextNode(name));
446    setEventsOnContactFolder(li);
447 }
448
449 function newAbCallback(http) {
450   if (http.readyState == 4
451       && http.status == 201) {
452      var name = http.callbackData;
453      appendAddressBook(name, "/" + name);
454   }
455   else
456     log ("ajax problem 4:" + http.status);
457 }
458
459 function newUserFolderCallback(folderData) {
460    var folder = $(folderData["folder"]);
461    if (!folder)
462       appendAddressBook(folderData["folderName"], folderData["folder"]);
463 }
464
465 function onAddressBookAdd(event) {
466    openUserFolderSelector(newUserFolderCallback, "contact");
467
468    event.preventDefault();
469 }
470
471 function onFolderUnsubscribeCB(folderId) {
472    var node = $(folderId);
473    node.parentNode.removeChild(node);
474    var personal = $("/personal");
475    personal.select();
476    onFolderSelectionChange();
477 }
478
479 function onAddressBookRemove(event) {
480   var selector = $("contactFolders");
481   var nodes = selector.getSelectedNodes();
482   if (nodes.length > 0) { 
483      nodes[0].deselect();
484      var folderId = nodes[0].getAttribute("id");
485      var folderIdElements = folderId.split(":");
486      if (folderIdElements.length > 1)
487         unsubscribeFromFolder(folderId, onFolderUnsubscribeCB, folderId);
488      else {
489         var abId = folderIdElements[0].substr(1);
490         deletePersonalAddressBook(abId);
491         var personal = $("/personal");
492         personal.select();
493         onFolderSelectionChange();
494      }
495   }
496
497   event.preventDefault();
498 }
499
500 function deletePersonalAddressBook(folderId) {
501    var label
502       = labels["Are you sure you want to delete the selected address book?"];
503    if (window.confirm(label.decodeEntities())) {
504       if (document.deletePersonalABAjaxRequest) {
505          document.deletePersonalABAjaxRequest.aborted = true;
506          document.deletePersonalABAjaxRequest.abort();
507       }
508       var url = ApplicationBaseURL + "/" + folderId + "/delete";
509       document.deletePersonalABAjaxRequest
510          = triggerAjaxRequest(url, deletePersonalAddressBookCallback,
511                               folderId);
512    }
513 }
514
515 function deletePersonalAddressBookCallback(http) {
516   if (http.readyState == 4) {
517      if (http.status == 200) {
518         var ul = $("contactFolders");
519         
520         var children = ul.childNodesWithTag("li");
521         var i = 0;
522         var done = false;
523         while (!done && i < children.length) {
524            var currentFolderId = children[i].getAttribute("id").substr(1);
525            if (currentFolderId == http.callbackData) {
526               ul.removeChild(children[i]);
527               done = true;
528            }
529            else
530               i++;
531         }
532      }
533      document.deletePersonalABAjaxRequest = null;
534   }
535   else
536      log ("ajax problem");
537 }
538
539 function configureDragHandles() {
540   var handle = $("dragHandle");
541   if (handle) {
542     handle.addInterface(SOGoDragHandlesInterface);
543     handle.leftBlock=$("contactFoldersList");
544     handle.rightBlock=$("rightPanel");
545   }
546
547   handle = $("rightDragHandle");
548   if (handle) {
549     handle.addInterface(SOGoDragHandlesInterface);
550     handle.upperBlock=$("contactsListContent");
551     handle.lowerBlock=$("contactView");
552   }
553 }
554
555 function lookupDeniedFolders() {
556   var list = $("contactFolders").childNodesWithTag("li");
557   for (var i = 0; i < list.length; i++) {
558      var folderID = list[i].getAttribute("id");
559      var url = URLForFolderID(folderID) + "/canAccessContent";
560      
561      triggerAjaxRequest(url, deniedFoldersLookupCallback, folderID);
562   }
563 }
564
565 function deniedFoldersLookupCallback(http) {
566    if (http.readyState == 4) {
567       var denied = ! isHttpStatus204(http.status);
568       var entry = $(http.callbackData);
569       if (denied)
570          entry.addClassName("denied");
571       else
572          entry.removeClassName("denied");
573    }
574 }
575
576 function configureAbToolbar() {
577   var toolbar = $("abToolbar");
578   var links = toolbar.childNodesWithTag("a");
579   Event.observe(links[0], "click", onAddressBookNew, false);
580   Event.observe(links[1], "click", onAddressBookAdd, false);
581   Event.observe(links[2], "click", onAddressBookRemove, false);
582 }
583
584 function configureContactFolders() {
585   var contactFolders = $("contactFolders");
586   if (contactFolders) {
587     Event.observe(contactFolders, "mousedown", listRowMouseDownHandler);
588     Event.observe(contactFolders, "click", onFolderSelectionChange);
589     var lis = contactFolders.childNodesWithTag("li");
590     for (var i = 0; i < lis.length; i++)
591       setEventsOnContactFolder(lis[i]);
592
593     lookupDeniedFolders();
594
595     var personalFolder = $("/personal");
596     personalFolder.select();
597   }
598 }
599
600 function setEventsOnContactFolder(node) {
601   Event.observe(node, "mousedown", listRowMouseDownHandler, false);
602   Event.observe(node, "click", onRowClick, false);
603   Event.observe(node, "contextmenu", onContactFoldersContextMenu.bindAsEventListener(node), false);
604 }
605
606 function onMenuSharing(event) {
607    var folders = $("contactFolders");
608    var selected = folders.getSelectedNodes()[0];
609    var title = this.innerHTML;
610    var url = URLForFolderID(selected.getAttribute("id"));
611
612    openAclWindow(url + "/acls", title);
613 }
614
615 function getMenus() {
616    var menus = {};
617    menus["contactFoldersMenu"] = new Array(null, "-", null,
618                                            null, "-", null, "-",
619                                            onMenuSharing);
620    menus["contactMenu"] = new Array(onMenuEditContact, "-",
621                                     onMenuWriteToContact, null, "-",
622                                     onMenuDeleteContact);
623    menus["searchMenu"] = new Array(setSearchCriteria);
624
625    return menus;
626 }
627
628 function configureSearchField() {
629    var searchValue = $("searchValue");
630
631    Event.observe(searchValue, "mousedown", onSearchMouseDown.bindAsEventListener(searchValue), false);
632    Event.observe(searchValue, "click", popupSearchMenu.bindAsEventListener(searchValue), false);
633    Event.observe(searchValue, "blur", onSearchBlur.bindAsEventListener(searchValue), false);
634    Event.observe(searchValue, "focus", onSearchFocus.bindAsEventListener(searchValue), false);
635    Event.observe(searchValue, "keydown", onSearchKeyDown.bindAsEventListener(searchValue), false);
636 }
637
638 function configureSelectionButtons() {
639    var container = $("contactSelectionButtons");
640    if (container) {
641       var buttons = container.childNodesWithTag("input");
642       for (var i = 0; i < buttons.length; i++)
643         Event.observe(buttons[i], "click", onConfirmContactSelection.bindAsEventListener(buttons[i]), false);
644    }
645 }
646
647 function initContacts(event) {
648    if (!document.body.hasClassName("popup")) {
649      configureAbToolbar();
650      configureSearchField();
651    }
652    else
653      configureSelectionButtons();
654    configureContactFolders();
655 //     initDnd();
656 }
657
658 addEvent(window, 'load', initContacts);