1 /*********************************************************************************
2 * SugarCRM Community Edition is a customer relationship management program developed by
3 * SugarCRM, Inc. Copyright (C) 2004-2011 SugarCRM Inc.
5 * This program is free software; you can redistribute it and/or modify it under
6 * the terms of the GNU Affero General Public License version 3 as published by the
7 * Free Software Foundation with the addition of the following permission added
8 * to Section 15 as permitted in Section 7(a): FOR ANY PART OF THE COVERED WORK
9 * IN WHICH THE COPYRIGHT IS OWNED BY SUGARCRM, SUGARCRM DISCLAIMS THE WARRANTY
10 * OF NON INFRINGEMENT OF THIRD PARTY RIGHTS.
12 * This program is distributed in the hope that it will be useful, but WITHOUT
13 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
14 * FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
17 * You should have received a copy of the GNU Affero General Public License along with
18 * this program; if not, see http://www.gnu.org/licenses or write to the Free
19 * Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
22 * You can contact SugarCRM, Inc. headquarters at 10050 North Wolfe Road,
23 * SW2-130, Cupertino, CA 95014, USA. or at email address contact@sugarcrm.com.
25 * The interactive user interfaces in modified source and object code versions
26 * of this program must display Appropriate Legal Notices, as required under
27 * Section 5 of the GNU Affero General Public License version 3.
29 * In accordance with Section 7(b) of the GNU Affero General Public License version 3,
30 * these Appropriate Legal Notices must retain the display of the "Powered by
31 * SugarCRM" logo. If the display of the logo is not reasonably feasible for
32 * technical reasons, the Appropriate Legal Notices must display the words
33 * "Powered by SugarCRM".
34 ********************************************************************************/
38 Event = YAHOO.util.Event,
39 Connect = YAHOO.util.Connect,
43 ///////////////////////////////////////////////////////////////////////////////
46 _contactCache : new Array(), // cache of contacts
47 _dd : new Array(), // filtered list, same format as _contactCache
48 _ddLists : new Array(), // list of Lists
49 _dd_mlUsed : new Array(), // contacts in mailing list edit view column1
50 _dd_mlAvailable : new Array(), // contacts in mailing list edit view column2
51 clickBubble : true, // hack to get around onclick event bubbling
56 itemSpacing : 'white-space:nowrap; padding:2px;',
62 * YUI bug fix 2527707. Causes nested datatable's in <tables> to cause 404 errors whens earching.
64 initFixForDatatableSort: function () {
65 //Workaround for YUI bug 2527707: http://yuilibrary.com/projects/yui2/ticket/913efafad48ce433199f3e72e4847b18, should be removed when YUI 2.8+ is used
66 YAHOO.widget.DataTable.prototype.getColumn = function(column) {
67 var oColumn = this._oColumnSet.getColumn(column);
70 // Validate TD element
71 var elCell = column.nodeName.toLowerCase() != "th" ? this.getTdEl(column) : false;
73 oColumn = this._oColumnSet.getColumn(elCell.cellIndex);
75 // Validate TH element
77 elCell = this.getThEl(column);
80 var allColumns = this._oColumnSet.flat;
81 for(var i=0, len=allColumns.length; i<len; i++) {
82 if(allColumns[i].getThEl().id === elCell.id) {
83 oColumn = allColumns[i];
94 cancelEdit : function() {
95 if(this.editContactDialog)
96 this.editContactDialog.hide();
97 if(this.editMailingListDialog)
98 this.editMailingListDialog.hide();
105 var t = document.getElementById('contactsFilter');
111 * handle context-menu Compose-to call
112 * @param string type 'contacts' or 'lists'
114 composeTo : function(type, waited) {
115 var activePanel = SUGAR.email2.innerLayout.get("activeTab").get("id")
116 if (activePanel.substring(0, 10) != "composeTab") {
117 SE.composeLayout.c0_composeNewEmail();
118 setTimeout("SE.addressBook.composeTo('" + type + "', true);");
119 SE.contextMenus.contactsContextMenu.hide();
122 var idx = activePanel.substring(10);
125 // determine if we have a selection to work with
126 if(type == 'contacts') {
127 var ids = SE.contactView.getSelectedRows();
129 rows[i] = SE.contactView.getRecord(ids[i]);
131 removeHiddenNodes(rows, SE.contactView);
135 if(rows.length > 0) {
136 SE.composeLayout.handleDrop(
137 (type == 'contacts') ? SE.contactView : SE.emailListsView,
138 null, rows, 'addressTO' + idx );
140 alert(app_strings.LBL_EMAIL_MENU_MAKE_SELECTION);
144 editContact : function() {
145 SE.contextMenus.contactsContextMenu.hide();
146 var element = SE.contactView.getSelectedNodes()[0];
148 if (element.className.indexOf('address-contact') > -1) {
149 elementId = element.id;
150 } else if (element.className.indexOf('address-exp-contact') > -1) {
151 elementId = element.id.substring(2);
157 * Filters contact entries based on user input
159 filter : function(inputEl) {
160 var ret = new Object();
161 var re = new RegExp(inputEl.value, "gi");
163 for(var i in this._contactCache) {
164 if(this._contactCache[i].name.match(re)) {
165 ret[i] = this._contactCache[i];
169 this.buildContactList(ret);
172 fullForm : function(id, module) {
173 document.location = "index.php?return_module=Emails&return_action=index&module=" + module + "&action=EditView&record=" + id;
177 * returns a formatted email address from the addressBook cache
179 getFormattedAddress : function(id) {
180 var o = this._contactCache[id];
181 var primaryEmail = '';
183 for(var i=0; i<o.email.length; i++) {
184 var currentEmail = o.email[i].email_address;
186 if(o.email[i].primary_address == 1) {
187 primaryEmail = o.email[i].email_address;
191 var finalEmail = (primaryEmail == "") ? currentEmail : primaryEmail;
192 var name = new String(o.name);
193 var finalName = name.replace(/(<([^>]+)>)/ig, "").replace(/'/gi,'\'');
194 var ret = finalName + " <" + finalEmail.replace(/'/gi,'\'') + ">";
200 * Sets up async call to query for matching contacts, users, etc.
202 searchContacts : function() {
203 var fn = document.getElementById('input_searchField').value;
204 var pe = document.getElementById('input_searchPerson').value;
206 var rb = document.getElementById('hasRelatedBean').checked;
209 var relatedBeanId = document.getElementById('data_parent_id' + idx).value;
210 var relatedBeanType = document.getElementById('data_parent_type' + idx).value;
211 this.addressBookDataModel.params['related_bean_id'] = relatedBeanId;
212 this.addressBookDataModel.params['related_bean_type'] = relatedBeanType;
214 this.addressBookDataModel.params['related_bean_id'] = '';
217 this.addressBookDataModel.params['search_field'] = fn;
218 this.addressBookDataModel.params['person'] = pe;
219 this.addressBookDataModel.params['emailUIAction'] = 'getAddressSearchResults';
220 this.grid._oDataSource = this.addressBookDataModel;
221 this.grid.getDataSource().sendRequest(SUGAR.util.paramsToUrl(this.addressBookDataModel.params), this.grid.onDataReturnInitializeTable, this.grid);
225 * Clear Search Crieteria For Addressbook
227 clearAddressBookSearch : function() {
228 document.getElementById('input_searchField').value = "";
229 document.getElementById('input_searchPerson').selectedIndex = 0;
233 * Opens modal select window to add contacts to addressbook
235 selectContactsDialogue : function(destId) {
236 if(!this.contactsDialogue) {
237 var dlg = this.contactsDialogue = new YAHOO.widget.Dialog("contactsDialogue", {
241 constraintoviewport: true,
243 buttons : [{text: app_strings.LBL_EMAIL_ADDRESS_BOOK_ADD, isDefault: true, handler: this.populateEmailAddressFieldsFromResultTable},
244 {text: app_strings.LBL_EMAIL_ADDRESS_BOOK_CLEAR, isDefault: true, handler: this.clearAllEmailAddressFieldsFromResultTable} ]
246 dlg.setHeader(app_strings.LBL_EMAIL_ADDRESS_BOOK_SELECT_TITLE);
248 var body = SUGAR.util.getAndRemove("contactsDialogueHTML");
249 dlg.setBody(body.innerHTML);
250 dlg.renderEvent.subscribe(function() {
251 var iev = YAHOO.util.Dom.get("contactsDialogueBody");
252 if (iev && !SUGAR.isIE) {
253 this.body.style.width = "950px";
258 dlg.beforeRenderEvent.subscribe(function() {
259 var dd = new YAHOO.util.DDProxy(dlg.element);
260 dd.setHandleElId(dlg.header);
261 dd.on('endDragEvent', function() {
267 var tp = new YAHOO.widget.TabView("contactsSearchTabs");
269 var tabContent = SUGAR.util.getAndRemove("searchForm");
270 tp.addTab(new YAHOO.widget.Tab({
271 label: app_strings.LBL_EMAIL_ADDRESS_BOOK_TITLE,
273 content : tabContent.innerHTML,
274 id : "addressSearchTab",
278 var addListenerFields = ['input_searchPerson','input_searchField' ]
279 YAHOO.util.Event.addListener(addListenerFields,"keydown", function(e){
280 if (e.keyCode == 13) {
281 YAHOO.util.Event.stopEvent(e);
282 SUGAR.email2.addressBook.searchContacts();
286 this.contactsDialogue.render();
289 //Quick Compose does not have an innerLayout component and will always be referenced with ix 0.
290 if (typeof(SUGAR.email2.innerLayout) == 'undefined')
294 var activePanel = SUGAR.email2.innerLayout.get("activeTab").get("id");
295 var idx = activePanel.substring(10);
297 SE.addressBook.idx = idx;
300 if ((hasRelatedBeanId = document.getElementById('data_parent_id' + idx).value) != '') {
301 document.getElementById('relatedBeanColumn').style.display = '';
302 var relatedBeanName = document.getElementById('data_parent_name' + idx).value;
303 var relatedBeanType = document.getElementById('data_parent_type' + idx).value;
304 relatedBeanId = document.getElementById('data_parent_id' + idx).value;
305 document.getElementById('relatedBeanInfo').innerHTML = ' ' + relatedBeanType + ' <b>' + relatedBeanName + '</b>';
306 SE.addressBook.relatedBeanType = relatedBeanType;
308 document.getElementById('relatedBeanColumn').style.display = 'none';
309 document.getElementById('hasRelatedBean').checked = false;
312 if (!SE.addressBook.grid)
314 if (hasRelatedBeanId) {
315 document.getElementById('hasRelatedBean').checked = true;
317 AddressSearchGridInit();
318 SE.addressBook.relatedBeanId = relatedBeanId;
322 if (typeof(relatedBeanId) != 'undefined' && relatedBeanId != SE.addressBook.relatedBeanId)
324 SE.addressBook.relatedBeanId = relatedBeanId;
325 document.getElementById('hasRelatedBean').checked = true;
327 if (document.getElementById('hasRelatedBean').checked == true)
329 SE.addressBook.addressBookDataModel.params['related_bean_id'] = relatedBeanId;
330 SE.addressBook.addressBookDataModel.params['related_bean_type'] = relatedBeanType;
332 SE.addressBook.addressBookDataModel.params['related_bean_id'] = '';
333 SE.addressBook.addressBookDataModel.params['related_bean_type'] = '';
335 SE.addressBook.addressBookDataModel.params['search_field'] = document.getElementById('input_searchField').value;;
336 SE.addressBook.addressBookDataModel.params['person'] = document.getElementById('input_searchPerson').value;
337 SE.addressBook.grid.getDataSource().sendRequest(SUGAR.util.paramsToUrl(SE.addressBook.addressBookDataModel.params), SE.addressBook.grid.onDataReturnInitializeTable, SE.addressBook.grid);
340 //Remove any lingering rows in the result set table if the module was closed.
341 SE.addressBook.gridResults.deleteRows(0, SUGAR.email2.addressBook.gridResults.getRecordSet().getLength());
343 SE.addressBook.populateResulstTableEmailAddresses();
345 this.contactsDialogue.show();
348 * Clear all email addresses from result table.
351 clearAllEmailAddressFieldsFromResultTable: function () {
352 SUGAR.email2.addressBook.gridResults.deleteRows(0, SUGAR.email2.addressBook.gridResults.getRecordSet().getLength());
353 //Unhighlight any rows currently selected if the emails were cleared.
354 SUGAR.email2.addressBook.grid.toggleSelectAll(false);
355 SUGAR.email2.addressBook.grid.reSelectRowsOnRender();
358 * Take all email address listed in the compose tab To|Cc|Bcc fields and re-populates the
359 * results table. This function is called when the address book is displayed.
361 populateResulstTableEmailAddresses: function () {
363 var idx = SE.addressBook.idx;
364 var emailFields = ['to','cc','bcc'];
366 for(var k=0;k<emailFields.length;k++)
368 var elKey = 'address' + emailFields[k].toUpperCase() + idx;
369 var allEmails = document.getElementById(elKey).value;
373 var formatedEmails = SE.composeLayout._getEmailArrayFromString(allEmails);
375 for (var i=0; i<formatedEmails.length; i++)
377 var t_name = formatedEmails[i].name;
378 var t_emailAddr = formatedEmails[i].email_address;
379 var displayEmail = t_name + ' <' + t_emailAddr + '>';
381 t_name = displayEmail = t_emailAddr;
383 var addressType = SE.addressBook.translateAddresType(emailFields[k],true);
384 SUGAR.email2.addressBook.gridResults.addRow({'type':addressType,'name':t_name,'email_address': t_emailAddr,
385 'display_email_address': displayEmail,'bean_id': -1,'idx' : SE.addressBook.idx});
391 * Checks all entries in the result table against a particular email address, returning true
392 * if the email address is found, false otherwise.
394 doesEmailAdddressExistInResultTable: function(emailAddress)
396 if(trim(emailAddress) == '')
399 var emailAddressFound = false;
400 var contacts = SE.addressBook.gridResults.getRecordSet().getRecords();
401 for (var i=0; i < contacts.length; i++)
403 var data = SE.addressBook.gridResults.getRecord(contacts[i]).getData();
404 //If we are adding to cc or bcc fields, make them visible.
405 if(data.email_address == emailAddress)
407 emailAddressFound = true;
412 return emailAddressFound;
415 * Takes all email addresses that the users wishes to add from the address book and populates the To
416 * fields on the compose tab.
418 populateEmailAddressFieldsFromResultTable: function()
420 //Clear the fields first, all email addresses are stored in the address book
421 var idx = SE.addressBook.idx;
422 var emailFields = ['to','cc','bcc'];
423 for(var k=0;k<emailFields.length;k++)
425 var elKey = 'address' + emailFields[k].toUpperCase() + idx;
426 document.getElementById(elKey).value = "";
429 var contacts = SE.addressBook.gridResults.getRecordSet().getRecords();
430 for (var i=0; i < contacts.length; i++)
432 var data = SE.addressBook.gridResults.getRecord(contacts[i]).getData();
434 var addressTypeKey = SE.addressBook.translateAddresType(data.type,false);
435 //If we are adding to cc or bcc fields, make them visible.
436 if(addressTypeKey =='cc' || addressTypeKey =='bcc')
437 SE.composeLayout.showHiddenAddress(addressTypeKey,data.idx);
438 //Construct the target id
439 var target_id = 'address' + addressTypeKey.toUpperCase() + data.idx
441 var target = document.getElementById(target_id);
442 target.value = SE.addressBook.smartAddEmailAddressToComposeField(target.value, data.display_email_address);
445 //Delete all rows from the result set table
446 SUGAR.email2.addressBook.gridResults.deleteRows(0, SUGAR.email2.addressBook.gridResults.getRecordSet().getLength());
449 SE.addressBook.contactsDialogue.hide()
452 * Insert contacts into the result table.
454 insertContactToResultTable : function(event,address_type) {
456 var contactsDialogue = SE.addressBook.contactsDialogue;
457 var contacts = SE.addressBook.grid.getSelectedRows();
459 var rows = SUGAR.email2.addressBook.grid.getRecordSet().getRecords();
460 for (var i = 0; i < rows.length; i++)
462 if (typeof(rows[i]) != "undefined" && rows[i].getData().checked )
464 var recId = SE.addressBook.grid.getRecord(rows[i]).getId();
465 SE.addressBook.insertContactRowToResultTable(recId,address_type);
466 SUGAR.email2.addressBook.grid.selectRow(rows[i]);
467 rows[i].setData("selected",true);
470 var checkBoxes = SUGAR.email2.addressBook.grid.get("element").getElementsByTagName('input');
471 for (var i = 0; i < checkBoxes.length; i++) {
472 checkBoxes[i].checked = false;
478 insertContactRowToResultTable : function(rowId, addressType) {
479 var data = SE.addressBook.grid.getRecord(rowId).getData();
480 if(SE.addressBook.doesGridResultsEntryExist(data.email) )
482 var name = data.name.replace(/'/gi,'\'').replace(/"/gi,'"');
483 var ea = name + ' <' + data.email.replace(/'/gi,'\'') + '>';
484 if(addressType == null)
485 addressType = app_strings.LBL_EMAIL_ADDRESS_BOOK_ADD_TO.replace(/:$/,''); //Default to To when using the plus icon.
486 SUGAR.email2.addressBook.gridResults.addRow({'type':addressType,'name':name,'email_address': data.email,'display_email_address': ea,'bean_id': data.bean_id,'idx' : SE.addressBook.idx});
489 * Remove a row from the gridsResult table.
491 removeRowFromGridResults : function(rowId,emailAddress)
493 var contacts = SE.addressBook.gridResults.getRecordSet().getRecords();
494 for (var i=0; i < contacts.length; i++)
496 var rec = SE.addressBook.gridResults.getRecord(contacts[i]);
497 var data = rec.getData();
498 if(data.email_address == emailAddress)
500 SUGAR.email2.addressBook.gridResults.deleteRow(rec.getId());
505 SUGAR.email2.addressBook.toggleSearchRowIcon(rowId,true);
508 * Translates between the addressType To|Cc|Bcc labels/keys.
510 translateAddresType: function(addressType,fromKey)
512 var displayTo = app_strings.LBL_EMAIL_ADDRESS_BOOK_ADD_TO.replace(/:$/,'');
513 var displayCc = app_strings.LBL_EMAIL_ADDRESS_BOOK_ADD_CC.replace(/:$/,'');
514 var displayBcc = app_strings.LBL_EMAIL_ADDRESS_BOOK_ADD_BCC.replace(/:$/,'');
515 var mappingObject = {};
518 mappingObject = {'to':displayTo, 'cc':displayCc, 'bcc':displayBcc};
521 mappingObject[displayTo] = 'to'; //Cant use object literal with variable variable.
522 mappingObject[displayCc] = 'cc';
523 mappingObject[displayBcc] = 'bcc';
526 return typeof(mappingObject[addressType]) != 'undefined' ? mappingObject[addressType] : '';
532 toggleSearchRowIcon : function(rowId,show)
536 var idToShow = rowId + '_add_img';
537 var idToHide = rowId + '_rm_img';
541 var idToShow = rowId + '_rm_img';
542 var idToHide = rowId + '_add_img';
546 Dom.addClass(idToHide, "yui-hidden");
547 Dom.removeClass(idToShow, "yui-hidden");
550 * Determine if an entry has already been added to the grid results table to prevent duplicates.
552 doesGridResultsEntryExist: function(emailAddrs)
555 var contactExists = false;
556 var contacts = SE.addressBook.gridResults.getRecordSet().getRecords();
557 for (var i=0; i < contacts.length; i++)
559 var data = SE.addressBook.gridResults.getRecord(contacts[i]).getData();
560 if(data.email_address == emailAddrs)
562 contactExists = true;
566 return contactExists;
570 * adds an email address to a string, but first checks if it exists
571 * @param string concat The string we are appending email addresses to
572 * @param string addr Email address to add
575 smartAddEmailAddressToComposeField : function(concat, addr) {
576 var re = new RegExp(addr);
578 if(!concat.match(re)) {
580 concat += "; " + addr;
589 //// END ADDRESS BOOK
590 ///////////////////////////////////////////////////////////////////////////////
594 ///////////////////////////////////////////////////////////////////////////////
597 * Auto-complete object
601 delimChar : [";", ","],
605 prehighlightClassName : "yui-ac-prehighlight",
608 instances : new Array(),
611 * Parses an addressBook entry looking for primary address. If not found, it will return the last found address.
612 * @param object Contact from AddressBook
615 getPrimaryAddress : function(contact) {
616 var address = app_strings.LBL_EMAIL_ADDRESS_BOOK_NOT_FOUND;
618 for(var eIndex in contact.email) {
619 address = contact.email[eIndex].email_address;
620 if(contact.email[eIndex].primary_address == 1) {
621 return contact.email[eIndex].email_address;
629 * initializes autocomplete widgets for a given compose view
632 init : function(idx) {
633 var ds = new YAHOO.widget.DS_JSArray(this.returnDataSource(SE.addressBook._contactCache), {
634 "queryMatchContains" : false,
635 "queryMatchSubset" : true
638 this.instances[idx] = {
645 // instantiate the autoComplete widgets
646 this.instances[idx]['to'] = new YAHOO.widget.AutoComplete('addressTO'+idx, "addressToAC"+idx, ds, this.config);
647 this.instances[idx]['cc'] = new YAHOO.widget.AutoComplete('addressCC'+idx, "addressCcAC"+idx, ds, this.config);
648 this.instances[idx]['bcc'] = new YAHOO.widget.AutoComplete('addressBCC'+idx, "addressBccAC"+idx, ds, this.config);
650 // enable hiding of interfering textareas
651 this.instances[idx]['to'].containerExpandEvent.subscribe(SE.autoComplete.toggleTextareaHide);
652 this.instances[idx]['cc'].containerExpandEvent.subscribe(SE.autoComplete.toggleTextareaHide);
653 this.instances[idx]['bcc'].containerExpandEvent.subscribe(SE.autoComplete.toggleTextareaHide);
655 // enable reshowing of hidden textareas
656 this.instances[idx]['to'].containerCollapseEvent.subscribe(SE.autoComplete.toggleTextareaShow);
657 this.instances[idx]['cc'].containerCollapseEvent.subscribe(SE.autoComplete.toggleTextareaShow);
658 this.instances[idx]['bcc'].containerCollapseEvent.subscribe(SE.autoComplete.toggleTextareaShow);
660 // enable refreshes of contact lists
661 this.instances[idx]['to'].textboxFocusEvent.subscribe(SE.autoComplete.refreshDataSource);
662 this.instances[idx]['cc'].textboxFocusEvent.subscribe(SE.autoComplete.refreshDataSource);
663 this.instances[idx]['bcc'].textboxFocusEvent.subscribe(SE.autoComplete.refreshDataSource);
666 refreshDataSource : function(sType, aArgs) {
667 var textBoxId = aArgs[0].getInputEl().id; // "addressTo0"
669 var refresh = SE.autoComplete.returnDataSource(SE.addressBook._contactCache);
671 if(textBoxId.indexOf("addressTO") > -1 || textBoxId.indexOf("addressCC") > -1) {
672 idx = textBoxId.substr(9);
674 idx = textBoxId.substr(10);
677 SE.autoComplete.instances[idx]['to'].dataSource.data = refresh;
678 SE.autoComplete.instances[idx]['cc'].dataSource.data = refresh;
679 SE.autoComplete.instances[idx]['bcc'].dataSource.data = refresh;
683 * Parses AddressBook entries to return an appropriate DataSource array for YUI.autoComplete
685 returnDataSource : function(contacts) {
686 var ret = new Array();
687 for(var id in contacts) {
688 if (contacts[id].name) {
689 var primary = this.getPrimaryAddress(contacts[id]);
691 ret[ret.length] = contacts[id].name.replace(/<[\/]*b>/gi, '') + " <" + primary + ">";
692 //ret[ret.length] = contacts[id].name + " <" + primary + ">";
694 for(var emailIndex in contacts[id].email) {
695 ret[ret.length] = contacts[id].email[emailIndex].email_address;
704 * Hides address textareas to prevent autocomplete dropdown from being obscured
706 toggleTextareaHide : function(sType, aArgs) {
707 var textBoxId = aArgs[0]._oTextbox.id; // "addressTo0"
711 if(textBoxId.indexOf("addressTO") > -1) {
713 } else if(textBoxId.indexOf("addressCC") > -1) {
716 idx = textBoxId.substr(9);
718 // follow through if not BCC
720 var cc = document.getElementById("addressCC" + idx);
721 var bcc = document.getElementById("addressBCC" + idx);
725 cc.style.visibility = 'hidden';
727 bcc.style.visibility = 'hidden';
734 * Redisplays the textareas after an address is commited
736 toggleTextareaShow : function(sType, aArgs) {
737 var textBoxId = aArgs[0]._oTextbox.id; // "addressTo0"
741 if(textBoxId.indexOf("addressTO") > -1) {
743 } else if(textBoxId.indexOf("addressCC") > -1) {
746 idx = textBoxId.substr(9);
748 // follow through if not BCC
750 document.getElementById("addressCC" + idx).style.visibility = 'visible';
751 document.getElementById("addressBCC" + idx).style.visibility = 'visible';
756 //// END AUTOCOMPLETE
757 ///////////////////////////////////////////////////////////////////////////////
759 ///////////////////////////////////////////////////////////////////////////////
762 * expands the options sidebar
765 currentInstanceId : 0,
768 outboundAccountErrors : null,
769 loadedTinyInstances : {}, //Tracks which tinyMCE editors have initalized with html content.
771 showAddressDetails : function(e) {
772 var linkElement = document.getElementById("More"+e.id);
773 var spanElement = document.getElementById("Detail"+e.id);
774 var emailAddressList = e.value;
775 if(e.value.length > 96)
777 var resultArray = SE.composeLayout._getEmailArrayFromString(emailAddressList);
778 var displayArray = [];
779 for (var i=0; i<resultArray.length; i++)
781 var t_name = resultArray[i].name;
782 var t_emailAddr = resultArray[i].email_address;
784 displayArray.push('<br/><' + t_emailAddr + '>');
786 displayArray.push(t_name + '<br/><' + t_emailAddr + '>');
789 var result = displayArray.join('<br/>');
791 linkElement.style.display = "inline";
792 linkElement.style.height="10px";
793 linkElement.style.overflow="visible";
794 spanElement.innerHTML = result;
797 linkElement.style.display = "none";
802 * Given a string of email address, return an array containing the name portion (if available)
805 _getEmailArrayFromString : function (emailAddressList){
808 while ((results = reg.exec(emailAddressList)) != null)
810 orignial = results[0];
811 parsedResult = results[0].replace(';', ':::::');
812 emailAddressList = emailAddressList.replace (orignial, parsedResult);
816 while ((results = reg.exec(emailAddressList)) != null)
818 orignial = results[0];
819 parsedResult = results[0].replace(',', ':::::');
820 emailAddressList = emailAddressList.replace (orignial, parsedResult);
822 //Administrator <johndoe@som.com> ;1@somwhe.com;2@somwherecomplex.com,3@somwherecomplex.com;4@somwherecomplex.com,5@somwherecomplex.com,
823 var emailArr = emailAddressList.split(":::::");
824 var resultsArray = [];
826 for (var i=0; i<emailArr.length; i++)
828 var rposition = emailArr[i].indexOf('<');
829 var lposition = emailArr[i].indexOf('>');
831 if(trim(emailArr[i]) != '')
833 if(rposition != -1 && lposition != -1)
835 var t_name = emailArr[i].substr(0, rposition-1);
836 var t_emailAddr = emailArr[i].substr(rposition+1, (lposition-1 - rposition) );
837 resultsArray.push({'name':t_name, 'email_address': t_emailAddr});
841 resultsArray.push({'name':'', 'email_address': emailArr[i]});
847 ///////////////////////////////////////////////////////////////////////////
850 * Prepare bucket DIV and yui-ext tab panels
852 _0_yui : function() {
853 var idx = this.currentInstanceId;
855 var composeTab = new YAHOO.SUGAR.ClosableTab({
856 label: mod_strings.LNK_NEW_SEND_EMAIL,
858 content : "<div id='htmleditordiv" + idx + "'/>",
859 id : "composeTab" + idx,
860 closeMsg: app_strings.LBL_EMAIL_CONFIRM_CLOSE,
863 SE.innerLayout.addTab(composeTab);
865 // get template engine with template
866 if (!SE.composeLayout.composeTemplate) {
867 SE.composeLayout.composeTemplate = new YAHOO.SUGAR.Template(SE.templates['compose']);
870 // create Tab inner layout
871 var composePanel = this.getComposeLayout();
872 composePanel.getUnitByPosition("right").collapse();
873 composePanel.autoSize();
877 * Generate the quick compose layout
878 * @method getQuickComposeLayout
879 * @param {Pannel} parentPanel Parent pannel
880 * @param {Object} o Options
883 getQuickComposeLayout : function (parentPanel,o) {
884 var idx = SE.composeLayout.currentInstanceId;
886 //Before rendering the parent pannel we need to initalize the grid layout
887 parentPanel.beforeRenderEvent.subscribe(function() {
889 YAHOO.util.Event.onAvailable('htmleditordiv' + idx, function() {
890 SE.composeLayout._createComposeLayout(idx);
891 SE.composeLayout[idx].set('height', 350);
892 SE.composeLayout[idx].render();
896 //Wait until the Compose Layout has rendered, then add the
897 //options tab and perform the tiny initialization.
898 parentPanel.renderEvent.subscribe(function() {
900 YAHOO.util.Event.onAvailable('htmleditordiv' + idx, function() {
901 SE.composeLayout._initComposeOptionTabs(idx);
902 SE.composeLayout[idx].getUnitByPosition("right").collapse();
904 SE.composeLayout._1_tiny(false);
906 //Init templates and address book
907 SE.composeLayout._2_final();
909 SE.composeLayout.quickCreateComposePackage(o);
914 //Check if we have the div override for the shortcut bar
915 if(typeof o.menu_id != 'undefined') {
916 parentPanel.render(o.menu_id);
918 parentPanel.render(document.body);
921 return SE.composeLayout[idx];
924 * Fill in all fields into the quick compose layout.
925 * @method quickCreateComposePackage
926 * @param {Object} o Options
929 quickCreateComposePackage: function(o)
931 //If we have a compose package fill in defaults.
932 if (typeof(o.composePackage) != 'undefined')
934 composePackage = o.composePackage; //Set the compose data object
935 //Hijack this method called by composePackage as it's not need for quick creates.
936 SE.composeLayout.c0_composeNewEmail = function(){};
937 SE.composeLayout.composePackage(); //Fill in defaults.
940 getComposeLayout : function() {
941 var idx = SE.composeLayout.currentInstanceId;
943 this._createComposeLayout(idx);
944 SE.composeLayout[idx].render();
945 this._initComposeOptionTabs(idx);
947 return SE.composeLayout[idx];
951 * Create the layout manager for the compose window.
953 _createComposeLayout : function(idx)
955 SE.composeLayout[idx] = new YAHOO.widget.Layout('htmleditordiv' + idx, {
956 parent: SE.complexLayout,
966 SE.composeLayout.composeTemplate.exec({
967 'app_strings':app_strings,
968 'mod_strings':mod_strings,
970 'linkbeans_options' : linkBeans,
971 'idx' : SE.composeLayout.currentInstanceId
982 body: "<div class='composeRightTabs' id='composeRightTabs" + idx + "'/>",
985 header: app_strings.LBL_EMAIL_OPTIONS
991 * Create compose tab which will populate the 'right' container in the compose window.
993 _initComposeOptionTabs : function(idx)
995 var cTabs = new YAHOO.widget.TabView("composeRightTabs" + idx);
996 var tab = new YAHOO.widget.Tab({
997 label: app_strings.LBL_EMAIL_ATTACHMENT,
999 content : SUGAR.util.getAndRemove("divAttachments" + idx).innerHTML,
1000 id : "divAttachments" + idx,
1004 tab.layout = SE.composeLayout[idx];
1006 tab.on("activeChange", function(o){
1008 this.layout.getUnitByPosition("right").set("header", app_strings.LBL_EMAIL_ATTACHMENT);
1014 tab = new YAHOO.widget.Tab({
1015 label: app_strings.LBL_EMAIL_OPTIONS,
1017 content : SUGAR.util.getAndRemove("divOptions" + idx).innerHTML,
1018 id : "divOptions" + idx,
1022 tab.layout = SE.composeLayout[idx];
1023 tab.on("activeChange", function(o){
1025 this.layout.getUnitByPosition("right").set("header", app_strings.LBL_EMAIL_OPTIONS);
1030 SE.composeLayout[idx].autoSize = function() {
1031 var pEl = this.get("element").parentNode.parentNode.parentNode;
1032 this.set("height", pEl.clientHeight-30);
1036 SE.composeLayout[idx].rightTabs = cTabs;
1038 isParentTypeValid : function(idx) {
1039 var parentTypeValue = document.getElementById('data_parent_type' + idx).value;
1040 var parentNameValue = document.getElementById('data_parent_name' + idx).value;
1041 if (trim(parentTypeValue) == ""){
1042 alert(mod_strings.LBL_ERROR_SELECT_MODULE);
1048 isParentTypeAndNameValid : function(idx) {
1049 var parentTypeValue = document.getElementById('data_parent_type' + idx).value;
1050 var parentNameValue = document.getElementById('data_parent_name' + idx).value;
1051 var parentIdValue = document.getElementById('data_parent_id' + idx).value;
1052 if ((trim(parentTypeValue) != "" && trim(parentNameValue) == "") ||
1053 (trim(parentTypeValue) != "" && trim(parentNameValue) != "" && parentIdValue == "")){
1054 alert(mod_strings.LBL_ERROR_SELECT_MODULE_SELECT);
1060 callopenpopupForEmail2 : function(idx,options) {
1062 var formName = 'emailCompose' + idx;
1064 if(typeof(options) != 'undefined' && typeof(options.form_name) != 'undefined')
1065 formName = options.form_name;
1067 var parentTypeValue = document.getElementById('data_parent_type' + idx).value;
1068 var parentNameValue = document.getElementById('data_parent_name' + idx).value;
1069 if (!SE.composeLayout.isParentTypeValid(idx)) {
1072 open_popup(document.getElementById('data_parent_type' + idx).value,600,400,'&tree=ProductsProd',true,false,
1074 call_back_function:"SE.composeLayout.popupAddEmail",
1076 field_to_name_array:{
1077 id:'data_parent_id' + idx,
1078 name:'data_parent_name' + idx,
1083 popupAddEmail : function(o)
1085 var nameKey = "data_parent_name" + SE.composeLayout.currentInstanceId;
1086 var data = o.name_to_value_array;
1087 if (typeof (data[nameKey]) != "undefined" && data[nameKey] != ""
1088 && typeof (data["email1"]) != "undefined" && data["email1"] != "" && data["email1"] != "undefined")
1090 var target = Dom.get("addressTO" + SE.composeLayout.currentInstanceId);
1091 target.value = SE.addressBook.smartAddEmailAddressToComposeField(target.value, data[nameKey] + "<" + data.email1 + ">");
1098 _1_tiny : function(isReplyForward) {
1099 var idx = SE.composeLayout.currentInstanceId;
1100 var elId = SE.tinyInstances.currentHtmleditor = 'htmleditor' + idx;
1101 SE.tinyInstances[elId] = { };
1102 SE.tinyInstances[elId].ready = false;
1104 if (!SUGAR.util.isTouchScreen()) {
1105 var t = tinyMCE.getInstanceById(elId);
1107 if(typeof(t) == 'undefined') {
1108 if (!SUGAR.util.isTouchScreen()) {
1109 tinyMCE.execCommand('mceAddControl', false, elId);
1111 YAHOO.util.Event.onAvailable(elId + "_parent", function() {
1112 SE.composeLayout.resizeEditorSetSignature(idx,!isReplyForward);
1117 resizeEditorSetSignature : function(idx,setSignature)
1119 var instance = SE.util.getTiny(SE.tinyInstances.currentHtmleditor);
1121 if(typeof(instance) == 'undefined' || (typeof(SE.composeLayout.loadedTinyInstances[idx]) != 'undefined' && SE.composeLayout.loadedTinyInstances[idx] == false)) {
1122 setTimeout("SE.composeLayout.resizeEditorSetSignature(" + idx + ",'"+isReplyForward+"');",500);
1126 SE.composeLayout.resizeEditor(idx);
1128 setTimeout("SUGAR.email2.composeLayout.setSignature("+idx+");",250);
1133 resizeEditor : function(idx)
1135 var cof = Dom.get('composeOverFrame' + idx);
1136 var head = Dom.get('composeHeaderTable' + idx);
1137 var targetHeight = cof.clientHeight - head.clientHeight;
1138 var instance = SE.util.getTiny('htmleditor' + idx);
1141 var parentEl = Dom.get(instance.editorId + '_parent');
1142 var toolbar = Dom.getElementsByClassName("mceFirst", "tr", parentEl)[0];
1143 var contentEl = instance.contentAreaContainer;
1144 var iFrame = contentEl.firstChild;
1145 var tinMceToolbarOffset = 18;
1146 iFrame.style.height = (targetHeight - toolbar.offsetHeight - tinMceToolbarOffset) + "px";
1149 setTimeout("SE.composeLayout.resizeEditor("+idx+");",1000);
1154 * Initializes d&d, auto-complete, email templates
1156 _2_final : function() {
1157 var idx = SE.composeLayout.currentInstanceId;
1159 if(this.emailTemplates) {
1160 this.setComposeOptions(idx);
1162 //populate email template cache
1163 AjaxObject.target = '';
1164 AjaxObject.startRequest(callbackComposeCache, urlStandard + "&emailUIAction=fillComposeCache");
1167 // handle drop targets for addressBook
1168 var to = new YAHOO.util.DDTarget('addressTO' +idx, 'addressBookDD', {notifyDrop:this.handleDrop});
1169 var cc = new YAHOO.util.DDTarget('addressCC' +idx, 'addressBookDD', {notifyDrop:this.handleDrop});
1170 var bcc = new YAHOO.util.DDTarget('addressBCC'+idx, 'addressBookDD', {notifyDrop:this.handleDrop});
1171 to.notifyDrop = cc.notifyDrop = bcc.notifyDrop = this.handleDrop;
1173 // auto-complete setup
1174 SE.autoComplete.init(idx);
1177 document.getElementById("addressTO" + idx).focus();
1181 * hide tinyMCE tool bar if send email as plaintext is checked
1183 renderTinyMCEToolBar : function (idx, hide) {
1185 document.getElementById('htmleditor' + idx + '_toolbar1').style.display = 'none';
1187 document.getElementById('htmleditor' + idx + '_toolbar1').style.display = '';
1191 c1_composeEmail : function(isReplyForward, retry) {
1195 if (!SUGAR.util.isTouchScreen() && (typeof(tinyMCE) == 'undefined' || typeof(tinyMCE.settings) == 'undefined')){
1196 setTimeout("SE.composeLayout.c1_composeEmail(" + isReplyForward + ", true);", 500);
1198 this._1_tiny(isReplyForward);
1201 if(isReplyForward) {
1202 this.replyForwardEmailStage2();
1208 * takes draft info and prepopulates
1210 c0_composeDraft : function() {
1211 this.getNewInstanceId();
1213 document.getElementById('_blank').innerHTML = '';
1214 var idx = SE.composeLayout.currentInstanceId;
1215 SE.composeLayout.draftObject = new Object();
1216 SE.composeLayout.draftObject.id = idx;
1217 SE.composeLayout.draftObject.isDraft = true;
1218 SE.composeLayout.currentInstanceId = idx;
1219 SE.tinyInstances.currentHtmleditor = 'htmleditor' + SE.composeLayout.currentInstanceId;
1220 SE.tinyInstances[SE.tinyInstances.currentHtmleditor] = new Object();
1221 SE.tinyInstances[SE.tinyInstances.currentHtmleditor].ready = false;
1223 SE.composeLayout._0_yui();
1224 SE.composeLayout._1_tiny(true);
1227 SE.composeLayout._2_final();
1229 /* Draft-specific final processing. Need a delay to allow Tiny to render before calling setText() */
1230 setTimeout("AjaxObject.handleReplyForwardForDraft(SE.o);", 1000);
1234 * Strip & Prep editor hidden fields
1236 c0_composeNewEmail : function() {
1237 this.getNewInstanceId();
1238 this.c1_composeEmail(false);
1242 * Sends async request to get the compose view.
1243 * Requests come from "reply" or "forwards"
1245 c0_replyForwardEmail : function(ieId, uid, mbox, type) {
1246 SE.composeLayout.replyForwardObj = new Object();
1247 SE.composeLayout.replyForwardObj.ieId = ieId;
1248 SE.composeLayout.replyForwardObj.uid = uid;
1249 SE.composeLayout.replyForwardObj.mbox = mbox;
1250 SE.composeLayout.replyForwardObj.type = type;
1252 if(mbox == 'sugar::Emails') {
1253 SE.composeLayout.replyForwardObj.sugarEmail = true;
1256 SE.composeLayout.getNewInstanceId();
1257 SE.composeLayout.c1_composeEmail(true);
1259 //// END COMPOSE FLOW
1260 ///////////////////////////////////////////////////////////////////////////
1263 * Called when a contact, email, or mailinglist is dropped
1264 * into one of the compose fields.
1266 handleDrop : function (source, event, data, target) {
1269 target = event.getTarget();
1271 data.nodes = [data.nodes];
1275 target = document.getElementById(target);
1279 if (target.id.indexOf('address') > -1) {
1280 // dropped onto email to/cc/bcc field
1281 for(var i in nodes) {
1282 var node = nodes[i].getData();
1284 if (node[1].indexOf('contact') > -1) {
1285 email = SE.addressBook.getFormattedAddress(node[0]);
1286 } else if (node[1].indexOf('address-email') > -1){
1287 email = node[3].replace(/ /gi, '');
1288 email = email.replace('<', '<').replace('>', '>');
1289 var tr = source.getTrEl(nodes[i]);
1290 while (tr && !Dom.hasClass(tr, "address-contact")) {
1291 tr = source.getPreviousTrEl(tr);
1293 var CID = source.getRecord(tr).getData()[0];
1294 var o = SE.addressBook._contactCache[CID];
1295 var name = new String(o.name);
1296 var finalName = name.replace(/(<([^>]+)>)/ig, "");
1297 email = finalName + email;
1299 target.value = SE.addressBook.smartAddEmailAddressToComposeField(target.value, email);
1305 /*/////////////////////////////////////////////////////////////////////////////
1306 /// EMAIL TEMPLATE CODE
1308 applyEmailTemplate : function (idx, id) {
1311 var box_title = mod_strings.LBL_EMAILTEMPLATE_MESSAGE_SHOW_TITLE;
1312 var box_msg = mod_strings.LBL_EMAILTEMPLATE_MESSAGE_SHOW_MSG;
1313 var box_none_msg = mod_strings.LBL_EMAILTEMPLATE_MESSAGE_CLEAR_MSG;
1316 var to_addr = document.getElementById('addressTO'+idx);
1317 if (to_addr.value.search(/[^;,]{6,}[;,][^;,]{6,}/) != -1)
1319 box_title = mod_strings.LBL_EMAILTEMPLATE_MESSAGE_WARNING_TITLE;
1320 box_msg = mod_strings.LBL_EMAILTEMPLATE_MESSAGE_MULTIPLE_RECIPIENTS + '<br /><br />' + box_msg;
1323 // id is selected index of email template drop-down
1324 if(id == '' || id == "0") {
1325 YAHOO.SUGAR.MessageBox.show({
1330 if(btn=='no'){return;};
1331 SUGAR.email2.composeLayout.processNoneResult(idx, id);},
1338 YAHOO.SUGAR.MessageBox.show({
1343 if(btn=='no'){return;};
1344 SUGAR.email2.composeLayout.processResult(idx, id);},
1350 processNoneResult : function(idx, id) {
1351 var tiny = SE.util.getTiny('htmleditor' + idx);
1352 var tinyHTML = tiny.getContent();
1353 var openTag = '<div><span><span>';
1354 var htmllow = tinyHTML.toLowerCase();
1355 var start = htmllow.indexOf(openTag);
1357 tinyHTML = tinyHTML.substr(start);
1358 tiny.setContent(tinyHTML);
1360 tiny.setContent('');
1364 processResult : function(idx , id){
1365 var post_data = {"module":"EmailTemplates","record":id};
1366 var global_rpcClient = new SugarRPCClient();
1368 result = global_rpcClient.call_method('retrieve', post_data, true);
1369 if(!result['record']) return;
1370 json_objects['email_template_object'] = result['record'];
1371 this.appendEmailTemplateJSON();
1373 // get attachments if any
1374 AjaxObject.target = '';
1375 AjaxObject.startRequest(callbackLoadAttachments, urlStandard + "&emailUIAction=getTemplateAttachments&parent_id=" + id);
1378 appendEmailTemplateJSON : function() {
1379 var idx = SE.composeLayout.currentInstanceId; // post increment
1381 // query based on template, contact_id0,related_to
1382 //jchi 09/10/2008 refix #7743
1383 if(json_objects['email_template_object']['fields']['subject'] != '' ) { // cn: bug 7743, don't stomp populated Subject Line
1384 document.getElementById('emailSubject' + idx).value = decodeURI(encodeURI(json_objects['email_template_object']['fields']['subject']));
1387 var text = decodeURI(encodeURI(json_objects['email_template_object']['fields']['body_html'])).replace(/<BR>/ig, '\n').replace(/<br>/gi, "\n").replace(/&/gi,'&').replace(/</gi,'<').replace(/>/gi,'>').replace(/'/gi,'\'').replace(/"/gi,'"');
1389 // cn: bug 14361 - text-only templates don't fill compose screen
1391 text = decodeURI(encodeURI(json_objects['email_template_object']['fields']['body'])).replace(/<BR>/ig, '\n').replace(/<br>/gi, "\n").replace(/&/gi,'&').replace(/</gi,'<').replace(/>/gi,'>').replace(/'/gi,'\'').replace(/"/gi,'"').replace(/\r\n/gi,"<br/>");
1394 var tiny = SE.util.getTiny('htmleditor' + idx);
1395 var tinyHTML = tiny.getContent();
1396 var openTag = '<div><span><span>';
1397 var closeTag = '</span></span></div>';
1398 var htmllow = tinyHTML.toLowerCase();
1399 var start = htmllow.indexOf(openTag);
1401 var htmlPart2 = tinyHTML.substr(start);
1402 tinyHTML = text + htmlPart2;
1403 tiny.setContent(tinyHTML);
1405 tiny.setContent(text);
1410 * Writes out the signature in the email editor
1412 setSignature : function(idx) {
1415 var hide = document.getElementById('setEditor' + idx).checked;
1416 SE.composeLayout.renderTinyMCEToolBar(idx,hide);
1417 //wait for signatures to load before trying to set them
1418 if (!SE.composeLayout.signatures) {
1419 setTimeout("SE.composeLayout.setSignature(" + idx + ");", 1000);
1424 var sel = document.getElementById('signatures' + idx);
1426 var sel = document.getElementById('signature_id');
1427 idx = SE.tinyInstances.currentHtmleditor;
1430 //Ensure that the tinyMCE html has been rendered.
1431 if(typeof(SE.composeLayout.loadedTinyInstances[idx]) != 'undefined' && SE.composeLayout.loadedTinyInstances[idx] == false) {
1432 setTimeout("SE.composeLayout.setSignature(" + idx + ");",1000);
1439 signature = sel.options[sel.selectedIndex].value;
1444 var openTag = '<div><span> </span>';
1445 var closeTag = '<span> </span></div>';
1446 var t = tinyMCE.getInstanceById('htmleditor' + idx);
1448 if(typeof(t) != 'undefined')
1450 t.contentDocument = t.contentWindow.document;
1451 var html = t.getContent();
1458 var htmllow = html.toLowerCase();
1459 var start = htmllow.indexOf(openTag);
1460 var end = htmllow.indexOf(closeTag) + closeTag.length;
1462 // selected "none" - remove signature from email
1463 if(signature == '') {
1465 var htmlPart1 = html.substr(0, start);
1466 var htmlPart2 = html.substr(end, html.length);
1468 html = htmlPart1 + htmlPart2;
1471 SE.signatures.lastAttemptedLoad = '';
1475 if(!SE.signatures.lastAttemptedLoad) // lazy load place holder
1476 SE.signatures.lastAttemptedLoad = '';
1478 SE.signatures.lastAttemptedLoad = signature;
1480 if(typeof(SE.signatures[signature]) == 'undefined') {
1482 SE.signatures.lastAttemptedLoad = ''; // reset this flag for recursion
1483 SE.signatures.targetInstance = (idx) ? idx : "";
1484 AjaxObject.target = '';
1485 AjaxObject.startRequest(callbackLoadSignature, urlStandard + "&emailUIAction=getSignature&id="+signature);
1487 var newSignature = this.prepareSignature(SE.signatures[signature]);
1489 // clear out old signature
1490 if(SE.signatures.lastAttemptedLoad && start > -1) {
1491 var htmlPart1 = html.substr(0, start);
1492 var htmlPart2 = html.substr(end, html.length);
1494 html = htmlPart1 + htmlPart2;
1498 start = html.indexOf('<div><hr></div>');
1499 if(SE.userPrefs.signatures.signature_prepend == 'true' && start > -1) {
1500 var htmlPart1 = html.substr(0, start);
1501 var htmlPart2 = html.substr(start, html.length);
1502 var newHtml = htmlPart1 + openTag + newSignature + closeTag + htmlPart2;
1503 } else if(SUGAR.email2.userPrefs.signatures.signature_prepend == 'true') {
1504 var newHtml = '<br/>' + openTag + newSignature + closeTag + html;
1506 var body = html.indexOf('</body>');
1508 var part1 = html.substr(0, body);
1509 var part2 = html.substr(body, html.length);
1510 var newHtml = part1 + openTag + newSignature + closeTag + part2;
1512 var newHtml = html + openTag + newSignature + closeTag;
1515 //tinyMCE.setContent(newHtml);
1516 t.setContent(newHtml);
1520 prepareSignature : function(str) {
1521 var signature = new String(str);
1523 signature = signature.replace(/</gi, '<');
1524 signature = signature.replace(/>/gi, '>');
1530 showAttachmentPanel : function(idx) {
1531 var east = SE.composeLayout[idx].getUnitByPosition("right");
1532 var tabs = SE.composeLayout[idx].rightTabs;
1534 tabs.set("activeTab", tabs.getTab(0));
1538 * expands sidebar and displays options panel
1540 showOptionsPanel : function(idx) {
1541 var east = SE.composeLayout[idx].getUnitByPosition("right");
1542 var tabs = SE.composeLayout[idx].rightTabs;
1544 tabs.set("activeTab", tabs.getTab(1));
1548 * Selects the Contacts tab
1550 showContactsPanel : function() {
1551 SE.complexLayout.regions.west.showPanel("contactsTab");
1555 * Generates fields for Select Document
1557 addDocumentField : function(idx) {
1558 var basket = document.getElementById('addedDocuments' + idx);
1560 var index = (basket.childNodes.length / 7) - 1;
1567 var test = document.getElementById('documentId' + idx + index);
1569 while(test != null) {
1571 test = document.getElementById('documentId' + idx + index);
1574 var documentCup = document.createElement("div");
1575 documentCup.id = 'documentCup' + idx + index;
1576 documentCup.innerHTML = "<input type='hidden' name='document" + idx + index + "' id='document" + idx + index + "' />" +
1577 // document id field
1578 "<input type='hidden' name='documentId" + idx + index + "' id='documentId" + idx + index + "' />" +
1579 // document name field
1580 "<input value='' size='15' disabled='true' type='text' name='documentName" + idx + index + "' id='documentName" + idx + index + "' />" +
1582 "<button class='button firstChild' type='button' name='documentSelect" + idx + index + "' id='documentSelect" + idx + index + "'" +
1583 "onclick='SE.composeLayout.selectDocument(\"" + index + "\");' value='" + app_strings.LBL_EMAIL_SELECT + "'>" +
1584 "<img src='index.php?entryPoint=getImage&themeName=" + SUGAR.themes.theme_name + "&imageName=id-ff-select.png' ></button>" +
1586 "<button class='button lastChild' type='button' name='documentRemove" + idx + index + "' id='documentRemove" + idx + index + "'" +
1587 "onclick='SE.composeLayout.deleteDocumentField(\"documentCup" + idx + index + "\");' value='" + app_strings.LBL_EMAIL_REMOVE + "'>" +
1588 "<img src='index.php?entryPoint=getImage&themeName=" + SUGAR.themes.theme_name + "&imageName=id-ff-clear.png' ></button>" +
1591 basket.appendChild(documentCup);
1592 //basket.innerHTML += out;
1597 * Makes async call to save a draft of the email
1598 * @param int Instance index
1600 saveDraft : function(tinyInstance) {
1601 this.sendEmail(tinyInstance, true);
1604 selectDocument : function(target) {
1605 URL="index.php?module=Emails&action=PopupDocuments&to_pdf=true&target=" + target;
1606 windowName = 'selectDocument';
1607 windowFeatures = 'width=800' + ',height=600' + ',resizable=1,scrollbars=1';
1609 win = SUGAR.util.openWindow(URL, windowName, windowFeatures);
1611 // put the focus on the popup if the browser supports the focus() method
1617 * Modal popup for file attachment dialogue
1619 addFileField : function() {
1620 if(!SE.addFileDialog){ // lazy initialize the dialog and only create it once
1621 SE.addFileDialog = new YAHOO.widget.Dialog("addFileDialog", {
1625 constraintoviewport: true,
1627 keylisteners : new YAHOO.util.KeyListener(document, { keys:27 }, {
1628 fn:function(){SE.addFileDialog.hide();}
1631 SE.addFileDialog.setHeader(app_strings.LBL_EMAIL_ATTACHMENTS);
1632 SE.addFileDialog.render();
1633 // SE.addFileDialog.addKeyListener(27, , SE.addFileDialog);
1635 Dom.removeClass("addFileDialog", "yui-hidden");
1637 SE.addFileDialog.show();
1641 * Async upload of file to temp dir
1643 uploadAttachment : function() {
1644 if(document.getElementById('email_attachment').value != "") {
1645 var formObject = document.getElementById('uploadAttachment');
1646 YAHOO.util.Connect.setForm(formObject, true, true);
1647 AjaxObject.target = '';
1648 AjaxObject.startRequest(callbackUploadAttachment, null);
1650 alert(app_strings.LBL_EMAIL_ERROR_NO_FILE);
1655 * Adds a SugarDocument to an outbound email. Action occurs in a popup window displaying a ListView from the Documents module
1656 * @param string target in focus compose layout
1658 setDocument : function(idx, target, documentId, documentName, docRevId) {
1659 // fields are named/id'd [fieldName][instanceId][index]
1660 var addedDocs = document.getElementById("addedDocuments" + idx);
1661 var docId = document.getElementById('documentId' + idx + target);
1662 var docName = document.getElementById('documentName' + idx + target);
1663 var docRevisionId = document.getElementById('document' + idx + target);
1664 docId.value = documentId;
1665 docName.value = documentName;
1666 docRevisionId.value = docRevId;
1670 * Removes the bucket div containing the document input fields
1672 deleteDocumentField : function(documentCup) {
1673 var f0 = document.getElementById(documentCup);
1674 f0.parentNode.removeChild(f0);
1678 * Removes a Template Attachment field
1682 deleteTemplateAttachmentField : function(idx, index) {
1683 // create not-in-array values for removal filtering
1684 var r = document.getElementById("templateAttachmentsRemove" + idx).value;
1690 r += document.getElementById('templateAttachmentId' + idx + index).value;
1691 document.getElementById("templateAttachmentsRemove" + idx).value = r;
1693 var target = 'templateAttachmentCup' + idx + index;
1694 d = document.getElementById(target);
1695 d.parentNode.removeChild(d);
1699 * Async removal of uploaded temp file
1700 * @param string index Should be a concatenation of idx and index
1703 deleteUploadAttachment : function(index, file) {
1704 var d = document.getElementById('email_attachment_bucket' + index);
1705 d.parentNode.removeChild(d);
1707 // make async call to delete cached file
1708 AjaxObject.target = '';
1709 AjaxObject.startRequest('', urlStandard + "&emailUIAction=removeUploadedAttachment&file="+file);
1713 * Attaches files coming from Email Templates
1715 addTemplateAttachmentField : function(idx) {
1717 document.getElementById('templateAttachmentsTitle' + idx).style.display = 'block';
1719 var basket = document.getElementById('addedTemplateAttachments' + idx);
1722 var index = basket.childNodes.length;
1729 var out = "<div id='templateAttachmentCup" + idx + index + "'>" +
1731 "<img src='index.php?entryPoint=getImage&themeName=" + SUGAR.themes.theme_name + "&imageName=minus.gif' " +
1732 "style='cursor:pointer' align='absmiddle' onclick='SUGAR.email2.composeLayout.deleteTemplateAttachmentField(\"" +
1733 idx + "\",\"" + index + "\");'/>" +
1735 "<img src='index.php?entryPoint=getImage&themeName=" + SUGAR.themes.theme_name + "&imageName=attachment.gif' " + "align='absmiddle' />" +
1736 // templateAttachment field
1737 "<input type='hidden' value='" + "' name='templateAttachment" + idx + index + "' id='templateAttachment" + idx + index + "' />" +
1739 "<input type='hidden' value='" + "' name='templateAttachmentId" + idx + index + "' id='templateAttachmentId" + idx + index + "' />" +
1741 "<span id='templateAttachmentName" + idx + index + "'" + "> </span>" +
1742 "<br id='br" + index + "></br>" +
1743 "<br id='brdoc" + index + "></br>" +
1745 basket.innerHTML = basket.innerHTML + out;
1751 * Sends one email via async call
1752 * @param int idx Editor instance ID
1753 * @param bool isDraft
1755 sendEmail : function(idx, isDraft) {
1757 //If the outbound account has an error message associate with it, alert the user and refuse to continue.
1758 var obAccountID = document.getElementById('addressFrom' + idx).value;
1760 if( typeof(SUGAR.email2.composeLayout.outboundAccountErrors[obAccountID]) != 'undefined' )
1762 overlay(app_strings.LBL_EMAIL_ERROR_DESC, SUGAR.email2.composeLayout.outboundAccountErrors[obAccountID], 'alert');
1767 var form = document.getElementById('emailCompose' + idx);
1768 var composeOptionsFormName = "composeOptionsForm" + idx;
1771 var t = SE.util.getTiny(SE.tinyInstances.currentHtmleditor);
1772 if (t != null || typeof(t) != "undefined") {
1773 var html = t.getContent();
1775 var html = "<p>" + document.getElementById('htmleditor' + idx).value + "</p>";
1778 var subj = document.getElementById('emailSubject' + idx).value;
1779 var to = trim(document.getElementById('addressTO' + idx).value);
1780 var cc = trim(document.getElementById('addressCC' + idx).value);
1781 var bcc = trim(document.getElementById('addressBCC' + idx).value);
1782 var email_id = document.getElementById('email_id' + idx).value;
1783 var composeType = document.getElementById('composeType').value;
1784 var parent_type = document.getElementById("parent_type").value;
1785 var parent_id = document.getElementById("parent_id").value;
1787 var el_uid = document.getElementById("uid");
1788 var uid = (el_uid == null) ? '' : el_uid.value;
1790 var el_ieId = document.getElementById("ieId");
1791 var ieId = (el_ieId == null) ? '' : el_ieId.value;
1793 var el_mbox = document.getElementById("mbox");
1794 var mbox = (el_mbox == null) ? '' : el_mbox.value;
1796 if (!isValidEmail(to) || !isValidEmail(cc) || !isValidEmail(bcc)) {
1797 alert(app_strings.LBL_EMAIL_COMPOSE_INVALID_ADDRESS);
1801 if (!SE.composeLayout.isParentTypeAndNameValid(idx)) {
1804 var parentTypeValue = document.getElementById('data_parent_type' + idx).value;
1805 var parentIdValue = document.getElementById('data_parent_id' + idx).value;
1806 parent_id = parentIdValue;
1807 parent_type = parentTypeValue;
1809 var in_draft = (document.getElementById('type' + idx).value == 'draft') ? true : false;
1810 // baseline viability check
1812 if(to == "" && cc == '' && bcc == '' && !isDraft) {
1813 alert(app_strings.LBL_EMAIL_COMPOSE_ERR_NO_RECIPIENTS);
1815 } else if(subj == '' && !isDraft) {
1816 if(!confirm(app_strings.LBL_EMAIL_COMPOSE_NO_SUBJECT)) {
1819 subj = app_strings.LBL_EMAIL_COMPOSE_NO_SUBJECT_LITERAL;
1821 } else if(html == '' && !isDraft) {
1822 if(!confirm(app_strings.LBL_EMAIL_COMPOSE_NO_BODY)) {
1827 SE.util.clearHiddenFieldValues('emailCompose' + idx);
1828 document.getElementById('data_parent_id' + idx).value = parentIdValue;
1829 var title = (isDraft) ? app_strings.LBL_EMAIL_SAVE_DRAFT : app_strings.LBL_EMAIL_SENDING_EMAIL;
1830 overlay(title, app_strings.LBL_EMAIL_ONE_MOMENT);
1831 html = html.replace(/</ig, "sugarLessThan");
1832 html = html.replace(/>/ig, "sugarGreaterThan");
1834 form.sendDescription.value = html;
1835 form.sendSubject.value = subj;
1836 form.sendTo.value = to;
1837 form.sendCc.value = cc;
1838 form.sendBcc.value = bcc;
1839 form.email_id.value = email_id;
1840 form.composeType.value = composeType;
1841 form.composeLayoutId.value = 'composeLayout' + idx;
1842 form.setEditor.value = (document.getElementById('setEditor' + idx).checked == false) ? 1 : 0;
1843 form.saveToSugar.value = 1;
1844 form.fromAccount.value = document.getElementById('addressFrom' + idx).value;
1845 form.parent_type.value = parent_type;
1846 form.parent_id.value = parent_id;
1847 form.uid.value = uid;
1848 form.ieId.value = ieId;
1849 form.mbox.value = mbox;
1851 // email attachments
1852 var addedFiles = document.getElementById('addedFiles' + idx);
1854 for(i=0; i<addedFiles.childNodes.length; i++) {
1855 var bucket = addedFiles.childNodes[i];
1857 for(j=0; j<bucket.childNodes.length; j++) {
1858 var node = bucket.childNodes[j];
1859 var nName = new String(node.name);
1861 if(node.type == 'hidden' && nName.match(/email_attachment/)) {
1862 if(form.attachments.value != '') {
1863 form.attachments.value += "::";
1865 form.attachments.value += node.value;
1872 var addedDocs = document.getElementById('addedDocuments' + idx);
1874 for(i=0; i<addedDocs.childNodes.length; i++) {
1875 var cNode = addedDocs.childNodes[i];
1876 for(j=0; j<cNode.childNodes.length; j++) {
1877 var node = cNode.childNodes[j];
1878 var nName = new String(node.name);
1879 if(node.type == 'hidden' && nName.match(/documentId/)) {
1880 if(form.documents.value != '') {
1881 form.documents.value += "::";
1883 form.documents.value += node.value;
1889 // template attachments
1890 var addedTemplateAttachments = document.getElementById('addedTemplateAttachments' + idx);
1891 if(addedTemplateAttachments) {
1892 for(i=0; i<addedTemplateAttachments.childNodes.length; i++) {
1893 var cNode = addedTemplateAttachments.childNodes[i];
1894 for(j=0; j<cNode.childNodes.length; j++) {
1895 var node = cNode.childNodes[j];
1896 var nName = new String(node.name);
1897 if(node.type == 'hidden' && nName.match(/templateAttachmentId/)) {
1898 if(form.templateAttachments.value != "") {
1899 form.templateAttachments.value += "::";
1901 form.templateAttachments.value += node.value;
1907 // remove attachments
1908 form.templateAttachmentsRemove.value = document.getElementById("templateAttachmentsRemove" + idx).value;
1910 YAHOO.util.Connect.setForm(form);
1912 AjaxObject.target = 'frameFlex';
1914 // sending a draft email
1915 if(!isDraft && in_draft) {
1917 SE.listView.removeRowByUid(email_id);
1920 var sendCallback = (isDraft) ? AjaxObject.composeLayout.callback.saveDraft : callbackSendEmail;
1921 var emailUiAction = (isDraft) ? "&emailUIAction=sendEmail&saveDraft=true" : "&emailUIAction=sendEmail";
1923 AjaxObject.startRequest(sendCallback, urlStandard + emailUiAction);
1927 * Handles clicking the email address link from a given view
1929 composePackage : function() {
1930 if(composePackage != null) {
1931 SE.composeLayout.c0_composeNewEmail();
1934 if(composePackage.to_email_addrs) {
1935 document.getElementById("addressTO" + SE.composeLayout.currentInstanceId).value = composePackage.to_email_addrs;
1937 if (composePackage.subject != null && composePackage.subject.length > 0) {
1938 document.getElementById("emailSubject" + SE.composeLayout.currentInstanceId).value = composePackage.subject;
1941 //If no parent fields are set in the composePackage, ensure they are cleared.
1942 var parentFields = ['parent_type','parent_name','parent_id'];
1943 for(var i=0;i<parentFields.length;i++)
1945 if ( typeof(composePackage[parentFields[i]]) == 'undefined' )
1946 composePackage[parentFields[i]] = "";
1949 document.getElementById("parent_type").value = composePackage.parent_type;
1950 document.getElementById('data_parent_type' + SE.composeLayout.currentInstanceId).value = composePackage.parent_type;
1951 document.getElementById("parent_id").value = composePackage.parent_id;
1952 document.getElementById('data_parent_id' + SE.composeLayout.currentInstanceId).value = composePackage.parent_id;
1953 document.getElementById('data_parent_name' + SE.composeLayout.currentInstanceId).value = composePackage.parent_name;
1955 if(composePackage.email_id != null && composePackage.email_id.length > 0) {
1956 document.getElementById("email_id" + SE.composeLayout.currentInstanceId).value = composePackage.email_id;
1958 if (composePackage.body != null && composePackage.body.length > 0) {
1959 var tiny = SE.util.getTiny('htmleditor' + SE.composeLayout.currentInstanceId);
1960 SE.composeLayout.loadedTinyInstances[SE.composeLayout.currentInstanceId] = false;
1961 setTimeout("SE.composeLayout.setContentOnThisTiny();", 3000);
1963 if (composePackage.attachments != null) {
1964 SE.composeLayout.loadAttachments(composePackage.attachments);
1967 if (composePackage.fromAccounts != null && composePackage.fromAccounts.status) {
1968 var addressFrom = document.getElementById('addressFrom' + SE.composeLayout.currentInstanceId);
1969 SE.util.emptySelectOptions(addressFrom);
1970 var fromAccountOpts = composePackage.fromAccounts.data;
1971 for(i=0; i<fromAccountOpts.length; i++) {
1972 var key = fromAccountOpts[i].value;
1973 var display = fromAccountOpts[i].text;
1974 var opt = new Option(display, key);
1975 if (fromAccountOpts[i].selected) {
1976 opt.selected = true;
1978 addressFrom.options.add(opt);
1985 setContentOnThisTiny : function() {
1986 var tiny = SE.util.getTiny('htmleditor' + SE.composeLayout.currentInstanceId);
1987 var tinyHTML = tiny.getContent();
1988 composePackage.body = decodeURI(encodeURI(composePackage.body));
1989 // cn: bug 14361 - text-only templates don't fill compose screen
1990 if(composePackage.body == '') {
1991 composePackage.body = decodeURI(encodeURI(composePackage.body)).replace(/<BR>/ig, '\n').replace(/<br>/gi, "\n").replace(/&/gi,'&').replace(/</gi,'<').replace(/>/gi,'>').replace(/'/gi,'\'').replace(/"/gi,'"');
1993 //Flag determines if we should clear the tiny contents or just append
1994 if (typeof(composePackage.clearBody) != 'undefined' && composePackage.clearBody)
1995 SE.composeLayout.tinyHTML = '';
1997 SE.composeLayout.tinyHTML = tinyHTML + composePackage.body;
1999 tiny.setContent(SE.composeLayout.tinyHTML);
2000 //Indicate that the contents has been loaded successfully.
2001 SE.composeLayout.loadedTinyInstances[SE.composeLayout.currentInstanceId] = true;
2004 * Confirms closure of a compose screen if "x" is clicked
2006 confirmClose : function(panel) {
2007 if(confirm(app_strings.LBL_EMAIL_CONFIRM_CLOSE)) {
2008 SE.composeLayout.closeCompose(panel.id);
2016 * forces close of a compose screen
2018 forceCloseCompose : function(id) {
2019 SE.composeLayout.closeCompose(id);
2021 // handle flow back to originating view
2022 if(composePackage) {
2023 // check if it's a module we need to return to
2024 if(composePackage.return_module && composePackage.return_action && composePackage.return_id) {
2025 if(confirm(app_strings.LBL_EMAIL_RETURN_TO_VIEW)) {
2026 var url = "index.php?module=" + composePackage.return_module + "&action=" + composePackage.return_action + "&record=" + composePackage.return_id;
2027 window.location = url;
2034 * closes the editor that just sent email
2035 * @param string id ID of composeLayout tab
2037 closeCompose : function(id) {
2038 // destroy tinyMCE instance
2039 var idx = id.substr(13, id.length);
2040 var instanceId = "htmleditor" + idx;
2041 tinyMCE.execCommand('mceRemoveControl', false, instanceId);
2043 // nullify DOM and namespace values.
2045 SE.composeLayout[idx] = null;
2046 SE.tinyInstances[instanceId] = null;
2047 var tabsArray = SE.innerLayout.get("tabs");
2048 for (i = 0 ; i < tabsArray.length ; i++) {
2049 if (tabsArray[i].get("id") == ('composeTab' + idx)) {
2050 tabsArray[i].close();
2054 //SE.innerLayout.getTab(idx).close();
2058 * Enable the quick search for the compose relate field or search tab
2060 enableQuickSearchRelate: function(idx,overides){
2062 if(typeof overides != 'undefined')
2064 var newModuleID = overides['moduleSelectField']; //data_parent_type_search
2065 var newModule = document.getElementById(newModuleID).value;
2066 var formName = overides['formName'];
2067 var fieldName = overides['fieldName'];
2068 var fieldId = overides['fieldId'];
2069 var fullName = formName + "_" + fieldName;
2070 var postBlurFunction = null;
2074 var newModule = document.getElementById('data_parent_type'+idx).value;
2075 var formName = 'emailCompose'+idx;
2076 var fieldName = 'data_parent_name'+idx;
2077 var fieldId = 'data_parent_id'+idx;
2078 var fullName = formName + "_" + fieldName;
2079 var postBlurFunction = "SE.composeLayout.qsAddAddress";
2082 if(typeof sqs_objects == 'undefined')
2083 window['sqs_objects'] = new Array;
2085 window['sqs_objects'][fullName] = {
2088 modules:[newModule],
2090 field_list:["name","id", "email1"],populate_list:[fieldName,fieldId],required_list:[fieldId],
2091 conditions:[{name:"name",op:"like_custom",end:"%",value:""}],
2092 post_onblur_function: postBlurFunction,
2093 order:"name","limit":"30","no_match_text":"No Match"};
2096 if(typeof QSProcessedFieldsArray != 'undefined')
2097 QSProcessedFieldsArray[fullName] = false;
2098 if (typeof(QSFieldsArray) != 'undefined' && typeof(QSFieldsArray[fullName]) != 'undefined') {
2099 QSFieldsArray[fullName].destroy();
2100 delete QSFieldsArray[fullName];
2102 if (Dom.get(fullName + "_results")) {
2103 Dom.get(fullName + "_results").parentNode.removeChild(Dom.get(fullName + "_results"));
2109 qsAddAddress : function(o) {
2110 if (o.name != "" && o.email1 != "")
2112 var target = Dom.get("addressTO" + SE.composeLayout.currentInstanceId);
2113 target.value = SE.addressBook.smartAddEmailAddressToComposeField(target.value, o.name + "<" + o.email1 + ">");
2117 * Returns a new instance ID, 0-index
2119 getNewInstanceId : function() {
2120 this.currentInstanceId = this.currentInstanceId + 1;
2121 return this.currentInstanceId;
2125 * Takes an array of objects that contain the filename and GUID of a Note (attachment or Sugar Document) and applies the values to the compose screen. Valid use-cases are applying an EmailTemplate or resuming a Draft Email.
2127 loadAttachments : function(result) {
2128 var idx = SE.composeLayout.currentInstanceId;
2130 if(typeof(result) == 'object') {
2131 //jchi #20680. Clean the former template attachments;
2132 var basket = document.getElementById('addedTemplateAttachments' + idx);
2133 if(basket.innerHTML != ''){
2134 confirm(mod_strings.LBL_CHECK_ATTACHMENTS, mod_strings.LBL_HAS_ATTACHMENTS, function(btn){
2136 basket.innerHTML = '';
2141 if(typeof result[i] == 'object') {
2142 var index = SE.composeLayout.addTemplateAttachmentField(idx);
2143 var bean = result[i];
2144 document.getElementById('templateAttachmentId' + idx + index).value = bean['id'];
2145 document.getElementById('templateAttachmentName' + idx + index).innerHTML += bean['filename'];
2152 * fills drop-down values for email templates and signatures
2154 setComposeOptions : function(idx) {
2155 // send from accounts
2156 var addressFrom = document.getElementById('addressFrom' + idx);
2158 if (addressFrom.options.length <= 0) {
2159 SE.util.emptySelectOptions(addressFrom);
2160 var fromAccountOpts = SE.composeLayout.fromAccounts;
2161 for (id = 0 ; id < fromAccountOpts.length ; id++) {
2162 var key = fromAccountOpts[id].value;
2163 var display = fromAccountOpts[id].text;
2164 var is_default = false;
2165 if(key == SUGAR.default_inbound_accnt_id)
2167 var opt = new Option(display, key);
2168 addressFrom.options.add(opt);
2169 addressFrom.options[id].selected = is_default; //Safari bug new Option(x,y,true) does not work.
2174 var et = document.getElementById('email_template' + idx);
2175 SE.util.emptySelectOptions(et);
2177 for(var key in this.emailTemplates) { // iterate through assoc array
2178 var display = this.emailTemplates[key];
2179 var opt = new Option(display, key);
2180 et.options.add(opt);
2184 var sigs = document.getElementById('signatures' + idx);
2185 SE.util.emptySelectOptions(sigs);
2187 for(var key in this.signatures) { // iterate through assoc array
2188 var display = this.signatures[key];
2189 var opt = new Option(display, key);
2191 if(key == SE.userPrefs.signatures.signature_default) {
2192 opt.selected = true;
2195 sigs.options.add(opt);
2198 // html/plain email?
2199 var htmlEmail = document.getElementById('setEditor' + idx);
2200 if(SE.userPrefs.emailSettings.sendPlainText == 1) {
2201 htmlEmail.checked = true;
2203 htmlEmail.checked = false;
2206 SE.tinyInstances[SE.tinyInstances.currentHtmleditor].ready = true;
2210 * After compose screen is rendered, async call to get email body from Sugar
2212 replyForwardEmailStage2 : function() {
2213 SE.util.clearHiddenFieldValues('emailUIForm');
2214 overlay(app_strings.LBL_EMAIL_RETRIEVING_MESSAGE, app_strings.LBL_EMAIL_ONE_MOMENT);
2216 var ieId = SE.composeLayout.replyForwardObj.ieId;
2217 var uid = SE.composeLayout.replyForwardObj.uid;
2218 var mbox = SE.composeLayout.replyForwardObj.mbox;
2219 var type = SE.composeLayout.replyForwardObj.type;
2220 var idx = SE.composeLayout.currentInstanceId;
2222 var sugarEmail = (SE.composeLayout.replyForwardObj.sugarEmail) ? '&sugarEmail=true' : "";
2224 document.getElementById('emailSubject' + idx).value = type;
2225 document.getElementById('emailUIAction').value = 'composeEmail';
2226 document.getElementById('composeType').value = type;
2227 document.getElementById('ieId').value = ieId;
2228 document.getElementById('uid').value = uid;
2229 document.getElementById('mbox').value = mbox;
2230 document.getElementById('setEditor' + idx).checked = SE.userPrefs.emailSettings.sendPlainText == 1 ? true : false;
2231 var formObject = document.getElementById('emailUIForm');
2232 YAHOO.util.Connect.setForm(formObject);
2234 var sendType = type;
2235 AjaxObject.startRequest(callbackReplyForward, urlStandard + "&composeType=" + type + sugarEmail);
2239 * Show the hidden cc or bcc fields
2241 showHiddenAddress: function(addrType,idx){
2243 Dom.removeClass(addrType+"_tr"+idx, "yui-hidden");
2244 Dom.addClass(addrType+"_span"+idx, "yui-hidden");
2245 Dom.addClass("bcc_cc_sep"+idx, "yui-hidden");
2246 this[addrType+'Hidden'+idx] = false;
2248 //After bcc or cc is added, move options below last addr field
2249 Dom.insertAfter("add_addr_options_tr"+idx, 'bcc_tr'+idx);
2251 //If both cc and bcc hidden, remove the empty row containing text.
2252 if( ( typeof(this['ccHidden'+idx]) != 'undefined' && typeof(this['bccHidden'+idx]) != 'undefined')
2253 && ( this['ccHidden'+idx] == false && this['bccHidden'+idx] == false) )
2254 Dom.addClass("add_addr_options_tr"+idx, "yui-hidden");
2256 // SE.composeLayout.resizeEditor(idx);
2259 * Hide the cc and bcc fields if they were shown.
2261 hideHiddenAddresses: function(idx){
2263 var addrTypes = ['cc','bcc'];
2264 for(var i = 0;i<addrTypes.length;i++)
2266 Dom.addClass(addrTypes[i] + "_tr"+idx, "yui-hidden");
2267 Dom.removeClass(addrTypes[i] + "_span"+idx, "yui-hidden");
2268 this[addrTypes[i] + 'Hidden'+idx] = true
2271 Dom.removeClass("bcc_cc_sep"+idx, "yui-hidden");
2272 Dom.removeClass("add_addr_options_tr"+idx, "yui-hidden");
2273 Dom.insertBefore("add_addr_options_tr"+idx, 'bcc_tr'+idx);
2277 //// END SE.composeLayout
2278 ///////////////////////////////////////////////////////////////////////////////
2279 ///////////////////////////////////////////////////////////////////////////////
2283 * Cleans serialized UID lists of duplicates
2287 cleanUids : function(str) {
2288 var seen = new Object();
2290 var arr = new String(str).split(",");
2292 for(var i=0; i<arr.length; i++) {
2297 clean += (clean != "") ? "," : "";
2299 seen[arr[i]] = true;
2306 * Clears hidden field values
2307 * @param string id ID of form element to clear
2309 clearHiddenFieldValues : function(id) {
2310 var form = document.getElementById(id);
2312 for(i=0; i<form.elements.length; i++) {
2313 if(form.elements[i].type == 'hidden') {
2314 var e = form.elements[i];
2315 if(e.name != 'action' && e.name != 'module' && e.name != 'to_pdf') {
2323 * Reduces a SELECT drop-down to 0 items to prepare for new ones
2325 emptySelectOptions : function(el) {
2327 for(i=el.childNodes.length - 1; i >= 0; i--) {
2328 if(el.childNodes[i]) {
2329 el.removeChild(el.childNodes[i]);
2336 * Returns the MBOX path in the manner php_imap expects:
2337 * ie: INBOX.DEBUG.test
2338 * @param string str Current serialized value, Home.personal.test.INBOX.DEBUG.test
2340 generateMboxPath : function(str) {
2341 var ex = str.split("::");
2343 /* we have a serialized MBOX path */
2347 for(var i=0; i<ex.length; i++) {
2348 if(ex[i] == 'INBOX') {
2360 /* we have a Sugar folder GUID - do nothing */
2368 * returns a SUGAR GUID by navigating the DOM tree a few moves backwards
2369 * @param HTMLElement el
2370 * @return string GUID of found element or empty on failure
2372 getGuidFromElement : function(el) {
2378 for(var i=0; i<iterations; i++) {
2380 if(el.id.match(SE.reGUID)) {
2392 * Returns the ID value for the current in-focus, active panel (in the innerLayout, not complexLayout)
2395 getPanelId : function() {
2396 return SE.innerLayout.get("activeTab").id ? SE.innerLayout.get("activeTab").id : "Preview";
2400 * wrapper to handle weirdness with IE
2401 * @param string instanceId
2402 * @return tinyMCE Controller object
2404 getTiny : function(instanceId) {
2405 if(instanceId == '') {
2409 var t = tinyMCE.getInstanceById(instanceId);
2413 YAHOO.util.Event.onContentReady(instanceId, function(t) { return t; });
2419 * Simple check for MSIE browser
2423 var nav = new String(navigator.appVersion);
2424 if(nav.match(/MSIE/)) {
2431 * Recursively removes an element from the DOM
2432 * @param HTMLElement
2434 removeElementRecursive : function(el) {
2435 this.emptySelectOptions(el);
2442 sleep : function(secs) {
2443 setTimeout("void(0);", secs);
2447 * Converts a <select> element to an Ext.form.combobox
2449 convertSelect : function(select) {
2450 alert('in convertSelect');
2451 if (typeof(select) == "string") {
2452 select = document.getElementById(select);
2456 findChildNode : function (parent, property, value) {
2457 for (i in parent.children) {
2458 var child = parent.children[i];
2459 if (child.data[property] && child.data[property] == value || child[property] && child[property] == value)
2461 var searchChild = SE.util.findChildNode(child, property, value);
2468 cascadeNodes : function (parent, fn, scope, args) {
2469 for (i in parent.children) {
2470 var child = parent.children[i];
2471 var s = scope ? scope : child;
2472 var a = args ? args : child;
2474 SE.util.cascadeNodes(child, fn, scope, args);
2481 ///////////////////////////////////////////////////////////////////////////////
2484 })();//End namespace